001/*
002 * #%L
003 * HAPI FHIR - Server Framework
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.rest.server.method;
021
022import ca.uhn.fhir.context.ConfigurationException;
023import ca.uhn.fhir.context.FhirContext;
024import ca.uhn.fhir.context.RuntimePrimitiveDatatypeDefinition;
025import ca.uhn.fhir.i18n.Msg;
026import ca.uhn.fhir.model.api.IQueryParameterAnd;
027import ca.uhn.fhir.model.api.IQueryParameterOr;
028import ca.uhn.fhir.model.api.IQueryParameterType;
029import ca.uhn.fhir.model.base.composite.BaseIdentifierDt;
030import ca.uhn.fhir.model.base.composite.BaseQuantityDt;
031import ca.uhn.fhir.model.primitive.StringDt;
032import ca.uhn.fhir.rest.annotation.OptionalParam;
033import ca.uhn.fhir.rest.api.Constants;
034import ca.uhn.fhir.rest.api.QualifiedParamList;
035import ca.uhn.fhir.rest.api.RestSearchParameterTypeEnum;
036import ca.uhn.fhir.rest.param.CompositeAndListParam;
037import ca.uhn.fhir.rest.param.CompositeOrListParam;
038import ca.uhn.fhir.rest.param.CompositeParam;
039import ca.uhn.fhir.rest.param.DateAndListParam;
040import ca.uhn.fhir.rest.param.DateOrListParam;
041import ca.uhn.fhir.rest.param.DateParam;
042import ca.uhn.fhir.rest.param.DateRangeParam;
043import ca.uhn.fhir.rest.param.HasAndListParam;
044import ca.uhn.fhir.rest.param.HasOrListParam;
045import ca.uhn.fhir.rest.param.HasParam;
046import ca.uhn.fhir.rest.param.NumberAndListParam;
047import ca.uhn.fhir.rest.param.NumberOrListParam;
048import ca.uhn.fhir.rest.param.NumberParam;
049import ca.uhn.fhir.rest.param.QuantityAndListParam;
050import ca.uhn.fhir.rest.param.QuantityOrListParam;
051import ca.uhn.fhir.rest.param.QuantityParam;
052import ca.uhn.fhir.rest.param.ReferenceAndListParam;
053import ca.uhn.fhir.rest.param.ReferenceOrListParam;
054import ca.uhn.fhir.rest.param.ReferenceParam;
055import ca.uhn.fhir.rest.param.SpecialAndListParam;
056import ca.uhn.fhir.rest.param.SpecialOrListParam;
057import ca.uhn.fhir.rest.param.SpecialParam;
058import ca.uhn.fhir.rest.param.StringAndListParam;
059import ca.uhn.fhir.rest.param.StringOrListParam;
060import ca.uhn.fhir.rest.param.StringParam;
061import ca.uhn.fhir.rest.param.TokenAndListParam;
062import ca.uhn.fhir.rest.param.TokenOrListParam;
063import ca.uhn.fhir.rest.param.TokenParam;
064import ca.uhn.fhir.rest.param.UriAndListParam;
065import ca.uhn.fhir.rest.param.UriOrListParam;
066import ca.uhn.fhir.rest.param.UriParam;
067import ca.uhn.fhir.rest.param.binder.CalendarBinder;
068import ca.uhn.fhir.rest.param.binder.DateBinder;
069import ca.uhn.fhir.rest.param.binder.FhirPrimitiveBinder;
070import ca.uhn.fhir.rest.param.binder.IParamBinder;
071import ca.uhn.fhir.rest.param.binder.QueryParameterAndBinder;
072import ca.uhn.fhir.rest.param.binder.QueryParameterOrBinder;
073import ca.uhn.fhir.rest.param.binder.QueryParameterTypeBinder;
074import ca.uhn.fhir.rest.param.binder.StringBinder;
075import ca.uhn.fhir.rest.server.exceptions.InternalErrorException;
076import ca.uhn.fhir.rest.server.exceptions.InvalidRequestException;
077import ca.uhn.fhir.util.ReflectionUtil;
078import org.apache.commons.lang3.builder.ToStringBuilder;
079import org.hl7.fhir.instance.model.api.IBaseResource;
080import org.hl7.fhir.instance.model.api.IPrimitiveType;
081
082import java.util.ArrayList;
083import java.util.Arrays;
084import java.util.Calendar;
085import java.util.Collection;
086import java.util.Collections;
087import java.util.Date;
088import java.util.HashMap;
089import java.util.HashSet;
090import java.util.List;
091import java.util.Set;
092
093public class SearchParameter extends BaseQueryParameter {
094
095        private static final String EMPTY_STRING = "";
096        private static final HashMap<RestSearchParameterTypeEnum, Set<String>> ourParamQualifiers;
097        private static final HashMap<Class<?>, RestSearchParameterTypeEnum> ourParamTypes;
098        static final String QUALIFIER_ANY_TYPE = ":*";
099
100        static {
101                ourParamTypes = new HashMap<>();
102                ourParamQualifiers = new HashMap<>();
103
104                ourParamTypes.put(StringParam.class, RestSearchParameterTypeEnum.STRING);
105                ourParamTypes.put(StringOrListParam.class, RestSearchParameterTypeEnum.STRING);
106                ourParamTypes.put(StringAndListParam.class, RestSearchParameterTypeEnum.STRING);
107                ourParamQualifiers.put(RestSearchParameterTypeEnum.STRING, Set.of(new String[] {
108                        Constants.PARAMQUALIFIER_STRING_EXACT,
109                        Constants.PARAMQUALIFIER_STRING_CONTAINS,
110                        Constants.PARAMQUALIFIER_MISSING,
111                        EMPTY_STRING
112                }));
113
114                ourParamTypes.put(UriParam.class, RestSearchParameterTypeEnum.URI);
115                ourParamTypes.put(UriOrListParam.class, RestSearchParameterTypeEnum.URI);
116                ourParamTypes.put(UriAndListParam.class, RestSearchParameterTypeEnum.URI);
117                // TODO: are these right for URI?
118                ourParamQualifiers.put(RestSearchParameterTypeEnum.URI, Set.of(new String[] {
119                        Constants.PARAMQUALIFIER_STRING_EXACT, Constants.PARAMQUALIFIER_MISSING, EMPTY_STRING
120                }));
121
122                ourParamTypes.put(TokenParam.class, RestSearchParameterTypeEnum.TOKEN);
123                ourParamTypes.put(TokenOrListParam.class, RestSearchParameterTypeEnum.TOKEN);
124                ourParamTypes.put(TokenAndListParam.class, RestSearchParameterTypeEnum.TOKEN);
125                ourParamQualifiers.put(RestSearchParameterTypeEnum.TOKEN, Set.of(new String[] {
126                        Constants.PARAMQUALIFIER_TOKEN_TEXT, Constants.PARAMQUALIFIER_MISSING, EMPTY_STRING
127                }));
128
129                ourParamTypes.put(DateParam.class, RestSearchParameterTypeEnum.DATE);
130                ourParamTypes.put(DateOrListParam.class, RestSearchParameterTypeEnum.DATE);
131                ourParamTypes.put(DateAndListParam.class, RestSearchParameterTypeEnum.DATE);
132                ourParamTypes.put(DateRangeParam.class, RestSearchParameterTypeEnum.DATE);
133                ourParamQualifiers.put(
134                                RestSearchParameterTypeEnum.DATE,
135                                Set.of(new String[] {Constants.PARAMQUALIFIER_MISSING, EMPTY_STRING}));
136
137                ourParamTypes.put(QuantityParam.class, RestSearchParameterTypeEnum.QUANTITY);
138                ourParamTypes.put(QuantityOrListParam.class, RestSearchParameterTypeEnum.QUANTITY);
139                ourParamTypes.put(QuantityAndListParam.class, RestSearchParameterTypeEnum.QUANTITY);
140                ourParamQualifiers.put(
141                                RestSearchParameterTypeEnum.QUANTITY,
142                                Set.of(new String[] {Constants.PARAMQUALIFIER_MISSING, EMPTY_STRING}));
143
144                ourParamTypes.put(NumberParam.class, RestSearchParameterTypeEnum.NUMBER);
145                ourParamTypes.put(NumberOrListParam.class, RestSearchParameterTypeEnum.NUMBER);
146                ourParamTypes.put(NumberAndListParam.class, RestSearchParameterTypeEnum.NUMBER);
147                ourParamQualifiers.put(
148                                RestSearchParameterTypeEnum.NUMBER,
149                                Set.of(new String[] {Constants.PARAMQUALIFIER_MISSING, EMPTY_STRING}));
150
151                ourParamTypes.put(ReferenceParam.class, RestSearchParameterTypeEnum.REFERENCE);
152                ourParamTypes.put(ReferenceOrListParam.class, RestSearchParameterTypeEnum.REFERENCE);
153                ourParamTypes.put(ReferenceAndListParam.class, RestSearchParameterTypeEnum.REFERENCE);
154                // --vvvv-- no empty because that gets added from OptionalParam#chainWhitelist
155                ourParamQualifiers.put(
156                                RestSearchParameterTypeEnum.REFERENCE, Set.of(new String[] {Constants.PARAMQUALIFIER_MISSING}));
157
158                ourParamTypes.put(CompositeParam.class, RestSearchParameterTypeEnum.COMPOSITE);
159                ourParamTypes.put(CompositeOrListParam.class, RestSearchParameterTypeEnum.COMPOSITE);
160                ourParamTypes.put(CompositeAndListParam.class, RestSearchParameterTypeEnum.COMPOSITE);
161                ourParamQualifiers.put(
162                                RestSearchParameterTypeEnum.COMPOSITE,
163                                Set.of(new String[] {Constants.PARAMQUALIFIER_MISSING, EMPTY_STRING}));
164
165                ourParamTypes.put(HasParam.class, RestSearchParameterTypeEnum.HAS);
166                ourParamTypes.put(HasOrListParam.class, RestSearchParameterTypeEnum.HAS);
167                ourParamTypes.put(HasAndListParam.class, RestSearchParameterTypeEnum.HAS);
168
169                ourParamTypes.put(SpecialParam.class, RestSearchParameterTypeEnum.SPECIAL);
170                ourParamTypes.put(SpecialOrListParam.class, RestSearchParameterTypeEnum.SPECIAL);
171                ourParamTypes.put(SpecialAndListParam.class, RestSearchParameterTypeEnum.SPECIAL);
172                ourParamQualifiers.put(
173                                RestSearchParameterTypeEnum.SPECIAL, Set.of(new String[] {Constants.PARAMQUALIFIER_MISSING}));
174        }
175
176        private List<Class<? extends IQueryParameterType>> myCompositeTypes = Collections.emptyList();
177        private List<Class<? extends IBaseResource>> myDeclaredTypes;
178        private String myDescription;
179        private String myName;
180        private IParamBinder<?> myParamBinder;
181        private RestSearchParameterTypeEnum myParamType;
182        private Set<String> myQualifierBlacklist;
183        private Set<String> myQualifierWhitelist;
184        private boolean myRequired;
185        private Class<?> myType;
186        private boolean mySupportsRepetition = false;
187
188        public SearchParameter() {}
189
190        public SearchParameter(String theName, boolean theRequired) {
191                this.myName = theName;
192                this.myRequired = theRequired;
193        }
194
195        /*
196         * (non-Javadoc)
197         *
198         * @see ca.uhn.fhir.rest.param.IParameter#encode(java.lang.Object)
199         */
200        @Override
201        public List<QualifiedParamList> encode(FhirContext theContext, Object theObject) throws InternalErrorException {
202                ArrayList<QualifiedParamList> retVal = new ArrayList<>();
203
204                // TODO: declaring method should probably have a generic type..
205                @SuppressWarnings("rawtypes")
206                IParamBinder paramBinder = myParamBinder;
207
208                @SuppressWarnings("unchecked")
209                List<IQueryParameterOr<?>> val = paramBinder.encode(theContext, theObject);
210                for (IQueryParameterOr<?> nextOr : val) {
211                        retVal.add(new QualifiedParamList(nextOr, theContext));
212                }
213
214                return retVal;
215        }
216
217        public List<Class<? extends IBaseResource>> getDeclaredTypes() {
218                return Collections.unmodifiableList(myDeclaredTypes);
219        }
220
221        public String getDescription() {
222                return myDescription;
223        }
224
225        /*
226         * (non-Javadoc)
227         *
228         * @see ca.uhn.fhir.rest.param.IParameter#getName()
229         */
230        @Override
231        public String getName() {
232                return myName;
233        }
234
235        @Override
236        public RestSearchParameterTypeEnum getParamType() {
237                return myParamType;
238        }
239
240        @Override
241        public Set<String> getQualifierBlacklist() {
242                return myQualifierBlacklist;
243        }
244
245        @Override
246        public Set<String> getQualifierWhitelist() {
247                return myQualifierWhitelist;
248        }
249
250        public Class<?> getType() {
251                return myType;
252        }
253
254        @Override
255        public boolean handlesMissing() {
256                return false;
257        }
258
259        @Override
260        public boolean isRequired() {
261                return myRequired;
262        }
263
264        /*
265         * (non-Javadoc)
266         *
267         * @see ca.uhn.fhir.rest.param.IParameter#parse(java.util.List)
268         */
269        @Override
270        public Object parse(FhirContext theContext, List<QualifiedParamList> theString)
271                        throws InternalErrorException, InvalidRequestException {
272                return myParamBinder.parse(theContext, getName(), theString);
273        }
274
275        public void setChainLists(String[] theChainWhitelist, String[] theChainBlacklist) {
276                myQualifierWhitelist = new HashSet<>(theChainWhitelist.length);
277                myQualifierWhitelist.add(QUALIFIER_ANY_TYPE);
278
279                for (String nextChain : theChainWhitelist) {
280                        if (nextChain.equals(OptionalParam.ALLOW_CHAIN_ANY)) {
281                                myQualifierWhitelist.add('.' + OptionalParam.ALLOW_CHAIN_ANY);
282                        } else if (nextChain.equals(EMPTY_STRING)) {
283                                myQualifierWhitelist.add(".");
284                        } else {
285                                myQualifierWhitelist.add('.' + nextChain);
286                        }
287                }
288
289                if (theChainBlacklist.length > 0) {
290                        myQualifierBlacklist = new HashSet<>(theChainBlacklist.length);
291                        for (String next : theChainBlacklist) {
292                                if (next.equals(EMPTY_STRING)) {
293                                        myQualifierBlacklist.add(EMPTY_STRING);
294                                } else {
295                                        myQualifierBlacklist.add('.' + next);
296                                }
297                        }
298                }
299        }
300
301        public void setCompositeTypes(Class<? extends IQueryParameterType>[] theCompositeTypes) {
302                myCompositeTypes = Arrays.asList(theCompositeTypes);
303        }
304
305        public void setDeclaredTypes(Class<? extends IBaseResource>[] theTypes) {
306                myDeclaredTypes = Arrays.asList(theTypes);
307        }
308
309        public void setDescription(String theDescription) {
310                myDescription = theDescription;
311        }
312
313        public void setName(String name) {
314                this.myName = name;
315        }
316
317        public void setRequired(boolean required) {
318                this.myRequired = required;
319        }
320
321        @Override
322        protected boolean supportsRepetition() {
323                return mySupportsRepetition;
324        }
325
326        @SuppressWarnings("unchecked")
327        public void setType(
328                        FhirContext theContext,
329                        final Class<?> theType,
330                        Class<? extends Collection<?>> theInnerCollectionType,
331                        Class<? extends Collection<?>> theOuterCollectionType) {
332
333                this.myType = theType;
334                if (IQueryParameterType.class.isAssignableFrom(theType)) {
335                        myParamBinder =
336                                        new QueryParameterTypeBinder((Class<? extends IQueryParameterType>) theType, myCompositeTypes);
337                } else if (IQueryParameterOr.class.isAssignableFrom(theType)) {
338                        myParamBinder =
339                                        new QueryParameterOrBinder((Class<? extends IQueryParameterOr<?>>) theType, myCompositeTypes);
340                } else if (IQueryParameterAnd.class.isAssignableFrom(theType)) {
341                        myParamBinder =
342                                        new QueryParameterAndBinder((Class<? extends IQueryParameterAnd<?>>) theType, myCompositeTypes);
343                        mySupportsRepetition = true;
344                } else if (String.class.equals(theType)) {
345                        myParamBinder = new StringBinder();
346                        myParamType = RestSearchParameterTypeEnum.STRING;
347                } else if (Date.class.equals(theType)) {
348                        myParamBinder = new DateBinder();
349                        myParamType = RestSearchParameterTypeEnum.DATE;
350                } else if (Calendar.class.equals(theType)) {
351                        myParamBinder = new CalendarBinder();
352                        myParamType = RestSearchParameterTypeEnum.DATE;
353                } else if (IPrimitiveType.class.isAssignableFrom(theType) && ReflectionUtil.isInstantiable(theType)) {
354                        RuntimePrimitiveDatatypeDefinition def = (RuntimePrimitiveDatatypeDefinition)
355                                        theContext.getElementDefinition((Class<? extends IPrimitiveType<?>>) theType);
356                        if (def.getNativeType() != null) {
357                                if (def.getNativeType().equals(Date.class)) {
358                                        myParamBinder = new FhirPrimitiveBinder((Class<IPrimitiveType<?>>) theType);
359                                        myParamType = RestSearchParameterTypeEnum.DATE;
360                                } else if (def.getNativeType().equals(String.class)) {
361                                        myParamBinder = new FhirPrimitiveBinder((Class<IPrimitiveType<?>>) theType);
362                                        myParamType = RestSearchParameterTypeEnum.STRING;
363                                }
364                        }
365                } else {
366                        throw new ConfigurationException(
367                                        Msg.code(354) + "Unsupported data type for parameter: " + theType.getCanonicalName());
368                }
369
370                RestSearchParameterTypeEnum typeEnum = ourParamTypes.get(theType);
371                if (typeEnum != null) {
372                        Set<String> builtInQualifiers = ourParamQualifiers.get(typeEnum);
373                        if (builtInQualifiers != null) {
374                                if (myQualifierWhitelist != null) {
375                                        HashSet<String> qualifierWhitelist = new HashSet<>();
376                                        qualifierWhitelist.addAll(myQualifierWhitelist);
377                                        qualifierWhitelist.addAll(builtInQualifiers);
378                                        myQualifierWhitelist = qualifierWhitelist;
379                                } else {
380                                        myQualifierWhitelist = Collections.unmodifiableSet(builtInQualifiers);
381                                }
382                        }
383                }
384
385                if (myParamType == null) {
386                        myParamType = typeEnum;
387                }
388
389                if (myParamType != null) {
390                        // ok
391                } else if (StringDt.class.isAssignableFrom(theType)) {
392                        myParamType = RestSearchParameterTypeEnum.STRING;
393                } else if (BaseIdentifierDt.class.isAssignableFrom(theType)) {
394                        myParamType = RestSearchParameterTypeEnum.TOKEN;
395                } else if (BaseQuantityDt.class.isAssignableFrom(theType)) {
396                        myParamType = RestSearchParameterTypeEnum.QUANTITY;
397                } else if (ReferenceParam.class.isAssignableFrom(theType)) {
398                        myParamType = RestSearchParameterTypeEnum.REFERENCE;
399                } else if (HasParam.class.isAssignableFrom(theType)) {
400                        myParamType = RestSearchParameterTypeEnum.STRING;
401                } else {
402                        throw new ConfigurationException(Msg.code(355) + "Unknown search parameter type: " + theType);
403                }
404
405                // NB: Once this is enabled, we should return true from handlesMissing if
406                // it's a collection theType
407                // if (theInnerCollectionType != null) {
408                // this.parser = new CollectionBinder(this.parser, theInnerCollectionType);
409                // }
410                //
411                // if (theOuterCollectionType != null) {
412                // this.parser = new CollectionBinder(this.parser, theOuterCollectionType);
413                // }
414
415        }
416
417        @Override
418        public String toString() {
419                ToStringBuilder retVal = new ToStringBuilder(this);
420                retVal.append("name", myName);
421                retVal.append("required", myRequired);
422                return retVal.toString();
423        }
424}