mirror of
https://github.com/StackExchange/dnscontrol.git
synced 2025-01-13 02:50:27 +08:00
154 lines
4.3 KiB
Go
154 lines
4.3 KiB
Go
package netcup
|
|
|
|
import (
|
|
"encoding/json"
|
|
"fmt"
|
|
|
|
"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/providers"
|
|
)
|
|
|
|
var features = providers.DocumentationNotes{
|
|
providers.CanGetZones: 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) (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",
|
|
})
|
|
}
|
|
|
|
// GetDomainCorrections returns the corrections for a domain.
|
|
func (api *netcupProvider) GetDomainCorrections(dc *models.DomainConfig) ([]*models.Correction, error) {
|
|
dc, err := dc.Copy()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
dc.Punycode()
|
|
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
|
|
|
|
// Check existing set
|
|
existingRecords, err := api.GetZoneRecords(domain)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// Normalize
|
|
models.PostProcessRecords(existingRecords)
|
|
// no need for txtutil.SplitSingleLongTxt in function GetDomainCorrections
|
|
// txtutil.SplitSingleLongTxt(dc.Records) // Autosplit long TXT records
|
|
|
|
var corrections []*models.Correction
|
|
var create, del, modify diff.Changeset
|
|
var differ diff.Differ
|
|
if !diff2.EnableDiff2 {
|
|
differ = diff.New(dc)
|
|
} else {
|
|
differ = diff.NewCompat(dc)
|
|
}
|
|
_, create, del, modify, err = differ.IncrementalDiff(existingRecords)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// 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
|
|
}
|