blob: f70382ddf97acd846dc52407d48e2545e23bb680 [file] [log] [blame]
Ray Milkey4f5de002014-12-17 19:26:11 -08001/*
Brian O'Connora09fe5b2017-08-03 21:12:30 -07002 * Copyright 2015-present Open Networking Foundation
Ray Milkey4f5de002014-12-17 19:26:11 -08003 *
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 */
Jonathan Hart9bb32ab2015-05-05 18:17:31 -070016package org.onosproject.rest.resources;
Ray Milkey4f5de002014-12-17 19:26:11 -080017
Jian Li9d616492016-03-09 10:52:49 -080018import com.fasterxml.jackson.databind.JsonNode;
19import com.fasterxml.jackson.databind.node.ArrayNode;
20import com.fasterxml.jackson.databind.node.ObjectNode;
Ray Milkey460f9b02016-03-29 11:56:19 -070021import com.google.common.collect.ArrayListMultimap;
22import com.google.common.collect.ListMultimap;
Ray Milkeyd43fe452015-05-29 09:35:12 -070023import org.onlab.util.ItemNotFoundException;
Jian Li2907ad22016-05-12 23:08:54 -070024import org.onosproject.app.ApplicationService;
25import org.onosproject.core.ApplicationId;
Ray Milkeyd43fe452015-05-29 09:35:12 -070026import org.onosproject.net.Device;
27import org.onosproject.net.DeviceId;
28import org.onosproject.net.device.DeviceService;
29import org.onosproject.net.flow.FlowEntry;
30import org.onosproject.net.flow.FlowRule;
31import org.onosproject.net.flow.FlowRuleService;
32import org.onosproject.rest.AbstractWebResource;
33
Jian Li9d616492016-03-09 10:52:49 -080034import javax.ws.rs.Consumes;
35import javax.ws.rs.DELETE;
36import javax.ws.rs.GET;
37import javax.ws.rs.POST;
38import javax.ws.rs.Path;
39import javax.ws.rs.PathParam;
40import javax.ws.rs.Produces;
Jian Li2907ad22016-05-12 23:08:54 -070041import javax.ws.rs.QueryParam;
Jian Li9d616492016-03-09 10:52:49 -080042import javax.ws.rs.core.Context;
43import javax.ws.rs.core.MediaType;
44import javax.ws.rs.core.Response;
45import javax.ws.rs.core.UriBuilder;
46import javax.ws.rs.core.UriInfo;
47import java.io.IOException;
48import java.io.InputStream;
Ray Milkey460f9b02016-03-29 11:56:19 -070049import java.util.ArrayList;
Thomas Vachuskaa1ae5e12016-03-28 10:36:30 -070050import java.util.List;
Jian Li9d616492016-03-09 10:52:49 -080051import java.util.stream.StreamSupport;
Ray Milkeyd43fe452015-05-29 09:35:12 -070052
Ray Milkey131c0682016-10-18 11:12:50 -070053import static org.onlab.util.Tools.nullIsIllegal;
Ray Milkey460f9b02016-03-29 11:56:19 -070054import static org.onlab.util.Tools.nullIsNotFound;
Ray Milkey86ee5e82018-04-02 15:33:07 -070055import static org.onlab.util.Tools.readTreeFromStream;
Ray Milkey460f9b02016-03-29 11:56:19 -070056
Ray Milkey4f5de002014-12-17 19:26:11 -080057/**
Thomas Vachuska0fa2aa12015-08-18 12:53:04 -070058 * Query and program flow rules.
Ray Milkey4f5de002014-12-17 19:26:11 -080059 */
60
61@Path("flows")
62public class FlowsWebResource extends AbstractWebResource {
Jian Li9d616492016-03-09 10:52:49 -080063
64 @Context
Jian Licc730a62016-05-10 16:36:16 -070065 private UriInfo uriInfo;
Jian Li9d616492016-03-09 10:52:49 -080066
Jian Licc730a62016-05-10 16:36:16 -070067 private static final String DEVICE_NOT_FOUND = "Device is not found";
68 private static final String FLOW_NOT_FOUND = "Flow is not found";
Jian Li2907ad22016-05-12 23:08:54 -070069 private static final String APP_ID_NOT_FOUND = "Application Id is not found";
Ray Milkey131c0682016-10-18 11:12:50 -070070 private static final String FLOW_ARRAY_REQUIRED = "Flows array was not specified";
Jian Licc730a62016-05-10 16:36:16 -070071 private static final String FLOWS = "flows";
72 private static final String DEVICE_ID = "deviceId";
73 private static final String FLOW_ID = "flowId";
Ray Milkey4f5de002014-12-17 19:26:11 -080074
Jian Licc730a62016-05-10 16:36:16 -070075 private final FlowRuleService service = get(FlowRuleService.class);
76 private final ObjectNode root = mapper().createObjectNode();
77 private final ArrayNode flowsNode = root.putArray(FLOWS);
Ray Milkey4f5de002014-12-17 19:26:11 -080078
79 /**
Jian Licc730a62016-05-10 16:36:16 -070080 * Gets all flow entries. Returns array of all flow rules in the system.
81 *
82 * @return 200 OK with a collection of flows
Jian Li2907ad22016-05-12 23:08:54 -070083 * @onos.rsModel FlowEntries
Ray Milkey4f5de002014-12-17 19:26:11 -080084 */
85 @GET
86 @Produces(MediaType.APPLICATION_JSON)
87 public Response getFlows() {
Ray Milkey4f5de002014-12-17 19:26:11 -080088 final Iterable<Device> devices = get(DeviceService.class).getDevices();
89 for (final Device device : devices) {
Phaneendra Mandaaec654c2015-09-23 19:02:23 +053090 final Iterable<FlowEntry> flowEntries = service.getFlowEntries(device.id());
91 if (flowEntries != null) {
92 for (final FlowEntry entry : flowEntries) {
Thomas Vachuska8683e012015-03-18 18:03:33 -070093 flowsNode.add(codec(FlowEntry.class).encode(entry, this));
Ray Milkey4f5de002014-12-17 19:26:11 -080094 }
95 }
96 }
97
Ray Milkey3f025692015-01-26 11:15:41 -080098 return ok(root).build();
Ray Milkey4f5de002014-12-17 19:26:11 -080099 }
100
101 /**
Jian Licc730a62016-05-10 16:36:16 -0700102 * Creates new flow rules. Creates and installs a new flow rules.<br>
Jonathan Hartf09a13a2017-01-26 09:52:14 -0800103 * Flow rule criteria and instruction description:
104 * https://wiki.onosproject.org/display/ONOS/Flow+Rules
Thomas Vachuskaa1ae5e12016-03-28 10:36:30 -0700105 *
Ray Milkeybb23e0b2016-08-02 17:00:21 -0700106 * @param appId application id
Jian Licc730a62016-05-10 16:36:16 -0700107 * @param stream flow rules JSON
Thomas Vachuskaa1ae5e12016-03-28 10:36:30 -0700108 * @return status of the request - CREATED if the JSON is correct,
109 * BAD_REQUEST if the JSON is invalid
Jian Licc730a62016-05-10 16:36:16 -0700110 * @onos.rsModel FlowsBatchPost
Thomas Vachuskaa1ae5e12016-03-28 10:36:30 -0700111 */
112 @POST
113 @Consumes(MediaType.APPLICATION_JSON)
114 @Produces(MediaType.APPLICATION_JSON)
Jian Li2907ad22016-05-12 23:08:54 -0700115 public Response createFlows(@QueryParam("appId") String appId, InputStream stream) {
Thomas Vachuskaa1ae5e12016-03-28 10:36:30 -0700116 try {
Ray Milkey86ee5e82018-04-02 15:33:07 -0700117 ObjectNode jsonTree = readTreeFromStream(mapper(), stream);
Ray Milkey131c0682016-10-18 11:12:50 -0700118 ArrayNode flowsArray = nullIsIllegal((ArrayNode) jsonTree.get(FLOWS),
119 FLOW_ARRAY_REQUIRED);
Jian Li2907ad22016-05-12 23:08:54 -0700120
Jonathan Hart9b80da82016-11-10 21:46:15 +0000121 if (appId != null) {
122 flowsArray.forEach(flowJson -> ((ObjectNode) flowJson).put("appId", appId));
123 }
Jian Li2907ad22016-05-12 23:08:54 -0700124
Thomas Vachuskaa1ae5e12016-03-28 10:36:30 -0700125 List<FlowRule> rules = codec(FlowRule.class).decode(flowsArray, this);
Ray Milkey131c0682016-10-18 11:12:50 -0700126
Thomas Vachuskaa1ae5e12016-03-28 10:36:30 -0700127 service.applyFlowRules(rules.toArray(new FlowRule[rules.size()]));
Ray Milkey460f9b02016-03-29 11:56:19 -0700128 rules.forEach(flowRule -> {
129 ObjectNode flowNode = mapper().createObjectNode();
130 flowNode.put(DEVICE_ID, flowRule.deviceId().toString())
Jayasree Ghosh28e1b3e2016-06-16 05:38:50 +0530131 .put(FLOW_ID, Long.toString(flowRule.id().value()));
Ray Milkey460f9b02016-03-29 11:56:19 -0700132 flowsNode.add(flowNode);
133 });
Thomas Vachuskaa1ae5e12016-03-28 10:36:30 -0700134 } catch (IOException ex) {
135 throw new IllegalArgumentException(ex);
136 }
Ray Milkey460f9b02016-03-29 11:56:19 -0700137 return Response.ok(root).build();
Thomas Vachuskaa1ae5e12016-03-28 10:36:30 -0700138 }
139
140 /**
Jian Licc730a62016-05-10 16:36:16 -0700141 * Gets flow entries of a device. Returns array of all flow rules for the
Thomas Vachuska0fa2aa12015-08-18 12:53:04 -0700142 * specified device.
Jian Licc730a62016-05-10 16:36:16 -0700143 *
Thomas Vachuska0fa2aa12015-08-18 12:53:04 -0700144 * @param deviceId device identifier
Jian Licc730a62016-05-10 16:36:16 -0700145 * @return 200 OK with a collection of flows of given device
Jian Li2907ad22016-05-12 23:08:54 -0700146 * @onos.rsModel FlowEntries
Ray Milkey4f5de002014-12-17 19:26:11 -0800147 */
148 @GET
149 @Produces(MediaType.APPLICATION_JSON)
Jian Li2907ad22016-05-12 23:08:54 -0700150 // TODO: we need to add "/device" suffix to the path to differentiate with appId
Ray Milkey4f5de002014-12-17 19:26:11 -0800151 @Path("{deviceId}")
152 public Response getFlowByDeviceId(@PathParam("deviceId") String deviceId) {
Phaneendra Mandaaec654c2015-09-23 19:02:23 +0530153 final Iterable<FlowEntry> flowEntries =
Ray Milkey4f5de002014-12-17 19:26:11 -0800154 service.getFlowEntries(DeviceId.deviceId(deviceId));
155
Jian Li9d616492016-03-09 10:52:49 -0800156 if (flowEntries == null || !flowEntries.iterator().hasNext()) {
Ray Milkey4f5de002014-12-17 19:26:11 -0800157 throw new ItemNotFoundException(DEVICE_NOT_FOUND);
158 }
Phaneendra Mandaaec654c2015-09-23 19:02:23 +0530159 for (final FlowEntry entry : flowEntries) {
Thomas Vachuska8683e012015-03-18 18:03:33 -0700160 flowsNode.add(codec(FlowEntry.class).encode(entry, this));
Ray Milkey4f5de002014-12-17 19:26:11 -0800161 }
Ray Milkey3f025692015-01-26 11:15:41 -0800162 return ok(root).build();
Ray Milkey4f5de002014-12-17 19:26:11 -0800163 }
164
165 /**
Jian Li2907ad22016-05-12 23:08:54 -0700166 * Gets flow rules. Returns the flow entry specified by the device id and
Thomas Vachuska0fa2aa12015-08-18 12:53:04 -0700167 * flow rule id.
Jian Licc730a62016-05-10 16:36:16 -0700168 *
Thomas Vachuska0fa2aa12015-08-18 12:53:04 -0700169 * @param deviceId device identifier
170 * @param flowId flow rule identifier
Jian Li2907ad22016-05-12 23:08:54 -0700171 * @return 200 OK with a collection of flows of given device and flow
172 * @onos.rsModel FlowEntries
Ray Milkey4f5de002014-12-17 19:26:11 -0800173 */
174 @GET
175 @Produces(MediaType.APPLICATION_JSON)
176 @Path("{deviceId}/{flowId}")
177 public Response getFlowByDeviceIdAndFlowId(@PathParam("deviceId") String deviceId,
178 @PathParam("flowId") long flowId) {
Phaneendra Mandaaec654c2015-09-23 19:02:23 +0530179 final Iterable<FlowEntry> flowEntries =
Ray Milkey4f5de002014-12-17 19:26:11 -0800180 service.getFlowEntries(DeviceId.deviceId(deviceId));
181
Jian Li9d616492016-03-09 10:52:49 -0800182 if (flowEntries == null || !flowEntries.iterator().hasNext()) {
Ray Milkey4f5de002014-12-17 19:26:11 -0800183 throw new ItemNotFoundException(DEVICE_NOT_FOUND);
184 }
Phaneendra Mandaaec654c2015-09-23 19:02:23 +0530185 for (final FlowEntry entry : flowEntries) {
Ray Milkey4f5de002014-12-17 19:26:11 -0800186 if (entry.id().value() == flowId) {
Thomas Vachuska8683e012015-03-18 18:03:33 -0700187 flowsNode.add(codec(FlowEntry.class).encode(entry, this));
Ray Milkey4f5de002014-12-17 19:26:11 -0800188 }
189 }
Ray Milkey3f025692015-01-26 11:15:41 -0800190 return ok(root).build();
Ray Milkey4f5de002014-12-17 19:26:11 -0800191 }
Ray Milkeyd43fe452015-05-29 09:35:12 -0700192
193 /**
Jian Li2907ad22016-05-12 23:08:54 -0700194 * Gets flow rules generated by an application.
195 * Returns the flow rule specified by the application id.
196 *
197 * @param appId application identifier
198 * @return 200 OK with a collection of flows of given application id
199 * @onos.rsModel FlowRules
200 */
201 @GET
202 @Produces(MediaType.APPLICATION_JSON)
203 @Path("application/{appId}")
204 public Response getFlowByAppId(@PathParam("appId") String appId) {
205 final ApplicationService appService = get(ApplicationService.class);
206 final ApplicationId idInstant = nullIsNotFound(appService.getId(appId), APP_ID_NOT_FOUND);
Bharath Thiruveedula99849dc2016-11-17 22:04:38 +0530207 final Iterable<FlowEntry> flowEntries = service.getFlowEntriesById(idInstant);
Jian Li2907ad22016-05-12 23:08:54 -0700208
Bharath Thiruveedula99849dc2016-11-17 22:04:38 +0530209 flowEntries.forEach(flow -> flowsNode.add(codec(FlowEntry.class).encode(flow, this)));
Jian Li2907ad22016-05-12 23:08:54 -0700210 return ok(root).build();
211 }
212
Bharath Thiruveedula99849dc2016-11-17 22:04:38 +0530213
Jian Li2907ad22016-05-12 23:08:54 -0700214 /**
215 * Removes flow rules by application ID.
216 * Removes a collection of flow rules generated by the given application.
217 *
218 * @param appId application identifier
219 * @return 204 NO CONTENT
220 */
221 @DELETE
222 @Produces(MediaType.APPLICATION_JSON)
223 @Path("application/{appId}")
224 public Response removeFlowByAppId(@PathParam("appId") String appId) {
225 final ApplicationService appService = get(ApplicationService.class);
226 final ApplicationId idInstant = nullIsNotFound(appService.getId(appId), APP_ID_NOT_FOUND);
227 service.removeFlowRulesById(idInstant);
228 return Response.noContent().build();
229 }
230
231 /**
Jian Licc730a62016-05-10 16:36:16 -0700232 * Creates new flow rule. Creates and installs a new flow rule for the
Andrea Campanella881f29f2016-03-03 19:18:42 -0800233 * specified device. <br>
Jonathan Hartf09a13a2017-01-26 09:52:14 -0800234 * Flow rule criteria and instruction description:
235 * https://wiki.onosproject.org/display/ONOS/Flow+Rules
Andrea Campanella881f29f2016-03-03 19:18:42 -0800236 *
Madan Jampani0dbac7a2015-06-25 10:37:45 -0700237 * @param deviceId device identifier
Jian Li2907ad22016-05-12 23:08:54 -0700238 * @param appId application identifier
Thomas Vachuska0fa2aa12015-08-18 12:53:04 -0700239 * @param stream flow rule JSON
Ray Milkeyeb5c7172015-06-23 14:59:27 -0700240 * @return status of the request - CREATED if the JSON is correct,
Ray Milkeyd43fe452015-05-29 09:35:12 -0700241 * BAD_REQUEST if the JSON is invalid
Jian Licc730a62016-05-10 16:36:16 -0700242 * @onos.rsModel FlowsPost
Ray Milkeyd43fe452015-05-29 09:35:12 -0700243 */
244 @POST
Ray Milkeyeb5c7172015-06-23 14:59:27 -0700245 @Path("{deviceId}")
Ray Milkeyd43fe452015-05-29 09:35:12 -0700246 @Consumes(MediaType.APPLICATION_JSON)
247 @Produces(MediaType.APPLICATION_JSON)
Ray Milkeyeb5c7172015-06-23 14:59:27 -0700248 public Response createFlow(@PathParam("deviceId") String deviceId,
Jian Li2907ad22016-05-12 23:08:54 -0700249 @QueryParam("appId") String appId,
Ray Milkeyeb5c7172015-06-23 14:59:27 -0700250 InputStream stream) {
Ray Milkeyd43fe452015-05-29 09:35:12 -0700251 try {
Ray Milkey86ee5e82018-04-02 15:33:07 -0700252 ObjectNode jsonTree = readTreeFromStream(mapper(), stream);
Ray Milkey5d915f42015-08-13 10:27:53 -0700253 JsonNode specifiedDeviceId = jsonTree.get("deviceId");
Ray Milkeyeb5c7172015-06-23 14:59:27 -0700254 if (specifiedDeviceId != null &&
255 !specifiedDeviceId.asText().equals(deviceId)) {
256 throw new IllegalArgumentException(
257 "Invalid deviceId in flow creation request");
258 }
Ray Milkey5d915f42015-08-13 10:27:53 -0700259 jsonTree.put("deviceId", deviceId);
Jian Li2907ad22016-05-12 23:08:54 -0700260
Jonathan Hart9b80da82016-11-10 21:46:15 +0000261 if (appId != null) {
262 jsonTree.put("appId", appId);
263 }
Jian Li2907ad22016-05-12 23:08:54 -0700264
Ray Milkey5d915f42015-08-13 10:27:53 -0700265 FlowRule rule = codec(FlowRule.class).decode(jsonTree, this);
Ray Milkeyd43fe452015-05-29 09:35:12 -0700266 service.applyFlowRules(rule);
Jian Li9d616492016-03-09 10:52:49 -0800267 UriBuilder locationBuilder = uriInfo.getBaseUriBuilder()
268 .path("flows")
269 .path(deviceId)
Ray Milkey90289b02016-03-31 15:23:28 -0700270 .path(Long.toString(rule.id().value()));
Jian Li9d616492016-03-09 10:52:49 -0800271
272 return Response
273 .created(locationBuilder.build())
274 .build();
275 } catch (IOException ex) {
Ray Milkey5d915f42015-08-13 10:27:53 -0700276 throw new IllegalArgumentException(ex);
Ray Milkeyd43fe452015-05-29 09:35:12 -0700277 }
Ray Milkeyeb5c7172015-06-23 14:59:27 -0700278 }
279
280 /**
Jian Li2907ad22016-05-12 23:08:54 -0700281 * Removes flow rule. Removes the specified flow rule.
Ray Milkeyeb5c7172015-06-23 14:59:27 -0700282 *
Thomas Vachuska0fa2aa12015-08-18 12:53:04 -0700283 * @param deviceId device identifier
284 * @param flowId flow rule identifier
Jian Lic2a542b2016-05-10 11:48:19 -0700285 * @return 204 NO CONTENT
Ray Milkeyeb5c7172015-06-23 14:59:27 -0700286 */
287 @DELETE
Ray Milkeyeb5c7172015-06-23 14:59:27 -0700288 @Path("{deviceId}/{flowId}")
Jian Lic2a542b2016-05-10 11:48:19 -0700289 public Response deleteFlowByDeviceIdAndFlowId(@PathParam("deviceId") String deviceId,
Jian Licc730a62016-05-10 16:36:16 -0700290 @PathParam("flowId") long flowId) {
Phaneendra Mandaaec654c2015-09-23 19:02:23 +0530291 final Iterable<FlowEntry> flowEntries =
Ray Milkeyeb5c7172015-06-23 14:59:27 -0700292 service.getFlowEntries(DeviceId.deviceId(deviceId));
293
Phaneendra Mandaaec654c2015-09-23 19:02:23 +0530294 if (!flowEntries.iterator().hasNext()) {
Ray Milkeyeb5c7172015-06-23 14:59:27 -0700295 throw new ItemNotFoundException(DEVICE_NOT_FOUND);
296 }
297
Phaneendra Mandaaec654c2015-09-23 19:02:23 +0530298 StreamSupport.stream(flowEntries.spliterator(), false)
Ray Milkeyeb5c7172015-06-23 14:59:27 -0700299 .filter(entry -> entry.id().value() == flowId)
300 .forEach(service::removeFlowRules);
Jian Lic2a542b2016-05-10 11:48:19 -0700301 return Response.noContent().build();
Ray Milkeyd43fe452015-05-29 09:35:12 -0700302 }
303
Ray Milkey460f9b02016-03-29 11:56:19 -0700304 /**
305 * Removes a batch of flow rules.
Ray Milkeybee35092016-04-12 10:01:26 -0700306 *
307 * @param stream stream for posted JSON
Jian Licc730a62016-05-10 16:36:16 -0700308 * @return 204 NO CONTENT
Ray Milkey460f9b02016-03-29 11:56:19 -0700309 */
310 @DELETE
Jian Lic2a542b2016-05-10 11:48:19 -0700311 public Response deleteFlows(InputStream stream) {
Ray Milkey460f9b02016-03-29 11:56:19 -0700312 ListMultimap<DeviceId, Long> deviceMap = ArrayListMultimap.create();
313 List<FlowEntry> rulesToRemove = new ArrayList<>();
314
315 try {
Ray Milkey86ee5e82018-04-02 15:33:07 -0700316 ObjectNode jsonTree = readTreeFromStream(mapper(), stream);
Ray Milkey460f9b02016-03-29 11:56:19 -0700317
318 JsonNode jsonFlows = jsonTree.get("flows");
319
320 jsonFlows.forEach(node -> {
321 DeviceId deviceId =
322 DeviceId.deviceId(
323 nullIsNotFound(node.get(DEVICE_ID),
Jian Licc730a62016-05-10 16:36:16 -0700324 DEVICE_NOT_FOUND).asText());
Ray Milkey460f9b02016-03-29 11:56:19 -0700325 long flowId = nullIsNotFound(node.get(FLOW_ID),
Jian Licc730a62016-05-10 16:36:16 -0700326 FLOW_NOT_FOUND).asLong();
Ray Milkey460f9b02016-03-29 11:56:19 -0700327 deviceMap.put(deviceId, flowId);
328
329 });
330 } catch (IOException ex) {
331 throw new IllegalArgumentException(ex);
332 }
333
334 deviceMap.keySet().forEach(deviceId -> {
335 List<Long> flowIds = deviceMap.get(deviceId);
336 Iterable<FlowEntry> entries = service.getFlowEntries(deviceId);
337 flowIds.forEach(flowId -> {
338 StreamSupport.stream(entries.spliterator(), false)
339 .filter(entry -> flowId == entry.id().value())
340 .forEach(rulesToRemove::add);
341 });
342 });
343
344 service.removeFlowRules(rulesToRemove.toArray(new FlowEntry[0]));
Jian Lic2a542b2016-05-10 11:48:19 -0700345 return Response.noContent().build();
Ray Milkey460f9b02016-03-29 11:56:19 -0700346 }
Ray Milkey4f5de002014-12-17 19:26:11 -0800347}