package com.limegroup.gnutella.gui;
import java.awt.IllegalComponentStateException;
import java.io.PrintWriter;
import java.io.StringWriter;
import com.limegroup.gnutella.ErrorService;
/**
* @author jum
*
* Implement a generic error handler that catches all errors thrown
* by ActionListeners in the AWT event dispatcher thread.
*/
public class DefaultErrorCatcher {
static void install() {
System.setProperty("sun.awt.exception.handler",
DefaultErrorCatcher.class.getName());
}
public void handle(Throwable ex) {
StringWriter sw = new StringWriter();
PrintWriter pw = new PrintWriter(sw);
ex.printStackTrace(pw);
pw.flush();
String bug = sw.toString();
if(!isIgnorable(ex, bug))
ErrorService.error(ex, "Uncaught event-thread error.");
else {
System.err.println("Ignoring error:");
ex.printStackTrace();
}
}
/**
* Determines if the message can be ignored.
*/
private boolean isIgnorable(Throwable bug, String msg) {
// ignore all overflows & out of memory errors,
// since they'll give us absolutely no debugging information
if(bug instanceof StackOverflowError)
return true;
if(bug instanceof OutOfMemoryError)
return true;
// no bug? kinda impossible, but shouldn't report.
if(msg == null)
return true;
// frickin' repaint manager stinks.
if(msg.indexOf("javax.swing.RepaintManager") != -1)
return true;
if(msg.indexOf("sun.awt.RepaintArea.paint") != -1)
return true;
// display manager on OSX goes out of whack
if(bug instanceof ArrayIndexOutOfBoundsException) {
if(msg.indexOf("apple.awt.CWindow.displayChanged") != -1)
return true;
if(msg.indexOf("javax.swing.plaf.basic.BasicTabbedPaneUI.getTabBounds") != -1)
return true;
}
// system clipboard can be held, preventing us from getting.
// throws a RuntimeException through stuff we don't control...
if(bug instanceof IllegalStateException) {
if(msg.indexOf("cannot open system clipboard") != -1)
return true;
}
// odd component exception
if(bug instanceof IllegalComponentStateException) {
if(msg.indexOf("component must be showing on the screen to determine its location") != -1)
return true;
}
// various NPEs we can ignore:
if(bug instanceof NullPointerException) {
if(msg.indexOf("MetalFileChooserUI") != -1)
return true;
if(msg.indexOf("WindowsFileChooserUI") != -1)
return true;
if(msg.indexOf("AquaDirectoryModel") != -1)
return true;
if(msg.indexOf("SizeRequirements.calculateAlignedPositions") != -1)
return true;
if(msg.indexOf("BasicTextUI.damageRange") != -1)
return true;
if(msg.indexOf("null pData") != -1)
return true;
if(msg.indexOf("disposed component") != -1)
return true;
}
// various InternalErrors we can ignore.
if(bug instanceof InternalError) {
if(msg.indexOf("getGraphics not implemented for this component") != -1)
return true;
}
// if we're not somewhere in the bug, ignore it.
// no need for us to debug sun's internal errors.
if(msg.indexOf("com.limegroup.gnutella") == -1)
return true;
// we intercept calls in various places -- check if the only
// com.limegroup.gnutella is from an intecepted call.
if(intercepts(msg, "com.limegroup.gnutella.tables.MouseEventConsumptionChecker"))
return true;
if(intercepts(msg, "com.limegroup.gnutella.gui.tables.LimeJTable.processMouseEvent"))
return true;
return false;
}
/**
* Determines if the given string is the only place where 'com.limegroup.gnutella' exists.
*/
private boolean intercepts(String msg, String inter) {
int i = msg.indexOf(inter);
// not intercepted at all?
if(i == -1)
return false;
// something before it?
if(msg.lastIndexOf("com.limegroup.gnutella", i) != -1)
return false;
i += inter.length();
if(i >= msg.length())
return false;
// something after it?
if(msg.indexOf("com.limegroup.gnutella", i) != -1)
return false;
// yup, it's the only com.limegroup.gnutella in there.
return true;
}
}