001/*
002 * #%L
003 * HAPI FHIR - Core Library
004 * %%
005 * Copyright (C) 2014 - 2025 Smile CDR, Inc.
006 * %%
007 * Licensed under the Apache License, Version 2.0 (the "License");
008 * you may not use this file except in compliance with the License.
009 * You may obtain a copy of the License at
010 *
011 *      http://www.apache.org/licenses/LICENSE-2.0
012 *
013 * Unless required by applicable law or agreed to in writing, software
014 * distributed under the License is distributed on an "AS IS" BASIS,
015 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
016 * See the License for the specific language governing permissions and
017 * limitations under the License.
018 * #L%
019 */
020package ca.uhn.fhir.context;
021
022import ca.uhn.fhir.i18n.Msg;
023import ca.uhn.fhir.model.api.ExtensionDt;
024import ca.uhn.fhir.model.api.IDatatype;
025import ca.uhn.fhir.model.base.composite.BaseResourceReferenceDt;
026import org.apache.commons.text.WordUtils;
027import org.hl7.fhir.instance.model.api.IBase;
028
029import java.util.ArrayList;
030import java.util.Collections;
031import java.util.HashMap;
032import java.util.List;
033import java.util.Map;
034import java.util.Set;
035
036public class RuntimeChildUndeclaredExtensionDefinition extends BaseRuntimeChildDefinition {
037
038        private static final String VALUE_REFERENCE = "valueReference";
039        private static final String VALUE_RESOURCE = "valueResource";
040        private static final org.slf4j.Logger ourLog =
041                        org.slf4j.LoggerFactory.getLogger(RuntimeChildUndeclaredExtensionDefinition.class);
042        private Map<String, BaseRuntimeElementDefinition<?>> myAttributeNameToDefinition;
043        private Map<Class<? extends IBase>, String> myDatatypeToAttributeName;
044        private Map<Class<? extends IBase>, BaseRuntimeElementDefinition<?>> myDatatypeToDefinition;
045
046        public RuntimeChildUndeclaredExtensionDefinition() {
047                // nothing
048        }
049
050        private void addReferenceBinding(
051                        FhirContext theContext,
052                        Map<Class<? extends IBase>, BaseRuntimeElementDefinition<?>> theClassToElementDefinitions,
053                        String value) {
054                BaseRuntimeElementDefinition<?> def = findResourceReferenceDefinition(theClassToElementDefinitions);
055
056                myAttributeNameToDefinition.put(value, def);
057                /*
058                 * Resource reference - The correct name is 'valueReference' in DSTU2 and 'valueResource' in DSTU1
059                 */
060                if (!value.equals(VALUE_RESOURCE)) {
061                        myDatatypeToAttributeName.put(theContext.getVersion().getResourceReferenceType(), value);
062                        myDatatypeToDefinition.put(BaseResourceReferenceDt.class, def);
063                        myDatatypeToDefinition.put(theContext.getVersion().getResourceReferenceType(), def);
064                }
065        }
066
067        @Override
068        public IAccessor getAccessor() {
069                return new IAccessor() {
070                        @Override
071                        public List<IBase> getValues(IBase theTarget) {
072                                ExtensionDt target = (ExtensionDt) theTarget;
073                                if (target.getValue() != null) {
074                                        return Collections.singletonList(target.getValue());
075                                }
076                                return new ArrayList<>(target.getUndeclaredExtensions());
077                        }
078                };
079        }
080
081        @Override
082        public BaseRuntimeElementDefinition<?> getChildByName(String theName) {
083                return myAttributeNameToDefinition.get(theName);
084        }
085
086        @Override
087        public BaseRuntimeElementDefinition<?> getChildElementDefinitionByDatatype(Class<? extends IBase> theType) {
088                return myDatatypeToDefinition.get(theType);
089        }
090
091        @Override
092        public String getChildNameByDatatype(Class<? extends IBase> theDatatype) {
093                return myDatatypeToAttributeName.get(theDatatype);
094        }
095
096        @Override
097        public String getElementName() {
098                return "extension";
099        }
100
101        @Override
102        public int getMax() {
103                return 1;
104        }
105
106        @Override
107        public int getMin() {
108                return 0;
109        }
110
111        @Override
112        public IMutator getMutator() {
113                return new IMutator() {
114                        @Override
115                        public void addValue(IBase theTarget, IBase theValue) {
116                                ExtensionDt target = (ExtensionDt) theTarget;
117                                target.setValue((IDatatype) theTarget);
118                        }
119
120                        @Override
121                        public void setValue(IBase theTarget, IBase theValue) {
122                                ExtensionDt target = (ExtensionDt) theTarget;
123                                target.setValue((IDatatype) theTarget);
124                        }
125                };
126        }
127
128        @Override
129        public Set<String> getValidChildNames() {
130                return myAttributeNameToDefinition.keySet();
131        }
132
133        @Override
134        public boolean isSummary() {
135                return false;
136        }
137
138        @Override
139        void sealAndInitialize(
140                        FhirContext theContext,
141                        Map<Class<? extends IBase>, BaseRuntimeElementDefinition<?>> theClassToElementDefinitions) {
142                Map<String, BaseRuntimeElementDefinition<?>> datatypeAttributeNameToDefinition = new HashMap<>();
143                myDatatypeToAttributeName = new HashMap<>();
144                myDatatypeToDefinition = new HashMap<>();
145
146                for (BaseRuntimeElementDefinition<?> next : theClassToElementDefinitions.values()) {
147                        if (next instanceof IRuntimeDatatypeDefinition) {
148
149                                myDatatypeToDefinition.put(next.getImplementingClass(), next);
150
151                                boolean isSpecialization = ((IRuntimeDatatypeDefinition) next).isSpecialization();
152                                if (isSpecialization) {
153                                        ourLog.trace("Not adding specialization: {}", next.getImplementingClass());
154                                }
155
156                                if (!next.isStandardType()) {
157                                        continue;
158                                }
159
160                                String qualifiedName = next.getImplementingClass().getName();
161
162                                /*
163                                 * We don't want user-defined custom datatypes ending up overriding the built in
164                                 * types here. It would probably be better for there to be a way for
165                                 * a datatype to indicate via its annotation that it's a built in
166                                 * type.
167                                 */
168                                if (!qualifiedName.startsWith("ca.uhn.fhir.model")) {
169                                        if (!qualifiedName.startsWith("org.hl7.fhir")) {
170                                                continue;
171                                        }
172                                }
173
174                                String attrName = createExtensionChildName(next);
175                                if (isSpecialization && datatypeAttributeNameToDefinition.containsKey(attrName)) {
176                                        continue;
177                                }
178
179                                if (datatypeAttributeNameToDefinition.containsKey(attrName)) {
180                                        BaseRuntimeElementDefinition<?> existing = datatypeAttributeNameToDefinition.get(attrName);
181                                        // We do allow built-in standard types to override extension types with the same element name,
182                                        // e.g. how EnumerationType extends CodeType but both serialize to "code". In this case,
183                                        // CodeType should win. If we aren't in a situation like that, there is a problem with the
184                                        // model so we should bail.
185                                        if (!existing.isStandardType()) {
186                                                throw new ConfigurationException(Msg.code(1734) + "More than one child of " + getElementName()
187                                                                + " matches attribute name " + attrName + ". Found ["
188                                                                + existing.getImplementingClass().getName() + "] and ["
189                                                                + next.getImplementingClass().getName() + "]");
190                                        }
191                                }
192
193                                datatypeAttributeNameToDefinition.put(attrName, next);
194                                datatypeAttributeNameToDefinition.put(attrName.toLowerCase(), next);
195                                myDatatypeToAttributeName.put(next.getImplementingClass(), attrName);
196                        }
197                }
198
199                myAttributeNameToDefinition = datatypeAttributeNameToDefinition;
200
201                /*
202                 * Resource reference - The correct name is 'valueReference' in DSTU2 and 'valueResource' in DSTU1
203                 */
204                addReferenceBinding(theContext, theClassToElementDefinitions, VALUE_RESOURCE);
205                addReferenceBinding(theContext, theClassToElementDefinitions, VALUE_REFERENCE);
206        }
207
208        public static String createExtensionChildName(BaseRuntimeElementDefinition<?> next) {
209                return "value" + WordUtils.capitalize(next.getName());
210        }
211}