2020-10-08 23:59:03 +00:00
|
|
|
package treeops
|
2020-09-20 12:40:09 +00:00
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
|
|
|
)
|
|
|
|
|
|
|
|
type PathPostFixer interface {
|
2020-10-11 00:24:22 +00:00
|
|
|
ConvertToPostfix([]*Token) ([]*PathElement, error)
|
2020-09-20 12:40:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type pathPostFixer struct {
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewPathPostFixer() PathPostFixer {
|
|
|
|
return &pathPostFixer{}
|
|
|
|
}
|
|
|
|
|
2020-10-11 00:24:22 +00:00
|
|
|
func popOpToResult(opStack []*Token, result []*PathElement) ([]*Token, []*PathElement) {
|
|
|
|
var operatorToPushToPostFix *Token
|
2020-09-20 12:40:09 +00:00
|
|
|
opStack, operatorToPushToPostFix = opStack[0:len(opStack)-1], opStack[len(opStack)-1]
|
2020-10-11 23:44:33 +00:00
|
|
|
var pathElement = PathElement{PathElementType: Operation, OperationType: operatorToPushToPostFix.OperationType}
|
2020-09-20 12:40:09 +00:00
|
|
|
return opStack, append(result, &pathElement)
|
|
|
|
}
|
|
|
|
|
2020-10-11 00:24:22 +00:00
|
|
|
func (p *pathPostFixer) ConvertToPostfix(infixTokens []*Token) ([]*PathElement, error) {
|
2020-09-20 12:40:09 +00:00
|
|
|
var result []*PathElement
|
|
|
|
// surround the whole thing with quotes
|
2020-10-16 01:29:26 +00:00
|
|
|
var opStack = []*Token{&Token{PathElementType: OpenBracket, OperationType: None, Value: "("}}
|
|
|
|
var tokens = append(infixTokens, &Token{PathElementType: CloseBracket, OperationType: None, Value: ")"})
|
2020-09-20 12:40:09 +00:00
|
|
|
|
|
|
|
for _, token := range tokens {
|
2020-10-16 01:29:26 +00:00
|
|
|
log.Debugf("postfix processing token %v", token.Value)
|
2020-10-11 00:24:22 +00:00
|
|
|
switch token.PathElementType {
|
2020-10-16 01:29:26 +00:00
|
|
|
case PathKey, SelfReference, Value:
|
2020-10-11 00:24:22 +00:00
|
|
|
var pathElement = PathElement{PathElementType: token.PathElementType, Value: token.Value, StringValue: token.StringValue}
|
2020-09-24 03:20:02 +00:00
|
|
|
result = append(result, &pathElement)
|
2020-10-16 01:29:26 +00:00
|
|
|
case OpenBracket, OpenCollect:
|
2020-09-20 12:40:09 +00:00
|
|
|
opStack = append(opStack, token)
|
2020-10-16 01:29:26 +00:00
|
|
|
case CloseCollect:
|
|
|
|
for len(opStack) > 0 && opStack[len(opStack)-1].PathElementType != OpenCollect {
|
|
|
|
opStack, result = popOpToResult(opStack, result)
|
|
|
|
}
|
|
|
|
if len(opStack) == 0 {
|
|
|
|
return nil, errors.New("Bad path expression, got close collect brackets without matching opening bracket")
|
|
|
|
}
|
|
|
|
// now we should have ( as the last element on the opStack, get rid of it
|
|
|
|
opStack = opStack[0 : len(opStack)-1]
|
|
|
|
//and append a collect to the opStack
|
|
|
|
opStack = append(opStack, &Token{PathElementType: Operation, OperationType: Collect})
|
2020-10-11 00:24:22 +00:00
|
|
|
case CloseBracket:
|
|
|
|
for len(opStack) > 0 && opStack[len(opStack)-1].PathElementType != OpenBracket {
|
2020-09-20 12:40:09 +00:00
|
|
|
opStack, result = popOpToResult(opStack, result)
|
|
|
|
}
|
|
|
|
if len(opStack) == 0 {
|
|
|
|
return nil, errors.New("Bad path expression, got close brackets without matching opening bracket")
|
|
|
|
}
|
|
|
|
// now we should have ( as the last element on the opStack, get rid of it
|
|
|
|
opStack = opStack[0 : len(opStack)-1]
|
2020-10-11 00:24:22 +00:00
|
|
|
|
2020-10-10 04:24:37 +00:00
|
|
|
default:
|
2020-10-11 23:44:33 +00:00
|
|
|
var currentPrecedence = token.OperationType.Precedence
|
2020-10-10 04:24:37 +00:00
|
|
|
// pop off higher precedent operators onto the result
|
2020-10-11 23:44:33 +00:00
|
|
|
for len(opStack) > 0 && opStack[len(opStack)-1].OperationType.Precedence >= currentPrecedence {
|
2020-10-10 04:24:37 +00:00
|
|
|
opStack, result = popOpToResult(opStack, result)
|
|
|
|
}
|
|
|
|
// add this operator to the opStack
|
|
|
|
opStack = append(opStack, token)
|
2020-09-20 12:40:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return result, nil
|
|
|
|
}
|