machinateur / twig-block-validator
A twig implementation/integration of the Shopware 6 IntelliJ plugin's twig template-hash feature.
Installs: 36
Dependents: 0
Suggesters: 0
Security: 0
Stars: 7
Watchers: 1
Forks: 1
Open Issues: 2
pkg:composer/machinateur/twig-block-validator
Requires
- php: >=8.2.0
- composer-runtime-api: *
- composer/semver: ^3.4
- machinateur/twig-comment-lexer: ^1.0
- symfony/cache: ^7.2 | ^6.4 | ^5.4
- symfony/console: ^7.2 | ^6.4 | ^5.4
- symfony/debug-bundle: ^7.2 | ^6.4 | ^5.4
- symfony/dependency-injection: ^7.2 | ^6.4 | ^5.4
- symfony/event-dispatcher: ^7.2 | ^6.4 | ^5.4
- symfony/finder: ^7.2 | ^6.4 | ^5.4
- symfony/framework-bundle: ^7.2 | ^6.4 | ^5.4
- symfony/http-kernel: ^7.2 | ^6.4 | ^5.4
- symfony/monolog-bundle: ^3.6
- symfony/runtime: ^7.2 | ^6.4 | ^5.4
- symfony/twig-bundle: ^7.2 | ^6.4 | ^5.4
- symfony/yaml: ^7.2 | ^6.4 | ^5.4
- twig/twig: ^3.15
Requires (Dev)
- phpunit/phpunit: ^9.5
Suggests
- humbug/box: *
This package is auto-updated.
Last update: 2025-10-07 08:27:06 UTC
README
A twig block validator, inspired by the Shopware 6 IntelliJ plugin's twig block versioning. Pure PHP.
Inspiration
After visiting SCUC 2025 in Cologne, where it was mentioned and recommended multiple times, I held the idea that making twig-block validation possible via CLI would be a great help with CI for large shopware projects.
When dealing with many plugins and templates, it can get confusing quite fast. Let this tool help you with those twig blocks and validate if they're still up to date with their parent's content.
It also provides a way to automatically add and update the block version comment on your templates. Existing blocks and their comments may also be inspected without running any validation or pattern matching.
This project is still an early version!
Feature ideas, bug reports and contributions welcome. Thanks!
Installation
Requirements:
- PHP 8.2or above
- Twig version ^3.15
- At least Shopware 6.4(optional)
In your shopware project or plugin, run:
composer require --dev machinateur/twig-block-validator
Also make sure the bundle is available in the desired environments, usually dev and test. So in config/bundles.php:
// ... Machinateur\Shopware\TwigBlockValidator\TwigBlockValidatorBundle::class => ['dev' => true, 'test' => true],
Usage
Basic command line options include:
- -c= Add a path/namespace to process
This is the main option to define which template namespace/path to validate or annotate (depending on the command).
Multiple values may be given. The format is @DemoVendor:.../path/to/stuff.
- -t= Add additional template context
By default, the template context is inherited from the project's twig template paths. It is possible to define your own using this option.
Multiple values may be given. The format is @DemoVendor:.../path/to/stuff.
- -r= Require a version
When you completely omit the -r|--use-version option, to basically require "no version" in the block comments.
Without an assigned value, the option will use the installed shopware storefront version as requirement to check against.
And with a specific version given, that one will be used.
Note: If shopware is not available, even with -r specified, the fallback behaviour will be to require "no version".
Examples with Shopware
This tool may also be used without Shopware, it supports both.
Annotate a shopware plugin
Use the following commands to execute the validator for an existing Shopware Storefront project.
bin/console twig:block:validate -c '@DemoVendor:custom/static-plugins/demo-plugin/src/Resources/views'
The above command will load all available templates based the paths available to your project,
then validate custom/static-plugins/demo-plugin/src/Resources/views with @DemoVendor namespace.
But at first you will need to annotate the existing blocks of your plugin, which is possible using the command below.
bin/console twig:block:annotate -y -c '@DemoVendor:custom/static-plugins/demo-plugin/src/Resources/views' var/twig-block-validator/templates
In version 0.1.x available paths had to be specified explicitly. Click to see more...
# validate bin/console debug:twig --format json --no-debug \ | jq -r -c '.loader_paths | to_entries[] | . as { key: $ns, value: $paths } | $paths | map( $ns + ":" + . ) | .[]' \ | xargs printf ' -t "%s"' \ | xargs bin/console twig:block:validate -c '@DemoVendor:custom/static-plugins/demo-plugin/src/Resources/views'
The above command will load all available templates based on the debug:twig command (JSON output), then validate @DemoVendor:custom/static-plugins/demo-plugin/src/Resources/views.
# annotate bin/console debug:twig --format json --no-debug \ | jq -r -c '.loader_paths | to_entries[] | . as { key: $ns, value: $paths } | $paths | map( $ns + ":" + . ) | .[]' \ | xargs printf ' -t "%s"' \ | xargs bin/console twig:block:annotate -y -c '@DemoVendor:custom/static-plugins/demo-plugin/src/Resources/views' var/twig-block-validator/templates
The above command will load all available templates based on the debug:twig command (JSON output), then annotate @DemoVendor:custom/static-plugins/demo-plugin/src/Resources/views
and output the changed templates to var/twig-block-validator/templates. Omit the path, to do update annotations in-place.
Next, those can be used to patch or compare the template source code, which is typically tracked via a VCS.
Validate a templates dir
To validate against the dev-dependency shopware/storefront:^6.4 installed at vendor/shopware/storefront/Resources/views,
expecting version 6.6 in the comments of tests/nested/ (__main__ as default namespace), run:
$ time bin/shopware twig:block:validate -t '@Storefront:vendor/shopware/storefront/Resources/views' -c tests/nested/ -r 6.7
 ! [NOTE] Adding namespace "__main__"...                                                                                
 ! [NOTE] Adding namespace "Storefront"...                                                                              
 ! [NOTE] Loading namespace "__main__".                                                                                 
 1/1 [▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓] 100% < 1 sec
 ! [NOTE] Collected 1 blocks across 1 templates.                                                                        
 ------------------------------ --------------------------------------- --------------------------- ------------------------------------------------------------------ ---------- ---------- 
  template                       parent template                         block                       hash                                                               version    mismatch  
 ------------------------------ --------------------------------------- --------------------------- ------------------------------------------------------------------ ---------- ---------- 
  @__main__/template.html.twig   @Storefront/storefront/base.html.twig   base_body_skip_to_content   c1954b12f0c43a0244c3d781256b6aa676b5699a9700c10983a468a68a0e5eb1   v6.6.6.0   [x]       
                                                                                                     4820b0cd1d32b5079aa8f3e988bd756bc0c52ea20a694353370b39a8ded2d5ac   6.7                  
 ------------------------------ --------------------------------------- --------------------------- ------------------------------------------------------------------ ---------- ---------- 
> 0.18s user 0.08s system 59% cpu 0.433 total (tested on Mac M1)
Here's the example template that would produce the above errors:
{% sw_extends "@Storefront/storefront/base.html.twig" %}
{# shopware-block: c1954b12f0c43a0244c3d781256b6aa676b5699a9700c10983a468a68a0e5eb1@v6.6.6.0 #}
{% block base_body_skip_to_content %}
    My overwrite content
{% endblock %}
I just generated a random SHA-265 for this test.
It's also possible to use twig-block here,
but since this was inspired by Shopware's PhpStorm plugin, shopware-block is also supported.
Which one is used is determined by shopware/storefront being installed (this is checked using composer).
In general, the version part of the comment is recommended, but since this tool is not strictly limited to working with
shopware, it is not enforced. The provided version (-r flag) will be the default version, if none is set for a block.
If not given, it will try to use the current shopware version, if available from the kernel.
Annotate @Storefront itself
The following command will go through the templates of shopware/storefront,
put the annotation comment for those blocks, that extend another template (and therefor have a parent),
and finally write the changed templates to ./var/cache/twig-block-validator/views.
$ bin/shopware twig:block:annotate -c '@Storefront:vendor/shopware/storefront/Resources/views' \
  -r 6.6.10.3 ./var/cache/twig-block-validator/views
 ! [NOTE] Adding namespace "Storefront"...                                                                              
 ! [NOTE] Loading namespace "Storefront".                                                                               
 324/324 [▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓] 100%  1 sec
 ! [NOTE] Collected 2390 blocks across 324 templates.                                                                   
 ! [NOTE] Loading namespace "Storefront".                                                                               
 324/324 [▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓] 100% < 1 sec
 ! [NOTE] Found 0 comments in 324 templates.                                                                            
This will also work, and annotate the storefront amongst its own templates. Just an experiment I tried.
How to validate
The validation can be performed by calling the CLI command twig:block:validate.
Here's it's synopsis:
$ bin/console twig:block:validate -h
Description:
  Validate block versions in twig templates
Usage:
  twig:block:validate [options]
Options:
  -t, --template=TEMPLATE          Twig template path to load (multiple values allowed)
  -c, --validate=VALIDATE          Twig template path to validate (multiple values allowed)
  -r, --use-version[=USE-VERSION]  The version number to require [default: false]
  -h, --help                       Display help for the given command. When no command is given display help for the list command
      --silent                     Do not output any message
  -q, --quiet                      Only errors are displayed. All other output is suppressed
  -V, --version                    Display this application version
      --ansi|--no-ansi             Force (or disable --no-ansi) ANSI output
  -n, --no-interaction             Do not ask any interactive question
  -e, --env=ENV                    The Environment name. [default: "dev"]
      --no-debug                   Switch off debug mode.
      --profile                    Enables profiling (requires debug).
  -v|vv|vvv, --verbose             Increase the verbosity of messages: 1 for normal output, 2 for more verbose output and 3 for debug
The exit-code of the validate command will be 0 when no violations were found and no twig errors occured, else 1.
This allows for better CI integration and other automated checks.
How to annotate
The annotation can be performed by calling the CLI command twig:block:annotate.
Here's it's synopsis:
$ bin/console twig:block:annotate -h
Description:
  Annotate block versions in twig templates
Usage:
  twig:block:annotate [options]
Options:
  -t, --template=TEMPLATE          Twig template path to load (multiple values allowed)
  -c, --validate=VALIDATE          Twig template path to validate (multiple values allowed)
  -r, --use-version[=USE-VERSION]  The version number to require [default: false]
  -y, --yes                        Silently approve annotating template in-place
  -h, --help                       Display help for the given command. When no command is given display help for the list command
      --silent                     Do not output any message
  -q, --quiet                      Only errors are displayed. All other output is suppressed
  -V, --version                    Display this application version
      --ansi|--no-ansi             Force (or disable --no-ansi) ANSI output
  -n, --no-interaction             Do not ask any interactive question
  -e, --env=ENV                    The Environment name. [default: "dev"]
      --no-debug                   Switch off debug mode.
      --profile                    Enables profiling (requires debug).
  -v|vv|vvv, --verbose             Increase the verbosity of messages: 1 for normal output, 2 for more verbose output and 3 for debug
Caution! Always back up your templates and use a VCS. A bug or user error can cause permanent loss of data!
How to inspect
$ bin/console twig:block:inspect -h 
Description:
  Inspect block comments in twig templates
Usage:
  twig:block:inspect [options]
Options:
  -t, --template=TEMPLATE          Twig template path to load (multiple values allowed)
  -c, --validate=VALIDATE          Twig template path to validate (multiple values allowed)
  -r, --use-version[=USE-VERSION]  The version number to require [default: false]
  -a, --check-all                  Check all available template paths
  -s, --strict-mode                Match comments in strict mode
  -h, --help                       Display help for the given command. When no command is given display help for the list command
      --silent                     Do not output any message
  -q, --quiet                      Only errors are displayed. All other output is suppressed
  -V, --version                    Display this application version
      --ansi|--no-ansi             Force (or disable --no-ansi) ANSI output
  -n, --no-interaction             Do not ask any interactive question
  -e, --env=ENV                    The Environment name. [default: "dev"]
      --no-debug                   Switch off debug mode.
      --profile                    Enables profiling (requires debug).
  -v|vv|vvv, --verbose             Increase the verbosity of messages: 1 for normal output, 2 for more verbose output and 3 for debug
Which CLI to use
There are a total of four different CLIs available:
- bin/console: A symfony console application.- Runs a symfony kernel with symfony framework and debug commands (in devortest).
- Only supports the twig extensions that are available in the symfony context.
 
- Runs a symfony kernel with symfony framework and debug commands (in 
- bin/twig-block-validate: A symfony command application.- Runs only the twig:block:validatecommand as standalone application.
- Only supports the twig extensions that are available in the symfony context.
 
- Runs only the 
- bin/twig-block-annotate: A symfony command application.- Runs only the twig:block:annotatecommand as standalone application.
- Only supports the twig extensions that are available in the symfony context.
 
- Runs only the 
- bin/twig-block-inspect: A symfony command application.- Runs only the twig:block:inspectcommand as standalone application.
- Only supports the twig extensions that are available in the symfony context.
 
- Runs only the 
These can be copied to the bin/ directory of your project (if not already present).
Shopware test container
I've put in a small docker-compose setup, at tests/shopware/ which may be used to test against a real shopware instance.
cd tests/shopware/ bash ./init.sh docker compose exec -it shop bash
Standalone
Note: The phar is still experimental, and does not support integrated use with shopware projects (yet).
The tool can be used as standalone, phar or source, for example in CI pipelines:
git clone https://github.com/machinateur/twig-block-validator.git
PHP is required. The source archive will need composer to be available
and may also be used instead of git clone, when not available in your environment.
Find the source archive and prebuilt phar attached to the latest release.
Build the phar
This instruction requires box to be installed globally.
# navigate to project cd twig-block-validator # set up env export APP_ENV=prod export APP_DEBUG=0 # prepare cache bin/console cache:clear rm -rf var/cache bin/console cache:warmup # check if everything is fine bin/box -V # remove old logs rm var/log/*.log || mkdir -p var/log # compile the phar box compile -vvv # run the phar ./build/twig-block-validator.phar
License
It's MIT.