Skip to content

Commit

Permalink
(#3318) Add ShouldProcess support to commands
Browse files Browse the repository at this point in the history
Some cmdlets should support ShouldProcess, this change ensures that they
do correctly support it, along with adding checks to the underlying
methods to make it work as users will expect.

As Chocolatey CLI already supports --dry-run itself, this is primarily
going to be useful for effectively unit testing the commands.
  • Loading branch information
vexx32 committed Oct 25, 2024
1 parent c8326fb commit 8034274
Show file tree
Hide file tree
Showing 5 changed files with 59 additions and 46 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -21,7 +21,7 @@

namespace Chocolatey.PowerShell.Commands
{
[Cmdlet(VerbsLifecycle.Install, "ChocolateyPath")]
[Cmdlet(VerbsLifecycle.Install, "ChocolateyPath", SupportsShouldProcess = true, ConfirmImpact = ConfirmImpact.Medium)]
[OutputType(typeof(void))]
public class InstallChocolateyPathCommand : ChocolateyCmdlet
{
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -21,7 +21,7 @@

namespace Chocolatey.PowerShell.Commands
{
[Cmdlet(VerbsCommon.Set, "EnvironmentVariable")]
[Cmdlet(VerbsCommon.Set, "EnvironmentVariable", SupportsShouldProcess = true, ConfirmImpact = ConfirmImpact.Medium)]
[OutputType(typeof(void))]
public sealed class SetEnvironmentVariableCommand : ChocolateyCmdlet
{
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -21,7 +21,7 @@

namespace Chocolatey.PowerShell.Commands
{
[Cmdlet(VerbsLifecycle.Uninstall, "ChocolateyPath")]
[Cmdlet(VerbsLifecycle.Uninstall, "ChocolateyPath", SupportsShouldProcess = true, ConfirmImpact = ConfirmImpact.Medium)]
[OutputType(typeof(void))]
public class UninstallChocolateyPathCommand : ChocolateyCmdlet
{
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -20,7 +20,7 @@

namespace Chocolatey.PowerShell.Commands
{
[Cmdlet(VerbsData.Update, "SessionEnvironment")]
[Cmdlet(VerbsData.Update, "SessionEnvironment", SupportsShouldProcess = true, ConfirmImpact = ConfirmImpact.Low)]
[OutputType(typeof(void))]
public sealed class UpdateSessionEnvironmentCommand : ChocolateyCmdlet
{
Expand Down
97 changes: 55 additions & 42 deletions src/Chocolatey.PowerShell/Helpers/EnvironmentHelper.cs
Original file line number Diff line number Diff line change
Expand Up @@ -139,7 +139,11 @@ public static void SetVariable(PSCmdlet cmdlet, string name, EnvironmentVariable
{
if (scope == EnvironmentVariableTarget.Process)
{
Environment.SetEnvironmentVariable(name, value);
if (cmdlet.ShouldProcess(name, "set Process environment variable"))
{
Environment.SetEnvironmentVariable(name, value);
}

return;
}

Expand All @@ -166,32 +170,38 @@ public static void SetVariable(PSCmdlet cmdlet, string name, EnvironmentVariable

cmdlet.WriteDebug($"Registry type for {name} is/will be {registryType}");

if (string.IsNullOrEmpty(value))
{
registryKey.DeleteValue(name, throwOnMissingValue: false);
}
else
if (cmdlet.ShouldProcess(name, $"set {scope} environment variable"))
{
registryKey.SetValue(name, value, registryType);
if (string.IsNullOrEmpty(value))
{
registryKey.DeleteValue(name, throwOnMissingValue: false);
}
else
{
registryKey.SetValue(name, value, registryType);
}
}
}

try
{
// Trigger environment refresh in explorer.exe:
// 1. Notify all windows of environment block change
NativeMethods.SendMessageTimeout(
hWnd: (IntPtr)NativeMethods.HWND_BROADCAST,
Msg: NativeMethods.WM_SETTINGCHANGE,
wParam: UIntPtr.Zero,
lParam: "Environment",
fuFlags: 2,
uTimeout: 5000,
out UIntPtr result);

// 2. Set a user environment variable making the system refresh
var setxPath = string.Format(@"{0}\System32\setx.exe", GetVariable(cmdlet, EnvironmentVariables.SystemRoot, EnvironmentVariableTarget.Process));
cmdlet.InvokeCommand.InvokeScript($"& \"{setxPath}\" {EnvironmentVariables.ChocolateyLastPathUpdate} \"{DateTime.Now.ToFileTime()}\"");
if (cmdlet.ShouldProcess("Environment variables", "Notify system of changes"))
{
// Trigger environment refresh in explorer.exe:
// 1. Notify all windows of environment block change
NativeMethods.SendMessageTimeout(
hWnd: (IntPtr)NativeMethods.HWND_BROADCAST,
Msg: NativeMethods.WM_SETTINGCHANGE,
wParam: UIntPtr.Zero,
lParam: "Environment",
fuFlags: 2,
uTimeout: 5000,
out UIntPtr result);

// 2. Set a user environment variable making the system refresh
var setxPath = string.Format(@"{0}\System32\setx.exe", GetVariable(cmdlet, EnvironmentVariables.SystemRoot, EnvironmentVariableTarget.Process));
cmdlet.InvokeCommand.InvokeScript($"& \"{setxPath}\" {EnvironmentVariables.ChocolateyLastPathUpdate} \"{DateTime.Now.ToFileTime()}\"");
}
}
catch (Exception error)
{
Expand Down Expand Up @@ -221,37 +231,40 @@ public static void UpdateSession(PSCmdlet cmdlet)
scopeList.Add(EnvironmentVariableTarget.User);
}

foreach (var scope in scopeList)
if (cmdlet.ShouldProcess("current process", "refresh all environment variables"))
{
foreach (var name in GetVariableNames(scope))
foreach (var scope in scopeList)
{
var value = GetVariable(cmdlet, name, scope);
if (!string.IsNullOrEmpty(value))
foreach (var name in GetVariableNames(scope))
{
SetVariable(cmdlet, name, EnvironmentVariableTarget.Process, value);
var value = GetVariable(cmdlet, name, scope);
if (!string.IsNullOrEmpty(value))
{
SetVariable(cmdlet, name, EnvironmentVariableTarget.Process, value);
}
}
}
}

// Update PATH, combining both scopes' values.
var paths = new string[2];
paths[0] = GetVariable(cmdlet, EnvironmentVariables.Path, EnvironmentVariableTarget.Machine);
paths[1] = GetVariable(cmdlet, EnvironmentVariables.Path, EnvironmentVariableTarget.User);
// Update PATH, combining both scopes' values.
var paths = new string[2];
paths[0] = GetVariable(cmdlet, EnvironmentVariables.Path, EnvironmentVariableTarget.Machine);
paths[1] = GetVariable(cmdlet, EnvironmentVariables.Path, EnvironmentVariableTarget.User);

SetVariable(cmdlet, EnvironmentVariables.Path, EnvironmentVariableTarget.Process, string.Join(";", paths));
SetVariable(cmdlet, EnvironmentVariables.Path, EnvironmentVariableTarget.Process, string.Join(";", paths));

// Preserve PSModulePath as it's almost always updated by process, preserve it
SetVariable(cmdlet, EnvironmentVariables.PSModulePath, EnvironmentVariableTarget.Process, psModulePath);
// Preserve PSModulePath as it's almost always updated by process, preserve it
SetVariable(cmdlet, EnvironmentVariables.PSModulePath, EnvironmentVariableTarget.Process, psModulePath);

// Preserve user and architecture
if (!string.IsNullOrEmpty(userName))
{
SetVariable(cmdlet, EnvironmentVariables.Username, EnvironmentVariableTarget.Process, userName);
}
// Preserve user and architecture
if (!string.IsNullOrEmpty(userName))
{
SetVariable(cmdlet, EnvironmentVariables.Username, EnvironmentVariableTarget.Process, userName);
}

if (!string.IsNullOrEmpty(architecture))
{
SetVariable(cmdlet, EnvironmentVariables.ProcessorArchitecture, EnvironmentVariableTarget.Process, architecture);
if (!string.IsNullOrEmpty(architecture))
{
SetVariable(cmdlet, EnvironmentVariables.ProcessorArchitecture, EnvironmentVariableTarget.Process, architecture);
}
}
}
}
Expand Down

0 comments on commit 8034274

Please sign in to comment.