001/*
002 * #%L
003 * HAPI FHIR - Client 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.client.method;
021
022import ca.uhn.fhir.context.ConfigurationException;
023import ca.uhn.fhir.context.FhirContext;
024import ca.uhn.fhir.i18n.Msg;
025import ca.uhn.fhir.rest.annotation.Patch;
026import ca.uhn.fhir.rest.annotation.ResourceParam;
027import ca.uhn.fhir.rest.api.PatchTypeEnum;
028import ca.uhn.fhir.rest.api.RequestTypeEnum;
029import ca.uhn.fhir.rest.api.RestOperationTypeEnum;
030import ca.uhn.fhir.rest.client.impl.BaseHttpClientInvocation;
031import ca.uhn.fhir.rest.server.exceptions.InternalErrorException;
032import ca.uhn.fhir.rest.server.exceptions.InvalidRequestException;
033import org.hl7.fhir.instance.model.api.IBaseResource;
034import org.hl7.fhir.instance.model.api.IIdType;
035
036import java.lang.annotation.Annotation;
037import java.lang.reflect.Method;
038import java.util.Arrays;
039import java.util.Collections;
040import java.util.List;
041import java.util.ListIterator;
042import java.util.Map;
043import java.util.Set;
044
045/**
046 * Base class for an operation that has a resource type but not a resource body in the
047 * request body
048 *
049 */
050public class PatchMethodBinding extends BaseOutcomeReturningMethodBindingWithResourceIdButNoResourceBody {
051
052        private int myPatchTypeParameterIndex = -1;
053        private int myResourceParamIndex;
054
055        public PatchMethodBinding(Method theMethod, FhirContext theContext, Object theProvider) {
056                super(
057                                theMethod,
058                                theContext,
059                                theProvider,
060                                Patch.class,
061                                theMethod.getAnnotation(Patch.class).type());
062
063                for (ListIterator<Class<?>> iter =
064                                                Arrays.asList(theMethod.getParameterTypes()).listIterator();
065                                iter.hasNext(); ) {
066                        int nextIndex = iter.nextIndex();
067                        Class<?> next = iter.next();
068                        if (next.equals(PatchTypeEnum.class)) {
069                                myPatchTypeParameterIndex = nextIndex;
070                        }
071                        for (Annotation nextAnnotation : theMethod.getParameterAnnotations()[nextIndex]) {
072                                if (nextAnnotation instanceof ResourceParam) {
073                                        myResourceParamIndex = nextIndex;
074                                }
075                        }
076                }
077
078                if (myPatchTypeParameterIndex == -1) {
079                        throw new ConfigurationException(Msg.code(1414) + "Method has no parameter of type "
080                                        + PatchTypeEnum.class.getName() + " - " + theMethod.toString());
081                }
082                if (myResourceParamIndex == -1) {
083                        throw new ConfigurationException(Msg.code(1415) + "Method has no parameter with @"
084                                        + ResourceParam.class.getSimpleName() + " annotation - " + theMethod.toString());
085                }
086        }
087
088        @Override
089        public RestOperationTypeEnum getRestOperationType() {
090                return RestOperationTypeEnum.PATCH;
091        }
092
093        @Override
094        protected Set<RequestTypeEnum> provideAllowableRequestTypes() {
095                return Collections.singleton(RequestTypeEnum.PATCH);
096        }
097
098        @Override
099        protected BaseHttpClientInvocation createClientInvocation(Object[] theArgs, IBaseResource theResource) {
100                StringBuilder urlExtension = new StringBuilder();
101                urlExtension.append(getContext().getResourceType(theResource));
102
103                return new HttpPostClientInvocation(getContext(), theResource, urlExtension.toString());
104        }
105
106        @Override
107        protected boolean allowVoidReturnType() {
108                return true;
109        }
110
111        @Override
112        public BaseHttpClientInvocation invokeClient(Object[] theArgs) throws InternalErrorException {
113                IIdType idDt = (IIdType) theArgs[getIdParameterIndex()];
114                if (idDt == null) {
115                        throw new NullPointerException(Msg.code(1416) + "ID can not be null");
116                }
117
118                if (idDt.hasResourceType() == false) {
119                        idDt = idDt.withResourceType(getResourceName());
120                } else if (getResourceName().equals(idDt.getResourceType()) == false) {
121                        throw new InvalidRequestException(Msg.code(1417) + "ID parameter has the wrong resource type, expected '"
122                                        + getResourceName() + "', found: " + idDt.getResourceType());
123                }
124
125                PatchTypeEnum patchType = (PatchTypeEnum) theArgs[myPatchTypeParameterIndex];
126                String body = (String) theArgs[myResourceParamIndex];
127
128                HttpPatchClientInvocation retVal = createPatchInvocation(getContext(), idDt, patchType, body);
129
130                for (int idx = 0; idx < theArgs.length; idx++) {
131                        IParameter nextParam = getParameters().get(idx);
132                        nextParam.translateClientArgumentIntoQueryArgument(getContext(), theArgs[idx], null, null);
133                }
134
135                return retVal;
136        }
137
138        public static HttpPatchClientInvocation createPatchInvocation(
139                        FhirContext theContext, IIdType theId, PatchTypeEnum thePatchType, String theBody) {
140                HttpPatchClientInvocation retVal =
141                                new HttpPatchClientInvocation(theContext, theId, thePatchType.getContentType(), theBody);
142                return retVal;
143        }
144
145        public static HttpPatchClientInvocation createPatchInvocation(
146                        FhirContext theContext, String theUrlPath, PatchTypeEnum thePatchType, String theBody) {
147                HttpPatchClientInvocation retVal =
148                                new HttpPatchClientInvocation(theContext, theUrlPath, thePatchType.getContentType(), theBody);
149                return retVal;
150        }
151
152        @Override
153        protected String getMatchingOperation() {
154                return null;
155        }
156
157        public static HttpPatchClientInvocation createPatchInvocation(
158                        FhirContext theContext,
159                        PatchTypeEnum thePatchType,
160                        String theBody,
161                        String theResourceType,
162                        Map<String, List<String>> theMatchParams) {
163                StringBuilder urlBuilder = MethodUtil.createUrl(theResourceType, theMatchParams);
164                String url = urlBuilder.toString();
165                HttpPatchClientInvocation retVal =
166                                new HttpPatchClientInvocation(theContext, url, thePatchType.getContentType(), theBody);
167                return retVal;
168        }
169}