9

问题

Symfony2 上是否可以按文件使用 assets_version

背景

我们使用 assets_version 和assets_version_format来管理文件版本并强制 CDN 和浏览器缓存上的缓存更新。

这就像魅力!但是,我们发现所有使用的静态资源只有一个assets_version参数。

这是一个问题,因为我们的 webapp 拥有大量静态资源,并且我们每天都在部署对 prod 环境的更改。这种情况会杀死缓存。:(

这是我们当前的配置:

配置.yml

framework:
   
    templating:
        engines: ['twig']
        assets_version: %assets_version%
        assets_version_format:  "stv%%2$s/%%1$s"

# Assetic Configuration
assetic:
    debug:          %kernel.debug%
    use_controller: false
    # java: /usr/bin/java
    filters:
         cssrewrite: ~
         closure:
             jar: %kernel.root_dir%/java/compiler.jar
         yui_css:
             jar: %kernel.root_dir%/java/yuicompressor-2.4.6.jar

sometemplate.html.twig

    {% stylesheets 'bundles/webapp/css/funCommon.css'
                       'bundles/webapp/css/funMobile.css'
                       filter='?yui_css'
        %}
        <link rel=stylesheet href='{{ asset_url }}'>
        {% endstylesheets %}
    
    {% javascripts 'bundles/webapp/js/app.js'
                   'bundles/webapp/js/utils.js'
                    filter='?closure'  %}
        <script src="{{ asset_url }}"></script>
        {% endjavascripts %}

{% javascripts 'bundles/webapp/js/moduleX.js'
                   'bundles/webapp/js/utilsX.js'
                    filter='?closure'  %}
        <script src="{{ asset_url }}"></script>
        {% endjavascripts %}

当我更改任何css 文件或模块 JS 或任何其他文件时,所有路径都会更改

我想通过 javascript/stylesheet twig 标签的参数来管理 assets_version_format 的版本参数。

这就是我要找的:

{% javascripts 'bundles/webapp/js/app.js'
               'bundles/webapp/js/utils.js'
                filter='?closure' **version='XX'** %}
    <script src="{{ asset_url }}"></script>
    {% endjavascripts %}
  
4

4 回答 4

17

经过几天的搜索,我找到了选项AsseticBundle

http://symfony.com/doc/2.0/reference/configuration/framework.html#full-default-configuration

使用该配置选项,我可以执行以下操作:

{% javascripts file package='packageName' %}

或者

{{asset(file,packageName)}}

样本:

配置.yml

framework:

    templating:
        engines: ['twig']
        assets_version: %assets_version%
        assets_version_format:  "stv%%2$s/%%1$s"
        packages:
             css:
                version: 6.1
                version_format: "stv%%2$s/%%1$s"
             jsApp:
                version: 4.2
                version_format: "stv%%2$s/%%1$s"

sometemplate.html.twig

<link rel=stylesheet href='{{ asset('bundles/webapp/css/funCommon.css','css') }}'>

{% javascripts 'bundles/webapp/js/app.js'
               'bundles/webapp/js/utils.js'
                filter='?closure'
                package='jsApp'
 %}
    <script src="{{ asset_url }}"></script>
 {% endjavascripts %}

这个的输出是:

<link rel=stylesheet href="http://static.domain.com/stv6.1/css/HASH.css">

<script src="http://static.domain.com/stv4.2/js/HASH.js"></script>

对我来说,这是按文件管理资产版本的最简单方法。

于 2013-07-29T18:59:47.827 回答
4

如果您尝试将assets_version参数与javascripts样式表帮助程序一起使用,您仍然需要使用资产帮助程序。

{% javascripts 'bundles/webapp/app.js'
               'bundles/webapp/utils.js'
               filter='?closure' %}
    <script src="{{ asset(asset_url) }}" type="text/javascript"></script>
{% endjavascripts %}

它不会自动添加到asset_url(这是一件好事)。

于 2015-01-12T10:47:03.737 回答
3

一个简单快速的解决方法是这样的:

{% set asset_version = 'xyz' %}

{% javascripts 'bundles/webapp/js/app.js'
           'bundles/webapp/js/utils.js'
            filter='?closure' %}
    <script src="{{ asset_url }}?{{ asset_version }}"></script>
{% endjavascripts %}

但是您可能希望将逻辑移动到接收asset_url 作为参数的树枝扩展。

正常的过程是生成文件的哈希值,然后将其存储在用户缓存中。

然后,您可以在自定义命令中将所有散列与其当前散列进行比较,并将最新的散列或其他内容附加到文件名以强制更新缓存。

于 2013-07-22T20:33:28.000 回答
0

以下解决方案将附加时间戳而不是版本号。重要的部分是他的时间戳只会在你清除缓存时改变。

首先创建一个编译器通道:

<?php

namespace App\Bundle\DependencyInjection\Compiler;

use Symfony\Component\DependencyInjection\Compiler\CompilerPassInterface;
use Symfony\Component\DependencyInjection\ContainerBuilder;
use Symfony\Component\DependencyInjection\DefinitionDecorator;

/**
 * Created by PhpStorm.
 * User: hpenny
 * Date: 15/03/17
 * Time: 2:33 PM
 */
class AssetCompilerPass implements CompilerPassInterface
{
    /**
     * You can modify the container here before it is dumped to PHP code.
     *
     * @param ContainerBuilder $container
     */
    public function process(ContainerBuilder $container)
    {
        $container->removeDefinition('assets._version__default');
        $decorator = new DefinitionDecorator('assets.static_version_strategy');
        $decorator->replaceArgument(0, time());
        $container->setDefinition('assets._version__default', $decorator);
    }
}

然后将其添加到您的主包中:

namespace App\Bundle;

use App\Bundle\DependencyInjection\Compiler\AssetCompilerPass;
use Symfony\Component\DependencyInjection\ContainerBuilder;
use Symfony\Component\HttpKernel\Bundle\Bundle;

class AppBundle extends Bundle
{
    public function build(ContainerBuilder $container)
    {
        parent::build($container);

        $container->addCompilerPass(new AssetCompilerPass());
    }
}

这仅适用于默认资产包。如果您正在使用该功能,则需要遍历您设置的不同包定义。

你需要更换assets._version_<package name>而不是assets._version__default

于 2017-03-15T04:44:33.047 回答