套接字不会绑定:没有这样的设备

use*_*868 16 python sockets

所以我有这条Python 3代码:

import socket
s = socket.socket(socket.AF_PACKET, socket.SOCK_RAW)
s.bind(('eth0', 0))
s.send(eth_packet)
Run Code Online (Sandbox Code Playgroud)

此代码适用于我的Raspberry Pi,但不适用于我的外部服务器.当我尝试在我的外部服务器上运行它时,我得到:

# sudo python3 test.py
s.send(eth_packet)
socket.error: [Errno 19] No such device
Run Code Online (Sandbox Code Playgroud)

我检查了网络接口输出(通过python脚本):外部服务器(debian):

['lo [index=1, IPv4=127.0.0.1, IPv6=::1]', 'eth0:0 [index=2, IPv4=xxxxx, IPv6=None]', 'eth0 [index=2, IPv4=yyyyyy, IPv6=zzzzzzz]']
Run Code Online (Sandbox Code Playgroud)

树莓派:

['lo [index=1, IPv4=127.0.0.1, IPv6=None]', 'eth0 [index=2, IPv4=rrrrr, IPv6=None]']
Run Code Online (Sandbox Code Playgroud)

有人可以解释发生了什么吗?我只是想发送一个手工制作的消息,但这个错误一直困扰着我,这可能是我服务器驱动程序的问题吗?这与ifconfig的结果相同.

编辑

好的,我在这个例子中使用了strace:

#!/usr/bin/env python3

import socket
import binascii
import struct

test= '000a959d6816'
packet= struct.pack("!6s", binascii.unhexlify(bytes(test, 'UTF-8')))
s = socket.socket(socket.AF_PACKET, socket.SOCK_RAW)
s.bind(('eth0', 0))
s.send(packet)
Run Code Online (Sandbox Code Playgroud)

这是关于strace的重要部分:

socket(PF_PACKET, SOCK_RAW, 0)          = 3
ioctl(3, SIOCGIFINDEX, {ifr_name="eth0", ifr_index=2}) = 0
bind(3, {sa_family=AF_PACKET, proto=0000, if2, pkttype=PACKET_HOST, addr(0)={0, }, 20) = 0
sendto(3, "\0\n\225\235h\26", 6, 0, NULL, 0) = -1 ENXIO (No such device or address)
open("test.py", O_RDONLY)               = 4
fstat(4, {st_mode=S_IFREG|0644, st_size=247, ...}) = 0
ioctl(4, SNDCTL_TMR_TIMEBASE or TCGETS, 0x7fff86c5f090) = -1 ENOTTY (Inappropriate ioctl for device)
fstat(4, {st_mode=S_IFREG|0644, st_size=247, ...}) = 0
lseek(4, 0, SEEK_CUR)                   = 0
dup(4)                                  = 5
fcntl(5, F_GETFL)                       = 0x8000 (flags O_RDONLY|O_LARGEFILE)
fstat(5, {st_mode=S_IFREG|0644, st_size=247, ...}) = 0
mmap(NULL, 4096, PROT_READ|PROT_WRITE, MAP_PRIVATE|MAP_ANONYMOUS, -1, 0) = 0x7fc1251c2000
lseek(5, 0, SEEK_CUR)                   = 0
read(5, "#!/usr/bin/env python3\n\nimport s"..., 4096) = 247
close(5)                                = 0
munmap(0x7fc1251c2000, 4096)            = 0
lseek(4, 0, SEEK_SET)                   = 0
lseek(4, 0, SEEK_CUR)                   = 0
read(4, "#!/usr/bin/env python3\n\nimport s"..., 4096) = 247
close(4)                                = 0
write(2, "Traceback (most recent call last"..., 143Traceback (most recent call last):
  File "test.py", line 11, in <module>
    s.send(packet)
socket.error: [Errno 6] No such device or address
) = 143
rt_sigaction(SIGINT, {SIG_DFL, [], SA_RESTORER, 0x7fc1264050a0}, {0x428787, [], SA_RESTORER, 0x7fc1264050a0}, 8) = 0
close(3)                                = 0
Run Code Online (Sandbox Code Playgroud)

Fig*_*gus 5

在接口上绑定带有系列PACKET的RAW套接字时,需要一个包含2个对象的元组:

(interfaceName, protoNumber)
Run Code Online (Sandbox Code Playgroud)

或5个对象:

(interfaceName, protoNumber, pkttype, hatype, haddr)
Run Code Online (Sandbox Code Playgroud)

您在protoNumber中指定0,但系统中可能不存在protoNumber 0.

有关数据包系列的文档:数据包(7)

sll_protocol是包含文件中定义的网络字节顺序的标准以太网协议类型.

尝试在linux/if_ether.h中找到正确的协议号.


小智 -3

这是一个可以发送手工消息的客户端。变量 ip 和 port 是服务器的端口和 IP 地址,以下程序中的值是示例

import socket
ip = "127.0.0.1"
port = 447
s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
s.connect((ip, port))
msg = ("hello ex. message")
s.send(msg.encode('ascii'))
rep = s.recv(1024)
rep1 = (rep.decode('ascii'))
print rep1
Run Code Online (Sandbox Code Playgroud)

服务器是

import socket
ip = "127.0.0.1"
port = 447
s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
s.bind((ip, port))
s.listen(15)
con, addr = s.accept()
msg = con.recv(1024)
msg1 = (msg.decode('ascii'))
print (msg1)
rep = ("got message")
con.send(rep.encode('ascii'))
Run Code Online (Sandbox Code Playgroud)

我希望这个答案有帮助,这就是您所寻找的。

  • 这个答案与上面的问题无关,上面的问题是关于原始套接字的。 (4认同)