Files

222 lines
6.9 KiB
Go
Raw Permalink Normal View History

// Copyright 2022 The Gitea Authors. All rights reserved.
// SPDX-License-Identifier: MIT
package templates
import (
"bufio"
"bytes"
"errors"
"fmt"
2026-01-24 13:11:49 +08:00
"html/template"
2023-04-08 14:21:50 +08:00
"io"
"path/filepath"
"regexp"
"strconv"
"strings"
2023-04-08 14:21:50 +08:00
"sync/atomic"
texttemplate "text/template"
"code.gitea.io/gitea/modules/assetfs"
"code.gitea.io/gitea/modules/log"
"code.gitea.io/gitea/modules/setting"
"code.gitea.io/gitea/modules/templates/scopedtmpl"
"code.gitea.io/gitea/modules/util"
)
type TemplateExecutor scopedtmpl.TemplateExecutor
type TplName string
2026-01-24 13:11:49 +08:00
type tmplRender struct {
templates atomic.Pointer[scopedtmpl.ScopedTemplate]
2023-04-30 20:22:23 +08:00
2026-01-24 13:11:49 +08:00
collectTemplateNames func() ([]string, error)
readTemplateContent func(name string) ([]byte, error)
2023-04-08 14:21:50 +08:00
}
2026-01-24 13:11:49 +08:00
func (h *tmplRender) Templates() *scopedtmpl.ScopedTemplate {
return h.templates.Load()
2023-04-08 14:21:50 +08:00
}
2026-01-24 13:11:49 +08:00
func (h *tmplRender) recompileTemplates(dummyFuncMap template.FuncMap) error {
tmpls := scopedtmpl.NewScopedTemplate()
2026-01-24 13:11:49 +08:00
tmpls.Funcs(dummyFuncMap)
names, err := h.collectTemplateNames()
if err != nil {
2026-01-24 13:11:49 +08:00
return err
}
2026-01-24 13:11:49 +08:00
for _, name := range names {
2023-04-08 14:21:50 +08:00
tmpl := tmpls.New(filepath.ToSlash(name))
2026-01-24 13:11:49 +08:00
buf, err := h.readTemplateContent(name)
2023-04-08 14:21:50 +08:00
if err != nil {
return err
}
if _, err = tmpl.Parse(string(buf)); err != nil {
return err
}
}
tmpls.Freeze()
2023-04-08 14:21:50 +08:00
h.templates.Store(tmpls)
return nil
}
2026-01-24 13:11:49 +08:00
func ReloadAllTemplates() error {
return errors.Join(PageRendererReload(), MailRendererReload())
}
2026-01-24 13:11:49 +08:00
func processStartupTemplateError(err error) {
if err == nil {
return
}
2026-01-24 13:11:49 +08:00
if setting.IsProd || setting.IsInTesting {
2023-05-25 11:47:30 +08:00
// in prod mode, Gitea must have correct templates to run
2026-01-24 13:11:49 +08:00
log.Fatal("Gitea can't run with template errors: %v", err)
2023-05-25 11:47:30 +08:00
}
2024-04-22 13:48:42 +02:00
// in dev mode, do not need to really exit, because the template errors could be fixed by developer soon and the templates get reloaded
2026-01-24 13:11:49 +08:00
log.Error("There are template errors but Gitea continues to run in dev mode: %v", err)
}
type templateErrorPrettier struct {
assets *assetfs.LayeredFS
}
var reGenericTemplateError = regexp.MustCompile(`^template: (.*):([0-9]+): (.*)`)
func (p *templateErrorPrettier) handleGenericTemplateError(err error) string {
groups := reGenericTemplateError.FindStringSubmatch(err.Error())
if len(groups) != 4 {
return ""
}
tmplName, lineStr, message := groups[1], groups[2], groups[3]
2026-03-08 15:35:50 +01:00
return p.makeDetailedError(message, tmplName, lineStr, "", "")
}
var reFuncNotDefinedError = regexp.MustCompile(`^template: (.*):([0-9]+): (function "(.*)" not defined)`)
func (p *templateErrorPrettier) handleFuncNotDefinedError(err error) string {
groups := reFuncNotDefinedError.FindStringSubmatch(err.Error())
if len(groups) != 5 {
return ""
}
tmplName, lineStr, message, funcName := groups[1], groups[2], groups[3], groups[4]
funcName, _ = strconv.Unquote(`"` + funcName + `"`)
2026-03-08 15:35:50 +01:00
return p.makeDetailedError(message, tmplName, lineStr, "", funcName)
}
var reUnexpectedOperandError = regexp.MustCompile(`^template: (.*):([0-9]+): (unexpected "(.*)" in operand)`)
func (p *templateErrorPrettier) handleUnexpectedOperandError(err error) string {
groups := reUnexpectedOperandError.FindStringSubmatch(err.Error())
if len(groups) != 5 {
return ""
}
tmplName, lineStr, message, unexpected := groups[1], groups[2], groups[3], groups[4]
unexpected, _ = strconv.Unquote(`"` + unexpected + `"`)
2026-03-08 15:35:50 +01:00
return p.makeDetailedError(message, tmplName, lineStr, "", unexpected)
}
var reExpectedEndError = regexp.MustCompile(`^template: (.*):([0-9]+): (expected end; found (.*))`)
func (p *templateErrorPrettier) handleExpectedEndError(err error) string {
groups := reExpectedEndError.FindStringSubmatch(err.Error())
if len(groups) != 5 {
return ""
}
tmplName, lineStr, message, unexpected := groups[1], groups[2], groups[3], groups[4]
2026-03-08 15:35:50 +01:00
return p.makeDetailedError(message, tmplName, lineStr, "", unexpected)
}
var (
reTemplateExecutingError = regexp.MustCompile(`^template: (.*):([1-9][0-9]*):([1-9][0-9]*): (executing .*)`)
reTemplateExecutingErrorMsg = regexp.MustCompile(`^executing "(.*)" at <(.*)>: `)
)
func (p *templateErrorPrettier) handleTemplateRenderingError(err error) string {
if groups := reTemplateExecutingError.FindStringSubmatch(err.Error()); len(groups) > 0 {
tmplName, lineStr, posStr, msgPart := groups[1], groups[2], groups[3], groups[4]
target := ""
if groups = reTemplateExecutingErrorMsg.FindStringSubmatch(msgPart); len(groups) > 0 {
target = groups[2]
}
return p.makeDetailedError(msgPart, tmplName, lineStr, posStr, target)
} else if execErr, ok := err.(texttemplate.ExecError); ok {
layerName := p.assets.GetFileLayerName(execErr.Name + ".tmpl")
return fmt.Sprintf("asset from: %s, %s", layerName, err.Error())
}
2023-10-24 04:54:59 +02:00
return err.Error()
}
func HandleTemplateRenderingError(err error) string {
p := &templateErrorPrettier{assets: AssetFS()}
return p.handleTemplateRenderingError(err)
}
const dashSeparator = "----------------------------------------------------------------------"
2026-03-08 15:35:50 +01:00
func (p *templateErrorPrettier) makeDetailedError(errMsg, tmplName, lineNumStr, posNumStr, target string) string {
code, layer, err := p.assets.ReadLayeredFile(tmplName + ".tmpl")
if err != nil {
return fmt.Sprintf("template error: %s, and unable to find template file %q", errMsg, tmplName)
}
2026-03-08 15:35:50 +01:00
line, err := strconv.Atoi(lineNumStr)
if err != nil {
2026-03-08 15:35:50 +01:00
return fmt.Sprintf("template error: %s, unable to parse template %q line number %s", errMsg, tmplName, lineNumStr)
}
2026-03-08 15:35:50 +01:00
pos, err := strconv.Atoi(util.IfZero(posNumStr, "-1"))
if err != nil {
2026-03-08 15:35:50 +01:00
return fmt.Sprintf("template error: %s, unable to parse template %q pos number %s", errMsg, tmplName, posNumStr)
}
2026-03-08 15:35:50 +01:00
detail := extractErrorLine(code, line, pos, target)
var msg string
if pos >= 0 {
msg = fmt.Sprintf("template error: %s:%s:%d:%d : %s", layer, tmplName, line, pos, errMsg)
} else {
msg = fmt.Sprintf("template error: %s:%s:%d : %s", layer, tmplName, line, errMsg)
2023-03-20 20:56:48 +00:00
}
return msg + "\n" + dashSeparator + "\n" + detail + "\n" + dashSeparator
}
func extractErrorLine(code []byte, lineNum, posNum int, target string) string {
b := bufio.NewReader(bytes.NewReader(code))
var line []byte
var err error
2025-06-18 03:48:09 +02:00
for i := range lineNum {
if line, err = b.ReadBytes('\n'); err != nil {
if i == lineNum-1 && errors.Is(err, io.EOF) {
err = nil
}
break
2023-03-20 20:56:48 +00:00
}
}
if err != nil {
return fmt.Sprintf("unable to find target line %d", lineNum)
}
line = bytes.TrimRight(line, "\r\n")
var indicatorLine []byte
targetBytes := []byte(target)
targetLen := len(targetBytes)
for i := 0; i < len(line); {
if posNum == -1 && target != "" && bytes.HasPrefix(line[i:], targetBytes) {
for j := 0; j < targetLen && i < len(line); j++ {
indicatorLine = append(indicatorLine, '^')
i++
}
} else if i == posNum {
indicatorLine = append(indicatorLine, '^')
i++
} else {
if line[i] == '\t' {
indicatorLine = append(indicatorLine, '\t')
} else {
indicatorLine = append(indicatorLine, ' ')
}
i++
}
}
// if the indicatorLine only contains spaces, trim it together
return strings.TrimRight(string(line)+"\n"+string(indicatorLine), " \t\r\n")
}