The old `HTMLRender` is not ideal. 1. It shouldn't be initialized multiple times, it consumes a lot of memory and is slow. 2. It shouldn't depend on short-lived requests, the `WatchLocalChanges` needs a long-running context. 3. It doesn't make sense to use FuncsMap slice. HTMLRender was designed to only work for GItea's specialized 400+ templates, so it's good to make it a global shared instance.
		
			
				
	
	
		
			220 lines
		
	
	
		
			7.0 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			220 lines
		
	
	
		
			7.0 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // Copyright 2018 The Gitea Authors. All rights reserved.
 | |
| // Copyright 2014 The Gogs Authors. All rights reserved.
 | |
| // SPDX-License-Identifier: MIT
 | |
| 
 | |
| package templates
 | |
| 
 | |
| import (
 | |
| 	"context"
 | |
| 	"fmt"
 | |
| 	"html"
 | |
| 	"html/template"
 | |
| 	"net/url"
 | |
| 	"strings"
 | |
| 	"time"
 | |
| 
 | |
| 	system_model "code.gitea.io/gitea/models/system"
 | |
| 	"code.gitea.io/gitea/modules/base"
 | |
| 	"code.gitea.io/gitea/modules/emoji"
 | |
| 	"code.gitea.io/gitea/modules/markup"
 | |
| 	"code.gitea.io/gitea/modules/setting"
 | |
| 	"code.gitea.io/gitea/modules/svg"
 | |
| 	"code.gitea.io/gitea/modules/templates/eval"
 | |
| 	"code.gitea.io/gitea/modules/timeutil"
 | |
| 	"code.gitea.io/gitea/modules/util"
 | |
| 	"code.gitea.io/gitea/services/gitdiff"
 | |
| )
 | |
| 
 | |
| // NewFuncMap returns functions for injecting to templates
 | |
| func NewFuncMap() template.FuncMap {
 | |
| 	return map[string]interface{}{
 | |
| 		"DumpVar": dumpVar,
 | |
| 
 | |
| 		// -----------------------------------------------------------------
 | |
| 		// html/template related functions
 | |
| 		"dict":        dict, // it's lowercase because this name has been widely used. Our other functions should have uppercase names.
 | |
| 		"Eval":        Eval,
 | |
| 		"Safe":        Safe,
 | |
| 		"Escape":      html.EscapeString,
 | |
| 		"QueryEscape": url.QueryEscape,
 | |
| 		"JSEscape":    template.JSEscapeString,
 | |
| 		"Str2html":    Str2html, // TODO: rename it to SanitizeHTML
 | |
| 		"URLJoin":     util.URLJoin,
 | |
| 		"DotEscape":   DotEscape,
 | |
| 
 | |
| 		"PathEscape":         url.PathEscape,
 | |
| 		"PathEscapeSegments": util.PathEscapeSegments,
 | |
| 
 | |
| 		// utils
 | |
| 		"StringUtils": NewStringUtils,
 | |
| 		"SliceUtils":  NewSliceUtils,
 | |
| 		"JsonUtils":   NewJsonUtils,
 | |
| 
 | |
| 		// -----------------------------------------------------------------
 | |
| 		// svg / avatar / icon
 | |
| 		"svg":            svg.RenderHTML,
 | |
| 		"avatar":         Avatar,
 | |
| 		"avatarHTML":     AvatarHTML,
 | |
| 		"avatarByAction": AvatarByAction,
 | |
| 		"avatarByEmail":  AvatarByEmail,
 | |
| 		"repoAvatar":     RepoAvatar,
 | |
| 		"EntryIcon":      base.EntryIcon,
 | |
| 		"MigrationIcon":  MigrationIcon,
 | |
| 		"ActionIcon":     ActionIcon,
 | |
| 
 | |
| 		"SortArrow": SortArrow,
 | |
| 
 | |
| 		// -----------------------------------------------------------------
 | |
| 		// time / number / format
 | |
| 		"FileSize":      base.FileSize,
 | |
| 		"CountFmt":      base.FormatNumberSI,
 | |
| 		"TimeSince":     timeutil.TimeSince,
 | |
| 		"TimeSinceUnix": timeutil.TimeSinceUnix,
 | |
| 		"DateTime":      timeutil.DateTime,
 | |
| 		"Sec2Time":      util.SecToTime,
 | |
| 		"LoadTimes": func(startTime time.Time) string {
 | |
| 			return fmt.Sprint(time.Since(startTime).Nanoseconds()/1e6) + "ms"
 | |
| 		},
 | |
| 
 | |
| 		// -----------------------------------------------------------------
 | |
| 		// setting
 | |
| 		"AppName": func() string {
 | |
| 			return setting.AppName
 | |
| 		},
 | |
| 		"AppSubUrl": func() string {
 | |
| 			return setting.AppSubURL
 | |
| 		},
 | |
| 		"AssetUrlPrefix": func() string {
 | |
| 			return setting.StaticURLPrefix + "/assets"
 | |
| 		},
 | |
| 		"AppUrl": func() string {
 | |
| 			// The usage of AppUrl should be avoided as much as possible,
 | |
| 			// because the AppURL(ROOT_URL) may not match user's visiting site and the ROOT_URL in app.ini may be incorrect.
 | |
| 			// And it's difficult for Gitea to guess absolute URL correctly with zero configuration,
 | |
| 			// because Gitea doesn't know whether the scheme is HTTP or HTTPS unless the reverse proxy could tell Gitea.
 | |
| 			return setting.AppURL
 | |
| 		},
 | |
| 		"AppVer": func() string {
 | |
| 			return setting.AppVer
 | |
| 		},
 | |
| 		"AppDomain": func() string { // documented in mail-templates.md
 | |
| 			return setting.Domain
 | |
| 		},
 | |
| 		"AssetVersion": func() string {
 | |
| 			return setting.AssetVersion
 | |
| 		},
 | |
| 		"DisableGravatar": func(ctx context.Context) bool {
 | |
| 			return system_model.GetSettingWithCacheBool(ctx, system_model.KeyPictureDisableGravatar)
 | |
| 		},
 | |
| 		"DefaultShowFullName": func() bool {
 | |
| 			return setting.UI.DefaultShowFullName
 | |
| 		},
 | |
| 		"ShowFooterTemplateLoadTime": func() bool {
 | |
| 			return setting.Other.ShowFooterTemplateLoadTime
 | |
| 		},
 | |
| 		"AllowedReactions": func() []string {
 | |
| 			return setting.UI.Reactions
 | |
| 		},
 | |
| 		"CustomEmojis": func() map[string]string {
 | |
| 			return setting.UI.CustomEmojisMap
 | |
| 		},
 | |
| 		"ThemeColorMetaTag": func() string {
 | |
| 			return setting.UI.ThemeColorMetaTag
 | |
| 		},
 | |
| 		"MetaAuthor": func() string {
 | |
| 			return setting.UI.Meta.Author
 | |
| 		},
 | |
| 		"MetaDescription": func() string {
 | |
| 			return setting.UI.Meta.Description
 | |
| 		},
 | |
| 		"MetaKeywords": func() string {
 | |
| 			return setting.UI.Meta.Keywords
 | |
| 		},
 | |
| 		"UseServiceWorker": func() bool {
 | |
| 			return setting.UI.UseServiceWorker
 | |
| 		},
 | |
| 		"EnableTimetracking": func() bool {
 | |
| 			return setting.Service.EnableTimetracking
 | |
| 		},
 | |
| 		"DisableGitHooks": func() bool {
 | |
| 			return setting.DisableGitHooks
 | |
| 		},
 | |
| 		"DisableWebhooks": func() bool {
 | |
| 			return setting.DisableWebhooks
 | |
| 		},
 | |
| 		"DisableImportLocal": func() bool {
 | |
| 			return !setting.ImportLocalPaths
 | |
| 		},
 | |
| 		"DefaultTheme": func() string {
 | |
| 			return setting.UI.DefaultTheme
 | |
| 		},
 | |
| 		"NotificationSettings": func() map[string]interface{} {
 | |
| 			return map[string]interface{}{
 | |
| 				"MinTimeout":            int(setting.UI.Notification.MinTimeout / time.Millisecond),
 | |
| 				"TimeoutStep":           int(setting.UI.Notification.TimeoutStep / time.Millisecond),
 | |
| 				"MaxTimeout":            int(setting.UI.Notification.MaxTimeout / time.Millisecond),
 | |
| 				"EventSourceUpdateTime": int(setting.UI.Notification.EventSourceUpdateTime / time.Millisecond),
 | |
| 			}
 | |
| 		},
 | |
| 		"MermaidMaxSourceCharacters": func() int {
 | |
| 			return setting.MermaidMaxSourceCharacters
 | |
| 		},
 | |
| 
 | |
| 		// -----------------------------------------------------------------
 | |
| 		// render
 | |
| 		"RenderCommitMessage":            RenderCommitMessage,
 | |
| 		"RenderCommitMessageLinkSubject": RenderCommitMessageLinkSubject,
 | |
| 
 | |
| 		"RenderCommitBody": RenderCommitBody,
 | |
| 		"RenderCodeBlock":  RenderCodeBlock,
 | |
| 		"RenderIssueTitle": RenderIssueTitle,
 | |
| 		"RenderEmoji":      RenderEmoji,
 | |
| 		"RenderEmojiPlain": emoji.ReplaceAliases,
 | |
| 		"ReactionToEmoji":  ReactionToEmoji,
 | |
| 		"RenderNote":       RenderNote,
 | |
| 
 | |
| 		"RenderMarkdownToHtml": RenderMarkdownToHtml,
 | |
| 		"RenderLabel":          RenderLabel,
 | |
| 		"RenderLabels":         RenderLabels,
 | |
| 
 | |
| 		// -----------------------------------------------------------------
 | |
| 		// misc
 | |
| 		"DiffLineTypeToStr":        DiffLineTypeToStr,
 | |
| 		"ShortSha":                 base.ShortSha,
 | |
| 		"ActionContent2Commits":    ActionContent2Commits,
 | |
| 		"IsMultilineCommitMessage": IsMultilineCommitMessage,
 | |
| 		"CommentMustAsDiff":        gitdiff.CommentMustAsDiff,
 | |
| 		"MirrorRemoteAddress":      mirrorRemoteAddress,
 | |
| 
 | |
| 		"FilenameIsImage": FilenameIsImage,
 | |
| 		"TabSizeClass":    TabSizeClass,
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // Safe render raw as HTML
 | |
| func Safe(raw string) template.HTML {
 | |
| 	return template.HTML(raw)
 | |
| }
 | |
| 
 | |
| // Str2html render Markdown text to HTML
 | |
| func Str2html(raw string) template.HTML {
 | |
| 	return template.HTML(markup.Sanitize(raw))
 | |
| }
 | |
| 
 | |
| // DotEscape wraps a dots in names with ZWJ [U+200D] in order to prevent autolinkers from detecting these as urls
 | |
| func DotEscape(raw string) string {
 | |
| 	return strings.ReplaceAll(raw, ".", "\u200d.\u200d")
 | |
| }
 | |
| 
 | |
| // Eval the expression and return the result, see the comment of eval.Expr for details.
 | |
| // To use this helper function in templates, pass each token as a separate parameter.
 | |
| //
 | |
| //	{{ $int64 := Eval $var "+" 1 }}
 | |
| //	{{ $float64 := Eval $var "+" 1.0 }}
 | |
| //
 | |
| // Golang's template supports comparable int types, so the int64 result can be used in later statements like {{if lt $int64 10}}
 | |
| func Eval(tokens ...any) (any, error) {
 | |
| 	n, err := eval.Expr(tokens...)
 | |
| 	return n.Value, err
 | |
| }
 |