REST API support added for t3-troubleshoot-simple

Change-Id: I1f31b48fb866e0a61e66b61ad2e6a30b5eddb04c
diff --git a/apps/t3/web/src/main/java/org/onosproject/t3/rest/T3WebApplication.java b/apps/t3/web/src/main/java/org/onosproject/t3/rest/T3WebApplication.java
new file mode 100644
index 0000000..5f6a548
--- /dev/null
+++ b/apps/t3/web/src/main/java/org/onosproject/t3/rest/T3WebApplication.java
@@ -0,0 +1,30 @@
+/*
+ * Copyright 2018-present Open Networking Foundation
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ *     http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.onosproject.t3.rest;
+
+
+import org.onlab.rest.AbstractWebApplication;
+
+import java.util.Set;
+
+public class T3WebApplication extends AbstractWebApplication {
+    @Override
+    public Set<Class<?>> getClasses() {
+        return getClasses(T3WebResource.class);
+    }
+
+}
diff --git a/apps/t3/web/src/main/java/org/onosproject/t3/rest/T3WebResource.java b/apps/t3/web/src/main/java/org/onosproject/t3/rest/T3WebResource.java
new file mode 100644
index 0000000..0fe7674
--- /dev/null
+++ b/apps/t3/web/src/main/java/org/onosproject/t3/rest/T3WebResource.java
@@ -0,0 +1,271 @@
+/*
+ * Copyright 2018-present Open Networking Foundation
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ *     http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.onosproject.t3.rest;
+
+import com.fasterxml.jackson.databind.ObjectMapper;
+import com.fasterxml.jackson.databind.node.ArrayNode;
+import com.fasterxml.jackson.databind.node.ObjectNode;
+
+import org.onlab.packet.EthType;
+import org.onosproject.net.ConnectPoint;
+import org.onosproject.net.HostId;
+import org.onosproject.net.flow.FlowEntry;
+import org.onosproject.net.flow.criteria.Criterion;
+import org.onosproject.net.group.Group;
+import org.onosproject.rest.AbstractWebResource;
+import org.onosproject.t3.api.GroupsInDevice;
+import org.onosproject.t3.api.StaticPacketTrace;
+import org.onosproject.t3.api.TroubleshootService;
+import org.slf4j.Logger;
+
+import javax.ws.rs.GET;
+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.util.List;
+import java.util.Set;
+
+import static org.slf4j.LoggerFactory.getLogger;
+
+/**
+ * Trellis Troubleshooting Tool REST API.
+ */
+@Path("t3")
+public class T3WebResource extends AbstractWebResource {
+    private final ObjectMapper mapper = new ObjectMapper();
+    private static final Logger LOG = getLogger(T3WebResource.class);
+
+    /**
+     * Returns the trace non verbose result for the given source and destination ips.
+     *
+     * @param srcHost source ip identifier
+     * @param dstHost destination ip identifier
+     * @param ethType ethernet type identifier
+     * @return 200 OK with component properties of given component and variable
+     */
+    @GET
+    @Produces(MediaType.APPLICATION_JSON)
+    @Path("simple/{srcHost}/{dstHost}/{ethType}")
+    public Response getT3Simple(@PathParam("srcHost") String srcHost, @PathParam("dstHost") String dstHost,
+                                @PathParam("ethType") String ethType) {
+
+        ObjectNode node;
+        try {
+            node = getT3JsonOutput(srcHost, dstHost, ethType, false);
+        } catch (IllegalArgumentException e) {
+            throw new IllegalArgumentException(e);
+        }
+        return Response.status(200).entity(node).build();
+
+    }
+
+    /**
+     * Returns the trace verbose result for the given source and destination ips.
+     *
+     * @param srcHost source ip identifier
+     * @param dstHost destination ip identifier
+     * @param ethType ethernet type identifier
+     * @return 200 OK with component properties of given component and variable
+     */
+    @GET
+    @Produces(MediaType.APPLICATION_JSON)
+    @Path("simple/{srcHost}/{dstHost}/{ethType}/verbose")
+    public Response getT3SimpleVerbose(@PathParam("srcHost") String srcHost, @PathParam("dstHost") String dstHost,
+                                       @PathParam("ethType") String ethType) {
+
+        ObjectNode node;
+        try {
+            node = getT3JsonOutput(srcHost, dstHost, ethType, true);
+        } catch (IllegalArgumentException e) {
+            throw new IllegalArgumentException(e);
+        }
+        return Response.status(200).entity(node).build();
+    }
+
+    /**
+     * Returns trace verbose or non verbose json output for the given trace.
+     *
+     * @param srcHost source ip identifier
+     * @param dstHost destination ip identifier
+     * @param ethType the ethernet Type
+     * @param verbose based on verbosity level
+     * @return a json representing the trace.
+     */
+    private ObjectNode getT3JsonOutput(String srcHost, String dstHost, String ethType, Boolean verbose) {
+        TroubleshootService troubleshootService = get(TroubleshootService.class);
+        final ObjectNode nodeOutput = mapper.createObjectNode();
+        //Tracing between host ips
+        ArrayNode ipList = mapper.createArrayNode();
+        ipList.add(srcHost);
+        ipList.add(dstHost);
+        nodeOutput.set("hostIps ", ipList);
+        EthType.EtherType type = EthType.EtherType.valueOf(ethType.toUpperCase());
+
+        //Build the traces
+        Set<StaticPacketTrace> traces = troubleshootService.trace(HostId.hostId(srcHost), HostId.hostId(dstHost), type);
+        traces.forEach(trace -> {
+
+            if (trace.getInitialPacket() != null) {
+                ArrayNode node = mapper.createArrayNode();
+                for (Criterion packet : trace.getInitialPacket().criteria()) {
+                    node.add(packet.toString());
+                }
+                nodeOutput.set("input packet", node);
+                nodeOutput.set("trace", getTraceJson(trace, verbose));
+            } else {
+                LOG.debug("cannot obtain trace between: ", srcHost, dstHost);
+                nodeOutput.set("failed trace", ipList);
+                nodeOutput.put("reason", trace.resultMessage());
+            }
+        });
+        return nodeOutput;
+    }
+
+    /**
+     * Returns verbose or non verbose json output for the given trace.      *
+     *
+     * @param trace     the trace
+     * @param verbosity based on verbosity level
+     * @return a json representing the trace.
+     */
+    public ObjectNode getTraceJson(StaticPacketTrace trace, boolean verbosity) {
+        ObjectNode nodeOutput = mapper.createObjectNode();
+        if (verbosity) {
+            nodeOutput.set("trace", getTrace(trace, verbosity));
+        } else {
+            ArrayNode nodePath = mapper.createArrayNode();
+            for (List<ConnectPoint> listPaths : trace.getCompletePaths()) {
+                ArrayNode node = mapper.createArrayNode();
+                for (ConnectPoint path : listPaths) {
+                    node.add(path.toString());
+                }
+                nodePath.add(node);
+            }
+            nodeOutput.set("paths", nodePath);
+        }
+
+        nodeOutput.put("result", trace.resultMessage());
+        return nodeOutput;
+    }
+
+    /**
+     * Returns verbose json output for the given trace.      *
+     *
+     * @param trace the trace
+     * @return a json representing the trace.
+     */
+    private ObjectNode getTrace(StaticPacketTrace trace, boolean verbose) {
+
+        ObjectNode nodeOutput = mapper.createObjectNode();
+
+        List<List<ConnectPoint>> paths = trace.getCompletePaths();
+        ArrayNode nodePath = mapper.createArrayNode();
+        for (List<ConnectPoint> path : paths) {
+            ArrayNode pathNode = mapper.createArrayNode();
+            for (ConnectPoint pathItr : path) {
+                pathNode.add(pathItr.toString());
+            }
+            nodePath.add(pathNode);
+
+            ConnectPoint previous = null;
+
+            if (path.size() == 1) {
+                ConnectPoint connectPoint = path.get(0);
+                nodeOutput.put("device", connectPoint.deviceId().toString());
+                nodeOutput.put("input", connectPoint.toString());
+                nodeOutput.put("flowCount", trace.getFlowsForDevice(connectPoint.deviceId()).size());
+                nodeOutput.set("flows", getFlowArray(trace, connectPoint, verbose));
+
+                List<GroupsInDevice> groupsInDevice = trace.getGroupOuputs(connectPoint.deviceId());
+
+                if (groupsInDevice != null) {
+                    groupsInDevice.forEach(output -> {
+                        nodeOutput.set("groups", getGroupObj(connectPoint, output, verbose));
+                    });
+                }
+
+            } else {
+                for (ConnectPoint connectPoint : path) {
+                    if (previous == null || !previous.deviceId().equals(connectPoint.deviceId())) {
+                        nodeOutput.put("device", connectPoint.deviceId().toString());
+                        nodeOutput.put("input", connectPoint.toString());
+                        nodeOutput.put("flows", trace.getFlowsForDevice(connectPoint.deviceId()).size());
+                        nodeOutput.put("verbose", verbose);
+                        nodeOutput.set("flows", getFlowArray(trace, connectPoint, verbose));
+                    } else {
+                        List<GroupsInDevice> groupsInDevice = trace.getGroupOuputs(connectPoint.deviceId());
+                        if (groupsInDevice != null) {
+                            groupsInDevice.forEach(output -> {
+                                nodeOutput.set("groups", getGroupObj(connectPoint, output, verbose));
+                            });
+
+                        }
+
+                    }
+                    previous = connectPoint;
+                }
+            }
+        }
+        nodeOutput.set("path", nodePath);
+        return nodeOutput;
+    }
+
+    //Return groups Object for a given trace and a specified level of verbosity
+    private ObjectNode getGroupObj(ConnectPoint connectPoint, GroupsInDevice output, boolean verbose) {
+        ArrayNode groupArray = mapper.createArrayNode();
+        ObjectNode groupsObj = mapper.createObjectNode();
+        if (output.getOutput().equals(connectPoint)) {
+            output.getGroups().forEach(group -> {
+                ObjectNode groups = mapper.createObjectNode();
+                if (verbose) {
+                    groups = codec(Group.class).encode(group, this);
+                } else {
+                    groups.put("groupId", group.id().toString());
+                }
+                groupArray.add(groups);
+            });
+            ArrayNode node = mapper.createArrayNode();
+            for (Criterion packet : output.getFinalPacket().criteria()) {
+                node.add(packet.toString());
+            }
+            groupsObj.set("outgoing packet", node);
+        }
+        groupsObj.set("groups", groupArray);
+        return groupsObj;
+    }
+
+    //Return flows Object for a given trace and a specified level of verbosity
+    private ArrayNode getFlowArray(StaticPacketTrace trace, ConnectPoint connectPoint, boolean verbose) {
+        ArrayNode flowArray = mapper.createArrayNode();
+        trace.getFlowsForDevice(connectPoint.deviceId()).forEach(f -> {
+            ObjectNode flows = mapper.createObjectNode();
+            if (verbose) {
+                flows = codec(FlowEntry.class).encode(f, this);
+            } else {
+                flows.put("flowId: ", f.id().toString());
+                flows.put("table: ", f.table().toString());
+                flows.put("selector: ", f.selector().criteria().toString());
+            }
+            flowArray.add(flows);
+        });
+        return flowArray;
+    }
+
+}
diff --git a/apps/t3/web/src/main/java/org/onosproject/t3/rest/package-info.java b/apps/t3/web/src/main/java/org/onosproject/t3/rest/package-info.java
new file mode 100644
index 0000000..e390f3e
--- /dev/null
+++ b/apps/t3/web/src/main/java/org/onosproject/t3/rest/package-info.java
@@ -0,0 +1,20 @@
+/*
+ * Copyright 2017-present Open Networking Foundation
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ *     http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+/**
+ * Troubleshooting toolkit for trellis fabrics implementation.
+ */
+package org.onosproject.t3.rest;
diff --git a/apps/t3/web/src/main/webapp/WEB-INF/web.xml b/apps/t3/web/src/main/webapp/WEB-INF/web.xml
new file mode 100644
index 0000000..d99a6ae
--- /dev/null
+++ b/apps/t3/web/src/main/webapp/WEB-INF/web.xml
@@ -0,0 +1,57 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<!--
+  ~ Copyright 2018-present Open Networking Foundation
+  ~
+  ~ Licensed under the Apache License, Version 2.0 (the "License");
+  ~ you may not use this file except in compliance with the License.
+  ~ You may obtain a copy of the License at
+  ~
+  ~     http://www.apache.org/licenses/LICENSE-2.0
+  ~
+  ~ Unless required by applicable law or agreed to in writing, software
+  ~ distributed under the License is distributed on an "AS IS" BASIS,
+  ~ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+  ~ See the License for the specific language governing permissions and
+  ~ limitations under the License.
+  -->
+<web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
+         xmlns="http://java.sun.com/xml/ns/javaee"
+         xmlns:web="http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd"
+         xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd"
+         id="ONOS" version="2.5">
+    <display-name>T3 REST API v1.0</display-name>
+
+    <security-constraint>
+        <web-resource-collection>
+            <web-resource-name>Secured</web-resource-name>
+            <url-pattern>/*</url-pattern>
+        </web-resource-collection>
+        <auth-constraint>
+            <role-name>admin</role-name>
+        </auth-constraint>
+    </security-constraint>
+
+    <security-role>
+        <role-name>admin</role-name>
+    </security-role>
+
+    <login-config>
+        <auth-method>BASIC</auth-method>
+        <realm-name>karaf</realm-name>
+    </login-config>
+
+    <servlet>
+        <servlet-name>JAX-RS Service</servlet-name>
+        <servlet-class>org.glassfish.jersey.servlet.ServletContainer</servlet-class>
+        <init-param>
+            <param-name>javax.ws.rs.Application</param-name>
+            <param-value>org.onosproject.t3.rest.T3WebApplication</param-value>
+        </init-param>
+        <load-on-startup>1</load-on-startup>
+    </servlet>
+
+    <servlet-mapping>
+        <servlet-name>JAX-RS Service</servlet-name>
+        <url-pattern>/*</url-pattern>
+    </servlet-mapping>
+</web-app>