将套接字绑定到具有未知 IP 的特定接口



我有一个有两个接口的主机。在我的特定情况下,我正在尝试使用 boost::asio::ip::multicast::join_group 加入多播组,这似乎仅在我使用包含本地地址的构造函数时才有效。但是,我事先不知道连接到将进行多播的远程主机的本地接口的 IP 地址。但是,我知道它将是eth1。当然,我可以使其可配置,但这似乎引入了一个无用的错误配置机会,看看必须如何为接口和我的应用程序配置相同的地址。

理想情况下,会有一种非常明显的方法可以从界面而不是我以某种方式错过的地址创建boost::asio::endpointboost::asio::address。或者,我当然对任何其他方法来推断一个接口 IP 同样满意,该接口 IP 既工作又能工作,有和没有提供 IP 的 DHCP 服务器。

有没有正确的方法可以做到,或者我应该相信用户永远不会摸索配置?

为了确保这不完全是XY问题,以下是我在测试加入多播组时使用的代码:

m_socket.open(boost_ip::udp::v4());
m_socket.bind(boost_ip::udp::endpoint(boost_ip::udp::v4(), listeningPort));
m_socket.set_option(boost::asio::ip::udp::socket::reuse_address(true));
m_socket.set_option(boost::asio::ip::multicast::join_group(
                  boost::asio::ip::address::from_string("225.x.x.10").to_v4(),  // remote
                  boost::asio::ip::address::from_string("192.x.x.3").to_v4())); // local

这确实有效,但是当我丢弃最后一行不依赖于当前本地地址(可能会在部署中更改)时,我不再收到任何数据包。

组播使用 IGMP 协议形成组播组。由于 IGMP 在网络层上运行,因此它需要加入组播组的端点的本地 IP 地址。

应用程序应等待指示已将 IP 地址分配给以太网接口的事件,然后调用 join_group 方法加入多播组。

我自己想出了问题的答案。这对我有用,但如果有人有更直接的解决方案,我会把这个问题留给开放。

我决定使用 <ifaddrs.h> 来查找接口的当前 ip,并使用它来加入多播组。这是我最终确定我的 ip 的代码:

#include <ifaddrs.h>
#include <boost/asio.hpp>
#include <cstring>
std::string getInterfaceAddress(const std::string & interfaceName)
{
    ifaddrs* firstNetIf = 0;
    getifaddrs(&firstNetIf);
    ifaddrs* netIf = 0;
    for(netIf = firstNetIf; netIf != 0; netIf = netIf->ifa_next)
    {
        if(netIf->ifa_addr->sa_family == AF_INET && std::strncmp(netIf->ifa_name, interfaceName.c_str(), interfaceName.length()) == 0)
        {
            break;
        }
    }
    unsigned long address =
            netIf != 0 ? reinterpret_cast<sockaddr_in*>(netIf->ifa_addr)->sin_addr.s_addr : 0;
    if(firstNetIf != 0)
    {
        freeifaddrs(firstNetIf);
    }
    return boost::asio::ip::address_v4(htonl(address)).to_string();
}

当然,就我而言,我可以与"eth1"进行比较并直接返回boost::asio::ip::address,但事实证明,此代码也可以以这种方式在其他地方使用。

最新更新