2021-04-20 06:25:08 +08:00
// Copyright 2017 The Gitea Authors. All rights reserved.
// Use of this source code is governed by a MIT-style
// license that can be found in the LICENSE file.
package markup
import (
2022-06-09 00:46:39 +03:00
"bytes"
2021-04-20 06:25:08 +08:00
"context"
"errors"
"fmt"
"io"
"path/filepath"
"strings"
"sync"
2021-06-20 23:39:12 +01:00
"code.gitea.io/gitea/modules/git"
2021-04-20 06:25:08 +08:00
"code.gitea.io/gitea/modules/setting"
)
// Init initialize regexps for markdown parsing
func Init ( ) {
NewSanitizer ( )
if len ( setting . Markdown . CustomURLSchemes ) > 0 {
CustomLinkURLSchemes ( setting . Markdown . CustomURLSchemes )
}
// since setting maybe changed extensions, this will reload all renderer extensions mapping
extRenderers = make ( map [ string ] Renderer )
for _ , renderer := range renderers {
for _ , ext := range renderer . Extensions ( ) {
extRenderers [ strings . ToLower ( ext ) ] = renderer
}
}
}
2022-06-08 09:59:16 +01:00
// Header holds the data about a header.
type Header struct {
Level int
Text string
ID string
}
2021-04-20 06:25:08 +08:00
// RenderContext represents a render context
type RenderContext struct {
2022-06-08 09:59:16 +01:00
Ctx context . Context
Filename string
Type string
IsWiki bool
URLPrefix string
Metas map [ string ] string
DefaultLink string
GitRepo * git . Repository
ShaExistCache map [ string ] bool
cancelFn func ( )
TableOfContents [ ] Header
2021-06-20 23:39:12 +01:00
}
// Cancel runs any cleanup functions that have been registered for this Ctx
func ( ctx * RenderContext ) Cancel ( ) {
if ctx == nil {
return
}
ctx . ShaExistCache = map [ string ] bool { }
if ctx . cancelFn == nil {
return
}
ctx . cancelFn ( )
}
// AddCancel adds the provided fn as a Cleanup for this Ctx
func ( ctx * RenderContext ) AddCancel ( fn func ( ) ) {
if ctx == nil {
return
}
oldCancelFn := ctx . cancelFn
if oldCancelFn == nil {
ctx . cancelFn = fn
return
}
ctx . cancelFn = func ( ) {
defer oldCancelFn ( )
fn ( )
}
2021-04-20 06:25:08 +08:00
}
// Renderer defines an interface for rendering markup file to HTML
type Renderer interface {
Name ( ) string // markup format name
Extensions ( ) [ ] string
2021-06-07 06:50:07 +08:00
NeedPostProcess ( ) bool
2021-06-23 23:09:51 +02:00
SanitizerRules ( ) [ ] setting . MarkupSanitizerRule
2022-03-06 16:41:54 +08:00
SanitizerDisabled ( ) bool
2021-04-20 06:25:08 +08:00
Render ( ctx * RenderContext , input io . Reader , output io . Writer ) error
}
2022-06-09 00:46:39 +03:00
// RendererContentDetector detects if the content can be rendered
// by specified renderer
type RendererContentDetector interface {
CanRender ( filename string , input io . Reader ) bool
}
2021-04-20 06:25:08 +08:00
var (
extRenderers = make ( map [ string ] Renderer )
renderers = make ( map [ string ] Renderer )
)
// RegisterRenderer registers a new markup file renderer
func RegisterRenderer ( renderer Renderer ) {
renderers [ renderer . Name ( ) ] = renderer
for _ , ext := range renderer . Extensions ( ) {
extRenderers [ strings . ToLower ( ext ) ] = renderer
}
}
// GetRendererByFileName get renderer by filename
func GetRendererByFileName ( filename string ) Renderer {
extension := strings . ToLower ( filepath . Ext ( filename ) )
return extRenderers [ extension ]
}
// GetRendererByType returns a renderer according type
func GetRendererByType ( tp string ) Renderer {
return renderers [ tp ]
}
2022-06-09 00:46:39 +03:00
// DetectRendererType detects the markup type of the content
func DetectRendererType ( filename string , input io . Reader ) string {
buf , err := io . ReadAll ( input )
if err != nil {
return ""
}
for _ , renderer := range renderers {
if detector , ok := renderer . ( RendererContentDetector ) ; ok && detector . CanRender ( filename , bytes . NewReader ( buf ) ) {
return renderer . Name ( )
}
}
return ""
}
2021-04-20 06:25:08 +08:00
// Render renders markup file to HTML with all specific handling stuff.
func Render ( ctx * RenderContext , input io . Reader , output io . Writer ) error {
if ctx . Type != "" {
return renderByType ( ctx , input , output )
} else if ctx . Filename != "" {
return renderFile ( ctx , input , output )
}
return errors . New ( "Render options both filename and type missing" )
}
// RenderString renders Markup string to HTML with all specific handling stuff and return string
func RenderString ( ctx * RenderContext , content string ) ( string , error ) {
var buf strings . Builder
if err := Render ( ctx , strings . NewReader ( content ) , & buf ) ; err != nil {
return "" , err
}
return buf . String ( ) , nil
}
2022-03-06 16:41:54 +08:00
type nopCloser struct {
io . Writer
}
func ( nopCloser ) Close ( ) error { return nil }
2021-06-07 06:50:07 +08:00
func render ( ctx * RenderContext , renderer Renderer , input io . Reader , output io . Writer ) error {
2021-04-20 06:25:08 +08:00
var wg sync . WaitGroup
var err error
pr , pw := io . Pipe ( )
defer func ( ) {
_ = pr . Close ( )
_ = pw . Close ( )
} ( )
2022-03-06 16:41:54 +08:00
var pr2 io . ReadCloser
var pw2 io . WriteCloser
if ! renderer . SanitizerDisabled ( ) {
pr2 , pw2 = io . Pipe ( )
defer func ( ) {
_ = pr2 . Close ( )
_ = pw2 . Close ( )
} ( )
wg . Add ( 1 )
go func ( ) {
err = SanitizeReader ( pr2 , renderer . Name ( ) , output )
_ = pr2 . Close ( )
wg . Done ( )
} ( )
} else {
pw2 = nopCloser { output }
}
2021-06-23 23:09:51 +02:00
wg . Add ( 1 )
go func ( ) {
if renderer . NeedPostProcess ( ) {
2021-06-07 06:50:07 +08:00
err = PostProcess ( ctx , pr , pw2 )
2021-06-23 23:09:51 +02:00
} else {
_ , err = io . Copy ( pw2 , pr )
}
_ = pr . Close ( )
_ = pw2 . Close ( )
wg . Done ( )
} ( )
2021-06-07 06:50:07 +08:00
if err1 := renderer . Render ( ctx , input , pw ) ; err1 != nil {
2021-04-20 06:25:08 +08:00
return err1
}
_ = pw . Close ( )
wg . Wait ( )
return err
}
// ErrUnsupportedRenderType represents
type ErrUnsupportedRenderType struct {
Type string
}
func ( err ErrUnsupportedRenderType ) Error ( ) string {
return fmt . Sprintf ( "Unsupported render type: %s" , err . Type )
}
func renderByType ( ctx * RenderContext , input io . Reader , output io . Writer ) error {
if renderer , ok := renderers [ ctx . Type ] ; ok {
return render ( ctx , renderer , input , output )
}
return ErrUnsupportedRenderType { ctx . Type }
}
// ErrUnsupportedRenderExtension represents the error when extension doesn't supported to render
type ErrUnsupportedRenderExtension struct {
Extension string
}
func ( err ErrUnsupportedRenderExtension ) Error ( ) string {
return fmt . Sprintf ( "Unsupported render extension: %s" , err . Extension )
}
func renderFile ( ctx * RenderContext , input io . Reader , output io . Writer ) error {
extension := strings . ToLower ( filepath . Ext ( ctx . Filename ) )
if renderer , ok := extRenderers [ extension ] ; ok {
return render ( ctx , renderer , input , output )
}
return ErrUnsupportedRenderExtension { extension }
}
// Type returns if markup format via the filename
func Type ( filename string ) string {
if parser := GetRendererByFileName ( filename ) ; parser != nil {
return parser . Name ( )
}
return ""
}
// IsMarkupFile reports whether file is a markup type file
func IsMarkupFile ( name , markup string ) bool {
if parser := GetRendererByFileName ( name ) ; parser != nil {
return parser . Name ( ) == markup
}
return false
}
// IsReadmeFile reports whether name looks like a README file
// based on its name. If an extension is provided, it will strictly
// match that extension.
// Note that the '.' should be provided in ext, e.g ".md"
func IsReadmeFile ( name string , ext ... string ) bool {
name = strings . ToLower ( name )
if len ( ext ) > 0 {
return name == "readme" + ext [ 0 ]
}
if len ( name ) < 6 {
return false
} else if len ( name ) == 6 {
return name == "readme"
}
return name [ : 7 ] == "readme."
}