A webmail client. Forked from https://git.sr.ht/~migadu/alps
Non puoi selezionare più di 25 argomenti Gli argomenti devono iniziare con una lettera o un numero, possono includere trattini ('-') e possono essere lunghi fino a 35 caratteri.
 
 
 
 

133 righe
2.6 KiB

  1. package koushin
  2. import (
  3. "fmt"
  4. "html/template"
  5. "path/filepath"
  6. "github.com/labstack/echo/v4"
  7. "github.com/yuin/gopher-lua"
  8. "layeh.com/gopher-luar"
  9. )
  10. type Plugin interface {
  11. Name() string
  12. Filters() template.FuncMap
  13. Render(name string, data interface{}) error
  14. Close() error
  15. }
  16. type luaPlugin struct {
  17. filename string
  18. state *lua.LState
  19. renderCallbacks map[string]*lua.LFunction
  20. filters template.FuncMap
  21. }
  22. func (p *luaPlugin) Name() string {
  23. return p.filename
  24. }
  25. func (p *luaPlugin) onRender(l *lua.LState) int {
  26. name := l.CheckString(1)
  27. f := l.CheckFunction(2)
  28. p.renderCallbacks[name] = f
  29. return 0
  30. }
  31. func (p *luaPlugin) setFilter(l *lua.LState) int {
  32. name := l.CheckString(1)
  33. f := l.CheckFunction(2)
  34. p.filters[name] = func(args... interface{}) string {
  35. luaArgs := make([]lua.LValue, len(args))
  36. for i, v := range args {
  37. luaArgs[i] = luar.New(l, v)
  38. }
  39. err := l.CallByParam(lua.P{
  40. Fn: f,
  41. NRet: 1,
  42. Protect: true,
  43. }, luaArgs...)
  44. if err != nil {
  45. panic(err) // TODO: better error handling?
  46. }
  47. ret := l.CheckString(-1)
  48. l.Pop(1)
  49. return ret
  50. }
  51. return 0
  52. }
  53. func (p *luaPlugin) Render(name string, data interface{}) error {
  54. f, ok := p.renderCallbacks[name]
  55. if !ok {
  56. return nil
  57. }
  58. err := p.state.CallByParam(lua.P{
  59. Fn: f,
  60. NRet: 0,
  61. Protect: true,
  62. }, luar.New(p.state, data))
  63. if err != nil {
  64. return err
  65. }
  66. return nil
  67. }
  68. func (p *luaPlugin) Filters() template.FuncMap {
  69. return p.filters
  70. }
  71. func (p *luaPlugin) Close() error {
  72. p.state.Close()
  73. return nil
  74. }
  75. func loadLuaPlugin(filename string) (*luaPlugin, error) {
  76. l := lua.NewState()
  77. p := &luaPlugin{
  78. filename: filename,
  79. state: l,
  80. renderCallbacks: make(map[string]*lua.LFunction),
  81. filters: make(template.FuncMap),
  82. }
  83. mt := l.NewTypeMetatable("koushin")
  84. l.SetGlobal("koushin", mt)
  85. l.SetField(mt, "on_render", l.NewFunction(p.onRender))
  86. l.SetField(mt, "set_filter", l.NewFunction(p.setFilter))
  87. if err := l.DoFile(filename); err != nil {
  88. l.Close()
  89. return nil, err
  90. }
  91. return p, nil
  92. }
  93. func loadAllLuaPlugins(log echo.Logger) ([]Plugin, error) {
  94. filenames, err := filepath.Glob("plugins/*.lua")
  95. if err != nil {
  96. return nil, fmt.Errorf("filepath.Glob failed: %v", err)
  97. }
  98. plugins := make([]Plugin, 0, len(filenames))
  99. for _, filename := range filenames {
  100. log.Printf("Loading Lua plugin '%v'", filename)
  101. p, err := loadLuaPlugin(filename)
  102. if err != nil {
  103. for _, p := range plugins {
  104. p.Close()
  105. }
  106. return nil, fmt.Errorf("failed to load Lua plugin '%v': %v", filename, err)
  107. }
  108. plugins = append(plugins, p)
  109. }
  110. return plugins, nil
  111. }