001/* 002 * Licensed to the Apache Software Foundation (ASF) under one or more 003 * contributor license agreements. See the NOTICE file distributed with 004 * this work for additional information regarding copyright ownership. 005 * The ASF licenses this file to You under the Apache License, Version 2.0 006 * (the "License"); you may not use this file except in compliance with 007 * the License. You may obtain a copy of the License at 008 * 009 * http://www.apache.org/licenses/LICENSE-2.0 010 * 011 * Unless required by applicable law or agreed to in writing, software 012 * distributed under the License is distributed on an "AS IS" BASIS, 013 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 014 * See the License for the specific language governing permissions and 015 * limitations under the License. 016 */ 017package org.apache.commons.lang3.exception; 018 019import java.util.List; 020import java.util.Set; 021 022import org.apache.commons.lang3.tuple.Pair; 023 024/** 025 * <p> 026 * An exception that provides an easy and safe way to add contextual information. 027 * </p><p> 028 * An exception trace itself is often insufficient to provide rapid diagnosis of the issue. 029 * Frequently what is needed is a select few pieces of local contextual data. 030 * Providing this data is tricky however, due to concerns over formatting and nulls. 031 * </p><p> 032 * The contexted exception approach allows the exception to be created together with a 033 * list of context label-value pairs. This additional information is automatically included in 034 * the message and printed stack trace. 035 * </p><p> 036 * An unchecked version of this exception is provided by ContextedRuntimeException. 037 * </p> 038 * <p> 039 * To use this class write code as follows: 040 * </p> 041 * <pre> 042 * try { 043 * ... 044 * } catch (Exception e) { 045 * throw new ContextedException("Error posting account transaction", e) 046 * .addContextValue("Account Number", accountNumber) 047 * .addContextValue("Amount Posted", amountPosted) 048 * .addContextValue("Previous Balance", previousBalance) 049 * } 050 * } 051 * </pre> 052 * <p> 053 * or improve diagnose data at a higher level: 054 * </p> 055 * <pre> 056 * try { 057 * ... 058 * } catch (ContextedException e) { 059 * throw e.setContextValue("Transaction Id", transactionId); 060 * } catch (Exception e) { 061 * if (e instanceof ExceptionContext) { 062 * e.setContextValue("Transaction Id", transactionId); 063 * } 064 * throw e; 065 * } 066 * } 067 * </pre> 068 * <p> 069 * The output in a printStacktrace() (which often is written to a log) would look something like the following: 070 * </p> 071 * <pre> 072 * org.apache.commons.lang3.exception.ContextedException: java.lang.Exception: Error posting account transaction 073 * Exception Context: 074 * [1:Account Number=null] 075 * [2:Amount Posted=100.00] 076 * [3:Previous Balance=-2.17] 077 * [4:Transaction Id=94ef1d15-d443-46c4-822b-637f26244899] 078 * 079 * --------------------------------- 080 * at org.apache.commons.lang3.exception.ContextedExceptionTest.testAddValue(ContextedExceptionTest.java:88) 081 * ..... (rest of trace) 082 * </pre> 083 * 084 * @see ContextedRuntimeException 085 * @since 3.0 086 */ 087public class ContextedException extends Exception implements ExceptionContext { 088 089 /** The serialization version. */ 090 private static final long serialVersionUID = 20110706L; 091 /** The context where the data is stored. */ 092 private final ExceptionContext exceptionContext; 093 094 /** 095 * Instantiates ContextedException without message or cause. 096 * <p> 097 * The context information is stored using a default implementation. 098 */ 099 public ContextedException() { 100 super(); 101 exceptionContext = new DefaultExceptionContext(); 102 } 103 104 /** 105 * Instantiates ContextedException with message, but without cause. 106 * <p> 107 * The context information is stored using a default implementation. 108 * 109 * @param message the exception message, may be null 110 */ 111 public ContextedException(final String message) { 112 super(message); 113 exceptionContext = new DefaultExceptionContext(); 114 } 115 116 /** 117 * Instantiates ContextedException with cause, but without message. 118 * <p> 119 * The context information is stored using a default implementation. 120 * 121 * @param cause the underlying cause of the exception, may be null 122 */ 123 public ContextedException(final Throwable cause) { 124 super(cause); 125 exceptionContext = new DefaultExceptionContext(); 126 } 127 128 /** 129 * Instantiates ContextedException with cause and message. 130 * <p> 131 * The context information is stored using a default implementation. 132 * 133 * @param message the exception message, may be null 134 * @param cause the underlying cause of the exception, may be null 135 */ 136 public ContextedException(final String message, final Throwable cause) { 137 super(message, cause); 138 exceptionContext = new DefaultExceptionContext(); 139 } 140 141 /** 142 * Instantiates ContextedException with cause, message, and ExceptionContext. 143 * 144 * @param message the exception message, may be null 145 * @param cause the underlying cause of the exception, may be null 146 * @param context the context used to store the additional information, null uses default implementation 147 */ 148 public ContextedException(final String message, final Throwable cause, ExceptionContext context) { 149 super(message, cause); 150 if (context == null) { 151 context = new DefaultExceptionContext(); 152 } 153 exceptionContext = context; 154 } 155 156 //----------------------------------------------------------------------- 157 /** 158 * Adds information helpful to a developer in diagnosing and correcting the problem. 159 * For the information to be meaningful, the value passed should have a reasonable 160 * toString() implementation. 161 * Different values can be added with the same label multiple times. 162 * <p> 163 * Note: This exception is only serializable if the object added is serializable. 164 * </p> 165 * 166 * @param label a textual label associated with information, {@code null} not recommended 167 * @param value information needed to understand exception, may be {@code null} 168 * @return {@code this}, for method chaining, not {@code null} 169 */ 170 @Override 171 public ContextedException addContextValue(final String label, final Object value) { 172 exceptionContext.addContextValue(label, value); 173 return this; 174 } 175 176 /** 177 * Sets information helpful to a developer in diagnosing and correcting the problem. 178 * For the information to be meaningful, the value passed should have a reasonable 179 * toString() implementation. 180 * Any existing values with the same labels are removed before the new one is added. 181 * <p> 182 * Note: This exception is only serializable if the object added as value is serializable. 183 * </p> 184 * 185 * @param label a textual label associated with information, {@code null} not recommended 186 * @param value information needed to understand exception, may be {@code null} 187 * @return {@code this}, for method chaining, not {@code null} 188 */ 189 @Override 190 public ContextedException setContextValue(final String label, final Object value) { 191 exceptionContext.setContextValue(label, value); 192 return this; 193 } 194 195 /** 196 * {@inheritDoc} 197 */ 198 @Override 199 public List<Object> getContextValues(final String label) { 200 return this.exceptionContext.getContextValues(label); 201 } 202 203 /** 204 * {@inheritDoc} 205 */ 206 @Override 207 public Object getFirstContextValue(final String label) { 208 return this.exceptionContext.getFirstContextValue(label); 209 } 210 211 /** 212 * {@inheritDoc} 213 */ 214 @Override 215 public List<Pair<String, Object>> getContextEntries() { 216 return this.exceptionContext.getContextEntries(); 217 } 218 219 /** 220 * {@inheritDoc} 221 */ 222 @Override 223 public Set<String> getContextLabels() { 224 return exceptionContext.getContextLabels(); 225 } 226 227 /** 228 * Provides the message explaining the exception, including the contextual data. 229 * 230 * @see java.lang.Throwable#getMessage() 231 * @return the message, never null 232 */ 233 @Override 234 public String getMessage() { 235 return getFormattedExceptionMessage(super.getMessage()); 236 } 237 238 /** 239 * Provides the message explaining the exception without the contextual data. 240 * 241 * @see java.lang.Throwable#getMessage() 242 * @return the message 243 * @since 3.0.1 244 */ 245 public String getRawMessage() { 246 return super.getMessage(); 247 } 248 249 /** 250 * {@inheritDoc} 251 */ 252 @Override 253 public String getFormattedExceptionMessage(final String baseMessage) { 254 return exceptionContext.getFormattedExceptionMessage(baseMessage); 255 } 256}