/* Version management * * Copyright (c) 2020 The OSCAR Team * * This file is subject to the terms and conditions of the GNU General Public * License. See the file COPYING in the main directory of the source code * for more details. */ #include "version.h" #include "git_info.h" #include "SleepLib/common.h" #include const int major_version = 1; // incompatible API changes const int minor_version = 1; // new features that don't break things const int revision_number = 0; // bugfixes, revisions const QString ReleaseStatus = "beta"; // testing/nightly/unstable, beta/untamed, rc/almost, r/stable #include "build_number.h" #define VERSION "1.1.0-beta-1" const QString VersionString = QString("%1.%2.%3-%4-%5").arg(major_version).arg(minor_version).arg(revision_number).arg(ReleaseStatus).arg(build_number); const QString ShortVersionString = QString("%1.%2.%3").arg(major_version).arg(minor_version).arg(revision_number); #ifdef Q_OS_MAC const QString PlatformString = "MacOSX"; #elif defined(Q_OS_WIN32) const QString PlatformString = "Win32"; #elif defined(Q_OS_WIN64) const QString PlatformString = "Win64"; #elif defined(Q_OS_LINUX) const QString PlatformString = "Linux"; #elif defined(Q_OS_HAIKU) const QString PlatformString = "Haiku"; #endif const QString & gitRevision() { return GIT_REVISION; } const QString & gitBranch() { return GIT_BRANCH; } QString getBranchVersion() { QString version = STR_TR_AppVersion; if (!((ReleaseStatus.compare("r", Qt::CaseInsensitive)==0) || (ReleaseStatus.compare("release", Qt::CaseInsensitive)==0))) version += " ("+GIT_REVISION + ")"; if (GIT_BRANCH != "master") { version += " [Branch: " + GIT_BRANCH + "]"; } #ifdef BROKEN_OPENGL_BUILD version += " ["+CSTR_GFX_BrokenGL+"]"; #endif return version; } QString getPrereleaseSuffix() { QString suffix; // Append branch if there is a branch specified if (gitBranch() != "master") { suffix += "-"+gitBranch(); } // Append "-test" if not release else if (!((ReleaseStatus.compare("r", Qt::CaseInsensitive)==0) || (ReleaseStatus.compare("rc", Qt::CaseInsensitive)==0) )) { suffix += "-test"; } return suffix; } int checkVersionStatus(QString statusstr) { bool ok; // because Qt Install Framework is dumb and doesn't handle beta/release strings in version numbers, // so we store them numerically instead int v =statusstr.toInt(&ok); if (ok) { return v; } if ((statusstr.compare("testing", Qt::CaseInsensitive) == 0) || (statusstr.compare("unstable", Qt::CaseInsensitive) == 0)) return 0; else if ((statusstr.compare("beta", Qt::CaseInsensitive) == 0) || (statusstr.compare("untamed", Qt::CaseInsensitive) == 0)) return 1; else if ((statusstr.compare("rc", Qt::CaseInsensitive) == 0) || (statusstr.compare("almost", Qt::CaseInsensitive) == 0)) return 2; else if ((statusstr.compare("r", Qt::CaseInsensitive) == 0) || (statusstr.compare("stable", Qt::CaseInsensitive) == 0)) return 3; // anything else is considered a test build return 0; } struct VersionStruct { short major; short minor; short revision; short status; short build; }; VersionStruct parseVersion(QString versionstring) { static VersionStruct version; QStringList parts = versionstring.split("."); bool ok, dodgy = false; if (parts.size() < 3) dodgy = true; short major = parts[0].toInt(&ok); if (!ok) dodgy = true; short minor = parts[1].toInt(&ok); if (!ok) dodgy = true; QStringList patchver = parts[2].split("-"); if (patchver.size() < 3) dodgy = true; short rev = patchver[0].toInt(&ok); if (!ok) dodgy = true; short build = patchver[2].toInt(&ok); if (!ok) dodgy = true; int status = checkVersionStatus(patchver[1]); if (!dodgy) { version.major = major; version.minor = minor; version.revision = rev; version.status = status; version.build = build; } return version; } // Compare supplied version string with current version // < 0 = this one is newer or version supplied is dodgy, 0 = same, and > 0 there is a newer version int compareVersion(const QString & version) { // v1.0.0-beta-2 QStringList parts = version.split("."); bool ok; if (parts.size() < 3) { // dodgy version string supplied. return -1; } int major = parts[0].toInt(&ok); if (!ok) return -1; int minor = parts[1].toInt(&ok); if (!ok) return -1; if (major > major_version) { return 1; } else if (major < major_version) { return -1; } if (minor > minor_version) { return 1; } else if (minor < minor_version) { return -1; } int build_index = 1; int build = 0; int status = 0; QStringList patchver = parts[2].split("-"); if (patchver.size() >= 3) { build_index = 2; status = checkVersionStatus(patchver[1]); } else if (patchver.size() < 2) { return -1; // dodgy version string supplied. } int rev = patchver[0].toInt(&ok); if (!ok) return -1; if (rev > revision_number) { return 1; } else if (rev < revision_number) { return -1; } build = patchver[build_index].toInt(&ok); if (!ok) return -1; int rstatus = checkVersionStatus(ReleaseStatus); if (patchver.size() == 3) { // read it if it's actually present. } if (status > rstatus) { return 1; } else if (status < rstatus) { return -1; } if (build > build_number) { return 1; } else if (build < build_number) { return -1; } // Versions match return 0; } bool isReleaseVersion() { return (ReleaseStatus == "r"); } static const Version s_Version(VERSION); const Version & getVersion() { return s_Version; } Version::Version(const QString & version_string) : mString(version_string), mIsValid(false) { ParseSemanticVersion(); FixLegacyVersions(); } Version::operator const QString &() const { return mString; } // Parse a version string as specified by Semantic Versioning 2.0.0, see https://semver.org/spec/v2.0.0.html void Version::ParseSemanticVersion() { // Use a C++11 raw string literal to keep the regular expression (mostly) legible. static const QRegularExpression re(R"(^(?P0|[1-9]\d*)\.(?P0|[1-9]\d*)\.(?P0|[1-9]\d*)(?:-(?P(?:0|[1-9]\d*|\d*[a-zA-Z-][0-9a-zA-Z-]*)(?:\.(?:0|[1-9]\d*|\d*[a-zA-Z-][0-9a-zA-Z-]*))*))?(?:\+(?P[0-9a-zA-Z-]+(?:\.[0-9a-zA-Z-]+)*))?$)"); QRegularExpressionMatch match = re.match(mString); while (match.hasMatch()) { bool ok; mMajor = match.captured("major").toInt(&ok); if (!ok) break; mMinor = match.captured("minor").toInt(&ok); if (!ok) break; mPatch = match.captured("patch").toInt(&ok); if (!ok) break; mPrerelease = match.captured("prerelease"); mBuild = match.captured("buildmetadata"); mIsValid = true; break; } } // Deal with non-Semantic-Versioning numbers used before 1.1.0-beta-2 to make sure they // will have proper (lower) precedence compared to later versions. // // TODO: THIS CAN PROBABLY BE REMOVED AFTER THE RELEASE OF 1.1.0, since the release // version will take precedence over all 1.1.0 prereleases, as well as 1.0.1 of any // release status. // // Right now we just need to make sure that 1.1.0-beta versions take precedence over // 1.1.0-testing. void Version::FixLegacyVersions() { if (mIsValid) { // Replace prerelease "testing" with "alpha" for backwards compatibility with 1.1.0-testing-* // versions: otherwise "testing" would take precedence over "beta". mPrerelease.replace("testing", "alpha"); // Technically the use of "r1" in "1.0.1-r1" could also be corrected, as the code // will incorrectly consider that release version to be a prerelease, but it doesn't // matter because 1.1.0 and later will take precedence either way. } } // Compare two version instances in accordance with Semantic Versionin 2.0.0 precedence rules. int Version::Compare(const Version & a, const Version & b) { int diff; diff = a.mMajor - b.mMajor; if (diff) return diff; diff = a.mMinor - b.mMinor; if (diff) return diff; diff = a.mPatch - b.mPatch; if (diff) return diff; // Version numbers are equal, now check prerelease status: if (a.IsReleaseVersion() && b.IsReleaseVersion()) return 0; // A pre-release version has lower prededence than a release version. diff = a.IsReleaseVersion() - b.IsReleaseVersion(); if (diff) return diff; // Both are prerelease versions, compare them: // The prerelease version may contain a series of dot-separated identifiers, // each of which is compared. QStringList ap = a.mPrerelease.split("."); QStringList bp = b.mPrerelease.split("."); int max = qMin(ap.size(), bp.size()); for (int i = 0; i < max; i++) { bool a_is_num, b_is_num; int ai = ap[i].toInt(&a_is_num); int bi = bp[i].toInt(&b_is_num); // Numeric identifiers always have lower precedence than non-numeric. diff = b_is_num - a_is_num; if (diff) return diff; if (a_is_num) { // Numeric identifiers are compared numerically. diff = ai - bi; if (diff) return diff; } else { // Non-numeric identifiers are compared lexically. diff = ap[i].compare(bp[i]); if (diff) return diff; } } // A larger set of pre-release fields has higher precedence (if the above were equal). diff = ap.size() - bp.size(); // We ignore build metadata in comparing semantic versions. return diff; }