4

我们通过 webapp 中的 wro 提供 javascript 资源(和其他资源)。在 PROD 环境中,浏览器获取(例如)app.js angular webapp 的内容,其中包含未来一年的“过期”标题。

这意味着对于后续请求,浏览器从缓存中获取它而不向服务器请求。如果我们部署新版本的 webapp,浏览器不会获取新版本,因为它是从本地缓存中获取的。

目标是配置 wro 或/和 spring,以便正确设置标头以使浏览器每次都执行请求,并且服务器返回未修改的 304。所以我们会让客户端在新部署时自动“更新”。有人已经做到了吗?

我们使用 Spring 的 Java 配置:

@Configuration
public class Wro4jConfiguration {   

@Value("${app.webapp.web.minimize}")
private String minimize;

@Value("${app.webapp.web.disableCache}")
private String disableCache;

@Autowired
private Environment env;

@Bean(name = "wroFilter")
public WroFilter wroFilter() {
    ConfigurableWroFilter filter = new ConfigurableWroFilter();
    filter.setWroManagerFactory(new Wro4jManagerFactory());
    filter.setWroConfigurationFactory(createProperties());
    return filter;
}

private PropertyWroConfigurationFactory createProperties() {
    Properties props = new Properties();
    props.setProperty("jmxEnabled", "false");
    props.setProperty("debug", String.valueOf(!env.acceptsProfiles(EnvConstants.PROD)));
    props.setProperty("gzipResources", "false");
    props.setProperty("ignoreMissingResources", "true");
    props.setProperty("minimizeEnabled", minimize);
    props.setProperty("resourceWatcherUpdatePeriod", "0");
    props.setProperty("modelUpdatePeriod", "0");
    props.setProperty("cacheGzippedContent", "false");
    // let's see if server-side cache is disabled (DEV only)
    if (Boolean.valueOf(disableCache)) {
        props.setProperty("resourceWatcherUpdatePeriod", "1");
        props.setProperty("modelUpdatePeriod", "5");
    }
    return new PropertyWroConfigurationFactory(props);
}
}
4

2 回答 2

3

默认情况下,WroFilter 设置以下标头:ETag(资源的 md5 校验和)、Cache-Control(public,max-age=315360000)、Expires(资源创建后 1 年)。

关于这些标题的重要性有很多细节。简短的解释是这样的:

当服务器从客户端请求中读取 ETag 时,服务器可以确定是发送文件(HTTP 200)还是告诉客户端只使用其本地副本(HTTP 304)。ETag 基本上只是一个文件的校验和,当文件的内容发生变化时,它会在语义上发生变化。如果只发送 ETag,客户端将始终需要发出请求。

Expires 和 Cache-Control 标头非常相似,客户端(和代理/缓存)使用它们来确定它是否甚至需要向服务器发出请求。

因此,您真正想要做的是使用 BOTH 标头 - 根据内容更改的频率将 Expires 标头设置为合理的值。然后配置要发送的 ETag,这样当客户端向服务器发送请求时,它可以更容易地确定是否将文件发回。

如果您希望客户端始终检查最新的资源版本,则不应发送 expires & cache-control 标头。

或者,有一种更激进的缓存技术:将资源的校验和编码到其路径中。结果,每次更改资源时,都会更改该资源的路径。这种方法保证客户端总是请求最新的版本。对于这种方法,理论上资源永远不会过期,因为每次更改资源时校验和都会更改。

于 2015-06-16T14:51:41.630 回答
2

根据 Alex 的信息和文档参考,我最终覆盖了 WroFilter.setResponseHeaders 以放置适当的过期值。这工作正常。Wro 已经负责设置 ETag、Date 等,所以我只覆盖过期延迟和日期。

@Configuration
public class Wro4jConfiguration {

    @Value("${app.webapp.web.browserCache.maxAgeInHours}")
    private String maxAgeInHours;

    @Bean(name = "wroFilter")
    public WroFilter wroFilter() {
        ConfigurableWroFilter filter = createFilter();

        filter.setWroManagerFactory(new Wro4jManagerFactory());
        filter.setWroConfigurationFactory(createProperties());

        return filter;
    }

    private ConfigurableWroFilter createFilter() {
        return new ConfigurableWroFilter() {

            private final int BROWSER_CACHE_HOURS = Integer.parseInt(maxAgeInHours);
            private final int BROWSER_CACHE_SECONDS = BROWSER_CACHE_HOURS * 60 * 60;

            @Override
            protected void setResponseHeaders(final HttpServletResponse response){
                super.setResponseHeaders(response);

                if (!getConfiguration().isDebug()) {
                    ZonedDateTime cacheExpires = ZonedDateTime.of(LocalDateTime.now(), ZoneId.of("GMT")).plusHours(BROWSER_CACHE_HOURS);
                    String cacheExpiresStr = cacheExpires.format(DateTimeFormatter.RFC_1123_DATE_TIME);

                    response.setHeader(HttpHeader.EXPIRES.toString(), cacheExpiresStr);
                    response.setHeader(HttpHeader.CACHE_CONTROL.toString(), "public,  max-age=" + BROWSER_CACHE_SECONDS);
                }
            }
        };
    }

    // Other config methods

}
于 2015-06-18T10:23:02.473 回答