/* * Copyright (c) 2013 Cisco Systems, Inc. and others. All rights reserved. * * This program and the accompanying materials are made available under the * terms of the Eclipse Public License v1.0 which accompanies this distribution, * and is available at http://www.eclipse.org/legal/epl-v10.html */ package org.opendaylight.protocol.pcep.impl; import static org.junit.Assert.assertEquals; import static org.junit.Assert.assertTrue; import static org.mockito.Matchers.any; import static org.mockito.Mockito.doAnswer; import static org.mockito.Mockito.doReturn; import static org.mockito.Mockito.mock; import com.google.common.collect.Lists; import io.netty.channel.Channel; import io.netty.channel.ChannelFuture; import io.netty.channel.ChannelHandler; import io.netty.channel.ChannelPipeline; import io.netty.channel.DefaultChannelPromise; import io.netty.channel.EventLoop; import io.netty.util.concurrent.DefaultPromise; import io.netty.util.concurrent.GlobalEventExecutor; import io.netty.util.concurrent.ScheduledFuture; import java.net.InetSocketAddress; import java.net.SocketAddress; import java.util.List; import java.util.concurrent.TimeUnit; import org.junit.After; import org.junit.Before; import org.junit.Ignore; import org.junit.Test; import org.mockito.Mock; import org.mockito.MockitoAnnotations; import org.mockito.invocation.InvocationOnMock; import org.mockito.stubbing.Answer; import org.opendaylight.protocol.pcep.spi.PCEPErrors; import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.pcep.message.rev131007.Keepalive; import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.pcep.message.rev131007.KeepaliveBuilder; import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.pcep.message.rev131007.Open; import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.pcep.message.rev131007.OpenBuilder; import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.pcep.message.rev131007.Pcerr; import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.pcep.types.rev131005.OpenMessage; import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.pcep.types.rev131005.keepalive.message.KeepaliveMessageBuilder; import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.pcep.types.rev131005.open.message.OpenMessageBuilder; import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.pcep.types.rev131005.pcerr.message.pcerr.message.Errors; import org.opendaylight.yangtools.yang.binding.Notification; public class FiniteStateMachineTest { private DefaultPCEPSessionNegotiator serverSession; @Mock private Channel clientListener; @Mock private EventLoop eventLoop; @Mock private ScheduledFuture future; @Mock private ChannelPipeline pipeline; @Mock private SocketAddress address; private final List receivedMsgs = Lists.newArrayList(); private Open openmsg; private Keepalive kamsg; @Before public void setUp() { MockitoAnnotations.initMocks(this); final org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.pcep.types.rev131005.open.object.Open localPrefs = new org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.pcep.types.rev131005.open.object.OpenBuilder().setKeepalive( (short) 1).build(); this.serverSession = new DefaultPCEPSessionNegotiator(new DefaultPromise(GlobalEventExecutor.INSTANCE), this.clientListener, new SimpleSessionListener(), (short) 1, 20, localPrefs); final ChannelFuture future = new DefaultChannelPromise(this.clientListener); doAnswer(new Answer() { @Override public Object answer(final InvocationOnMock invocation) { final Object[] args = invocation.getArguments(); FiniteStateMachineTest.this.receivedMsgs.add((Notification) args[0]); return future; } }).when(this.clientListener).writeAndFlush(any(Notification.class)); doReturn("TestingChannel").when(this.clientListener).toString(); doReturn(this.pipeline).when(this.clientListener).pipeline(); doReturn(this.address).when(this.clientListener).localAddress(); doReturn(this.address).when(this.clientListener).remoteAddress(); doReturn(this.eventLoop).when(this.clientListener).eventLoop(); doReturn(true).when(this.future).cancel(false); doReturn(this.future).when(this.eventLoop).schedule(any(Runnable.class), any(long.class), any(TimeUnit.class)); doReturn(this.pipeline).when(this.pipeline).replace(any(ChannelHandler.class), any(String.class), any(ChannelHandler.class)); doReturn(true).when(this.clientListener).isActive(); doReturn(mock(ChannelFuture.class)).when(this.clientListener).close(); doReturn(InetSocketAddress.createUnresolved("127.0.0.1", 4189)).when(this.clientListener).remoteAddress(); doReturn(InetSocketAddress.createUnresolved("127.0.0.1", 4189)).when(this.clientListener).localAddress(); this.openmsg = new OpenBuilder().setOpenMessage( new OpenMessageBuilder().setOpen( new org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.pcep.types.rev131005.open.object.OpenBuilder().setDeadTimer( (short) 45).setKeepalive((short) 15).build()).build()).build(); this.kamsg = new KeepaliveBuilder().setKeepaliveMessage(new KeepaliveMessageBuilder().build()).build(); } /** * Both PCEs accept session characteristics. Also tests KeepAliveTimer and error message and when pce attempts to * establish pce session for the 2nd time. * * @throws Exception */ @Test public void testSessionCharsAccBoth() throws Exception { this.serverSession.channelActive(null); assertEquals(1, this.receivedMsgs.size()); assertTrue(this.receivedMsgs.get(0) instanceof Open); this.serverSession.handleMessage(this.openmsg); assertEquals(2, this.receivedMsgs.size()); assertTrue(this.receivedMsgs.get(1) instanceof Keepalive); this.serverSession.handleMessage(this.kamsg); assertEquals(this.serverSession.getState(), DefaultPCEPSessionNegotiator.State.Finished); } /** * Mock PCE does not accept session characteristics the first time. * * @throws Exception */ @Test public void testSessionCharsAccMe() throws Exception { this.serverSession.channelActive(null); assertEquals(1, this.receivedMsgs.size()); assertTrue(this.receivedMsgs.get(0) instanceof Open); final Open remote = (Open) this.receivedMsgs.get(0); this.serverSession.handleMessage(this.openmsg); assertEquals(2, this.receivedMsgs.size()); assertTrue(this.receivedMsgs.get(1) instanceof Keepalive); this.serverSession.handleMessage(Util.createErrorMessage(PCEPErrors.NON_ACC_NEG_SESSION_CHAR, remote.getOpenMessage().getOpen())); assertEquals(3, this.receivedMsgs.size()); assertTrue(this.receivedMsgs.get(2) instanceof Open); this.serverSession.handleMessage(this.kamsg); assertEquals(this.serverSession.getState(), DefaultPCEPSessionNegotiator.State.Finished); } /** * Sending different PCEP Message than Open in session establishment phase. * * @throws Exception */ @Test public void testErrorOneOne() throws Exception { this.serverSession.channelActive(null); assertEquals(1, this.receivedMsgs.size()); assertTrue(this.receivedMsgs.get(0) instanceof Open); this.serverSession.handleMessage(this.kamsg); for (final Notification m : this.receivedMsgs) { if (m instanceof Pcerr) { final Errors obj = ((Pcerr) m).getPcerrMessage().getErrors().get(0); assertEquals(new Short((short) 1), obj.getErrorObject().getType()); assertEquals(new Short((short) 1), obj.getErrorObject().getValue()); } } } /** * KeepWaitTimer expired. * * @throws Exception */ @Test public void testErrorOneSeven() throws Exception { this.serverSession.channelActive(null); assertEquals(1, this.receivedMsgs.size()); assertTrue(this.receivedMsgs.get(0) instanceof Open); this.serverSession.handleMessage(this.openmsg); Thread.sleep(1000); for (final Notification m : this.receivedMsgs) { if (m instanceof Pcerr) { final Errors obj = ((Pcerr) m).getPcerrMessage().getErrors().get(0); assertEquals(new Short((short) 1), obj.getErrorObject().getType()); assertEquals(new Short((short) 7), obj.getErrorObject().getValue()); } } } /************* Tests commented because of their long duration (tested timers) **************/ /** * OpenWait timer expired. * * @throws InterruptedException */ @Test @Ignore public void testErrorOneTwo() throws InterruptedException { this.serverSession.channelActive(null); assertEquals(1, this.receivedMsgs.size()); assertTrue(this.receivedMsgs.get(0) instanceof OpenMessage); Thread.sleep(60 * 1000); for (final Notification m : this.receivedMsgs) { if (m instanceof Pcerr) { final Errors obj = ((Pcerr) m).getPcerrMessage().getErrors().get(0); assertEquals(new Short((short) 1), obj.getErrorObject().getType()); assertEquals(new Short((short) 2), obj.getErrorObject().getValue()); } } } @Test @Ignore public void testUnknownMessage() throws InterruptedException { final SimpleSessionListener client = new SimpleSessionListener(); final PCEPSessionImpl s = new PCEPSessionImpl(client, 5, this.clientListener, this.openmsg.getOpenMessage().getOpen(), this.openmsg.getOpenMessage().getOpen()); s.handleMalformedMessage(PCEPErrors.CAPABILITY_NOT_SUPPORTED); assertEquals(1, s.getUnknownMessagesTimes().size()); Thread.sleep(10000); s.handleMalformedMessage(PCEPErrors.CAPABILITY_NOT_SUPPORTED); assertEquals(2, s.getUnknownMessagesTimes().size()); Thread.sleep(10000); s.handleMalformedMessage(PCEPErrors.CAPABILITY_NOT_SUPPORTED); assertEquals(3, s.getUnknownMessagesTimes().size()); Thread.sleep(20000); s.handleMalformedMessage(PCEPErrors.CAPABILITY_NOT_SUPPORTED); assertEquals(4, s.getUnknownMessagesTimes().size()); Thread.sleep(30000); s.handleMalformedMessage(PCEPErrors.CAPABILITY_NOT_SUPPORTED); assertEquals(3, s.getUnknownMessagesTimes().size()); Thread.sleep(10000); s.handleMalformedMessage(PCEPErrors.CAPABILITY_NOT_SUPPORTED); assertEquals(3, s.getUnknownMessagesTimes().size()); Thread.sleep(5000); s.handleMalformedMessage(PCEPErrors.CAPABILITY_NOT_SUPPORTED); assertEquals(4, s.getUnknownMessagesTimes().size()); Thread.sleep(1000); s.handleMalformedMessage(PCEPErrors.CAPABILITY_NOT_SUPPORTED); assertEquals(5, s.getUnknownMessagesTimes().size()); Thread.sleep(1000); s.handleMalformedMessage(PCEPErrors.CAPABILITY_NOT_SUPPORTED); synchronized (client) { while (client.up) { client.wait(); } } assertTrue(!client.up); } @After public void tearDown() { } }