Plugins/Porkbun.ps1
function Get-CurrentPluginType { 'dns-01' } function Add-DnsTxt { [CmdletBinding()] param ( [Parameter(Mandatory, Position = 0)] [string] $RecordName, [Parameter(Mandatory, Position = 1)] [string] $TxtValue, [Parameter(Mandatory, Position = 2)] [SecureString] $PorkbunAPIKey, [Parameter(Mandatory, Position = 3)] [SecureString] $PorkbunSecret, [string] $PorkbunAPIHost = 'api.porkbun.com', [Parameter(ValueFromRemainingArguments)] $ExtraParams ) [string] $APIKey = [PSCredential]::new('Username', $PorkbunAPIKey).GetNetworkCredential().Password [string] $APISecret = [PSCredential]::new('Username', $PorkbunSecret).GetNetworkCredential().Password $domainQuery = @{ LongName = $RecordName PorkbunAPIKey = $APIKey PorkbunSecret = $APISecret PorkbunAPIHost = $PorkbunAPIHost } $DomainInfo = Get-PorkbunDomainInfo @domainQuery # Get the portion of the full name that is the domain name (e.g. 'record.name.sub.example.com' will become 'example.com') [string] $DomainName = $DomainInfo.Domain # Get the portion of the full name that will become the record name (e.g. 'record.name.sub.example.com' will become 'record.name.sub') [string] $RecordNameShort = $RecordName -ireplace "\.?$([regex]::Escape($DomainName.TrimEnd('.')))$",'' # Get any existing TXT record(s) that already match what we want to create [object[]] $EqualRecords = @($DomainInfo.Records | Where-Object { ($_.type -EQ 'TXT') -AND ($_.name -eq $RecordName) -AND ($_.content -EQ $TxtValue) }) if ($EqualRecords.Count -EQ 0) { Write-Debug 'This record does not exist yet, creating it' Write-Verbose "Creating record `"$RecordNameShort`" with value `"$TxtValue`" on domain `"$DomainName`"" $queryParams = @{ Uri = "https://$PorkbunAPIHost/api/json/v3/dns/create/$DomainName" Method = 'POST' BodyObject = @{ name = $RecordNameShort type = 'TXT' content = $TxtValue apikey = $APIKey secretapikey = $APISecret } } try { $ResultData = Invoke-Porkbun @queryParams if ($ResultData.status -NE 'SUCCESS') { throw "API returned result $($ResultData.status)" } Write-Debug 'Successfully created record.' } catch { throw } } else { Write-Debug 'A record already exists with this value, so no creation is necessary' return } <# .SYNOPSIS Add a DNS TXT record to Porkbun. .DESCRIPTION Adds or edits a DNS TXT record using Porkbun's API. .PARAMETER RecordName The fully qualified name of the TXT record. .PARAMETER TxtValue The value of the TXT record. .PARAMETER PorkbunAPIKey The API key to use, obtained from https://porkbun.com/account/api .PARAMETER PorkbunSecret The API secret key corresponding to the API key, also obtained from https://porkbun.com/account/api (not accessible after being generated) .PARAMETER ExtraParams This parameter can be ignored and is only used to prevent errors when splatting with more parameters than this function supports. .EXAMPLE Add-DnsTxt '_acme-challenge.example.com' 'txt-value' Adds a TXT record for the specified site with the specified value. Does nothing if the record already exists. #> } function Remove-DnsTxt { [CmdletBinding()] param ( [Parameter(Mandatory, Position = 0)] [string] $RecordName, [Parameter(Mandatory, Position = 1)] [string] $TxtValue, [Parameter(Mandatory, Position = 2)] [SecureString] $PorkbunAPIKey, [Parameter(Mandatory, Position = 3)] [SecureString] $PorkbunSecret, [string] $PorkbunAPIHost = 'api.porkbun.com', [Parameter(ValueFromRemainingArguments)] $ExtraParams ) [string] $APIKey = [PSCredential]::new('Username', $PorkbunAPIKey).GetNetworkCredential().Password [string] $APISecret = [PSCredential]::new('Username', $PorkbunSecret).GetNetworkCredential().Password $domainQuery = @{ LongName = $RecordName PorkbunAPIKey = $APIKey PorkbunSecret = $APISecret PorkbunAPIHost = $PorkbunAPIHost } $DomainInfo = Get-PorkbunDomainInfo @domainQuery # Get the portion of the full name that is the domain name (e.g. 'record.name.sub.example.com' will become 'example.com') [string] $DomainName = $DomainInfo.Domain # Get any existing TXT record(s) that have matching content [object[]] $EqualRecords = @($DomainInfo.Records | Where-Object { ($_.type -EQ 'TXT') -AND ($_.name -eq $RecordName) -AND ($_.content -EQ $TxtValue) }) if ($EqualRecords.Count -EQ 0) { Write-Debug 'There are no records with this content, so no deletion is necessary' return } else { Write-Debug "Found $($EqualRecords.Count) record(s) to delete." foreach($RecordToDelete in $EqualRecords) { $RecordID = $RecordToDelete.id Write-Verbose "Deleting record ID `"$RecordID`" on domain `"$DomainName`"" $queryParams = @{ Uri = "https://$PorkbunAPIHost/api/json/v3/dns/delete/$DomainName/$RecordID" Method = 'POST' BodyObject = @{ apikey = $APIKey secretapikey = $APISecret } } try { $ResultData = Invoke-Porkbun @queryParams if ($ResultData.status -NE 'SUCCESS') { throw "API returned result $($ResultData.status)" } Write-Debug 'Successfully deleted record.' } catch { throw } } } <# .SYNOPSIS Remove a DNS TXT record from Porkbun. .DESCRIPTION Removes a DNS TXT record using Porkbun's API. .PARAMETER RecordName The fully qualified name of the TXT record. .PARAMETER TxtValue The value of the TXT record. .PARAMETER PorkbunAPIKey The API key to use, obtained from https://porkbun.com/account/api .PARAMETER PorkbunSecret The API secret key corresponding to the API key, also obtained from https://porkbun.com/account/api (not accessible after being generated) .EXAMPLE Remove-DnsTxt '_acme-challenge.example.com' 'txt-value' Removes a TXT record for the specified site with the specified value. Does nothing if the record does not exist. #> } function Save-DnsTxt { [CmdletBinding()] param ( [Parameter(ValueFromRemainingArguments)] $ExtraParams ) <# .SYNOPSIS Not required. .DESCRIPTION This provider does not require calling this function to commit changes to DNS records. .PARAMETER ExtraParams This parameter can be ignored and is only used to prevent errors when splatting with more parameters than this function supports. .EXAMPLE Save-DnsTxt Commits changes for pending DNS TXT record modifications. (Not required) #> } ################################## # Helper Functions ################################## # API Documentation: https://porkbun.com/api/json/v3/documentation function Invoke-Porkbun { [CmdletBinding()] param( [Parameter(Mandatory)] [string]$Uri, [string]$Method, [hashtable]$BodyObject ) # Porkbun seems to have some sort of rate limiting that results in HTTP 503 errors # when you hit it (or maybe it's just overloaded?). But there's no documentation # about it and there are no Retry-After headers in the response. So we're just # going to implement a dumb retry mechanic. $queryParams = @{ Uri = $Uri Method = $Method Body = ($BodyObject | ConvertTo-Json) ErrorAction = 'Stop' Verbose = $false } # sanitize credentials for logging $BodyObject.apikey = 'REDACTED' $BodyObject.secretapikey = 'REDACTED' for ($i=0; $i -lt 3; $i++) { try { Write-Debug "POST $($queryParams.Uri)`n$($BodyObject|ConvertTo-Json)" $ResultData = Invoke-RestMethod @queryParams @script:UseBasic return $ResultData } catch { if ($_.Exception.Response.StatusCode -eq 503) { Write-Debug "API Temporarily Unavailable. Waiting and Re-trying." Start-Sleep -Seconds 2 continue } else { throw } } } throw "API responding as Temporarily Unavailable. Gave up retrying." } function Get-PorkbunDomainInfo { [CmdletBinding()] param ( [Parameter(Mandatory)] [string] $LongName, [Parameter(Mandatory)] [string] $PorkbunAPIKey, [Parameter(Mandatory)] [string] $PorkbunSecret, [string] $PorkbunAPIHost = 'api.porkbun.com' ) Write-Debug "Looking for domain `"$LongName`"" [string[]] $Sections = $LongName.Split('.') [int] $MaxIndex = $Sections.Count - 1 for ([int]$i = 0; $i -lt $MaxIndex; $i++) { [string] $NameToCheck = [string]::Join('.', $Sections[$i .. $MaxIndex]) Write-Debug "Querying API for `"$NameToCheck`"" try { $queryParams = @{ Uri = "https://$PorkbunAPIHost/api/json/v3/dns/retrieve/$NameToCheck" Method = 'POST' BodyObject = @{ apikey = $PorkbunAPIKey secretapikey = $PorkbunSecret } } $ResultData = Invoke-Porkbun @queryParams } catch { if ($_.Exception.Response.StatusCode -eq 400) { Write-Debug "Could not find domain `"$NameToCheck`"." continue } else { Write-Debug "Something went wrong while checking domain `"$NameToCheck`"" throw } } if ($ResultData.status -NE 'SUCCESS') { Write-Debug "API returned status $($ResultData.status) for domain `"$NameToCheck`"" continue } Write-Debug "Found domain `"$NameToCheck`"" $domainInfo = [pscustomobject]@{ Domain = $NameToCheck Records = $ResultData.records } return $domainInfo } throw "No matching domain could be found for `"$LongName`" on this Porkbun account. Check that the domain is correct, that your API key and secret are entered correctly, and that you've enabled API access for this domain in the settings." <# .SYNOPSIS Finds the domain and existing records. .DESCRIPTION Uses the Porkbun API to find the relevant domain and existing records for this full name. .PARAMETER LongName The combined record/domain name to query for. .PARAMETER PorkbunAPIKey The API key to use, obtained from https://porkbun.com/account/api .PARAMETER PorkbunSecret The API secret key corresponding to the API key, also obtained from https://porkbun.com/account/api (not accessible after being generated) .OUTPUTS An object containing a 'Domain' property, which contains the base domain name, and a 'Records' property which contains all currently existing records for this domain, including ones for other subdomains. .EXAMPLE Get-PorkbunDomainInfo -PorkbunAPIKey (key) -PorkbunAPISecret (secret) -LongName 'long.name.for.example.com' Will return a Domain of 'example.com' and any records for that entire domain. #> } |