2018-03-28 09:36:36 +00:00
|
|
|
#include "PresetUpdater.hpp"
|
|
|
|
|
2018-04-12 18:04:48 +00:00
|
|
|
#include <algorithm>
|
2018-03-28 09:36:36 +00:00
|
|
|
#include <thread>
|
2018-04-24 16:06:42 +00:00
|
|
|
#include <unordered_map>
|
2018-04-23 09:16:47 +00:00
|
|
|
#include <ostream>
|
2018-04-13 13:08:58 +00:00
|
|
|
#include <stdexcept>
|
|
|
|
#include <boost/format.hpp>
|
2018-03-28 09:36:36 +00:00
|
|
|
#include <boost/algorithm/string.hpp>
|
2018-04-11 11:12:08 +00:00
|
|
|
#include <boost/filesystem.hpp>
|
2018-03-28 09:36:36 +00:00
|
|
|
#include <boost/filesystem/fstream.hpp>
|
2018-04-17 09:54:59 +00:00
|
|
|
#include <boost/log/trivial.hpp>
|
2018-03-28 09:36:36 +00:00
|
|
|
|
|
|
|
#include <wx/app.h>
|
|
|
|
#include <wx/event.h>
|
2018-04-12 18:04:48 +00:00
|
|
|
#include <wx/msgdlg.h>
|
2018-04-17 14:59:53 +00:00
|
|
|
|
|
|
|
#include "libslic3r/libslic3r.h"
|
2018-03-28 09:36:36 +00:00
|
|
|
#include "libslic3r/Utils.hpp"
|
|
|
|
#include "slic3r/GUI/GUI.hpp"
|
|
|
|
#include "slic3r/GUI/PresetBundle.hpp"
|
2018-04-24 16:06:42 +00:00
|
|
|
#include "slic3r/GUI/UpdateDialogs.hpp"
|
|
|
|
#include "slic3r/GUI/ConfigWizard.hpp"
|
2018-03-28 09:36:36 +00:00
|
|
|
#include "slic3r/Utils/Http.hpp"
|
2018-04-12 18:04:48 +00:00
|
|
|
#include "slic3r/Config/Version.hpp"
|
|
|
|
#include "slic3r/Config/Snapshot.hpp"
|
2018-09-20 23:33:41 +00:00
|
|
|
#include "slic3r/GUI/GUI_App.hpp"
|
2018-03-28 09:36:36 +00:00
|
|
|
|
|
|
|
namespace fs = boost::filesystem;
|
2018-04-12 18:04:48 +00:00
|
|
|
using Slic3r::GUI::Config::Index;
|
|
|
|
using Slic3r::GUI::Config::Version;
|
|
|
|
using Slic3r::GUI::Config::Snapshot;
|
|
|
|
using Slic3r::GUI::Config::SnapshotDB;
|
2018-03-28 09:36:36 +00:00
|
|
|
|
|
|
|
|
|
|
|
namespace Slic3r {
|
|
|
|
|
|
|
|
|
|
|
|
enum {
|
|
|
|
SLIC3R_VERSION_BODY_MAX = 256,
|
|
|
|
};
|
|
|
|
|
2018-04-13 13:08:58 +00:00
|
|
|
static const char *INDEX_FILENAME = "index.idx";
|
|
|
|
static const char *TMP_EXTENSION = ".download";
|
|
|
|
|
2018-04-12 18:04:48 +00:00
|
|
|
|
2018-04-24 16:06:42 +00:00
|
|
|
struct Update
|
2018-04-17 14:59:53 +00:00
|
|
|
{
|
2018-04-24 16:06:42 +00:00
|
|
|
fs::path source;
|
|
|
|
fs::path target;
|
|
|
|
Version version;
|
2018-04-17 14:59:53 +00:00
|
|
|
|
2018-04-24 16:06:42 +00:00
|
|
|
Update(fs::path &&source, fs::path &&target, const Version &version) :
|
|
|
|
source(std::move(source)),
|
|
|
|
target(std::move(target)),
|
|
|
|
version(version)
|
|
|
|
{}
|
2018-04-17 14:59:53 +00:00
|
|
|
|
2018-04-24 16:06:42 +00:00
|
|
|
std::string name() const { return source.stem().string(); }
|
2018-04-17 14:59:53 +00:00
|
|
|
|
2018-04-24 16:06:42 +00:00
|
|
|
friend std::ostream& operator<<(std::ostream& os , const Update &self) {
|
|
|
|
os << "Update(" << self.source.string() << " -> " << self.target.string() << ')';
|
|
|
|
return os;
|
2018-04-17 14:59:53 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2018-04-24 16:06:42 +00:00
|
|
|
struct Incompat
|
2018-04-12 18:04:48 +00:00
|
|
|
{
|
2018-04-24 16:06:42 +00:00
|
|
|
fs::path bundle;
|
2018-04-12 18:04:48 +00:00
|
|
|
Version version;
|
|
|
|
|
2018-04-24 16:06:42 +00:00
|
|
|
Incompat(fs::path &&bundle, const Version &version) :
|
|
|
|
bundle(std::move(bundle)),
|
2018-04-12 18:04:48 +00:00
|
|
|
version(version)
|
|
|
|
{}
|
|
|
|
|
2018-04-24 16:06:42 +00:00
|
|
|
std::string name() const { return bundle.stem().string(); }
|
2018-04-16 14:52:11 +00:00
|
|
|
|
2018-04-24 16:06:42 +00:00
|
|
|
friend std::ostream& operator<<(std::ostream& os , const Incompat &self) {
|
|
|
|
os << "Incompat(" << self.bundle.string() << ')';
|
2018-04-23 09:16:47 +00:00
|
|
|
return os;
|
|
|
|
}
|
2018-04-12 18:04:48 +00:00
|
|
|
};
|
|
|
|
|
2018-04-24 16:06:42 +00:00
|
|
|
struct Updates
|
|
|
|
{
|
|
|
|
std::vector<Incompat> incompats;
|
|
|
|
std::vector<Update> updates;
|
|
|
|
};
|
2018-04-12 18:04:48 +00:00
|
|
|
|
|
|
|
|
2018-03-28 09:36:36 +00:00
|
|
|
struct PresetUpdater::priv
|
|
|
|
{
|
2018-04-13 13:08:58 +00:00
|
|
|
std::vector<Index> index_db;
|
|
|
|
|
|
|
|
bool enabled_version_check;
|
|
|
|
bool enabled_config_update;
|
|
|
|
std::string version_check_url;
|
2018-04-17 15:11:56 +00:00
|
|
|
bool had_config_update;
|
2018-03-28 09:36:36 +00:00
|
|
|
|
|
|
|
fs::path cache_path;
|
2018-04-12 18:04:48 +00:00
|
|
|
fs::path rsrc_path;
|
|
|
|
fs::path vendor_path;
|
|
|
|
|
2018-03-28 09:36:36 +00:00
|
|
|
bool cancel;
|
|
|
|
std::thread thread;
|
|
|
|
|
2018-09-20 23:33:41 +00:00
|
|
|
priv();
|
2018-03-28 09:36:36 +00:00
|
|
|
|
2018-04-13 13:08:58 +00:00
|
|
|
void set_download_prefs(AppConfig *app_config);
|
|
|
|
bool get_file(const std::string &url, const fs::path &target_path) const;
|
|
|
|
void prune_tmps() const;
|
|
|
|
void sync_version() const;
|
|
|
|
void sync_config(const std::set<VendorProfile> vendors) const;
|
2018-04-12 18:04:48 +00:00
|
|
|
|
|
|
|
void check_install_indices() const;
|
2018-04-23 11:58:03 +00:00
|
|
|
Updates get_config_updates() const;
|
2018-04-18 09:40:43 +00:00
|
|
|
void perform_updates(Updates &&updates, bool snapshot = true) const;
|
2018-06-11 15:34:06 +00:00
|
|
|
|
|
|
|
static void copy_file(const fs::path &from, const fs::path &to);
|
2018-03-28 09:36:36 +00:00
|
|
|
};
|
|
|
|
|
2018-09-20 23:33:41 +00:00
|
|
|
PresetUpdater::priv::priv() :
|
2018-04-17 15:11:56 +00:00
|
|
|
had_config_update(false),
|
2018-03-28 09:36:36 +00:00
|
|
|
cache_path(fs::path(Slic3r::data_dir()) / "cache"),
|
2018-04-12 18:04:48 +00:00
|
|
|
rsrc_path(fs::path(resources_dir()) / "profiles"),
|
|
|
|
vendor_path(fs::path(Slic3r::data_dir()) / "vendor"),
|
2018-03-28 09:36:36 +00:00
|
|
|
cancel(false)
|
2018-04-13 13:08:58 +00:00
|
|
|
{
|
2018-09-20 23:33:41 +00:00
|
|
|
set_download_prefs(GUI::wxGetApp().app_config);
|
2018-04-13 13:08:58 +00:00
|
|
|
check_install_indices();
|
|
|
|
index_db = std::move(Index::load_db());
|
|
|
|
}
|
|
|
|
|
2018-04-23 09:16:47 +00:00
|
|
|
// Pull relevant preferences from AppConfig
|
2018-04-13 13:08:58 +00:00
|
|
|
void PresetUpdater::priv::set_download_prefs(AppConfig *app_config)
|
|
|
|
{
|
|
|
|
enabled_version_check = app_config->get("version_check") == "1";
|
2018-04-25 15:43:01 +00:00
|
|
|
version_check_url = app_config->version_check_url();
|
2018-05-15 10:14:26 +00:00
|
|
|
enabled_config_update = app_config->get("preset_update") == "1" && !app_config->legacy_datadir();
|
2018-04-13 13:08:58 +00:00
|
|
|
}
|
2018-03-28 09:36:36 +00:00
|
|
|
|
2018-04-23 09:16:47 +00:00
|
|
|
// Downloads a file (http get operation). Cancels if the Updater is being destroyed.
|
2018-04-13 13:08:58 +00:00
|
|
|
bool PresetUpdater::priv::get_file(const std::string &url, const fs::path &target_path) const
|
2018-03-28 09:36:36 +00:00
|
|
|
{
|
2018-04-13 13:08:58 +00:00
|
|
|
bool res = false;
|
|
|
|
fs::path tmp_path = target_path;
|
2018-04-20 09:05:00 +00:00
|
|
|
tmp_path += (boost::format(".%1%%2%") % get_current_pid() % TMP_EXTENSION).str();
|
|
|
|
|
2018-04-23 09:16:47 +00:00
|
|
|
BOOST_LOG_TRIVIAL(info) << boost::format("Get: `%1%`\n\t-> `%2%`\n\tvia tmp path `%3%`")
|
|
|
|
% url
|
|
|
|
% target_path.string()
|
|
|
|
% tmp_path.string();
|
2018-04-13 13:08:58 +00:00
|
|
|
|
|
|
|
Http::get(url)
|
|
|
|
.on_progress([this](Http::Progress, bool &cancel) {
|
|
|
|
if (cancel) { cancel = true; }
|
|
|
|
})
|
2018-04-23 09:16:47 +00:00
|
|
|
.on_error([&](std::string body, std::string error, unsigned http_status) {
|
|
|
|
(void)body;
|
|
|
|
BOOST_LOG_TRIVIAL(error) << boost::format("Error getting: `%1%`: HTTP %2%, %3%")
|
|
|
|
% url
|
|
|
|
% http_status
|
2018-04-24 16:06:42 +00:00
|
|
|
% error;
|
2018-04-23 09:16:47 +00:00
|
|
|
})
|
2018-06-06 08:52:19 +00:00
|
|
|
.on_complete([&](std::string body, unsigned /* http_status */) {
|
2018-04-13 13:08:58 +00:00
|
|
|
fs::fstream file(tmp_path, std::ios::out | std::ios::binary | std::ios::trunc);
|
|
|
|
file.write(body.c_str(), body.size());
|
2018-04-16 14:52:11 +00:00
|
|
|
file.close();
|
2018-04-13 13:08:58 +00:00
|
|
|
fs::rename(tmp_path, target_path);
|
|
|
|
res = true;
|
|
|
|
})
|
|
|
|
.perform_sync();
|
|
|
|
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2018-04-23 09:16:47 +00:00
|
|
|
// Remove leftover paritally downloaded files, if any.
|
2018-04-13 13:08:58 +00:00
|
|
|
void PresetUpdater::priv::prune_tmps() const
|
|
|
|
{
|
|
|
|
for (fs::directory_iterator it(cache_path); it != fs::directory_iterator(); ++it) {
|
|
|
|
if (it->path().extension() == TMP_EXTENSION) {
|
2018-04-23 09:16:47 +00:00
|
|
|
BOOST_LOG_TRIVIAL(debug) << "Cache prune: " << it->path().string();
|
2018-04-13 13:08:58 +00:00
|
|
|
fs::remove(it->path());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-23 09:16:47 +00:00
|
|
|
// Get Slic3rPE version available online, save in AppConfig.
|
2018-04-13 13:08:58 +00:00
|
|
|
void PresetUpdater::priv::sync_version() const
|
|
|
|
{
|
|
|
|
if (! enabled_version_check) { return; }
|
|
|
|
|
2018-04-23 09:16:47 +00:00
|
|
|
BOOST_LOG_TRIVIAL(info) << boost::format("Downloading Slic3rPE online version from: `%1%`") % version_check_url;
|
|
|
|
|
2018-04-13 13:08:58 +00:00
|
|
|
Http::get(version_check_url)
|
2018-03-28 09:36:36 +00:00
|
|
|
.size_limit(SLIC3R_VERSION_BODY_MAX)
|
|
|
|
.on_progress([this](Http::Progress, bool &cancel) {
|
|
|
|
cancel = this->cancel;
|
|
|
|
})
|
2018-04-23 09:16:47 +00:00
|
|
|
.on_error([&](std::string body, std::string error, unsigned http_status) {
|
|
|
|
(void)body;
|
|
|
|
BOOST_LOG_TRIVIAL(error) << boost::format("Error getting: `%1%`: HTTP %2%, %3%")
|
|
|
|
% version_check_url
|
|
|
|
% http_status
|
2018-04-24 16:06:42 +00:00
|
|
|
% error;
|
2018-04-23 09:16:47 +00:00
|
|
|
})
|
2018-06-06 08:52:19 +00:00
|
|
|
.on_complete([&](std::string body, unsigned /* http_status */) {
|
2018-03-28 09:36:36 +00:00
|
|
|
boost::trim(body);
|
2018-04-23 09:16:47 +00:00
|
|
|
BOOST_LOG_TRIVIAL(info) << boost::format("Got Slic3rPE online version: `%1%`. Sending to GUI thread...") % body;
|
2018-09-20 23:33:41 +00:00
|
|
|
// wxCommandEvent* evt = new wxCommandEvent(version_online_event);
|
|
|
|
// evt->SetString(body);
|
|
|
|
// GUI::get_app()->QueueEvent(evt);
|
|
|
|
GUI::wxGetApp().app_config->set("version_online", body);
|
|
|
|
GUI::wxGetApp().app_config->save();
|
2018-03-28 09:36:36 +00:00
|
|
|
})
|
|
|
|
.perform_sync();
|
2018-04-13 13:08:58 +00:00
|
|
|
}
|
2018-03-28 09:36:36 +00:00
|
|
|
|
2018-04-23 09:16:47 +00:00
|
|
|
// Download vendor indices. Also download new bundles if an index indicates there's a new one available.
|
|
|
|
// Both are saved in cache.
|
2018-04-13 13:08:58 +00:00
|
|
|
void PresetUpdater::priv::sync_config(const std::set<VendorProfile> vendors) const
|
|
|
|
{
|
2018-04-23 09:16:47 +00:00
|
|
|
BOOST_LOG_TRIVIAL(info) << "Syncing configuration cache";
|
2018-04-13 13:08:58 +00:00
|
|
|
|
|
|
|
if (!enabled_config_update) { return; }
|
2018-03-28 09:36:36 +00:00
|
|
|
|
|
|
|
// Donwload vendor preset bundles
|
2018-04-13 13:08:58 +00:00
|
|
|
for (const auto &index : index_db) {
|
2018-03-28 09:36:36 +00:00
|
|
|
if (cancel) { return; }
|
|
|
|
|
2018-04-13 13:08:58 +00:00
|
|
|
const auto vendor_it = vendors.find(VendorProfile(index.vendor()));
|
2018-04-23 09:16:47 +00:00
|
|
|
if (vendor_it == vendors.end()) {
|
|
|
|
BOOST_LOG_TRIVIAL(warning) << "No such vendor: " << index.vendor();
|
|
|
|
continue;
|
|
|
|
}
|
2018-04-13 13:08:58 +00:00
|
|
|
|
|
|
|
const VendorProfile &vendor = *vendor_it;
|
2018-04-23 09:16:47 +00:00
|
|
|
if (vendor.config_update_url.empty()) {
|
|
|
|
BOOST_LOG_TRIVIAL(info) << "Vendor has no config_update_url: " << vendor.name;
|
|
|
|
continue;
|
|
|
|
}
|
2018-04-13 13:08:58 +00:00
|
|
|
|
|
|
|
// Download a fresh index
|
2018-04-23 09:16:47 +00:00
|
|
|
BOOST_LOG_TRIVIAL(info) << "Downloading index for vendor: " << vendor.name;
|
2018-04-13 13:08:58 +00:00
|
|
|
const auto idx_url = vendor.config_update_url + "/" + INDEX_FILENAME;
|
|
|
|
const auto idx_path = cache_path / (vendor.id + ".idx");
|
|
|
|
if (! get_file(idx_url, idx_path)) { continue; }
|
|
|
|
if (cancel) { return; }
|
|
|
|
|
|
|
|
// Load the fresh index up
|
|
|
|
Index new_index;
|
|
|
|
new_index.load(idx_path);
|
2018-03-28 09:36:36 +00:00
|
|
|
|
2018-04-13 13:08:58 +00:00
|
|
|
// See if a there's a new version to download
|
|
|
|
const auto recommended_it = new_index.recommended();
|
2018-04-23 09:16:47 +00:00
|
|
|
if (recommended_it == new_index.end()) {
|
2018-05-01 08:35:07 +00:00
|
|
|
BOOST_LOG_TRIVIAL(error) << boost::format("No recommended version for vendor: %1%, invalid index?") % vendor.name;
|
2018-04-23 09:16:47 +00:00
|
|
|
continue;
|
|
|
|
}
|
2018-04-13 13:08:58 +00:00
|
|
|
const auto recommended = recommended_it->config_version;
|
2018-03-28 09:36:36 +00:00
|
|
|
|
2018-06-19 16:26:38 +00:00
|
|
|
BOOST_LOG_TRIVIAL(debug) << boost::format("Got index for vendor: %1%: current version: %2%, recommended version: %3%")
|
2018-04-23 09:16:47 +00:00
|
|
|
% vendor.name
|
|
|
|
% vendor.config_version.to_string()
|
|
|
|
% recommended.to_string();
|
2018-04-13 13:08:58 +00:00
|
|
|
|
|
|
|
if (vendor.config_version >= recommended) { continue; }
|
|
|
|
|
|
|
|
// Download a fresh bundle
|
2018-04-23 09:16:47 +00:00
|
|
|
BOOST_LOG_TRIVIAL(info) << "Downloading new bundle for vendor: " << vendor.name;
|
2018-04-13 13:08:58 +00:00
|
|
|
const auto bundle_url = (boost::format("%1%/%2%.ini") % vendor.config_update_url % recommended.to_string()).str();
|
|
|
|
const auto bundle_path = cache_path / (vendor.id + ".ini");
|
|
|
|
if (! get_file(bundle_url, bundle_path)) { continue; }
|
|
|
|
if (cancel) { return; }
|
2018-03-28 09:36:36 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-23 09:16:47 +00:00
|
|
|
// Install indicies from resources. Only installs those that are either missing or older than in resources.
|
2018-04-12 18:04:48 +00:00
|
|
|
void PresetUpdater::priv::check_install_indices() const
|
|
|
|
{
|
2018-04-23 09:16:47 +00:00
|
|
|
BOOST_LOG_TRIVIAL(info) << "Checking if indices need to be installed from resources...";
|
|
|
|
|
2018-04-12 18:04:48 +00:00
|
|
|
for (fs::directory_iterator it(rsrc_path); it != fs::directory_iterator(); ++it) {
|
|
|
|
const auto &path = it->path();
|
|
|
|
if (path.extension() == ".idx") {
|
|
|
|
const auto path_in_cache = cache_path / path.filename();
|
|
|
|
|
|
|
|
if (! fs::exists(path_in_cache)) {
|
2018-04-23 09:16:47 +00:00
|
|
|
BOOST_LOG_TRIVIAL(info) << "Install index from resources: " << path.filename();
|
2018-06-11 15:34:06 +00:00
|
|
|
copy_file(path, path_in_cache);
|
2018-04-17 08:28:32 +00:00
|
|
|
} else {
|
|
|
|
Index idx_rsrc, idx_cache;
|
|
|
|
idx_rsrc.load(path);
|
|
|
|
idx_cache.load(path_in_cache);
|
|
|
|
|
|
|
|
if (idx_cache.version() < idx_rsrc.version()) {
|
2018-04-23 09:16:47 +00:00
|
|
|
BOOST_LOG_TRIVIAL(info) << "Update index from resources: " << path.filename();
|
2018-06-11 15:34:06 +00:00
|
|
|
copy_file(path, path_in_cache);
|
2018-04-17 08:28:32 +00:00
|
|
|
}
|
2018-04-12 18:04:48 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-23 09:16:47 +00:00
|
|
|
// Generates a list of bundle updates that are to be performed
|
2018-04-23 11:58:03 +00:00
|
|
|
Updates PresetUpdater::priv::get_config_updates() const
|
2018-04-12 18:04:48 +00:00
|
|
|
{
|
|
|
|
Updates updates;
|
2018-04-23 11:58:03 +00:00
|
|
|
|
2018-04-23 09:16:47 +00:00
|
|
|
BOOST_LOG_TRIVIAL(info) << "Checking for cached configuration updates...";
|
2018-04-12 18:04:48 +00:00
|
|
|
|
|
|
|
for (const auto idx : index_db) {
|
2018-04-24 16:06:42 +00:00
|
|
|
auto bundle_path = vendor_path / (idx.vendor() + ".ini");
|
2018-04-12 18:04:48 +00:00
|
|
|
|
|
|
|
if (! fs::exists(bundle_path)) {
|
2018-04-23 09:16:47 +00:00
|
|
|
BOOST_LOG_TRIVIAL(info) << "Bundle not present for index, skipping: " << idx.vendor();
|
2018-04-12 18:04:48 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Perform a basic load and check the version
|
|
|
|
const auto vp = VendorProfile::from_ini(bundle_path, false);
|
|
|
|
|
|
|
|
const auto ver_current = idx.find(vp.config_version);
|
|
|
|
if (ver_current == idx.end()) {
|
2018-07-24 13:27:31 +00:00
|
|
|
auto message = (boost::format("Preset bundle `%1%` version not found in index: %2%") % idx.vendor() % vp.config_version.to_string()).str();
|
|
|
|
BOOST_LOG_TRIVIAL(error) << message;
|
|
|
|
throw std::runtime_error(message);
|
2018-04-12 18:04:48 +00:00
|
|
|
}
|
|
|
|
|
2018-07-21 07:32:45 +00:00
|
|
|
// Getting a recommended version from the latest index, wich may have been downloaded
|
|
|
|
// from the internet, or installed / updated from the installation resources.
|
2018-04-12 18:04:48 +00:00
|
|
|
const auto recommended = idx.recommended();
|
|
|
|
if (recommended == idx.end()) {
|
2018-05-01 08:35:07 +00:00
|
|
|
BOOST_LOG_TRIVIAL(error) << boost::format("No recommended version for vendor: %1%, invalid index?") % idx.vendor();
|
2018-04-12 18:04:48 +00:00
|
|
|
}
|
|
|
|
|
2018-04-23 09:16:47 +00:00
|
|
|
BOOST_LOG_TRIVIAL(debug) << boost::format("Vendor: %1%, version installed: %2%, version cached: %3%")
|
|
|
|
% vp.name
|
2018-04-24 16:06:42 +00:00
|
|
|
% ver_current->config_version.to_string()
|
|
|
|
% recommended->config_version.to_string();
|
2018-04-23 09:16:47 +00:00
|
|
|
|
2018-04-12 18:04:48 +00:00
|
|
|
if (! ver_current->is_current_slic3r_supported()) {
|
2018-04-23 09:16:47 +00:00
|
|
|
BOOST_LOG_TRIVIAL(warning) << "Current Slic3r incompatible with installed bundle: " << bundle_path.string();
|
2018-04-24 16:06:42 +00:00
|
|
|
updates.incompats.emplace_back(std::move(bundle_path), *ver_current);
|
2018-04-12 18:04:48 +00:00
|
|
|
} else if (recommended->config_version > ver_current->config_version) {
|
|
|
|
// Config bundle update situation
|
|
|
|
|
2018-04-23 11:58:03 +00:00
|
|
|
// Check if the update is already present in a snapshot
|
|
|
|
const auto recommended_snap = SnapshotDB::singleton().snapshot_with_vendor_preset(vp.name, recommended->config_version);
|
|
|
|
if (recommended_snap != SnapshotDB::singleton().end()) {
|
|
|
|
BOOST_LOG_TRIVIAL(info) << boost::format("Bundle update %1% %2% already found in snapshot %3%, skipping...")
|
|
|
|
% vp.name
|
|
|
|
% recommended->config_version.to_string()
|
|
|
|
% recommended_snap->id;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2018-06-19 16:26:38 +00:00
|
|
|
auto path_src = cache_path / (idx.vendor() + ".ini");
|
2018-07-21 07:32:45 +00:00
|
|
|
auto path_in_rsrc = rsrc_path / (idx.vendor() + ".ini");
|
2018-06-19 16:26:38 +00:00
|
|
|
if (! fs::exists(path_src)) {
|
|
|
|
if (! fs::exists(path_in_rsrc)) {
|
|
|
|
BOOST_LOG_TRIVIAL(warning) << boost::format("Index for vendor %1% indicates update, but bundle found in neither cache nor resources")
|
2018-07-21 07:32:45 +00:00
|
|
|
% idx.vendor();
|
2018-06-19 16:26:38 +00:00
|
|
|
continue;
|
|
|
|
} else {
|
|
|
|
path_src = std::move(path_in_rsrc);
|
2018-07-21 07:32:45 +00:00
|
|
|
path_in_rsrc.clear();
|
2018-06-19 16:26:38 +00:00
|
|
|
}
|
2018-04-13 13:08:58 +00:00
|
|
|
}
|
|
|
|
|
2018-07-21 07:32:45 +00:00
|
|
|
auto new_vp = VendorProfile::from_ini(path_src, false);
|
|
|
|
bool found = false;
|
2018-06-19 16:26:38 +00:00
|
|
|
if (new_vp.config_version == recommended->config_version) {
|
|
|
|
updates.updates.emplace_back(std::move(path_src), std::move(bundle_path), *recommended);
|
2018-07-21 07:32:45 +00:00
|
|
|
found = true;
|
|
|
|
} else if (! path_in_rsrc.empty() && fs::exists(path_in_rsrc)) {
|
|
|
|
new_vp = VendorProfile::from_ini(path_in_rsrc, false);
|
|
|
|
if (new_vp.config_version == recommended->config_version) {
|
|
|
|
updates.updates.emplace_back(std::move(path_in_rsrc), std::move(bundle_path), *recommended);
|
|
|
|
found = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (! found)
|
2018-06-19 16:26:38 +00:00
|
|
|
BOOST_LOG_TRIVIAL(warning) << boost::format("Index for vendor %1% indicates update (%2%) but the new bundle was found neither in cache nor resources")
|
2018-05-02 15:13:39 +00:00
|
|
|
% idx.vendor()
|
2018-06-19 16:26:38 +00:00
|
|
|
% recommended->config_version.to_string();
|
2018-04-12 18:04:48 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return updates;
|
|
|
|
}
|
|
|
|
|
2018-04-18 09:40:43 +00:00
|
|
|
void PresetUpdater::priv::perform_updates(Updates &&updates, bool snapshot) const
|
2018-04-16 14:52:11 +00:00
|
|
|
{
|
2018-04-24 16:06:42 +00:00
|
|
|
if (updates.incompats.size() > 0) {
|
|
|
|
if (snapshot) {
|
|
|
|
BOOST_LOG_TRIVIAL(info) << "Taking a snapshot...";
|
2018-09-20 23:33:41 +00:00
|
|
|
SnapshotDB::singleton().take_snapshot(*GUI::wxGetApp().app_config, Snapshot::SNAPSHOT_DOWNGRADE);
|
2018-04-24 16:06:42 +00:00
|
|
|
}
|
2018-04-23 09:16:47 +00:00
|
|
|
|
2018-04-24 16:06:42 +00:00
|
|
|
BOOST_LOG_TRIVIAL(info) << boost::format("Deleting %1% incompatible bundles") % updates.incompats.size();
|
|
|
|
|
|
|
|
for (const auto &incompat : updates.incompats) {
|
|
|
|
BOOST_LOG_TRIVIAL(info) << '\t' << incompat;
|
|
|
|
fs::remove(incompat.bundle);
|
|
|
|
}
|
2018-04-18 09:40:43 +00:00
|
|
|
}
|
2018-04-24 16:06:42 +00:00
|
|
|
else if (updates.updates.size() > 0) {
|
|
|
|
if (snapshot) {
|
|
|
|
BOOST_LOG_TRIVIAL(info) << "Taking a snapshot...";
|
2018-09-20 23:33:41 +00:00
|
|
|
SnapshotDB::singleton().take_snapshot(*GUI::wxGetApp().app_config, Snapshot::SNAPSHOT_UPGRADE);
|
2018-04-24 16:06:42 +00:00
|
|
|
}
|
2018-04-16 14:52:11 +00:00
|
|
|
|
2018-04-24 16:06:42 +00:00
|
|
|
BOOST_LOG_TRIVIAL(info) << boost::format("Performing %1% updates") % updates.updates.size();
|
2018-04-23 09:16:47 +00:00
|
|
|
|
2018-04-24 16:06:42 +00:00
|
|
|
for (const auto &update : updates.updates) {
|
|
|
|
BOOST_LOG_TRIVIAL(info) << '\t' << update;
|
2018-04-16 14:52:11 +00:00
|
|
|
|
2018-06-11 15:34:06 +00:00
|
|
|
copy_file(update.source, update.target);
|
2018-04-16 14:52:11 +00:00
|
|
|
|
2018-04-24 16:06:42 +00:00
|
|
|
PresetBundle bundle;
|
|
|
|
bundle.load_configbundle(update.target.string(), PresetBundle::LOAD_CFGBNDLE_SYSTEM);
|
2018-04-16 14:52:11 +00:00
|
|
|
|
2018-05-18 12:58:24 +00:00
|
|
|
BOOST_LOG_TRIVIAL(info) << boost::format("Deleting %1% conflicting presets")
|
|
|
|
% (bundle.prints.size() + bundle.filaments.size() + bundle.printers.size());
|
|
|
|
|
2018-04-24 16:06:42 +00:00
|
|
|
auto preset_remover = [](const Preset &preset) {
|
2018-05-18 12:58:24 +00:00
|
|
|
BOOST_LOG_TRIVIAL(info) << '\t' << preset.file;
|
2018-04-24 16:06:42 +00:00
|
|
|
fs::remove(preset.file);
|
|
|
|
};
|
|
|
|
|
|
|
|
for (const auto &preset : bundle.prints) { preset_remover(preset); }
|
|
|
|
for (const auto &preset : bundle.filaments) { preset_remover(preset); }
|
|
|
|
for (const auto &preset : bundle.printers) { preset_remover(preset); }
|
2018-05-16 17:41:35 +00:00
|
|
|
|
|
|
|
// Also apply the `obsolete_presets` property, removing obsolete ini files
|
2018-05-18 12:58:24 +00:00
|
|
|
|
|
|
|
BOOST_LOG_TRIVIAL(info) << boost::format("Deleting %1% obsolete presets")
|
|
|
|
% (bundle.obsolete_presets.prints.size() + bundle.obsolete_presets.filaments.size() + bundle.obsolete_presets.printers.size());
|
|
|
|
|
2018-05-16 17:41:35 +00:00
|
|
|
auto obsolete_remover = [](const char *subdir, const std::string &preset) {
|
|
|
|
auto path = fs::path(Slic3r::data_dir()) / subdir / preset;
|
|
|
|
path += ".ini";
|
2018-05-18 12:58:24 +00:00
|
|
|
BOOST_LOG_TRIVIAL(info) << '\t' << path.string();
|
2018-05-16 17:41:35 +00:00
|
|
|
fs::remove(path);
|
|
|
|
};
|
|
|
|
|
|
|
|
for (const auto &name : bundle.obsolete_presets.prints) { obsolete_remover("print", name); }
|
|
|
|
for (const auto &name : bundle.obsolete_presets.filaments) { obsolete_remover("filament", name); }
|
2018-07-31 13:09:57 +00:00
|
|
|
for (const auto &name : bundle.obsolete_presets.filaments) { obsolete_remover("sla_material", name); }
|
2018-05-16 17:41:35 +00:00
|
|
|
for (const auto &name : bundle.obsolete_presets.printers) { obsolete_remover("printer", name); }
|
2018-04-24 16:06:42 +00:00
|
|
|
}
|
2018-04-16 14:52:11 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-06-11 15:34:06 +00:00
|
|
|
void PresetUpdater::priv::copy_file(const fs::path &source, const fs::path &target)
|
|
|
|
{
|
|
|
|
static const auto perms = fs::owner_read | fs::owner_write | fs::group_read | fs::others_read; // aka 644
|
|
|
|
|
|
|
|
// Make sure the file has correct permission both before and after we copy over it
|
2018-06-11 16:30:40 +00:00
|
|
|
if (fs::exists(target)) {
|
|
|
|
fs::permissions(target, perms);
|
|
|
|
}
|
2018-06-11 15:34:06 +00:00
|
|
|
fs::copy_file(source, target, fs::copy_option::overwrite_if_exists);
|
|
|
|
fs::permissions(target, perms);
|
|
|
|
}
|
|
|
|
|
2018-04-12 18:04:48 +00:00
|
|
|
|
2018-09-20 23:33:41 +00:00
|
|
|
PresetUpdater::PresetUpdater() :
|
|
|
|
p(new priv())
|
2018-04-13 13:08:58 +00:00
|
|
|
{}
|
2018-03-28 09:36:36 +00:00
|
|
|
|
|
|
|
|
|
|
|
// Public
|
|
|
|
|
|
|
|
PresetUpdater::~PresetUpdater()
|
|
|
|
{
|
|
|
|
if (p && p->thread.joinable()) {
|
2018-04-23 09:16:47 +00:00
|
|
|
// This will stop transfers being done by the thread, if any.
|
|
|
|
// Cancelling takes some time, but should complete soon enough.
|
2018-03-28 09:36:36 +00:00
|
|
|
p->cancel = true;
|
|
|
|
p->thread.join();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-17 14:59:53 +00:00
|
|
|
void PresetUpdater::sync(PresetBundle *preset_bundle)
|
2018-03-28 09:36:36 +00:00
|
|
|
{
|
2018-09-20 23:33:41 +00:00
|
|
|
p->set_download_prefs(GUI::wxGetApp().app_config);
|
2018-04-13 13:08:58 +00:00
|
|
|
if (!p->enabled_version_check && !p->enabled_config_update) { return; }
|
2018-03-28 09:36:36 +00:00
|
|
|
|
|
|
|
// Copy the whole vendors data for use in the background thread
|
|
|
|
// Unfortunatelly as of C++11, it needs to be copied again
|
|
|
|
// into the closure (but perhaps the compiler can elide this).
|
|
|
|
std::set<VendorProfile> vendors = preset_bundle->vendors;
|
|
|
|
|
|
|
|
p->thread = std::move(std::thread([this, vendors]() {
|
2018-04-13 13:08:58 +00:00
|
|
|
this->p->prune_tmps();
|
|
|
|
this->p->sync_version();
|
|
|
|
this->p->sync_config(std::move(vendors));
|
2018-03-28 09:36:36 +00:00
|
|
|
}));
|
|
|
|
}
|
|
|
|
|
2018-04-17 14:59:53 +00:00
|
|
|
void PresetUpdater::slic3r_update_notify()
|
|
|
|
{
|
2018-04-23 09:16:47 +00:00
|
|
|
if (! p->enabled_version_check) { return; }
|
|
|
|
|
|
|
|
if (p->had_config_update) {
|
|
|
|
BOOST_LOG_TRIVIAL(info) << "New Slic3r version available, but there was a configuration update, notification won't be displayed";
|
|
|
|
return;
|
|
|
|
}
|
2018-04-17 14:59:53 +00:00
|
|
|
|
2018-09-20 23:33:41 +00:00
|
|
|
auto* app_config = GUI::wxGetApp().app_config;
|
2018-04-17 14:59:53 +00:00
|
|
|
const auto ver_slic3r = Semver::parse(SLIC3R_VERSION);
|
2018-04-20 12:53:11 +00:00
|
|
|
const auto ver_online_str = app_config->get("version_online");
|
|
|
|
const auto ver_online = Semver::parse(ver_online_str);
|
|
|
|
const auto ver_online_seen = Semver::parse(app_config->get("version_online_seen"));
|
2018-04-17 14:59:53 +00:00
|
|
|
if (! ver_slic3r) {
|
|
|
|
throw std::runtime_error("Could not parse Slic3r version string: " SLIC3R_VERSION);
|
|
|
|
}
|
|
|
|
|
2018-04-20 12:53:11 +00:00
|
|
|
if (ver_online) {
|
|
|
|
// Only display the notification if the version available online is newer AND if we haven't seen it before
|
|
|
|
if (*ver_online > *ver_slic3r && (! ver_online_seen || *ver_online_seen < *ver_online)) {
|
2018-04-24 16:06:42 +00:00
|
|
|
GUI::MsgUpdateSlic3r notification(*ver_slic3r, *ver_online);
|
2018-04-20 12:53:11 +00:00
|
|
|
notification.ShowModal();
|
|
|
|
if (notification.disable_version_check()) {
|
|
|
|
app_config->set("version_check", "0");
|
|
|
|
p->enabled_version_check = false;
|
|
|
|
}
|
2018-04-17 14:59:53 +00:00
|
|
|
}
|
2018-04-20 12:53:11 +00:00
|
|
|
app_config->set("version_online_seen", ver_online_str);
|
2018-04-17 14:59:53 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-24 16:06:42 +00:00
|
|
|
bool PresetUpdater::config_update() const
|
2018-04-11 11:12:08 +00:00
|
|
|
{
|
2018-04-24 16:06:42 +00:00
|
|
|
if (! p->enabled_config_update) { return true; }
|
2018-04-13 13:08:58 +00:00
|
|
|
|
2018-04-23 11:58:03 +00:00
|
|
|
auto updates = p->get_config_updates();
|
2018-04-24 16:06:42 +00:00
|
|
|
if (updates.incompats.size() > 0) {
|
|
|
|
BOOST_LOG_TRIVIAL(info) << boost::format("%1% bundles incompatible. Asking for action...") % updates.incompats.size();
|
|
|
|
|
2018-04-24 16:40:06 +00:00
|
|
|
std::unordered_map<std::string, wxString> incompats_map;
|
2018-04-24 16:06:42 +00:00
|
|
|
for (const auto &incompat : updates.incompats) {
|
|
|
|
auto vendor = incompat.name();
|
2018-07-23 10:34:07 +00:00
|
|
|
|
|
|
|
const auto min_slic3r = incompat.version.min_slic3r_version;
|
|
|
|
const auto max_slic3r = incompat.version.max_slic3r_version;
|
|
|
|
wxString restrictions;
|
|
|
|
if (min_slic3r != Semver::zero() && max_slic3r != Semver::inf()) {
|
|
|
|
restrictions = wxString::Format(_(L("requires min. %s and max. %s")),
|
|
|
|
min_slic3r.to_string(),
|
|
|
|
max_slic3r.to_string()
|
|
|
|
);
|
|
|
|
} else if (min_slic3r != Semver::zero()) {
|
|
|
|
restrictions = wxString::Format(_(L("requires min. %s")), min_slic3r.to_string());
|
|
|
|
} else {
|
|
|
|
restrictions = wxString::Format(_(L("requires max. %s")), max_slic3r.to_string());
|
|
|
|
}
|
|
|
|
|
2018-04-24 16:40:06 +00:00
|
|
|
incompats_map.emplace(std::make_pair(std::move(vendor), std::move(restrictions)));
|
2018-04-24 16:06:42 +00:00
|
|
|
}
|
2018-04-12 18:04:48 +00:00
|
|
|
|
2018-06-22 13:27:04 +00:00
|
|
|
p->had_config_update = true; // This needs to be done before a dialog is shown because of OnIdle() + CallAfter() in Perl
|
|
|
|
|
2018-04-24 16:06:42 +00:00
|
|
|
GUI::MsgDataIncompatible dlg(std::move(incompats_map));
|
|
|
|
const auto res = dlg.ShowModal();
|
|
|
|
if (res == wxID_REPLACE) {
|
|
|
|
BOOST_LOG_TRIVIAL(info) << "User wants to re-configure...";
|
|
|
|
p->perform_updates(std::move(updates));
|
|
|
|
GUI::ConfigWizard wizard(nullptr, GUI::ConfigWizard::RR_DATA_INCOMPAT);
|
2018-07-23 10:34:07 +00:00
|
|
|
if (! wizard.run(GUI::get_preset_bundle(), this)) {
|
2018-04-24 16:06:42 +00:00
|
|
|
return false;
|
|
|
|
}
|
2018-07-23 12:19:40 +00:00
|
|
|
GUI::load_current_presets();
|
2018-04-24 16:06:42 +00:00
|
|
|
} else {
|
|
|
|
BOOST_LOG_TRIVIAL(info) << "User wants to exit Slic3r, bye...";
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (updates.updates.size() > 0) {
|
|
|
|
BOOST_LOG_TRIVIAL(info) << boost::format("Update of %1% bundles available. Asking for confirmation ...") % updates.updates.size();
|
2018-04-12 18:04:48 +00:00
|
|
|
|
2018-04-24 16:06:42 +00:00
|
|
|
std::unordered_map<std::string, std::string> updates_map;
|
|
|
|
for (const auto &update : updates.updates) {
|
|
|
|
auto vendor = update.name();
|
|
|
|
auto ver_str = update.version.config_version.to_string();
|
2018-04-12 18:04:48 +00:00
|
|
|
if (! update.version.comment.empty()) {
|
2018-04-24 16:06:42 +00:00
|
|
|
ver_str += std::string(" (") + update.version.comment + ")";
|
2018-04-12 18:04:48 +00:00
|
|
|
}
|
2018-04-24 16:40:06 +00:00
|
|
|
updates_map.emplace(std::make_pair(std::move(vendor), std::move(ver_str)));
|
2018-04-12 18:04:48 +00:00
|
|
|
}
|
2018-04-11 11:12:08 +00:00
|
|
|
|
2018-06-22 13:27:04 +00:00
|
|
|
p->had_config_update = true; // Ditto, see above
|
|
|
|
|
2018-04-24 16:06:42 +00:00
|
|
|
GUI::MsgUpdateConfig dlg(std::move(updates_map));
|
|
|
|
|
2018-04-12 18:04:48 +00:00
|
|
|
const auto res = dlg.ShowModal();
|
2018-04-25 13:20:46 +00:00
|
|
|
if (res == wxID_OK) {
|
2018-04-23 09:16:47 +00:00
|
|
|
BOOST_LOG_TRIVIAL(debug) << "User agreed to perform the update";
|
2018-04-17 14:59:53 +00:00
|
|
|
p->perform_updates(std::move(updates));
|
2018-06-06 08:52:19 +00:00
|
|
|
|
|
|
|
// Reload global configuration
|
2018-09-20 23:33:41 +00:00
|
|
|
auto *app_config = GUI::wxGetApp().app_config;
|
2018-06-06 08:52:19 +00:00
|
|
|
GUI::get_preset_bundle()->load_presets(*app_config);
|
|
|
|
GUI::load_current_presets();
|
2018-04-23 09:16:47 +00:00
|
|
|
} else {
|
|
|
|
BOOST_LOG_TRIVIAL(info) << "User refused the update";
|
2018-04-16 14:52:11 +00:00
|
|
|
}
|
2018-04-23 09:16:47 +00:00
|
|
|
} else {
|
|
|
|
BOOST_LOG_TRIVIAL(info) << "No configuration updates available.";
|
2018-04-16 14:52:11 +00:00
|
|
|
}
|
2018-04-24 16:06:42 +00:00
|
|
|
|
|
|
|
return true;
|
2018-04-16 14:52:11 +00:00
|
|
|
}
|
2018-04-12 18:04:48 +00:00
|
|
|
|
2018-04-24 16:06:42 +00:00
|
|
|
void PresetUpdater::install_bundles_rsrc(std::vector<std::string> bundles, bool snapshot) const
|
2018-04-16 14:52:11 +00:00
|
|
|
{
|
|
|
|
Updates updates;
|
2018-04-12 18:04:48 +00:00
|
|
|
|
2018-04-23 09:16:47 +00:00
|
|
|
BOOST_LOG_TRIVIAL(info) << boost::format("Installing %1% bundles from resources ...") % bundles.size();
|
|
|
|
|
2018-04-16 14:52:11 +00:00
|
|
|
for (const auto &bundle : bundles) {
|
|
|
|
auto path_in_rsrc = p->rsrc_path / bundle;
|
|
|
|
auto path_in_vendors = p->vendor_path / bundle;
|
2018-04-24 16:06:42 +00:00
|
|
|
updates.updates.emplace_back(std::move(path_in_rsrc), std::move(path_in_vendors), Version());
|
2018-04-11 11:12:08 +00:00
|
|
|
}
|
2018-04-16 14:52:11 +00:00
|
|
|
|
2018-04-18 09:40:43 +00:00
|
|
|
p->perform_updates(std::move(updates), snapshot);
|
2018-04-11 11:12:08 +00:00
|
|
|
}
|
|
|
|
|
2018-03-28 09:36:36 +00:00
|
|
|
|
|
|
|
}
|