2016-08-23 08:31:50 +08:00
|
|
|
package gandi
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
|
|
|
|
gandiclient "github.com/prasmussen/gandi-api/client"
|
|
|
|
gandidomain "github.com/prasmussen/gandi-api/domain"
|
|
|
|
gandizone "github.com/prasmussen/gandi-api/domain/zone"
|
|
|
|
gandirecord "github.com/prasmussen/gandi-api/domain/zone/record"
|
|
|
|
gandiversion "github.com/prasmussen/gandi-api/domain/zone/version"
|
2017-01-12 03:38:07 +08:00
|
|
|
|
|
|
|
"github.com/StackExchange/dnscontrol/models"
|
2017-03-28 06:03:01 +08:00
|
|
|
"github.com/miekg/dns/dnsutil"
|
2016-08-23 08:31:50 +08:00
|
|
|
)
|
|
|
|
|
|
|
|
// fetchDomainList gets list of domains for account. Cache ids for easy lookup.
|
|
|
|
func (c *GandiApi) fetchDomainList() error {
|
2016-12-17 04:10:27 +08:00
|
|
|
if c.domainIndex != nil {
|
|
|
|
return nil
|
|
|
|
}
|
2016-08-23 08:31:50 +08:00
|
|
|
c.domainIndex = map[string]int64{}
|
|
|
|
gc := gandiclient.New(c.ApiKey, gandiclient.Production)
|
|
|
|
domain := gandidomain.New(gc)
|
|
|
|
domains, err := domain.List()
|
|
|
|
if err != nil {
|
|
|
|
// fmt.Println(err)
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
for _, d := range domains {
|
|
|
|
c.domainIndex[d.Fqdn] = d.Id
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// fetchDomainInfo gets information about a domain.
|
|
|
|
func (c *GandiApi) fetchDomainInfo(fqdn string) (*gandidomain.DomainInfo, error) {
|
|
|
|
gc := gandiclient.New(c.ApiKey, gandiclient.Production)
|
|
|
|
domain := gandidomain.New(gc)
|
|
|
|
return domain.Info(fqdn)
|
|
|
|
}
|
|
|
|
|
|
|
|
// getRecordsForDomain returns a list of records for a zone.
|
2017-03-28 06:03:01 +08:00
|
|
|
func (c *GandiApi) getZoneRecords(zoneid int64, origin string) ([]*models.RecordConfig, error) {
|
2016-08-23 08:31:50 +08:00
|
|
|
gc := gandiclient.New(c.ApiKey, gandiclient.Production)
|
|
|
|
record := gandirecord.New(gc)
|
2017-01-12 03:38:07 +08:00
|
|
|
recs, err := record.List(zoneid, 0)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
rcs := make([]*models.RecordConfig, 0, len(recs))
|
|
|
|
for _, r := range recs {
|
2017-03-28 06:03:01 +08:00
|
|
|
rcs = append(rcs, convert(r, origin))
|
2017-01-12 03:38:07 +08:00
|
|
|
}
|
|
|
|
return rcs, nil
|
2016-08-23 08:31:50 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// listZones retrieves the list of zones.
|
|
|
|
func (c *GandiApi) listZones() ([]*gandizone.ZoneInfoBase, error) {
|
|
|
|
gc := gandiclient.New(c.ApiKey, gandiclient.Production)
|
|
|
|
zone := gandizone.New(gc)
|
|
|
|
return zone.List()
|
|
|
|
}
|
|
|
|
|
|
|
|
// setZone assigns a particular zone to a domain.
|
|
|
|
func (c *GandiApi) setZones(domainname string, zone_id int64) (*gandidomain.DomainInfo, error) {
|
|
|
|
gc := gandiclient.New(c.ApiKey, gandiclient.Production)
|
|
|
|
zone := gandizone.New(gc)
|
|
|
|
return zone.Set(domainname, zone_id)
|
|
|
|
}
|
|
|
|
|
|
|
|
// getZoneInfo gets ZoneInfo about a zone.
|
|
|
|
func (c *GandiApi) getZoneInfo(zoneid int64) (*gandizone.ZoneInfo, error) {
|
|
|
|
gc := gandiclient.New(c.ApiKey, gandiclient.Production)
|
|
|
|
zone := gandizone.New(gc)
|
|
|
|
return zone.Info(zoneid)
|
|
|
|
}
|
|
|
|
|
|
|
|
// createZone creates an entirely new zone.
|
|
|
|
func (c *GandiApi) createZone(name string) (*gandizone.ZoneInfo, error) {
|
|
|
|
gc := gandiclient.New(c.ApiKey, gandiclient.Production)
|
|
|
|
zone := gandizone.New(gc)
|
|
|
|
return zone.Create(name)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *GandiApi) getEditableZone(domainname string, zoneinfo *gandizone.ZoneInfo) (int64, error) {
|
|
|
|
var zone_id int64
|
|
|
|
if zoneinfo.Domains < 2 {
|
|
|
|
// If there is only on{ domain linked to this zone, use it.
|
|
|
|
zone_id = zoneinfo.Id
|
|
|
|
fmt.Printf("Using zone id=%d named %#v\n", zone_id, zoneinfo.Name)
|
|
|
|
return zone_id, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// We can't use the zone_id given to us. Let's make/find a new one.
|
|
|
|
zones, err := c.listZones()
|
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
zonename := fmt.Sprintf("%s dnscontrol", domainname)
|
|
|
|
for _, z := range zones {
|
|
|
|
if z.Name == zonename {
|
|
|
|
zone_id = z.Id
|
|
|
|
fmt.Printf("Recycling zone id=%d named %#v\n", zone_id, z.Name)
|
|
|
|
return zone_id, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
zoneinfo, err = c.createZone(zonename)
|
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
zone_id = zoneinfo.Id
|
|
|
|
fmt.Printf("Created zone id=%d named %#v\n", zone_id, zoneinfo.Name)
|
|
|
|
return zone_id, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// makeEditableZone
|
|
|
|
func (c *GandiApi) makeEditableZone(zone_id int64) (int64, error) {
|
|
|
|
gc := gandiclient.New(c.ApiKey, gandiclient.Production)
|
|
|
|
version := gandiversion.New(gc)
|
|
|
|
return version.New(zone_id, 0)
|
|
|
|
}
|
|
|
|
|
|
|
|
// setZoneRecords
|
|
|
|
func (c *GandiApi) setZoneRecords(zone_id, version_id int64, records []gandirecord.RecordSet) ([]*gandirecord.RecordInfo, error) {
|
|
|
|
gc := gandiclient.New(c.ApiKey, gandiclient.Production)
|
|
|
|
record := gandirecord.New(gc)
|
|
|
|
return record.SetRecords(zone_id, version_id, records)
|
|
|
|
}
|
|
|
|
|
|
|
|
// activateVersion
|
|
|
|
func (c *GandiApi) activateVersion(zone_id, version_id int64) (bool, error) {
|
|
|
|
gc := gandiclient.New(c.ApiKey, gandiclient.Production)
|
|
|
|
version := gandiversion.New(gc)
|
|
|
|
return version.Set(zone_id, version_id)
|
|
|
|
}
|
|
|
|
|
2017-03-28 06:03:01 +08:00
|
|
|
func (c *GandiApi) createGandiZone(domainname string, zoneID int64, records []gandirecord.RecordSet) error {
|
2016-08-23 08:31:50 +08:00
|
|
|
|
|
|
|
// Get the zone_id of the zone we'll be updating.
|
2017-03-28 06:03:01 +08:00
|
|
|
zoneinfo, err := c.getZoneInfo(zoneID)
|
2016-08-23 08:31:50 +08:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
//fmt.Println("ZONEINFO:", zoneinfo)
|
2017-03-28 06:03:01 +08:00
|
|
|
zoneID, err = c.getEditableZone(domainname, zoneinfo)
|
2016-08-23 08:31:50 +08:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2017-03-28 06:03:01 +08:00
|
|
|
// Get the versionID of the zone we're updating.
|
|
|
|
versionID, err := c.makeEditableZone(zoneID)
|
2016-08-23 08:31:50 +08:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update the new version.
|
2017-03-28 06:03:01 +08:00
|
|
|
_, err = c.setZoneRecords(zoneID, versionID, records)
|
2016-08-23 08:31:50 +08:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Activate zone version
|
2017-03-28 06:03:01 +08:00
|
|
|
_, err = c.activateVersion(zoneID, versionID)
|
2016-08-23 08:31:50 +08:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2017-03-28 06:03:01 +08:00
|
|
|
_, err = c.setZones(domainname, zoneID)
|
2016-08-23 08:31:50 +08:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
2017-01-12 03:38:07 +08:00
|
|
|
|
2017-08-23 03:45:40 +08:00
|
|
|
// convert takes a DNS record from Gandi and returns our native RecordConfig format.
|
2017-03-28 06:03:01 +08:00
|
|
|
func convert(r *gandirecord.RecordInfo, origin string) *models.RecordConfig {
|
2017-08-06 01:21:57 +08:00
|
|
|
rc := &models.RecordConfig{
|
2017-03-28 06:03:01 +08:00
|
|
|
NameFQDN: dnsutil.AddOrigin(r.Name, origin),
|
|
|
|
Name: r.Name,
|
2017-01-12 03:38:07 +08:00
|
|
|
Type: r.Type,
|
|
|
|
Original: r,
|
|
|
|
Target: r.Value,
|
|
|
|
TTL: uint32(r.Ttl),
|
|
|
|
}
|
2017-08-06 01:21:57 +08:00
|
|
|
switch r.Type {
|
2017-08-23 03:45:40 +08:00
|
|
|
case "A", "AAAA", "NS", "CNAME", "PTR", "TXT":
|
2017-08-29 22:10:50 +08:00
|
|
|
case "SRV":
|
|
|
|
var err error
|
|
|
|
rc.SrvPriority, rc.SrvWeight, rc.SrvPort, rc.Target, err = models.SplitCombinedSrvValue(r.Value)
|
|
|
|
if err != nil {
|
|
|
|
panic(fmt.Sprintf("gandi.convert bad srv value format: %#v (%s)", r.Value, err))
|
|
|
|
}
|
2017-08-06 01:21:57 +08:00
|
|
|
// no-op
|
|
|
|
case "MX":
|
|
|
|
var err error
|
|
|
|
rc.MxPreference, rc.Target, err = models.SplitCombinedMxValue(r.Value)
|
|
|
|
if err != nil {
|
|
|
|
panic(fmt.Sprintf("gandi.convert bad mx value format: %#v", r.Value))
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
panic(fmt.Sprintf("gandi.convert unimplemented rtype %v", r.Type))
|
|
|
|
// We panic so that we quickly find any switch statements
|
|
|
|
// that have not been updated for a new RR type.
|
|
|
|
}
|
|
|
|
return rc
|
2017-01-12 03:38:07 +08:00
|
|
|
}
|