| /*! JSON v3.3.2 | https://bestiejs.github.io/json3 | Copyright 2012-2015, Kit Cambridge, Benjamin Tan | http://kit.mit-license.org */ |
| ;(function () { |
| // Detect the `define` function exposed by asynchronous module loaders. The |
| // strict `define` check is necessary for compatibility with `r.js`. |
| var isLoader = typeof define === "function" && define.amd; |
| |
| // A set of types used to distinguish objects from primitives. |
| var objectTypes = { |
| "function": true, |
| "object": true |
| }; |
| |
| // Detect the `exports` object exposed by CommonJS implementations. |
| var freeExports = objectTypes[typeof exports] && exports && !exports.nodeType && exports; |
| |
| // Use the `global` object exposed by Node (including Browserify via |
| // `insert-module-globals`), Narwhal, and Ringo as the default context, |
| // and the `window` object in browsers. Rhino exports a `global` function |
| // instead. |
| var root = objectTypes[typeof window] && window || this, |
| freeGlobal = freeExports && objectTypes[typeof module] && module && !module.nodeType && typeof global == "object" && global; |
| |
| if (freeGlobal && (freeGlobal.global === freeGlobal || freeGlobal.window === freeGlobal || freeGlobal.self === freeGlobal)) { |
| root = freeGlobal; |
| } |
| |
| // Public: Initializes JSON 3 using the given `context` object, attaching the |
| // `stringify` and `parse` functions to the specified `exports` object. |
| function runInContext(context, exports) { |
| context || (context = root.Object()); |
| exports || (exports = root.Object()); |
| |
| // Native constructor aliases. |
| var Number = context.Number || root.Number, |
| String = context.String || root.String, |
| Object = context.Object || root.Object, |
| Date = context.Date || root.Date, |
| SyntaxError = context.SyntaxError || root.SyntaxError, |
| TypeError = context.TypeError || root.TypeError, |
| Math = context.Math || root.Math, |
| nativeJSON = context.JSON || root.JSON; |
| |
| // Delegate to the native `stringify` and `parse` implementations. |
| if (typeof nativeJSON == "object" && nativeJSON) { |
| exports.stringify = nativeJSON.stringify; |
| exports.parse = nativeJSON.parse; |
| } |
| |
| // Convenience aliases. |
| var objectProto = Object.prototype, |
| getClass = objectProto.toString, |
| isProperty = objectProto.hasOwnProperty, |
| undefined; |
| |
| // Internal: Contains `try...catch` logic used by other functions. |
| // This prevents other functions from being deoptimized. |
| function attempt(func, errorFunc) { |
| try { |
| func(); |
| } catch (exception) { |
| if (errorFunc) { |
| errorFunc(); |
| } |
| } |
| } |
| |
| // Test the `Date#getUTC*` methods. Based on work by @Yaffle. |
| var isExtended = new Date(-3509827334573292); |
| attempt(function () { |
| // The `getUTCFullYear`, `Month`, and `Date` methods return nonsensical |
| // results for certain dates in Opera >= 10.53. |
| isExtended = isExtended.getUTCFullYear() == -109252 && isExtended.getUTCMonth() === 0 && isExtended.getUTCDate() === 1 && |
| isExtended.getUTCHours() == 10 && isExtended.getUTCMinutes() == 37 && isExtended.getUTCSeconds() == 6 && isExtended.getUTCMilliseconds() == 708; |
| }); |
| |
| // Internal: Determines whether the native `JSON.stringify` and `parse` |
| // implementations are spec-compliant. Based on work by Ken Snyder. |
| function has(name) { |
| if (has[name] != null) { |
| // Return cached feature test result. |
| return has[name]; |
| } |
| var isSupported; |
| if (name == "bug-string-char-index") { |
| // IE <= 7 doesn't support accessing string characters using square |
| // bracket notation. IE 8 only supports this for primitives. |
| isSupported = "a"[0] != "a"; |
| } else if (name == "json") { |
| // Indicates whether both `JSON.stringify` and `JSON.parse` are |
| // supported. |
| isSupported = has("json-stringify") && has("date-serialization") && has("json-parse"); |
| } else if (name == "date-serialization") { |
| // Indicates whether `Date`s can be serialized accurately by `JSON.stringify`. |
| isSupported = has("json-stringify") && isExtended; |
| if (isSupported) { |
| var stringify = exports.stringify; |
| attempt(function () { |
| isSupported = |
| // JSON 2, Prototype <= 1.7, and older WebKit builds incorrectly |
| // serialize extended years. |
| stringify(new Date(-8.64e15)) == '"-271821-04-20T00:00:00.000Z"' && |
| // The milliseconds are optional in ES 5, but required in 5.1. |
| stringify(new Date(8.64e15)) == '"+275760-09-13T00:00:00.000Z"' && |
| // Firefox <= 11.0 incorrectly serializes years prior to 0 as negative |
| // four-digit years instead of six-digit years. Credits: @Yaffle. |
| stringify(new Date(-621987552e5)) == '"-000001-01-01T00:00:00.000Z"' && |
| // Safari <= 5.1.5 and Opera >= 10.53 incorrectly serialize millisecond |
| // values less than 1000. Credits: @Yaffle. |
| stringify(new Date(-1)) == '"1969-12-31T23:59:59.999Z"'; |
| }); |
| } |
| } else { |
| var value, serialized = '{"a":[1,true,false,null,"\\u0000\\b\\n\\f\\r\\t"]}'; |
| // Test `JSON.stringify`. |
| if (name == "json-stringify") { |
| var stringify = exports.stringify, stringifySupported = typeof stringify == "function"; |
| if (stringifySupported) { |
| // A test function object with a custom `toJSON` method. |
| (value = function () { |
| return 1; |
| }).toJSON = value; |
| attempt(function () { |
| stringifySupported = |
| // Firefox 3.1b1 and b2 serialize string, number, and boolean |
| // primitives as object literals. |
| stringify(0) === "0" && |
| // FF 3.1b1, b2, and JSON 2 serialize wrapped primitives as object |
| // literals. |
| stringify(new Number()) === "0" && |
| stringify(new String()) == '""' && |
| // FF 3.1b1, 2 throw an error if the value is `null`, `undefined`, or |
| // does not define a canonical JSON representation (this applies to |
| // objects with `toJSON` properties as well, *unless* they are nested |
| // within an object or array). |
| stringify(getClass) === undefined && |
| // IE 8 serializes `undefined` as `"undefined"`. Safari <= 5.1.7 and |
| // FF 3.1b3 pass this test. |
| stringify(undefined) === undefined && |
| // Safari <= 5.1.7 and FF 3.1b3 throw `Error`s and `TypeError`s, |
| // respectively, if the value is omitted entirely. |
| stringify() === undefined && |
| // FF 3.1b1, 2 throw an error if the given value is not a number, |
| // string, array, object, Boolean, or `null` literal. This applies to |
| // objects with custom `toJSON` methods as well, unless they are nested |
| // inside object or array literals. YUI 3.0.0b1 ignores custom `toJSON` |
| // methods entirely. |
| stringify(value) === "1" && |
| stringify([value]) == "[1]" && |
| // Prototype <= 1.6.1 serializes `[undefined]` as `"[]"` instead of |
| // `"[null]"`. |
| stringify([undefined]) == "[null]" && |
| // YUI 3.0.0b1 fails to serialize `null` literals. |
| stringify(null) == "null" && |
| // FF 3.1b1, 2 halts serialization if an array contains a function: |
| // `[1, true, getClass, 1]` serializes as "[1,true,],". FF 3.1b3 |
| // elides non-JSON values from objects and arrays, unless they |
| // define custom `toJSON` methods. |
| stringify([undefined, getClass, null]) == "[null,null,null]" && |
| // Simple serialization test. FF 3.1b1 uses Unicode escape sequences |
| // where character escape codes are expected (e.g., `\b` => `\u0008`). |
| stringify({ "a": [value, true, false, null, "\x00\b\n\f\r\t"] }) == serialized && |
| // FF 3.1b1 and b2 ignore the `filter` and `width` arguments. |
| stringify(null, value) === "1" && |
| stringify([1, 2], null, 1) == "[\n 1,\n 2\n]"; |
| }, function () { |
| stringifySupported = false; |
| }); |
| } |
| isSupported = stringifySupported; |
| } |
| // Test `JSON.parse`. |
| if (name == "json-parse") { |
| var parse = exports.parse, parseSupported; |
| if (typeof parse == "function") { |
| attempt(function () { |
| // FF 3.1b1, b2 will throw an exception if a bare literal is provided. |
| // Conforming implementations should also coerce the initial argument to |
| // a string prior to parsing. |
| if (parse("0") === 0 && !parse(false)) { |
| // Simple parsing test. |
| value = parse(serialized); |
| parseSupported = value["a"].length == 5 && value["a"][0] === 1; |
| if (parseSupported) { |
| attempt(function () { |
| // Safari <= 5.1.2 and FF 3.1b1 allow unescaped tabs in strings. |
| parseSupported = !parse('"\t"'); |
| }); |
| if (parseSupported) { |
| attempt(function () { |
| // FF 4.0 and 4.0.1 allow leading `+` signs and leading |
| // decimal points. FF 4.0, 4.0.1, and IE 9-10 also allow |
| // certain octal literals. |
| parseSupported = parse("01") !== 1; |
| }); |
| } |
| if (parseSupported) { |
| attempt(function () { |
| // FF 4.0, 4.0.1, and Rhino 1.7R3-R4 allow trailing decimal |
| // points. These environments, along with FF 3.1b1 and 2, |
| // also allow trailing commas in JSON objects and arrays. |
| parseSupported = parse("1.") !== 1; |
| }); |
| } |
| } |
| } |
| }, function () { |
| parseSupported = false; |
| }); |
| } |
| isSupported = parseSupported; |
| } |
| } |
| return has[name] = !!isSupported; |
| } |
| has["bug-string-char-index"] = has["date-serialization"] = has["json"] = has["json-stringify"] = has["json-parse"] = null; |
| |
| if (!has("json")) { |
| // Common `[[Class]]` name aliases. |
| var functionClass = "[object Function]", |
| dateClass = "[object Date]", |
| numberClass = "[object Number]", |
| stringClass = "[object String]", |
| arrayClass = "[object Array]", |
| booleanClass = "[object Boolean]"; |
| |
| // Detect incomplete support for accessing string characters by index. |
| var charIndexBuggy = has("bug-string-char-index"); |
| |
| // Internal: Normalizes the `for...in` iteration algorithm across |
| // environments. Each enumerated key is yielded to a `callback` function. |
| var forOwn = function (object, callback) { |
| var size = 0, Properties, dontEnums, property; |
| |
| // Tests for bugs in the current environment's `for...in` algorithm. The |
| // `valueOf` property inherits the non-enumerable flag from |
| // `Object.prototype` in older versions of IE, Netscape, and Mozilla. |
| (Properties = function () { |
| this.valueOf = 0; |
| }).prototype.valueOf = 0; |
| |
| // Iterate over a new instance of the `Properties` class. |
| dontEnums = new Properties(); |
| for (property in dontEnums) { |
| // Ignore all properties inherited from `Object.prototype`. |
| if (isProperty.call(dontEnums, property)) { |
| size++; |
| } |
| } |
| Properties = dontEnums = null; |
| |
| // Normalize the iteration algorithm. |
| if (!size) { |
| // A list of non-enumerable properties inherited from `Object.prototype`. |
| dontEnums = ["valueOf", "toString", "toLocaleString", "propertyIsEnumerable", "isPrototypeOf", "hasOwnProperty", "constructor"]; |
| // IE <= 8, Mozilla 1.0, and Netscape 6.2 ignore shadowed non-enumerable |
| // properties. |
| forOwn = function (object, callback) { |
| var isFunction = getClass.call(object) == functionClass, property, length; |
| var hasProperty = !isFunction && typeof object.constructor != "function" && objectTypes[typeof object.hasOwnProperty] && object.hasOwnProperty || isProperty; |
| for (property in object) { |
| // Gecko <= 1.0 enumerates the `prototype` property of functions under |
| // certain conditions; IE does not. |
| if (!(isFunction && property == "prototype") && hasProperty.call(object, property)) { |
| callback(property); |
| } |
| } |
| // Manually invoke the callback for each non-enumerable property. |
| for (length = dontEnums.length; property = dontEnums[--length];) { |
| if (hasProperty.call(object, property)) { |
| callback(property); |
| } |
| } |
| }; |
| } else { |
| // No bugs detected; use the standard `for...in` algorithm. |
| forOwn = function (object, callback) { |
| var isFunction = getClass.call(object) == functionClass, property, isConstructor; |
| for (property in object) { |
| if (!(isFunction && property == "prototype") && isProperty.call(object, property) && !(isConstructor = property === "constructor")) { |
| callback(property); |
| } |
| } |
| // Manually invoke the callback for the `constructor` property due to |
| // cross-environment inconsistencies. |
| if (isConstructor || isProperty.call(object, (property = "constructor"))) { |
| callback(property); |
| } |
| }; |
| } |
| return forOwn(object, callback); |
| }; |
| |
| // Public: Serializes a JavaScript `value` as a JSON string. The optional |
| // `filter` argument may specify either a function that alters how object and |
| // array members are serialized, or an array of strings and numbers that |
| // indicates which properties should be serialized. The optional `width` |
| // argument may be either a string or number that specifies the indentation |
| // level of the output. |
| if (!has("json-stringify") && !has("date-serialization")) { |
| // Internal: A map of control characters and their escaped equivalents. |
| var Escapes = { |
| 92: "\\\\", |
| 34: '\\"', |
| 8: "\\b", |
| 12: "\\f", |
| 10: "\\n", |
| 13: "\\r", |
| 9: "\\t" |
| }; |
| |
| // Internal: Converts `value` into a zero-padded string such that its |
| // length is at least equal to `width`. The `width` must be <= 6. |
| var leadingZeroes = "000000"; |
| var toPaddedString = function (width, value) { |
| // The `|| 0` expression is necessary to work around a bug in |
| // Opera <= 7.54u2 where `0 == -0`, but `String(-0) !== "0"`. |
| return (leadingZeroes + (value || 0)).slice(-width); |
| }; |
| |
| // Internal: Serializes a date object. |
| var serializeDate = function (value) { |
| var getData, year, month, date, time, hours, minutes, seconds, milliseconds; |
| // Define additional utility methods if the `Date` methods are buggy. |
| if (!isExtended) { |
| var floor = Math.floor; |
| // A mapping between the months of the year and the number of days between |
| // January 1st and the first of the respective month. |
| var Months = [0, 31, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334]; |
| // Internal: Calculates the number of days between the Unix epoch and the |
| // first day of the given month. |
| var getDay = function (year, month) { |
| return Months[month] + 365 * (year - 1970) + floor((year - 1969 + (month = +(month > 1))) / 4) - floor((year - 1901 + month) / 100) + floor((year - 1601 + month) / 400); |
| }; |
| getData = function (value) { |
| // Manually compute the year, month, date, hours, minutes, |
| // seconds, and milliseconds if the `getUTC*` methods are |
| // buggy. Adapted from @Yaffle's `date-shim` project. |
| date = floor(value / 864e5); |
| for (year = floor(date / 365.2425) + 1970 - 1; getDay(year + 1, 0) <= date; year++); |
| for (month = floor((date - getDay(year, 0)) / 30.42); getDay(year, month + 1) <= date; month++); |
| date = 1 + date - getDay(year, month); |
| // The `time` value specifies the time within the day (see ES |
| // 5.1 section 15.9.1.2). The formula `(A % B + B) % B` is used |
| // to compute `A modulo B`, as the `%` operator does not |
| // correspond to the `modulo` operation for negative numbers. |
| time = (value % 864e5 + 864e5) % 864e5; |
| // The hours, minutes, seconds, and milliseconds are obtained by |
| // decomposing the time within the day. See section 15.9.1.10. |
| hours = floor(time / 36e5) % 24; |
| minutes = floor(time / 6e4) % 60; |
| seconds = floor(time / 1e3) % 60; |
| milliseconds = time % 1e3; |
| }; |
| } else { |
| getData = function (value) { |
| year = value.getUTCFullYear(); |
| month = value.getUTCMonth(); |
| date = value.getUTCDate(); |
| hours = value.getUTCHours(); |
| minutes = value.getUTCMinutes(); |
| seconds = value.getUTCSeconds(); |
| milliseconds = value.getUTCMilliseconds(); |
| }; |
| } |
| serializeDate = function (value) { |
| if (value > -1 / 0 && value < 1 / 0) { |
| // Dates are serialized according to the `Date#toJSON` method |
| // specified in ES 5.1 section 15.9.5.44. See section 15.9.1.15 |
| // for the ISO 8601 date time string format. |
| getData(value); |
| // Serialize extended years correctly. |
| value = (year <= 0 || year >= 1e4 ? (year < 0 ? "-" : "+") + toPaddedString(6, year < 0 ? -year : year) : toPaddedString(4, year)) + |
| "-" + toPaddedString(2, month + 1) + "-" + toPaddedString(2, date) + |
| // Months, dates, hours, minutes, and seconds should have two |
| // digits; milliseconds should have three. |
| "T" + toPaddedString(2, hours) + ":" + toPaddedString(2, minutes) + ":" + toPaddedString(2, seconds) + |
| // Milliseconds are optional in ES 5.0, but required in 5.1. |
| "." + toPaddedString(3, milliseconds) + "Z"; |
| year = month = date = hours = minutes = seconds = milliseconds = null; |
| } else { |
| value = null; |
| } |
| return value; |
| }; |
| return serializeDate(value); |
| }; |
| |
| // For environments with `JSON.stringify` but buggy date serialization, |
| // we override the native `Date#toJSON` implementation with a |
| // spec-compliant one. |
| if (has("json-stringify") && !has("date-serialization")) { |
| // Internal: the `Date#toJSON` implementation used to override the native one. |
| function dateToJSON (key) { |
| return serializeDate(this); |
| } |
| |
| // Public: `JSON.stringify`. See ES 5.1 section 15.12.3. |
| var nativeStringify = exports.stringify; |
| exports.stringify = function (source, filter, width) { |
| var nativeToJSON = Date.prototype.toJSON; |
| Date.prototype.toJSON = dateToJSON; |
| var result = nativeStringify(source, filter, width); |
| Date.prototype.toJSON = nativeToJSON; |
| return result; |
| } |
| } else { |
| // Internal: Double-quotes a string `value`, replacing all ASCII control |
| // characters (characters with code unit values between 0 and 31) with |
| // their escaped equivalents. This is an implementation of the |
| // `Quote(value)` operation defined in ES 5.1 section 15.12.3. |
| var unicodePrefix = "\\u00"; |
| var escapeChar = function (character) { |
| var charCode = character.charCodeAt(0), escaped = Escapes[charCode]; |
| if (escaped) { |
| return escaped; |
| } |
| return unicodePrefix + toPaddedString(2, charCode.toString(16)); |
| }; |
| var reEscape = /[\x00-\x1f\x22\x5c]/g; |
| var quote = function (value) { |
| reEscape.lastIndex = 0; |
| return '"' + |
| ( |
| reEscape.test(value) |
| ? value.replace(reEscape, escapeChar) |
| : value |
| ) + |
| '"'; |
| }; |
| |
| // Internal: Recursively serializes an object. Implements the |
| // `Str(key, holder)`, `JO(value)`, and `JA(value)` operations. |
| var serialize = function (property, object, callback, properties, whitespace, indentation, stack) { |
| var value, type, className, results, element, index, length, prefix, result; |
| attempt(function () { |
| // Necessary for host object support. |
| value = object[property]; |
| }); |
| if (typeof value == "object" && value) { |
| if (value.getUTCFullYear && getClass.call(value) == dateClass && value.toJSON === Date.prototype.toJSON) { |
| value = serializeDate(value); |
| } else if (typeof value.toJSON == "function") { |
| value = value.toJSON(property); |
| } |
| } |
| if (callback) { |
| // If a replacement function was provided, call it to obtain the value |
| // for serialization. |
| value = callback.call(object, property, value); |
| } |
| // Exit early if value is `undefined` or `null`. |
| if (value == undefined) { |
| return value === undefined ? value : "null"; |
| } |
| type = typeof value; |
| // Only call `getClass` if the value is an object. |
| if (type == "object") { |
| className = getClass.call(value); |
| } |
| switch (className || type) { |
| case "boolean": |
| case booleanClass: |
| // Booleans are represented literally. |
| return "" + value; |
| case "number": |
| case numberClass: |
| // JSON numbers must be finite. `Infinity` and `NaN` are serialized as |
| // `"null"`. |
| return value > -1 / 0 && value < 1 / 0 ? "" + value : "null"; |
| case "string": |
| case stringClass: |
| // Strings are double-quoted and escaped. |
| return quote("" + value); |
| } |
| // Recursively serialize objects and arrays. |
| if (typeof value == "object") { |
| // Check for cyclic structures. This is a linear search; performance |
| // is inversely proportional to the number of unique nested objects. |
| for (length = stack.length; length--;) { |
| if (stack[length] === value) { |
| // Cyclic structures cannot be serialized by `JSON.stringify`. |
| throw TypeError(); |
| } |
| } |
| // Add the object to the stack of traversed objects. |
| stack.push(value); |
| results = []; |
| // Save the current indentation level and indent one additional level. |
| prefix = indentation; |
| indentation += whitespace; |
| if (className == arrayClass) { |
| // Recursively serialize array elements. |
| for (index = 0, length = value.length; index < length; index++) { |
| element = serialize(index, value, callback, properties, whitespace, indentation, stack); |
| results.push(element === undefined ? "null" : element); |
| } |
| result = results.length ? (whitespace ? "[\n" + indentation + results.join(",\n" + indentation) + "\n" + prefix + "]" : ("[" + results.join(",") + "]")) : "[]"; |
| } else { |
| // Recursively serialize object members. Members are selected from |
| // either a user-specified list of property names, or the object |
| // itself. |
| forOwn(properties || value, function (property) { |
| var element = serialize(property, value, callback, properties, whitespace, indentation, stack); |
| if (element !== undefined) { |
| // According to ES 5.1 section 15.12.3: "If `gap` {whitespace} |
| // is not the empty string, let `member` {quote(property) + ":"} |
| // be the concatenation of `member` and the `space` character." |
| // The "`space` character" refers to the literal space |
| // character, not the `space` {width} argument provided to |
| // `JSON.stringify`. |
| results.push(quote(property) + ":" + (whitespace ? " " : "") + element); |
| } |
| }); |
| result = results.length ? (whitespace ? "{\n" + indentation + results.join(",\n" + indentation) + "\n" + prefix + "}" : ("{" + results.join(",") + "}")) : "{}"; |
| } |
| // Remove the object from the traversed object stack. |
| stack.pop(); |
| return result; |
| } |
| }; |
| |
| // Public: `JSON.stringify`. See ES 5.1 section 15.12.3. |
| exports.stringify = function (source, filter, width) { |
| var whitespace, callback, properties, className; |
| if (objectTypes[typeof filter] && filter) { |
| className = getClass.call(filter); |
| if (className == functionClass) { |
| callback = filter; |
| } else if (className == arrayClass) { |
| // Convert the property names array into a makeshift set. |
| properties = {}; |
| for (var index = 0, length = filter.length, value; index < length;) { |
| value = filter[index++]; |
| className = getClass.call(value); |
| if (className == "[object String]" || className == "[object Number]") { |
| properties[value] = 1; |
| } |
| } |
| } |
| } |
| if (width) { |
| className = getClass.call(width); |
| if (className == numberClass) { |
| // Convert the `width` to an integer and create a string containing |
| // `width` number of space characters. |
| if ((width -= width % 1) > 0) { |
| if (width > 10) { |
| width = 10; |
| } |
| for (whitespace = ""; whitespace.length < width;) { |
| whitespace += " "; |
| } |
| } |
| } else if (className == stringClass) { |
| whitespace = width.length <= 10 ? width : width.slice(0, 10); |
| } |
| } |
| // Opera <= 7.54u2 discards the values associated with empty string keys |
| // (`""`) only if they are used directly within an object member list |
| // (e.g., `!("" in { "": 1})`). |
| return serialize("", (value = {}, value[""] = source, value), callback, properties, whitespace, "", []); |
| }; |
| } |
| } |
| |
| // Public: Parses a JSON source string. |
| if (!has("json-parse")) { |
| var fromCharCode = String.fromCharCode; |
| |
| // Internal: A map of escaped control characters and their unescaped |
| // equivalents. |
| var Unescapes = { |
| 92: "\\", |
| 34: '"', |
| 47: "/", |
| 98: "\b", |
| 116: "\t", |
| 110: "\n", |
| 102: "\f", |
| 114: "\r" |
| }; |
| |
| // Internal: Stores the parser state. |
| var Index, Source; |
| |
| // Internal: Resets the parser state and throws a `SyntaxError`. |
| var abort = function () { |
| Index = Source = null; |
| throw SyntaxError(); |
| }; |
| |
| // Internal: Returns the next token, or `"$"` if the parser has reached |
| // the end of the source string. A token may be a string, number, `null` |
| // literal, or Boolean literal. |
| var lex = function () { |
| var source = Source, length = source.length, value, begin, position, isSigned, charCode; |
| while (Index < length) { |
| charCode = source.charCodeAt(Index); |
| switch (charCode) { |
| case 9: case 10: case 13: case 32: |
| // Skip whitespace tokens, including tabs, carriage returns, line |
| // feeds, and space characters. |
| Index++; |
| break; |
| case 123: case 125: case 91: case 93: case 58: case 44: |
| // Parse a punctuator token (`{`, `}`, `[`, `]`, `:`, or `,`) at |
| // the current position. |
| value = charIndexBuggy ? source.charAt(Index) : source[Index]; |
| Index++; |
| return value; |
| case 34: |
| // `"` delimits a JSON string; advance to the next character and |
| // begin parsing the string. String tokens are prefixed with the |
| // sentinel `@` character to distinguish them from punctuators and |
| // end-of-string tokens. |
| for (value = "@", Index++; Index < length;) { |
| charCode = source.charCodeAt(Index); |
| if (charCode < 32) { |
| // Unescaped ASCII control characters (those with a code unit |
| // less than the space character) are not permitted. |
| abort(); |
| } else if (charCode == 92) { |
| // A reverse solidus (`\`) marks the beginning of an escaped |
| // control character (including `"`, `\`, and `/`) or Unicode |
| // escape sequence. |
| charCode = source.charCodeAt(++Index); |
| switch (charCode) { |
| case 92: case 34: case 47: case 98: case 116: case 110: case 102: case 114: |
| // Revive escaped control characters. |
| value += Unescapes[charCode]; |
| Index++; |
| break; |
| case 117: |
| // `\u` marks the beginning of a Unicode escape sequence. |
| // Advance to the first character and validate the |
| // four-digit code point. |
| begin = ++Index; |
| for (position = Index + 4; Index < position; Index++) { |
| charCode = source.charCodeAt(Index); |
| // A valid sequence comprises four hexdigits (case- |
| // insensitive) that form a single hexadecimal value. |
| if (!(charCode >= 48 && charCode <= 57 || charCode >= 97 && charCode <= 102 || charCode >= 65 && charCode <= 70)) { |
| // Invalid Unicode escape sequence. |
| abort(); |
| } |
| } |
| // Revive the escaped character. |
| value += fromCharCode("0x" + source.slice(begin, Index)); |
| break; |
| default: |
| // Invalid escape sequence. |
| abort(); |
| } |
| } else { |
| if (charCode == 34) { |
| // An unescaped double-quote character marks the end of the |
| // string. |
| break; |
| } |
| charCode = source.charCodeAt(Index); |
| begin = Index; |
| // Optimize for the common case where a string is valid. |
| while (charCode >= 32 && charCode != 92 && charCode != 34) { |
| charCode = source.charCodeAt(++Index); |
| } |
| // Append the string as-is. |
| value += source.slice(begin, Index); |
| } |
| } |
| if (source.charCodeAt(Index) == 34) { |
| // Advance to the next character and return the revived string. |
| Index++; |
| return value; |
| } |
| // Unterminated string. |
| abort(); |
| default: |
| // Parse numbers and literals. |
| begin = Index; |
| // Advance past the negative sign, if one is specified. |
| if (charCode == 45) { |
| isSigned = true; |
| charCode = source.charCodeAt(++Index); |
| } |
| // Parse an integer or floating-point value. |
| if (charCode >= 48 && charCode <= 57) { |
| // Leading zeroes are interpreted as octal literals. |
| if (charCode == 48 && ((charCode = source.charCodeAt(Index + 1)), charCode >= 48 && charCode <= 57)) { |
| // Illegal octal literal. |
| abort(); |
| } |
| isSigned = false; |
| // Parse the integer component. |
| for (; Index < length && ((charCode = source.charCodeAt(Index)), charCode >= 48 && charCode <= 57); Index++); |
| // Floats cannot contain a leading decimal point; however, this |
| // case is already accounted for by the parser. |
| if (source.charCodeAt(Index) == 46) { |
| position = ++Index; |
| // Parse the decimal component. |
| for (; position < length; position++) { |
| charCode = source.charCodeAt(position); |
| if (charCode < 48 || charCode > 57) { |
| break; |
| } |
| } |
| if (position == Index) { |
| // Illegal trailing decimal. |
| abort(); |
| } |
| Index = position; |
| } |
| // Parse exponents. The `e` denoting the exponent is |
| // case-insensitive. |
| charCode = source.charCodeAt(Index); |
| if (charCode == 101 || charCode == 69) { |
| charCode = source.charCodeAt(++Index); |
| // Skip past the sign following the exponent, if one is |
| // specified. |
| if (charCode == 43 || charCode == 45) { |
| Index++; |
| } |
| // Parse the exponential component. |
| for (position = Index; position < length; position++) { |
| charCode = source.charCodeAt(position); |
| if (charCode < 48 || charCode > 57) { |
| break; |
| } |
| } |
| if (position == Index) { |
| // Illegal empty exponent. |
| abort(); |
| } |
| Index = position; |
| } |
| // Coerce the parsed value to a JavaScript number. |
| return +source.slice(begin, Index); |
| } |
| // A negative sign may only precede numbers. |
| if (isSigned) { |
| abort(); |
| } |
| // `true`, `false`, and `null` literals. |
| var temp = source.slice(Index, Index + 4); |
| if (temp == "true") { |
| Index += 4; |
| return true; |
| } else if (temp == "fals" && source.charCodeAt(Index + 4 ) == 101) { |
| Index += 5; |
| return false; |
| } else if (temp == "null") { |
| Index += 4; |
| return null; |
| } |
| // Unrecognized token. |
| abort(); |
| } |
| } |
| // Return the sentinel `$` character if the parser has reached the end |
| // of the source string. |
| return "$"; |
| }; |
| |
| // Internal: Parses a JSON `value` token. |
| var get = function (value) { |
| var results, hasMembers; |
| if (value == "$") { |
| // Unexpected end of input. |
| abort(); |
| } |
| if (typeof value == "string") { |
| if ((charIndexBuggy ? value.charAt(0) : value[0]) == "@") { |
| // Remove the sentinel `@` character. |
| return value.slice(1); |
| } |
| // Parse object and array literals. |
| if (value == "[") { |
| // Parses a JSON array, returning a new JavaScript array. |
| results = []; |
| for (;;) { |
| value = lex(); |
| // A closing square bracket marks the end of the array literal. |
| if (value == "]") { |
| break; |
| } |
| // If the array literal contains elements, the current token |
| // should be a comma separating the previous element from the |
| // next. |
| if (hasMembers) { |
| if (value == ",") { |
| value = lex(); |
| if (value == "]") { |
| // Unexpected trailing `,` in array literal. |
| abort(); |
| } |
| } else { |
| // A `,` must separate each array element. |
| abort(); |
| } |
| } else { |
| hasMembers = true; |
| } |
| // Elisions and leading commas are not permitted. |
| if (value == ",") { |
| abort(); |
| } |
| results.push(get(value)); |
| } |
| return results; |
| } else if (value == "{") { |
| // Parses a JSON object, returning a new JavaScript object. |
| results = {}; |
| for (;;) { |
| value = lex(); |
| // A closing curly brace marks the end of the object literal. |
| if (value == "}") { |
| break; |
| } |
| // If the object literal contains members, the current token |
| // should be a comma separator. |
| if (hasMembers) { |
| if (value == ",") { |
| value = lex(); |
| if (value == "}") { |
| // Unexpected trailing `,` in object literal. |
| abort(); |
| } |
| } else { |
| // A `,` must separate each object member. |
| abort(); |
| } |
| } else { |
| hasMembers = true; |
| } |
| // Leading commas are not permitted, object property names must be |
| // double-quoted strings, and a `:` must separate each property |
| // name and value. |
| if (value == "," || typeof value != "string" || (charIndexBuggy ? value.charAt(0) : value[0]) != "@" || lex() != ":") { |
| abort(); |
| } |
| results[value.slice(1)] = get(lex()); |
| } |
| return results; |
| } |
| // Unexpected token encountered. |
| abort(); |
| } |
| return value; |
| }; |
| |
| // Internal: Updates a traversed object member. |
| var update = function (source, property, callback) { |
| var element = walk(source, property, callback); |
| if (element === undefined) { |
| delete source[property]; |
| } else { |
| source[property] = element; |
| } |
| }; |
| |
| // Internal: Recursively traverses a parsed JSON object, invoking the |
| // `callback` function for each value. This is an implementation of the |
| // `Walk(holder, name)` operation defined in ES 5.1 section 15.12.2. |
| var walk = function (source, property, callback) { |
| var value = source[property], length; |
| if (typeof value == "object" && value) { |
| // `forOwn` can't be used to traverse an array in Opera <= 8.54 |
| // because its `Object#hasOwnProperty` implementation returns `false` |
| // for array indices (e.g., `![1, 2, 3].hasOwnProperty("0")`). |
| if (getClass.call(value) == arrayClass) { |
| for (length = value.length; length--;) { |
| update(getClass, forOwn, value, length, callback); |
| } |
| } else { |
| forOwn(value, function (property) { |
| update(value, property, callback); |
| }); |
| } |
| } |
| return callback.call(source, property, value); |
| }; |
| |
| // Public: `JSON.parse`. See ES 5.1 section 15.12.2. |
| exports.parse = function (source, callback) { |
| var result, value; |
| Index = 0; |
| Source = "" + source; |
| result = get(lex()); |
| // If a JSON string contains multiple tokens, it is invalid. |
| if (lex() != "$") { |
| abort(); |
| } |
| // Reset the parser state. |
| Index = Source = null; |
| return callback && getClass.call(callback) == functionClass ? walk((value = {}, value[""] = result, value), "", callback) : result; |
| }; |
| } |
| } |
| |
| exports.runInContext = runInContext; |
| return exports; |
| } |
| |
| if (freeExports && !isLoader) { |
| // Export for CommonJS environments. |
| runInContext(root, freeExports); |
| } else { |
| // Export for web browsers and JavaScript engines. |
| var nativeJSON = root.JSON, |
| previousJSON = root.JSON3, |
| isRestored = false; |
| |
| var JSON3 = runInContext(root, (root.JSON3 = { |
| // Public: Restores the original value of the global `JSON` object and |
| // returns a reference to the `JSON3` object. |
| "noConflict": function () { |
| if (!isRestored) { |
| isRestored = true; |
| root.JSON = nativeJSON; |
| root.JSON3 = previousJSON; |
| nativeJSON = previousJSON = null; |
| } |
| return JSON3; |
| } |
| })); |
| |
| root.JSON = { |
| "parse": JSON3.parse, |
| "stringify": JSON3.stringify |
| }; |
| } |
| |
| // Export for asynchronous module loaders. |
| if (isLoader) { |
| define(function () { |
| return JSON3; |
| }); |
| } |
| }).call(this); |