Skip to content

Commit

Permalink
text/template: shut down lexing goroutine on error
Browse files Browse the repository at this point in the history
When a parse error occurred, the lexing goroutine would lay idle.
It's not likely a problem but if the program is for some reason
accepting badly formed data repeatedly, it's wasteful.

The solution is easy: Just drain the input on error. We know this
will succeed because the input is always a string and is therefore
guaranteed finite.

With debugging prints in the package tests I've shown this is effective,
shutting down 79 goroutines that would otherwise linger, out of 123 total.

Fixes #10574.

Change-Id: I8aa536e327b219189a7e7f604a116fa562ae1c39
Reviewed-on: https://go-review.googlesource.com/9658
Reviewed-by: Russ Cox <[email protected]>
  • Loading branch information
robpike committed May 5, 2015
1 parent 5a828cf commit 64c39a3
Show file tree
Hide file tree
Showing 3 changed files with 41 additions and 0 deletions.
12 changes: 12 additions & 0 deletions src/text/template/parse/lex.go
Original file line number Diff line number Diff line change
Expand Up @@ -167,12 +167,23 @@ func (l *lexer) errorf(format string, args ...interface{}) stateFn {
}

// nextItem returns the next item from the input.
// Called by the parser, not in the lexing goroutine.
func (l *lexer) nextItem() item {
item := <-l.items
l.lastPos = item.pos
return item
}

// drain drains the output so the lexing goroutine will exit.
// Called by the parser, not in the lexing goroutine.
func (l *lexer) drain() {
if l == nil {

This comment has been minimized.

Copy link
@gravis

gravis May 6, 2015

how l could be nil here?

This comment has been minimized.

Copy link
@robpike

robpike May 7, 2015

Author Contributor

It can't. Well spotted.

This was added during testing but is no longer necessary. See CL 9841.

return
}
for range l.items {
}
}

// lex creates a new scanner for the input string.
func lex(name, input, left, right string) *lexer {
if left == "" {
Expand All @@ -197,6 +208,7 @@ func (l *lexer) run() {
for l.state = lexText; l.state != nil; {
l.state = l.state(l)
}
close(l.items)
}

// state functions
Expand Down
28 changes: 28 additions & 0 deletions src/text/template/parse/lex_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -466,3 +466,31 @@ func TestPos(t *testing.T) {
}
}
}

// Test that an error shuts down the lexing goroutine.
func TestShutdown(t *testing.T) {
// We need to duplicate template.Parse here to hold on to the lexer.
const text = "erroneous{{define}}{{else}}1234"
lexer := lex("foo", text, "{{", "}}")
_, err := New("root").parseLexer(lexer, text)
if err == nil {
t.Fatalf("expected error")
}
// The error should have drained the input. Therefore, the lexer should be shut down.
token, ok := <-lexer.items
if ok {
t.Fatalf("input was not drained; got %v", token)
}
}

// parseLexer is a local version of parse that lets us pass in the lexer instead of building it.
// We expect an error, so the tree set and funcs list are explicitly nil.
func (t *Tree) parseLexer(lex *lexer, text string) (tree *Tree, err error) {
defer t.recover(&err)
t.ParseName = t.Name
t.startParse(nil, lex)
t.parse(nil)
t.add(nil)
t.stopParse()
return t, nil
}
1 change: 1 addition & 0 deletions src/text/template/parse/parse.go
Original file line number Diff line number Diff line change
Expand Up @@ -196,6 +196,7 @@ func (t *Tree) recover(errp *error) {
panic(e)
}
if t != nil {
t.lex.drain()
t.stopParse()
}
*errp = e.(error)
Expand Down

0 comments on commit 64c39a3

Please sign in to comment.