mirror of
https://github.com/StackExchange/dnscontrol.git
synced 2025-01-13 10:58:17 +08:00
06ee4d6fb1
This: * adds a global -v flag for verbosity * refactors the "printer" package to have a DefaultPrinter and package functions that call it, similar to net/http's DefaultServeMux * adds printer tests * moves current users of Debugf to Printf * moves most users of the "log" package to use "printer" * demotes noticably noisy log messages to "Debugf", like "IGNORE"- and "NO_PURGE"-related messages
202 lines
5.5 KiB
Go
202 lines
5.5 KiB
Go
package gandi
|
|
|
|
import (
|
|
"encoding/json"
|
|
"fmt"
|
|
"sort"
|
|
|
|
"github.com/StackExchange/dnscontrol/models"
|
|
"github.com/StackExchange/dnscontrol/pkg/printer"
|
|
"github.com/StackExchange/dnscontrol/providers"
|
|
"github.com/StackExchange/dnscontrol/providers/diff"
|
|
"github.com/pkg/errors"
|
|
|
|
"strings"
|
|
|
|
gandidomain "github.com/prasmussen/gandi-api/domain"
|
|
gandirecord "github.com/prasmussen/gandi-api/domain/zone/record"
|
|
)
|
|
|
|
/*
|
|
|
|
Gandi API DNS provider:
|
|
|
|
Info required in `creds.json`:
|
|
- apikey
|
|
|
|
*/
|
|
|
|
var features = providers.DocumentationNotes{
|
|
providers.CanUseCAA: providers.Can(),
|
|
providers.CanUsePTR: providers.Can(),
|
|
providers.CanUseSRV: providers.Can(),
|
|
providers.CantUseNOPURGE: providers.Cannot(),
|
|
providers.DocCreateDomains: providers.Cannot("Can only manage domains registered through their service"),
|
|
providers.DocOfficiallySupported: providers.Cannot(),
|
|
}
|
|
|
|
func init() {
|
|
providers.RegisterDomainServiceProviderType("GANDI", newDsp, features)
|
|
providers.RegisterRegistrarType("GANDI", newReg)
|
|
}
|
|
|
|
// GandiApi is the API handle for this module.
|
|
type GandiApi struct {
|
|
ApiKey string
|
|
domainIndex map[string]int64 // Map of domainname to index
|
|
nameservers map[string][]*models.Nameserver
|
|
ZoneId int64
|
|
}
|
|
|
|
type gandiRecord struct {
|
|
gandirecord.RecordInfo
|
|
}
|
|
|
|
func (c *GandiApi) getDomainInfo(domain string) (*gandidomain.DomainInfo, error) {
|
|
if err := c.fetchDomainList(); err != nil {
|
|
return nil, err
|
|
}
|
|
_, ok := c.domainIndex[domain]
|
|
if !ok {
|
|
return nil, errors.Errorf("%s not listed in zones for gandi account", domain)
|
|
}
|
|
return c.fetchDomainInfo(domain)
|
|
}
|
|
|
|
// GetNameservers returns the nameservers for domain.
|
|
func (c *GandiApi) GetNameservers(domain string) ([]*models.Nameserver, error) {
|
|
domaininfo, err := c.getDomainInfo(domain)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
ns := []*models.Nameserver{}
|
|
for _, nsname := range domaininfo.Nameservers {
|
|
ns = append(ns, &models.Nameserver{Name: nsname})
|
|
}
|
|
return ns, nil
|
|
}
|
|
|
|
// GetDomainCorrections returns a list of corrections recommended for this domain.
|
|
func (c *GandiApi) GetDomainCorrections(dc *models.DomainConfig) ([]*models.Correction, error) {
|
|
dc.Punycode()
|
|
domaininfo, err := c.getDomainInfo(dc.Name)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
foundRecords, err := c.getZoneRecords(domaininfo.ZoneId, dc.Name)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
expectedRecordSets := make([]gandirecord.RecordSet, 0, len(dc.Records))
|
|
recordsToKeep := make([]*models.RecordConfig, 0, len(dc.Records))
|
|
for _, rec := range dc.Records {
|
|
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 {
|
|
return nil, errors.Errorf("ERROR: Gandi does not support TTLs > 30 days (TTL=%d)", rec.TTL)
|
|
}
|
|
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. %s will not be added.\n", rec.GetTargetField())
|
|
}
|
|
continue
|
|
}
|
|
rs := gandirecord.RecordSet{
|
|
"type": rec.Type,
|
|
"name": rec.GetLabel(),
|
|
"value": rec.GetTargetCombined(),
|
|
"ttl": rec.TTL,
|
|
}
|
|
expectedRecordSets = append(expectedRecordSets, rs)
|
|
recordsToKeep = append(recordsToKeep, rec)
|
|
}
|
|
dc.Records = recordsToKeep
|
|
|
|
// Normalize
|
|
models.PostProcessRecords(foundRecords)
|
|
|
|
differ := diff.New(dc)
|
|
_, create, del, mod := differ.IncrementalDiff(foundRecords)
|
|
|
|
// Print a list of changes. Generate an actual change that is the zone
|
|
changes := false
|
|
desc := ""
|
|
for _, i := range create {
|
|
changes = true
|
|
desc += "\n" + i.String()
|
|
}
|
|
for _, i := range del {
|
|
changes = true
|
|
desc += "\n" + i.String()
|
|
}
|
|
for _, i := range mod {
|
|
changes = true
|
|
desc += "\n" + i.String()
|
|
}
|
|
|
|
msg := fmt.Sprintf("GENERATE_ZONE: %s (%d records)%s", dc.Name, len(dc.Records), desc)
|
|
corrections := []*models.Correction{}
|
|
if changes {
|
|
corrections = append(corrections,
|
|
&models.Correction{
|
|
Msg: msg,
|
|
F: func() error {
|
|
printer.Printf("CREATING ZONE: %v\n", dc.Name)
|
|
return c.createGandiZone(dc.Name, domaininfo.ZoneId, expectedRecordSets)
|
|
},
|
|
})
|
|
}
|
|
|
|
return corrections, nil
|
|
}
|
|
|
|
func newDsp(conf map[string]string, metadata json.RawMessage) (providers.DNSServiceProvider, error) {
|
|
return newGandi(conf, metadata)
|
|
}
|
|
|
|
func newReg(conf map[string]string) (providers.Registrar, error) {
|
|
return newGandi(conf, nil)
|
|
}
|
|
|
|
func newGandi(m map[string]string, metadata json.RawMessage) (*GandiApi, error) {
|
|
api := &GandiApi{}
|
|
api.ApiKey = m["apikey"]
|
|
if api.ApiKey == "" {
|
|
return nil, errors.Errorf("missing Gandi apikey")
|
|
}
|
|
|
|
return api, nil
|
|
}
|
|
|
|
// GetRegistrarCorrections returns a list of corrections for this registrar.
|
|
func (c *GandiApi) GetRegistrarCorrections(dc *models.DomainConfig) ([]*models.Correction, error) {
|
|
domaininfo, err := c.getDomainInfo(dc.Name)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
sort.Strings(domaininfo.Nameservers)
|
|
found := strings.Join(domaininfo.Nameservers, ",")
|
|
desiredNs := []string{}
|
|
for _, d := range dc.Nameservers {
|
|
desiredNs = append(desiredNs, d.Name)
|
|
}
|
|
sort.Strings(desiredNs)
|
|
desired := strings.Join(desiredNs, ",")
|
|
if found != desired {
|
|
return []*models.Correction{
|
|
{
|
|
Msg: fmt.Sprintf("Change Nameservers from '%s' to '%s'", found, desired),
|
|
F: func() (err error) {
|
|
_, err = c.setDomainNameservers(dc.Name, desiredNs)
|
|
return
|
|
}},
|
|
}, nil
|
|
}
|
|
return nil, nil
|
|
}
|