2016-08-23 08:31:50 +08:00
|
|
|
package gandi
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
2017-03-28 06:03:01 +08:00
|
|
|
"log"
|
2016-08-23 08:31:50 +08:00
|
|
|
|
|
|
|
"github.com/StackExchange/dnscontrol/models"
|
|
|
|
"github.com/StackExchange/dnscontrol/providers"
|
|
|
|
"github.com/StackExchange/dnscontrol/providers/diff"
|
|
|
|
|
2017-03-28 06:03:01 +08:00
|
|
|
"strings"
|
|
|
|
|
2016-12-17 04:10:27 +08:00
|
|
|
gandidomain "github.com/prasmussen/gandi-api/domain"
|
2016-08-23 08:31:50 +08:00
|
|
|
gandirecord "github.com/prasmussen/gandi-api/domain/zone/record"
|
|
|
|
)
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
|
|
|
Gandi API DNS provider:
|
|
|
|
|
|
|
|
Info required in `creds.json`:
|
|
|
|
- apikey
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
type GandiApi struct {
|
|
|
|
ApiKey string
|
|
|
|
domainIndex map[string]int64 // Map of domainname to index
|
|
|
|
nameservers map[string][]*models.Nameserver
|
|
|
|
ZoneId int64
|
|
|
|
}
|
|
|
|
|
2017-01-12 03:38:07 +08:00
|
|
|
type gandiRecord struct {
|
2016-08-23 08:31:50 +08:00
|
|
|
gandirecord.RecordInfo
|
|
|
|
}
|
|
|
|
|
2016-12-17 04:10:27 +08:00
|
|
|
func (c *GandiApi) getDomainInfo(domain string) (*gandidomain.DomainInfo, error) {
|
|
|
|
if err := c.fetchDomainList(); err != nil {
|
|
|
|
return nil, err
|
2016-08-23 08:31:50 +08:00
|
|
|
}
|
2016-12-17 04:10:27 +08:00
|
|
|
_, ok := c.domainIndex[domain]
|
2016-08-23 08:31:50 +08:00
|
|
|
if !ok {
|
2016-12-17 04:10:27 +08:00
|
|
|
return nil, fmt.Errorf("%s not listed in zones for gandi account", domain)
|
2016-08-23 08:31:50 +08:00
|
|
|
}
|
2016-12-17 04:10:27 +08:00
|
|
|
return c.fetchDomainInfo(domain)
|
|
|
|
}
|
2016-08-23 08:31:50 +08:00
|
|
|
|
2016-12-17 04:10:27 +08:00
|
|
|
func (c *GandiApi) GetNameservers(domain string) ([]*models.Nameserver, error) {
|
|
|
|
domaininfo, err := c.getDomainInfo(domain)
|
2016-08-23 08:31:50 +08:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2016-12-17 04:10:27 +08:00
|
|
|
ns := []*models.Nameserver{}
|
2016-08-23 08:31:50 +08:00
|
|
|
for _, nsname := range domaininfo.Nameservers {
|
2016-12-17 04:10:27 +08:00
|
|
|
ns = append(ns, &models.Nameserver{Name: nsname})
|
|
|
|
}
|
|
|
|
return ns, nil
|
|
|
|
}
|
2017-01-12 03:38:07 +08:00
|
|
|
|
2016-12-17 04:10:27 +08:00
|
|
|
func (c *GandiApi) GetDomainCorrections(dc *models.DomainConfig) ([]*models.Correction, error) {
|
2017-03-28 06:03:01 +08:00
|
|
|
dc.Punycode()
|
|
|
|
dc.CombineMXs()
|
2016-12-17 04:10:27 +08:00
|
|
|
domaininfo, err := c.getDomainInfo(dc.Name)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
2016-08-23 08:31:50 +08:00
|
|
|
}
|
2017-03-28 06:03:01 +08:00
|
|
|
foundRecords, err := c.getZoneRecords(domaininfo.ZoneId, dc.Name)
|
2016-08-23 08:31:50 +08:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2017-01-12 03:38:07 +08:00
|
|
|
|
2017-03-28 06:03:01 +08:00
|
|
|
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 {
|
|
|
|
log.Printf("WARNING: Gandi does not support ttls < 300. %s will not be set to %d.", rec.NameFQDN, rec.TTL)
|
|
|
|
rec.TTL = 300
|
2016-08-23 08:31:50 +08:00
|
|
|
}
|
2017-01-12 03:38:07 +08:00
|
|
|
if rec.Type == "TXT" {
|
|
|
|
rec.Target = "\"" + rec.Target + "\"" // FIXME(tlim): Should do proper quoting.
|
2016-08-23 08:31:50 +08:00
|
|
|
}
|
2017-03-28 06:03:01 +08:00
|
|
|
if rec.Type == "NS" && rec.Name == "@" {
|
|
|
|
if !strings.HasSuffix(rec.Target, ".gandi.net.") {
|
|
|
|
log.Printf("WARNING: Gandi does not support changing apex NS records. %s will not be added.", rec.Target)
|
|
|
|
}
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
rs := gandirecord.RecordSet{
|
|
|
|
"type": rec.Type,
|
|
|
|
"name": rec.Name,
|
|
|
|
"value": rec.Target,
|
|
|
|
"ttl": rec.TTL,
|
|
|
|
}
|
|
|
|
expectedRecordSets = append(expectedRecordSets, rs)
|
|
|
|
recordsToKeep = append(recordsToKeep, rec)
|
2016-08-23 08:31:50 +08:00
|
|
|
}
|
2017-03-28 06:03:01 +08:00
|
|
|
dc.Records = recordsToKeep
|
2017-01-12 03:38:07 +08:00
|
|
|
differ := diff.New(dc)
|
|
|
|
_, create, del, mod := differ.IncrementalDiff(foundRecords)
|
2016-08-23 08:31:50 +08:00
|
|
|
|
|
|
|
// Print a list of changes. Generate an actual change that is the zone
|
|
|
|
changes := false
|
2017-03-28 06:03:01 +08:00
|
|
|
desc := ""
|
2016-08-23 08:31:50 +08:00
|
|
|
for _, i := range create {
|
|
|
|
changes = true
|
2017-03-28 06:03:01 +08:00
|
|
|
desc += "\n" + i.String()
|
2016-08-23 08:31:50 +08:00
|
|
|
}
|
|
|
|
for _, i := range del {
|
|
|
|
changes = true
|
2017-03-28 06:03:01 +08:00
|
|
|
desc += "\n" + i.String()
|
2016-08-23 08:31:50 +08:00
|
|
|
}
|
|
|
|
for _, i := range mod {
|
|
|
|
changes = true
|
2017-03-28 06:03:01 +08:00
|
|
|
desc += "\n" + i.String()
|
2016-08-23 08:31:50 +08:00
|
|
|
}
|
|
|
|
|
2017-03-28 06:03:01 +08:00
|
|
|
msg := fmt.Sprintf("GENERATE_ZONE: %s (%d records)%s", dc.Name, len(dc.Records), desc)
|
2016-08-23 08:31:50 +08:00
|
|
|
corrections := []*models.Correction{}
|
|
|
|
if changes {
|
|
|
|
corrections = append(corrections,
|
|
|
|
&models.Correction{
|
|
|
|
Msg: msg,
|
|
|
|
F: func() error {
|
|
|
|
fmt.Printf("CREATING ZONE: %v\n", dc.Name)
|
|
|
|
return c.createGandiZone(dc.Name, domaininfo.ZoneId, expectedRecordSets)
|
|
|
|
},
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
return corrections, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func newGandi(m map[string]string, metadata json.RawMessage) (providers.DNSServiceProvider, error) {
|
|
|
|
api := &GandiApi{}
|
|
|
|
api.ApiKey = m["apikey"]
|
|
|
|
if api.ApiKey == "" {
|
|
|
|
return nil, fmt.Errorf("Gandi apikey must be provided.")
|
|
|
|
}
|
|
|
|
|
|
|
|
return api, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
providers.RegisterDomainServiceProviderType("GANDI", newGandi)
|
|
|
|
}
|