001/*
002 * #%L
003 * HAPI FHIR Storage api
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.jpa.provider;
021
022import ca.uhn.fhir.i18n.Msg;
023import ca.uhn.fhir.jpa.api.dao.IFhirResourceDao;
024import ca.uhn.fhir.jpa.api.model.DaoMethodOutcome;
025import ca.uhn.fhir.jpa.api.model.ExpungeOptions;
026import ca.uhn.fhir.jpa.api.model.ExpungeOutcome;
027import ca.uhn.fhir.jpa.model.util.JpaConstants;
028import ca.uhn.fhir.model.api.annotation.Description;
029import ca.uhn.fhir.rest.annotation.At;
030import ca.uhn.fhir.rest.annotation.ConditionalUrlParam;
031import ca.uhn.fhir.rest.annotation.Create;
032import ca.uhn.fhir.rest.annotation.Delete;
033import ca.uhn.fhir.rest.annotation.History;
034import ca.uhn.fhir.rest.annotation.IdParam;
035import ca.uhn.fhir.rest.annotation.Offset;
036import ca.uhn.fhir.rest.annotation.Operation;
037import ca.uhn.fhir.rest.annotation.OperationParam;
038import ca.uhn.fhir.rest.annotation.Patch;
039import ca.uhn.fhir.rest.annotation.Read;
040import ca.uhn.fhir.rest.annotation.ResourceParam;
041import ca.uhn.fhir.rest.annotation.Since;
042import ca.uhn.fhir.rest.annotation.Update;
043import ca.uhn.fhir.rest.annotation.Validate;
044import ca.uhn.fhir.rest.api.EncodingEnum;
045import ca.uhn.fhir.rest.api.MethodOutcome;
046import ca.uhn.fhir.rest.api.PatchTypeEnum;
047import ca.uhn.fhir.rest.api.ValidationModeEnum;
048import ca.uhn.fhir.rest.api.server.IBundleProvider;
049import ca.uhn.fhir.rest.api.server.RequestDetails;
050import ca.uhn.fhir.rest.param.DateRangeParam;
051import ca.uhn.fhir.rest.param.HistorySearchDateRangeParam;
052import ca.uhn.fhir.rest.server.IResourceProvider;
053import ca.uhn.fhir.rest.server.exceptions.InvalidRequestException;
054import ca.uhn.fhir.rest.server.provider.ProviderConstants;
055import ca.uhn.fhir.util.CoverageIgnore;
056import ca.uhn.fhir.util.ParametersUtil;
057import jakarta.servlet.http.HttpServletRequest;
058import org.hl7.fhir.instance.model.api.IBaseMetaType;
059import org.hl7.fhir.instance.model.api.IBaseParameters;
060import org.hl7.fhir.instance.model.api.IBaseResource;
061import org.hl7.fhir.instance.model.api.IIdType;
062import org.hl7.fhir.instance.model.api.IPrimitiveType;
063
064import java.util.Date;
065
066import static ca.uhn.fhir.jpa.model.util.JpaConstants.OPERATION_META_ADD;
067import static ca.uhn.fhir.jpa.model.util.JpaConstants.OPERATION_META_DELETE;
068import static ca.uhn.fhir.rest.server.provider.ProviderConstants.OPERATION_META;
069
070public abstract class BaseJpaResourceProvider<T extends IBaseResource> extends BaseJpaProvider
071                implements IResourceProvider {
072
073        private IFhirResourceDao<T> myDao;
074
075        public BaseJpaResourceProvider() {
076                // nothing
077        }
078
079        @CoverageIgnore
080        public BaseJpaResourceProvider(IFhirResourceDao<T> theDao) {
081                myDao = theDao;
082        }
083
084        protected IBaseParameters doExpunge(
085                        IIdType theIdParam,
086                        IPrimitiveType<? extends Integer> theLimit,
087                        IPrimitiveType<? extends Boolean> theExpungeDeletedResources,
088                        IPrimitiveType<? extends Boolean> theExpungeOldVersions,
089                        IPrimitiveType<? extends Boolean> theExpungeEverything,
090                        RequestDetails theRequest) {
091
092                ExpungeOptions options =
093                                createExpungeOptions(theLimit, theExpungeDeletedResources, theExpungeOldVersions, theExpungeEverything);
094
095                ExpungeOutcome outcome;
096                if (theIdParam != null) {
097                        outcome = getDao().expunge(theIdParam, options, theRequest);
098                } else {
099                        outcome = getDao().expunge(options, theRequest);
100                }
101
102                return createExpungeResponse(outcome);
103        }
104
105        public IFhirResourceDao<T> getDao() {
106                return myDao;
107        }
108
109        public void setDao(IFhirResourceDao<T> theDao) {
110                myDao = theDao;
111        }
112
113        @History
114        public IBundleProvider getHistoryForResourceInstance(
115                        HttpServletRequest theRequest,
116                        @Offset Integer theOffset,
117                        @IdParam IIdType theId,
118                        @Since Date theSince,
119                        @At DateRangeParam theAt,
120                        RequestDetails theRequestDetails) {
121
122                startRequest(theRequest);
123                try {
124                        DateRangeParam sinceOrAt = processSinceOrAt(theSince, theAt);
125                        return myDao.history(
126                                        theId,
127                                        new HistorySearchDateRangeParam(theRequestDetails.getParameters(), sinceOrAt, theOffset),
128                                        theRequestDetails);
129                } finally {
130                        endRequest(theRequest);
131                }
132        }
133
134        @History
135        public IBundleProvider getHistoryForResourceType(
136                        HttpServletRequest theRequest,
137                        @Offset Integer theOffset,
138                        @Since Date theSince,
139                        @At DateRangeParam theAt,
140                        RequestDetails theRequestDetails) {
141                startRequest(theRequest);
142                try {
143                        DateRangeParam sinceOrAt = processSinceOrAt(theSince, theAt);
144                        return myDao.history(
145                                        sinceOrAt.getLowerBoundAsInstant(),
146                                        sinceOrAt.getUpperBoundAsInstant(),
147                                        theOffset,
148                                        theRequestDetails);
149                } finally {
150                        endRequest(theRequest);
151                }
152        }
153
154        @Override
155        public Class<? extends IBaseResource> getResourceType() {
156                return myDao.getResourceType();
157        }
158
159        @Patch
160        public DaoMethodOutcome patch(
161                        HttpServletRequest theRequest,
162                        @IdParam IIdType theId,
163                        @ConditionalUrlParam String theConditionalUrl,
164                        RequestDetails theRequestDetails,
165                        @ResourceParam String theBody,
166                        PatchTypeEnum thePatchType,
167                        @ResourceParam IBaseParameters theRequestBody) {
168                startRequest(theRequest);
169                try {
170                        return myDao.patch(theId, theConditionalUrl, thePatchType, theBody, theRequestBody, theRequestDetails);
171                } finally {
172                        endRequest(theRequest);
173                }
174        }
175
176        @Read(version = true)
177        public T read(HttpServletRequest theRequest, @IdParam IIdType theId, RequestDetails theRequestDetails) {
178                startRequest(theRequest);
179                try {
180                        return myDao.read(theId, theRequestDetails);
181                } finally {
182                        endRequest(theRequest);
183                }
184        }
185
186        @Create
187        public MethodOutcome create(
188                        HttpServletRequest theRequest,
189                        @ResourceParam T theResource,
190                        @ConditionalUrlParam String theConditional,
191                        RequestDetails theRequestDetails) {
192                startRequest(theRequest);
193                try {
194                        if (theConditional != null) {
195                                return getDao().create(theResource, theConditional, theRequestDetails);
196                        } else {
197                                return getDao().create(theResource, theRequestDetails);
198                        }
199                } finally {
200                        endRequest(theRequest);
201                }
202        }
203
204        @Delete()
205        public MethodOutcome delete(
206                        HttpServletRequest theRequest,
207                        @IdParam IIdType theResource,
208                        @ConditionalUrlParam(supportsMultiple = true) String theConditional,
209                        RequestDetails theRequestDetails) {
210                startRequest(theRequest);
211                try {
212                        if (theConditional != null) {
213                                return getDao().deleteByUrl(theConditional, theRequestDetails);
214                        } else {
215                                return getDao().delete(theResource, theRequestDetails);
216                        }
217                } finally {
218                        endRequest(theRequest);
219                }
220        }
221
222        @Operation(
223                        name = ProviderConstants.OPERATION_EXPUNGE,
224                        idempotent = false,
225                        returnParameters = {
226                                @OperationParam(name = JpaConstants.OPERATION_EXPUNGE_OUT_PARAM_EXPUNGE_COUNT, typeName = "integer")
227                        })
228        public IBaseParameters expunge(
229                        @IdParam IIdType theIdParam,
230                        @OperationParam(name = ProviderConstants.OPERATION_EXPUNGE_PARAM_LIMIT, typeName = "integer")
231                                        IPrimitiveType<Integer> theLimit,
232                        @OperationParam(
233                                                        name = ProviderConstants.OPERATION_EXPUNGE_PARAM_EXPUNGE_DELETED_RESOURCES,
234                                                        typeName = "boolean")
235                                        IPrimitiveType<Boolean> theExpungeDeletedResources,
236                        @OperationParam(
237                                                        name = ProviderConstants.OPERATION_EXPUNGE_PARAM_EXPUNGE_PREVIOUS_VERSIONS,
238                                                        typeName = "boolean")
239                                        IPrimitiveType<Boolean> theExpungeOldVersions,
240                        RequestDetails theRequest) {
241                return doExpunge(theIdParam, theLimit, theExpungeDeletedResources, theExpungeOldVersions, null, theRequest);
242        }
243
244        @Operation(
245                        name = ProviderConstants.OPERATION_EXPUNGE,
246                        idempotent = false,
247                        returnParameters = {
248                                @OperationParam(name = JpaConstants.OPERATION_EXPUNGE_OUT_PARAM_EXPUNGE_COUNT, typeName = "integer")
249                        })
250        public IBaseParameters expunge(
251                        @OperationParam(name = ProviderConstants.OPERATION_EXPUNGE_PARAM_LIMIT, typeName = "integer")
252                                        IPrimitiveType<Integer> theLimit,
253                        @OperationParam(
254                                                        name = ProviderConstants.OPERATION_EXPUNGE_PARAM_EXPUNGE_DELETED_RESOURCES,
255                                                        typeName = "boolean")
256                                        IPrimitiveType<Boolean> theExpungeDeletedResources,
257                        @OperationParam(
258                                                        name = ProviderConstants.OPERATION_EXPUNGE_PARAM_EXPUNGE_PREVIOUS_VERSIONS,
259                                                        typeName = "boolean")
260                                        IPrimitiveType<Boolean> theExpungeOldVersions,
261                        RequestDetails theRequest) {
262                return doExpunge(null, theLimit, theExpungeDeletedResources, theExpungeOldVersions, null, theRequest);
263        }
264
265        @Description("Request a global list of tags, profiles, and security labels")
266        @Operation(
267                        name = OPERATION_META,
268                        idempotent = true,
269                        returnParameters = {@OperationParam(name = "return", typeName = "Meta")})
270        public IBaseParameters meta(RequestDetails theRequestDetails) {
271                Class metaType = getContext().getElementDefinition("Meta").getImplementingClass();
272                IBaseMetaType metaGetOperation = getDao().metaGetOperation(metaType, theRequestDetails);
273                IBaseParameters parameters = ParametersUtil.newInstance(getContext());
274                ParametersUtil.addParameterToParameters(getContext(), parameters, "return", metaGetOperation);
275                return parameters;
276        }
277
278        @Description("Request a list of tags, profiles, and security labels for a specfic resource instance")
279        @Operation(
280                        name = OPERATION_META,
281                        idempotent = true,
282                        returnParameters = {@OperationParam(name = "return", typeName = "Meta")})
283        public IBaseParameters meta(@IdParam IIdType theId, RequestDetails theRequestDetails) {
284                Class metaType = getContext().getElementDefinition("Meta").getImplementingClass();
285                IBaseMetaType metaGetOperation = getDao().metaGetOperation(metaType, theId, theRequestDetails);
286
287                IBaseParameters parameters = ParametersUtil.newInstance(getContext());
288                ParametersUtil.addParameterToParameters(getContext(), parameters, "return", metaGetOperation);
289                return parameters;
290        }
291
292        @Description("Add tags, profiles, and/or security labels to a resource")
293        @Operation(
294                        name = OPERATION_META_ADD,
295                        idempotent = false,
296                        returnParameters = {@OperationParam(name = "return", typeName = "Meta")})
297        public IBaseParameters metaAdd(
298                        @IdParam IIdType theId,
299                        @OperationParam(name = "meta", typeName = "Meta") IBaseMetaType theMeta,
300                        RequestDetails theRequestDetails) {
301                if (theMeta == null) {
302                        throw new InvalidRequestException(Msg.code(554) + "Input contains no parameter with name 'meta'");
303                }
304                IBaseMetaType metaAddOperation = getDao().metaAddOperation(theId, theMeta, theRequestDetails);
305                IBaseParameters parameters = ParametersUtil.newInstance(getContext());
306                ParametersUtil.addParameterToParameters(getContext(), parameters, "return", metaAddOperation);
307                return parameters;
308        }
309
310        @Description("Delete tags, profiles, and/or security labels from a resource")
311        @Operation(
312                        name = OPERATION_META_DELETE,
313                        idempotent = false,
314                        returnParameters = {@OperationParam(name = "return", typeName = "Meta")})
315        public IBaseParameters metaDelete(
316                        @IdParam IIdType theId,
317                        @OperationParam(name = "meta", typeName = "Meta") IBaseMetaType theMeta,
318                        RequestDetails theRequestDetails) {
319                if (theMeta == null) {
320                        throw new InvalidRequestException(Msg.code(555) + "Input contains no parameter with name 'meta'");
321                }
322                IBaseMetaType metaDelete = getDao().metaDeleteOperation(theId, theMeta, theRequestDetails);
323                IBaseParameters parameters = ParametersUtil.newInstance(getContext());
324                ParametersUtil.addParameterToParameters(getContext(), parameters, "return", metaDelete);
325                return parameters;
326        }
327
328        @Update
329        public MethodOutcome update(
330                        HttpServletRequest theRequest,
331                        @ResourceParam T theResource,
332                        @IdParam IIdType theId,
333                        @ConditionalUrlParam String theConditional,
334                        RequestDetails theRequestDetails) {
335                startRequest(theRequest);
336                try {
337                        if (theConditional != null) {
338                                return getDao().update(theResource, theConditional, theRequestDetails);
339                        } else {
340                                return getDao().update(theResource, theRequestDetails);
341                        }
342                } finally {
343                        endRequest(theRequest);
344                }
345        }
346
347        @Validate
348        public MethodOutcome validate(
349                        @ResourceParam T theResource,
350                        @ResourceParam String theRawResource,
351                        @ResourceParam EncodingEnum theEncoding,
352                        @Validate.Mode ValidationModeEnum theMode,
353                        @Validate.Profile String theProfile,
354                        RequestDetails theRequestDetails) {
355                return validate(theResource, null, theRawResource, theEncoding, theMode, theProfile, theRequestDetails);
356        }
357
358        @Validate
359        public MethodOutcome validate(
360                        @ResourceParam T theResource,
361                        @IdParam IIdType theId,
362                        @ResourceParam String theRawResource,
363                        @ResourceParam EncodingEnum theEncoding,
364                        @Validate.Mode ValidationModeEnum theMode,
365                        @Validate.Profile String theProfile,
366                        RequestDetails theRequestDetails) {
367                return getDao().validate(
368                                                theResource, theId, theRawResource, theEncoding, theMode, theProfile, theRequestDetails);
369        }
370}