mirror of
https://github.com/StackExchange/dnscontrol.git
synced 2025-01-11 09:59:59 +08:00
5e06c249d6
Co-authored-by: Yannik Sembritzki <yannik@sembritzki.org> Co-authored-by: Tom Limoncelli <tlimoncelli@stackoverflow.com>
198 lines
7.1 KiB
Go
198 lines
7.1 KiB
Go
package providers
|
|
|
|
import (
|
|
"encoding/json"
|
|
"fmt"
|
|
"log"
|
|
|
|
"github.com/StackExchange/dnscontrol/v3/models"
|
|
)
|
|
|
|
// Registrar is an interface for a domain registrar. It can return a list of needed corrections to be applied in the future. Implement this only if the provider is a "registrar" (i.e. can update the NS records of the parent to a domain).
|
|
type Registrar interface {
|
|
models.Registrar
|
|
}
|
|
|
|
// DNSServiceProvider is able to generate a set of corrections that need to be made to correct records for a domain. Implement this only if the provider is a DNS Service Provider (can update records in a DNS zone).
|
|
type DNSServiceProvider interface {
|
|
models.DNSProvider
|
|
}
|
|
|
|
// ZoneCreator should be implemented by providers that have the ability to create zones
|
|
// (used for automatically creating zones if they don't exist)
|
|
type ZoneCreator interface {
|
|
EnsureZoneExists(domain string) error
|
|
}
|
|
|
|
// ZoneLister should be implemented by providers that have the
|
|
// ability to list the zones they manage. This facilitates using the
|
|
// "get-zones" command for "all" zones.
|
|
type ZoneLister interface {
|
|
ListZones() ([]string, error)
|
|
}
|
|
|
|
// RegistrarInitializer is a function to create a registrar. Function will be passed the unprocessed json payload from the configuration file for the given provider.
|
|
type RegistrarInitializer func(map[string]string) (Registrar, error)
|
|
|
|
// RegistrarTypes stores initializer for each registrar.
|
|
var RegistrarTypes = map[string]RegistrarInitializer{}
|
|
|
|
// DspInitializer is a function to create a DNS service provider. Function will be passed the unprocessed json payload from the configuration file for the given provider.
|
|
type DspInitializer func(map[string]string, json.RawMessage) (DNSServiceProvider, error)
|
|
|
|
// RecordAuditor is a function that verifies that all the records
|
|
// are supportable by this provider. It returns a list of errors
|
|
// detailing records that this provider can not support.
|
|
type RecordAuditor func([]*models.RecordConfig) []error
|
|
|
|
// DspFuncs lists functions registered with a provider.
|
|
type DspFuncs struct {
|
|
Initializer DspInitializer
|
|
RecordAuditor RecordAuditor
|
|
}
|
|
|
|
// DNSProviderTypes stores initializer for each DSP.
|
|
var DNSProviderTypes = map[string]DspFuncs{}
|
|
|
|
// RegisterRegistrarType adds a registrar type to the registry by providing a suitable initialization function.
|
|
func RegisterRegistrarType(name string, init RegistrarInitializer, pm ...ProviderMetadata) {
|
|
if _, ok := RegistrarTypes[name]; ok {
|
|
log.Fatalf("Cannot register registrar type %q multiple times", name)
|
|
}
|
|
RegistrarTypes[name] = init
|
|
unwrapProviderCapabilities(name, pm)
|
|
}
|
|
|
|
// RegisterDomainServiceProviderType adds a dsp to the registry with the given initialization function.
|
|
func RegisterDomainServiceProviderType(name string, fns DspFuncs, pm ...ProviderMetadata) {
|
|
if _, ok := DNSProviderTypes[name]; ok {
|
|
log.Fatalf("Cannot register registrar type %q multiple times", name)
|
|
}
|
|
DNSProviderTypes[name] = fns
|
|
unwrapProviderCapabilities(name, pm)
|
|
}
|
|
|
|
// CreateRegistrar initializes a registrar instance from given credentials.
|
|
func CreateRegistrar(rType string, config map[string]string) (Registrar, error) {
|
|
var err error
|
|
rType, err = beCompatible(rType, config)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
initer, ok := RegistrarTypes[rType]
|
|
if !ok {
|
|
return nil, fmt.Errorf("no such registrar type: %q", rType)
|
|
}
|
|
return initer(config)
|
|
}
|
|
|
|
// CreateDNSProvider initializes a dns provider instance from given credentials.
|
|
func CreateDNSProvider(providerTypeName string, config map[string]string, meta json.RawMessage) (DNSServiceProvider, error) {
|
|
var err error
|
|
providerTypeName, err = beCompatible(providerTypeName, config)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
p, ok := DNSProviderTypes[providerTypeName]
|
|
if !ok {
|
|
return nil, fmt.Errorf("no such DNS service provider: %q", providerTypeName)
|
|
}
|
|
return p.Initializer(config, meta)
|
|
}
|
|
|
|
// beCompatible looks up
|
|
func beCompatible(n string, config map[string]string) (string, error) {
|
|
// Pre 4.0: If n is a placeholder, substitute the TYPE from creds.json.
|
|
// 4.0: Require TYPE from creds.json.
|
|
|
|
ct := config["TYPE"]
|
|
// If a placeholder value was specified...
|
|
if n == "" || n == "-" {
|
|
// But no TYPE exists in creds.json...
|
|
if ct == "" {
|
|
return "-", fmt.Errorf("creds.json entry missing TYPE field")
|
|
}
|
|
// Otherwise, use the value from creds.json.
|
|
return ct, nil
|
|
}
|
|
|
|
// Pre 4.0: The user specified the name manually.
|
|
// Cross check to detect user-error.
|
|
if ct != "" && n != ct {
|
|
return "", fmt.Errorf("creds.json entry mismatch: specified=%q TYPE=%q", n, ct)
|
|
}
|
|
// Seems like the user did it the right way. Return the original value.
|
|
return n, nil
|
|
|
|
// NB(tlim): My hope is that in 4.0 this entire function will simply be the
|
|
// following, but I may be wrong:
|
|
//return config["TYPE"], nil
|
|
}
|
|
|
|
// AuditRecords calls the RecordAudit function for a provider.
|
|
func AuditRecords(dType string, rcs models.Records) []error {
|
|
p, ok := DNSProviderTypes[dType]
|
|
if !ok {
|
|
return []error{fmt.Errorf("unknown DNS service provider type: %q", dType)}
|
|
}
|
|
if p.RecordAuditor == nil {
|
|
return []error{fmt.Errorf("DNS service provider type %q has no RecordAuditor", dType)}
|
|
}
|
|
return p.RecordAuditor(rcs)
|
|
}
|
|
|
|
// None is a basic provider type that does absolutely nothing. Can be useful as a placeholder for third parties or unimplemented providers.
|
|
type None struct{}
|
|
|
|
// GetRegistrarCorrections returns corrections to update registrars.
|
|
func (n None) GetRegistrarCorrections(dc *models.DomainConfig) ([]*models.Correction, error) {
|
|
return nil, nil
|
|
}
|
|
|
|
// GetNameservers returns the current nameservers for a domain.
|
|
func (n None) GetNameservers(string) ([]*models.Nameserver, error) {
|
|
return nil, nil
|
|
}
|
|
|
|
// GetZoneRecords gets the records of a zone and returns them in RecordConfig format.
|
|
func (n None) GetZoneRecords(domain string) (models.Records, error) {
|
|
return nil, fmt.Errorf("not implemented")
|
|
// This enables the get-zones subcommand.
|
|
// Implement this by extracting the code from GetDomainCorrections into
|
|
// a single function. For most providers this should be relatively easy.
|
|
}
|
|
|
|
// GetDomainCorrections returns corrections to update a domain.
|
|
func (n None) GetDomainCorrections(dc *models.DomainConfig) ([]*models.Correction, error) {
|
|
return nil, nil
|
|
}
|
|
|
|
func init() {
|
|
RegisterRegistrarType("NONE", func(map[string]string) (Registrar, error) {
|
|
return None{}, nil
|
|
})
|
|
}
|
|
|
|
// CustomRType stores an rtype that is only valid for this DSP.
|
|
type CustomRType struct {
|
|
Name string
|
|
Provider string
|
|
RealType string
|
|
}
|
|
|
|
// RegisterCustomRecordType registers a record type that is only valid for one provider.
|
|
// provider is the registered type of provider this is valid with
|
|
// name is the record type as it will appear in the js. (should be something like $PROVIDER_FOO)
|
|
// realType is the record type it will be replaced with after validation
|
|
func RegisterCustomRecordType(name, provider, realType string) {
|
|
customRecordTypes[name] = &CustomRType{Name: name, Provider: provider, RealType: realType}
|
|
}
|
|
|
|
// GetCustomRecordType returns a registered custom record type, or nil if none
|
|
func GetCustomRecordType(rType string) *CustomRType {
|
|
return customRecordTypes[rType]
|
|
}
|
|
|
|
var customRecordTypes = map[string]*CustomRType{}
|