-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathparse-data.js
112 lines (103 loc) · 3.3 KB
/
parse-data.js
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
const debug = require('debug')(__filename.split('/').slice(-1).join())
const nativize = require('./nativize')
const { escape } = require('xregexp')
/*
This is the regex matching the section of in the intpu which aligns
with a slot in the template. It's a quoted string or the minimal
matching text which doesn't contain quotes - that ends up using the
next literal in the template as the ending delimiter.
slotRE is really a string containing a regex, but we write it as
/foo/.source to avoid another doubling of the backslashes.
*/
const slotRE = /"([^"\\]|\\\"|\\)*"|[^"]*?/.source //eslint-disable-line
function makeRE (parsed, index, varMap) {
const out = []
for (const part of parsed) {
if (typeof (part) === 'string') {
out.push(escape(part))
} else {
const groupName = 'var_' + index + '_' + part.count
out.push('(?<' + groupName + '>' + slotRE + ')') // or quoted anything
varMap[groupName] = part
}
}
return out.join('')
}
function mergeTemplates (templates) {
const re = []
for (const t of templates) {
re.push('(?<t_' + t.index + '>' + t.re + ')')
}
re.push('(?<ws>\\s+)')
re.push('(?<junk>.+?)')
return re.join('|')
}
function * parse (mapper, text, reftable) {
// //console.log('re=%o', re)
while (true) {
// console.log('')
debug('parse() next chunk')
const m = mapper.mergedRE.exec(text)
if (!m) break
const line = m[0]
debug('match=%o', m)
if (m.groups.junk) {
// console.log('Junk %o', m.groups.junk)
continue
}
if (m.groups.ws) {
// console.log('Whitespace %o', m.groups.ws)
continue
}
const b = {}
let tnum
let id
let subject
for (const [key, value] of Object.entries(m.groups)) {
if (!value) continue // to unused ones are set to undefined
if (key.startsWith('t_')) {
tnum = parseInt(key.slice(2))
// console.log('match line %o', line)
}
const slot = mapper.varMap[key]
if (slot) {
let v = value
if (v.startsWith('"')) {
// should only be possible if it's a fully quoted string
v = JSON.parse(v)
}
if (slot.type === 'ref') {
v = v.trim().toLowerCase()
if (slot.name === 'id') {
id = v
} else if (slot.name === 'subject') {
subject = v
} else {
reftable.setValueToId(b, slot.name, v)
debug('after reftable.setValueToId %O', b)
}
} else {
const native = nativize(slot.type, v)
b[slot.name] = native
}
}
}
if (subject) {
const ref = reftable.getObject(subject)
if (!ref) throw Error('subject reference not found: ' + JSON.stringify(subject))
debug('[subject] match, overlaying %O on %O', b, ref)
Object.assign(ref, b)
continue
}
debug('matched template %d text %O, vars %O', tnum, line, b)
debug('mapper %O', mapper)
const t = mapper.templates[tnum]
const newObj = t.local.input(b)
b._forwardTo = newObj // so forward references can resolve
// and how that the newObj is set, we can resolve references
if (id) reftable.gotId(id, newObj)
yield newObj
}
// console.log('DONE, at pos', mapper.mergedRE.lastIndex, text.length)
}
module.exports = { makeRE, mergeTemplates, parse }