2016-08-23 08:31:50 +08:00
|
|
|
package route53
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
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
|
|
|
"errors"
|
2016-08-23 08:31:50 +08:00
|
|
|
"fmt"
|
2020-11-12 13:30:40 +08:00
|
|
|
"log"
|
2017-08-05 22:56:42 +08:00
|
|
|
"sort"
|
2016-08-23 08:31:50 +08:00
|
|
|
"strings"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/aws/aws-sdk-go/aws"
|
|
|
|
"github.com/aws/aws-sdk-go/aws/credentials"
|
|
|
|
"github.com/aws/aws-sdk-go/aws/session"
|
|
|
|
r53 "github.com/aws/aws-sdk-go/service/route53"
|
2017-08-05 22:56:42 +08:00
|
|
|
r53d "github.com/aws/aws-sdk-go/service/route53domains"
|
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
|
|
|
|
2020-04-15 04:47:30 +08:00
|
|
|
"github.com/StackExchange/dnscontrol/v3/models"
|
|
|
|
"github.com/StackExchange/dnscontrol/v3/pkg/diff"
|
|
|
|
"github.com/StackExchange/dnscontrol/v3/providers"
|
2016-08-23 08:31:50 +08:00
|
|
|
)
|
|
|
|
|
2020-10-26 21:25:30 +08:00
|
|
|
type route53Provider struct {
|
2020-02-18 21:59:18 +08:00
|
|
|
client *r53.Route53
|
|
|
|
registrar *r53d.Route53Domains
|
|
|
|
delegationSet *string
|
|
|
|
zones map[string]*r53.HostedZone
|
|
|
|
originalRecords []*r53.ResourceRecordSet
|
2016-08-23 08:31:50 +08:00
|
|
|
}
|
|
|
|
|
2017-08-05 22:56:42 +08:00
|
|
|
func newRoute53Reg(conf map[string]string) (providers.Registrar, error) {
|
|
|
|
return newRoute53(conf, nil)
|
|
|
|
}
|
|
|
|
|
|
|
|
func newRoute53Dsp(conf map[string]string, metadata json.RawMessage) (providers.DNSServiceProvider, error) {
|
|
|
|
return newRoute53(conf, metadata)
|
|
|
|
}
|
|
|
|
|
2020-10-26 21:25:30 +08:00
|
|
|
func newRoute53(m map[string]string, metadata json.RawMessage) (*route53Provider, error) {
|
2018-09-05 00:58:02 +08:00
|
|
|
keyID, secretKey, tokenID := m["KeyId"], m["SecretKey"], m["Token"]
|
2017-05-04 01:30:05 +08:00
|
|
|
|
2017-08-05 22:56:42 +08:00
|
|
|
// Route53 uses a global endpoint and route53domains
|
|
|
|
// currently only has a single regional endpoint in us-east-1
|
|
|
|
// http://docs.aws.amazon.com/general/latest/gr/rande.html#r53_region
|
2017-05-04 01:30:05 +08:00
|
|
|
config := &aws.Config{
|
2017-08-05 22:56:42 +08:00
|
|
|
Region: aws.String("us-east-1"),
|
2017-05-04 01:30:05 +08:00
|
|
|
}
|
|
|
|
|
2018-09-05 00:58:02 +08:00
|
|
|
// Token is optional and left empty unless required
|
2018-01-10 01:53:16 +08:00
|
|
|
if keyID != "" || secretKey != "" {
|
2018-09-05 00:58:02 +08:00
|
|
|
config.Credentials = credentials.NewStaticCredentials(keyID, secretKey, tokenID)
|
2016-08-23 08:31:50 +08:00
|
|
|
}
|
2019-12-04 23:17:28 +08:00
|
|
|
sess := session.Must(session.NewSession(config))
|
2016-08-23 08:31:50 +08:00
|
|
|
|
2020-07-07 08:18:24 +08:00
|
|
|
var dls *string
|
2019-02-20 01:30:39 +08:00
|
|
|
if val, ok := m["DelegationSet"]; ok {
|
|
|
|
fmt.Printf("ROUTE53 DelegationSet %s configured\n", val)
|
|
|
|
dls = sPtr(val)
|
|
|
|
}
|
2020-10-26 21:25:30 +08:00
|
|
|
api := &route53Provider{client: r53.New(sess), registrar: r53d.New(sess), delegationSet: dls}
|
2017-05-04 01:30:05 +08:00
|
|
|
err := api.getZones()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2016-08-23 08:31:50 +08:00
|
|
|
return api, nil
|
|
|
|
}
|
|
|
|
|
2018-01-05 08:19:35 +08:00
|
|
|
var features = providers.DocumentationNotes{
|
ROUTE53: Support Route53's ALIAS record type (#239) (#301)
* Stable comparison of metadata (#239)
Iterating over a map in Go never produces twice the same ordering.
Thus when comparing two metadata map with more than one key, the
`differ` is always finding differences.
To properly compare records metadata, we need to iterate the maps
in a deterministic way.
Signed-off-by: Brice Figureau <brice@daysofwonder.com>
* Support for Route53 ALIAS record type (#239)
Route53 ALIAS doesn't behave like a regular ALIAS, and is much more
limited as its target can only be some specific AWS resources or
another record in the same zone.
According to #239, this change adds a new directive R53_ALIAS which
implements this specific alias. This record type can only be used
with the Route53 provider.
This directive usage looks like this:
```js
D("example.com", REGISTRAR, DnsProvider("ROUTE53"),
R53_ALIAS("foo1", "A", "bar") // record in same zone
R53_ALIAS("foo2", "A",
"blahblah.elasticloadbalancing.us-west-1.amazonaws.com",
R53_ZONE('Z368ELLRRE2KJ0')) // ELB in us-west-1
```
Unfortunately, Route53 requires indicating the hosted zone id
where the target is defined (those are listed in AWS documentation,
see the R53_ALIAS documentation for links).
2018-01-16 18:53:12 +08:00
|
|
|
providers.CanUseAlias: providers.Cannot("R53 does not provide a generic ALIAS functionality. Use R53_ALIAS instead."),
|
2017-09-15 04:13:17 +08:00
|
|
|
providers.DocCreateDomains: providers.Can(),
|
2018-01-05 08:19:35 +08:00
|
|
|
providers.DocDualHost: providers.Can(),
|
2017-09-15 04:13:17 +08:00
|
|
|
providers.DocOfficiallySupported: providers.Can(),
|
2018-01-05 08:19:35 +08:00
|
|
|
providers.CanUsePTR: providers.Can(),
|
|
|
|
providers.CanUseSRV: providers.Can(),
|
|
|
|
providers.CanUseTXTMulti: providers.Can(),
|
|
|
|
providers.CanUseCAA: providers.Can(),
|
ROUTE53: Support Route53's ALIAS record type (#239) (#301)
* Stable comparison of metadata (#239)
Iterating over a map in Go never produces twice the same ordering.
Thus when comparing two metadata map with more than one key, the
`differ` is always finding differences.
To properly compare records metadata, we need to iterate the maps
in a deterministic way.
Signed-off-by: Brice Figureau <brice@daysofwonder.com>
* Support for Route53 ALIAS record type (#239)
Route53 ALIAS doesn't behave like a regular ALIAS, and is much more
limited as its target can only be some specific AWS resources or
another record in the same zone.
According to #239, this change adds a new directive R53_ALIAS which
implements this specific alias. This record type can only be used
with the Route53 provider.
This directive usage looks like this:
```js
D("example.com", REGISTRAR, DnsProvider("ROUTE53"),
R53_ALIAS("foo1", "A", "bar") // record in same zone
R53_ALIAS("foo2", "A",
"blahblah.elasticloadbalancing.us-west-1.amazonaws.com",
R53_ZONE('Z368ELLRRE2KJ0')) // ELB in us-west-1
```
Unfortunately, Route53 requires indicating the hosted zone id
where the target is defined (those are listed in AWS documentation,
see the R53_ALIAS documentation for links).
2018-01-16 18:53:12 +08:00
|
|
|
providers.CanUseRoute53Alias: providers.Can(),
|
2020-02-18 21:59:18 +08:00
|
|
|
providers.CanGetZones: providers.Can(),
|
2017-09-15 04:13:17 +08:00
|
|
|
}
|
|
|
|
|
2016-08-23 08:31:50 +08:00
|
|
|
func init() {
|
2018-01-05 08:19:35 +08:00
|
|
|
providers.RegisterDomainServiceProviderType("ROUTE53", newRoute53Dsp, features)
|
2017-08-05 22:56:42 +08:00
|
|
|
providers.RegisterRegistrarType("ROUTE53", newRoute53Reg)
|
2018-02-01 20:39:11 +08:00
|
|
|
providers.RegisterCustomRecordType("R53_ALIAS", "ROUTE53", "")
|
2016-08-23 08:31:50 +08:00
|
|
|
}
|
2017-08-05 22:56:42 +08:00
|
|
|
|
2016-08-23 08:31:50 +08:00
|
|
|
func sPtr(s string) *string {
|
|
|
|
return &s
|
|
|
|
}
|
2017-01-12 03:38:07 +08:00
|
|
|
|
2019-02-20 01:30:39 +08:00
|
|
|
func withRetry(f func() error) {
|
|
|
|
const maxRetries = 23
|
|
|
|
// TODO: exponential backoff
|
|
|
|
const sleepTime = 5 * time.Second
|
2020-07-07 08:18:24 +08:00
|
|
|
var currentRetry int
|
2019-02-20 01:30:39 +08:00
|
|
|
for {
|
|
|
|
err := f()
|
|
|
|
if err == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if strings.Contains(err.Error(), "Rate exceeded") {
|
|
|
|
currentRetry++
|
|
|
|
if currentRetry >= maxRetries {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
fmt.Printf("============ Route53 rate limit exceeded. Waiting %s to retry.\n", sleepTime)
|
|
|
|
time.Sleep(sleepTime)
|
|
|
|
} else {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-02-21 04:07:34 +08:00
|
|
|
// ListZones lists the zones on this account.
|
2020-10-26 21:25:30 +08:00
|
|
|
func (r *route53Provider) ListZones() ([]string, error) {
|
2020-02-21 04:07:34 +08:00
|
|
|
var zones []string
|
|
|
|
// Assumes r.zones was filled already by newRoute53().
|
|
|
|
for i := range r.zones {
|
|
|
|
zones = append(zones, i)
|
|
|
|
}
|
|
|
|
return zones, nil
|
|
|
|
}
|
|
|
|
|
2020-10-26 21:25:30 +08:00
|
|
|
func (r *route53Provider) getZones() error {
|
2016-08-23 08:31:50 +08:00
|
|
|
var nextMarker *string
|
|
|
|
r.zones = make(map[string]*r53.HostedZone)
|
|
|
|
for {
|
2019-02-20 01:30:39 +08:00
|
|
|
var out *r53.ListHostedZonesOutput
|
|
|
|
var err error
|
|
|
|
withRetry(func() error {
|
|
|
|
inp := &r53.ListHostedZonesInput{Marker: nextMarker}
|
|
|
|
out, err = r.client.ListHostedZones(inp)
|
|
|
|
return err
|
|
|
|
})
|
2017-05-04 01:30:05 +08:00
|
|
|
if err != nil && strings.Contains(err.Error(), "is not authorized") {
|
2020-08-31 07:52:37 +08:00
|
|
|
return errors.New("check your credentials, you're not authorized to perform actions on Route 53 AWS Service")
|
2017-05-04 01:30:05 +08:00
|
|
|
} else if err != nil {
|
2016-08-23 08:31:50 +08:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
for _, z := range out.HostedZones {
|
|
|
|
domain := strings.TrimSuffix(*z.Name, ".")
|
|
|
|
r.zones[domain] = z
|
|
|
|
}
|
|
|
|
if out.NextMarker != nil {
|
|
|
|
nextMarker = out.NextMarker
|
|
|
|
} else {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2017-01-04 04:26:08 +08:00
|
|
|
type errNoExist struct {
|
|
|
|
domain string
|
|
|
|
}
|
|
|
|
|
|
|
|
func (e errNoExist) Error() string {
|
|
|
|
return fmt.Sprintf("Domain %s not found in your route 53 account", e.domain)
|
|
|
|
}
|
|
|
|
|
2020-10-26 21:25:30 +08:00
|
|
|
func (r *route53Provider) GetNameservers(domain string) ([]*models.Nameserver, error) {
|
2017-05-04 01:30:05 +08:00
|
|
|
|
2016-12-17 04:10:27 +08:00
|
|
|
zone, ok := r.zones[domain]
|
|
|
|
if !ok {
|
2017-01-04 04:26:08 +08:00
|
|
|
return nil, errNoExist{domain}
|
2016-12-17 04:10:27 +08:00
|
|
|
}
|
2019-02-20 01:30:39 +08:00
|
|
|
var z *r53.GetHostedZoneOutput
|
|
|
|
var err error
|
|
|
|
withRetry(func() error {
|
|
|
|
z, err = r.client.GetHostedZone(&r53.GetHostedZoneInput{Id: zone.Id})
|
|
|
|
return err
|
|
|
|
})
|
2016-12-17 04:10:27 +08:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2020-03-01 23:33:24 +08:00
|
|
|
|
|
|
|
var nss []string
|
2017-05-04 01:30:05 +08:00
|
|
|
if z.DelegationSet != nil {
|
|
|
|
for _, nsPtr := range z.DelegationSet.NameServers {
|
2020-03-01 23:33:24 +08:00
|
|
|
nss = append(nss, *nsPtr)
|
2017-05-04 01:30:05 +08:00
|
|
|
}
|
2016-12-17 04:10:27 +08:00
|
|
|
}
|
2020-03-01 23:33:24 +08:00
|
|
|
return models.ToNameservers(nss)
|
2016-12-17 04:10:27 +08:00
|
|
|
}
|
|
|
|
|
2020-02-18 21:59:18 +08:00
|
|
|
// GetZoneRecords gets the records of a zone and returns them in RecordConfig format.
|
2020-10-26 21:25:30 +08:00
|
|
|
func (r *route53Provider) GetZoneRecords(domain string) (models.Records, error) {
|
2016-12-17 04:10:27 +08:00
|
|
|
|
2020-02-18 21:59:18 +08:00
|
|
|
zone, ok := r.zones[domain]
|
2016-08-23 08:31:50 +08:00
|
|
|
if !ok {
|
2020-02-18 21:59:18 +08:00
|
|
|
return nil, errNoExist{domain}
|
2016-08-23 08:31:50 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
records, err := r.fetchRecordSets(zone.Id)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2020-02-18 21:59:18 +08:00
|
|
|
r.originalRecords = records
|
2016-08-23 08:31:50 +08:00
|
|
|
|
2017-01-12 03:38:07 +08:00
|
|
|
var existingRecords = []*models.RecordConfig{}
|
2016-08-23 08:31:50 +08:00
|
|
|
for _, set := range records {
|
2020-11-14 05:32:40 +08:00
|
|
|
rts, err := nativeToRecords(set, domain)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
existingRecords = append(existingRecords, rts...)
|
2016-08-23 08:31:50 +08:00
|
|
|
}
|
2020-02-18 21:59:18 +08:00
|
|
|
return existingRecords, nil
|
|
|
|
}
|
|
|
|
|
2020-10-26 21:25:30 +08:00
|
|
|
func (r *route53Provider) GetDomainCorrections(dc *models.DomainConfig) ([]*models.Correction, error) {
|
2020-02-18 21:59:18 +08:00
|
|
|
dc.Punycode()
|
|
|
|
|
|
|
|
var corrections = []*models.Correction{}
|
|
|
|
|
|
|
|
existingRecords, err := r.GetZoneRecords(dc.Name)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
zone, ok := r.zones[dc.Name]
|
|
|
|
if !ok {
|
|
|
|
return nil, errNoExist{dc.Name}
|
|
|
|
}
|
|
|
|
|
2016-08-23 08:31:50 +08:00
|
|
|
for _, want := range dc.Records {
|
ROUTE53: Support Route53's ALIAS record type (#239) (#301)
* Stable comparison of metadata (#239)
Iterating over a map in Go never produces twice the same ordering.
Thus when comparing two metadata map with more than one key, the
`differ` is always finding differences.
To properly compare records metadata, we need to iterate the maps
in a deterministic way.
Signed-off-by: Brice Figureau <brice@daysofwonder.com>
* Support for Route53 ALIAS record type (#239)
Route53 ALIAS doesn't behave like a regular ALIAS, and is much more
limited as its target can only be some specific AWS resources or
another record in the same zone.
According to #239, this change adds a new directive R53_ALIAS which
implements this specific alias. This record type can only be used
with the Route53 provider.
This directive usage looks like this:
```js
D("example.com", REGISTRAR, DnsProvider("ROUTE53"),
R53_ALIAS("foo1", "A", "bar") // record in same zone
R53_ALIAS("foo2", "A",
"blahblah.elasticloadbalancing.us-west-1.amazonaws.com",
R53_ZONE('Z368ELLRRE2KJ0')) // ELB in us-west-1
```
Unfortunately, Route53 requires indicating the hosted zone id
where the target is defined (those are listed in AWS documentation,
see the R53_ALIAS documentation for links).
2018-01-16 18:53:12 +08:00
|
|
|
// update zone_id to current zone.id if not specified by the user
|
|
|
|
if want.Type == "R53_ALIAS" && want.R53Alias["zone_id"] == "" {
|
|
|
|
want.R53Alias["zone_id"] = getZoneID(zone, want)
|
|
|
|
}
|
2016-08-23 08:31:50 +08:00
|
|
|
}
|
|
|
|
|
2017-11-08 06:12:17 +08:00
|
|
|
// Normalize
|
2018-01-05 08:19:35 +08:00
|
|
|
models.PostProcessRecords(existingRecords)
|
2017-11-08 06:12:17 +08:00
|
|
|
|
2018-01-10 01:53:16 +08:00
|
|
|
// diff
|
ROUTE53: Support Route53's ALIAS record type (#239) (#301)
* Stable comparison of metadata (#239)
Iterating over a map in Go never produces twice the same ordering.
Thus when comparing two metadata map with more than one key, the
`differ` is always finding differences.
To properly compare records metadata, we need to iterate the maps
in a deterministic way.
Signed-off-by: Brice Figureau <brice@daysofwonder.com>
* Support for Route53 ALIAS record type (#239)
Route53 ALIAS doesn't behave like a regular ALIAS, and is much more
limited as its target can only be some specific AWS resources or
another record in the same zone.
According to #239, this change adds a new directive R53_ALIAS which
implements this specific alias. This record type can only be used
with the Route53 provider.
This directive usage looks like this:
```js
D("example.com", REGISTRAR, DnsProvider("ROUTE53"),
R53_ALIAS("foo1", "A", "bar") // record in same zone
R53_ALIAS("foo2", "A",
"blahblah.elasticloadbalancing.us-west-1.amazonaws.com",
R53_ZONE('Z368ELLRRE2KJ0')) // ELB in us-west-1
```
Unfortunately, Route53 requires indicating the hosted zone id
where the target is defined (those are listed in AWS documentation,
see the R53_ALIAS documentation for links).
2018-01-16 18:53:12 +08:00
|
|
|
differ := diff.New(dc, getAliasMap)
|
2020-08-21 03:49:00 +08:00
|
|
|
namesToUpdate, err := differ.ChangedGroups(existingRecords)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2016-08-23 08:31:50 +08:00
|
|
|
|
|
|
|
if len(namesToUpdate) == 0 {
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
2018-09-04 22:55:27 +08:00
|
|
|
updates := map[models.RecordKey][]*models.RecordConfig{}
|
2019-06-27 11:45:34 +08:00
|
|
|
|
|
|
|
// for each name we need to update, collect relevant records from our desired domain state
|
2016-08-23 08:31:50 +08:00
|
|
|
for k := range namesToUpdate {
|
|
|
|
updates[k] = nil
|
|
|
|
for _, rc := range dc.Records {
|
2018-09-04 22:55:27 +08:00
|
|
|
if rc.Key() == k {
|
2016-08-23 08:31:50 +08:00
|
|
|
updates[k] = append(updates[k], rc)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-12 13:30:40 +08:00
|
|
|
// updateOrder is the order that the updates will happen.
|
|
|
|
// The order should be sorted by NameFQDN, then Type, with R53_ALIAS_*
|
|
|
|
// types sorted after all other types. R53_ALIAS_* needs to be last
|
|
|
|
// because they are order dependent (aliases must refer to labels
|
|
|
|
// that already exist).
|
|
|
|
var updateOrder []models.RecordKey
|
|
|
|
// Collect the keys
|
2020-11-14 05:32:40 +08:00
|
|
|
for k := range updates {
|
2020-11-12 13:30:40 +08:00
|
|
|
updateOrder = append(updateOrder, k)
|
|
|
|
}
|
|
|
|
// Sort themm
|
|
|
|
sort.Slice(updateOrder, func(i, j int) bool {
|
|
|
|
if updateOrder[i].Type == updateOrder[j].Type {
|
|
|
|
return updateOrder[i].NameFQDN < updateOrder[j].NameFQDN
|
|
|
|
}
|
|
|
|
|
|
|
|
if strings.HasPrefix(updateOrder[i].Type, "R53_ALIAS_") {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
if strings.HasPrefix(updateOrder[j].Type, "R53_ALIAS_") {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
if updateOrder[i].NameFQDN == updateOrder[j].NameFQDN {
|
|
|
|
return updateOrder[i].Type < updateOrder[j].Type
|
|
|
|
}
|
|
|
|
return updateOrder[i].NameFQDN < updateOrder[j].NameFQDN
|
|
|
|
})
|
|
|
|
|
2019-06-27 11:45:34 +08:00
|
|
|
// we collect all changes into one of two categories now:
|
|
|
|
// pure deletions where we delete an entire record set,
|
|
|
|
// or changes where we upsert an entire record set.
|
2017-03-23 03:08:23 +08:00
|
|
|
dels := []*r53.Change{}
|
2020-11-12 13:30:40 +08:00
|
|
|
delDesc := []string{}
|
2016-08-23 08:31:50 +08:00
|
|
|
changes := []*r53.Change{}
|
2020-01-10 21:47:20 +08:00
|
|
|
changeDesc := []string{}
|
2019-06-27 11:45:34 +08:00
|
|
|
|
2020-11-12 13:30:40 +08:00
|
|
|
for _, k := range updateOrder {
|
|
|
|
recs := updates[k]
|
|
|
|
// If there are no records in our desired state for a key, this
|
|
|
|
// indicates we should delete all records at that key.
|
2016-08-23 08:31:50 +08:00
|
|
|
if len(recs) == 0 {
|
2020-11-12 13:30:40 +08:00
|
|
|
// To delete, we submit the original resource set we got from r53.
|
|
|
|
var rrset *r53.ResourceRecordSet
|
|
|
|
// Find the original resource set:
|
2020-02-18 21:59:18 +08:00
|
|
|
for _, r := range r.originalRecords {
|
2018-09-08 01:46:44 +08:00
|
|
|
if unescape(r.Name) == k.NameFQDN && (*r.Type == k.Type || k.Type == "R53_ALIAS_"+*r.Type) {
|
2016-08-23 08:31:50 +08:00
|
|
|
rrset = r
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
2018-08-24 05:46:50 +08:00
|
|
|
if rrset == nil {
|
2020-11-12 13:30:40 +08:00
|
|
|
// This should not happen.
|
2020-08-31 07:52:37 +08:00
|
|
|
return nil, fmt.Errorf("no record set found to delete. Name: '%s'. Type: '%s'", k.NameFQDN, k.Type)
|
2018-08-24 05:46:50 +08:00
|
|
|
}
|
2020-11-12 13:30:40 +08:00
|
|
|
// Assemble the change and add it to the list:
|
|
|
|
chg := &r53.Change{
|
|
|
|
Action: sPtr("DELETE"),
|
|
|
|
ResourceRecordSet: rrset,
|
2016-08-23 08:31:50 +08:00
|
|
|
}
|
2020-11-12 13:30:40 +08:00
|
|
|
dels = append(dels, chg)
|
|
|
|
delDesc = append(delDesc, strings.Join(namesToUpdate[k], "\n"))
|
|
|
|
} else {
|
|
|
|
// If it isn't a delete, it must be either a change or create. In
|
|
|
|
// either case, we build a new record set from the desired state and
|
|
|
|
// UPSERT it.
|
|
|
|
|
|
|
|
if strings.HasPrefix(k.Type, "R53_ALIAS_") {
|
|
|
|
// Each R53_ALIAS_* requires an individual change.
|
|
|
|
if len(recs) != 1 {
|
|
|
|
log.Fatal("Only one R53_ALIAS_ permitted on a label")
|
|
|
|
}
|
|
|
|
for _, r := range recs {
|
|
|
|
rrset := aliasToRRSet(zone, r)
|
|
|
|
rrset.Name = sPtr(k.NameFQDN)
|
|
|
|
// Assemble the change and add it to the list:
|
|
|
|
chg := &r53.Change{
|
|
|
|
Action: sPtr("UPSERT"),
|
|
|
|
ResourceRecordSet: rrset,
|
|
|
|
}
|
|
|
|
changes = append(changes, chg)
|
|
|
|
changeDesc = append(changeDesc, strings.Join(namesToUpdate[k], "\n"))
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// All other keys combine their updates into one rrset:
|
|
|
|
rrset := &r53.ResourceRecordSet{
|
|
|
|
Name: sPtr(k.NameFQDN),
|
|
|
|
Type: sPtr(k.Type),
|
|
|
|
}
|
|
|
|
for _, r := range recs {
|
|
|
|
val := r.GetTargetCombined()
|
ROUTE53: Support Route53's ALIAS record type (#239) (#301)
* Stable comparison of metadata (#239)
Iterating over a map in Go never produces twice the same ordering.
Thus when comparing two metadata map with more than one key, the
`differ` is always finding differences.
To properly compare records metadata, we need to iterate the maps
in a deterministic way.
Signed-off-by: Brice Figureau <brice@daysofwonder.com>
* Support for Route53 ALIAS record type (#239)
Route53 ALIAS doesn't behave like a regular ALIAS, and is much more
limited as its target can only be some specific AWS resources or
another record in the same zone.
According to #239, this change adds a new directive R53_ALIAS which
implements this specific alias. This record type can only be used
with the Route53 provider.
This directive usage looks like this:
```js
D("example.com", REGISTRAR, DnsProvider("ROUTE53"),
R53_ALIAS("foo1", "A", "bar") // record in same zone
R53_ALIAS("foo2", "A",
"blahblah.elasticloadbalancing.us-west-1.amazonaws.com",
R53_ZONE('Z368ELLRRE2KJ0')) // ELB in us-west-1
```
Unfortunately, Route53 requires indicating the hosted zone id
where the target is defined (those are listed in AWS documentation,
see the R53_ALIAS documentation for links).
2018-01-16 18:53:12 +08:00
|
|
|
rr := &r53.ResourceRecord{
|
|
|
|
Value: &val,
|
|
|
|
}
|
|
|
|
rrset.ResourceRecords = append(rrset.ResourceRecords, rr)
|
|
|
|
i := int64(r.TTL)
|
|
|
|
rrset.TTL = &i // TODO: make sure that ttls are consistent within a set
|
2016-08-23 08:31:50 +08:00
|
|
|
}
|
2020-11-12 13:30:40 +08:00
|
|
|
// Assemble the change and add it to the list:
|
|
|
|
chg := &r53.Change{
|
|
|
|
Action: sPtr("UPSERT"),
|
|
|
|
ResourceRecordSet: rrset,
|
|
|
|
}
|
|
|
|
changes = append(changes, chg)
|
|
|
|
changeDesc = append(changeDesc, strings.Join(namesToUpdate[k], "\n"))
|
2016-08-23 08:31:50 +08:00
|
|
|
}
|
2020-11-12 13:30:40 +08:00
|
|
|
|
2016-08-23 08:31:50 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-07-15 04:03:04 +08:00
|
|
|
addCorrection := func(msg string, req *r53.ChangeResourceRecordSetsInput) {
|
2017-03-23 03:08:23 +08:00
|
|
|
corrections = append(corrections,
|
|
|
|
&models.Correction{
|
2017-07-15 04:03:04 +08:00
|
|
|
Msg: msg,
|
2017-03-23 03:08:23 +08:00
|
|
|
F: func() error {
|
2019-02-20 01:30:39 +08:00
|
|
|
var err error
|
2017-03-23 03:08:23 +08:00
|
|
|
req.HostedZoneId = zone.Id
|
2019-02-20 01:30:39 +08:00
|
|
|
withRetry(func() error {
|
|
|
|
_, err = r.client.ChangeResourceRecordSets(req)
|
|
|
|
return err
|
|
|
|
})
|
2017-03-23 03:08:23 +08:00
|
|
|
return err
|
|
|
|
},
|
|
|
|
})
|
|
|
|
}
|
2017-07-15 04:03:04 +08:00
|
|
|
|
2019-06-27 13:21:23 +08:00
|
|
|
getBatchSize := func(size, max int) int {
|
|
|
|
if size > max {
|
2019-06-27 11:45:34 +08:00
|
|
|
return max
|
|
|
|
}
|
|
|
|
return size
|
|
|
|
}
|
|
|
|
|
|
|
|
for len(dels) > 0 {
|
|
|
|
batchSize := getBatchSize(len(dels), 1000)
|
|
|
|
batch := dels[:batchSize]
|
|
|
|
dels = dels[batchSize:]
|
2020-01-10 21:47:20 +08:00
|
|
|
delDescBatch := delDesc[:batchSize]
|
|
|
|
delDesc = delDesc[batchSize:]
|
|
|
|
|
|
|
|
delDescBatchStr := "\n" + strings.Join(delDescBatch, "\n") + "\n"
|
|
|
|
|
2019-06-27 11:45:34 +08:00
|
|
|
delReq := &r53.ChangeResourceRecordSetsInput{
|
|
|
|
ChangeBatch: &r53.ChangeBatch{Changes: batch},
|
|
|
|
}
|
2020-01-10 21:47:20 +08:00
|
|
|
addCorrection(delDescBatchStr, delReq)
|
2017-03-23 03:08:23 +08:00
|
|
|
}
|
2017-07-15 04:03:04 +08:00
|
|
|
|
2019-06-27 11:45:34 +08:00
|
|
|
for len(changes) > 0 {
|
|
|
|
batchSize := getBatchSize(len(changes), 500)
|
|
|
|
batch := changes[:batchSize]
|
|
|
|
changes = changes[batchSize:]
|
2020-01-10 21:47:20 +08:00
|
|
|
changeDescBatch := changeDesc[:batchSize]
|
|
|
|
changeDesc = changeDesc[batchSize:]
|
|
|
|
changeDescBatchStr := "\n" + strings.Join(changeDescBatch, "\n") + "\n"
|
|
|
|
|
2019-06-27 11:45:34 +08:00
|
|
|
changeReq := &r53.ChangeResourceRecordSetsInput{
|
|
|
|
ChangeBatch: &r53.ChangeBatch{Changes: batch},
|
|
|
|
}
|
2020-01-10 21:47:20 +08:00
|
|
|
addCorrection(changeDescBatchStr, changeReq)
|
2017-03-23 03:08:23 +08:00
|
|
|
}
|
2016-08-23 08:31:50 +08:00
|
|
|
|
|
|
|
return corrections, nil
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2020-11-14 05:32:40 +08:00
|
|
|
func nativeToRecords(set *r53.ResourceRecordSet, origin string) ([]*models.RecordConfig, error) {
|
2018-02-16 01:02:50 +08:00
|
|
|
results := []*models.RecordConfig{}
|
|
|
|
if set.AliasTarget != nil {
|
|
|
|
rc := &models.RecordConfig{
|
|
|
|
Type: "R53_ALIAS",
|
|
|
|
TTL: 300,
|
|
|
|
R53Alias: map[string]string{
|
|
|
|
"type": *set.Type,
|
|
|
|
"zone_id": *set.AliasTarget.HostedZoneId,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
rc.SetLabelFromFQDN(unescape(set.Name), origin)
|
|
|
|
rc.SetTarget(aws.StringValue(set.AliasTarget.DNSName))
|
|
|
|
results = append(results, rc)
|
2018-03-09 00:59:40 +08:00
|
|
|
} else if set.TrafficPolicyInstanceId != nil {
|
|
|
|
// skip traffic policy records
|
2018-02-16 01:02:50 +08:00
|
|
|
} else {
|
|
|
|
for _, rec := range set.ResourceRecords {
|
|
|
|
switch rtype := *set.Type; rtype {
|
|
|
|
case "SOA":
|
|
|
|
continue
|
2020-07-18 01:02:42 +08:00
|
|
|
case "SPF":
|
|
|
|
// route53 uses a custom record type for SPF
|
|
|
|
rtype = "TXT"
|
|
|
|
fallthrough
|
2018-02-16 01:02:50 +08:00
|
|
|
default:
|
|
|
|
rc := &models.RecordConfig{TTL: uint32(*set.TTL)}
|
|
|
|
rc.SetLabelFromFQDN(unescape(set.Name), origin)
|
2020-07-18 01:02:42 +08:00
|
|
|
if err := rc.PopulateFromString(rtype, *rec.Value, origin); err != nil {
|
2020-11-14 05:32:40 +08:00
|
|
|
return nil, fmt.Errorf("unparsable record received from R53: %w", err)
|
2018-02-16 01:02:50 +08:00
|
|
|
}
|
|
|
|
results = append(results, rc)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2020-11-14 05:32:40 +08:00
|
|
|
return results, nil
|
2018-02-16 01:02:50 +08:00
|
|
|
}
|
|
|
|
|
ROUTE53: Support Route53's ALIAS record type (#239) (#301)
* Stable comparison of metadata (#239)
Iterating over a map in Go never produces twice the same ordering.
Thus when comparing two metadata map with more than one key, the
`differ` is always finding differences.
To properly compare records metadata, we need to iterate the maps
in a deterministic way.
Signed-off-by: Brice Figureau <brice@daysofwonder.com>
* Support for Route53 ALIAS record type (#239)
Route53 ALIAS doesn't behave like a regular ALIAS, and is much more
limited as its target can only be some specific AWS resources or
another record in the same zone.
According to #239, this change adds a new directive R53_ALIAS which
implements this specific alias. This record type can only be used
with the Route53 provider.
This directive usage looks like this:
```js
D("example.com", REGISTRAR, DnsProvider("ROUTE53"),
R53_ALIAS("foo1", "A", "bar") // record in same zone
R53_ALIAS("foo2", "A",
"blahblah.elasticloadbalancing.us-west-1.amazonaws.com",
R53_ZONE('Z368ELLRRE2KJ0')) // ELB in us-west-1
```
Unfortunately, Route53 requires indicating the hosted zone id
where the target is defined (those are listed in AWS documentation,
see the R53_ALIAS documentation for links).
2018-01-16 18:53:12 +08:00
|
|
|
func getAliasMap(r *models.RecordConfig) map[string]string {
|
|
|
|
if r.Type != "R53_ALIAS" {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return r.R53Alias
|
|
|
|
}
|
|
|
|
|
|
|
|
func aliasToRRSet(zone *r53.HostedZone, r *models.RecordConfig) *r53.ResourceRecordSet {
|
2020-11-12 13:30:40 +08:00
|
|
|
target := r.GetTargetField()
|
ROUTE53: Support Route53's ALIAS record type (#239) (#301)
* Stable comparison of metadata (#239)
Iterating over a map in Go never produces twice the same ordering.
Thus when comparing two metadata map with more than one key, the
`differ` is always finding differences.
To properly compare records metadata, we need to iterate the maps
in a deterministic way.
Signed-off-by: Brice Figureau <brice@daysofwonder.com>
* Support for Route53 ALIAS record type (#239)
Route53 ALIAS doesn't behave like a regular ALIAS, and is much more
limited as its target can only be some specific AWS resources or
another record in the same zone.
According to #239, this change adds a new directive R53_ALIAS which
implements this specific alias. This record type can only be used
with the Route53 provider.
This directive usage looks like this:
```js
D("example.com", REGISTRAR, DnsProvider("ROUTE53"),
R53_ALIAS("foo1", "A", "bar") // record in same zone
R53_ALIAS("foo2", "A",
"blahblah.elasticloadbalancing.us-west-1.amazonaws.com",
R53_ZONE('Z368ELLRRE2KJ0')) // ELB in us-west-1
```
Unfortunately, Route53 requires indicating the hosted zone id
where the target is defined (those are listed in AWS documentation,
see the R53_ALIAS documentation for links).
2018-01-16 18:53:12 +08:00
|
|
|
zoneID := getZoneID(zone, r)
|
|
|
|
targetHealth := false
|
2020-11-12 13:30:40 +08:00
|
|
|
rrset := &r53.ResourceRecordSet{
|
|
|
|
Type: sPtr(r.R53Alias["type"]),
|
|
|
|
AliasTarget: &r53.AliasTarget{
|
|
|
|
DNSName: &target,
|
|
|
|
HostedZoneId: aws.String(zoneID),
|
|
|
|
EvaluateTargetHealth: &targetHealth,
|
|
|
|
},
|
ROUTE53: Support Route53's ALIAS record type (#239) (#301)
* Stable comparison of metadata (#239)
Iterating over a map in Go never produces twice the same ordering.
Thus when comparing two metadata map with more than one key, the
`differ` is always finding differences.
To properly compare records metadata, we need to iterate the maps
in a deterministic way.
Signed-off-by: Brice Figureau <brice@daysofwonder.com>
* Support for Route53 ALIAS record type (#239)
Route53 ALIAS doesn't behave like a regular ALIAS, and is much more
limited as its target can only be some specific AWS resources or
another record in the same zone.
According to #239, this change adds a new directive R53_ALIAS which
implements this specific alias. This record type can only be used
with the Route53 provider.
This directive usage looks like this:
```js
D("example.com", REGISTRAR, DnsProvider("ROUTE53"),
R53_ALIAS("foo1", "A", "bar") // record in same zone
R53_ALIAS("foo2", "A",
"blahblah.elasticloadbalancing.us-west-1.amazonaws.com",
R53_ZONE('Z368ELLRRE2KJ0')) // ELB in us-west-1
```
Unfortunately, Route53 requires indicating the hosted zone id
where the target is defined (those are listed in AWS documentation,
see the R53_ALIAS documentation for links).
2018-01-16 18:53:12 +08:00
|
|
|
}
|
|
|
|
return rrset
|
|
|
|
}
|
|
|
|
|
|
|
|
func getZoneID(zone *r53.HostedZone, r *models.RecordConfig) string {
|
|
|
|
zoneID := r.R53Alias["zone_id"]
|
|
|
|
if zoneID == "" {
|
|
|
|
zoneID = aws.StringValue(zone.Id)
|
|
|
|
}
|
|
|
|
if strings.HasPrefix(zoneID, "/hostedzone/") {
|
|
|
|
zoneID = strings.TrimPrefix(zoneID, "/hostedzone/")
|
|
|
|
}
|
|
|
|
return zoneID
|
|
|
|
}
|
|
|
|
|
2020-10-26 21:25:30 +08:00
|
|
|
func (r *route53Provider) GetRegistrarCorrections(dc *models.DomainConfig) ([]*models.Correction, error) {
|
2017-08-05 22:56:42 +08:00
|
|
|
corrections := []*models.Correction{}
|
|
|
|
actualSet, err := r.getRegistrarNameservers(&dc.Name)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
sort.Strings(actualSet)
|
|
|
|
actual := strings.Join(actualSet, ",")
|
|
|
|
|
|
|
|
expectedSet := []string{}
|
|
|
|
for _, ns := range dc.Nameservers {
|
|
|
|
expectedSet = append(expectedSet, ns.Name)
|
|
|
|
}
|
|
|
|
sort.Strings(expectedSet)
|
|
|
|
expected := strings.Join(expectedSet, ",")
|
|
|
|
|
|
|
|
if actual != expected {
|
|
|
|
return []*models.Correction{
|
|
|
|
{
|
|
|
|
Msg: fmt.Sprintf("Update nameservers %s -> %s", actual, expected),
|
|
|
|
F: func() error {
|
2017-11-27 23:22:08 +08:00
|
|
|
_, err := r.updateRegistrarNameservers(dc.Name, expectedSet)
|
|
|
|
return err
|
2017-08-05 22:56:42 +08:00
|
|
|
},
|
|
|
|
},
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return corrections, nil
|
|
|
|
}
|
|
|
|
|
2020-10-26 21:25:30 +08:00
|
|
|
func (r *route53Provider) getRegistrarNameservers(domainName *string) ([]string, error) {
|
2019-02-20 01:30:39 +08:00
|
|
|
var domainDetail *r53d.GetDomainDetailOutput
|
|
|
|
var err error
|
|
|
|
withRetry(func() error {
|
|
|
|
domainDetail, err = r.registrar.GetDomainDetail(&r53d.GetDomainDetailInput{DomainName: domainName})
|
|
|
|
return err
|
|
|
|
})
|
2017-08-05 22:56:42 +08:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
nameservers := []string{}
|
|
|
|
for _, ns := range domainDetail.Nameservers {
|
|
|
|
nameservers = append(nameservers, *ns.Name)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nameservers, nil
|
|
|
|
}
|
|
|
|
|
2020-10-26 21:25:30 +08:00
|
|
|
func (r *route53Provider) updateRegistrarNameservers(domainName string, nameservers []string) (*string, error) {
|
2017-08-05 22:56:42 +08:00
|
|
|
servers := []*r53d.Nameserver{}
|
|
|
|
for i := range nameservers {
|
|
|
|
servers = append(servers, &r53d.Nameserver{Name: &nameservers[i]})
|
|
|
|
}
|
2019-02-20 01:30:39 +08:00
|
|
|
var domainUpdate *r53d.UpdateDomainNameserversOutput
|
|
|
|
var err error
|
|
|
|
withRetry(func() error {
|
|
|
|
domainUpdate, err = r.registrar.UpdateDomainNameservers(&r53d.UpdateDomainNameserversInput{
|
|
|
|
DomainName: &domainName, Nameservers: servers})
|
|
|
|
return err
|
|
|
|
})
|
2017-08-05 22:56:42 +08:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return domainUpdate.OperationId, nil
|
|
|
|
}
|
|
|
|
|
2020-10-26 21:25:30 +08:00
|
|
|
func (r *route53Provider) fetchRecordSets(zoneID *string) ([]*r53.ResourceRecordSet, error) {
|
2016-08-23 08:31:50 +08:00
|
|
|
if zoneID == nil || *zoneID == "" {
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
var next *string
|
|
|
|
var nextType *string
|
|
|
|
var records []*r53.ResourceRecordSet
|
|
|
|
for {
|
|
|
|
listInput := &r53.ListResourceRecordSetsInput{
|
|
|
|
HostedZoneId: zoneID,
|
|
|
|
StartRecordName: next,
|
|
|
|
StartRecordType: nextType,
|
|
|
|
MaxItems: sPtr("100"),
|
|
|
|
}
|
2019-02-20 01:30:39 +08:00
|
|
|
var list *r53.ListResourceRecordSetsOutput
|
|
|
|
var err error
|
|
|
|
withRetry(func() error {
|
|
|
|
list, err = r.client.ListResourceRecordSets(listInput)
|
|
|
|
return err
|
|
|
|
})
|
2016-08-23 08:31:50 +08:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2019-02-20 01:30:39 +08:00
|
|
|
|
2016-08-23 08:31:50 +08:00
|
|
|
records = append(records, list.ResourceRecordSets...)
|
|
|
|
if list.NextRecordName != nil {
|
|
|
|
next = list.NextRecordName
|
|
|
|
nextType = list.NextRecordType
|
|
|
|
} else {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return records, nil
|
|
|
|
}
|
|
|
|
|
2018-01-10 01:53:16 +08:00
|
|
|
// we have to process names from route53 to match what we expect and to remove their odd octal encoding
|
2016-08-23 08:31:50 +08:00
|
|
|
func unescape(s *string) string {
|
|
|
|
if s == nil {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
name := strings.TrimSuffix(*s, ".")
|
2018-01-10 01:53:16 +08:00
|
|
|
name = strings.Replace(name, `\052`, "*", -1) // TODO: escape all octal sequences
|
2016-08-23 08:31:50 +08:00
|
|
|
return name
|
|
|
|
}
|
2017-01-04 04:26:08 +08:00
|
|
|
|
2020-10-26 21:25:30 +08:00
|
|
|
func (r *route53Provider) EnsureDomainExists(domain string) error {
|
2017-01-04 04:26:08 +08:00
|
|
|
if _, ok := r.zones[domain]; ok {
|
|
|
|
return nil
|
|
|
|
}
|
2019-02-20 01:30:39 +08:00
|
|
|
if r.delegationSet != nil {
|
|
|
|
fmt.Printf("Adding zone for %s to route 53 account with delegationSet %s\n", domain, *r.delegationSet)
|
|
|
|
} else {
|
|
|
|
fmt.Printf("Adding zone for %s to route 53 account\n", domain)
|
|
|
|
}
|
2017-01-04 04:26:08 +08:00
|
|
|
in := &r53.CreateHostedZoneInput{
|
|
|
|
Name: &domain,
|
2019-02-20 01:30:39 +08:00
|
|
|
DelegationSetId: r.delegationSet,
|
2017-01-04 04:26:08 +08:00
|
|
|
CallerReference: sPtr(fmt.Sprint(time.Now().UnixNano())),
|
|
|
|
}
|
2019-02-20 01:30:39 +08:00
|
|
|
var err error
|
|
|
|
withRetry(func() error {
|
|
|
|
_, err := r.client.CreateHostedZone(in)
|
|
|
|
return err
|
|
|
|
})
|
2017-01-04 04:26:08 +08:00
|
|
|
return err
|
|
|
|
}
|