2020-10-22 21:44:21 +08:00
|
|
|
package hetzner
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
2020-11-02 08:30:16 +08:00
|
|
|
"strings"
|
2020-10-22 21:44:21 +08:00
|
|
|
|
|
|
|
"github.com/StackExchange/dnscontrol/v3/models"
|
|
|
|
"github.com/StackExchange/dnscontrol/v3/pkg/diff"
|
2021-03-08 02:19:22 +08:00
|
|
|
"github.com/StackExchange/dnscontrol/v3/pkg/txtutil"
|
2020-10-22 21:44:21 +08:00
|
|
|
"github.com/StackExchange/dnscontrol/v3/providers"
|
|
|
|
)
|
|
|
|
|
|
|
|
var features = providers.DocumentationNotes{
|
|
|
|
providers.CanGetZones: providers.Can(),
|
|
|
|
providers.CanUseAlias: providers.Cannot(),
|
|
|
|
providers.CanUseCAA: providers.Can(),
|
|
|
|
providers.CanUseDS: providers.Cannot(),
|
|
|
|
providers.CanUsePTR: providers.Cannot(),
|
|
|
|
providers.CanUseSRV: providers.Can(),
|
|
|
|
providers.CanUseSSHFP: providers.Cannot(),
|
|
|
|
providers.CanUseTLSA: providers.Cannot(),
|
2022-03-03 00:19:15 +08:00
|
|
|
providers.DocCreateDomains: providers.Can(),
|
|
|
|
providers.DocDualHost: providers.Can(),
|
|
|
|
providers.DocOfficiallySupported: providers.Cannot(),
|
2020-10-22 21:44:21 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
func init() {
|
2021-03-08 02:19:22 +08:00
|
|
|
fns := providers.DspFuncs{
|
2021-05-05 02:15:31 +08:00
|
|
|
Initializer: New,
|
2021-03-09 09:14:30 +08:00
|
|
|
RecordAuditor: AuditRecords,
|
2021-03-08 02:19:22 +08:00
|
|
|
}
|
|
|
|
providers.RegisterDomainServiceProviderType("HETZNER", fns, features)
|
2020-10-22 21:44:21 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// New creates a new API handle.
|
|
|
|
func New(settings map[string]string, _ json.RawMessage) (providers.DNSServiceProvider, error) {
|
|
|
|
if settings["api_key"] == "" {
|
|
|
|
return nil, fmt.Errorf("missing HETZNER api_key")
|
|
|
|
}
|
|
|
|
|
2020-10-26 21:25:30 +08:00
|
|
|
api := &hetznerProvider{}
|
2020-10-22 21:44:21 +08:00
|
|
|
|
|
|
|
api.apiKey = settings["api_key"]
|
|
|
|
|
|
|
|
if settings["rate_limited"] == "true" {
|
2020-11-17 01:26:52 +08:00
|
|
|
// backwards compatibility
|
|
|
|
settings["start_with_default_rate_limit"] = "true"
|
|
|
|
}
|
|
|
|
if settings["start_with_default_rate_limit"] == "true" {
|
2020-10-22 21:44:21 +08:00
|
|
|
api.startRateLimited()
|
|
|
|
}
|
|
|
|
|
2020-11-17 01:26:52 +08:00
|
|
|
quota := settings["optimize_for_rate_limit_quota"]
|
|
|
|
err := api.requestRateLimiter.setOptimizeForRateLimitQuota(quota)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("unexpected value for optimize_for_rate_limit_quota: %w", err)
|
|
|
|
}
|
|
|
|
|
2020-10-22 21:44:21 +08:00
|
|
|
return api, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// EnsureDomainExists creates the domain if it does not exist.
|
2020-10-26 21:25:30 +08:00
|
|
|
func (api *hetznerProvider) EnsureDomainExists(domain string) error {
|
2020-10-22 21:44:21 +08:00
|
|
|
domains, err := api.ListZones()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, d := range domains {
|
|
|
|
if d == domain {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-08-02 02:44:17 +08:00
|
|
|
// reset zone cache
|
|
|
|
api.zones = nil
|
2020-10-22 21:44:21 +08:00
|
|
|
return api.createZone(domain)
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetDomainCorrections returns the corrections for a domain.
|
2020-10-26 21:25:30 +08:00
|
|
|
func (api *hetznerProvider) GetDomainCorrections(dc *models.DomainConfig) ([]*models.Correction, error) {
|
2020-10-22 21:44:21 +08:00
|
|
|
dc, err := dc.Copy()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
err = dc.Punycode()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
domain := dc.Name
|
|
|
|
|
|
|
|
// Get existing records
|
|
|
|
existingRecords, err := api.GetZoneRecords(domain)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Normalize
|
|
|
|
models.PostProcessRecords(existingRecords)
|
2021-03-08 02:19:22 +08:00
|
|
|
txtutil.SplitSingleLongTxt(dc.Records) // Autosplit long TXT records
|
2020-10-22 21:44:21 +08:00
|
|
|
|
|
|
|
differ := diff.New(dc)
|
|
|
|
_, create, del, modify, err := differ.IncrementalDiff(existingRecords)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
var corrections []*models.Correction
|
|
|
|
|
|
|
|
zone, err := api.getZone(domain)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, m := range del {
|
|
|
|
record := m.Existing.Original.(*record)
|
|
|
|
corr := &models.Correction{
|
|
|
|
Msg: m.String(),
|
|
|
|
F: func() error {
|
|
|
|
return api.deleteRecord(*record)
|
|
|
|
},
|
|
|
|
}
|
|
|
|
corrections = append(corrections, corr)
|
|
|
|
}
|
|
|
|
|
2020-11-02 08:30:16 +08:00
|
|
|
var createRecords []record
|
|
|
|
createDescription := []string{"Batch creation of records:"}
|
2020-10-22 21:44:21 +08:00
|
|
|
for _, m := range create {
|
|
|
|
record := fromRecordConfig(m.Desired, zone)
|
2020-11-02 08:30:16 +08:00
|
|
|
createRecords = append(createRecords, *record)
|
|
|
|
createDescription = append(createDescription, m.String())
|
|
|
|
}
|
|
|
|
if len(createRecords) > 0 {
|
2020-10-22 21:44:21 +08:00
|
|
|
corr := &models.Correction{
|
2020-11-02 08:30:16 +08:00
|
|
|
Msg: strings.Join(createDescription, "\n\t"),
|
2020-10-22 21:44:21 +08:00
|
|
|
F: func() error {
|
2020-11-02 08:30:16 +08:00
|
|
|
return api.bulkCreateRecords(createRecords)
|
2020-10-22 21:44:21 +08:00
|
|
|
},
|
|
|
|
}
|
|
|
|
corrections = append(corrections, corr)
|
|
|
|
}
|
|
|
|
|
2020-11-02 08:30:16 +08:00
|
|
|
var modifyRecords []record
|
|
|
|
modifyDescription := []string{"Batch modification of records:"}
|
2020-10-22 21:44:21 +08:00
|
|
|
for _, m := range modify {
|
|
|
|
id := m.Existing.Original.(*record).ID
|
|
|
|
record := fromRecordConfig(m.Desired, zone)
|
|
|
|
record.ID = id
|
2020-11-02 08:30:16 +08:00
|
|
|
modifyRecords = append(modifyRecords, *record)
|
|
|
|
modifyDescription = append(modifyDescription, m.String())
|
|
|
|
}
|
|
|
|
if len(modifyRecords) > 0 {
|
2020-10-22 21:44:21 +08:00
|
|
|
corr := &models.Correction{
|
2020-11-02 08:30:16 +08:00
|
|
|
Msg: strings.Join(modifyDescription, "\n\t"),
|
2020-10-22 21:44:21 +08:00
|
|
|
F: func() error {
|
2020-11-02 08:30:16 +08:00
|
|
|
return api.bulkUpdateRecords(modifyRecords)
|
2020-10-22 21:44:21 +08:00
|
|
|
},
|
|
|
|
}
|
|
|
|
corrections = append(corrections, corr)
|
|
|
|
}
|
|
|
|
|
|
|
|
return corrections, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetNameservers returns the nameservers for a domain.
|
2020-10-26 21:25:30 +08:00
|
|
|
func (api *hetznerProvider) GetNameservers(domain string) ([]*models.Nameserver, error) {
|
2020-10-22 21:44:21 +08:00
|
|
|
zone, err := api.getZone(domain)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
nameserver := make([]*models.Nameserver, len(zone.NameServers))
|
|
|
|
for i := range zone.NameServers {
|
|
|
|
nameserver[i] = &models.Nameserver{Name: zone.NameServers[i]}
|
|
|
|
}
|
|
|
|
return nameserver, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetZoneRecords gets the records of a zone and returns them in RecordConfig format.
|
2020-10-26 21:25:30 +08:00
|
|
|
func (api *hetznerProvider) GetZoneRecords(domain string) (models.Records, error) {
|
2020-10-22 21:44:21 +08:00
|
|
|
records, err := api.getAllRecords(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
|
|
|
|
}
|
|
|
|
|
|
|
|
// ListZones lists the zones on this account.
|
2020-10-26 21:25:30 +08:00
|
|
|
func (api *hetznerProvider) ListZones() ([]string, error) {
|
2020-10-22 21:44:21 +08:00
|
|
|
if err := api.getAllZones(); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
var zones []string
|
|
|
|
for i := range api.zones {
|
|
|
|
zones = append(zones, i)
|
|
|
|
}
|
|
|
|
return zones, nil
|
|
|
|
}
|