#include <util/memory/tempbuf.h>
#include <util/generic/singleton.h>
#include <util/generic/yexception.h>
#include <util/network/ip.h>
#if defined(_unix_)
#include <unistd.h>
#include <ifaddrs.h>
#include <netdb.h>
#endif
#if defined(_win_)
#include <WinSock2.h>
#endif
#include "defaults.h"
#include "yassert.h"
#include "hostname.h"
namespace {
struct THostNameHolder {
inline THostNameHolder() {
TTempBuf hostNameBuf;
if (gethostname(hostNameBuf.Data(), hostNameBuf.Size() - 1)) {
ythrow TSystemError() << "can not get host name";
}
HostName = hostNameBuf.Data();
}
TString HostName;
};
struct TFQDNHostNameHolder {
inline TFQDNHostNameHolder() {
char buf[1024];
memset(buf, 0, sizeof(buf));
int res = gethostname(buf, sizeof(buf) - 1);
if (res) {
ythrow TSystemError() << "can not get hostname";
}
#ifdef _darwin_
// On Darwin gethostname returns fqdn, see hostname.c in shell_cmds:
// https://github.com/apple-oss-distributions/shell_cmds/blob/main/hostname/hostname.c
// There are macs in the wild that don't have fqdn hostnames, but
// which have search domains in their resolv.conf, so any attempt to
// resolve AI_CANONNAME for the short hostname will result in the
// EAI_NONAME error.
// It seems using gethostname is enough to emulate the result of
// `hostname -f`, which still works on those macs.
FQDNHostName = buf;
#else
// On Linux `hostname -f` calls getaddrinfo with AI_CANONNAME flag
// to find the fqdn and will fail on any error.
// Hosts often have a short hostname and fqdn is resolved over dns.
// It is also very common to have a short hostname alias in
// /etc/hosts, which works as a fallback (e.g. no fqdn in search
// domains, otherwise `hostname -f` fails with an error and it is
// obvious the host is not configured correctly).
// Note that getaddrinfo may sometimes return EAI_NONAME even when
// host actually has fqdn, but dns is temporary unavailable, so we
// cannot ignore any errors on Linux.
struct addrinfo hints;
struct addrinfo* ais{nullptr};
memset(&hints, 0, sizeof(hints));
hints.ai_family = AF_UNSPEC;
hints.ai_flags = AI_CANONNAME;
res = getaddrinfo(buf, nullptr, &hints, &ais);
if (res) {
ythrow TSystemError() << "can not get FQDN (return code is " << res << ", hostname is \"" << buf << "\")";
}
FQDNHostName = ais->ai_canonname;
FQDNHostName.to_lower();
freeaddrinfo(ais);
#endif
}
TString FQDNHostName;
};
}
const TString& HostName() {
return (Singleton<THostNameHolder>())->HostName;
}
const char* GetHostName() {
return HostName().data();
}
const TString& FQDNHostName() {
return (Singleton<TFQDNHostNameHolder>())->FQDNHostName;
}
const char* GetFQDNHostName() {
return FQDNHostName().data();
}
bool IsFQDN(const TString& name) {
TString absName = name;
if (!absName.EndsWith('.')) {
absName.append(".");
}
try {
// ResolveHost() can't be used since it is ipv4-only, port is not important
TNetworkAddress addr(absName, 0);
} catch (const TNetworkResolutionError&) {
return false;
}
return true;
}