forked from dannyvankooten/grender
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathgrender.go
150 lines (118 loc) · 3.37 KB
/
grender.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
package grender
import (
"encoding/json"
"encoding/xml"
"fmt"
"html/template"
"net/http"
"regexp"
)
const (
// ContentHTML HTTP header value for HTML data
ContentHTML = "text/html"
// ContentJSON HTTP header value for JSON data
ContentJSON = "application/json"
// ContentType HTTP header name for defining the content type
ContentType = "Content-Type"
// ContentText header value for Text data.
ContentText = "text/plain"
// ContentXML header value for XML data.
ContentXML = "text/xml"
// DefaultCharset for when no specific Charset options was given
DefaultCharset = "UTF-8"
)
var extendsRegex *regexp.Regexp
// Grender provides functions for easily writing HTML templates & JSON out to a HTTP Response.
type Grender struct {
options Options
templates map[string]*template.Template
}
// Options holds the configuration options for a Renderer
type Options struct {
// With Debug set to true, templates will be recompiled before every render call.
Debug bool
// The glob string to your templates
TemplatesGlob string
// The Glob string for additional templates
PartialsGlob string
// The function map to pass to each HTML template
Funcs template.FuncMap
// Charset for responses
Charset string
}
func init() {
var err error
extendsRegex, err = regexp.Compile(`{{\/\*\s+extends\s+"(.*)"\s+\*\/}}`)
if err != nil {
panic(err)
}
}
// New creates a new Renderer with the given options
func New(optsarg ...Options) *Grender {
var opts Options
if len(optsarg) > 0 {
opts = optsarg[0]
} else {
opts = Options{}
}
if opts.Charset == "" {
opts.Charset = "UTF-8"
}
r := &Grender{
options: opts,
}
r.compileTemplatesFromDir()
return r
}
// HTML executes the template and writes to the responsewriter
func (r *Grender) HTML(w http.ResponseWriter, statusCode int, templateName string, data interface{}) error {
// re-compile on every render call when Debug is true
if r.options.Debug {
r.compileTemplatesFromDir()
}
tmpl, ok := r.templates[templateName]
if !ok {
return fmt.Errorf("unrecognised template %s", templateName)
}
// send response headers + body
w.Header().Set("Content-Type", ContentHTML+"; charset="+r.options.Charset)
out := bufPool.Get()
defer bufPool.Put(out)
// execute template
err := tmpl.Execute(out, data)
if err != nil {
return err
}
w.WriteHeader(statusCode)
out.WriteTo(w)
return nil
}
// JSON renders the data as a JSON HTTP response to the ResponseWriter
func (r *Grender) JSON(w http.ResponseWriter, statusCode int, data interface{}) error {
w.Header().Set("Content-Type", ContentJSON+"; charset="+r.options.Charset)
// do nothing if nil data
if data == nil {
return nil
}
w.WriteHeader(statusCode)
err := json.NewEncoder(w).Encode(data)
return err
}
// XML writes the data as a XML HTTP response to the ResponseWriter
func (r *Grender) XML(w http.ResponseWriter, statusCode int, data interface{}) error {
w.Header().Set("Content-Type", ContentXML+"; charset="+r.options.Charset)
// do nothing if nil data
if data == nil {
return nil
}
w.WriteHeader(statusCode)
err := xml.NewEncoder(w).Encode(data)
return err
}
// Text writes the data as a JSON HTTP response to the ResponseWriter
func (r *Grender) Text(w http.ResponseWriter, statusCode int, data string) error {
w.Header().Set("Content-Type", ContentText+"; charset="+r.options.Charset)
w.WriteHeader(statusCode)
w.Write([]byte(data))
return nil
}