1
0
mirror of https://github.com/golang/go synced 2024-11-22 03:34:40 -07:00

template: support string, int and float literals

This enables customizing the behavior of formatters
with logic such as {"template"|import} or even
{Field1 Field2 "%.2f 0x%X"|printf}

Thanks to Roger Peppe for some debate on this.

R=golang-dev, r, r
CC=golang-dev
https://golang.org/cl/4536059
This commit is contained in:
Gustavo Niemeyer 2011-05-19 09:24:27 -03:00
parent bddd092dc3
commit 29b246c644
2 changed files with 154 additions and 23 deletions

View File

@ -76,6 +76,10 @@
executed sequentially, with each formatter receiving the bytes executed sequentially, with each formatter receiving the bytes
emitted by the one to its left. emitted by the one to its left.
As well as field names, one may use literals with Go syntax.
Integer, floating-point, and string literals are supported.
Raw strings may not span newlines.
The delimiter strings get their default value, "{" and "}", from The delimiter strings get their default value, "{" and "}", from
JSON-template. They may be set to any non-empty, space-free JSON-template. They may be set to any non-empty, space-free
string using the SetDelims method. Their value can be printed string using the SetDelims method. Their value can be printed
@ -91,6 +95,7 @@ import (
"io/ioutil" "io/ioutil"
"os" "os"
"reflect" "reflect"
"strconv"
"strings" "strings"
"unicode" "unicode"
"utf8" "utf8"
@ -151,10 +156,13 @@ type literalElement struct {
// A variable invocation to be evaluated // A variable invocation to be evaluated
type variableElement struct { type variableElement struct {
linenum int linenum int
word []string // The fields in the invocation. args []interface{} // The fields and literals in the invocation.
fmts []string // Names of formatters to apply. len(fmts) > 0 fmts []string // Names of formatters to apply. len(fmts) > 0
} }
// A variableElement arg to be evaluated as a field name
type fieldName string
// A .section block, possibly with a .or // A .section block, possibly with a .or
type sectionElement struct { type sectionElement struct {
linenum int // of .section itself linenum int // of .section itself
@ -245,6 +253,31 @@ func equal(s []byte, n int, t []byte) bool {
return true return true
} }
// isQuote returns true if c is a string- or character-delimiting quote character.
func isQuote(c byte) bool {
return c == '"' || c == '`' || c == '\''
}
// endQuote returns the end quote index for the quoted string that
// starts at n, or -1 if no matching end quote is found before the end
// of the line.
func endQuote(s []byte, n int) int {
quote := s[n]
for n++; n < len(s); n++ {
switch s[n] {
case '\\':
if quote == '"' || quote == '\'' {
n++
}
case '\n':
return -1
case quote:
return n
}
}
return -1
}
// nextItem returns the next item from the input buffer. If the returned // nextItem returns the next item from the input buffer. If the returned
// item is empty, we are at EOF. The item will be either a // item is empty, we are at EOF. The item will be either a
// delimited string or a non-empty string between delimited // delimited string or a non-empty string between delimited
@ -282,6 +315,14 @@ func (t *Template) nextItem() []byte {
if t.buf[i] == '\n' { if t.buf[i] == '\n' {
break break
} }
if isQuote(t.buf[i]) {
i = endQuote(t.buf, i)
if i == -1 {
t.parseError("unmatched quote")
return nil
}
continue
}
if equal(t.buf, i, t.rdelim) { if equal(t.buf, i, t.rdelim) {
i += len(t.rdelim) i += len(t.rdelim)
right = i right = i
@ -333,23 +374,33 @@ func (t *Template) nextItem() []byte {
return item return item
} }
// Turn a byte array into a white-space-split array of strings. // Turn a byte array into a white-space-split array of strings,
// taking into account quoted strings.
func words(buf []byte) []string { func words(buf []byte) []string {
s := make([]string, 0, 5) s := make([]string, 0, 5)
p := 0 // position in buf for i := 0; i < len(buf); {
// one word per loop // One word per loop
for i := 0; ; i++ { for i < len(buf) && white(buf[i]) {
// skip white space i++
for ; p < len(buf) && white(buf[p]); p++ {
} }
// grab word if i == len(buf) {
start := p
for ; p < len(buf) && !white(buf[p]); p++ {
}
if start == p { // no text left
break break
} }
s = append(s, string(buf[start:p])) // Got a word
start := i
if isQuote(buf[i]) {
i = endQuote(buf, i)
if i < 0 {
i = len(buf)
} else {
i++
}
} else {
for i < len(buf) && !white(buf[i]) {
i++
}
}
s = append(s, string(buf[start:i]))
} }
return s return s
} }
@ -381,11 +432,17 @@ func (t *Template) analyze(item []byte) (tok int, w []string) {
t.parseError("empty directive") t.parseError("empty directive")
return return
} }
if len(w) > 0 && w[0][0] != '.' { first := w[0]
if first[0] != '.' {
tok = tokVariable tok = tokVariable
return return
} }
switch w[0] { if len(first) > 1 && first[1] >= '0' && first[1] <= '9' {
// Must be a float.
tok = tokVariable
return
}
switch first {
case ".meta-left", ".meta-right", ".space", ".tab": case ".meta-left", ".meta-right", ".space", ".tab":
tok = tokLiteral tok = tokLiteral
return return
@ -447,6 +504,37 @@ func (t *Template) newVariable(words []string) *variableElement {
formatters = strings.Split(lastWord[bar+1:], "|", -1) formatters = strings.Split(lastWord[bar+1:], "|", -1)
} }
args := make([]interface{}, len(words))
// Build argument list, processing any literals
for i, word := range words {
var lerr os.Error
switch word[0] {
case '"', '`', '\'':
v, err := strconv.Unquote(word)
if err == nil && word[0] == '\'' {
args[i] = []int(v)[0]
} else {
args[i], lerr = v, err
}
case '.', '+', '-', '0', '1', '2', '3', '4', '5', '6', '7', '8', '9':
v, err := strconv.Btoi64(word, 0)
if err == nil {
args[i] = v
} else {
v, err := strconv.Atof64(word)
args[i], lerr = v, err
}
default:
args[i] = fieldName(word)
}
if lerr != nil {
t.parseError("invalid literal: %q: %s", word, lerr)
}
}
// We could remember the function address here and avoid the lookup later, // We could remember the function address here and avoid the lookup later,
// but it's more dynamic to let the user change the map contents underfoot. // but it's more dynamic to let the user change the map contents underfoot.
// We do require the name to be present, though. // We do require the name to be present, though.
@ -457,7 +545,8 @@ func (t *Template) newVariable(words []string) *variableElement {
t.parseError("unknown formatter: %q", f) t.parseError("unknown formatter: %q", f)
} }
} }
return &variableElement{t.linenum, words, formatters}
return &variableElement{t.linenum, args, formatters}
} }
// Grab the next item. If it's simple, just append it to the template. // Grab the next item. If it's simple, just append it to the template.
@ -753,7 +842,7 @@ func (t *Template) varValue(name string, st *state) reflect.Value {
func (t *Template) format(wr io.Writer, fmt string, val []interface{}, v *variableElement, st *state) { func (t *Template) format(wr io.Writer, fmt string, val []interface{}, v *variableElement, st *state) {
fn := t.formatter(fmt) fn := t.formatter(fmt)
if fn == nil { if fn == nil {
t.execError(st, v.linenum, "missing formatter %s for variable %s", fmt, v.word[0]) t.execError(st, v.linenum, "missing formatter %s for variable", fmt)
} }
fn(wr, fmt, val...) fn(wr, fmt, val...)
} }
@ -761,12 +850,15 @@ func (t *Template) format(wr io.Writer, fmt string, val []interface{}, v *variab
// Evaluate a variable, looking up through the parent if necessary. // Evaluate a variable, looking up through the parent if necessary.
// If it has a formatter attached ({var|formatter}) run that too. // If it has a formatter attached ({var|formatter}) run that too.
func (t *Template) writeVariable(v *variableElement, st *state) { func (t *Template) writeVariable(v *variableElement, st *state) {
// Turn the words of the invocation into values. // Resolve field names
val := make([]interface{}, len(v.word)) val := make([]interface{}, len(v.args))
for i, word := range v.word { for i, arg := range v.args {
val[i] = t.varValue(word, st).Interface() if name, ok := arg.(fieldName); ok {
val[i] = t.varValue(string(name), st).Interface()
} else {
val[i] = arg
}
} }
for i, fmt := range v.fmts[:len(v.fmts)-1] { for i, fmt := range v.fmts[:len(v.fmts)-1] {
b := &st.buf[i&1] b := &st.buf[i&1]
b.Reset() b.Reset()

View File

@ -94,10 +94,15 @@ func multiword(w io.Writer, format string, value ...interface{}) {
} }
} }
func printf(w io.Writer, format string, v ...interface{}) {
io.WriteString(w, fmt.Sprintf(v[0].(string), v[1:]...))
}
var formatters = FormatterMap{ var formatters = FormatterMap{
"uppercase": writer(uppercase), "uppercase": writer(uppercase),
"+1": writer(plus1), "+1": writer(plus1),
"multiword": multiword, "multiword": multiword,
"printf": printf,
} }
var tests = []*Test{ var tests = []*Test{
@ -138,6 +143,36 @@ var tests = []*Test{
out: "nil pointer: <nil>=77\n", out: "nil pointer: <nil>=77\n",
}, },
&Test{
in: `{"Strings" ":"} {""} {"\t\u0123 \x23\\"} {"\"}{\\"}`,
out: "Strings: \t\u0123 \x23\\ \"}{\\",
},
&Test{
in: "{`Raw strings` `:`} {``} {`\\t\\u0123 \\x23\\`} {`}{\\`}",
out: "Raw strings: \\t\\u0123 \\x23\\ }{\\",
},
&Test{
in: "Characters: {'a'} {'\\u0123'} {' '} {'}'} {'{'}",
out: "Characters: 97 291 32 125 123",
},
&Test{
in: "Integers: {1} {-2} {+42} {0777} {0x0a}",
out: "Integers: 1 -2 42 511 10",
},
&Test{
in: "Floats: {.5} {-.5} {1.1} {-2.2} {+42.1} {1e10} {1.2e-3} {1.2e3} {-1.2e3}",
out: "Floats: 0.5 -0.5 1.1 -2.2 42.1 1e+10 0.0012 1200 -1200",
},
// Method at top level // Method at top level
&Test{ &Test{
in: "ptrmethod={PointerMethod}\n", in: "ptrmethod={PointerMethod}\n",
@ -723,6 +758,10 @@ var formatterTests = []Test{
in: "{Integer|||||}", // empty string is a valid formatter in: "{Integer|||||}", // empty string is a valid formatter
out: "77", out: "77",
}, },
{
in: `{"%.02f 0x%02X" 1.1 10|printf}`,
out: "1.10 0x0A",
},
} }
func TestFormatters(t *testing.T) { func TestFormatters(t *testing.T) {