aboutsummaryrefslogtreecommitdiff
path: root/libphobos/src/std/json.d
diff options
context:
space:
mode:
Diffstat (limited to 'libphobos/src/std/json.d')
-rw-r--r--libphobos/src/std/json.d1031
1 files changed, 792 insertions, 239 deletions
diff --git a/libphobos/src/std/json.d b/libphobos/src/std/json.d
index 8ba0f05..39f89a6 100644
--- a/libphobos/src/std/json.d
+++ b/libphobos/src/std/json.d
@@ -6,8 +6,8 @@ JavaScript Object Notation
Copyright: Copyright Jeremie Pelletier 2008 - 2009.
License: $(HTTP www.boost.org/LICENSE_1_0.txt, Boost License 1.0).
Authors: Jeremie Pelletier, David Herberth
-References: $(LINK http://json.org/)
-Source: $(PHOBOSSRC std/_json.d)
+References: $(LINK http://json.org/), $(LINK http://seriot.ch/parsing_json.html)
+Source: $(PHOBOSSRC std/json.d)
*/
/*
Copyright Jeremie Pelletier 2008 - 2009.
@@ -39,7 +39,7 @@ import std.traits;
long x;
if (const(JSONValue)* code = "code" in j)
{
- if (code.type() == JSON_TYPE.INTEGER)
+ if (code.type() == JSONType.integer)
x = code.integer;
else
x = to!int(code.str);
@@ -77,31 +77,47 @@ enum JSONOptions
specialFloatLiterals = 0x1, /// encode NaN and Inf float values as strings
escapeNonAsciiChars = 0x2, /// encode non ascii characters with an unicode escape sequence
doNotEscapeSlashes = 0x4, /// do not escape slashes ('/')
+ strictParsing = 0x8, /// Strictly follow RFC-8259 grammar when parsing
}
/**
JSON type enumeration
*/
-enum JSON_TYPE : byte
+enum JSONType : byte
{
- /// Indicates the type of a $(D JSONValue).
- NULL,
- STRING, /// ditto
- INTEGER, /// ditto
- UINTEGER,/// ditto
- FLOAT, /// ditto
- OBJECT, /// ditto
- ARRAY, /// ditto
- TRUE, /// ditto
- FALSE /// ditto
+ /// Indicates the type of a `JSONValue`.
+ null_,
+ string, /// ditto
+ integer, /// ditto
+ uinteger, /// ditto
+ float_, /// ditto
+ array, /// ditto
+ object, /// ditto
+ true_, /// ditto
+ false_, /// ditto
+ // FIXME: Find some way to deprecate the enum members below, which does NOT
+ // create lots of spam-like deprecation warnings, which can't be fixed
+ // by the user. See discussion on this issue at
+ // https://forum.dlang.org/post/feudrhtxkaxxscwhhhff@forum.dlang.org
+ /* deprecated("Use .null_") */ NULL = null_,
+ /* deprecated("Use .string") */ STRING = string,
+ /* deprecated("Use .integer") */ INTEGER = integer,
+ /* deprecated("Use .uinteger") */ UINTEGER = uinteger,
+ /* deprecated("Use .float_") */ FLOAT = float_,
+ /* deprecated("Use .array") */ ARRAY = array,
+ /* deprecated("Use .object") */ OBJECT = object,
+ /* deprecated("Use .true_") */ TRUE = true_,
+ /* deprecated("Use .false_") */ FALSE = false_,
}
+deprecated("Use JSONType and the new enum member names") alias JSON_TYPE = JSONType;
+
/**
JSON value node
*/
struct JSONValue
{
- import std.exception : enforceEx, enforce;
+ import std.exception : enforce;
union Store
{
@@ -113,12 +129,12 @@ struct JSONValue
JSONValue[] array;
}
private Store store;
- private JSON_TYPE type_tag;
+ private JSONType type_tag;
/**
- Returns the JSON_TYPE of the value stored in this structure.
+ Returns the JSONType of the value stored in this structure.
*/
- @property JSON_TYPE type() const pure nothrow @safe @nogc
+ @property JSONType type() const pure nothrow @safe @nogc
{
return type_tag;
}
@@ -127,23 +143,23 @@ struct JSONValue
{
string s = "{ \"language\": \"D\" }";
JSONValue j = parseJSON(s);
- assert(j.type == JSON_TYPE.OBJECT);
- assert(j["language"].type == JSON_TYPE.STRING);
+ assert(j.type == JSONType.object);
+ assert(j["language"].type == JSONType.string);
}
/***
- * Value getter/setter for $(D JSON_TYPE.STRING).
- * Throws: $(D JSONException) for read access if $(D type) is not
- * $(D JSON_TYPE.STRING).
+ * Value getter/setter for `JSONType.string`.
+ * Throws: `JSONException` for read access if `type` is not
+ * `JSONType.string`.
*/
- @property string str() const pure @trusted
+ @property string str() const pure @trusted return scope
{
- enforce!JSONException(type == JSON_TYPE.STRING,
+ enforce!JSONException(type == JSONType.string,
"JSONValue is not a string");
return store.str;
}
/// ditto
- @property string str(string v) pure nothrow @nogc @safe
+ @property string str(return string v) pure nothrow @nogc @trusted return // TODO make @safe
{
assign(v);
return v;
@@ -162,13 +178,13 @@ struct JSONValue
}
/***
- * Value getter/setter for $(D JSON_TYPE.INTEGER).
- * Throws: $(D JSONException) for read access if $(D type) is not
- * $(D JSON_TYPE.INTEGER).
+ * Value getter/setter for `JSONType.integer`.
+ * Throws: `JSONException` for read access if `type` is not
+ * `JSONType.integer`.
*/
- @property inout(long) integer() inout pure @safe
+ @property long integer() const pure @safe
{
- enforce!JSONException(type == JSON_TYPE.INTEGER,
+ enforce!JSONException(type == JSONType.integer,
"JSONValue is not an integer");
return store.integer;
}
@@ -180,13 +196,13 @@ struct JSONValue
}
/***
- * Value getter/setter for $(D JSON_TYPE.UINTEGER).
- * Throws: $(D JSONException) for read access if $(D type) is not
- * $(D JSON_TYPE.UINTEGER).
+ * Value getter/setter for `JSONType.uinteger`.
+ * Throws: `JSONException` for read access if `type` is not
+ * `JSONType.uinteger`.
*/
- @property inout(ulong) uinteger() inout pure @safe
+ @property ulong uinteger() const pure @safe
{
- enforce!JSONException(type == JSON_TYPE.UINTEGER,
+ enforce!JSONException(type == JSONType.uinteger,
"JSONValue is not an unsigned integer");
return store.uinteger;
}
@@ -198,14 +214,14 @@ struct JSONValue
}
/***
- * Value getter/setter for $(D JSON_TYPE.FLOAT). Note that despite
+ * Value getter/setter for `JSONType.float_`. Note that despite
* the name, this is a $(B 64)-bit `double`, not a 32-bit `float`.
- * Throws: $(D JSONException) for read access if $(D type) is not
- * $(D JSON_TYPE.FLOAT).
+ * Throws: `JSONException` for read access if `type` is not
+ * `JSONType.float_`.
*/
- @property inout(double) floating() inout pure @safe
+ @property double floating() const pure @safe
{
- enforce!JSONException(type == JSON_TYPE.FLOAT,
+ enforce!JSONException(type == JSONType.float_,
"JSONValue is not a floating type");
return store.floating;
}
@@ -217,9 +233,41 @@ struct JSONValue
}
/***
- * Value getter/setter for $(D JSON_TYPE.OBJECT).
- * Throws: $(D JSONException) for read access if $(D type) is not
- * $(D JSON_TYPE.OBJECT).
+ * Value getter/setter for boolean stored in JSON.
+ * Throws: `JSONException` for read access if `this.type` is not
+ * `JSONType.true_` or `JSONType.false_`.
+ */
+ @property bool boolean() const pure @safe
+ {
+ if (type == JSONType.true_) return true;
+ if (type == JSONType.false_) return false;
+
+ throw new JSONException("JSONValue is not a boolean type");
+ }
+ /// ditto
+ @property bool boolean(bool v) pure nothrow @safe @nogc
+ {
+ assign(v);
+ return v;
+ }
+ ///
+ @safe unittest
+ {
+ JSONValue j = true;
+ assert(j.boolean == true);
+
+ j.boolean = false;
+ assert(j.boolean == false);
+
+ j.integer = 12;
+ import std.exception : assertThrown;
+ assertThrown!JSONException(j.boolean);
+ }
+
+ /***
+ * Value getter/setter for `JSONType.object`.
+ * Throws: `JSONException` for read access if `type` is not
+ * `JSONType.object`.
* Note: this is @system because of the following pattern:
---
auto a = &(json.object());
@@ -227,22 +275,22 @@ struct JSONValue
(*a)["hello"] = "world"; // segmentation fault
---
*/
- @property ref inout(JSONValue[string]) object() inout pure @system
+ @property ref inout(JSONValue[string]) object() inout pure @system return
{
- enforce!JSONException(type == JSON_TYPE.OBJECT,
+ enforce!JSONException(type == JSONType.object,
"JSONValue is not an object");
return store.object;
}
/// ditto
- @property JSONValue[string] object(JSONValue[string] v) pure nothrow @nogc @safe
+ @property JSONValue[string] object(return JSONValue[string] v) pure nothrow @nogc @trusted // TODO make @safe
{
assign(v);
return v;
}
/***
- * Value getter for $(D JSON_TYPE.OBJECT).
- * Unlike $(D object), this retrieves the object by value and can be used in @safe code.
+ * Value getter for `JSONType.object`.
+ * Unlike `object`, this retrieves the object by value and can be used in @safe code.
*
* A caveat is that, if the returned value is null, modifications will not be visible:
* ---
@@ -252,20 +300,20 @@ struct JSONValue
* assert("hello" !in json.object);
* ---
*
- * Throws: $(D JSONException) for read access if $(D type) is not
- * $(D JSON_TYPE.OBJECT).
+ * Throws: `JSONException` for read access if `type` is not
+ * `JSONType.object`.
*/
@property inout(JSONValue[string]) objectNoRef() inout pure @trusted
{
- enforce!JSONException(type == JSON_TYPE.OBJECT,
+ enforce!JSONException(type == JSONType.object,
"JSONValue is not an object");
return store.object;
}
/***
- * Value getter/setter for $(D JSON_TYPE.ARRAY).
- * Throws: $(D JSONException) for read access if $(D type) is not
- * $(D JSON_TYPE.ARRAY).
+ * Value getter/setter for `JSONType.array`.
+ * Throws: `JSONException` for read access if `type` is not
+ * `JSONType.array`.
* Note: this is @system because of the following pattern:
---
auto a = &(json.array());
@@ -275,20 +323,20 @@ struct JSONValue
*/
@property ref inout(JSONValue[]) array() inout pure @system
{
- enforce!JSONException(type == JSON_TYPE.ARRAY,
+ enforce!JSONException(type == JSONType.array,
"JSONValue is not an array");
return store.array;
}
/// ditto
- @property JSONValue[] array(JSONValue[] v) pure nothrow @nogc @safe
+ @property JSONValue[] array(return JSONValue[] v) pure nothrow @nogc @trusted scope // TODO make @safe
{
assign(v);
return v;
}
/***
- * Value getter for $(D JSON_TYPE.ARRAY).
- * Unlike $(D array), this retrieves the array by value and can be used in @safe code.
+ * Value getter for `JSONType.array`.
+ * Unlike `array`, this retrieves the array by value and can be used in @safe code.
*
* A caveat is that, if you append to the returned array, the new values aren't visible in the
* JSONValue:
@@ -299,38 +347,137 @@ struct JSONValue
* assert(json.array.length == 1);
* ---
*
- * Throws: $(D JSONException) for read access if $(D type) is not
- * $(D JSON_TYPE.ARRAY).
+ * Throws: `JSONException` for read access if `type` is not
+ * `JSONType.array`.
*/
@property inout(JSONValue[]) arrayNoRef() inout pure @trusted
{
- enforce!JSONException(type == JSON_TYPE.ARRAY,
+ enforce!JSONException(type == JSONType.array,
"JSONValue is not an array");
return store.array;
}
- /// Test whether the type is $(D JSON_TYPE.NULL)
+ /// Test whether the type is `JSONType.null_`
@property bool isNull() const pure nothrow @safe @nogc
{
- return type == JSON_TYPE.NULL;
+ return type == JSONType.null_;
}
- private void assign(T)(T arg) @safe
+ /***
+ * Generic type value getter
+ * A convenience getter that returns this `JSONValue` as the specified D type.
+ * Note: only numeric, `bool`, `string`, `JSONValue[string]` and `JSONValue[]` types are accepted
+ * Throws: `JSONException` if `T` cannot hold the contents of this `JSONValue`
+ * `ConvException` in case of integer overflow when converting to `T`
+ */
+ @property inout(T) get(T)() inout const pure @safe
+ {
+ static if (is(immutable T == immutable string))
+ {
+ return str;
+ }
+ else static if (is(immutable T == immutable bool))
+ {
+ return boolean;
+ }
+ else static if (isFloatingPoint!T)
+ {
+ switch (type)
+ {
+ case JSONType.float_:
+ return cast(T) floating;
+ case JSONType.uinteger:
+ return cast(T) uinteger;
+ case JSONType.integer:
+ return cast(T) integer;
+ default:
+ throw new JSONException("JSONValue is not a number type");
+ }
+ }
+ else static if (isIntegral!T)
+ {
+ switch (type)
+ {
+ case JSONType.uinteger:
+ return uinteger.to!T;
+ case JSONType.integer:
+ return integer.to!T;
+ default:
+ throw new JSONException("JSONValue is not a an integral type");
+ }
+ }
+ else
+ {
+ static assert(false, "Unsupported type");
+ }
+ }
+ // This specialization is needed because arrayNoRef requires inout
+ @property inout(T) get(T : JSONValue[])() inout pure @trusted /// ditto
+ {
+ return arrayNoRef;
+ }
+ /// ditto
+ @property inout(T) get(T : JSONValue[string])() inout pure @trusted
+ {
+ return object;
+ }
+ ///
+ @safe unittest
+ {
+ import std.exception;
+ import std.conv;
+ string s =
+ `{
+ "a": 123,
+ "b": 3.1415,
+ "c": "text",
+ "d": true,
+ "e": [1, 2, 3],
+ "f": { "a": 1 },
+ "g": -45,
+ "h": ` ~ ulong.max.to!string ~ `,
+ }`;
+
+ struct a { }
+
+ immutable json = parseJSON(s);
+ assert(json["a"].get!double == 123.0);
+ assert(json["a"].get!int == 123);
+ assert(json["a"].get!uint == 123);
+ assert(json["b"].get!double == 3.1415);
+ assertThrown!JSONException(json["b"].get!int);
+ assert(json["c"].get!string == "text");
+ assert(json["d"].get!bool == true);
+ assertNotThrown(json["e"].get!(JSONValue[]));
+ assertNotThrown(json["f"].get!(JSONValue[string]));
+ static assert(!__traits(compiles, json["a"].get!a));
+ assertThrown!JSONException(json["e"].get!float);
+ assertThrown!JSONException(json["d"].get!(JSONValue[string]));
+ assertThrown!JSONException(json["f"].get!(JSONValue[]));
+ assert(json["g"].get!int == -45);
+ assertThrown!ConvException(json["g"].get!uint);
+ assert(json["h"].get!ulong == ulong.max);
+ assertThrown!ConvException(json["h"].get!uint);
+ assertNotThrown(json["h"].get!float);
+ }
+
+ private void assign(T)(T arg)
{
static if (is(T : typeof(null)))
{
- type_tag = JSON_TYPE.NULL;
+ type_tag = JSONType.null_;
}
else static if (is(T : string))
{
- type_tag = JSON_TYPE.STRING;
+ type_tag = JSONType.string;
string t = arg;
() @trusted { store.str = t; }();
}
- else static if (isSomeString!T) // issue 15884
+ // https://issues.dlang.org/show_bug.cgi?id=15884
+ else static if (isSomeString!T)
{
- type_tag = JSON_TYPE.STRING;
- // FIXME: std.array.array(Range) is not deduced as 'pure'
+ type_tag = JSONType.string;
+ // FIXME: std.Array.Array(Range) is not deduced as 'pure'
() @trusted {
import std.utf : byUTF;
store.str = cast(immutable)(arg.byUTF!char.array);
@@ -338,27 +485,27 @@ struct JSONValue
}
else static if (is(T : bool))
{
- type_tag = arg ? JSON_TYPE.TRUE : JSON_TYPE.FALSE;
+ type_tag = arg ? JSONType.true_ : JSONType.false_;
}
else static if (is(T : ulong) && isUnsigned!T)
{
- type_tag = JSON_TYPE.UINTEGER;
+ type_tag = JSONType.uinteger;
store.uinteger = arg;
}
else static if (is(T : long))
{
- type_tag = JSON_TYPE.INTEGER;
+ type_tag = JSONType.integer;
store.integer = arg;
}
else static if (isFloatingPoint!T)
{
- type_tag = JSON_TYPE.FLOAT;
+ type_tag = JSONType.float_;
store.floating = arg;
}
else static if (is(T : Value[Key], Key, Value))
{
static assert(is(Key : string), "AA key must be string");
- type_tag = JSON_TYPE.OBJECT;
+ type_tag = JSONType.object;
static if (is(Value : JSONValue))
{
JSONValue[string] t = arg;
@@ -374,7 +521,7 @@ struct JSONValue
}
else static if (isArray!T)
{
- type_tag = JSON_TYPE.ARRAY;
+ type_tag = JSONType.array;
static if (is(ElementEncodingType!T : JSONValue))
{
JSONValue[] t = arg;
@@ -401,7 +548,7 @@ struct JSONValue
private void assignRef(T)(ref T arg) if (isStaticArray!T)
{
- type_tag = JSON_TYPE.ARRAY;
+ type_tag = JSONType.array;
static if (is(ElementEncodingType!T : JSONValue))
{
store.array = arg;
@@ -416,15 +563,15 @@ struct JSONValue
}
/**
- * Constructor for $(D JSONValue). If $(D arg) is a $(D JSONValue)
- * its value and type will be copied to the new $(D JSONValue).
- * Note that this is a shallow copy: if type is $(D JSON_TYPE.OBJECT)
- * or $(D JSON_TYPE.ARRAY) then only the reference to the data will
+ * Constructor for `JSONValue`. If `arg` is a `JSONValue`
+ * its value and type will be copied to the new `JSONValue`.
+ * Note that this is a shallow copy: if type is `JSONType.object`
+ * or `JSONType.array` then only the reference to the data will
* be copied.
- * Otherwise, $(D arg) must be implicitly convertible to one of the
- * following types: $(D typeof(null)), $(D string), $(D ulong),
- * $(D long), $(D double), an associative array $(D V[K]) for any $(D V)
- * and $(D K) i.e. a JSON object, any array or $(D bool). The type will
+ * Otherwise, `arg` must be implicitly convertible to one of the
+ * following types: `typeof(null)`, `string`, `ulong`,
+ * `long`, `double`, an associative array `V[K]` for any `V`
+ * and `K` i.e. a JSON object, any array or `bool`. The type will
* be set accordingly.
*/
this(T)(T arg) if (!isStaticArray!T)
@@ -449,10 +596,10 @@ struct JSONValue
j = JSONValue(42);
j = JSONValue( [1, 2, 3] );
- assert(j.type == JSON_TYPE.ARRAY);
+ assert(j.type == JSONType.array);
j = JSONValue( ["language": "D"] );
- assert(j.type == JSON_TYPE.OBJECT);
+ assert(j.type == JSONType.object);
}
void opAssign(T)(T arg) if (!isStaticArray!T && !is(T : JSONValue))
@@ -467,12 +614,12 @@ struct JSONValue
/***
* Array syntax for json arrays.
- * Throws: $(D JSONException) if $(D type) is not $(D JSON_TYPE.ARRAY).
+ * Throws: `JSONException` if `type` is not `JSONType.array`.
*/
ref inout(JSONValue) opIndex(size_t i) inout pure @safe
{
auto a = this.arrayNoRef;
- enforceEx!JSONException(i < a.length,
+ enforce!JSONException(i < a.length,
"JSONValue array index is out of range");
return a[i];
}
@@ -486,9 +633,9 @@ struct JSONValue
/***
* Hash syntax for json objects.
- * Throws: $(D JSONException) if $(D type) is not $(D JSON_TYPE.OBJECT).
+ * Throws: `JSONException` if `type` is not `JSONType.object`.
*/
- ref inout(JSONValue) opIndex(string k) inout pure @safe
+ ref inout(JSONValue) opIndex(return string k) inout pure @safe
{
auto o = this.objectNoRef;
return *enforce!JSONException(k in o,
@@ -502,20 +649,20 @@ struct JSONValue
}
/***
- * Operator sets $(D value) for element of JSON object by $(D key).
+ * Operator sets `value` for element of JSON object by `key`.
*
* If JSON value is null, then operator initializes it with object and then
- * sets $(D value) for it.
+ * sets `value` for it.
*
- * Throws: $(D JSONException) if $(D type) is not $(D JSON_TYPE.OBJECT)
- * or $(D JSON_TYPE.NULL).
+ * Throws: `JSONException` if `type` is not `JSONType.object`
+ * or `JSONType.null_`.
*/
- void opIndexAssign(T)(auto ref T value, string key) pure
+ void opIndexAssign(T)(auto ref T value, string key)
{
- enforceEx!JSONException(type == JSON_TYPE.OBJECT || type == JSON_TYPE.NULL,
+ enforce!JSONException(type == JSONType.object || type == JSONType.null_,
"JSONValue must be object or null");
JSONValue[string] aa = null;
- if (type == JSON_TYPE.OBJECT)
+ if (type == JSONType.object)
{
aa = this.objectNoRef;
}
@@ -531,10 +678,10 @@ struct JSONValue
assert( j["language"].str == "Perl" );
}
- void opIndexAssign(T)(T arg, size_t i) pure
+ void opIndexAssign(T)(T arg, size_t i)
{
auto a = this.arrayNoRef;
- enforceEx!JSONException(i < a.length,
+ enforce!JSONException(i < a.length,
"JSONValue array index is out of range");
a[i] = arg;
this.array = a;
@@ -547,7 +694,7 @@ struct JSONValue
assert( j[1].str == "D" );
}
- JSONValue opBinary(string op : "~", T)(T arg) @safe
+ JSONValue opBinary(string op : "~", T)(T arg)
{
auto a = this.arrayNoRef;
static if (isArray!T)
@@ -564,7 +711,7 @@ struct JSONValue
}
}
- void opOpAssign(string op : "~", T)(T arg) @safe
+ void opOpAssign(string op : "~", T)(T arg)
{
auto a = this.arrayNoRef;
static if (isArray!T)
@@ -583,16 +730,16 @@ struct JSONValue
}
/**
- * Support for the $(D in) operator.
+ * Support for the `in` operator.
*
* Tests wether a key can be found in an object.
*
* Returns:
- * when found, the $(D const(JSONValue)*) that matches to the key,
- * otherwise $(D null).
+ * when found, the `const(JSONValue)*` that matches to the key,
+ * otherwise `null`.
*
- * Throws: $(D JSONException) if the right hand side argument $(D JSON_TYPE)
- * is not $(D OBJECT).
+ * Throws: `JSONException` if the right hand side argument `JSONType`
+ * is not `object`.
*/
auto opBinaryRight(string op : "in")(string k) const @safe
{
@@ -615,30 +762,67 @@ struct JSONValue
// Default doesn't work well since store is a union. Compare only
// what should be in store.
// This is @trusted to remain nogc, nothrow, fast, and usable from @safe code.
- if (type_tag != rhs.type_tag) return false;
final switch (type_tag)
{
- case JSON_TYPE.STRING:
- return store.str == rhs.store.str;
- case JSON_TYPE.INTEGER:
- return store.integer == rhs.store.integer;
- case JSON_TYPE.UINTEGER:
- return store.uinteger == rhs.store.uinteger;
- case JSON_TYPE.FLOAT:
- return store.floating == rhs.store.floating;
- case JSON_TYPE.OBJECT:
- return store.object == rhs.store.object;
- case JSON_TYPE.ARRAY:
- return store.array == rhs.store.array;
- case JSON_TYPE.TRUE:
- case JSON_TYPE.FALSE:
- case JSON_TYPE.NULL:
- return true;
+ case JSONType.integer:
+ switch (rhs.type_tag)
+ {
+ case JSONType.integer:
+ return store.integer == rhs.store.integer;
+ case JSONType.uinteger:
+ return store.integer == rhs.store.uinteger;
+ case JSONType.float_:
+ return store.integer == rhs.store.floating;
+ default:
+ return false;
+ }
+ case JSONType.uinteger:
+ switch (rhs.type_tag)
+ {
+ case JSONType.integer:
+ return store.uinteger == rhs.store.integer;
+ case JSONType.uinteger:
+ return store.uinteger == rhs.store.uinteger;
+ case JSONType.float_:
+ return store.uinteger == rhs.store.floating;
+ default:
+ return false;
+ }
+ case JSONType.float_:
+ switch (rhs.type_tag)
+ {
+ case JSONType.integer:
+ return store.floating == rhs.store.integer;
+ case JSONType.uinteger:
+ return store.floating == rhs.store.uinteger;
+ case JSONType.float_:
+ return store.floating == rhs.store.floating;
+ default:
+ return false;
+ }
+ case JSONType.string:
+ return type_tag == rhs.type_tag && store.str == rhs.store.str;
+ case JSONType.object:
+ return type_tag == rhs.type_tag && store.object == rhs.store.object;
+ case JSONType.array:
+ return type_tag == rhs.type_tag && store.array == rhs.store.array;
+ case JSONType.true_:
+ case JSONType.false_:
+ case JSONType.null_:
+ return type_tag == rhs.type_tag;
}
}
- /// Implements the foreach $(D opApply) interface for json arrays.
+ ///
+ @safe unittest
+ {
+ assert(JSONValue(0u) == JSONValue(0));
+ assert(JSONValue(0u) == JSONValue(0.0));
+ assert(JSONValue(0) == JSONValue(0.0));
+ }
+
+ /// Implements the foreach `opApply` interface for json arrays.
int opApply(scope int delegate(size_t index, ref JSONValue) dg) @system
{
int result;
@@ -653,10 +837,10 @@ struct JSONValue
return result;
}
- /// Implements the foreach $(D opApply) interface for json objects.
+ /// Implements the foreach `opApply` interface for json objects.
int opApply(scope int delegate(string key, ref JSONValue) dg) @system
{
- enforce!JSONException(type == JSON_TYPE.OBJECT,
+ enforce!JSONException(type == JSONType.object,
"JSONValue is not an object");
int result;
@@ -671,7 +855,7 @@ struct JSONValue
}
/***
- * Implicitly calls $(D toJSON) on this JSONValue.
+ * Implicitly calls `toJSON` on this JSONValue.
*
* $(I options) can be used to tweak the conversion behavior.
*/
@@ -680,8 +864,14 @@ struct JSONValue
return toJSON(this, false, options);
}
+ ///
+ void toString(Out)(Out sink, in JSONOptions options = JSONOptions.none) const
+ {
+ toJSON(sink, this, false, options);
+ }
+
/***
- * Implicitly calls $(D toJSON) on this JSONValue, like $(D toString), but
+ * Implicitly calls `toJSON` on this JSONValue, like `toString`, but
* also passes $(I true) as $(I pretty) argument.
*
* $(I options) can be used to tweak the conversion behavior
@@ -690,11 +880,47 @@ struct JSONValue
{
return toJSON(this, true, options);
}
+
+ ///
+ void toPrettyString(Out)(Out sink, in JSONOptions options = JSONOptions.none) const
+ {
+ toJSON(sink, this, true, options);
+ }
+}
+
+// https://issues.dlang.org/show_bug.cgi?id=20874
+@system unittest
+{
+ static struct MyCustomType
+ {
+ public string toString () const @system { return null; }
+ alias toString this;
+ }
+
+ static struct B
+ {
+ public JSONValue asJSON() const @system { return JSONValue.init; }
+ alias asJSON this;
+ }
+
+ if (false) // Just checking attributes
+ {
+ JSONValue json;
+ MyCustomType ilovedlang;
+ json = ilovedlang;
+ json["foo"] = ilovedlang;
+ auto s = ilovedlang in json;
+
+ B b;
+ json ~= b;
+ json ~ b;
+ }
}
/**
Parses a serialized string and returns a tree of JSON values.
-Throws: $(LREF JSONException) if the depth exceeds the max depth.
+Throws: $(LREF JSONException) if string does not follow the JSON grammar or the depth exceeds the max depth,
+ $(LREF ConvException) if a number in the input cannot be represented by a native D type.
Params:
json = json-formatted string to parse
maxDepth = maximum depth of nesting allowed, -1 disables depth checking
@@ -703,10 +929,10 @@ Params:
JSONValue parseJSON(T)(T json, int maxDepth = -1, JSONOptions options = JSONOptions.none)
if (isInputRange!T && !isInfinite!T && isSomeChar!(ElementEncodingType!T))
{
- import std.ascii : isWhite, isDigit, isHexDigit, toUpper, toLower;
- import std.typecons : Yes;
+ import std.ascii : isDigit, isHexDigit, toUpper, toLower;
+ import std.typecons : Nullable, Yes;
JSONValue root;
- root.type_tag = JSON_TYPE.NULL;
+ root.type_tag = JSONType.null_;
// Avoid UTF decoding when possible, as it is unnecessary when
// processing JSON.
@@ -715,17 +941,37 @@ if (isInputRange!T && !isInfinite!T && isSomeChar!(ElementEncodingType!T))
else
alias Char = Unqual!(ElementType!T);
- if (json.empty) return root;
-
int depth = -1;
- Char next = 0;
+ Nullable!Char next;
int line = 1, pos = 0;
+ immutable bool strict = (options & JSONOptions.strictParsing) != 0;
void error(string msg)
{
throw new JSONException(msg, line, pos);
}
+ if (json.empty)
+ {
+ if (strict)
+ {
+ error("Empty JSON body");
+ }
+ return root;
+ }
+
+ bool isWhite(dchar c)
+ {
+ if (strict)
+ {
+ // RFC 7159 has a stricter definition of whitespace than general ASCII.
+ return c == ' ' || c == '\t' || c == '\n' || c == '\r';
+ }
+ import std.ascii : isWhite;
+ // Accept ASCII NUL as whitespace in non-strict mode.
+ return c == 0 || isWhite(c);
+ }
+
Char popChar()
{
if (json.empty) error("Unexpected end of data.");
@@ -755,17 +1001,35 @@ if (isInputRange!T && !isInfinite!T && isSomeChar!(ElementEncodingType!T))
Char peekChar()
{
- if (!next)
+ if (next.isNull)
{
if (json.empty) return '\0';
next = popChar();
}
+ return next.get;
+ }
+
+ Nullable!Char peekCharNullable()
+ {
+ if (next.isNull && !json.empty)
+ {
+ next = popChar();
+ }
return next;
}
void skipWhitespace()
{
- while (isWhite(peekChar())) next = 0;
+ while (true)
+ {
+ auto c = peekCharNullable();
+ if (c.isNull ||
+ !isWhite(c.get))
+ {
+ return;
+ }
+ next.nullify();
+ }
}
Char getChar(bool SkipWhitespace = false)()
@@ -773,10 +1037,10 @@ if (isInputRange!T && !isInfinite!T && isSomeChar!(ElementEncodingType!T))
static if (SkipWhitespace) skipWhitespace();
Char c;
- if (next)
+ if (!next.isNull)
{
- c = next;
- next = 0;
+ c = next.get;
+ next.nullify();
}
else
c = popChar();
@@ -784,11 +1048,11 @@ if (isInputRange!T && !isInfinite!T && isSomeChar!(ElementEncodingType!T))
return c;
}
- void checkChar(bool SkipWhitespace = true, bool CaseSensitive = true)(char c)
+ void checkChar(bool SkipWhitespace = true)(char c, bool caseSensitive = true)
{
static if (SkipWhitespace) skipWhitespace();
auto c2 = getChar();
- static if (!CaseSensitive) c2 = toLower(c2);
+ if (!caseSensitive) c2 = toLower(c2);
if (c2 != c) error(text("Found '", c2, "' when expecting '", c, "'."));
}
@@ -819,7 +1083,6 @@ if (isInputRange!T && !isInfinite!T && isSomeChar!(ElementEncodingType!T))
string parseString()
{
- import std.ascii : isControl;
import std.uni : isSurrogateHi, isSurrogateLo;
import std.utf : encode, decode;
@@ -880,8 +1143,12 @@ if (isInputRange!T && !isInfinite!T && isSomeChar!(ElementEncodingType!T))
default:
// RFC 7159 states that control characters U+0000 through
// U+001F must not appear unescaped in a JSON string.
+ // Note: std.ascii.isControl can't be used for this test
+ // because it considers ASCII DEL (0x7f) to be a control
+ // character but RFC 7159 does not.
+ // Accept unescaped ASCII NULs in non-strict mode.
auto c = getChar();
- if (isControl(c))
+ if (c < 0x20 && (strict || c != 0))
error("Illegal control character.");
str.put(c);
goto Next;
@@ -927,6 +1194,11 @@ if (isInputRange!T && !isInfinite!T && isSomeChar!(ElementEncodingType!T))
JSONValue[string] obj;
do
{
+ skipWhitespace();
+ if (!strict && peekChar() == '}')
+ {
+ break;
+ }
checkChar('"');
string name = parseString();
checkChar(':');
@@ -943,13 +1215,18 @@ if (isInputRange!T && !isInfinite!T && isSomeChar!(ElementEncodingType!T))
case '[':
if (testChar(']'))
{
- value.type_tag = JSON_TYPE.ARRAY;
+ value.type_tag = JSONType.array;
break;
}
JSONValue[] arr;
do
{
+ skipWhitespace();
+ if (!strict && peekChar() == ']')
+ {
+ break;
+ }
JSONValue element;
parseValue(element);
arr ~= element;
@@ -968,12 +1245,11 @@ if (isInputRange!T && !isInfinite!T && isSomeChar!(ElementEncodingType!T))
tryGetSpecialFloat(str, value.store.floating))
{
// found a special float, its value was placed in value.store.floating
- value.type_tag = JSON_TYPE.FLOAT;
+ value.type_tag = JSONType.float_;
break;
}
- value.type_tag = JSON_TYPE.STRING;
- value.store.str = str;
+ value.assign(str);
break;
case '0': .. case '9':
@@ -1001,7 +1277,18 @@ if (isInputRange!T && !isInfinite!T && isSomeChar!(ElementEncodingType!T))
isNegative = true;
}
- readInteger();
+ if (strict && c == '0')
+ {
+ number.put('0');
+ if (isDigit(peekChar()))
+ {
+ error("Additional digits not allowed after initial zero digit");
+ }
+ }
+ else
+ {
+ readInteger();
+ }
if (testChar('.'))
{
@@ -1023,44 +1310,63 @@ if (isInputRange!T && !isInfinite!T && isSomeChar!(ElementEncodingType!T))
string data = number.data;
if (isFloat)
{
- value.type_tag = JSON_TYPE.FLOAT;
+ value.type_tag = JSONType.float_;
value.store.floating = parse!double(data);
}
else
{
if (isNegative)
+ {
value.store.integer = parse!long(data);
+ value.type_tag = JSONType.integer;
+ }
else
- value.store.uinteger = parse!ulong(data);
-
- value.type_tag = !isNegative && value.store.uinteger & (1UL << 63) ?
- JSON_TYPE.UINTEGER : JSON_TYPE.INTEGER;
+ {
+ // only set the correct union member to not confuse CTFE
+ ulong u = parse!ulong(data);
+ if (u & (1UL << 63))
+ {
+ value.store.uinteger = u;
+ value.type_tag = JSONType.uinteger;
+ }
+ else
+ {
+ value.store.integer = u;
+ value.type_tag = JSONType.integer;
+ }
+ }
}
break;
- case 't':
case 'T':
- value.type_tag = JSON_TYPE.TRUE;
- checkChar!(false, false)('r');
- checkChar!(false, false)('u');
- checkChar!(false, false)('e');
+ if (strict) goto default;
+ goto case;
+ case 't':
+ value.type_tag = JSONType.true_;
+ checkChar!false('r', strict);
+ checkChar!false('u', strict);
+ checkChar!false('e', strict);
break;
- case 'f':
case 'F':
- value.type_tag = JSON_TYPE.FALSE;
- checkChar!(false, false)('a');
- checkChar!(false, false)('l');
- checkChar!(false, false)('s');
- checkChar!(false, false)('e');
+ if (strict) goto default;
+ goto case;
+ case 'f':
+ value.type_tag = JSONType.false_;
+ checkChar!false('a', strict);
+ checkChar!false('l', strict);
+ checkChar!false('s', strict);
+ checkChar!false('e', strict);
break;
- case 'n':
case 'N':
- value.type_tag = JSON_TYPE.NULL;
- checkChar!(false, false)('u');
- checkChar!(false, false)('l');
- checkChar!(false, false)('l');
+ if (strict) goto default;
+ goto case;
+ case 'n':
+ value.type_tag = JSONType.null_;
+ checkChar!false('u', strict);
+ checkChar!false('l', strict);
+ checkChar!false('l', strict);
break;
default:
@@ -1071,16 +1377,21 @@ if (isInputRange!T && !isInfinite!T && isSomeChar!(ElementEncodingType!T))
}
parseValue(root);
+ if (strict)
+ {
+ skipWhitespace();
+ if (!peekCharNullable().isNull) error("Trailing non-whitespace characters");
+ }
return root;
}
@safe unittest
{
enum issue15742objectOfObject = `{ "key1": { "key2": 1 }}`;
- static assert(parseJSON(issue15742objectOfObject).type == JSON_TYPE.OBJECT);
+ static assert(parseJSON(issue15742objectOfObject).type == JSONType.object);
enum issue15742arrayOfArray = `[[1]]`;
- static assert(parseJSON(issue15742arrayOfArray).type == JSON_TYPE.ARRAY);
+ static assert(parseJSON(issue15742arrayOfArray).type == JSONType.array);
}
@safe unittest
@@ -1110,9 +1421,15 @@ if (isInputRange!T && !isInfinite!T && isSomeChar!(ElementEncodingType!T))
assert(json["key1"]["key2"].integer == 1);
}
+// https://issues.dlang.org/show_bug.cgi?id=20527
+@safe unittest
+{
+ static assert(parseJSON(`{"a" : 2}`)["a"].integer == 2);
+}
+
/**
Parses a serialized string and returns a tree of JSON values.
-Throws: $(REF JSONException, std,json) if the depth exceeds the max depth.
+Throws: $(LREF JSONException) if the depth exceeds the max depth.
Params:
json = json-formatted string to parse
options = enable decoding string representations of NaN/Inf as float values
@@ -1128,15 +1445,26 @@ Takes a tree of JSON values and returns the serialized string.
Any Object types will be serialized in a key-sorted order.
-If $(D pretty) is false no whitespaces are generated.
-If $(D pretty) is true serialized string is formatted to be human-readable.
-Set the $(LREF JSONOptions.specialFloatLiterals) flag is set in $(D options) to encode NaN/Infinity as strings.
+If `pretty` is false no whitespaces are generated.
+If `pretty` is true serialized string is formatted to be human-readable.
+Set the $(LREF JSONOptions.specialFloatLiterals) flag is set in `options` to encode NaN/Infinity as strings.
*/
string toJSON(const ref JSONValue root, in bool pretty = false, in JSONOptions options = JSONOptions.none) @safe
{
auto json = appender!string();
+ toJSON(json, root, pretty, options);
+ return json.data;
+}
- void toStringImpl(Char)(string str) @safe
+///
+void toJSON(Out)(
+ auto ref Out json,
+ const ref JSONValue root,
+ in bool pretty = false,
+ in JSONOptions options = JSONOptions.none)
+if (isOutputRange!(Out,char))
+{
+ void toStringImpl(Char)(string str)
{
json.put('"');
@@ -1166,7 +1494,7 @@ string toJSON(const ref JSONValue root, in bool pretty = false, in JSONOptions o
// Make sure we do UTF decoding iff we want to
// escape Unicode characters.
assert(((options & JSONOptions.escapeNonAsciiChars) != 0)
- == is(Char == dchar));
+ == is(Char == dchar), "JSONOptions.escapeNonAsciiChars needs dchar strings");
with (JSONOptions) if (isControl(c) ||
((options & escapeNonAsciiChars) >= escapeNonAsciiChars && c >= 0x80))
@@ -1197,7 +1525,7 @@ string toJSON(const ref JSONValue root, in bool pretty = false, in JSONOptions o
json.put('"');
}
- void toString(string str) @safe
+ void toString(string str)
{
// Avoid UTF decoding when possible, as it is unnecessary when
// processing JSON.
@@ -1207,7 +1535,19 @@ string toJSON(const ref JSONValue root, in bool pretty = false, in JSONOptions o
toStringImpl!char(str);
}
- void toValue(ref in JSONValue value, ulong indentLevel) @safe
+ // recursive @safe inference is broken here
+ // workaround: if json.put is @safe, we should be too,
+ // so annotate the recursion as @safe manually
+ static if (isSafe!({ json.put(""); }))
+ {
+ void delegate(ref const JSONValue, ulong) @safe toValue;
+ }
+ else
+ {
+ void delegate(ref const JSONValue, ulong) @system toValue;
+ }
+
+ void toValueImpl(ref const JSONValue value, ulong indentLevel)
{
void putTabs(ulong additionalIndent = 0)
{
@@ -1228,7 +1568,7 @@ string toJSON(const ref JSONValue root, in bool pretty = false, in JSONOptions o
final switch (value.type)
{
- case JSON_TYPE.OBJECT:
+ case JSONType.object:
auto obj = value.objectNoRef;
if (!obj.length)
{
@@ -1257,7 +1597,7 @@ string toJSON(const ref JSONValue root, in bool pretty = false, in JSONOptions o
}
import std.algorithm.sorting : sort;
- // @@@BUG@@@ 14439
+ // https://issues.dlang.org/show_bug.cgi?id=14439
// auto names = obj.keys; // aa.keys can't be called in @safe code
auto names = new string[obj.length];
size_t i = 0;
@@ -1275,7 +1615,7 @@ string toJSON(const ref JSONValue root, in bool pretty = false, in JSONOptions o
}
break;
- case JSON_TYPE.ARRAY:
+ case JSONType.array:
auto arr = value.arrayNoRef;
if (arr.empty)
{
@@ -1297,20 +1637,20 @@ string toJSON(const ref JSONValue root, in bool pretty = false, in JSONOptions o
}
break;
- case JSON_TYPE.STRING:
+ case JSONType.string:
toString(value.str);
break;
- case JSON_TYPE.INTEGER:
+ case JSONType.integer:
json.put(to!string(value.store.integer));
break;
- case JSON_TYPE.UINTEGER:
+ case JSONType.uinteger:
json.put(to!string(value.store.uinteger));
break;
- case JSON_TYPE.FLOAT:
- import std.math : isNaN, isInfinity;
+ case JSONType.float_:
+ import std.math.traits : isNaN, isInfinity;
auto val = value.store.floating;
@@ -1340,34 +1680,41 @@ string toJSON(const ref JSONValue root, in bool pretty = false, in JSONOptions o
}
else
{
- import std.format : format;
+ import std.algorithm.searching : canFind;
+ import std.format : sformat;
// The correct formula for the number of decimal digits needed for lossless round
// trips is actually:
// ceil(log(pow(2.0, double.mant_dig - 1)) / log(10.0) + 1) == (double.dig + 2)
// Anything less will round off (1 + double.epsilon)
- json.put("%.18g".format(val));
+ char[25] buf;
+ auto result = buf[].sformat!"%.18g"(val);
+ json.put(result);
+ if (!result.canFind('e') && !result.canFind('.'))
+ json.put(".0");
}
break;
- case JSON_TYPE.TRUE:
+ case JSONType.true_:
json.put("true");
break;
- case JSON_TYPE.FALSE:
+ case JSONType.false_:
json.put("false");
break;
- case JSON_TYPE.NULL:
+ case JSONType.null_:
json.put("null");
break;
}
}
+ toValue = &toValueImpl;
+
toValue(root, 0);
- return json.data;
}
-@safe unittest // bugzilla 12897
+ // https://issues.dlang.org/show_bug.cgi?id=12897
+@safe unittest
{
JSONValue jv0 = JSONValue("test测试");
assert(toJSON(jv0, false, JSONOptions.escapeNonAsciiChars) == `"test\u6D4B\u8BD5"`);
@@ -1381,6 +1728,71 @@ string toJSON(const ref JSONValue root, in bool pretty = false, in JSONOptions o
assert(toJSON(jv1, false, JSONOptions.none) == `"été"`);
}
+// https://issues.dlang.org/show_bug.cgi?id=20511
+@system unittest
+{
+ import std.format.write : formattedWrite;
+ import std.range : nullSink, outputRangeObject;
+
+ outputRangeObject!(const(char)[])(nullSink)
+ .formattedWrite!"%s"(JSONValue.init);
+}
+
+// Issue 16432 - JSON incorrectly parses to string
+@safe unittest
+{
+ // Floating points numbers are rounded to the nearest integer and thus get
+ // incorrectly parsed
+
+ import std.math.operations : isClose;
+
+ string s = "{\"rating\": 3.0 }";
+ JSONValue j = parseJSON(s);
+ assert(j["rating"].type == JSONType.float_);
+ j = j.toString.parseJSON;
+ assert(j["rating"].type == JSONType.float_);
+ assert(isClose(j["rating"].floating, 3.0));
+
+ s = "{\"rating\": -3.0 }";
+ j = parseJSON(s);
+ assert(j["rating"].type == JSONType.float_);
+ j = j.toString.parseJSON;
+ assert(j["rating"].type == JSONType.float_);
+ assert(isClose(j["rating"].floating, -3.0));
+
+ // https://issues.dlang.org/show_bug.cgi?id=13660
+ auto jv1 = JSONValue(4.0);
+ auto textual = jv1.toString();
+ auto jv2 = parseJSON(textual);
+ assert(jv1.type == JSONType.float_);
+ assert(textual == "4.0");
+ assert(jv2.type == JSONType.float_);
+}
+
+@safe unittest
+{
+ // Adapted from https://github.com/dlang/phobos/pull/5005
+ // Result from toString is not checked here, because this
+ // might differ (%e-like or %f-like output) depending
+ // on OS and compiler optimization.
+ import std.math.operations : isClose;
+
+ // test positive extreme values
+ JSONValue j;
+ j["rating"] = 1e18 - 65;
+ assert(isClose(j.toString.parseJSON["rating"].floating, 1e18 - 65));
+
+ j["rating"] = 1e18 - 64;
+ assert(isClose(j.toString.parseJSON["rating"].floating, 1e18 - 64));
+
+ // negative extreme values
+ j["rating"] = -1e18 + 65;
+ assert(isClose(j.toString.parseJSON["rating"].floating, -1e18 + 65));
+
+ j["rating"] = -1e18 + 64;
+ assert(isClose(j.toString.parseJSON["rating"].floating, -1e18 + 64));
+}
+
/**
Exception thrown on JSON errors
*/
@@ -1405,7 +1817,7 @@ class JSONException : Exception
{
import std.exception;
JSONValue jv = "123";
- assert(jv.type == JSON_TYPE.STRING);
+ assert(jv.type == JSONType.string);
assertNotThrown(jv.str);
assertThrown!JSONException(jv.integer);
assertThrown!JSONException(jv.uinteger);
@@ -1416,19 +1828,19 @@ class JSONException : Exception
assertThrown!JSONException(jv[2]);
jv = -3;
- assert(jv.type == JSON_TYPE.INTEGER);
+ assert(jv.type == JSONType.integer);
assertNotThrown(jv.integer);
jv = cast(uint) 3;
- assert(jv.type == JSON_TYPE.UINTEGER);
+ assert(jv.type == JSONType.uinteger);
assertNotThrown(jv.uinteger);
jv = 3.0;
- assert(jv.type == JSON_TYPE.FLOAT);
+ assert(jv.type == JSONType.float_);
assertNotThrown(jv.floating);
jv = ["key" : "value"];
- assert(jv.type == JSON_TYPE.OBJECT);
+ assert(jv.type == JSONType.object);
assertNotThrown(jv.object);
assertNotThrown(jv["key"]);
assert("key" in jv);
@@ -1442,82 +1854,81 @@ class JSONException : Exception
{
static assert(is(typeof(value) == JSONValue));
assert(key == "key");
- assert(value.type == JSON_TYPE.STRING);
+ assert(value.type == JSONType.string);
assertNotThrown(value.str);
assert(value.str == "value");
}
jv = [3, 4, 5];
- assert(jv.type == JSON_TYPE.ARRAY);
+ assert(jv.type == JSONType.array);
assertNotThrown(jv.array);
assertNotThrown(jv[2]);
foreach (size_t index, value; jv)
{
static assert(is(typeof(value) == JSONValue));
- assert(value.type == JSON_TYPE.INTEGER);
+ assert(value.type == JSONType.integer);
assertNotThrown(value.integer);
assert(index == (value.integer-3));
}
jv = null;
- assert(jv.type == JSON_TYPE.NULL);
+ assert(jv.type == JSONType.null_);
assert(jv.isNull);
jv = "foo";
assert(!jv.isNull);
jv = JSONValue("value");
- assert(jv.type == JSON_TYPE.STRING);
+ assert(jv.type == JSONType.string);
assert(jv.str == "value");
JSONValue jv2 = JSONValue("value");
- assert(jv2.type == JSON_TYPE.STRING);
+ assert(jv2.type == JSONType.string);
assert(jv2.str == "value");
JSONValue jv3 = JSONValue("\u001c");
- assert(jv3.type == JSON_TYPE.STRING);
+ assert(jv3.type == JSONType.string);
assert(jv3.str == "\u001C");
}
+// https://issues.dlang.org/show_bug.cgi?id=11504
@system unittest
{
- // Bugzilla 11504
-
JSONValue jv = 1;
- assert(jv.type == JSON_TYPE.INTEGER);
+ assert(jv.type == JSONType.integer);
jv.str = "123";
- assert(jv.type == JSON_TYPE.STRING);
+ assert(jv.type == JSONType.string);
assert(jv.str == "123");
jv.integer = 1;
- assert(jv.type == JSON_TYPE.INTEGER);
+ assert(jv.type == JSONType.integer);
assert(jv.integer == 1);
jv.uinteger = 2u;
- assert(jv.type == JSON_TYPE.UINTEGER);
+ assert(jv.type == JSONType.uinteger);
assert(jv.uinteger == 2u);
jv.floating = 1.5;
- assert(jv.type == JSON_TYPE.FLOAT);
+ assert(jv.type == JSONType.float_);
assert(jv.floating == 1.5);
jv.object = ["key" : JSONValue("value")];
- assert(jv.type == JSON_TYPE.OBJECT);
+ assert(jv.type == JSONType.object);
assert(jv.object == ["key" : JSONValue("value")]);
jv.array = [JSONValue(1), JSONValue(2), JSONValue(3)];
- assert(jv.type == JSON_TYPE.ARRAY);
+ assert(jv.type == JSONType.array);
assert(jv.array == [JSONValue(1), JSONValue(2), JSONValue(3)]);
jv = true;
- assert(jv.type == JSON_TYPE.TRUE);
+ assert(jv.type == JSONType.true_);
jv = false;
- assert(jv.type == JSON_TYPE.FALSE);
+ assert(jv.type == JSONType.false_);
enum E{True = true}
jv = E.True;
- assert(jv.type == JSON_TYPE.TRUE);
+ assert(jv.type == JSONType.true_);
}
@system pure unittest
@@ -1653,32 +2064,32 @@ class JSONException : Exception
@system pure unittest
{
- // Bugzilla 12969
+ // https://issues.dlang.org/show_bug.cgi?id=12969
JSONValue jv;
jv["int"] = 123;
- assert(jv.type == JSON_TYPE.OBJECT);
+ assert(jv.type == JSONType.object);
assert("int" in jv);
assert(jv["int"].integer == 123);
jv["array"] = [1, 2, 3, 4, 5];
- assert(jv["array"].type == JSON_TYPE.ARRAY);
+ assert(jv["array"].type == JSONType.array);
assert(jv["array"][2].integer == 3);
jv["str"] = "D language";
- assert(jv["str"].type == JSON_TYPE.STRING);
+ assert(jv["str"].type == JSONType.string);
assert(jv["str"].str == "D language");
jv["bool"] = false;
- assert(jv["bool"].type == JSON_TYPE.FALSE);
+ assert(jv["bool"].type == JSONType.false_);
assert(jv.object.length == 4);
jv = [5, 4, 3, 2, 1];
- assert( jv.type == JSON_TYPE.ARRAY );
- assert( jv[3].integer == 2 );
+ assert(jv.type == JSONType.array);
+ assert(jv[3].integer == 2);
}
@safe unittest
@@ -1702,7 +2113,7 @@ EOF";
@safe unittest
{
import std.exception : assertThrown;
- import std.math : isNaN, isInfinity;
+ import std.math.traits : isNaN, isInfinity;
// expected representations of NaN and Inf
enum {
@@ -1753,28 +2164,30 @@ pure nothrow @safe @nogc unittest
assert(testVal.isNull);
}
-pure nothrow @safe unittest // issue 15884
+// https://issues.dlang.org/show_bug.cgi?id=15884
+pure nothrow @safe unittest
{
import std.typecons;
void Test(C)() {
C[] a = ['x'];
JSONValue testVal = a;
- assert(testVal.type == JSON_TYPE.STRING);
+ assert(testVal.type == JSONType.string);
testVal = a.idup;
- assert(testVal.type == JSON_TYPE.STRING);
+ assert(testVal.type == JSONType.string);
}
Test!char();
Test!wchar();
Test!dchar();
}
-@safe unittest // issue 15885
+// https://issues.dlang.org/show_bug.cgi?id=15885
+@safe unittest
{
enum bool realInDoublePrecision = real.mant_dig == double.mant_dig;
static bool test(const double num0)
{
- import std.math : feqrel;
+ import std.math.operations : feqrel;
const json0 = JSONValue(num0);
const num1 = to!double(toJSON(json0));
static if (realInDoublePrecision)
@@ -1802,34 +2215,39 @@ pure nothrow @safe unittest // issue 15884
assert(test(3*minSub));
}
-@safe unittest // issue 17555
+// https://issues.dlang.org/show_bug.cgi?id=17555
+@safe unittest
{
import std.exception : assertThrown;
assertThrown!JSONException(parseJSON("\"a\nb\""));
}
-@safe unittest // issue 17556
+// https://issues.dlang.org/show_bug.cgi?id=17556
+@safe unittest
{
auto v = JSONValue("\U0001D11E");
auto j = toJSON(v, false, JSONOptions.escapeNonAsciiChars);
assert(j == `"\uD834\uDD1E"`);
}
-@safe unittest // issue 5904
+// https://issues.dlang.org/show_bug.cgi?id=5904
+@safe unittest
{
string s = `"\uD834\uDD1E"`;
auto j = parseJSON(s);
assert(j.str == "\U0001D11E");
}
-@safe unittest // issue 17557
+// https://issues.dlang.org/show_bug.cgi?id=17557
+@safe unittest
{
assert(parseJSON("\"\xFF\"").str == "\xFF");
assert(parseJSON("\"\U0001D11E\"").str == "\U0001D11E");
}
-@safe unittest // issue 17553
+// https://issues.dlang.org/show_bug.cgi?id=17553
+@safe unittest
{
auto v = JSONValue("\xFF");
assert(toJSON(v) == "\"\xFF\"");
@@ -1842,10 +2260,145 @@ pure nothrow @safe unittest // issue 15884
assert(parseJSON("\"\U0001D11E\"".byChar).str == "\U0001D11E");
}
-@safe unittest // JSONOptions.doNotEscapeSlashes (issue 17587)
+// JSONOptions.doNotEscapeSlashes (https://issues.dlang.org/show_bug.cgi?id=17587)
+@safe unittest
{
assert(parseJSON(`"/"`).toString == `"\/"`);
assert(parseJSON(`"\/"`).toString == `"\/"`);
assert(parseJSON(`"/"`).toString(JSONOptions.doNotEscapeSlashes) == `"/"`);
assert(parseJSON(`"\/"`).toString(JSONOptions.doNotEscapeSlashes) == `"/"`);
}
+
+// JSONOptions.strictParsing (https://issues.dlang.org/show_bug.cgi?id=16639)
+@safe unittest
+{
+ import std.exception : assertThrown;
+
+ // Unescaped ASCII NULs
+ assert(parseJSON("[\0]").type == JSONType.array);
+ assertThrown!JSONException(parseJSON("[\0]", JSONOptions.strictParsing));
+ assert(parseJSON("\"\0\"").str == "\0");
+ assertThrown!JSONException(parseJSON("\"\0\"", JSONOptions.strictParsing));
+
+ // Unescaped ASCII DEL (0x7f) in strings
+ assert(parseJSON("\"\x7f\"").str == "\x7f");
+ assert(parseJSON("\"\x7f\"", JSONOptions.strictParsing).str == "\x7f");
+
+ // "true", "false", "null" case sensitivity
+ assert(parseJSON("true").type == JSONType.true_);
+ assert(parseJSON("true", JSONOptions.strictParsing).type == JSONType.true_);
+ assert(parseJSON("True").type == JSONType.true_);
+ assertThrown!JSONException(parseJSON("True", JSONOptions.strictParsing));
+ assert(parseJSON("tRUE").type == JSONType.true_);
+ assertThrown!JSONException(parseJSON("tRUE", JSONOptions.strictParsing));
+
+ assert(parseJSON("false").type == JSONType.false_);
+ assert(parseJSON("false", JSONOptions.strictParsing).type == JSONType.false_);
+ assert(parseJSON("False").type == JSONType.false_);
+ assertThrown!JSONException(parseJSON("False", JSONOptions.strictParsing));
+ assert(parseJSON("fALSE").type == JSONType.false_);
+ assertThrown!JSONException(parseJSON("fALSE", JSONOptions.strictParsing));
+
+ assert(parseJSON("null").type == JSONType.null_);
+ assert(parseJSON("null", JSONOptions.strictParsing).type == JSONType.null_);
+ assert(parseJSON("Null").type == JSONType.null_);
+ assertThrown!JSONException(parseJSON("Null", JSONOptions.strictParsing));
+ assert(parseJSON("nULL").type == JSONType.null_);
+ assertThrown!JSONException(parseJSON("nULL", JSONOptions.strictParsing));
+
+ // Whitespace characters
+ assert(parseJSON("[\f\v]").type == JSONType.array);
+ assertThrown!JSONException(parseJSON("[\f\v]", JSONOptions.strictParsing));
+ assert(parseJSON("[ \t\r\n]").type == JSONType.array);
+ assert(parseJSON("[ \t\r\n]", JSONOptions.strictParsing).type == JSONType.array);
+
+ // Empty input
+ assert(parseJSON("").type == JSONType.null_);
+ assertThrown!JSONException(parseJSON("", JSONOptions.strictParsing));
+
+ // Numbers with leading '0's
+ assert(parseJSON("01").integer == 1);
+ assertThrown!JSONException(parseJSON("01", JSONOptions.strictParsing));
+ assert(parseJSON("-01").integer == -1);
+ assertThrown!JSONException(parseJSON("-01", JSONOptions.strictParsing));
+ assert(parseJSON("0.01").floating == 0.01);
+ assert(parseJSON("0.01", JSONOptions.strictParsing).floating == 0.01);
+ assert(parseJSON("0e1").floating == 0);
+ assert(parseJSON("0e1", JSONOptions.strictParsing).floating == 0);
+
+ // Trailing characters after JSON value
+ assert(parseJSON(`""asdf`).str == "");
+ assertThrown!JSONException(parseJSON(`""asdf`, JSONOptions.strictParsing));
+ assert(parseJSON("987\0").integer == 987);
+ assertThrown!JSONException(parseJSON("987\0", JSONOptions.strictParsing));
+ assert(parseJSON("987\0\0").integer == 987);
+ assertThrown!JSONException(parseJSON("987\0\0", JSONOptions.strictParsing));
+ assert(parseJSON("[]]").type == JSONType.array);
+ assertThrown!JSONException(parseJSON("[]]", JSONOptions.strictParsing));
+ assert(parseJSON("123 \t\r\n").integer == 123); // Trailing whitespace is OK
+ assert(parseJSON("123 \t\r\n", JSONOptions.strictParsing).integer == 123);
+}
+
+@system unittest
+{
+ import std.algorithm.iteration : map;
+ import std.array : array;
+ import std.exception : assertThrown;
+
+ string s = `{ "a" : [1,2,3,], }`;
+ JSONValue j = parseJSON(s);
+ assert(j["a"].array().map!(i => i.integer()).array == [1,2,3]);
+
+ assertThrown(parseJSON(s, -1, JSONOptions.strictParsing));
+}
+
+@system unittest
+{
+ import std.algorithm.iteration : map;
+ import std.array : array;
+ import std.exception : assertThrown;
+
+ string s = `{ "a" : { } , }`;
+ JSONValue j = parseJSON(s);
+ assert("a" in j);
+ auto t = j["a"].object();
+ assert(t.empty);
+
+ assertThrown(parseJSON(s, -1, JSONOptions.strictParsing));
+}
+
+// https://issues.dlang.org/show_bug.cgi?id=20330
+@safe unittest
+{
+ import std.array : appender;
+
+ string s = `{"a":[1,2,3]}`;
+ JSONValue j = parseJSON(s);
+
+ auto app = appender!string();
+ j.toString(app);
+
+ assert(app.data == s, app.data);
+}
+
+// https://issues.dlang.org/show_bug.cgi?id=20330
+@safe unittest
+{
+ import std.array : appender;
+ import std.format.write : formattedWrite;
+
+ string s =
+`{
+ "a": [
+ 1,
+ 2,
+ 3
+ ]
+}`;
+ JSONValue j = parseJSON(s);
+
+ auto app = appender!string();
+ j.toPrettyString(app);
+
+ assert(app.data == s, app.data);
+}