package tmcsim.simulationmanager;
import java.io.File;
import java.rmi.Naming;
import java.rmi.RemoteException;
import java.rmi.server.UnicastRemoteObject;
import java.util.TreeMap;
import java.util.TreeSet;
import java.util.Vector;
import java.util.logging.Level;
import java.util.logging.Logger;
import tmcsim.cadmodels.CMSDiversion;
import tmcsim.cadmodels.CMSInfo;
import tmcsim.client.cadclientgui.data.Incident;
import tmcsim.client.cadclientgui.data.IncidentEvent;
import tmcsim.common.CADEnums.PARAMICS_STATUS;
import tmcsim.common.CADEnums.SCRIPT_STATUS;
import tmcsim.common.ScriptException;
import tmcsim.common.SimulationException;
import tmcsim.interfaces.CoordinatorInterface;
import tmcsim.interfaces.SimulationManagerInterface;
/**
* SimulationManagerModel is the model class for the Simulation Manager. All
* communication between the Coordinator and the Simulation Manager passes
* through this object. The view passes requests through local methods, and the
* Coordinator calls remote functions in this object to update the viewed
* simulation data.
*
* At construction, the SimulationManagerModel registers itself with the
* coordinator. Administrative commands for the simulation are received from the
* SimlationManagerView class, and then the appropriate Coordinator remote
* method is executed. During a simulation, the Coordinator calls remote methods
* contained in the SimulationManagerInterface. For a description of those
* methods, see the interface's documentation. After construction, this model
* class must have its setView() method called to set the reference to its view
* class.
*
* @see SimulationManagerView
* @see SimulationManagerInterface
* @author Matthew Cechini
* @version $Revision: 1.3 $ $Date: 2006/06/06 20:46:41 $
*/
@SuppressWarnings("serial")
public class SimulationManagerModel extends UnicastRemoteObject
implements SimulationManagerInterface
{
/**
* Error Logger.
*/
private Logger simManagerLogger = Logger.getLogger("tmcsim.simulationmanager");
/**
* RMI interface for communication with the remote Coordinator.
*/
private static CoordinatorInterface theCoorInt;
/**
* The SimulationManagerView object.
*/
private SimulationManagerView theSimManagerView;
/**
* Constructor. Establishes the RMI communication with the Coordinator.
*
* @param hostname Host name of the CAD Simulator.
* @param portNumber Port number of the CAD Simulator RMI communication.
* @throws RemoteException if error in RMI communication
* @throws SimulationException if there is an error in registering RMI
* methods.
*/
public SimulationManagerModel(String hostname, String portNumber)
throws RemoteException, SimulationException
{
super();
connect(hostname, portNumber);
}
/**
* Connect to the Coordinator's RMI object, and register this object for
* callback with the Coordinator.
*
* @param hostname Host name of the CAD Simulator.
* @param portNumber Port number of the CAD Simulator RMI communication.
* @throws SimulationException if there is an error creating the RMI
* connection.
*/
protected void connect(String hostname, String portNumber)
throws SimulationException
{
String coorIntURL = "";
try
{
coorIntURL = "rmi://" + hostname + ":" + portNumber + "/coordinator";
theCoorInt = (CoordinatorInterface) Naming.lookup(coorIntURL);
theCoorInt.registerForCallback(this);
} catch (Exception e)
{
simManagerLogger.logp(Level.SEVERE, "SimulationManagerModel",
"establishRMIConnection", "Unable to establish RMI "
+ "communication with the CAD Simulator. URL <" + coorIntURL + ">", e);
throw new SimulationException(SimulationException.CAD_SIM_CONNECT, e);
}
}
/**
* This method unregisters this SimulationManager from the Coordinator and
* closes the RMI communication.
*/
public void disconnect()
{
try
{
theCoorInt.unregisterForCallback(this);
theCoorInt = null;
} catch (Exception ex)
{
simManagerLogger.logp(Level.SEVERE, "SimulationManagerModel",
"closeSimManager", "Exception in unregistering Simulation"
+ "Manager from the CAD Simulator.", ex.getMessage());
}
}
/**
* Set the local reference to the SimulationManagerView object. The view is
* updated with the current simulation time, script status, and Paramics
* connection status.
*
* @param newView The instance of the SimulationManagerView class.
* @throws SimulationException if there is an error in RMI communication to
* the CAD Simulator.
*/
public void setView(SimulationManagerView newView) throws SimulationException
{
try
{
theSimManagerView = newView;
theSimManagerView.tick(theCoorInt.getCurrentSimulationTime());
theSimManagerView.setScriptStatus(theCoorInt.getScriptStatus());
theSimManagerView.setParamicsStatus(theCoorInt.getParamicsStatus());
initialize();
} catch (RemoteException re)
{
simManagerLogger.logp(Level.SEVERE, "SimulationManagerModel",
"startSimulation", "Unable to communicate with the "
+ "CAD Simulator.", re);
throw new SimulationException(SimulationException.CAD_SIM_COMM, re);
}
}
/**
* This method initializes the SimulationManager with all current simulation
* data from the Coordinator. If the simulation is running when the
* Simulation Manager is initialized, all previously occured incidents,
* events, and current diversion must be displayed. This method first loads
* the list of current incidents from the coordinator, then the triggered
* events, followed by all CMSInfo objects. The SimulationManagerView is
* notified of any current diversions within the CMSInfo objects.
*
* @throws SimulationException if there is an error in RMI communication or
* if the SimulationManagerView reference has not been set.
*/
protected void initialize() throws SimulationException
{
try
{
//Load all incidents from Coordinator
loadIncidents();
//Load all triggered incidents from Coordinator
TreeMap> tempEvents = theCoorInt.getTriggeredEvents();
for (Integer key : tempEvents.keySet())
{
for (IncidentEvent ie : tempEvents.get(key))
{
eventOccured(key, ie);
}
}
//Load all current diversions from the Coordinator
TreeSet cmsIDs = theCoorInt.getCMSIDs();
CMSInfo cmsinfo = null;
for (String cms_id : cmsIDs)
{
cmsinfo = theCoorInt.getCMSDiversionInfo(cms_id);
for (CMSDiversion div : cmsinfo.possibleDiversions)
{
if (div.getCurrDiv() != 0)
{
theSimManagerView.addDiversion(cmsinfo, div);
}
}
}
//Send the list of CMS IDs to the View.
theSimManagerView.setCMS_IDList(cmsIDs.toArray());
} catch (Exception e)
{
simManagerLogger.logp(Level.SEVERE, "SimulationManagerModel",
"initialize", "Unable to initialize the SimulationManager.", e);
throw new SimulationException(SimulationException.CAD_SIM_COMM, e);
}
}
public void tick(long theTime) throws RemoteException
{
if (theSimManagerView != null)
{
theSimManagerView.tick(theTime);
}
}
public void incidentAdded(Incident newIncident) throws RemoteException
{
if (theSimManagerView != null)
{
Integer logNum = new Integer(newIncident.getLogNumber());
theSimManagerView.addIncident(newIncident);
theSimManagerView.addIncidentTab(logNum);
}
}
public void incidentStarted(Integer logNumber) throws RemoteException
{
if (theSimManagerView != null)
{
theSimManagerView.startIncident(logNumber);
}
}
public void incidentRemoved(Integer logNumber) throws RemoteException
{
if (theSimManagerView != null)
{
theSimManagerView.removeIncident(logNumber);
theSimManagerView.removeIncidentTab(logNumber);
}
}
public void eventOccured(Integer logNumber, IncidentEvent theEvent) throws RemoteException
{
if (theSimManagerView != null)
{
theSimManagerView.addIncidentEvent(logNumber, theEvent);
}
}
public void setScriptStatus(SCRIPT_STATUS newStatus) throws RemoteException
{
if (theSimManagerView != null)
{
theSimManagerView.setScriptStatus(newStatus);
}
}
public void setParamicsStatus(PARAMICS_STATUS newStatus) throws RemoteException
{
if (theSimManagerView != null)
{
theSimManagerView.setParamicsStatus(newStatus);
}
}
/**
* This method passes the view's request to start the simulation on to the
* remote coordinator.
*
* @throws ScriptException if an error occurs in started the simulation.
* @throws SimulationException if there is an error in RMI communication to
* the CAD Simulator.
*/
public void startSimulation() throws ScriptException, SimulationException
{
try
{
theCoorInt.startSimulation();
} catch (RemoteException re)
{
simManagerLogger.logp(Level.SEVERE, "SimulationManagerModel",
"startSimulation", "Unable to communicate with the "
+ "CAD Simulator.", re);
throw new SimulationException(SimulationException.CAD_SIM_COMM, re);
}
}
/**
* This method passes the view's request to reset the simulation on to the
* remote coordinator. The View's simulation time is reset to 0. The view's
* incident tabs are cleared and incident list reset and initialized with
* the current list of incidents. The list of diversions is also reset.
*
* @throws SimulationException if there is an error in RMI communication to
* the CAD Simulator.
*/
public void resetSimulation() throws SimulationException
{
try
{
theCoorInt.resetSimulation();
tick(0);
theSimManagerView.resetIncidentTabs();
theSimManagerView.resetIncidents();
theSimManagerView.resetDiversions();
loadIncidents();
} catch (RemoteException re)
{
simManagerLogger.logp(Level.SEVERE, "SimulationManagerModel",
"resetSimulation", "Unable to communicate with the "
+ "CAD Simulator.", re);
throw new SimulationException(SimulationException.CAD_SIM_COMM, re);
}
}
/**
* This method passes the view's request to pause the simulation on to the
* remote coordinator.
*
* @throws SimulationException if there is an error in RMI communication to
* the CAD Simulator.
*/
public void pauseSimulation() throws SimulationException
{
try
{
theCoorInt.pauseSimulation();
} catch (RemoteException re)
{
simManagerLogger.logp(Level.SEVERE, "SimulationManagerModel",
"pauseSimulation", "Unable to communicate with the "
+ "CAD Simulator.", re);
throw new SimulationException(SimulationException.CAD_SIM_COMM, re);
}
}
/**
* This method passes the view's request to goto a new simulation time on to
* the remote coordinator.
*
* @param time Simulation time
* @throws SimulationException if there is an error in RMI communication to
* the CAD Simulator.
*/
public void gotoSimulationTime(long time) throws SimulationException
{
try
{
theCoorInt.gotoSimulationTime(time);
} catch (RemoteException re)
{
simManagerLogger.logp(Level.SEVERE, "SimulationManagerModel",
"gotoSimulationTime", "Unable to communicate with the "
+ "CAD Simulator.", re);
throw new SimulationException(SimulationException.CAD_SIM_COMM, re);
}
}
/**
* This method passes the view's request to load a script file on to the
* remote Coordinator. If this is successful, the View's incident tabs are
* cleared and incident list reset and initialized with the current list of
* incidents. If the load is not successful, the View's incident tabs and
* incident list are cleared. The list of diversions is also reset.
*
* @param scriptFile the File chosen by the user in the open file dialog.
* @throws ScriptException if an error occurs in reading the script file.
* @throws SimulationException if there is an error in RMI communication to
* the CAD Simulator.
*/
public void loadScript(File scriptFile) throws ScriptException, SimulationException
{
try
{
theCoorInt.loadScriptFile(scriptFile);
tick(0);
theSimManagerView.resetIncidentTabs();
theSimManagerView.resetIncidents();
theSimManagerView.resetDiversions();
loadIncidents();
} catch (ScriptException se)
{
theSimManagerView.resetIncidentTabs();
theSimManagerView.resetIncidents();
theSimManagerView.resetDiversions();
throw se;
} catch (RemoteException re)
{
simManagerLogger.logp(Level.SEVERE, "SimulationManagerModel",
"loadScript", "Unable to communicate with the "
+ "CAD Simulator.", re);
throw new SimulationException(SimulationException.CAD_SIM_COMM, re);
}
}
/**
* This method passes the view's request to create a connection between the
* CADSimulator and the Paramics Communicator on to the remote Coordinator.
*
* @throws SimulationException if there is an error in RMI communication to
* the CAD Simulator.
*/
public void connectToParamics() throws SimulationException
{
try
{
theCoorInt.connectToParamics();
} catch (RemoteException re)
{
simManagerLogger.logp(Level.SEVERE, "SimulationManagerModel",
"connectToParamics", "Unable to communicate with the "
+ "CAD Simulator.", re);
throw new SimulationException(SimulationException.CAD_SIM_COMM, re);
}
}
/**
* This method passes the view's request to drop the connection between the
* CADSimulator and the Paramics Communicator on to the remote Coordinator.
*
* @throws SimulationException if there is an error in RMI communication to
* the CAD Simulator.
*/
public void disconnectFromParamics() throws SimulationException
{
try
{
theCoorInt.disconnectFromParamics();
} catch (RemoteException re)
{
simManagerLogger.logp(Level.SEVERE, "SimulationManagerModel",
"disconnectFromParamics", "Unable to communicate with the "
+ "CAD Simulator.", re);
throw new SimulationException(SimulationException.CAD_SIM_COMM, re);
}
}
/**
* This method passes the view's request to load a paramics network on to
* the remote Coordinator.
*
* @param networkID The unique network ID that is being loaded
*
* @throws ScriptException if there is an error in loading the network
* @throws SimulationException if there is an error in RMI communication to
* the CAD Simulator.
*/
public void loadParamicsNetwork(int networkID) throws ScriptException, SimulationException
{
try
{
theCoorInt.loadParamicsNetwork(networkID);
} catch (RemoteException re)
{
simManagerLogger.logp(Level.SEVERE, "SimulationManagerModel",
"loadParamicsNetwork", "Unable to communicate with the "
+ "CAD Simulator.", re);
throw new SimulationException(SimulationException.CAD_SIM_COMM, re);
}
}
/**
* This method passes the view's request to get the value of the currently
* loaded paramics network on to the remote Coordinator.
*
* @return Value of the loaded paramics network.
* @throws SimulationException if there is an error in RMI communication to
* the CAD Simulator.
*/
public int getParamicsNetworkLoaded() throws SimulationException
{
try
{
return theCoorInt.getParamicsNetworkLoaded();
} catch (RemoteException re)
{
simManagerLogger.logp(Level.SEVERE, "SimulationManagerModel",
"getParamicsNetworkLoaded", "Unable to communicate with the "
+ "CAD Simulator.", re);
throw new SimulationException(SimulationException.CAD_SIM_COMM, re);
}
}
/**
* This method passes the view's request to trigger an incident on to the
* remote Coordinator.
*
* @throws ScriptException if an error occurs in triggering an event.
* @throws SimulationException if there is an error in RMI communication to
* the CAD Simulator.
*/
public void triggerIncident(int logNumber) throws ScriptException, SimulationException
{
try
{
theCoorInt.triggerIncident(logNumber);
} catch (RemoteException re)
{
simManagerLogger.logp(Level.SEVERE, "SimulationManagerModel",
"startIncident", "Unable to communicate with the "
+ "CAD Simulator.", re);
throw new SimulationException(SimulationException.CAD_SIM_COMM, re);
}
}
/**
* This method passes the view's request to add an incident into the
* simulation on to the remote Coordinator.
*
* @param newIncident Incident to add to the simulation
* @throws SimulationException if there is an error in RMI communication to
* the CAD Simulator.
*/
public void addIncident(final Incident newIncident) throws SimulationException
{
try
{
theCoorInt.addIncident(newIncident);
} catch (RemoteException re)
{
simManagerLogger.logp(Level.SEVERE, "SimulationManagerModel",
"addIncident", "Unable to communicate with the "
+ "CAD Simulator.", re);
throw new SimulationException(SimulationException.CAD_SIM_COMM, re);
}
}
/**
* This method passes the view's request to delete an incident from the
* simulation on to the remote Coordinator.
*
* @throws ScriptException if an error occurs in deleting an event.
* @throws SimulationException if there is an error in RMI communication to
* the CAD Simulator.
*/
public void deleteIncident(int logNumber) throws ScriptException, SimulationException
{
try
{
theCoorInt.deleteIncident(logNumber);
} catch (RemoteException re)
{
simManagerLogger.logp(Level.SEVERE, "SimulationManagerModel",
"deleteIncident", "Unable to communicate with the "
+ "CAD Simulator.", re);
throw new SimulationException(SimulationException.CAD_SIM_COMM, re);
}
}
/**
* This method passes the view's request to reschedule an incident on to the
* remote Coordinator.
*
* @param newTime New simulation time (in seconds).
* @throws ScriptException if the Incident has already started or the time
* for recheduling has already passed.
* @throws SimulationException if there is an error in RMI communication to
* the CAD Simulator.
*/
public void rescheduleIncident(long newTime, int logNumber) throws ScriptException, SimulationException
{
try
{
theCoorInt.rescheduleIncident(logNumber, newTime);
} catch (RemoteException re)
{
simManagerLogger.logp(Level.SEVERE, "SimulationManagerModel",
"rescheduleIncident", "Unable to communicate with the "
+ "CAD Simulator.", re);
throw new SimulationException(SimulationException.CAD_SIM_COMM, re);
}
}
/**
* This method passes the view's request to get the current list of
* Incidents loaded into the simulation on to the remote Coordinator.
*
* @return Vector The Vector of currently loaded incidents.
* @throws SimulationException if there is an error in RMI communication to
* the CAD Simulator.
*/
public Vector getIncidentList() throws SimulationException
{
try
{
return theCoorInt.getIncidentList();
} catch (RemoteException re)
{
simManagerLogger.logp(Level.SEVERE, "SimulationManagerModel",
"getIncidentList", "Unable to communicate with the "
+ "CAD Simulator.", re);
throw new SimulationException(SimulationException.CAD_SIM_COMM, re);
}
}
/**
* This method passes the view's request to get the CMSInfo object that
* corresponds to a unique CMS ID String on to the remote Coordinator.
*
* @param cms_id Unique CMS ID String.
* @return CMSInfo object corresponding to the parameter CMS id.
* @throws SimulationException if there is an error in RMI communication to
* the CAD Simulator.
*/
public CMSInfo getCMSDiversionInfo(String cms_id) throws SimulationException
{
try
{
return theCoorInt.getCMSDiversionInfo(cms_id);
} catch (RemoteException re)
{
simManagerLogger.logp(Level.SEVERE, "SimulationManagerModel",
"getCMSDiversionInfo", "Unable to communicate with the "
+ "CAD Simulator.", re);
throw new SimulationException(SimulationException.CAD_SIM_COMM, re);
}
}
/**
* This method passes the view's request to update diversions for a CMS on
* to the remote Coordinator.
*
* @param diversion CMS diversions information to apply.
* @throws SimulationException if there is an error in RMI communication to
* the CAD Simulator.
*/
public void applyDiversions(CMSInfo diversion) throws SimulationException
{
try
{
theCoorInt.applyDiversions(diversion);
} catch (RemoteException re)
{
simManagerLogger.logp(Level.SEVERE, "SimulationManagerModel",
"applyDiversions", "Unable to communicate with the "
+ "CAD Simulator.", re);
throw new SimulationException(SimulationException.CAD_SIM_COMM, re);
}
}
/**
* This method loads the current list of incidents from the Coordinator into
* the SimulationManager. For each incident in the simulation, notify the
* view to add a new incident tab. If the Incident has begun in the
* simulation, call the incidentStarted() method to update the view
* accordingly.
*
* @throws SimulationException if there is an error in RMI communication to
* the CAD Simulator.
*/
private void loadIncidents() throws SimulationException
{
try
{
for (Incident inc : theCoorInt.getIncidentList())
{
Integer logNum = new Integer(inc.getLogNumber());
theSimManagerView.addIncidentTab(logNum);
theSimManagerView.addIncident(inc);
if (inc.getSecondsToStart() < theCoorInt.getCurrentSimulationTime())
{
incidentStarted(logNum);
}
}
} catch (RemoteException re)
{
simManagerLogger.logp(Level.SEVERE, "SimulationManagerModel",
"loadIncidentListTable", "Unable to communicate with the "
+ "CAD Simulator.", re);
throw new SimulationException(SimulationException.CAD_SIM_COMM, re);
}
}
}