mirror of
https://github.com/neovim/neovim.git
synced 2024-12-20 11:15:14 -07:00
737f58e232
In the api_info() output: :new|put =map(filter(api_info().functions, '!has_key(v:val,''deprecated_since'')'), 'v:val') ... {'return_type': 'ArrayOf(Integer, 2)', 'name': 'nvim_win_get_position', 'method': v:true, 'parameters': [['Window', 'window']], 'since': 1} The `ArrayOf(Integer, 2)` return type didn't break clients when we added it, which is evidence that clients don't use the `return_type` field, thus renaming Dictionary => Dict in api_info() is not (in practice) a breaking change.
929 lines
33 KiB
Lua
929 lines
33 KiB
Lua
local t = require('test.testutil')
|
|
local n = require('test.functional.testnvim')()
|
|
|
|
local clear = n.clear
|
|
local fn = n.fn
|
|
local api = n.api
|
|
local eq = t.eq
|
|
local eval = n.eval
|
|
local command = n.command
|
|
local exc_exec = n.exc_exec
|
|
local pcall_err = t.pcall_err
|
|
local NIL = vim.NIL
|
|
local source = n.source
|
|
|
|
describe('json_decode() function', function()
|
|
local restart = function(...)
|
|
clear(...)
|
|
source([[
|
|
language C
|
|
function Eq(exp, act)
|
|
let act = a:act
|
|
let exp = a:exp
|
|
if type(exp) != type(act)
|
|
return 0
|
|
endif
|
|
if type(exp) == type({})
|
|
if sort(keys(exp)) !=# sort(keys(act))
|
|
return 0
|
|
endif
|
|
if sort(keys(exp)) ==# ['_TYPE', '_VAL']
|
|
let exp_typ = v:msgpack_types[exp._TYPE]
|
|
let act_typ = act._TYPE
|
|
if exp_typ isnot act_typ
|
|
return 0
|
|
endif
|
|
return Eq(exp._VAL, act._VAL)
|
|
else
|
|
return empty(filter(copy(exp), '!Eq(v:val, act[v:key])'))
|
|
endif
|
|
else
|
|
if type(exp) == type([])
|
|
if len(exp) != len(act)
|
|
return 0
|
|
endif
|
|
return empty(filter(copy(exp), '!Eq(v:val, act[v:key])'))
|
|
endif
|
|
return exp ==# act
|
|
endif
|
|
return 1
|
|
endfunction
|
|
function EvalEq(exp, act_expr)
|
|
let act = eval(a:act_expr)
|
|
if Eq(a:exp, act)
|
|
return 1
|
|
else
|
|
return string(act)
|
|
endif
|
|
endfunction
|
|
]])
|
|
end
|
|
before_each(restart)
|
|
|
|
local speq = function(expected, actual_expr)
|
|
eq(1, fn.EvalEq(expected, actual_expr))
|
|
end
|
|
|
|
it('accepts readfile()-style list', function()
|
|
eq(
|
|
{ Test = 1 },
|
|
fn.json_decode({
|
|
'{',
|
|
'\t"Test": 1',
|
|
'}',
|
|
})
|
|
)
|
|
end)
|
|
|
|
it('accepts strings with newlines', function()
|
|
eq(
|
|
{ Test = 1 },
|
|
fn.json_decode([[
|
|
{
|
|
"Test": 1
|
|
}
|
|
]])
|
|
)
|
|
end)
|
|
|
|
it('parses null, true, false', function()
|
|
eq(NIL, fn.json_decode('null'))
|
|
eq(true, fn.json_decode('true'))
|
|
eq(false, fn.json_decode('false'))
|
|
end)
|
|
|
|
it('fails to parse incomplete null, true, false', function()
|
|
eq('Vim(call):E474: Expected null: n', exc_exec('call json_decode("n")'))
|
|
eq('Vim(call):E474: Expected null: nu', exc_exec('call json_decode("nu")'))
|
|
eq('Vim(call):E474: Expected null: nul', exc_exec('call json_decode("nul")'))
|
|
eq('Vim(call):E474: Expected null: nul\n\t', exc_exec('call json_decode("nul\\n\\t")'))
|
|
|
|
eq('Vim(call):E474: Expected true: t', exc_exec('call json_decode("t")'))
|
|
eq('Vim(call):E474: Expected true: tr', exc_exec('call json_decode("tr")'))
|
|
eq('Vim(call):E474: Expected true: tru', exc_exec('call json_decode("tru")'))
|
|
eq('Vim(call):E474: Expected true: tru\t\n', exc_exec('call json_decode("tru\\t\\n")'))
|
|
|
|
eq('Vim(call):E474: Expected false: f', exc_exec('call json_decode("f")'))
|
|
eq('Vim(call):E474: Expected false: fa', exc_exec('call json_decode("fa")'))
|
|
eq('Vim(call):E474: Expected false: fal', exc_exec('call json_decode("fal")'))
|
|
eq('Vim(call):E474: Expected false: fal <', exc_exec('call json_decode(" fal <")'))
|
|
eq('Vim(call):E474: Expected false: fals', exc_exec('call json_decode("fals")'))
|
|
end)
|
|
|
|
it('parses integer numbers', function()
|
|
eq(100000, fn.json_decode('100000'))
|
|
eq(-100000, fn.json_decode('-100000'))
|
|
eq(100000, fn.json_decode(' 100000 '))
|
|
eq(-100000, fn.json_decode(' -100000 '))
|
|
eq(0, fn.json_decode('0'))
|
|
eq(0, fn.json_decode('-0'))
|
|
end)
|
|
|
|
it('fails to parse +numbers and .number', function()
|
|
eq('Vim(call):E474: Unidentified byte: +1000', exc_exec('call json_decode("+1000")'))
|
|
eq('Vim(call):E474: Unidentified byte: .1000', exc_exec('call json_decode(".1000")'))
|
|
end)
|
|
|
|
it('fails to parse numbers with leading zeroes', function()
|
|
eq('Vim(call):E474: Leading zeroes are not allowed: 00.1', exc_exec('call json_decode("00.1")'))
|
|
eq('Vim(call):E474: Leading zeroes are not allowed: 01', exc_exec('call json_decode("01")'))
|
|
eq('Vim(call):E474: Leading zeroes are not allowed: -01', exc_exec('call json_decode("-01")'))
|
|
eq(
|
|
'Vim(call):E474: Leading zeroes are not allowed: -001.0',
|
|
exc_exec('call json_decode("-001.0")')
|
|
)
|
|
end)
|
|
|
|
it('fails to parse incomplete numbers', function()
|
|
eq('Vim(call):E474: Missing number after minus sign: -.1', exc_exec('call json_decode("-.1")'))
|
|
eq('Vim(call):E474: Missing number after minus sign: -', exc_exec('call json_decode("-")'))
|
|
eq('Vim(call):E474: Missing number after decimal dot: -1.', exc_exec('call json_decode("-1.")'))
|
|
eq('Vim(call):E474: Missing number after decimal dot: 0.', exc_exec('call json_decode("0.")'))
|
|
eq('Vim(call):E474: Missing exponent: 0.0e', exc_exec('call json_decode("0.0e")'))
|
|
eq('Vim(call):E474: Missing exponent: 0.0e+', exc_exec('call json_decode("0.0e+")'))
|
|
eq('Vim(call):E474: Missing exponent: 0.0e-', exc_exec('call json_decode("0.0e-")'))
|
|
eq('Vim(call):E474: Missing exponent: 0.0e-', exc_exec('call json_decode("0.0e-")'))
|
|
eq(
|
|
'Vim(call):E474: Missing number after decimal dot: 1.e5',
|
|
exc_exec('call json_decode("1.e5")')
|
|
)
|
|
eq(
|
|
'Vim(call):E474: Missing number after decimal dot: 1.e+5',
|
|
exc_exec('call json_decode("1.e+5")')
|
|
)
|
|
eq(
|
|
'Vim(call):E474: Missing number after decimal dot: 1.e+',
|
|
exc_exec('call json_decode("1.e+")')
|
|
)
|
|
end)
|
|
|
|
it('parses floating-point numbers', function()
|
|
-- Also test method call (->) syntax
|
|
eq('100000.0', eval('"100000.0"->json_decode()->string()'))
|
|
eq(100000.5, fn.json_decode('100000.5'))
|
|
eq(-100000.5, fn.json_decode('-100000.5'))
|
|
eq(-100000.5e50, fn.json_decode('-100000.5e50'))
|
|
eq(100000.5e50, fn.json_decode('100000.5e50'))
|
|
eq(100000.5e50, fn.json_decode('100000.5e+50'))
|
|
eq(-100000.5e-50, fn.json_decode('-100000.5e-50'))
|
|
eq(100000.5e-50, fn.json_decode('100000.5e-50'))
|
|
eq(100000e-50, fn.json_decode('100000e-50'))
|
|
eq(0.5, fn.json_decode('0.5'))
|
|
eq(0.005, fn.json_decode('0.005'))
|
|
eq(0.005, fn.json_decode('0.00500'))
|
|
eq(0.5, fn.json_decode('0.00500e+002'))
|
|
eq(0.00005, fn.json_decode('0.00500e-002'))
|
|
|
|
eq(-0.0, fn.json_decode('-0.0'))
|
|
eq(-0.0, fn.json_decode('-0.0e0'))
|
|
eq(-0.0, fn.json_decode('-0.0e+0'))
|
|
eq(-0.0, fn.json_decode('-0.0e-0'))
|
|
eq(-0.0, fn.json_decode('-0e-0'))
|
|
eq(-0.0, fn.json_decode('-0e-2'))
|
|
eq(-0.0, fn.json_decode('-0e+2'))
|
|
|
|
eq(0.0, fn.json_decode('0.0'))
|
|
eq(0.0, fn.json_decode('0.0e0'))
|
|
eq(0.0, fn.json_decode('0.0e+0'))
|
|
eq(0.0, fn.json_decode('0.0e-0'))
|
|
eq(0.0, fn.json_decode('0e-0'))
|
|
eq(0.0, fn.json_decode('0e-2'))
|
|
eq(0.0, fn.json_decode('0e+2'))
|
|
end)
|
|
|
|
it('fails to parse numbers with spaces inside', function()
|
|
eq(
|
|
'Vim(call):E474: Missing number after minus sign: - 1000',
|
|
exc_exec('call json_decode("- 1000")')
|
|
)
|
|
eq('Vim(call):E474: Missing number after decimal dot: 0. ', exc_exec('call json_decode("0. ")'))
|
|
eq(
|
|
'Vim(call):E474: Missing number after decimal dot: 0. 0',
|
|
exc_exec('call json_decode("0. 0")')
|
|
)
|
|
eq('Vim(call):E474: Missing exponent: 0.0e 1', exc_exec('call json_decode("0.0e 1")'))
|
|
eq('Vim(call):E474: Missing exponent: 0.0e+ 1', exc_exec('call json_decode("0.0e+ 1")'))
|
|
eq('Vim(call):E474: Missing exponent: 0.0e- 1', exc_exec('call json_decode("0.0e- 1")'))
|
|
end)
|
|
|
|
it('fails to parse "," and ":"', function()
|
|
eq('Vim(call):E474: Comma not inside container: , ', exc_exec('call json_decode(" , ")'))
|
|
eq('Vim(call):E474: Colon not inside container: : ', exc_exec('call json_decode(" : ")'))
|
|
end)
|
|
|
|
it('parses empty containers', function()
|
|
eq({}, fn.json_decode('[]'))
|
|
eq('[]', eval('string(json_decode("[]"))'))
|
|
end)
|
|
|
|
it('fails to parse "[" and "{"', function()
|
|
eq('Vim(call):E474: Unexpected end of input: {', exc_exec('call json_decode("{")'))
|
|
eq('Vim(call):E474: Unexpected end of input: [', exc_exec('call json_decode("[")'))
|
|
end)
|
|
|
|
it('fails to parse "}" and "]"', function()
|
|
eq('Vim(call):E474: No container to close: ]', exc_exec('call json_decode("]")'))
|
|
eq('Vim(call):E474: No container to close: }', exc_exec('call json_decode("}")'))
|
|
end)
|
|
|
|
it('fails to parse containers which are closed by different brackets', function()
|
|
eq(
|
|
'Vim(call):E474: Closing dictionary with square bracket: ]',
|
|
exc_exec('call json_decode("{]")')
|
|
)
|
|
eq('Vim(call):E474: Closing list with curly bracket: }', exc_exec('call json_decode("[}")'))
|
|
end)
|
|
|
|
it('fails to parse concat inside container', function()
|
|
eq(
|
|
'Vim(call):E474: Expected comma before list item: []]',
|
|
exc_exec('call json_decode("[[][]]")')
|
|
)
|
|
eq(
|
|
'Vim(call):E474: Expected comma before list item: {}]',
|
|
exc_exec('call json_decode("[{}{}]")')
|
|
)
|
|
eq('Vim(call):E474: Expected comma before list item: ]', exc_exec('call json_decode("[1 2]")'))
|
|
eq(
|
|
'Vim(call):E474: Expected comma before dictionary key: ": 4}',
|
|
exc_exec('call json_decode("{\\"1\\": 2 \\"3\\": 4}")')
|
|
)
|
|
eq(
|
|
'Vim(call):E474: Expected colon before dictionary value: , "3" 4}',
|
|
exc_exec('call json_decode("{\\"1\\" 2, \\"3\\" 4}")')
|
|
)
|
|
end)
|
|
|
|
it('fails to parse containers with leading comma or colon', function()
|
|
eq('Vim(call):E474: Leading comma: ,}', exc_exec('call json_decode("{,}")'))
|
|
eq('Vim(call):E474: Leading comma: ,]', exc_exec('call json_decode("[,]")'))
|
|
eq('Vim(call):E474: Using colon not in dictionary: :]', exc_exec('call json_decode("[:]")'))
|
|
eq('Vim(call):E474: Unexpected colon: :}', exc_exec('call json_decode("{:}")'))
|
|
end)
|
|
|
|
it('fails to parse containers with trailing comma', function()
|
|
eq('Vim(call):E474: Trailing comma: ]', exc_exec('call json_decode("[1,]")'))
|
|
eq('Vim(call):E474: Trailing comma: }', exc_exec('call json_decode("{\\"1\\": 2,}")'))
|
|
end)
|
|
|
|
it('fails to parse dictionaries with missing value', function()
|
|
eq('Vim(call):E474: Expected value after colon: }', exc_exec('call json_decode("{\\"1\\":}")'))
|
|
eq('Vim(call):E474: Expected value: }', exc_exec('call json_decode("{\\"1\\"}")'))
|
|
end)
|
|
|
|
it('fails to parse containers with two commas or colons', function()
|
|
eq(
|
|
'Vim(call):E474: Duplicate comma: , "2": 2}',
|
|
exc_exec('call json_decode("{\\"1\\": 1,, \\"2\\": 2}")')
|
|
)
|
|
eq(
|
|
'Vim(call):E474: Duplicate comma: , "2", 2]',
|
|
exc_exec('call json_decode("[\\"1\\", 1,, \\"2\\", 2]")')
|
|
)
|
|
eq(
|
|
'Vim(call):E474: Duplicate colon: : 2}',
|
|
exc_exec('call json_decode("{\\"1\\": 1, \\"2\\":: 2}")')
|
|
)
|
|
eq(
|
|
'Vim(call):E474: Comma after colon: , 2}',
|
|
exc_exec('call json_decode("{\\"1\\": 1, \\"2\\":, 2}")')
|
|
)
|
|
eq(
|
|
'Vim(call):E474: Unexpected colon: : "2": 2}',
|
|
exc_exec('call json_decode("{\\"1\\": 1,: \\"2\\": 2}")')
|
|
)
|
|
eq(
|
|
'Vim(call):E474: Unexpected colon: :, "2": 2}',
|
|
exc_exec('call json_decode("{\\"1\\": 1:, \\"2\\": 2}")')
|
|
)
|
|
end)
|
|
|
|
it('fails to parse concat of two values', function()
|
|
eq('Vim(call):E474: Trailing characters: []', exc_exec('call json_decode("{}[]")'))
|
|
end)
|
|
|
|
it('parses containers', function()
|
|
eq({ 1 }, fn.json_decode('[1]'))
|
|
eq({ NIL, 1 }, fn.json_decode('[null, 1]'))
|
|
eq({ ['1'] = 2 }, fn.json_decode('{"1": 2}'))
|
|
eq(
|
|
{ ['1'] = 2, ['3'] = { { ['4'] = { ['5'] = { {}, 1 } } } } },
|
|
fn.json_decode('{"1": 2, "3": [{"4": {"5": [[], 1]}}]}')
|
|
)
|
|
end)
|
|
|
|
it('fails to parse incomplete strings', function()
|
|
eq('Vim(call):E474: Expected string end: \t"', exc_exec('call json_decode("\\t\\"")'))
|
|
eq('Vim(call):E474: Expected string end: \t"abc', exc_exec('call json_decode("\\t\\"abc")'))
|
|
eq(
|
|
'Vim(call):E474: Unfinished escape sequence: \t"abc\\',
|
|
exc_exec('call json_decode("\\t\\"abc\\\\")')
|
|
)
|
|
eq(
|
|
'Vim(call):E474: Unfinished unicode escape sequence: \t"abc\\u',
|
|
exc_exec('call json_decode("\\t\\"abc\\\\u")')
|
|
)
|
|
eq(
|
|
'Vim(call):E474: Unfinished unicode escape sequence: \t"abc\\u0',
|
|
exc_exec('call json_decode("\\t\\"abc\\\\u0")')
|
|
)
|
|
eq(
|
|
'Vim(call):E474: Unfinished unicode escape sequence: \t"abc\\u00',
|
|
exc_exec('call json_decode("\\t\\"abc\\\\u00")')
|
|
)
|
|
eq(
|
|
'Vim(call):E474: Unfinished unicode escape sequence: \t"abc\\u000',
|
|
exc_exec('call json_decode("\\t\\"abc\\\\u000")')
|
|
)
|
|
eq(
|
|
'Vim(call):E474: Expected four hex digits after \\u: \\u" ',
|
|
exc_exec('call json_decode("\\t\\"abc\\\\u\\" ")')
|
|
)
|
|
eq(
|
|
'Vim(call):E474: Expected four hex digits after \\u: \\u0" ',
|
|
exc_exec('call json_decode("\\t\\"abc\\\\u0\\" ")')
|
|
)
|
|
eq(
|
|
'Vim(call):E474: Expected four hex digits after \\u: \\u00" ',
|
|
exc_exec('call json_decode("\\t\\"abc\\\\u00\\" ")')
|
|
)
|
|
eq(
|
|
'Vim(call):E474: Expected four hex digits after \\u: \\u000" ',
|
|
exc_exec('call json_decode("\\t\\"abc\\\\u000\\" ")')
|
|
)
|
|
eq(
|
|
'Vim(call):E474: Expected string end: \t"abc\\u0000',
|
|
exc_exec('call json_decode("\\t\\"abc\\\\u0000")')
|
|
)
|
|
end)
|
|
|
|
it('fails to parse unknown escape sequences', function()
|
|
eq(
|
|
'Vim(call):E474: Unknown escape sequence: \\a"',
|
|
exc_exec('call json_decode("\\t\\"\\\\a\\"")')
|
|
)
|
|
end)
|
|
|
|
it('parses strings properly', function()
|
|
eq('\n', fn.json_decode('"\\n"'))
|
|
eq('', fn.json_decode('""'))
|
|
eq('\\/"\t\b\n\r\f', fn.json_decode([["\\\/\"\t\b\n\r\f"]]))
|
|
eq('/a', fn.json_decode([["\/a"]]))
|
|
-- Unicode characters: 2-byte, 3-byte, 4-byte
|
|
eq(
|
|
{
|
|
'«',
|
|
'ફ',
|
|
'\240\144\128\128',
|
|
},
|
|
fn.json_decode({
|
|
'[',
|
|
'"«",',
|
|
'"ફ",',
|
|
'"\240\144\128\128"',
|
|
']',
|
|
})
|
|
)
|
|
end)
|
|
|
|
it('fails on strings with invalid bytes', function()
|
|
eq(
|
|
'Vim(call):E474: Only UTF-8 strings allowed: \255"',
|
|
exc_exec('call json_decode("\\t\\"\\xFF\\"")')
|
|
)
|
|
eq(
|
|
'Vim(call):E474: ASCII control characters cannot be present inside string: ',
|
|
exc_exec('call json_decode(["\\"\\n\\""])')
|
|
)
|
|
-- 0xC2 starts 2-byte unicode character
|
|
eq(
|
|
'Vim(call):E474: Only UTF-8 strings allowed: \194"',
|
|
exc_exec('call json_decode("\\t\\"\\xC2\\"")')
|
|
)
|
|
-- 0xE0 0xAA starts 3-byte unicode character
|
|
eq(
|
|
'Vim(call):E474: Only UTF-8 strings allowed: \224"',
|
|
exc_exec('call json_decode("\\t\\"\\xE0\\"")')
|
|
)
|
|
eq(
|
|
'Vim(call):E474: Only UTF-8 strings allowed: \224\170"',
|
|
exc_exec('call json_decode("\\t\\"\\xE0\\xAA\\"")')
|
|
)
|
|
-- 0xF0 0x90 0x80 starts 4-byte unicode character
|
|
eq(
|
|
'Vim(call):E474: Only UTF-8 strings allowed: \240"',
|
|
exc_exec('call json_decode("\\t\\"\\xF0\\"")')
|
|
)
|
|
eq(
|
|
'Vim(call):E474: Only UTF-8 strings allowed: \240\144"',
|
|
exc_exec('call json_decode("\\t\\"\\xF0\\x90\\"")')
|
|
)
|
|
eq(
|
|
'Vim(call):E474: Only UTF-8 strings allowed: \240\144\128"',
|
|
exc_exec('call json_decode("\\t\\"\\xF0\\x90\\x80\\"")')
|
|
)
|
|
-- 0xF9 0x80 0x80 0x80 starts 5-byte unicode character
|
|
eq(
|
|
'Vim(call):E474: Only UTF-8 strings allowed: \249"',
|
|
exc_exec('call json_decode("\\t\\"\\xF9\\"")')
|
|
)
|
|
eq(
|
|
'Vim(call):E474: Only UTF-8 strings allowed: \249\128"',
|
|
exc_exec('call json_decode("\\t\\"\\xF9\\x80\\"")')
|
|
)
|
|
eq(
|
|
'Vim(call):E474: Only UTF-8 strings allowed: \249\128\128"',
|
|
exc_exec('call json_decode("\\t\\"\\xF9\\x80\\x80\\"")')
|
|
)
|
|
eq(
|
|
'Vim(call):E474: Only UTF-8 strings allowed: \249\128\128\128"',
|
|
exc_exec('call json_decode("\\t\\"\\xF9\\x80\\x80\\x80\\"")')
|
|
)
|
|
-- 0xFC 0x90 0x80 0x80 0x80 starts 6-byte unicode character
|
|
eq(
|
|
'Vim(call):E474: Only UTF-8 strings allowed: \252"',
|
|
exc_exec('call json_decode("\\t\\"\\xFC\\"")')
|
|
)
|
|
eq(
|
|
'Vim(call):E474: Only UTF-8 strings allowed: \252\144"',
|
|
exc_exec('call json_decode("\\t\\"\\xFC\\x90\\"")')
|
|
)
|
|
eq(
|
|
'Vim(call):E474: Only UTF-8 strings allowed: \252\144\128"',
|
|
exc_exec('call json_decode("\\t\\"\\xFC\\x90\\x80\\"")')
|
|
)
|
|
eq(
|
|
'Vim(call):E474: Only UTF-8 strings allowed: \252\144\128\128"',
|
|
exc_exec('call json_decode("\\t\\"\\xFC\\x90\\x80\\x80\\"")')
|
|
)
|
|
eq(
|
|
'Vim(call):E474: Only UTF-8 strings allowed: \252\144\128\128\128"',
|
|
exc_exec('call json_decode("\\t\\"\\xFC\\x90\\x80\\x80\\x80\\"")')
|
|
)
|
|
-- Specification does not allow unquoted characters above 0x10FFFF
|
|
eq(
|
|
'Vim(call):E474: Only UTF-8 code points up to U+10FFFF are allowed to appear unescaped: \249\128\128\128\128"',
|
|
exc_exec('call json_decode("\\t\\"\\xF9\\x80\\x80\\x80\\x80\\"")')
|
|
)
|
|
eq(
|
|
'Vim(call):E474: Only UTF-8 code points up to U+10FFFF are allowed to appear unescaped: \252\144\128\128\128\128"',
|
|
exc_exec('call json_decode("\\t\\"\\xFC\\x90\\x80\\x80\\x80\\x80\\"")')
|
|
)
|
|
-- '"\249\128\128\128\128"',
|
|
-- '"\252\144\128\128\128\128"',
|
|
end)
|
|
|
|
it('parses surrogate pairs properly', function()
|
|
eq('\240\144\128\128', fn.json_decode('"\\uD800\\uDC00"'))
|
|
eq('\237\160\128a\237\176\128', fn.json_decode('"\\uD800a\\uDC00"'))
|
|
eq('\237\160\128\t\237\176\128', fn.json_decode('"\\uD800\\t\\uDC00"'))
|
|
|
|
eq('\237\160\128', fn.json_decode('"\\uD800"'))
|
|
eq('\237\160\128a', fn.json_decode('"\\uD800a"'))
|
|
eq('\237\160\128\t', fn.json_decode('"\\uD800\\t"'))
|
|
|
|
eq('\237\176\128', fn.json_decode('"\\uDC00"'))
|
|
eq('\237\176\128a', fn.json_decode('"\\uDC00a"'))
|
|
eq('\237\176\128\t', fn.json_decode('"\\uDC00\\t"'))
|
|
|
|
eq('\237\176\128', fn.json_decode('"\\uDC00"'))
|
|
eq('a\237\176\128', fn.json_decode('"a\\uDC00"'))
|
|
eq('\t\237\176\128', fn.json_decode('"\\t\\uDC00"'))
|
|
|
|
eq('\237\160\128¬', fn.json_decode('"\\uD800\\u00AC"'))
|
|
|
|
eq('\237\160\128\237\160\128', fn.json_decode('"\\uD800\\uD800"'))
|
|
end)
|
|
|
|
local sp_decode_eq = function(expected, json)
|
|
api.nvim_set_var('__json', json)
|
|
speq(expected, 'json_decode(g:__json)')
|
|
command('unlet! g:__json')
|
|
end
|
|
|
|
it('parses strings with NUL properly', function()
|
|
sp_decode_eq('\000', '"\\u0000"')
|
|
sp_decode_eq('\000\n\000', '"\\u0000\\n\\u0000"')
|
|
sp_decode_eq('\000«\000', '"\\u0000\\u00AB\\u0000"')
|
|
end)
|
|
|
|
it('parses dictionaries with duplicate keys to special maps', function()
|
|
sp_decode_eq({ _TYPE = 'map', _VAL = { { 'a', 1 }, { 'a', 2 } } }, '{"a": 1, "a": 2}')
|
|
sp_decode_eq(
|
|
{ _TYPE = 'map', _VAL = { { 'b', 3 }, { 'a', 1 }, { 'a', 2 } } },
|
|
'{"b": 3, "a": 1, "a": 2}'
|
|
)
|
|
sp_decode_eq(
|
|
{ _TYPE = 'map', _VAL = { { 'b', 3 }, { 'a', 1 }, { 'c', 4 }, { 'a', 2 } } },
|
|
'{"b": 3, "a": 1, "c": 4, "a": 2}'
|
|
)
|
|
sp_decode_eq(
|
|
{ _TYPE = 'map', _VAL = { { 'b', 3 }, { 'a', 1 }, { 'c', 4 }, { 'a', 2 }, { 'c', 4 } } },
|
|
'{"b": 3, "a": 1, "c": 4, "a": 2, "c": 4}'
|
|
)
|
|
sp_decode_eq(
|
|
{ { _TYPE = 'map', _VAL = { { 'b', 3 }, { 'a', 1 }, { 'c', 4 }, { 'a', 2 }, { 'c', 4 } } } },
|
|
'[{"b": 3, "a": 1, "c": 4, "a": 2, "c": 4}]'
|
|
)
|
|
sp_decode_eq({
|
|
{
|
|
d = {
|
|
_TYPE = 'map',
|
|
_VAL = { { 'b', 3 }, { 'a', 1 }, { 'c', 4 }, { 'a', 2 }, { 'c', 4 } },
|
|
},
|
|
},
|
|
}, '[{"d": {"b": 3, "a": 1, "c": 4, "a": 2, "c": 4}}]')
|
|
sp_decode_eq({
|
|
1,
|
|
{
|
|
d = {
|
|
_TYPE = 'map',
|
|
_VAL = { { 'b', 3 }, { 'a', 1 }, { 'c', 4 }, { 'a', 2 }, { 'c', 4 } },
|
|
},
|
|
},
|
|
}, '[1, {"d": {"b": 3, "a": 1, "c": 4, "a": 2, "c": 4}}]')
|
|
sp_decode_eq({
|
|
1,
|
|
{
|
|
a = {},
|
|
d = {
|
|
_TYPE = 'map',
|
|
_VAL = {
|
|
{ 'b', 3 },
|
|
{ 'a', 1 },
|
|
{ 'c', 4 },
|
|
{ 'a', 2 },
|
|
{
|
|
'c',
|
|
4,
|
|
},
|
|
},
|
|
},
|
|
},
|
|
}, '[1, {"a": [], "d": {"b": 3, "a": 1, "c": 4, "a": 2, "c": 4}}]')
|
|
sp_decode_eq(
|
|
{ _TYPE = 'map', _VAL = { { '', 3 }, { 'a', 1 }, { 'c', 4 }, { 'd', 2 }, { '', 4 } } },
|
|
'{"": 3, "a": 1, "c": 4, "d": 2, "": 4}'
|
|
)
|
|
sp_decode_eq(
|
|
{ { _TYPE = 'map', _VAL = { { '', 3 }, { 'a', 1 }, { 'c', 4 }, { 'd', 2 }, { '', 4 } } } },
|
|
'[{"": 3, "a": 1, "c": 4, "d": 2, "": 4}]'
|
|
)
|
|
end)
|
|
|
|
it('parses dictionaries with empty keys', function()
|
|
eq({ [''] = 4 }, fn.json_decode('{"": 4}'))
|
|
eq(
|
|
{ b = 3, a = 1, c = 4, d = 2, [''] = 4 },
|
|
fn.json_decode('{"b": 3, "a": 1, "c": 4, "d": 2, "": 4}')
|
|
)
|
|
end)
|
|
|
|
it('parses dictionaries with keys with NUL bytes to special maps', function()
|
|
sp_decode_eq({ _TYPE = 'map', _VAL = { { 'a\000\nb', 4 } } }, '{"a\\u0000\\nb": 4}')
|
|
sp_decode_eq({ _TYPE = 'map', _VAL = { { 'a\000\nb\n', 4 } } }, '{"a\\u0000\\nb\\n": 4}')
|
|
sp_decode_eq({
|
|
_TYPE = 'map',
|
|
_VAL = {
|
|
{ 'b', 3 },
|
|
{ 'a', 1 },
|
|
{ 'c', 4 },
|
|
{ 'd', 2 },
|
|
{ '\000', 4 },
|
|
},
|
|
}, '{"b": 3, "a": 1, "c": 4, "d": 2, "\\u0000": 4}')
|
|
end)
|
|
|
|
it('parses U+00C3 correctly', function()
|
|
eq('\195\131', fn.json_decode('"\195\131"'))
|
|
end)
|
|
|
|
it('fails to parse empty string', function()
|
|
eq('Vim(call):E474: Attempt to decode a blank string', exc_exec('call json_decode("")'))
|
|
eq('Vim(call):E474: Attempt to decode a blank string', exc_exec('call json_decode([])'))
|
|
eq('Vim(call):E474: Attempt to decode a blank string', exc_exec('call json_decode([""])'))
|
|
eq('Vim(call):E474: Attempt to decode a blank string', exc_exec('call json_decode(" ")'))
|
|
eq('Vim(call):E474: Attempt to decode a blank string', exc_exec('call json_decode("\\t")'))
|
|
eq('Vim(call):E474: Attempt to decode a blank string', exc_exec('call json_decode("\\n")'))
|
|
eq(
|
|
'Vim(call):E474: Attempt to decode a blank string',
|
|
exc_exec('call json_decode(" \\t\\n \\n\\t\\t \\n\\t\\n \\n \\t\\n\\t ")')
|
|
)
|
|
end)
|
|
|
|
it('accepts all spaces in every position where space may be put', function()
|
|
local s =
|
|
' \t\n\r \t\r\n \n\t\r \n\r\t \r\t\n \r\n\t\t \n\r\t \r\n\t\n \r\t\n\r \t\r \n\t\r\n \n \t\r\n \r\t\n\t \r\n\t\r \n\r \t\n\r\t \r \t\n\r \n\t\r\t \n\r\t\n \r\n \t\r\n\t'
|
|
local str = ('%s{%s"key"%s:%s[%s"val"%s,%s"val2"%s]%s,%s"key2"%s:%s1%s}%s'):gsub('%%s', s)
|
|
eq({ key = { 'val', 'val2' }, key2 = 1 }, fn.json_decode(str))
|
|
end)
|
|
|
|
it('does not overflow when writing error message about decoding ["", ""]', function()
|
|
eq(
|
|
'Vim(call):E474: Attempt to decode a blank string',
|
|
pcall_err(command, 'call json_decode(["", ""])')
|
|
)
|
|
end)
|
|
end)
|
|
|
|
describe('json_encode() function', function()
|
|
before_each(function()
|
|
clear()
|
|
command('language C')
|
|
end)
|
|
|
|
it('dumps strings', function()
|
|
eq('"Test"', fn.json_encode('Test'))
|
|
eq('""', fn.json_encode(''))
|
|
eq('"\\t"', fn.json_encode('\t'))
|
|
eq('"\\n"', fn.json_encode('\n'))
|
|
eq('"\\u001B"', fn.json_encode('\27'))
|
|
eq('"þÿþ"', fn.json_encode('þÿþ'))
|
|
end)
|
|
|
|
it('dumps blobs', function()
|
|
eq('[]', eval('json_encode(0z)'))
|
|
eq('[222, 173, 190, 239]', eval('json_encode(0zDEADBEEF)'))
|
|
end)
|
|
|
|
it('dumps numbers', function()
|
|
eq('0', fn.json_encode(0))
|
|
eq('10', fn.json_encode(10))
|
|
eq('-10', fn.json_encode(-10))
|
|
end)
|
|
|
|
it('dumps floats', function()
|
|
-- Also test method call (->) syntax
|
|
eq('0.0', eval('0.0->json_encode()'))
|
|
eq('10.5', fn.json_encode(10.5))
|
|
eq('-10.5', fn.json_encode(-10.5))
|
|
eq('-1.0e-5', fn.json_encode(-1e-5))
|
|
eq('1.0e50', eval('1.0e50->json_encode()'))
|
|
end)
|
|
|
|
it('fails to dump NaN and infinite values', function()
|
|
eq(
|
|
'Vim(call):E474: Unable to represent NaN value in JSON',
|
|
exc_exec('call json_encode(str2float("nan"))')
|
|
)
|
|
eq(
|
|
'Vim(call):E474: Unable to represent infinity in JSON',
|
|
exc_exec('call json_encode(str2float("inf"))')
|
|
)
|
|
eq(
|
|
'Vim(call):E474: Unable to represent infinity in JSON',
|
|
exc_exec('call json_encode(-str2float("inf"))')
|
|
)
|
|
end)
|
|
|
|
it('dumps lists', function()
|
|
eq('[]', fn.json_encode({}))
|
|
eq('[[]]', fn.json_encode({ {} }))
|
|
eq('[[], []]', fn.json_encode({ {}, {} }))
|
|
end)
|
|
|
|
it('dumps dictionaries', function()
|
|
eq('{}', eval('json_encode({})'))
|
|
eq('{"d": []}', fn.json_encode({ d = {} }))
|
|
eq('{"d": [], "e": []}', fn.json_encode({ d = {}, e = {} }))
|
|
-- Empty keys are allowed per JSON spec (and Vim dicts, and msgpack).
|
|
eq('{"": []}', fn.json_encode({ [''] = {} }))
|
|
end)
|
|
|
|
it('cannot dump generic mapping with generic mapping keys and values', function()
|
|
command('let todump = {"_TYPE": v:msgpack_types.map, "_VAL": []}')
|
|
command('let todumpv1 = {"_TYPE": v:msgpack_types.map, "_VAL": []}')
|
|
command('let todumpv2 = {"_TYPE": v:msgpack_types.map, "_VAL": []}')
|
|
command('call add(todump._VAL, [todumpv1, todumpv2])')
|
|
eq('Vim(call):E474: Invalid key in special dictionary', exc_exec('call json_encode(todump)'))
|
|
end)
|
|
|
|
it('cannot dump generic mapping with ext key', function()
|
|
command('let todump = {"_TYPE": v:msgpack_types.ext, "_VAL": [5, ["",""]]}')
|
|
command('let todump = {"_TYPE": v:msgpack_types.map, "_VAL": [[todump, 1]]}')
|
|
eq('Vim(call):E474: Invalid key in special dictionary', exc_exec('call json_encode(todump)'))
|
|
end)
|
|
|
|
it('cannot dump generic mapping with array key', function()
|
|
command('let todump = {"_TYPE": v:msgpack_types.array, "_VAL": [5, [""]]}')
|
|
command('let todump = {"_TYPE": v:msgpack_types.map, "_VAL": [[todump, 1]]}')
|
|
eq('Vim(call):E474: Invalid key in special dictionary', exc_exec('call json_encode(todump)'))
|
|
end)
|
|
|
|
it('cannot dump generic mapping with UINT64_MAX key', function()
|
|
command('let todump = {"_TYPE": v:msgpack_types.integer}')
|
|
command('let todump._VAL = [1, 3, 0x7FFFFFFF, 0x7FFFFFFF]')
|
|
command('let todump = {"_TYPE": v:msgpack_types.map, "_VAL": [[todump, 1]]}')
|
|
eq('Vim(call):E474: Invalid key in special dictionary', exc_exec('call json_encode(todump)'))
|
|
end)
|
|
|
|
it('cannot dump generic mapping with floating-point key', function()
|
|
command('let todump = {"_TYPE": v:msgpack_types.float, "_VAL": 0.125}')
|
|
command('let todump = {"_TYPE": v:msgpack_types.map, "_VAL": [[todump, 1]]}')
|
|
eq('Vim(call):E474: Invalid key in special dictionary', exc_exec('call json_encode(todump)'))
|
|
end)
|
|
|
|
it('can dump generic mapping with STR special key and NUL', function()
|
|
command('let todump = {"_TYPE": v:msgpack_types.string, "_VAL": ["\\n"]}')
|
|
command('let todump = {"_TYPE": v:msgpack_types.map, "_VAL": [[todump, 1]]}')
|
|
eq('{"\\u0000": 1}', eval('json_encode(todump)'))
|
|
end)
|
|
|
|
it('can dump STR special mapping with NUL and NL', function()
|
|
command('let todump = {"_TYPE": v:msgpack_types.string, "_VAL": ["\\n", ""]}')
|
|
eq('"\\u0000\\n"', eval('json_encode(todump)'))
|
|
end)
|
|
|
|
it('cannot dump special ext mapping', function()
|
|
command('let todump = {"_TYPE": v:msgpack_types.ext, "_VAL": [5, ["",""]]}')
|
|
eq('Vim(call):E474: Unable to convert EXT string to JSON', exc_exec('call json_encode(todump)'))
|
|
end)
|
|
|
|
it('can dump special array mapping', function()
|
|
command('let todump = {"_TYPE": v:msgpack_types.array, "_VAL": [5, [""]]}')
|
|
eq('[5, [""]]', eval('json_encode(todump)'))
|
|
end)
|
|
|
|
it('can dump special UINT64_MAX mapping', function()
|
|
command('let todump = {"_TYPE": v:msgpack_types.integer}')
|
|
command('let todump._VAL = [1, 3, 0x7FFFFFFF, 0x7FFFFFFF]')
|
|
eq('18446744073709551615', eval('json_encode(todump)'))
|
|
end)
|
|
|
|
it('can dump special INT64_MIN mapping', function()
|
|
command('let todump = {"_TYPE": v:msgpack_types.integer}')
|
|
command('let todump._VAL = [-1, 2, 0, 0]')
|
|
eq('-9223372036854775808', eval('json_encode(todump)'))
|
|
end)
|
|
|
|
it('can dump special BOOLEAN true mapping', function()
|
|
command('let todump = {"_TYPE": v:msgpack_types.boolean, "_VAL": 1}')
|
|
eq('true', eval('json_encode(todump)'))
|
|
end)
|
|
|
|
it('can dump special BOOLEAN false mapping', function()
|
|
command('let todump = {"_TYPE": v:msgpack_types.boolean, "_VAL": 0}')
|
|
eq('false', eval('json_encode(todump)'))
|
|
end)
|
|
|
|
it('can dump special NIL mapping', function()
|
|
command('let todump = {"_TYPE": v:msgpack_types.nil, "_VAL": 0}')
|
|
eq('null', eval('json_encode(todump)'))
|
|
end)
|
|
|
|
it('fails to dump a function reference', function()
|
|
eq(
|
|
'Vim(call):E474: Error while dumping encode_tv2json() argument, itself: attempt to dump function reference',
|
|
exc_exec('call json_encode(function("tr"))')
|
|
)
|
|
end)
|
|
|
|
it('fails to dump a partial', function()
|
|
command('function T() dict\nendfunction')
|
|
eq(
|
|
'Vim(call):E474: Error while dumping encode_tv2json() argument, itself: attempt to dump function reference',
|
|
exc_exec('call json_encode(function("T", [1, 2], {}))')
|
|
)
|
|
end)
|
|
|
|
it('fails to dump a function reference in a list', function()
|
|
eq(
|
|
'Vim(call):E474: Error while dumping encode_tv2json() argument, index 0: attempt to dump function reference',
|
|
exc_exec('call json_encode([function("tr")])')
|
|
)
|
|
end)
|
|
|
|
it('fails to dump a recursive list', function()
|
|
command('let todump = [[[]]]')
|
|
command('call add(todump[0][0], todump)')
|
|
eq(
|
|
'Vim(call):E724: unable to correctly dump variable with self-referencing container',
|
|
exc_exec('call json_encode(todump)')
|
|
)
|
|
end)
|
|
|
|
it('fails to dump a recursive dict', function()
|
|
command('let todump = {"d": {"d": {}}}')
|
|
command('call extend(todump.d.d, {"d": todump})')
|
|
eq(
|
|
'Vim(call):E724: unable to correctly dump variable with self-referencing container',
|
|
exc_exec('call json_encode([todump])')
|
|
)
|
|
end)
|
|
|
|
it('can dump dict with two same dicts inside', function()
|
|
command('let inter = {}')
|
|
command('let todump = {"a": inter, "b": inter}')
|
|
eq('{"a": {}, "b": {}}', eval('json_encode(todump)'))
|
|
end)
|
|
|
|
it('can dump list with two same lists inside', function()
|
|
command('let inter = []')
|
|
command('let todump = [inter, inter]')
|
|
eq('[[], []]', eval('json_encode(todump)'))
|
|
end)
|
|
|
|
it('fails to dump a recursive list in a special dict', function()
|
|
command('let todump = {"_TYPE": v:msgpack_types.array, "_VAL": []}')
|
|
command('call add(todump._VAL, todump)')
|
|
eq(
|
|
'Vim(call):E724: unable to correctly dump variable with self-referencing container',
|
|
exc_exec('call json_encode(todump)')
|
|
)
|
|
end)
|
|
|
|
it('fails to dump a recursive (val) map in a special dict', function()
|
|
command('let todump = {"_TYPE": v:msgpack_types.map, "_VAL": []}')
|
|
command('call add(todump._VAL, ["", todump])')
|
|
eq(
|
|
'Vim(call):E724: unable to correctly dump variable with self-referencing container',
|
|
exc_exec('call json_encode([todump])')
|
|
)
|
|
end)
|
|
|
|
it('fails to dump a recursive (val) map in a special dict, _VAL reference', function()
|
|
command('let todump = {"_TYPE": v:msgpack_types.map, "_VAL": [["", []]]}')
|
|
command('call add(todump._VAL[0][1], todump._VAL)')
|
|
eq(
|
|
'Vim(call):E724: unable to correctly dump variable with self-referencing container',
|
|
exc_exec('call json_encode(todump)')
|
|
)
|
|
end)
|
|
|
|
it('fails to dump a recursive (val) special list in a special dict', function()
|
|
command('let todump = {"_TYPE": v:msgpack_types.array, "_VAL": []}')
|
|
command('call add(todump._VAL, ["", todump._VAL])')
|
|
eq(
|
|
'Vim(call):E724: unable to correctly dump variable with self-referencing container',
|
|
exc_exec('call json_encode(todump)')
|
|
)
|
|
end)
|
|
|
|
it('fails when called with no arguments', function()
|
|
eq(
|
|
'Vim(call):E119: Not enough arguments for function: json_encode',
|
|
exc_exec('call json_encode()')
|
|
)
|
|
end)
|
|
|
|
it('fails when called with two arguments', function()
|
|
eq(
|
|
'Vim(call):E118: Too many arguments for function: json_encode',
|
|
exc_exec('call json_encode(["", ""], 1)')
|
|
)
|
|
end)
|
|
|
|
it('ignores improper values in &isprint', function()
|
|
api.nvim_set_option_value('isprint', '1', {})
|
|
eq(1, eval('"\1" =~# "\\\\p"'))
|
|
eq('"\\u0001"', fn.json_encode('\1'))
|
|
end)
|
|
|
|
it('fails when using surrogate character in a UTF-8 string', function()
|
|
eq(
|
|
'Vim(call):E474: UTF-8 string contains code point which belongs to a surrogate pair: \237\160\128',
|
|
exc_exec('call json_encode("\237\160\128")')
|
|
)
|
|
eq(
|
|
'Vim(call):E474: UTF-8 string contains code point which belongs to a surrogate pair: \237\175\191',
|
|
exc_exec('call json_encode("\237\175\191")')
|
|
)
|
|
end)
|
|
|
|
it('dumps control characters as expected', function()
|
|
eq(
|
|
[["\u0000\u0001\u0002\u0003\u0004\u0005\u0006\u0007\b\t\n\u000B\f\r\u000E\u000F\u0010\u0011\u0012\u0013"]],
|
|
eval(
|
|
'json_encode({"_TYPE": v:msgpack_types.string, "_VAL": ["\n\1\2\3\4\5\6\7\8\9", "\11\12\13\14\15\16\17\18\19"]})'
|
|
)
|
|
)
|
|
end)
|
|
|
|
it('can dump NULL string', function()
|
|
eq('""', eval('json_encode($XXX_UNEXISTENT_VAR_XXX)'))
|
|
end)
|
|
|
|
it('can dump NULL blob', function()
|
|
eq('[]', eval('json_encode(v:_null_blob)'))
|
|
end)
|
|
|
|
it('can dump NULL list', function()
|
|
eq('[]', eval('json_encode(v:_null_list)'))
|
|
end)
|
|
|
|
it('can dump NULL dict', function()
|
|
eq('{}', eval('json_encode(v:_null_dict)'))
|
|
end)
|
|
|
|
it('fails to parse NULL strings and lists', function()
|
|
eq(
|
|
'Vim(call):E474: Attempt to decode a blank string',
|
|
exc_exec('call json_decode($XXX_UNEXISTENT_VAR_XXX)')
|
|
)
|
|
eq(
|
|
'Vim(call):E474: Attempt to decode a blank string',
|
|
exc_exec('call json_decode(v:_null_list)')
|
|
)
|
|
end)
|
|
end)
|