123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111 |
- package main
- import (
- "fmt"
- "net"
- "os/exec"
- "strings"
- "github.com/go-check/check"
- )
- func startServerContainer(c *check.C, proto string, port int) string {
- pStr := fmt.Sprintf("%d:%d", port, port)
- bCmd := fmt.Sprintf("nc -lp %d && echo bye", port)
- cmd := []string{"-d", "-p", pStr, "busybox", "sh", "-c", bCmd}
- if proto == "udp" {
- cmd = append(cmd, "-u")
- }
- name := "server"
- if err := waitForContainer(name, cmd...); err != nil {
- c.Fatalf("Failed to launch server container: %v", err)
- }
- return name
- }
- func getExternalAddress(c *check.C) net.IP {
- iface, err := net.InterfaceByName("eth0")
- if err != nil {
- c.Skip(fmt.Sprintf("Test not running with `make test`. Interface eth0 not found: %v", err))
- }
- ifaceAddrs, err := iface.Addrs()
- if err != nil || len(ifaceAddrs) == 0 {
- c.Fatalf("Error retrieving addresses for eth0: %v (%d addresses)", err, len(ifaceAddrs))
- }
- ifaceIP, _, err := net.ParseCIDR(ifaceAddrs[0].String())
- if err != nil {
- c.Fatalf("Error retrieving the up for eth0: %s", err)
- }
- return ifaceIP
- }
- func getContainerLogs(c *check.C, containerID string) string {
- runCmd := exec.Command(dockerBinary, "logs", containerID)
- out, _, err := runCommandWithOutput(runCmd)
- if err != nil {
- c.Fatal(out, err)
- }
- return strings.Trim(out, "\r\n")
- }
- func getContainerStatus(c *check.C, containerID string) string {
- out, err := inspectField(containerID, "State.Running")
- c.Assert(err, check.IsNil)
- return out
- }
- func (s *DockerSuite) TestNetworkNat(c *check.C) {
- testRequires(c, SameHostDaemon, NativeExecDriver)
- srv := startServerContainer(c, "tcp", 8080)
- // Spawn a new container which connects to the server through the
- // interface address.
- endpoint := getExternalAddress(c)
- runCmd := exec.Command(dockerBinary, "run", "busybox", "sh", "-c", fmt.Sprintf("echo hello world | nc -w 30 %s 8080", endpoint))
- if out, _, err := runCommandWithOutput(runCmd); err != nil {
- c.Fatalf("Failed to connect to server: %v (output: %q)", err, string(out))
- }
- result := getContainerLogs(c, srv)
- // Ideally we'd like to check for "hello world" but sometimes
- // nc doesn't show the data it received so instead let's look for
- // the output of the 'echo bye' that should be printed once
- // the nc command gets a connection
- expected := "bye"
- if !strings.Contains(result, expected) {
- c.Fatalf("Unexpected output. Expected: %q, received: %q", expected, result)
- }
- }
- func (s *DockerSuite) TestNetworkLocalhostTCPNat(c *check.C) {
- testRequires(c, SameHostDaemon, NativeExecDriver)
- srv := startServerContainer(c, "tcp", 8081)
- // Attempt to connect from the host to the listening container.
- conn, err := net.Dial("tcp", "localhost:8081")
- if err != nil {
- c.Fatalf("Failed to connect to container (%v)", err)
- }
- if _, err := conn.Write([]byte("hello world\n")); err != nil {
- c.Fatal(err)
- }
- conn.Close()
- result := getContainerLogs(c, srv)
- // Ideally we'd like to check for "hello world" but sometimes
- // nc doesn't show the data it received so instead let's look for
- // the output of the 'echo bye' that should be printed once
- // the nc command gets a connection
- expected := "bye"
- if !strings.Contains(result, expected) {
- c.Fatalf("Unexpected output. Expected: %q, received: %q", expected, result)
- }
- }
|