2010-03-22 12:18:58 +01:00
|
|
|
/* JSON parser based on the grammar described at http://json.org/. */
|
|
|
|
|
|
|
|
/* ===== Syntactical Elements ===== */
|
|
|
|
|
2010-05-23 15:56:57 +02:00
|
|
|
start
|
2010-05-31 12:40:24 +02:00
|
|
|
= _ object:object { return object; }
|
2010-03-22 12:18:58 +01:00
|
|
|
|
|
|
|
object
|
2010-05-31 12:40:24 +02:00
|
|
|
= "{" _ "}" _ { return {}; }
|
|
|
|
/ "{" _ members:members "}" _ { return members; }
|
2010-03-22 12:18:58 +01:00
|
|
|
|
2010-05-23 15:56:57 +02:00
|
|
|
members
|
2010-05-31 12:40:24 +02:00
|
|
|
= head:pair tail:("," _ pair)* {
|
2010-05-23 15:56:57 +02:00
|
|
|
var result = {};
|
2013-10-16 08:08:35 +02:00
|
|
|
result[head[0]] = head[1];
|
2010-05-31 12:40:24 +02:00
|
|
|
for (var i = 0; i < tail.length; i++) {
|
2013-10-16 08:08:35 +02:00
|
|
|
result[tail[i][2][0]] = tail[i][2][1];
|
2010-05-23 15:56:57 +02:00
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
2010-03-22 12:18:58 +01:00
|
|
|
|
2010-05-23 15:56:57 +02:00
|
|
|
pair
|
2010-05-31 12:40:24 +02:00
|
|
|
= name:string ":" _ value:value { return [name, value]; }
|
2010-03-22 12:18:58 +01:00
|
|
|
|
|
|
|
array
|
2010-05-31 12:40:24 +02:00
|
|
|
= "[" _ "]" _ { return []; }
|
|
|
|
/ "[" _ elements:elements "]" _ { return elements; }
|
2010-03-22 12:18:58 +01:00
|
|
|
|
2010-05-23 15:56:57 +02:00
|
|
|
elements
|
2010-05-31 12:40:24 +02:00
|
|
|
= head:value tail:("," _ value)* {
|
2013-10-16 08:08:35 +02:00
|
|
|
var result = [head];
|
2010-05-31 12:40:24 +02:00
|
|
|
for (var i = 0; i < tail.length; i++) {
|
2013-10-16 08:08:35 +02:00
|
|
|
result.push(tail[i][2]);
|
2010-05-23 15:56:57 +02:00
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
2010-03-22 12:18:58 +01:00
|
|
|
|
|
|
|
value
|
2010-05-23 10:41:43 +02:00
|
|
|
= string
|
2010-03-22 12:18:58 +01:00
|
|
|
/ number
|
|
|
|
/ object
|
|
|
|
/ array
|
2013-08-20 07:49:44 +02:00
|
|
|
/ "true" _ { return true; }
|
|
|
|
/ "false" _ { return false; }
|
2013-10-16 08:08:35 +02:00
|
|
|
/ "null" _ { return null; }
|
2010-03-22 12:18:58 +01:00
|
|
|
|
|
|
|
/* ===== Lexical Elements ===== */
|
|
|
|
|
|
|
|
string "string"
|
2010-05-31 12:40:24 +02:00
|
|
|
= '"' '"' _ { return ""; }
|
|
|
|
/ '"' chars:chars '"' _ { return chars; }
|
2010-03-22 12:18:58 +01:00
|
|
|
|
2010-05-23 15:56:57 +02:00
|
|
|
chars
|
2010-05-31 12:40:24 +02:00
|
|
|
= chars:char+ { return chars.join(""); }
|
2010-03-22 12:18:58 +01:00
|
|
|
|
|
|
|
char
|
|
|
|
// In the original JSON grammar: "any-Unicode-character-except-"-or-\-or-control-character"
|
2010-05-23 10:41:43 +02:00
|
|
|
= [^"\\\0-\x1F\x7f]
|
2010-03-22 12:18:58 +01:00
|
|
|
/ '\\"' { return '"'; }
|
|
|
|
/ "\\\\" { return "\\"; }
|
|
|
|
/ "\\/" { return "/"; }
|
|
|
|
/ "\\b" { return "\b"; }
|
|
|
|
/ "\\f" { return "\f"; }
|
|
|
|
/ "\\n" { return "\n"; }
|
|
|
|
/ "\\r" { return "\r"; }
|
|
|
|
/ "\\t" { return "\t"; }
|
2012-12-01 16:38:13 +01:00
|
|
|
/ "\\u" digits:$(hexDigit hexDigit hexDigit hexDigit) {
|
|
|
|
return String.fromCharCode(parseInt("0x" + digits));
|
2010-03-22 12:18:58 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
number "number"
|
2012-12-01 16:38:13 +01:00
|
|
|
= parts:$(int frac exp) _ { return parseFloat(parts); }
|
|
|
|
/ parts:$(int frac) _ { return parseFloat(parts); }
|
|
|
|
/ parts:$(int exp) _ { return parseFloat(parts); }
|
|
|
|
/ parts:$(int) _ { return parseFloat(parts); }
|
2010-03-22 12:18:58 +01:00
|
|
|
|
|
|
|
int
|
2012-12-01 16:38:13 +01:00
|
|
|
= digit19 digits
|
|
|
|
/ digit
|
|
|
|
/ "-" digit19 digits
|
|
|
|
/ "-" digit
|
2010-03-22 12:18:58 +01:00
|
|
|
|
2010-05-23 15:56:57 +02:00
|
|
|
frac
|
2012-12-01 16:38:13 +01:00
|
|
|
= "." digits
|
2010-03-22 12:18:58 +01:00
|
|
|
|
2010-05-23 15:56:57 +02:00
|
|
|
exp
|
2012-12-01 16:38:13 +01:00
|
|
|
= e digits
|
2010-03-22 12:18:58 +01:00
|
|
|
|
2010-05-23 15:56:57 +02:00
|
|
|
digits
|
2012-12-01 16:38:13 +01:00
|
|
|
= digit+
|
2010-03-22 12:18:58 +01:00
|
|
|
|
2010-05-23 15:56:57 +02:00
|
|
|
e
|
2012-12-01 16:38:13 +01:00
|
|
|
= [eE] [+-]?
|
2010-03-22 12:18:58 +01:00
|
|
|
|
|
|
|
/*
|
|
|
|
* The following rules are not present in the original JSON gramar, but they are
|
|
|
|
* assumed to exist implicitly.
|
2010-04-19 20:00:22 +02:00
|
|
|
*
|
|
|
|
* FIXME: Define them according to ECMA-262, 5th ed.
|
2010-03-22 12:18:58 +01:00
|
|
|
*/
|
|
|
|
|
2010-05-23 15:56:57 +02:00
|
|
|
digit
|
|
|
|
= [0-9]
|
2010-03-22 12:18:58 +01:00
|
|
|
|
2010-05-23 15:56:57 +02:00
|
|
|
digit19
|
|
|
|
= [1-9]
|
2010-03-22 12:18:58 +01:00
|
|
|
|
2010-05-23 15:56:57 +02:00
|
|
|
hexDigit
|
|
|
|
= [0-9a-fA-F]
|
2010-03-22 12:18:58 +01:00
|
|
|
|
|
|
|
/* ===== Whitespace ===== */
|
|
|
|
|
2010-05-23 15:56:57 +02:00
|
|
|
_ "whitespace"
|
|
|
|
= whitespace*
|
2010-03-22 12:18:58 +01:00
|
|
|
|
|
|
|
// Whitespace is undefined in the original JSON grammar, so I assume a simple
|
2010-04-19 20:00:22 +02:00
|
|
|
// conventional definition consistent with ECMA-262, 5th ed.
|
2010-05-23 15:56:57 +02:00
|
|
|
whitespace
|
|
|
|
= [ \t\n\r]
|