portmap integration test: echo server runs in separate process
this way we're not mixing goroutines and namespaces
This commit is contained in:
parent
556e509097
commit
008024125a
74
plugins/meta/portmap/echosvr/echosvr_test.go
Normal file
74
plugins/meta/portmap/echosvr/echosvr_test.go
Normal 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")))
|
||||||
|
})
|
||||||
|
})
|
13
plugins/meta/portmap/echosvr/init_test.go
Normal file
13
plugins/meta/portmap/echosvr/init_test.go
Normal 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")
|
||||||
|
}
|
32
plugins/meta/portmap/echosvr/main.go
Normal file
32
plugins/meta/portmap/echosvr/main.go
Normal 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()
|
||||||
|
}
|
@ -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())
|
||||||
|
|
||||||
|
@ -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
|
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user