File:  [ELWIX - Embedded LightWeight unIX -] / embedaddon / mtr / ui / report.c
Revision 1.1.1.3 (vendor branch): download - view: text, annotated - select for diffs - revision graph
Wed Sep 27 11:18:58 2023 UTC (18 months, 1 week ago) by misho
Branches: mtr, MAIN
CVS tags: v0_95, HEAD
Version 0.95

    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: 
   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.
   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>
   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
   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 320
   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->af, addr));
   69:         else if (ctl->dns && ctl->show_ips)
   70:             return snprintf(dst, dst_len, "%s (%s)", host->h_name,
   71:                             strlongip(ctl->af, 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++)
   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]);
   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 temporary 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: 
  187:         /* Print list of all hosts that have responded from ttl = at + 1 away */
  188:         for (z = 0; z < MAX_PATH; z++) {
  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,
  194:                   ctl->af)) == 0) {
  195:                 break;
  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:             }
  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 (mpls->labels && z == 1 && ctl->enablempls)
  219:                     print_mpls(mpls);
  220:                 if (is_printii(ctl)) {
  221:                     snprint_addr(ctl, name, sizeof(name), addr2);
  222:                     printf("     %s%s\n", fmt_ipinfo(ctl, addr2), name);
  223:                 }
  224:                 if (ctl->enablempls)
  225:                     print_mpls(mplss);
  226: #else
  227:                 int k;
  228:                 if (mpls->labels && z == 1 && ctl->enablempls) {
  229:                     for (k = 0; k < mpls->labels; k++) {
  230:                         printf
  231:                             ("    |  |+-- [MPLS: Lbl %lu TC %u S %u TTL %u]\n",
  232:                              mpls->label[k], mpls->tc[k], mpls->s[k],
  233:                              mpls->ttl[k]);
  234:                     }
  235:                 }
  236: 
  237:                 if (z == 1) {
  238:                     printf("    |  `|-- %s\n", strlongip(ctl->af, addr2));
  239:                     for (k = 0; k < mplss->labels && ctl->enablempls; k++) {
  240:                         printf
  241:                             ("    |   +-- [MPLS: Lbl %lu TC %u S %u TTL %u]\n",
  242:                              mplss->label[k], mplss->tc[k], mplss->s[k],
  243:                              mplss->ttl[k]);
  244:                     }
  245:                 } else {
  246:                     printf("    |   |-- %s\n", strlongip(ctl->af, addr2));
  247:                     for (k = 0; k < mplss->labels && ctl->enablempls; k++) {
  248:                         printf
  249:                             ("    |   +-- [MPLS: Lbl %lu TC %u S %u TTL %u]\n",
  250:                              mplss->label[k], mplss->tc[k], mplss->s[k],
  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++) {
  268:                 printf("    |   +-- [MPLS: Lbl %lu TC %u S %u TTL %u]\n",
  269:                        mpls->label[k], mpls->tc[k], mpls->s[k],
  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: 
  290: #ifdef HAVE_JANSSON
  291: void json_open(
  292:     void)
  293: {
  294: }
  295: 
  296: void json_close(struct mtr_ctl *ctl)
  297: {
  298:     int i, j, at, max;
  299:     int ret;
  300:     char buf[128];
  301:     json_t *jreport, *jmtr, *jhubs, *jh;
  302:     ip_t *addr;
  303:     char name[MAX_FORMAT_STR];
  304: 
  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: 
  313:     if (ctl->cpacketsize >= 0) {
  314:         snprintf(buf, sizeof(buf), "%d", ctl->cpacketsize);
  315:     } else {
  316:         snprintf(buf, sizeof(buf), "rand(%d-%d)", MINPACKET, -ctl->cpacketsize);
  317:     }
  318:     ret = json_object_set_new(jmtr, "psize", json_string(buf));
  319:     if (ret == -1)
  320:         goto on_error;
  321: 
  322:     if (ctl->bitpattern >= 0) {
  323:         snprintf(buf, sizeof(buf), "0x%02X", (unsigned char)(ctl->bitpattern));
  324:     } else {
  325:         snprintf(buf, sizeof(buf), "rand(0x00-FF)");
  326:     }
  327: 
  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;
  335: 
  336:     max = net_max(ctl);
  337:     at = net_min(ctl);
  338:     for (; at < max; at++) {
  339:         addr = net_addr(at);
  340:         snprint_addr(ctl, name, sizeof(name), addr);
  341: 
  342:         jh = json_pack("{si ss}", "count", at + 1, "host", name);
  343:         if (!jh)
  344:             goto on_error;
  345: 
  346: #ifdef HAVE_IPINFO
  347:         if (!ctl->ipinfo_no) {
  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;
  355:         }
  356: #endif
  357: 
  358:         for (i = 0; i < MAXFLD; i++) {
  359:             j = ctl->fld_index[ctl->fld_active[i]];
  360: 
  361:             if (j <= 0)
  362:                 continue; /* Field nr 0, " " shouldn't be printed in this method. */
  363: 
  364:             if (strchr(data_fields[j].format, 'f')) {
  365:                 ret = json_object_set_new(
  366:                     jh, data_fields[j].title,
  367:                     json_real(data_fields[j].net_xxx(at) / 1000.0));
  368:             } else {
  369:                 ret = json_object_set_new(
  370:                     jh, data_fields[j].title,
  371:                     json_integer(data_fields[j].net_xxx(at)));
  372:             }
  373:             if (ret == -1)
  374:                 goto on_error;
  375:         }
  376: 
  377:         ret = json_array_append_new(jhubs, jh);
  378:         if (ret == -1)
  379:             goto on_error;
  380:     }
  381: 
  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 prompt 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
  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 temporary 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, z, w;
  474:     ip_t *addr;
  475:     ip_t *addr2 = NULL;
  476:     char name[MAX_FORMAT_STR];
  477: 
  478:     for (i = 0; i < MAXFLD; i++) {
  479:         j = ctl->fld_index[ctl->fld_active[i]];
  480:         if (j < 0)
  481:             continue;
  482:     }
  483: 
  484:     max = net_max(ctl);
  485:     at = net_min(ctl);
  486:     for (; at < max; at++) {
  487:         addr = net_addr(at);
  488:         snprint_addr(ctl, name, sizeof(name), addr);
  489: 
  490:         if (at == net_min(ctl)) {
  491:             printf("Mtr_Version,Start_Time,Status,Host,Hop,Ip,");
  492: #ifdef HAVE_IPINFO
  493:             if (!ctl->ipinfo_no) {
  494:                 printf("Asn,");
  495:             }
  496: #endif
  497:             for (i = 0; i < MAXFLD; i++) {
  498:                 j = ctl->fld_index[ctl->fld_active[i]];
  499:                 if (j < 0)
  500:                     continue;
  501:                 printf("%s,", data_fields[j].title);
  502:             }
  503:             printf("\n");
  504:         }
  505: #ifdef HAVE_IPINFO
  506:         if (!ctl->ipinfo_no) {
  507:             char *fmtinfo = fmt_ipinfo(ctl, addr);
  508:             fmtinfo = trim(fmtinfo, '\0');
  509:             printf("MTR.%s,%lld,%s,%s,%d,%s,%s", PACKAGE_VERSION,
  510:                    (long long) now, "OK", ctl->Hostname, at + 1, name,
  511:                    fmtinfo);
  512:         } else
  513: #endif
  514:             printf("MTR.%s,%lld,%s,%s,%d,%s", PACKAGE_VERSION,
  515:                    (long long) now, "OK", ctl->Hostname, at + 1, name);
  516: 
  517:         for (i = 0; i < MAXFLD; i++) {
  518:             j = ctl->fld_index[ctl->fld_active[i]];
  519:             if (j < 0)
  520:                 continue;
  521: 
  522:             /* 1000.0 is a temporary hack for stats usec to ms, impacted net_loss. */
  523:             if (strchr(data_fields[j].format, 'f')) {
  524:                 printf(",%.2f",
  525:                        (double) (data_fields[j].net_xxx(at) / 1000.0));
  526:             } else {
  527:                 printf(",%d", data_fields[j].net_xxx(at));
  528:             }
  529:         }
  530:         printf("\n");
  531:         if (ctl->reportwide == 0)
  532:             continue;
  533:         
  534:         for (z = 0; z < MAX_PATH; z++) {
  535:             int found = 0;
  536:             addr2 = net_addrs(at, z);
  537:             snprint_addr(ctl, name, sizeof(name), addr2);
  538:             if ((addrcmp
  539:                     ((void *) &ctl->unspec_addr, (void *) addr2,
  540:                      ctl->af)) == 0) {
  541:                 break;
  542:             } else if ((addrcmp
  543:                     ((void *) addr, (void *) addr2,
  544:                      ctl->af)) == 0) {
  545:                 continue; /* Latest Host is already printed */
  546:             } else {
  547:                 for (w = 0; w < z; w++)
  548:                     /* Ok... checking if there are ips repeated on same hop */
  549:                     if ((addrcmp
  550:                             ((void *) addr2, (void *) net_addrs(at, w),
  551:                              ctl->af)) == 0) {
  552:                         found = 1;
  553:                         break;
  554:                     }
  555: 
  556:                 if (!found) {
  557: #ifdef HAVE_IPINFO
  558:                     if (!ctl->ipinfo_no) {
  559:                         char *fmtinfo = fmt_ipinfo(ctl, addr2);
  560:                         fmtinfo = trim(fmtinfo, '\0');
  561:                         printf("MTR.%s,%lld,%s,%s,%d,%s,%s", PACKAGE_VERSION,
  562:                             (long long) now, "OK", ctl->Hostname, at + 1, name,
  563:                             fmtinfo);
  564:                     } else
  565: #endif
  566:                         printf("MTR.%s,%lld,%s,%s,%d,%s", PACKAGE_VERSION,
  567:                            (long long) now, "OK", ctl->Hostname, at + 1, name);
  568: 
  569:                     /* Use values associated with the first ip discovered for this hop */
  570:                     for (i = 0; i < MAXFLD; i++) {
  571:                         j = ctl->fld_index[ctl->fld_active[i]];
  572:                         if (j < 0)
  573:                             continue;
  574: 
  575:                         /* 1000.0 is a temporary hack for stats usec to ms, impacted net_loss. */
  576:                         if (strchr(data_fields[j].format, 'f')) {
  577:                             printf(",%.2f",
  578:                                    (double) (data_fields[j].net_xxx(at) / 1000.0));
  579:                         } else {
  580:                             printf(",%d", data_fields[j].net_xxx(at));
  581:                         }
  582:                     }
  583:                     printf("\n");
  584:                 }
  585:             }    
  586:         }
  587:     }
  588: }

FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>