yoake/internal/servetpl/middleware.go

181 lines
4.7 KiB
Go

package servetpl
import (
"bytes"
"errors"
"fmt"
"html/template"
"io"
"log"
"mime"
"net/http"
"os"
"path"
"path/filepath"
"strings"
textTemplate "text/template"
"github.com/eternal-flame-AD/yoake/config"
"github.com/eternal-flame-AD/yoake/internal/servetpl/funcmap"
"github.com/eternal-flame-AD/yoake/internal/session"
"github.com/labstack/echo/v4"
)
type TemplatePath struct {
File string
Name string
}
type Context struct {
CleanPath string
Config func() config.C
C echo.Context
Request *http.Request
Response *echo.Response
WriteHeaders func(headers ...string) error
Session session.Provider
Global map[string]interface{}
}
type bodyBuffer struct {
resp *echo.Response
bodyBuf bytes.Buffer
committed bool
}
func (b *bodyBuffer) Write(p []byte) (int, error) {
if !b.committed {
return b.bodyBuf.Write(p)
}
return b.resp.Write(p)
}
func (b *bodyBuffer) WriteHeader(headers ...string) error {
if b.committed {
return nil
}
for _, header := range headers {
h := strings.SplitN(header, ":", 2)
if len(h) != 2 {
return fmt.Errorf("invalid header %s", header)
}
h[0] = strings.TrimSpace(h[0])
h[1] = strings.TrimSpace(h[1])
b.resp.Header().Set(h[0], h[1])
}
b.committed = true
if _, err := io.Copy(b.resp, &b.bodyBuf); err != nil {
return err
}
return nil
}
var (
errUndefinedTemplate = errors.New("undefined template")
errTplExtNotStripped = errors.New("this is a template file and should be requested without the template extension inflix")
)
func ServeTemplateDir(dir string) echo.MiddlewareFunc {
dir = filepath.Clean(dir)
var tplFiles []TemplatePath
if err := filepath.Walk(dir, func(file string, stat os.FileInfo, err error) error {
if err != nil {
return err
}
if stat.IsDir() {
return nil
}
ext := path.Ext(file)
secondExt := path.Ext(strings.TrimSuffix(file, ext))
if secondExt == ".tpl" {
relPath, err := filepath.Rel(dir, file)
if err != nil {
return err
}
tplFiles = append(tplFiles, TemplatePath{File: file, Name: "/" + relPath})
}
return nil
}); err != nil {
log.Panicf("templates failed to parse: %s", err)
}
templates := template.New("").Funcs(funcmap.GetFuncMap())
textTemplates := textTemplate.New("").Funcs(funcmap.GetFuncMap())
for _, file := range tplFiles {
log.Printf("parsing template: %s", file.Name)
if path.Ext(file.File) == ".html" {
if _, err := ParseTemplateFileAs[template.FuncMap](templates, file.Name, file.File); err != nil {
log.Panicf("templates failed to parse: %s", err)
}
} else {
if _, err := ParseTemplateFileAs[textTemplate.FuncMap](textTemplates, file.Name, file.File); err != nil {
log.Panicf("templates failed to parse: %s", err)
}
}
}
dispatchTemplate := func(file string) func(wr io.Writer, data any) error {
ext := path.Ext(file)
tplName := file[:len(file)-len(ext)] + ".tpl" + ext
if path.Ext(file[:len(file)-len(ext)]) == ".tpl" {
/* reject requests for the template source file */
log.Printf("rejecting request for template source file: %s", file)
if _, err := os.Stat(filepath.Join(dir, file)); err == nil {
return func(wr io.Writer, data any) error {
return errTplExtNotStripped
}
}
}
tplPath := filepath.Join(dir, tplName)
if _, err := os.Stat(tplPath); err == nil {
//log.Printf("dispatch template: %s(%s) ext=%s", tplName, tplPath, ext)
// template file is still there, execute
if ext == ".html" {
return func(wr io.Writer, data any) error { return templates.ExecuteTemplate(wr, tplName, data) }
} else {
return func(wr io.Writer, data any) error { return textTemplates.ExecuteTemplate(wr, tplName, data) }
}
}
return func(wr io.Writer, data any) error { return errUndefinedTemplate }
}
return func(next echo.HandlerFunc) echo.HandlerFunc {
return func(c echo.Context) error {
req, resp := c.Request(), c.Response()
p := path.Clean("/" + req.URL.Path)
ext := path.Ext(p)
c.Response().Header().Set(echo.HeaderContentType, mime.TypeByExtension(ext))
body := &bodyBuffer{resp: resp}
defer body.WriteHeader()
sess, sessClose := session.ManagedSession(c)
defer sessClose()
if err := dispatchTemplate(p)(body, Context{
Config: config.Config,
C: c,
CleanPath: p,
Request: req,
Response: resp,
WriteHeaders: body.WriteHeader,
Session: sess,
Global: map[string]interface{}{},
}); err == errUndefinedTemplate {
return next(c)
} else if errors.Is(err, funcmap.ErrEarlyTermination) {
return nil
} else if errors.Is(err, errTplExtNotStripped) {
c.String(http.StatusBadRequest, err.Error())
} else if err != nil {
c.Response().Write([]byte(fmt.Sprintf("<!-- ERROR: %v -->", err)))
return err
}
return nil
}
}
}