mirror of
https://github.com/nim-lang/Nim.git
synced 2026-04-27 09:43:58 +00:00
Add some enhancements to jsonutils.nim (#15133)
* Add some enhancements to `jsonutils.nim`
* Use `jsonutils.nim` hookable API to add a possibility to deserialize
JSON arrays directly to `HashSet` and `OrderedSet` types and
respectively to serialize those types to JSON arrays.
* Also add a possibility to deserialize JSON `null` objects to Nim
option objects and respectively to serialize Nim option object to JSON
object if some or to JSON `null` object if none.
* Move serialization/deserialization functionality for `Table` and
`OrderedTable` types from `jsonutils.nim` to `tables.nim` via the
hookable API.
* Add object `jsonutils.Joptions` and parameter from its type to
`jsonutils.fromJson` procedure to control whether to allow
deserializing JSON objects to Nim objects when the JSON has some
extra or missing keys.
* Add unit tests for the added functionalities to `tjsonutils.nim`.
* improve fromJsonFields
* Add changelog entry for the jsonutils enhancements
* Add TODO in `jsonutils.nim`
* Added an entry to "Future directions" section in `jsonutils.nim` as
suggestion for future support of serialization and de-serialization of
nested variant objects.
* Added currently disabled test case in `tjsonutils.nim` for testing
serialization and de-serialization of nested variant objects.
* Move JSON hooks to `jsonutils.nim`
Move `fromJsonHook` and `toJsonHook` procedures for different types to
`jsonutils.nim` module to avoid a dependency of collections modules to
the `json.nim` module.
The hooks are removed from the following modules:
* `tables.nim`
* `sets.nim`
* `options.nim`
* `strtabs.nim`
* Add some tests about `StringTableRef`
Add tests for `StringTableRef`'s `fromJsonHook` and `toJsonHook` to
`tjsonutils.nim`.
* Disable a warning in `jsonutils.nim`
Mark `fun` template in `jsonutils` module with `{.used.}` pragma in
order to disable `[XDeclaredButNotUsed]` hint. The template is actually
used by the `initCaseObject` macro in the same module.
Co-authored-by: Timothee Cour <timothee.cour2@gmail.com>
This commit is contained in:
@@ -13,8 +13,7 @@ proc testRoundtrip[T](t: T, expected: string) =
|
||||
t2.fromJson(j)
|
||||
doAssert t2.toJson == j
|
||||
|
||||
import tables
|
||||
import strtabs
|
||||
import tables, sets, algorithm, sequtils, options, strtabs
|
||||
|
||||
type Foo = ref object
|
||||
id: int
|
||||
@@ -119,5 +118,187 @@ template fn() =
|
||||
testRoundtrip(Foo[int](t1: false, z2: 7)): """{"t1":false,"z2":7}"""
|
||||
# pending https://github.com/nim-lang/Nim/issues/14698, test with `type Foo[T] = ref object`
|
||||
|
||||
block testHashSet:
|
||||
testRoundtrip(HashSet[string]()): "[]"
|
||||
testRoundtrip([""].toHashSet): """[""]"""
|
||||
testRoundtrip(["one"].toHashSet): """["one"]"""
|
||||
|
||||
var s: HashSet[string]
|
||||
fromJson(s, parseJson("""["one","two"]"""))
|
||||
doAssert s == ["one", "two"].toHashSet
|
||||
|
||||
let jsonNode = toJson(s)
|
||||
doAssert jsonNode.elems.mapIt(it.str).sorted == @["one", "two"]
|
||||
|
||||
block testOrderedSet:
|
||||
testRoundtrip(["one", "two", "three"].toOrderedSet):
|
||||
"""["one","two","three"]"""
|
||||
|
||||
block testOption:
|
||||
testRoundtrip(some("test")): "\"test\""
|
||||
testRoundtrip(none[string]()): "null"
|
||||
testRoundtrip(some(42)): "42"
|
||||
testRoundtrip(none[int]()): "null"
|
||||
|
||||
block testStrtabs:
|
||||
testRoundtrip(newStringTable(modeStyleInsensitive)):
|
||||
"""{"mode":"modeStyleInsensitive","table":{}}"""
|
||||
|
||||
testRoundtrip(
|
||||
newStringTable("name", "John", "surname", "Doe", modeCaseSensitive)):
|
||||
"""{"mode":"modeCaseSensitive","table":{"name":"John","surname":"Doe"}}"""
|
||||
|
||||
block testJoptions:
|
||||
type
|
||||
AboutLifeUniverseAndEverythingElse = object
|
||||
question: string
|
||||
answer: int
|
||||
|
||||
block testExceptionOnExtraKeys:
|
||||
var guide: AboutLifeUniverseAndEverythingElse
|
||||
let json = parseJson(
|
||||
"""{"question":"6*9=?","answer":42,"author":"Douglas Adams"}""")
|
||||
doAssertRaises ValueError, fromJson(guide, json)
|
||||
doAssertRaises ValueError,
|
||||
fromJson(guide, json, Joptions(allowMissingKeys: true))
|
||||
|
||||
type
|
||||
A = object
|
||||
a1,a2,a3: int
|
||||
var a: A
|
||||
let j = parseJson("""{"a3": 1, "a4": 2}""")
|
||||
doAssertRaises ValueError,
|
||||
fromJson(a, j, Joptions(allowMissingKeys: true))
|
||||
|
||||
block testExceptionOnMissingKeys:
|
||||
var guide: AboutLifeUniverseAndEverythingElse
|
||||
let json = parseJson("""{"answer":42}""")
|
||||
doAssertRaises ValueError, fromJson(guide, json)
|
||||
doAssertRaises ValueError,
|
||||
fromJson(guide, json, Joptions(allowExtraKeys: true))
|
||||
|
||||
block testAllowExtraKeys:
|
||||
var guide: AboutLifeUniverseAndEverythingElse
|
||||
let json = parseJson(
|
||||
"""{"question":"6*9=?","answer":42,"author":"Douglas Adams"}""")
|
||||
fromJson(guide, json, Joptions(allowExtraKeys: true))
|
||||
doAssert guide == AboutLifeUniverseAndEverythingElse(
|
||||
question: "6*9=?", answer: 42)
|
||||
|
||||
block testAllowMissingKeys:
|
||||
var guide = AboutLifeUniverseAndEverythingElse(
|
||||
question: "6*9=?", answer: 54)
|
||||
let json = parseJson("""{"answer":42}""")
|
||||
fromJson(guide, json, Joptions(allowMissingKeys: true))
|
||||
doAssert guide == AboutLifeUniverseAndEverythingElse(
|
||||
question: "6*9=?", answer: 42)
|
||||
|
||||
block testAllowExtraAndMissingKeys:
|
||||
var guide = AboutLifeUniverseAndEverythingElse(
|
||||
question: "6*9=?", answer: 54)
|
||||
let json = parseJson(
|
||||
"""{"answer":42,"author":"Douglas Adams"}""")
|
||||
fromJson(guide, json, Joptions(
|
||||
allowExtraKeys: true, allowMissingKeys: true))
|
||||
doAssert guide == AboutLifeUniverseAndEverythingElse(
|
||||
question: "6*9=?", answer: 42)
|
||||
|
||||
type
|
||||
Foo = object
|
||||
a: array[2, string]
|
||||
case b: bool
|
||||
of false: f: float
|
||||
of true: t: tuple[i: int, s: string]
|
||||
case c: range[0 .. 2]
|
||||
of 0: c0: int
|
||||
of 1: c1: float
|
||||
of 2: c2: string
|
||||
|
||||
block testExceptionOnMissingDiscriminantKey:
|
||||
var foo: Foo
|
||||
let json = parseJson("""{"a":["one","two"]}""")
|
||||
doAssertRaises ValueError, fromJson(foo, json)
|
||||
|
||||
block testDoNotResetMissingFieldsWhenHaveDiscriminantKey:
|
||||
var foo = Foo(a: ["one", "two"], b: true, t: (i: 42, s: "s"),
|
||||
c: 0, c0: 1)
|
||||
let json = parseJson("""{"b":true,"c":2}""")
|
||||
fromJson(foo, json, Joptions(allowMissingKeys: true))
|
||||
doAssert foo.a == ["one", "two"]
|
||||
doAssert foo.b
|
||||
doAssert foo.t == (i: 42, s: "s")
|
||||
doAssert foo.c == 2
|
||||
doAssert foo.c2 == ""
|
||||
|
||||
block testAllowMissingDiscriminantKeys:
|
||||
var foo: Foo
|
||||
let json = parseJson("""{"a":["one","two"],"c":1,"c1":3.14159}""")
|
||||
fromJson(foo, json, Joptions(allowMissingKeys: true))
|
||||
doAssert foo.a == ["one", "two"]
|
||||
doAssert not foo.b
|
||||
doAssert foo.f == 0.0
|
||||
doAssert foo.c == 1
|
||||
doAssert foo.c1 == 3.14159
|
||||
|
||||
block testExceptionOnWrongDiscirminatBranchInJson:
|
||||
var foo = Foo(b: false, f: 3.14159, c: 0, c0: 42)
|
||||
let json = parseJson("""{"c2": "hello"}""")
|
||||
doAssertRaises ValueError,
|
||||
fromJson(foo, json, Joptions(allowMissingKeys: true))
|
||||
# Test that the original fields are not reset.
|
||||
doAssert not foo.b
|
||||
doAssert foo.f == 3.14159
|
||||
doAssert foo.c == 0
|
||||
doAssert foo.c0 == 42
|
||||
|
||||
block testNoExceptionOnRightDiscriminantBranchInJson:
|
||||
var foo = Foo(b: false, f: 0, c:1, c1: 0)
|
||||
let json = parseJson("""{"f":2.71828,"c1": 3.14159}""")
|
||||
fromJson(foo, json, Joptions(allowMissingKeys: true))
|
||||
doAssert not foo.b
|
||||
doAssert foo.f == 2.71828
|
||||
doAssert foo.c == 1
|
||||
doAssert foo.c1 == 3.14159
|
||||
|
||||
block testAllowExtraKeysInJsonOnWrongDisciriminatBranch:
|
||||
var foo = Foo(b: false, f: 3.14159, c: 0, c0: 42)
|
||||
let json = parseJson("""{"c2": "hello"}""")
|
||||
fromJson(foo, json, Joptions(allowMissingKeys: true,
|
||||
allowExtraKeys: true))
|
||||
# Test that the original fields are not reset.
|
||||
doAssert not foo.b
|
||||
doAssert foo.f == 3.14159
|
||||
doAssert foo.c == 0
|
||||
doAssert foo.c0 == 42
|
||||
|
||||
when false:
|
||||
## TODO: Implement support for nested variant objects allowing the tests
|
||||
## bellow to pass.
|
||||
block testNestedVariantObjects:
|
||||
type
|
||||
Variant = object
|
||||
case b: bool
|
||||
of false:
|
||||
case bf: bool
|
||||
of false: bff: int
|
||||
of true: bft: float
|
||||
of true:
|
||||
case bt: bool
|
||||
of false: btf: string
|
||||
of true: btt: char
|
||||
|
||||
testRoundtrip(Variant(b: false, bf: false, bff: 42)):
|
||||
"""{"b": false, "bf": false, "bff": 42}"""
|
||||
testRoundtrip(Variant(b: false, bf: true, bft: 3.14159)):
|
||||
"""{"b": false, "bf": true, "bft": 3.14159}"""
|
||||
testRoundtrip(Variant(b: true, bt: false, btf: "test")):
|
||||
"""{"b": true, "bt": false, "btf": "test"}"""
|
||||
testRoundtrip(Variant(b: true, bt: true, btt: 'c')):
|
||||
"""{"b": true, "bt": true, "btt": "c"}"""
|
||||
|
||||
# TODO: Add additional tests with missing and extra JSON keys, both when
|
||||
# allowed and forbidden analogous to the tests for the not nested
|
||||
# variant objects.
|
||||
|
||||
static: fn()
|
||||
fn()
|
||||
|
||||
Reference in New Issue
Block a user