LCOV - code coverage report
Current view: top level - modules - json.sl (source / functions) Hit Total Coverage
Test: all.lcov Lines: 107 110 97.3 %
Date: 2022-08-02 14:41:00 Functions: 13 13 100.0 %

          Line data    Source code
       1             : % -*- mode: slang; mode: fold -*-
       2             : 
       3           1 : import("json");
       4             : 
       5             : %{{{ Type Handlers for json_encode
       6             : 
       7           1 : private variable Type_Map = Ref_Type[0];
       8             : 
       9             : private define add_type_handler ()
      10             : {
      11           6 :    variable func = ();
      12           6 :    loop (_NARGS-1)
      13             :      {
      14          18 :         variable type = ();
      15          18 :         variable idx = __class_id (type);
      16          18 :         variable n = length (Type_Map);
      17          18 :         if (idx >= n)
      18             :           {
      19           6 :              variable new_map = Ref_Type[idx+1];
      20           6 :              new_map[[0:n-1]] = Type_Map;
      21           6 :              Type_Map = new_map;
      22             :           }
      23          18 :         Type_Map[idx] = func;
      24             :      }
      25             : }
      26             : 
      27             : private define get_encode_func (type)
      28             : {
      29             :    try
      30             :      {
      31          64 :         variable func = Type_Map[__class_id (type)];
      32          64 :         if (func == NULL) throw IndexError;
      33          64 :         return func;
      34             :      }
      35             :    catch IndexError:
      36           0 :      throw Json_Invalid_Json_Error, "$type does not represent a JSON data structure"$;
      37             : }
      38             : %}}}
      39             : 
      40             : private define json_encode_string (indent, q, data) %{{{
      41             : {
      42           4 :    return _json_encode_string (data);
      43             : }
      44           1 : add_type_handler (String_Type, BString_Type, &json_encode_string);
      45             : %}}}
      46             : 
      47             : private define json_encode_boolean (indent, q, data) %{{{
      48             : {
      49           4 :    if (data == 1) return "true"B;
      50           3 :    if (data == 0) return "false"B;
      51           1 :    throw Json_Invalid_Json_Error, sprintf (`invalid boolean value '\%03o'; only '\000' and '\001' are allowed`, data);
      52             : }
      53           1 : add_type_handler (UChar_Type, &json_encode_boolean);
      54             : %}}}
      55             : 
      56             : private define json_encode_null (indent, q, data) %{{{
      57             : {
      58           2 :    return "null"B;
      59             : }
      60           1 : add_type_handler (Null_Type, &json_encode_null);
      61             : %}}}
      62             : 
      63             : private define json_encode_number (indent, q, data) %{{{
      64             : {
      65          34 :    return string (data);
      66             : }
      67           1 : add_type_handler (
      68             :         Char_Type, % UChar_Type,
      69             :         Short_Type, UShort_Type,
      70             :         Int_Type, UInt_Type,
      71             :         Long_Type, ULong_Type,
      72             : #ifexists LLong_Type
      73             :         LLong_Type, ULLong_Type,
      74             : #endif
      75             :         Float_Type, Double_Type,
      76             :         &json_encode_number);
      77             : %}}}
      78             : 
      79             : private define json_encode_object (indent, q, object) %{{{
      80             : {
      81           8 :    variable json = "{"B;
      82           8 :    variable keys = get_struct_field_names (object);
      83           8 :    variable n_values = length (keys);
      84           8 :    if (n_values)
      85             :      {
      86             :         % pvs indent KEY nsep VAL vsep pvs indent KEY nsep VAL vsep
      87             :         % ... pvs indent KEY nsep VAL pvs
      88             : 
      89           7 :         variable new_indent = bstrcat (indent, q.indent);
      90           7 :         variable sep = bstrcat (q.vsep, q.post_vsep, new_indent);
      91           7 :         variable nsep = q.nsep;
      92             : 
      93           7 :         variable key = keys[0];
      94           7 :         variable val = get_struct_field(object, key);
      95           7 :         variable type = typeof (val);
      96           7 :         variable func = get_encode_func (type);
      97           7 :         json = bstrcat (__tmp(json), q.post_vsep, new_indent,
      98             :                         _json_encode_string (key), nsep, (@func)(new_indent, q, val));
      99             : 
     100             :         variable i;
     101           7 :         _for i (1, n_values-1)
     102             :           {
     103           9 :              key = keys[i];
     104           9 :              val = get_struct_field(object, key);
     105           9 :              variable next_type = typeof (val);
     106           9 :              if (next_type == String_Type)
     107             :                {
     108           1 :                   json = bstrcat (__tmp(json), sep, _json_encode_string (key),
     109             :                                   nsep, _json_encode_string (val));
     110           1 :                   continue;
     111             :                }
     112             : 
     113           8 :              if (next_type != type)
     114           6 :                (type, func) = (next_type, get_encode_func (next_type));
     115           8 :              json = bstrcat (__tmp(json), sep, _json_encode_string (key),
     116             :                              nsep, (@func)(new_indent, q, val));
     117             :           }
     118           7 :         json = bstrcat (__tmp(json), q.post_vsep);
     119             :      }
     120           8 :    return bstrcat (__tmp(json), indent, "}");
     121             : }
     122           1 : add_type_handler (Struct_Type, &json_encode_object);
     123             : %}}}
     124             : 
     125             : private define json_encode_array (indent, q, array) %{{{
     126             : {
     127          20 :    variable json = "["B;
     128          20 :    variable n_values = length (array);
     129          20 :    if (n_values)
     130             :      {
     131             :         %  pvs, new_indent, VALUE, vsep, pvs, new_indent, VALUE, vsep, pvs, ..., new_indent, VALUE, pvs
     132             : 
     133          17 :         variable new_indent = bstrcat (indent, q.indent);
     134          17 :         variable sep = bstrcat (q.vsep, q.post_vsep, new_indent);
     135             : 
     136          17 :         variable i = 0;
     137          17 :         variable a = array[i];
     138          17 :         variable type = typeof (a);
     139          17 :         variable func = get_encode_func (type);
     140          17 :         json = bstrcat (__tmp(json), q.post_vsep,
     141             :                         new_indent, (@func)(new_indent, q, a));
     142             : 
     143          16 :         if ((typeof (array) == Array_Type) && not any (_isnull (array)))
     144             :           {
     145           3 :              if (type == String_Type)
     146           1 :                _for i (1, n_values-1)
     147           1 :                  json = bstrcat (__tmp(json), sep, _json_encode_string (array[i]));
     148             :              else
     149           2 :                _for i (1, n_values-1)
     150           4 :                  json = bstrcat (__tmp(json), sep, (@func)(new_indent, q, array[i]));
     151             :           }
     152          13 :         else _for i (1, n_values-1)
     153             :           {
     154          27 :              a = array[i];
     155          27 :              variable next_type = typeof (a);
     156          27 :              if (next_type == String_Type)
     157             :                {
     158           8 :                   json = bstrcat (__tmp(json), sep, _json_encode_string (a));
     159           8 :                   continue;
     160             :                }
     161             : 
     162          19 :              if (next_type != type)
     163          17 :                (type, func) = (next_type, get_encode_func (next_type));
     164          19 :              json = bstrcat (__tmp(json), sep, (@func)(new_indent, q, a));
     165             :           }
     166             : 
     167          16 :         json = bstrcat (__tmp(json), q.post_vsep);
     168             :      }
     169          19 :    return bstrcat (__tmp(json), indent, "]");
     170             : }
     171           1 : add_type_handler (List_Type, Array_Type, &json_encode_array);
     172             : %}}}
     173             : 
     174             : private define default_handler (indent, q, data) %{{{
     175             : {
     176           1 :    if (0 < __is_numeric (data) < 3)
     177           0 :      return json_encode_number (data);
     178             : 
     179           1 :    if (is_struct_type (data))
     180           0 :      return json_encode_object (data);
     181             : 
     182           1 :    variable type = _typeof (data);
     183           1 :    throw Json_Invalid_Json_Error, "$type does not represent a JSON data structure"$;
     184             : }
     185           1 : Type_Map[where (_isnull (Type_Map))] = &default_handler;
     186             : %}}}
     187             : 
     188             : % process_qualifiers %{{{
     189             : 
     190             : private define split_post_vsep (post_vsep) %{{{
     191             : {
     192          17 :    variable indent = "";
     193          17 :    variable parts = strchop (post_vsep, '\n', 0);
     194          17 :    if (length (parts) > 1)
     195             :      {
     196           5 :         indent = parts[-1];
     197           5 :         parts[-1] = "";
     198           5 :         post_vsep = strjoin (parts, "\n");
     199             :      }
     200          17 :   return (indent, post_vsep);
     201             : }
     202             : %}}}
     203             : 
     204             : private define only_whitespace (s)
     205             : {
     206          85 :    return ""B + str_delete_chars (s, "^ \t\n\r");
     207             : }
     208             : 
     209             : private define process_qualifiers ()
     210             : {
     211          17 :    variable post_vsep = split_post_vsep (qualifier ("post_vsep", ""));
     212          17 :    variable indent = ();  % other return value from split_post_vsep
     213          17 :    variable q = struct {
     214          17 :       pre_nsep  = only_whitespace (qualifier ("pre_nsep", "")),
     215          17 :       post_nsep = only_whitespace (qualifier ("post_nsep", "")),
     216          17 :       pre_vsep  = only_whitespace (qualifier ("pre_vsep", "")),
     217          17 :       post_vsep = only_whitespace (post_vsep),
     218          17 :       indent    = only_whitespace (indent),
     219             :    };
     220          17 :    return struct {
     221          17 :       nsep = q.pre_nsep + ":" + q.post_nsep,
     222          17 :       vsep = q.pre_vsep + ",",
     223          17 :       @q
     224             :    };
     225             : }
     226             : %}}}
     227             : 
     228             : define json_encode (data)
     229             : {
     230          17 :    variable q = process_qualifiers (;; __qualifiers);
     231          17 :    variable func = get_encode_func (typeof (data));
     232          17 :    variable json = (@func)(""B, q, data);
     233          15 :    return typecast (json, String_Type);
     234             : }

Generated by: LCOV version 1.13