Public/Utility/New-OATHTokenSerial.ps1
<# .SYNOPSIS Generates unique serial numbers for OATH tokens .DESCRIPTION Creates unique, properly formatted serial numbers for use with OATH hardware tokens. Provides several options for format and ensures uniqueness when checking against existing tokens. .PARAMETER Prefix A prefix to add to the generated serial number (e.g., company identifier) .PARAMETER Length The total length of the numeric portion of the serial number .PARAMETER Format The format of the serial number (Numeric, Alphanumeric, Hexadecimal) .PARAMETER CheckExisting If specified, checks that generated serial numbers don't already exist in the tenant .PARAMETER Count The number of serial numbers to generate .EXAMPLE New-OATHTokenSerial Generates a new random 8-digit numeric serial number .EXAMPLE New-OATHTokenSerial -Prefix "ACME-" -Length 6 -Format Alphanumeric -Count 5 Generates 5 unique alphanumeric serial numbers with "ACME-" prefix, like "ACME-A7B23X" .EXAMPLE New-OATHTokenSerial -Prefix "YK" -Format Hexadecimal -CheckExisting Generates a hexadecimal serial number with "YK" prefix and checks that it doesn't exist in the tenant .NOTES Requires Microsoft.Graph.Authentication module and appropriate permissions when using -CheckExisting: - Policy.ReadWrite.AuthenticationMethod #> function New-OATHTokenSerial { [CmdletBinding()] [OutputType([string[]])] param( [Parameter()] [string]$Prefix = "", [Parameter()] [ValidateRange(4, 16)] [int]$Length = 8, [Parameter()] [ValidateSet('Numeric', 'Alphanumeric', 'Hexadecimal')] [string]$Format = 'Numeric', [Parameter()] [switch]$CheckExisting, [Parameter()] [ValidateRange(1, 1000)] [int]$Count = 1 ) begin { # Initialize skip processing flag when checking existing tokens $script:skipProcessing = $false # Generate character sets based on selected format function Get-CharacterSet { param([string]$Format) switch ($Format) { 'Numeric' { return '0123456789' } 'Alphanumeric' { return 'ABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789' } 'Hexadecimal' { return '0123456789ABCDEF' } default { throw "Invalid format: $Format" } } } $charSet = Get-CharacterSet -Format $Format $existingSerials = @() # Retrieve existing tokens if needed if ($CheckExisting) { if (-not (Test-MgConnection)) { $script:skipProcessing = $true Write-Warning "Not connected to Microsoft Graph. Cannot check for existing tokens." # Not throwing - we'll continue without checking duplicates } if (-not $script:skipProcessing) { try { Write-Verbose "Retrieving existing tokens to check for duplicate serial numbers..." $tokens = Get-OATHToken $existingSerials = $tokens | Select-Object -ExpandProperty SerialNumber Write-Verbose "Found $($existingSerials.Count) existing tokens" } catch { Write-Warning "Failed to retrieve existing tokens: $_" # Don't throw - we'll continue and just not check for duplicates } } } } process { $results = @() $generateCount = 0 $attemptCount = 0 $maxAttempts = $Count * 3 # Allow for some retries in case of duplicates while ($generateCount -lt $Count -and $attemptCount -lt $maxAttempts) { $attemptCount++ # Generate a random serial $serial = "" for ($i = 0; $i -lt $Length; $i++) { $randomIndex = Get-Random -Minimum 0 -Maximum $charSet.Length $serial += $charSet[$randomIndex] } # Add prefix if specified $completeSerial = "$Prefix$serial" # Check if it already exists if ($CheckExisting -and $existingSerials -contains $completeSerial) { Write-Verbose "Serial $completeSerial already exists, generating another..." continue } # Check if we've already generated this one if ($results -contains $completeSerial) { Write-Verbose "Serial $completeSerial is a duplicate of a previously generated serial, generating another..." continue } # Add to results and track to avoid duplicates $results += $completeSerial $existingSerials += $completeSerial $generateCount++ } if ($generateCount -lt $Count) { Write-Warning "Could only generate $generateCount unique serial numbers out of $Count requested after $maxAttempts attempts" } return $results } } |