dnscontrol/providers/softlayer/softlayerProvider.go
Tom Limoncelli 489be2e3dc
ROUTE53: fix R53_ZONE() handling for domains (#2306)
Co-authored-by: Tom Limoncelli <tal@whatexit.org>
2023-05-02 13:04:59 -04:00

393 lines
10 KiB
Go

package softlayer
import (
"encoding/json"
"fmt"
"regexp"
"strings"
"github.com/StackExchange/dnscontrol/v3/models"
"github.com/StackExchange/dnscontrol/v3/pkg/diff"
"github.com/StackExchange/dnscontrol/v3/pkg/diff2"
"github.com/StackExchange/dnscontrol/v3/pkg/printer"
"github.com/StackExchange/dnscontrol/v3/providers"
"github.com/softlayer/softlayer-go/datatypes"
"github.com/softlayer/softlayer-go/filter"
"github.com/softlayer/softlayer-go/services"
"github.com/softlayer/softlayer-go/session"
)
// softlayerProvider is the protocol handle for this provider.
type softlayerProvider struct {
Session *session.Session
}
var features = providers.DocumentationNotes{
providers.CanGetZones: providers.Unimplemented(),
providers.CanUseLOC: providers.Cannot(),
providers.CanUseSRV: providers.Can(),
}
func init() {
fns := providers.DspFuncs{
Initializer: newReg,
RecordAuditor: AuditRecords,
}
providers.RegisterDomainServiceProviderType("SOFTLAYER", fns, features)
}
func newReg(conf map[string]string, _ json.RawMessage) (providers.DNSServiceProvider, error) {
printer.Warnf("The SOFTLAYER provider is unmaintained: https://github.com/StackExchange/dnscontrol/issues/1079")
s := session.New(conf["username"], conf["api_key"], conf["endpoint_url"], conf["timeout"])
if len(s.UserName) == 0 || len(s.APIKey) == 0 {
return nil, fmt.Errorf("SoftLayer UserName and APIKey must be provided")
}
// s.Debug = true
api := &softlayerProvider{
Session: s,
}
return api, nil
}
// GetNameservers returns the nameservers for a domain.
func (s *softlayerProvider) GetNameservers(domain string) ([]*models.Nameserver, error) {
// Always use the same nameservers for softlayer
return models.ToNameservers([]string{"ns1.softlayer.com", "ns2.softlayer.com"})
}
// GetZoneRecords gets all the records for domainName and converts
// them to model.RecordConfig.
func (s *softlayerProvider) GetZoneRecords(domainName string, meta map[string]string) (models.Records, error) {
domain, err := s.getDomain(&domainName)
if err != nil {
return nil, err
}
actual, err := s.getExistingRecords(domain)
if err != nil {
return nil, err
}
return actual, nil
}
// GetZoneRecordsCorrections returns a list of corrections that will turn existing records into dc.Records.
func (s *softlayerProvider) GetZoneRecordsCorrections(dc *models.DomainConfig, actual models.Records) ([]*models.Correction, error) {
domain, err := s.getDomain(&dc.Name)
if err != nil {
return nil, err
}
var corrections []*models.Correction
var create, deletes, modify diff.Changeset
if !diff2.EnableDiff2 {
_, create, deletes, modify, err = diff.New(dc).IncrementalDiff(actual)
} else {
_, create, deletes, modify, err = diff.NewCompat(dc).IncrementalDiff(actual)
}
if err != nil {
return nil, err
}
for _, del := range deletes {
existing := del.Existing.Original.(datatypes.Dns_Domain_ResourceRecord)
corrections = append(corrections, &models.Correction{
Msg: del.String(),
F: s.deleteRecordFunc(*existing.Id),
})
}
for _, cre := range create {
corrections = append(corrections, &models.Correction{
Msg: cre.String(),
F: s.createRecordFunc(cre.Desired, domain),
})
}
for _, mod := range modify {
existing := mod.Existing.Original.(datatypes.Dns_Domain_ResourceRecord)
corrections = append(corrections, &models.Correction{
Msg: mod.String(),
F: s.updateRecordFunc(&existing, mod.Desired),
})
}
return corrections, nil
}
func (s *softlayerProvider) getDomain(name *string) (*datatypes.Dns_Domain, error) {
// FIXME(tlim) Memoize this
domains, err := services.GetAccountService(s.Session).
Filter(filter.Path("domains.name").Eq(name).Build()).
Mask("resourceRecords").
GetDomains()
if err != nil {
return nil, err
}
if len(domains) == 0 {
return nil, fmt.Errorf("didn't find a domain matching %s", *name)
} else if len(domains) > 1 {
return nil, fmt.Errorf("found %d domains matching %s", len(domains), *name)
}
return &domains[0], nil
}
func (s *softlayerProvider) getExistingRecords(domain *datatypes.Dns_Domain) (models.Records, error) {
actual := []*models.RecordConfig{}
for _, record := range domain.ResourceRecords {
recType := strings.ToUpper(*record.Type)
if recType == "SOA" {
continue
}
recConfig := &models.RecordConfig{
Type: recType,
TTL: uint32(*record.Ttl),
Original: record,
}
recConfig.SetTarget(*record.Data)
switch recType {
case "SRV":
var service, protocol = "", "_tcp"
if record.Weight != nil {
recConfig.SrvWeight = uint16(*record.Weight)
}
if record.Port != nil {
recConfig.SrvPort = uint16(*record.Port)
}
if record.Priority != nil {
recConfig.SrvPriority = uint16(*record.Priority)
}
if record.Protocol != nil {
protocol = *record.Protocol
}
if record.Service != nil {
service = *record.Service
}
recConfig.SetLabel(fmt.Sprintf("%s.%s", service, strings.ToLower(protocol)), *domain.Name)
case "TXT":
recConfig.TxtStrings = append(recConfig.TxtStrings, *record.Data)
fallthrough
case "MX":
if record.MxPriority != nil {
recConfig.MxPreference = uint16(*record.MxPriority)
}
fallthrough
default:
recConfig.SetLabel(*record.Host, *domain.Name)
}
actual = append(actual, recConfig)
}
return actual, nil
}
func (s *softlayerProvider) createRecordFunc(desired *models.RecordConfig, domain *datatypes.Dns_Domain) func() error {
var ttl, preference, domainID = verifyMinTTL(int(desired.TTL)), int(desired.MxPreference), *domain.Id
var weight, priority, port = int(desired.SrvWeight), int(desired.SrvPriority), int(desired.SrvPort)
var host, data, newType = desired.GetLabel(), desired.GetTargetField(), desired.Type
var err error
srvRegexp := regexp.MustCompile(`^_(?P<Service>\w+)\.\_(?P<Protocol>\w+)$`)
return func() error {
newRecord := datatypes.Dns_Domain_ResourceRecord{
DomainId: &domainID,
Ttl: &ttl,
Type: &newType,
Data: &data,
Host: &host,
}
switch newType {
case "MX":
service := services.GetDnsDomainResourceRecordMxTypeService(s.Session)
newRecord.MxPriority = &preference
newMx := datatypes.Dns_Domain_ResourceRecord_MxType{
Dns_Domain_ResourceRecord: newRecord,
}
_, err = service.CreateObject(&newMx)
case "SRV":
service := services.GetDnsDomainResourceRecordSrvTypeService(s.Session)
result := srvRegexp.FindStringSubmatch(host)
if len(result) != 3 {
return fmt.Errorf("SRV Record must match format \"_service._protocol\" not %s", host)
}
var serviceName, protocol = result[1], strings.ToLower(result[2])
newSrv := datatypes.Dns_Domain_ResourceRecord_SrvType{
Dns_Domain_ResourceRecord: newRecord,
Service: &serviceName,
Port: &port,
Priority: &priority,
Protocol: &protocol,
Weight: &weight,
}
_, err = service.CreateObject(&newSrv)
default:
service := services.GetDnsDomainResourceRecordService(s.Session)
_, err = service.CreateObject(&newRecord)
}
return err
}
}
func (s *softlayerProvider) deleteRecordFunc(resID int) func() error {
// seems to be no problem deleting MX and SRV records via common interface
return func() error {
_, err := services.GetDnsDomainResourceRecordService(s.Session).
Id(resID).
DeleteObject()
return err
}
}
func (s *softlayerProvider) updateRecordFunc(existing *datatypes.Dns_Domain_ResourceRecord, desired *models.RecordConfig) func() error {
var ttl, preference = verifyMinTTL(int(desired.TTL)), int(desired.MxPreference)
var priority, weight, port = int(desired.SrvPriority), int(desired.SrvWeight), int(desired.SrvPort)
return func() error {
var changes = false
var err error
switch desired.Type {
case "MX":
service := services.GetDnsDomainResourceRecordMxTypeService(s.Session)
updated := datatypes.Dns_Domain_ResourceRecord_MxType{}
label := desired.GetLabel()
if label != *existing.Host {
updated.Host = &label
changes = true
}
target := desired.GetTargetField()
if target != *existing.Data {
updated.Data = &target
changes = true
}
if ttl != *existing.Ttl {
updated.Ttl = &ttl
changes = true
}
if preference != *existing.MxPriority {
updated.MxPriority = &preference
changes = true
}
if !changes {
return fmt.Errorf("didn't find changes when I expect some")
}
_, err = service.Id(*existing.Id).EditObject(&updated)
case "SRV":
service := services.GetDnsDomainResourceRecordSrvTypeService(s.Session)
updated := datatypes.Dns_Domain_ResourceRecord_SrvType{}
label := desired.GetLabel()
if label != *existing.Host {
updated.Host = &label
changes = true
}
target := desired.GetTargetField()
if target != *existing.Data {
updated.Data = &target
changes = true
}
if ttl != *existing.Ttl {
updated.Ttl = &ttl
changes = true
}
if priority != *existing.Priority {
updated.Priority = &priority
changes = true
}
if weight != *existing.Weight {
updated.Weight = &weight
changes = true
}
if port != *existing.Port {
updated.Port = &port
changes = true
}
// TODO: handle service & protocol - or does that just result in a
// delete and recreate?
if !changes {
return fmt.Errorf("didn't find changes when I expect some")
}
_, err = service.Id(*existing.Id).EditObject(&updated)
default:
service := services.GetDnsDomainResourceRecordService(s.Session)
updated := datatypes.Dns_Domain_ResourceRecord{}
label := desired.GetLabel()
if label != *existing.Host {
updated.Host = &label
changes = true
}
target := desired.GetTargetField()
if target != *existing.Data {
updated.Data = &target
changes = true
}
if ttl != *existing.Ttl {
updated.Ttl = &ttl
changes = true
}
if !changes {
return fmt.Errorf("didn't find changes when I expect some")
}
_, err = service.Id(*existing.Id).EditObject(&updated)
}
return err
}
}
func verifyMinTTL(ttl int) int {
const minTTL = 60
if ttl < minTTL {
printer.Printf("\nMODIFY TTL to Min supported TTL value: (ttl=%d) -> (ttl=%d)\n", ttl, minTTL)
return minTTL
}
return ttl
}