001/*
002 * ============================================================================
003 * Copyright © 2002-2023 by Thomas Thrien.
004 * All Rights Reserved.
005 * ============================================================================
006 *
007 * Licensed to the public under the agreements of the GNU Lesser General Public
008 * License, version 3.0 (the "License"). You may obtain a copy of the License at
009 *
010 *      http://www.gnu.org/licenses/lgpl.html
011 *
012 * Unless required by applicable law or agreed to in writing, software
013 * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
014 * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
015 * License for the specific language governing permissions and limitations
016 * under the License.
017 */
018
019package org.tquadrat.foundation.util.stringconverter;
020
021import static java.lang.String.format;
022import static org.apiguardian.api.API.Status.STABLE;
023import static org.tquadrat.foundation.lang.Objects.nonNull;
024
025import java.io.Serial;
026import java.time.Duration;
027import java.time.format.DateTimeParseException;
028
029import org.apiguardian.api.API;
030import org.tquadrat.foundation.annotation.ClassVersion;
031import org.tquadrat.foundation.lang.StringConverter;
032
033/**
034 *  An implementation of
035 *  {@link StringConverter}
036 *  for
037 *  {@link Duration}
038 *  values.<br>
039 *  <br>While the method
040 *  {@link StringConverter#toString(Object) toString(Duration)}
041 *  simply uses
042 *  {@link Duration#toString()},
043 *  the method
044 *  {@link #fromString(CharSequence)}
045 *  uses
046 *  {@link Duration#parse(CharSequence)}
047 *  to create the {@code Duration} instance for the given value. The formats
048 *  accepted are based on the ISO-8601 duration format {@code PnDTnHnMn.nS}
049 *  with days considered to be exactly 24 hours.<br>
050 *  <br>The string starts with an optional sign, denoted by the ASCII negative
051 *  ('-') or positive ('+') symbol. If negative, the whole period is
052 *  negated.<br>
053 *  <br>The ASCII letter &quot;P&quot; is next in upper or lower case. There
054 *  are then four sections, each consisting of a number and a suffix.<br>
055 *  <br>The sections have suffixes in ASCII of &quot;D&quot;, &quot;H&quot;,
056 *  &quot;M&quot; and &quot;S&quot; for days, hours, minutes and seconds,
057 *  accepted in upper or lower case.<br>
058 *  <br>The suffixes must occur in order. The ASCII letter &quot;T&quot; must
059 *  occur before the first occurrence, if any, of an hour, minute or second
060 *  section.<br>
061 *  <br>At least one of the four sections must be present, and if &quot;T&quot;
062 *  is present there must be at least one section after the &quot;T&quot;.<br>
063 *  <br>The number part of each section must consist of one or more ASCII
064 *  digits. The number may be prefixed by the ASCII negative or positive
065 *  symbol. The number of days, hours and minutes must parse to a {@code long}.
066 *  The number of seconds must parse to a {@code long} with optional fraction.
067 *  The decimal point may be either a dot or a comma. The fractional part may
068 *  have from zero to 9 digits.
069 *
070 *  @note The leading plus/minus sign, and negative values for other units are
071 *      not originally part of the ISO-8601 standard.
072 *
073 *  @see Duration
074 *
075 *  @extauthor Thomas Thrien - thomas.thrien@tquadrat.org
076 *  @version $Id: DurationStringConverter.java 1060 2023-09-24 19:21:40Z tquadrat $
077 *  @since 0.0.7
078 *
079 *  @UMLGraph.link
080 */
081@ClassVersion( sourceVersion = "$Id: DurationStringConverter.java 1060 2023-09-24 19:21:40Z tquadrat $" )
082@API( status = STABLE, since = "0.0.7" )
083public final class DurationStringConverter implements StringConverter<Duration>
084{
085        /*-----------*\
086    ====** Constants **========================================================
087        \*-----------*/
088    /**
089     *  The error message for an invalid duration String: {@value}.
090     */
091    public static final String MSG_InvalidDuration = "'%1$s' cannot be parsed as a valid duration";
092
093        /*------------------------*\
094    ====** Static Initialisations **===========================================
095        \*------------------------*/
096    /**
097     *  The serial version UID for objects of this class: {@value}.
098     *
099     *  @hidden
100     */
101    @Serial
102    private static final long serialVersionUID = 1L;
103
104    /**
105     *  An instance of this class.
106     */
107    public static final DurationStringConverter INSTANCE = new DurationStringConverter();
108
109        /*--------------*\
110    ====** Constructors **=====================================================
111        \*--------------*/
112    /**
113     *  Creates a new instance of {@code DurationStringConverter}.
114     */
115    public DurationStringConverter() {}
116
117        /*---------*\
118    ====** Methods **==========================================================
119        \*---------*/
120    /**
121     *  {@inheritDoc}
122     */
123    @Override
124    public final Duration fromString( final CharSequence source ) throws IllegalArgumentException
125    {
126        Duration retValue = null;
127        if( nonNull( source ) )
128        {
129            try
130            {
131                retValue = Duration.parse( source );
132            }
133            catch( final DateTimeParseException e )
134            {
135                throw new IllegalArgumentException( format( MSG_InvalidDuration, source ), e );
136            }
137        }
138
139        //---* Done *----------------------------------------------------------
140        return retValue;
141    }   //  fromString()
142
143    /**
144     *  This method is used by the
145     *  {@link java.util.ServiceLoader}
146     *  to obtain the instance for this
147     *  {@link org.tquadrat.foundation.lang.StringConverter}
148     *  implementation.
149     *
150     *  @return The instance for this {@code StringConverter} implementation.
151     */
152    public static final DurationStringConverter provider() { return INSTANCE; }
153}
154//  class DurationStringConverter
155
156/*
157 *  End of File
158 */