-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathsearch.ts
127 lines (111 loc) · 3.36 KB
/
search.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
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
import { aql, literal, join } from 'arangojs/aql'
import { Query, Collection, Term, Type } from './lib/structs'
import { parseQuery } from './parse'
export function buildSearch(query: Query): any {
/* parse string query */
query.terms =
typeof query.terms == 'string' ? parseQuery(query.terms) : query.terms
/* build boolean pieces */
let ANDS = buildOps(query.collections, query.terms, '+')
let ORS = buildOps(query.collections, query.terms, '?')
let NOTS = buildOps(query.collections, query.terms, '-')
/* handle combinations */
if (ANDS && ORS) {
ANDS = aql`${ANDS} OR (${ANDS} AND ${ORS})`
ORS = undefined
}
if (!!NOTS) {
NOTS = aql`${ANDS || ORS ? literal(' AND ') : undefined}
${NOTS.phrases ? literal(' NOT ') : undefined} ${NOTS.phrases}
${NOTS.phrases && NOTS.tokens ? literal(' AND ') : undefined} ${
NOTS.tokens
}`
}
const cols = query.collections.map((c) => c.name)
/* if an empty query.terms string or array is passed, SEARCH true, bringing
* back all documents in view */
return aql`
SEARCH
${ANDS}
${ORS}
${NOTS}
${(!ANDS && !ORS && !NOTS) || undefined}
OPTIONS { collections: ${cols} }
SORT TFIDF(doc) DESC`
}
function buildOps(collections: Collection[], terms: Term[], op: string): any {
const opWord: string = op == '+' ? ' AND ' : ' OR '
let queryTerms: any = terms.filter((t: Term) => t.op == op)
if (!queryTerms.length) return
/* phrases */
let phrases = queryTerms.filter((qT: Term) => qT.type == Type.phrase)
phrases = buildPhrases(phrases, collections, opWord)
/* tokens */
let tokens = queryTerms.filter((qT: { type: string }) => qT.type === 'tok')
tokens = tokens && buildTokens(tokens, collections)
/* if (!phrases && !tokens) return */
if (op == '-') return { phrases, tokens }
if (phrases && tokens) return join([phrases, tokens], opWord)
return tokens || phrases
}
function buildPhrases(
phrases: Term[],
collections: Collection[],
opWord: string,
): any {
if (!phrases.length) return undefined
return join(
phrases.map((phrase: any) => buildPhrase(phrase, collections)),
opWord,
)
}
function buildPhrase(phrase: Term, collections: Collection[]): any {
const phrases = []
collections.forEach((coll) =>
coll.keys.forEach((k: string) =>
phrases.push(
aql`PHRASE(doc.${k}, ${phrase.val.slice(1, -1)}, ${coll.analyzer})`,
),
),
)
return aql`(${join(phrases, ' OR ')})`
}
function buildTokens(tokens: Term[], collections: Collection[]): any {
if (!tokens.length) return
const opWordMap = {
'+': 'ALL',
'?': 'ANY',
'-': 'NONE',
}
const mapped = tokens.reduce((a, v) => {
const op = a[opWordMap[v.op]]
a[opWordMap[v.op]] = op ? op + ' ' + v.val : v.val
return a
}, {})
const makeTokenAnalyzers = (
tokens: Term[],
op: string,
analyzer: string,
keys: string[],
) => {
return join(
keys.map(
(k) => aql`
ANALYZER(
TOKENS(${tokens}, ${analyzer})
${literal(op)} IN doc.${k}, ${analyzer})`,
),
' OR ',
)
}
let remapped = []
collections.forEach((col) => {
remapped.push(
...Object.keys(mapped).map((op) =>
makeTokenAnalyzers(mapped[op], op, col.analyzer, col.keys),
),
)
})
return aql`MIN_MATCH(${join(remapped, ', ')},
${tokens[0].op === '-' ? collections.length : 1})`
}