2017-03-19 09:58:47 +08:00
|
|
|
package dnsimple
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
|
|
|
"sort"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"github.com/StackExchange/dnscontrol/models"
|
|
|
|
"github.com/StackExchange/dnscontrol/providers"
|
|
|
|
"github.com/StackExchange/dnscontrol/providers/diff"
|
|
|
|
"github.com/miekg/dns/dnsutil"
|
|
|
|
|
|
|
|
dnsimpleapi "github.com/dnsimple/dnsimple-go/dnsimple"
|
|
|
|
)
|
|
|
|
|
|
|
|
const stateRegistered = "registered"
|
|
|
|
|
|
|
|
var defaultNameServerNames = []string{
|
|
|
|
"ns1.dnsimple.com",
|
|
|
|
"ns2.dnsimple.com",
|
|
|
|
"ns3.dnsimple.com",
|
|
|
|
"ns4.dnsimple.com",
|
|
|
|
}
|
|
|
|
|
|
|
|
type DnsimpleApi struct {
|
|
|
|
AccountToken string // The account access token
|
|
|
|
BaseURL string // An alternate base URI
|
|
|
|
accountId string // Account id cache
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *DnsimpleApi) GetNameservers(domainName string) ([]*models.Nameserver, error) {
|
|
|
|
return models.StringsToNameservers(defaultNameServerNames), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *DnsimpleApi) GetDomainCorrections(dc *models.DomainConfig) ([]*models.Correction, error) {
|
|
|
|
corrections := []*models.Correction{}
|
|
|
|
|
|
|
|
records, err := c.getRecords(dc.Name)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
var actual []*models.RecordConfig
|
|
|
|
for _, r := range records {
|
|
|
|
if r.Type == "SOA" || r.Type == "NS" {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if r.Name == "" {
|
|
|
|
r.Name = "@"
|
|
|
|
}
|
|
|
|
if r.Type == "CNAME" || r.Type == "MX" {
|
|
|
|
r.Content += "."
|
|
|
|
}
|
|
|
|
rec := &models.RecordConfig{
|
2017-07-20 03:53:40 +08:00
|
|
|
NameFQDN: dnsutil.AddOrigin(r.Name, dc.Name),
|
|
|
|
Type: r.Type,
|
|
|
|
Target: r.Content,
|
|
|
|
TTL: uint32(r.TTL),
|
|
|
|
MxPreference: uint16(r.Priority),
|
|
|
|
Original: r,
|
2017-03-19 09:58:47 +08:00
|
|
|
}
|
|
|
|
actual = append(actual, rec)
|
|
|
|
}
|
|
|
|
removeOtherNS(dc)
|
|
|
|
differ := diff.New(dc)
|
|
|
|
_, create, delete, modify := differ.IncrementalDiff(actual)
|
|
|
|
|
|
|
|
for _, del := range delete {
|
|
|
|
rec := del.Existing.Original.(dnsimpleapi.ZoneRecord)
|
|
|
|
corrections = append(corrections, &models.Correction{
|
|
|
|
Msg: del.String(),
|
|
|
|
F: c.deleteRecordFunc(rec.ID, dc.Name),
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, cre := range create {
|
|
|
|
rec := cre.Desired
|
|
|
|
corrections = append(corrections, &models.Correction{
|
|
|
|
Msg: cre.String(),
|
|
|
|
F: c.createRecordFunc(rec, dc.Name),
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, mod := range modify {
|
|
|
|
old := mod.Existing.Original.(dnsimpleapi.ZoneRecord)
|
|
|
|
new := mod.Desired
|
|
|
|
corrections = append(corrections, &models.Correction{
|
|
|
|
Msg: mod.String(),
|
|
|
|
F: c.updateRecordFunc(&old, new, dc.Name),
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
return corrections, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *DnsimpleApi) GetRegistrarCorrections(dc *models.DomainConfig) ([]*models.Correction, error) {
|
|
|
|
corrections := []*models.Correction{}
|
|
|
|
|
|
|
|
nameServers, err := c.getNameservers(dc.Name)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
actual := strings.Join(nameServers, ",")
|
|
|
|
|
|
|
|
expectedSet := []string{}
|
|
|
|
for _, ns := range dc.Nameservers {
|
|
|
|
expectedSet = append(expectedSet, ns.Name)
|
|
|
|
}
|
|
|
|
sort.Strings(expectedSet)
|
|
|
|
expected := strings.Join(expectedSet, ",")
|
|
|
|
|
|
|
|
if actual != expected {
|
|
|
|
return []*models.Correction{
|
|
|
|
{
|
|
|
|
Msg: fmt.Sprintf("Update nameservers %s -> %s", actual, expected),
|
|
|
|
F: c.updateNameserversFunc(expectedSet, dc.Name),
|
|
|
|
},
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return corrections, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// DNSimple calls
|
|
|
|
|
|
|
|
func (c *DnsimpleApi) getClient() *dnsimpleapi.Client {
|
|
|
|
client := dnsimpleapi.NewClient(dnsimpleapi.NewOauthTokenCredentials(c.AccountToken))
|
|
|
|
if c.BaseURL != "" {
|
|
|
|
client.BaseURL = c.BaseURL
|
|
|
|
}
|
|
|
|
return client
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *DnsimpleApi) getAccountId() (string, error) {
|
|
|
|
if c.accountId == "" {
|
|
|
|
client := c.getClient()
|
|
|
|
whoamiResponse, err := client.Identity.Whoami()
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
if whoamiResponse.Data.User != nil && whoamiResponse.Data.Account == nil {
|
|
|
|
return "", fmt.Errorf("DNSimple token appears to be a user token. Please supply an account token")
|
|
|
|
}
|
|
|
|
c.accountId = strconv.Itoa(whoamiResponse.Data.Account.ID)
|
|
|
|
}
|
|
|
|
return c.accountId, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *DnsimpleApi) getRecords(domainName string) ([]dnsimpleapi.ZoneRecord, error) {
|
|
|
|
client := c.getClient()
|
|
|
|
|
|
|
|
accountId, err := c.getAccountId()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
recordsResponse, err := client.Zones.ListRecords(accountId, domainName, nil)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return recordsResponse.Data, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Returns the name server names that should be used. If the domain is registered
|
|
|
|
// then this method will return the delegation name servers. If this domain
|
|
|
|
// is hosted only, then it will return the default DNSimple name servers.
|
|
|
|
func (c *DnsimpleApi) getNameservers(domainName string) ([]string, error) {
|
|
|
|
client := c.getClient()
|
|
|
|
|
|
|
|
accountId, err := c.getAccountId()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
domainResponse, err := client.Domains.GetDomain(accountId, domainName)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if domainResponse.Data.State == stateRegistered {
|
|
|
|
|
|
|
|
delegationResponse, err := client.Registrar.GetDomainDelegation(accountId, domainName)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return *delegationResponse.Data, nil
|
|
|
|
} else {
|
|
|
|
return defaultNameServerNames, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Returns a function that can be invoked to change the delegation of the domain to the given name server names.
|
|
|
|
func (c *DnsimpleApi) updateNameserversFunc(nameServerNames []string, domainName string) func() error {
|
|
|
|
return func() error {
|
|
|
|
client := c.getClient()
|
|
|
|
|
|
|
|
accountId, err := c.getAccountId()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
nameServers := dnsimpleapi.Delegation(nameServerNames)
|
|
|
|
|
|
|
|
_, err = client.Registrar.ChangeDomainDelegation(accountId, domainName, &nameServers)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Returns a function that can be invoked to create a record in a zone.
|
|
|
|
func (c *DnsimpleApi) createRecordFunc(rc *models.RecordConfig, domainName string) func() error {
|
|
|
|
return func() error {
|
|
|
|
client := c.getClient()
|
|
|
|
|
|
|
|
accountId, err := c.getAccountId()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
record := dnsimpleapi.ZoneRecord{
|
|
|
|
Name: dnsutil.TrimDomainName(rc.NameFQDN, domainName),
|
|
|
|
Type: rc.Type,
|
|
|
|
Content: rc.Target,
|
|
|
|
TTL: int(rc.TTL),
|
2017-07-20 03:53:40 +08:00
|
|
|
Priority: int(rc.MxPreference),
|
2017-03-19 09:58:47 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
_, err = client.Zones.CreateRecord(accountId, domainName, record)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Returns a function that can be invoked to delete a record in a zone.
|
|
|
|
func (c *DnsimpleApi) deleteRecordFunc(recordId int, domainName string) func() error {
|
|
|
|
return func() error {
|
|
|
|
client := c.getClient()
|
|
|
|
|
|
|
|
accountId, err := c.getAccountId()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
_, err = client.Zones.DeleteRecord(accountId, domainName, recordId)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Returns a function that can be invoked to update a record in a zone.
|
|
|
|
func (c *DnsimpleApi) updateRecordFunc(old *dnsimpleapi.ZoneRecord, rc *models.RecordConfig, domainName string) func() error {
|
|
|
|
return func() error {
|
|
|
|
client := c.getClient()
|
|
|
|
|
|
|
|
accountId, err := c.getAccountId()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
record := dnsimpleapi.ZoneRecord{
|
|
|
|
Name: dnsutil.TrimDomainName(rc.NameFQDN, domainName),
|
|
|
|
Type: rc.Type,
|
|
|
|
Content: rc.Target,
|
|
|
|
TTL: int(rc.TTL),
|
2017-07-20 03:53:40 +08:00
|
|
|
Priority: int(rc.MxPreference),
|
2017-03-19 09:58:47 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
_, err = client.Zones.UpdateRecord(accountId, domainName, old.ID, record)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// constructors
|
|
|
|
|
|
|
|
func newReg(conf map[string]string) (providers.Registrar, error) {
|
|
|
|
return newProvider(conf, nil)
|
|
|
|
}
|
|
|
|
|
|
|
|
func newDsp(conf map[string]string, metadata json.RawMessage) (providers.DNSServiceProvider, error) {
|
|
|
|
return newProvider(conf, metadata)
|
|
|
|
}
|
|
|
|
|
|
|
|
func newProvider(m map[string]string, metadata json.RawMessage) (*DnsimpleApi, error) {
|
|
|
|
api := &DnsimpleApi{}
|
|
|
|
api.AccountToken = m["token"]
|
|
|
|
if api.AccountToken == "" {
|
|
|
|
return nil, fmt.Errorf("DNSimple token must be provided.")
|
|
|
|
}
|
|
|
|
|
|
|
|
if m["baseurl"] != "" {
|
|
|
|
api.BaseURL = m["baseurl"]
|
|
|
|
}
|
|
|
|
|
|
|
|
return api, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
providers.RegisterRegistrarType("DNSIMPLE", newReg)
|
2017-07-20 02:41:18 +08:00
|
|
|
providers.RegisterDomainServiceProviderType("DNSIMPLE", newDsp, providers.CanUsePTR)
|
2017-03-19 09:58:47 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// remove all non-dnsimple NS records from our desired state.
|
|
|
|
// if any are found, print a warning
|
|
|
|
func removeOtherNS(dc *models.DomainConfig) {
|
|
|
|
newList := make([]*models.RecordConfig, 0, len(dc.Records))
|
|
|
|
for _, rec := range dc.Records {
|
2017-03-21 11:28:43 +08:00
|
|
|
if rec.Type == "NS" {
|
|
|
|
// apex NS inside dnsimple are expected.
|
|
|
|
if rec.NameFQDN == dc.Name && strings.HasSuffix(rec.Target, ".dnsimple.com.") {
|
|
|
|
continue
|
2017-03-19 09:58:47 +08:00
|
|
|
}
|
2017-03-21 11:28:43 +08:00
|
|
|
fmt.Printf("Warning: dnsimple.com does not allow NS records to be modified. %s will not be added.\n", rec.Target)
|
2017-03-19 09:58:47 +08:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
newList = append(newList, rec)
|
|
|
|
}
|
|
|
|
dc.Records = newList
|
|
|
|
}
|