renderer.go 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203
  1. package renderer
  2. import (
  3. "encoding/json"
  4. "fmt"
  5. "html/template"
  6. "log"
  7. "net/http"
  8. "net/url"
  9. "path"
  10. "path/filepath"
  11. "reflect"
  12. "strings"
  13. "github.com/gorilla/schema"
  14. )
  15. type Renderer interface {
  16. Render(http.ResponseWriter, *http.Request, interface{}, ...url.Values) error
  17. }
  18. type JSONRenderer struct{}
  19. type htmlTemplateData struct {
  20. Data interface{}
  21. Options url.Values
  22. }
  23. type JsonResponse struct {
  24. Result []byte
  25. Error []byte
  26. }
  27. var (
  28. currRenderer Renderer
  29. Render map[string]func(http.ResponseWriter, *http.Request, interface{}, ...url.Values)
  30. funcMap = template.FuncMap{
  31. "query": query,
  32. }
  33. )
  34. func init() {
  35. htmlRenderer, err := NewHTMLRenderer("templates/")
  36. if err != nil {
  37. panic(err)
  38. }
  39. jsonRenderer, err := NewJSONRenderer()
  40. if err != nil {
  41. panic(err)
  42. }
  43. Render = make(map[string]func(http.ResponseWriter, *http.Request, interface{}, ...url.Values))
  44. Render["html"] = func(w http.ResponseWriter, r *http.Request, data interface{}, options ...url.Values) {
  45. htmlRenderer.Render(w, r, data, options...)
  46. }
  47. Render["json"] = func(w http.ResponseWriter, r *http.Request, data interface{}, options ...url.Values) {
  48. jsonRenderer.Render(w, r, data, options...)
  49. }
  50. }
  51. func query(values ...string) template.URL {
  52. var urlValues url.Values
  53. urlValues = make(url.Values)
  54. urlValues.Set("format", "html")
  55. for i := 0; i < len(values); i += 2 {
  56. urlValues.Add(values[i], values[i+1])
  57. }
  58. return template.URL(urlValues.Encode())
  59. }
  60. // FIXME: Not safe. We should check if it responds to the error interface instead.
  61. func isErrorType(data interface{}) bool {
  62. return strings.Contains(strings.ToLower(reflect.TypeOf(data).String()), "error")
  63. }
  64. func NewJSONRenderer() (*JSONRenderer, error) {
  65. return &JSONRenderer{}, nil
  66. }
  67. func (rend *JSONRenderer) Render(w http.ResponseWriter, r *http.Request, data interface{}, options ...url.Values) error {
  68. w.Header().Set("Content-Type", "application/json; charset=utf-8")
  69. if isErrorType(data) {
  70. j, err := json.Marshal(JsonResponse{nil, []byte(data.(error).Error())})
  71. if err != nil {
  72. return err
  73. }
  74. w.WriteHeader(http.StatusInternalServerError)
  75. w.Write(j)
  76. } else {
  77. result, err := json.Marshal(data)
  78. if err != nil {
  79. return err
  80. }
  81. response, err := json.Marshal(JsonResponse{result, nil})
  82. if err != nil {
  83. return err
  84. }
  85. w.Write(response)
  86. }
  87. return nil
  88. }
  89. type HTMLRenderer struct {
  90. TemplatePath string
  91. templates map[string]*template.Template
  92. }
  93. func NewHTMLRenderer(templatePath string) (*HTMLRenderer, error) {
  94. r := &HTMLRenderer{
  95. TemplatePath: templatePath,
  96. templates: make(map[string]*template.Template),
  97. }
  98. fns, err := filepath.Glob(filepath.Join(templatePath, "*.tpl"))
  99. if err != nil {
  100. panic(err)
  101. }
  102. lfns, err := filepath.Glob(filepath.Join(templatePath, "layout", "*.tpl"))
  103. if err != nil {
  104. panic(err)
  105. }
  106. for _, fn := range fns {
  107. tplName := filepath.Base(fn)
  108. tplName = strings.TrimSuffix(tplName, path.Ext(tplName))
  109. tplName = strings.TrimSuffix(tplName, path.Ext(tplName))
  110. files := append(lfns, fn)
  111. if err != nil {
  112. return nil, err
  113. }
  114. r.templates[tplName] = template.New(tplName)
  115. r.templates[tplName] = template.Must(r.templates[tplName].Funcs(funcMap).ParseFiles(files...))
  116. }
  117. return r, nil
  118. }
  119. func Use(r Renderer) {
  120. currRenderer = r
  121. }
  122. func (rend *HTMLRenderer) writeError(w http.ResponseWriter, r *http.Request, err error) {
  123. log.Println(rend.templates)
  124. t, ok := rend.templates["error"]
  125. if !ok {
  126. panic(fmt.Errorf("Error template not found! Can't proceed, sorry."))
  127. }
  128. w.Header().Set("Content-Type", "text/html; charset=utf-8")
  129. e := t.ExecuteTemplate(w, "base", err)
  130. if e != nil {
  131. panic(e)
  132. }
  133. }
  134. func (rend *HTMLRenderer) Render(w http.ResponseWriter, r *http.Request, data interface{}, options ...url.Values) {
  135. if data != nil && isErrorType(data) {
  136. rend.writeError(w, r, data.(error))
  137. } else {
  138. t, ok := rend.templates[options[0]["tpl_content"][0]]
  139. if !ok {
  140. rend.writeError(w, r, fmt.Errorf("Template %s not found", options[0]["tpl_content"][0]))
  141. }
  142. w.Header().Set("Content-Type", "text/html; charset=utf-8")
  143. err := t.ExecuteTemplate(w, options[0]["tpl_layout"][0], &htmlTemplateData{data, options[0]})
  144. if err != nil {
  145. rend.writeError(w, r, err)
  146. }
  147. }
  148. }
  149. func Decode(dst interface{}, r *http.Request) error {
  150. switch r.Header.Get("Content-type") {
  151. case "application/x-www-form-urlencoded":
  152. if err := r.ParseForm(); err != nil {
  153. return err
  154. }
  155. decoder := schema.NewDecoder()
  156. if err := decoder.Decode(dst, r.PostForm); err != nil {
  157. return err
  158. }
  159. case "application/json; charset=utf-8":
  160. decoder := json.NewDecoder(r.Body)
  161. err := decoder.Decode(&dst)
  162. if err != nil {
  163. return err
  164. }
  165. }
  166. return nil
  167. }