-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathparse.ts
110 lines (98 loc) · 2.27 KB
/
parse.ts
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
import { Term, Type, Operator, Char } from './lib/structs'
export function parseQuery(queryString: string): Term[] {
const queryRgx: RegExp = /[+?-]?(["'(]).+?(\1|\))|[^"'()\s]+/g
let matches = queryString.match(queryRgx)
if (!matches) return []
return matches.map((match) => {
/* strip op */
let op = Operator.OR
if (/[+?-]/.test(match[0])) {
op = Operator[match[0]]
match = match.substring(1)
}
if (match[0] == '"' || match[0] == "'") {
return {
type: Type.phrase,
val: match,
op,
}
} else {
return {
type: Type.token,
val: match,
op,
}
}
})
}
export function parseQueryEXP(queryString: string): Term[] {
return lex(queryString)
}
function lex(queryString: string) {
const lexes = []
let curToken = ''
let curTokenType: Type = Type.token
let curOperator: Operator = Operator.OR
let isToken = false
for (let i = 0; i < queryString.length; i++) {
const ch = new Char(queryString[i])
if (ch.isSpace()) {
if (curToken.length) {
if (curTokenType === Type.token) {
isToken = true
} else {
curToken += ch.char
}
}
} else if (ch.isOperator()) {
if (curToken.length) {
throw new ParseError(
ch.char,
i,
`Logical operator must precede query term.`,
)
}
curOperator = Operator[ch.char]
} else if (ch.isQuote()) {
if (curToken.length && curTokenType === Type.phrase) {
isToken = true
} else {
curTokenType = Type.phrase
}
} else {
curToken += ch.char
}
if (isToken) {
resetCur()
}
}
if (curToken.length) {
resetCur()
}
function resetCur() {
lexes.push({
type: curTokenType,
val: curToken,
op: curOperator,
})
curToken = ''
curTokenType = Type.token
curOperator = Operator.OR
isToken = false
}
return lexes
}
export class ParseError extends Error {
char: string
pos: number
constructor(char: string, pos: number, message: string) {
super(message)
this.char = char
this.pos = pos
}
print() {
return `err: “ ${this.char} ” in position ${this.pos} ${this.message}`
}
// U+201C “
// U+201D ”
}