mirror of
				https://github.com/caddyserver/caddy.git
				synced 2025-10-26 00:02:45 -04:00 
			
		
		
		
	
		
			
				
	
	
		
			411 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			411 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // Copyright 2015 Matthew Holt and The Caddy Authors
 | |
| //
 | |
| // Licensed under the Apache License, Version 2.0 (the "License");
 | |
| // you may not use this file except in compliance with the License.
 | |
| // You may obtain a copy of the License at
 | |
| //
 | |
| //     http://www.apache.org/licenses/LICENSE-2.0
 | |
| //
 | |
| // Unless required by applicable law or agreed to in writing, software
 | |
| // distributed under the License is distributed on an "AS IS" BASIS,
 | |
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | |
| // See the License for the specific language governing permissions and
 | |
| // limitations under the License.
 | |
| 
 | |
| package templates
 | |
| 
 | |
| import (
 | |
| 	"bytes"
 | |
| 	"fmt"
 | |
| 	"io"
 | |
| 	"net"
 | |
| 	"net/http"
 | |
| 	"os"
 | |
| 	"path"
 | |
| 	"strconv"
 | |
| 	"strings"
 | |
| 	"sync"
 | |
| 	"text/template"
 | |
| 
 | |
| 	"github.com/Masterminds/sprig/v3"
 | |
| 	"github.com/alecthomas/chroma/formatters/html"
 | |
| 	"github.com/caddyserver/caddy/v2"
 | |
| 	"github.com/caddyserver/caddy/v2/modules/caddyhttp"
 | |
| 	"github.com/yuin/goldmark"
 | |
| 	highlighting "github.com/yuin/goldmark-highlighting"
 | |
| 	"github.com/yuin/goldmark/extension"
 | |
| 	"github.com/yuin/goldmark/parser"
 | |
| 	gmhtml "github.com/yuin/goldmark/renderer/html"
 | |
| )
 | |
| 
 | |
| // TemplateContext is the TemplateContext with which HTTP templates are executed.
 | |
| type TemplateContext struct {
 | |
| 	Root       http.FileSystem
 | |
| 	Req        *http.Request
 | |
| 	Args       []interface{} // defined by arguments to funcInclude
 | |
| 	RespHeader WrappedHeader
 | |
| 
 | |
| 	config *Templates
 | |
| }
 | |
| 
 | |
| // NewTemplate returns a new template intended to be evaluated with this
 | |
| // context, as it is initialized with configuration from this context.
 | |
| func (c TemplateContext) NewTemplate(tplName string) *template.Template {
 | |
| 	tpl := template.New(tplName)
 | |
| 
 | |
| 	// customize delimiters, if applicable
 | |
| 	if c.config != nil && len(c.config.Delimiters) == 2 {
 | |
| 		tpl.Delims(c.config.Delimiters[0], c.config.Delimiters[1])
 | |
| 	}
 | |
| 
 | |
| 	// add sprig library
 | |
| 	tpl.Funcs(sprigFuncMap)
 | |
| 
 | |
| 	// add our own library
 | |
| 	tpl.Funcs(template.FuncMap{
 | |
| 		"include":          c.funcInclude,
 | |
| 		"httpInclude":      c.funcHTTPInclude,
 | |
| 		"stripHTML":        c.funcStripHTML,
 | |
| 		"markdown":         c.funcMarkdown,
 | |
| 		"splitFrontMatter": c.funcSplitFrontMatter,
 | |
| 		"listFiles":        c.funcListFiles,
 | |
| 		"env":              c.funcEnv,
 | |
| 		"placeholder":      c.funcPlaceholder,
 | |
| 		"fileExists":       c.funcFileExists,
 | |
| 		"httpError":        c.funcHTTPError,
 | |
| 	})
 | |
| 
 | |
| 	return tpl
 | |
| }
 | |
| 
 | |
| // OriginalReq returns the original, unmodified, un-rewritten request as
 | |
| // it originally came in over the wire.
 | |
| func (c TemplateContext) OriginalReq() http.Request {
 | |
| 	or, _ := c.Req.Context().Value(caddyhttp.OriginalRequestCtxKey).(http.Request)
 | |
| 	return or
 | |
| }
 | |
| 
 | |
| // funcInclude returns the contents of filename relative to the site root.
 | |
| // Note that included files are NOT escaped, so you should only include
 | |
| // trusted files. If it is not trusted, be sure to use escaping functions
 | |
| // in your template.
 | |
| func (c TemplateContext) funcInclude(filename string, args ...interface{}) (string, error) {
 | |
| 	if c.Root == nil {
 | |
| 		return "", fmt.Errorf("root file system not specified")
 | |
| 	}
 | |
| 
 | |
| 	file, err := c.Root.Open(filename)
 | |
| 	if err != nil {
 | |
| 		return "", err
 | |
| 	}
 | |
| 	defer file.Close()
 | |
| 
 | |
| 	bodyBuf := bufPool.Get().(*bytes.Buffer)
 | |
| 	bodyBuf.Reset()
 | |
| 	defer bufPool.Put(bodyBuf)
 | |
| 
 | |
| 	_, err = io.Copy(bodyBuf, file)
 | |
| 	if err != nil {
 | |
| 		return "", err
 | |
| 	}
 | |
| 
 | |
| 	c.Args = args
 | |
| 
 | |
| 	err = c.executeTemplateInBuffer(filename, bodyBuf)
 | |
| 	if err != nil {
 | |
| 		return "", err
 | |
| 	}
 | |
| 
 | |
| 	return bodyBuf.String(), nil
 | |
| }
 | |
| 
 | |
| // funcHTTPInclude returns the body of a virtual (lightweight) request
 | |
| // to the given URI on the same server. Note that included bodies
 | |
| // are NOT escaped, so you should only include trusted resources.
 | |
| // If it is not trusted, be sure to use escaping functions yourself.
 | |
| func (c TemplateContext) funcHTTPInclude(uri string) (string, error) {
 | |
| 	// prevent virtual request loops by counting how many levels
 | |
| 	// deep we are; and if we get too deep, return an error
 | |
| 	recursionCount := 1
 | |
| 	if numStr := c.Req.Header.Get(recursionPreventionHeader); numStr != "" {
 | |
| 		num, err := strconv.Atoi(numStr)
 | |
| 		if err != nil {
 | |
| 			return "", fmt.Errorf("parsing %s: %v", recursionPreventionHeader, err)
 | |
| 		}
 | |
| 		if num >= 3 {
 | |
| 			return "", fmt.Errorf("virtual request cycle")
 | |
| 		}
 | |
| 		recursionCount = num + 1
 | |
| 	}
 | |
| 
 | |
| 	buf := bufPool.Get().(*bytes.Buffer)
 | |
| 	buf.Reset()
 | |
| 	defer bufPool.Put(buf)
 | |
| 
 | |
| 	virtReq, err := http.NewRequest("GET", uri, nil)
 | |
| 	if err != nil {
 | |
| 		return "", err
 | |
| 	}
 | |
| 	virtReq.Host = c.Req.Host
 | |
| 	virtReq.Header = c.Req.Header.Clone()
 | |
| 	virtReq.Trailer = c.Req.Trailer.Clone()
 | |
| 	virtReq.Header.Set(recursionPreventionHeader, strconv.Itoa(recursionCount))
 | |
| 
 | |
| 	vrw := &virtualResponseWriter{body: buf, header: make(http.Header)}
 | |
| 	server := c.Req.Context().Value(caddyhttp.ServerCtxKey).(http.Handler)
 | |
| 
 | |
| 	server.ServeHTTP(vrw, virtReq)
 | |
| 	if vrw.status >= 400 {
 | |
| 		return "", fmt.Errorf("http %d", vrw.status)
 | |
| 	}
 | |
| 
 | |
| 	err = c.executeTemplateInBuffer(uri, buf)
 | |
| 	if err != nil {
 | |
| 		return "", err
 | |
| 	}
 | |
| 
 | |
| 	return buf.String(), nil
 | |
| }
 | |
| 
 | |
| func (c TemplateContext) executeTemplateInBuffer(tplName string, buf *bytes.Buffer) error {
 | |
| 	tpl := c.NewTemplate(tplName)
 | |
| 
 | |
| 	parsedTpl, err := tpl.Parse(buf.String())
 | |
| 	if err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 
 | |
| 	buf.Reset() // reuse buffer for output
 | |
| 
 | |
| 	return parsedTpl.Execute(buf, c)
 | |
| }
 | |
| 
 | |
| func (c TemplateContext) funcPlaceholder(name string) string {
 | |
| 	repl := c.Req.Context().Value(caddy.ReplacerCtxKey).(*caddy.Replacer)
 | |
| 	value, _ := repl.GetString(name)
 | |
| 	return value
 | |
| }
 | |
| 
 | |
| func (TemplateContext) funcEnv(varName string) string {
 | |
| 	return os.Getenv(varName)
 | |
| }
 | |
| 
 | |
| // Cookie gets the value of a cookie with name name.
 | |
| func (c TemplateContext) Cookie(name string) string {
 | |
| 	cookies := c.Req.Cookies()
 | |
| 	for _, cookie := range cookies {
 | |
| 		if cookie.Name == name {
 | |
| 			return cookie.Value
 | |
| 		}
 | |
| 	}
 | |
| 	return ""
 | |
| }
 | |
| 
 | |
| // RemoteIP gets the IP address of the client making the request.
 | |
| func (c TemplateContext) RemoteIP() string {
 | |
| 	ip, _, err := net.SplitHostPort(c.Req.RemoteAddr)
 | |
| 	if err != nil {
 | |
| 		return c.Req.RemoteAddr
 | |
| 	}
 | |
| 	return ip
 | |
| }
 | |
| 
 | |
| // Host returns the hostname portion of the Host header
 | |
| // from the HTTP request.
 | |
| func (c TemplateContext) Host() (string, error) {
 | |
| 	host, _, err := net.SplitHostPort(c.Req.Host)
 | |
| 	if err != nil {
 | |
| 		if !strings.Contains(c.Req.Host, ":") {
 | |
| 			// common with sites served on the default port 80
 | |
| 			return c.Req.Host, nil
 | |
| 		}
 | |
| 		return "", err
 | |
| 	}
 | |
| 	return host, nil
 | |
| }
 | |
| 
 | |
| // funcStripHTML returns s without HTML tags. It is fairly naive
 | |
| // but works with most valid HTML inputs.
 | |
| func (TemplateContext) funcStripHTML(s string) string {
 | |
| 	var buf bytes.Buffer
 | |
| 	var inTag, inQuotes bool
 | |
| 	var tagStart int
 | |
| 	for i, ch := range s {
 | |
| 		if inTag {
 | |
| 			if ch == '>' && !inQuotes {
 | |
| 				inTag = false
 | |
| 			} else if ch == '<' && !inQuotes {
 | |
| 				// false start
 | |
| 				buf.WriteString(s[tagStart:i])
 | |
| 				tagStart = i
 | |
| 			} else if ch == '"' {
 | |
| 				inQuotes = !inQuotes
 | |
| 			}
 | |
| 			continue
 | |
| 		}
 | |
| 		if ch == '<' {
 | |
| 			inTag = true
 | |
| 			tagStart = i
 | |
| 			continue
 | |
| 		}
 | |
| 		buf.WriteRune(ch)
 | |
| 	}
 | |
| 	if inTag {
 | |
| 		// false start
 | |
| 		buf.WriteString(s[tagStart:])
 | |
| 	}
 | |
| 	return buf.String()
 | |
| }
 | |
| 
 | |
| // funcMarkdown renders the markdown body as HTML. The resulting
 | |
| // HTML is NOT escaped so that it can be rendered as HTML.
 | |
| func (TemplateContext) funcMarkdown(input interface{}) (string, error) {
 | |
| 	inputStr := toString(input)
 | |
| 
 | |
| 	md := goldmark.New(
 | |
| 		goldmark.WithExtensions(
 | |
| 			extension.GFM,
 | |
| 			extension.Footnote,
 | |
| 			highlighting.NewHighlighting(
 | |
| 				highlighting.WithFormatOptions(
 | |
| 					html.WithClasses(true),
 | |
| 				),
 | |
| 			),
 | |
| 		),
 | |
| 		goldmark.WithParserOptions(
 | |
| 			parser.WithAutoHeadingID(),
 | |
| 		),
 | |
| 		goldmark.WithRendererOptions(
 | |
| 			gmhtml.WithUnsafe(), // TODO: this is not awesome, maybe should be configurable?
 | |
| 		),
 | |
| 	)
 | |
| 
 | |
| 	buf := bufPool.Get().(*bytes.Buffer)
 | |
| 	buf.Reset()
 | |
| 	defer bufPool.Put(buf)
 | |
| 
 | |
| 	err := md.Convert([]byte(inputStr), buf)
 | |
| 	if err != nil {
 | |
| 		return "", err
 | |
| 	}
 | |
| 
 | |
| 	return buf.String(), nil
 | |
| }
 | |
| 
 | |
| // splitFrontMatter parses front matter out from the beginning of input,
 | |
| // and returns the separated key-value pairs and the body/content. input
 | |
| // must be a "stringy" value.
 | |
| func (TemplateContext) funcSplitFrontMatter(input interface{}) (parsedMarkdownDoc, error) {
 | |
| 	meta, body, err := extractFrontMatter(toString(input))
 | |
| 	if err != nil {
 | |
| 		return parsedMarkdownDoc{}, err
 | |
| 	}
 | |
| 	return parsedMarkdownDoc{Meta: meta, Body: body}, nil
 | |
| }
 | |
| 
 | |
| // funcListFiles reads and returns a slice of names from the given
 | |
| // directory relative to the root of c.
 | |
| func (c TemplateContext) funcListFiles(name string) ([]string, error) {
 | |
| 	if c.Root == nil {
 | |
| 		return nil, fmt.Errorf("root file system not specified")
 | |
| 	}
 | |
| 
 | |
| 	dir, err := c.Root.Open(path.Clean(name))
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	defer dir.Close()
 | |
| 
 | |
| 	stat, err := dir.Stat()
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	if !stat.IsDir() {
 | |
| 		return nil, fmt.Errorf("%v is not a directory", name)
 | |
| 	}
 | |
| 
 | |
| 	dirInfo, err := dir.Readdir(0)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 
 | |
| 	names := make([]string, len(dirInfo))
 | |
| 	for i, fileInfo := range dirInfo {
 | |
| 		names[i] = fileInfo.Name()
 | |
| 	}
 | |
| 
 | |
| 	return names, nil
 | |
| }
 | |
| 
 | |
| // funcFileExists returns true if filename can be opened successfully.
 | |
| func (c TemplateContext) funcFileExists(filename string) (bool, error) {
 | |
| 	if c.Root == nil {
 | |
| 		return false, fmt.Errorf("root file system not specified")
 | |
| 	}
 | |
| 	file, err := c.Root.Open(filename)
 | |
| 	if err == nil {
 | |
| 		file.Close()
 | |
| 		return true, nil
 | |
| 	}
 | |
| 	return false, nil
 | |
| }
 | |
| 
 | |
| // funcHTTPError returns a structured HTTP handler error. EXPERIMENTAL.
 | |
| // TODO: Requires https://github.com/golang/go/issues/34201 to be fixed (Go 1.17).
 | |
| // Example usage might be: `{{if not (fileExists $includeFile)}}{{httpError 404}}{{end}}`
 | |
| func (c TemplateContext) funcHTTPError(statusCode int) (bool, error) {
 | |
| 	return false, caddyhttp.Error(statusCode, nil)
 | |
| }
 | |
| 
 | |
| // WrappedHeader wraps niladic functions so that they
 | |
| // can be used in templates. (Template functions must
 | |
| // return a value.)
 | |
| type WrappedHeader struct{ http.Header }
 | |
| 
 | |
| // Add adds a header field value, appending val to
 | |
| // existing values for that field. It returns an
 | |
| // empty string.
 | |
| func (h WrappedHeader) Add(field, val string) string {
 | |
| 	h.Header.Add(field, val)
 | |
| 	return ""
 | |
| }
 | |
| 
 | |
| // Set sets a header field value, overwriting any
 | |
| // other values for that field. It returns an
 | |
| // empty string.
 | |
| func (h WrappedHeader) Set(field, val string) string {
 | |
| 	h.Header.Set(field, val)
 | |
| 	return ""
 | |
| }
 | |
| 
 | |
| // Del deletes a header field. It returns an empty string.
 | |
| func (h WrappedHeader) Del(field string) string {
 | |
| 	h.Header.Del(field)
 | |
| 	return ""
 | |
| }
 | |
| 
 | |
| func toString(input interface{}) string {
 | |
| 	switch v := input.(type) {
 | |
| 	case string:
 | |
| 		return v
 | |
| 	case fmt.Stringer:
 | |
| 		return v.String()
 | |
| 	case error:
 | |
| 		return v.Error()
 | |
| 	default:
 | |
| 		return fmt.Sprintf("%v", input)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| var bufPool = sync.Pool{
 | |
| 	New: func() interface{} {
 | |
| 		return new(bytes.Buffer)
 | |
| 	},
 | |
| }
 | |
| 
 | |
| // at time of writing, sprig.FuncMap() makes a copy, thus
 | |
| // involves iterating the whole map, so do it just once
 | |
| var sprigFuncMap = sprig.TxtFuncMap()
 | |
| 
 | |
| const recursionPreventionHeader = "Caddy-Templates-Include"
 |