SemanticVersioning/BuildGeneratorEditorWindow.cs

331 lines
14 KiB
C#

using System;
using System.Collections.Generic;
using System.Diagnostics;
using UnityEditor;
using UnityEngine;
public class BuildGeneratorEditorWindow : EditorWindow
{
private VersionDefinition editorVersion = default;
private VersionDefinition gitVersion = default;
[MenuItem("Window/Build Generator")]
private static void ShowWindow()
{
BuildGeneratorEditorWindow window = GetWindow<BuildGeneratorEditorWindow>();
window.titleContent = new GUIContent("Build Generator");
window.editorVersion = new(PlayerSettings.bundleVersion);
try { window.gitVersion = GitProcess.GetLatestBuildVersion(); } catch { }
window.Show();
}
private void OnGUI()
{
if (GUILayout.Button($"Re-evaluate versions"))
{
editorVersion = new(PlayerSettings.bundleVersion);
gitVersion = GitProcess.GetLatestBuildVersion();
}
EditorGUILayout.Space();
EditorGUILayout.BeginHorizontal();
EditorGUILayout.BeginVertical();
EditorGUILayout.LabelField($"Git Version: {editorVersion} ({editorVersion.BuildNumber})", new GUIStyle(GUI.skin.label) { alignment = TextAnchor.MiddleCenter });
if (GUILayout.Button($"Create Release")) CommitVersion(GitProcess.GetUpcomingReleaseVersion());
if (GUILayout.Button($"Create Release Candidate")) CommitVersion(GitProcess.GetUpcomingReleaseCandidateVersion());
EditorGUILayout.EndVertical();
EditorGUILayout.BeginVertical();
EditorGUILayout.LabelField($"Editor Version: {editorVersion} ({editorVersion.BuildNumber})", new GUIStyle(GUI.skin.label) { alignment = TextAnchor.MiddleCenter });
if (GUILayout.Button($"Create Release")) CommitVersion(editorVersion.ToReleaseVersion());
if (GUILayout.Button($"Create Release Candidate")) CommitVersion(editorVersion.IncreaseReleaseCandidate());
EditorGUILayout.EndVertical();
EditorGUILayout.EndHorizontal();
EditorGUILayout.Space();
if (GUILayout.Button($"Push All")) { GitProcess.Push(); GitProcess.PushTags(); }
EditorGUILayout.Space();
EditorGUILayout.LabelField("Force New Version", new GUIStyle(GUI.skin.label) { alignment = TextAnchor.MiddleCenter });
EditorGUILayout.BeginHorizontal();
if (IncrementButton("Major", editorVersion.IncreaseMajor()))
CommitVersion(editorVersion.IncreaseMajor());
if (IncrementButton("Minor", editorVersion.IncreaseMinor()))
CommitVersion(editorVersion.IncreaseMinor());
if (IncrementButton("Patch", editorVersion.IncreasePatch()))
CommitVersion(editorVersion.IncreasePatch());
if (IncrementButton("Release Candidate", editorVersion.IncreaseReleaseCandidate()))
CommitVersion(editorVersion.IncreaseReleaseCandidate());
EditorGUILayout.EndHorizontal();
}
private void CommitVersion(VersionDefinition versionDefinition)
{
if (editorVersion >= versionDefinition)
{
EditorUtility.DisplayDialog(
"Release Conflict",
$"You can't create a new version because the old version({editorVersion}) is either higher or the same as the new version({versionDefinition}). Please make new commits to create new releases.",
"Return");
return;
}
if (!EditorUtility.DisplayDialog(
"Release Confirmation",
$"Are you sure to commit a new incremental version of {versionDefinition}?",
"Yes",
"Cancel")
)
return;
ApplyVersion(versionDefinition);
GitProcess.Add("ProjectSettings\\ProjectSettings.asset");
GitProcess.Add("Assets\\Settings\\Build Profiles\\**");
GitProcess.Commit($"chore: Bump Version to {versionDefinition}");
GitProcess.CreateTag(versionDefinition);
GitProcess.PushTag(versionDefinition);
}
private bool IncrementButton(string fieldName, VersionDefinition resultDefinition)
{
EditorGUILayout.BeginVertical();
bool isButtonPressed = GUILayout.Button($"Add Incremental {fieldName}");
EditorGUILayout.LabelField($"{editorVersion} -> {resultDefinition}", new GUIStyle(GUI.skin.label) { alignment = TextAnchor.MiddleCenter });
EditorGUILayout.EndVertical();
return isButtonPressed && EditorUtility.DisplayDialog("Increment Confirmation", $"Are you sure to add an incremental {fieldName} version?", "Yes", "Cancel");
}
private void ApplyVersion(VersionDefinition versionDefinition)
{
editorVersion = versionDefinition;
PlayerSettings.Android.bundleVersionCode = versionDefinition.BuildNumber;
PlayerSettings.iOS.buildNumber = versionDefinition.BuildNumber.ToString();
PlayerSettings.bundleVersion = versionDefinition.ToString();
AssetDatabase.SaveAssets();
}
private struct VersionDefinition
{
public const int MAX_VALUE = 99;
public const string BUILD_NUMBER_FORMAT = "00";
public const int DEFAULT_MAJOR = 0;
public const int DEFAULT_MINOR = 0;
public const int DEFAULT_PATCH = 0;
public const int DEFAULT_RELEASE_CANDIDATE = 1;
public const int RELEASE_RC_VALUE = 99;
private readonly int _major;
private readonly int _minor;
private readonly int _patch;
private readonly int _releaseCandidate;
public readonly int Major => _major > MAX_VALUE ? MAX_VALUE : _major;
public readonly int Minor => _minor > MAX_VALUE ? MAX_VALUE : _minor;
public readonly int Patch => _patch > MAX_VALUE ? MAX_VALUE : _patch;
public readonly int ReleaseCandidate => _releaseCandidate > MAX_VALUE ? MAX_VALUE : _releaseCandidate;
public readonly bool IsRelease => ReleaseCandidate == RELEASE_RC_VALUE;
public VersionDefinition(int? major = null, int? minor = null, int? patch = null, int? releaseCandidate = null)
{
_major = major ?? DEFAULT_MAJOR;
_minor = minor ?? DEFAULT_MINOR;
_patch = patch ?? DEFAULT_PATCH;
_releaseCandidate = releaseCandidate ?? DEFAULT_RELEASE_CANDIDATE;
if (_major == 0 && _minor == 0)
_minor = _releaseCandidate = 1;
}
public VersionDefinition(string versionString)
{
if (versionString.StartsWith('v'))
versionString = versionString[1..];
_major = DEFAULT_MAJOR;
_minor = DEFAULT_MINOR;
_patch = DEFAULT_PATCH;
_releaseCandidate = RELEASE_RC_VALUE;
string[] releaseCandidateVersionStrings = versionString.Split("-rc");
string[] versionNumbers = releaseCandidateVersionStrings[0].Split('.');
if (versionNumbers.Length > 0 && int.TryParse(versionNumbers[0], out int major)) _major = major;
if (versionNumbers.Length > 1 && int.TryParse(versionNumbers[1], out int minor)) _minor = minor;
if (versionNumbers.Length > 2 && int.TryParse(versionNumbers[2], out int patch)) _patch = patch;
if (releaseCandidateVersionStrings.Length > 1 && int.TryParse(releaseCandidateVersionStrings[1], out int releaseCandidate)) _releaseCandidate = releaseCandidate;
if (_major == 0 && _minor == 0)
_minor = _releaseCandidate = 1;
}
public readonly VersionDefinition IncreaseMajor() => new(Major + 1, DEFAULT_MINOR, DEFAULT_PATCH, 0);
public readonly VersionDefinition IncreaseMinor() => new(Major, Minor + 1, DEFAULT_PATCH, 0);
public readonly VersionDefinition IncreasePatch() => new(Major, Minor, Patch + 1, 0);
public readonly VersionDefinition IncreaseReleaseCandidate() => new(Major, Minor, Patch, ReleaseCandidate + 1);
public readonly VersionDefinition ToReleaseVersion() => new(Major, Minor, Patch, RELEASE_RC_VALUE);
public readonly int BuildNumber => int.Parse($"{Major.ToString(BUILD_NUMBER_FORMAT)}{Minor.ToString(BUILD_NUMBER_FORMAT)}{Patch.ToString(BUILD_NUMBER_FORMAT)}{ReleaseCandidate.ToString(BUILD_NUMBER_FORMAT)}");
public override readonly string ToString()
{
if (IsRelease)
return $"{Major}.{Minor}.{Patch}";
return $"{Major}.{Minor}.{Patch}-rc{ReleaseCandidate}";
}
public static bool operator >(VersionDefinition left, VersionDefinition right) => left.BuildNumber > right.BuildNumber;
public static bool operator <(VersionDefinition left, VersionDefinition right) => left.BuildNumber < right.BuildNumber;
public static bool operator >=(VersionDefinition left, VersionDefinition right) => left.BuildNumber >= right.BuildNumber;
public static bool operator <=(VersionDefinition left, VersionDefinition right) => left.BuildNumber <= right.BuildNumber;
public static bool operator ==(VersionDefinition left, VersionDefinition right) => left.BuildNumber == right.BuildNumber;
public static bool operator !=(VersionDefinition left, VersionDefinition right) => left.BuildNumber != right.BuildNumber;
public override readonly bool Equals(object obj)
=> obj is VersionDefinition versionDefinition && BuildNumber == versionDefinition.BuildNumber;
public override readonly int GetHashCode()
=> HashCode.Combine(Major, Minor, Patch, ReleaseCandidate);
}
private static class GitProcess
{
public static VersionDefinition GetLatestBuildVersion() => new(GetLatestBuildVersionString());
public static VersionDefinition GetUpcomingReleaseVersion() => GetUpcomingReleaseCandidateVersion().ToReleaseVersion();
public static VersionDefinition GetUpcomingReleaseCandidateVersion()
{
VersionDefinition latestVersionDefinition = GetLatestBuildVersion();
string[] commits = GetCommitsSinceLastTag();
return ApplySemVer(latestVersionDefinition, commits);
}
public static VersionDefinition GetVersion(string version, IList<string> gitLogs)
=> ApplySemVer(new(version), gitLogs);
public static VersionDefinition ApplySemVer(VersionDefinition latestVersionDefinition, IList<string> commits)
{
VersionDefinition result = latestVersionDefinition.IsRelease ? latestVersionDefinition.IncreasePatch() : latestVersionDefinition;
for (int i = commits.Count - 1; i >= 0; i--)
{
string line = commits[i];
if (string.IsNullOrWhiteSpace(line))
continue;
string commitTitle = line.Length > 9 ? line[9..] : line;
if (commitTitle.StartsWith("breaking change", StringComparison.InvariantCultureIgnoreCase))
{
result = result.IncreaseMajor();
break;
}
if (latestVersionDefinition.Minor != result.Minor)
continue;
if (commitTitle.StartsWith("feat", StringComparison.OrdinalIgnoreCase))
{
result = result.IncreaseMinor();
continue;
}
if (latestVersionDefinition.Patch != result.Patch)
continue;
if (commitTitle.StartsWith("fix", StringComparison.OrdinalIgnoreCase))
{
result = result.IncreasePatch();
continue;
}
if (latestVersionDefinition.ReleaseCandidate == result.ReleaseCandidate)
result = result.IncreaseReleaseCandidate();
}
return result;
}
public static string GetLatestBuildVersionString()
=> RunGitCommand($"describe --tags --abbrev=0");
public static string[] GetCommits()
=> RunGitCommand("log --oneline").Replace("\r", "").Split('\n');
public static string[] GetCommitsSinceLastTag()
=> RunGitCommand($"log {GetLatestBuildVersionString()}..HEAD --oneline").Replace("\r", "").Split('\n');
public static string CreateTag(VersionDefinition versionDefinition)
=> CreateTag($"v{versionDefinition}", $"{(versionDefinition.IsRelease ? "Release" : "Build")} {versionDefinition} with build number: {versionDefinition.BuildNumber}\"");
public static string Push()
=> RunGitCommand($"push");
public static string Add(IList<string> files)
=> RunGitCommand($"add \"{string.Join("\" \"", files)}\"");
public static string Add(string file)
=> RunGitCommand($"add \"{file}\"");
public static string Commit(string message)
=> RunGitCommand($"commit -m \"{message}\"");
public static string PushTags()
=> RunGitCommand($"push --tags");
public static string PushTag(VersionDefinition versionDefinition, string remote = "origin")
=> RunGitCommand($"push {remote} tag v{versionDefinition}");
public static string CreateTag(string title, string message)
=> RunGitCommand($"tag -a {title} -m \"{message}");
public static string Unstage(IList<string> files)
=> RunGitCommand($"reset \"{string.Join("\" \"", files)}\"");
public static string Unstage(string file)
=> RunGitCommand($"reset \"{file}\"");
public static string Unstage()
=> RunGitCommand($"reset");
private static string RunGitCommand(string gitArgs)
{
ProcessStartInfo processStartInfo = new()
{
FileName = "git",
Arguments = gitArgs,
RedirectStandardOutput = true,
UseShellExecute = false,
CreateNoWindow = true
};
try
{
Process process = Process.Start(processStartInfo);
string output = process.StandardOutput.ReadToEnd();
if (output.EndsWith('\n')) output = output[0..^1];
if (output.EndsWith('\r')) output = output[0..^1];
process.WaitForExit();
return output;
}
catch (Exception ex)
{
Console.WriteLine("An error occurred: " + ex.Message);
throw;
}
}
}
}