Annotation of embedaddon/mtr/ui/report.c, revision 1.1.1.2
1.1 misho 1: /*
2: mtr -- a network diagnostic tool
3: Copyright (C) 1997,1998 Matt Kimball
4:
5: This program is free software; you can redistribute it and/or modify
6: it under the terms of the GNU General Public License version 2 as
7: published by the Free Software Foundation.
8:
9: This program is distributed in the hope that it will be useful,
10: but WITHOUT ANY WARRANTY; without even the implied warranty of
11: MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12: GNU General Public License for more details.
13:
1.1.1.2 ! misho 14: You should have received a copy of the GNU General Public License along
! 15: with this program; if not, write to the Free Software Foundation, Inc.,
! 16: 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
1.1 misho 17: */
18:
19: #include "config.h"
20:
21: #include <sys/types.h>
22: #include <stdio.h>
23: #include <netdb.h>
24: #include <netinet/in.h>
25: #include <sys/socket.h>
26: #include <string.h>
27: #include <strings.h>
28: #include <time.h>
1.1.1.2 ! misho 29: #ifdef HAVE_JANSSON
! 30: #include <jansson.h>
! 31: #endif
! 32: #ifdef HAVE_ERROR_H
! 33: #include <error.h>
! 34: #else
! 35: #include "portability/error.h"
! 36: #endif
1.1 misho 37:
38: #include "mtr.h"
39: #include "report.h"
40: #include "net.h"
41: #include "dns.h"
42: #include "asn.h"
43: #include "utils.h"
44:
45: #define MAXLOADBAL 5
46: #define MAX_FORMAT_STR 81
47:
48:
49: void report_open(
50: void)
51: {
52: const time_t now = time(NULL);
53: const char *t = iso_time(&now);
54:
55: printf("Start: %s\n", t);
56: }
57:
58: static size_t snprint_addr(
59: struct mtr_ctl *ctl,
60: char *dst,
61: size_t dst_len,
62: ip_t * addr)
63: {
64: if (addrcmp((void *) addr, (void *) &ctl->unspec_addr, ctl->af)) {
65: struct hostent *host =
66: ctl->dns ? addr2host((void *) addr, ctl->af) : NULL;
67: if (!host)
68: return snprintf(dst, dst_len, "%s", strlongip(ctl, addr));
69: else if (ctl->dns && ctl->show_ips)
70: return snprintf(dst, dst_len, "%s (%s)", host->h_name,
71: strlongip(ctl, addr));
72: else
73: return snprintf(dst, dst_len, "%s", host->h_name);
74: } else
75: return snprintf(dst, dst_len, "%s", "???");
76: }
77:
78:
79: #ifdef HAVE_IPINFO
80: static void print_mpls(
81: struct mplslen *mpls)
82: {
83: int k;
84: for (k = 0; k < mpls->labels; k++)
1.1.1.2 ! misho 85: printf(" [MPLS: Lbl %lu TC %u S %cu TTL %u]\n",
! 86: mpls->label[k], mpls->tc[k], mpls->s[k], mpls->ttl[k]);
1.1 misho 87: }
88: #endif
89:
90: void report_close(
91: struct mtr_ctl *ctl)
92: {
93: int i, j, at, max, z, w;
94: struct mplslen *mpls, *mplss;
95: ip_t *addr;
96: ip_t *addr2 = NULL;
97: char name[MAX_FORMAT_STR];
98: char buf[1024];
99: char fmt[16];
100: size_t len = 0;
101: size_t len_hosts = 33;
102: #ifdef HAVE_IPINFO
103: int len_tmp;
104: const size_t iiwidth_len = get_iiwidth_len();
105: #endif
106:
107: if (ctl->reportwide) {
108: /* get the longest hostname */
109: len_hosts = strlen(ctl->LocalHostname);
110: max = net_max(ctl);
111: at = net_min(ctl);
112: for (; at < max; at++) {
113: size_t nlen;
114: addr = net_addr(at);
115: if ((nlen = snprint_addr(ctl, name, sizeof(name), addr)))
116: if (len_hosts < nlen)
117: len_hosts = nlen;
118: }
119: }
120: #ifdef HAVE_IPINFO
121: len_tmp = len_hosts;
122: if (ctl->ipinfo_no >= 0 && iiwidth_len) {
123: ctl->ipinfo_no %= iiwidth_len;
124: if (ctl->reportwide) {
125: len_hosts++; /* space */
126: len_tmp += get_iiwidth(ctl->ipinfo_no);
127: if (!ctl->ipinfo_no)
128: len_tmp += 2; /* align header: AS */
129: }
130: }
131: snprintf(fmt, sizeof(fmt), "HOST: %%-%ds", len_tmp);
132: #else
133: snprintf(fmt, sizeof(fmt), "HOST: %%-%zus", len_hosts);
134: #endif
135: snprintf(buf, sizeof(buf), fmt, ctl->LocalHostname);
136: len = ctl->reportwide ? strlen(buf) : len_hosts;
137: for (i = 0; i < MAXFLD; i++) {
138: j = ctl->fld_index[ctl->fld_active[i]];
139: if (j < 0)
140: continue;
141:
142: snprintf(fmt, sizeof(fmt), "%%%ds", data_fields[j].length);
143: snprintf(buf + len, sizeof(buf), fmt, data_fields[j].title);
144: len += data_fields[j].length;
145: }
146: printf("%s\n", buf);
147:
148: max = net_max(ctl);
149: at = net_min(ctl);
150: for (; at < max; at++) {
151: addr = net_addr(at);
152: mpls = net_mpls(at);
153: snprint_addr(ctl, name, sizeof(name), addr);
154:
155: #ifdef HAVE_IPINFO
156: if (is_printii(ctl)) {
157: snprintf(fmt, sizeof(fmt), " %%2d. %%s%%-%zus", len_hosts);
158: snprintf(buf, sizeof(buf), fmt, at + 1, fmt_ipinfo(ctl, addr),
159: name);
160: } else {
161: #endif
162: snprintf(fmt, sizeof(fmt), " %%2d.|-- %%-%zus", len_hosts);
163: snprintf(buf, sizeof(buf), fmt, at + 1, name);
164: #ifdef HAVE_IPINFO
165: }
166: #endif
167: len = ctl->reportwide ? strlen(buf) : len_hosts;
168: for (i = 0; i < MAXFLD; i++) {
169: j = ctl->fld_index[ctl->fld_active[i]];
170: if (j < 0)
171: continue;
172:
173: /* 1000.0 is a temporay hack for stats usec to ms, impacted net_loss. */
174: if (strchr(data_fields[j].format, 'f')) {
175: snprintf(buf + len, sizeof(buf), data_fields[j].format,
176: data_fields[j].net_xxx(at) / 1000.0);
177: } else {
178: snprintf(buf + len, sizeof(buf), data_fields[j].format,
179: data_fields[j].net_xxx(at));
180: }
181: len += data_fields[j].length;
182: }
183: printf("%s\n", buf);
184:
185: /* This feature shows 'loadbalances' on routes */
186:
1.1.1.2 ! misho 187: /* Print list of all hosts that have responded from ttl = at + 1 away */
! 188: for (z = 0; z < MAX_PATH; z++) {
1.1 misho 189: int found = 0;
190: addr2 = net_addrs(at, z);
191: mplss = net_mplss(at, z);
192: if ((addrcmp
193: ((void *) &ctl->unspec_addr, (void *) addr2,
1.1.1.2 ! misho 194: ctl->af)) == 0) {
1.1 misho 195: break;
1.1.1.2 ! misho 196: } else if ((addrcmp
! 197: ((void *) addr, (void *) addr2,
! 198: ctl->af)) == 0) {
! 199: continue; /* Latest Host is already printed */
! 200: } else {
! 201: snprint_addr(ctl, name, sizeof(name), addr2);
! 202: snprintf(fmt, sizeof(fmt), " %%-%zus", len_hosts);
! 203: snprintf(buf, sizeof(buf), fmt, name);
! 204: printf("%s\n", buf);
! 205: }
1.1 misho 206: for (w = 0; w < z; w++)
207: /* Ok... checking if there are ips repeated on same hop */
208: if ((addrcmp
209: ((void *) addr2, (void *) net_addrs(at, w),
210: ctl->af)) == 0) {
211: found = 1;
212: break;
213: }
214:
215: if (!found) {
216:
217: #ifdef HAVE_IPINFO
218: if (is_printii(ctl)) {
219: if (mpls->labels && z == 1 && ctl->enablempls)
220: print_mpls(mpls);
221: snprint_addr(ctl, name, sizeof(name), addr2);
222: printf(" %s%s\n", fmt_ipinfo(ctl, addr2), name);
223: if (ctl->enablempls)
224: print_mpls(mplss);
225: }
226: #else
227: int k;
228: if (mpls->labels && z == 1 && ctl->enablempls) {
229: for (k = 0; k < mpls->labels; k++) {
230: printf
1.1.1.2 ! misho 231: (" | |+-- [MPLS: Lbl %lu TC %u S %u TTL %u]\n",
! 232: mpls->label[k], mpls->tc[k], mpls->s[k],
1.1 misho 233: mpls->ttl[k]);
234: }
235: }
236:
237: if (z == 1) {
238: printf(" | `|-- %s\n", strlongip(ctl, addr2));
239: for (k = 0; k < mplss->labels && ctl->enablempls; k++) {
240: printf
1.1.1.2 ! misho 241: (" | +-- [MPLS: Lbl %lu TC %u S %u TTL %u]\n",
! 242: mplss->label[k], mplss->tc[k], mplss->s[k],
1.1 misho 243: mplss->ttl[k]);
244: }
245: } else {
246: printf(" | |-- %s\n", strlongip(ctl, addr2));
247: for (k = 0; k < mplss->labels && ctl->enablempls; k++) {
248: printf
1.1.1.2 ! misho 249: (" | +-- [MPLS: Lbl %lu TC %u S %u TTL %u]\n",
! 250: mplss->label[k], mplss->tc[k], mplss->s[k],
1.1 misho 251: mplss->ttl[k]);
252: }
253: }
254: #endif
255: }
256: }
257:
258: /* No multipath */
259: #ifdef HAVE_IPINFO
260: if (is_printii(ctl)) {
261: if (mpls->labels && z == 1 && ctl->enablempls)
262: print_mpls(mpls);
263: }
264: #else
265: if (mpls->labels && z == 1 && ctl->enablempls) {
266: int k;
267: for (k = 0; k < mpls->labels; k++) {
1.1.1.2 ! misho 268: printf(" | +-- [MPLS: Lbl %lu TC %u S %u TTL %u]\n",
! 269: mpls->label[k], mpls->tc[k], mpls->s[k],
1.1 misho 270: mpls->ttl[k]);
271: }
272: }
273: #endif
274: }
275: }
276:
277:
278: void txt_open(
279: void)
280: {
281: }
282:
283:
284: void txt_close(
285: struct mtr_ctl *ctl)
286: {
287: report_close(ctl);
288: }
289:
1.1.1.2 ! misho 290: #ifdef HAVE_JANSSON
1.1 misho 291: void json_open(
292: void)
293: {
294: }
295:
1.1.1.2 ! misho 296: void json_close(struct mtr_ctl *ctl)
1.1 misho 297: {
1.1.1.2 ! misho 298: int i, j, at, max;
! 299: int ret;
! 300: char buf[128];
! 301: json_t *jreport, *jmtr, *jhubs, *jh;
1.1 misho 302: ip_t *addr;
303: char name[MAX_FORMAT_STR];
304:
1.1.1.2 ! misho 305: jmtr = json_pack("{ss ss si si}",
! 306: "src", ctl->LocalHostname,
! 307: "dst", ctl->Hostname,
! 308: "tos", ctl->tos,
! 309: "tests", ctl->MaxPing);
! 310: if (!jmtr)
! 311: goto on_error;
! 312:
1.1 misho 313: if (ctl->cpacketsize >= 0) {
1.1.1.2 ! misho 314: snprintf(buf, sizeof(buf), "%d", ctl->cpacketsize);
1.1 misho 315: } else {
1.1.1.2 ! misho 316: snprintf(buf, sizeof(buf), "rand(%d-%d)", MINPACKET, -ctl->cpacketsize);
1.1 misho 317: }
1.1.1.2 ! misho 318: ret = json_object_set_new(jmtr, "psize", json_string(buf));
! 319: if (ret == -1)
! 320: goto on_error;
! 321:
1.1 misho 322: if (ctl->bitpattern >= 0) {
1.1.1.2 ! misho 323: snprintf(buf, sizeof(buf), "0x%02X", (unsigned char)(ctl->bitpattern));
1.1 misho 324: } else {
1.1.1.2 ! misho 325: snprintf(buf, sizeof(buf), "rand(0x00-FF)");
1.1 misho 326: }
327:
1.1.1.2 ! misho 328: ret = json_object_set_new(jmtr, "bitpattern", json_string(buf));
! 329: if (ret == -1)
! 330: goto on_error;
! 331:
! 332: jhubs = json_array();
! 333: if (!jhubs)
! 334: goto on_error;
1.1 misho 335:
336: max = net_max(ctl);
1.1.1.2 ! misho 337: at = net_min(ctl);
1.1 misho 338: for (; at < max; at++) {
339: addr = net_addr(at);
340: snprint_addr(ctl, name, sizeof(name), addr);
341:
1.1.1.2 ! misho 342: jh = json_pack("{si ss}", "count", at + 1, "host", name);
! 343: if (!jh)
! 344: goto on_error;
! 345:
1.1 misho 346: #ifdef HAVE_IPINFO
347: if(!ctl->ipinfo_no) {
1.1.1.2 ! misho 348: char* fmtinfo = fmt_ipinfo(ctl, addr);
! 349: if (fmtinfo != NULL)
! 350: fmtinfo = trim(fmtinfo, '\0');
! 351:
! 352: ret = json_object_set_new(jh, "ASN", json_string(fmtinfo));
! 353: if (ret == -1)
! 354: goto on_error;
1.1 misho 355: }
356: #endif
357:
1.1.1.2 ! misho 358: for (i = 0; i < MAXFLD; i++) {
1.1 misho 359: j = ctl->fld_index[ctl->fld_active[i]];
360:
361: if (j <= 0)
1.1.1.2 ! misho 362: continue; /* Field nr 0, " " shouldn't be printed in this method. */
1.1 misho 363:
364: if (strchr(data_fields[j].format, 'f')) {
1.1.1.2 ! misho 365: ret = json_object_set_new(
! 366: jh, data_fields[j].title,
! 367: json_real(data_fields[j].net_xxx(at) / 1000.0));
1.1 misho 368: } else {
1.1.1.2 ! misho 369: ret = json_object_set_new(
! 370: jh, data_fields[j].title,
! 371: json_integer(data_fields[j].net_xxx(at)));
1.1 misho 372: }
1.1.1.2 ! misho 373: if (ret == -1)
! 374: goto on_error;
1.1 misho 375: }
1.1.1.2 ! misho 376:
! 377: ret = json_array_append_new(jhubs, jh);
! 378: if (ret == -1)
! 379: goto on_error;
1.1 misho 380: }
381:
1.1.1.2 ! misho 382: jreport = json_pack("{s{so so}}", "report", "mtr", jmtr, "hubs", jhubs);
! 383:
! 384: ret = json_dumpf(jreport, stdout, JSON_INDENT(4) | JSON_REAL_PRECISION(5));
! 385: if (ret == -1)
! 386: goto on_error;
! 387:
! 388: printf("\n"); // bash promt should be on new line
! 389: json_decref(jreport);
! 390: return;
! 391: on_error:
! 392: error(EXIT_FAILURE, 0, "json_close failed");
! 393: }
! 394: #endif
1.1 misho 395:
396:
397: void xml_open(
398: void)
399: {
400: }
401:
402:
403: void xml_close(
404: struct mtr_ctl *ctl)
405: {
406: int i, j, at, max;
407: ip_t *addr;
408: char name[MAX_FORMAT_STR];
409:
410: printf("<?xml version=\"1.0\"?>\n");
411: printf("<MTR SRC=\"%s\" DST=\"%s\"", ctl->LocalHostname,
412: ctl->Hostname);
413: printf(" TOS=\"0x%X\"", ctl->tos);
414: if (ctl->cpacketsize >= 0) {
415: printf(" PSIZE=\"%d\"", ctl->cpacketsize);
416: } else {
417: printf(" PSIZE=\"rand(%d-%d)\"", MINPACKET, -ctl->cpacketsize);
418: }
419: if (ctl->bitpattern >= 0) {
420: printf(" BITPATTERN=\"0x%02X\"",
421: (unsigned char) (ctl->bitpattern));
422: } else {
423: printf(" BITPATTERN=\"rand(0x00-FF)\"");
424: }
425: printf(" TESTS=\"%d\">\n", ctl->MaxPing);
426:
427: max = net_max(ctl);
428: at = net_min(ctl);
429: for (; at < max; at++) {
430: addr = net_addr(at);
431: snprint_addr(ctl, name, sizeof(name), addr);
432:
433: printf(" <HUB COUNT=\"%d\" HOST=\"%s\">\n", at + 1, name);
434: for (i = 0; i < MAXFLD; i++) {
435: const char *title;
436:
437: j = ctl->fld_index[ctl->fld_active[i]];
438: if (j <= 0)
439: continue; /* Field nr 0, " " shouldn't be printed in this method. */
440:
441: snprintf(name, sizeof(name), "%s%s%s", " <%s>",
442: data_fields[j].format, "</%s>\n");
443:
444: /* XML doesn't allow "%" in tag names, rename Loss% to just Loss */
445: title = data_fields[j].title;
446: if (strcmp(data_fields[j].title, "Loss%") == 0) {
447: title = "Loss";
448: }
449:
450: /* 1000.0 is a temporay hack for stats usec to ms, impacted net_loss. */
451: if (strchr(data_fields[j].format, 'f')) {
452: printf(name,
453: title, data_fields[j].net_xxx(at) / 1000.0, title);
454: } else {
455: printf(name, title, data_fields[j].net_xxx(at), title);
456: }
457: }
458: printf(" </HUB>\n");
459: }
460: printf("</MTR>\n");
461: }
462:
463:
464: void csv_open(
465: void)
466: {
467: }
468:
469: void csv_close(
470: struct mtr_ctl *ctl,
471: time_t now)
472: {
473: int i, j, at, max;
474: ip_t *addr;
475: char name[MAX_FORMAT_STR];
476:
477: for (i = 0; i < MAXFLD; i++) {
478: j = ctl->fld_index[ctl->fld_active[i]];
479: if (j < 0)
480: continue;
481: }
482:
483: max = net_max(ctl);
484: at = net_min(ctl);
485: for (; at < max; at++) {
486: addr = net_addr(at);
487: snprint_addr(ctl, name, sizeof(name), addr);
488:
489: if (at == net_min(ctl)) {
490: printf("Mtr_Version,Start_Time,Status,Host,Hop,Ip,");
491: #ifdef HAVE_IPINFO
492: if (!ctl->ipinfo_no) {
493: printf("Asn,");
494: }
495: #endif
496: for (i = 0; i < MAXFLD; i++) {
497: j = ctl->fld_index[ctl->fld_active[i]];
498: if (j < 0)
499: continue;
500: printf("%s,", data_fields[j].title);
501: }
502: printf("\n");
503: }
504: #ifdef HAVE_IPINFO
505: if (!ctl->ipinfo_no) {
506: char *fmtinfo = fmt_ipinfo(ctl, addr);
507: fmtinfo = trim(fmtinfo, '\0');
508: printf("MTR.%s,%lld,%s,%s,%d,%s,%s", PACKAGE_VERSION,
509: (long long) now, "OK", ctl->Hostname, at + 1, name,
510: fmtinfo);
511: } else
512: #endif
513: printf("MTR.%s,%lld,%s,%s,%d,%s", PACKAGE_VERSION,
514: (long long) now, "OK", ctl->Hostname, at + 1, name);
515:
516: for (i = 0; i < MAXFLD; i++) {
517: j = ctl->fld_index[ctl->fld_active[i]];
518: if (j < 0)
519: continue;
520:
521: /* 1000.0 is a temporay hack for stats usec to ms, impacted net_loss. */
522: if (strchr(data_fields[j].format, 'f')) {
523: printf(",%.2f",
524: (double) (data_fields[j].net_xxx(at) / 1000.0));
525: } else {
526: printf(",%d", data_fields[j].net_xxx(at));
527: }
528: }
529: printf("\n");
530: }
531: }
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>