0
0
mirror of https://github.com/vim/vim.git synced 2025-07-26 11:04:33 -04:00

patch 8.0.0171: JS style JSON does not support single quotes

Problem:    JS style JSON does not support single quotes.
Solution:   Allow for single quotes. (Yasuhiro Matsumoto, closes #1371)
This commit is contained in:
Bram Moolenaar 2017-01-11 21:50:08 +01:00
parent e32abbe42c
commit ee142add22
5 changed files with 26 additions and 7 deletions

View File

@ -5229,6 +5229,7 @@ join({list} [, {sep}]) *join()*
js_decode({string}) *js_decode()* js_decode({string}) *js_decode()*
This is similar to |json_decode()| with these differences: This is similar to |json_decode()| with these differences:
- Object key names do not have to be in quotes. - Object key names do not have to be in quotes.
- Strings can be in single quotes.
- Empty items in an array (between two commas) are allowed and - Empty items in an array (between two commas) are allowed and
result in v:none items. result in v:none items.

View File

@ -378,7 +378,7 @@ json_skip_white(js_read_T *reader)
} }
static int static int
json_decode_string(js_read_T *reader, typval_T *res) json_decode_string(js_read_T *reader, typval_T *res, int quote)
{ {
garray_T ga; garray_T ga;
int len; int len;
@ -389,8 +389,8 @@ json_decode_string(js_read_T *reader, typval_T *res)
if (res != NULL) if (res != NULL)
ga_init2(&ga, 1, 200); ga_init2(&ga, 1, 200);
p = reader->js_buf + reader->js_used + 1; /* skip over " */ p = reader->js_buf + reader->js_used + 1; /* skip over " or ' */
while (*p != '"') while (*p != quote)
{ {
/* The JSON is always expected to be utf-8, thus use utf functions /* The JSON is always expected to be utf-8, thus use utf functions
* here. The string is converted below if needed. */ * here. The string is converted below if needed. */
@ -504,7 +504,7 @@ json_decode_string(js_read_T *reader, typval_T *res)
} }
reader->js_used = (int)(p - reader->js_buf); reader->js_used = (int)(p - reader->js_buf);
if (*p == '"') if (*p == quote)
{ {
++reader->js_used; ++reader->js_used;
if (res != NULL) if (res != NULL)
@ -620,7 +620,8 @@ json_decode_item(js_read_T *reader, typval_T *res, int options)
if (top_item != NULL && top_item->jd_type == JSON_OBJECT_KEY if (top_item != NULL && top_item->jd_type == JSON_OBJECT_KEY
&& (options & JSON_JS) && (options & JSON_JS)
&& reader->js_buf[reader->js_used] != '"') && reader->js_buf[reader->js_used] != '"'
&& reader->js_buf[reader->js_used] != '\'')
{ {
char_u *key; char_u *key;
@ -690,7 +691,17 @@ json_decode_item(js_read_T *reader, typval_T *res, int options)
continue; continue;
case '"': /* string */ case '"': /* string */
retval = json_decode_string(reader, cur_item); retval = json_decode_string(reader, cur_item, *p);
break;
case '\'':
if (options & JSON_JS)
retval = json_decode_string(reader, cur_item, *p);
else
{
EMSG(_(e_invarg));
retval = FAIL;
}
break; break;
case ',': /* comma: empty item */ case ',': /* comma: empty item */

View File

@ -181,7 +181,7 @@ test_fill_called_on_string(void)
reader.js_buf = (char_u *)" \"foo"; reader.js_buf = (char_u *)" \"foo";
reader.js_end = reader.js_buf + STRLEN(reader.js_buf); reader.js_end = reader.js_buf + STRLEN(reader.js_buf);
reader.js_cookie = " \"foobar\" "; reader.js_cookie = " \"foobar\" ";
assert(json_decode_string(&reader, NULL) == OK); assert(json_decode_string(&reader, NULL, '"') == OK);
} }
#endif #endif

View File

@ -145,6 +145,8 @@ func Test_json_decode()
call assert_equal("", json_decode('""')) call assert_equal("", json_decode('""'))
call assert_equal({'n': 1}, json_decode('{"n":1,}')) call assert_equal({'n': 1}, json_decode('{"n":1,}'))
call assert_fails("call json_decode(\"{'n':'1',}\")", 'E474:')
call assert_fails("call json_decode(\"'n'\")", 'E474:')
call assert_fails('call json_decode("\"")', "E474:") call assert_fails('call json_decode("\"")', "E474:")
call assert_fails('call json_decode("blah")', "E474:") call assert_fails('call json_decode("blah")', "E474:")
@ -255,8 +257,11 @@ func Test_js_decode()
call assert_equal(v:none, js_decode('')) call assert_equal(v:none, js_decode(''))
call assert_equal(type(v:none), type(js_decode(''))) call assert_equal(type(v:none), type(js_decode('')))
call assert_equal("", js_decode('""')) call assert_equal("", js_decode('""'))
call assert_equal("", js_decode("''"))
call assert_equal('n', js_decode("'n'"))
call assert_equal({'n': 1}, js_decode('{"n":1,}')) call assert_equal({'n': 1}, js_decode('{"n":1,}'))
call assert_equal({'n': '1'}, js_decode("{'n':'1',}"))
call assert_fails('call js_decode("\"")', "E474:") call assert_fails('call js_decode("\"")', "E474:")
call assert_fails('call js_decode("blah")', "E474:") call assert_fails('call js_decode("blah")', "E474:")

View File

@ -764,6 +764,8 @@ static char *(features[]) =
static int included_patches[] = static int included_patches[] =
{ /* Add new patch number below this line */ { /* Add new patch number below this line */
/**/
171,
/**/ /**/
170, 170,
/**/ /**/