mirror of https://github.com/0xERR0R/blocky.git
101 lines
2.5 KiB
Go
101 lines
2.5 KiB
Go
package resolver
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"strings"
|
|
"sync/atomic"
|
|
|
|
"github.com/0xERR0R/blocky/config"
|
|
"github.com/0xERR0R/blocky/log"
|
|
"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 start 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] = fmt.Sprintf("%s", s.resolver)
|
|
}
|
|
|
|
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) {
|
|
logger := log.WithPrefix(request.Log, strictResolverType)
|
|
|
|
// 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")
|
|
}
|