Do you test your projects with automated tests? If not, would you like to start? Do you work with application, integration, functional, unit, and Selenium layers and drive you crazy? Do you spend more time writing tests than the actual code behind them?
I want my tests to be simple, effective, and fun to write and maintain. Today, we look at one approach used by PHP itself, nikic/php-parser
. It's so good I'm surprised not everyone is using it.
You can spend a year building house of your dreams. You already live in a flat, so you have a place to sleep. There is no pressure. You go there every weekend, add a window here and there, add doors, prepare holes in walls for electric heating cables... good 2-4 years.
How would the situation change if you'd build houses as a developer (the building one)? Your job is to build 100 flats per year. You can't fool around with the color of the walls inside each room. You have to be effective.
The first case of building one house for a year - you only work on 1 PHP project at once. It's only natural to try-out all the testing layers you can Google. There are a couple of tests that test the product checkout process, a couple of integration tests to check the component is rendered correctly, mocking to "decouple" one part from another.
Every month I work with ~5 private projects, and I maintain 35 open-sources packages. I used to have very complicated tests for all possible application parts, but that turned out to take more time to maintain to develop, and it slowed down my productivity brutally the more tests I had.
Let's look at the PHP test with name 001. Give it 60 seconds:
--TEST--
Trivial "Hello World" test
--FILE--
<?php echo "Hello World"?>
--EXPECT--
Hello World
Do you need a PHP-core developer to explain the whole testing process? You don't; you get it.
When we call:
<?php echo "Hello World"?>
We get the output:
Hello World
The 1st line is just a description, useful for a more complicated case.
It's like a smartphone or door handle in testing.
This kind of testing gives you confidence, and that's by far the most important feeling that builds senior code bases.
You're probably thinking, "but how do I apply this to my unique startup that does not compare strings"? Of course, there is a place for the complex test that checks your checkout process work. The goal is not to narrow all your tests to 1 size to fit em all.
The goal is to find what startup is different from other projects. Is your specialty to build e-commerce websites, or is it a recommendation of the next best product? Is it an instant delivery of warm food or a reliable video conference for massive numbers of users?
Find your domain, because in this domain will be placed 80 % of your tests. If you pick the right domain and make these tests simple, effective, and fun to write and maintain, your developers will enjoy writing them, and your code will naturally grow.
Let's look at the projects you know:
Enough theory, let's do the practice.
We have a fixture file with your domain, /fixture/first_try.php
:
input
-----
expected output
Then we need to run Test Case:
<?php
use PHPUnit\Framework\TestCase;
final class FirstTryTest extends TestCase
{
public function test(): void
{
$fixtureContent = file_get_contents(__DIR__ . '/fixture/first_try.php');
[$input, $expectedOutput] = explode("\n-----\n", $fixtureContent);
// test your main domain service
$output = $this->processInputInYourDomain($input);
$this->assertSame($expectedOutput, $output);
}
}
And that's it :) See 3v4l.org code sample.
-/tests/fixture/before/input_string.php
-/tests/fixture/after/some_string_print.php
+/tests/fixture/change_string.php
"But what happens when we add a new property to the output? Do we have to change all the files manually? That's crazy."
It would be crazy. I tried to update 60 files in php-parser when I only added typed properties... oh, that was too much work. At file 50, I figured out there is an automated way exactly my case. We will look at how we can turn these tests into snapshot tests that update themselves in the next post.
PS.: Do you want to automate part with loading and splitting fixture? Checkout symplify/easy-testing package.
Happy coding!
Do you learn from my contents or use open-souce packages like Rector every day?
Consider supporting it on GitHub Sponsors.
I'd really appreciate it!