GNPy integration
Installs the path with the best OSRn value according to gnpy, while also setting the power at each stage.

usage:
odtn-connect-gnpy-command http 127.0.0.1 8080 test test
odtn-gnpy-connection-command -b netconf:127.0.0.1:1101/1 netconf:127.0.0.1:1102/1

Change-Id: I82b6d57d07ab5a3873ce37eb44639ebaa85c733f
diff --git a/apps/odtn/service/BUILD b/apps/odtn/service/BUILD
index abf6ee6..5226c66 100644
--- a/apps/odtn/service/BUILD
+++ b/apps/odtn/service/BUILD
@@ -5,6 +5,17 @@
     "//apps/yang:onos-apps-yang",
     "//apps/optical-model:onos-apps-optical-model",
     "//protocols/netconf/api:onos-protocols-netconf-api",
+    "@jersey_client//jar",
+    "@jersey_server//jar",
+    "@jersey_common//jar",
+    "@jersey_security//jar",
+    "@httpclient_osgi//jar",
+    "@httpcore_osgi//jar",
+    "@javax_ws_rs_api//jar",
+    "@hk2_api//jar",
+    "@aopalliance_repackaged//jar",
+    "@javax_inject//jar",
+    "//utils/rest:onlab-rest",
 ]
 
 osgi_jar_with_tests(
diff --git a/apps/odtn/service/src/main/java/org/onosproject/odtn/GnpyService.java b/apps/odtn/service/src/main/java/org/onosproject/odtn/GnpyService.java
new file mode 100644
index 0000000..4016b3e
--- /dev/null
+++ b/apps/odtn/service/src/main/java/org/onosproject/odtn/GnpyService.java
@@ -0,0 +1,66 @@
+/*
+ * 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.odtn;
+
+import org.apache.commons.lang3.tuple.Pair;
+import org.onosproject.net.ConnectPoint;
+import org.onosproject.net.intent.IntentId;
+
+/**
+ * Interface for Integration with GNPy optical tool.
+ */
+public interface GnpyService {
+
+
+    /**
+     * Connectes to an instance of GNPY optical planning and simulation tool.
+     *
+     * @param protocol the protocol, eg HTTP or HTTPS
+     * @param ip       the ip of the service
+     * @param port     the port fo the service
+     * @param username the username of the service
+     * @param password the password.
+     * @return true if connection was successful
+     */
+    boolean connectGnpy(String protocol, String ip, String port, String username, String password);
+
+    /**
+     * Disconnects for the connected GNPy instance.
+     *
+     * @return true if successful
+     */
+    boolean disconnectGnpy();
+
+    /**
+     * Checks connectivity an instance of GNPY optical planning and simulation tool.
+     *
+     * @return true if connection was successful
+     */
+    boolean isConnected();
+
+    /**
+     * Obtains the best connectivity from A to B according to GSNR from GNPY, possibly bidirectional.
+     *
+     * @param ingress       the ingress connect point
+     * @param egress        the egress connect point
+     * @param bidirectional true if bidirectional connectivity is required.
+     * @return the Id of the intent that was installed as a result of the computation.
+     */
+    Pair<IntentId, Double> obtainConnectivity(ConnectPoint ingress, ConnectPoint egress, boolean bidirectional);
+
+
+}
diff --git a/apps/odtn/service/src/main/java/org/onosproject/odtn/cli/impl/ConnectGnpyCommand.java b/apps/odtn/service/src/main/java/org/onosproject/odtn/cli/impl/ConnectGnpyCommand.java
new file mode 100644
index 0000000..a397b9f
--- /dev/null
+++ b/apps/odtn/service/src/main/java/org/onosproject/odtn/cli/impl/ConnectGnpyCommand.java
@@ -0,0 +1,84 @@
+/*
+ * 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.odtn.cli.impl;
+
+import com.google.common.annotations.Beta;
+import com.google.common.base.Preconditions;
+import org.apache.karaf.shell.api.action.Argument;
+import org.apache.karaf.shell.api.action.Command;
+import org.apache.karaf.shell.api.action.Option;
+import org.apache.karaf.shell.api.action.lifecycle.Service;
+import org.onosproject.cli.AbstractShellCommand;
+import org.onosproject.odtn.GnpyService;
+import org.slf4j.Logger;
+
+import static org.slf4j.LoggerFactory.getLogger;
+
+@Beta
+@Service
+@Command(scope = "onos", name = "odtn-connect-gnpy-command",
+        description = "show tapi context command")
+public class ConnectGnpyCommand extends AbstractShellCommand {
+
+    private static final Logger log = getLogger(ConnectGnpyCommand.class);
+
+    @Argument(index = 0, name = "protocol",
+            description = "protocol for requests, e.g. HTTP or HTTPS",
+            required = true, multiValued = false)
+    String protocol = "";
+
+    @Argument(index = 1, name = "ip",
+            description = "Ip of GNPy instance",
+            required = true, multiValued = false)
+    String ip = "";
+
+    @Argument(index = 2, name = "port",
+            description = "Protocol of GNPy instance",
+            required = true, multiValued = false)
+    String port = "";
+
+    @Argument(index = 3, name = "username",
+            description = "Username of GNPy instance",
+            required = true, multiValued = false)
+    String username = "";
+
+    @Argument(index = 4, name = "password",
+            description = "Password of GNPy instance",
+            required = true, multiValued = false)
+    String password = "";
+
+    @Option(name = "-d", aliases = "--disconnect",
+            description = "If this argument is passed the connection with gNPY is removed.",
+            required = false, multiValued = false)
+    private boolean disconnect = false;
+
+    @Override
+    protected void doExecute() {
+        GnpyService service = get(GnpyService.class);
+        String msg;
+        if (disconnect) {
+            msg = service.disconnectGnpy() ? "gnpy disconnect" : "error in disconnecting gnpy";
+        } else {
+            Preconditions.checkNotNull(ip, "Ip must be specified");
+            Preconditions.checkNotNull(password, "password must be specified");
+            msg = service.connectGnpy(protocol, ip, port, username, password) ? "gnpy connected" :
+                    "error in connecting gnpy, please check logs";
+        }
+        print(msg);
+    }
+
+}
diff --git a/apps/odtn/service/src/main/java/org/onosproject/odtn/cli/impl/GnpyConnectionRequestCommand.java b/apps/odtn/service/src/main/java/org/onosproject/odtn/cli/impl/GnpyConnectionRequestCommand.java
new file mode 100644
index 0000000..c434fdb
--- /dev/null
+++ b/apps/odtn/service/src/main/java/org/onosproject/odtn/cli/impl/GnpyConnectionRequestCommand.java
@@ -0,0 +1,95 @@
+/*
+ * 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.odtn.cli.impl;
+
+import com.google.common.annotations.Beta;
+import org.apache.commons.lang3.tuple.Pair;
+import org.apache.karaf.shell.api.action.Argument;
+import org.apache.karaf.shell.api.action.Command;
+import org.apache.karaf.shell.api.action.Completion;
+import org.apache.karaf.shell.api.action.Option;
+import org.apache.karaf.shell.api.action.lifecycle.Service;
+import org.onosproject.cli.AbstractShellCommand;
+import org.onosproject.cli.net.ConnectPointCompleter;
+import org.onosproject.net.ConnectPoint;
+import org.onosproject.net.intent.IntentId;
+import org.onosproject.odtn.GnpyService;
+import org.slf4j.Logger;
+
+import static com.google.common.base.Preconditions.checkArgument;
+import static org.slf4j.LoggerFactory.getLogger;
+
+@Beta
+@Service
+@Command(scope = "onos", name = "odtn-gnpy-connection-command",
+        description = "show tapi context command")
+public class GnpyConnectionRequestCommand extends AbstractShellCommand {
+
+    private static final Logger log = getLogger(GnpyConnectionRequestCommand.class);
+
+    @Argument(index = 0, name = "ingress",
+            description = "Ingress Device/Port Description",
+            required = true, multiValued = false)
+    @Completion(ConnectPointCompleter.class)
+    String ingressString = "";
+
+    @Argument(index = 1, name = "egress",
+            description = "Egress Device/Port Description",
+            required = true, multiValued = false)
+    @Completion(ConnectPointCompleter.class)
+    String egressString = "";
+
+    @Option(name = "-b", aliases = "--bidirectional",
+            description = "If this argument is passed the optical link created will be bidirectional, " +
+                    "else the link will be unidirectional.",
+            required = false, multiValued = false)
+    private boolean bidirectional = false;
+
+    @Override
+    protected void doExecute() {
+        GnpyService service = get(GnpyService.class);
+
+        if (!service.isConnected()) {
+            print("gNPY is not connected, please issue `odtn-connect-gnpy-command` first");
+            return;
+        }
+
+        ConnectPoint ingress = createConnectPoint(ingressString);
+        ConnectPoint egress = createConnectPoint(egressString);
+
+        Pair<IntentId, Double> intentIdAndOsnr =
+                service.obtainConnectivity(ingress, egress, bidirectional);
+
+        if (intentIdAndOsnr != null) {
+            print("Optical Connectivity from %s to %s submitted through GNPy. \n", ingress, egress);
+            print("Expected GSNR %.2f dB", intentIdAndOsnr.getRight().doubleValue());
+            print("Intent: %s", intentIdAndOsnr.getLeft());
+        } else {
+            print("Error in submitting Optical Connectivity submitted through GNPy, please see logs");
+        }
+    }
+
+    private ConnectPoint createConnectPoint(String devicePortString) {
+        String[] splitted = devicePortString.split("/");
+
+        checkArgument(splitted.length == 2,
+                      "Connect point must be in \"deviceUri/portNumber\" format");
+
+        return ConnectPoint.deviceConnectPoint(devicePortString);
+    }
+
+}
diff --git a/apps/odtn/service/src/main/java/org/onosproject/odtn/impl/GnpyManager.java b/apps/odtn/service/src/main/java/org/onosproject/odtn/impl/GnpyManager.java
new file mode 100644
index 0000000..a474dfa
--- /dev/null
+++ b/apps/odtn/service/src/main/java/org/onosproject/odtn/impl/GnpyManager.java
@@ -0,0 +1,546 @@
+/*
+ * 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.odtn.impl;
+
+import com.fasterxml.jackson.core.JsonEncoding;
+import com.fasterxml.jackson.core.JsonFactory;
+import com.fasterxml.jackson.core.JsonGenerator;
+import com.fasterxml.jackson.databind.JsonNode;
+import com.fasterxml.jackson.databind.ObjectMapper;
+import com.fasterxml.jackson.databind.ObjectReader;
+import com.google.common.annotations.Beta;
+import org.apache.commons.lang3.tuple.Pair;
+import org.onlab.graph.ScalarWeight;
+import org.onosproject.core.ApplicationId;
+import org.onosproject.core.CoreService;
+import org.onosproject.net.ChannelSpacing;
+import org.onosproject.net.ConnectPoint;
+import org.onosproject.net.DefaultPath;
+import org.onosproject.net.Device;
+import org.onosproject.net.DeviceId;
+import org.onosproject.net.Direction;
+import org.onosproject.net.GridType;
+import org.onosproject.net.Link;
+import org.onosproject.net.OchSignal;
+import org.onosproject.net.Path;
+import org.onosproject.net.behaviour.PowerConfig;
+import org.onosproject.net.device.DeviceService;
+import org.onosproject.net.intent.Intent;
+import org.onosproject.net.intent.IntentEvent;
+import org.onosproject.net.intent.IntentId;
+import org.onosproject.net.intent.IntentListener;
+import org.onosproject.net.intent.IntentService;
+import org.onosproject.net.link.LinkService;
+import org.onosproject.net.provider.ProviderId;
+import org.onosproject.odtn.GnpyService;
+import org.onosproject.store.service.AtomicCounter;
+import org.onosproject.store.service.StorageService;
+import org.osgi.service.component.annotations.Activate;
+import org.osgi.service.component.annotations.Component;
+import org.osgi.service.component.annotations.Reference;
+import org.osgi.service.component.annotations.ReferenceCardinality;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+
+import javax.ws.rs.core.MediaType;
+import java.io.ByteArrayInputStream;
+import java.io.ByteArrayOutputStream;
+import java.io.IOException;
+import java.util.ArrayList;
+import java.util.HashMap;
+import java.util.Iterator;
+import java.util.List;
+import java.util.Map;
+import java.util.Set;
+import java.util.stream.Collectors;
+import java.util.stream.StreamSupport;
+
+import static java.lang.Math.log10;
+import static org.onosproject.net.ChannelSpacing.CHL_50GHZ;
+import static org.onosproject.net.ChannelSpacing.CHL_6P25GHZ;
+import static org.onosproject.net.optical.util.OpticalIntentUtility.createOpticalIntent;
+
+/**
+ * Implementation of GnpyService.
+ */
+@Beta
+@Component(immediate = true, service = GnpyService.class)
+public class GnpyManager implements GnpyService {
+
+    private final Logger log = LoggerFactory.getLogger(getClass());
+
+    @Reference(cardinality = ReferenceCardinality.MANDATORY)
+    protected IntentService intentService;
+
+    @Reference(cardinality = ReferenceCardinality.MANDATORY)
+    protected DeviceService deviceService;
+
+    @Reference(cardinality = ReferenceCardinality.MANDATORY)
+    protected CoreService coreService;
+
+    @Reference(cardinality = ReferenceCardinality.MANDATORY)
+    protected LinkService linkService;
+
+    @Reference(cardinality = ReferenceCardinality.MANDATORY)
+    protected StorageService storageService;
+
+    HttpUtil gnpyHttpUtil;
+
+    private static final String APP_ID = "org.onosproject.odtn-service";
+
+    private static final ProviderId PROVIDER_ID = new ProviderId("odtn", "gnpy");
+
+    private ApplicationId appId;
+
+    private AtomicCounter counter;
+
+    private Map<IntentId, GnpyPowerInfo> intentsPowerMap = new HashMap<>();
+
+    @Activate
+    protected void activate() {
+        log.info("Started");
+        appId = coreService.getAppId(APP_ID);
+        counter = storageService.getAtomicCounter("GNPy-connection-counter");
+    }
+
+
+    @Override
+    public boolean connectGnpy(String protocol, String ip, String port, String username, String password) {
+        gnpyHttpUtil = new HttpUtil(protocol, ip, port);
+        gnpyHttpUtil.connect(username, password);
+        return gnpyHttpUtil.get("/gnpy-experimental", MediaType.APPLICATION_JSON_TYPE) != null;
+    }
+
+    @Override
+    public boolean disconnectGnpy() {
+        gnpyHttpUtil.disconnect();
+        gnpyHttpUtil = null;
+        return true;
+    }
+
+    @Override
+    public boolean isConnected() {
+        return gnpyHttpUtil != null;
+    }
+
+    @Override
+    public Pair<IntentId, Double> obtainConnectivity(ConnectPoint ingress, ConnectPoint egress, boolean bidirectional) {
+        ByteArrayOutputStream connectivityRequest = createGnpyRequest(ingress, egress, bidirectional);
+        String response = gnpyHttpUtil.post(null, "/gnpy-experimental",
+                                            new ByteArrayInputStream(connectivityRequest.toByteArray()),
+                                            MediaType.APPLICATION_JSON_TYPE, String.class);
+        ObjectMapper om = new ObjectMapper();
+        final ObjectReader reader = om.reader();
+        JsonNode jsonNode;
+        try {
+            jsonNode = reader.readTree(response);
+            if (jsonNode == null) {
+                log.error("JsonNode is null for response {}", response);
+                return null;
+            }
+            log.info("Response {}", response);
+            String bestPath;
+            try {
+                bestPath = getBestOsnrPathKey(jsonNode);
+            } catch (IllegalStateException e) {
+                log.error("Exception while contacting GNPy", e);
+                return null;
+            }
+            OchSignal ochSignal = createOchSignal(jsonNode);
+            Map<DeviceId, Double> deviceAtoBPowerMap = new HashMap<>();
+            Map<DeviceId, Double> deviceBtoAPowerMap = new HashMap<>();
+            //TODO this list is currently only populated in the forward direction
+            List<DeviceId> deviceIds = getDeviceAndPopulatePowerMap(jsonNode, deviceAtoBPowerMap,
+                                                                    deviceBtoAPowerMap, bestPath);
+            Path suggestedPath = createSuggestedPath(deviceIds);
+            log.info("Suggested path {}", suggestedPath);
+
+            Intent intent = createOpticalIntent(ingress, egress, deviceService,
+                                                null, appId, bidirectional, ochSignal, suggestedPath);
+
+            intentsPowerMap.put(intent.id(), new GnpyPowerInfo(deviceAtoBPowerMap, deviceBtoAPowerMap,
+                                                               getLaunchPower(jsonNode), suggestedPath.links(),
+                                                               ingress, egress, ochSignal));
+            intentService.submit(intent);
+            return Pair.of(intent.id(), getOsnr(jsonNode, bestPath));
+        } catch (IOException e) {
+            log.error("Exception while reading response {}", response, e);
+            return null;
+        }
+    }
+
+    private String getBestOsnrPathKey(JsonNode connectivityReply) throws IllegalStateException {
+        Double bestOsnr = -100.0;
+        String bestPathId = "";
+        if (connectivityReply.get("result").asText().contains("Service error")) {
+            throw new IllegalStateException(connectivityReply.get("result").asText());
+        }
+        Iterator<JsonNode> paths = connectivityReply.get("result").get("response")
+                .elements();
+        while (paths.hasNext()) {
+            JsonNode path = paths.next();
+            String respId = path.get("response-id").asText();
+            double osnr = getOsnr(connectivityReply, respId);
+            if (osnr > bestOsnr) {
+                bestOsnr = osnr;
+                bestPathId = respId;
+            }
+        }
+        return bestPathId;
+    }
+
+    protected Path createSuggestedPath(List<DeviceId> deviceIds) {
+        List<Link> listLinks = new ArrayList<>();
+        for (int i = 0; i < deviceIds.size() - 1; i++) {
+            Set<Link> links = linkService.getDeviceLinks(deviceIds.get(i));
+
+            for (Link link : links) {
+                if (link.dst().deviceId().equals(deviceIds.get(i + 1))) {
+                    listLinks.add(link);
+                }
+            }
+        }
+        return new DefaultPath(PROVIDER_ID, listLinks, new ScalarWeight(1));
+
+    }
+
+    protected List<DeviceId> getDeviceAndPopulatePowerMap(JsonNode connectivityReply,
+                                                          Map<DeviceId, Double> deviceAtoBPowerMap,
+                                                          Map<DeviceId, Double> deviceBtoAPowerMap,
+                                                          String name) {
+        List<DeviceId> deviceIds = new ArrayList<>();
+        if (connectivityReply.has("result")
+                && connectivityReply.get("result").has("response")) {
+            JsonNode response = connectivityReply.get("result").get("response");
+            //getting the a-b path.
+            Iterator<JsonNode> paths = connectivityReply.get("result").get("response")
+                    .elements();
+            while (paths.hasNext()) {
+                JsonNode path = paths.next();
+                if (path.get("response-id").asText().equals(name)) {
+                    Iterator<JsonNode> elements = path.get("path-properties")
+                            .get("reversed-path-route-objects").elements();
+                    Iterable<JsonNode> iterable = () -> elements;
+                    List<JsonNode> elementsList = StreamSupport
+                            .stream(iterable.spliterator(), false)
+                            .collect(Collectors.toList());
+                    Iterator<JsonNode> reversePathRoute = path.get("path-properties")
+                            .get("reversed-path-route-objects").elements();
+                    Iterable<JsonNode> reversedIterable = () -> reversePathRoute;
+                    List<JsonNode> reversedElementsList = StreamSupport
+                            .stream(reversedIterable.spliterator(), false)
+                            .collect(Collectors.toList());
+                    for (int i = 0; i < elementsList.size() - 1; i++) {
+                        if (elementsList.get(i).get("path-route-object").has("num-unnum-hop")) {
+                            String elementId = elementsList.get(i).get("path-route-object")
+                                    .get("num-unnum-hop").get("node-id")
+                                    .asText();
+                            //TODO this is a workaround until we understand better the
+                            // topology mapping between ONOS and GNPy
+                            if (elementId.startsWith("netconf:")) {
+                                double power = -99;
+                                if (!elementsList.get(i).get("path-route-object")
+                                        .get("num-unnum-hop").get("gnpy-node-type")
+                                        .asText().equals("transceiver")) {
+                                    power = getPerHopPower(elementsList.get(i + 2));
+                                }
+                                deviceAtoBPowerMap.put(DeviceId.deviceId(elementId), power);
+                                for (int j = 0; j < reversedElementsList.size() - 1; j++) {
+                                    if (reversedElementsList.get(j).get("path-route-object").has("num-unnum-hop")) {
+                                        String reversedElementId = reversedElementsList.get(j).get("path-route-object")
+                                                .get("num-unnum-hop").get("node-id")
+                                                .asText();
+                                        double reversePower = -99;
+                                        if (reversedElementId.equals(elementId)) {
+                                            reversePower = getPerHopPower(reversedElementsList.get(j + 2));
+                                            deviceBtoAPowerMap.put(DeviceId.deviceId(elementId), reversePower);
+                                        }
+                                    }
+                                }
+                                deviceIds.add(DeviceId.deviceId(elementId));
+                            }
+                        }
+                    }
+                    break;
+                }
+            }
+        } else {
+            log.warn("Can't retrieve devices {}", connectivityReply);
+        }
+        return deviceIds;
+    }
+
+    protected OchSignal createOchSignal(JsonNode connectivityReply) throws IllegalArgumentException {
+        if (connectivityReply.has("result")
+                && connectivityReply.get("result").has("response")) {
+            Iterator<JsonNode> elements = connectivityReply.get("result").get("response").elements()
+                    .next().get("path-properties").get("path-route-objects").elements();
+            Iterable<JsonNode> iterable = () -> elements;
+            List<JsonNode> elementsList = StreamSupport
+                    .stream(iterable.spliterator(), false)
+                    .collect(Collectors.toList());
+            int n = 0;
+            int m = 0;
+            for (JsonNode node : elementsList) {
+                if (node.get("path-route-object").has("label-hop")) {
+                    n = node.get("path-route-object").get("label-hop").get("N").asInt();
+                    m = node.get("path-route-object").get("label-hop").get("M").asInt();
+                    break;
+                }
+            }
+            int offset = 193100;
+
+            double centralFreq = offset + (n * CHL_6P25GHZ.frequency().asGHz());
+            try {
+                int multiplier = getMultplier(centralFreq, GridType.DWDM, CHL_50GHZ);
+                return new OchSignal(GridType.DWDM, CHL_50GHZ, multiplier, 4);
+            } catch (RuntimeException e) {
+                /* catching RuntimeException as both NullPointerException (thrown by
+                 * checkNotNull) and IllegalArgumentException (thrown by checkArgument)
+                 * are subclasses of RuntimeException.
+                 */
+                throw new IllegalArgumentException(e);
+            }
+        }
+        return null;
+    }
+
+    protected double getLaunchPower(JsonNode connectivityReply) {
+        double power = -99;
+        if (connectivityReply.has("result")
+                && connectivityReply.get("result").has("response")) {
+            Iterator<JsonNode> elements = connectivityReply.get("result").get("response")
+                    .elements().next().get("path-properties").get("path-metric").elements();
+            Iterable<JsonNode> iterable = () -> elements;
+            List<JsonNode> elementsList = StreamSupport
+                    .stream(iterable.spliterator(), false)
+                    .collect(Collectors.toList());
+            for (JsonNode node : elementsList) {
+                if (node.has("metric-type") &&
+                        node.get("metric-type").asText().equals("reference_power")) {
+                    power = node.get("accumulative-value").asDouble();
+                    break;
+                }
+            }
+        }
+        return 10 * log10(power * 1000);
+    }
+
+    protected double getPerHopPower(JsonNode pathRouteObj) {
+        double power = -99;
+        if (pathRouteObj.get("path-route-object").has("target-channel-power")) {
+            power = pathRouteObj.get("path-route-object")
+                    .get("target-channel-power").get("value")
+                    .asDouble();
+        }
+        return power;
+    }
+
+    protected double getOsnr(JsonNode connectivityReply, String name) {
+        double osnr = -1;
+        if (connectivityReply.has("result")
+                && connectivityReply.get("result").has("response")) {
+            Iterator<JsonNode> paths = connectivityReply.get("result").get("response")
+                    .elements();
+            while (paths.hasNext()) {
+                JsonNode path = paths.next();
+                if (path.get("response-id").asText().equals(name)) {
+                    Iterator<JsonNode> elements = path.get("path-properties").get("path-metric").elements();
+                    Iterable<JsonNode> iterable = () -> elements;
+                    List<JsonNode> elementsList = StreamSupport
+                            .stream(iterable.spliterator(), false)
+                            .collect(Collectors.toList());
+                    for (JsonNode node : elementsList) {
+                        if (node.has("metric-type") &&
+                                node.get("metric-type").asText().equals("OSNR-0.1nm")) {
+                            osnr = node.get("accumulative-value").asDouble();
+                            break;
+                        }
+                    }
+                    if (osnr != -1) {
+                        break;
+                    }
+                }
+            }
+        }
+        return osnr;
+    }
+
+    private int getMultplier(double wavelength, GridType gridType, ChannelSpacing channelSpacing) {
+        long baseFreq;
+        switch (gridType) {
+            case DWDM:
+                baseFreq = 193100;
+                break;
+            case CWDM:
+            case FLEX:
+            case UNKNOWN:
+            default:
+                baseFreq = 0L;
+                break;
+        }
+        return (int) ((wavelength - baseFreq) / (channelSpacing.frequency().asGHz()));
+    }
+
+    protected ByteArrayOutputStream createGnpyRequest(ConnectPoint ingress,
+                                                      ConnectPoint egress, boolean bidirectional) {
+        /*
+        {
+  "path-request": [
+    {
+      "request-id": "first",
+      "source": "trx-Amsterdam",
+      "destination": "trx-Bremen",
+      "src-tp-id": "trx-Amsterdam",
+      "dst-tp-id": "trx-Bremen",
+      "bidirectional": false,
+      "path-constraints": {
+        "te-bandwidth": {
+          "technology": "flexi-grid",
+          "trx_type": "Voyager",
+          "trx_mode": null,
+          "effective-freq-slot": [
+            {
+              "N": "null",
+              "M": "null"
+            }
+          ],
+          "spacing": 50000000000.0,
+          "max-nb-of-channel": null,
+          "output-power": null,
+          "path_bandwidth": 100000000000.0
+        }
+      }
+    }
+    ]
+    }
+         */
+        ByteArrayOutputStream stream = new ByteArrayOutputStream();
+        try {
+            JsonGenerator generator = getJsonGenerator(stream);
+            generator.writeStartObject();
+            generator.writeArrayFieldStart("path-request");
+            generator.writeStartObject();
+            generator.writeStringField("request-id", "onos-" + counter.getAndIncrement());
+            generator.writeStringField("source", ingress.deviceId().toString());
+            generator.writeStringField("destination", egress.deviceId().toString());
+            generator.writeStringField("src-tp-id", ingress.deviceId().toString());
+            generator.writeStringField("dst-tp-id", egress.deviceId().toString());
+            generator.writeBooleanField("bidirectional", bidirectional);
+            generator.writeObjectFieldStart("path-constraints");
+            generator.writeObjectFieldStart("te-bandwidth");
+            generator.writeStringField("technology", "flexi-grid");
+            generator.writeStringField("trx_type", "Cassini"); //TODO make variable
+            generator.writeNullField("trx_mode");
+            generator.writeArrayFieldStart("effective-freq-slot");
+            generator.writeStartObject();
+            generator.writeStringField("N", "null");
+            generator.writeStringField("M", "null");
+            generator.writeEndObject();
+            generator.writeEndArray();
+            generator.writeNumberField("spacing", 50000000000.0);
+            generator.writeNullField("max-nb-of-channel");
+            generator.writeNullField("output-power");
+            generator.writeNumberField("path_bandwidth", 100000000000.0);
+            generator.writeEndObject();
+            generator.writeEndObject();
+            generator.writeEndObject();
+            generator.writeEndArray();
+            generator.writeEndObject();
+            generator.close();
+            return stream;
+        } catch (IOException e) {
+            log.error("Cant' create json", e);
+        }
+        return stream;
+
+    }
+
+    private JsonGenerator getJsonGenerator(ByteArrayOutputStream stream) throws IOException {
+        JsonFactory factory = new JsonFactory();
+        return factory.createGenerator(stream, JsonEncoding.UTF8);
+    }
+
+    /**
+     * Internal listener for tracking the intent deletion events.
+     */
+    private class InternalIntentListener implements IntentListener {
+
+        @Override
+        public boolean isRelevant(IntentEvent event) {
+            return intentsPowerMap.keySet().contains(event.subject().id());
+        }
+
+        @Override
+        public void event(IntentEvent event) {
+            setPathPower(event.subject());
+
+        }
+    }
+
+    private void setPathPower(Intent intent) {
+        GnpyPowerInfo powerInfo = intentsPowerMap.get(intent.id());
+        for (Link link : powerInfo.path()) {
+            Device ingressDev = deviceService.getDevice(link.src().deviceId());
+            if (ingressDev.is(PowerConfig.class)) {
+                if (powerInfo.deviceAtoBPowerMap().get(link.src().deviceId()) != -99) {
+                    log.info("Configuring power {} for {}",
+                             powerInfo.deviceAtoBPowerMap().get(link.src().deviceId()),
+                             link.src().deviceId());
+                    ingressDev.as(PowerConfig.class)
+                            .setTargetPower(link.src().port(), powerInfo.ochSignal(),
+                                            powerInfo.deviceAtoBPowerMap()
+                                                    .get(link.src().deviceId()));
+                } else {
+                    log.warn("Can't determine power for {}", link.src().deviceId());
+                }
+            }
+            Device egressDev = deviceService.getDevice(link.dst().deviceId());
+            if (egressDev.is(PowerConfig.class)) {
+                if (powerInfo.deviceBtoAPowerMap().get(link.dst().deviceId()) != -99) {
+                    log.info("Configuring power {} for {}",
+                             powerInfo.deviceBtoAPowerMap().get(link.dst().deviceId()),
+                             link.dst().deviceId());
+                    egressDev.as(PowerConfig.class)
+                            .setTargetPower(link.dst().port(), powerInfo.ochSignal(),
+                                            powerInfo.deviceBtoAPowerMap()
+                                                    .get(link.dst().deviceId()));
+                } else {
+                    log.warn("Can't determine power for {}", link.dst().deviceId());
+                }
+            }
+        }
+        Device ingressDevice = deviceService.getDevice(powerInfo.ingress().deviceId());
+        if (ingressDevice.is(PowerConfig.class)) {
+            if (powerInfo.launchPower() != -99) {
+                log.info("Configuring ingress with power {} for {}",
+                         powerInfo.launchPower(), ingressDevice);
+                ingressDevice.as(PowerConfig.class)
+                        .setTargetPower(powerInfo.ingress().port(), Direction.ALL, powerInfo.launchPower());
+            }
+        }
+        Device egressDevice = deviceService.getDevice(powerInfo.ingress().deviceId());
+        if (egressDevice.is(PowerConfig.class)) {
+            if (powerInfo.launchPower() != -99) {
+                log.info("Configuring egress with power {} for {}",
+                         powerInfo.launchPower(), ingressDevice);
+                ingressDevice.as(PowerConfig.class)
+                        .setTargetPower(powerInfo.ingress().port(), Direction.ALL, powerInfo.launchPower());
+            }
+        }
+    }
+}
diff --git a/apps/odtn/service/src/main/java/org/onosproject/odtn/impl/GnpyPowerInfo.java b/apps/odtn/service/src/main/java/org/onosproject/odtn/impl/GnpyPowerInfo.java
new file mode 100644
index 0000000..2923ec0
--- /dev/null
+++ b/apps/odtn/service/src/main/java/org/onosproject/odtn/impl/GnpyPowerInfo.java
@@ -0,0 +1,127 @@
+/*
+ * 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.odtn.impl;
+
+import com.google.common.annotations.Beta;
+import org.onosproject.net.ConnectPoint;
+import org.onosproject.net.DeviceId;
+import org.onosproject.net.Link;
+import org.onosproject.net.OchSignal;
+
+import java.util.List;
+import java.util.Map;
+
+/**
+ * This class contains information given by gNPY for a given path.
+ */
+@Beta
+public final class GnpyPowerInfo {
+
+    private Map<DeviceId, Double> deviceAtoBPowerMap;
+    private Map<DeviceId, Double> deviceBtoAPowerMap;
+    private double launchPower;
+    private List<Link> path;
+    private ConnectPoint ingress;
+    private ConnectPoint egress;
+    private OchSignal ochSignal;
+
+    /**
+     * Creates the class with information.
+     *
+     * @param deviceAtoBPowerMap the power in a to b direction
+     * @param deviceBtoAPowerMap the power in b to a direction
+     * @param launchPower        the power at the TXs
+     * @param path               the pat
+     * @param ingress            the ingress device (A)
+     * @param egress             the egress device (B)
+     * @param ochSignal          the signal
+     */
+    public GnpyPowerInfo(Map<DeviceId, Double> deviceAtoBPowerMap,
+                         Map<DeviceId, Double> deviceBtoAPowerMap, double launchPower,
+                         List<Link> path, ConnectPoint ingress, ConnectPoint egress,
+                         OchSignal ochSignal) {
+        this.deviceAtoBPowerMap = deviceAtoBPowerMap;
+        this.deviceBtoAPowerMap = deviceBtoAPowerMap;
+        this.launchPower = launchPower;
+        this.path = path;
+        this.ingress = ingress;
+        this.egress = egress;
+        this.ochSignal = ochSignal;
+    }
+
+    /**
+     * Retrieve the ingress connect point of the path (A).
+     *
+     * @return ingress connect point
+     */
+    public ConnectPoint ingress() {
+        return ingress;
+    }
+
+    /**
+     * Retrieve the egress connect point of the path (B).
+     *
+     * @return egress connect point
+     */
+    public ConnectPoint egress() {
+        return egress;
+    }
+
+    /**
+     * Retrieve the ochSignal.
+     *
+     * @return signal
+     */
+    public OchSignal ochSignal() {
+        return ochSignal;
+    }
+
+    /**
+     * Retrieve the the power in a to b direction.
+     *
+     * @return power map a to b
+     */
+    public Map<DeviceId, Double> deviceAtoBPowerMap() {
+        return deviceAtoBPowerMap;
+    }
+
+    /**
+     * Retrieve the power in b to a direction.
+     *
+     * @return power map b to a
+     */
+    public Map<DeviceId, Double> deviceBtoAPowerMap() {
+        return deviceBtoAPowerMap;
+    }
+
+    /**
+     * Retrieve the launch power at the TX, both A and B.
+     *
+     * @return ingress connect point
+     */
+    public double launchPower() {
+        return launchPower;
+    }
+
+    /**
+     * Retrieve the set of links for the path in the network.
+     *
+     * @return links fo the path
+     */
+    public List<Link> path() {
+        return path;
+    }
+}
diff --git a/apps/odtn/service/src/main/java/org/onosproject/odtn/impl/HttpUtil.java b/apps/odtn/service/src/main/java/org/onosproject/odtn/impl/HttpUtil.java
new file mode 100644
index 0000000..1da0e5d
--- /dev/null
+++ b/apps/odtn/service/src/main/java/org/onosproject/odtn/impl/HttpUtil.java
@@ -0,0 +1,236 @@
+/*
+ * Copyright 2016-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.odtn.impl;
+
+import com.google.common.annotations.Beta;
+import org.apache.commons.io.IOUtils;
+import org.glassfish.jersey.client.authentication.HttpAuthenticationFeature;
+import org.onosproject.net.DeviceId;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+
+import javax.net.ssl.SSLContext;
+import javax.net.ssl.TrustManager;
+import javax.net.ssl.X509TrustManager;
+import javax.ws.rs.ProcessingException;
+import javax.ws.rs.client.Client;
+import javax.ws.rs.client.ClientBuilder;
+import javax.ws.rs.client.Entity;
+import javax.ws.rs.client.WebTarget;
+import javax.ws.rs.core.MediaType;
+import javax.ws.rs.core.Response;
+import javax.ws.rs.core.Response.Status;
+import java.io.ByteArrayInputStream;
+import java.io.IOException;
+import java.io.InputStream;
+import java.nio.charset.StandardCharsets;
+import java.security.KeyManagementException;
+import java.security.NoSuchAlgorithmException;
+import java.security.cert.CertificateException;
+import java.security.cert.X509Certificate;
+
+/**
+ * The implementation of HttpUtils.
+ */
+@Beta
+public class HttpUtil {
+
+    private static final Logger log = LoggerFactory.getLogger(HttpUtil.class);
+    private static final String XML = "xml";
+    private static final String JSON = "json";
+    protected static final String DOUBLESLASH = "//";
+    protected static final String COLON = ":";
+    private static final int STATUS_OK = Response.Status.OK.getStatusCode();
+    private static final int STATUS_CREATED = Response.Status.CREATED.getStatusCode();
+    private static final int STATUS_ACCEPTED = Response.Status.ACCEPTED.getStatusCode();
+    private static final String HTTPS = "https";
+
+    private Client client = null;
+    private String protocol = null;
+    private String ip = null;
+    private String port = null;
+
+    public HttpUtil(String protocol, String ip, String port) {
+        //TODO check not null
+        this.protocol = protocol.equals("") ? HTTPS : protocol;
+        this.ip = ip;
+        this.port = port;
+    }
+
+    public void connect(String username, String password) {
+        client = ignoreSslClient();
+        authenticate(client, username, password);
+    }
+
+    public void disconnect() {
+        protocol = "";
+        ip = "";
+        port = "";
+        client = null;
+    }
+
+    public int post(String request, InputStream payload, MediaType mediaType) {
+        Response response = getResponse(request, payload, mediaType);
+        if (response == null) {
+            return Status.NO_CONTENT.getStatusCode();
+        }
+        return response.getStatus();
+    }
+
+    public <T> T post(DeviceId device, String request, InputStream payload, MediaType mediaType,
+                      Class<T> responseClass) {
+        Response response = getResponse(request, payload, mediaType);
+        if (response != null && response.hasEntity()) {
+            // Do not read the entity if the responseClass is of type Response. This would allow the
+            // caller to receive the Response directly and try to read its appropriate entity locally.
+            return responseClass == Response.class ? (T) response : response.readEntity(responseClass);
+        }
+        log.error("Response from device {} for request {} contains no entity", device, request);
+        return null;
+    }
+
+    private Response getResponse(String request, InputStream payload, MediaType mediaType) {
+
+        WebTarget wt = getWebTarget(request);
+
+        Response response = null;
+        if (payload != null) {
+            try {
+                response = wt.request(mediaType)
+                        .post(Entity.entity(IOUtils.toString(payload, StandardCharsets.UTF_8), mediaType));
+            } catch (IOException e) {
+                log.error("Cannot do POST {} request on GNPY because can't read payload", request);
+            }
+        } else {
+            response = wt.request(mediaType).post(Entity.entity(null, mediaType));
+        }
+        return response;
+    }
+
+    public int put(String request, InputStream payload, MediaType mediaType) {
+
+        WebTarget wt = getWebTarget(request);
+
+        Response response = null;
+        if (payload != null) {
+            try {
+                response = wt.request(mediaType).put(Entity.entity(IOUtils.
+                        toString(payload, StandardCharsets.UTF_8), mediaType));
+            } catch (IOException e) {
+                log.error("Cannot do POST {} request on GNPY because can't read payload", request);
+            }
+        } else {
+            response = wt.request(mediaType).put(Entity.entity(null, mediaType));
+        }
+
+        if (response == null) {
+            return Status.NO_CONTENT.getStatusCode();
+        }
+        return response.getStatus();
+    }
+
+    public InputStream get(String request, MediaType mediaType) {
+        WebTarget wt = getWebTarget(request);
+
+        Response s = wt.request(mediaType).get();
+
+        if (checkReply(s)) {
+            return new ByteArrayInputStream(s.readEntity((String.class)).getBytes(StandardCharsets.UTF_8));
+        }
+        return null;
+    }
+
+    public int delete(DeviceId device, String request, InputStream payload, MediaType mediaType) {
+
+        WebTarget wt = getWebTarget(request);
+
+        // FIXME: do we need to delete an entry by enclosing data in DELETE
+        // request?
+        // wouldn't it be nice to use PUT to implement the similar concept?
+        Response response = null;
+        try {
+            response = wt.request(mediaType).delete();
+        } catch (ProcessingException procEx) {
+            log.error("Cannot issue DELETE {} request on device {}", request, device);
+            return Status.SERVICE_UNAVAILABLE.getStatusCode();
+        }
+
+        return response.getStatus();
+    }
+
+    private void authenticate(Client client, String username, String password) {
+        client.register(HttpAuthenticationFeature.basic(username, password));
+
+    }
+
+    protected WebTarget getWebTarget(String request) {
+        log.debug("Sending request to URL {} ", getUrlString(request));
+        return client.target(getUrlString(request));
+    }
+
+    protected String getUrlString(String request) {
+        return protocol + COLON + DOUBLESLASH + ip + COLON + port + request;
+    }
+
+    private boolean checkReply(Response response) {
+        if (response != null) {
+            boolean statusCode = checkStatusCode(response.getStatus());
+            if (!statusCode && response.hasEntity()) {
+                log.error("Failed request, HTTP error msg : " + response.readEntity(String.class));
+            }
+            return statusCode;
+        }
+        log.error("Null reply from device");
+        return false;
+    }
+
+    private boolean checkStatusCode(int statusCode) {
+        if (statusCode == STATUS_OK || statusCode == STATUS_CREATED || statusCode == STATUS_ACCEPTED) {
+            return true;
+        } else {
+            log.error("Failed request, HTTP error code : " + statusCode);
+            return false;
+        }
+    }
+
+    private Client ignoreSslClient() {
+        SSLContext sslcontext = null;
+
+        try {
+            sslcontext = SSLContext.getInstance("TLS");
+            sslcontext.init(null, new TrustManager[]{new X509TrustManager() {
+                @Override
+                public void checkClientTrusted(X509Certificate[] arg0, String arg1) throws CertificateException {
+                }
+
+                @Override
+                public void checkServerTrusted(X509Certificate[] arg0, String arg1) throws CertificateException {
+                }
+
+                @Override
+                public X509Certificate[] getAcceptedIssuers() {
+                    return new X509Certificate[0];
+                }
+            } }, new java.security.SecureRandom());
+        } catch (NoSuchAlgorithmException | KeyManagementException e) {
+            throw new IllegalStateException(e);
+        }
+
+        return ClientBuilder.newBuilder().sslContext(sslcontext).hostnameVerifier((s1, s2) -> true).build();
+    }
+
+}
diff --git a/apps/odtn/service/src/main/resources/gnpy-response.json b/apps/odtn/service/src/main/resources/gnpy-response.json
new file mode 100644
index 0000000..2446e4b
--- /dev/null
+++ b/apps/odtn/service/src/main/resources/gnpy-response.json
@@ -0,0 +1,1010 @@
+{
+  "result": {
+    "response": [
+      {
+        "response-id": "first",
+        "path-properties": {
+          "path-metric": [
+            {
+              "metric-type": "SNR-bandwidth",
+              "accumulative-value": 16.92
+            },
+            {
+              "metric-type": "SNR-0.1nm",
+              "accumulative-value": 21.0
+            },
+            {
+              "metric-type": "OSNR-bandwidth",
+              "accumulative-value": 16.92
+            },
+            {
+              "metric-type": "OSNR-0.1nm",
+              "accumulative-value": 21.0
+            },
+            {
+              "metric-type": "reference_power",
+              "accumulative-value": 0.001
+            },
+            {
+              "metric-type": "path_bandwidth",
+              "accumulative-value": 100000000000.0
+            }
+          ],
+          "z-a-path-metric": [
+            {
+              "metric-type": "SNR-bandwidth",
+              "accumulative-value": 16.92
+            },
+            {
+              "metric-type": "SNR-0.1nm",
+              "accumulative-value": 21.0
+            },
+            {
+              "metric-type": "OSNR-bandwidth",
+              "accumulative-value": 16.92
+            },
+            {
+              "metric-type": "OSNR-0.1nm",
+              "accumulative-value": 21.0
+            },
+            {
+              "metric-type": "reference_power",
+              "accumulative-value": 0.001
+            },
+            {
+              "metric-type": "path_bandwidth",
+              "accumulative-value": 100000000000.0
+            }
+          ],
+          "path-route-objects": [
+            {
+              "path-route-object": {
+                "index": 0,
+                "num-unnum-hop": {
+                  "node-id": "trx-Amsterdam",
+                  "link-tp-id": "trx-Amsterdam",
+                  "gnpy-node-type": "transceiver"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 1,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 2,
+                "transponder": {
+                  "transponder-type": "Voyager",
+                  "transponder-mode": "mode 1"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 3,
+                "num-unnum-hop": {
+                  "node-id": "roadm-Amsterdam-AD",
+                  "link-tp-id": "roadm-Amsterdam-AD",
+                  "gnpy-node-type": "ROADM"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 4,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 5,
+                "target-channel-power": {
+                  "value": -12
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 6,
+                "num-unnum-hop": {
+                  "node-id": "splice-(roadm-Amsterdam-AD)-(patch-(roadm-Amsterdam-AD)-(roadm-Amsterdam-L2))",
+                  "link-tp-id": "splice-(roadm-Amsterdam-AD)-(patch-(roadm-Amsterdam-AD)-(roadm-Amsterdam-L2))"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 7,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 8,
+                "num-unnum-hop": {
+                  "node-id": "patch-(roadm-Amsterdam-AD)-(roadm-Amsterdam-L2)",
+                  "link-tp-id": "patch-(roadm-Amsterdam-AD)-(roadm-Amsterdam-L2)"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 9,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 10,
+                "num-unnum-hop": {
+                  "node-id": "splice-(patch-(roadm-Amsterdam-AD)-(roadm-Amsterdam-L2))-(roadm-Amsterdam-L2)",
+                  "link-tp-id": "splice-(patch-(roadm-Amsterdam-AD)-(roadm-Amsterdam-L2))-(roadm-Amsterdam-L2)"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 11,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 12,
+                "num-unnum-hop": {
+                  "node-id": "roadm-Amsterdam-L2",
+                  "link-tp-id": "roadm-Amsterdam-L2",
+                  "gnpy-node-type": "ROADM"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 13,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 14,
+                "target-channel-power": {
+                  "value": -23
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 15,
+                "num-unnum-hop": {
+                  "node-id": "roadm-Amsterdam-L2-booster",
+                  "link-tp-id": "roadm-Amsterdam-L2-booster",
+                  "gnpy-node-type": "EDFA"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 16,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 17,
+                "target-channel-power": {
+                  "value": -1
+                },
+                "output-voa": {
+                  "value": 12.0
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 18,
+                "num-unnum-hop": {
+                  "node-id": "fiber-Amsterdam-Cologne",
+                  "link-tp-id": "fiber-Amsterdam-Cologne"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 19,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 20,
+                "num-unnum-hop": {
+                  "node-id": "roadm-Cologne-L1-preamp",
+                  "link-tp-id": "roadm-Cologne-L1-preamp",
+                  "gnpy-node-type": "EDFA"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 21,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 22,
+                "target-channel-power": {
+                  "value": 1.0
+                },
+                "output-voa": {
+                  "value": 0.0
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 23,
+                "num-unnum-hop": {
+                  "node-id": "roadm-Cologne-L1",
+                  "link-tp-id": "roadm-Cologne-L1",
+                  "gnpy-node-type": "ROADM"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 24,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 25,
+                "target-channel-power": {
+                  "value": -12
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 26,
+                "num-unnum-hop": {
+                  "node-id": "splice-(roadm-Cologne-L1)-(patch-(roadm-Cologne-L1)-(roadm-Cologne-L2))",
+                  "link-tp-id": "splice-(roadm-Cologne-L1)-(patch-(roadm-Cologne-L1)-(roadm-Cologne-L2))"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 27,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 28,
+                "num-unnum-hop": {
+                  "node-id": "patch-(roadm-Cologne-L1)-(roadm-Cologne-L2)",
+                  "link-tp-id": "patch-(roadm-Cologne-L1)-(roadm-Cologne-L2)"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 29,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 30,
+                "num-unnum-hop": {
+                  "node-id": "splice-(patch-(roadm-Cologne-L1)-(roadm-Cologne-L2))-(roadm-Cologne-L2)",
+                  "link-tp-id": "splice-(patch-(roadm-Cologne-L1)-(roadm-Cologne-L2))-(roadm-Cologne-L2)"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 31,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 32,
+                "num-unnum-hop": {
+                  "node-id": "roadm-Cologne-L2",
+                  "link-tp-id": "roadm-Cologne-L2",
+                  "gnpy-node-type": "ROADM"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 33,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 34,
+                "target-channel-power": {
+                  "value": -23
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 35,
+                "num-unnum-hop": {
+                  "node-id": "roadm-Cologne-L2-booster",
+                  "link-tp-id": "roadm-Cologne-L2-booster",
+                  "gnpy-node-type": "EDFA"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 36,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 37,
+                "target-channel-power": {
+                  "value": -1
+                },
+                "output-voa": {
+                  "value": 12.0
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 38,
+                "num-unnum-hop": {
+                  "node-id": "fiber-Cologne-Bremen",
+                  "link-tp-id": "fiber-Cologne-Bremen"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 39,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 40,
+                "num-unnum-hop": {
+                  "node-id": "roadm-Bremen-L1-preamp",
+                  "link-tp-id": "roadm-Bremen-L1-preamp",
+                  "gnpy-node-type": "EDFA"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 41,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 42,
+                "target-channel-power": {
+                  "value": 1.0
+                },
+                "output-voa": {
+                  "value": 0.0
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 43,
+                "num-unnum-hop": {
+                  "node-id": "roadm-Bremen-L1",
+                  "link-tp-id": "roadm-Bremen-L1",
+                  "gnpy-node-type": "ROADM"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 44,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 45,
+                "target-channel-power": {
+                  "value": -12
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 46,
+                "num-unnum-hop": {
+                  "node-id": "splice-(roadm-Bremen-L1)-(patch-(roadm-Bremen-L1)-(roadm-Bremen-AD))",
+                  "link-tp-id": "splice-(roadm-Bremen-L1)-(patch-(roadm-Bremen-L1)-(roadm-Bremen-AD))"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 47,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 48,
+                "num-unnum-hop": {
+                  "node-id": "patch-(roadm-Bremen-L1)-(roadm-Bremen-AD)",
+                  "link-tp-id": "patch-(roadm-Bremen-L1)-(roadm-Bremen-AD)"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 49,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 50,
+                "num-unnum-hop": {
+                  "node-id": "splice-(patch-(roadm-Bremen-L1)-(roadm-Bremen-AD))-(roadm-Bremen-AD)",
+                  "link-tp-id": "splice-(patch-(roadm-Bremen-L1)-(roadm-Bremen-AD))-(roadm-Bremen-AD)"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 51,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 52,
+                "num-unnum-hop": {
+                  "node-id": "roadm-Bremen-AD",
+                  "link-tp-id": "roadm-Bremen-AD",
+                  "gnpy-node-type": "ROADM"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 53,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 54,
+                "target-channel-power": {
+                  "value": -25
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 55,
+                "num-unnum-hop": {
+                  "node-id": "trx-Bremen",
+                  "link-tp-id": "trx-Bremen",
+                  "gnpy-node-type": "transceiver"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 56,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 57,
+                "transponder": {
+                  "transponder-type": "Voyager",
+                  "transponder-mode": "mode 1"
+                }
+              }
+            }
+          ]
+        }
+      },
+      {
+        "response-id": "second",
+        "path-properties": {
+          "path-metric": [
+            {
+              "metric-type": "SNR-bandwidth",
+              "accumulative-value": 19.38
+            },
+            {
+              "metric-type": "SNR-0.1nm",
+              "accumulative-value": 23.46
+            },
+            {
+              "metric-type": "OSNR-bandwidth",
+              "accumulative-value": 19.38
+            },
+            {
+              "metric-type": "OSNR-0.1nm",
+              "accumulative-value": 23.47
+            },
+            {
+              "metric-type": "reference_power",
+              "accumulative-value": 0.001
+            },
+            {
+              "metric-type": "path_bandwidth",
+              "accumulative-value": 100000000000.0
+            }
+          ],
+          "z-a-path-metric": [
+            {
+              "metric-type": "SNR-bandwidth",
+              "accumulative-value": 19.38
+            },
+            {
+              "metric-type": "SNR-0.1nm",
+              "accumulative-value": 23.46
+            },
+            {
+              "metric-type": "OSNR-bandwidth",
+              "accumulative-value": 19.38
+            },
+            {
+              "metric-type": "OSNR-0.1nm",
+              "accumulative-value": 23.47
+            },
+            {
+              "metric-type": "reference_power",
+              "accumulative-value": 0.001
+            },
+            {
+              "metric-type": "path_bandwidth",
+              "accumulative-value": 100000000000.0
+            }
+          ],
+          "path-route-objects": [
+            {
+              "path-route-object": {
+                "index": 0,
+                "num-unnum-hop": {
+                  "node-id": "trx-Amsterdam",
+                  "link-tp-id": "trx-Amsterdam",
+                  "gnpy-node-type": "transceiver"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 1,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 2,
+                "transponder": {
+                  "transponder-type": "Voyager",
+                  "transponder-mode": "mode 1"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 3,
+                "num-unnum-hop": {
+                  "node-id": "roadm-Amsterdam-AD",
+                  "link-tp-id": "roadm-Amsterdam-AD",
+                  "gnpy-node-type": "ROADM"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 4,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 5,
+                "target-channel-power": {
+                  "value": -12
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 6,
+                "num-unnum-hop": {
+                  "node-id": "splice-(roadm-Amsterdam-AD)-(patch-(roadm-Amsterdam-AD)-(roadm-Amsterdam-L1))",
+                  "link-tp-id": "splice-(roadm-Amsterdam-AD)-(patch-(roadm-Amsterdam-AD)-(roadm-Amsterdam-L1))"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 7,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 8,
+                "num-unnum-hop": {
+                  "node-id": "patch-(roadm-Amsterdam-AD)-(roadm-Amsterdam-L1)",
+                  "link-tp-id": "patch-(roadm-Amsterdam-AD)-(roadm-Amsterdam-L1)"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 9,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 10,
+                "num-unnum-hop": {
+                  "node-id": "splice-(patch-(roadm-Amsterdam-AD)-(roadm-Amsterdam-L1))-(roadm-Amsterdam-L1)",
+                  "link-tp-id": "splice-(patch-(roadm-Amsterdam-AD)-(roadm-Amsterdam-L1))-(roadm-Amsterdam-L1)"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 11,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 12,
+                "num-unnum-hop": {
+                  "node-id": "roadm-Amsterdam-L1",
+                  "link-tp-id": "roadm-Amsterdam-L1",
+                  "gnpy-node-type": "ROADM"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 13,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 14,
+                "target-channel-power": {
+                  "value": -23
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 15,
+                "num-unnum-hop": {
+                  "node-id": "roadm-Amsterdam-L1-booster",
+                  "link-tp-id": "roadm-Amsterdam-L1-booster",
+                  "gnpy-node-type": "EDFA"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 16,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 17,
+                "target-channel-power": {
+                  "value": -1
+                },
+                "output-voa": {
+                  "value": 12.0
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 18,
+                "num-unnum-hop": {
+                  "node-id": "fiber-Amsterdam-Bremen",
+                  "link-tp-id": "fiber-Amsterdam-Bremen"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 19,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 20,
+                "num-unnum-hop": {
+                  "node-id": "roadm-Bremen-L2-preamp",
+                  "link-tp-id": "roadm-Bremen-L2-preamp",
+                  "gnpy-node-type": "EDFA"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 21,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 22,
+                "target-channel-power": {
+                  "value": 1.0
+                },
+                "output-voa": {
+                  "value": 0.0
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 23,
+                "num-unnum-hop": {
+                  "node-id": "roadm-Bremen-L2",
+                  "link-tp-id": "roadm-Bremen-L2",
+                  "gnpy-node-type": "ROADM"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 24,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 25,
+                "target-channel-power": {
+                  "value": -12
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 26,
+                "num-unnum-hop": {
+                  "node-id": "splice-(roadm-Bremen-L2)-(patch-(roadm-Bremen-L2)-(roadm-Bremen-AD))",
+                  "link-tp-id": "splice-(roadm-Bremen-L2)-(patch-(roadm-Bremen-L2)-(roadm-Bremen-AD))"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 27,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 28,
+                "num-unnum-hop": {
+                  "node-id": "patch-(roadm-Bremen-L2)-(roadm-Bremen-AD)",
+                  "link-tp-id": "patch-(roadm-Bremen-L2)-(roadm-Bremen-AD)"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 29,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 30,
+                "num-unnum-hop": {
+                  "node-id": "splice-(patch-(roadm-Bremen-L2)-(roadm-Bremen-AD))-(roadm-Bremen-AD)",
+                  "link-tp-id": "splice-(patch-(roadm-Bremen-L2)-(roadm-Bremen-AD))-(roadm-Bremen-AD)"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 31,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 32,
+                "num-unnum-hop": {
+                  "node-id": "roadm-Bremen-AD",
+                  "link-tp-id": "roadm-Bremen-AD",
+                  "gnpy-node-type": "ROADM"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 33,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 34,
+                "target-channel-power": {
+                  "value": -25
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 35,
+                "num-unnum-hop": {
+                  "node-id": "trx-Bremen",
+                  "link-tp-id": "trx-Bremen",
+                  "gnpy-node-type": "transceiver"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 36,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 37,
+                "transponder": {
+                  "transponder-type": "Voyager",
+                  "transponder-mode": "mode 1"
+                }
+              }
+            }
+          ]
+        }
+      }
+    ]
+  }
+}
diff --git a/apps/odtn/service/src/test/java/org/onosproject/odtn/impl/GnpyManagerTest.java b/apps/odtn/service/src/test/java/org/onosproject/odtn/impl/GnpyManagerTest.java
new file mode 100644
index 0000000..438491e
--- /dev/null
+++ b/apps/odtn/service/src/test/java/org/onosproject/odtn/impl/GnpyManagerTest.java
@@ -0,0 +1,205 @@
+/*
+ * Copyright 2019-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.odtn.impl;
+
+import com.fasterxml.jackson.databind.JsonNode;
+import com.fasterxml.jackson.databind.ObjectMapper;
+import com.google.common.collect.ImmutableSet;
+import org.junit.Before;
+import org.junit.Test;
+import org.onosproject.core.ApplicationId;
+import org.onosproject.core.CoreServiceAdapter;
+import org.onosproject.core.DefaultApplicationId;
+import org.onosproject.net.ConnectPoint;
+import org.onosproject.net.DefaultLink;
+import org.onosproject.net.DeviceId;
+import org.onosproject.net.GridType;
+import org.onosproject.net.Link;
+import org.onosproject.net.OchSignal;
+import org.onosproject.net.Path;
+import org.onosproject.net.link.LinkServiceAdapter;
+import org.onosproject.net.provider.ProviderId;
+import org.onosproject.store.service.TestStorageService;
+
+import java.io.IOException;
+import java.util.HashMap;
+import java.util.Iterator;
+import java.util.List;
+import java.util.Map;
+import java.util.Set;
+import java.util.stream.Collectors;
+import java.util.stream.StreamSupport;
+
+import static junit.framework.Assert.assertTrue;
+import static junit.framework.TestCase.assertEquals;
+
+/**
+ * Test for parsing gNPY json files.
+ */
+public class GnpyManagerTest {
+
+    private static final String REQUEST = "{\"path-request\":[{\"request-id\":\"onos-0\"," +
+            "\"source\":\"netconf:10.0.254.93:830\"," +
+            "\"destination\":\"netconf:10.0.254.94:830\",\"src-tp-id\":" +
+            "\"netconf:10.0.254.93:830\",\"dst-tp-id\":" +
+            "\"netconf:10.0.254.94:830\",\"bidirectional\":true," +
+            "\"path-constraints\":{\"te-bandwidth\":" +
+            "{\"technology\":\"flexi-grid\",\"trx_type\":\"Cassini\"," +
+            "\"trx_mode\":null,\"effective-freq-slot\":" +
+            "[{\"N\":\"null\",\"M\":\"null\"}],\"spacing\":5.0E10," +
+            "\"max-nb-of-channel\":null,\"output-power\":null," +
+            "\"path_bandwidth\":1.0E11}}}]}";
+
+    private ConnectPoint tx1 = ConnectPoint.fromString("netconf:10.0.254.93:830/1");
+    private ConnectPoint rdm1tx1 = ConnectPoint.fromString("netconf:10.0.254.107:830/1");
+    private ConnectPoint rdm1ln1 = ConnectPoint.fromString("netconf:10.0.254.107:830/2");
+    private ConnectPoint ln1rdm1 = ConnectPoint.fromString("netconf:10.0.254.101:830/1");
+    private ConnectPoint ln1ln2 = ConnectPoint.fromString("netconf:10.0.254.101:830/2");
+    private ConnectPoint ln2ln1 = ConnectPoint.fromString("netconf:10.0.254.102:830/1");
+    private ConnectPoint ln2rdm2 = ConnectPoint.fromString("netconf:10.0.254.102:830/2");
+    private ConnectPoint rdm2ln2 = ConnectPoint.fromString("netconf:10.0.254.225:830/1");
+    private ConnectPoint rdm2tx2 = ConnectPoint.fromString("netconf:10.0.254.225:830/2");
+    private ConnectPoint tx2 = ConnectPoint.fromString("netconf:10.0.254.94:830/1");
+    private Link tx1rdm1Link = DefaultLink.builder().type(Link.Type.OPTICAL)
+            .providerId(ProviderId.NONE).src(rdm1tx1).dst(tx1).build();
+    private Link rmd1ln1Link = DefaultLink.builder().type(Link.Type.OPTICAL)
+            .providerId(ProviderId.NONE).src(ln1rdm1).dst(rdm1ln1).build();
+    private Link ln1ln2Link = DefaultLink.builder().type(Link.Type.OPTICAL)
+            .providerId(ProviderId.NONE).src(ln2ln1).dst(ln1ln2).build();
+    private Link ln2rdm2Link = DefaultLink.builder().type(Link.Type.OPTICAL)
+            .providerId(ProviderId.NONE).src(rdm2ln2).dst(ln2rdm2).build();
+    private Link tx2rmd2Link = DefaultLink.builder().type(Link.Type.OPTICAL)
+            .providerId(ProviderId.NONE).src(tx2).dst(rdm2tx2).build();
+
+    private GnpyManager manager;
+    private JsonNode reply;
+
+    @Before
+    public void setUp() throws Exception {
+        ObjectMapper mapper = new ObjectMapper();
+        reply = mapper.readTree(this.getClass().getResourceAsStream("gnpy-response.json"));
+        manager = new GnpyManager();
+        manager.storageService = new TestStorageService();
+        manager.linkService = new InternalLinkService();
+        manager.coreService = new InternalCoreService();
+        manager.activate();
+    }
+
+    @Test
+    public void testCreateSuggestedPath() throws IOException {
+        Map<DeviceId, Double> deviceAtoBPowerMap = new HashMap<>();
+        Map<DeviceId, Double> deviceBtoAPowerMap = new HashMap<>();
+        List<DeviceId> deviceIds = manager.getDeviceAndPopulatePowerMap(reply, deviceAtoBPowerMap,
+                                                                        deviceBtoAPowerMap, "second");
+        Path path = manager.createSuggestedPath(deviceIds);
+        assertTrue(path.links().contains(tx1rdm1Link));
+        assertTrue(path.links().contains(rmd1ln1Link));
+        assertTrue(path.links().contains(ln1ln2Link));
+        assertTrue(path.links().contains(ln2rdm2Link));
+        assertTrue(path.links().contains(tx2rmd2Link));
+        assertEquals(path.src(), tx2);
+        assertEquals(path.dst(), tx1);
+
+    }
+
+    @Test
+    public void testgetDevicePowerMap() throws IOException {
+        Map<DeviceId, Double> deviceAtoBPowerMap = new HashMap<>();
+        Map<DeviceId, Double> deviceBtoAPowerMap = new HashMap<>();
+        manager.getDeviceAndPopulatePowerMap(reply, deviceAtoBPowerMap, deviceBtoAPowerMap, "second");
+        assertEquals(-25.0, deviceAtoBPowerMap.get(DeviceId.deviceId("netconf:10.0.254.107:830")));
+        assertEquals(-12.0, deviceAtoBPowerMap.get(DeviceId.deviceId("netconf:10.0.254.225:830")));
+        assertEquals(-12.0, deviceBtoAPowerMap.get(DeviceId.deviceId("netconf:10.0.254.225:830")));
+        assertEquals(-25.0, deviceBtoAPowerMap.get(DeviceId.deviceId("netconf:10.0.254.107:830")));
+    }
+
+    @Test
+    public void testGetLaunchPower() throws IOException {
+        double power = manager.getLaunchPower(reply);
+        assertEquals(0.0, power);
+    }
+
+    @Test
+    public void testGetPerHopPower() throws IOException {
+        JsonNode response = reply.get("result").get("response");
+        //getting the a-b path.
+        JsonNode responseObj = response.elements()
+                .next();
+        Iterator<JsonNode> elements = responseObj.get("path-properties")
+                .get("path-route-objects").elements();
+        Iterable<JsonNode> iterable = () -> elements;
+        List<JsonNode> elementsList = StreamSupport
+                .stream(iterable.spliterator(), false)
+                .collect(Collectors.toList());
+        double power = manager.getPerHopPower(elementsList.get(5));
+        assertEquals(-12.0, power);
+    }
+
+    @Test
+    public void testGetOsnr() throws IOException {
+        double osnr = manager.getOsnr(reply, "second");
+        assertEquals(23.47, osnr);
+    }
+
+    @Test
+    public void testCreateOchSignal() throws IOException {
+        OchSignal signal = manager.createOchSignal(reply);
+        System.out.println(signal);
+        assertEquals(signal.gridType(), GridType.DWDM);
+        assertEquals(signal.slotWidth().asGHz(), 50.000);
+        assertEquals(-35, signal.spacingMultiplier());
+    }
+
+    @Test
+    public void testCreateGnpyRequest() {
+        ConnectPoint ingress = ConnectPoint.fromString("netconf:10.0.254.93:830/1");
+        ConnectPoint egress = ConnectPoint.fromString("netconf:10.0.254.94:830/1");
+        String output = manager.createGnpyRequest(ingress, egress, true).toString();
+        System.out.println(output);
+        assertEquals("Json to create network connectivity is wrong", REQUEST, output);
+    }
+
+    private class InternalLinkService extends LinkServiceAdapter {
+        @Override
+        public Set<Link> getDeviceLinks(DeviceId deviceId) {
+            if (deviceId.equals(DeviceId.deviceId("netconf:10.0.254.94:830"))) {
+                return ImmutableSet.of(tx2rmd2Link);
+            } else if (deviceId.equals(DeviceId.deviceId("netconf:10.0.254.107:830"))) {
+                return ImmutableSet.of(tx1rdm1Link);
+            } else if (deviceId.equals(DeviceId.deviceId("netconf:10.0.254.101:830"))) {
+                return ImmutableSet.of(rmd1ln1Link);
+            } else if (deviceId.equals(DeviceId.deviceId("netconf:10.0.254.102:830"))) {
+                return ImmutableSet.of(ln1ln2Link);
+            } else if (deviceId.equals(DeviceId.deviceId("netconf:10.0.254.225:830"))) {
+                return ImmutableSet.of(ln2rdm2Link);
+            }
+            return ImmutableSet.of();
+        }
+    }
+
+    private class InternalCoreService extends CoreServiceAdapter {
+        @Override
+        public ApplicationId getAppId(String name) {
+            return new DefaultApplicationId(1, name);
+        }
+    }
+//    private class InternalStoreService extends StorageServiceAdapter {
+//        @Override
+//        public AtomicCounterBuilder atomicCounterBuilder() {
+//            return TestAtomicCounter.builder();
+//        }
+//    }
+}
diff --git a/apps/odtn/service/src/test/resources/org/onosproject/odtn/impl/gnpy-response.json b/apps/odtn/service/src/test/resources/org/onosproject/odtn/impl/gnpy-response.json
new file mode 100644
index 0000000..72a1273
--- /dev/null
+++ b/apps/odtn/service/src/test/resources/org/onosproject/odtn/impl/gnpy-response.json
@@ -0,0 +1,1900 @@
+{
+  "result": {
+    "response": [
+      {
+        "response-id": "first",
+        "path-properties": {
+          "path-metric": [
+            {
+              "metric-type": "SNR-bandwidth",
+              "accumulative-value": 16.92
+            },
+            {
+              "metric-type": "SNR-0.1nm",
+              "accumulative-value": 21.0
+            },
+            {
+              "metric-type": "OSNR-bandwidth",
+              "accumulative-value": 16.92
+            },
+            {
+              "metric-type": "OSNR-0.1nm",
+              "accumulative-value": 21.0
+            },
+            {
+              "metric-type": "reference_power",
+              "accumulative-value": 0.001
+            },
+            {
+              "metric-type": "path_bandwidth",
+              "accumulative-value": 100000000000.0
+            }
+          ],
+          "z-a-path-metric": [
+            {
+              "metric-type": "SNR-bandwidth",
+              "accumulative-value": 16.92
+            },
+            {
+              "metric-type": "SNR-0.1nm",
+              "accumulative-value": 21.0
+            },
+            {
+              "metric-type": "OSNR-bandwidth",
+              "accumulative-value": 16.92
+            },
+            {
+              "metric-type": "OSNR-0.1nm",
+              "accumulative-value": 21.0
+            },
+            {
+              "metric-type": "reference_power",
+              "accumulative-value": 0.001
+            },
+            {
+              "metric-type": "path_bandwidth",
+              "accumulative-value": 100000000000.0
+            }
+          ],
+          "path-route-objects": [
+            {
+              "path-route-object": {
+                "index": 0,
+                "num-unnum-hop": {
+                  "node-id": "netconf:10.0.254.93:830",
+                  "link-tp-id": "netconf:10.0.254.93:830",
+                  "gnpy-node-type": "transceiver"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 1,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 2,
+                "transponder": {
+                  "transponder-type": "Cassini",
+                  "transponder-mode": "dp-qpsk"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 3,
+                "num-unnum-hop": {
+                  "node-id": "netconf:10.0.254.107:830",
+                  "link-tp-id": "netconf:10.0.254.107:830",
+                  "gnpy-node-type": "ROADM"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 4,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 5,
+                "target-channel-power": {
+                  "value": -12
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 6,
+                "num-unnum-hop": {
+                  "node-id": "splice-(roadm-Amsterdam-AD)-(patch-(roadm-Amsterdam-AD)-(roadm-Amsterdam-L2))",
+                  "link-tp-id": "splice-(roadm-Amsterdam-AD)-(patch-(roadm-Amsterdam-AD)-(roadm-Amsterdam-L2))"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 7,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 8,
+                "num-unnum-hop": {
+                  "node-id": "patch-(roadm-Amsterdam-AD)-(roadm-Amsterdam-L2)",
+                  "link-tp-id": "patch-(roadm-Amsterdam-AD)-(roadm-Amsterdam-L2)"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 9,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 10,
+                "num-unnum-hop": {
+                  "node-id": "splice-(patch-(roadm-Amsterdam-AD)-(roadm-Amsterdam-L2))-(roadm-Amsterdam-L2)",
+                  "link-tp-id": "splice-(patch-(roadm-Amsterdam-AD)-(roadm-Amsterdam-L2))-(roadm-Amsterdam-L2)"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 11,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 12,
+                "num-unnum-hop": {
+                  "node-id": "netconf:10.0.254.235:830",
+                  "link-tp-id": "netconf:10.0.254.235:830",
+                  "gnpy-node-type": "ROADM"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 13,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 14,
+                "target-channel-power": {
+                  "value": -23
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 15,
+                "num-unnum-hop": {
+                  "node-id": "roadm-Amsterdam-L2-booster",
+                  "link-tp-id": "roadm-Amsterdam-L2-booster",
+                  "gnpy-node-type": "EDFA"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 16,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 17,
+                "target-channel-power": {
+                  "value": -1
+                },
+                "output-voa": {
+                  "value": 12.0
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 18,
+                "num-unnum-hop": {
+                  "node-id": "fiber-Amsterdam-Cologne",
+                  "link-tp-id": "fiber-Amsterdam-Cologne"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 19,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 20,
+                "num-unnum-hop": {
+                  "node-id": "roadm-Cologne-L1-preamp",
+                  "link-tp-id": "roadm-Cologne-L1-preamp",
+                  "gnpy-node-type": "EDFA"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 21,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 22,
+                "target-channel-power": {
+                  "value": 1.0
+                },
+                "output-voa": {
+                  "value": 0.0
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 23,
+                "num-unnum-hop": {
+                  "node-id": "netconf:10.0.254.99:830",
+                  "link-tp-id": "netconf:10.0.254.99:830",
+                  "gnpy-node-type": "ROADM"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 24,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 25,
+                "target-channel-power": {
+                  "value": -12
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 26,
+                "num-unnum-hop": {
+                  "node-id": "splice-(roadm-Cologne-L1)-(patch-(roadm-Cologne-L1)-(roadm-Cologne-L2))",
+                  "link-tp-id": "splice-(roadm-Cologne-L1)-(patch-(roadm-Cologne-L1)-(roadm-Cologne-L2))"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 27,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 28,
+                "num-unnum-hop": {
+                  "node-id": "patch-(roadm-Cologne-L1)-(roadm-Cologne-L2)",
+                  "link-tp-id": "patch-(roadm-Cologne-L1)-(roadm-Cologne-L2)"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 29,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 30,
+                "num-unnum-hop": {
+                  "node-id": "splice-(patch-(roadm-Cologne-L1)-(roadm-Cologne-L2))-(roadm-Cologne-L2)",
+                  "link-tp-id": "splice-(patch-(roadm-Cologne-L1)-(roadm-Cologne-L2))-(roadm-Cologne-L2)"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 31,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 32,
+                "num-unnum-hop": {
+                  "node-id": "netconf:10.0.254.104:830",
+                  "link-tp-id": "netconf:10.0.254.104:830",
+                  "gnpy-node-type": "ROADM"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 33,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 34,
+                "target-channel-power": {
+                  "value": -23
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 35,
+                "num-unnum-hop": {
+                  "node-id": "roadm-Cologne-L2-booster",
+                  "link-tp-id": "roadm-Cologne-L2-booster",
+                  "gnpy-node-type": "EDFA"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 36,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 37,
+                "target-channel-power": {
+                  "value": -1
+                },
+                "output-voa": {
+                  "value": 12.0
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 38,
+                "num-unnum-hop": {
+                  "node-id": "fiber-Cologne-Bremen",
+                  "link-tp-id": "fiber-Cologne-Bremen"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 39,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 40,
+                "num-unnum-hop": {
+                  "node-id": "roadm-Bremen-L1-preamp",
+                  "link-tp-id": "roadm-Bremen-L1-preamp",
+                  "gnpy-node-type": "EDFA"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 41,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 42,
+                "target-channel-power": {
+                  "value": 1.0
+                },
+                "output-voa": {
+                  "value": 0.0
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 43,
+                "num-unnum-hop": {
+                  "node-id": "netconf:10.0.254.100:830",
+                  "link-tp-id": "netconf:10.0.254.100:830",
+                  "gnpy-node-type": "ROADM"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 44,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 45,
+                "target-channel-power": {
+                  "value": -12
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 46,
+                "num-unnum-hop": {
+                  "node-id": "splice-(roadm-Bremen-L1)-(patch-(roadm-Bremen-L1)-(roadm-Bremen-AD))",
+                  "link-tp-id": "splice-(roadm-Bremen-L1)-(patch-(roadm-Bremen-L1)-(roadm-Bremen-AD))"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 47,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 48,
+                "num-unnum-hop": {
+                  "node-id": "patch-(roadm-Bremen-L1)-(roadm-Bremen-AD)",
+                  "link-tp-id": "patch-(roadm-Bremen-L1)-(roadm-Bremen-AD)"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 49,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 50,
+                "num-unnum-hop": {
+                  "node-id": "splice-(patch-(roadm-Bremen-L1)-(roadm-Bremen-AD))-(roadm-Bremen-AD)",
+                  "link-tp-id": "splice-(patch-(roadm-Bremen-L1)-(roadm-Bremen-AD))-(roadm-Bremen-AD)"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 51,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 52,
+                "num-unnum-hop": {
+                  "node-id": "netconf:10.0.254.225:830",
+                  "link-tp-id": "netconf:10.0.254.225:830",
+                  "gnpy-node-type": "ROADM"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 53,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 54,
+                "target-channel-power": {
+                  "value": -25
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 55,
+                "num-unnum-hop": {
+                  "node-id": "netconf:10.0.254.94:830",
+                  "link-tp-id": "netconf:10.0.254.94:830",
+                  "gnpy-node-type": "transceiver"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 56,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 57,
+                "transponder": {
+                  "transponder-type": "Cassini",
+                  "transponder-mode": "dp-qpsk"
+                }
+              }
+            }
+          ],
+          "reversed-path-route-objects": [
+            {
+              "path-route-object": {
+                "index": 0,
+                "num-unnum-hop": {
+                  "node-id": "netconf:10.0.254.94:830",
+                  "link-tp-id": "netconf:10.0.254.94:830",
+                  "gnpy-node-type": "transceiver"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 1,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 2,
+                "transponder": {
+                  "transponder-type": "Cassini",
+                  "transponder-mode": "dp-qpsk"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 3,
+                "num-unnum-hop": {
+                  "node-id": "netconf:10.0.254.225:830",
+                  "link-tp-id": "netconf:10.0.254.225:830",
+                  "gnpy-node-type": "ROADM"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 4,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 5,
+                "target-channel-power": {
+                  "value": -12
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 6,
+                "num-unnum-hop": {
+                  "node-id": "splice-(roadm-Bremen-AD)-(patch-(roadm-Bremen-AD)-(roadm-Bremen-L1))",
+                  "link-tp-id": "splice-(roadm-Bremen-AD)-(patch-(roadm-Bremen-AD)-(roadm-Bremen-L1))"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 7,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 8,
+                "num-unnum-hop": {
+                  "node-id": "patch-(roadm-Bremen-AD)-(roadm-Bremen-L1)",
+                  "link-tp-id": "patch-(roadm-Bremen-AD)-(roadm-Bremen-L1)"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 9,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 10,
+                "num-unnum-hop": {
+                  "node-id": "splice-(patch-(roadm-Bremen-AD)-(roadm-Bremen-L1))-(roadm-Bremen-L1)",
+                  "link-tp-id": "splice-(patch-(roadm-Bremen-AD)-(roadm-Bremen-L1))-(roadm-Bremen-L1)"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 11,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 12,
+                "num-unnum-hop": {
+                  "node-id": "netconf:10.0.254.100:830",
+                  "link-tp-id": "netconf:10.0.254.100:830",
+                  "gnpy-node-type": "ROADM"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 13,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 14,
+                "target-channel-power": {
+                  "value": -23
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 15,
+                "num-unnum-hop": {
+                  "node-id": "roadm-Bremen-L1-booster",
+                  "link-tp-id": "roadm-Bremen-L1-booster",
+                  "gnpy-node-type": "EDFA"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 16,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 17,
+                "target-channel-power": {
+                  "value": -1
+                },
+                "output-voa": {
+                  "value": 12.0
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 18,
+                "num-unnum-hop": {
+                  "node-id": "fiber-Bremen-Cologne",
+                  "link-tp-id": "fiber-Bremen-Cologne"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 19,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 20,
+                "num-unnum-hop": {
+                  "node-id": "roadm-Cologne-L2-preamp",
+                  "link-tp-id": "roadm-Cologne-L2-preamp",
+                  "gnpy-node-type": "EDFA"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 21,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 22,
+                "target-channel-power": {
+                  "value": 1.0
+                },
+                "output-voa": {
+                  "value": 0.0
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 23,
+                "num-unnum-hop": {
+                  "node-id": "netconf:10.0.254.104:830",
+                  "link-tp-id": "netconf:10.0.254.104:830",
+                  "gnpy-node-type": "ROADM"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 24,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 25,
+                "target-channel-power": {
+                  "value": -12
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 26,
+                "num-unnum-hop": {
+                  "node-id": "splice-(roadm-Cologne-L2)-(patch-(roadm-Cologne-L2)-(roadm-Cologne-L1))",
+                  "link-tp-id": "splice-(roadm-Cologne-L2)-(patch-(roadm-Cologne-L2)-(roadm-Cologne-L1))"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 27,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 28,
+                "num-unnum-hop": {
+                  "node-id": "patch-(roadm-Cologne-L2)-(roadm-Cologne-L1)",
+                  "link-tp-id": "patch-(roadm-Cologne-L2)-(roadm-Cologne-L1)"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 29,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 30,
+                "num-unnum-hop": {
+                  "node-id": "splice-(patch-(roadm-Cologne-L2)-(roadm-Cologne-L1))-(roadm-Cologne-L1)",
+                  "link-tp-id": "splice-(patch-(roadm-Cologne-L2)-(roadm-Cologne-L1))-(roadm-Cologne-L1)"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 31,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 32,
+                "num-unnum-hop": {
+                  "node-id": "netconf:10.0.254.99:830",
+                  "link-tp-id": "netconf:10.0.254.99:830",
+                  "gnpy-node-type": "ROADM"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 33,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 34,
+                "target-channel-power": {
+                  "value": -23
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 35,
+                "num-unnum-hop": {
+                  "node-id": "roadm-Cologne-L1-booster",
+                  "link-tp-id": "roadm-Cologne-L1-booster",
+                  "gnpy-node-type": "EDFA"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 36,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 37,
+                "target-channel-power": {
+                  "value": -1
+                },
+                "output-voa": {
+                  "value": 12.0
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 38,
+                "num-unnum-hop": {
+                  "node-id": "fiber-Cologne-Amsterdam",
+                  "link-tp-id": "fiber-Cologne-Amsterdam"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 39,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 40,
+                "num-unnum-hop": {
+                  "node-id": "roadm-Amsterdam-L2-preamp",
+                  "link-tp-id": "roadm-Amsterdam-L2-preamp",
+                  "gnpy-node-type": "EDFA"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 41,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 42,
+                "target-channel-power": {
+                  "value": 1.0
+                },
+                "output-voa": {
+                  "value": 0.0
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 43,
+                "num-unnum-hop": {
+                  "node-id": "netconf:10.0.254.235:830",
+                  "link-tp-id": "netconf:10.0.254.235:830",
+                  "gnpy-node-type": "ROADM"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 44,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 45,
+                "target-channel-power": {
+                  "value": -12
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 46,
+                "num-unnum-hop": {
+                  "node-id": "splice-(roadm-Amsterdam-L2)-(patch-(roadm-Amsterdam-L2)-(roadm-Amsterdam-AD))",
+                  "link-tp-id": "splice-(roadm-Amsterdam-L2)-(patch-(roadm-Amsterdam-L2)-(roadm-Amsterdam-AD))"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 47,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 48,
+                "num-unnum-hop": {
+                  "node-id": "patch-(roadm-Amsterdam-L2)-(roadm-Amsterdam-AD)",
+                  "link-tp-id": "patch-(roadm-Amsterdam-L2)-(roadm-Amsterdam-AD)"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 49,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 50,
+                "num-unnum-hop": {
+                  "node-id": "splice-(patch-(roadm-Amsterdam-L2)-(roadm-Amsterdam-AD))-(roadm-Amsterdam-AD)",
+                  "link-tp-id": "splice-(patch-(roadm-Amsterdam-L2)-(roadm-Amsterdam-AD))-(roadm-Amsterdam-AD)"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 51,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 52,
+                "num-unnum-hop": {
+                  "node-id": "netconf:10.0.254.107:830",
+                  "link-tp-id": "netconf:10.0.254.107:830",
+                  "gnpy-node-type": "ROADM"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 53,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 54,
+                "target-channel-power": {
+                  "value": -25
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 55,
+                "num-unnum-hop": {
+                  "node-id": "netconf:10.0.254.93:830",
+                  "link-tp-id": "netconf:10.0.254.93:830",
+                  "gnpy-node-type": "transceiver"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 56,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 57,
+                "transponder": {
+                  "transponder-type": "Cassini",
+                  "transponder-mode": "dp-qpsk"
+                }
+              }
+            }
+          ]
+        }
+      },
+      {
+        "response-id": "second",
+        "path-properties": {
+          "path-metric": [
+            {
+              "metric-type": "SNR-bandwidth",
+              "accumulative-value": 19.38
+            },
+            {
+              "metric-type": "SNR-0.1nm",
+              "accumulative-value": 23.46
+            },
+            {
+              "metric-type": "OSNR-bandwidth",
+              "accumulative-value": 19.38
+            },
+            {
+              "metric-type": "OSNR-0.1nm",
+              "accumulative-value": 23.47
+            },
+            {
+              "metric-type": "reference_power",
+              "accumulative-value": 0.001
+            },
+            {
+              "metric-type": "path_bandwidth",
+              "accumulative-value": 100000000000.0
+            }
+          ],
+          "z-a-path-metric": [
+            {
+              "metric-type": "SNR-bandwidth",
+              "accumulative-value": 19.38
+            },
+            {
+              "metric-type": "SNR-0.1nm",
+              "accumulative-value": 23.46
+            },
+            {
+              "metric-type": "OSNR-bandwidth",
+              "accumulative-value": 19.38
+            },
+            {
+              "metric-type": "OSNR-0.1nm",
+              "accumulative-value": 23.47
+            },
+            {
+              "metric-type": "reference_power",
+              "accumulative-value": 0.001
+            },
+            {
+              "metric-type": "path_bandwidth",
+              "accumulative-value": 100000000000.0
+            }
+          ],
+          "path-route-objects": [
+            {
+              "path-route-object": {
+                "index": 0,
+                "num-unnum-hop": {
+                  "node-id": "netconf:10.0.254.93:830",
+                  "link-tp-id": "netconf:10.0.254.93:830",
+                  "gnpy-node-type": "transceiver"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 1,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 2,
+                "transponder": {
+                  "transponder-type": "Cassini",
+                  "transponder-mode": "dp-qpsk"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 3,
+                "num-unnum-hop": {
+                  "node-id": "netconf:10.0.254.107:830",
+                  "link-tp-id": "netconf:10.0.254.107:830",
+                  "gnpy-node-type": "ROADM"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 4,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 5,
+                "target-channel-power": {
+                  "value": -12
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 6,
+                "num-unnum-hop": {
+                  "node-id": "splice-(roadm-Amsterdam-AD)-(patch-(roadm-Amsterdam-AD)-(roadm-Amsterdam-L1))",
+                  "link-tp-id": "splice-(roadm-Amsterdam-AD)-(patch-(roadm-Amsterdam-AD)-(roadm-Amsterdam-L1))"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 7,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 8,
+                "num-unnum-hop": {
+                  "node-id": "patch-(roadm-Amsterdam-AD)-(roadm-Amsterdam-L1)",
+                  "link-tp-id": "patch-(roadm-Amsterdam-AD)-(roadm-Amsterdam-L1)"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 9,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 10,
+                "num-unnum-hop": {
+                  "node-id": "splice-(patch-(roadm-Amsterdam-AD)-(roadm-Amsterdam-L1))-(roadm-Amsterdam-L1)",
+                  "link-tp-id": "splice-(patch-(roadm-Amsterdam-AD)-(roadm-Amsterdam-L1))-(roadm-Amsterdam-L1)"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 11,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 12,
+                "num-unnum-hop": {
+                  "node-id": "netconf:10.0.254.101:830",
+                  "link-tp-id": "netconf:10.0.254.101:830",
+                  "gnpy-node-type": "ROADM"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 13,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 14,
+                "target-channel-power": {
+                  "value": -23
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 15,
+                "num-unnum-hop": {
+                  "node-id": "roadm-Amsterdam-L1-booster",
+                  "link-tp-id": "roadm-Amsterdam-L1-booster",
+                  "gnpy-node-type": "EDFA"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 16,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 17,
+                "target-channel-power": {
+                  "value": -1
+                },
+                "output-voa": {
+                  "value": 12.0
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 18,
+                "num-unnum-hop": {
+                  "node-id": "fiber-Amsterdam-Bremen",
+                  "link-tp-id": "fiber-Amsterdam-Bremen"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 19,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 20,
+                "num-unnum-hop": {
+                  "node-id": "roadm-Bremen-L2-preamp",
+                  "link-tp-id": "roadm-Bremen-L2-preamp",
+                  "gnpy-node-type": "EDFA"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 21,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 22,
+                "target-channel-power": {
+                  "value": 1.0
+                },
+                "output-voa": {
+                  "value": 0.0
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 23,
+                "num-unnum-hop": {
+                  "node-id": "netconf:10.0.254.102:830",
+                  "link-tp-id": "netconf:10.0.254.102:830",
+                  "gnpy-node-type": "ROADM"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 24,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 25,
+                "target-channel-power": {
+                  "value": -12
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 26,
+                "num-unnum-hop": {
+                  "node-id": "splice-(roadm-Bremen-L2)-(patch-(roadm-Bremen-L2)-(roadm-Bremen-AD))",
+                  "link-tp-id": "splice-(roadm-Bremen-L2)-(patch-(roadm-Bremen-L2)-(roadm-Bremen-AD))"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 27,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 28,
+                "num-unnum-hop": {
+                  "node-id": "patch-(roadm-Bremen-L2)-(roadm-Bremen-AD)",
+                  "link-tp-id": "patch-(roadm-Bremen-L2)-(roadm-Bremen-AD)"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 29,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 30,
+                "num-unnum-hop": {
+                  "node-id": "splice-(patch-(roadm-Bremen-L2)-(roadm-Bremen-AD))-(roadm-Bremen-AD)",
+                  "link-tp-id": "splice-(patch-(roadm-Bremen-L2)-(roadm-Bremen-AD))-(roadm-Bremen-AD)"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 31,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 32,
+                "num-unnum-hop": {
+                  "node-id": "netconf:10.0.254.225:830",
+                  "link-tp-id": "netconf:10.0.254.225:830",
+                  "gnpy-node-type": "ROADM"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 33,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 34,
+                "target-channel-power": {
+                  "value": -25
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 35,
+                "num-unnum-hop": {
+                  "node-id": "netconf:10.0.254.94:830",
+                  "link-tp-id": "netconf:10.0.254.94:830",
+                  "gnpy-node-type": "transceiver"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 36,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 37,
+                "transponder": {
+                  "transponder-type": "Cassini",
+                  "transponder-mode": "dp-qpsk"
+                }
+              }
+            }
+          ],
+          "reversed-path-route-objects": [
+            {
+              "path-route-object": {
+                "index": 0,
+                "num-unnum-hop": {
+                  "node-id": "netconf:10.0.254.94:830",
+                  "link-tp-id": "netconf:10.0.254.94:830",
+                  "gnpy-node-type": "transceiver"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 1,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 2,
+                "transponder": {
+                  "transponder-type": "Cassini",
+                  "transponder-mode": "dp-qpsk"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 3,
+                "num-unnum-hop": {
+                  "node-id": "netconf:10.0.254.225:830",
+                  "link-tp-id": "netconf:10.0.254.225:830",
+                  "gnpy-node-type": "ROADM"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 4,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 5,
+                "target-channel-power": {
+                  "value": -12
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 6,
+                "num-unnum-hop": {
+                  "node-id": "splice-(roadm-Bremen-AD)-(patch-(roadm-Bremen-AD)-(roadm-Bremen-L2))",
+                  "link-tp-id": "splice-(roadm-Bremen-AD)-(patch-(roadm-Bremen-AD)-(roadm-Bremen-L2))"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 7,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 8,
+                "num-unnum-hop": {
+                  "node-id": "patch-(roadm-Bremen-AD)-(roadm-Bremen-L2)",
+                  "link-tp-id": "patch-(roadm-Bremen-AD)-(roadm-Bremen-L2)"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 9,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 10,
+                "num-unnum-hop": {
+                  "node-id": "splice-(patch-(roadm-Bremen-AD)-(roadm-Bremen-L2))-(roadm-Bremen-L2)",
+                  "link-tp-id": "splice-(patch-(roadm-Bremen-AD)-(roadm-Bremen-L2))-(roadm-Bremen-L2)"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 11,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 12,
+                "num-unnum-hop": {
+                  "node-id": "netconf:10.0.254.102:830",
+                  "link-tp-id": "netconf:10.0.254.102:830",
+                  "gnpy-node-type": "ROADM"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 13,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 14,
+                "target-channel-power": {
+                  "value": -23
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 15,
+                "num-unnum-hop": {
+                  "node-id": "roadm-Bremen-L2-booster",
+                  "link-tp-id": "roadm-Bremen-L2-booster",
+                  "gnpy-node-type": "EDFA"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 16,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 17,
+                "target-channel-power": {
+                  "value": -1
+                },
+                "output-voa": {
+                  "value": 12.0
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 18,
+                "num-unnum-hop": {
+                  "node-id": "fiber-Bremen-Amsterdam",
+                  "link-tp-id": "fiber-Bremen-Amsterdam"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 19,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 20,
+                "num-unnum-hop": {
+                  "node-id": "roadm-Amsterdam-L1-preamp",
+                  "link-tp-id": "roadm-Amsterdam-L1-preamp",
+                  "gnpy-node-type": "EDFA"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 21,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 22,
+                "target-channel-power": {
+                  "value": 1.0
+                },
+                "output-voa": {
+                  "value": 0.0
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 23,
+                "num-unnum-hop": {
+                  "node-id": "netconf:10.0.254.101:830",
+                  "link-tp-id": "netconf:10.0.254.101:830",
+                  "gnpy-node-type": "ROADM"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 24,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 25,
+                "target-channel-power": {
+                  "value": -12
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 26,
+                "num-unnum-hop": {
+                  "node-id": "splice-(roadm-Amsterdam-L1)-(patch-(roadm-Amsterdam-L1)-(roadm-Amsterdam-AD))",
+                  "link-tp-id": "splice-(roadm-Amsterdam-L1)-(patch-(roadm-Amsterdam-L1)-(roadm-Amsterdam-AD))"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 27,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 28,
+                "num-unnum-hop": {
+                  "node-id": "patch-(roadm-Amsterdam-L1)-(roadm-Amsterdam-AD)",
+                  "link-tp-id": "patch-(roadm-Amsterdam-L1)-(roadm-Amsterdam-AD)"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 29,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 30,
+                "num-unnum-hop": {
+                  "node-id": "splice-(patch-(roadm-Amsterdam-L1)-(roadm-Amsterdam-AD))-(roadm-Amsterdam-AD)",
+                  "link-tp-id": "splice-(patch-(roadm-Amsterdam-L1)-(roadm-Amsterdam-AD))-(roadm-Amsterdam-AD)"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 31,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 32,
+                "num-unnum-hop": {
+                  "node-id": "netconf:10.0.254.107:830",
+                  "link-tp-id": "netconf:10.0.254.107:830",
+                  "gnpy-node-type": "ROADM"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 33,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 34,
+                "target-channel-power": {
+                  "value": -25
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 35,
+                "num-unnum-hop": {
+                  "node-id": "netconf:10.0.254.93:830",
+                  "link-tp-id": "netconf:10.0.254.93:830",
+                  "gnpy-node-type": "transceiver"
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 36,
+                "label-hop": {
+                  "N": -284,
+                  "M": 4
+                }
+              }
+            },
+            {
+              "path-route-object": {
+                "index": 37,
+                "transponder": {
+                  "transponder-type": "Cassini",
+                  "transponder-mode": "dp-qpsk"
+                }
+              }
+            }
+          ]
+        }
+      }
+    ]
+  }
+}