* plugin/sign: a plugin that signs zones Sign is a plugin that signs zone data (on disk). The README.md details what exactly happens to should be accurate related to the code. Signs are signed with a CSK, resigning and first time signing is all handled by *sign* plugin. Logging with a test zone looks something like this: ~~~ txt [INFO] plugin/sign: Signing "miek.nl." because open plugin/sign/testdata/db.miek.nl.signed: no such file or directory [INFO] plugin/sign: Signed "miek.nl." with key tags "59725" in 11.670985ms, saved in "plugin/sign/testdata/db.miek.nl.signed". Next: 2019-07-20T15:49:06.560Z [INFO] plugin/file: Successfully reloaded zone "miek.nl." in "plugin/sign/testdata/db.miek.nl.signed" with serial 1563636548 [INFO] plugin/sign: Signing "miek.nl." because resign was: 10m0s ago [INFO] plugin/sign: Signed "miek.nl." with key tags "59725" in 2.055895ms, saved in "plugin/sign/testdata/db.miek.nl.signed". Next: 2019-07-20T16:09:06.560Z [INFO] plugin/file: Successfully reloaded zone "miek.nl." in "plugin/sign/testdata/db.miek.nl.signed" with serial 1563637748 ~~~ Signed-off-by: Miek Gieben <miek@miek.nl> * Adjust readme and remove timestamps Signed-off-by: Miek Gieben <miek@miek.nl> * Comment on the newline Signed-off-by: Miek Gieben <miek@miek.nl> * Update plugin/sign/README.md Co-Authored-By: Michael Grosser <development@stp-ip.net>
222 lines
6.3 KiB
Go
222 lines
6.3 KiB
Go
package sign
|
|
|
|
import (
|
|
"fmt"
|
|
"io"
|
|
"os"
|
|
"path/filepath"
|
|
"time"
|
|
|
|
"github.com/coredns/coredns/plugin/file"
|
|
"github.com/coredns/coredns/plugin/file/tree"
|
|
clog "github.com/coredns/coredns/plugin/pkg/log"
|
|
|
|
"github.com/miekg/dns"
|
|
)
|
|
|
|
var log = clog.NewWithPlugin("sign")
|
|
|
|
// Signer holds the data needed to sign a zone file.
|
|
type Signer struct {
|
|
keys []Pair
|
|
origin string
|
|
dbfile string
|
|
directory string
|
|
jitter time.Duration
|
|
|
|
signedfile string
|
|
stop chan struct{}
|
|
|
|
expiration uint32
|
|
inception uint32
|
|
ttl uint32
|
|
}
|
|
|
|
// Sign signs a zone file according to the parameters in s.
|
|
func (s *Signer) Sign(now time.Time) (*file.Zone, error) {
|
|
rd, err := os.Open(s.dbfile)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
z, err := Parse(rd, s.origin, s.dbfile)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
s.inception, s.expiration = lifetime(now, s.jitter)
|
|
|
|
s.ttl = z.Apex.SOA.Header().Ttl
|
|
z.Apex.SOA.Serial = uint32(now.Unix())
|
|
|
|
for _, pair := range s.keys {
|
|
pair.Public.Header().Ttl = s.ttl // set TTL on key so it matches the RRSIG.
|
|
z.Insert(pair.Public)
|
|
z.Insert(pair.Public.ToDS(dns.SHA1))
|
|
z.Insert(pair.Public.ToDS(dns.SHA256))
|
|
z.Insert(pair.Public.ToDS(dns.SHA1).ToCDS())
|
|
z.Insert(pair.Public.ToDS(dns.SHA256).ToCDS())
|
|
z.Insert(pair.Public.ToCDNSKEY())
|
|
}
|
|
|
|
names, apex := names(s.origin, z)
|
|
ln := len(names)
|
|
|
|
var nsec *dns.NSEC
|
|
if apex {
|
|
nsec = NSEC(s.origin, names[(ln+1)%ln], s.ttl, []uint16{dns.TypeSOA, dns.TypeNS, dns.TypeRRSIG, dns.TypeNSEC})
|
|
z.Insert(nsec)
|
|
}
|
|
|
|
for _, pair := range s.keys {
|
|
rrsig, err := pair.signRRs([]dns.RR{z.Apex.SOA}, s.origin, s.ttl, s.inception, s.expiration)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
z.Insert(rrsig)
|
|
// NS apex may not be set if RR's have been discarded because the origin doesn't match.
|
|
if len(z.Apex.NS) > 0 {
|
|
rrsig, err = pair.signRRs(z.Apex.NS, s.origin, s.ttl, s.inception, s.expiration)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
z.Insert(rrsig)
|
|
}
|
|
if apex {
|
|
rrsig, err = pair.signRRs([]dns.RR{nsec}, s.origin, s.ttl, s.inception, s.expiration)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
z.Insert(rrsig)
|
|
}
|
|
}
|
|
|
|
// We are walking the tree in the same direction, so names[] can be used here to indicated the next element.
|
|
i := 1
|
|
err = z.Walk(func(e *tree.Elem, zrrs map[uint16][]dns.RR) error {
|
|
if !apex && e.Name() == s.origin {
|
|
nsec := NSEC(e.Name(), names[(ln+i)%ln], s.ttl, append(e.Types(), dns.TypeNS, dns.TypeSOA, dns.TypeNSEC, dns.TypeRRSIG))
|
|
z.Insert(nsec)
|
|
} else {
|
|
nsec := NSEC(e.Name(), names[(ln+i)%ln], s.ttl, append(e.Types(), dns.TypeNSEC, dns.TypeRRSIG))
|
|
z.Insert(nsec)
|
|
}
|
|
|
|
for t, rrs := range zrrs {
|
|
if t == dns.TypeRRSIG {
|
|
continue
|
|
}
|
|
for _, pair := range s.keys {
|
|
rrsig, err := pair.signRRs(rrs, s.origin, s.ttl, s.inception, s.expiration)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
e.Insert(rrsig)
|
|
}
|
|
}
|
|
i++
|
|
return nil
|
|
})
|
|
return z, err
|
|
}
|
|
|
|
// resign checks if the signed zone exists, or needs resigning.
|
|
func (s *Signer) resign() error {
|
|
signedfile := filepath.Join(s.directory, s.signedfile)
|
|
rd, err := os.Open(signedfile)
|
|
if err != nil && os.IsNotExist(err) {
|
|
return err
|
|
}
|
|
|
|
now := time.Now().UTC()
|
|
return resign(rd, now)
|
|
}
|
|
|
|
// resign will scan rd and check the signature on the SOA record. We will resign on the basis
|
|
// of 2 conditions:
|
|
// * either the inception is more than 6 days ago, or
|
|
// * we only have 1 week left on the signature
|
|
//
|
|
// All SOA signatures will be checked. If the SOA isn't found in the first 100
|
|
// records, we will resign the zone.
|
|
func resign(rd io.Reader, now time.Time) (why error) {
|
|
zp := dns.NewZoneParser(rd, ".", "resign")
|
|
zp.SetIncludeAllowed(true)
|
|
i := 0
|
|
|
|
for rr, ok := zp.Next(); ok; rr, ok = zp.Next() {
|
|
if err := zp.Err(); err != nil {
|
|
return err
|
|
}
|
|
|
|
switch x := rr.(type) {
|
|
case *dns.RRSIG:
|
|
if x.TypeCovered != dns.TypeSOA {
|
|
continue
|
|
}
|
|
incep, _ := time.Parse("20060102150405", dns.TimeToString(x.Inception))
|
|
// If too long ago, resign.
|
|
if now.Sub(incep) >= 0 && now.Sub(incep) > DurationResignDays {
|
|
return fmt.Errorf("inception %q was more than: %s ago from %s: %s", incep.Format(timeFmt), DurationResignDays, now.Format(timeFmt), now.Sub(incep))
|
|
}
|
|
// Inception hasn't even start yet.
|
|
if now.Sub(incep) < 0 {
|
|
return fmt.Errorf("inception %q date is in the future: %s", incep.Format(timeFmt), now.Sub(incep))
|
|
}
|
|
|
|
expire, _ := time.Parse("20060102150405", dns.TimeToString(x.Expiration))
|
|
if expire.Sub(now) < DurationExpireDays {
|
|
return fmt.Errorf("expiration %q is less than: %s away from %s: %s", expire.Format(timeFmt), DurationExpireDays, now.Format(timeFmt), expire.Sub(now))
|
|
}
|
|
}
|
|
i++
|
|
if i > 100 {
|
|
// 100 is a random number. A SOA record should be the first in the zonefile, but RFC 1035 doesn't actually mandate this. So it could
|
|
// be 3rd or even later. The number 100 looks crazy high enough that it will catch all weird zones, but not high enough to keep the CPU
|
|
// busy with parsing all the time.
|
|
return fmt.Errorf("no SOA RRSIG found in first 100 records")
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func signAndLog(s *Signer, why error) {
|
|
now := time.Now().UTC()
|
|
z, err := s.Sign(now)
|
|
log.Infof("Signing %q because %s", s.origin, why)
|
|
if err != nil {
|
|
log.Warningf("Error signing %q with key tags %q in %s: %s, next: %s", s.origin, keyTag(s.keys), time.Since(now), err, now.Add(DurationRefreshHours).Format(timeFmt))
|
|
return
|
|
}
|
|
|
|
if err := s.write(z); err != nil {
|
|
log.Warningf("Error signing %q: failed to move zone file into place: %s", s.origin, err)
|
|
return
|
|
}
|
|
log.Infof("Successfully signed zone %q in %q with key tags %q and %d SOA serial, elapsed %f, next: %s", s.origin, filepath.Join(s.directory, s.signedfile), keyTag(s.keys), z.Apex.SOA.Serial, time.Since(now).Seconds(), now.Add(DurationRefreshHours).Format(timeFmt))
|
|
}
|
|
|
|
// refresh checks every val if some zones need to be resigned.
|
|
func (s *Signer) refresh(val time.Duration) {
|
|
tick := time.NewTicker(val)
|
|
defer tick.Stop()
|
|
for {
|
|
select {
|
|
case <-s.stop:
|
|
return
|
|
case <-tick.C:
|
|
why := s.resign()
|
|
if why == nil {
|
|
continue
|
|
}
|
|
signAndLog(s, why)
|
|
}
|
|
}
|
|
}
|
|
|
|
func lifetime(now time.Time, jitter time.Duration) (uint32, uint32) {
|
|
incep := uint32(now.Add(DurationSignatureInceptionHours).Add(jitter).Unix())
|
|
expir := uint32(now.Add(DurationSignatureExpireDays).Unix())
|
|
return incep, expir
|
|
}
|