ICMP是網(wǎng)絡(luò)協(xié)議的一部分,屬于網(wǎng)絡(luò)層協(xié)議(OSI模型中的第3層)。它的作用是在IP網(wǎng)絡(luò)中傳輸控制消息,以響應(yīng)網(wǎng)絡(luò)故障或傳輸異常。在ICMP消息的流程中,碰巧也會(huì)涉及到端口號(hào)。在本文中,我們將從多個(gè)方面深入探討icmp端口號(hào)是多少。
一、ICMP端口號(hào)介紹
ICMP消息是基于IP協(xié)議的,其消息格式中并沒(méi)有端口號(hào)這個(gè)字段。因此,我們?cè)趥鬏擨CMP消息時(shí),并不需要設(shè)置端口號(hào)。然而,在收到ICMP消息時(shí),是需要通過(guò)端口號(hào)將消息傳遞給相應(yīng)的進(jìn)程的。
ICMP消息是通過(guò)IP數(shù)據(jù)包傳輸?shù)?,在傳輸時(shí)使用數(shù)據(jù)包的類(lèi)型碼來(lái)確定如何處理數(shù)據(jù)包。在很多情況下,ICMP消息可以看作一個(gè)特定類(lèi)型的數(shù)據(jù)包。因此,在處理ICMP消息時(shí),使用的就是與處理對(duì)應(yīng)類(lèi)型的IP數(shù)據(jù)包時(shí)相同的端口號(hào)。
二、ICMP端口號(hào)范圍
ICMP消息所使用的端口號(hào)分為兩個(gè)范圍:系統(tǒng)保留端口和動(dòng)態(tài)私有端口。
系統(tǒng)保留端口由IANA(Internet Assigned Numbers Authority)分配,范圍是0~1023,主要用于標(biāo)準(zhǔn)化的服務(wù)和應(yīng)用程序。一般而言,這些端口必須由特權(quán)進(jìn)程(超級(jí)用戶或管理員)打開(kāi)。
動(dòng)態(tài)私有端口由客戶端或服務(wù)器分配,范圍是49152~65535。在傳輸ICMP消息時(shí),大多數(shù)情況下會(huì)使用這個(gè)范圍內(nèi)的端口號(hào)。
三、ICMP端口號(hào)的使用
在處理ICMP消息時(shí),使用的端口號(hào)并不是與ICMP消息綁定的。相反,它是與進(jìn)程綁定的。這就是為什么需要將收到的ICMP消息傳遞到相應(yīng)進(jìn)程的原因。
在大多數(shù)情況下,系統(tǒng)會(huì)使用動(dòng)態(tài)分配的端口來(lái)發(fā)送ICMP消息。這個(gè)過(guò)程通常是由內(nèi)核處理的,因此并不需要我們手動(dòng)設(shè)置端口號(hào)。我們可以通過(guò)捕獲ICMP消息,然后查看源IP地址與端口號(hào)來(lái)確認(rèn)消息來(lái)源。同時(shí),我們也可以使用ICMP的類(lèi)型碼和代碼來(lái)判斷消息的類(lèi)型,然后根據(jù)消息類(lèi)型來(lái)執(zhí)行相應(yīng)的操作。
四、代碼示例
import os import socket import struct import time # 構(gòu)造ICMP消息 def icmp_echo_request(): # 定義ICMP類(lèi)型和代碼 icmp_type = 8 # ICMP Echo Request icmp_code = 0 # must be zero # 構(gòu)造ICMP消息頭部 icmp_header = struct.pack('!BBHHH', icmp_type, icmp_code, 0, 0, 0) # 構(gòu)造ICMP數(shù)據(jù) icmp_data = b'hello, world!' # 計(jì)算校驗(yàn)和 checksum = icmp_checksum(icmp_header + icmp_data) # 重新構(gòu)造ICMP頭部,將計(jì)算出的校驗(yàn)和寫(xiě)入ICMP頭部 icmp_header = struct.pack('!BBHHH', icmp_type, icmp_code, checksum, 0, 0) # 最終構(gòu)造ICMP消息 icmp_msg = icmp_header + icmp_data return icmp_msg # 計(jì)算ICMP校驗(yàn)和 def icmp_checksum(data): # data需要是16位的整數(shù)倍,所以我們需要將數(shù)據(jù)進(jìn)行填充 data_len = len(data) if (data_len % 2) != 0: data += b'\x00' # 計(jì)算校驗(yàn)和 sum = 0 for i in range(0, data_len, 2): word = (data[i]<<8) + data[i+1] sum += word sum = (sum>>16) + (sum & 0xffff) sum += (sum>>16) checksum = ~sum & 0xffff return checksum # 主函數(shù) def main(): # 構(gòu)造ICMP消息 icmp_msg = icmp_echo_request() # 構(gòu)造IP數(shù)據(jù)報(bào) dest_addr = socket.gethostbyname('www.baidu.com') # 創(chuàng)建socket對(duì)象 icmp_socket = socket.socket(socket.AF_INET, socket.SOCK_RAW, socket.IPPROTO_ICMP) # 設(shè)置超時(shí)時(shí)間 icmp_socket.settimeout(5) # 發(fā)送ICMP消息 icmp_socket.sendto(icmp_msg, (dest_addr, 0)) # 接收ICMP消息 try: recv_data, addr = icmp_socket.recvfrom(1024) print('recv from', addr) # 解析ICMP消息,獲取信息類(lèi)型碼和代碼 icmp_type, icmp_code = struct.unpack('BB', recv_data[20:22]) print('type:', icmp_type, 'code:', icmp_code) except socket.timeout: print('timeout') icmp_socket.close() if __name__ == '__main__': main()