org.jboss.as.ee.concurrent.ManagedThreadFactoryImpl Maven / Gradle / Ivy
/*
* Copyright The WildFly Authors
* SPDX-License-Identifier: Apache-2.0
*/
package org.jboss.as.ee.concurrent;
import org.glassfish.enterprise.concurrent.AbstractManagedThread;
import org.glassfish.enterprise.concurrent.ContextServiceImpl;
import org.glassfish.enterprise.concurrent.internal.ManagedFutureTask;
import org.glassfish.enterprise.concurrent.spi.ContextHandle;
import org.wildfly.security.auth.server.SecurityIdentity;
import org.wildfly.security.manager.WildFlySecurityManager;
import jakarta.enterprise.concurrent.ManagedThreadFactory;
import java.security.AccessControlContext;
import java.security.AccessController;
import java.security.PrivilegedAction;
/**
* {@link ManagedThreadFactory} implementation ensuring {@link SecurityIdentity} propagation into new threads.
* @author Jan Kalina
* @author emmartins
*/
public class ManagedThreadFactoryImpl extends org.glassfish.enterprise.concurrent.ManagedThreadFactoryImpl {
/**
* the priority set on new threads
*/
private final int priority;
/**
* the factory's ACC
*/
private final AccessControlContext accessControlContext;
public ManagedThreadFactoryImpl(String name, ContextServiceImpl contextService, int priority) {
super(name, contextService, priority);
this.priority = priority;
this.accessControlContext = AccessController.getContext();
}
/**
*
* @return the priority set on new threads
*/
public int getPriority() {
return priority;
}
protected AbstractManagedThread createThread(Runnable r, final ContextHandle contextHandleForSetup) {
if (contextHandleForSetup != null) {
// app thread, do identity wrap
r = SecurityIdentityUtils.doIdentityWrap(r);
}
// use the factory's acc as privileged, otherwise the new thread inherits current thread's acc
final AbstractManagedThread t = AccessController.doPrivileged(new CreateThreadAction(r, contextHandleForSetup), accessControlContext);
// reset thread classloader to prevent leaks
if (!WildFlySecurityManager.isChecking()) {
t.setContextClassLoader(null);
} else {
AccessController.doPrivileged((PrivilegedAction) () -> {
t.setContextClassLoader(null);
return null;
});
}
return t;
}
@Override
public void taskStarting(Thread t, ManagedFutureTask task) {
super.taskStarting(t, task);
if (t instanceof ManagedThread) {
((ManagedThread)t).task = task;
}
}
@Override
public void taskDone(Thread t) {
super.taskDone(t);
if (t instanceof ManagedThread) {
((ManagedThread)t).task = null;
}
}
private final class CreateThreadAction implements PrivilegedAction {
private final Runnable r;
private final ContextHandle contextHandleForSetup;
private CreateThreadAction(Runnable r, ContextHandle contextHandleForSetup) {
this.r = r;
this.contextHandleForSetup = contextHandleForSetup;
}
@Override
public AbstractManagedThread run() {
return new ManagedThread(r, contextHandleForSetup);
}
}
/**
* Managed thread extension, to allow canceling the task running in the thread.
* @author emmartins
*/
public class ManagedThread extends org.glassfish.enterprise.concurrent.ManagedThreadFactoryImpl.ManagedThread {
volatile ManagedFutureTask task = null;
/**
*
* @param target
* @param contextHandleForSetup
*/
public ManagedThread(Runnable target, ContextHandle contextHandleForSetup) {
super(target, contextHandleForSetup);
}
/**
* Cancel the task running in the thread.
* @return
*/
boolean cancelTask() {
final ManagedFutureTask task = this.task;
if (task != null) {
return task.cancel(true);
}
return false;
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy