Open o-l-a-v opened 2 years ago
Thanks @o-l-a-v for opening this issue, we are investigating this-- to confirm have you hit this issue with our v3 previews?
Don't remember if I've tried with v3 yet.
Should be easy enough for you to reproduce? :)
Tested with beta 3.0.12, it installs to %OneDriveCommercial%\Documents
from both Windows PowerShell and PowerShell 7.2.2.
\WindowsPowerShell
with Windows PowerShell, \PowerShell
with v7.2.2.
# Import module downloaded from PowerShellGallery, extracted with 7-Zip
Import-Module -Name ('{0}\powershellget.3.0.12-beta\PowerShellGet.psd1' -f [System.Environment]::GetFolderPath('Desktop'))
# Check imported modules
Get-Module
# Install a module in user context
PowerShellGet\Install-PSResource -Name 'Az.Cdn' -Scope 'CurrentUser' -Repository 'PSGallery' -Quiet -Confirm:$false -TrustRepository -Reinstall
Related issues from PowerShell/PowerShell:
Thanks @o-l-a-v we are planning to do a deeper dive into these (and other path) issues after our next release
Honestly, whether or not the changes in PowerShell/15552 are made, having a parameter set which gives us the ability to specify the -InstallPath
instead of a -Scope
when calling Install-PSResource
would allow users who really want or need to do this to just change their $Env:PSModulePath
and set a $PSDefaultParameterValues
to make it happen -- without needing to learn to use Save-PSResource
instead.
On company developer laptops, OneDrive routinely causes "Access to the cloud file is denied" errors when trying to upgrade modules there, and starts needlessly mirroring the files back to the cloud, slowing down install even more. Not to mention that every time I remove an old version of Az or Microsoft.Graph it causes that scary warning about how something has deleted thousands of files from my OneDrive...
I was more than happy to solve this in OneDrive "Choose Folders" option, but that doesn't work remotely the way I expected. I just wanted to exclude the Modules folder from syncing entirely but unchecking it in that screen does very different things than that.
Having modules roam between machines (via OneDrive) is desired functionality for me! Please don't turn it off!
I can see that some might want to turn it off, which they can at the moment by setting $env:PSModulesPath
.
I can also see that some sort of alternative mechanism might be desired - eg. some sort of placeholder file, etc. although not all modules are available from the gallery, so quite a bit of design required there.
I would strongly recommend caution when breaking existing workflows by disabling/altering the currently enabled-by-default feature.
@fowl2 thanks for reaching out and providing insight into your use case with this default behavior. This is being discussed and worked on from the PowerShell project side, on this issue linked here: https://github.com/PowerShell/PowerShell/issues/15552
If you can share this comment there, that would be great thanks.
After some more tinkering I've found a solution that works for me.
I settled on %LOCALAPPDATA%\Microsoft\PowerShell\Modules
.
The desired directory must be created prior to step 4.
PSModulePath
to desired directoryImportant for PowerShell to automagically look for modules in desired directory.
Save-Package
for installing modulesUse PackageManagement
cmdlet Save-Package
, which let's you specify path.
Save-Module
.Save-Package
.I later found out that PackageManagement
and PowerShellGet
does not actually use $env:PSModulePath
when searching for installed modules, while Microsoft.PowerShell.Core\Get-Module
and Import-Module
does. Opened an issue on this here:
The problem with Microsoft.PowerShell.Core\Get-Module
though, for me at least, is that it does not return enough attributes on the module, like "Author".
Az
, and you want to filter out modules made by others than Microsoft Corporation
.Got it faster by using .NET classes
Got it even faster by using <string>.Contains()
vs -like
, and .Where({<filter>},'First')
.
I have not researched the behavior of PowerShellGet v3, but in v2, when the Update-Module
command calls the Install-Module
command internally, it uses the value of the InstalledLocation
from the PSGetModuleInfo.xml
file located within the directory where the module is installed to determine the value of the Scope
parameter. This file contains the full path of the module directory. However, if the document directory is synchronized with OneDrive and the same Microsoft account is used on multiple machines, or if the PC is re-setup and the username changes, it is possible that the value of InstalledLocation
and the actual path where the module exists may differ.
Summary of the new feature / enhancement
Behavior today
Default install location for PowerShell scripts and modules when specifying user context, is:
%USERPROFILE%\Documents\WindowsPowerShell\Modules
%USERPROFILE%\Documents\PowerShell\Modules
But if you have OneDrive for Business set up with Known Folder Move (KFM), default install location for user context is:
%OneDriveCommercial%\Documents\WindowsPowerShell\Modules
%OneDriveCommercial%\Documents\PowerShell\Modules
Why is it a problem
This is not ideal, as you'll end up with hundreds or thousands of small files that will be synced up and down to OneDrive, which might cause OneDrive sync issues, and other performance hits.
I currently install all modules to AllUsers scope for this reason. Currently 2.7 GB, 12 617 files, 2 315 folders.
Screenshot
![image](https://user-images.githubusercontent.com/6450056/162154199-f95ee0f9-dc2d-4939-aa96-b6735cadeafb.png)If I did not care about this myself, I'd be using more than 1 / 10 of the capacity / max number of files recommendation for the OneDrive client, just for PowerShell modules.
Proposed technical implementation details
In my opinion, there is no reason to install PowerShell modules from PowerShell Gallery to OneDrive by default when KFM is active. A publicly available PowerShell module is nothing unique that needs to be backed up/ synced.
Option 1 - Cmdlet to set PSResourceLocation for Process/User/Machine
Add cmdlet to set PSResourceLocation for scope Process/User/Machine. For instance:
It could also:
[System.Environment]::GetEnvironmentVariable('PSModulePath','<scope>')
.PackageManagement
,PowerShellGet
andMicrosoft.PowerShell.PSResourceGet
from old path given scope to new path.Option 2 - Use first path in
$env:PSModulePath
if setIf I've set
[System.Environment]::GetEnvironmentVariable('PSModulePath','User').Split(';')[0]
to be somewhere else than the default location for<scope>
, use it.Option 3 - Don't follow KFM redirect
Users must opt in to install PowerShell modules to OneDrive, instead of current default behavior.
%USERPROFILE%\Documents\WindowsPowerShell\Modules
%USERPROFILE%\Documents\PowerShell\Modules
Option 4 - Change default location for user scope to
%LOCALAPPDATA%
Change default location for user context to:
%LOCALAPPDATA%\WindowsPowerShell\Modules
%LOCALAPPDATA%\PowerShell\Modules