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.ResourceDef;
024import ca.uhn.fhir.util.UrlUtil;
025import org.hl7.fhir.instance.model.api.IAnyResource;
026import org.hl7.fhir.instance.model.api.IBase;
027import org.hl7.fhir.instance.model.api.IBaseResource;
028import org.hl7.fhir.instance.model.api.IDomainResource;
029
030import java.util.ArrayList;
031import java.util.Collections;
032import java.util.Comparator;
033import java.util.HashMap;
034import java.util.LinkedHashMap;
035import java.util.List;
036import java.util.Map;
037
038public class RuntimeResourceDefinition extends BaseRuntimeElementCompositeDefinition<IBaseResource> {
039
040        private Class<? extends IBaseResource> myBaseType;
041        private Map<String, List<RuntimeSearchParam>> myCompartmentNameToSearchParams;
042        private FhirContext myContext;
043        private String myId;
044        private Map<String, RuntimeSearchParam> myNameToSearchParam = new LinkedHashMap<String, RuntimeSearchParam>();
045        private IBaseResource myProfileDef;
046        private String myResourceProfile;
047        private List<RuntimeSearchParam> mySearchParams;
048        private final FhirVersionEnum myStructureVersion;
049        private volatile RuntimeResourceDefinition myBaseDefinition;
050
051        public RuntimeResourceDefinition(
052                        FhirContext theContext,
053                        String theResourceName,
054                        Class<? extends IBaseResource> theClass,
055                        ResourceDef theResourceAnnotation,
056                        boolean theStandardType,
057                        Map<Class<? extends IBase>, BaseRuntimeElementDefinition<?>> theClassToElementDefinitions) {
058                super(theResourceName, theClass, theStandardType, theContext, theClassToElementDefinitions);
059                myContext = theContext;
060                myResourceProfile = theResourceAnnotation.profile();
061                myId = theResourceAnnotation.id();
062
063                IBaseResource instance;
064                try {
065                        instance = theClass.getConstructor().newInstance();
066                } catch (Exception e) {
067                        throw new ConfigurationException(
068                                        Msg.code(1730)
069                                                        + myContext
070                                                                        .getLocalizer()
071                                                                        .getMessage(getClass(), "nonInstantiableType", theClass.getName(), e.toString()),
072                                        e);
073                }
074                myStructureVersion = instance.getStructureFhirVersionEnum();
075                if (myStructureVersion != theContext.getVersion().getVersion()) {
076                        if (myStructureVersion == FhirVersionEnum.R5
077                                        && theContext.getVersion().getVersion() == FhirVersionEnum.R4B) {
078                                // TODO: remove this exception once we've bumped FHIR core to a new version
079                                // TODO: also fix the TODO in ModelScanner
080                                // TODO: also fix the TODO in RestfulServerUtils
081                                // TODO: also fix the TODO in BaseParser
082                        } else {
083                                throw new ConfigurationException(Msg.code(1731)
084                                                + myContext
085                                                                .getLocalizer()
086                                                                .getMessage(
087                                                                                getClass(),
088                                                                                "typeWrongVersion",
089                                                                                theContext.getVersion().getVersion(),
090                                                                                theClass.getName(),
091                                                                                myStructureVersion));
092                        }
093                }
094        }
095
096        public void addSearchParam(RuntimeSearchParam theParam) {
097                myNameToSearchParam.put(theParam.getName(), theParam);
098        }
099
100        /**
101         * If this definition refers to a class which extends another resource definition type, this
102         * method will return the definition of the topmost resource. For example, if this definition
103         * refers to MyPatient2, which extends MyPatient, which in turn extends Patient, this method
104         * will return the resource definition for Patient.
105         * <p>
106         * If the definition has no parent, returns <code>this</code>
107         * </p>
108         */
109        public RuntimeResourceDefinition getBaseDefinition() {
110                validateSealed();
111                if (myBaseDefinition == null) {
112                        myBaseDefinition = myContext.getResourceDefinition(myBaseType);
113                }
114                return myBaseDefinition;
115        }
116
117        @Override
118        public ca.uhn.fhir.context.BaseRuntimeElementDefinition.ChildTypeEnum getChildType() {
119                return ChildTypeEnum.RESOURCE;
120        }
121
122        public String getId() {
123                return myId;
124        }
125
126        /**
127         * Express {@link #getImplementingClass()} as theClass (to prevent casting warnings)
128         */
129        @SuppressWarnings("unchecked")
130        public <T> Class<T> getImplementingClass(Class<T> theClass) {
131                if (!theClass.isAssignableFrom(getImplementingClass())) {
132                        throw new ConfigurationException(
133                                        Msg.code(1732) + "Unable to convert " + getImplementingClass() + " to " + theClass);
134                }
135                return (Class<T>) getImplementingClass();
136        }
137
138        @Deprecated
139        public String getResourceProfile() {
140                return myResourceProfile;
141        }
142
143        public String getResourceProfile(String theServerBase) {
144                validateSealed();
145                String profile;
146                if (!myResourceProfile.isEmpty()) {
147                        profile = myResourceProfile;
148                } else if (!myId.isEmpty()) {
149                        profile = myId;
150                } else {
151                        return "";
152                }
153
154                if (!UrlUtil.isValid(profile)) {
155                        String resourceName = "/StructureDefinition/";
156                        String profileWithUrl = theServerBase + resourceName + profile;
157                        if (UrlUtil.isValid(profileWithUrl)) {
158                                return profileWithUrl;
159                        }
160                }
161                return profile;
162        }
163
164        public RuntimeSearchParam getSearchParam(String theName) {
165                validateSealed();
166                return myNameToSearchParam.get(theName);
167        }
168
169        public List<RuntimeSearchParam> getSearchParams() {
170                validateSealed();
171                return mySearchParams;
172        }
173
174        /**
175         * Will not return null
176         */
177        public List<RuntimeSearchParam> getSearchParamsForCompartmentName(String theCompartmentName) {
178                validateSealed();
179                List<RuntimeSearchParam> retVal = myCompartmentNameToSearchParams.get(theCompartmentName);
180                if (retVal == null) {
181                        return Collections.emptyList();
182                }
183                return retVal;
184        }
185
186        public FhirVersionEnum getStructureVersion() {
187                return myStructureVersion;
188        }
189
190        public boolean isBundle() {
191                return "Bundle".equals(getName());
192        }
193
194        @SuppressWarnings("unchecked")
195        @Override
196        public void sealAndInitialize(
197                        FhirContext theContext,
198                        Map<Class<? extends IBase>, BaseRuntimeElementDefinition<?>> theClassToElementDefinitions) {
199                super.sealAndInitialize(theContext, theClassToElementDefinitions);
200
201                myNameToSearchParam = Collections.unmodifiableMap(myNameToSearchParam);
202
203                ArrayList<RuntimeSearchParam> searchParams = new ArrayList<RuntimeSearchParam>(myNameToSearchParam.values());
204                Collections.sort(searchParams, new Comparator<RuntimeSearchParam>() {
205                        @Override
206                        public int compare(RuntimeSearchParam theArg0, RuntimeSearchParam theArg1) {
207                                return theArg0.getName().compareTo(theArg1.getName());
208                        }
209                });
210                mySearchParams = Collections.unmodifiableList(searchParams);
211
212                Map<String, List<RuntimeSearchParam>> compartmentNameToSearchParams = new HashMap<>();
213                for (RuntimeSearchParam next : searchParams) {
214                        if (next.getProvidesMembershipInCompartments() != null) {
215                                for (String nextCompartment : next.getProvidesMembershipInCompartments()) {
216
217                                        if (nextCompartment.startsWith("Base FHIR compartment definition for ")) {
218                                                nextCompartment = nextCompartment.substring("Base FHIR compartment definition for ".length());
219                                        }
220
221                                        if (!compartmentNameToSearchParams.containsKey(nextCompartment)) {
222                                                compartmentNameToSearchParams.put(nextCompartment, new ArrayList<>());
223                                        }
224                                        List<RuntimeSearchParam> searchParamsForCompartment =
225                                                        compartmentNameToSearchParams.get(nextCompartment);
226                                        searchParamsForCompartment.add(next);
227
228                                        /*
229                                         * If one search parameter marks an SP as making a resource
230                                         * a part of a compartment, let's also denote all other
231                                         * SPs with the same path the same way. This behaviour is
232                                         * used by AuthorizationInterceptor
233                                         */
234                                        String nextPath = massagePathForCompartmentSimilarity(next.getPath());
235                                        for (RuntimeSearchParam nextAlternate : searchParams) {
236                                                String nextAlternatePath = massagePathForCompartmentSimilarity(nextAlternate.getPath());
237                                                if (nextAlternatePath.equals(nextPath)) {
238                                                        if (!nextAlternate.getName().equals(next.getName())) {
239                                                                searchParamsForCompartment.add(nextAlternate);
240                                                        }
241                                                }
242                                        }
243                                }
244                        }
245                }
246
247                // Make the map of lists completely unmodifiable
248                for (String nextKey : new ArrayList<>(compartmentNameToSearchParams.keySet())) {
249                        List<RuntimeSearchParam> nextList = compartmentNameToSearchParams.get(nextKey);
250                        compartmentNameToSearchParams.put(nextKey, Collections.unmodifiableList(nextList));
251                }
252                myCompartmentNameToSearchParams = Collections.unmodifiableMap(compartmentNameToSearchParams);
253
254                Class<?> target = getImplementingClass();
255                myBaseType = (Class<? extends IBaseResource>) target;
256                do {
257                        target = target.getSuperclass();
258                        if (IBaseResource.class.isAssignableFrom(target) && target.getAnnotation(ResourceDef.class) != null) {
259                                myBaseType = (Class<? extends IBaseResource>) target;
260                        }
261                } while (target.equals(Object.class) == false);
262
263                /*
264                 * See #504:
265                 * Bundle types may not have extensions
266                 */
267                if (hasExtensions()) {
268                        if (IAnyResource.class.isAssignableFrom(getImplementingClass())) {
269                                if (!IDomainResource.class.isAssignableFrom(getImplementingClass())) {
270                                        throw new ConfigurationException(Msg.code(1733) + "Class \"" + getImplementingClass()
271                                                        + "\" is invalid. This resource type is not a DomainResource, it must not have extensions");
272                                }
273                        }
274                }
275        }
276
277        private String massagePathForCompartmentSimilarity(String thePath) {
278                String path = thePath;
279                if (path.matches(".*\\.where\\(resolve\\(\\) is [a-zA-Z]+\\)")) {
280                        path = path.substring(0, path.indexOf(".where"));
281                }
282                return path;
283        }
284
285        @Deprecated
286        public synchronized IBaseResource toProfile() {
287                validateSealed();
288                if (myProfileDef != null) {
289                        return myProfileDef;
290                }
291
292                IBaseResource retVal = myContext.getVersion().generateProfile(this, null);
293                myProfileDef = retVal;
294
295                return retVal;
296        }
297
298        public synchronized IBaseResource toProfile(String theServerBase) {
299                validateSealed();
300                if (myProfileDef != null) {
301                        return myProfileDef;
302                }
303
304                IBaseResource retVal = myContext.getVersion().generateProfile(this, theServerBase);
305                myProfileDef = retVal;
306
307                return retVal;
308        }
309}