blob: d817cf968097f7b96c5524b497b4a953fab215fd [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;
tom85ff08b2014-09-22 17:14:18 -07007import com.hazelcast.core.HazelcastInstance;
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;
tomb41d1ac2014-09-24 01:51:24 -070011import org.onlab.onos.cluster.MastershipServiceAdapter;
Ayaka Koshibea7f044e2014-09-23 16:56:20 -070012import org.onlab.onos.cluster.NodeId;
Yuta HIGUCHIb3b2ac42014-09-21 23:37:11 -070013import org.onlab.onos.event.Event;
tom0872a172014-09-23 11:24:26 -070014import org.onlab.onos.event.impl.TestEventDispatcher;
Yuta HIGUCHIb3b2ac42014-09-21 23:37:11 -070015import org.onlab.onos.net.Device;
16import org.onlab.onos.net.DeviceId;
17import org.onlab.onos.net.MastershipRole;
18import org.onlab.onos.net.Port;
19import org.onlab.onos.net.PortNumber;
20import org.onlab.onos.net.device.DefaultDeviceDescription;
21import org.onlab.onos.net.device.DefaultPortDescription;
22import org.onlab.onos.net.device.DeviceAdminService;
23import org.onlab.onos.net.device.DeviceDescription;
24import org.onlab.onos.net.device.DeviceEvent;
25import org.onlab.onos.net.device.DeviceListener;
26import org.onlab.onos.net.device.DeviceProvider;
27import org.onlab.onos.net.device.DeviceProviderRegistry;
28import org.onlab.onos.net.device.DeviceProviderService;
29import org.onlab.onos.net.device.DeviceService;
30import org.onlab.onos.net.device.PortDescription;
31import org.onlab.onos.net.provider.AbstractProvider;
32import org.onlab.onos.net.provider.ProviderId;
tom0755a362014-09-24 11:54:43 -070033import org.onlab.onos.store.common.StoreService;
Yuta HIGUCHIb3b2ac42014-09-21 23:37:11 -070034import org.onlab.onos.store.device.impl.DistributedDeviceStore;
tom0872a172014-09-23 11:24:26 -070035import org.onlab.onos.store.impl.StoreManager;
Yuta HIGUCHIb4139d82014-09-23 18:41:33 -070036import org.onlab.onos.store.impl.TestStoreManager;
Yuta HIGUCHIb3b2ac42014-09-21 23:37:11 -070037
38import java.util.ArrayList;
39import java.util.Iterator;
40import java.util.List;
Ayaka Koshibea7f044e2014-09-23 16:56:20 -070041import java.util.Set;
Yuta HIGUCHIb3b2ac42014-09-21 23:37:11 -070042import static org.junit.Assert.*;
43import static org.onlab.onos.net.Device.Type.SWITCH;
44import static org.onlab.onos.net.DeviceId.deviceId;
45import static org.onlab.onos.net.device.DeviceEvent.Type.*;
46
Yuta HIGUCHIb4139d82014-09-23 18:41:33 -070047// FIXME This test is slow starting up Hazelcast on each test cases.
Yuta HIGUCHIb3b2ac42014-09-21 23:37:11 -070048// FIXME DistributedDeviceStore should have it's own test cases.
tomb41d1ac2014-09-24 01:51:24 -070049
Yuta HIGUCHIb3b2ac42014-09-21 23:37:11 -070050/**
51 * Test codifying the device service & device provider service contracts.
52 */
Yuta HIGUCHIb3b2ac42014-09-21 23:37:11 -070053public class DistributedDeviceManagerTest {
54
55 private static final ProviderId PID = new ProviderId("of", "foo");
56 private static final DeviceId DID1 = deviceId("of:foo");
57 private static final DeviceId DID2 = deviceId("of:bar");
58 private static final String MFR = "whitebox";
59 private static final String HW = "1.1.x";
60 private static final String SW1 = "3.8.1";
61 private static final String SW2 = "3.9.5";
62 private static final String SN = "43311-12345";
63
64 private static final PortNumber P1 = PortNumber.portNumber(1);
65 private static final PortNumber P2 = PortNumber.portNumber(2);
66 private static final PortNumber P3 = PortNumber.portNumber(3);
67
68 private DeviceManager mgr;
69
tom0872a172014-09-23 11:24:26 -070070 protected StoreManager storeManager;
Yuta HIGUCHIb3b2ac42014-09-21 23:37:11 -070071 protected DeviceService service;
72 protected DeviceAdminService admin;
73 protected DeviceProviderRegistry registry;
74 protected DeviceProviderService providerService;
75 protected TestProvider provider;
76 protected TestListener listener = new TestListener();
77 private DistributedDeviceStore dstore;
78
79 @Before
80 public void setUp() {
81 mgr = new DeviceManager();
82 service = mgr;
83 admin = mgr;
84 registry = mgr;
Yuta HIGUCHIb4139d82014-09-23 18:41:33 -070085 // TODO should find a way to clean Hazelcast instance without shutdown.
86 Config config = TestStoreManager.getTestConfig();
tom0872a172014-09-23 11:24:26 -070087
88 storeManager = new TestStoreManager(Hazelcast.newHazelcastInstance(config));
89 storeManager.activate();
90
91 dstore = new TestDistributedDeviceStore(storeManager);
Yuta HIGUCHIb3b2ac42014-09-21 23:37:11 -070092 dstore.activate();
93 mgr.store = dstore;
94 mgr.eventDispatcher = new TestEventDispatcher();
Ayaka Koshibea7f044e2014-09-23 16:56:20 -070095 mgr.mastershipService = new TestMastershipService();
Yuta HIGUCHIb3b2ac42014-09-21 23:37:11 -070096 mgr.activate();
97
98 service.addListener(listener);
99
100 provider = new TestProvider();
101 providerService = registry.register(provider);
102 assertTrue("provider should be registered",
103 registry.getProviders().contains(provider.id()));
104 }
105
106 @After
107 public void tearDown() {
108 registry.unregister(provider);
109 assertFalse("provider should not be registered",
110 registry.getProviders().contains(provider.id()));
111 service.removeListener(listener);
112 mgr.deactivate();
113
114 dstore.deactivate();
tom0872a172014-09-23 11:24:26 -0700115 storeManager.deactivate();
Yuta HIGUCHIb3b2ac42014-09-21 23:37:11 -0700116 }
117
118 private void connectDevice(DeviceId deviceId, String swVersion) {
119 DeviceDescription description =
120 new DefaultDeviceDescription(deviceId.uri(), SWITCH, MFR,
121 HW, swVersion, SN);
122 providerService.deviceConnected(deviceId, description);
123 assertNotNull("device should be found", service.getDevice(DID1));
124 }
125
126 @Test
127 public void deviceConnected() {
128 assertNull("device should not be found", service.getDevice(DID1));
129 connectDevice(DID1, SW1);
130 validateEvents(DEVICE_ADDED);
131
132 assertEquals("only one device expected", 1, Iterables.size(service.getDevices()));
133 Iterator<Device> it = service.getDevices().iterator();
134 assertNotNull("one device expected", it.next());
135 assertFalse("only one device expected", it.hasNext());
136
137 assertEquals("incorrect device count", 1, service.getDeviceCount());
138 assertTrue("device should be available", service.isAvailable(DID1));
139 }
140
141 @Test
142 public void deviceDisconnected() {
143 connectDevice(DID1, SW1);
144 connectDevice(DID2, SW1);
145 validateEvents(DEVICE_ADDED, DEVICE_ADDED);
146 assertTrue("device should be available", service.isAvailable(DID1));
147
148 // Disconnect
149 providerService.deviceDisconnected(DID1);
150 assertNotNull("device should not be found", service.getDevice(DID1));
151 assertFalse("device should not be available", service.isAvailable(DID1));
152 validateEvents(DEVICE_AVAILABILITY_CHANGED);
153
154 // Reconnect
155 connectDevice(DID1, SW1);
156 validateEvents(DEVICE_AVAILABILITY_CHANGED);
157
158 assertEquals("incorrect device count", 2, service.getDeviceCount());
159 }
160
161 @Test
162 public void deviceUpdated() {
163 connectDevice(DID1, SW1);
164 validateEvents(DEVICE_ADDED);
165
166 connectDevice(DID1, SW2);
167 validateEvents(DEVICE_UPDATED);
168 }
169
170 @Test
171 public void getRole() {
172 connectDevice(DID1, SW1);
173 assertEquals("incorrect role", MastershipRole.MASTER, service.getRole(DID1));
174 }
175
176 @Test
Yuta HIGUCHIb3b2ac42014-09-21 23:37:11 -0700177 public void updatePorts() {
178 connectDevice(DID1, SW1);
179 List<PortDescription> pds = new ArrayList<>();
180 pds.add(new DefaultPortDescription(P1, true));
181 pds.add(new DefaultPortDescription(P2, true));
182 pds.add(new DefaultPortDescription(P3, true));
183 providerService.updatePorts(DID1, pds);
184 validateEvents(DEVICE_ADDED, PORT_ADDED, PORT_ADDED, PORT_ADDED);
185 pds.clear();
186
187 pds.add(new DefaultPortDescription(P1, false));
188 pds.add(new DefaultPortDescription(P3, true));
189 providerService.updatePorts(DID1, pds);
190 validateEvents(PORT_UPDATED, PORT_REMOVED);
191 }
192
193 @Test
194 public void updatePortStatus() {
195 connectDevice(DID1, SW1);
196 List<PortDescription> pds = new ArrayList<>();
197 pds.add(new DefaultPortDescription(P1, true));
198 pds.add(new DefaultPortDescription(P2, true));
199 providerService.updatePorts(DID1, pds);
200 validateEvents(DEVICE_ADDED, PORT_ADDED, PORT_ADDED);
201
202 providerService.portStatusChanged(DID1, new DefaultPortDescription(P1, false));
203 validateEvents(PORT_UPDATED);
204 providerService.portStatusChanged(DID1, new DefaultPortDescription(P1, false));
205 assertTrue("no events expected", listener.events.isEmpty());
206 }
207
208 @Test
209 public void getPorts() {
210 connectDevice(DID1, SW1);
211 List<PortDescription> pds = new ArrayList<>();
212 pds.add(new DefaultPortDescription(P1, true));
213 pds.add(new DefaultPortDescription(P2, true));
214 providerService.updatePorts(DID1, pds);
215 validateEvents(DEVICE_ADDED, PORT_ADDED, PORT_ADDED);
216 assertEquals("wrong port count", 2, service.getPorts(DID1).size());
217
218 Port port = service.getPort(DID1, P1);
219 assertEquals("incorrect port", P1, service.getPort(DID1, P1).number());
220 assertEquals("incorrect state", true, service.getPort(DID1, P1).isEnabled());
221 }
222
223 @Test
224 public void removeDevice() {
225 connectDevice(DID1, SW1);
226 connectDevice(DID2, SW2);
227 assertEquals("incorrect device count", 2, service.getDeviceCount());
228 admin.removeDevice(DID1);
229 assertNull("device should not be found", service.getDevice(DID1));
230 assertNotNull("device should be found", service.getDevice(DID2));
231 assertEquals("incorrect device count", 1, service.getDeviceCount());
232
233 }
234
235 protected void validateEvents(Enum... types) {
236 int i = 0;
237 assertEquals("wrong events received", types.length, listener.events.size());
238 for (Event event : listener.events) {
239 assertEquals("incorrect event type", types[i], event.type());
240 i++;
241 }
242 listener.events.clear();
243 }
244
245
246 private class TestProvider extends AbstractProvider implements DeviceProvider {
247 private Device deviceReceived;
248 private MastershipRole roleReceived;
249
250 public TestProvider() {
251 super(PID);
252 }
253
254 @Override
255 public void triggerProbe(Device device) {
256 }
257
258 @Override
259 public void roleChanged(Device device, MastershipRole newRole) {
260 deviceReceived = device;
261 roleReceived = newRole;
262 }
263 }
264
265 private static class TestListener implements DeviceListener {
266 final List<DeviceEvent> events = new ArrayList<>();
267
268 @Override
269 public void event(DeviceEvent event) {
270 events.add(event);
271 }
272 }
273
tom85ff08b2014-09-22 17:14:18 -0700274 private class TestDistributedDeviceStore extends DistributedDeviceStore {
tom0872a172014-09-23 11:24:26 -0700275 public TestDistributedDeviceStore(StoreService storeService) {
276 this.storeService = storeService;
277 }
278 }
279
tomb41d1ac2014-09-24 01:51:24 -0700280 private static class TestMastershipService extends MastershipServiceAdapter {
Yuta HIGUCHIb4139d82014-09-23 18:41:33 -0700281
Ayaka Koshibea7f044e2014-09-23 16:56:20 -0700282 @Override
tomb41d1ac2014-09-24 01:51:24 -0700283 public MastershipRole getLocalRole(DeviceId deviceId) {
284 return MastershipRole.MASTER;
Ayaka Koshibea7f044e2014-09-23 16:56:20 -0700285 }
286
287 @Override
288 public Set<DeviceId> getDevicesOf(NodeId nodeId) {
289 return Sets.newHashSet(DID1, DID2);
290 }
291
292 @Override
293 public MastershipRole requestRoleFor(DeviceId deviceId) {
294 return MastershipRole.MASTER;
295 }
Ayaka Koshibea7f044e2014-09-23 16:56:20 -0700296 }
Yuta HIGUCHIb3b2ac42014-09-21 23:37:11 -0700297}