Public/Device/New-PanDevice.ps1
function New-PanDevice { <# .SYNOPSIS Creates a new PanDevice object and adds/persists to the PanDeviceDb. .DESCRIPTION Creates a new PanDevice object and adds/persists to the PanDeviceDb. .NOTES Regardless of how the PanDevice is created, the in-memory and on-disk secrets (password and API key) are stored encrypted. In-memory as a SecureString. On-disk as a serialized SecureString that is only decryptable by the user account with which it was created. .INPUTS None .OUTPUTS PanDevice or $false .EXAMPLE # Using Username and Password PSCredential, most secure with no username or password visible on command-line, prompted for both. New-PanDevice -Name "fw.lab.local" -Credential $(Get-Credential) -Keygen # Using Username and Password PSCredential, including username "admin" on the command-line, prompted for password New-PanDevice -Name "fw.lab.local" -Credential "admin" -Keygen # Specifying both username and password in plaintext. Not prompted for anything. Included for non-interactive support. Avoid where possible. New-PanDevice -Name "fw.lab.local" -Username "admin" -Password "admin123" -Keygen .EXAMPLE # Using API key with PSCredential, most secure with no key visible on command-line, prompted instead. Username is ignored. New-PanDevice -Name "acme-edge-fw1.acme.net" -KeyCredential $(Get-Credential) -Label "acme-edge-fw1","Azure" # Or pre-populate the ignored username New-PanDevice -Name "acme-edge-fw1.acme.net" -KeyCredential "throwaway" -Label "acme-edge-fw1","Azure" # Specifying API key in plaintext. Not prompted for anything. Included for non-interactive support. Avoid where possible. New-PanDevice -Name "acme-edge-fw1.acme.net" -Key "A1E2I3O4U5LONGAPIKEY" -Label "acme-edge-fw1","Azure" .EXAMPLE # Add some labels while creating New-PanDevice -Name "acme-edge-fw1.acme.net" -Credential $(Get-Credential) -Label "acme-edge-fw1","Azure" .EXAMPLE # Non-interactive example for creating new PanDevice using environment variables and -NoPersist. # -NoPersist avoids writing the PanDevice to disk for later use. New-PanDevice -Name $Env:MYPANHOST -Key $Env:MYPANKEY -NoPersist #> [CmdletBinding()] # OutputType of [PanDevice] fails given typing issues [OutputType([Bool])] param( [parameter(Mandatory=$true,Position=0,HelpMessage='Name or IP address of PAN device')] [String] $Name, [parameter(Mandatory=$true,Position=1,ParameterSetName="UserPass",HelpMessage='Non-interactive: Username against which API key will be generated. To be stored as secure [PSCredential]')] [String] $Username, [parameter(Mandatory=$true,Position=2,ParameterSetName="UserPass",HelpMessage='Non-interactive: Password associated with Username against which API key will be generated. To be stored as secure [PSCredential]')] [String] $Password, [parameter(Mandatory=$true,Position=1,ParameterSetName="Credential",HelpMessage='Interactive: PowerShell [PSCredential] against which API key will be generated')] # "Credential Attribute" to be able to pre-specify a username, like -Credential "John" and be prompted securely for only a password # https://learn.microsoft.com/en-us/powershell/module/microsoft.powershell.core/about/about_functions_advanced_parameters?view=powershell-5.1#attributes-of-parameters [System.Management.Automation.Credential()] [PSCredential] $Credential, [parameter(ParameterSetName="UserPass",HelpMessage='When specifying -Username and -Password, use this switch parameter to generate an API key')] [parameter(ParameterSetName="Credential",HelpMessage='When specifying -Credential, use this switch parameter to generate an API key')] [Switch] $Keygen, [parameter(Mandatory=$true,Position=1,ParameterSetName="Key",HelpMessage='Non-interactive: Pre-generated API key. To be stored as [SecureString]')] [String] $Key, [parameter(Mandatory=$true,Position=1,ParameterSetName="KeyCredential",HelpMessage='Interactive: Pre-generated API key. To be stored as [SecureString]. Username portion of PSCredential is ignored.')] [System.Management.Automation.Credential()] [PSCredential] $KeyCredential, [parameter(HelpMessage='PowerPAN locally significant label(s) to facilitate session-ease, friendly name, and grouping')] [System.Collections.Generic.List[String]] $Label = [System.Collections.Generic.List[String]]@(), [parameter(HelpMessage='Default is to disable x.509 certificate validation. Use this switch parameter to enable x.509 certificate validation')] [Switch] $ValidateCertificate = $false, [parameter(HelpMessage='Default is "https". Choose "http" or "https"')] [ValidatePattern("http|https")] [String] $Protocol = "https", [parameter(HelpMessage='Default is 443. Choose 1 - 65535')] [ValidateRange(1,65535)] [Int] $Port = 443, [parameter(HelpMessage='Default is to persist created [PanDevice] to PanDeviceDb. Use this switch parameter to not add PanDevice to PanDeviceDb. Commonly enabled with scripts.')] [Switch] $NoPersist = $false, [parameter(HelpMessage='Internal module use only. Use this switch parameter during unserializing to prevent adding session-specific Label.')] [Switch] $ImportMode = $false ) # Propagate -Debug and -Verbose to this module function, https://tinyurl.com/y5dcbb34 if($PSBoundParameters.Debug) { $DebugPreference = 'Continue' } if($PSBoundParameters.Verbose) { $VerbosePreference = 'Continue' } # Announce Write-Debug ($MyInvocation.MyCommand.Name + ':') # Update -Label parameter based on -ImportMode parameter if($ImportMode.IsPresent) { Write-Debug ($MyInvocation.MyCommand.Name + ': ImportMode: Not adding session-based Label') } else { Write-Debug ($MyInvocation.MyCommand.Name + ': Adding session-based Label') $Label.Add("session-$(GetPanSessionGuid)") } # Key parameter set :: -Key parameter present. API key previously generated, does not need to be created. # KeyCredential parameter set :: -KeyCredential parameter present. API key previously generated, does not need to be created. # For KeyCredential, ignore the Username portion of the PSCredential. if($PSCmdlet.ParameterSetName -eq 'Key' -or $PSCmdlet.ParameterSetName -eq 'KeyCredential') { if($PSCmdlet.ParameterSetName -eq 'Key') { Write-Debug ($MyInvocation.MyCommand.Name + ': Key parameter set') # Convert plaintext key to [SecureString] immediately and $null out original. $SecureKey = ConvertTo-SecureString -String $Key -AsPlainText -Force $Key = $null } elseif($PSCmdlet.ParameterSetName -eq 'KeyCredential') { # With KeyCredential, the PSCredential.Password representing API key is already a [SecureString]. $SecureKey = $KeyCredential.Password } # Create PanDevice $Device = [PanDevice]::New($Name, $SecureKey, $Label, $ValidateCertificate.IsPresent, $Protocol, $Port) } # UserPass or Credential parameter set, optional -Keygen parameter valid for both parameter sets elseif($PSCmdlet.ParameterSetName -eq 'UserPass' -or $PSCmdlet.ParameterSetName -eq 'Credential') { # UserPass parameter set :: -Username and -Password if($PSCmdlet.ParameterSetName -eq 'UserPass') { Write-Debug ($MyInvocation.MyCommand.Name + ': UserPass parameter set') # Convert password to [SecureString] immediately and $null out plaintext password variable $SecurePassword = ConvertTo-SecureString $Password -AsPlainText -Force $Password = $null # Build PSCredential from $UserName and $SecurePassword $Credential = New-Object -TypeName PSCredential -ArgumentList $Username, $SecurePassword # Create base PanDevice object $Device = [PanDevice]::New($Name, $Credential, $Label, $ValidateCertificate.IsPresent, $Protocol, $Port) } # Credential parameter set :: -Credential elseif($PSCmdlet.ParameterSetName -eq 'Credential') { Write-Debug ($MyInvocation.MyCommand.Name + ': Credential parameter set') # Create base PanDevice object $Device = [PanDevice]::New($Name, $Credential, $Label, $ValidateCertificate.IsPresent, $Protocol, $Port) } # Optionally generate API key if($Keygen.IsPresent) { Write-Debug ($MyInvocation.MyCommand.Name + ': -Keygen: Generating API key') $PanResponse = Invoke-PanXApi -Device $Device -Keygen if($PanResponse.Status -eq 'success'){ Write-Debug ($MyInvocation.MyCommand.Name + ': -Keygen: API key generation successful') $Device.Key = ConvertTo-SecureString -String $PanResponse.Result.key -AsPlainText -Force # Test API key Write-Debug ($MyInvocation.MyCommand.Name + ': -Keygen: Testing generated API key') $PanResponse = Invoke-PanXApi -Device $Device -Op -Cmd '<show><system><info></info></system></show>' if($PanResponse.Status -eq 'success'){ Write-Debug ($MyInvocation.MyCommand.Name + ': Keygen: Generated API key tested successfully') Write-Debug ("`t DeviceName: {0} Family: {1} Model: {2}" -f $PanResponse.Result.system.devicename,$PanResponse.Result.system.family,$PanResponse.Result.system.model) if($PanResponse.Result.system.family -eq 'vm') { Write-Debug ("`t VM-License: {0} VM-Mode: {1}" -f $PanResponse.Result.system.'vm-license',$PanResponse.Result.system.'vm-mode') } Write-Debug ("`t Serial: {0} Software Version: {1}" -f $PanResponse.Result.system.serial,$PanResponse.Result.system.'sw-version') } else { Write-Error ('Error testing generated API key. Status: {0} Code: {1} Message: {2}' -f $Response.Status,$Response.Code,$Response.Message) return $false } } else { Write-Error ('Error generating API key. Status: {0} Code: {1} Message: {2}' -f $Response.Status,$Response.Code,$Response.Message) return $false } } # End optional generate API key } # End UserPass / Credential parameter set # Completed building PanDevice for all parameter set. Determine whether to add to PanDeviceDb. if($NoPersist.IsPresent) { Write-Debug ($MyInvocation.MyCommand.Name + ': NoPersist: Not adding to PanDeviceDb') return $Device } else { # Determine whether to add to PanDeviceDb with -ImportMode if($ImportMode.IsPresent) { Write-Debug ($MyInvocation.MyCommand.Name + ': ImportMode: Adding to PanDeviceDb') Add-PanDevice -Device $Device -ImportMode } else { Write-Debug ($MyInvocation.MyCommand.Name + ': Adding to PanDeviceDb') Add-PanDevice -Device $Device } return $Device } } # End New-PanDevice |