mirror of
https://github.com/StackExchange/dnscontrol.git
synced 2025-01-11 01:47:53 +08:00
253 lines
7.5 KiB
Go
253 lines
7.5 KiB
Go
package commands
|
|
|
|
import (
|
|
"fmt"
|
|
"log"
|
|
"os"
|
|
|
|
"github.com/urfave/cli/v2"
|
|
|
|
"github.com/StackExchange/dnscontrol/v3/models"
|
|
"github.com/StackExchange/dnscontrol/v3/pkg/nameservers"
|
|
"github.com/StackExchange/dnscontrol/v3/pkg/normalize"
|
|
"github.com/StackExchange/dnscontrol/v3/pkg/notifications"
|
|
"github.com/StackExchange/dnscontrol/v3/pkg/printer"
|
|
"github.com/StackExchange/dnscontrol/v3/providers"
|
|
"github.com/StackExchange/dnscontrol/v3/providers/config"
|
|
)
|
|
|
|
var _ = cmd(catMain, func() *cli.Command {
|
|
var args PreviewArgs
|
|
return &cli.Command{
|
|
Name: "preview",
|
|
Usage: "read live configuration and identify changes to be made, without applying them",
|
|
Action: func(ctx *cli.Context) error {
|
|
return exit(Preview(args))
|
|
},
|
|
Flags: args.flags(),
|
|
}
|
|
}())
|
|
|
|
// PreviewArgs contains all data/flags needed to run preview, independently of CLI
|
|
type PreviewArgs struct {
|
|
GetDNSConfigArgs
|
|
GetCredentialsArgs
|
|
FilterArgs
|
|
Notify bool
|
|
WarnChanges bool
|
|
}
|
|
|
|
func (args *PreviewArgs) flags() []cli.Flag {
|
|
flags := args.GetDNSConfigArgs.flags()
|
|
flags = append(flags, args.GetCredentialsArgs.flags()...)
|
|
flags = append(flags, args.FilterArgs.flags()...)
|
|
flags = append(flags, &cli.BoolFlag{
|
|
Name: "notify",
|
|
Destination: &args.Notify,
|
|
Usage: `set to true to send notifications to configured destinations`,
|
|
})
|
|
flags = append(flags, &cli.BoolFlag{
|
|
Name: "expect-no-changes",
|
|
Destination: &args.WarnChanges,
|
|
Usage: `set to true for non-zero return code if there are changes`,
|
|
})
|
|
return flags
|
|
}
|
|
|
|
var _ = cmd(catMain, func() *cli.Command {
|
|
var args PushArgs
|
|
return &cli.Command{
|
|
Name: "push",
|
|
Usage: "identify changes to be made, and perform them",
|
|
Action: func(ctx *cli.Context) error {
|
|
return exit(Push(args))
|
|
},
|
|
Flags: args.flags(),
|
|
}
|
|
}())
|
|
|
|
// PushArgs contains all data/flags needed to run push, independently of CLI
|
|
type PushArgs struct {
|
|
PreviewArgs
|
|
Interactive bool
|
|
}
|
|
|
|
func (args *PushArgs) flags() []cli.Flag {
|
|
flags := args.PreviewArgs.flags()
|
|
flags = append(flags, &cli.BoolFlag{
|
|
Name: "i",
|
|
Destination: &args.Interactive,
|
|
Usage: "Interactive. Confirm or Exclude each correction before they run",
|
|
})
|
|
return flags
|
|
}
|
|
|
|
// Preview implements the preview subcommand.
|
|
func Preview(args PreviewArgs) error {
|
|
return run(args, false, false, printer.DefaultPrinter)
|
|
}
|
|
|
|
// Push implements the push subcommand.
|
|
func Push(args PushArgs) error {
|
|
return run(args.PreviewArgs, true, args.Interactive, printer.DefaultPrinter)
|
|
}
|
|
|
|
// run is the main routine common to preview/push
|
|
func run(args PreviewArgs, push bool, interactive bool, out printer.CLI) error {
|
|
// TODO: make truly CLI independent. Perhaps return results on a channel as they occur
|
|
cfg, err := GetDNSConfig(args.GetDNSConfigArgs)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
errs := normalize.ValidateAndNormalizeConfig(cfg)
|
|
if PrintValidationErrors(errs) {
|
|
return fmt.Errorf("Exiting due to validation errors")
|
|
}
|
|
// TODO:
|
|
notifier, err := InitializeProviders(args.CredsFile, cfg, args.Notify)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
anyErrors := false
|
|
totalCorrections := 0
|
|
DomainLoop:
|
|
for _, domain := range cfg.Domains {
|
|
if !args.shouldRunDomain(domain.Name) {
|
|
continue
|
|
}
|
|
out.StartDomain(domain.Name)
|
|
nsList, err := nameservers.DetermineNameservers(domain)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
domain.Nameservers = nsList
|
|
nameservers.AddNSRecords(domain)
|
|
for _, provider := range domain.DNSProviderInstances {
|
|
dc, err := domain.Copy()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
shouldrun := args.shouldRunProvider(provider.Name, dc)
|
|
out.StartDNSProvider(provider.Name, !shouldrun)
|
|
if !shouldrun {
|
|
continue
|
|
}
|
|
corrections, err := provider.Driver.GetDomainCorrections(dc)
|
|
out.EndProvider(len(corrections), err)
|
|
if err != nil {
|
|
anyErrors = true
|
|
continue DomainLoop
|
|
}
|
|
totalCorrections += len(corrections)
|
|
anyErrors = printOrRunCorrections(domain.Name, provider.Name, corrections, out, push, interactive, notifier) || anyErrors
|
|
}
|
|
run := args.shouldRunProvider(domain.RegistrarName, domain)
|
|
out.StartRegistrar(domain.RegistrarName, !run)
|
|
if !run {
|
|
continue
|
|
}
|
|
if len(domain.Nameservers) == 0 && domain.Metadata["no_ns"] != "true" {
|
|
out.Warnf("No nameservers declared; skipping registrar. Add {no_ns:'true'} to force.\n")
|
|
continue
|
|
}
|
|
dc, err := domain.Copy()
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
corrections, err := domain.RegistrarInstance.Driver.GetRegistrarCorrections(dc)
|
|
out.EndProvider(len(corrections), err)
|
|
if err != nil {
|
|
anyErrors = true
|
|
continue
|
|
}
|
|
totalCorrections += len(corrections)
|
|
anyErrors = printOrRunCorrections(domain.Name, domain.RegistrarName, corrections, out, push, interactive, notifier) || anyErrors
|
|
}
|
|
if os.Getenv("TEAMCITY_VERSION") != "" {
|
|
fmt.Fprintf(os.Stderr, "##teamcity[buildStatus status='SUCCESS' text='%d corrections']", totalCorrections)
|
|
}
|
|
notifier.Done()
|
|
out.Printf("Done. %d corrections.\n", totalCorrections)
|
|
if anyErrors {
|
|
return fmt.Errorf("Completed with errors")
|
|
}
|
|
if totalCorrections != 0 && args.WarnChanges {
|
|
return fmt.Errorf("There are pending changes")
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// InitializeProviders takes a creds file path and a DNSConfig object. Creates all providers with the proper types, and returns them.
|
|
// nonDefaultProviders is a list of providers that should not be run unless explicitly asked for by flags.
|
|
func InitializeProviders(credsFile string, cfg *models.DNSConfig, notifyFlag bool) (notify notifications.Notifier, err error) {
|
|
var providerConfigs map[string]map[string]string
|
|
var notificationCfg map[string]string
|
|
defer func() {
|
|
notify = notifications.Init(notificationCfg)
|
|
}()
|
|
providerConfigs, err = config.LoadProviderConfigs(credsFile)
|
|
if err != nil {
|
|
return
|
|
}
|
|
if notifyFlag {
|
|
notificationCfg = providerConfigs["notifications"]
|
|
}
|
|
isNonDefault := map[string]bool{}
|
|
for name, vals := range providerConfigs {
|
|
// add "_exclude_from_defaults":"true" to a provider to exclude it from being run unless
|
|
// -providers=all or -providers=name
|
|
if vals["_exclude_from_defaults"] == "true" {
|
|
isNonDefault[name] = true
|
|
}
|
|
}
|
|
registrars := map[string]providers.Registrar{}
|
|
dnsProviders := map[string]providers.DNSServiceProvider{}
|
|
for _, d := range cfg.Domains {
|
|
if registrars[d.RegistrarName] == nil {
|
|
rCfg := cfg.RegistrarsByName[d.RegistrarName]
|
|
r, err := providers.CreateRegistrar(rCfg.Type, providerConfigs[d.RegistrarName])
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
registrars[d.RegistrarName] = r
|
|
}
|
|
d.RegistrarInstance.Driver = registrars[d.RegistrarName]
|
|
d.RegistrarInstance.IsDefault = !isNonDefault[d.RegistrarName]
|
|
for _, pInst := range d.DNSProviderInstances {
|
|
if dnsProviders[pInst.Name] == nil {
|
|
dCfg := cfg.DNSProvidersByName[pInst.Name]
|
|
prov, err := providers.CreateDNSProvider(dCfg.Type, providerConfigs[dCfg.Name], dCfg.Metadata)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
dnsProviders[pInst.Name] = prov
|
|
}
|
|
pInst.Driver = dnsProviders[pInst.Name]
|
|
pInst.IsDefault = !isNonDefault[pInst.Name]
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
func printOrRunCorrections(domain string, provider string, corrections []*models.Correction, out printer.CLI, push bool, interactive bool, notifier notifications.Notifier) (anyErrors bool) {
|
|
anyErrors = false
|
|
if len(corrections) == 0 {
|
|
return false
|
|
}
|
|
for i, correction := range corrections {
|
|
out.PrintCorrection(i, correction)
|
|
var err error
|
|
if push {
|
|
if interactive && !out.PromptToRun() {
|
|
continue
|
|
}
|
|
err = correction.F()
|
|
out.EndCorrection(err)
|
|
if err != nil {
|
|
anyErrors = true
|
|
}
|
|
}
|
|
notifier.Notify(domain, provider, correction.Msg, err, !push)
|
|
}
|
|
return anyErrors
|
|
}
|