| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519 | // Package jsonpath implements Stefan Goener's JSONPath http://goessner.net/articles/JsonPath///// A jsonpath applies to any JSON decoded data using interface{} when// decoded with encoding/json (http://golang.org/pkg/encoding/json/) :////    var bookstore interface{}//    err := json.Unmarshal(data, &bookstore)//    authors, err := jsonpath.Read(bookstore, "$..authors")//// A jsonpath expression can be prepared to be reused multiple times :////    allAuthors, err := jsonpath.Prepare("$..authors")//    ...//    var bookstore interface{}//    err = json.Unmarshal(data, &bookstore)//    authors, err := allAuthors(bookstore)//// The type of the values returned by the `Read` method or `Prepare`// functions depends on the jsonpath expression.//// Limitations//// No support for subexpressions and filters.// Strings in brackets must use double quotes.// It cannot operate on JSON decoded struct fields.//package jsonpathimport (	"errors"	"fmt"	"strconv"	"strings"	"text/scanner")// Read a path from a decoded JSON array or object ([]interface{} or map[string]interface{})// and returns the corresponding value or an error.//// The returned value type depends on the requested path and the JSON value.func Read(value interface{}, path string) (interface{}, error) {	filter, err := Prepare(path)	if err != nil {		return nil, err	}	return filter(value)}// Prepare will parse the path and return a filter function that can then be applied to decoded JSON values.func Prepare(path string) (FilterFunc, error) {	p := newScanner(path)	if err := p.parse(); err != nil {		return nil, err	}	return p.prepareFilterFunc(), nil}// FilterFunc applies a prepared json path to a JSON decoded valuetype FilterFunc func(value interface{}) (interface{}, error)// short variables// p: the parser context// r: root node => @// c: current node => $// a: the list of actions to apply next// v: value// actionFunc applies a transformation to current value (possibily using root)// then applies the next action from actions (using next()) to the output of the transformationtype actionFunc func(r, c interface{}, a actions) (interface{}, error)// a list of action functions to apply one after the othertype actions []actionFunc// next applies the next action functionfunc (a actions) next(r, c interface{}) (interface{}, error) {	return a[0](r, c, a[1:])}// call applies the next action function without taking it outfunc (a actions) call(r, c interface{}) (interface{}, error) {	return a[0](r, c, a)}type exprFunc func(r, c interface{}) (interface{}, error)type parser struct {	scanner scanner.Scanner	path    string	actions actions}func (p *parser) prepareFilterFunc() FilterFunc {	actions := p.actions	return func(value interface{}) (interface{}, error) {		return actions.next(value, value)	}}func newScanner(path string) *parser {	return &parser{path: path}}func (p *parser) scan() rune {	return p.scanner.Scan()}func (p *parser) text() string {	return p.scanner.TokenText()}func (p *parser) column() int {	return p.scanner.Position.Column}func (p *parser) peek() rune {	return p.scanner.Peek()}func (p *parser) add(action actionFunc) {	p.actions = append(p.actions, action)}func (p *parser) parse() error {	p.scanner.Init(strings.NewReader(p.path))	if p.scan() != '$' {		return errors.New("path must start with a '$'")	}	return p.parsePath()}func (p *parser) parsePath() (err error) {	for err == nil {		switch p.scan() {		case '.':			p.scanner.Mode = scanner.ScanIdents			switch p.scan() {			case scanner.Ident:				err = p.parseObjAccess()			case '*':				err = p.prepareWildcard()			case '.':				err = p.parseDeep()			default:				err = fmt.Errorf("expected JSON child identifier after '.' at %d", p.column())			}		case '[':			err = p.parseBracket()		case scanner.EOF:			// the end, add a last func that just return current node			p.add(func(r, c interface{}, a actions) (interface{}, error) { return c, nil })			return nil		default:			err = fmt.Errorf("unexcepted token %s at %d", p.text(), p.column())		}	}	return}func (p *parser) parseObjAccess() error {	ident := p.text()	column := p.scanner.Position.Column	p.add(func(r, c interface{}, a actions) (interface{}, error) {		obj, ok := c.(map[string]interface{})		if !ok {			return nil, fmt.Errorf("expected JSON object to access child '%s' at %d", ident, column)		}		if c, ok = obj[ident]; !ok {			return nil, fmt.Errorf("child '%s' not found in JSON object at %d", ident, column)		}		return a.next(r, c)	})	return nil}func (p *parser) prepareWildcard() error {	p.add(func(r, c interface{}, a actions) (interface{}, error) {		values := []interface{}{}		if obj, ok := c.(map[string]interface{}); ok {			for _, v := range obj {				v, err := a.next(r, v)				if err != nil {					continue				}				values = append(values, v)			}		} else if array, ok := c.([]interface{}); ok {			for _, v := range array {				v, err := a.next(r, v)				if err != nil {					continue				}				values = append(values, v)			}		}		return values, nil	})	return nil}func (p *parser) parseDeep() (err error) {	p.add(func(r, c interface{}, a actions) (interface{}, error) {		return recSearch(r, c, a, []interface{}{}), nil	})	p.scanner.Mode = scanner.ScanIdents	switch p.scan() {	case scanner.Ident:		return p.parseObjAccess()	case '*':		p.add(func(r, c interface{}, a actions) (interface{}, error) { return a.next(r, c) })		return nil	case '[':		return p.parseBracket()	case scanner.EOF:		return fmt.Errorf("cannot end with a scan '..' at %d", p.column())	default:		return fmt.Errorf("unexpected token '%s' after deep search '..' at %d", p.text(), p.column())	}}// bracket contains filter, wildcard or array accessfunc (p *parser) parseBracket() error {	if p.peek() == '?' {		return p.parseFilter()	} else if p.peek() == '*' {		p.scan() // eat *		if p.scan() != ']' {			return fmt.Errorf("expected closing bracket after [* at %d", p.column())		}		return p.prepareWildcard()	}	return p.parseArray()}// array contains either a union [,,,], a slice [::] or a single element.// Each element can be an int, a string or an expression.// TODO optimize map/array access (by detecting the type of indexes)func (p *parser) parseArray() error {	var indexes []interface{} // string, int or exprFunc	var mode string           // slice or union	p.scanner.Mode = scanner.ScanIdents | scanner.ScanStrings | scanner.ScanIntsparse:	for {		// parse value		switch p.scan() {		case scanner.Int:			index, err := strconv.Atoi(p.text())			if err != nil {				return fmt.Errorf("%s at %d", err.Error(), p.column())			}			indexes = append(indexes, index)		case '-':			if p.scan() != scanner.Int {				return fmt.Errorf("expect an int after the minus '-' sign at %d", p.column())			}			index, err := strconv.Atoi(p.text())			if err != nil {				return fmt.Errorf("%s at %d", err.Error(), p.column())			}			indexes = append(indexes, -index)		case scanner.Ident:			indexes = append(indexes, p.text())		case scanner.String:			s, err := strconv.Unquote(p.text())			if err != nil {				return fmt.Errorf("bad string %s at %d", err, p.column())			}			indexes = append(indexes, s)		case '(':			filter, err := p.parseExpression()			if err != nil {				return err			}			indexes = append(indexes, filter)		case ':': // when slice value is ommited			if mode == "" {				mode = "slice"				indexes = append(indexes, 0)			} else if mode == "slice" {				indexes = append(indexes, 0)			} else {				return fmt.Errorf("unexpected ':' after %s at %d", mode, p.column())			}			continue // skip separator parsing, it's done		case ']': // when slice value is ommited			if mode == "slice" {				indexes = append(indexes, 0)			} else if len(indexes) == 0 {				return fmt.Errorf("expected at least one key, index or expression at %d", p.column())			}			break parse		case scanner.EOF:			return fmt.Errorf("unexpected end of path at %d", p.column())		default:			return fmt.Errorf("unexpected token '%s' at %d", p.text(), p.column())		}		// parse separator		switch p.scan() {		case ',':			if mode == "" {				mode = "union"			} else if mode != "union" {				return fmt.Errorf("unexpeted ',' in %s at %d", mode, p.column())			}		case ':':			if mode == "" {				mode = "slice"			} else if mode != "slice" {				return fmt.Errorf("unexpected ':' in %s at %d", mode, p.column())			}		case ']':			break parse		case scanner.EOF:			return fmt.Errorf("unexpected end of path at %d", p.column())		default:			return fmt.Errorf("unexpected token '%s' at %d", p.text(), p.column())		}	}	if mode == "slice" {		if len(indexes) > 3 {			return fmt.Errorf("bad range syntax [start:end:step] at %d", p.column())		}		p.add(prepareSlice(indexes, p.column()))	} else if len(indexes) == 1 {		p.add(prepareIndex(indexes[0], p.column()))	} else {		p.add(prepareUnion(indexes, p.column()))	}	return nil}func (p *parser) parseFilter() error {	return errors.New("Filters are not (yet) implemented")}func (p *parser) parseExpression() (exprFunc, error) {	return nil, errors.New("Expression are not (yet) implemented")}func recSearch(r, c interface{}, a actions, acc []interface{}) []interface{} {	if obj, ok := c.(map[string]interface{}); ok {		for _, c := range obj {			if result, err := a.next(r, c); err == nil {				acc = append(acc, result)			}			acc = recSearch(r, c, a, acc)		}	} else if array, ok := c.([]interface{}); ok {		for _, c := range array {			if result, err := a.next(r, c); err == nil {				acc = append(acc, result)			}			acc = recSearch(r, c, a, acc)		}	}	return acc}func prepareIndex(index interface{}, column int) actionFunc {	return func(r, c interface{}, a actions) (interface{}, error) {		if obj, ok := c.(map[string]interface{}); ok {			key, err := indexAsString(index, r, c)			if err != nil {				return nil, err			}			if c, ok = obj[key]; !ok {				return nil, fmt.Errorf("no key '%s' for object at %d", key, column)			}			return a.next(r, c)		} else if array, ok := c.([]interface{}); ok {			index, err := indexAsInt(index, r, c)			if err != nil {				return nil, err			}			if index < 0 || index >= len(array) {				return nil, fmt.Errorf("out of bound array access at %d", column)			}			return a.next(r, array[index])		}		return nil, fmt.Errorf("expected array or object at %d", column)	}}func prepareSlice(indexes []interface{}, column int) actionFunc {	return func(r, c interface{}, a actions) (interface{}, error) {		array, ok := c.([]interface{})		if !ok {			return nil, fmt.Errorf("expected JSON array at %d", column)		}		var err error		var start, end, step int		if start, err = indexAsInt(indexes[0], r, c); err != nil {			return nil, err		}		if end, err = indexAsInt(indexes[1], r, c); err != nil {			return nil, err		}		if len(indexes) > 2 {			if step, err = indexAsInt(indexes[2], r, c); err != nil {				return nil, err			}		}		max := len(array)		start = negmax(start, max)		if end == 0 {			end = max		} else {			end = negmax(end, max)		}		if start > end {			return nil, fmt.Errorf("cannot start range at %d and end at %d", start, end)		}		if step == 0 {			step = 1		}		var values []interface{}		if step > 0 {			for i := start; i < end; i += step {				values = append(values, array[i])			}		} else { // reverse order on negative step			for i := end - 1; i >= start; i += step {				values = append(values, array[i])			}		}		return values, nil	}}func prepareUnion(indexes []interface{}, column int) actionFunc {	return func(r, c interface{}, a actions) (interface{}, error) {		if obj, ok := c.(map[string]interface{}); ok {			var values []interface{}			for _, index := range indexes {				key, err := indexAsString(index, r, c)				if err != nil {					return nil, err				}				if c, ok = obj[key]; !ok {					return nil, fmt.Errorf("no key '%s' for object at %d", key, column)				}				if c, err = a.next(r, c); err != nil {					return nil, err				}				values = append(values, c)			}			return values, nil		} else if array, ok := c.([]interface{}); ok {			var values []interface{}			for _, index := range indexes {				index, err := indexAsInt(index, r, c)				if err != nil {					return nil, err				}				if index < 0 || index >= len(array) {					return nil, fmt.Errorf("out of bound array access at %d", column)				}				if c, err = a.next(r, array[index]); err != nil {					return nil, err				}				values = append(values, c)			}			return values, nil		}		return nil, fmt.Errorf("expected array or object at %d", column)	}}func negmax(n, max int) int {	if n < 0 {		n = max + n		if n < 0 {			n = 0		}	} else if n > max {		return max	}	return n}func indexAsInt(index, r, c interface{}) (int, error) {	switch i := index.(type) {	case int:		return i, nil	case exprFunc:		index, err := i(r, c)		if err != nil {			return 0, err		}		switch i := index.(type) {		case int:			return i, nil		default:			return 0, fmt.Errorf("expected expression to return an index for array access")		}	default:		return 0, fmt.Errorf("expected index value (integer or expression returning an integer) for array access")	}}func indexAsString(key, r, c interface{}) (string, error) {	switch s := key.(type) {	case string:		return s, nil	case exprFunc:		key, err := s(r, c)		if err != nil {			return "", err		}		switch s := key.(type) {		case string:			return s, nil		default:			return "", fmt.Errorf("expected expression to return a key for object access")		}	default:		return "", fmt.Errorf("expected key value (string or expression returning a string) for object access")	}}
 |