2018-04-06 14:49:33 +00:00
|
|
|
#include "Version.hpp"
|
|
|
|
|
2019-03-27 11:14:34 +00:00
|
|
|
#include <cctype>
|
|
|
|
|
2018-04-06 14:49:33 +00:00
|
|
|
#include <boost/algorithm/string/predicate.hpp>
|
|
|
|
#include <boost/algorithm/string/trim.hpp>
|
|
|
|
#include <boost/nowide/fstream.hpp>
|
|
|
|
|
2018-12-06 11:52:28 +00:00
|
|
|
#include "libslic3r/libslic3r.h"
|
|
|
|
#include "libslic3r/Config.hpp"
|
|
|
|
#include "libslic3r/FileParserError.hpp"
|
|
|
|
#include "libslic3r/Utils.hpp"
|
2018-04-06 14:49:33 +00:00
|
|
|
|
|
|
|
namespace Slic3r {
|
|
|
|
namespace GUI {
|
|
|
|
namespace Config {
|
|
|
|
|
2018-04-25 11:44:06 +00:00
|
|
|
static const Semver s_current_slic3r_semver(SLIC3R_VERSION);
|
2018-04-06 14:49:33 +00:00
|
|
|
|
2018-04-13 14:15:30 +00:00
|
|
|
// Optimized lexicographic compare of two pre-release versions, ignoring the numeric suffix.
|
|
|
|
static int compare_prerelease(const char *p1, const char *p2)
|
|
|
|
{
|
|
|
|
for (;;) {
|
|
|
|
char c1 = *p1 ++;
|
|
|
|
char c2 = *p2 ++;
|
|
|
|
bool a1 = std::isalpha(c1) && c1 != 0;
|
|
|
|
bool a2 = std::isalpha(c2) && c2 != 0;
|
|
|
|
if (a1) {
|
|
|
|
if (a2) {
|
|
|
|
if (c1 != c2)
|
|
|
|
return (c1 < c2) ? -1 : 1;
|
|
|
|
} else
|
|
|
|
return 1;
|
|
|
|
} else {
|
|
|
|
if (a2)
|
|
|
|
return -1;
|
|
|
|
else
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// This shall never happen.
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Version::is_slic3r_supported(const Semver &slic3r_version) const
|
|
|
|
{
|
|
|
|
if (! slic3r_version.in_range(min_slic3r_version, max_slic3r_version))
|
|
|
|
return false;
|
|
|
|
// Now verify, whether the configuration pre-release status is compatible with the Slic3r's pre-release status.
|
|
|
|
// Alpha Slic3r will happily load any configuration, while beta Slic3r will ignore alpha configurations etc.
|
|
|
|
const char *prerelease_slic3r = slic3r_version.prerelease();
|
|
|
|
const char *prerelease_config = this->config_version.prerelease();
|
|
|
|
if (prerelease_config == nullptr)
|
|
|
|
// Released config is always supported.
|
|
|
|
return true;
|
|
|
|
else if (prerelease_slic3r == nullptr)
|
|
|
|
// Released slic3r only supports released configs.
|
|
|
|
return false;
|
|
|
|
// Compare the pre-release status of Slic3r against the config.
|
|
|
|
// If the prerelease status of slic3r is lexicographically lower or equal
|
|
|
|
// to the prerelease status of the config, accept it.
|
|
|
|
return compare_prerelease(prerelease_slic3r, prerelease_config) != 1;
|
|
|
|
}
|
|
|
|
|
2018-04-06 14:49:33 +00:00
|
|
|
bool Version::is_current_slic3r_supported() const
|
|
|
|
{
|
2018-04-25 11:44:06 +00:00
|
|
|
return this->is_slic3r_supported(s_current_slic3r_semver);
|
2018-04-06 14:49:33 +00:00
|
|
|
}
|
|
|
|
|
2018-04-13 14:15:30 +00:00
|
|
|
#if 0
|
|
|
|
//TODO: This test should be moved to a unit test, once we have C++ unit tests in place.
|
|
|
|
static int version_test()
|
|
|
|
{
|
|
|
|
Version v;
|
|
|
|
v.config_version = *Semver::parse("1.1.2");
|
|
|
|
v.min_slic3r_version = *Semver::parse("1.38.0");
|
|
|
|
v.max_slic3r_version = Semver::inf();
|
|
|
|
assert(v.is_slic3r_supported(*Semver::parse("1.38.0")));
|
|
|
|
assert(! v.is_slic3r_supported(*Semver::parse("1.38.0-alpha")));
|
|
|
|
assert(! v.is_slic3r_supported(*Semver::parse("1.37.0-alpha")));
|
|
|
|
// Test the prerelease status.
|
|
|
|
assert(v.is_slic3r_supported(*Semver::parse("1.39.0-alpha")));
|
|
|
|
assert(v.is_slic3r_supported(*Semver::parse("1.39.0-alpha1")));
|
|
|
|
assert(v.is_slic3r_supported(*Semver::parse("1.39.0-alpha1")));
|
|
|
|
assert(v.is_slic3r_supported(*Semver::parse("1.39.0-beta")));
|
|
|
|
assert(v.is_slic3r_supported(*Semver::parse("1.39.0-beta1")));
|
|
|
|
assert(v.is_slic3r_supported(*Semver::parse("1.39.0-beta1")));
|
|
|
|
assert(v.is_slic3r_supported(*Semver::parse("1.39.0-rc2")));
|
|
|
|
assert(v.is_slic3r_supported(*Semver::parse("1.39.0")));
|
|
|
|
v.config_version = *Semver::parse("1.1.2-alpha");
|
|
|
|
assert(v.is_slic3r_supported(*Semver::parse("1.39.0-alpha")));
|
|
|
|
assert(v.is_slic3r_supported(*Semver::parse("1.39.0-alpha1")));
|
|
|
|
assert(! v.is_slic3r_supported(*Semver::parse("1.39.0-beta")));
|
|
|
|
assert(! v.is_slic3r_supported(*Semver::parse("1.39.0-beta1")));
|
|
|
|
assert(! v.is_slic3r_supported(*Semver::parse("1.39.0-beta1")));
|
|
|
|
assert(! v.is_slic3r_supported(*Semver::parse("1.39.0-rc2")));
|
|
|
|
assert(! v.is_slic3r_supported(*Semver::parse("1.39.0")));
|
|
|
|
v.config_version = *Semver::parse("1.1.2-alpha1");
|
|
|
|
assert(v.is_slic3r_supported(*Semver::parse("1.39.0-alpha")));
|
|
|
|
assert(v.is_slic3r_supported(*Semver::parse("1.39.0-alpha1")));
|
|
|
|
assert(! v.is_slic3r_supported(*Semver::parse("1.39.0-beta")));
|
|
|
|
assert(! v.is_slic3r_supported(*Semver::parse("1.39.0-beta1")));
|
|
|
|
assert(! v.is_slic3r_supported(*Semver::parse("1.39.0-beta1")));
|
|
|
|
assert(! v.is_slic3r_supported(*Semver::parse("1.39.0-rc2")));
|
|
|
|
assert(! v.is_slic3r_supported(*Semver::parse("1.39.0")));
|
|
|
|
v.config_version = *Semver::parse("1.1.2-beta");
|
|
|
|
assert(v.is_slic3r_supported(*Semver::parse("1.39.0-alpha")));
|
|
|
|
assert(v.is_slic3r_supported(*Semver::parse("1.39.0-alpha1")));
|
|
|
|
assert(v.is_slic3r_supported(*Semver::parse("1.39.0-beta")));
|
|
|
|
assert(v.is_slic3r_supported(*Semver::parse("1.39.0-beta1")));
|
|
|
|
assert(v.is_slic3r_supported(*Semver::parse("1.39.0-beta1")));
|
|
|
|
assert(! v.is_slic3r_supported(*Semver::parse("1.39.0-rc")));
|
|
|
|
assert(! v.is_slic3r_supported(*Semver::parse("1.39.0-rc2")));
|
|
|
|
assert(! v.is_slic3r_supported(*Semver::parse("1.39.0")));
|
|
|
|
v.config_version = *Semver::parse("1.1.2-rc");
|
|
|
|
assert(v.is_slic3r_supported(*Semver::parse("1.39.0-alpha")));
|
|
|
|
assert(v.is_slic3r_supported(*Semver::parse("1.39.0-alpha1")));
|
|
|
|
assert(v.is_slic3r_supported(*Semver::parse("1.39.0-beta")));
|
|
|
|
assert(v.is_slic3r_supported(*Semver::parse("1.39.0-beta1")));
|
|
|
|
assert(v.is_slic3r_supported(*Semver::parse("1.39.0-beta1")));
|
|
|
|
assert(v.is_slic3r_supported(*Semver::parse("1.39.0-rc")));
|
|
|
|
assert(v.is_slic3r_supported(*Semver::parse("1.39.0-rc2")));
|
|
|
|
assert(! v.is_slic3r_supported(*Semver::parse("1.39.0")));
|
|
|
|
v.config_version = *Semver::parse("1.1.2-rc2");
|
|
|
|
assert(v.is_slic3r_supported(*Semver::parse("1.39.0-alpha")));
|
|
|
|
assert(v.is_slic3r_supported(*Semver::parse("1.39.0-alpha1")));
|
|
|
|
assert(v.is_slic3r_supported(*Semver::parse("1.39.0-beta")));
|
|
|
|
assert(v.is_slic3r_supported(*Semver::parse("1.39.0-beta1")));
|
|
|
|
assert(v.is_slic3r_supported(*Semver::parse("1.39.0-beta1")));
|
|
|
|
assert(v.is_slic3r_supported(*Semver::parse("1.39.0-rc")));
|
|
|
|
assert(v.is_slic3r_supported(*Semver::parse("1.39.0-rc2")));
|
|
|
|
assert(! v.is_slic3r_supported(*Semver::parse("1.39.0")));
|
|
|
|
// Test the upper boundary.
|
|
|
|
v.config_version = *Semver::parse("1.1.2");
|
|
|
|
v.max_slic3r_version = *Semver::parse("1.39.3-beta1");
|
|
|
|
assert(v.is_slic3r_supported(*Semver::parse("1.38.0")));
|
|
|
|
assert(! v.is_slic3r_supported(*Semver::parse("1.38.0-alpha")));
|
|
|
|
assert(! v.is_slic3r_supported(*Semver::parse("1.38.0-alpha1")));
|
|
|
|
assert(! v.is_slic3r_supported(*Semver::parse("1.37.0-alpha")));
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
static int version_test_run = version_test();
|
|
|
|
#endif
|
|
|
|
|
2018-04-06 14:49:33 +00:00
|
|
|
inline char* left_trim(char *c)
|
|
|
|
{
|
|
|
|
for (; *c == ' ' || *c == '\t'; ++ c);
|
|
|
|
return c;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline char* right_trim(char *start)
|
|
|
|
{
|
|
|
|
char *end = start + strlen(start) - 1;
|
|
|
|
for (; end >= start && (*end == ' ' || *end == '\t'); -- end);
|
|
|
|
*(++ end) = 0;
|
|
|
|
return end;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline std::string unquote_value(char *value, char *end, const std::string &path, int idx_line)
|
|
|
|
{
|
|
|
|
std::string svalue;
|
|
|
|
if (value == end) {
|
|
|
|
// Empty string is a valid string.
|
|
|
|
} else if (*value == '"') {
|
2018-04-11 13:17:41 +00:00
|
|
|
if (++ value > -- end || *end != '"')
|
2018-04-06 14:49:33 +00:00
|
|
|
throw file_parser_error("String not enquoted correctly", path, idx_line);
|
|
|
|
*end = 0;
|
|
|
|
if (! unescape_string_cstyle(value, svalue))
|
|
|
|
throw file_parser_error("Invalid escape sequence inside a quoted string", path, idx_line);
|
2018-04-11 13:17:41 +00:00
|
|
|
} else
|
|
|
|
svalue.assign(value, end);
|
2018-04-06 14:49:33 +00:00
|
|
|
return svalue;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline std::string unquote_version_comment(char *value, char *end, const std::string &path, int idx_line)
|
|
|
|
{
|
|
|
|
std::string svalue;
|
|
|
|
if (value == end) {
|
|
|
|
// Empty string is a valid string.
|
|
|
|
} else if (*value == '"') {
|
2018-04-11 13:17:41 +00:00
|
|
|
if (++ value > -- end || *end != '"')
|
2018-04-06 14:49:33 +00:00
|
|
|
throw file_parser_error("Version comment not enquoted correctly", path, idx_line);
|
|
|
|
*end = 0;
|
|
|
|
if (! unescape_string_cstyle(value, svalue))
|
|
|
|
throw file_parser_error("Invalid escape sequence inside a quoted version comment", path, idx_line);
|
2018-04-11 13:17:41 +00:00
|
|
|
} else
|
|
|
|
svalue.assign(value, end);
|
2018-04-06 14:49:33 +00:00
|
|
|
return svalue;
|
|
|
|
}
|
|
|
|
|
2018-04-09 15:03:37 +00:00
|
|
|
size_t Index::load(const boost::filesystem::path &path)
|
2018-04-06 14:49:33 +00:00
|
|
|
{
|
|
|
|
m_configs.clear();
|
2018-04-09 15:03:37 +00:00
|
|
|
m_vendor = path.stem().string();
|
2019-06-03 09:31:32 +00:00
|
|
|
m_path = path;
|
2018-04-06 14:49:33 +00:00
|
|
|
|
2018-04-09 15:03:37 +00:00
|
|
|
boost::nowide::ifstream ifs(path.string());
|
2018-04-06 14:49:33 +00:00
|
|
|
std::string line;
|
|
|
|
size_t idx_line = 0;
|
|
|
|
Version ver;
|
|
|
|
while (std::getline(ifs, line)) {
|
2019-06-13 14:33:50 +00:00
|
|
|
#ifndef _MSVCVER
|
|
|
|
// On a Unix system, getline does not remove the trailing carriage returns, if the index is shared over a Windows filesystem. Remove them manually.
|
|
|
|
while (! line.empty() && line.back() == '\r')
|
|
|
|
line.pop_back();
|
|
|
|
#endif
|
2018-04-06 14:49:33 +00:00
|
|
|
++ idx_line;
|
|
|
|
// Skip the initial white spaces.
|
|
|
|
char *key = left_trim(const_cast<char*>(line.data()));
|
2018-04-11 13:17:41 +00:00
|
|
|
if (*key == '#')
|
|
|
|
// Skip a comment line.
|
|
|
|
continue;
|
2018-04-06 14:49:33 +00:00
|
|
|
// Right trim the line.
|
|
|
|
char *end = right_trim(key);
|
2018-04-11 13:17:41 +00:00
|
|
|
if (key == end)
|
|
|
|
// Skip an empty line.
|
|
|
|
continue;
|
2018-04-06 14:49:33 +00:00
|
|
|
// Keyword may only contain alphanumeric characters. Semantic version may in addition contain "+.-".
|
|
|
|
char *key_end = key;
|
2018-04-11 13:17:41 +00:00
|
|
|
bool maybe_semver = true;
|
|
|
|
for (; *key_end != 0; ++ key_end) {
|
|
|
|
if (std::isalnum(*key_end) || strchr("+.-", *key_end) != nullptr) {
|
|
|
|
// It may be a semver.
|
|
|
|
} else if (*key_end == '_') {
|
|
|
|
// Cannot be a semver, but it may be a key.
|
|
|
|
maybe_semver = false;
|
|
|
|
} else
|
|
|
|
// End of semver or keyword.
|
|
|
|
break;
|
2018-04-06 14:49:33 +00:00
|
|
|
}
|
2018-04-11 13:17:41 +00:00
|
|
|
if (*key_end != 0 && *key_end != ' ' && *key_end != '\t' && *key_end != '=')
|
2018-04-06 14:49:33 +00:00
|
|
|
throw file_parser_error("Invalid keyword or semantic version", path, idx_line);
|
2018-04-11 13:17:41 +00:00
|
|
|
char *value = left_trim(key_end);
|
|
|
|
bool key_value_pair = *value == '=';
|
|
|
|
if (key_value_pair)
|
|
|
|
value = left_trim(value + 1);
|
|
|
|
*key_end = 0;
|
2018-04-06 14:49:33 +00:00
|
|
|
boost::optional<Semver> semver;
|
|
|
|
if (maybe_semver)
|
|
|
|
semver = Semver::parse(key);
|
2018-04-11 13:17:41 +00:00
|
|
|
if (key_value_pair) {
|
2018-04-06 14:49:33 +00:00
|
|
|
if (semver)
|
2018-04-11 13:17:41 +00:00
|
|
|
throw file_parser_error("Key cannot be a semantic version", path, idx_line);\
|
2018-04-06 14:49:33 +00:00
|
|
|
// Verify validity of the key / value pair.
|
2018-04-11 13:17:41 +00:00
|
|
|
std::string svalue = unquote_value(value, end, path.string(), idx_line);
|
|
|
|
if (strcmp(key, "min_slic3r_version") == 0 || strcmp(key, "max_slic3r_version") == 0) {
|
2018-04-06 14:49:33 +00:00
|
|
|
if (! svalue.empty())
|
2018-04-11 13:17:41 +00:00
|
|
|
semver = Semver::parse(svalue);
|
2018-04-06 14:49:33 +00:00
|
|
|
if (! semver)
|
|
|
|
throw file_parser_error(std::string(key) + " must referece a valid semantic version", path, idx_line);
|
2018-04-11 13:17:41 +00:00
|
|
|
if (strcmp(key, "min_slic3r_version") == 0)
|
2018-04-06 14:49:33 +00:00
|
|
|
ver.min_slic3r_version = *semver;
|
|
|
|
else
|
|
|
|
ver.max_slic3r_version = *semver;
|
|
|
|
} else {
|
|
|
|
// Ignore unknown keys, as there may come new keys in the future.
|
|
|
|
}
|
2018-04-11 13:17:41 +00:00
|
|
|
continue;
|
2018-04-06 14:49:33 +00:00
|
|
|
}
|
|
|
|
if (! semver)
|
|
|
|
throw file_parser_error("Invalid semantic version", path, idx_line);
|
|
|
|
ver.config_version = *semver;
|
2018-04-09 15:03:37 +00:00
|
|
|
ver.comment = (end <= key_end) ? "" : unquote_version_comment(value, end, path.string(), idx_line);
|
2018-04-06 14:49:33 +00:00
|
|
|
m_configs.emplace_back(ver);
|
|
|
|
}
|
|
|
|
|
2018-04-09 15:03:37 +00:00
|
|
|
// Sort the configs by their version.
|
|
|
|
std::sort(m_configs.begin(), m_configs.end(), [](const Version &v1, const Version &v2) { return v1.config_version < v2.config_version; });
|
2018-04-06 14:49:33 +00:00
|
|
|
return m_configs.size();
|
|
|
|
}
|
|
|
|
|
2018-04-13 12:49:33 +00:00
|
|
|
Semver Index::version() const
|
|
|
|
{
|
|
|
|
Semver ver = Semver::zero();
|
|
|
|
for (const Version &cv : m_configs)
|
|
|
|
if (cv.config_version >= ver)
|
|
|
|
ver = cv.config_version;
|
|
|
|
return ver;
|
|
|
|
}
|
|
|
|
|
2018-04-12 18:04:48 +00:00
|
|
|
Index::const_iterator Index::find(const Semver &ver) const
|
2018-04-09 15:03:37 +00:00
|
|
|
{
|
|
|
|
Version key;
|
|
|
|
key.config_version = ver;
|
|
|
|
auto it = std::lower_bound(m_configs.begin(), m_configs.end(), key,
|
|
|
|
[](const Version &v1, const Version &v2) { return v1.config_version < v2.config_version; });
|
|
|
|
return (it == m_configs.end() || it->config_version == ver) ? it : m_configs.end();
|
|
|
|
}
|
|
|
|
|
2018-04-06 14:49:33 +00:00
|
|
|
Index::const_iterator Index::recommended() const
|
|
|
|
{
|
|
|
|
int idx = -1;
|
2018-04-13 13:08:58 +00:00
|
|
|
const_iterator highest = this->end();
|
2018-04-06 14:49:33 +00:00
|
|
|
for (const_iterator it = this->begin(); it != this->end(); ++ it)
|
|
|
|
if (it->is_current_slic3r_supported() &&
|
2018-04-13 13:08:58 +00:00
|
|
|
(highest == this->end() || highest->config_version < it->config_version))
|
2018-04-06 14:49:33 +00:00
|
|
|
highest = it;
|
|
|
|
return highest;
|
|
|
|
}
|
|
|
|
|
2018-04-09 15:03:37 +00:00
|
|
|
std::vector<Index> Index::load_db()
|
|
|
|
{
|
2018-04-12 18:04:48 +00:00
|
|
|
boost::filesystem::path cache_dir = boost::filesystem::path(Slic3r::data_dir()) / "cache";
|
2018-04-09 15:03:37 +00:00
|
|
|
|
|
|
|
std::vector<Index> index_db;
|
|
|
|
std::string errors_cummulative;
|
2018-04-12 18:04:48 +00:00
|
|
|
for (auto &dir_entry : boost::filesystem::directory_iterator(cache_dir))
|
2019-02-03 14:30:37 +00:00
|
|
|
if (Slic3r::is_idx_file(dir_entry)) {
|
2018-04-09 15:03:37 +00:00
|
|
|
Index idx;
|
|
|
|
try {
|
|
|
|
idx.load(dir_entry.path());
|
|
|
|
} catch (const std::runtime_error &err) {
|
|
|
|
errors_cummulative += err.what();
|
|
|
|
errors_cummulative += "\n";
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
index_db.emplace_back(std::move(idx));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (! errors_cummulative.empty())
|
|
|
|
throw std::runtime_error(errors_cummulative);
|
|
|
|
return index_db;
|
|
|
|
}
|
|
|
|
|
2018-04-06 14:49:33 +00:00
|
|
|
} // namespace Config
|
|
|
|
} // namespace GUI
|
|
|
|
} // namespace Slic3r
|