* Add VersionAssemblyName to package manifest * Fix/improve nullability * Ensure package version from manifest is set when package migration exists * Set versionAssemblyName in umbracopackage template * Use Assembly.Load instead of ITypeFinder * Use AssemblyLoadContext to get asesmbly by name * Get version from package migration assembly * Hide unknown package version * Set versionAssemblyName in umbracopackage-rcl template
597 lines
22 KiB
C#
597 lines
22 KiB
C#
using System.Text.RegularExpressions;
|
|
using System.Diagnostics.CodeAnalysis;
|
|
#if !NETSTANDARD
|
|
using System.Globalization;
|
|
using System.Runtime.Serialization;
|
|
using System.Security.Permissions;
|
|
#endif
|
|
|
|
/*
|
|
Copyright (c) 2013 Max Hauser
|
|
|
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
of this software and associated documentation files (the "Software"), to deal
|
|
in the Software without restriction, including without limitation the rights
|
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
copies of the Software, and to permit persons to whom the Software is
|
|
furnished to do so, subject to the following conditions:
|
|
|
|
The above copyright notice and this permission notice shall be included in
|
|
all copies or substantial portions of the Software.
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
THE SOFTWARE.
|
|
*/
|
|
namespace Umbraco.Cms.Core.Semver
|
|
{
|
|
/// <summary>
|
|
/// A semantic version implementation.
|
|
/// Conforms to v2.0.0 of http://semver.org/
|
|
/// </summary>
|
|
#if NETSTANDARD
|
|
public sealed class SemVersion : IComparable<SemVersion>, IComparable
|
|
#else
|
|
[Serializable]
|
|
public sealed class SemVersion : IComparable<SemVersion>, IComparable, ISerializable
|
|
#endif
|
|
{
|
|
private static Regex parseEx =
|
|
new(
|
|
@"^(?<major>\d+)" +
|
|
@"(\.(?<minor>\d+))?" +
|
|
@"(\.(?<patch>\d+))?" +
|
|
@"(\-(?<pre>[0-9A-Za-z\-\.]+))?" +
|
|
@"(\+(?<build>[0-9A-Za-z\-\.]+))?$",
|
|
#if NETSTANDARD
|
|
RegexOptions.CultureInvariant | RegexOptions.ExplicitCapture);
|
|
#else
|
|
RegexOptions.CultureInvariant | RegexOptions.Compiled | RegexOptions.ExplicitCapture);
|
|
#endif
|
|
|
|
#if !NETSTANDARD
|
|
/// <summary>
|
|
/// Initializes a new instance of the <see cref="SemVersion" /> class.
|
|
/// </summary>
|
|
/// <param name="info"></param>
|
|
/// <param name="context"></param>
|
|
/// <exception cref="ArgumentNullException"></exception>
|
|
private SemVersion(SerializationInfo info, StreamingContext context)
|
|
{
|
|
if (info == null)
|
|
{
|
|
throw new ArgumentNullException("info");
|
|
}
|
|
|
|
SemVersion semVersion = Parse(info.GetString("SemVersion")!);
|
|
Major = semVersion.Major;
|
|
Minor = semVersion.Minor;
|
|
Patch = semVersion.Patch;
|
|
Prerelease = semVersion.Prerelease;
|
|
Build = semVersion.Build;
|
|
}
|
|
#endif
|
|
|
|
/// <summary>
|
|
/// Initializes a new instance of the <see cref="SemVersion" /> class.
|
|
/// </summary>
|
|
/// <param name="major">The major version.</param>
|
|
/// <param name="minor">The minor version.</param>
|
|
/// <param name="patch">The patch version.</param>
|
|
/// <param name="prerelease">The prerelease version (eg. "alpha").</param>
|
|
/// <param name="build">The build eg ("nightly.232").</param>
|
|
public SemVersion(int major, int minor = 0, int patch = 0, string prerelease = "", string build = "")
|
|
{
|
|
Major = major;
|
|
Minor = minor;
|
|
Patch = patch;
|
|
|
|
Prerelease = prerelease ?? string.Empty;
|
|
Build = build ?? string.Empty;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Initializes a new instance of the <see cref="SemVersion" /> class.
|
|
/// </summary>
|
|
/// <param name="version">
|
|
/// The <see cref="System.Version" /> that is used to initialize
|
|
/// the Major, Minor, Patch and Build properties.
|
|
/// </param>
|
|
public SemVersion(Version version)
|
|
{
|
|
if (version == null)
|
|
{
|
|
throw new ArgumentNullException("version");
|
|
}
|
|
|
|
Major = version.Major;
|
|
Minor = version.Minor;
|
|
|
|
if (version.Revision >= 0)
|
|
{
|
|
Patch = version.Revision;
|
|
}
|
|
|
|
Prerelease = string.Empty;
|
|
|
|
if (version.Build > 0)
|
|
{
|
|
Build = version.Build.ToString();
|
|
}
|
|
else
|
|
{
|
|
Build = string.Empty;
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Parses the specified string to a semantic version.
|
|
/// </summary>
|
|
/// <param name="version">The version string.</param>
|
|
/// <param name="strict">If set to <c>true</c> minor and patch version are required, else they default to 0.</param>
|
|
/// <returns>The SemVersion object.</returns>
|
|
/// <exception cref="System.InvalidOperationException">When a invalid version string is passed.</exception>
|
|
public static SemVersion Parse(string version, bool strict = false)
|
|
{
|
|
Match match = parseEx.Match(version);
|
|
if (!match.Success)
|
|
{
|
|
throw new ArgumentException("Invalid version.", "version");
|
|
}
|
|
|
|
#if NETSTANDARD
|
|
var major = int.Parse(match.Groups["major"].Value);
|
|
#else
|
|
var major = int.Parse(match.Groups["major"].Value, CultureInfo.InvariantCulture);
|
|
#endif
|
|
|
|
Group minorMatch = match.Groups["minor"];
|
|
var minor = 0;
|
|
if (minorMatch.Success)
|
|
{
|
|
#if NETSTANDARD
|
|
minor = int.Parse(minorMatch.Value);
|
|
#else
|
|
minor = int.Parse(minorMatch.Value, CultureInfo.InvariantCulture);
|
|
#endif
|
|
}
|
|
else if (strict)
|
|
{
|
|
throw new InvalidOperationException("Invalid version (no minor version given in strict mode)");
|
|
}
|
|
|
|
Group patchMatch = match.Groups["patch"];
|
|
var patch = 0;
|
|
if (patchMatch.Success)
|
|
{
|
|
#if NETSTANDARD
|
|
patch = int.Parse(patchMatch.Value);
|
|
#else
|
|
patch = int.Parse(patchMatch.Value, CultureInfo.InvariantCulture);
|
|
#endif
|
|
}
|
|
else if (strict)
|
|
{
|
|
throw new InvalidOperationException("Invalid version (no patch version given in strict mode)");
|
|
}
|
|
|
|
var prerelease = match.Groups["pre"].Value;
|
|
var build = match.Groups["build"].Value;
|
|
|
|
return new SemVersion(major, minor, patch, prerelease, build);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Parses the specified string to a semantic version.
|
|
/// </summary>
|
|
/// <param name="version">The version string.</param>
|
|
/// <param name="semver">
|
|
/// When the method returns, contains a SemVersion instance equivalent
|
|
/// to the version string passed in, if the version string was valid, or <c>null</c> if the
|
|
/// version string was not valid.
|
|
/// </param>
|
|
/// <param name="strict">If set to <c>true</c> minor and patch version are required, else they default to 0.</param>
|
|
/// <returns><c>False</c> when a invalid version string is passed, otherwise <c>true</c>.</returns>
|
|
public static bool TryParse(string version, [NotNullWhen(true)] out SemVersion? semver, bool strict = false)
|
|
{
|
|
try
|
|
{
|
|
semver = Parse(version, strict);
|
|
return true;
|
|
}
|
|
catch (Exception)
|
|
{
|
|
semver = null;
|
|
return false;
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Tests the specified versions for equality.
|
|
/// </summary>
|
|
/// <param name="versionA">The first version.</param>
|
|
/// <param name="versionB">The second version.</param>
|
|
/// <returns>If versionA is equal to versionB <c>true</c>, else <c>false</c>.</returns>
|
|
public static bool Equals(SemVersion versionA, SemVersion versionB)
|
|
{
|
|
if (ReferenceEquals(versionA, null))
|
|
{
|
|
return ReferenceEquals(versionB, null);
|
|
}
|
|
|
|
return versionA.Equals(versionB);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Compares the specified versions.
|
|
/// </summary>
|
|
/// <param name="versionA">The version to compare to.</param>
|
|
/// <param name="versionB">The version to compare against.</param>
|
|
/// <returns>
|
|
/// If versionA < versionB <c>< 0</c>, if versionA > versionB <c>> 0</c>,
|
|
/// if versionA is equal to versionB <c>0</c>.
|
|
/// </returns>
|
|
public static int Compare(SemVersion versionA, SemVersion versionB)
|
|
{
|
|
if (ReferenceEquals(versionA, null))
|
|
{
|
|
return ReferenceEquals(versionB, null) ? 0 : -1;
|
|
}
|
|
|
|
return versionA.CompareTo(versionB);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Make a copy of the current instance with optional altered fields.
|
|
/// </summary>
|
|
/// <param name="major">The major version.</param>
|
|
/// <param name="minor">The minor version.</param>
|
|
/// <param name="patch">The patch version.</param>
|
|
/// <param name="prerelease">The prerelease text.</param>
|
|
/// <param name="build">The build text.</param>
|
|
/// <returns>The new version object.</returns>
|
|
public SemVersion Change(int? major = null, int? minor = null, int? patch = null, string? prerelease = null, string? build = null) =>
|
|
new(
|
|
major ?? Major,
|
|
minor ?? Minor,
|
|
patch ?? Patch,
|
|
prerelease ?? Prerelease,
|
|
build ?? Build);
|
|
|
|
/// <summary>
|
|
/// Gets the major version.
|
|
/// </summary>
|
|
/// <value>
|
|
/// The major version.
|
|
/// </value>
|
|
public int Major { get; private set; }
|
|
|
|
/// <summary>
|
|
/// Gets the minor version.
|
|
/// </summary>
|
|
/// <value>
|
|
/// The minor version.
|
|
/// </value>
|
|
public int Minor { get; private set; }
|
|
|
|
/// <summary>
|
|
/// Gets the patch version.
|
|
/// </summary>
|
|
/// <value>
|
|
/// The patch version.
|
|
/// </value>
|
|
public int Patch { get; private set; }
|
|
|
|
/// <summary>
|
|
/// Gets the pre-release version.
|
|
/// </summary>
|
|
/// <value>
|
|
/// The pre-release version.
|
|
/// </value>
|
|
public string Prerelease { get; private set; }
|
|
|
|
/// <summary>
|
|
/// Gets the build version.
|
|
/// </summary>
|
|
/// <value>
|
|
/// The build version.
|
|
/// </value>
|
|
public string Build { get; private set; }
|
|
|
|
/// <summary>
|
|
/// Returns a <see cref="System.String" /> that represents this instance.
|
|
/// </summary>
|
|
/// <returns>
|
|
/// A <see cref="System.String" /> that represents this instance.
|
|
/// </returns>
|
|
public override string ToString()
|
|
{
|
|
var version = string.Empty + Major + "." + Minor + "." + Patch;
|
|
if (!string.IsNullOrEmpty(Prerelease))
|
|
{
|
|
version += "-" + Prerelease;
|
|
}
|
|
|
|
if (!string.IsNullOrEmpty(Build))
|
|
{
|
|
version += "+" + Build;
|
|
}
|
|
|
|
return version;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Compares the current instance with another object of the same type and returns an integer that indicates
|
|
/// whether the current instance precedes, follows, or occurs in the same position in the sort order as the
|
|
/// other object.
|
|
/// </summary>
|
|
/// <param name="obj">An object to compare with this instance.</param>
|
|
/// <returns>
|
|
/// A value that indicates the relative order of the objects being compared.
|
|
/// The return value has these meanings: Value Meaning Less than zero
|
|
/// This instance precedes <paramref name="obj" /> in the sort order.
|
|
/// Zero This instance occurs in the same position in the sort order as <paramref name="obj" />. i
|
|
/// Greater than zero This instance follows <paramref name="obj" /> in the sort order.
|
|
/// </returns>
|
|
public int CompareTo(object? obj) => CompareTo((SemVersion?)obj);
|
|
|
|
/// <summary>
|
|
/// Compares the current instance with another object of the same type and returns an integer that indicates
|
|
/// whether the current instance precedes, follows, or occurs in the same position in the sort order as the
|
|
/// other object.
|
|
/// </summary>
|
|
/// <param name="other">An object to compare with this instance.</param>
|
|
/// <returns>
|
|
/// A value that indicates the relative order of the objects being compared.
|
|
/// The return value has these meanings: Value Meaning Less than zero
|
|
/// This instance precedes <paramref name="other" /> in the sort order.
|
|
/// Zero This instance occurs in the same position in the sort order as <paramref name="other" />. i
|
|
/// Greater than zero This instance follows <paramref name="other" /> in the sort order.
|
|
/// </returns>
|
|
public int CompareTo(SemVersion? other)
|
|
{
|
|
if (ReferenceEquals(other, null))
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
var r = CompareByPrecedence(other);
|
|
if (r != 0)
|
|
{
|
|
return r;
|
|
}
|
|
|
|
r = CompareComponent(Build, other.Build);
|
|
return r;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Compares to semantic versions by precedence. This does the same as a Equals, but ignores the build information.
|
|
/// </summary>
|
|
/// <param name="other">The semantic version.</param>
|
|
/// <returns><c>true</c> if the version precedence matches.</returns>
|
|
public bool PrecedenceMatches(SemVersion other) => CompareByPrecedence(other) == 0;
|
|
|
|
/// <summary>
|
|
/// Compares to semantic versions by precedence. This does the same as a Equals, but ignores the build information.
|
|
/// </summary>
|
|
/// <param name="other">The semantic version.</param>
|
|
/// <returns>
|
|
/// A value that indicates the relative order of the objects being compared.
|
|
/// The return value has these meanings: Value Meaning Less than zero
|
|
/// This instance precedes <paramref name="other" /> in the version precedence.
|
|
/// Zero This instance has the same precedence as <paramref name="other" />. i
|
|
/// Greater than zero This instance has creater precedence as <paramref name="other" />.
|
|
/// </returns>
|
|
public int CompareByPrecedence(SemVersion other)
|
|
{
|
|
if (ReferenceEquals(other, null))
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
var r = Major.CompareTo(other.Major);
|
|
if (r != 0)
|
|
{
|
|
return r;
|
|
}
|
|
|
|
r = Minor.CompareTo(other.Minor);
|
|
if (r != 0)
|
|
{
|
|
return r;
|
|
}
|
|
|
|
r = Patch.CompareTo(other.Patch);
|
|
if (r != 0)
|
|
{
|
|
return r;
|
|
}
|
|
|
|
r = CompareComponent(Prerelease, other.Prerelease, true);
|
|
return r;
|
|
}
|
|
|
|
private static int CompareComponent(string a, string b, bool lower = false)
|
|
{
|
|
var aEmpty = string.IsNullOrEmpty(a);
|
|
var bEmpty = string.IsNullOrEmpty(b);
|
|
if (aEmpty && bEmpty)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
if (aEmpty)
|
|
{
|
|
return lower ? 1 : -1;
|
|
}
|
|
|
|
if (bEmpty)
|
|
{
|
|
return lower ? -1 : 1;
|
|
}
|
|
|
|
var aComps = a.Split('.');
|
|
var bComps = b.Split('.');
|
|
|
|
var minLen = Math.Min(aComps.Length, bComps.Length);
|
|
for (var i = 0; i < minLen; i++)
|
|
{
|
|
var ac = aComps[i];
|
|
var bc = bComps[i];
|
|
int anum, bnum;
|
|
var isanum = int.TryParse(ac, out anum);
|
|
var isbnum = int.TryParse(bc, out bnum);
|
|
int r;
|
|
if (isanum && isbnum)
|
|
{
|
|
r = anum.CompareTo(bnum);
|
|
if (r != 0)
|
|
{
|
|
return anum.CompareTo(bnum);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (isanum)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
if (isbnum)
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
r = string.CompareOrdinal(ac, bc);
|
|
if (r != 0)
|
|
{
|
|
return r;
|
|
}
|
|
}
|
|
}
|
|
|
|
return aComps.Length.CompareTo(bComps.Length);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Determines whether the specified <see cref="System.Object" /> is equal to this instance.
|
|
/// </summary>
|
|
/// <param name="obj">The <see cref="System.Object" /> to compare with this instance.</param>
|
|
/// <returns>
|
|
/// <c>true</c> if the specified <see cref="System.Object" /> is equal to this instance; otherwise, <c>false</c>.
|
|
/// </returns>
|
|
public override bool Equals(object? obj)
|
|
{
|
|
if (ReferenceEquals(obj, null))
|
|
{
|
|
return false;
|
|
}
|
|
|
|
if (ReferenceEquals(this, obj))
|
|
{
|
|
return true;
|
|
}
|
|
|
|
var other = (SemVersion)obj;
|
|
|
|
return Major == other.Major &&
|
|
Minor == other.Minor &&
|
|
Patch == other.Patch &&
|
|
string.Equals(Prerelease, other.Prerelease, StringComparison.Ordinal) &&
|
|
string.Equals(Build, other.Build, StringComparison.Ordinal);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Returns a hash code for this instance.
|
|
/// </summary>
|
|
/// <returns>
|
|
/// A hash code for this instance, suitable for use in hashing algorithms and data structures like a hash table.
|
|
/// </returns>
|
|
public override int GetHashCode()
|
|
{
|
|
unchecked
|
|
{
|
|
var result = Major.GetHashCode();
|
|
result = (result * 31) + Minor.GetHashCode();
|
|
result = (result * 31) + Patch.GetHashCode();
|
|
result = (result * 31) + Prerelease.GetHashCode();
|
|
result = (result * 31) + Build.GetHashCode();
|
|
return result;
|
|
}
|
|
}
|
|
|
|
#if !NETSTANDARD
|
|
[SecurityPermission(SecurityAction.Demand, SerializationFormatter = true)]
|
|
public void GetObjectData(SerializationInfo info, StreamingContext context)
|
|
{
|
|
if (info == null)
|
|
{
|
|
throw new ArgumentNullException("info");
|
|
}
|
|
|
|
info.AddValue("SemVersion", ToString());
|
|
}
|
|
#endif
|
|
|
|
/// <summary>
|
|
/// Implicit conversion from string to SemVersion.
|
|
/// </summary>
|
|
/// <param name="version">The semantic version.</param>
|
|
/// <returns>The SemVersion object.</returns>
|
|
public static implicit operator SemVersion(string version) => Parse(version);
|
|
|
|
/// <summary>
|
|
/// The override of the equals operator.
|
|
/// </summary>
|
|
/// <param name="left">The left value.</param>
|
|
/// <param name="right">The right value.</param>
|
|
/// <returns>If left is equal to right <c>true</c>, else <c>false</c>.</returns>
|
|
public static bool operator ==(SemVersion left, SemVersion right) => Equals(left, right);
|
|
|
|
/// <summary>
|
|
/// The override of the un-equal operator.
|
|
/// </summary>
|
|
/// <param name="left">The left value.</param>
|
|
/// <param name="right">The right value.</param>
|
|
/// <returns>If left is not equal to right <c>true</c>, else <c>false</c>.</returns>
|
|
public static bool operator !=(SemVersion left, SemVersion right) => !Equals(left, right);
|
|
|
|
/// <summary>
|
|
/// The override of the greater operator.
|
|
/// </summary>
|
|
/// <param name="left">The left value.</param>
|
|
/// <param name="right">The right value.</param>
|
|
/// <returns>If left is greater than right <c>true</c>, else <c>false</c>.</returns>
|
|
public static bool operator >(SemVersion left, SemVersion right) => Compare(left, right) > 0;
|
|
|
|
/// <summary>
|
|
/// The override of the greater than or equal operator.
|
|
/// </summary>
|
|
/// <param name="left">The left value.</param>
|
|
/// <param name="right">The right value.</param>
|
|
/// <returns>If left is greater than or equal to right <c>true</c>, else <c>false</c>.</returns>
|
|
public static bool operator >=(SemVersion left, SemVersion right) => left == right || left > right;
|
|
|
|
/// <summary>
|
|
/// The override of the less operator.
|
|
/// </summary>
|
|
/// <param name="left">The left value.</param>
|
|
/// <param name="right">The right value.</param>
|
|
/// <returns>If left is less than right <c>true</c>, else <c>false</c>.</returns>
|
|
public static bool operator <(SemVersion left, SemVersion right) => Compare(left, right) < 0;
|
|
|
|
/// <summary>
|
|
/// The override of the less than or equal operator.
|
|
/// </summary>
|
|
/// <param name="left">The left value.</param>
|
|
/// <param name="right">The right value.</param>
|
|
/// <returns>If left is less than or equal to right <c>true</c>, else <c>false</c>.</returns>
|
|
public static bool operator <=(SemVersion left, SemVersion right) => left == right || left < right;
|
|
}
|
|
}
|