001/* 002 * ============================================================================ 003 * Copyright © 2002-2023 by Thomas Thrien. 004 * All Rights Reserved. 005 * ============================================================================ 006 * Licensed to the public under the agreements of the GNU Lesser General Public 007 * License, version 3.0 (the "License"). You may obtain a copy of the License at 008 * 009 * http://www.gnu.org/licenses/lgpl.html 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, WITHOUT 013 * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the 014 * License for the specific language governing permissions and limitations 015 * under the License. 016 */ 017 018package org.tquadrat.foundation.exception; 019 020import static java.lang.String.format; 021import static org.apiguardian.api.API.Status.STABLE; 022import static org.tquadrat.foundation.lang.Objects.nonNull; 023import static org.tquadrat.foundation.lang.Objects.requireNonNullArgument; 024 025import java.io.Serial; 026 027import org.apiguardian.api.API; 028import org.tquadrat.foundation.annotation.ClassVersion; 029 030/** 031 * This implementation of 032 * {@link Error} 033 * should be thrown in all cases where an unexpected exception was caught. 034 * Unlike an 035 * {@link ImpossibleExceptionError}, 036 * the exception is <i>possible</i>, but not expected – probably because the 037 * programmer has taken (at least thought to have …) appropriate 038 * precautions against its occurrence. An example for this could be an 039 * implementation of 040 * {@link Object#clone() clone()} 041 * for a particular class: with implementing that method (and declaring the 042 * interface 043 * {@link Cloneable} 044 * …), instances of that class do support cloning and therefore that 045 * exception should not be thrown. Unfortunately, changes somewhere else in 046 * the source might prevent one of the attributes of the respective class from 047 * being cloned, so that exception could be thrown again …<br> 048 * <br>Another example could be this code sequence:<pre><code> … 049 * File file = new File( … ); 050 * if( file.exists() ) 051 * { 052 * try( var inputStream = new FileInputStream( file ) ) 053 * { 054 * … 055 * } 056 * catch( FileNotFoundException e ) 057 * { 058 * throw new UnexpectedExceptionError( e ); 059 * } 060 * }</code></pre> 061 * 062 * @extauthor Thomas Thrien - thomas.thrien@tquadrat.org 063 * @version $Id: UnexpectedExceptionError.java 1060 2023-09-24 19:21:40Z tquadrat $ 064 * @since 0.0.5 065 * 066 * @UMLGraph.link 067 */ 068@ClassVersion( sourceVersion = "$Id: UnexpectedExceptionError.java 1060 2023-09-24 19:21:40Z tquadrat $" ) 069@API( status = STABLE, since = "0.0.5" ) 070public sealed class UnexpectedExceptionError extends AssertionError 071 permits ImpossibleExceptionError 072{ 073 /*-----------*\ 074 ====** Constants **======================================================== 075 \*-----------*/ 076 /** 077 * The message for this exception. 078 */ 079 private static final String MSG_UnexpectedException = "The Exception '%1$s' was not expected to occur in this context"; 080 081 /*------------------------*\ 082 ====** Static Initialisations **=========================================== 083 \*------------------------*/ 084 /** 085 * The serial version UID for objects of this class: {@value}. 086 * 087 * @hidden 088 */ 089 @Serial 090 private static final long serialVersionUID = 2233464685783981770L; 091 092 /*--------------*\ 093 ====** Constructors **===================================================== 094 \*--------------*/ 095 /** 096 * Creates a new instance of the Error. The message is a constant, only 097 * the causing exception can be given. 098 * 099 * @param cause The causing exception. 100 */ 101 public UnexpectedExceptionError( final Throwable cause ) 102 { 103 super( format( MSG_UnexpectedException, requireNonNullArgument( cause, "cause" ).getClass().getName() ) , cause ); 104 } // UnexpectedExceptionError() 105 106 /** 107 * Creates a new instance of the Error. This constructor should be used in 108 * cases where an enhanced message is useful or necessary. 109 * 110 * @param message The message for the error. 111 * @param cause The causing exception. 112 */ 113 public UnexpectedExceptionError( final String message, final Throwable cause ) 114 { 115 super( nonNull( message ) && !message.isEmpty() ? message : format( MSG_UnexpectedException, requireNonNullArgument( cause, "cause" ).getClass().getName() ), requireNonNullArgument( cause, "cause" ) ); 116 } // UnexpectedExceptionError() 117} 118// class UnexpectedExceptionError 119 120/* 121 * End of File 122 */