jcov 1.0.1 → 1.1

Sign up to get free protection for your applications and to get access to all the features.
@@ -0,0 +1,92 @@
1
+ module JCov
2
+
3
+ class Loader
4
+
5
+ attr_accessor :coverable_files
6
+
7
+ def initialize(coverable_files, options={})
8
+ @options = options
9
+ @coverable_files = coverable_files
10
+ @file_cache = {}
11
+
12
+ create_parser
13
+ end
14
+
15
+ def load(file)
16
+ if @file_cache[file]
17
+ # reuse previously loaded file
18
+ content = @file_cache[file]
19
+ else
20
+ content = File.read(file)
21
+
22
+ # is this a file we need to instrument?
23
+ if coverable_files.include? file
24
+
25
+ # run it through the js parser to get coverage data
26
+ calculate_coverage_data file, content
27
+
28
+ # update the content with the coverage instrumentations
29
+ content = instrument_script(content, file)
30
+
31
+ # cache the file if it's reloaded
32
+ @file_cache[file] = content
33
+ end
34
+ end
35
+
36
+ content
37
+ end
38
+
39
+ def coverage_data
40
+ if @coverage_data.nil?
41
+ # set up coverage data structure
42
+ @coverage_data = {}
43
+ coverable_files.each {|file| @coverage_data[file] = {} }
44
+ end
45
+ @coverage_data
46
+ end
47
+
48
+ private
49
+
50
+ def create_parser
51
+ context = JCov::Context::InstrumentationContext.new(coverage_data)
52
+ @parser = context.create
53
+
54
+ @parser.load(File.expand_path('../js/parser.js', __FILE__))
55
+ @parser.load(File.expand_path('../../../vendor/acorn.js', __FILE__))
56
+ @parser.load(File.expand_path('../../../vendor/walk.js', __FILE__))
57
+ end
58
+
59
+ def instrument_script(content, filename)
60
+ lines = coverage_data[filename] || {}
61
+ line_number = 0
62
+ output = ""
63
+
64
+ StringIO.new(content).each_line do |line|
65
+ line_number += 1
66
+ if !lines[line_number].nil? # nil values are ones set to be ignored by ignoreLine
67
+ column = lines[line_number]
68
+ output << line[0...column]
69
+ output << "_coverage_tick('#{filename}', #{line_number});"
70
+ output << line[column..-1]
71
+ lines[line_number] = 0 # reset to zero for counting
72
+ else
73
+ output << line
74
+ end
75
+ end
76
+
77
+ puts output if @options[:dump]
78
+
79
+ output
80
+ end
81
+
82
+ def calculate_coverage_data(filename, content)
83
+ @parser['filename'] = filename
84
+ @parser['code'] = content
85
+ @parser.eval("JCov.calculateCoverageData(code, filename);")
86
+ rescue V8::Error => e
87
+ message = e.message.split(' at').first
88
+ raise JCov::ParseError.new("#{message} in #{filename}")
89
+ end
90
+ end
91
+
92
+ end
@@ -15,7 +15,7 @@ module JCov::Reporter
15
15
  # report a warning message if we're not checking any files for coverage
16
16
  puts "No files were checked for coverage. Maybe your ignore list in #{config.filename} is too inclusive?"
17
17
  elsif report_coverage?
18
- report_file_coverage if options.report
18
+ report_file_coverage if config.report
19
19
  report_total_coverage
20
20
  report_threshold_errors if config.threshold
21
21
  passed?
@@ -30,13 +30,9 @@ module JCov::Reporter
30
30
  @coverage_runner.config
31
31
  end
32
32
 
33
- def options
34
- @coverage_runner.options
35
- end
36
-
37
33
  # no_coverage.nil? because no_coverage gets set to false when set because it's prefixed as 'no'
38
34
  def report_coverage?
39
- options.report || options.no_coverage.nil? && options.coverage && options.test.nil? && options.args.empty?
35
+ config.report || config.no_coverage.nil? && config.coverage && config.test.nil? && config.args.empty?
40
36
  end
41
37
 
42
38
  # passes if any are true:
@@ -80,7 +76,7 @@ module JCov::Reporter
80
76
  percent = 100
81
77
  coverage_string = "(EMPTY)"
82
78
  end
83
- if options.test.nil? || percent > 0 # only show ran files if we're doing a focused test
79
+ if config.test.nil? || percent > 0 # only show ran files if we're doing a focused test
84
80
  printf "%-#{filename_length}s %-10s %3s%\n", file, coverage_string, percent
85
81
  end
86
82
  end
@@ -7,7 +7,7 @@ module JCov::Reporter
7
7
 
8
8
  class HTMLReporter
9
9
 
10
- def initialize coverage_runner
10
+ def initialize(coverage_runner)
11
11
  @coverage_runner = coverage_runner
12
12
 
13
13
  @report_index_template = File.read File.expand_path('../report.html.erb', __FILE__)
@@ -1,27 +1,23 @@
1
1
  module JCov
2
2
 
3
3
  class Runner
4
- attr_reader :context
5
4
  attr_reader :config
6
- attr_reader :options
5
+ attr_reader :coverage
7
6
 
8
- def initialize(config, options)
7
+ def initialize(config)
9
8
  @config = config
10
- @options = options
11
-
12
- setup_context
13
9
  end
14
10
 
11
+ # which tests shall we run?
15
12
  def tests
16
- # which tests shall we run?
17
13
  if @tests.nil?
18
- if options.args.size > 0
19
- @tests = options.args
14
+ if config.args.size > 0
15
+ @tests = config.args
20
16
  else
21
17
  @tests = Dir.glob(File.join(config.test_directory, "**", "*.js"))
22
18
  end
23
- if options.test # limit to a single or group of tests
24
- @tests = @tests.grep(/#{options.test}/)
19
+ if config.test # limit to a single or group of tests
20
+ @tests = @tests.grep(/#{config.test}/)
25
21
  end
26
22
  # remove the runner if it's in there
27
23
  @tests.delete(config.test_runner)
@@ -30,69 +26,33 @@ module JCov
30
26
  @tests
31
27
  end
32
28
 
33
- def print s
34
- Kernel.print s
35
- end
36
-
37
- def println *s
38
- Kernel.puts s
39
- end
40
-
41
- def load file
42
- content = File.read(file)
43
-
44
- context.eval(content, file)
45
- end
46
-
47
- # for JSpec
48
- def readFile file
49
- File.read(file)
50
- end
51
-
52
- # so we can do our dotted line output
53
- def putc char
54
- $stdout.putc(char);
55
- $stdout.flush();
56
- end
57
-
58
29
  def failure_count
59
- context.eval(config.error_field)
30
+ @context.eval(config.error_field)
60
31
  end
61
32
 
62
33
  def run
63
- context.load(config.test_runner)
34
+ setup
35
+
36
+ @context.load(config.test_runner)
64
37
 
65
38
  failure_count
66
39
  end
67
40
 
68
41
  private
69
42
 
70
- def setup_context
71
- # create a V8 context for this runner
72
- @context = V8::Context.new
73
-
74
- # create the jcov context object
75
- @context["JCov"] = {}
76
-
77
- setup_object_proxies
78
- setup_method_proxies
79
- end
43
+ def setup
44
+ # for coverage reporting
45
+ @coverage = JCov::Coverage.new(config)
80
46
 
81
- def setup_object_proxies
82
- context['JCov']['tests'] = tests
83
- context['JCov']['config'] = config
84
- context['JCov']['options'] = options.__hash__
85
- end
47
+ # v8 context for running
48
+ run_context = JCov::Context::RunContext.new(@coverage.loader)
49
+ @context = run_context.create
86
50
 
87
- def setup_method_proxies
88
- # these will become global methods in the context
89
- %w{print
90
- println
91
- load
92
- readFile
93
- putc}.each do |method|
94
- context[method] = self.method(method)
95
- end
51
+ # create the jcov context object
52
+ @context['JCov'] = {}
53
+ @context['JCov']['tests'] = tests
54
+ @context['JCov']['config'] = config
55
+ @context['JCov']['options'] = config
96
56
  end
97
57
 
98
58
  end
@@ -1,3 +1,3 @@
1
1
  module JCov
2
- VERSION = "1.0.1"
2
+ VERSION = "1.1"
3
3
  end
@@ -0,0 +1,1711 @@
1
+ // Acorn is a tiny, fast JavaScript parser written in JavaScript.
2
+ //
3
+ // Acorn was written by Marijn Haverbeke and released under an MIT
4
+ // license. The Unicode regexps (for identifiers and whitespace) were
5
+ // taken from [Esprima](http://esprima.org) by Ariya Hidayat.
6
+ //
7
+ // Git repositories for Acorn are available at
8
+ //
9
+ // http://marijnhaverbeke.nl/git/acorn
10
+ // https://github.com/marijnh/acorn.git
11
+ //
12
+ // Please use the [github bug tracker][ghbt] to report issues.
13
+ //
14
+ // [ghbt]: https://github.com/marijnh/acorn/issues
15
+ //
16
+ // This file defines the main parser interface. The library also comes
17
+ // with a [error-tolerant parser][dammit] and an
18
+ // [abstract syntax tree walker][walk], defined in other files.
19
+ //
20
+ // [dammit]: acorn_loose.js
21
+ // [walk]: util/walk.js
22
+
23
+ (function(exports) {
24
+ "use strict";
25
+
26
+ exports.version = "0.1.01";
27
+
28
+ // The main exported interface (under `self.acorn` when in the
29
+ // browser) is a `parse` function that takes a code string and
30
+ // returns an abstract syntax tree as specified by [Mozilla parser
31
+ // API][api], with the caveat that the SpiderMonkey-specific syntax
32
+ // (`let`, `yield`, inline XML, etc) is not recognized.
33
+ //
34
+ // [api]: https://developer.mozilla.org/en-US/docs/SpiderMonkey/Parser_API
35
+
36
+ var options, input, inputLen, sourceFile;
37
+
38
+ exports.parse = function(inpt, opts) {
39
+ input = String(inpt); inputLen = input.length;
40
+ setOptions(opts);
41
+ initTokenState();
42
+ return parseTopLevel(options.program);
43
+ };
44
+
45
+ // A second optional argument can be given to further configure
46
+ // the parser process. These options are recognized:
47
+
48
+ var defaultOptions = exports.defaultOptions = {
49
+ // `ecmaVersion` indicates the ECMAScript version to parse. Must
50
+ // be either 3 or 5. This
51
+ // influences support for strict mode, the set of reserved words, and
52
+ // support for getters and setter.
53
+ ecmaVersion: 5,
54
+ // Turn on `strictSemicolons` to prevent the parser from doing
55
+ // automatic semicolon insertion.
56
+ strictSemicolons: false,
57
+ // When `allowTrailingCommas` is false, the parser will not allow
58
+ // trailing commas in array and object literals.
59
+ allowTrailingCommas: true,
60
+ // By default, reserved words are not enforced. Enable
61
+ // `forbidReserved` to enforce them.
62
+ forbidReserved: false,
63
+ // When `locations` is on, `loc` properties holding objects with
64
+ // `start` and `end` properties in `{line, column}` form (with
65
+ // line being 1-based and column 0-based) will be attached to the
66
+ // nodes.
67
+ locations: false,
68
+ // A function can be passed as `onComment` option, which will
69
+ // cause Acorn to call that function with `(block, text, start,
70
+ // end)` parameters whenever a comment is skipped. `block` is a
71
+ // boolean indicating whether this is a block (`/* */`) comment,
72
+ // `text` is the content of the comment, and `start` and `end` are
73
+ // character offsets that denote the start and end of the comment.
74
+ // When the `locations` option is on, two more parameters are
75
+ // passed, the full `{line, column}` locations of the start and
76
+ // end of the comments.
77
+ onComment: null,
78
+ // Nodes have their start and end characters offsets recorded in
79
+ // `start` and `end` properties (directly on the node, rather than
80
+ // the `loc` object, which holds line/column data. To also add a
81
+ // [semi-standardized][range] `range` property holding a `[start,
82
+ // end]` array with the same numbers, set the `ranges` option to
83
+ // `true`.
84
+ //
85
+ // [range]: https://bugzilla.mozilla.org/show_bug.cgi?id=745678
86
+ ranges: false,
87
+ // It is possible to parse multiple files into a single AST by
88
+ // passing the tree produced by parsing the first file as
89
+ // `program` option in subsequent parses. This will add the
90
+ // toplevel forms of the parsed file to the `Program` (top) node
91
+ // of an existing parse tree.
92
+ program: null,
93
+ // When `location` is on, you can pass this to record the source
94
+ // file in every node's `loc` object.
95
+ sourceFile: null
96
+ };
97
+
98
+ function setOptions(opts) {
99
+ options = opts || {};
100
+ for (var opt in defaultOptions) if (!options.hasOwnProperty(opt))
101
+ options[opt] = defaultOptions[opt];
102
+ sourceFile = options.sourceFile || null;
103
+ }
104
+
105
+ // The `getLineInfo` function is mostly useful when the
106
+ // `locations` option is off (for performance reasons) and you
107
+ // want to find the line/column position for a given character
108
+ // offset. `input` should be the code string that the offset refers
109
+ // into.
110
+
111
+ var getLineInfo = exports.getLineInfo = function(input, offset) {
112
+ for (var line = 1, cur = 0;;) {
113
+ lineBreak.lastIndex = cur;
114
+ var match = lineBreak.exec(input);
115
+ if (match && match.index < offset) {
116
+ ++line;
117
+ cur = match.index + match[0].length;
118
+ } else break;
119
+ }
120
+ return {line: line, column: offset - cur};
121
+ };
122
+
123
+ // Acorn is organized as a tokenizer and a recursive-descent parser.
124
+ // The `tokenize` export provides an interface to the tokenizer.
125
+ // Because the tokenizer is optimized for being efficiently used by
126
+ // the Acorn parser itself, this interface is somewhat crude and not
127
+ // very modular. Performing another parse or call to `tokenize` will
128
+ // reset the internal state, and invalidate existing tokenizers.
129
+
130
+ exports.tokenize = function(inpt, opts) {
131
+ input = String(inpt); inputLen = input.length;
132
+ setOptions(opts);
133
+ initTokenState();
134
+
135
+ var t = {};
136
+ function getToken(forceRegexp) {
137
+ readToken(forceRegexp);
138
+ t.start = tokStart; t.end = tokEnd;
139
+ t.startLoc = tokStartLoc; t.endLoc = tokEndLoc;
140
+ t.type = tokType; t.value = tokVal;
141
+ return t;
142
+ }
143
+ getToken.jumpTo = function(pos, reAllowed) {
144
+ tokPos = pos;
145
+ if (options.locations) {
146
+ tokCurLine = tokLineStart = lineBreak.lastIndex = 0;
147
+ var match;
148
+ while ((match = lineBreak.exec(input)) && match.index < pos) {
149
+ ++tokCurLine;
150
+ tokLineStart = match.index + match[0].length;
151
+ }
152
+ }
153
+ var ch = input.charAt(pos - 1);
154
+ tokRegexpAllowed = reAllowed;
155
+ skipSpace();
156
+ };
157
+ return getToken;
158
+ };
159
+
160
+ // State is kept in (closure-)global variables. We already saw the
161
+ // `options`, `input`, and `inputLen` variables above.
162
+
163
+ // The current position of the tokenizer in the input.
164
+
165
+ var tokPos;
166
+
167
+ // The start and end offsets of the current token.
168
+
169
+ var tokStart, tokEnd;
170
+
171
+ // When `options.locations` is true, these hold objects
172
+ // containing the tokens start and end line/column pairs.
173
+
174
+ var tokStartLoc, tokEndLoc;
175
+
176
+ // The type and value of the current token. Token types are objects,
177
+ // named by variables against which they can be compared, and
178
+ // holding properties that describe them (indicating, for example,
179
+ // the precedence of an infix operator, and the original name of a
180
+ // keyword token). The kind of value that's held in `tokVal` depends
181
+ // on the type of the token. For literals, it is the literal value,
182
+ // for operators, the operator name, and so on.
183
+
184
+ var tokType, tokVal;
185
+
186
+ // Interal state for the tokenizer. To distinguish between division
187
+ // operators and regular expressions, it remembers whether the last
188
+ // token was one that is allowed to be followed by an expression.
189
+ // (If it is, a slash is probably a regexp, if it isn't it's a
190
+ // division operator. See the `parseStatement` function for a
191
+ // caveat.)
192
+
193
+ var tokRegexpAllowed;
194
+
195
+ // When `options.locations` is true, these are used to keep
196
+ // track of the current line, and know when a new line has been
197
+ // entered.
198
+
199
+ var tokCurLine, tokLineStart;
200
+
201
+ // These store the position of the previous token, which is useful
202
+ // when finishing a node and assigning its `end` position.
203
+
204
+ var lastStart, lastEnd, lastEndLoc;
205
+
206
+ // This is the parser's state. `inFunction` is used to reject
207
+ // `return` statements outside of functions, `labels` to verify that
208
+ // `break` and `continue` have somewhere to jump to, and `strict`
209
+ // indicates whether strict mode is on.
210
+
211
+ var inFunction, labels, strict;
212
+
213
+ // This function is used to raise exceptions on parse errors. It
214
+ // takes an offset integer (into the current `input`) to indicate
215
+ // the location of the error, attaches the position to the end
216
+ // of the error message, and then raises a `SyntaxError` with that
217
+ // message.
218
+
219
+ function raise(pos, message) {
220
+ var loc = getLineInfo(input, pos);
221
+ message += " (" + loc.line + ":" + loc.column + ")";
222
+ var err = new SyntaxError(message);
223
+ err.pos = pos; err.loc = loc; err.raisedAt = tokPos;
224
+ throw err;
225
+ }
226
+
227
+ // ## Token types
228
+
229
+ // The assignment of fine-grained, information-carrying type objects
230
+ // allows the tokenizer to store the information it has about a
231
+ // token in a way that is very cheap for the parser to look up.
232
+
233
+ // All token type variables start with an underscore, to make them
234
+ // easy to recognize.
235
+
236
+ // These are the general types. The `type` property is only used to
237
+ // make them recognizeable when debugging.
238
+
239
+ var _num = {type: "num"}, _regexp = {type: "regexp"}, _string = {type: "string"};
240
+ var _name = {type: "name"}, _eof = {type: "eof"};
241
+
242
+ // Keyword tokens. The `keyword` property (also used in keyword-like
243
+ // operators) indicates that the token originated from an
244
+ // identifier-like word, which is used when parsing property names.
245
+ //
246
+ // The `beforeExpr` property is used to disambiguate between regular
247
+ // expressions and divisions. It is set on all token types that can
248
+ // be followed by an expression (thus, a slash after them would be a
249
+ // regular expression).
250
+ //
251
+ // `isLoop` marks a keyword as starting a loop, which is important
252
+ // to know when parsing a label, in order to allow or disallow
253
+ // continue jumps to that label.
254
+
255
+ var _break = {keyword: "break"}, _case = {keyword: "case", beforeExpr: true}, _catch = {keyword: "catch"};
256
+ var _continue = {keyword: "continue"}, _debugger = {keyword: "debugger"}, _default = {keyword: "default"};
257
+ var _do = {keyword: "do", isLoop: true}, _else = {keyword: "else", beforeExpr: true};
258
+ var _finally = {keyword: "finally"}, _for = {keyword: "for", isLoop: true}, _function = {keyword: "function"};
259
+ var _if = {keyword: "if"}, _return = {keyword: "return", beforeExpr: true}, _switch = {keyword: "switch"};
260
+ var _throw = {keyword: "throw", beforeExpr: true}, _try = {keyword: "try"}, _var = {keyword: "var"};
261
+ var _while = {keyword: "while", isLoop: true}, _with = {keyword: "with"}, _new = {keyword: "new", beforeExpr: true};
262
+ var _this = {keyword: "this"};
263
+
264
+ // The keywords that denote values.
265
+
266
+ var _null = {keyword: "null", atomValue: null}, _true = {keyword: "true", atomValue: true};
267
+ var _false = {keyword: "false", atomValue: false};
268
+
269
+ // Some keywords are treated as regular operators. `in` sometimes
270
+ // (when parsing `for`) needs to be tested against specifically, so
271
+ // we assign a variable name to it for quick comparing.
272
+
273
+ var _in = {keyword: "in", binop: 7, beforeExpr: true};
274
+
275
+ // Map keyword names to token types.
276
+
277
+ var keywordTypes = {"break": _break, "case": _case, "catch": _catch,
278
+ "continue": _continue, "debugger": _debugger, "default": _default,
279
+ "do": _do, "else": _else, "finally": _finally, "for": _for,
280
+ "function": _function, "if": _if, "return": _return, "switch": _switch,
281
+ "throw": _throw, "try": _try, "var": _var, "while": _while, "with": _with,
282
+ "null": _null, "true": _true, "false": _false, "new": _new, "in": _in,
283
+ "instanceof": {keyword: "instanceof", binop: 7, beforeExpr: true}, "this": _this,
284
+ "typeof": {keyword: "typeof", prefix: true, beforeExpr: true},
285
+ "void": {keyword: "void", prefix: true, beforeExpr: true},
286
+ "delete": {keyword: "delete", prefix: true, beforeExpr: true}};
287
+
288
+ // Punctuation token types. Again, the `type` property is purely for debugging.
289
+
290
+ var _bracketL = {type: "[", beforeExpr: true}, _bracketR = {type: "]"}, _braceL = {type: "{", beforeExpr: true};
291
+ var _braceR = {type: "}"}, _parenL = {type: "(", beforeExpr: true}, _parenR = {type: ")"};
292
+ var _comma = {type: ",", beforeExpr: true}, _semi = {type: ";", beforeExpr: true};
293
+ var _colon = {type: ":", beforeExpr: true}, _dot = {type: "."}, _question = {type: "?", beforeExpr: true};
294
+
295
+ // Operators. These carry several kinds of properties to help the
296
+ // parser use them properly (the presence of these properties is
297
+ // what categorizes them as operators).
298
+ //
299
+ // `binop`, when present, specifies that this operator is a binary
300
+ // operator, and will refer to its precedence.
301
+ //
302
+ // `prefix` and `postfix` mark the operator as a prefix or postfix
303
+ // unary operator. `isUpdate` specifies that the node produced by
304
+ // the operator should be of type UpdateExpression rather than
305
+ // simply UnaryExpression (`++` and `--`).
306
+ //
307
+ // `isAssign` marks all of `=`, `+=`, `-=` etcetera, which act as
308
+ // binary operators with a very low precedence, that should result
309
+ // in AssignmentExpression nodes.
310
+
311
+ var _slash = {binop: 10, beforeExpr: true}, _eq = {isAssign: true, beforeExpr: true};
312
+ var _assign = {isAssign: true, beforeExpr: true}, _plusmin = {binop: 9, prefix: true, beforeExpr: true};
313
+ var _incdec = {postfix: true, prefix: true, isUpdate: true}, _prefix = {prefix: true, beforeExpr: true};
314
+ var _bin1 = {binop: 1, beforeExpr: true}, _bin2 = {binop: 2, beforeExpr: true};
315
+ var _bin3 = {binop: 3, beforeExpr: true}, _bin4 = {binop: 4, beforeExpr: true};
316
+ var _bin5 = {binop: 5, beforeExpr: true}, _bin6 = {binop: 6, beforeExpr: true};
317
+ var _bin7 = {binop: 7, beforeExpr: true}, _bin8 = {binop: 8, beforeExpr: true};
318
+ var _bin10 = {binop: 10, beforeExpr: true};
319
+
320
+ // Provide access to the token types for external users of the
321
+ // tokenizer.
322
+
323
+ exports.tokTypes = {bracketL: _bracketL, bracketR: _bracketR, braceL: _braceL, braceR: _braceR,
324
+ parenL: _parenL, parenR: _parenR, comma: _comma, semi: _semi, colon: _colon,
325
+ dot: _dot, question: _question, slash: _slash, eq: _eq, name: _name, eof: _eof,
326
+ num: _num, regexp: _regexp, string: _string};
327
+ for (var kw in keywordTypes) exports.tokTypes[kw] = keywordTypes[kw];
328
+
329
+ // This is a trick taken from Esprima. It turns out that, on
330
+ // non-Chrome browsers, to check whether a string is in a set, a
331
+ // predicate containing a big ugly `switch` statement is faster than
332
+ // a regular expression, and on Chrome the two are about on par.
333
+ // This function uses `eval` (non-lexical) to produce such a
334
+ // predicate from a space-separated string of words.
335
+ //
336
+ // It starts by sorting the words by length.
337
+
338
+ function makePredicate(words) {
339
+ words = words.split(" ");
340
+ var f = "", cats = [];
341
+ out: for (var i = 0; i < words.length; ++i) {
342
+ for (var j = 0; j < cats.length; ++j)
343
+ if (cats[j][0].length == words[i].length) {
344
+ cats[j].push(words[i]);
345
+ continue out;
346
+ }
347
+ cats.push([words[i]]);
348
+ }
349
+ function compareTo(arr) {
350
+ if (arr.length == 1) return f += "return str === " + JSON.stringify(arr[0]) + ";";
351
+ f += "switch(str){";
352
+ for (var i = 0; i < arr.length; ++i) f += "case " + JSON.stringify(arr[i]) + ":";
353
+ f += "return true}return false;";
354
+ }
355
+
356
+ // When there are more than three length categories, an outer
357
+ // switch first dispatches on the lengths, to save on comparisons.
358
+
359
+ if (cats.length > 3) {
360
+ cats.sort(function(a, b) {return b.length - a.length;});
361
+ f += "switch(str.length){";
362
+ for (var i = 0; i < cats.length; ++i) {
363
+ var cat = cats[i];
364
+ f += "case " + cat[0].length + ":";
365
+ compareTo(cat);
366
+ }
367
+ f += "}";
368
+
369
+ // Otherwise, simply generate a flat `switch` statement.
370
+
371
+ } else {
372
+ compareTo(words);
373
+ }
374
+ return new Function("str", f);
375
+ }
376
+
377
+ // The ECMAScript 3 reserved word list.
378
+
379
+ var isReservedWord3 = makePredicate("abstract boolean byte char class double enum export extends final float goto implements import int interface long native package private protected public short static super synchronized throws transient volatile");
380
+
381
+ // ECMAScript 5 reserved words.
382
+
383
+ var isReservedWord5 = makePredicate("class enum extends super const export import");
384
+
385
+ // The additional reserved words in strict mode.
386
+
387
+ var isStrictReservedWord = makePredicate("implements interface let package private protected public static yield");
388
+
389
+ // The forbidden variable names in strict mode.
390
+
391
+ var isStrictBadIdWord = makePredicate("eval arguments");
392
+
393
+ // And the keywords.
394
+
395
+ var isKeyword = makePredicate("break case catch continue debugger default do else finally for function if return switch throw try var while with null true false instanceof typeof void delete new in this");
396
+
397
+ // ## Character categories
398
+
399
+ // Big ugly regular expressions that match characters in the
400
+ // whitespace, identifier, and identifier-start categories. These
401
+ // are only applied when a character is found to actually have a
402
+ // code point above 128.
403
+
404
+ var nonASCIIwhitespace = /[\u1680\u180e\u2000-\u200a\u2028\u2029\u202f\u205f\u3000\ufeff]/;
405
+ var nonASCIIidentifierStartChars = "\xaa\xb5\xba\xc0-\xd6\xd8-\xf6\xf8-\u02c1\u02c6-\u02d1\u02e0-\u02e4\u02ec\u02ee\u0370-\u0374\u0376\u0377\u037a-\u037d\u0386\u0388-\u038a\u038c\u038e-\u03a1\u03a3-\u03f5\u03f7-\u0481\u048a-\u0527\u0531-\u0556\u0559\u0561-\u0587\u05d0-\u05ea\u05f0-\u05f2\u0620-\u064a\u066e\u066f\u0671-\u06d3\u06d5\u06e5\u06e6\u06ee\u06ef\u06fa-\u06fc\u06ff\u0710\u0712-\u072f\u074d-\u07a5\u07b1\u07ca-\u07ea\u07f4\u07f5\u07fa\u0800-\u0815\u081a\u0824\u0828\u0840-\u0858\u08a0\u08a2-\u08ac\u0904-\u0939\u093d\u0950\u0958-\u0961\u0971-\u0977\u0979-\u097f\u0985-\u098c\u098f\u0990\u0993-\u09a8\u09aa-\u09b0\u09b2\u09b6-\u09b9\u09bd\u09ce\u09dc\u09dd\u09df-\u09e1\u09f0\u09f1\u0a05-\u0a0a\u0a0f\u0a10\u0a13-\u0a28\u0a2a-\u0a30\u0a32\u0a33\u0a35\u0a36\u0a38\u0a39\u0a59-\u0a5c\u0a5e\u0a72-\u0a74\u0a85-\u0a8d\u0a8f-\u0a91\u0a93-\u0aa8\u0aaa-\u0ab0\u0ab2\u0ab3\u0ab5-\u0ab9\u0abd\u0ad0\u0ae0\u0ae1\u0b05-\u0b0c\u0b0f\u0b10\u0b13-\u0b28\u0b2a-\u0b30\u0b32\u0b33\u0b35-\u0b39\u0b3d\u0b5c\u0b5d\u0b5f-\u0b61\u0b71\u0b83\u0b85-\u0b8a\u0b8e-\u0b90\u0b92-\u0b95\u0b99\u0b9a\u0b9c\u0b9e\u0b9f\u0ba3\u0ba4\u0ba8-\u0baa\u0bae-\u0bb9\u0bd0\u0c05-\u0c0c\u0c0e-\u0c10\u0c12-\u0c28\u0c2a-\u0c33\u0c35-\u0c39\u0c3d\u0c58\u0c59\u0c60\u0c61\u0c85-\u0c8c\u0c8e-\u0c90\u0c92-\u0ca8\u0caa-\u0cb3\u0cb5-\u0cb9\u0cbd\u0cde\u0ce0\u0ce1\u0cf1\u0cf2\u0d05-\u0d0c\u0d0e-\u0d10\u0d12-\u0d3a\u0d3d\u0d4e\u0d60\u0d61\u0d7a-\u0d7f\u0d85-\u0d96\u0d9a-\u0db1\u0db3-\u0dbb\u0dbd\u0dc0-\u0dc6\u0e01-\u0e30\u0e32\u0e33\u0e40-\u0e46\u0e81\u0e82\u0e84\u0e87\u0e88\u0e8a\u0e8d\u0e94-\u0e97\u0e99-\u0e9f\u0ea1-\u0ea3\u0ea5\u0ea7\u0eaa\u0eab\u0ead-\u0eb0\u0eb2\u0eb3\u0ebd\u0ec0-\u0ec4\u0ec6\u0edc-\u0edf\u0f00\u0f40-\u0f47\u0f49-\u0f6c\u0f88-\u0f8c\u1000-\u102a\u103f\u1050-\u1055\u105a-\u105d\u1061\u1065\u1066\u106e-\u1070\u1075-\u1081\u108e\u10a0-\u10c5\u10c7\u10cd\u10d0-\u10fa\u10fc-\u1248\u124a-\u124d\u1250-\u1256\u1258\u125a-\u125d\u1260-\u1288\u128a-\u128d\u1290-\u12b0\u12b2-\u12b5\u12b8-\u12be\u12c0\u12c2-\u12c5\u12c8-\u12d6\u12d8-\u1310\u1312-\u1315\u1318-\u135a\u1380-\u138f\u13a0-\u13f4\u1401-\u166c\u166f-\u167f\u1681-\u169a\u16a0-\u16ea\u16ee-\u16f0\u1700-\u170c\u170e-\u1711\u1720-\u1731\u1740-\u1751\u1760-\u176c\u176e-\u1770\u1780-\u17b3\u17d7\u17dc\u1820-\u1877\u1880-\u18a8\u18aa\u18b0-\u18f5\u1900-\u191c\u1950-\u196d\u1970-\u1974\u1980-\u19ab\u19c1-\u19c7\u1a00-\u1a16\u1a20-\u1a54\u1aa7\u1b05-\u1b33\u1b45-\u1b4b\u1b83-\u1ba0\u1bae\u1baf\u1bba-\u1be5\u1c00-\u1c23\u1c4d-\u1c4f\u1c5a-\u1c7d\u1ce9-\u1cec\u1cee-\u1cf1\u1cf5\u1cf6\u1d00-\u1dbf\u1e00-\u1f15\u1f18-\u1f1d\u1f20-\u1f45\u1f48-\u1f4d\u1f50-\u1f57\u1f59\u1f5b\u1f5d\u1f5f-\u1f7d\u1f80-\u1fb4\u1fb6-\u1fbc\u1fbe\u1fc2-\u1fc4\u1fc6-\u1fcc\u1fd0-\u1fd3\u1fd6-\u1fdb\u1fe0-\u1fec\u1ff2-\u1ff4\u1ff6-\u1ffc\u2071\u207f\u2090-\u209c\u2102\u2107\u210a-\u2113\u2115\u2119-\u211d\u2124\u2126\u2128\u212a-\u212d\u212f-\u2139\u213c-\u213f\u2145-\u2149\u214e\u2160-\u2188\u2c00-\u2c2e\u2c30-\u2c5e\u2c60-\u2ce4\u2ceb-\u2cee\u2cf2\u2cf3\u2d00-\u2d25\u2d27\u2d2d\u2d30-\u2d67\u2d6f\u2d80-\u2d96\u2da0-\u2da6\u2da8-\u2dae\u2db0-\u2db6\u2db8-\u2dbe\u2dc0-\u2dc6\u2dc8-\u2dce\u2dd0-\u2dd6\u2dd8-\u2dde\u2e2f\u3005-\u3007\u3021-\u3029\u3031-\u3035\u3038-\u303c\u3041-\u3096\u309d-\u309f\u30a1-\u30fa\u30fc-\u30ff\u3105-\u312d\u3131-\u318e\u31a0-\u31ba\u31f0-\u31ff\u3400-\u4db5\u4e00-\u9fcc\ua000-\ua48c\ua4d0-\ua4fd\ua500-\ua60c\ua610-\ua61f\ua62a\ua62b\ua640-\ua66e\ua67f-\ua697\ua6a0-\ua6ef\ua717-\ua71f\ua722-\ua788\ua78b-\ua78e\ua790-\ua793\ua7a0-\ua7aa\ua7f8-\ua801\ua803-\ua805\ua807-\ua80a\ua80c-\ua822\ua840-\ua873\ua882-\ua8b3\ua8f2-\ua8f7\ua8fb\ua90a-\ua925\ua930-\ua946\ua960-\ua97c\ua984-\ua9b2\ua9cf\uaa00-\uaa28\uaa40-\uaa42\uaa44-\uaa4b\uaa60-\uaa76\uaa7a\uaa80-\uaaaf\uaab1\uaab5\uaab6\uaab9-\uaabd\uaac0\uaac2\uaadb-\uaadd\uaae0-\uaaea\uaaf2-\uaaf4\uab01-\uab06\uab09-\uab0e\uab11-\uab16\uab20-\uab26\uab28-\uab2e\uabc0-\uabe2\uac00-\ud7a3\ud7b0-\ud7c6\ud7cb-\ud7fb\uf900-\ufa6d\ufa70-\ufad9\ufb00-\ufb06\ufb13-\ufb17\ufb1d\ufb1f-\ufb28\ufb2a-\ufb36\ufb38-\ufb3c\ufb3e\ufb40\ufb41\ufb43\ufb44\ufb46-\ufbb1\ufbd3-\ufd3d\ufd50-\ufd8f\ufd92-\ufdc7\ufdf0-\ufdfb\ufe70-\ufe74\ufe76-\ufefc\uff21-\uff3a\uff41-\uff5a\uff66-\uffbe\uffc2-\uffc7\uffca-\uffcf\uffd2-\uffd7\uffda-\uffdc";
406
+ var nonASCIIidentifierChars = "\u0371-\u0374\u0483-\u0487\u0591-\u05bd\u05bf\u05c1\u05c2\u05c4\u05c5\u05c7\u0610-\u061a\u0620-\u0649\u0672-\u06d3\u06e7-\u06e8\u06fb-\u06fc\u0730-\u074a\u0800-\u0814\u081b-\u0823\u0825-\u0827\u0829-\u082d\u0840-\u0857\u08e4-\u08fe\u0900-\u0903\u093a-\u093c\u093e-\u094f\u0951-\u0957\u0962-\u0963\u0966-\u096f\u0981-\u0983\u09bc\u09be-\u09c4\u09c7\u09c8\u09d7\u09df-\u09e0\u0a01-\u0a03\u0a3c\u0a3e-\u0a42\u0a47\u0a48\u0a4b-\u0a4d\u0a51\u0a66-\u0a71\u0a75\u0a81-\u0a83\u0abc\u0abe-\u0ac5\u0ac7-\u0ac9\u0acb-\u0acd\u0ae2-\u0ae3\u0ae6-\u0aef\u0b01-\u0b03\u0b3c\u0b3e-\u0b44\u0b47\u0b48\u0b4b-\u0b4d\u0b56\u0b57\u0b5f-\u0b60\u0b66-\u0b6f\u0b82\u0bbe-\u0bc2\u0bc6-\u0bc8\u0bca-\u0bcd\u0bd7\u0be6-\u0bef\u0c01-\u0c03\u0c46-\u0c48\u0c4a-\u0c4d\u0c55\u0c56\u0c62-\u0c63\u0c66-\u0c6f\u0c82\u0c83\u0cbc\u0cbe-\u0cc4\u0cc6-\u0cc8\u0cca-\u0ccd\u0cd5\u0cd6\u0ce2-\u0ce3\u0ce6-\u0cef\u0d02\u0d03\u0d46-\u0d48\u0d57\u0d62-\u0d63\u0d66-\u0d6f\u0d82\u0d83\u0dca\u0dcf-\u0dd4\u0dd6\u0dd8-\u0ddf\u0df2\u0df3\u0e34-\u0e3a\u0e40-\u0e45\u0e50-\u0e59\u0eb4-\u0eb9\u0ec8-\u0ecd\u0ed0-\u0ed9\u0f18\u0f19\u0f20-\u0f29\u0f35\u0f37\u0f39\u0f41-\u0f47\u0f71-\u0f84\u0f86-\u0f87\u0f8d-\u0f97\u0f99-\u0fbc\u0fc6\u1000-\u1029\u1040-\u1049\u1067-\u106d\u1071-\u1074\u1082-\u108d\u108f-\u109d\u135d-\u135f\u170e-\u1710\u1720-\u1730\u1740-\u1750\u1772\u1773\u1780-\u17b2\u17dd\u17e0-\u17e9\u180b-\u180d\u1810-\u1819\u1920-\u192b\u1930-\u193b\u1951-\u196d\u19b0-\u19c0\u19c8-\u19c9\u19d0-\u19d9\u1a00-\u1a15\u1a20-\u1a53\u1a60-\u1a7c\u1a7f-\u1a89\u1a90-\u1a99\u1b46-\u1b4b\u1b50-\u1b59\u1b6b-\u1b73\u1bb0-\u1bb9\u1be6-\u1bf3\u1c00-\u1c22\u1c40-\u1c49\u1c5b-\u1c7d\u1cd0-\u1cd2\u1d00-\u1dbe\u1e01-\u1f15\u200c\u200d\u203f\u2040\u2054\u20d0-\u20dc\u20e1\u20e5-\u20f0\u2d81-\u2d96\u2de0-\u2dff\u3021-\u3028\u3099\u309a\ua640-\ua66d\ua674-\ua67d\ua69f\ua6f0-\ua6f1\ua7f8-\ua800\ua806\ua80b\ua823-\ua827\ua880-\ua881\ua8b4-\ua8c4\ua8d0-\ua8d9\ua8f3-\ua8f7\ua900-\ua909\ua926-\ua92d\ua930-\ua945\ua980-\ua983\ua9b3-\ua9c0\uaa00-\uaa27\uaa40-\uaa41\uaa4c-\uaa4d\uaa50-\uaa59\uaa7b\uaae0-\uaae9\uaaf2-\uaaf3\uabc0-\uabe1\uabec\uabed\uabf0-\uabf9\ufb20-\ufb28\ufe00-\ufe0f\ufe20-\ufe26\ufe33\ufe34\ufe4d-\ufe4f\uff10-\uff19\uff3f";
407
+ var nonASCIIidentifierStart = new RegExp("[" + nonASCIIidentifierStartChars + "]");
408
+ var nonASCIIidentifier = new RegExp("[" + nonASCIIidentifierStartChars + nonASCIIidentifierChars + "]");
409
+
410
+ // Whether a single character denotes a newline.
411
+
412
+ var newline = /[\n\r\u2028\u2029]/;
413
+
414
+ // Matches a whole line break (where CRLF is considered a single
415
+ // line break). Used to count lines.
416
+
417
+ var lineBreak = /\r\n|[\n\r\u2028\u2029]/g;
418
+
419
+ // Test whether a given character code starts an identifier.
420
+
421
+ function isIdentifierStart(code) {
422
+ if (code < 65) return code === 36;
423
+ if (code < 91) return true;
424
+ if (code < 97) return code === 95;
425
+ if (code < 123)return true;
426
+ return code >= 0xaa && nonASCIIidentifierStart.test(String.fromCharCode(code));
427
+ }
428
+
429
+ // Test whether a given character is part of an identifier.
430
+
431
+ function isIdentifierChar(code) {
432
+ if (code < 48) return code === 36;
433
+ if (code < 58) return true;
434
+ if (code < 65) return false;
435
+ if (code < 91) return true;
436
+ if (code < 97) return code === 95;
437
+ if (code < 123)return true;
438
+ return code >= 0xaa && nonASCIIidentifier.test(String.fromCharCode(code));
439
+ }
440
+
441
+ // ## Tokenizer
442
+
443
+ // These are used when `options.locations` is on, for the
444
+ // `tokStartLoc` and `tokEndLoc` properties.
445
+
446
+ function line_loc_t() {
447
+ this.line = tokCurLine;
448
+ this.column = tokPos - tokLineStart;
449
+ }
450
+
451
+ // Reset the token state. Used at the start of a parse.
452
+
453
+ function initTokenState() {
454
+ tokCurLine = 1;
455
+ tokPos = tokLineStart = 0;
456
+ tokRegexpAllowed = true;
457
+ skipSpace();
458
+ }
459
+
460
+ // Called at the end of every token. Sets `tokEnd`, `tokVal`, and
461
+ // `tokRegexpAllowed`, and skips the space after the token, so that
462
+ // the next one's `tokStart` will point at the right position.
463
+
464
+ function finishToken(type, val) {
465
+ tokEnd = tokPos;
466
+ if (options.locations) tokEndLoc = new line_loc_t;
467
+ tokType = type;
468
+ skipSpace();
469
+ tokVal = val;
470
+ tokRegexpAllowed = type.beforeExpr;
471
+ }
472
+
473
+ function skipBlockComment() {
474
+ var startLoc = options.onComment && options.locations && new line_loc_t;
475
+ var start = tokPos, end = input.indexOf("*/", tokPos += 2);
476
+ if (end === -1) raise(tokPos - 2, "Unterminated comment");
477
+ tokPos = end + 2;
478
+ if (options.locations) {
479
+ lineBreak.lastIndex = start;
480
+ var match;
481
+ while ((match = lineBreak.exec(input)) && match.index < tokPos) {
482
+ ++tokCurLine;
483
+ tokLineStart = match.index + match[0].length;
484
+ }
485
+ }
486
+ if (options.onComment)
487
+ options.onComment(true, input.slice(start + 2, end), start, tokPos,
488
+ startLoc, options.locations && new line_loc_t);
489
+ }
490
+
491
+ function skipLineComment() {
492
+ var start = tokPos;
493
+ var startLoc = options.onComment && options.locations && new line_loc_t;
494
+ var ch = input.charCodeAt(tokPos+=2);
495
+ while (tokPos < inputLen && ch !== 10 && ch !== 13 && ch !== 8232 && ch !== 8329) {
496
+ ++tokPos;
497
+ ch = input.charCodeAt(tokPos);
498
+ }
499
+ if (options.onComment)
500
+ options.onComment(false, input.slice(start + 2, tokPos), start, tokPos,
501
+ startLoc, options.locations && new line_loc_t);
502
+ }
503
+
504
+ // Called at the start of the parse and after every token. Skips
505
+ // whitespace and comments, and.
506
+
507
+ function skipSpace() {
508
+ while (tokPos < inputLen) {
509
+ var ch = input.charCodeAt(tokPos);
510
+ if (ch === 32) { // ' '
511
+ ++tokPos;
512
+ } else if(ch === 13) {
513
+ ++tokPos;
514
+ var next = input.charCodeAt(tokPos);
515
+ if(next === 10) {
516
+ ++tokPos;
517
+ }
518
+ if(options.locations) {
519
+ ++tokCurLine;
520
+ tokLineStart = tokPos;
521
+ }
522
+ } else if (ch === 10) {
523
+ ++tokPos;
524
+ ++tokCurLine;
525
+ tokLineStart = tokPos;
526
+ } else if(ch < 14 && ch > 8) {
527
+ ++tokPos;
528
+ } else if (ch === 47) { // '/'
529
+ var next = input.charCodeAt(tokPos+1);
530
+ if (next === 42) { // '*'
531
+ skipBlockComment();
532
+ } else if (next === 47) { // '/'
533
+ skipLineComment();
534
+ } else break;
535
+ } else if ((ch < 14 && ch > 8) || ch === 32 || ch === 160) { // ' ', '\xa0'
536
+ ++tokPos;
537
+ } else if (ch >= 5760 && nonASCIIwhitespace.test(String.fromCharCode(ch))) {
538
+ ++tokPos;
539
+ } else {
540
+ break;
541
+ }
542
+ }
543
+ }
544
+
545
+ // ### Token reading
546
+
547
+ // This is the function that is called to fetch the next token. It
548
+ // is somewhat obscure, because it works in character codes rather
549
+ // than characters, and because operator parsing has been inlined
550
+ // into it.
551
+ //
552
+ // All in the name of speed.
553
+ //
554
+ // The `forceRegexp` parameter is used in the one case where the
555
+ // `tokRegexpAllowed` trick does not work. See `parseStatement`.
556
+
557
+ function readToken_dot() {
558
+ var next = input.charCodeAt(tokPos+1);
559
+ if (next >= 48 && next <= 57) return readNumber(true);
560
+ ++tokPos;
561
+ return finishToken(_dot);
562
+ }
563
+
564
+ function readToken_slash() { // '/'
565
+ var next = input.charCodeAt(tokPos+1);
566
+ if (tokRegexpAllowed) {++tokPos; return readRegexp();}
567
+ if (next === 61) return finishOp(_assign, 2);
568
+ return finishOp(_slash, 1);
569
+ }
570
+
571
+ function readToken_mult_modulo() { // '%*'
572
+ var next = input.charCodeAt(tokPos+1);
573
+ if (next === 61) return finishOp(_assign, 2);
574
+ return finishOp(_bin10, 1);
575
+ }
576
+
577
+ function readToken_pipe_amp(code) { // '|&'
578
+ var next = input.charCodeAt(tokPos+1);
579
+ if (next === code) return finishOp(code === 124 ? _bin1 : _bin2, 2);
580
+ if (next === 61) return finishOp(_assign, 2);
581
+ return finishOp(code === 124 ? _bin3 : _bin5, 1);
582
+ }
583
+
584
+ function readToken_caret() { // '^'
585
+ var next = input.charCodeAt(tokPos+1);
586
+ if (next === 61) return finishOp(_assign, 2);
587
+ return finishOp(_bin4, 1);
588
+ }
589
+
590
+ function readToken_plus_min(code) { // '+-'
591
+ var next = input.charCodeAt(tokPos+1);
592
+ if (next === code) return finishOp(_incdec, 2);
593
+ if (next === 61) return finishOp(_assign, 2);
594
+ return finishOp(_plusmin, 1);
595
+ }
596
+
597
+ function readToken_lt_gt(code) { // '<>'
598
+ var next = input.charCodeAt(tokPos+1);
599
+ var size = 1;
600
+ if (next === code) {
601
+ size = code === 62 && input.charCodeAt(tokPos+2) === 62 ? 3 : 2;
602
+ if (input.charCodeAt(tokPos + size) === 61) return finishOp(_assign, size + 1);
603
+ return finishOp(_bin8, size);
604
+ }
605
+ if (next === 61)
606
+ size = input.charCodeAt(tokPos+2) === 61 ? 3 : 2;
607
+ return finishOp(_bin7, size);
608
+ }
609
+
610
+ function readToken_eq_excl(code) { // '=!'
611
+ var next = input.charCodeAt(tokPos+1);
612
+ if (next === 61) return finishOp(_bin6, input.charCodeAt(tokPos+2) === 61 ? 3 : 2);
613
+ return finishOp(code === 61 ? _eq : _prefix, 1);
614
+ }
615
+
616
+ function getTokenFromCode(code) {
617
+ switch(code) {
618
+ // The interpretation of a dot depends on whether it is followed
619
+ // by a digit.
620
+ case 46: // '.'
621
+ return readToken_dot();
622
+
623
+ // Punctuation tokens.
624
+ case 40: ++tokPos; return finishToken(_parenL);
625
+ case 41: ++tokPos; return finishToken(_parenR);
626
+ case 59: ++tokPos; return finishToken(_semi);
627
+ case 44: ++tokPos; return finishToken(_comma);
628
+ case 91: ++tokPos; return finishToken(_bracketL);
629
+ case 93: ++tokPos; return finishToken(_bracketR);
630
+ case 123: ++tokPos; return finishToken(_braceL);
631
+ case 125: ++tokPos; return finishToken(_braceR);
632
+ case 58: ++tokPos; return finishToken(_colon);
633
+ case 63: ++tokPos; return finishToken(_question);
634
+
635
+ // '0x' is a hexadecimal number.
636
+ case 48: // '0'
637
+ var next = input.charCodeAt(tokPos+1);
638
+ if (next === 120 || next === 88) return readHexNumber();
639
+ // Anything else beginning with a digit is an integer, octal
640
+ // number, or float.
641
+ case 49: case 50: case 51: case 52: case 53: case 54: case 55: case 56: case 57: // 1-9
642
+ return readNumber(false);
643
+
644
+ // Quotes produce strings.
645
+ case 34: case 39: // '"', "'"
646
+ return readString(code);
647
+
648
+ // Operators are parsed inline in tiny state machines. '=' (61) is
649
+ // often referred to. `finishOp` simply skips the amount of
650
+ // characters it is given as second argument, and returns a token
651
+ // of the type given by its first argument.
652
+
653
+ case 47: // '/'
654
+ return readToken_slash(code);
655
+
656
+ case 37: case 42: // '%*'
657
+ return readToken_mult_modulo();
658
+
659
+ case 124: case 38: // '|&'
660
+ return readToken_pipe_amp(code);
661
+
662
+ case 94: // '^'
663
+ return readToken_caret();
664
+
665
+ case 43: case 45: // '+-'
666
+ return readToken_plus_min(code);
667
+
668
+ case 60: case 62: // '<>'
669
+ return readToken_lt_gt(code);
670
+
671
+ case 61: case 33: // '=!'
672
+ return readToken_eq_excl(code);
673
+
674
+ case 126: // '~'
675
+ return finishOp(_prefix, 1);
676
+ }
677
+
678
+ return false;
679
+ }
680
+
681
+ function readToken(forceRegexp) {
682
+ if (!forceRegexp) tokStart = tokPos;
683
+ else tokPos = tokStart + 1;
684
+ if (options.locations) tokStartLoc = new line_loc_t;
685
+ if (forceRegexp) return readRegexp();
686
+ if (tokPos >= inputLen) return finishToken(_eof);
687
+
688
+ var code = input.charCodeAt(tokPos);
689
+ // Identifier or keyword. '\uXXXX' sequences are allowed in
690
+ // identifiers, so '\' also dispatches to that.
691
+ if (isIdentifierStart(code) || code === 92 /* '\' */) return readWord();
692
+
693
+ var tok = getTokenFromCode(code);
694
+
695
+ if (tok === false) {
696
+ // If we are here, we either found a non-ASCII identifier
697
+ // character, or something that's entirely disallowed.
698
+ var ch = String.fromCharCode(code);
699
+ if (ch === "\\" || nonASCIIidentifierStart.test(ch)) return readWord();
700
+ raise(tokPos, "Unexpected character '" + ch + "'");
701
+ }
702
+ return tok;
703
+ }
704
+
705
+ function finishOp(type, size) {
706
+ var str = input.slice(tokPos, tokPos + size);
707
+ tokPos += size;
708
+ finishToken(type, str);
709
+ }
710
+
711
+ // Parse a regular expression. Some context-awareness is necessary,
712
+ // since a '/' inside a '[]' set does not end the expression.
713
+
714
+ function readRegexp() {
715
+ var content = "", escaped, inClass, start = tokPos;
716
+ for (;;) {
717
+ if (tokPos >= inputLen) raise(start, "Unterminated regular expression");
718
+ var ch = input.charAt(tokPos);
719
+ if (newline.test(ch)) raise(start, "Unterminated regular expression");
720
+ if (!escaped) {
721
+ if (ch === "[") inClass = true;
722
+ else if (ch === "]" && inClass) inClass = false;
723
+ else if (ch === "/" && !inClass) break;
724
+ escaped = ch === "\\";
725
+ } else escaped = false;
726
+ ++tokPos;
727
+ }
728
+ var content = input.slice(start, tokPos);
729
+ ++tokPos;
730
+ // Need to use `readWord1` because '\uXXXX' sequences are allowed
731
+ // here (don't ask).
732
+ var mods = readWord1();
733
+ if (mods && !/^[gmsiy]*$/.test(mods)) raise(start, "Invalid regexp flag");
734
+ return finishToken(_regexp, new RegExp(content, mods));
735
+ }
736
+
737
+ // Read an integer in the given radix. Return null if zero digits
738
+ // were read, the integer value otherwise. When `len` is given, this
739
+ // will return `null` unless the integer has exactly `len` digits.
740
+
741
+ function readInt(radix, len) {
742
+ var start = tokPos, total = 0;
743
+ for (var i = 0, e = len == null ? Infinity : len; i < e; ++i) {
744
+ var code = input.charCodeAt(tokPos), val;
745
+ if (code >= 97) val = code - 97 + 10; // a
746
+ else if (code >= 65) val = code - 65 + 10; // A
747
+ else if (code >= 48 && code <= 57) val = code - 48; // 0-9
748
+ else val = Infinity;
749
+ if (val >= radix) break;
750
+ ++tokPos;
751
+ total = total * radix + val;
752
+ }
753
+ if (tokPos === start || len != null && tokPos - start !== len) return null;
754
+
755
+ return total;
756
+ }
757
+
758
+ function readHexNumber() {
759
+ tokPos += 2; // 0x
760
+ var val = readInt(16);
761
+ if (val == null) raise(tokStart + 2, "Expected hexadecimal number");
762
+ if (isIdentifierStart(input.charCodeAt(tokPos))) raise(tokPos, "Identifier directly after number");
763
+ return finishToken(_num, val);
764
+ }
765
+
766
+ // Read an integer, octal integer, or floating-point number.
767
+
768
+ function readNumber(startsWithDot) {
769
+ var start = tokPos, isFloat = false, octal = input.charCodeAt(tokPos) === 48;
770
+ if (!startsWithDot && readInt(10) === null) raise(start, "Invalid number");
771
+ if (input.charCodeAt(tokPos) === 46) {
772
+ ++tokPos;
773
+ readInt(10);
774
+ isFloat = true;
775
+ }
776
+ var next = input.charCodeAt(tokPos);
777
+ if (next === 69 || next === 101) { // 'eE'
778
+ next = input.charCodeAt(++tokPos);
779
+ if (next === 43 || next === 45) ++tokPos; // '+-'
780
+ if (readInt(10) === null) raise(start, "Invalid number")
781
+ isFloat = true;
782
+ }
783
+ if (isIdentifierStart(input.charCodeAt(tokPos))) raise(tokPos, "Identifier directly after number");
784
+
785
+ var str = input.slice(start, tokPos), val;
786
+ if (isFloat) val = parseFloat(str);
787
+ else if (!octal || str.length === 1) val = parseInt(str, 10);
788
+ else if (/[89]/.test(str) || strict) raise(start, "Invalid number");
789
+ else val = parseInt(str, 8);
790
+ return finishToken(_num, val);
791
+ }
792
+
793
+ // Read a string value, interpreting backslash-escapes.
794
+
795
+ function readString(quote) {
796
+ tokPos++;
797
+ var out = "";
798
+ for (;;) {
799
+ if (tokPos >= inputLen) raise(tokStart, "Unterminated string constant");
800
+ var ch = input.charCodeAt(tokPos);
801
+ if (ch === quote) {
802
+ ++tokPos;
803
+ return finishToken(_string, out);
804
+ }
805
+ if (ch === 92) { // '\'
806
+ ch = input.charCodeAt(++tokPos);
807
+ var octal = /^[0-7]+/.exec(input.slice(tokPos, tokPos + 3));
808
+ if (octal) octal = octal[0];
809
+ while (octal && parseInt(octal, 8) > 255) octal = octal.slice(0, octal.length - 1);
810
+ if (octal === "0") octal = null;
811
+ ++tokPos;
812
+ if (octal) {
813
+ if (strict) raise(tokPos - 2, "Octal literal in strict mode");
814
+ out += String.fromCharCode(parseInt(octal, 8));
815
+ tokPos += octal.length - 1;
816
+ } else {
817
+ switch (ch) {
818
+ case 110: out += "\n"; break; // 'n' -> '\n'
819
+ case 114: out += "\r"; break; // 'r' -> '\r'
820
+ case 120: out += String.fromCharCode(readHexChar(2)); break; // 'x'
821
+ case 117: out += String.fromCharCode(readHexChar(4)); break; // 'u'
822
+ case 85: out += String.fromCharCode(readHexChar(8)); break; // 'U'
823
+ case 116: out += "\t"; break; // 't' -> '\t'
824
+ case 98: out += "\b"; break; // 'b' -> '\b'
825
+ case 118: out += "\v"; break; // 'v' -> '\u000b'
826
+ case 102: out += "\f"; break; // 'f' -> '\f'
827
+ case 48: out += "\0"; break; // 0 -> '\0'
828
+ case 13: if (input.charCodeAt(tokPos) === 10) ++tokPos; // '\r\n'
829
+ case 10: // ' \n'
830
+ if (options.locations) { tokLineStart = tokPos; ++tokCurLine; }
831
+ break;
832
+ default: out += String.fromCharCode(ch); break;
833
+ }
834
+ }
835
+ } else {
836
+ if (ch === 13 || ch === 10 || ch === 8232 || ch === 8329) raise(tokStart, "Unterminated string constant");
837
+ out += String.fromCharCode(ch); // '\'
838
+ ++tokPos;
839
+ }
840
+ }
841
+ }
842
+
843
+ // Used to read character escape sequences ('\x', '\u', '\U').
844
+
845
+ function readHexChar(len) {
846
+ var n = readInt(16, len);
847
+ if (n === null) raise(tokStart, "Bad character escape sequence");
848
+ return n;
849
+ }
850
+
851
+ // Used to signal to callers of `readWord1` whether the word
852
+ // contained any escape sequences. This is needed because words with
853
+ // escape sequences must not be interpreted as keywords.
854
+
855
+ var containsEsc;
856
+
857
+ // Read an identifier, and return it as a string. Sets `containsEsc`
858
+ // to whether the word contained a '\u' escape.
859
+ //
860
+ // Only builds up the word character-by-character when it actually
861
+ // containeds an escape, as a micro-optimization.
862
+
863
+ function readWord1() {
864
+ containsEsc = false;
865
+ var word, first = true, start = tokPos;
866
+ for (;;) {
867
+ var ch = input.charCodeAt(tokPos);
868
+ if (isIdentifierChar(ch)) {
869
+ if (containsEsc) word += input.charAt(tokPos);
870
+ ++tokPos;
871
+ } else if (ch === 92) { // "\"
872
+ if (!containsEsc) word = input.slice(start, tokPos);
873
+ containsEsc = true;
874
+ if (input.charCodeAt(++tokPos) != 117) // "u"
875
+ raise(tokPos, "Expecting Unicode escape sequence \\uXXXX");
876
+ ++tokPos;
877
+ var esc = readHexChar(4);
878
+ var escStr = String.fromCharCode(esc);
879
+ if (!escStr) raise(tokPos - 1, "Invalid Unicode escape");
880
+ if (!(first ? isIdentifierStart(esc) : isIdentifierChar(esc)))
881
+ raise(tokPos - 4, "Invalid Unicode escape");
882
+ word += escStr;
883
+ } else {
884
+ break;
885
+ }
886
+ first = false;
887
+ }
888
+ return containsEsc ? word : input.slice(start, tokPos);
889
+ }
890
+
891
+ // Read an identifier or keyword token. Will check for reserved
892
+ // words when necessary.
893
+
894
+ function readWord() {
895
+ var word = readWord1();
896
+ var type = _name;
897
+ if (!containsEsc) {
898
+ if (isKeyword(word)) type = keywordTypes[word];
899
+ else if (options.forbidReserved &&
900
+ (options.ecmaVersion === 3 ? isReservedWord3 : isReservedWord5)(word) ||
901
+ strict && isStrictReservedWord(word))
902
+ raise(tokStart, "The keyword '" + word + "' is reserved");
903
+ }
904
+ return finishToken(type, word);
905
+ }
906
+
907
+ // ## Parser
908
+
909
+ // A recursive descent parser operates by defining functions for all
910
+ // syntactic elements, and recursively calling those, each function
911
+ // advancing the input stream and returning an AST node. Precedence
912
+ // of constructs (for example, the fact that `!x[1]` means `!(x[1])`
913
+ // instead of `(!x)[1]` is handled by the fact that the parser
914
+ // function that parses unary prefix operators is called first, and
915
+ // in turn calls the function that parses `[]` subscripts — that
916
+ // way, it'll receive the node for `x[1]` already parsed, and wraps
917
+ // *that* in the unary operator node.
918
+ //
919
+ // Acorn uses an [operator precedence parser][opp] to handle binary
920
+ // operator precedence, because it is much more compact than using
921
+ // the technique outlined above, which uses different, nesting
922
+ // functions to specify precedence, for all of the ten binary
923
+ // precedence levels that JavaScript defines.
924
+ //
925
+ // [opp]: http://en.wikipedia.org/wiki/Operator-precedence_parser
926
+
927
+ // ### Parser utilities
928
+
929
+ // Continue to the next token.
930
+
931
+ function next() {
932
+ lastStart = tokStart;
933
+ lastEnd = tokEnd;
934
+ lastEndLoc = tokEndLoc;
935
+ readToken();
936
+ }
937
+
938
+ // Enter strict mode. Re-reads the next token to please pedantic
939
+ // tests ("use strict"; 010; -- should fail).
940
+
941
+ function setStrict(strct) {
942
+ strict = strct;
943
+ tokPos = lastEnd;
944
+ skipSpace();
945
+ readToken();
946
+ }
947
+
948
+ // Start an AST node, attaching a start offset.
949
+
950
+ function node_t() {
951
+ this.type = null;
952
+ this.start = tokStart;
953
+ this.end = null;
954
+ }
955
+
956
+ function node_loc_t() {
957
+ this.start = tokStartLoc;
958
+ this.end = null;
959
+ if (sourceFile !== null) this.source = sourceFile;
960
+ }
961
+
962
+ function startNode() {
963
+ var node = new node_t();
964
+ if (options.locations)
965
+ node.loc = new node_loc_t();
966
+ if (options.ranges)
967
+ node.range = [tokStart, 0];
968
+ return node;
969
+ }
970
+
971
+ // Start a node whose start offset information should be based on
972
+ // the start of another node. For example, a binary operator node is
973
+ // only started after its left-hand side has already been parsed.
974
+
975
+ function startNodeFrom(other) {
976
+ var node = new node_t();
977
+ node.start = other.start;
978
+ if (options.locations) {
979
+ node.loc = new node_loc_t();
980
+ node.loc.start = other.loc.start;
981
+ }
982
+ if (options.ranges)
983
+ node.range = [other.range[0], 0];
984
+
985
+ return node;
986
+ }
987
+
988
+ // Finish an AST node, adding `type` and `end` properties.
989
+
990
+ function finishNode(node, type) {
991
+ node.type = type;
992
+ node.end = lastEnd;
993
+ if (options.locations)
994
+ node.loc.end = lastEndLoc;
995
+ if (options.ranges)
996
+ node.range[1] = lastEnd;
997
+ return node;
998
+ }
999
+
1000
+ // Test whether a statement node is the string literal `"use strict"`.
1001
+
1002
+ function isUseStrict(stmt) {
1003
+ return options.ecmaVersion >= 5 && stmt.type === "ExpressionStatement" &&
1004
+ stmt.expression.type === "Literal" && stmt.expression.value === "use strict";
1005
+ }
1006
+
1007
+ // Predicate that tests whether the next token is of the given
1008
+ // type, and if yes, consumes it as a side effect.
1009
+
1010
+ function eat(type) {
1011
+ if (tokType === type) {
1012
+ next();
1013
+ return true;
1014
+ }
1015
+ }
1016
+
1017
+ // Test whether a semicolon can be inserted at the current position.
1018
+
1019
+ function canInsertSemicolon() {
1020
+ return !options.strictSemicolons &&
1021
+ (tokType === _eof || tokType === _braceR || newline.test(input.slice(lastEnd, tokStart)));
1022
+ }
1023
+
1024
+ // Consume a semicolon, or, failing that, see if we are allowed to
1025
+ // pretend that there is a semicolon at this position.
1026
+
1027
+ function semicolon() {
1028
+ if (!eat(_semi) && !canInsertSemicolon()) unexpected();
1029
+ }
1030
+
1031
+ // Expect a token of a given type. If found, consume it, otherwise,
1032
+ // raise an unexpected token error.
1033
+
1034
+ function expect(type) {
1035
+ if (tokType === type) next();
1036
+ else unexpected();
1037
+ }
1038
+
1039
+ // Raise an unexpected token error.
1040
+
1041
+ function unexpected() {
1042
+ raise(tokStart, "Unexpected token");
1043
+ }
1044
+
1045
+ // Verify that a node is an lval — something that can be assigned
1046
+ // to.
1047
+
1048
+ function checkLVal(expr) {
1049
+ if (expr.type !== "Identifier" && expr.type !== "MemberExpression")
1050
+ raise(expr.start, "Assigning to rvalue");
1051
+ if (strict && expr.type === "Identifier" && isStrictBadIdWord(expr.name))
1052
+ raise(expr.start, "Assigning to " + expr.name + " in strict mode");
1053
+ }
1054
+
1055
+ // ### Statement parsing
1056
+
1057
+ // Parse a program. Initializes the parser, reads any number of
1058
+ // statements, and wraps them in a Program node. Optionally takes a
1059
+ // `program` argument. If present, the statements will be appended
1060
+ // to its body instead of creating a new node.
1061
+
1062
+ function parseTopLevel(program) {
1063
+ lastStart = lastEnd = tokPos;
1064
+ if (options.locations) lastEndLoc = new line_loc_t;
1065
+ inFunction = strict = null;
1066
+ labels = [];
1067
+ readToken();
1068
+
1069
+ var node = program || startNode(), first = true;
1070
+ if (!program) node.body = [];
1071
+ while (tokType !== _eof) {
1072
+ var stmt = parseStatement();
1073
+ node.body.push(stmt);
1074
+ if (first && isUseStrict(stmt)) setStrict(true);
1075
+ first = false;
1076
+ }
1077
+ return finishNode(node, "Program");
1078
+ }
1079
+
1080
+ var loopLabel = {kind: "loop"}, switchLabel = {kind: "switch"};
1081
+
1082
+ // Parse a single statement.
1083
+ //
1084
+ // If expecting a statement and finding a slash operator, parse a
1085
+ // regular expression literal. This is to handle cases like
1086
+ // `if (foo) /blah/.exec(foo);`, where looking at the previous token
1087
+ // does not help.
1088
+
1089
+ function parseStatement() {
1090
+ if (tokType === _slash)
1091
+ readToken(true);
1092
+
1093
+ var starttype = tokType, node = startNode();
1094
+
1095
+ // Most types of statements are recognized by the keyword they
1096
+ // start with. Many are trivial to parse, some require a bit of
1097
+ // complexity.
1098
+
1099
+ switch (starttype) {
1100
+ case _break: case _continue:
1101
+ next();
1102
+ var isBreak = starttype === _break;
1103
+ if (eat(_semi) || canInsertSemicolon()) node.label = null;
1104
+ else if (tokType !== _name) unexpected();
1105
+ else {
1106
+ node.label = parseIdent();
1107
+ semicolon();
1108
+ }
1109
+
1110
+ // Verify that there is an actual destination to break or
1111
+ // continue to.
1112
+ for (var i = 0; i < labels.length; ++i) {
1113
+ var lab = labels[i];
1114
+ if (node.label == null || lab.name === node.label.name) {
1115
+ if (lab.kind != null && (isBreak || lab.kind === "loop")) break;
1116
+ if (node.label && isBreak) break;
1117
+ }
1118
+ }
1119
+ if (i === labels.length) raise(node.start, "Unsyntactic " + starttype.keyword);
1120
+ return finishNode(node, isBreak ? "BreakStatement" : "ContinueStatement");
1121
+
1122
+ case _debugger:
1123
+ next();
1124
+ semicolon();
1125
+ return finishNode(node, "DebuggerStatement");
1126
+
1127
+ case _do:
1128
+ next();
1129
+ labels.push(loopLabel);
1130
+ node.body = parseStatement();
1131
+ labels.pop();
1132
+ expect(_while);
1133
+ node.test = parseParenExpression();
1134
+ semicolon();
1135
+ return finishNode(node, "DoWhileStatement");
1136
+
1137
+ // Disambiguating between a `for` and a `for`/`in` loop is
1138
+ // non-trivial. Basically, we have to parse the init `var`
1139
+ // statement or expression, disallowing the `in` operator (see
1140
+ // the second parameter to `parseExpression`), and then check
1141
+ // whether the next token is `in`. When there is no init part
1142
+ // (semicolon immediately after the opening parenthesis), it is
1143
+ // a regular `for` loop.
1144
+
1145
+ case _for:
1146
+ next();
1147
+ labels.push(loopLabel);
1148
+ expect(_parenL);
1149
+ if (tokType === _semi) return parseFor(node, null);
1150
+ if (tokType === _var) {
1151
+ var init = startNode();
1152
+ next();
1153
+ parseVar(init, true);
1154
+ if (init.declarations.length === 1 && eat(_in))
1155
+ return parseForIn(node, init);
1156
+ return parseFor(node, init);
1157
+ }
1158
+ var init = parseExpression(false, true);
1159
+ if (eat(_in)) {checkLVal(init); return parseForIn(node, init);}
1160
+ return parseFor(node, init);
1161
+
1162
+ case _function:
1163
+ next();
1164
+ return parseFunction(node, true);
1165
+
1166
+ case _if:
1167
+ next();
1168
+ node.test = parseParenExpression();
1169
+ node.consequent = parseStatement();
1170
+ node.alternate = eat(_else) ? parseStatement() : null;
1171
+ return finishNode(node, "IfStatement");
1172
+
1173
+ case _return:
1174
+ if (!inFunction) raise(tokStart, "'return' outside of function");
1175
+ next();
1176
+
1177
+ // In `return` (and `break`/`continue`), the keywords with
1178
+ // optional arguments, we eagerly look for a semicolon or the
1179
+ // possibility to insert one.
1180
+
1181
+ if (eat(_semi) || canInsertSemicolon()) node.argument = null;
1182
+ else { node.argument = parseExpression(); semicolon(); }
1183
+ return finishNode(node, "ReturnStatement");
1184
+
1185
+ case _switch:
1186
+ next();
1187
+ node.discriminant = parseParenExpression();
1188
+ node.cases = [];
1189
+ expect(_braceL);
1190
+ labels.push(switchLabel);
1191
+
1192
+ // Statements under must be grouped (by label) in SwitchCase
1193
+ // nodes. `cur` is used to keep the node that we are currently
1194
+ // adding statements to.
1195
+
1196
+ for (var cur, sawDefault; tokType != _braceR;) {
1197
+ if (tokType === _case || tokType === _default) {
1198
+ var isCase = tokType === _case;
1199
+ if (cur) finishNode(cur, "SwitchCase");
1200
+ node.cases.push(cur = startNode());
1201
+ cur.consequent = [];
1202
+ next();
1203
+ if (isCase) cur.test = parseExpression();
1204
+ else {
1205
+ if (sawDefault) raise(lastStart, "Multiple default clauses"); sawDefault = true;
1206
+ cur.test = null;
1207
+ }
1208
+ expect(_colon);
1209
+ } else {
1210
+ if (!cur) unexpected();
1211
+ cur.consequent.push(parseStatement());
1212
+ }
1213
+ }
1214
+ if (cur) finishNode(cur, "SwitchCase");
1215
+ next(); // Closing brace
1216
+ labels.pop();
1217
+ return finishNode(node, "SwitchStatement");
1218
+
1219
+ case _throw:
1220
+ next();
1221
+ if (newline.test(input.slice(lastEnd, tokStart)))
1222
+ raise(lastEnd, "Illegal newline after throw");
1223
+ node.argument = parseExpression();
1224
+ semicolon();
1225
+ return finishNode(node, "ThrowStatement");
1226
+
1227
+ case _try:
1228
+ next();
1229
+ node.block = parseBlock();
1230
+ node.handlers = [];
1231
+ while (tokType === _catch) {
1232
+ var clause = startNode();
1233
+ next();
1234
+ expect(_parenL);
1235
+ clause.param = parseIdent();
1236
+ if (strict && isStrictBadIdWord(clause.param.name))
1237
+ raise(clause.param.start, "Binding " + clause.param.name + " in strict mode");
1238
+ expect(_parenR);
1239
+ clause.guard = null;
1240
+ clause.body = parseBlock();
1241
+ node.handlers.push(finishNode(clause, "CatchClause"));
1242
+ }
1243
+ node.finalizer = eat(_finally) ? parseBlock() : null;
1244
+ if (!node.handlers.length && !node.finalizer)
1245
+ raise(node.start, "Missing catch or finally clause");
1246
+ return finishNode(node, "TryStatement");
1247
+
1248
+ case _var:
1249
+ next();
1250
+ node = parseVar(node);
1251
+ semicolon();
1252
+ return node;
1253
+
1254
+ case _while:
1255
+ next();
1256
+ node.test = parseParenExpression();
1257
+ labels.push(loopLabel);
1258
+ node.body = parseStatement();
1259
+ labels.pop();
1260
+ return finishNode(node, "WhileStatement");
1261
+
1262
+ case _with:
1263
+ if (strict) raise(tokStart, "'with' in strict mode");
1264
+ next();
1265
+ node.object = parseParenExpression();
1266
+ node.body = parseStatement();
1267
+ return finishNode(node, "WithStatement");
1268
+
1269
+ case _braceL:
1270
+ return parseBlock();
1271
+
1272
+ case _semi:
1273
+ next();
1274
+ return finishNode(node, "EmptyStatement");
1275
+
1276
+ // If the statement does not start with a statement keyword or a
1277
+ // brace, it's an ExpressionStatement or LabeledStatement. We
1278
+ // simply start parsing an expression, and afterwards, if the
1279
+ // next token is a colon and the expression was a simple
1280
+ // Identifier node, we switch to interpreting it as a label.
1281
+
1282
+ default:
1283
+ var maybeName = tokVal, expr = parseExpression();
1284
+ if (starttype === _name && expr.type === "Identifier" && eat(_colon)) {
1285
+ for (var i = 0; i < labels.length; ++i)
1286
+ if (labels[i].name === maybeName) raise(expr.start, "Label '" + maybeName + "' is already declared");
1287
+ var kind = tokType.isLoop ? "loop" : tokType === _switch ? "switch" : null;
1288
+ labels.push({name: maybeName, kind: kind});
1289
+ node.body = parseStatement();
1290
+ labels.pop();
1291
+ node.label = expr;
1292
+ return finishNode(node, "LabeledStatement");
1293
+ } else {
1294
+ node.expression = expr;
1295
+ semicolon();
1296
+ return finishNode(node, "ExpressionStatement");
1297
+ }
1298
+ }
1299
+ }
1300
+
1301
+ // Used for constructs like `switch` and `if` that insist on
1302
+ // parentheses around their expression.
1303
+
1304
+ function parseParenExpression() {
1305
+ expect(_parenL);
1306
+ var val = parseExpression();
1307
+ expect(_parenR);
1308
+ return val;
1309
+ }
1310
+
1311
+ // Parse a semicolon-enclosed block of statements, handling `"use
1312
+ // strict"` declarations when `allowStrict` is true (used for
1313
+ // function bodies).
1314
+
1315
+ function parseBlock(allowStrict) {
1316
+ var node = startNode(), first = true, strict = false, oldStrict;
1317
+ node.body = [];
1318
+ expect(_braceL);
1319
+ while (!eat(_braceR)) {
1320
+ var stmt = parseStatement();
1321
+ node.body.push(stmt);
1322
+ if (first && isUseStrict(stmt)) {
1323
+ oldStrict = strict;
1324
+ setStrict(strict = true);
1325
+ }
1326
+ first = false
1327
+ }
1328
+ if (strict && !oldStrict) setStrict(false);
1329
+ return finishNode(node, "BlockStatement");
1330
+ }
1331
+
1332
+ // Parse a regular `for` loop. The disambiguation code in
1333
+ // `parseStatement` will already have parsed the init statement or
1334
+ // expression.
1335
+
1336
+ function parseFor(node, init) {
1337
+ node.init = init;
1338
+ expect(_semi);
1339
+ node.test = tokType === _semi ? null : parseExpression();
1340
+ expect(_semi);
1341
+ node.update = tokType === _parenR ? null : parseExpression();
1342
+ expect(_parenR);
1343
+ node.body = parseStatement();
1344
+ labels.pop();
1345
+ return finishNode(node, "ForStatement");
1346
+ }
1347
+
1348
+ // Parse a `for`/`in` loop.
1349
+
1350
+ function parseForIn(node, init) {
1351
+ node.left = init;
1352
+ node.right = parseExpression();
1353
+ expect(_parenR);
1354
+ node.body = parseStatement();
1355
+ labels.pop();
1356
+ return finishNode(node, "ForInStatement");
1357
+ }
1358
+
1359
+ // Parse a list of variable declarations.
1360
+
1361
+ function parseVar(node, noIn) {
1362
+ node.declarations = [];
1363
+ node.kind = "var";
1364
+ for (;;) {
1365
+ var decl = startNode();
1366
+ decl.id = parseIdent();
1367
+ if (strict && isStrictBadIdWord(decl.id.name))
1368
+ raise(decl.id.start, "Binding " + decl.id.name + " in strict mode");
1369
+ decl.init = eat(_eq) ? parseExpression(true, noIn) : null;
1370
+ node.declarations.push(finishNode(decl, "VariableDeclarator"));
1371
+ if (!eat(_comma)) break;
1372
+ }
1373
+ return finishNode(node, "VariableDeclaration");
1374
+ }
1375
+
1376
+ // ### Expression parsing
1377
+
1378
+ // These nest, from the most general expression type at the top to
1379
+ // 'atomic', nondivisible expression types at the bottom. Most of
1380
+ // the functions will simply let the function(s) below them parse,
1381
+ // and, *if* the syntactic construct they handle is present, wrap
1382
+ // the AST node that the inner parser gave them in another node.
1383
+
1384
+ // Parse a full expression. The arguments are used to forbid comma
1385
+ // sequences (in argument lists, array literals, or object literals)
1386
+ // or the `in` operator (in for loops initalization expressions).
1387
+
1388
+ function parseExpression(noComma, noIn) {
1389
+ var expr = parseMaybeAssign(noIn);
1390
+ if (!noComma && tokType === _comma) {
1391
+ var node = startNodeFrom(expr);
1392
+ node.expressions = [expr];
1393
+ while (eat(_comma)) node.expressions.push(parseMaybeAssign(noIn));
1394
+ return finishNode(node, "SequenceExpression");
1395
+ }
1396
+ return expr;
1397
+ }
1398
+
1399
+ // Parse an assignment expression. This includes applications of
1400
+ // operators like `+=`.
1401
+
1402
+ function parseMaybeAssign(noIn) {
1403
+ var left = parseMaybeConditional(noIn);
1404
+ if (tokType.isAssign) {
1405
+ var node = startNodeFrom(left);
1406
+ node.operator = tokVal;
1407
+ node.left = left;
1408
+ next();
1409
+ node.right = parseMaybeAssign(noIn);
1410
+ checkLVal(left);
1411
+ return finishNode(node, "AssignmentExpression");
1412
+ }
1413
+ return left;
1414
+ }
1415
+
1416
+ // Parse a ternary conditional (`?:`) operator.
1417
+
1418
+ function parseMaybeConditional(noIn) {
1419
+ var expr = parseExprOps(noIn);
1420
+ if (eat(_question)) {
1421
+ var node = startNodeFrom(expr);
1422
+ node.test = expr;
1423
+ node.consequent = parseExpression(true);
1424
+ expect(_colon);
1425
+ node.alternate = parseExpression(true, noIn);
1426
+ return finishNode(node, "ConditionalExpression");
1427
+ }
1428
+ return expr;
1429
+ }
1430
+
1431
+ // Start the precedence parser.
1432
+
1433
+ function parseExprOps(noIn) {
1434
+ return parseExprOp(parseMaybeUnary(noIn), -1, noIn);
1435
+ }
1436
+
1437
+ // Parse binary operators with the operator precedence parsing
1438
+ // algorithm. `left` is the left-hand side of the operator.
1439
+ // `minPrec` provides context that allows the function to stop and
1440
+ // defer further parser to one of its callers when it encounters an
1441
+ // operator that has a lower precedence than the set it is parsing.
1442
+
1443
+ function parseExprOp(left, minPrec, noIn) {
1444
+ var prec = tokType.binop;
1445
+ if (prec != null && (!noIn || tokType !== _in)) {
1446
+ if (prec > minPrec) {
1447
+ var node = startNodeFrom(left);
1448
+ node.left = left;
1449
+ node.operator = tokVal;
1450
+ next();
1451
+ node.right = parseExprOp(parseMaybeUnary(noIn), prec, noIn);
1452
+ var node = finishNode(node, /&&|\|\|/.test(node.operator) ? "LogicalExpression" : "BinaryExpression");
1453
+ return parseExprOp(node, minPrec, noIn);
1454
+ }
1455
+ }
1456
+ return left;
1457
+ }
1458
+
1459
+ // Parse unary operators, both prefix and postfix.
1460
+
1461
+ function parseMaybeUnary(noIn) {
1462
+ if (tokType.prefix) {
1463
+ var node = startNode(), update = tokType.isUpdate;
1464
+ node.operator = tokVal;
1465
+ node.prefix = true;
1466
+ next();
1467
+ node.argument = parseMaybeUnary(noIn);
1468
+ if (update) checkLVal(node.argument);
1469
+ else if (strict && node.operator === "delete" &&
1470
+ node.argument.type === "Identifier")
1471
+ raise(node.start, "Deleting local variable in strict mode");
1472
+ return finishNode(node, update ? "UpdateExpression" : "UnaryExpression");
1473
+ }
1474
+ var expr = parseExprSubscripts();
1475
+ while (tokType.postfix && !canInsertSemicolon()) {
1476
+ var node = startNodeFrom(expr);
1477
+ node.operator = tokVal;
1478
+ node.prefix = false;
1479
+ node.argument = expr;
1480
+ checkLVal(expr);
1481
+ next();
1482
+ expr = finishNode(node, "UpdateExpression");
1483
+ }
1484
+ return expr;
1485
+ }
1486
+
1487
+ // Parse call, dot, and `[]`-subscript expressions.
1488
+
1489
+ function parseExprSubscripts() {
1490
+ return parseSubscripts(parseExprAtom());
1491
+ }
1492
+
1493
+ function parseSubscripts(base, noCalls) {
1494
+ if (eat(_dot)) {
1495
+ var node = startNodeFrom(base);
1496
+ node.object = base;
1497
+ node.property = parseIdent(true);
1498
+ node.computed = false;
1499
+ return parseSubscripts(finishNode(node, "MemberExpression"), noCalls);
1500
+ } else if (eat(_bracketL)) {
1501
+ var node = startNodeFrom(base);
1502
+ node.object = base;
1503
+ node.property = parseExpression();
1504
+ node.computed = true;
1505
+ expect(_bracketR);
1506
+ return parseSubscripts(finishNode(node, "MemberExpression"), noCalls);
1507
+ } else if (!noCalls && eat(_parenL)) {
1508
+ var node = startNodeFrom(base);
1509
+ node.callee = base;
1510
+ node.arguments = parseExprList(_parenR, false);
1511
+ return parseSubscripts(finishNode(node, "CallExpression"), noCalls);
1512
+ } else return base;
1513
+ }
1514
+
1515
+ // Parse an atomic expression — either a single token that is an
1516
+ // expression, an expression started by a keyword like `function` or
1517
+ // `new`, or an expression wrapped in punctuation like `()`, `[]`,
1518
+ // or `{}`.
1519
+
1520
+ function parseExprAtom() {
1521
+ switch (tokType) {
1522
+ case _this:
1523
+ var node = startNode();
1524
+ next();
1525
+ return finishNode(node, "ThisExpression");
1526
+ case _name:
1527
+ return parseIdent();
1528
+ case _num: case _string: case _regexp:
1529
+ var node = startNode();
1530
+ node.value = tokVal;
1531
+ node.raw = input.slice(tokStart, tokEnd);
1532
+ next();
1533
+ return finishNode(node, "Literal");
1534
+
1535
+ case _null: case _true: case _false:
1536
+ var node = startNode();
1537
+ node.value = tokType.atomValue;
1538
+ node.raw = tokType.keyword
1539
+ next();
1540
+ return finishNode(node, "Literal");
1541
+
1542
+ case _parenL:
1543
+ var tokStartLoc1 = tokStartLoc, tokStart1 = tokStart;
1544
+ next();
1545
+ var val = parseExpression();
1546
+ val.start = tokStart1;
1547
+ val.end = tokEnd;
1548
+ if (options.locations) {
1549
+ val.loc.start = tokStartLoc1;
1550
+ val.loc.end = tokEndLoc;
1551
+ }
1552
+ if (options.ranges)
1553
+ val.range = [tokStart1, tokEnd];
1554
+ expect(_parenR);
1555
+ return val;
1556
+
1557
+ case _bracketL:
1558
+ var node = startNode();
1559
+ next();
1560
+ node.elements = parseExprList(_bracketR, true, true);
1561
+ return finishNode(node, "ArrayExpression");
1562
+
1563
+ case _braceL:
1564
+ return parseObj();
1565
+
1566
+ case _function:
1567
+ var node = startNode();
1568
+ next();
1569
+ return parseFunction(node, false);
1570
+
1571
+ case _new:
1572
+ return parseNew();
1573
+
1574
+ default:
1575
+ unexpected();
1576
+ }
1577
+ }
1578
+
1579
+ // New's precedence is slightly tricky. It must allow its argument
1580
+ // to be a `[]` or dot subscript expression, but not a call — at
1581
+ // least, not without wrapping it in parentheses. Thus, it uses the
1582
+
1583
+ function parseNew() {
1584
+ var node = startNode();
1585
+ next();
1586
+ node.callee = parseSubscripts(parseExprAtom(), true);
1587
+ if (eat(_parenL)) node.arguments = parseExprList(_parenR, false);
1588
+ else node.arguments = [];
1589
+ return finishNode(node, "NewExpression");
1590
+ }
1591
+
1592
+ // Parse an object literal.
1593
+
1594
+ function parseObj() {
1595
+ var node = startNode(), first = true, sawGetSet = false;
1596
+ node.properties = [];
1597
+ next();
1598
+ while (!eat(_braceR)) {
1599
+ if (!first) {
1600
+ expect(_comma);
1601
+ if (options.allowTrailingCommas && eat(_braceR)) break;
1602
+ } else first = false;
1603
+
1604
+ var prop = {key: parsePropertyName()}, isGetSet = false, kind;
1605
+ if (eat(_colon)) {
1606
+ prop.value = parseExpression(true);
1607
+ kind = prop.kind = "init";
1608
+ } else if (options.ecmaVersion >= 5 && prop.key.type === "Identifier" &&
1609
+ (prop.key.name === "get" || prop.key.name === "set")) {
1610
+ isGetSet = sawGetSet = true;
1611
+ kind = prop.kind = prop.key.name;
1612
+ prop.key = parsePropertyName();
1613
+ if (tokType !== _parenL) unexpected();
1614
+ prop.value = parseFunction(startNode(), false);
1615
+ } else unexpected();
1616
+
1617
+ // getters and setters are not allowed to clash — either with
1618
+ // each other or with an init property — and in strict mode,
1619
+ // init properties are also not allowed to be repeated.
1620
+
1621
+ if (prop.key.type === "Identifier" && (strict || sawGetSet)) {
1622
+ for (var i = 0; i < node.properties.length; ++i) {
1623
+ var other = node.properties[i];
1624
+ if (other.key.name === prop.key.name) {
1625
+ var conflict = kind == other.kind || isGetSet && other.kind === "init" ||
1626
+ kind === "init" && (other.kind === "get" || other.kind === "set");
1627
+ if (conflict && !strict && kind === "init" && other.kind === "init") conflict = false;
1628
+ if (conflict) raise(prop.key.start, "Redefinition of property");
1629
+ }
1630
+ }
1631
+ }
1632
+ node.properties.push(prop);
1633
+ }
1634
+ return finishNode(node, "ObjectExpression");
1635
+ }
1636
+
1637
+ function parsePropertyName() {
1638
+ if (tokType === _num || tokType === _string) return parseExprAtom();
1639
+ return parseIdent(true);
1640
+ }
1641
+
1642
+ // Parse a function declaration or literal (depending on the
1643
+ // `isStatement` parameter).
1644
+
1645
+ function parseFunction(node, isStatement) {
1646
+ if (tokType === _name) node.id = parseIdent();
1647
+ else if (isStatement) unexpected();
1648
+ else node.id = null;
1649
+ node.params = [];
1650
+ var first = true;
1651
+ expect(_parenL);
1652
+ while (!eat(_parenR)) {
1653
+ if (!first) expect(_comma); else first = false;
1654
+ node.params.push(parseIdent());
1655
+ }
1656
+
1657
+ // Start a new scope with regard to labels and the `inFunction`
1658
+ // flag (restore them to their old value afterwards).
1659
+ var oldInFunc = inFunction, oldLabels = labels;
1660
+ inFunction = true; labels = [];
1661
+ node.body = parseBlock(true);
1662
+ inFunction = oldInFunc; labels = oldLabels;
1663
+
1664
+ // If this is a strict mode function, verify that argument names
1665
+ // are not repeated, and it does not try to bind the words `eval`
1666
+ // or `arguments`.
1667
+ if (strict || node.body.body.length && isUseStrict(node.body.body[0])) {
1668
+ for (var i = node.id ? -1 : 0; i < node.params.length; ++i) {
1669
+ var id = i < 0 ? node.id : node.params[i];
1670
+ if (isStrictReservedWord(id.name) || isStrictBadIdWord(id.name))
1671
+ raise(id.start, "Defining '" + id.name + "' in strict mode");
1672
+ if (i >= 0) for (var j = 0; j < i; ++j) if (id.name === node.params[j].name)
1673
+ raise(id.start, "Argument name clash in strict mode");
1674
+ }
1675
+ }
1676
+
1677
+ return finishNode(node, isStatement ? "FunctionDeclaration" : "FunctionExpression");
1678
+ }
1679
+
1680
+ // Parses a comma-separated list of expressions, and returns them as
1681
+ // an array. `close` is the token type that ends the list, and
1682
+ // `allowEmpty` can be turned on to allow subsequent commas with
1683
+ // nothing in between them to be parsed as `null` (which is needed
1684
+ // for array literals).
1685
+
1686
+ function parseExprList(close, allowTrailingComma, allowEmpty) {
1687
+ var elts = [], first = true;
1688
+ while (!eat(close)) {
1689
+ if (!first) {
1690
+ expect(_comma);
1691
+ if (allowTrailingComma && options.allowTrailingCommas && eat(close)) break;
1692
+ } else first = false;
1693
+
1694
+ if (allowEmpty && tokType === _comma) elts.push(null);
1695
+ else elts.push(parseExpression(true));
1696
+ }
1697
+ return elts;
1698
+ }
1699
+
1700
+ // Parse the next token as an identifier. If `liberal` is true (used
1701
+ // when parsing properties), it will also convert keywords into
1702
+ // identifiers.
1703
+
1704
+ function parseIdent(liberal) {
1705
+ var node = startNode();
1706
+ node.name = tokType === _name ? tokVal : (liberal && !options.forbidReserved && tokType.keyword) || unexpected();
1707
+ next();
1708
+ return finishNode(node, "Identifier");
1709
+ }
1710
+
1711
+ })(typeof exports === "undefined" ? (self.acorn = {}) : exports);