Public/Unprotect-Document.ps1
function Unprotect-Document { # .SYNOPSIS # Decrypts data or file encrypted with PsModuleBase for the current user. (Wrapper for [PsModuleBase]::UnprotectDataset) [Diagnostics.CodeAnalysis.SuppressMessageAttribute("PSAvoidUsingWriteHost", "")] # Write-Host is used by called method for feedback [CmdletBinding()] Param ( [Parameter(Mandatory = $true, ParameterSetName = 'File', ValueFromPipeline = $true, ValueFromPipelineByPropertyName = $true)] [Alias('FullName')] [string[]]$Path, [Parameter(Mandatory = $true, ParameterSetName = 'Content', ValueFromPipelineByPropertyName = $true)] [string]$Content, [ValidateScript({ # Allow OutPath not existing if Type=Content, but require it to be a dir if it exists # If Type=File, it MUST exist and be a dir. This validation is better done inside the process block. if (Test-Path $_) { Test-Path $_ -PathType Container } else { $true } })] [string]$OutPath ) process { if ($PSCmdlet.ParameterSetName -eq 'Content') { Write-Verbose "Attempting to unprotect content..." # Validate OutPath if provided for content if ($OutPath -and !(Test-Path -LiteralPath $OutPath -PathType Container)) { $PSCmdlet.WriteError( (New-ErrorRecord -Message "Output directory specified by -OutPath does not exist: '$OutPath'" -ErrorId 'OutPathNotFoundContent' -Category ObjectNotFound -TargetObject $OutPath) ) return # Stop processing this content item } try { # Pass $PSCmdlet for error writing context $decryptedContentOrPath = [PsModuleBase]::UnprotectDataset($Content, $OutPath, $PSCmdlet) # Output depends on whether OutPath was used for content if ($OutPath) { # If OutPath was used, method returns null on success (file written) or error # If it returns a path, it means it wrote the file if ($null -eq $decryptedContentOrPath) { # Check if an error was written by the method, otherwise assume success but no output needed } else { # Should not happen if OutPath was specified for content, but handle defensively # Write-Output $decryptedContentOrPath } } elseif ($null -ne $decryptedContentOrPath) { # No OutPath, method returns decrypted string Write-Output $decryptedContentOrPath } # Errors are written by UnprotectDataset via $PSCmdlet.WriteError } catch { # Catch unexpected errors in the wrapper itself $PSCmdlet.WriteError($_) } } elseif ($PSCmdlet.ParameterSetName -eq 'File') { # Use Resolve-PathEx carefully or replace with simpler Resolve-Path if sufficient foreach ($fileInfo in Resolve-PathEx -Path $Path -Type File -Mode AnyWarning -Provider FileSystem) { if ($fileInfo.Success) { foreach ($resolvedPath in $fileInfo.Path) { # Determine root output directory $outputDirectory = $OutPath # Use explicit OutPath if provided if (!$outputDirectory) { # Default to same directory as input file $outputDirectory = Split-Path -Path $resolvedPath -Parent } # Ensure output directory exists if (!(Test-Path -LiteralPath $outputDirectory -PathType Container)) { $PSCmdlet.WriteError( (New-ErrorRecord -Message "Output directory does not exist: '$outputDirectory' for input file '$resolvedPath'" -ErrorId 'OutPathNotFoundFile' -Category ObjectNotFound -TargetObject $outputDirectory) ) continue # Skip this file } Write-Verbose "Unprotecting file: $resolvedPath to directory $outputDirectory" try { $jsonFileContent = [System.IO.File]::ReadAllText($resolvedPath) # Pass $PSCmdlet for error writing context # UnprotectDataset will handle writing the file and Write-Host output $unprotectedFilePath = [PsModuleBase]::UnprotectDataset($jsonFileContent, $outputDirectory, $PSCmdlet) # Optionally output the path of the created file if ($unprotectedFilePath) { Write-Output $unprotectedFilePath } # Errors are written by UnprotectDataset via $PSCmdlet.WriteError } catch { # Catch errors from file reading or unexpected wrapper errors $PSCmdlet.WriteError($_) } } } else { $PSCmdlet.WriteWarning("Could not resolve or access path: $($fileInfo.Input). Message: $($fileInfo.Message)") } } } } } |