jsoninfo: in /interfaces show 'state' text as 'up' boolean
[olsrd.git] / lib / jsoninfo / src / olsrd_jsoninfo.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.h"
43
44 #include <unistd.h>
45 #include <ctype.h>
46 #include <libgen.h>
47
48 #include "ipcalc.h"
49 #include "builddata.h"
50 #include "neighbor_table.h"
51 #include "mpr_selector_set.h"
52 #include "mid_set.h"
53 #include "routing_table.h"
54 #include "lq_plugin.h"
55 #include "gateway.h"
56 #include "olsrd_plugin.h"
57 #include "../../info/info_types.h"
58 #include "../../info/http_headers.h"
59 #include "gateway_default_handler.h"
60 #include "egressTypes.h"
61 #include "olsrd_jsoninfo_helpers.h"
62
63 struct timeval start_time;
64
65 void plugin_init(const char *plugin_name) {
66   /* Get start time */
67   gettimeofday(&start_time, NULL);
68
69   if (!strlen(uuidfile)) {
70     strscpy(uuidfile, "uuid.txt", sizeof(uuidfile));
71   }
72   read_uuid_from_file(plugin_name, uuidfile);
73 }
74
75 bool isCommand(const char *str, unsigned long long siw) {
76   const char * cmd;
77   switch (siw) {
78     case SIW_OLSRD_CONF:
79       cmd = "/olsrd.conf";
80       break;
81
82     case SIW_ALL:
83       cmd = "/all";
84       break;
85
86     case SIW_RUNTIME_ALL:
87       cmd = "/runtime";
88       break;
89
90     case SIW_STARTUP_ALL:
91       cmd = "/startup";
92       break;
93
94     case SIW_NEIGHBORS:
95       cmd = "/neighbors";
96       break;
97
98     case SIW_LINKS:
99       cmd = "/links";
100       break;
101
102     case SIW_ROUTES:
103       cmd = "/routes";
104       break;
105
106     case SIW_HNA:
107       cmd = "/hna";
108       break;
109
110     case SIW_MID:
111       cmd = "/mid";
112       break;
113
114     case SIW_TOPOLOGY:
115       cmd = "/topology";
116       break;
117
118     case SIW_GATEWAYS:
119       cmd = "/gateways";
120       break;
121
122     case SIW_INTERFACES:
123       cmd = "/interfaces";
124       break;
125
126     case SIW_2HOP:
127       cmd = "/2hop";
128       break;
129
130     case SIW_SGW:
131       cmd = "/sgw";
132       break;
133
134     case SIW_VERSION:
135       cmd = "/version";
136       break;
137
138     case SIW_CONFIG:
139       cmd = "/config";
140       break;
141
142     case SIW_PLUGINS:
143       cmd = "/plugins";
144       break;
145
146     case SIW_NEIGHBORS_FREIFUNK:
147       cmd = "/neighbours";
148       break;
149
150     default:
151       return false;
152   }
153
154   return !strcmp(str, cmd);
155 }
156
157 const char * determine_mime_type(unsigned int send_what) {
158   return (send_what & SIW_OLSRD_CONF) ? "text/plain; charset=utf-8" : "application/json; charset=utf-8";
159 }
160
161 void output_start(struct autobuf *abuf) {
162   /* global variables for tracking when to put a comma in for JSON */
163   abuf_json_reset_entry_number_and_depth();
164   abuf_json_mark_output(true, abuf);
165
166   abuf_json_int(abuf, "systemTime", time(NULL));
167   abuf_json_int(abuf, "timeSinceStartup", now_times);
168   if (*uuid) {
169     abuf_json_string(abuf, "uuid", uuid);
170   }
171 }
172
173 void output_end(struct autobuf *abuf) {
174   abuf_json_mark_output(false, abuf);
175   abuf_puts(abuf, "\n");
176 }
177
178 void output_error(struct autobuf *abuf, unsigned int status, const char * req, bool http_headers) {
179   struct autobuf abufInternal;
180
181   if (http_headers || (status == INFO_HTTP_OK)) {
182     return;
183   }
184
185   abuf_init(&abufInternal, 1024);
186
187   output_start(abuf);
188
189   switch (status) {
190     case INFO_HTTP_NOTFOUND:
191       abuf_appendf(&abufInternal, "Invalid request '%s'", req);
192       abuf_json_string(abuf, "error", abufInternal.buf);
193       break;
194
195     case INFO_HTTP_NOCONTENT:
196       abuf_json_string(abuf, "error", "no content");
197       break;
198
199     default:
200       abuf_appendf(&abufInternal, "Unknown status %d for request '%s'", status, req);
201       abuf_json_string(abuf, "error", abufInternal.buf);
202       break;
203   }
204
205   output_end(abuf);
206 }
207
208 static void print_msg_params(struct autobuf *abuf, struct olsr_msg_params *params, const char * name) {
209   assert(abuf);
210   assert(params);
211   assert(name);
212
213   abuf_json_mark_object(true, false, abuf, name);
214   abuf_json_float(abuf, "emissionInterval", params->emission_interval);
215   abuf_json_float(abuf, "validityTime", params->validity_time);
216   abuf_json_mark_object(false, false, abuf, NULL);
217 }
218
219 static void print_hna_array_entry(struct autobuf *abuf, union olsr_ip_addr *gw, union olsr_ip_addr *ip, uint8_t prefix_len, long long validityTime) {
220   assert(abuf);
221
222   abuf_json_mark_array_entry(true, abuf);
223   abuf_json_ip_address(abuf, "gateway", gw);
224   abuf_json_ip_address(abuf, "destination", ip);
225   abuf_json_int(abuf, "genmask", prefix_len);
226   abuf_json_int(abuf, "validityTime", validityTime);
227   abuf_json_mark_array_entry(false, abuf);
228 }
229
230 static void print_link_quality_multipliers_array_entry(struct autobuf *abuf, struct olsr_lq_mult *mult) {
231   assert(abuf);
232   assert(mult);
233
234   abuf_json_mark_array_entry(true, abuf);
235   abuf_json_ip_address(abuf, "route", &mult->addr);
236   abuf_json_float(abuf, "multiplier", mult->value / 65535.0);
237   abuf_json_mark_array_entry(false, abuf);
238 }
239
240 static void print_ipc_net_array_entry(struct autobuf *abuf, struct ip_prefix_list *ipc_nets) {
241   assert(abuf);
242   assert(ipc_nets);
243
244   abuf_json_mark_array_entry(true, abuf);
245   abuf_json_boolean(abuf, "host", (ipc_nets->net.prefix_len == olsr_cnf->maxplen));
246   abuf_json_ip_address(abuf, "ipAddress", &ipc_nets->net.prefix);
247   abuf_json_int(abuf, "genmask", ipc_nets->net.prefix_len);
248   abuf_json_mark_array_entry(false, abuf);
249 }
250
251 static void print_interface_config(struct autobuf *abuf, const char * name, struct if_config_options* id) {
252   assert(abuf);
253   assert(name);
254
255   abuf_json_mark_object(true, false, abuf, name);
256   if (id) {
257     struct olsr_lq_mult *mult;
258
259     abuf_json_ip_address(abuf, "ipv4Broadcast", &id->ipv4_multicast);
260     abuf_json_ip_address(abuf, "ipv6Multicast", &id->ipv6_multicast);
261
262     abuf_json_ip_address(abuf, "ipv4Source", &id->ipv4_src);
263     abuf_json_ip_address(abuf, "ipv6Source", &id->ipv6_src.prefix);
264     abuf_json_int(abuf, "ipv6SourcePrefixLength", id->ipv6_src.prefix_len);
265
266     abuf_json_string(abuf, "mode", ((id->mode < IF_MODE_MESH) || (id->mode >= IF_MODE_CNT)) ? "" : OLSR_IF_MODE[id->mode]);
267
268     abuf_json_int(abuf, "weightValue", id->weight.value);
269     abuf_json_boolean(abuf, "weightFixed", id->weight.fixed);
270     print_msg_params(abuf, &id->hello_params, "hello");
271     print_msg_params(abuf, &id->tc_params, "tc");
272     print_msg_params(abuf, &id->mid_params, "mid");
273     print_msg_params(abuf, &id->hna_params, "hna");
274     abuf_json_mark_object(true, true, abuf, "linkQualityMultipliers");
275     for (mult = olsr_cnf->interface_defaults->lq_mult; mult != NULL ; mult = mult->next) {
276       print_link_quality_multipliers_array_entry(abuf, mult);
277     }
278     abuf_json_mark_object(false, true, abuf, NULL);
279     abuf_json_int(abuf, "linkQualityMultipliersCount", id->orig_lq_mult_cnt);
280     abuf_json_boolean(abuf, "autoDetectChanges", id->autodetect_chg);
281   }
282   abuf_json_mark_object(false, false, abuf, NULL);
283 }
284
285 static void print_interface_olsr(struct autobuf *abuf, const char * name, struct interface_olsr * rifs) {
286   struct ipaddr_str addrbuf;
287
288   assert(abuf);
289   assert(name);
290
291   abuf_json_mark_object(true, false, abuf, name);
292   abuf_json_boolean(abuf, "up", rifs != NULL);
293   if (!rifs) {
294     abuf_json_mark_object(false, false, abuf, NULL);
295     return;
296   }
297
298   abuf_json_string(abuf, "ipv4Address", ip4_to_string(&addrbuf, rifs->int_addr.sin_addr));
299   abuf_json_string(abuf, "ipv4Netmask", ip4_to_string(&addrbuf, rifs->int_netmask.sin_addr));
300   abuf_json_string(abuf, "ipv4Broadcast", ip4_to_string(&addrbuf, rifs->int_broadaddr.sin_addr));
301   abuf_json_string(abuf, "mode", ((rifs->mode < IF_MODE_MESH) || (rifs->mode >= IF_MODE_CNT)) ? "" : OLSR_IF_MODE[rifs->mode]);
302
303   abuf_json_string(abuf, "ipv6Address", ip6_to_string(&addrbuf, &rifs->int6_addr.sin6_addr));
304   abuf_json_string(abuf, "ipv6Multicast", ip6_to_string(&addrbuf, &rifs->int6_multaddr.sin6_addr));
305
306   abuf_json_ip_address(abuf, "ipAddress", &rifs->ip_addr);
307   abuf_json_boolean(abuf, "emulatedInterface", rifs->is_hcif);
308
309   abuf_json_int(abuf, "olsrSocket", rifs->olsr_socket);
310   abuf_json_int(abuf, "sendSocket", rifs->send_socket);
311
312   abuf_json_int(abuf, "metric", rifs->int_metric);
313   abuf_json_int(abuf, "mtu", rifs->int_mtu);
314   abuf_json_int(abuf, "flags", rifs->int_flags);
315   abuf_json_int(abuf, "index", rifs->if_index);
316   abuf_json_boolean(abuf, "wireless", rifs->is_wireless);
317   abuf_json_string(abuf, "name", rifs->int_name);
318   abuf_json_int(abuf, "seqNum", rifs->olsr_seqnum);
319
320
321   abuf_json_int(abuf, "helloTime", rifs->hello_gen_timer ? (long) (rifs->hello_gen_timer->timer_clock - now_times) : 0);
322   abuf_json_int(abuf, "hnaTime", rifs->hna_gen_timer ? (long) (rifs->hna_gen_timer->timer_clock - now_times) : 0);
323   abuf_json_int(abuf, "midTime", rifs->mid_gen_timer ? (long) (rifs->mid_gen_timer->timer_clock - now_times) : 0);
324   abuf_json_int(abuf, "tcTime", rifs->tc_gen_timer ? (long) (rifs->tc_gen_timer->timer_clock - now_times) : 0);
325
326 #ifdef __linux__
327
328
329
330
331   abuf_json_boolean(abuf, "icmpRedirectBackup", rifs->nic_state.redirect);
332
333
334   abuf_json_boolean(abuf, "spoofFilterBackup", rifs->nic_state.spoof);
335
336 #endif /* __linux__ */
337
338   abuf_json_int(abuf, "helloEmissionInterval", rifs->hello_etime);
339   abuf_json_mark_object(true, false, abuf, "validityTimes");
340   abuf_json_int(abuf, "hello", me_to_reltime(rifs->valtimes.hello));
341   abuf_json_int(abuf, "tc", me_to_reltime(rifs->valtimes.tc));
342   abuf_json_int(abuf, "mid", me_to_reltime(rifs->valtimes.mid));
343   abuf_json_int(abuf, "hna", me_to_reltime(rifs->valtimes.hna));
344   abuf_json_mark_object(false, false, abuf, NULL);
345
346   abuf_json_int(abuf, "forwardingTimeout", rifs->fwdtimer);
347
348
349   abuf_json_int(abuf, "sgwZeroBwTimeout", rifs->sgw_sgw_zero_bw_timeout);
350
351
352   // netbuf
353
354
355   // gen_properties
356
357
358   abuf_json_int(abuf, "ttlIndex", rifs->ttl_index);
359
360
361   abuf_json_boolean(abuf, "immediateSendTc", rifs->immediate_send_tc);
362
363   abuf_json_mark_object(false, false, abuf, NULL);
364 }
365
366 #ifdef __linux__
367 static void ipc_print_gateway_entry(struct autobuf *abuf, bool ipv6, struct gateway_entry * current_gw, struct gateway_entry * gw) {
368   struct tc_entry* tc;
369
370   assert(abuf);
371   assert(gw);
372
373   tc = olsr_lookup_tc_entry(&gw->originator);
374
375   abuf_json_boolean(abuf, "selected", current_gw && (current_gw == gw));
376   abuf_json_boolean(abuf, "selectable", isGwSelectable(gw, ipv6));
377   abuf_json_ip_address(abuf, "originator", &gw->originator);
378   abuf_json_ip_address(abuf, "prefix", &gw->external_prefix.prefix);
379   abuf_json_int(abuf, "prefixLen", gw->external_prefix.prefix_len);
380   abuf_json_int(abuf, "uplink", gw->uplink);
381   abuf_json_int(abuf, "downlink", gw->downlink);
382   abuf_json_int(abuf, "cost", gw->path_cost);
383   abuf_json_boolean(abuf, "IPv4", gw->ipv4);
384   abuf_json_boolean(abuf, "IPv4-NAT", gw->ipv4nat);
385   abuf_json_boolean(abuf, "IPv6", gw->ipv6);
386   abuf_json_int(abuf, "expireTime", gw->expire_timer ? (gw->expire_timer->timer_clock - now_times) : 0);
387   abuf_json_int(abuf, "cleanupTime", gw->cleanup_timer ? (gw->cleanup_timer->timer_clock - now_times) : 0);
388
389   abuf_json_int(abuf, "pathcost", !tc ? ROUTE_COST_BROKEN : tc->path_cost);
390   abuf_json_int(abuf, "hops", !tc ? 0 : tc->hops);
391 }
392 #endif /* __linux__ */
393
394 static void ipc_print_neighbors_internal(struct autobuf *abuf, bool list_2hop) {
395   struct neighbor_entry *neigh;
396
397   assert(abuf);
398
399   if (!list_2hop) {
400     abuf_json_mark_object(true, true, abuf, "neighbors");
401   } else {
402     abuf_json_mark_object(true, true, abuf, "2hop");
403   }
404
405   /* Neighbors */
406   OLSR_FOR_ALL_NBR_ENTRIES(neigh) {
407     struct neighbor_2_list_entry *list_2;
408     int thop_cnt = 0;
409
410     abuf_json_mark_array_entry(true, abuf);
411
412     abuf_json_ip_address(abuf, "ipAddress", &neigh->neighbor_main_addr);
413     abuf_json_boolean(abuf, "symmetric", (neigh->status == SYM));
414     abuf_json_int(abuf, "willingness", neigh->willingness);
415     abuf_json_boolean(abuf, "isMultiPointRelay", neigh->is_mpr);
416     abuf_json_boolean(abuf, "wasMultiPointRelay", neigh->was_mpr);
417     abuf_json_boolean(abuf, "multiPointRelaySelector", olsr_lookup_mprs_set(&neigh->neighbor_main_addr) != NULL);
418     abuf_json_boolean(abuf, "skip", neigh->skip);
419     abuf_json_int(abuf, "neighbor2nocov", neigh->neighbor_2_nocov);
420     abuf_json_int(abuf, "linkcount", neigh->linkcount);
421
422     if (list_2hop) {
423       abuf_json_mark_object(true, true, abuf, "twoHopNeighbors");
424     }
425
426     thop_cnt = 0;
427     for (list_2 = neigh->neighbor_2_list.next; list_2 != &neigh->neighbor_2_list; list_2 = list_2->next) {
428       if (list_2hop && list_2->neighbor_2) {
429         abuf_json_ip_address(abuf, NULL, &list_2->neighbor_2->neighbor_2_addr);
430       }
431       thop_cnt++;
432     }
433
434     if (list_2hop) {
435       abuf_json_mark_object(false, true, abuf, NULL);
436     }
437     abuf_json_int(abuf, "twoHopNeighborCount", thop_cnt);
438
439     abuf_json_mark_array_entry(false, abuf);
440   } OLSR_FOR_ALL_NBR_ENTRIES_END(neigh);
441   abuf_json_mark_object(false, true, abuf, NULL);
442 }
443
444 void ipc_print_neighbors(struct autobuf *abuf) {
445   ipc_print_neighbors_internal(abuf, false);
446 }
447
448 void ipc_print_links(struct autobuf *abuf) {
449   struct link_entry *my_link;
450
451   abuf_json_mark_object(true, true, abuf, "links");
452
453   OLSR_FOR_ALL_LINK_ENTRIES(my_link) {
454     struct lqtextbuffer lqBuffer;
455     const char* lqString = get_link_entry_text(my_link, '\t', &lqBuffer);
456     char * nlqString = strrchr(lqString, '\t');
457
458     if (nlqString) {
459       *nlqString = '\0';
460       nlqString++;
461     }
462
463     abuf_json_mark_array_entry(true, abuf);
464
465     abuf_json_ip_address(abuf, "localIP", &my_link->local_iface_addr);
466     abuf_json_ip_address(abuf, "remoteIP", &my_link->neighbor_iface_addr);
467     abuf_json_string(abuf, "olsrInterface", (my_link->inter && my_link->inter->int_name) ? my_link->inter->int_name : "");
468     abuf_json_string(abuf, "ifName", my_link->if_name ? my_link->if_name : "");
469     abuf_json_int(abuf, "validityTime", my_link->link_timer ? (long) (my_link->link_timer->timer_clock - now_times) : 0);
470     abuf_json_int(abuf, "symmetryTime", my_link->link_sym_timer ? (long) (my_link->link_sym_timer->timer_clock - now_times) : 0);
471     abuf_json_int(abuf, "asymmetryTime", my_link->ASYM_time);
472     abuf_json_int(abuf, "vtime", (long) my_link->vtime);
473     // neighbor (no need to print, can be looked up via neighbours)
474     abuf_json_string(abuf, "currentLinkStatus", linkTypeToString(lookup_link_status(my_link)));
475     abuf_json_string(abuf, "previousLinkStatus", linkTypeToString(my_link->prev_status));
476
477     abuf_json_float(abuf, "hysteresis", my_link->L_link_quality);
478     abuf_json_boolean(abuf, "pending", my_link->L_link_pending != 0);
479     abuf_json_int(abuf, "lostLinkTime", (long) my_link->L_LOST_LINK_time);
480     abuf_json_int(abuf, "helloTime", my_link->link_hello_timer ? (long) (my_link->link_hello_timer->timer_clock - now_times) : 0);
481     abuf_json_int(abuf, "lastHelloTime", (long) my_link->last_htime);
482     abuf_json_boolean(abuf, "seqnoValid", my_link->olsr_seqno_valid);
483     abuf_json_int(abuf, "seqno", my_link->olsr_seqno);
484
485     abuf_json_int(abuf, "lossHelloInterval", (long) my_link->loss_helloint);
486     abuf_json_int(abuf, "lossTime", my_link->link_loss_timer ? (long) (my_link->link_loss_timer->timer_clock - now_times) : 0);
487
488     abuf_json_int(abuf, "lossMultiplier", (long) my_link->loss_link_multiplier);
489
490     abuf_json_int(abuf, "linkCost", MIN(my_link->linkcost, LINK_COST_BROKEN));
491
492     abuf_json_float(abuf, "linkQuality", atof(lqString));
493     abuf_json_float(abuf, "neighborLinkQuality", nlqString ? atof(nlqString) : 0.0);
494
495     abuf_json_mark_array_entry(false, abuf);
496   } OLSR_FOR_ALL_LINK_ENTRIES_END(my_link);
497   abuf_json_mark_object(false, true, abuf, NULL);
498 }
499
500 void ipc_print_routes(struct autobuf *abuf) {
501   struct rt_entry *rt;
502
503   abuf_json_mark_object(true, true, abuf, "routes");
504
505   /* Walk the route table */
506   OLSR_FOR_ALL_RT_ENTRIES(rt) {
507     struct lqtextbuffer costbuffer;
508
509     if (rt->rt_best) {
510       abuf_json_mark_array_entry(true, abuf);
511       abuf_json_ip_address(abuf, "destination", &rt->rt_dst.prefix);
512       abuf_json_int(abuf, "genmask", rt->rt_dst.prefix_len);
513       abuf_json_ip_address(abuf, "gateway", &rt->rt_best->rtp_nexthop.gateway);
514       abuf_json_int(abuf, "metric", rt->rt_best->rtp_metric.hops);
515       abuf_json_float(abuf, "etx", atof(get_linkcost_text(rt->rt_best->rtp_metric.cost, true, &costbuffer)));
516       abuf_json_int(abuf, "rtpMetricCost", MIN(ROUTE_COST_BROKEN, rt->rt_best->rtp_metric.cost));
517       abuf_json_string(abuf, "networkInterface", if_ifwithindex_name(rt->rt_best->rtp_nexthop.iif_index));
518       abuf_json_mark_array_entry(false, abuf);
519     }
520   } OLSR_FOR_ALL_RT_ENTRIES_END(rt);
521
522   abuf_json_mark_object(false, true, abuf, NULL);
523 }
524
525 void ipc_print_topology(struct autobuf *abuf) {
526   struct tc_entry *tc;
527
528   abuf_json_mark_object(true, true, abuf, "topology");
529
530   /* Topology */
531   OLSR_FOR_ALL_TC_ENTRIES(tc) {
532     struct tc_edge_entry *tc_edge;
533     OLSR_FOR_ALL_TC_EDGE_ENTRIES(tc, tc_edge) {
534       if (tc_edge->edge_inv) {
535         struct lqtextbuffer lqbuffer;
536         const char* lqString = get_tc_edge_entry_text(tc_edge, '\t', &lqbuffer);
537         char * nlqString = strrchr(lqString, '\t');
538
539         if (nlqString) {
540           *nlqString = '\0';
541           nlqString++;
542         }
543
544         abuf_json_mark_array_entry(true, abuf);
545
546         // vertex_node
547         abuf_json_ip_address(abuf, "lastHopIP", &tc->addr);
548         // cand_tree_node
549         abuf_json_int(abuf, "pathCost", MIN(tc->path_cost, ROUTE_COST_BROKEN));
550         // path_list_node
551         // edge_tree
552         // prefix_tree
553         // next_hop
554         // edge_gc_timer
555         abuf_json_int(abuf, "validityTime", tc->validity_timer ? (tc->validity_timer->timer_clock - now_times) : 0);
556         abuf_json_int(abuf, "refCount", tc->refcount);
557         abuf_json_int(abuf, "msgSeq", tc->msg_seq);
558         abuf_json_int(abuf, "msgHops", tc->msg_hops);
559         abuf_json_int(abuf, "hops", tc->hops);
560         abuf_json_int(abuf, "ansn", tc->ansn);
561         abuf_json_int(abuf, "tcIgnored", tc->ignored);
562
563         abuf_json_int(abuf, "errSeq", tc->err_seq);
564         abuf_json_boolean(abuf, "errSeqValid", tc->err_seq_valid);
565
566         // edge_node
567         abuf_json_ip_address(abuf, "destinationIP", &tc_edge->T_dest_addr);
568         // tc
569         abuf_json_int(abuf, "tcEdgeCost", MIN(LINK_COST_BROKEN, tc_edge->cost));
570         abuf_json_int(abuf, "ansnEdge", tc_edge->ansn);
571         abuf_json_float(abuf, "linkQuality", atof(lqString));
572         abuf_json_float(abuf, "neighborLinkQuality", nlqString ? atof(nlqString) : 0.0);
573
574         abuf_json_mark_array_entry(false, abuf);
575       }
576     } OLSR_FOR_ALL_TC_EDGE_ENTRIES_END(tc, tc_edge);
577   } OLSR_FOR_ALL_TC_ENTRIES_END(tc);
578
579   abuf_json_mark_object(false, true, abuf, NULL);
580 }
581
582 void ipc_print_hna(struct autobuf *abuf) {
583   struct ip_prefix_list *hna;
584   struct hna_entry *tmp_hna;
585
586   abuf_json_mark_object(true, true, abuf, "hna");
587
588   /* Announced HNA entries */
589   for (hna = olsr_cnf->hna_entries; hna != NULL ; hna = hna->next) {
590     print_hna_array_entry( //
591         abuf, //
592         &olsr_cnf->main_addr, //
593         &hna->net.prefix, //
594         hna->net.prefix_len, //
595         0);
596   }
597
598   OLSR_FOR_ALL_HNA_ENTRIES(tmp_hna) {
599     struct hna_net *tmp_net;
600
601     /* Check all networks */
602     for (tmp_net = tmp_hna->networks.next; tmp_net != &tmp_hna->networks; tmp_net = tmp_net->next) {
603       print_hna_array_entry( //
604           abuf, //
605           &tmp_hna->A_gateway_addr, //
606           &tmp_net->hna_prefix.prefix, //
607           tmp_net->hna_prefix.prefix_len, //
608           tmp_net->hna_net_timer ? (tmp_net->hna_net_timer->timer_clock - now_times) : 0);
609     }
610   } OLSR_FOR_ALL_HNA_ENTRIES_END(tmp_hna);
611
612   abuf_json_mark_object(false, true, abuf, NULL);
613 }
614
615 void ipc_print_mid(struct autobuf *abuf) {
616   int idx;
617
618   abuf_json_mark_object(true, true, abuf, "mid");
619
620   /* MID */
621   for (idx = 0; idx < HASHSIZE; idx++) {
622     struct mid_entry * entry = mid_set[idx].next;
623
624     while (entry != &mid_set[idx]) {
625       abuf_json_mark_array_entry(true, abuf);
626
627       abuf_json_mark_object(true, false, abuf, "main");
628       abuf_json_ip_address(abuf, "ipAddress", &entry->main_addr);
629       abuf_json_int(abuf, "validityTime", entry->mid_timer ? (entry->mid_timer->timer_clock - now_times) : 0);
630       abuf_json_mark_object(false, false, abuf, NULL); // main
631
632       {
633         struct mid_address * alias = entry->aliases;
634
635         abuf_json_mark_object(true, true, abuf, "aliases");
636         while (alias) {
637           abuf_json_mark_array_entry(true, abuf);
638           abuf_json_ip_address(abuf, "ipAddress", &alias->alias);
639           abuf_json_int(abuf, "validityTime", alias->vtime - now_times);
640           abuf_json_mark_array_entry(false, abuf);
641
642           alias = alias->next_alias;
643         }
644         abuf_json_mark_object(false, true, abuf, NULL); // aliases
645       }
646       abuf_json_mark_array_entry(false, abuf); // entry
647
648       entry = entry->next;
649     }
650   }
651   abuf_json_mark_object(false, true, abuf, NULL); // mid
652 }
653
654 #ifdef __linux__
655
656 static void ipc_print_gateways_ipvx(struct autobuf *abuf, bool ipv6) {
657   assert(abuf);
658
659   abuf_json_mark_object(true, true, abuf, ipv6 ? "ipv6" : "ipv4");
660
661   if (olsr_cnf->smart_gw_active) {
662     struct gateway_entry * current_gw = olsr_get_inet_gateway(ipv6);
663     struct gateway_entry * gw;
664     OLSR_FOR_ALL_GATEWAY_ENTRIES(gw) {
665       if ((!ipv6 && !gw->ipv4) || (ipv6 && !gw->ipv6)) {
666         /* gw does not advertise the requested IP version */
667         continue;
668       }
669
670       abuf_json_mark_array_entry(true, abuf);
671       ipc_print_gateway_entry(abuf, ipv6, current_gw, gw);
672       abuf_json_mark_array_entry(false, abuf);
673     } OLSR_FOR_ALL_GATEWAY_ENTRIES_END(gw)
674   }
675
676   abuf_json_mark_object(false, true, abuf, NULL);
677 }
678 #endif /* __linux__ */
679
680 void ipc_print_gateways(struct autobuf *abuf) {
681 #ifndef __linux__
682   abuf_json_string(abuf, "error", "Gateway mode is only supported in Linux");
683 #else /* __linux__ */
684   abuf_json_mark_object(true, false, abuf, "gateways");
685
686   ipc_print_gateways_ipvx(abuf, false);
687   ipc_print_gateways_ipvx(abuf, true);
688
689   abuf_json_mark_object(false, false, abuf, NULL);
690 #endif /* __linux__ */
691 }
692
693 #ifdef __linux__
694
695 /** interface names for smart gateway tunnel interfaces, IPv4 */
696 extern struct interfaceName * sgwTunnel4InterfaceNames;
697
698 /** interface names for smart gateway tunnel interfaces, IPv6 */
699 extern struct interfaceName * sgwTunnel6InterfaceNames;
700
701 /**
702  * Construct the sgw table for a given ip version
703  *
704  * @param abuf the string buffer
705  * @param ipv6 true for IPv6, false for IPv4
706  * @param fmtv the format for printing
707  */
708 static void sgw_ipvx(struct autobuf *abuf, bool ipv6) {
709   struct interfaceName * sgwTunnelInterfaceNames;
710
711   assert(abuf);
712
713   sgwTunnelInterfaceNames = !ipv6 ? sgwTunnel4InterfaceNames : sgwTunnel6InterfaceNames;
714
715   abuf_json_mark_object(true, true, abuf, ipv6 ? "ipv6" : "ipv4");
716
717   if (olsr_cnf->smart_gw_active && sgwTunnelInterfaceNames) {
718     struct gateway_entry * current_gw = olsr_get_inet_gateway(ipv6);
719     int i;
720     for (i = 0; i < olsr_cnf->smart_gw_use_count; i++) {
721       struct interfaceName * node = &sgwTunnelInterfaceNames[i];
722       struct gateway_entry * gw = node->gw;
723
724       if (!gw) {
725         continue;
726       }
727
728       abuf_json_mark_array_entry(true, abuf);
729       ipc_print_gateway_entry(abuf, ipv6, current_gw, gw);
730       abuf_json_ip_address(abuf, "destination", &gw->originator);
731       abuf_json_string(abuf, "tunnel", node->name);
732       abuf_json_int(abuf, "tableNr", node->tableNr);
733       abuf_json_int(abuf, "ruleNr", node->ruleNr);
734       abuf_json_int(abuf, "bypassRuleNr", node->bypassRuleNr);
735       abuf_json_mark_array_entry(false, abuf);
736     }
737   }
738
739   abuf_json_mark_object(false, true, abuf, NULL);
740 }
741 #endif /* __linux__ */
742
743 void ipc_print_sgw(struct autobuf *abuf) {
744 #ifndef __linux__
745   abuf_json_string(abuf, "error", "Gateway mode is only supported in Linux");
746 #else
747   abuf_json_mark_object(true, false, abuf, "sgw");
748
749   sgw_ipvx(abuf, false);
750   sgw_ipvx(abuf, true);
751
752   abuf_json_mark_object(false, false, abuf, NULL);
753 #endif /* __linux__ */
754 }
755
756 void ipc_print_version(struct autobuf *abuf) {
757   abuf_json_mark_object(true, false, abuf, "version");
758
759   abuf_json_string(abuf, "version", olsrd_version);
760
761   abuf_json_string(abuf, "date", build_date);
762   abuf_json_string(abuf, "host", build_host);
763   abuf_json_string(abuf, "gitDescriptor", git_descriptor);
764   abuf_json_string(abuf, "gitSha", git_sha);
765   abuf_json_string(abuf, "releaseVersion", release_version);
766   abuf_json_string(abuf, "sourceHash", source_hash);
767
768   abuf_json_mark_object(false, false, abuf, NULL);
769 }
770
771 void ipc_print_olsrd_conf(struct autobuf *abuf) {
772   olsrd_write_cnf_autobuf(abuf, olsr_cnf);
773 }
774
775 void ipc_print_interfaces(struct autobuf *abuf) {
776   struct olsr_if *ifs;
777
778   abuf_json_mark_object(true, true, abuf, "interfaces");
779   for (ifs = olsr_cnf->interfaces; ifs != NULL ; ifs = ifs->next) {
780     abuf_json_mark_array_entry(true, abuf);
781     abuf_json_string(abuf, "name", ifs->name);
782     abuf_json_boolean(abuf, "configured", ifs->configured);
783     abuf_json_boolean(abuf, "hostEmulation", ifs->host_emul);
784     abuf_json_ip_address(abuf, "hostEmulationAddress", &ifs->hemu_ip);
785     print_interface_olsr(abuf, "olsrInterface", ifs->interf);
786     print_interface_config(abuf, "InterfaceConfiguration", ifs->cnf);
787     print_interface_config(abuf, "InterfaceConfigurationDefaults", ifs->cnfi);
788     abuf_json_mark_array_entry(false, abuf);
789   }
790   abuf_json_mark_object(false, true, abuf, NULL); // interfaces
791 }
792
793 void ipc_print_twohop(struct autobuf *abuf) {
794   ipc_print_neighbors_internal(abuf, true);
795 }
796
797 void ipc_print_config(struct autobuf *abuf) {
798   abuf_json_mark_object(true, false, abuf, "config");
799
800   abuf_json_string(abuf, "configurationFile", olsr_cnf->configuration_file);
801   abuf_json_int(abuf, "olsrPort", olsr_cnf->olsrport);
802   abuf_json_int(abuf, "debugLevel", olsr_cnf->debug_level);
803   abuf_json_boolean(abuf, "noFork", olsr_cnf->no_fork);
804   abuf_json_string(abuf, "pidFile", olsr_cnf->pidfile);
805   abuf_json_boolean(abuf, "hostEmulation", olsr_cnf->host_emul);
806   abuf_json_int(abuf, "ipVersion", olsr_cnf->ip_version);
807   abuf_json_boolean(abuf, "allowNoInt", olsr_cnf->allow_no_interfaces);
808   abuf_json_int(abuf, "tosValue", olsr_cnf->tos);
809   abuf_json_int(abuf, "rtProto", olsr_cnf->rt_proto);
810   abuf_json_int(abuf, "rtTable", olsr_cnf->rt_table);
811   abuf_json_int(abuf, "rtTableDefault", olsr_cnf->rt_table_default);
812   abuf_json_int(abuf, "rtTableTunnel", olsr_cnf->rt_table_tunnel);
813   abuf_json_int(abuf, "rtTablePriority", olsr_cnf->rt_table_pri);
814   abuf_json_int(abuf, "rtTableTunnelPriority", olsr_cnf->rt_table_tunnel_pri);
815   abuf_json_int(abuf, "rtTableDefauiltOlsrPriority", olsr_cnf->rt_table_defaultolsr_pri);
816   abuf_json_int(abuf, "rtTableDefaultPriority", olsr_cnf->rt_table_default_pri);
817   abuf_json_int(abuf, "willingness", olsr_cnf->willingness);
818   abuf_json_boolean(abuf, "willingnessAuto", olsr_cnf->willingness_auto);
819   // ipc_connections: later
820   abuf_json_boolean(abuf, "useHysteresis", olsr_cnf->use_hysteresis);
821   abuf_json_string(abuf, "fibMetric", ((olsr_cnf->fib_metric < FIBM_FLAT) || (olsr_cnf->fib_metric >= FIBM_CNT)) ? "" : FIB_METRIC_TXT[olsr_cnf->fib_metric]);
822   abuf_json_string(abuf, "fibMetricDefault", FIB_METRIC_TXT[olsr_cnf->fib_metric_default]);
823   abuf_json_float(abuf, "hystScaling", olsr_cnf->hysteresis_param.scaling);
824   abuf_json_float(abuf, "hystThrLow", olsr_cnf->hysteresis_param.thr_low);
825   abuf_json_float(abuf, "hystThrHigh", olsr_cnf->hysteresis_param.thr_high);
826   // plugins: later
827   // hna_entries
828   {
829     struct ip_prefix_list *hna;
830
831     abuf_json_mark_object(true, true, abuf, "hna");
832
833     /* Announced HNA entries */
834     for (hna = olsr_cnf->hna_entries; hna; hna = hna->next) {
835         print_hna_array_entry( //
836             abuf, //
837             &olsr_cnf->main_addr, //
838             &hna->net.prefix, //
839             hna->net.prefix_len, //
840             0);
841     }
842     abuf_json_mark_object(false, true, abuf, NULL);
843   }
844   // ipc_nets: later
845   // interface_defaults: later
846   // interfaces: later
847   abuf_json_float(abuf, "pollrate", olsr_cnf->pollrate);
848   abuf_json_float(abuf, "nicChgsPollInt", olsr_cnf->nic_chgs_pollrate);
849   abuf_json_boolean(abuf, "clearScreen", olsr_cnf->clear_screen);
850   abuf_json_int(abuf, "tcRedundancy", olsr_cnf->tc_redundancy);
851   abuf_json_int(abuf, "mprCoverage", olsr_cnf->mpr_coverage);
852   abuf_json_int(abuf, "linkQualityLevel", olsr_cnf->lq_level);
853   abuf_json_boolean(abuf, "linkQualityFishEye", olsr_cnf->lq_fish);
854   abuf_json_float(abuf, "linkQualityAging", olsr_cnf->lq_aging);
855   abuf_json_string(abuf, "linkQualityAlgorithm", olsr_cnf->lq_algorithm);
856
857   abuf_json_float(abuf, "minTCVTime", olsr_cnf->min_tc_vtime);
858
859   abuf_json_boolean(abuf, "setIpForward", olsr_cnf->set_ip_forward);
860
861   abuf_json_string(abuf, "lockFile", olsr_cnf->lock_file);
862   abuf_json_boolean(abuf, "useNiit", olsr_cnf->use_niit);
863
864   abuf_json_boolean(abuf, "smartGateway", olsr_cnf->smart_gw_active);
865   abuf_json_boolean(abuf, "smartGatewayAlwaysRemoveServerTunnel", olsr_cnf->smart_gw_always_remove_server_tunnel);
866   abuf_json_boolean(abuf, "smartGatewayAllowNAT", olsr_cnf->smart_gw_allow_nat);
867   abuf_json_boolean(abuf, "smartGatewayUplinkNAT", olsr_cnf->smart_gw_uplink_nat);
868   abuf_json_int(abuf, "smartGatewayUseCount", olsr_cnf->smart_gw_use_count);
869   abuf_json_int(abuf, "smartGatewayTakeDownPercentage", olsr_cnf->smart_gw_takedown_percentage);
870   abuf_json_string(abuf, "smartGatewayInstanceId", olsr_cnf->smart_gw_instance_id);
871   abuf_json_string(abuf, "smartGatewayPolicyRoutingScript", olsr_cnf->smart_gw_policyrouting_script);
872   // smart_gw_egress_interfaces
873   {
874     struct sgw_egress_if * egressif = olsr_cnf->smart_gw_egress_interfaces;
875
876     abuf_json_mark_object(true, true, abuf, "smartGatewayEgressInterfaces");
877     while (egressif) {
878       abuf_json_mark_array_entry(true, abuf);
879       abuf_json_string(abuf, "interface", egressif->name);
880       abuf_json_mark_array_entry(false, abuf);
881
882       egressif = egressif->next;
883     }
884     abuf_json_mark_object(false, true, abuf, NULL);
885   }
886   abuf_json_int(abuf, "smartGatewayEgressInterfacesCount", olsr_cnf->smart_gw_egress_interfaces_count);
887   abuf_json_string(abuf, "smartGatewayEgressFile", olsr_cnf->smart_gw_egress_file);
888   abuf_json_int(abuf, "smartGatewayEgressFilePeriod", olsr_cnf->smart_gw_egress_file_period);
889   abuf_json_string(abuf, "smartGatewayStatusFile", olsr_cnf->smart_gw_status_file);
890   abuf_json_int(abuf, "smartGatewayTablesOffset", olsr_cnf->smart_gw_offset_tables);
891   abuf_json_int(abuf, "smartGatewayRulesOffset", olsr_cnf->smart_gw_offset_rules);
892   abuf_json_int(abuf, "smartGatewayPeriod", olsr_cnf->smart_gw_period);
893   abuf_json_int(abuf, "smartGatewayStableCount", olsr_cnf->smart_gw_stablecount);
894   abuf_json_int(abuf, "smartGatewayThreshold", olsr_cnf->smart_gw_thresh);
895   abuf_json_int(abuf, "smartGatewayWeightExitLinkUp", olsr_cnf->smart_gw_weight_exitlink_up);
896   abuf_json_int(abuf, "smartGatewayWeightExitLinkDown", olsr_cnf->smart_gw_weight_exitlink_down);
897   abuf_json_int(abuf, "smartGatewayWeightEtx", olsr_cnf->smart_gw_weight_etx);
898   abuf_json_int(abuf, "smartGatewayDividerEtx", olsr_cnf->smart_gw_divider_etx);
899   abuf_json_int(abuf, "smartGatewayMaxCostMaxEtx", olsr_cnf->smart_gw_path_max_cost_etx_max);
900   abuf_json_string(abuf, "smartGatewayUplink", ((olsr_cnf->smart_gw_type < GW_UPLINK_NONE) || (olsr_cnf->smart_gw_type >= GW_UPLINK_CNT)) ? "" : GW_UPLINK_TXT[olsr_cnf->smart_gw_type]);
901   abuf_json_int(abuf, "smartGatewayUplinkKbps", olsr_cnf->smart_gw_uplink);
902   abuf_json_int(abuf, "smartGatewayDownlinkKbps", olsr_cnf->smart_gw_downlink);
903   abuf_json_boolean(abuf, "smartGatewayBandwidthZero", olsr_cnf->smart_gateway_bandwidth_zero);
904   abuf_json_ip_address(abuf, "smartGatewayPrefix", &olsr_cnf->smart_gw_prefix.prefix);
905   abuf_json_int(abuf, "smartGatewayPrefixLength", olsr_cnf->smart_gw_prefix.prefix_len);
906
907
908   abuf_json_ip_address(abuf, "mainIp", &olsr_cnf->main_addr);
909   abuf_json_ip_address(abuf, "unicastSourceIpAddress", &olsr_cnf->unicast_src_ip);
910   abuf_json_boolean(abuf, "srcIpRoutes", olsr_cnf->use_src_ip_routes);
911
912
913   abuf_json_int(abuf, "maxPrefixLength", olsr_cnf->maxplen);
914   abuf_json_int(abuf, "ipSize", olsr_cnf->ipsize);
915   abuf_json_boolean(abuf, "delgw", olsr_cnf->del_gws);
916   abuf_json_float(abuf, "willingnessUpdateInterval", olsr_cnf->will_int);
917   abuf_json_float(abuf, "maxSendMessageJitter", olsr_cnf->max_jitter);
918   abuf_json_int(abuf, "exitValue", olsr_cnf->exit_value);
919   abuf_json_float(abuf, "maxTcValidTime", olsr_cnf->max_tc_vtime);
920
921   abuf_json_int(abuf, "niit4to6InterfaceIndex", olsr_cnf->niit4to6_if_index);
922   abuf_json_int(abuf, "niit6to4InterfaceIndex", olsr_cnf->niit6to4_if_index);
923
924
925   abuf_json_boolean(abuf, "hasIpv4Gateway", olsr_cnf->has_ipv4_gateway);
926   abuf_json_boolean(abuf, "hasIpv6Gateway", olsr_cnf->has_ipv6_gateway);
927
928   abuf_json_int(abuf, "ioctlSocket", olsr_cnf->ioctl_s);
929 #ifdef __linux__
930   abuf_json_int(abuf, "routeNetlinkSocket", olsr_cnf->rtnl_s);
931   abuf_json_int(abuf, "routeMonitorSocket", olsr_cnf->rt_monitor_socket);
932 #endif /* __linux__ */
933
934 #if defined __FreeBSD__ || defined __FreeBSD_kernel__ || defined __APPLE__ || defined __NetBSD__ || defined __OpenBSD__
935   abuf_json_int(abuf, "routeChangeSocket", olsr_cnf->rts);
936 #endif /* defined __FreeBSD__ || defined __FreeBSD_kernel__ || defined __APPLE__ || defined __NetBSD__ || defined __OpenBSD__ */
937   abuf_json_float(abuf, "linkQualityNatThreshold", olsr_cnf->lq_nat_thresh);
938
939
940   // Other settings
941   abuf_json_int(abuf, "brokenLinkCost", LINK_COST_BROKEN);
942   abuf_json_int(abuf, "brokenRouteCost", ROUTE_COST_BROKEN);
943
944
945   // IpcConnect section
946   abuf_json_int(abuf, "ipcConnectMaxConnections", olsr_cnf->ipc_connections);
947   {
948     struct ip_prefix_list *ipc_nets;
949
950     abuf_json_mark_object(true, true, abuf, "ipcConnectAllowed");
951     for (ipc_nets = olsr_cnf->ipc_nets; ipc_nets; ipc_nets = ipc_nets->next) {
952       print_ipc_net_array_entry(abuf, ipc_nets);
953     }
954     abuf_json_mark_object(false, true, abuf, NULL);
955   }
956
957
958   // plugins section: use /plugins
959
960
961   // InterfaceDefaults section
962   print_interface_config(abuf, "interfaceDefaults", olsr_cnf->interface_defaults);
963
964
965   // Interface(s) section: use /interfaces
966
967
968   // OS section
969 #if defined _WIN32 || defined _WIN64
970   abuf_json_string(abuf, "os", "Windows");
971 #elif defined __gnu_linux__
972   abuf_json_string(abuf, "os", "GNU/Linux");
973 #elif defined __ANDROID__
974   abuf_json_string(abuf, "os", "Android");
975 #elif defined __APPLE__
976   abuf_json_string(abuf, "os", "Mac OS X");
977 #elif defined __NetBSD__
978   abuf_json_string(abuf, "os", "NetBSD");
979 #elif defined __OpenBSD__
980   abuf_json_string(abuf, "os", "OpenBSD");
981 #elif defined __FreeBSD__ || defined __FreeBSD_kernel__
982   abuf_json_string(abuf, "os", "FreeBSD");
983 #else /* OS detection */
984   abuf_json_string(abuf, "os", "Undefined");
985 #endif /* OS detection */
986
987   abuf_json_int(abuf, "startTime", start_time.tv_sec);
988
989   abuf_json_mark_object(false, false, abuf, NULL);
990 }
991
992 void ipc_print_plugins(struct autobuf *abuf) {
993   abuf_json_mark_object(true, true, abuf, "plugins");
994   if (olsr_cnf->plugins) {
995     struct plugin_entry *plugin;
996
997     for (plugin = olsr_cnf->plugins; plugin; plugin = plugin->next) {
998       struct plugin_param *param;
999
1000       abuf_json_mark_array_entry(true, abuf);
1001       abuf_json_string(abuf, "plugin", plugin->name);
1002
1003       abuf_json_mark_object(true, false, abuf, "parameters");
1004       for (param = plugin->params; param; param = param->next) {
1005         abuf_json_string(abuf, param->key, param->value);
1006       }
1007       abuf_json_mark_object(false, false, abuf, NULL);
1008
1009       abuf_json_mark_array_entry(false, abuf);
1010     }
1011   }
1012   abuf_json_mark_object(false, true, abuf, NULL);
1013 }