Skip to content

Instantly share code, notes, and snippets.

@emilwojcik93
Last active February 11, 2025 16:26
Show Gist options
  • Save emilwojcik93/aa5f9b7f6a28608896d43c5d77c3575d to your computer and use it in GitHub Desktop.
Save emilwojcik93/aa5f9b7f6a28608896d43c5d77c3575d to your computer and use it in GitHub Desktop.
This script searches for a certificate by thumbprint or issuer, exports it to a Base-64 encoded .crt file, and optionally installs it in WSL.
<#
.SYNOPSIS
This script searches for certificates by thumbprint or issuer, exports them to Base-64 encoded .crt files, and optionally installs them in WSL.
.DESCRIPTION
The script searches for certificates with the specified thumbprints or issuers. If found, it exports the certificates to Base-64 encoded .crt files. The script can also install the certificates in WSL if the --install parameter is provided.
.PARAMETER Thumbprint
The thumbprints of the certificates to search for. Default is an array of predefined thumbprints.
.PARAMETER Issuer
The issuers of the certificates to search for. Default is an array of predefined issuers.
.PARAMETER ExportPath
The path where the certificates will be exported. Default is the current working directory.
.PARAMETER Install
If provided, the script will install the certificates in WSL. Default is $true.
.PARAMETER ExportOnly
If provided, the script will only export the certificates to files and not install them in WSL.
.PARAMETER Verbose
Enables verbose output.
.EXAMPLE
.\Manage-Certificates.ps1 -Verbose
.EXAMPLE
.\Manage-Certificates.ps1 -Thumbprint "DSA1235SAGWE451256121GF113WSA124" -ExportPath "C:\path\to\output.crt" -Verbose
.EXAMPLE
.\Manage-Certificates.ps1 -Issuer "CN=Example Root CA" -ExportPath "C:\path\to\output.crt" -Verbose
.EXAMPLE
.\Manage-Certificates.ps1 -Install -Verbose
.EXAMPLE
.\Manage-Certificates.ps1 -ExportOnly -Verbose
#>
param (
[string[]]$Thumbprint,
[string[]]$Issuer,
[string]$ExportPath = "$env:Temp\certs",
[switch]$Install,
[switch]$ExportOnly,
[switch]$Verbose
)
if ($Verbose) {
$VerbosePreference = "Continue"
}
function Get-AllCertificates {
$certificateType = [System.Security.Cryptography.X509Certificates.X509Certificate2]
$includedStores = @("TrustedPublisher", "Root", "CA", "AuthRoot")
$certificates = $includedStores.ForEach({
Get-ChildItem Cert:\CurrentUser\$_ | Where-Object { $_ -is $certificateType }
})
return $certificates
}
function Get-CertificateByThumbprint {
param (
[string]$Thumbprint
)
Write-Verbose "Looking for certificate with thumbprint: $Thumbprint"
$cert = Get-ChildItem -Path Cert:\LocalMachine\Root | Where-Object { $_.Thumbprint -eq $Thumbprint }
return $cert
}
function Get-CertificateByIssuer {
param (
[string]$Issuer
)
Write-Verbose "Looking for certificate with issuer: $Issuer"
$cert = Get-ChildItem -Path Cert:\LocalMachine\Root | Where-Object { $_.Issuer -like "*$Issuer*" }
return $cert
}
function Export-CertificateToFile {
param (
[System.Security.Cryptography.X509Certificates.X509Certificate2]$Cert,
[string]$FilePath
)
Write-Verbose "Exporting certificate to file: $FilePath"
$tempCertPath = [System.IO.Path]::GetTempFileName()
Export-Certificate -Cert $Cert -FilePath $tempCertPath -Type CERT
# Convert the certificate to Base-64 encoded X.509 format
$certBytes = [System.IO.File]::ReadAllBytes($tempCertPath)
$base64CertContent = [System.Convert]::ToBase64String($certBytes)
# Insert line breaks every 64 characters
$formattedCertContent = ($base64CertContent -split "(.{64})" | Where-Object { $_ -ne "" }) -join "`n"
$base64CertHeader = "-----BEGIN CERTIFICATE-----"
$base64CertFooter = "-----END CERTIFICATE-----"
$base64Cert = $base64CertHeader + "`n" + $formattedCertContent + "`n" + $base64CertFooter
Set-Content -Path $FilePath -Value $base64Cert -ErrorAction Stop
Remove-Item -Force $tempCertPath -ErrorAction Stop
}
function Check-ExportPath {
param (
[string]$FilePath
)
$directory = [System.IO.Path]::GetDirectoryName($FilePath)
if (-not (Test-Path -Path $directory)) {
try {
New-Item -ItemType Directory -Path $directory -Force | Out-Null
} catch {
throw "The directory '$directory' cannot be created."
}
}
if (-not (Test-Path -Path $FilePath -PathType Leaf)) {
try {
New-Item -ItemType File -Path $FilePath -Force | Out-Null
} catch {
throw "The file '$FilePath' cannot be created."
}
}
if (-not (Test-Path -Path $FilePath -PathType Leaf)) {
throw "The file '$FilePath' is not writable."
}
}
function Install-CertificatesInWSL {
param (
[string[]]$CertFileNames,
[string]$ExportPath
)
# Get the default WSL distribution from the registry
$defaultDistroGuid = Get-ItemProperty -Path "HKCU:\Software\Microsoft\Windows\CurrentVersion\Lxss" | Select-Object -ExpandProperty DefaultDistribution
$defaultDistroKey = "HKCU:\Software\Microsoft\Windows\CurrentVersion\Lxss\$defaultDistroGuid"
$defaultDistro = Get-ItemProperty -Path $defaultDistroKey | Select-Object -ExpandProperty DistributionName
# Output the default distribution
Write-Output "Default WSL Distribution: $defaultDistro"
# Define the WSL directory path for custom CA certificates
$wslCertsPath = "\\wsl$\$defaultDistro\usr\local\share\ca-certificates"
# Copy the certificates to $wslCertsPath
foreach ($certFileName in $CertFileNames) {
$destinationCertPath = Join-Path -Path $wslCertsPath -ChildPath $CertFileName
Write-Verbose "Copying certificate to WSL path: $destinationCertPath"
wsl --cd "${ExportPath}" -d $defaultDistro -u root -e bash -c "cp ${certFileName} /usr/local/share/ca-certificates"
if (Test-Path -Path "$destinationCertPath") {
Write-Output "Certificate copied to $destinationCertPath"
} else {
Write-Output "Failed to copy certificate to $destinationCertPath"
exit 1
}
}
# Run the WSL command and capture the output
$output = wsl --cd "$PSScriptRoot" -d $defaultDistro -u root -e bash -c 'if [[ $(command -v update-ca-certificates) ]]; then echo true; else echo false; fi;'
# Check the output and display the appropriate message
if ($output -eq "true") {
Write-Output "Success: update-ca-certificates is available in WSL."
# Install the CA certificates using update-ca-certificates
wsl --cd "$PSScriptRoot" -d $defaultDistro -u root -e bash -c "update-ca-certificates"
Write-Output "CA certificates installed using update-ca-certificates."
} else {
Write-Output "Failed: update-ca-certificates is not available in WSL."
# Append the CA certificates directly to ca-certificates.crt
foreach ($certFileName in $CertFileNames) {
wsl --cd "$PSScriptRoot" -d $defaultDistro -u root -e bash -c "cat /usr/local/share/ca-certificates/$certFileName | sudo tee -a /etc/ssl/certs/ca-certificates.crt > /dev/null"
Write-Output "CA certificate $certFileName appended directly to /etc/ssl/certs/ca-certificates.crt."
}
}
}
function Main {
param (
[string[]]$Thumbprint,
[string[]]$Issuer,
[string]$ExportPath = "$env:Temp\certs",
[switch]$Install,
[switch]$ExportOnly,
[switch]$Verbose
)
$certificates = @()
if ($Thumbprint -or $Issuer) {
foreach ($thumbprint in $Thumbprint) {
$cert = Get-CertificateByThumbprint -Thumbprint $thumbprint
if ($cert) {
$certificates += $cert
} else {
Write-Verbose "Certificate with thumbprint $thumbprint not found."
}
}
foreach ($issuer in $Issuer) {
$cert = Get-CertificateByIssuer -Issuer $issuer
if ($cert) {
$certificates += $cert
} else {
Write-Verbose "Certificate with issuer $issuer not found."
}
}
} else {
$certificates = Get-AllCertificates
}
$uniqueCertificates = $certificates | Sort-Object Thumbprint -Unique
$certFileNames = @()
foreach ($cert in $uniqueCertificates) {
Write-Verbose "Certificate details: $($cert | Format-List | Out-String)"
$certName = ($cert.Subject -match 'CN=([^,]+)') | Out-Null; $certName = $matches[1] -replace '[^\w\s]', '' -replace '\s', '_'
$certFileName = "${certName}.crt"
Write-Verbose "Extracted certificate name: $certName"
Write-Verbose "Certificate file name: $certFileName"
$sourceCertPath = Join-Path -Path $ExportPath -ChildPath $certFileName
try {
Check-ExportPath -FilePath $sourceCertPath
} catch {
Write-Verbose "The specified path '$sourceCertPath' is not writable. Using default path in the user's home directory."
$sourceCertPath = "${env:UserProfile}\${certFileName}"
Check-ExportPath -FilePath $sourceCertPath
}
Export-CertificateToFile -Cert $cert -FilePath $sourceCertPath
Write-Output "Certificate exported to $sourceCertPath"
$certFileNames += $certFileName
}
if (-not $ExportOnly) {
Install-CertificatesInWSL -CertFileNames $certFileNames -ExportPath $ExportPath
# Remove the temporary directory for certs
if (Test-Path -Path $ExportPath) {
Remove-Item -Recurse -Force -Path $ExportPath
Write-Output "Temporary directory for certs removed: $ExportPath"
}
}
}
# Call the main function with parameters
Main -Thumbprint $Thumbprint -Issuer $Issuer -ExportPath $ExportPath -Install:$Install -ExportOnly:$ExportOnly -Verbose:$Verbose
@emilwojcik93
Copy link
Author

Manage-Certificates.ps1

This PowerShell script searches for a certificate by thumbprint or issuer, exports it to a Base-64 encoded .crt file, and optionally installs it in WSL.

Usage

.\Manage-Certificates.ps1 [options]

Options

  • -Thumbprint: The thumbprint of the certificate to search for.
  • -Issuer: The issuer of the certificate to search for.
  • -ExportPath: The path where the certificate will be exported. Default is the current working directory.
  • -CertFileName: The name of the exported certificate file. Default is "custom_ssl_ca.crt".
  • -Install: If provided, the script will install the certificate in WSL.
  • -ExportOnly: If provided, the script will only export the certificate to a file and not install it in WSL.
  • -Verbose: Enables verbose output.

Running the Script from the Internet:

Use Invoke-RestMethod to download and execute the script. Here is how you can do it:

# Using Invoke-RestMethod
irm https://gist.githubusercontent.com/emilwojcik93/aa5f9b7f6a28608896d43c5d77c3575d/raw | iex

Note

If it doesn't work, then try to Set-ExecutionPolicy via PowerShell (Admin)

Set-ExecutionPolicy -ExecutionPolicy Bypass -Scope Process -Force; irm https://gist.githubusercontent.com/emilwojcik93/aa5f9b7f6a28608896d43c5d77c3575d/raw | iex

Note

To execute the script from the Internet with additional parameters, please run

&([ScriptBlock]::Create((irm https://gist.githubusercontent.com/emilwojcik93/aa5f9b7f6a28608896d43c5d77c3575d/raw))) -Thumbprint "THUMBPRINT" -Issuer "ISSUER" -Verbose

Example of execution

PS > &([ScriptBlock]::Create((irm https://gist.githubusercontent.com/emilwojcik93/aa5f9b7f6a28608896d43c5d77c3575d/raw))) -Thumbprint "THUMBPRINT_VALUE" -Verbose
VERBOSE: Looking for certificate with thumbprint: THUMBPRINT_VALUE
VERBOSE: Exporting certificate to file: C:\Users\UserName\Downloads\custom_ssl_ca.crt

    Directory: C:\Users\UserName\AppData\Local\Temp

Mode                 LastWriteTime         Length Name
----                 -------------         ------ ----
-a---            2/5/2025 10:14 AM           1500 tmpnn0udc.tmp
Certificate exported to C:\Users\UserName\Downloads\custom_ssl_ca.crt
Default WSL Distribution: Ubuntu
Certificate copied to \\wsl$\Ubuntu\usr\local\share\ca-certificates\custom_ssl_ca.crt
Success: update-ca-certificates is available in WSL.
Updating certificates in /etc/ssl/certs...
rehash: warning: skipping ca-certificates.crt,it does not contain exactly one certificate or CRL
1 added, 0 removed; done.
Running hooks in /etc/ca-certificates/update.d...
done.
CA certificate installed using update-ca-certificates.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment