Chapters
Twig: For a Good time with Templates
Scroll down to the script below, click on any sentence (including terminal blocks) to jump to that spot in the video!
Unless you're building a pure API, Twig is your new best friend. Rarely do you find a library that's this much fun to use. It's also really easy, so let me just give you a quick intro.
{{ SaySomething }}, {% doSomething %}
Twig has two syntaxes: {{ }}
- which is the "say something" tag - and {% %}
- which is the "do something" tag. If you're printing something, you aways write {{
then a variable name, a string or any expression: Twig looks a lot like JavaScript.
But if you're writing code that won't print something - like an if
statement a for
loop, or setting a variable, you'll use {% %}
.
Head over to Twig's website at twig.sensiolabs.org, click Documentation and scroll down. Ah, this is a list of everything Twig does.
Look at the Tags column first: this is the short list of all "do something" tags. Click on if
to see some usage. Do something tags are always {%
and then if
or set
or one of these "tags". The for
tag - used as {% for %}
is for looping. You'll probably end up only using 5 or 6 of these commonly.
Twig also has other things like functions... which are exactly like functions in every language ever. Filters are a bit more interesting: check out lower
. Really, these are functions, but with a trendier syntax: just print something, then use the pipe (|
) to pass that value into a filter. You can have filter after filter. And, you can create your own.
The dump() Function
Let's make some magic happen in our Twig template. Our Aquanauts will take notes about each genus, and those will render on this page. Create a cool $notes
variable with some hardcoded text and pass it into our Twig template:
Show Lines
|
// ... lines 1 - 8 |
class GenusController extends Controller | |
{ | |
Show Lines
|
// ... lines 11 - 13 |
public function showAction($genusName) | |
{ | |
$notes = [ | |
'Octopus asked me a riddle, outsmarted me', | |
'I counted 8 legs... as they wrapped around me', | |
'Inked!' | |
]; | |
return $this->render('genus/show.html.twig', array( | |
'name' => $genusName, | |
'notes' => $notes | |
)); | |
} | |
} |
But before we loop over this, I want to show you a small piece of awesome: the dump()
function:
Show Lines
|
// ... lines 1 - 2 |
{{ dump() }} |
This is like var_dump()
in PHP, but better, and you can use it without any arguments to print details about every available variable.
Refresh the browser! There's the name
variable, notes
and a bonus: app
- a global variable that Symfony adds to every template. More on that in the future. With the dump()
function, you can expand the variables in really cool ways. Oh, and bonus time: you can also use dump()
in PHP code: Symfony gives us that function.
Go Deeper!
See more usage examples of the dump()
function in The dump Function for Debugging
of the separate Twig screencast.
chapter.
The for Tag
To print out the notes, add a <ul>
and open up a for
tag with {% for note in notes %}
. Close it with an {% endfor %}
tag. Now, it's simple: print out each note, which is a string:
Show Lines
|
// ... lines 1 - 2 |
<ul> | |
{% for note in notes %} | |
<li>{{ note }}</li> | |
{% endfor %} | |
</ul> |
Back to the browser to see what we've got. Refresh! Well, it's not pretty yet, but it is working. Open the source: it's still just this html, there's no HTML layout. Time to fix that.
18 Comments
Hey Junaid,
Hm, this error looks unrelated to the dump function... What if you comment out this dump function? Does everything work fine or do you still see this/another error? Can you show your dump function and a bit of content around it?
Cheers!
Hey victor
If i remove the dump function, everyhing works fine. But i initially got the following error and i think the problem is with the PHP v7.2 that i am running:
Warning: count(): Parameter must be an array or an object that implements Countable
500 Internal Server Error - ContextErrorException
Got this error right at the beginning after finishing the project installation and then trying to run server. Can you please help in this regard. Thanks in advance. Cheers!
Yo Junaid,
OK, let's fix this problem first. Looks like you call count() function somewhere, and the value you passing to it is not an array or an object which implements Countable interface. Can you find where you call this count() function? You can debug value which is passed to count() with:
dump($someVar); die;
count($someVar);
If you still have a problem with dump(), try to use var_dump() instead.
Cheers!
Hey Victor,
I am not calling count() function anywhere in my code as i told you i got this error right at the beginning otherwise i would know where is the bug. Besides after researching a bit, i came to know that this is somehow related to PHP V 7.2 but i couldn't just figure out how to resolve this issue. But when i migrated to PHP V 5.6 everything worked as it should without any error (at least this one was not there anymore). I specifically got this error when i created a couple of Symfony projects, one in V 2.8.1 and another in V 3.0.1.
Cheers!
Hey Junaid,
Hm, this sound like a bug in vendors, probably in Symfony core or in other dependencies, I found a few related issues: https://github.com/symfony/... . I think this is fixed now, so probably you just need to update dependencies to the latest versions. Anyway, it's difficult to say where's this problem exactly without debugging. If you wonder, you can follow backtrace of the calls from Symfony on the error page to see from where exactly this error came.
Cheers!
Hi there!
I am having some trouble getting the variable being passed by my controller into the twig template. I'm returning it exactly as shown in the tutorial and PHPStorm can see it for auto completion but when I actually run the code in my browser it tells me the variable can't be found.
Hi Pad!
Oh that *is* weird - especially since PHPStorm can see the variables. Try this: remove all the variables (so that there aren't any errors anymore) and then just execute {{ dump() }}. This will dump out *all* the variables you have available to you. If there's some weird issue - you should see that your variables are *not* in this list. This isn't a solution exactly - but hopefully it'll take you to the next step of debugging.
And btw, there really *shouldn't* be anything weird happening here - there's no magic to watch out for - it's probably some tiny error somewhere :). Let me know how it goes!
Hello,
I'm unable to call dump() in the twig template. I can use {{ name }} and see the array of notes in {{ notes }}, however adding {{ dump() }} anywhere in the twig file, either by itself or in addition to example code as you have shown results in a 500 error.
I have tried flushing the cache and that doesn't seem to help. Additionally, I noticed when you started typing 'dump', you had annotations pop-up. My does not do that.
Hey Michael,
I bet you use it in prod, but keep in mind that the dump() Twig function works only in dev mode. It's just for security reasons, i.e. to prevent leaking data (credentials, etc.). Actually, dump() function should be used only for debug reasons, but not for the production, where you should look over logs or use some monitoring services.
Cheers!
I have a vagrant box I'm using for this, so I'm not using the built in web server. I simply modified the app.php and set dev to true and it worked as expected. Thank you for pointing me in the right direction.
I've started testing my app in the production environment. It took me forever to narrow down the following error that started appearing:<br />Oops! An Error Occurred<br />The server returned a "500 Internal Server Error".<br />Something is broken. Please let us know what you were doing when this error occurred. We will fix it as soon as possible. Sorry for any inconvenience caused.<br />
It turned out that I had a dump function in one of the templates. <br />{{ dump() }}<br />
Actually I had the dump function enclosed in an if statement:
{% if app.environment == 'dev' %}
{{ dump() }}
{{ dump(app.session.all) }}
{{ dump(app.request.attributes) }}
{% endif %}
The documentation seems to indicate there is a dump function in the 'prod' environment, but that nothing will be dumped (it should indicate that it doesn't exist and will throw a difficlut-to-find error!):
http://symfony.com/doc/current/templating/debug.html
<blockquote>
"The variables will only be dumped if Twig's debug setting (in config.yml) is true. By default this means that the variables will be dumped in the dev environment but not the prod environment."
</blockquote>
It turns out that the dump function doesn't exist in the production (with debug=false) environment.<br />bin/console debug:twig --env=prod // no dump listed in the functions<br />
Thus, even enclosing the dump function in a check for the environment doesn't work as it still throws the non-specific error above. I'm assuming Symfony/twig's internal compiling of the template is failing when it hits the 'debug' in the 'prod' environment.
Questions:
1) Is there any logging for this type of error? When the above error was output, there were no errors written to the apache or php logs. I think this is just Symfony throwing out a '500' error on its own while fully executing as far as apache/php are concerned. Does symfony log these errors somewhere? (It was VERY difficult to debug this issue, because the issue happened in the 'prod' environment, but when I'd switch to 'dev' or 'prod'/test, everything suddenly worked, so there was nothing to debug!)
2) I'm lazy and would like to keep my 'dump' functions in my code for all time and just protect them with the 'if' statement as above. Is there a way to create a 'silent' dump function that does nothing in the 'prod' non-testing environment?
3) Is #2 bad practice?
Hi Terry!
Wow, you're right! This is absolutely a bug in the documentation - I've just verified the same behavior as you: https://github.com/symfony/symfony-docs/issues/7937
Now, let's get to your questions:
1) Yes! Production errors are logged to var/logs/prod.log. This should show the issue (it did in my case). If an error is super-fatal (which it is not in this case), you will only see error in your web server log.
2) The reason the dump()
function isn't available is because the DebugBundle that provides it is not enabled in the prod environment. So yea, you could do this - and I don't see a big problem with it (to answer question #3). The simplest way would be to (A) create a Twig extension that implements a dump() function that does nothing then (B) register this as a service in config_prod.yml
. We don't normally register services in that file... but this is an abnormal situation: we want that service (i.e. Twig extension) to be registered only in the prod environment, where the normal dump() doesn't exist.
Sorry it took you so long to track down! Like with all errors, if you don't know where to look, then you're totally guessing (which sucks!)
Cheers!
Thanks much. Knowing about the var/logs/prod.log will be very helpful and hopefully save me a lot of time in the future!
For #2 - Haven't had a chance to try it yet, but what will happen if I need to see errors in the "prod" environment and set the debug parameter to "true" in the "prod" environment. Will one dump override the other?
<br />$kernel = new AppKernel('prod', false); // no dump conflict<br />vs<br />$kernel = new AppKernel('prod', true); // twig dump conflict?<br />
BTW: In the first sentence of #2 you put "dev" but meant "prod" (for any future readers)
Yo Terry Caliendo!
BTW: In the first sentence of #2 you put "dev" but meant "prod"
Thank you! I fixed that above!
If both dump() functions are enabled, one will override the other. I believe you function will win, as it's registered second - but it's not a normal situation to have two functions competing with each other. And that false
versus true
flag to AppKernel
is what normally configures Symfony to hide or show errors. But, this flag isn't what controls the dump()
function being available: it's the environment. If you look in AppKernel, the DebugBundle (which provides the function) is only enabled in the dev and test environments. If I did want to see my dumps temporarily, I would actually create a temporary new front controller with new AppKernel('dev', true)
and run that - go full debug/dev mode :). But, I don't need to do this very often - the prod.log file usually has what I need (actually, we configure Monolog to send all errors to Slack, so we're basically pinged when something goes into this file) to be able to replicate something locally. But, we also never have bugs on production... ;)
Cheers!
Hi Ryan,
I read the Docs from:
http://symfony.com/doc/curr...
and I can't find the docs for: path(), url(), and asset() function in Twig docs or somethere else.
are these depreciated?
Thanks for your advice
Teo
Hey Teo!
They are definitely still used and relevant - check out this page: http://symfony.com/doc/curr.... The one you linked to is more of a marketing "preview" of Symfony. The link I posted is the real templating/Twig documentation in Symfony :). And it has all the details about each of those functions. We also have a reference section for all the custom things that are added to Twig by Symfony: http://symfony.com/doc/curr...
I hope that helps! Cheers!
"Houston: no signs of life"
Start the conversation!
What PHP libraries does this tutorial use?
// composer.json
{
"require": {
"php": ">=5.5.9",
"symfony/symfony": "3.1.*", // v3.1.4
"doctrine/orm": "^2.5", // v2.7.2
"doctrine/doctrine-bundle": "^1.6", // 1.6.4
"doctrine/doctrine-cache-bundle": "^1.2", // 1.3.0
"symfony/swiftmailer-bundle": "^2.3", // v2.3.11
"symfony/monolog-bundle": "^2.8", // 2.11.1
"symfony/polyfill-apcu": "^1.0", // v1.2.0
"sensio/distribution-bundle": "^5.0", // v5.0.22
"sensio/framework-extra-bundle": "^3.0.2", // v3.0.16
"incenteev/composer-parameter-handler": "^2.0", // v2.1.2
"composer/package-versions-deprecated": "^1.11" // 1.11.99
},
"require-dev": {
"sensio/generator-bundle": "^3.0", // v3.0.7
"symfony/phpunit-bridge": "^3.0" // v3.1.3
}
}
Hey There,
I am getting the following error while try to use dump function in the template:
[2/2] Twig_Error_Runtime: An exception has been thrown during the rendering of a template ("Notice: A non well formed numeric value encountered") in genus\show.html.twig at line 1
[1/2] ContextErrorException: Notice: A non well formed numeric value encountered
LOGS:
CRITICAL - Uncaught PHP Exception Twig_Error_Runtime: "An exception has been thrown during the rendering of a template ("Notice: A non well formed numeric value encountered") in "genus/show.html.twig" at line 1." at C:\Data\Literature\Programming\Source Code\Symfony\Symfony 3\aqua_note\vendor\twig\twig\lib\Twig\Template.php line 396
I actually got an error of type ContextErrorException (which i think is somehow related to php v7.2) during the initial setup. But i ignored it as everything worked normally as it should till now.
Thanks in advance!