org.geppetto.frontend.controllers.ConnectionHandler.java Source code

Java tutorial

Introduction

Here is the source code for org.geppetto.frontend.controllers.ConnectionHandler.java

Source

/*******************************************************************************
 * The MIT License (MIT)
 * 
 * Copyright (c) 2011 - 2015 OpenWorm.
 * http://openworm.org
 * 
 * All rights reserved. This program and the accompanying materials
 * are made available under the terms of the MIT License
 * which accompanies this distribution, and is available at
 * http://opensource.org/licenses/MIT
 *
 * Contributors:
 *        OpenWorm - http://openworm.org/people.html
 *
 * Permission is hereby granted, free of charge, to any person obtaining a copy
 * of this software and associated documentation files (the "Software"), to deal
 * in the Software without restriction, including without limitation the rights 
 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 
 * copies of the Software, and to permit persons to whom the Software is 
 * furnished to do so, subject to the following conditions:
 * 
 * The above copyright notice and this permission notice shall be included in 
 * all copies or substantial portions of the Software.
 * 
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. 
 * IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, 
 * DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR 
 * OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE 
 * USE OR OTHER DEALINGS IN THE SOFTWARE.
 *******************************************************************************/
package org.geppetto.frontend.controllers;

import java.io.BufferedReader;
import java.io.File;
import java.io.IOException;
import java.io.InputStreamReader;
import java.lang.reflect.Type;
import java.net.MalformedURLException;
import java.net.URL;
import java.nio.file.Path;
import java.util.Date;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Properties;

import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.geppetto.core.beans.PathConfiguration;
import org.geppetto.core.common.GeppettoErrorCodes;
import org.geppetto.core.common.GeppettoExecutionException;
import org.geppetto.core.common.GeppettoInitializationException;
import org.geppetto.core.data.DataManagerHelper;
import org.geppetto.core.data.IGeppettoDataManager;
import org.geppetto.core.data.model.IAspectConfiguration;
import org.geppetto.core.data.model.IExperiment;
import org.geppetto.core.data.model.IGeppettoProject;
import org.geppetto.core.data.model.ResultsFormat;
import org.geppetto.core.manager.IGeppettoManager;
import org.geppetto.core.manager.Scope;
import org.geppetto.core.model.runtime.AspectSubTreeNode;
import org.geppetto.core.model.runtime.RuntimeTreeRoot;
import org.geppetto.core.model.state.visitors.SerializeTreeVisitor;
import org.geppetto.core.services.ModelFormat;
import org.geppetto.core.services.registry.ServicesRegistry;
import org.geppetto.core.utilities.URLReader;
import org.geppetto.core.utilities.Zipper;
import org.geppetto.frontend.Resources;
import org.geppetto.frontend.messages.OutboundMessages;
import org.springframework.beans.factory.annotation.Autowired;

import com.fasterxml.jackson.core.JsonProcessingException;
import com.fasterxml.jackson.core.type.TypeReference;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.google.gson.Gson;
import com.google.gson.GsonBuilder;
import com.google.gson.JsonDeserializationContext;
import com.google.gson.JsonDeserializer;
import com.google.gson.JsonElement;
import com.google.gson.JsonParseException;

/**
 * Class that handles the Web Socket connections the servlet is receiving. FIXME: REMOVE ALL THE MANUAL CONSTRUCTION OF JSON STRINGS, USE GSON INSTEAD
 *
 * @author Jesus R. Martinez (jesus@metacell.us)
 * @author matteocantarelli
 * 
 */
public class ConnectionHandler {

    private static Log logger = LogFactory.getLog(ConnectionHandler.class);

    @Autowired
    private SimulationServerConfig simulationServerConfig;

    private WebsocketConnection websocketConnection;

    private IGeppettoManager geppettoManager;

    // the geppetto project active for this connection
    private IGeppettoProject geppettoProject;

    /**
     * @param websocketConnection
     * @param geppettoManager
     */
    protected ConnectionHandler(WebsocketConnection websocketConnection, IGeppettoManager geppettoManager) {
        this.websocketConnection = websocketConnection;
        // FIXME This is extremely ugly, a session based geppetto manager is autowired in the websocketconnection
        // but a session bean cannot travel outside a conenction thread so a new one is instantiated and initialised
        this.geppettoManager = new GeppettoManager(geppettoManager);

    }

    /**
     * @param requestID
     * @param projectId
     */
    public void loadProjectFromId(String requestID, long projectId, long experimentId) {
        IGeppettoDataManager dataManager = DataManagerHelper.getDataManager();
        try {
            IGeppettoProject geppettoProject = dataManager.getGeppettoProjectById(projectId);
            if (geppettoProject == null) {
                websocketConnection.sendMessage(requestID, OutboundMessages.ERROR_LOADING_PROJECT,
                        "Project not found");
            } else {
                loadGeppettoProject(requestID, geppettoProject, experimentId);
            }
        } catch (NumberFormatException e) {
            websocketConnection.sendMessage(requestID, OutboundMessages.ERROR_LOADING_PROJECT, "");
        }
    }

    /**
     * @param requestID
     * @param projectContent
     */
    public void loadProjectFromContent(String requestID, String projectContent) {
        IGeppettoProject geppettoProject = DataManagerHelper.getDataManager().getProjectFromJson(getGson(),
                projectContent);
        loadGeppettoProject(requestID, geppettoProject, -1l);
    }

    /**
     * @param requestID
     * @param urlString
     */
    public void loadProjectFromURL(String requestID, String urlString) {
        URL url;
        try {
            url = URLReader.getURL(urlString);
            BufferedReader reader = new BufferedReader(new InputStreamReader(url.openStream()));
            IGeppettoProject geppettoProject = DataManagerHelper.getDataManager().getProjectFromJson(getGson(),
                    reader);
            loadGeppettoProject(requestID, geppettoProject, -1l);
        } catch (IOException e) {
            error(e, "Could not load geppetto project");
        }
    }

    /**
     * @param requestID
     * @param geppettoProject
     */
    public void loadGeppettoProject(String requestID, IGeppettoProject geppettoProject, long experimentId) {
        try {
            geppettoManager.loadProject(requestID, geppettoProject);
            // serialize project prior to sending it to client
            Gson gson = new Gson();
            String json = gson.toJson(geppettoProject);
            setConnectionProject(geppettoProject);
            websocketConnection.sendMessage(requestID, OutboundMessages.PROJECT_LOADED, json);

            if (experimentId != -1) {
                loadExperiment(requestID, experimentId, geppettoProject.getId());
            } else if (geppettoProject.getActiveExperimentId() != -1) {
                loadExperiment(requestID, geppettoProject.getActiveExperimentId(), geppettoProject.getId());
            }

        } catch (MalformedURLException | GeppettoInitializationException | GeppettoExecutionException e) {
            error(e, "Could not load geppetto project");
        }

    }

    /**
     * @param projectId
     */
    public void newExperiment(String requestID, long projectId) {
        if (DataManagerHelper.getDataManager().isDefault()) {
            info(requestID, Resources.UNSUPPORTED_OPERATION.toString());
        } else {
            IGeppettoProject project = retrieveGeppettoProject(projectId);

            try {
                IExperiment experiment = geppettoManager.newExperiment(requestID, project);
                Gson gson = new Gson();
                String json = gson.toJson(experiment);
                websocketConnection.sendMessage(requestID, OutboundMessages.EXPERIMENT_CREATED, json);

            } catch (GeppettoExecutionException e) {
                error(e, "Error creating a new experiment");
            }
        }
    }

    /**
     * @param requestID
     * @param experimentID
     * @param projectId
     */
    public void loadExperiment(String requestID, long experimentID, long projectId) {
        websocketConnection.sendMessage(requestID, OutboundMessages.EXPERIMENT_LOADING, "");
        try {
            IGeppettoProject geppettoProject = retrieveGeppettoProject(projectId);
            IExperiment experiment = retrieveExperiment(experimentID, geppettoProject);
            // run the matched experiment
            if (experiment != null) {
                RuntimeTreeRoot runtimeTree = geppettoManager.loadExperiment(requestID, experiment);

                SerializeTreeVisitor serializeTreeVisitor = new SerializeTreeVisitor();
                runtimeTree.apply(serializeTreeVisitor);
                String scene = serializeTreeVisitor.getSerializedTree();
                // TODO This is ugly, the tree visitor should not have "scene" inside, also it should be called runtimetree
                String message = "{\"experimentId\":" + experimentID + "," + scene.substring(1);
                websocketConnection.sendMessage(requestID, OutboundMessages.EXPERIMENT_LOADED, message);
                logger.info("The experiment " + experimentID
                        + " was loaded and the runtime tree was sent to the client");

            } else {
                error(null, "Error loading experiment, the experiment " + experimentID
                        + " was not found in project " + projectId);
            }

        } catch (GeppettoExecutionException e) {
            error(e, "Error loading experiment");
        }
    }

    /**
     * Run the Experiment
     */
    public void runExperiment(String requestID, long experimentID, long projectId) {
        if (DataManagerHelper.getDataManager().isDefault()) {
            info(requestID, Resources.UNSUPPORTED_OPERATION.toString());
        }
        IGeppettoProject geppettoProject = retrieveGeppettoProject(projectId);
        IExperiment experiment = retrieveExperiment(experimentID, geppettoProject);
        if (geppettoProject.isVolatile()) {
            info(requestID, Resources.VOLATILE_PROJECT.toString());
            return;
        } else {
            try {
                // run the matched experiment
                if (experiment != null) {
                    geppettoManager.runExperiment(requestID, experiment);
                } else {
                    error(null, "Error running experiment, the experiment " + experimentID
                            + " was not found in project " + projectId);
                }

            } catch (GeppettoExecutionException e) {
                error(e, "Error running experiment");
            }
        }
    }

    /**
     * Adds watch lists with variables to be watched
     * 
     * @param requestID
     * @param jsonLists
     * @throws GeppettoExecutionException
     * @throws GeppettoInitializationException
     */
    public void setWatchedVariables(String requestID, List<String> variables, long experimentID, long projectId)
            throws GeppettoExecutionException, GeppettoInitializationException {
        if (DataManagerHelper.getDataManager().isDefault()) {
            info(requestID, Resources.UNSUPPORTED_OPERATION.toString());
        } else {
            IGeppettoProject geppettoProject = retrieveGeppettoProject(projectId);
            IExperiment experiment = retrieveExperiment(experimentID, geppettoProject);

            geppettoManager.setWatchedVariables(variables, experiment, geppettoProject);

            // serialize watch-lists
            ObjectMapper mapper = new ObjectMapper();
            String serializedLists;
            try {
                serializedLists = mapper.writer().writeValueAsString(variables);

                // send to the client the watch lists were added
                websocketConnection.sendMessage(requestID, OutboundMessages.WATCHED_VARIABLES_SET, serializedLists);
            } catch (JsonProcessingException e) {
                error(e, "There was an error serializing the watched lists");
            }
        }
    }

    /**
     * @param requestID
     * @param experimentID
     * @param projectId
     * @throws GeppettoExecutionException
     */
    public void clearWatchLists(String requestID, long experimentID, long projectId)
            throws GeppettoExecutionException {
        IGeppettoProject geppettoProject = retrieveGeppettoProject(projectId);
        IExperiment experiment = retrieveExperiment(experimentID, geppettoProject);
        geppettoManager.clearWatchLists(experiment, geppettoProject);
        websocketConnection.sendMessage(requestID, OutboundMessages.CLEAR_WATCH, "");
    }

    /**
     * @param requestID
     */
    public void getVersionNumber(String requestID) {
        Properties prop = new Properties();
        try {
            prop.load(ConnectionHandler.class.getResourceAsStream("/Geppetto.properties"));
            websocketConnection.sendMessage(requestID, OutboundMessages.GEPPETTO_VERSION,
                    prop.getProperty("Geppetto.version"));
        } catch (IOException e) {
            error(e, "Unable to read GEPPETTO.properties file");
        }
    }

    /**
     * @param requestID
     * @param experimentId
     * @param projectId
     */
    public void playExperiment(String requestID, long experimentId, long projectId) {
        IGeppettoProject geppettoProject = retrieveGeppettoProject(projectId);
        IExperiment experiment = retrieveExperiment(experimentId, geppettoProject);

        if (experiment != null) {
            Map<String, AspectSubTreeNode> simulationTree;
            try {
                simulationTree = geppettoManager.playExperiment(requestID, experiment);

                String simulationTreeString = "[";
                for (Map.Entry<String, AspectSubTreeNode> entry : simulationTree.entrySet()) {
                    SerializeTreeVisitor updateClientVisitor = new SerializeTreeVisitor();
                    entry.getValue().apply(updateClientVisitor);
                    String simTree = updateClientVisitor.getSerializedTree();
                    // remove first and last bracket of sim tree before adding to string
                    // that'll be send to client
                    String formattedTree = simTree.substring(1, simTree.length() - 1);
                    simulationTreeString += "{\"aspectInstancePath\":" + '"' + entry.getKey() + '"' + ","
                            + formattedTree + "},";
                }
                simulationTreeString = simulationTreeString.substring(0, simulationTreeString.length() - 1);
                simulationTreeString += "]";

                websocketConnection.sendMessage(requestID, OutboundMessages.PLAY_EXPERIMENT, simulationTreeString);
            } catch (GeppettoExecutionException e) {
                error(e, "Error playing the experiment " + experimentId);
            }
        } else {
            error(null, "Error playing experiment, the experiment " + experimentId + " was not found in project "
                    + projectId);
        }
    }

    /**
     * @param requestID
     * @param aspectInstancePath
     */
    public void getModelTree(String requestID, String aspectInstancePath, long experimentID, long projectId) {
        IGeppettoProject geppettoProject = retrieveGeppettoProject(projectId);
        IExperiment experiment = retrieveExperiment(experimentID, geppettoProject);
        Map<String, AspectSubTreeNode> modelTree;
        try {
            modelTree = geppettoManager.getModelTree(aspectInstancePath, experiment, geppettoProject);

            String modelTreeString = "[";
            for (Map.Entry<String, AspectSubTreeNode> entry : modelTree.entrySet()) {
                SerializeTreeVisitor updateClientVisitor = new SerializeTreeVisitor();
                entry.getValue().apply(updateClientVisitor);
                String simTree = updateClientVisitor.getSerializedTree();
                // remove first and last bracket of sim tree before adding to string
                // that'll be send to client
                String formattedTree = simTree.substring(1, simTree.length() - 1);
                modelTreeString += "{\"aspectInstancePath\":" + '"' + entry.getKey() + '"' + "," + formattedTree
                        + "},";

                // reset flags
                ModelTreeExitVisitor exitVisitor = new ModelTreeExitVisitor();
                entry.getValue().apply(exitVisitor);
            }
            modelTreeString = modelTreeString.substring(0, modelTreeString.length() - 1);
            modelTreeString += "]";

            websocketConnection.sendMessage(requestID, OutboundMessages.GET_MODEL_TREE, modelTreeString);
        } catch (GeppettoExecutionException e) {
            error(e, "Error populating the model tree for " + aspectInstancePath);
        }
    }

    /**
     * @param requestID
     * @param aspectInstancePath
     */
    public void getSimulationTree(String requestID, String aspectInstancePath, long experimentID, long projectId) {
        IGeppettoProject geppettoProject = retrieveGeppettoProject(projectId);
        IExperiment experiment = retrieveExperiment(experimentID, geppettoProject);
        Map<String, AspectSubTreeNode> simulationTree;
        try {
            simulationTree = geppettoManager.getSimulationTree(aspectInstancePath, experiment, geppettoProject);

            String simulationTreeString = "[";
            for (Map.Entry<String, AspectSubTreeNode> entry : simulationTree.entrySet()) {
                SerializeTreeVisitor updateClientVisitor = new SerializeTreeVisitor();
                entry.getValue().apply(updateClientVisitor);
                String simTree = updateClientVisitor.getSerializedTree();
                // remove first and last bracket of sim tree before adding to string
                // that'll be send to client
                String formattedTree = simTree.substring(1, simTree.length() - 1);
                simulationTreeString += "{\"aspectInstancePath\":" + '"' + entry.getKey() + '"' + ","
                        + formattedTree + "},";
            }
            simulationTreeString = simulationTreeString.substring(0, simulationTreeString.length() - 1);
            simulationTreeString += "]";

            websocketConnection.sendMessage(requestID, OutboundMessages.GET_SIMULATION_TREE, simulationTreeString);
        } catch (GeppettoExecutionException e) {
            error(e, "Error populating the simulation tree for " + aspectInstancePath);
        }

    }

    /**
     * @param requestID
     * @param aspectInstancePath
     * @param format
     * 
     */
    public void downloadModel(String requestID, String aspectInstancePath, String format, long experimentID,
            long projectId) {
        IGeppettoProject geppettoProject = retrieveGeppettoProject(projectId);
        IExperiment experiment = retrieveExperiment(experimentID, geppettoProject);

        ModelFormat modelFormat = ServicesRegistry.getModelFormat(format);
        try {

            if (modelFormat == null && format != null) {
                // FIXME There is a method called ERROR for sending errors to the GUI, also the error code and the outbound message are different
                // things, there's no need to have a separate message for each error
                websocketConnection.sendMessage(requestID, OutboundMessages.ERROR_DOWNLOADING_MODEL, "");
            } else {
                // Convert model
                File file = geppettoManager.downloadModel(aspectInstancePath, modelFormat, experiment,
                        geppettoProject);

                // Zip folder
                Zipper zipper = new Zipper(PathConfiguration.createExperimentTmpPath(Scope.CONNECTION, projectId,
                        experimentID, aspectInstancePath, file.getName() + ".zip"));
                Path path = zipper.getZipFromDirectory(file);

                // Send zip file to the client
                websocketConnection.sendBinaryMessage(requestID, path);
                websocketConnection.sendMessage(requestID, OutboundMessages.DOWNLOAD_MODEL, "");
            }
        } catch (GeppettoExecutionException | IOException e) {
            error(e, "Error downloading model for " + aspectInstancePath + " in format " + format);
        }
    }

    /**
     * @param requestID
     * @param aspectInstancePath
     * 
     */
    public void getSupportedOuputs(String requestID, String aspectInstancePath, long experimentID, long projectId) {
        IGeppettoProject geppettoProject = retrieveGeppettoProject(projectId);
        IExperiment experiment = retrieveExperiment(experimentID, geppettoProject);
        try {
            List<ModelFormat> supportedOutputs = geppettoManager.getSupportedOuputs(aspectInstancePath, experiment,
                    geppettoProject);

            String supportedOutputsString = "[";
            for (ModelFormat supportedOutput : supportedOutputs) {
                supportedOutputsString += "\"" + supportedOutput.getModelFormat() + "\",";
            }
            supportedOutputsString = supportedOutputsString.substring(0, supportedOutputsString.length() - 1);
            supportedOutputsString += "]";

            websocketConnection.sendMessage(requestID, OutboundMessages.GET_SUPPORTED_OUTPUTS,
                    supportedOutputsString);
        } catch (GeppettoExecutionException e) {
            error(e, "Error getting supported outputs for " + aspectInstancePath);
        }
    }

    /**
     * @param requestID
     * @param url
     * @param visitor
     */
    public void sendScriptData(String requestID, URL url, WebsocketConnection visitor) {
        try {
            String line = null;
            StringBuilder sb = new StringBuilder();

            BufferedReader br = new BufferedReader(new InputStreamReader(url.openStream()));

            while ((line = br.readLine()) != null) {
                sb.append(line + "\n");
            }
            String script = sb.toString();

            websocketConnection.sendMessage(requestID, OutboundMessages.SCRIPT_FETCHED, script);
        } catch (IOException e) {
            error(e, "Error while reading the script at " + url);
        }
    }

    public void userBecameIdle(String requestID) {
        closeProject();
    }

    /**
     * @param requestID
     * @param modelPath
     * @param modelParameters
     * @param projectId
     * @param experimentID
     */
    public void setParameters(String requestID, String modelPath, Map<String, String> modelParameters,
            long projectId, long experimentID) {
        if (DataManagerHelper.getDataManager().isDefault()) {
            info(requestID, Resources.UNSUPPORTED_OPERATION.toString());
            return;
        }
        IGeppettoProject geppettoProject = retrieveGeppettoProject(projectId);
        IExperiment experiment = retrieveExperiment(experimentID, geppettoProject);
        if (geppettoProject.isVolatile()) {
            info(requestID, Resources.VOLATILE_PROJECT.toString());
            return;
        } else {

            try {
                AspectSubTreeNode modelTreeNode = geppettoManager.setModelParameters(modelPath, modelParameters,
                        experiment, geppettoProject);
                String modelTreeString = "[";
                SerializeTreeVisitor updateClientVisitor = new SerializeTreeVisitor();
                modelTreeNode.apply(updateClientVisitor);
                String simTree = updateClientVisitor.getSerializedTree();
                String formattedTree = simTree.substring(1, simTree.length());

                modelTreeString += "{\"aspectInstancePath\":" + '"' + modelPath + '"' + "," + formattedTree + "}";

                modelTreeString = modelTreeString.substring(0, modelTreeString.length() - 1);
                modelTreeString += "]";

                // reset flags
                ModelTreeExitVisitor exitVisitor = new ModelTreeExitVisitor();
                modelTreeNode.apply(exitVisitor);

                websocketConnection.sendMessage(requestID, OutboundMessages.UPDATE_MODEL_TREE, modelTreeString);
            } catch (GeppettoExecutionException e) {
                error(e, "There was an error setting parameters");
            }
        }
    }

    /**
     * @param experimentID
     * @param geppettoProject
     * @return
     */
    private IExperiment retrieveExperiment(long experimentID, IGeppettoProject geppettoProject) {
        IExperiment theExperiment = null;
        // Look for experiment that matches id passed
        for (IExperiment e : geppettoProject.getExperiments()) {
            if (e.getId() == experimentID) {
                // The experiment is found
                theExperiment = e;
                break;
            }
        }
        return theExperiment;
    }

    /**
     * @param projectId
     * @return
     */
    private IGeppettoProject retrieveGeppettoProject(long projectId) {
        IGeppettoDataManager dataManager = DataManagerHelper.getDataManager();
        return dataManager.getGeppettoProjectById(projectId);
    }

    /**
     * @param type
     * @param jsonPacket
     * @return
     * @throws GeppettoExecutionException
     */
    public <T> T fromJSON(final TypeReference<T> type, String jsonPacket) throws GeppettoExecutionException {
        T data = null;

        try {
            data = new ObjectMapper().readValue(jsonPacket, type);
        } catch (IOException e) {
            error(e, "Error deserializing the JSON document");
        }

        return data;
    }

    /**
     * @return
     */
    private Gson getGson() {
        GsonBuilder builder = new GsonBuilder();
        builder.registerTypeAdapter(Date.class, new JsonDeserializer<Date>() {
            @Override
            public Date deserialize(JsonElement json, Type typeOfT, JsonDeserializationContext context)
                    throws JsonParseException {
                return new Date(json.getAsJsonPrimitive().getAsLong());
            }
        });
        return builder.create();
    }

    /**
     * @return
     */
    public SimulationServerConfig getSimulationServerConfig() {
        return simulationServerConfig;
    }

    /**
     * @param exception
     * @param errorMessage
     */
    private void error(Exception exception, String errorMessage) {
        String exceptionMessage = "";
        if (exception != null) {
            exceptionMessage = exception.getCause() == null ? exception.getMessage() : exception.toString();
        }
        Error error = new Error(GeppettoErrorCodes.EXCEPTION, errorMessage, exceptionMessage);
        logger.error(errorMessage, exception);
        websocketConnection.sendMessage(null, OutboundMessages.ERROR, getGson().toJson(error));

    }

    /**
     * @param requestID 
     * @param exception
     * @param errorMessage
     */
    private void info(String requestID, String message) {
        logger.info(message);
        websocketConnection.sendMessage(requestID, OutboundMessages.INFO_MESSAGE, getGson().toJson(message));

    }

    private class Error {
        public Error(GeppettoErrorCodes errorCode, String errorMessage, String jsonExceptionMsg) {
            this.error_code = errorCode.toString();
            message = errorMessage;
            exception = jsonExceptionMsg;
        }

        String error_code;
        String message;
        String exception;
    }

    /**
     * @param requestID
     * @param projectId
     */
    public void checkExperimentStatus(String requestID, String projectId) {
        IGeppettoDataManager dataManager = DataManagerHelper.getDataManager();
        try {
            IGeppettoProject geppettoProject = dataManager.getGeppettoProjectById(Long.parseLong(projectId));
            if (geppettoProject != null) {
                List<? extends IExperiment> experiments = geppettoManager.checkExperimentsStatus(requestID,
                        geppettoProject);
                String status = "[";
                for (IExperiment e : experiments) {
                    // FIXME
                    status += "{\"projectID\":" + '"' + projectId + '"' + ",\"experimentID\":" + '"' + e.getId()
                            + '"' + ",\"status\":" + '"' + e.getStatus().toString() + '"' + "},";

                }
                status = status.substring(0, status.length() - 1);
                status += "]";
                websocketConnection.sendMessage(requestID, OutboundMessages.EXPERIMENT_STATUS, status);
            } else {
                String msg = "Check Experiment: Cannot find project " + projectId;
                error(new GeppettoExecutionException(msg), msg);
            }
        } catch (NumberFormatException e) {
            error(e, "Check Experiment: Errror parsing project id");
        }
    }

    /**
     * @param requestID
     * @param experimentId
     * @param projectId
     */
    public void deleteExperiment(String requestID, long experimentId, long projectId) {
        if (DataManagerHelper.getDataManager().isDefault()) {
            info(requestID, Resources.UNSUPPORTED_OPERATION.toString());
        } else {
            IGeppettoProject geppettoProject = retrieveGeppettoProject(projectId);
            IExperiment experiment = retrieveExperiment(experimentId, geppettoProject);

            if (experiment != null) {
                try {
                    geppettoManager.deleteExperiment(requestID, experiment);
                } catch (GeppettoExecutionException e) {
                    error(e, "Error while deleting the experiment");
                }
                String update = "{\"id\":" + '"' + experiment.getId() + '"' + ",\"name\":" + '"'
                        + experiment.getName() + '"' + "}";
                websocketConnection.sendMessage(requestID, OutboundMessages.DELETE_EXPERIMENT, update);
            } else {
                error(null, "Error deleting experiment, the experiment " + experimentId
                        + " was not found in project " + projectId);
            }
        }
    }

    /**
     * @param requestID
     * @param projectId
     */
    public void persistProject(String requestID, long projectId) {
        if (DataManagerHelper.getDataManager().isDefault()) {
            info(requestID, Resources.UNSUPPORTED_OPERATION.toString());
        } else {
            try {
                IGeppettoProject geppettoProject = retrieveGeppettoProject(projectId);

                if (geppettoProject != null) {

                    geppettoManager.persistProject(requestID, geppettoProject);
                    PersistedProject persistedProject = new PersistedProject(geppettoProject.getId(),
                            geppettoProject.getActiveExperimentId());
                    websocketConnection.sendMessage(requestID, OutboundMessages.PROJECT_PERSISTED,
                            getGson().toJson(persistedProject));
                } else {
                    error(null, "Error persisting project  " + projectId + ".");
                }
            } catch (GeppettoExecutionException e) {
                error(e, "Error persisting project");
            }
        }
    }

    class PersistedProject {
        long projectID;
        long activeExperimentID;

        public PersistedProject(long projectID, long activeExperimentID) {
            super();
            this.projectID = projectID;
            this.activeExperimentID = activeExperimentID;
        }

    }

    /**
     * @param requestID
     * @param key
     */
    public void linkDropBox(String requestID, String key) {
        try {
            geppettoManager.linkDropBoxAccount(key);
            websocketConnection.sendMessage(requestID, OutboundMessages.DROPBOX_LINKED, null);
        } catch (Exception e) {
            error(e, "Unable to link dropbox account.");
        }
    }

    /**
     * @param requestID
     * @param key
     */
    public void unLinkDropBox(String requestID, String key) {
        try {
            geppettoManager.unlinkDropBoxAccount(key);
            websocketConnection.sendMessage(null, OutboundMessages.DROPBOX_UNLINKED, null);
        } catch (Exception e) {
            error(e, "Unable to unlink dropbox account.");
        }
    }

    /**
     * @param aspectPath
     * @param projectId
     * @param experimentId
     * @param format
     */
    public void uploadModel(String aspectPath, long projectId, long experimentId, String format) {
        IGeppettoProject geppettoProject = retrieveGeppettoProject(projectId);
        IExperiment experiment = retrieveExperiment(experimentId, geppettoProject);
        ModelFormat modelFormat = ServicesRegistry.getModelFormat(format);
        try {
            geppettoManager.uploadModelToDropBox(aspectPath, experiment, geppettoProject, modelFormat);
            websocketConnection.sendMessage(null, OutboundMessages.MODEL_UPLOADED, null);
        } catch (Exception e) {
            error(e, "Unable to upload results for aspect : " + aspectPath);
        }
    }

    /**
     * @param aspectPath
     * @param projectId
     * @param experimentId
     * @param format
     */
    public void uploadResults(String aspectPath, long projectId, long experimentId, String format) {
        IGeppettoProject geppettoProject = retrieveGeppettoProject(projectId);
        IExperiment experiment = retrieveExperiment(experimentId, geppettoProject);
        ResultsFormat resultsFormat = ServicesRegistry.getResultsFormat(format);
        try {
            geppettoManager.uploadResultsToDropBox(aspectPath, experiment, geppettoProject, resultsFormat);
            websocketConnection.sendMessage(null, OutboundMessages.RESULTS_UPLOADED, null);
        } catch (GeppettoExecutionException e) {
            error(e, "Unable to upload results for aspect : " + aspectPath);
        }
    }

    /**
     * @param requestID
     * @param aspectPath
     * @param projectId
     * @param experimentId
     * @param format
     */
    public void downloadResults(String requestID, String aspectPath, long projectId, long experimentId,
            String format) {
        IGeppettoProject geppettoProject = retrieveGeppettoProject(projectId);
        IExperiment experiment = retrieveExperiment(experimentId, geppettoProject);
        ResultsFormat resultsFormat = ServicesRegistry.getResultsFormat(format);
        try {
            if (resultsFormat == null) {
                websocketConnection.sendMessage(requestID, OutboundMessages.ERROR_DOWNLOADING_RESULTS, "");
            } else {
                // Convert model
                URL url = geppettoManager.downloadResults(aspectPath, resultsFormat, experiment, geppettoProject);

                if (url != null) {
                    // Zip folder
                    Zipper zipper = new Zipper(PathConfiguration.createExperimentTmpPath(Scope.CONNECTION,
                            projectId, experimentId, aspectPath, URLReader.getFileName(url)));
                    Path path = zipper.getZipFromFile(url);

                    // Send zip file to the client
                    websocketConnection.sendBinaryMessage(requestID, path);
                    websocketConnection.sendMessage(requestID, OutboundMessages.DOWNLOAD_RESULTS, "");
                } else {
                    error(new GeppettoExecutionException(
                            "Results of type " + format + " not found in the current experiment"),
                            "Error downloading results for " + aspectPath + " in format " + format);
                }
            }
        } catch (GeppettoExecutionException | IOException e) {
            error(e, "Error downloading results for " + aspectPath + " in format " + format);
        }
    }

    /**
     * @param requestID
     * @param projectId
     * @param properties
     */
    public void saveProjectProperties(String requestID, long projectId, Map<String, String> properties) {
        if (DataManagerHelper.getDataManager().isDefault()) {
            info(requestID, Resources.UNSUPPORTED_OPERATION.toString());
        }
        IGeppettoProject geppettoProject = retrieveGeppettoProject(projectId);
        if (geppettoProject.isVolatile()) {
            info(requestID, Resources.VOLATILE_PROJECT.toString());
            return;
        } else {
            IGeppettoDataManager dataManager = DataManagerHelper.getDataManager();
            if (properties != null) {
                for (String p : properties.keySet()) {
                    switch (p) {
                    case "name": {
                        geppettoProject.setName(properties.get(p));
                        break;
                    }
                    }
                }
            }
            dataManager.saveEntity(geppettoProject);
            websocketConnection.sendMessage(requestID, OutboundMessages.PROJECT_PROPS_SAVED, "");
        }
    }

    /**
     * @param requestID
     * @param projectId
     * @param experimentId
     * @param properties
     */
    public void saveExperimentProperties(String requestID, long projectId, long experimentId,
            Map<String, String> properties) {
        if (DataManagerHelper.getDataManager().isDefault()) {
            info(requestID, Resources.UNSUPPORTED_OPERATION.toString());
        }
        IGeppettoProject geppettoProject = retrieveGeppettoProject(projectId);
        IExperiment experiment = retrieveExperiment(experimentId, geppettoProject);
        if (geppettoProject.isVolatile()) {
            info(requestID, Resources.VOLATILE_PROJECT.toString());
            return;
        } else {
            IGeppettoDataManager dataManager = DataManagerHelper.getDataManager();
            for (String p : properties.keySet()) {
                switch (p) {
                case "name": {
                    experiment.setName(properties.get(p));
                    dataManager.saveEntity(experiment);
                    break;
                }
                case "description": {
                    experiment.setDescription(properties.get(p));
                    dataManager.saveEntity(experiment);
                    break;
                }
                case "script": {
                    experiment.setScript(properties.get(p));
                    dataManager.saveEntity(experiment);
                    break;
                }
                case "timeStep": {
                    String aspectPath = properties.get("aspectInstancePath");
                    for (IAspectConfiguration aspectConfiguration : experiment.getAspectConfigurations()) {
                        if (aspectConfiguration.getAspect().getInstancePath().equals(aspectPath)) {
                            aspectConfiguration.getSimulatorConfiguration()
                                    .setTimestep(Float.parseFloat(properties.get(p)));
                            dataManager.saveEntity(aspectConfiguration.getSimulatorConfiguration());
                            break;
                        }
                    }
                    break;
                }
                case "length": {
                    String aspectPath = properties.get("aspectInstancePath");
                    for (IAspectConfiguration aspectConfiguration : experiment.getAspectConfigurations()) {
                        if (aspectConfiguration.getAspect().getInstancePath().equals(aspectPath)) {
                            aspectConfiguration.getSimulatorConfiguration()
                                    .setLength(Float.parseFloat(properties.get(p)));
                            dataManager.saveEntity(aspectConfiguration.getSimulatorConfiguration());
                            break;
                        }
                    }
                    break;
                }
                case "simulatorId": {
                    String aspectPath = properties.get("aspectInstancePath");
                    for (IAspectConfiguration aspectConfiguration : experiment.getAspectConfigurations()) {
                        if (aspectConfiguration.getAspect().getInstancePath().equals(aspectPath)) {
                            aspectConfiguration.getSimulatorConfiguration().setSimulatorId(properties.get(p));
                            dataManager.saveEntity(aspectConfiguration.getSimulatorConfiguration());
                            break;
                        }
                    }
                    break;
                }
                case "conversionServiceId": {
                    String aspectPath = properties.get("aspectInstancePath");
                    for (IAspectConfiguration aspectConfiguration : experiment.getAspectConfigurations()) {
                        if (aspectConfiguration.getAspect().getInstancePath().equals(aspectPath)) {
                            aspectConfiguration.getSimulatorConfiguration()
                                    .setConversionServiceId(properties.get(p));
                            dataManager.saveEntity(aspectConfiguration.getSimulatorConfiguration());
                            break;
                        }
                    }
                    break;
                }
                case "aspectInstancePath": {
                    break;
                }
                default: {
                    if (p.startsWith("SP$")) {
                        // This is a simulator parameter
                        String aspectPath = properties.get("aspectInstancePath");
                        for (IAspectConfiguration aspectConfiguration : experiment.getAspectConfigurations()) {
                            if (aspectConfiguration.getAspect().getInstancePath().equals(aspectPath)) {

                                Map<String, String> parameters = aspectConfiguration.getSimulatorConfiguration()
                                        .getParameters();
                                if (parameters == null) {
                                    parameters = new HashMap<String, String>();
                                    aspectConfiguration.getSimulatorConfiguration().setParameters(parameters);
                                }
                                parameters.put(p.substring(p.indexOf("$") + 1), properties.get(p));
                                dataManager.saveEntity(aspectConfiguration.getSimulatorConfiguration());
                                break;
                            }
                        }
                        break;
                    } else {
                        String msg = "Cannot find parameter " + p + " in the experiment";
                        error(new GeppettoExecutionException(msg), msg);
                    }
                }
                }
            }
        }
    }

    /**
     * 
     */
    public void closeProject() {
        try {
            geppettoManager.closeProject(null, geppettoProject);
        } catch (GeppettoExecutionException e) {
            logger.error("Error while closing the project", e);
        }

        ConnectionsManager.getInstance().removeConnection(websocketConnection);

    }

    /**
     * @param geppettoProject
     * @throws GeppettoExecutionException
     */
    public void setConnectionProject(IGeppettoProject geppettoProject) throws GeppettoExecutionException {
        if (this.geppettoProject != null) {
            geppettoManager.closeProject(null, this.geppettoProject);
        }
        this.geppettoProject = geppettoProject;
    }

}