从 restful 网络服务中的拦截器访问 SevletContext
Access SevletContext from interceptor in restful web service
下面是我的 jersey 网络服务页面 (application.java),它调用了一个拦截器
@Context ServletContext context
@Path("/application")
public class application{
@POST
@Path("/getData")
@Produces(MediaType.APPLICATION_JSON)
@Consumes(MediaType.APPLICATION_JSON)
@Interceptors(Interceptor.class)
public int metthod(Object input) throws Exception{
// do something
}
}
如果我使用 @Context ServletContext 上下文,ServletContext 可以在 appliation.java 上访问,但是如果我在拦截器 class 中使用相同的上下文,我会得到 NullPointerException。
如何从拦截器访问 ServletContext class。
谢谢
<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns="http://java.sun.com/xml/ns/javaee" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd" version="2.5">
<display-name>ws</display-name>
<description>Web Services</description>
<servlet>
<servlet-name>jersey-servlet</servlet-name>
<servlet-class>com.sun.jersey.spi.container.servlet.ServletContainer</servlet-class>
<init-param>
<param-name>jersey.config.server.provider.packages</param-name>
<param-value>com.java.files;com.fasterxml.jackson.jaxrs.json</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>
<listener>
<listener-class>com.java.listener.appServletContextListener</listener-class>
</listener>
<servlet-mapping>
<servlet-name>jersey-servlet</servlet-name>
<url-pattern>/*</url-pattern>
</servlet-mapping>
<welcome-file-list>
<welcome-file>index.html</welcome-file>
<welcome-file>index.jsp</welcome-file>
<welcome-file>index.htm</welcome-file>
</welcome-file-list>
我将尝试帮助您解决查询并在答案中执行此操作,以便它可以格式化并更易于阅读。如果我有什么误解,请对此发表评论。
根据你的问题,我认为你没有使用 Weblogic 和 Jersey 以外的任何其他框架。
我必须在这里做一些假设:
- 您的拦截器是通过某种 XML 标记为您创建的,并且您告诉它拦截对资源方法的所有调用?
- 这是在您的 xml 配置中静态连接在一起的
是否可以更新您的问题并添加如何创建拦截器?
听起来,Weblogic 提供了一个容器,它是 运行 为您提供的 jax.rs 参考实现 (jersey)。 (如果我错了请纠正我)。
本质问题是Weblogic没有上下文的概念。如果你看到你的注释,你可以看到 Annotation 是一个 jax-rs 注释:
javax.ws.rs.core.Context
这意味着您可以将此注释与 jax-rs 框架的参考实现一起使用,这将是球衣。 Jersey 为您管理 bean(creation/injection/interception 等)并且是请求范围的,这就是为什么它可以为您注入各种有趣的属性,这些属性与您的 Web 服务有关。
这也是 @Context 注释在您的资源中起作用的原因,因为您的资源是球衣处理的。但是,您的拦截器似乎是 Weblogic 拦截器,由 weblogic 处理。它没有挂接到您的球衣逻辑中,因此注入在这里不起作用。
有一些方法可以解决这个问题,例如使用 DI 框架和作用域 bean(例如 guice 可以为您做这件事),但这不是必需的。对于您的问题,有一种更加集成和流畅的拦截资源调用的方式。
网络服务不应该从应用程序内部调用(例如,它对 HTTP 请求作出反应)。这意味着,您将希望使用 jersey 拦截器拦截这些调用,而不是 weblogic 拦截器。为此,您可以看到这两个接口:
javax.ws.rs.container.ContainerRequestFilter
javax.ws.rs.container.ContainerResponseFilter
您猜对了 - 其中一个在您的资源获取请求之前拦截请求,另一个在您的资源处理完请求后拦截。
您可以在此处阅读更多相关信息:https://jersey.java.net/documentation/latest/filters-and-interceptors.html
所以在您的用例中,您希望在将请求传递给您的资源之前拦截该请求。如果你的 ServletContextMap 有一些值,你会想要中止请求(return 它给用户一些错误)。
这样做的好处是过滤器由 jersey 处理,因此它们可以由 jersey 注入你的上下文。
考虑一下:
public class MyFilter implements ContainerRequestFilter {
@Context
private ServletContext context;
@Override
public void filter(ContainerRequestContext requestContext) throws IOException {
boolean result = evaluateMyMap();
if(!result) {
// This is the jersey way of not calling proceed. It will return a 400 response
// to the user with the message
requestContext.abortWith(Response.status(Status.BAD_REQUEST)
.entity("The Hashmap in the Context has some wonky value").build());
}
}
private boolean evaluateMyMap() {
// access your context here and do your map evanulation
// return true if you want the request to be processed, false otherwise
return false;
}
}
在过滤器 class 中,您可以访问注入的上下文,您也可以访问您的地图。
请注意,您的过滤器也必须在 jersey 中注册。如何做到这一点,我不确定(因为我以前没有使用过 weblogic)
我希望这能解释您的问题以及解决方法。如果您需要更多 advise/help,请告诉我。
此致,
阿图尔
下面是我的 jersey 网络服务页面 (application.java),它调用了一个拦截器
@Context ServletContext context
@Path("/application")
public class application{
@POST
@Path("/getData")
@Produces(MediaType.APPLICATION_JSON)
@Consumes(MediaType.APPLICATION_JSON)
@Interceptors(Interceptor.class)
public int metthod(Object input) throws Exception{
// do something
}
}
如果我使用 @Context ServletContext 上下文,ServletContext 可以在 appliation.java 上访问,但是如果我在拦截器 class 中使用相同的上下文,我会得到 NullPointerException。
如何从拦截器访问 ServletContext class。
谢谢
<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns="http://java.sun.com/xml/ns/javaee" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd" version="2.5">
<display-name>ws</display-name>
<description>Web Services</description>
<servlet>
<servlet-name>jersey-servlet</servlet-name>
<servlet-class>com.sun.jersey.spi.container.servlet.ServletContainer</servlet-class>
<init-param>
<param-name>jersey.config.server.provider.packages</param-name>
<param-value>com.java.files;com.fasterxml.jackson.jaxrs.json</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>
<listener>
<listener-class>com.java.listener.appServletContextListener</listener-class>
</listener>
<servlet-mapping>
<servlet-name>jersey-servlet</servlet-name>
<url-pattern>/*</url-pattern>
</servlet-mapping>
<welcome-file-list>
<welcome-file>index.html</welcome-file>
<welcome-file>index.jsp</welcome-file>
<welcome-file>index.htm</welcome-file>
</welcome-file-list>
我将尝试帮助您解决查询并在答案中执行此操作,以便它可以格式化并更易于阅读。如果我有什么误解,请对此发表评论。
根据你的问题,我认为你没有使用 Weblogic 和 Jersey 以外的任何其他框架。
我必须在这里做一些假设:
- 您的拦截器是通过某种 XML 标记为您创建的,并且您告诉它拦截对资源方法的所有调用?
- 这是在您的 xml 配置中静态连接在一起的
是否可以更新您的问题并添加如何创建拦截器?
听起来,Weblogic 提供了一个容器,它是 运行 为您提供的 jax.rs 参考实现 (jersey)。 (如果我错了请纠正我)。
本质问题是Weblogic没有上下文的概念。如果你看到你的注释,你可以看到 Annotation 是一个 jax-rs 注释:
javax.ws.rs.core.Context
这意味着您可以将此注释与 jax-rs 框架的参考实现一起使用,这将是球衣。 Jersey 为您管理 bean(creation/injection/interception 等)并且是请求范围的,这就是为什么它可以为您注入各种有趣的属性,这些属性与您的 Web 服务有关。
这也是 @Context 注释在您的资源中起作用的原因,因为您的资源是球衣处理的。但是,您的拦截器似乎是 Weblogic 拦截器,由 weblogic 处理。它没有挂接到您的球衣逻辑中,因此注入在这里不起作用。
有一些方法可以解决这个问题,例如使用 DI 框架和作用域 bean(例如 guice 可以为您做这件事),但这不是必需的。对于您的问题,有一种更加集成和流畅的拦截资源调用的方式。
网络服务不应该从应用程序内部调用(例如,它对 HTTP 请求作出反应)。这意味着,您将希望使用 jersey 拦截器拦截这些调用,而不是 weblogic 拦截器。为此,您可以看到这两个接口:
javax.ws.rs.container.ContainerRequestFilter
javax.ws.rs.container.ContainerResponseFilter
您猜对了 - 其中一个在您的资源获取请求之前拦截请求,另一个在您的资源处理完请求后拦截。
您可以在此处阅读更多相关信息:https://jersey.java.net/documentation/latest/filters-and-interceptors.html
所以在您的用例中,您希望在将请求传递给您的资源之前拦截该请求。如果你的 ServletContextMap 有一些值,你会想要中止请求(return 它给用户一些错误)。
这样做的好处是过滤器由 jersey 处理,因此它们可以由 jersey 注入你的上下文。
考虑一下:
public class MyFilter implements ContainerRequestFilter {
@Context
private ServletContext context;
@Override
public void filter(ContainerRequestContext requestContext) throws IOException {
boolean result = evaluateMyMap();
if(!result) {
// This is the jersey way of not calling proceed. It will return a 400 response
// to the user with the message
requestContext.abortWith(Response.status(Status.BAD_REQUEST)
.entity("The Hashmap in the Context has some wonky value").build());
}
}
private boolean evaluateMyMap() {
// access your context here and do your map evanulation
// return true if you want the request to be processed, false otherwise
return false;
}
}
在过滤器 class 中,您可以访问注入的上下文,您也可以访问您的地图。
请注意,您的过滤器也必须在 jersey 中注册。如何做到这一点,我不确定(因为我以前没有使用过 weblogic)
我希望这能解释您的问题以及解决方法。如果您需要更多 advise/help,请告诉我。
此致, 阿图尔