Add a Lua API to set template filters

This commit is contained in:
Simon Ser 2019-12-09 17:54:24 +01:00
parent f42cb45457
commit 76599232dc
No known key found for this signature in database
GPG key ID: 0FDE7BE0E88F5E48
3 changed files with 50 additions and 11 deletions

View file

@ -2,6 +2,7 @@ package koushin
import ( import (
"fmt" "fmt"
"html/template"
"path/filepath" "path/filepath"
"github.com/labstack/echo/v4" "github.com/labstack/echo/v4"
@ -11,6 +12,7 @@ import (
type Plugin interface { type Plugin interface {
Name() string Name() string
Filters() template.FuncMap
Render(name string, data interface{}) error Render(name string, data interface{}) error
Close() error Close() error
} }
@ -19,6 +21,7 @@ type luaPlugin struct {
filename string filename string
state *lua.LState state *lua.LState
renderCallbacks map[string]*lua.LFunction renderCallbacks map[string]*lua.LFunction
filters template.FuncMap
} }
func (p *luaPlugin) Name() string { func (p *luaPlugin) Name() string {
@ -32,23 +35,53 @@ func (p *luaPlugin) onRender(l *lua.LState) int {
return 0 return 0
} }
func (p *luaPlugin) setFilter(l *lua.LState) int {
name := l.CheckString(1)
f := l.CheckFunction(2)
p.filters[name] = func(args... interface{}) string {
luaArgs := make([]lua.LValue, len(args))
for i, v := range args {
luaArgs[i] = luar.New(l, v)
}
err := l.CallByParam(lua.P{
Fn: f,
NRet: 1,
Protect: true,
}, luaArgs...)
if err != nil {
panic(err) // TODO: better error handling?
}
ret := l.CheckString(-1)
l.Pop(1)
return ret
}
return 0
}
func (p *luaPlugin) Render(name string, data interface{}) error { func (p *luaPlugin) Render(name string, data interface{}) error {
f, ok := p.renderCallbacks[name] f, ok := p.renderCallbacks[name]
if !ok { if !ok {
return nil return nil
} }
if err := p.state.CallByParam(lua.P{ err := p.state.CallByParam(lua.P{
Fn: f, Fn: f,
NRet: 0, NRet: 0,
Protect: true, Protect: true,
}, luar.New(p.state, data)); err != nil { }, luar.New(p.state, data))
if err != nil {
return err return err
} }
return nil return nil
} }
func (p *luaPlugin) Filters() template.FuncMap {
return p.filters
}
func (p *luaPlugin) Close() error { func (p *luaPlugin) Close() error {
p.state.Close() p.state.Close()
return nil return nil
@ -60,12 +93,13 @@ func loadLuaPlugin(filename string) (*luaPlugin, error) {
filename: filename, filename: filename,
state: l, state: l,
renderCallbacks: make(map[string]*lua.LFunction), renderCallbacks: make(map[string]*lua.LFunction),
filters: make(template.FuncMap),
} }
mt := l.NewTypeMetatable("koushin") mt := l.NewTypeMetatable("koushin")
l.SetGlobal("koushin", mt) l.SetGlobal("koushin", mt)
l.SetField(mt, "on_render", l.NewFunction(p.onRender)) l.SetField(mt, "on_render", l.NewFunction(p.onRender))
// TODO: set_filter l.SetField(mt, "set_filter", l.NewFunction(p.setFilter))
if err := l.DoFile(filename); err != nil { if err := l.DoFile(filename); err != nil {
l.Close() l.Close()

View file

@ -128,16 +128,16 @@ func New(e *echo.Echo, options *Options) error {
return err return err
} }
e.Renderer, err = loadTemplates(e.Logger, options.Theme)
if err != nil {
return fmt.Errorf("failed to load templates: %v", err)
}
s.plugins, err = loadAllLuaPlugins(e.Logger) s.plugins, err = loadAllLuaPlugins(e.Logger)
if err != nil { if err != nil {
return fmt.Errorf("failed to load plugins: %v", err) return fmt.Errorf("failed to load plugins: %v", err)
} }
e.Renderer, err = loadTemplates(e.Logger, options.Theme, s.plugins)
if err != nil {
return fmt.Errorf("failed to load templates: %v", err)
}
e.HTTPErrorHandler = func(err error, c echo.Context) { e.HTTPErrorHandler = func(err error, c echo.Context) {
code := http.StatusInternalServerError code := http.StatusInternalServerError
if he, ok := err.(*echo.HTTPError); ok { if he, ok := err.(*echo.HTTPError); ok {

View file

@ -27,15 +27,20 @@ func (t *tmpl) Render(w io.Writer, name string, data interface{}, ectx echo.Cont
return t.t.ExecuteTemplate(w, name, data) return t.t.ExecuteTemplate(w, name, data)
} }
func loadTemplates(logger echo.Logger, themeName string) (*tmpl, error) { func loadTemplates(logger echo.Logger, themeName string, plugins []Plugin) (*tmpl, error) {
base, err := template.New("").Funcs(template.FuncMap{ base := template.New("").Funcs(template.FuncMap{
"tuple": func(values ...interface{}) []interface{} { "tuple": func(values ...interface{}) []interface{} {
return values return values
}, },
"pathescape": func(s string) string { "pathescape": func(s string) string {
return url.PathEscape(s) return url.PathEscape(s)
}, },
}).ParseGlob("public/*.html") })
for _, p := range plugins {
base = base.Funcs(p.Filters())
}
base, err := base.ParseGlob("public/*.html")
if err != nil { if err != nil {
return nil, err return nil, err
} }