/* Copyright (C) 2016, 2017 "IoT.bzh" author: José Bollo Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. */ #include #include "wrap-json.h" #define STACKCOUNT 32 #define STRCOUNT 8 enum { wrap_json_error_none, wrap_json_error_null_object, wrap_json_error_truncated, wrap_json_error_internal_error, wrap_json_error_out_of_memory, wrap_json_error_invalid_character, wrap_json_error_too_long, wrap_json_error_too_deep, wrap_json_error_null_spec, wrap_json_error_null_key, wrap_json_error_null_string, wrap_json_error_out_of_range, wrap_json_error_incomplete, wrap_json_error_missfit_type, wrap_json_error_key_not_found, _wrap_json_error_count_ }; static const char ignore_all[] = " \t\n\r,:"; static const char pack_accept_arr[] = "][{snbiIfoO"; static const char pack_accept_key[] = "s}"; #define pack_accept_any (&pack_accept_arr[1]) static const char unpack_accept_arr[] = "*!][{snbiIfFoO"; static const char unpack_accept_key[] = "*!s}"; #define unpack_accept_any (&unpack_accept_arr[3]) static const char *pack_errors[_wrap_json_error_count_] = { [wrap_json_error_none] = "unknown error", [wrap_json_error_null_object] = "null object", [wrap_json_error_truncated] = "truncated", [wrap_json_error_internal_error] = "internal error", [wrap_json_error_out_of_memory] = "out of memory", [wrap_json_error_invalid_character] = "invalid character", [wrap_json_error_too_long] = "too long", [wrap_json_error_too_deep] = "too deep", [wrap_json_error_null_spec] = "spec is NULL", [wrap_json_error_null_key] = "key is NULL", [wrap_json_error_null_string] = "string is NULL", [wrap_json_error_out_of_range] = "array too small", [wrap_json_error_incomplete] = "incomplete container", [wrap_json_error_missfit_type] = "missfit of type", [wrap_json_error_key_not_found] = "key not found" }; int wrap_json_get_error_position(int rc) { if (rc < 0) rc = -rc; return (rc >> 4) + 1; } int wrap_json_get_error_code(int rc) { if (rc < 0) rc = -rc; return rc & 15; } const char *wrap_json_get_error_string(int rc) { rc = wrap_json_get_error_code(rc); if (rc >= sizeof pack_errors / sizeof *pack_errors) rc = 0; return pack_errors[rc]; } static inline const char *skip(const char *d) { while (*d && strchr(ignore_all, *d)) d++; return d; } int wrap_json_vpack(struct json_object **result, const char *desc, va_list args) { /* TODO: the case of structs with key being single char should be optimized */ int nstr, notnull, nullable, rc; size_t sz, dsz, ssz; char *s; char c; const char *d; char buffer[256]; struct { const char *str; size_t sz; } strs[STRCOUNT]; struct { struct json_object *cont, *key; const char *acc; char type; } stack[STACKCOUNT], *top; struct json_object *obj; ssz = sizeof buffer; s = buffer; top = stack; top->key = NULL; top->cont = NULL; top->acc = pack_accept_any; top->type = 0; d = desc; if (!d) goto null_spec; d = skip(d); for(;;) { c = *d; if (!c) goto truncated; if (!strchr(top->acc, c)) goto invalid_character; d = skip(++d); switch(c) { case 's': nullable = 0; notnull = 0; nstr = 0; sz = 0; for (;;) { strs[nstr].str = va_arg(args, const char*); if (strs[nstr].str) notnull = 1; if (*d == '?') { d = skip(++d); nullable = 1; } switch(*d) { case '%': strs[nstr].sz = va_arg(args, size_t); d = skip(++d); break; case '#': strs[nstr].sz = (size_t)va_arg(args, int); d = skip(++d); break; default: strs[nstr].sz = strs[nstr].str ? strlen(strs[nstr].str) : 0; break; } sz += strs[nstr++].sz; if (*d == '?') { d = skip(++d); nullable = 1; } if (*d != '+') break; if (nstr >= STRCOUNT) goto too_long; d = skip(++d); } if (*d == '*') nullable = 1; if (notnull) { if (sz > ssz) { ssz += ssz; if (ssz < sz) ssz = sz; s = alloca(sz); } dsz = sz; while (nstr) { nstr--; dsz -= strs[nstr].sz; memcpy(&s[dsz], strs[nstr].str, strs[nstr].sz); } obj = json_object_new_string_len(s, (int)sz); if (!obj) goto out_of_memory; } else if (nullable) obj = NULL; else goto null_string; break; case 'n': obj = NULL; break; case 'b': obj = json_object_new_boolean(va_arg(args, int)); if (!obj) goto out_of_memory; break; case 'i': obj = json_object_new_int(va_arg(args, int)); if (!obj) goto out_of_memory; break; case 'I': obj = json_object_new_int64(va_arg(args, int64_t)); if (!obj) goto out_of_memory; break; case 'f': obj = json_object_new_double(va_arg(args, double)); if (!obj) goto out_of_memory; break; case 'o': case 'O': obj = va_arg(args, struct json_object*); if (*d == '?') d = skip(++d); else if (*d != '*' && !obj) goto null_object; if (c == 'O') json_object_get(obj); break; case '[': case '{': if (++top >= &stack[STACKCOUNT]) goto too_deep; top->key = NULL; if (c == '[') { top->type = ']'; top->acc = pack
type: books
books:
-
    id: meta-agl
    title: AGL Layers
    description: Meta AGL documentation
    keywords:
    author: "AGL"
    version: master
    chapters:
    - url: ../agl-layers-overview.md
      name: Overview
    - url: ../meta-agl.md
      destination: meta-agl.md
      name: meta-agl
*xf[10]; struct json_object *xo[10]; size_t xz[10]; void u(const char *value, const char *desc, ...) { unsigned m, k; int rc; va_list args; struct json_object *obj, *o; memset(xs, 0, sizeof xs); memset(xi, 0, sizeof xi); memset(xI, 0, sizeof xI); memset(xf, 0, sizeof xf); memset(xo, 0, sizeof xo); memset(xz, 0, sizeof xz); obj = json_tokener_parse(value); va_start(args, desc); rc = wrap_json_vunpack(obj, desc, args); va_end(args); if (rc) printf(" ERROR[char %d err %d] %s\n\n", wrap_json_get_error_position(rc), wrap_json_get_error_code(rc), wrap_json_get_error_string(rc)); else { value = NULL; printf(" SUCCESS"); va_start(args, desc); k = m = 0; while(*desc) { switch(*desc) { case '{': m = (m << 1) | 1; k = 1; break; case '}': m = m >> 1; k = m&1; break; case '[': m = m << 1; k = 0; break; case ']': m = m >> 1; k = m&1; break; case 's': printf(" s:%s", k ? va_arg(args, const char*) : *(va_arg(args, const char**)?:&value)); k ^= m&1; break; case '%': printf(" %%:%zu", *va_arg(args, size_t*)); k = m&1; break; case 'n': printf(" n"); k = m&1; break; case 'b': printf(" b:%d", *va_arg(args, int*)); k = m&1; break; case 'i': printf(" i:%d", *va_arg(args, int*)); k = m&1; break; case 'I': printf(" I:%lld", *va_arg(args, int64_t*)); k = m&1; break; case 'f': printf(" f:%f", *va_arg(args, double*)); k = m&1; break; case 'F': printf(" F:%f", *va_arg(args, double*)); k = m&1; break; case 'o': printf(" o:%s", json_object_to_json_string(*va_arg(args, struct json_object**))); k = m&1; break; case 'O': o = *va_arg(args, struct json_object**); printf(" O:%s", json_object_to_json_string(o)); json_object_put(o); k = m&1; break; default: break; } desc++; } va_end(args); printf("\n\n"); } json_object_put(obj); } #define P(...) do{ printf("pack(%s)\n",#__VA_ARGS__); p(__VA_ARGS__); } while(0) #define U(...) do{ printf("unpack(%s)\n",#__VA_ARGS__); u(__VA_ARGS__); } while(0) int main() { char buffer[4] = {'t', 'e', 's', 't'}; P("n"); P("b", 1); P("b", 0); P("i", 1); P("I", (uint64_t)0x123456789abcdef); P("f", 3.14); P("s", "test"); P("s?", "test"); P("s?", NULL); P("s#", "test asdf", 4); P("s%", "test asdf", (size_t)4); P("s#", buffer, 4); P("s%", buffer, (size_t)4); P("s++", "te", "st", "ing"); P("s#+#+", "test", 1, "test", 2, "test"); P("s%+%+", "test", (size_t)1, "test", (size_t)2, "test"); P("{}", 1.0); P("[]", 1.0); P("o", json_object_new_int(1)); P("o?", json_object_new_int(1)); P("o?", NULL); P("O", json_object_new_int(1)); P("O?", json_object_new_int(1)); P("O?", NULL); P("{s:[]}", "foo"); P("{s+#+: []}", "foo", "barbar", 3, "baz"); P("{s:s,s:o,s:O}", "a", NULL, "b", NULL, "c", NULL); P("{s:**}", "a", NULL); P("{s:s*,s:o*,s:O*}", "a", NULL, "b", NULL, "c", NULL); P("[i,i,i]", 0, 1, 2); P("[s,o,O]", NULL, NULL, NULL); P("[**]", NULL); P("[s*,o*,O*]", NULL, NULL, NULL); P(" s ", "test"); P("[ ]"); P("[ i , i, i ] ", 1, 2, 3); P("{\n\n1"); P("[}"); P("{]"); P("["); P("{"); P("[i]a", 42); P("ia", 42); P("s", NULL); P("+", NULL); P(NULL); P("{s:i}", NULL, 1); P("{ {}: s }", "foo"); P("{ s: {}, s:[ii{} }", "foo", "bar", 12, 13); P("[[[[[ [[[[[ [[[[ }]]]] ]]]] ]]]]]"); U("true", "b", &xi[0]); U("false", "b", &xi[0]); U("null", "n"); U("42", "i", &xi[0]); U("123456789", "I", &xI[0]); U("3.14", "f", &xf[0]); U("12345", "F", &xf[0]); U("3.14", "F", &xf[0]); U("\"foo\"", "s", &xs[0]); U("\"foo\"", "s%", &xs[0], &xz[0]); U("{}", "{}"); U("[]", "[]"); U("{}", "o", &xo[0]); U("{}", "O", &xo[0]); U("{\"foo\":42}", "{si}", "foo", &xi[0]); U("[1,2,3]", "[i,i,i]", &xi[0], &xi[1], &xi[2]); U("{\"a\":1,\"b\":2,\"c\":3}", "{s:i, s:i, s:i}", "a", &xi[0], "b", &xi[1], "c", &xi[2]); U("42", "z"); U("null", "[i]"); U("[]", "[}"); U("{}", "{]"); U("[]", "["); U("{}", "{"); U("[42]", "[i]a", &xi[0]); U("42", "ia", &xi[0]); U("[]", NULL); U("\"foo\"", "s", NULL); U("42", "s", NULL); U("42", "n"); U("42", "b", NULL); U("42", "f", NULL); U("42", "[i]", NULL); U("42", "{si}", "foo", NULL); U("\"foo\"", "n"); U("\"foo\"", "b", NULL); U("\"foo\"", "i", NULL); U("\"foo\"", "I", NULL); U("\"foo\"", "f", NULL); U("\"foo\"", "F", NULL); U("true", "s", NULL); U("true", "n"); U("true", "i", NULL); U("true", "I", NULL); U("true", "f", NULL); U("true", "F", NULL); U("[42]", "[ii]", &xi[0], &xi[1]); U("{\"foo\":42}", "{si}", NULL, &xi[0]); U("{\"foo\":42}", "{si}", "baz", &xi[0]); U("[1,2,3]", "[iii!]", &xi[0], &xi[1], &xi[2]); U("[1,2,3]", "[ii!]", &xi[0], &xi[1]); U("[1,2,3]", "[ii]", &xi[0], &xi[1]); U("[1,2,3]", "[ii*]", &xi[0], &xi[1]); U("{\"foo\":42,\"baz\":45}", "{sisi}", "baz", &xi[0], "foo", &xi[1]); U("{\"foo\":42,\"baz\":45}", "{sisi*}", "baz", &xi[0], "foo", &xi[1]); U("{\"foo\":42,\"baz\":45}", "{sisi!}", "baz", &xi[0], "foo", &xi[1]); U("{\"foo\":42,\"baz\":45}", "{si}", "baz", &xi[0], "foo", &xi[1]); U("{\"foo\":42,\"baz\":45}", "{si*}", "baz", &xi[0], "foo", &xi[1]); U("{\"foo\":42,\"baz\":45}", "{si!}", "baz", &xi[0], "foo", &xi[1]); U("[1,{\"foo\":2,\"bar\":null},[3,4]]", "[i{sisn}[ii]]", &xi[0], "foo", &xi[1], "bar", &xi[2], &xi[3]); U("[1,2,3]", "[ii!i]", &xi[0], &xi[1], &xi[2]); U("[1,2,3]", "[ii*i]", &xi[0], &xi[1], &xi[2]); U("{\"foo\":1,\"bar\":2}", "{si!si}", "foo", &xi[1], "bar", &xi[2]); U("{\"foo\":1,\"bar\":2}", "{si*si}", "foo", &xi[1], "bar", &xi[2]); U("{\"foo\":{\"baz\":null,\"bar\":null}}", "{s{sn!}}", "foo", "bar"); U("[[1,2,3]]", "[[ii!]]", &xi[0], &xi[1]); U("{}", "{s?i}", "foo", &xi[0]); U("{\"foo\":1}", "{s?i}", "foo", &xi[0]); U("{}", "{s?[ii]s?{s{si!}}}", "foo", &xi[0], &xi[1], "bar", "baz", "quux", &xi[2]); U("{\"foo\":[1,2]}", "{s?[ii]s?{s{si!}}}", "foo", &xi[0], &xi[1], "bar", "baz", "quux", &xi[2]); U("{\"bar\":{\"baz\":{\"quux\":15}}}", "{s?[ii]s?{s{si!}}}", "foo", &xi[0], &xi[1], "bar", "baz", "quux", &xi[2]); U("{\"foo\":{\"bar\":4}}", "{s?{s?i}}", "foo", "bar", &xi[0]); U("{\"foo\":{}}", "{s?{s?i}}", "foo", "bar", &xi[0]); U("{}", "{s?{s?i}}", "foo", "bar", &xi[0]); U("{\"foo\":42,\"baz\":45}", "{s?isi!}", "baz", &xi[0], "foo", &xi[1]); U("{\"foo\":42}", "{s?isi!}", "baz", &xi[0], "foo", &xi[1]); return 0; } #endif #if 0 /* Unpack the same item twice */ j = json_pack("{s:s, s:i, s:b}", "foo", "bar", "baz", 42, "quux", 1); if(!json_unpack_ex(j, &error, 0, "{s:s,s:s!}", "foo", &s, "foo", &s)) fail("json_unpack object with strict validation failed"); { const char *possible_errors[] = { "2 object item(s) left unpacked: baz, quux", "2 object item(s) left unpacked: quux, baz" }; check_errors(possible_errors, 2, "", 1, 10, 10); } json_decref(j); #endif