2021-03-09 09:14:30 +08:00
|
|
|
package gandiv5
|
2020-01-21 03:13:32 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
|
|
|
|
Gandi API V5 LiveDNS provider:
|
|
|
|
|
|
|
|
Documentation: https://api.gandi.net/docs/
|
|
|
|
Endpoint: https://api.gandi.net/
|
|
|
|
|
|
|
|
Settings from `creds.json`:
|
|
|
|
- apikey
|
|
|
|
- sharing_id (optional)
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
|
|
|
"os"
|
|
|
|
"sort"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
|
2023-05-21 01:21:45 +08:00
|
|
|
"github.com/StackExchange/dnscontrol/v4/models"
|
|
|
|
"github.com/StackExchange/dnscontrol/v4/pkg/diff"
|
|
|
|
"github.com/StackExchange/dnscontrol/v4/pkg/diff2"
|
|
|
|
"github.com/StackExchange/dnscontrol/v4/pkg/printer"
|
|
|
|
"github.com/StackExchange/dnscontrol/v4/pkg/txtutil"
|
|
|
|
"github.com/StackExchange/dnscontrol/v4/providers"
|
2022-08-15 08:46:56 +08:00
|
|
|
"github.com/go-gandi/go-gandi"
|
|
|
|
"github.com/go-gandi/go-gandi/config"
|
|
|
|
"github.com/miekg/dns/dnsutil"
|
2020-01-21 03:13:32 +08:00
|
|
|
)
|
|
|
|
|
|
|
|
// Section 1: Register this provider in the system.
|
|
|
|
|
|
|
|
// init registers the provider to dnscontrol.
|
|
|
|
func init() {
|
2021-03-08 02:19:22 +08:00
|
|
|
fns := providers.DspFuncs{
|
2021-03-09 09:14:30 +08:00
|
|
|
Initializer: newDsp,
|
|
|
|
RecordAuditor: AuditRecords,
|
2021-03-08 02:19:22 +08:00
|
|
|
}
|
|
|
|
providers.RegisterDomainServiceProviderType("GANDI_V5", fns, features)
|
2020-01-21 03:13:32 +08:00
|
|
|
providers.RegisterRegistrarType("GANDI_V5", newReg)
|
|
|
|
}
|
|
|
|
|
|
|
|
// features declares which features and options are available.
|
|
|
|
var features = providers.DocumentationNotes{
|
2022-03-03 00:19:15 +08:00
|
|
|
providers.CanGetZones: providers.Can(),
|
2020-03-10 22:13:20 +08:00
|
|
|
providers.CanUseAlias: providers.Can("Only on the bare domain. Otherwise CNAME will be substituted"),
|
2020-01-21 03:13:32 +08:00
|
|
|
providers.CanUseCAA: providers.Can(),
|
2021-04-17 22:49:06 +08:00
|
|
|
providers.CanUseDS: providers.Cannot("Only supports DS records at the apex"),
|
|
|
|
providers.CanUseDSForChildren: providers.Can(),
|
2023-03-17 02:04:20 +08:00
|
|
|
providers.CanUseLOC: providers.Cannot(),
|
2020-01-21 03:13:32 +08:00
|
|
|
providers.CanUsePTR: providers.Can(),
|
|
|
|
providers.CanUseSRV: providers.Can(),
|
2020-03-01 22:36:12 +08:00
|
|
|
providers.CanUseSSHFP: providers.Can(),
|
|
|
|
providers.CanUseTLSA: providers.Can(),
|
2020-01-21 03:13:32 +08:00
|
|
|
providers.CantUseNOPURGE: providers.Cannot(),
|
|
|
|
providers.DocCreateDomains: providers.Cannot("Can only manage domains registered through their service"),
|
|
|
|
providers.DocOfficiallySupported: providers.Cannot(),
|
|
|
|
}
|
|
|
|
|
2020-03-01 22:36:12 +08:00
|
|
|
// DNSSEC: platform supports it, but it doesn't fit our GetDomainCorrections
|
|
|
|
// model, so deferring for now.
|
|
|
|
|
2020-01-21 03:13:32 +08:00
|
|
|
// Section 2: Define the API client.
|
|
|
|
|
2020-10-26 21:25:30 +08:00
|
|
|
// gandiv5Provider is the gandiv5Provider handle used to store any client-related state.
|
|
|
|
type gandiv5Provider struct {
|
2020-01-21 03:13:32 +08:00
|
|
|
apikey string
|
|
|
|
sharingid string
|
|
|
|
debug bool
|
|
|
|
}
|
|
|
|
|
|
|
|
// newDsp generates a DNS Service Provider client handle.
|
|
|
|
func newDsp(conf map[string]string, metadata json.RawMessage) (providers.DNSServiceProvider, error) {
|
|
|
|
return newHelper(conf, metadata)
|
|
|
|
}
|
|
|
|
|
|
|
|
// newReg generates a Registrar Provider client handle.
|
|
|
|
func newReg(conf map[string]string) (providers.Registrar, error) {
|
|
|
|
return newHelper(conf, nil)
|
|
|
|
}
|
|
|
|
|
|
|
|
// newHelper generates a handle.
|
2020-10-26 21:25:30 +08:00
|
|
|
func newHelper(m map[string]string, metadata json.RawMessage) (*gandiv5Provider, error) {
|
|
|
|
api := &gandiv5Provider{}
|
2020-01-21 03:13:32 +08:00
|
|
|
api.apikey = m["apikey"]
|
|
|
|
if api.apikey == "" {
|
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
|
|
|
return nil, fmt.Errorf("missing Gandi apikey")
|
2020-01-21 03:13:32 +08:00
|
|
|
}
|
|
|
|
api.sharingid = m["sharing_id"]
|
|
|
|
debug, err := strconv.ParseBool(os.Getenv("GANDI_V5_DEBUG"))
|
|
|
|
if err == nil {
|
|
|
|
api.debug = debug
|
|
|
|
}
|
|
|
|
|
|
|
|
return api, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Section 3: Domain Service Provider (DSP) related functions
|
|
|
|
|
2023-01-22 06:22:26 +08:00
|
|
|
// // ListZones lists the zones on this account.
|
|
|
|
// This no longer works. Until we can figure out why, we're removing this
|
|
|
|
// feature for Gandi.
|
|
|
|
// func (client *gandiv5Provider) ListZones() ([]string, error) {
|
|
|
|
// g := gandi.NewLiveDNSClient(config.Config{
|
|
|
|
// APIKey: client.apikey,
|
|
|
|
// SharingID: client.sharingid,
|
|
|
|
// Debug: client.debug,
|
|
|
|
// })
|
|
|
|
|
|
|
|
// listResp, err := g.ListDomains()
|
|
|
|
// if err != nil {
|
|
|
|
// return nil, err
|
|
|
|
// }
|
|
|
|
|
|
|
|
// zones := make([]string, len(listResp))
|
|
|
|
// fmt.Printf("DEBUG: HERE START\n")
|
|
|
|
// for i, zone := range listResp {
|
|
|
|
// fmt.Printf("DEBUG: HERE %d: %v\n", i, zone.FQDN)
|
|
|
|
// zone := zone
|
|
|
|
// zones[i] = zone.FQDN
|
|
|
|
// }
|
|
|
|
// fmt.Printf("DEBUG: HERE END\n")
|
|
|
|
// return zones, nil
|
|
|
|
// }
|
2021-04-03 22:31:25 +08:00
|
|
|
|
2020-01-21 03:13:32 +08:00
|
|
|
// GetZoneRecords gathers the DNS records and converts them to
|
|
|
|
// dnscontrol's format.
|
2023-05-03 01:04:59 +08:00
|
|
|
func (client *gandiv5Provider) GetZoneRecords(domain string, meta map[string]string) (models.Records, error) {
|
2022-03-08 02:00:39 +08:00
|
|
|
g := gandi.NewLiveDNSClient(config.Config{
|
|
|
|
APIKey: client.apikey,
|
|
|
|
SharingID: client.sharingid,
|
|
|
|
Debug: client.debug,
|
|
|
|
})
|
2020-01-21 03:13:32 +08:00
|
|
|
|
|
|
|
// Get all the existing records:
|
2020-03-28 00:09:56 +08:00
|
|
|
records, err := g.GetDomainRecords(domain)
|
2020-01-21 03:13:32 +08:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Convert them to DNScontrol's native format:
|
|
|
|
existingRecords := []*models.RecordConfig{}
|
|
|
|
for _, rr := range records {
|
2022-07-22 02:30:26 +08:00
|
|
|
rrs, err := nativeToRecords(rr, domain)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
existingRecords = append(existingRecords, rrs...)
|
2020-01-21 03:13:32 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return existingRecords, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// PrepDesiredRecords munges any records to best suit this provider.
|
|
|
|
func PrepDesiredRecords(dc *models.DomainConfig) {
|
|
|
|
// Sort through the dc.Records, eliminate any that can't be
|
|
|
|
// supported; modify any that need adjustments to work with the
|
|
|
|
// provider. We try to do minimal changes otherwise it gets
|
|
|
|
// confusing.
|
|
|
|
|
|
|
|
recordsToKeep := make([]*models.RecordConfig, 0, len(dc.Records))
|
|
|
|
for _, rec := range dc.Records {
|
2020-03-10 22:13:20 +08:00
|
|
|
if rec.Type == "ALIAS" && rec.Name != "@" {
|
|
|
|
// GANDI only permits aliases on a naked domain.
|
|
|
|
// Therefore, we change this to a CNAME.
|
|
|
|
rec.Type = "CNAME"
|
|
|
|
}
|
2020-01-21 03:13:32 +08:00
|
|
|
if rec.TTL < 300 {
|
|
|
|
printer.Warnf("Gandi does not support ttls < 300. Setting %s from %d to 300\n", rec.GetLabelFQDN(), rec.TTL)
|
|
|
|
rec.TTL = 300
|
|
|
|
}
|
|
|
|
if rec.TTL > 2592000 {
|
|
|
|
printer.Warnf("Gandi does not support ttls > 30 days. Setting %s from %d to 2592000\n", rec.GetLabelFQDN(), rec.TTL)
|
|
|
|
rec.TTL = 2592000
|
|
|
|
}
|
|
|
|
if rec.Type == "TXT" {
|
|
|
|
rec.SetTarget("\"" + rec.GetTargetField() + "\"") // FIXME(tlim): Should do proper quoting.
|
|
|
|
}
|
|
|
|
if rec.Type == "NS" && rec.GetLabel() == "@" {
|
|
|
|
if !strings.HasSuffix(rec.GetTargetField(), ".gandi.net.") {
|
|
|
|
printer.Warnf("Gandi does not support changing apex NS records. Ignoring %s\n", rec.GetTargetField())
|
|
|
|
}
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
recordsToKeep = append(recordsToKeep, rec)
|
|
|
|
}
|
|
|
|
dc.Records = recordsToKeep
|
|
|
|
}
|
|
|
|
|
2023-04-15 03:22:23 +08:00
|
|
|
// GetZoneRecordsCorrections returns a list of corrections that will turn existing records into dc.Records.
|
|
|
|
func (client *gandiv5Provider) GetZoneRecordsCorrections(dc *models.DomainConfig, existing models.Records) ([]*models.Correction, error) {
|
2020-01-21 03:13:32 +08:00
|
|
|
if client.debug {
|
|
|
|
debugRecords("GenDC input", existing)
|
|
|
|
}
|
|
|
|
|
2023-04-15 03:22:23 +08:00
|
|
|
PrepDesiredRecords(dc)
|
2021-03-08 02:19:22 +08:00
|
|
|
txtutil.SplitSingleLongTxt(dc.Records) // Autosplit long TXT records
|
|
|
|
|
2022-12-12 04:02:58 +08:00
|
|
|
var corrections []*models.Correction
|
2022-12-12 06:28:58 +08:00
|
|
|
if !diff2.EnableDiff2 {
|
2022-12-12 04:02:58 +08:00
|
|
|
|
|
|
|
// diff existing vs. current.
|
|
|
|
differ := diff.New(dc)
|
|
|
|
keysToUpdate, err := differ.ChangedGroups(existing)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if client.debug {
|
|
|
|
diff.DebugKeyMapMap("GenDC diff", keysToUpdate)
|
|
|
|
}
|
|
|
|
if len(keysToUpdate) == 0 {
|
|
|
|
return nil, nil
|
|
|
|
}
|
2020-01-21 03:13:32 +08:00
|
|
|
|
2022-12-12 04:02:58 +08:00
|
|
|
// Regroup data by FQDN. ChangedGroups returns data grouped by label:RType tuples.
|
|
|
|
affectedLabels, msgsForLabel := gatherAffectedLabels(keysToUpdate)
|
|
|
|
_, desiredRecords := dc.Records.GroupedByFQDN()
|
|
|
|
doesLabelExist := existing.FQDNMap()
|
2020-01-21 03:13:32 +08:00
|
|
|
|
2022-12-12 04:02:58 +08:00
|
|
|
g := gandi.NewLiveDNSClient(config.Config{
|
|
|
|
APIKey: client.apikey,
|
|
|
|
SharingID: client.sharingid,
|
|
|
|
Debug: client.debug,
|
|
|
|
})
|
2020-01-21 03:13:32 +08:00
|
|
|
|
2022-12-12 04:02:58 +08:00
|
|
|
// For any key with an update, delete or replace those records.
|
|
|
|
for label := range affectedLabels {
|
|
|
|
if len(desiredRecords[label]) == 0 {
|
|
|
|
// No records matching this key? This can only mean that all
|
|
|
|
// the records were deleted. Delete them.
|
|
|
|
|
|
|
|
msgs := strings.Join(msgsForLabel[label], "\n")
|
2020-01-21 03:13:32 +08:00
|
|
|
domain := dc.Name
|
|
|
|
shortname := dnsutil.TrimDomainName(label, dc.Name)
|
|
|
|
corrections = append(corrections,
|
|
|
|
&models.Correction{
|
2022-12-12 04:02:58 +08:00
|
|
|
Msg: msgs,
|
2020-01-21 03:13:32 +08:00
|
|
|
F: func() error {
|
2022-12-12 04:02:58 +08:00
|
|
|
err := g.DeleteDomainRecordsByName(domain, shortname)
|
2020-01-21 03:13:32 +08:00
|
|
|
if err != nil {
|
2022-12-12 04:02:58 +08:00
|
|
|
return err
|
2020-01-21 03:13:32 +08:00
|
|
|
}
|
|
|
|
return nil
|
|
|
|
},
|
|
|
|
})
|
|
|
|
|
|
|
|
} else {
|
2022-12-12 04:02:58 +08:00
|
|
|
// Replace all the records at a label with our new records.
|
|
|
|
|
|
|
|
// Generate the new data in Gandi's format.
|
|
|
|
ns := recordsToNative(desiredRecords[label], dc.Name)
|
|
|
|
|
|
|
|
if doesLabelExist[label] {
|
|
|
|
// Records exist for this label. Replace them with what we have.
|
2020-01-21 03:13:32 +08:00
|
|
|
|
|
|
|
msg := strings.Join(msgsForLabel[label], "\n")
|
|
|
|
domain := dc.Name
|
|
|
|
shortname := dnsutil.TrimDomainName(label, dc.Name)
|
|
|
|
corrections = append(corrections,
|
|
|
|
&models.Correction{
|
|
|
|
Msg: msg,
|
|
|
|
F: func() error {
|
2022-12-12 04:02:58 +08:00
|
|
|
res, err := g.UpdateDomainRecordsByName(domain, shortname, ns)
|
2020-01-21 03:13:32 +08:00
|
|
|
if 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
|
|
|
return fmt.Errorf("%+v: %w", res, err)
|
2020-01-21 03:13:32 +08:00
|
|
|
}
|
|
|
|
return nil
|
|
|
|
},
|
|
|
|
})
|
2022-12-12 04:02:58 +08:00
|
|
|
|
|
|
|
} else {
|
|
|
|
// First time putting data on this label. Create it.
|
|
|
|
|
|
|
|
// We have to create the label one rtype at a time.
|
2022-12-12 06:28:58 +08:00
|
|
|
ns := recordsToNative(desiredRecords[label], dc.Name)
|
2022-12-12 04:02:58 +08:00
|
|
|
for _, n := range ns {
|
|
|
|
msg := strings.Join(msgsForLabel[label], "\n")
|
|
|
|
domain := dc.Name
|
|
|
|
shortname := dnsutil.TrimDomainName(label, dc.Name)
|
|
|
|
rtype := n.RrsetType
|
|
|
|
ttl := n.RrsetTTL
|
|
|
|
values := n.RrsetValues
|
|
|
|
corrections = append(corrections,
|
|
|
|
&models.Correction{
|
|
|
|
Msg: msg,
|
|
|
|
F: func() error {
|
|
|
|
res, err := g.CreateDomainRecord(domain, shortname, rtype, ttl, values)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("%+v: %w", res, err)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
},
|
|
|
|
})
|
|
|
|
}
|
2020-01-21 03:13:32 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-12-12 04:02:58 +08:00
|
|
|
|
|
|
|
// NB(tlim): This sort is just to make updates look pretty. It is
|
|
|
|
// cosmetic. The risk here is that there may be some updates that
|
|
|
|
// require a specific order (for example a delete before an add).
|
|
|
|
// However the code doesn't seem to have such situation. All tests
|
|
|
|
// pass. That said, if this breaks anything, the easiest fix might
|
|
|
|
// be to just remove the sort.
|
|
|
|
sort.Slice(corrections, func(i, j int) bool { return diff.CorrectionLess(corrections, i, j) })
|
|
|
|
|
|
|
|
return corrections, nil
|
2020-01-21 03:13:32 +08:00
|
|
|
}
|
|
|
|
|
2022-12-12 06:28:58 +08:00
|
|
|
g := gandi.NewLiveDNSClient(config.Config{
|
|
|
|
APIKey: client.apikey,
|
|
|
|
SharingID: client.sharingid,
|
|
|
|
Debug: client.debug,
|
|
|
|
})
|
|
|
|
|
|
|
|
// Gandi is a "ByLabel" API with the odd exception that changes must be
|
|
|
|
// done one label:rtype at a time.
|
|
|
|
instructions, err := diff2.ByLabel(existing, dc, nil)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
for _, inst := range instructions {
|
|
|
|
switch inst.Type {
|
|
|
|
|
2023-02-01 20:27:00 +08:00
|
|
|
case diff2.REPORT:
|
|
|
|
corrections = append(corrections, &models.Correction{Msg: inst.MsgsJoined})
|
|
|
|
|
2022-12-12 06:28:58 +08:00
|
|
|
case diff2.CREATE:
|
|
|
|
// We have to create the label one rtype at a time.
|
2023-02-28 14:25:09 +08:00
|
|
|
// In other words, this is a ByRecordSet API for creation, even though
|
2023-04-15 03:22:23 +08:00
|
|
|
// this is very ByLabel()-ish for everything else.
|
2022-12-12 06:28:58 +08:00
|
|
|
natives := recordsToNative(inst.New, dc.Name)
|
|
|
|
for _, n := range natives {
|
|
|
|
label := inst.Key.NameFQDN
|
|
|
|
rtype := n.RrsetType
|
|
|
|
domain := dc.Name
|
|
|
|
shortname := dnsutil.TrimDomainName(label, dc.Name)
|
|
|
|
ttl := n.RrsetTTL
|
|
|
|
values := n.RrsetValues
|
|
|
|
key := models.RecordKey{NameFQDN: label, Type: rtype}
|
|
|
|
msg := strings.Join(inst.MsgsByKey[key], "\n")
|
|
|
|
corrections = append(corrections,
|
|
|
|
&models.Correction{
|
|
|
|
Msg: msg,
|
|
|
|
F: func() error {
|
|
|
|
res, err := g.CreateDomainRecord(domain, shortname, rtype, ttl, values)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("%+v: %w", res, err)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
},
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
case diff2.CHANGE:
|
|
|
|
msgs := strings.Join(inst.Msgs, "\n")
|
|
|
|
domain := dc.Name
|
|
|
|
label := inst.Key.NameFQDN
|
|
|
|
shortname := dnsutil.TrimDomainName(label, dc.Name)
|
|
|
|
ns := recordsToNative(inst.New, dc.Name)
|
|
|
|
corrections = append(corrections,
|
|
|
|
&models.Correction{
|
|
|
|
Msg: msgs,
|
|
|
|
F: func() error {
|
|
|
|
res, err := g.UpdateDomainRecordsByName(domain, shortname, ns)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("%+v: %w", res, err)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
},
|
|
|
|
})
|
|
|
|
|
|
|
|
case diff2.DELETE:
|
|
|
|
msgs := strings.Join(inst.Msgs, "\n")
|
|
|
|
domain := dc.Name
|
|
|
|
label := inst.Key.NameFQDN
|
|
|
|
shortname := dnsutil.TrimDomainName(label, dc.Name)
|
|
|
|
corrections = append(corrections,
|
|
|
|
&models.Correction{
|
|
|
|
Msg: msgs,
|
|
|
|
F: func() error {
|
|
|
|
err := g.DeleteDomainRecordsByName(domain, shortname)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
},
|
|
|
|
})
|
2023-02-01 20:27:00 +08:00
|
|
|
|
|
|
|
default:
|
|
|
|
panic(fmt.Sprintf("unhandled inst.Type %s", inst.Type))
|
2022-12-12 06:28:58 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
2020-07-01 17:55:20 +08:00
|
|
|
|
2020-01-21 03:13:32 +08:00
|
|
|
return corrections, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// debugRecords prints a list of RecordConfig.
|
|
|
|
func debugRecords(note string, recs []*models.RecordConfig) {
|
2022-06-18 21:01:02 +08:00
|
|
|
printer.Debugf(note)
|
2020-01-21 03:13:32 +08:00
|
|
|
for k, v := range recs {
|
2022-06-18 21:01:02 +08:00
|
|
|
printer.Printf(" %v: %v %v %v %v\n", k, v.GetLabel(), v.Type, v.TTL, v.GetTargetCombined())
|
2020-01-21 03:13:32 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// gatherAffectedLabels takes the output of diff.ChangedGroups and
|
|
|
|
// regroups it by FQDN of the label, not by Key. It also returns
|
|
|
|
// a list of all the FQDNs.
|
|
|
|
func gatherAffectedLabels(groups map[models.RecordKey][]string) (labels map[string]bool, msgs map[string][]string) {
|
|
|
|
labels = map[string]bool{}
|
|
|
|
msgs = map[string][]string{}
|
|
|
|
for k, v := range groups {
|
|
|
|
labels[k.NameFQDN] = true
|
|
|
|
msgs[k.NameFQDN] = append(msgs[k.NameFQDN], v...)
|
|
|
|
}
|
|
|
|
return labels, msgs
|
|
|
|
}
|
|
|
|
|
|
|
|
// Section 3: Registrar-related functions
|
|
|
|
|
|
|
|
// GetNameservers returns a list of nameservers for domain.
|
2020-10-26 21:25:30 +08:00
|
|
|
func (client *gandiv5Provider) GetNameservers(domain string) ([]*models.Nameserver, error) {
|
2022-03-08 02:00:39 +08:00
|
|
|
g := gandi.NewLiveDNSClient(config.Config{
|
|
|
|
APIKey: client.apikey,
|
|
|
|
SharingID: client.sharingid,
|
|
|
|
Debug: client.debug,
|
|
|
|
})
|
2020-01-21 03:13:32 +08:00
|
|
|
nameservers, err := g.GetDomainNS(domain)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2020-03-01 23:33:24 +08:00
|
|
|
return models.ToNameservers(nameservers)
|
2020-01-21 03:13:32 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// GetRegistrarCorrections returns a list of corrections for this registrar.
|
2020-10-26 21:25:30 +08:00
|
|
|
func (client *gandiv5Provider) GetRegistrarCorrections(dc *models.DomainConfig) ([]*models.Correction, error) {
|
2022-03-08 02:00:39 +08:00
|
|
|
gd := gandi.NewDomainClient(config.Config{
|
|
|
|
APIKey: client.apikey,
|
|
|
|
SharingID: client.sharingid,
|
|
|
|
Debug: client.debug,
|
|
|
|
})
|
2020-01-21 03:13:32 +08:00
|
|
|
|
|
|
|
existingNs, err := gd.GetNameServers(dc.Name)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
sort.Strings(existingNs)
|
|
|
|
existing := strings.Join(existingNs, ",")
|
|
|
|
|
|
|
|
desiredNs := models.NameserversToStrings(dc.Nameservers)
|
|
|
|
sort.Strings(desiredNs)
|
|
|
|
desired := strings.Join(desiredNs, ",")
|
|
|
|
|
|
|
|
if existing != desired {
|
|
|
|
return []*models.Correction{
|
|
|
|
{
|
|
|
|
Msg: fmt.Sprintf("Change Nameservers from '%s' to '%s'", existing, desired),
|
|
|
|
F: func() (err error) {
|
|
|
|
err = gd.UpdateNameServers(dc.Name, desiredNs)
|
|
|
|
return
|
|
|
|
}},
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
return nil, nil
|
|
|
|
}
|