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.tx; 021 022import ca.uhn.fhir.i18n.Msg; 023import ca.uhn.fhir.interceptor.api.HookParams; 024import ca.uhn.fhir.interceptor.api.IInterceptorBroadcaster; 025import ca.uhn.fhir.interceptor.api.Pointcut; 026import ca.uhn.fhir.interceptor.model.RequestPartitionId; 027import ca.uhn.fhir.jpa.api.model.ResourceVersionConflictResolutionStrategy; 028import ca.uhn.fhir.jpa.dao.DaoFailureUtil; 029import ca.uhn.fhir.jpa.model.config.PartitionSettings; 030import ca.uhn.fhir.jpa.partition.IRequestPartitionHelperSvc; 031import ca.uhn.fhir.rest.api.server.RequestDetails; 032import ca.uhn.fhir.rest.api.server.storage.TransactionDetails; 033import ca.uhn.fhir.rest.server.exceptions.BaseServerResponseException; 034import ca.uhn.fhir.rest.server.exceptions.InternalErrorException; 035import ca.uhn.fhir.rest.server.exceptions.ResourceVersionConflictException; 036import ca.uhn.fhir.rest.server.servlet.ServletRequestDetails; 037import ca.uhn.fhir.rest.server.util.CompositeInterceptorBroadcaster; 038import ca.uhn.fhir.util.ICallable; 039import ca.uhn.fhir.util.SleepUtil; 040import com.google.common.annotations.VisibleForTesting; 041import jakarta.annotation.Nonnull; 042import jakarta.annotation.Nullable; 043import org.apache.commons.lang3.Validate; 044import org.apache.commons.lang3.exception.ExceptionUtils; 045import org.hibernate.exception.ConstraintViolationException; 046import org.hl7.fhir.instance.model.api.IBaseOperationOutcome; 047import org.slf4j.Logger; 048import org.slf4j.LoggerFactory; 049import org.springframework.beans.factory.annotation.Autowired; 050import org.springframework.dao.DataIntegrityViolationException; 051import org.springframework.dao.PessimisticLockingFailureException; 052import org.springframework.orm.ObjectOptimisticLockingFailureException; 053import org.springframework.transaction.PlatformTransactionManager; 054import org.springframework.transaction.TransactionStatus; 055import org.springframework.transaction.annotation.Isolation; 056import org.springframework.transaction.annotation.Propagation; 057import org.springframework.transaction.support.TransactionCallback; 058import org.springframework.transaction.support.TransactionCallbackWithoutResult; 059import org.springframework.transaction.support.TransactionOperations; 060import org.springframework.transaction.support.TransactionSynchronizationManager; 061import org.springframework.transaction.support.TransactionTemplate; 062 063import java.util.Objects; 064import java.util.concurrent.Callable; 065 066/** 067 * @see IHapiTransactionService for an explanation of this class 068 */ 069public class HapiTransactionService implements IHapiTransactionService { 070 071 public static final String XACT_USERDATA_KEY_RESOLVED_TAG_DEFINITIONS = 072 HapiTransactionService.class.getName() + "_RESOLVED_TAG_DEFINITIONS"; 073 public static final String XACT_USERDATA_KEY_EXISTING_SEARCH_PARAMS = 074 HapiTransactionService.class.getName() + "_EXISTING_SEARCH_PARAMS"; 075 private static final Logger ourLog = LoggerFactory.getLogger(HapiTransactionService.class); 076 private static final ThreadLocal<RequestPartitionId> ourRequestPartitionThreadLocal = new ThreadLocal<>(); 077 private static final ThreadLocal<HapiTransactionService> ourExistingTransaction = new ThreadLocal<>(); 078 079 /** 080 * Default value for {@link #setTransactionPropagationWhenChangingPartitions(Propagation)} 081 * 082 * @since 7.6.0 083 */ 084 public static final Propagation DEFAULT_TRANSACTION_PROPAGATION_WHEN_CHANGING_PARTITIONS = Propagation.REQUIRED; 085 086 @Autowired 087 protected IInterceptorBroadcaster myInterceptorBroadcaster; 088 089 @Autowired 090 protected PlatformTransactionManager myTransactionManager; 091 092 @Autowired 093 protected IRequestPartitionHelperSvc myRequestPartitionHelperSvc; 094 095 @Autowired 096 protected PartitionSettings myPartitionSettings; 097 098 private Propagation myTransactionPropagationWhenChangingPartitions = 099 DEFAULT_TRANSACTION_PROPAGATION_WHEN_CHANGING_PARTITIONS; 100 101 private SleepUtil mySleepUtil = new SleepUtil(); 102 103 @VisibleForTesting 104 public void setInterceptorBroadcaster(IInterceptorBroadcaster theInterceptorBroadcaster) { 105 myInterceptorBroadcaster = theInterceptorBroadcaster; 106 } 107 108 @VisibleForTesting 109 public void setSleepUtil(SleepUtil theSleepUtil) { 110 mySleepUtil = theSleepUtil; 111 } 112 113 @Override 114 public IExecutionBuilder withRequest(@Nullable RequestDetails theRequestDetails) { 115 return buildExecutionBuilder(theRequestDetails); 116 } 117 118 @Override 119 public IExecutionBuilder withSystemRequest() { 120 return buildExecutionBuilder(null); 121 } 122 123 protected IExecutionBuilder buildExecutionBuilder(@Nullable RequestDetails theRequestDetails) { 124 return new ExecutionBuilder(theRequestDetails); 125 } 126 127 /** 128 * @deprecated Use {@link #withRequest(RequestDetails)} with fluent call instead 129 */ 130 @Deprecated 131 public <T> T execute( 132 @Nullable RequestDetails theRequestDetails, 133 @Nullable TransactionDetails theTransactionDetails, 134 @Nonnull TransactionCallback<T> theCallback) { 135 return execute(theRequestDetails, theTransactionDetails, theCallback, null); 136 } 137 138 /** 139 * @deprecated Use {@link #withRequest(RequestDetails)} with fluent call instead 140 */ 141 @Deprecated 142 public void execute( 143 @Nullable RequestDetails theRequestDetails, 144 @Nullable TransactionDetails theTransactionDetails, 145 @Nonnull Propagation thePropagation, 146 @Nonnull Isolation theIsolation, 147 @Nonnull Runnable theCallback) { 148 TransactionCallbackWithoutResult callback = new TransactionCallbackWithoutResult() { 149 @Override 150 protected void doInTransactionWithoutResult(TransactionStatus status) { 151 theCallback.run(); 152 } 153 }; 154 execute(theRequestDetails, theTransactionDetails, callback, null, thePropagation, theIsolation); 155 } 156 157 /** 158 * @deprecated Use {@link #withRequest(RequestDetails)} with fluent call instead 159 */ 160 @Deprecated 161 @Override 162 public <T> T withRequest( 163 @Nullable RequestDetails theRequestDetails, 164 @Nullable TransactionDetails theTransactionDetails, 165 @Nonnull Propagation thePropagation, 166 @Nonnull Isolation theIsolation, 167 @Nonnull ICallable<T> theCallback) { 168 169 TransactionCallback<T> callback = tx -> theCallback.call(); 170 return execute(theRequestDetails, theTransactionDetails, callback, null, thePropagation, theIsolation); 171 } 172 173 /** 174 * @deprecated Use {@link #withRequest(RequestDetails)} with fluent call instead 175 */ 176 @Deprecated 177 public <T> T execute( 178 @Nullable RequestDetails theRequestDetails, 179 @Nullable TransactionDetails theTransactionDetails, 180 @Nonnull TransactionCallback<T> theCallback, 181 @Nullable Runnable theOnRollback) { 182 return execute(theRequestDetails, theTransactionDetails, theCallback, theOnRollback, null, null); 183 } 184 185 @SuppressWarnings("ConstantConditions") 186 /** 187 * @deprecated Use {@link #withRequest(RequestDetails)} with fluent call instead 188 */ 189 @Deprecated 190 public <T> T execute( 191 @Nullable RequestDetails theRequestDetails, 192 @Nullable TransactionDetails theTransactionDetails, 193 @Nonnull TransactionCallback<T> theCallback, 194 @Nullable Runnable theOnRollback, 195 @Nullable Propagation thePropagation, 196 @Nullable Isolation theIsolation) { 197 return withRequest(theRequestDetails) 198 .withTransactionDetails(theTransactionDetails) 199 .withPropagation(thePropagation) 200 .withIsolation(theIsolation) 201 .onRollback(theOnRollback) 202 .execute(theCallback); 203 } 204 205 /** 206 * @deprecated Use {@link #withRequest(RequestDetails)} with fluent call instead 207 */ 208 @Deprecated 209 public <T> T execute( 210 @Nullable RequestDetails theRequestDetails, 211 @Nullable TransactionDetails theTransactionDetails, 212 @Nonnull TransactionCallback<T> theCallback, 213 @Nullable Runnable theOnRollback, 214 @Nonnull Propagation thePropagation, 215 @Nonnull Isolation theIsolation, 216 RequestPartitionId theRequestPartitionId) { 217 return withRequest(theRequestDetails) 218 .withTransactionDetails(theTransactionDetails) 219 .withPropagation(thePropagation) 220 .withIsolation(theIsolation) 221 .withRequestPartitionId(theRequestPartitionId) 222 .onRollback(theOnRollback) 223 .execute(theCallback); 224 } 225 226 public boolean isCustomIsolationSupported() { 227 return false; 228 } 229 230 @VisibleForTesting 231 public void setRequestPartitionSvcForUnitTest(IRequestPartitionHelperSvc theRequestPartitionHelperSvc) { 232 myRequestPartitionHelperSvc = theRequestPartitionHelperSvc; 233 } 234 235 public PlatformTransactionManager getTransactionManager() { 236 return myTransactionManager; 237 } 238 239 @VisibleForTesting 240 public void setTransactionManager(PlatformTransactionManager theTransactionManager) { 241 myTransactionManager = theTransactionManager; 242 } 243 244 @VisibleForTesting 245 public void setPartitionSettingsForUnitTest(PartitionSettings thePartitionSettings) { 246 myPartitionSettings = thePartitionSettings; 247 } 248 249 @Nullable 250 protected <T> T doExecute(ExecutionBuilder theExecutionBuilder, TransactionCallback<T> theCallback) { 251 final RequestPartitionId requestPartitionId = theExecutionBuilder.getEffectiveRequestPartitionId(); 252 RequestPartitionId previousRequestPartitionId = null; 253 if (requestPartitionId != null) { 254 previousRequestPartitionId = ourRequestPartitionThreadLocal.get(); 255 ourRequestPartitionThreadLocal.set(requestPartitionId); 256 } 257 258 ourLog.trace("Starting doExecute for RequestPartitionId {}", requestPartitionId); 259 if (isCompatiblePartition(previousRequestPartitionId, requestPartitionId)) { 260 if (ourExistingTransaction.get() == this && canReuseExistingTransaction(theExecutionBuilder)) { 261 /* 262 * If we're already in an active transaction, and it's for the right partition, 263 * and it's not a read-only transaction, we don't need to open a new transaction 264 * so let's just add a method to the stack trace that makes this obvious. 265 */ 266 return executeInExistingTransaction(theCallback); 267 } 268 } 269 270 HapiTransactionService previousExistingTransaction = ourExistingTransaction.get(); 271 try { 272 ourExistingTransaction.set(this); 273 274 if (isRequiresNewTransactionWhenChangingPartitions()) { 275 return executeInNewTransactionForPartitionChange( 276 theExecutionBuilder, theCallback, requestPartitionId, previousRequestPartitionId); 277 } else { 278 return doExecuteInTransaction( 279 theExecutionBuilder, theCallback, requestPartitionId, previousRequestPartitionId); 280 } 281 } finally { 282 ourExistingTransaction.set(previousExistingTransaction); 283 } 284 } 285 286 protected boolean isRequiresNewTransactionWhenChangingPartitions() { 287 return myTransactionPropagationWhenChangingPartitions == Propagation.REQUIRES_NEW; 288 } 289 290 @Override 291 public boolean isCompatiblePartition( 292 RequestPartitionId theRequestPartitionId, RequestPartitionId theOtherRequestPartitionId) { 293 return !myPartitionSettings.isPartitioningEnabled() 294 || !isRequiresNewTransactionWhenChangingPartitions() 295 || Objects.equals(theRequestPartitionId, theOtherRequestPartitionId); 296 } 297 298 @Nullable 299 private <T> T executeInNewTransactionForPartitionChange( 300 ExecutionBuilder theExecutionBuilder, 301 TransactionCallback<T> theCallback, 302 RequestPartitionId requestPartitionId, 303 RequestPartitionId previousRequestPartitionId) { 304 ourLog.trace("executeInNewTransactionForPartitionChange"); 305 theExecutionBuilder.myPropagation = myTransactionPropagationWhenChangingPartitions; 306 return doExecuteInTransaction(theExecutionBuilder, theCallback, requestPartitionId, previousRequestPartitionId); 307 } 308 309 private boolean isThrowableOrItsSubclassPresent(Throwable theThrowable, Class<? extends Throwable> theClass) { 310 return ExceptionUtils.indexOfType(theThrowable, theClass) != -1; 311 } 312 313 private boolean isThrowablePresent(Throwable theThrowable, Class<? extends Throwable> theClass) { 314 return ExceptionUtils.indexOfThrowable(theThrowable, theClass) != -1; 315 } 316 317 private boolean isRetriable(Throwable theThrowable) { 318 return isThrowablePresent(theThrowable, ResourceVersionConflictException.class) 319 || isThrowablePresent(theThrowable, DataIntegrityViolationException.class) 320 || isThrowablePresent(theThrowable, ConstraintViolationException.class) 321 || isThrowablePresent(theThrowable, ObjectOptimisticLockingFailureException.class) 322 // calling isThrowableOrItsSubclassPresent instead of isThrowablePresent for 323 // PessimisticLockingFailureException, because we want to retry on its subclasses as well, especially 324 // CannotAcquireLockException, which is thrown in some deadlock situations which we want to retry 325 || isThrowableOrItsSubclassPresent(theThrowable, PessimisticLockingFailureException.class); 326 } 327 328 @Nullable 329 private <T> T doExecuteInTransaction( 330 ExecutionBuilder theExecutionBuilder, 331 TransactionCallback<T> theCallback, 332 RequestPartitionId requestPartitionId, 333 RequestPartitionId previousRequestPartitionId) { 334 ourLog.trace("doExecuteInTransaction"); 335 try { 336 for (int i = 0; ; i++) { 337 try { 338 339 return doExecuteCallback(theExecutionBuilder, theCallback); 340 341 } catch (Exception e) { 342 if (!isRetriable(e)) { 343 ourLog.debug("Unexpected transaction exception. Will not be retried.", e); 344 throw e; 345 } else { 346 347 ourLog.debug("Version conflict detected", e); 348 349 if (theExecutionBuilder.myOnRollback != null) { 350 theExecutionBuilder.myOnRollback.run(); 351 } 352 353 int maxRetries = 0; 354 355 /* 356 * If two client threads both concurrently try to add the same tag that isn't 357 * known to the system already, they'll both try to create a row in HFJ_TAG_DEF, 358 * which is the tag definition table. In that case, a constraint error will be 359 * thrown by one of the client threads, so we auto-retry in order to avoid 360 * annoying spurious failures for the client. 361 */ 362 if (DaoFailureUtil.isTagStorageFailure(e)) { 363 maxRetries = 3; 364 } 365 366 if (maxRetries == 0) { 367 IInterceptorBroadcaster compositeBroadcaster = 368 CompositeInterceptorBroadcaster.newCompositeBroadcaster( 369 myInterceptorBroadcaster, theExecutionBuilder.myRequestDetails); 370 if (compositeBroadcaster.hasHooks(Pointcut.STORAGE_VERSION_CONFLICT)) { 371 HookParams params = new HookParams() 372 .add(RequestDetails.class, theExecutionBuilder.myRequestDetails) 373 .addIfMatchesType( 374 ServletRequestDetails.class, theExecutionBuilder.myRequestDetails); 375 ResourceVersionConflictResolutionStrategy conflictResolutionStrategy = 376 (ResourceVersionConflictResolutionStrategy) 377 compositeBroadcaster.callHooksAndReturnObject( 378 Pointcut.STORAGE_VERSION_CONFLICT, params); 379 if (conflictResolutionStrategy != null && conflictResolutionStrategy.isRetry()) { 380 maxRetries = conflictResolutionStrategy.getMaxRetries(); 381 } 382 } 383 } 384 385 if (i < maxRetries) { 386 if (theExecutionBuilder.myTransactionDetails != null) { 387 theExecutionBuilder 388 .myTransactionDetails 389 .getRollbackUndoActions() 390 .forEach(Runnable::run); 391 theExecutionBuilder.myTransactionDetails.clearRollbackUndoActions(); 392 theExecutionBuilder.myTransactionDetails.clearResolvedItems(); 393 theExecutionBuilder.myTransactionDetails.clearUserData( 394 XACT_USERDATA_KEY_RESOLVED_TAG_DEFINITIONS); 395 theExecutionBuilder.myTransactionDetails.clearUserData( 396 XACT_USERDATA_KEY_EXISTING_SEARCH_PARAMS); 397 } 398 double sleepAmount = (250.0d * i) * Math.random(); 399 long sleepAmountLong = (long) sleepAmount; 400 mySleepUtil.sleepAtLeast(sleepAmountLong, false); 401 402 ourLog.info( 403 "About to start a transaction retry due to conflict or constraint error. Sleeping {}ms first.", 404 sleepAmountLong); 405 continue; 406 } 407 408 IBaseOperationOutcome oo = null; 409 if (e instanceof ResourceVersionConflictException) { 410 oo = ((ResourceVersionConflictException) e).getOperationOutcome(); 411 } 412 413 if (maxRetries > 0) { 414 String msg = 415 "Max retries (" + maxRetries + ") exceeded for version conflict: " + e.getMessage(); 416 ourLog.info(msg, maxRetries); 417 throw new ResourceVersionConflictException(Msg.code(549) + msg); 418 } 419 420 throw new ResourceVersionConflictException(Msg.code(550) + e.getMessage(), e, oo); 421 } 422 } 423 } 424 } finally { 425 if (requestPartitionId != null) { 426 ourRequestPartitionThreadLocal.set(previousRequestPartitionId); 427 } 428 } 429 } 430 431 public void setTransactionPropagationWhenChangingPartitions( 432 Propagation theTransactionPropagationWhenChangingPartitions) { 433 Validate.notNull(theTransactionPropagationWhenChangingPartitions); 434 myTransactionPropagationWhenChangingPartitions = theTransactionPropagationWhenChangingPartitions; 435 } 436 437 @Nullable 438 protected <T> T doExecuteCallback(ExecutionBuilder theExecutionBuilder, TransactionCallback<T> theCallback) { 439 try { 440 TransactionTemplate txTemplate = new TransactionTemplate(myTransactionManager); 441 442 if (theExecutionBuilder.myPropagation != null) { 443 txTemplate.setPropagationBehavior(theExecutionBuilder.myPropagation.value()); 444 } 445 446 if (isCustomIsolationSupported() 447 && theExecutionBuilder.myIsolation != null 448 && theExecutionBuilder.myIsolation != Isolation.DEFAULT) { 449 txTemplate.setIsolationLevel(theExecutionBuilder.myIsolation.value()); 450 } 451 452 if (theExecutionBuilder.myReadOnly) { 453 txTemplate.setReadOnly(true); 454 } 455 456 return txTemplate.execute(theCallback); 457 } catch (MyException e) { 458 if (e.getCause() instanceof RuntimeException) { 459 throw (RuntimeException) e.getCause(); 460 } else { 461 throw new InternalErrorException(Msg.code(551) + e); 462 } 463 } 464 } 465 466 protected class ExecutionBuilder implements IExecutionBuilder, TransactionOperations, Cloneable { 467 468 private final RequestDetails myRequestDetails; 469 private Isolation myIsolation; 470 private Propagation myPropagation; 471 private boolean myReadOnly; 472 private TransactionDetails myTransactionDetails; 473 private Runnable myOnRollback; 474 protected RequestPartitionId myRequestPartitionId; 475 476 protected ExecutionBuilder(RequestDetails theRequestDetails) { 477 myRequestDetails = theRequestDetails; 478 } 479 480 @Override 481 public ExecutionBuilder withIsolation(Isolation theIsolation) { 482 assert myIsolation == null; 483 myIsolation = theIsolation; 484 return this; 485 } 486 487 @Override 488 public ExecutionBuilder withTransactionDetails(TransactionDetails theTransactionDetails) { 489 assert myTransactionDetails == null; 490 myTransactionDetails = theTransactionDetails; 491 return this; 492 } 493 494 @Override 495 public ExecutionBuilder withPropagation(Propagation thePropagation) { 496 assert myPropagation == null; 497 myPropagation = thePropagation; 498 return this; 499 } 500 501 @Override 502 public ExecutionBuilder withRequestPartitionId(RequestPartitionId theRequestPartitionId) { 503 assert myRequestPartitionId == null; 504 myRequestPartitionId = theRequestPartitionId; 505 return this; 506 } 507 508 @Override 509 public ExecutionBuilder readOnly() { 510 myReadOnly = true; 511 return this; 512 } 513 514 @Override 515 public ExecutionBuilder onRollback(Runnable theOnRollback) { 516 assert myOnRollback == null; 517 myOnRollback = theOnRollback; 518 return this; 519 } 520 521 @Override 522 public void execute(Runnable theTask) { 523 TransactionCallback<Void> task = tx -> { 524 theTask.run(); 525 return null; 526 }; 527 execute(task); 528 } 529 530 @Override 531 public <T> T execute(Callable<T> theTask) { 532 TransactionCallback<T> callback = tx -> invokeCallableAndHandleAnyException(theTask); 533 return execute(callback); 534 } 535 536 @Override 537 public <T> T execute(@Nonnull TransactionCallback<T> callback) { 538 assert callback != null; 539 540 return doExecute(this, callback); 541 } 542 543 @VisibleForTesting 544 public RequestPartitionId getRequestPartitionIdForTesting() { 545 return myRequestPartitionId; 546 } 547 548 @VisibleForTesting 549 public RequestDetails getRequestDetailsForTesting() { 550 return myRequestDetails; 551 } 552 553 public Propagation getPropagation() { 554 return myPropagation; 555 } 556 557 @Nullable 558 protected RequestPartitionId getEffectiveRequestPartitionId() { 559 final RequestPartitionId requestPartitionId; 560 if (myRequestPartitionId != null) { 561 requestPartitionId = myRequestPartitionId; 562 } else if (myRequestDetails != null) { 563 requestPartitionId = myRequestPartitionHelperSvc.determineGenericPartitionForRequest(myRequestDetails); 564 } else { 565 requestPartitionId = null; 566 } 567 return requestPartitionId; 568 } 569 } 570 571 /** 572 * This is just an unchecked exception so that we can catch checked exceptions inside TransactionTemplate 573 * and rethrow them outside of it 574 */ 575 static class MyException extends RuntimeException { 576 577 public MyException(Throwable theThrowable) { 578 super(theThrowable); 579 } 580 } 581 582 /** 583 * Returns true if we already have an active transaction associated with the current thread, AND 584 * either it's non-read-only or we only need a read-only transaction, AND 585 * the newly requested transaction has a propagation of REQUIRED 586 */ 587 private static boolean canReuseExistingTransaction(ExecutionBuilder theExecutionBuilder) { 588 return TransactionSynchronizationManager.isActualTransactionActive() 589 && (!TransactionSynchronizationManager.isCurrentTransactionReadOnly() || theExecutionBuilder.myReadOnly) 590 && (theExecutionBuilder.myPropagation == null 591 || theExecutionBuilder.myPropagation 592 == DEFAULT_TRANSACTION_PROPAGATION_WHEN_CHANGING_PARTITIONS); 593 } 594 595 @Nullable 596 private static <T> T executeInExistingTransaction(@Nonnull TransactionCallback<T> theCallback) { 597 ourLog.trace("executeInExistingTransaction"); 598 // TODO we could probably track the TransactionStatus we need as a thread local like we do our partition id. 599 return theCallback.doInTransaction(null); 600 } 601 602 /** 603 * Invokes {@link Callable#call()} and rethrows any exceptions thrown by that method. 604 * If the exception extends {@link BaseServerResponseException} it is rethrown unmodified. 605 * Otherwise, it's wrapped in a {@link InternalErrorException}. 606 */ 607 public static <T> T invokeCallableAndHandleAnyException(Callable<T> theTask) { 608 try { 609 return theTask.call(); 610 } catch (BaseServerResponseException e) { 611 throw e; 612 } catch (Exception e) { 613 throw new InternalErrorException(Msg.code(2223) + e.getMessage(), e); 614 } 615 } 616 617 public static <T> T executeWithDefaultPartitionInContext(@Nonnull ICallable<T> theCallback) { 618 RequestPartitionId previousRequestPartitionId = ourRequestPartitionThreadLocal.get(); 619 ourRequestPartitionThreadLocal.set(RequestPartitionId.defaultPartition()); 620 try { 621 return theCallback.call(); 622 } finally { 623 ourRequestPartitionThreadLocal.set(previousRequestPartitionId); 624 } 625 } 626 627 public static RequestPartitionId getRequestPartitionAssociatedWithThread() { 628 return ourRequestPartitionThreadLocal.get(); 629 } 630 631 /** 632 * Throws an {@link IllegalArgumentException} if a transaction is active 633 */ 634 public static void noTransactionAllowed() { 635 Validate.isTrue( 636 !TransactionSynchronizationManager.isActualTransactionActive(), 637 "Transaction must not be active but found an active transaction"); 638 } 639 640 /** 641 * Throws an {@link IllegalArgumentException} if no transaction is active 642 */ 643 public static void requireTransaction() { 644 Validate.isTrue( 645 TransactionSynchronizationManager.isActualTransactionActive(), 646 "Transaction required here but no active transaction found"); 647 } 648}