mirror of
https://github.com/mjl-/mox.git
synced 2025-01-26 06:45:53 +03:00
d1b87cdb0d
since we are now at go1.21 as minimum.
84 lines
2.3 KiB
Go
84 lines
2.3 KiB
Go
package tlsrpt
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"log/slog"
|
|
"time"
|
|
|
|
"github.com/mjl-/mox/dns"
|
|
"github.com/mjl-/mox/mlog"
|
|
"github.com/mjl-/mox/stub"
|
|
)
|
|
|
|
var (
|
|
MetricLookup stub.HistogramVec = stub.HistogramVecIgnore{}
|
|
)
|
|
|
|
var (
|
|
ErrNoRecord = errors.New("tlsrpt: no tlsrpt dns txt record")
|
|
ErrMultipleRecords = errors.New("tlsrpt: multiple tlsrpt records") // Must be treated as if domain does not implement TLSRPT.
|
|
ErrDNS = errors.New("tlsrpt: temporary error")
|
|
ErrRecordSyntax = errors.New("tlsrpt: record syntax error")
|
|
)
|
|
|
|
// Lookup looks up a TLSRPT DNS TXT record for domain at "_smtp._tls.<domain>" and
|
|
// parses it.
|
|
func Lookup(ctx context.Context, elog *slog.Logger, resolver dns.Resolver, domain dns.Domain) (rrecord *Record, rtxt string, rerr error) {
|
|
log := mlog.New("tlsrpt", elog)
|
|
start := time.Now()
|
|
defer func() {
|
|
result := "ok"
|
|
if rerr != nil {
|
|
if errors.Is(rerr, ErrNoRecord) {
|
|
result = "notfound"
|
|
} else if errors.Is(rerr, ErrMultipleRecords) {
|
|
result = "multiple"
|
|
} else if errors.Is(rerr, ErrDNS) {
|
|
result = "temperror"
|
|
} else if errors.Is(rerr, ErrRecordSyntax) {
|
|
result = "malformed"
|
|
} else {
|
|
result = "error"
|
|
}
|
|
}
|
|
MetricLookup.ObserveLabels(float64(time.Since(start))/float64(time.Second), result)
|
|
log.Debugx("tlsrpt lookup result", rerr,
|
|
slog.Any("domain", domain),
|
|
slog.Any("record", rrecord),
|
|
slog.Duration("duration", time.Since(start)))
|
|
}()
|
|
|
|
name := "_smtp._tls." + domain.ASCII + "."
|
|
txts, _, err := dns.WithPackage(resolver, "tlsrpt").LookupTXT(ctx, name)
|
|
if dns.IsNotFound(err) {
|
|
return nil, "", ErrNoRecord
|
|
} else if err != nil {
|
|
return nil, "", fmt.Errorf("%w: %s", ErrDNS, err)
|
|
}
|
|
|
|
var text string
|
|
var record *Record
|
|
for _, txt := range txts {
|
|
r, istlsrpt, err := ParseRecord(txt)
|
|
if !istlsrpt {
|
|
// This is a loose but probably reasonable interpretation of ../rfc/8460:375 which
|
|
// wants us to discard otherwise valid records that start with e.g. "v=TLSRPTv1 ;"
|
|
// (note the space before the ";") when multiple TXT records were returned.
|
|
continue
|
|
}
|
|
if err != nil {
|
|
return nil, "", fmt.Errorf("parsing record: %w", err)
|
|
}
|
|
if record != nil {
|
|
return nil, "", ErrMultipleRecords
|
|
}
|
|
record = r
|
|
text = txt
|
|
}
|
|
if record == nil {
|
|
return nil, "", ErrNoRecord
|
|
}
|
|
return record, text, nil
|
|
}
|