Phpstorm Symfony Phpunit



The PhpStorm Symfony Plugin To get really crazy, you'll want to install the amazing, incredible Symfony plugin. This thing makes Symfony development so absurdly fun, I'm going to walk you through its installation right now. In Preferences, search for Symfony and click the plugins option.

Here, I’m going to talk about configuring XDebug with PHPStorm and Docker.

  1. Starting with PhpStorm 6.0, PHPUnit Skeleton Generator is no longer available in the IDE. PHPUnit tests creation functionality has been completely re-worked so that PhpStorm generates tests itself and you can benefit from its flexibility and code insight.
  2. Symfony 4.2 skeleton with fully configured Docker, PHPSTORM, Xdebug, PHPUnit. Symfony 4.2 skeleton with fully configured Docker, PHPSTORM, Xdebug, PHPUnit.

ℹ️ Important note: I will not talk about using it in an HTTP context. I have struggled so much in the past and never succeeded in configuring this, and I will probably never do it anyway.
I’m only using it when testing, with PHPUnit or Behat, and that’s perfect: it forces me to write more tests.

Having a working Docker environment

First of all, if we talk about Docker, you may refer to the series of blog posts I wrote about Docker, it might help you.

I will now consider you have a working PHP + Docker environment.

I also consider you already installed the Docker plugin for your PHPStorm IDE.

Make sure XDebug is available

Of course now you have a working Docker setup, but remember that XDebug must be present.

I usually install it in my Docker images by adding a RUN statement with (echo ' | pecl install xdebug).
This echo thing is a trick to force pecl to execute in a “non-interactive” mode, in order to let the Docker image be built automatically with no user interaction (which is not possible).

If you have PHP 7.2+, you can even make your debug-based test scripts cross-compatible with any platform thanks to extension loading by name.TL;DR: it means that you can do php -dzend_extension=xdebug instead of php -dzend_extension=xdebug.so for UNIX and php -dzend_extension=xdebug.dll for Windows. Yeah, it’s just about removing the extension.

To know whether XDebug is available, run php -dzend_extension=xdebug -i | grep xdebug.

If all XDebug options are displayed with their default/configured values, it works!

Start configuring!

Okay, let’s see ALL the steps that I go through to set this up.

First, the PHP interpreter

Let’s consider we have a php container running, configured in our docker-compose.yaml file.

PHPStorm will need a PHP interpreter.

For this, go to the File | Settings | Languages & Frameworks | PHP menu.

You should see something like this:

Now, you need to click on the [...] button at the right of the CLI Interpreter section in order to create/use a PHP interpreter.

You might have an existing PHP interpreter, but for the sake of the example, I’ll show you how to configure your PHP Docker container.

Add a new “Remote Interpreter” by clicking on the big + sign:

You should configure the remote interpreter to use Docker Compose and have something similar to this:

If it comes that you have the Server section to be empty, click on New and you may see something like this:

Note: I am using Windows, and I deliberately checked the Expose daemon on tcp://localhost:2375 without TLS checkbox in Docker For Windows configuration.I may update this post in the future for linux-specific config, so check out your docker machine in the first place to see if it can be linked to PHPStorm instead of using the legacy daemon tcp socket.

Here is my final configuration:

Some notes:

  • You should decide on whether you want PHPStorm to use docker-compose run --rm php or docker-compose exec php.
    run guarantees an isolated container, but is slower because it needs to start the container first.
    exec is faster because it connects to a running container, but may have concurrence issues, even though it’s rare (feel free to share experience on that!).
  • You can add Environment Variables for each interpreter. That’s nice if you want a “test-optimized” or a “profiling-optimized” one, etc.
  • Note the Debugger extension field: if you write xdebug (or xdebug.so or xdebug.dll for PHP<=7.1), PHPStorm will automatically append -dzend_extension=xdebug when running a script in “Debug Mode”.

Phew! Now we have PHP, let’s install PHPUnit!

PHPUnit

You may have noticed that I’m a big Symfony fan, so we will take the example of a Symfony project here.

First, I do composer require phpunit. This will install the symfony/test-pack package, which is a package that requires a few other packages, with in particular the symfony/phpunit-bridge package.

The Symfony PHPUnit Bridge component comes with a modified version of PHPUnit (TL;DR: it’s a wrapper around PHPUnit) that will allow you to not require PHPUnit in your composer.json file. This might save some dependencies issues, because PHPUnit and your project might depend on same packages with different versions, and you don’t want that. Apart from that, the component provides some other nice features you may check on the docs.

Apart this “dependencies conflicts” theory, the PHPUnit team decided to use PHP-Scoper for their PHAR version, so if you use the phpunit.phar file, you will not have any conflict either, and that’s okay.

The good thing about this is that requiring it like this on a modern Symfony 4/5 project will install the PHPUnit Bridge Flex recipe that commes with a nice bin/phpunit script.
Very convenient for command-line, but don’t use it with PHPStorm (I will talk about this later).

Right after your composer require phpunit, execute bin/phpunit --version.

The wrapper provided by Symfony will install PHPUnit, find a good version for your system.

You can still override the version in the phpunit.xml.dist file created by the Flex recipe. I personally always update.

By default (as of the time I write this post), PHPUnit is installed via a big composer create-project command (you can find it here) into your bin/.phpunit/phpunit-{version}/ directory.

This point is important, because we will configure the PHPUnit “Run configuration” in PHPStorm by using this specific configuration.

Now! Comes the moment where we move back to PHPStorm!

First, click on the Add Configuration... button on the top-right section of your PHPStorm screen:

Then, add a PHPUnit configuration:

An empty PHPUnit configuration never works.

PHPStorm needs many things for it to work:

Phpunit setup
  • A working PHP Interpreter
  • A way to execute PHPUnit (autoloader, include path, executable…)
  • An optional configuration file (but we must set it anyway, else PHPStorm’s PHPUnit process will not use the phpunit.xml.dist file at all)

So, to make it work, configure PHPUnit:

Add a new PHPUnit configuration with a Remote Interpreter:

Select the remote interpreter you created with your Docker configuration.

Then, we will tell PHPStorm to look for the PHPUnit executable.

When we ran bin/phpunit --version, the bridge installed PHPUnit in bin/.phpunit/phpunit-{version}, remember? We will pick the executable from there.Important: remember that the script path will be inside the Docker container, so be careful about paths.

Also remember to fill the Default configuration file field, else PHPUnit will not use any config.

Note: You might have slight differences in paths and versions depending on your Docker and PHPUnit configuration.

After that: PHPUnit is configured!

Phpstorm Symfony Phpunit

And for XDebug, we installed it.

Run a PHPUnit script in Debug mode with step-by-step debugging

To run in Debug mode, you have the little bug icon next to the “Run” button in your “Run configuration” top bar:

Click on it, and see if your project is tested!

Now, we’ll do step-by-step debugging, thanks to XDebug and PHPStorm’s integration.

Find some piece of code you want to test, and add a breakpoint. To do so, you can left-click in the left gutter of the line you want to stop by when running the test, or you can also place the cursor on the line and press Ctrl+F8 (Windows keymap).

Now, DEBUG!

The test will execute and PHPStorm will open a brand new tab so you can debug everything: stack trace, variables state, etc.

You can now start your step-by-step debugging, thanks to these buttons:

These buttons allow you to execute current line and go to next instruction, step into the function/method call, step out of the current function, etc.

Have fun doing better debugging than dump($var);exit;!

Late notes:

Why do I choose to not use bin/phpunit?

Well, because it simply does not run with PHPStorm.

Phpunit setup

The reason might be that as it is a wrapper around PHPUnit, Symfony adds features to it (or removes some), especially the one that installs PHPUnit in the first place.
Then, PHPStorm needs to execute this script before finishing the config, because it needs to know which version of PHPUnit is installed (either for auto-completion, autoload, or maybe PHPUnit-specific stuff I am not aware of).
The wrapper does not return the same contents than the native PHPUnit script, so PHPStorm will consider it either not working or incompatible.

This is why I use the native PHPUnit script that is installed by executing bin/phpunit --version

You probably all know PHPStorm. Just a quick reminder to start this article: PHPStorm is an IDE (Integrated Development Environment) engineered by JetBrains for web developers. This full-featured web development IDE has become increasingly popular over the last couple of years.

Recently, I began to use it myself.

I used Emacs before. But over time PHPStorm integrated such wonderful features that I decided to give it a try.

I used to have a fully customized version of Emacs for PHP:

  • PHPCS, PHPStan and PHPMD linting
  • Integrated Symfony console
  • PHP-CS-Fixer fixes on buffer save
  • PHPUnit integration
  • Xdebug debugging

If you want to retrieve these configurations have a look at here.

After switching to PHPStorm, I was quite satisfied. Indeed, almost all of these features were natively integrated with very few configurations.

But one wasn’t integrated as I wanted: PHP-CS-Fixer fixes on demand.

If you never heard about PHP-CS-Fixer, you really should have a look at this useful tool.
In the official documentation, PHP-CS-Fixer is described as the following:

The PHP Coding Standards Fixer (PHP CS Fixer) tool fixes your code to follow standards; whether you want to follow PHP coding standards as defined in the PSR-1, PSR-2, etc, or other community-driven ones like the Symfony one. You can also define your (team’s) style through configuration.

Phpunit

Phpunit Setup

As I’m writing this article, PHPStorm still doesn’t handle external formatters. Therefore PHP-CS-Fixer fixes could only be triggered using the intentions menu (Alt+Enter or the light bulb).

I didn’t want to wait for PHPStorm’s intentions guessings. Indeed, PHPStorm is taking time to guess theses intentions, and sometimes even doesn’t guess it. Fortunately, I came up with a tiny solution that I’m going to explain.

Cookbook

This kind of solution has already been described in a lot of articles over the internet.

But I wanted to go further and get my PHP-CS-Fixer using relevant configuration files.

Phpstorm Docker Phpunit

For example, if I’m in a project that’s holding a configuration file (eg: .php_cs or .php_cs.dist), I want fixes to be done using that configuration. On the other hand, if none of these files are present in the project, I want my default configuration file to be used.

In that way, I’ll be compliant with each project’s code style and I'll have my coding style on every other PHP files.

1. Install PHP-CS-Fixer

I’m not going to detail this step as it’s already done very well on the official documentation.

2. Write the wrapper PHP script

The trick in this solution is to write a custom PHP script that will act as a wrapper of PHP-CS-Fixer and will execute some custom logic.

Thanks to that script, we’ll be free to execute any custom logic we want.

Below is the content of my own script /home/user/.mtarld/phpcsfixer/phpcsfixer:

As you can see, we just dynamically tell PHP-CS-Fixer which configuration to use when it’s executed.

This script could therefore be executed like the following:

⚠️ Don’t forget to add execution rights to your script!

3. Create a new PHPStorm External tool

The hardest part is done. Now, we just have to bind PHPStorm with our script.

To do that, in the settings popup, navigate to Tools > External Tools and click on the + button to add a new external tool.

Then, configure it like the following:

Important fields are:

  • Program: The location of your custom script.
  • Arguments: Arguments given to the script.
    Here we are giving the project root path and the current file path
  • Open console for tool output: If console should pop on command execution
    As we don’t wanna be bothered on each fix, we’ll leave it unchecked

4. Add a handy key binding

Last but not least, specify a key binding that will be used to trigger our custom PHP-CS-Fixer.

To do that, in the settings popup, navigate to Keymap, unfold External Tools twice, and double click on PHPCS Fixer (or whatever the name you gave it). Finally, bind your favorite keys.

And voilà!

As a result, you’ll just have to press your favorite keybinding, and the file will be fixed.

Moreover, following this very same cookbook, you’ll be able to easily extend PHPStorm integrating other external tools into it. Therefore, feel free to customize it as you like!