REST API Support for DHCP Relay
Change-Id: Ib2ab8990a077f316c6292cc439452e671751b29a
diff --git a/apps/dhcprelay/web/BUCK b/apps/dhcprelay/web/BUCK
index d173841..42980c3 100644
--- a/apps/dhcprelay/web/BUCK
+++ b/apps/dhcprelay/web/BUCK
@@ -7,6 +7,7 @@
'//utils/rest:onlab-rest',
'//lib:javax.ws.rs-api',
'//lib:jersey-server',
+ '//lib:jackson-databind',
'//core/store/serializers:onos-core-serializers',
'//apps/route-service/api:onos-apps-route-service-api',
'//apps/dhcprelay/app:onos-apps-dhcprelay-app',
@@ -20,4 +21,4 @@
api_version = '1.0',
api_description = 'REST API for DHCPRelay',
api_package = 'org.onosproject.dhcprelay.rest',
-)
\ No newline at end of file
+)
diff --git a/apps/dhcprelay/web/src/main/java/org/onosproject/dhcprelay/rest/DhcpRelayWebResource.java b/apps/dhcprelay/web/src/main/java/org/onosproject/dhcprelay/rest/DhcpRelayWebResource.java
index 9338a4b..1a48ea5 100644
--- a/apps/dhcprelay/web/src/main/java/org/onosproject/dhcprelay/rest/DhcpRelayWebResource.java
+++ b/apps/dhcprelay/web/src/main/java/org/onosproject/dhcprelay/rest/DhcpRelayWebResource.java
@@ -19,7 +19,15 @@
import org.onlab.packet.IpAddress;
import org.onlab.packet.IpPrefix;
+import org.onlab.util.Tools;
+import com.fasterxml.jackson.databind.ObjectMapper;
+import com.fasterxml.jackson.databind.node.ArrayNode;
+import com.fasterxml.jackson.databind.node.ObjectNode;
import org.onosproject.dhcprelay.api.DhcpRelayService;
+import org.onosproject.dhcprelay.api.DhcpServerInfo;
+import org.onosproject.dhcprelay.cli.DhcpRelayCommand;
+import org.onosproject.dhcprelay.store.DhcpRecord;
+import org.onosproject.dhcprelay.store.DhcpRelayCounters;
import org.onosproject.rest.AbstractWebResource;
import org.onosproject.routeservice.Route;
import org.onosproject.routeservice.RouteStore;
@@ -27,11 +35,18 @@
import org.slf4j.Logger;
import javax.ws.rs.DELETE;
+import javax.ws.rs.GET;
+import javax.ws.rs.PUT;
import javax.ws.rs.Path;
import javax.ws.rs.PathParam;
+import javax.ws.rs.Produces;
+import javax.ws.rs.core.MediaType;
import javax.ws.rs.core.Response;
import java.io.IOException;
+import java.util.Collection;
+import java.util.Map;
+import java.util.List;
import java.util.Optional;
import static org.slf4j.LoggerFactory.getLogger;
@@ -42,6 +57,13 @@
@Path("fpm-delete")
public class DhcpRelayWebResource extends AbstractWebResource {
private static final Logger LOG = getLogger(DhcpRelayWebResource.class);
+ private final ObjectMapper mapper = new ObjectMapper();
+ private final DhcpRelayService dhcpDelayService = get(DhcpRelayService.class);
+ private static final String NA = "N/A";
+ List<DhcpServerInfo> defaultDhcpServerInfoList = dhcpDelayService.getDefaultDhcpServerInfoList();
+ List<DhcpServerInfo> indirectDhcpServerInfoList = dhcpDelayService.getIndirectDhcpServerInfoList();
+ Collection<DhcpRecord> records = dhcpDelayService.getDhcpRecords();
+
/**
* Deletes the fpm route from fpm record.
@@ -74,5 +96,170 @@
return Response.noContent().build();
}
+ /**
+ * Returns the response object with list of dhcp servers without counters.
+ *
+ * @return 200 OK with component properties of given component and variable
+ */
+ @GET
+ @Produces(MediaType.APPLICATION_JSON)
+ @Path("dhcp-relay")
+ public Response getDhcpServers() {
+ ObjectNode node = getdhcpRelayJsonOutput(null, null);
+ return Response.status(200).entity(node).build();
+ }
+
+ /**
+ * Returns dhcp servers details with counters.
+ *
+ * @param counter source ip identifier
+ * @return 200 OK with component properties of given component and variable
+ */
+ @GET
+ @Produces(MediaType.APPLICATION_JSON)
+ @Path("dhcp-relay/{counter}")
+ public Response getDhcpRelayCounter(@PathParam("counter") String counter) {
+ ObjectNode node = getdhcpRelayJsonOutput(counter, null);
+ return Response.status(200).entity(node).build();
+ }
+
+ /**
+ * To reset the dhcp relay counters.
+ *
+ * @param counter type String
+ * @param reset type String
+ * @return 200 OK with component properties of given component and variable
+ */
+ @PUT
+ @Produces(MediaType.APPLICATION_JSON)
+ @Path("dhcp-relay/{counter}/{reset}")
+ public Response resetDhcpRelayCounter(@PathParam("counter") String counter, @PathParam("reset") String reset) {
+ ObjectNode node = getdhcpRelayJsonOutput(counter, reset);
+ return Response.status(200).entity(node).build();
+ }
+
+
+ /**
+ * To create json output.
+ *
+ * @param counter type String
+ * @param reset type String
+ * @return node type ObjectNode.
+ */
+ private ObjectNode getdhcpRelayJsonOutput(String counter, String reset) {
+ ObjectNode node = mapper.createObjectNode();
+ ObjectNode dhcpRelayServerNode = mapper.createObjectNode();
+ if (!defaultDhcpServerInfoList.isEmpty()) {
+ ArrayNode defaultDhcpServers = listServers(defaultDhcpServerInfoList);
+ dhcpRelayServerNode.put("Default-DHCP-Server", defaultDhcpServers);
+ }
+ if (!indirectDhcpServerInfoList.isEmpty()) {
+ ArrayNode indirectDhcpServers = listServers(indirectDhcpServerInfoList);
+ dhcpRelayServerNode.put("Indirect-DHCP-Server", indirectDhcpServers);
+ }
+
+ ArrayNode dhcpRecords = dhcpRelayRecords(records);
+ dhcpRelayServerNode.put("DHCP-Relay-Records([D]:Directly-Connected)", dhcpRecords);
+ if (counter != null && !counter.equals("counter")) {
+ ArrayNode counterArray = dhcpRelayCounters(reset);
+ dhcpRelayServerNode.put("DHCP-Relay-Counter", counterArray);
+ }
+ node.put("Default-DHCP-Servers", dhcpRelayServerNode);
+
+ return node;
+
+ }
+ /**
+ * To get the liset of dhcp servers.
+ *
+ * @param dhcpServerInfoList type List
+ * @return servers type ArrayNode.
+ */
+ private ArrayNode listServers(List<DhcpServerInfo> dhcpServerInfoList) {
+ ArrayNode servers = mapper.createArrayNode();
+ dhcpServerInfoList.forEach(dhcpServerInfo -> {
+ ObjectNode serverNode = mapper.createObjectNode();
+ String connectPoint = dhcpServerInfo.getDhcpServerConnectPoint()
+ .map(cp -> cp.toString()).orElse(NA);
+ String serverMac = dhcpServerInfo.getDhcpConnectMac()
+ .map(mac -> mac.toString()).orElse(NA);
+ String gatewayAddress;
+ String serverIp;
+
+ switch (dhcpServerInfo.getVersion()) {
+ case DHCP_V4:
+ gatewayAddress = dhcpServerInfo.getDhcpGatewayIp4()
+ .map(gw -> gw.toString()).orElse(null);
+ serverIp = dhcpServerInfo.getDhcpServerIp4()
+ .map(ip -> ip.toString()).orElse(NA);
+ break;
+ case DHCP_V6:
+ gatewayAddress = dhcpServerInfo.getDhcpGatewayIp6()
+ .map(gw -> gw.toString()).orElse(null);
+ serverIp = dhcpServerInfo.getDhcpServerIp6()
+ .map(ip -> ip.toString()).orElse(NA);
+ break;
+ default:
+ return;
+ }
+
+ serverNode.put("connectPoint", connectPoint);
+ if (gatewayAddress != null) {
+ serverNode.put("server", serverIp.concat(" via ").concat(gatewayAddress));
+ } else {
+ serverNode.put("server", serverIp);
+ }
+ serverNode.put("mac", serverMac);
+ servers.add(serverNode);
+ });
+ return servers;
+}
+
+ /**
+ * To get the list of dhcp relay records.
+ *
+ * @param records type Collections
+ * @return dhcpRelayRecords type ArrayNode.
+ */
+ private ArrayNode dhcpRelayRecords(Collection<DhcpRecord> records) {
+ DhcpRelayCommand dhcpRelayCommand = new DhcpRelayCommand();
+ ArrayNode dhcpRelayRecords = mapper.createArrayNode();
+ records.forEach(record -> {
+ ObjectNode dhcpRecord = mapper.createObjectNode();
+ dhcpRecord.put("id", record.macAddress() + "/" + record.vlanId());
+ dhcpRecord.put("locations", record.locations().toString());
+ dhcpRecord.put("last-seen", Tools.timeAgo(record.lastSeen()));
+ dhcpRecord.put("IPv4", dhcpRelayCommand.ip4State(record));
+ dhcpRecord.put("IPv6", dhcpRelayCommand.ip6State(record));
+ dhcpRelayRecords.add(dhcpRecord);
+ });
+ return dhcpRelayRecords;
+ }
+
+ /**
+ * To get the details of dhcp relay counters.
+ *
+ * @param reset type String
+ * @return counterArray type ArrayNode.
+ */
+ private ArrayNode dhcpRelayCounters(String reset) {
+ ObjectNode counters = mapper.createObjectNode();
+ ObjectNode counterPackets = mapper.createObjectNode();
+ ArrayNode counterArray = mapper.createArrayNode();
+ records.forEach(record -> {
+ DhcpRelayCounters v6Counters = record.getV6Counters();
+ if (reset != null && reset.equals("reset")) {
+ v6Counters.resetCounters();
+ }
+ Map<String, Integer> countersMap = v6Counters.getCounters();
+ countersMap.forEach((name, value) -> {
+ counterPackets.put(name, value);
+
+ });
+ counters.put(record.locations().toString(), counterPackets);
+ counterArray.add(counters);
+ });
+ return counterArray;
+ }
}