如何删除HTTP响应头?

问题描述 投票:26回答:4

我有一个情况,其中一个响应标头Content-Disposition必须删除。所以我想写一个servlet过滤器来做到这一点。但我意识到HttpServletResponse只有setHeader()方法但没有方法可以删除它。我怎样才能做到这一点?

servlets servlet-filters response-headers
4个回答
42
投票

之后,您无法通过标准Servlet API删除标头。最好的办法就是阻止设置标题。您可以通过创建一个Filter来替换ServletResponse,并使用自定义的HttpServletResponseWrapper实现,只要标题名称为setHeader(),就会跳过Content-Disposition的工作。

基本上:

@Override
public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {
    chain.doFilter(request, new HttpServletResponseWrapper((HttpServletResponse) response) {
        public void setHeader(String name, String value) {
            if (!name.equalsIgnoreCase("Content-Disposition")) {
                super.setHeader(name, value);
            }
        }
    });
}

只需将该过滤器映射到感兴趣的URL模式即可运行。


3
投票

这可能不符合Servlet API,但将值设置为null可以在GlassFish 4上运行,也可能在Tomcat上运行,因为它就是GlassFish下面的内容。

我们确实需要更新Servlet API规范,以添加允许删除标头的方法或正式支持使用带有空值的setHeader。

这很重要的一个例子是,如果您在Web应用程序上使用安全约束(SSL / TLS),那么静态资源缓存会因容器将自动添加标头以防止缓存而变得复杂(您可以尝试使用disableProxyCaching和Tomcat / GlassFish上的securePagesWithPragma)。我已经有一个用于缓存控制的servlet过滤器,非常适合非安全内容,所以我想将缓存控制全部保存在一个地方,只需将Prama和Cache-Control设置为null即可清除任何容器添加的标头。


1
投票

这对我使用Spring 4不起作用。我试图删除Expires响应头。对于每一页。像这样:

public class CachingFilter implements Filter {
    private final Log logger = LogFactory.getLog(getClass());

    public CachingFilter() {}
    public void destroy() {}

    @Override
    public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {
        logger.debug("doFilter()");
        chain.doFilter(request, new HttpServletResponseWrapper((HttpServletResponse) response) {
            public void setHeader(String name, String value) {
                logger.debug("setHeader(" + name + ","+value+")");

                if (!name.equalsIgnoreCase("Expires")) {
                    super.setHeader(name, value);
                }
            }
        });
    }

    public void init(FilterConfig fConfig) throws ServletException {}
}

以下是我添加过滤器的方法:

public class AppConfig implements WebApplicationInitializer {
    private static final String DISPATCHER_SERVLET_NAME = "dispatcher";
    private static final String DISPATCHER_SERVLET_MAPPING = "/";

    @Override
    public void onStartup(ServletContext servletContext) throws ServletException {
        AnnotationConfigWebApplicationContext rootContext = new AnnotationConfigWebApplicationContext();
        rootContext.register(AppContext.class);

        ServletRegistration.Dynamic dispatcher = servletContext.addServlet(DISPATCHER_SERVLET_NAME, new DispatcherServlet(rootContext));
        dispatcher.setLoadOnStartup(1);
        dispatcher.addMapping(DISPATCHER_SERVLET_MAPPING);

        EnumSet<DispatcherType> dispatcherTypes = EnumSet.of(DispatcherType.REQUEST, DispatcherType.FORWARD);

        FilterRegistration.Dynamic noCache = servletContext.addFilter("noCacheFilter", new CachingFilter());
        noCache.addMappingForUrlPatterns(dispatcherTypes, true, "/*");

        servletContext.addListener(new ContextLoaderListener(rootContext));
    }
}

为Expires和Cache-Control调用setHeader(),但我无法覆盖Expires过滤器值或Cache-Control值。我可以添加到Cache-Control值。如果我在Cache-Control上调用setHeader,它将变成一个值数组。但我需要删除标题。


1
投票

作为其他回应。设置后无法删除标题,至少不是标准(glassfish允许清除标题,将其值设置为null)。所以在一天结束时你会有两个选择:

  1. 使用response.reset()重置响应 - 这有效地删除了所有标头以及任何缓冲数据,具体取决于您的情况可能是一个很好的选择(在我的情况下是在验证验证错误之后)。如果响应已经提交,您将收到IllegalStateException。
  2. 将标题设置为空字符串,显然这不会删除标题。但http规范在Accept-Encoding,TE(传输编码)和HOS​​T头中只有一些定义和空值,因此根据您的需要,您可以在应用层中控制它。
© www.soinside.com 2019 - 2024. All rights reserved.