beginTransaction(); try { $result = $fn(); $db->commit(); return $result; } catch (Throwable $e) { $db->rollBack(); throw $e; } } /** * Wrapper around atomic() to retry after a retryable exception occurred * * Certain transactions might need to be retried. This is especially useful * in highly concurrent requests where a deadlocks is thrown by the database * without waiting for the lock to be freed (e.g. due to MySQL/MariaDB deadlock * detection) * * @template T * @param callable $fn * @psalm-param callable():T $fn * @param IDBConnection $db * @param int $maxRetries * * @return mixed the result of the passed callable * @psalm-return T * * @throws Exception for possible errors of commit or rollback or the custom operations within the closure * @throws Throwable any other error caused by the closure * * @since 27.0.0 */ protected function atomicRetry(callable $fn, IDBConnection $db, int $maxRetries = 3): mixed { for ($i = 0; $i < $maxRetries; $i++) { try { return $this->atomic($fn, $db); } catch (DbalException $e) { if (!$e->isRetryable() || $i === ($maxRetries - 1)) { throw $e; } logger('core')->warning('Retrying operation after retryable exception.', [ 'exception' => $e ]); } } } }