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.util;
021
022import ca.uhn.fhir.context.BaseRuntimeChildDefinition;
023import ca.uhn.fhir.context.FhirContext;
024import ca.uhn.fhir.context.FhirVersionEnum;
025import ca.uhn.fhir.context.RuntimeResourceDefinition;
026import ca.uhn.fhir.context.RuntimeSearchParam;
027import ca.uhn.fhir.i18n.Msg;
028import jakarta.annotation.Nonnull;
029import jakarta.annotation.Nullable;
030import org.apache.commons.collections4.CollectionUtils;
031import org.apache.commons.lang3.Validate;
032import org.hl7.fhir.instance.model.api.IBase;
033import org.hl7.fhir.instance.model.api.IBaseResource;
034import org.hl7.fhir.instance.model.api.IPrimitiveType;
035
036import java.util.ArrayList;
037import java.util.Collections;
038import java.util.List;
039import java.util.Optional;
040import java.util.Set;
041import java.util.stream.Collectors;
042
043public class SearchParameterUtil {
044
045        public static List<String> getBaseAsStrings(FhirContext theContext, IBaseResource theResource) {
046                Validate.notNull(theContext, "theContext must not be null");
047                Validate.notNull(theResource, "theResource must not be null");
048                RuntimeResourceDefinition def = theContext.getResourceDefinition(theResource);
049
050                BaseRuntimeChildDefinition base = def.getChildByName("base");
051                List<IBase> baseValues = base.getAccessor().getValues(theResource);
052                List<String> retVal = new ArrayList<>();
053                for (IBase next : baseValues) {
054                        IPrimitiveType<?> nextPrimitive = (IPrimitiveType<?>) next;
055                        retVal.add(nextPrimitive.getValueAsString());
056                }
057
058                return retVal;
059        }
060
061        /**
062         * Given the resource type, fetch its patient-based search parameter name
063         * 1. Attempt to find one called 'patient'
064         * 2. If that fails, find one called 'subject'
065         * 3. If that fails, find one by Patient Compartment.
066         * 3.1 If that returns exactly 1 result then return it
067         * 3.2 If that doesn't return exactly 1 result and is R4, fall to 3.3, otherwise, 3.5
068         * 3.3 If that returns >1 result, throw an error
069         * 3.4 If that returns 1 result, return it
070         * 3.5 Find the search parameters by patient compartment using the R4 FHIR path, and return it if there is 1 result,
071         *     otherwise, fall to 3.3
072         */
073        public static Optional<RuntimeSearchParam> getOnlyPatientSearchParamForResourceType(
074                        FhirContext theFhirContext, String theResourceType) {
075                RuntimeSearchParam myPatientSearchParam = null;
076                RuntimeResourceDefinition runtimeResourceDefinition = theFhirContext.getResourceDefinition(theResourceType);
077                myPatientSearchParam = runtimeResourceDefinition.getSearchParam("patient");
078                if (myPatientSearchParam == null) {
079                        myPatientSearchParam = runtimeResourceDefinition.getSearchParam("subject");
080                        if (myPatientSearchParam == null) {
081                                final List<RuntimeSearchParam> searchParamsForCurrentVersion =
082                                                runtimeResourceDefinition.getSearchParamsForCompartmentName("Patient");
083                                final List<RuntimeSearchParam> searchParamsToUse;
084                                // We want to handle a narrow code path in which attempting to process SearchParameters for a non-R4
085                                // resource would have failed, and instead make another attempt to process them with the R4-equivalent
086                                // FHIR path.
087                                if (FhirVersionEnum.R4 == theFhirContext.getVersion().getVersion()
088                                                || searchParamsForCurrentVersion.size() == 1) {
089                                        searchParamsToUse = searchParamsForCurrentVersion;
090                                } else {
091                                        searchParamsToUse =
092                                                        checkR4PatientCompartmentForMatchingSearchParam(runtimeResourceDefinition, theResourceType);
093                                }
094                                myPatientSearchParam =
095                                                validateSearchParamsAndReturnOnlyOne(runtimeResourceDefinition, searchParamsToUse);
096                        }
097                }
098                return Optional.of(myPatientSearchParam);
099        }
100
101        @Nonnull
102        private static List<RuntimeSearchParam> checkR4PatientCompartmentForMatchingSearchParam(
103                        RuntimeResourceDefinition theRuntimeResourceDefinition, String theResourceType) {
104                final RuntimeSearchParam patientSearchParamForR4 =
105                                FhirContext.forR4Cached().getResourceDefinition(theResourceType).getSearchParam("patient");
106
107                return Optional.ofNullable(patientSearchParamForR4)
108                                .map(patientSearchParamForR4NonNull ->
109                                                theRuntimeResourceDefinition.getSearchParamsForCompartmentName("Patient").stream()
110                                                                .filter(searchParam -> searchParam.getPath() != null)
111                                                                .filter(searchParam ->
112                                                                                searchParam.getPath().equals(patientSearchParamForR4NonNull.getPath()))
113                                                                .collect(Collectors.toList()))
114                                .orElse(Collections.emptyList());
115        }
116
117        /**
118         * Given the resource type, fetch all its patient-based search parameter name that's available
119         */
120        public static Set<String> getPatientSearchParamsForResourceType(
121                        FhirContext theFhirContext, String theResourceType) {
122                RuntimeResourceDefinition runtimeResourceDefinition = theFhirContext.getResourceDefinition(theResourceType);
123
124                List<RuntimeSearchParam> searchParams =
125                                new ArrayList<>(runtimeResourceDefinition.getSearchParamsForCompartmentName("Patient"));
126                // add patient search parameter for resources that's not in the compartment
127                RuntimeSearchParam myPatientSearchParam = runtimeResourceDefinition.getSearchParam("patient");
128                if (myPatientSearchParam != null) {
129                        searchParams.add(myPatientSearchParam);
130                }
131                RuntimeSearchParam mySubjectSearchParam = runtimeResourceDefinition.getSearchParam("subject");
132                if (mySubjectSearchParam != null) {
133                        searchParams.add(mySubjectSearchParam);
134                }
135                if (CollectionUtils.isEmpty(searchParams)) {
136                        String errorMessage = String.format(
137                                        "Resource type [%s] is not eligible for this type of export, as it contains no Patient compartment, and no `patient` or `subject` search parameter",
138                                        runtimeResourceDefinition.getId());
139                        throw new IllegalArgumentException(Msg.code(2222) + errorMessage);
140                }
141                // deduplicate list of searchParams and get their names
142                return searchParams.stream().map(RuntimeSearchParam::getName).collect(Collectors.toSet());
143        }
144
145        /**
146         * Search the resource definition for a compartment named 'patient' and return its related Search Parameter.
147         */
148        public static RuntimeSearchParam getOnlyPatientCompartmentRuntimeSearchParam(
149                        FhirContext theFhirContext, String theResourceType) {
150                RuntimeResourceDefinition resourceDefinition = theFhirContext.getResourceDefinition(theResourceType);
151                return getOnlyPatientCompartmentRuntimeSearchParam(resourceDefinition);
152        }
153
154        public static RuntimeSearchParam getOnlyPatientCompartmentRuntimeSearchParam(
155                        RuntimeResourceDefinition runtimeResourceDefinition) {
156                return validateSearchParamsAndReturnOnlyOne(
157                                runtimeResourceDefinition, runtimeResourceDefinition.getSearchParamsForCompartmentName("Patient"));
158        }
159
160        public static RuntimeSearchParam getOnlyPatientCompartmentRuntimeSearchParam(
161                        RuntimeResourceDefinition runtimeResourceDefinition, List<RuntimeSearchParam> theSearchParams) {
162                return validateSearchParamsAndReturnOnlyOne(runtimeResourceDefinition, theSearchParams);
163        }
164
165        @Nonnull
166        private static RuntimeSearchParam validateSearchParamsAndReturnOnlyOne(
167                        RuntimeResourceDefinition theRuntimeResourceDefinition, List<RuntimeSearchParam> theSearchParams) {
168                final RuntimeSearchParam patientSearchParam;
169                if (CollectionUtils.isEmpty(theSearchParams)) {
170                        String errorMessage = String.format(
171                                        "Resource type [%s] for ID [%s] and version: [%s] is not eligible for this type of export, as it contains no Patient compartment, and no `patient` or `subject` search parameter",
172                                        theRuntimeResourceDefinition.getName(),
173                                        theRuntimeResourceDefinition.getId(),
174                                        theRuntimeResourceDefinition.getStructureVersion());
175                        throw new IllegalArgumentException(Msg.code(1774) + errorMessage);
176                } else if (theSearchParams.size() == 1) {
177                        patientSearchParam = theSearchParams.get(0);
178                } else {
179                        String errorMessage = String.format(
180                                        "Resource type [%s] for ID [%s] and version: [%s] has more than one Search Param which references a patient compartment. We are unable to disambiguate which patient search parameter we should be searching by.",
181                                        theRuntimeResourceDefinition.getName(),
182                                        theRuntimeResourceDefinition.getId(),
183                                        theRuntimeResourceDefinition.getStructureVersion());
184                        throw new IllegalArgumentException(Msg.code(1775) + errorMessage);
185                }
186                return patientSearchParam;
187        }
188
189        public static List<RuntimeSearchParam> getAllPatientCompartmentRuntimeSearchParamsForResourceType(
190                        FhirContext theFhirContext, String theResourceType) {
191                RuntimeResourceDefinition runtimeResourceDefinition = theFhirContext.getResourceDefinition(theResourceType);
192                return getAllPatientCompartmentRuntimeSearchParams(runtimeResourceDefinition);
193        }
194
195        public static List<RuntimeSearchParam> getAllPatientCompartmenRuntimeSearchParams(FhirContext theFhirContext) {
196                return theFhirContext.getResourceTypes().stream()
197                                .flatMap(type ->
198                                                getAllPatientCompartmentRuntimeSearchParamsForResourceType(theFhirContext, type).stream())
199                                .collect(Collectors.toList());
200        }
201
202        public static Set<String> getAllResourceTypesThatAreInPatientCompartment(FhirContext theFhirContext) {
203                return theFhirContext.getResourceTypes().stream()
204                                .filter(type -> CollectionUtils.isNotEmpty(
205                                                getAllPatientCompartmentRuntimeSearchParamsForResourceType(theFhirContext, type)))
206                                .collect(Collectors.toSet());
207        }
208
209        private static List<RuntimeSearchParam> getAllPatientCompartmentRuntimeSearchParams(
210                        RuntimeResourceDefinition theRuntimeResourceDefinition) {
211                List<RuntimeSearchParam> patient = theRuntimeResourceDefinition.getSearchParamsForCompartmentName("Patient");
212                return patient;
213        }
214
215        /**
216         * Return true if any search parameter in the resource can point at a patient, false otherwise
217         */
218        public static boolean isResourceTypeInPatientCompartment(FhirContext theFhirContext, String theResourceType) {
219                RuntimeResourceDefinition runtimeResourceDefinition = theFhirContext.getResourceDefinition(theResourceType);
220                return CollectionUtils.isNotEmpty(getAllPatientCompartmentRuntimeSearchParams(runtimeResourceDefinition));
221        }
222
223        @Nullable
224        public static String getCode(FhirContext theContext, IBaseResource theResource) {
225                return getStringChild(theContext, theResource, "code");
226        }
227
228        @Nullable
229        public static String getURL(FhirContext theContext, IBaseResource theResource) {
230                return getStringChild(theContext, theResource, "url");
231        }
232
233        @Nullable
234        public static String getExpression(FhirContext theFhirContext, IBaseResource theResource) {
235                return getStringChild(theFhirContext, theResource, "expression");
236        }
237
238        private static String getStringChild(FhirContext theFhirContext, IBaseResource theResource, String theChildName) {
239                Validate.notNull(theFhirContext, "theContext must not be null");
240                Validate.notNull(theResource, "theResource must not be null");
241                RuntimeResourceDefinition def = theFhirContext.getResourceDefinition(theResource);
242
243                BaseRuntimeChildDefinition base = def.getChildByName(theChildName);
244                return base.getAccessor()
245                                .getFirstValueOrNull(theResource)
246                                .map(t -> ((IPrimitiveType<?>) t))
247                                .map(t -> t.getValueAsString())
248                                .orElse(null);
249        }
250
251        public static String stripModifier(String theSearchParam) {
252                String retval;
253                int colonIndex = theSearchParam.indexOf(":");
254                if (colonIndex == -1) {
255                        retval = theSearchParam;
256                } else {
257                        retval = theSearchParam.substring(0, colonIndex);
258                }
259                return retval;
260        }
261}