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.BaseRuntimeElementDefinition;
023import ca.uhn.fhir.context.ConfigurationException;
024import ca.uhn.fhir.context.FhirContext;
025import ca.uhn.fhir.i18n.Msg;
026import ca.uhn.fhir.interceptor.api.IInterceptorBroadcaster;
027import ca.uhn.fhir.model.api.Include;
028import ca.uhn.fhir.model.api.TagList;
029import ca.uhn.fhir.model.api.annotation.Description;
030import ca.uhn.fhir.rest.annotation.At;
031import ca.uhn.fhir.rest.annotation.ConditionalUrlParam;
032import ca.uhn.fhir.rest.annotation.Count;
033import ca.uhn.fhir.rest.annotation.Elements;
034import ca.uhn.fhir.rest.annotation.GraphQLQueryBody;
035import ca.uhn.fhir.rest.annotation.GraphQLQueryUrl;
036import ca.uhn.fhir.rest.annotation.IdParam;
037import ca.uhn.fhir.rest.annotation.IncludeParam;
038import ca.uhn.fhir.rest.annotation.Offset;
039import ca.uhn.fhir.rest.annotation.Operation;
040import ca.uhn.fhir.rest.annotation.OperationParam;
041import ca.uhn.fhir.rest.annotation.OptionalParam;
042import ca.uhn.fhir.rest.annotation.Patch;
043import ca.uhn.fhir.rest.annotation.RawParam;
044import ca.uhn.fhir.rest.annotation.RequiredParam;
045import ca.uhn.fhir.rest.annotation.ResourceParam;
046import ca.uhn.fhir.rest.annotation.ServerBase;
047import ca.uhn.fhir.rest.annotation.Since;
048import ca.uhn.fhir.rest.annotation.Sort;
049import ca.uhn.fhir.rest.annotation.TransactionParam;
050import ca.uhn.fhir.rest.annotation.Validate;
051import ca.uhn.fhir.rest.api.Constants;
052import ca.uhn.fhir.rest.api.EncodingEnum;
053import ca.uhn.fhir.rest.api.PatchTypeEnum;
054import ca.uhn.fhir.rest.api.SearchContainedModeEnum;
055import ca.uhn.fhir.rest.api.SearchTotalModeEnum;
056import ca.uhn.fhir.rest.api.SummaryEnum;
057import ca.uhn.fhir.rest.api.ValidationModeEnum;
058import ca.uhn.fhir.rest.api.server.RequestDetails;
059import ca.uhn.fhir.rest.param.binder.CollectionBinder;
060import ca.uhn.fhir.rest.server.method.OperationParameter.IOperationParamConverter;
061import ca.uhn.fhir.rest.server.method.ResourceParameter.Mode;
062import ca.uhn.fhir.rest.server.servlet.ServletRequestDetails;
063import ca.uhn.fhir.util.ParametersUtil;
064import ca.uhn.fhir.util.ReflectionUtil;
065import jakarta.servlet.ServletRequest;
066import jakarta.servlet.ServletResponse;
067import org.hl7.fhir.instance.model.api.IBaseResource;
068import org.hl7.fhir.instance.model.api.IPrimitiveType;
069
070import java.lang.annotation.Annotation;
071import java.lang.reflect.Method;
072import java.util.ArrayList;
073import java.util.Collection;
074import java.util.Date;
075import java.util.List;
076
077import static org.apache.commons.lang3.StringUtils.isNotBlank;
078
079public class MethodUtil {
080
081        /**
082         * Non instantiable
083         */
084        private MethodUtil() {
085                // nothing
086        }
087
088        public static void extractDescription(SearchParameter theParameter, Annotation[] theAnnotations) {
089                for (Annotation annotation : theAnnotations) {
090                        if (annotation instanceof Description) {
091                                Description desc = (Description) annotation;
092                                String description = ParametersUtil.extractDescription(desc);
093                                theParameter.setDescription(description);
094                        }
095                }
096        }
097
098        @SuppressWarnings("unchecked")
099        public static List<IParameter> getResourceParameters(
100                        final FhirContext theContext, Method theMethod, Object theProvider) {
101                List<IParameter> parameters = new ArrayList<>();
102
103                Class<?>[] parameterTypes = theMethod.getParameterTypes();
104                int paramIndex = 0;
105                for (Annotation[] nextParameterAnnotations : theMethod.getParameterAnnotations()) {
106
107                        IParameter param = null;
108                        Class<?> declaredParameterType = parameterTypes[paramIndex];
109                        Class<?> parameterType = declaredParameterType;
110                        Class<? extends java.util.Collection<?>> outerCollectionType = null;
111                        Class<? extends java.util.Collection<?>> innerCollectionType = null;
112                        if (TagList.class.isAssignableFrom(parameterType)) {
113                                // TagList is handled directly within the method bindings
114                                param = new NullParameter();
115                        } else {
116                                if (Collection.class.isAssignableFrom(parameterType)) {
117                                        innerCollectionType = (Class<? extends java.util.Collection<?>>) parameterType;
118                                        parameterType = ReflectionUtil.getGenericCollectionTypeOfMethodParameter(theMethod, paramIndex);
119                                        if (parameterType == null && theMethod.getDeclaringClass().isSynthetic()) {
120                                                try {
121                                                        theMethod = theMethod
122                                                                        .getDeclaringClass()
123                                                                        .getSuperclass()
124                                                                        .getMethod(theMethod.getName(), parameterTypes);
125                                                        parameterType =
126                                                                        ReflectionUtil.getGenericCollectionTypeOfMethodParameter(theMethod, paramIndex);
127                                                } catch (NoSuchMethodException e) {
128                                                        throw new ConfigurationException(Msg.code(400) + "A method with name '"
129                                                                        + theMethod.getName() + "' does not exist for super class '"
130                                                                        + theMethod.getDeclaringClass().getSuperclass() + "'");
131                                                }
132                                        }
133                                        declaredParameterType = parameterType;
134                                }
135                                if (Collection.class.isAssignableFrom(parameterType)) {
136                                        outerCollectionType = innerCollectionType;
137                                        innerCollectionType = (Class<? extends java.util.Collection<?>>) parameterType;
138                                        parameterType = ReflectionUtil.getGenericCollectionTypeOfMethodParameter(theMethod, paramIndex);
139                                        declaredParameterType = parameterType;
140                                }
141                                if (Collection.class.isAssignableFrom(parameterType)) {
142                                        throw new ConfigurationException(
143                                                        Msg.code(401) + "Argument #" + paramIndex + " of Method '" + theMethod.getName()
144                                                                        + "' in type '"
145                                                                        + theMethod.getDeclaringClass().getCanonicalName()
146                                                                        + "' is of an invalid generic type (can not be a collection of a collection of a collection)");
147                                }
148
149                                /*
150                                 * If the user is trying to bind IPrimitiveType they are probably
151                                 * trying to write code that is compatible across versions of FHIR.
152                                 * We'll try and come up with an appropriate subtype to give
153                                 * them.
154                                 *
155                                 * This gets tested in HistoryR4Test
156                                 */
157                                if (IPrimitiveType.class.equals(parameterType)) {
158                                        Class<?> genericType =
159                                                        ReflectionUtil.getGenericCollectionTypeOfMethodParameter(theMethod, paramIndex);
160                                        if (Date.class.equals(genericType)) {
161                                                BaseRuntimeElementDefinition<?> dateTimeDef = theContext.getElementDefinition("dateTime");
162                                                parameterType = dateTimeDef.getImplementingClass();
163                                        } else if (String.class.equals(genericType) || genericType == null) {
164                                                BaseRuntimeElementDefinition<?> dateTimeDef = theContext.getElementDefinition("string");
165                                                parameterType = dateTimeDef.getImplementingClass();
166                                        }
167                                }
168                        }
169
170                        if (ServletRequest.class.isAssignableFrom(parameterType)) {
171                                param = new ServletRequestParameter();
172                        } else if (ServletResponse.class.isAssignableFrom(parameterType)) {
173                                param = new ServletResponseParameter();
174                        } else if (parameterType.equals(RequestDetails.class)
175                                        || parameterType.equals(ServletRequestDetails.class)) {
176                                param = new RequestDetailsParameter();
177                        } else if (parameterType.equals(IInterceptorBroadcaster.class)) {
178                                param = new InterceptorBroadcasterParameter();
179                        } else if (parameterType.equals(SummaryEnum.class)) {
180                                param = new SummaryEnumParameter();
181                        } else if (parameterType.equals(PatchTypeEnum.class)) {
182                                param = new PatchTypeParameter();
183                        } else if (parameterType.equals(SearchContainedModeEnum.class)) {
184                                param = new SearchContainedModeParameter();
185                        } else if (parameterType.equals(SearchTotalModeEnum.class)) {
186                                param = new SearchTotalModeParameter();
187                        } else {
188                                for (int i = 0; i < nextParameterAnnotations.length && param == null; i++) {
189                                        Annotation nextAnnotation = nextParameterAnnotations[i];
190
191                                        if (nextAnnotation instanceof RequiredParam) {
192                                                SearchParameter parameter = new SearchParameter();
193                                                parameter.setName(((RequiredParam) nextAnnotation).name());
194                                                parameter.setRequired(true);
195                                                parameter.setDeclaredTypes(((RequiredParam) nextAnnotation).targetTypes());
196                                                parameter.setCompositeTypes(((RequiredParam) nextAnnotation).compositeTypes());
197                                                parameter.setChainLists(
198                                                                ((RequiredParam) nextAnnotation).chainWhitelist(),
199                                                                ((RequiredParam) nextAnnotation).chainBlacklist());
200                                                parameter.setType(theContext, parameterType, innerCollectionType, outerCollectionType);
201                                                MethodUtil.extractDescription(parameter, nextParameterAnnotations);
202                                                param = parameter;
203                                        } else if (nextAnnotation instanceof OptionalParam) {
204                                                SearchParameter parameter = new SearchParameter();
205                                                parameter.setName(((OptionalParam) nextAnnotation).name());
206                                                parameter.setRequired(false);
207                                                parameter.setDeclaredTypes(((OptionalParam) nextAnnotation).targetTypes());
208                                                parameter.setCompositeTypes(((OptionalParam) nextAnnotation).compositeTypes());
209                                                parameter.setChainLists(
210                                                                ((OptionalParam) nextAnnotation).chainWhitelist(),
211                                                                ((OptionalParam) nextAnnotation).chainBlacklist());
212                                                parameter.setType(theContext, parameterType, innerCollectionType, outerCollectionType);
213                                                MethodUtil.extractDescription(parameter, nextParameterAnnotations);
214                                                param = parameter;
215                                        } else if (nextAnnotation instanceof RawParam) {
216                                                param = new RawParamsParameter(parameters);
217                                        } else if (nextAnnotation instanceof IncludeParam) {
218                                                Class<? extends Collection<Include>> instantiableCollectionType;
219                                                Class<?> specType;
220
221                                                if (parameterType == String.class) {
222                                                        instantiableCollectionType = null;
223                                                        specType = String.class;
224                                                } else if ((parameterType != Include.class)
225                                                                || innerCollectionType == null
226                                                                || outerCollectionType != null) {
227                                                        throw new ConfigurationException(Msg.code(402) + "Method '" + theMethod.getName()
228                                                                        + "' is annotated with @" + IncludeParam.class.getSimpleName()
229                                                                        + " but has a type other than Collection<" + Include.class.getSimpleName() + ">");
230                                                } else {
231                                                        instantiableCollectionType = (Class<? extends Collection<Include>>)
232                                                                        CollectionBinder.getInstantiableCollectionType(
233                                                                                        innerCollectionType, "Method '" + theMethod.getName() + "'");
234                                                        specType = parameterType;
235                                                }
236
237                                                param = new IncludeParameter(
238                                                                (IncludeParam) nextAnnotation, instantiableCollectionType, specType);
239                                        } else if (nextAnnotation instanceof ResourceParam) {
240                                                Mode mode;
241                                                if (IBaseResource.class.isAssignableFrom(parameterType)) {
242                                                        mode = Mode.RESOURCE;
243                                                } else if (String.class.equals(parameterType)) {
244                                                        mode = ResourceParameter.Mode.BODY;
245                                                } else if (byte[].class.equals(parameterType)) {
246                                                        mode = ResourceParameter.Mode.BODY_BYTE_ARRAY;
247                                                } else if (EncodingEnum.class.equals(parameterType)) {
248                                                        mode = Mode.ENCODING;
249                                                } else {
250                                                        StringBuilder b = new StringBuilder();
251                                                        b.append("Method '");
252                                                        b.append(theMethod.getName());
253                                                        b.append("' is annotated with @");
254                                                        b.append(ResourceParam.class.getSimpleName());
255                                                        b.append(" but has a type that is not an implementation of ");
256                                                        b.append(IBaseResource.class.getCanonicalName());
257                                                        b.append(" or String or byte[]");
258                                                        throw new ConfigurationException(Msg.code(403) + b.toString());
259                                                }
260                                                boolean methodIsOperation = theMethod.getAnnotation(Operation.class) != null;
261                                                boolean methodIsPatch = theMethod.getAnnotation(Patch.class) != null;
262                                                param = new ResourceParameter(
263                                                                (Class<? extends IBaseResource>) parameterType,
264                                                                theProvider,
265                                                                mode,
266                                                                methodIsOperation,
267                                                                methodIsPatch);
268                                        } else if (nextAnnotation instanceof IdParam) {
269                                                param = new NullParameter();
270                                        } else if (nextAnnotation instanceof ServerBase) {
271                                                param = new ServerBaseParamBinder();
272                                        } else if (nextAnnotation instanceof Elements) {
273                                                param = new ElementsParameter();
274                                        } else if (nextAnnotation instanceof Since) {
275                                                param = new SinceParameter();
276                                                ((SinceParameter) param)
277                                                                .setType(theContext, parameterType, innerCollectionType, outerCollectionType);
278                                        } else if (nextAnnotation instanceof At) {
279                                                param = new AtParameter();
280                                                ((AtParameter) param)
281                                                                .setType(theContext, parameterType, innerCollectionType, outerCollectionType);
282                                        } else if (nextAnnotation instanceof Count) {
283                                                param = new CountParameter();
284                                        } else if (nextAnnotation instanceof Offset) {
285                                                param = new OffsetParameter();
286                                        } else if (nextAnnotation instanceof GraphQLQueryUrl) {
287                                                param = new GraphQLQueryUrlParameter();
288                                        } else if (nextAnnotation instanceof GraphQLQueryBody) {
289                                                param = new GraphQLQueryBodyParameter();
290                                        } else if (nextAnnotation instanceof Sort) {
291                                                param = new SortParameter(theContext);
292                                        } else if (nextAnnotation instanceof TransactionParam) {
293                                                param = new TransactionParameter(theContext);
294                                        } else if (nextAnnotation instanceof ConditionalUrlParam) {
295                                                param = new ConditionalParamBinder(((ConditionalUrlParam) nextAnnotation).supportsMultiple());
296                                        } else if (nextAnnotation instanceof OperationParam) {
297                                                Operation op = theMethod.getAnnotation(Operation.class);
298                                                if (op == null) {
299                                                        throw new ConfigurationException(Msg.code(404)
300                                                                        + "@OperationParam detected on method that is not annotated with @Operation: "
301                                                                        + theMethod.toGenericString());
302                                                }
303
304                                                OperationParam operationParam = (OperationParam) nextAnnotation;
305                                                String description = ParametersUtil.extractDescription(nextParameterAnnotations);
306                                                List<String> examples = ParametersUtil.extractExamples(nextParameterAnnotations);
307                                                ;
308                                                param = new OperationParameter(
309                                                                theContext,
310                                                                op.name(),
311                                                                operationParam.name(),
312                                                                operationParam.min(),
313                                                                operationParam.max(),
314                                                                description,
315                                                                examples);
316                                                if (isNotBlank(operationParam.typeName())) {
317                                                        BaseRuntimeElementDefinition<?> elementDefinition =
318                                                                        theContext.getElementDefinition(operationParam.typeName());
319                                                        if (elementDefinition == null) {
320                                                                elementDefinition = theContext.getResourceDefinition(operationParam.typeName());
321                                                        }
322                                                        org.apache.commons.lang3.Validate.notNull(
323                                                                        elementDefinition,
324                                                                        "Unknown type name in @OperationParam: typeName=\"%s\"",
325                                                                        operationParam.typeName());
326
327                                                        Class<?> newParameterType = elementDefinition.getImplementingClass();
328                                                        if (!declaredParameterType.isAssignableFrom(newParameterType)) {
329                                                                throw new ConfigurationException(Msg.code(405) + "Non assignable parameter typeName=\""
330                                                                                + operationParam.typeName() + "\" specified on method " + theMethod);
331                                                        }
332                                                        parameterType = newParameterType;
333                                                }
334                                        } else if (nextAnnotation instanceof Validate.Mode) {
335                                                if (parameterType.equals(ValidationModeEnum.class) == false) {
336                                                        throw new ConfigurationException(Msg.code(406) + "Parameter annotated with @"
337                                                                        + Validate.class.getSimpleName() + "." + Validate.Mode.class.getSimpleName()
338                                                                        + " must be of type " + ValidationModeEnum.class.getName());
339                                                }
340                                                String description = ParametersUtil.extractDescription(nextParameterAnnotations);
341                                                List<String> examples = ParametersUtil.extractExamples(nextParameterAnnotations);
342                                                param = new OperationParameter(
343                                                                                theContext,
344                                                                                Constants.EXTOP_VALIDATE,
345                                                                                Constants.EXTOP_VALIDATE_MODE,
346                                                                                0,
347                                                                                1,
348                                                                                description,
349                                                                                examples)
350                                                                .setConverter(new IOperationParamConverter() {
351                                                                        @Override
352                                                                        public Object incomingServer(Object theObject) {
353                                                                                if (isNotBlank(theObject.toString())) {
354                                                                                        ValidationModeEnum retVal =
355                                                                                                        ValidationModeEnum.forCode(theObject.toString());
356                                                                                        if (retVal == null) {
357                                                                                                OperationParameter.throwInvalidMode(theObject.toString());
358                                                                                        }
359                                                                                        return retVal;
360                                                                                }
361                                                                                return null;
362                                                                        }
363
364                                                                        @Override
365                                                                        public Object outgoingClient(Object theObject) {
366                                                                                return ParametersUtil.createString(
367                                                                                                theContext, ((ValidationModeEnum) theObject).getCode());
368                                                                        }
369                                                                });
370                                        } else if (nextAnnotation instanceof Validate.Profile) {
371                                                if (parameterType.equals(String.class) == false) {
372                                                        throw new ConfigurationException(Msg.code(407) + "Parameter annotated with @"
373                                                                        + Validate.class.getSimpleName() + "." + Validate.Profile.class.getSimpleName()
374                                                                        + " must be of type " + String.class.getName());
375                                                }
376                                                String description = ParametersUtil.extractDescription(nextParameterAnnotations);
377                                                List<String> examples = ParametersUtil.extractExamples(nextParameterAnnotations);
378                                                param = new OperationParameter(
379                                                                                theContext,
380                                                                                Constants.EXTOP_VALIDATE,
381                                                                                Constants.EXTOP_VALIDATE_PROFILE,
382                                                                                0,
383                                                                                1,
384                                                                                description,
385                                                                                examples)
386                                                                .setConverter(new IOperationParamConverter() {
387                                                                        @Override
388                                                                        public Object incomingServer(Object theObject) {
389                                                                                return theObject.toString();
390                                                                        }
391
392                                                                        @Override
393                                                                        public Object outgoingClient(Object theObject) {
394                                                                                return ParametersUtil.createString(theContext, theObject.toString());
395                                                                        }
396                                                                });
397                                        } else {
398                                                continue;
399                                        }
400                                }
401                        }
402
403                        if (param == null) {
404                                throw new ConfigurationException(
405                                                Msg.code(408) + "Parameter #" + ((paramIndex + 1)) + "/" + (parameterTypes.length)
406                                                                + " of method '" + theMethod.getName() + "' on type '"
407                                                                + theMethod.getDeclaringClass().getCanonicalName()
408                                                                + "' has no recognized FHIR interface parameter nextParameterAnnotations. Don't know how to handle this parameter");
409                        }
410
411                        param.initializeTypes(theMethod, outerCollectionType, innerCollectionType, parameterType);
412                        parameters.add(param);
413
414                        paramIndex++;
415                }
416                return parameters;
417        }
418}