package com.apigee.sdk.apm.android.crashlogging.internal; import java.io.BufferedWriter; import java.io.FileWriter; import java.io.PrintWriter; import java.io.StringWriter; import java.io.Writer; import java.lang.Thread.UncaughtExceptionHandler; import java.util.Date; import java.util.UUID; import android.util.Log; import com.apigee.sdk.apm.android.crashlogging.Constants; import com.apigee.sdk.apm.android.crashlogging.CrashManagerListener; import com.apigee.sdk.apm.android.model.ClientLog; /** * <h4>Description</h4> * * Internal helper class to catch exceptions. Saves the stack trace * as a file and executes callback methods to ask the app for * additional information and meta data (see CrashManagerListener). * * <h4>License</h4> * * <pre> * Copyright (c) 2009 nullwire aps * Copyright (c) 2011,2012 Codenauts UG * * Permission is hereby granted, free of charge, to any person * obtaining a copy of this software and associated documentation * files (the "Software"), to deal in the Software without * restriction, including without limitation the rights to use, * copy, modify, merge, publish, distribute, sublicense, and/or sell * copies of the Software, and to permit persons to whom the * Software is furnished to do so, subject to the following * conditions: * * The above copyright notice and this permission notice shall be * included in all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES * OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT * HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, * WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR * OTHER DEALINGS IN THE SOFTWARE. * </pre> * * @author Mads Kristiansen * @author Glen Humphrey * @author Evan Charlton * @author Peter Hewitt * @author Thomas Dohmke **/ public class ExceptionHandler implements UncaughtExceptionHandler { private boolean ignoreDefaultHandler = false; private CrashManagerListener listener; private UncaughtExceptionHandler defaultExceptionHandler; public ExceptionHandler(UncaughtExceptionHandler defaultExceptionHandler, CrashManagerListener listener, boolean ignoreDefaultHandler) { this.defaultExceptionHandler = defaultExceptionHandler; this.ignoreDefaultHandler = ignoreDefaultHandler; this.listener = listener; } public static void saveException(Throwable exception, CrashManagerListener listener) { final Date now = new Date(); final Writer result = new StringWriter(); final PrintWriter printWriter = new PrintWriter(result); exception.printStackTrace(printWriter); try { // Create filename from a random uuid String filename = UUID.randomUUID().toString(); String path = Constants.FILES_PATH + "/" + filename + ".stacktrace"; Log.d(ClientLog.TAG_MONITORING_CLIENT, "Writing unhandled exception to: " + path); StringBuilder sb = new StringBuilder(); sb.append("Package: " + Constants.APP_PACKAGE + "\n"); sb.append("Version: " + Constants.APP_VERSION + "\n"); sb.append("Android: " + Constants.ANDROID_VERSION + "\n"); sb.append("Manufacturer: " + Constants.PHONE_MANUFACTURER + "\n"); sb.append("Model: " + Constants.PHONE_MODEL + "\n"); sb.append("Date: " + now + "\n"); sb.append("\n"); sb.append(result.toString()); String exceptionReport = sb.toString(); //Uncomment following line to see crash report contents logged //Log.d(ClientLog.TAG_MONITORING_CLIENT, exceptionReport); // Write the stacktrace to disk BufferedWriter write = new BufferedWriter(new FileWriter(path)); write.write(exceptionReport); write.flush(); write.close(); if (listener != null) { writeValueToFile(limitedString(listener.getUserID()), filename + ".user"); writeValueToFile(limitedString(listener.getContact()), filename + ".contact"); writeValueToFile(listener.getDescription(), filename + ".description"); } } catch (Exception another) { Log.e(ClientLog.TAG_MONITORING_CLIENT, "Error saving exception stacktrace!\n", another); } } public void uncaughtException(Thread thread, Throwable exception) { saveException(exception, listener); if (!ignoreDefaultHandler) { defaultExceptionHandler.uncaughtException(thread, exception); } else { android.os.Process.killProcess(android.os.Process.myPid()); System.exit(10); } } private static void writeValueToFile(String value, String filename) { try { String path = Constants.FILES_PATH + "/" + filename; if (value.trim().length() > 0) { BufferedWriter writer = new BufferedWriter(new FileWriter(path)); writer.write(value); writer.flush(); writer.close(); } } catch (Exception e) { } } private static String limitedString(String string) { if ((string != null) && (string.length() > 255)) { string = string.substring(0, 255); } return string; } }