2021-04-20 06:25:08 +08:00
|
|
|
// Copyright 2017 The Gitea Authors. All rights reserved.
|
2022-11-27 13:20:29 -05:00
|
|
|
// 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"
|
2025-06-30 16:12:25 +08:00
|
|
|
"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
|
|
|
|
|
}
|
|
|
|
|
|
2022-06-16 11:33:23 +08:00
|
|
|
// 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
|
2022-06-16 11:33:23 +08:00
|
|
|
type ExternalRenderer interface {
|
2025-10-23 16:01:38 +08:00
|
|
|
GetExternalRendererOptions() ExternalRendererOptions
|
2022-06-16 11:33:23 +08:00
|
|
|
}
|
|
|
|
|
|
2022-06-09 00:46:39 +03:00
|
|
|
// RendererContentDetector detects if the content can be rendered
|
|
|
|
|
// by specified renderer
|
|
|
|
|
type RendererContentDetector interface {
|
2025-06-30 16:12:25 +08:00
|
|
|
CanRender(filename string, sniffedType typesniffer.SniffedType, prefetchBuf []byte) bool
|
2022-06-09 00:46:39 +03:00
|
|
|
}
|
|
|
|
|
|
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()
|
|
|
|
|
}
|
|
|
|
|
}
|
2022-06-09 00:46:39 +03:00
|
|
|
for _, renderer := range renderers {
|
2025-06-30 16:12:25 +08:00
|
|
|
if detector, ok := renderer.(RendererContentDetector); ok && detector.CanRender(filename, sniffedType, prefetchBuf) {
|
2022-06-09 00:46:39 +03:00
|
|
|
return renderer.Name()
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return ""
|
|
|
|
|
}
|
|
|
|
|
|
2023-03-24 07:12:23 +01:00
|
|
|
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)
|
|
|
|
|
}
|
2023-03-24 07:12:23 +01:00
|
|
|
}
|
2026-01-26 10:34:38 +08:00
|
|
|
return exts
|
2023-03-24 07:12:23 +01:00
|
|
|
}
|