Add compatibility assistant module
This commit is contained in:
193
compat/selfcheck.go
Normal file
193
compat/selfcheck.go
Normal file
@@ -0,0 +1,193 @@
|
||||
package compat
|
||||
|
||||
import (
|
||||
"context"
|
||||
"encoding/hex"
|
||||
"fmt"
|
||||
"net"
|
||||
"strings"
|
||||
"sync"
|
||||
"time"
|
||||
|
||||
"github.com/safing/portbase/log"
|
||||
"github.com/safing/portbase/rng"
|
||||
"github.com/safing/portmaster/network/packet"
|
||||
)
|
||||
|
||||
var (
|
||||
selfcheckLock sync.Mutex
|
||||
|
||||
SystemIntegrationCheckDstIP = net.IPv4(127, 65, 67, 75)
|
||||
SystemIntegrationCheckProtocol = packet.AnyHostInternalProtocol61
|
||||
|
||||
systemIntegrationCheckDialNet = fmt.Sprintf("ip4:%d", uint8(SystemIntegrationCheckProtocol))
|
||||
systemIntegrationCheckDialIP = SystemIntegrationCheckDstIP.String()
|
||||
systemIntegrationCheckPackets = make(chan packet.Packet, 1)
|
||||
systemIntegrationCheckWaitDuration = 3 * time.Second
|
||||
|
||||
DNSCheckInternalDomainScope string
|
||||
dnsCheckReceivedDomain = make(chan string, 1)
|
||||
dnsCheckWaitDuration = 3 * time.Second
|
||||
dnsCheckAnswerLock sync.Mutex
|
||||
dnsCheckAnswer net.IP
|
||||
|
||||
DNSTestDomain = "one.one.one.one."
|
||||
DNSTestExpectedIP = net.IPv4(1, 1, 1, 1)
|
||||
)
|
||||
|
||||
func selfcheck(ctx context.Context) (issue *systemIssue, err error) {
|
||||
selfcheckLock.Lock()
|
||||
defer selfcheckLock.Unlock()
|
||||
|
||||
// Step 1: Check if the system integration sees a packet.
|
||||
|
||||
// Empty recv channel.
|
||||
select {
|
||||
case <-systemIntegrationCheckPackets:
|
||||
case <-ctx.Done():
|
||||
return nil, context.Canceled
|
||||
default:
|
||||
}
|
||||
|
||||
// Send packet.
|
||||
conn, err := net.DialTimeout(
|
||||
systemIntegrationCheckDialNet,
|
||||
systemIntegrationCheckDialIP,
|
||||
time.Second,
|
||||
)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("failed to create system integration conn: %w", err)
|
||||
}
|
||||
_, err = conn.Write([]byte("SELF-CHECK"))
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("failed to send system integration packet: %w", err)
|
||||
}
|
||||
|
||||
// Wait for packet.
|
||||
select {
|
||||
case <-systemIntegrationCheckPackets:
|
||||
// Check passed!
|
||||
log.Tracef("compat: self-check #1: system integration check passed")
|
||||
case <-time.After(systemIntegrationCheckWaitDuration):
|
||||
return systemIntegrationIssue, fmt.Errorf("self-check #1: system integration check failed: did not receive test packet after %s", systemIntegrationCheckWaitDuration)
|
||||
case <-ctx.Done():
|
||||
return nil, context.Canceled
|
||||
}
|
||||
|
||||
// Step 2: Check if a DNS request arrives at the nameserver
|
||||
// This step necessary also includes some setup for step 3.
|
||||
|
||||
// Generate random subdomain.
|
||||
randomSubdomainBytes, err := rng.Bytes(16)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("self-check #2: failed to get random bytes for subdomain check: %w", err)
|
||||
}
|
||||
randomSubdomain := "a" + strings.ToLower(hex.EncodeToString(randomSubdomainBytes)) + "b"
|
||||
|
||||
// Generate random answer.
|
||||
var B, C, D uint64
|
||||
B, err = rng.Number(255)
|
||||
if err == nil {
|
||||
C, err = rng.Number(255)
|
||||
}
|
||||
if err == nil {
|
||||
D, err = rng.Number(255)
|
||||
}
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("self-check #2: failed to get random number for subdomain check response: %w", err)
|
||||
}
|
||||
randomAnswer := net.IPv4(127, byte(B), byte(C), byte(D))
|
||||
func() {
|
||||
dnsCheckAnswerLock.Lock()
|
||||
defer dnsCheckAnswerLock.Unlock()
|
||||
dnsCheckAnswer = randomAnswer
|
||||
}()
|
||||
|
||||
// Setup variables for lookup worker.
|
||||
var (
|
||||
dnsCheckReturnedIP net.IP
|
||||
dnsCheckLookupError = make(chan error)
|
||||
)
|
||||
|
||||
// Empty recv channel.
|
||||
select {
|
||||
case <-dnsCheckReceivedDomain:
|
||||
case <-ctx.Done():
|
||||
return nil, context.Canceled
|
||||
default:
|
||||
}
|
||||
|
||||
// Start worker for the DNS lookup.
|
||||
module.StartWorker("dns check lookup", func(_ context.Context) error {
|
||||
ips, err := net.LookupIP(randomSubdomain + DNSCheckInternalDomainScope)
|
||||
if err == nil && len(ips) > 0 {
|
||||
dnsCheckReturnedIP = ips[0]
|
||||
}
|
||||
select {
|
||||
case dnsCheckLookupError <- err:
|
||||
case <-time.After(dnsCheckWaitDuration * 2):
|
||||
case <-ctx.Done():
|
||||
}
|
||||
|
||||
return nil
|
||||
})
|
||||
|
||||
// Wait for the resolver to receive the query.
|
||||
select {
|
||||
case receivedTestDomain := <-dnsCheckReceivedDomain:
|
||||
if receivedTestDomain != randomSubdomain {
|
||||
return systemCompatibilityIssue, fmt.Errorf("self-check #2: dns integration check failed: received unmatching subdomain %q", receivedTestDomain)
|
||||
}
|
||||
case <-time.After(dnsCheckWaitDuration):
|
||||
return systemCompatibilityIssue, fmt.Errorf("self-check #2: dns integration check failed: did not receive test query after %s", dnsCheckWaitDuration)
|
||||
}
|
||||
log.Tracef("compat: self-check #2: dns integration query check passed")
|
||||
|
||||
// Step 3: Have the nameserver respond with random data in the answer section.
|
||||
|
||||
// Wait for the reply from the resolver.
|
||||
select {
|
||||
case err := <-dnsCheckLookupError:
|
||||
if err != nil {
|
||||
return systemCompatibilityIssue, fmt.Errorf("self-check #3: dns integration check failed: failed to receive test response: %w", err)
|
||||
}
|
||||
case <-time.After(dnsCheckWaitDuration):
|
||||
return systemCompatibilityIssue, fmt.Errorf("self-check #3: dns integration check failed: did not receive test response after %s", dnsCheckWaitDuration)
|
||||
case <-ctx.Done():
|
||||
return nil, context.Canceled
|
||||
}
|
||||
|
||||
// Check response.
|
||||
if !dnsCheckReturnedIP.Equal(randomAnswer) {
|
||||
return systemCompatibilityIssue, fmt.Errorf("self-check #3: dns integration check failed: received unmatching response %q", dnsCheckReturnedIP)
|
||||
}
|
||||
log.Tracef("compat: self-check #3: dns integration response check passed")
|
||||
|
||||
return nil, nil
|
||||
}
|
||||
|
||||
/*
|
||||
|
||||
* Check if the system integration sees a packet:
|
||||
* Send raw IP packet with random content and protocol, report finding to compat module.
|
||||
* use `Dial("ip4:61", "127.65.67.75")`.
|
||||
* Firewall reports back the data seen on `ip4:61` to IP `127.65.67.75`.
|
||||
* If this fails, the system integration is broken. -> Integration Issue
|
||||
* Check if a DNS request arrives at the nameserver:
|
||||
* Send A question for `[random-subdomain].self-check.portmaster.home.arpa.`.
|
||||
* Nameserver reports back the data seen.
|
||||
* If this fails, redirection to the nameserver fails.
|
||||
* This means there is another software interfering with DNS. -> Compatibility Issue
|
||||
* Have the nameserver respond with random data in the answer section.
|
||||
* Compat provides nameserver with random response data.
|
||||
* Compat module checks if the received data matches.
|
||||
* If this fails, redirection to the nameserver fails.
|
||||
* This means there is another software interfering with DNS on the return path. -> Compatibility Issue
|
||||
* DROPPED: If resolvers are reported failing, but we are online:
|
||||
* Send out plain DNS requests to one.one.one.one. and dns.quad9.net via the Go standard lookup and check if the responses are correct.
|
||||
* If not, something is blocking the Portmaster -> Secure DNS Issue
|
||||
* Discuss if this is necessary:
|
||||
* Does this improve from only having a failed TCP connection to the resolver?
|
||||
* Could another program block port 853, but fully leave requests for one.one.one.one. to port 53 alone?
|
||||
|
||||
*/
|
||||
Reference in New Issue
Block a user