mirror of
https://github.com/prometheus/prometheus.git
synced 2025-03-05 20:59:13 -08:00
Merge pull request #660 from prometheus/fabxc/pql/parse-errs
Fix and improve parsing error output.
This commit is contained in:
commit
6649306e63
|
@ -15,7 +15,6 @@ package promql
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"reflect"
|
|
||||||
"strings"
|
"strings"
|
||||||
"unicode"
|
"unicode"
|
||||||
"unicode/utf8"
|
"unicode/utf8"
|
||||||
|
@ -35,6 +34,8 @@ func (i item) String() string {
|
||||||
return "EOF"
|
return "EOF"
|
||||||
case i.typ == itemError:
|
case i.typ == itemError:
|
||||||
return i.val
|
return i.val
|
||||||
|
case i.typ == itemIdentifier || i.typ == itemMetricIdentifier:
|
||||||
|
return fmt.Sprintf("%q", i.val)
|
||||||
case i.typ.isKeyword():
|
case i.typ.isKeyword():
|
||||||
return fmt.Sprintf("<%s>", i.val)
|
return fmt.Sprintf("<%s>", i.val)
|
||||||
case i.typ.isOperator():
|
case i.typ.isOperator():
|
||||||
|
@ -183,6 +184,16 @@ var key = map[string]itemType{
|
||||||
// These are the default string representations for common items. It does not
|
// These are the default string representations for common items. It does not
|
||||||
// imply that those are the only character sequences that can be lexed to such an item.
|
// imply that those are the only character sequences that can be lexed to such an item.
|
||||||
var itemTypeStr = map[itemType]string{
|
var itemTypeStr = map[itemType]string{
|
||||||
|
itemLeftParen: "(",
|
||||||
|
itemRightParen: ")",
|
||||||
|
itemLeftBrace: "{",
|
||||||
|
itemRightBrace: "}",
|
||||||
|
itemLeftBracket: "[",
|
||||||
|
itemRightBracket: "]",
|
||||||
|
itemComma: ",",
|
||||||
|
itemAssign: "=",
|
||||||
|
itemSemicolon: ";",
|
||||||
|
|
||||||
itemSUB: "-",
|
itemSUB: "-",
|
||||||
itemADD: "+",
|
itemADD: "+",
|
||||||
itemMUL: "*",
|
itemMUL: "*",
|
||||||
|
@ -209,7 +220,39 @@ func (t itemType) String() string {
|
||||||
if s, ok := itemTypeStr[t]; ok {
|
if s, ok := itemTypeStr[t]; ok {
|
||||||
return s
|
return s
|
||||||
}
|
}
|
||||||
return reflect.TypeOf(t).Name()
|
return fmt.Sprintf("<item %d>", t)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (i item) desc() string {
|
||||||
|
if _, ok := itemTypeStr[i.typ]; ok {
|
||||||
|
return i.String()
|
||||||
|
}
|
||||||
|
if i.typ == itemEOF {
|
||||||
|
return i.typ.desc()
|
||||||
|
}
|
||||||
|
return fmt.Sprintf("%s %s", i.typ.desc(), i)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t itemType) desc() string {
|
||||||
|
switch t {
|
||||||
|
case itemError:
|
||||||
|
return "error"
|
||||||
|
case itemEOF:
|
||||||
|
return "end of input"
|
||||||
|
case itemComment:
|
||||||
|
return "comment"
|
||||||
|
case itemIdentifier:
|
||||||
|
return "identifier"
|
||||||
|
case itemMetricIdentifier:
|
||||||
|
return "metric identifier"
|
||||||
|
case itemString:
|
||||||
|
return "string"
|
||||||
|
case itemNumber:
|
||||||
|
return "number"
|
||||||
|
case itemDuration:
|
||||||
|
return "duration"
|
||||||
|
}
|
||||||
|
return fmt.Sprintf("%q", t)
|
||||||
}
|
}
|
||||||
|
|
||||||
const eof = -1
|
const eof = -1
|
||||||
|
@ -377,7 +420,7 @@ func lexStatements(l *lexer) stateFn {
|
||||||
l.next()
|
l.next()
|
||||||
l.emit(itemEQL)
|
l.emit(itemEQL)
|
||||||
} else if t == '~' {
|
} else if t == '~' {
|
||||||
return l.errorf("unrecognized character after '=': %#U", t)
|
return l.errorf("unexpected character after '=': %q", t)
|
||||||
} else {
|
} else {
|
||||||
l.emit(itemAssign)
|
l.emit(itemAssign)
|
||||||
}
|
}
|
||||||
|
@ -385,7 +428,7 @@ func lexStatements(l *lexer) stateFn {
|
||||||
if t := l.next(); t == '=' {
|
if t := l.next(); t == '=' {
|
||||||
l.emit(itemNEQ)
|
l.emit(itemNEQ)
|
||||||
} else {
|
} else {
|
||||||
return l.errorf("unrecognized character after '!': %#U", t)
|
return l.errorf("unexpected character after '!': %q", t)
|
||||||
}
|
}
|
||||||
case r == '<':
|
case r == '<':
|
||||||
if t := l.peek(); t == '=' {
|
if t := l.peek(); t == '=' {
|
||||||
|
@ -401,7 +444,7 @@ func lexStatements(l *lexer) stateFn {
|
||||||
} else {
|
} else {
|
||||||
l.emit(itemGTR)
|
l.emit(itemGTR)
|
||||||
}
|
}
|
||||||
case '0' <= r && r <= '9' || r == '.':
|
case unicode.IsDigit(r) || (r == '.' && unicode.IsDigit(l.peek())):
|
||||||
l.backup()
|
l.backup()
|
||||||
return lexNumberOrDuration
|
return lexNumberOrDuration
|
||||||
case r == '"' || r == '\'':
|
case r == '"' || r == '\'':
|
||||||
|
@ -422,7 +465,7 @@ func lexStatements(l *lexer) stateFn {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
fallthrough
|
fallthrough
|
||||||
case isAlphaNumeric(r):
|
case isAlphaNumeric(r) || r == ':':
|
||||||
l.backup()
|
l.backup()
|
||||||
return lexKeywordOrIdentifier
|
return lexKeywordOrIdentifier
|
||||||
case r == '(':
|
case r == '(':
|
||||||
|
@ -433,7 +476,7 @@ func lexStatements(l *lexer) stateFn {
|
||||||
l.emit(itemRightParen)
|
l.emit(itemRightParen)
|
||||||
l.parenDepth--
|
l.parenDepth--
|
||||||
if l.parenDepth < 0 {
|
if l.parenDepth < 0 {
|
||||||
return l.errorf("unexpected right parenthesis %#U", r)
|
return l.errorf("unexpected right parenthesis %q", r)
|
||||||
}
|
}
|
||||||
return lexStatements
|
return lexStatements
|
||||||
case r == '{':
|
case r == '{':
|
||||||
|
@ -442,20 +485,20 @@ func lexStatements(l *lexer) stateFn {
|
||||||
return lexInsideBraces(l)
|
return lexInsideBraces(l)
|
||||||
case r == '[':
|
case r == '[':
|
||||||
if l.bracketOpen {
|
if l.bracketOpen {
|
||||||
return l.errorf("unexpected left bracket %#U", r)
|
return l.errorf("unexpected left bracket %q", r)
|
||||||
}
|
}
|
||||||
l.emit(itemLeftBracket)
|
l.emit(itemLeftBracket)
|
||||||
l.bracketOpen = true
|
l.bracketOpen = true
|
||||||
return lexDuration
|
return lexDuration
|
||||||
case r == ']':
|
case r == ']':
|
||||||
if !l.bracketOpen {
|
if !l.bracketOpen {
|
||||||
return l.errorf("unexpected right bracket %#U", r)
|
return l.errorf("unexpected right bracket %q", r)
|
||||||
}
|
}
|
||||||
l.emit(itemRightBracket)
|
l.emit(itemRightBracket)
|
||||||
l.bracketOpen = false
|
l.bracketOpen = false
|
||||||
|
|
||||||
default:
|
default:
|
||||||
return l.errorf("unrecognized character in statement: %#U", r)
|
return l.errorf("unexpected character: %q", r)
|
||||||
}
|
}
|
||||||
return lexStatements
|
return lexStatements
|
||||||
}
|
}
|
||||||
|
@ -469,10 +512,10 @@ func lexInsideBraces(l *lexer) stateFn {
|
||||||
|
|
||||||
switch r := l.next(); {
|
switch r := l.next(); {
|
||||||
case r == eof:
|
case r == eof:
|
||||||
return l.errorf("unexpected EOF inside braces")
|
return l.errorf("unexpected end of input inside braces")
|
||||||
case isSpace(r):
|
case isSpace(r):
|
||||||
return lexSpace
|
return lexSpace
|
||||||
case isAlphaNumeric(r):
|
case unicode.IsLetter(r) || r == '_':
|
||||||
l.backup()
|
l.backup()
|
||||||
return lexIdentifier
|
return lexIdentifier
|
||||||
case r == ',':
|
case r == ',':
|
||||||
|
@ -494,16 +537,16 @@ func lexInsideBraces(l *lexer) stateFn {
|
||||||
case nr == '=':
|
case nr == '=':
|
||||||
l.emit(itemNEQ)
|
l.emit(itemNEQ)
|
||||||
default:
|
default:
|
||||||
return l.errorf("unrecognized character after '!' inside braces: %#U", nr)
|
return l.errorf("unexpected character after '!' inside braces: %q", nr)
|
||||||
}
|
}
|
||||||
case r == '{':
|
case r == '{':
|
||||||
return l.errorf("unexpected left brace %#U", r)
|
return l.errorf("unexpected left brace %q", r)
|
||||||
case r == '}':
|
case r == '}':
|
||||||
l.emit(itemRightBrace)
|
l.emit(itemRightBrace)
|
||||||
l.braceOpen = false
|
l.braceOpen = false
|
||||||
return lexStatements
|
return lexStatements
|
||||||
default:
|
default:
|
||||||
return l.errorf("unrecognized character inside braces: %#U", r)
|
return l.errorf("unexpected character inside braces: %q", r)
|
||||||
}
|
}
|
||||||
return lexInsideBraces
|
return lexInsideBraces
|
||||||
}
|
}
|
||||||
|
@ -553,7 +596,11 @@ func lexDuration(l *lexer) stateFn {
|
||||||
return l.errorf("missing unit character in duration")
|
return l.errorf("missing unit character in duration")
|
||||||
}
|
}
|
||||||
// Next two chars must be a valid unit and a non-alphanumeric.
|
// Next two chars must be a valid unit and a non-alphanumeric.
|
||||||
if l.accept("smhdwy") && !isAlphaNumeric(l.peek()) {
|
if l.accept("smhdwy") {
|
||||||
|
if isAlphaNumeric(l.next()) {
|
||||||
|
return l.errorf("bad duration syntax: %q", l.input[l.start:l.pos])
|
||||||
|
}
|
||||||
|
l.backup()
|
||||||
l.emit(itemDuration)
|
l.emit(itemDuration)
|
||||||
return lexStatements
|
return lexStatements
|
||||||
}
|
}
|
||||||
|
@ -576,7 +623,11 @@ func lexNumberOrDuration(l *lexer) stateFn {
|
||||||
return lexStatements
|
return lexStatements
|
||||||
}
|
}
|
||||||
// Next two chars must be a valid unit and a non-alphanumeric.
|
// Next two chars must be a valid unit and a non-alphanumeric.
|
||||||
if l.accept("smhdwy") && !isAlphaNumeric(l.peek()) {
|
if l.accept("smhdwy") {
|
||||||
|
if isAlphaNumeric(l.next()) {
|
||||||
|
return l.errorf("bad number or duration syntax: %q", l.input[l.start:l.pos])
|
||||||
|
}
|
||||||
|
l.backup()
|
||||||
l.emit(itemDuration)
|
l.emit(itemDuration)
|
||||||
return lexStatements
|
return lexStatements
|
||||||
}
|
}
|
||||||
|
@ -605,7 +656,8 @@ func (l *lexer) scanNumber() bool {
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
// lexIdentifier scans an alphanumeric identifier.
|
// lexIdentifier scans an alphanumeric identifier. The next character
|
||||||
|
// is known to be a letter.
|
||||||
func lexIdentifier(l *lexer) stateFn {
|
func lexIdentifier(l *lexer) stateFn {
|
||||||
for isAlphaNumeric(l.next()) {
|
for isAlphaNumeric(l.next()) {
|
||||||
// absorb
|
// absorb
|
||||||
|
@ -651,5 +703,5 @@ func isEndOfLine(r rune) bool {
|
||||||
|
|
||||||
// isAlphaNumeric reports whether r is an alphabetic, digit, or underscore.
|
// isAlphaNumeric reports whether r is an alphabetic, digit, or underscore.
|
||||||
func isAlphaNumeric(r rune) bool {
|
func isAlphaNumeric(r rune) bool {
|
||||||
return r == '_' || unicode.IsLetter(r) || unicode.IsDigit(r)
|
return r == '_' || ('a' <= r && r <= 'z') || ('A' <= r && r <= 'Z') || unicode.IsDigit(r)
|
||||||
}
|
}
|
||||||
|
|
|
@ -245,6 +245,9 @@ var tests = []struct {
|
||||||
},
|
},
|
||||||
// Test Selector.
|
// Test Selector.
|
||||||
{
|
{
|
||||||
|
input: `台北`,
|
||||||
|
fail: true,
|
||||||
|
}, {
|
||||||
input: `{foo="bar"}`,
|
input: `{foo="bar"}`,
|
||||||
expected: []item{
|
expected: []item{
|
||||||
{itemLeftBrace, 0, `{`},
|
{itemLeftBrace, 0, `{`},
|
||||||
|
|
|
@ -101,7 +101,7 @@ func (p *parser) parseExpr() (expr Expr, err error) {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
if expr != nil {
|
if expr != nil {
|
||||||
p.errorf("expression read but input remaining")
|
p.errorf("could not parse remaining input %.15q...", p.lex.input[p.lex.lastPos:])
|
||||||
}
|
}
|
||||||
expr = p.expr()
|
expr = p.expr()
|
||||||
}
|
}
|
||||||
|
@ -132,6 +132,9 @@ func (p *parser) next() item {
|
||||||
}
|
}
|
||||||
p.token[0] = t
|
p.token[0] = t
|
||||||
}
|
}
|
||||||
|
if p.token[p.peekCount].typ == itemError {
|
||||||
|
p.errorf("%s", p.token[p.peekCount].val)
|
||||||
|
}
|
||||||
return p.token[p.peekCount]
|
return p.token[p.peekCount]
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -175,28 +178,23 @@ func (p *parser) error(err error) {
|
||||||
}
|
}
|
||||||
|
|
||||||
// expect consumes the next token and guarantees it has the required type.
|
// expect consumes the next token and guarantees it has the required type.
|
||||||
func (p *parser) expect(expected itemType, context string) item {
|
func (p *parser) expect(exp itemType, context string) item {
|
||||||
token := p.next()
|
token := p.next()
|
||||||
if token.typ != expected {
|
if token.typ != exp {
|
||||||
p.unexpected(token, context)
|
p.errorf("unexpected %s in %s, expected %s", token.desc(), context, exp.desc())
|
||||||
}
|
}
|
||||||
return token
|
return token
|
||||||
}
|
}
|
||||||
|
|
||||||
// expectOneOf consumes the next token and guarantees it has one of the required types.
|
// expectOneOf consumes the next token and guarantees it has one of the required types.
|
||||||
func (p *parser) expectOneOf(expected1, expected2 itemType, context string) item {
|
func (p *parser) expectOneOf(exp1, exp2 itemType, context string) item {
|
||||||
token := p.next()
|
token := p.next()
|
||||||
if token.typ != expected1 && token.typ != expected2 {
|
if token.typ != exp1 && token.typ != exp2 {
|
||||||
p.unexpected(token, context)
|
p.errorf("unexpected %s in %s, expected %s or %s", token.desc(), context, exp1.desc(), exp2.desc())
|
||||||
}
|
}
|
||||||
return token
|
return token
|
||||||
}
|
}
|
||||||
|
|
||||||
// unexpected complains about the token and terminates processing.
|
|
||||||
func (p *parser) unexpected(token item, context string) {
|
|
||||||
p.errorf("unexpected %s in %s", token, context)
|
|
||||||
}
|
|
||||||
|
|
||||||
// recover is the handler that turns panics into returns from the top level of Parse.
|
// recover is the handler that turns panics into returns from the top level of Parse.
|
||||||
func (p *parser) recover(errp *error) {
|
func (p *parser) recover(errp *error) {
|
||||||
e := recover()
|
e := recover()
|
||||||
|
@ -303,10 +301,11 @@ func (p *parser) recordStmt() *RecordStmt {
|
||||||
|
|
||||||
// expr parses any expression.
|
// expr parses any expression.
|
||||||
func (p *parser) expr() Expr {
|
func (p *parser) expr() Expr {
|
||||||
const ctx = "binary expression"
|
|
||||||
|
|
||||||
// Parse the starting expression.
|
// Parse the starting expression.
|
||||||
expr := p.unaryExpr()
|
expr := p.unaryExpr()
|
||||||
|
if expr == nil {
|
||||||
|
p.errorf("no valid expression found")
|
||||||
|
}
|
||||||
|
|
||||||
// Loop through the operations and construct a binary operation tree based
|
// Loop through the operations and construct a binary operation tree based
|
||||||
// on the operators' precedence.
|
// on the operators' precedence.
|
||||||
|
@ -354,6 +353,9 @@ func (p *parser) expr() Expr {
|
||||||
|
|
||||||
// Parse the next operand.
|
// Parse the next operand.
|
||||||
rhs := p.unaryExpr()
|
rhs := p.unaryExpr()
|
||||||
|
if rhs == nil {
|
||||||
|
p.errorf("missing right-hand side in binary expression")
|
||||||
|
}
|
||||||
|
|
||||||
// Assign the new root based on the precendence of the LHS and RHS operators.
|
// Assign the new root based on the precendence of the LHS and RHS operators.
|
||||||
if lhs, ok := expr.(*BinaryExpr); ok && lhs.Op.precedence() < op.precedence() {
|
if lhs, ok := expr.(*BinaryExpr); ok && lhs.Op.precedence() < op.precedence() {
|
||||||
|
@ -497,7 +499,6 @@ func (p *parser) primaryExpr() Expr {
|
||||||
p.backup()
|
p.backup()
|
||||||
return p.aggrExpr()
|
return p.aggrExpr()
|
||||||
}
|
}
|
||||||
p.errorf("invalid primary expression")
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -535,7 +536,7 @@ func (p *parser) aggrExpr() *AggregateExpr {
|
||||||
|
|
||||||
agop := p.next()
|
agop := p.next()
|
||||||
if !agop.typ.isAggregator() {
|
if !agop.typ.isAggregator() {
|
||||||
p.errorf("%s is not an aggregation operator", agop)
|
p.errorf("expected aggregation operator but got %s", agop)
|
||||||
}
|
}
|
||||||
var grouping clientmodel.LabelNames
|
var grouping clientmodel.LabelNames
|
||||||
var keepExtra bool
|
var keepExtra bool
|
||||||
|
@ -650,7 +651,7 @@ func (p *parser) labelMatchers(operators ...itemType) metric.LabelMatchers {
|
||||||
|
|
||||||
op := p.next().typ
|
op := p.next().typ
|
||||||
if !op.isOperator() {
|
if !op.isOperator() {
|
||||||
p.errorf("item %s is not a valid operator for label matching", op)
|
p.errorf("expected label matching operator but got %s", op)
|
||||||
}
|
}
|
||||||
var validOp = false
|
var validOp = false
|
||||||
for _, allowedOp := range operators {
|
for _, allowedOp := range operators {
|
||||||
|
@ -849,10 +850,10 @@ func (p *parser) checkType(node Node) (typ ExprType) {
|
||||||
case *Call:
|
case *Call:
|
||||||
nargs := len(n.Func.ArgTypes)
|
nargs := len(n.Func.ArgTypes)
|
||||||
if na := nargs - n.Func.OptionalArgs; na > len(n.Args) {
|
if na := nargs - n.Func.OptionalArgs; na > len(n.Args) {
|
||||||
p.errorf("expected at least %d arguments in call to %q, got %d", na, n.Func.Name, len(n.Args))
|
p.errorf("expected at least %d argument(s) in call to %q, got %d", na, n.Func.Name, len(n.Args))
|
||||||
}
|
}
|
||||||
if nargs < len(n.Args) {
|
if nargs < len(n.Args) {
|
||||||
p.errorf("expected at most %d arguments in call to %q, got %d", nargs, n.Func.Name, len(n.Args))
|
p.errorf("expected at most %d argument(s) in call to %q, got %d", nargs, n.Func.Name, len(n.Args))
|
||||||
}
|
}
|
||||||
for i, arg := range n.Args {
|
for i, arg := range n.Args {
|
||||||
p.expectType(arg, n.Func.ArgTypes[i], fmt.Sprintf("call to function %q", n.Func.Name))
|
p.expectType(arg, n.Func.ArgTypes[i], fmt.Sprintf("call to function %q", n.Func.Name))
|
||||||
|
|
|
@ -17,6 +17,7 @@ import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"math"
|
"math"
|
||||||
"reflect"
|
"reflect"
|
||||||
|
"strings"
|
||||||
"testing"
|
"testing"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
@ -25,9 +26,10 @@ import (
|
||||||
)
|
)
|
||||||
|
|
||||||
var testExpr = []struct {
|
var testExpr = []struct {
|
||||||
input string
|
input string // The input to be parsed.
|
||||||
expected Expr
|
expected Expr // The expected expression AST.
|
||||||
fail bool
|
fail bool // Whether parsing is supposed to fail.
|
||||||
|
errMsg string // If not empty the parsing error has to contain this string.
|
||||||
}{
|
}{
|
||||||
// Scalars and scalar-to-scalar operations.
|
// Scalars and scalar-to-scalar operations.
|
||||||
{
|
{
|
||||||
|
@ -122,39 +124,77 @@ var testExpr = []struct {
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
}, {
|
}, {
|
||||||
input: "", fail: true,
|
input: "",
|
||||||
|
fail: true,
|
||||||
|
errMsg: "no expression found in input",
|
||||||
}, {
|
}, {
|
||||||
input: "# just a comment\n\n", fail: true,
|
input: "# just a comment\n\n",
|
||||||
|
fail: true,
|
||||||
|
errMsg: "no expression found in input",
|
||||||
}, {
|
}, {
|
||||||
input: "1+", fail: true,
|
input: "1+",
|
||||||
|
fail: true,
|
||||||
|
errMsg: "missing right-hand side in binary expression",
|
||||||
}, {
|
}, {
|
||||||
input: "2.5.", fail: true,
|
input: ".",
|
||||||
|
fail: true,
|
||||||
|
errMsg: "unexpected character: '.'",
|
||||||
}, {
|
}, {
|
||||||
input: "100..4", fail: true,
|
input: "2.5.",
|
||||||
|
fail: true,
|
||||||
|
errMsg: "could not parse remaining input \".\"...",
|
||||||
}, {
|
}, {
|
||||||
input: "0deadbeef", fail: true,
|
input: "100..4",
|
||||||
|
fail: true,
|
||||||
|
errMsg: "could not parse remaining input \".4\"...",
|
||||||
}, {
|
}, {
|
||||||
input: "1 /", fail: true,
|
input: "0deadbeef",
|
||||||
|
fail: true,
|
||||||
|
errMsg: "bad number or duration syntax: \"0de\"",
|
||||||
}, {
|
}, {
|
||||||
input: "*1", fail: true,
|
input: "1 /",
|
||||||
|
fail: true,
|
||||||
|
errMsg: "missing right-hand side in binary expression",
|
||||||
}, {
|
}, {
|
||||||
input: "(1))", fail: true,
|
input: "*1",
|
||||||
|
fail: true,
|
||||||
|
errMsg: "no valid expression found",
|
||||||
}, {
|
}, {
|
||||||
input: "((1)", fail: true,
|
input: "(1))",
|
||||||
|
fail: true,
|
||||||
|
errMsg: "could not parse remaining input \")\"...",
|
||||||
}, {
|
}, {
|
||||||
input: "(", fail: true,
|
input: "((1)",
|
||||||
|
fail: true,
|
||||||
|
errMsg: "unclosed left parenthesis",
|
||||||
}, {
|
}, {
|
||||||
input: "1 and 1", fail: true,
|
input: "(",
|
||||||
|
fail: true,
|
||||||
|
errMsg: "unclosed left parenthesis",
|
||||||
}, {
|
}, {
|
||||||
input: "1 or 1", fail: true,
|
input: "1 and 1",
|
||||||
|
fail: true,
|
||||||
|
errMsg: "AND and OR not allowed in binary scalar expression",
|
||||||
}, {
|
}, {
|
||||||
input: "1 !~ 1", fail: true,
|
input: "1 or 1",
|
||||||
|
fail: true,
|
||||||
|
errMsg: "AND and OR not allowed in binary scalar expression",
|
||||||
}, {
|
}, {
|
||||||
input: "1 =~ 1", fail: true,
|
input: "1 !~ 1",
|
||||||
|
fail: true,
|
||||||
|
errMsg: "could not parse remaining input \"!~ 1\"...",
|
||||||
}, {
|
}, {
|
||||||
input: "-some_metric", fail: true,
|
input: "1 =~ 1",
|
||||||
|
fail: true,
|
||||||
|
errMsg: "could not parse remaining input \"=~ 1\"...",
|
||||||
}, {
|
}, {
|
||||||
input: `-"string"`, fail: true,
|
input: "-some_metric",
|
||||||
|
fail: true,
|
||||||
|
errMsg: "expected type scalar in unary expression, got vector",
|
||||||
|
}, {
|
||||||
|
input: `-"string"`,
|
||||||
|
fail: true,
|
||||||
|
errMsg: "expected type scalar in unary expression, got string",
|
||||||
},
|
},
|
||||||
// Vector binary operations.
|
// Vector binary operations.
|
||||||
{
|
{
|
||||||
|
@ -397,25 +437,45 @@ var testExpr = []struct {
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
}, {
|
}, {
|
||||||
input: "foo and 1", fail: true,
|
input: "foo and 1",
|
||||||
|
fail: true,
|
||||||
|
errMsg: "AND and OR not allowed in binary scalar expression",
|
||||||
}, {
|
}, {
|
||||||
input: "1 and foo", fail: true,
|
input: "1 and foo",
|
||||||
|
fail: true,
|
||||||
|
errMsg: "AND and OR not allowed in binary scalar expression",
|
||||||
}, {
|
}, {
|
||||||
input: "foo or 1", fail: true,
|
input: "foo or 1",
|
||||||
|
fail: true,
|
||||||
|
errMsg: "AND and OR not allowed in binary scalar expression",
|
||||||
}, {
|
}, {
|
||||||
input: "1 or foo", fail: true,
|
input: "1 or foo",
|
||||||
|
fail: true,
|
||||||
|
errMsg: "AND and OR not allowed in binary scalar expression",
|
||||||
}, {
|
}, {
|
||||||
input: "1 or on(bar) foo", fail: true,
|
input: "1 or on(bar) foo",
|
||||||
|
fail: true,
|
||||||
|
errMsg: "vector matching only allowed between vectors",
|
||||||
}, {
|
}, {
|
||||||
input: "foo == on(bar) 10", fail: true,
|
input: "foo == on(bar) 10",
|
||||||
|
fail: true,
|
||||||
|
errMsg: "vector matching only allowed between vectors",
|
||||||
}, {
|
}, {
|
||||||
input: "foo and on(bar) group_left(baz) bar", fail: true,
|
input: "foo and on(bar) group_left(baz) bar",
|
||||||
|
fail: true,
|
||||||
|
errMsg: "no grouping allowed for AND and OR operations",
|
||||||
}, {
|
}, {
|
||||||
input: "foo and on(bar) group_right(baz) bar", fail: true,
|
input: "foo and on(bar) group_right(baz) bar",
|
||||||
|
fail: true,
|
||||||
|
errMsg: "no grouping allowed for AND and OR operations",
|
||||||
}, {
|
}, {
|
||||||
input: "foo or on(bar) group_left(baz) bar", fail: true,
|
input: "foo or on(bar) group_left(baz) bar",
|
||||||
|
fail: true,
|
||||||
|
errMsg: "no grouping allowed for AND and OR operations",
|
||||||
}, {
|
}, {
|
||||||
input: "foo or on(bar) group_right(baz) bar", fail: true,
|
input: "foo or on(bar) group_right(baz) bar",
|
||||||
|
fail: true,
|
||||||
|
errMsg: "no grouping allowed for AND and OR operations",
|
||||||
},
|
},
|
||||||
// Test vector selector.
|
// Test vector selector.
|
||||||
{
|
{
|
||||||
|
@ -470,31 +530,59 @@ var testExpr = []struct {
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
}, {
|
}, {
|
||||||
input: `{`, fail: true,
|
input: `{`,
|
||||||
|
fail: true,
|
||||||
|
errMsg: "unexpected end of input inside braces",
|
||||||
}, {
|
}, {
|
||||||
input: `}`, fail: true,
|
input: `}`,
|
||||||
|
fail: true,
|
||||||
|
errMsg: "unexpected character: '}'",
|
||||||
}, {
|
}, {
|
||||||
input: `some{`, fail: true,
|
input: `some{`,
|
||||||
|
fail: true,
|
||||||
|
errMsg: "unexpected end of input inside braces",
|
||||||
}, {
|
}, {
|
||||||
input: `some}`, fail: true,
|
input: `some}`,
|
||||||
|
fail: true,
|
||||||
|
errMsg: "could not parse remaining input \"}\"...",
|
||||||
}, {
|
}, {
|
||||||
input: `some_metric{a=b}`, fail: true,
|
input: `some_metric{a=b}`,
|
||||||
|
fail: true,
|
||||||
|
errMsg: "unexpected identifier \"b\" in label matching, expected string",
|
||||||
}, {
|
}, {
|
||||||
input: `some_metric{a:b="b"}`, fail: true,
|
input: `some_metric{a:b="b"}`,
|
||||||
|
fail: true,
|
||||||
|
errMsg: "unexpected character inside braces: ':'",
|
||||||
}, {
|
}, {
|
||||||
input: `foo{a*"b"}`, fail: true,
|
input: `foo{a*"b"}`,
|
||||||
|
fail: true,
|
||||||
|
errMsg: "unexpected character inside braces: '*'",
|
||||||
}, {
|
}, {
|
||||||
input: `foo{a>="b"}`, fail: true,
|
input: `foo{a>="b"}`,
|
||||||
|
fail: true,
|
||||||
|
// TODO(fabxc): willingly lexing wrong tokens allows for more precrise error
|
||||||
|
// messages from the parser - consider if this is an option.
|
||||||
|
errMsg: "unexpected character inside braces: '>'",
|
||||||
}, {
|
}, {
|
||||||
input: `foo{gibberish}`, fail: true,
|
input: `foo{gibberish}`,
|
||||||
|
fail: true,
|
||||||
|
errMsg: "expected label matching operator but got }",
|
||||||
}, {
|
}, {
|
||||||
input: `foo{1}`, fail: true,
|
input: `foo{1}`,
|
||||||
|
fail: true,
|
||||||
|
errMsg: "unexpected character inside braces: '1'",
|
||||||
}, {
|
}, {
|
||||||
input: `{}`, fail: true,
|
input: `{}`,
|
||||||
|
fail: true,
|
||||||
|
errMsg: "vector selector must contain label matchers or metric name",
|
||||||
}, {
|
}, {
|
||||||
input: `foo{__name__="bar"}`, fail: true,
|
input: `foo{__name__="bar"}`,
|
||||||
}, {
|
fail: true,
|
||||||
input: `:foo`, fail: true,
|
errMsg: "metric name must not be set twice: \"foo\" or \"bar\"",
|
||||||
|
// }, {
|
||||||
|
// input: `:foo`,
|
||||||
|
// fail: true,
|
||||||
|
// errMsg: "bla",
|
||||||
},
|
},
|
||||||
// Test matrix selector.
|
// Test matrix selector.
|
||||||
{
|
{
|
||||||
|
@ -559,25 +647,45 @@ var testExpr = []struct {
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
}, {
|
}, {
|
||||||
input: `foo[5mm]`, fail: true,
|
input: `foo[5mm]`,
|
||||||
|
fail: true,
|
||||||
|
errMsg: "bad duration syntax: \"5mm\"",
|
||||||
}, {
|
}, {
|
||||||
input: `foo[0m]`, fail: true,
|
input: `foo[0m]`,
|
||||||
|
fail: true,
|
||||||
|
errMsg: "duration must be greater than 0",
|
||||||
}, {
|
}, {
|
||||||
input: `foo[5m30s]`, fail: true,
|
input: `foo[5m30s]`,
|
||||||
|
fail: true,
|
||||||
|
errMsg: "bad duration syntax: \"5m3\"",
|
||||||
}, {
|
}, {
|
||||||
input: `foo[5m] OFFSET 1h30m`, fail: true,
|
input: `foo[5m] OFFSET 1h30m`,
|
||||||
|
fail: true,
|
||||||
|
errMsg: "bad number or duration syntax: \"1h3\"",
|
||||||
}, {
|
}, {
|
||||||
input: `foo[]`, fail: true,
|
input: `foo[]`,
|
||||||
|
fail: true,
|
||||||
|
errMsg: "missing unit character in duration",
|
||||||
}, {
|
}, {
|
||||||
input: `foo[1]`, fail: true,
|
input: `foo[1]`,
|
||||||
|
fail: true,
|
||||||
|
errMsg: "missing unit character in duration",
|
||||||
}, {
|
}, {
|
||||||
input: `some_metric[5m] OFFSET 1`, fail: true,
|
input: `some_metric[5m] OFFSET 1`,
|
||||||
|
fail: true,
|
||||||
|
errMsg: "unexpected number \"1\" in matrix selector, expected duration",
|
||||||
}, {
|
}, {
|
||||||
input: `some_metric[5m] OFFSET 1mm`, fail: true,
|
input: `some_metric[5m] OFFSET 1mm`,
|
||||||
|
fail: true,
|
||||||
|
errMsg: "bad number or duration syntax: \"1mm\"",
|
||||||
}, {
|
}, {
|
||||||
input: `some_metric[5m] OFFSET`, fail: true,
|
input: `some_metric[5m] OFFSET`,
|
||||||
|
fail: true,
|
||||||
|
errMsg: "unexpected end of input in matrix selector, expected duration",
|
||||||
}, {
|
}, {
|
||||||
input: `(foo + bar)[5m]`, fail: true,
|
input: `(foo + bar)[5m]`,
|
||||||
|
fail: true,
|
||||||
|
errMsg: "could not parse remaining input \"[5m]\"...",
|
||||||
},
|
},
|
||||||
// Test aggregation.
|
// Test aggregation.
|
||||||
{
|
{
|
||||||
|
@ -692,21 +800,37 @@ var testExpr = []struct {
|
||||||
Grouping: clientmodel.LabelNames{"foo"},
|
Grouping: clientmodel.LabelNames{"foo"},
|
||||||
},
|
},
|
||||||
}, {
|
}, {
|
||||||
input: `sum some_metric by (test)`, fail: true,
|
input: `sum some_metric by (test)`,
|
||||||
|
fail: true,
|
||||||
|
errMsg: "unexpected identifier \"some_metric\" in aggregation, expected \"(\"",
|
||||||
}, {
|
}, {
|
||||||
input: `sum (some_metric) by test`, fail: true,
|
input: `sum (some_metric) by test`,
|
||||||
|
fail: true,
|
||||||
|
errMsg: "unexpected identifier \"test\" in grouping opts, expected \"(\"",
|
||||||
}, {
|
}, {
|
||||||
input: `sum (some_metric) by ()`, fail: true,
|
input: `sum (some_metric) by ()`,
|
||||||
|
fail: true,
|
||||||
|
errMsg: "unexpected \")\" in grouping opts, expected identifier",
|
||||||
}, {
|
}, {
|
||||||
input: `sum (some_metric) by test`, fail: true,
|
input: `sum (some_metric) by test`,
|
||||||
|
fail: true,
|
||||||
|
errMsg: "unexpected identifier \"test\" in grouping opts, expected \"(\"",
|
||||||
}, {
|
}, {
|
||||||
input: `some_metric[5m] OFFSET`, fail: true,
|
input: `some_metric[5m] OFFSET`,
|
||||||
|
fail: true,
|
||||||
|
errMsg: "unexpected end of input in matrix selector, expected duration",
|
||||||
}, {
|
}, {
|
||||||
input: `sum () by (test)`, fail: true,
|
input: `sum () by (test)`,
|
||||||
|
fail: true,
|
||||||
|
errMsg: "no valid expression found",
|
||||||
}, {
|
}, {
|
||||||
input: "MIN keeping_extra (some_metric) by (foo)", fail: true,
|
input: "MIN keeping_extra (some_metric) by (foo)",
|
||||||
|
fail: true,
|
||||||
|
errMsg: "could not parse remaining input \"by (foo)\"...",
|
||||||
}, {
|
}, {
|
||||||
input: "MIN by(test) (some_metric) keeping_extra", fail: true,
|
input: "MIN by(test) (some_metric) keeping_extra",
|
||||||
|
fail: true,
|
||||||
|
errMsg: "could not parse remaining input \"keeping_extra\"...",
|
||||||
},
|
},
|
||||||
// Test function calls.
|
// Test function calls.
|
||||||
{
|
{
|
||||||
|
@ -770,21 +894,30 @@ var testExpr = []struct {
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
}, {
|
}, {
|
||||||
input: "floor()", fail: true,
|
input: "floor()",
|
||||||
|
fail: true,
|
||||||
|
errMsg: "expected at least 1 argument(s) in call to \"floor\", got 0",
|
||||||
}, {
|
}, {
|
||||||
input: "floor(some_metric, other_metric)", fail: true,
|
input: "floor(some_metric, other_metric)",
|
||||||
|
fail: true,
|
||||||
|
errMsg: "expected at most 1 argument(s) in call to \"floor\", got 2",
|
||||||
}, {
|
}, {
|
||||||
input: "floor(1)", fail: true,
|
input: "floor(1)",
|
||||||
|
fail: true,
|
||||||
|
errMsg: "expected type vector in call to function \"floor\", got scalar",
|
||||||
}, {
|
}, {
|
||||||
input: "non_existant_function_far_bar()", fail: true,
|
input: "non_existant_function_far_bar()",
|
||||||
|
fail: true,
|
||||||
|
errMsg: "unknown function with name \"non_existant_function_far_bar\"",
|
||||||
}, {
|
}, {
|
||||||
input: "rate(some_metric)", fail: true,
|
input: "rate(some_metric)",
|
||||||
|
fail: true,
|
||||||
|
errMsg: "expected type matrix in call to function \"rate\", got vector",
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestParseExpressions(t *testing.T) {
|
func TestParseExpressions(t *testing.T) {
|
||||||
for _, test := range testExpr {
|
for _, test := range testExpr {
|
||||||
|
|
||||||
parser := newParser(test.input)
|
parser := newParser(test.input)
|
||||||
|
|
||||||
expr, err := parser.parseExpr()
|
expr, err := parser.parseExpr()
|
||||||
|
@ -793,6 +926,10 @@ func TestParseExpressions(t *testing.T) {
|
||||||
t.Fatalf("could not parse: %s", err)
|
t.Fatalf("could not parse: %s", err)
|
||||||
}
|
}
|
||||||
if test.fail && err != nil {
|
if test.fail && err != nil {
|
||||||
|
if !strings.Contains(err.Error(), test.errMsg) {
|
||||||
|
t.Errorf("unexpected error on input '%s'", test.input)
|
||||||
|
t.Fatalf("expected error to contain %q but got %q", test.errMsg, err)
|
||||||
|
}
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -804,6 +941,10 @@ func TestParseExpressions(t *testing.T) {
|
||||||
|
|
||||||
if test.fail {
|
if test.fail {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
if !strings.Contains(err.Error(), test.errMsg) {
|
||||||
|
t.Errorf("unexpected error on input '%s'", test.input)
|
||||||
|
t.Fatalf("expected error to contain %q but got %q", test.errMsg, err)
|
||||||
|
}
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
t.Errorf("error on input '%s'", test.input)
|
t.Errorf("error on input '%s'", test.input)
|
||||||
|
|
|
@ -72,6 +72,9 @@ func Tree(node Node) string {
|
||||||
}
|
}
|
||||||
|
|
||||||
func tree(node Node, level string) string {
|
func tree(node Node, level string) string {
|
||||||
|
if node == nil {
|
||||||
|
return fmt.Sprintf("%s |---- %T\n", level, node)
|
||||||
|
}
|
||||||
typs := strings.Split(fmt.Sprintf("%T", node), ".")[1]
|
typs := strings.Split(fmt.Sprintf("%T", node), ".")[1]
|
||||||
|
|
||||||
var t string
|
var t string
|
||||||
|
|
|
@ -77,7 +77,7 @@ func TestQuery(t *testing.T) {
|
||||||
{
|
{
|
||||||
queryStr: "expr=(badexpression",
|
queryStr: "expr=(badexpression",
|
||||||
status: http.StatusOK,
|
status: http.StatusOK,
|
||||||
bodyRe: `{"type":"error","value":"Parse error at char 15: unexpected unclosed left parenthesis in paren expression","version":1}`,
|
bodyRe: `{"type":"error","value":"Parse error at char 15: unclosed left parenthesis","version":1}`,
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue