Commit ab52ad89 authored by Caleb Spare's avatar Caleb Spare Committed by Brad Fitzpatrick

encoding/json: add Encoder.DisableHTMLEscaping

This provides a way to disable the escaping of <, >, and & in JSON
strings.

Fixes #14749.

Change-Id: I1afeb0244455fc8b06c6cce920444532f229555b
Reviewed-on: https://go-review.googlesource.com/21796
Run-TryBot: Caleb Spare <cespare@gmail.com>
TryBot-Result: Gobot Gobot <gobot@golang.org>
Reviewed-by: default avatarBrad Fitzpatrick <bradfitz@golang.org>
parent 97360096
This diff is collapsed.
...@@ -376,41 +376,45 @@ func TestDuplicatedFieldDisappears(t *testing.T) { ...@@ -376,41 +376,45 @@ func TestDuplicatedFieldDisappears(t *testing.T) {
func TestStringBytes(t *testing.T) { func TestStringBytes(t *testing.T) {
// Test that encodeState.stringBytes and encodeState.string use the same encoding. // Test that encodeState.stringBytes and encodeState.string use the same encoding.
es := &encodeState{}
var r []rune var r []rune
for i := '\u0000'; i <= unicode.MaxRune; i++ { for i := '\u0000'; i <= unicode.MaxRune; i++ {
r = append(r, i) r = append(r, i)
} }
s := string(r) + "\xff\xff\xffhello" // some invalid UTF-8 too s := string(r) + "\xff\xff\xffhello" // some invalid UTF-8 too
es.string(s)
esBytes := &encodeState{} for _, escapeHTML := range []bool{true, false} {
esBytes.stringBytes([]byte(s)) es := &encodeState{}
es.string(s, escapeHTML)
enc := es.Buffer.String() esBytes := &encodeState{}
encBytes := esBytes.Buffer.String() esBytes.stringBytes([]byte(s), escapeHTML)
if enc != encBytes {
i := 0
for i < len(enc) && i < len(encBytes) && enc[i] == encBytes[i] {
i++
}
enc = enc[i:]
encBytes = encBytes[i:]
i = 0
for i < len(enc) && i < len(encBytes) && enc[len(enc)-i-1] == encBytes[len(encBytes)-i-1] {
i++
}
enc = enc[:len(enc)-i]
encBytes = encBytes[:len(encBytes)-i]
if len(enc) > 20 { enc := es.Buffer.String()
enc = enc[:20] + "..." encBytes := esBytes.Buffer.String()
} if enc != encBytes {
if len(encBytes) > 20 { i := 0
encBytes = encBytes[:20] + "..." for i < len(enc) && i < len(encBytes) && enc[i] == encBytes[i] {
} i++
}
enc = enc[i:]
encBytes = encBytes[i:]
i = 0
for i < len(enc) && i < len(encBytes) && enc[len(enc)-i-1] == encBytes[len(encBytes)-i-1] {
i++
}
enc = enc[:len(enc)-i]
encBytes = encBytes[:len(encBytes)-i]
t.Errorf("encodings differ at %#q vs %#q", enc, encBytes) if len(enc) > 20 {
enc = enc[:20] + "..."
}
if len(encBytes) > 20 {
encBytes = encBytes[:20] + "..."
}
t.Errorf("with escapeHTML=%t, encodings differ at %#q vs %#q",
escapeHTML, enc, encBytes)
}
} }
} }
......
...@@ -166,8 +166,9 @@ func nonSpace(b []byte) bool { ...@@ -166,8 +166,9 @@ func nonSpace(b []byte) bool {
// An Encoder writes JSON values to an output stream. // An Encoder writes JSON values to an output stream.
type Encoder struct { type Encoder struct {
w io.Writer w io.Writer
err error err error
escapeHTML bool
indentBuf *bytes.Buffer indentBuf *bytes.Buffer
indentPrefix string indentPrefix string
...@@ -176,7 +177,7 @@ type Encoder struct { ...@@ -176,7 +177,7 @@ type Encoder struct {
// NewEncoder returns a new encoder that writes to w. // NewEncoder returns a new encoder that writes to w.
func NewEncoder(w io.Writer) *Encoder { func NewEncoder(w io.Writer) *Encoder {
return &Encoder{w: w} return &Encoder{w: w, escapeHTML: true}
} }
// Encode writes the JSON encoding of v to the stream, // Encode writes the JSON encoding of v to the stream,
...@@ -189,7 +190,7 @@ func (enc *Encoder) Encode(v interface{}) error { ...@@ -189,7 +190,7 @@ func (enc *Encoder) Encode(v interface{}) error {
return enc.err return enc.err
} }
e := newEncodeState() e := newEncodeState()
err := e.marshal(v) err := e.marshal(v, encOpts{escapeHTML: enc.escapeHTML})
if err != nil { if err != nil {
return err return err
} }
...@@ -225,6 +226,12 @@ func (enc *Encoder) Indent(prefix, indent string) { ...@@ -225,6 +226,12 @@ func (enc *Encoder) Indent(prefix, indent string) {
enc.indentValue = indent enc.indentValue = indent
} }
// DisableHTMLEscaping causes the encoder not to escape angle brackets
// ("<" and ">") or ampersands ("&") in JSON strings.
func (enc *Encoder) DisableHTMLEscaping() {
enc.escapeHTML = false
}
// RawMessage is a raw encoded JSON value. // RawMessage is a raw encoded JSON value.
// It implements Marshaler and Unmarshaler and can // It implements Marshaler and Unmarshaler and can
// be used to delay JSON decoding or precompute a JSON encoding. // be used to delay JSON decoding or precompute a JSON encoding.
......
...@@ -87,6 +87,39 @@ func TestEncoderIndent(t *testing.T) { ...@@ -87,6 +87,39 @@ func TestEncoderIndent(t *testing.T) {
} }
} }
func TestEncoderDisableHTMLEscaping(t *testing.T) {
var c C
var ct CText
for _, tt := range []struct {
name string
v interface{}
wantEscape string
want string
}{
{"c", c, `"\u003c\u0026\u003e"`, `"<&>"`},
{"ct", ct, `"\"\u003c\u0026\u003e\""`, `"\"<&>\""`},
{`"<&>"`, "<&>", `"\u003c\u0026\u003e"`, `"<&>"`},
} {
var buf bytes.Buffer
enc := NewEncoder(&buf)
if err := enc.Encode(tt.v); err != nil {
t.Fatalf("Encode(%s): %s", tt.name, err)
}
if got := strings.TrimSpace(buf.String()); got != tt.wantEscape {
t.Errorf("Encode(%s) = %#q, want %#q", tt.name, got, tt.wantEscape)
}
buf.Reset()
enc.DisableHTMLEscaping()
if err := enc.Encode(tt.v); err != nil {
t.Fatalf("DisableHTMLEscaping Encode(%s): %s", tt.name, err)
}
if got := strings.TrimSpace(buf.String()); got != tt.want {
t.Errorf("DisableHTMLEscaping Encode(%s) = %#q, want %#q",
tt.name, got, tt.want)
}
}
}
func TestDecoder(t *testing.T) { func TestDecoder(t *testing.T) {
for i := 0; i <= len(streamTest); i++ { for i := 0; i <= len(streamTest); i++ {
// Use stream without newlines as input, // Use stream without newlines as input,
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment