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.FhirContext;
023import ca.uhn.fhir.i18n.Msg;
024import ca.uhn.fhir.model.api.IResource;
025import ca.uhn.fhir.model.api.ResourceMetadataKeyEnum;
026import ca.uhn.fhir.model.primitive.IdDt;
027import ca.uhn.fhir.model.valueset.BundleTypeEnum;
028import ca.uhn.fhir.rest.annotation.History;
029import ca.uhn.fhir.rest.api.Constants;
030import ca.uhn.fhir.rest.api.RestOperationTypeEnum;
031import ca.uhn.fhir.rest.api.server.IBundleProvider;
032import ca.uhn.fhir.rest.api.server.IRestfulServer;
033import ca.uhn.fhir.rest.api.server.RequestDetails;
034import ca.uhn.fhir.rest.param.ParameterUtil;
035import ca.uhn.fhir.rest.server.IResourceProvider;
036import ca.uhn.fhir.rest.server.exceptions.InternalErrorException;
037import ca.uhn.fhir.rest.server.exceptions.InvalidRequestException;
038import jakarta.annotation.Nonnull;
039import org.apache.commons.lang3.StringUtils;
040import org.hl7.fhir.instance.model.api.IBaseResource;
041import org.hl7.fhir.instance.model.api.IPrimitiveType;
042
043import java.lang.reflect.Method;
044import java.lang.reflect.Modifier;
045import java.util.Date;
046import java.util.List;
047
048import static org.apache.commons.lang3.StringUtils.isBlank;
049
050public class HistoryMethodBinding extends BaseResourceReturningMethodBinding {
051
052        private final Integer myIdParamIndex;
053        private final RestOperationTypeEnum myResourceOperationType;
054        private final String myResourceName;
055
056        public HistoryMethodBinding(Method theMethod, FhirContext theContext, Object theProvider) {
057                super(toReturnType(theMethod, theProvider), theMethod, theContext, theProvider);
058
059                myIdParamIndex = ParameterUtil.findIdParameterIndex(theMethod, getContext());
060
061                History historyAnnotation = theMethod.getAnnotation(History.class);
062                Class<? extends IBaseResource> type = historyAnnotation.type();
063                if (Modifier.isInterface(type.getModifiers())) {
064                        if (theProvider instanceof IResourceProvider) {
065                                type = ((IResourceProvider) theProvider).getResourceType();
066                                if (myIdParamIndex != null) {
067                                        myResourceOperationType = RestOperationTypeEnum.HISTORY_INSTANCE;
068                                } else {
069                                        myResourceOperationType = RestOperationTypeEnum.HISTORY_TYPE;
070                                }
071                        } else {
072                                myResourceOperationType = RestOperationTypeEnum.HISTORY_SYSTEM;
073                        }
074                } else {
075                        if (myIdParamIndex != null) {
076                                myResourceOperationType = RestOperationTypeEnum.HISTORY_INSTANCE;
077                        } else {
078                                myResourceOperationType = RestOperationTypeEnum.HISTORY_TYPE;
079                        }
080                }
081
082                if (type != IBaseResource.class && type != IResource.class) {
083                        myResourceName = theContext.getResourceType(type);
084                } else {
085                        myResourceName = null;
086                }
087        }
088
089        @Override
090        protected BundleTypeEnum getResponseBundleType() {
091                return BundleTypeEnum.HISTORY;
092        }
093
094        @Nonnull
095        @Override
096        public RestOperationTypeEnum getRestOperationType() {
097                return myResourceOperationType;
098        }
099
100        @Override
101        public ReturnTypeEnum getReturnType() {
102                return ReturnTypeEnum.BUNDLE;
103        }
104
105        @Override
106        protected boolean isOffsetModeHistory() {
107                return true;
108        }
109
110        // ObjectUtils.equals is replaced by a JDK7 method..
111        @Override
112        public MethodMatchEnum incomingServerRequestMatchesMethod(RequestDetails theRequest) {
113                if (!Constants.PARAM_HISTORY.equals(theRequest.getOperation())) {
114                        return MethodMatchEnum.NONE;
115                }
116                if (theRequest.getResourceName() == null) {
117                        if (myResourceOperationType == RestOperationTypeEnum.HISTORY_SYSTEM) {
118                                return MethodMatchEnum.EXACT;
119                        } else {
120                                return MethodMatchEnum.NONE;
121                        }
122                }
123                if (!StringUtils.equals(theRequest.getResourceName(), myResourceName)) {
124                        return MethodMatchEnum.NONE;
125                }
126
127                boolean haveIdParam = theRequest.getId() != null && !theRequest.getId().isEmpty();
128                boolean wantIdParam = myIdParamIndex != null;
129                if (haveIdParam != wantIdParam) {
130                        return MethodMatchEnum.NONE;
131                }
132
133                if (theRequest.getId() == null) {
134                        if (myResourceOperationType != RestOperationTypeEnum.HISTORY_TYPE) {
135                                return MethodMatchEnum.NONE;
136                        }
137                } else if (theRequest.getId().hasVersionIdPart()) {
138                        return MethodMatchEnum.NONE;
139                }
140
141                return MethodMatchEnum.EXACT;
142        }
143
144        @Override
145        public IBundleProvider invokeServer(
146                        IRestfulServer<?> theServer, RequestDetails theRequest, Object[] theMethodParams)
147                        throws InvalidRequestException, InternalErrorException {
148                if (myIdParamIndex != null) {
149                        theMethodParams[myIdParamIndex] = theRequest.getId();
150                }
151
152                Object response = invokeServerMethod(theRequest, theMethodParams);
153
154                final IBundleProvider resources = toResourceList(response);
155
156                /*
157                 * We wrap the response so we can verify that it has the ID and version set,
158                 * as is the contract for history
159                 */
160                return new IBundleProvider() {
161
162                        @Override
163                        public String getCurrentPageId() {
164                                return resources.getCurrentPageId();
165                        }
166
167                        @Override
168                        public String getNextPageId() {
169                                return resources.getNextPageId();
170                        }
171
172                        @Override
173                        public String getPreviousPageId() {
174                                return resources.getPreviousPageId();
175                        }
176
177                        @Override
178                        public IPrimitiveType<Date> getPublished() {
179                                return resources.getPublished();
180                        }
181
182                        @Nonnull
183                        @Override
184                        public List<IBaseResource> getResources(
185                                        int theFromIndex, int theToIndex, ResponsePage.ResponsePageBuilder theResponsePageBuilder) {
186                                List<IBaseResource> retVal = resources.getResources(theFromIndex, theToIndex, theResponsePageBuilder);
187                                int index = theFromIndex;
188                                for (IBaseResource nextResource : retVal) {
189                                        if (nextResource.getIdElement() == null
190                                                        || isBlank(nextResource.getIdElement().getIdPart())) {
191                                                throw new InternalErrorException(Msg.code(410) + "Server provided resource at index " + index
192                                                                + " with no ID set (using IResource#setId(IdDt))");
193                                        }
194                                        if (isBlank(nextResource.getIdElement().getVersionIdPart()) && nextResource instanceof IResource) {
195                                                // TODO: Use of a deprecated method should be resolved.
196                                                IdDt versionId = ResourceMetadataKeyEnum.VERSION_ID.get(nextResource);
197                                                if (versionId == null || versionId.isEmpty()) {
198                                                        throw new InternalErrorException(Msg.code(411) + "Server provided resource at index "
199                                                                        + index + " with no Version ID set (using IResource#setId(IdDt))");
200                                                }
201                                        }
202                                        index++;
203                                }
204                                return retVal;
205                        }
206
207                        @Override
208                        public String getUuid() {
209                                return resources.getUuid();
210                        }
211
212                        @Override
213                        public Integer preferredPageSize() {
214                                return resources.preferredPageSize();
215                        }
216
217                        @Override
218                        public Integer size() {
219                                return resources.size();
220                        }
221                };
222        }
223
224        private static Class<? extends IBaseResource> toReturnType(Method theMethod, Object theProvider) {
225                if (theProvider instanceof IResourceProvider) {
226                        return ((IResourceProvider) theProvider).getResourceType();
227                }
228                History historyAnnotation = theMethod.getAnnotation(History.class);
229                Class<? extends IBaseResource> type = historyAnnotation.type();
230                if (type != IBaseResource.class && type != IResource.class) {
231                        return type;
232                }
233                return null;
234        }
235}