blob: 36057275ab8357e8da7dc823cb4f492c10ac2002 [file] [log] [blame]
Charles Chan114aec72017-06-19 14:00:53 -07001/*
Brian O'Connor0947d7e2017-08-03 21:12:30 -07002 * Copyright 2017-present Open Networking Foundation
Charles Chan114aec72017-06-19 14:00:53 -07003 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17package org.onosproject.segmentrouting;
18
Charles Chan3ed34d82017-06-22 18:03:14 -070019import com.fasterxml.jackson.databind.JsonNode;
20import com.fasterxml.jackson.databind.ObjectMapper;
Charles Chan114aec72017-06-19 14:00:53 -070021import com.google.common.collect.Lists;
22import com.google.common.collect.Maps;
23import com.google.common.collect.Sets;
24import org.junit.Before;
25import org.junit.Test;
26import org.onlab.packet.IpAddress;
27import org.onlab.packet.IpPrefix;
28import org.onlab.packet.MacAddress;
29import org.onlab.packet.VlanId;
Charles Chan6c624992017-08-18 17:11:34 -070030import org.onosproject.net.config.ConfigApplyDelegate;
Charles Chan873661e2017-11-30 15:37:50 -080031import org.onosproject.net.host.HostLocationProbingService;
Ray Milkeyc2d43be2017-08-03 11:58:29 -070032import org.onosproject.net.intf.Interface;
Charles Chan114aec72017-06-19 14:00:53 -070033import org.onosproject.net.ConnectPoint;
34import org.onosproject.net.DefaultHost;
35import org.onosproject.net.DeviceId;
36import org.onosproject.net.Host;
37import org.onosproject.net.HostId;
38import org.onosproject.net.HostLocation;
39import org.onosproject.net.PortNumber;
40import org.onosproject.net.config.NetworkConfigRegistryAdapter;
Charles Chan114aec72017-06-19 14:00:53 -070041import org.onosproject.net.flow.TrafficTreatment;
Charles Chan114aec72017-06-19 14:00:53 -070042import org.onosproject.net.host.HostEvent;
43import org.onosproject.net.host.InterfaceIpAddress;
44import org.onosproject.net.provider.ProviderId;
45import org.onosproject.segmentrouting.config.DeviceConfiguration;
Charles Chan3ed34d82017-06-22 18:03:14 -070046import org.onosproject.segmentrouting.config.SegmentRoutingDeviceConfig;
Charles Chand66d6712018-03-29 16:03:41 -070047import org.onosproject.store.service.StorageService;
48import org.onosproject.store.service.TestConsistentMap;
Charles Chan114aec72017-06-19 14:00:53 -070049
50import java.util.Map;
Charles Chan114aec72017-06-19 14:00:53 -070051import java.util.Set;
Charles Chan114aec72017-06-19 14:00:53 -070052
Charles Chand66d6712018-03-29 16:03:41 -070053import static org.easymock.EasyMock.createMock;
54import static org.easymock.EasyMock.expect;
55import static org.easymock.EasyMock.replay;
Charles Chan114aec72017-06-19 14:00:53 -070056import static org.junit.Assert.*;
57
Charles Chan6c624992017-08-18 17:11:34 -070058/**r
Charles Chan114aec72017-06-19 14:00:53 -070059 * Unit test for {@link HostHandler}.
60 */
61public class HostHandlerTest {
Charles Chan114aec72017-06-19 14:00:53 -070062 private HostHandler hostHandler;
63
64 // Mocked routing and bridging tables
Charles Chan6c624992017-08-18 17:11:34 -070065 private static final Map<MockBridgingTableKey, MockBridgingTableValue> BRIDGING_TABLE =
66 Maps.newConcurrentMap();
67 private static final Map<MockRoutingTableKey, MockRoutingTableValue> ROUTING_TABLE =
68 Maps.newConcurrentMap();
Charles Chan910be6a2017-08-23 14:46:43 -070069 private static final Map<ConnectPoint, Set<IpPrefix>> SUBNET_TABLE = Maps.newConcurrentMap();
Charles Chan114aec72017-06-19 14:00:53 -070070 // Mocked Next Id
Charles Chan6c624992017-08-18 17:11:34 -070071 private static final Map<Integer, TrafficTreatment> NEXT_TABLE = Maps.newConcurrentMap();
Charles Chan114aec72017-06-19 14:00:53 -070072
Charles Chand9265a32017-06-16 15:19:24 -070073 // Host Mac, VLAN
Charles Chan114aec72017-06-19 14:00:53 -070074 private static final ProviderId PROVIDER_ID = ProviderId.NONE;
75 private static final MacAddress HOST_MAC = MacAddress.valueOf("00:00:00:00:00:01");
76 private static final VlanId HOST_VLAN_UNTAGGED = VlanId.NONE;
77 private static final HostId HOST_ID_UNTAGGED = HostId.hostId(HOST_MAC, HOST_VLAN_UNTAGGED);
78 private static final VlanId HOST_VLAN_TAGGED = VlanId.vlanId((short) 20);
79 private static final HostId HOST_ID_TAGGED = HostId.hostId(HOST_MAC, HOST_VLAN_TAGGED);
Charles Chand9265a32017-06-16 15:19:24 -070080 // Host IP
81 private static final IpAddress HOST_IP11 = IpAddress.valueOf("10.0.1.1");
82 private static final IpAddress HOST_IP21 = IpAddress.valueOf("10.0.2.1");
83 private static final IpAddress HOST_IP12 = IpAddress.valueOf("10.0.1.2");
84 private static final IpAddress HOST_IP13 = IpAddress.valueOf("10.0.1.3");
85 private static final IpAddress HOST_IP14 = IpAddress.valueOf("10.0.1.4");
Charles Chanceb2a2e2017-09-12 18:57:47 -070086 private static final IpAddress HOST_IP32 = IpAddress.valueOf("10.0.3.2");
Charles Chand9265a32017-06-16 15:19:24 -070087 // Device
88 private static final DeviceId DEV1 = DeviceId.deviceId("of:0000000000000001");
89 private static final DeviceId DEV2 = DeviceId.deviceId("of:0000000000000002");
Charles Chanabfe7e02017-08-09 16:50:15 -070090 private static final DeviceId DEV3 = DeviceId.deviceId("of:0000000000000003");
91 private static final DeviceId DEV4 = DeviceId.deviceId("of:0000000000000004");
92 private static final DeviceId DEV5 = DeviceId.deviceId("of:0000000000000005");
93 private static final DeviceId DEV6 = DeviceId.deviceId("of:0000000000000006");
Charles Chand9265a32017-06-16 15:19:24 -070094 // Port
95 private static final PortNumber P1 = PortNumber.portNumber(1);
96 private static final PortNumber P2 = PortNumber.portNumber(2);
97 private static final PortNumber P3 = PortNumber.portNumber(3);
Charles Chan3ed34d82017-06-22 18:03:14 -070098 private static final PortNumber P9 = PortNumber.portNumber(9);
Charles Chand9265a32017-06-16 15:19:24 -070099 // Connect Point
100 private static final ConnectPoint CP11 = new ConnectPoint(DEV1, P1);
101 private static final HostLocation HOST_LOC11 = new HostLocation(CP11, 0);
102 private static final ConnectPoint CP12 = new ConnectPoint(DEV1, P2);
103 private static final HostLocation HOST_LOC12 = new HostLocation(CP12, 0);
104 private static final ConnectPoint CP13 = new ConnectPoint(DEV1, P3);
105 private static final HostLocation HOST_LOC13 = new HostLocation(CP13, 0);
106 private static final ConnectPoint CP21 = new ConnectPoint(DEV2, P1);
107 private static final HostLocation HOST_LOC21 = new HostLocation(CP21, 0);
108 private static final ConnectPoint CP22 = new ConnectPoint(DEV2, P2);
109 private static final HostLocation HOST_LOC22 = new HostLocation(CP22, 0);
Charles Chan3ed34d82017-06-22 18:03:14 -0700110 // Connect Point for dual-homed host failover
111 private static final ConnectPoint CP31 = new ConnectPoint(DEV3, P1);
112 private static final HostLocation HOST_LOC31 = new HostLocation(CP31, 0);
Charles Chanceb2a2e2017-09-12 18:57:47 -0700113 private static final ConnectPoint CP32 = new ConnectPoint(DEV3, P2);
114 private static final HostLocation HOST_LOC32 = new HostLocation(CP32, 0);
Charles Chan3ed34d82017-06-22 18:03:14 -0700115 private static final ConnectPoint CP41 = new ConnectPoint(DEV4, P1);
116 private static final HostLocation HOST_LOC41 = new HostLocation(CP41, 0);
117 private static final ConnectPoint CP39 = new ConnectPoint(DEV3, P9);
118 private static final ConnectPoint CP49 = new ConnectPoint(DEV4, P9);
Charles Chanabfe7e02017-08-09 16:50:15 -0700119 // Conenct Point for mastership test
120 private static final ConnectPoint CP51 = new ConnectPoint(DEV5, P1);
121 private static final HostLocation HOST_LOC51 = new HostLocation(CP51, 0);
122 private static final ConnectPoint CP61 = new ConnectPoint(DEV6, P1);
123 private static final HostLocation HOST_LOC61 = new HostLocation(CP61, 0);
Charles Chand9265a32017-06-16 15:19:24 -0700124 // Interface VLAN
Charles Chan114aec72017-06-19 14:00:53 -0700125 private static final VlanId INTF_VLAN_UNTAGGED = VlanId.vlanId((short) 10);
Charles Chan114aec72017-06-19 14:00:53 -0700126 private static final Set<VlanId> INTF_VLAN_TAGGED = Sets.newHashSet(VlanId.vlanId((short) 20));
127 private static final VlanId INTF_VLAN_NATIVE = VlanId.vlanId((short) 30);
Charles Chan3ed34d82017-06-22 18:03:14 -0700128 private static final Set<VlanId> INTF_VLAN_PAIR = Sets.newHashSet(VlanId.vlanId((short) 10),
129 VlanId.vlanId((short) 20), VlanId.vlanId((short) 30));
Charles Chanceb2a2e2017-09-12 18:57:47 -0700130 private static final VlanId INTF_VLAN_OTHER = VlanId.vlanId((short) 40);
Charles Chand9265a32017-06-16 15:19:24 -0700131 // Interface subnet
132 private static final IpPrefix INTF_PREFIX1 = IpPrefix.valueOf("10.0.1.254/24");
133 private static final IpPrefix INTF_PREFIX2 = IpPrefix.valueOf("10.0.2.254/24");
Charles Chanceb2a2e2017-09-12 18:57:47 -0700134 private static final IpPrefix INTF_PREFIX3 = IpPrefix.valueOf("10.0.3.254/24");
Charles Chand9265a32017-06-16 15:19:24 -0700135 private static final InterfaceIpAddress INTF_IP1 =
136 new InterfaceIpAddress(INTF_PREFIX1.address(), INTF_PREFIX1);
137 private static final InterfaceIpAddress INTF_IP2 =
138 new InterfaceIpAddress(INTF_PREFIX2.address(), INTF_PREFIX2);
Charles Chanceb2a2e2017-09-12 18:57:47 -0700139 private static final InterfaceIpAddress INTF_IP3 =
140 new InterfaceIpAddress(INTF_PREFIX3.address(), INTF_PREFIX3);
Charles Chan6c624992017-08-18 17:11:34 -0700141 // Interfaces
142 private static final Interface INTF11 =
143 new Interface(null, CP11, Lists.newArrayList(INTF_IP1), MacAddress.NONE, null,
144 INTF_VLAN_UNTAGGED, null, null);
145 private static final Interface INTF12 =
146 new Interface(null, CP12, Lists.newArrayList(INTF_IP1), MacAddress.NONE, null,
147 INTF_VLAN_UNTAGGED, null, null);
148 private static final Interface INTF13 =
149 new Interface(null, CP13, Lists.newArrayList(INTF_IP2), MacAddress.NONE, null,
150 null, INTF_VLAN_TAGGED, INTF_VLAN_NATIVE);
151 private static final Interface INTF21 =
152 new Interface(null, CP21, Lists.newArrayList(INTF_IP1), MacAddress.NONE, null,
153 INTF_VLAN_UNTAGGED, null, null);
154 private static final Interface INTF22 =
155 new Interface(null, CP22, Lists.newArrayList(INTF_IP1), MacAddress.NONE, null,
156 INTF_VLAN_UNTAGGED, null, null);
157 private static final Interface INTF31 =
158 new Interface(null, CP31, Lists.newArrayList(INTF_IP1), MacAddress.NONE, null,
159 INTF_VLAN_UNTAGGED, null, null);
Charles Chanceb2a2e2017-09-12 18:57:47 -0700160 private static final Interface INTF32 =
161 new Interface(null, CP32, Lists.newArrayList(INTF_IP3), MacAddress.NONE, null,
162 INTF_VLAN_OTHER, null, null);
Charles Chan6c624992017-08-18 17:11:34 -0700163 private static final Interface INTF39 =
164 new Interface(null, CP39, Lists.newArrayList(INTF_IP1), MacAddress.NONE, null,
165 null, INTF_VLAN_PAIR, null);
166 private static final Interface INTF41 =
167 new Interface(null, CP41, Lists.newArrayList(INTF_IP1), MacAddress.NONE, null,
168 INTF_VLAN_UNTAGGED, null, null);
169 private static final Interface INTF49 =
170 new Interface(null, CP49, Lists.newArrayList(INTF_IP1), MacAddress.NONE, null,
171 null, INTF_VLAN_PAIR, null);
Charles Chand9265a32017-06-16 15:19:24 -0700172 // Host
173 private static final Host HOST1 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC,
174 HOST_VLAN_UNTAGGED, Sets.newHashSet(HOST_LOC11, HOST_LOC21), Sets.newHashSet(HOST_IP11),
175 false);
Charles Chan114aec72017-06-19 14:00:53 -0700176
Charles Chan6c624992017-08-18 17:11:34 -0700177 // A set of hosts
178 private static final Set<Host> HOSTS = Sets.newHashSet(HOST1);
179 // A set of devices of which we have mastership
180 private static final Set<DeviceId> LOCAL_DEVICES = Sets.newHashSet(DEV1, DEV2, DEV3, DEV4);
181 // A set of interfaces
182 private static final Set<Interface> INTERFACES = Sets.newHashSet(INTF11, INTF12, INTF13, INTF21,
Charles Chanceb2a2e2017-09-12 18:57:47 -0700183 INTF22, INTF31, INTF32, INTF39, INTF41, INTF49);
Charles Chan6c624992017-08-18 17:11:34 -0700184
Charles Chan873661e2017-11-30 15:37:50 -0800185 private MockLocationProbingService mockLocationProbingService;
186
Charles Chan114aec72017-06-19 14:00:53 -0700187 @Before
188 public void setUp() throws Exception {
Charles Chan6c624992017-08-18 17:11:34 -0700189 // Initialize pairDevice and pairLocalPort config
190 ObjectMapper mapper = new ObjectMapper();
191 ConfigApplyDelegate delegate = config -> { };
192
193 SegmentRoutingDeviceConfig dev3Config = new SegmentRoutingDeviceConfig();
194 JsonNode dev3Tree = mapper.createObjectNode();
195 dev3Config.init(DEV3, "host-handler-test", dev3Tree, mapper, delegate);
196 dev3Config.setPairDeviceId(DEV4).setPairLocalPort(P9);
197
198 SegmentRoutingDeviceConfig dev4Config = new SegmentRoutingDeviceConfig();
199 JsonNode dev4Tree = mapper.createObjectNode();
200 dev4Config.init(DEV4, "host-handler-test", dev4Tree, mapper, delegate);
201 dev4Config.setPairDeviceId(DEV3).setPairLocalPort(P9);
202
203 MockNetworkConfigRegistry mockNetworkConfigRegistry = new MockNetworkConfigRegistry();
204 mockNetworkConfigRegistry.applyConfig(dev3Config);
205 mockNetworkConfigRegistry.applyConfig(dev4Config);
206
207 // Initialize Segment Routing Manager
208 SegmentRoutingManager srManager = new MockSegmentRoutingManager(NEXT_TABLE);
Charles Chand66d6712018-03-29 16:03:41 -0700209 srManager.storageService = createMock(StorageService.class);
210 expect(srManager.storageService.consistentMapBuilder()).andReturn(new TestConsistentMap.Builder<>()).anyTimes();
211 replay(srManager.storageService);
Charles Chan114aec72017-06-19 14:00:53 -0700212 srManager.cfgService = new NetworkConfigRegistryAdapter();
213 srManager.deviceConfiguration = new DeviceConfiguration(srManager);
Charles Chan6c624992017-08-18 17:11:34 -0700214 srManager.flowObjectiveService = new MockFlowObjectiveService(BRIDGING_TABLE, NEXT_TABLE);
215 srManager.routingRulePopulator = new MockRoutingRulePopulator(srManager, ROUTING_TABLE);
Charles Chanc4d68882018-03-15 16:41:10 -0700216 srManager.defaultRoutingHandler = new MockDefaultRoutingHandler(srManager, SUBNET_TABLE, ROUTING_TABLE);
Charles Chan6c624992017-08-18 17:11:34 -0700217 srManager.interfaceService = new MockInterfaceService(INTERFACES);
218 srManager.mastershipService = new MockMastershipService(LOCAL_DEVICES);
219 srManager.hostService = new MockHostService(HOSTS);
220 srManager.cfgService = mockNetworkConfigRegistry;
Charles Chan873661e2017-11-30 15:37:50 -0800221 mockLocationProbingService = new MockLocationProbingService();
222 srManager.probingService = mockLocationProbingService;
Saurav Dase6c448a2018-01-18 12:07:33 -0800223 srManager.linkHandler = new MockLinkHandler(srManager);
Charles Chan114aec72017-06-19 14:00:53 -0700224
225 hostHandler = new HostHandler(srManager);
226
Charles Chan6c624992017-08-18 17:11:34 -0700227 ROUTING_TABLE.clear();
228 BRIDGING_TABLE.clear();
Charles Chan114aec72017-06-19 14:00:53 -0700229 }
230
231 @Test
232 public void init() throws Exception {
Charles Chand9265a32017-06-16 15:19:24 -0700233 hostHandler.init(DEV1);
Charles Chan6c624992017-08-18 17:11:34 -0700234 assertEquals(1, ROUTING_TABLE.size());
235 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP11.toIpPrefix())));
236 assertEquals(1, BRIDGING_TABLE.size());
237 assertNotNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_UNTAGGED)));
Charles Chand9265a32017-06-16 15:19:24 -0700238
239 hostHandler.init(DEV2);
Charles Chan6c624992017-08-18 17:11:34 -0700240 assertEquals(2, ROUTING_TABLE.size());
241 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP11.toIpPrefix())));
242 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV2, HOST_IP11.toIpPrefix())));
243 assertEquals(2, BRIDGING_TABLE.size());
244 assertNotNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_UNTAGGED)));
245 assertNotNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV2, HOST_MAC, INTF_VLAN_UNTAGGED)));
Charles Chand9265a32017-06-16 15:19:24 -0700246 }
247
248 @Test(expected = IllegalArgumentException.class)
249 public void testHostAddedAtWrongLocation() throws Exception {
250 hostHandler.processHostAddedAtLocation(HOST1, HOST_LOC13);
251 }
252
253
254 @Test()
255 public void testHostAddedAtCorrectLocation() throws Exception {
256 hostHandler.processHostAddedAtLocation(HOST1, HOST_LOC11);
Charles Chan6c624992017-08-18 17:11:34 -0700257 assertEquals(1, ROUTING_TABLE.size());
258 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP11.toIpPrefix())));
259 assertEquals(1, BRIDGING_TABLE.size());
260 assertNotNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_UNTAGGED)));
Charles Chan114aec72017-06-19 14:00:53 -0700261 }
262
263 @Test
264 public void testHostAdded() throws Exception {
265 Host subject;
266
267 // Untagged host discovered on untagged port
268 // Expect: add one routing rule and one bridging rule
269 subject = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
Charles Chand9265a32017-06-16 15:19:24 -0700270 Sets.newHashSet(HOST_LOC11), Sets.newHashSet(HOST_IP11), false);
Charles Chan114aec72017-06-19 14:00:53 -0700271 hostHandler.processHostAddedEvent(new HostEvent(HostEvent.Type.HOST_ADDED, subject));
Charles Chan6c624992017-08-18 17:11:34 -0700272 assertEquals(1, ROUTING_TABLE.size());
273 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP11.toIpPrefix())));
274 assertEquals(1, BRIDGING_TABLE.size());
275 assertNotNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_UNTAGGED)));
Charles Chan114aec72017-06-19 14:00:53 -0700276
277 // Untagged host discovered on tagged/native port
278 // Expect: add one routing rule and one bridging rule
279 subject = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
Charles Chand9265a32017-06-16 15:19:24 -0700280 Sets.newHashSet(HOST_LOC13), Sets.newHashSet(HOST_IP21), false);
Charles Chan114aec72017-06-19 14:00:53 -0700281 hostHandler.processHostAddedEvent(new HostEvent(HostEvent.Type.HOST_ADDED, subject));
Charles Chan6c624992017-08-18 17:11:34 -0700282 assertEquals(2, ROUTING_TABLE.size());
283 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP21.toIpPrefix())));
284 assertEquals(2, BRIDGING_TABLE.size());
285 assertNotNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_NATIVE)));
Charles Chan114aec72017-06-19 14:00:53 -0700286
287 // Tagged host discovered on untagged port
288 // Expect: ignore the host. No rule is added.
289 subject = new DefaultHost(PROVIDER_ID, HOST_ID_TAGGED, HOST_MAC, HOST_VLAN_TAGGED,
Charles Chand9265a32017-06-16 15:19:24 -0700290 Sets.newHashSet(HOST_LOC11), Sets.newHashSet(HOST_IP11), false);
Charles Chan114aec72017-06-19 14:00:53 -0700291 hostHandler.processHostAddedEvent(new HostEvent(HostEvent.Type.HOST_ADDED, subject));
Charles Chan6c624992017-08-18 17:11:34 -0700292 assertEquals(2, ROUTING_TABLE.size());
293 assertEquals(2, BRIDGING_TABLE.size());
Charles Chan114aec72017-06-19 14:00:53 -0700294
295 // Tagged host discovered on tagged port with the same IP
296 // Expect: update existing route, add one bridging rule
297 subject = new DefaultHost(PROVIDER_ID, HOST_ID_TAGGED, HOST_MAC, HOST_VLAN_TAGGED,
Charles Chand9265a32017-06-16 15:19:24 -0700298 Sets.newHashSet(HOST_LOC13), Sets.newHashSet(HOST_IP21), false);
Charles Chan114aec72017-06-19 14:00:53 -0700299 hostHandler.processHostAddedEvent(new HostEvent(HostEvent.Type.HOST_ADDED, subject));
Charles Chan6c624992017-08-18 17:11:34 -0700300 assertEquals(2, ROUTING_TABLE.size());
301 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP21.toIpPrefix())));
302 assertEquals(HOST_VLAN_TAGGED, ROUTING_TABLE.get(new MockRoutingTableKey(HOST_LOC13.deviceId(),
Charles Chand9265a32017-06-16 15:19:24 -0700303 HOST_IP21.toIpPrefix())).vlanId);
Charles Chan6c624992017-08-18 17:11:34 -0700304 assertEquals(3, BRIDGING_TABLE.size());
305 assertNotNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, HOST_VLAN_TAGGED)));
Charles Chand9265a32017-06-16 15:19:24 -0700306 }
307
308 @Test
309 public void testDualHomedHostAdded() throws Exception {
310 // Add a dual-homed host that has 2 locations
311 // Expect: add two routing rules and two bridging rules
312 Host subject = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
313 Sets.newHashSet(HOST_LOC11, HOST_LOC21), Sets.newHashSet(HOST_IP11), false);
314 hostHandler.processHostAddedEvent(new HostEvent(HostEvent.Type.HOST_ADDED, subject));
Charles Chan6c624992017-08-18 17:11:34 -0700315 assertEquals(2, ROUTING_TABLE.size());
316 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP11.toIpPrefix())));
317 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV2, HOST_IP11.toIpPrefix())));
318 assertEquals(2, BRIDGING_TABLE.size());
319 assertNotNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_UNTAGGED)));
320 assertNotNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV2, HOST_MAC, INTF_VLAN_UNTAGGED)));
Charles Chan114aec72017-06-19 14:00:53 -0700321 }
322
323 @Test
Charles Chanceb2a2e2017-09-12 18:57:47 -0700324 public void testSingleHomedHostAddedOnPairLeaf() throws Exception {
325 Host host1 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
326 Sets.newHashSet(HOST_LOC32), Sets.newHashSet(HOST_IP32), false);
327
328 // Add a single-homed host with one location
329 // Expect: the pair link should not be utilized
330 hostHandler.processHostAddedEvent(new HostEvent(HostEvent.Type.HOST_ADDED, host1));
331 assertEquals(1, ROUTING_TABLE.size());
332 assertEquals(P2, ROUTING_TABLE.get(new MockRoutingTableKey(DEV3, HOST_IP32.toIpPrefix())).portNumber);
333 assertEquals(1, BRIDGING_TABLE.size());
334 assertEquals(P2, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV3, HOST_MAC, INTF_VLAN_OTHER)).portNumber);
335 }
336
337 @Test
338 public void testDualHomedHostAddedOneByOne() throws Exception {
339 Host host1 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
340 Sets.newHashSet(HOST_LOC31), Sets.newHashSet(HOST_IP11), false);
341 Host host2 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
342 Sets.newHashSet(HOST_LOC31, HOST_LOC41), Sets.newHashSet(HOST_IP11), false);
343
344 // Add a dual-homed host with one location
345 // Expect: the pair link is utilized temporarily before the second location is discovered
346 hostHandler.processHostAddedEvent(new HostEvent(HostEvent.Type.HOST_ADDED, host1));
347 assertEquals(2, ROUTING_TABLE.size());
348 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV3, HOST_IP11.toIpPrefix())).portNumber);
349 assertEquals(P9, ROUTING_TABLE.get(new MockRoutingTableKey(DEV4, HOST_IP11.toIpPrefix())).portNumber);
350 assertEquals(2, BRIDGING_TABLE.size());
351 assertEquals(P1, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV3, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
352 assertEquals(P9, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV4, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
Charles Chan873661e2017-11-30 15:37:50 -0800353 // Expect probe to be sent out on pair device
354 assertTrue(mockLocationProbingService.verifyProbe(host1, CP41, HostLocationProbingService.ProbeMode.DISCOVER));
Charles Chanceb2a2e2017-09-12 18:57:47 -0700355
356 // Add the second location of dual-homed host
357 // Expect: no longer use the pair link
358 hostHandler.processHostMovedEvent(new HostEvent(HostEvent.Type.HOST_MOVED, host2, host1));
359 assertEquals(2, ROUTING_TABLE.size());
360 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV3, HOST_IP11.toIpPrefix())).portNumber);
Charles Chan873661e2017-11-30 15:37:50 -0800361 assertEquals(P9, ROUTING_TABLE.get(new MockRoutingTableKey(DEV4, HOST_IP11.toIpPrefix())).portNumber);
Charles Chanceb2a2e2017-09-12 18:57:47 -0700362 assertEquals(2, BRIDGING_TABLE.size());
363 assertEquals(P1, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV3, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
Charles Chan873661e2017-11-30 15:37:50 -0800364 assertEquals(P9, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV4, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
365 // FIXME: Delay event handling a little bit to wait for the previous redirection flows to be completed
366 // The permanent solution would be introducing CompletableFuture and wait for it
367 Thread.sleep(HostHandler.HOST_MOVED_DELAY_MS + 50);
368 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV4, HOST_IP11.toIpPrefix())).portNumber);
Charles Chanceb2a2e2017-09-12 18:57:47 -0700369 assertEquals(P1, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV4, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
370 }
371
372 @Test
Charles Chan114aec72017-06-19 14:00:53 -0700373 public void testHostRemoved() throws Exception {
374 Host subject = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
Charles Chand9265a32017-06-16 15:19:24 -0700375 Sets.newHashSet(HOST_LOC11), Sets.newHashSet(HOST_IP11), false);
Charles Chan114aec72017-06-19 14:00:53 -0700376
377 // Add a host
378 // Expect: add one routing rule and one bridging rule
379 hostHandler.processHostAddedEvent(new HostEvent(HostEvent.Type.HOST_ADDED, subject));
Charles Chan6c624992017-08-18 17:11:34 -0700380 assertEquals(1, ROUTING_TABLE.size());
381 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP11.toIpPrefix())));
382 assertEquals(1, BRIDGING_TABLE.size());
383 assertNotNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_UNTAGGED)));
Charles Chan114aec72017-06-19 14:00:53 -0700384
385 // Remove the host
386 // Expect: add the routing rule and the bridging rule
Charles Chand9265a32017-06-16 15:19:24 -0700387 hostHandler.processHostRemovedEvent(new HostEvent(HostEvent.Type.HOST_REMOVED, subject));
Charles Chan6c624992017-08-18 17:11:34 -0700388 assertEquals(0, ROUTING_TABLE.size());
389 assertEquals(0, BRIDGING_TABLE.size());
Charles Chand9265a32017-06-16 15:19:24 -0700390 }
391
392 @Test
393 public void testDualHomedHostRemoved() throws Exception {
394 // Add a dual-homed host that has 2 locations
395 // Expect: add two routing rules and two bridging rules
396 Host subject = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
397 Sets.newHashSet(HOST_LOC11, HOST_LOC21), Sets.newHashSet(HOST_IP11), false);
398 hostHandler.processHostAddedEvent(new HostEvent(HostEvent.Type.HOST_ADDED, subject));
Charles Chan6c624992017-08-18 17:11:34 -0700399 assertEquals(2, ROUTING_TABLE.size());
400 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP11.toIpPrefix())));
401 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV2, HOST_IP11.toIpPrefix())));
402 assertEquals(2, BRIDGING_TABLE.size());
403 assertNotNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_UNTAGGED)));
404 assertNotNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV2, HOST_MAC, INTF_VLAN_UNTAGGED)));
Charles Chand9265a32017-06-16 15:19:24 -0700405
406 // Remove a dual-homed host that has 2 locations
407 // Expect: all routing and bridging rules are removed
408 hostHandler.processHostRemovedEvent(new HostEvent(HostEvent.Type.HOST_REMOVED, subject));
Charles Chan6c624992017-08-18 17:11:34 -0700409 assertEquals(0, ROUTING_TABLE.size());
410 assertEquals(0, BRIDGING_TABLE.size());
Charles Chan114aec72017-06-19 14:00:53 -0700411 }
412
413 @Test
414 public void testHostMoved() throws Exception {
415 Host host1 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
Charles Chand9265a32017-06-16 15:19:24 -0700416 Sets.newHashSet(HOST_LOC11), Sets.newHashSet(HOST_IP11), false);
Charles Chan114aec72017-06-19 14:00:53 -0700417 Host host2 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
Charles Chand9265a32017-06-16 15:19:24 -0700418 Sets.newHashSet(HOST_LOC21), Sets.newHashSet(HOST_IP11), false);
Charles Chan114aec72017-06-19 14:00:53 -0700419 Host host3 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
Charles Chand9265a32017-06-16 15:19:24 -0700420 Sets.newHashSet(HOST_LOC13), Sets.newHashSet(HOST_IP11), false);
Charles Chan114aec72017-06-19 14:00:53 -0700421
422 // Add a host
Charles Chan3ed34d82017-06-22 18:03:14 -0700423 // Expect: add one new routing rule, one new bridging rule
Charles Chan114aec72017-06-19 14:00:53 -0700424 hostHandler.processHostAddedEvent(new HostEvent(HostEvent.Type.HOST_ADDED, host1));
Charles Chan6c624992017-08-18 17:11:34 -0700425 assertEquals(1, ROUTING_TABLE.size());
426 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP11.toIpPrefix())));
427 assertNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP21.toIpPrefix())));
428 assertNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP13.toIpPrefix())));
429 assertEquals(1, BRIDGING_TABLE.size());
430 assertNotNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_UNTAGGED)));
431 assertNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV2, HOST_MAC, INTF_VLAN_UNTAGGED)));
Charles Chan114aec72017-06-19 14:00:53 -0700432
Charles Chand9265a32017-06-16 15:19:24 -0700433 // Move the host to CP13, which has different subnet setting
Charles Chan114aec72017-06-19 14:00:53 -0700434 // Expect: remove routing rule. Change vlan in bridging rule.
Charles Chand9265a32017-06-16 15:19:24 -0700435 hostHandler.processHostMovedEvent(new HostEvent(HostEvent.Type.HOST_MOVED, host3, host1));
Charles Chan6c624992017-08-18 17:11:34 -0700436 assertEquals(0, ROUTING_TABLE.size());
437 assertEquals(1, BRIDGING_TABLE.size());
438 assertNotNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_NATIVE)));
439 assertNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_UNTAGGED)));
Charles Chan114aec72017-06-19 14:00:53 -0700440
Charles Chand9265a32017-06-16 15:19:24 -0700441 // Move the host to CP21, which has same subnet setting
Charles Chan114aec72017-06-19 14:00:53 -0700442 // Expect: add a new routing rule. Change vlan in bridging rule.
Charles Chand9265a32017-06-16 15:19:24 -0700443 hostHandler.processHostMovedEvent(new HostEvent(HostEvent.Type.HOST_MOVED, host2, host3));
Charles Chan6c624992017-08-18 17:11:34 -0700444 assertEquals(1, ROUTING_TABLE.size());
445 assertNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP11.toIpPrefix())));
446 assertNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP11.toIpPrefix())));
447 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV2, HOST_IP11.toIpPrefix())));
448 assertEquals(1, BRIDGING_TABLE.size());
449 assertNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_UNTAGGED)));
450 assertNotNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV2, HOST_MAC, INTF_VLAN_UNTAGGED)));
Charles Chand9265a32017-06-16 15:19:24 -0700451 }
452
453 @Test
454 public void testDualHomedHostMoved() throws Exception {
455 Host host1 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
456 Sets.newHashSet(HOST_LOC11, HOST_LOC21), Sets.newHashSet(HOST_IP11, HOST_IP12), false);
457 Host host2 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
458 Sets.newHashSet(HOST_LOC12, HOST_LOC22), Sets.newHashSet(HOST_IP11, HOST_IP12), false);
459 Host host3 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
460 Sets.newHashSet(HOST_LOC11, HOST_LOC21), Sets.newHashSet(HOST_IP13, HOST_IP14), false);
461 Host host4 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
462 Sets.newHashSet(HOST_LOC11, HOST_LOC22), Sets.newHashSet(HOST_IP12, HOST_IP13), false);
463
464 // Add a host with IP11, IP12 and LOC11, LOC21
465 // Expect: 4 routing rules and 2 bridging rules
466 hostHandler.processHostAddedEvent(new HostEvent(HostEvent.Type.HOST_ADDED, host1));
Charles Chan6c624992017-08-18 17:11:34 -0700467 assertEquals(4, ROUTING_TABLE.size());
468 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP11.toIpPrefix())).portNumber);
469 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP12.toIpPrefix())).portNumber);
470 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV2, HOST_IP11.toIpPrefix())).portNumber);
471 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV2, HOST_IP12.toIpPrefix())).portNumber);
472 assertEquals(2, BRIDGING_TABLE.size());
473 assertEquals(P1, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
474 assertEquals(P1, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV2, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
Charles Chand9265a32017-06-16 15:19:24 -0700475
476 // Move the host to LOC12, LOC22 and keep the IP
477 // Expect: 4 routing rules and 2 bridging rules all at the new location
478 hostHandler.processHostMovedEvent(new HostEvent(HostEvent.Type.HOST_MOVED, host2, host1));
Charles Chan6c624992017-08-18 17:11:34 -0700479 assertEquals(4, ROUTING_TABLE.size());
480 assertEquals(P2, ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP11.toIpPrefix())).portNumber);
481 assertEquals(P2, ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP12.toIpPrefix())).portNumber);
482 assertEquals(P2, ROUTING_TABLE.get(new MockRoutingTableKey(DEV2, HOST_IP11.toIpPrefix())).portNumber);
483 assertEquals(P2, ROUTING_TABLE.get(new MockRoutingTableKey(DEV2, HOST_IP12.toIpPrefix())).portNumber);
484 assertEquals(2, BRIDGING_TABLE.size());
485 assertEquals(P2, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
486 assertEquals(P2, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV2, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
Charles Chand9265a32017-06-16 15:19:24 -0700487
488 // Move the host to LOC11, LOC21 and change the IP to IP13, IP14 at the same time
489 // Expect: 4 routing rules and 2 bridging rules all at the new location
490 hostHandler.processHostMovedEvent(new HostEvent(HostEvent.Type.HOST_MOVED, host3, host2));
Charles Chan6c624992017-08-18 17:11:34 -0700491 assertEquals(4, ROUTING_TABLE.size());
492 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP13.toIpPrefix())).portNumber);
493 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP14.toIpPrefix())).portNumber);
494 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV2, HOST_IP13.toIpPrefix())).portNumber);
495 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV2, HOST_IP14.toIpPrefix())).portNumber);
496 assertEquals(2, BRIDGING_TABLE.size());
497 assertEquals(P1, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
498 assertEquals(P1, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV2, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
Charles Chand9265a32017-06-16 15:19:24 -0700499
500 // Move the host to LOC11, LOC22 and change the IP to IP12, IP13 at the same time
501 // Expect: 4 routing rules and 2 bridging rules all at the new location
502 hostHandler.processHostMovedEvent(new HostEvent(HostEvent.Type.HOST_MOVED, host4, host3));
Charles Chan6c624992017-08-18 17:11:34 -0700503 assertEquals(4, ROUTING_TABLE.size());
504 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP12.toIpPrefix())).portNumber);
505 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP13.toIpPrefix())).portNumber);
506 assertEquals(P2, ROUTING_TABLE.get(new MockRoutingTableKey(DEV2, HOST_IP12.toIpPrefix())).portNumber);
507 assertEquals(P2, ROUTING_TABLE.get(new MockRoutingTableKey(DEV2, HOST_IP13.toIpPrefix())).portNumber);
508 assertEquals(2, BRIDGING_TABLE.size());
509 assertEquals(P1, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
510 assertEquals(P2, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV2, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
Charles Chan114aec72017-06-19 14:00:53 -0700511 }
512
513 @Test
Charles Chanabfe7e02017-08-09 16:50:15 -0700514 public void testHostMoveToInvalidLocation() throws Exception {
515 Host host1 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
516 Sets.newHashSet(HOST_LOC11), Sets.newHashSet(HOST_IP11), false);
517 Host host2 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
518 Sets.newHashSet(HOST_LOC51), Sets.newHashSet(HOST_IP11), false);
519
520 // Add a host
521 // Expect: add one new routing rule, one new bridging rule
522 hostHandler.processHostAddedEvent(new HostEvent(HostEvent.Type.HOST_ADDED, host1));
Charles Chan6c624992017-08-18 17:11:34 -0700523 assertEquals(1, ROUTING_TABLE.size());
524 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP11.toIpPrefix())));
525 assertNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP21.toIpPrefix())));
526 assertEquals(1, BRIDGING_TABLE.size());
527 assertNotNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_UNTAGGED)));
528 assertNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV2, HOST_MAC, INTF_VLAN_UNTAGGED)));
Charles Chanabfe7e02017-08-09 16:50:15 -0700529
530 // Move the host to an invalid location
531 // Expect: Old flow is removed. New flow is not created
532 hostHandler.processHostMovedEvent(new HostEvent(HostEvent.Type.HOST_MOVED, host2, host1));
Charles Chan6c624992017-08-18 17:11:34 -0700533 assertEquals(0, ROUTING_TABLE.size());
534 assertEquals(0, BRIDGING_TABLE.size());
Charles Chanabfe7e02017-08-09 16:50:15 -0700535
536 // Move the host to a valid location
537 // Expect: add one new routing rule, one new bridging rule
538 hostHandler.processHostMovedEvent(new HostEvent(HostEvent.Type.HOST_MOVED, host1, host2));
Charles Chan6c624992017-08-18 17:11:34 -0700539 assertEquals(1, ROUTING_TABLE.size());
540 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP11.toIpPrefix())));
541 assertNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP21.toIpPrefix())));
542 assertEquals(1, BRIDGING_TABLE.size());
543 assertNotNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_UNTAGGED)));
544 assertNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV2, HOST_MAC, INTF_VLAN_UNTAGGED)));
Charles Chanabfe7e02017-08-09 16:50:15 -0700545 }
546
547 @Test
548 public void testDualHomedHostMoveToInvalidLocation() throws Exception {
549 Host host1 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
550 Sets.newHashSet(HOST_LOC11, HOST_LOC21), Sets.newHashSet(HOST_IP11), false);
551 Host host2 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
552 Sets.newHashSet(HOST_LOC11, HOST_LOC51), Sets.newHashSet(HOST_IP11), false);
553 Host host3 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
554 Sets.newHashSet(HOST_LOC61, HOST_LOC51), Sets.newHashSet(HOST_IP11), false);
555
556 // Add a host
557 // Expect: add two new routing rules, two new bridging rules
558 hostHandler.processHostAddedEvent(new HostEvent(HostEvent.Type.HOST_ADDED, host1));
Charles Chan6c624992017-08-18 17:11:34 -0700559 assertEquals(2, ROUTING_TABLE.size());
560 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP11.toIpPrefix())).portNumber);
561 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV2, HOST_IP11.toIpPrefix())).portNumber);
562 assertEquals(2, BRIDGING_TABLE.size());
563 assertEquals(P1, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
564 assertEquals(P1, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV2, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
Charles Chanabfe7e02017-08-09 16:50:15 -0700565
566 // Move first host location to an invalid location
567 // Expect: One routing and one bridging flow
568 hostHandler.processHostMovedEvent(new HostEvent(HostEvent.Type.HOST_MOVED, host2, host1));
Charles Chan6c624992017-08-18 17:11:34 -0700569 assertEquals(1, ROUTING_TABLE.size());
570 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP11.toIpPrefix())).portNumber);
571 assertEquals(1, BRIDGING_TABLE.size());
572 assertEquals(P1, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
Charles Chanabfe7e02017-08-09 16:50:15 -0700573
574 // Move second host location to an invalid location
575 // Expect: No routing or bridging rule
576 hostHandler.processHostMovedEvent(new HostEvent(HostEvent.Type.HOST_MOVED, host3, host2));
Charles Chan6c624992017-08-18 17:11:34 -0700577 assertEquals(0, ROUTING_TABLE.size());
578 assertEquals(0, BRIDGING_TABLE.size());
Charles Chanabfe7e02017-08-09 16:50:15 -0700579
580 // Move second host location back to a valid location
581 // Expect: One routing and one bridging flow
582 hostHandler.processHostMovedEvent(new HostEvent(HostEvent.Type.HOST_MOVED, host2, host3));
Charles Chan6c624992017-08-18 17:11:34 -0700583 assertEquals(1, ROUTING_TABLE.size());
584 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP11.toIpPrefix())).portNumber);
585 assertEquals(1, BRIDGING_TABLE.size());
586 assertEquals(P1, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
Charles Chanabfe7e02017-08-09 16:50:15 -0700587
588 // Move first host location back to a valid location
589 // Expect: Two routing and two bridging flow
590 hostHandler.processHostMovedEvent(new HostEvent(HostEvent.Type.HOST_MOVED, host1, host2));
Charles Chan6c624992017-08-18 17:11:34 -0700591 assertEquals(2, ROUTING_TABLE.size());
592 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP11.toIpPrefix())).portNumber);
593 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV2, HOST_IP11.toIpPrefix())).portNumber);
594 assertEquals(2, BRIDGING_TABLE.size());
595 assertEquals(P1, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
596 assertEquals(P1, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV2, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
Charles Chanabfe7e02017-08-09 16:50:15 -0700597 }
598
599 @Test
Charles Chan3ed34d82017-06-22 18:03:14 -0700600 public void testDualHomingSingleLocationFail() throws Exception {
601 Host host1 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
602 Sets.newHashSet(HOST_LOC31, HOST_LOC41), Sets.newHashSet(HOST_IP11, HOST_IP12), false);
603 Host host2 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
604 Sets.newHashSet(HOST_LOC31), Sets.newHashSet(HOST_IP11, HOST_IP12), false);
605
606 // Add a host
607 // Expect: add four new routing rules, two new bridging rules
608 hostHandler.processHostAddedEvent(new HostEvent(HostEvent.Type.HOST_ADDED, host1));
Charles Chan6c624992017-08-18 17:11:34 -0700609 assertEquals(4, ROUTING_TABLE.size());
610 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV3, HOST_IP11.toIpPrefix())).portNumber);
611 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV3, HOST_IP12.toIpPrefix())).portNumber);
612 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV4, HOST_IP11.toIpPrefix())).portNumber);
613 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV4, HOST_IP12.toIpPrefix())).portNumber);
614 assertEquals(2, BRIDGING_TABLE.size());
615 assertEquals(P1, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV3, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
616 assertEquals(P1, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV4, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
Charles Chan3ed34d82017-06-22 18:03:14 -0700617
618 // Host becomes single-homed
619 // Expect: redirect flows from host location to pair link
620 hostHandler.processHostMovedEvent(new HostEvent(HostEvent.Type.HOST_MOVED, host2, host1));
Charles Chan6c624992017-08-18 17:11:34 -0700621 assertEquals(4, ROUTING_TABLE.size());
622 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV3, HOST_IP11.toIpPrefix())).portNumber);
623 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV3, HOST_IP12.toIpPrefix())).portNumber);
624 assertEquals(P9, ROUTING_TABLE.get(new MockRoutingTableKey(DEV4, HOST_IP11.toIpPrefix())).portNumber);
625 assertEquals(P9, ROUTING_TABLE.get(new MockRoutingTableKey(DEV4, HOST_IP12.toIpPrefix())).portNumber);
626 assertEquals(2, BRIDGING_TABLE.size());
627 assertEquals(P1, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV3, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
628 assertEquals(P9, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV4, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
Charles Chan3ed34d82017-06-22 18:03:14 -0700629
630 // Host becomes dual-homed again
631 // Expect: Redirect flows from pair link back to host location
632 hostHandler.processHostMovedEvent(new HostEvent(HostEvent.Type.HOST_MOVED, host1, host2));
Charles Chan6c624992017-08-18 17:11:34 -0700633 assertEquals(4, ROUTING_TABLE.size());
634 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV3, HOST_IP11.toIpPrefix())).portNumber);
635 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV3, HOST_IP12.toIpPrefix())).portNumber);
Charles Chan873661e2017-11-30 15:37:50 -0800636 assertEquals(P9, ROUTING_TABLE.get(new MockRoutingTableKey(DEV4, HOST_IP11.toIpPrefix())).portNumber);
637 assertEquals(P9, ROUTING_TABLE.get(new MockRoutingTableKey(DEV4, HOST_IP12.toIpPrefix())).portNumber);
Charles Chan6c624992017-08-18 17:11:34 -0700638 assertEquals(2, BRIDGING_TABLE.size());
639 assertEquals(P1, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV3, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
Charles Chan873661e2017-11-30 15:37:50 -0800640 assertEquals(P9, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV4, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
641 // FIXME: Delay event handling a little bit to wait for the previous redirection flows to be completed
642 // The permanent solution would be introducing CompletableFuture and wait for it
643 Thread.sleep(HostHandler.HOST_MOVED_DELAY_MS + 50);
644 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV4, HOST_IP11.toIpPrefix())).portNumber);
645 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV4, HOST_IP12.toIpPrefix())).portNumber);
Charles Chan6c624992017-08-18 17:11:34 -0700646 assertEquals(P1, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV4, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
Charles Chan3ed34d82017-06-22 18:03:14 -0700647 }
648
649 @Test
650 public void testDualHomingBothLocationFail() throws Exception {
651 Host host1 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
652 Sets.newHashSet(HOST_LOC31, HOST_LOC41), Sets.newHashSet(HOST_IP11, HOST_IP12), false);
653 Host host2 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
654 Sets.newHashSet(HOST_LOC31), Sets.newHashSet(HOST_IP11, HOST_IP12), false);
655
656 // Add a host
657 // Expect: add four new routing rules, two new bridging rules
658 hostHandler.processHostAddedEvent(new HostEvent(HostEvent.Type.HOST_ADDED, host1));
Charles Chan6c624992017-08-18 17:11:34 -0700659 assertEquals(4, ROUTING_TABLE.size());
660 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV3, HOST_IP11.toIpPrefix())).portNumber);
661 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV3, HOST_IP12.toIpPrefix())).portNumber);
662 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV4, HOST_IP11.toIpPrefix())).portNumber);
663 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV4, HOST_IP12.toIpPrefix())).portNumber);
664 assertEquals(2, BRIDGING_TABLE.size());
665 assertEquals(P1, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV3, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
666 assertEquals(P1, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV4, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
Charles Chan3ed34d82017-06-22 18:03:14 -0700667
668 // Host becomes single-homed
669 // Expect: redirect flows from host location to pair link
670 hostHandler.processHostMovedEvent(new HostEvent(HostEvent.Type.HOST_MOVED, host2, host1));
Charles Chan6c624992017-08-18 17:11:34 -0700671 assertEquals(4, ROUTING_TABLE.size());
672 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV3, HOST_IP11.toIpPrefix())).portNumber);
673 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV3, HOST_IP12.toIpPrefix())).portNumber);
674 assertEquals(P9, ROUTING_TABLE.get(new MockRoutingTableKey(DEV4, HOST_IP11.toIpPrefix())).portNumber);
675 assertEquals(P9, ROUTING_TABLE.get(new MockRoutingTableKey(DEV4, HOST_IP12.toIpPrefix())).portNumber);
676 assertEquals(2, BRIDGING_TABLE.size());
677 assertEquals(P1, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV3, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
678 assertEquals(P9, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV4, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
Charles Chan3ed34d82017-06-22 18:03:14 -0700679
680 // Host loses both locations
681 // Expect: Remove last location and all previous redirection flows
682 hostHandler.processHostRemovedEvent(new HostEvent(HostEvent.Type.HOST_REMOVED, host2));
Charles Chan6c624992017-08-18 17:11:34 -0700683 assertEquals(0, ROUTING_TABLE.size());
684 assertEquals(0, BRIDGING_TABLE.size());
Charles Chan3ed34d82017-06-22 18:03:14 -0700685 }
686
687 @Test
Charles Chan114aec72017-06-19 14:00:53 -0700688 public void testHostUpdated() throws Exception {
689 Host host1 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
Charles Chand9265a32017-06-16 15:19:24 -0700690 Sets.newHashSet(HOST_LOC11), Sets.newHashSet(HOST_IP11), false);
Charles Chan114aec72017-06-19 14:00:53 -0700691 Host host2 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
Charles Chand9265a32017-06-16 15:19:24 -0700692 Sets.newHashSet(HOST_LOC11), Sets.newHashSet(HOST_IP21), false);
Charles Chan114aec72017-06-19 14:00:53 -0700693 Host host3 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
Charles Chand9265a32017-06-16 15:19:24 -0700694 Sets.newHashSet(HOST_LOC11), Sets.newHashSet(HOST_IP12), false);
Charles Chan114aec72017-06-19 14:00:53 -0700695
696 // Add a host
Charles Chand9265a32017-06-16 15:19:24 -0700697 // Expect: add one new routing rule. Add one new bridging rule.
Charles Chan114aec72017-06-19 14:00:53 -0700698 hostHandler.processHostAddedEvent(new HostEvent(HostEvent.Type.HOST_ADDED, host1));
Charles Chan6c624992017-08-18 17:11:34 -0700699 assertEquals(1, ROUTING_TABLE.size());
700 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP11.toIpPrefix())));
701 assertNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP21.toIpPrefix())));
702 assertNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP12.toIpPrefix())));
703 assertEquals(1, BRIDGING_TABLE.size());
704 assertNotNull(BRIDGING_TABLE.get(new MockBridgingTableKey(HOST_LOC11.deviceId(), HOST_MAC,
705 INTF_VLAN_UNTAGGED)));
Charles Chan114aec72017-06-19 14:00:53 -0700706
707 // Update the host IP to same subnet
708 // Expect: update routing rule with new IP. No change to bridging rule.
709 hostHandler.processHostUpdatedEvent(new HostEvent(HostEvent.Type.HOST_UPDATED, host3, host1));
Charles Chan6c624992017-08-18 17:11:34 -0700710 assertEquals(1, ROUTING_TABLE.size());
711 assertNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP11.toIpPrefix())));
712 assertNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP21.toIpPrefix())));
713 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP12.toIpPrefix())));
714 assertEquals(1, BRIDGING_TABLE.size());
715 assertNotNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_UNTAGGED)));
Charles Chan114aec72017-06-19 14:00:53 -0700716
717 // Update the host IP to different subnet
718 // Expect: Remove routing rule. No change to bridging rule.
719 hostHandler.processHostUpdatedEvent(new HostEvent(HostEvent.Type.HOST_UPDATED, host2, host3));
Charles Chan6c624992017-08-18 17:11:34 -0700720 assertEquals(0, ROUTING_TABLE.size());
721 assertEquals(1, BRIDGING_TABLE.size());
722 assertNotNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_UNTAGGED)));
Charles Chand9265a32017-06-16 15:19:24 -0700723 }
724
725 @Test
Charles Chanb3c1faf2017-11-20 08:46:24 -0800726 public void testDelayedIpAndLocation() throws Exception {
727 Host host1 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
728 Sets.newHashSet(HOST_LOC31), Sets.newHashSet(), false);
729 Host host2 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
730 Sets.newHashSet(HOST_LOC31), Sets.newHashSet(HOST_IP11), false);
731 Host host3 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
732 Sets.newHashSet(HOST_LOC31, HOST_LOC41), Sets.newHashSet(HOST_IP11), false);
733
734 // Add a dual-home host with only one location and no IP
735 // Expect: only bridging redirection works
736 hostHandler.processHostAddedEvent(new HostEvent(HostEvent.Type.HOST_ADDED, host1));
737 assertEquals(0, ROUTING_TABLE.size());
738 assertEquals(2, BRIDGING_TABLE.size());
739 assertEquals(P1, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV3, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
740 assertEquals(P9, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV4, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
741
742 // Discover IP
743 // Expect: routing redirection should also work
Charles Chan873661e2017-11-30 15:37:50 -0800744 hostHandler.processHostUpdatedEvent(new HostEvent(HostEvent.Type.HOST_UPDATED, host2, host1));
Charles Chanb3c1faf2017-11-20 08:46:24 -0800745 assertEquals(2, ROUTING_TABLE.size());
746 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV3, HOST_IP11.toIpPrefix())).portNumber);
747 assertEquals(P9, ROUTING_TABLE.get(new MockRoutingTableKey(DEV4, HOST_IP11.toIpPrefix())).portNumber);
748 assertEquals(2, BRIDGING_TABLE.size());
749 assertEquals(P1, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV3, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
750 assertEquals(P9, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV4, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
Charles Chan873661e2017-11-30 15:37:50 -0800751 // Expect probe to be sent out on pair device
752 assertTrue(mockLocationProbingService.verifyProbe(host2, CP41, HostLocationProbingService.ProbeMode.DISCOVER));
Charles Chanb3c1faf2017-11-20 08:46:24 -0800753
754 // Discover location
755 // Expect: cancel all redirections
Charles Chan873661e2017-11-30 15:37:50 -0800756 hostHandler.processHostMovedEvent(new HostEvent(HostEvent.Type.HOST_MOVED, host3, host2));
Charles Chanb3c1faf2017-11-20 08:46:24 -0800757 assertEquals(2, ROUTING_TABLE.size());
758 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV3, HOST_IP11.toIpPrefix())).portNumber);
Charles Chan873661e2017-11-30 15:37:50 -0800759 assertEquals(P9, ROUTING_TABLE.get(new MockRoutingTableKey(DEV4, HOST_IP11.toIpPrefix())).portNumber);
Charles Chanb3c1faf2017-11-20 08:46:24 -0800760 assertEquals(2, BRIDGING_TABLE.size());
761 assertEquals(P1, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV3, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
Charles Chan873661e2017-11-30 15:37:50 -0800762 assertEquals(P9, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV4, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
763 // FIXME: Delay event handling a little bit to wait for the previous redirection flows to be completed
764 // The permanent solution would be introducing CompletableFuture and wait for it
765 Thread.sleep(HostHandler.HOST_MOVED_DELAY_MS + 50);
766 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV4, HOST_IP11.toIpPrefix())).portNumber);
Charles Chanb3c1faf2017-11-20 08:46:24 -0800767 assertEquals(P1, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV4, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
768 }
769
770 @Test
771 public void testDelayedIpAndLocation2() throws Exception {
772 Host host1 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
773 Sets.newHashSet(HOST_LOC31), Sets.newHashSet(), false);
774 Host host2 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
775 Sets.newHashSet(HOST_LOC31, HOST_LOC41), Sets.newHashSet(), false);
776 Host host3 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
777 Sets.newHashSet(HOST_LOC31, HOST_LOC41), Sets.newHashSet(HOST_IP11), false);
778
779 // Add a dual-home host with only one location and no IP
780 // Expect: only bridging redirection works
781 hostHandler.processHostAddedEvent(new HostEvent(HostEvent.Type.HOST_ADDED, host1));
782 assertEquals(0, ROUTING_TABLE.size());
783 assertEquals(2, BRIDGING_TABLE.size());
784 assertEquals(P1, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV3, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
785 assertEquals(P9, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV4, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
786
787 // Discover Location
788 // Expect: cancel bridging redirections
789 hostHandler.processHostAddedEvent(new HostEvent(HostEvent.Type.HOST_MOVED, host2, host1));
790 assertEquals(0, ROUTING_TABLE.size());
791 assertEquals(2, BRIDGING_TABLE.size());
792 assertEquals(P1, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV3, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
793 assertEquals(P1, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV4, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
794
795 // Discover IP
796 // Expect: add IP rules to both location
797 hostHandler.processHostAddedEvent(new HostEvent(HostEvent.Type.HOST_UPDATED, host3, host2));
798 assertEquals(2, ROUTING_TABLE.size());
799 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV3, HOST_IP11.toIpPrefix())).portNumber);
800 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV4, HOST_IP11.toIpPrefix())).portNumber);
801 assertEquals(2, BRIDGING_TABLE.size());
802 assertEquals(P1, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV3, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
803 assertEquals(P1, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV4, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
804 }
805
806 @Test
Charles Chand9265a32017-06-16 15:19:24 -0700807 public void testDualHomedHostUpdated() throws Exception {
808 Host host1 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
809 Sets.newHashSet(HOST_LOC11, HOST_LOC21), Sets.newHashSet(HOST_IP11, HOST_IP12), false);
810 Host host2 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
811 Sets.newHashSet(HOST_LOC11, HOST_LOC21), Sets.newHashSet(HOST_IP11, HOST_IP21), false);
812 Host host3 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
813 Sets.newHashSet(HOST_LOC11, HOST_LOC21), Sets.newHashSet(HOST_IP13, HOST_IP14), false);
814
815 // Add a dual-homed host with two locations and two IPs
816 // Expect: add four new routing rules. Add two new bridging rules
817 hostHandler.processHostAddedEvent(new HostEvent(HostEvent.Type.HOST_ADDED, host1));
Charles Chan6c624992017-08-18 17:11:34 -0700818 assertEquals(4, ROUTING_TABLE.size());
819 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP11.toIpPrefix())));
820 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP12.toIpPrefix())));
821 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV2, HOST_IP11.toIpPrefix())));
822 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV2, HOST_IP12.toIpPrefix())));
823 assertEquals(2, BRIDGING_TABLE.size());
824 assertNotNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_UNTAGGED)));
825 assertNotNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV2, HOST_MAC, INTF_VLAN_UNTAGGED)));
Charles Chand9265a32017-06-16 15:19:24 -0700826
827 // Update both host IPs
828 // Expect: update routing rules with new IP. No change to bridging rule.
829 hostHandler.processHostUpdatedEvent(new HostEvent(HostEvent.Type.HOST_UPDATED, host3, host1));
Charles Chan6c624992017-08-18 17:11:34 -0700830 assertEquals(4, ROUTING_TABLE.size());
831 assertNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP11.toIpPrefix())));
832 assertNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP12.toIpPrefix())));
833 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP13.toIpPrefix())));
834 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP14.toIpPrefix())));
835 assertNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV2, HOST_IP11.toIpPrefix())));
836 assertNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV2, HOST_IP12.toIpPrefix())));
837 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV2, HOST_IP13.toIpPrefix())));
838 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV2, HOST_IP14.toIpPrefix())));
839 assertEquals(2, BRIDGING_TABLE.size());
840 assertNotNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_UNTAGGED)));
841 assertNotNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV2, HOST_MAC, INTF_VLAN_UNTAGGED)));
Charles Chand9265a32017-06-16 15:19:24 -0700842
843 // Update one of the host IP to different subnet
844 // Expect: update routing rule with new IP. No change to bridging rule.
845 hostHandler.processHostUpdatedEvent(new HostEvent(HostEvent.Type.HOST_UPDATED, host2, host3));
Charles Chan6c624992017-08-18 17:11:34 -0700846 assertEquals(2, ROUTING_TABLE.size());
847 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP11.toIpPrefix())));
848 assertNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP21.toIpPrefix())));
849 assertNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP12.toIpPrefix())));
850 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV2, HOST_IP11.toIpPrefix())));
851 assertNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV2, HOST_IP21.toIpPrefix())));
852 assertNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV2, HOST_IP12.toIpPrefix())));
853 assertEquals(2, BRIDGING_TABLE.size());
854 assertNotNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_UNTAGGED)));
855 assertNotNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV2, HOST_MAC, INTF_VLAN_UNTAGGED)));
Charles Chand9265a32017-06-16 15:19:24 -0700856 }
Ray Milkeyc2d43be2017-08-03 11:58:29 -0700857}