Add a Lua API to set template filters
This commit is contained in:
parent
f42cb45457
commit
76599232dc
3 changed files with 50 additions and 11 deletions
40
plugin.go
40
plugin.go
|
@ -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()
|
||||||
|
|
10
server.go
10
server.go
|
@ -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 {
|
||||||
|
|
11
template.go
11
template.go
|
@ -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
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue