Public/Device/Invoke-PanCommit.ps1

function Invoke-PanCommit {
<#
.SYNOPSIS
Commit, validate, or check for pending changes on a PanDevice.
.DESCRIPTION
Commit or validate configuration changes on a PanDevice. This cmdlet provides a wrapper around the PAN-OS XML API commit and validate functionality.

1) Choose -Commit or -Validate
2) Choose -Full or -Partial
3) Choose to -Force a commit (does not apply to validate). -Force is the PAN-OS "Force Commit" function, not the PowerShell override.

From there, specify additional optional parameters. There are many optional parameters to scope a -Partial commit or validate.
Successful commit and validation requests return a PanJob. The PanJob can be used to monitor the status and final result of commit or validation.

Separately, check for pending changes with standalone -PendingChanges switch. Returns $True or $False.
.NOTES
Invoke-PanCommit abstracts several related capabilities within PAN-OS into a single cmdlet.
Config mode commit, commit partial which are part of XML-API type=commit
Config mode validate, validate partial which are part of XML-API type=op (operational)
Exec mode check pending changes which is part of XML-API type=op (operational)

These logically grouped capabilities are built-into a single cmdlet instead of many constituent cmdlets.
.INPUTS
PanDevice[]
   You can pipe a PanDevice to this cmdlet
.OUTPUTS
PanJob
.EXAMPLE
PS> Get-PanDevice '10.0.0.1' | Invoke-PanCommit -Commit -Full
Standard commit of full configuration. Returns a PanJob.
.EXAMPLE
PS> Get-PanDevice '10.0.0.1' | Invoke-PanCommit -Commit -Partial -Admin "JohnnyU" -Description "Emergency Change"
Partial commit only committing changes by "JohnnyU" with a description. Returns a PanJob.
.EXAMPLE
PS> Get-PanDevice '10.0.0.1' | Invoke-PanCommit -Validate -Full
Standard validation of full configuration. Returns a PanJob.
.EXAMPLE
PS> Get-PanDevice '10.0.0.1' | Invoke-PanCommit -PendingChanges
Returns $True is changes are pending in candidate configuration. $False if there are no pending changes.
#>

    [CmdletBinding()]
    param(
        [parameter(Mandatory=$true,Position=0,ValueFromPipeline=$true,HelpMessage='PanDevice(s) to commit configuration changes.')]
        [PanDevice[]] $Device,
        [parameter(Mandatory=$true,Position=1,ParameterSetName='Commit-Full',HelpMessage='Commit switch.')]
        [parameter(Mandatory=$true,Position=1,ParameterSetName='Commit-Partial',HelpMessage='Commit switch.')]
        [Switch] $Commit,
        [parameter(Mandatory=$true,Position=1,ParameterSetName='Validate-Full',HelpMessage='Validate switch. Validate the commit without committing.')]
        [parameter(Mandatory=$true,Position=1,ParameterSetName='Validate-Partial',HelpMessage='Validate switch. Validate the commit without committing.')]
        [Switch] $Validate,
        ####
        #### Experiment with default value of $Full:$true to see if can avoid -Commit -Full explicitness on the PowerShell commandline.
        ####
        [parameter(Mandatory=$true,Position=2,ParameterSetName='Commit-Full',HelpMessage='Partial switch.')]
        [parameter(Mandatory=$true,Position=2,ParameterSetName='Validate-Full',HelpMessage='Partial switch.')]
        [Switch] $Full,
        [parameter(Mandatory=$true,Position=2,ParameterSetName='Commit-Partial',HelpMessage='Partial switch.')]
        [parameter(Mandatory=$true,Position=2,ParameterSetName='Validate-Partial',HelpMessage='Partial switch.')]
        [Switch] $Partial,
        [parameter(ParameterSetName='Commit-Full',HelpMessage='PAN-OS "Force Commit", not PowerShell override.')]
        [parameter(ParameterSetName='Commit-Partial',HelpMessage='PAN-OS "Force Commit", not PowerShell override.')]
        [Switch] $Force,
        [parameter(ParameterSetName='Commit-Full',HelpMessage='Description for commit.')]
        [parameter(ParameterSetName='Commit-Partial',HelpMessage='Description for commit.')]
        [String] $Description,
        [parameter(ParameterSetName='Commit-Partial',HelpMessage='Limit scope to specific Admin(s) changes.')]
        [parameter(ParameterSetName='Validate-Partial',HelpMessage='Limit scope to specific Admin(s) changes.')]
        [String[]] $Admin,
        [parameter(ParameterSetName='Commit-Partial',HelpMessage='Limit scope to specific vsys(s).')]
        [parameter(ParameterSetName='Validate-Partial',HelpMessage='Limit scope to specific vsys(s).')]
        [String[]] $Vsys,
        [parameter(ParameterSetName='Commit-Partial',HelpMessage='Not clear on use but supported by CLI & API.')]
        [parameter(ParameterSetName='Validate-Partial',HelpMessage='Not clear on use but supported by CLI & API.')]
        [Switch] $NoVsys,
        [parameter(ParameterSetName='Commit-Partial',HelpMessage='Exclude Device & Network from scope.')]
        [parameter(ParameterSetName='Validate-Partial',HelpMessage='Exclude Device & Network from scope.')]
        [Switch] $ExcludeDeviceAndNetwork,
        [parameter(ParameterSetName='Commit-Partial',HelpMessage='Exclude Shared Objects from scope.')]
        [parameter(ParameterSetName='Validate-Partial',HelpMessage='Exclude Shared Objects from scope.')]
        [Switch] $ExcludeSharedObject,
        [parameter(ParameterSetName='Commit-Partial',HelpMessage='XPath(s) to include in scope.')]
        [parameter(ParameterSetName='Validate-Partial',HelpMessage='XPath(s) to include in scope.')]
        [String[]] $XPath,
        [parameter(ParameterSetName='Pending-Changes',HelpMessage='Determine pending changes in candidate configuration.')]
        [Switch] $PendingChanges
    )

    Begin {
        # 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 + ':')
    } # Begin block

    Process {
        foreach($DeviceCur in $Device) {
            Write-Debug ($MyInvocation.MyCommand.Name + (': Device: {0}' -f $DeviceCur.Name))
            
            # ParameterSet name Pending-Changes
            if($PSCmdlet.ParameterSetName -eq 'Pending-Changes') {
                Write-Debug ($MyInvocation.MyCommand.Name + ': -PendingChanges')
                $Cmd = '<check><pending-changes></pending-changes></check>'
                $Response = Invoke-PanXApi -Device $DeviceCur -Op -Cmd $Cmd
                if($Response.result -eq 'no') {
                    Write-Debug ($MyInvocation.MyCommand.Name + (': Device : {0} has NO pending changes to be committed.' -f $DeviceCur.Name))
                    return $False
                }
                elseif($Response.result -eq 'yes') {
                    Write-Debug ($MyInvocation.MyCommand.Name + (': Device : {0} HAS pending changes to be committed.' -f $DeviceCur.Name))
                    return $True
                }
            }
            # Any other ParameterSetName (all the Commit- and Validate-)
            else {
                # -Commit and -Validate base
                # $XmlDoc used to build and chain XML elements together
                $XmlDoc = [System.Xml.XmlDocument]::new()
                # Set the Root to <commit> or <validate>
                # $XmlRoot will contain the final XML to be used in Cmd. Build it slowly.
                if($PSBoundParameters.Commit.IsPresent) {
                    Write-Debug ($MyInvocation.MyCommand.Name + ': -Commit')
                    $XmlRoot = $XmlDoc.CreateElement('commit')
                } 
                else {
                    Write-Debug ($MyInvocation.MyCommand.Name + ': -Validate')
                    $XmlRoot = $XmlDoc.CreateElement('validate')
                }
                $XmlDoc.AppendChild($XmlRoot) | Out-Null

                # -Force
                # Force is only relevant for Commit operation, not Validate. ParameterSet definitions control where it can be used
                if($PSBoundParameters.Force.IsPresent) {
                    Write-Debug ($MyInvocation.MyCommand.Name + ': -Force')
                    $XmlForce = $XmlDoc.CreateElement('force')
                    # Trick to get a closing </force> tag on the valueless <force />
                    # https://stackoverflow.com/questions/45270479/force-xmldocument-to-save-empty-elements-with-an-explicit-closing-tag
                    $XmlForce.AppendChild($XmlDoc.CreateWhitespace('')) | Out-Null
                    # Append to root
                    $XmlRoot.AppendChild($XmlForce) | Out-Null
                    # Force requested, XmlForce is go-foward point of working tree
                    $XmlWork = $XmlForce
                }
                else {
                    # Force not requested, XmlRoot is go-foward point of working tree
                    $XmlWork = $XmlRoot
                }

                # -Full and -Partial
                # PAN-OS Continues to amaze
                # Commit partial is <commit><partial>...</partial></commit>
                # Validate partial is <validate><partial>...</partial</validate>
                # Commit full is <commit>...</commit> (no <full></full>, the odd one out)
                # Valid full is <validate><full>...</full></validate>
                if($PSCmdlet.ParameterSetName -like '*-Partial') {
                    Write-Debug ($MyInvocation.MyCommand.Name + ': -Partial')
                    $XmlPartial = $XmlDoc.CreateElement('partial')
                    # Trick for closing tag
                    $XmlPartial.AppendChild($XmlDoc.CreateWhitespace('')) | Out-Null
                    $XmlWork.AppendChild($XmlPartial) | Out-Null
                    # Partial requested, XmlPartial is go-forward point of working tree
                    $XmlWork = $XmlPartial
            
                }
                elseif($PSCmdlet.ParameterSetName -eq 'Validate-Full') {
                    Write-Debug ($MyInvocation.MyCommand.Name + ': (Validate) -Full')
                    $XmlFull = $XmlDoc.CreateElement('full')
                    # Trick for closing tag
                    $XmlFull.AppendChild($XmlDoc.CreateWhitespace('')) | Out-Null
                    $XmlWork.AppendChild($XmlFull) | Out-Null
                    # Validate full, XmlFull is go-forward point of working tree
                    $XmlWork = $XmlFull
                }
                elseif($PSCmdlet.ParameterSetName -eq 'Commit-Full') {
                    Write-Debug ($MyInvocation.MyCommand.Name + ': (Commit) -Full')
                    # Nothing to do for Commit-Full. Go-forward point of working tree does not change
                }

                # -Description <description>My Description</description>
                if($PSBoundParameters.Description) {
                    Write-Debug ($MyInvocation.MyCommand.Name + ': -Description')
                    $XmlDescription = $XmlDoc.CreateElement('description')
                    $XmlDescription.InnerText = $PSBoundParameters.Description
                    $XmlWork.AppendChild($XmlDescription) | Out-Null
                }

                # -Admin <admin><member>Admin1</member><member>Admin2</member></admin>
                if($PSBoundParameters.Admin) {
                    Write-Debug ($MyInvocation.MyCommand.Name + (': -Admin (Count:{0})' -f $PSBoundParameters.Admin.Count))
                    $XmlAdmin = $XmlDoc.CreateElement('admin')
                    foreach($AdminCur in $PSBoundParameters.Admin) {
                        $XmlMember = $XmlDoc.CreateElement('member')
                        $XmlMember.InnerText = $AdminCur
                        # Add the <member> to the <admin>
                        $XmlAdmin.AppendChild($XmlMember) | Out-Null
                    }
                    $XmlWork.AppendChild($XmlAdmin) | Out-Null
                    # # Go-forward point of working tree does not change
                }

                # -Vsys <vsys><member>vsys1</member><member>vsys2</member></vsys>
                if($PSBoundParameters.Vsys) {
                    Write-Debug ($MyInvocation.MyCommand.Name + (': -Vsys (Count:{0})' -f $PSBoundParameters.Vsys.Count))
                    $XmlVsys = $XmlDoc.CreateElement('vsys')
                    foreach($VsysCur in $PSBoundParameters.Vsys) {
                        $XmlMember = $XmlDoc.CreateElement('member')
                        $XmlMember.InnerText = $VsysCur
                        # Add the <member> to the <vsys>
                        $XmlVsys.AppendChild($XmlMember) | Out-Null
                    }
                    $XmlWork.AppendChild($XmlVsys) | Out-Null
                    # # Go-forward point of working tree does not change
                }

                # -NoVsys <novsys></novsys>
                if($PSBoundParameters.NoVsys.IsPresent) {
                    Write-Debug ($MyInvocation.MyCommand.Name + ': -NoVsys')
                    $XmlNoVsys = $XmlDoc.CreateElement('novsys')
                    # Force a closing tag
                    $XmlNoVsys.AppendChild($XmlDoc.CreateWhitespace('')) | Out-Null
                    $XmlWork.AppendChild($XmlNoVsys) | Out-Null
                    # Go-forward point of working tree does not change
                }

                # -ExcludeDeviceAndNetwork <device-and-network>excluded</device-and-network>
                if($PSBoundParameters.ExcludeDeviceAndNetwork.IsPresent) {
                    Write-Debug ($MyInvocation.MyCommand.Name + ': -ExcludeDeviceAndNetwork')
                    $XmlDeviceAndNetwork = $XmlDoc.CreateElement('device-and-network')
                    $XmlDeviceAndNetwork.InnerText = 'excluded'
                    $XmlWork.AppendChild($XmlDeviceAndNetwork) | Out-Null
                    # Go-forward point of working tree does not change
                }

                # -ExcludeSharedObject <shared-object>excluded</shared-object>
                if($PSBoundParameters.ExcludeSharedObject.IsPresent) {
                    Write-Debug ($MyInvocation.MyCommand.Name + ': -ExcludeSharedObject')
                    $XmlSharedObject = $XmlDoc.CreateElement('shared-object')
                    $XmlSharedObject.InnerText = 'excluded'
                    $XmlWork.AppendChild($XmlSharedObject) | Out-Null
                    # Go-forward point of working tree does not change
                }

                # -XPath <object-xpaths><member>/config/devices/entry[@name='localhost.localdomain']/vsys/entry[@name='vsys1']/service</member><member>...</member></object-xpaths>
                if($PSBoundParameters.XPath) {
                    Write-Debug ($MyInvocation.MyCommand.Name + (': -XPath (Count:{0})' -f $PSBoundParameters.XPath.Count))
                    $XmlObjectXPaths = $XmlDoc.CreateElement('object-xpaths')
                    foreach($XPathCur in $PSBoundParameters.XPath) {
                        $XmlMember = $XmlDoc.CreateElement('member')
                        $XmlMember.InnerText = $XPathCur
                        # Add the <member> to the <object-xpaths>
                        $XmlObjectXPaths.AppendChild($XmlMember) | Out-Null
                    }
                    $XmlWork.AppendChild($XmlObjectXPaths) | Out-Null
                    # # Go-forward point of working tree does not change
                }

                # Finished building XML. Make the request.
                if($PSCmdlet.ParameterSetName -like 'Commit-Full') {
                    Write-Debug ($MyInvocation.MyCommand.Name + ': -Commit Cmd: {0}' -f $XmlRoot.OuterXml)
                    $Response = Invoke-PanXApi -Device $DeviceCur -Commit -Cmd $XmlRoot.OuterXml
                }
                elseif($PSCmdlet.ParameterSetName -like 'Commit-Partial') {
                    Write-Debug ($MyInvocation.MyCommand.Name + ': -Commit -Action "partial" Cmd: {0}' -f $XmlRoot.OuterXml)
                    $Response = Invoke-PanXApi -Device $DeviceCur -Commit -Action 'partial' -Cmd $XmlRoot.OuterXml
                }
                elseif($PSCmdlet.ParameterSetName -like 'Validate-*') {
                    Write-Debug ($MyInvocation.MyCommand.Name + ': -Validate Cmd: {0}' -f $XmlRoot.OuterXml)
                    $Response = Invoke-PanXApi -Device $DeviceCur -Op -Cmd $XmlRoot.OuterXml
                }
                # PAN-OS responses are same for commit and validate operations. Can use same logic for both
                # Validate responses are identical to below, just change word "commit" to "validate"
                # PAN-OS gives two different types of responses on success based on whether *pending changes* or not
                # Pending Changes:
                # <response status="success" code="19">
                # <result>
                # <msg>
                # <line>Commit job enqueued with jobid 4</line>
                # </msg>
                # <job>4</job>
                # </result>
                # </response>
                # NO Pending Changes:
                # <response status="success" code="19"><msg>There are no changes to commit.</msg></response>
                if($Response.Status -eq 'success') {
                    # If pending changes and a job
                    if($Response.Result.job) {
                    # Inform an interactive user of the JobID using Write-Host given operation is asynchronous
                    Write-Host $Response.Result.msg.line
                    # Send a PanJob object down the pipeline
                    Get-PanJob -Device $DeviceCur -Id $Response.Result.job
                    }
                    # No pending changes
                    else {
                        Write-Warning $Response.Message
                    }
                }
                # If request to commit did not succeed.
                else {
                    Write-Error ('Commit/Validate request failed. Status: {0} Code: {1} Message: {2}' -f $Response.Status,$Response.Code,$Response.Message)
                }
            } # else Any other ParameterSetName
        } # foreach $DeviceCur
    } # Process block
    End {
    } # End block
} # Function