美文网首页
Python网络编程3--实现IP源站路由

Python网络编程3--实现IP源站路由

作者: 净坛使者_猪悟能 | 来源:发表于2021-08-09 12:49 被阅读0次

一、IP报文首部格式

IP首部
  • 字段解释
    • Version
      4:表示为IPV4;
      6:表示为IPV6。
    • IHL
      首部长度,如果不带Option字段,则为20,最长为60,该值限制了记录路由选项。以4字节为一个单位。
    • Type of Service
      服务类型。只有在有QoS差分服务要求时这个字段才起作用。
    • Total Length
      总长度,整个IP数据报的长度,包括首部和数据之和,单位为字节,最长65535,总长度必须不超过最大传输单元MTU。
    • Identification
      标识,主机每发一个报文,加1,分片重组时会用到该字段。
    • Flags
      标志位:
      Bit 0: 保留位,必须为0。
      Bit 1: DF(Don't Fragment),能否分片位,0表示可以分片,1表示不能分片。
      Bit 2: MF(More Fragment),表示是否该报文为最后一片,0表示最后一片,1代表后面还有。
    • Fragment Offset
      片偏移:分片重组时会用到该字段。表示较长的分组在分片后,某片在原分组中的相对位置。以8个字节为偏移单位。
    • Time to Live
      生存时间:可经过的最多路由数,即数据包在网络中可通过的路由器数的最大值。
    • Protocol
      协议:下一层协议。指出此数据包携带的数据使用何种协议,以便目的主机的IP层将数据部分上交给哪个进程处理。
    • Header Checksum
      首部检验和,只检验数据包的首部,不检验数据部分。这里不采用CRC检验码,而采用简单的计算方法。
    • Source Address
      源IP地址。
    • Destination Address
      目的IP地址。
    • Options
      可变 选项字段,用来支持排错,测量以及安全等措施,内容丰富。选项字段长度可变,从1字节到40字节不等,取决于所选项的功能。
    • Padding
      可变 填充字段,全填0。

二、实现IP源站路由

2.1 源站路由

  源站路由可以事先规定IP数据包所经过得路由器,每经过一个路由器就改变数据包的目的地址(下一跳)
  使用IP头部中的option字段记录路由IP。该字段最大40字节,因此最多存放9个IP,记录格式如下:


源站路由

Type:占1字节,code 的值此处设为137。

  • 严格:0x89,清单中的相连两个地址,经过的路由器必须是直连的(因此此时路由器不需要有路由就能转发该数据包),如果所指下一跳不在路由器的直连网络中,将返回一个“源站路由失败”的ICMP差错报文。
  • 宽松:0x83,清单中的相连两个地址,可以经过几个路由器到达

length:占1字节,记录整个选项的长度。
pointer:指针项,占1个字节,指向下一个被处理的源站地址,最小值为4。

2.2 运行过程

  发送主机从应用程序接收源站路由清单,最后一个表项(它是数据报的最终目的地址),剩余的为所有经过的下一跳,每经过一个设备都会检查是否是最终目标,如不是则从列表中读取下一项作为数据包下一跳的目的地址,同时数据包每从一个路由器发出,就记录其出接口的地址,用其替换掉清单中的上一跳地址,数据包返回时任然按照原来的路径返回。
  如下图,主机S上的发送应用程序发送一份数据报给D,指定源路由为R1,R2和R3。#表示指针字段,其值分别是4、8、12和16(一个值表示一个32位IP)。长度字段恒为1 5(三个IP地址加上三个字节首部)。可以看出,每一跳IP数据报中的目的地址都发生改变。


IP源路由

2.3 Python实现

  • 实验环境
    在Ensp上搭建测试网络,其中各路由器上只有直连路由。
    源路由实验
  • Python脚本
#!/usr/bin/python3.4
# -*- coding=utf-8 -*-

from kamene.all import *
import struct

def ip_sec(ip):
    '''对IP地址进行Bytes转换'''
    ip_l = ip.split(".")
    a = struct.pack(">B", int(ip_l[0]))
    b = struct.pack(">B", int(ip_l[1]))
    c = struct.pack(">B", int(ip_l[2]))
    d = struct.pack(">B", int(ip_l[3]))
    return a + b + c + d

def try_lsrr(dst,lsrr_hop):
    r2=ip_sec(lsrr_hop[1])
    D=ip_sec(dst)
    ip_options = b'\x83\x0B\x04'+r2+D+b"\x00"
    #\x83表示宽松源站路由,\x0B表示长度,\x04表示指针,紧跟着四个字节的IP地址(真正的目的地址),\x00补齐8字节边作为结束位。
    #长度为:3(前options前3字节)+8(r2+D)字节
    pkt = IP(dst=lsrr_hop[0], options=IPOption(ip_options))/ICMP(type=8,code=0)
    #目的地址为源站路由的地址,正真的目的地址放在IP选项内
    a=sr1(pkt,timeout = 1, verbose=True)
    result = a
if __name__ == '__main__':
    dest=input("目的IP>>>")
    lsr_route=input("经过的下一跳>>>")
    lsr_hop=lsr_route.split(" ")
    try_lsrr(dest, lsr_hop)
  • 执行效果
    在Pycharm上执行脚本。
    运行效果
    在R1-R2间抓包,可以看到此时request包当前目的为172.16.10.2,IP option中下一跳指向172.16.20.2,实际目标地址为172.16.30.2。
    R1-R2
    在R2-R3之间抓包,此时request包当前目的变为了172.16.20.2,IP Options中记录上一跳的出接口地址172.16.20.1,实际目的地址为172.16.30.2.
    R2-R3
    在R3-R4之间抓包,此时request包马上到达目标地址172.16.30.2,IP Options中记录着前两跳的出接口地址。
    R3-R4
    再看reply包,目的地址为172.16.16.30,IP Options中下一跳为172.16.20.1,实际目标地址为192.168.10.112.
    Reply

相关文章

  • Python网络编程3--实现IP源站路由

    一、IP报文首部格式 字段解释Version4:表示为IPV4;6:表示为IPV6。IHL首部长度,如果不带Opt...

  • s9python网络编程

    python之路——网络编程计算机网络ftp作业 网络编程 mac地址 arp协议:通过ip地址找mac地址 ip...

  • Python网络编程 —— IP、UDP

    Python网络编程 —— IP、UDP IP ip地址: 在网络中标识一台唯一的设备 ip地址的作用: 通过ip...

  • 路由器跳板功能简单说明

    在某些网络运维的过程中,因为网络设置,源IP不能直接访问目的IP,这时需要借助跳板连接目的IP。 路由器的运维管理...

  • 网络的基础知识

    一、数据包在网络层的传输 源IP地址和目的IP地址一定是不会变的,源IP地址是路由器上的外网IP地址 中途会一直变...

  • 计算机网络基础-知识扩展

    网络的书籍:UNIX环境高级编程Linux网络编程 ip是不是唯一的,是不是可变的 1.全网搜索企业级的路由器 交...

  • 计算机网络基础-知识扩展

    网络的书籍:UNIX环境高级编程Linux网络编程 ip是不是唯一的,是不是可变的 1.全网搜索企业级的路由器 交...

  • 【Python入门】39.网络编程

    摘要:网络编程的概念介绍;TCP/IP协议的基本介绍;介绍Python的socket库,在Python中进行TCP...

  • Java学习路线?

    基础知识 编程语言: Java Python 基本算法 基本网络知识:TCP/IP HTTP/HTTPS 基本的...

  • 网络层(七)路由算法和路由协议

    1.什么是路由?“路由是从源主机到目的主机的路径”,这个说法其实不准确。发出IP分组的源主机,会将IP分组送到该主...

网友评论

      本文标题:Python网络编程3--实现IP源站路由

      本文链接:https://www.haomeiwen.com/subject/vmszvltx.html