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 "P" 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 "D", "H", 056 * "M" and "S" 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 "T" 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 "T" 062 * is present there must be at least one section after the "T".<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 */