// Copyright (c) 2007-2018 ppy Pty Ltd . // Licensed under the MIT Licence - https://raw.githubusercontent.com/ppy/osu/master/LICENCE using System; using System.Collections.Generic; using System.Configuration; using System.Diagnostics; using System.IO; using System.Linq; using Newtonsoft.Json; using osu.Framework.IO.Network; using FileWebRequest = osu.Framework.IO.Network.FileWebRequest; using WebRequest = osu.Framework.IO.Network.WebRequest; namespace osu.Desktop.Deploy { internal static class Program { private static string packages => Path.Combine(Environment.GetFolderPath(Environment.SpecialFolder.UserProfile), ".nuget", "packages"); private static string nugetPath => Path.Combine(packages, @"nuget.commandline\4.5.1\tools\NuGet.exe"); private static string squirrelPath => Path.Combine(packages, @"squirrel.windows\1.7.8\tools\Squirrel.exe"); private const string msbuild_path = @"C:\Program Files (x86)\Microsoft Visual Studio\2017\Community\MSBuild\15.0\Bin\MSBuild.exe"; public static string StagingFolder = ConfigurationManager.AppSettings["StagingFolder"]; public static string ReleasesFolder = ConfigurationManager.AppSettings["ReleasesFolder"]; public static string GitHubAccessToken = ConfigurationManager.AppSettings["GitHubAccessToken"]; public static string GitHubUsername = ConfigurationManager.AppSettings["GitHubUsername"]; public static string GitHubRepoName = ConfigurationManager.AppSettings["GitHubRepoName"]; public static string SolutionName = ConfigurationManager.AppSettings["SolutionName"]; public static string ProjectName = ConfigurationManager.AppSettings["ProjectName"]; public static string NuSpecName = ConfigurationManager.AppSettings["NuSpecName"]; public static string TargetNames = ConfigurationManager.AppSettings["TargetName"]; public static string PackageName = ConfigurationManager.AppSettings["PackageName"]; public static string IconName = ConfigurationManager.AppSettings["IconName"]; public static string CodeSigningCertificate = ConfigurationManager.AppSettings["CodeSigningCertificate"]; public static string GitHubApiEndpoint => $"https://api.github.com/repos/{GitHubUsername}/{GitHubRepoName}/releases"; public static string GitHubReleasePage => $"https://github.com/{GitHubUsername}/{GitHubRepoName}/releases"; /// /// How many previous build deltas we want to keep when publishing. /// private const int keep_delta_count = 4; private static string codeSigningCmd => string.IsNullOrEmpty(codeSigningPassword) ? "" : $"-n \"/a /f {codeSigningCertPath} /p {codeSigningPassword} /t http://timestamp.comodoca.com/authenticode\""; private static string homeDir => Environment.GetFolderPath(Environment.SpecialFolder.UserProfile); private static string codeSigningCertPath => Path.Combine(homeDir, CodeSigningCertificate); private static string solutionPath => Environment.CurrentDirectory; private static string stagingPath => Path.Combine(solutionPath, StagingFolder); private static string iconPath => Path.Combine(solutionPath, ProjectName, IconName); private static string nupkgFilename(string ver) => $"{PackageName}.{ver}.nupkg"; private static string nupkgDistroFilename(string ver) => $"{PackageName}-{ver}-full.nupkg"; private static readonly Stopwatch sw = new Stopwatch(); private static string codeSigningPassword; private static bool interactive; public static void Main(string[] args) { interactive = args.Length == 0; displayHeader(); findSolutionPath(); if (!Directory.Exists(ReleasesFolder)) { write("WARNING: No release directory found. Make sure you want this!", ConsoleColor.Yellow); Directory.CreateDirectory(ReleasesFolder); } checkGitHubReleases(); refreshDirectory(StagingFolder); //increment build number until we have a unique one. string verBase = DateTime.Now.ToString("yyyy.Mdd."); int increment = 0; while (Directory.GetFiles(ReleasesFolder, $"*{verBase}{increment}*").Any()) increment++; string version = $"{verBase}{increment}"; Console.ForegroundColor = ConsoleColor.White; Console.Write($"Ready to deploy {version}!"); pauseIfInteractive(); sw.Start(); if (!string.IsNullOrEmpty(CodeSigningCertificate)) { Console.Write("Enter code signing password: "); codeSigningPassword = args.Length > 0 ? args[0] : readLineMasked(); } write("Updating AssemblyInfo..."); updateCsprojVersion(version); write("Running build process..."); foreach (string targetName in TargetNames.Split(',')) runCommand(msbuild_path, $"/v:quiet /m /t:{targetName.Replace('.', '_')} /p:OutputPath={stagingPath};Targets=\"Clean;Build\";Configuration=Release {SolutionName}.sln"); write("Creating NuGet deployment package..."); runCommand(nugetPath, $"pack {NuSpecName} -Version {version} -Properties Configuration=Deploy -OutputDirectory {stagingPath} -BasePath {stagingPath}"); //prune once before checking for files so we can avoid erroring on files which aren't even needed for this build. pruneReleases(); checkReleaseFiles(); write("Running squirrel build..."); runCommand(squirrelPath, $"--releasify {stagingPath}\\{nupkgFilename(version)} --setupIcon {iconPath} --icon {iconPath} {codeSigningCmd} --no-msi"); //prune again to clean up before upload. pruneReleases(); //rename setup to install. File.Copy(Path.Combine(ReleasesFolder, "Setup.exe"), Path.Combine(ReleasesFolder, "install.exe"), true); File.Delete(Path.Combine(ReleasesFolder, "Setup.exe")); uploadBuild(version); //reset assemblyinfo. updateCsprojVersion("0.0.0"); write("Done!", ConsoleColor.White); pauseIfInteractive(); } private static void displayHeader() { Console.ForegroundColor = ConsoleColor.Red; Console.WriteLine(); Console.WriteLine(" Please note that OSU! and PPY are registered trademarks and as such covered by trademark law."); Console.WriteLine(" Do not distribute builds of this project publicly that make use of these."); Console.ResetColor(); Console.WriteLine(); } /// /// Ensure we have all the files in the release directory which are expected to be there. /// This should have been accounted for in earlier steps, and just serves as a verification step. /// private static void checkReleaseFiles() { if (!canGitHub) return; var releaseLines = getReleaseLines(); //ensure we have all files necessary foreach (var l in releaseLines) if (!File.Exists(Path.Combine(ReleasesFolder, l.Filename))) error($"Local file missing {l.Filename}"); } private static IEnumerable getReleaseLines() => File.ReadAllLines(Path.Combine(ReleasesFolder, "RELEASES")).Select(l => new ReleaseLine(l)); private static void pruneReleases() { if (!canGitHub) return; write("Pruning RELEASES..."); var releaseLines = getReleaseLines().ToList(); var fulls = releaseLines.Where(l => l.Filename.Contains("-full")).Reverse().Skip(1); //remove any FULL releases (except most recent) foreach (var l in fulls) { write($"- Removing old release {l.Filename}", ConsoleColor.Yellow); File.Delete(Path.Combine(ReleasesFolder, l.Filename)); releaseLines.Remove(l); } //remove excess deltas var deltas = releaseLines.Where(l => l.Filename.Contains("-delta")).ToArray(); if (deltas.Length > keep_delta_count) { foreach (var l in deltas.Take(deltas.Length - keep_delta_count)) { write($"- Removing old delta {l.Filename}", ConsoleColor.Yellow); File.Delete(Path.Combine(ReleasesFolder, l.Filename)); releaseLines.Remove(l); } } var lines = new List(); releaseLines.ForEach(l => lines.Add(l.ToString())); File.WriteAllLines(Path.Combine(ReleasesFolder, "RELEASES"), lines); } private static void uploadBuild(string version) { if (!canGitHub || string.IsNullOrEmpty(CodeSigningCertificate)) return; write("Publishing to GitHub..."); write($"- Creating release {version}...", ConsoleColor.Yellow); var req = new JsonWebRequest($"{GitHubApiEndpoint}") { Method = HttpMethod.POST, }; req.AddRaw(JsonConvert.SerializeObject(new GitHubRelease { Name = version, Draft = true, PreRelease = true })); req.AuthenticatedBlockingPerform(); var assetUploadUrl = req.ResponseObject.UploadUrl.Replace("{?name,label}", "?name={0}"); foreach (var a in Directory.GetFiles(ReleasesFolder).Reverse()) //reverse to upload RELEASES first. { write($"- Adding asset {a}...", ConsoleColor.Yellow); var upload = new WebRequest(assetUploadUrl, Path.GetFileName(a)) { Method = HttpMethod.POST, Timeout = 240000, ContentType = "application/octet-stream", }; upload.AddRaw(File.ReadAllBytes(a)); upload.AuthenticatedBlockingPerform(); } openGitHubReleasePage(); } private static void openGitHubReleasePage() => Process.Start(GitHubReleasePage); private static bool canGitHub => !string.IsNullOrEmpty(GitHubAccessToken); private static void checkGitHubReleases() { if (!canGitHub) return; write("Checking GitHub releases..."); var req = new JsonWebRequest>($"{GitHubApiEndpoint}"); req.AuthenticatedBlockingPerform(); var lastRelease = req.ResponseObject.FirstOrDefault(); if (lastRelease == null) return; if (lastRelease.Draft) { openGitHubReleasePage(); error("There's a pending draft release! You probably don't want to push a build with this present."); } //there's a previous release for this project. var assetReq = new JsonWebRequest>($"{GitHubApiEndpoint}/{lastRelease.Id}/assets"); assetReq.AuthenticatedBlockingPerform(); var assets = assetReq.ResponseObject; //make sure our RELEASES file is the same as the last build on the server. var releaseAsset = assets.FirstOrDefault(a => a.Name == "RELEASES"); //if we don't have a RELEASES asset then the previous release likely wasn't a Squirrel one. if (releaseAsset == null) return; write($"Last GitHub release was {lastRelease.Name}."); bool requireDownload = false; if (!File.Exists(Path.Combine(ReleasesFolder, nupkgDistroFilename(lastRelease.Name)))) { write("Last version's package not found locally.", ConsoleColor.Red); requireDownload = true; } else { var lastReleases = new RawFileWebRequest($"{GitHubApiEndpoint}/assets/{releaseAsset.Id}"); lastReleases.AuthenticatedBlockingPerform(); if (File.ReadAllText(Path.Combine(ReleasesFolder, "RELEASES")) != lastReleases.ResponseString) { write("Server's RELEASES differed from ours.", ConsoleColor.Red); requireDownload = true; } } if (!requireDownload) return; write("Refreshing local releases directory..."); refreshDirectory(ReleasesFolder); foreach (var a in assets) { if (a.Name.EndsWith(".exe")) continue; write($"- Downloading {a.Name}...", ConsoleColor.Yellow); new FileWebRequest(Path.Combine(ReleasesFolder, a.Name), $"{GitHubApiEndpoint}/assets/{a.Id}").AuthenticatedBlockingPerform(); } } private static void refreshDirectory(string directory) { if (Directory.Exists(directory)) Directory.Delete(directory, true); Directory.CreateDirectory(directory); } private static void updateCsprojVersion(string version) { var toUpdate = new[] { "", "" }; string file = Path.Combine(ProjectName, $"{ProjectName}.csproj"); var l1 = File.ReadAllLines(file); List l2 = new List(); foreach (var l in l1) { string line = l; foreach (var tag in toUpdate) { int startIndex = l.IndexOf(tag, StringComparison.InvariantCulture); if (startIndex == -1) continue; startIndex += tag.Length; int endIndex = l.IndexOf("<", startIndex, StringComparison.InvariantCulture); line = $"{l.Substring(0, startIndex)}{version}{l.Substring(endIndex)}"; } l2.Add(line); } File.WriteAllLines(file, l2); } /// /// Find the base path of the active solution (git checkout location) /// private static void findSolutionPath() { string path = Path.GetDirectoryName(Environment.CommandLine.Replace("\"", "").Trim()); if (string.IsNullOrEmpty(path)) path = Environment.CurrentDirectory; while (!File.Exists(Path.Combine(path, $"{SolutionName}.sln"))) path = path.Remove(path.LastIndexOf(Path.DirectorySeparatorChar)); path += Path.DirectorySeparatorChar; Environment.CurrentDirectory = path; } private static bool runCommand(string command, string args) { var psi = new ProcessStartInfo(command, args) { WorkingDirectory = solutionPath, CreateNoWindow = true, RedirectStandardOutput = true, RedirectStandardError = true, UseShellExecute = false, WindowStyle = ProcessWindowStyle.Hidden }; Process p = Process.Start(psi); if (p == null) return false; string output = p.StandardOutput.ReadToEnd(); output += p.StandardError.ReadToEnd(); if (p.ExitCode == 0) return true; write(output); error($"Command {command} {args} failed!"); return false; } private static string readLineMasked() { var fg = Console.ForegroundColor; Console.ForegroundColor = Console.BackgroundColor; var ret = Console.ReadLine(); Console.ForegroundColor = fg; return ret; } private static void error(string message) { Console.ForegroundColor = ConsoleColor.Red; Console.WriteLine($"FATAL ERROR: {message}"); pauseIfInteractive(); Environment.Exit(-1); } private static void pauseIfInteractive() { if (interactive) Console.ReadLine(); else Console.WriteLine(); } private static void write(string message, ConsoleColor col = ConsoleColor.Gray) { if (sw.ElapsedMilliseconds > 0) { Console.ForegroundColor = ConsoleColor.Green; Console.Write(sw.ElapsedMilliseconds.ToString().PadRight(8)); } Console.ForegroundColor = col; Console.WriteLine(message); } public static void AuthenticatedBlockingPerform(this WebRequest r) { r.AddHeader("Authorization", $"token {GitHubAccessToken}"); r.Perform(); } } internal class RawFileWebRequest : WebRequest { public RawFileWebRequest(string url) : base(url) { } protected override string Accept => "application/octet-stream"; } internal class ReleaseLine { public string Hash; public string Filename; public int Filesize; public ReleaseLine(string line) { var split = line.Split(' '); Hash = split[0]; Filename = split[1]; Filesize = int.Parse(split[2]); } public override string ToString() => $"{Hash} {Filename} {Filesize}"; } }