Files
OrcaSlicer-KX/src/slic3r/Utils/InstanceID.cpp
SoftFever 565ac58fd5 Revamp OrcaSlicer updater (#10884)
* Revamp OrcaSlicer updater
2025-09-28 10:33:33 +08:00

210 lines
4.7 KiB
C++

#include "InstanceID.hpp"
#include <algorithm>
#include <cctype>
#include <mutex>
#include <optional>
#include <string>
#include <utility>
#include <vector>
#include <boost/algorithm/string.hpp>
#include <boost/log/trivial.hpp>
#include <boost/uuid/random_generator.hpp>
#include <boost/uuid/string_generator.hpp>
#include <boost/uuid/uuid.hpp>
#include <boost/uuid/uuid_io.hpp>
#include <boost/filesystem/operations.hpp>
#include <boost/filesystem/path.hpp>
#include <boost/nowide/fstream.hpp>
#include "libslic3r/AppConfig.hpp"
#include "libslic3r/Utils.hpp"
namespace Slic3r {
namespace instance_id {
namespace {
constexpr const char* CONFIG_KEY = "updater_iid";
constexpr const char* LEGACY_KEY = "iid";
std::mutex& cache_mutex()
{
static std::mutex mtx;
return mtx;
}
std::string& cached_iid()
{
static std::string value;
return value;
}
bool& cache_ready()
{
static bool ready = false;
return ready;
}
std::optional<std::string> normalize_uuid(std::string value)
{
boost::algorithm::trim(value);
boost::algorithm::to_lower(value);
if (value.size() != 36)
return std::nullopt;
try {
const boost::uuids::uuid parsed = boost::uuids::string_generator()(value);
if (parsed.version() != boost::uuids::uuid::version_random_number_based)
return std::nullopt;
return value;
} catch (...) {
return std::nullopt;
}
}
std::optional<std::string> read_config_value(AppConfig& config)
{
const auto read_key = [&](const char* key) -> std::optional<std::string> {
const std::string raw = config.get(key);
if (raw.empty())
return std::nullopt;
if (auto normalized = normalize_uuid(raw))
return normalized;
return std::nullopt;
};
if (auto value = read_key(CONFIG_KEY))
return value;
return read_key(LEGACY_KEY);
}
void write_config_value(AppConfig& config, const std::string& value)
{
config.set(CONFIG_KEY, value);
if (config.get(LEGACY_KEY) != value)
config.set(LEGACY_KEY, value);
}
void prune_config_value(AppConfig& config)
{
if (config.has(CONFIG_KEY))
config.erase("app", CONFIG_KEY);
if (config.has(LEGACY_KEY))
config.erase("app", LEGACY_KEY);
}
boost::filesystem::path storage_path()
{
const std::string& base_dir = Slic3r::data_dir();
if (base_dir.empty())
return {};
return boost::filesystem::path(base_dir) / ".orcaslicer_machine_id";
}
std::optional<std::string> read_storage_file()
{
const auto path = storage_path();
if (path.empty() || !boost::filesystem::exists(path))
return std::nullopt;
boost::nowide::ifstream file(path.string());
if (!file)
return std::nullopt;
std::string value;
std::getline(file, value);
file.close();
return normalize_uuid(value);
}
bool write_storage_file(const std::string& value)
{
if (value.empty())
return false;
const auto path = storage_path();
if (path.empty())
return false;
const auto parent = path.parent_path();
boost::system::error_code ec;
if (!parent.empty() && !boost::filesystem::exists(parent))
boost::filesystem::create_directories(parent, ec);
if (ec)
return false;
boost::nowide::ofstream file(path.string(), std::ios::trunc);
if (!file)
return false;
file << value;
file.close();
return file.good();
}
std::optional<std::string> read_secure()
{
return read_storage_file();
}
bool write_secure(const std::string& value)
{
return write_storage_file(value);
}
std::string generate_uuid()
{
const auto uuid = boost::uuids::random_generator()();
return boost::uuids::to_string(uuid);
}
} // namespace
std::string ensure(AppConfig& config)
{
std::lock_guard<std::mutex> lock(cache_mutex());
if (cache_ready())
return cached_iid();
if (auto secure = read_secure()) {
cached_iid() = *secure;
cache_ready() = true;
prune_config_value(config);
return cached_iid();
}
if (auto from_config = read_config_value(config)) {
cached_iid() = *from_config;
cache_ready() = true;
if (!write_secure(cached_iid()))
write_config_value(config, cached_iid());
else
prune_config_value(config);
return cached_iid();
}
cached_iid() = generate_uuid();
cache_ready() = true;
if (!write_secure(cached_iid()))
write_config_value(config, cached_iid());
else
prune_config_value(config);
return cached_iid();
}
void reset_cache_for_tests()
{
std::lock_guard<std::mutex> lock(cache_mutex());
cached_iid().clear();
cache_ready() = false;
}
} // namespace instance_id
} // namespace Slic3r