44

I'm trying to use a ContainerRequestFilter to enforce some authentication on a Tomcat based Jersey application. I followed this document. Problem : the filter is never triggered

The filter class :

@Provider
public class AuthFilter implements ContainerRequestFilter {

    @Override
    public void filter(ContainerRequestContext requestContext)
        throws IOException {

        // MY AUTHENTICATION CODE GOES HERE

    }

The web.xml file :

<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xmlns="http://java.sun.com/xml/ns/javaee"
    xmlns:web="http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd"
    xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_3_0.xsd"
    id="debate-rest"
    version="3.0">
  <display-name>rest</display-name>
   <servlet>  
    <servlet-name>Jersey REST Service</servlet-name>  
    <servlet-class>org.glassfish.jersey.servlet.ServletContainer</servlet-class>  
    <init-param>
        <param-name>com.sun.jersey.spi.container.ContainerRequestFilters</param-name>
        <param-value>com.hck.debate.rest.security.AuthFilter</param-value>
    </init-param>
    <init-param>  
      <param-name>jersey.config.server.provider.packages</param-name>  
      <param-value>com.hck.debate.rest.controller</param-value>  
    </init-param>  
    <init-param>  
        <param-name>com.sun.jersey.api.json.POJOMappingFeature</param-name>  
        <param-value>true</param-value>  
    </init-param>      
    <load-on-startup>1</load-on-startup>  
  </servlet>  
  <servlet-mapping>  
    <servlet-name>Jersey REST Service</servlet-name>  
    <url-pattern>/*</url-pattern>  
  </servlet-mapping>
koyaga
  • 1,005
  • 1
  • 9
  • 14

10 Answers10

37

Okay, I didn't get that the jersey.config.server.provider.packages init param needs to reference not only service classes (API endpoints) but ALL the classes including filters.

Now it works :

<init-param>  
  <param-name>jersey.config.server.provider.packages</param-name>  
  <param-value>com.hck.debate.rest.controller;com.hck.debate.rest.security</param-value>
</init-param>
<init-param>
    <param-name>com.sun.jersey.spi.container.ContainerRequestFilters</param-name>
    <param-value>com.hck.debate.rest.security.AuthFilter</param-value>
</init-param>
guipivoto
  • 18,327
  • 9
  • 60
  • 75
koyaga
  • 1,005
  • 1
  • 9
  • 14
  • Your above code is working as it is with TomCat, maybe it's different with Glas Fish – AZ_ Aug 26 '13 at 09:54
  • 3
    The default behavior is to scan recursively, so you can also just use the single base package `com.hck.debate.rest` – Paul Samsotha Nov 05 '15 at 04:43
  • 5
    Also note that `com.sun.jersey.spi.container.ContainerRequestFilters` and `com.sun.jersey.api.json.POJOMappingFeature` are useless. These are Jersey 1.x properties. They have no impact on Jersey 2.x – Paul Samsotha Nov 05 '15 at 04:44
  • Also note that by default, the package scanning is recursive. So instead of listing multiple packages, just declare a base package, like `com.hck.debate.rest`, and it will recursively scan all sub-packages. – Paul Samsotha Jan 11 '19 at 06:45
  • And just an FYI, the scanning is to pick up classes annotated with `@Provider` and `@Path`, so you need to make sure to also have the `@Provider` annotation. – Paul Samsotha Jan 11 '19 at 06:47
23

I also had to add the @Provider JAX-RS annotation to my filters.
This makes the filter discoverable during JAX-RS scanning phase.

@Provider
public class MyAppFilter implements ContainerRequestFilter {
    // filter logic
}
kapex
  • 28,903
  • 6
  • 107
  • 121
uris
  • 5,993
  • 3
  • 25
  • 24
5

Some hints:

  1. Make sure you're using JAX-RS version 2.17.
  2. Make sure you're using the right imports in your filter:

    • import javax.ws.rs.container.ContainerRequestContext;
    • import javax.ws.rs.container.ContainerRequestFilter;
  3. Add the @Provider annotation

ACV
  • 9,964
  • 5
  • 76
  • 81
3

We were missing the below call in our ResourceConfig implementation class:

register(CorrelationIdFilter.class);
anothernode
  • 5,100
  • 13
  • 43
  • 62
2

The minimum requirements to work filters with jersey:

  • add @Provider annotation to filter class
  • the namespace of filter class has to be included in 'jersey.config.server.provider.packages' init-param

Any other settings aren't required (e.g. 'com.sun.jersey.spi.container.ContainerRequestFilters' init-param or ResourceConfig)

crazyman
  • 979
  • 8
  • 8
2

I had the same problem for JAX-RS 2 , jersey and the below annotation fixed it

 @PreMatching
Tamas Rev
  • 7,008
  • 5
  • 32
  • 49
virtuvious
  • 2,362
  • 2
  • 21
  • 22
0

Instead of using the @Provider annotation (which did not work in my case), you can register your ContainerRequestFilter manually with your JerseyServletFactory:

JerseyServletFactory jerseyServletFactory = new JerseyServletFactory(config);
HttpServlet myServiceServlet = jerseyServletFactory.create(myResource);

// Register your ContainerRequestFilter like this
jerseyServletFactory.addRequestFilter(new MyFilter());

httpServer.register(myServiceServlet, "/api");
httpServer.start();
Pierre-Luc Pineault
  • 8,993
  • 6
  • 40
  • 55
  • and where is this piece of code supposed to be placed? – jon Mar 09 '16 at 19:40
  • @jon During initialization. I've place my code between the creation and the start of the `JettyHttpServer`, with the other stuff that needs to get initialized on start (Datasources, Resources, DAOs, etc.) – Pierre-Luc Pineault Mar 09 '16 at 19:55
  • I did the same, now i want to call destroy (for some custom cleanup task) of servlet. I am not able to get the reference back of this filter to call destroy in shutdown event. – Atul Kumar Dec 06 '17 at 11:52
0

If you were stuck like me, note that TomEE 1.7.X uses JAX-RS 1.1, which does not include ContainerRequestFilter.

Managarm
  • 1,070
  • 3
  • 12
  • 25
0

For anybody having this problem in MULE ESB. Remember to register path with:

<jersey:resources doc:name="REST">
   <component doc:name="rest component">
     <spring-object bean="endpoit"/>
   </component>
   <jersey:package packageName="path @Provider-s"/>
</jersey:resources >
HuTa
  • 168
  • 1
  • 8
-1

Instead of javax.ws.rs, i used com.sun.jersey and it worked

import com.sun.jersey.spi.container.ContainerRequestFilter import com.sun.jersey.spi.container.ContainerRequest

Dropwizard users need to do this

environment.jersey().getResourceConfig()
           .getContainerRequestFilters()
           .add(filter);