A CDI Portable Extension for Java EE 7 (and probably Java EE 8) that allows you to use @RolesAllowed({"role-name"})
on CDI Beans and their Methods.
Makes this work:
@ApplicationScoped
@RolesAllowed({"required-role"})
public class MyBusinessLogic {
public void wahoo() {
// only invokable by users with required-role
}
}
If the person doesn't have the the required role, the DefaultJavaXSecurityFailureHandler
class throws a new WebApplicationException(Response.Status.UNAUTHORIZED)
. This behavior is customizable, see below.
A common mistake is that someone sees javax.annotation.security and applies it to a CDI bean, thinking since it belongs to a general javax package, and not a EJB package. Well it doesn't sadly, it only works for EJB (Stateless, Stateful, Singleton, and MDB).
All files in this project are licensed Apache Source License 2.0. Please consider submitting any changes you make back to this project!
You must be in a servlet lifecycle. Said differently: you have to be handling an HTTP Request. If the bean is being called by another initiator like an MDB or Timer, you'll run into some problems (and you probably have a bug in your program too).
Maven Coordinates:
<dependency>
<groupId>com.github.exabrial</groupId>
<artifactId>javax-security-cdi-extension</artifactId>
<version>1.0.0</version>
<scope>runtime</scope>
</dependency>
If you are customizing the behavior, your scope
will need to be compile
.
You must have authentication setup in your webapp.
Example WEB-INF/web.xml
for Basic auth:
<?xml version="1.0" encoding="UTF-8"?>
<web-app
version="3.1"
xmlns="http://xmlns.jcp.org/xml/ns/javaee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_3_1.xsd">
<login-config>
<auth-method>BASIC</auth-method>
</login-config>
</web-app>
If you have a CDI @Producer
for SLF4J loggers, the extension will produce useful warning logs on failures. You probably want to know if a bunch of login failures are happening in your apps, so I suggest doing this.
Example CDI Logger Producer:
@Produces
@Dependent
public Logger createLogger(final InjectionPoint injectionPoint) {
Class<?> declaringClass = injectionPoint.getMember().getDeclaringClass();
return LoggerFactory.getLogger(declaringClass);
}
If you want to disable security (maybe for localhost development), you can disable it by creating a Boolean CDI Producer with the qualifer @com.github.exabrial.cdi.javaxsecurity.SkipSecurity
:
Example of skipping security:
@Produces
@SkipSecurity
public Boolean skipSecurity() {
return Boolean.TRUE;
}
You can implement the com.github.exabrial.cdi.javaxsecurity.JavaXSecurityFailureHandler
interface and create custom behavior when things go wrong. Implement the interface, then mark it as an @Alternative
and with @Priority
.
Example:
@ApplicationScoped
@Alternative
@Priority
public class MyJavaXSecurityFailureHandler implements JavaXSecurityFailureHandler {
@Override
public void authenticationFailure() {
// ...
}
@Override
public void authorizationFailure(Class<? extends Object> targetClass, Method targetMethod, String roleName) {
// ...
}
}