/*
* GeoTools - The Open Source Java GIS Toolkit
* http://geotools.org
*
* (C) 2007-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;
// OpenGIS dependencies
import org.opengis.metadata.citation.Citation;
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;
// Geotools dependencies
import org.geotools.factory.Hints;
import org.geotools.util.GenericName;
import org.geotools.metadata.iso.citation.Citations;
import org.geotools.resources.i18n.ErrorKeys;
import org.geotools.resources.i18n.Errors;
/**
* Wraps {@linkplain AllAuthoritiesFactory all factories} in a {@code "http://www.opengis.net/"}
* name space. An exemple of complete URL is {@code "http://www.opengis.net/gml/srs/epsg.xml#4326"}.
* <p>
* Implementation note: this class requires some cooperation from the
* {@link AllAuthoritiesFactory#getSeparator} method, since the separator is not the usual
* {@value org.geotools.util.GenericName#DEFAULT_SEPARATOR} character.
*
* @since 2.4
* @source $URL$
* @version $Id$
* @author Martin Desruisseaux
*
* @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 HTTP_AuthorityFactory extends AuthorityFactoryAdapter implements CRSAuthorityFactory,
CSAuthorityFactory, DatumAuthorityFactory, CoordinateOperationAuthorityFactory
{
/**
* The base URL, which is {@value}.
*/
public static final String BASE_URL = "http://www.opengis.net/gml/srs/";
/**
* Creates a default wrapper.
*/
public HTTP_AuthorityFactory() {
this((Hints) null);
}
/**
* Creates a wrapper using the specified hints. For strict compliance with OGC
* definition of CRS defined by URL, 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 HTTP_AuthorityFactory(final Hints userHints) {
this(getFactory(userHints, "http"));
}
/**
* Creates a wrapper around the specified factory. The supplied factory is given unchanged
* to the {@linkplain AuthorityFactoryAdapter#AuthorityFactoryAdapter(AuthorityFactory)
* super class constructor}.
*/
public HTTP_AuthorityFactory(final AllAuthoritiesFactory factory) {
super(factory);
}
/**
* Returns {@code false} if {@link Hints#FORCE_LONGITUDE_FIRST_AXIS_ORDER} should be set to
* {@link Boolean#FALSE}. This method compares {@link Hints#FORCE_AXIS_ORDER_HONORING} with
* the specified authority.
*
* @param hints The hints to use (may be {@code null}).
* @param authority The authority factory under creation.
*
* @todo Should not looks at system hints; this is {@link ReferencingFactoryFinder}'s job.
*/
static boolean defaultAxisOrderHints(final Hints hints, final String authority) {
Object value = null;
if (hints != null) {
value = hints.get(Hints.FORCE_AXIS_ORDER_HONORING);
}
if (value == null) {
value = Hints.getSystemDefault(Hints.FORCE_AXIS_ORDER_HONORING);
}
if (value instanceof CharSequence) {
final String list = value.toString();
int i = 0;
while ((i = list.indexOf(authority, i)) >= 0) {
if (i==0 || !Character.isJavaIdentifierPart(list.charAt(i - 1))) {
final int j = i + authority.length();
if (j==list.length() || !Character.isJavaIdentifierPart(list.charAt(j))) {
// Found the authority in the list: we need to use the global setting.
return true;
}
}
i++;
}
}
return false;
}
/**
* Returns a factory from the specified hints. Used by {@link URN_AuthorityFactory}
* constructor as well.
*/
static AllAuthoritiesFactory getFactory(Hints hints, final String authority) {
if (!defaultAxisOrderHints(hints, authority)) {
hints = new Hints(hints);
hints.put(Hints.FORCE_LONGITUDE_FIRST_AXIS_ORDER, Boolean.FALSE);
}
return new AllAuthoritiesFactory(hints);
}
/**
* Returns the authority, which contains the {@code "http://www.opengis.net"} identifier.
*/
@Override
public Citation getAuthority() {
return Citations.HTTP_OGC;
}
/**
* Removes the URL base ({@value #BASE_URL}) 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(String code) throws FactoryException {
code = code.trim();
final int length = BASE_URL.length();
if (code.regionMatches(true, 0, BASE_URL, 0, length)) {
code = code.substring(length);
if (code.indexOf('/') < 0) {
final int split = code.indexOf('#');
if (split >= 0 && code.indexOf('#', split+1) < 0) {
String authority = code.substring(0, split).trim();
final int ext = authority.lastIndexOf('.');
if (ext > 0) {
// Removes the extension part (typically ".xml")
authority = authority.substring(0, ext);
}
code = code.substring(split + 1).trim();
code = authority + GenericName.DEFAULT_SEPARATOR + code;
return code;
}
}
}
throw new NoSuchAuthorityCodeException(Errors.format(ErrorKeys.ILLEGAL_ARGUMENT_$2,
"code", code), BASE_URL, code);
}
}