// Package acme provides a means of performing Let's Encrypt DNS challenges via a DNSConfig package acme import ( "crypto/x509" "encoding/pem" "fmt" "io/ioutil" "log" "net/url" "sort" "strings" "time" "github.com/StackExchange/dnscontrol/v3/models" "github.com/StackExchange/dnscontrol/v3/pkg/nameservers" "github.com/StackExchange/dnscontrol/v3/pkg/notifications" "github.com/go-acme/lego/certcrypto" "github.com/go-acme/lego/certificate" "github.com/go-acme/lego/challenge" "github.com/go-acme/lego/challenge/dns01" "github.com/go-acme/lego/lego" acmelog "github.com/go-acme/lego/log" ) // CertConfig describes a certificate's configuration. type CertConfig struct { CertName string `json:"cert_name"` Names []string `json:"names"` UseECC bool `json:"use_ecc"` MustStaple bool `json:"must_staple"` } // Client is an interface for systems that issue or renew certs. type Client interface { IssueOrRenewCert(config *CertConfig, renewUnder int, verbose bool) (bool, error) } type certManager struct { email string acmeDirectory string acmeHost string storage Storage cfg *models.DNSConfig domains map[string]*models.DomainConfig originalDomains []*models.DomainConfig notifier notifications.Notifier account *Account waitedOnce bool } const ( // LetsEncryptLive is the endpoint for updates (production). LetsEncryptLive = "https://acme-v02.api.letsencrypt.org/directory" // LetsEncryptStage is the endpoint for the staging area. LetsEncryptStage = "https://acme-staging-v02.api.letsencrypt.org/directory" ) // New is a factory for acme clients. func New(cfg *models.DNSConfig, directory string, email string, server string, notify notifications.Notifier) (Client, error) { return commonNew(cfg, directoryStorage(directory), email, server, notify) } func commonNew(cfg *models.DNSConfig, storage Storage, email string, server string, notify notifications.Notifier) (Client, error) { u, err := url.Parse(server) if err != nil || u.Host == "" { return nil, fmt.Errorf("ACME directory '%s' is not a valid URL", server) } c := &certManager{ storage: storage, email: email, acmeDirectory: server, acmeHost: u.Host, cfg: cfg, domains: map[string]*models.DomainConfig{}, notifier: notify, } acct, err := c.getOrCreateAccount() if err != nil { return nil, err } c.account = acct return c, nil } // NewVault is a factory for new vaunt clients. func NewVault(cfg *models.DNSConfig, vaultPath string, email string, server string, notify notifications.Notifier) (Client, error) { storage, err := makeVaultStorage(vaultPath) if err != nil { return nil, err } return commonNew(cfg, storage, email, server, notify) } // IssueOrRenewCert will obtain a certificate with the given name if it does not exist, // or renew it if it is close enough to the expiration date. // It will return true if it issued or updated the certificate. func (c *certManager) IssueOrRenewCert(cfg *CertConfig, renewUnder int, verbose bool) (bool, error) { if !verbose { acmelog.Logger = log.New(ioutil.Discard, "", 0) } defer c.finalCleanUp() log.Printf("Checking certificate [%s]", cfg.CertName) existing, err := c.storage.GetCertificate(cfg.CertName) if err != nil { return false, err } var client *lego.Client var action = func() (*certificate.Resource, error) { return client.Certificate.Obtain(certificate.ObtainRequest{ Bundle: true, Domains: cfg.Names, MustStaple: cfg.MustStaple, }) } if existing == nil { log.Println("No existing cert found. Issuing new...") } else { names, daysLeft, err := getCertInfo(existing.Certificate) if err != nil { return false, err } log.Printf("Found existing cert. %0.2f days remaining.", daysLeft) namesOK := dnsNamesEqual(cfg.Names, names) if daysLeft >= float64(renewUnder) && namesOK { log.Println("Nothing to do") //nothing to do return false, nil } if !namesOK { log.Println("DNS Names don't match expected set. Reissuing.") } else { log.Println("Renewing cert") action = func() (*certificate.Resource, error) { return client.Certificate.Renew(*existing, true, cfg.MustStaple) } } } kt := certcrypto.RSA2048 if cfg.UseECC { kt = certcrypto.EC256 } config := lego.NewConfig(c.account) config.CADirURL = c.acmeDirectory config.Certificate.KeyType = kt client, err = lego.NewClient(config) if err != nil { return false, err } client.Challenge.Remove(challenge.HTTP01) client.Challenge.Remove(challenge.TLSALPN01) client.Challenge.SetDNS01Provider(c, dns01.WrapPreCheck(c.preCheckDNS)) certResource, err := action() if err != nil { return false, err } fmt.Printf("Obtained certificate for %s\n", cfg.CertName) if err = c.storage.StoreCertificate(cfg.CertName, certResource); err != nil { return true, err } return true, nil } func getCertInfo(pemBytes []byte) (names []string, remaining float64, err error) { block, _ := pem.Decode(pemBytes) if block == nil { return nil, 0, fmt.Errorf("invalid certificate PEM data") } cert, err := x509.ParseCertificate(block.Bytes) if err != nil { return nil, 0, err } // FIXME(tlim): should use time.Until instead of t.Sub(time.Now()) (S1024) var daysLeft = float64(cert.NotAfter.Sub(time.Now())) / float64(time.Hour*24) return cert.DNSNames, daysLeft, nil } // checks two lists of sans to make sure they have all the same names in them. func dnsNamesEqual(a []string, b []string) bool { if len(a) != len(b) { return false } sort.Strings(a) sort.Strings(b) for i, s := range a { if b[i] != s { return false } } return true } func (c *certManager) Present(domain, token, keyAuth string) (e error) { d := c.cfg.DomainContainingFQDN(domain) name := d.Name if seen := c.domains[name]; seen != nil { // we've already pre-processed this domain, just need to add to it. d = seen } else { // one-time tasks to get this domain ready. // if multiple validations on a single domain, we don't need to rebuild all this. // fix NS records for this domain's DNS providers nsList, err := nameservers.DetermineNameservers(d) if err != nil { return err } d.Nameservers = nsList nameservers.AddNSRecords(d) // make sure we have the latest config before we change anything. // alternately, we could avoid a lot of this trouble if we really really trusted no-purge in all cases if err := c.ensureNoPendingCorrections(d); err != nil { return err } // copy domain and work from copy from now on. That way original config can be used to "restore" when we are all done. copy, err := d.Copy() if err != nil { return err } c.originalDomains = append(c.originalDomains, d) c.domains[name] = copy d = copy } fqdn, val := dns01.GetRecord(domain, keyAuth) txt := &models.RecordConfig{Type: "TXT"} txt.SetTargetTXT(val) txt.SetLabelFromFQDN(fqdn, d.Name) d.Records = append(d.Records, txt) return c.getAndRunCorrections(d) } func (c *certManager) ensureNoPendingCorrections(d *models.DomainConfig) error { corrections, err := c.getCorrections(d) if err != nil { return err } if len(corrections) != 0 { // TODO: maybe allow forcing through this check. for _, c := range corrections { fmt.Println(c.Msg) } return fmt.Errorf("found %d pending corrections for %s. Not going to proceed issuing certificates", len(corrections), d.Name) } return nil } // IgnoredProviders is a lit of provider names that should not be used to fill challenges. var IgnoredProviders = map[string]bool{} func (c *certManager) getCorrections(d *models.DomainConfig) ([]*models.Correction, error) { cs := []*models.Correction{} for _, p := range d.DNSProviderInstances { if IgnoredProviders[p.Name] { continue } dc, err := d.Copy() if err != nil { return nil, err } corrections, err := p.Driver.GetDomainCorrections(dc) if err != nil { return nil, err } for _, c := range corrections { c.Msg = fmt.Sprintf("[%s] %s", p.Name, strings.TrimSpace(c.Msg)) } cs = append(cs, corrections...) } return cs, nil } func (c *certManager) getAndRunCorrections(d *models.DomainConfig) error { cs, err := c.getCorrections(d) if err != nil { return err } fmt.Printf("%d corrections\n", len(cs)) for _, corr := range cs { fmt.Printf("Running [%s]\n", corr.Msg) err = corr.F() c.notifier.Notify(d.Name, "certs", corr.Msg, err, false) if err != nil { return err } } return nil } func (c *certManager) CleanUp(domain, token, keyAuth string) error { // do nothing for now. We will do a final clean up step at the very end. return nil } func (c *certManager) finalCleanUp() error { log.Println("Cleaning up all records we made") var lastError error for _, d := range c.originalDomains { if err := c.getAndRunCorrections(d); err != nil { log.Printf("ERROR cleaning up: %s", err) lastError = err } } return lastError }