2016-08-23 08:31:50 +08:00
|
|
|
package namedotcom
|
|
|
|
|
|
|
|
import (
|
Switch to Go 1.13 error wrapping (#604)
* Replaced errors.Wrap with fmt.Errorf (#589)
* Find: errors\.Wrap\(([^,]+),\s+(["`][^"`]*)(["`])\)
Replace: fmt.Errorf($2: %w$3, $1)
* Replaced errors.Wrapf with fmt.Errorf (#589)
* Find: errors\.Wrapf\(([^,]+),\s+(["`][^"`]*)(["`])\)
Replace: fmt.Errorf($2: %w$3, $1)
* Find: errors\.Wrapf\(([^,]+),\s+(["`][^"`]*)(["`])(,[^)]+)\)
* Replace: fmt.Errorf($2: %w$3$4, $1)
* Replaced errors.Errorf with fmt.Errorf (#589)
* Find: errors\.Errorf
Replace: fmt.Errorf
* Cleaned up remaining imports
* Cleanup
* Regenerate provider support matrix
This was broken by #533 ... and it's now the third time this has been missed.
2020-01-29 00:06:56 +08:00
|
|
|
"errors"
|
2016-08-23 08:31:50 +08:00
|
|
|
"fmt"
|
2018-01-11 20:23:59 +08:00
|
|
|
"regexp"
|
2017-07-20 03:53:40 +08:00
|
|
|
"strings"
|
2016-08-23 08:31:50 +08:00
|
|
|
|
2018-01-11 05:49:20 +08:00
|
|
|
"github.com/namedotcom/go/namecom"
|
|
|
|
|
2020-04-15 04:47:30 +08:00
|
|
|
"github.com/StackExchange/dnscontrol/v3/models"
|
|
|
|
"github.com/StackExchange/dnscontrol/v3/pkg/diff"
|
2016-08-23 08:31:50 +08:00
|
|
|
)
|
|
|
|
|
|
|
|
var defaultNameservers = []*models.Nameserver{
|
|
|
|
{Name: "ns1.name.com"},
|
|
|
|
{Name: "ns2.name.com"},
|
|
|
|
{Name: "ns3.name.com"},
|
|
|
|
{Name: "ns4.name.com"},
|
|
|
|
}
|
|
|
|
|
2020-02-18 21:59:18 +08:00
|
|
|
// GetZoneRecords gets the records of a zone and returns them in RecordConfig format.
|
2020-10-26 21:25:30 +08:00
|
|
|
func (n *namedotcomProvider) GetZoneRecords(domain string) (models.Records, error) {
|
2020-02-22 04:03:27 +08:00
|
|
|
records, err := n.getRecords(domain)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
actual := make([]*models.RecordConfig, len(records))
|
|
|
|
for i, r := range records {
|
|
|
|
actual[i] = toRecord(r, domain)
|
|
|
|
}
|
|
|
|
|
|
|
|
return actual, nil
|
2020-02-18 21:59:18 +08:00
|
|
|
}
|
|
|
|
|
2018-02-16 01:02:50 +08:00
|
|
|
// GetDomainCorrections gathers correctios that would bring n to match dc.
|
2020-10-26 21:25:30 +08:00
|
|
|
func (n *namedotcomProvider) GetDomainCorrections(dc *models.DomainConfig) ([]*models.Correction, error) {
|
2017-08-05 23:08:22 +08:00
|
|
|
dc.Punycode()
|
2020-02-22 04:03:27 +08:00
|
|
|
|
|
|
|
actual, err := n.GetZoneRecords(dc.Name)
|
2016-08-23 08:31:50 +08:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2017-07-20 01:23:17 +08:00
|
|
|
for _, rec := range dc.Records {
|
|
|
|
if rec.Type == "ALIAS" {
|
|
|
|
rec.Type = "ANAME"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-01-12 03:38:07 +08:00
|
|
|
checkNSModifications(dc)
|
2016-08-23 08:31:50 +08:00
|
|
|
|
2017-11-08 06:12:17 +08:00
|
|
|
// Normalize
|
2018-01-05 08:19:35 +08:00
|
|
|
models.PostProcessRecords(actual)
|
2017-11-08 06:12:17 +08:00
|
|
|
|
2017-01-12 03:38:07 +08:00
|
|
|
differ := diff.New(dc)
|
2020-08-21 03:49:00 +08:00
|
|
|
_, create, del, mod, err := differ.IncrementalDiff(actual)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2016-08-23 08:31:50 +08:00
|
|
|
corrections := []*models.Correction{}
|
|
|
|
|
|
|
|
for _, d := range del {
|
2018-01-11 05:49:20 +08:00
|
|
|
rec := d.Existing.Original.(*namecom.Record)
|
|
|
|
c := &models.Correction{Msg: d.String(), F: func() error { return n.deleteRecord(rec.ID, dc.Name) }}
|
2016-08-23 08:31:50 +08:00
|
|
|
corrections = append(corrections, c)
|
|
|
|
}
|
|
|
|
for _, cre := range create {
|
2017-01-14 03:30:04 +08:00
|
|
|
rec := cre.Desired
|
2016-08-23 08:31:50 +08:00
|
|
|
c := &models.Correction{Msg: cre.String(), F: func() error { return n.createRecord(rec, dc.Name) }}
|
|
|
|
corrections = append(corrections, c)
|
|
|
|
}
|
|
|
|
for _, chng := range mod {
|
2018-01-11 05:49:20 +08:00
|
|
|
old := chng.Existing.Original.(*namecom.Record)
|
2017-01-12 03:38:07 +08:00
|
|
|
new := chng.Desired
|
2016-08-23 08:31:50 +08:00
|
|
|
c := &models.Correction{Msg: chng.String(), F: func() error {
|
2018-01-11 05:49:20 +08:00
|
|
|
err := n.deleteRecord(old.ID, dc.Name)
|
2016-08-23 08:31:50 +08:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return n.createRecord(new, dc.Name)
|
|
|
|
}}
|
|
|
|
corrections = append(corrections, c)
|
|
|
|
}
|
|
|
|
return corrections, nil
|
|
|
|
}
|
|
|
|
|
2017-01-12 03:38:07 +08:00
|
|
|
func checkNSModifications(dc *models.DomainConfig) {
|
|
|
|
newList := make([]*models.RecordConfig, 0, len(dc.Records))
|
|
|
|
for _, rec := range dc.Records {
|
2018-03-20 05:18:58 +08:00
|
|
|
if rec.Type == "NS" && rec.GetLabel() == "@" {
|
2017-04-20 05:20:26 +08:00
|
|
|
continue // Apex NS records are automatically created for the domain's nameservers and cannot be managed otherwise via the name.com API.
|
2017-01-12 03:38:07 +08:00
|
|
|
}
|
|
|
|
newList = append(newList, rec)
|
|
|
|
}
|
|
|
|
dc.Records = newList
|
2016-08-23 08:31:50 +08:00
|
|
|
}
|
2017-01-12 03:38:07 +08:00
|
|
|
|
2018-02-16 01:02:50 +08:00
|
|
|
func toRecord(r *namecom.Record, origin string) *models.RecordConfig {
|
2017-07-20 03:53:40 +08:00
|
|
|
rc := &models.RecordConfig{
|
2017-01-12 03:38:07 +08:00
|
|
|
Type: r.Type,
|
2018-01-11 05:49:20 +08:00
|
|
|
TTL: r.TTL,
|
2017-01-12 03:38:07 +08:00
|
|
|
Original: r,
|
2016-08-23 08:31:50 +08:00
|
|
|
}
|
2018-02-16 01:02:50 +08:00
|
|
|
if !strings.HasSuffix(r.Fqdn, ".") {
|
Switch to Go 1.13 error wrapping (#604)
* Replaced errors.Wrap with fmt.Errorf (#589)
* Find: errors\.Wrap\(([^,]+),\s+(["`][^"`]*)(["`])\)
Replace: fmt.Errorf($2: %w$3, $1)
* Replaced errors.Wrapf with fmt.Errorf (#589)
* Find: errors\.Wrapf\(([^,]+),\s+(["`][^"`]*)(["`])\)
Replace: fmt.Errorf($2: %w$3, $1)
* Find: errors\.Wrapf\(([^,]+),\s+(["`][^"`]*)(["`])(,[^)]+)\)
* Replace: fmt.Errorf($2: %w$3$4, $1)
* Replaced errors.Errorf with fmt.Errorf (#589)
* Find: errors\.Errorf
Replace: fmt.Errorf
* Cleaned up remaining imports
* Cleanup
* Regenerate provider support matrix
This was broken by #533 ... and it's now the third time this has been missed.
2020-01-29 00:06:56 +08:00
|
|
|
panic(fmt.Errorf("namedotcom suddenly changed protocol. Bailing. (%v)", r.Fqdn))
|
2018-02-16 01:02:50 +08:00
|
|
|
}
|
|
|
|
fqdn := r.Fqdn[:len(r.Fqdn)-1]
|
|
|
|
rc.SetLabelFromFQDN(fqdn, origin)
|
|
|
|
switch rtype := r.Type; rtype { // #rtype_variations
|
2018-01-11 20:23:59 +08:00
|
|
|
case "TXT":
|
2018-02-16 01:02:50 +08:00
|
|
|
rc.SetTargetTXTs(decodeTxt(r.Answer))
|
2017-07-20 03:53:40 +08:00
|
|
|
case "MX":
|
2018-02-16 01:02:50 +08:00
|
|
|
if err := rc.SetTargetMX(uint16(r.Priority), r.Answer); err != nil {
|
Switch to Go 1.13 error wrapping (#604)
* Replaced errors.Wrap with fmt.Errorf (#589)
* Find: errors\.Wrap\(([^,]+),\s+(["`][^"`]*)(["`])\)
Replace: fmt.Errorf($2: %w$3, $1)
* Replaced errors.Wrapf with fmt.Errorf (#589)
* Find: errors\.Wrapf\(([^,]+),\s+(["`][^"`]*)(["`])\)
Replace: fmt.Errorf($2: %w$3, $1)
* Find: errors\.Wrapf\(([^,]+),\s+(["`][^"`]*)(["`])(,[^)]+)\)
* Replace: fmt.Errorf($2: %w$3$4, $1)
* Replaced errors.Errorf with fmt.Errorf (#589)
* Find: errors\.Errorf
Replace: fmt.Errorf
* Cleaned up remaining imports
* Cleanup
* Regenerate provider support matrix
This was broken by #533 ... and it's now the third time this has been missed.
2020-01-29 00:06:56 +08:00
|
|
|
panic(fmt.Errorf("unparsable MX record received from ndc: %w", err))
|
2018-02-16 01:02:50 +08:00
|
|
|
}
|
2017-07-20 03:53:40 +08:00
|
|
|
case "SRV":
|
2018-02-16 01:02:50 +08:00
|
|
|
if err := rc.SetTargetSRVPriorityString(uint16(r.Priority), r.Answer+"."); err != nil {
|
Switch to Go 1.13 error wrapping (#604)
* Replaced errors.Wrap with fmt.Errorf (#589)
* Find: errors\.Wrap\(([^,]+),\s+(["`][^"`]*)(["`])\)
Replace: fmt.Errorf($2: %w$3, $1)
* Replaced errors.Wrapf with fmt.Errorf (#589)
* Find: errors\.Wrapf\(([^,]+),\s+(["`][^"`]*)(["`])\)
Replace: fmt.Errorf($2: %w$3, $1)
* Find: errors\.Wrapf\(([^,]+),\s+(["`][^"`]*)(["`])(,[^)]+)\)
* Replace: fmt.Errorf($2: %w$3$4, $1)
* Replaced errors.Errorf with fmt.Errorf (#589)
* Find: errors\.Errorf
Replace: fmt.Errorf
* Cleaned up remaining imports
* Cleanup
* Regenerate provider support matrix
This was broken by #533 ... and it's now the third time this has been missed.
2020-01-29 00:06:56 +08:00
|
|
|
panic(fmt.Errorf("unparsable SRV record received from ndc: %w", err))
|
2018-02-16 01:02:50 +08:00
|
|
|
}
|
|
|
|
default: // "A", "AAAA", "ANAME", "CNAME", "NS"
|
|
|
|
if err := rc.PopulateFromString(rtype, r.Answer, r.Fqdn); err != nil {
|
Switch to Go 1.13 error wrapping (#604)
* Replaced errors.Wrap with fmt.Errorf (#589)
* Find: errors\.Wrap\(([^,]+),\s+(["`][^"`]*)(["`])\)
Replace: fmt.Errorf($2: %w$3, $1)
* Replaced errors.Wrapf with fmt.Errorf (#589)
* Find: errors\.Wrapf\(([^,]+),\s+(["`][^"`]*)(["`])\)
Replace: fmt.Errorf($2: %w$3, $1)
* Find: errors\.Wrapf\(([^,]+),\s+(["`][^"`]*)(["`])(,[^)]+)\)
* Replace: fmt.Errorf($2: %w$3$4, $1)
* Replaced errors.Errorf with fmt.Errorf (#589)
* Find: errors\.Errorf
Replace: fmt.Errorf
* Cleaned up remaining imports
* Cleanup
* Regenerate provider support matrix
This was broken by #533 ... and it's now the third time this has been missed.
2020-01-29 00:06:56 +08:00
|
|
|
panic(fmt.Errorf("unparsable record received from ndc: %w", err))
|
2018-02-16 01:02:50 +08:00
|
|
|
}
|
2017-07-20 03:53:40 +08:00
|
|
|
}
|
|
|
|
return rc
|
2016-08-23 08:31:50 +08:00
|
|
|
}
|
|
|
|
|
2020-10-26 21:25:30 +08:00
|
|
|
func (n *namedotcomProvider) getRecords(domain string) ([]*namecom.Record, error) {
|
2018-01-11 05:49:20 +08:00
|
|
|
var (
|
|
|
|
err error
|
|
|
|
records []*namecom.Record
|
|
|
|
response *namecom.ListRecordsResponse
|
|
|
|
)
|
2016-08-23 08:31:50 +08:00
|
|
|
|
2018-01-11 05:49:20 +08:00
|
|
|
request := &namecom.ListRecordsRequest{
|
|
|
|
DomainName: domain,
|
|
|
|
Page: 1,
|
2016-08-23 08:31:50 +08:00
|
|
|
}
|
2018-01-11 05:49:20 +08:00
|
|
|
|
|
|
|
for request.Page > 0 {
|
|
|
|
response, err = n.client.ListRecords(request)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
records = append(records, response.Records...)
|
|
|
|
request.Page = response.NextPage
|
2016-08-23 08:31:50 +08:00
|
|
|
}
|
|
|
|
|
2018-01-11 05:49:20 +08:00
|
|
|
for _, rc := range records {
|
2017-07-21 04:39:40 +08:00
|
|
|
if rc.Type == "CNAME" || rc.Type == "ANAME" || rc.Type == "MX" || rc.Type == "NS" {
|
2018-01-11 05:49:20 +08:00
|
|
|
rc.Answer = rc.Answer + "."
|
2016-08-23 08:31:50 +08:00
|
|
|
}
|
|
|
|
}
|
2018-01-11 05:49:20 +08:00
|
|
|
return records, nil
|
2016-08-23 08:31:50 +08:00
|
|
|
}
|
|
|
|
|
2020-10-26 21:25:30 +08:00
|
|
|
func (n *namedotcomProvider) createRecord(rc *models.RecordConfig, domain string) error {
|
2018-01-11 05:49:20 +08:00
|
|
|
record := &namecom.Record{
|
|
|
|
DomainName: domain,
|
2018-03-20 05:18:58 +08:00
|
|
|
Host: rc.GetLabel(),
|
2018-01-11 05:49:20 +08:00
|
|
|
Type: rc.Type,
|
2018-03-20 05:18:58 +08:00
|
|
|
Answer: rc.GetTargetField(),
|
2018-01-11 05:49:20 +08:00
|
|
|
TTL: rc.TTL,
|
|
|
|
Priority: uint32(rc.MxPreference),
|
2016-08-23 08:31:50 +08:00
|
|
|
}
|
2017-08-05 03:26:29 +08:00
|
|
|
switch rc.Type { // #rtype_variations
|
2018-01-11 20:23:59 +08:00
|
|
|
case "A", "AAAA", "ANAME", "CNAME", "MX", "NS":
|
2021-05-05 02:15:31 +08:00
|
|
|
// nothing
|
|
|
|
case "TXT":
|
2021-03-08 02:19:22 +08:00
|
|
|
// record.Answer = encodeTxt(rc.TxtStrings)
|
2017-07-20 03:53:40 +08:00
|
|
|
case "SRV":
|
2019-11-15 00:25:20 +08:00
|
|
|
if rc.GetTargetField() == "." {
|
|
|
|
return errors.New("SRV records with empty targets are not supported (as of 2019-11-05, the API returns 'Parameter Value Error - Invalid Srv Format')")
|
|
|
|
}
|
2018-03-20 05:18:58 +08:00
|
|
|
record.Answer = fmt.Sprintf("%d %d %v", rc.SrvWeight, rc.SrvPort, rc.GetTargetField())
|
2018-01-11 05:49:20 +08:00
|
|
|
record.Priority = uint32(rc.SrvPriority)
|
2017-07-20 03:53:40 +08:00
|
|
|
default:
|
|
|
|
panic(fmt.Sprintf("createRecord rtype %v unimplemented", rc.Type))
|
2017-08-05 03:26:29 +08:00
|
|
|
// We panic so that we quickly find any switch statements
|
|
|
|
// that have not been updated for a new RR type.
|
2017-07-20 03:53:40 +08:00
|
|
|
}
|
2018-01-11 05:49:20 +08:00
|
|
|
_, err := n.client.CreateRecord(record)
|
|
|
|
return err
|
2016-08-23 08:31:50 +08:00
|
|
|
}
|
|
|
|
|
2021-03-08 02:19:22 +08:00
|
|
|
// // makeTxt encodes TxtStrings for sending in the CREATE/MODIFY API:
|
|
|
|
// func encodeTxt(txts []string) string {
|
|
|
|
// ans := txts[0]
|
|
|
|
|
|
|
|
// if len(txts) > 1 {
|
|
|
|
// ans = ""
|
|
|
|
// for _, t := range txts {
|
|
|
|
// ans += `"` + strings.Replace(t, `"`, `\"`, -1) + `"`
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
// return ans
|
|
|
|
// }
|
2018-02-16 01:02:50 +08:00
|
|
|
|
2020-07-07 08:18:24 +08:00
|
|
|
// finds a string surrounded by quotes that might contain an escaped quote character.
|
2018-02-16 01:02:50 +08:00
|
|
|
var quotedStringRegexp = regexp.MustCompile(`"((?:[^"\\]|\\.)*)"`)
|
|
|
|
|
|
|
|
// decodeTxt decodes the TXT record as received from name.com and
|
|
|
|
// returns the list of strings.
|
|
|
|
func decodeTxt(s string) []string {
|
|
|
|
|
|
|
|
if len(s) >= 2 && s[0] == '"' && s[len(s)-1] == '"' {
|
|
|
|
txtStrings := []string{}
|
|
|
|
for _, t := range quotedStringRegexp.FindAllStringSubmatch(s, -1) {
|
|
|
|
txtString := strings.Replace(t[1], `\"`, `"`, -1)
|
|
|
|
txtStrings = append(txtStrings, txtString)
|
|
|
|
}
|
|
|
|
return txtStrings
|
|
|
|
}
|
|
|
|
return []string{s}
|
|
|
|
}
|
|
|
|
|
2020-10-26 21:25:30 +08:00
|
|
|
func (n *namedotcomProvider) deleteRecord(id int32, domain string) error {
|
2018-01-11 05:49:20 +08:00
|
|
|
request := &namecom.DeleteRecordRequest{
|
|
|
|
DomainName: domain,
|
|
|
|
ID: id,
|
2016-08-23 08:31:50 +08:00
|
|
|
}
|
2018-01-11 05:49:20 +08:00
|
|
|
|
|
|
|
_, err := n.client.DeleteRecord(request)
|
|
|
|
return err
|
2016-08-23 08:31:50 +08:00
|
|
|
}
|