All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.jboss.ejb.protocol.remote.RetryExecutorWrapper Maven / Gradle / Ivy

Go to download

This artifact provides a single jar that contains all classes required to use remote Jakarta Enterprise Beans and Jakarta Messaging, including all dependencies. It is intended for use by those not using maven, maven users should just import the Jakarta Enterprise Beans and Jakarta Messaging BOM's instead (shaded JAR's cause lots of problems with maven, as it is very easy to inadvertently end up with different versions on classes on the class path).

There is a newer version: 35.0.0.Final
Show newest version
package org.jboss.ejb.protocol.remote;

import org.jboss.ejb._private.Logs;
import org.jboss.ejb.client.EJBClientContext;
import org.wildfly.common.context.ContextManager;
import org.wildfly.discovery.Discovery;
import org.wildfly.security.auth.client.AuthenticationContext;

import java.util.concurrent.Executor;
import java.util.function.Supplier;

/**
 * A class for queuing retry operations as well as transferring various necessary contexts from calling thread to executor thread.
 *
 * @author Stuart Douglas
 * @author Richard Achmatowicz
 */
class RetryExecutorWrapper {

    private final Object lock = new Object();
    private Task last = null;

    /**
     * A version of getExecutor() which does not modify the thread contexts of the executor thread
     *
     * @param executor the executor used to execute the runnable
     * @return the modified executor
     */
    Executor getExecutor(Executor executor) {
        if (Logs.INVOCATION.isTraceEnabled()) {
            Logs.INVOCATION.tracef("RetryExecutorWrapper: calling getExecutor(executor= %s)", executor.getClass().getName());
        }
        return runnable -> {
            synchronized (lock) {
                // create a new task
                Task task = new Task(runnable, executor);
                if (last != null) {
                    last.next = task;
                    last = task;
                } else {
                    last = task;
                    executor.execute(task);
                }
            }
        };
    }

    /**
     * A version of getExecutor which allows transferring thread contexts from the calling tread to the executor thread
     *
     * @param executor  the executor used to execute the runnable
     * @param ejbClientContext the EJBClientContext to attach to the executor thread
     * @param discovery the Discovery context to attach to the executor thread
     * @param authenticationContext the AuthenticationContext to attach to the executor thread
     * @return the modified executor
     */
    Executor getExecutor(Executor executor, EJBClientContext ejbClientContext, Discovery discovery, AuthenticationContext authenticationContext) {
        if (Logs.INVOCATION.isTraceEnabled()) {
            Logs.INVOCATION.tracef("RetryExecutorWrapper: calling getExecutor(executor= %s, ejbClientContext = %s, discovery = %s, authenticationContext = %s)",
                    executor.getClass().getName(), ejbClientContext, discovery, authenticationContext);
        }
        return runnable -> {
            synchronized (lock) {
                // provide the caller's context to the executor thread
                Runnable runnableWithContext = wrapExecutorThreadWithCallerContext(runnable, ejbClientContext, discovery, authenticationContext);

                // create a new task
                Task task = new Task(runnableWithContext, executor);
                if (last != null) {
                    last.next = task;
                    last = task;
                } else {
                    last = task;
                    executor.execute(task);
                }
            }
        };
    }

    private Runnable wrapExecutorThreadWithCallerContext(Runnable runnable, EJBClientContext callerEJBClientContext, Discovery callerDiscovery, AuthenticationContext callerAuthenticationContext) {

        Runnable callerContextTask = () -> {

            // get a copy of the executor thread context
            EJBClientContext executorEJBClientContext = EJBClientContext.getContextManager().getThreadDefault();
            Discovery executorDiscovery = Discovery.getContextManager().getThreadDefault();
            AuthenticationContext executorAuthenticationContext = AuthenticationContext.getContextManager().getThreadDefault();

            // set the context on the executor thread
            EJBClientContext.getContextManager().setThreadDefault(callerEJBClientContext);
            Discovery.getContextManager().setThreadDefault(callerDiscovery);
            AuthenticationContext.getContextManager().setThreadDefault(callerAuthenticationContext);

            // run the code
            runnable.run();

            // reset the original executor context
            EJBClientContext.getContextManager().setThreadDefault(executorEJBClientContext);
            Discovery.getContextManager().setThreadDefault(executorDiscovery);
            AuthenticationContext.getContextManager().setThreadDefault(executorAuthenticationContext);
        };
        return callerContextTask;
    }


    private class Task implements Runnable {

        private final Runnable runnable;
        private final Executor delegate;
        private Task next;

        private Task(Runnable runnable, Executor delegate) {
            this.runnable = runnable;
            this.delegate = delegate;
        }

        @Override
        public void run() {
            try {
                runnable.run();
            } catch (Throwable t) {
                Logs.MAIN.taskFailed(runnable, t);
            } finally {
                synchronized (lock) {
                    if (last == this) {
                        last = null;
                    }
                    if (next != null) {
                        next.delegate.execute(next);
                    }
                }
            }
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy