2016-08-23 08:31:50 +08:00
|
|
|
package namedotcom
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"regexp"
|
|
|
|
"sort"
|
|
|
|
"strings"
|
|
|
|
|
2020-04-15 04:47:30 +08:00
|
|
|
"github.com/StackExchange/dnscontrol/v3/models"
|
2018-01-11 05:49:20 +08:00
|
|
|
"github.com/namedotcom/go/namecom"
|
2016-08-23 08:31:50 +08:00
|
|
|
)
|
|
|
|
|
2016-12-17 04:10:27 +08:00
|
|
|
var nsRegex = regexp.MustCompile(`ns([1-4])[a-z]{3}\.name\.com`)
|
|
|
|
|
2018-02-16 01:02:50 +08:00
|
|
|
// GetNameservers gets the nameservers set on a domain.
|
2020-10-26 21:25:30 +08:00
|
|
|
func (n *namedotcomProvider) GetNameservers(domain string) ([]*models.Nameserver, error) {
|
2018-01-10 01:53:16 +08:00
|
|
|
// This is an interesting edge case. Name.com expects you to SET the nameservers to ns[1-4].name.com,
|
|
|
|
// but it will internally set it to ns1xyz.name.com, where xyz is a uniqueish 3 letters.
|
|
|
|
// In order to avoid endless loops, we will use the unique nameservers if present, or else the generic ones if not.
|
2016-12-17 04:10:27 +08:00
|
|
|
nss, err := n.getNameserversRaw(domain)
|
2016-08-23 08:31:50 +08:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2016-12-17 04:10:27 +08:00
|
|
|
toUse := []string{"ns1.name.com", "ns2.name.com", "ns3.name.com", "ns4.name.com"}
|
|
|
|
for _, ns := range nss {
|
|
|
|
if matches := nsRegex.FindStringSubmatch(ns); len(matches) == 2 && len(matches[1]) == 1 {
|
2018-01-10 01:53:16 +08:00
|
|
|
idx := matches[1][0] - '1' // regex ensures proper range
|
2016-12-17 04:10:27 +08:00
|
|
|
toUse[idx] = matches[0]
|
|
|
|
}
|
|
|
|
}
|
2020-03-01 23:33:24 +08:00
|
|
|
return models.ToNameservers(toUse)
|
2016-12-17 04:10:27 +08:00
|
|
|
}
|
|
|
|
|
2020-10-26 21:25:30 +08:00
|
|
|
func (n *namedotcomProvider) getNameserversRaw(domain string) ([]string, error) {
|
2018-01-11 05:49:20 +08:00
|
|
|
request := &namecom.GetDomainRequest{
|
|
|
|
DomainName: domain,
|
2016-12-17 04:10:27 +08:00
|
|
|
}
|
2018-01-11 05:49:20 +08:00
|
|
|
|
|
|
|
response, err := n.client.GetDomain(request)
|
|
|
|
if err != nil {
|
2016-12-17 04:10:27 +08:00
|
|
|
return nil, err
|
2016-08-23 08:31:50 +08:00
|
|
|
}
|
2018-01-11 05:49:20 +08:00
|
|
|
|
|
|
|
sort.Strings(response.Nameservers)
|
|
|
|
return response.Nameservers, nil
|
2016-12-17 04:10:27 +08:00
|
|
|
}
|
|
|
|
|
2018-02-16 01:02:50 +08:00
|
|
|
// GetRegistrarCorrections gathers corrections that would being n to match dc.
|
2020-10-26 21:25:30 +08:00
|
|
|
func (n *namedotcomProvider) GetRegistrarCorrections(dc *models.DomainConfig) ([]*models.Correction, error) {
|
2016-12-17 04:10:27 +08:00
|
|
|
nss, err := n.getNameserversRaw(dc.Name)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
foundNameservers := strings.Join(nss, ",")
|
2016-08-23 08:31:50 +08:00
|
|
|
expected := []string{}
|
|
|
|
for _, ns := range dc.Nameservers {
|
2019-05-16 00:57:17 +08:00
|
|
|
expected = append(expected, ns.Name)
|
2016-08-23 08:31:50 +08:00
|
|
|
}
|
|
|
|
sort.Strings(expected)
|
|
|
|
expectedNameservers := strings.Join(expected, ",")
|
|
|
|
|
|
|
|
if foundNameservers != expectedNameservers {
|
|
|
|
return []*models.Correction{
|
|
|
|
{
|
|
|
|
Msg: fmt.Sprintf("Update nameservers %s -> %s", foundNameservers, expectedNameservers),
|
|
|
|
F: n.updateNameservers(expected, dc.Name),
|
|
|
|
},
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
2020-10-26 21:25:30 +08:00
|
|
|
func (n *namedotcomProvider) updateNameservers(ns []string, domain string) func() error {
|
2016-08-23 08:31:50 +08:00
|
|
|
return func() error {
|
2018-01-11 05:49:20 +08:00
|
|
|
request := &namecom.SetNameserversRequest{
|
|
|
|
DomainName: domain,
|
|
|
|
Nameservers: ns,
|
2016-08-23 08:31:50 +08:00
|
|
|
}
|
2018-01-11 05:49:20 +08:00
|
|
|
|
|
|
|
_, err := n.client.SetNameservers(request)
|
|
|
|
return err
|
2016-08-23 08:31:50 +08:00
|
|
|
}
|
|
|
|
}
|