mirror of
				https://github.com/juanfont/headscale.git
				synced 2025-10-28 10:51:44 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			156 lines
		
	
	
		
			3.9 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			156 lines
		
	
	
		
			3.9 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package dockertestutil
 | |
| 
 | |
| import (
 | |
| 	"errors"
 | |
| 	"fmt"
 | |
| 	"log"
 | |
| 	"net"
 | |
| 
 | |
| 	"github.com/juanfont/headscale/hscontrol/util"
 | |
| 	"github.com/ory/dockertest/v3"
 | |
| 	"github.com/ory/dockertest/v3/docker"
 | |
| )
 | |
| 
 | |
| var ErrContainerNotFound = errors.New("container not found")
 | |
| 
 | |
| func GetFirstOrCreateNetwork(pool *dockertest.Pool, name string) (*dockertest.Network, error) {
 | |
| 	networks, err := pool.NetworksByName(name)
 | |
| 	if err != nil {
 | |
| 		return nil, fmt.Errorf("looking up network names: %w", err)
 | |
| 	}
 | |
| 	if len(networks) == 0 {
 | |
| 		if _, err := pool.CreateNetwork(name); err == nil {
 | |
| 			// Create does not give us an updated version of the resource, so we need to
 | |
| 			// get it again.
 | |
| 			networks, err := pool.NetworksByName(name)
 | |
| 			if err != nil {
 | |
| 				return nil, err
 | |
| 			}
 | |
| 
 | |
| 			return &networks[0], nil
 | |
| 		} else {
 | |
| 			return nil, fmt.Errorf("creating network: %w", err)
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return &networks[0], nil
 | |
| }
 | |
| 
 | |
| func AddContainerToNetwork(
 | |
| 	pool *dockertest.Pool,
 | |
| 	network *dockertest.Network,
 | |
| 	testContainer string,
 | |
| ) error {
 | |
| 	containers, err := pool.Client.ListContainers(docker.ListContainersOptions{
 | |
| 		All: true,
 | |
| 		Filters: map[string][]string{
 | |
| 			"name": {testContainer},
 | |
| 		},
 | |
| 	})
 | |
| 	if err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 
 | |
| 	err = pool.Client.ConnectNetwork(network.Network.ID, docker.NetworkConnectionOptions{
 | |
| 		Container: containers[0].ID,
 | |
| 	})
 | |
| 	if err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 
 | |
| 	// TODO(kradalby): This doesn't work reliably, but calling the exact same functions
 | |
| 	// seem to work fine...
 | |
| 	// if container, ok := pool.ContainerByName("/" + testContainer); ok {
 | |
| 	// 	err := container.ConnectToNetwork(network)
 | |
| 	// 	if err != nil {
 | |
| 	// 		return err
 | |
| 	// 	}
 | |
| 	// }
 | |
| 
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| // RandomFreeHostPort asks the kernel for a free open port that is ready to use.
 | |
| // (from https://github.com/phayes/freeport)
 | |
| func RandomFreeHostPort() (int, error) {
 | |
| 	addr, err := net.ResolveTCPAddr("tcp", "localhost:0")
 | |
| 	if err != nil {
 | |
| 		return 0, err
 | |
| 	}
 | |
| 
 | |
| 	listener, err := net.ListenTCP("tcp", addr)
 | |
| 	if err != nil {
 | |
| 		return 0, err
 | |
| 	}
 | |
| 	defer listener.Close()
 | |
| 	//nolint:forcetypeassert
 | |
| 	return listener.Addr().(*net.TCPAddr).Port, nil
 | |
| }
 | |
| 
 | |
| // CleanUnreferencedNetworks removes networks that are not referenced by any containers.
 | |
| func CleanUnreferencedNetworks(pool *dockertest.Pool) error {
 | |
| 	filter := "name=hs-"
 | |
| 	networks, err := pool.NetworksByName(filter)
 | |
| 	if err != nil {
 | |
| 		return fmt.Errorf("getting networks by filter %q: %w", filter, err)
 | |
| 	}
 | |
| 
 | |
| 	for _, network := range networks {
 | |
| 		if network.Network.Containers == nil || len(network.Network.Containers) == 0 {
 | |
| 			err := pool.RemoveNetwork(&network)
 | |
| 			if err != nil {
 | |
| 				log.Printf("removing network %s: %s", network.Network.Name, err)
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| // CleanImagesInCI removes images if running in CI.
 | |
| func CleanImagesInCI(pool *dockertest.Pool) error {
 | |
| 	if !util.IsCI() {
 | |
| 		log.Println("Skipping image cleanup outside of CI")
 | |
| 		return nil
 | |
| 	}
 | |
| 
 | |
| 	images, err := pool.Client.ListImages(docker.ListImagesOptions{})
 | |
| 	if err != nil {
 | |
| 		return fmt.Errorf("getting images: %w", err)
 | |
| 	}
 | |
| 
 | |
| 	for _, image := range images {
 | |
| 		log.Printf("removing image: %s, %v", image.ID, image.RepoTags)
 | |
| 		_ = pool.Client.RemoveImage(image.ID)
 | |
| 	}
 | |
| 
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| // DockerRestartPolicy sets the restart policy for containers.
 | |
| func DockerRestartPolicy(config *docker.HostConfig) {
 | |
| 	config.RestartPolicy = docker.RestartPolicy{
 | |
| 		Name: "unless-stopped",
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // DockerAllowLocalIPv6 allows IPv6 traffic within the container.
 | |
| func DockerAllowLocalIPv6(config *docker.HostConfig) {
 | |
| 	config.NetworkMode = "default"
 | |
| 	config.Sysctls = map[string]string{
 | |
| 		"net.ipv6.conf.all.disable_ipv6": "0",
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // DockerAllowNetworkAdministration gives the container network administration capabilities.
 | |
| func DockerAllowNetworkAdministration(config *docker.HostConfig) {
 | |
| 	config.CapAdd = append(config.CapAdd, "NET_ADMIN")
 | |
| 	config.Privileged = true
 | |
| }
 | |
| 
 | |
| // DockerMemoryLimit sets memory limit and disables OOM kill for containers.
 | |
| func DockerMemoryLimit(config *docker.HostConfig) {
 | |
| 	config.Memory = 2 * 1024 * 1024 * 1024 // 2GB in bytes
 | |
| 	config.OOMKillDisable = true
 | |
| }
 |