mirror of
https://github.com/moby/moby.git
synced 2026-01-11 10:41:43 +00:00
Use a cancelReadCloser to automatically close the reader when the context is cancelled. Consumers are still recommended to manually close the reader, but the cancelReadCloser makes the Close idempotent. Signed-off-by: Sebastiaan van Stijn <github@gone.nl>
164 lines
4.3 KiB
Go
164 lines
4.3 KiB
Go
package container
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"fmt"
|
|
"io"
|
|
"net"
|
|
"strconv"
|
|
"strings"
|
|
"testing"
|
|
|
|
"github.com/moby/moby/api/types/network"
|
|
"github.com/moby/moby/client"
|
|
"github.com/moby/moby/v2/integration/internal/container"
|
|
"gotest.tools/v3/assert"
|
|
is "gotest.tools/v3/assert/cmp"
|
|
"gotest.tools/v3/poll"
|
|
"gotest.tools/v3/skip"
|
|
)
|
|
|
|
func TestNetworkNat(t *testing.T) {
|
|
skip.If(t, testEnv.DaemonInfo.OSType == "windows", "FIXME")
|
|
skip.If(t, testEnv.IsRemoteDaemon)
|
|
|
|
ctx := setupTest(t)
|
|
|
|
const msg = "it works"
|
|
const port = 8080
|
|
startServerContainer(ctx, t, msg, port)
|
|
|
|
endpoint := getExternalAddress(t)
|
|
|
|
var conn net.Conn
|
|
addr := net.JoinHostPort(endpoint.String(), strconv.Itoa(port))
|
|
poll.WaitOn(t, func(t poll.LogT) poll.Result {
|
|
var err error
|
|
conn, err = net.Dial("tcp", addr)
|
|
if err != nil {
|
|
return poll.Continue("waiting for %s to be accessible: %v", addr, err)
|
|
}
|
|
return poll.Success()
|
|
})
|
|
defer func() {
|
|
assert.Check(t, conn.Close())
|
|
}()
|
|
|
|
data, err := io.ReadAll(conn)
|
|
assert.NilError(t, err)
|
|
assert.Check(t, is.Equal(msg, strings.TrimSpace(string(data))))
|
|
}
|
|
|
|
func TestNetworkLocalhostTCPNat(t *testing.T) {
|
|
skip.If(t, testEnv.IsRemoteDaemon)
|
|
|
|
ctx := setupTest(t)
|
|
|
|
const msg = "hi yall"
|
|
const port = 8081
|
|
startServerContainer(ctx, t, msg, port)
|
|
|
|
var conn net.Conn
|
|
addr := net.JoinHostPort("localhost", strconv.Itoa(port))
|
|
poll.WaitOn(t, func(t poll.LogT) poll.Result {
|
|
var err error
|
|
conn, err = net.Dial("tcp", addr)
|
|
if err != nil {
|
|
return poll.Continue("waiting for %s to be accessible: %v", addr, err)
|
|
}
|
|
return poll.Success()
|
|
})
|
|
defer func() {
|
|
assert.Check(t, conn.Close())
|
|
}()
|
|
|
|
data, err := io.ReadAll(conn)
|
|
assert.NilError(t, err)
|
|
assert.Check(t, is.Equal(msg, strings.TrimSpace(string(data))))
|
|
}
|
|
|
|
func TestNetworkLoopbackNat(t *testing.T) {
|
|
skip.If(t, testEnv.GitHubActions, "FIXME: https://github.com/moby/moby/issues/41561")
|
|
skip.If(t, testEnv.DaemonInfo.OSType == "windows", "FIXME")
|
|
skip.If(t, testEnv.IsRemoteDaemon)
|
|
|
|
ctx := setupTest(t)
|
|
|
|
msg := "it works"
|
|
serverContainerID := startServerContainer(ctx, t, msg, 8080)
|
|
|
|
endpoint := getExternalAddress(t)
|
|
|
|
apiClient := testEnv.APIClient()
|
|
|
|
cID := container.Run(ctx, t, apiClient,
|
|
container.WithCmd("sh", "-c", fmt.Sprintf("stty raw && nc -w 1 %s 8080", endpoint.String())),
|
|
container.WithTty(true),
|
|
container.WithNetworkMode("container:"+serverContainerID),
|
|
)
|
|
|
|
poll.WaitOn(t, container.IsStopped(ctx, apiClient, cID))
|
|
|
|
logs, err := apiClient.ContainerLogs(ctx, cID, client.ContainerLogsOptions{
|
|
ShowStdout: true,
|
|
})
|
|
assert.NilError(t, err)
|
|
defer logs.Close()
|
|
|
|
var b bytes.Buffer
|
|
_, err = io.Copy(&b, logs)
|
|
assert.NilError(t, err)
|
|
|
|
assert.Check(t, is.Equal(msg, strings.TrimSpace(b.String())))
|
|
}
|
|
|
|
func startServerContainer(ctx context.Context, t *testing.T, msg string, port uint16) string {
|
|
t.Helper()
|
|
apiClient := testEnv.APIClient()
|
|
|
|
return container.Run(ctx, t, apiClient,
|
|
container.WithName("server-"+t.Name()),
|
|
container.WithCmd("sh", "-c", fmt.Sprintf("echo %q | nc -lp %d", msg, port)),
|
|
container.WithExposedPorts(fmt.Sprintf("%d/tcp", port)),
|
|
func(c *container.TestContainerConfig) {
|
|
c.HostConfig.PortBindings = network.PortMap{
|
|
network.MustParsePort(fmt.Sprintf("%d/tcp", port)): []network.PortBinding{
|
|
{
|
|
HostPort: fmt.Sprintf("%d", port),
|
|
},
|
|
},
|
|
}
|
|
},
|
|
)
|
|
}
|
|
|
|
// getExternalAddress() returns the external IP-address from eth0. If eth0 has
|
|
// multiple IP-addresses, it returns the first IPv4 IP-address; if no IPv4
|
|
// address is present, it returns the first IP-address found.
|
|
func getExternalAddress(t *testing.T) net.IP {
|
|
t.Helper()
|
|
iface, err := net.InterfaceByName("eth0")
|
|
skip.If(t, err != nil, "Test not running with `make test-integration`. Interface eth0 not found: %s", err)
|
|
|
|
ifaceAddrs, err := iface.Addrs()
|
|
assert.NilError(t, err)
|
|
assert.Check(t, len(ifaceAddrs) != 0)
|
|
|
|
if len(ifaceAddrs) > 1 {
|
|
// Prefer IPv4 address if multiple addresses found, as rootlesskit
|
|
// does not handle IPv6 currently https://github.com/moby/moby/pull/41908#issuecomment-774200001
|
|
for _, a := range ifaceAddrs {
|
|
ifaceIP, _, err := net.ParseCIDR(a.String())
|
|
assert.NilError(t, err)
|
|
if ifaceIP.To4() != nil {
|
|
return ifaceIP
|
|
}
|
|
}
|
|
}
|
|
ifaceIP, _, err := net.ParseCIDR(ifaceAddrs[0].String())
|
|
assert.NilError(t, err)
|
|
|
|
return ifaceIP
|
|
}
|