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 &hellip;) 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 *  &hellip;), 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 &hellip;<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 */