forked from TrueCloudLab/distribution
a685e3fc98
Vndr has a simpler configuration and allows pointing to forked packages. Additionally other docker projects are now using vndr making vendoring in distribution more consistent. Updates letsencrypt to use fork. No longer uses sub-vendored packages. Signed-off-by: Derek McGowan <derek@mcgstyle.net> (github: dmcgowan)
603 lines
15 KiB
Go
603 lines
15 KiB
Go
package jmespath
|
|
|
|
import (
|
|
"encoding/json"
|
|
"fmt"
|
|
"strconv"
|
|
"strings"
|
|
)
|
|
|
|
type astNodeType int
|
|
|
|
//go:generate stringer -type astNodeType
|
|
const (
|
|
ASTEmpty astNodeType = iota
|
|
ASTComparator
|
|
ASTCurrentNode
|
|
ASTExpRef
|
|
ASTFunctionExpression
|
|
ASTField
|
|
ASTFilterProjection
|
|
ASTFlatten
|
|
ASTIdentity
|
|
ASTIndex
|
|
ASTIndexExpression
|
|
ASTKeyValPair
|
|
ASTLiteral
|
|
ASTMultiSelectHash
|
|
ASTMultiSelectList
|
|
ASTOrExpression
|
|
ASTAndExpression
|
|
ASTNotExpression
|
|
ASTPipe
|
|
ASTProjection
|
|
ASTSubexpression
|
|
ASTSlice
|
|
ASTValueProjection
|
|
)
|
|
|
|
// ASTNode represents the abstract syntax tree of a JMESPath expression.
|
|
type ASTNode struct {
|
|
nodeType astNodeType
|
|
value interface{}
|
|
children []ASTNode
|
|
}
|
|
|
|
func (node ASTNode) String() string {
|
|
return node.PrettyPrint(0)
|
|
}
|
|
|
|
// PrettyPrint will pretty print the parsed AST.
|
|
// The AST is an implementation detail and this pretty print
|
|
// function is provided as a convenience method to help with
|
|
// debugging. You should not rely on its output as the internal
|
|
// structure of the AST may change at any time.
|
|
func (node ASTNode) PrettyPrint(indent int) string {
|
|
spaces := strings.Repeat(" ", indent)
|
|
output := fmt.Sprintf("%s%s {\n", spaces, node.nodeType)
|
|
nextIndent := indent + 2
|
|
if node.value != nil {
|
|
if converted, ok := node.value.(fmt.Stringer); ok {
|
|
// Account for things like comparator nodes
|
|
// that are enums with a String() method.
|
|
output += fmt.Sprintf("%svalue: %s\n", strings.Repeat(" ", nextIndent), converted.String())
|
|
} else {
|
|
output += fmt.Sprintf("%svalue: %#v\n", strings.Repeat(" ", nextIndent), node.value)
|
|
}
|
|
}
|
|
lastIndex := len(node.children)
|
|
if lastIndex > 0 {
|
|
output += fmt.Sprintf("%schildren: {\n", strings.Repeat(" ", nextIndent))
|
|
childIndent := nextIndent + 2
|
|
for _, elem := range node.children {
|
|
output += elem.PrettyPrint(childIndent)
|
|
}
|
|
}
|
|
output += fmt.Sprintf("%s}\n", spaces)
|
|
return output
|
|
}
|
|
|
|
var bindingPowers = map[tokType]int{
|
|
tEOF: 0,
|
|
tUnquotedIdentifier: 0,
|
|
tQuotedIdentifier: 0,
|
|
tRbracket: 0,
|
|
tRparen: 0,
|
|
tComma: 0,
|
|
tRbrace: 0,
|
|
tNumber: 0,
|
|
tCurrent: 0,
|
|
tExpref: 0,
|
|
tColon: 0,
|
|
tPipe: 1,
|
|
tOr: 2,
|
|
tAnd: 3,
|
|
tEQ: 5,
|
|
tLT: 5,
|
|
tLTE: 5,
|
|
tGT: 5,
|
|
tGTE: 5,
|
|
tNE: 5,
|
|
tFlatten: 9,
|
|
tStar: 20,
|
|
tFilter: 21,
|
|
tDot: 40,
|
|
tNot: 45,
|
|
tLbrace: 50,
|
|
tLbracket: 55,
|
|
tLparen: 60,
|
|
}
|
|
|
|
// Parser holds state about the current expression being parsed.
|
|
type Parser struct {
|
|
expression string
|
|
tokens []token
|
|
index int
|
|
}
|
|
|
|
// NewParser creates a new JMESPath parser.
|
|
func NewParser() *Parser {
|
|
p := Parser{}
|
|
return &p
|
|
}
|
|
|
|
// Parse will compile a JMESPath expression.
|
|
func (p *Parser) Parse(expression string) (ASTNode, error) {
|
|
lexer := NewLexer()
|
|
p.expression = expression
|
|
p.index = 0
|
|
tokens, err := lexer.tokenize(expression)
|
|
if err != nil {
|
|
return ASTNode{}, err
|
|
}
|
|
p.tokens = tokens
|
|
parsed, err := p.parseExpression(0)
|
|
if err != nil {
|
|
return ASTNode{}, err
|
|
}
|
|
if p.current() != tEOF {
|
|
return ASTNode{}, p.syntaxError(fmt.Sprintf(
|
|
"Unexpected token at the end of the expresssion: %s", p.current()))
|
|
}
|
|
return parsed, nil
|
|
}
|
|
|
|
func (p *Parser) parseExpression(bindingPower int) (ASTNode, error) {
|
|
var err error
|
|
leftToken := p.lookaheadToken(0)
|
|
p.advance()
|
|
leftNode, err := p.nud(leftToken)
|
|
if err != nil {
|
|
return ASTNode{}, err
|
|
}
|
|
currentToken := p.current()
|
|
for bindingPower < bindingPowers[currentToken] {
|
|
p.advance()
|
|
leftNode, err = p.led(currentToken, leftNode)
|
|
if err != nil {
|
|
return ASTNode{}, err
|
|
}
|
|
currentToken = p.current()
|
|
}
|
|
return leftNode, nil
|
|
}
|
|
|
|
func (p *Parser) parseIndexExpression() (ASTNode, error) {
|
|
if p.lookahead(0) == tColon || p.lookahead(1) == tColon {
|
|
return p.parseSliceExpression()
|
|
}
|
|
indexStr := p.lookaheadToken(0).value
|
|
parsedInt, err := strconv.Atoi(indexStr)
|
|
if err != nil {
|
|
return ASTNode{}, err
|
|
}
|
|
indexNode := ASTNode{nodeType: ASTIndex, value: parsedInt}
|
|
p.advance()
|
|
if err := p.match(tRbracket); err != nil {
|
|
return ASTNode{}, err
|
|
}
|
|
return indexNode, nil
|
|
}
|
|
|
|
func (p *Parser) parseSliceExpression() (ASTNode, error) {
|
|
parts := []*int{nil, nil, nil}
|
|
index := 0
|
|
current := p.current()
|
|
for current != tRbracket && index < 3 {
|
|
if current == tColon {
|
|
index++
|
|
p.advance()
|
|
} else if current == tNumber {
|
|
parsedInt, err := strconv.Atoi(p.lookaheadToken(0).value)
|
|
if err != nil {
|
|
return ASTNode{}, err
|
|
}
|
|
parts[index] = &parsedInt
|
|
p.advance()
|
|
} else {
|
|
return ASTNode{}, p.syntaxError(
|
|
"Expected tColon or tNumber" + ", received: " + p.current().String())
|
|
}
|
|
current = p.current()
|
|
}
|
|
if err := p.match(tRbracket); err != nil {
|
|
return ASTNode{}, err
|
|
}
|
|
return ASTNode{
|
|
nodeType: ASTSlice,
|
|
value: parts,
|
|
}, nil
|
|
}
|
|
|
|
func (p *Parser) match(tokenType tokType) error {
|
|
if p.current() == tokenType {
|
|
p.advance()
|
|
return nil
|
|
}
|
|
return p.syntaxError("Expected " + tokenType.String() + ", received: " + p.current().String())
|
|
}
|
|
|
|
func (p *Parser) led(tokenType tokType, node ASTNode) (ASTNode, error) {
|
|
switch tokenType {
|
|
case tDot:
|
|
if p.current() != tStar {
|
|
right, err := p.parseDotRHS(bindingPowers[tDot])
|
|
return ASTNode{
|
|
nodeType: ASTSubexpression,
|
|
children: []ASTNode{node, right},
|
|
}, err
|
|
}
|
|
p.advance()
|
|
right, err := p.parseProjectionRHS(bindingPowers[tDot])
|
|
return ASTNode{
|
|
nodeType: ASTValueProjection,
|
|
children: []ASTNode{node, right},
|
|
}, err
|
|
case tPipe:
|
|
right, err := p.parseExpression(bindingPowers[tPipe])
|
|
return ASTNode{nodeType: ASTPipe, children: []ASTNode{node, right}}, err
|
|
case tOr:
|
|
right, err := p.parseExpression(bindingPowers[tOr])
|
|
return ASTNode{nodeType: ASTOrExpression, children: []ASTNode{node, right}}, err
|
|
case tAnd:
|
|
right, err := p.parseExpression(bindingPowers[tAnd])
|
|
return ASTNode{nodeType: ASTAndExpression, children: []ASTNode{node, right}}, err
|
|
case tLparen:
|
|
name := node.value
|
|
var args []ASTNode
|
|
for p.current() != tRparen {
|
|
expression, err := p.parseExpression(0)
|
|
if err != nil {
|
|
return ASTNode{}, err
|
|
}
|
|
if p.current() == tComma {
|
|
if err := p.match(tComma); err != nil {
|
|
return ASTNode{}, err
|
|
}
|
|
}
|
|
args = append(args, expression)
|
|
}
|
|
if err := p.match(tRparen); err != nil {
|
|
return ASTNode{}, err
|
|
}
|
|
return ASTNode{
|
|
nodeType: ASTFunctionExpression,
|
|
value: name,
|
|
children: args,
|
|
}, nil
|
|
case tFilter:
|
|
return p.parseFilter(node)
|
|
case tFlatten:
|
|
left := ASTNode{nodeType: ASTFlatten, children: []ASTNode{node}}
|
|
right, err := p.parseProjectionRHS(bindingPowers[tFlatten])
|
|
return ASTNode{
|
|
nodeType: ASTProjection,
|
|
children: []ASTNode{left, right},
|
|
}, err
|
|
case tEQ, tNE, tGT, tGTE, tLT, tLTE:
|
|
right, err := p.parseExpression(bindingPowers[tokenType])
|
|
if err != nil {
|
|
return ASTNode{}, err
|
|
}
|
|
return ASTNode{
|
|
nodeType: ASTComparator,
|
|
value: tokenType,
|
|
children: []ASTNode{node, right},
|
|
}, nil
|
|
case tLbracket:
|
|
tokenType := p.current()
|
|
var right ASTNode
|
|
var err error
|
|
if tokenType == tNumber || tokenType == tColon {
|
|
right, err = p.parseIndexExpression()
|
|
if err != nil {
|
|
return ASTNode{}, err
|
|
}
|
|
return p.projectIfSlice(node, right)
|
|
}
|
|
// Otherwise this is a projection.
|
|
if err := p.match(tStar); err != nil {
|
|
return ASTNode{}, err
|
|
}
|
|
if err := p.match(tRbracket); err != nil {
|
|
return ASTNode{}, err
|
|
}
|
|
right, err = p.parseProjectionRHS(bindingPowers[tStar])
|
|
if err != nil {
|
|
return ASTNode{}, err
|
|
}
|
|
return ASTNode{
|
|
nodeType: ASTProjection,
|
|
children: []ASTNode{node, right},
|
|
}, nil
|
|
}
|
|
return ASTNode{}, p.syntaxError("Unexpected token: " + tokenType.String())
|
|
}
|
|
|
|
func (p *Parser) nud(token token) (ASTNode, error) {
|
|
switch token.tokenType {
|
|
case tJSONLiteral:
|
|
var parsed interface{}
|
|
err := json.Unmarshal([]byte(token.value), &parsed)
|
|
if err != nil {
|
|
return ASTNode{}, err
|
|
}
|
|
return ASTNode{nodeType: ASTLiteral, value: parsed}, nil
|
|
case tStringLiteral:
|
|
return ASTNode{nodeType: ASTLiteral, value: token.value}, nil
|
|
case tUnquotedIdentifier:
|
|
return ASTNode{
|
|
nodeType: ASTField,
|
|
value: token.value,
|
|
}, nil
|
|
case tQuotedIdentifier:
|
|
node := ASTNode{nodeType: ASTField, value: token.value}
|
|
if p.current() == tLparen {
|
|
return ASTNode{}, p.syntaxErrorToken("Can't have quoted identifier as function name.", token)
|
|
}
|
|
return node, nil
|
|
case tStar:
|
|
left := ASTNode{nodeType: ASTIdentity}
|
|
var right ASTNode
|
|
var err error
|
|
if p.current() == tRbracket {
|
|
right = ASTNode{nodeType: ASTIdentity}
|
|
} else {
|
|
right, err = p.parseProjectionRHS(bindingPowers[tStar])
|
|
}
|
|
return ASTNode{nodeType: ASTValueProjection, children: []ASTNode{left, right}}, err
|
|
case tFilter:
|
|
return p.parseFilter(ASTNode{nodeType: ASTIdentity})
|
|
case tLbrace:
|
|
return p.parseMultiSelectHash()
|
|
case tFlatten:
|
|
left := ASTNode{
|
|
nodeType: ASTFlatten,
|
|
children: []ASTNode{{nodeType: ASTIdentity}},
|
|
}
|
|
right, err := p.parseProjectionRHS(bindingPowers[tFlatten])
|
|
if err != nil {
|
|
return ASTNode{}, err
|
|
}
|
|
return ASTNode{nodeType: ASTProjection, children: []ASTNode{left, right}}, nil
|
|
case tLbracket:
|
|
tokenType := p.current()
|
|
//var right ASTNode
|
|
if tokenType == tNumber || tokenType == tColon {
|
|
right, err := p.parseIndexExpression()
|
|
if err != nil {
|
|
return ASTNode{}, nil
|
|
}
|
|
return p.projectIfSlice(ASTNode{nodeType: ASTIdentity}, right)
|
|
} else if tokenType == tStar && p.lookahead(1) == tRbracket {
|
|
p.advance()
|
|
p.advance()
|
|
right, err := p.parseProjectionRHS(bindingPowers[tStar])
|
|
if err != nil {
|
|
return ASTNode{}, err
|
|
}
|
|
return ASTNode{
|
|
nodeType: ASTProjection,
|
|
children: []ASTNode{{nodeType: ASTIdentity}, right},
|
|
}, nil
|
|
} else {
|
|
return p.parseMultiSelectList()
|
|
}
|
|
case tCurrent:
|
|
return ASTNode{nodeType: ASTCurrentNode}, nil
|
|
case tExpref:
|
|
expression, err := p.parseExpression(bindingPowers[tExpref])
|
|
if err != nil {
|
|
return ASTNode{}, err
|
|
}
|
|
return ASTNode{nodeType: ASTExpRef, children: []ASTNode{expression}}, nil
|
|
case tNot:
|
|
expression, err := p.parseExpression(bindingPowers[tNot])
|
|
if err != nil {
|
|
return ASTNode{}, err
|
|
}
|
|
return ASTNode{nodeType: ASTNotExpression, children: []ASTNode{expression}}, nil
|
|
case tLparen:
|
|
expression, err := p.parseExpression(0)
|
|
if err != nil {
|
|
return ASTNode{}, err
|
|
}
|
|
if err := p.match(tRparen); err != nil {
|
|
return ASTNode{}, err
|
|
}
|
|
return expression, nil
|
|
case tEOF:
|
|
return ASTNode{}, p.syntaxErrorToken("Incomplete expression", token)
|
|
}
|
|
|
|
return ASTNode{}, p.syntaxErrorToken("Invalid token: "+token.tokenType.String(), token)
|
|
}
|
|
|
|
func (p *Parser) parseMultiSelectList() (ASTNode, error) {
|
|
var expressions []ASTNode
|
|
for {
|
|
expression, err := p.parseExpression(0)
|
|
if err != nil {
|
|
return ASTNode{}, err
|
|
}
|
|
expressions = append(expressions, expression)
|
|
if p.current() == tRbracket {
|
|
break
|
|
}
|
|
err = p.match(tComma)
|
|
if err != nil {
|
|
return ASTNode{}, err
|
|
}
|
|
}
|
|
err := p.match(tRbracket)
|
|
if err != nil {
|
|
return ASTNode{}, err
|
|
}
|
|
return ASTNode{
|
|
nodeType: ASTMultiSelectList,
|
|
children: expressions,
|
|
}, nil
|
|
}
|
|
|
|
func (p *Parser) parseMultiSelectHash() (ASTNode, error) {
|
|
var children []ASTNode
|
|
for {
|
|
keyToken := p.lookaheadToken(0)
|
|
if err := p.match(tUnquotedIdentifier); err != nil {
|
|
if err := p.match(tQuotedIdentifier); err != nil {
|
|
return ASTNode{}, p.syntaxError("Expected tQuotedIdentifier or tUnquotedIdentifier")
|
|
}
|
|
}
|
|
keyName := keyToken.value
|
|
err := p.match(tColon)
|
|
if err != nil {
|
|
return ASTNode{}, err
|
|
}
|
|
value, err := p.parseExpression(0)
|
|
if err != nil {
|
|
return ASTNode{}, err
|
|
}
|
|
node := ASTNode{
|
|
nodeType: ASTKeyValPair,
|
|
value: keyName,
|
|
children: []ASTNode{value},
|
|
}
|
|
children = append(children, node)
|
|
if p.current() == tComma {
|
|
err := p.match(tComma)
|
|
if err != nil {
|
|
return ASTNode{}, nil
|
|
}
|
|
} else if p.current() == tRbrace {
|
|
err := p.match(tRbrace)
|
|
if err != nil {
|
|
return ASTNode{}, nil
|
|
}
|
|
break
|
|
}
|
|
}
|
|
return ASTNode{
|
|
nodeType: ASTMultiSelectHash,
|
|
children: children,
|
|
}, nil
|
|
}
|
|
|
|
func (p *Parser) projectIfSlice(left ASTNode, right ASTNode) (ASTNode, error) {
|
|
indexExpr := ASTNode{
|
|
nodeType: ASTIndexExpression,
|
|
children: []ASTNode{left, right},
|
|
}
|
|
if right.nodeType == ASTSlice {
|
|
right, err := p.parseProjectionRHS(bindingPowers[tStar])
|
|
return ASTNode{
|
|
nodeType: ASTProjection,
|
|
children: []ASTNode{indexExpr, right},
|
|
}, err
|
|
}
|
|
return indexExpr, nil
|
|
}
|
|
func (p *Parser) parseFilter(node ASTNode) (ASTNode, error) {
|
|
var right, condition ASTNode
|
|
var err error
|
|
condition, err = p.parseExpression(0)
|
|
if err != nil {
|
|
return ASTNode{}, err
|
|
}
|
|
if err := p.match(tRbracket); err != nil {
|
|
return ASTNode{}, err
|
|
}
|
|
if p.current() == tFlatten {
|
|
right = ASTNode{nodeType: ASTIdentity}
|
|
} else {
|
|
right, err = p.parseProjectionRHS(bindingPowers[tFilter])
|
|
if err != nil {
|
|
return ASTNode{}, err
|
|
}
|
|
}
|
|
|
|
return ASTNode{
|
|
nodeType: ASTFilterProjection,
|
|
children: []ASTNode{node, right, condition},
|
|
}, nil
|
|
}
|
|
|
|
func (p *Parser) parseDotRHS(bindingPower int) (ASTNode, error) {
|
|
lookahead := p.current()
|
|
if tokensOneOf([]tokType{tQuotedIdentifier, tUnquotedIdentifier, tStar}, lookahead) {
|
|
return p.parseExpression(bindingPower)
|
|
} else if lookahead == tLbracket {
|
|
if err := p.match(tLbracket); err != nil {
|
|
return ASTNode{}, err
|
|
}
|
|
return p.parseMultiSelectList()
|
|
} else if lookahead == tLbrace {
|
|
if err := p.match(tLbrace); err != nil {
|
|
return ASTNode{}, err
|
|
}
|
|
return p.parseMultiSelectHash()
|
|
}
|
|
return ASTNode{}, p.syntaxError("Expected identifier, lbracket, or lbrace")
|
|
}
|
|
|
|
func (p *Parser) parseProjectionRHS(bindingPower int) (ASTNode, error) {
|
|
current := p.current()
|
|
if bindingPowers[current] < 10 {
|
|
return ASTNode{nodeType: ASTIdentity}, nil
|
|
} else if current == tLbracket {
|
|
return p.parseExpression(bindingPower)
|
|
} else if current == tFilter {
|
|
return p.parseExpression(bindingPower)
|
|
} else if current == tDot {
|
|
err := p.match(tDot)
|
|
if err != nil {
|
|
return ASTNode{}, err
|
|
}
|
|
return p.parseDotRHS(bindingPower)
|
|
} else {
|
|
return ASTNode{}, p.syntaxError("Error")
|
|
}
|
|
}
|
|
|
|
func (p *Parser) lookahead(number int) tokType {
|
|
return p.lookaheadToken(number).tokenType
|
|
}
|
|
|
|
func (p *Parser) current() tokType {
|
|
return p.lookahead(0)
|
|
}
|
|
|
|
func (p *Parser) lookaheadToken(number int) token {
|
|
return p.tokens[p.index+number]
|
|
}
|
|
|
|
func (p *Parser) advance() {
|
|
p.index++
|
|
}
|
|
|
|
func tokensOneOf(elements []tokType, token tokType) bool {
|
|
for _, elem := range elements {
|
|
if elem == token {
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
func (p *Parser) syntaxError(msg string) SyntaxError {
|
|
return SyntaxError{
|
|
msg: msg,
|
|
Expression: p.expression,
|
|
Offset: p.lookaheadToken(0).position,
|
|
}
|
|
}
|
|
|
|
// Create a SyntaxError based on the provided token.
|
|
// This differs from syntaxError() which creates a SyntaxError
|
|
// based on the current lookahead token.
|
|
func (p *Parser) syntaxErrorToken(msg string, t token) SyntaxError {
|
|
return SyntaxError{
|
|
msg: msg,
|
|
Expression: p.expression,
|
|
Offset: t.position,
|
|
}
|
|
}
|