rename 'msValid' JSON key to 'validityTime' for clarity
[olsrd.git] / lib / jsoninfo / src / olsrd_jsoninfo.c
1
2 /*
3  * The olsr.org Optimized Link-State Routing daemon(olsrd)
4  * Copyright (c) 2004, Andreas Tonnesen(andreto@olsr.org)
5  *                     includes code by Bruno Randolf
6  *                     includes code by Andreas Lopatic
7  *                     includes code by Sven-Ola Tuecke
8  *                     includes code by Lorenz Schori
9  *                     includes bugs by Markus Kittenberger
10  *                     includes bugs by Hans-Christoph Steiner
11  * All rights reserved.
12  *
13  * Redistribution and use in source and binary forms, with or without
14  * modification, are permitted provided that the following conditions
15  * are met:
16  *
17  * * Redistributions of source code must retain the above copyright
18  *   notice, this list of conditions and the following disclaimer.
19  * * Redistributions in binary form must reproduce the above copyright
20  *   notice, this list of conditions and the following disclaimer in
21  *   the documentation and/or other materials provided with the
22  *   distribution.
23  * * Neither the name of olsr.org, olsrd nor the names of its
24  *   contributors may be used to endorse or promote products derived
25  *   from this software without specific prior written permission.
26  *
27  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
28  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
29  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
30  * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
31  * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
32  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
33  * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
34  * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
35  * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
36  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
37  * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
38  * POSSIBILITY OF SUCH DAMAGE.
39  *
40  * Visit http://www.olsr.org for more information.
41  *
42  * If you find this software useful feel free to make a donation
43  * to the project. For more information see the website or contact
44  * the copyright holders.
45  *
46  */
47
48 /*
49  * Dynamic linked library for the olsr.org olsr daemon
50  */
51
52
53 #include <sys/types.h>
54 #include <sys/socket.h>
55 #ifndef _WIN32
56 #include <sys/select.h>
57 #endif
58 #include <netinet/in.h>
59 #include <arpa/inet.h>
60 #include <sys/time.h>
61 #include <time.h>
62 #include <math.h>
63 #include <stdio.h>
64 #include <string.h>
65 #include <stdlib.h>
66 #include <unistd.h>
67 #include <errno.h>
68
69 #include "ipcalc.h"
70 #include "olsr.h"
71 #include "olsr_types.h"
72 #include "neighbor_table.h"
73 #include "two_hop_neighbor_table.h"
74 #include "mpr_selector_set.h"
75 #include "tc_set.h"
76 #include "hna_set.h"
77 #include "mid_set.h"
78 #include "link_set.h"
79 #include "net_olsr.h"
80 #include "lq_plugin.h"
81 #include "common/autobuf.h"
82 #include "gateway.h"
83
84 #include "olsrd_jsoninfo.h"
85 #include "olsrd_plugin.h"
86
87 #ifdef _WIN32
88 #define close(x) closesocket(x)
89 #endif
90
91 static int ipc_socket;
92
93 /* IPC initialization function */
94 static int plugin_ipc_init(void);
95
96 static void abuf_json_open_object(struct autobuf *abuf, const char* header);
97 static void abuf_json_close_object(struct autobuf *abuf);
98 static void abuf_json_open_array(struct autobuf *abuf, const char* header);
99 static void abuf_json_close_array(struct autobuf *abuf);
100 static void abuf_json_open_array_entry(struct autobuf *abuf);
101 static void abuf_json_close_array_entry(struct autobuf *abuf);
102 static void abuf_json_boolean(struct autobuf *abuf, const char* key, int value);
103 static void abuf_json_string(struct autobuf *abuf, const char* key, const char* value);
104 static void abuf_json_int(struct autobuf *abuf, const char* key, long value);
105 static void abuf_json_float(struct autobuf *abuf, const char* key, float value);
106
107 static void send_info(unsigned int /*send_what*/, int /*socket*/);
108 static void ipc_action(int, void *, unsigned int);
109 static void ipc_print_neighbors(struct autobuf *);
110 static void ipc_print_links(struct autobuf *);
111 static void ipc_print_routes(struct autobuf *);
112 static void ipc_print_topology(struct autobuf *);
113 static void ipc_print_hna(struct autobuf *);
114 static void ipc_print_mid(struct autobuf *);
115 static void ipc_print_gateways(struct autobuf *);
116 static void ipc_print_config(struct autobuf *);
117 static void ipc_print_interfaces(struct autobuf *);
118 static void ipc_print_plugins(struct autobuf *);
119 static void ipc_print_olsrd_conf(struct autobuf *abuf);
120
121 #define TXT_IPC_BUFSIZE 256
122
123 #define SIW_NEIGHBORS 0x0001
124 #define SIW_LINKS 0x0002
125 #define SIW_ROUTES 0x0004
126 #define SIW_HNA 0x0008
127 #define SIW_MID 0x0010
128 #define SIW_TOPOLOGY 0x0020
129 #define SIW_GATEWAYS 0x0040
130 #define SIW_INTERFACES 0x0080
131 #define SIW_CONFIG 0x0100
132 #define SIW_ALL 0x00FF
133
134 /* these don't change at runtime, so they are not part of ALL/status */
135 #define SIW_PLUGINS 0x0200
136 #define SIW_OLSRD_CONF 0x0400
137
138 #define MAX_CLIENTS 3
139
140 static char *outbuffer[MAX_CLIENTS];
141 static size_t outbuffer_size[MAX_CLIENTS];
142 static size_t outbuffer_written[MAX_CLIENTS];
143 static int outbuffer_socket[MAX_CLIENTS];
144 static int outbuffer_count;
145
146 static struct timeval start_time;
147 static struct timer_entry *writetimer_entry;
148
149
150 /* JSON support functions */
151
152
153 /* JSON does not tolerate commas dangling at the end of arrays, so we need to
154  * count which entry number we're at in order to make sure we don't tack a
155  * dangling comma on at the end */
156 static int entrynumber = 0;
157 static int arrayentrynumber = 0;
158
159 static void
160 abuf_json_open_object(struct autobuf *abuf, const char* header)
161 {
162   entrynumber = 0;
163   abuf_appendf(abuf, "{\"%s\": {", header);
164 }
165
166 static void
167 abuf_json_close_object(struct autobuf *abuf)
168 {
169   abuf_appendf(abuf, "\t}\n}\n");
170 }
171
172 static void
173 abuf_json_open_array(struct autobuf *abuf, const char* header)
174 {
175   arrayentrynumber = 0;
176   abuf_appendf(abuf, "{\"%s\": [\n", header);
177 }
178
179 static void
180 abuf_json_close_array(struct autobuf *abuf)
181 {
182   abuf_appendf(abuf, "]}\n");
183 }
184
185 static void
186 abuf_json_open_array_entry(struct autobuf *abuf)
187 {
188   entrynumber = 0;
189   if (arrayentrynumber)
190     abuf_appendf(abuf, ",\n{");
191   else
192     abuf_appendf(abuf, "{");
193   arrayentrynumber++;
194 }
195
196 static void
197 abuf_json_close_array_entry(struct autobuf *abuf)
198 {
199   abuf_appendf(abuf, "}");
200 }
201
202 static void
203 abuf_json_boolean(struct autobuf *abuf, const char* key, int value)
204 {
205   if (entrynumber)
206     abuf_appendf(abuf, ",\n");
207   else
208     abuf_appendf(abuf, "\n");
209   abuf_appendf(abuf, "\t\"%s\": %s", key, value ? "true" : "false");
210   entrynumber++;
211 }
212
213 static void
214 abuf_json_string(struct autobuf *abuf, const char* key, const char* value)
215 {
216   if (entrynumber)
217     abuf_appendf(abuf, ",\n");
218   else
219     abuf_appendf(abuf, "\n");
220   abuf_appendf(abuf, "\t\"%s\": \"%s\"", key, value);
221   entrynumber++;
222 }
223
224 static void
225 abuf_json_int(struct autobuf *abuf, const char* key, long value)
226 {
227   if (entrynumber)
228     abuf_appendf(abuf, ",\n");
229   else
230     abuf_appendf(abuf, "\n");
231   abuf_appendf(abuf, "\t\"%s\": %li", key, value);
232   entrynumber++;
233 }
234
235 static void
236 abuf_json_float(struct autobuf *abuf, const char* key, float value)
237 {
238   if (entrynumber)
239     abuf_appendf(abuf, ",\n");
240   else
241     abuf_appendf(abuf, "\n");
242   abuf_appendf(abuf, "\t\"%s\": %.03f", key, value);
243   entrynumber++;
244 }
245
246
247 /**
248  *Do initialization here
249  *
250  *This function is called by the my_init
251  *function in uolsrd_plugin.c
252  */
253 int
254 olsrd_plugin_init(void)
255 {
256   /* Initial IPC value */
257   ipc_socket = -1;
258
259   /* Get start time */
260   gettimeofday(&start_time, NULL);
261
262   plugin_ipc_init();
263   return 1;
264 }
265
266 /**
267  * destructor - called at unload
268  */
269 void
270 olsr_plugin_exit(void)
271 {
272   if (ipc_socket != -1)
273     close(ipc_socket);
274 }
275
276 static int
277 plugin_ipc_init(void)
278 {
279   union olsr_sockaddr sst;
280   uint32_t yes = 1;
281   socklen_t addrlen;
282
283   /* Init ipc socket */
284   if ((ipc_socket = socket(olsr_cnf->ip_version, SOCK_STREAM, 0)) == -1) {
285 #ifndef NODEBUG
286     olsr_printf(1, "(JSONINFO) socket()=%s\n", strerror(errno));
287 #endif
288     return 0;
289   } else {
290     if (setsockopt(ipc_socket, SOL_SOCKET, SO_REUSEADDR, (char *)&yes, sizeof(yes)) < 0) {
291 #ifndef NODEBUG
292       olsr_printf(1, "(JSONINFO) setsockopt()=%s\n", strerror(errno));
293 #endif
294       return 0;
295     }
296 #if (defined __FreeBSD__ || defined __FreeBSD_kernel__) && defined SO_NOSIGPIPE
297     if (setsockopt(ipc_socket, SOL_SOCKET, SO_NOSIGPIPE, (char *)&yes, sizeof(yes)) < 0) {
298       perror("SO_REUSEADDR failed");
299       return 0;
300     }
301 #endif
302     /* Bind the socket */
303
304     /* complete the socket structure */
305     memset(&sst, 0, sizeof(sst));
306     if (olsr_cnf->ip_version == AF_INET) {
307       sst.in4.sin_family = AF_INET;
308       addrlen = sizeof(struct sockaddr_in);
309 #ifdef SIN6_LEN
310       sst.in4.sin_len = addrlen;
311 #endif
312       sst.in4.sin_addr.s_addr = jsoninfo_listen_ip.v4.s_addr;
313       sst.in4.sin_port = htons(ipc_port);
314     } else {
315       sst.in6.sin6_family = AF_INET6;
316       addrlen = sizeof(struct sockaddr_in6);
317 #ifdef SIN6_LEN
318       sst.in6.sin6_len = addrlen;
319 #endif
320       sst.in6.sin6_addr = jsoninfo_listen_ip.v6;
321       sst.in6.sin6_port = htons(ipc_port);
322     }
323
324     /* bind the socket to the port number */
325     if (bind(ipc_socket, &sst.in, addrlen) == -1) {
326 #ifndef NODEBUG
327       olsr_printf(1, "(JSONINFO) bind()=%s\n", strerror(errno));
328 #endif
329       return 0;
330     }
331
332     /* show that we are willing to listen */
333     if (listen(ipc_socket, 1) == -1) {
334 #ifndef NODEBUG
335       olsr_printf(1, "(JSONINFO) listen()=%s\n", strerror(errno));
336 #endif
337       return 0;
338     }
339
340     /* Register with olsrd */
341     add_olsr_socket(ipc_socket, &ipc_action, NULL, NULL, SP_PR_READ);
342
343 #ifndef NODEBUG
344     olsr_printf(2, "(JSONINFO) listening on port %d\n", ipc_port);
345 #endif
346   }
347   return 1;
348 }
349
350 static void
351 ipc_action(int fd, void *data __attribute__ ((unused)), unsigned int flags __attribute__ ((unused)))
352 {
353   union olsr_sockaddr pin;
354
355   char addr[INET6_ADDRSTRLEN];
356   fd_set rfds;
357   struct timeval tv;
358   unsigned int send_what = 0;
359   int ipc_connection;
360
361   socklen_t addrlen = sizeof(pin);
362
363   if ((ipc_connection = accept(fd, &pin.in, &addrlen)) == -1) {
364 #ifndef NODEBUG
365     olsr_printf(1, "(JSONINFO) accept()=%s\n", strerror(errno));
366 #endif
367     return;
368   }
369
370   tv.tv_sec = tv.tv_usec = 0;
371   if (olsr_cnf->ip_version == AF_INET) {
372     if (inet_ntop(olsr_cnf->ip_version, &pin.in4.sin_addr, addr, INET6_ADDRSTRLEN) == NULL)
373       addr[0] = '\0';
374     if (!ip4equal(&pin.in4.sin_addr, &jsoninfo_accept_ip.v4) && jsoninfo_accept_ip.v4.s_addr != INADDR_ANY) {
375 #ifdef JSONINFO_ALLOW_LOCALHOST
376       if (pin.in4.sin_addr.s_addr != INADDR_LOOPBACK) {
377 #endif
378         olsr_printf(1, "(JSONINFO) From host(%s) not allowed!\n", addr);
379         close(ipc_connection);
380         return;
381 #ifdef JSONINFO_ALLOW_LOCALHOST
382       }
383 #endif
384     }
385   } else {
386     if (inet_ntop(olsr_cnf->ip_version, &pin.in6.sin6_addr, addr, INET6_ADDRSTRLEN) == NULL)
387       addr[0] = '\0';
388     /* Use in6addr_any (::) in olsr.conf to allow anybody. */
389     if (!ip6equal(&in6addr_any, &jsoninfo_accept_ip.v6) && !ip6equal(&pin.in6.sin6_addr, &jsoninfo_accept_ip.v6)) {
390       olsr_printf(1, "(JSONINFO) From host(%s) not allowed!\n", addr);
391       close(ipc_connection);
392       return;
393     }
394   }
395
396 #ifndef NODEBUG
397   olsr_printf(2, "(JSONINFO) Connect from %s\n", addr);
398 #endif
399
400   /* purge read buffer to prevent blocking on linux */
401   FD_ZERO(&rfds);
402   FD_SET((unsigned int)ipc_connection, &rfds);  /* Win32 needs the cast here */
403   if (0 <= select(ipc_connection + 1, &rfds, NULL, NULL, &tv)) {
404     char requ[128];
405     ssize_t s = recv(ipc_connection, (void *)&requ, sizeof(requ), 0);   /* Win32 needs the cast here */
406     if (0 < s) {
407       requ[s] = 0;
408       /* print out the requested tables */
409       if (0 != strstr(requ, "/status")) send_what = SIW_ALL;
410       else { /* included in /status */
411         if (0 != strstr(requ, "/neighbors")) send_what |= SIW_NEIGHBORS;
412         if (0 != strstr(requ, "/links")) send_what |= SIW_LINKS;
413         if (0 != strstr(requ, "/routes")) send_what |= SIW_ROUTES;
414         if (0 != strstr(requ, "/hna")) send_what |= SIW_HNA;
415         if (0 != strstr(requ, "/mid")) send_what |= SIW_MID;
416         if (0 != strstr(requ, "/topology")) send_what |= SIW_TOPOLOGY;
417         if (0 != strstr(requ, "/gateways")) send_what |= SIW_GATEWAYS;
418         if (0 != strstr(requ, "/interfaces")) send_what |= SIW_INTERFACES;
419       }
420       /* these don't change during runtime, so leave them out of ALL */
421       if (0 != strstr(requ, "/config")) send_what |= SIW_CONFIG;
422       if (0 != strstr(requ, "/plugins")) send_what |= SIW_PLUGINS;
423       if (0 != strstr(requ, "/olsrd.conf")) send_what |= SIW_OLSRD_CONF;
424     }
425     if ( send_what == 0 ) send_what = SIW_ALL;
426   }
427
428   send_info(send_what, ipc_connection);
429 }
430
431 static void
432 ipc_print_neighbors(struct autobuf *abuf)
433 {
434   struct ipaddr_str buf1;
435   struct neighbor_entry *neigh;
436   struct neighbor_2_list_entry *list_2;
437   int thop_cnt;
438
439   abuf_json_open_array(abuf, "neighbors");
440
441   /* Neighbors */
442   OLSR_FOR_ALL_NBR_ENTRIES(neigh) {
443     abuf_json_open_array_entry(abuf);
444     abuf_json_string(abuf, "ipv4Address",
445                      olsr_ip_to_string(&buf1, &neigh->neighbor_main_addr));
446     abuf_json_boolean(abuf, "symmetric", (neigh->status == SYM));
447     abuf_json_boolean(abuf, "multiPointRelay", neigh->is_mpr);
448     abuf_json_boolean(abuf, "multiPointRelaySelector",
449                       olsr_lookup_mprs_set(&neigh->neighbor_main_addr) != NULL);
450     abuf_json_int(abuf, "willingness", neigh->willingness);
451     abuf_appendf(abuf, ",\n");
452
453     thop_cnt = 0;
454     if (neigh->neighbor_2_list.next) {
455       abuf_appendf(abuf, "\t\"twoHopNeighbors\": [");
456       for (list_2 = neigh->neighbor_2_list.next; list_2 != &neigh->neighbor_2_list; list_2 = list_2->next) {
457         if (thop_cnt)
458           abuf_appendf(abuf, ", ");
459         abuf_appendf(abuf,
460                      "\"%s\"",
461                      olsr_ip_to_string(&buf1, &list_2->neighbor_2->neighbor_2_addr));
462         thop_cnt++;
463       }
464     }
465     abuf_appendf(abuf, "]");
466     abuf_json_int(abuf, "twoHopNeighborCount", thop_cnt);
467     abuf_json_close_array_entry(abuf);
468   }
469   OLSR_FOR_ALL_NBR_ENTRIES_END(neigh);
470   abuf_json_close_array(abuf);
471 }
472
473 static void
474 ipc_print_links(struct autobuf *abuf)
475 {
476   struct ipaddr_str buf1, buf2;
477   struct lqtextbuffer lqbuffer1, lqbuffer2;
478
479   struct link_entry *my_link = NULL;
480
481   abuf_json_open_array(abuf, "links");
482   OLSR_FOR_ALL_LINK_ENTRIES(my_link) {
483     const char* lqs;
484     int diff = (unsigned int)(my_link->link_timer->timer_clock - now_times);
485
486     abuf_json_open_array_entry(abuf);
487     abuf_json_string(abuf, "localIP",
488                      olsr_ip_to_string(&buf1, &my_link->local_iface_addr));
489     abuf_json_string(abuf, "remoteIP",
490                      olsr_ip_to_string(&buf2, &my_link->neighbor_iface_addr));
491     abuf_json_int(abuf, "validityTime", diff);
492     lqs = get_link_entry_text(my_link, '\t', &lqbuffer1);
493     abuf_json_float(abuf, "linkQuality", atof(lqs));
494     abuf_json_float(abuf, "neighborLinkQuality", atof(strrchr(lqs, '\t')));
495     abuf_json_float(abuf, "linkCost",
496                     atof(get_linkcost_text(my_link->linkcost, false, &lqbuffer2)));
497     abuf_json_close_array_entry(abuf);
498   }
499   OLSR_FOR_ALL_LINK_ENTRIES_END(my_link);
500   abuf_json_close_array(abuf);
501 }
502
503 static void
504 ipc_print_routes(struct autobuf *abuf)
505 {
506   struct ipaddr_str buf1, buf2;
507   struct rt_entry *rt;
508   struct lqtextbuffer lqbuffer;
509
510   //abuf_puts(abuf, "Table: Routes\nDestination\tGateway IP\tMetric\tETX\tInterface\n");
511   abuf_json_open_array(abuf, "routes");
512
513   /* Walk the route table */
514   OLSR_FOR_ALL_RT_ENTRIES(rt) {
515     abuf_json_open_array_entry(abuf);
516     abuf_json_string(abuf, "destination",
517                      olsr_ip_to_string(&buf1, &rt->rt_dst.prefix));
518     abuf_json_int(abuf, "genmask", rt->rt_dst.prefix_len);
519     abuf_json_string(abuf, "gateway",
520                      olsr_ip_to_string(&buf2, &rt->rt_best->rtp_nexthop.gateway));
521     abuf_json_int(abuf, "metric", rt->rt_best->rtp_metric.hops);
522     abuf_json_float(abuf, "expectedTransmissionCount",
523                     atof(get_linkcost_text(rt->rt_best->rtp_metric.cost, true, &lqbuffer)));
524     abuf_json_string(abuf, "interface",
525                      if_ifwithindex_name(rt->rt_best->rtp_nexthop.iif_index));
526     abuf_json_close_array_entry(abuf);
527   }
528   OLSR_FOR_ALL_RT_ENTRIES_END(rt);
529
530   abuf_json_close_array(abuf);
531 }
532
533 static void
534 ipc_print_topology(struct autobuf *abuf)
535 {
536   struct tc_entry *tc;
537
538   abuf_json_open_array(abuf, "topology");
539   //abuf_puts(abuf, "Table: Topology\nDest. IP\tLast hop IP\tLQ\tNLQ\tCost\tVTime\n");
540
541   /* Topology */
542   OLSR_FOR_ALL_TC_ENTRIES(tc) {
543     struct tc_edge_entry *tc_edge;
544     OLSR_FOR_ALL_TC_EDGE_ENTRIES(tc, tc_edge) {
545       if (tc_edge->edge_inv) {
546         struct ipaddr_str dstbuf, addrbuf;
547         struct lqtextbuffer lqbuffer1, lqbuffer2;
548         uint32_t vt = tc->validity_timer != NULL ? (tc->validity_timer->timer_clock - now_times) : 0;
549         int diff = (int)(vt);
550         const char* lqs;
551         abuf_json_open_array_entry(abuf);
552         abuf_json_string(abuf, "destinationIP",
553                          olsr_ip_to_string(&dstbuf, &tc_edge->T_dest_addr));
554         abuf_json_string(abuf, "lastHopIP",
555                          olsr_ip_to_string(&addrbuf, &tc->addr));
556         lqs = get_tc_edge_entry_text(tc_edge, '\t', &lqbuffer1);
557         abuf_json_float(abuf, "linkQuality", atof(lqs));
558         abuf_json_float(abuf, "neighborLinkQuality", atof(strrchr(lqs, '\t')));
559         abuf_json_float(abuf, "cost", 
560                         atof(get_linkcost_text(tc_edge->cost, false, &lqbuffer2)));
561         abuf_json_int(abuf, "validityTime", diff);
562         abuf_json_close_array_entry(abuf);
563       }
564     }
565     OLSR_FOR_ALL_TC_EDGE_ENTRIES_END(tc, tc_edge);
566   }
567   OLSR_FOR_ALL_TC_ENTRIES_END(tc);
568
569   abuf_json_close_array(abuf);
570 }
571
572 static void
573 ipc_print_hna(struct autobuf *abuf)
574 {
575   struct ip_prefix_list *hna;
576   struct hna_entry *tmp_hna;
577   struct hna_net *tmp_net;
578   struct ipaddr_str buf, mainaddrbuf;
579
580   abuf_json_open_array(abuf, "hna");
581
582   /* Announced HNA entries */
583   if (olsr_cnf->ip_version == AF_INET) {
584     for (hna = olsr_cnf->hna_entries; hna != NULL; hna = hna->next) {
585       abuf_json_open_array_entry(abuf);
586       abuf_json_string(abuf, "destination",
587                        olsr_ip_to_string(&buf, &hna->net.prefix));
588       abuf_json_int(abuf, "genmask", hna->net.prefix_len);
589       abuf_json_string(abuf, "gateway",
590                        olsr_ip_to_string(&mainaddrbuf, &olsr_cnf->main_addr));
591       abuf_json_close_array_entry(abuf);
592     }
593   } else {
594     for (hna = olsr_cnf->hna_entries; hna != NULL; hna = hna->next) {
595       abuf_json_open_array_entry(abuf);
596       abuf_json_string(abuf, "destination",
597                        olsr_ip_to_string(&buf, &hna->net.prefix));
598       abuf_json_int(abuf, "genmask", hna->net.prefix_len);
599       abuf_json_string(abuf, "gateway",
600                        olsr_ip_to_string(&mainaddrbuf, &olsr_cnf->main_addr));
601       abuf_json_close_array_entry(abuf);
602     }
603   }
604
605   /* HNA entries */
606   OLSR_FOR_ALL_HNA_ENTRIES(tmp_hna) {
607
608     /* Check all networks */
609     for (tmp_net = tmp_hna->networks.next; tmp_net != &tmp_hna->networks; tmp_net = tmp_net->next) {
610       uint32_t vt = tmp_net->hna_net_timer != NULL ? (tmp_net->hna_net_timer->timer_clock - now_times) : 0;
611       int diff = (int)(vt);
612       abuf_json_open_array_entry(abuf);
613       abuf_json_string(abuf, "destination",
614                        olsr_ip_to_string(&buf, &tmp_net->hna_prefix.prefix)),
615       abuf_json_int(abuf, "genmask", tmp_net->hna_prefix.prefix_len);
616       abuf_json_string(abuf, "gateway",
617                        olsr_ip_to_string(&mainaddrbuf, &tmp_hna->A_gateway_addr));
618       abuf_json_int(abuf, "validityTime", diff);
619       abuf_json_close_array_entry(abuf);
620     }
621   }
622   OLSR_FOR_ALL_HNA_ENTRIES_END(tmp_hna);
623
624   abuf_json_close_array(abuf);
625 }
626
627 static void
628 ipc_print_mid(struct autobuf *abuf)
629 {
630   int idx;
631   struct mid_entry *entry;
632   struct mid_address *alias;
633
634   abuf_json_open_array(abuf, "mid");
635
636   /* MID */
637   for (idx = 0; idx < HASHSIZE; idx++) {
638     entry = mid_set[idx].next;
639
640     while (entry != &mid_set[idx]) {
641       struct ipaddr_str buf, buf2;
642       alias = entry->aliases;
643
644       while (alias) {
645         uint32_t vt = alias->vtime - now_times;
646         int diff = (int)(vt);
647
648         abuf_json_open_array_entry(abuf);
649         abuf_json_string(abuf, "ipv4Address",
650                          olsr_ip_to_string(&buf, &entry->main_addr));
651         abuf_json_string(abuf, "alias",
652                          olsr_ip_to_string(&buf2, &alias->alias));
653         abuf_json_open_array_entry(abuf);
654         abuf_json_int(abuf, "validityTime", diff);
655         abuf_json_close_array_entry(abuf);
656         alias = alias->next_alias;
657       }
658       entry = entry->next;
659     }
660   }
661   abuf_json_close_array(abuf);
662 }
663
664 static void
665 ipc_print_gateways(struct autobuf *abuf)
666 {
667 #ifndef linux
668   abuf_json_string(abuf, "error", "Gateway mode is only supported in Linux");
669 #else
670   static const char IPV4[] = "ipv4";
671   static const char IPV4_NAT[] = "ipv4(n)";
672   static const char IPV6[] = "ipv6";
673   static const char NONE[] = "-";
674
675   struct ipaddr_str buf;
676   struct gateway_entry *gw;
677   struct lqtextbuffer lqbuf;
678
679   // Status IP ETX Hopcount Uplink-Speed Downlink-Speed ipv4/ipv4-nat/- ipv6/- ipv6-prefix/-
680   abuf_json_open_array(abuf, "gateways");
681   //abuf_puts(abuf, "Table: Gateways\nStatus\tGateway IP\tETX\tHopcnt\tUplink\tDownlnk\tIPv4\tIPv6\tPrefix\n");
682   OLSR_FOR_ALL_GATEWAY_ENTRIES(gw) {
683     char v4 = '-', v6 = '-';
684     bool autoV4 = false, autoV6 = false;
685     const char *v4type = NONE, *v6type = NONE;
686     struct tc_entry *tc;
687
688     if ((tc = olsr_lookup_tc_entry(&gw->originator)) == NULL) {
689       continue;
690     }
691
692     if (gw == olsr_get_ipv4_inet_gateway(&autoV4)) {
693       v4 = autoV4 ? 'a' : 's';
694     } else if (gw->ipv4 && (olsr_cnf->ip_version == AF_INET || olsr_cnf->use_niit)
695                && (olsr_cnf->smart_gw_allow_nat || !gw->ipv4nat)) {
696       v4 = 'u';
697     }
698
699     if (gw == olsr_get_ipv6_inet_gateway(&autoV6)) {
700       v6 = autoV6 ? 'a' : 's';
701     } else if (gw->ipv6 && olsr_cnf->ip_version == AF_INET6) {
702       v6 = 'u';
703     }
704
705     if (gw->ipv4) {
706       v4type = gw->ipv4nat ? IPV4_NAT : IPV4;
707     }
708     if (gw->ipv6) {
709       v6type = IPV6;
710     }
711
712     abuf_appendf(abuf, "%c%c\t%s\t%s\t%d\t%u\t%u\t%s\t%s\t%s\n",
713                  v4, v6, olsr_ip_to_string(&buf, &gw->originator),
714                  get_linkcost_text(tc->path_cost, true, &lqbuf), tc->hops,
715                  gw->uplink, gw->downlink, v4type, v6type,
716                  gw->external_prefix.prefix_len == 0 ? NONE : olsr_ip_prefix_to_string(&gw->external_prefix));
717   }
718   OLSR_FOR_ALL_GATEWAY_ENTRIES_END(gw)
719   abuf_json_close_array(abuf);
720 #endif
721 }
722
723
724 static void
725 ipc_print_plugins(struct autobuf *abuf)
726 {
727   struct plugin_entry *pentry;
728   struct plugin_param *pparam;
729   abuf_json_open_array(abuf, "plugins");
730   if (olsr_cnf->plugins)
731     for (pentry = olsr_cnf->plugins; pentry; pentry = pentry->next) {
732       abuf_json_open_array_entry(abuf);
733       abuf_json_string(abuf, "plugin", pentry->name);
734       for (pparam = pentry->params; pparam; pparam = pparam->next) {
735        abuf_json_string(abuf, pparam->key, pparam->value);
736       }
737       abuf_json_close_array_entry(abuf);
738     }
739   abuf_json_close_array(abuf);
740 }
741
742
743 static void
744 ipc_print_config(struct autobuf *abuf)
745 {
746   struct ipaddr_str mainaddrbuf;
747   struct ip_prefix_list *ipcn;
748
749   abuf_json_open_object(abuf, "config");
750
751   abuf_json_int(abuf, "olsrPort", olsr_cnf->olsrport);
752   abuf_json_int(abuf, "debugLevel", olsr_cnf->debug_level);
753   abuf_json_int(abuf, "noFork", olsr_cnf->no_fork);
754   abuf_json_boolean(abuf, "hostEmulation", olsr_cnf->host_emul);
755   abuf_json_int(abuf, "ipVersion", olsr_cnf->ip_version);
756   abuf_json_boolean(abuf, "allowNoInterfaces", olsr_cnf->allow_no_interfaces);
757   abuf_json_int(abuf, "typeOfService", olsr_cnf->tos);
758   abuf_json_int(abuf, "rtProto", olsr_cnf->rt_proto);
759   abuf_json_int(abuf, "rtTable", olsr_cnf->rt_table);
760   abuf_json_int(abuf, "rtTableDefault", olsr_cnf->rt_table_default);
761   abuf_json_int(abuf, "rtTableTunnel", olsr_cnf->rt_table_tunnel);
762   abuf_json_int(abuf, "rtTablePriority", olsr_cnf->rt_table_pri);
763   abuf_json_int(abuf, "rtTableTunnelPriority", olsr_cnf->rt_table_tunnel_pri);
764   abuf_json_int(abuf, "rtTableDefauiltOlsrPriority", olsr_cnf->rt_table_defaultolsr_pri);
765   abuf_json_int(abuf, "rtTableDefaultPriority", olsr_cnf->rt_table_default_pri);
766   abuf_json_int(abuf, "willingness", olsr_cnf->willingness);
767   abuf_json_boolean(abuf, "willingnessAuto", olsr_cnf->willingness_auto);
768
769   abuf_json_string(abuf, "fibMetrics", FIB_METRIC_TXT[olsr_cnf->fib_metric]);
770   /*
771   struct if_config_options *interface_defaults;
772   */
773   abuf_json_int(abuf, "ipcConnections", olsr_cnf->ipc_connections);
774   if (olsr_cnf->ipc_connections)
775     for (ipcn = olsr_cnf->ipc_nets; ipcn != NULL; ipcn = ipcn->next) {
776       abuf_json_string(abuf, "ipcAllowedAddress",
777                        olsr_ip_to_string(&mainaddrbuf, &ipcn->net.prefix));
778       abuf_json_int(abuf, "ipcAllowedAddressMask", ipcn->net.prefix_len);
779     }
780
781   // keep all time in ms, so convert these two, which are in seconds
782   abuf_json_int(abuf, "pollRate", olsr_cnf->pollrate * 1000);
783   abuf_json_int(abuf, "nicChangePollInterval", olsr_cnf->nic_chgs_pollrate * 1000);
784   abuf_json_boolean(abuf, "clearScreen", olsr_cnf->clear_screen);
785   abuf_json_int(abuf, "tcRedundancy", olsr_cnf->tc_redundancy);
786   abuf_json_int(abuf, "mprCoverage", olsr_cnf->mpr_coverage);
787
788   if (olsr_cnf->lq_level == 0) {
789     abuf_json_boolean(abuf, "useHysteresis", olsr_cnf->use_hysteresis);
790     if (olsr_cnf->use_hysteresis) {
791       abuf_json_float(abuf, "hysteresisScaling", olsr_cnf->hysteresis_param.scaling);
792       abuf_json_float(abuf, "hysteresisLowThreshold", olsr_cnf->hysteresis_param.thr_low);
793       abuf_json_float(abuf, "hysteresisHighThreshold", olsr_cnf->hysteresis_param.thr_high);
794     }
795   }
796   abuf_json_int(abuf, "linkQualityLevel", olsr_cnf->lq_level);
797   abuf_json_int(abuf, "linkQualityFisheye", olsr_cnf->lq_fish);
798   abuf_json_float(abuf, "linkQualityAging", olsr_cnf->lq_aging);
799   abuf_json_string(abuf, "linkQualityAlgorithm", olsr_cnf->lq_algorithm);
800   // keep all time in ms, so convert this from seconds
801   abuf_json_int(abuf, "minTcValidTime", olsr_cnf->min_tc_vtime * 1000);
802   abuf_json_boolean(abuf, "setIpForward", olsr_cnf->set_ip_forward);
803   abuf_json_string(abuf, "lockFile", olsr_cnf->lock_file);
804   abuf_json_boolean(abuf, "useNiit", olsr_cnf->use_niit);
805
806   abuf_json_boolean(abuf, "smartGateway", olsr_cnf->smart_gw_active);
807   if (olsr_cnf->smart_gw_active) {
808     abuf_json_boolean(abuf, "smartGatewayAllowNat", olsr_cnf->smart_gw_allow_nat);
809     abuf_json_boolean(abuf, "smartGatewayUplinkNat", olsr_cnf->smart_gw_uplink_nat);
810     abuf_json_int(abuf, "smartGatewayPeriod", olsr_cnf->smart_gw_period);
811     abuf_json_int(abuf, "smartGatewayStableCount", olsr_cnf->smart_gw_stablecount);
812     abuf_json_int(abuf, "smartGatewayThreshold", olsr_cnf->smart_gw_thresh);
813     abuf_json_int(abuf, "smartGatewayUplink", olsr_cnf->smart_gw_uplink);
814     abuf_json_int(abuf, "smartGatewayDownlink", olsr_cnf->smart_gw_downlink);
815     abuf_json_int(abuf, "smartGatewayType", olsr_cnf->smart_gw_type);
816     abuf_json_string(abuf, "smartGatewayPrefix",
817                      olsr_ip_to_string(&mainaddrbuf, &olsr_cnf->smart_gw_prefix.prefix));
818     abuf_json_int(abuf, "smartGatewayPrefixLength", olsr_cnf->smart_gw_prefix.prefix_len);
819   }
820
821   abuf_json_string(abuf, "mainIpAddress",
822                    olsr_ip_to_string(&mainaddrbuf, &olsr_cnf->main_addr));
823   abuf_json_string(abuf, "unicastSourceIpAddress",
824                    olsr_ip_to_string(&mainaddrbuf, &olsr_cnf->unicast_src_ip));
825
826   abuf_json_boolean(abuf, "useSourceIpRoutes", olsr_cnf->use_src_ip_routes);
827
828   abuf_json_int(abuf, "maxPrefixLength", olsr_cnf->maxplen);
829   abuf_json_int(abuf, "ipSize", olsr_cnf->ipsize);
830   abuf_json_boolean(abuf, "deleteInternetGatewaysAtStartup", olsr_cnf->del_gws);
831   // keep all time in ms, so convert this from seconds
832   abuf_json_int(abuf, "willingnessUpdateInterval", olsr_cnf->will_int * 1000);
833   abuf_json_float(abuf, "maxSendMessageJitter", olsr_cnf->max_jitter);
834   abuf_json_int(abuf, "exitValue", olsr_cnf->exit_value);
835   // keep all time in ms, so convert this from seconds
836   abuf_json_int(abuf, "maxTcValidTime", olsr_cnf->max_tc_vtime * 1000);
837
838   abuf_json_int(abuf, "niit4to6InterfaceIndex", olsr_cnf->niit4to6_if_index);
839   abuf_json_int(abuf, "niit6to4InterfaceIndex", olsr_cnf->niit6to4_if_index);
840
841   abuf_json_boolean(abuf, "hasIpv4Gateway", olsr_cnf->has_ipv4_gateway);
842   abuf_json_boolean(abuf, "hasIpv6Gateway", olsr_cnf->has_ipv6_gateway);
843
844   abuf_json_int(abuf, "ioctlSocket", olsr_cnf->ioctl_s);
845 #ifdef __linux__
846   abuf_json_int(abuf, "routeNetlinkSocket", olsr_cnf->rtnl_s);
847   abuf_json_int(abuf, "routeMonitorSocket", olsr_cnf->rt_monitor_socket);
848 #endif
849
850 #if defined __FreeBSD__ || defined __FreeBSD_kernel__ || defined __APPLE__ || defined __NetBSD__ || defined __OpenBSD__
851   abuf_json_int(abuf, "routeChangeSocket", olsr_cnf->rts);
852 #endif
853   abuf_json_float(abuf, "linkQualityNatThreshold", olsr_cnf->lq_nat_thresh);
854
855   abuf_json_string(abuf, "olsrdVersion", olsrd_version);
856   abuf_json_string(abuf, "olsrdBuildDate", build_date);
857   abuf_json_string(abuf, "olsrdBuildHost", build_host);
858
859 #if defined _WIN32 || defined _WIN64
860   abuf_json_string(abuf, "os", "Windows");
861 #elif defined __gnu_linux__
862   abuf_json_string(abuf, "os", "GNU/Linux");
863 #elif defined __ANDROID__
864   abuf_json_string(abuf, "os", "Android");
865 #elif defined __APPLE__
866   abuf_json_string(abuf, "os", "Mac OS X");
867 #elif defined __NetBSD__
868   abuf_json_string(abuf, "os", "NetBSD");
869 #elif defined __OpenBSD__
870   abuf_json_string(abuf, "os", "OpenBSD");
871 #elif defined __FreeBSD__ || defined __FreeBSD_kernel__
872   abuf_json_string(abuf, "os", "FreeBSD");
873 #else
874   abuf_json_string(abuf, "os", "Undefined");
875 #endif
876
877   abuf_json_int(abuf, "systemTime", time(NULL));
878   abuf_json_int(abuf, "startTime", start_time.tv_sec);
879
880   abuf_json_close_object(abuf);
881 }
882
883 static void
884 ipc_print_interfaces(struct autobuf *abuf)
885 {
886   const struct olsr_if *ifs;
887   abuf_json_open_array(abuf, "interfaces");
888   //abuf_puts(abuf, "Table: Interfaces\nName\tState\tMTU\tWLAN\tSrc-Adress\tMask\tDst-Adress\n");
889   for (ifs = olsr_cnf->interfaces; ifs != NULL; ifs = ifs->next) {
890     const struct interface *const rifs = ifs->interf;
891     abuf_json_open_array_entry(abuf);
892     abuf_json_string(abuf, "name", ifs->name);
893     if (!rifs) {
894       abuf_json_string(abuf, "state", "down");
895     } else {
896       abuf_json_string(abuf, "state", "up");
897       abuf_json_int(abuf, "mtu", rifs->int_mtu);
898       abuf_json_boolean(abuf, "wireless", rifs->is_wireless);
899
900       if (olsr_cnf->ip_version == AF_INET) {
901         struct ipaddr_str addrbuf, maskbuf, bcastbuf;
902         abuf_json_string(abuf, "ipv4Address",
903                              ip4_to_string(&addrbuf, rifs->int_addr.sin_addr));
904         abuf_json_string(abuf, "netmask",
905                              ip4_to_string(&maskbuf, rifs->int_netmask.sin_addr));
906         abuf_json_string(abuf, "broadcast",
907                              ip4_to_string(&bcastbuf, rifs->int_broadaddr.sin_addr));
908       } else {
909         struct ipaddr_str addrbuf, maskbuf;
910         abuf_json_string(abuf, "ipv6Address",
911                              ip6_to_string(&addrbuf, &rifs->int6_addr.sin6_addr));
912         abuf_json_string(abuf, "multicast",
913                              ip6_to_string(&maskbuf, &rifs->int6_multaddr.sin6_addr));
914       }
915     }
916     abuf_json_close_array_entry(abuf);
917   }
918   abuf_json_close_array(abuf);
919 }
920
921
922 static void
923 ipc_print_olsrd_conf(struct autobuf *abuf)
924 {
925   olsrd_write_cnf_autobuf(abuf, olsr_cnf);
926 }
927
928
929 static void
930 jsoninfo_write_data(void *foo __attribute__ ((unused)))
931 {
932   fd_set set;
933   int result, i, j, max;
934   struct timeval tv;
935
936   FD_ZERO(&set);
937   max = 0;
938   for (i=0; i<outbuffer_count; i++) {
939     /* And we cast here since we get a warning on Win32 */
940     FD_SET((unsigned int)(outbuffer_socket[i]), &set);
941
942     if (outbuffer_socket[i] > max) {
943       max = outbuffer_socket[i];
944     }
945   }
946
947   tv.tv_sec = 0;
948   tv.tv_usec = 0;
949
950   result = select(max + 1, NULL, &set, NULL, &tv);
951   if (result <= 0) {
952     return;
953   }
954
955   for (i=0; i<outbuffer_count; i++) {
956     if (FD_ISSET(outbuffer_socket[i], &set)) {
957       result = send(outbuffer_socket[i],
958                     outbuffer[i] + outbuffer_written[i],
959                     outbuffer_size[i] - outbuffer_written[i],
960                     0);
961       if (result > 0) {
962         outbuffer_written[i] += result;
963       }
964
965       if (result <= 0 || outbuffer_written[i] == outbuffer_size[i]) {
966         /* close this socket and cleanup*/
967         close(outbuffer_socket[i]);
968         free (outbuffer[i]);
969
970         for (j=i+1; j<outbuffer_count; j++) {
971           outbuffer[j-1] = outbuffer[j];
972           outbuffer_size[j-1] = outbuffer_size[j];
973           outbuffer_socket[j-1] = outbuffer_socket[j];
974           outbuffer_written[j-1] = outbuffer_written[j];
975         }
976         outbuffer_count--;
977       }
978     }
979   }
980   if (outbuffer_count == 0) {
981     olsr_stop_timer(writetimer_entry);
982   }
983 }
984
985 static void
986 send_info(unsigned int send_what, int the_socket)
987 {
988   struct autobuf abuf;
989
990   abuf_init(&abuf, 32768);
991
992   /* wrap everything in a JSON array to handle multiple elements*/
993   abuf_puts(&abuf, "\n[\n");
994
995   /* Print tables to IPC socket */
996
997   if ((send_what & SIW_LINKS) == SIW_LINKS) {
998     ipc_print_links(&abuf);
999     // if this is being output with others, then include commas
1000     if (send_what != SIW_LINKS) abuf_puts(&abuf, ",");
1001   }
1002   if ((send_what & SIW_NEIGHBORS) == SIW_NEIGHBORS) {
1003     ipc_print_neighbors(&abuf);
1004     if (send_what != SIW_NEIGHBORS) abuf_puts(&abuf, ",");
1005   }
1006   if ((send_what & SIW_TOPOLOGY) == SIW_TOPOLOGY) {
1007     ipc_print_topology(&abuf);
1008     if (send_what != SIW_TOPOLOGY) abuf_puts(&abuf, ",");
1009   }
1010   if ((send_what & SIW_HNA) == SIW_HNA) {
1011     ipc_print_hna(&abuf);
1012     if (send_what != SIW_HNA) abuf_puts(&abuf, ",");
1013   }
1014   if ((send_what & SIW_MID) == SIW_MID) {
1015     ipc_print_mid(&abuf);
1016     if (send_what != SIW_MID) abuf_puts(&abuf, ",");
1017   }
1018   if ((send_what & SIW_ROUTES) == SIW_ROUTES) {
1019     ipc_print_routes(&abuf);
1020     if (send_what != SIW_ROUTES) abuf_puts(&abuf, ",");
1021   }
1022   if ((send_what & SIW_GATEWAYS) == SIW_GATEWAYS) {
1023     ipc_print_gateways(&abuf);
1024     if (send_what != SIW_GATEWAYS) abuf_puts(&abuf, ",");
1025   }
1026   if ((send_what & SIW_INTERFACES) == SIW_INTERFACES) {
1027     ipc_print_interfaces(&abuf);
1028     if (send_what != SIW_INTERFACES) abuf_puts(&abuf, ",");
1029   }
1030   if ((send_what & SIW_CONFIG) == SIW_CONFIG) {
1031     ipc_print_config(&abuf);
1032     if (send_what != SIW_CONFIG) abuf_puts(&abuf, ",");
1033   }
1034   if ((send_what & SIW_PLUGINS) == SIW_PLUGINS) {
1035     ipc_print_plugins(&abuf);
1036   }
1037
1038   /* end of JSON array for status */
1039   abuf_puts(&abuf, "]\n");
1040
1041   /* this outputs the olsrd.conf text directly, not JSON */
1042   if ((send_what & SIW_OLSRD_CONF) == SIW_OLSRD_CONF) {
1043     ipc_print_olsrd_conf(&abuf);
1044   }
1045
1046   outbuffer[outbuffer_count] = olsr_malloc(abuf.len, "txt output buffer");
1047   outbuffer_size[outbuffer_count] = abuf.len;
1048   outbuffer_written[outbuffer_count] = 0;
1049   outbuffer_socket[outbuffer_count] = the_socket;
1050
1051   memcpy(outbuffer[outbuffer_count], abuf.buf, abuf.len);
1052   outbuffer_count++;
1053
1054   if (outbuffer_count == 1) {
1055     writetimer_entry = olsr_start_timer(100,
1056                                         0,
1057                                         OLSR_TIMER_PERIODIC,
1058                                         &jsoninfo_write_data,
1059                                         NULL,
1060                                         0);
1061   }
1062
1063   abuf_free(&abuf);
1064 }
1065
1066 /*
1067  * Local Variables:
1068  * mode: c
1069  * style: linux
1070  * c-basic-offset: 2
1071  * indent-tabs-mode: nil
1072  * End:
1073  */