/* Base class for displaying exceptions, warnings, and messages. Copyright (c) 2003-2014 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.util; import java.io.BufferedReader; import java.io.IOException; import java.io.InputStreamReader; import java.util.Locale; /////////////////////////////////////////////////////////////////// //// MessageHandler /** This is a class that is used to report errors. It provides a set of static methods that are called to report errors. However, the actual reporting of the errors is deferred to an instance of this class that is set using the setMessageHandler() method. Normally there is only one instance, set up by the application, so the class is a singleton. But this is not enforced.

This base class simply writes the errors to System.err. When an applet or application starts up, it may wish to set a subclass of this class as the message handler, to allow a nicer way of reporting errors. For example, a Swing application will probably want to report errors in a dialog box, using for example the derived class GraphicalMessageHandler. @see ptolemy.gui.GraphicalMessageHandler @author Edward A. Lee, Steve Neuendorffer, Elaine Cheong @version $Id: MessageHandler.java 70402 2014-10-23 00:52:20Z cxh $ @since Ptolemy II 4.0 @Pt.ProposedRating Green (cxh) @Pt.AcceptedRating Green (cxh) */ public class MessageHandler { /////////////////////////////////////////////////////////////////// //// public methods //// /** Defer to the set message handler to show the specified * error message. * *

Note that within Ptolemy, most user code should not call * this method directly. Instead, throw an exception, which will * be caught by the system elsewhere and include information * about what object caused the error. * @param info The message. */ public static void error(String info) { _handler._error(info); } /** Defer to the set message handler to * show the specified message and throwable information. * If the throwable is an instance of CancelException, then it * is not shown. By default, only the message of the throwable * is thrown. The stack trace information is only shown if the * user clicks on the "Display Stack Trace" button. * *

Note that within Ptolemy, most user code should not call * this method directly. Instead, throw an exception, which will * be caught by the system elsewhere and include information * about what object caused the error. * @param info The message. * @param throwable The throwable. * @see CancelException */ public static void error(String info, Throwable throwable) { // Sometimes you find that errors are reported multiple times. // To find out who is calling this method, uncomment the following. // System.out.println("------ reporting error:" + throwable); // throwable.printStackTrace(); // System.out.println("------ called from:"); // (new Exception()).printStackTrace(); try { _handler._error(info, throwable); } catch (Throwable throwable2) { // An applet was throwing an exception while handling // the error - so we print the original message if _error() fails. if (_handler instanceof SimpleMessageHandler) { throw new RuntimeException(throwable); } else { System.err.println("Internal Error, exception thrown while " + "handling error: \"" + info + "\"\n"); throwable.printStackTrace(); System.err.println("Internal Error:\n"); throwable2.printStackTrace(); } } } /** Return the message handler instance that is used by the static * methods in this class. * @return The message handler. * @see #setMessageHandler(MessageHandler) */ public static MessageHandler getMessageHandler() { return _handler; } /** If the nightly build is running, then return true. * *

This method merely checks to see if the * "ptolemy.ptII.isRunningNightlyBuild" property exists and is not empty * or if the "ptolemy.ptII.batchMode" property exists and is not empty * and the property "ptolemyII.ptII.testingMessageHandler" is not set. * *

To run the test suite in the Nightly Build mode, use

*
     *  make nightly
     *  
* @return True if the nightly build is running. */ public static boolean isRunningNightlyBuild() { if ((StringUtilities.getProperty("ptolemy.ptII.isRunningNightlyBuild") .length() > 0 || StringUtilities.getProperty( "ptolemy.ptII.batchMode").length() > 0) && StringUtilities.getProperty( "ptolemy.ptII.testingMessageHandler").length() == 0) { return true; } return false; } /** Defer to the set message handler to show the specified * message. * @param info The message. */ public static void message(String info) { _handler._message(info); } /** Set the message handler instance that is used by the static * methods in this class. If the given handler is null, then * do nothing. * @param handler The message handler. * @see #getMessageHandler() */ public static void setMessageHandler(MessageHandler handler) { if (handler != null) { _handler = handler; } } /** Return a short description of the throwable. * @param throwable The throwable * @return If the throwable is an Exception, return "Exception", * if it is an Error, return "Error", if it is a Throwable, return * "Throwable". */ public static String shortDescription(Throwable throwable) { String throwableType = null; if (throwable instanceof Exception) { throwableType = "Exception"; } else if (throwable instanceof Error) { throwableType = "Error"; } else { throwableType = "Throwable"; } return throwableType; } /** Defer to the set message handler to * show the specified message in a modal dialog. If the user * clicks on the "Cancel" button, then throw an exception. * This gives the user the option of not continuing the * execution, something that is particularly useful if continuing * execution will result in repeated warnings. * *

Note that within Ptolemy, most user code should not call * this method directly. Instead, throw an exception, which will * be caught by the system elsewhere and include information * about what object caused the warning. * * @param info The message. * @exception CancelException If the user clicks on the "Cancel" button. */ public static void warning(String info) throws CancelException { _handler._warning(info); } /** Show the specified message and throwable information * in a modal dialog. If the user * clicks on the "Cancel" button, then throw an exception. * This gives the user the option of not continuing the * execution, something that is particularly useful if continuing * execution will result in repeated warnings. * By default, only the message of the throwable * is thrown. The stack trace information is only shown if the * user clicks on the "Display Stack Trace" button. * *

Note that within Ptolemy, most user code should not call * this method directly. Instead, throw an exception, which will * be caught by the system elsewhere and include information * about what object caused the warning. * * @param info The message. * @param throwable The throwable associated with this warning. * @exception CancelException If the user clicks on the "Cancel" button. */ public static void warning(String info, Throwable throwable) throws CancelException { _handler._warning(info + ": " + throwable.getMessage(), throwable); } /** Ask the user a yes/no question, and return true if the answer * is yes. This method returns true without asking the user if * the property "ptolemy.ptII.isRunningNightlyBuild" is set. * In the regression tests, there is no user to answer the question. * @param question The yes/no question. * @return True if the answer is yes. */ public static boolean yesNoQuestion(String question) { if (!isRunningNightlyBuild()) { return _handler._yesNoQuestion(question); } else { return true; } } /** Ask the user a yes/no/cancel question, and return true if the * answer is yes. If the user clicks on the "Cancel" button, * then throw an exception. * * @param question The yes/no/cancel question. * @return True if the answer is yes. * @exception ptolemy.util.CancelException If the user clicks on * the "Cancel" button. */ public static boolean yesNoCancelQuestion(String question) throws ptolemy.util.CancelException { return yesNoCancelQuestion(question, "Yes", "No", "Cancel"); } /** Ask the user a question with three possible answers; * return true if the answer is the first one and false if * the answer is the second one; throw an exception if the * user selects the third one. * * @param question The question. * @param trueOption The option for which to return true. * @param falseOption The option for which to return false. * @param exceptionOption The option for which to throw an exception. * @return True if the answer is the first option, false if it is the second. * @exception ptolemy.util.CancelException If the user selects the third option. */ public static boolean yesNoCancelQuestion(String question, String trueOption, String falseOption, String exceptionOption) throws ptolemy.util.CancelException { return _handler._yesNoCancelQuestion(question, trueOption, falseOption, exceptionOption); } /////////////////////////////////////////////////////////////////// //// protected methods //// /** Show the specified error message. * @param info The message. */ protected void _error(String info) { System.err.println(info); } /** Show the specified message and throwable information. * If the throwable is an instance of CancelException, then nothing * is not shown. By default, only the message of the exception * is thrown. The stack trace information is only shown if the * user clicks on the "Display Stack Trace" button. * * @param info The message. * @param throwable The throwable. * @see CancelException */ protected void _error(String info, Throwable throwable) { if (throwable instanceof CancelException) { return; } System.err.println(info); throwable.printStackTrace(); } /** Display the warning message. In this base class, the * the default handler merely prints the warning to stderr. * @param info The message. */ protected void _message(String info) { System.err.println(info); } /** Show the specified message. In this base class, the message * is printed to standard error. *

Derived classes might show the specified message in a modal * dialog. If the user clicks on the "Cancel" button, then throw * an exception. This gives the user the option of not * continuing the execution, something that is particularly * useful if continuing execution will result in repeated * warnings. * @param info The message. * @exception CancelException If the user clicks on the "Cancel" button. */ protected void _warning(String info) throws CancelException { _error(info); } /** Display the warning message and throwable information. In * this base class, the the default handler merely prints the * warning to stderr. If the user clicks on the "Cancel" button, * then throw an exception. This gives the user the option of * not continuing the execution, something that is particularly * useful if continuing execution will result in repeated * warnings. By default, only the message of the throwable is * thrown. The stack trace information is only shown if the user * clicks on the "Display Stack Trace" button. * @param info The message. * @param throwable The Throwable. * @exception CancelException If the user clicks on the "Cancel" button. */ protected void _warning(String info, Throwable throwable) throws CancelException { _error(info, throwable); } /** Ask the user a yes/no question, and return true if the answer * is yes. In this base class, this prints the question on standard * output and looks for the reply on standard input. * @param question The yes/no question to be asked. * @return True if the answer is yes. */ protected boolean _yesNoQuestion(String question) { System.out.print(question); System.out.print(" (yes or no) "); BufferedReader stdIn = new BufferedReader(new InputStreamReader( System.in)); try { String reply = stdIn.readLine(); if (reply == null) { return false; } else if (reply.trim().toLowerCase(Locale.getDefault()) .equals("yes")) { return true; } } catch (IOException ex) { } return false; } /** Ask the user a question with three possible answers; * return true if the answer is the first one and false if * the answer is the second one; throw an exception if the * user selects the third one. * @param question The question. * @param trueOption The option for which to return true. * @param falseOption The option for which to return false. * @param exceptionOption The option for which to throw an exception. * @return True if the answer is the first option, false if it is the second. * @exception ptolemy.util.CancelException If the user selects the third option. */ protected boolean _yesNoCancelQuestion(String question, String trueOption, String falseOption, String exceptionOption) throws ptolemy.util.CancelException { System.out.print(question + " (" + trueOption + " or " + falseOption + " or " + exceptionOption + ") "); BufferedReader stdIn = new BufferedReader(new InputStreamReader( System.in)); try { String reply = stdIn.readLine(); if (reply == null) { return false; } else { if (reply.trim().toLowerCase(Locale.getDefault()) .equals(trueOption.toLowerCase(Locale.getDefault()))) { return true; } else if (reply .trim() .toLowerCase(Locale.getDefault()) .equals(exceptionOption.toLowerCase(Locale.getDefault()))) { throw new ptolemy.util.CancelException("Cancelled: " + question); } } } catch (IOException ex) { } return false; } /////////////////////////////////////////////////////////////////// //// private variables //// // The message handler. private static MessageHandler _handler = new MessageHandler(); }