home: imp code; imp docs; fix bugs;

This commit is contained in:
Maksim Kazantsev 2026-06-15 15:04:44 +03:00
parent 5e7bd02bff
commit ef139f5601
5 changed files with 397 additions and 382 deletions

View file

@ -301,7 +301,7 @@ type pendingRequests struct {
// and HTTPS. When adding new properties, update the [tlsConfigSettings.clone]
// and [tlsConfigSettings.setPrivateFieldsAndCompare] methods as necessary.
type tlsConfigSettings struct {
// Status is the current status of the configuration. It is never nil.
// Status is the current status of the configuration.
Status tlsConfigStatus `yaml:"-" json:"-"`
// Enabled indicates whether encryption (DoT/DoH/HTTPS) is enabled.
@ -390,6 +390,7 @@ func (c *tlsConfigSettings) clone() (clone *tlsConfigSettings) {
// [tlsConfigSettings.DNSCryptConfigFile]
// [tlsConfigSettings.OverrideTLSCiphers]
// [tlsConfigSettings.PortDNSCrypt]
// [tlsConfigSettings.Status]
//
// The following properties are skipped as they are set by
// [tlsManager.loadTLSConfig]:
@ -399,7 +400,6 @@ func (c *tlsConfigSettings) clone() (clone *tlsConfigSettings) {
func (c *tlsConfigSettings) setPrivateFieldsAndCompare(
conf *tlsConfigSettings,
status tlsConfigStatus,
servePlain aghalg.NullBool,
) (equal bool) {
conf.OverrideTLSCiphers = slices.Clone(c.OverrideTLSCiphers)
@ -407,10 +407,6 @@ func (c *tlsConfigSettings) setPrivateFieldsAndCompare(
conf.PortDNSCrypt = c.PortDNSCrypt
conf.Status = status
if servePlain != aghalg.NBNull {
conf.ServePlainDNS = servePlain == aghalg.NBTrue
}
// TODO(a.garipov): Define a custom comparer.
return cmp.Equal(c, conf)
}

View file

@ -278,34 +278,6 @@ func (m *tlsManager) reload(ctx context.Context) {
m.web.tlsConfigChanged(context.Background(), m.extTLSConf)
}
// reconfigureDNSServer updates the DNS server configuration using extTLSConf.
// extTLSConf must not be nil.
func (web *webAPI) reconfigureDNSServer(
ctx context.Context,
extTLSConf *tlsConfigSettings,
) (err error) {
newConf, err := newServerConfig(
&config.DNS,
config.Clients.Sources,
extTLSConf,
config.HTTPConfig.DoH,
web.tlsManager,
web.httpReg,
globalContext.clients.storage,
web.tlsManager.confModifier,
)
if err != nil {
return fmt.Errorf("generating forwarding dns server config: %w", err)
}
err = globalContext.dnsServer.Reconfigure(ctx, newConf)
if err != nil {
return fmt.Errorf("starting forwarding dns server: %w", err)
}
return nil
}
// loadTLSConfig loads and validates the TLS configuration. It also sets
// [tlsConfigSettings.CertificateChainData] and
// [tlsConfigSettings.PrivateKeyData] properties. The returned error is also
@ -451,8 +423,8 @@ type tlsConfigSettingsExt struct {
ServePlainDNS aghalg.NullBool `yaml:"-" json:"serve_plain_dns"`
}
// setConfig updates manager TLS configuration with the given one. newConf must
// not be nil.
// setConfig updates manager TLS configuration with the given one. newConf and
// status must not be nil.
func (m *tlsManager) setConfig(
ctx context.Context,
newConf *tlsConfigSettings,
@ -462,7 +434,9 @@ func (m *tlsManager) setConfig(
m.mu.Lock()
defer m.mu.Unlock()
if !m.extTLSConf.setPrivateFieldsAndCompare(newConf, *status, servePlain) {
m.extTLSConf.updatePlainDNS(newConf, servePlain)
if !m.extTLSConf.setPrivateFieldsAndCompare(newConf, *status) {
m.logger.InfoContext(ctx, "config has changed, restarting https server")
restartHTTPS = true
} else {
@ -489,6 +463,27 @@ func (m *tlsManager) setConfig(
return restartHTTPS
}
// updatePlainDNS checks the old value of [tlsConfigSettings.ServePlainDNS] in
// c and if it differs from servePlain, sets the value of servePlain in
// newTLSConf.ServePlainDNS. newTLSConf must not be nil.
func (c *tlsConfigSettings) updatePlainDNS(
newTLSConf *tlsConfigSettings,
servePlain aghalg.NullBool,
) {
if servePlain != aghalg.NBNull {
func() {
config.Lock()
defer config.Unlock()
config.DNS.ServePlainDNS = servePlain == aghalg.NBTrue
}()
newTLSConf.ServePlainDNS = servePlain == aghalg.NBTrue
} else {
newTLSConf.ServePlainDNS = c.ServePlainDNS
}
}
// validateCertChain verifies certs using the first as the main one and others
// as intermediate. srvName stands for the expected DNS name. certs must not
// be empty.

View file

@ -6,26 +6,17 @@ import (
"crypto/rsa"
"crypto/tls"
"crypto/x509"
"encoding/base64"
"encoding/json"
"encoding/pem"
"fmt"
"math/big"
"net"
"net/http"
"net/http/httptest"
"net/netip"
"os"
"path/filepath"
"testing"
"time"
"github.com/AdguardTeam/AdGuardHome/internal/agh"
"github.com/AdguardTeam/AdGuardHome/internal/aghalg"
"github.com/AdguardTeam/AdGuardHome/internal/aghtls"
"github.com/AdguardTeam/AdGuardHome/internal/client"
"github.com/AdguardTeam/AdGuardHome/internal/dnsforward"
"github.com/AdguardTeam/golibs/netutil"
"github.com/AdguardTeam/golibs/testutil"
"github.com/AdguardTeam/golibs/timeutil"
"github.com/stretchr/testify/assert"
@ -305,329 +296,3 @@ func TestTLSManager_Reload(t *testing.T) {
extTLSConf = m.extendedTLSConfig()
assertCertSerialNumber(t, extTLSConf, snAfter)
}
func TestTLSManager_HandleTLSStatus(t *testing.T) {
var (
ctx = testutil.ContextWithTimeout(t, testTimeout)
err error
)
testCertChain := requireReadFile(t, testCertificatePath)
testPrivateKeyData := requireReadFile(t, testPrivateKeyPath)
m, err := newTLSManager(ctx, &tlsManagerConfig{
logger: testLogger,
confModifier: agh.EmptyConfigModifier{},
manager: aghtls.EmptyManager{},
tlsSettings: tlsConfigSettings{
Enabled: true,
CertificateChain: string(testCertChain),
PrivateKey: string(testPrivateKeyData),
},
servePlainDNS: false,
})
require.NoError(t, err)
web := newTestWeb(t, &webConfig{tlsManager: m})
m.setWebAPI(web)
w := httptest.NewRecorder()
r := httptest.NewRequest(http.MethodGet, "/control/tls/status", nil)
web.handleTLSStatus(w, r)
res := &tlsConfigSettingsExt{}
err = json.NewDecoder(w.Body).Decode(res)
require.NoError(t, err)
wantCertificateChain := base64.StdEncoding.EncodeToString(testCertChain)
assert.True(t, res.Enabled)
assert.Equal(t, wantCertificateChain, res.CertificateChain)
assert.True(t, res.PrivateKeySaved)
}
func TestValidateTLSSettings(t *testing.T) {
storeGlobals(t)
var (
ctx = testutil.ContextWithTimeout(t, testTimeout)
err error
)
m, err := newTLSManager(ctx, &tlsManagerConfig{
logger: testLogger,
confModifier: agh.EmptyConfigModifier{},
manager: aghtls.EmptyManager{},
servePlainDNS: false,
})
require.NoError(t, err)
web := newTestWeb(t, &webConfig{tlsManager: m})
m.setWebAPI(web)
tcpLn, err := net.Listen("tcp", ":0")
require.NoError(t, err)
testutil.CleanupAndRequireSuccess(t, tcpLn.Close)
tcpAddr := testutil.RequireTypeAssert[*net.TCPAddr](t, tcpLn.Addr())
busyTCPPort := tcpAddr.Port
udpLn, err := net.ListenPacket("udp", ":0")
require.NoError(t, err)
testutil.CleanupAndRequireSuccess(t, udpLn.Close)
udpAddr := testutil.RequireTypeAssert[*net.UDPAddr](t, udpLn.LocalAddr())
busyUDPPort := udpAddr.Port
testCases := []struct {
name string
wantErr string
setts tlsConfigSettingsExt
}{{
name: "basic",
wantErr: "",
setts: tlsConfigSettingsExt{},
}, {
name: "disabled_all",
wantErr: "plain DNS is required in case encryption protocols are disabled",
setts: tlsConfigSettingsExt{
ServePlainDNS: aghalg.NBFalse,
},
}, {
name: "busy_https_port",
wantErr: fmt.Sprintf("port %d for HTTPS is not available", busyTCPPort),
setts: tlsConfigSettingsExt{
tlsConfigSettings: tlsConfigSettings{
Enabled: true,
PortHTTPS: uint16(busyTCPPort),
},
},
}, {
name: "busy_dot_port",
wantErr: fmt.Sprintf("port %d for DNS-over-TLS is not available", busyTCPPort),
setts: tlsConfigSettingsExt{
tlsConfigSettings: tlsConfigSettings{
Enabled: true,
PortDNSOverTLS: uint16(busyTCPPort),
},
},
}, {
name: "busy_doq_port",
wantErr: fmt.Sprintf("port %d for DNS-over-QUIC is not available", busyUDPPort),
setts: tlsConfigSettingsExt{
tlsConfigSettings: tlsConfigSettings{
Enabled: true,
PortDNSOverQUIC: uint16(busyUDPPort),
},
},
}, {
name: "duplicate_port",
wantErr: "validating tcp ports: duplicated values: [4433]",
setts: tlsConfigSettingsExt{
tlsConfigSettings: tlsConfigSettings{
Enabled: true,
PortHTTPS: 4433,
PortDNSOverTLS: 4433,
},
},
}}
for _, tc := range testCases {
t.Run(tc.name, func(t *testing.T) {
err = web.validateTLSSettings(tc.setts)
testutil.AssertErrorMsg(t, tc.wantErr, err)
})
}
}
func TestTLSManager_HandleTLSValidate(t *testing.T) {
storeGlobals(t)
var (
ctx = testutil.ContextWithTimeout(t, testTimeout)
err error
)
m, err := newTLSManager(ctx, &tlsManagerConfig{
logger: testLogger,
confModifier: agh.EmptyConfigModifier{},
manager: aghtls.EmptyManager{},
tlsSettings: tlsConfigSettings{
Enabled: true,
CertificatePath: testCertificatePath,
PrivateKeyPath: testPrivateKeyPath,
},
servePlainDNS: false,
})
require.NoError(t, err)
web := newTestWeb(t, &webConfig{tlsManager: m})
m.setWebAPI(web)
setts := &tlsConfigSettingsExt{
tlsConfigSettings: tlsConfigSettings{
Enabled: true,
CertificatePath: testCertificatePath,
PrivateKeyPath: testPrivateKeyPath,
},
}
req, err := json.Marshal(setts)
require.NoError(t, err)
w := httptest.NewRecorder()
r := httptest.NewRequest(http.MethodPost, "/control/tls/validate", bytes.NewReader(req))
web.handleTLSValidate(w, r)
res := &tlsConfigStatus{}
err = json.NewDecoder(w.Body).Decode(res)
require.NoError(t, err)
testCertChainData := requireReadFile(t, testCertificatePath)
testPrivateKeyData := requireReadFile(t, testPrivateKeyPath)
cert, err := tls.X509KeyPair(testCertChainData, testPrivateKeyData)
require.NoError(t, err)
wantIssuer := cert.Leaf.Issuer.String()
assert.Equal(t, wantIssuer, res.Issuer)
}
func TestTLSManager_HandleTLSConfigure(t *testing.T) {
// Store the global state before making any changes.
storeGlobals(t)
var (
ctx = testutil.ContextWithTimeout(t, testTimeout)
err error
)
globalContext.dnsServer, err = dnsforward.NewServer(dnsforward.DNSCreateParams{
Logger: testLogger,
})
require.NoError(t, err)
err = globalContext.dnsServer.Prepare(
testutil.ContextWithTimeout(t, testTimeout),
&dnsforward.ServerConfig{
TLSConf: &dnsforward.TLSConfig{},
Config: dnsforward.Config{
UpstreamMode: dnsforward.UpstreamModeLoadBalance,
EDNSClientSubnet: &dnsforward.EDNSClientSubnet{Enabled: false},
ClientsContainer: dnsforward.EmptyClientsContainer{},
},
ServePlainDNS: true,
})
require.NoError(t, err)
globalContext.clients.storage, err = client.NewStorage(ctx, &client.StorageConfig{
BaseLogger: testLogger,
Logger: testLogger,
Clock: timeutil.SystemClock{},
})
require.NoError(t, err)
config.DNS.BindHosts = []netip.Addr{netutil.IPv4Localhost()}
config.DNS.Port = 0
const wantSerialNumber int64 = 1
// Prepare the TLS manager configuration.
tmpDir := t.TempDir()
certPath := filepath.Join(tmpDir, "cert.pem")
keyPath := filepath.Join(tmpDir, "key.pem")
certDER, key := newCertAndKey(t, wantSerialNumber)
writeCertAndKey(t, certDER, certPath, key, keyPath)
// Initialize the TLS manager and assert its configuration.
m, err := newTLSManager(ctx, &tlsManagerConfig{
logger: testLogger,
confModifier: agh.EmptyConfigModifier{},
manager: aghtls.EmptyManager{},
tlsSettings: tlsConfigSettings{
Enabled: true,
CertificatePath: certPath,
PrivateKeyPath: keyPath,
},
servePlainDNS: true,
})
require.NoError(t, err)
web := newTestWeb(t, &webConfig{tlsManager: m})
m.setWebAPI(web)
extTLSConf := m.extendedTLSConfig()
assertCertSerialNumber(t, extTLSConf, wantSerialNumber)
// Prepare a request with the new TLS configuration.
setts := &tlsConfigSettingsExt{
tlsConfigSettings: tlsConfigSettings{
Enabled: true,
PortHTTPS: 4433,
CertificatePath: testCertificatePath,
PrivateKeyPath: testPrivateKeyPath,
},
}
req, err := json.Marshal(setts)
require.NoError(t, err)
r := httptest.NewRequest(http.MethodPost, "/control/tls/configure", bytes.NewReader(req))
w := httptest.NewRecorder()
// Reconfigure the TLS manager.
web.handleTLSConfigure(w, r)
// The [webAPI.handleTLSConfigure] method will start the DNS server and
// it should be stopped after the test ends.
testutil.CleanupAndRequireSuccess(t, func() (err error) {
return globalContext.dnsServer.Stop(testutil.ContextWithTimeout(t, testTimeout))
})
res := &tlsConfig{
tlsConfigStatus: &tlsConfigStatus{},
}
err = json.NewDecoder(w.Body).Decode(res)
require.NoError(t, err)
testCertChainData := requireReadFile(t, testCertificatePath)
testPrivateKeyData := requireReadFile(t, testPrivateKeyPath)
cert, err := tls.X509KeyPair(testCertChainData, testPrivateKeyData)
require.NoError(t, err)
wantIssuer := cert.Leaf.Issuer.String()
assert.Equal(t, wantIssuer, res.tlsConfigStatus.Issuer)
// Assert that the Web API's TLS configuration has been updated.
//
// TODO(s.chzhen): Remove when [httpsServer.cond] is removed.
assert.Eventually(t, func() bool {
web.httpsServer.condLock.Lock()
defer web.httpsServer.condLock.Unlock()
cert = web.httpsServer.cert
if cert.Leaf == nil {
return false
}
assert.Equal(t, wantIssuer, cert.Leaf.Issuer.String())
return true
}, testTimeout, testTimeout/10)
}
// requireReadFile reads the file at the specified path and returns its content.
//
// TODO(m.kazantsev): Move to golibs/testutil.
func requireReadFile(tb testing.TB, path string) (data []byte) {
tb.Helper()
data, err := os.ReadFile(path)
require.NoError(tb, err)
return data
}

View file

@ -208,10 +208,10 @@ func newWebAPI(ctx context.Context, conf *webAPIConfig) (w *webAPI) {
mux.Handle("/install.html", w.preInstallHandler(clientFS))
w.registerInstallHandlers()
} else {
w.registerTLSHandlers()
w.registerControlHandlers()
}
w.registerTLSHandlers()
w.httpsServer.cond = sync.NewCond(&w.httpsServer.condLock)
return w
@ -733,15 +733,6 @@ func (web *webAPI) handleTLSConfigure(w http.ResponseWriter, r *http.Request) {
restartHTTPS = web.tlsManager.setConfig(ctx, newTLSConf, status, req.ServePlainDNS)
if req.ServePlainDNS != aghalg.NBNull {
func() {
config.Lock()
defer config.Unlock()
config.DNS.ServePlainDNS = req.ServePlainDNS == aghalg.NBTrue
}()
}
err = web.reconfigureDNSServer(ctx, newTLSConf)
if err != nil {
web.logger.ErrorContext(ctx, "reconfiguring dns server", slogutil.KeyError, err)
@ -771,3 +762,31 @@ func (web *webAPI) handleTLSConfigure(w http.ResponseWriter, r *http.Request) {
go web.tlsConfigChanged(context.Background(), &req.tlsConfigSettings)
}
}
// reconfigureDNSServer updates the DNS server configuration using extTLSConf.
// extTLSConf must not be nil.
func (web *webAPI) reconfigureDNSServer(
ctx context.Context,
extTLSConf *tlsConfigSettings,
) (err error) {
newConf, err := newServerConfig(
&config.DNS,
config.Clients.Sources,
extTLSConf,
config.HTTPConfig.DoH,
web.tlsManager,
web.httpReg,
globalContext.clients.storage,
web.tlsManager.confModifier,
)
if err != nil {
return fmt.Errorf("generating forwarding dns server config: %w", err)
}
err = globalContext.dnsServer.Reconfigure(ctx, newConf)
if err != nil {
return fmt.Errorf("starting forwarding dns server: %w", err)
}
return nil
}

View file

@ -3,21 +3,34 @@ package home
import (
"bufio"
"bytes"
"crypto/tls"
"encoding/base64"
"encoding/json"
"fmt"
"net"
"net/http"
"net/http/httptest"
"net/netip"
"net/url"
"os"
"path"
"path/filepath"
"strconv"
"testing"
"testing/fstest"
"github.com/AdguardTeam/AdGuardHome/internal/agh"
"github.com/AdguardTeam/AdGuardHome/internal/aghalg"
"github.com/AdguardTeam/AdGuardHome/internal/aghhttp"
"github.com/AdguardTeam/AdGuardHome/internal/aghos"
"github.com/AdguardTeam/AdGuardHome/internal/aghtls"
"github.com/AdguardTeam/AdGuardHome/internal/aghuser"
"github.com/AdguardTeam/AdGuardHome/internal/client"
"github.com/AdguardTeam/AdGuardHome/internal/dnsforward"
"github.com/AdguardTeam/golibs/netutil"
"github.com/AdguardTeam/golibs/netutil/urlutil"
"github.com/AdguardTeam/golibs/testutil"
"github.com/AdguardTeam/golibs/timeutil"
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require"
"golang.org/x/crypto/bcrypt"
@ -347,3 +360,330 @@ func readH2CResponse(tb testing.TB, framer *http2.Framer, decoder *testDecoder)
break
}
}
func TestWebAPI_HandleTLSConfigure(t *testing.T) {
// Store the global state before making any changes.
storeGlobals(t)
var (
ctx = testutil.ContextWithTimeout(t, testTimeout)
err error
)
globalContext.dnsServer, err = dnsforward.NewServer(dnsforward.DNSCreateParams{
Logger: testLogger,
})
require.NoError(t, err)
err = globalContext.dnsServer.Prepare(
testutil.ContextWithTimeout(t, testTimeout),
&dnsforward.ServerConfig{
TLSConf: &dnsforward.TLSConfig{},
Config: dnsforward.Config{
UpstreamMode: dnsforward.UpstreamModeLoadBalance,
EDNSClientSubnet: &dnsforward.EDNSClientSubnet{Enabled: false},
ClientsContainer: dnsforward.EmptyClientsContainer{},
},
ServePlainDNS: true,
})
require.NoError(t, err)
globalContext.clients.storage, err = client.NewStorage(ctx, &client.StorageConfig{
BaseLogger: testLogger,
Logger: testLogger,
Clock: timeutil.SystemClock{},
})
require.NoError(t, err)
config.DNS.BindHosts = []netip.Addr{netutil.IPv4Localhost()}
config.DNS.Port = 0
const wantSerialNumber int64 = 1
// Prepare the TLS manager configuration.
tmpDir := t.TempDir()
certPath := filepath.Join(tmpDir, "cert.pem")
keyPath := filepath.Join(tmpDir, "key.pem")
certDER, key := newCertAndKey(t, wantSerialNumber)
writeCertAndKey(t, certDER, certPath, key, keyPath)
// Initialize the TLS manager and assert its configuration.
m, err := newTLSManager(ctx, &tlsManagerConfig{
logger: testLogger,
confModifier: agh.EmptyConfigModifier{},
manager: aghtls.EmptyManager{},
tlsSettings: tlsConfigSettings{
Enabled: true,
CertificatePath: certPath,
PrivateKeyPath: keyPath,
ServePlainDNS: true,
},
servePlainDNS: true,
})
require.NoError(t, err)
web := newTestWeb(t, &webConfig{tlsManager: m})
m.setWebAPI(web)
extTLSConf := m.extendedTLSConfig()
assertCertSerialNumber(t, extTLSConf, wantSerialNumber)
// Prepare a request with the new TLS configuration.
setts := &tlsConfigSettingsExt{
tlsConfigSettings: tlsConfigSettings{
Enabled: true,
PortHTTPS: 4433,
CertificatePath: testCertificatePath,
PrivateKeyPath: testPrivateKeyPath,
},
}
req, err := json.Marshal(setts)
require.NoError(t, err)
r := httptest.NewRequest(http.MethodPost, "/control/tls/configure", bytes.NewReader(req))
w := httptest.NewRecorder()
// Reconfigure the TLS manager.
web.handleTLSConfigure(w, r)
// The [webAPI.handleTLSConfigure] method will start the DNS server and
// it should be stopped after the test ends.
testutil.CleanupAndRequireSuccess(t, func() (err error) {
return globalContext.dnsServer.Stop(testutil.ContextWithTimeout(t, testTimeout))
})
res := &tlsConfig{
tlsConfigStatus: &tlsConfigStatus{},
}
err = json.NewDecoder(w.Body).Decode(res)
require.NoError(t, err)
testCertChainData := requireReadFile(t, testCertificatePath)
testPrivateKeyData := requireReadFile(t, testPrivateKeyPath)
cert, err := tls.X509KeyPair(testCertChainData, testPrivateKeyData)
require.NoError(t, err)
wantIssuer := cert.Leaf.Issuer.String()
assert.Equal(t, wantIssuer, res.tlsConfigStatus.Issuer)
// Assert that the Web API's TLS configuration has been updated.
//
// TODO(s.chzhen): Remove when [httpsServer.cond] is removed.
assert.Eventually(t, func() bool {
web.httpsServer.condLock.Lock()
defer web.httpsServer.condLock.Unlock()
cert = web.httpsServer.cert
if cert.Leaf == nil {
return false
}
assert.Equal(t, wantIssuer, cert.Leaf.Issuer.String())
return true
}, testTimeout, testTimeout/10)
}
func TestWebAPI_HandleTLSStatus(t *testing.T) {
var (
ctx = testutil.ContextWithTimeout(t, testTimeout)
err error
)
testCertChain := requireReadFile(t, testCertificatePath)
testPrivateKeyData := requireReadFile(t, testPrivateKeyPath)
m, err := newTLSManager(ctx, &tlsManagerConfig{
logger: testLogger,
confModifier: agh.EmptyConfigModifier{},
manager: aghtls.EmptyManager{},
tlsSettings: tlsConfigSettings{
Enabled: true,
CertificateChain: string(testCertChain),
PrivateKey: string(testPrivateKeyData),
},
servePlainDNS: false,
})
require.NoError(t, err)
web := newTestWeb(t, &webConfig{tlsManager: m})
m.setWebAPI(web)
w := httptest.NewRecorder()
r := httptest.NewRequest(http.MethodGet, "/control/tls/status", nil)
web.handleTLSStatus(w, r)
res := &tlsConfigSettingsExt{}
err = json.NewDecoder(w.Body).Decode(res)
require.NoError(t, err)
wantCertificateChain := base64.StdEncoding.EncodeToString(testCertChain)
assert.True(t, res.Enabled)
assert.Equal(t, wantCertificateChain, res.CertificateChain)
assert.True(t, res.PrivateKeySaved)
}
func TestWebAPI_ValidateTLSSettings(t *testing.T) {
storeGlobals(t)
var (
ctx = testutil.ContextWithTimeout(t, testTimeout)
err error
)
m, err := newTLSManager(ctx, &tlsManagerConfig{
logger: testLogger,
confModifier: agh.EmptyConfigModifier{},
manager: aghtls.EmptyManager{},
servePlainDNS: false,
})
require.NoError(t, err)
web := newTestWeb(t, &webConfig{tlsManager: m})
m.setWebAPI(web)
tcpLn, err := net.Listen("tcp", ":0")
require.NoError(t, err)
testutil.CleanupAndRequireSuccess(t, tcpLn.Close)
tcpAddr := testutil.RequireTypeAssert[*net.TCPAddr](t, tcpLn.Addr())
busyTCPPort := tcpAddr.Port
udpLn, err := net.ListenPacket("udp", ":0")
require.NoError(t, err)
testutil.CleanupAndRequireSuccess(t, udpLn.Close)
udpAddr := testutil.RequireTypeAssert[*net.UDPAddr](t, udpLn.LocalAddr())
busyUDPPort := udpAddr.Port
testCases := []struct {
name string
wantErr string
setts tlsConfigSettingsExt
}{{
name: "basic",
wantErr: "",
setts: tlsConfigSettingsExt{},
}, {
name: "disabled_all",
wantErr: "plain DNS is required in case encryption protocols are disabled",
setts: tlsConfigSettingsExt{
ServePlainDNS: aghalg.NBFalse,
},
}, {
name: "busy_https_port",
wantErr: fmt.Sprintf("port %d for HTTPS is not available", busyTCPPort),
setts: tlsConfigSettingsExt{
tlsConfigSettings: tlsConfigSettings{
Enabled: true,
PortHTTPS: uint16(busyTCPPort),
},
},
}, {
name: "busy_dot_port",
wantErr: fmt.Sprintf("port %d for DNS-over-TLS is not available", busyTCPPort),
setts: tlsConfigSettingsExt{
tlsConfigSettings: tlsConfigSettings{
Enabled: true,
PortDNSOverTLS: uint16(busyTCPPort),
},
},
}, {
name: "busy_doq_port",
wantErr: fmt.Sprintf("port %d for DNS-over-QUIC is not available", busyUDPPort),
setts: tlsConfigSettingsExt{
tlsConfigSettings: tlsConfigSettings{
Enabled: true,
PortDNSOverQUIC: uint16(busyUDPPort),
},
},
}, {
name: "duplicate_port",
wantErr: "validating tcp ports: duplicated values: [4433]",
setts: tlsConfigSettingsExt{
tlsConfigSettings: tlsConfigSettings{
Enabled: true,
PortHTTPS: 4433,
PortDNSOverTLS: 4433,
},
},
}}
for _, tc := range testCases {
t.Run(tc.name, func(t *testing.T) {
err = web.validateTLSSettings(tc.setts)
testutil.AssertErrorMsg(t, tc.wantErr, err)
})
}
}
func TestWebAPI_HandleTLSValidate(t *testing.T) {
storeGlobals(t)
var (
ctx = testutil.ContextWithTimeout(t, testTimeout)
err error
)
m, err := newTLSManager(ctx, &tlsManagerConfig{
logger: testLogger,
confModifier: agh.EmptyConfigModifier{},
manager: aghtls.EmptyManager{},
tlsSettings: tlsConfigSettings{
Enabled: true,
CertificatePath: testCertificatePath,
PrivateKeyPath: testPrivateKeyPath,
},
servePlainDNS: false,
})
require.NoError(t, err)
web := newTestWeb(t, &webConfig{tlsManager: m})
m.setWebAPI(web)
setts := &tlsConfigSettingsExt{
tlsConfigSettings: tlsConfigSettings{
Enabled: true,
CertificatePath: testCertificatePath,
PrivateKeyPath: testPrivateKeyPath,
},
}
req, err := json.Marshal(setts)
require.NoError(t, err)
w := httptest.NewRecorder()
r := httptest.NewRequest(http.MethodPost, "/control/tls/validate", bytes.NewReader(req))
web.handleTLSValidate(w, r)
res := &tlsConfigStatus{}
err = json.NewDecoder(w.Body).Decode(res)
require.NoError(t, err)
testCertChainData := requireReadFile(t, testCertificatePath)
testPrivateKeyData := requireReadFile(t, testPrivateKeyPath)
cert, err := tls.X509KeyPair(testCertChainData, testPrivateKeyData)
require.NoError(t, err)
wantIssuer := cert.Leaf.Issuer.String()
assert.Equal(t, wantIssuer, res.Issuer)
}
// requireReadFile reads the file at the specified path and returns its content.
//
// TODO(m.kazantsev): Move to golibs/testutil.
func requireReadFile(tb testing.TB, path string) (data []byte) {
tb.Helper()
data, err := os.ReadFile(path)
require.NoError(tb, err)
return data
}