-
Notifications
You must be signed in to change notification settings - Fork 908
/
Copy pathSet-EnvironmentVariable.ps1
127 lines (102 loc) · 4.36 KB
/
Set-EnvironmentVariable.ps1
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
# Copyright © 2017 - 2021 Chocolatey Software, Inc.
# Copyright © 2015 - 2017 RealDimensions Software, LLC
# Copyright © 2011 - 2015 RealDimensions Software, LLC & original authors/contributors from https://github.com/chocolatey/chocolatey
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
function Set-EnvironmentVariable {
<#
.SYNOPSIS
**NOTE:** Administrative Access Required when `-Scope 'Machine'.`
DO NOT USE. Not part of the public API. Use
`Install-ChocolateyEnvironmentVariable` instead.
.DESCRIPTION
Saves an environment variable.
.NOTES
This command will assert UAC/Admin privileges on the machine if
`-Scope 'Machine'`.
.INPUTS
None
.OUTPUTS
None
.PARAMETER Name
.PARAMETER Value
.PARAMETER IgnoredArguments
Allows splatting with arguments that do not apply. Do not use directly.
.LINK
Install-ChocolateyEnvironmentVariable
.LINK
Uninstall-ChocolateyEnvironmentVariable
.LINK
Install-ChocolateyPath
.LINK
Get-EnvironmentVariable
#>
param (
[parameter(Mandatory=$true, Position=0)][string] $Name,
[parameter(Mandatory=$false, Position=1)][string] $Value,
[parameter(Mandatory=$false, Position=2)]
[System.EnvironmentVariableTarget] $Scope,
[parameter(ValueFromRemainingArguments = $true)][Object[]] $ignoredArguments
)
Write-FunctionCallLogMessage -Invocation $MyInvocation -Parameters $PSBoundParameters
if ($Scope -eq [System.EnvironmentVariableTarget]::Process -or $Value -eq $null -or $Value -eq '') {
return [Environment]::SetEnvironmentVariable($Name, $Value, $Scope)
}
[string]$keyHive = 'HKEY_LOCAL_MACHINE'
[string]$registryKey = "SYSTEM\CurrentControlSet\Control\Session Manager\Environment\"
[Microsoft.Win32.RegistryKey] $win32RegistryKey = [Microsoft.Win32.Registry]::LocalMachine.OpenSubKey($registryKey)
if ($Scope -eq [System.EnvironmentVariableTarget]::User) {
$keyHive = 'HKEY_CURRENT_USER'
$registryKey = "Environment"
[Microsoft.Win32.RegistryKey] $win32RegistryKey = [Microsoft.Win32.Registry]::CurrentUser.OpenSubKey($registryKey)
}
[Microsoft.Win32.RegistryValueKind]$registryType = [Microsoft.Win32.RegistryValueKind]::String
try {
if ($win32RegistryKey.GetValueNames() -contains $Name)
{
$registryType = $win32RegistryKey.GetValueKind($Name)
}
} catch {
# the value doesn't yet exist
# move along, nothing to see here
}
Write-Debug "Registry type for $Name is/will be $registryType"
if ($Name -eq 'PATH') {
$registryType = [Microsoft.Win32.RegistryValueKind]::ExpandString
}
[Microsoft.Win32.Registry]::SetValue($keyHive + "\" + $registryKey, $Name, $Value, $registryType)
try {
# make everything refresh
# because sometimes explorer.exe just doesn't get the message that things were updated.
if (-not ("win32.nativemethods" -as [type])) {
# import sendmessagetimeout from win32
add-type -Namespace Win32 -Name NativeMethods -MemberDefinition @"
[DllImport("user32.dll", SetLastError = true, CharSet = CharSet.Auto)]
public static extern IntPtr SendMessageTimeout(
IntPtr hWnd, uint Msg, UIntPtr wParam, string lParam,
uint fuFlags, uint uTimeout, out UIntPtr lpdwResult);
"@
}
$HWND_BROADCAST = [intptr]0xffff;
$WM_SETTINGCHANGE = 0x1a;
$result = [uintptr]::zero
# notify all windows of environment block change
[win32.nativemethods]::SendMessageTimeout($HWND_BROADCAST, $WM_SETTINGCHANGE, [uintptr]::Zero, "Environment", 2, 5000, [ref]$result) | Out-Null
# Set a user environment variable making the system refresh
$setx = "$($env:SystemRoot)\System32\setx.exe"
& "$setx" ChocolateyLastPathUpdate `"$((Get-Date).ToFileTime())`" | Out-Null
} catch {
Write-Warning "Failure attempting to let Explorer know about updated environment settings.`n $($_.Exception.Message)"
}
Update-SessionEnvironment
}