safing-portmaster/resolver/main.go
2022-03-16 11:47:44 +01:00

191 lines
4.8 KiB
Go

package resolver
import (
"context"
"fmt"
"net"
"strings"
"sync"
"time"
"github.com/tevino/abool"
"github.com/safing/portbase/log"
"github.com/safing/portbase/modules"
"github.com/safing/portbase/notifications"
"github.com/safing/portbase/utils/debug"
_ "github.com/safing/portmaster/core/base"
"github.com/safing/portmaster/intel"
)
var module *modules.Module
func init() {
module = modules.Register("resolver", prep, start, nil, "base", "netenv")
}
func prep() error {
intel.SetReverseResolver(ResolveIPAndValidate)
if err := registerAPI(); err != nil {
return err
}
if err := prepEnvResolver(); err != nil {
return err
}
return prepConfig()
}
func start() error {
// load resolvers from config and environment
loadResolvers()
// reload after network change
err := module.RegisterEventHook(
"netenv",
"network changed",
"update nameservers",
func(_ context.Context, _ interface{}) error {
loadResolvers()
log.Debug("resolver: reloaded nameservers due to network change")
return nil
},
)
if err != nil {
return err
}
// reload after config change
prevNameservers := strings.Join(configuredNameServers(), " ")
err = module.RegisterEventHook(
"config",
"config change",
"update nameservers",
func(_ context.Context, _ interface{}) error {
newNameservers := strings.Join(configuredNameServers(), " ")
if newNameservers != prevNameservers {
prevNameservers = newNameservers
loadResolvers()
log.Debug("resolver: reloaded nameservers due to config change")
}
return nil
},
)
if err != nil {
return err
}
module.StartServiceWorker(
"mdns handler",
5*time.Second,
listenToMDNS,
)
module.StartServiceWorker("name record delayed cache writer", 0, recordDatabase.DelayedCacheWriter)
module.StartServiceWorker("ip info delayed cache writer", 0, ipInfoDatabase.DelayedCacheWriter)
return nil
}
var localAddrFactory func(network string) net.Addr
// SetLocalAddrFactory supplies the intel package with a function to get permitted local addresses for connections.
func SetLocalAddrFactory(laf func(network string) net.Addr) {
if localAddrFactory == nil {
localAddrFactory = laf
}
}
func getLocalAddr(network string) net.Addr {
if localAddrFactory != nil {
return localAddrFactory(network)
}
return nil
}
var (
failingResolverNotification *notifications.Notification
failingResolverNotificationSet = abool.New()
failingResolverNotificationLock sync.Mutex
)
func notifyAboutFailingResolvers(err error) {
failingResolverNotificationLock.Lock()
defer failingResolverNotificationLock.Unlock()
failingResolverNotificationSet.Set()
// Check if already set.
if failingResolverNotification != nil {
return
}
// Create new notification.
n := &notifications.Notification{
EventID: "resolver:all-configured-resolvers-failed",
Type: notifications.Error,
Title: "Detected DNS Compatibility Issue",
Message: "Portmaster detected that something is interfering with its Secure DNS resolver. This could be a firewall or another secure DNS resolver software. Please check if you are running incompatible [software](https://docs.safing.io/portmaster/install/status/software-compatibility). Otherwise, please report the issue via [GitHub](https://github.com/safing/portmaster/issues) or send a mail to [support@safing.io](mailto:support@safing.io) so we can help you out.",
ShowOnSystem: true,
}
notifications.Notify(n)
failingResolverNotification = n
n.AttachToModule(module)
// Report the raw error as module error.
module.NewErrorMessage("resolving", err).Report()
}
func resetFailingResolversNotification() {
if failingResolverNotificationSet.IsNotSet() {
return
}
failingResolverNotificationLock.Lock()
defer failingResolverNotificationLock.Unlock()
if failingResolverNotification != nil {
failingResolverNotification.Delete()
failingResolverNotification = nil
}
}
// AddToDebugInfo adds the system status to the given debug.Info.
func AddToDebugInfo(di *debug.Info) {
resolversLock.Lock()
defer resolversLock.Unlock()
content := make([]string, 0, (len(globalResolvers)*4)-1)
var working, total int
for i, resolver := range globalResolvers {
// Count for summary.
total++
failing := resolver.Conn.IsFailing()
if !failing {
working++
}
// Add section.
if resolver.Info.Name != "" {
content = append(content, resolver.Info.Name)
} else {
content = append(content, resolver.Info.IP.String())
}
content = append(content, fmt.Sprintf(" %s", resolver.Info.ID()))
content = append(content, fmt.Sprintf(" Failing: %v", resolver.Conn.IsFailing()))
// Add a empty line for all but the last entry.
if i+1 < len(globalResolvers) {
content = append(content, "")
}
}
di.AddSection(
fmt.Sprintf("Resolvers: %d/%d", working, total),
debug.UseCodeSection|debug.AddContentLineBreaks,
content...,
)
}