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.annotation.Child;
024import ca.uhn.fhir.model.api.annotation.Description;
025import jakarta.annotation.Nonnull;
026import org.apache.commons.lang3.StringUtils;
027import org.apache.commons.lang3.Validate;
028import org.hl7.fhir.instance.model.api.IBase;
029import org.hl7.fhir.instance.model.api.IBaseDatatype;
030import org.hl7.fhir.instance.model.api.IBaseReference;
031import org.hl7.fhir.instance.model.api.IBaseResource;
032import org.hl7.fhir.instance.model.api.IPrimitiveType;
033
034import java.lang.reflect.Field;
035import java.util.ArrayList;
036import java.util.Collections;
037import java.util.HashMap;
038import java.util.List;
039import java.util.Map;
040import java.util.Set;
041
042public class RuntimeChildChoiceDefinition extends BaseRuntimeDeclaredChildDefinition {
043
044        private List<Class<? extends IBase>> myChoiceTypes;
045        private Map<String, BaseRuntimeElementDefinition<?>> myNameToChildDefinition;
046        private Map<Class<? extends IBase>, String> myDatatypeToElementName;
047        private Map<Class<? extends IBase>, BaseRuntimeElementDefinition<?>> myDatatypeToElementDefinition;
048        private String myReferenceSuffix;
049        private List<Class<? extends IBaseResource>> myResourceTypes;
050        private List<Class<? extends IBase>> mySpecializationChoiceTypes = Collections.emptyList();
051
052        /**
053         * Constructor
054         */
055        public RuntimeChildChoiceDefinition(
056                        Field theField,
057                        String theElementName,
058                        Child theChildAnnotation,
059                        Description theDescriptionAnnotation,
060                        List<Class<? extends IBase>> theChoiceTypes) {
061                super(theField, theChildAnnotation, theDescriptionAnnotation, theElementName);
062
063                myChoiceTypes = Collections.unmodifiableList(theChoiceTypes);
064        }
065
066        /**
067         * Constructor
068         *
069         * For extension, if myChoiceTypes will be set some other way
070         */
071        RuntimeChildChoiceDefinition(
072                        Field theField, String theElementName, Child theChildAnnotation, Description theDescriptionAnnotation) {
073                super(theField, theChildAnnotation, theDescriptionAnnotation, theElementName);
074        }
075
076        void setChoiceTypes(
077                        @Nonnull List<Class<? extends IBase>> theChoiceTypes,
078                        @Nonnull List<Class<? extends IBase>> theSpecializationChoiceTypes) {
079                Validate.notNull(theChoiceTypes, "theChoiceTypes must not be null");
080                Validate.notNull(theSpecializationChoiceTypes, "theSpecializationChoiceTypes must not be null");
081                myChoiceTypes = Collections.unmodifiableList(theChoiceTypes);
082                mySpecializationChoiceTypes = Collections.unmodifiableList(theSpecializationChoiceTypes);
083        }
084
085        public List<Class<? extends IBase>> getChoices() {
086                return myChoiceTypes;
087        }
088
089        @Override
090        public Set<String> getValidChildNames() {
091                return myNameToChildDefinition.keySet();
092        }
093
094        @Override
095        public BaseRuntimeElementDefinition<?> getChildByName(String theName) {
096                assert myNameToChildDefinition.containsKey(theName)
097                                : "Can't find child '" + theName + "' in names: " + myNameToChildDefinition.keySet();
098
099                return myNameToChildDefinition.get(theName);
100        }
101
102        @SuppressWarnings("unchecked")
103        @Override
104        void sealAndInitialize(
105                        FhirContext theContext,
106                        Map<Class<? extends IBase>, BaseRuntimeElementDefinition<?>> theClassToElementDefinitions) {
107                myNameToChildDefinition = new HashMap<>();
108                myDatatypeToElementName = new HashMap<>();
109                myDatatypeToElementDefinition = new HashMap<>();
110                myResourceTypes = new ArrayList<>();
111
112                myReferenceSuffix = "Reference";
113
114                sealAndInitializeChoiceTypes(theContext, theClassToElementDefinitions, mySpecializationChoiceTypes, true);
115                sealAndInitializeChoiceTypes(theContext, theClassToElementDefinitions, myChoiceTypes, false);
116
117                myNameToChildDefinition = Collections.unmodifiableMap(myNameToChildDefinition);
118                myDatatypeToElementName = Collections.unmodifiableMap(myDatatypeToElementName);
119                myDatatypeToElementDefinition = Collections.unmodifiableMap(myDatatypeToElementDefinition);
120                myResourceTypes = Collections.unmodifiableList(myResourceTypes);
121        }
122
123        private void sealAndInitializeChoiceTypes(
124                        FhirContext theContext,
125                        Map<Class<? extends IBase>, BaseRuntimeElementDefinition<?>> theClassToElementDefinitions,
126                        List<Class<? extends IBase>> choiceTypes,
127                        boolean theIsSpecilization) {
128                for (Class<? extends IBase> next : choiceTypes) {
129
130                        String elementName = null;
131                        BaseRuntimeElementDefinition<?> nextDef;
132                        boolean nonPreferred = false;
133                        if (IBaseResource.class.isAssignableFrom(next)) {
134                                elementName = getElementName() + StringUtils.capitalize(next.getSimpleName());
135                                nextDef = findResourceReferenceDefinition(theClassToElementDefinitions);
136
137                                if (!theIsSpecilization) {
138                                        myNameToChildDefinition.put(getElementName() + "Reference", nextDef);
139                                        myNameToChildDefinition.put(getElementName() + "Resource", nextDef);
140                                }
141
142                                myResourceTypes.add((Class<? extends IBaseResource>) next);
143
144                        } else {
145                                nextDef = theClassToElementDefinitions.get(next);
146                                BaseRuntimeElementDefinition<?> nextDefForChoice = nextDef;
147
148                                /*
149                                 * In HAPI 1.3 the following applied:
150                                 * Elements which are called foo[x] and have a choice which is a profiled datatype must use the
151                                 * unprofiled datatype as the element name. E.g. if foo[x] allows markdown as a datatype, it calls the
152                                 * element fooString when encoded, because markdown is a profile of string. This is according to the
153                                 * FHIR spec
154                                 *
155                                 * Note that as of HAPI 1.4 this applies only to non-primitive datatypes after discussion
156                                 * with Grahame.
157                                 */
158                                if (nextDef instanceof IRuntimeDatatypeDefinition) {
159                                        IRuntimeDatatypeDefinition nextDefDatatype = (IRuntimeDatatypeDefinition) nextDef;
160                                        if (nextDefDatatype.getProfileOf() != null && !IPrimitiveType.class.isAssignableFrom(next)) {
161                                                nextDefForChoice = null;
162                                                nonPreferred = true;
163                                                Class<? extends IBaseDatatype> profileType = nextDefDatatype.getProfileOf();
164                                                BaseRuntimeElementDefinition<?> elementDef = theClassToElementDefinitions.get(profileType);
165                                                elementName = getElementName() + StringUtils.capitalize(elementDef.getName());
166                                        }
167                                }
168                                if (nextDefForChoice != null) {
169                                        elementName = getElementName() + StringUtils.capitalize(nextDefForChoice.getName());
170                                }
171                        }
172
173                        // I don't see how elementName could be null here, but eclipse complains..
174                        if (!theIsSpecilization) {
175                                if (elementName != null) {
176                                        if (!myNameToChildDefinition.containsKey(elementName) || !nonPreferred) {
177                                                myNameToChildDefinition.put(elementName, nextDef);
178                                        }
179                                }
180
181                                /*
182                                 * If this is a resource reference, the element name is "fooNameReference"
183                                 */
184                                if (IBaseResource.class.isAssignableFrom(next) || IBaseReference.class.isAssignableFrom(next)) {
185                                        next = theContext.getVersion().getResourceReferenceType();
186                                        elementName = getElementName() + myReferenceSuffix;
187                                        myNameToChildDefinition.put(elementName, nextDef);
188                                }
189                        }
190
191                        myDatatypeToElementDefinition.put(next, nextDef);
192
193                        if (myDatatypeToElementName.containsKey(next)) {
194                                String existing = myDatatypeToElementName.get(next);
195                                if (!existing.equals(elementName)) {
196                                        throw new ConfigurationException(
197                                                        Msg.code(1693) + "Already have element name " + existing + " for datatype "
198                                                                        + next.getSimpleName() + " in " + getElementName() + ", cannot add " + elementName);
199                                }
200                        } else {
201                                myDatatypeToElementName.put(next, elementName);
202                        }
203                }
204        }
205
206        public List<Class<? extends IBaseResource>> getResourceTypes() {
207                return myResourceTypes;
208        }
209
210        @Override
211        public String getChildNameByDatatype(Class<? extends IBase> theDatatype) {
212                return myDatatypeToElementName.get(theDatatype);
213        }
214
215        @Override
216        public BaseRuntimeElementDefinition<?> getChildElementDefinitionByDatatype(Class<? extends IBase> theDatatype) {
217                return myDatatypeToElementDefinition.get(theDatatype);
218        }
219
220        public Set<Class<? extends IBase>> getValidChildTypes() {
221                return Collections.unmodifiableSet((myDatatypeToElementDefinition.keySet()));
222        }
223}