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.i18n.Msg;
025import ca.uhn.fhir.model.api.IResource;
026import ca.uhn.fhir.model.api.Include;
027import ca.uhn.fhir.parser.DataFormatException;
028import ca.uhn.fhir.rest.annotation.AddTags;
029import ca.uhn.fhir.rest.annotation.Create;
030import ca.uhn.fhir.rest.annotation.Delete;
031import ca.uhn.fhir.rest.annotation.DeleteTags;
032import ca.uhn.fhir.rest.annotation.GetPage;
033import ca.uhn.fhir.rest.annotation.GraphQL;
034import ca.uhn.fhir.rest.annotation.History;
035import ca.uhn.fhir.rest.annotation.Metadata;
036import ca.uhn.fhir.rest.annotation.Operation;
037import ca.uhn.fhir.rest.annotation.Patch;
038import ca.uhn.fhir.rest.annotation.Read;
039import ca.uhn.fhir.rest.annotation.Search;
040import ca.uhn.fhir.rest.annotation.Transaction;
041import ca.uhn.fhir.rest.annotation.Update;
042import ca.uhn.fhir.rest.annotation.Validate;
043import ca.uhn.fhir.rest.api.MethodOutcome;
044import ca.uhn.fhir.rest.api.RestOperationTypeEnum;
045import ca.uhn.fhir.rest.api.server.IBundleProvider;
046import ca.uhn.fhir.rest.api.server.IRestfulServer;
047import ca.uhn.fhir.rest.api.server.RequestDetails;
048import ca.uhn.fhir.rest.server.BundleProviders;
049import ca.uhn.fhir.rest.server.IResourceProvider;
050import ca.uhn.fhir.rest.server.exceptions.BaseServerResponseException;
051import ca.uhn.fhir.rest.server.exceptions.InternalErrorException;
052import ca.uhn.fhir.util.ReflectionUtil;
053import jakarta.annotation.Nonnull;
054import org.hl7.fhir.instance.model.api.IAnyResource;
055import org.hl7.fhir.instance.model.api.IBaseBundle;
056import org.hl7.fhir.instance.model.api.IBaseResource;
057
058import java.io.IOException;
059import java.lang.reflect.InvocationTargetException;
060import java.lang.reflect.Method;
061import java.util.ArrayList;
062import java.util.Collection;
063import java.util.HashSet;
064import java.util.List;
065import java.util.Set;
066import java.util.TreeSet;
067import java.util.stream.Collectors;
068
069import static org.apache.commons.lang3.StringUtils.isNotBlank;
070
071public abstract class BaseMethodBinding {
072
073        private static final org.slf4j.Logger ourLog = org.slf4j.LoggerFactory.getLogger(BaseMethodBinding.class);
074        private final List<BaseQueryParameter> myQueryParameters;
075        private FhirContext myContext;
076        private Method myMethod;
077        private List<IParameter> myParameters;
078        private Object myProvider;
079        private boolean mySupportsConditional;
080        private boolean mySupportsConditionalMultiple;
081
082        public BaseMethodBinding(Method theMethod, FhirContext theContext, Object theProvider) {
083                assert theMethod != null;
084                assert theContext != null;
085
086                myMethod = theMethod;
087                myContext = theContext;
088                myProvider = theProvider;
089                myParameters = MethodUtil.getResourceParameters(theContext, theMethod, theProvider);
090                myQueryParameters = myParameters.stream()
091                                .filter(t -> t instanceof BaseQueryParameter)
092                                .map(t -> (BaseQueryParameter) t)
093                                .collect(Collectors.toList());
094
095                for (IParameter next : myParameters) {
096                        if (next instanceof ConditionalParamBinder) {
097                                mySupportsConditional = true;
098                                if (((ConditionalParamBinder) next).isSupportsMultiple()) {
099                                        mySupportsConditionalMultiple = true;
100                                }
101                                break;
102                        }
103                }
104
105                // This allows us to invoke methods on private classes
106                myMethod.setAccessible(true);
107        }
108
109        protected List<BaseQueryParameter> getQueryParameters() {
110                return myQueryParameters;
111        }
112
113        protected Object[] createMethodParams(RequestDetails theRequest) {
114                Object[] params = new Object[getParameters().size()];
115                for (int i = 0; i < getParameters().size(); i++) {
116                        IParameter param = getParameters().get(i);
117                        if (param != null) {
118                                params[i] = param.translateQueryParametersIntoServerArgument(theRequest, this);
119                        }
120                }
121                return params;
122        }
123
124        protected Object[] createParametersForServerRequest(RequestDetails theRequest) {
125                Object[] params = new Object[getParameters().size()];
126                for (int i = 0; i < getParameters().size(); i++) {
127                        IParameter param = getParameters().get(i);
128                        if (param == null) {
129                                continue;
130                        }
131                        params[i] = param.translateQueryParametersIntoServerArgument(theRequest, this);
132                }
133                return params;
134        }
135
136        /**
137         * Subclasses may override to declare that they apply to all resource types
138         */
139        public boolean isGlobalMethod() {
140                return false;
141        }
142
143        public List<Class<?>> getAllowableParamAnnotations() {
144                return null;
145        }
146
147        public FhirContext getContext() {
148                return myContext;
149        }
150
151        public Set<String> getIncludes() {
152                return doGetIncludesOrRevIncludes(false);
153        }
154
155        public Set<String> getRevIncludes() {
156                return doGetIncludesOrRevIncludes(true);
157        }
158
159        private Set<String> doGetIncludesOrRevIncludes(boolean reverse) {
160                Set<String> retVal = new TreeSet<>();
161                for (IParameter next : myParameters) {
162                        if (next instanceof IncludeParameter) {
163                                IncludeParameter includeParameter = (IncludeParameter) next;
164                                if (includeParameter.isReverse() == reverse) {
165                                        retVal.addAll(includeParameter.getAllow());
166                                }
167                        }
168                }
169                return retVal;
170        }
171
172        public Method getMethod() {
173                return myMethod;
174        }
175
176        public List<IParameter> getParameters() {
177                return myParameters;
178        }
179
180        /**
181         * For unit tests only
182         */
183        public void setParameters(List<IParameter> theParameters) {
184                myParameters = theParameters;
185        }
186
187        public Object getProvider() {
188                return myProvider;
189        }
190
191        @SuppressWarnings({"unchecked", "rawtypes"})
192        public Set<Include> getRequestIncludesFromParams(Object[] params) {
193                if (params == null || params.length == 0) {
194                        return null;
195                }
196                int index = 0;
197                boolean match = false;
198                for (IParameter parameter : myParameters) {
199                        if (parameter instanceof IncludeParameter) {
200                                match = true;
201                                break;
202                        }
203                        index++;
204                }
205                if (!match) {
206                        return null;
207                }
208                if (index >= params.length) {
209                        ourLog.warn("index out of parameter range (should never happen");
210                        return null;
211                }
212                if (params[index] instanceof Set) {
213                        return (Set<Include>) params[index];
214                }
215                if (params[index] instanceof Iterable) {
216                        Set includes = new HashSet<Include>();
217                        for (Object o : (Iterable) params[index]) {
218                                if (o instanceof Include) {
219                                        includes.add(o);
220                                }
221                        }
222                        return includes;
223                }
224                ourLog.warn("include params wasn't Set or Iterable, it was {}", params[index].getClass());
225                return null;
226        }
227
228        /**
229         * Returns the name of the resource this method handles, or <code>null</code> if this method is not resource specific
230         */
231        public abstract String getResourceName();
232
233        @Nonnull
234        public abstract RestOperationTypeEnum getRestOperationType();
235
236        /**
237         * Determine which operation is being fired for a specific request
238         *
239         * @param theRequestDetails The request
240         */
241        public RestOperationTypeEnum getRestOperationType(RequestDetails theRequestDetails) {
242                return getRestOperationType();
243        }
244
245        public abstract MethodMatchEnum incomingServerRequestMatchesMethod(RequestDetails theRequest);
246
247        public abstract Object invokeServer(IRestfulServer<?> theServer, RequestDetails theRequest)
248                        throws BaseServerResponseException, IOException;
249
250        protected final Object invokeServerMethod(RequestDetails theRequest, Object[] theMethodParams) {
251                // Handle server action interceptors
252                RestOperationTypeEnum operationType = getRestOperationType(theRequest);
253                if (operationType != null) {
254
255                        populateRequestDetailsForInterceptor(theRequest, theMethodParams);
256
257                        // Interceptor invoke: SERVER_INCOMING_REQUEST_PRE_HANDLED
258                        PageMethodBinding.callPreHandledHooks(theRequest);
259                }
260
261                // Actually invoke the method
262                try {
263                        Method method = getMethod();
264                        return method.invoke(getProvider(), theMethodParams);
265                } catch (InvocationTargetException e) {
266                        if (e.getCause() instanceof BaseServerResponseException) {
267                                throw (BaseServerResponseException) e.getCause();
268                        }
269                        if (e.getTargetException() instanceof DataFormatException) {
270                                throw (DataFormatException) e.getTargetException();
271                        }
272                        throw new InternalErrorException(Msg.code(389) + "Failed to call access method: " + e.getCause(), e);
273                } catch (Exception e) {
274                        throw new InternalErrorException(Msg.code(390) + "Failed to call access method: " + e.getCause(), e);
275                }
276        }
277
278        /**
279         * Does this method have a parameter annotated with {@link ConditionalParamBinder}. Note that many operations don't actually support this paramter, so this will only return true occasionally.
280         */
281        public boolean isSupportsConditional() {
282                return mySupportsConditional;
283        }
284
285        /**
286         * Does this method support conditional operations over multiple objects (basically for conditional delete)
287         */
288        public boolean isSupportsConditionalMultiple() {
289                return mySupportsConditionalMultiple;
290        }
291
292        /**
293         * Subclasses may override this method (but should also call super) to provide method specifics to the
294         * interceptors.
295         * @param theRequestDetails The server request details
296         * @param theMethodParams   The method params as generated by the specific method binding
297         */
298        protected void populateRequestDetailsForInterceptor(RequestDetails theRequestDetails, Object[] theMethodParams) {
299                // nothing by default
300        }
301
302        protected IBundleProvider toResourceList(Object response) throws InternalErrorException {
303                if (response == null) {
304                        return BundleProviders.newEmptyList();
305                } else if (response instanceof IBundleProvider) {
306                        return (IBundleProvider) response;
307                } else if (response instanceof IBaseResource) {
308                        return BundleProviders.newList((IBaseResource) response);
309                } else if (response instanceof Collection) {
310                        List<IBaseResource> retVal = new ArrayList<IBaseResource>();
311                        for (Object next : ((Collection<?>) response)) {
312                                retVal.add((IBaseResource) next);
313                        }
314                        return BundleProviders.newList(retVal);
315                } else if (response instanceof MethodOutcome) {
316                        IBaseResource retVal = ((MethodOutcome) response).getOperationOutcome();
317                        if (retVal == null) {
318                                retVal = getContext().getResourceDefinition("OperationOutcome").newInstance();
319                        }
320                        return BundleProviders.newList(retVal);
321                } else {
322                        throw new InternalErrorException(Msg.code(391) + "Unexpected return type: "
323                                        + response.getClass().getCanonicalName());
324                }
325        }
326
327        public void close() {
328                // subclasses may override
329        }
330
331        @SuppressWarnings("unchecked")
332        public static BaseMethodBinding bindMethod(Method theMethod, FhirContext theContext, Object theProvider) {
333                Read read = theMethod.getAnnotation(Read.class);
334                Search search = theMethod.getAnnotation(Search.class);
335                Metadata conformance = theMethod.getAnnotation(Metadata.class);
336                Create create = theMethod.getAnnotation(Create.class);
337                Update update = theMethod.getAnnotation(Update.class);
338                Delete delete = theMethod.getAnnotation(Delete.class);
339                History history = theMethod.getAnnotation(History.class);
340                Validate validate = theMethod.getAnnotation(Validate.class);
341                AddTags addTags = theMethod.getAnnotation(AddTags.class);
342                DeleteTags deleteTags = theMethod.getAnnotation(DeleteTags.class);
343                Transaction transaction = theMethod.getAnnotation(Transaction.class);
344                Operation operation = theMethod.getAnnotation(Operation.class);
345                GetPage getPage = theMethod.getAnnotation(GetPage.class);
346                Patch patch = theMethod.getAnnotation(Patch.class);
347                GraphQL graphQL = theMethod.getAnnotation(GraphQL.class);
348
349                // ** if you add another annotation above, also add it to the next line:
350                if (!verifyMethodHasZeroOrOneOperationAnnotation(
351                                theMethod,
352                                read,
353                                search,
354                                conformance,
355                                create,
356                                update,
357                                delete,
358                                history,
359                                validate,
360                                addTags,
361                                deleteTags,
362                                transaction,
363                                operation,
364                                getPage,
365                                patch,
366                                graphQL)) {
367                        return null;
368                }
369
370                if (getPage != null) {
371                        return new PageMethodBinding(theContext, theMethod);
372                }
373
374                if (graphQL != null) {
375                        return new GraphQLMethodBinding(theMethod, graphQL.type(), theContext, theProvider);
376                }
377
378                Class<? extends IBaseResource> returnType;
379
380                Class<? extends IBaseResource> returnTypeFromRp = null;
381                if (theProvider instanceof IResourceProvider) {
382                        returnTypeFromRp = ((IResourceProvider) theProvider).getResourceType();
383                        if (!verifyIsValidResourceReturnType(returnTypeFromRp)) {
384                                throw new ConfigurationException(
385                                                Msg.code(392) + "getResourceType() from " + IResourceProvider.class.getSimpleName() + " type "
386                                                                + theMethod.getDeclaringClass().getCanonicalName() + " returned "
387                                                                + toLogString(returnTypeFromRp) + " - Must return a resource type");
388                        }
389                }
390
391                Class<?> returnTypeFromMethod = theMethod.getReturnType();
392                if (MethodOutcome.class.isAssignableFrom(returnTypeFromMethod)) {
393                        // returns a method outcome
394                } else if (IBundleProvider.class.equals(returnTypeFromMethod)) {
395                        // returns a bundle provider
396                } else if (void.class.equals(returnTypeFromMethod)) {
397                        // returns a bundle
398                } else if (Collection.class.isAssignableFrom(returnTypeFromMethod)) {
399                        returnTypeFromMethod = ReflectionUtil.getGenericCollectionTypeOfMethodReturnType(theMethod);
400                        if (returnTypeFromMethod == null) {
401                                ourLog.trace("Method {} returns a non-typed list, can't verify return type", theMethod);
402                        } else if (!verifyIsValidResourceReturnType(returnTypeFromMethod)
403                                        && !isResourceInterface(returnTypeFromMethod)) {
404                                throw new ConfigurationException(
405                                                Msg.code(393) + "Method '" + theMethod.getName() + "' from "
406                                                                + IResourceProvider.class.getSimpleName() + " type "
407                                                                + theMethod.getDeclaringClass().getCanonicalName()
408                                                                + " returns a collection with generic type " + toLogString(returnTypeFromMethod)
409                                                                + " - Must return a resource type or a collection (List, Set) with a resource type parameter (e.g. List<Patient> or List<IBaseResource> )");
410                        }
411                } else if (IBaseBundle.class.isAssignableFrom(returnTypeFromMethod) && returnTypeFromRp == null) {
412                        // If a plain provider method returns a Bundle, we'll assume it to be a system
413                        // level operation and not a type/instance level operation on the Bundle type.
414                        returnTypeFromMethod = null;
415                } else {
416                        if (!isResourceInterface(returnTypeFromMethod) && !verifyIsValidResourceReturnType(returnTypeFromMethod)) {
417                                throw new ConfigurationException(Msg.code(394) + "Method '" + theMethod.getName() + "' from "
418                                                + IResourceProvider.class.getSimpleName() + " type "
419                                                + theMethod.getDeclaringClass().getCanonicalName()
420                                                + " returns " + toLogString(returnTypeFromMethod)
421                                                + " - Must return a resource type (eg Patient, Bundle, " + IBundleProvider.class.getSimpleName()
422                                                + ", etc., see the documentation for more details)");
423                        }
424                }
425
426                Class<? extends IBaseResource> returnTypeFromAnnotation = IBaseResource.class;
427                String returnTypeNameFromAnnotation = null;
428                if (read != null) {
429                        returnTypeFromAnnotation = read.type();
430                        returnTypeNameFromAnnotation = read.typeName();
431                } else if (search != null) {
432                        returnTypeFromAnnotation = search.type();
433                        returnTypeNameFromAnnotation = search.typeName();
434                } else if (history != null) {
435                        returnTypeFromAnnotation = history.type();
436                        returnTypeNameFromAnnotation = history.typeName();
437                } else if (delete != null) {
438                        returnTypeFromAnnotation = delete.type();
439                        returnTypeNameFromAnnotation = delete.typeName();
440                } else if (patch != null) {
441                        returnTypeFromAnnotation = patch.type();
442                        returnTypeNameFromAnnotation = patch.typeName();
443                } else if (create != null) {
444                        returnTypeFromAnnotation = create.type();
445                        returnTypeNameFromAnnotation = create.typeName();
446                } else if (update != null) {
447                        returnTypeFromAnnotation = update.type();
448                        returnTypeNameFromAnnotation = update.typeName();
449                } else if (validate != null) {
450                        returnTypeFromAnnotation = validate.type();
451                        returnTypeNameFromAnnotation = validate.typeName();
452                } else if (addTags != null) {
453                        returnTypeFromAnnotation = addTags.type();
454                        returnTypeNameFromAnnotation = addTags.typeName();
455                } else if (deleteTags != null) {
456                        returnTypeFromAnnotation = deleteTags.type();
457                        returnTypeNameFromAnnotation = deleteTags.typeName();
458                }
459
460                if (isNotBlank(returnTypeNameFromAnnotation)) {
461                        returnTypeFromAnnotation = theContext
462                                        .getResourceDefinition(returnTypeNameFromAnnotation)
463                                        .getImplementingClass();
464                }
465
466                if (returnTypeFromRp != null) {
467                        if (returnTypeFromAnnotation != null && !isResourceInterface(returnTypeFromAnnotation)) {
468                                if (returnTypeFromMethod != null && !returnTypeFromRp.isAssignableFrom(returnTypeFromMethod)) {
469                                        throw new ConfigurationException(Msg.code(395) + "Method '" + theMethod.getName() + "' in type "
470                                                        + theMethod.getDeclaringClass().getCanonicalName() + " returns type "
471                                                        + returnTypeFromMethod.getCanonicalName() + " - Must return "
472                                                        + returnTypeFromRp.getCanonicalName()
473                                                        + " (or a subclass of it) per IResourceProvider contract");
474                                }
475                                if (!returnTypeFromRp.isAssignableFrom(returnTypeFromAnnotation)) {
476                                        throw new ConfigurationException(Msg.code(396) + "Method '" + theMethod.getName() + "' in type "
477                                                        + theMethod.getDeclaringClass().getCanonicalName() + " claims to return type "
478                                                        + returnTypeFromAnnotation.getCanonicalName() + " per method annotation - Must return "
479                                                        + returnTypeFromRp.getCanonicalName()
480                                                        + " (or a subclass of it) per IResourceProvider contract");
481                                }
482                                returnType = returnTypeFromAnnotation;
483                        } else {
484                                returnType = returnTypeFromRp;
485                        }
486                } else {
487                        if (!isResourceInterface(returnTypeFromAnnotation)) {
488                                if (!verifyIsValidResourceReturnType(returnTypeFromAnnotation)) {
489                                        throw new ConfigurationException(Msg.code(397) + "Method '" + theMethod.getName() + "' from "
490                                                        + IResourceProvider.class.getSimpleName() + " type "
491                                                        + theMethod.getDeclaringClass().getCanonicalName() + " returns "
492                                                        + toLogString(returnTypeFromAnnotation)
493                                                        + " according to annotation - Must return a resource type");
494                                }
495                                returnType = returnTypeFromAnnotation;
496                        } else {
497                                returnType = (Class<? extends IBaseResource>) returnTypeFromMethod;
498                        }
499                }
500
501                if (read != null) {
502                        return new ReadMethodBinding(returnType, theMethod, theContext, theProvider);
503                } else if (search != null) {
504                        return new SearchMethodBinding(returnType, returnTypeFromRp, theMethod, theContext, theProvider);
505                } else if (conformance != null) {
506                        return new ConformanceMethodBinding(theMethod, theContext, theProvider);
507                } else if (create != null) {
508                        return new CreateMethodBinding(theMethod, theContext, theProvider);
509                } else if (update != null) {
510                        return new UpdateMethodBinding(theMethod, theContext, theProvider);
511                } else if (delete != null) {
512                        return new DeleteMethodBinding(theMethod, theContext, theProvider);
513                } else if (patch != null) {
514                        return new PatchMethodBinding(theMethod, theContext, theProvider);
515                } else if (history != null) {
516                        return new HistoryMethodBinding(theMethod, theContext, theProvider);
517                } else if (validate != null) {
518                        return new ValidateMethodBindingDstu2Plus(
519                                        returnType, returnTypeFromRp, theMethod, theContext, theProvider, validate);
520                } else if (transaction != null) {
521                        return new TransactionMethodBinding(theMethod, theContext, theProvider);
522                } else if (operation != null) {
523                        return new OperationMethodBinding(
524                                        returnType, returnTypeFromRp, theMethod, theContext, theProvider, operation);
525                } else {
526                        throw new ConfigurationException(
527                                        Msg.code(398) + "Did not detect any FHIR annotations on method '" + theMethod.getName()
528                                                        + "' on type: " + theMethod.getDeclaringClass().getCanonicalName());
529                }
530        }
531
532        private static boolean isResourceInterface(Class<?> theReturnTypeFromMethod) {
533                return theReturnTypeFromMethod != null
534                                && (theReturnTypeFromMethod.equals(IBaseResource.class)
535                                                || theReturnTypeFromMethod.equals(IResource.class)
536                                                || theReturnTypeFromMethod.equals(IAnyResource.class));
537        }
538
539        private static String toLogString(Class<?> theType) {
540                if (theType == null) {
541                        return null;
542                }
543                return theType.getCanonicalName();
544        }
545
546        private static boolean verifyIsValidResourceReturnType(Class<?> theReturnType) {
547                if (theReturnType == null) {
548                        return false;
549                }
550                if (!IBaseResource.class.isAssignableFrom(theReturnType)) {
551                        return false;
552                }
553                return true;
554        }
555
556        public static boolean verifyMethodHasZeroOrOneOperationAnnotation(Method theNextMethod, Object... theAnnotations) {
557                Object obj1 = null;
558                for (Object object : theAnnotations) {
559                        if (object != null) {
560                                if (obj1 == null) {
561                                        obj1 = object;
562                                } else {
563                                        throw new ConfigurationException(Msg.code(399) + "Method " + theNextMethod.getName() + " on type '"
564                                                        + theNextMethod.getDeclaringClass().getSimpleName() + " has annotations @"
565                                                        + obj1.getClass().getSimpleName() + " and @"
566                                                        + object.getClass().getSimpleName() + ". Can not have both.");
567                                }
568                        }
569                }
570                if (obj1 == null) {
571                        return false;
572                }
573                return true;
574        }
575}