Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

geo/wkt: simplify parser grammar and improve error messages #60561

Merged
merged 1 commit into from
Feb 16, 2021
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
145 changes: 139 additions & 6 deletions pkg/geo/wkt/lex.go
Original file line number Diff line number Diff line change
Expand Up @@ -36,21 +36,28 @@ type ParseError struct {
problem string
pos int
str string
hint string
}

func (e *ParseError) Error() string {
return fmt.Sprintf("%s at pos %d\n%s\n%s^", e.problem, e.pos, e.str, strings.Repeat(" ", e.pos))
// TODO(ayang): print only relevant line (with line and pos no) instead of entire input
err := fmt.Sprintf("%s at pos %d\n%s\n%s^", e.problem, e.pos, e.str, strings.Repeat(" ", e.pos))
if e.hint != "" {
err += fmt.Sprintf("\nHINT: %s", e.hint)
}
return err
}

// Constant expected by parser when lexer reaches EOF.
const eof = 0

type wktLex struct {
line string
pos int
lastPos int
ret geom.T
lastErr error
line string
pos int
lastPos int
ret geom.T
curLayout geom.Layout
lastErr error
}

// Lex lexes a token from the input.
Expand Down Expand Up @@ -224,10 +231,136 @@ func (l *wktLex) trimLeft() {
}
}

func getDefaultLayoutForStride(stride int) geom.Layout {
switch stride {
case 2:
return geom.XY
case 3:
return geom.XYZ
case 4:
return geom.XYZM
default:
// This should never happen.
panic("unsupported stride")
}
}

func (l *wktLex) validateStrideAndSetLayoutIfNoLayout(stride int) bool {
if !l.validateStride(stride) {
return false
}
l.setLayoutIfNoLayout(getDefaultLayoutForStride(stride))
return true
}

func (l *wktLex) validateStride(stride int) bool {
if !l.isValidStrideForLayout(stride) {
l.setIncorrectStrideError(stride, "")
return false
}
return true
}

func (l *wktLex) isValidStrideForLayout(stride int) bool {
switch l.curLayout {
case geom.NoLayout:
return true
case geom.XY:
return stride == 2
case geom.XYM:
return stride == 3
case geom.XYZ:
return stride == 3
case geom.XYZM:
return stride == 4
default:
// This should never happen.
panic("unknown geom.Layout")
}
}

func (l *wktLex) setLayout(layout geom.Layout) bool {
if layout == l.curLayout {
return true
}
if l.curLayout != geom.NoLayout {
l.setIncorrectLayoutError(layout, "")
return false
}
l.setLayoutIfNoLayout(layout)
return true
}

func (l *wktLex) setLayoutEmptyInCollection() bool {
if l.curLayout == geom.XY {
return true
}
if l.curLayout == geom.NoLayout {
l.curLayout = geom.XY
return true
}
l.setIncorrectLayoutError(geom.XY, "EMPTY is XY layout in base geometry type collection")
return false
}

func (l *wktLex) setLayoutIfNoLayout(layout geom.Layout) {
switch l.curLayout {
case geom.NoLayout:
l.curLayout = layout
case geom.XY, geom.XYM, geom.XYZ, geom.XYZM:
break
default:
// This should never happen.
panic("unknown geom.Layout")
}
}

func (l *wktLex) setLexError(expectedTokType string) {
l.lastErr = &LexError{expectedTokType: expectedTokType, pos: l.lastPos, str: l.line}
}

func getLayoutName(layout geom.Layout) string {
switch layout {
case geom.XY:
return "XY"
case geom.XYM:
return "XYM"
case geom.XYZ:
return "XYZ"
case geom.XYZM:
return "XYZM"
default:
// This should never happen.
panic("unknown geom.Layout")
}
}

func (l *wktLex) setIncorrectStrideError(incorrectStride int, hint string) {
problem := fmt.Sprintf("mixed dimensionality, parsed layout is %s so expecting %d coords but got %d coords",
getLayoutName(l.curLayout), l.curLayout.Stride(), incorrectStride)
l.setParseError(problem, hint)
}

func (l *wktLex) setIncorrectLayoutError(incorrectLayout geom.Layout, hint string) {
problem := fmt.Sprintf("mixed dimensionality, parsed layout is %s but encountered layout of %s",
getLayoutName(l.curLayout), getLayoutName(incorrectLayout))
l.setParseError(problem, hint)
}

func (l *wktLex) setParseError(problem string, hint string) {
// Lex errors take precedence.
if l.lastErr != nil {
return
}
errProblem := "syntax error: " + problem
l.lastErr = &ParseError{
problem: errProblem,
pos: l.lastPos,
str: l.line,
hint: hint,
}
}

func (l *wktLex) Error(s string) {
// NB: Lex errors are set in the Lex function.
if l.lastErr == nil {
Expand Down
Loading