#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;
}