adding support for --wrapScalar=false in properties encoder (#1241)

* adding support for --wrapScalar=false in properties encoder

* altering encoder and decoder tests somewhat

* adding .idea

* Revert "altering encoder and decoder tests somewhat"

This reverts commit e3655130e2.

* adding test scenario for encoding with wrapped scalars
This commit is contained in:
Daniel Carbone 2022-06-24 21:22:03 -05:00 committed by GitHub
parent 06d2aaad80
commit 98b411f82e
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
9 changed files with 232 additions and 79 deletions

3
.gitignore vendored
View File

@ -54,3 +54,6 @@ yq.1
# debian pkg # debian pkg
_build _build
debian/files debian/files
# intellij
/.idea

View File

@ -93,7 +93,7 @@ func configureEncoder(format yqlib.PrinterOutputFormat) yqlib.Encoder {
case yqlib.JSONOutputFormat: case yqlib.JSONOutputFormat:
return yqlib.NewJONEncoder(indent, colorsEnabled) return yqlib.NewJONEncoder(indent, colorsEnabled)
case yqlib.PropsOutputFormat: case yqlib.PropsOutputFormat:
return yqlib.NewPropertiesEncoder() return yqlib.NewPropertiesEncoder(unwrapScalar)
case yqlib.CSVOutputFormat: case yqlib.CSVOutputFormat:
return yqlib.NewCsvEncoder(',') return yqlib.NewCsvEncoder(',')
case yqlib.TSVOutputFormat: case yqlib.TSVOutputFormat:

View File

@ -17,7 +17,7 @@ Given a sample.yml file of:
```yaml ```yaml
# block comments don't come through # block comments don't come through
person: # neither do comments on maps person: # neither do comments on maps
name: Mike # comments on values appear name: Mike Wazowski # comments on values appear
pets: pets:
- cat # comments on array values appear - cat # comments on array values appear
food: [pizza] # comments on arrays do not food: [pizza] # comments on arrays do not
@ -32,7 +32,36 @@ yq -o=props sample.yml
will output will output
```properties ```properties
# comments on values appear # comments on values appear
person.name = Mike person.name = Mike Wazowski
# comments on array values appear
person.pets.0 = cat
person.food.0 = pizza
```
## Encode properties: scalar encapsulation
Note that string values with blank characters in them are encapsulated with double quotes
Given a sample.yml file of:
```yaml
# block comments don't come through
person: # neither do comments on maps
name: Mike Wazowski # comments on values appear
pets:
- cat # comments on array values appear
food: [pizza] # comments on arrays do not
emptyArray: []
emptyMap: []
```
then
```bash
yq -o=props --unwrapScalar=false sample.yml
```
will output
```properties
# comments on values appear
person.name = "Mike Wazowski"
# comments on array values appear # comments on array values appear
person.pets.0 = cat person.pets.0 = cat
@ -44,7 +73,7 @@ Given a sample.yml file of:
```yaml ```yaml
# block comments don't come through # block comments don't come through
person: # neither do comments on maps person: # neither do comments on maps
name: Mike # comments on values appear name: Mike Wazowski # comments on values appear
pets: pets:
- cat # comments on array values appear - cat # comments on array values appear
food: [pizza] # comments on arrays do not food: [pizza] # comments on arrays do not
@ -58,7 +87,7 @@ yq -o=props '... comments = ""' sample.yml
``` ```
will output will output
```properties ```properties
person.name = Mike person.name = Mike Wazowski
person.pets.0 = cat person.pets.0 = cat
person.food.0 = pizza person.food.0 = pizza
``` ```
@ -70,7 +99,7 @@ Given a sample.yml file of:
```yaml ```yaml
# block comments don't come through # block comments don't come through
person: # neither do comments on maps person: # neither do comments on maps
name: Mike # comments on values appear name: Mike Wazowski # comments on values appear
pets: pets:
- cat # comments on array values appear - cat # comments on array values appear
food: [pizza] # comments on arrays do not food: [pizza] # comments on arrays do not
@ -85,7 +114,7 @@ yq -o=props '(.. | select( (tag == "!!map" or tag =="!!seq") and length == 0)) =
will output will output
```properties ```properties
# comments on values appear # comments on values appear
person.name = Mike person.name = Mike Wazowski
# comments on array values appear # comments on array values appear
person.pets.0 = cat person.pets.0 = cat
@ -98,7 +127,7 @@ emptyMap =
Given a sample.properties file of: Given a sample.properties file of:
```properties ```properties
# comments on values appear # comments on values appear
person.name = Mike person.name = Mike Wazowski
# comments on array values appear # comments on array values appear
person.pets.0 = cat person.pets.0 = cat
@ -112,7 +141,7 @@ yq -p=props sample.properties
will output will output
```yaml ```yaml
person: person:
name: Mike # comments on values appear name: Mike Wazowski # comments on values appear
pets: pets:
- cat # comments on array values appear - cat # comments on array values appear
food: food:
@ -123,7 +152,7 @@ person:
Given a sample.properties file of: Given a sample.properties file of:
```properties ```properties
# comments on values appear # comments on values appear
person.name = Mike person.name = Mike Wazowski
# comments on array values appear # comments on array values appear
person.pets.0 = cat person.pets.0 = cat
@ -137,7 +166,7 @@ yq -p=props -o=props '.person.pets.0 = "dog"' sample.properties
will output will output
```properties ```properties
# comments on values appear # comments on values appear
person.name = Mike person.name = Mike Wazowski
# comments on array values appear # comments on array values appear
person.pets.0 = dog person.pets.0 = dog

View File

@ -12,10 +12,13 @@ import (
) )
type propertiesEncoder struct { type propertiesEncoder struct {
unwrapScalar bool
} }
func NewPropertiesEncoder() Encoder { func NewPropertiesEncoder(unwrapScalar bool) Encoder {
return &propertiesEncoder{} return &propertiesEncoder{
unwrapScalar: unwrapScalar,
}
} }
func (pe *propertiesEncoder) CanHandleAliases() bool { func (pe *propertiesEncoder) CanHandleAliases() bool {
@ -75,7 +78,13 @@ func (pe *propertiesEncoder) doEncode(p *properties.Properties, node *yaml.Node,
p.SetComment(path, headAndLineComment(node)) p.SetComment(path, headAndLineComment(node))
switch node.Kind { switch node.Kind {
case yaml.ScalarNode: case yaml.ScalarNode:
_, _, err := p.Set(path, node.Value) var nodeValue string
if pe.unwrapScalar || !strings.Contains(node.Value, " ") {
nodeValue = node.Value
} else {
nodeValue = fmt.Sprintf("%q", node.Value)
}
_, _, err := p.Set(path, nodeValue)
return err return err
case yaml.DocumentNode: case yaml.DocumentNode:
return pe.doEncode(p, node.Content[0], path) return pe.doEncode(p, node.Content[0], path)

View File

@ -9,11 +9,11 @@ import (
"github.com/mikefarah/yq/v4/test" "github.com/mikefarah/yq/v4/test"
) )
func yamlToProps(sampleYaml string) string { func yamlToProps(sampleYaml string, unwrapScalar bool) string {
var output bytes.Buffer var output bytes.Buffer
writer := bufio.NewWriter(&output) writer := bufio.NewWriter(&output)
var propsEncoder = NewPropertiesEncoder() var propsEncoder = NewPropertiesEncoder(unwrapScalar)
inputs, err := readDocuments(strings.NewReader(sampleYaml), "sample.yml", 0, NewYamlDecoder()) inputs, err := readDocuments(strings.NewReader(sampleYaml), "sample.yml", 0, NewYamlDecoder())
if err != nil { if err != nil {
panic(err) panic(err)
@ -28,51 +28,100 @@ func yamlToProps(sampleYaml string) string {
return strings.TrimSuffix(output.String(), "\n") return strings.TrimSuffix(output.String(), "\n")
} }
func TestPropertiesEncoderSimple(t *testing.T) { func TestPropertiesEncoderSimple_Unwrapped(t *testing.T) {
var sampleYaml = `a: 'bob cool'` var sampleYaml = `a: 'bob cool'`
var expectedProps = `a = bob cool` var expectedProps = `a = bob cool`
var actualProps = yamlToProps(sampleYaml) var actualProps = yamlToProps(sampleYaml, true)
test.AssertResult(t, expectedProps, actualProps) test.AssertResult(t, expectedProps, actualProps)
} }
func TestPropertiesEncoderSimpleWithComments(t *testing.T) { func TestPropertiesEncoderSimple_Wrapped(t *testing.T) {
var sampleYaml = `a: 'bob cool'`
var expectedProps = `a = "bob cool"`
var actualProps = yamlToProps(sampleYaml, false)
test.AssertResult(t, expectedProps, actualProps)
}
func TestPropertiesEncoderSimpleWithComments_Unwrapped(t *testing.T) {
var sampleYaml = `a: 'bob cool' # line` var sampleYaml = `a: 'bob cool' # line`
var expectedProps = `# line var expectedProps = `# line
a = bob cool` a = bob cool`
var actualProps = yamlToProps(sampleYaml) var actualProps = yamlToProps(sampleYaml, true)
test.AssertResult(t, expectedProps, actualProps) test.AssertResult(t, expectedProps, actualProps)
} }
func TestPropertiesEncoderDeep(t *testing.T) { func TestPropertiesEncoderSimpleWithComments_Wrapped(t *testing.T) {
var sampleYaml = `a: 'bob cool' # line`
var expectedProps = `# line
a = "bob cool"`
var actualProps = yamlToProps(sampleYaml, false)
test.AssertResult(t, expectedProps, actualProps)
}
func TestPropertiesEncoderDeep_Unwrapped(t *testing.T) {
var sampleYaml = `a: var sampleYaml = `a:
b: "bob cool" b: "bob cool"
` `
var expectedProps = `a.b = bob cool` var expectedProps = `a.b = bob cool`
var actualProps = yamlToProps(sampleYaml) var actualProps = yamlToProps(sampleYaml, true)
test.AssertResult(t, expectedProps, actualProps) test.AssertResult(t, expectedProps, actualProps)
} }
func TestPropertiesEncoderDeepWithComments(t *testing.T) { func TestPropertiesEncoderDeep_Wrapped(t *testing.T) {
var sampleYaml = `a:
b: "bob cool"
`
var expectedProps = `a.b = "bob cool"`
var actualProps = yamlToProps(sampleYaml, false)
test.AssertResult(t, expectedProps, actualProps)
}
func TestPropertiesEncoderDeepWithComments_Unwrapped(t *testing.T) {
var sampleYaml = `a: # a thing var sampleYaml = `a: # a thing
b: "bob cool" # b thing b: "bob cool" # b thing
` `
var expectedProps = `# b thing var expectedProps = `# b thing
a.b = bob cool` a.b = bob cool`
var actualProps = yamlToProps(sampleYaml) var actualProps = yamlToProps(sampleYaml, true)
test.AssertResult(t, expectedProps, actualProps) test.AssertResult(t, expectedProps, actualProps)
} }
func TestPropertiesEncoderArray(t *testing.T) { func TestPropertiesEncoderDeepWithComments_Wrapped(t *testing.T) {
var sampleYaml = `a: # a thing
b: "bob cool" # b thing
`
var expectedProps = `# b thing
a.b = "bob cool"`
var actualProps = yamlToProps(sampleYaml, false)
test.AssertResult(t, expectedProps, actualProps)
}
func TestPropertiesEncoderArray_Unwrapped(t *testing.T) {
var sampleYaml = `a: var sampleYaml = `a:
b: [{c: dog}, {c: cat}] b: [{c: dog}, {c: cat}]
` `
var expectedProps = `a.b.0.c = dog var expectedProps = `a.b.0.c = dog
a.b.1.c = cat` a.b.1.c = cat`
var actualProps = yamlToProps(sampleYaml) var actualProps = yamlToProps(sampleYaml, true)
test.AssertResult(t, expectedProps, actualProps)
}
func TestPropertiesEncoderArray_Wrapped(t *testing.T) {
var sampleYaml = `a:
b: [{c: dog named jim}, {c: cat named jam}]
`
var expectedProps = `a.b.0.c = "dog named jim"
a.b.1.c = "cat named jam"`
var actualProps = yamlToProps(sampleYaml, false)
test.AssertResult(t, expectedProps, actualProps) test.AssertResult(t, expectedProps, actualProps)
} }

View File

@ -9,7 +9,7 @@ import (
"github.com/mikefarah/yq/v4/test" "github.com/mikefarah/yq/v4/test"
) )
var complexExpectYaml = `D0, P[], (!!map)::a: Easy! as one two three const complexExpectYaml = `D0, P[], (!!map)::a: Easy! as one two three
b: b:
c: 2 c: 2
d: d:
@ -96,11 +96,15 @@ func decodeJSON(t *testing.T, jsonString string) *CandidateNode {
} }
func testJSONScenario(t *testing.T, s formatScenario) { func testJSONScenario(t *testing.T, s formatScenario) {
if s.scenarioType == "encode" || s.scenarioType == "roundtrip" { switch s.scenarioType {
case "encode", "decode":
test.AssertResultWithContext(t, s.expected, processFormatScenario(s, NewYamlDecoder(), NewJONEncoder(s.indent, false)), s.description) test.AssertResultWithContext(t, s.expected, processFormatScenario(s, NewYamlDecoder(), NewJONEncoder(s.indent, false)), s.description)
} else { case "":
var actual = resultToString(t, decodeJSON(t, s.input)) var actual = resultToString(t, decodeJSON(t, s.input))
test.AssertResultWithContext(t, s.expected, actual, s.description) test.AssertResultWithContext(t, s.expected, actual, s.description)
default:
panic(fmt.Sprintf("unhandled scenario type %q", s.scenarioType))
} }
} }
@ -135,14 +139,17 @@ func documentJSONDecodeScenario(t *testing.T, w *bufio.Writer, s formatScenario)
func documentJSONScenario(t *testing.T, w *bufio.Writer, i interface{}) { func documentJSONScenario(t *testing.T, w *bufio.Writer, i interface{}) {
s := i.(formatScenario) s := i.(formatScenario)
if s.skipDoc { if s.skipDoc {
return return
} }
if s.scenarioType == "encode" { switch s.scenarioType {
documentJSONEncodeScenario(w, s) case "":
} else {
documentJSONDecodeScenario(t, w, s) documentJSONDecodeScenario(t, w, s)
case "encode":
documentJSONEncodeScenario(w, s)
default:
panic(fmt.Sprintf("unhandled scenario type %q", s.scenarioType))
} }
} }

View File

@ -15,7 +15,7 @@ func configureEncoder(format PrinterOutputFormat, indent int) Encoder {
case JSONOutputFormat: case JSONOutputFormat:
return NewJONEncoder(indent, false) return NewJONEncoder(indent, false)
case PropsOutputFormat: case PropsOutputFormat:
return NewPropertiesEncoder() return NewPropertiesEncoder(true)
case CSVOutputFormat: case CSVOutputFormat:
return NewCsvEncoder(',') return NewCsvEncoder(',')
case TSVOutputFormat: case TSVOutputFormat:

View File

@ -8,9 +8,9 @@ import (
"github.com/mikefarah/yq/v4/test" "github.com/mikefarah/yq/v4/test"
) )
var samplePropertiesYaml = `# block comments don't come through const samplePropertiesYaml = `# block comments don't come through
person: # neither do comments on maps person: # neither do comments on maps
name: Mike # comments on values appear name: Mike Wazowski # comments on values appear
pets: pets:
- cat # comments on array values appear - cat # comments on array values appear
food: [pizza] # comments on arrays do not food: [pizza] # comments on arrays do not
@ -18,37 +18,45 @@ emptyArray: []
emptyMap: [] emptyMap: []
` `
var expectedProperties = `# comments on values appear const expectedPropertiesUnwrapped = `# comments on values appear
person.name = Mike person.name = Mike Wazowski
# comments on array values appear # comments on array values appear
person.pets.0 = cat person.pets.0 = cat
person.food.0 = pizza person.food.0 = pizza
` `
var expectedUpdatedProperties = `# comments on values appear const expectedPropertiesWrapped = `# comments on values appear
person.name = Mike person.name = "Mike Wazowski"
# comments on array values appear
person.pets.0 = cat
person.food.0 = pizza
`
const expectedUpdatedProperties = `# comments on values appear
person.name = Mike Wazowski
# comments on array values appear # comments on array values appear
person.pets.0 = dog person.pets.0 = dog
person.food.0 = pizza person.food.0 = pizza
` `
var expectedDecodedYaml = `person: const expectedDecodedYaml = `person:
name: Mike # comments on values appear name: Mike Wazowski # comments on values appear
pets: pets:
- cat # comments on array values appear - cat # comments on array values appear
food: food:
- pizza - pizza
` `
var expectedPropertiesNoComments = `person.name = Mike const expectedPropertiesNoComments = `person.name = Mike Wazowski
person.pets.0 = cat person.pets.0 = cat
person.food.0 = pizza person.food.0 = pizza
` `
var expectedPropertiesWithEmptyMapsAndArrays = `# comments on values appear const expectedPropertiesWithEmptyMapsAndArrays = `# comments on values appear
person.name = Mike person.name = Mike Wazowski
# comments on array values appear # comments on array values appear
person.pets.0 = cat person.pets.0 = cat
@ -62,7 +70,14 @@ var propertyScenarios = []formatScenario{
description: "Encode properties", description: "Encode properties",
subdescription: "Note that empty arrays and maps are not encoded by default.", subdescription: "Note that empty arrays and maps are not encoded by default.",
input: samplePropertiesYaml, input: samplePropertiesYaml,
expected: expectedProperties, expected: expectedPropertiesUnwrapped,
},
{
description: "Encode properties: scalar encapsulation",
subdescription: "Note that string values with blank characters in them are encapsulated with double quotes",
input: samplePropertiesYaml,
expected: expectedPropertiesWrapped,
scenarioType: "encode-wrapped",
}, },
{ {
description: "Encode properties: no comments", description: "Encode properties: no comments",
@ -79,7 +94,7 @@ var propertyScenarios = []formatScenario{
}, },
{ {
description: "Decode properties", description: "Decode properties",
input: expectedProperties, input: expectedPropertiesUnwrapped,
expected: expectedDecodedYaml, expected: expectedDecodedYaml,
scenarioType: "decode", scenarioType: "decode",
}, },
@ -92,7 +107,7 @@ var propertyScenarios = []formatScenario{
}, },
{ {
description: "Roundtrip", description: "Roundtrip",
input: expectedProperties, input: expectedPropertiesUnwrapped,
expression: `.person.pets.0 = "dog"`, expression: `.person.pets.0 = "dog"`,
expected: expectedUpdatedProperties, expected: expectedUpdatedProperties,
scenarioType: "roundtrip", scenarioType: "roundtrip",
@ -106,7 +121,7 @@ var propertyScenarios = []formatScenario{
}, },
} }
func documentEncodePropertyScenario(w *bufio.Writer, s formatScenario) { func documentUnwrappedEncodePropertyScenario(w *bufio.Writer, s formatScenario) {
writeOrPanic(w, fmt.Sprintf("## %v\n", s.description)) writeOrPanic(w, fmt.Sprintf("## %v\n", s.description))
if s.subdescription != "" { if s.subdescription != "" {
@ -128,7 +143,32 @@ func documentEncodePropertyScenario(w *bufio.Writer, s formatScenario) {
} }
writeOrPanic(w, "will output\n") writeOrPanic(w, "will output\n")
writeOrPanic(w, fmt.Sprintf("```properties\n%v```\n\n", processFormatScenario(s, NewYamlDecoder(), NewPropertiesEncoder()))) writeOrPanic(w, fmt.Sprintf("```properties\n%v```\n\n", processFormatScenario(s, NewYamlDecoder(), NewPropertiesEncoder(true))))
}
func documentWrappedEncodePropertyScenario(w *bufio.Writer, s formatScenario) {
writeOrPanic(w, fmt.Sprintf("## %v\n", s.description))
if s.subdescription != "" {
writeOrPanic(w, s.subdescription)
writeOrPanic(w, "\n\n")
}
writeOrPanic(w, "Given a sample.yml file of:\n")
writeOrPanic(w, fmt.Sprintf("```yaml\n%v\n```\n", s.input))
writeOrPanic(w, "then\n")
expression := s.expression
if expression != "" {
writeOrPanic(w, fmt.Sprintf("```bash\nyq -o=props --unwrapScalar=false '%v' sample.yml\n```\n", expression))
} else {
writeOrPanic(w, "```bash\nyq -o=props --unwrapScalar=false sample.yml\n```\n")
}
writeOrPanic(w, "will output\n")
writeOrPanic(w, fmt.Sprintf("```properties\n%v```\n\n", processFormatScenario(s, NewYamlDecoder(), NewPropertiesEncoder(false))))
} }
func documentDecodePropertyScenario(w *bufio.Writer, s formatScenario) { func documentDecodePropertyScenario(w *bufio.Writer, s formatScenario) {
@ -178,7 +218,7 @@ func documentRoundTripPropertyScenario(w *bufio.Writer, s formatScenario) {
writeOrPanic(w, "will output\n") writeOrPanic(w, "will output\n")
writeOrPanic(w, fmt.Sprintf("```properties\n%v```\n\n", processFormatScenario(s, NewPropertiesDecoder(), NewPropertiesEncoder()))) writeOrPanic(w, fmt.Sprintf("```properties\n%v```\n\n", processFormatScenario(s, NewPropertiesDecoder(), NewPropertiesEncoder(true))))
} }
func documentPropertyScenario(t *testing.T, w *bufio.Writer, i interface{}) { func documentPropertyScenario(t *testing.T, w *bufio.Writer, i interface{}) {
@ -186,24 +226,35 @@ func documentPropertyScenario(t *testing.T, w *bufio.Writer, i interface{}) {
if s.skipDoc { if s.skipDoc {
return return
} }
if s.scenarioType == "decode" { switch s.scenarioType {
case "":
documentUnwrappedEncodePropertyScenario(w, s)
case "decode":
documentDecodePropertyScenario(w, s) documentDecodePropertyScenario(w, s)
} else if s.scenarioType == "roundtrip" { case "encode-wrapped":
documentWrappedEncodePropertyScenario(w, s)
case "roundtrip":
documentRoundTripPropertyScenario(w, s) documentRoundTripPropertyScenario(w, s)
} else {
documentEncodePropertyScenario(w, s)
}
default:
panic(fmt.Sprintf("unhandled scenario type %q", s.scenarioType))
}
} }
func TestPropertyScenarios(t *testing.T) { func TestPropertyScenarios(t *testing.T) {
for _, s := range propertyScenarios { for _, s := range propertyScenarios {
if s.scenarioType == "decode" { switch s.scenarioType {
case "":
test.AssertResultWithContext(t, s.expected, processFormatScenario(s, NewYamlDecoder(), NewPropertiesEncoder(true)), s.description)
case "decode":
test.AssertResultWithContext(t, s.expected, processFormatScenario(s, NewPropertiesDecoder(), NewYamlEncoder(2, false, true, true)), s.description) test.AssertResultWithContext(t, s.expected, processFormatScenario(s, NewPropertiesDecoder(), NewYamlEncoder(2, false, true, true)), s.description)
} else if s.scenarioType == "roundtrip" { case "encode-wrapped":
test.AssertResultWithContext(t, s.expected, processFormatScenario(s, NewPropertiesDecoder(), NewPropertiesEncoder()), s.description) test.AssertResultWithContext(t, s.expected, processFormatScenario(s, NewYamlDecoder(), NewPropertiesEncoder(false)), s.description)
} else { case "roundtrip":
test.AssertResultWithContext(t, s.expected, processFormatScenario(s, NewYamlDecoder(), NewPropertiesEncoder()), s.description) test.AssertResultWithContext(t, s.expected, processFormatScenario(s, NewPropertiesDecoder(), NewPropertiesEncoder(true)), s.description)
default:
panic(fmt.Sprintf("unhandled scenario type %q", s.scenarioType))
} }
} }
genericScenarios := make([]interface{}, len(propertyScenarios)) genericScenarios := make([]interface{}, len(propertyScenarios))

View File

@ -8,7 +8,7 @@ import (
"github.com/mikefarah/yq/v4/test" "github.com/mikefarah/yq/v4/test"
) )
var inputXMLWithComments = ` const inputXMLWithComments = `
<!-- before cat --> <!-- before cat -->
<cat> <cat>
<!-- in cat before --> <!-- in cat before -->
@ -26,7 +26,7 @@ for x --></x>
</cat> </cat>
<!-- after cat --> <!-- after cat -->
` `
var inputXMLWithCommentsWithSubChild = ` const inputXMLWithCommentsWithSubChild = `
<!-- before cat --> <!-- before cat -->
<cat> <cat>
<!-- in cat before --> <!-- in cat before -->
@ -45,7 +45,7 @@ for x --></x>
<!-- after cat --> <!-- after cat -->
` `
var expectedDecodeYamlWithSubChild = `# before cat const expectedDecodeYamlWithSubChild = `# before cat
cat: cat:
# in cat before # in cat before
x: "3" # multi x: "3" # multi
@ -65,7 +65,7 @@ cat:
# after cat # after cat
` `
var inputXMLWithCommentsWithArray = ` const inputXMLWithCommentsWithArray = `
<!-- before cat --> <!-- before cat -->
<cat> <cat>
<!-- in cat before --> <!-- in cat before -->
@ -85,7 +85,7 @@ for x --></x>
<!-- after cat --> <!-- after cat -->
` `
var expectedDecodeYamlWithArray = `# before cat const expectedDecodeYamlWithArray = `# before cat
cat: cat:
# in cat before # in cat before
x: "3" # multi x: "3" # multi
@ -109,7 +109,7 @@ cat:
# after cat # after cat
` `
var expectedDecodeYamlWithComments = `# before cat const expectedDecodeYamlWithComments = `# before cat
cat: cat:
# in cat before # in cat before
x: "3" # multi x: "3" # multi
@ -126,7 +126,7 @@ cat:
# after cat # after cat
` `
var expectedRoundtripXMLWithComments = `<!-- before cat --><cat><!-- in cat before --> const expectedRoundtripXMLWithComments = `<!-- before cat --><cat><!-- in cat before -->
<x>3<!-- multi <x>3<!-- multi
line comment line comment
for x --></x><!-- before y --> for x --></x><!-- before y -->
@ -137,7 +137,7 @@ in d before -->
</cat><!-- after cat --> </cat><!-- after cat -->
` `
var yamlWithComments = `# above_cat const yamlWithComments = `# above_cat
cat: # inline_cat cat: # inline_cat
# above_array # above_array
array: # inline_array array: # inline_array
@ -147,31 +147,31 @@ cat: # inline_cat
# below_cat # below_cat
` `
var expectedXMLWithComments = `<!-- above_cat inline_cat --><cat><!-- above_array inline_array --> const expectedXMLWithComments = `<!-- above_cat inline_cat --><cat><!-- above_array inline_array -->
<array>val1<!-- inline_val1 --></array> <array>val1<!-- inline_val1 --></array>
<array><!-- above_val2 -->val2<!-- inline_val2 --></array> <array><!-- above_val2 -->val2<!-- inline_val2 --></array>
</cat><!-- below_cat --> </cat><!-- below_cat -->
` `
var inputXMLWithNamespacedAttr = ` const inputXMLWithNamespacedAttr = `
<?xml version="1.0"?> <?xml version="1.0"?>
<map xmlns="some-namespace" xmlns:xsi="some-instance" xsi:schemaLocation="some-url"> <map xmlns="some-namespace" xmlns:xsi="some-instance" xsi:schemaLocation="some-url">
</map> </map>
` `
var expectedYAMLWithNamespacedAttr = `map: const expectedYAMLWithNamespacedAttr = `map:
+xmlns: some-namespace +xmlns: some-namespace
+xmlns:xsi: some-instance +xmlns:xsi: some-instance
+some-instance:schemaLocation: some-url +some-instance:schemaLocation: some-url
` `
var expectedYAMLWithRawNamespacedAttr = `map: const expectedYAMLWithRawNamespacedAttr = `map:
+xmlns: some-namespace +xmlns: some-namespace
+xmlns:xsi: some-instance +xmlns:xsi: some-instance
+xsi:schemaLocation: some-url +xsi:schemaLocation: some-url
` `
var xmlWithCustomDtd = ` const xmlWithCustomDtd = `
<?xml version="1.0"?> <?xml version="1.0"?>
<!DOCTYPE root [ <!DOCTYPE root [
<!ENTITY writer "Blah."> <!ENTITY writer "Blah.">
@ -181,7 +181,7 @@ var xmlWithCustomDtd = `
<item>&writer;&copyright;</item> <item>&writer;&copyright;</item>
</root>` </root>`
var expectedDtd = `root: const expectedDtd = `root:
item: '&writer;&copyright;' item: '&writer;&copyright;'
` `
@ -336,6 +336,8 @@ var xmlScenarios = []formatScenario{
func testXMLScenario(t *testing.T, s formatScenario) { func testXMLScenario(t *testing.T, s formatScenario) {
switch s.scenarioType { switch s.scenarioType {
case "", "decode":
test.AssertResultWithContext(t, s.expected, processFormatScenario(s, NewXMLDecoder("+", "+content", false, false, false), NewYamlEncoder(4, false, true, true)), s.description)
case "encode": case "encode":
test.AssertResultWithContext(t, s.expected, processFormatScenario(s, NewYamlDecoder(), NewXMLEncoder(2, "+", "+content")), s.description) test.AssertResultWithContext(t, s.expected, processFormatScenario(s, NewYamlDecoder(), NewXMLEncoder(2, "+", "+content")), s.description)
case "roundtrip": case "roundtrip":
@ -344,8 +346,9 @@ func testXMLScenario(t *testing.T, s formatScenario) {
test.AssertResultWithContext(t, s.expected, processFormatScenario(s, NewXMLDecoder("+", "+content", false, true, false), NewYamlEncoder(2, false, true, true)), s.description) test.AssertResultWithContext(t, s.expected, processFormatScenario(s, NewXMLDecoder("+", "+content", false, true, false), NewYamlEncoder(2, false, true, true)), s.description)
case "decode-raw-token": case "decode-raw-token":
test.AssertResultWithContext(t, s.expected, processFormatScenario(s, NewXMLDecoder("+", "+content", false, true, true), NewYamlEncoder(2, false, true, true)), s.description) test.AssertResultWithContext(t, s.expected, processFormatScenario(s, NewXMLDecoder("+", "+content", false, true, true), NewYamlEncoder(2, false, true, true)), s.description)
default: default:
test.AssertResultWithContext(t, s.expected, processFormatScenario(s, NewXMLDecoder("+", "+content", false, false, false), NewYamlEncoder(4, false, true, true)), s.description) panic(fmt.Sprintf("unhandled scenario type %q", s.scenarioType))
} }
} }
@ -356,6 +359,8 @@ func documentXMLScenario(t *testing.T, w *bufio.Writer, i interface{}) {
return return
} }
switch s.scenarioType { switch s.scenarioType {
case "", "decode":
documentXMLDecodeScenario(w, s)
case "encode": case "encode":
documentXMLEncodeScenario(w, s) documentXMLEncodeScenario(w, s)
case "roundtrip": case "roundtrip":
@ -364,10 +369,10 @@ func documentXMLScenario(t *testing.T, w *bufio.Writer, i interface{}) {
documentXMLDecodeKeepNsScenario(w, s) documentXMLDecodeKeepNsScenario(w, s)
case "decode-raw-token": case "decode-raw-token":
documentXMLDecodeKeepNsRawTokenScenario(w, s) documentXMLDecodeKeepNsRawTokenScenario(w, s)
default:
documentXMLDecodeScenario(w, s)
}
default:
panic(fmt.Sprintf("unhandled scenario type %q", s.scenarioType))
}
} }
func documentXMLDecodeScenario(w *bufio.Writer, s formatScenario) { func documentXMLDecodeScenario(w *bufio.Writer, s formatScenario) {