diff --git a/.github/actions/spelling/expect.txt b/.github/actions/spelling/expect.txt index 1f26c71dd8..580cb0be4b 100644 --- a/.github/actions/spelling/expect.txt +++ b/.github/actions/spelling/expect.txt @@ -10,6 +10,7 @@ aicli AICLIC ajor alreadyinstalled +amd amrutha anonymized APARTMENTTHREADED @@ -19,6 +20,7 @@ apiset appinstallertest appname Archs +arget argumentlist ARMNT arp @@ -49,6 +51,7 @@ blogs bluetooth bomgar BOMs +boundparms brk Buf BUILTINS @@ -97,6 +100,7 @@ dirs diskfull dnld Dobbeleer +dsc dustojnikhummer dvinns dw @@ -138,6 +142,8 @@ GHS gity Globals Google +hackathon +hashtable helplib helplibrary hhx @@ -212,6 +218,7 @@ LTDA lw lz malware +maxvalue MBH mdmp megamorf @@ -249,6 +256,7 @@ nonetwork normer NOSEPARATOR NOTAPROPERTY +notmatch npp nsis nuffing @@ -260,6 +268,7 @@ ofile Outptr Packagedx packageinuse +parametermap pathparts pathpaths Patil @@ -273,11 +282,15 @@ pkgmgr pkindex PMS positionals +powershellgallery powertoys pri processthreads productcode +pscustomobject pseudocode +psm +psobject pvk pvm pwabuilder @@ -298,6 +311,7 @@ REFIID regexes REGSAM restsource +rgex rhs riid roblox @@ -351,6 +365,7 @@ Tlg tombstoned tpl transitioning +trimstart UCase ucasemap UChars @@ -389,6 +404,7 @@ Webserver website WERSJA wesome +whatif windir windowsdeveloper winerror @@ -406,5 +422,6 @@ WZDNCRFJ XPLATSTR xsi yamlcreateps +ype Zanollo zy diff --git a/.gitignore b/.gitignore index 3e759b75bf..20106aeb70 100644 --- a/.gitignore +++ b/.gitignore @@ -8,6 +8,7 @@ *.user *.userosscache *.sln.docstates +settings.json # User-specific files (MonoDevelop/Xamarin Studio) *.userprefs diff --git a/tools/PowerShell/Microsoft.WinGet.Client/crescendo/Get-WinGetVersion.json b/tools/PowerShell/Microsoft.WinGet.Client/crescendo/Get-WinGetVersion.json new file mode 100644 index 0000000000..f9247e461e --- /dev/null +++ b/tools/PowerShell/Microsoft.WinGet.Client/crescendo/Get-WinGetVersion.json @@ -0,0 +1,68 @@ +{ + "$schema": "https://aka.ms/crescendo/schema.json", + "Commands": [ + { + "Verb": "Get", + "Noun": "WinGetVersion", + "OriginalName": "winget", + "OriginalCommandElements": [ + "--version" + ], + "OutputHandlers": [ + { + "ParameterSetName": "default", + "StreamOutput": true, + "HandlerType": "Inline", + "Handler": "$Input.trimstart('v') -as [version]" + } + ] + }, + { + "Verb": "Get", + "Noun": "WinGetSource", + "OriginalName": "winget", + "DefaultParameterSetName": "default", + "Parameters": [ + { + "Name": "Export", + "ParameterSetName": ["export"], + "ParameterType": "switch", + "OriginalName": "Export" + }, + { + "Name": "default", + "ParameterSetName": ["default"], + "ParameterType": "switch", + "OriginalName": "default" + } + ], + "OriginalCommandElements": [ + "source", "export" + ], + "OutputHandlers": [ + { + "ParameterSetName": "default", + "StreamOutput": true, + "HandlerType": "Inline", + "Handler": "$Input | ConvertFrom-Json | add-member -pass -typename WinGetSource" + } + ] + }, + { + "Verb": "Get", + "Noun": "WinGetSource", + "OriginalName": "winget", + "OriginalCommandElements": [ + "source", "export" + ], + "OutputHandlers": [ + { + "ParameterSetName": "default", + "StreamOutput": true, + "HandlerType": "Inline", + "Handler": "$Input | ConvertFrom-Json | add-member -pass -typename WinGetSource" + } + ] + } + ] +} \ No newline at end of file diff --git a/tools/PowerShell/Microsoft.WinGet.Client/crescendo/WindowsPackageManager.format.ps1xml b/tools/PowerShell/Microsoft.WinGet.Client/crescendo/WindowsPackageManager.format.ps1xml new file mode 100644 index 0000000000..6b42692259 --- /dev/null +++ b/tools/PowerShell/Microsoft.WinGet.Client/crescendo/WindowsPackageManager.format.ps1xml @@ -0,0 +1,26 @@ + + + + WinGetSourceTable + + WinGetSource + + + + + + + + + + + Name + Arg + + + + + + + + diff --git a/tools/PowerShell/Microsoft.WinGet.Client/crescendo/WindowsPackageManager.psd1 b/tools/PowerShell/Microsoft.WinGet.Client/crescendo/WindowsPackageManager.psd1 new file mode 100644 index 0000000000..50926a8796 --- /dev/null +++ b/tools/PowerShell/Microsoft.WinGet.Client/crescendo/WindowsPackageManager.psd1 @@ -0,0 +1,139 @@ +# +# Module manifest for module 'WindowsPackageManager' +# +# Generated by: denelon +# +# Generated on: 10/12/2021 +# + +@{ + +# Script module or binary module file associated with this manifest. +RootModule = 'WindowsPackageManager.psm1' + +# Version number of this module. +ModuleVersion = '0.0.1' + +# Supported PSEditions +# CompatiblePSEditions = @() + +# ID used to uniquely identify this module +GUID = '79355747-76a0-4d6d-b182-4a8715b8fc59' + +# Author of this module +Author = 'denelon' + +# Company or vendor of this module +CompanyName = 'Microsoft Corporation' + +# Copyright statement for this module +Copyright = 'Copyright Microsoft Corporation. All rights reserved.' + +# Description of the functionality provided by this module +# Description = '' + +# Minimum version of the PowerShell engine required by this module +PowerShellVersion = '5.1.0' + +# Name of the PowerShell host required by this module +# PowerShellHostName = '' + +# Minimum version of the PowerShell host required by this module +# PowerShellHostVersion = '' + +# Minimum version of Microsoft .NET Framework required by this module. This prerequisite is valid for the PowerShell Desktop edition only. +# DotNetFrameworkVersion = '' + +# Minimum version of the common language runtime (CLR) required by this module. This prerequisite is valid for the PowerShell Desktop edition only. +# ClrVersion = '' + +# Processor architecture (None, X86, Amd64) required by this module +# ProcessorArchitecture = '' + +# Modules that must be imported into the global environment prior to importing this module +# RequiredModules = @() + +# Assemblies that must be loaded prior to importing this module +# RequiredAssemblies = @() + +# Script files (.ps1) that are run in the caller's environment prior to importing this module. +# ScriptsToProcess = @() + +# Type files (.ps1xml) to be loaded when importing this module +# TypesToProcess = @() + +# Format files (.ps1xml) to be loaded when importing this module +# FormatsToProcess = @() + +# Modules to import as nested modules of the module specified in RootModule/ModuleToProcess +# NestedModules = @() + +# Functions to export from this module, for best performance, do not use wild cards and do not delete the entry, use an empty array if there are no functions to export. +FunctionsToExport = 'Get-WinGetVersion', 'Get-WinGetSource' + +# Cmdlets to export from this module, for best performance, do not use wild cards and do not delete the entry, use an empty array if there are no cmdlets to export. +CmdletsToExport = @() + +# Variables to export from this module +# VariablesToExport = @() + +# Aliases to export from this module, for best performance, do not use wild cards and do not delete the entry, use an empty array if there are no aliases to export. +AliasesToExport = @() + +# DSC resources to export from this module +# DscResourcesToExport = @() + +# List of all modules packaged with this module +# ModuleList = @() + +# List of all files packaged with this module +# FileList = @() + +# Private data to pass to the module specified in RootModule/ModuleToProcess. This may also contain a PSData hashtable with additional module metadata used by PowerShell. +PrivateData = @{ + + PSData = @{ + + # Tags applied to this module. These help with module discovery in online galleries. + Tags = 'CrescendoBuilt' + + # A URL to the license for this module. + # LicenseUri = '' + + # A URL to the main website for this project. + # ProjectUri = '' + + # A URL to an icon representing this module. + # IconUri = '' + + # ReleaseNotes of this module + # ReleaseNotes = '' + + # Prerelease string of this module + # Prerelease = '' + + # Flag to indicate whether the module requires explicit user acceptance for install/update/save + # RequireLicenseAcceptance = $false + + # External dependent modules of this module + # ExternalModuleDependencies = @() + + } # End of PSData hashtable + + + # CrescendoVersion + CrescendoVersion = '0.6.1' + + # CrescendoGenerated + CrescendoGenerated = '10/12/2021 13:21:52' + +} # End of PrivateData hashtable + +# HelpInfo URI of this module +# HelpInfoURI = '' + +# Default prefix for commands exported from this module. Override the default prefix using Import-Module -Prefix. +# DefaultCommandPrefix = '' + +} + diff --git a/tools/PowerShell/Microsoft.WinGet.Client/crescendo/WindowsPackageManager.psm1 b/tools/PowerShell/Microsoft.WinGet.Client/crescendo/WindowsPackageManager.psm1 new file mode 100644 index 0000000000..321a55152e --- /dev/null +++ b/tools/PowerShell/Microsoft.WinGet.Client/crescendo/WindowsPackageManager.psm1 @@ -0,0 +1,196 @@ +# Module created by Microsoft.PowerShell.Crescendo +class PowerShellCustomFunctionAttribute : System.Attribute { + [bool]$RequiresElevation + [string]$Source + PowerShellCustomFunctionAttribute() { $this.RequiresElevation = $false; $this.Source = "Microsoft.PowerShell.Crescendo" } + PowerShellCustomFunctionAttribute([bool]$rElevation) { + $this.RequiresElevation = $rElevation + $this.Source = "Microsoft.PowerShell.Crescendo" + } +} + + + +function Get-WinGetVersion +{ +[PowerShellCustomFunctionAttribute(RequiresElevation=$False)] +[CmdletBinding()] + +param( ) + +BEGIN { + $__PARAMETERMAP = @{} + $__outputHandlers = @{ + default = @{ StreamOutput = $True; Handler = { $Input.trimstart('v') -as [version] } } + } +} + +PROCESS { + $__commandArgs = @( + '--version' + ) + $__boundparms = $PSBoundParameters + $MyInvocation.MyCommand.Parameters.Values.Where({$_.SwitchParameter -and $_.Name -notmatch "Debug|Whatif|Confirm|Verbose" -and ! $PSBoundParameters[$_.Name]}).ForEach({$PSBoundParameters[$_.Name] = [switch]::new($false)}) + if ($PSBoundParameters["Debug"]){wait-debugger} + foreach ($paramName in $PSBoundParameters.Keys|Sort-Object {$__PARAMETERMAP[$_].OriginalPosition}) { + $value = $PSBoundParameters[$paramName] + $param = $__PARAMETERMAP[$paramName] + if ($param) { + if ( $value -is [switch] ) { $__commandArgs += if ( $value.IsPresent ) { $param.OriginalName } else { $param.DefaultMissingValue } } + elseif ( $param.NoGap ) { $__commandArgs += "{0}""{1}""" -f $param.OriginalName, $value } + else { $__commandArgs += $param.OriginalName; $__commandArgs += $value |Foreach-Object {$_}} + } + } + $__commandArgs = $__commandArgs|Where-Object {$_} + if ($PSBoundParameters["Debug"]){wait-debugger} + if ( $PSBoundParameters["Verbose"]) { + Write-Verbose -Verbose -Message winget + $__commandArgs | Write-Verbose -Verbose + } + $__handlerInfo = $__outputHandlers[$PSCmdlet.ParameterSetName] + if (! $__handlerInfo ) { + $__handlerInfo = $__outputHandlers["Default"] # Guaranteed to be present + } + $__handler = $__handlerInfo.Handler + if ( $PSCmdlet.ShouldProcess("winget $__commandArgs")) { + if ( $__handlerInfo.StreamOutput ) { + & "winget" $__commandArgs | & $__handler + } + else { + $result = & "winget" $__commandArgs + & $__handler $result + } + } + } # end PROCESS + +<# +.SYNOPSIS +Windows Package Manager v1.1.12702 +Copyright (c) Microsoft Corporation. All rights reserved. + +The winget command line utility enables installing applications and other packages from the command line. + +usage: winget [] [] + +The following commands are available: + install Installs the given package + show Shows information about a package + source Manage sources of packages + search Find and show basic info of packages + list Display installed packages + upgrade Upgrades the given package + uninstall Uninstalls the given package + hash Helper to hash installer files + validate Validates a manifest file + settings Open settings or set administrator settings + features Shows the status of experimental features + export Exports a list of the installed packages + import Installs all the packages in a file + +For more details on a specific command, pass it the help argument. [-?] + +The following options are available: + -v,--version Display the version of the tool + --info Display general info of the tool + +More help can be found at: https://aka.ms/winget-command-help + +.DESCRIPTION See help for winget + +#> +} + + + + +function Get-WinGetSource +{ +[PowerShellCustomFunctionAttribute(RequiresElevation=$False)] +[CmdletBinding()] + +param( ) + +BEGIN { + $__PARAMETERMAP = @{} + $__outputHandlers = @{ + default = @{ StreamOutput = $True; Handler = { $Input | ConvertFrom-Json | add-member -pass -typename WinGetSource } } + } +} + +PROCESS { + $__commandArgs = @( + 'source' + 'export' + ) + $__boundparms = $PSBoundParameters + $MyInvocation.MyCommand.Parameters.Values.Where({$_.SwitchParameter -and $_.Name -notmatch "Debug|Whatif|Confirm|Verbose" -and ! $PSBoundParameters[$_.Name]}).ForEach({$PSBoundParameters[$_.Name] = [switch]::new($false)}) + if ($PSBoundParameters["Debug"]){wait-debugger} + foreach ($paramName in $PSBoundParameters.Keys|Sort-Object {$__PARAMETERMAP[$_].OriginalPosition}) { + $value = $PSBoundParameters[$paramName] + $param = $__PARAMETERMAP[$paramName] + if ($param) { + if ( $value -is [switch] ) { $__commandArgs += if ( $value.IsPresent ) { $param.OriginalName } else { $param.DefaultMissingValue } } + elseif ( $param.NoGap ) { $__commandArgs += "{0}""{1}""" -f $param.OriginalName, $value } + else { $__commandArgs += $param.OriginalName; $__commandArgs += $value |Foreach-Object {$_}} + } + } + $__commandArgs = $__commandArgs|Where-Object {$_} + if ($PSBoundParameters["Debug"]){wait-debugger} + if ( $PSBoundParameters["Verbose"]) { + Write-Verbose -Verbose -Message winget + $__commandArgs | Write-Verbose -Verbose + } + $__handlerInfo = $__outputHandlers[$PSCmdlet.ParameterSetName] + if (! $__handlerInfo ) { + $__handlerInfo = $__outputHandlers["Default"] # Guaranteed to be present + } + $__handler = $__handlerInfo.Handler + if ( $PSCmdlet.ShouldProcess("winget $__commandArgs")) { + if ( $__handlerInfo.StreamOutput ) { + & "winget" $__commandArgs | & $__handler + } + else { + $result = & "winget" $__commandArgs + & $__handler $result + } + } + } # end PROCESS + +<# +.SYNOPSIS +Windows Package Manager v1.1.12702 +Copyright (c) Microsoft Corporation. All rights reserved. + +The winget command line utility enables installing applications and other packages from the command line. + +usage: winget [] [] + +The following commands are available: + install Installs the given package + show Shows information about a package + source Manage sources of packages + search Find and show basic info of packages + list Display installed packages + upgrade Upgrades the given package + uninstall Uninstalls the given package + hash Helper to hash installer files + validate Validates a manifest file + settings Open settings or set administrator settings + features Shows the status of experimental features + export Exports a list of the installed packages + import Installs all the packages in a file + +For more details on a specific command, pass it the help argument. [-?] + +The following options are available: + -v,--version Display the version of the tool + --info Display general info of the tool + +More help can be found at: https://aka.ms/winget-command-help + +.DESCRIPTION See help for winget + +#> +} + + diff --git a/tools/PowerShell/Microsoft.WinGet.Client/doc/README.md b/tools/PowerShell/Microsoft.WinGet.Client/doc/README.md new file mode 100644 index 0000000000..d0b1722291 --- /dev/null +++ b/tools/PowerShell/Microsoft.WinGet.Client/doc/README.md @@ -0,0 +1 @@ +Placeholder for PlatyPS generated help files diff --git a/tools/PowerShell/Microsoft.WinGet.Client/src/Library/Add-WinGetSource.ps1 b/tools/PowerShell/Microsoft.WinGet.Client/src/Library/Add-WinGetSource.ps1 new file mode 100644 index 0000000000..186c8b2f7d --- /dev/null +++ b/tools/PowerShell/Microsoft.WinGet.Client/src/Library/Add-WinGetSource.ps1 @@ -0,0 +1,64 @@ +Function Add-WinGetSource +{ + <# + .SYNOPSIS + Adds a source to the Windows Package Manager. + + .DESCRIPTION + By running this cmdlet with the will add a new source to the Windows Package Manager. + + .PARAMETER Name + Used to specify the Name of the source + + .PARAMETER Arg + Used to specify the URL for the source + + .PARAMETER Type + Used to specify the Type of source to add + Currently the only two valid values are "Microsoft.Rest" and "Microsoft.PreIndexed.Package" + + .PARAMETER Header + Used to specify the value to pass as the "Windows-Package-Manager" HTTP header for a REST source. + + .PARAMETER AcceptSourceAgreement + Used to explicitly accept any agreement required by the source. + + .PARAMETER VerboseLog + Used to provide verbose logging for the Windows Package Manager. + + .EXAMPLE + Add-WinGetSource -Name "custom" -Argument "https://contoso.com/" -Type "Microsoft.Rest" + + This example adds a new source to the Windows Package Manager named "custom" + + .EXAMPLE + Add-WinGetSource -Name "custom" -Argument "https://contoso.com/" -Type "Microsoft.Rest" -Header "string" + + This example adds a new source to the Windows Package Manager named "custom" and passes the value "string" in the Windows-Package-Manager HTTP header to the source. + #> + + PARAM( + [Parameter(Mandatory=$true)] [string] $Name, + [Parameter(Mandatory=$true)] [string] $Argument, + [Parameter(Mandatory=$true)] [ValidateSet("Microsoft.Rest", "Microsoft.PreIndexed.Package")] [string] $Type, + [Parameter()] [ValidateLength(1, 1024)] $Header, + [Parameter()] [switch] $AcceptSourceAgreement + ) + BEGIN + { + [string[]] $WinGetArgs = "Source", "Add" + $WinGetArgs += "--Name", $Name + $WinGetArgs += "--Arg", $Argument + $WinGetArgs += "--Type", $Type + } + PROCESS + { + & "WinGet" $WingetArgs + } + END + { + return + } +} + +Export-ModuleMember -Function Add-WinGetSource \ No newline at end of file diff --git a/tools/PowerShell/Microsoft.WinGet.Client/src/Library/Disable-WinGetLocalManifest.ps1 b/tools/PowerShell/Microsoft.WinGet.Client/src/Library/Disable-WinGetLocalManifest.ps1 new file mode 100644 index 0000000000..a5e77e2fc8 --- /dev/null +++ b/tools/PowerShell/Microsoft.WinGet.Client/src/Library/Disable-WinGetLocalManifest.ps1 @@ -0,0 +1,35 @@ +Function Disable-WinGetLocalManifest +{ + <# + .SYNOPSIS + Disables the Windows Package Manager to work with local manifest files. + + .DESCRIPTION + By running this cmdlet the Windows Package Manager will configured not to support working with manifests on the file system. + + .PARAMETER VerboseLog + Used to provide verbose logging for the Windows Package Manager. + + .EXAMPLE + Disable-WinGetLocalManifest + + This cmdlet must be executed in an administrative terminal. + #> + PARAM ( + [Parameter()] [switch] $VerboseLog + + ) + BEGIN + { + [string[]] $WinGetArgs = "Settings", "--disable", "LocalManifestFiles" + $WinGetArgs += "--Verbose-Logs", $VerboseLog + } + PROCESS{ + & "WinGet" $WingetArgs + } + END{ + return + } +} + +Export-ModuleMember -Function Disable-WinGetLocalManifest \ No newline at end of file diff --git a/tools/PowerShell/Microsoft.WinGet.Client/src/Library/Edit-WinGetClientSetting.ps1 b/tools/PowerShell/Microsoft.WinGet.Client/src/Library/Edit-WinGetClientSetting.ps1 new file mode 100644 index 0000000000..e79eb0672c --- /dev/null +++ b/tools/PowerShell/Microsoft.WinGet.Client/src/Library/Edit-WinGetClientSetting.ps1 @@ -0,0 +1,36 @@ +Function Edit-WinGetClientSetting +{ + <# + .SYNOPSIS + Opens the Windows Package Manager settings.json file in the default editor. + + .DESCRIPTION + By running this cmdlet will open the Windows Package Manager settings.json in the application configured for editing JSON files. + + .PARAMETER VerboseLog + Used to provide verbose logging for the Windows Package Manager. + + .EXAMPLE + Edit-WinGetClientSetting + + The settings.json file for the Windows Package Manager will be opened in the default .json editor + #> + + + PARAM ( + [Parameter()] [switch] $VerboseLog + ) + BEGIN + { + [string[]] $WinGetArgs = "Settings" + $WinGetArgs += "--Verbose-Logs", $VerboseLog + } + PROCESS{ + & "WinGet" $WingetArgs + } + END{ + return + } +} + +Export-ModuleMember -Function Edit-WinGetClientSetting \ No newline at end of file diff --git a/tools/PowerShell/Microsoft.WinGet.Client/src/Library/Enable-WinGetLocalManifest.ps1 b/tools/PowerShell/Microsoft.WinGet.Client/src/Library/Enable-WinGetLocalManifest.ps1 new file mode 100644 index 0000000000..897bb92c8a --- /dev/null +++ b/tools/PowerShell/Microsoft.WinGet.Client/src/Library/Enable-WinGetLocalManifest.ps1 @@ -0,0 +1,30 @@ +Function Enable-WinGetLocalManifest +{ + <# + .SYNOPSIS + Enables the Windows Package Manager to work with local manifest files. + + .DESCRIPTION + By running this cmdlet the Windows Package Manager will configured to support working with manifests on the file system. + + .PARAMETER VerboseLog + Used to provide verbose logging for the Windows Package Manager. + #> + + PARAM ( + [Parameter()] [switch] $VerboseLog + ) + BEGIN + { + [string[]] $WinGetArgs = "Settings", "--enable", "LocalManifestFiles" + $WinGetArgs += "--Verbose-Logs", $VerboseLog + } + PROCESS{ + & "WinGet" $WingetArgs + } + END{ + return + } +} + +Export-ModuleMember -Function Enable-WinGetLocalManifest \ No newline at end of file diff --git a/tools/PowerShell/Microsoft.WinGet.Client/src/Library/Find-WinGetPackage.ps1 b/tools/PowerShell/Microsoft.WinGet.Client/src/Library/Find-WinGetPackage.ps1 new file mode 100644 index 0000000000..583fee866f --- /dev/null +++ b/tools/PowerShell/Microsoft.WinGet.Client/src/Library/Find-WinGetPackage.ps1 @@ -0,0 +1,144 @@ +Function Find-WinGetPackage{ + <# + .SYNOPSIS + Searches for a package on configured sources. + Additional options can be provided to filter the output, much like the search command. + + .DESCRIPTION + By running this cmdlet with the required inputs, it will retrieve the packages installed on the local system. + + .PARAMETER Filter + Used to search across multiple fields of the package. + + .PARAMETER Id + Used to specify the Id of the package + + .PARAMETER Name + Used to specify the Name of the package + + .PARAMETER Moniker + Used to specify the Moniker of the package + + .PARAMETER Tag + Used to specify the Tag of the package + + .PARAMETER Command + Used to specify the Command of the package + + .PARAMETER Exact + Used to specify an exact match for any parameters provided. Many of the other parameters may be used for case insensitive substring matches if Exact is not specified. + + .PARAMETER Source + Name of the Windows Package Manager private source. Can be identified by running: "Get-WinGetSource" and using the source Name + + .PARAMETER Count + Used to specify the maximum number of packages to return + + .PARAMETER Header + Used to specify the value to pass as the "Windows-Package-Manager" HTTP header for a REST source. + + .PARAMETER VerboseLog + Used to provide verbose logging for the Windows Package Manager. + + .PARAMETER AcceptSourceAgreement + Used to accept any source agreement required for the source. + + .EXAMPLE + Find-WinGetPackage -id "Publisher.Package" + + This example searches for a package containing "Publisher.Package" as a valid identifier on all configured sources. + + .EXAMPLE + Find-WinGetPackage -id "Publisher.Package" -source "Private" + + This example searches for a package containing "Publisher.Package" as a valid identifier from the source named "Private". + + .EXAMPLE + Find-WinGetPackage -Name "Package" + + This example searches for a package containing "Package" as a valid name on all configured sources. + #> + PARAM( + [Parameter(Position=0)] $Filter, + [Parameter()] $Id, + [Parameter()] $Name, + [Parameter()] $Moniker, + [Parameter()] $Tag, + [Parameter()] $Command, + [Parameter()] [switch] $Exact, + [Parameter()] $Source, + [Parameter()] [ValidateRange(1, [int]::maxvalue)][int]$Count, + [Parameter()] [ValidateLength(1, 1024)]$Header, + [Parameter()] [switch] $VerboseLog, + [Parameter()] [switch] $AcceptSourceAgreement + ) + BEGIN + { + [string[]] $WinGetArgs = @("Search") + [WinGetPackage[]] $Result = @() + [string[]] $IndexTitles = @("Name", "Id", "Version", "Available", "Source") + + if($PSBoundParameters.ContainsKey('Filter')){ + ## Search across Name, ID, moniker, and tags + $WinGetArgs += $Filter + } + if($PSBoundParameters.ContainsKey('Id')){ + ## Search for the ID + $WinGetArgs += "--Id", $Id.Replace("…", "") + } + if($PSBoundParameters.ContainsKey('Name')){ + ## Search for the Name + $WinGetArgs += "--Name", $Name.Replace("…", "") + } + if($PSBoundParameters.ContainsKey('Moniker')){ + ## Search for the Moniker + $WinGetArgs += "--Moniker", $Moniker.Replace("…", "") + } + if($PSBoundParameters.ContainsKey('Tag')){ + ## Search for the Tag + $WinGetArgs += "--Tag", $Tag.Replace("…", "") + } + if($PSBoundParameters.ContainsKey('Command')){ + ## Search for the Moniker + $WinGetArgs += "--Command", $Command.Replace("…", "") + } + if($Exact){ + ## Search using exact values specified (case sensitive) + $WinGetArgs += "--Exact" + } + if($PSBoundParameters.ContainsKey('Source')){ + ## Search for the Source + $WinGetArgs += "--Source", $Source.Replace("…", "") + } + if($PSBoundParameters.ContainsKey('Count')){ + ## Specify the number of results to return + $WinGetArgs += "--Count", $Count + } + if($PSBoundParameters.ContainsKey('Header')){ + ## Pass the value specified as the Windows-Package-Manager HTTP header + $WinGetArgs += "--header", $Header + } + if($PSBoundParameters.ContainsKey('VerboseLog')){ + ## Search using exact values specified (case sensitive) + $WinGetArgs += "--VerboseLog", $VerboseLog + } + if($AcceptSourceAgreement){ + ## Accept source agreements + $WinGetArgs += "--accept-source-agreements" + } + } + PROCESS + { + $List = Invoke-WinGetCommand -WinGetArgs $WinGetArgs -IndexTitles $IndexTitles + + foreach ($Obj in $List) { + $Result += [WinGetPackage]::New($Obj) + } + } + END + { + return $Result + } +} + +Export-ModuleMember -Function Find-WinGetPackage \ No newline at end of file diff --git a/tools/PowerShell/Microsoft.WinGet.Client/src/Library/Get-WinGetPackage.ps1 b/tools/PowerShell/Microsoft.WinGet.Client/src/Library/Get-WinGetPackage.ps1 new file mode 100644 index 0000000000..7a4e0e5af6 --- /dev/null +++ b/tools/PowerShell/Microsoft.WinGet.Client/src/Library/Get-WinGetPackage.ps1 @@ -0,0 +1,133 @@ +Function Get-WinGetPackage{ + <# + .SYNOPSIS + Gets installed packages on the local system. displays the packages installed on the system, as well as whether an update is available. + Additional options can be provided to filter the output, much like the search command. + + .DESCRIPTION + By running this cmdlet with the required inputs, it will retrieve the packages installed on the local system. + + .PARAMETER Filter + Used to search across multiple fields of the package. + + .PARAMETER Id + Used to specify the Id of the package + + .PARAMETER Name + Used to specify the Name of the package + + .PARAMETER Moniker + Used to specify the Moniker of the package + + .PARAMETER Tag + Used to specify the Tag of the package + + .PARAMETER Command + Used to specify the Command of the package + + .PARAMETER Count + Used to specify the maximum number of packages to return + + .PARAMETER Exact + Used to specify an exact match for any parameters provided. Many of the other parameters may be used for case insensitive substring matches if Exact is not specified. + + .PARAMETER Source + Name of the Windows Package Manager private source. Can be identified by running: "Get-WinGetSource" and using the source Name + + .PARAMETER Header + Used to specify the value to pass as the "Windows-Package-Manager" HTTP header for a REST source. + + .PARAMETER AcceptSourceAgreement + Used to accept any source agreements required by a REST source. + + .EXAMPLE + Get-WinGetPackage -id "Publisher.Package" + + This example expects only a single configured REST source with a package containing "Publisher.Package" as a valid identifier. + + .EXAMPLE + Get-WinGetPackage -id "Publisher.Package" -source "Private" + + This example expects the REST source named "Private" with a package containing "Publisher.Package" as a valid identifier. + + .EXAMPLE + Get-WinGetPackage -Name "Package" + + This example expects the REST source named "Private" with a package containing "Package" as a valid name. + #> + + PARAM( + [Parameter(Position=0)] $Filter, + [Parameter()] $Name, + [Parameter()] $Id, + [Parameter()] $Moniker, + [Parameter()] $Tag, + [Parameter()] $Source, + [Parameter()] $Command, + [Parameter()] [ValidateRange(1, [int]::maxvalue)][int]$Count, + [Parameter()] [switch]$Exact, + [Parameter()] [ValidateLength(1, 1024)]$Header, + [Parameter()] [switch]$AcceptSourceAgreement + ) + BEGIN + { + [string[]] $WinGetArgs = @("List") + [WinGetPackage[]]$Result = @() + [string[]] $IndexTitles = @("Name", "Id", "Version", "Available", "Source") + + if($Filter){ + ## Search across Name, ID, moniker, and tags + $WinGetArgs += $Filter + } + if($PSBoundParameters.ContainsKey('Name')){ + ## Search for the Name + $WinGetArgs += "--Name", $Name.Replace("…", "") + } + if($PSBoundParameters.ContainsKey('Id')){ + ## Search for the ID + $WinGetArgs += "--Id", $Id.Replace("…", "") + } + if($PSBoundParameters.ContainsKey('Moniker')){ + ## Search for the Moniker + $WinGetArgs += "--Moniker", $Moniker.Replace("…", "") + } + if($PSBoundParameters.ContainsKey('Tag')){ + ## Search for the Tag + $WinGetArgs += "--Tag", $Tag.Replace("…", "") + } + if($PSBoundParameters.ContainsKey('Source')){ + ## Search for the Source + $WinGetArgs += "--Source", $Source.Replace("…", "") + } + if($PSBoundParameters.ContainsKey('Count')){ + ## Specify the number of results to return + $WinGetArgs += "--Count", $Count + } + if($Exact){ + ## Search using exact values specified (case sensitive) + $WinGetArgs += "--Exact" + } + if($PSBoundParameters.ContainsKey('Header')){ + ## Pass the value specified as the Windows-Package-Manager HTTP header + $WinGetArgs += "--header", $Header + } + if($AcceptSourceAgreement){ + ## Accept source agreements + $WinGetArgs += "--accept-source-agreements" + } + } + PROCESS + { + $List = Invoke-WinGetCommand -WinGetArgs $WinGetArgs -IndexTitles $IndexTitles + + foreach ($Obj in $List) { + $Result += [WinGetPackage]::New($Obj) + } + } + END + { + return $Result + } +} + +Export-ModuleMember -Function Get-WinGetPackage \ No newline at end of file diff --git a/tools/PowerShell/Microsoft.WinGet.Client/src/Library/Get-WinGetSource.ps1 b/tools/PowerShell/Microsoft.WinGet.Client/src/Library/Get-WinGetSource.ps1 new file mode 100644 index 0000000000..9b31993678 --- /dev/null +++ b/tools/PowerShell/Microsoft.WinGet.Client/src/Library/Get-WinGetSource.ps1 @@ -0,0 +1,80 @@ +Function Get-WinGetSource +{ + <# + .SYNOPSIS + Gets the configured sources. + Additional options can be provided to filter the output, much like the search command. + + .DESCRIPTION + By running this cmdlet with the will retrieve the sources configured on the local system. + + .PARAMETER Filter + Used to search for the source by name. + + .PARAMETER Name + Used to specify the Name of the package + + .PARAMETER VerboseLog + Used to provide verbose logging for the Windows Package Manager. + + .EXAMPLE + Get-WinGetSource "winget" + + This example returns the configured source named "winget". + + .EXAMPLE + Get-WinGetSource "winget" -Name "msstore" + + This example returns the configured source named "msstore". + + .EXAMPLE + Find-WinGetPackage -Name "Package" + + This example searches for a package containing "Package" as a valid name on all configured sources. + #> + + PARAM( + [Parameter()] [string] $Name, + [Parameter()] [switch] $VerboseLog + ) + BEGIN + { + [string[]] $WinGetArgs = "Source", "Export" + [WinGetSource[]] $Result = @() + [string[]] $IndexTitles = @("Name", "Argument") + + if($PSBoundParameters.ContainsKey('Filter')){ + ## Search for the Name + $WinGetArgs += "--Filter", $Filter.Replace("…", "") + } + if($PSBoundParameters.ContainsKey('Name')){ + ## Search for the Name + $WinGetArgs += "--Name", $Name.Replace("…", "") + } + if($VerboseLog){ + ## Search for the Name + $WinGetArgs += "--Verbose-Logs" + } + } + PROCESS + { + $List = Invoke-WinGetCommand -WinGetArgs $WinGetArgs -IndexTitles $IndexTitles -JSON + + foreach ($Obj in $List) { + #$Result += [WinGetSource]::New($Obj.Name, $Obj.Argument) + $Result += @{ + Name = $Obj.Name + Argument = $Obj.Arg + Data = $Obj.Data + Identifier = $Obj.Identifier + Type = $Obj.Type + } + } + } + END + { + return $Result + } +} + +Export-ModuleMember -Function Get-WinGetSource \ No newline at end of file diff --git a/tools/PowerShell/Microsoft.WinGet.Client/src/Library/Get-WinGetVersion.ps1 b/tools/PowerShell/Microsoft.WinGet.Client/src/Library/Get-WinGetVersion.ps1 new file mode 100644 index 0000000000..375e414102 --- /dev/null +++ b/tools/PowerShell/Microsoft.WinGet.Client/src/Library/Get-WinGetVersion.ps1 @@ -0,0 +1,32 @@ +Function Get-WinGetVersion { + <# + .SYNOPSIS + Gets the version from the Windows Package Manager. + + .DESCRIPTION + By running this cmdlet, it will retrieve the version of the Windows Package Manager installed on the local system. + + .EXAMPLE + Get-WinGetVersion + #> + + BEGIN + { + [string[]] $WinGetArgs = "--version" + } + PROCESS + { + try { + $Result = [version](& "winget" $WinGetArgs).trimstart("v") + } + catch { + $Result = [version]"0.0.0.0" + } + } + END + { + return $Result + } +} + +Export-ModuleMember -Function Get-WinGetVersion \ No newline at end of file diff --git a/tools/PowerShell/Microsoft.WinGet.Client/src/Library/Install-WinGetPackage.ps1 b/tools/PowerShell/Microsoft.WinGet.Client/src/Library/Install-WinGetPackage.ps1 new file mode 100644 index 0000000000..85dbe9bef7 --- /dev/null +++ b/tools/PowerShell/Microsoft.WinGet.Client/src/Library/Install-WinGetPackage.ps1 @@ -0,0 +1,180 @@ +Function Install-WinGetPackage +{ + <# + .SYNOPSIS + Installs a package on the local system. + Additional options can be provided to filter the output, much like the search command. + + .DESCRIPTION + By running this cmdlet with the required inputs, it will retrieve the packages installed on the local system. + + .PARAMETER Filter + Used to search across multiple fields of the package. + + .PARAMETER Id + Used to specify the Id of the package + + .PARAMETER Name + Used to specify the Name of the package + + .PARAMETER Moniker + Used to specify the Moniker of the package + + .PARAMETER Tag + Used to specify the Tag of the package + + .PARAMETER Command + Used to specify the Command of the package + + .PARAMETER Scope + Used to specify install scope (user or machine) + + .PARAMETER Exact + Used to specify an exact match for any parameters provided. Many of the other parameters may be used for case insensitive substring matches if Exact is not specified. + + .PARAMETER Source + Name of the Windows Package Manager private source. Can be identified by running: "Get-WinGetSource" and using the source Name + + .PARAMETER Interactive + Used to specify the installer should be run in interactive mode. + + .PARAMETER Silent + Used to specify the installer should be run in silent mode with no user input. + + .PARAMETER Locale + Used to specify the locale for localized package installer. + + .PARAMETER Log + Used to specify the location for the log location if it is supported by the package installer. + + .PARAMETER Header + Used to specify the value to pass as the "Windows-Package-Manager" HTTP header for a REST source. + + .PARAMETER Version + Used to specify the Version of the package + + .PARAMETER VerboseLog + Used to provide verbose logging for the Windows Package Manager. + + .PARAMETER AcceptPackageAgreement + Used to accept any package agreement required for the package. + + .PARAMETER AcceptSourceAgreement + Used to explicitly accept any agreement required by the source. + + .PARAMETER Local + Used to install from a local manifest + + .EXAMPLE + Install-WinGetPackage -id "Publisher.Package" + + This example expects only a single package containing "Publisher.Package" as a valid identifier. + + .EXAMPLE + Install-WinGetPackage -id "Publisher.Package" -source "Private" + + This example expects the source named "Private" contains a package with "Publisher.Package" as a valid identifier. + + .EXAMPLE + Install-WinGetPackage -Name "Package" + + This example expects a configured source contains a package with "Package" as a valid name. + #> + + PARAM( + [Parameter(Position=0)] $Filter, + [Parameter()] $Name, + [Parameter()] $Id, + [Parameter()] $Moniker, + [Parameter()] $Source, + [Parameter()] [ValidateSet("User", "Machine")] $Scope, + [Parameter()] [switch] $Interactive, + [Parameter()] [switch] $Silent, + [Parameter()] [string] $Version, + [Parameter()] [switch] $Exact, + [Parameter()] [switch] $Override, + [Parameter()] [System.IO.FileInfo] $Location, + [Parameter()] [switch] $Force, + [Parameter()] [ValidatePattern("^([a-zA-Z]{2,3}|[iI]-[a-zA-Z]+|[xX]-[a-zA-Z]{1,8})(-[a-zA-Z]{1,8})*$")] [string] $Locale, + [Parameter()] [System.IO.FileInfo] $Log, ## This is a path of where to create a log. + [Parameter()] [switch] $AcceptSourceAgreements, + [Parameter()] [switch] $Local # This is for installing local manifests + ) + BEGIN + { + [string[]] $WinGetArgs = "Install" + IF($PSBoundParameters.ContainsKey('Filter')){ + IF($Local) { + $WinGetArgs += "--Manifest" + } + $WinGetArgs += $Filter + } + IF($PSBoundParameters.ContainsKey('Name')){ + $WinGetArgs += "--Name", $Name + } + IF($PSBoundParameters.ContainsKey('Id')){ + $WinGetArgs += "--Id", $Id + } + IF($PSBoundParameters.ContainsKey('Moniker')){ + $WinGetArgs += "--Moniker", $Moniker + } + IF($PSBoundParameters.ContainsKey('Source')){ + $WinGetArgs += "--Source", $Source + } + IF($PSBoundParameters.ContainsKey('Scope')){ + $WinGetArgs += "--Scope", $Scope + } + IF($Interactive){ + $WinGetArgs += "--Interactive" + } + IF($Silent){ + $WinGetArgs += "--Silent" + } + IF($PSBoundParameters.ContainsKey('Locale')){ + $WinGetArgs += "--locale", $Locale + } + if($PSBoundParameters.ContainsKey('Version')){ + $WinGetArgs += "--Version", $Version + } + if($Exact){ + $WinGetArgs += "--Exact" + } + if($PSBoundParameters.ContainsKey('Log')){ + $WinGetArgs += "--Log", $Log + } + if($PSBoundParameters.ContainsKey('Override')){ + $WinGetArgs += "--override", $Override + } + if($PSBoundParameters.ContainsKey('Location')){ + $WinGetArgs += "--Location", $Location + } + if($Force){ + $WinGetArgs += "--Force" + } + } + PROCESS + { + ## Exact, ID and Source - Talk with Demitrius tomorrow to better understand this. + IF(!$Local) { + $Result = Find-WinGetPackage -Filter $Filter -Name $Name -Id $Id -Moniker $Moniker -Tag $Tag -Command $Command -Source $Source + } + + if($Result.count -eq 1 -or $Local) { + & "WinGet" $WingetArgs + $Result = "" + } + elseif($Result.count -lt 1){ + Write-Host "Unable to locate package for installation" + $Result = "" + } + else { + Write-Host "Multiple packages found matching input criteria. Please refine the input." + } + } + END + { + return $Result + } +} + +Export-ModuleMember -Function Install-WinGetPackage \ No newline at end of file diff --git a/tools/PowerShell/Microsoft.WinGet.Client/src/Library/Invoke-WinGetCommand.ps1 b/tools/PowerShell/Microsoft.WinGet.Client/src/Library/Invoke-WinGetCommand.ps1 new file mode 100644 index 0000000000..43f45fecc1 --- /dev/null +++ b/tools/PowerShell/Microsoft.WinGet.Client/src/Library/Invoke-WinGetCommand.ps1 @@ -0,0 +1,234 @@ +filter Assert-WhiteSpaceIsNull { + IF ([string]::IsNullOrWhiteSpace($_)){$null} + ELSE {$_} +} + +class WinGetSource +{ + [string] $Name + [string] $Argument + [string] $Data + [string] $Identifier + [string] $Type + + WinGetSource () + { } + + WinGetSource ([string]$a, [string]$b, [string]$c, [string]$d, [string]$e) + { + $this.Name = $a.TrimEnd() | Assert-WhiteSpaceIsNull + $this.Argument = $b.TrimEnd() | Assert-WhiteSpaceIsNull + $this.Data = $c.TrimEnd() | Assert-WhiteSpaceIsNull + $this.Identifier = $d.TrimEnd() | Assert-WhiteSpaceIsNull + $this.Type = $e.TrimEnd() | Assert-WhiteSpaceIsNull + } + + WinGetSource ([string[]]$a) + { + $this.name = $a[0].TrimEnd() | Assert-WhiteSpaceIsNull + $this.Argument = $a[1].TrimEnd() | Assert-WhiteSpaceIsNull + $this.Data = $a[2].TrimEnd() | Assert-WhiteSpaceIsNull + $this.Identifier = $a[3].TrimEnd() | Assert-WhiteSpaceIsNull + $this.Type = $a[4].TrimEnd() | Assert-WhiteSpaceIsNull + } + + WinGetSource ([WinGetSource]$a) + { + $this.Name = $a.Name.TrimEnd() | Assert-WhiteSpaceIsNull + $this.Argument = $a.Argument.TrimEnd() | Assert-WhiteSpaceIsNull + $this.Data = $a.Data.TrimEnd() | Assert-WhiteSpaceIsNull + $this.Identifier = $a.Identifier.TrimEnd() | Assert-WhiteSpaceIsNull + $this.Type = $a.Type.TrimEnd() | Assert-WhiteSpaceIsNull + + } + + [WinGetSource[]] Add ([WinGetSource]$a) + { + $FirstValue = [WinGetSource]::New($this) + $SecondValue = [WinGetSource]::New($a) + + [WinGetSource[]] $Combined = @([WinGetSource]::New($FirstValue), [WinGetSource]::New($SecondValue)) + + Return $Combined + } + + [WinGetSource[]] Add ([String[]]$a) + { + $FirstValue = [WinGetSource]::New($this) + $SecondValue = [WinGetSource]::New($a) + + [WinGetSource[]] $Combined = @([WinGetSource]::New($FirstValue), [WinGetSource]::New($SecondValue)) + + Return $Combined + } +} + +class WinGetPackage +{ + [string]$Name + [string]$Id + [string]$Version + [string]$Available + [string]$Source + [string]$Match + + WinGetPackage ([string] $a, [string]$b, [string]$c, [string]$d, [string]$e) + { + $this.Name = $a.TrimEnd() | Assert-WhiteSpaceIsNull + $this.Id = $b.TrimEnd() | Assert-WhiteSpaceIsNull + $this.Version = $c.TrimEnd() | Assert-WhiteSpaceIsNull + $this.Available = $d.TrimEnd() | Assert-WhiteSpaceIsNull + $this.Source = $e.TrimEnd() | Assert-WhiteSpaceIsNull + } + + WinGetPackage ([WinGetPackage] $a) { + $this.Name = $a.Name | Assert-WhiteSpaceIsNull + $this.Id = $a.Id | Assert-WhiteSpaceIsNull + $this.Version = $a.Version | Assert-WhiteSpaceIsNull + $this.Available = $a.Available | Assert-WhiteSpaceIsNull + $this.Source = $a.Source | Assert-WhiteSpaceIsNull + + } + WinGetPackage ([psobject] $a) { + $this.Name = $a.Name | Assert-WhiteSpaceIsNull + $this.Id = $a.Id | Assert-WhiteSpaceIsNull + $this.Version = $a.Version | Assert-WhiteSpaceIsNull + $this.Available = $a.Available | Assert-WhiteSpaceIsNull + $this.Source = $a.Source | Assert-WhiteSpaceIsNull + } + + WinGetSource ([string[]]$a) + { + $this.name = $a[0].TrimEnd() | Assert-WhiteSpaceIsNull + $this.Id = $a[1].TrimEnd() | Assert-WhiteSpaceIsNull + $this.Version = $a[2].TrimEnd() | Assert-WhiteSpaceIsNull + $this.Available = $a[3].TrimEnd() | Assert-WhiteSpaceIsNull + $this.Source = $a[4].TrimEnd() | Assert-WhiteSpaceIsNull + } + + + [WinGetPackage[]] Add ([WinGetPackage] $a) + { + $FirstValue = [WinGetPackage]::New($this) + $SecondValue = [WinGetPackage]::New($a) + + [WinGetPackage[]]$Result = @([WinGetPackage]::New($FirstValue), [WinGetPackage]::New($SecondValue)) + + Return $Result + } + + [WinGetPackage[]] Add ([String[]]$a) + { + $FirstValue = [WinGetPackage]::New($this) + $SecondValue = [WinGetPackage]::New($a) + + [WinGetPackage[]] $Combined = @([WinGetPackage]::New($FirstValue), [WinGetPackage]::New($SecondValue)) + + Return $Combined + } +} +Function Invoke-WinGetCommand +{ + PARAM( + [Parameter(Position=0, Mandatory=$true)] [string[]]$WinGetArgs, + [Parameter(Position=0, Mandatory=$true)] [string[]]$IndexTitles, + [Parameter()] [switch] $JSON + ) + BEGIN + { + $Index = @() + $Result = @() + $i = 0 + $IndexTitlesCount = $IndexTitles.Count + $Offset = 0 + $Found = $false + + ## Remove two characters from the string length and add "..." to the end (only if there is the three below characters present). + [string[]]$WinGetSourceListRaw = & "WinGet" $WingetArgs | out-string -stream | foreach-object{$_ -replace ("$([char]915)$([char]199)$([char]170)", "$([char]199)")} + } + PROCESS + { + if($JSON){ + ## If expecting JSON content, return the object + return $WinGetSourceListRaw | ConvertFrom-Json + } + + ## Gets the indexing of each title + $rgex = $IndexTitles -join "|" + for ($Offset=0; $Offset -lt $WinGetSourceListRaw.Length; $Offset++) { + if($WinGetSourceListRaw[$Offset].Split(" ")[0].Trim() -match $rgex) { + $Found = $true + break + } + } + if(!$Found) { + Write-Error -Message "No results were found." -TargetObject $WinGetSourceListRaw + return + } + + foreach ($IndexTitle in $IndexTitles) { + ## Creates an array of titles and their string location + $IndexStart = $WinGetSourceListRaw[$Offset].IndexOf($IndexTitle) + $IndexEnds = "" + + IF($IndexStart -ne "-1") { + $Index += [pscustomobject]@{ + Title = $IndexTitle + Start = $IndexStart + Ends = $IndexEnds + } + } + } + + ## Orders the Object based on Index value + $Index = $Index | Sort-Object Start + + ## Sets the end of string value + while ($i -lt $IndexTitlesCount) { + $i ++ + + ## Sets the End of string value (if not null) + if($Index[$i].Start) { + $Index[$i-1].Ends = ($Index[$i].Start -1) - $Index[$i-1].Start + } + } + + ## Builds the WinGetSource Object with contents + $i = $Offset + 2 + while($i -lt $WinGetSourceListRaw.Length) { + $row = $WinGetSourceListRaw[$i] + try { + [bool] $TestNotTitles = $WinGetSourceListRaw[0] -ne $row + [bool] $TestNotHyphenLine = $WinGetSourceListRaw[1] -ne $row -and !$Row.Contains("---") + [bool] $TestNotNoResults = $row -ne "No package found matching input criteria." + } + catch {Wait-Debugger} + + if(!$TestNotNoResults) { + Write-LogEntry -LogEntry "No package found matching input criteria." -Severity 1 + } + + ## If this is the first pass containing titles or the table line, skip. + if($TestNotTitles -and $TestNotHyphenLine -and $TestNotNoResults) { + $List = @{} + + foreach($item in $Index) { + if($Item.Ends) { + $List[$Item.Title] = $row.SubString($item.Start,$Item.Ends) + } + else { + $List[$item.Title] = $row.SubString($item.Start, $row.Length - $Item.Start) + } + } + + $result += [pscustomobject]$list + } + $i++ + } + } + END + { + return $Result + } +} + diff --git a/tools/PowerShell/Microsoft.WinGet.Client/src/Library/Remove-WinGetSource.ps1 b/tools/PowerShell/Microsoft.WinGet.Client/src/Library/Remove-WinGetSource.ps1 new file mode 100644 index 0000000000..24a5ed375f --- /dev/null +++ b/tools/PowerShell/Microsoft.WinGet.Client/src/Library/Remove-WinGetSource.ps1 @@ -0,0 +1,37 @@ +Function Remove-WinGetSource +{ + <# + .SYNOPSIS + Removes a source from the Windows Package Manager. + + .DESCRIPTION + By running this cmdlet will remove an existing source from the Windows Package Manager. + + .PARAMETER Name + Used to specify the Name of the source + + .EXAMPLE + Remove-WinGetSource -Name "Private" + + This will remove the source named "Private" from the Windows Package Manager + #> + + PARAM( + [Parameter(Mandatory=$true)] [string] $Name + ) + BEGIN + { + [string[]] $WinGetArgs = "Source", "remove" + $WinGetArgs += "--Name", $Name + } + PROCESS + { + WinGet $WingetArgs + } + END + { + return + } +} + +Export-ModuleMember -Function Remove-WinGetSource \ No newline at end of file diff --git a/tools/PowerShell/Microsoft.WinGet.Client/src/Library/Reset-WinGetSource.ps1 b/tools/PowerShell/Microsoft.WinGet.Client/src/Library/Reset-WinGetSource.ps1 new file mode 100644 index 0000000000..0864d2b54b --- /dev/null +++ b/tools/PowerShell/Microsoft.WinGet.Client/src/Library/Reset-WinGetSource.ps1 @@ -0,0 +1,42 @@ +Function Reset-WinGetSource +{ + <# + .SYNOPSIS + Resets the sources for the Windows Package Manager. + + .DESCRIPTION + By running this cmdlet will reset sources for the Windows Package Manager. + + .PARAMETER Name + Used to specify the Name of the source + + .EXAMPLE + Reset-WinGetSource + + This will reset the default sources for the Windows Package Manager + + .EXAMPLE + Reset-WinGetSource -Name "Private" + + This will reset the source named "Private" for the Windows Package Manager + #> + + PARAM( + [Parameter()] [string] $Name + ) + BEGIN + { + [string[]] $WinGetArgs = "Source", "reset" + $WinGetArgs += "--Name", $Name + } + PROCESS + { + WinGet $WingetArgs + } + END + { + return + } +} + +Export-ModuleMember -Function Reset-WinGetSource \ No newline at end of file diff --git a/tools/PowerShell/Microsoft.WinGet.Client/src/Library/Uninstall-WinGetPackage.ps1 b/tools/PowerShell/Microsoft.WinGet.Client/src/Library/Uninstall-WinGetPackage.ps1 new file mode 100644 index 0000000000..0e1dcac608 --- /dev/null +++ b/tools/PowerShell/Microsoft.WinGet.Client/src/Library/Uninstall-WinGetPackage.ps1 @@ -0,0 +1,156 @@ +Function Uninstall-WinGetPackage{ + <# + .SYNOPSIS + Uninstalls a package from the local system. + Additional options can be provided to filter the output, much like the search command. + + .DESCRIPTION + By running this cmdlet with the required inputs, it will uninstall a package installed on the local system. + + .PARAMETER Filter + Used to search across multiple fields of the package. + + .PARAMETER Id + Used to specify the Id of the package + + .PARAMETER Name + Used to specify the Name of the package + + .PARAMETER Moniker + Used to specify the Moniker of the package + + .PARAMETER Version + Used to specify the Version of the package + + .PARAMETER Exact + Used to specify an exact match for any parameters provided. Many of the other parameters may be used for case insensitive substring matches if Exact is not specified. + + .PARAMETER Source + Name of the Windows Package Manager private source. Can be identified by running: "Get-WinGetSource" and using the source Name + + .PARAMETER Interactive + Used to specify the uninstaller should be run in interactive mode. + + .PARAMETER Silent + Used to specify the uninstaller should be run in silent mode with no user input. + + .PARAMETER Log + Used to specify the location for the log location if it is supported by the package uninstaller. + + .PARAMETER VerboseLog + Used to provide verbose logging for the Windows Package Manager. + + .PARAMETER Header + Used to specify the value to pass as the "Windows-Package-Manager" HTTP header for a REST source. + + .PARAMETER AcceptSourceAgreement + Used to explicitly accept any agreement required by the source. + + .PARAMETER Local + Used to uninstall from a local manifest + + .EXAMPLE + Uninstall-WinGetPackage -id "Publisher.Package" + + This example expects only a single configured REST source with a package containing "Publisher.Package" as a valid identifier. + + .EXAMPLE + Uninstall-WinGetPackage -id "Publisher.Package" -source "Private" + + This example expects the REST source named "Private" with a package containing "Publisher.Package" as a valid identifier. + + .EXAMPLE + Uninstall-WinGetPackage -Name "Package" + + This example expects a configured source contains a package with "Package" as a valid name. + #> + + PARAM( + [Parameter(Position=0)] $Filter, + [Parameter()] $Name, + [Parameter()] $Id, + [Parameter()] $Moniker, + [Parameter()] $Source, + [Parameter()] [switch] $Interactive, + [Parameter()] [switch] $Silent, + [Parameter()] [string] $Version, + [Parameter()] [switch] $Exact, + [Parameter()] [switch] $Override, + [Parameter()] [System.IO.FileInfo] $Location, + [Parameter()] [switch] $Force, + [Parameter()] [System.IO.FileInfo] $Log, ## This is a path of where to create a log. + [Parameter()] [switch] $AcceptSourceAgreements, + [Parameter()] [switch] $Local # This is for installing local manifests + ) + BEGIN + { + [string[]] $WinGetArgs = "Uninstall" + IF($PSBoundParameters.ContainsKey('Filter')){ + IF($Local) { + $WinGetArgs += "--Manifest" + } + $WinGetArgs += $Filter + } + IF($PSBoundParameters.ContainsKey('Name')){ + $WinGetArgs += "--Name", $Name + } + IF($PSBoundParameters.ContainsKey('Id')){ + $WinGetArgs += "--Id", $Id + } + IF($PSBoundParameters.ContainsKey('Moniker')){ + $WinGetArgs += "--Moniker", $Moniker + } + IF($PSBoundParameters.ContainsKey('Source')){ + $WinGetArgs += "--Source", $Source + } + IF($Interactive){ + $WinGetArgs += "--Interactive" + } + IF($Silent){ + $WinGetArgs += "--Silent" + } + if($PSBoundParameters.ContainsKey('Version')){ + $WinGetArgs += "--Version", $Version + } + if($Exact){ + $WinGetArgs += "--Exact" + } + if($PSBoundParameters.ContainsKey('Log')){ + $WinGetArgs += "--Log", $Log + } + if($PSBoundParameters.ContainsKey('Location')){ + $WinGetArgs += "--Location", $Location + } + if($Force){ + $WinGetArgs += "--Force" + } + } + PROCESS + { + ## Exact, ID and Source - Talk with tomorrow to better understand this. + IF(!$Local) { + $Result = Find-WinGetPackage -Filter $Filter -Name $Name -Id $Id -Moniker $Moniker -Tag $Tag -Command $Command -Source $Source + } + + if($Result.count -eq 1 -or $Local) { + & "WinGet" $WingetArgs + $Result = "" + } + elseif($Result.count -lt 1){ + Write-Host "Unable to locate package for uninstallation" + $Result = "" + } + else { + Write-Host "Multiple packages found matching input criteria. Please refine the input." + } + } + END + { + return $Result + } +} + +New-Alias -Name Remove-WinGetPackage -Value Uninstall-WinGetPackage + +Export-ModuleMember -Function Uninstall-WinGetPackage +Export-ModuleMember -Alias Remove-WinGetPackage diff --git a/tools/PowerShell/Microsoft.WinGet.Client/src/Library/Upgrade-WinGetPackage.ps1 b/tools/PowerShell/Microsoft.WinGet.Client/src/Library/Upgrade-WinGetPackage.ps1 new file mode 100644 index 0000000000..3db80f4b91 --- /dev/null +++ b/tools/PowerShell/Microsoft.WinGet.Client/src/Library/Upgrade-WinGetPackage.ps1 @@ -0,0 +1,185 @@ +Function Upgrade-WinGetPackage +{ + <# + .SYNOPSIS + Upgrades a package on the local system. + Additional options can be provided to filter the output, much like the search command. + + .DESCRIPTION + By running this cmdlet with the required inputs, it will retrieve the packages installed on the local system. + + .PARAMETER Filter + Used to search across multiple fields of the package. + + .PARAMETER Id + Used to specify the Id of the package + + .PARAMETER Name + Used to specify the Name of the package + + .PARAMETER Moniker + Used to specify the Moniker of the package + + .PARAMETER Tag + Used to specify the Tag of the package + + .PARAMETER Command + Used to specify the Command of the package + + .PARAMETER Channel + Used to specify the channel of the package. Note this is not yet implemented in Windows Package Manager as of version 1.1.0. + + .PARAMETER Scope + Used to specify install scope (user or machine) + + .PARAMETER Exact + Used to specify an exact match for any parameters provided. Many of the other parameters may be used for case insensitive substring matches if Exact is not specified. + + .PARAMETER Source + Name of the Windows Package Manager private source. Can be identified by running: "Get-WinGetSource" and using the source Name + + .PARAMETER Manifest + Path to the manifest on the local file system. Requires local manifest setting to be enabled. + + .PARAMETER Interactive + Used to specify the installer should be run in interactive mode. + + .PARAMETER Silent + Used to specify the installer should be run in silent mode with no user input. + + .PARAMETER Locale + Used to specify the locale for localized package installer. + + .PARAMETER Log + Used to specify the location for the log location if it is supported by the package installer. + + .PARAMETER Override + Used to override switches passed to installer. + + .PARAMETER Force + Used to force the upgrade when the Windows Package Manager would ordinarily not upgrade the package. + + .PARAMETER Location + Used to specify the location for the package to be upgraded. + + .PARAMETER Header + Used to specify the value to pass as the "Windows-Package-Manager" HTTP header for a REST source. + + .PARAMETER Version + Used to specify the Version of the package + + .PARAMETER VerboseLog + Used to provide verbose logging for the Windows Package Manager. + + .PARAMETER AcceptPackageAgreement + Used to accept any source package required for the package. + + .PARAMETER AcceptSourceAgreement + + .EXAMPLE + Upgrade-WinGetPackage -id "Publisher.Package" + + This example expects only a single package containing "Publisher.Package" as a valid identifier. + + .EXAMPLE + Upgrade-WinGetPackage -id "Publisher.Package" -source "Private" + + This example expects the source named "Private" contains a package with "Publisher.Package" as a valid identifier. + + .EXAMPLE + Upgrade-WinGetPackage -Name "Package" + + This example expects the source named "Private" contains a package with "Package" as a valid name. + #> + + PARAM( + [Parameter(Position=0)] $Filter, + [Parameter()] $Name, + [Parameter()] $Id, + [Parameter()] $Moniker, + [Parameter()] $Source, + [Parameter()] [ValidateSet("User", "Machine")] $Scope, + [Parameter()] [switch] $Interactive, + [Parameter()] [switch] $Silent, + [Parameter()] [string] $Version, + [Parameter()] [switch] $Exact, + [Parameter()] [switch] $Override, + [Parameter()] [System.IO.FileInfo] $Location, + [Parameter()] [switch] $Force, + [Parameter()] [ValidatePattern("^([a-zA-Z]{2,3}|[iI]-[a-zA-Z]+|[xX]-[a-zA-Z]{1,8})(-[a-zA-Z]{1,8})*$")] [string] $Locale, + [Parameter()] [System.IO.FileInfo] $Log, ## This is a path of where to create a log. + [Parameter()] [switch] $AcceptSourceAgreements + ) + BEGIN + { + [string[]] $WinGetArgs = "Install" + IF($PSBoundParameters.ContainsKey('Filter')){ + $WinGetArgs += $Filter + } + IF($PSBoundParameters.ContainsKey('Name')){ + $WinGetArgs += "--Name", $Name + } + IF($PSBoundParameters.ContainsKey('Id')){ + $WinGetArgs += "--Id", $Id + } + IF($PSBoundParameters.ContainsKey('Moniker')){ + $WinGetArgs += "--Moniker", $Moniker + } + IF($PSBoundParameters.ContainsKey('Source')){ + $WinGetArgs += "--Source", $Source + } + IF($PSBoundParameters.ContainsKey('Scope')){ + $WinGetArgs += "--Scope", $Scope + } + IF($Interactive){ + $WinGetArgs += "--Interactive" + } + IF($Silent){ + $WinGetArgs += "--Silent" + } + IF($PSBoundParameters.ContainsKey('Locale')){ + $WinGetArgs += "--locale", $Locale + } + if($PSBoundParameters.ContainsKey('Version')){ + $WinGetArgs += "--Version", $Version + } + if($Exact){ + $WinGetArgs += "--Exact" + } + if($PSBoundParameters.ContainsKey('Log')){ + $WinGetArgs += "--Log", $Log + } + if($PSBoundParameters.ContainsKey('Override')){ + $WinGetArgs += "--override", $Override + } + if($PSBoundParameters.ContainsKey('Location')){ + $WinGetArgs += "--Location", $Location + } + if($Force){ + $WinGetArgs += "--Force" + } + } + PROCESS + { + ## Exact, ID and Source - Talk with Demitrius tomorrow to better understand this. + $Result = Find-WinGetPackage -Filter $Filter -Name $Name -Id $Id -Moniker $Moniker -Tag $Tag -Command $Command -Source $Source + + if($Result.count -eq 1) { + & "WinGet" $WingetArgs + $Result = "" + } + elseif($Result.count -lt 1){ + Write-Host "Unable to locate package for installation" + $Result = "" + } + else { + Write-Host "Multiple packages found matching input criteria. Please refine the input." + } + } + END + { + return $Result + } +} + +Export-ModuleMember -Function Install-WinGetPackage \ No newline at end of file diff --git a/tools/PowerShell/Microsoft.WinGet.Client/src/Library/Write-LogEntry.ps1 b/tools/PowerShell/Microsoft.WinGet.Client/src/Library/Write-LogEntry.ps1 new file mode 100644 index 0000000000..7759e1293a --- /dev/null +++ b/tools/PowerShell/Microsoft.WinGet.Client/src/Library/Write-LogEntry.ps1 @@ -0,0 +1,51 @@ +Function Write-LogEntry +{ + PARAM( + [Parameter(Position=0, Mandatory=$true)] [string] $LogEntry, + [Parameter(Position=1, Mandatory=$false)] [int] $Severity=1, + [Parameter(Position=2, Mandatory=$false)] [string] $FontColor="", + [Parameter(Position=3, Mandatory=$false)] [int] $Indent = 0, + [Parameter(Position=4, Mandatory=$false)] [switch] $NoNewLine + ) + BEGIN + { + if($FontColor -eq "") { + switch ($Severity) { + "1" { + ## Informational Response + $FontColor = "White" + $MessagePreFix = "" + } + "2" { + ## Warning Response + $FontColor = "Yellow" + $MessagePreFix = "WARNING: " + } + "3" { + ## Error Response + $FontColor = "Red" + $MessagePreFix = "ERROR: " + } + } + } + ## Combines the logging message and the message type as a prefix + $LogEntry = $MessagePreFix + $LogEntry + + ## Indents the message when viewed on the screen. + $LogEntry = $LogEntry.PadLeft($LogEntry.Length + (2 * $Indent) ) + } + PROCESS + { + ## Writes logging to the screen + if($NoNewLine) { + Write-Host -Object $LogEntry -ForegroundColor $FontColor -NoNewline + } + else { + Write-Host -Object $LogEntry -ForegroundColor $FontColor + } + } + END + { + return + } +} \ No newline at end of file diff --git a/tools/PowerShell/Microsoft.WinGet.Client/src/Microsoft.WinGet.Client.psd1 b/tools/PowerShell/Microsoft.WinGet.Client/src/Microsoft.WinGet.Client.psd1 new file mode 100644 index 0000000000..0dd90dc0f3 --- /dev/null +++ b/tools/PowerShell/Microsoft.WinGet.Client/src/Microsoft.WinGet.Client.psd1 @@ -0,0 +1,123 @@ +# +# Module manifest for module 'WindowsPackageManager' +# +# Generated on: 10/12/2021 +# + +@{ + +# Script module or binary module file associated with this manifest. +RootModule = 'Microsoft.WinGet.Client.psm1' + +# Version number of this module. +ModuleVersion = '0.1.0' + +# Supported PSEditions +# CompatiblePSEditions = @() + +# ID used to uniquely identify this module +GUID = 'b22b7711-6329-420a-87fd-bc47ad9951c7' + +# Author of this module +# Author = '' + +# Company or vendor of this module +CompanyName = 'Microsoft' + +# Copyright statement for this module +Copyright = '(c) 2021 Microsoft. All rights reserved.' + +# Description of the functionality provided by this module +Description = 'This module provides support for working with the Windows Package Manager. It was built as a Hackathon project for the Windows Package Manager.' + +# Minimum version of the Windows PowerShell engine required by this module +PowerShellVersion = '5.1' + +# Name of the Windows PowerShell host required by this module +# PowerShellHostName = '' + +# Minimum version of the Windows PowerShell host required by this module +# PowerShellHostVersion = '' + +# Minimum version of Microsoft .NET Framework required by this module. This prerequisite is valid for the PowerShell Desktop edition only. +# DotNetFrameworkVersion = '' + +# Minimum version of the common language runtime (CLR) required by this module. This prerequisite is valid for the PowerShell Desktop edition only. +# CLRVersion = '' + +# Processor architecture (None, X86, Amd64) required by this module +# ProcessorArchitecture = '' + +# Modules that must be imported into the global environment prior to importing this module +# RequiredModules = @() + +# Assemblies that must be loaded prior to importing this module +# RequiredAssemblies = @() + +# Script files (.ps1) that are run in the caller's environment prior to importing this module. +# ScriptsToProcess = @() + +# Type files (.ps1xml) to be loaded when importing this module +# TypesToProcess = @() + +# Format files (.ps1xml) to be loaded when importing this module +# FormatsToProcess = @() + +# Modules to import as nested modules of the module specified in RootModule/ModuleToProcess +# NestedModules = @() + +# Functions to export from this module, for best performance, do not use wild cards and do not delete the entry, use an empty array if there are no functions to export. +FunctionsToExport = @("Get-WinGetPackage", "Get-WinGetSource", "Get-WinGetVersion", "Add-WinGetSource") + +# Cmdlets to export from this module, for best performance, do not use wild cards and do not delete the entry, use an empty array if there are no cmdlets to export. +CmdletsToExport = @() + +# Variables to export from this module +VariablesToExport = '*' + +# Aliases to export from this module, for best performance, do not use wild cards and do not delete the entry, use an empty array if there are no aliases to export. +AliasesToExport = @() + +# DSC resources to export from this module +# DscResourcesToExport = @() + +# List of all modules packaged with this module +# ModuleList = @() + +# List of all files packaged with this module +# FileList = @() + +# Private data to pass to the module specified in RootModule/ModuleToProcess. This may also contain a PSData hashtable with additional module metadata used by PowerShell. +PrivateData = @{ + + PSData = @{ + + # Tags applied to this module. These help with module discovery in online galleries. + Tags = @("winget") + + # A URL to the license for this module. + # LicenseUri = '' + + # A URL to the main website for this project. + ProjectUri = 'https://github.com/microsoft/winget-cli' + + # A URL to an icon representing this module. + # IconUri = '' + + # ReleaseNotes of this module + # ReleaseNotes = '' + + PreRelease = 'alpha' + + } # End of PSData hashtable + +} # End of PrivateData hashtable + +# HelpInfo URI of this module +# HelpInfoURI = '' + +# Default prefix for commands exported from this module. Override the default prefix using Import-Module -Prefix. +# DefaultCommandPrefix = '' + +} + diff --git a/tools/PowerShell/Microsoft.WinGet.Client/src/Microsoft.WinGet.Client.psm1 b/tools/PowerShell/Microsoft.WinGet.Client/src/Microsoft.WinGet.Client.psm1 new file mode 100644 index 0000000000..1006824caa --- /dev/null +++ b/tools/PowerShell/Microsoft.WinGet.Client/src/Microsoft.WinGet.Client.psm1 @@ -0,0 +1,16 @@ +. $PSScriptRoot\Library\Get-WinGetVersion.ps1 + +$RequiredVersion = "1.1.12653" + +$version = Get-WinGetVersion +if ($version -lt $RequiredVersion) { + # Need to do localization + Write-Host "Windows Package Manager is missing. For more information on installing the Windows Package Manager. `n Visit: https://github.com/microsoft/winget-cli#installing-the-client" + throw [WinGetVersionMismatch]::new("Requires Windows Package Manager $RequiredVersion or later to be installed.") +} + +Get-ChildItem -Path $PSScriptRoot\Library -Filter *.ps1 | foreach-object { . $_.FullName } + +class WinGetVersionMismatch : Exception { + WinGetVersionMismatch([string] $message) : base($message) {} +} diff --git a/tools/PowerShell/Microsoft.WinGet.Client/test/README.md b/tools/PowerShell/Microsoft.WinGet.Client/test/README.md new file mode 100644 index 0000000000..2323f662dd --- /dev/null +++ b/tools/PowerShell/Microsoft.WinGet.Client/test/README.md @@ -0,0 +1 @@ +Placeholder for test collateral \ No newline at end of file diff --git a/tools/PowerShell/Microsoft.WinGet/doc/README.md b/tools/PowerShell/Microsoft.WinGet/doc/README.md new file mode 100644 index 0000000000..d0b1722291 --- /dev/null +++ b/tools/PowerShell/Microsoft.WinGet/doc/README.md @@ -0,0 +1 @@ +Placeholder for PlatyPS generated help files diff --git a/tools/PowerShell/Microsoft.WinGet/src/Microsoft.WinGet.psd1 b/tools/PowerShell/Microsoft.WinGet/src/Microsoft.WinGet.psd1 new file mode 100644 index 0000000000..66ae9b67a3 --- /dev/null +++ b/tools/PowerShell/Microsoft.WinGet/src/Microsoft.WinGet.psd1 @@ -0,0 +1,133 @@ +# +# Module manifest for module 'Microsoft.WinGet' +# +# Generated by: denelon +# +# Generated on: 10/15/2021 +# + +@{ + + # Script module or binary module file associated with this manifest. + RootModule = '.\Microsoft.WinGet.psm1' + + # Version number of this module. + ModuleVersion = '0.1.0' + + # Supported PSEditions + # CompatiblePSEditions = @() + + # ID used to uniquely identify this module + GUID = 'd9b4b2a9-11c0-4b64-bde2-e1938f81e82a' + + # Author of this module + # Author = '' + + # Company or vendor of this module + CompanyName = 'Microsoft' + + # Copyright statement for this module + Copyright = '(c) Microsoft. All rights reserved.' + + # Description of the functionality provided by this module + Description = 'This module provides support for working with the Windows Package Manager. It was built as a Hackathon project for the Windows Package Manager.' + + # Minimum version of the PowerShell engine required by this module + PowerShellVersion = '5.1' + + # Name of the PowerShell host required by this module + # PowerShellHostName = '' + + # Minimum version of the PowerShell host required by this module + # PowerShellHostVersion = '' + + # Minimum version of Microsoft .NET Framework required by this module. This prerequisite is valid for the PowerShell Desktop edition only. + # DotNetFrameworkVersion = '' + + # Minimum version of the common language runtime (CLR) required by this module. This prerequisite is valid for the PowerShell Desktop edition only. + # ClrVersion = '' + + # Processor architecture (None, X86, Amd64) required by this module + # ProcessorArchitecture = '' + + # Modules that must be imported into the global environment prior to importing this module + # RequiredModules = @() + + # Assemblies that must be loaded prior to importing this module + # RequiredAssemblies = @() + + # Script files (.ps1) that are run in the caller's environment prior to importing this module. + # ScriptsToProcess = @() + + # Type files (.ps1xml) to be loaded when importing this module + # TypesToProcess = @() + + # Format files (.ps1xml) to be loaded when importing this module + # FormatsToProcess = @() + + # Modules to import as nested modules of the module specified in RootModule/ModuleToProcess + # NestedModules = @() + + # Functions to export from this module, for best performance, do not use wildcards and do not delete the entry, use an empty array if there are no functions to export. + FunctionsToExport = @() + + # Cmdlets to export from this module, for best performance, do not use wildcards and do not delete the entry, use an empty array if there are no cmdlets to export. + CmdletsToExport = @() + + # Variables to export from this module + VariablesToExport = '*' + + # Aliases to export from this module, for best performance, do not use wildcards and do not delete the entry, use an empty array if there are no aliases to export. + AliasesToExport = @() + + # DSC resources to export from this module + # DscResourcesToExport = @() + + # List of all modules packaged with this module + ModuleList = @("Microsoft.WinGet.Client", "Microsoft.WinGet.Create", "Microsoft.WinGet.Source") + + # List of all files packaged with this module + # FileList = @() + + # Private data to pass to the module specified in RootModule/ModuleToProcess. This may also contain a PSData hashtable with additional module metadata used by PowerShell. + PrivateData = @{ + + PSData = @{ + + # Tags applied to this module. These help with module discovery in online galleries. + Tags = @("winget", "wingetcreate") + + # A URL to the license for this module. + # LicenseUri = '' + + # A URL to the main website for this project. + ProjectUri = 'https://github.com/microsoft/winget-cli' + + # A URL to an icon representing this module. + # IconUri = '' + + # ReleaseNotes of this module + # ReleaseNotes = '' + + # Prerelease string of this module + Prerelease = 'alpha' + + # Flag to indicate whether the module requires explicit user acceptance for install/update/save + # RequireLicenseAcceptance = $false + + # External dependent modules of this module + # ExternalModuleDependencies = @() + + } # End of PSData hashtable + + } # End of PrivateData hashtable + + # HelpInfo URI of this module + # HelpInfoURI = '' + + # Default prefix for commands exported from this module. Override the default prefix using Import-Module -Prefix. + # DefaultCommandPrefix = '' + + } + + \ No newline at end of file diff --git a/tools/PowerShell/Microsoft.WinGet/src/Microsoft.WinGet.psm1 b/tools/PowerShell/Microsoft.WinGet/src/Microsoft.WinGet.psm1 new file mode 100644 index 0000000000..9ca6db43ab --- /dev/null +++ b/tools/PowerShell/Microsoft.WinGet/src/Microsoft.WinGet.psm1 @@ -0,0 +1,4 @@ + +Import-Module -Name Microsoft.WinGet.Client -Global -MaximumVersion 0.99.99 +Import-Module -Name Microsoft.WinGet.Create -Global -MaximumVersion 0.99.99 +Import-Module -Name Microsoft.WinGet.Source -Global -MaximumVersion 0.99.99 \ No newline at end of file diff --git a/tools/PowerShell/Microsoft.WinGet/test/README.md b/tools/PowerShell/Microsoft.WinGet/test/README.md new file mode 100644 index 0000000000..2323f662dd --- /dev/null +++ b/tools/PowerShell/Microsoft.WinGet/test/README.md @@ -0,0 +1 @@ +Placeholder for test collateral \ No newline at end of file diff --git a/tools/PowerShell/README.md b/tools/PowerShell/README.md new file mode 100644 index 0000000000..228bef6c0b --- /dev/null +++ b/tools/PowerShell/README.md @@ -0,0 +1,179 @@ +# Windows Package Manager PowerShell Modules + +## History +This set of PowerShell modules was originally built as a Microsoft Hackathon project in 2021. + +We started this project as an exploration to design the right set of cmdlets with PowerShell approved nouns and verbs. As we began to explore wrapping the Windows Package Manager we identified several areas of complexity. Rather than slow down progress on building PowerShell cmdlets, we decided to forge ahead and call out the areas where the experience with pipelines is sub-optimal. + +For example, the Windows Package Manager (later referenced as just CLI) CLI was designed for displaying output in the standard width Windows Terminal. As such, long package names and "Id"s are truncated with a single width ellipsis character. Users wanting to use the PowerShell pipeline to pass values to another cmdlet will likely encounter undesired behavior due to this truncation. We have decided to declare the "Microsoft.WinGet" module as an alpha release until these problems have been resolved. We are planning to change the status of "PreRelease" to beta once we believe we have addressed the issues related to piping the output to other cmdlets. + +The module and associated cmdlets were essentially handcrafted as the Hackathon team came up to speed with PowerShell idioms. The module and associated cmdlets in the Microsoft.WinGet.Client/crescendo directory were crafted using the "Microsoft.PowerShell.Crescendo" module. Several manual changes were applied to these files as well, but the idea was to leverage crescendo to speed up development. + +As we continued, we also identified work being done in support of private REST sources. We decided to create separate modules to speed up development in that are of the product as well as standardizing some of the module work. + +The "Microsoft.WinGet" module is a top level module to organize the others. They child modules are "Microsoft.WinGet.Client" intended to represent the [native PowerShell module](https://github.com/microsoft/winget-cli/issues/221) feature request. The "Microsoft.WinGet.Create" module in the [winget-cli-restsource repository](https://github.com/microsoft/winget-cli-restsource) is expected to support building and modifying manifests in private sources (related to the Windows Package Manager Manifest Creator [REST support feature](https://github.com/microsoft/winget-create/issues/3)). The third module "Microsoft.WinGet.Source" is intended to simplify working the private REST sources like the [reference implementation](https://github.com/microsoft/winget-cli-restsource). A fourth module may be created for the [winget create tool](https://github.com/microsoft/wingetcreate). + +## Terms and Conventions used + +### Package +The term "package" is used to reference an application or program. For example, Windows Terminal is a package. It's identifier "Microsoft.WindowsTerminal" is available in the [Windows Package Manager Community App Repository](https://github.com/microsoft/winget-pkgs). + + +### Manifest +The term "manifest" is used to reference the metadata about an application or program. In the [Windows Package Manager Community App Repository](https://github.com/microsoft/winget-pkgs) manifests are represented as YAML files. In a REST source, manifests are represented as JSON structured data. For the sake of transparency, the manifests downloaded by the Windows Package Manager from the default "**winget**" source, the manifests are taken from the GitHub repository and merged into a single YAML file per package version. + + +### Conventions +We attempted to make use of the approved verbs for PowerShell. In areas where the verb may have been contentious, we decided to at least be consistent with ourselves. The Open verb is used to open a directory in file explorer (Windows Package Manager Manifest Creator installer cache). The Edit verb is used to open the settings.json file for the Windows Package Manager Manifest Creator and the Windows Package Manager. + +### Status +The modules should be treated as experimental at this stage of development. They are essentially calling the Windows Package Manager executable and attempting to parse text output that wasn't designed for PowerShell. + +### Future +We expect to enhance the COM interface to support JSON output in the future so the client module can provide rich PowerShell objects. + +--- +>Drafted in October and preserved for context. + +## WinGet Modules +* Microsoft.Winget +* Microsoft.WinGet.Client +* Microsoft.WinGet.Create +* Microsoft.WinGet.Source + +## Microsoft.WinGet.Client cmdlets + +### Add-WinGetSource +Adds a source for the Windows Package Manager to use + +### Disable-WinGetLocalManifest +>This must be run in administrator mode + +### Edit-WinGetClientSetting +Open Windows Package Manager settings file + +### Enable-WinGetLocalManifest +>This must be run in administrator mode + +### Find-WinGetPackage +Searches for packages in configured sources + +### Get-WinGetPackage +Displays the list of packages installed on the local system + +### Get-WinGetSource +Displays the list of sources configured for the Windows Package Manager + +### Get-WinGetVersion +Gets the version for the Windows Package Manager Manifest Creator + +### Install-WinGetPackage +Installs the given package + +### Remove-WinGetSource +Removes a configured source from the Windows Package Manager + +### Reset-WinGetSource +Resets the default sources for the Windows Package Manager + +### Uninstall-WinGetPackage +Uninstalls a package from the local system + +### Upgrade-WinGetPackage +Upgrades a package installed on the local system + +### Get-WinGetInstaller (ToDo) +Displays the installer Install-WinGetPackage would select for the local system + +--- +## Microsoft.WinGet.Create Module cmdlets + +### Get-WinGetCreateVersion (ToDo) +Gets the version for the Windows Package Manager Manifest Creator + +### Edit-WinGetCreateSetting (ToDo) +Open Windows Package Manager Manifest Creator settings file + +### New-WinGetManifest -Path (ToDo) +Creates a new Manifest + +### Set-WinGetManifest - Path (ToDo) +Updates fields of an existing manifest + +### Submit-WinGetManifest -Path (ToDo) +Submits a manifest to the Windows Package Manager App Repository for validation + +### Add-WinGetManifestVersion (ToDo) +Adds a version to an existing manifest + +### Test-WinGetManifest (ToDo) +Validates a manifest + +### Open-WinGetCreateCache (ToDo) +Opens the cache folder storing the downloaded installers + +### Get-WinGetCreateCache (ToDo) +Lists out all the downloaded installers stored in cache + +### Clear-WinGetCreateCache (ToDo) +Deletes all downloaded installers in the cache folder + +--- +## Microsoft.WinGet.Source Module cmdlets + +** Focus on Private Repository (Rest), If there is time.. then we can look at future support of Public WinGet Source. + +### Add-WinGetManifest +Example: Add-WinGetManifest -Path "C:\Folder\File.json" -Source PrivateRepo +* Submit a Manifest to a repository +* -Version: Returns only a specific version of the Manifest as a manifest. +* If Version not specified, then return the latest version only as part of the manifest. + +### Get-WinGetManifest +Example: Get-WinGetManifest -Id Microsoft.PowerToys -Source PrivateRepo +* Gets a Manifest +* Needs to have the returned results "Beautified" when shown to the screen. + +### Set-WinGetManifestVersion +Set-WinGetManifestVersion [-Id] Microsoft.PowerToys [--Source] PrivateRepo --ShortDescription "New Description" [--Version] "33.0.0.0" +--Source {(PrivateRepo)} + +### Add-WinGetManifestVersion + Does not overwrite previously existing values. User must run "Set" to modify. + +### Get-WinGetManifestVersion +Example: Get-WinGetManifestVersion [-Id] Microsoft.PowerToys -Version 3.0.0.0 +Returns the values of a specific version in a Manifest + +### Remove-WinGetManifestVersion +This will remove manifest versions from a manifest located in the Private Source only. + +### Remove-WinGetManifest +This will remove manifests from the Private Source only. + +--- + +## ToDo + +This is not an exhaustive list, but is here as guide for work that needs to be performed. + +* Localization Support +* Handle error messages from the client in several scenarios +* Handle Group Policy messages +* Header is only valid in the context of a single source parameter. We may want to validate in the cmdlet +* Validation needs to be built +* The Modules should be moved to a new GitHub repository patterned after [Crescendo](https://www.powershellgallery.com/packages/PSPackageProject/0.1.18) +* Ultimately, the child modules will be moved to their respective GitHub repositories and the parent will stay in it's own repository. +* Validation should be implemented +* A CI Pipeline should be built to sign and publish the modules to the PowerShell gallery +* Packaging with [PSPackageProject](https://www.powershellgallery.com/packages/PSPackageProject/0.1.18) +* Documentation with [platyPS](https://www.powershellgallery.com/packages/platyPS/0.14.2) +* Support changes from https://github.com/microsoft/winget-cli/issues/1597 + +--- + +## Parking Lot + +PowerShell modules should provide at least parity with the Windows Package Manager + +The CLI is inefficient at returning multiple versions / installers. If we expand beyond what it can do, when we inevitably switch over to the CLI as the tool for running, then the community will see this as a loss of functionality. \ No newline at end of file