在Linux上使用python接收多播UDP数据报

时间:2013-03-07 17:42:00

标签: python linux sockets udp multicast

我有一个硬件设备,可以在我的网络上发送多播数据。我编写了一个python脚本来接收数据并打印出来。但是,我发现它只能在我的Windows XP PC上运行,并且无法在我的Ubuntu Linux 10.04 PC上运行。在Linux下,没有收到任何内容。它只围绕while循环,并且从未收到任何数据。我的代码发布在下面。你能看出为什么这在Linux下不起作用的原因吗?谢谢,拉布。

# Multicast client
# Adapted from: http://chaos.weblogs.us/archives/164
# on 05/03/2013

import socket

ANY = "0.0.0.0" 
MCAST_ADDR = "224.0.33.154"
MCAST_PORT = 31800

# Create a UDP socket
sock = socket.socket(socket.AF_INET, socket.SOCK_DGRAM, socket.IPPROTO_UDP)

# Allow multiple sockets to use the same PORT number
sock.setsockopt(socket.SOL_SOCKET,socket.SO_REUSEADDR,1)

# Bind to the port that we know will receive multicast data
sock.bind((ANY,MCAST_PORT))

# Tell the kernel that we are a multicast socket
sock.setsockopt(socket.IPPROTO_IP, socket.IP_MULTICAST_TTL, 255)

# Tell the kernel that we want to add ourselves to a multicast group
# The address for the multicast group is the third param
status = sock.setsockopt(socket.IPPROTO_IP,
socket.IP_ADD_MEMBERSHIP,
socket.inet_aton(MCAST_ADDR) + socket.inet_aton(ANY));

# setblocking(0) is equiv to settimeout(0.0) which means we poll the socket.
# But this will raise an error if recv() or send() can't immediately find or send data. 
sock.setblocking(0)

while 1:
    try:
        data, addr = sock.recvfrom(1024)
    except socket.error as e:
        pass
    else:
        print "From: ", addr
        print "Data: ", data

以下是我的Windows PC的一些示例输出:

From:  ('0.0.0.0', 31801)
Data:  EDCP

请注意,远程硬件设备没有IP地址,并且使用地址0.0.0.0

编辑:我现在发现这在我的Windows笔记本电脑上也不起作用。因此,它看起来并不像操作系统那样具体。此外,我尝试运行另一个脚本将组播数据发送到同一个多播地址和端口。我可以从一台PC发送,其他所有人都可以使用我的接收脚本正确接收。但是,只有我的一台Windows PC才能从相关的硬件设备接收数据。我想知道它是否与以太网适配器或它们的配置有关。这可能与硬件设备的IP地址为0.0.0.0并且这些以太网适配器和/或我的接收器脚本需要告知接收带有此地址的消息有关吗?在Linux PC上运行Wireshark可以看到来自硬件设备的数据。

2 个答案:

答案 0 :(得分:0)

尝试绑定到多播组地址:

sock.bind((MCAST_ADDR,MCAST_PORT))

此外,您不需要在接收器上设置多播TTL,也可以在发送器上设置,也可以选择。

答案 1 :(得分:0)

我在同样的问题上打了两天。 Wireshark看到了数据包,但我的代码却没有。各种来源的所谓“确定性”答案都不适合我。密钥来自https://serverfault.com/questions/163244/linux-kernel-not-passing-through-multicast-udp-packets

运行“ip maddr”显示类似于您的代码没有将多播地址添加到任何接口。我强迫它添加smcroute(参见上面的链接)。仍然没有快乐。数据包的源IP为172.22 ...我的接口是172.17 ...我在该网卡上添加了一个172.22地址。答对了!现在我的代码收到了数据包。

现在如何让程序在没有smcroute的情况下工作?我注释掉了setsockopt()调用。仍然有效。将多播地址与smcroute取消链接 - 失败。取消注释setsockopt()调用并用我的172.22地址替换“ANY”。成功了!

要点:

  1. 确保您在与传入数据包相同的网段上拥有IP。
  2. 在IP_ADD_MEMBERSHIP调用中使用该地址而不是INADDR_ANY。
  3. 如果你只有一个网卡,你可能不需要做2)。我有三个,必须这样做。

    如果它有密切关系,我使用的是Ubuntu 12.04。我不需要像其他人描述的那样更改任何默认的/etc/sysctl.conf设置。我试过了。他们没有帮助,所以我将它们重置为安装默认值。