1
0
Fork 0
mirror of https://github.com/badaix/snapcast synced 2025-02-22 14:54:30 +01:00
snapcast/common/utils.hpp
2025-01-09 15:06:35 +01:00

458 lines
14 KiB
C++

/***
This file is part of snapcast
Copyright (C) 2014-2025 Johannes Pohl
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see <http://www.gnu.org/licenses/>.
***/
#pragma once
// local headers
#include "common/utils/string_utils.hpp"
// standard headers
#include <cstring>
#include <fstream>
#include <iomanip>
#include <memory>
#ifndef WINDOWS
#include <net/if.h>
#include <netinet/in.h>
#include <sys/ioctl.h>
#include <sys/utsname.h>
#include <unistd.h>
#endif
#include <sstream>
#include <string>
#include <sys/stat.h>
#include <sys/types.h>
#if !defined(WINDOWS) && !defined(FREEBSD)
#include <sys/sysinfo.h>
#endif
#ifdef MACOS
#include <IOKit/IOCFPlugIn.h>
#include <IOKit/IOTypes.h>
#include <ifaddrs.h>
#include <net/if_dl.h>
#endif
#ifdef ANDROID
#include <sys/system_properties.h>
#endif
#ifdef WINDOWS
#include <chrono>
#include <direct.h>
#include <iphlpapi.h>
#include <versionhelpers.h>
#include <windows.h>
#include <winsock2.h>
#endif
namespace strutils = utils::string;
#ifndef WINDOWS
static std::string execGetOutput(const std::string& cmd)
{
std::shared_ptr<::FILE> pipe(popen((cmd + " 2> /dev/null").c_str(), "r"), [](::FILE* stream)
{
if (stream != nullptr)
pclose(stream);
});
if (!pipe)
return "";
char buffer[1024];
std::string result;
while (feof(pipe.get()) == 0)
{
if (fgets(buffer, 1024, pipe.get()) != nullptr)
result += buffer;
}
return strutils::trim(result);
}
#endif
#ifdef ANDROID
static std::string getProp(const std::string& key, const std::string& def = "")
{
std::string result(def);
char cresult[PROP_VALUE_MAX + 1];
if (__system_property_get(key.c_str(), cresult) > 0)
result = cresult;
return result;
}
#endif
static std::string getOS()
{
static std::string os;
if (!os.empty())
return os;
#ifdef ANDROID
os = strutils::trim_copy("Android " + getProp("ro.build.version.release"));
#elif WINDOWS
if (/*IsWindows10OrGreater()*/ FALSE)
os = "Windows 10";
else if (IsWindows8Point1OrGreater())
os = "Windows 8.1";
else if (IsWindows8OrGreater())
os = "Windows 8";
else if (IsWindows7SP1OrGreater())
os = "Windows 7 SP1";
else if (IsWindows7OrGreater())
os = "Windows 7";
else if (IsWindowsVistaSP2OrGreater())
os = "Windows Vista SP2";
else if (IsWindowsVistaSP1OrGreater())
os = "Windows Vista SP1";
else if (IsWindowsVistaOrGreater())
os = "Windows Vista";
else if (IsWindowsXPSP3OrGreater())
os = "Windows XP SP3";
else if (IsWindowsXPSP2OrGreater())
os = "Windows XP SP2";
else if (IsWindowsXPSP1OrGreater())
os = "Windows XP SP1";
else if (IsWindowsXPOrGreater())
os = "Windows XP";
else
os = "Unknown Windows";
#else
os = execGetOutput("lsb_release -d");
if ((os.find(':') != std::string::npos) && (os.find("lsb_release") == std::string::npos))
os = strutils::trim_copy(os.substr(os.find(':') + 1));
#endif
#ifndef WINDOWS
if (os.empty())
{
os = strutils::trim_copy(execGetOutput("grep /etc/os-release /etc/openwrt_release -e PRETTY_NAME -e DISTRIB_DESCRIPTION"));
if (os.find('=') != std::string::npos)
{
os = strutils::trim_copy(os.substr(os.find('=') + 1));
os.erase(std::remove(os.begin(), os.end(), '"'), os.end());
os.erase(std::remove(os.begin(), os.end(), '\''), os.end());
}
}
if (os.empty())
{
utsname u;
uname(&u);
os = u.sysname;
}
#endif
strutils::trim(os);
return os;
}
static std::string getHostName()
{
#ifdef ANDROID
std::string result = getProp("net.hostname");
if (!result.empty())
return result;
result = getProp("ro.product.model");
if (!result.empty())
return result;
#endif
char hostname[1024];
hostname[1023] = '\0';
gethostname(hostname, 1023);
return hostname;
}
static std::string getArch()
{
std::string arch;
#ifdef ANDROID
arch = getProp("ro.product.cpu.abi");
if (!arch.empty())
return arch;
#endif
#ifndef WINDOWS
arch = execGetOutput("arch");
if (arch.empty())
arch = execGetOutput("uname -i");
if (arch.empty() || (arch == "unknown"))
arch = execGetOutput("uname -m");
#else
SYSTEM_INFO sysInfo;
GetSystemInfo(&sysInfo);
switch (sysInfo.wProcessorArchitecture)
{
case PROCESSOR_ARCHITECTURE_AMD64:
arch = "amd64";
break;
case PROCESSOR_ARCHITECTURE_ARM:
arch = "arm";
break;
case PROCESSOR_ARCHITECTURE_IA64:
arch = "ia64";
break;
case PROCESSOR_ARCHITECTURE_INTEL:
arch = "intel";
break;
default:
case PROCESSOR_ARCHITECTURE_UNKNOWN:
arch = "unknown";
break;
}
#endif
return strutils::trim_copy(arch);
}
// Seems not to be used
// static std::chrono::seconds uptime()
// {
// #ifndef WINDOWS
// #ifndef FREEBSD
// struct sysinfo info;
// sysinfo(&info);
// return std::chrono::seconds(info.uptime);
// #else
// std::string uptime = execGetOutput("sysctl kern.boottime");
// if ((uptime.find(" sec = ") != std::string::npos) && (uptime.find(",") != std::string::npos))
// {
// uptime = strutils::trim_copy(uptime.substr(uptime.find(" sec = ") + 7));
// uptime.resize(uptime.find(","));
// timeval now;
// gettimeofday(&now, NULL);
// try
// {
// return std::chrono::seconds(now.tv_sec - cpt::stoul(uptime));
// }
// catch (...)
// {
// }
// }
// return 0s;
// #endif
// #else
// return std::chrono::duration_cast<std::chrono::seconds>(std::chrono::milliseconds(GetTickCount()));
// #endif
// }
/// http://stackoverflow.com/questions/2174768/generating-random-uuids-in-linux
static std::string generateUUID()
{
static bool initialized(false);
if (!initialized)
{
std::srand(static_cast<unsigned int>(std::time(nullptr)));
initialized = true;
}
std::stringstream ss;
ss << std::setfill('0') << std::hex << std::setw(4) << (std::rand() % 0xffff) << std::setw(4) << (std::rand() % 0xffff) << "-" << std::setw(4)
<< (std::rand() % 0xffff) << "-" << std::setw(4) << (std::rand() % 0xffff) << "-" << std::setw(4) << (std::rand() % 0xffff) << "-" << std::setw(4)
<< (std::rand() % 0xffff) << std::setw(4) << (std::rand() % 0xffff) << std::setw(4) << (std::rand() % 0xffff);
return ss.str();
}
#ifndef WINDOWS
/// https://gist.github.com/OrangeTide/909204
static std::string getMacAddress(int sock)
{
struct ifreq ifr;
struct ifconf ifc;
char buf[16384];
int success = 0;
if (sock < 0)
return "";
ifc.ifc_len = sizeof(buf);
ifc.ifc_buf = buf;
if (ioctl(sock, SIOCGIFCONF, &ifc) != 0)
return "";
struct ifreq* it = ifc.ifc_req;
for (int i = 0; i < ifc.ifc_len;)
{
/// some systems have ifr_addr.sa_len and adjust the length that way, but not mine. weird */
#ifdef FREEBSD
size_t len = IFNAMSIZ + it->ifr_addr.sa_len;
#else
size_t len = sizeof(*it);
#endif
strcpy(ifr.ifr_name, it->ifr_name);
if (ioctl(sock, SIOCGIFFLAGS, &ifr) == 0)
{
if (!(ifr.ifr_flags & IFF_LOOPBACK)) // don't count loopback
{
#ifdef MACOS
/// Dirty Mac version
struct ifaddrs *ifap, *ifaptr;
unsigned char* ptr;
if (getifaddrs(&ifap) == 0)
{
for (ifaptr = ifap; ifaptr != NULL; ifaptr = ifaptr->ifa_next)
{
// std::cout << ifaptr->ifa_name << ", " << ifreq->ifr_name << "\n";
if (strcmp(ifaptr->ifa_name, it->ifr_name) != 0)
continue;
if (ifaptr->ifa_addr->sa_family == AF_LINK)
{
ptr = (unsigned char*)LLADDR((struct sockaddr_dl*)ifaptr->ifa_addr);
char mac[19];
sprintf(mac, "%02x:%02x:%02x:%02x:%02x:%02x", *ptr, *(ptr + 1), *(ptr + 2), *(ptr + 3), *(ptr + 4), *(ptr + 5));
if (strcmp(mac, "00:00:00:00:00:00") == 0)
continue;
freeifaddrs(ifap);
return mac;
}
}
freeifaddrs(ifap);
}
#endif
#ifdef FREEBSD
if (ioctl(sock, SIOCGIFMAC, &ifr) == 0)
#else
if (ioctl(sock, SIOCGIFHWADDR, &ifr) == 0)
#endif
{
success = 1;
break;
}
else
{
std::stringstream ss;
ss << "/sys/class/net/" << ifr.ifr_name << "/address";
std::ifstream infile(ss.str().c_str());
std::string line;
if (infile.good() && std::getline(infile, line))
{
strutils::trim(line);
if ((line.size() == 17) && (line[2] == ':'))
return line;
}
}
}
}
else
{ /* handle error */
}
it = (struct ifreq*)((char*)it + len);
i += len;
}
if (!success)
return "";
char mac[19];
#ifndef FREEBSD
sprintf(mac, "%02x:%02x:%02x:%02x:%02x:%02x", (unsigned char)ifr.ifr_hwaddr.sa_data[0], (unsigned char)ifr.ifr_hwaddr.sa_data[1],
(unsigned char)ifr.ifr_hwaddr.sa_data[2], (unsigned char)ifr.ifr_hwaddr.sa_data[3], (unsigned char)ifr.ifr_hwaddr.sa_data[4],
(unsigned char)ifr.ifr_hwaddr.sa_data[5]);
#else
sprintf(mac, "%02x:%02x:%02x:%02x:%02x:%02x", (unsigned char)ifr.ifr_ifru.ifru_addr.sa_data[0], (unsigned char)ifr.ifr_ifru.ifru_addr.sa_data[1],
(unsigned char)ifr.ifr_ifru.ifru_addr.sa_data[2], (unsigned char)ifr.ifr_ifru.ifru_addr.sa_data[3],
(unsigned char)ifr.ifr_ifru.ifru_addr.sa_data[4], (unsigned char)ifr.ifr_ifru.ifru_addr.sa_data[5]);
#endif
return mac;
}
#else
static std::string getMacAddress(const std::string& address)
{
IP_ADAPTER_INFO* first;
IP_ADAPTER_INFO* pos;
ULONG bufferLength = sizeof(IP_ADAPTER_INFO);
first = (IP_ADAPTER_INFO*)malloc(bufferLength);
if (GetAdaptersInfo(first, &bufferLength) == ERROR_BUFFER_OVERFLOW)
{
free(first);
first = (IP_ADAPTER_INFO*)malloc(bufferLength);
}
char mac[19];
if (GetAdaptersInfo(first, &bufferLength) == NO_ERROR)
for (pos = first; pos != NULL; pos = pos->Next)
{
IP_ADDR_STRING* firstAddr = &pos->IpAddressList;
IP_ADDR_STRING* posAddr;
for (posAddr = firstAddr; posAddr != NULL; posAddr = posAddr->Next)
if (_stricmp(posAddr->IpAddress.String, address.c_str()) == 0)
{
sprintf(mac, "%02x:%02x:%02x:%02x:%02x:%02x", pos->Address[0], pos->Address[1], pos->Address[2], pos->Address[3], pos->Address[4],
pos->Address[5]);
free(first);
return mac;
}
}
else
free(first);
return mac;
}
#endif
static std::string getHostId(const std::string& defaultId = "")
{
std::string result = strutils::trim_copy(defaultId);
if (!result.empty() // default provided
&& (result != "00:00:00:00:00:00") // default mac returned by getMaxAddress if it fails
&& (result != "02:00:00:00:00:00") // the Android API will return "02:00:00:00:00:00" for WifiInfo.getMacAddress()
&& (result != "ac:de:48:00:11:22") // iBridge interface on new MacBook Pro (later 2016)
)
return result;
#ifdef MACOS
/// https://stackoverflow.com/questions/933460/unique-hardware-id-in-mac-os-x
/// About this Mac, Hardware-UUID
char buf[64];
io_registry_entry_t ioRegistryRoot = IORegistryEntryFromPath(kIOMasterPortDefault, "IOService:/");
CFStringRef uuidCf = (CFStringRef)IORegistryEntryCreateCFProperty(ioRegistryRoot, CFSTR(kIOPlatformUUIDKey), kCFAllocatorDefault, 0);
IOObjectRelease(ioRegistryRoot);
if (CFStringGetCString(uuidCf, buf, 64, kCFStringEncodingMacRoman))
result = buf;
CFRelease(uuidCf);
#elif ANDROID
result = getProp("ro.serialno");
#endif
// #else
// // on embedded platforms it's
// // - either not there
// // - or not unique, or changes during boot
// // - or changes during boot
// std::ifstream infile("/var/lib/dbus/machine-id");
// if (infile.good())
// std::getline(infile, result);
// #endif
strutils::trim(result);
if (!result.empty())
return result;
/// The host name should be unique enough in a LAN
return getHostName();
}