xref: /openbsd-src/usr.sbin/rpki-client/output-json.c (revision fc405d53b73a2d73393cb97f684863d17b583e38)
1 /*	$OpenBSD: output-json.c,v 1.37 2023/05/03 09:54:25 claudio Exp $ */
2 /*
3  * Copyright (c) 2019 Claudio Jeker <claudio@openbsd.org>
4  *
5  * Permission to use, copy, modify, and distribute this software for any
6  * purpose with or without fee is hereby granted, provided that the above
7  * copyright notice and this permission notice appear in all copies.
8  *
9  * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
10  * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
11  * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
12  * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
13  * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
14  * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
15  * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
16  */
17 
18 #include <stdlib.h>
19 #include <unistd.h>
20 #include <time.h>
21 #include <netdb.h>
22 
23 #include "extern.h"
24 #include "json.h"
25 
26 static void
27 outputheader_json(struct stats *st)
28 {
29 	char		 hn[NI_MAXHOST], tbuf[26];
30 	struct tm	*tp;
31 	time_t		 t;
32 	int		 i;
33 
34 	time(&t);
35 	tp = gmtime(&t);
36 	strftime(tbuf, sizeof tbuf, "%FT%TZ", tp);
37 
38 	gethostname(hn, sizeof hn);
39 
40 	json_do_object("metadata");
41 
42 	json_do_string("buildmachine", hn);
43 	json_do_string("buildtime", tbuf);
44 	json_do_int("elapsedtime", st->elapsed_time.tv_sec);
45 	json_do_int("usertime", st->user_time.tv_sec);
46 	json_do_int("systemtime", st->system_time.tv_sec);
47 	json_do_int("roas", st->repo_tal_stats.roas);
48 	json_do_int("failedroas", st->repo_tal_stats.roas_fail);
49 	json_do_int("invalidroas", st->repo_tal_stats.roas_invalid);
50 	json_do_int("aspas", st->repo_tal_stats.aspas);
51 	json_do_int("failedaspas", st->repo_tal_stats.aspas_fail);
52 	json_do_int("invalidaspas", st->repo_tal_stats.aspas_invalid);
53 	json_do_int("bgpsec_pubkeys", st->repo_tal_stats.brks);
54 	json_do_int("certificates", st->repo_tal_stats.certs);
55 	json_do_int("invalidcertificates", st->repo_tal_stats.certs_fail);
56 	json_do_int("taks", st->repo_tal_stats.taks);
57 	json_do_int("tals", st->tals);
58 	json_do_int("invalidtals", talsz - st->tals);
59 
60 	json_do_array("talfiles");
61 	for (i = 0; i < talsz; i++)
62 		json_do_string("name", tals[i]);
63 	json_do_end();
64 
65 	json_do_int("manifests", st->repo_tal_stats.mfts);
66 	json_do_int("failedmanifests", st->repo_tal_stats.mfts_fail);
67 	json_do_int("stalemanifests", st->repo_tal_stats.mfts_stale);
68 	json_do_int("crls", st->repo_tal_stats.crls);
69 	json_do_int("gbrs", st->repo_tal_stats.gbrs);
70 	json_do_int("repositories", st->repos);
71 	json_do_int("vrps", st->repo_tal_stats.vrps);
72 	json_do_int("uniquevrps", st->repo_tal_stats.vrps_uniqs);
73 	json_do_int("vaps", st->repo_tal_stats.vaps);
74 	json_do_int("uniquevaps", st->repo_tal_stats.vaps_uniqs);
75 	json_do_int("cachedir_del_files", st->repo_stats.del_files);
76 	json_do_int("cachedir_superfluous_files", st->repo_stats.extra_files);
77 	json_do_int("cachedir_del_dirs", st->repo_stats.del_dirs);
78 
79 	json_do_end();
80 }
81 
82 static void
83 print_vap(struct vap *v, enum afi afi)
84 {
85 	size_t i;
86 	int found = 0;
87 
88 	json_do_object("aspa");
89 	json_do_int("customer_asid", v->custasid);
90 	json_do_int("expires", v->expires);
91 
92 	json_do_array("providers");
93 	for (i = 0; i < v->providersz; i++) {
94 		if (v->providers[i].afi != 0 && v->providers[i].afi != afi)
95 			continue;
96 		found = 1;
97 		json_do_int("provider", v->providers[i].as);
98 	}
99 	if (!found)
100 		json_do_int("provider", 0);
101 	json_do_end();
102 }
103 
104 static void
105 output_aspa(struct vap_tree *vaps)
106 {
107 	struct vap	*v;
108 
109 	json_do_object("provider_authorizations");
110 
111 	json_do_array("ipv4");
112 	RB_FOREACH(v, vap_tree, vaps) {
113 		print_vap(v, AFI_IPV4);
114 	}
115 	json_do_end();
116 
117 	json_do_array("ipv6");
118 	RB_FOREACH(v, vap_tree, vaps) {
119 		print_vap(v, AFI_IPV6);
120 	}
121 	json_do_end();
122 }
123 
124 int
125 output_json(FILE *out, struct vrp_tree *vrps, struct brk_tree *brks,
126     struct vap_tree *vaps, struct stats *st)
127 {
128 	char		 buf[64];
129 	struct vrp	*v;
130 	struct brk	*b;
131 
132 	json_do_start(out);
133 	outputheader_json(st);
134 
135 	json_do_array("roas");
136 	RB_FOREACH(v, vrp_tree, vrps) {
137 		ip_addr_print(&v->addr, v->afi, buf, sizeof(buf));
138 
139 		json_do_object("roa");
140 		json_do_int("asn", v->asid);
141 		json_do_string("prefix", buf);
142 		json_do_int("maxLength", v->maxlength);
143 		json_do_string("ta", taldescs[v->talid]);
144 		json_do_int("expires", v->expires);
145 		json_do_end();
146 	}
147 	json_do_end();
148 
149 	json_do_array("bgpsec_keys");
150 	RB_FOREACH(b, brk_tree, brks) {
151 		json_do_object("brks");
152 		json_do_int("asn", b->asid);
153 		json_do_string("ski", b->ski);
154 		json_do_string("pubkey", b->pubkey);
155 		json_do_string("ta", taldescs[b->talid]);
156 		json_do_int("expires", b->expires);
157 		json_do_end();
158 	}
159 	json_do_end();
160 
161 	if (!excludeaspa)
162 		output_aspa(vaps);
163 
164 	return json_do_finish();
165 }
166