mirror of
https://github.com/FairRootGroup/FairMQ.git
synced 2025-10-13 00:31:14 +00:00
191 lines
5.6 KiB
C++
191 lines
5.6 KiB
C++
/********************************************************************************
|
|
* Copyright (C) 2017-2021 GSI Helmholtzzentrum fuer Schwerionenforschung GmbH *
|
|
* *
|
|
* This software is distributed under the terms of the *
|
|
* GNU Lesser General Public Licence (LGPL) version 3, *
|
|
* copied verbatim in the file "LICENSE" *
|
|
********************************************************************************/
|
|
|
|
#include <fairlogger/Logger.h>
|
|
#include <fairmq/tools/Network.h>
|
|
|
|
#ifndef _GNU_SOURCE
|
|
#define _GNU_SOURCE // To get defns of NI_MAXSERV and NI_MAXHOST
|
|
#endif
|
|
|
|
#include <algorithm>
|
|
#include <array>
|
|
#include <boost/algorithm/string.hpp> // trim
|
|
#include <boost/asio.hpp>
|
|
#include <cstdio>
|
|
#include <exception>
|
|
#include <fstream>
|
|
#include <ifaddrs.h>
|
|
#include <iostream>
|
|
#include <map>
|
|
#include <netdb.h>
|
|
#include <stdexcept>
|
|
#include <string>
|
|
#include <sys/socket.h>
|
|
#include <sys/types.h>
|
|
|
|
using namespace std;
|
|
|
|
namespace fair::mq::tools {
|
|
|
|
// returns a map with network interface names as keys and their IP addresses as values
|
|
map<string, string> getHostIPs()
|
|
{
|
|
map<string, string> addressMap;
|
|
ifaddrs* ifaddr = nullptr;
|
|
ifaddrs* ifa = nullptr;
|
|
int s = 0;
|
|
array<char, NI_MAXHOST> host{};
|
|
|
|
if (getifaddrs(&ifaddr) == -1) {
|
|
perror("getifaddrs");
|
|
throw runtime_error("getifaddrs failed");
|
|
}
|
|
|
|
for (ifa = ifaddr; ifa != nullptr; ifa = ifa->ifa_next) {
|
|
if (ifa->ifa_addr == nullptr) {
|
|
continue;
|
|
}
|
|
|
|
if (ifa->ifa_addr->sa_family == AF_INET) {
|
|
s = getnameinfo(ifa->ifa_addr,
|
|
sizeof(sockaddr_in),
|
|
host.data(),
|
|
NI_MAXHOST,
|
|
nullptr,
|
|
0,
|
|
NI_NUMERICHOST);
|
|
if (s != 0) {
|
|
cout << "getnameinfo() failed: " << gai_strerror(s) << endl;
|
|
throw runtime_error("getnameinfo() failed");
|
|
}
|
|
|
|
addressMap.insert({ifa->ifa_name, host.data()});
|
|
}
|
|
}
|
|
|
|
freeifaddrs(ifaddr);
|
|
|
|
return addressMap;
|
|
}
|
|
|
|
// get IP address of a given interface name
|
|
string getInterfaceIP(const string& interface)
|
|
{
|
|
try {
|
|
auto IPs = getHostIPs();
|
|
if (IPs.count(interface) > 0) {
|
|
return IPs[interface];
|
|
}
|
|
LOG(error) << "Could not find provided network interface: \""
|
|
<< interface << "\"!, exiting.";
|
|
return "";
|
|
} catch (runtime_error& re) {
|
|
cout << "could not get interface IP: " << re.what();
|
|
return "";
|
|
}
|
|
}
|
|
|
|
// get name of the default route interface
|
|
string getDefaultRouteNetworkInterface()
|
|
{
|
|
const int BUFSIZE(128);
|
|
array<char, BUFSIZE> buffer{};
|
|
string interfaceName;
|
|
|
|
#ifdef __APPLE__ // MacOS
|
|
unique_ptr<FILE, decltype(pclose)*> file(
|
|
popen("route -n get default | grep interface | cut -d \":\" -f 2", "r"), pclose);
|
|
#else // Linux
|
|
ifstream is("/proc/net/route");
|
|
string line;
|
|
|
|
// discard header
|
|
getline(is, line);
|
|
|
|
// check each line, until 00000000 destination is found
|
|
while (!is.eof()) {
|
|
getline(is, line);
|
|
size_t pos = line.find('\t');
|
|
|
|
if (pos == string::npos) {
|
|
break;
|
|
}
|
|
|
|
if (line.substr(pos + 1, 8) == "00000000") {
|
|
interfaceName = line.substr(0, pos);
|
|
LOG(debug) << "Detected network interface name for the default route: "
|
|
<< interfaceName;
|
|
return interfaceName;
|
|
}
|
|
}
|
|
|
|
LOG(debug) << "could not get network interface of the default route from /proc/net/route, "
|
|
"going to try via 'ip route'";
|
|
|
|
unique_ptr<FILE, decltype(pclose)*> file(
|
|
popen("ip route | grep default | cut -d \" \" -f 5 | head -n 1", "r"), pclose);
|
|
#endif
|
|
|
|
if (!file) {
|
|
LOG(error) << "Could not detect default route network interface name - popen() failed!";
|
|
return "";
|
|
}
|
|
|
|
while (feof(file.get()) == 0) {
|
|
if (fgets(buffer.data(), BUFSIZE, file.get()) != nullptr) {
|
|
interfaceName += buffer.data();
|
|
}
|
|
}
|
|
|
|
boost::algorithm::trim(interfaceName);
|
|
|
|
if (interfaceName.empty()) {
|
|
LOG(debug) << "Could not detect default route network interface name from /proc/net/route "
|
|
"nor 'ip route'";
|
|
throw DefaultRouteDetectionError("Could not detect default route network interface name "
|
|
"from /proc/net/route nor 'ip route'");
|
|
} else {
|
|
LOG(debug) << "Detected network interface name for the default route: " << interfaceName;
|
|
}
|
|
|
|
return interfaceName;
|
|
}
|
|
|
|
string getIpFromHostname(const string& hostname)
|
|
{
|
|
boost::asio::io_context ioc;
|
|
|
|
using namespace boost::asio::ip;
|
|
|
|
try {
|
|
tcp::resolver resolver(ioc);
|
|
tcp::resolver::query query(hostname, "");
|
|
tcp::resolver::iterator end;
|
|
|
|
auto it = find_if(static_cast<basic_resolver_iterator<tcp>>(resolver.resolve(query)),
|
|
end,
|
|
[](const tcp::endpoint& ep) { return ep.address().is_v4(); });
|
|
|
|
if (it != end) {
|
|
stringstream ss;
|
|
ss << static_cast<tcp::endpoint>(*it).address();
|
|
return ss.str();
|
|
}
|
|
|
|
LOG(warn) << "could not find ipv4 address for hostname '" << hostname << "'";
|
|
|
|
return "";
|
|
} catch (exception& e) {
|
|
LOG(error) << "could not resolve hostname '" << hostname << "', reason: " << e.what();
|
|
return "";
|
|
}
|
|
}
|
|
|
|
} // namespace fair::mq::tools
|