OSCAR-code/oscar/version.cpp
2020-01-15 17:00:21 -05:00

350 lines
9.8 KiB
C++

/* 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 <QRegularExpression>
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"(^(?P<major>0|[1-9]\d*)\.(?P<minor>0|[1-9]\d*)\.(?P<patch>0|[1-9]\d*)(?:-(?P<prerelease>(?: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<buildmetadata>[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;
}