Private/Device/Software/NewPanSoftware.ps1

function NewPanSoftware {
<#
.SYNOPSIS
Returns a PanSoftware object.
.DESCRIPTION
Returns a PanSoftware object.
.NOTES
.INPUTS
None
.OUTPUTS
PanSoftware
.EXAMPLE
.NOTES
See help for NewPanJob for time zone related context.
#>

    [CmdletBinding()]
    param(
       [parameter(Mandatory=$true,HelpMessage='PanResponse')]
       [PanResponse] $Response,
       [parameter(Mandatory=$true,HelpMessage='PanDevice')]
       [PanDevice] $Device,
       [parameter(Mandatory=$true,HelpMessage='Time zone name (tz database / tzdata format) of PanDevice. Ex. America/Chicago')]
        [String] $TimeZoneName
    )
 
    # 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 + ':')

    # Determine TimeZoneInfo from the passed TimeZoneName before main processing loop
    # Offset will be determined for each software given daylight savings impact
    # Use the internal ConvertFromPanTimeZone helper to accommodate cross-platform nuances
    $TimeZoneInfo = ConvertFromPanTimeZone -Name $PSBoundParameters.TimeZoneName
    # If it cannot be found, assume (likely incorrectly) UTC (it's a standard)
    if(-not $TimeZoneInfo) {
        $TimeZoneInfo = [TimeZoneInfo]::FindSystemTimeZoneById('UTC')
    }
 
    # Container for processed software items
    $SoftwareAgg = [System.Collections.Generic.List[PanSoftware]]@()

    foreach($EntryCur in $PSBoundParameters.Response.Result.'sw-updates'.versions.entry) {
        $SoftwareNew = [PanSoftware]::new()
        $SoftwareNew.Version = $EntryCur.version
        $SoftwareNew.Filename = $EntryCur.filename
        $SoftwareNew.Size = $EntryCur.size
  
        # Released
        # PAN-OS XML-API released-on return format example below, does not include timezone indicator
        # 2025/02/20 21:05:20
        # In firewall local time
        $Regex = '(\d+)/(\d+)/(\d+) (\d+):(\d+):(\d+)'
        if($EntryCur.'released-on' -match $Regex) {
            # Calculate the offset for released-on to feed into future DateTimeOffset
            # Need a DateTime to assess whether DST is/was in effect at that DateTime
            $ReleasedDateTime = [DateTime]::new($Matches[1],$Matches[2],$Matches[3],$Matches[4],$Matches[5],$Matches[6],0,0)
            # If DST in effect, offset is the time zone default offset + 1
            if($TimeZoneInfo.IsDaylightSavingTime($ReleasedDateTime)) {
                $DstModification = New-TimeSpan -Hours 1
                $Offset = $TimeZoneInfo.BaseUtcOffset.Add($DstModification)
            }
            # If DST NOT in effect, offset is the time zone default offset
            else {
                $Offset = $TimeZoneInfo.BaseUtcOffset
            }

            $SoftwareNew.Released = [DateTimeOffset]::new($ReleasedDateTime,$Offset)
        }
        
        $SoftwareNew.ReleaseNotes = $EntryCur.'release-notes'.'#cdata-section'
        $SoftwareNew.Downloaded = if($EntryCur.downloaded -eq 'yes') {$True} else {$False}
        $SoftwareNew.Uploaded = if($EntryCur.uploaded -eq 'yes') {$True} else {$False}
        $SoftwareNew.Current = if($EntryCur.current -eq 'yes') {$True} else {$False}
        $SoftwareNew.Latest = if($EntryCur.latest -eq 'yes') {$True} else {$False}
        $SoftwareNew.ReleaseType = $EntryCur.'release-type'
        $SoftwareNew.Sha256 = $EntryCur.sha256
        $SoftwareNew.Device = $PSBoundParameters.Device

        # Add to aggregate
        $SoftwareAgg.Add($SoftwareNew)
    }

    return $SoftwareAgg | Sort-Object -Property 'Version'
} # Function