mirror of
https://github.com/juanfont/headscale.git
synced 2025-05-23 01:15:27 +02:00
make derp test lest special
Signed-off-by: Kristoffer Dalby <kristoffer@tailscale.com>
This commit is contained in:
parent
3e30f82acd
commit
52d005b248
@ -1,18 +1,12 @@
|
|||||||
package integration
|
package integration
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
|
||||||
"log"
|
|
||||||
"net/url"
|
|
||||||
"strings"
|
"strings"
|
||||||
"testing"
|
"testing"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/juanfont/headscale/hscontrol/util"
|
|
||||||
"github.com/juanfont/headscale/integration/dockertestutil"
|
|
||||||
"github.com/juanfont/headscale/integration/hsic"
|
"github.com/juanfont/headscale/integration/hsic"
|
||||||
"github.com/juanfont/headscale/integration/tsic"
|
"github.com/juanfont/headscale/integration/tsic"
|
||||||
"github.com/ory/dockertest/v3"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
type ClientsSpec struct {
|
type ClientsSpec struct {
|
||||||
@ -20,21 +14,18 @@ type ClientsSpec struct {
|
|||||||
WebsocketDERP int
|
WebsocketDERP int
|
||||||
}
|
}
|
||||||
|
|
||||||
type EmbeddedDERPServerScenario struct {
|
|
||||||
*Scenario
|
|
||||||
|
|
||||||
tsicNetworks map[string]*dockertest.Network
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestDERPServerScenario(t *testing.T) {
|
func TestDERPServerScenario(t *testing.T) {
|
||||||
spec := map[string]ClientsSpec{
|
spec := ScenarioSpec{
|
||||||
"user1": {
|
NodesPerUser: 1,
|
||||||
Plain: len(MustTestVersions),
|
Users: []string{"user1", "user2", "user3"},
|
||||||
WebsocketDERP: 0,
|
Networks: map[string][]string{
|
||||||
|
"usernet1": {"user1"},
|
||||||
|
"usernet2": {"user2"},
|
||||||
|
"usernet3": {"user3"},
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
derpServerScenario(t, spec, func(scenario *EmbeddedDERPServerScenario) {
|
derpServerScenario(t, spec, false, func(scenario *Scenario) {
|
||||||
allClients, err := scenario.ListTailscaleClients()
|
allClients, err := scenario.ListTailscaleClients()
|
||||||
assertNoErrListClients(t, err)
|
assertNoErrListClients(t, err)
|
||||||
t.Logf("checking %d clients for websocket connections", len(allClients))
|
t.Logf("checking %d clients for websocket connections", len(allClients))
|
||||||
@ -52,14 +43,17 @@ func TestDERPServerScenario(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestDERPServerWebsocketScenario(t *testing.T) {
|
func TestDERPServerWebsocketScenario(t *testing.T) {
|
||||||
spec := map[string]ClientsSpec{
|
spec := ScenarioSpec{
|
||||||
"user1": {
|
NodesPerUser: 1,
|
||||||
Plain: 0,
|
Users: []string{"user1", "user2", "user3"},
|
||||||
WebsocketDERP: 2,
|
Networks: map[string][]string{
|
||||||
|
"usernet1": []string{"user1"},
|
||||||
|
"usernet2": []string{"user2"},
|
||||||
|
"usernet3": []string{"user3"},
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
derpServerScenario(t, spec, func(scenario *EmbeddedDERPServerScenario) {
|
derpServerScenario(t, spec, true, func(scenario *Scenario) {
|
||||||
allClients, err := scenario.ListTailscaleClients()
|
allClients, err := scenario.ListTailscaleClients()
|
||||||
assertNoErrListClients(t, err)
|
assertNoErrListClients(t, err)
|
||||||
t.Logf("checking %d clients for websocket connections", len(allClients))
|
t.Logf("checking %d clients for websocket connections", len(allClients))
|
||||||
@ -83,23 +77,22 @@ func TestDERPServerWebsocketScenario(t *testing.T) {
|
|||||||
//nolint:thelper
|
//nolint:thelper
|
||||||
func derpServerScenario(
|
func derpServerScenario(
|
||||||
t *testing.T,
|
t *testing.T,
|
||||||
spec map[string]ClientsSpec,
|
spec ScenarioSpec,
|
||||||
furtherAssertions ...func(*EmbeddedDERPServerScenario),
|
websocket bool,
|
||||||
|
furtherAssertions ...func(*Scenario),
|
||||||
) {
|
) {
|
||||||
IntegrationSkip(t)
|
IntegrationSkip(t)
|
||||||
// t.Parallel()
|
// t.Parallel()
|
||||||
|
|
||||||
baseScenario, err := NewScenario(dockertestMaxWait())
|
scenario, err := NewScenario(spec)
|
||||||
assertNoErr(t, err)
|
assertNoErr(t, err)
|
||||||
|
|
||||||
scenario := EmbeddedDERPServerScenario{
|
|
||||||
Scenario: baseScenario,
|
|
||||||
tsicNetworks: map[string]*dockertest.Network{},
|
|
||||||
}
|
|
||||||
defer scenario.ShutdownAssertNoPanics(t)
|
defer scenario.ShutdownAssertNoPanics(t)
|
||||||
|
|
||||||
err = scenario.CreateHeadscaleEnv(
|
err = scenario.CreateHeadscaleEnv(
|
||||||
spec,
|
[]tsic.Option{
|
||||||
|
tsic.WithWebsocketDERP(websocket),
|
||||||
|
},
|
||||||
hsic.WithTestName("derpserver"),
|
hsic.WithTestName("derpserver"),
|
||||||
hsic.WithExtraPorts([]string{"3478/udp"}),
|
hsic.WithExtraPorts([]string{"3478/udp"}),
|
||||||
hsic.WithEmbeddedDERPServerOnly(),
|
hsic.WithEmbeddedDERPServerOnly(),
|
||||||
@ -185,180 +178,6 @@ func derpServerScenario(
|
|||||||
t.Logf("Run2: %d successful pings out of %d", success, len(allClients)*len(allHostnames))
|
t.Logf("Run2: %d successful pings out of %d", success, len(allClients)*len(allHostnames))
|
||||||
|
|
||||||
for _, check := range furtherAssertions {
|
for _, check := range furtherAssertions {
|
||||||
check(&scenario)
|
check(scenario)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *EmbeddedDERPServerScenario) CreateHeadscaleEnv(
|
|
||||||
users map[string]ClientsSpec,
|
|
||||||
opts ...hsic.Option,
|
|
||||||
) error {
|
|
||||||
hsServer, err := s.Headscale(opts...)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
headscaleEndpoint := hsServer.GetEndpoint()
|
|
||||||
headscaleURL, err := url.Parse(headscaleEndpoint)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
headscaleURL.Host = fmt.Sprintf("%s:%s", hsServer.GetHostname(), headscaleURL.Port())
|
|
||||||
|
|
||||||
err = hsServer.WaitForRunning()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
hash, err := util.GenerateRandomStringDNSSafe(scenarioHashLength)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
for userName, clientCount := range users {
|
|
||||||
err = s.CreateUser(userName)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
if clientCount.Plain > 0 {
|
|
||||||
// Containers that use default DERP config
|
|
||||||
err = s.CreateTailscaleIsolatedNodesInUser(
|
|
||||||
hash,
|
|
||||||
userName,
|
|
||||||
"all",
|
|
||||||
clientCount.Plain,
|
|
||||||
)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if clientCount.WebsocketDERP > 0 {
|
|
||||||
// Containers that use DERP-over-WebSocket
|
|
||||||
// Note that these clients *must* be built
|
|
||||||
// from source, which is currently
|
|
||||||
// only done for HEAD.
|
|
||||||
err = s.CreateTailscaleIsolatedNodesInUser(
|
|
||||||
hash,
|
|
||||||
userName,
|
|
||||||
tsic.VersionHead,
|
|
||||||
clientCount.WebsocketDERP,
|
|
||||||
tsic.WithWebsocketDERP(true),
|
|
||||||
)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
key, err := s.CreatePreAuthKey(userName, true, false)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
err = s.RunTailscaleUp(userName, headscaleURL.String(), key.GetKey())
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *EmbeddedDERPServerScenario) CreateTailscaleIsolatedNodesInUser(
|
|
||||||
hash string,
|
|
||||||
userStr string,
|
|
||||||
requestedVersion string,
|
|
||||||
count int,
|
|
||||||
opts ...tsic.Option,
|
|
||||||
) error {
|
|
||||||
hsServer, err := s.Headscale()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
if user, ok := s.users[userStr]; ok {
|
|
||||||
for clientN := 0; clientN < count; clientN++ {
|
|
||||||
networkName := fmt.Sprintf("tsnet-%s-%s-%d",
|
|
||||||
hash,
|
|
||||||
userStr,
|
|
||||||
clientN,
|
|
||||||
)
|
|
||||||
network, err := dockertestutil.GetFirstOrCreateNetwork(
|
|
||||||
s.pool,
|
|
||||||
networkName,
|
|
||||||
)
|
|
||||||
if err != nil {
|
|
||||||
return fmt.Errorf("failed to create or get %s network: %w", networkName, err)
|
|
||||||
}
|
|
||||||
|
|
||||||
s.tsicNetworks[networkName] = network
|
|
||||||
|
|
||||||
err = hsServer.ConnectToNetwork(network)
|
|
||||||
if err != nil {
|
|
||||||
return fmt.Errorf("failed to connect headscale to %s network: %w", networkName, err)
|
|
||||||
}
|
|
||||||
|
|
||||||
version := requestedVersion
|
|
||||||
if requestedVersion == "all" {
|
|
||||||
version = MustTestVersions[clientN%len(MustTestVersions)]
|
|
||||||
}
|
|
||||||
|
|
||||||
cert := hsServer.GetCert()
|
|
||||||
|
|
||||||
opts = append(opts,
|
|
||||||
tsic.WithCACert(cert),
|
|
||||||
)
|
|
||||||
|
|
||||||
user.createWaitGroup.Go(func() error {
|
|
||||||
tsClient, err := tsic.New(
|
|
||||||
s.pool,
|
|
||||||
version,
|
|
||||||
opts...,
|
|
||||||
)
|
|
||||||
if err != nil {
|
|
||||||
return fmt.Errorf(
|
|
||||||
"failed to create tailscale (%s) node: %w",
|
|
||||||
tsClient.Hostname(),
|
|
||||||
err,
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
err = tsClient.WaitForNeedsLogin()
|
|
||||||
if err != nil {
|
|
||||||
return fmt.Errorf(
|
|
||||||
"failed to wait for tailscaled (%s) to need login: %w",
|
|
||||||
tsClient.Hostname(),
|
|
||||||
err,
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
s.mu.Lock()
|
|
||||||
user.Clients[tsClient.Hostname()] = tsClient
|
|
||||||
s.mu.Unlock()
|
|
||||||
|
|
||||||
return nil
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
if err := user.createWaitGroup.Wait(); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
return fmt.Errorf("failed to add tailscale nodes: %w", errNoUserAvailable)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *EmbeddedDERPServerScenario) Shutdown() {
|
|
||||||
for _, network := range s.tsicNetworks {
|
|
||||||
err := s.pool.RemoveNetwork(network)
|
|
||||||
if err != nil {
|
|
||||||
log.Printf("failed to remove DERP network %s", network.Network.Name)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
s.Scenario.Shutdown()
|
|
||||||
}
|
|
||||||
|
Loading…
Reference in New Issue
Block a user