Files

109 lines
2.9 KiB
Go
Raw Permalink Normal View History

2021-04-20 06:25:08 +08:00
// Copyright 2017 The Gitea Authors. All rights reserved.
// SPDX-License-Identifier: MIT
2021-04-20 06:25:08 +08:00
package markup
import (
"io"
2024-11-24 16:18:57 +08:00
"path"
2021-04-20 06:25:08 +08:00
"strings"
"code.gitea.io/gitea/modules/setting"
"code.gitea.io/gitea/modules/typesniffer"
2021-04-20 06:25:08 +08:00
)
// Renderer defines an interface for rendering markup file to HTML
type Renderer interface {
2026-01-26 10:34:38 +08:00
Name() string // markup format name, also the renderer type, also the external tool name
FileNamePatterns() []string
2021-06-23 23:09:51 +02:00
SanitizerRules() []setting.MarkupSanitizerRule
2021-04-20 06:25:08 +08:00
Render(ctx *RenderContext, input io.Reader, output io.Writer) error
}
// PostProcessRenderer defines an interface for renderers who need post process
type PostProcessRenderer interface {
NeedPostProcess() bool
}
2025-10-23 16:01:38 +08:00
type ExternalRendererOptions struct {
SanitizerDisabled bool
DisplayInIframe bool
ContentSandbox string
}
2024-11-04 18:59:50 +08:00
// ExternalRenderer defines an interface for external renderers
type ExternalRenderer interface {
2025-10-23 16:01:38 +08:00
GetExternalRendererOptions() ExternalRendererOptions
}
// RendererContentDetector detects if the content can be rendered
// by specified renderer
type RendererContentDetector interface {
CanRender(filename string, sniffedType typesniffer.SniffedType, prefetchBuf []byte) bool
}
2021-04-20 06:25:08 +08:00
var (
2026-01-26 10:34:38 +08:00
fileNameRenderers = make(map[string]Renderer)
renderers = make(map[string]Renderer)
2021-04-20 06:25:08 +08:00
)
// RegisterRenderer registers a new markup file renderer
func RegisterRenderer(renderer Renderer) {
2026-01-26 10:34:38 +08:00
// TODO: need to handle conflicts
2021-04-20 06:25:08 +08:00
renderers[renderer.Name()] = renderer
2026-01-26 10:34:38 +08:00
}
func RefreshFileNamePatterns() {
// TODO: need to handle conflicts
fileNameRenderers = make(map[string]Renderer)
for _, renderer := range renderers {
for _, ext := range renderer.FileNamePatterns() {
fileNameRenderers[strings.ToLower(ext)] = renderer
}
2021-04-20 06:25:08 +08:00
}
}
2026-01-26 10:34:38 +08:00
func DetectRendererTypeByFilename(filename string) Renderer {
basename := path.Base(strings.ToLower(filename))
ext1 := path.Ext(basename)
if renderer := fileNameRenderers[basename]; renderer != nil {
return renderer
}
if renderer := fileNameRenderers["*"+ext1]; renderer != nil {
return renderer
}
if basename, ok := strings.CutSuffix(basename, ext1); ok {
ext2 := path.Ext(basename)
if renderer := fileNameRenderers["*"+ext2+ext1]; renderer != nil {
return renderer
}
}
return nil
2021-04-20 06:25:08 +08:00
}
2026-01-26 10:34:38 +08:00
// DetectRendererTypeByPrefetch detects the markup type of the content
func DetectRendererTypeByPrefetch(filename string, sniffedType typesniffer.SniffedType, prefetchBuf []byte) string {
if filename != "" {
byExt := DetectRendererTypeByFilename(filename)
if byExt != nil {
return byExt.Name()
}
}
for _, renderer := range renderers {
if detector, ok := renderer.(RendererContentDetector); ok && detector.CanRender(filename, sniffedType, prefetchBuf) {
return renderer.Name()
}
}
return ""
}
func PreviewableExtensions() []string {
2026-01-26 10:34:38 +08:00
exts := make([]string, 0, len(fileNameRenderers))
for p := range fileNameRenderers {
if s, ok := strings.CutPrefix(p, "*"); ok {
exts = append(exts, s)
}
}
2026-01-26 10:34:38 +08:00
return exts
}