portmap integration test: echo server runs in separate process

this way we're not mixing goroutines and namespaces
This commit is contained in:
Gabriel Rosenhouse 2017-09-05 23:36:12 -07:00
parent 556e509097
commit 008024125a
5 changed files with 168 additions and 73 deletions

View File

@ -0,0 +1,74 @@
package main_test
import (
"fmt"
"io/ioutil"
"net"
"os/exec"
"strings"
. "github.com/onsi/ginkgo"
. "github.com/onsi/gomega"
"github.com/onsi/gomega/gbytes"
"github.com/onsi/gomega/gexec"
)
var binaryPath string
var _ = SynchronizedBeforeSuite(func() []byte {
binaryPath, err := gexec.Build("github.com/containernetworking/plugins/plugins/meta/portmap/echosvr")
Expect(err).NotTo(HaveOccurred())
return []byte(binaryPath)
}, func(data []byte) {
binaryPath = string(data)
})
var _ = SynchronizedAfterSuite(func() {}, func() {
gexec.CleanupBuildArtifacts()
})
var _ = Describe("Echosvr", func() {
var session *gexec.Session
BeforeEach(func() {
var err error
cmd := exec.Command(binaryPath)
session, err = gexec.Start(cmd, GinkgoWriter, GinkgoWriter)
Expect(err).NotTo(HaveOccurred())
})
AfterEach(func() {
session.Terminate().Wait()
})
It("starts and doesn't terminate immediately", func() {
Consistently(session).ShouldNot(gexec.Exit())
})
tryConnect := func() (net.Conn, error) {
programOutput := session.Out.Contents()
addr := strings.TrimSpace(string(programOutput))
conn, err := net.Dial("tcp", addr)
if err != nil {
return nil, err
}
return conn, err
}
It("prints its listening address to stdout", func() {
Eventually(session.Out).Should(gbytes.Say("\n"))
conn, err := tryConnect()
Expect(err).NotTo(HaveOccurred())
conn.Close()
})
It("will echo data back to us", func() {
Eventually(session.Out).Should(gbytes.Say("\n"))
conn, err := tryConnect()
Expect(err).NotTo(HaveOccurred())
defer conn.Close()
fmt.Fprintf(conn, "hello")
Expect(ioutil.ReadAll(conn)).To(Equal([]byte("hello")))
})
})

View File

@ -0,0 +1,13 @@
package main_test
import (
. "github.com/onsi/ginkgo"
. "github.com/onsi/gomega"
"testing"
)
func TestEchosvr(t *testing.T) {
RegisterFailHandler(Fail)
RunSpecs(t, "Echosvr Suite")
}

View File

@ -0,0 +1,32 @@
package main
import (
"fmt"
"net"
)
func main() {
listener, err := net.Listen("tcp", ":")
if err != nil {
panic(err)
}
_, port, err := net.SplitHostPort(listener.Addr().String())
if err != nil {
panic(err)
}
fmt.Printf("127.0.0.1:%s\n", port)
for {
conn, err := listener.Accept()
if err != nil {
panic(err)
}
go handleConnection(conn)
}
}
func handleConnection(conn net.Conn) {
buf := make([]byte, 512)
nBytesRead, _ := conn.Read(buf)
conn.Write(buf[0:nBytesRead])
conn.Close()
}

View File

@ -28,19 +28,20 @@ import (
"github.com/coreos/go-iptables/iptables" "github.com/coreos/go-iptables/iptables"
. "github.com/onsi/ginkgo" . "github.com/onsi/ginkgo"
. "github.com/onsi/gomega" . "github.com/onsi/gomega"
"github.com/onsi/gomega/gexec"
"github.com/vishvananda/netlink" "github.com/vishvananda/netlink"
) )
const TIMEOUT = 90 const TIMEOUT = 90
var _ = Describe("portmap integration tests", func() { var _ = Describe("portmap integration tests", func() {
rand.Seed(time.Now().UTC().UnixNano()) var (
configList *libcni.NetworkConfigList
var configList *libcni.NetworkConfigList cniConf *libcni.CNIConfig
var cniConf *libcni.CNIConfig targetNS ns.NetNS
var targetNS ns.NetNS containerPort int
var containerPort int session *gexec.Session
var closeChan chan interface{} )
BeforeEach(func() { BeforeEach(func() {
var err error var err error
@ -80,12 +81,12 @@ var _ = Describe("portmap integration tests", func() {
fmt.Fprintln(GinkgoWriter, "namespace:", targetNS.Path()) fmt.Fprintln(GinkgoWriter, "namespace:", targetNS.Path())
// Start an echo server and get the port // Start an echo server and get the port
containerPort, closeChan, err = RunEchoServerInNS(targetNS) containerPort, session, err = StartEchoServerInNamespace(targetNS)
Expect(err).NotTo(HaveOccurred()) Expect(err).NotTo(HaveOccurred())
}) })
AfterEach(func() { AfterEach(func() {
session.Terminate().Wait()
if targetNS != nil { if targetNS != nil {
targetNS.Close() targetNS.Close()
} }
@ -163,7 +164,7 @@ var _ = Describe("portmap integration tests", func() {
snatOK := testEchoServer(fmt.Sprintf("%s:%d", "127.0.0.1", hostPort)) snatOK := testEchoServer(fmt.Sprintf("%s:%d", "127.0.0.1", hostPort))
// Cleanup // Cleanup
close(closeChan) session.Terminate()
err = deleteNetwork() err = deleteNetwork()
Expect(err).NotTo(HaveOccurred()) Expect(err).NotTo(HaveOccurred())

View File

@ -15,89 +15,64 @@
package main package main
import ( import (
"fmt" "math/rand"
"net" "net"
"time" "os/exec"
"path/filepath"
"strconv"
"strings"
"github.com/containernetworking/plugins/pkg/ns" "github.com/containernetworking/plugins/pkg/ns"
. "github.com/onsi/ginkgo" . "github.com/onsi/ginkgo"
"github.com/onsi/ginkgo/config"
. "github.com/onsi/gomega" . "github.com/onsi/gomega"
"github.com/onsi/gomega/gbytes"
"github.com/onsi/gomega/gexec"
"testing" "testing"
) )
func TestPortmap(t *testing.T) { func TestPortmap(t *testing.T) {
rand.Seed(config.GinkgoConfig.RandomSeed)
RegisterFailHandler(Fail) RegisterFailHandler(Fail)
RunSpecs(t, "portmap Suite") RunSpecs(t, "portmap Suite")
} }
// OpenEchoServer opens a server that listens until closeChan is closed. var echoServerBinaryPath string
// It opens on a random port and sends the port number on portChan when
// the server is up and running. If an error is encountered, closes portChan.
// If closeChan is closed, closes the socket.
func OpenEchoServer(portChan chan<- int, closeChan <-chan interface{}) error {
laddr, err := net.ResolveTCPAddr("tcp", "0.0.0.0:0")
if err != nil {
close(portChan)
return err
}
sock, err := net.ListenTCP("tcp", laddr)
if err != nil {
close(portChan)
return err
}
defer sock.Close()
switch addr := sock.Addr().(type) { var _ = SynchronizedBeforeSuite(func() []byte {
case *net.TCPAddr: binaryPath, err := gexec.Build("github.com/containernetworking/plugins/plugins/meta/portmap/echosvr")
portChan <- addr.Port Expect(err).NotTo(HaveOccurred())
default: return []byte(binaryPath)
close(portChan) }, func(data []byte) {
return fmt.Errorf("addr cast failed!") echoServerBinaryPath = string(data)
} })
for {
select {
case <-closeChan:
break
default:
}
sock.SetDeadline(time.Now().Add(time.Second)) var _ = SynchronizedAfterSuite(func() {}, func() {
con, err := sock.AcceptTCP() gexec.CleanupBuildArtifacts()
if err != nil { })
if opErr, ok := err.(*net.OpError); ok && opErr.Timeout() {
continue
}
continue
}
buf := make([]byte, 512) func startInNetNS(binPath string, netNS ns.NetNS) (*gexec.Session, error) {
con.Read(buf) baseName := filepath.Base(netNS.Path())
con.Write(buf) // we are relying on the netNS path living in /var/run/netns
con.Close() // where `ip netns exec` can find it
} cmd := exec.Command("ip", "netns", "exec", baseName, binPath)
session, err := gexec.Start(cmd, GinkgoWriter, GinkgoWriter)
return session, err
} }
func RunEchoServerInNS(netNS ns.NetNS) (int, chan interface{}, error) { func StartEchoServerInNamespace(netNS ns.NetNS) (int, *gexec.Session, error) {
portChan := make(chan int) session, err := startInNetNS(echoServerBinaryPath, netNS)
closeChan := make(chan interface{}) Expect(err).NotTo(HaveOccurred())
go func() { // wait for it to print it's address on stdout
err := netNS.Do(func(ns.NetNS) error { Eventually(session.Out).Should(gbytes.Say("\n"))
OpenEchoServer(portChan, closeChan) _, portString, err := net.SplitHostPort(strings.TrimSpace(string(session.Out.Contents())))
return nil Expect(err).NotTo(HaveOccurred())
})
// Somehow the ns.Do failed
if err != nil {
close(portChan)
}
}()
portNum := <-portChan port, err := strconv.Atoi(portString)
if portNum == 0 { Expect(err).NotTo(HaveOccurred())
return 0, nil, fmt.Errorf("failed to execute server") return port, session, nil
}
return portNum, closeChan, nil
} }