jsoninfo: move json support functions into their own file
[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 static int entrynumber[16] = { 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 };
61 static int currentjsondepth = 0;
62
63 static void abuf_json_new_indent(struct autobuf *abuf) {
64   int i = currentjsondepth;
65
66   if (!i) {
67     return;
68   }
69
70   abuf_puts(abuf, "\n");
71   while (i-- > 0) {
72     abuf_puts(abuf, "  ");
73   }
74 }
75
76 void abuf_json_reset_entry_number_and_depth(void) {
77   entrynumber[0] = 0;
78   currentjsondepth = 0;
79 }
80
81 void abuf_json_mark_output(bool open, struct autobuf *abuf) {
82   if (open) {
83     assert(!currentjsondepth);
84     abuf_json_new_indent(abuf);
85     abuf_puts(abuf, "{");
86     currentjsondepth++;
87     entrynumber[currentjsondepth] = 0;
88   } else {
89     entrynumber[currentjsondepth] = 0;
90     currentjsondepth--;
91     assert(!currentjsondepth);
92     abuf_json_new_indent(abuf);
93     abuf_puts(abuf, "\n}");
94   }
95 }
96
97 void abuf_json_mark_object(bool open, bool array, struct autobuf *abuf, const char* header) {
98   if (open) {
99     if (entrynumber[currentjsondepth]) {
100       abuf_appendf(abuf, ",");
101     }
102     abuf_json_new_indent(abuf);
103     if (header) {
104       abuf_appendf(abuf, "\"%s\": %s", header, array ? "[" : "{");
105     } else {
106       abuf_appendf(abuf, "%s", array ? "[" : "{");
107     }
108     entrynumber[currentjsondepth]++;
109     currentjsondepth++;
110     entrynumber[currentjsondepth] = 0;
111   } else {
112     entrynumber[currentjsondepth] = 0;
113     currentjsondepth--;
114     abuf_json_new_indent(abuf);
115     abuf_appendf(abuf, "%s", array ? "]" : "}");
116   }
117 }
118
119 void abuf_json_mark_array_entry(bool open, struct autobuf *abuf) {
120   abuf_json_mark_object(open, false, abuf, NULL);
121 }
122
123 void abuf_json_insert_comma(struct autobuf *abuf) {
124   if (entrynumber[currentjsondepth])
125     abuf_appendf(abuf, ",");
126 }
127
128 void abuf_json_boolean(struct autobuf *abuf, const char* key, int value) {
129   abuf_json_insert_comma(abuf);
130   abuf_json_new_indent(abuf);
131   abuf_appendf(abuf, "\"%s\": %s", key, value ? "true" : "false");
132   entrynumber[currentjsondepth]++;
133 }
134
135 void abuf_json_string(struct autobuf *abuf, const char* key, const char* value) {
136   abuf_json_insert_comma(abuf);
137   abuf_json_new_indent(abuf);
138   abuf_appendf(abuf, "\"%s\": \"%s\"", key, value);
139   entrynumber[currentjsondepth]++;
140 }
141
142 void abuf_json_int(struct autobuf *abuf, const char* key, long value) {
143   abuf_json_insert_comma(abuf);
144   abuf_json_new_indent(abuf);
145   abuf_appendf(abuf, "\"%s\": %li", key, value);
146   entrynumber[currentjsondepth]++;
147 }
148
149 void abuf_json_float(struct autobuf *abuf, const char* key, float value) {
150   abuf_json_insert_comma(abuf);
151   abuf_json_new_indent(abuf);
152   abuf_appendf(abuf, "\"%s\": %.03f", key, (double) value);
153   entrynumber[currentjsondepth]++;
154 }
155
156 /* Linux specific functions for getting system info */
157
158 #ifdef __linux__
159 static int get_string_from_file(const char* filename, char* buf, int len) {
160   int bytes = -1;
161   int fd = open(filename, O_RDONLY);
162   if (fd > -1) {
163     bytes = read(fd, buf, len);
164     if (bytes < len)
165       buf[bytes - 1] = '\0'; // remove trailing \n
166     else
167       buf[len - 1] = '\0';
168     close(fd);
169   }
170   return bytes;
171 }
172
173 static int abuf_json_sysdata(struct autobuf *abuf, const char* key, const char* syspath) {
174   int ret = -1;
175   char buf[256];
176   *buf = 0;
177   ret = get_string_from_file(syspath, buf, 256);
178   if (*buf)
179     abuf_json_string(abuf, key, buf);
180   return ret;
181 }
182
183 void abuf_json_sys_class_net(struct autobuf *abuf, const char* key, const char* ifname, const char* datapoint) {
184   char filename[256];
185   snprintf(filename, 255, "/sys/class/net/%s/%s", ifname, datapoint);
186   abuf_json_sysdata(abuf, key, filename);
187 }
188 #endif /* __linux__ */
189
190 int read_uuid_from_file(const char * name, const char *file) {
191   FILE *f;
192   char* end;
193   int r = 0;
194   size_t chars;
195
196   memset(uuid, 0, sizeof(uuid));
197
198   f = fopen(file, "r");
199   olsr_printf(1, "(%s) Reading UUID from '%s'\n", name, file);
200   if (f == NULL) {
201     olsr_printf(1, "(%s) Could not open '%s': %s\n", name, file, strerror(errno));
202     return -1;
203   }
204   chars = fread(uuid, 1, UUIDLEN, f);
205   if (chars > 0) {
206     uuid[chars] = '\0'; /* null-terminate the string */
207
208     /* we only use the first line of the file */
209     end = strchr(uuid, '\n');
210     if (end)
211       *end = 0;
212     r = 0;
213   } else {
214     olsr_printf(1, "(%s) Could not read UUID from '%s': %s\n", name, file, strerror(errno));
215     r = -1;
216   }
217
218   fclose(f);
219   return r;
220 }