blocky/resolver/strict_resolver.go

100 lines
2.5 KiB
Go

package resolver
import (
"context"
"errors"
"fmt"
"strings"
"sync/atomic"
"github.com/0xERR0R/blocky/config"
"github.com/0xERR0R/blocky/model"
"github.com/0xERR0R/blocky/util"
"github.com/sirupsen/logrus"
)
const (
strictResolverType = "strict"
)
// StrictResolver delegates the DNS message strictly to the first configured upstream resolver
// if it can't provide the answer in time the next resolver is used
type StrictResolver struct {
configurable[*config.UpstreamGroup]
typed
resolvers atomic.Pointer[[]*upstreamResolverStatus]
}
// NewStrictResolver creates a new strict resolver instance
func NewStrictResolver(
ctx context.Context, cfg config.UpstreamGroup, bootstrap *Bootstrap,
) (*StrictResolver, error) {
r := newStrictResolver(
cfg,
[]Resolver{bootstrap}, // if init strategy is fast, use bootstrap until init finishes
)
return initGroupResolvers(ctx, r, cfg, bootstrap)
}
func newStrictResolver(
cfg config.UpstreamGroup, resolvers []Resolver,
) *StrictResolver {
r := StrictResolver{
configurable: withConfig(&cfg),
typed: withType(strictResolverType),
}
r.setResolvers(newUpstreamResolverStatuses(resolvers))
return &r
}
func (r *StrictResolver) setResolvers(resolvers []*upstreamResolverStatus) {
r.resolvers.Store(&resolvers)
}
func (r *StrictResolver) Name() string {
return r.String()
}
func (r *StrictResolver) String() string {
resolvers := *r.resolvers.Load()
upstreams := make([]string, len(resolvers))
for i, s := range resolvers {
upstreams[i] = s.resolver.String()
}
return fmt.Sprintf("%s upstreams '%s (%s)'", strictResolverType, r.cfg.Name, strings.Join(upstreams, ","))
}
// Resolve sends the query request in a strict order to the upstream resolvers
func (r *StrictResolver) Resolve(ctx context.Context, request *model.Request) (*model.Response, error) {
ctx, logger := r.log(ctx)
// start with first resolver
for _, resolver := range *r.resolvers.Load() {
logger.Debugf("using %s as resolver", resolver.resolver)
resp, err := resolver.resolve(ctx, request)
if err != nil {
// log error and try next upstream
logger.WithField("resolver", resolver.resolver).Debug("resolution failed from resolver, cause: ", err)
continue
}
logger.WithFields(logrus.Fields{
"resolver": *resolver,
"answer": util.AnswerToString(resp.Res.Answer),
}).Debug("using response from resolver")
return resp, nil
}
return nil, errors.New("resolution was not successful, no resolver returned an answer in time")
}