2017-10-12 21:21:36 +08:00
|
|
|
package vultr
|
|
|
|
|
|
|
|
import (
|
2019-07-15 22:31:55 +08:00
|
|
|
"context"
|
2017-10-12 21:21:36 +08:00
|
|
|
"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"
|
2017-10-12 21:21:36 +08:00
|
|
|
"fmt"
|
2019-07-15 22:31:55 +08:00
|
|
|
"strconv"
|
2017-10-12 21:21:36 +08:00
|
|
|
"strings"
|
|
|
|
|
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
|
|
|
"github.com/vultr/govultr"
|
|
|
|
|
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"
|
2017-10-12 21:21:36 +08:00
|
|
|
)
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
|
|
|
Vultr API DNS provider:
|
|
|
|
|
|
|
|
Info required in `creds.json`:
|
|
|
|
- token
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
2018-01-05 08:19:35 +08:00
|
|
|
var features = providers.DocumentationNotes{
|
2017-10-12 21:21:36 +08:00
|
|
|
providers.CanUseAlias: providers.Cannot(),
|
2018-01-05 08:19:35 +08:00
|
|
|
providers.CanUseCAA: providers.Can(),
|
2017-10-12 21:21:36 +08:00
|
|
|
providers.CanUsePTR: providers.Cannot(),
|
2018-01-05 08:19:35 +08:00
|
|
|
providers.CanUseSRV: providers.Can(),
|
|
|
|
providers.CanUseTLSA: providers.Cannot(),
|
2019-07-16 00:28:37 +08:00
|
|
|
providers.CanUseSSHFP: providers.Can(),
|
2018-01-05 08:19:35 +08:00
|
|
|
providers.DocCreateDomains: providers.Can(),
|
|
|
|
providers.DocOfficiallySupported: providers.Cannot(),
|
2020-02-29 22:04:00 +08:00
|
|
|
providers.CanGetZones: providers.Can(),
|
2017-10-12 21:21:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
func init() {
|
2021-03-08 02:19:22 +08:00
|
|
|
fns := providers.DspFuncs{
|
2021-05-05 02:15:31 +08:00
|
|
|
Initializer: NewProvider,
|
2021-03-09 09:14:30 +08:00
|
|
|
RecordAuditor: AuditRecords,
|
2021-03-08 02:19:22 +08:00
|
|
|
}
|
|
|
|
providers.RegisterDomainServiceProviderType("VULTR", fns, features)
|
2017-10-12 21:21:36 +08:00
|
|
|
}
|
|
|
|
|
2020-10-26 21:25:30 +08:00
|
|
|
// vultrProvider represents the Vultr DNSServiceProvider.
|
|
|
|
type vultrProvider struct {
|
2019-07-15 22:31:55 +08:00
|
|
|
client *govultr.Client
|
2017-10-12 21:21:36 +08:00
|
|
|
token string
|
|
|
|
}
|
|
|
|
|
2019-07-15 22:31:55 +08:00
|
|
|
// defaultNS contains the default nameservers for Vultr.
|
2017-10-12 21:21:36 +08:00
|
|
|
var defaultNS = []string{
|
|
|
|
"ns1.vultr.com",
|
|
|
|
"ns2.vultr.com",
|
|
|
|
}
|
|
|
|
|
2019-07-15 22:31:55 +08:00
|
|
|
// NewProvider initializes a Vultr DNSServiceProvider.
|
|
|
|
func NewProvider(m map[string]string, metadata json.RawMessage) (providers.DNSServiceProvider, error) {
|
|
|
|
token := m["token"]
|
|
|
|
if token == "" {
|
2020-08-31 07:52:37 +08:00
|
|
|
return nil, fmt.Errorf("missing Vultr API token")
|
2017-10-12 21:21:36 +08:00
|
|
|
}
|
|
|
|
|
2019-07-15 22:31:55 +08:00
|
|
|
client := govultr.NewClient(nil, token)
|
|
|
|
client.SetUserAgent("dnscontrol")
|
2017-10-12 21:21:36 +08:00
|
|
|
|
2019-07-15 22:31:55 +08:00
|
|
|
_, err := client.Account.GetInfo(context.Background())
|
2020-10-26 21:25:30 +08:00
|
|
|
return &vultrProvider{client, token}, err
|
2017-10-12 21:21:36 +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 (api *vultrProvider) GetZoneRecords(domain string) (models.Records, error) {
|
2020-02-29 22:04:00 +08:00
|
|
|
records, err := api.client.DNSRecord.List(context.Background(), domain)
|
2017-10-12 21:21:36 +08:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2020-02-29 22:04:00 +08:00
|
|
|
curRecords := make(models.Records, len(records))
|
2017-10-12 21:21:36 +08:00
|
|
|
for i := range records {
|
2020-02-29 22:04:00 +08:00
|
|
|
r, err := toRecordConfig(domain, &records[i])
|
2017-10-12 21:21:36 +08:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
curRecords[i] = r
|
|
|
|
}
|
|
|
|
|
2020-02-29 22:04:00 +08:00
|
|
|
return curRecords, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetDomainCorrections gets the corrections for a DomainConfig.
|
2020-10-26 21:25:30 +08:00
|
|
|
func (api *vultrProvider) GetDomainCorrections(dc *models.DomainConfig) ([]*models.Correction, error) {
|
2020-02-29 22:04:00 +08:00
|
|
|
dc.Punycode()
|
|
|
|
|
|
|
|
curRecords, err := api.GetZoneRecords(dc.Name)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2018-01-05 08:19:35 +08:00
|
|
|
models.PostProcessRecords(curRecords)
|
2017-11-08 06:12:17 +08:00
|
|
|
|
2017-10-12 21:21:36 +08:00
|
|
|
differ := diff.New(dc)
|
2020-08-21 03:49:00 +08:00
|
|
|
_, create, delete, modify, err := differ.IncrementalDiff(curRecords)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2017-10-12 21:21:36 +08:00
|
|
|
|
2019-07-15 22:31:55 +08:00
|
|
|
var corrections []*models.Correction
|
2017-10-12 21:21:36 +08:00
|
|
|
|
|
|
|
for _, mod := range delete {
|
2019-07-15 22:31:55 +08:00
|
|
|
id := mod.Existing.Original.(*govultr.DNSRecord).RecordID
|
2017-10-12 21:21:36 +08:00
|
|
|
corrections = append(corrections, &models.Correction{
|
|
|
|
Msg: fmt.Sprintf("%s; Vultr RecordID: %v", mod.String(), id),
|
|
|
|
F: func() error {
|
2019-07-15 22:31:55 +08:00
|
|
|
return api.client.DNSRecord.Delete(context.Background(), dc.Name, strconv.Itoa(id))
|
2017-10-12 21:21:36 +08:00
|
|
|
},
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, mod := range create {
|
2019-07-15 22:31:55 +08:00
|
|
|
r := toVultrRecord(dc, mod.Desired, 0)
|
2017-10-12 21:21:36 +08:00
|
|
|
corrections = append(corrections, &models.Correction{
|
|
|
|
Msg: mod.String(),
|
|
|
|
F: func() error {
|
2020-10-09 04:12:56 +08:00
|
|
|
return api.client.DNSRecord.Create(context.Background(), dc.Name, r.Type, r.Name, r.Data, r.TTL, vultrPriority(r))
|
2017-10-12 21:21:36 +08:00
|
|
|
},
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, mod := range modify {
|
2019-07-15 22:31:55 +08:00
|
|
|
r := toVultrRecord(dc, mod.Desired, mod.Existing.Original.(*govultr.DNSRecord).RecordID)
|
2017-10-12 21:21:36 +08:00
|
|
|
corrections = append(corrections, &models.Correction{
|
2019-07-15 22:31:55 +08:00
|
|
|
Msg: fmt.Sprintf("%s; Vultr RecordID: %v", mod.String(), r.RecordID),
|
2017-10-12 21:21:36 +08:00
|
|
|
F: func() error {
|
2019-07-15 22:31:55 +08:00
|
|
|
return api.client.DNSRecord.Update(context.Background(), dc.Name, r)
|
2017-10-12 21:21:36 +08:00
|
|
|
},
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
return corrections, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetNameservers gets the Vultr nameservers for a domain
|
2020-10-26 21:25:30 +08:00
|
|
|
func (api *vultrProvider) GetNameservers(domain string) ([]*models.Nameserver, error) {
|
2020-03-26 21:59:59 +08:00
|
|
|
return models.ToNameservers(defaultNS)
|
2017-10-12 21:21:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// EnsureDomainExists adds a domain to the Vutr DNS service if it does not exist
|
2020-10-26 21:25:30 +08:00
|
|
|
func (api *vultrProvider) EnsureDomainExists(domain string) error {
|
2019-07-15 22:31:55 +08:00
|
|
|
if ok, err := api.isDomainInAccount(domain); err != nil {
|
2017-10-12 21:21:36 +08:00
|
|
|
return err
|
2019-07-15 22:31:55 +08:00
|
|
|
} else if ok {
|
|
|
|
return nil
|
2017-10-12 21:21:36 +08:00
|
|
|
}
|
|
|
|
|
2019-07-15 22:31:55 +08:00
|
|
|
// Vultr requires an initial IP, use a dummy one.
|
|
|
|
return api.client.DNSDomain.Create(context.Background(), domain, "0.0.0.0")
|
2017-10-12 21:21:36 +08:00
|
|
|
}
|
|
|
|
|
2020-10-26 21:25:30 +08:00
|
|
|
func (api *vultrProvider) isDomainInAccount(domain string) (bool, error) {
|
2019-07-15 22:31:55 +08:00
|
|
|
domains, err := api.client.DNSDomain.List(context.Background())
|
2017-10-12 21:21:36 +08:00
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
for _, d := range domains {
|
|
|
|
if d.Domain == domain {
|
2019-07-15 22:31:55 +08:00
|
|
|
return true, nil
|
2017-10-12 21:21:36 +08:00
|
|
|
}
|
|
|
|
}
|
2019-07-15 22:31:55 +08:00
|
|
|
return false, nil
|
2017-10-12 21:21:36 +08:00
|
|
|
}
|
|
|
|
|
2019-07-15 22:31:55 +08:00
|
|
|
// toRecordConfig converts a Vultr DNSRecord to a RecordConfig. #rtype_variations
|
2020-02-29 22:04:00 +08:00
|
|
|
func toRecordConfig(domain string, r *govultr.DNSRecord) (*models.RecordConfig, error) {
|
|
|
|
origin, data := domain, r.Data
|
2017-10-12 21:21:36 +08:00
|
|
|
rc := &models.RecordConfig{
|
|
|
|
TTL: uint32(r.TTL),
|
|
|
|
Original: r,
|
|
|
|
}
|
2020-02-29 22:04:00 +08:00
|
|
|
rc.SetLabel(r.Name, domain)
|
2017-10-12 21:21:36 +08:00
|
|
|
|
2018-02-16 01:02:50 +08:00
|
|
|
switch rtype := r.Type; rtype {
|
|
|
|
case "CNAME", "NS":
|
|
|
|
rc.Type = r.Type
|
2019-07-15 22:31:55 +08:00
|
|
|
// Make target into a FQDN if it is a CNAME, NS, MX, or SRV.
|
2018-02-16 01:02:50 +08:00
|
|
|
if !strings.HasSuffix(data, ".") {
|
|
|
|
data = data + "."
|
2017-10-12 21:21:36 +08:00
|
|
|
}
|
2020-10-09 04:12:38 +08:00
|
|
|
return rc, rc.SetTarget(data)
|
2018-02-16 01:02:50 +08:00
|
|
|
case "CAA":
|
2019-07-15 22:31:55 +08:00
|
|
|
// Vultr returns CAA records in the format "[flag] [tag] [value]".
|
2018-02-16 01:02:50 +08:00
|
|
|
return rc, rc.SetTargetCAAString(data)
|
|
|
|
case "MX":
|
|
|
|
if !strings.HasSuffix(data, ".") {
|
|
|
|
data = data + "."
|
2017-10-12 21:21:36 +08:00
|
|
|
}
|
2020-10-09 04:12:56 +08:00
|
|
|
return rc, rc.SetTargetMX(uint16(vultrPriority(r)), data)
|
2018-02-16 01:02:50 +08:00
|
|
|
case "SRV":
|
2019-07-15 22:31:55 +08:00
|
|
|
// Vultr returns SRV records in the format "[weight] [port] [target]".
|
2020-10-09 04:12:56 +08:00
|
|
|
return rc, rc.SetTargetSRVPriorityString(uint16(vultrPriority(r)), data)
|
2018-02-16 01:02:50 +08:00
|
|
|
case "TXT":
|
2019-07-15 22:31:55 +08:00
|
|
|
// Remove quotes if it is a TXT record.
|
2018-02-16 01:02:50 +08:00
|
|
|
if !strings.HasPrefix(data, `"`) || !strings.HasSuffix(data, `"`) {
|
2020-08-31 07:52:37 +08:00
|
|
|
return nil, errors.New("unexpected lack of quotes in TXT record from Vultr")
|
2017-10-12 21:21:36 +08:00
|
|
|
}
|
2018-02-16 01:02:50 +08:00
|
|
|
return rc, rc.SetTargetTXT(data[1 : len(data)-1])
|
|
|
|
default:
|
|
|
|
return rc, rc.PopulateFromString(rtype, r.Data, origin)
|
2017-10-12 21:21:36 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-07-15 22:31:55 +08:00
|
|
|
// toVultrRecord converts a RecordConfig converted by toRecordConfig back to a Vultr DNSRecord. #rtype_variations
|
|
|
|
func toVultrRecord(dc *models.DomainConfig, rc *models.RecordConfig, vultrID int) *govultr.DNSRecord {
|
2018-03-20 05:18:58 +08:00
|
|
|
name := rc.GetLabel()
|
2019-07-15 22:31:55 +08:00
|
|
|
// Vultr uses a blank string to represent the apex domain.
|
2017-10-12 21:21:36 +08:00
|
|
|
if name == "@" {
|
|
|
|
name = ""
|
|
|
|
}
|
|
|
|
|
2018-02-16 01:02:50 +08:00
|
|
|
data := rc.GetTargetField()
|
2017-10-12 21:21:36 +08:00
|
|
|
|
2019-07-15 22:31:55 +08:00
|
|
|
// Vultr does not use a period suffix for CNAME, NS, or MX.
|
2017-10-12 21:21:36 +08:00
|
|
|
if strings.HasSuffix(data, ".") {
|
|
|
|
data = data[:len(data)-1]
|
|
|
|
}
|
|
|
|
|
2020-10-09 04:12:56 +08:00
|
|
|
var priority *int
|
2017-10-12 21:21:36 +08:00
|
|
|
|
|
|
|
if rc.Type == "MX" {
|
2020-10-09 04:12:56 +08:00
|
|
|
tmp := int(rc.MxPreference)
|
|
|
|
priority = &tmp
|
2017-10-12 21:21:36 +08:00
|
|
|
}
|
|
|
|
if rc.Type == "SRV" {
|
2020-10-09 04:12:56 +08:00
|
|
|
tmp := int(rc.SrvPriority)
|
|
|
|
priority = &tmp
|
2017-10-12 21:21:36 +08:00
|
|
|
}
|
|
|
|
|
2019-07-15 22:31:55 +08:00
|
|
|
r := &govultr.DNSRecord{
|
|
|
|
RecordID: vultrID,
|
2017-10-12 21:21:36 +08:00
|
|
|
Type: rc.Type,
|
|
|
|
Name: name,
|
|
|
|
Data: data,
|
|
|
|
TTL: int(rc.TTL),
|
|
|
|
Priority: priority,
|
|
|
|
}
|
2018-03-20 05:18:58 +08:00
|
|
|
switch rtype := rc.Type; rtype { // #rtype_variations
|
|
|
|
case "SRV":
|
2019-07-15 22:31:55 +08:00
|
|
|
r.Data = fmt.Sprintf("%v %v %s", rc.SrvWeight, rc.SrvPort, rc.GetTargetField())
|
2018-03-20 05:18:58 +08:00
|
|
|
case "CAA":
|
2018-02-16 01:02:50 +08:00
|
|
|
r.Data = fmt.Sprintf(`%v %s "%s"`, rc.CaaFlag, rc.CaaTag, rc.GetTargetField())
|
2019-07-16 00:28:37 +08:00
|
|
|
case "SSHFP":
|
|
|
|
r.Data = fmt.Sprintf("%d %d %s", rc.SshfpAlgorithm, rc.SshfpFingerprint, rc.GetTargetField())
|
2018-03-20 05:18:58 +08:00
|
|
|
default:
|
2017-10-12 21:21:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return r
|
|
|
|
}
|
2020-10-09 04:12:56 +08:00
|
|
|
|
|
|
|
func vultrPriority(r *govultr.DNSRecord) int {
|
|
|
|
if r.Priority == nil {
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
return *r.Priority
|
|
|
|
}
|