1 /* Simple library for printing JSON data.
2    Copyright (C) 2014-2022 Free Software Foundation, Inc.
3    This file is part of the GNU C Library.
4 
5    The GNU C Library is free software; you can redistribute it and/or
6    modify it under the terms of the GNU Lesser General Public
7    License as published by the Free Software Foundation; either
8    version 2.1 of the License, or (at your option) any later version.
9 
10    The GNU C Library is distributed in the hope that it will be useful,
11    but WITHOUT ANY WARRANTY; without even the implied warranty of
12    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
13    Lesser General Public License for more details.
14 
15    You should have received a copy of the GNU Lesser General Public
16    License along with the GNU C Library; if not, see
17    <https://www.gnu.org/licenses/>.  */
18 
19 #include <string.h>
20 
21 #include "json-lib.h"
22 
23 void
json_init(json_ctx_t * ctx,unsigned int indent_level,FILE * fp)24 json_init (json_ctx_t *ctx, unsigned int indent_level, FILE *fp)
25 {
26   ctx->indent_level = indent_level;
27   ctx->fp = fp;
28   ctx->first_element = true;
29 }
30 
31 static void
do_indent(json_ctx_t * ctx)32 do_indent (json_ctx_t *ctx)
33 {
34   char indent_buf[ctx->indent_level + 1];
35 
36   memset (indent_buf, ' ', ctx->indent_level + 1);
37   indent_buf[ctx->indent_level] = '\0';
38 
39   fputs (indent_buf, ctx->fp);
40 }
41 
42 void
json_document_begin(json_ctx_t * ctx)43 json_document_begin (json_ctx_t *ctx)
44 {
45   do_indent (ctx);
46 
47   fputs ("{\n", ctx->fp);
48 
49   ctx->indent_level++;
50   ctx->first_element = true;
51 }
52 
53 void
json_document_end(json_ctx_t * ctx)54 json_document_end (json_ctx_t *ctx)
55 {
56   ctx->indent_level--;
57 
58   do_indent (ctx);
59 
60   fputs ("\n}", ctx->fp);
61 }
62 
63 void
json_attr_object_begin(json_ctx_t * ctx,const char * name)64 json_attr_object_begin (json_ctx_t *ctx, const char *name)
65 {
66   if (!ctx->first_element)
67     fprintf (ctx->fp, ",\n");
68 
69   do_indent (ctx);
70 
71   fprintf (ctx->fp, "\"%s\": {\n", name);
72 
73   ctx->indent_level++;
74   ctx->first_element = true;
75 }
76 
77 void
json_attr_object_end(json_ctx_t * ctx)78 json_attr_object_end (json_ctx_t *ctx)
79 {
80   ctx->indent_level--;
81   ctx->first_element = false;
82 
83   fputs ("\n", ctx->fp);
84 
85   do_indent (ctx);
86 
87   fputs ("}", ctx->fp);
88 }
89 
90 void
json_attr_string(json_ctx_t * ctx,const char * name,const char * s)91 json_attr_string (json_ctx_t *ctx, const char *name, const char *s)
92 {
93   if (!ctx->first_element)
94     fprintf (ctx->fp, ",\n");
95   else
96     ctx->first_element = false;
97 
98   do_indent (ctx);
99 
100   fprintf (ctx->fp, "\"%s\": \"%s\"", name, s);
101 }
102 
103 void
json_attr_uint(json_ctx_t * ctx,const char * name,uint64_t d)104 json_attr_uint (json_ctx_t *ctx, const char *name, uint64_t d)
105 {
106   if (!ctx->first_element)
107     fprintf (ctx->fp, ",\n");
108   else
109     ctx->first_element = false;
110 
111   do_indent (ctx);
112 
113   fprintf (ctx->fp, "\"%s\": %" PRIu64 , name, d);
114 }
115 
116 void
json_attr_int(json_ctx_t * ctx,const char * name,int64_t d)117 json_attr_int (json_ctx_t *ctx, const char *name, int64_t d)
118 {
119   if (!ctx->first_element)
120     fprintf (ctx->fp, ",\n");
121   else
122     ctx->first_element = false;
123 
124   do_indent (ctx);
125 
126   fprintf (ctx->fp, "\"%s\": %" PRId64 , name, d);
127 }
128 
129 void
json_attr_double(json_ctx_t * ctx,const char * name,double d)130 json_attr_double (json_ctx_t *ctx, const char *name, double d)
131 {
132   if (!ctx->first_element)
133     fprintf (ctx->fp, ",\n");
134   else
135     ctx->first_element = false;
136 
137   do_indent (ctx);
138 
139   fprintf (ctx->fp, "\"%s\": %g", name, d);
140 }
141 
142 void
json_array_begin(json_ctx_t * ctx,const char * name)143 json_array_begin (json_ctx_t *ctx, const char *name)
144 {
145   if (!ctx->first_element)
146     fprintf (ctx->fp, ",\n");
147 
148   do_indent (ctx);
149 
150   fprintf (ctx->fp, "\"%s\": [", name);
151 
152   ctx->indent_level++;
153   ctx->first_element = true;
154 }
155 
156 void
json_array_end(json_ctx_t * ctx)157 json_array_end (json_ctx_t *ctx)
158 {
159   ctx->indent_level--;
160   ctx->first_element = false;
161 
162   fputs ("]", ctx->fp);
163 }
164 
165 void
json_element_string(json_ctx_t * ctx,const char * s)166 json_element_string (json_ctx_t *ctx, const char *s)
167 {
168   if (!ctx->first_element)
169     fprintf (ctx->fp, ", \"%s\"", s);
170   else
171     {
172       fprintf (ctx->fp, "\"%s\"", s);
173       ctx->first_element = false;
174     }
175 }
176 
177 void
json_element_uint(json_ctx_t * ctx,uint64_t d)178 json_element_uint (json_ctx_t *ctx, uint64_t d)
179 {
180   if (!ctx->first_element)
181     fprintf (ctx->fp, ", %" PRIu64, d);
182   else
183     {
184       fprintf (ctx->fp, "%" PRIu64, d);
185       ctx->first_element = false;
186     }
187 }
188 
189 void
json_element_int(json_ctx_t * ctx,int64_t d)190 json_element_int (json_ctx_t *ctx, int64_t d)
191 {
192   if (!ctx->first_element)
193     fprintf (ctx->fp, ", %" PRId64, d);
194   else
195     {
196       fprintf (ctx->fp, "%" PRId64, d);
197       ctx->first_element = false;
198     }
199 }
200 
201 void
json_element_double(json_ctx_t * ctx,double d)202 json_element_double (json_ctx_t *ctx, double d)
203 {
204   if (!ctx->first_element)
205     fprintf (ctx->fp, ", %g", d);
206   else
207     {
208       fprintf (ctx->fp, "%g", d);
209       ctx->first_element = false;
210     }
211 }
212 
213 void
json_element_object_begin(json_ctx_t * ctx)214 json_element_object_begin (json_ctx_t *ctx)
215 {
216   if (!ctx->first_element)
217     fprintf (ctx->fp, ",");
218 
219   fputs ("\n", ctx->fp);
220 
221   do_indent (ctx);
222 
223   fputs ("{\n", ctx->fp);
224 
225   ctx->indent_level++;
226   ctx->first_element = true;
227 }
228 
229 void
json_element_object_end(json_ctx_t * ctx)230 json_element_object_end (json_ctx_t *ctx)
231 {
232   ctx->indent_level--;
233   ctx->first_element = false;
234 
235   fputs ("\n", ctx->fp);
236 
237   do_indent (ctx);
238 
239   fputs ("}", ctx->fp);
240 }
241