blob: be484effe282f790c08e315193556f118fb30820 [file] [log] [blame]
Jian Li5c411232015-12-16 15:29:16 -08001/*
2 * Copyright 2014-2015 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.resources;
17
18import com.fasterxml.jackson.databind.JsonNode;
19import com.fasterxml.jackson.databind.node.ArrayNode;
20import com.fasterxml.jackson.databind.node.ObjectNode;
21import org.onosproject.net.DeviceId;
22import org.onosproject.net.meter.DefaultMeterRequest;
23import org.onosproject.net.meter.Meter;
24import org.onosproject.net.meter.MeterId;
25import org.onosproject.net.meter.MeterRequest;
26import org.onosproject.net.meter.MeterService;
27import org.onosproject.rest.AbstractWebResource;
28import org.slf4j.Logger;
29
30import javax.ws.rs.Consumes;
31import javax.ws.rs.DELETE;
32import javax.ws.rs.GET;
33import javax.ws.rs.POST;
34import javax.ws.rs.Path;
35import javax.ws.rs.PathParam;
36import javax.ws.rs.Produces;
37import javax.ws.rs.core.MediaType;
38import javax.ws.rs.core.Response;
39import java.io.IOException;
40import java.io.InputStream;
41import java.net.URI;
42import java.net.URISyntaxException;
43
44import static org.onlab.util.Tools.nullIsNotFound;
45import static org.slf4j.LoggerFactory.getLogger;
46
47/**
48 * Query and program meter rules.
49 */
50@Path("meters")
51public class MetersWebResource extends AbstractWebResource {
52 private final Logger log = getLogger(getClass());
53 public static final String DEVICE_INVALID = "Invalid deviceId in meter creation request";
54
55 final MeterService meterService = get(MeterService.class);
56 final ObjectNode root = mapper().createObjectNode();
57 final ArrayNode metersNode = root.putArray("meters");
58
59 /**
60 * Returns all meters of all devices.
61 *
62 * @return array of all the meters in the system
63 * @onos.rsModel Meters
64 */
65 @GET
66 @Produces(MediaType.APPLICATION_JSON)
67 public Response getMeters() {
68 final Iterable<Meter> meters = meterService.getAllMeters();
69 if (meters != null) {
70 meters.forEach(meter -> metersNode.add(codec(Meter.class).encode(meter, this)));
71 }
72 return ok(root).build();
73 }
74
75 /**
Jian Lia4faf4f2016-01-06 00:22:52 -080076 * Returns a collection of meters by the device id.
77 *
78 * @param deviceId device identifier
79 * @return array of meters which belongs to specified device
80 * @onos.rsModel Meters
81 */
82 @GET
83 @Produces(MediaType.APPLICATION_JSON)
84 @Path("{deviceId}")
85 public Response getMetersByDeviceId(@PathParam("deviceId") String deviceId) {
86 DeviceId did = DeviceId.deviceId(deviceId);
87 final Iterable<Meter> meters = meterService.getMeters(did);
88 if (meters != null) {
89 meters.forEach(meter -> metersNode.add(codec(Meter.class).encode(meter, this)));
90 }
91 return ok(root).build();
92 }
93
94 /**
Jian Li5c411232015-12-16 15:29:16 -080095 * Returns a meter by the meter id.
96 *
97 * @param deviceId device identifier
Jian Lia4faf4f2016-01-06 00:22:52 -080098 * @return a meter, return 404 if no entry has been found
Jian Li5c411232015-12-16 15:29:16 -080099 * @onos.rsModel Meter
100 */
101 @GET
102 @Produces(MediaType.APPLICATION_JSON)
103 @Path("{deviceId}/{meterId}")
104 public Response getMeterByDeviceIdAndMeterId(@PathParam("deviceId") String deviceId,
105 @PathParam("meterId") String meterId) {
106 DeviceId did = DeviceId.deviceId(deviceId);
107 MeterId mid = MeterId.meterId(Long.valueOf(meterId));
108
109 final Meter meter = nullIsNotFound(meterService.getMeter(did, mid),
110 "Meter is not found for " + mid.id());
111
112 metersNode.add(codec(Meter.class).encode(meter, this));
113 return ok(root).build();
114 }
115
116 /**
117 * Create new meter rule. Creates and installs a new meter rule for the
118 * specified device.
119 *
120 * @param deviceId device identifier
121 * @param stream meter rule JSON
122 * @return status of the request - CREATED if the JSON is correct,
123 * BAD_REQUEST if the JSON is invalid
124 * @onos.rsModel MeterPost
125 */
126 @POST
127 @Path("{deviceId}")
128 @Consumes(MediaType.APPLICATION_JSON)
129 @Produces(MediaType.APPLICATION_JSON)
130 public Response createMeter(@PathParam("deviceId") String deviceId,
131 InputStream stream) {
132 URI location;
133 try {
134 ObjectNode jsonTree = (ObjectNode) mapper().readTree(stream);
135 JsonNode specifiedDeviceId = jsonTree.get("deviceId");
136
137 if (specifiedDeviceId != null &&
138 !specifiedDeviceId.asText().equals(deviceId)) {
139 throw new IllegalArgumentException(DEVICE_INVALID);
140 }
141 jsonTree.put("deviceId", deviceId);
142 final Meter tmpMeter = codec(Meter.class).decode(jsonTree, this);
143 final MeterRequest meterRequest = meterToMeterRequest(tmpMeter, "ADD");
144 final Meter meter = meterService.submit(meterRequest);
145 location = new URI(Long.toString(meter.id().id()));
146 } catch (IOException | URISyntaxException ex) {
147 throw new IllegalArgumentException(ex);
148 }
149
150 return Response
151 .created(location)
152 .build();
153 }
154
155 /**
156 * Removes the specified meter.
157 *
158 * @param deviceId device identifier
159 * @param meterId meter identifier
160 */
161 @DELETE
162 @Produces(MediaType.APPLICATION_JSON)
163 @Path("{deviceId}/{meterId}")
164 public void deleteMeterByDeviceIdAndMeterId(@PathParam("deviceId") String deviceId,
165 @PathParam("meterId") String meterId) {
166 DeviceId did = DeviceId.deviceId(deviceId);
167 MeterId mid = MeterId.meterId(Long.valueOf(meterId));
168 final Meter tmpMeter = meterService.getMeter(did, mid);
169 if (tmpMeter != null) {
170 final MeterRequest meterRequest = meterToMeterRequest(tmpMeter, "REMOVE");
171 meterService.withdraw(meterRequest, tmpMeter.id());
172 }
173 }
174
175 /**
176 * Convert a meter instance to meterRequest instance with a certain operation.
177 *
178 * @param meter meter instance
179 * @param operation operation
180 * @return converted meterRequest instance
181 */
182 private MeterRequest meterToMeterRequest(Meter meter, String operation) {
183 MeterRequest.Builder builder;
184 MeterRequest meterRequest;
185
186 if (meter == null) {
187 return null;
188 }
189
190 if (meter.isBurst()) {
191 builder = DefaultMeterRequest.builder()
192 .fromApp(meter.appId())
193 .forDevice(meter.deviceId())
194 .withUnit(meter.unit())
195 .withBands(meter.bands())
196 .burst();
197 } else {
198 builder = DefaultMeterRequest.builder()
199 .fromApp(meter.appId())
200 .forDevice(meter.deviceId())
201 .withUnit(meter.unit())
202 .withBands(meter.bands());
203 }
204
205 switch (operation) {
206 case "ADD":
207 meterRequest = builder.add();
208 break;
209 case "REMOVE":
210 meterRequest = builder.remove();
211 break;
212 default:
213 log.warn("Invalid operation {}.", operation);
214 return null;
215 }
216
217 return meterRequest;
218 }
219}