001/*
002 *    GeoAPI - Java interfaces for OGC/ISO standards
003 *    Copyright © 2018-2023 Open Geospatial Consortium, Inc.
004 *    http://www.geoapi.org
005 *
006 *    Licensed under the Apache License, Version 2.0 (the "License");
007 *    you may not use this file except in compliance with the License.
008 *    You may obtain a copy of the License at
009 *
010 *        http://www.apache.org/licenses/LICENSE-2.0
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,
014 *    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
015 *    See the License for the specific language governing permissions and
016 *    limitations under the License.
017 */
018package org.opengis.geoapi.schema;
019
020import java.util.Map;
021import java.util.HashMap;
022
023
024/**
025 * Departures in GeoAPI interfaces compared to OGC/ISO schemas.
026 * Each {@code Departures} instance is initialized with a default set of departure information.
027 * More departure can be added by calls to {@code add(…)} methods.
028 *
029 * @author  Martin Desruisseaux (Geomatys)
030 * @since   3.1
031 * @version 3.1
032 */
033public class Departures {
034    /**
035     * ISO 19115-2 classes merged with ISO 19115-1 classes. For example, ISO 19115-2 defines {@code MI_Band}
036     * as an extension of ISO 19115-1 {@code MD_Band}, but GeoAPI merges those two types in a single interface
037     * for simplicity.
038     *
039     * <p>Keys or extension types (e.g. the {@code "MI_*"} types defined by the metadata extension for imagery)
040     * and values are the base types in which the extension has been merged (e.g. the {@code "MD_*"} types defined
041     * by metadata fundamentals).</p>
042     */
043    private final Map<String,String> mergedTypes;
044
045    /**
046     * Changes in the spelling of an identifier. The differences may be for historical reasons,
047     * or in a few cases because of a misspelling in the XSD file compared to the UML.
048     *
049     * <p>Keys are the spellings used in GeoAPI {@link org.opengis.annotation.UML} annotations
050     * and values are the spellings used in XSD files.</p>
051     */
052    final Map<String,String> spellingChanges;
053
054    /**
055     * Creates new collections of departure information. All maps in public fields ({@link #mergedTypes},
056     * <i>etc.</i>) are initialized with new instances and populated.
057     */
058    public Departures() {
059        Map<String,String> m = new HashMap<>(12);
060        // ………Merge what…………………………………………………………Into……………………………………………
061        m.put("MI_Band_Type",                 "MD_Band_Type");
062        m.put("MI_CoverageDescription_Type",  "MD_CoverageDescription_Type");
063        m.put("MI_Georectified_Type",         "MD_Georectified_Type");
064        m.put("MI_Georeferenceable_Type",     "MD_Georeferenceable_Type");
065        m.put("LE_Source_Type",               "LI_Source_Type");
066        m.put("LE_ProcessStep_Type",          "LI_ProcessStep_Type");
067        m.put("AbstractMX_File_Type",         "MX_DataFile_Type");
068        m.put("Abstract_DataQuality_Type",    "DQ_DataQuality_Type");
069        m.put("Abstract_QualityElement_Type", "AbstractDQ_Element_Type");
070        mergedTypes = m;
071
072        m = new HashMap<>(12);
073        m.put("MI_EnvironmentalRecord.meteorologicalConditions", "meterologicalConditions");    // Misspelling in ISO 19115-3:2016
074        m.put("MI_Requirement.satisfiedPlan",                    "satisifiedPlan");             // Misspelling in ISO 19115-3:2016
075        m.put("LI_ProcessStep.stepDateTime",                     "dateTime");                   // Spelling change in XSD files
076        m.put("DQ_Result.valueType",                             "valueRecordType");            // TODO: verify in ISO 19157
077        spellingChanges = m;
078    }
079
080    /**
081     * Adds a class to be retrofitted into another class. For example, ISO 19115-2 defines {@code MI_Band} as
082     * an extension of ISO 19115-1 {@code MD_Band}, but GeoAPI merges those two types in a single interface
083     * for simplicity.
084     *
085     * @param  toRetrofit  name of the type to retrofit into another type. Example: {@code MI_Band}.
086     * @param  target      name of a type which will receive the properties of the retrofitted type.
087     *                     Example: {@code MD_Band}.
088     */
089    public void addMergedType(final String toRetrofit, final String target) {
090        if (mergedTypes.putIfAbsent(toRetrofit, target) != null) {
091            throw new IllegalArgumentException(toRetrofit + " is already retrofitted.");
092        }
093    }
094
095    /**
096     * Changes the spelling of an identifier. The differences may be for historical reasons,
097     * or in a few cases because of a misspelling in the XSD file compared to the UML.
098     *
099     * @param  uml  the spelling in the UML, including class name. Example: {@code "LI_ProcessStep.stepDateTime"}.
100     * @param  xsd  the spelling in the XSD file. Example: {@code "dateTime"}.
101     */
102    public void addSpellingChange(final String uml, final String xsd) {
103        if (spellingChanges.put(uml, xsd) != null) {
104            throw new IllegalArgumentException("A spelling change is already declared for " + uml);
105        }
106    }
107
108    /**
109     * Returns the name of a class merging the given class and its (usually) parent class.
110     * For example, {@code "MI_Band_Type"} is renamed as {@code "MD_Band_Type"}.
111     * We do that because we use only one class for representing those two distinct ISO types.
112     * Note that not all ISO 19115-2 types extend an ISO 19115-1 type, so we need to apply a case-by-case approach.
113     * If there is no merge to apply, then this method returns the given name unchanged.
114     *
115     * @param  name  name of a class to potentially merge with its parent class.
116     * @return the merged class name (may be the given name) together with other information.
117     */
118    final MergeInfo nameOfMergedType(final String name) {
119        String target = mergedTypes.remove(name);
120        if (target == null) {
121            return new MergeInfo(name, false);
122        } else {
123            return new MergeInfo(target, name.startsWith("Abstract"));
124        }
125    }
126
127    /**
128     * Information about a type that may have been retrofitted into another type.
129     * For example, ISO 19115-2 defines {@code MI_Band} as an extension of ISO 19115-1 {@code MD_Band},
130     * but GeoAPI merges those two types in a single interface for simplicity.
131     * Sometimes the merge also implies to change properties order.
132     */
133    static final class MergeInfo {
134        /**
135         * Name of the merged type.
136         */
137        final String typeName;
138
139        /**
140         * Whether we will need to reorder properties. Reordering will be needed if the properties of
141         * parent type are retrofitted into the properties of the child type instead of the converse.
142         * We identifies this situation by the {@code "Abstract"} prefix in type to retrofit.
143         */
144        private final boolean needToReorderProperties;
145
146        /**
147         * Names of properties to keep last, or {@code null} if none.
148         * This is set to a non-null value only if {@link #needToReorderProperties} is {@code true}.
149         */
150        private String[] propertiesToKeepLast;
151
152        /**
153         * Creates information for a type.
154         *
155         * @param  typeName  name of the merged type.
156         * @param  reorder   whether we will need to reorder properties.
157         */
158        private MergeInfo(final String typeName, final boolean reorder) {
159            this.typeName = typeName;
160            needToReorderProperties = reorder;
161        }
162
163        /**
164         * Invoked before properties are added in the given map. This method does nothing
165         * in the common case where there is no merge operation to prepare.
166         *
167         * @param  properties  map where a property will be added.
168         */
169        final void beforeAddProperties(final Map<String,?> properties) {
170            if (needToReorderProperties) {
171                propertiesToKeepLast = properties.keySet().toArray(String[]::new);
172            }
173        }
174
175        /**
176         * Invoked after properties are added in the given map. If a merge operation has been applied,
177         * then this method may reorder entries in the given map by moving last the properties recorded
178         * in this {@code MergeInfo}.
179         *
180         * @param  <E>         type of values in the map.
181         * @param  properties  map where a property has been added.
182         */
183        final <E> void afterAddProperties(final Map<String,E> properties) {
184            if (propertiesToKeepLast != null) {
185                for (final String p : propertiesToKeepLast) {
186                    if (p != null) {
187                        final E e = properties.remove(p);
188                        if (e == null) {
189                            throw new IllegalArgumentException("Missing property for " + p);
190                        }
191                        properties.put(p, e);
192                    }
193                }
194            }
195        }
196    }
197}