blob: f6b3b1b752327c99ad7dfa00585920326734368d [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one or more contributor license
* agreements. See the NOTICE file distributed with this work for additional information regarding
* copyright ownership. The ASF licenses this file to You under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance with the License. You may obtain a
* copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software distributed under the License
* is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express
* or implied. See the License for the specific language governing permissions and limitations under
* the License.
*/
package org.apache.geode.management.internal.cli.util;
/**
* Helper class to navigate through the nested causes of an exception. Provides method to
* <ul>
* <li>find index of an Exception of a specific type
* <li>find Root Cause of the Exception
* <li>retrieve a nested cause at a specific index/depth
* <li>find a cause by specific type
* </ul>
*
* @since GemFire 7.0
*/
public class CauseFinder {
private static final int START_INDEX = 0;
/**
* Returns the index of a first nested cause which is of the given Throwable type or its sub-type
* depending on <code>isSubtypeOk</code> value.
* <p />
* NOTE: It looks for the "nested" causes & doesn't try to match the <code>causeClass</code> of
* the <code>parent</code>.
* <p />
*
* Returns -1 if a cause with the given Throwable type is not found.
*
* @param parent parent Throwable instance
* @param causeClass type of the nested Throwable cause
* @param isSubtypeOk whether any matching sub-type is required or exact match is required
* @return index/depth of the cause starting from the 'top most' Throwable in the stack. -1 if
* can't find it.
*/
public static int indexOfCause(Throwable parent, Class<? extends Throwable> causeClass,
final boolean isSubtypeOk) {
return indexOfCause(parent, causeClass, START_INDEX, isSubtypeOk);
}
/**
* Returns the index of a first nested cause which is of the given Throwable type or its sub-type
* depending on <code>isSubtypeOk</code> value.
* <p />
* NOTE: It looks for the "nested" causes & doesn't try to match the <code>causeClass</code> of
* the <code>parent</code>.
* <p />
*
* Returns -1 if a cause with the given Throwable type is not found.
*
* @param parent parent Throwable instance
* @param causeClass type of the nested Throwable cause
* @param cindex variable to store current index/depth of the cause
* @param isSubtypeOk whether any matching sub-type is required or exact match is required
* @return index/depth of the cause starting from the 'top most' Throwable in the stack. -1 if
* can't find it.
*/
private static int indexOfCause(Throwable parent, Class<? extends Throwable> causeClass,
final int cindex, final boolean isSubtypeOk) {
int resultIndex = cindex;
Throwable cause = parent.getCause();
// (cause is not null & cause is not of type causeClass)
if (cause != null && !isMatching(cause, causeClass, isSubtypeOk)) {
// recurse deeper
resultIndex = indexOfCause(cause, causeClass, cindex + 1, isSubtypeOk);
} else if (cause != null && isMatching(cause, causeClass, isSubtypeOk)) {
resultIndex = cindex != -1 ? cindex + 1 : cindex;
} else if (cause == null) {
resultIndex = -1;
}
return resultIndex;
}
/**
* Returns whether the given <code>cause</code> is assignable or has same type as that of
* <code>causeClass</code> depending on <code>isSubtypeOk</code> value.
*
* @param cause parent Throwable instance
* @param causeClass type of the nested Throwable cause
* @param isSubtypeOk whether any matching sub-type is required or exact match is required
* @return true if <code>cause</code> is assignable or has same type as that of
* <code>causeClass</code>. false otherwise
*/
// Note: No not null check on 'Throwable cause' - currently called after null check
private static boolean isMatching(Throwable cause, Class<? extends Throwable> causeClass,
final boolean isSubtypeOk) {
return isSubtypeOk ? causeClass.isInstance(cause) : causeClass == cause.getClass();
}
/**
* Returns nested 'root' cause of the given parent Throwable. Will return the same Throwable if it
* has no 'cause'.
*
* @param parent Throwable whose root cause is to be found
* @return nested root cause.
* @throws IllegalArgumentException when parent is <code>null</code>
*/
public static Throwable getRootCause(Throwable parent) {
return getRootCause(parent, 0);
}
/**
* Returns nested 'root' cause of the given parent Throwable. Will return the same Throwable if it
* has no 'cause'. If <code>depth</code> is 0, does a <code>null</code> check on the given
* <code>parent Throwable</code> & if <code>parent</code> is <code>null</code>, throws
* {@link IllegalArgumentException}.
*
* @param parent Throwable whose root cause is to be found
* @param depth recurse depth indicator
* @return nested root cause.
* @throws IllegalArgumentException when parent is <code>null</code> at <code>depth = 0</code>
*/
// Note: private method
private static Throwable getRootCause(Throwable parent, int depth) {
if (depth == 0) {
if (parent == null) {
throw new IllegalArgumentException("Given parent Throwable is null.");
}
}
Throwable theCause = parent.getCause();
if (theCause != null) {
// recurse deeper
return getRootCause(theCause, depth + 1);
}
return parent;
}
/**
* Returns cause at the specified depth/index starting from the 'top most' Throwable
* <code>parent</code> in the stack. Returns <code>null</code> if it can't find it.
*
* @param parent Throwable to use to find the cause at given index/depth
* @param requiredIndex index/depth of nesting the cause
* @return cause at the specified index starting from the 'top most' Throwable in the stack.
* <code>null</code> if it can't find it.
*/
public static Throwable causeAt(Throwable parent, final int requiredIndex) {
//
// TODO:
// NEED TO CHECK WHETHER RIGHT PARENT CAUSE RETRUNED
//
if (parent != null && requiredIndex > 0) {
// recurse deeper
return causeAt(parent.getCause(), requiredIndex - 1);
}
// when there isn't required depth
if (requiredIndex > 0) {
return null;
}
return parent;
}
/**
* Returns the first occurrence of nested cause of <code>parent</code> which matches the specified
* <code>causeType</code> or its sub-type depending on <code>isSubtypeOk</code> value.
*
* @param parent Throwable to use to find the cause of given type
* @param causeType type of the nested Throwable cause
* @param isSubtypeOk whether any matching sub-type is required or exact match is required
* @return the first occurrence of nested cause which matches the specified <code>causeType</code>
* or its sub-type. <code>null</code> if it can't find it.
*/
public static Throwable causeByType(Throwable parent, Class<? extends Throwable> causeType,
boolean isSubtypeOk) {
Throwable cause = null;
int foundAtIndex = indexOfCause(parent, causeType, isSubtypeOk);
if (foundAtIndex != -1) {
cause = causeAt(parent, foundAtIndex);
}
return cause;
}
}