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.dao;
021
022import ca.uhn.fhir.i18n.Msg;
023import ca.uhn.fhir.jpa.api.dao.IFhirResourceDao;
024import ca.uhn.fhir.jpa.api.dao.IJpaDao;
025import ca.uhn.fhir.jpa.api.model.DaoMethodOutcome;
026import ca.uhn.fhir.jpa.api.model.DeleteMethodOutcome;
027import ca.uhn.fhir.jpa.dao.tx.HapiTransactionService;
028import ca.uhn.fhir.jpa.model.cross.IBasePersistedResource;
029import ca.uhn.fhir.jpa.patch.FhirPatch;
030import ca.uhn.fhir.jpa.patch.JsonPatchUtils;
031import ca.uhn.fhir.jpa.patch.XmlPatchUtils;
032import ca.uhn.fhir.parser.StrictErrorHandler;
033import ca.uhn.fhir.rest.api.DeleteCascadeModeEnum;
034import ca.uhn.fhir.rest.api.PatchTypeEnum;
035import ca.uhn.fhir.rest.api.RestOperationTypeEnum;
036import ca.uhn.fhir.rest.api.server.RequestDetails;
037import ca.uhn.fhir.rest.api.server.storage.IDeleteExpungeJobSubmitter;
038import ca.uhn.fhir.rest.api.server.storage.IResourcePersistentId;
039import ca.uhn.fhir.rest.api.server.storage.TransactionDetails;
040import ca.uhn.fhir.rest.server.RestfulServerUtils;
041import ca.uhn.fhir.rest.server.exceptions.InvalidRequestException;
042import ca.uhn.fhir.rest.server.exceptions.MethodNotAllowedException;
043import ca.uhn.fhir.rest.server.exceptions.PreconditionFailedException;
044import ca.uhn.fhir.rest.server.exceptions.ResourceNotFoundException;
045import ca.uhn.fhir.rest.server.exceptions.ResourceVersionConflictException;
046import ca.uhn.fhir.rest.server.exceptions.UnprocessableEntityException;
047import jakarta.annotation.Nonnull;
048import org.hl7.fhir.instance.model.api.IBaseParameters;
049import org.hl7.fhir.instance.model.api.IBaseResource;
050import org.hl7.fhir.instance.model.api.IIdType;
051import org.springframework.beans.factory.annotation.Autowired;
052import org.springframework.transaction.support.TransactionSynchronizationManager;
053
054import java.util.Collections;
055import java.util.List;
056import java.util.Set;
057
058import static org.apache.commons.lang3.StringUtils.isNotBlank;
059
060public abstract class BaseStorageResourceDao<T extends IBaseResource> extends BaseStorageDao
061                implements IFhirResourceDao<T>, IJpaDao<T> {
062        public static final StrictErrorHandler STRICT_ERROR_HANDLER = new StrictErrorHandler();
063
064        @Autowired
065        protected abstract HapiTransactionService getTransactionService();
066
067        @Autowired
068        protected abstract MatchResourceUrlService getMatchResourceUrlService();
069
070        @Autowired
071        protected abstract IStorageResourceParser getStorageResourceParser();
072
073        @Autowired
074        protected abstract IDeleteExpungeJobSubmitter getDeleteExpungeJobSubmitter();
075
076        @Override
077        public DaoMethodOutcome patch(
078                        IIdType theId,
079                        String theConditionalUrl,
080                        PatchTypeEnum thePatchType,
081                        String thePatchBody,
082                        IBaseParameters theFhirPatchBody,
083                        RequestDetails theRequestDetails) {
084                TransactionDetails transactionDetails = new TransactionDetails();
085                return getTransactionService()
086                                .execute(
087                                                theRequestDetails,
088                                                transactionDetails,
089                                                tx -> patchInTransaction(
090                                                                theId,
091                                                                theConditionalUrl,
092                                                                true,
093                                                                thePatchType,
094                                                                thePatchBody,
095                                                                theFhirPatchBody,
096                                                                theRequestDetails,
097                                                                transactionDetails));
098        }
099
100        @Override
101        public DaoMethodOutcome patchInTransaction(
102                        IIdType theId,
103                        String theConditionalUrl,
104                        boolean thePerformIndexing,
105                        PatchTypeEnum thePatchType,
106                        String thePatchBody,
107                        IBaseParameters theFhirPatchBody,
108                        RequestDetails theRequestDetails,
109                        TransactionDetails theTransactionDetails) {
110                assert TransactionSynchronizationManager.isActualTransactionActive();
111
112                IBasePersistedResource entityToUpdate;
113                IIdType resourceId;
114                if (isNotBlank(theConditionalUrl)) {
115
116                        Set<IResourcePersistentId> match = getMatchResourceUrlService()
117                                        .processMatchUrl(theConditionalUrl, getResourceType(), theTransactionDetails, theRequestDetails);
118                        if (match.size() > 1) {
119                                String msg = getContext()
120                                                .getLocalizer()
121                                                .getMessageSanitized(
122                                                                BaseStorageDao.class,
123                                                                "transactionOperationWithMultipleMatchFailure",
124                                                                "PATCH",
125                                                                getResourceName(),
126                                                                theConditionalUrl,
127                                                                match.size());
128                                throw new PreconditionFailedException(Msg.code(972) + msg);
129                        } else if (match.size() == 1) {
130                                IResourcePersistentId pid = match.iterator().next();
131                                entityToUpdate = readEntityLatestVersion(pid, theRequestDetails, theTransactionDetails);
132                                resourceId = entityToUpdate.getIdDt();
133                        } else {
134                                String msg = getContext()
135                                                .getLocalizer()
136                                                .getMessageSanitized(BaseStorageDao.class, "invalidMatchUrlNoMatches", theConditionalUrl);
137                                throw new ResourceNotFoundException(Msg.code(973) + msg);
138                        }
139
140                } else {
141                        resourceId = theId;
142                        entityToUpdate = readEntityLatestVersion(theId, theRequestDetails, theTransactionDetails);
143                        if (theId.hasVersionIdPart()) {
144                                if (theId.getVersionIdPartAsLong() != entityToUpdate.getVersion()) {
145                                        throw new ResourceVersionConflictException(Msg.code(974) + "Version " + theId.getVersionIdPart()
146                                                        + " is not the most recent version of this resource, unable to apply patch");
147                                }
148                        }
149                }
150
151                validateResourceType(entityToUpdate, getResourceName());
152
153                if (entityToUpdate.isDeleted()) {
154                        throw createResourceGoneException(entityToUpdate);
155                }
156
157                IBaseResource resourceToUpdate = getStorageResourceParser().toResource(entityToUpdate, false);
158                if (resourceToUpdate == null) {
159                        // If this is null, we are presumably in a FHIR transaction bundle with both a create and a patch on the
160                        // same
161                        // resource. This is weird but not impossible.
162                        resourceToUpdate = theTransactionDetails.getResolvedResource(resourceId);
163                }
164
165                IBaseResource destination;
166                switch (thePatchType) {
167                        case JSON_PATCH:
168                                destination = JsonPatchUtils.apply(getContext(), resourceToUpdate, thePatchBody);
169                                break;
170                        case XML_PATCH:
171                                destination = XmlPatchUtils.apply(getContext(), resourceToUpdate, thePatchBody);
172                                break;
173                        case FHIR_PATCH_XML:
174                        case FHIR_PATCH_JSON:
175                        default:
176                                IBaseParameters fhirPatchJson = theFhirPatchBody;
177                                new FhirPatch(getContext()).apply(resourceToUpdate, fhirPatchJson);
178                                destination = resourceToUpdate;
179                                break;
180                }
181
182                @SuppressWarnings("unchecked")
183                T destinationCasted = (T) destination;
184                myFhirContext
185                                .newJsonParser()
186                                .setParserErrorHandler(STRICT_ERROR_HANDLER)
187                                .encodeResourceToString(destinationCasted);
188
189                preProcessResourceForStorage(destinationCasted, theRequestDetails, theTransactionDetails, true);
190
191                return doUpdateForUpdateOrPatch(
192                                theRequestDetails,
193                                resourceId,
194                                theConditionalUrl,
195                                thePerformIndexing,
196                                false,
197                                destinationCasted,
198                                entityToUpdate,
199                                RestOperationTypeEnum.PATCH,
200                                theTransactionDetails);
201        }
202
203        @Override
204        @Nonnull
205        public abstract Class<T> getResourceType();
206
207        @Override
208        @Nonnull
209        protected abstract String getResourceName();
210
211        protected abstract IBasePersistedResource readEntityLatestVersion(
212                        IResourcePersistentId thePersistentId,
213                        RequestDetails theRequestDetails,
214                        TransactionDetails theTransactionDetails);
215
216        protected abstract IBasePersistedResource readEntityLatestVersion(
217                        IIdType theId, RequestDetails theRequestDetails, TransactionDetails theTransactionDetails);
218
219        protected DaoMethodOutcome doUpdateForUpdateOrPatch(
220                        RequestDetails theRequest,
221                        IIdType theResourceId,
222                        String theMatchUrl,
223                        boolean thePerformIndexing,
224                        boolean theForceUpdateVersion,
225                        T theResource,
226                        IBasePersistedResource theEntity,
227                        RestOperationTypeEnum theOperationType,
228                        TransactionDetails theTransactionDetails) {
229                if (theResourceId.hasVersionIdPart()
230                                && Long.parseLong(theResourceId.getVersionIdPart()) != theEntity.getVersion()) {
231                        throw new ResourceVersionConflictException(
232                                        Msg.code(989) + "Trying to update " + theResourceId + " but this is not the current version");
233                }
234
235                if (theResourceId.hasResourceType() && !theResourceId.getResourceType().equals(getResourceName())) {
236                        throw new UnprocessableEntityException(Msg.code(990) + "Invalid resource ID["
237                                        + theEntity.getIdDt().toUnqualifiedVersionless() + "] of type[" + theEntity.getResourceType()
238                                        + "] - Does not match expected [" + getResourceName() + "]");
239                }
240
241                IBaseResource oldResource;
242                if (getStorageSettings().isMassIngestionMode()) {
243                        oldResource = null;
244                } else {
245                        oldResource = getStorageResourceParser().toResource(theEntity, false);
246                }
247
248                /*
249                 * Mark the entity as not deleted - This is also done in the actual updateInternal()
250                 * method later on so it usually doesn't matter whether we do it here, but in the
251                 * case of a transaction with multiple PUTs we don't get there until later so
252                 * having this here means that a transaction can have a reference in one
253                 * resource to another resource in the same transaction that is being
254                 * un-deleted by the transaction. Wacky use case, sure. But it's real.
255                 *
256                 * See SystemProviderR4Test#testTransactionReSavesPreviouslyDeletedResources
257                 * for a test that needs this.
258                 */
259                boolean wasDeleted = theEntity.isDeleted();
260                theEntity.setNotDeleted();
261
262                /*
263                 * If we aren't indexing, that means we're doing this inside a transaction.
264                 * The transaction will do the actual storage to the database a bit later on,
265                 * after placeholder IDs have been replaced, by calling {@link #updateInternal}
266                 * directly. So we just bail now.
267                 */
268                if (!thePerformIndexing) {
269                        theResource.setId(theEntity.getIdDt().getValue());
270                        DaoMethodOutcome outcome = toMethodOutcome(
271                                                        theRequest, theEntity, theResource, theMatchUrl, theOperationType)
272                                        .setCreated(wasDeleted);
273                        outcome.setPreviousResource(oldResource);
274                        if (!outcome.isNop()) {
275                                // Technically this may not end up being right since we might not increment if the
276                                // contents turn out to be the same
277                                outcome.setId(outcome.getId()
278                                                .withVersion(Long.toString(outcome.getId().getVersionIdPartAsLong() + 1)));
279                        }
280                        return outcome;
281                }
282
283                /*
284                 * Otherwise, we're not in a transaction
285                 */
286                return updateInternal(
287                                theRequest,
288                                theResource,
289                                theMatchUrl,
290                                thePerformIndexing,
291                                theForceUpdateVersion,
292                                theEntity,
293                                theResourceId,
294                                oldResource,
295                                theOperationType,
296                                theTransactionDetails);
297        }
298
299        public static void validateResourceType(IBasePersistedResource<?> theEntity, String theResourceName) {
300                if (!theResourceName.equals(theEntity.getResourceType())) {
301                        throw new ResourceNotFoundException(Msg.code(935) + "Resource with ID "
302                                        + theEntity.getIdDt().getIdPart() + " exists but it is not of type " + theResourceName
303                                        + ", found resource of type " + theEntity.getResourceType());
304                }
305        }
306
307        protected DeleteMethodOutcome deleteExpunge(String theUrl, RequestDetails theRequest) {
308                if (!getStorageSettings().canDeleteExpunge()) {
309                        throw new MethodNotAllowedException(Msg.code(963) + "_expunge is not enabled on this server: "
310                                        + getStorageSettings().cannotDeleteExpungeReason());
311                }
312
313                RestfulServerUtils.DeleteCascadeDetails cascadeDelete =
314                                RestfulServerUtils.extractDeleteCascadeParameter(theRequest);
315                boolean cascade = false;
316                Integer cascadeMaxRounds = null;
317                if (cascadeDelete.getMode() == DeleteCascadeModeEnum.DELETE) {
318                        cascade = true;
319                        cascadeMaxRounds = cascadeDelete.getMaxRounds();
320                        if (cascadeMaxRounds == null) {
321                                cascadeMaxRounds = myStorageSettings.getMaximumDeleteConflictQueryCount();
322                        } else if (myStorageSettings.getMaximumDeleteConflictQueryCount() != null
323                                        && myStorageSettings.getMaximumDeleteConflictQueryCount() < cascadeMaxRounds) {
324                                cascadeMaxRounds = myStorageSettings.getMaximumDeleteConflictQueryCount();
325                        }
326                }
327
328                List<String> urlsToDeleteExpunge = Collections.singletonList(theUrl);
329                try {
330                        String jobId = getDeleteExpungeJobSubmitter()
331                                        .submitJob(
332                                                        getStorageSettings().getExpungeBatchSize(),
333                                                        urlsToDeleteExpunge,
334                                                        cascade,
335                                                        cascadeMaxRounds,
336                                                        theRequest);
337                        return new DeleteMethodOutcome(createInfoOperationOutcome("Delete job submitted with id " + jobId));
338                } catch (InvalidRequestException e) {
339                        throw new InvalidRequestException(Msg.code(965) + "Invalid Delete Expunge Request: " + e.getMessage(), e);
340                }
341        }
342}