/* An application for editing ptolemy models visually.
Copyright (c) 1999-2008 The Regents of the University of California.
All rights reserved.
Permission is hereby granted, without written agreement and without
license or royalty fees, to use, copy, modify, and distribute this
software and its documentation for any purpose, provided that the above
copyright notice and the following two paragraphs appear in all copies
of this software.
IN NO EVENT SHALL THE UNIVERSITY OF CALIFORNIA BE LIABLE TO ANY PARTY
FOR DIRECT, INDIRECT, SPECIAL, INCIDENTAL, OR CONSEQUENTIAL DAMAGES
ARISING OUT OF THE USE OF THIS SOFTWARE AND ITS DOCUMENTATION, EVEN IF
THE UNIVERSITY OF CALIFORNIA HAS BEEN ADVISED OF THE POSSIBILITY OF
SUCH DAMAGE.
THE UNIVERSITY OF CALIFORNIA SPECIFICALLY DISCLAIMS ANY WARRANTIES,
INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. THE SOFTWARE
PROVIDED HEREUNDER IS ON AN "AS IS" BASIS, AND THE UNIVERSITY OF
CALIFORNIA HAS NO OBLIGATION TO PROVIDE MAINTENANCE, SUPPORT, UPDATES,
ENHANCEMENTS, OR MODIFICATIONS.
PT_COPYRIGHT_VERSION_2
COPYRIGHTENDKEY
*/
package ptolemy.vergil;
import java.io.File;
import java.io.IOException;
import java.net.URL;
import java.util.LinkedList;
import java.util.List;
import javax.swing.SwingUtilities;
import ptolemy.actor.gui.Configuration;
import ptolemy.actor.gui.Effigy;
import ptolemy.actor.gui.MoMLApplication;
import ptolemy.actor.gui.ModelDirectory;
import ptolemy.actor.gui.PtolemyEffigy;
import ptolemy.actor.gui.PtolemyPreferences;
import ptolemy.actor.gui.UserActorLibrary;
import ptolemy.data.expr.Parameter;
import ptolemy.gui.GraphicalMessageHandler;
import ptolemy.kernel.util.IllegalActionException;
import ptolemy.moml.MoMLParser;
import ptolemy.util.MessageHandler;
//////////////////////////////////////////////////////////////////////////
//// VergilApplication
/**
This application opens run control panels for models specified on the
command line.
<p>
The exact facilities that are available are determined by an optional
command line argument that names a directory in ptolemy/configs that
contains a configuration.xml file. For example, if we call vergil
-ptiny, then we will use ptolemy/configs/ptiny/configuration.xml and
ptolemy/configs/ptiny/intro.htm. The default configuration is
ptolemy/configs/full/configuration.xml, which is loaded before any
other command-line arguments are processed.
<p>This application also takes an optional command line argument pair
<code>-configuration <i>configurationFile.xml</i></code> that names a configuration
to be read. For example,
<pre>
$PTII/bin/vergil -configuration ptolemy/configs/ptiny/configuration.xml
</pre>
and
<pre>
$PTII/bin/vergil -ptiny
</pre>
are equivalent
<p>
If there are no command-line arguments at all, then the configuration
file is augmented by the MoML file ptolemy/configs/full/welcomeWindow.xml
@author Edward A. Lee, Steve Neuendorffer, Christopher Hylands, contributor: Chad Berkeley
@version $Id: VergilApplication.java,v 1.168.4.1 2008/03/25 22:32:13 cxh Exp $
@since Ptolemy II 1.0
@Pt.ProposedRating Yellow (eal)
@Pt.AcceptedRating Red (eal)
@see ptolemy.actor.gui.ModelFrame
@see ptolemy.actor.gui.RunTableau
@see ptolemy.actor.gui.PtExecuteApplication
*/
public class VergilApplication extends MoMLApplication {
/** Parse the specified command-line arguments, creating models
* and frames to interact with them.
* Look for configurations in "ptolemy/configs"
* @param args The command-line arguments.
* @exception Exception If command line arguments have problems.
*/
public VergilApplication(String[] args) throws Exception {
super("ptolemy/configs", args);
// Create register an error handler with the parser so that
// MoML errors are tolerated more than the default.
MoMLParser.setErrorHandler(new VergilErrorHandler());
}
/** Parse the specified command-line arguments, creating models
* and frames to interact with them.
* @param basePath The basePath to look for configurations
* in, usually "ptolemy/configs", but other tools might
* have other configurations in other directories
* @param args The command-line arguments.
* @exception Exception If command line arguments have problems.
*/
public VergilApplication(String basePath, String[] args) throws Exception {
super(basePath, args);
// Create register an error handler with the parser so that
// MoML errors are tolerated more than the default.
MoMLParser.setErrorHandler(new VergilErrorHandler());
}
///////////////////////////////////////////////////////////////////
//// public methods ////
/** Create a new instance of this application, passing it the
* command-line arguments.
* @param args The command-line arguments.
*/
public static void main(final String[] args) {
// FIXME: Java superstition dictates that if you want something
// to work, you should invoke it in event thread. Otherwise,
// weird things happens at the user interface level. This
// seems to prevent occasional errors rending HTML under Web Start.
try {
// NOTE: This is unfortunate... It would be nice
// if this could be run inside a PtolemyThread, since
// getting read access the workspace is much more efficient
// in PtolemyThread.
SwingUtilities.invokeLater(new Runnable() {
public void run() {
try {
new VergilApplication(args);
} catch (Throwable throwable) {
// If we get an Error or and Exception while
// configuring, we will end up here.
_errorAndExit("Command failed", args, throwable);
}
}
});
} catch (Throwable throwable2) {
// We are not likely to get here, but just to be safe
// we try to print the error message and display it in a
// graphical widget.
_errorAndExit("Command failed", args, throwable2);
}
// If the -test arg was set, then exit after 2 seconds.
if (_test) {
try {
Thread.sleep(2000);
} catch (InterruptedException e) {
}
System.exit(0);
}
}
/**
* Open the MoML file at the given location as a new library in the
* actor library for this application.
*
* An alternate class can be used to build the library if reading the
* MoML is not desired. The class must extend ptolemy.moml.LibraryBuilder
* and the _alternateLibraryBuilder property must be set with the 'value'
* set to the class that extends LibraryBuilder.
*
* @param configuration The configuration where we look for the
* actor library.
* @param file The MoML file to open.
* @exception Exception If there is a problem opening the configuration,
* opening the MoML file, or opening the MoML file as a new library.
* @deprecated Use {@link ptolemy.actor.gui.UserActorLibrary#openLibrary(Configuration, File)}
*/
public static void openLibrary(Configuration configuration, File file)
throws Exception {
MoMLParser.setErrorHandler(new VergilErrorHandler());
UserActorLibrary.openLibrary(configuration, file);
}
///////////////////////////////////////////////////////////////////
//// protected methods ////
/** Return a default Configuration. The initial default configuration
* is the MoML file full/configuration.xml under the _basePath
* directory, which is usually ptolemy/configs.
* using different command line arguments can change the value
* Usually, we also open the user library, which is located
* in the directory returned by
* {@link ptolemy.util.StringUtilities#preferencesDirectory()}
* If the configuration contains a top level Parameter named
* _hideUserLibrary, then we do not open the user library.
*
* @return A default configuration.
* @exception Exception If the configuration cannot be opened.
*/
protected Configuration _createDefaultConfiguration() throws Exception {
try {
if (_configurationURL == null) {
_configurationURL = specToURL(_basePath
+ "/full/configuration.xml");
}
} catch (IOException ex) {
try {
// If we ship HyVisual without a full installation, then
// we try the hyvisual configuration.
// vergil -help needs this.
// FIXME: we could do better than this and either
// search for configurations or go through a list
// of them.
_configurationURL = specToURL(_basePath
+ "/hyvisual/configuration.xml");
} catch (IOException ex2) {
// Throw the original exception.
throw ex;
}
}
// This has the side effects of merging properties from ptII.properties
Configuration configuration = super._createDefaultConfiguration();
try {
configuration = readConfiguration(_configurationURL);
} catch (Exception ex) {
throw new Exception("Failed to read configuration '"
+ _configurationURL + "'", ex);
}
// Read the user preferences, if any.
PtolemyPreferences.setDefaultPreferences(configuration);
// If _hideUserLibraryAttribute is not present, or is false,
// call openUserLibrary(). openUserLibrary() will open either the
// user library or the library named by the _alternateLibraryBuilder.
Parameter hideUserLibraryAttribute = (Parameter) configuration
.getAttribute("_hideUserLibrary", Parameter.class);
if ((hideUserLibraryAttribute == null)
|| hideUserLibraryAttribute.getExpression().equals("false")) {
// Load the user library.
try {
MoMLParser.setErrorHandler(new VergilErrorHandler());
UserActorLibrary.openUserLibrary(configuration);
} catch (Exception ex) {
MessageHandler.error("Failed to display user library.", ex);
}
}
return configuration;
}
/** Return a default Configuration to use when there are no command-line
* arguments. If the configuration contains a parameter
* _applicationBlankPtolemyEffigyAtStartup
* then we create an empty up an empty PtolemyEffigy.
* @return A configuration for when there no command-line arguments.
* @exception Exception If the configuration cannot be opened.
*/
protected Configuration _createEmptyConfiguration() throws Exception {
Configuration configuration = _createDefaultConfiguration();
URL welcomeURL = null;
URL introURL = null;
ModelDirectory directory = configuration.getDirectory();
Parameter applicationBlankPtolemyEffigyAtStartup = (Parameter) configuration
.getAttribute("_applicationBlankPtolemyEffigyAtStartup",
Parameter.class);
if ((applicationBlankPtolemyEffigyAtStartup != null)
&& applicationBlankPtolemyEffigyAtStartup.getExpression()
.equals("true")) {
PtolemyEffigy.Factory factory = new PtolemyEffigy.Factory(
directory, directory.uniqueName("ptolemyEffigy"));
Effigy effigy = factory.createEffigy(directory, null, null);
configuration.createPrimaryTableau(effigy);
}
try {
// First, we see if we can find the welcome window by
// looking at the default configuration.
// FIXME: this seems wrong, we should be able to get
// an attribute from the configuration that names the
// welcome window.
String configurationURLString = _configurationURL.toExternalForm();
String base = configurationURLString.substring(0,
configurationURLString.lastIndexOf("/"));
welcomeURL = specToURL(base + "/welcomeWindow.xml");
introURL = specToURL(base + "/intro.htm");
_parser.reset();
_parser.setContext(configuration);
_parser.parse(welcomeURL, welcomeURL);
} catch (Throwable throwable) {
// OK, that did not work, try a different method.
if (_configurationSubdirectory == null) {
_configurationSubdirectory = "full";
}
// FIXME: This code is Dog slow for some reason.
welcomeURL = specToURL(_basePath + "/" + _configurationSubdirectory
+ "/welcomeWindow.xml");
introURL = specToURL(_basePath + "/" + _configurationSubdirectory
+ "/intro.htm");
_parser.reset();
_parser.setContext(configuration);
_parser.parse(welcomeURL, welcomeURL);
}
Effigy doc = (Effigy) configuration.getEntity("directory.doc");
doc.identifier.setExpression(introURL.toExternalForm());
return configuration;
}
/** Parse the command-line arguments.
* @exception Exception If an argument is not understood or triggers
* an error.
*/
protected void _parseArgs(final String[] args) throws Exception {
_commandTemplate = "vergil [ options ] [file ...]";
// VergilApplication.super._parseArgs(args)
// calls _createDefaultConfiguration() asap,
// so delay calling it until we process
// the arguments and possibly get a configuration.
List processedArgsList = new LinkedList();
for (int i = 0; i < args.length; i++) {
// Parse any configuration specific args first so that we can
// set up the configuration for later use by the parent class.
if (!_configurationParseArg(args[i])) {
// If we did not parse the arg, the
// add it to the list of args to be passed
// to the super class.
processedArgsList.add(args[i]);
}
}
if (_expectingConfiguration) {
throw new IllegalActionException("Missing configuration");
}
String[] processedArgs = (String[]) processedArgsList
.toArray(new String[processedArgsList.size()]);
super._parseArgs(processedArgs);
}
/** Return a string summarizing the command-line arguments.
* @return A usage string.
*/
protected String _usage() {
return _configurationUsage(_commandTemplate, _commandOptions,
new String[] {});
}
///////////////////////////////////////////////////////////////////
//// protected variables ////
// _commandOptions is static because
// _usage() may be called from the constructor of the parent class,
// in which case non-static variables are null?
/** The command-line options that take arguments. */
protected static String[][] _commandOptions = { { "-configuration",
"<configuration URL, defaults to ptolemy/configs/full/configuration.xml>" }, };
///////////////////////////////////////////////////////////////////
//// private methods ////
/** Parse a command-line argument. Usually, we would name this
* method _parseArg(), but we want to handle any arguments that
* handle configuration changes before calling the parent class
* _parseArg() because the parent class depends either having a
* configuration to work with, or the parent class sets up a
* configuration.
* @return True if the argument is understood, false otherwise.
* @exception Exception If something goes wrong.
*/
private boolean _configurationParseArg(String arg) throws Exception {
if (arg.startsWith("-conf")) {
_expectingConfiguration = true;
} else if (arg.startsWith("-")) {
// If the argument names a directory in ptolemy/configs
// that contains a file named configuration.xml that can
// be found either as a URL or in the classpath, then
// assume that it is a configuration. For example, -ptiny
// will look for ptolemy/configs/ptiny/configuration.xml
// If the argument does not name a configuration, then
// we return false so that the argument can be processed
// by the parent class.
try {
_configurationSubdirectory = arg.substring(1);
String potentialConfiguration = _basePath + "/"
+ _configurationSubdirectory + "/configuration.xml";
// This will throw an Exception if we can't find the config.
_configurationURL = specToURL(potentialConfiguration);
} catch (Throwable ex) {
// The argument did not name a configuration, let the parent
// class have a shot.
return false;
}
} else if (_expectingConfiguration) {
_expectingConfiguration = false;
_configurationURL = specToURL(arg);
} else {
return false;
}
return true;
}
///////////////////////////////////////////////////////////////////
//// private methods ////
// Print out an error message and stack trace on stderr and then
// display a dialog box. This method is used as a fail safe
// in case there are problems with the configuration
// We use a Throwable here instead of an Exception because
// we might get an Error or and Exception. For example, if we
// are using JNI, then we might get a java.lang.UnsatisfiedLinkError,
// which is an Error, not and Exception.
private static void _errorAndExit(String message, String[] args,
Throwable throwable) {
StringBuffer argsBuffer = new StringBuffer("Command failed");
if (args.length > 0) {
argsBuffer.append("\nArguments: " + args[0]);
for (int i = 1; i < args.length; i++) {
argsBuffer.append(" " + args[i]);
}
argsBuffer.append("\n");
}
// First, print out the stack trace so that
// if the next step fails the user has
// a chance of seeing the message.
System.out.println(argsBuffer.toString());
throwable.printStackTrace();
// Display the error message in a stack trace
// If there are problems with the configuration,
// then there is a chance that we have not
// registered the GraphicalMessageHandler yet
// so we do so now so that we are sure
// the user can see the message.
// One way to test this is to run vergil -configuration foo
MessageHandler.setMessageHandler(new GraphicalMessageHandler());
MessageHandler.error(argsBuffer.toString(), throwable);
System.exit(0);
}
///////////////////////////////////////////////////////////////////
//// private variables ////
// The subdirectory (if any) of ptolemy/configs where the configuration
// may be found. For example if vergil was called with -ptiny,
// then this variable will be set to "ptiny", and the configuration
// should be at ptolemy/configs/ptiny/configuration.xml
private String _configurationSubdirectory;
// URL of the configuration to read.
// The URL may absolute, or relative to the Ptolemy II tree root.
// We use the URL instead of the string so that if the configuration
// is set as a command line argument, we can use the processed value
// from the command line instead of calling specToURL() again, which
// might be expensive.
private URL _configurationURL;
// Flag indicating that the previous argument was -conf
private boolean _expectingConfiguration = false;
}