-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathparser.go
206 lines (171 loc) · 6.14 KB
/
parser.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
package dockerlang
import "fmt"
type Stack struct {
Elements []AST
}
func NewStack() *Stack {
return &Stack{Elements: []AST{}}
}
func (s *Stack) Push(e AST) {
s.Elements = append(s.Elements, e)
}
func (s *Stack) Pop() AST {
e := s.Peek()
if e == nil {
return nil
}
s.Elements = s.Elements[:len(s.Elements)-1]
return e
}
func (s *Stack) Peek() AST {
if len(s.Elements) == 0 {
return nil
}
return s.Elements[len(s.Elements)-1]
}
func (s *Stack) Length() int {
return len(s.Elements)
}
func (c *Compterpreter) Parse() error {
// build the global StackTree, for all expressions in the global scope as part of an implicit anonymous function
var (
opsStack = NewStack()
exprStack = NewStack()
parenCount = 0
)
// TODO make the root expression an EXIT_OPERATOR which will operate
// on all program roots.
c.StackTree = &Expr{}
for idx, token := range c.Tokens {
if parenCount < 0 {
return SyntaxError("too many close parens", "current token:", string(token.Value))
}
switch token.Type {
case OPERATOR:
opsStack.Push(&Expr{Op: token.Value, Arity: OP_TO_ARITY[token.Value]})
case INT:
exprStack.Push(&Literal{Type: INT, Value: token.Value})
case IDENTIFIER:
// is this a declaration or a reference?
identifier, err := c.StackTree.ParseIdentifier(token, opsStack)
if err != nil {
return err
}
exprStack.Push(identifier)
case PUNCTUATION:
switch token.Value {
case "(":
// push the open paren to the expression stack so we know when an expression
// or array is starting.
exprStack.Push(&Expr{Op: token.Value})
// check to see if the next token is *not* an operator/keyword
if idx + 1 < len(c.Tokens) {
nextToken := c.Tokens[idx+1]
if nextToken.Type != OPERATOR && nextToken.Type != KEYWORD {
// add the implicit list operator to the ops stack since an
// open paren followed by a non-operator or keyword is assumed
// to be an array
opsStack.Push(&Expr{Op: IMPLICIT_LIST_OPERATOR, Arity: -1})
}
}
case ")":
// shit gets real
var opsExpr = opsStack.Pop().(*Expr)
for {
// make sure we're not popping nil into exprs
if exprStack.Peek() == nil {
return SyntaxError()
}
// pop the expression off the expr stack
exp := exprStack.Pop()
// check whether this popped expression is an open paren
// if it is, stop the loop
if e, ok := exp.(*Expr); ok && e.Op == "(" {
break
}
// add this expression to the expression operands array`
opsExpr.Operands = append([]AST{exp, opsExpr.Operands...)
}
// push modified ops expr onto the expr stack
exprStack.Push(opsExpr)
parenCount--
default:
// whatever
}
}
// if there is only 1 element in the expressionStack, we have successfully parsed
// a single expression
if exprStack.Length() == 1 && parenCount == 0 {
// there should be nothing on the operations stack!
if opsStack.Peek() != nil {
fmt.Println("error in loop")
fmt.Println(opsStack.Peek())
fmt.Println(exprStack.Peek())
// oh noooo!
return SyntaxError()
}
// add this expression to the sequential list of expressions in the
// programs execution
c.StackTree.Operands = append(
c.StackTree.Operands,
exprStack.Pop().(*Expr),
)
}
}
// there should be nothing on the expression stack or operation stack
if exprStack.Length() != 0 || opsStack.Peek() != nil {
fmt.Println("SOMETHING IS AWRY")
fmt.Println(opsStack.Peek())
fmt.Println(exprStack.Peek())
// oh no!
return SyntaxError()
}
return nil
}
// TODO (cw,mr|4.11.2018) refactor this so that there is better separation of concerns
// i.e. it *would* be useful to have a function for adding local, global, args, etc. to
// an expression, but we might not want to do all this parsing in that function.
// NOTE: once we implement functions, we are going to want to check globals and args!
func (e *Expr) ParseIdentifier(token Token, opsStack *Stack) (*Identifier, error) {
var (
isDefined bool = false
knownIdentifier *Identifier
)
// check all locals to see if we've already defined this identifier
for name, ast := range e.Locals {
if token.Value == name {
// this means we have already defined this identifer
isDefined = true
knownIdentifier = ast.(*Identifier)
break
}
}
prev := opsStack.Peek().(*Expr)
// this is an identifier reference
if prev.Op != VARIABLE_INITIALIZATION && prev.Op != FUNCTION_KEYWORD {
if !isDefined {
return nil, SyntaxError("idk what", "'", token.Value, "'", "is...maybe you forgot to define it :)")
}
// we are assuming that if an identifier is defined, then it is also bounded (or whatever)
return knownIdentifier, nil
}
// if we are here, this is an identifier definition
// we are trying to re-define this identifier
if isDefined {
return nil, SyntaxError("oops, i think you've already defined", "'", token.Value, "'")
}
// actually define this identifier
switch prev.Op {
case VARIABLE_INITIALIZATION:
knownIdentifier = &Identifier{Type: VARIABLE_IDENTIFIER, Name: token.Value, Bound: true}
case FUNCTION_KEYWORD:
knownIdentifier = &Identifier{Type: FUNCTION_IDENTIFIER, Name: token.Value, Bound: true}
}
// TODO (cw, mr|4.11.2018) maybe put this in the Expr constructor once one exists
if e.Locals == nil {
e.Locals = map[string]AST{}
}
// add this to the local scope of the current expression
e.Locals[knownIdentifier.Name] = knownIdentifier
return knownIdentifier, nil
}