2018-01-10 01:53:16 +08:00
|
|
|
// Package nameservers provides logic for dynamically finding nameservers for a domain, and configuring NS records for them.
|
2016-12-17 04:10:27 +08:00
|
|
|
package nameservers
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
2018-01-10 01:53:16 +08:00
|
|
|
"strconv"
|
2022-08-02 02:44:17 +08:00
|
|
|
"strings"
|
2018-01-10 01:53:16 +08:00
|
|
|
|
2023-05-21 01:21:45 +08:00
|
|
|
"github.com/StackExchange/dnscontrol/v4/models"
|
|
|
|
"github.com/StackExchange/dnscontrol/v4/pkg/printer"
|
2016-12-17 04:10:27 +08:00
|
|
|
)
|
|
|
|
|
2018-01-10 01:53:16 +08:00
|
|
|
// DetermineNameservers will find all nameservers we should use for a domain. It follows the following rules:
|
|
|
|
// 1. All explicitly defined NAMESERVER records will be used.
|
|
|
|
// 2. Each DSP declares how many nameservers to use. Default is all. 0 indicates to use none.
|
2018-02-02 00:45:53 +08:00
|
|
|
func DetermineNameservers(dc *models.DomainConfig) ([]*models.Nameserver, error) {
|
2022-08-02 02:44:17 +08:00
|
|
|
return DetermineNameserversForProviders(dc, dc.DNSProviderInstances)
|
|
|
|
}
|
|
|
|
|
|
|
|
// DetermineNameserversForProviders is like DetermineNameservers, for a subset of providers.
|
|
|
|
func DetermineNameserversForProviders(dc *models.DomainConfig, providers []*models.DNSProviderInstance) ([]*models.Nameserver, error) {
|
2018-01-10 01:53:16 +08:00
|
|
|
// always take explicit
|
2016-12-17 04:10:27 +08:00
|
|
|
ns := dc.Nameservers
|
2022-08-02 02:44:17 +08:00
|
|
|
for _, dnsProvider := range providers {
|
2018-02-02 00:45:53 +08:00
|
|
|
n := dnsProvider.NumberOfNameservers
|
2016-12-17 04:10:27 +08:00
|
|
|
if n == 0 {
|
|
|
|
continue
|
|
|
|
}
|
2022-08-21 08:59:02 +08:00
|
|
|
if !printer.SkinnyReport {
|
|
|
|
fmt.Printf("----- Getting nameservers from: %s\n", dnsProvider.Name)
|
|
|
|
}
|
2018-02-02 00:45:53 +08:00
|
|
|
nss, err := dnsProvider.Driver.GetNameservers(dc.Name)
|
2016-12-17 04:10:27 +08:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2020-03-01 23:33:24 +08:00
|
|
|
// Clean up the nameservers due to
|
|
|
|
// https://github.com/StackExchange/dnscontrol/issues/491
|
|
|
|
// In the far future, this warning will become a fatal error.
|
2020-03-11 04:53:17 +08:00
|
|
|
for i := range nss {
|
2020-03-01 23:33:24 +08:00
|
|
|
if strings.HasSuffix(nss[i].Name, ".") {
|
|
|
|
models.WarnNameserverDot(dnsProvider.Name, fmt.Sprintf("DetermineNameservers (%s) (%s)", dc.Name, nss[i].Name))
|
|
|
|
nss[i].Name = strings.TrimSuffix(nss[i].Name, ".")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-12-17 04:10:27 +08:00
|
|
|
take := len(nss)
|
|
|
|
if n > 0 && n < take {
|
|
|
|
take = n
|
|
|
|
}
|
|
|
|
for i := 0; i < take; i++ {
|
|
|
|
ns = append(ns, nss[i])
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ns, nil
|
|
|
|
}
|
|
|
|
|
2018-01-10 01:53:16 +08:00
|
|
|
// AddNSRecords creates NS records on a domain corresponding to the nameservers specified.
|
2016-12-17 04:10:27 +08:00
|
|
|
func AddNSRecords(dc *models.DomainConfig) {
|
|
|
|
ttl := uint32(300)
|
|
|
|
if ttls, ok := dc.Metadata["ns_ttl"]; ok {
|
|
|
|
t, err := strconv.ParseUint(ttls, 10, 32)
|
|
|
|
if err != nil {
|
2020-01-30 02:47:32 +08:00
|
|
|
fmt.Printf("WARNING: ns_ttl for %s (%s) is not a valid int", dc.Name, ttls)
|
2016-12-17 04:10:27 +08:00
|
|
|
} else {
|
|
|
|
ttl = uint32(t)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for _, ns := range dc.Nameservers {
|
|
|
|
rc := &models.RecordConfig{
|
|
|
|
Type: "NS",
|
|
|
|
Metadata: map[string]string{},
|
|
|
|
TTL: ttl,
|
|
|
|
}
|
2018-03-20 05:18:58 +08:00
|
|
|
rc.SetLabel("@", dc.Name)
|
|
|
|
t := ns.Name
|
|
|
|
if !strings.HasSuffix(t, ".") {
|
2018-03-22 23:52:52 +08:00
|
|
|
t += "."
|
2016-12-17 04:10:27 +08:00
|
|
|
}
|
2018-03-20 05:18:58 +08:00
|
|
|
rc.SetTarget(t)
|
|
|
|
|
2016-12-17 04:10:27 +08:00
|
|
|
dc.Records = append(dc.Records, rc)
|
|
|
|
}
|
|
|
|
}
|