-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathserver_lexer.go
186 lines (148 loc) · 3.42 KB
/
server_lexer.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
// Copyright 2014 The imapsrv Authors. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the imapsrv.LICENSE file.
package imap
import (
"bufio"
"fmt"
"strconv"
)
type lexer struct {
reader *bufio.Reader
current byte
}
// The lexer produces tokens
type token struct {
value string
tokType tokenType
}
// Token types
type tokenType int
const (
stringTokenType = iota
eolTokenType
invalidTokenType
)
// Literal tokens
var invalidToken = &token{"", invalidTokenType}
var eolToken = &token{"", eolTokenType}
// Ascii codes
const (
endOfInput = 0x00
CR = 0x0d
lf = 0x0a
space = 0x20
doubleQuote = 0x22
zero = 0x30
nine = 0x39
leftCurly = 0x7b
rightCurly = 0x7d
backslash = 0x5c
)
// Create an IMAP lexer
func createLexer(in *bufio.Reader) *lexer {
// Fake the first character - use a space that will be skipped
return &lexer{reader: in, current: space}
}
// Get the next token
func (l *lexer) next() *token {
// Skip whitespace
l.skipSpace()
// Consider the first character - this gives the type of argument
switch l.current {
case CR:
l.consumeEol()
return eolToken
case doubleQuote:
l.consume()
return l.qstring()
case leftCurly:
l.consume()
return l.literal()
default:
return l.astring()
}
}
// Read a quoted string
func (l *lexer) qstring() *token {
var buffer = make([]byte, 0, 16)
// Collect the characters that are within double quotes
for l.current != doubleQuote {
switch l.current {
case CR, lf:
err := parseError(fmt.Sprintf(
"Unexpected character %q in quoted string", l.current))
panic(err)
case backslash:
l.consume()
buffer = append(buffer, l.current)
default:
buffer = append(buffer, l.current)
}
// Get the next character
l.consume()
}
// Ignore the closing quote
l.consume()
return &token{string(buffer), stringTokenType}
}
// Parse a length tagged literal
func (l *lexer) literal() *token {
lengthBuffer := make([]byte, 0, 8)
// Get the length of the literal
for l.current != rightCurly {
if l.current < zero || l.current > nine {
err := parseError(fmt.Sprintf(
"Unexpected character %q in literal length", l.current))
panic(err)
}
lengthBuffer = append(lengthBuffer, l.current)
l.consume()
}
// Extract the literal length as an int
length, err := strconv.ParseInt(string(lengthBuffer), 10, 32)
if err != nil {
panic(parseError(err.Error()))
}
// Consume the right curly and the newline that should follow
l.consumeEol()
buffer := make([]byte, 0, 64)
// Read the literal
for length > 0 {
buffer = append(buffer, l.current)
length -= 1
l.consume()
}
return &token{string(buffer), stringTokenType}
}
// A very loose interpretation of a non-quoted string
func (l *lexer) astring() *token {
buffer := make([]byte, 0, 16)
for l.current > space && l.current < 0x7f {
buffer = append(buffer, l.current)
l.consume()
}
return &token{string(buffer), stringTokenType}
}
// Skip whitespace
func (l *lexer) skipSpace() {
for l.current == space || l.current == lf {
l.consume()
}
}
// Consume until end of line
func (l *lexer) consumeEol() {
// Consume until the linefeed
for l.current != lf {
l.consume()
}
}
// Move forward 1 byte
func (l *lexer) consume() {
var err error
l.current, err = l.reader.ReadByte()
// Panic with a parser error if the read fails
if err != nil {
panic(parseError(err.Error()))
}
}