MetaNull.Queue.psm1
# Module Constants $User = [Security.Principal.WindowsIdentity]::GetCurrent() $Principal = [Security.Principal.WindowsPrincipal]::new($User) $Role = [Security.Principal.WindowsBuiltInRole]::Administrator if($Principal.IsInRole($Role)) { # Current use is Administrator $PSDriveRoot = 'HKLM:\SOFTWARE\MetaNull\PowerShell\MetaNull.Queue' } else { # Current user is not Administrator $PSDriveRoot = 'HKCU:\SOFTWARE\MetaNull\PowerShell\MetaNull.Queue' } if(-not (Test-Path $PSDriveRoot)) { New-Item -Path $PSDriveRoot -Force | Out-Null } New-Variable MetaNull -Scope script -Value @{ Queue = @{ PSDriveRoot = $PSDriveRoot Lock = New-Object Object Drive = New-PSDrive -Name 'MetaNull' -Scope Script -PSProvider Registry -Root $PSDriveRoot } } if(-not (Test-Path MetaNull:\Queues)) { New-Item -Path MetaNull:\Queues -Force | Out-Null } Function Test-IsAdministrator { <# .SYNOPSIS Tests if the current user has Administrative rights #> [CmdletBinding()] [OutputType([bool])] param() Process { $User = [Security.Principal.WindowsIdentity]::GetCurrent() $Principal = [Security.Principal.WindowsPrincipal]::new($User) $Role = [Security.Principal.WindowsBuiltInRole]::Administrator return $Principal.IsInRole($Role) } } Function Clear-Queue { <# .SYNOPSIS Remove all Commands from the queue #> [CmdletBinding()] [OutputType([pscustomobject])] param( [Parameter(Mandatory, ValueFromPipeline, ValueFromPipelineByPropertyName, Position = 0)] [guid] $Id ) Process { $BackupErrorActionPreference = $ErrorActionPreference $ErrorActionPreference = 'Stop' try { # Find the queue if(-not "$Id" -or -not (Test-Path "MetaNull:\Queues\$Id")) { throw "Queue $Id not found" } # Remove the command [System.Threading.Monitor]::Enter($MetaNull.Queue.Lock) try { Remove-Item "MetaNull:\Queues\$Id\Commands\*" -Force -Recurse } finally { [System.Threading.Monitor]::Exit($MetaNull.Queue.Lock) } } finally { $ErrorActionPreference = $BackupErrorActionPreference } } } Function Get-Queue { <# .SYNOPSIS Returns the list of Queues .DESCRIPTION Returns the list of Queues .PARAMETER Id The Id of the Queue to return .PARAMETER Name The filter to apply on the name of the Queues Filter supports wildcards .EXAMPLE # Get all the Queues Get-Queue .EXAMPLE # Get the Queue with the Id '00000000-0000-0000-0000-000000000000' Get-Queue -Id '00000000-0000-0000-0000-000000000000' .EXAMPLE # Get the Queues with the name starting with 'Queue' Get-Queue -Name 'Queue*' #> [CmdletBinding()] [OutputType([PSCustomObject])] param( [Parameter(Mandatory = $false, ValueFromPipeline, ValueFromPipelineByPropertyName, Position = 0)] [guid] $Id = [guid]::Empty, [Parameter(Mandatory = $false, Position = 1)] [SupportsWildcards()] [ArgumentCompleter( { param ( $commandName, $parameterName, $wordToComplete, $commandAst, $fakeBoundParameters ) $NameList = Get-ChildItem -Path "MetaNull:\Queues" | Get-ItemProperty | Select-Object Name | Select-Object -ExpandProperty Name $NameList -like *_ } )] [string] $Name = '*' ) Process { $BackupErrorActionPreference = $ErrorActionPreference $ErrorActionPreference = 'Stop' try { # Set the filter to '*' or the 'Id' $Filter = '*' if($Id -ne [guid]::Empty) { $Filter = $Id.ToString() } [System.Threading.Monitor]::Enter($MetaNull.Queue.Lock) try { # Get the queue(s) Get-Item -Path "MetaNull:\Queues\$Filter" | Foreach-Object { $Queue = $_ | Get-ItemProperty | Select-Object * | Select-Object -ExcludeProperty PS* $Queue | Add-Member -MemberType NoteProperty -Name 'RegistryKey' -Value $RegistryKey $Queue | Add-Member -MemberType NoteProperty -Name 'Commands' -Value @() # Return the queue object $Queue | Write-Output } | Where-Object { # Filter the queue(s) by 'Name' $_.Name -like $Name } | ForEach-Object { # Add command(s) to the queue object $_.Commands = Get-ChildItem "MetaNull:\Queues\$($_.Id)\Commands" | Foreach-Object { $Command = $_ | Get-ItemProperty | Select-Object * | Select-Object -ExcludeProperty PS* $Command | Add-Member -MemberType NoteProperty -Name 'RegistryKey' -Value $RegistryKey $Command | Write-Output } # Return the Queue object $_ | write-output } } finally { [System.Threading.Monitor]::Exit($MetaNull.Queue.Lock) } } finally { $ErrorActionPreference = $BackupErrorActionPreference } } } Function New-Queue { <# .SYNOPSIS Create a Queue #> [CmdletBinding()] [OutputType([guid])] param( [Parameter(Mandatory)] [string] $Name, [Parameter(Mandatory=$false)] [AllowNull()] [AllowEmptyString()] [string] $Description, [Parameter(Mandatory=$false)] [ValidateSet('Iddle','Running','Disabled','Suspended')] [string] $Status = 'Iddle' ) Process { $BackupErrorActionPreference = $ErrorActionPreference $ErrorActionPreference = 'Stop' try { # Create the queue object $Guid = [guid]::NewGuid().ToString() $Properties = @{ Id = $Guid Name = $Name Description = $Description Status = $Status } # Store the queue into the registry [System.Threading.Monitor]::Enter($MetaNull.Queue.Lock) try { $Item = New-Item -Path "MetaNull:\Queues\$Guid" -Force New-Item -Path "MetaNull:\Queues\$Guid\Commands" -Force | Out-Null $Properties.GetEnumerator() | ForEach-Object { $Item | New-ItemProperty -Name $_.Key -Value $_.Value | Out-Null } return $Guid } finally { [System.Threading.Monitor]::Exit($MetaNull.Queue.Lock) } } finally { $ErrorActionPreference = $BackupErrorActionPreference } } } Function Pop-QueueCommand { <# .SYNOPSIS Remove a Command from the top or bottom of the queue #> [CmdletBinding()] [OutputType([pscustomobject])] param( [Parameter(Mandatory, ValueFromPipeline, ValueFromPipelineByPropertyName, Position = 0)] [guid] $Id, [Parameter(Mandatory = $false)] [switch] $Unshift ) Process { $BackupErrorActionPreference = $ErrorActionPreference $ErrorActionPreference = 'Stop' [System.Threading.Monitor]::Enter($MetaNull.Queue.Lock) try { # Collect the existing commands $Commands = Get-ChildItem "MetaNull:\Queues\$Id\Commands" | Foreach-Object { $_ | Get-ItemProperty | Select-Object * | Select-Object -ExcludeProperty PS* | Write-Output } | Sort-Object -Property Index # Select which command to remove if($Unshift.IsPresent -and $Unshift) { $Command = $Commands | Select-Object -First 1 } else { $Command = $Commands | Select-Object -Last 1 } if(-not $Command -or -not $Command.Index) { return } # Remove the command from the registry Remove-Item -Force -Recurse "MetaNull:\Queues\$Id\Commands\$($Command.Index)" # Return the popped/unshifted command $Command | Write-Output } finally { [System.Threading.Monitor]::Exit($MetaNull.Queue.Lock) $ErrorActionPreference = $BackupErrorActionPreference } } } Function Push-QueueCommand { <# .SYNOPSIS Add a new Command at the end of a queue #> [CmdletBinding(DefaultParameterSetName='REG_MULTI_SZ')] [OutputType([int])] param( [Parameter(Mandatory, ValueFromPipeline, ValueFromPipelineByPropertyName, Position = 0)] [guid] $Id, [Parameter(Mandatory, Position = 1, ParameterSetName = 'REG_MULTI_SZ')] [string[]] $Commands, [Parameter(Mandatory, Position = 1, ParameterSetName = 'REG_SZ')] [string[]] $Command, [Parameter(Mandatory, Position = 1, ParameterSetName = 'REG_EXPAND_SZ')] [string[]] $ExpandableCommand, [Parameter(Mandatory = $false, Position = 2)] [AllowEmptyString()] [AllowNull()] [string] $Name, [Parameter(Mandatory = $false)] [switch] $Unique ) Process { $BackupErrorActionPreference = $ErrorActionPreference $ErrorActionPreference = 'Stop' [System.Threading.Monitor]::Enter($MetaNull.Queue.Lock) try { # Collect the existing commands $CommandList = [object[]](Get-ChildItem "MetaNull:\Queues\$Id\Commands" -ErrorAction SilentlyContinue | Foreach-Object { $_ | Get-ItemProperty | Select-Object * | Select-Object -ExcludeProperty PS* | Write-Output } | Sort-Object -Property Index) # Check if the command is already present if($Unique.IsPresent -and $Unique -and ($Commands | Where-Object { $_.Command -eq $Command })) { throw "Command already present in queue $Id" } # Find the last command index $LastCommandIndex = ($CommandList | Select-Object -Last 1 | Select-Object -ExpandProperty Index) + 1 # Create the new command $Properties = @{ Index = $LastCommandIndex Name = $Name # Command = $Command # This value is set separately, as it allows different types (REG_SZ, REG_MULTI_SZ, REG_EXPAND_SZ) } # Add the new command to the registry $Item = New-Item "MetaNull:\Queues\$Id\Commands\$LastCommandIndex" -Force switch($PSCmdlet.ParameterSetName) { 'REG_EXPAND_SZ' { $Item | New-ItemProperty -Name Command -Value $ExpandableCommand -Type ExpandString | Out-Null } 'REG_SZ' { $Item | New-ItemProperty -Name Command -Value $Command -Type String | Out-Null } default { $Item | New-ItemProperty -Name Command -Value ([string[]]$Commands) -Type MultiString | Out-Null } } $Properties.GetEnumerator() | ForEach-Object { $Item | New-ItemProperty -Name $_.Key -Value $_.Value | Out-Null } # Return the command Index return $LastCommandIndex } finally { [System.Threading.Monitor]::Exit($MetaNull.Queue.Lock) $ErrorActionPreference = $BackupErrorActionPreference } } } Function Remove-Queue { <# .SYNOPSIS Returns the list of Queues #> [CmdletBinding()] [OutputType([void])] param( [Parameter(Mandatory, ValueFromPipeline, ValueFromPipelineByPropertyName, Position = 0)] [guid] $Id, [Parameter(Mandatory = $false)] [switch] $Force ) Process { $BackupErrorActionPreference = $ErrorActionPreference $ErrorActionPreference = 'Stop' try { # Test if the queue exists if(-not "$Id" -or -not (Test-Path "MetaNull:\Queues\$Id")) { throw "Queue $Id not found" } # Remove the queue [System.Threading.Monitor]::Enter($MetaNull.Queue.Lock) try { $DoForce = $Force.IsPresent -and $Force Remove-Item "MetaNull:\Queues\$Id" -Recurse -Force:$DoForce } finally { [System.Threading.Monitor]::Exit($MetaNull.Queue.Lock) } } finally { $ErrorActionPreference = $BackupErrorActionPreference } } } |