mirror of
https://github.com/StackExchange/dnscontrol.git
synced 2024-11-10 09:12:47 +08:00
134 lines
4.1 KiB
Go
134 lines
4.1 KiB
Go
package netcup
|
|
|
|
import (
|
|
"encoding/json"
|
|
"fmt"
|
|
|
|
"github.com/StackExchange/dnscontrol/v4/models"
|
|
"github.com/StackExchange/dnscontrol/v4/pkg/diff"
|
|
"github.com/StackExchange/dnscontrol/v4/providers"
|
|
)
|
|
|
|
var features = providers.DocumentationNotes{
|
|
// The default for unlisted capabilities is 'Cannot'.
|
|
// See providers/capabilities.go for the entire list of capabilities.
|
|
providers.CanGetZones: providers.Cannot(),
|
|
providers.CanConcur: providers.Cannot(),
|
|
providers.CanUseCAA: providers.Can(),
|
|
providers.CanUseLOC: providers.Cannot(),
|
|
providers.CanUsePTR: providers.Cannot(),
|
|
providers.CanUseSRV: providers.Can(),
|
|
providers.DocCreateDomains: providers.Cannot(),
|
|
providers.DocDualHost: providers.Cannot(),
|
|
providers.DocOfficiallySupported: providers.Cannot(),
|
|
}
|
|
|
|
func init() {
|
|
fns := providers.DspFuncs{
|
|
Initializer: New,
|
|
RecordAuditor: AuditRecords,
|
|
}
|
|
providers.RegisterDomainServiceProviderType("NETCUP", fns, features)
|
|
}
|
|
|
|
// New creates a new API handle.
|
|
func New(settings map[string]string, _ json.RawMessage) (providers.DNSServiceProvider, error) {
|
|
if settings["api-key"] == "" || settings["api-password"] == "" || settings["customer-number"] == "" {
|
|
return nil, fmt.Errorf("missing netcup login parameters")
|
|
}
|
|
|
|
api := &netcupProvider{}
|
|
err := api.login(settings["api-key"], settings["api-password"], settings["customer-number"])
|
|
if err != nil {
|
|
return nil, fmt.Errorf("login to netcup DNS failed, please check your credentials: %v", err)
|
|
}
|
|
return api, nil
|
|
}
|
|
|
|
// GetZoneRecords gets the records of a zone and returns them in RecordConfig format.
|
|
func (api *netcupProvider) GetZoneRecords(domain string, meta map[string]string) (models.Records, error) {
|
|
records, err := api.getRecords(domain)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
existingRecords := make([]*models.RecordConfig, len(records))
|
|
for i := range records {
|
|
existingRecords[i] = toRecordConfig(domain, &records[i])
|
|
}
|
|
|
|
return existingRecords, nil
|
|
}
|
|
|
|
// GetNameservers returns the nameservers for a domain.
|
|
// As netcup doesn't support setting nameservers over this API, these are static.
|
|
// Domains not managed by netcup DNS will return an error
|
|
func (api *netcupProvider) GetNameservers(domain string) ([]*models.Nameserver, error) {
|
|
return models.ToNameservers([]string{
|
|
"root-dns.netcup.net",
|
|
"second-dns.netcup.net",
|
|
"third-dns.netcup.net",
|
|
})
|
|
}
|
|
|
|
// GetZoneRecordsCorrections returns a list of corrections that will turn existing records into dc.Records.
|
|
func (api *netcupProvider) GetZoneRecordsCorrections(dc *models.DomainConfig, existingRecords models.Records) ([]*models.Correction, error) {
|
|
domain := dc.Name
|
|
|
|
// Setting the TTL is not supported for netcup
|
|
for _, r := range dc.Records {
|
|
r.TTL = 0
|
|
}
|
|
|
|
// Filter out types we can't modify (like NS)
|
|
newRecords := models.Records{}
|
|
for _, r := range dc.Records {
|
|
if r.Type != "NS" {
|
|
newRecords = append(newRecords, r)
|
|
}
|
|
}
|
|
dc.Records = newRecords
|
|
|
|
toReport, create, del, modify, err := diff.NewCompat(dc).IncrementalDiff(existingRecords)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
// Start corrections with the reports
|
|
corrections := diff.GenerateMessageCorrections(toReport)
|
|
|
|
// Deletes first so changing type works etc.
|
|
for _, m := range del {
|
|
req := m.Existing.Original.(*record)
|
|
corr := &models.Correction{
|
|
Msg: fmt.Sprintf("%s, Netcup ID: %s", m.String(), req.ID),
|
|
F: func() error {
|
|
return api.deleteRecord(domain, req)
|
|
},
|
|
}
|
|
corrections = append(corrections, corr)
|
|
}
|
|
|
|
for _, m := range create {
|
|
req := fromRecordConfig(m.Desired)
|
|
corr := &models.Correction{
|
|
Msg: m.String(),
|
|
F: func() error {
|
|
return api.createRecord(domain, req)
|
|
},
|
|
}
|
|
corrections = append(corrections, corr)
|
|
}
|
|
for _, m := range modify {
|
|
id := m.Existing.Original.(*record).ID
|
|
req := fromRecordConfig(m.Desired)
|
|
req.ID = id
|
|
corr := &models.Correction{
|
|
Msg: fmt.Sprintf("%s, Netcup ID: %s: ", m.String(), id),
|
|
F: func() error {
|
|
return api.modifyRecord(domain, req)
|
|
},
|
|
}
|
|
corrections = append(corrections, corr)
|
|
}
|
|
|
|
return corrections, nil
|
|
}
|