| # Tests of json module. |
| |
| load("assert.star", "assert") |
| load("json.star", "json") |
| |
| assert.eq(dir(json), ["decode", "encode", "indent"]) |
| |
| # Some of these cases were inspired by github.com/nst/JSONTestSuite. |
| |
| ## json.encode |
| |
| assert.eq(json.encode(None), "null") |
| assert.eq(json.encode(True), "true") |
| assert.eq(json.encode(False), "false") |
| assert.eq(json.encode(-123), "-123") |
| assert.eq(json.encode(12345*12345*12345*12345*12345*12345), "3539537889086624823140625") |
| assert.eq(json.encode(float(12345*12345*12345*12345*12345*12345)), "3.539537889086625e+24") |
| assert.eq(json.encode(12.345e67), "1.2345e+68") |
| assert.eq(json.encode("hello"), '"hello"') |
| assert.eq(json.encode([1, 2, 3]), "[1,2,3]") |
| assert.eq(json.encode((1, 2, 3)), "[1,2,3]") |
| assert.eq(json.encode(range(3)), "[0,1,2]") # a built-in iterable |
| assert.eq(json.encode(dict(x = 1, y = "two")), '{"x":1,"y":"two"}') |
| assert.eq(json.encode(dict(y = "two", x = 1)), '{"x":1,"y":"two"}') # key, not insertion, order |
| assert.eq(json.encode(struct(x = 1, y = "two")), '{"x":1,"y":"two"}') # a user-defined HasAttrs |
| assert.eq(json.encode("πΉ"[:1]), '"\\ufffd"') # invalid UTF-8 -> replacement char |
| |
| def encode_error(expr, error): |
| assert.fails(lambda: json.encode(expr), error) |
| |
| encode_error(float("NaN"), "json.encode: cannot encode non-finite float nan") |
| encode_error({1: "two"}, "dict has int key, want string") |
| encode_error(len, "cannot encode builtin_function_or_method as JSON") |
| encode_error(struct(x=[1, {"x": len}]), # nested failure |
| 'in field .x: at list index 1: in dict key "x": cannot encode...') |
| encode_error(struct(x=[1, {"x": len}]), # nested failure |
| 'in field .x: at list index 1: in dict key "x": cannot encode...') |
| encode_error({1: 2}, 'dict has int key, want string') |
| |
| recursive_map = {} |
| recursive_map["r"] = recursive_map |
| encode_error(recursive_map, 'json.encode: in dict key "r": cycle in JSON structure') |
| |
| recursive_list = [] |
| recursive_list.append(recursive_list) |
| encode_error(recursive_list, 'json.encode: at list index 0: cycle in JSON structure') |
| |
| recursive_tuple = (1, 2, []) |
| recursive_tuple[2].append(recursive_tuple) |
| encode_error(recursive_tuple, 'json.encode: at tuple index 2: at list index 0: cycle in JSON structure') |
| |
| ## json.decode |
| |
| assert.eq(json.decode("null"), None) |
| assert.eq(json.decode("true"), True) |
| assert.eq(json.decode("false"), False) |
| assert.eq(json.decode("-123"), -123) |
| assert.eq(json.decode("-0"), -0) |
| assert.eq(json.decode("3539537889086624823140625"), 3539537889086624823140625) |
| assert.eq(json.decode("3539537889086624823140625.0"), float(3539537889086624823140625)) |
| assert.eq(json.decode("3.539537889086625e+24"), 3.539537889086625e+24) |
| assert.eq(json.decode("0e+1"), 0) |
| assert.eq(json.decode("-0.0"), -0.0) |
| assert.eq(json.decode( |
| "-0.000000000000000000000000000000000000000000000000000000000000000000000000000001"), |
| -0.000000000000000000000000000000000000000000000000000000000000000000000000000001) |
| assert.eq(json.decode('[]'), []) |
| assert.eq(json.decode('[1]'), [1]) |
| assert.eq(json.decode('[1,2,3]'), [1, 2, 3]) |
| assert.eq(json.decode('{"one": 1, "two": 2}'), dict(one=1, two=2)) |
| assert.eq(json.decode('{"foo\\u0000bar": 42}'), {"foo\x00bar": 42}) |
| assert.eq(json.decode('"\\ud83d\\ude39\\ud83d\\udc8d"'), "πΉπ") |
| assert.eq(json.decode('"\\u0123"'), 'Δ£') |
| assert.eq(json.decode('"\x7f"'), "\x7f") |
| |
| def decode_error(expr, error): |
| assert.fails(lambda: json.decode(expr), error) |
| |
| decode_error('truefalse', |
| "json.decode: at offset 4, unexpected character 'f' after value") |
| |
| decode_error('"abc', "unclosed string literal") |
| decode_error('"ab\\gc"', "invalid character 'g' in string escape code") |
| decode_error("'abc'", "unexpected character '\\\\''") |
| |
| decode_error("1.2.3", "invalid number: 1.2.3") |
| decode_error("+1", "unexpected character '\\+'") |
| decode_error("-abc", "invalid number: -") |
| decode_error("-", "invalid number: -") |
| decode_error("-00", "invalid number: -00") |
| decode_error("00", "invalid number: 00") |
| decode_error("--1", "invalid number: --1") |
| decode_error("-+1", "invalid number: -\\+1") |
| decode_error("1e1e1", "invalid number: 1e1e1") |
| decode_error("0123", "invalid number: 0123") |
| decode_error("000.123", "invalid number: 000.123") |
| decode_error("-0123", "invalid number: -0123") |
| decode_error("-000.123", "invalid number: -000.123") |
| decode_error("0x123", "unexpected character 'x' after value") |
| |
| decode_error('[1, 2 ', "unexpected end of file") |
| decode_error('[1, 2, ', "unexpected end of file") |
| decode_error('[1, 2, ]', "unexpected character ']'") |
| decode_error('[1, 2, }', "unexpected character '}'") |
| decode_error('[1, 2}', "got '}', want ',' or ']'") |
| |
| decode_error('{"one": 1', "unexpected end of file") |
| decode_error('{"one" 1', "after object key, got '1', want ':'") |
| decode_error('{"one": 1 "two": 2', "in object, got '\"', want ',' or '}'") |
| decode_error('{"one": 1,', "unexpected end of file") |
| decode_error('{"one": 1, }', "unexpected character '}'") |
| decode_error('{"one": 1]', "in object, got ']', want ',' or '}'") |
| |
| ## json.decode with default specified |
| |
| assert.eq(json.decode('{"valid": "json"}', default = "default value"), {"valid": "json"}) |
| assert.eq(json.decode('{"valid": "json"}', "default value"), {"valid": "json"}) |
| assert.eq(json.decode('{"invalid": "json"', default = "default value"), "default value") |
| assert.eq(json.decode('{"invalid": "json"', "default value"), "default value") |
| assert.eq(json.decode('{"invalid": "json"', default = None), None) |
| assert.eq(json.decode('{"invalid": "json"', None), None) |
| |
| assert.fails( |
| lambda: json.decode(x = '{"invalid": "json"', default = "default value"), |
| "unexpected keyword argument x" |
| ) |
| |
| def codec(x): |
| return json.decode(json.encode(x)) |
| |
| # string round-tripping |
| strings = [ |
| "πΏ", # U+1F63F CRYING_CAT_FACE |
| "π±βπ€", # CAT FACE + ZERO WIDTH JOINER + BUST IN SILHOUETTE |
| ] |
| assert.eq(codec(strings), strings) |
| |
| # codepoints is a string with every 16-bit code point. |
| codepoints = ''.join(['%c' % c for c in range(65536)]) |
| assert.eq(codec(codepoints), codepoints) |
| |
| # number round-tripping |
| numbers = [ |
| 0, 1, -1, +1, 1.23e45, -1.23e-45, |
| 3539537889086624823140625, |
| float(3539537889086624823140625), |
| ] |
| assert.eq(codec(numbers), numbers) |
| |
| ## json.indent |
| |
| s = json.encode(dict(x = 1, y = ["one", "two"])) |
| |
| assert.eq(json.indent(s), '''{ |
| "x": 1, |
| "y": [ |
| "one", |
| "two" |
| ] |
| }''') |
| |
| assert.eq(json.decode(json.indent(s)), {"x": 1, "y": ["one", "two"]}) |
| |
| assert.eq(json.indent(s, prefix='ΒΆ', indent='βββ'), '''{ |
| ΒΆβββ"x": 1, |
| ΒΆβββ"y": [ |
| ΒΆββββββ"one", |
| ΒΆββββββ"two" |
| ΒΆβββ] |
| ΒΆ}''') |
| |
| assert.fails(lambda: json.indent("!@#$%^& this is not json"), 'invalid character') |
| --- |