diff options
Diffstat (limited to 'api/api_test.go')
-rw-r--r-- | api/api_test.go | 588 |
1 files changed, 0 insertions, 588 deletions
diff --git a/api/api_test.go b/api/api_test.go deleted file mode 100644 index 7cbd9f0..0000000 --- a/api/api_test.go +++ /dev/null @@ -1,588 +0,0 @@ -// Copyright (c) 2017 Cisco and/or its affiliates. -// -// Licensed under the Apache License, Version 2.0 (the "License"); -// you may not use this file except in compliance with the License. -// You may obtain a copy of the License at: -// -// http://www.apache.org/licenses/LICENSE-2.0 -// -// Unless required by applicable law or agreed to in writing, software -// distributed under the License is distributed on an "AS IS" BASIS, -// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -// See the License for the specific language governing permissions and -// limitations under the License. - -package api_test - -import ( - "testing" - "time" - - "git.fd.io/govpp.git/adapter/mock" - "git.fd.io/govpp.git/api" - "git.fd.io/govpp.git/core" - "git.fd.io/govpp.git/core/bin_api/vpe" - "git.fd.io/govpp.git/examples/bin_api/interfaces" - "git.fd.io/govpp.git/examples/bin_api/memif" - "git.fd.io/govpp.git/examples/bin_api/tap" - - . "github.com/onsi/gomega" -) - -type testCtx struct { - mockVpp *mock.VppAdapter - conn *core.Connection - ch *api.Channel -} - -func setupTest(t *testing.T) *testCtx { - RegisterTestingT(t) - - ctx := &testCtx{ - mockVpp: &mock.VppAdapter{}, - } - - var err error - ctx.conn, err = core.Connect(ctx.mockVpp) - Expect(err).ShouldNot(HaveOccurred()) - - ctx.ch, err = ctx.conn.NewAPIChannel() - Expect(err).ShouldNot(HaveOccurred()) - - return ctx -} - -func (ctx *testCtx) teardownTest() { - ctx.ch.Close() - ctx.conn.Disconnect() -} - -func TestRequestReplyTapConnect(t *testing.T) { - ctx := setupTest(t) - defer ctx.teardownTest() - - ctx.mockVpp.MockReply(&tap.TapConnectReply{ - Retval: 10, - SwIfIndex: 1, - }) - request := &tap.TapConnect{ - TapName: []byte("test-tap-name"), - UseRandomMac: 1, - } - reply := &tap.TapConnectReply{} - - err := ctx.ch.SendRequest(request).ReceiveReply(reply) - Expect(err).ShouldNot(HaveOccurred()) - Expect(reply.Retval).To(BeEquivalentTo(10), "Incorrect retval value for TapConnectReply") - Expect(reply.SwIfIndex).To(BeEquivalentTo(1), "Incorrect SwIfIndex value for TapConnectReply") -} - -func TestRequestReplyTapModify(t *testing.T) { - ctx := setupTest(t) - defer ctx.teardownTest() - - ctx.mockVpp.MockReply(&tap.TapModifyReply{ - Retval: 15, - SwIfIndex: 2, - }) - request := &tap.TapModify{ - TapName: []byte("test-tap-modify"), - UseRandomMac: 1, - CustomDevInstance: 1, - } - reply := &tap.TapModifyReply{} - - err := ctx.ch.SendRequest(request).ReceiveReply(reply) - Expect(err).ShouldNot(HaveOccurred()) - Expect(reply.Retval).To(BeEquivalentTo(15), "Incorrect retval value for TapModifyReply") - Expect(reply.SwIfIndex).To(BeEquivalentTo(2), "Incorrect SwIfIndex value for TapModifyReply") -} - -func TestRequestReplyTapDelete(t *testing.T) { - ctx := setupTest(t) - defer ctx.teardownTest() - - ctx.mockVpp.MockReply(&tap.TapDeleteReply{ - Retval: 20, - }) - request := &tap.TapDelete{ - SwIfIndex: 3, - } - reply := &tap.TapDeleteReply{} - - err := ctx.ch.SendRequest(request).ReceiveReply(reply) - Expect(err).ShouldNot(HaveOccurred()) - Expect(reply.Retval).To(BeEquivalentTo(20), "Incorrect retval value for TapDeleteReply") -} - -func TestRequestReplySwInterfaceTapDump(t *testing.T) { - ctx := setupTest(t) - defer ctx.teardownTest() - - byteName := []byte("dev-name-test") - ctx.mockVpp.MockReply(&tap.SwInterfaceTapDetails{ - SwIfIndex: 25, - DevName: byteName, - }) - request := &tap.SwInterfaceTapDump{} - reply := &tap.SwInterfaceTapDetails{} - - err := ctx.ch.SendRequest(request).ReceiveReply(reply) - Expect(err).ShouldNot(HaveOccurred()) - Expect(reply.SwIfIndex).To(BeEquivalentTo(25), "Incorrect SwIfIndex value for SwInterfaceTapDetails") - Expect(reply.DevName).ToNot(BeNil(), "Incorrect DevName value for SwInterfaceTapDetails") -} - -func TestRequestReplyMemifCreate(t *testing.T) { - ctx := setupTest(t) - defer ctx.teardownTest() - - ctx.mockVpp.MockReply(&memif.MemifCreateReply{ - Retval: 22, - SwIfIndex: 4, - }) - request := &memif.MemifCreate{ - Role: 10, - ID: 12, - RingSize: 8000, - BufferSize: 50, - } - reply := &memif.MemifCreateReply{} - - err := ctx.ch.SendRequest(request).ReceiveReply(reply) - Expect(err).ShouldNot(HaveOccurred()) - Expect(reply.Retval).To(BeEquivalentTo(22), "Incorrect Retval value for MemifCreate") - Expect(reply.SwIfIndex).To(BeEquivalentTo(4), "Incorrect SwIfIndex value for MemifCreate") -} - -func TestRequestReplyMemifDelete(t *testing.T) { - ctx := setupTest(t) - defer ctx.teardownTest() - - ctx.mockVpp.MockReply(&memif.MemifDeleteReply{ - Retval: 24, - }) - request := &memif.MemifDelete{ - SwIfIndex: 15, - } - reply := &memif.MemifDeleteReply{} - - err := ctx.ch.SendRequest(request).ReceiveReply(reply) - Expect(err).ShouldNot(HaveOccurred()) - Expect(reply.Retval).To(BeEquivalentTo(24), "Incorrect Retval value for MemifDelete") -} - -func TestRequestReplyMemifDetails(t *testing.T) { - ctx := setupTest(t) - defer ctx.teardownTest() - - ctx.mockVpp.MockReply(&memif.MemifDetails{ - SwIfIndex: 25, - IfName: []byte("memif-name"), - Role: 0, - }) - request := &memif.MemifDump{} - reply := &memif.MemifDetails{} - - err := ctx.ch.SendRequest(request).ReceiveReply(reply) - Expect(err).ShouldNot(HaveOccurred()) - Expect(reply.SwIfIndex).To(BeEquivalentTo(25), "Incorrect SwIfIndex value for MemifDetails") - Expect(reply.IfName).ToNot(BeEmpty(), "MemifDetails IfName is empty byte array") - Expect(reply.Role).To(BeEquivalentTo(0), "Incorrect Role value for MemifDetails") -} - -func TestMultiRequestReplySwInterfaceTapDump(t *testing.T) { - ctx := setupTest(t) - defer ctx.teardownTest() - - // mock reply - msgs := []api.Message{} - for i := 1; i <= 10; i++ { - msgs = append(msgs, &tap.SwInterfaceTapDetails{ - SwIfIndex: uint32(i), - DevName: []byte("dev-name-test"), - }) - } - ctx.mockVpp.MockReply(msgs...) - ctx.mockVpp.MockReply(&vpe.ControlPingReply{}) - - reqCtx := ctx.ch.SendMultiRequest(&tap.SwInterfaceTapDump{}) - cnt := 0 - for { - msg := &tap.SwInterfaceTapDetails{} - stop, err := reqCtx.ReceiveReply(msg) - if stop { - break // break out of the loop - } - Expect(err).ShouldNot(HaveOccurred()) - cnt++ - } - Expect(cnt).To(BeEquivalentTo(10)) -} - -func TestMultiRequestReplySwInterfaceMemifDump(t *testing.T) { - ctx := setupTest(t) - defer ctx.teardownTest() - - // mock reply - msgs := []api.Message{} - for i := 1; i <= 10; i++ { - msgs = append(msgs, &memif.MemifDetails{ - SwIfIndex: uint32(i), - }) - } - ctx.mockVpp.MockReply(msgs...) - ctx.mockVpp.MockReply(&vpe.ControlPingReply{}) - - reqCtx := ctx.ch.SendMultiRequest(&memif.MemifDump{}) - cnt := 0 - for { - msg := &memif.MemifDetails{} - stop, err := reqCtx.ReceiveReply(msg) - if stop { - break // break out of the loop - } - Expect(err).ShouldNot(HaveOccurred()) - cnt++ - } - Expect(cnt).To(BeEquivalentTo(10)) -} - -func TestNotifications(t *testing.T) { - ctx := setupTest(t) - defer ctx.teardownTest() - - // subscribe for notification - notifChan := make(chan api.Message, 1) - subs, err := ctx.ch.SubscribeNotification(notifChan, interfaces.NewSwInterfaceSetFlags) - Expect(err).ShouldNot(HaveOccurred()) - - // mock the notification and force its delivery - ctx.mockVpp.MockReply(&interfaces.SwInterfaceSetFlags{ - SwIfIndex: 3, - AdminUpDown: 1, - }) - ctx.mockVpp.SendMsg(0, []byte("")) - - // receive the notification - var notif *interfaces.SwInterfaceSetFlags - Eventually(func() *interfaces.SwInterfaceSetFlags { - select { - case n := <-notifChan: - notif = n.(*interfaces.SwInterfaceSetFlags) - return notif - default: - return nil - } - }).ShouldNot(BeNil()) - - // verify the received notifications - Expect(notif.SwIfIndex).To(BeEquivalentTo(3), "Incorrect SwIfIndex value for SwInterfaceSetFlags") - Expect(notif.AdminUpDown).To(BeEquivalentTo(1), "Incorrect AdminUpDown value for SwInterfaceSetFlags") - - ctx.ch.UnsubscribeNotification(subs) -} - -func TestNotificationEvent(t *testing.T) { - ctx := setupTest(t) - defer ctx.teardownTest() - - // subscribe for notification - notifChan := make(chan api.Message, 1) - subs, err := ctx.ch.SubscribeNotification(notifChan, interfaces.NewSwInterfaceEvent) - Expect(err).ShouldNot(HaveOccurred()) - - // mock the notification and force its delivery - ctx.mockVpp.MockReply(&interfaces.SwInterfaceEvent{ - SwIfIndex: 2, - LinkUpDown: 1, - }) - ctx.mockVpp.SendMsg(0, []byte("")) - - // receive the notification - var notif *interfaces.SwInterfaceEvent - Eventually(func() *interfaces.SwInterfaceEvent { - select { - case n := <-notifChan: - notif = n.(*interfaces.SwInterfaceEvent) - return notif - default: - return nil - } - }).ShouldNot(BeNil()) - - // verify the received notifications - Expect(notif.SwIfIndex).To(BeEquivalentTo(2), "Incorrect SwIfIndex value for SwInterfaceSetFlags") - Expect(notif.LinkUpDown).To(BeEquivalentTo(1), "Incorrect LinkUpDown value for SwInterfaceSetFlags") - - ctx.ch.UnsubscribeNotification(subs) -} - -func TestCheckMessageCompatibility(t *testing.T) { - ctx := setupTest(t) - defer ctx.teardownTest() - - err := ctx.ch.CheckMessageCompatibility(&interfaces.SwInterfaceSetFlags{}) - Expect(err).ShouldNot(HaveOccurred()) -} -func TestSetReplyTimeout(t *testing.T) { - ctx := setupTest(t) - defer ctx.teardownTest() - - ctx.ch.SetReplyTimeout(time.Millisecond) - - // first one request should work - ctx.mockVpp.MockReply(&vpe.ControlPingReply{}) - err := ctx.ch.SendRequest(&vpe.ControlPing{}).ReceiveReply(&vpe.ControlPingReply{}) - Expect(err).ShouldNot(HaveOccurred()) - - // no other reply ready - expect timeout - err = ctx.ch.SendRequest(&vpe.ControlPing{}).ReceiveReply(&vpe.ControlPingReply{}) - Expect(err).Should(HaveOccurred()) - Expect(err.Error()).To(ContainSubstring("timeout")) -} - -func TestSetReplyTimeoutMultiRequest(t *testing.T) { - ctx := setupTest(t) - defer ctx.teardownTest() - - ctx.ch.SetReplyTimeout(time.Millisecond) - - msgs := []api.Message{} - for i := 1; i <= 3; i++ { - msgs = append(msgs, &interfaces.SwInterfaceDetails{ - SwIfIndex: uint32(i), - InterfaceName: []byte("if-name-test"), - }) - } - ctx.mockVpp.MockReply(msgs...) - ctx.mockVpp.MockReply(&vpe.ControlPingReply{}) - - cnt := 0 - sendMultiRequest := func() error { - reqCtx := ctx.ch.SendMultiRequest(&interfaces.SwInterfaceDump{}) - for { - msg := &interfaces.SwInterfaceDetails{} - stop, err := reqCtx.ReceiveReply(msg) - if stop { - break // break out of the loop - } - if err != nil { - return err - } - cnt++ - } - return nil - } - - // first one request should work - err := sendMultiRequest() - Expect(err).ShouldNot(HaveOccurred()) - - // no other reply ready - expect timeout - err = sendMultiRequest() - Expect(err).Should(HaveOccurred()) - Expect(err.Error()).To(ContainSubstring("timeout")) - - Expect(cnt).To(BeEquivalentTo(3)) -} - -func TestReceiveReplyNegative(t *testing.T) { - ctx := setupTest(t) - defer ctx.teardownTest() - - // invalid context 1 - reqCtx1 := &api.RequestCtx{} - err := reqCtx1.ReceiveReply(&vpe.ControlPingReply{}) - Expect(err).Should(HaveOccurred()) - Expect(err.Error()).To(ContainSubstring("invalid request context")) - - // invalid context 2 - reqCtx2 := &api.MultiRequestCtx{} - _, err = reqCtx2.ReceiveReply(&vpe.ControlPingReply{}) - Expect(err).Should(HaveOccurred()) - Expect(err.Error()).To(ContainSubstring("invalid request context")) - - // NU - reqCtx3 := &api.RequestCtx{} - err = reqCtx3.ReceiveReply(nil) - Expect(err).Should(HaveOccurred()) - Expect(err.Error()).To(ContainSubstring("invalid request context")) -} - -func TestMultiRequestDouble(t *testing.T) { - ctx := setupTest(t) - defer ctx.teardownTest() - - // mock reply - msgs := []mock.MsgWithContext{} - for i := 1; i <= 3; i++ { - msgs = append(msgs, mock.MsgWithContext{ - Msg: &interfaces.SwInterfaceDetails{ - SwIfIndex: uint32(i), - InterfaceName: []byte("if-name-test"), - }, - Multipart: true, - SeqNum: 1, - }) - } - msgs = append(msgs, mock.MsgWithContext{Msg: &vpe.ControlPingReply{}, Multipart: true, SeqNum: 1}) - - for i := 1; i <= 3; i++ { - msgs = append(msgs, - mock.MsgWithContext{ - Msg: &interfaces.SwInterfaceDetails{ - SwIfIndex: uint32(i), - InterfaceName: []byte("if-name-test"), - }, - Multipart: true, - SeqNum: 2, - }) - } - msgs = append(msgs, mock.MsgWithContext{Msg: &vpe.ControlPingReply{}, Multipart: true, SeqNum: 2}) - - ctx.mockVpp.MockReplyWithContext(msgs...) - - cnt := 0 - var sendMultiRequest = func() error { - reqCtx := ctx.ch.SendMultiRequest(&interfaces.SwInterfaceDump{}) - for { - msg := &interfaces.SwInterfaceDetails{} - stop, err := reqCtx.ReceiveReply(msg) - if stop { - break // break out of the loop - } - if err != nil { - return err - } - cnt++ - } - return nil - } - - err := sendMultiRequest() - Expect(err).ShouldNot(HaveOccurred()) - - err = sendMultiRequest() - Expect(err).ShouldNot(HaveOccurred()) - - Expect(cnt).To(BeEquivalentTo(6)) -} - -func TestReceiveReplyAfterTimeout(t *testing.T) { - ctx := setupTest(t) - defer ctx.teardownTest() - - ctx.ch.SetReplyTimeout(time.Millisecond) - - // first one request should work - ctx.mockVpp.MockReplyWithContext(mock.MsgWithContext{Msg: &vpe.ControlPingReply{}, SeqNum: 1}) - err := ctx.ch.SendRequest(&vpe.ControlPing{}).ReceiveReply(&vpe.ControlPingReply{}) - Expect(err).ShouldNot(HaveOccurred()) - - err = ctx.ch.SendRequest(&vpe.ControlPing{}).ReceiveReply(&vpe.ControlPingReply{}) - Expect(err).Should(HaveOccurred()) - Expect(err.Error()).To(ContainSubstring("timeout")) - - ctx.mockVpp.MockReplyWithContext( - // simulating late reply - mock.MsgWithContext{Msg: &vpe.ControlPingReply{}, SeqNum: 2}, - // normal reply for next request - mock.MsgWithContext{Msg: &tap.TapConnectReply{}, SeqNum: 3}) - - req := &tap.TapConnect{ - TapName: []byte("test-tap-name"), - UseRandomMac: 1, - } - reply := &tap.TapConnectReply{} - - // should succeed - err = ctx.ch.SendRequest(req).ReceiveReply(reply) - Expect(err).ShouldNot(HaveOccurred()) -} - -func TestReceiveReplyAfterTimeoutMultiRequest(t *testing.T) { - /* - TODO: fix mock adapter - This test will fail because mock adapter will stop sending replies - when it encounters control_ping_reply from multi request, - thus never sending reply for next request - */ - t.Skip() - - ctx := setupTest(t) - defer ctx.teardownTest() - - ctx.ch.SetReplyTimeout(time.Millisecond * 100) - - // first one request should work - ctx.mockVpp.MockReplyWithContext(mock.MsgWithContext{Msg: &vpe.ControlPingReply{}, SeqNum: 1}) - err := ctx.ch.SendRequest(&vpe.ControlPing{}).ReceiveReply(&vpe.ControlPingReply{}) - Expect(err).ShouldNot(HaveOccurred()) - - cnt := 0 - var sendMultiRequest = func() error { - reqCtx := ctx.ch.SendMultiRequest(&interfaces.SwInterfaceDump{}) - for { - msg := &interfaces.SwInterfaceDetails{} - stop, err := reqCtx.ReceiveReply(msg) - if stop { - break // break out of the loop - } - if err != nil { - return err - } - cnt++ - } - return nil - } - - err = sendMultiRequest() - Expect(err).Should(HaveOccurred()) - Expect(err.Error()).To(ContainSubstring("timeout")) - Expect(cnt).To(BeEquivalentTo(0)) - - // simulating late replies - msgs := []mock.MsgWithContext{} - for i := 1; i <= 3; i++ { - msgs = append(msgs, mock.MsgWithContext{ - Msg: &interfaces.SwInterfaceDetails{ - SwIfIndex: uint32(i), - InterfaceName: []byte("if-name-test"), - }, - Multipart: true, - SeqNum: 2, - }) - } - msgs = append(msgs, mock.MsgWithContext{Msg: &vpe.ControlPingReply{}, Multipart: true, SeqNum: 2}) - ctx.mockVpp.MockReplyWithContext(msgs...) - - // normal reply for next request - ctx.mockVpp.MockReplyWithContext(mock.MsgWithContext{Msg: &tap.TapConnectReply{}, SeqNum: 3}) - - req := &tap.TapConnect{ - TapName: []byte("test-tap-name"), - UseRandomMac: 1, - } - reply := &tap.TapConnectReply{} - - // should succeed - err = ctx.ch.SendRequest(req).ReceiveReply(reply) - Expect(err).ShouldNot(HaveOccurred()) -} - -func TestInvalidMessageID(t *testing.T) { - ctx := setupTest(t) - defer ctx.teardownTest() - - // first one request should work - ctx.mockVpp.MockReply(&vpe.ShowVersionReply{}) - err := ctx.ch.SendRequest(&vpe.ShowVersion{}).ReceiveReply(&vpe.ShowVersionReply{}) - Expect(err).ShouldNot(HaveOccurred()) - - // second should fail with error invalid message ID - ctx.mockVpp.MockReply(&vpe.ShowVersionReply{}) - err = ctx.ch.SendRequest(&vpe.ControlPing{}).ReceiveReply(&vpe.ControlPingReply{}) - Expect(err).Should(HaveOccurred()) - Expect(err.Error()).To(ContainSubstring("invalid message ID")) -} |