2020-01-13 23:18:14 +00:00
|
|
|
/* Version management
|
|
|
|
*
|
2024-01-13 20:27:48 +00:00
|
|
|
* Copyright (c) 2019-2024 The OSCAR Team
|
2020-01-13 23:18:14 +00:00
|
|
|
*
|
|
|
|
* 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"
|
2020-01-16 22:37:43 +00:00
|
|
|
#include "VERSION"
|
2020-01-13 23:37:32 +00:00
|
|
|
#include "git_info.h"
|
2020-01-15 22:00:21 +00:00
|
|
|
#include <QRegularExpression>
|
|
|
|
|
2020-01-13 23:18:14 +00:00
|
|
|
|
2020-01-16 22:37:43 +00:00
|
|
|
// Initialize the Version instance with build metadata, if any.
|
|
|
|
#ifdef GIT_REVISION
|
|
|
|
#ifdef GIT_BRANCH
|
|
|
|
#define BUILD_METADATA "+" GIT_BRANCH "-" GIT_REVISION
|
|
|
|
#else
|
|
|
|
#define BUILD_METADATA "+" GIT_REVISION
|
|
|
|
#endif
|
|
|
|
#else
|
|
|
|
#define BUILD_METADATA ""
|
|
|
|
#endif
|
|
|
|
static const Version s_Version(VERSION BUILD_METADATA);
|
2020-01-13 23:37:32 +00:00
|
|
|
|
2020-01-16 18:58:18 +00:00
|
|
|
// Technically this is the date and time that version.cpp was compiled, but since
|
|
|
|
// it gets recompiled whenever git_info.h changes, it's about as close as we can
|
|
|
|
// come without forcing recompiles every single build.
|
|
|
|
static const QString s_BuildDateTime = __DATE__ " " __TIME__;
|
2020-01-13 23:37:32 +00:00
|
|
|
|
2020-01-16 21:06:11 +00:00
|
|
|
|
2020-01-15 18:18:00 +00:00
|
|
|
QString getPrereleaseSuffix()
|
|
|
|
{
|
|
|
|
QString suffix;
|
|
|
|
|
2020-01-17 20:59:29 +00:00
|
|
|
if (getVersion().IsReleaseVersion() || getVersion().PrereleaseType().compare("rc") == 0) {
|
|
|
|
// No suffix for release or rc versions.
|
|
|
|
suffix = "";
|
|
|
|
} else {
|
|
|
|
#ifdef GIT_TAG
|
|
|
|
// If this commit has a tag, then it's a full testing (alpha/beta/etc.) release.
|
|
|
|
// Put preferences/data in "-test".
|
|
|
|
suffix = "-test";
|
|
|
|
#else
|
|
|
|
// Otherwise it's a development build, which will be identified by its branch in most cases.
|
|
|
|
#ifdef GIT_BRANCH
|
|
|
|
suffix = "-" GIT_BRANCH;
|
2020-01-17 21:25:01 +00:00
|
|
|
#else
|
|
|
|
#ifdef GIT_REVISION
|
2020-01-17 20:59:29 +00:00
|
|
|
// If we've checked out an older version, we're in a headless state and not on any branch.
|
|
|
|
// If the older version was a previous testing release, it should be tagged, in which case
|
|
|
|
// it's treated as a testing release above.
|
|
|
|
//
|
|
|
|
// Otherwise this is probably being used for regression testing an older build.
|
|
|
|
suffix = "-" GIT_REVISION;
|
2020-01-17 21:25:01 +00:00
|
|
|
#else
|
2020-01-17 20:59:29 +00:00
|
|
|
// In theory someone might try to build a prerelease from a tarball, so we don't have any
|
|
|
|
// revision information. Just put it in an "-unreleased" sandbox.
|
|
|
|
suffix = "-unreleased";
|
2020-01-17 21:25:01 +00:00
|
|
|
#endif
|
2020-01-17 20:59:29 +00:00
|
|
|
#endif
|
|
|
|
#endif
|
2020-01-15 18:18:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return suffix;
|
|
|
|
}
|
|
|
|
|
2020-01-16 18:58:18 +00:00
|
|
|
const QString & getBuildDateTime()
|
|
|
|
{
|
|
|
|
return s_BuildDateTime;
|
|
|
|
}
|
|
|
|
|
2020-01-15 22:00:21 +00:00
|
|
|
const Version & getVersion()
|
|
|
|
{
|
|
|
|
return s_Version;
|
|
|
|
}
|
|
|
|
|
2020-01-16 16:33:39 +00:00
|
|
|
// Alternate formatting of the version string for display or logging
|
|
|
|
const QString Version::minimalString() const
|
|
|
|
{
|
Update version display throughout to use the new information and be consistent.
The full version now includes the build/git information embedded within
it as build metadata according to the Semantic Versioning 2.0.0 spec,
for example: "1.1.0-beta-1+branch-name-a1b2c3d".
Now the full version string, with all detail is always displayed
EXCEPT for release versions, in which case just the simple version
number ("1.1.0") is displayed in the primary UI.
- Main window title: simple version for release versions, full version
string otherwise
- Notifications: same as main window title
- System tray: same as main window title
- About window title: same as main window title
- About window release notes: always include full version string
- Reports: always include full version string
- Under the logo (about dialog, profile selector, new profile
window): removed, as it is largely redundant and can
interfere with the window geometry.
- Database upgrade alert: same as main window title
- Database newer alert: same as main window title
The full version string is also included within the preference and
profile .xml files, but because build metadata is ignored in version
comparisons, differences in builds will not cause any spurious
alerts. However, changes in prerelease versions will continue to
be significant, as they should be.
2020-01-16 18:05:55 +00:00
|
|
|
return toString().section("+", 0, 0);
|
2020-01-16 16:33:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
const QString Version::displayString() const
|
|
|
|
{
|
|
|
|
if (IsReleaseVersion())
|
|
|
|
return minimalString();
|
|
|
|
else
|
|
|
|
return toString();
|
|
|
|
}
|
|
|
|
|
2020-01-16 00:27:03 +00:00
|
|
|
// This is application-specific interpretation of the prerelease data.
|
|
|
|
const QString Version::PrereleaseType() const
|
|
|
|
{
|
|
|
|
// Extract the first identifier
|
Update version display throughout to use the new information and be consistent.
The full version now includes the build/git information embedded within
it as build metadata according to the Semantic Versioning 2.0.0 spec,
for example: "1.1.0-beta-1+branch-name-a1b2c3d".
Now the full version string, with all detail is always displayed
EXCEPT for release versions, in which case just the simple version
number ("1.1.0") is displayed in the primary UI.
- Main window title: simple version for release versions, full version
string otherwise
- Notifications: same as main window title
- System tray: same as main window title
- About window title: same as main window title
- About window release notes: always include full version string
- Reports: always include full version string
- Under the logo (about dialog, profile selector, new profile
window): removed, as it is largely redundant and can
interfere with the window geometry.
- Database upgrade alert: same as main window title
- Database newer alert: same as main window title
The full version string is also included within the preference and
profile .xml files, but because build metadata is ignored in version
comparisons, differences in builds will not cause any spurious
alerts. However, changes in prerelease versions will continue to
be significant, as they should be.
2020-01-16 18:05:55 +00:00
|
|
|
QString type = mPrerelease.section(".", 0, 0);
|
2020-01-16 00:27:03 +00:00
|
|
|
|
|
|
|
// Remove any "-2", etc. that's included in the first identifier rather than as a dot-separated identifier
|
Update version display throughout to use the new information and be consistent.
The full version now includes the build/git information embedded within
it as build metadata according to the Semantic Versioning 2.0.0 spec,
for example: "1.1.0-beta-1+branch-name-a1b2c3d".
Now the full version string, with all detail is always displayed
EXCEPT for release versions, in which case just the simple version
number ("1.1.0") is displayed in the primary UI.
- Main window title: simple version for release versions, full version
string otherwise
- Notifications: same as main window title
- System tray: same as main window title
- About window title: same as main window title
- About window release notes: always include full version string
- Reports: always include full version string
- Under the logo (about dialog, profile selector, new profile
window): removed, as it is largely redundant and can
interfere with the window geometry.
- Database upgrade alert: same as main window title
- Database newer alert: same as main window title
The full version string is also included within the preference and
profile .xml files, but because build metadata is ignored in version
comparisons, differences in builds will not cause any spurious
alerts. However, changes in prerelease versions will continue to
be significant, as they should be.
2020-01-16 18:05:55 +00:00
|
|
|
type = type.section("-", 0, 0);
|
2020-01-16 00:27:03 +00:00
|
|
|
|
|
|
|
return type.toLower();
|
|
|
|
}
|
|
|
|
|
2020-01-16 21:06:11 +00:00
|
|
|
// 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.
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// ===================================================================================================
|
|
|
|
// Version class for parsing and comparing version strings as specified by Semantic Versioning 2.0.0
|
|
|
|
// See https://semver.org/spec/v2.0.0.html
|
|
|
|
|
|
|
|
Version::Version(const QString & version_string) : mString(version_string), mIsValid(false)
|
|
|
|
{
|
|
|
|
ParseSemanticVersion();
|
|
|
|
FixLegacyVersions();
|
|
|
|
}
|
|
|
|
|
|
|
|
Version::operator const QString &() const
|
|
|
|
{
|
|
|
|
return toString();
|
|
|
|
}
|
|
|
|
|
|
|
|
const QString & Version::toString() const
|
|
|
|
{
|
|
|
|
return mString;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Parse a version string as specified by Semantic Versioning 2.0.0.
|
2020-01-15 22:00:21 +00:00
|
|
|
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;
|
|
|
|
}
|
2020-01-16 16:54:41 +00:00
|
|
|
|
|
|
|
// If we ever encounter any really old version whose version isn't valid, its
|
|
|
|
// major version will be 0, so it will correctly be considered older than
|
|
|
|
// valid versions.
|
2020-01-15 22:00:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// 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;
|
|
|
|
}
|