Skip to content

Commit cd66d2c

Browse files
committed
Merge pull request #14 from bouk/add-html-escaping
Add support for HTML escaped string outputting
2 parents 7c661d7 + c5c411f commit cd66d2c

File tree

5 files changed

+82
-11
lines changed

5 files changed

+82
-11
lines changed

README.md

+3-1
Original file line numberDiff line numberDiff line change
@@ -26,7 +26,9 @@ An ego template is made up of several types of blocks:
2626

2727
* **Code Block** - These blocks execute raw Go code: `<% var foo = "bar" %>`
2828

29-
* **Print Block** - These blocks print a Go expression: `<%= myVar %>`
29+
* **Print Block** - These blocks print a Go expression. They use `html.EscapeString` to escape it before outputting: `<%= myVar %>`
30+
31+
* **Raw Print Block** - These blocks print a Go expression raw into the HTML: `<%== "<script>" %>`
3032

3133
* **Header Block** - These blocks allow you to import packages: `<%% import "encoding/json" %%>`
3234

scanner.go

+22-1
Original file line numberDiff line numberDiff line change
@@ -63,7 +63,18 @@ func (s *Scanner) scanCodeBlock() (Block, error) {
6363
case '%':
6464
return s.scanHeaderBlock()
6565
case '=':
66-
return s.scanPrintBlock()
66+
ch, err := s.read()
67+
if err == io.EOF {
68+
return nil, io.ErrUnexpectedEOF
69+
} else if err != nil {
70+
return nil, err
71+
}
72+
if ch == '=' {
73+
return s.scanRawPrintBlock()
74+
} else {
75+
s.unread()
76+
return s.scanPrintBlock()
77+
}
6778
}
6879

6980
// Otherwise read the contents of the code block.
@@ -98,6 +109,16 @@ func (s *Scanner) scanHeaderBlock() (Block, error) {
98109
return b, nil
99110
}
100111

112+
func (s *Scanner) scanRawPrintBlock() (Block, error) {
113+
b := &RawPrintBlock{Pos: s.pos}
114+
content, err := s.scanContent()
115+
if err != nil {
116+
return nil, err
117+
}
118+
b.Content = content
119+
return b, nil
120+
}
121+
101122
func (s *Scanner) scanPrintBlock() (Block, error) {
102123
b := &PrintBlock{Pos: s.pos}
103124
content, err := s.scanContent()

scanner_test.go

+23-5
Original file line numberDiff line numberDiff line change
@@ -79,6 +79,13 @@ func TestScannerCodeBlockUnexpectedEOF_4(t *testing.T) {
7979
assert.Equal(t, err, io.ErrUnexpectedEOF)
8080
}
8181

82+
// Ensure that a print code block that ends unexpectedly returns an error.
83+
func TestScannerCodeBlockUnexpectedEOF_5(t *testing.T) {
84+
s := NewScanner(bytes.NewBufferString(`<%=`), "tmpl.ego")
85+
_, err := s.Scan()
86+
assert.Equal(t, err, io.ErrUnexpectedEOF)
87+
}
88+
8289
// Ensure that a header block can be scanned.
8390
func TestScannerHeaderBlock(t *testing.T) {
8491
s := NewScanner(bytes.NewBufferString(`<%% import "foo" %%>`), "tmpl.ego")
@@ -127,22 +134,33 @@ func TestScannerHeaderBlockUnexpectedEOF_5(t *testing.T) {
127134

128135
// Ensure that a print block can be scanned.
129136
func TestScannerPrintBlock(t *testing.T) {
130-
s := NewScanner(bytes.NewBufferString(`<%= myNum %>`), "tmpl.ego")
137+
s := NewScanner(bytes.NewBufferString(`<%== myNum %>`), "tmpl.ego")
131138
b, err := s.Scan()
132139
assert.NoError(t, err)
133-
if b, ok := b.(*PrintBlock); assert.True(t, ok) {
140+
if b, ok := b.(*RawPrintBlock); assert.True(t, ok) {
134141
assert.Equal(t, b.Content, ` myNum `)
135142
assert.Equal(t, b.Pos, Pos{Path: "tmpl.ego", LineNo: 1})
136143
}
137144
}
138145

139146
// Ensure that a print block that ends unexpectedly returns an error.
140147
func TestScannerPrintBlockUnexpectedEOF(t *testing.T) {
141-
s := NewScanner(bytes.NewBufferString(`<%= `), "tmpl.ego")
148+
s := NewScanner(bytes.NewBufferString(`<%== `), "tmpl.ego")
142149
_, err := s.Scan()
143150
assert.Equal(t, err, io.ErrUnexpectedEOF)
144151
}
145152

153+
// Ensure that an escaped print block can be scanned.
154+
func TestScannerEscapedPrintBlock(t *testing.T) {
155+
s := NewScanner(bytes.NewBufferString(`<%= myNum %>`), "tmpl.ego")
156+
b, err := s.Scan()
157+
assert.NoError(t, err)
158+
if b, ok := b.(*PrintBlock); assert.True(t, ok) {
159+
assert.Equal(t, b.Content, ` myNum `)
160+
assert.Equal(t, b.Pos, Pos{Path: "tmpl.ego", LineNo: 1})
161+
}
162+
}
163+
146164
// Ensure that a declaration block can be scanned.
147165
func TestScannerDeclarationBlock(t *testing.T) {
148166
s := NewScanner(bytes.NewBufferString(`<%! MyTemplate(w io.Writer) error %>`), "tmpl.ego")
@@ -163,11 +181,11 @@ func TestScannerDeclarationBlockUnexpectedEOF(t *testing.T) {
163181

164182
// Ensure that line numbers are tracked correctly.
165183
func TestScannerMultiline(t *testing.T) {
166-
s := NewScanner(bytes.NewBufferString("hello\nworld<%= x \n\n %>goodbye"), "tmpl.ego")
184+
s := NewScanner(bytes.NewBufferString("hello\nworld<%== x \n\n %>goodbye"), "tmpl.ego")
167185
b, _ := s.Scan()
168186
assert.Equal(t, b.(*TextBlock).Pos, Pos{Path: "tmpl.ego", LineNo: 1})
169187
b, _ = s.Scan()
170-
assert.Equal(t, b.(*PrintBlock).Pos, Pos{Path: "tmpl.ego", LineNo: 2})
188+
assert.Equal(t, b.(*RawPrintBlock).Pos, Pos{Path: "tmpl.ego", LineNo: 2})
171189
b, _ = s.Scan()
172190
assert.Equal(t, b.(*TextBlock).Pos, Pos{Path: "tmpl.ego", LineNo: 4})
173191
}

template.go

+33-3
Original file line numberDiff line numberDiff line change
@@ -14,7 +14,7 @@ import (
1414

1515
// Template represents an entire Ego template.
1616
// A template consists of a declaration block followed by zero or more blocks.
17-
// Blocks can be either a TextBlock, a PrintBlock, or a CodeBlock.
17+
// Blocks can be either a TextBlock, a PrintBlock, a RawPrintBlock, or a CodeBlock.
1818
type Template struct {
1919
Path string
2020
Blocks []Block
@@ -79,6 +79,15 @@ func (t *Template) nonHeaderBlocks() []Block {
7979
return blocks
8080
}
8181

82+
func (t *Template) hasEscapedPrintBlock() bool {
83+
for _, b := range t.Blocks {
84+
if _, ok := b.(*PrintBlock); ok {
85+
return true
86+
}
87+
}
88+
return false
89+
}
90+
8291
// normalize joins together adjacent text blocks.
8392
func (t *Template) normalize() {
8493
var a []Block
@@ -102,6 +111,7 @@ func (b *DeclarationBlock) block() {}
102111
func (b *TextBlock) block() {}
103112
func (b *CodeBlock) block() {}
104113
func (b *HeaderBlock) block() {}
114+
func (b *RawPrintBlock) block() {}
105115
func (b *PrintBlock) block() {}
106116

107117
// DeclarationBlock represents a block that declaration the function signature.
@@ -158,15 +168,27 @@ func (b *HeaderBlock) write(buf *bytes.Buffer) error {
158168
return nil
159169
}
160170

161-
// PrintBlock represents a block of the template that is printed out to the writer.
171+
// RawPrintBlock represents a block of the template that is printed out to the writer.
172+
type RawPrintBlock struct {
173+
Pos Pos
174+
Content string
175+
}
176+
177+
func (b *RawPrintBlock) write(buf *bytes.Buffer) error {
178+
b.Pos.write(buf)
179+
fmt.Fprintf(buf, `_, _ = fmt.Fprintf(w, "%%v", %s)`+"\n", b.Content)
180+
return nil
181+
}
182+
183+
// PrintBlock represents a block that will HTML escape the contents before outputting
162184
type PrintBlock struct {
163185
Pos Pos
164186
Content string
165187
}
166188

167189
func (b *PrintBlock) write(buf *bytes.Buffer) error {
168190
b.Pos.write(buf)
169-
fmt.Fprintf(buf, `_, _ = fmt.Fprintf(w, "%%v", %s)`+"\n", b.Content)
191+
fmt.Fprintf(buf, `_, _ = fmt.Fprint(w, html.EscapeString(fmt.Sprintf("%%v", %s)))`+"\n", b.Content)
170192
return nil
171193
}
172194

@@ -237,7 +259,15 @@ func (p *Package) writeHeader(w io.Writer) error {
237259
var decls = map[string]bool{`:"fmt"`: true, `:"io"`: true}
238260
fmt.Fprint(&buf, "import (\n")
239261
fmt.Fprintln(&buf, `"fmt"`)
262+
for _, t := range p.Templates {
263+
if t.hasEscapedPrintBlock() {
264+
fmt.Fprintln(&buf, `"html"`)
265+
decls["html"] = true
266+
break
267+
}
268+
}
240269
fmt.Fprintln(&buf, `"io"`)
270+
241271
for _, d := range f.Decls {
242272
d, ok := d.(*ast.GenDecl)
243273
if !ok || d.Tok != token.IMPORT {

template_test.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -20,7 +20,7 @@ func TestTemplate_Write(t *testing.T) {
2020
&DeclarationBlock{Content: " func MyTemplate(w io.Writer, nums []int) error "},
2121
&CodeBlock{Content: " for _, num := range nums {"},
2222
&TextBlock{Content: " <p>"},
23-
&PrintBlock{Content: "num + 1"},
23+
&RawPrintBlock{Content: "num + 1"},
2424
&TextBlock{Content: " </p>"},
2525
&CodeBlock{Content: " }"},
2626
&TextBlock{Content: "</html>"},

0 commit comments

Comments
 (0)