import socket import threading import protocol import sys LOCAL_PORT = 9000 TIMEOUT = 2 # 秒 class StopWaitChat: def __init__(self, local_port, peer_addr): # IPv6 UDP socket self.sock = socket.socket(socket.AF_INET6, socket.SOCK_DGRAM) # 绑定本地 IPv6 任意地址 self.sock.bind(("::", local_port)) # 设置 socket 超时(用于停等协议) self.sock.settimeout(TIMEOUT) self.peer_addr = peer_addr # 停等协议序号 self.send_seq = 0 self.recv_seq = 0 def sender(self): """ 发送线程(停等协议) """ while True: try: msg = input(">> ").encode() except EOFError: break pkt = protocol.make_packet( self.send_seq, protocol.TYPE_DATA, msg ) # 停等:一直发,直到收到正确 ACK while True: self.sock.sendto(pkt, self.peer_addr) try: data, _ = self.sock.recvfrom(1024) seq, ptype, _ = protocol.parse_packet(data) if ptype == protocol.TYPE_ACK and seq == self.send_seq: # 正确 ACK self.send_seq ^= 1 break except socket.timeout: print("[timeout] retransmit") def receiver(self): """ 接收线程(必须捕获 timeout) """ while True: try: data, addr = self.sock.recvfrom(1024) except socket.timeout: continue seq, ptype, payload = protocol.parse_packet(data) if ptype == protocol.TYPE_DATA: if seq == self.recv_seq: print("\npeer:", payload.decode()) self.recv_seq ^= 1 # 无论是否重复,都返回 ACK ack = protocol.make_packet(seq, protocol.TYPE_ACK) self.sock.sendto(ack, addr) def main(): if len(sys.argv) != 3: print("用法: python3 chat.py <对方IPv6地址或域名> <对方端口>") sys.exit(1) peer_ip = sys.argv[1] peer_port = int(sys.argv[2]) chat = StopWaitChat( LOCAL_PORT, (peer_ip, peer_port) ) # 启动接收线程 recv_thread = threading.Thread( target=chat.receiver, daemon=True ) recv_thread.start() # 主线程负责发送 chat.sender() if __name__ == "__main__": main()