DHCP协议及Python脚本模拟测试

如今几乎我们身边所有的设备都已经支持互联网通信功能,都知道要在互联网中通信需要给设备分配IP地址,目前IP地址的设置方法主要有以下两种:

  • 静态设置IP
  • 动态设置IP

DHCP动态主机配置协议

DHCP(Dynamic Host Configuration Protocol)是工作在应用层的一个局域网协议,其作用是用来给内部网络分配IP地址。

DHCP工作流程

我们把一个网络中的某一台设备叫做DHCP服务器,它有一个地址池,里边存放着可以使用的IP地址;把需要IP地址的设备叫做DHCP客户端。
(1)DHCP客户端在局域网内广播一个DHCP Discover数据包(带着自己的MAC地址),目的是发现能够给它提供IP地址的DHCP服务器。
(2)DHCP服务器收到DHCP Discover数据包后向请求的DHCP服务端发送一个DHCP Offer数据包(包含分给请求者的IP地址)。
(3)DHCP客户端接收到DHCP Offer数据包后(同时接收到来自多个DHCP服务器的DHCP Offer只处理第一个),广播一个DHCP Request数据包,声明自己所用的DHCP服务端和IP地址是哪一个。
(4)DHCP服务器收到DHCP Request数据包后,判断数据包中的IP地址是不是和自己的IP地址相同,如果相同则发送一个DHCP ACK数据包。
关于DHCP的具体内容可以参考DHCP详解

用Python程序模拟DHCP客户端

DHCP协议数据包并不能单独在网络中传播,需要其他各层协议的配合,毕竟互联网协议是分工明确相互合作的嘛。其次,DHCP协议的前身是引导协议(BootStrap Protocol),BOOTP协议为连接道网络中的设备自动分配IP地址,但是后来被更加复杂、更加强大的DHCP取代,然而在一些工具中,比如Wireshark和Scapy中还是将DHCP和BOOTP拆成两部分来看。
如图:在这里插入图片描述字段含义可以参考:
在这里插入图片描述
在构造数据包之前,我们需要了解一下DHCP数据包的相关知识:
- DHCP协议采用UDP协议
- DHCP客户端发送请求是到服务端的67端口,而响应消息是发给客户端的68端口。

DHCP Discovery数据包的构造

# DHCP Discover包
import binascii
from random import randint
from scapy.all import *
from scapy.layers.dhcp import BOOTP, DHCP
from scapy.layers.inet import *

client_mac = str(RandMAC())        # 本机已经有IP了,为了模拟,所以我们随机产生一个MAC地址,并转为str类型
client_mac_id = binascii.unhexlify(client_mac.replace(':', ''))
print("client_mac: " + client_mac)
client_xid = randint(1, 9999999)    # DHCP发起一次请求是选择的一个随机数
DHCP_Discover = DHCP(options=[("message-type", "discover"), "end"])
BOOTP_Discover = BOOTP(chaddr=client_mac_id, xid=client_xid)
UDP_Discover = UDP(sport=68, dport=67)
IP_Discover = IP(src='0.0.0.0', dst='255.255.255.255')  # 此时没有IP,故IP=‘0.0.0.0’
Ether_Discover = Ether(src=client_mac, dst='ff:ff:ff:ff:ff:ff')
Discover = Ether_Discover / IP_Discover / UDP_Discover / BOOTP_Discover / DHCP_Discover
ans = srp(Discover, iface='以太网')
print("\n\n\nSending DHCPDISCOVER on " + " 以太网")
print(ans.summary())

wireshark捕获到我们发出去的数据包如下![在这里插入图片描述](https://img-blog.csdnimg.cn/20210416181122223.png?x-oss-process=image/watermark,type_Zm在这里插入图片描述

DHCP Request数据包的构造

Ether_Request = Ether(src=pkt[Ether].dst, dst='ff:ff:ff:ff:ff:ff')
IP_Request = IP(src='0:0:0:0', dst='255.255.255.255')
UDP_Request = UDP(sport=68, dport=67)
BOOTP_Request = BOOTP(chaddr=pkt[BOOTP].chaddr,xid=pkt[BOOTP].xid)
DHCP_Request = DHCP(options=[("message-type","request"),("server_id",pkt[DHCP].options[1][1]),("requested_addr",pkt[BOOTP].yiaddr),"end"])
Request = Ether_Request / IP_Request / UDP_Request / BOOTP_Request /DHCP_Request

完整代码如下(我这里用的是kali linux所以网卡为wlan0):

from scapy.all import *

def detect_dhcp(pkt):
    if DHCP in pkt:
        if pkt[DHCP].options[0][1] == 2:
            Ether_Request = Ether(src=pkt[Ether].dst, dst='ff:ff:ff:ff:ff:ff')
            IP_Request = IP(src='0:0:0:0', dst='255.255.255.255')
            UDP_Request = UDP(sport=68, dport=67)
            BOOTP_Request = BOOTP(chaddr=pkt[BOOTP].chaddr,xid=pkt[BOOTP].xid)
            DHCP_Request = DHCP(options=[("message-type","request"),("server_id",pkt[DHCP].options[1][1]),("requested_addr",pkt[BOOTP].yiaddr),"end"])
            Request = Ether_Request / IP_Request / UDP_Request / BOOTP_Request /DHCP_Request
            sendp(Request,iface='wlan0')
            print(pkt[BOOTP].yiaddr + "正在分配")
        if pkt[DHCP].options[0][1] == 5:
            print(pkt[BOOTP].yiaddr + "已经分配")



sniff(filter='src port 67',iface='wlan0',prn=detect_dhcp,count=10)

DHCP模拟

#!/usr/bin/python3
from scapy.all import *
import binascii
import _thread
import time
# 为线程定义一个函数
def sent_discover( threadName, delay):
    while(1):
        xid_random = random.randint(1, 900000000)
        mac_random = str(RandMAC())
        client_mac_id = binascii.unhexlify(mac_random.replace(':', ''))
        dhcp_discover = Ether(src=mac_random, dst="ff:ff:ff:ff:ff:ff") / IP(src="0.0.0.0", dst="255.255.255.255") / UDP(sport=68, dport=67) / BOOTP(chaddr=client_mac_id, xid=xid_random) / DHCP(options=[("message-type", "discover"), "end"])
        sendp(dhcp_discover, iface='wlan0')
        time.sleep(delay)
def sniff_discover(threadName, delay):
    def detect_dhcp(pkt):
        if DHCP in pkt:
            if pkt[DHCP].options[0][1] == 2:
                Ether_Request = Ether(src=pkt[Ether].dst, dst="ff:ff:ff:ff:ff:ff")
                IP_Request = IP(src="0.0.0.0", dst="255.255.255.255")
                UDP_Request = UDP(sport=68, dport=67)
                BOOTP_Request = BOOTP(chaddr=pkt[BOOTP].chaddr, xid=pkt[BOOTP].xid)
                DHCP_Request = DHCP(options=[("message-type", 'request'), ("server_id", pkt[DHCP].options[1][1]),
                                             ("requested_addr", pkt[BOOTP].yiaddr), "end"])
                Request = Ether_Request / IP_Request / UDP_Request / BOOTP_Request / DHCP_Request
                sendp(Request, iface='wlan0')
                print(pkt[BOOTP].yiaddr + "正在分配")
            if pkt[DHCP].options[0][1] == 5:
                print(pkt[BOOTP].yiaddr + "已经分配")
    sniff(filter="src port 67", iface='wlan0', prn=detect_dhcp)
try:
    _thread.start_new_thread( sniff_discover, ("Thread-1", 0, ) )
    _thread.start_new_thread( sent_discover, ("Thread-2", 10, ) )
except:
    print ("Error: 无法启动线程")
while 1:
    pass
  • 4
    点赞
  • 24
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
说明: 1, 暂未实现重传机制, 所以若抓包无响应, 请尝试停止后重发. 2, dhcp状态显示采用1s定时器刷新, 所以状态显示可能存在延时的情况; 3, xcap通过pcap导入报文会有部分字段自动变化, 且导入的报文DHCP数据部分无法正常解析, 建议通过新建的方式解决; 4, 添加报文格式举例: 1,2 说明: 1表示报文组1, 选中报文组后, 在状态栏会显示报文组的索引, 2表示第三个报文, 即索引为3的报文. 版本记录: V1.0.1(基础版本) 1, 支持连接xcap并读取报文功能; 2, 支持刷新按钮自动更新报文功能; 3, 支持选择网卡功能; 4, 支持通过pcap文件打开报文功能(已废弃); 5, 支持指定服务器交互; 6, 支持dhcp交互状态显示; 7, 支持输入框通过正则表达式限制输入字符; 8, 支持选择特定报文操作; V1.0.2 1, 将状态修改为自动显示, 即动态识别报文类型并显示结果; 2, 解决解析option字段, 若字段中存在多个value时存在丢失的问题; 3, 增加鼠标点击状态显示气泡信息; 4, 增加隔行显示不同颜色; V1.0.3 1, 修改dhcp的状态机, 之前的版本是收到报文则发送request, 之后收到报文则认为收到ack. 现修改为只有收到offer报文才发送request报文 , 收到ack报文才结束. 2, 增加dhcpv6功能; 3, 优化代码; V1.0.4 1, 修改request报文由于校验和和报文长度未初始化导致构造错误的问题 V1.0.5 1, 增加服务器地址的气泡提示; 2, 增加自动填充的气泡提示; 3, 添加的报文默认为选中状态; 4, 选择网卡下拉框中将虚拟网卡排放靠后; 5, 关闭程序时自动保存设置; V1.0.6 1, 优化代码, 将字段设置使用统一的函数处理; 2, 状态气泡显示格式化; 3, 双击表格表头实现全选和反选; 4, 增加renew(50%), rebind(87.5%)和release的自动发送功能; 5, 增加手动释放按钮和实现; 6, 增加部分打印信息用于调试; 暂未实现报文重传机制, 计划下一个版本实现 V1.0.7 1, 实现discover/solicit报文自动重传机制 2, renew, rebind以及release修改为手动发送 3, 解决报文发送错乱问题 4, 增加decline报文的发送 5, 解决设备无故发送discover报文问题 问题解决: 1, 停止后再次发送数据会出现数据错乱 分析: 停止客户端的时候, 删除过滤器是通过callback函数删除的, 这里应该是通过filter来进行删除. self.widget.sniff.del_filter(self.callback)修改为 self.widget.sniff.del_filter(self.filter) 2, 设备无故发送discover问题 分析: 由于发送discover报文使用的定时器, 定时器是通过判断当前的direction来确定是否重传的, 而当定时器老化时, 可能正好收到报文导 致direction被修改, 所以导致错误的发送discover报文的问题. 将接收逻辑修改为重传时判断当前状态是否为discover报文, 若是则重传, 否则不重传. V1.0.8 1, 增加inform实现 V1.0.9 1, 增加报文五元组的源mac地址和xid的气泡显示; 2, 增加步长和报文限制功能; 问题解决: 1, 修改ipv6报文添加失败的问题. 由于ipv4报文为xid, ipv6报文为trid, 需要区分处理. V1.0.10 1, 在发送dhcpv6报文之前, 先发送na报文触发服务器学习nd消息. V1.0.11 1, 增加发送solicit/request前, 自动响应ns报文. 自动响应ns报文的目标地址为solicit/request报文源mac地址生成的ipv6地址 2, 实现dhcpv6的renew续约功能. 3, 解决ipv6地址转换格式化不正确, 导致无法响应ns报文问题. 4, 增加日志输出到dhcp.log文件. V1.0.12 1, 增加dhcpv6的续约功能, 通过renew和rebind实现续约, 增加release、decline报文的实现; V1.0.13 1, 解决服务器无法设置ipv6地址的问题. 之前的输入框只允许输入数字和., 修改为运行输入数字.:和a-f 2, 解决多个客户端时, 若选中其中的部分客户端发送时报错. 由于客户端采用的是列表中包含元组的形式, 即[(row, [client1, client2])], 这样实际客户端无法直接通过row索引到clients, 导致列表读 取时溢出. 譬如有1、2、3三行数据, 这里只选中了第三行, 限制为1, 那么如果点击发送, 则clients = [(row, [client1]], 此时clients[2] 就会溢出. 所以这里讲clients修改为字典, 即通过row来索引客户端client = {3: [client1]} 3, 将数据发送放到线程中, 规避模拟大量客户端时界面假死的问题. 4, 当客户端限制小于等于50, 则气泡显示trid和ip地址信息. 当大于50, 则气泡显示获取ip地址的数量. V1.0.14 1, 解决监听报文使用的网卡不正确问题. V1.0.15 1, 解决dhcpv6的响应报文的IANA中包含Status code选项导致程序无法解析的问题. 兼容性处理, 即option为IAAddress时按照IAAddress解析, 当option为status code时按照Status code解析 V1.0.16 1, 解决DHCPv6的client_id的duid处理, 支持任意格式的duid.
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值