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}