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

org.jboss.as.ejb3.inflow.JBossMessageEndpointFactory Maven / Gradle / Ivy

There is a newer version: 33.0.2.Final
Show newest version
/*
 * JBoss, Home of Professional Open Source.
 * Copyright (c) 2011, Red Hat, Inc., and individual contributors
 * as indicated by the @author tags. See the copyright.txt file in the
 * distribution for a full listing of individual contributors.
 *
 * This is free software; you can redistribute it and/or modify it
 * under the terms of the GNU Lesser General Public License as
 * published by the Free Software Foundation; either version 2.1 of
 * the License, or (at your option) any later version.
 *
 * This software is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
 * License along with this software; if not, write to the Free
 * Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
 * 02110-1301 USA, or see the FSF site: http://www.fsf.org.
 */
package org.jboss.as.ejb3.inflow;

import java.lang.reflect.Method;
import java.security.PrivilegedAction;
import java.util.concurrent.atomic.AtomicInteger;

import javax.resource.spi.UnavailableException;
import javax.resource.spi.endpoint.MessageEndpoint;
import javax.resource.spi.endpoint.MessageEndpointFactory;
import javax.transaction.xa.XAResource;

import org.jboss.invocation.proxy.ProxyConfiguration;
import org.jboss.invocation.proxy.ProxyFactory;
import org.wildfly.security.manager.WildFlySecurityManager;

import static java.security.AccessController.doPrivileged;
import static java.util.concurrent.TimeUnit.MILLISECONDS;

/**
 * @author Carlo de Wolf
 */
public class JBossMessageEndpointFactory implements MessageEndpointFactory {
    private static final AtomicInteger PROXY_ID = new AtomicInteger(0);
    private final MessageEndpointService service;
    private final ProxyFactory factory;
    private final Class endpointClass;

    public JBossMessageEndpointFactory(final ClassLoader classLoader, final MessageEndpointService service, final Class ejbClass, final Class messageListenerInterface) {
        // todo: generics bug; only Object.class is a Class.  Everything else is Class aka Class
        this.service = service;
        final ProxyConfiguration configuration = new ProxyConfiguration()
                .setClassLoader(classLoader)
                .setProxyName(ejbClass.getName() + "$$$endpoint" + PROXY_ID.incrementAndGet())
                .setSuperClass(ejbClass)
                .setProtectionDomain(ejbClass.getProtectionDomain())
                .addAdditionalInterface(MessageEndpoint.class)
                .addAdditionalInterface(messageListenerInterface);
        this.factory = new ProxyFactory(configuration);
        this.endpointClass = ejbClass;
    }

    @Override
    public MessageEndpoint createEndpoint(XAResource xaResource) throws UnavailableException {
        return createEndpoint(xaResource, 0);
    }

    @Override
    public MessageEndpoint createEndpoint(XAResource xaResource, long timeout) throws UnavailableException {
        Object delegate = service.obtain(timeout, MILLISECONDS);
        MessageEndpointInvocationHandler handler = new MessageEndpointInvocationHandler(service, delegate, xaResource);
        // New instance creation leads to component initialization which needs to have the TCCL that corresponds to the
        // component classloader. @see https://issues.jboss.org/browse/WFLY-3989
        final ClassLoader oldTCCL = WildFlySecurityManager.getCurrentContextClassLoaderPrivileged();

        try {
            WildFlySecurityManager.setCurrentContextClassLoaderPrivileged(this.factory.getClassLoader());

            if (System.getSecurityManager() == null) {
                return createEndpoint(factory, handler);
            } else {
                return doPrivileged((PrivilegedAction) () -> {
                    return createEndpoint(factory, handler);
                });
            }

        } finally {
            // reset TCCL
            WildFlySecurityManager.setCurrentContextClassLoaderPrivileged(oldTCCL);
        }

    }

    @Override
    public boolean isDeliveryTransacted(Method method) throws NoSuchMethodException {
        return service.isDeliveryTransacted(method);
    }

    @Override
    public String getActivationName() {
        return service.getActivationName();
    }

    @Override
    public Class getEndpointClass() {
        // The MDB class is the message endpoint class
        return this.endpointClass;
    }

    private MessageEndpoint createEndpoint(ProxyFactory factory, MessageEndpointInvocationHandler handler) {
        try {
            return (MessageEndpoint) factory.newInstance(handler);
        } catch (InstantiationException e) {
            throw new RuntimeException(e);
        } catch (IllegalAccessException e) {
            throw new RuntimeException(e);
        }
    }
}