0

我正在尝试在 Python 中实现基于 ICMP 的 Traceroute。

我找到了一个非常有用的指南(https://blogs.oracle.com/ksplice/entry/learning_by_doing_writing_your),它允许我创建一个基于 UDP 的 Traceroute(下面的代码),所以只需要修改。我尝试将发送套接字更改为 ICMP,但是没有异常我无法运行任何东西。

注意 - 下面的代码有效,但是这是一个 UDP 跟踪路由(发送一个 UDP 数据包并接收一个 ICMP 数据包),我需要我的程序发送一个 ICMP 数据包并接收一个 ICMP 数据包。本质上,UDP 套接字需要更改为 ICMP 套接字。

我想这不是最常见的尝试和实现的事情,并且在网上研究如何做到这一点时遇到了麻烦。如果有人可以提供一些见解,将不胜感激:-)

#!/usr/bin/python

    import socket

    def main(dest_name):
        dest_addr = socket.gethostbyname(dest_name)
        port = 33434
        max_hops = 30
        icmp = socket.getprotobyname('icmp')
        udp = socket.getprotobyname('udp')
        ttl = 1
        while True:
            recv_socket = socket.socket(socket.AF_INET, socket.SOCK_RAW, icmp)
            send_socket = socket.socket(socket.AF_INET, socket.SOCK_DGRAM, udp)
            send_socket.setsockopt(socket.SOL_IP, socket.IP_TTL, ttl)
            recv_socket.bind(("", port))
            send_socket.sendto("", (dest_name, port))
            curr_addr = None
            curr_name = None
            try:
                _, curr_addr = recv_socket.recvfrom(512)
                curr_addr = curr_addr[0]
                try:
                    curr_name = socket.gethostbyaddr(curr_addr)[0]
                except socket.error:
                    curr_name = curr_addr
            except socket.error:
                pass
            finally:
                send_socket.close()
                recv_socket.close()

            if curr_addr is not None:
                curr_host = "%s (%s)" % (curr_name, curr_addr)
            else:
                curr_host = "*"
            print "%d\t%s" % (ttl, curr_host)

            ttl += 1
            if curr_addr == dest_addr or ttl > max_hops:
                break

    if __name__ == "__main__":
        main('google.com')
4

1 回答 1

4

我认为 udp 是你最好的方法

由于 RFC 791,traceroute 在发送 icmp 数据包时无效,导致原作者 Van Jacobson 使用 udp 重新实现。查看此页面上的脚注以进一步澄清http://www.inetdaemon.com/tutorials/troubleshooting/tools/traceroute/definition.shtml

如果你还坚持发送icmp,试试这个库

http://code.activestate.com/recipes/409689-icmplib-library-for-creating-and-reading-icmp-pack/

于 2012-04-16T09:17:55.230 回答