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.interceptor; 021 022import ca.uhn.fhir.interceptor.api.Hook; 023import ca.uhn.fhir.interceptor.api.Interceptor; 024import ca.uhn.fhir.interceptor.api.Pointcut; 025import ca.uhn.fhir.jpa.api.model.ResourceVersionConflictResolutionStrategy; 026import ca.uhn.fhir.rest.api.server.RequestDetails; 027import ca.uhn.fhir.rest.api.server.SystemRequestDetails; 028import org.apache.commons.lang3.Validate; 029 030/** 031 * This interceptor looks for a header on incoming requests called <code>X-Retry-On-Version-Conflict</code> and 032 * if present, it will instruct the server to automatically retry JPA server operations that would have 033 * otherwise failed with a {@link ca.uhn.fhir.rest.server.exceptions.ResourceVersionConflictException} (HTTP 409). 034 * <p> 035 * The format of the header is:<br/> 036 * <code>X-Retry-On-Version-Conflict: retry; max-retries=100</code> 037 * </p> 038 */ 039@Interceptor 040public class UserRequestRetryVersionConflictsInterceptor { 041 042 /** Deprecated and moved to {@link ca.uhn.fhir.rest.api.Constants#HEADER_RETRY_ON_VERSION_CONFLICT} */ 043 @Deprecated 044 public static final String HEADER_NAME = "X-Retry-On-Version-Conflict"; 045 046 /** Deprecated and moved to {@link ca.uhn.fhir.rest.api.Constants#HEADER_MAX_RETRIES} */ 047 @Deprecated 048 public static final String MAX_RETRIES = "max-retries"; 049 050 /** Deprecated and moved to {@link ca.uhn.fhir.rest.api.Constants#HEADER_RETRY} */ 051 @Deprecated 052 public static final String RETRY = "retry"; 053 054 @Hook(value = Pointcut.STORAGE_VERSION_CONFLICT, order = 100) 055 public ResourceVersionConflictResolutionStrategy check(RequestDetails theRequestDetails) { 056 ResourceVersionConflictResolutionStrategy retVal = new ResourceVersionConflictResolutionStrategy(); 057 boolean shouldSetRetries = theRequestDetails != null && theRequestDetails.isRetry(); 058 if (shouldSetRetries) { 059 retVal.setRetry(true); 060 int maxRetries = Math.min(100, theRequestDetails.getMaxRetries()); 061 retVal.setMaxRetries(maxRetries); 062 } 063 064 return retVal; 065 } 066 067 /** 068 * Convenience method to add a retry header to a system request 069 */ 070 public static void addRetryHeader(SystemRequestDetails theRequestDetails, int theMaxRetries) { 071 Validate.inclusiveBetween(1, Integer.MAX_VALUE, theMaxRetries, "Max retries must be > 0"); 072 theRequestDetails.setRetry(true); 073 theRequestDetails.setMaxRetries(theMaxRetries); 074 } 075}