2018-07-03 10:44:04 +00:00
// Package duckdns Adds lego support for http://duckdns.org.
2018-04-06 15:04:03 +00:00
// See http://www.duckdns.org/spec.jsp for more info on updating TXT records.
package duckdns
import (
"errors"
"fmt"
"io/ioutil"
2018-05-31 07:30:04 +00:00
"github.com/xenolf/lego/acme"
2018-07-03 10:44:04 +00:00
"github.com/xenolf/lego/platform/config/env"
2018-04-06 15:04:03 +00:00
)
// DNSProvider adds and removes the record for the DNS challenge
type DNSProvider struct {
2018-07-03 10:44:04 +00:00
// The api token
2018-04-06 15:04:03 +00:00
token string
}
// NewDNSProvider returns a new DNS provider using
// environment variable DUCKDNS_TOKEN for adding and removing the DNS record.
func NewDNSProvider ( ) ( * DNSProvider , error ) {
2018-07-03 10:44:04 +00:00
values , err := env . Get ( "DUCKDNS_TOKEN" )
if err != nil {
return nil , fmt . Errorf ( "DuckDNS: %v" , err )
}
2018-04-06 15:04:03 +00:00
2018-07-03 10:44:04 +00:00
return NewDNSProviderCredentials ( values [ "DUCKDNS_TOKEN" ] )
2018-04-06 15:04:03 +00:00
}
// NewDNSProviderCredentials uses the supplied credentials to return a
// DNSProvider instance configured for http://duckdns.org .
func NewDNSProviderCredentials ( duckdnsToken string ) ( * DNSProvider , error ) {
if duckdnsToken == "" {
2018-07-03 10:44:04 +00:00
return nil , errors . New ( "DuckDNS: credentials missing" )
2018-04-06 15:04:03 +00:00
}
return & DNSProvider { token : duckdnsToken } , nil
}
// makeDuckdnsURL creates a url to clear the set or unset the TXT record.
// txt == "" will clear the TXT record.
func makeDuckdnsURL ( domain , token , txt string ) string {
requestBase := fmt . Sprintf ( "https://www.duckdns.org/update?domains=%s&token=%s" , domain , token )
if txt == "" {
return requestBase + "&clear=true"
}
return requestBase + "&txt=" + txt
}
func issueDuckdnsRequest ( url string ) error {
2018-05-31 07:30:04 +00:00
response , err := acme . HTTPClient . Get ( url )
2018-04-06 15:04:03 +00:00
if err != nil {
return err
}
defer response . Body . Close ( )
bodyBytes , err := ioutil . ReadAll ( response . Body )
if err != nil {
return err
}
body := string ( bodyBytes )
if body != "OK" {
return fmt . Errorf ( "Request to change TXT record for duckdns returned the following result (%s) this does not match expectation (OK) used url [%s]" , body , url )
}
return nil
}
// Present creates a TXT record to fulfil the dns-01 challenge.
// In duckdns you only have one TXT record shared with
// the domain and all sub domains.
//
// To update the TXT record we just need to make one simple get request.
func ( d * DNSProvider ) Present ( domain , token , keyAuth string ) error {
2018-05-31 07:30:04 +00:00
_ , txtRecord , _ := acme . DNS01Record ( domain , keyAuth )
2018-04-06 15:04:03 +00:00
url := makeDuckdnsURL ( domain , d . token , txtRecord )
return issueDuckdnsRequest ( url )
}
// CleanUp clears duckdns TXT record
func ( d * DNSProvider ) CleanUp ( domain , token , keyAuth string ) error {
url := makeDuckdnsURL ( domain , d . token , "" )
return issueDuckdnsRequest ( url )
}