Plugins/TotalUptime.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)]
        [pscredential]$TotalUptimeCredential,
        [string]$TotalUptimeApiRoot = 'https://api.totaluptime.com',
        [Parameter(ValueFromRemainingArguments)]
        $ExtraParams
    )

    #Find apex domain for record
    $DomainMetadata = Find-TotalUptimeDomain -RecordName $RecordName -TotalUptimeCredential $TotalUptimeCredential -TotalUptimeApiRoot $TotalUptimeApiRoot
    if (-not $DomainMetadata) {
        throw "Unable to find TotalUptime DNS Zone for $RecordName"
    }
    Write-Debug "Found Domain ID $($DomainMetadata.id) for record $RecordName"

    #strip domain component from record
    $recShort = $RecordName -ireplace "\.?$([regex]::Escape($DomainMetadata.domainName.TrimEnd('.')))$",''
    if ($recShort -eq [string]::Empty) {
        $recShort = '@'
    }
    Write-Debug "Stripped record name to $recShort"

    $txtRecord = Find-TotalUptimeTXTRecord -DomainID $DomainMetadata.id -RecordName $recShort -TxtContent $TxtValue -TotalUptimeCredential $TotalUptimeCredential -TotalUptimeApiRoot $TotalUptimeApiRoot
    if ($txtRecord) {
        Write-Debug 'Record already exists, nothing to do'
        return
    }

    #Build Request
    $reqParams = @{
        Uri = "$TotalUptimeApiRoot/CloudDNS/Domain/$($DomainMetadata.id)/TXTRecord"
        Method = 'Post'
        Headers = @{
            Authorization  = ("Basic {0}" -f (ConvertTo-TotalUptimeBasicHTTPAuthString -Credential $TotalUptimeCredential));
            Accept         = "application/json"
            'Content-Type' = "application/x-www-form-urlencoded"
        }
        Body = @{
            txtHostName = $recShort;
            txtText     = $TxtValue;
            txtTTL      = 60;
        } | ConvertTo-Json -Compress
        Verbose = $false
        ErrorAction = 'Stop'
    }
    Write-Debug "POST $($reqParams.Uri)`n$($reqParams.Body)"
    $Response = Invoke-RestMethod @reqParams @script:UseBasic

    #failure throw error
    if ($Response.status -ne 'Success') {
        throw $Response.message
    } else {
        Write-Debug $Response.message
    }

    <#
    .SYNOPSIS
        Add a DNS TXT record to TotalUptime

    .DESCRIPTION
        Add a DNS TXT record to TotalUptime

    .PARAMETER RecordName
        The fully qualified name of the TXT record.

    .PARAMETER TxtValue
        The value of the TXT record.

    .PARAMETER TotalUptimeCredential
        The API username and password required to authenticate.

    .PARAMETER TotalUptimeApiRoute
        The API root URL. defaults to https://api.totaluptime.com

    .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' -TotalUptimeCredential (Get-Credential)

        Adds a TXT record for the specified site with the specified value.
    #>

}

function Remove-DnsTxt {
    [CmdletBinding()]
    param(
        [Parameter(Mandatory, Position = 0)]
        [string]$RecordName,
        [Parameter(Mandatory, Position = 1)]
        [string]$TxtValue,
        [Parameter(Mandatory)]
        [pscredential]$TotalUptimeCredential,
        [string]$TotalUptimeApiRoot = 'https://api.totaluptime.com',
        [Parameter(ValueFromRemainingArguments)]
        $ExtraParams
    )

    #Find apex domain for record
    $DomainMetadata = Find-TotalUptimeDomain -RecordName $RecordName -TotalUptimeCredential $TotalUptimeCredential -TotalUptimeApiRoot $TotalUptimeApiRoot
    if (-not $DomainMetadata) {
        throw "Unable to find TotalUptime DNS Zone for $RecordName"
    }

    #Format recordname as short form
    $recShort = $RecordName -ireplace "\.?$([regex]::Escape($DomainMetadata.domainName.TrimEnd('.')))$",''
    if ($recShort -eq [string]::Empty) {
        $recShort = '@'
    }

    $txtRecord = Find-TotalUptimeTXTRecord -DomainID $DomainMetadata.id -RecordName $recShort -TxtContent $TxtValue -TotalUptimeCredential $TotalUptimeCredential -TotalUptimeApiRoot $TotalUptimeApiRoot
    if (-not $txtRecord) {
        Write-Debug 'No record found, nothing to do'
        return
    }

    #Build Request
    $reqParams = @{
        Uri = "$TotalUptimeApiRoot/CloudDNS/Domain/$($DomainMetadata.id)/TXTRecord/$($txtRecord.id)"
        Method = 'Delete'
        Headers = @{
            Authorization = 'Basic {0}' -f (ConvertTo-TotalUptimeBasicHTTPAuthString -Credential $TotalUptimeCredential)
            Accept        = 'application/json'
        }
        Verbose = $false
        ErrorAction = 'Stop'
    }

    Write-Debug "DELETE $($reqParams.Uri)"
    $Response = Invoke-RestMethod @reqParams @script:UseBasic

    # throw error on failure
    if ($Response.status -ne 'Success') {
        throw $Response.message
    } else {
        Write-Debug $Response.message
    }


    <#
    .SYNOPSIS
        Remove a DNS TXT record from TotalUptime

    .DESCRIPTION
        Remove a DNS TXT record from TotalUptime

    .PARAMETER RecordName
        The fully qualified name of the TXT record.

    .PARAMETER TxtValue
        The value of the TXT record.

    .PARAMETER TotalUptimeCredential
        The API username and password required to authenticate.

    .PARAMETER TotalUptimeApiRoute
        The API root URL. defaults to https://api.totaluptime.com

    .PARAMETER ExtraParams
        This parameter can be ignored and is only used to prevent errors when splatting with more parameters than this function supports.

    .EXAMPLE
        Remove-DnsTxt '_acme-challenge.example.com' 'txt-value' -TotalUptimeCredential (Get-Credential)

        Removes a TXT record for the specified site with the specified value.
    #>

}

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.
    #>

}

############################
# Helper Functions
############################

# API Documentation
# https://totaluptime.com/api/v2/

# Auth function
function ConvertTo-TotalUptimeBasicHTTPAuthString {
    [CmdletBinding()]
    param(
        [Parameter(Mandatory = $true)]
        [PSCredential]$Credential
    )
    #Converts a PSCredential Object to a HTTP Basic Auth String
    $base64AuthString = [Convert]::ToBase64String([Text.Encoding]::ASCII.GetBytes(("{0}:{1}" -f $Credential.GetNetworkCredential().UserName, $Credential.GetNetworkCredential().Password)))
    return $base64AuthString
}

# Search Functions

function Find-TotalUptimeDomain {
    [CmdletBinding()]
    param (
        [Parameter(Mandatory, Position = 0)]
        [string]$RecordName,
        [Parameter(Mandatory)]
        [pscredential]$TotalUptimeCredential,
        [string]$TotalUptimeApiRoot = 'https://api.totaluptime.com'
    )

    #retrieve domains in account
    $Domains = Get-TotalUptimeDomains -TotalUptimeCredential $TotalUptimeCredential -TotalUptimeApiRoot $TotalUptimeApiRoot -EA Stop
    $DomainNames = $Domains.rows.domainName

    #check if domain is in domain list
    $pieces = $RecordName.Split('.')
    for ($i = 0; $i -lt ($pieces.Count - 1); $i++) {
        $domainTest = $pieces[$i..($pieces.Count - 1)] -join '.'
        Write-Debug "Checking $domainTest"
        if ($domainTest -in $DomainNames) {
            Write-Debug "Found $domainTest"
            return $domains.rows | Where-Object domainName -eq $domainTest
        }
    }
    Write-Debug "Unable to find matching domain"
    return $null
}

function Find-TotalUptimeTXTRecord {
    [CmdletBinding()]
    param (
        [Parameter(Mandatory, Position = 0)]
        [string]$RecordName,
        [Parameter(Mandatory, Position = 1)]
        [string]$TxtContent,
        [Parameter(Mandatory, Position = 2)]
        [string]$DomainID,
        [Parameter(Mandatory)]
        [pscredential]$TotalUptimeCredential,
        [string]$TotalUptimeApiRoot = 'https://api.totaluptime.com'
    )
    #retrieve TXT records for Domain
    $TXTRecords = Get-TotalUptimeTXTRecords -DomainID $DomainID -TotalUptimeCredential $TotalUptimeCredential -TotalUptimeApiRoot $TotalUptimeApiRoot

    #filter by name an content
    $TXTRecords.rows | Where-Object { ($_.txtHostName -eq $RecordName) -and ($_.txtText -eq $TxtContent) }
}

# API Calls

function Get-TotalUptimeDomains {
    [CmdletBinding()]
    param(
        [Parameter(Mandatory)]
        [pscredential]$TotalUptimeCredential,
        [string]$TotalUptimeApiRoot = 'https://api.totaluptime.com'
    )

    #Build Request
    $reqParams = @{
        Uri = "$TotalUptimeApiRoot/CloudDNS/Domain/All"
        Headers = @{
            Authorization = 'Basic {0}' -f (ConvertTo-TotalUptimeBasicHTTPAuthString -Credential $TotalUptimeCredential)
            Accept = 'application/json'
        }
        Verbose = $false
        ErrorAction = 'Stop'
    }

    #Execute Request
    Write-Debug "GET $($reqParams.Uri)"
    Invoke-RestMethod @reqParams @script:UseBasic
}

function Get-TotalUptimeTXTRecords {
    [CmdletBinding()]
    param(
        [Parameter(Mandatory)]$DomainID,
        [pscredential]$TotalUptimeCredential,
        [string]$TotalUptimeApiRoot = 'https://api.totaluptime.com'
    )

    #Build Request
    $reqParams = @{
        Uri = "$TotalUptimeApiRoot/CloudDNS/Domain/$DomainID/TXTRecord/All"
        Headers = @{
            Authorization = ("Basic {0}" -f (ConvertTo-TotalUptimeBasicHTTPAuthString -Credential $TotalUptimeCredential));
            Accept        = "application/json"
        }
        Verbose = $false
        ErrorAction = 'Stop'
    }

    #Execute Request
    Write-Debug "GET $($reqParams.Uri)"
    Invoke-RestMethod @reqParams @script:UseBasic
}