【分享】一个集成tracert和ping的网络监测工具

简介: 最近接到一个需求,需求背景是这样的:目前Windows平台下本身都有tracert和ping的实现,而且可以直接在cmd下使用。需求中有两个要求:1. Windows平台中的tracert执行速度太慢,一次tracert可能要花十几分钟。

最近接到一个需求,需求背景是这样的:目前Windows平台下本身都有tracert和ping的实现,而且可以直接在cmd下使用。

需求中有两个要求:

1. Windows平台中的tracert执行速度太慢,一次tracert可能要花十几分钟。所以,需要一个快速的tracert实现。

2.将tracert和ping结合起来:tracert出来的节点,需要ping一下,看看当前节点的连通性。

 

接到这个需求后,开始的时候有点无从下手,只能先从研究tracert的原理出发,搞清楚原理后才能去提升速率。

tracert原理:

Tracert 命令用 IP 生存时间 (TTL) 字段和 ICMP 错误消息来确定从一个主机到网络上其他主机的路由。

首先,tracert送出一个TTL是1的IP 数据包到目的地,当路径上的第一个路由器收到这个数据包时,它将TTL减1。此时,TTL变为0,所以该路由器会将此数据包丢掉,并送回一个「ICMP time exceeded」消息(包括发IP包的源地址,IP包的所有内容及路由器的IP地址),tracert 收到这个消息后,便知道这个路由器存在于这个路径上,接着tracert 再送出另一个TTL是2 的数据包,发现第2 个路由器,以此往复。。。。。

tracert 每次将送出的数据包的TTL 加1来发现另一个路由器,这个重复的动作一直持续到某个数据包 抵达目的地。当数据包到达目的地后,该主机则不会送回ICMP time exceeded消息,一旦到达目的地,由于tracert通过UDP数据包向不常见端口(30000以上)发送数据包,因此会收到「ICMP port unreachable」消息,故可判断到达目的地。

tracert 有一个固定的时间等待响应(ICMP TTL到期消息)。如果这个时间过了,它将打印出一系列的*号表明:在这个路径上,这个设备不能在给定的时间内发出ICMP TTL到期消息的响应。然后,Tracert给TTL记数器加1,继续进行。

读懂tracert的原理后,其实不难发现,TTL=1到TTL=30的请求其实关联性不大,我们要加速实现tracert可以从这里入手。

看一下tracert结果的格式:可以发现,每一次tracert最多会有30个节点,每个节点包含3个时间数据,经查阅是三次请求的响应时间

 

网上查阅了一些资料,发现可以基于ICMP.dll实现tracert,网友给出了代码:

  1 //TraceRoute3.cpp
  2 #include <stdio.h>
  3 #include <winsock2.h>
  4 #include <windows.h>
  5 #include <IPHlpApi.h>
  6 //增加静态链接库ws2_32.lib
  7 #pragma comment(lib,"ws2_32.lib")
  8 //声明3个函数类型的指针
  9 typedef HANDLE (WINAPI *lpIcmpCreateFile)(VOID);
 10 typedef BOOL (WINAPI *lpIcmpCloseHandle)(HANDLE  IcmpHandle);
 11 typedef DWORD (WINAPI *lpIcmpSendEcho)(
 12     HANDLE                   IcmpHandle,
 13     IPAddr                   DestinationAddress,
 14     LPVOID                   RequestData,
 15     WORD                     RequestSize,
 16     PIP_OPTION_INFORMATION   RequestOptions,
 17     LPVOID                   ReplyBuffer,
 18     DWORD                    ReplySize,
 19     DWORD                    Timeout
 20     );
 21 int main(int argc,char* argv[]){
 22  if(argc!=2){
 23   printf("Usage: %s destIP/n",argv[0]);
 24   exit(-1);
 25  }
 26  WSADATA wsa;
 27  if(WSAStartup(MAKEWORD(2,2),&wsa)!=0){
 28   printf("WSAStartup failed./n");
 29   exit(-1);
 30  }
 31  //转换IP地址到整数
 32  unsigned long ip = inet_addr(argv[1]);
 33  if(ip==INADDR_NONE){
 34   //用户可能输入的是域名
 35   hostent* pHost = gethostbyname(argv[1]);
 36   //如果域名无法解析
 37   if(pHost==NULL){
 38    printf("Invalid IP or domain name: %s/n", argv[1]);
 39    exit(-1);
 40   }
 41   //取域名的第一个IP地址
 42   ip = *(unsigned long*)pHost->h_addr_list[0];
 43   printf("trace route to %s(%s)/n/n",argv[1],inet_ntoa(*(in_addr*)&ip));
 44  }else{
 45   printf("trace route to %s/n/n",argv[1]);
 46  }
 47  //载入ICMP.DLL动态库
 48  HMODULE hIcmpDll = LoadLibrary("icmp.dll");
 49  if(hIcmpDll==NULL){
 50   printf("fail to load icmp.dll/n");
 51   exit(-1);
 52  }
 53  //定义3个函数指针
 54  lpIcmpCreateFile IcmpCreateFile;
 55  lpIcmpCloseHandle IcmpCloseHandle;
 56  lpIcmpSendEcho IcmpSendEcho;
 57  //从ICMP.DLL中获取所需的函数入口地址
 58  IcmpCreateFile = (lpIcmpCreateFile)GetProcAddress(hIcmpDll,"IcmpCreateFile");
 59  IcmpCloseHandle = (lpIcmpCloseHandle)GetProcAddress(hIcmpDll,"IcmpCloseHandle");
 60  IcmpSendEcho = (lpIcmpSendEcho)GetProcAddress(hIcmpDll,"IcmpSendEcho");
 61  //打开ICMP句柄
 62  HANDLE hIcmp;
 63  if ((hIcmp = IcmpCreateFile()) == INVALID_HANDLE_VALUE){
 64   printf("/tUnable to open ICMP file./n");
 65   exit(-1);
 66  }
 67  //设置IP报头的TTL值
 68  IP_OPTION_INFORMATION IpOption;
 69  ZeroMemory(&IpOption,sizeof(IP_OPTION_INFORMATION));
 70  IpOption.Ttl = 1;
 71  //设置要发送的数据
 72  char SendData[32];
 73  memset(SendData,'0',sizeof(SendData));
 74  //设置接收缓冲区
 75  char ReplyBuffer[sizeof(ICMP_ECHO_REPLY)+32];
 76  PICMP_ECHO_REPLY pEchoReply = (PICMP_ECHO_REPLY)ReplyBuffer;
 77  BOOL bLoop = TRUE;
 78  int iMaxHop = 30;
 79  while(bLoop && iMaxHop--){
 80   printf("%2d: ",IpOption.Ttl);
 81   //发送ICMP回显请求
 82   if(IcmpSendEcho(hIcmp,(IPAddr)ip, SendData, sizeof(SendData), &IpOption, 
 83    ReplyBuffer, sizeof(ReplyBuffer), 3000)!=0){
 84     if(pEchoReply->RoundTripTime==0){
 85      printf("/t<1ms");
 86     }else{
 87      printf("/t%dms",pEchoReply->RoundTripTime);
 88     }
 89     printf("/t%s/n",inet_ntoa(*(in_addr*)&(pEchoReply->Address)));
 90     //判断是否完成路由路径探测
 91     if((unsigned long)pEchoReply->Address==ip){
 92      printf("/nTrace complete./n");
 93      bLoop = FALSE;
 94     }
 95   }else{
 96    printf("/t*/tRequest time out./n");
 97   }
 98   IpOption.Ttl++;
 99  }
100  IcmpCloseHandle(hIcmp);
101  FreeLibrary(hIcmpDll);
102  WSACleanup();
103  return 0;
104 }
View Code

运行后可以看到,这里实现了和windows的dos里面一模一样的tracert,但距离我们想要的“快速”还有一定差距。

我们可以在这个基础上做改进:使用多线程来实现最多30跳跃点的请求,每次请求分别执行三次,最后我们可以直接使用90个线程来执行程序,最终将结果

按固定位置填充到定义的数组中。

至于结合ping就比较简单了,直接调用 windows下的ping命令。

最后,需要注意的是,我们使用c++做成一个exe供程序直接调用,这里需要接收外部参数:网址,是否执行ping命令,ping命令的-l参数,ping命令的-n参数

最终结果:

最终效果非常好,执行效率相比windows下的tracert快了非常多,而且非常方便。

注意:在某些系统上执行的时候会报错,需要添加两个dll文件:msvcp120d.dll和msvcr120d.dll到本机的windows/system32或者windows/SysWOW64下

 小工具分享给大家,下载地址

 

 

目录
相关文章
|
2月前
|
运维 网络协议 安全
【Shell 命令集合 网络通讯 】Linux 网络抓包工具 tcpdump命令 使用指南
【Shell 命令集合 网络通讯 】Linux 网络抓包工具 tcpdump命令 使用指南
50 0
|
2月前
|
网络协议 Linux 网络安全
curl(http命令行工具):Linux下最强大的网络数据传输工具
curl(http命令行工具):Linux下最强大的网络数据传输工具
64 0
|
2月前
|
数据采集 JavaScript 前端开发
实用工具推荐:适用于 TypeScript 网络爬取的常用爬虫框架与库
实用工具推荐:适用于 TypeScript 网络爬取的常用爬虫框架与库
|
14天前
|
IDE 测试技术 项目管理
集成开发环境(IDE)的使用:提升Visual Basic开发效率的工具和技巧
【4月更文挑战第27天】本文探讨了如何使用Visual Basic IDE提升开发效率,包括理解IDE组件、利用代码编辑器的智能功能、通过界面设计器设计GUI、使用调试和测试工具、有效管理项目与版本控制、掌握快捷键和宏、定制IDE以及利用学习资源。通过充分利用这些工具和技巧,开发者能更快地编写高质量代码,高效管理项目,从而提升整体开发效率。随着IDE的持续发展,开发者应不断学习新特性以适应进步。
|
1天前
|
Cloud Native 关系型数据库 分布式数据库
数据库性能诊断工具DBdoctor通过阿里云PolarDB产品生态集成认证
DBdoctor(V3.1.0)成功通过阿里云PolarDB分布式版(V2.3)集成认证,展现优秀兼容性和稳定性。此工具是聚好看科技的内核级数据库性能诊断产品,运用eBPF技术诊断SQL执行,提供智能巡检、根因分析和优化建议。最新版V3.1.1增加了对PolarDB-X和OceanBase的支持,以及基于cost的索引诊断功能。PolarDB-X是阿里巴巴的高性能云原生分布式数据库,兼容MySQL生态。用户可通过提供的下载地址、在线试用链接和部署指南体验DBdoctor。
|
10天前
|
敏捷开发 监控 测试技术
探索自动化测试工具Selenium Grid的高效集成策略
【4月更文挑战第30天】在现代Web应用的快速迭代和持续部署中,测试自动化已成为确保产品质量的关键。Selenium Grid作为一款支持多种浏览器和操作系统的测试工具,提供了并行执行测试用例的能力,极大地提升了测试效率。本文将深入探讨如何高效地将Selenium Grid集成到现有的测试框架中,以及实施过程中的最佳实践,帮助团队最大化测试覆盖率,同时降低资源消耗。
|
10天前
|
中间件 测试技术 API
探索自动化测试工具的新边界:Selenium与Appium的集成实践
【4月更文挑战第30天】 随着移动应用和Web应用的不断融合,传统的自动化测试工具需要适应新的测试环境。本文将详细分析Selenium和Appium这两款流行的自动化测试工具的集成实践,探讨如何构建一个能够同时支持Web和移动端应用的自动化测试框架。通过对比两者的技术架构、功能特性以及在实际项目中的集成过程,我们旨在为读者提供一个清晰的指导,帮助他们在复杂的应用环境中实现高效、稳定的自动化测试流程。
|
10天前
|
弹性计算 运维 Shell
自动化网络扫描工具发现活跃主机
【4月更文挑战第30天】
9 0
|
11天前
|
分布式计算 DataWorks NoSQL
DataWorks产品使用合集之DataWorks 集成工具是否支持对 MongoDB 的单字段更新操作
DataWorks作为一站式的数据开发与治理平台,提供了从数据采集、清洗、开发、调度、服务化、质量监控到安全管理的全套解决方案,帮助企业构建高效、规范、安全的大数据处理体系。以下是对DataWorks产品使用合集的概述,涵盖数据处理的各个环节。
21 0
|
12天前
|
运维 监控 Linux
【专栏】Linux中的ping命令不仅用于基础网络连通性检查,Linux 中这些高级 ping 命令可以提高工作效率!
【4月更文挑战第28天】Linux中的ping命令不仅用于基础网络连通性检查,还有许多高级功能。了解如`-c`(设置数据包数量)、`-i`(设置间隔时间)和`-w`(设置超时时间)等选项能提升效率。进阶技巧包括自定义数据包大小(`-s`)、详细统计信息(`-v`)、持续ping(`-t`)、指定源地址(`-S`)和多目标ping。这些在网络性能测试、故障排查和监控中极其有用。注意权限、参数选择,并结合其他工具以准确解读结果。提升网络管理技能,善用ping命令的全部潜力。

热门文章

最新文章