opentofu/config/lang/lex.go

310 lines
6.4 KiB
Go
Raw Normal View History

2015-01-11 14:38:45 -06:00
package lang
import (
"bytes"
"fmt"
"unicode"
"unicode/utf8"
2015-01-12 02:28:47 -06:00
"github.com/hashicorp/terraform/config/lang/ast"
2015-01-11 14:38:45 -06:00
)
//go:generate go tool yacc -p parser lang.y
// The parser expects the lexer to return 0 on EOF.
const lexEOF = 0
// The parser uses the type <prefix>Lex as a lexer. It must provide
// the methods Lex(*<prefix>SymType) int and Error(string).
type parserLex struct {
Err error
Input string
mode parserMode
2015-01-11 14:38:45 -06:00
interpolationDepth int
pos int
width int
2015-01-12 02:28:47 -06:00
col, line int
lastLine int
astPos *ast.Pos
}
// parserToken is the token yielded to the parser. The value can be
// determined within the parser type based on the enum value returned
// from Lex.
type parserToken struct {
Value interface{}
Pos ast.Pos
2015-01-11 14:38:45 -06:00
}
// parserMode keeps track of what mode we're in for the parser. We have
// two modes: literal and interpolation. Literal mode is when strings
// don't have to be quoted, and interpolations are defined as ${foo}.
// Interpolation mode means that strings have to be quoted and unquoted
// things are identifiers, such as foo("bar").
type parserMode uint8
const (
parserModeInvalid parserMode = 0
parserModeLiteral = 1 << iota
parserModeInterpolation
)
2015-01-11 14:38:45 -06:00
// The parser calls this method to get each new token.
func (x *parserLex) Lex(yylval *parserSymType) int {
if x.mode == parserModeInvalid {
x.mode = parserModeLiteral
}
2015-01-12 02:28:47 -06:00
defer func() {
if yylval.token != nil && yylval.token.Pos.Column == 0 {
yylval.token.Pos = *x.astPos
}
}()
x.astPos = nil
return x.lex(yylval)
}
func (x *parserLex) lex(yylval *parserSymType) int {
switch x.mode {
case parserModeLiteral:
return x.lexModeLiteral(yylval)
case parserModeInterpolation:
return x.lexModeInterpolation(yylval)
default:
x.Error(fmt.Sprintf("Unknown parse mode: %s", x.mode))
return lexEOF
}
}
func (x *parserLex) lexModeLiteral(yylval *parserSymType) int {
2015-01-11 14:38:45 -06:00
for {
c := x.next()
if c == lexEOF {
return lexEOF
}
// Are we starting an interpolation?
if c == '$' && x.peek() == '{' {
x.next()
x.interpolationDepth++
x.mode = parserModeInterpolation
2015-01-11 14:38:45 -06:00
return PROGRAM_BRACKET_LEFT
}
// We're just a normal string that isn't part of any interpolation yet.
x.backup()
result, terminated := x.lexString(yylval, x.interpolationDepth > 0)
// If the string terminated and we're within an interpolation already
// then that means that we finished a nested string, so pop
// back out to interpolation mode.
if terminated && x.interpolationDepth > 0 {
x.mode = parserModeInterpolation
// If the string is empty, just skip it. We're still in
// an interpolation so we do this to avoid empty nodes.
2015-01-12 02:28:47 -06:00
if yylval.token.Value.(string) == "" {
return x.lex(yylval)
}
}
return result
}
}
func (x *parserLex) lexModeInterpolation(yylval *parserSymType) int {
for {
c := x.next()
if c == lexEOF {
return lexEOF
2015-01-11 15:03:37 -06:00
}
// Ignore all whitespace
if unicode.IsSpace(c) {
continue
}
2015-01-11 14:38:45 -06:00
// If we see a double quote and we're in an interpolation, then
// we are lexing a string.
2015-01-11 15:03:37 -06:00
if c == '"' {
result, terminated := x.lexString(yylval, true)
if !terminated {
// The string didn't end, which means that we're in the
// middle of starting another interpolation.
x.mode = parserModeLiteral
// If the string is empty and we're starting an interpolation,
// then just skip it to avoid empty string AST nodes
2015-01-12 02:28:47 -06:00
if yylval.token.Value.(string) == "" {
return x.lex(yylval)
}
}
return result
2015-01-11 14:38:45 -06:00
}
switch c {
case '}':
x.interpolationDepth--
x.mode = parserModeLiteral
2015-01-11 14:38:45 -06:00
return PROGRAM_BRACKET_RIGHT
2015-01-11 15:03:37 -06:00
case '(':
return PAREN_LEFT
case ')':
return PAREN_RIGHT
case ',':
return COMMA
2015-01-11 14:38:45 -06:00
default:
x.backup()
2015-01-11 15:03:37 -06:00
return x.lexId(yylval)
2015-01-11 14:38:45 -06:00
}
}
}
func (x *parserLex) lexId(yylval *parserSymType) int {
var b bytes.Buffer
for {
c := x.next()
if c == lexEOF {
break
}
// If this isn't a character we want in an ID, return out.
// One day we should make this a regexp.
if c != '_' &&
c != '-' &&
c != '.' &&
c != '*' &&
!unicode.IsLetter(c) &&
!unicode.IsNumber(c) {
x.backup()
break
}
if _, err := b.WriteRune(c); err != nil {
x.Error(err.Error())
return lexEOF
}
}
2015-01-12 02:28:47 -06:00
yylval.token = &parserToken{Value: b.String()}
2015-01-11 14:38:45 -06:00
return IDENTIFIER
}
func (x *parserLex) lexString(yylval *parserSymType, quoted bool) (int, bool) {
2015-01-11 14:38:45 -06:00
var b bytes.Buffer
terminated := false
2015-01-11 14:38:45 -06:00
for {
c := x.next()
if c == lexEOF {
if quoted {
x.Error("unterminated string")
}
2015-01-11 14:38:45 -06:00
break
}
// Behavior is a bit different if we're lexing within a quoted string.
if quoted {
// If its a double quote, we've reached the end of the string
if c == '"' {
terminated = true
2015-01-11 14:38:45 -06:00
break
}
// Let's check to see if we're escaping anything.
if c == '\\' {
switch n := x.next(); n {
case '\\':
fallthrough
case '"':
c = n
case 'n':
c = '\n'
default:
x.backup()
}
}
}
// If we hit a '}' and we're in a program, then end it.
if c == '}' && x.interpolationDepth > 0 {
x.backup()
break
}
// If we hit a dollar sign, then check if we're starting
// another interpolation. If so, then we're done.
if c == '$' && x.peek() == '{' {
x.backup()
break
}
if _, err := b.WriteRune(c); err != nil {
x.Error(err.Error())
return lexEOF, false
2015-01-11 14:38:45 -06:00
}
}
2015-01-12 02:28:47 -06:00
yylval.token = &parserToken{Value: b.String()}
return STRING, terminated
2015-01-11 14:38:45 -06:00
}
// Return the next rune for the lexer.
func (x *parserLex) next() rune {
if int(x.pos) >= len(x.Input) {
x.width = 0
return lexEOF
}
r, w := utf8.DecodeRuneInString(x.Input[x.pos:])
x.width = w
x.pos += x.width
2015-01-12 02:28:47 -06:00
if x.line == 0 {
x.line = 1
x.col = 1
} else {
x.col += 1
}
if r == '\n' {
x.lastLine = x.col
x.line += 1
x.col = 1
}
if x.astPos == nil {
x.astPos = &ast.Pos{Column: x.col, Line: x.line}
}
2015-01-11 14:38:45 -06:00
return r
}
// peek returns but does not consume the next rune in the input
func (x *parserLex) peek() rune {
r := x.next()
x.backup()
return r
}
// backup steps back one rune. Can only be called once per next.
func (x *parserLex) backup() {
x.pos -= x.width
2015-01-12 02:28:47 -06:00
x.col -= 1
// If we are at column 0, we're backing up across a line boundary
// so we need to be careful to get the proper value.
if x.col == 0 {
x.col = x.lastLine
x.line -= 1
}
2015-01-11 14:38:45 -06:00
}
// The parser calls this method on a parse error.
func (x *parserLex) Error(s string) {
x.Err = fmt.Errorf("parse error: %s", s)
}