jsoninfo: helpers: firm up asserts on currentjsondepth
[olsrd.git] / lib / jsoninfo / src / olsrd_jsoninfo_helpers.c
1 /*
2  * The olsr.org Optimized Link-State Routing daemon(olsrd)
3  * Copyright (c) 2004
4  *
5  * All rights reserved.
6  *
7  * Redistribution and use in source and binary forms, with or without
8  * modification, are permitted provided that the following conditions
9  * are met:
10  *
11  * * Redistributions of source code must retain the above copyright
12  *   notice, this list of conditions and the following disclaimer.
13  * * Redistributions in binary form must reproduce the above copyright
14  *   notice, this list of conditions and the following disclaimer in
15  *   the documentation and/or other materials provided with the
16  *   distribution.
17  * * Neither the name of olsr.org, olsrd nor the names of its
18  *   contributors may be used to endorse or promote products derived
19  *   from this software without specific prior written permission.
20  *
21  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
22  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
23  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
24  * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
25  * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
26  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
27  * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
28  * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
29  * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
30  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
31  * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
32  * POSSIBILITY OF SUCH DAMAGE.
33  *
34  * Visit http://www.olsr.org for more information.
35  *
36  * If you find this software useful feel free to make a donation
37  * to the project. For more information see the website or contact
38  * the copyright holders.
39  *
40  */
41
42 #include "olsrd_jsoninfo_helpers.h"
43 #include "olsr.h"
44
45 #include <stdbool.h>
46 #include <assert.h>
47 #include <unistd.h>
48
49 #ifdef __linux__
50 #include <fcntl.h>
51 #endif /* __linux__ */
52
53 char uuid[UUIDLEN + 1];
54
55 /* JSON support functions */
56
57 /* JSON does not allow commas dangling at the end of arrays, so we need to
58  * count which entry number we're at in order to make sure we don't tack a
59  * dangling comma on at the end */
60 #define ENTRY_NUMBER_MAX_DEPTH 16
61 static int entrynumber[ENTRY_NUMBER_MAX_DEPTH] = { 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 };
62 static int currentjsondepth = 0;
63
64 static void abuf_json_new_indent(struct autobuf *abuf) {
65   if (currentjsondepth) {
66     int i = currentjsondepth;
67
68     abuf_puts(abuf, "\n");
69     while (i-- > 0) {
70       abuf_puts(abuf, "  ");
71     }
72   }
73 }
74
75 void abuf_json_reset_entry_number_and_depth(void) {
76   entrynumber[0] = 0;
77   currentjsondepth = 0;
78 }
79
80 void abuf_json_mark_output(bool open, struct autobuf *abuf) {
81   if (open) {
82     assert(!currentjsondepth);
83     abuf_json_new_indent(abuf);
84     abuf_puts(abuf, "{");
85     currentjsondepth++;
86     entrynumber[currentjsondepth] = 0;
87   } else {
88     assert(currentjsondepth == 1);
89     entrynumber[currentjsondepth] = 0;
90     currentjsondepth--;
91     abuf_json_new_indent(abuf);
92     abuf_puts(abuf, "\n}");
93   }
94 }
95
96 void abuf_json_mark_object(bool open, bool array, struct autobuf *abuf, const char* header) {
97   if (open) {
98     if (entrynumber[currentjsondepth]) {
99       abuf_appendf(abuf, ",");
100     }
101     abuf_json_new_indent(abuf);
102     if (header) {
103       abuf_appendf(abuf, "\"%s\": %s", header, array ? "[" : "{");
104     } else {
105       abuf_appendf(abuf, "%s", array ? "[" : "{");
106     }
107     entrynumber[currentjsondepth]++;
108     currentjsondepth++;
109     assert(currentjsondepth < ENTRY_NUMBER_MAX_DEPTH);
110     entrynumber[currentjsondepth] = 0;
111   } else {
112     entrynumber[currentjsondepth] = 0;
113     currentjsondepth--;
114     assert(currentjsondepth >= 0);
115     abuf_json_new_indent(abuf);
116     abuf_appendf(abuf, "%s", array ? "]" : "}");
117   }
118 }
119
120 void abuf_json_mark_array_entry(bool open, struct autobuf *abuf) {
121   abuf_json_mark_object(open, false, abuf, NULL);
122 }
123
124 void abuf_json_insert_comma(struct autobuf *abuf) {
125   if (entrynumber[currentjsondepth])
126     abuf_appendf(abuf, ",");
127 }
128
129 void abuf_json_boolean(struct autobuf *abuf, const char* key, int value) {
130   abuf_json_insert_comma(abuf);
131   abuf_json_new_indent(abuf);
132   abuf_appendf(abuf, "\"%s\": %s", key, value ? "true" : "false");
133   entrynumber[currentjsondepth]++;
134 }
135
136 void abuf_json_string(struct autobuf *abuf, const char* key, const char* value) {
137   abuf_json_insert_comma(abuf);
138   abuf_json_new_indent(abuf);
139   abuf_appendf(abuf, "\"%s\": \"%s\"", key, value);
140   entrynumber[currentjsondepth]++;
141 }
142
143 void abuf_json_int(struct autobuf *abuf, const char* key, long value) {
144   abuf_json_insert_comma(abuf);
145   abuf_json_new_indent(abuf);
146   abuf_appendf(abuf, "\"%s\": %li", key, value);
147   entrynumber[currentjsondepth]++;
148 }
149
150 void abuf_json_float(struct autobuf *abuf, const char* key, float value) {
151   abuf_json_insert_comma(abuf);
152   abuf_json_new_indent(abuf);
153   abuf_appendf(abuf, "\"%s\": %.03f", key, (double) value);
154   entrynumber[currentjsondepth]++;
155 }
156
157 /* Linux specific functions for getting system info */
158
159 #ifdef __linux__
160 static int get_string_from_file(const char* filename, char* buf, int len) {
161   int bytes = -1;
162   int fd = open(filename, O_RDONLY);
163   if (fd > -1) {
164     bytes = read(fd, buf, len);
165     if (bytes < len)
166       buf[bytes - 1] = '\0'; // remove trailing \n
167     else
168       buf[len - 1] = '\0';
169     close(fd);
170   }
171   return bytes;
172 }
173
174 static int abuf_json_sysdata(struct autobuf *abuf, const char* key, const char* syspath) {
175   int ret = -1;
176   char buf[256];
177   *buf = 0;
178   ret = get_string_from_file(syspath, buf, 256);
179   if (*buf)
180     abuf_json_string(abuf, key, buf);
181   return ret;
182 }
183
184 void abuf_json_sys_class_net(struct autobuf *abuf, const char* key, const char* ifname, const char* datapoint) {
185   char filename[256];
186   snprintf(filename, 255, "/sys/class/net/%s/%s", ifname, datapoint);
187   abuf_json_sysdata(abuf, key, filename);
188 }
189 #endif /* __linux__ */
190
191 int read_uuid_from_file(const char * name, const char *file) {
192   FILE *f;
193   char* end;
194   int r = 0;
195   size_t chars;
196
197   memset(uuid, 0, sizeof(uuid));
198
199   f = fopen(file, "r");
200   olsr_printf(1, "(%s) Reading UUID from '%s'\n", name, file);
201   if (f == NULL) {
202     olsr_printf(1, "(%s) Could not open '%s': %s\n", name, file, strerror(errno));
203     return -1;
204   }
205   chars = fread(uuid, 1, UUIDLEN, f);
206   if (chars > 0) {
207     uuid[chars] = '\0'; /* null-terminate the string */
208
209     /* we only use the first line of the file */
210     end = strchr(uuid, '\n');
211     if (end)
212       *end = 0;
213     r = 0;
214   } else {
215     olsr_printf(1, "(%s) Could not read UUID from '%s': %s\n", name, file, strerror(errno));
216     r = -1;
217   }
218
219   fclose(f);
220   return r;
221 }