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