mirror of
https://github.com/StackExchange/dnscontrol.git
synced 2024-11-11 01:33:58 +08:00
7b8d608019
* Stable comparison of metadata (#239) Iterating over a map in Go never produces twice the same ordering. Thus when comparing two metadata map with more than one key, the `differ` is always finding differences. To properly compare records metadata, we need to iterate the maps in a deterministic way. Signed-off-by: Brice Figureau <brice@daysofwonder.com> * Support for Route53 ALIAS record type (#239) Route53 ALIAS doesn't behave like a regular ALIAS, and is much more limited as its target can only be some specific AWS resources or another record in the same zone. According to #239, this change adds a new directive R53_ALIAS which implements this specific alias. This record type can only be used with the Route53 provider. This directive usage looks like this: ```js D("example.com", REGISTRAR, DnsProvider("ROUTE53"), R53_ALIAS("foo1", "A", "bar") // record in same zone R53_ALIAS("foo2", "A", "blahblah.elasticloadbalancing.us-west-1.amazonaws.com", R53_ZONE('Z368ELLRRE2KJ0')) // ELB in us-west-1 ``` Unfortunately, Route53 requires indicating the hosted zone id where the target is defined (those are listed in AWS documentation, see the R53_ALIAS documentation for links).
136 lines
4.2 KiB
Go
136 lines
4.2 KiB
Go
package providers
|
|
|
|
import (
|
|
"log"
|
|
)
|
|
|
|
// Capability is a bitmasked set of "features" that a provider supports. Only use constants from this package.
|
|
type Capability uint32
|
|
|
|
const (
|
|
// If you add something to this list, you probably want to add it to pkg/normalize/validate.go checkProviderCapabilities() or somewhere near there.
|
|
|
|
// CanUseAlias indicates the provider support ALIAS records (or flattened CNAMES). Up to the provider to translate them to the appropriate record type.
|
|
CanUseAlias Capability = iota
|
|
|
|
// CanUseCAA indicates the provider can handle CAA records
|
|
CanUseCAA
|
|
|
|
// CanUsePTR indicates the provider can handle PTR records
|
|
CanUsePTR
|
|
|
|
// CanUseSRV indicates the provider can handle SRV records
|
|
CanUseSRV
|
|
|
|
// CanUseTLSA indicates the provider can handle TLSA records
|
|
CanUseTLSA
|
|
|
|
// CanUseTXTMulti indicates the provider can handle TXT records with multiple strings
|
|
CanUseTXTMulti
|
|
|
|
// CantUseNOPURGE indicates NO_PURGE is broken for this provider. To make it
|
|
// work would require complex emulation of an incremental update mechanism,
|
|
// so it is easier to simply mark this feature as not working for this
|
|
// provider.
|
|
CantUseNOPURGE
|
|
|
|
// DocOfficiallySupported means it is actively used and maintained by stack exchange
|
|
DocOfficiallySupported
|
|
// DocDualHost means provider allows full management of apex NS records, so we can safely dual-host with anothe provider
|
|
DocDualHost
|
|
// DocCreateDomains means provider can add domains with the `dnscontrol create-domains` command
|
|
DocCreateDomains
|
|
|
|
// CanUseRoute53Alias indicates the provider support the specific R53_ALIAS records that only the Route53 provider supports
|
|
CanUseRoute53Alias
|
|
)
|
|
|
|
var providerCapabilities = map[string]map[Capability]bool{}
|
|
|
|
// ProviderHasCabability returns true if provider has capability.
|
|
func ProviderHasCabability(pType string, cap Capability) bool {
|
|
if providerCapabilities[pType] == nil {
|
|
return false
|
|
}
|
|
return providerCapabilities[pType][cap]
|
|
}
|
|
|
|
// DocumentationNote is a way for providers to give more detail about what features they support.
|
|
type DocumentationNote struct {
|
|
HasFeature bool
|
|
Unimplemented bool
|
|
Comment string
|
|
Link string
|
|
}
|
|
|
|
// DocumentationNotes is a full list of notes for a single provider
|
|
type DocumentationNotes map[Capability]*DocumentationNote
|
|
|
|
// ProviderMetadata is a common interface for DocumentationNotes and Capability to be used interchangably
|
|
type ProviderMetadata interface{}
|
|
|
|
// Notes is a collection of all documentation notes, keyed by provider type
|
|
var Notes = map[string]DocumentationNotes{}
|
|
|
|
func unwrapProviderCapabilities(pName string, meta []ProviderMetadata) {
|
|
if providerCapabilities[pName] == nil {
|
|
providerCapabilities[pName] = map[Capability]bool{}
|
|
}
|
|
for _, pm := range meta {
|
|
switch x := pm.(type) {
|
|
case Capability:
|
|
providerCapabilities[pName][x] = true
|
|
case DocumentationNotes:
|
|
if Notes[pName] == nil {
|
|
Notes[pName] = DocumentationNotes{}
|
|
}
|
|
for k, v := range x {
|
|
Notes[pName][k] = v
|
|
providerCapabilities[pName][k] = v.HasFeature
|
|
}
|
|
default:
|
|
log.Fatalf("Unrecognized ProviderMetadata type: %T", pm)
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
// Can is a small helper for concisely creating Documentation Notes
|
|
// comments are variadic for easy ommission. First is comment, second is link, the rest are ignored.
|
|
func Can(comments ...string) *DocumentationNote {
|
|
n := &DocumentationNote{
|
|
HasFeature: true,
|
|
}
|
|
n.addStrings(comments)
|
|
return n
|
|
}
|
|
|
|
// Cannot is a small helper for concisely creating Documentation Notes
|
|
// comments are variadic for easy ommission. First is comment, second is link, the rest are ignored.
|
|
func Cannot(comments ...string) *DocumentationNote {
|
|
n := &DocumentationNote{
|
|
HasFeature: false,
|
|
}
|
|
n.addStrings(comments)
|
|
return n
|
|
}
|
|
|
|
// Unimplemented is a small helper for concisely creating Documentation Notes
|
|
// comments are variadic for easy ommission. First is comment, second is link, the rest are ignored.
|
|
func Unimplemented(comments ...string) *DocumentationNote {
|
|
n := &DocumentationNote{
|
|
HasFeature: false,
|
|
Unimplemented: true,
|
|
}
|
|
n.addStrings(comments)
|
|
return n
|
|
}
|
|
|
|
func (n *DocumentationNote) addStrings(comments []string) {
|
|
if len(comments) > 0 {
|
|
n.Comment = comments[0]
|
|
}
|
|
if len(comments) > 1 {
|
|
n.Link = comments[1]
|
|
}
|
|
}
|