Content inside gojax
(The raw file follows this syntax highlighted file.)

package main

import (
	"flag"
	"fmt"
	"git.ondollo.com/gojax"
	"io/ioutil"
	"os"
	// "strings"
)

var root = flag.String("root", "root", "")
var runTests = flag.Bool("t", false, "")
var inPath = flag.String("in", "", "")
var stringIn = flag.String("s", "", "")
var path = flag.String("p", "/*", "")
var allResults = flag.Bool("allResults", false, "")

var xStringIn = flag.String("xs", "", "")
var xStdin = flag.Bool("xstdin", false, "")

func main() {
	flag.Parse()

	if len(*xStringIn) > 0 {
		x, err := gojax.FromXML(*xStringIn)
		if err != nil {
			fmt.Println(err)
			os.Exit(1)
		}
		fmt.Println(x.JSON())
		os.Exit(0)
	}
	if *xStdin {
		bs, err := ioutil.ReadAll(os.Stdin)
		if err != nil {
			fmt.Println("Failed to read from STDIN:", err)
			os.Exit(1)
		}
		x, err := gojax.FromXML(string(bs))
		if err != nil {
			fmt.Println(err)
			os.Exit(1)
		}
		fmt.Println(x.JSON())
		os.Exit(0)
	}

	postProcess := false
	jsonFromFlags := ""
	if len(flag.Args()) == 1 && flag.Args()[0] == "-" {
		bs, err := ioutil.ReadAll(os.Stdin)
		if err != nil {
			fmt.Println("Failed to read from STDIN:", err)
			os.Exit(1)
		}
		jsonFromFlags = string(bs)
		postProcess = true
	}
	if len(*stringIn) > 0 {
		jsonFromFlags = *stringIn
		postProcess = true
	}
	if len(*inPath) > 0 && len(flag.Args()) == 0 {
		bs, err := ioutil.ReadFile(*inPath)
		if err != nil {
			fmt.Println("Failed to read given input file:", err)
			os.Exit(1)
		}
		if len(bs) == 0 {
			fmt.Println("Given file is empty:", *inPath)
			os.Exit(1)
		}
		jsonFromFlags = string(bs)
		postProcess = true
	}
	if postProcess {
		x, err := gojax.FromJSON(*root, jsonFromFlags)
		if err != nil {
			fmt.Println(err)
			os.Exit(1)
		}
		xns, err := x.Path(*path)
		if err != nil {
			fmt.Println(err)
			os.Exit(1)
		}
		if *allResults {
			for _, xn := range xns {
				fmt.Println(xn)
			}
		} else {
			if len(xns) > 0 {
				fmt.Println(xns[0])
			} else {
				fmt.Println("No results")
			}
		}
		os.Exit(0)
	}
	tests := 0
	failures := 0
	options := gojax.DefaultTestOptions
	test := func(s string, j string, r string) {
		tests += 1
		x, err := gojax.FromJSONWithOptions(`r`, j, options)
		if err != nil {
			failures += 1
			fmt.Println("Fail with error", s, err, x)
			//gojax.FromDebug(`r`, j)
		} else if x.StringWithOptions(options) != r {
			failures += 1
			fmt.Printf(`Fail: %s
  FROM: %s
  EXP: %s
  GOT: %s
`, s, j, r, x.StringWithOptions(options))
			//gojax.FromDebug(`r`, j)
		}
	}
	xtest := func(s string, xml string, r string) {
		tests += 1
		x, err := gojax.FromXMLWithOptions(xml, options)
		if err != nil {
			failures += 1
			fmt.Println("Fail with error", s, err, x)
			//gojax.FromDebug(`r`, j)
		} else if x.JSON() != r {
			failures += 1
			fmt.Printf(`Fail: %s
  FROM: %s
  EXP: %s
  GOT: %s
`, s, xml, r, x.JSON())
			//gojax.FromDebug(`r`, j)
		}
	}
	xxtest := func(s string, xml string, r string) {
		tests += 1
		x, err := gojax.FromXMLWithOptions(xml, options)
		if err != nil {
			failures += 1
			fmt.Println("Fail with error", s, err, x)
			//gojax.FromDebug(`r`, j)
		} else if x.StringWithOptions(options) != r {
			failures += 1
			fmt.Printf(`Fail: %s
  FROM: %s
  EXP: %s
  GOT: %s
`, s, xml, r, x.StringWithOptions(options))
			//gojax.FromDebug(`r`, j)
		}
	}
	_nop := func(s string, j string, r string) { /*fmt.Println("Unsupported:", s, j)*/ }
	if *runTests {

		//fmt.Println(strings.Repeat("\n", 40))
		xtest("empty short node", `<a/>`, `{"a":""}`)
		xtest("empty paired node", `<a></a>`, `{"a":""}`)
		xtest("paired node with a sentence", `<a>This is a sentence.</a>`, `{"a":"This is a sentence."}`)
		xtest("paired node with cdata",
			`<a><![CDATA[something that should be <i>just</i> consumed]]></a>`,
			`{"a":"something that should be <i>just</i> consumed"}`)
		xtest("simple paired node", `<a>1</a>`, `{"a":"1"}`)
		xtest("simple paired/short nodes", `<a><b/></a>`, `{"a":{"b":""}}`)
		xtest("simple paired/paired nodes", `<a><b></b></a>`, `{"a":{"b":""}}`)
		xtest("simple paired/paired nodes with inner content", `<a><b>1</b></a>`, `{"a":{"b":"1"}}`)

		xtest("more than one child in simple node", `<a><b></b><c></c></a>`, `{"a":{"b":"","c":""}}`)
		xtest("more than one child with content in simple node", `<a><b>1</b><c>2</c></a>`, `{"a":{"b":"1","c":"2"}}`)
		xtest("more than one child with content and repeats in simple node",
			`<a><b>1</b><c>1</c><b>1</b></a>`, `{"a":{"b":["1","1"],"c":"1"}}`)

		xtest("empty short node w/attr", `<a b="1"/>`, `{"a":""}`)
		xtest("empty short node w/attrs", `<a b="1" c="2"/>`, `{"a":""}`)
		xtest("empty paired node w/attr", `<a b="1"></a>`, `{"a":""}`)
		xtest("empty paired node w/attrs", `<a b="1" c="2"></a>`, `{"a":""}`)
		xtest("simple paired node w/attr", `<a b="1">1</a>`, `{"a":"1"}`)
		xtest("simple paired node w/attrs", `<a b="1" c="2">1</a>`, `{"a":"1"}`)
		xtest("simple paired/short nodes w/attr", `<a><b c="1"/></a>`, `{"a":{"b":""}}`)
		xtest("simple paired/short nodes w/attrs", `<a><b c="1" d="2"/></a>`, `{"a":{"b":""}}`)
		xtest("simple paired/paired nodes w/attr", `<a><b c="1"></b></a>`, `{"a":{"b":""}}`)
		xtest("simple paired/paired nodes w/attrs", `<a><b c="1" d="2"></b></a>`, `{"a":{"b":""}}`)
		xtest("simple paired/paired nodes with inner content w/attr", `<a><b c="2">1</b></a>`, `{"a":{"b":"1"}}`)
		xtest("simple paired/paired nodes with inner content w/attrs", `<a><b c="2" d="3">1</b></a>`, `{"a":{"b":"1"}}`)

		xtest("more than one child in simple node w/attr", `<a><b d="1"></b><c e="2"></c></a>`, `{"a":{"b":"","c":""}}`)
		xtest("more than one child in simple node w/attrs",
			`<a><b d="1" e="2"></b><c f="3" g="4"></c></a>`, `{"a":{"b":"","c":""}}`)
		xtest("more than one child with content in simple node w/attr",
			`<a><b d="1">1</b><c e="2">2</c></a>`, `{"a":{"b":"1","c":"2"}}`)
		xtest("more than one child with content in simple node w/attrs",
			`<a><b d="1" e="2">1</b><c f="3" g="4">2</c></a>`, `{"a":{"b":"1","c":"2"}}`)
		xtest("more than one child with content and repeats in simple node w/attr",
			`<a><b d="1">1</b><c e="2">1</c><b f="3">1</b></a>`, `{"a":{"b":["1","1"],"c":"1"}}`)
		xtest("more than one child with content and repeats in simple node w/attrs",
			`<a><b d="1" e="2">1</b><c f="3" g="4">1</c><b h="5" i="6">1</b></a>`,
			`{"a":{"b":["1","1"],"c":"1"}}`)

		xtest("node w/children and w/attr", `<a id="1"><b/></a>`, `{"a":{"id":"1","b":""}}`)

		xtest("complex",
			`<a><b>1</b><c>2</c><d/><e>3</e><f>4</f></a>`,
			`{"a":{"b":"1","c":"2","d":"","e":"3","f":"4"}}`)

		xtest("repeated paired/paired nodes",
			`<a><b><c>1</c></b><b><c>2</c></b></a>`,
			`{"a":{"b":[{"c":"1"},{"c":"2"}]}}`)

		xxtest("match paired node with markup",
			`<a>something that should be & just consumed</a>`,
			`<a><![CDATA[something that should be & just consumed]]></a>`,
		)
		xxtest("match paired node with cdata",
			`<a><![CDATA[something that should be <i>just</i> consumed]]></a>`,
			`<a><![CDATA[something that should be <i>just</i> consumed]]></a>`,
		)

		_nop("empty object", `{}`, `<r></r>`)
		test("basic digit", `{"a":1}`, `<r><a>1</a></r>`)
		test("basic string", `{"a":"b"}`, `<r><a>b</a></r>`)
		test("basic string with markup", `{"a":"b<c>d</c>"}`, `<r><a><![CDATA[b<c>d</c>]]></a></r>`)
		_nop("basic object", `{"a":{}}`, `<r><a></a></r>`)
		_nop("basic array", `{"a":[]}`, `<r><a></a></r>`)
		test("basic true", `{"a":true}`, `<r><a>true</a></r>`)
		test("basic false", `{"a":false}`, `<r><a>false</a></r>`)
		test("basic null", `{"a":null}`, `<r><a>null</a></r>`)

		test("basic id with digit", `{"a#theID":1}`, `<r><a id="theID">1</a></r>`)

		test("basic object digit", `{"a":{"b":1}}`, `<r><a><b>1</b></a></r>`)
		test("basic object string", `{"a":{"b":"c"}}`, `<r><a><b>c</b></a></r>`)
		test("basic object true", `{"a":{"b":true}}`, `<r><a><b>true</b></a></r>`)
		test("basic object false", `{"a":{"b":false}}`, `<r><a><b>false</b></a></r>`)
		test("basic object null", `{"a":{"b":null}}`, `<r><a><b>null</b></a></r>`)

		test("basic array digit", `{"a":[1]}`, `<r><a><n index="0">1</n></a></r>`)
		test("basic array string", `{"a":["b"]}`, `<r><a><n index="0">b</n></a></r>`)
		test("basic array true", `{"a":[true]}`, `<r><a><n index="0">true</n></a></r>`)
		test("basic array false", `{"a":[false]}`, `<r><a><n index="0">false</n></a></r>`)
		test("basic array null", `{"a":[null]}`, `<r><a><n index="0">null</n></a></r>`)

		test("basic array object digit", `{"a":[{"z":1}]}`, `<r><a><n index="0"><z>1</z></n></a></r>`)
		test("basic array of objects digit", `{"a":[{"z":1},{"y":2}]}`, `<r><a><n index="0"><z>1</z></n><n index="1"><y>2</y></n></a></r>`)
		test("basic array object string", `{"a":[{"z":"b"}]}`, `<r><a><n index="0"><z>b</z></n></a></r>`)
		test("basic array object true", `{"a":[{"z":true}]}`, `<r><a><n index="0"><z>true</z></n></a></r>`)
		test("basic array object false", `{"a":[{"z":false}]}`, `<r><a><n index="0"><z>false</z></n></a></r>`)
		test("basic array object null", `{"a":[{"z":null}]}`, `<r><a><n index="0"><z>null</z></n></a></r>`)

		test("basic array array digit", `{"a":[[1]]}`, `<r><a><n index="0"><n index="0">1</n></n></a></r>`)
		test("basic array array string", `{"a":[["b"]]}`, `<r><a><n index="0"><n index="0">b</n></n></a></r>`)
		test("basic array array true", `{"a":[[true]]}`, `<r><a><n index="0"><n index="0">true</n></n></a></r>`)
		test("basic array array false", `{"a":[[false]]}`, `<r><a><n index="0"><n index="0">false</n></n></a></r>`)
		test("basic array array null", `{"a":[[null]]}`, `<r><a><n index="0"><n index="0">null</n></n></a></r>`)

		options.WrapArrayElements = false
		test("basic array digit", `{"a":[1]}`, `<r><a>1</a></r>`)
		test("basic array string", `{"a":["b"]}`, `<r><a>b</a></r>`)
		test("basic array true", `{"a":[true]}`, `<r><a>true</a></r>`)
		test("basic array false", `{"a":[false]}`, `<r><a>false</a></r>`)
		test("basic array null", `{"a":[null]}`, `<r><a>null</a></r>`)

		test("basic array object digit", `{"a":[{"z":1}]}`, `<r><a><z>1</z></a></r>`)
		test("basic array of objects digit", `{"a":[{"z":1},{"y":2}]}`, `<r><a><z>1</z><y>2</y></a></r>`)
		test("basic array object string", `{"a":[{"z":"b"}]}`, `<r><a><z>b</z></a></r>`)
		test("basic array object true", `{"a":[{"z":true}]}`, `<r><a><z>true</z></a></r>`)
		test("basic array object false", `{"a":[{"z":false}]}`, `<r><a><z>false</z></a></r>`)
		test("basic array object null", `{"a":[{"z":null}]}`, `<r><a><z>null</z></a></r>`)

		test("basic array array digit", `{"a":[[1]]}`, `<r><a>1</a></r>`)
		test("basic array array string", `{"a":[["b"]]}`, `<r><a>b</a></r>`)
		test("basic array array true", `{"a":[[true]]}`, `<r><a>true</a></r>`)
		test("basic array array false", `{"a":[[false]]}`, `<r><a>false</a></r>`)
		test("basic array array null", `{"a":[[null]]}`, `<r><a>null</a></r>`)

		//test("invalid input", `{`, ``)
		//test("invalid input", `{"asd:1,"b":2}`, ``)
		//test("invalid input", `{"asd":1,"b":2.2}`, ``)

		r := gojax.NewXMLNode("root#123[a=1]")
		b := gojax.NewXMLNode("b")
		r.Add(b)
		n2 := gojax.NewXMLContentNode("2")
		b.Add(n2)
		c := gojax.NewXMLNode("c")
		r.Add(c)
		d := gojax.NewXMLNode("d")
		c.Add(d)
		n3 := gojax.NewXMLContentNode("3")
		d.Add(n3)
		tests += 1
		if r.StringWithOptions(options) != `<root a="1" id="123"><b>2</b><c><d>3</d></c></root>` {
			fmt.Println("Fail: simple node creation through API")
			fmt.Println(`  <root a="1" id="123"><b>2</b><c><d>3</d></c></root>`)
			fmt.Println(" ", r)
			failures += 1
		}

		rc := gojax.NewXMLNode("r")
		n2c := gojax.NewXMLContentNode("2")
		rc.Add(n2c)
		tests += 1
		if rc.StringWithOptions(options) != `<r>2</r>` || rc.AnyContent() != "2" {
			fmt.Println("Fail: extract content from a non-content node who's only child *is* a content node")
			fmt.Println(`  "2"`)
			fmt.Printf("  %q\n", rc.AnyContent())
			failures += 1
		}

		xa := gojax.NewXMLNode("a")
		xb := gojax.NewXMLNode("b")
		xc := gojax.NewXMLNode("c")
		xa.Add(xb)
		xa.Add(xc)
		xb.AddBefore(xc)
		tests += 1
		if xa.StringWithOptions(options) != `<a><c></c><b></b></a>` {
			fmt.Println("Fail: AddBefore")
			fmt.Println(`  <a><c></c><b></b></a>`)
			fmt.Println("  ", xa)
			failures += 1
		}

		xa = gojax.NewXMLNode("a")
		xb = gojax.NewXMLNode("b")
		xc = gojax.NewXMLNode("c")
		xa.Add(xb)
		err := xa.Prepend(xc)
		tests += 1
		if err != nil {
			fmt.Println("Fail: Prepend")
			fmt.Println(`  `, err)
			failures += 1
		}
		if xa.StringWithOptions(options) != `<a><c></c><b></b></a>` {
			fmt.Println("Fail: Prepend")
			fmt.Println(`  <a><c></c><b></b></a>`)
			fmt.Println("  ", xa)
			failures += 1
		}

		raf, err := gojax.FromJSON(`r`, `{"a":[{"b":{"c":1}}, {"b":{"c":1}}, {"b":{"c":2}}]}`)
		tests += 1
		if err != nil {
			fmt.Println("Failed to create tree to use decscendant content as a node filter")
			failures += 1
		}
		xnsaf, err := raf.Path(`//b[c=2]`)
		if err != nil {
			fmt.Println("Failed to call Path to use decscendant content as a node filter")
			failures += 1
		}
		if len(xnsaf) != 1 || xnsaf[0].StringWithOptions(options) != `<b><c>2</c></b>` {
			fmt.Println("Failed to use decscendant content as a node filter")
			fmt.Println(xnsaf)
			failures += 1
		}

		/*
			x, err := gojax.FromJSON("root", `{"a":{"b":1}}`)
			if err != nil {
				fmt.Println("Failed 2:", err)
			}
			fmt.Println("Paths:", x)
			fmt.Println(x.Path("/"))
			fmt.Println(x.Path("/root"))
			fmt.Println(x.Path("/root/a"))
			fmt.Println(x.Path("/root/a/b"))
			fmt.Println(x.Path("//a"))
			fmt.Println(x.Path("//b"))
			fmt.Println(x.Path("/root//b"))
			fmt.Println(x.Path("//a/b"))
			x, err = gojax.FromJSON("root", `{"a":[{"b":"c"},{"b":"d"}]}`)
			if err != nil {
				fmt.Println("Failed 2:", err)
			}
			fmt.Println(x.Path("//n"))
			fmt.Println(x.Path("//n[index=0]"))
			x, err = gojax.FromJSON("root", `{"a":1}`)
			if err != nil {
				fmt.Println("Failed 3:", err)
			}
			newx := x.Clone()
			newa, err := gojax.First(newx.Path(`//a`))
			if err != nil {
				fmt.Println("Failed 4:", err)
			}
			newa.Name = "b"
			xa, err := gojax.First(x.Path(`//a`))
			if err != nil {
				fmt.Println("Failed 5:", err)
			}
			xa.Name = "z"
			fmt.Println("Was 'a', is 'z'", x)
			fmt.Println("Was 'a', is 'b'", newx)
			x, err = gojax.FromJSON("root", `{"a":[{"b":"c"},{"b":"d"}]}`)
			if err != nil {
				fmt.Println("Failed 6:", err)
			}
			fmt.Println("Should return only one n with index 1")
			fmt.Println(x.Path("//n[position()=1]"))
			fmt.Println("Should return nothing")
			fmt.Println(x.Path("//n[position()=2]"))
			x, err = gojax.FromJSON("root", `{"a":0,"b":1,"c":{"a":2,"b":3},"d":{"b":4}}`)
			if err != nil {
				fmt.Println("Failed 7:", err)
			}
			fmt.Println("Should return two b with content 1 and 3")
			fmt.Println(x.Path("//b[position()=1]"))
			fmt.Println("Should return two b with content 1 and 3")
			fmt.Println(x.Path("//b[position()>0]"))
			fmt.Println("Should return three b with content 1 and 3 and 4")
			fmt.Println(x.Path("//b[position()>-1]"))
			fmt.Println("Should return one b with content 4")
			fmt.Println(x.Path("//b[position()<1]"))
			fmt.Println("Should return nothing")
			fmt.Println(x.Path("//b[position()=2]"))
			r = gojax.NewXMLNode("root")
			r.Set("a", "1")
			fmt.Println("Should return root node")
			fmt.Println(r.Path("/root[a=1]"))
			fmt.Println("Should return nothing")
			fmt.Println(r.Path("/root[meta=no_attrs]"))
			testFirstOr()
			sampleConstruction(`{"a":"b"}`)
			sampleConstruction(`{"a":"b","c":"d"}`)
			sampleConstruction(`{"a":{"b":"c","d":"e"}}`)
			sampleConstruction(`{"a":{
					"b":{"c":{"d":"e"}},
					"f":{"g":{"h":"i"}}
				}}`)
			sampleConstruction(`{"a":{
						"b":{"c":"d"},
						"e":{"f":"g"}
					}}`)
			sampleConstruction(`{"a":["b"]}`)
			sampleConstruction(`{"a":[{"b":"c"},{"d":"e"}]}`)
			testUnwrappedOr()
		*/
		fmt.Println("\n")
		fmt.Println("Passed:", (tests - failures), "out of", tests)
		fmt.Println("\n")
	}
}

func testFirstOr() {
	defer func() {
		if r := recover(); r != nil {
			fmt.Println("Caught:", r)
		}
	}()
	r := gojax.NewXMLNode("root")
	fmt.Println("Should return root node")
	fmt.Println(gojax.FirstOr(r, `/root`))
	fmt.Println("Should panic and return nothing")
	fmt.Println(gojax.FirstOr(r, `/not_a_node`))
}

func testUnwrappedOr() {
	defer func() {
		if r := recover(); r != nil {
			fmt.Println("Caught:", r)
		}
	}()
	fmt.Println("Should return root node")
	fmt.Println(gojax.UnwrappedOr(`{"a":"b","c":"d"}`))
	fmt.Println("Should convert name to a_b")
	fmt.Println(gojax.UnwrappedOr(`{"a-b":"c"}`))
	fmt.Println("Should convert name to a_b")
	fmt.Println(gojax.UnwrappedOr(`{"a b":"c"}`))
	fmt.Println("Should convert name to ns:a_b")
	fmt.Println(gojax.UnwrappedOr(`{"ns:a b":"c"}`))
	fmt.Println("Should panic and return nothing")
	fmt.Println(gojax.UnwrappedOr(`[]`))
}

func sampleConstruction(s string) {
	x, err := gojax.FromJSON("sample", s)
	if err != nil {
		panic(fmt.Sprintf("Failed to create a sample construction: %s %s %s", err, s, x))
	}
	fmt.Println("Input:")
	fmt.Println("\t", s)
	fmt.Println("Output:")
	fmt.Println("\t", x)
}


The raw file follows...


package main

import (
	"flag"
	"fmt"
	"git.ondollo.com/gojax"
	"io/ioutil"
	"os"
	// "strings"
)

var root = flag.String("root", "root", "")
var runTests = flag.Bool("t", false, "")
var inPath = flag.String("in", "", "")
var stringIn = flag.String("s", "", "")
var path = flag.String("p", "/*", "")
var allResults = flag.Bool("allResults", false, "")

var xStringIn = flag.String("xs", "", "")
var xStdin = flag.Bool("xstdin", false, "")

func main() {
	flag.Parse()

	if len(*xStringIn) > 0 {
		x, err := gojax.FromXML(*xStringIn)
		if err != nil {
			fmt.Println(err)
			os.Exit(1)
		}
		fmt.Println(x.JSON())
		os.Exit(0)
	}
	if *xStdin {
		bs, err := ioutil.ReadAll(os.Stdin)
		if err != nil {
			fmt.Println("Failed to read from STDIN:", err)
			os.Exit(1)
		}
		x, err := gojax.FromXML(string(bs))
		if err != nil {
			fmt.Println(err)
			os.Exit(1)
		}
		fmt.Println(x.JSON())
		os.Exit(0)
	}

	postProcess := false
	jsonFromFlags := ""
	if len(flag.Args()) == 1 && flag.Args()[0] == "-" {
		bs, err := ioutil.ReadAll(os.Stdin)
		if err != nil {
			fmt.Println("Failed to read from STDIN:", err)
			os.Exit(1)
		}
		jsonFromFlags = string(bs)
		postProcess = true
	}
	if len(*stringIn) > 0 {
		jsonFromFlags = *stringIn
		postProcess = true
	}
	if len(*inPath) > 0 && len(flag.Args()) == 0 {
		bs, err := ioutil.ReadFile(*inPath)
		if err != nil {
			fmt.Println("Failed to read given input file:", err)
			os.Exit(1)
		}
		if len(bs) == 0 {
			fmt.Println("Given file is empty:", *inPath)
			os.Exit(1)
		}
		jsonFromFlags = string(bs)
		postProcess = true
	}
	if postProcess {
		x, err := gojax.FromJSON(*root, jsonFromFlags)
		if err != nil {
			fmt.Println(err)
			os.Exit(1)
		}
		xns, err := x.Path(*path)
		if err != nil {
			fmt.Println(err)
			os.Exit(1)
		}
		if *allResults {
			for _, xn := range xns {
				fmt.Println(xn)
			}
		} else {
			if len(xns) > 0 {
				fmt.Println(xns[0])
			} else {
				fmt.Println("No results")
			}
		}
		os.Exit(0)
	}
	tests := 0
	failures := 0
	options := gojax.DefaultTestOptions
	test := func(s string, j string, r string) {
		tests += 1
		x, err := gojax.FromJSONWithOptions(`r`, j, options)
		if err != nil {
			failures += 1
			fmt.Println("Fail with error", s, err, x)
			//gojax.FromDebug(`r`, j)
		} else if x.StringWithOptions(options) != r {
			failures += 1
			fmt.Printf(`Fail: %s
  FROM: %s
  EXP: %s
  GOT: %s
`, s, j, r, x.StringWithOptions(options))
			//gojax.FromDebug(`r`, j)
		}
	}
	xtest := func(s string, xml string, r string) {
		tests += 1
		x, err := gojax.FromXMLWithOptions(xml, options)
		if err != nil {
			failures += 1
			fmt.Println("Fail with error", s, err, x)
			//gojax.FromDebug(`r`, j)
		} else if x.JSON() != r {
			failures += 1
			fmt.Printf(`Fail: %s
  FROM: %s
  EXP: %s
  GOT: %s
`, s, xml, r, x.JSON())
			//gojax.FromDebug(`r`, j)
		}
	}
	xxtest := func(s string, xml string, r string) {
		tests += 1
		x, err := gojax.FromXMLWithOptions(xml, options)
		if err != nil {
			failures += 1
			fmt.Println("Fail with error", s, err, x)
			//gojax.FromDebug(`r`, j)
		} else if x.StringWithOptions(options) != r {
			failures += 1
			fmt.Printf(`Fail: %s
  FROM: %s
  EXP: %s
  GOT: %s
`, s, xml, r, x.StringWithOptions(options))
			//gojax.FromDebug(`r`, j)
		}
	}
	_nop := func(s string, j string, r string) { /*fmt.Println("Unsupported:", s, j)*/ }
	if *runTests {

		//fmt.Println(strings.Repeat("\n", 40))
		xtest("empty short node", `<a/>`, `{"a":""}`)
		xtest("empty paired node", `<a></a>`, `{"a":""}`)
		xtest("paired node with a sentence", `<a>This is a sentence.</a>`, `{"a":"This is a sentence."}`)
		xtest("paired node with cdata",
			`<a><![CDATA[something that should be <i>just</i> consumed]]></a>`,
			`{"a":"something that should be <i>just</i> consumed"}`)
		xtest("simple paired node", `<a>1</a>`, `{"a":"1"}`)
		xtest("simple paired/short nodes", `<a><b/></a>`, `{"a":{"b":""}}`)
		xtest("simple paired/paired nodes", `<a><b></b></a>`, `{"a":{"b":""}}`)
		xtest("simple paired/paired nodes with inner content", `<a><b>1</b></a>`, `{"a":{"b":"1"}}`)

		xtest("more than one child in simple node", `<a><b></b><c></c></a>`, `{"a":{"b":"","c":""}}`)
		xtest("more than one child with content in simple node", `<a><b>1</b><c>2</c></a>`, `{"a":{"b":"1","c":"2"}}`)
		xtest("more than one child with content and repeats in simple node",
			`<a><b>1</b><c>1</c><b>1</b></a>`, `{"a":{"b":["1","1"],"c":"1"}}`)

		xtest("empty short node w/attr", `<a b="1"/>`, `{"a":""}`)
		xtest("empty short node w/attrs", `<a b="1" c="2"/>`, `{"a":""}`)
		xtest("empty paired node w/attr", `<a b="1"></a>`, `{"a":""}`)
		xtest("empty paired node w/attrs", `<a b="1" c="2"></a>`, `{"a":""}`)
		xtest("simple paired node w/attr", `<a b="1">1</a>`, `{"a":"1"}`)
		xtest("simple paired node w/attrs", `<a b="1" c="2">1</a>`, `{"a":"1"}`)
		xtest("simple paired/short nodes w/attr", `<a><b c="1"/></a>`, `{"a":{"b":""}}`)
		xtest("simple paired/short nodes w/attrs", `<a><b c="1" d="2"/></a>`, `{"a":{"b":""}}`)
		xtest("simple paired/paired nodes w/attr", `<a><b c="1"></b></a>`, `{"a":{"b":""}}`)
		xtest("simple paired/paired nodes w/attrs", `<a><b c="1" d="2"></b></a>`, `{"a":{"b":""}}`)
		xtest("simple paired/paired nodes with inner content w/attr", `<a><b c="2">1</b></a>`, `{"a":{"b":"1"}}`)
		xtest("simple paired/paired nodes with inner content w/attrs", `<a><b c="2" d="3">1</b></a>`, `{"a":{"b":"1"}}`)

		xtest("more than one child in simple node w/attr", `<a><b d="1"></b><c e="2"></c></a>`, `{"a":{"b":"","c":""}}`)
		xtest("more than one child in simple node w/attrs",
			`<a><b d="1" e="2"></b><c f="3" g="4"></c></a>`, `{"a":{"b":"","c":""}}`)
		xtest("more than one child with content in simple node w/attr",
			`<a><b d="1">1</b><c e="2">2</c></a>`, `{"a":{"b":"1","c":"2"}}`)
		xtest("more than one child with content in simple node w/attrs",
			`<a><b d="1" e="2">1</b><c f="3" g="4">2</c></a>`, `{"a":{"b":"1","c":"2"}}`)
		xtest("more than one child with content and repeats in simple node w/attr",
			`<a><b d="1">1</b><c e="2">1</c><b f="3">1</b></a>`, `{"a":{"b":["1","1"],"c":"1"}}`)
		xtest("more than one child with content and repeats in simple node w/attrs",
			`<a><b d="1" e="2">1</b><c f="3" g="4">1</c><b h="5" i="6">1</b></a>`,
			`{"a":{"b":["1","1"],"c":"1"}}`)

		xtest("node w/children and w/attr", `<a id="1"><b/></a>`, `{"a":{"id":"1","b":""}}`)

		xtest("complex",
			`<a><b>1</b><c>2</c><d/><e>3</e><f>4</f></a>`,
			`{"a":{"b":"1","c":"2","d":"","e":"3","f":"4"}}`)

		xtest("repeated paired/paired nodes",
			`<a><b><c>1</c></b><b><c>2</c></b></a>`,
			`{"a":{"b":[{"c":"1"},{"c":"2"}]}}`)

		xxtest("match paired node with markup",
			`<a>something that should be & just consumed</a>`,
			`<a><![CDATA[something that should be & just consumed]]></a>`,
		)
		xxtest("match paired node with cdata",
			`<a><![CDATA[something that should be <i>just</i> consumed]]></a>`,
			`<a><![CDATA[something that should be <i>just</i> consumed]]></a>`,
		)

		_nop("empty object", `{}`, `<r></r>`)
		test("basic digit", `{"a":1}`, `<r><a>1</a></r>`)
		test("basic string", `{"a":"b"}`, `<r><a>b</a></r>`)
		test("basic string with markup", `{"a":"b<c>d</c>"}`, `<r><a><![CDATA[b<c>d</c>]]></a></r>`)
		_nop("basic object", `{"a":{}}`, `<r><a></a></r>`)
		_nop("basic array", `{"a":[]}`, `<r><a></a></r>`)
		test("basic true", `{"a":true}`, `<r><a>true</a></r>`)
		test("basic false", `{"a":false}`, `<r><a>false</a></r>`)
		test("basic null", `{"a":null}`, `<r><a>null</a></r>`)

		test("basic id with digit", `{"a#theID":1}`, `<r><a id="theID">1</a></r>`)

		test("basic object digit", `{"a":{"b":1}}`, `<r><a><b>1</b></a></r>`)
		test("basic object string", `{"a":{"b":"c"}}`, `<r><a><b>c</b></a></r>`)
		test("basic object true", `{"a":{"b":true}}`, `<r><a><b>true</b></a></r>`)
		test("basic object false", `{"a":{"b":false}}`, `<r><a><b>false</b></a></r>`)
		test("basic object null", `{"a":{"b":null}}`, `<r><a><b>null</b></a></r>`)

		test("basic array digit", `{"a":[1]}`, `<r><a><n index="0">1</n></a></r>`)
		test("basic array string", `{"a":["b"]}`, `<r><a><n index="0">b</n></a></r>`)
		test("basic array true", `{"a":[true]}`, `<r><a><n index="0">true</n></a></r>`)
		test("basic array false", `{"a":[false]}`, `<r><a><n index="0">false</n></a></r>`)
		test("basic array null", `{"a":[null]}`, `<r><a><n index="0">null</n></a></r>`)

		test("basic array object digit", `{"a":[{"z":1}]}`, `<r><a><n index="0"><z>1</z></n></a></r>`)
		test("basic array of objects digit", `{"a":[{"z":1},{"y":2}]}`, `<r><a><n index="0"><z>1</z></n><n index="1"><y>2</y></n></a></r>`)
		test("basic array object string", `{"a":[{"z":"b"}]}`, `<r><a><n index="0"><z>b</z></n></a></r>`)
		test("basic array object true", `{"a":[{"z":true}]}`, `<r><a><n index="0"><z>true</z></n></a></r>`)
		test("basic array object false", `{"a":[{"z":false}]}`, `<r><a><n index="0"><z>false</z></n></a></r>`)
		test("basic array object null", `{"a":[{"z":null}]}`, `<r><a><n index="0"><z>null</z></n></a></r>`)

		test("basic array array digit", `{"a":[[1]]}`, `<r><a><n index="0"><n index="0">1</n></n></a></r>`)
		test("basic array array string", `{"a":[["b"]]}`, `<r><a><n index="0"><n index="0">b</n></n></a></r>`)
		test("basic array array true", `{"a":[[true]]}`, `<r><a><n index="0"><n index="0">true</n></n></a></r>`)
		test("basic array array false", `{"a":[[false]]}`, `<r><a><n index="0"><n index="0">false</n></n></a></r>`)
		test("basic array array null", `{"a":[[null]]}`, `<r><a><n index="0"><n index="0">null</n></n></a></r>`)

		options.WrapArrayElements = false
		test("basic array digit", `{"a":[1]}`, `<r><a>1</a></r>`)
		test("basic array string", `{"a":["b"]}`, `<r><a>b</a></r>`)
		test("basic array true", `{"a":[true]}`, `<r><a>true</a></r>`)
		test("basic array false", `{"a":[false]}`, `<r><a>false</a></r>`)
		test("basic array null", `{"a":[null]}`, `<r><a>null</a></r>`)

		test("basic array object digit", `{"a":[{"z":1}]}`, `<r><a><z>1</z></a></r>`)
		test("basic array of objects digit", `{"a":[{"z":1},{"y":2}]}`, `<r><a><z>1</z><y>2</y></a></r>`)
		test("basic array object string", `{"a":[{"z":"b"}]}`, `<r><a><z>b</z></a></r>`)
		test("basic array object true", `{"a":[{"z":true}]}`, `<r><a><z>true</z></a></r>`)
		test("basic array object false", `{"a":[{"z":false}]}`, `<r><a><z>false</z></a></r>`)
		test("basic array object null", `{"a":[{"z":null}]}`, `<r><a><z>null</z></a></r>`)

		test("basic array array digit", `{"a":[[1]]}`, `<r><a>1</a></r>`)
		test("basic array array string", `{"a":[["b"]]}`, `<r><a>b</a></r>`)
		test("basic array array true", `{"a":[[true]]}`, `<r><a>true</a></r>`)
		test("basic array array false", `{"a":[[false]]}`, `<r><a>false</a></r>`)
		test("basic array array null", `{"a":[[null]]}`, `<r><a>null</a></r>`)

		//test("invalid input", `{`, ``)
		//test("invalid input", `{"asd:1,"b":2}`, ``)
		//test("invalid input", `{"asd":1,"b":2.2}`, ``)

		r := gojax.NewXMLNode("root#123[a=1]")
		b := gojax.NewXMLNode("b")
		r.Add(b)
		n2 := gojax.NewXMLContentNode("2")
		b.Add(n2)
		c := gojax.NewXMLNode("c")
		r.Add(c)
		d := gojax.NewXMLNode("d")
		c.Add(d)
		n3 := gojax.NewXMLContentNode("3")
		d.Add(n3)
		tests += 1
		if r.StringWithOptions(options) != `<root a="1" id="123"><b>2</b><c><d>3</d></c></root>` {
			fmt.Println("Fail: simple node creation through API")
			fmt.Println(`  <root a="1" id="123"><b>2</b><c><d>3</d></c></root>`)
			fmt.Println(" ", r)
			failures += 1
		}

		rc := gojax.NewXMLNode("r")
		n2c := gojax.NewXMLContentNode("2")
		rc.Add(n2c)
		tests += 1
		if rc.StringWithOptions(options) != `<r>2</r>` || rc.AnyContent() != "2" {
			fmt.Println("Fail: extract content from a non-content node who's only child *is* a content node")
			fmt.Println(`  "2"`)
			fmt.Printf("  %q\n", rc.AnyContent())
			failures += 1
		}

		xa := gojax.NewXMLNode("a")
		xb := gojax.NewXMLNode("b")
		xc := gojax.NewXMLNode("c")
		xa.Add(xb)
		xa.Add(xc)
		xb.AddBefore(xc)
		tests += 1
		if xa.StringWithOptions(options) != `<a><c></c><b></b></a>` {
			fmt.Println("Fail: AddBefore")
			fmt.Println(`  <a><c></c><b></b></a>`)
			fmt.Println("  ", xa)
			failures += 1
		}

		xa = gojax.NewXMLNode("a")
		xb = gojax.NewXMLNode("b")
		xc = gojax.NewXMLNode("c")
		xa.Add(xb)
		err := xa.Prepend(xc)
		tests += 1
		if err != nil {
			fmt.Println("Fail: Prepend")
			fmt.Println(`  `, err)
			failures += 1
		}
		if xa.StringWithOptions(options) != `<a><c></c><b></b></a>` {
			fmt.Println("Fail: Prepend")
			fmt.Println(`  <a><c></c><b></b></a>`)
			fmt.Println("  ", xa)
			failures += 1
		}

		raf, err := gojax.FromJSON(`r`, `{"a":[{"b":{"c":1}}, {"b":{"c":1}}, {"b":{"c":2}}]}`)
		tests += 1
		if err != nil {
			fmt.Println("Failed to create tree to use decscendant content as a node filter")
			failures += 1
		}
		xnsaf, err := raf.Path(`//b[c=2]`)
		if err != nil {
			fmt.Println("Failed to call Path to use decscendant content as a node filter")
			failures += 1
		}
		if len(xnsaf) != 1 || xnsaf[0].StringWithOptions(options) != `<b><c>2</c></b>` {
			fmt.Println("Failed to use decscendant content as a node filter")
			fmt.Println(xnsaf)
			failures += 1
		}

		/*
			x, err := gojax.FromJSON("root", `{"a":{"b":1}}`)
			if err != nil {
				fmt.Println("Failed 2:", err)
			}
			fmt.Println("Paths:", x)
			fmt.Println(x.Path("/"))
			fmt.Println(x.Path("/root"))
			fmt.Println(x.Path("/root/a"))
			fmt.Println(x.Path("/root/a/b"))
			fmt.Println(x.Path("//a"))
			fmt.Println(x.Path("//b"))
			fmt.Println(x.Path("/root//b"))
			fmt.Println(x.Path("//a/b"))
			x, err = gojax.FromJSON("root", `{"a":[{"b":"c"},{"b":"d"}]}`)
			if err != nil {
				fmt.Println("Failed 2:", err)
			}
			fmt.Println(x.Path("//n"))
			fmt.Println(x.Path("//n[index=0]"))
			x, err = gojax.FromJSON("root", `{"a":1}`)
			if err != nil {
				fmt.Println("Failed 3:", err)
			}
			newx := x.Clone()
			newa, err := gojax.First(newx.Path(`//a`))
			if err != nil {
				fmt.Println("Failed 4:", err)
			}
			newa.Name = "b"
			xa, err := gojax.First(x.Path(`//a`))
			if err != nil {
				fmt.Println("Failed 5:", err)
			}
			xa.Name = "z"
			fmt.Println("Was 'a', is 'z'", x)
			fmt.Println("Was 'a', is 'b'", newx)
			x, err = gojax.FromJSON("root", `{"a":[{"b":"c"},{"b":"d"}]}`)
			if err != nil {
				fmt.Println("Failed 6:", err)
			}
			fmt.Println("Should return only one n with index 1")
			fmt.Println(x.Path("//n[position()=1]"))
			fmt.Println("Should return nothing")
			fmt.Println(x.Path("//n[position()=2]"))
			x, err = gojax.FromJSON("root", `{"a":0,"b":1,"c":{"a":2,"b":3},"d":{"b":4}}`)
			if err != nil {
				fmt.Println("Failed 7:", err)
			}
			fmt.Println("Should return two b with content 1 and 3")
			fmt.Println(x.Path("//b[position()=1]"))
			fmt.Println("Should return two b with content 1 and 3")
			fmt.Println(x.Path("//b[position()>0]"))
			fmt.Println("Should return three b with content 1 and 3 and 4")
			fmt.Println(x.Path("//b[position()>-1]"))
			fmt.Println("Should return one b with content 4")
			fmt.Println(x.Path("//b[position()<1]"))
			fmt.Println("Should return nothing")
			fmt.Println(x.Path("//b[position()=2]"))
			r = gojax.NewXMLNode("root")
			r.Set("a", "1")
			fmt.Println("Should return root node")
			fmt.Println(r.Path("/root[a=1]"))
			fmt.Println("Should return nothing")
			fmt.Println(r.Path("/root[meta=no_attrs]"))
			testFirstOr()
			sampleConstruction(`{"a":"b"}`)
			sampleConstruction(`{"a":"b","c":"d"}`)
			sampleConstruction(`{"a":{"b":"c","d":"e"}}`)
			sampleConstruction(`{"a":{
					"b":{"c":{"d":"e"}},
					"f":{"g":{"h":"i"}}
				}}`)
			sampleConstruction(`{"a":{
						"b":{"c":"d"},
						"e":{"f":"g"}
					}}`)
			sampleConstruction(`{"a":["b"]}`)
			sampleConstruction(`{"a":[{"b":"c"},{"d":"e"}]}`)
			testUnwrappedOr()
		*/
		fmt.Println("\n")
		fmt.Println("Passed:", (tests - failures), "out of", tests)
		fmt.Println("\n")
	}
}

func testFirstOr() {
	defer func() {
		if r := recover(); r != nil {
			fmt.Println("Caught:", r)
		}
	}()
	r := gojax.NewXMLNode("root")
	fmt.Println("Should return root node")
	fmt.Println(gojax.FirstOr(r, `/root`))
	fmt.Println("Should panic and return nothing")
	fmt.Println(gojax.FirstOr(r, `/not_a_node`))
}

func testUnwrappedOr() {
	defer func() {
		if r := recover(); r != nil {
			fmt.Println("Caught:", r)
		}
	}()
	fmt.Println("Should return root node")
	fmt.Println(gojax.UnwrappedOr(`{"a":"b","c":"d"}`))
	fmt.Println("Should convert name to a_b")
	fmt.Println(gojax.UnwrappedOr(`{"a-b":"c"}`))
	fmt.Println("Should convert name to a_b")
	fmt.Println(gojax.UnwrappedOr(`{"a b":"c"}`))
	fmt.Println("Should convert name to ns:a_b")
	fmt.Println(gojax.UnwrappedOr(`{"ns:a b":"c"}`))
	fmt.Println("Should panic and return nothing")
	fmt.Println(gojax.UnwrappedOr(`[]`))
}

func sampleConstruction(s string) {
	x, err := gojax.FromJSON("sample", s)
	if err != nil {
		panic(fmt.Sprintf("Failed to create a sample construction: %s %s %s", err, s, x))
	}
	fmt.Println("Input:")
	fmt.Println("\t", s)
	fmt.Println("Output:")
	fmt.Println("\t", x)
}