gecko/js/src/json.cpp

1135 lines
33 KiB
C++
Raw Normal View History

/* -*- Mode: C; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*-
* vim: set ts=8 sw=4 et tw=99:
*
* ***** BEGIN LICENSE BLOCK *****
2008-10-06 13:54:12 -07:00
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
*
* The contents of this file are subject to the Mozilla Public License Version
* 1.1 (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
* http://www.mozilla.org/MPL/
*
* Software distributed under the License is distributed on an "AS IS" basis,
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
* for the specific language governing rights and limitations under the
* License.
*
* The Original Code is SpiderMonkey JSON.
*
* The Initial Developer of the Original Code is
* Mozilla Corporation.
* Portions created by the Initial Developer are Copyright (C) 1998-1999
* the Initial Developer. All Rights Reserved.
*
* Contributor(s):
* Robert Sayre <sayrer@gmail.com>
*
* Alternatively, the contents of this file may be used under the terms of
* either of the GNU General Public License Version 2 or later (the "GPL"),
* or the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
* in which case the provisions of the GPL or the LGPL are applicable instead
* of those above. If you wish to allow use of your version of this file only
* under the terms of either the GPL or the LGPL, and not to allow others to
* use your version of this file under the terms of the MPL, indicate your
* decision by deleting the provisions above and replace them with the notice
* and other provisions required by the GPL or the LGPL. If you do not delete
* the provisions above, a recipient may use your version of this file under
* the terms of any one of the MPL, the GPL or the LGPL.
*
* ***** END LICENSE BLOCK ***** */
#include <string.h> /* memset */
2008-10-06 13:54:12 -07:00
#include "jsapi.h"
#include "jsarena.h"
#include "jsarray.h"
#include "jsatom.h"
#include "jsbool.h"
#include "jscntxt.h"
#include "jsdtoa.h"
#include "jsinterp.h"
#include "jsiter.h"
#include "jsnum.h"
#include "jsobj.h"
#include "jsprf.h"
#include "jsscan.h"
#include "jsstr.h"
#include "jstypes.h"
#include "jsstdint.h"
2008-10-06 13:54:12 -07:00
#include "jsutil.h"
#include "json.h"
JSClass js_JSONClass = {
js_JSON_str,
JSCLASS_HAS_CACHED_PROTO(JSProto_JSON),
JS_PropertyStub, JS_PropertyStub, JS_PropertyStub, JS_PropertyStub,
JS_EnumerateStub, JS_ResolveStub, JS_ConvertStub, JS_FinalizeStub,
JSCLASS_NO_OPTIONAL_MEMBERS
};
JSBool
js_json_parse(JSContext *cx, uintN argc, jsval *vp)
{
JSString *s = NULL;
jsval *argv = vp + 2;
jsval reviver = JSVAL_VOID;
JSAutoTempValueRooter(cx, 1, &reviver);
if (!JS_ConvertArguments(cx, argc, argv, "S / v", &s, &reviver))
2008-10-06 13:54:12 -07:00
return JS_FALSE;
JSONParser *jp = js_BeginJSONParse(cx, vp);
JSBool ok = jp != NULL;
if (ok) {
ok = js_ConsumeJSONText(cx, jp, JS_GetStringChars(s), JS_GetStringLength(s));
ok &= js_FinishJSONParse(cx, jp, reviver);
2008-10-06 13:54:12 -07:00
}
return ok;
}
class StringifyClosure : JSAutoTempValueRooter
2008-10-06 13:54:12 -07:00
{
public:
StringifyClosure(JSContext *cx, size_t len, jsval *vec)
: JSAutoTempValueRooter(cx, len, vec), cx(cx), s(vec)
2008-10-06 13:54:12 -07:00
{
}
JSContext *cx;
jsval *s;
};
static JSBool
WriteCallback(const jschar *buf, uint32 len, void *data)
2008-10-06 13:54:12 -07:00
{
StringifyClosure *sc = static_cast<StringifyClosure*>(data);
JSString *s1 = JSVAL_TO_STRING(sc->s[0]);
JSString *s2 = js_NewStringCopyN(sc->cx, buf, len);
2008-10-06 13:54:12 -07:00
if (!s2)
return JS_FALSE;
sc->s[1] = STRING_TO_JSVAL(s2);
2008-10-06 13:54:12 -07:00
s1 = js_ConcatStrings(sc->cx, s1, s2);
if (!s1)
return JS_FALSE;
sc->s[0] = STRING_TO_JSVAL(s1);
sc->s[1] = JSVAL_VOID;
2008-10-06 13:54:12 -07:00
return JS_TRUE;
}
JSBool
js_json_stringify(JSContext *cx, uintN argc, jsval *vp)
{
jsval *argv = vp + 2;
2008-10-06 13:54:12 -07:00
// Must throw an Error if there isn't a first arg
if (!JS_ConvertArguments(cx, argc, argv, "v", vp))
2008-10-06 13:54:12 -07:00
return JS_FALSE;
if (!js_TryJSON(cx, vp))
2008-10-06 13:54:12 -07:00
return JS_FALSE;
2008-10-06 13:54:12 -07:00
JSString *s = JS_NewStringCopyN(cx, "", 0);
if (!s)
return JS_FALSE;
2008-10-06 13:54:12 -07:00
jsval vec[2] = {STRING_TO_JSVAL(s), JSVAL_VOID};
StringifyClosure sc(cx, 2, vec);
JSAutoTempValueRooter resultTvr(cx, 1, sc.s);
JSBool ok = js_Stringify(cx, vp, NULL, &WriteCallback, &sc, 0);
*vp = *sc.s;
2008-10-06 13:54:12 -07:00
return ok;
}
JSBool
js_TryJSON(JSContext *cx, jsval *vp)
{
// Checks whether the return value implements toJSON()
JSBool ok = JS_TRUE;
2008-10-06 13:54:12 -07:00
if (!JSVAL_IS_PRIMITIVE(*vp)) {
JSObject *obj = JSVAL_TO_OBJECT(*vp);
ok = js_TryMethod(cx, obj, cx->runtime->atomState.toJSONAtom, 0, NULL, vp);
}
2008-10-06 13:54:12 -07:00
return ok;
}
static const jschar quote = jschar('"');
static const jschar backslash = jschar('\\');
static const jschar unicodeEscape[] = {'\\', 'u', '0', '0'};
static JSBool
write_string(JSContext *cx, JSONWriteCallback callback, void *data, const jschar *buf, uint32 len)
{
if (!callback(&quote, 1, data))
return JS_FALSE;
uint32 mark = 0;
uint32 i;
for (i = 0; i < len; ++i) {
if (buf[i] == quote || buf[i] == backslash) {
if (!callback(&buf[mark], i - mark, data) || !callback(&backslash, 1, data) ||
!callback(&buf[i], 1, data)) {
return JS_FALSE;
}
mark = i + 1;
} else if (buf[i] <= 31 || buf[i] == 127) {
if (!callback(&buf[mark], i - mark, data) || !callback(unicodeEscape, 4, data))
return JS_FALSE;
char ubuf[3];
size_t len = JS_snprintf(ubuf, sizeof(ubuf), "%.2x", buf[i]);
2008-10-06 13:54:12 -07:00
JS_ASSERT(len == 2);
jschar wbuf[3];
size_t wbufSize = JS_ARRAY_LENGTH(wbuf);
if (!js_InflateStringToBuffer(cx, ubuf, len, wbuf, &wbufSize) ||
!callback(wbuf, wbufSize, data)) {
2008-10-06 13:54:12 -07:00
return JS_FALSE;
}
2008-10-06 13:54:12 -07:00
mark = i + 1;
}
}
if (mark < len && !callback(&buf[mark], len - mark, data))
return JS_FALSE;
if (!callback(&quote, 1, data))
return JS_FALSE;
return JS_TRUE;
}
static JSBool
stringify_leaf(JSContext *cx, jsval *vp,
JSONWriteCallback callback, void *data, JSType type)
{
JSString *outputString;
JSString *s = js_ValueToString(cx, *vp);
if (!s)
return JS_FALSE;
if (type == JSTYPE_STRING)
return write_string(cx, callback, data, JS_GetStringChars(s), JS_GetStringLength(s));
if (type == JSTYPE_NUMBER) {
if (JSVAL_IS_DOUBLE(*vp)) {
jsdouble d = *JSVAL_TO_DOUBLE(*vp);
if (!JSDOUBLE_IS_FINITE(d))
outputString = JS_NewStringCopyN(cx, "null", 4);
else
outputString = s;
} else {
outputString = s;
}
} else if (type == JSTYPE_BOOLEAN) {
outputString = s;
} else if (JSVAL_IS_NULL(*vp)) {
outputString = JS_NewStringCopyN(cx, "null", 4);
} else if (JSVAL_IS_VOID(*vp) || type == JSTYPE_FUNCTION || type == JSTYPE_XML) {
outputString = JS_NewStringCopyN(cx, "undefined", 9);
} else {
JS_NOT_REACHED("A type we don't know about");
outputString = JS_NewStringCopyN(cx, "undefined", 9);
}
if (!outputString)
return JS_FALSE;
return callback(JS_GetStringChars(outputString), JS_GetStringLength(outputString), data);
}
static JSBool
stringify(JSContext *cx, jsval *vp, JSObject *replacer,
JSONWriteCallback callback, void *data, uint32 depth)
2008-10-06 13:54:12 -07:00
{
if (depth > JSON_MAX_DEPTH) {
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL, JSMSG_JSON_BAD_STRINGIFY);
2008-10-06 13:54:12 -07:00
return JS_FALSE; /* encoding error */
}
2008-10-06 13:54:12 -07:00
JSBool ok = JS_TRUE;
JSObject *obj = JSVAL_TO_OBJECT(*vp);
JSBool isArray = JS_IsArrayObject(cx, obj);
jschar output = jschar(isArray ? '[' : '{');
if (!callback(&output, 1, data))
return JS_FALSE;
2008-10-08 10:31:07 -07:00
2008-10-06 13:54:12 -07:00
JSObject *iterObj = NULL;
jsint i = 0;
jsuint length = 0;
if (isArray) {
if (!js_GetLengthProperty(cx, obj, &length))
2008-10-06 13:54:12 -07:00
return JS_FALSE;
} else {
if (!js_ValueToIterator(cx, JSITER_ENUMERATE, vp))
return JS_FALSE;
iterObj = JSVAL_TO_OBJECT(*vp);
}
jsval outputValue = JSVAL_VOID;
JSAutoTempValueRooter tvr(cx, 1, &outputValue);
jsval key;
JSBool memberWritten = JS_FALSE;
do {
outputValue = JSVAL_VOID;
if (isArray) {
if ((jsuint)i >= length)
break;
jsid index;
if (!js_IndexToId(cx, i, &index))
return JS_FALSE;
ok = OBJ_GET_PROPERTY(cx, obj, index, &outputValue);
if (!ok)
break;
i++;
2008-10-06 13:54:12 -07:00
} else {
ok = js_CallIteratorNext(cx, iterObj, &key);
if (!ok)
break;
if (key == JSVAL_HOLE)
break;
JSString *ks;
if (JSVAL_IS_STRING(key)) {
ks = JSVAL_TO_STRING(key);
} else {
ks = js_ValueToString(cx, key);
2008-10-06 13:54:12 -07:00
if (!ks) {
ok = JS_FALSE;
break;
}
}
// Don't include prototype properties, since this operation is
// supposed to be implemented as if by ES3.1 Object.keys()
jsid id;
jsval v = JS_FALSE;
if (!js_ValueToStringId(cx, STRING_TO_JSVAL(ks), &id) ||
!js_HasOwnProperty(cx, obj->map->ops->lookupProperty, obj, id, &v)) {
ok = JS_FALSE;
break;
}
2008-10-06 13:54:12 -07:00
if (v != JSVAL_TRUE)
continue;
ok = JS_GetPropertyById(cx, obj, id, &outputValue);
if (!ok)
break;
}
2008-10-06 13:54:12 -07:00
// if this is an array, holes are transmitted as null
if (isArray && outputValue == JSVAL_VOID) {
outputValue = JSVAL_NULL;
} else if (JSVAL_IS_OBJECT(outputValue)) {
ok = js_TryJSON(cx, &outputValue);
if (!ok)
break;
}
JSType type = JS_TypeOfValue(cx, outputValue);
// elide undefined values and functions and XML
if (outputValue == JSVAL_VOID || type == JSTYPE_FUNCTION || type == JSTYPE_XML)
2008-10-06 13:54:12 -07:00
continue;
// output a comma unless this is the first member to write
if (memberWritten) {
output = jschar(',');
ok = callback(&output, 1, data);
if (!ok)
break;
2008-10-06 13:54:12 -07:00
}
memberWritten = JS_TRUE;
// Be careful below, this string is weakly rooted.
JSString *s;
// If this isn't an array, we need to output a key
if (!isArray) {
s = js_ValueToString(cx, key);
2008-10-06 13:54:12 -07:00
if (!s) {
ok = JS_FALSE;
break;
}
ok = write_string(cx, callback, data, JS_GetStringChars(s), JS_GetStringLength(s));
if (!ok)
break;
output = jschar(':');
ok = callback(&output, 1, data);
if (!ok)
break;
}
if (!JSVAL_IS_PRIMITIVE(outputValue)) {
// recurse
ok = stringify(cx, &outputValue, replacer, callback, data, depth + 1);
2008-10-06 13:54:12 -07:00
} else {
ok = stringify_leaf(cx, &outputValue, callback, data, type);
2008-10-06 13:54:12 -07:00
}
} while (ok);
if (iterObj) {
// Always close the iterator, but make sure not to stomp on OK
ok &= js_CloseIterator(cx, *vp);
}
if (!ok)
2008-10-06 13:54:12 -07:00
return JS_FALSE;
2008-10-06 13:54:12 -07:00
output = jschar(isArray ? ']' : '}');
ok = callback(&output, 1, data);
return ok;
}
JSBool
js_Stringify(JSContext *cx, jsval *vp, JSObject *replacer,
JSONWriteCallback callback, void *data, uint32 depth)
{
JSBool ok;
JSType type = JS_TypeOfValue(cx, *vp);
if (JSVAL_IS_PRIMITIVE(*vp) || type == JSTYPE_FUNCTION || type == JSTYPE_XML) {
ok = stringify_leaf(cx, vp, callback, data, type);
} else {
ok = stringify(cx, vp, replacer, callback, data, depth);
}
2008-10-06 13:54:12 -07:00
return ok;
}
// helper to determine whether a character could be part of a number
2008-10-08 10:31:07 -07:00
static JSBool IsNumChar(jschar c)
2008-10-06 13:54:12 -07:00
{
return ((c <= '9' && c >= '0') || c == '.' || c == '-' || c == '+' || c == 'e' || c == 'E');
}
static JSBool HandleData(JSContext *cx, JSONParser *jp, JSONDataType type);
static JSBool PopState(JSContext *cx, JSONParser *jp);
static JSBool
DestroyIdArrayOnError(JSContext *cx, JSIdArray *ida) {
JS_DestroyIdArray(cx, ida);
return JS_FALSE;
}
static JSBool
Walk(JSContext *cx, jsid id, JSObject *holder, jsval reviver, jsval *vp)
{
JS_CHECK_RECURSION(cx, return JS_FALSE);
if (!OBJ_GET_PROPERTY(cx, holder, id, vp))
return JS_FALSE;
JSObject *obj;
if (!JSVAL_IS_PRIMITIVE(*vp) && !js_IsCallable(obj = JSVAL_TO_OBJECT(*vp), cx)) {
jsval propValue = JSVAL_VOID;
JSAutoTempValueRooter tvr(cx, 1, &propValue);
if(OBJ_IS_ARRAY(cx, obj)) {
jsuint length = 0;
if (!js_GetLengthProperty(cx, obj, &length))
return JS_FALSE;
for (jsuint i = 0; i < length; i++) {
jsid index;
if (!js_IndexToId(cx, i, &index))
return JS_FALSE;
if (!Walk(cx, index, obj, reviver, &propValue))
return JS_FALSE;
if (!OBJ_DEFINE_PROPERTY(cx, obj, index, propValue,
NULL, NULL, JSPROP_ENUMERATE, NULL)) {
return JS_FALSE;
}
}
} else {
JSIdArray *ida = JS_Enumerate(cx, obj);
if (!ida)
return JS_FALSE;
JSAutoTempValueRooter idaroot(cx, JS_ARRAY_LENGTH(ida), (jsval*)ida);
for(jsint i = 0; i < ida->length; i++) {
jsid idName = ida->vector[i];
if (!Walk(cx, idName, obj, reviver, &propValue))
return DestroyIdArrayOnError(cx, ida);
if (propValue == JSVAL_VOID) {
if (!js_DeleteProperty(cx, obj, idName, &propValue))
return DestroyIdArrayOnError(cx, ida);
} else {
if (!OBJ_DEFINE_PROPERTY(cx, obj, idName, propValue,
NULL, NULL, JSPROP_ENUMERATE, NULL)) {
return DestroyIdArrayOnError(cx, ida);
}
}
}
JS_DestroyIdArray(cx, ida);
}
}
// return reviver.call(holder, key, value);
jsval value = *vp;
JSString *key = js_ValueToString(cx, ID_TO_VALUE(id));
if (!key)
return JS_FALSE;
jsval vec[2] = {STRING_TO_JSVAL(key), value};
jsval reviverResult;
if (!JS_CallFunctionValue(cx, holder, reviver, 2, vec, &reviverResult))
return JS_FALSE;
*vp = reviverResult;
return JS_TRUE;
}
static JSBool
Revive(JSContext *cx, jsval reviver, jsval *vp)
{
JSObject *obj = js_NewObject(cx, &js_ObjectClass, NULL, NULL, 0);
if (!obj)
return JS_FALSE;
jsval v = OBJECT_TO_JSVAL(obj);
JSAutoTempValueRooter tvr(cx, 1, &v);
if (!OBJ_DEFINE_PROPERTY(cx, obj, ATOM_TO_JSID(cx->runtime->atomState.emptyAtom),
*vp, NULL, NULL, JSPROP_ENUMERATE, NULL)) {
return JS_FALSE;
}
return Walk(cx, ATOM_TO_JSID(cx->runtime->atomState.emptyAtom), obj, reviver, vp);
}
#define JSON_INITIAL_BUFSIZE 256
2008-10-06 13:54:12 -07:00
JSONParser *
js_BeginJSONParse(JSContext *cx, jsval *rootVal)
{
if (!cx)
return NULL;
JSObject *arr = js_NewArrayObject(cx, 0, NULL);
2008-10-06 13:54:12 -07:00
if (!arr)
return NULL;
JSONParser *jp = (JSONParser*) JS_malloc(cx, sizeof(JSONParser));
if (!jp)
2008-10-08 10:31:07 -07:00
return NULL;
memset(jp, 0, sizeof *jp);
2008-10-06 13:54:12 -07:00
jp->objectStack = arr;
if (!js_AddRoot(cx, &jp->objectStack, "JSON parse stack"))
2008-10-06 13:54:12 -07:00
goto bad;
jp->statep = jp->stateStack;
*jp->statep = JSON_PARSE_STATE_INIT;
jp->rootVal = rootVal;
js_InitStringBuffer(&jp->objectKey);
js_InitStringBuffer(&jp->buffer);
2008-10-06 13:54:12 -07:00
if (!jp->buffer.grow(&jp->buffer, JSON_INITIAL_BUFSIZE)) {
JS_ReportOutOfMemory(cx);
goto bad;
}
2008-10-06 13:54:12 -07:00
return jp;
2008-10-06 13:54:12 -07:00
bad:
js_FinishJSONParse(cx, jp, JSVAL_NULL);
2008-10-06 13:54:12 -07:00
return NULL;
}
JSBool
js_FinishJSONParse(JSContext *cx, JSONParser *jp, jsval reviver)
2008-10-06 13:54:12 -07:00
{
if (!jp)
return JS_TRUE;
2009-04-21 18:09:44 -07:00
JSBool early_ok = JS_TRUE;
// Check for unprocessed primitives at the root. This doesn't happen for
// strings because a closing quote triggers value processing.
if ((jp->statep - jp->stateStack) == 1) {
if (*jp->statep == JSON_PARSE_STATE_KEYWORD) {
2009-04-21 18:09:44 -07:00
early_ok = HandleData(cx, jp, JSON_DATA_KEYWORD);
2009-04-21 22:50:51 -07:00
if (early_ok)
PopState(cx, jp);
} else if (*jp->statep == JSON_PARSE_STATE_NUMBER) {
2009-04-21 18:09:44 -07:00
early_ok = HandleData(cx, jp, JSON_DATA_NUMBER);
2009-04-21 22:50:51 -07:00
if (early_ok)
PopState(cx, jp);
}
}
js_FinishStringBuffer(&jp->objectKey);
js_FinishStringBuffer(&jp->buffer);
/* This internal API is infallible, in spite of its JSBool return type. */
js_RemoveRoot(cx->runtime, &jp->objectStack);
2008-10-06 13:54:12 -07:00
JSBool ok = *jp->statep == JSON_PARSE_STATE_FINISHED;
jsval *vp = jp->rootVal;
2008-10-06 13:54:12 -07:00
JS_free(cx, jp);
2009-04-21 18:09:44 -07:00
if (!early_ok)
return JS_FALSE;
if (!ok) {
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL, JSMSG_JSON_BAD_PARSE);
return JS_FALSE;
}
if (!JSVAL_IS_PRIMITIVE(reviver) && js_IsCallable(JSVAL_TO_OBJECT(reviver), cx))
ok = Revive(cx, reviver, vp);
2008-10-06 13:54:12 -07:00
return ok;
}
static JSBool
PushState(JSContext *cx, JSONParser *jp, JSONParserState state)
2008-10-06 13:54:12 -07:00
{
if (*jp->statep == JSON_PARSE_STATE_FINISHED) {
// extra input
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL, JSMSG_JSON_BAD_PARSE);
return JS_FALSE;
}
2008-10-06 13:54:12 -07:00
jp->statep++;
if ((uint32)(jp->statep - jp->stateStack) >= JS_ARRAY_LENGTH(jp->stateStack)) {
// too deep
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL, JSMSG_JSON_BAD_PARSE);
return JS_FALSE;
}
2008-10-06 13:54:12 -07:00
*jp->statep = state;
return JS_TRUE;
}
static JSBool
PopState(JSContext *cx, JSONParser *jp)
2008-10-06 13:54:12 -07:00
{
jp->statep--;
if (jp->statep < jp->stateStack) {
jp->statep = jp->stateStack;
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL, JSMSG_JSON_BAD_PARSE);
2008-10-06 13:54:12 -07:00
return JS_FALSE;
2008-10-08 10:31:07 -07:00
}
2008-10-06 13:54:12 -07:00
if (*jp->statep == JSON_PARSE_STATE_INIT)
*jp->statep = JSON_PARSE_STATE_FINISHED;
return JS_TRUE;
}
static JSBool
PushValue(JSContext *cx, JSONParser *jp, JSObject *parent, jsval value)
{
JSBool ok;
if (OBJ_IS_ARRAY(cx, parent)) {
jsuint len;
ok = js_GetLengthProperty(cx, parent, &len);
if (ok) {
jsid index;
if (!js_IndexToId(cx, len, &index))
return JS_FALSE;
ok = OBJ_DEFINE_PROPERTY(cx, parent, index, value,
NULL, NULL, JSPROP_ENUMERATE, NULL);
}
2008-10-06 13:54:12 -07:00
} else {
ok = JS_DefineUCProperty(cx, parent, jp->objectKey.base,
STRING_BUFFER_OFFSET(&jp->objectKey), value,
2008-10-06 13:54:12 -07:00
NULL, NULL, JSPROP_ENUMERATE);
js_RewindStringBuffer(&jp->objectKey);
2008-10-06 13:54:12 -07:00
}
return ok;
}
static JSBool
PushObject(JSContext *cx, JSONParser *jp, JSObject *obj)
{
jsuint len;
if (!js_GetLengthProperty(cx, jp->objectStack, &len))
2008-10-06 13:54:12 -07:00
return JS_FALSE;
if (len >= JSON_MAX_DEPTH) {
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL, JSMSG_JSON_BAD_PARSE);
return JS_FALSE;
}
2008-10-06 13:54:12 -07:00
jsval v = OBJECT_TO_JSVAL(obj);
JSAutoTempValueRooter tvr(cx, v);
2008-10-06 13:54:12 -07:00
// Check if this is the root object
if (len == 0) {
2008-10-08 10:31:07 -07:00
*jp->rootVal = v;
// This property must be enumerable to keep the array dense
if (!OBJ_DEFINE_PROPERTY(cx, jp->objectStack, INT_TO_JSID(0), *jp->rootVal,
NULL, NULL, JSPROP_ENUMERATE, NULL)) {
2008-10-06 13:54:12 -07:00
return JS_FALSE;
}
2008-10-06 13:54:12 -07:00
return JS_TRUE;
}
jsval p;
if (!OBJ_GET_PROPERTY(cx, jp->objectStack, INT_TO_JSID(len - 1), &p))
2008-10-06 13:54:12 -07:00
return JS_FALSE;
2008-10-06 13:54:12 -07:00
JS_ASSERT(JSVAL_IS_OBJECT(p));
JSObject *parent = JSVAL_TO_OBJECT(p);
if (!PushValue(cx, jp, parent, v))
2008-10-06 13:54:12 -07:00
return JS_FALSE;
// This property must be enumerable to keep the array dense
if (!OBJ_DEFINE_PROPERTY(cx, jp->objectStack, INT_TO_JSID(len), v,
NULL, NULL, JSPROP_ENUMERATE, NULL)) {
2008-10-06 13:54:12 -07:00
return JS_FALSE;
}
2008-10-06 13:54:12 -07:00
return JS_TRUE;
}
static JSBool
OpenObject(JSContext *cx, JSONParser *jp)
{
JSObject *obj = js_NewObject(cx, &js_ObjectClass, NULL, NULL, 0);
2008-10-06 13:54:12 -07:00
if (!obj)
return JS_FALSE;
return PushObject(cx, jp, obj);
}
static JSBool
OpenArray(JSContext *cx, JSONParser *jp)
{
// Add an array to an existing array or object
JSObject *arr = js_NewArrayObject(cx, 0, NULL);
2008-10-06 13:54:12 -07:00
if (!arr)
return JS_FALSE;
return PushObject(cx, jp, arr);
}
static JSBool
CloseObject(JSContext *cx, JSONParser *jp)
{
jsuint len;
if (!js_GetLengthProperty(cx, jp->objectStack, &len))
2008-10-06 13:54:12 -07:00
return JS_FALSE;
if (!js_SetLengthProperty(cx, jp->objectStack, len - 1))
2008-10-06 13:54:12 -07:00
return JS_FALSE;
return JS_TRUE;
}
static JSBool
CloseArray(JSContext *cx, JSONParser *jp)
{
return CloseObject(cx, jp);
2008-10-06 13:54:12 -07:00
}
static JSBool
PushPrimitive(JSContext *cx, JSONParser *jp, jsval value)
{
JSAutoTempValueRooter tvr(cx, 1, &value);
jsuint len;
if (!js_GetLengthProperty(cx, jp->objectStack, &len))
return JS_FALSE;
if (len > 0) {
jsval o;
if (!OBJ_GET_PROPERTY(cx, jp->objectStack, INT_TO_JSID(len - 1), &o))
return JS_FALSE;
JS_ASSERT(!JSVAL_IS_PRIMITIVE(o));
return PushValue(cx, jp, JSVAL_TO_OBJECT(o), value);
}
// root value must be primitive
*jp->rootVal = value;
return JS_TRUE;
}
2008-10-06 13:54:12 -07:00
static JSBool
HandleNumber(JSContext *cx, JSONParser *jp, const jschar *buf, uint32 len)
{
const jschar *ep;
2008-10-08 10:31:07 -07:00
double val;
if (!js_strtod(cx, buf, buf + len, &ep, &val))
2008-10-06 13:54:12 -07:00
return JS_FALSE;
if (ep != buf + len) {
// bad number input
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL, JSMSG_JSON_BAD_PARSE);
return JS_FALSE;
}
2008-10-06 13:54:12 -07:00
jsval numVal;
if (!JS_NewNumberValue(cx, val, &numVal))
return JS_FALSE;
return PushPrimitive(cx, jp, numVal);
2008-10-06 13:54:12 -07:00
}
static JSBool
HandleString(JSContext *cx, JSONParser *jp, const jschar *buf, uint32 len)
{
JSString *str = js_NewStringCopyN(cx, buf, len);
if (!str)
2008-10-06 13:54:12 -07:00
return JS_FALSE;
return PushPrimitive(cx, jp, STRING_TO_JSVAL(str));
2008-10-06 13:54:12 -07:00
}
static JSBool
HandleKeyword(JSContext *cx, JSONParser *jp, const jschar *buf, uint32 len)
{
jsval keyword;
JSTokenType tt = js_CheckKeyword(buf, len);
if (tt != TOK_PRIMARY) {
// bad keyword
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL, JSMSG_JSON_BAD_PARSE);
2008-10-06 13:54:12 -07:00
return JS_FALSE;
}
2008-10-06 13:54:12 -07:00
if (buf[0] == 'n') {
2008-10-06 13:54:12 -07:00
keyword = JSVAL_NULL;
} else if (buf[0] == 't') {
2008-10-06 13:54:12 -07:00
keyword = JSVAL_TRUE;
} else if (buf[0] == 'f') {
2008-10-06 13:54:12 -07:00
keyword = JSVAL_FALSE;
} else {
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL, JSMSG_JSON_BAD_PARSE);
2008-10-06 13:54:12 -07:00
return JS_FALSE;
}
2008-10-06 13:54:12 -07:00
return PushPrimitive(cx, jp, keyword);
2008-10-06 13:54:12 -07:00
}
static JSBool
HandleData(JSContext *cx, JSONParser *jp, JSONDataType type)
2008-10-06 13:54:12 -07:00
{
JSBool ok;
2008-10-06 13:54:12 -07:00
switch (type) {
case JSON_DATA_STRING:
ok = HandleString(cx, jp, jp->buffer.base, STRING_BUFFER_OFFSET(&jp->buffer));
break;
case JSON_DATA_KEYSTRING:
js_AppendUCString(&jp->objectKey, jp->buffer.base, STRING_BUFFER_OFFSET(&jp->buffer));
ok = STRING_BUFFER_OK(&jp->objectKey);
if (!ok)
JS_ReportOutOfMemory(cx);
break;
case JSON_DATA_NUMBER:
ok = HandleNumber(cx, jp, jp->buffer.base, STRING_BUFFER_OFFSET(&jp->buffer));
break;
default:
JS_ASSERT(type == JSON_DATA_KEYWORD);
ok = HandleKeyword(cx, jp, jp->buffer.base, STRING_BUFFER_OFFSET(&jp->buffer));
break;
}
2008-10-06 13:54:12 -07:00
if (ok) {
ok = STRING_BUFFER_OK(&jp->buffer);
if (ok)
js_RewindStringBuffer(&jp->buffer);
else
JS_ReportOutOfMemory(cx);
}
return ok;
2008-10-06 13:54:12 -07:00
}
JSBool
js_ConsumeJSONText(JSContext *cx, JSONParser *jp, const jschar *data, uint32 len)
{
uint32 i;
if (*jp->statep == JSON_PARSE_STATE_INIT) {
PushState(cx, jp, JSON_PARSE_STATE_VALUE);
2008-10-06 13:54:12 -07:00
}
2008-10-08 10:31:07 -07:00
for (i = 0; i < len; i++) {
2008-10-06 13:54:12 -07:00
jschar c = data[i];
switch (*jp->statep) {
case JSON_PARSE_STATE_VALUE:
if (c == ']') {
// empty array
if (!PopState(cx, jp))
return JS_FALSE;
if (*jp->statep != JSON_PARSE_STATE_ARRAY) {
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL, JSMSG_JSON_BAD_PARSE);
return JS_FALSE;
2008-10-06 13:54:12 -07:00
}
if (!CloseArray(cx, jp) || !PopState(cx, jp))
return JS_FALSE;
2008-10-06 13:54:12 -07:00
break;
}
2008-10-06 13:54:12 -07:00
if (c == '}') {
// we should only find these in OBJECT_KEY state
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL, JSMSG_JSON_BAD_PARSE);
return JS_FALSE;
}
2008-10-06 13:54:12 -07:00
if (c == '"') {
*jp->statep = JSON_PARSE_STATE_STRING;
2008-10-06 13:54:12 -07:00
break;
}
2008-10-06 13:54:12 -07:00
if (IsNumChar(c)) {
*jp->statep = JSON_PARSE_STATE_NUMBER;
js_FastAppendChar(&jp->buffer, c);
2008-10-06 13:54:12 -07:00
break;
}
2008-10-06 13:54:12 -07:00
if (JS7_ISLET(c)) {
*jp->statep = JSON_PARSE_STATE_KEYWORD;
js_FastAppendChar(&jp->buffer, c);
2008-10-06 13:54:12 -07:00
break;
}
2008-10-08 10:31:07 -07:00
// fall through in case the value is an object or array
case JSON_PARSE_STATE_OBJECT_VALUE:
if (c == '{') {
*jp->statep = JSON_PARSE_STATE_OBJECT;
if (!OpenObject(cx, jp) || !PushState(cx, jp, JSON_PARSE_STATE_OBJECT_PAIR))
return JS_FALSE;
} else if (c == '[') {
*jp->statep = JSON_PARSE_STATE_ARRAY;
if (!OpenArray(cx, jp) || !PushState(cx, jp, JSON_PARSE_STATE_VALUE))
return JS_FALSE;
} else if (!JS_ISXMLSPACE(c)) {
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL, JSMSG_JSON_BAD_PARSE);
return JS_FALSE;
}
break;
2008-10-08 10:31:07 -07:00
case JSON_PARSE_STATE_OBJECT:
if (c == '}') {
if (!CloseObject(cx, jp) || !PopState(cx, jp))
return JS_FALSE;
} else if (c == ',') {
if (!PushState(cx, jp, JSON_PARSE_STATE_OBJECT_PAIR))
return JS_FALSE;
} else if (c == ']' || !JS_ISXMLSPACE(c)) {
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL, JSMSG_JSON_BAD_PARSE);
return JS_FALSE;
}
break;
2008-10-08 10:31:07 -07:00
case JSON_PARSE_STATE_ARRAY :
if (c == ']') {
if (!CloseArray(cx, jp) || !PopState(cx, jp))
return JS_FALSE;
} else if (c == ',') {
if (!PushState(cx, jp, JSON_PARSE_STATE_VALUE))
return JS_FALSE;
} else if (!JS_ISXMLSPACE(c)) {
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL, JSMSG_JSON_BAD_PARSE);
return JS_FALSE;
}
break;
2008-10-08 10:31:07 -07:00
case JSON_PARSE_STATE_OBJECT_PAIR :
if (c == '"') {
// we want to be waiting for a : when the string has been read
*jp->statep = JSON_PARSE_STATE_OBJECT_IN_PAIR;
if (!PushState(cx, jp, JSON_PARSE_STATE_STRING))
return JS_FALSE;
} else if (c == '}') {
// pop off the object pair state and the object state
if (!CloseObject(cx, jp) || !PopState(cx, jp) || !PopState(cx, jp))
return JS_FALSE;
} else if (c == ']' || !JS_ISXMLSPACE(c)) {
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL, JSMSG_JSON_BAD_PARSE);
return JS_FALSE;
}
break;
2008-10-06 13:54:12 -07:00
case JSON_PARSE_STATE_OBJECT_IN_PAIR:
if (c == ':') {
*jp->statep = JSON_PARSE_STATE_VALUE;
} else if (!JS_ISXMLSPACE(c)) {
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL, JSMSG_JSON_BAD_PARSE);
return JS_FALSE;
}
break;
case JSON_PARSE_STATE_STRING:
if (c == '"') {
if (!PopState(cx, jp))
return JS_FALSE;
JSONDataType jdt;
if (*jp->statep == JSON_PARSE_STATE_OBJECT_IN_PAIR) {
jdt = JSON_DATA_KEYSTRING;
} else {
jdt = JSON_DATA_STRING;
}
if (!HandleData(cx, jp, jdt))
return JS_FALSE;
} else if (c == '\\') {
*jp->statep = JSON_PARSE_STATE_STRING_ESCAPE;
} else {
js_FastAppendChar(&jp->buffer, c);
}
break;
2008-10-08 10:31:07 -07:00
case JSON_PARSE_STATE_STRING_ESCAPE:
switch (c) {
case '"':
case '\\':
case '/':
break;
case 'b' : c = '\b'; break;
case 'f' : c = '\f'; break;
case 'n' : c = '\n'; break;
case 'r' : c = '\r'; break;
case 't' : c = '\t'; break;
default :
if (c == 'u') {
2008-10-06 13:54:12 -07:00
jp->numHex = 0;
jp->hexChar = 0;
*jp->statep = JSON_PARSE_STATE_STRING_HEX;
continue;
} else {
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL, JSMSG_JSON_BAD_PARSE);
return JS_FALSE;
2008-10-06 13:54:12 -07:00
}
}
2008-10-08 10:31:07 -07:00
js_FastAppendChar(&jp->buffer, c);
*jp->statep = JSON_PARSE_STATE_STRING;
break;
2008-10-08 10:31:07 -07:00
case JSON_PARSE_STATE_STRING_HEX:
if (('0' <= c) && (c <= '9')) {
jp->hexChar = (jp->hexChar << 4) | (c - '0');
} else if (('a' <= c) && (c <= 'f')) {
jp->hexChar = (jp->hexChar << 4) | (c - 'a' + 0x0a);
} else if (('A' <= c) && (c <= 'F')) {
jp->hexChar = (jp->hexChar << 4) | (c - 'A' + 0x0a);
} else {
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL, JSMSG_JSON_BAD_PARSE);
return JS_FALSE;
}
if (++(jp->numHex) == 4) {
js_FastAppendChar(&jp->buffer, jp->hexChar);
jp->hexChar = 0;
jp->numHex = 0;
*jp->statep = JSON_PARSE_STATE_STRING;
}
break;
2008-10-08 10:31:07 -07:00
case JSON_PARSE_STATE_KEYWORD:
if (JS7_ISLET(c)) {
js_FastAppendChar(&jp->buffer, c);
} else {
// this character isn't part of the keyword, process it again
i--;
if (!PopState(cx, jp))
return JS_FALSE;
if (!HandleData(cx, jp, JSON_DATA_KEYWORD))
return JS_FALSE;
}
break;
2008-10-08 10:31:07 -07:00
case JSON_PARSE_STATE_NUMBER:
if (IsNumChar(c)) {
js_FastAppendChar(&jp->buffer, c);
} else {
// this character isn't part of the number, process it again
i--;
if (!PopState(cx, jp))
return JS_FALSE;
if (!HandleData(cx, jp, JSON_DATA_NUMBER))
return JS_FALSE;
}
break;
2008-10-06 13:54:12 -07:00
case JSON_PARSE_STATE_FINISHED:
if (!JS_ISXMLSPACE(c)) {
// extra input
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL, JSMSG_JSON_BAD_PARSE);
return JS_FALSE;
}
break;
2008-10-08 10:31:07 -07:00
default:
JS_NOT_REACHED("Invalid JSON parser state");
}
if (!STRING_BUFFER_OK(&jp->buffer)) {
JS_ReportOutOfMemory(cx);
return JS_FALSE;
}
2008-10-06 13:54:12 -07:00
}
*jp->buffer.ptr = 0;
2008-10-06 13:54:12 -07:00
return JS_TRUE;
}
#if JS_HAS_TOSOURCE
static JSBool
json_toSource(JSContext *cx, uintN argc, jsval *vp)
{
*vp = ATOM_KEY(CLASS_ATOM(cx, JSON));
return JS_TRUE;
}
#endif
static JSFunctionSpec json_static_methods[] = {
#if JS_HAS_TOSOURCE
JS_FN(js_toSource_str, json_toSource, 0, 0),
#endif
JS_FN("parse", js_json_parse, 1, 0),
JS_FN("stringify", js_json_stringify, 1, 0),
2008-10-06 13:54:12 -07:00
JS_FS_END
};
JSObject *
js_InitJSONClass(JSContext *cx, JSObject *obj)
{
JSObject *JSON;
JSON = JS_NewObject(cx, &js_JSONClass, NULL, obj);
if (!JSON)
return NULL;
if (!JS_DefineProperty(cx, obj, js_JSON_str, OBJECT_TO_JSVAL(JSON),
JS_PropertyStub, JS_PropertyStub, 0))
2008-10-06 13:54:12 -07:00
return NULL;
if (!JS_DefineFunctions(cx, JSON, json_static_methods))
return NULL;
return JSON;
}