blob: 4a57d28b9ba506b95a34bc06d7b9dcbe4607b8ac [file] [log] [blame]
Hyunsun Moonb6febbe2016-02-12 15:59:53 -08001/*
Brian O'Connor8e57fd52016-04-09 01:19:45 -07002 * Copyright 2016-present Open Networking Laboratory
Hyunsun Moonb6febbe2016-02-12 15:59:53 -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 */
alshabibb4d31712016-06-01 18:51:03 -070016package org.opencord.cordvtn.rest;
Hyunsun Moonb6febbe2016-02-12 15:59:53 -080017
Hyunsun Mooneaf75e62016-09-27 16:40:23 -070018import com.fasterxml.jackson.databind.JsonNode;
19import com.fasterxml.jackson.databind.node.ArrayNode;
20import com.fasterxml.jackson.databind.node.ObjectNode;
21import org.onlab.osgi.DefaultServiceDirectory;
Hyunsun Moonb6febbe2016-02-12 15:59:53 -080022import org.onosproject.rest.AbstractWebResource;
Hyunsun Mooneaf75e62016-09-27 16:40:23 -070023import org.opencord.cordvtn.api.CordVtnAdminService;
24import org.opencord.cordvtn.api.SubnetId;
25import org.openstack4j.core.transport.ObjectMapperSingleton;
26import org.openstack4j.model.network.Subnet;
27import org.openstack4j.openstack.networking.domain.NeutronSubnet;
Hyunsun Moonb6febbe2016-02-12 15:59:53 -080028import org.slf4j.Logger;
29import org.slf4j.LoggerFactory;
30
31import javax.ws.rs.Consumes;
32import javax.ws.rs.DELETE;
Hyunsun Mooneaf75e62016-09-27 16:40:23 -070033import javax.ws.rs.GET;
Hyunsun Moonb6febbe2016-02-12 15:59:53 -080034import javax.ws.rs.POST;
35import javax.ws.rs.PUT;
36import javax.ws.rs.Path;
37import javax.ws.rs.PathParam;
38import javax.ws.rs.Produces;
Hyunsun Mooneaf75e62016-09-27 16:40:23 -070039import javax.ws.rs.core.Context;
Hyunsun Moonb6febbe2016-02-12 15:59:53 -080040import javax.ws.rs.core.MediaType;
41import javax.ws.rs.core.Response;
Hyunsun Mooneaf75e62016-09-27 16:40:23 -070042import javax.ws.rs.core.UriBuilder;
43import javax.ws.rs.core.UriInfo;
Hyunsun Moonb6febbe2016-02-12 15:59:53 -080044import java.io.InputStream;
Hyunsun Mooneaf75e62016-09-27 16:40:23 -070045import java.util.Set;
46
47import static com.fasterxml.jackson.databind.SerializationFeature.INDENT_OUTPUT;
48import static javax.ws.rs.core.Response.Status.NOT_FOUND;
49import static javax.ws.rs.core.Response.created;
50import static javax.ws.rs.core.Response.noContent;
51import static javax.ws.rs.core.Response.status;
Hyunsun Moonb6febbe2016-02-12 15:59:53 -080052
53/**
Hyunsun Mooneaf75e62016-09-27 16:40:23 -070054 * Neutron ML2 mechanism driver implementation for the subnet resource.
Hyunsun Moonb6febbe2016-02-12 15:59:53 -080055 */
56@Path("subnets")
57public class NeutronMl2SubnetsWebResource extends AbstractWebResource {
58 protected final Logger log = LoggerFactory.getLogger(getClass());
Hyunsun Moonb6febbe2016-02-12 15:59:53 -080059
Hyunsun Mooneaf75e62016-09-27 16:40:23 -070060 private static final String MESSAGE = "Received subnets %s request";
61 private static final String SUBNET = "subnet";
62 private static final String SUBNETS = "subnets";
63
64 private final CordVtnAdminService adminService =
65 DefaultServiceDirectory.getService(CordVtnAdminService.class);
66
67 @Context
68 private UriInfo uriInfo;
69
70 /**
71 * Creates a subnet from the JSON input stream.
72 *
73 * @param input subnet JSON input stream
74 * @return 201 CREATED if the JSON is correct, 400 BAD_REQUEST if the JSON
75 * is invalid or duplicated subnet already exists
76 */
Hyunsun Moonb6febbe2016-02-12 15:59:53 -080077 @POST
78 @Consumes(MediaType.APPLICATION_JSON)
79 @Produces(MediaType.APPLICATION_JSON)
80 public Response createSubnet(InputStream input) {
Hyunsun Mooneaf75e62016-09-27 16:40:23 -070081 log.trace(String.format(MESSAGE, "CREATE"));
82
83 final NeutronSubnet subnet = readSubnet(input);
84 adminService.createSubnet(subnet);
85 UriBuilder locationBuilder = uriInfo.getBaseUriBuilder()
86 .path(SUBNETS)
87 .path(subnet.getId());
88
89 // TODO fix networking-onos to send Network UPDATE when subnet created
90 return created(locationBuilder.build()).build();
Hyunsun Moonb6febbe2016-02-12 15:59:53 -080091 }
92
Hyunsun Mooneaf75e62016-09-27 16:40:23 -070093 /**
94 * Updates the subnet with the specified identifier.
95 *
96 * @param id subnet identifier
97 * @param input subnet JSON input stream
98 * @return 200 OK with the updated subnet, 400 BAD_REQUEST if the requested
99 * subnet does not exist
100 */
Hyunsun Moonb6febbe2016-02-12 15:59:53 -0800101 @PUT
102 @Path("{id}")
103 @Produces(MediaType.APPLICATION_JSON)
104 @Consumes(MediaType.APPLICATION_JSON)
105 public Response updateSubnet(@PathParam("id") String id, InputStream input) {
Hyunsun Mooneaf75e62016-09-27 16:40:23 -0700106 log.trace(String.format(MESSAGE, "UPDATE " + id));
Hyunsun Moonb6febbe2016-02-12 15:59:53 -0800107
Hyunsun Mooneaf75e62016-09-27 16:40:23 -0700108 final NeutronSubnet subnet = readSubnet(input);
109 adminService.updateSubnet(subnet);
110
111 ObjectNode result = this.mapper().createObjectNode();
112 return ok(result.set(SUBNET, writeSubnet(subnet))).build();
Hyunsun Moonb6febbe2016-02-12 15:59:53 -0800113 }
114
Hyunsun Mooneaf75e62016-09-27 16:40:23 -0700115 /**
116 * Returns all subnets.
117 *
118 * @return 200 OK with set of subnets
119 */
120 @GET
121 @Consumes(MediaType.APPLICATION_JSON)
122 @Produces(MediaType.APPLICATION_JSON)
123 public Response getSubnets() {
124 log.trace(String.format(MESSAGE, "GET"));
125
126 Set<Subnet> subnets = adminService.getSubnets();
127 ArrayNode arrayNodes = mapper().createArrayNode();
128 subnets.stream().forEach(subnet -> {
129 arrayNodes.add(writeSubnet(subnet));
130 });
131
132 ObjectNode result = this.mapper().createObjectNode();
133 return ok(result.set(SUBNETS, arrayNodes)).build();
134 }
135
136 /**
137 * Returns the subnet with the given subnet id.
138 *
139 * @param id subnet id
140 * @return 200 OK with the subnet, 404 NOT_FOUND if the subnet does not exist
141 */
142 @GET
143 @Path("{id}")
144 @Consumes(MediaType.APPLICATION_JSON)
145 @Produces(MediaType.APPLICATION_JSON)
146 public Response getSubnet(@PathParam("id") String id) {
147 log.trace(String.format(MESSAGE, "GET " + id));
148
149 Subnet subnet = adminService.getSubnet(SubnetId.of(id));
150 if (subnet == null) {
151 return status(NOT_FOUND).build();
152 }
153
154 ObjectNode result = this.mapper().createObjectNode();
155 return ok(result.set(SUBNET, writeSubnet(subnet))).build();
156 }
157
158 /**
159 * Removes the subnet.
160 *
161 * @param id subnet identifier
162 * @return 204 NO_CONTENT, 400 BAD_REQUEST if the subnet does not exist
163 */
Hyunsun Moonb6febbe2016-02-12 15:59:53 -0800164 @DELETE
165 @Path("{id}")
Hyunsun Mooneaf75e62016-09-27 16:40:23 -0700166 @Consumes(MediaType.APPLICATION_JSON)
167 @Produces(MediaType.APPLICATION_JSON)
Hyunsun Moonb6febbe2016-02-12 15:59:53 -0800168 public Response deleteSubnet(@PathParam("id") String id) {
Hyunsun Mooneaf75e62016-09-27 16:40:23 -0700169 log.trace(String.format(MESSAGE, "DELETE " + id));
170
171 adminService.removeSubnet(SubnetId.of(id));
172 return noContent().build();
173 }
174
175 private NeutronSubnet readSubnet(InputStream input) {
176 try {
177 JsonNode jsonTree = mapper().enable(INDENT_OUTPUT).readTree(input);
178 log.trace(mapper().writeValueAsString(jsonTree));
179 return ObjectMapperSingleton.getContext(NeutronSubnet.class)
180 .readerFor(NeutronSubnet.class)
181 .readValue(jsonTree);
182 } catch (Exception e) {
183 throw new IllegalArgumentException();
184 }
185 }
186
187 private ObjectNode writeSubnet(Subnet subnet) {
188 try {
189 String strSubnet = ObjectMapperSingleton.getContext(NeutronSubnet.class)
190 .writerFor(NeutronSubnet.class)
191 .writeValueAsString(subnet);
192 log.trace(strSubnet);
193 return (ObjectNode) mapper().readTree(strSubnet.getBytes());
194 } catch (Exception e) {
195 throw new IllegalStateException();
196 }
Hyunsun Moonb6febbe2016-02-12 15:59:53 -0800197 }
198}