Parser.js
2.55 KB
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
/*
MIT License http://www.opensource.org/licenses/mit-license.php
Author Tobias Koppers @sokra
*/
function ignoreFunction() {}
function createReturningFunction(value) {
return function() {
return value;
};
}
function Parser(states) {
this.states = this.compileStates(states);
}
Parser.prototype.compileStates = function(states) {
var result = {};
Object.keys(states).forEach(function(name) {
result[name] = this.compileState(states[name], states);
}, this);
return result;
};
Parser.prototype.compileState = function(state, states) {
var regExps = [];
function iterator(str, value) {
regExps.push({
groups: Parser.getGroupCount(str),
regExp: str,
value: value
});
}
function processState(statePart) {
if(Array.isArray(statePart)) {
statePart.forEach(processState);
} else if(typeof statePart === "object") {
Object.keys(statePart).forEach(function(key) {
iterator(key, statePart[key]);
});
} else if(typeof statePart === "string") {
processState(states[statePart]);
} else {
throw new Error("Unexpected 'state' format");
}
}
processState(state);
var total = regExps.map(function(r) {
return "(" + r.regExp + ")";
}).join("|");
var actions = [];
var pos = 1;
regExps.forEach(function(r) {
var fn;
if(typeof r.value === "function") {
fn = r.value;
} else if(typeof r.value === "string") {
fn = createReturningFunction(r.value);
} else {
fn = ignoreFunction;
}
actions.push({
name: r.regExp,
fn: fn,
pos: pos,
pos2: pos + r.groups + 1
});
pos += r.groups + 1;
});
return {
regExp: new RegExp(total, "g"),
actions: actions
};
};
Parser.getGroupCount = function(regExpStr) {
return new RegExp("(" + regExpStr + ")|^$").exec("").length - 2;
};
Parser.prototype.parse = function(initialState, string, context) {
context = context || {};
var currentState = initialState;
var currentIndex = 0;
for(;;) {
var state = this.states[currentState];
var regExp = state.regExp;
regExp.lastIndex = currentIndex;
var match = regExp.exec(string);
if(!match) return context;
var actions = state.actions;
currentIndex = state.regExp.lastIndex;
for(var i = 0; i < actions.length; i++) {
var action = actions[i];
if(match[action.pos]) {
var ret = action.fn.apply(context, Array.prototype.slice.call(match, action.pos, action.pos2).concat([state.regExp.lastIndex - match[0].length, match[0].length]));
if(ret) {
if(!(ret in this.states))
throw new Error("State '" + ret + "' doesn't exist");
currentState = ret;
}
break;
}
}
}
};
module.exports = Parser;