renderer.go 2.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127
  1. package renderer
  2. import (
  3. "encoding/json"
  4. "fmt"
  5. "html/template"
  6. "net/http"
  7. "path"
  8. "path/filepath"
  9. "reflect"
  10. "strings"
  11. )
  12. type Renderer interface {
  13. Render(http.ResponseWriter, *http.Request, interface{}, ...string) error
  14. }
  15. type JSONRenderer struct{}
  16. var (
  17. currRenderer Renderer
  18. Render map[string]func(http.ResponseWriter, *http.Request, interface{}, ...string)
  19. )
  20. func init() {
  21. htmlRenderer, err := NewHTMLRenderer("templates/")
  22. if err != nil {
  23. panic(err)
  24. }
  25. Render = make(map[string]func(http.ResponseWriter, *http.Request, interface{}, ...string))
  26. Render["html"] = func(w http.ResponseWriter, r *http.Request, data interface{}, options ...string) {
  27. htmlRenderer.Render(w, r, data, options...)
  28. }
  29. }
  30. // FIXME: Not safe. We should check if it responds to the error interface instead.
  31. func isErrorType(data interface{}) bool {
  32. return strings.Contains(strings.ToLower(reflect.TypeOf(data).String()), "error")
  33. }
  34. func NewJSONRenderer() *JSONRenderer {
  35. return &JSONRenderer{}
  36. }
  37. func (r *JSONRenderer) Render(w http.ResponseWriter, layout string, name string, data interface{}) error {
  38. w.Header().Set("Content-Type", "application/json; charset=utf-8")
  39. j, err := json.Marshal(data)
  40. if err != nil {
  41. return err
  42. }
  43. w.Write(j)
  44. return nil
  45. }
  46. type HTMLRenderer struct {
  47. TemplatePath string
  48. templates map[string]*template.Template
  49. }
  50. func NewHTMLRenderer(templatePath string) (*HTMLRenderer, error) {
  51. r := &HTMLRenderer{
  52. TemplatePath: templatePath,
  53. templates: make(map[string]*template.Template),
  54. }
  55. fns, err := filepath.Glob(filepath.Join(templatePath, "*.tpl"))
  56. if err != nil {
  57. panic(err)
  58. }
  59. lfns, err := filepath.Glob(filepath.Join(templatePath, "layout", "*.tpl"))
  60. if err != nil {
  61. panic(err)
  62. }
  63. for _, fn := range fns {
  64. tplName := filepath.Base(fn)
  65. tplName = strings.TrimSuffix(tplName, path.Ext(tplName))
  66. tplName = strings.TrimSuffix(tplName, path.Ext(tplName))
  67. files := append(lfns, fn)
  68. if err != nil {
  69. return nil, err
  70. }
  71. r.templates[tplName] = template.Must(r.templates[tplName].ParseFiles(files...))
  72. if err != nil {
  73. return nil, err
  74. }
  75. }
  76. return r, nil
  77. }
  78. func Use(r Renderer) {
  79. currRenderer = r
  80. }
  81. func (rend *HTMLRenderer) writeError(w http.ResponseWriter, r *http.Request, err error) {
  82. t, ok := rend.templates["error"]
  83. if !ok {
  84. panic(fmt.Errorf("Error template not found! Can't proceed, sorry."))
  85. }
  86. w.Header().Set("Content-Type", "text/html; charset=utf-8")
  87. e := t.ExecuteTemplate(w, "base", err)
  88. if e != nil {
  89. panic(e)
  90. }
  91. }
  92. func (rend *HTMLRenderer) Render(w http.ResponseWriter, r *http.Request, data interface{}, options ...string) {
  93. if isErrorType(data) {
  94. rend.writeError(w, r, data.(error))
  95. } else {
  96. t, ok := rend.templates[options[1]]
  97. if !ok {
  98. rend.writeError(w, r, fmt.Errorf("Template %s not found", options[1]))
  99. }
  100. w.Header().Set("Content-Type", "text/html; charset=utf-8")
  101. err := t.ExecuteTemplate(w, options[0], data)
  102. if err != nil {
  103. rend.writeError(w, r, err)
  104. }
  105. }
  106. }