blob: 5d50eac8783ca0f55155fb47d120e682d0f251a6 [file] [log] [blame]
Ray Milkey4f5de002014-12-17 19:26:11 -08001/*
2 * Copyright 2014 Open Networking Laboratory
3 *
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 */
16package org.onosproject.rest;
17
18import java.util.HashMap;
19import java.util.HashSet;
20import java.util.Set;
21
22import org.hamcrest.Description;
23import org.hamcrest.TypeSafeMatcher;
24import org.junit.After;
25import org.junit.Before;
26import org.junit.Test;
27import org.onlab.osgi.ServiceDirectory;
28import org.onlab.osgi.TestServiceDirectory;
29import org.onlab.packet.MacAddress;
30import org.onlab.rest.BaseResource;
31import org.onosproject.codec.CodecService;
32import org.onosproject.codec.impl.CodecManager;
33import org.onosproject.core.DefaultGroupId;
34import org.onosproject.core.GroupId;
35import org.onosproject.net.DefaultDevice;
36import org.onosproject.net.Device;
37import org.onosproject.net.DeviceId;
38import org.onosproject.net.device.DeviceService;
39import org.onosproject.net.flow.DefaultTrafficSelector;
40import org.onosproject.net.flow.DefaultTrafficTreatment;
41import org.onosproject.net.flow.FlowEntry;
42import org.onosproject.net.flow.FlowId;
43import org.onosproject.net.flow.FlowRuleService;
44import org.onosproject.net.flow.TrafficSelector;
45import org.onosproject.net.flow.TrafficTreatment;
46import org.onosproject.net.flow.criteria.Criterion;
47import org.onosproject.net.flow.instructions.Instruction;
48import org.onosproject.net.flow.instructions.L0ModificationInstruction;
49
50import com.eclipsesource.json.JsonArray;
51import com.eclipsesource.json.JsonObject;
52import com.eclipsesource.json.JsonValue;
53import com.google.common.collect.ImmutableSet;
54import com.sun.jersey.api.client.UniformInterfaceException;
55import com.sun.jersey.api.client.WebResource;
56import com.sun.jersey.test.framework.JerseyTest;
57
58import static org.easymock.EasyMock.anyObject;
59import static org.easymock.EasyMock.createMock;
60import static org.easymock.EasyMock.expect;
61import static org.easymock.EasyMock.replay;
62import static org.easymock.EasyMock.verify;
63import static org.hamcrest.Matchers.containsString;
64import static org.hamcrest.Matchers.hasSize;
65import static org.hamcrest.Matchers.is;
66import static org.hamcrest.Matchers.not;
67import static org.hamcrest.Matchers.notNullValue;
68import static org.junit.Assert.assertThat;
69import static org.junit.Assert.fail;
70
71/**
72 * Unit tests for Flows REST APIs.
73 */
74public class FlowsResourceTest extends JerseyTest {
75 final FlowRuleService mockFlowService = createMock(FlowRuleService.class);
76 final HashMap<DeviceId, Set<FlowEntry>> rules = new HashMap<>();
77
78 final DeviceService mockDeviceService = createMock(DeviceService.class);
79
80 final DeviceId deviceId1 = DeviceId.deviceId("1");
81 final DeviceId deviceId2 = DeviceId.deviceId("2");
82 final DeviceId deviceId3 = DeviceId.deviceId("3");
83 final Device device1 = new DefaultDevice(null, deviceId1, Device.Type.OTHER,
84 "", "", "", "", null);
85 final Device device2 = new DefaultDevice(null, deviceId2, Device.Type.OTHER,
86 "", "", "", "", null);
87
88 final MockFlowEntry flow1 = new MockFlowEntry(deviceId1, 1);
89 final MockFlowEntry flow2 = new MockFlowEntry(deviceId1, 2);
90
91 final MockFlowEntry flow3 = new MockFlowEntry(deviceId2, 3);
92 final MockFlowEntry flow4 = new MockFlowEntry(deviceId2, 4);
93
94 final MockFlowEntry flow5 = new MockFlowEntry(deviceId2, 5);
95 final MockFlowEntry flow6 = new MockFlowEntry(deviceId2, 6);
96
97 /**
98 * Mock class for a flow entry.
99 */
100 private static class MockFlowEntry implements FlowEntry {
101 final DeviceId deviceId;
102 final long baseValue;
103 TrafficTreatment treatment;
104 TrafficSelector selector;
105
106 public MockFlowEntry(DeviceId deviceId, long id) {
107 this.deviceId = deviceId;
108 this.baseValue = id * 100;
109 }
110
111 @Override
112 public FlowEntryState state() {
113 return FlowEntryState.ADDED;
114 }
115
116 @Override
117 public long life() {
118 return baseValue + 11;
119 }
120
121 @Override
122 public long packets() {
123 return baseValue + 22;
124 }
125
126 @Override
127 public long bytes() {
128 return baseValue + 33;
129 }
130
131 @Override
132 public long lastSeen() {
133 return baseValue + 44;
134 }
135
136 @Override
137 public int errType() {
138 return 0;
139 }
140
141 @Override
142 public int errCode() {
143 return 0;
144 }
145
146 @Override
147 public FlowId id() {
148 final long id = baseValue + 55;
149 return FlowId.valueOf(id);
150 }
151
152 @Override
153 public short appId() {
154 return 2;
155 }
156
157 @Override
158 public GroupId groupId() {
159 return new DefaultGroupId(3);
160 }
161
162 @Override
163 public int priority() {
164 return (int) (baseValue + 66);
165 }
166
167 @Override
168 public DeviceId deviceId() {
169 return deviceId;
170 }
171
172 @Override
173 public TrafficSelector selector() {
174 return selector;
175 }
176
177 @Override
178 public TrafficTreatment treatment() {
179 return treatment;
180 }
181
182 @Override
183 public int timeout() {
184 return (int) (baseValue + 77);
185 }
186
187 @Override
188 public boolean isPermanent() {
189 return false;
190 }
191 }
192
193 public FlowsResourceTest() {
194 super("org.onosproject.rest");
195 }
196
197 /**
198 * Populates some flows used as testing data.
199 */
200 private void setupMockFlows() {
201 flow2.treatment = DefaultTrafficTreatment.builder()
202 .add(new L0ModificationInstruction.ModLambdaInstruction(
203 L0ModificationInstruction.L0SubType.LAMBDA, (short) 4))
204 .add(new L0ModificationInstruction.ModLambdaInstruction(
205 L0ModificationInstruction.L0SubType.LAMBDA, (short) 5))
206 .setEthDst(MacAddress.BROADCAST)
207 .build();
208 flow2.selector = DefaultTrafficSelector.builder()
209 .matchEthType((short) 3)
210 .matchIPProtocol((byte) 9)
211 .build();
212 flow4.treatment = DefaultTrafficTreatment.builder()
213 .add(new L0ModificationInstruction.ModLambdaInstruction(
214 L0ModificationInstruction.L0SubType.LAMBDA, (short) 6))
215 .build();
216 final Set<FlowEntry> flows1 = new HashSet<>();
217 flows1.add(flow1);
218 flows1.add(flow2);
219
220 final Set<FlowEntry> flows2 = new HashSet<>();
221 flows1.add(flow3);
222 flows1.add(flow4);
223
224 rules.put(deviceId1, flows1);
225 rules.put(deviceId2, flows2);
226
227 expect(mockFlowService.getFlowEntries(deviceId1))
228 .andReturn(rules.get(deviceId1)).anyTimes();
229 expect(mockFlowService.getFlowEntries(deviceId2))
230 .andReturn(rules.get(deviceId2)).anyTimes();
231 }
232
233 /**
234 * Sets up the global values for all the tests.
235 */
236 @Before
237 public void setUp() {
238 // Mock device service
239 expect(mockDeviceService.getDevice(deviceId1))
240 .andReturn(device1);
241 expect(mockDeviceService.getDevice(deviceId2))
242 .andReturn(device2);
243 expect(mockDeviceService.getDevices())
244 .andReturn(ImmutableSet.of(device1, device2));
245
246 // Register the services needed for the test
247 final CodecManager codecService = new CodecManager();
248 codecService.activate();
249 ServiceDirectory testDirectory =
250 new TestServiceDirectory()
251 .add(FlowRuleService.class, mockFlowService)
252 .add(DeviceService.class, mockDeviceService)
253 .add(CodecService.class, codecService);
254
255 BaseResource.setServiceDirectory(testDirectory);
256 }
257
258 /**
259 * Cleans up and verifies the mocks.
260 *
261 * @throws Exception if the super teardown fails.
262 */
263 @After
264 public void tearDown() throws Exception {
265 super.tearDown();
266 verify(mockFlowService);
267 }
268
269 /**
270 * Hamcrest matcher to check that a flow representation in JSON matches
271 * the actual flow entry.
272 */
273 public static class FlowJsonMatcher extends TypeSafeMatcher<JsonObject> {
274 private final FlowEntry flow;
275 private String reason = "";
276
277 public FlowJsonMatcher(FlowEntry flowValue) {
278 flow = flowValue;
279 }
280
281 @Override
282 public boolean matchesSafely(JsonObject jsonFlow) {
283 // check id
284 final String jsonId = jsonFlow.get("id").asString();
285 final String flowId = Long.toString(flow.id().value());
286 if (!jsonId.equals(flowId)) {
287 reason = "id " + flow.id().toString();
288 return false;
289 }
290
291 // check application id
292 final int jsonAppId = jsonFlow.get("appId").asInt();
293 if (jsonAppId != flow.appId()) {
294 reason = "appId " + Short.toString(flow.appId());
295 return false;
296 }
297
298 // check device id
299 final String jsonDeviceId = jsonFlow.get("deviceId").asString();
300 if (!jsonDeviceId.equals(flow.deviceId().toString())) {
301 reason = "deviceId " + flow.deviceId();
302 return false;
303 }
304
305 // check treatment and instructions array
306 if (flow.treatment() != null) {
307 final JsonObject jsonTreatment = jsonFlow.get("treatment").asObject();
308 final JsonArray jsonInstructions = jsonTreatment.get("instructions").asArray();
309 if (flow.treatment().instructions().size() != jsonInstructions.size()) {
310 reason = "instructions array size of " +
311 Integer.toString(flow.treatment().instructions().size());
312 return false;
313 }
314 for (final Instruction instruction : flow.treatment().instructions()) {
315 boolean instructionFound = false;
316 final String instructionString = instruction.toString();
317 for (int instructionIndex = 0; instructionIndex < jsonInstructions.size(); instructionIndex++) {
318 final JsonValue value = jsonInstructions.get(instructionIndex);
319 if (value.asString().equals(instructionString)) {
320 instructionFound = true;
321 }
322 }
323 if (!instructionFound) {
324 reason = "instruction " + instructionString;
325 return false;
326 }
327 }
328 }
329
330 // check selector and criteria array
331 if (flow.selector() != null) {
332 final JsonObject jsonTreatment = jsonFlow.get("selector").asObject();
333 final JsonArray jsonCriteria = jsonTreatment.get("criteria").asArray();
334 if (flow.selector().criteria().size() != jsonCriteria.size()) {
335 reason = "criteria array size of " +
336 Integer.toString(flow.selector().criteria().size());
337 return false;
338 }
339 for (final Criterion criterion : flow.selector().criteria()) {
340 boolean criterionFound = false;
341 final String criterionString = criterion.toString();
342 for (int criterionIndex = 0; criterionIndex < jsonCriteria.size(); criterionIndex++) {
343 final JsonValue value = jsonCriteria.get(criterionIndex);
344 if (value.asString().equals(criterionString)) {
345 criterionFound = true;
346 }
347 }
348 if (!criterionFound) {
349 reason = "criterion " + criterionString;
350 return false;
351 }
352 }
353 }
354
355 return true;
356 }
357
358 @Override
359 public void describeTo(Description description) {
360 description.appendText(reason);
361 }
362 }
363
364 /**
365 * Factory to allocate a flow matcher.
366 *
367 * @param flow flow object we are looking for
368 * @return matcher
369 */
370 private static FlowJsonMatcher matchesFlow(FlowEntry flow) {
371 return new FlowJsonMatcher(flow);
372 }
373
374 /**
375 * Hamcrest matcher to check that a flow is represented properly in a JSON
376 * array of flows.
377 */
378 public static class FlowJsonArrayMatcher extends TypeSafeMatcher<JsonArray> {
379 private final FlowEntry flow;
380 private String reason = "";
381
382 public FlowJsonArrayMatcher(FlowEntry flowValue) {
383 flow = flowValue;
384 }
385
386 @Override
387 public boolean matchesSafely(JsonArray json) {
388 boolean flowFound = false;
389
390 for (int jsonFlowIndex = 0; jsonFlowIndex < json.size();
391 jsonFlowIndex++) {
392
393 final JsonObject jsonFlow = json.get(jsonFlowIndex).asObject();
394
395 final String flowId = Long.toString(flow.id().value());
396 final String jsonFlowId = jsonFlow.get("id").asString();
397 if (jsonFlowId.equals(flowId)) {
398 flowFound = true;
399
400 // We found the correct flow, check attribute values
401 assertThat(jsonFlow, matchesFlow(flow));
402 }
403 }
404 if (!flowFound) {
405 reason = "Flow with id " + flow.id().toString() + " not found";
406 return false;
407 } else {
408 return true;
409 }
410 }
411
412 @Override
413 public void describeTo(Description description) {
414 description.appendText(reason);
415 }
416 }
417
418 /**
419 * Factory to allocate a flow array matcher.
420 *
421 * @param flow flow object we are looking for
422 * @return matcher
423 */
424 private static FlowJsonArrayMatcher hasFlow(FlowEntry flow) {
425 return new FlowJsonArrayMatcher(flow);
426 }
427
428 /**
429 * Tests the result of the rest api GET when there are no flows.
430 */
431 @Test
432 public void testFlowsEmptyArray() {
433 expect(mockFlowService.getFlowEntries(deviceId1))
434 .andReturn(null).anyTimes();
435 expect(mockFlowService.getFlowEntries(deviceId2))
436 .andReturn(null).anyTimes();
437 replay(mockFlowService);
438 replay(mockDeviceService);
439 final WebResource rs = resource();
440 final String response = rs.path("flows").get(String.class);
441 assertThat(response, is("{\"flows\":[]}"));
442 }
443
444 /**
445 * Tests the result of the rest api GET when there are active flows.
446 */
447 @Test
448 public void testFlowsPopulatedArray() {
449 setupMockFlows();
450 replay(mockFlowService);
451 replay(mockDeviceService);
452 final WebResource rs = resource();
453 final String response = rs.path("flows").get(String.class);
454 final JsonObject result = JsonObject.readFrom(response);
455 assertThat(result, notNullValue());
456
457 assertThat(result.names(), hasSize(1));
458 assertThat(result.names().get(0), is("flows"));
459 final JsonArray jsonFlows = result.get("flows").asArray();
460 assertThat(jsonFlows, notNullValue());
461 assertThat(jsonFlows, hasFlow(flow1));
462 assertThat(jsonFlows, hasFlow(flow2));
463 assertThat(jsonFlows, hasFlow(flow3));
464 assertThat(jsonFlows, hasFlow(flow4));
465 }
466
467 /**
468 * Tests the result of a rest api GET for a device.
469 */
470 @Test
471 public void testFlowsSingleDevice() {
472 setupMockFlows();
473 final Set<FlowEntry> flows = new HashSet<>();
474 flows.add(flow5);
475 flows.add(flow6);
476 expect(mockFlowService.getFlowEntries(anyObject()))
477 .andReturn(flows).anyTimes();
478 replay(mockFlowService);
479 replay(mockDeviceService);
480 final WebResource rs = resource();
481 final String response = rs.path("flows/" + deviceId3).get(String.class);
482 final JsonObject result = JsonObject.readFrom(response);
483 assertThat(result, notNullValue());
484
485 assertThat(result.names(), hasSize(1));
486 assertThat(result.names().get(0), is("flows"));
487 final JsonArray jsonFlows = result.get("flows").asArray();
488 assertThat(jsonFlows, notNullValue());
489 assertThat(jsonFlows, hasFlow(flow5));
490 assertThat(jsonFlows, hasFlow(flow6));
491 }
492
493 /**
494 * Tests the result of a rest api GET for a device.
495 */
496 @Test
497 public void testFlowsSingleDeviceWithFlowId() {
498 setupMockFlows();
499 final Set<FlowEntry> flows = new HashSet<>();
500 flows.add(flow5);
501 flows.add(flow6);
502 expect(mockFlowService.getFlowEntries(anyObject()))
503 .andReturn(flows).anyTimes();
504 replay(mockFlowService);
505 replay(mockDeviceService);
506 final WebResource rs = resource();
507 final String response = rs.path("flows/" + deviceId3 + "/"
508 + Long.toString(flow5.id().value())).get(String.class);
509 final JsonObject result = JsonObject.readFrom(response);
510 assertThat(result, notNullValue());
511
512 assertThat(result.names(), hasSize(1));
513 assertThat(result.names().get(0), is("flows"));
514 final JsonArray jsonFlows = result.get("flows").asArray();
515 assertThat(jsonFlows, notNullValue());
516 assertThat(jsonFlows, hasFlow(flow5));
517 assertThat(jsonFlows, not(hasFlow(flow6)));
518 }
519
520 /**
521 * Tests that a fetch of a non-existent device object throws an exception.
522 */
523 @Test
524 public void testBadGet() {
525 expect(mockFlowService.getFlowEntries(deviceId1))
526 .andReturn(null).anyTimes();
527 expect(mockFlowService.getFlowEntries(deviceId2))
528 .andReturn(null).anyTimes();
529 replay(mockFlowService);
530 replay(mockDeviceService);
531
532 WebResource rs = resource();
533 try {
534 rs.path("flows/0").get(String.class);
535 fail("Fetch of non-existent device did not throw an exception");
536 } catch (UniformInterfaceException ex) {
537 assertThat(ex.getMessage(),
538 containsString("returned a response status of"));
539 }
540 }
541}