参考了网上几个 traceroute的实现版本,存在一些缺陷,比如没有做超时处理,或者只能在window下使用。自己用boost实现了一个traceroute小工具,在window下正常运行。

先来看下面实现的原理。这些说明来自维基百科。traceroute,现代Linux系统称为tracepath, Windows 系统称为tracert,是一种电脑网络工具。它可显示 数据包 在IP网络经过的路由器的 IP地址。 程序利用增加存活时间(TTL)值来实现其功能的。每当数据包经过一个路由器,其存活时间就会减1。当其存活时间是0时,主机便取消数据包,并传送一个 ICMP TTL数据包给原数据包的发出者。

下面是具体实现的源代码:

  1 //
  2 // traceroute.cpp
  3 // ~~~~~~~~
  4 //
  5 // BoSheng Lu 2013-04-25
  6 //
  7 // Version 1.0.
  8 //
 10 #include <boost/asio.hpp>
 11 #include <boost/bind.hpp>
 12 #include<boost/thread.hpp>
 13 #include <istream>
 14 #include <iostream>
 15 #include<iomanip>
 16 #include<string> 
 17 #include <ostream>
 19 #include "icmp_header.hpp"
 20 #include "ipv4_header.hpp"
 22 using boost::asio::ip::icmp;
 23 using boost::asio::deadline_timer;
 24 namespace posix_time = boost::posix_time;
 26 class traceroute
 27 {
 28 public:
 29     traceroute(boost::asio::io_service& io_service, const char* destination)
 30         : resolver_(io_service), socket_(io_service, icmp::v4()),
 31         sequence_number_(0), max_hop_(30)
 32     {
 33         icmp::resolver::query query(icmp::v4(), destination, "");
 34         destination_ = *resolver_.resolve(query);
 36         std::cout << "\nTracing route to " << destination 
 37         << " [" << destination_.address().to_string() << "]"
 38         << " with a maximum of " << max_hop_ << " hops.\n" << std::endl;
 40 #if defined(BOOST_WINDOWS)
 41         int timeout = 5000;
 42         if( setsockopt(socket_.native(), SOL_SOCKET, SO_RCVTIMEO, 
 43             (const char*)&timeout, sizeof(timeout)))
 44         {
 45              std::cout << "RCVTIMEO not set properly." << std::endl;
 46             throw std::runtime_error("RCVTIMEO not set properly");
 47         }
 48 #else
 49         struct timeval tv;
 50         tv.tv_sec  = 5; 
 51         tv.tv_usec = 0;         
 52         if( setsockopt(socket_.native(), SOL_SOCKET, SO_RCVTIMEO, &tv, sizeof(tv)))
 53         {
 54             std::cout << "RCVTIMEO not set properly." << std::endl;
 55             throw std::runtime_error("RCVTIMEO not set properly");
 56         }
 57 #endif
 59         reach_dest_host_ = false;
 60         int ttl = 1;
 61         while (!reach_dest_host_ && max_hop_--)
 62         {
 63             const boost::asio::ip::unicast::hops option(ttl);
 64             socket_.set_option(option);
 66             boost::asio::ip::unicast::hops op;
 67             socket_.get_option(op);
 68             if( ttl !=  op.value() )
 69             {
 70                 std::cout << "TTL not set properly. Should be "
 71                     << ttl << " but was set to "
 72                     << op.value() << '.' << std::endl;
 73                 throw std::runtime_error("TTL not set properly");        
 74             }
 76             std::cout << std::setw(3) << ttl << std::flush;
 77             start_send();
 78             start_receive();
 80             ttl++;
 81         }//while
 82     }
 84 private:
 85     void start_send()
 86     {
 87         std::string body("\"Hello!\" from Asio ping.");
 89         // Create an ICMP header for an echo request.
 90         icmp_header echo_request;
 91         echo_request.type(icmp_header::echo_request);
 92         echo_request.code(0);
 93         echo_request.identifier(get_identifier());
 94         echo_request.sequence_number(++sequence_number_);
 95         compute_checksum(echo_request, body.begin(), body.end());
 97         // Encode the request packet.
 98         boost::asio::streambuf request_buffer;
 99         std::ostream os(&request_buffer);
100         os << echo_request << body;
102         // Send the request.
103         time_sent_ = posix_time::microsec_clock::universal_time();
104         socket_.send_to(request_buffer.data(), destination_);  
105     }
108     void start_receive()
109     {
110         // Discard any data already in the buffer.
111         reply_buffer_.consume(reply_buffer_.size());
113         // Wait for a reply. We prepare the buffer to receive up to 64KB.
114         boost::system::error_code ec;
115         boost::asio::socket_base::message_flags flags = 0;
116         size_t len = socket_.receive(reply_buffer_.prepare(65536), flags, ec);
118         handle_receive(len, ec);
119     }
121     void handle_receive(std::size_t length, boost::system::error_code& ec)
122     {
123         // The actual number of bytes received is committed to the buffer so that we





    
124         // can extract it using a std::istream object.
125         reply_buffer_.commit(length);
127         // Decode the reply packet.
128         std::istream is(&reply_buffer_);
129         ipv4_header ipv4_hdr;
130         is >> ipv4_hdr;
132         if(ec)
133         {
134             std::cout << std::setw(9) << '*' << '\t' << "Request timed out." << std::endl;
135             return;
136         }
138         posix_time::ptime now = posix_time::microsec_clock::universal_time();
139         long long dwRoundTripTime = (now - time_sent_).total_milliseconds();
141         if(dwRoundTripTime)
142         {
143             std::cout << std::setw(6) << dwRoundTripTime << " ms"  << std::flush;
144         }
145         else
146         {
147             std::cout << std::setw(6) << "<1" << " ms" << std::flush;
148         }
150         boost::this_thread::sleep(boost::posix_time::seconds(2)); 
152         std::cout << '\t' << ipv4_hdr.source_address() << std::endl;
154         // Print out some information about the reply packet.
155         if( ipv4_hdr.source_address().to_string() == destination_.address().to_string() )
156         {
157             reach_dest_host_ = true;
158             std::cout << std::endl;
159             std::cout << std::endl;
160             std::cout << "traceroute sucess!" << std::endl;
162             icmp_header icmp_hdr;
163             is >> icmp_hdr;
165             if (is && icmp_hdr.type() == icmp_header::echo_reply
166                 && icmp_hdr.identifier() == get_identifier()
167                 && icmp_hdr.sequence_number() == sequence_number_)
168             {
169                 // Print out some information about the reply packet.
170                 std::cout << length - ipv4_hdr.header_length()
171                  << " bytes from " << ipv4_hdr.source_address()
172                 << ": icmp_seq=" << icmp_hdr.sequence_number()
173                 << ", ttl=" << ipv4_hdr.time_to_live()
174                 << ", time=" << dwRoundTripTime << " ms"
175                 << std::endl;
176             }
177         }
178      }
180     static unsigned short get_identifier()
181     {
182 #if defined(BOOST_WINDOWS)
183         return static_cast<unsigned short>(::GetCurrentProcessId());
184 #else
185         return static_cast<unsigned short>(::getpid());
186 #endif
187     }
189     icmp::resolver resolver_;
190     icmp::endpoint destination_;
191     icmp::socket socket_;
192     unsigned short sequence_number_;
193     posix_time::ptime time_sent_;
194     boost::asio::streambuf reply_buffer_;
195     bool reach_dest_host_;
196     std::size_t max_hop_;
197 };
199 int main(int argc, char* argv[])
200 {
201     try
202     {
203         if(argc != 2)
204         {
205             std::cerr << "Usage: ping <host>" << std::endl;
206 #if !defined(BOOST_WINDOWS)
207             std::cerr << "(You may need to run this program as root.)" << std::endl;
208 #endif
209             return 1;
210         }
212         boost::asio::io_service io_service;
213         traceroute p(io_service, argv[1]);
214         io_service.run();
215     }
216     catch (std::exception& e)
217     {
218         std::cerr << "Exception: " << e.what() << std::endl;
219     }
220 }

这个源码实现使用了boost库附带的boost asio使用示例中的代码,下面也将这个例子中使用到的代码附上,方便使用。至于boost库的编译使用另请参考其他的文章。

下面是 ipv4_header.hpp

// ipv4_header.hpp // ~~~~~~~~~~~~~~~ // Distributed under the Boost Software License, Version 1.0. (See accompanying // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt ) #ifndef IPV4_HEADER_HPP #define IPV4_HEADER_HPP #include <algorithm> #include <boost/asio/ip/address_v4.hpp> // Packet header for IPv4. // The wire format of an IPv4 header is: // 0 8 16 31 // +-------+-------+---------------+------------------------------+ --- // | | | | | ^ // |version|header | type of | total length in bytes | | // | (4) | length| service | | | // +-------+-------+---------------+-+-+-+------------------------+ | // | | | | | | | // | identification |0|D|M| fragment offset | | // | | |F|F| | | // +---------------+---------------+-+-+-+------------------------+ | // | | | | | // | time to live | protocol | header checksum | 20 bytes // | | | | | // +---------------+---------------+------------------------------+ | // | | | // | source IPv4 address | | // | | | // +--------------------------------------------------------------+ | // | | | // | destination IPv4 address | | // | | v // +--------------------------------------------------------------+ --- // | | ^ // | | | // / options (if any) / 0 - 40 // / / bytes // | | | // | | v // +--------------------------------------------------------------+ --- class ipv4_header public : ipv4_header() { std::fill(rep_, rep_ + sizeof (rep_), 0 ); } unsigned char version() const { return (rep_[ 0 ] >> 4 ) & 0xF ; } unsigned short header_length() const { return (rep_[ 0 ] & 0xF ) * 4 ; } unsigned char type_of_service() const { return rep_[ 1 ]; } unsigned short total_length() const { return decode( 2 , 3 ); } unsigned short identification() const { return decode( 4 , 5 ); } bool dont_fragment() const { return (rep_[ 6 ] & 0x40 ) != 0 ; } bool more_fragments() const { return (rep_[ 6 ] & 0x20 ) != 0 ; } unsigned short fragment_offset() const { return decode( 6 , 7 ) & 0x1FFF ; } unsigned int time_to_live() const { return rep_[ 8 ]; } unsigned char protocol() const { return rep_[ 9 ]; } unsigned short header_checksum() const { return decode( 10 , 11 ); } boost::asio::ip::address_v4 source_address() const boost::asio::ip::address_v4::bytes_type bytes = { { rep_[ 12 ], rep_[ 13 ], rep_[ 14 ], rep_[ 15 ] } }; return boost::asio::ip::address_v4(bytes); boost::asio::ip::address_v4 destination_address() const boost::asio::ip::address_v4::bytes_type bytes = { { rep_[ 16 ], rep_[ 17 ], rep_[ 18 ], rep_[ 19 ] } }; return boost::asio::ip::address_v4(bytes); friend std::istream & operator >>(std::istream& is , ipv4_header& header) is .read(reinterpret_cast< char *>(header.rep_), 20 ); if (header.version() != 4 ) is .setstate(std::ios::failbit); std::streamsize options_length = header.header_length() - 20 ; if (options_length < 0 || options_length > 40 ) is .setstate(std::ios::failbit); is .read(reinterpret_cast< char *>(header.rep_) + 20 , options_length); return is ; private : unsigned short decode( int a, int b) const { return (rep_[a] << 8 ) + rep_[b]; } unsigned char rep_[ 60 ]; #endif // IPV4_HEADER_HPP

下面是 icmp_header.hpp

// icmp_header.hpp // ~~~~~~~~~~~~~~~ // Distributed under the Boost Software License, Version 1.0. (See accompanying // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt ) #ifndef ICMP_HEADER_HPP #define ICMP_HEADER_HPP #include <istream> #include <ostream> #include <algorithm> // ICMP header for both IPv4 and IPv6. // The wire format of an ICMP header is: // 0 8 16 31 // +---------------+---------------+------------------------------+ --- // | | | | ^ // | type | code | checksum | | // | | | | | // +---------------+---------------+------------------------------+ 8 bytes // | | | | // | identifier | sequence number | | // | | | v // +-------------------------------+------------------------------+ --- class icmp_header public : enum { echo_reply = 0 , destination_unreachable = 3 , source_quench = 4 , redirect = 5 , echo_request = 8 , time_exceeded = 11 , parameter_problem = 12 , timestamp_request = 13 , timestamp_reply = 14 , info_request = 15 , info_reply = 16 , address_request = 17 , address_reply = 18 }; icmp_header() { std::fill(rep_, rep_ + sizeof (rep_), 0 ); } unsigned char type() const { return rep_[ 0 ]; } unsigned char code() const { return rep_[ 1 ]; } unsigned short checksum() const { return decode( 2 , 3 ); } unsigned short identifier() const { return decode( 4 , 5 ); } unsigned short sequence_number() const { return decode( 6 , 7 ); } void type(unsigned char n) { rep_[ 0 ] = n; } void code(unsigned char n) { rep_[ 1 ] = n; } void checksum(unsigned short n) { encode( 2 , 3 , n); } void identifier(unsigned short n) { encode( 4 , 5 , n); } void sequence_number(unsigned short n) { encode( 6 , 7 , n); } friend std::istream & operator >>(std::istream& is , icmp_header& header) { return is .read(reinterpret_cast< char *>(header.rep_), 8 ); } friend std::ostream & operator <<(std::ostream& os, const icmp_header& header) { return os.write(reinterpret_cast< const char *>(header.rep_), 8 ); } private : unsigned short decode( int a, int b) const { return (rep_[a] << 8 ) + rep_[b]; } void encode( int a, int b, unsigned short n) rep_[a] = static_cast<unsigned char >(n >> 8 ); rep_[b] = static_cast<unsigned char >(n & 0xFF ); unsigned char rep_[ 8 ]; template <typename Iterator> void compute_checksum(icmp_header& header, Iterator body_begin, Iterator body_end) unsigned int sum = (header.type() << 8 ) + header.code() + header.identifier() + header.sequence_number(); Iterator body_iter = body_begin; while (body_iter != body_end) sum += (static_cast<unsigned char >(*body_iter++) << 8 ); if (body_iter != body_end) sum += static_cast<unsigned char >(*body_iter++ ); sum = (sum >> 16 ) + (sum & 0xFFFF ); sum += (sum >> 16 ); header.checksum(static_cast <unsigned short >(~ sum)); #endif // ICMP_HEADER_HPP