A webmail client. Forked from https://git.sr.ht/~migadu/alps
Nie możesz wybrać więcej, niż 25 tematów Tematy muszą się zaczynać od litery lub cyfry, mogą zawierać myślniki ('-') i mogą mieć do 35 znaków.
 
 
 
 

181 wiersze
3.7 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 luaRoute struct {
  11. method string
  12. path string
  13. f *lua.LFunction
  14. }
  15. type luaPlugin struct {
  16. filename string
  17. state *lua.LState
  18. renderCallbacks map[string]*lua.LFunction
  19. filters template.FuncMap
  20. routes []luaRoute
  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) setRoute(l *lua.LState) int {
  54. method := l.CheckString(1)
  55. path := l.CheckString(2)
  56. f := l.CheckFunction(3)
  57. p.routes = append(p.routes, luaRoute{method, path, f})
  58. return 0
  59. }
  60. func (p *luaPlugin) inject(name string, data RenderData) error {
  61. f, ok := p.renderCallbacks[name]
  62. if !ok {
  63. return nil
  64. }
  65. err := p.state.CallByParam(lua.P{
  66. Fn: f,
  67. NRet: 0,
  68. Protect: true,
  69. }, luar.New(p.state, data))
  70. if err != nil {
  71. return err
  72. }
  73. return nil
  74. }
  75. func (p *luaPlugin) Inject(ctx *Context, name string, data RenderData) error {
  76. if err := p.inject("*", data); err != nil {
  77. return err
  78. }
  79. return p.inject(name, data)
  80. }
  81. func (p *luaPlugin) LoadTemplate(t *template.Template) error {
  82. t.Funcs(p.filters)
  83. paths, err := filepath.Glob(filepath.Dir(p.filename) + "/public/*.html")
  84. if err != nil {
  85. return err
  86. }
  87. if len(paths) > 0 {
  88. if _, err := t.ParseFiles(paths...); err != nil {
  89. return err
  90. }
  91. }
  92. return nil
  93. }
  94. func (p *luaPlugin) SetRoutes(group *echo.Group) {
  95. for _, r := range p.routes {
  96. group.Match([]string{r.method}, r.path, func(ctx echo.Context) error {
  97. err := p.state.CallByParam(lua.P{
  98. Fn: r.f,
  99. NRet: 0,
  100. Protect: true,
  101. }, luar.New(p.state, ctx))
  102. if err != nil {
  103. return fmt.Errorf("Lua plugin error: %v", err)
  104. }
  105. return nil
  106. })
  107. }
  108. _, name := filepath.Split(filepath.Dir(p.filename))
  109. group.Static("/plugins/"+name+"/assets", filepath.Dir(p.filename)+"/public/assets")
  110. }
  111. func (p *luaPlugin) Close() error {
  112. p.state.Close()
  113. return nil
  114. }
  115. func loadLuaPlugin(filename string) (*luaPlugin, error) {
  116. l := lua.NewState()
  117. p := &luaPlugin{
  118. filename: filename,
  119. state: l,
  120. renderCallbacks: make(map[string]*lua.LFunction),
  121. filters: make(template.FuncMap),
  122. }
  123. mt := l.NewTypeMetatable("koushin")
  124. l.SetGlobal("koushin", mt)
  125. l.SetField(mt, "on_render", l.NewFunction(p.onRender))
  126. l.SetField(mt, "set_filter", l.NewFunction(p.setFilter))
  127. l.SetField(mt, "set_route", l.NewFunction(p.setRoute))
  128. if err := l.DoFile(filename); err != nil {
  129. l.Close()
  130. return nil, err
  131. }
  132. return p, nil
  133. }
  134. func loadAllLuaPlugins(log echo.Logger) ([]Plugin, error) {
  135. filenames, err := filepath.Glob(pluginDir + "/*/main.lua")
  136. if err != nil {
  137. return nil, fmt.Errorf("filepath.Glob failed: %v", err)
  138. }
  139. plugins := make([]Plugin, 0, len(filenames))
  140. for _, filename := range filenames {
  141. log.Printf("Loading Lua plugin '%v'", filename)
  142. p, err := loadLuaPlugin(filename)
  143. if err != nil {
  144. for _, p := range plugins {
  145. p.Close()
  146. }
  147. return nil, fmt.Errorf("failed to load Lua plugin '%v': %v", filename, err)
  148. }
  149. plugins = append(plugins, p)
  150. }
  151. return plugins, nil
  152. }