Annotation of embedaddon/mtr/ui/report.c, revision 1.1

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

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