From 0681d0a75bb54be9c8b017bb6f9daddbda863b76 Mon Sep 17 00:00:00 2001 From: Pat Hartl Date: Sun, 15 Jan 2023 04:29:47 -0600 Subject: [PATCH] Save all scripts on a game and allow them to run as admin. --- .../InstallController.cs | 41 +++++++++++++++-- .../PowerShellRuntime.cs | 46 ++++++++++++------- .../UninstallController.cs | 3 +- LANCommander.SDK/Enums/ScriptType.cs | 10 ++++ LANCommander.SDK/Models/Game.cs | 1 + LANCommander.SDK/Models/Script.cs | 17 +++++++ 6 files changed, 97 insertions(+), 21 deletions(-) create mode 100644 LANCommander.SDK/Enums/ScriptType.cs create mode 100644 LANCommander.SDK/Models/Script.cs diff --git a/LANCommander.Playnite.Extension/InstallController.cs b/LANCommander.Playnite.Extension/InstallController.cs index da38b67..cf83a09 100644 --- a/LANCommander.Playnite.Extension/InstallController.cs +++ b/LANCommander.Playnite.Extension/InstallController.cs @@ -1,6 +1,7 @@ using Playnite.SDK; using Playnite.SDK.Models; using Playnite.SDK.Plugins; +using LANCommander.SDK.Enums; using LANCommander.SDK.Extensions; using System; using System.Collections.Generic; @@ -13,6 +14,7 @@ using ICSharpCode.SharpZipLib.Zip; using ICSharpCode.SharpZipLib.Core; using YamlDotNet.Serialization; using YamlDotNet.Serialization.NamingConventions; +using LANCommander.SDK.Models; namespace LANCommander.PlaynitePlugin { @@ -20,9 +22,9 @@ namespace LANCommander.PlaynitePlugin { private LANCommanderLibraryPlugin Plugin; private PowerShellRuntime PowerShellRuntime; - private Game PlayniteGame; + private Playnite.SDK.Models.Game PlayniteGame; - public LANCommanderInstallController(LANCommanderLibraryPlugin plugin, Game game) : base(game) + public LANCommanderInstallController(LANCommanderLibraryPlugin plugin, Playnite.SDK.Models.Game game) : base(game) { Name = "Install using LANCommander"; Plugin = plugin; @@ -49,9 +51,14 @@ namespace LANCommander.PlaynitePlugin File.WriteAllText(Path.Combine(installDirectory, "_manifest.yml"), GetManifest(gameId)); + SaveScript(game, installDirectory, ScriptType.Install); + SaveScript(game, installDirectory, ScriptType.Uninstall); + SaveScript(game, installDirectory, ScriptType.NameChange); + SaveScript(game, installDirectory, ScriptType.KeyChange); + try { - PowerShellRuntime.RunInstallScript(PlayniteGame); + PowerShellRuntime.RunScript(PlayniteGame, ScriptType.Install); } catch { } @@ -169,5 +176,33 @@ namespace LANCommander.PlaynitePlugin return yaml; } + + private void SaveScript(LANCommander.SDK.Models.Game game, string installationDirectory, ScriptType type) + { + var script = game.Scripts.FirstOrDefault(s => s.Type == type); + + if (script == null) + return; + + if (script.RequiresAdmin) + script.Contents = "# Requires Admin" + "\r\n\r\n" + script.Contents; + + Dictionary filenames = new Dictionary() { + { ScriptType.Install, "_install.ps1" }, + { ScriptType.Uninstall, "_uninstall.ps1" }, + { ScriptType.NameChange, "_changename.ps1" }, + { ScriptType.KeyChange, "_changekey.ps1" } + }; + + if (!filenames.ContainsKey(type)) + return; + + var filename = Path.Combine(installationDirectory, filenames[type]); + + if (File.Exists(filename)) + File.Delete(filename); + + File.WriteAllText(filename, script.Contents); + } } } diff --git a/LANCommander.Playnite.Extension/PowerShellRuntime.cs b/LANCommander.Playnite.Extension/PowerShellRuntime.cs index 4eed616..a78a13e 100644 --- a/LANCommander.Playnite.Extension/PowerShellRuntime.cs +++ b/LANCommander.Playnite.Extension/PowerShellRuntime.cs @@ -1,4 +1,5 @@ -using Playnite.SDK.Models; +using LANCommander.SDK.Enums; +using Playnite.SDK.Models; using System; using System.Collections.Generic; using System.Diagnostics; @@ -12,35 +13,46 @@ namespace LANCommander.PlaynitePlugin { internal class PowerShellRuntime { - public void RunScript(string script) + public void RunScript(string path) + { + var process = new Process(); + process.StartInfo.FileName = "powershell.exe"; + process.StartInfo.Arguments = $@"-File ""{path}"""; + process.Start(); + process.WaitForExit(); + } + + public void RunScriptAsAdmin(string path) { var process = new Process(); process.StartInfo.FileName = "powershell.exe"; process.StartInfo.UseShellExecute = true; process.StartInfo.Verb = "runas"; - process.StartInfo.Arguments = $@"-ExecutionPolicy Unrestricted -File ""{script}"""; + process.StartInfo.Arguments = $@"-ExecutionPolicy Unrestricted -File ""{path}"""; process.Start(); process.WaitForExit(); } - public void RunInstallScript(Game game) + public void RunScript(Game game, ScriptType type) { - var scriptPath = Path.Combine(game.InstallDirectory, "_install.ps1"); + Dictionary filenames = new Dictionary() { + { ScriptType.Install, "_install.ps1" }, + { ScriptType.Uninstall, "_uninstall.ps1" }, + { ScriptType.NameChange, "_changename.ps1" }, + { ScriptType.KeyChange, "_changekey.ps1" } + }; - if (!File.Exists(scriptPath)) - throw new FileNotFoundException(scriptPath); + var path = Path.Combine(game.InstallDirectory, filenames[type]); - RunScript(scriptPath); - } + if (File.Exists(path)) + { + var contents = File.ReadAllText(path); - public void RunUninstallScript(Game game) - { - var scriptPath = Path.Combine(game.InstallDirectory, "_uninstall.ps1"); - - if (!File.Exists(scriptPath)) - throw new FileNotFoundException(scriptPath); - - RunScript(scriptPath); + if (contents.StartsWith("# Requires Admin")) + RunScriptAsAdmin(path); + else + RunScript(path); + } } } } diff --git a/LANCommander.Playnite.Extension/UninstallController.cs b/LANCommander.Playnite.Extension/UninstallController.cs index 6fd98c4..158e111 100644 --- a/LANCommander.Playnite.Extension/UninstallController.cs +++ b/LANCommander.Playnite.Extension/UninstallController.cs @@ -11,6 +11,7 @@ using System.Threading; using System.Threading.Tasks; using ICSharpCode.SharpZipLib.Zip; using ICSharpCode.SharpZipLib.Core; +using LANCommander.SDK.Enums; namespace LANCommander.PlaynitePlugin { @@ -30,7 +31,7 @@ namespace LANCommander.PlaynitePlugin { try { - PowerShellRuntime.RunUninstallScript(Game); + PowerShellRuntime.RunScript(Game, ScriptType.Uninstall); } catch { } diff --git a/LANCommander.SDK/Enums/ScriptType.cs b/LANCommander.SDK/Enums/ScriptType.cs new file mode 100644 index 0000000..e1f9b60 --- /dev/null +++ b/LANCommander.SDK/Enums/ScriptType.cs @@ -0,0 +1,10 @@ +namespace LANCommander.SDK.Enums +{ + public enum ScriptType + { + Install, + Uninstall, + NameChange, + KeyChange + } +} diff --git a/LANCommander.SDK/Models/Game.cs b/LANCommander.SDK/Models/Game.cs index 3a22f07..6d504d0 100644 --- a/LANCommander.SDK/Models/Game.cs +++ b/LANCommander.SDK/Models/Game.cs @@ -14,5 +14,6 @@ namespace LANCommander.SDK.Models public virtual Company Publisher { get; set; } public virtual Company Developer { get; set; } public virtual IEnumerable Archives { get; set; } + public virtual IEnumerable