/* * JBoss, Home of Professional Open Source. * Copyright 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.test.integration.ejb.security; import java.util.concurrent.Callable; import javax.ejb.EJBAccessException; import javax.naming.InitialContext; import org.hamcrest.CoreMatchers; import org.jboss.arquillian.container.test.api.Deployer; import org.jboss.arquillian.container.test.api.Deployment; import org.jboss.arquillian.container.test.api.OperateOnDeployment; import org.jboss.arquillian.junit.Arquillian; import org.jboss.arquillian.test.api.ArquillianResource; import org.jboss.as.arquillian.api.ServerSetup; import org.jboss.as.test.categories.CommonCriteria; import org.jboss.as.test.integration.ejb.security.runasprincipal.Caller; import org.jboss.as.test.integration.ejb.security.runasprincipal.CallerRunAsPrincipal; import org.jboss.as.test.integration.ejb.security.runasprincipal.CallerWithIdentity; import org.jboss.as.test.integration.ejb.security.runasprincipal.SingletonBean; import org.jboss.as.test.integration.ejb.security.runasprincipal.StatelessBBean; import org.jboss.as.test.integration.ejb.security.runasprincipal.WhoAmI; import org.jboss.as.test.integration.ejb.security.runasprincipal.transitive.SimpleSingletonBean; import org.jboss.as.test.integration.ejb.security.runasprincipal.transitive.SingletonStartupBean; import org.jboss.as.test.integration.ejb.security.runasprincipal.transitive.StatelessSingletonUseBean; import org.jboss.as.test.integration.security.common.AbstractSecurityDomainSetup; import org.jboss.as.test.shared.integration.ejb.security.Util; import org.jboss.as.test.shared.util.AssumeTestGroupUtil; import org.jboss.logging.Logger; import org.jboss.shrinkwrap.api.Archive; import org.jboss.shrinkwrap.api.ShrinkWrap; import org.jboss.shrinkwrap.api.asset.StringAsset; import org.jboss.shrinkwrap.api.spec.WebArchive; import org.junit.Assert; import org.junit.BeforeClass; import org.junit.Test; import org.junit.experimental.categories.Category; import org.junit.runner.RunWith; /** * Migration of test from EJB3 testsuite [JBQA-5451] Testing calling with runasprincipal annotation (ejbthree1945) * * @author Carlo de Wolf, Ondrej Chaloupka */ @RunWith(Arquillian.class) @ServerSetup({EjbSecurityDomainSetup.class}) @Category(CommonCriteria.class) public class RunAsPrincipalTestCase { private static final Logger log = Logger.getLogger(RunAsPrincipalTestCase.class); private static final String STARTUP_SINGLETON_DEPLOYMENT = "startup-transitive-singleton"; private static final String DEPLOYMENT = "runasprincipal-test"; @ArquillianResource public Deployer deployer; @BeforeClass public static void beforeClass() { AssumeTestGroupUtil.assumeElytronProfileTestsEnabled(); } @Deployment(name = STARTUP_SINGLETON_DEPLOYMENT, managed = false, testable = false) public static Archive<?> runAsStartupTransitiveDeployment() { // using JavaArchive doesn't work, because of a bug in Arquillian, it only deploys wars properly final WebArchive war = ShrinkWrap.create(WebArchive.class, STARTUP_SINGLETON_DEPLOYMENT + ".war") .addClass(WhoAmI.class) .addClass(StatelessBBean.class) .addClass(SingletonStartupBean.class) .addPackage(Assert.class.getPackage()) .addClass(Util.class) .addClass(Entry.class) .addClass(RunAsPrincipalTestCase.class) .addClasses(AbstractSecurityDomainSetup.class, EjbSecurityDomainSetup.class) .addAsWebInfResource(RunAsPrincipalTestCase.class.getPackage(), "jboss-ejb3.xml", "jboss-ejb3.xml") .addAsManifestResource(new StringAsset("Dependencies: org.jboss.as.controller-client,org.jboss.dmr\n"), "MANIFEST.MF"); war.addPackage(CommonCriteria.class.getPackage()); return war; } @Deployment public static Archive<?> runAsDeployment() { //XXX This is hack related to Elytron profile assumption in beforeClass() method. // Arquillian does deploy before executing @BeforeClass annotated methods. So let's create a dummy deployment if (! AssumeTestGroupUtil.CONDITION_SKIP_ELYTRON_PROFILE.get()) { return ShrinkWrap.create(WebArchive.class).addAsWebResource(new StringAsset(""), "index.html"); } // using JavaArchive doesn't work, because of a bug in Arquillian, it only deploys wars properly final WebArchive war = ShrinkWrap.create(WebArchive.class, DEPLOYMENT + ".war") .addPackage(WhoAmI.class.getPackage()) .addClass(SimpleSingletonBean.class) .addClass(StatelessSingletonUseBean.class) .addClass(Util.class) .addClass(Entry.class) .addClass(RunAsPrincipalTestCase.class) .addClasses(AbstractSecurityDomainSetup.class, EjbSecurityDomainSetup.class) .addAsWebInfResource(RunAsPrincipalTestCase.class.getPackage(), "jboss-ejb3.xml", "jboss-ejb3.xml") .addAsManifestResource(new StringAsset("Dependencies: org.jboss.as.controller-client,org.jboss.dmr\n"), "MANIFEST.MF"); war.addPackage(CommonCriteria.class.getPackage()); return war; } private WhoAmI lookupCallerWithIdentity() throws Exception { return (WhoAmI)new InitialContext().lookup("java:module/" + CallerWithIdentity.class.getSimpleName() + "!" + WhoAmI.class.getName()); } private WhoAmI lookupSingleCallerWithIdentity() throws Exception { return (WhoAmI)new InitialContext().lookup("java:module/" + SingletonBean.class.getSimpleName() + "!" + WhoAmI.class.getName()); } private WhoAmI lookupCaller() throws Exception { return (WhoAmI)new InitialContext().lookup("java:module/" + Caller.class.getSimpleName() + "!" + WhoAmI.class.getName()); } private WhoAmI lookupCallerRunAsPrincipal() throws Exception { return (WhoAmI)new InitialContext().lookup("java:module/" + CallerRunAsPrincipal.class.getSimpleName() + "!" + WhoAmI.class.getName()); } private WhoAmI lookupSingletonUseBeanWithIdentity() throws Exception { return (WhoAmI) new InitialContext().lookup("java:module/" + StatelessSingletonUseBean.class.getSimpleName() + "!" + WhoAmI.class.getName()); } @Test public void testJackInABox() throws Exception { final Callable<Void> callable = () -> { WhoAmI bean = lookupCallerWithIdentity(); String actual = bean.getCallerPrincipal(); Assert.assertEquals("jackinabox", actual); return null; }; Util.switchIdentitySCF("user1", "password1", callable); } @Test public void testSingletonPostconstructSecurity() throws Exception { final Callable<Void> callable = () -> { WhoAmI bean = lookupSingleCallerWithIdentity(); String actual = bean.getCallerPrincipal(); Assert.assertEquals("Helloween", actual); return null; }; Util.switchIdentitySCF("user1", "password1", callable); } @Test public void testRunAsPrincipal() throws Exception { WhoAmI bean = lookupCallerRunAsPrincipal(); try { String actual = bean.getCallerPrincipal(); Assert.fail("Expected EJBAccessException and it was get identity: " + actual); } catch (EJBAccessException e) { // good } } @Test public void testAnonymous() throws Exception { final Callable<Void> callable = () -> { WhoAmI bean = lookupCaller(); String actual = bean.getCallerPrincipal(); Assert.assertEquals("anonymous", actual); return null; }; Util.switchIdentitySCF("user1", "password1", callable); } @Test @OperateOnDeployment(STARTUP_SINGLETON_DEPLOYMENT) public void testStartupSingletonPostconstructSecurityNotPropagating() { try { deployer.deploy(STARTUP_SINGLETON_DEPLOYMENT); Assert.fail("Deployment should fail"); } catch (Exception dex) { Throwable t = checkEjbException(dex); log.trace("Expected deployment error because the Singleton has nosecurity context per itself", dex.getCause()); Assert.assertThat(t.getMessage(), t.getMessage(), CoreMatchers.containsString("WFLYEJB0364")); } finally { deployer.undeploy(STARTUP_SINGLETON_DEPLOYMENT); } } @Test public void testSingletonPostconstructSecurityNotPropagating() throws Exception { final Callable<Void> callable = () -> { WhoAmI bean = lookupSingletonUseBeanWithIdentity(); //To load the singleton bean.getCallerPrincipal(); Assert.fail("Deployment should fail"); return null; }; try { Util.switchIdentitySCF("user1", "password1", callable); } catch (Exception dex) { Throwable t = checkEjbException(dex); log.trace("Expected deployment error because the Singleton has nosecurity context per itself", dex.getCause()); Assert.assertThat(t.getMessage(), t.getMessage(), CoreMatchers.containsString("WFLYEJB0364")); } } private Throwable checkEjbException(Throwable ex) { if (ex instanceof EJBAccessException) { return ex; } if (ex.getCause() != null) { return checkEjbException(ex.getCause()); } return ex; } }