icmp详解
Ping 是一个常用的网络诊断工具,其工作原理基于 ICMP(Internet Control Message Protocol,互联网控制消息协议)
- 发送 ICMP Echo Request:
- ping 命令会向目标主机发送 ICMP Echo Request(类型 8)数据包。
- 这个数据包包含一个标识符、序列号和可选的数据负载。
- 等待 ICMP Echo Reply:
- 如果目标主机正常运行且网络连接正常,它会回复一个 ICMP Echo Reply(类型 0)数据包。
- 回复包含与请求包相同的标识符和序列号。
- 计算往返时间(RTT):
- ping 程序记录发送请求和接收回复的时间差,这就是往返时间(Round-Trip Time, RTT)。
- 重复过程:
- 通常,ping 会发送多个请求(默认情况下通常是 4 个),以获得更可靠的结果。
- 统计和报告:
- ping 最后会显示统计信息,如平均 RTT、最小和最大 RTT、丢包率等。
主要用途:
- 检测网络连通性:确认目标主机是否可达。
- 测量网络延迟:通过 RTT 了解网络响应速度。
- 诊断网络问题:如丢包、路由问题等。
ICMP 报文格式
ICMP头部通常是8字节长,紧跟在IP头部之后。
0 1 2 3
0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
| Type | Code | Checksum |
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
| Identifier | Sequence Number |
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
具体到每个字段:
- Type (8 bits-uint8_t):
- 表示ICMP消息的类型。
- 类型 0:回显应答(Echo Reply)
- 类型 3:目标不可达(Destination Unreachable)
- 类型 5:重定向(Redirect)
- 类型 8:回显请求(Echo Request)
- 类型 11:超时(Time Exceeded)
- Code (8 bits):
- 进一步细分ICMP消息的类型。
- 对于Echo请求和回复,这个值通常为0。
- Checksum (16 bits):
- 用于检测ICMP报文在传输过程中是否发生错误。
- 计算方法包括ICMP头和数据。
- Identifier (16 bits):
- 用于匹配请求和回复。
- 通常是发送进程的ID。
- Sequence Number (16 bits):
- 用于匹配请求和回复。
- 每发送一个新的Echo请求,这个值就会增加。
因此实际上可以简单的定义一个结构体来表示ICMP头部:
1
2
3
4
5
6
7
struct icmp_header {
uint8_t type;
uint8_t code;
uint16_t checksum;
uint16_t identifier;
uint16_t sequence_number;
};
在Python中,我们可以使用struct模块来打包和解包这些字段:
1
2
3
4
5
6
7
8
9
10
import struct
# 打包ICMP头
icmp_header = struct.pack('!BBHHH',
8, # Type (8 for Echo Request)
0, # Code
0, # Checksum (初始值为0)
1234, # Identifier
1 # Sequence Number
)
也能注意到,ping的时候其实没有指定端口号,这是因为ICMP是一个基于IP层的协议,不需要端口号。
数据包按照普通的 IP 路由规则传输到目标主机,路由器根据 IP 地址进行转发,不关心 ICMP 的内容。
当目标主机收到 IP 数据包时,它会检查 IP 头部的协议字段,看到协议字段为 1,操作系统知道这是 ICMP 数据包。然后,数据包会被传递给操作系统的 ICMP 处理模块。
因此,配合socket_raw, 其实可以实现一个简单的ping client
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
import os
import struct
import time
import select
import socket
ICMP_ECHO_REQUEST = 8
def checksum(packet):
if len(packet) % 2 != 0:
packet += b'\0'
res = sum(struct.unpack("!H", packet[i:i+2])[0] for i in range(0, len(packet), 2))
res = (res >> 16) + (res & 0xffff)
res += res >> 16
return (~res) & 0xffff
def create_packet(id, seq = 1):
"""Create a new echo request packet based on the given id."""
header = struct.pack('!BBHHH', ICMP_ECHO_REQUEST, 0, 0, id, seq)
data = b'Q' * 192
my_checksum = checksum(header + data)
header = struct.pack('!BBHHH', ICMP_ECHO_REQUEST, 0, my_checksum, id, seq)
return header + data
def ping(host):
"""
Returns either the delay (in seconds) or None on timeout.
"""
dest_addr = socket.gethostbyname(host)
my_socket = socket.socket(socket.AF_INET, socket.SOCK_RAW, socket.IPPROTO_ICMP)
# my_socket.setsockopt(socket.SOL_SOCKET, socket.SO_RCVBUF, 65536)
my_id = os.getpid() & 0xFFFF
packet = create_packet(my_id)
my_socket.sendto(packet, (dest_addr, 1)) # Port number is irrelevant for ICMP
start_time = time.time()
while True:
ready = select.select([my_socket], [], [], 4) # Timeout of 4 seconds
if not ready[0]: # Timeout
return None
received_packet, addr = my_socket.recvfrom(1024)
icmp_header = received_packet[20:28]
type, code, checksum, packet_id, sequence = struct.unpack('BBHHH', icmp_header)
if packet_id == my_id:
return time.time() - start_time
my_socket.close()
def main():
host = '127.0.0.1' # input("Enter the host to ping: ")
print(f"Pinging {host}")
for i in range(4): # Send 4 ping requests
delay = ping(host)
if delay is None:
print("Request timed out.")
else:
delay *= 1000 # Convert to milliseconds
print(f"Received reply in {delay:.2f}ms")
time.sleep(1)
if __name__ == "__main__":
main()
python的struct
Python的struct
模块用于在字节串和Python数据类型之间进行转换。它非常有用,特别是在处理二进制数据、网络协议或文件格式时。让我详细解释一下如何使用struct
模块:
- 基本用法:
1
2
3
4
5
6
7
import struct
# 打包(将Python数据类型转换为字节串)
packed = struct.pack(format, v1, v2, ...)
# 解包(将字节串转换为Python数据类型)
unpacked = struct.unpack(format, buffer)
- 格式字符串:
格式字符串指定了如何解释字节。一些常用的格式字符包括:
B
: 无符号字符(1字节)H
: 无符号短整型(2字节)I
: 无符号整型(4字节)Q
: 无符号长长整型(8字节)h
: 有符号短整型(2字节)i
: 有符号整型(4字节)q
: 有符号长长整型(8字节)f
: 单精度浮点数(4字节)d
: 双精度浮点数(8字节)s
: 字符串(字节数组)
- 字节顺序:
<
: 小端序>
: 大端序!
: 网络字节序(大端序)=
: 本机字节序
- 示例:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
import struct
# 打包
# 格式:!BBHIH 表示网络字节序,依次为:无符号字符、无符号字符、无符号短整型、无符号整型、无符号短整型
packed = struct.pack('!BBHIH', 8, 0, 1000, 12345, 1)
print(packed) # b'\x08\x00\x03\xe8\x00\x0009\x00\x01'
# 解包
unpacked = struct.unpack('!BBHIH', packed)
print(unpacked) # (8, 0, 1000, 12345, 1)
# 字符串处理
name = "Alice"
# 表示一个无符号整数(4字节), 5s: 表示一个5字节的字符串(字节串)
packed_name = struct.pack('!I5s', len(name), name.encode('utf-8'))
print(packed_name) # b'\x00\x00\x00\x05Alice'
length, name = struct.unpack('!I5s', packed_name)
print(length, name.decode('utf-8')) # 5 Alice
# 浮点数
packed_float = struct.pack('!f', 3.14)
print(packed_float) # b'@I\x0f\xd0'
unpacked_float = struct.unpack('!f', packed_float)
print(unpacked_float) # (3.140000104904175,)
- 在网络编程中的应用(以ICMP为例):
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
import struct
def create_icmp_header(id, seq):
# 创建ICMP头(类型8表示Echo请求,代码0)
header = struct.pack('!BBHHH', 8, 0, 0, id, seq)
# 计算校验和
checksum = calculate_checksum(header)
# 重新打包,这次包含计算好的校验和
header = struct.pack('!BBHHH', 8, 0, checksum, id, seq)
return header
def calculate_checksum(data):
# 校验和计算(简化版)
s = 0
for i in range(0, len(data), 2):
w = (data[i] << 8) + (data[i+1] if i+1 < len(data) else 0)
s = s + w
s = (s >> 16) + (s & 0xffff)
s = ~s & 0xffff
return s
# 使用示例
icmp_header = create_icmp_header(1234, 1)
print(icmp_header)
# 解析ICMP头
type, code, checksum, id, seq = struct.unpack('!BBHHH', icmp_header)
print(f"Type: {type}, Code: {code}, Checksum: {checksum}, ID: {id}, Sequence: {seq}")
ip协议
/*
IPv4 Header Format
0 1 2 3
0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|Version| IHL |Type of Service| Total Length |
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
| Identification |Flags| Fragment Offset |
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
| Time to Live | Protocol | Header Checksum |
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
| Source IP Address |
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
| Destination IP Address |
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
| Options | Padding |
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
Version: 4 bits
IHL: 4 bits (Internet Header Length)
Type of Service:8 bits
Total Length: 16 bits
Identification: 16 bits
Flags: 3 bits
Fragment Offset:13 bits
Time to Live: 8 bits
Protocol: 8 bits
Header Checksum:16 bits
Source IP: 32 bits
Destination IP: 32 bits
Options: variable length, optional
Padding: variable length, used to ensure 32-bit alignment
*/
IPv4头部的详细字段描述:
- 版本(Version): 4位
- 表示IP协议的版本,IPv4为4
- 首部长度(Internet Header Length, IHL): 4位
- 表示IP头部的长度,以32位字(4字节)为单位
- 最小值为5(20字节),最大值为15(60字节)
- 服务类型(Type of Service): 8位
- 用于指定服务质量
- 总长度(Total Length): 16位
- 整个IP数据报的长度,包括头部和数据,以字节为单位
- 最大值为65535字节
- 标识(Identification): 16位
- 用于标识数据报片段,同一数据报的所有片段该值相同
- 标志(Flags): 3位
- 包括保留位、不分片(DF)和更多片段(MF)
- 片偏移(Fragment Offset): 13位
- 表示该片段在原始数据报中的位置,以8字节为单位
- 生存时间(Time to Live, TTL): 8位
- 限制数据报在网络中的生存时间,每经过一个路由器就减1
- 协议(Protocol): 8位
- 指示上层协议,如TCP(6)或UDP(17)
- 首部校验和(Header Checksum): 16位
- 用于检验IP头部的完整性
源IP地址(Source IP Address): 32位
- 目的IP地址(Destination IP Address): 32位
需要注意的是,虽然IP头部的标准长度是20字节(如图所示),但它可以包含额外的选项字段,最长可达60字节。这就是为什么我们需要”首部长度”(IHL)字段来指明实际的头部长度。
这种结构化的格式使得IP协议能够高效地在网络中传输数据,同时提供了必要的信息用于路由、分片和重组等网络操作。
OSI Model TCP/IP Model
+-----------------------+ +------------------+
| 7. Application | | |
|-----------------------| | |
| 6. Presentation | | Application |
|-----------------------| | |
| 5. Session | | |
|-----------------------| +------------------+
| 4. Transport | | Transport |
|-----------------------| +------------------+
| 3. Network | | Internet |
|-----------------------| +------------------+
| 2. Data Link | | |
|-----------------------| | Network Access |
| 1. Physical | | |
+-----------------------+ +------------------+
OSI Model Layers:
7. Application: High-level APIs, resource sharing, remote file access
6. Presentation: Data translation, encryption, compression
5. Session: Interhost communication, managing sessions
4. Transport: End-to-end connections, reliability, flow control
3. Network: Path determination and logical addressing
2. Data Link: Physical addressing, error detection and correction
1. Physical: Media, signal and binary transmission
TCP/IP Model Layers:
4. Application: Combines OSI layers 5-7, handles high-level protocols
3. Transport: Corresponds to OSI layer 4, uses TCP and UDP
2. Internet: Corresponds to OSI layer 3, uses IP
1. Network Access:Combines OSI layers 1-2, handles physical and data link
This post is licensed under CC BY 4.0 by the author.