Public/UI/Show-OATHTokenMenu.ps1
<# .SYNOPSIS Displays the main OATH token management menu .DESCRIPTION Presents a text-based menu interface for managing OATH tokens in Microsoft Entra ID. Provides options for listing, adding, removing, and managing tokens. .PARAMETER DefaultAction The default action to perform if no menu selection is made. Valid values are Main, Get, Add, Remove. .PARAMETER NonInteractive Run in non-interactive mode. Requires DefaultAction to be specified. .EXAMPLE Show-OATHTokenMenu Displays the main menu and waits for user input. .EXAMPLE Show-OATHTokenMenu -DefaultAction Get Opens the Get OATH menu directly. .NOTES Requires Microsoft.Graph.Authentication module and appropriate permissions: - Policy.ReadWrite.AuthenticationMethod - Directory.Read.All #> function Show-OATHTokenMenu { [CmdletBinding()] param( [Parameter()] [ValidateSet('Main', 'Get', 'Add', 'Remove')] [string]$DefaultAction = 'Main', [Parameter()] [switch]$NonInteractive ) begin { # Initialize the skip processing flag at the start of each function call $script:skipProcessing = $false # Ensure we're connected to Graph if (-not (Test-MgConnection)) { # Set a flag to indicate we should skip processing $script:skipProcessing = $true # Return here only exits the begin block, not the function return } # Set console colors $script:headerColor = [System.ConsoleColor]::Cyan $script:promptColor = [System.ConsoleColor]::Yellow $script:successColor = [System.ConsoleColor]::Green $script:errorColor = [System.ConsoleColor]::Red } process { # Skip all processing if the connection check failed if ($script:skipProcessing) { return $null } # Main menu function function Show-MainMenu { Clear-Host Write-Host "===== OATH Token Management =====" -ForegroundColor $headerColor Write-Host "1) Get OATH" -ForegroundColor $promptColor Write-Host "2) Add OATH" -ForegroundColor $promptColor Write-Host "3) Remove OATH" -ForegroundColor $promptColor Write-Host "0) Exit" -ForegroundColor $promptColor Write-Host "" $choice = Read-Host "Enter your choice" switch ($choice) { "1" { Show-GetMenu } "2" { Show-AddMenu } "3" { Show-RemoveMenu } "0" { return } default { Write-Host "Invalid option. Please try again." -ForegroundColor $errorColor Start-Sleep -Seconds 2 Show-MainMenu } } } # Get OATH menu function Show-GetMenu { Clear-Host Write-Host "===== Get OATH Menu =====" -ForegroundColor $headerColor Write-Host "1) List All" -ForegroundColor $promptColor Write-Host "2) List Available" -ForegroundColor $promptColor Write-Host "3) List Activated" -ForegroundColor $promptColor Write-Host "4) Export to CSV" -ForegroundColor $promptColor Write-Host "5) Find by Serial Number" -ForegroundColor $promptColor Write-Host "6) Find by User ID/UPN" -ForegroundColor $promptColor Write-Host "0) Return to main menu" -ForegroundColor $promptColor Write-Host "" $choice = Read-Host "Enter your choice" switch ($choice) { "1" { Write-Host "Getting all tokens..." -ForegroundColor $headerColor $tokens = Get-OATHToken Write-Host "Found $($tokens.Count) tokens." -ForegroundColor $successColor $tokens | Format-Table -AutoSize Write-Host "Press any key to continue..." -ForegroundColor $promptColor [void][System.Console]::ReadKey($true) Show-GetMenu } "2" { Write-Host "Getting available tokens..." -ForegroundColor $headerColor $tokens = Get-OATHToken -AvailableOnly Write-Host "Found $($tokens.Count) available tokens." -ForegroundColor $successColor $tokens | Format-Table -AutoSize Write-Host "Press any key to continue..." -ForegroundColor $promptColor [void][System.Console]::ReadKey($true) Show-GetMenu } "3" { Write-Host "Getting activated tokens..." -ForegroundColor $headerColor $tokens = Get-OATHToken -ActivatedOnly Write-Host "Found $($tokens.Count) activated tokens." -ForegroundColor $successColor $tokens | Format-Table -AutoSize Write-Host "Press any key to continue..." -ForegroundColor $promptColor [void][System.Console]::ReadKey($true) Show-GetMenu } "4" { $csvPath = Read-Host "Enter CSV file path (press Enter for default)" if ([string]::IsNullOrWhiteSpace($csvPath)) { $timestamp = Get-Date -Format 'yyyyMMdd_HHmmss' $csvPath = Join-Path -Path $PWD -ChildPath "OATHTokens_$timestamp.csv" } Write-Host "Exporting tokens to: $csvPath" -ForegroundColor $headerColor $result = Export-OATHToken -FilePath $csvPath if ($result) { Write-Host "Tokens exported successfully." -ForegroundColor $successColor } else { Write-Host "Failed to export tokens." -ForegroundColor $errorColor } Write-Host "Press any key to continue..." -ForegroundColor $promptColor [void][System.Console]::ReadKey($true) Show-GetMenu } "5" { $serialNumber = Read-Host "Enter token serial number (can include wildcards)" Write-Host "Searching for tokens with serial number: $serialNumber" -ForegroundColor $headerColor $tokens = Get-OATHToken -SerialNumber $serialNumber Write-Host "Found $($tokens.Count) matching tokens." -ForegroundColor $successColor $tokens | Format-Table -AutoSize Write-Host "Press any key to continue..." -ForegroundColor $promptColor [void][System.Console]::ReadKey($true) Show-GetMenu } "6" { $userId = Read-Host "Enter user ID or UPN" Write-Host "Searching for tokens assigned to: $userId" -ForegroundColor $headerColor $tokens = Get-OATHToken -UserId $userId Write-Host "Found $($tokens.Count) tokens assigned to this user." -ForegroundColor $successColor $tokens | Format-Table -AutoSize Write-Host "Press any key to continue..." -ForegroundColor $promptColor [void][System.Console]::ReadKey($true) Show-GetMenu } "0" { Show-MainMenu } default { Write-Host "Invalid option. Please try again." -ForegroundColor $errorColor Start-Sleep -Seconds 2 Show-GetMenu } } } # Add OATH menu function Show-AddMenu { Clear-Host Write-Host "===== Add OATH Menu =====" -ForegroundColor $headerColor Write-Host "1) Add OATH Token" -ForegroundColor $promptColor Write-Host "2) Assign OATH User" -ForegroundColor $promptColor Write-Host "3) Activate OATH Token" -ForegroundColor $promptColor Write-Host "4) Bulk Import OATH Tokens" -ForegroundColor $promptColor Write-Host "5) Activate with TOTP" -ForegroundColor $promptColor Write-Host "0) Return to main menu" -ForegroundColor $promptColor Write-Host "" $choice = Read-Host "Enter your choice" switch ($choice) { "1" { # Add token with optional user assignment $serialNumber = Read-Host "Enter token serial number" $secretKey = Read-Host "Enter secret key" $secretFormat = Read-Host "Enter secret format (Base32, Hex, Text) or leave blank for Base32" $userId = Read-Host "Enter user ID or UPN (optional, leave blank for no assignment)" if ([string]::IsNullOrWhiteSpace($secretFormat)) { $secretFormat = "Base32" } Write-Host "Adding token with serial number: $serialNumber" -ForegroundColor $headerColor $params = @{ SerialNumber = $serialNumber SecretKey = $secretKey SecretFormat = $secretFormat } # Only include UserId if it was provided if (-not [string]::IsNullOrWhiteSpace($userId)) { $params['UserId'] = $userId Write-Host "Token will be assigned to: $userId" -ForegroundColor $headerColor } $result = Add-OATHToken @params if ($result) { if ([string]::IsNullOrWhiteSpace($userId)) { Write-Host "Token added successfully." -ForegroundColor $successColor } else { # Check if the token has the assignedTo property set if ($result.assignedTo -or $result.userId) { Write-Host "Token added and assigned successfully." -ForegroundColor $successColor } else { Write-Host "Token added but assignment may have failed." -ForegroundColor $errorColor } } } else { Write-Host "Failed to add token." -ForegroundColor $errorColor } Write-Host "Press any key to continue..." -ForegroundColor $promptColor [void][System.Console]::ReadKey($true) Show-AddMenu } "2" { # Assign existing token to user $tokenId = Read-Host "Enter token ID" $userId = Read-Host "Enter user ID or UPN" Write-Host "Assigning token to user..." -ForegroundColor $headerColor $result = Set-OATHTokenUser -TokenId $tokenId -UserId $userId if ($result) { Write-Host "Token assigned successfully." -ForegroundColor $successColor } else { Write-Host "Failed to assign token." -ForegroundColor $errorColor } Write-Host "Press any key to continue..." -ForegroundColor $promptColor [void][System.Console]::ReadKey($true) Show-AddMenu } "3" { # Activate token with verification code $tokenId = Read-Host "Enter token ID" $userId = Read-Host "Enter user ID or UPN" $verificationCode = Read-Host "Enter verification code" Write-Host "Activating token..." -ForegroundColor $headerColor $result = Set-OATHTokenActive -TokenId $tokenId -UserId $userId -VerificationCode $verificationCode if ($result) { Write-Host "Token activated successfully." -ForegroundColor $successColor } else { Write-Host "Failed to activate token." -ForegroundColor $errorColor } Write-Host "Press any key to continue..." -ForegroundColor $promptColor [void][System.Console]::ReadKey($true) Show-AddMenu } "4" { # Bulk import tokens (can include both unassigned and assigned tokens) $jsonPath = Read-Host "Enter JSON file path (press Enter for default tokens.json)" if ([string]::IsNullOrWhiteSpace($jsonPath)) { $jsonPath = Join-Path -Path $PWD -ChildPath "tokens.json" } if (-not (Test-Path -Path $jsonPath)) { Write-Host "File not found: $jsonPath" -ForegroundColor $errorColor } else { Write-Host "Importing tokens from: $jsonPath" -ForegroundColor $headerColor # Ask if tokens should be assigned to users if they have assignTo properties $assignToUsers = $true # Default is now true $assignPrompt = Read-Host "Skip user assignments? (Y/N, default: N)" if (-not [string]::IsNullOrWhiteSpace($assignPrompt) -and $assignPrompt.ToUpper() -eq 'Y') { $assignToUsers = $false } $result = Import-OATHToken -FilePath $jsonPath -Format JSON -AssignToUsers:$assignToUsers if ($result) { Write-Host "Tokens imported successfully." -ForegroundColor $successColor } else { Write-Host "Failed to import tokens." -ForegroundColor $errorColor } } Write-Host "Press any key to continue..." -ForegroundColor $promptColor [void][System.Console]::ReadKey($true) Show-AddMenu } "5" { # Activate token with TOTP $tokenId = Read-Host "Enter token ID" $userId = Read-Host "Enter user ID or UPN" $secret = Read-Host "Enter token secret" $secretFormat = Read-Host "Enter secret format (Base32, Hex, Text) or leave blank for Base32" if ([string]::IsNullOrWhiteSpace($secretFormat)) { $secretFormat = "Base32" } Write-Host "Activating token with generated TOTP code..." -ForegroundColor $headerColor $result = Set-OATHTokenActive -TokenId $tokenId -UserId $userId -Secret $secret -SecretFormat $secretFormat if ($result) { Write-Host "Token activated successfully with generated TOTP code." -ForegroundColor $successColor } else { Write-Host "Failed to activate token with TOTP." -ForegroundColor $errorColor } Write-Host "Press any key to continue..." -ForegroundColor $promptColor [void][System.Console]::ReadKey($true) Show-AddMenu } "0" { Show-MainMenu } default { Write-Host "Invalid option. Please try again." -ForegroundColor $errorColor Start-Sleep -Seconds 2 Show-AddMenu } } } # Remove OATH menu function Show-RemoveMenu { Clear-Host Write-Host "===== Remove OATH Menu =====" -ForegroundColor $headerColor Write-Host "1) Remove OATH" -ForegroundColor $promptColor Write-Host "2) Bulk Remove OATH" -ForegroundColor $promptColor Write-Host "3) Unassign OATH token" -ForegroundColor $promptColor Write-Host "0) Return to main menu" -ForegroundColor $promptColor Write-Host "" $choice = Read-Host "Enter your choice" switch ($choice) { "1" { # Remove single token $tokenIdentifier = Read-Host "Enter token ID or serial number" # Check if the input looks like a GUID if ($tokenIdentifier -match '^[0-9a-fA-F]{8}-[0-9a-fA-F]{4}-[0-9a-fA-F]{4}-[0-9a-fA-F]{4}-[0-9a-fA-F]{12}$') { Write-Host "Removing token by ID..." -ForegroundColor $headerColor $result = Remove-OATHToken -TokenId $tokenIdentifier } else { Write-Host "Removing token by serial number..." -ForegroundColor $headerColor $result = Remove-OATHToken -SerialNumber $tokenIdentifier } if ($result) { Write-Host "Token removed successfully." -ForegroundColor $successColor } else { Write-Host "Failed to remove token." -ForegroundColor $errorColor } Write-Host "Press any key to continue..." -ForegroundColor $promptColor [void][System.Console]::ReadKey($true) Show-RemoveMenu } "2" { # Bulk remove tokens $jsonPath = Read-Host "Enter JSON file path with token IDs to remove (press Enter for default tokens_remove.json)" if ([string]::IsNullOrWhiteSpace($jsonPath)) { $jsonPath = Join-Path -Path $PWD -ChildPath "tokens_remove.json" } if (-not (Test-Path -Path $jsonPath)) { Write-Host "File not found: $jsonPath" -ForegroundColor $errorColor } else { try { $jsonContent = Get-Content -Path $jsonPath -Raw | ConvertFrom-Json if (-not $jsonContent.remove -or $jsonContent.remove.Count -eq 0) { Write-Host "No token IDs found in the file. Expected 'remove' array property." -ForegroundColor $errorColor } else { $tokenIds = $jsonContent.remove Write-Host "Removing $($tokenIds.Count) tokens..." -ForegroundColor $headerColor $removedCount = 0 $failedCount = 0 foreach ($tokenId in $tokenIds) { $result = Remove-OATHToken -TokenId $tokenId -Force if ($result) { $removedCount++ } else { $failedCount++ } } Write-Host "Bulk removal completed: $removedCount removed, $failedCount failed." -ForegroundColor $successColor } } catch { Write-Host "Error processing JSON file: $_" -ForegroundColor $errorColor } } Write-Host "Press any key to continue..." -ForegroundColor $promptColor [void][System.Console]::ReadKey($true) Show-RemoveMenu } "3" { # Unassign token $tokenId = Read-Host "Enter token ID to unassign" Write-Host "Unassigning token..." -ForegroundColor $headerColor $result = Set-OATHTokenUser -TokenId $tokenId -Unassign if ($result) { Write-Host "Token unassigned successfully." -ForegroundColor $successColor } else { Write-Host "Failed to unassign token." -ForegroundColor $errorColor } Write-Host "Press any key to continue..." -ForegroundColor $promptColor [void][System.Console]::ReadKey($true) Show-RemoveMenu } "0" { Show-MainMenu } default { Write-Host "Invalid option. Please try again." -ForegroundColor $errorColor Start-Sleep -Seconds 2 Show-RemoveMenu } } } # Start with the specified menu or the main menu by default switch ($DefaultAction) { 'Get' { Show-GetMenu } 'Add' { Show-AddMenu } 'Remove' { Show-RemoveMenu } default { Show-MainMenu } } } } # Add alias for backward compatibility - only if it doesn't already exist if (-not (Get-Alias -Name 'Show-HardwareOathTokenMenu' -ErrorAction SilentlyContinue)) { New-Alias -Name 'Show-HardwareOathTokenMenu' -Value 'Show-OATHTokenMenu' } |