blob: 3ad2d52b49bfa35b5db6bf0c3c0cacb37c013dae [file] [log] [blame]
Charles Chan2e2e3402017-06-19 14:00:53 -07001/*
Brian O'Connora09fe5b2017-08-03 21:12:30 -07002 * Copyright 2017-present Open Networking Foundation
Charles Chan2e2e3402017-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 Chan65238242017-06-22 18:03:14 -070019import com.fasterxml.jackson.databind.JsonNode;
20import com.fasterxml.jackson.databind.ObjectMapper;
Charles Chan2e2e3402017-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 Chan48f89a52017-08-18 17:11:34 -070030import org.onosproject.net.config.ConfigApplyDelegate;
Ray Milkeyfacf2862017-08-03 11:58:29 -070031import org.onosproject.net.intf.Interface;
Charles Chan2e2e3402017-06-19 14:00:53 -070032import org.onosproject.net.ConnectPoint;
33import org.onosproject.net.DefaultHost;
34import org.onosproject.net.DeviceId;
35import org.onosproject.net.Host;
36import org.onosproject.net.HostId;
37import org.onosproject.net.HostLocation;
38import org.onosproject.net.PortNumber;
39import org.onosproject.net.config.NetworkConfigRegistryAdapter;
Charles Chan2e2e3402017-06-19 14:00:53 -070040import org.onosproject.net.flow.TrafficTreatment;
Charles Chan2e2e3402017-06-19 14:00:53 -070041import org.onosproject.net.host.HostEvent;
42import org.onosproject.net.host.InterfaceIpAddress;
43import org.onosproject.net.provider.ProviderId;
44import org.onosproject.segmentrouting.config.DeviceConfiguration;
Charles Chan65238242017-06-22 18:03:14 -070045import org.onosproject.segmentrouting.config.SegmentRoutingDeviceConfig;
Charles Chan2e2e3402017-06-19 14:00:53 -070046
47import java.util.Map;
Charles Chan2e2e3402017-06-19 14:00:53 -070048import java.util.Set;
Charles Chan2e2e3402017-06-19 14:00:53 -070049
50import static org.junit.Assert.*;
51
Charles Chan48f89a52017-08-18 17:11:34 -070052/**r
Charles Chan2e2e3402017-06-19 14:00:53 -070053 * Unit test for {@link HostHandler}.
54 */
55public class HostHandlerTest {
Charles Chan2e2e3402017-06-19 14:00:53 -070056 private HostHandler hostHandler;
57
58 // Mocked routing and bridging tables
Charles Chan48f89a52017-08-18 17:11:34 -070059 private static final Map<MockBridgingTableKey, MockBridgingTableValue> BRIDGING_TABLE =
60 Maps.newConcurrentMap();
61 private static final Map<MockRoutingTableKey, MockRoutingTableValue> ROUTING_TABLE =
62 Maps.newConcurrentMap();
Charles Chan2e2e3402017-06-19 14:00:53 -070063 // Mocked Next Id
Charles Chan48f89a52017-08-18 17:11:34 -070064 private static final Map<Integer, TrafficTreatment> NEXT_TABLE = Maps.newConcurrentMap();
Charles Chan2e2e3402017-06-19 14:00:53 -070065
Charles Chanf9a52702017-06-16 15:19:24 -070066 // Host Mac, VLAN
Charles Chan2e2e3402017-06-19 14:00:53 -070067 private static final ProviderId PROVIDER_ID = ProviderId.NONE;
68 private static final MacAddress HOST_MAC = MacAddress.valueOf("00:00:00:00:00:01");
69 private static final VlanId HOST_VLAN_UNTAGGED = VlanId.NONE;
70 private static final HostId HOST_ID_UNTAGGED = HostId.hostId(HOST_MAC, HOST_VLAN_UNTAGGED);
71 private static final VlanId HOST_VLAN_TAGGED = VlanId.vlanId((short) 20);
72 private static final HostId HOST_ID_TAGGED = HostId.hostId(HOST_MAC, HOST_VLAN_TAGGED);
Charles Chanf9a52702017-06-16 15:19:24 -070073 // Host IP
74 private static final IpAddress HOST_IP11 = IpAddress.valueOf("10.0.1.1");
75 private static final IpAddress HOST_IP21 = IpAddress.valueOf("10.0.2.1");
76 private static final IpAddress HOST_IP12 = IpAddress.valueOf("10.0.1.2");
77 private static final IpAddress HOST_IP13 = IpAddress.valueOf("10.0.1.3");
78 private static final IpAddress HOST_IP14 = IpAddress.valueOf("10.0.1.4");
79 // Device
80 private static final DeviceId DEV1 = DeviceId.deviceId("of:0000000000000001");
81 private static final DeviceId DEV2 = DeviceId.deviceId("of:0000000000000002");
Charles Chan9f08b102017-08-09 16:50:15 -070082 private static final DeviceId DEV3 = DeviceId.deviceId("of:0000000000000003");
83 private static final DeviceId DEV4 = DeviceId.deviceId("of:0000000000000004");
84 private static final DeviceId DEV5 = DeviceId.deviceId("of:0000000000000005");
85 private static final DeviceId DEV6 = DeviceId.deviceId("of:0000000000000006");
Charles Chanf9a52702017-06-16 15:19:24 -070086 // Port
87 private static final PortNumber P1 = PortNumber.portNumber(1);
88 private static final PortNumber P2 = PortNumber.portNumber(2);
89 private static final PortNumber P3 = PortNumber.portNumber(3);
Charles Chan65238242017-06-22 18:03:14 -070090 private static final PortNumber P9 = PortNumber.portNumber(9);
Charles Chanf9a52702017-06-16 15:19:24 -070091 // Connect Point
92 private static final ConnectPoint CP11 = new ConnectPoint(DEV1, P1);
93 private static final HostLocation HOST_LOC11 = new HostLocation(CP11, 0);
94 private static final ConnectPoint CP12 = new ConnectPoint(DEV1, P2);
95 private static final HostLocation HOST_LOC12 = new HostLocation(CP12, 0);
96 private static final ConnectPoint CP13 = new ConnectPoint(DEV1, P3);
97 private static final HostLocation HOST_LOC13 = new HostLocation(CP13, 0);
98 private static final ConnectPoint CP21 = new ConnectPoint(DEV2, P1);
99 private static final HostLocation HOST_LOC21 = new HostLocation(CP21, 0);
100 private static final ConnectPoint CP22 = new ConnectPoint(DEV2, P2);
101 private static final HostLocation HOST_LOC22 = new HostLocation(CP22, 0);
Charles Chan65238242017-06-22 18:03:14 -0700102 // Connect Point for dual-homed host failover
103 private static final ConnectPoint CP31 = new ConnectPoint(DEV3, P1);
104 private static final HostLocation HOST_LOC31 = new HostLocation(CP31, 0);
105 private static final ConnectPoint CP41 = new ConnectPoint(DEV4, P1);
106 private static final HostLocation HOST_LOC41 = new HostLocation(CP41, 0);
107 private static final ConnectPoint CP39 = new ConnectPoint(DEV3, P9);
108 private static final ConnectPoint CP49 = new ConnectPoint(DEV4, P9);
Charles Chan9f08b102017-08-09 16:50:15 -0700109 // Conenct Point for mastership test
110 private static final ConnectPoint CP51 = new ConnectPoint(DEV5, P1);
111 private static final HostLocation HOST_LOC51 = new HostLocation(CP51, 0);
112 private static final ConnectPoint CP61 = new ConnectPoint(DEV6, P1);
113 private static final HostLocation HOST_LOC61 = new HostLocation(CP61, 0);
Charles Chanf9a52702017-06-16 15:19:24 -0700114 // Interface VLAN
Charles Chan2e2e3402017-06-19 14:00:53 -0700115 private static final VlanId INTF_VLAN_UNTAGGED = VlanId.vlanId((short) 10);
Charles Chan2e2e3402017-06-19 14:00:53 -0700116 private static final Set<VlanId> INTF_VLAN_TAGGED = Sets.newHashSet(VlanId.vlanId((short) 20));
117 private static final VlanId INTF_VLAN_NATIVE = VlanId.vlanId((short) 30);
Charles Chan65238242017-06-22 18:03:14 -0700118 private static final Set<VlanId> INTF_VLAN_PAIR = Sets.newHashSet(VlanId.vlanId((short) 10),
119 VlanId.vlanId((short) 20), VlanId.vlanId((short) 30));
Charles Chanf9a52702017-06-16 15:19:24 -0700120 // Interface subnet
121 private static final IpPrefix INTF_PREFIX1 = IpPrefix.valueOf("10.0.1.254/24");
122 private static final IpPrefix INTF_PREFIX2 = IpPrefix.valueOf("10.0.2.254/24");
123 private static final InterfaceIpAddress INTF_IP1 =
124 new InterfaceIpAddress(INTF_PREFIX1.address(), INTF_PREFIX1);
125 private static final InterfaceIpAddress INTF_IP2 =
126 new InterfaceIpAddress(INTF_PREFIX2.address(), INTF_PREFIX2);
Charles Chan48f89a52017-08-18 17:11:34 -0700127 // Interfaces
128 private static final Interface INTF11 =
129 new Interface(null, CP11, Lists.newArrayList(INTF_IP1), MacAddress.NONE, null,
130 INTF_VLAN_UNTAGGED, null, null);
131 private static final Interface INTF12 =
132 new Interface(null, CP12, Lists.newArrayList(INTF_IP1), MacAddress.NONE, null,
133 INTF_VLAN_UNTAGGED, null, null);
134 private static final Interface INTF13 =
135 new Interface(null, CP13, Lists.newArrayList(INTF_IP2), MacAddress.NONE, null,
136 null, INTF_VLAN_TAGGED, INTF_VLAN_NATIVE);
137 private static final Interface INTF21 =
138 new Interface(null, CP21, Lists.newArrayList(INTF_IP1), MacAddress.NONE, null,
139 INTF_VLAN_UNTAGGED, null, null);
140 private static final Interface INTF22 =
141 new Interface(null, CP22, Lists.newArrayList(INTF_IP1), MacAddress.NONE, null,
142 INTF_VLAN_UNTAGGED, null, null);
143 private static final Interface INTF31 =
144 new Interface(null, CP31, Lists.newArrayList(INTF_IP1), MacAddress.NONE, null,
145 INTF_VLAN_UNTAGGED, null, null);
146 private static final Interface INTF39 =
147 new Interface(null, CP39, Lists.newArrayList(INTF_IP1), MacAddress.NONE, null,
148 null, INTF_VLAN_PAIR, null);
149 private static final Interface INTF41 =
150 new Interface(null, CP41, Lists.newArrayList(INTF_IP1), MacAddress.NONE, null,
151 INTF_VLAN_UNTAGGED, null, null);
152 private static final Interface INTF49 =
153 new Interface(null, CP49, Lists.newArrayList(INTF_IP1), MacAddress.NONE, null,
154 null, INTF_VLAN_PAIR, null);
Charles Chanf9a52702017-06-16 15:19:24 -0700155 // Host
156 private static final Host HOST1 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC,
157 HOST_VLAN_UNTAGGED, Sets.newHashSet(HOST_LOC11, HOST_LOC21), Sets.newHashSet(HOST_IP11),
158 false);
Charles Chan2e2e3402017-06-19 14:00:53 -0700159
Charles Chan48f89a52017-08-18 17:11:34 -0700160 // A set of hosts
161 private static final Set<Host> HOSTS = Sets.newHashSet(HOST1);
162 // A set of devices of which we have mastership
163 private static final Set<DeviceId> LOCAL_DEVICES = Sets.newHashSet(DEV1, DEV2, DEV3, DEV4);
164 // A set of interfaces
165 private static final Set<Interface> INTERFACES = Sets.newHashSet(INTF11, INTF12, INTF13, INTF21,
166 INTF22, INTF31, INTF39, INTF41, INTF49);
167
Charles Chan2e2e3402017-06-19 14:00:53 -0700168 @Before
169 public void setUp() throws Exception {
Charles Chan48f89a52017-08-18 17:11:34 -0700170 // Initialize pairDevice and pairLocalPort config
171 ObjectMapper mapper = new ObjectMapper();
172 ConfigApplyDelegate delegate = config -> { };
173
174 SegmentRoutingDeviceConfig dev3Config = new SegmentRoutingDeviceConfig();
175 JsonNode dev3Tree = mapper.createObjectNode();
176 dev3Config.init(DEV3, "host-handler-test", dev3Tree, mapper, delegate);
177 dev3Config.setPairDeviceId(DEV4).setPairLocalPort(P9);
178
179 SegmentRoutingDeviceConfig dev4Config = new SegmentRoutingDeviceConfig();
180 JsonNode dev4Tree = mapper.createObjectNode();
181 dev4Config.init(DEV4, "host-handler-test", dev4Tree, mapper, delegate);
182 dev4Config.setPairDeviceId(DEV3).setPairLocalPort(P9);
183
184 MockNetworkConfigRegistry mockNetworkConfigRegistry = new MockNetworkConfigRegistry();
185 mockNetworkConfigRegistry.applyConfig(dev3Config);
186 mockNetworkConfigRegistry.applyConfig(dev4Config);
187
188 // Initialize Segment Routing Manager
189 SegmentRoutingManager srManager = new MockSegmentRoutingManager(NEXT_TABLE);
Charles Chan2e2e3402017-06-19 14:00:53 -0700190 srManager.cfgService = new NetworkConfigRegistryAdapter();
191 srManager.deviceConfiguration = new DeviceConfiguration(srManager);
Charles Chan48f89a52017-08-18 17:11:34 -0700192 srManager.flowObjectiveService = new MockFlowObjectiveService(BRIDGING_TABLE, NEXT_TABLE);
193 srManager.routingRulePopulator = new MockRoutingRulePopulator(srManager, ROUTING_TABLE);
194 srManager.interfaceService = new MockInterfaceService(INTERFACES);
195 srManager.mastershipService = new MockMastershipService(LOCAL_DEVICES);
196 srManager.hostService = new MockHostService(HOSTS);
197 srManager.cfgService = mockNetworkConfigRegistry;
Charles Chan2e2e3402017-06-19 14:00:53 -0700198
199 hostHandler = new HostHandler(srManager);
200
Charles Chan48f89a52017-08-18 17:11:34 -0700201 ROUTING_TABLE.clear();
202 BRIDGING_TABLE.clear();
Charles Chan2e2e3402017-06-19 14:00:53 -0700203 }
204
205 @Test
206 public void init() throws Exception {
Charles Chanf9a52702017-06-16 15:19:24 -0700207 hostHandler.init(DEV1);
Charles Chan48f89a52017-08-18 17:11:34 -0700208 assertEquals(1, ROUTING_TABLE.size());
209 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP11.toIpPrefix())));
210 assertEquals(1, BRIDGING_TABLE.size());
211 assertNotNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_UNTAGGED)));
Charles Chanf9a52702017-06-16 15:19:24 -0700212
213 hostHandler.init(DEV2);
Charles Chan48f89a52017-08-18 17:11:34 -0700214 assertEquals(2, ROUTING_TABLE.size());
215 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP11.toIpPrefix())));
216 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV2, HOST_IP11.toIpPrefix())));
217 assertEquals(2, BRIDGING_TABLE.size());
218 assertNotNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_UNTAGGED)));
219 assertNotNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV2, HOST_MAC, INTF_VLAN_UNTAGGED)));
Charles Chanf9a52702017-06-16 15:19:24 -0700220 }
221
222 @Test(expected = IllegalArgumentException.class)
223 public void testHostAddedAtWrongLocation() throws Exception {
224 hostHandler.processHostAddedAtLocation(HOST1, HOST_LOC13);
225 }
226
227
228 @Test()
229 public void testHostAddedAtCorrectLocation() throws Exception {
230 hostHandler.processHostAddedAtLocation(HOST1, HOST_LOC11);
Charles Chan48f89a52017-08-18 17:11:34 -0700231 assertEquals(1, ROUTING_TABLE.size());
232 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP11.toIpPrefix())));
233 assertEquals(1, BRIDGING_TABLE.size());
234 assertNotNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_UNTAGGED)));
Charles Chan2e2e3402017-06-19 14:00:53 -0700235 }
236
237 @Test
238 public void testHostAdded() throws Exception {
239 Host subject;
240
241 // Untagged host discovered on untagged port
242 // Expect: add one routing rule and one bridging rule
243 subject = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
Charles Chanf9a52702017-06-16 15:19:24 -0700244 Sets.newHashSet(HOST_LOC11), Sets.newHashSet(HOST_IP11), false);
Charles Chan2e2e3402017-06-19 14:00:53 -0700245 hostHandler.processHostAddedEvent(new HostEvent(HostEvent.Type.HOST_ADDED, subject));
Charles Chan48f89a52017-08-18 17:11:34 -0700246 assertEquals(1, ROUTING_TABLE.size());
247 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP11.toIpPrefix())));
248 assertEquals(1, BRIDGING_TABLE.size());
249 assertNotNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_UNTAGGED)));
Charles Chan2e2e3402017-06-19 14:00:53 -0700250
251 // Untagged host discovered on tagged/native port
252 // Expect: add one routing rule and one bridging rule
253 subject = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
Charles Chanf9a52702017-06-16 15:19:24 -0700254 Sets.newHashSet(HOST_LOC13), Sets.newHashSet(HOST_IP21), false);
Charles Chan2e2e3402017-06-19 14:00:53 -0700255 hostHandler.processHostAddedEvent(new HostEvent(HostEvent.Type.HOST_ADDED, subject));
Charles Chan48f89a52017-08-18 17:11:34 -0700256 assertEquals(2, ROUTING_TABLE.size());
257 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP21.toIpPrefix())));
258 assertEquals(2, BRIDGING_TABLE.size());
259 assertNotNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_NATIVE)));
Charles Chan2e2e3402017-06-19 14:00:53 -0700260
261 // Tagged host discovered on untagged port
262 // Expect: ignore the host. No rule is added.
263 subject = new DefaultHost(PROVIDER_ID, HOST_ID_TAGGED, HOST_MAC, HOST_VLAN_TAGGED,
Charles Chanf9a52702017-06-16 15:19:24 -0700264 Sets.newHashSet(HOST_LOC11), Sets.newHashSet(HOST_IP11), false);
Charles Chan2e2e3402017-06-19 14:00:53 -0700265 hostHandler.processHostAddedEvent(new HostEvent(HostEvent.Type.HOST_ADDED, subject));
Charles Chan48f89a52017-08-18 17:11:34 -0700266 assertEquals(2, ROUTING_TABLE.size());
267 assertEquals(2, BRIDGING_TABLE.size());
Charles Chan2e2e3402017-06-19 14:00:53 -0700268
269 // Tagged host discovered on tagged port with the same IP
270 // Expect: update existing route, add one bridging rule
271 subject = new DefaultHost(PROVIDER_ID, HOST_ID_TAGGED, HOST_MAC, HOST_VLAN_TAGGED,
Charles Chanf9a52702017-06-16 15:19:24 -0700272 Sets.newHashSet(HOST_LOC13), Sets.newHashSet(HOST_IP21), false);
Charles Chan2e2e3402017-06-19 14:00:53 -0700273 hostHandler.processHostAddedEvent(new HostEvent(HostEvent.Type.HOST_ADDED, subject));
Charles Chan48f89a52017-08-18 17:11:34 -0700274 assertEquals(2, ROUTING_TABLE.size());
275 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP21.toIpPrefix())));
276 assertEquals(HOST_VLAN_TAGGED, ROUTING_TABLE.get(new MockRoutingTableKey(HOST_LOC13.deviceId(),
Charles Chanf9a52702017-06-16 15:19:24 -0700277 HOST_IP21.toIpPrefix())).vlanId);
Charles Chan48f89a52017-08-18 17:11:34 -0700278 assertEquals(3, BRIDGING_TABLE.size());
279 assertNotNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, HOST_VLAN_TAGGED)));
Charles Chanf9a52702017-06-16 15:19:24 -0700280 }
281
282 @Test
283 public void testDualHomedHostAdded() throws Exception {
284 // Add a dual-homed host that has 2 locations
285 // Expect: add two routing rules and two bridging rules
286 Host subject = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
287 Sets.newHashSet(HOST_LOC11, HOST_LOC21), Sets.newHashSet(HOST_IP11), false);
288 hostHandler.processHostAddedEvent(new HostEvent(HostEvent.Type.HOST_ADDED, subject));
Charles Chan48f89a52017-08-18 17:11:34 -0700289 assertEquals(2, ROUTING_TABLE.size());
290 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP11.toIpPrefix())));
291 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV2, HOST_IP11.toIpPrefix())));
292 assertEquals(2, BRIDGING_TABLE.size());
293 assertNotNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_UNTAGGED)));
294 assertNotNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV2, HOST_MAC, INTF_VLAN_UNTAGGED)));
Charles Chan2e2e3402017-06-19 14:00:53 -0700295 }
296
297 @Test
298 public void testHostRemoved() throws Exception {
299 Host subject = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
Charles Chanf9a52702017-06-16 15:19:24 -0700300 Sets.newHashSet(HOST_LOC11), Sets.newHashSet(HOST_IP11), false);
Charles Chan2e2e3402017-06-19 14:00:53 -0700301
302 // Add a host
303 // Expect: add one routing rule and one bridging rule
304 hostHandler.processHostAddedEvent(new HostEvent(HostEvent.Type.HOST_ADDED, subject));
Charles Chan48f89a52017-08-18 17:11:34 -0700305 assertEquals(1, ROUTING_TABLE.size());
306 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP11.toIpPrefix())));
307 assertEquals(1, BRIDGING_TABLE.size());
308 assertNotNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_UNTAGGED)));
Charles Chan2e2e3402017-06-19 14:00:53 -0700309
310 // Remove the host
311 // Expect: add the routing rule and the bridging rule
Charles Chanf9a52702017-06-16 15:19:24 -0700312 hostHandler.processHostRemovedEvent(new HostEvent(HostEvent.Type.HOST_REMOVED, subject));
Charles Chan48f89a52017-08-18 17:11:34 -0700313 assertEquals(0, ROUTING_TABLE.size());
314 assertEquals(0, BRIDGING_TABLE.size());
Charles Chanf9a52702017-06-16 15:19:24 -0700315 }
316
317 @Test
318 public void testDualHomedHostRemoved() throws Exception {
319 // Add a dual-homed host that has 2 locations
320 // Expect: add two routing rules and two bridging rules
321 Host subject = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
322 Sets.newHashSet(HOST_LOC11, HOST_LOC21), Sets.newHashSet(HOST_IP11), false);
323 hostHandler.processHostAddedEvent(new HostEvent(HostEvent.Type.HOST_ADDED, subject));
Charles Chan48f89a52017-08-18 17:11:34 -0700324 assertEquals(2, ROUTING_TABLE.size());
325 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP11.toIpPrefix())));
326 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV2, HOST_IP11.toIpPrefix())));
327 assertEquals(2, BRIDGING_TABLE.size());
328 assertNotNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_UNTAGGED)));
329 assertNotNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV2, HOST_MAC, INTF_VLAN_UNTAGGED)));
Charles Chanf9a52702017-06-16 15:19:24 -0700330
331 // Remove a dual-homed host that has 2 locations
332 // Expect: all routing and bridging rules are removed
333 hostHandler.processHostRemovedEvent(new HostEvent(HostEvent.Type.HOST_REMOVED, subject));
Charles Chan48f89a52017-08-18 17:11:34 -0700334 assertEquals(0, ROUTING_TABLE.size());
335 assertEquals(0, BRIDGING_TABLE.size());
Charles Chan2e2e3402017-06-19 14:00:53 -0700336 }
337
338 @Test
339 public void testHostMoved() throws Exception {
340 Host host1 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
Charles Chanf9a52702017-06-16 15:19:24 -0700341 Sets.newHashSet(HOST_LOC11), Sets.newHashSet(HOST_IP11), false);
Charles Chan2e2e3402017-06-19 14:00:53 -0700342 Host host2 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
Charles Chanf9a52702017-06-16 15:19:24 -0700343 Sets.newHashSet(HOST_LOC21), Sets.newHashSet(HOST_IP11), false);
Charles Chan2e2e3402017-06-19 14:00:53 -0700344 Host host3 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
Charles Chanf9a52702017-06-16 15:19:24 -0700345 Sets.newHashSet(HOST_LOC13), Sets.newHashSet(HOST_IP11), false);
Charles Chan2e2e3402017-06-19 14:00:53 -0700346
347 // Add a host
Charles Chan65238242017-06-22 18:03:14 -0700348 // Expect: add one new routing rule, one new bridging rule
Charles Chan2e2e3402017-06-19 14:00:53 -0700349 hostHandler.processHostAddedEvent(new HostEvent(HostEvent.Type.HOST_ADDED, host1));
Charles Chan48f89a52017-08-18 17:11:34 -0700350 assertEquals(1, ROUTING_TABLE.size());
351 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP11.toIpPrefix())));
352 assertNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP21.toIpPrefix())));
353 assertNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP13.toIpPrefix())));
354 assertEquals(1, BRIDGING_TABLE.size());
355 assertNotNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_UNTAGGED)));
356 assertNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV2, HOST_MAC, INTF_VLAN_UNTAGGED)));
Charles Chan2e2e3402017-06-19 14:00:53 -0700357
Charles Chanf9a52702017-06-16 15:19:24 -0700358 // Move the host to CP13, which has different subnet setting
Charles Chan2e2e3402017-06-19 14:00:53 -0700359 // Expect: remove routing rule. Change vlan in bridging rule.
Charles Chanf9a52702017-06-16 15:19:24 -0700360 hostHandler.processHostMovedEvent(new HostEvent(HostEvent.Type.HOST_MOVED, host3, host1));
Charles Chan48f89a52017-08-18 17:11:34 -0700361 assertEquals(0, ROUTING_TABLE.size());
362 assertEquals(1, BRIDGING_TABLE.size());
363 assertNotNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_NATIVE)));
364 assertNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_UNTAGGED)));
Charles Chan2e2e3402017-06-19 14:00:53 -0700365
Charles Chanf9a52702017-06-16 15:19:24 -0700366 // Move the host to CP21, which has same subnet setting
Charles Chan2e2e3402017-06-19 14:00:53 -0700367 // Expect: add a new routing rule. Change vlan in bridging rule.
Charles Chanf9a52702017-06-16 15:19:24 -0700368 hostHandler.processHostMovedEvent(new HostEvent(HostEvent.Type.HOST_MOVED, host2, host3));
Charles Chan48f89a52017-08-18 17:11:34 -0700369 assertEquals(1, ROUTING_TABLE.size());
370 assertNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP11.toIpPrefix())));
371 assertNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP11.toIpPrefix())));
372 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV2, HOST_IP11.toIpPrefix())));
373 assertEquals(1, BRIDGING_TABLE.size());
374 assertNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_UNTAGGED)));
375 assertNotNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV2, HOST_MAC, INTF_VLAN_UNTAGGED)));
Charles Chanf9a52702017-06-16 15:19:24 -0700376 }
377
378 @Test
379 public void testDualHomedHostMoved() throws Exception {
380 Host host1 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
381 Sets.newHashSet(HOST_LOC11, HOST_LOC21), Sets.newHashSet(HOST_IP11, HOST_IP12), false);
382 Host host2 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
383 Sets.newHashSet(HOST_LOC12, HOST_LOC22), Sets.newHashSet(HOST_IP11, HOST_IP12), false);
384 Host host3 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
385 Sets.newHashSet(HOST_LOC11, HOST_LOC21), Sets.newHashSet(HOST_IP13, HOST_IP14), false);
386 Host host4 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
387 Sets.newHashSet(HOST_LOC11, HOST_LOC22), Sets.newHashSet(HOST_IP12, HOST_IP13), false);
388
389 // Add a host with IP11, IP12 and LOC11, LOC21
390 // Expect: 4 routing rules and 2 bridging rules
391 hostHandler.processHostAddedEvent(new HostEvent(HostEvent.Type.HOST_ADDED, host1));
Charles Chan48f89a52017-08-18 17:11:34 -0700392 assertEquals(4, ROUTING_TABLE.size());
393 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP11.toIpPrefix())).portNumber);
394 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP12.toIpPrefix())).portNumber);
395 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV2, HOST_IP11.toIpPrefix())).portNumber);
396 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV2, HOST_IP12.toIpPrefix())).portNumber);
397 assertEquals(2, BRIDGING_TABLE.size());
398 assertEquals(P1, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
399 assertEquals(P1, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV2, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
Charles Chanf9a52702017-06-16 15:19:24 -0700400
401 // Move the host to LOC12, LOC22 and keep the IP
402 // Expect: 4 routing rules and 2 bridging rules all at the new location
403 hostHandler.processHostMovedEvent(new HostEvent(HostEvent.Type.HOST_MOVED, host2, host1));
Charles Chan48f89a52017-08-18 17:11:34 -0700404 assertEquals(4, ROUTING_TABLE.size());
405 assertEquals(P2, ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP11.toIpPrefix())).portNumber);
406 assertEquals(P2, ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP12.toIpPrefix())).portNumber);
407 assertEquals(P2, ROUTING_TABLE.get(new MockRoutingTableKey(DEV2, HOST_IP11.toIpPrefix())).portNumber);
408 assertEquals(P2, ROUTING_TABLE.get(new MockRoutingTableKey(DEV2, HOST_IP12.toIpPrefix())).portNumber);
409 assertEquals(2, BRIDGING_TABLE.size());
410 assertEquals(P2, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
411 assertEquals(P2, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV2, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
Charles Chanf9a52702017-06-16 15:19:24 -0700412
413 // Move the host to LOC11, LOC21 and change the IP to IP13, IP14 at the same time
414 // Expect: 4 routing rules and 2 bridging rules all at the new location
415 hostHandler.processHostMovedEvent(new HostEvent(HostEvent.Type.HOST_MOVED, host3, host2));
Charles Chan48f89a52017-08-18 17:11:34 -0700416 assertEquals(4, ROUTING_TABLE.size());
417 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP13.toIpPrefix())).portNumber);
418 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP14.toIpPrefix())).portNumber);
419 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV2, HOST_IP13.toIpPrefix())).portNumber);
420 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV2, HOST_IP14.toIpPrefix())).portNumber);
421 assertEquals(2, BRIDGING_TABLE.size());
422 assertEquals(P1, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
423 assertEquals(P1, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV2, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
Charles Chanf9a52702017-06-16 15:19:24 -0700424
425 // Move the host to LOC11, LOC22 and change the IP to IP12, IP13 at the same time
426 // Expect: 4 routing rules and 2 bridging rules all at the new location
427 hostHandler.processHostMovedEvent(new HostEvent(HostEvent.Type.HOST_MOVED, host4, host3));
Charles Chan48f89a52017-08-18 17:11:34 -0700428 assertEquals(4, ROUTING_TABLE.size());
429 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP12.toIpPrefix())).portNumber);
430 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP13.toIpPrefix())).portNumber);
431 assertEquals(P2, ROUTING_TABLE.get(new MockRoutingTableKey(DEV2, HOST_IP12.toIpPrefix())).portNumber);
432 assertEquals(P2, ROUTING_TABLE.get(new MockRoutingTableKey(DEV2, HOST_IP13.toIpPrefix())).portNumber);
433 assertEquals(2, BRIDGING_TABLE.size());
434 assertEquals(P1, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
435 assertEquals(P2, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV2, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
Charles Chan2e2e3402017-06-19 14:00:53 -0700436 }
437
438 @Test
Charles Chan9f08b102017-08-09 16:50:15 -0700439 public void testHostMoveToInvalidLocation() throws Exception {
440 Host host1 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
441 Sets.newHashSet(HOST_LOC11), Sets.newHashSet(HOST_IP11), false);
442 Host host2 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
443 Sets.newHashSet(HOST_LOC51), Sets.newHashSet(HOST_IP11), false);
444
445 // Add a host
446 // Expect: add one new routing rule, one new bridging rule
447 hostHandler.processHostAddedEvent(new HostEvent(HostEvent.Type.HOST_ADDED, host1));
Charles Chan48f89a52017-08-18 17:11:34 -0700448 assertEquals(1, ROUTING_TABLE.size());
449 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP11.toIpPrefix())));
450 assertNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP21.toIpPrefix())));
451 assertEquals(1, BRIDGING_TABLE.size());
452 assertNotNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_UNTAGGED)));
453 assertNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV2, HOST_MAC, INTF_VLAN_UNTAGGED)));
Charles Chan9f08b102017-08-09 16:50:15 -0700454
455 // Move the host to an invalid location
456 // Expect: Old flow is removed. New flow is not created
457 hostHandler.processHostMovedEvent(new HostEvent(HostEvent.Type.HOST_MOVED, host2, host1));
Charles Chan48f89a52017-08-18 17:11:34 -0700458 assertEquals(0, ROUTING_TABLE.size());
459 assertEquals(0, BRIDGING_TABLE.size());
Charles Chan9f08b102017-08-09 16:50:15 -0700460
461 // Move the host to a valid location
462 // Expect: add one new routing rule, one new bridging rule
463 hostHandler.processHostMovedEvent(new HostEvent(HostEvent.Type.HOST_MOVED, host1, host2));
Charles Chan48f89a52017-08-18 17:11:34 -0700464 assertEquals(1, ROUTING_TABLE.size());
465 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP11.toIpPrefix())));
466 assertNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP21.toIpPrefix())));
467 assertEquals(1, BRIDGING_TABLE.size());
468 assertNotNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_UNTAGGED)));
469 assertNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV2, HOST_MAC, INTF_VLAN_UNTAGGED)));
Charles Chan9f08b102017-08-09 16:50:15 -0700470 }
471
472 @Test
473 public void testDualHomedHostMoveToInvalidLocation() throws Exception {
474 Host host1 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
475 Sets.newHashSet(HOST_LOC11, HOST_LOC21), Sets.newHashSet(HOST_IP11), false);
476 Host host2 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
477 Sets.newHashSet(HOST_LOC11, HOST_LOC51), Sets.newHashSet(HOST_IP11), false);
478 Host host3 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
479 Sets.newHashSet(HOST_LOC61, HOST_LOC51), Sets.newHashSet(HOST_IP11), false);
480
481 // Add a host
482 // Expect: add two new routing rules, two new bridging rules
483 hostHandler.processHostAddedEvent(new HostEvent(HostEvent.Type.HOST_ADDED, host1));
Charles Chan48f89a52017-08-18 17:11:34 -0700484 assertEquals(2, ROUTING_TABLE.size());
485 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP11.toIpPrefix())).portNumber);
486 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV2, HOST_IP11.toIpPrefix())).portNumber);
487 assertEquals(2, BRIDGING_TABLE.size());
488 assertEquals(P1, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
489 assertEquals(P1, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV2, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
Charles Chan9f08b102017-08-09 16:50:15 -0700490
491 // Move first host location to an invalid location
492 // Expect: One routing and one bridging flow
493 hostHandler.processHostMovedEvent(new HostEvent(HostEvent.Type.HOST_MOVED, host2, host1));
Charles Chan48f89a52017-08-18 17:11:34 -0700494 assertEquals(1, ROUTING_TABLE.size());
495 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP11.toIpPrefix())).portNumber);
496 assertEquals(1, BRIDGING_TABLE.size());
497 assertEquals(P1, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
Charles Chan9f08b102017-08-09 16:50:15 -0700498
499 // Move second host location to an invalid location
500 // Expect: No routing or bridging rule
501 hostHandler.processHostMovedEvent(new HostEvent(HostEvent.Type.HOST_MOVED, host3, host2));
Charles Chan48f89a52017-08-18 17:11:34 -0700502 assertEquals(0, ROUTING_TABLE.size());
503 assertEquals(0, BRIDGING_TABLE.size());
Charles Chan9f08b102017-08-09 16:50:15 -0700504
505 // Move second host location back to a valid location
506 // Expect: One routing and one bridging flow
507 hostHandler.processHostMovedEvent(new HostEvent(HostEvent.Type.HOST_MOVED, host2, host3));
Charles Chan48f89a52017-08-18 17:11:34 -0700508 assertEquals(1, ROUTING_TABLE.size());
509 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP11.toIpPrefix())).portNumber);
510 assertEquals(1, BRIDGING_TABLE.size());
511 assertEquals(P1, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
Charles Chan9f08b102017-08-09 16:50:15 -0700512
513 // Move first host location back to a valid location
514 // Expect: Two routing and two bridging flow
515 hostHandler.processHostMovedEvent(new HostEvent(HostEvent.Type.HOST_MOVED, host1, host2));
Charles Chan48f89a52017-08-18 17:11:34 -0700516 assertEquals(2, ROUTING_TABLE.size());
517 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP11.toIpPrefix())).portNumber);
518 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV2, HOST_IP11.toIpPrefix())).portNumber);
519 assertEquals(2, BRIDGING_TABLE.size());
520 assertEquals(P1, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
521 assertEquals(P1, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV2, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
Charles Chan9f08b102017-08-09 16:50:15 -0700522 }
523
524 @Test
Charles Chan65238242017-06-22 18:03:14 -0700525 public void testDualHomingSingleLocationFail() throws Exception {
526 Host host1 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
527 Sets.newHashSet(HOST_LOC31, HOST_LOC41), Sets.newHashSet(HOST_IP11, HOST_IP12), false);
528 Host host2 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
529 Sets.newHashSet(HOST_LOC31), Sets.newHashSet(HOST_IP11, HOST_IP12), false);
530
531 // Add a host
532 // Expect: add four new routing rules, two new bridging rules
533 hostHandler.processHostAddedEvent(new HostEvent(HostEvent.Type.HOST_ADDED, host1));
Charles Chan48f89a52017-08-18 17:11:34 -0700534 assertEquals(4, ROUTING_TABLE.size());
535 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV3, HOST_IP11.toIpPrefix())).portNumber);
536 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV3, HOST_IP12.toIpPrefix())).portNumber);
537 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV4, HOST_IP11.toIpPrefix())).portNumber);
538 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV4, HOST_IP12.toIpPrefix())).portNumber);
539 assertEquals(2, BRIDGING_TABLE.size());
540 assertEquals(P1, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV3, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
541 assertEquals(P1, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV4, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
Charles Chan65238242017-06-22 18:03:14 -0700542
543 // Host becomes single-homed
544 // Expect: redirect flows from host location to pair link
545 hostHandler.processHostMovedEvent(new HostEvent(HostEvent.Type.HOST_MOVED, host2, host1));
Charles Chan48f89a52017-08-18 17:11:34 -0700546 assertEquals(4, ROUTING_TABLE.size());
547 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV3, HOST_IP11.toIpPrefix())).portNumber);
548 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV3, HOST_IP12.toIpPrefix())).portNumber);
549 assertEquals(P9, ROUTING_TABLE.get(new MockRoutingTableKey(DEV4, HOST_IP11.toIpPrefix())).portNumber);
550 assertEquals(P9, ROUTING_TABLE.get(new MockRoutingTableKey(DEV4, HOST_IP12.toIpPrefix())).portNumber);
551 assertEquals(2, BRIDGING_TABLE.size());
552 assertEquals(P1, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV3, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
553 assertEquals(P9, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV4, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
Charles Chan65238242017-06-22 18:03:14 -0700554
555 // Host becomes dual-homed again
556 // Expect: Redirect flows from pair link back to host location
557 hostHandler.processHostMovedEvent(new HostEvent(HostEvent.Type.HOST_MOVED, host1, host2));
Charles Chan48f89a52017-08-18 17:11:34 -0700558 assertEquals(4, ROUTING_TABLE.size());
559 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV3, HOST_IP11.toIpPrefix())).portNumber);
560 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV3, HOST_IP12.toIpPrefix())).portNumber);
561 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV4, HOST_IP11.toIpPrefix())).portNumber);
562 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV4, HOST_IP12.toIpPrefix())).portNumber);
563 assertEquals(2, BRIDGING_TABLE.size());
564 assertEquals(P1, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV3, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
565 assertEquals(P1, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV4, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
Charles Chan65238242017-06-22 18:03:14 -0700566 }
567
568 @Test
569 public void testDualHomingBothLocationFail() throws Exception {
570 Host host1 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
571 Sets.newHashSet(HOST_LOC31, HOST_LOC41), Sets.newHashSet(HOST_IP11, HOST_IP12), false);
572 Host host2 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
573 Sets.newHashSet(HOST_LOC31), Sets.newHashSet(HOST_IP11, HOST_IP12), false);
574
575 // Add a host
576 // Expect: add four new routing rules, two new bridging rules
577 hostHandler.processHostAddedEvent(new HostEvent(HostEvent.Type.HOST_ADDED, host1));
Charles Chan48f89a52017-08-18 17:11:34 -0700578 assertEquals(4, ROUTING_TABLE.size());
579 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV3, HOST_IP11.toIpPrefix())).portNumber);
580 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV3, HOST_IP12.toIpPrefix())).portNumber);
581 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV4, HOST_IP11.toIpPrefix())).portNumber);
582 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV4, HOST_IP12.toIpPrefix())).portNumber);
583 assertEquals(2, BRIDGING_TABLE.size());
584 assertEquals(P1, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV3, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
585 assertEquals(P1, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV4, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
Charles Chan65238242017-06-22 18:03:14 -0700586
587 // Host becomes single-homed
588 // Expect: redirect flows from host location to pair link
589 hostHandler.processHostMovedEvent(new HostEvent(HostEvent.Type.HOST_MOVED, host2, host1));
Charles Chan48f89a52017-08-18 17:11:34 -0700590 assertEquals(4, ROUTING_TABLE.size());
591 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV3, HOST_IP11.toIpPrefix())).portNumber);
592 assertEquals(P1, ROUTING_TABLE.get(new MockRoutingTableKey(DEV3, HOST_IP12.toIpPrefix())).portNumber);
593 assertEquals(P9, ROUTING_TABLE.get(new MockRoutingTableKey(DEV4, HOST_IP11.toIpPrefix())).portNumber);
594 assertEquals(P9, ROUTING_TABLE.get(new MockRoutingTableKey(DEV4, HOST_IP12.toIpPrefix())).portNumber);
595 assertEquals(2, BRIDGING_TABLE.size());
596 assertEquals(P1, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV3, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
597 assertEquals(P9, BRIDGING_TABLE.get(new MockBridgingTableKey(DEV4, HOST_MAC, INTF_VLAN_UNTAGGED)).portNumber);
Charles Chan65238242017-06-22 18:03:14 -0700598
599 // Host loses both locations
600 // Expect: Remove last location and all previous redirection flows
601 hostHandler.processHostRemovedEvent(new HostEvent(HostEvent.Type.HOST_REMOVED, host2));
Charles Chan48f89a52017-08-18 17:11:34 -0700602 assertEquals(0, ROUTING_TABLE.size());
603 assertEquals(0, BRIDGING_TABLE.size());
Charles Chan65238242017-06-22 18:03:14 -0700604 }
605
606 @Test
Charles Chan2e2e3402017-06-19 14:00:53 -0700607 public void testHostUpdated() throws Exception {
608 Host host1 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
Charles Chanf9a52702017-06-16 15:19:24 -0700609 Sets.newHashSet(HOST_LOC11), Sets.newHashSet(HOST_IP11), false);
Charles Chan2e2e3402017-06-19 14:00:53 -0700610 Host host2 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
Charles Chanf9a52702017-06-16 15:19:24 -0700611 Sets.newHashSet(HOST_LOC11), Sets.newHashSet(HOST_IP21), false);
Charles Chan2e2e3402017-06-19 14:00:53 -0700612 Host host3 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
Charles Chanf9a52702017-06-16 15:19:24 -0700613 Sets.newHashSet(HOST_LOC11), Sets.newHashSet(HOST_IP12), false);
Charles Chan2e2e3402017-06-19 14:00:53 -0700614
615 // Add a host
Charles Chanf9a52702017-06-16 15:19:24 -0700616 // Expect: add one new routing rule. Add one new bridging rule.
Charles Chan2e2e3402017-06-19 14:00:53 -0700617 hostHandler.processHostAddedEvent(new HostEvent(HostEvent.Type.HOST_ADDED, host1));
Charles Chan48f89a52017-08-18 17:11:34 -0700618 assertEquals(1, ROUTING_TABLE.size());
619 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP11.toIpPrefix())));
620 assertNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP21.toIpPrefix())));
621 assertNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP12.toIpPrefix())));
622 assertEquals(1, BRIDGING_TABLE.size());
623 assertNotNull(BRIDGING_TABLE.get(new MockBridgingTableKey(HOST_LOC11.deviceId(), HOST_MAC,
624 INTF_VLAN_UNTAGGED)));
Charles Chan2e2e3402017-06-19 14:00:53 -0700625
626 // Update the host IP to same subnet
627 // Expect: update routing rule with new IP. No change to bridging rule.
628 hostHandler.processHostUpdatedEvent(new HostEvent(HostEvent.Type.HOST_UPDATED, host3, host1));
Charles Chan48f89a52017-08-18 17:11:34 -0700629 assertEquals(1, ROUTING_TABLE.size());
630 assertNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP11.toIpPrefix())));
631 assertNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP21.toIpPrefix())));
632 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP12.toIpPrefix())));
633 assertEquals(1, BRIDGING_TABLE.size());
634 assertNotNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_UNTAGGED)));
Charles Chan2e2e3402017-06-19 14:00:53 -0700635
636 // Update the host IP to different subnet
637 // Expect: Remove routing rule. No change to bridging rule.
638 hostHandler.processHostUpdatedEvent(new HostEvent(HostEvent.Type.HOST_UPDATED, host2, host3));
Charles Chan48f89a52017-08-18 17:11:34 -0700639 assertEquals(0, ROUTING_TABLE.size());
640 assertEquals(1, BRIDGING_TABLE.size());
641 assertNotNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_UNTAGGED)));
Charles Chanf9a52702017-06-16 15:19:24 -0700642 }
643
644 @Test
645 public void testDualHomedHostUpdated() throws Exception {
646 Host host1 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
647 Sets.newHashSet(HOST_LOC11, HOST_LOC21), Sets.newHashSet(HOST_IP11, HOST_IP12), false);
648 Host host2 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
649 Sets.newHashSet(HOST_LOC11, HOST_LOC21), Sets.newHashSet(HOST_IP11, HOST_IP21), false);
650 Host host3 = new DefaultHost(PROVIDER_ID, HOST_ID_UNTAGGED, HOST_MAC, HOST_VLAN_UNTAGGED,
651 Sets.newHashSet(HOST_LOC11, HOST_LOC21), Sets.newHashSet(HOST_IP13, HOST_IP14), false);
652
653 // Add a dual-homed host with two locations and two IPs
654 // Expect: add four new routing rules. Add two new bridging rules
655 hostHandler.processHostAddedEvent(new HostEvent(HostEvent.Type.HOST_ADDED, host1));
Charles Chan48f89a52017-08-18 17:11:34 -0700656 assertEquals(4, ROUTING_TABLE.size());
657 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP11.toIpPrefix())));
658 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP12.toIpPrefix())));
659 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV2, HOST_IP11.toIpPrefix())));
660 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV2, HOST_IP12.toIpPrefix())));
661 assertEquals(2, BRIDGING_TABLE.size());
662 assertNotNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_UNTAGGED)));
663 assertNotNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV2, HOST_MAC, INTF_VLAN_UNTAGGED)));
Charles Chanf9a52702017-06-16 15:19:24 -0700664
665 // Update both host IPs
666 // Expect: update routing rules with new IP. No change to bridging rule.
667 hostHandler.processHostUpdatedEvent(new HostEvent(HostEvent.Type.HOST_UPDATED, host3, host1));
Charles Chan48f89a52017-08-18 17:11:34 -0700668 assertEquals(4, ROUTING_TABLE.size());
669 assertNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP11.toIpPrefix())));
670 assertNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP12.toIpPrefix())));
671 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP13.toIpPrefix())));
672 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP14.toIpPrefix())));
673 assertNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV2, HOST_IP11.toIpPrefix())));
674 assertNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV2, HOST_IP12.toIpPrefix())));
675 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV2, HOST_IP13.toIpPrefix())));
676 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV2, HOST_IP14.toIpPrefix())));
677 assertEquals(2, BRIDGING_TABLE.size());
678 assertNotNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_UNTAGGED)));
679 assertNotNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV2, HOST_MAC, INTF_VLAN_UNTAGGED)));
Charles Chanf9a52702017-06-16 15:19:24 -0700680
681 // Update one of the host IP to different subnet
682 // Expect: update routing rule with new IP. No change to bridging rule.
683 hostHandler.processHostUpdatedEvent(new HostEvent(HostEvent.Type.HOST_UPDATED, host2, host3));
Charles Chan48f89a52017-08-18 17:11:34 -0700684 assertEquals(2, ROUTING_TABLE.size());
685 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP11.toIpPrefix())));
686 assertNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP21.toIpPrefix())));
687 assertNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV1, HOST_IP12.toIpPrefix())));
688 assertNotNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV2, HOST_IP11.toIpPrefix())));
689 assertNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV2, HOST_IP21.toIpPrefix())));
690 assertNull(ROUTING_TABLE.get(new MockRoutingTableKey(DEV2, HOST_IP12.toIpPrefix())));
691 assertEquals(2, BRIDGING_TABLE.size());
692 assertNotNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV1, HOST_MAC, INTF_VLAN_UNTAGGED)));
693 assertNotNull(BRIDGING_TABLE.get(new MockBridgingTableKey(DEV2, HOST_MAC, INTF_VLAN_UNTAGGED)));
Charles Chanf9a52702017-06-16 15:19:24 -0700694 }
695
696 @Test
697 public void testBridgingFwdObjBuilder() throws Exception {
698 assertNotNull(hostHandler.bridgingFwdObjBuilder(DEV2, HOST_MAC, HOST_VLAN_UNTAGGED, P1, false));
699 assertNull(hostHandler.bridgingFwdObjBuilder(DEV2, HOST_MAC, HOST_VLAN_UNTAGGED, P3, false));
Charles Chan2e2e3402017-06-19 14:00:53 -0700700 }
Ray Milkeyfacf2862017-08-03 11:58:29 -0700701}