All Downloads are FREE. Search and download functionalities are using the official Maven repository.

xyz.thepathfinder.android.Model Maven / Gradle / Ivy

The newest version!
package xyz.thepathfinder.android;

import com.google.gson.JsonObject;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.util.LinkedList;
import java.util.List;
import java.util.Queue;

/**
 * Basic class for dealing with models from the Pathfinder server.
 *
 * @param  The type of model receiving the {@link ModelListener}'s updates.
 * @param  a type of {@link ModelListener}
 * @author David Robinson
 */
public abstract class Model, E extends ModelListener> extends Listenable {

    /**
     * Logs actions performed by the class.
     */
    private static final Logger logger = LoggerFactory.getLogger(Model.class);

    /**
     * The path of the model.
     */
    private final Path path;

    /**
     * A pathfinder services object to have access to the model registry
     * and the web socket.
     */
    private final PathfinderServices services;

    /**
     * Whether or not the model has connected with the Pathfinder server.
     */
    private boolean isConnected;

    /**
     * Messages saved to be sent later, after the model's path is fully defined.
     */
    private Queue messageBacklog;

    /**
     * Creates a basic object that all pathfinder models should use.
     *
     * @param path     to the model on the Pathfinder server.
     * @param type     of the model.
     * @param services a pathfinder services object.
     */
    public Model(String path, ModelType type, PathfinderServices services) {
        this.path = new Path(path, type);
        this.services = services;
        this.isConnected = false;

        if (path == null) {
            this.messageBacklog = new LinkedList();
        }
    }

    /**
     * Returns the path of the model.
     *
     * @return path of the model.
     */
    protected Path getPath() {
        return this.path;
    }

    /**
     * Returns true if the model's path is unknown. This occurs if the model has
     * not been created on the pathfinder server.
     *
     * @return true if the model's path is unknown and false if it is known.
     */
    public boolean isPathUnknown() {
        return this.path.getPathName() == null;
    }

    /**
     * Returns the string of the path of the model.
     *
     * @return the path.
     */
    public String getPathName() {
        return this.path.getPathName();
    }

    /**
     * Set the path of the model. This method may not be called after the path becomes known.
     *
     * @param path of the model.
     * @throws IllegalStateException if the path is already known.
     */
    protected void setPathName(String path) {
        if (this.isPathUnknown()) {
            logger.info("Setting path to: " + path);
            this.path.setPathName(path);
            this.getServices().getRegistry().registerModel(this);

            logger.info("Flushing " + this.getPathName() + "'s message backlog");
            for (JsonObject json : this.messageBacklog) {
                json.addProperty("id", Integer.parseInt(this.getName()));
                this.getServices().getConnection().sendMessage(json.toString());
            }
            logger.info("End flushing backlog");
            this.messageBacklog = null;
        } else {
            logger.error("Illegal State Exception: The path of a model may not be set after becoming known");
            throw new IllegalStateException("The path of a model may not be set after becoming known");
        }
    }

    /**
     * Returns the path of a child of this model.
     *
     * @param name the name to added on to the path.
     * @param type of the model.
     * @return the path of the child.
     */
    protected Path getChildPath(String name, ModelType type) {
        return this.path.getChildPath(name, type);
    }

    /**
     * Returns the path of the parent of this model.
     *
     * @return the path to the parent cluster.
     */
    public Path getParentPath() {
        return this.path.getParentPath();
    }

    /**
     * Returns the name of the model. If the path of this model is
     * "/default/cluster1/subcluster1/transport3" the name is
     * "transport3".
     *
     * @return the name of the model.
     */
    public String getName() {
        return this.path.getName();
    }

    /**
     * Returns the parent cluster of this model.
     *
     * @return the parent cluster of this model. If the default cluster it returns null.
     */
    public Cluster getParentCluster() {
        Path parentPath = this.getParentPath();
        return Cluster.getInstance(parentPath.getPathName(), services);
    }

    /**
     * Returns the type of the model
     *
     * @return type of model
     */
    public ModelType getModelType() {
        return this.path.getModelType();
    }

    /**
     * Returns the pathfinder services object.
     *
     * @return a pathfinder services object.
     */
    protected PathfinderServices getServices() {
        return this.services;
    }

    /**
     * Returns if the model has connected to the Pathfinder server.
     *
     * @return true if the model has connected to the Pathfinder server, false otherwise.
     */
    public boolean isConnected() {
        return this.isConnected;
    }

    /**
     * Sets if the model has connected with the Pathfinder server.
     *
     * @param connected has connected with the Pathfinder server.
     */
    protected void setConnected(boolean connected) {
        this.isConnected = connected;
    }

    /**
     * Invokes the model unspecific notifications.
     *
     * @param reason message type
     * @param json   the message
     * @return true if model changed in any way, false otherwise.
     */
    private boolean updateType(String reason, JsonObject json) {
        boolean updated = false;
        JsonObject value = null;

        logger.info("Reason for update is: " + reason + ", " + json);

        if (json.has("value") && !json.has("route")) {
            value = json.getAsJsonObject("value");
        } else if (reason == null) {
            value = json;
        }

        logger.info("Value is: " + value);

        if (value != null) {
            updated = this.updateFields(value);
        }

        logger.info("Finished updating " + this.getPathName() + "'s fields.");
        List listeners = this.getListeners();

        if (updated && !"Updated".equals(reason)) {
            logger.info("Model " + this.getPathName() + " updated");
            for (E listener : listeners) {
                logger.info("Calling updated");
                listener.updated(this.getThis());
                logger.info("Finished Calling updated");
            }
        }

        logger.info("Reason: " + reason);
        if (reason == null) {
            return updated;
        }

        if (reason.equals("Updated")) {
            logger.info("Model " + this.getPathName() + " updated");
            for (E listener : listeners) {
                listener.updated(this.getThis());
            }
            return updated;
        }

        if (reason.equals("Routed")) {
            logger.info("Model " + this.getPathName() + " routed");
            this.route(json, this.getServices());
            return true;
        }

        if (reason.equals("Model")) {
            logger.info("Model " + this.getPathName() + " connected");
            for (E listener : listeners) {
                listener.connected(this.getThis());
            }
            return updated;
        }

        if (reason.equals("Subscribed")) {
            logger.info("Model " + this.getPathName() + " subscribed");
            for (E listener : listeners) {
                listener.subscribed(this.getThis());
            }
            return updated;
        }

        if (reason.equals("RouteSubscribed")) {
            logger.info("Model " + this.getPathName() + " route subscribed");
            for (E listener : listeners) {
                listener.routeSubscribed(this.getThis());
            }
            return updated;
        }

        if (reason.equals("Unsubscribed")) {
            logger.info("Model " + this.getPathName() + " unsubscribed");
            for (E listener : listeners) {
                listener.unsubscribed(this.getThis());
            }
            return updated;
        }

        if (reason.equals("RouteUnsubscribed")) {
            logger.info("Model " + this.getPathName() + " route unsubscribed");
            for (E listener : listeners) {
                listener.routeUnsubscribed(this.getThis());
            }
            return updated;
        }

        if (reason.equals("Created")) {
            logger.info("Model " + this.getPathName() + " created");
            for (E listener : listeners) {
                listener.created(this.getThis());
            }
            return updated;
        }

        if (reason.equals("Deleted")) {
            logger.info("Model " + this.getPathName() + " deleted");
            for (E listener : listeners) {
                this.setConnected(false);
                listener.deleted(this.getThis());
            }
            return updated;
        }

        if (reason.equals("Error") && value != null) {
            logger.warn("Model " + this.getPathName() + " received error: " + value.get("reason").getAsString());
            for (E listener : listeners) {
                listener.error(value.get("reason").getAsString());
            }
            return updated;
        }

        logger.warn("Invalid message sent to " + this.getPathName() + " with type: " + reason + "\nJson: " + json);
        return updated;
    }

    /**
     * {@inheritDoc}
     */
    @Override
    protected boolean notifyUpdate(String reason, JsonObject json) {
        if (!this.isPathUnknown()) {
            this.setConnected(true);
        }
        return this.updateType(reason, json);
    }

    /**
     * Sends a json message through the web socket connection if connected. Otherwise, it stores the message.
     *
     * @param json message to be sent.
     */
    protected void sendMessage(JsonObject json) {
        if (!this.isPathUnknown()) {
            this.getServices().getConnection().sendMessage(json.toString());
        } else {
            this.messageBacklog.offer(json);
        }
    }

    /**
     * Converts the {@link Model} to JSON.
     *
     * @return the {@link Model} as JSON.
     */
    public abstract JsonObject toJson();

    /**
     * Updates the fields of the model.
     *
     * @param json of the model.
     * @return true if model changed in any way, false otherwise.
     */
    protected abstract boolean updateFields(JsonObject json);

    /**
     * Updates the models routes.
     *
     * @param json     of the model.
     * @param services pathfinder services object.
     */
    protected abstract void route(JsonObject json, PathfinderServices services);

    /**
     * Returns the current object represented by this.
     *
     * @return this.
     */
    protected abstract T getThis();
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy