/*
* GeoTools - The Open Source Java GIS Toolkit
* http://geotools.org
*
* (C) 2006-2008, Open Source Geospatial Foundation (OSGeo)
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation;
* version 2.1 of the License.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*/
package org.geotools.referencing.factory;
import java.util.Arrays;
import java.util.List;
import java.util.SortedMap;
import java.util.TreeMap;
import org.opengis.metadata.citation.Citation;
import org.opengis.referencing.AuthorityFactory;
import org.opengis.referencing.FactoryException;
import org.opengis.referencing.NoSuchAuthorityCodeException;
import org.opengis.referencing.crs.CRSAuthorityFactory;
import org.opengis.referencing.cs.CSAuthorityFactory;
import org.opengis.referencing.datum.DatumAuthorityFactory;
import org.opengis.referencing.operation.CoordinateOperationAuthorityFactory;
import org.geotools.util.Version;
import org.geotools.factory.Hints;
import org.geotools.metadata.iso.citation.Citations;
/**
* Wraps {@linkplain AllAuthoritiesFactory all factories} in a {@code "urn:ogc:def"}
* name space. An exemple of complete URN is {@code "urn:ogc:def:crs:EPSG:6.8:4326"}.
* <p>
* Users don't need to create an instance of this class, since one is automatically
* registered for use in {@link org.opengis.referencing.ReferencingFactoryFinder}.
*
* @since 2.4
* @source $URL$
* @version $Id$
* @author Justin Deoliveira
* @author Martin Desruisseaux
*
* @see <A HREF="https://portal.opengeospatial.org/files/?artifact_id=8814">URNs of definitions
* in OGC namespace</A>
*
* @deprecated This class will move in a <code>org.geotools.referencing.factory.<strong>web</strong></code>
* package in Geotools 2.5, in order to put together other web-related factories.
* Don't use this class directly. You should not need to anyway - use
* {@link org.geotools.referencing.ReferencingFactoryFinder} instead, which will
* continue to work no matter where this class is located.
*/
public class URN_AuthorityFactory extends AuthorityFactoryAdapter implements CRSAuthorityFactory,
CSAuthorityFactory, DatumAuthorityFactory, CoordinateOperationAuthorityFactory
{
/**
* The backing factory. Will be used as a fallback if no object
* is available for some specific version of an EPSG database.
*/
private final AllAuthoritiesFactory factory;
/**
* The authority factories by versions. Factories will be created by
* {@link #createVersionedFactory} when first needed.
*/
private final SortedMap<Version, AuthorityFactory> byVersions =
new TreeMap<Version, AuthorityFactory>();
/**
* The last code processed, or {@code null} if none.
*/
private transient URN_Parser last;
/**
* Creates a default wrapper.
*/
public URN_AuthorityFactory() {
this((Hints) null);
}
/**
* Creates a wrapper using the specified hints. For strict compliance with OGC definition
* of {@code "urn:ogc:def"} namespace, the supplied hints should contains at least the
* {@link Hints#FORCE_LONGITUDE_FIRST_AXIS_ORDER FORCE_LONGITUDE_FIRST_AXIS_ORDER} hint
* with value {@link Boolean#FALSE FALSE}.
*
* @param userHints The hints to be given to backing factories.
*/
public URN_AuthorityFactory(final Hints userHints) {
this(HTTP_AuthorityFactory.getFactory(userHints, "urn"));
}
/**
* Creates a wrapper around the specified factory. The supplied factory is given unchanged
* to the {@linkplain AuthorityFactoryAdapter#AuthorityFactoryAdapter(AuthorityFactory)
* super class constructor}.
*/
public URN_AuthorityFactory(final AllAuthoritiesFactory factory) {
super(factory);
this.factory = factory;
}
/**
* Returns the authority, which contains {@code "urn:ogc:def"} and {@code "urn:x-ogc:def"}
* identifiers.
*/
@Override
public Citation getAuthority() {
return Citations.URN_OGC;
}
/**
* Parses the specified code. For performance reason, returns the last result if applicable.
*
* @param code The URN to parse.
* @return parser The parser.
* @throws NoSuchAuthorityCodeException if the URN syntax is invalid.
*/
private URN_Parser getParser(final String code) throws NoSuchAuthorityCodeException {
/*
* Take a local copy of the field in order to protect against changes.
* This avoid the need for synchronization (URN_Parsers are immutable,
* so it doesn't matter if the 'last' reference is changed concurrently).
*/
URN_Parser parser = last;
if (parser == null || !parser.urn.equals(code)) {
last = parser = new URN_Parser(code);
}
return parser;
}
/**
* Returns an object factory for the specified code. This method invokes one of the
* <code>get</code><var>Type</var><code>AuthorityFactory</code> methods where
* <var>Type</var> is inferred from the code.
*
* @param code The authority code given to this class.
* @return A factory for the specified authority code (never {@code null}).
* @throws FactoryException if no suitable factory were found.
*/
@Override
protected AuthorityFactory getAuthorityFactory(final String code) throws FactoryException {
if (code != null) {
return getAuthorityFactory(getParser(code).type.type.asSubclass(AuthorityFactory.class), code);
} else {
return super.getAuthorityFactory(code);
}
}
/**
* Returns the datum factory to use for the specified URN. If the URN contains a version
* string, then this method will try to fetch a factory for that particular version. The
* {@link #createVersionedFactory} method may be invoked for that purpose. If no factory
* is provided for that specific version, then the
* {@linkplain AuthorityFactoryAdapter#getDatumAuthorityFactory default one} is used.
*
* @param code The URN given to this class.
* @return A factory for the specified URN (never {@code null}).
* @throws FactoryException if no datum factory is available.
*/
@Override
protected DatumAuthorityFactory getDatumAuthorityFactory(final String code)
throws FactoryException
{
if (code != null) {
final URN_Parser parser = getParser(code);
parser.logWarningIfTypeMismatch(DatumAuthorityFactory.class);
final AuthorityFactory factory = getVersionedFactory(parser);
if (factory instanceof DatumAuthorityFactory) {
return (DatumAuthorityFactory) factory;
}
}
return super.getDatumAuthorityFactory(code);
}
/**
* Returns the coordinate system factory to use for the specified URN. If the URN contains a
* version string, then this method will try to fetch a factory for that particular version.
* The {@link #createVersionedFactory} method may be invoked for that purpose. If no factory
* is provided for that specific version, then the
* {@linkplain AuthorityFactoryAdapter#getCSAuthorityFactory default one} is used.
*
* @param code The URN given to this class.
* @return A factory for the specified URN (never {@code null}).
* @throws FactoryException if no coordinate system factory is available.
*/
@Override
protected CSAuthorityFactory getCSAuthorityFactory(final String code)
throws FactoryException
{
if (code != null) {
final URN_Parser parser = getParser(code);
parser.logWarningIfTypeMismatch(CSAuthorityFactory.class);
final AuthorityFactory factory = getVersionedFactory(parser);
if (factory instanceof CSAuthorityFactory) {
return (CSAuthorityFactory) factory;
}
}
return super.getCSAuthorityFactory(code);
}
/**
* Returns the coordinate reference system factory to use for the specified URN.
* If the URN contains a version string, then this method will try to fetch a factory
* for that particular version. The {@link #createVersionedFactory} method may be
* invoked for that purpose. If no factory is provided for that specific version, then
* the {@linkplain AuthorityFactoryAdapter#getCRSAuthorityFactory default one} is used.
*
* @param code The URN given to this class.
* @return A factory for the specified URN (never {@code null}).
* @throws FactoryException if no coordinate reference system factory is available.
*/
@Override
protected CRSAuthorityFactory getCRSAuthorityFactory(final String code)
throws FactoryException
{
if (code != null) {
final URN_Parser parser = getParser(code);
parser.logWarningIfTypeMismatch(CRSAuthorityFactory.class);
final AuthorityFactory factory = getVersionedFactory(parser);
if (factory instanceof CRSAuthorityFactory) {
return (CRSAuthorityFactory) factory;
}
}
return super.getCRSAuthorityFactory(code);
}
/**
* Returns the coordinate operation factory to use for the specified URN. If the URN
* contains a version string, then this method will try to fetch a factory for that
* particular version. The {@link #createVersionedFactory} method may be invoked for
* that purpose. If no factory is provided for that specific version, then the
* {@linkplain AuthorityFactoryAdapter#getCoordinateOperationAuthorityFactory default one}
* is used.
*
* @param code The URN given to this class.
* @return A factory for the specified URN (never {@code null}).
* @throws FactoryException if no coordinate operation factory is available.
*/
@Override
protected CoordinateOperationAuthorityFactory getCoordinateOperationAuthorityFactory(final String code)
throws FactoryException
{
if (code != null) {
final URN_Parser parser = getParser(code);
parser.logWarningIfTypeMismatch(CoordinateOperationAuthorityFactory.class);
final AuthorityFactory factory = getVersionedFactory(parser);
if (factory instanceof CoordinateOperationAuthorityFactory) {
return (CoordinateOperationAuthorityFactory) factory;
}
}
return super.getCoordinateOperationAuthorityFactory(code);
}
/**
* Returns an authority factory for the specified version, or {@code null} if none.
* This method invokes {@link #createVersionedFactory} the first time it is invoked
* for a given version and cache the factory.
*
* @throws FactoryException if an error occured while creating the factory.
*/
private AuthorityFactory getVersionedFactory(final URN_Parser parser)
throws FactoryException
{
final Version version = parser.version;
if (version == null) {
return null;
}
AuthorityFactory factory;
synchronized (byVersions) {
factory = byVersions.get(version);
if (factory == null) {
factory = createVersionedFactory(version);
if (factory != null) {
byVersions.put(version, factory);
}
}
}
return factory;
}
/**
* Invoked when a factory is requested for a specific version. This method should create
* a factory for the exact version specified by the argument, or return {@code null} if
* no such factory is available. In the later case, this class will fallback on the factory
* specified at {@linkplain #URN_AuthorityFactory(AuthorityFactory, String, Citation)
* construction time}.
*
* @param version The version for the factory to create.
* @return The factory, of {@code null} if there is none for the specified version.
* @throws FactoryException if an error occured while creating the factory.
*/
protected AuthorityFactory createVersionedFactory(final Version version)
throws FactoryException
{
final Hints hints = new Hints(factory.getImplementationHints());
hints.put(Hints.VERSION, version);
final List<AuthorityFactory> factories = Arrays.asList(new AuthorityFactory[] {
new AllAuthoritiesFactory(hints),
factory
});
return FallbackAuthorityFactory.create(factories);
}
/**
* Removes the URN base ({@code "urn:ogc:def"}) from the specified code
* before to pass it to the wrapped factories.
*
* @param code The code given to this factory.
* @return The code to give to the underlying factories.
* @throws FactoryException if the code can't be converted.
*/
@Override
protected String toBackingFactoryCode(final String code) throws FactoryException {
return getParser(code).getAuthorityCode();
}
}