001/*
002 * ============================================================================
003 * Copyright © 2002-2020 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.lang;
019
020import static org.apiguardian.api.API.Status.STABLE;
021import static org.tquadrat.foundation.lang.Objects.isNull;
022import static org.tquadrat.foundation.lang.internal.StringConverterService.listInstances;
023import static org.tquadrat.foundation.lang.internal.StringConverterService.retrieveConverterForClass;
024import static org.tquadrat.foundation.lang.internal.StringConverterService.retrieveConverterForEnum;
025
026import java.io.Serializable;
027import java.util.Collection;
028import java.util.Optional;
029
030import org.apiguardian.api.API;
031import org.tquadrat.foundation.annotation.ClassVersion;
032
033/**
034 *  <p>{@summary Defines the conversion between Strings and object instances,
035 *  where the concrete behaviour is defined by the implementation of the
036 *  interface.}</p>
037 *  <p>Implementations of {@code StringConverter} are meant as a tool to
038 *  translate Strings to Objects for technical purposes, not necessarily for
039 *  human convenience.</p>
040 *  <p>Each implementation should ensure that</p>
041 *  <pre><code>  StringConverter&lt;T&gt; c = &hellip;
042 *  T v = &hellip;
043 *  true == ( v.equals( c.fromString( c.toString( v ) );</code></pre>
044 *  <p>is always valid.</p>
045 *  <p>If that condition could not be met, this must be clearly
046 *  documented.</p>
047 *  <p>For the other direction (from String to object back to String), this
048 *  condition is weaker, as long as the result is equivalent to the original,
049 *  as for numbers that can be represented as decimal, octal or
050 *  hexadecimal.</p>
051 *  <p>An implementation of {@code StringConverter} usually holds no state and
052 *  it is possible to reuse an instance. Therefore it is recommended to provide
053 *  a {@code public static final} field {@code INSTANCE} that is initialised
054 *  with a reference for a single instance of the implementation.
055 *
056 *  @note Both
057 *      {@code fromString(CharSequence)} and {@code toString(Object)} will
058 *      return {@code null} if called with a {@code null} argument.
059 *
060 *  @note Usually the {@code java.util.Locale.ROOT} locale is used when locale
061 *      specific conversion or parsing is required; this means in particular
062 *      that numbers will use the decimal point as separator.
063 *
064 *  @note In order to make an implementation of this interface visible for
065 *      {@code forClass()}, {@code forEnum()} and {@code list()}, that
066 *      implementation needs to be exposed as a service provider for
067 *      {@code org.tquadrat.foundation.lang.StringConverter}.
068 *
069 *  @note If the implementation is published as a service provider, and if it
070 *      has a static {@code INSTANCE} field holding the reference to an
071 *      instance of the converter, the implementation should also provide a
072 *      static final method {@code provider()} that returns that reference.
073 *
074 *  @note Two implementations of {@code StringConverter} for the same subject
075 *      class may cause unpredictable results if both are exposed as service
076 *      providers.
077 *
078 *  @note When an implementation of {@code StringConverter} is published as a
079 *      service, the subject class usually is guessed from the return value of
080 *      {@code fromString()}, but sometimes this does not work reliably. In
081 *      that case it is required that the implementation of
082 *      {@code StringConverter} provides a method
083 *      {@code public Collection<Class<?>> getSubjectClass()} that returns the
084 *      respective classes.<br>
085 *      This is definitely required, when the concrete implementation of the
086 *      string converter is derived from an abstract base class; this abstract
087 *      class must be visible for this module, too.
088 *
089 *  @extauthor Thomas Thrien - thomas.thrien@tquadrat.org
090 *  @version $Id: StringConverter.java 1107 2024-03-01 15:33:11Z tquadrat $
091 *  @since 0.1.0
092 *
093 *  @param  <T> The Object type for the conversion.
094 *
095 *  @UMLGraph.link
096 */
097@SuppressWarnings( "InterfaceMayBeAnnotatedFunctional" )
098@ClassVersion( sourceVersion = "$Id: StringConverter.java 1107 2024-03-01 15:33:11Z tquadrat $" )
099@API( status = STABLE, since = "0.1.0" )
100public interface StringConverter<T> extends Serializable
101{
102        /*-----------*\
103    ====** Constants **========================================================
104        \*-----------*/
105    /**
106     *  The name for the method that returns the subject classes: {@value}.
107     */
108    @SuppressWarnings( "StaticMethodOnlyUsedInOneClass" )
109    public static final String METHOD_NAME_GetSubjectClass = "getSubjectClass";
110
111    /**
112     *  The name for the method that returns the instance: {@value}.
113     */
114    @SuppressWarnings( "StaticMethodOnlyUsedInOneClass" )
115    public static final String METHOD_NAME_Provider = "provider";
116
117        /*---------*\
118    ====** Methods **==========================================================
119        \*---------*/
120    /**
121     *  Returns an instance of {@code StringConverter} for the given
122     *  {@link Class}.
123     *  If there is no converter for the given type, or the type is
124     *  {@code null}, the return value is
125     *  {@link Optional#empty()}.
126     *
127     *  @param  <C> The class a converter is needed for.
128     *  @param  type    The instance of the class a converter is needed for.
129     *  @return An instance of
130     *      {@link Optional}
131     *      that holds the instance of {@code StringConverter}.
132     */
133    @API( status = STABLE, since = "0.1.0" )
134    public static <C> Optional<StringConverter<C>> forClass( final Class<C> type )
135    {
136        return retrieveConverterForClass( type );
137    }   //  forClass()
138
139    /**
140     *  Returns an instance of {@code StringConverter} for the given
141     *  {@link Enum} type.
142     *  If there is no converter for the given type in the registry, a new
143     *  instance of {@code StringConverter} will be created, based on a
144     *  default implementation.
145     *
146     *  @param  <E> The class a converter is needed for.
147     *  @param  type    The instance of the class a converter is needed for.
148     *  @return The requested instance of {@code StringConverter}.
149     */
150    @API( status = STABLE, since = "0.0.6" )
151    public static <E extends Enum<E>> StringConverter<E> forEnum( final Class<E> type )
152    {
153        return retrieveConverterForEnum( type );
154    }   //  forEnum()
155
156    /**
157     *  Converts the given String to an object instance.
158     *
159     *  @param  source  The String representation for the object instance;
160     *      can be {@code null}.
161     *  @return The resulting object instance; will be {@code null} if
162     *      {@code source} was already {@code null}.
163     *  @throws IllegalArgumentException    The format of the given String is
164     *      invalid and cannot be parsed into the object instance.
165     */
166    public T fromString( final CharSequence source ) throws IllegalArgumentException;
167
168    /**
169     *  Converts the given object instance to a String.
170     *
171     *  @note Even if an implementation of
172     *      {@link org.tquadrat.foundation.lang.Stringer}
173     *      might exist for {@code T}, it cannot be used to implement this
174     *      method;
175     *      {@link org.tquadrat.foundation.lang.Stringer#toString(Object)}
176     *      will never return {@code null} (if implemented accordingly), for a
177     *      {@code null} argument, it will return the String &quot;null&quot;.
178     *      That contradicts the contract for this method.
179     *
180     *  @param  source  The object to convert; can be {@code null}.
181     *  @return The resulting String; will be {@code null} if {@code source}
182     *      was already {@code null}.
183     */
184    public default String toString( final T source )
185    {
186        final var retValue = isNull( source ) ? null : source.toString();
187
188        //---* Done *----------------------------------------------------------
189        return retValue;
190    }   //  toString()
191
192    /**
193     *  Returns the classes for that an instance of {@code StringConverter} is
194     *  already registered,
195     *
196     *  @return The classes with a string converter.
197     */
198    @API( status = STABLE, since = "0.1.0" )
199    public static Collection<Class<?>> list() { return listInstances(); }
200}
201//  interface StringConverter
202
203/*
204 *  End of File
205 */