org.geotoolkit.io.wkt.ReferencingParser Maven / Gradle / Ivy
/*
* Geotoolkit.org - An Open Source Java GIS Toolkit
* http://www.geotoolkit.org
*
* (C) 2002-2012, Open Source Geospatial Foundation (OSGeo)
* (C) 2009-2012, Geomatys
*
* 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.geotoolkit.io.wkt;
import java.util.Map;
import java.util.List;
import java.util.Locale;
import java.util.HashMap;
import java.util.ArrayList;
import java.text.ParseException;
import java.text.ParsePosition;
import javax.measure.unit.Unit;
import javax.measure.quantity.Angle;
import javax.measure.quantity.Length;
import javax.measure.quantity.Quantity;
import org.opengis.metadata.citation.Citation;
import org.opengis.parameter.ParameterValue;
import org.opengis.parameter.ParameterValueGroup;
import org.opengis.parameter.ParameterDescriptor;
import org.opengis.parameter.ParameterNotFoundException;
import org.opengis.referencing.IdentifiedObject;
import org.opengis.util.FactoryException;
import org.opengis.util.NoSuchIdentifierException;
// While start import is usually a deprecated practice, we use such a large amount
// of interfaces in those packages that it we choose to exceptionnaly use * here.
import org.opengis.referencing.cs.*;
import org.opengis.referencing.crs.*;
import org.opengis.referencing.datum.*;
import org.opengis.referencing.operation.*;
import org.geotoolkit.factory.Hints;
import org.geotoolkit.factory.FactoryFinder;
import org.geotoolkit.metadata.iso.citation.Citations;
import org.geotoolkit.referencing.NamedIdentifier;
import org.geotoolkit.referencing.datum.BursaWolfParameters;
import org.geotoolkit.referencing.cs.AbstractCS;
import org.geotoolkit.referencing.cs.DefaultCoordinateSystemAxis;
import org.geotoolkit.referencing.operation.DefiningConversion;
import org.geotoolkit.referencing.factory.ReferencingFactoryContainer;
import org.geotoolkit.measure.Units;
import org.geotoolkit.resources.Errors;
import static java.util.Collections.singletonMap;
import static javax.measure.unit.SI.METRE;
import static javax.measure.unit.SI.RADIAN;
import static javax.measure.unit.NonSI.DEGREE_ANGLE;
import static org.geotoolkit.util.ArgumentChecks.ensureNonNull;
import static org.geotoolkit.util.collection.XCollections.hashMapCapacity;
import static org.geotoolkit.referencing.datum.DefaultGeodeticDatum.WGS84;
import static org.geotoolkit.referencing.datum.DefaultPrimeMeridian.GREENWICH;
import static org.geotoolkit.referencing.datum.DefaultGeodeticDatum.BURSA_WOLF_KEY;
import static org.geotoolkit.referencing.datum.DefaultVerticalDatum.getVerticalDatumTypeFromLegacyCode;
/**
* Well
* Known Text (WKT) parser for referencing objects. This include, but is not limited too,
* {@linkplain CoordinateReferenceSystem Coordinate Reference System} and {@linkplain MathTransform
* Math Transform} objects. Note that math transforms are part of the WKT {@code "FITTED_CS"} element.
*
* {@section Default axis names}
* The default axis names differ depending on whatever the parsing shall be strictly compliant to
* the legacy WKT specification, or whatever ISO 19111 identifiers shall be used instead. The
* following table compares the names:
*
*
* CRS type WKT defaults ISO abbreviations
* Geographic Lon, Lat λ, φ
* Vertical H h
* Projected X, Y x, y
* Geocentric X, Y, Z X, Y, Z
*
*
* The default behavior is to use the legacy WKT identifiers, for compliance with the WKT
* specification. This behavior can be changed by call to {@link #setISOConform(boolean)}.
* Note that Geotk referencing factories like
* {@link org.geotoolkit.referencing.factory.wkt.WKTParsingAuthorityFactory} perform the
* above-cited {@code setISOConform(true)} method call on their internal parser instance,
* for ISO compliance.
*
* @author Rémi Eve (IRD)
* @author Martin Desruisseaux (IRD, Geomatys)
* @version 3.20
*
* @see Well Know Text specification
* @see OGC WKT Coordinate System Issues
*
* @since 2.0
* @level advanced
* @module
*/
public class ReferencingParser extends MathTransformParser {
/**
* {@code true} in order to allows the non-standard Oracle syntax. Oracle puts the Bursa-Wolf
* parameters straight into the {@code DATUM} elements, without enclosing them in a
* {@code TOWGS84} element.
*/
private static final boolean ALLOW_ORACLE_SYNTAX = true;
/**
* {@code true} if the authority declared in the {@code AUTHORITY[, ]} element
* should be assigned to the name. A previous Geotk version assigned the authority to the name.
* However experience show that it is often wrong in practice, since peoples declare EPSG codes
* but still use WKT name much shorter than the EPSG name (for example "WGS84"
* instead than "World Geodetic System 1984"). Even our own Geotk implementation
* make such substitution through the {@link Formatter#getName(IdentifiedObject)} method.
*
* @since 3.19
*/
private static final boolean ASSIGN_AUTHORITY_TO_NAME = false;
/**
* The factory to use for creating {@linkplain Datum datum}.
*/
private final DatumFactory datumFactory;
/**
* The factory to use for creating {@linkplain CoordinateSystem coordinate systems}.
*/
private final CSFactory csFactory;
/**
* The factory to use for creating {@linkplain CoordinateReferenceSystem
* coordinate reference systems}.
*/
private final CRSFactory crsFactory;
/**
* If non-null, forces {@code PRIMEM} and {@code PARAMETER} angular units to the given value
* instead than inferring it from the context. This field is occasionally set to
* {@code NonSI.DEGREE_ANGLE} for compatibility with ESRI softwares.
*
* Note that this value does not apply to {@code AXIS} elements.
*
* @since 3.20
*/
private Unit forcedAngularUnit;
/**
* {@code true} if ISO axis identifiers should be used instead than the one defined
* by the WKT specification.
*
* @since 3.18
*/
private boolean isoConform;
/**
* {@code true} if {@code AXIS[...]} elements should be ignored. This is sometime used
* for simulating a "force longitude first axis order" behavior. It is also used for
* compatibility with ESRI softwares which ignore axis elements.
*/
private boolean axisIgnored;
/**
* The list of {@linkplain AxisDirection axis directions} from their name.
* Instantiated at construction time and never modified after that point.
*/
private final Map directions;
/**
* The value of the last {@linkplain #directions} map created.
* We keep this reference only on the assumption that the same
* map will often be reused.
*/
private static Map lastDirections;
/**
* Creates a parser using the default set of symbols and factories.
*/
public ReferencingParser() {
this(Symbols.DEFAULT, (Hints) null);
}
/**
* Creates a parser using the specified set of symbols.
* Default factories are fetching according the given hints.
*
* @param symbols The symbols for parsing and formatting numbers.
* @param hints The hints to be used for fetching the factories, or
* {@code null} for the system-wide default hints.
*/
public ReferencingParser(final Symbols symbols, final Hints hints) {
this(symbols,
FactoryFinder.getDatumFactory(hints),
FactoryFinder.getCSFactory(hints),
FactoryFinder.getCRSFactory(hints),
FactoryFinder.getMathTransformFactory(hints));
}
/**
* Constructs a parser for the specified set of symbols using the specified set of factories.
*
* @param symbols The symbols for parsing and formatting numbers.
* @param factories The factories to use.
*/
public ReferencingParser(final Symbols symbols, final ReferencingFactoryContainer factories) {
this(symbols,
factories.getDatumFactory(),
factories.getCSFactory(),
factories.getCRSFactory(),
factories.getMathTransformFactory());
}
/**
* Constructs a parser for the specified set of symbols using the specified set of factories.
*
* @param symbols The symbols for parsing and formatting numbers.
* @param datumFactory The factory to use for creating {@linkplain Datum datum}.
* @param csFactory The factory to use for creating {@linkplain CoordinateSystem coordinate systems}.
* @param crsFactory The factory to use for creating {@linkplain CoordinateReferenceSystem coordinate reference systems}.
* @param mtFactory The factory to use for creating {@linkplain MathTransform math transform} objects.
*/
public ReferencingParser(final Symbols symbols,
final DatumFactory datumFactory,
final CSFactory csFactory,
final CRSFactory crsFactory,
final MathTransformFactory mtFactory)
{
super(symbols, mtFactory);
this.datumFactory = datumFactory;
this.csFactory = csFactory;
this.crsFactory = crsFactory;
ensureNonNull("datumFactory", datumFactory);
ensureNonNull("csFactory", csFactory);
ensureNonNull("crsFactory", crsFactory);
/*
* Gets the map of axis directions.
*/
final AxisDirection[] values = AxisDirection.values();
Map directions =
new HashMap(hashMapCapacity(values.length));
final Locale locale = symbols.locale;
for (int i=0; i existing = lastDirections;
if (directions.equals(existing)) {
directions = existing;
} else {
lastDirections = directions;
}
}
this.directions = directions;
}
/**
* If non-null, forces {@code PRIMEM} and {@code PARAMETER} angular units to the returned
* value instead than inferring it from the context. The default value is {@code null},
* which mean that the angular units are inferred from the context as required by the
* WKT specification.
*
* @return The angular unit, or {@code null} for inferring it from the context.
*
* @since 3.20
*/
public Unit getForcedAngularUnit() {
return forcedAngularUnit;
}
/**
* If non-null, forces {@code PRIMEM} and {@code PARAMETER} angular units to the given
* value instead than inferring it from the context. This property is occasionally set
* to {@link javax.measure.unit.NonSI#DEGREE_ANGLE} for compatibility with ESRI and GDAL
* softwares. Note that this value does not apply to {@code AXIS} elements.
*
* @param angularUnit The new angular unit, or {@code null} for restoring the default behavior.
*
* @see Convention#ESRI
* @see Convention#PROJ4
*
* @since 3.20
*/
public void setForcedAngularUnit(final Unit angularUnit) {
forcedAngularUnit = angularUnit;
}
/**
* Returns {@code true} if the default names of {@code AXIS[...]} elements shall be ISO 19111
* identifiers. The default value is {@code false}, which mean that the identifiers specified
* by the WKT specification are used.
*
* @return {@code true} if the default identifiers of {@code AXIS[...]} elements shall be
* conform to ISO 19111.
*
* @since 3.18
*/
public boolean isISOConform() {
return isoConform;
}
/**
* Sets whatever the default names of {@code AXIS[...]} elements shall be ISO identifiers.
*
* @param conform {@code true} if the default identifiers of {@code AXIS[...]} elements shall
* be conform to ISO 19111.
*
* @since 3.18
*/
public void setISOConform(final boolean conform) {
isoConform = conform;
}
/**
* Returns {@code true} if {@code AXIS[...]} elements will be ignored during parsing.
* The default value is {@code false}.
*
* @return {@code true} if {@code AXIS[...]} elements will be ignored during parsing.
*
* @since 3.00
*/
public boolean isAxisIgnored() {
return axisIgnored;
}
/**
* Sets whatever {@code AXIS[...]} elements will be ignored during parsing. The default
* value is {@code false} as we would expect from a WKT compliant parser. However this
* flag may occasionally be set to {@code true} for compatibility with ESRI softwares,
* which ignore {@code AXIS} elements. It may also be used as a way to force the longitude
* axis to be first.
*
* Note that {@code AXIS} elements still need to be well formed even when this flag is set
* to {@code true}; invalid axis will continue to cause a {@link ParseException} despite
* their content being ignored.
*
* @param ignored {@code true} if {@code AXIS[...]} elements should be ignored during parsing.
*
* @since 3.00
*/
public void setAxisIgnored(final boolean ignored) {
axisIgnored = ignored;
}
/**
* Parses a coordinate reference system element.
*
* @param text The text to be parsed.
* @return The coordinate reference system.
* @throws ParseException if the string can't be parsed.
*/
public final CoordinateReferenceSystem parseCoordinateReferenceSystem(final String text)
throws ParseException
{
final Element element = getTree(text, new ParsePosition(0));
final CoordinateReferenceSystem crs = parseCoordinateReferenceSystem(element);
element.close();
return crs;
}
/**
* Parses a coordinate reference system element.
*
* @param parent The parent element.
* @return The next element as a {@link CoordinateReferenceSystem} object.
* @throws ParseException if the next element can't be parsed.
*/
private CoordinateReferenceSystem parseCoordinateReferenceSystem(final Element element)
throws ParseException
{
final Object key = element.peek();
if (key instanceof Element) {
final String keyword = keyword((Element) key);
switch (keyword.hashCode()) {
/*
* Note: the following cases are copied in the parseObject(Element) method in
* order to take advantage of a single switch statement. If new cases are added
* here, then they must be added in parseObject(Element) as well.
*/
case 2098816550: if ( "GEOGCS".equals(keyword)) return parseGeoGCS (element); break;
case -1926479731: if ( "PROJCS".equals(keyword)) return parseProjCS (element); break;
case 2098812706: if ( "GEOCCS".equals(keyword)) return parseGeoCCS (element); break;
case 1069641278: if ( "VERT_CS".equals(keyword)) return parseVertCS (element); break;
case -1611514396: if ( "LOCAL_CS".equals(keyword)) return parseLocalCS (element); break;
case 182967770: if ( "COMPD_CS".equals(keyword)) return parseCompdCS (element); break;
case 414930797: if ("FITTED_CS".equals(keyword)) return parseFittedCS(element); break;
}
}
throw element.parseFailed(null, Errors.format(Errors.Keys.UNKNOWN_TYPE_$1, key));
}
/**
* Parses the next element in the specified Well Know Text (WKT) tree.
*
* @param element The element to be parsed.
* @return The object.
* @throws ParseException if the element can't be parsed.
*/
@Override
Object parse(final Element element) throws ParseException {
final Object key = element.peek();
if (key instanceof Element) {
final String keyword = keyword((Element) key);
switch (keyword.hashCode()) {
case 2023329: if ( "AXIS".equals(keyword)) return parseAxis (element, METRE, true); break;
case -1926655538: if ( "PRIMEM".equals(keyword)) return parsePrimem (element, DEGREE_ANGLE); break;
case -414856156: if ( "TOWGS84".equals(keyword)) return parseToWGS84 (element); break;
case -1262236878: if ( "SPHEROID".equals(keyword)) return parseSpheroid (element); break;
case 1321414593: if ( "VERT_DATUM".equals(keyword)) return parseVertDatum (element); break;
case 519534171: if ("LOCAL_DATUM".equals(keyword)) return parseLocalDatum(element); break;
case 64819279: if ( "DATUM".equals(keyword)) return parseDatum (element, GREENWICH); break;
/*
* Note: the following cases are copied from parseCoordinateReferenceSystem(Element)
* method in order to take advantage of a single switch statement. If new cases are
* added here, then they must be added in the above method first.
*/
case 2098816550: if ( "GEOGCS".equals(keyword)) return parseGeoGCS (element); break;
case -1926479731: if ( "PROJCS".equals(keyword)) return parseProjCS (element); break;
case 2098812706: if ( "GEOCCS".equals(keyword)) return parseGeoCCS (element); break;
case 1069641278: if ( "VERT_CS".equals(keyword)) return parseVertCS (element); break;
case -1611514396: if ( "LOCAL_CS".equals(keyword)) return parseLocalCS (element); break;
case 182967770: if ( "COMPD_CS".equals(keyword)) return parseCompdCS (element); break;
case 414930797: if ("FITTED_CS".equals(keyword)) return parseFittedCS(element); break;
/*
* Note: the following cases are copied from MathTransformParser in order to take
* advantage of a single switch statement. If new cases are added there, then the
* superclass must be updated first.
*/
case 1954077369: if ( "PARAM_MT".equals(keyword)) return parseParamMT (element); break;
case 1889286834: if ( "CONCAT_MT".equals(keyword)) return parseConcatMT (element); break;
case -1910641354: if ( "INVERSE_MT".equals(keyword)) return parseInverseMT (element); break;
case -219294638: if ("PASSTHROUGH_MT".equals(keyword)) return parsePassThroughMT(element); break;
}
}
throw element.parseFailed(null, Errors.format(Errors.Keys.UNKNOWN_TYPE_$1, key));
}
/**
* Returns the properties to be given to the parsed object. This method is invoked
* automatically by the parser for the root element only. This method expect on input
* the properties parsed from the {@code AUTHORITY} element, and returns on output the
* properties to give to the object to be created. The default implementation returns
* the {@code properties} map unchanged. Subclasses may override this method in order
* to add or change properties.
*
* Example: if a subclass want to add automatically an authority code when no
* {@code AUTHORITY} element was explicitly set in the WKT, then it may test for the
* {@link IdentifiedObject#IDENTIFIERS_KEY} key and add automatically an entry if this
* key was missing.
*
* @param properties The properties parsed from the WKT file. Entries can be added, removed
* or modified directly in this map.
* @return The properties to be given to the parsed object. This is usually {@code properties}
* (maybe after modifications), but could also be a new map.
*
* @since 2.3
*/
protected Map alterProperties(final Map properties) {
return properties;
}
/**
* Parses an optional {@code "AUTHORITY"} element.
* This element has the following pattern:
*
* {@preformat text
* AUTHORITY["", ""]
* }
*
* @param parent The parent element.
* @param name The name of the parent object being parsed.
* @return A properties map with the parent name and the optional authority code.
* @throws ParseException if the {@code "AUTHORITY"} can't be parsed.
*/
private Map parseAuthority(final Element parent, final String name)
throws ParseException
{
final boolean isRoot = parent.isRoot();
final Element element = parent.pullOptionalElement("AUTHORITY");
if (element == null && !isRoot) {
return singletonMap(IdentifiedObject.NAME_KEY, (Object) name);
}
Map properties = new HashMap(4);
properties.put(IdentifiedObject.NAME_KEY, name);
if (element != null) {
final String auth = element.pullString("name");
final String code = element.pullObject("code").toString(); // Accepts Integer as well as String.
element.close();
final Citation authority = Citations.fromName(auth);
if (ASSIGN_AUTHORITY_TO_NAME) {
properties.put(IdentifiedObject.NAME_KEY, new NamedIdentifier(authority, name));
}
properties.put(IdentifiedObject.IDENTIFIERS_KEY, new NamedIdentifier(authority, code));
}
if (isRoot) {
properties = alterProperties(properties);
}
return properties;
}
/**
* Parses a {@code "UNIT"} element.
* This element has the following pattern:
*
* {@preformat text
* UNIT["", {,}]
* }
*
* @param parent The parent element.
* @param unit The contextual unit. Usually {@link javax.measure.unit.SI#METRE} or
* {@link javax.measure.unit.SI#RADIAN}.
* @return The {@code "UNIT"} element as an {@link Unit} object.
* @throws ParseException if the {@code "UNIT"} can't be parsed.
*
* @todo Authority code is currently ignored. We may consider to create a subclass of
* {@link Unit} which implements {@link IdentifiedObject} in a future version.
*/
private Unit parseUnit(final Element parent, final Unit unit)
throws ParseException
{
final Element element = parent.pullElement("UNIT");
final String name = element.pullString("name");
final double factor = element.pullDouble("factor");
final Map properties = parseAuthority(element, name); // NOSONAR: Ignored for now.
element.close();
return Units.multiply(unit, factor);
}
/**
* Parses an {@code "AXIS"} element.
* This element has the following pattern:
*
* {@preformat text
* AXIS["", NORTH | SOUTH | EAST | WEST | UP | DOWN | OTHER]
* }
*
* {@note There is no AUTHORITY element for AXIS element in OGC specification. However, we
* accept it anyway in order to make the parser more tolerant to non-100% compliant
* WKT. Note that AXIS is really the only element without such AUTHORITY clause and
* the EPSG database provides authority code for all axis.}
*
* @param parent The parent element.
* @param unit The contextual unit. Usually {@link javax.measure.unit.NonSI#DEGREE_ANGLE}
* or {@link javax.measure.unit.SI#METRE}.
* @param required {@code true} if the axis is mandatory,
* or {@code false} if it is optional.
* @return The {@code "AXIS"} element as a {@link CoordinateSystemAxis} object, or {@code null}
* if the axis was not required and there is no axis object.
* @throws ParseException if the {@code "AXIS"} element can't be parsed.
*/
private CoordinateSystemAxis parseAxis(final Element parent,
final Unit> unit,
final boolean required)
throws ParseException
{
final Element element;
if (required) {
element = parent.pullElement("AXIS");
} else {
element = parent.pullOptionalElement("AXIS");
if (element == null) {
return null;
}
}
final String name = element.pullString("name");
final Element orientation = element.pullVoidElement("orientation");
final Map properties = parseAuthority(element, name); // See javadoc
element.close();
final AxisDirection direction = directions.get(keyword(orientation));
if (direction == null) {
throw element.parseFailed(null, Errors.format(Errors.Keys.UNKNOWN_TYPE_$1, orientation));
}
try {
return createAxis(properties, name, direction, unit);
} catch (FactoryException exception) {
throw element.parseFailed(exception, null);
}
}
/**
* Creates an axis. If the name matches one of pre-defined axis, the pre-defined one
* will be returned. This replacement help to get more success when comparing a CS
* built from WKT against a CS built from one of Geotk constants.
*
* @param properties Name and other properties to give to the new object.
* If {@code null}, the abbreviation will be used as the axis name.
* @param abbreviation The coordinate axis abbreviation.
* @param direction The axis direction.
* @param unit The coordinate axis unit.
* @throws FactoryException if the axis can't be created.
*/
private CoordinateSystemAxis createAxis(Map properties,
final String abbreviation,
final AxisDirection direction,
final Unit> unit)
throws FactoryException
{
final CoordinateSystemAxis candidate =
DefaultCoordinateSystemAxis.getPredefined(abbreviation, direction);
if (candidate != null && unit.equals(candidate.getUnit())) {
return candidate;
}
if (properties == null) {
properties = singletonMap(IdentifiedObject.NAME_KEY, abbreviation);
}
return csFactory.createCoordinateSystemAxis(properties, abbreviation, direction, unit);
}
/**
* Parses a {@code "PRIMEM"} element. This element has the following pattern:
*
* {@preformat text
* PRIMEM["", {,}]
* }
*
* @param parent The parent element.
* @param angularUnit The contextual unit.
* @return The {@code "PRIMEM"} element as a {@link PrimeMeridian} object.
* @throws ParseException if the {@code "PRIMEM"} element can't be parsed.
*/
private PrimeMeridian parsePrimem(final Element parent, Unit angularUnit)
throws ParseException
{
if (forcedAngularUnit != null) {
angularUnit = forcedAngularUnit;
}
final Element element = parent.pullElement("PRIMEM");
final String name = element.pullString("name");
final double longitude = element.pullDouble("longitude");
final Map properties = parseAuthority(element, name);
element.close();
try {
return datumFactory.createPrimeMeridian(properties, longitude, angularUnit);
} catch (FactoryException exception) {
throw element.parseFailed(exception, null);
}
}
/**
* Parses an optional {@code "TOWGS84"} element.
* This element has the following pattern:
*
* {@preformat text
* TOWGS84[, , , , , , ]
* }
*
* @param parent The parent element.
* @return The {@code "TOWGS84"} element as a {@link BursaWolfParameters} object,
* or {@code null} if no {@code "TOWGS84"} has been found.
* @throws ParseException if the {@code "TOWGS84"} can't be parsed.
*/
private static BursaWolfParameters parseToWGS84(final Element parent)
throws ParseException
{
final Element element = parent.pullOptionalElement("TOWGS84");
if (element == null) {
return null;
}
final BursaWolfParameters info = new BursaWolfParameters(WGS84);
info.dx = element.pullDouble("dx");
info.dy = element.pullDouble("dy");
info.dz = element.pullDouble("dz");
if (element.peek() != null) {
info.ex = element.pullDouble("ex");
info.ey = element.pullDouble("ey");
info.ez = element.pullDouble("ez");
info.ppm = element.pullDouble("ppm");
}
element.close();
return info;
}
/**
* Parses a {@code "SPHEROID"} element. This element has the following pattern:
*
* {@preformat text
* SPHEROID["", , {,}]
* }
*
* @param parent The parent element.
* @return The {@code "SPHEROID"} element as an {@link Ellipsoid} object.
* @throws ParseException if the {@code "SPHEROID"} element can't be parsed.
*/
private Ellipsoid parseSpheroid(final Element parent) throws ParseException {
Element element = parent.pullElement("SPHEROID");
String name = element.pullString("name");
double semiMajorAxis = element.pullDouble("semiMajorAxis");
double inverseFlattening = element.pullDouble("inverseFlattening");
Map properties = parseAuthority(element, name);
element.close();
if (inverseFlattening == 0) {
// Inverse flattening null is an OGC convention for a sphere.
inverseFlattening = Double.POSITIVE_INFINITY;
}
try {
return datumFactory.createFlattenedSphere(properties,
semiMajorAxis, inverseFlattening, METRE);
} catch (FactoryException exception) {
throw element.parseFailed(exception, null);
}
}
/**
* Parses a {@code "PROJECTION"} element. This element has the following pattern:
*
* {@preformat text
* PROJECTION["" {,}]
* }
*
* @param parent The parent element.
* @param ellipsoid The ellipsoid, or {@code null} if none.
* @param linearUnit The linear unit of the parent {@code PROJCS} element, or {@code null}.
* @param angularUnit The angular unit of the parent {@code GEOCS} element, or {@code null}.
* @return The {@code "PROJECTION"} element as a {@link ParameterValueGroup} object.
* @throws ParseException if the {@code "PROJECTION"} element can't be parsed.
*/
private ParameterValueGroup parseProjection(final Element parent,
final Ellipsoid ellipsoid,
final Unit linearUnit,
final Unit angularUnit)
throws ParseException
{
final Element element = parent.pullElement("PROJECTION");
final String classification = element.pullString("name");
final Map properties = parseAuthority(element, classification); // NOSONAR: Ignored for now.
element.close();
/*
* Set the list of parameters. NOTE: Parameters are defined in
* the parent Element (usually a "PROJCS" element), not in this
* "PROJECTION" element.
*
* We will set the semi-major and semi-minor parameters from the
* ellipsoid first. If those values were explicitly specified in
* a "PARAMETER" statement, they will overwrite the values inferred
* from the ellipsoid.
*/
final ParameterValueGroup parameters;
try {
parameters = mtFactory.getDefaultParameters(classification);
} catch (NoSuchIdentifierException exception) {
throw element.parseFailed(exception, null);
}
Element param = parent;
try {
if (ellipsoid != null) {
final Unit axisUnit = ellipsoid.getAxisUnit();
parameters.parameter("semi_major").setValue(ellipsoid.getSemiMajorAxis(), axisUnit);
parameters.parameter("semi_minor").setValue(ellipsoid.getSemiMinorAxis(), axisUnit);
}
while ((param = parent.pullOptionalElement("PARAMETER")) != null) {
final String paramName = param.pullString("name");
final ParameterValue> parameter = parameters.parameter(paramName);
final ParameterDescriptor> descriptor = parameter.getDescriptor();
final Class> valueClass = descriptor.getValueClass();
if (valueClass == String.class) {
parameter.setValue(param.pullString("value"));
} else if (valueClass == Boolean.class) {
parameter.setValue(param.pullBoolean("value"));
} else {
/*
* Usually, projection parameters contain only double values. The above
* check for other types was done as a safety, but having those types in
* a PROJECTION[...] element is unusual. Consequently we make the double
* type the default for all unknown types.
*/
final double paramValue = param.pullDouble("value");
final Unit> expected = descriptor.getUnit();
Unit> unit = null;
if (expected != null && !Unit.ONE.equals(expected)) {
if (linearUnit != null && METRE.isCompatible(expected)) {
unit = linearUnit;
} else if (angularUnit != null && RADIAN.isCompatible(expected)) {
unit = angularUnit;
}
}
if (unit != null) {
parameter.setValue(paramValue, unit);
} else {
parameter.setValue(paramValue);
}
}
param.close();
}
} catch (ParameterNotFoundException exception) {
throw param.parseFailed(exception, Errors.format(
Errors.Keys.UNEXPECTED_PARAMETER_$1, exception.getParameterName()));
}
return parameters;
}
/**
* Parses a {@code "DATUM"} element. This element has the following pattern:
*
* {@preformat text
* DATUM["", {,} {,}]
* }
*
* @param parent The parent element.
* @param meridian the prime meridian.
* @return The {@code "DATUM"} element as a {@link GeodeticDatum} object.
* @throws ParseException if the {@code "DATUM"} element can't be parsed.
*/
private GeodeticDatum parseDatum(final Element parent,
final PrimeMeridian meridian)
throws ParseException
{
Element element = parent.pullElement("DATUM");
String name = element.pullString("name");
Ellipsoid ellipsoid = parseSpheroid(element);
BursaWolfParameters toWGS84 = parseToWGS84(element); // Optional; may be null.
Map properties = parseAuthority(element, name);
if (ALLOW_ORACLE_SYNTAX && (toWGS84 == null) && (element.peek() instanceof Number)) {
toWGS84 = new BursaWolfParameters(WGS84);
toWGS84.dx = element.pullDouble("dx");
toWGS84.dy = element.pullDouble("dy");
toWGS84.dz = element.pullDouble("dz");
toWGS84.ex = element.pullDouble("ex");
toWGS84.ey = element.pullDouble("ey");
toWGS84.ez = element.pullDouble("ez");
toWGS84.ppm = element.pullDouble("ppm");
}
element.close();
if (toWGS84 != null) {
if (!(properties instanceof HashMap,?>)) {
properties = new HashMap(properties);
}
properties.put(BURSA_WOLF_KEY, toWGS84);
}
try {
return datumFactory.createGeodeticDatum(properties, ellipsoid, meridian);
} catch (FactoryException exception) {
throw element.parseFailed(exception, null);
}
}
/**
* Parses a {@code "VERT_DATUM"} element. This element has the following pattern:
*
* {@preformat text
* VERT_DATUM["", {,}]
* }
*
* @param parent The parent element.
* @return The {@code "VERT_DATUM"} element as a {@link VerticalDatum} object.
* @throws ParseException if the {@code "VERT_DATUM"} element can't be parsed.
*/
private VerticalDatum parseVertDatum(final Element parent) throws ParseException {
final Element element = parent.pullElement("VERT_DATUM");
final String name = element.pullString ("name");
final int datum = element.pullInteger("datum");
final Map properties = parseAuthority(element, name);
element.close();
final VerticalDatumType type = getVerticalDatumTypeFromLegacyCode(datum);
if (type == null) {
throw element.parseFailed(null, Errors.format(Errors.Keys.UNKNOWN_TYPE_$1, datum));
}
try {
return datumFactory.createVerticalDatum(properties, type);
} catch (FactoryException exception) {
throw element.parseFailed(exception, null);
}
}
/**
* Parses a {@code "LOCAL_DATUM"} element. This element has the following pattern:
*
* {@preformat text
* LOCAL_DATUM["", {,}]
* }
*
* @param parent The parent element.
* @return The {@code "LOCAL_DATUM"} element as an {@link EngineeringDatum} object.
* @throws ParseException if the {@code "LOCAL_DATUM"} element can't be parsed.
*
* @todo The vertical datum type is currently ignored.
*/
private EngineeringDatum parseLocalDatum(final Element parent) throws ParseException {
final Element element = parent.pullElement("LOCAL_DATUM");
final String name = element.pullString ("name");
final int datum = element.pullInteger("datum"); // NOSONAR: Ignored for now.
final Map properties = parseAuthority(element, name);
element.close();
try {
return datumFactory.createEngineeringDatum(properties);
} catch (FactoryException exception) {
throw element.parseFailed(exception, null);
}
}
/**
* Parses a {@code "LOCAL_CS"} element.
* This element has the following pattern:
*
* {@preformat text
* LOCAL_CS["", , , , {,}* {,}]
* }
*
* @param parent The parent element.
* @return The {@code "LOCAL_CS"} element as an {@link EngineeringCRS} object.
* @throws ParseException if the {@code "LOCAL_CS"} element can't be parsed.
*
* @todo The coordinate system used is always a Geotk implementation, since we don't
* know which method to invokes in the {@link CSFactory} (is it a Cartesian
* coordinate system? a spherical one? etc.).
*/
private EngineeringCRS parseLocalCS(final Element parent) throws ParseException {
Element element = parent.pullElement("LOCAL_CS");
String name = element.pullString("name");
EngineeringDatum datum = parseLocalDatum(element);
Unit linearUnit = parseUnit(element, METRE);
CoordinateSystemAxis axis = parseAxis(element, linearUnit, true);
List list = new ArrayList();
do {
list.add(axis);
axis = parseAxis(element, linearUnit, false);
} while (axis != null);
final Map properties = parseAuthority(element, name);
element.close();
final CoordinateSystem cs;
cs = new AbstractCS(singletonMap("name", name),
list.toArray(new CoordinateSystemAxis[list.size()]));
try {
return crsFactory.createEngineeringCRS(properties, datum, cs);
} catch (FactoryException exception) {
throw element.parseFailed(exception, null);
}
}
/**
* Parses a {@code "GEOCCS"} element.
* This element has the following pattern:
*
* {@preformat text
* GEOCCS["", , ,
* {, , ,} {,}]
* }
*
* @param parent The parent element.
* @return The {@code "GEOCCS"} element as a {@link GeocentricCRS} object.
* @throws ParseException if the {@code "GEOCCS"} element can't be parsed.
*/
private GeocentricCRS parseGeoCCS(final Element parent) throws ParseException {
final Element element = parent.pullElement("GEOCCS");
final String name = element.pullString("name");
final Map properties = parseAuthority(element, name);
final PrimeMeridian meridian = parsePrimem (element, DEGREE_ANGLE);
final GeodeticDatum datum = parseDatum (element, meridian);
final Unit linearUnit = parseUnit (element, METRE);
CoordinateSystemAxis axis0, axis1 = null, axis2 = null;
axis0 = parseAxis(element, linearUnit, false);
try {
if (axis0 != null) {
axis1 = parseAxis(element, linearUnit, true);
axis2 = parseAxis(element, linearUnit, true);
}
if (axis0 == null || axisIgnored) {
// Those default values are part of WKT specification.
axis0 = createAxis(null, "X", AxisDirection.OTHER, linearUnit);
axis1 = createAxis(null, "Y", AxisDirection.EAST, linearUnit);
axis2 = createAxis(null, "Z", AxisDirection.NORTH, linearUnit);
}
element.close();
CartesianCS cs = csFactory.createCartesianCS(properties, axis0, axis1, axis2);
cs = Convention.replace(cs, false);
return crsFactory.createGeocentricCRS(properties, datum, cs);
} catch (FactoryException exception) {
throw element.parseFailed(exception, null);
}
}
/**
* Parses an optional {@code "VERT_CS"} element.
* This element has the following pattern:
*
* {@preformat text
* VERT_CS["", , , {,} {,}]
* }
*
* @param parent The parent element.
* @return The {@code "VERT_CS"} element as a {@link VerticalCRS} object.
* @throws ParseException if the {@code "VERT_CS"} element can't be parsed.
*/
private VerticalCRS parseVertCS(final Element parent) throws ParseException {
final Element element = parent.pullElement("VERT_CS");
if (element == null) {
return null;
}
String name = element.pullString("name");
VerticalDatum datum = parseVertDatum(element);
Unit linearUnit = parseUnit(element, METRE);
CoordinateSystemAxis axis = parseAxis(element, linearUnit, false);
Map properties = parseAuthority(element, name);
element.close();
try {
if (axis == null || axisIgnored) {
axis = createAxis(null, isoConform ? "h" : "H", AxisDirection.UP, linearUnit);
}
return crsFactory.createVerticalCRS(properties, datum,
csFactory.createVerticalCS(singletonMap("name", name), axis));
} catch (FactoryException exception) {
throw element.parseFailed(exception, null);
}
}
/**
* Parses a {@code "GEOGCS"} element. This element has the following pattern:
*
* {@preformat text
* GEOGCS["", , , {,} {,}]
* }
*
* @param parent The parent element.
* @return The {@code "GEOGCS"} element as a {@link GeographicCRS} object.
* @throws ParseException if the {@code "GEOGCS"} element can't be parsed.
*/
private GeographicCRS parseGeoGCS(final Element parent) throws ParseException {
Element element = parent.pullElement("GEOGCS");
String name = element.pullString("name");
Map properties = parseAuthority(element, name);
Unit angularUnit = parseUnit (element, RADIAN);
PrimeMeridian meridian = parsePrimem (element, angularUnit);
GeodeticDatum datum = parseDatum (element, meridian);
CoordinateSystemAxis axis0 = parseAxis (element, angularUnit, false);
CoordinateSystemAxis axis1 = null;
try {
if (axis0 != null) {
axis1 = parseAxis(element, angularUnit, true);
}
if (axis0 == null || axisIgnored) {
// The (Lon,Lat) default values are part of WKT specification.
axis0 = createAxis(null, isoConform ? "\u03BB" : "Lon", AxisDirection.EAST, angularUnit);
axis1 = createAxis(null, isoConform ? "\u03C6" : "Lat", AxisDirection.NORTH, angularUnit);
}
element.close();
return crsFactory.createGeographicCRS(properties, datum,
csFactory.createEllipsoidalCS(properties, axis0, axis1));
} catch (FactoryException exception) {
throw element.parseFailed(exception, null);
}
}
/**
* Parses a {@code "PROJCS"} element.
* This element has the following pattern:
*
* {@preformat text
* PROJCS["", , , {,}*,
* {,}{,}]
* }
*
* @param parent The parent element.
* @return The {@code "PROJCS"} element as a {@link ProjectedCRS} object.
* @throws ParseException if the {@code "GEOGCS"} element can't be parsed.
*/
private ProjectedCRS parseProjCS(final Element parent) throws ParseException {
Element element = parent.pullElement("PROJCS");
String name = element.pullString("name");
Map properties = parseAuthority(element, name);
GeographicCRS geoCRS = parseGeoGCS(element);
Ellipsoid ellipsoid = geoCRS.getDatum().getEllipsoid();
Unit linearUnit = parseUnit(element, METRE);
ParameterValueGroup projection = parseProjection(element, ellipsoid, linearUnit,
(forcedAngularUnit != null) ? forcedAngularUnit :
geoCRS.getCoordinateSystem().getAxis(0).getUnit().asType(Angle.class));
CoordinateSystemAxis axis0 = parseAxis(element, linearUnit, false);
CoordinateSystemAxis axis1 = null;
try {
if (axis0 != null) {
axis1 = parseAxis(element, linearUnit, true);
}
if (axis0 == null || axisIgnored) {
// The (X,Y) default values are part of WKT specification.
axis0 = createAxis(null, isoConform ? "x" : "X", AxisDirection.EAST, linearUnit);
axis1 = createAxis(null, isoConform ? "y" : "Y", AxisDirection.NORTH, linearUnit);
}
element.close();
final Conversion conversion = new DefiningConversion(name, projection);
return crsFactory.createProjectedCRS(properties, geoCRS, conversion,
csFactory.createCartesianCS(properties, axis0, axis1));
} catch (FactoryException exception) {
throw element.parseFailed(exception, null);
}
}
/**
* Parses a {@code "COMPD_CS"} element.
* This element has the following pattern:
*
* {@preformat text
* COMPD_CS["", , {,}]
* }
*
* @param parent The parent element.
* @return The {@code "COMPD_CS"} element as a {@link CompoundCRS} object.
* @throws ParseException if the {@code "COMPD_CS"} element can't be parsed.
*/
private CompoundCRS parseCompdCS(final Element parent) throws ParseException {
final CoordinateReferenceSystem[] CRS = new CoordinateReferenceSystem[2];
Element element = parent.pullElement("COMPD_CS");
String name = element.pullString("name");
Map properties = parseAuthority(element, name);
CRS[0] = parseCoordinateReferenceSystem(element);
CRS[1] = parseCoordinateReferenceSystem(element);
element.close();
try {
return crsFactory.createCompoundCRS(properties, CRS);
} catch (FactoryException exception) {
throw element.parseFailed(exception, null);
}
}
/**
* Parses a {@code "FITTED_CS"} element.
* This element has the following pattern:
*
* {@preformat text
* FITTED_CS["", , ]
* }
*
* @param parent The parent element.
* @return The {@code "FITTED_CS"} element as a {@link CompoundCRS} object.
* @throws ParseException if the {@code "COMPD_CS"} element can't be parsed.
*/
private DerivedCRS parseFittedCS(final Element parent) throws ParseException {
Element element = parent.pullElement("FITTED_CS");
String name = element.pullString("name");
Map properties = parseAuthority(element, name);
final MathTransform toBase = parseMathTransform(element, true);
final CoordinateReferenceSystem base = parseCoordinateReferenceSystem(element);
final OperationMethod method = getOperationMethod();
element.close();
/*
* WKT provides no informations about the underlying CS of a derived CRS.
* We have to guess some reasonable one with arbitrary units. We try to
* construct the one which contains as few information as possible, in
* order to avoid providing wrong informations.
*/
final CoordinateSystemAxis[] axis = new CoordinateSystemAxis[toBase.getSourceDimensions()];
final StringBuilder buffer = new StringBuilder(name);
buffer.append(" axis ");
final int start = buffer.length();
try {
for (int i=0; i