Remotely/Server.Installer/Program.cs
2021-07-29 07:58:08 -07:00

361 lines
16 KiB
C#

using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.Logging;
using Remotely.Shared.Enums;
using Remotely.Shared.Services;
using Remotely.Shared.Utilities;
using Server.Installer.Models;
using Server.Installer.Services;
using System;
using System.IO;
using System.Threading.Tasks;
namespace Server.Installer
{
public class Program
{
public static IServiceProvider Services { get; set; }
public static async Task Main(string[] args)
{
Console.ForegroundColor = ConsoleColor.Cyan;
Console.WriteLine(AppConstants.RemotelyAscii);
Console.ForegroundColor = ConsoleColor.Gray;
Console.WriteLine("(https://remotely.one)");
Console.WriteLine();
Console.WriteLine();
if (!ParseCliParams(args, out var cliParams))
{
ShowHelpText();
return;
}
if (EnvironmentHelper.Platform != Platform.Windows &&
EnvironmentHelper.Platform != Platform.Linux)
{
ConsoleHelper.WriteError("Remotely Server can only be installed on Linux or Windows.");
return;
}
BuildServices();
var elevationDetector = Services.GetRequiredService<IElevationDetector>();
var serverInstaller = Services.GetRequiredService<IServerInstaller>();
var githubApi = Services.GetRequiredService<IGitHubApi>();
if (!elevationDetector.IsElevated())
{
ConsoleHelper.WriteError("The installer process must be elevated. On Linux, run with sudo. " +
"On Windows, run from a command line that was opened with \"Run as admin\".");
return;
}
ConsoleHelper.WriteLine("Thank you for trying Remotely! This installer will guide you " +
"through deploying the Remotely server onto this machine.");
ConsoleHelper.WriteLine("There are two ways to create the server files. You can download the pre-built package " +
"from the latest public release, or you can use your GitHub credentials to build a customized package through " +
"an integration with GitHub Actions. The pre-built packages will not have your server's URL embedded in the " +
"desktop clients, and end users will need to type it in manually.");
ConsoleHelper.WriteLine("If using GitHub Actions, you will need to enter a GitHub Personal Access Token, " +
"which will allow this app to access your fork of the Remotely repo. You can generate a PAT at " +
"https://github.com/settings/tokens. You need to give it the \"repo\" scope.");
ConsoleHelper.WriteLine("Be sure to retain your GitHub Personal Access Token if you want to re-use it " +
"for upgrading in the future. The installer does not save it locally.");
ConsoleHelper.WriteLine("If you haven't already, please go to the Actions tab in your Remotely repo " +
"and enable them. If not, this process will fail.");
while (cliParams.UsePrebuiltPackage is null)
{
var usePrebuiltPackage = ConsoleHelper.ReadLine("Download pre-built package (yes/no)?",
subprompt: "If no, a customized server package will be created through GitHub Actions.");
if (ConsoleHelper.TryParseBoolLike(usePrebuiltPackage, out var result))
{
cliParams.UsePrebuiltPackage = result;
}
}
while (string.IsNullOrWhiteSpace(cliParams.InstallDirectory))
{
cliParams.InstallDirectory = ConsoleHelper.ReadLine("Which directory should the server files be extracted to (e.g. /var/www/remotely/)?").Trim();
}
while (cliParams.ServerUrl is null)
{
var url = ConsoleHelper.ReadLine("What is your server's public URL (e.g. https://app.remotely.one)?").Trim();
if (Uri.TryCreate(url, UriKind.Absolute, out var serverUrl))
{
cliParams.ServerUrl = serverUrl;
}
}
while (cliParams.WebServer is null)
{
var webServerType = ConsoleHelper.GetSelection("Which web server will be used?",
"Caddy on Ubuntu",
"Nginx on Ubuntu",
"Caddy on CentOS",
"Nginx on CentOS",
"IIS on Windows Server 2016+");
if (Enum.TryParse<WebServerType>(webServerType, out var result))
{
cliParams.WebServer = result;
}
}
if (cliParams.UsePrebuiltPackage == false)
{
while (string.IsNullOrWhiteSpace(cliParams.GitHubUsername))
{
cliParams.GitHubUsername = ConsoleHelper.ReadLine("What is your GitHub username?").Trim();
}
while (string.IsNullOrWhiteSpace(cliParams.GitHubPat))
{
cliParams.GitHubPat = ConsoleHelper.ReadLine("What GitHub Personal Access Token should be used?").Trim();
}
while (cliParams.CreateNew is null)
{
var createNew = ConsoleHelper.ReadLine("Create new build (yes/no)?",
subprompt: "If no, the latest existing build artifact on GitHub will be used.");
if (ConsoleHelper.TryParseBoolLike(createNew, out var result))
{
cliParams.CreateNew = result;
}
}
if (cliParams.CreateNew == true)
{
if (cliParams.Reference?.Contains("latest", StringComparison.OrdinalIgnoreCase) == true)
{
cliParams.Reference = await githubApi.GetLatestReleaseTag();
}
while (string.IsNullOrWhiteSpace(cliParams.Reference))
{
var selection = ConsoleHelper.GetSelection("Which version would you like to build?",
"Latest official release",
"Preview changes (i.e. master branch)",
"Specific release");
if (int.TryParse(selection, out var result))
{
switch (result)
{
case 0:
cliParams.Reference = await githubApi.GetLatestReleaseTag();
break;
case 1:
cliParams.Reference = "master";
break;
case 2:
ConsoleHelper.WriteLine("Enter the GitHub branch or tag name from which to build " +
"(e.g. \"master\" or \"v2021.04.25.0953\").");
cliParams.Reference = ConsoleHelper.ReadLine("Input Reference").Trim();
break;
default:
break;
}
}
}
}
ConsoleHelper.WriteLine($"Performing server install. GitHub User: {cliParams.GitHubUsername}. " +
$"Server URL: {cliParams.ServerUrl}. Installation Directory: {cliParams.InstallDirectory}. " +
$"Web Server: {cliParams.WebServer}. Create New Build: {cliParams.CreateNew}. " +
$"Git Reference: {cliParams.Reference}");
}
else
{
ConsoleHelper.WriteLine($"Server URL: {cliParams.ServerUrl}. " +
$"Installation Directory: {cliParams.InstallDirectory}. Web Server: {cliParams.WebServer}.");
}
await serverInstaller.PerformInstall(cliParams);
ConsoleHelper.WriteLine("Installation completed.");
}
private static void BuildServices()
{
var services = new ServiceCollection();
services.AddSingleton<IGitHubApi, GitHubApi>();
services.AddSingleton<IServerInstaller, ServerInstaller>();
if (EnvironmentHelper.IsWindows)
{
services.AddSingleton<IElevationDetector, ElevationDetectorWin>();
}
else if (EnvironmentHelper.IsLinux)
{
services.AddSingleton<IElevationDetector, ElevationDetectorLinux>();
}
else
{
throw new NotSupportedException("Operating system not supported.");
}
Services = services.BuildServiceProvider();
}
private static bool ParseCliParams(string[] args, out CliParams cliParams)
{
cliParams = new CliParams();
for (var i = 0; i < args.Length; i += 2)
{
try
{
var key = args[i].Trim();
if (i == args.Length - 1)
{
ConsoleHelper.WriteError("An argument is missing a value.");
return false;
}
var value = args[i + 1].Trim();
switch (key)
{
case "--help":
case "-h":
case "/h":
ShowHelpText();
Environment.Exit(0);
break;
case "--use-prebuilt":
case "-b":
{
if (bool.TryParse(value, out var result))
{
cliParams.UsePrebuiltPackage = result;
continue;
}
ConsoleHelper.WriteError("--use-prebuilt parameter is invalid. Must be true/yes or false/no.");
return false;
}
case "--web-server":
case "-w":
{
if (int.TryParse(value, out var webServerResult))
{
cliParams.WebServer = (WebServerType)webServerResult;
continue;
}
ConsoleHelper.WriteError($"--web-server parameter is invalid. Must be a " +
$"number (0 - {Enum.GetValues<WebServerType>().Length}).");
return false;
}
case "--server-url":
case "-s":
{
if (Uri.TryCreate(value, UriKind.Absolute, out var result))
{
cliParams.ServerUrl = result;
continue;
}
ConsoleHelper.WriteError("--server-url parameter is invalid. Must be a valid URL (e.g. https://app.remotely.one).");
return false;
}
case "--install-directory":
case "-i":
cliParams.InstallDirectory = value;
continue;
case "--github-username":
case "-u":
cliParams.GitHubUsername = value;
continue;
case "--github-pat":
case "-p":
cliParams.GitHubPat = value;
continue;
case "--reference":
case "-r":
cliParams.Reference = value;
continue;
case "--create-new":
case "-c":
{
if (bool.TryParse(value, out var result))
{
cliParams.CreateNew = result;
continue;
}
ConsoleHelper.WriteError("--create-new parameter is invalid. Must be true/yes or false/no.");
return false;
}
default:
return false;
}
}
catch (Exception ex)
{
ConsoleHelper.WriteError($"Error while parsing command line arguments: {ex.Message}");
return false;
}
}
return true;
}
private static void ShowHelpText()
{
ConsoleHelper.WriteLine("Remotely Server Installer", 0, ConsoleColor.Cyan);
ConsoleHelper.WriteLine("Builds a customized Remotely server using GitHub actions " +
"and installs the server on the local machine.", 1);
ConsoleHelper.WriteLine("Usage:");
ConsoleHelper.WriteLine("\tNo Parameters - Run the installer interactively.", 2);
ConsoleHelper.WriteLine("\t--use-prebuilt, -b (true/false or yes/no) Whether to use the pre-built server package from the " +
"latest public release, or to create a customized package through GitHub Actions. The pre-built package " +
"will not contain your server's URL in the desktop clients, and end users will need to type it in manually.", 1);
ConsoleHelper.WriteLine("\t--github-username, -u Your GitHub username, where the forked Remotely repo exists.", 1);
ConsoleHelper.WriteLine("\t--github-pat, -p The GitHub Personal Access Token to use for authentication. " +
"Create one at ttps://github.com/settings/tokens.", 1);
ConsoleHelper.WriteLine("\t--server-url, -s The public URL where your Remotely server will be accessed (e.g. https://app.remotely.one).", 1);
ConsoleHelper.WriteLine("\t--install-directory, -i The directory path where the server files will be installed (e.g. /var/www/remotely/).", 1);
ConsoleHelper.WriteLine("Enter the GitHub branch or tag name from which to build. For example, you can enter " +
" \"master\" to build the latest changes from the default branch. Or you can enter a release tag like \"v2021.04.13.1604\".", 1);
ConsoleHelper.WriteLine("\t--reference, -r Use \"latest\" to build from the latest full release. Otherwise, you can enter the " +
"name of a branch or tag from which to build. For example, you can enter \"master\" to build the most recent preview changes " +
"from the default branch. Or you can enter a specific release tag like \"v2021.04.13.1604\".", 1);
ConsoleHelper.WriteLine("\t--create-new, -c (true/false or yes/no) Whether to run a new build. If false, the latest existing build artifact will be used.", 1);
ConsoleHelper.WriteLine("\t--web-server, -w Number. The web server that will be used as a reverse proxy to forward " +
"requests to the Remotely server. Select the appropriate option for your operating system and web server. " +
"0 = Caddy on Ubuntu. 1 = Nginx on Ubuntu. 2 = Caddy on CentOS. 3 = Nginx on CentOS. 4 = IIS on Windows Server 2016+.", 1);
ConsoleHelper.WriteLine("Example (build latest release):");
ConsoleHelper.WriteLine("sudo ./Remotely_Server_Installer -b false -u lucent-sea -p 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 -s https://app.remotely.one -i /var/www/remotely/ -r latest -c true -w 0", 1);
ConsoleHelper.WriteLine("Example (use pre-built package):");
ConsoleHelper.WriteLine("sudo ./Remotely_Server_Installer -b true -s https://app.remotely.one -i /var/www/remotely/ -w 0");
}
}
}