Private/ConvertFromPanTimeZone.ps1
function ConvertFromPanTimeZone { <# .SYNOPSIS Returns a TimeZoneInfo object from a PAN time zone string. Internal helper cmdlet. .DESCRIPTION Returns a TimeZoneInfo object from a PAN time zone string. Internal helper cmdlet. .NOTES PowerShell 6+ and the corresponding .NET 6+ have robust cross-platform time zone support within the TimeZoneInfo class supporting both the Windows time zone style "AUS Eastern Standard Time" and IANA style 'Australia/Sydney'. https://devblogs.microsoft.com/dotnet/date-time-and-time-zone-enhancements-in-net-6/#time-zone-conversion-apis Windows PowerShell 5.1 and corresponding .NET 4.5/4.6 TimeZoneInfo class IN CONSTRAST only support Windows style "AUS Eastern Standard Time" time zones. PAN-OS configuration stores an IANA style, posing a problem for Windows PowerShell 5.1. This helper cmdlet accepts a time zone string and returns a TimeZoneInfo object. When used on PowerShell 6+, the native [TimeZoneInfo]::FindSystemTimeZoneById() will be used When used on PowerShell 5.1, an additional external assembly (TimeZoneConverter.dll) will be loaded at runtime and used. TimeZoneConverter will map IANA style time zone names to the only supported Windows time zone names when running on Windows PowerShell 5.1. The TimeZoneConverter NuGet package (and assembly) is how many developers worked around the limitation prior to .NET 6+. https://www.nuget.org/packages/TimeZoneConverter/ https://github.com/mattjohnsonpint/TimeZoneConverter .INPUTS None .OUTPUTS TimeZoneInfo .EXAMPLE # Running on PowerShell 6+ returns a TimeZoneInfo object for 'America/Chicago' ConvertFromPanTimeZone -Name 'America/Chicago' .EXAMPLE # Running on Windows PowerShell 5.1 returns a TimeZoneInfo object for 'Central Standard Time' # (thanks to TimeZoneConverter) ConvertFromPanTimeZone -Name 'America/Chicago' #> [CmdletBinding()] param( [parameter(Mandatory=$true,Position=0,HelpMessage='PAN-OS time zone name')] [String] $Name ) # 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 + ':') # PowerShell 6+ (PowerPAN supports 7+ LTS releases) if($PSVersionTable.PSVersion.Major -ge 8) { Write-Debug ($MyInvocation.MyCommand.Name + ': PowerShell 6+') # Use .NET native FindSystemTimeZoneById to return a TimeZoneInfo object return [TimeZoneInfo]::FindSystemTimeZoneById($PSBoundParameters.Name) } # PowerShell 5.1 else { # If the assemply is already loaded, don't load it again Write-Debug ($MyInvocation.MyCommand.Name + ': < PowerShell 6+ (likely 5.1)') if(([System.AppDomain]::CurrentDomain.GetAssemblies() | Where-Object {$_.FullName -like 'TimeZoneConverter*'}).Count -gt 0) { Write-Debug ($MyInvocation.MyCommand.Name + ': TimeZoneConverter assembly already loaded') } # TimeZoneConvert assembly NOT loaded, load it else { Write-Debug ($MyInvocation.MyCommand.Name + ': Loading TimeZoneConverter assembly') # Assembly included in the module, use ModuleBase to get there $AssemblyDllPath = (Get-Module -Name 'PowerPAN').ModuleBase + '/Assembly/timezoneconverter.7.0.0/net462/TimeZoneConverter.dll' Add-Type -Path $AssemblyDllPath } # Use [TimeZoneConvert.TZConvert]::GetTimeZoneInfo() capabilities to return a TimeZoneInfo object return [TimeZoneConverter.TZConvert]::GetTimeZoneInfo($Name) } } |