diff --git a/.gitattributes b/.gitattributes
index fe3a31b..2c6f6d9 100644
--- a/.gitattributes
+++ b/.gitattributes
@@ -1,6 +1,5 @@
# Remove PHP source from zip downloads,
-# only keeping twigexpress.phar, doc & test folders
-/dev export-ignore
+# only keeping twigexpress.phar|json and demo folder
/doc export-ignore
/src export-ignore
/test export-ignore
diff --git a/CHANGELOG.md b/CHANGELOG.md
index 7c5ec38..1893d9d 100644
--- a/CHANGELOG.md
+++ b/CHANGELOG.md
@@ -1,5 +1,10 @@
# Changelog
+## 2.1.3
+
+- Fix main download by removing test-specific config in `twigexpress.json`.
+- Update Twig lib to 1.35.3.
+
## 2.1.2
- Remove separate 'download' branch, in favour of 'master'.
diff --git a/LICENSE b/LICENSE
index 78f7ce1..1264cb4 100644
--- a/LICENSE
+++ b/LICENSE
@@ -1,6 +1,6 @@
The MIT License (MIT)
-Copyright (c) 2016-2017 Kaliop
+Copyright (c) 2016-2018 Kaliop
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
diff --git a/README.md b/README.md
index 8f89c59..23d49b7 100644
--- a/README.md
+++ b/README.md
@@ -1,50 +1,56 @@
TwigExpress
===========
-TwigExpress packages the [Twig templating engine][TWIG_HOME], and a few other tools, in a single file. Our goal is to make it easy to get started with Twig if you’re a designer or a front-end developer, without having to install a big PHP framework or a CMS.
+TwigExpress packages the [Twig templating engine][TWIG_HOME], and a few other tools, in a single file.
+
+Our goal is to make it easy to get started with Twig if you’re a designer or a front-end developer, without having to install a heavy PHP framework or CMS.
Main features:
-- Browse and serve files
+- Browse and serve files (⚠ only for development!)
- Render Twig templates
- Error pages with code excerpt, source view
- And a few extra tools for prototyping (dummy text, Markdown to HTML…):
-**⚠ Do not use in production ⚠**
-We recommend using TwigExpress for:
-
-1. discovering (or play with) Twig;
-2. building HTML prototypes with Twig, CSS and JS.
-
Table of contents
-----------------
-1. [Installation](#installation)
+1. [Running TwigExpress](#running-twigexpress)
2. [Adding content](#adding-content)
3. [Writing Twig templates](#writing-twig-templates)
↪ [Getting Started with Twig][DOC_INTRO]
↪ [TwigExpress-specific features][DOC_EXTRAS]
4. [Configuration](#configuration)
- ↪ [TwigExpress configuration reference][DOC_CONFIG]
+ ↪ [TwigExpress configuration reference][DOC_CONFIG]
↪ [Using TwigExpress with Apache][DOC_APACHE]
5. [Library and license info](#library-and-license-info)
-Installation
-------------
+Running TwigExpress
+-------------------
### Requirements
-- PHP 5.4+ available on the command line.
- On macOS, you should have PHP installed already.
- On Windows, one easy way to install PHP (and other tools) is [XAMPP](https://www.apachefriends.org/download.html).
+PHP 5.4+ available on the command line.
+
+- On macOS, you should have PHP installed already.
+- On Windows, one easy way to install PHP (and other tools) is [XAMPP](https://www.apachefriends.org/download.html).
-### Installation and usage
+### Download TwigExpress
-1. [Download a ZIP of this repo][DOWNLOAD] and unzip it.
-2. Open a Terminal or Command Prompt *in that folder*, and run:
- `php --server localhost:8000 twigexpress.phar`.
-3. Load `http://localhost:8000/` in a web browser to browse files.
- Any file ending in `.twig` will be interpreted as a Twig template.
+- [Download a ZIP of this repo][DOWNLOAD] and unzip.
+- Recommended: rename `twig-express-master` to your project’s name.
+
+### Usage
+
+Open your project folder in a Terminal or Command Prompt and run this command:
+
+```sh
+php --server localhost:8000 twigexpress.phar
+```
+
+Now load [http://localhost:8000/](http://localhost:8000/) in a web browser to browse your files.
+
+Any file ending in `.twig` will be interpreted as a Twig template. You can check out the example pages in the `demo` directory.
Adding content
--------------
@@ -60,7 +66,7 @@ myproject/
twigexpress.phar
```
-If you don’t care about the test and info files, you can remove them, keeping only this:
+If you don’t care about the demo and read-me, you can keep these only:
```
myproject/
@@ -68,14 +74,26 @@ myproject/
twigexpress.phar
```
-You can add your own content anywhere. By content we mean: Twig templates (which must have the `.twig` extension), CSS, scripts, images, etc. File names and URLs will match, but you should omit the `.twig` extension in URLs. We’re also using `index.twig` and `index.html` files as directory index, if they exist. For example:
-
-- `myproject/index.twig` → can be accessed at `http://localhost:8000/`
-- `myproject/some/page.twig` → `http://localhost:8000/some/page`
-- `myproject/css/styles.css` → `http://localhost:8000/css/styles.css`
-
-There is no enforced convention for where to place stylesheets, JavaScript, etc. Feel free to organize your static assets and templates however you want.
-
+You can add your own content anywhere. This content can be Twig templates (which must have the `.twig` extension), CSS, scripts, images, etc.
+
+
Example file path | +Corresponding URL | +
---|---|
myproject/index.twig |
+ http://localhost:8000/ |
+
myproject/some/page.twig |
+ http://localhost:8000/some/page |
+
myproject/css/styles.css |
+ http://localhost:8000/css/styles.css |
+
_context
Key | Value |
---|---|
{{ key }} | {{ val|json_encode|replace({'\\/':'/'}) }} |
{% verbatim %}{% include 'demo/test.json.twig' with {hello:'goodbye'} %}{% endverbatim %}-
Output:
-{% include 'demo/test.json.twig' with {hello:'goodbye'} %}-
Or load this template directly: demo/test.json
- -{% verbatim %}{{ source('LICENSE', ignore_missing=true) }}{% endverbatim %}-
Output:
-{{ source('LICENSE', ignore_missing=true) }}+
It’s split in two files:
+ + +_context
Key | Value |
---|---|
{{ key }} | {{ val|json_encode|replace({'\\/':'/'}) }} |
Including a template (interpreted as Twig):
+{% verbatim %}{% include 'demo/test.json.twig' with {hello:'goodbye'} %}{% endverbatim %}+
{% include 'demo/test.json.twig' with {hello:'goodbye'} %}+ +
Including a file (not interpreted):
+{% verbatim %}{{ source('demo/test.json.twig') }}{% endverbatim %}+
{{ source('demo/test.json.twig') }}{% endblock %} diff --git a/demo/test.json.twig b/demo/test.json.twig index 19365e5..304ecd1 100644 --- a/demo/test.json.twig +++ b/demo/test.json.twig @@ -1,5 +1,4 @@ -{% set hello = hello|default('world') %} { - "hello": "{{ hello }}", + "hello": "{{ hello|default('world') }}", "list": [{% for value in 1..20 %}"value = {{ value }}"{% if not loop.last %}, {% endif %}{% endfor %}] } diff --git a/dev/package.json b/dev/package.json deleted file mode 100644 index 37fdca8..0000000 --- a/dev/package.json +++ /dev/null @@ -1,13 +0,0 @@ -{ - "scripts": { - "build": "npm run js && npm run css && npm run phar", - "js": "uglifyjs js/*.js > ../src/tpl/layout.min.js", - "css": "cat css/*.css | csso > ../src/tpl/layout.min.css", - "phar": "php makephar.php", - "serve": "cd .. && php --server localhost:8088 twigexpress.phar" - }, - "dependencies": { - "csso-cli": "^1.0.0", - "uglify-js": "^3.0.13" - } -} diff --git a/dev/README-DEV.md b/src/README-DEV.md similarity index 84% rename from dev/README-DEV.md rename to src/README-DEV.md index 5661a53..6e8eb0e 100644 --- a/dev/README-DEV.md +++ b/src/README-DEV.md @@ -4,14 +4,14 @@ Building TwigExpress ## Minify CSS and JS ``` -$ cd dev +# From the 'src' directory $ npm install -$ npm run build +$ npm start ``` ## Making the PHAR -The command should already be included in `npm run build`, but you can run manually: +The command should already be included in `npm start`, but you can run manually: ``` php makephar.php diff --git a/dev/css/1-core.css b/src/css/1-core.css similarity index 94% rename from dev/css/1-core.css rename to src/css/1-core.css index 0f87d81..ae9daa0 100644 --- a/dev/css/1-core.css +++ b/src/css/1-core.css @@ -69,17 +69,15 @@ } ::selection { + text-decoration: none; color: #fff; - background-color: #579; - color: var(--white, #fff); - background-color: var(--dark-red, #579); + background-color: #d85b5c; } ::-moz-selection { + text-decoration: none; color: #fff; - background-color: #579; - color: var(--white, #fff); - background-color: var(--dark-red, #579); + background-color: #d85b5c; } body { diff --git a/dev/css/2-header.css b/src/css/2-header.css similarity index 100% rename from dev/css/2-header.css rename to src/css/2-header.css diff --git a/dev/css/3-content.css b/src/css/3-content.css similarity index 100% rename from dev/css/3-content.css rename to src/css/3-content.css diff --git a/dev/css/9-code.css b/src/css/9-code.css similarity index 100% rename from dev/css/9-code.css rename to src/css/9-code.css diff --git a/dev/css/9-links.css b/src/css/9-links.css similarity index 100% rename from dev/css/9-links.css rename to src/css/9-links.css diff --git a/dev/js/1-highlight.js b/src/js/1-highlight.js similarity index 100% rename from dev/js/1-highlight.js rename to src/js/1-highlight.js diff --git a/dev/js/9-headings.js b/src/js/9-headings.js similarity index 100% rename from dev/js/9-headings.js rename to src/js/9-headings.js diff --git a/dev/js/9-highlight.js b/src/js/9-highlight.js similarity index 100% rename from dev/js/9-highlight.js rename to src/js/9-highlight.js diff --git a/src/lib/Parsedown/README.md b/src/lib/Parsedown/README.md deleted file mode 100644 index 6f9f649..0000000 --- a/src/lib/Parsedown/README.md +++ /dev/null @@ -1,57 +0,0 @@ -## Parsedown - -[![Build Status](https://img.shields.io/travis/erusev/parsedown/master.svg?style=flat-square)](https://travis-ci.org/erusev/parsedown) - - -Better Markdown Parser in PHP - -[Demo](http://parsedown.org/demo) | -[Benchmarks](http://parsedown.org/speed) | -[Tests](http://parsedown.org/tests/) | -[Documentation](https://github.com/erusev/parsedown/wiki/) - -### Features - -* Super Fast -* [GitHub flavored](https://help.github.com/articles/github-flavored-markdown) -* Extensible -* Tested in 5.3 to 5.6 -* [Markdown Extra extension](https://github.com/erusev/parsedown-extra) - -### Installation - -Include `Parsedown.php` or install [the composer package](https://packagist.org/packages/erusev/parsedown). - -### Example - -``` php -$Parsedown = new Parsedown(); - -echo $Parsedown->text('Hello _Parsedown_!'); # prints:
Hello Parsedown!
-``` - -More examples in [the wiki](https://github.com/erusev/parsedown/wiki/) and in [this video tutorial](http://youtu.be/wYZBY8DEikI). - -### Questions - -**How does Parsedown work?** - -It tries to read Markdown like a human. First, it looks at the lines. It’s interested in how the lines start. This helps it recognise blocks. It knows, for example, that if a line start with a `-` then it perhaps belong to a list. Once it recognises the blocks, it continues to the content. As it reads, it watches out for special characters. This helps it recognise inline elements (or inlines). - -We call this approach "line based". We believe that Parsedown is the first Markdown parser to use it. Since the release of Parsedown, other developers have used the same approach to develop other Markdown parsers in PHP and in other languages. - -**Is it compliant with CommonMark?** - -It passes most of the CommonMark tests. Most of the tests that don't pass deal with cases that are quite uncommon. Still, as CommonMark matures, compliance should improve. - -**Who uses it?** - -[phpDocumentor](http://www.phpdoc.org/), [October CMS](http://octobercms.com/), [Bolt CMS](http://bolt.cm/), [Kirby CMS](http://getkirby.com/), [Grav CMS](http://getgrav.org/), [Statamic CMS](http://www.statamic.com/), [RaspberryPi.org](http://www.raspberrypi.org/) and [more](https://www.versioneye.com/php/erusev:parsedown/references). - -**How can I help?** - -Use it, star it, share it and if you feel generous, [donate](https://www.paypal.com/cgi-bin/webscr?cmd=_s-xclick&hosted_button_id=528P3NZQMP8N2). - ---- - -You might also like [Caret](http://caret.io) - our Markdown editor for the desktop. diff --git a/dev/makephar.php b/src/makephar.php similarity index 89% rename from dev/makephar.php rename to src/makephar.php index 86e4432..d249e34 100644 --- a/dev/makephar.php +++ b/src/makephar.php @@ -1,9 +1,7 @@ php/tpl/layout.min.js", + "css": "cat css/*.css | csso > php/tpl/layout.min.css", + "phar": "php makephar.php", + "serve": "cd .. && php --server localhost:8088 twigexpress.phar" + }, + "dependencies": { + "csso-cli": "^1.1.0", + "uglify-js": "^3.3.23" + } +} diff --git a/src/Controller.php b/src/php/Controller.php similarity index 100% rename from src/Controller.php rename to src/php/Controller.php diff --git a/src/TwigEnv.php b/src/php/TwigEnv.php similarity index 100% rename from src/TwigEnv.php rename to src/php/TwigEnv.php diff --git a/src/Utils.php b/src/php/Utils.php similarity index 100% rename from src/Utils.php rename to src/php/Utils.php diff --git a/src/index.php b/src/php/index.php similarity index 76% rename from src/index.php rename to src/php/index.php index a95d155..27420b6 100644 --- a/src/index.php +++ b/src/php/index.php @@ -2,9 +2,13 @@ if (PHP_SAPI === 'cli') { $msg = <<Hello Parsedown!
+``` + +More examples in [the wiki](https://github.com/erusev/parsedown/wiki/) and in [this video tutorial](http://youtu.be/wYZBY8DEikI). + +### Security + +Parsedown is capable of escaping user-input within the HTML that it generates. Additionally Parsedown will apply sanitisation to additional scripting vectors (such as scripting link destinations) that are introduced by the markdown syntax itself. + +To tell Parsedown that it is processing untrusted user-input, use the following: +```php +$parsedown = new Parsedown; +$parsedown->setSafeMode(true); +``` + +If instead, you wish to allow HTML within untrusted user-input, but still want output to be free from XSS it is recommended that you make use of a HTML sanitiser that allows HTML tags to be whitelisted, like [HTML Purifier](http://htmlpurifier.org/). + +In both cases you should strongly consider employing defence-in-depth measures, like [deploying a Content-Security-Policy](https://scotthelme.co.uk/content-security-policy-an-introduction/) (a browser security feature) so that your page is likely to be safe even if an attacker finds a vulnerability in one of the first lines of defence above. + +#### Security of Parsedown Extensions + +Safe mode does not necessarily yield safe results when using extensions to Parsedown. Extensions should be evaluated on their own to determine their specific safety against XSS. + +### Escaping HTML +> ⚠️ **WARNING:** This method isn't safe from XSS! + +If you wish to escape HTML **in trusted input**, you can use the following: +```php +$parsedown = new Parsedown; +$parsedown->setMarkupEscaped(true); +``` + +Beware that this still allows users to insert unsafe scripting vectors, such as links like `[xss](javascript:alert%281%29)`. + +### Questions + +**How does Parsedown work?** + +It tries to read Markdown like a human. First, it looks at the lines. It’s interested in how the lines start. This helps it recognise blocks. It knows, for example, that if a line starts with a `-` then perhaps it belongs to a list. Once it recognises the blocks, it continues to the content. As it reads, it watches out for special characters. This helps it recognise inline elements (or inlines). + +We call this approach "line based". We believe that Parsedown is the first Markdown parser to use it. Since the release of Parsedown, other developers have used the same approach to develop other Markdown parsers in PHP and in other languages. + +**Is it compliant with CommonMark?** + +It passes most of the CommonMark tests. Most of the tests that don't pass deal with cases that are quite uncommon. Still, as CommonMark matures, compliance should improve. + +**Who uses it?** + +[Laravel Framework](https://laravel.com/), [Bolt CMS](http://bolt.cm/), [Grav CMS](http://getgrav.org/), [Herbie CMS](http://www.getherbie.org/), [Kirby CMS](http://getkirby.com/), [October CMS](http://octobercms.com/), [Pico CMS](http://picocms.org), [Statamic CMS](http://www.statamic.com/), [phpDocumentor](http://www.phpdoc.org/), [RaspberryPi.org](http://www.raspberrypi.org/), [Symfony demo](https://github.com/symfony/symfony-demo) and [more](https://packagist.org/packages/erusev/parsedown/dependents). + +**How can I help?** + +Use it, star it, share it and if you feel generous, [donate](https://www.paypal.com/cgi-bin/webscr?cmd=_s-xclick&hosted_button_id=528P3NZQMP8N2). diff --git a/src/lib/Twig/Autoloader.php b/src/php/lib/Twig/Autoloader.php similarity index 100% rename from src/lib/Twig/Autoloader.php rename to src/php/lib/Twig/Autoloader.php diff --git a/src/lib/Twig/BaseNodeVisitor.php b/src/php/lib/Twig/BaseNodeVisitor.php similarity index 100% rename from src/lib/Twig/BaseNodeVisitor.php rename to src/php/lib/Twig/BaseNodeVisitor.php diff --git a/src/php/lib/Twig/CHANGELOG b/src/php/lib/Twig/CHANGELOG new file mode 100644 index 0000000..e270288 --- /dev/null +++ b/src/php/lib/Twig/CHANGELOG @@ -0,0 +1,1003 @@ +* 1.35.3 (2018-03-20) + + * fixed block names unicity + * fixed counting children of SimpleXMLElement objects + * added missing else clause to avoid infinite loops + * fixed .. (range operator) in sandbox policy + +* 1.35.2 (2018-03-03) + + * fixed a regression in the way the profiler is registered in templates + +* 1.35.1 (2018-03-02) + + * added an exception when using "===" instead of "same as" + * fixed possible array to string conversion concealing actual error + * made variable names deterministic in compiled templates + * fixed length filter when passing an instance of IteratorAggregate + * fixed Environment::resolveTemplate to accept instances of TemplateWrapper + +* 1.35.0 (2017-09-27) + + * added Twig_Profiler_Profile::reset() + * fixed use TokenParser to return an empty Node + * added RuntimeExtensionInterface + * added circular reference detection when loading templates + +* 1.34.4 (2017-07-04) + + * added support for runtime loaders in IntegrationTestCase + * fixed deprecation when using Twig_Profiler_Dumper_Html + +* 1.34.3 (2017-06-07) + + * fixed namespaces introduction + +* 1.34.2 (2017-06-05) + + * fixed namespaces introduction + +* 1.34.1 (2017-06-05) + + * fixed namespaces introduction + +* 1.34.0 (2017-06-05) + + * added support for PHPUnit 6 when testing extensions + * fixed PHP 7.2 compatibility + * fixed template name generation in Twig_Environment::createTemplate() + * removed final tag on Twig_TokenParser_Include + * added namespaced aliases for all (non-deprecated) classes and interfaces + * dropped HHVM support + * dropped PHP 5.2 support + +* 1.33.2 (2017-04-20) + + * fixed edge case in the method cache for Twig attributes + +* 1.33.1 (2017-04-18) + + * fixed the empty() test + +* 1.33.0 (2017-03-22) + + * fixed a race condition handling when writing cache files + * "length" filter now returns string length when applied to an object that does + not implement \Countable but provides __toString() + * "empty" test will now consider the return value of the __toString() method for + objects implement __toString() but not \Countable + * fixed JS escaping for unicode characters with higher code points + +* 1.32.0 (2017-02-26) + + * fixed deprecation notice in Twig_Util_DeprecationCollector + * added a PSR-11 compatible runtime loader + * added `side` argument to `trim` to allow left or right trimming only. + +* 1.31.0 (2017-01-11) + + * added Twig_NodeCaptureInterface for nodes that capture all output + * fixed marking the environment as initialized too early + * fixed C89 compat for the C extension + * turned fatal error into exception when a previously generated cache is corrupted + * fixed offline cache warm-ups for embedded templates + +* 1.30.0 (2016-12-23) + + * added Twig_FactoryRuntimeLoader + * deprecated function/test/filter/tag overriding + * deprecated the "disable_c_ext" attribute on Twig_Node_Expression_GetAttr + +* 1.29.0 (2016-12-13) + + * fixed sandbox being left enabled if an exception is thrown while rendering + * marked some classes as being final (via @final) + * made Twig_Error report real source path when possible + * added support for {{ _self }} to provide an upgrade path from 1.x to 2.0 (replaces {{ _self.templateName }}) + * deprecated silent display of undefined blocks + * deprecated support for mbstring.func_overload != 0 + +* 1.28.2 (2016-11-23) + + * fixed precedence between getFoo() and isFoo() in Twig_Template::getAttribute() + * improved a deprecation message + +* 1.28.1 (2016-11-18) + + * fixed block() function when used with a template argument + +* 1.28.0 (2016-11-17) + + * added support for the PHP 7 null coalescing operator for the ?? Twig implementation + * exposed a way to access template data and methods in a portable way + * changed context access to use the PHP 7 null coalescing operator when available + * added the "with" tag + * added support for a custom template on the block() function + * added "is defined" support for block() and constant() + * optimized the way attributes are fetched + +* 1.27.0 (2016-10-25) + + * deprecated Twig_Parser::getEnvironment() + * deprecated Twig_Parser::addHandler() and Twig_Parser::addNodeVisitor() + * deprecated Twig_Compiler::addIndentation() + * fixed regression when registering two extensions having the same class name + * deprecated Twig_LoaderInterface::getSource() (implement Twig_SourceContextLoaderInterface instead) + * fixed the filesystem loader with relative paths + * deprecated Twig_Node::getLine() in favor of Twig_Node::getTemplateLine() + * deprecated Twig_Template::getSource() in favor of Twig_Template::getSourceContext() + * deprecated Twig_Node::getFilename() in favor of Twig_Node::getTemplateName() + * deprecated the "filename" escaping strategy (use "name" instead) + * added Twig_Source to hold information about the original template + * deprecated Twig_Error::getTemplateFile() and Twig_Error::setTemplateFile() in favor of Twig_Error::getTemplateName() and Twig_Error::setTemplateName() + * deprecated Parser::getFilename() + * fixed template paths when a template name contains a protocol like vfs:// + * improved debugging with Twig_Sandbox_SecurityError exceptions for disallowed methods and properties + +* 1.26.1 (2016-10-05) + + * removed template source code from generated template classes when debug is disabled + * fixed default implementation of Twig_Template::getDebugInfo() for better BC + * fixed regression on static calls for functions/filters/tests + +* 1.26.0 (2016-10-02) + + * added template cache invalidation based on more environment options + * added a missing deprecation notice + * fixed template paths when a template is stored in a PHAR file + * allowed filters/functions/tests implementation to use a different class than the extension they belong to + * deprecated Twig_ExtensionInterface::getName() + +* 1.25.0 (2016-09-21) + + * changed the way we store template source in template classes + * removed usage of realpath in cache keys + * fixed Twig cache sharing when used with different versions of PHP + * removed embed parent workaround for simple use cases + * deprecated the ability to store non Node instances in Node::$nodes + * deprecated Twig_Environment::getLexer(), Twig_Environment::getParser(), Twig_Environment::getCompiler() + * deprecated Twig_Compiler::getFilename() + +* 1.24.2 (2016-09-01) + + * fixed static callables + * fixed a potential PHP warning when loading the cache + * fixed a case where the autoescaping does not work as expected + +* 1.24.1 (2016-05-30) + + * fixed reserved keywords (forbids true, false, null and none keywords for variables names) + * fixed support for PHP7 (Throwable support) + * marked the following methods as being internals on Twig_Environment: + getFunctions(), getFilters(), getTests(), getFunction(), getFilter(), getTest(), + getTokenParsers(), getTags(), getNodeVisitors(), getUnaryOperators(), getBinaryOperators(), + getFunctions(), getFilters(), getGlobals(), initGlobals(), initExtensions(), and initExtension() + +* 1.24.0 (2016-01-25) + + * adding support for the ?? operator + * fixed the defined test when used on a constant, a map, or a sequence + * undeprecated _self (should only be used to get the template name, not the template instance) + * fixed parsing on PHP7 + +* 1.23.3 (2016-01-11) + + * fixed typo + +* 1.23.2 (2015-01-11) + + * added versions in deprecated messages + * made file cache tolerant for trailing (back)slashes on directory configuration + * deprecated unused Twig_Node_Expression_ExtensionReference class + +* 1.23.1 (2015-11-05) + + * fixed some exception messages which triggered PHP warnings + * fixed BC on Twig_Test_NodeTestCase + +* 1.23.0 (2015-10-29) + + * deprecated the possibility to override an extension by registering another one with the same name + * deprecated Twig_ExtensionInterface::getGlobals() (added Twig_Extension_GlobalsInterface for BC) + * deprecated Twig_ExtensionInterface::initRuntime() (added Twig_Extension_InitRuntimeInterface for BC) + * deprecated Twig_Environment::computeAlternatives() + +* 1.22.3 (2015-10-13) + + * fixed regression when using null as a cache strategy + * improved performance when checking template freshness + * fixed warnings when loaded templates do not exist + * fixed template class name generation to prevent possible collisions + * fixed logic for custom escapers to call them even on integers and null values + * changed template cache names to take into account the Twig C extension + +* 1.22.2 (2015-09-22) + + * fixed a race condition in template loading + +* 1.22.1 (2015-09-15) + + * fixed regression in template_from_string + +* 1.22.0 (2015-09-13) + + * made Twig_Test_IntegrationTestCase more flexible + * added an option to force PHP bytecode invalidation when writing a compiled template into the cache + * fixed the profiler duration for the root node + * changed template cache names to take into account enabled extensions + * deprecated Twig_Environment::clearCacheFiles(), Twig_Environment::getCacheFilename(), + Twig_Environment::writeCacheFile(), and Twig_Environment::getTemplateClassPrefix() + * added a way to override the filesystem template cache system + * added a way to get the original template source from Twig_Template + +* 1.21.2 (2015-09-09) + + * fixed variable names for the deprecation triggering code + * fixed escaping strategy detection based on filename + * added Traversable support for replace, merge, and sort + * deprecated support for character by character replacement for the "replace" filter + +* 1.21.1 (2015-08-26) + + * fixed regression when using the deprecated Twig_Test_* classes + +* 1.21.0 (2015-08-24) + + * added deprecation notices for deprecated features + * added a deprecation "framework" for filters/functions/tests and test fixtures + +* 1.20.0 (2015-08-12) + + * forbid access to the Twig environment from templates and internal parts of Twig_Template + * fixed limited RCEs when in sandbox mode + * deprecated Twig_Template::getEnvironment() + * deprecated the _self variable for usage outside of the from and import tags + * added Twig_BaseNodeVisitor to ease the compatibility of node visitors + between 1.x and 2.x + +* 1.19.0 (2015-07-31) + + * fixed wrong error message when including an undefined template in a child template + * added support for variadic filters, functions, and tests + * added support for extra positional arguments in macros + * added ignore_missing flag to the source function + * fixed batch filter with zero items + * deprecated Twig_Environment::clearTemplateCache() + * fixed sandbox disabling when using the include function + +* 1.18.2 (2015-06-06) + + * fixed template/line guessing in exceptions for nested templates + * optimized the number of inodes and the size of realpath cache when using the cache + +* 1.18.1 (2015-04-19) + + * fixed memory leaks in the C extension + * deprecated Twig_Loader_String + * fixed the slice filter when used with a SimpleXMLElement object + * fixed filesystem loader when trying to load non-files (like directories) + +* 1.18.0 (2015-01-25) + + * fixed some error messages where the line was wrong (unknown variables or argument names) + * added a new way to customize the main Module node (via empty nodes) + * added Twig_Environment::createTemplate() to create a template from a string + * added a profiler + * fixed filesystem loader cache when different file paths are used for the same template + +* 1.17.0 (2015-01-14) + + * added a 'filename' autoescaping strategy, which dynamically chooses the + autoescaping strategy for a template based on template file extension. + +* 1.16.3 (2014-12-25) + + * fixed regression for dynamic parent templates + * fixed cache management with statcache + * fixed a regression in the slice filter + +* 1.16.2 (2014-10-17) + + * fixed timezone on dates as strings + * fixed 2-words test names when a custom node class is not used + * fixed macros when using an argument named like a PHP super global (like GET or POST) + * fixed date_modify when working with DateTimeImmutable + * optimized for loops + * fixed multi-byte characters handling in the split filter + * fixed a regression in the in operator + * fixed a regression in the slice filter + +* 1.16.1 (2014-10-10) + + * improved error reporting in a sandboxed template + * fixed missing error file/line information under certain circumstances + * fixed wrong error line number in some error messages + * fixed the in operator to use strict comparisons + * sped up the slice filter + * fixed for mb function overload mb_substr acting different + * fixed the attribute() function when passing a variable for the arguments + +* 1.16.0 (2014-07-05) + + * changed url_encode to always encode according to RFC 3986 + * fixed inheritance in a 'use'-hierarchy + * removed the __toString policy check when the sandbox is disabled + * fixed recursively calling blocks in templates with inheritance + +* 1.15.1 (2014-02-13) + + * fixed the conversion of the special '0000-00-00 00:00' date + * added an error message when trying to import an undefined block from a trait + * fixed a C extension crash when accessing defined but uninitialized property. + +* 1.15.0 (2013-12-06) + + * made ignoreStrictCheck in Template::getAttribute() works with __call() methods throwing BadMethodCallException + * added min and max functions + * added the round filter + * fixed a bug that prevented the optimizers to be enabled/disabled selectively + * fixed first and last filters for UTF-8 strings + * added a source function to include the content of a template without rendering it + * fixed the C extension sandbox behavior when get or set is prepend to method name + +* 1.14.2 (2013-10-30) + + * fixed error filename/line when an error occurs in an included file + * allowed operators that contain whitespaces to have more than one whitespace + * allowed tests to be made of 1 or 2 words (like "same as" or "divisible by") + +* 1.14.1 (2013-10-15) + + * made it possible to use named operators as variables + * fixed the possibility to have a variable named 'matches' + * added support for PHP 5.5 DateTimeInterface + +* 1.14.0 (2013-10-03) + + * fixed usage of the html_attr escaping strategy to avoid double-escaping with the html strategy + * added new operators: ends with, starts with, and matches + * fixed some compatibility issues with HHVM + * added a way to add custom escaping strategies + * fixed the C extension compilation on Windows + * fixed the batch filter when using a fill argument with an exact match of elements to batch + * fixed the filesystem loader cache when a template name exists in several namespaces + * fixed template_from_string when the template includes or extends other ones + * fixed a crash of the C extension on an edge case + +* 1.13.2 (2013-08-03) + + * fixed the error line number for an error occurs in and embedded template + * fixed crashes of the C extension on some edge cases + +* 1.13.1 (2013-06-06) + + * added the possibility to ignore the filesystem constructor argument in Twig_Loader_Filesystem + * fixed Twig_Loader_Chain::exists() for a loader which implements Twig_ExistsLoaderInterface + * adjusted backtrace call to reduce memory usage when an error occurs + * added support for object instances as the second argument of the constant test + * fixed the include function when used in an assignment + +* 1.13.0 (2013-05-10) + + * fixed getting a numeric-like item on a variable ('09' for instance) + * fixed getting a boolean or float key on an array, so it is consistent with PHP's array access: + `{{ array[false] }}` behaves the same as `echo $array[false];` (equals `$array[0]`) + * made the escape filter 20% faster for happy path (escaping string for html with UTF-8) + * changed ☃ to § in tests + * enforced usage of named arguments after positional ones + +* 1.12.3 (2013-04-08) + + * fixed a security issue in the filesystem loader where it was possible to include a template one + level above the configured path + * fixed fatal error that should be an exception when adding a filter/function/test too late + * added a batch filter + * added support for encoding an array as query string in the url_encode filter + +* 1.12.2 (2013-02-09) + + * fixed the timezone used by the date filter and function when the given date contains a timezone (like 2010-01-28T15:00:00+02:00) + * fixed globals when getGlobals is called early on + * added the first and last filter + +* 1.12.1 (2013-01-15) + + * added support for object instances as the second argument of the constant function + * relaxed globals management to avoid a BC break + * added support for {{ some_string[:2] }} + +* 1.12.0 (2013-01-08) + + * added verbatim as an alias for the raw tag to avoid confusion with the raw filter + * fixed registration of tests and functions as anonymous functions + * fixed globals management + +* 1.12.0-RC1 (2012-12-29) + + * added an include function (does the same as the include tag but in a more flexible way) + * added the ability to use any PHP callable to define filters, functions, and tests + * added a syntax error when using a loop variable that is not defined + * added the ability to set default values for macro arguments + * added support for named arguments for filters, tests, and functions + * moved filters/functions/tests syntax errors to the parser + * added support for extended ternary operator syntaxes + +* 1.11.1 (2012-11-11) + + * fixed debug info line numbering (was off by 2) + * fixed escaping when calling a macro inside another one (regression introduced in 1.9.1) + * optimized variable access on PHP 5.4 + * fixed a crash of the C extension when an exception was thrown from a macro called without being imported (using _self.XXX) + +* 1.11.0 (2012-11-07) + + * fixed macro compilation when a variable name is a PHP reserved keyword + * changed the date filter behavior to always apply the default timezone, except if false is passed as the timezone + * fixed bitwise operator precedences + * added the template_from_string function + * fixed default timezone usage for the date function + * optimized the way Twig exceptions are managed (to make them faster) + * added Twig_ExistsLoaderInterface (implementing this interface in your loader make the chain loader much faster) + +* 1.10.3 (2012-10-19) + + * fixed wrong template location in some error messages + * reverted a BC break introduced in 1.10.2 + * added a split filter + +* 1.10.2 (2012-10-15) + + * fixed macro calls on PHP 5.4 + +* 1.10.1 (2012-10-15) + + * made a speed optimization to macro calls when imported via the "import" tag + * fixed C extension compilation on Windows + * fixed a segfault in the C extension when using DateTime objects + +* 1.10.0 (2012-09-28) + + * extracted functional tests framework to make it reusable for third-party extensions + * added namespaced templates support in Twig_Loader_Filesystem + * added Twig_Loader_Filesystem::prependPath() + * fixed an error when a token parser pass a closure as a test to the subparse() method + +* 1.9.2 (2012-08-25) + + * fixed the in operator for objects that contain circular references + * fixed the C extension when accessing a public property of an object implementing the \ArrayAccess interface + +* 1.9.1 (2012-07-22) + + * optimized macro calls when auto-escaping is on + * fixed wrong parent class for Twig_Function_Node + * made Twig_Loader_Chain more explicit about problems + +* 1.9.0 (2012-07-13) + + * made the parsing independent of the template loaders + * fixed exception trace when an error occurs when rendering a child template + * added escaping strategies for CSS, URL, and HTML attributes + * fixed nested embed tag calls + * added the date_modify filter + +* 1.8.3 (2012-06-17) + + * fixed paths in the filesystem loader when passing a path that ends with a slash or a backslash + * fixed escaping when a project defines a function named html or js + * fixed chmod mode to apply the umask correctly + +* 1.8.2 (2012-05-30) + + * added the abs filter + * fixed a regression when using a number in template attributes + * fixed compiler when mbstring.func_overload is set to 2 + * fixed DateTimeZone support in date filter + +* 1.8.1 (2012-05-17) + + * fixed a regression when dealing with SimpleXMLElement instances in templates + * fixed "is_safe" value for the "dump" function when "html_errors" is not defined in php.ini + * switched to use mbstring whenever possible instead of iconv (you might need to update your encoding as mbstring and iconv encoding names sometimes differ) + +* 1.8.0 (2012-05-08) + + * enforced interface when adding tests, filters, functions, and node visitors from extensions + * fixed a side-effect of the date filter where the timezone might be changed + * simplified usage of the autoescape tag; the only (optional) argument is now the escaping strategy or false (with a BC layer) + * added a way to dynamically change the auto-escaping strategy according to the template "filename" + * changed the autoescape option to also accept a supported escaping strategy (for BC, true is equivalent to html) + * added an embed tag + +* 1.7.0 (2012-04-24) + + * fixed a PHP warning when using CIFS + * fixed template line number in some exceptions + * added an iterable test + * added an error when defining two blocks with the same name in a template + * added the preserves_safety option for filters + * fixed a PHP notice when trying to access a key on a non-object/array variable + * enhanced error reporting when the template file is an instance of SplFileInfo + * added Twig_Environment::mergeGlobals() + * added compilation checks to avoid misuses of the sandbox tag + * fixed filesystem loader freshness logic for high traffic websites + * fixed random function when charset is null + +* 1.6.5 (2012-04-11) + + * fixed a regression when a template only extends another one without defining any blocks + +* 1.6.4 (2012-04-02) + + * fixed PHP notice in Twig_Error::guessTemplateLine() introduced in 1.6.3 + * fixed performance when compiling large files + * optimized parent template creation when the template does not use dynamic inheritance + +* 1.6.3 (2012-03-22) + + * fixed usage of Z_ADDREF_P for PHP 5.2 in the C extension + * fixed compilation of numeric values used in templates when using a locale where the decimal separator is not a dot + * made the strategy used to guess the real template file name and line number in exception messages much faster and more accurate + +* 1.6.2 (2012-03-18) + + * fixed sandbox mode when used with inheritance + * added preserveKeys support for the slice filter + * fixed the date filter when a DateTime instance is passed with a specific timezone + * added a trim filter + +* 1.6.1 (2012-02-29) + + * fixed Twig C extension + * removed the creation of Twig_Markup instances when not needed + * added a way to set the default global timezone for dates + * fixed the slice filter on strings when the length is not specified + * fixed the creation of the cache directory in case of a race condition + +* 1.6.0 (2012-02-04) + + * fixed raw blocks when used with the whitespace trim option + * made a speed optimization to macro calls when imported via the "from" tag + * fixed globals, parsers, visitors, filters, tests, and functions management in Twig_Environment when a new one or new extension is added + * fixed the attribute function when passing arguments + * added slice notation support for the [] operator (syntactic sugar for the slice operator) + * added a slice filter + * added string support for the reverse filter + * fixed the empty test and the length filter for Twig_Markup instances + * added a date function to ease date comparison + * fixed unary operators precedence + * added recursive parsing support in the parser + * added string and integer handling for the random function + +* 1.5.1 (2012-01-05) + + * fixed a regression when parsing strings + +* 1.5.0 (2012-01-04) + + * added Traversable objects support for the join filter + +* 1.5.0-RC2 (2011-12-30) + + * added a way to set the default global date interval format + * fixed the date filter for DateInterval instances (setTimezone() does not exist for them) + * refactored Twig_Template::display() to ease its extension + * added a number_format filter + +* 1.5.0-RC1 (2011-12-26) + + * removed the need to quote hash keys + * allowed hash keys to be any expression + * added a do tag + * added a flush tag + * added support for dynamically named filters and functions + * added a dump function to help debugging templates + * added a nl2br filter + * added a random function + * added a way to change the default format for the date filter + * fixed the lexer when an operator ending with a letter ends a line + * added string interpolation support + * enhanced exceptions for unknown filters, functions, tests, and tags + +* 1.4.0 (2011-12-07) + + * fixed lexer when using big numbers (> PHP_INT_MAX) + * added missing preserveKeys argument to the reverse filter + * fixed macros containing filter tag calls + +* 1.4.0-RC2 (2011-11-27) + + * removed usage of Reflection in Twig_Template::getAttribute() + * added a C extension that can optionally replace Twig_Template::getAttribute() + * added negative timestamp support to the date filter + +* 1.4.0-RC1 (2011-11-20) + + * optimized variable access when using PHP 5.4 + * changed the precedence of the .. operator to be more consistent with languages that implements such a feature like Ruby + * added an Exception to Twig_Loader_Array::isFresh() method when the template does not exist to be consistent with other loaders + * added Twig_Function_Node to allow more complex functions to have their own Node class + * added Twig_Filter_Node to allow more complex filters to have their own Node class + * added Twig_Test_Node to allow more complex tests to have their own Node class + * added a better error message when a template is empty but contain a BOM + * fixed "in" operator for empty strings + * fixed the "defined" test and the "default" filter (now works with more than one call (foo.bar.foo) and for both values of the strict_variables option) + * changed the way extensions are loaded (addFilter/addFunction/addGlobal/addTest/addNodeVisitor/addTokenParser/addExtension can now be called in any order) + * added Twig_Environment::display() + * made the escape filter smarter when the encoding is not supported by PHP + * added a convert_encoding filter + * moved all node manipulations outside the compile() Node method + * made several speed optimizations + +* 1.3.0 (2011-10-08) + +no changes + +* 1.3.0-RC1 (2011-10-04) + + * added an optimization for the parent() function + * added cache reloading when auto_reload is true and an extension has been modified + * added the possibility to force the escaping of a string already marked as safe (instance of Twig_Markup) + * allowed empty templates to be used as traits + * added traits support for the "parent" function + +* 1.2.0 (2011-09-13) + +no changes + +* 1.2.0-RC1 (2011-09-10) + + * enhanced the exception when a tag remains unclosed + * added support for empty Countable objects for the "empty" test + * fixed algorithm that determines if a template using inheritance is valid (no output between block definitions) + * added better support for encoding problems when escaping a string (available as of PHP 5.4) + * added a way to ignore a missing template when using the "include" tag ({% include "foo" ignore missing %}) + * added support for an array of templates to the "include" and "extends" tags ({% include ['foo', 'bar'] %}) + * added support for bitwise operators in expressions + * added the "attribute" function to allow getting dynamic attributes on variables + * added Twig_Loader_Chain + * added Twig_Loader_Array::setTemplate() + * added an optimization for the set tag when used to capture a large chunk of static text + * changed name regex to match PHP one "[a-zA-Z_\x7f-\xff][a-zA-Z0-9_\x7f-\xff]*" (works for blocks, tags, functions, filters, and macros) + * removed the possibility to use the "extends" tag from a block + * added "if" modifier support to "for" loops + +* 1.1.2 (2011-07-30) + + * fixed json_encode filter on PHP 5.2 + * fixed regression introduced in 1.1.1 ({{ block(foo|lower) }}) + * fixed inheritance when using conditional parents + * fixed compilation of templates when the body of a child template is not empty + * fixed output when a macro throws an exception + * fixed a parsing problem when a large chunk of text is enclosed in a comment tag + * added PHPDoc for all Token parsers and Core extension functions + +* 1.1.1 (2011-07-17) + + * added a performance optimization in the Optimizer (also helps to lower the number of nested level calls) + * made some performance improvement for some edge cases + +* 1.1.0 (2011-06-28) + + * fixed json_encode filter + +* 1.1.0-RC3 (2011-06-24) + + * fixed method case-sensitivity when using the sandbox mode + * added timezone support for the date filter + * fixed possible security problems with NUL bytes + +* 1.1.0-RC2 (2011-06-16) + + * added an exception when the template passed to "use" is not a string + * made 'a.b is defined' not throw an exception if a is not defined (in strict mode) + * added {% line \d+ %} directive + +* 1.1.0-RC1 (2011-05-28) + +Flush your cache after upgrading. + + * fixed date filter when using a timestamp + * fixed the defined test for some cases + * fixed a parsing problem when a large chunk of text is enclosed in a raw tag + * added support for horizontal reuse of template blocks (see docs for more information) + * added whitespace control modifier to all tags (see docs for more information) + * added null as an alias for none (the null test is also an alias for the none test now) + * made TRUE, FALSE, NONE equivalent to their lowercase counterparts + * wrapped all compilation and runtime exceptions with Twig_Error_Runtime and added logic to guess the template name and line + * moved display() method to Twig_Template (generated templates should now use doDisplay() instead) + +* 1.0.0 (2011-03-27) + + * fixed output when using mbstring + * fixed duplicate call of methods when using the sandbox + * made the charset configurable for the escape filter + +* 1.0.0-RC2 (2011-02-21) + + * changed the way {% set %} works when capturing (the content is now marked as safe) + * added support for macro name in the endmacro tag + * make Twig_Error compatible with PHP 5.3.0 > + * fixed an infinite loop on some Windows configurations + * fixed the "length" filter for numbers + * fixed Template::getAttribute() as properties in PHP are case sensitive + * removed coupling between Twig_Node and Twig_Template + * fixed the ternary operator precedence rule + +* 1.0.0-RC1 (2011-01-09) + +Backward incompatibilities: + + * the "items" filter, which has been deprecated for quite a long time now, has been removed + * the "range" filter has been converted to a function: 0|range(10) -> range(0, 10) + * the "constant" filter has been converted to a function: {{ some_date|date('DATE_W3C'|constant) }} -> {{ some_date|date(constant('DATE_W3C')) }} + * the "cycle" filter has been converted to a function: {{ ['odd', 'even']|cycle(i) }} -> {{ cycle(['odd', 'even'], i) }} + * the "for" tag does not support "joined by" anymore + * the "autoescape" first argument is now "true"/"false" (instead of "on"/"off") + * the "parent" tag has been replaced by a "parent" function ({{ parent() }} instead of {% parent %}) + * the "display" tag has been replaced by a "block" function ({{ block('title') }} instead of {% display title %}) + * removed the grammar and simple token parser (moved to the Twig Extensions repository) + +Changes: + + * added "needs_context" option for filters and functions (the context is then passed as a first argument) + * added global variables support + * made macros return their value instead of echoing directly (fixes calling a macro in sandbox mode) + * added the "from" tag to import macros as functions + * added support for functions (a function is just syntactic sugar for a getAttribute() call) + * made macros callable when sandbox mode is enabled + * added an exception when a macro uses a reserved name + * the "default" filter now uses the "empty" test instead of just checking for null + * added the "empty" test + +* 0.9.10 (2010-12-16) + +Backward incompatibilities: + + * The Escaper extension is enabled by default, which means that all displayed + variables are now automatically escaped. You can revert to the previous + behavior by removing the extension via $env->removeExtension('escaper') + or just set the 'autoescape' option to 'false'. + * removed the "without loop" attribute for the "for" tag (not needed anymore + as the Optimizer take care of that for most cases) + * arrays and hashes have now a different syntax + * arrays keep the same syntax with square brackets: [1, 2] + * hashes now use curly braces (["a": "b"] should now be written as {"a": "b"}) + * support for "arrays with keys" and "hashes without keys" is not supported anymore ([1, "foo": "bar"] or {"foo": "bar", 1}) + * the i18n extension is now part of the Twig Extensions repository + +Changes: + + * added the merge filter + * removed 'is_escaper' option for filters (a left over from the previous version) -- you must use 'is_safe' now instead + * fixed usage of operators as method names (like is, in, and not) + * changed the order of execution for node visitors + * fixed default() filter behavior when used with strict_variables set to on + * fixed filesystem loader compatibility with PHAR files + * enhanced error messages when an unexpected token is parsed in an expression + * fixed filename not being added to syntax error messages + * added the autoescape option to enable/disable autoescaping + * removed the newline after a comment (mimics PHP behavior) + * added a syntax error exception when parent block is used on a template that does not extend another one + * made the Escaper extension enabled by default + * fixed sandbox extension when used with auto output escaping + * fixed escaper when wrapping a Twig_Node_Print (the original class must be preserved) + * added an Optimizer extension (enabled by default; optimizes "for" loops and "raw" filters) + * added priority to node visitors + +* 0.9.9 (2010-11-28) + +Backward incompatibilities: + * the self special variable has been renamed to _self + * the odd and even filters are now tests: + {{ foo|odd }} must now be written {{ foo is odd }} + * the "safe" filter has been renamed to "raw" + * in Node classes, + sub-nodes are now accessed via getNode() (instead of property access) + attributes via getAttribute() (instead of array access) + * the urlencode filter had been renamed to url_encode + * the include tag now merges the passed variables with the current context by default + (the old behavior is still possible by adding the "only" keyword) + * moved Exceptions to Twig_Error_* (Twig_SyntaxError/Twig_RuntimeError are now Twig_Error_Syntax/Twig_Error_Runtime) + * removed support for {{ 1 < i < 3 }} (use {{ i > 1 and i < 3 }} instead) + * the "in" filter has been removed ({{ a|in(b) }} should now be written {{ a in b }}) + +Changes: + * added file and line to Twig_Error_Runtime exceptions thrown from Twig_Template + * changed trans tag to accept any variable for the plural count + * fixed sandbox mode (__toString() method check was not enforced if called implicitly from complex statements) + * added the ** (power) operator + * changed the algorithm used for parsing expressions + * added the spaceless tag + * removed trim_blocks option + * added support for is*() methods for attributes (foo.bar now looks for foo->getBar() or foo->isBar()) + * changed all exceptions to extend Twig_Error + * fixed unary expressions ({{ not(1 or 0) }}) + * fixed child templates (with an extend tag) that uses one or more imports + * added support for {{ 1 not in [2, 3] }} (more readable than the current {{ not (1 in [2, 3]) }}) + * escaping has been rewritten + * the implementation of template inheritance has been rewritten + (blocks can now be called individually and still work with inheritance) + * fixed error handling for if tag when a syntax error occurs within a subparse process + * added a way to implement custom logic for resolving token parsers given a tag name + * fixed js escaper to be stricter (now uses a whilelist-based js escaper) + * added the following filers: "constant", "trans", "replace", "json_encode" + * added a "constant" test + * fixed objects with __toString() not being autoescaped + * fixed subscript expressions when calling __call() (methods now keep the case) + * added "test" feature (accessible via the "is" operator) + * removed the debug tag (should be done in an extension) + * fixed trans tag when no vars are used in plural form + * fixed race condition when writing template cache + * added the special _charset variable to reference the current charset + * added the special _context variable to reference the current context + * renamed self to _self (to avoid conflict) + * fixed Twig_Template::getAttribute() for protected properties + +* 0.9.8 (2010-06-28) + +Backward incompatibilities: + * the trans tag plural count is now attached to the plural tag: + old: `{% trans count %}...{% plural %}...{% endtrans %}` + new: `{% trans %}...{% plural count %}...{% endtrans %}` + + * added a way to translate strings coming from a variable ({% trans var %}) + * fixed trans tag when used with the Escaper extension + * fixed default cache umask + * removed Twig_Template instances from the debug tag output + * fixed objects with __isset() defined + * fixed set tag when used with a capture + * fixed type hinting for Twig_Environment::addFilter() method + +* 0.9.7 (2010-06-12) + +Backward incompatibilities: + * changed 'as' to '=' for the set tag ({% set title as "Title" %} must now be {% set title = "Title" %}) + * removed the sandboxed attribute of the include tag (use the new sandbox tag instead) + * refactored the Node system (if you have custom nodes, you will have to update them to use the new API) + + * added self as a special variable that refers to the current template (useful for importing macros from the current template) + * added Twig_Template instance support to the include tag + * added support for dynamic and conditional inheritance ({% extends some_var %} and {% extends standalone ? "minimum" : "base" %}) + * added a grammar sub-framework to ease the creation of custom tags + * fixed the for tag for large arrays (some loop variables are now only available for arrays and objects that implement the Countable interface) + * removed the Twig_Resource::resolveMissingFilter() method + * fixed the filter tag which did not apply filtering to included files + * added a bunch of unit tests + * added a bunch of phpdoc + * added a sandbox tag in the sandbox extension + * changed the date filter to support any date format supported by DateTime + * added strict_variable setting to throw an exception when an invalid variable is used in a template (disabled by default) + * added the lexer, parser, and compiler as arguments to the Twig_Environment constructor + * changed the cache option to only accepts an explicit path to a cache directory or false + * added a way to add token parsers, filters, and visitors without creating an extension + * added three interfaces: Twig_NodeInterface, Twig_TokenParserInterface, and Twig_FilterInterface + * changed the generated code to match the new coding standards + * fixed sandbox mode (__toString() method check was not enforced if called implicitly from a simple statement like {{ article }}) + * added an exception when a child template has a non-empty body (as it is always ignored when rendering) + +* 0.9.6 (2010-05-12) + + * fixed variables defined outside a loop and for which the value changes in a for loop + * fixed the test suite for PHP 5.2 and older versions of PHPUnit + * added support for __call() in expression resolution + * fixed node visiting for macros (macros are now visited by visitors as any other node) + * fixed nested block definitions with a parent call (rarely useful but nonetheless supported now) + * added the cycle filter + * fixed the Lexer when mbstring.func_overload is used with an mbstring.internal_encoding different from ASCII + * added a long-syntax for the set tag ({% set foo %}...{% endset %}) + * unit tests are now powered by PHPUnit + * added support for gettext via the `i18n` extension + * fixed twig_capitalize_string_filter() and fixed twig_length_filter() when used with UTF-8 values + * added a more useful exception if an if tag is not closed properly + * added support for escaping strategy in the autoescape tag + * fixed lexer when a template has a big chunk of text between/in a block + +* 0.9.5 (2010-01-20) + +As for any new release, don't forget to remove all cached templates after +upgrading. + +If you have defined custom filters, you MUST upgrade them for this release. To +upgrade, replace "array" with "new Twig_Filter_Function", and replace the +environment constant by the "needs_environment" option: + + // before + 'even' => array('twig_is_even_filter', false), + 'escape' => array('twig_escape_filter', true), + + // after + 'even' => new Twig_Filter_Function('twig_is_even_filter'), + 'escape' => new Twig_Filter_Function('twig_escape_filter', array('needs_environment' => true)), + +If you have created NodeTransformer classes, you will need to upgrade them to +the new interface (please note that the interface is not yet considered +stable). + + * fixed list nodes that did not extend the Twig_NodeListInterface + * added the "without loop" option to the for tag (it disables the generation of the loop variable) + * refactored node transformers to node visitors + * fixed automatic-escaping for blocks + * added a way to specify variables to pass to an included template + * changed the automatic-escaping rules to be more sensible and more configurable in custom filters (the documentation lists all the rules) + * improved the filter system to allow object methods to be used as filters + * changed the Array and String loaders to actually make use of the cache mechanism + * included the default filter function definitions in the extension class files directly (Core, Escaper) + * added the // operator (like the floor() PHP function) + * added the .. operator (as a syntactic sugar for the range filter when the step is 1) + * added the in operator (as a syntactic sugar for the in filter) + * added the following filters in the Core extension: in, range + * added support for arrays (same behavior as in PHP, a mix between lists and dictionaries, arrays and hashes) + * enhanced some error messages to provide better feedback in case of parsing errors + +* 0.9.4 (2009-12-02) + +If you have custom loaders, you MUST upgrade them for this release: The +Twig_Loader base class has been removed, and the Twig_LoaderInterface has also +been changed (see the source code for more information or the documentation). + + * added support for DateTime instances for the date filter + * fixed loop.last when the array only has one item + * made it possible to insert newlines in tag and variable blocks + * fixed a bug when a literal '\n' were present in a template text + * fixed bug when the filename of a template contains */ + * refactored loaders + +* 0.9.3 (2009-11-11) + +This release is NOT backward compatible with the previous releases. + + The loaders do not take the cache and autoReload arguments anymore. Instead, + the Twig_Environment class has two new options: cache and auto_reload. + Upgrading your code means changing this kind of code: + + $loader = new Twig_Loader_Filesystem('/path/to/templates', '/path/to/compilation_cache', true); + $twig = new Twig_Environment($loader); + + to something like this: + + $loader = new Twig_Loader_Filesystem('/path/to/templates'); + $twig = new Twig_Environment($loader, array( + 'cache' => '/path/to/compilation_cache', + 'auto_reload' => true, + )); + + * deprecated the "items" filter as it is not needed anymore + * made cache and auto_reload options of Twig_Environment instead of arguments of Twig_Loader + * optimized template loading speed + * removed output when an error occurs in a template and render() is used + * made major speed improvements for loops (up to 300% on even the smallest loops) + * added properties as part of the sandbox mode + * added public properties support (obj.item can now be the item property on the obj object) + * extended set tag to support expression as value ({% set foo as 'foo' ~ 'bar' %} ) + * fixed bug when \ was used in HTML + +* 0.9.2 (2009-10-29) + + * made some speed optimizations + * changed the cache extension to .php + * added a js escaping strategy + * added support for short block tag + * changed the filter tag to allow chained filters + * made lexer more flexible as you can now change the default delimiters + * added set tag + * changed default directory permission when cache dir does not exist (more secure) + * added macro support + * changed filters first optional argument to be a Twig_Environment instance instead of a Twig_Template instance + * made Twig_Autoloader::autoload() a static method + * avoid writing template file if an error occurs + * added $ escaping when outputting raw strings + * enhanced some error messages to ease debugging + * fixed empty cache files when the template contains an error + +* 0.9.1 (2009-10-14) + + * fixed a bug in PHP 5.2.6 + * fixed numbers with one than one decimal + * added support for method calls with arguments ({{ foo.bar('a', 43) }}) + * made small speed optimizations + * made minor tweaks to allow better extensibility and flexibility + +* 0.9.0 (2009-10-12) + + * Initial release diff --git a/src/lib/Twig/Cache/Filesystem.php b/src/php/lib/Twig/Cache/Filesystem.php similarity index 100% rename from src/lib/Twig/Cache/Filesystem.php rename to src/php/lib/Twig/Cache/Filesystem.php diff --git a/src/lib/Twig/Cache/Null.php b/src/php/lib/Twig/Cache/Null.php similarity index 100% rename from src/lib/Twig/Cache/Null.php rename to src/php/lib/Twig/Cache/Null.php diff --git a/src/lib/Twig/CacheInterface.php b/src/php/lib/Twig/CacheInterface.php similarity index 100% rename from src/lib/Twig/CacheInterface.php rename to src/php/lib/Twig/CacheInterface.php diff --git a/src/lib/Twig/Compiler.php b/src/php/lib/Twig/Compiler.php similarity index 97% rename from src/lib/Twig/Compiler.php rename to src/php/lib/Twig/Compiler.php index e90bc98..803eb89 100644 --- a/src/lib/Twig/Compiler.php +++ b/src/php/lib/Twig/Compiler.php @@ -25,6 +25,7 @@ class Twig_Compiler implements Twig_CompilerInterface protected $sourceOffset; protected $sourceLine; protected $filename; + private $varNameSalt = 0; public function __construct(Twig_Environment $env) { @@ -78,6 +79,7 @@ public function compile(Twig_NodeInterface $node, $indentation = 0) // source code starts at 1 (as we then increment it when we encounter new lines) $this->sourceLine = 1; $this->indentation = $indentation; + $this->varNameSalt = 0; if ($node instanceof Twig_Node_Module) { // to be removed in 2.0 @@ -276,7 +278,7 @@ public function outdent($step = 1) public function getVarName() { - return sprintf('__internal_%s', hash('sha256', uniqid(mt_rand(), true), false)); + return sprintf('__internal_%s', hash('sha256', __METHOD__.$this->varNameSalt++)); } } diff --git a/src/lib/Twig/CompilerInterface.php b/src/php/lib/Twig/CompilerInterface.php similarity index 100% rename from src/lib/Twig/CompilerInterface.php rename to src/php/lib/Twig/CompilerInterface.php diff --git a/src/lib/Twig/ContainerRuntimeLoader.php b/src/php/lib/Twig/ContainerRuntimeLoader.php similarity index 100% rename from src/lib/Twig/ContainerRuntimeLoader.php rename to src/php/lib/Twig/ContainerRuntimeLoader.php diff --git a/src/lib/Twig/Environment.php b/src/php/lib/Twig/Environment.php similarity index 98% rename from src/lib/Twig/Environment.php rename to src/php/lib/Twig/Environment.php index 5de2f27..e4e8abd 100644 --- a/src/lib/Twig/Environment.php +++ b/src/php/lib/Twig/Environment.php @@ -16,11 +16,11 @@ */ class Twig_Environment { - const VERSION = '1.35.0'; - const VERSION_ID = 13500; + const VERSION = '1.35.3'; + const VERSION_ID = 13503; const MAJOR_VERSION = 1; const MINOR_VERSION = 35; - const RELEASE_VERSION = 0; + const RELEASE_VERSION = 3; const EXTRA_VERSION = ''; protected $charset; @@ -132,14 +132,14 @@ public function __construct(Twig_LoaderInterface $loader = null, $options = arra // For BC if (is_string($this->originalCache)) { $r = new ReflectionMethod($this, 'writeCacheFile'); - if ($r->getDeclaringClass()->getName() !== __CLASS__) { + if (__CLASS__ !== $r->getDeclaringClass()->getName()) { @trigger_error('The Twig_Environment::writeCacheFile method is deprecated since version 1.22 and will be removed in Twig 2.0.', E_USER_DEPRECATED); $this->bcWriteCacheFile = true; } $r = new ReflectionMethod($this, 'getCacheFilename'); - if ($r->getDeclaringClass()->getName() !== __CLASS__) { + if (__CLASS__ !== $r->getDeclaringClass()->getName()) { @trigger_error('The Twig_Environment::getCacheFilename method is deprecated since version 1.22 and will be removed in Twig 2.0.', E_USER_DEPRECATED); $this->bcGetCacheFilename = true; @@ -562,12 +562,12 @@ public function isTemplateFresh($name, $time) /** * Tries to load a template consecutively from an array. * - * Similar to loadTemplate() but it also accepts Twig_TemplateInterface instances and an array - * of templates where each is tried to be loaded. + * Similar to loadTemplate() but it also accepts instances of Twig_Template and + * Twig_TemplateWrapper, and an array of templates where each is tried to be loaded. * - * @param string|Twig_Template|array $names A template or an array of templates to try consecutively + * @param string|Twig_Template|Twig_TemplateWrapper|array $names A template or an array of templates to try consecutively * - * @return Twig_Template + * @return Twig_Template|Twig_TemplateWrapper * * @throws Twig_Error_Loader When none of the templates can be found * @throws Twig_Error_Syntax When an error occurred during compilation @@ -583,6 +583,10 @@ public function resolveTemplate($names) return $name; } + if ($name instanceof Twig_TemplateWrapper) { + return $name; + } + try { return $this->loadTemplate($name); } catch (Twig_Error_Loader $e) { diff --git a/src/lib/Twig/Error.php b/src/php/lib/Twig/Error.php similarity index 100% rename from src/lib/Twig/Error.php rename to src/php/lib/Twig/Error.php diff --git a/src/lib/Twig/Error/Loader.php b/src/php/lib/Twig/Error/Loader.php similarity index 100% rename from src/lib/Twig/Error/Loader.php rename to src/php/lib/Twig/Error/Loader.php diff --git a/src/lib/Twig/Error/Runtime.php b/src/php/lib/Twig/Error/Runtime.php similarity index 100% rename from src/lib/Twig/Error/Runtime.php rename to src/php/lib/Twig/Error/Runtime.php diff --git a/src/lib/Twig/Error/Syntax.php b/src/php/lib/Twig/Error/Syntax.php similarity index 100% rename from src/lib/Twig/Error/Syntax.php rename to src/php/lib/Twig/Error/Syntax.php diff --git a/src/lib/Twig/ExistsLoaderInterface.php b/src/php/lib/Twig/ExistsLoaderInterface.php similarity index 100% rename from src/lib/Twig/ExistsLoaderInterface.php rename to src/php/lib/Twig/ExistsLoaderInterface.php diff --git a/src/lib/Twig/ExpressionParser.php b/src/php/lib/Twig/ExpressionParser.php similarity index 97% rename from src/lib/Twig/ExpressionParser.php rename to src/php/lib/Twig/ExpressionParser.php index 4906f90..2b84f3f 100644 --- a/src/lib/Twig/ExpressionParser.php +++ b/src/php/lib/Twig/ExpressionParser.php @@ -199,11 +199,14 @@ public function parsePrimaryExpression() break; } + // no break default: if ($token->test(Twig_Token::PUNCTUATION_TYPE, '[')) { $node = $this->parseArrayExpression(); } elseif ($token->test(Twig_Token::PUNCTUATION_TYPE, '{')) { $node = $this->parseHashExpression(); + } elseif ($token->test(Twig_Token::OPERATOR_TYPE, '=') && ('==' === $this->parser->getStream()->look(-1)->getValue() || '!=' === $this->parser->getStream()->look(-1)->getValue())) { + throw new Twig_Error_Syntax(sprintf('Unexpected operator of value "%s". Did you try to use "===" or "!==" for strict comparison? Use "is same as(value)" instead.', $token->getValue()), $token->getLine(), $this->parser->getStream()->getSourceContext()); } else { throw new Twig_Error_Syntax(sprintf('Unexpected token "%s" of value "%s".', Twig_Token::typeToEnglish($token->getType()), $token->getValue()), $token->getLine(), $this->parser->getStream()->getSourceContext()); } @@ -313,7 +316,7 @@ public function parsePostfixExpression($node) { while (true) { $token = $this->parser->getCurrentToken(); - if ($token->getType() == Twig_Token::PUNCTUATION_TYPE) { + if (Twig_Token::PUNCTUATION_TYPE == $token->getType()) { if ('.' == $token->getValue() || '[' == $token->getValue()) { $node = $this->parseSubscriptExpression($node); } elseif ('|' == $token->getValue()) { @@ -384,14 +387,14 @@ public function parseSubscriptExpression($node) $lineno = $token->getLine(); $arguments = new Twig_Node_Expression_Array(array(), $lineno); $type = Twig_Template::ANY_CALL; - if ($token->getValue() == '.') { + if ('.' == $token->getValue()) { $token = $stream->next(); if ( - $token->getType() == Twig_Token::NAME_TYPE + Twig_Token::NAME_TYPE == $token->getType() || - $token->getType() == Twig_Token::NUMBER_TYPE + Twig_Token::NUMBER_TYPE == $token->getType() || - ($token->getType() == Twig_Token::OPERATOR_TYPE && preg_match(Twig_Lexer::REGEX_NAME, $token->getValue())) + (Twig_Token::OPERATOR_TYPE == $token->getType() && preg_match(Twig_Lexer::REGEX_NAME, $token->getValue())) ) { $arg = new Twig_Node_Expression_Constant($token->getValue(), $lineno); diff --git a/src/lib/Twig/Extension.php b/src/php/lib/Twig/Extension.php similarity index 100% rename from src/lib/Twig/Extension.php rename to src/php/lib/Twig/Extension.php diff --git a/src/lib/Twig/Extension/Core.php b/src/php/lib/Twig/Extension/Core.php similarity index 98% rename from src/lib/Twig/Extension/Core.php rename to src/php/lib/Twig/Extension/Core.php index 34f9ef7..6e5569a 100644 --- a/src/lib/Twig/Extension/Core.php +++ b/src/php/lib/Twig/Extension/Core.php @@ -661,7 +661,7 @@ function twig_slice(Twig_Environment $env, $item, $start, $length = null, $prese if ($start >= 0 && $length >= 0 && $item instanceof Iterator) { try { - return iterator_to_array(new LimitIterator($item, $start, $length === null ? -1 : $length), $preserveKeys); + return iterator_to_array(new LimitIterator($item, $start, null === $length ? -1 : $length), $preserveKeys); } catch (OutOfBoundsException $exception) { return array(); } @@ -1216,7 +1216,7 @@ function _twig_escape_html_attr_callback($matches) * The following replaces characters undefined in HTML with the * hex entity for the Unicode replacement character. */ - if (($ord <= 0x1f && $chr != "\t" && $chr != "\n" && $chr != "\r") || ($ord >= 0x7f && $ord <= 0x9f)) { + if (($ord <= 0x1f && "\t" != $chr && "\n" != $chr && "\r" != $chr) || ($ord >= 0x7f && $ord <= 0x9f)) { return '�'; } @@ -1224,7 +1224,7 @@ function _twig_escape_html_attr_callback($matches) * Check if the current character to escape has a name entity we should * replace it with while grabbing the hex value of the character. */ - if (strlen($chr) == 1) { + if (1 == strlen($chr)) { $hex = strtoupper(substr('00'.bin2hex($chr), -2)); } else { $chr = twig_convert_encoding($chr, 'UTF-16BE', 'UTF-8'); @@ -1263,6 +1263,10 @@ function twig_length_filter(Twig_Environment $env, $thing) return mb_strlen($thing, $env->getCharset()); } + if ($thing instanceof \SimpleXMLElement) { + return count($thing); + } + if (is_object($thing) && method_exists($thing, '__toString') && !$thing instanceof \Countable) { return mb_strlen((string) $thing, $env->getCharset()); } @@ -1271,6 +1275,10 @@ function twig_length_filter(Twig_Environment $env, $thing) return count($thing); } + if ($thing instanceof \IteratorAggregate) { + return iterator_count($thing); + } + return 1; } @@ -1362,6 +1370,10 @@ function twig_length_filter(Twig_Environment $env, $thing) return strlen($thing); } + if ($thing instanceof \SimpleXMLElement) { + return count($thing); + } + if (is_object($thing) && method_exists($thing, '__toString') && !$thing instanceof \Countable) { return strlen((string) $thing); } @@ -1370,6 +1382,10 @@ function twig_length_filter(Twig_Environment $env, $thing) return count($thing); } + if ($thing instanceof \IteratorAggregate) { + return iterator_count($thing); + } + return 1; } @@ -1444,7 +1460,7 @@ function twig_test_empty($value) * ** {# evaluates to true if the foo variable is an array or a traversable object #} - * {% if foo is traversable %} + * {% if foo is iterable %} * {# ... #} * {% endif %} *diff --git a/src/lib/Twig/Extension/Debug.php b/src/php/lib/Twig/Extension/Debug.php similarity index 100% rename from src/lib/Twig/Extension/Debug.php rename to src/php/lib/Twig/Extension/Debug.php diff --git a/src/lib/Twig/Extension/Escaper.php b/src/php/lib/Twig/Extension/Escaper.php similarity index 100% rename from src/lib/Twig/Extension/Escaper.php rename to src/php/lib/Twig/Extension/Escaper.php diff --git a/src/lib/Twig/Extension/GlobalsInterface.php b/src/php/lib/Twig/Extension/GlobalsInterface.php similarity index 100% rename from src/lib/Twig/Extension/GlobalsInterface.php rename to src/php/lib/Twig/Extension/GlobalsInterface.php diff --git a/src/lib/Twig/Extension/InitRuntimeInterface.php b/src/php/lib/Twig/Extension/InitRuntimeInterface.php similarity index 100% rename from src/lib/Twig/Extension/InitRuntimeInterface.php rename to src/php/lib/Twig/Extension/InitRuntimeInterface.php diff --git a/src/lib/Twig/Extension/Optimizer.php b/src/php/lib/Twig/Extension/Optimizer.php similarity index 100% rename from src/lib/Twig/Extension/Optimizer.php rename to src/php/lib/Twig/Extension/Optimizer.php diff --git a/src/lib/Twig/Extension/Profiler.php b/src/php/lib/Twig/Extension/Profiler.php similarity index 100% rename from src/lib/Twig/Extension/Profiler.php rename to src/php/lib/Twig/Extension/Profiler.php diff --git a/src/lib/Twig/Extension/Sandbox.php b/src/php/lib/Twig/Extension/Sandbox.php similarity index 100% rename from src/lib/Twig/Extension/Sandbox.php rename to src/php/lib/Twig/Extension/Sandbox.php diff --git a/src/lib/Twig/Extension/Staging.php b/src/php/lib/Twig/Extension/Staging.php similarity index 100% rename from src/lib/Twig/Extension/Staging.php rename to src/php/lib/Twig/Extension/Staging.php diff --git a/src/lib/Twig/Extension/StringLoader.php b/src/php/lib/Twig/Extension/StringLoader.php similarity index 100% rename from src/lib/Twig/Extension/StringLoader.php rename to src/php/lib/Twig/Extension/StringLoader.php diff --git a/src/lib/Twig/ExtensionInterface.php b/src/php/lib/Twig/ExtensionInterface.php similarity index 100% rename from src/lib/Twig/ExtensionInterface.php rename to src/php/lib/Twig/ExtensionInterface.php diff --git a/src/lib/Twig/FactoryRuntimeLoader.php b/src/php/lib/Twig/FactoryRuntimeLoader.php similarity index 100% rename from src/lib/Twig/FactoryRuntimeLoader.php rename to src/php/lib/Twig/FactoryRuntimeLoader.php diff --git a/src/lib/Twig/FileExtensionEscapingStrategy.php b/src/php/lib/Twig/FileExtensionEscapingStrategy.php similarity index 100% rename from src/lib/Twig/FileExtensionEscapingStrategy.php rename to src/php/lib/Twig/FileExtensionEscapingStrategy.php diff --git a/src/lib/Twig/Filter.php b/src/php/lib/Twig/Filter.php similarity index 100% rename from src/lib/Twig/Filter.php rename to src/php/lib/Twig/Filter.php diff --git a/src/lib/Twig/Filter/Function.php b/src/php/lib/Twig/Filter/Function.php similarity index 100% rename from src/lib/Twig/Filter/Function.php rename to src/php/lib/Twig/Filter/Function.php diff --git a/src/lib/Twig/Filter/Method.php b/src/php/lib/Twig/Filter/Method.php similarity index 100% rename from src/lib/Twig/Filter/Method.php rename to src/php/lib/Twig/Filter/Method.php diff --git a/src/lib/Twig/Filter/Node.php b/src/php/lib/Twig/Filter/Node.php similarity index 100% rename from src/lib/Twig/Filter/Node.php rename to src/php/lib/Twig/Filter/Node.php diff --git a/src/lib/Twig/FilterCallableInterface.php b/src/php/lib/Twig/FilterCallableInterface.php similarity index 100% rename from src/lib/Twig/FilterCallableInterface.php rename to src/php/lib/Twig/FilterCallableInterface.php diff --git a/src/lib/Twig/FilterInterface.php b/src/php/lib/Twig/FilterInterface.php similarity index 100% rename from src/lib/Twig/FilterInterface.php rename to src/php/lib/Twig/FilterInterface.php diff --git a/src/lib/Twig/Function.php b/src/php/lib/Twig/Function.php similarity index 100% rename from src/lib/Twig/Function.php rename to src/php/lib/Twig/Function.php diff --git a/src/lib/Twig/Function/Function.php b/src/php/lib/Twig/Function/Function.php similarity index 100% rename from src/lib/Twig/Function/Function.php rename to src/php/lib/Twig/Function/Function.php diff --git a/src/lib/Twig/Function/Method.php b/src/php/lib/Twig/Function/Method.php similarity index 100% rename from src/lib/Twig/Function/Method.php rename to src/php/lib/Twig/Function/Method.php diff --git a/src/lib/Twig/Function/Node.php b/src/php/lib/Twig/Function/Node.php similarity index 100% rename from src/lib/Twig/Function/Node.php rename to src/php/lib/Twig/Function/Node.php diff --git a/src/lib/Twig/FunctionCallableInterface.php b/src/php/lib/Twig/FunctionCallableInterface.php similarity index 100% rename from src/lib/Twig/FunctionCallableInterface.php rename to src/php/lib/Twig/FunctionCallableInterface.php diff --git a/src/lib/Twig/FunctionInterface.php b/src/php/lib/Twig/FunctionInterface.php similarity index 100% rename from src/lib/Twig/FunctionInterface.php rename to src/php/lib/Twig/FunctionInterface.php diff --git a/src/lib/Twig/LICENSE b/src/php/lib/Twig/LICENSE similarity index 97% rename from src/lib/Twig/LICENSE rename to src/php/lib/Twig/LICENSE index b6e17a1..e401cb9 100644 --- a/src/lib/Twig/LICENSE +++ b/src/php/lib/Twig/LICENSE @@ -1,4 +1,4 @@ -Copyright (c) 2009-2017 by the Twig Team. +Copyright (c) 2009-2018 by the Twig Team. Some rights reserved. diff --git a/src/lib/Twig/Lexer.php b/src/php/lib/Twig/Lexer.php similarity index 98% rename from src/lib/Twig/Lexer.php rename to src/php/lib/Twig/Lexer.php index 85390b2..41211eb 100644 --- a/src/lib/Twig/Lexer.php +++ b/src/php/lib/Twig/Lexer.php @@ -235,7 +235,7 @@ protected function lexExpression() $this->moveCursor($match[0]); if ($this->cursor >= $this->end) { - throw new Twig_Error_Syntax(sprintf('Unclosed "%s".', $this->state === self::STATE_BLOCK ? 'block' : 'variable'), $this->currentVarBlockLine, $this->source); + throw new Twig_Error_Syntax(sprintf('Unclosed "%s".', self::STATE_BLOCK === $this->state ? 'block' : 'variable'), $this->currentVarBlockLine, $this->source); } } @@ -337,12 +337,15 @@ protected function lexString() $this->moveCursor($match[0]); } elseif (preg_match(self::REGEX_DQ_STRING_DELIM, $this->code, $match, null, $this->cursor)) { list($expect, $lineno) = array_pop($this->brackets); - if ($this->code[$this->cursor] != '"') { + if ('"' != $this->code[$this->cursor]) { throw new Twig_Error_Syntax(sprintf('Unclosed "%s".', $expect), $lineno, $this->source); } $this->popState(); ++$this->cursor; + } else { + // unlexable + throw new Twig_Error_Syntax(sprintf('Unexpected character "%s".', $this->code[$this->cursor]), $this->lineno, $this->source); } } diff --git a/src/lib/Twig/LexerInterface.php b/src/php/lib/Twig/LexerInterface.php similarity index 100% rename from src/lib/Twig/LexerInterface.php rename to src/php/lib/Twig/LexerInterface.php diff --git a/src/lib/Twig/Loader/Array.php b/src/php/lib/Twig/Loader/Array.php similarity index 100% rename from src/lib/Twig/Loader/Array.php rename to src/php/lib/Twig/Loader/Array.php diff --git a/src/lib/Twig/Loader/Chain.php b/src/php/lib/Twig/Loader/Chain.php similarity index 100% rename from src/lib/Twig/Loader/Chain.php rename to src/php/lib/Twig/Loader/Chain.php diff --git a/src/lib/Twig/Loader/Filesystem.php b/src/php/lib/Twig/Loader/Filesystem.php similarity index 99% rename from src/lib/Twig/Loader/Filesystem.php rename to src/php/lib/Twig/Loader/Filesystem.php index 1275044..9149bc3 100644 --- a/src/lib/Twig/Loader/Filesystem.php +++ b/src/php/lib/Twig/Loader/Filesystem.php @@ -279,7 +279,7 @@ private function isAbsolutePath($file) { return strspn($file, '/\\', 0, 1) || (strlen($file) > 3 && ctype_alpha($file[0]) - && substr($file, 1, 1) === ':' + && ':' === substr($file, 1, 1) && strspn($file, '/\\', 2, 1) ) || null !== parse_url($file, PHP_URL_SCHEME) diff --git a/src/lib/Twig/Loader/String.php b/src/php/lib/Twig/Loader/String.php similarity index 100% rename from src/lib/Twig/Loader/String.php rename to src/php/lib/Twig/Loader/String.php diff --git a/src/lib/Twig/LoaderInterface.php b/src/php/lib/Twig/LoaderInterface.php similarity index 100% rename from src/lib/Twig/LoaderInterface.php rename to src/php/lib/Twig/LoaderInterface.php diff --git a/src/lib/Twig/Markup.php b/src/php/lib/Twig/Markup.php similarity index 100% rename from src/lib/Twig/Markup.php rename to src/php/lib/Twig/Markup.php diff --git a/src/lib/Twig/Node.php b/src/php/lib/Twig/Node.php similarity index 100% rename from src/lib/Twig/Node.php rename to src/php/lib/Twig/Node.php diff --git a/src/lib/Twig/Node/AutoEscape.php b/src/php/lib/Twig/Node/AutoEscape.php similarity index 100% rename from src/lib/Twig/Node/AutoEscape.php rename to src/php/lib/Twig/Node/AutoEscape.php diff --git a/src/lib/Twig/Node/Block.php b/src/php/lib/Twig/Node/Block.php similarity index 100% rename from src/lib/Twig/Node/Block.php rename to src/php/lib/Twig/Node/Block.php diff --git a/src/lib/Twig/Node/BlockReference.php b/src/php/lib/Twig/Node/BlockReference.php similarity index 100% rename from src/lib/Twig/Node/BlockReference.php rename to src/php/lib/Twig/Node/BlockReference.php diff --git a/src/lib/Twig/Node/Body.php b/src/php/lib/Twig/Node/Body.php similarity index 100% rename from src/lib/Twig/Node/Body.php rename to src/php/lib/Twig/Node/Body.php diff --git a/src/lib/Twig/Node/CheckSecurity.php b/src/php/lib/Twig/Node/CheckSecurity.php similarity index 100% rename from src/lib/Twig/Node/CheckSecurity.php rename to src/php/lib/Twig/Node/CheckSecurity.php diff --git a/src/lib/Twig/Node/Do.php b/src/php/lib/Twig/Node/Do.php similarity index 100% rename from src/lib/Twig/Node/Do.php rename to src/php/lib/Twig/Node/Do.php diff --git a/src/lib/Twig/Node/Embed.php b/src/php/lib/Twig/Node/Embed.php similarity index 100% rename from src/lib/Twig/Node/Embed.php rename to src/php/lib/Twig/Node/Embed.php diff --git a/src/lib/Twig/Node/Expression.php b/src/php/lib/Twig/Node/Expression.php similarity index 100% rename from src/lib/Twig/Node/Expression.php rename to src/php/lib/Twig/Node/Expression.php diff --git a/src/lib/Twig/Node/Expression/Array.php b/src/php/lib/Twig/Node/Expression/Array.php similarity index 100% rename from src/lib/Twig/Node/Expression/Array.php rename to src/php/lib/Twig/Node/Expression/Array.php diff --git a/src/lib/Twig/Node/Expression/AssignName.php b/src/php/lib/Twig/Node/Expression/AssignName.php similarity index 100% rename from src/lib/Twig/Node/Expression/AssignName.php rename to src/php/lib/Twig/Node/Expression/AssignName.php diff --git a/src/lib/Twig/Node/Expression/Binary.php b/src/php/lib/Twig/Node/Expression/Binary.php similarity index 100% rename from src/lib/Twig/Node/Expression/Binary.php rename to src/php/lib/Twig/Node/Expression/Binary.php diff --git a/src/lib/Twig/Node/Expression/Binary/Add.php b/src/php/lib/Twig/Node/Expression/Binary/Add.php similarity index 100% rename from src/lib/Twig/Node/Expression/Binary/Add.php rename to src/php/lib/Twig/Node/Expression/Binary/Add.php diff --git a/src/lib/Twig/Node/Expression/Binary/And.php b/src/php/lib/Twig/Node/Expression/Binary/And.php similarity index 100% rename from src/lib/Twig/Node/Expression/Binary/And.php rename to src/php/lib/Twig/Node/Expression/Binary/And.php diff --git a/src/lib/Twig/Node/Expression/Binary/BitwiseAnd.php b/src/php/lib/Twig/Node/Expression/Binary/BitwiseAnd.php similarity index 100% rename from src/lib/Twig/Node/Expression/Binary/BitwiseAnd.php rename to src/php/lib/Twig/Node/Expression/Binary/BitwiseAnd.php diff --git a/src/lib/Twig/Node/Expression/Binary/BitwiseOr.php b/src/php/lib/Twig/Node/Expression/Binary/BitwiseOr.php similarity index 100% rename from src/lib/Twig/Node/Expression/Binary/BitwiseOr.php rename to src/php/lib/Twig/Node/Expression/Binary/BitwiseOr.php diff --git a/src/lib/Twig/Node/Expression/Binary/BitwiseXor.php b/src/php/lib/Twig/Node/Expression/Binary/BitwiseXor.php similarity index 100% rename from src/lib/Twig/Node/Expression/Binary/BitwiseXor.php rename to src/php/lib/Twig/Node/Expression/Binary/BitwiseXor.php diff --git a/src/lib/Twig/Node/Expression/Binary/Concat.php b/src/php/lib/Twig/Node/Expression/Binary/Concat.php similarity index 100% rename from src/lib/Twig/Node/Expression/Binary/Concat.php rename to src/php/lib/Twig/Node/Expression/Binary/Concat.php diff --git a/src/lib/Twig/Node/Expression/Binary/Div.php b/src/php/lib/Twig/Node/Expression/Binary/Div.php similarity index 100% rename from src/lib/Twig/Node/Expression/Binary/Div.php rename to src/php/lib/Twig/Node/Expression/Binary/Div.php diff --git a/src/lib/Twig/Node/Expression/Binary/EndsWith.php b/src/php/lib/Twig/Node/Expression/Binary/EndsWith.php similarity index 100% rename from src/lib/Twig/Node/Expression/Binary/EndsWith.php rename to src/php/lib/Twig/Node/Expression/Binary/EndsWith.php diff --git a/src/lib/Twig/Node/Expression/Binary/Equal.php b/src/php/lib/Twig/Node/Expression/Binary/Equal.php similarity index 100% rename from src/lib/Twig/Node/Expression/Binary/Equal.php rename to src/php/lib/Twig/Node/Expression/Binary/Equal.php diff --git a/src/lib/Twig/Node/Expression/Binary/FloorDiv.php b/src/php/lib/Twig/Node/Expression/Binary/FloorDiv.php similarity index 100% rename from src/lib/Twig/Node/Expression/Binary/FloorDiv.php rename to src/php/lib/Twig/Node/Expression/Binary/FloorDiv.php diff --git a/src/lib/Twig/Node/Expression/Binary/Greater.php b/src/php/lib/Twig/Node/Expression/Binary/Greater.php similarity index 100% rename from src/lib/Twig/Node/Expression/Binary/Greater.php rename to src/php/lib/Twig/Node/Expression/Binary/Greater.php diff --git a/src/lib/Twig/Node/Expression/Binary/GreaterEqual.php b/src/php/lib/Twig/Node/Expression/Binary/GreaterEqual.php similarity index 100% rename from src/lib/Twig/Node/Expression/Binary/GreaterEqual.php rename to src/php/lib/Twig/Node/Expression/Binary/GreaterEqual.php diff --git a/src/lib/Twig/Node/Expression/Binary/In.php b/src/php/lib/Twig/Node/Expression/Binary/In.php similarity index 100% rename from src/lib/Twig/Node/Expression/Binary/In.php rename to src/php/lib/Twig/Node/Expression/Binary/In.php diff --git a/src/lib/Twig/Node/Expression/Binary/Less.php b/src/php/lib/Twig/Node/Expression/Binary/Less.php similarity index 100% rename from src/lib/Twig/Node/Expression/Binary/Less.php rename to src/php/lib/Twig/Node/Expression/Binary/Less.php diff --git a/src/lib/Twig/Node/Expression/Binary/LessEqual.php b/src/php/lib/Twig/Node/Expression/Binary/LessEqual.php similarity index 100% rename from src/lib/Twig/Node/Expression/Binary/LessEqual.php rename to src/php/lib/Twig/Node/Expression/Binary/LessEqual.php diff --git a/src/lib/Twig/Node/Expression/Binary/Matches.php b/src/php/lib/Twig/Node/Expression/Binary/Matches.php similarity index 100% rename from src/lib/Twig/Node/Expression/Binary/Matches.php rename to src/php/lib/Twig/Node/Expression/Binary/Matches.php diff --git a/src/lib/Twig/Node/Expression/Binary/Mod.php b/src/php/lib/Twig/Node/Expression/Binary/Mod.php similarity index 100% rename from src/lib/Twig/Node/Expression/Binary/Mod.php rename to src/php/lib/Twig/Node/Expression/Binary/Mod.php diff --git a/src/lib/Twig/Node/Expression/Binary/Mul.php b/src/php/lib/Twig/Node/Expression/Binary/Mul.php similarity index 100% rename from src/lib/Twig/Node/Expression/Binary/Mul.php rename to src/php/lib/Twig/Node/Expression/Binary/Mul.php diff --git a/src/lib/Twig/Node/Expression/Binary/NotEqual.php b/src/php/lib/Twig/Node/Expression/Binary/NotEqual.php similarity index 100% rename from src/lib/Twig/Node/Expression/Binary/NotEqual.php rename to src/php/lib/Twig/Node/Expression/Binary/NotEqual.php diff --git a/src/lib/Twig/Node/Expression/Binary/NotIn.php b/src/php/lib/Twig/Node/Expression/Binary/NotIn.php similarity index 100% rename from src/lib/Twig/Node/Expression/Binary/NotIn.php rename to src/php/lib/Twig/Node/Expression/Binary/NotIn.php diff --git a/src/lib/Twig/Node/Expression/Binary/Or.php b/src/php/lib/Twig/Node/Expression/Binary/Or.php similarity index 100% rename from src/lib/Twig/Node/Expression/Binary/Or.php rename to src/php/lib/Twig/Node/Expression/Binary/Or.php diff --git a/src/lib/Twig/Node/Expression/Binary/Power.php b/src/php/lib/Twig/Node/Expression/Binary/Power.php similarity index 100% rename from src/lib/Twig/Node/Expression/Binary/Power.php rename to src/php/lib/Twig/Node/Expression/Binary/Power.php diff --git a/src/lib/Twig/Node/Expression/Binary/Range.php b/src/php/lib/Twig/Node/Expression/Binary/Range.php similarity index 100% rename from src/lib/Twig/Node/Expression/Binary/Range.php rename to src/php/lib/Twig/Node/Expression/Binary/Range.php diff --git a/src/lib/Twig/Node/Expression/Binary/StartsWith.php b/src/php/lib/Twig/Node/Expression/Binary/StartsWith.php similarity index 100% rename from src/lib/Twig/Node/Expression/Binary/StartsWith.php rename to src/php/lib/Twig/Node/Expression/Binary/StartsWith.php diff --git a/src/lib/Twig/Node/Expression/Binary/Sub.php b/src/php/lib/Twig/Node/Expression/Binary/Sub.php similarity index 100% rename from src/lib/Twig/Node/Expression/Binary/Sub.php rename to src/php/lib/Twig/Node/Expression/Binary/Sub.php diff --git a/src/lib/Twig/Node/Expression/BlockReference.php b/src/php/lib/Twig/Node/Expression/BlockReference.php similarity index 100% rename from src/lib/Twig/Node/Expression/BlockReference.php rename to src/php/lib/Twig/Node/Expression/BlockReference.php diff --git a/src/lib/Twig/Node/Expression/Call.php b/src/php/lib/Twig/Node/Expression/Call.php similarity index 100% rename from src/lib/Twig/Node/Expression/Call.php rename to src/php/lib/Twig/Node/Expression/Call.php diff --git a/src/lib/Twig/Node/Expression/Conditional.php b/src/php/lib/Twig/Node/Expression/Conditional.php similarity index 100% rename from src/lib/Twig/Node/Expression/Conditional.php rename to src/php/lib/Twig/Node/Expression/Conditional.php diff --git a/src/lib/Twig/Node/Expression/Constant.php b/src/php/lib/Twig/Node/Expression/Constant.php similarity index 100% rename from src/lib/Twig/Node/Expression/Constant.php rename to src/php/lib/Twig/Node/Expression/Constant.php diff --git a/src/lib/Twig/Node/Expression/ExtensionReference.php b/src/php/lib/Twig/Node/Expression/ExtensionReference.php similarity index 100% rename from src/lib/Twig/Node/Expression/ExtensionReference.php rename to src/php/lib/Twig/Node/Expression/ExtensionReference.php diff --git a/src/lib/Twig/Node/Expression/Filter.php b/src/php/lib/Twig/Node/Expression/Filter.php similarity index 100% rename from src/lib/Twig/Node/Expression/Filter.php rename to src/php/lib/Twig/Node/Expression/Filter.php diff --git a/src/lib/Twig/Node/Expression/Filter/Default.php b/src/php/lib/Twig/Node/Expression/Filter/Default.php similarity index 100% rename from src/lib/Twig/Node/Expression/Filter/Default.php rename to src/php/lib/Twig/Node/Expression/Filter/Default.php diff --git a/src/lib/Twig/Node/Expression/Function.php b/src/php/lib/Twig/Node/Expression/Function.php similarity index 100% rename from src/lib/Twig/Node/Expression/Function.php rename to src/php/lib/Twig/Node/Expression/Function.php diff --git a/src/lib/Twig/Node/Expression/GetAttr.php b/src/php/lib/Twig/Node/Expression/GetAttr.php similarity index 100% rename from src/lib/Twig/Node/Expression/GetAttr.php rename to src/php/lib/Twig/Node/Expression/GetAttr.php diff --git a/src/lib/Twig/Node/Expression/MethodCall.php b/src/php/lib/Twig/Node/Expression/MethodCall.php similarity index 100% rename from src/lib/Twig/Node/Expression/MethodCall.php rename to src/php/lib/Twig/Node/Expression/MethodCall.php diff --git a/src/lib/Twig/Node/Expression/Name.php b/src/php/lib/Twig/Node/Expression/Name.php similarity index 100% rename from src/lib/Twig/Node/Expression/Name.php rename to src/php/lib/Twig/Node/Expression/Name.php diff --git a/src/lib/Twig/Node/Expression/NullCoalesce.php b/src/php/lib/Twig/Node/Expression/NullCoalesce.php similarity index 100% rename from src/lib/Twig/Node/Expression/NullCoalesce.php rename to src/php/lib/Twig/Node/Expression/NullCoalesce.php diff --git a/src/lib/Twig/Node/Expression/Parent.php b/src/php/lib/Twig/Node/Expression/Parent.php similarity index 100% rename from src/lib/Twig/Node/Expression/Parent.php rename to src/php/lib/Twig/Node/Expression/Parent.php diff --git a/src/lib/Twig/Node/Expression/TempName.php b/src/php/lib/Twig/Node/Expression/TempName.php similarity index 100% rename from src/lib/Twig/Node/Expression/TempName.php rename to src/php/lib/Twig/Node/Expression/TempName.php diff --git a/src/lib/Twig/Node/Expression/Test.php b/src/php/lib/Twig/Node/Expression/Test.php similarity index 100% rename from src/lib/Twig/Node/Expression/Test.php rename to src/php/lib/Twig/Node/Expression/Test.php diff --git a/src/lib/Twig/Node/Expression/Test/Constant.php b/src/php/lib/Twig/Node/Expression/Test/Constant.php similarity index 100% rename from src/lib/Twig/Node/Expression/Test/Constant.php rename to src/php/lib/Twig/Node/Expression/Test/Constant.php diff --git a/src/lib/Twig/Node/Expression/Test/Defined.php b/src/php/lib/Twig/Node/Expression/Test/Defined.php similarity index 100% rename from src/lib/Twig/Node/Expression/Test/Defined.php rename to src/php/lib/Twig/Node/Expression/Test/Defined.php diff --git a/src/lib/Twig/Node/Expression/Test/Divisibleby.php b/src/php/lib/Twig/Node/Expression/Test/Divisibleby.php similarity index 100% rename from src/lib/Twig/Node/Expression/Test/Divisibleby.php rename to src/php/lib/Twig/Node/Expression/Test/Divisibleby.php diff --git a/src/lib/Twig/Node/Expression/Test/Even.php b/src/php/lib/Twig/Node/Expression/Test/Even.php similarity index 100% rename from src/lib/Twig/Node/Expression/Test/Even.php rename to src/php/lib/Twig/Node/Expression/Test/Even.php diff --git a/src/lib/Twig/Node/Expression/Test/Null.php b/src/php/lib/Twig/Node/Expression/Test/Null.php similarity index 100% rename from src/lib/Twig/Node/Expression/Test/Null.php rename to src/php/lib/Twig/Node/Expression/Test/Null.php diff --git a/src/lib/Twig/Node/Expression/Test/Odd.php b/src/php/lib/Twig/Node/Expression/Test/Odd.php similarity index 100% rename from src/lib/Twig/Node/Expression/Test/Odd.php rename to src/php/lib/Twig/Node/Expression/Test/Odd.php diff --git a/src/lib/Twig/Node/Expression/Test/Sameas.php b/src/php/lib/Twig/Node/Expression/Test/Sameas.php similarity index 100% rename from src/lib/Twig/Node/Expression/Test/Sameas.php rename to src/php/lib/Twig/Node/Expression/Test/Sameas.php diff --git a/src/lib/Twig/Node/Expression/Unary.php b/src/php/lib/Twig/Node/Expression/Unary.php similarity index 100% rename from src/lib/Twig/Node/Expression/Unary.php rename to src/php/lib/Twig/Node/Expression/Unary.php diff --git a/src/lib/Twig/Node/Expression/Unary/Neg.php b/src/php/lib/Twig/Node/Expression/Unary/Neg.php similarity index 100% rename from src/lib/Twig/Node/Expression/Unary/Neg.php rename to src/php/lib/Twig/Node/Expression/Unary/Neg.php diff --git a/src/lib/Twig/Node/Expression/Unary/Not.php b/src/php/lib/Twig/Node/Expression/Unary/Not.php similarity index 100% rename from src/lib/Twig/Node/Expression/Unary/Not.php rename to src/php/lib/Twig/Node/Expression/Unary/Not.php diff --git a/src/lib/Twig/Node/Expression/Unary/Pos.php b/src/php/lib/Twig/Node/Expression/Unary/Pos.php similarity index 100% rename from src/lib/Twig/Node/Expression/Unary/Pos.php rename to src/php/lib/Twig/Node/Expression/Unary/Pos.php diff --git a/src/lib/Twig/Node/Flush.php b/src/php/lib/Twig/Node/Flush.php similarity index 100% rename from src/lib/Twig/Node/Flush.php rename to src/php/lib/Twig/Node/Flush.php diff --git a/src/lib/Twig/Node/For.php b/src/php/lib/Twig/Node/For.php similarity index 100% rename from src/lib/Twig/Node/For.php rename to src/php/lib/Twig/Node/For.php diff --git a/src/lib/Twig/Node/ForLoop.php b/src/php/lib/Twig/Node/ForLoop.php similarity index 100% rename from src/lib/Twig/Node/ForLoop.php rename to src/php/lib/Twig/Node/ForLoop.php diff --git a/src/lib/Twig/Node/If.php b/src/php/lib/Twig/Node/If.php similarity index 100% rename from src/lib/Twig/Node/If.php rename to src/php/lib/Twig/Node/If.php diff --git a/src/lib/Twig/Node/Import.php b/src/php/lib/Twig/Node/Import.php similarity index 100% rename from src/lib/Twig/Node/Import.php rename to src/php/lib/Twig/Node/Import.php diff --git a/src/lib/Twig/Node/Include.php b/src/php/lib/Twig/Node/Include.php similarity index 100% rename from src/lib/Twig/Node/Include.php rename to src/php/lib/Twig/Node/Include.php diff --git a/src/lib/Twig/Node/Macro.php b/src/php/lib/Twig/Node/Macro.php similarity index 100% rename from src/lib/Twig/Node/Macro.php rename to src/php/lib/Twig/Node/Macro.php diff --git a/src/lib/Twig/Node/Module.php b/src/php/lib/Twig/Node/Module.php similarity index 100% rename from src/lib/Twig/Node/Module.php rename to src/php/lib/Twig/Node/Module.php diff --git a/src/lib/Twig/Node/Print.php b/src/php/lib/Twig/Node/Print.php similarity index 100% rename from src/lib/Twig/Node/Print.php rename to src/php/lib/Twig/Node/Print.php diff --git a/src/lib/Twig/Node/Sandbox.php b/src/php/lib/Twig/Node/Sandbox.php similarity index 100% rename from src/lib/Twig/Node/Sandbox.php rename to src/php/lib/Twig/Node/Sandbox.php diff --git a/src/lib/Twig/Node/SandboxedPrint.php b/src/php/lib/Twig/Node/SandboxedPrint.php similarity index 100% rename from src/lib/Twig/Node/SandboxedPrint.php rename to src/php/lib/Twig/Node/SandboxedPrint.php diff --git a/src/lib/Twig/Node/Set.php b/src/php/lib/Twig/Node/Set.php similarity index 100% rename from src/lib/Twig/Node/Set.php rename to src/php/lib/Twig/Node/Set.php diff --git a/src/lib/Twig/Node/SetTemp.php b/src/php/lib/Twig/Node/SetTemp.php similarity index 100% rename from src/lib/Twig/Node/SetTemp.php rename to src/php/lib/Twig/Node/SetTemp.php diff --git a/src/lib/Twig/Node/Spaceless.php b/src/php/lib/Twig/Node/Spaceless.php similarity index 100% rename from src/lib/Twig/Node/Spaceless.php rename to src/php/lib/Twig/Node/Spaceless.php diff --git a/src/lib/Twig/Node/Text.php b/src/php/lib/Twig/Node/Text.php similarity index 100% rename from src/lib/Twig/Node/Text.php rename to src/php/lib/Twig/Node/Text.php diff --git a/src/lib/Twig/Node/With.php b/src/php/lib/Twig/Node/With.php similarity index 100% rename from src/lib/Twig/Node/With.php rename to src/php/lib/Twig/Node/With.php diff --git a/src/lib/Twig/NodeCaptureInterface.php b/src/php/lib/Twig/NodeCaptureInterface.php similarity index 100% rename from src/lib/Twig/NodeCaptureInterface.php rename to src/php/lib/Twig/NodeCaptureInterface.php diff --git a/src/lib/Twig/NodeInterface.php b/src/php/lib/Twig/NodeInterface.php similarity index 100% rename from src/lib/Twig/NodeInterface.php rename to src/php/lib/Twig/NodeInterface.php diff --git a/src/lib/Twig/NodeOutputInterface.php b/src/php/lib/Twig/NodeOutputInterface.php similarity index 100% rename from src/lib/Twig/NodeOutputInterface.php rename to src/php/lib/Twig/NodeOutputInterface.php diff --git a/src/lib/Twig/NodeTraverser.php b/src/php/lib/Twig/NodeTraverser.php similarity index 100% rename from src/lib/Twig/NodeTraverser.php rename to src/php/lib/Twig/NodeTraverser.php diff --git a/src/lib/Twig/NodeVisitor/Escaper.php b/src/php/lib/Twig/NodeVisitor/Escaper.php similarity index 100% rename from src/lib/Twig/NodeVisitor/Escaper.php rename to src/php/lib/Twig/NodeVisitor/Escaper.php diff --git a/src/lib/Twig/NodeVisitor/Optimizer.php b/src/php/lib/Twig/NodeVisitor/Optimizer.php similarity index 97% rename from src/lib/Twig/NodeVisitor/Optimizer.php rename to src/php/lib/Twig/NodeVisitor/Optimizer.php index 2fa19f3..c55e40f 100644 --- a/src/lib/Twig/NodeVisitor/Optimizer.php +++ b/src/php/lib/Twig/NodeVisitor/Optimizer.php @@ -56,7 +56,7 @@ protected function doEnterNode(Twig_Node $node, Twig_Environment $env) if (PHP_VERSION_ID < 50400 && self::OPTIMIZE_VAR_ACCESS === (self::OPTIMIZE_VAR_ACCESS & $this->optimizers) && !$env->isStrictVariables() && !$env->hasExtension('Twig_Extension_Sandbox')) { if ($this->inABody) { if (!$node instanceof Twig_Node_Expression) { - if (get_class($node) !== 'Twig_Node') { + if ('Twig_Node' !== get_class($node)) { array_unshift($this->prependedNodes, array()); } } else { @@ -88,7 +88,7 @@ protected function doLeaveNode(Twig_Node $node, Twig_Environment $env) if ($node instanceof Twig_Node_Body) { $this->inABody = false; } elseif ($this->inABody) { - if (!$expression && get_class($node) !== 'Twig_Node' && $prependedNodes = array_shift($this->prependedNodes)) { + if (!$expression && 'Twig_Node' !== get_class($node) && $prependedNodes = array_shift($this->prependedNodes)) { $nodes = array(); foreach (array_unique($prependedNodes) as $name) { $nodes[] = new Twig_Node_SetTemp($name, $node->getTemplateLine()); diff --git a/src/lib/Twig/NodeVisitor/SafeAnalysis.php b/src/php/lib/Twig/NodeVisitor/SafeAnalysis.php similarity index 100% rename from src/lib/Twig/NodeVisitor/SafeAnalysis.php rename to src/php/lib/Twig/NodeVisitor/SafeAnalysis.php diff --git a/src/lib/Twig/NodeVisitor/Sandbox.php b/src/php/lib/Twig/NodeVisitor/Sandbox.php similarity index 90% rename from src/lib/Twig/NodeVisitor/Sandbox.php rename to src/php/lib/Twig/NodeVisitor/Sandbox.php index b631b29..71aa4f0 100644 --- a/src/lib/Twig/NodeVisitor/Sandbox.php +++ b/src/php/lib/Twig/NodeVisitor/Sandbox.php @@ -48,6 +48,11 @@ protected function doEnterNode(Twig_Node $node, Twig_Environment $env) $this->functions[$node->getAttribute('name')] = $node; } + // the .. operator is equivalent to the range() function + if ($node instanceof Twig_Node_Expression_Binary_Range && !isset($this->functions['range'])) { + $this->functions['range'] = $node; + } + // wrap print to check __toString() calls if ($node instanceof Twig_Node_Print) { return new Twig_Node_SandboxedPrint($node->getNode('expr'), $node->getTemplateLine(), $node->getNodeTag()); diff --git a/src/lib/Twig/NodeVisitorInterface.php b/src/php/lib/Twig/NodeVisitorInterface.php similarity index 100% rename from src/lib/Twig/NodeVisitorInterface.php rename to src/php/lib/Twig/NodeVisitorInterface.php diff --git a/src/lib/Twig/Parser.php b/src/php/lib/Twig/Parser.php similarity index 96% rename from src/lib/Twig/Parser.php rename to src/php/lib/Twig/Parser.php index e796f19..6de879a 100644 --- a/src/lib/Twig/Parser.php +++ b/src/php/lib/Twig/Parser.php @@ -31,6 +31,7 @@ class Twig_Parser implements Twig_ParserInterface protected $importedSymbols; protected $traits; protected $embeddedTemplates = array(); + private $varNameSalt = 0; public function __construct(Twig_Environment $env) { @@ -49,7 +50,7 @@ public function getEnvironment() public function getVarName() { - return sprintf('__internal_%s', hash('sha256', uniqid(mt_rand(), true), false)); + return sprintf('__internal_%s', hash('sha256', __METHOD__.$this->stream->getSourceContext()->getCode().$this->varNameSalt++)); } /** @@ -98,6 +99,7 @@ public function parse(Twig_TokenStream $stream, $test = null, $dropNeedle = fals $this->blockStack = array(); $this->importedSymbols = array(array()); $this->embeddedTemplates = array(); + $this->varNameSalt = 0; try { $body = $this->subparse($test, $dropNeedle); @@ -153,7 +155,7 @@ public function subparse($test, $dropNeedle = false) $this->stream->next(); $token = $this->getCurrentToken(); - if ($token->getType() !== Twig_Token::NAME_TYPE) { + if (Twig_Token::NAME_TYPE !== $token->getType()) { throw new Twig_Error_Syntax('A block must start with a tag name.', $token->getLine(), $this->stream->getSourceContext()); } @@ -383,7 +385,7 @@ protected function filterBodyNodes(Twig_NodeInterface $node) throw new Twig_Error_Syntax('A template that extends another one cannot start with a byte order mark (BOM); it must be removed.', $node->getTemplateLine(), $this->stream->getSourceContext()); } - throw new Twig_Error_Syntax('A template that extends another one cannot include contents outside Twig blocks. Did you forget to put the contents inside a {% block %} tag?', $node->getTemplateLine(), $this->stream->getSourceContext()); + throw new Twig_Error_Syntax('A template that extends another one cannot include content outside Twig blocks. Did you forget to put the content inside a {% block %} tag?', $node->getTemplateLine(), $this->stream->getSourceContext()); } // bypass nodes that will "capture" the output diff --git a/src/lib/Twig/ParserInterface.php b/src/php/lib/Twig/ParserInterface.php similarity index 100% rename from src/lib/Twig/ParserInterface.php rename to src/php/lib/Twig/ParserInterface.php diff --git a/src/lib/Twig/Profiler/Dumper/Base.php b/src/php/lib/Twig/Profiler/Dumper/Base.php similarity index 100% rename from src/lib/Twig/Profiler/Dumper/Base.php rename to src/php/lib/Twig/Profiler/Dumper/Base.php diff --git a/src/lib/Twig/Profiler/Dumper/Blackfire.php b/src/php/lib/Twig/Profiler/Dumper/Blackfire.php similarity index 100% rename from src/lib/Twig/Profiler/Dumper/Blackfire.php rename to src/php/lib/Twig/Profiler/Dumper/Blackfire.php diff --git a/src/lib/Twig/Profiler/Dumper/Html.php b/src/php/lib/Twig/Profiler/Dumper/Html.php similarity index 100% rename from src/lib/Twig/Profiler/Dumper/Html.php rename to src/php/lib/Twig/Profiler/Dumper/Html.php diff --git a/src/lib/Twig/Profiler/Dumper/Text.php b/src/php/lib/Twig/Profiler/Dumper/Text.php similarity index 100% rename from src/lib/Twig/Profiler/Dumper/Text.php rename to src/php/lib/Twig/Profiler/Dumper/Text.php diff --git a/src/lib/Twig/Profiler/Node/EnterProfile.php b/src/php/lib/Twig/Profiler/Node/EnterProfile.php similarity index 100% rename from src/lib/Twig/Profiler/Node/EnterProfile.php rename to src/php/lib/Twig/Profiler/Node/EnterProfile.php diff --git a/src/lib/Twig/Profiler/Node/LeaveProfile.php b/src/php/lib/Twig/Profiler/Node/LeaveProfile.php similarity index 100% rename from src/lib/Twig/Profiler/Node/LeaveProfile.php rename to src/php/lib/Twig/Profiler/Node/LeaveProfile.php diff --git a/src/lib/Twig/Profiler/NodeVisitor/Profiler.php b/src/php/lib/Twig/Profiler/NodeVisitor/Profiler.php similarity index 96% rename from src/lib/Twig/Profiler/NodeVisitor/Profiler.php rename to src/php/lib/Twig/Profiler/NodeVisitor/Profiler.php index a395ae7..5db41fe 100644 --- a/src/lib/Twig/Profiler/NodeVisitor/Profiler.php +++ b/src/php/lib/Twig/Profiler/NodeVisitor/Profiler.php @@ -55,7 +55,7 @@ protected function doLeaveNode(Twig_Node $node, Twig_Environment $env) private function getVarName() { - return sprintf('__internal_%s', hash('sha256', uniqid(mt_rand(), true), false)); + return sprintf('__internal_%s', hash('sha256', $this->extensionName)); } public function getPriority() diff --git a/src/lib/Twig/Profiler/Profile.php b/src/php/lib/Twig/Profiler/Profile.php similarity index 100% rename from src/lib/Twig/Profiler/Profile.php rename to src/php/lib/Twig/Profiler/Profile.php diff --git a/src/lib/Twig/RuntimeLoaderInterface.php b/src/php/lib/Twig/RuntimeLoaderInterface.php similarity index 100% rename from src/lib/Twig/RuntimeLoaderInterface.php rename to src/php/lib/Twig/RuntimeLoaderInterface.php diff --git a/src/lib/Twig/Sandbox/SecurityError.php b/src/php/lib/Twig/Sandbox/SecurityError.php similarity index 100% rename from src/lib/Twig/Sandbox/SecurityError.php rename to src/php/lib/Twig/Sandbox/SecurityError.php diff --git a/src/lib/Twig/Sandbox/SecurityNotAllowedFilterError.php b/src/php/lib/Twig/Sandbox/SecurityNotAllowedFilterError.php similarity index 100% rename from src/lib/Twig/Sandbox/SecurityNotAllowedFilterError.php rename to src/php/lib/Twig/Sandbox/SecurityNotAllowedFilterError.php diff --git a/src/lib/Twig/Sandbox/SecurityNotAllowedFunctionError.php b/src/php/lib/Twig/Sandbox/SecurityNotAllowedFunctionError.php similarity index 100% rename from src/lib/Twig/Sandbox/SecurityNotAllowedFunctionError.php rename to src/php/lib/Twig/Sandbox/SecurityNotAllowedFunctionError.php diff --git a/src/lib/Twig/Sandbox/SecurityNotAllowedMethodError.php b/src/php/lib/Twig/Sandbox/SecurityNotAllowedMethodError.php similarity index 100% rename from src/lib/Twig/Sandbox/SecurityNotAllowedMethodError.php rename to src/php/lib/Twig/Sandbox/SecurityNotAllowedMethodError.php diff --git a/src/lib/Twig/Sandbox/SecurityNotAllowedPropertyError.php b/src/php/lib/Twig/Sandbox/SecurityNotAllowedPropertyError.php similarity index 100% rename from src/lib/Twig/Sandbox/SecurityNotAllowedPropertyError.php rename to src/php/lib/Twig/Sandbox/SecurityNotAllowedPropertyError.php diff --git a/src/lib/Twig/Sandbox/SecurityNotAllowedTagError.php b/src/php/lib/Twig/Sandbox/SecurityNotAllowedTagError.php similarity index 100% rename from src/lib/Twig/Sandbox/SecurityNotAllowedTagError.php rename to src/php/lib/Twig/Sandbox/SecurityNotAllowedTagError.php diff --git a/src/lib/Twig/Sandbox/SecurityPolicy.php b/src/php/lib/Twig/Sandbox/SecurityPolicy.php similarity index 100% rename from src/lib/Twig/Sandbox/SecurityPolicy.php rename to src/php/lib/Twig/Sandbox/SecurityPolicy.php diff --git a/src/lib/Twig/Sandbox/SecurityPolicyInterface.php b/src/php/lib/Twig/Sandbox/SecurityPolicyInterface.php similarity index 100% rename from src/lib/Twig/Sandbox/SecurityPolicyInterface.php rename to src/php/lib/Twig/Sandbox/SecurityPolicyInterface.php diff --git a/src/lib/Twig/SimpleFilter.php b/src/php/lib/Twig/SimpleFilter.php similarity index 100% rename from src/lib/Twig/SimpleFilter.php rename to src/php/lib/Twig/SimpleFilter.php diff --git a/src/lib/Twig/SimpleFunction.php b/src/php/lib/Twig/SimpleFunction.php similarity index 100% rename from src/lib/Twig/SimpleFunction.php rename to src/php/lib/Twig/SimpleFunction.php diff --git a/src/lib/Twig/SimpleTest.php b/src/php/lib/Twig/SimpleTest.php similarity index 100% rename from src/lib/Twig/SimpleTest.php rename to src/php/lib/Twig/SimpleTest.php diff --git a/src/lib/Twig/Source.php b/src/php/lib/Twig/Source.php similarity index 100% rename from src/lib/Twig/Source.php rename to src/php/lib/Twig/Source.php diff --git a/src/lib/Twig/SourceContextLoaderInterface.php b/src/php/lib/Twig/SourceContextLoaderInterface.php similarity index 100% rename from src/lib/Twig/SourceContextLoaderInterface.php rename to src/php/lib/Twig/SourceContextLoaderInterface.php diff --git a/src/lib/Twig/Template.php b/src/php/lib/Twig/Template.php similarity index 99% rename from src/lib/Twig/Template.php rename to src/php/lib/Twig/Template.php index 6456341..3709232 100644 --- a/src/lib/Twig/Template.php +++ b/src/php/lib/Twig/Template.php @@ -568,6 +568,8 @@ protected function getAttribute($object, $item, array $arguments = array(), $typ if (null === $object) { $message = sprintf('Impossible to invoke a method ("%s") on a null variable.', $item); + } elseif (is_array($object)) { + $message = sprintf('Impossible to invoke a method ("%s") on an array.', $item); } else { $message = sprintf('Impossible to invoke a method ("%s") on a %s variable ("%s").', $item, gettype($object), $object); } @@ -696,7 +698,7 @@ protected function getAttribute($object, $item, array $arguments = array(), $typ } @trigger_error($message, E_USER_DEPRECATED); - return $ret === '' ? '' : new Twig_Markup($ret, $this->env->getCharset()); + return '' === $ret ? '' : new Twig_Markup($ret, $this->env->getCharset()); } return $ret; diff --git a/src/lib/Twig/TemplateInterface.php b/src/php/lib/Twig/TemplateInterface.php similarity index 100% rename from src/lib/Twig/TemplateInterface.php rename to src/php/lib/Twig/TemplateInterface.php diff --git a/src/lib/Twig/TemplateWrapper.php b/src/php/lib/Twig/TemplateWrapper.php similarity index 100% rename from src/lib/Twig/TemplateWrapper.php rename to src/php/lib/Twig/TemplateWrapper.php diff --git a/src/lib/Twig/Test.php b/src/php/lib/Twig/Test.php similarity index 100% rename from src/lib/Twig/Test.php rename to src/php/lib/Twig/Test.php diff --git a/src/lib/Twig/Test/Function.php b/src/php/lib/Twig/Test/Function.php similarity index 100% rename from src/lib/Twig/Test/Function.php rename to src/php/lib/Twig/Test/Function.php diff --git a/src/lib/Twig/Test/IntegrationTestCase.php b/src/php/lib/Twig/Test/IntegrationTestCase.php similarity index 100% rename from src/lib/Twig/Test/IntegrationTestCase.php rename to src/php/lib/Twig/Test/IntegrationTestCase.php diff --git a/src/lib/Twig/Test/Method.php b/src/php/lib/Twig/Test/Method.php similarity index 100% rename from src/lib/Twig/Test/Method.php rename to src/php/lib/Twig/Test/Method.php diff --git a/src/lib/Twig/Test/Node.php b/src/php/lib/Twig/Test/Node.php similarity index 100% rename from src/lib/Twig/Test/Node.php rename to src/php/lib/Twig/Test/Node.php diff --git a/src/lib/Twig/Test/NodeTestCase.php b/src/php/lib/Twig/Test/NodeTestCase.php similarity index 100% rename from src/lib/Twig/Test/NodeTestCase.php rename to src/php/lib/Twig/Test/NodeTestCase.php diff --git a/src/lib/Twig/TestCallableInterface.php b/src/php/lib/Twig/TestCallableInterface.php similarity index 100% rename from src/lib/Twig/TestCallableInterface.php rename to src/php/lib/Twig/TestCallableInterface.php diff --git a/src/lib/Twig/TestInterface.php b/src/php/lib/Twig/TestInterface.php similarity index 100% rename from src/lib/Twig/TestInterface.php rename to src/php/lib/Twig/TestInterface.php diff --git a/src/lib/Twig/Token.php b/src/php/lib/Twig/Token.php similarity index 100% rename from src/lib/Twig/Token.php rename to src/php/lib/Twig/Token.php diff --git a/src/lib/Twig/TokenParser.php b/src/php/lib/Twig/TokenParser.php similarity index 100% rename from src/lib/Twig/TokenParser.php rename to src/php/lib/Twig/TokenParser.php diff --git a/src/lib/Twig/TokenParser/AutoEscape.php b/src/php/lib/Twig/TokenParser/AutoEscape.php similarity index 100% rename from src/lib/Twig/TokenParser/AutoEscape.php rename to src/php/lib/Twig/TokenParser/AutoEscape.php diff --git a/src/lib/Twig/TokenParser/Block.php b/src/php/lib/Twig/TokenParser/Block.php similarity index 100% rename from src/lib/Twig/TokenParser/Block.php rename to src/php/lib/Twig/TokenParser/Block.php diff --git a/src/lib/Twig/TokenParser/Do.php b/src/php/lib/Twig/TokenParser/Do.php similarity index 100% rename from src/lib/Twig/TokenParser/Do.php rename to src/php/lib/Twig/TokenParser/Do.php diff --git a/src/lib/Twig/TokenParser/Embed.php b/src/php/lib/Twig/TokenParser/Embed.php similarity index 100% rename from src/lib/Twig/TokenParser/Embed.php rename to src/php/lib/Twig/TokenParser/Embed.php diff --git a/src/lib/Twig/TokenParser/Extends.php b/src/php/lib/Twig/TokenParser/Extends.php similarity index 100% rename from src/lib/Twig/TokenParser/Extends.php rename to src/php/lib/Twig/TokenParser/Extends.php diff --git a/src/lib/Twig/TokenParser/Filter.php b/src/php/lib/Twig/TokenParser/Filter.php similarity index 100% rename from src/lib/Twig/TokenParser/Filter.php rename to src/php/lib/Twig/TokenParser/Filter.php diff --git a/src/lib/Twig/TokenParser/Flush.php b/src/php/lib/Twig/TokenParser/Flush.php similarity index 100% rename from src/lib/Twig/TokenParser/Flush.php rename to src/php/lib/Twig/TokenParser/Flush.php diff --git a/src/lib/Twig/TokenParser/For.php b/src/php/lib/Twig/TokenParser/For.php similarity index 98% rename from src/lib/Twig/TokenParser/For.php rename to src/php/lib/Twig/TokenParser/For.php index 63bf41d..8e737c5 100644 --- a/src/lib/Twig/TokenParser/For.php +++ b/src/php/lib/Twig/TokenParser/For.php @@ -40,7 +40,7 @@ public function parse(Twig_Token $token) $stream->expect(Twig_Token::BLOCK_END_TYPE); $body = $this->parser->subparse(array($this, 'decideForFork')); - if ($stream->next()->getValue() == 'else') { + if ('else' == $stream->next()->getValue()) { $stream->expect(Twig_Token::BLOCK_END_TYPE); $else = $this->parser->subparse(array($this, 'decideForEnd'), true); } else { diff --git a/src/lib/Twig/TokenParser/From.php b/src/php/lib/Twig/TokenParser/From.php similarity index 100% rename from src/lib/Twig/TokenParser/From.php rename to src/php/lib/Twig/TokenParser/From.php diff --git a/src/lib/Twig/TokenParser/If.php b/src/php/lib/Twig/TokenParser/If.php similarity index 100% rename from src/lib/Twig/TokenParser/If.php rename to src/php/lib/Twig/TokenParser/If.php diff --git a/src/lib/Twig/TokenParser/Import.php b/src/php/lib/Twig/TokenParser/Import.php similarity index 100% rename from src/lib/Twig/TokenParser/Import.php rename to src/php/lib/Twig/TokenParser/Import.php diff --git a/src/lib/Twig/TokenParser/Include.php b/src/php/lib/Twig/TokenParser/Include.php similarity index 100% rename from src/lib/Twig/TokenParser/Include.php rename to src/php/lib/Twig/TokenParser/Include.php diff --git a/src/lib/Twig/TokenParser/Macro.php b/src/php/lib/Twig/TokenParser/Macro.php similarity index 100% rename from src/lib/Twig/TokenParser/Macro.php rename to src/php/lib/Twig/TokenParser/Macro.php diff --git a/src/lib/Twig/TokenParser/Sandbox.php b/src/php/lib/Twig/TokenParser/Sandbox.php similarity index 100% rename from src/lib/Twig/TokenParser/Sandbox.php rename to src/php/lib/Twig/TokenParser/Sandbox.php diff --git a/src/lib/Twig/TokenParser/Set.php b/src/php/lib/Twig/TokenParser/Set.php similarity index 100% rename from src/lib/Twig/TokenParser/Set.php rename to src/php/lib/Twig/TokenParser/Set.php diff --git a/src/lib/Twig/TokenParser/Spaceless.php b/src/php/lib/Twig/TokenParser/Spaceless.php similarity index 100% rename from src/lib/Twig/TokenParser/Spaceless.php rename to src/php/lib/Twig/TokenParser/Spaceless.php diff --git a/src/lib/Twig/TokenParser/Use.php b/src/php/lib/Twig/TokenParser/Use.php similarity index 100% rename from src/lib/Twig/TokenParser/Use.php rename to src/php/lib/Twig/TokenParser/Use.php diff --git a/src/lib/Twig/TokenParser/With.php b/src/php/lib/Twig/TokenParser/With.php similarity index 100% rename from src/lib/Twig/TokenParser/With.php rename to src/php/lib/Twig/TokenParser/With.php diff --git a/src/lib/Twig/TokenParserBroker.php b/src/php/lib/Twig/TokenParserBroker.php similarity index 95% rename from src/lib/Twig/TokenParserBroker.php rename to src/php/lib/Twig/TokenParserBroker.php index a640135..0d7d6e5 100644 --- a/src/lib/Twig/TokenParserBroker.php +++ b/src/php/lib/Twig/TokenParserBroker.php @@ -61,12 +61,12 @@ public function removeTokenParser(Twig_TokenParserInterface $parser) } } - public function addTokenParserBroker(Twig_TokenParserBroker $broker) + public function addTokenParserBroker(self $broker) { $this->brokers[] = $broker; } - public function removeTokenParserBroker(Twig_TokenParserBroker $broker) + public function removeTokenParserBroker(self $broker) { if (false !== $pos = array_search($broker, $this->brokers)) { unset($this->brokers[$pos]); diff --git a/src/lib/Twig/TokenParserBrokerInterface.php b/src/php/lib/Twig/TokenParserBrokerInterface.php similarity index 100% rename from src/lib/Twig/TokenParserBrokerInterface.php rename to src/php/lib/Twig/TokenParserBrokerInterface.php diff --git a/src/lib/Twig/TokenParserInterface.php b/src/php/lib/Twig/TokenParserInterface.php similarity index 100% rename from src/lib/Twig/TokenParserInterface.php rename to src/php/lib/Twig/TokenParserInterface.php diff --git a/src/lib/Twig/TokenStream.php b/src/php/lib/Twig/TokenStream.php similarity index 98% rename from src/lib/Twig/TokenStream.php rename to src/php/lib/Twig/TokenStream.php index e1bd7ca..81c043c 100644 --- a/src/lib/Twig/TokenStream.php +++ b/src/php/lib/Twig/TokenStream.php @@ -139,7 +139,7 @@ public function test($primary, $secondary = null) */ public function isEOF() { - return $this->tokens[$this->current]->getType() === Twig_Token::EOF_TYPE; + return Twig_Token::EOF_TYPE === $this->tokens[$this->current]->getType(); } /** diff --git a/src/lib/Twig/Util/DeprecationCollector.php b/src/php/lib/Twig/Util/DeprecationCollector.php similarity index 100% rename from src/lib/Twig/Util/DeprecationCollector.php rename to src/php/lib/Twig/Util/DeprecationCollector.php diff --git a/src/lib/Twig/Util/TemplateDirIterator.php b/src/php/lib/Twig/Util/TemplateDirIterator.php similarity index 100% rename from src/lib/Twig/Util/TemplateDirIterator.php rename to src/php/lib/Twig/Util/TemplateDirIterator.php diff --git a/src/php/tpl/layout.min.css b/src/php/tpl/layout.min.css new file mode 100644 index 0000000..0feaddd --- /dev/null +++ b/src/php/tpl/layout.min.css @@ -0,0 +1 @@ +:root{--pad: 2.4rem;--pad-small: 1.5rem;--code: Source Code Pro, Menlo, Consolas, Ubuntu Mono, monospace}.lightColors{--black: #1d1f21;--white: #fafefc;--lighter-black: #373b41;--dark-gray: #707880;--mid-gray: #8a94a0;--base-gray: #b4b8b6;--lighter-gray: #dce0da;--lightest-gray: #eef2ef;--orange: #ea712f;--yellow: #f0c674;--red: #d85b5c;--dark-red: #c44e5f;--green: #a5f38f;--dark-green: #6fad77;--purple: #b294bb;--light-purple: #b8c0f0;--blue: #81a2be;--light-blue: #84c0df;--dim-blue: #5b7a95;--aqua: #7acbc6;--vibrant-blue: #2070d0;--vibrant-red: #ff6666;--vibrant-orange: #ffbb16;--vibrant-yellow: #ffea00;--vibrant-green: #44ea3c}.Content pre,.darkColors{--white: #1d1f21;--black: #f6f8fa;--lighter-black: #c8c4be;--dark-gray: #8F877F;--mid-gray: #756B5F;--base-gray: #4B4749;--lighter-gray: #231F25;--lightest-gray: #110D10;--orange: #1567CF;--yellow: #0F398B;--red: #3d7b88;--dark-red: #2d8b7e;--green: #5A0C70;--dark-green: #905288;--purple: #4D6B44;--light-purple: #473F0F;--blue: #7E5D41;--light-blue: #7B3F20;--dim-blue: #A4856A;--aqua: #754148;--vibrant-blue: #DF8F2F;--vibrant-red: #009999;--vibrant-orange: #0044E9;--vibrant-yellow: #0015FF;--vibrant-green: #BB15C3}::selection{text-decoration:none;color:#fff;background-color:#d85b5c}::-moz-selection{text-decoration:none;color:#fff;background-color:#d85b5c}body{box-sizing:border-box;margin:0;font-family:-apple-system,BlinkMacSystemFont,Segoe UI,Roboto,Helvetica Neue,Arial,sans-serif;line-height:1.5;color:#000;background-color:#fff;color:var(--lighter-black, #000);background-color:var(--white, #fff)}main{padding:1rem;padding:var(--pad)}main>*+:not(:empty){margin-top:1rem;margin-top:var(--pad)}@media (max-width:400px){:root{font-size:90%;--pad: 1.5rem;--pad-small: 1rem}}@media (min-width:900px){:root{padding:2rem;background-color:var(--lightest-gray)}body{max-width:75rem;margin:0 auto;border:solid 1px var(--lighter-gray)}}@media (min-width:1400px){:root{font-size:112.5%}}h1,h2{margin:1.25rem 0;font-size:150%;font-weight:600;line-height:1.3;color:var(--lighter-black)}h2{margin:2rem 0 1rem;font-size:125%}h3,h4,h5,h6{margin:1.5rem 0 1rem;font-size:100%}a{color:#07a;color:var(--vibrant-blue, #07a)}blockquote{padding-left:1rem;border-left:solid .5rem;border-color:var(--vibrant-orange);line-height:1.5}blockquote,p{margin:1em 0}code,pre{font-family:monospace,monospace;font-family:var(--code)}pre{white-space:pre-wrap;-webkit-tab-size:2;-moz-tab-size:2;tab-size:2}code{font-size:93.75%;color:var(--dark-red)}pre code{display:block}pre mark{color:#fff;background:#000}.error{color:var(--red)}.icon{display:inline-block;width:1em;height:1em;vertical-align:-.15em;fill:currentColor}body>header{padding:1rem;padding:var(--pad);padding-bottom:0;line-height:1.2}.Content:empty,body>header:empty{display:none}body>header>:first-child{margin-top:calc(var(--pad)*-.15)}body>header>:last-child{margin-bottom:0!important}body>header nav{line-height:1;font-size:1.125rem;cursor:default}body>header nav.border{padding-bottom:1em;padding-bottom:var(--pad-small);border-bottom:dashed 1px;border-color:var(--lighter-gray);margin-bottom:1em;margin-bottom:var(--pad-small)}body>header nav>*{display:inline-block;margin-right:.1em}body>header nav .sep{padding:0 .15em;color:var(--base-gray)}body>header nav .item{padding:4px 1px;font-weight:600}body>header nav,body>header nav span.item{color:var(--mid-gray)}body>header nav a{text-decoration:none}body>header nav a:focus,body>header nav a:hover{text-decoration:underline}.Content{box-sizing:content-box;max-width:52em;margin:.5rem auto 1rem;margin-bottom:var(--pad-small)}.Content>:first-child{margin-top:0!important}.Content>:last-child{margin-bottom:0!important}.Content h1{margin:1em 0 .66em;font-size:1.8em}.Content h2{margin-top:1.25em;padding:.75rem 0;border-bottom:solid 1px;border-color:var(--lighter-gray)}@media (min-width:900px){.Content h1{font-size:2.2em}}.Code>code,.Content a code,.Content h1 code,.Content h2 code,.Content h3 code,.Content h4 code{color:inherit}.Content table{margin:1.5em 0;border-collapse:collapse}.Content td,.Content th{text-align:left;padding:.5em 1em;border:solid 1px #f0f0f0}.Content pre{margin:1rem -1rem;margin:1rem calc(var(--pad)*-1);padding:.75rem 1rem;padding:.75rem var(--pad);white-space:pre-wrap;font-family:monospace,monospace;line-height:1.5;color:var(--base-gray);background:var(--black)}@media (min-width:900px){.Content pre{margin-left:0;margin-right:0;padding:var(--pad-small);border-radius:2px}}.TestResult{margin:1em 0;padding:1em;border:solid 1px #aaa;font-size:85%;background:#f4f4f4}.TestResult pre{margin:0;padding:0;background-color:transparent}.Content .LinkTable{width:100%;margin:3em 0}.Content .LinkTable td{padding:0}.Content .LinkTable a{display:block;padding:.5em 1em;text-decoration:none}.Code{position:relative;overflow:hidden;margin:0 -1rem;margin:0 calc(var(--pad)*-1);padding:2rem 1rem;padding:2rem var(--pad);white-space:pre-wrap;font-family:inherit;line-height:1.5;color:var(--base-gray);background:var(--black)}header:empty+main>.Code:first-child{margin-top:-1rem;margin-top:calc(var(--pad)*-1)}.Code:last-child{margin-bottom:-1rem;margin-bottom:calc(var(--pad)*-1)}.Code:last-child>code{margin:.5rem 0}.Code>code.markdown,.Code>code.txt{color:var(--lighter-gray)}.Code--mark code>:not(mark),.Code--mark mark>:not([class*=hljs-template-]){filter:saturate(0%)}.Code--numbers>code{padding-left:1rem}@media (min-width:600px){.Code--numbers>code{padding-left:1.5rem}}.Code [data-num]{display:none}.Code--numbers [data-num]{position:absolute;z-index:9;display:inline-block;width:3ch;overflow:hidden;margin-left:-4.5ch;text-align:right;color:var(--dark-gray)}.Code--numbers [data-num]::before{content:attr(data-num)}.hljs{display:block}.hljs-emphasis{font-style:italic}.hljs-strong{font-weight:700}.hljs-subst{color:var(--base-gray)}.hljs-attr,.hljs-name,.hljs-title{color:var(--purple)}.hljs-comment,.hljs-meta,.hljs-meta .hljs-keyword{color:var(--dark-gray)}.hljs-deletion,.hljs-link,.hljs-literal,.hljs-number,.hljs-symbol{color:var(--red)}.hljs-addition,.hljs-doctag,.hljs-regexp,.hljs-selector-attr,.hljs-selector-pseudo{color:var(--green)}.hljs-attribute,.hljs-code,.hljs-selector-id{color:var(--purple)}.hljs-bullet,.hljs-keyword,.hljs-selector-tag{color:var(--blue)}.hljs-string,.hljs-variable{color:var(--aqua)}.hljs-built_in,.hljs-builtin-name,.hljs-quote,.hljs-selector-class,.hljs-type{color:var(--orange)}.hljs-section{color:var(--vibrant-orange)}.hljs-tag{color:var(--blue)}.hljs-tag .hljs-name{color:var(--light-blue)}.hljs-tag .hljs-attr{color:var(--purple)}.hljs-tag .hljs-string{color:var(--mid-gray)}.hljs-template-comment{color:var(--light-purple)}.hljs-template-tag{color:var(--vibrant-orange)}.hljs-template-tag .hljs-keyword,.hljs-template-tag .hljs-name{color:var(--vibrant-yellow)}.hljs-template-variable{color:var(--green)}.hljs-template-variable .hljs-name{color:var(--vibrant-green)}.Links{line-height:1.3}.Links-list{margin:0 0 1rem;margin-bottom:var(--pad-small);padding:0;list-style:none}.Links-list a,.Links-list li{position:relative}.Links-list li+li{margin-top:-1px}.Links-list .icon{position:absolute;left:1rem;top:50%;margin-top:-.55em}.Links-list a{display:block;border:solid 1px;border-color:var(--lightest-gray, #ddd);padding:.5rem 1rem .65rem 3rem;text-decoration:none;font-size:112.5%;color:#000;color:var(--dim-blue, #000)}.Links-list a:focus,.Links-list a:hover{outline:0;color:#fff;background-color:#333;background-color:var(--dim-blue, #333)}@media (max-width:600px){.Links-list a{padding-top:.6rem;padding-bottom:.75rem}}.Links-list--folders a{color:var(--dark-red, #000)}.Links-list--folders a:focus,.Links-list--folders a:hover{color:#fff;background-color:#333;background-color:var(--dark-red, #333)}@media (min-width:750px){.Links-list:not(.Links-list--cols){width:calc(50% - .375em)}.Links-list--cols{-webkit-column-count:2;-moz-column-count:2;column-count:2;-webkit-column-gap:.75em;-moz-column-gap:.75em;column-gap:.75em}.Links-list--cols li{-webkit-column-break-inside:avoid;page-break-inside:avoid;break-inside:avoid}} diff --git a/src/tpl/layout.min.js b/src/php/tpl/layout.min.js similarity index 100% rename from src/tpl/layout.min.js rename to src/php/tpl/layout.min.js diff --git a/src/tpl/layout.twig b/src/php/tpl/layout.twig similarity index 100% rename from src/tpl/layout.twig rename to src/php/tpl/layout.twig diff --git a/src/tpl/layout.min.css b/src/tpl/layout.min.css deleted file mode 100644 index ada5d09..0000000 --- a/src/tpl/layout.min.css +++ /dev/null @@ -1 +0,0 @@ -:root{--pad: 2.4rem;--pad-small: 1.5rem;--code: Source Code Pro, Menlo, Consolas, Ubuntu Mono, monospace}.lightColors{--black: #1d1f21;--white: #fafefc;--lighter-black: #373b41;--dark-gray: #707880;--mid-gray: #8a94a0;--base-gray: #b4b8b6;--lighter-gray: #dce0da;--lightest-gray: #eef2ef;--orange: #ea712f;--yellow: #f0c674;--red: #d85b5c;--dark-red: #c44e5f;--green: #a5f38f;--dark-green: #6fad77;--purple: #b294bb;--light-purple: #b8c0f0;--blue: #81a2be;--light-blue: #84c0df;--dim-blue: #5b7a95;--aqua: #7acbc6;--vibrant-blue: #2070d0;--vibrant-red: #ff6666;--vibrant-orange: #ffbb16;--vibrant-yellow: #ffea00;--vibrant-green: #44ea3c}.Content pre,.darkColors{--white: #1d1f21;--black: #f6f8fa;--lighter-black: #c8c4be;--dark-gray: #8F877F;--mid-gray: #756B5F;--base-gray: #4B4749;--lighter-gray: #231F25;--lightest-gray: #110D10;--orange: #1567CF;--yellow: #0F398B;--red: #3d7b88;--dark-red: #2d8b7e;--green: #5A0C70;--dark-green: #905288;--purple: #4D6B44;--light-purple: #473F0F;--blue: #7E5D41;--light-blue: #7B3F20;--dim-blue: #A4856A;--aqua: #754148;--vibrant-blue: #DF8F2F;--vibrant-red: #009999;--vibrant-orange: #0044E9;--vibrant-yellow: #0015FF;--vibrant-green: #BB15C3}::selection{color:#fff;background-color:#579;color:var(--white, #fff);background-color:var(--dark-red, #579)}::-moz-selection{color:#fff;background-color:#579;color:var(--white, #fff);background-color:var(--dark-red, #579)}body{box-sizing:border-box;margin:0;font-family:-apple-system,BlinkMacSystemFont,Segoe UI,Roboto,Helvetica Neue,Arial,sans-serif;line-height:1.5;color:#000;background-color:#fff;color:var(--lighter-black, #000);background-color:var(--white, #fff)}main{padding:1rem;padding:var(--pad)}main>*+:not(:empty){margin-top:1rem;margin-top:var(--pad)}@media (max-width:400px){:root{font-size:90%;--pad: 1.5rem;--pad-small: 1rem}}@media (min-width:900px){:root{padding:2rem;background-color:var(--lightest-gray)}body{max-width:75rem;margin:0 auto;border:solid 1px var(--lighter-gray)}}@media (min-width:1400px){:root{font-size:112.5%}}h1,h2{margin:1.25rem 0;font-size:150%;font-weight:600;line-height:1.3;color:var(--lighter-black)}h2{margin:2rem 0 1rem;font-size:125%}h3,h4,h5,h6{margin:1.5rem 0 1rem;font-size:100%}a{color:#07a;color:var(--vibrant-blue, #07a)}blockquote{padding-left:1rem;border-left:solid .5rem;border-color:var(--vibrant-orange);line-height:1.5}blockquote,p{margin:1em 0}code,pre{font-family:monospace,monospace;font-family:var(--code)}pre{white-space:pre-wrap;-webkit-tab-size:2;-moz-tab-size:2;tab-size:2}code{font-size:93.75%;color:var(--dark-red)}pre code{display:block}pre mark{color:#fff;background:#000}.error{color:var(--red)}.icon{display:inline-block;width:1em;height:1em;vertical-align:-.15em;fill:currentColor}body>header{padding:1rem;padding:var(--pad);padding-bottom:0;line-height:1.2}.Content:empty,body>header:empty{display:none}body>header>:first-child{margin-top:calc(var(--pad)*-.15)}body>header>:last-child{margin-bottom:0!important}body>header nav{line-height:1;font-size:1.125rem;cursor:default}body>header nav.border{padding-bottom:1em;padding-bottom:var(--pad-small);border-bottom:dashed 1px;border-color:var(--lighter-gray);margin-bottom:1em;margin-bottom:var(--pad-small)}body>header nav>*{display:inline-block;margin-right:.1em}body>header nav .sep{padding:0 .15em;color:var(--base-gray)}body>header nav .item{padding:4px 1px;font-weight:600}body>header nav,body>header nav span.item{color:var(--mid-gray)}body>header nav a{text-decoration:none}body>header nav a:focus,body>header nav a:hover{text-decoration:underline}.Content{box-sizing:content-box;max-width:52em;margin:.5rem auto 1rem;margin-bottom:var(--pad-small)}.Content>:first-child{margin-top:0!important}.Content>:last-child{margin-bottom:0!important}.Content h1{margin:1em 0 .66em;font-size:1.8em}.Content h2{margin-top:1.25em;padding:.75rem 0;border-bottom:solid 1px;border-color:var(--lighter-gray)}@media (min-width:900px){.Content h1{font-size:2.2em}}.Code>code,.Content a code,.Content h1 code,.Content h2 code,.Content h3 code,.Content h4 code{color:inherit}.Content table{margin:1.5em 0;border-collapse:collapse}.Content td,.Content th{text-align:left;padding:.5em 1em;border:solid 1px #f0f0f0}.Content pre{margin:1rem -1rem;margin:1rem calc(var(--pad)*-1);padding:.75rem 1rem;padding:.75rem var(--pad);white-space:pre-wrap;font-family:monospace,monospace;line-height:1.5;color:var(--base-gray);background:var(--black)}@media (min-width:900px){.Content pre{margin-left:0;margin-right:0;padding:var(--pad-small);border-radius:2px}}.TestResult{margin:1em 0;padding:1em;border:solid 1px #aaa;font-size:85%;background:#f4f4f4}.TestResult pre{margin:0;padding:0;background-color:transparent}.Content .LinkTable{width:100%;margin:3em 0}.Content .LinkTable td{padding:0}.Content .LinkTable a{display:block;padding:.5em 1em;text-decoration:none}.Code{position:relative;overflow:hidden;margin:0 -1rem;margin:0 calc(var(--pad)*-1);padding:2rem 1rem;padding:2rem var(--pad);white-space:pre-wrap;font-family:inherit;line-height:1.5;color:var(--base-gray);background:var(--black)}header:empty+main>.Code:first-child{margin-top:-1rem;margin-top:calc(var(--pad)*-1)}.Code:last-child{margin-bottom:-1rem;margin-bottom:calc(var(--pad)*-1)}.Code:last-child>code{margin:.5rem 0}.Code>code.markdown,.Code>code.txt{color:var(--lighter-gray)}.Code--mark code>:not(mark),.Code--mark mark>:not([class*=hljs-template-]){filter:saturate(0%)}.Code--numbers>code{padding-left:1rem}@media (min-width:600px){.Code--numbers>code{padding-left:1.5rem}}.Code [data-num]{display:none}.Code--numbers [data-num]{position:absolute;z-index:9;display:inline-block;width:3ch;overflow:hidden;margin-left:-4.5ch;text-align:right;color:var(--dark-gray)}.Code--numbers [data-num]::before{content:attr(data-num)}.hljs{display:block}.hljs-emphasis{font-style:italic}.hljs-strong{font-weight:700}.hljs-subst{color:var(--base-gray)}.hljs-attr,.hljs-name,.hljs-title{color:var(--purple)}.hljs-comment,.hljs-meta,.hljs-meta .hljs-keyword{color:var(--dark-gray)}.hljs-deletion,.hljs-link,.hljs-literal,.hljs-number,.hljs-symbol{color:var(--red)}.hljs-addition,.hljs-doctag,.hljs-regexp,.hljs-selector-attr,.hljs-selector-pseudo{color:var(--green)}.hljs-attribute,.hljs-code,.hljs-selector-id{color:var(--purple)}.hljs-bullet,.hljs-keyword,.hljs-selector-tag{color:var(--blue)}.hljs-string,.hljs-variable{color:var(--aqua)}.hljs-built_in,.hljs-builtin-name,.hljs-quote,.hljs-selector-class,.hljs-type{color:var(--orange)}.hljs-section{color:var(--vibrant-orange)}.hljs-tag{color:var(--blue)}.hljs-tag .hljs-name{color:var(--light-blue)}.hljs-tag .hljs-attr{color:var(--purple)}.hljs-tag .hljs-string{color:var(--mid-gray)}.hljs-template-comment{color:var(--light-purple)}.hljs-template-tag{color:var(--vibrant-orange)}.hljs-template-tag .hljs-keyword,.hljs-template-tag .hljs-name{color:var(--vibrant-yellow)}.hljs-template-variable{color:var(--green)}.hljs-template-variable .hljs-name{color:var(--vibrant-green)}.Links{line-height:1.3}.Links-list{margin:0 0 1rem;margin-bottom:var(--pad-small);padding:0;list-style:none}.Links-list a,.Links-list li{position:relative}.Links-list li+li{margin-top:-1px}.Links-list .icon{position:absolute;left:1rem;top:50%;margin-top:-.55em}.Links-list a{display:block;border:solid 1px;border-color:var(--lightest-gray, #ddd);padding:.5rem 1rem .65rem 3rem;text-decoration:none;font-size:112.5%;color:#000;color:var(--dim-blue, #000)}.Links-list a:focus,.Links-list a:hover{outline:0;color:#fff;background-color:#333;background-color:var(--dim-blue, #333)}@media (max-width:600px){.Links-list a{padding-top:.6rem;padding-bottom:.75rem}}.Links-list--folders a{color:var(--dark-red, #000)}.Links-list--folders a:focus,.Links-list--folders a:hover{color:#fff;background-color:#333;background-color:var(--dark-red, #333)}@media (min-width:750px){.Links-list:not(.Links-list--cols){width:calc(50% - .375em)}.Links-list--cols{-webkit-column-count:2;-moz-column-count:2;column-count:2;-webkit-column-gap:.75em;-moz-column-gap:.75em;column-gap:.75em}.Links-list--cols li{-webkit-column-break-inside:avoid;page-break-inside:avoid;break-inside:avoid}} diff --git a/test/index.twig b/test/index.twig index 7691685..dbb3612 100644 --- a/test/index.twig +++ b/test/index.twig @@ -2,7 +2,11 @@ {% block content %}
Scope: We’re only testing features specific to TwigExpress. Manual tests for now.
+twigexpress.json
.Page | @@ -19,4 +23,17 @@ {% endif %} {% endfor %}
---|
{ + "debug_mode": true, + "global_vars": { + "test_data": { + "some_number": 3.1415926535898, + "with spaces": "Hey there" + } + }, + "namespaces": { + "test_namespace": "./test" + } +}{% endblock %} diff --git a/twigexpress.json b/twigexpress.json index 8cf59b1..159e1c3 100644 --- a/twigexpress.json +++ b/twigexpress.json @@ -1,12 +1,7 @@ { "debug_mode": true, "global_vars": { - "test_data": { - "some_number": 3.1415926535898, - "with spaces": "Hey there" - } + "some_global_var": "I'm set in twigexpress.json" }, - "namespaces": { - "test_namespace": "./test" - } + "namespaces": {} } diff --git a/twigexpress.phar b/twigexpress.phar index c3c9ca0..41f85cc 100644 Binary files a/twigexpress.phar and b/twigexpress.phar differ