blob: 3f8c0a8f85604740670718797c179687f0f11d61 [file] [log] [blame]
Yuta HIGUCHIb3b2ac42014-09-21 23:37:11 -07001package org.onlab.onos.net.device.impl;
2
tom0872a172014-09-23 11:24:26 -07003import com.google.common.collect.Iterables;
Ayaka Koshibea7f044e2014-09-23 16:56:20 -07004import com.google.common.collect.Sets;
tom0872a172014-09-23 11:24:26 -07005import com.hazelcast.config.Config;
6import com.hazelcast.core.Hazelcast;
Yuta HIGUCHIb113c452014-09-24 00:12:03 -07007
Yuta HIGUCHIb3b2ac42014-09-21 23:37:11 -07008import org.junit.After;
9import org.junit.Before;
Yuta HIGUCHIb3b2ac42014-09-21 23:37:11 -070010import org.junit.Test;
Yuta HIGUCHIb113c452014-09-24 00:12:03 -070011import org.onlab.onos.cluster.DefaultControllerNode;
tomb41d1ac2014-09-24 01:51:24 -070012import org.onlab.onos.cluster.MastershipServiceAdapter;
Ayaka Koshibea7f044e2014-09-23 16:56:20 -070013import org.onlab.onos.cluster.NodeId;
Yuta HIGUCHIb3b2ac42014-09-21 23:37:11 -070014import org.onlab.onos.event.Event;
Yuta HIGUCHIb113c452014-09-24 00:12:03 -070015import org.onlab.onos.event.EventDeliveryService;
tom0872a172014-09-23 11:24:26 -070016import org.onlab.onos.event.impl.TestEventDispatcher;
Yuta HIGUCHIb3b2ac42014-09-21 23:37:11 -070017import org.onlab.onos.net.Device;
18import org.onlab.onos.net.DeviceId;
19import org.onlab.onos.net.MastershipRole;
20import org.onlab.onos.net.Port;
21import org.onlab.onos.net.PortNumber;
22import org.onlab.onos.net.device.DefaultDeviceDescription;
23import org.onlab.onos.net.device.DefaultPortDescription;
24import org.onlab.onos.net.device.DeviceAdminService;
25import org.onlab.onos.net.device.DeviceDescription;
26import org.onlab.onos.net.device.DeviceEvent;
27import org.onlab.onos.net.device.DeviceListener;
28import org.onlab.onos.net.device.DeviceProvider;
29import org.onlab.onos.net.device.DeviceProviderRegistry;
30import org.onlab.onos.net.device.DeviceProviderService;
31import org.onlab.onos.net.device.DeviceService;
32import org.onlab.onos.net.device.PortDescription;
33import org.onlab.onos.net.provider.AbstractProvider;
34import org.onlab.onos.net.provider.ProviderId;
Yuta HIGUCHIb5df76d2014-09-27 20:54:00 -070035import org.onlab.onos.store.common.StoreManager;
Yuta HIGUCHIad4c2182014-09-29 11:16:23 -070036import org.onlab.onos.store.common.StoreService;
Yuta HIGUCHIb5df76d2014-09-27 20:54:00 -070037import org.onlab.onos.store.common.TestStoreManager;
Yuta HIGUCHIb3b2ac42014-09-21 23:37:11 -070038import org.onlab.onos.store.device.impl.DistributedDeviceStore;
Yuta HIGUCHIb113c452014-09-24 00:12:03 -070039import org.onlab.packet.IpPrefix;
Yuta HIGUCHIb3b2ac42014-09-21 23:37:11 -070040
41import java.util.ArrayList;
Yuta HIGUCHIb113c452014-09-24 00:12:03 -070042import java.util.HashSet;
Yuta HIGUCHIb3b2ac42014-09-21 23:37:11 -070043import java.util.Iterator;
44import java.util.List;
Yuta HIGUCHIb113c452014-09-24 00:12:03 -070045import java.util.Map.Entry;
Ayaka Koshibea7f044e2014-09-23 16:56:20 -070046import java.util.Set;
Yuta HIGUCHI3f6d29b2014-09-25 01:41:15 -070047import java.util.concurrent.BlockingQueue;
Yuta HIGUCHIb113c452014-09-24 00:12:03 -070048import java.util.concurrent.ConcurrentHashMap;
49import java.util.concurrent.ConcurrentMap;
Yuta HIGUCHI3f6d29b2014-09-25 01:41:15 -070050import java.util.concurrent.LinkedBlockingQueue;
51import java.util.concurrent.TimeUnit;
Yuta HIGUCHIb113c452014-09-24 00:12:03 -070052
Yuta HIGUCHIb3b2ac42014-09-21 23:37:11 -070053import static org.junit.Assert.*;
54import static org.onlab.onos.net.Device.Type.SWITCH;
55import static org.onlab.onos.net.DeviceId.deviceId;
56import static org.onlab.onos.net.device.DeviceEvent.Type.*;
57
Yuta HIGUCHIb4139d82014-09-23 18:41:33 -070058// FIXME This test is slow starting up Hazelcast on each test cases.
Yuta HIGUCHIb3b2ac42014-09-21 23:37:11 -070059// FIXME DistributedDeviceStore should have it's own test cases.
tomb41d1ac2014-09-24 01:51:24 -070060
Yuta HIGUCHIb3b2ac42014-09-21 23:37:11 -070061/**
62 * Test codifying the device service & device provider service contracts.
63 */
Yuta HIGUCHIb3b2ac42014-09-21 23:37:11 -070064public class DistributedDeviceManagerTest {
65
66 private static final ProviderId PID = new ProviderId("of", "foo");
67 private static final DeviceId DID1 = deviceId("of:foo");
68 private static final DeviceId DID2 = deviceId("of:bar");
69 private static final String MFR = "whitebox";
70 private static final String HW = "1.1.x";
71 private static final String SW1 = "3.8.1";
72 private static final String SW2 = "3.9.5";
73 private static final String SN = "43311-12345";
74
75 private static final PortNumber P1 = PortNumber.portNumber(1);
76 private static final PortNumber P2 = PortNumber.portNumber(2);
77 private static final PortNumber P3 = PortNumber.portNumber(3);
78
Yuta HIGUCHIb113c452014-09-24 00:12:03 -070079 private static final DefaultControllerNode SELF
80 = new DefaultControllerNode(new NodeId("foobar"),
81 IpPrefix.valueOf("127.0.0.1"));
82
83
Yuta HIGUCHIb3b2ac42014-09-21 23:37:11 -070084 private DeviceManager mgr;
85
tom0872a172014-09-23 11:24:26 -070086 protected StoreManager storeManager;
Yuta HIGUCHIb3b2ac42014-09-21 23:37:11 -070087 protected DeviceService service;
88 protected DeviceAdminService admin;
89 protected DeviceProviderRegistry registry;
90 protected DeviceProviderService providerService;
91 protected TestProvider provider;
92 protected TestListener listener = new TestListener();
93 private DistributedDeviceStore dstore;
Yuta HIGUCHIb113c452014-09-24 00:12:03 -070094 private TestMastershipManager masterManager;
95 private EventDeliveryService eventService;
Yuta HIGUCHIb3b2ac42014-09-21 23:37:11 -070096
97 @Before
98 public void setUp() {
99 mgr = new DeviceManager();
100 service = mgr;
101 admin = mgr;
102 registry = mgr;
Yuta HIGUCHIb4139d82014-09-23 18:41:33 -0700103 // TODO should find a way to clean Hazelcast instance without shutdown.
104 Config config = TestStoreManager.getTestConfig();
tom0872a172014-09-23 11:24:26 -0700105
Yuta HIGUCHIb113c452014-09-24 00:12:03 -0700106 masterManager = new TestMastershipManager();
107
tom0872a172014-09-23 11:24:26 -0700108 storeManager = new TestStoreManager(Hazelcast.newHazelcastInstance(config));
109 storeManager.activate();
110
Yuta HIGUCHI672488d2014-10-07 09:23:43 -0700111 dstore = new TestDistributedDeviceStore(storeManager);
Yuta HIGUCHIb3b2ac42014-09-21 23:37:11 -0700112 dstore.activate();
Yuta HIGUCHIb113c452014-09-24 00:12:03 -0700113
Yuta HIGUCHIb3b2ac42014-09-21 23:37:11 -0700114 mgr.store = dstore;
Yuta HIGUCHIb113c452014-09-24 00:12:03 -0700115 eventService = new TestEventDispatcher();
116 mgr.eventDispatcher = eventService;
117 mgr.mastershipService = masterManager;
Yuta HIGUCHIb3b2ac42014-09-21 23:37:11 -0700118 mgr.activate();
119
120 service.addListener(listener);
121
122 provider = new TestProvider();
123 providerService = registry.register(provider);
124 assertTrue("provider should be registered",
125 registry.getProviders().contains(provider.id()));
126 }
127
128 @After
129 public void tearDown() {
130 registry.unregister(provider);
131 assertFalse("provider should not be registered",
132 registry.getProviders().contains(provider.id()));
133 service.removeListener(listener);
134 mgr.deactivate();
135
136 dstore.deactivate();
tom0872a172014-09-23 11:24:26 -0700137 storeManager.deactivate();
Yuta HIGUCHIb3b2ac42014-09-21 23:37:11 -0700138 }
139
140 private void connectDevice(DeviceId deviceId, String swVersion) {
141 DeviceDescription description =
142 new DefaultDeviceDescription(deviceId.uri(), SWITCH, MFR,
143 HW, swVersion, SN);
144 providerService.deviceConnected(deviceId, description);
145 assertNotNull("device should be found", service.getDevice(DID1));
146 }
147
148 @Test
149 public void deviceConnected() {
150 assertNull("device should not be found", service.getDevice(DID1));
151 connectDevice(DID1, SW1);
152 validateEvents(DEVICE_ADDED);
153
154 assertEquals("only one device expected", 1, Iterables.size(service.getDevices()));
155 Iterator<Device> it = service.getDevices().iterator();
156 assertNotNull("one device expected", it.next());
157 assertFalse("only one device expected", it.hasNext());
158
159 assertEquals("incorrect device count", 1, service.getDeviceCount());
160 assertTrue("device should be available", service.isAvailable(DID1));
161 }
162
163 @Test
164 public void deviceDisconnected() {
165 connectDevice(DID1, SW1);
166 connectDevice(DID2, SW1);
Yuta HIGUCHIfec9e192014-09-28 14:58:02 -0700167 validateEvents(DEVICE_ADDED, DEVICE_ADDED);
Yuta HIGUCHIb3b2ac42014-09-21 23:37:11 -0700168 assertTrue("device should be available", service.isAvailable(DID1));
169
170 // Disconnect
171 providerService.deviceDisconnected(DID1);
172 assertNotNull("device should not be found", service.getDevice(DID1));
173 assertFalse("device should not be available", service.isAvailable(DID1));
174 validateEvents(DEVICE_AVAILABILITY_CHANGED);
175
176 // Reconnect
177 connectDevice(DID1, SW1);
178 validateEvents(DEVICE_AVAILABILITY_CHANGED);
179
180 assertEquals("incorrect device count", 2, service.getDeviceCount());
181 }
182
183 @Test
184 public void deviceUpdated() {
185 connectDevice(DID1, SW1);
Yuta HIGUCHIfec9e192014-09-28 14:58:02 -0700186 validateEvents(DEVICE_ADDED);
Yuta HIGUCHIb3b2ac42014-09-21 23:37:11 -0700187
188 connectDevice(DID1, SW2);
Yuta HIGUCHIfec9e192014-09-28 14:58:02 -0700189 validateEvents(DEVICE_UPDATED);
Yuta HIGUCHIb3b2ac42014-09-21 23:37:11 -0700190 }
191
192 @Test
193 public void getRole() {
194 connectDevice(DID1, SW1);
195 assertEquals("incorrect role", MastershipRole.MASTER, service.getRole(DID1));
196 }
197
198 @Test
Yuta HIGUCHIb3b2ac42014-09-21 23:37:11 -0700199 public void updatePorts() {
200 connectDevice(DID1, SW1);
201 List<PortDescription> pds = new ArrayList<>();
202 pds.add(new DefaultPortDescription(P1, true));
203 pds.add(new DefaultPortDescription(P2, true));
204 pds.add(new DefaultPortDescription(P3, true));
205 providerService.updatePorts(DID1, pds);
Yuta HIGUCHIfec9e192014-09-28 14:58:02 -0700206 validateEvents(DEVICE_ADDED, PORT_ADDED, PORT_ADDED, PORT_ADDED);
Yuta HIGUCHIb3b2ac42014-09-21 23:37:11 -0700207 pds.clear();
208
209 pds.add(new DefaultPortDescription(P1, false));
210 pds.add(new DefaultPortDescription(P3, true));
211 providerService.updatePorts(DID1, pds);
212 validateEvents(PORT_UPDATED, PORT_REMOVED);
213 }
214
215 @Test
216 public void updatePortStatus() {
217 connectDevice(DID1, SW1);
218 List<PortDescription> pds = new ArrayList<>();
219 pds.add(new DefaultPortDescription(P1, true));
220 pds.add(new DefaultPortDescription(P2, true));
221 providerService.updatePorts(DID1, pds);
Yuta HIGUCHIfec9e192014-09-28 14:58:02 -0700222 validateEvents(DEVICE_ADDED, PORT_ADDED, PORT_ADDED);
Yuta HIGUCHIb3b2ac42014-09-21 23:37:11 -0700223
224 providerService.portStatusChanged(DID1, new DefaultPortDescription(P1, false));
225 validateEvents(PORT_UPDATED);
226 providerService.portStatusChanged(DID1, new DefaultPortDescription(P1, false));
227 assertTrue("no events expected", listener.events.isEmpty());
228 }
229
230 @Test
231 public void getPorts() {
232 connectDevice(DID1, SW1);
233 List<PortDescription> pds = new ArrayList<>();
234 pds.add(new DefaultPortDescription(P1, true));
235 pds.add(new DefaultPortDescription(P2, true));
236 providerService.updatePorts(DID1, pds);
Yuta HIGUCHIfec9e192014-09-28 14:58:02 -0700237 validateEvents(DEVICE_ADDED, PORT_ADDED, PORT_ADDED);
Yuta HIGUCHIb3b2ac42014-09-21 23:37:11 -0700238 assertEquals("wrong port count", 2, service.getPorts(DID1).size());
239
240 Port port = service.getPort(DID1, P1);
241 assertEquals("incorrect port", P1, service.getPort(DID1, P1).number());
242 assertEquals("incorrect state", true, service.getPort(DID1, P1).isEnabled());
243 }
244
245 @Test
246 public void removeDevice() {
247 connectDevice(DID1, SW1);
248 connectDevice(DID2, SW2);
249 assertEquals("incorrect device count", 2, service.getDeviceCount());
250 admin.removeDevice(DID1);
Yuta HIGUCHIfec9e192014-09-28 14:58:02 -0700251 validateEvents(DEVICE_ADDED, DEVICE_ADDED, DEVICE_REMOVED);
Yuta HIGUCHIb3b2ac42014-09-21 23:37:11 -0700252 assertNull("device should not be found", service.getDevice(DID1));
253 assertNotNull("device should be found", service.getDevice(DID2));
254 assertEquals("incorrect device count", 1, service.getDeviceCount());
Yuta HIGUCHIb3b2ac42014-09-21 23:37:11 -0700255 }
256
257 protected void validateEvents(Enum... types) {
Yuta HIGUCHI3f6d29b2014-09-25 01:41:15 -0700258 for (Enum type : types) {
259 try {
260 Event event = listener.events.poll(1, TimeUnit.SECONDS);
261 assertNotNull("Timed out waiting for " + event, event);
262 assertEquals("incorrect event type", type, event.type());
263 } catch (InterruptedException e) {
264 fail("Unexpected interrupt");
265 }
Yuta HIGUCHIb3b2ac42014-09-21 23:37:11 -0700266 }
Yuta HIGUCHI3f6d29b2014-09-25 01:41:15 -0700267 assertTrue("Unexpected events left", listener.events.isEmpty());
Yuta HIGUCHIb3b2ac42014-09-21 23:37:11 -0700268 listener.events.clear();
269 }
270
271
272 private class TestProvider extends AbstractProvider implements DeviceProvider {
273 private Device deviceReceived;
274 private MastershipRole roleReceived;
275
276 public TestProvider() {
277 super(PID);
278 }
279
280 @Override
281 public void triggerProbe(Device device) {
282 }
283
284 @Override
285 public void roleChanged(Device device, MastershipRole newRole) {
286 deviceReceived = device;
287 roleReceived = newRole;
288 }
289 }
290
291 private static class TestListener implements DeviceListener {
Yuta HIGUCHI3f6d29b2014-09-25 01:41:15 -0700292 final BlockingQueue<DeviceEvent> events = new LinkedBlockingQueue<>();
Yuta HIGUCHIb3b2ac42014-09-21 23:37:11 -0700293
294 @Override
295 public void event(DeviceEvent event) {
296 events.add(event);
297 }
298 }
299
tom85ff08b2014-09-22 17:14:18 -0700300 private class TestDistributedDeviceStore extends DistributedDeviceStore {
Yuta HIGUCHIb113c452014-09-24 00:12:03 -0700301
Yuta HIGUCHI672488d2014-10-07 09:23:43 -0700302 public TestDistributedDeviceStore(StoreService storeService) {
Yuta HIGUCHIad4c2182014-09-29 11:16:23 -0700303 this.storeService = storeService;
tom0872a172014-09-23 11:24:26 -0700304 }
305 }
306
Yuta HIGUCHIb113c452014-09-24 00:12:03 -0700307 private static class TestMastershipManager extends MastershipServiceAdapter {
Yuta HIGUCHIb4139d82014-09-23 18:41:33 -0700308
Yuta HIGUCHIb113c452014-09-24 00:12:03 -0700309 private ConcurrentMap<DeviceId, NodeId> masters = new ConcurrentHashMap<>();
310
311 public TestMastershipManager() {
312 // SELF master of all initially
313 masters.put(DID1, SELF.id());
314 masters.put(DID1, SELF.id());
315 }
Ayaka Koshibea7f044e2014-09-23 16:56:20 -0700316 @Override
tomb41d1ac2014-09-24 01:51:24 -0700317 public MastershipRole getLocalRole(DeviceId deviceId) {
318 return MastershipRole.MASTER;
Ayaka Koshibea7f044e2014-09-23 16:56:20 -0700319 }
320
321 @Override
322 public Set<DeviceId> getDevicesOf(NodeId nodeId) {
Yuta HIGUCHIb113c452014-09-24 00:12:03 -0700323 HashSet<DeviceId> set = Sets.newHashSet();
324 for (Entry<DeviceId, NodeId> e : masters.entrySet()) {
325 if (e.getValue().equals(nodeId)) {
326 set.add(e.getKey());
327 }
328 }
329 return set;
Ayaka Koshibea7f044e2014-09-23 16:56:20 -0700330 }
331
332 @Override
333 public MastershipRole requestRoleFor(DeviceId deviceId) {
Yuta HIGUCHIb113c452014-09-24 00:12:03 -0700334 if (SELF.id().equals(masters.get(deviceId))) {
335 return MastershipRole.MASTER;
336 } else {
337 return MastershipRole.STANDBY;
338 }
339 }
340
341 @Override
342 public void relinquishMastership(DeviceId deviceId) {
343 masters.remove(deviceId, SELF.id());
Ayaka Koshibea7f044e2014-09-23 16:56:20 -0700344 }
Ayaka Koshibea7f044e2014-09-23 16:56:20 -0700345 }
Yuta HIGUCHIb3b2ac42014-09-21 23:37:11 -0700346}