You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
159 lines
4.3 KiB
159 lines
4.3 KiB
#include "dvlnet/zerotier_native.h" |
|
|
|
#include <SDL.h> |
|
#include <atomic> |
|
|
|
#ifdef USE_SDL1 |
|
#include "utils/sdl2_to_1_2_backports.h" |
|
#else |
|
#include "utils/sdl2_backports.h" |
|
#endif |
|
|
|
#if (defined(_WIN64) || defined(_WIN32)) && !defined(NXDK) |
|
#include "utils/stdcompat/filesystem.hpp" |
|
#ifdef DVL_HAS_FILESYSTEM |
|
#define DVL_ZT_SYMLINK |
|
#endif |
|
#endif |
|
|
|
#ifdef DVL_ZT_SYMLINK |
|
#include <shlobj.h> |
|
#include <sodium.h> |
|
|
|
#include "utils/str_cat.hpp" |
|
#include "utils/utf8.hpp" |
|
#endif |
|
|
|
#include <ZeroTierSockets.h> |
|
#include <cstdlib> |
|
|
|
#include "utils/log.hpp" |
|
#include "utils/paths.h" |
|
|
|
#include "dvlnet/zerotier_lwip.h" |
|
|
|
namespace devilution { |
|
namespace net { |
|
|
|
namespace { |
|
|
|
#ifdef DVL_ZT_SYMLINK |
|
bool HasMultiByteChars(string_view path) |
|
{ |
|
return std::any_of(path.begin(), path.end(), IsTrailUtf8CodeUnit); |
|
} |
|
|
|
std::string ComputeAlternateFolderName(string_view path) |
|
{ |
|
const size_t hashSize = crypto_generichash_BYTES; |
|
unsigned char hash[hashSize]; |
|
|
|
const int status = crypto_generichash(hash, hashSize, |
|
reinterpret_cast<const unsigned char *>(path.data()), path.size(), |
|
nullptr, 0); |
|
|
|
if (status != 0) |
|
return ""; |
|
|
|
return fmt::format("{:02x}", fmt::join(hash, "")); |
|
} |
|
|
|
std::string ToZTCompliantPath(string_view configPath) |
|
{ |
|
if (!HasMultiByteChars(configPath)) |
|
return std::string(configPath); |
|
|
|
char commonAppDataPath[MAX_PATH]; |
|
if (!SUCCEEDED(SHGetFolderPathA(NULL, CSIDL_COMMON_APPDATA, NULL, 0, commonAppDataPath))) { |
|
LogVerbose("Failed to retrieve common application data path"); |
|
return std::string(configPath); |
|
} |
|
|
|
std::error_code err; |
|
std::string alternateConfigPath = StrCat(commonAppDataPath, "\\diasurgical\\devilution"); |
|
std::filesystem::create_directories(alternateConfigPath, err); |
|
if (err) { |
|
LogVerbose("Failed to create directories in ZT-compliant config path"); |
|
return std::string(configPath); |
|
} |
|
|
|
std::string alternateFolderName = ComputeAlternateFolderName(configPath); |
|
if (alternateFolderName == "") { |
|
LogVerbose("Failed to hash config path for ZT"); |
|
return std::string(configPath); |
|
} |
|
|
|
std::string symlinkPath = StrCat(alternateConfigPath, "\\", alternateFolderName); |
|
bool symlinkExists = std::filesystem::exists(std::filesystem::u8path(symlinkPath), err); |
|
if (err) { |
|
LogVerbose("Failed to determine if symlink for ZT-compliant config path exists"); |
|
return std::string(configPath); |
|
} |
|
|
|
if (!symlinkExists) { |
|
std::filesystem::create_directory_symlink( |
|
std::filesystem::u8path(configPath), |
|
std::filesystem::u8path(symlinkPath), |
|
err); |
|
|
|
if (err) { |
|
LogVerbose("Failed to create symlink for ZT-compliant config path"); |
|
return std::string(configPath); |
|
} |
|
} |
|
|
|
return StrCat(symlinkPath, "\\"); |
|
} |
|
#endif |
|
|
|
} // namespace |
|
|
|
// static constexpr uint64_t zt_earth = 0x8056c2e21c000001; |
|
static constexpr uint64_t ZtNetwork = 0xa84ac5c10a7ebb5f; |
|
|
|
static std::atomic_bool zt_network_ready(false); |
|
static std::atomic_bool zt_node_online(false); |
|
static std::atomic_bool zt_joined(false); |
|
|
|
static void Callback(void *ptr) |
|
{ |
|
zts_event_msg_t *msg = reinterpret_cast<zts_event_msg_t *>(ptr); |
|
// printf("callback %i\n", msg->eventCode); |
|
if (msg->event_code == ZTS_EVENT_NODE_ONLINE) { |
|
Log("ZeroTier: ZTS_EVENT_NODE_ONLINE, nodeId={:x}", (unsigned long long)msg->node->node_id); |
|
zt_node_online = true; |
|
if (!zt_joined) { |
|
zts_net_join(ZtNetwork); |
|
zt_joined = true; |
|
} |
|
} else if (msg->event_code == ZTS_EVENT_NODE_OFFLINE) { |
|
Log("ZeroTier: ZTS_EVENT_NODE_OFFLINE"); |
|
zt_node_online = false; |
|
} else if (msg->event_code == ZTS_EVENT_NETWORK_READY_IP6) { |
|
Log("ZeroTier: ZTS_EVENT_NETWORK_READY_IP6, networkId={:x}", (unsigned long long)msg->network->net_id); |
|
zt_ip6setup(); |
|
zt_network_ready = true; |
|
} else if (msg->event_code == ZTS_EVENT_ADDR_ADDED_IP6) { |
|
print_ip6_addr(&(msg->addr->addr)); |
|
} |
|
} |
|
|
|
bool zerotier_network_ready() |
|
{ |
|
return zt_network_ready && zt_node_online; |
|
} |
|
|
|
void zerotier_network_start() |
|
{ |
|
std::string configPath = paths::ConfigPath(); |
|
#ifdef DVL_ZT_SYMLINK |
|
configPath = ToZTCompliantPath(configPath); |
|
#endif |
|
std::string ztpath = configPath + "zerotier"; |
|
zts_init_from_storage(ztpath.c_str()); |
|
zts_init_set_event_handler(&Callback); |
|
zts_node_start(); |
|
} |
|
|
|
} // namespace net |
|
} // namespace devilution
|
|
|