2018-02-16 01:02:50 +08:00
|
|
|
package models
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
|
|
|
|
"golang.org/x/net/idna"
|
|
|
|
)
|
|
|
|
|
|
|
|
// DomainConfig describes a DNS domain (tecnically a DNS zone).
|
|
|
|
type DomainConfig struct {
|
2020-08-27 01:45:02 +08:00
|
|
|
Name string `json:"name"` // NO trailing "."
|
|
|
|
RegistrarName string `json:"registrar"`
|
|
|
|
DNSProviderNames map[string]int `json:"dnsProviders"`
|
2018-02-16 01:02:50 +08:00
|
|
|
|
2020-08-18 23:14:34 +08:00
|
|
|
Metadata map[string]string `json:"meta,omitempty"`
|
|
|
|
Records Records `json:"records"`
|
|
|
|
Nameservers []*Nameserver `json:"nameservers,omitempty"`
|
|
|
|
KeepUnknown bool `json:"keepunknown,omitempty"`
|
|
|
|
IgnoredNames []string `json:"ignored_names,omitempty"`
|
|
|
|
IgnoredTargets []*IgnoreTarget `json:"ignored_targets,omitempty"`
|
2020-09-28 04:37:42 +08:00
|
|
|
AutoDNSSEC string `json:"auto_dnssec,omitempty"` // "", "on", "off"
|
2019-05-06 23:12:42 +08:00
|
|
|
//DNSSEC bool `json:"dnssec,omitempty"`
|
2018-02-16 01:02:50 +08:00
|
|
|
|
|
|
|
// These fields contain instantiated provider instances once everything is linked up.
|
|
|
|
// This linking is in two phases:
|
2020-07-07 08:18:24 +08:00
|
|
|
// 1. Metadata (name/type) is available just from the dnsconfig. Validation can use that.
|
2018-02-16 01:02:50 +08:00
|
|
|
// 2. Final driver instances are loaded after we load credentials. Any actual provider interaction requires that.
|
|
|
|
RegistrarInstance *RegistrarInstance `json:"-"`
|
|
|
|
DNSProviderInstances []*DNSProviderInstance `json:"-"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// Copy returns a deep copy of the DomainConfig.
|
|
|
|
func (dc *DomainConfig) Copy() (*DomainConfig, error) {
|
|
|
|
newDc := &DomainConfig{}
|
|
|
|
// provider instances are interfaces that gob hates if you don't register them.
|
|
|
|
// and the specific types are not gob encodable since nothing is exported.
|
|
|
|
// should find a better solution for this now.
|
|
|
|
//
|
|
|
|
// current strategy: remove everything, gob copy it. Then set both to stored copy.
|
|
|
|
reg := dc.RegistrarInstance
|
|
|
|
dnsps := dc.DNSProviderInstances
|
|
|
|
dc.RegistrarInstance = nil
|
|
|
|
dc.DNSProviderInstances = nil
|
|
|
|
err := copyObj(dc, newDc)
|
|
|
|
dc.RegistrarInstance = reg
|
|
|
|
newDc.RegistrarInstance = reg
|
|
|
|
dc.DNSProviderInstances = dnsps
|
|
|
|
newDc.DNSProviderInstances = dnsps
|
|
|
|
return newDc, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Filter removes all records that don't match the filter f.
|
|
|
|
func (dc *DomainConfig) Filter(f func(r *RecordConfig) bool) {
|
|
|
|
recs := []*RecordConfig{}
|
|
|
|
for _, r := range dc.Records {
|
|
|
|
if f(r) {
|
|
|
|
recs = append(recs, r)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
dc.Records = recs
|
|
|
|
}
|
|
|
|
|
|
|
|
// Punycode will convert all records to punycode format.
|
|
|
|
// It will encode:
|
|
|
|
// - Name
|
|
|
|
// - NameFQDN
|
|
|
|
// - Target (CNAME and MX only)
|
|
|
|
func (dc *DomainConfig) Punycode() error {
|
|
|
|
for _, rec := range dc.Records {
|
2018-03-20 05:18:58 +08:00
|
|
|
t, err := idna.ToASCII(rec.GetLabelFQDN())
|
2018-02-16 01:02:50 +08:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2018-03-20 05:18:58 +08:00
|
|
|
rec.SetLabelFromFQDN(t, dc.Name)
|
2018-02-16 01:02:50 +08:00
|
|
|
switch rec.Type { // #rtype_variations
|
2019-03-29 19:01:52 +08:00
|
|
|
case "ALIAS", "MX", "NS", "CNAME", "PTR", "SRV", "URL", "URL301", "FRAME", "R53_ALIAS":
|
2018-03-20 05:18:58 +08:00
|
|
|
// These rtypes are hostnames, therefore need to be converted (unlike, for example, an AAAA record)
|
|
|
|
t, err := idna.ToASCII(rec.GetTargetField())
|
|
|
|
rec.SetTarget(t)
|
2018-02-16 01:02:50 +08:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2020-05-30 22:40:21 +08:00
|
|
|
case "A", "AAAA", "CAA", "DS", "NAPTR", "SOA", "SSHFP", "TXT", "TLSA", "AZURE_ALIAS":
|
2018-02-16 01:02:50 +08:00
|
|
|
// Nothing to do.
|
|
|
|
default:
|
|
|
|
msg := fmt.Sprintf("Punycode rtype %v unimplemented", rec.Type)
|
|
|
|
panic(msg)
|
|
|
|
// We panic so that we quickly find any switch statements
|
|
|
|
// that have not been updated for a new RR type.
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|