--- title: 'ATDC: Lesson 6 - Builders and custom assertions' permalink: /atdc/6-builders-custom-assertions --- {% block head_meta %} {% endblock %} {% block content %} In yesterday's lesson, you created your first Kernel test and used it to ensure the posts are returned from `PostNodeRepository` in the desired order. This is how we're creating the posts currently: ```language-php $this->createNode([ 'created' => (new DrupalDateTime('-1 week'))->getTimestamp(), 'title' => 'Post one', 'type' => 'post', ]); $this->createNode([ 'created' => (new DrupalDateTime('-8 days'))->getTimestamp(), 'title' => 'Post two', 'type' => 'post', ]); $this->createNode([ 'created' => (new DrupalDateTime('yesterday'))->getTimestamp(), 'title' => 'Post three', 'type' => 'post', ]); ``` The Builder pattern is another design pattern I like, which makes it easier to build complex objects. Let's create a Builder class to create the posts. ## Creating a PostBuilder class This is how I'd like to create a post using a `PostBuilder`: ```language-php PostBuilder::create() ->setCreatedDate('-1 week') ->setTitle('Post one') ->getPost(); ``` This makes it easier to do by creating named methods for each value we want to set and not relying on array keys whilst also moving implementation details like using `DrupalDateTime` to set the `created` date. To do this, create a new class at `src/Builder/PostBuilder.php`: ```language-php created = new DrupalDateTime($time); return $this; } public function setTitle(string $title): self { $this->title = $title; return $this; } ``` Again, by returning `$this`, we can keep chaining methods. Finally, create the `getPost()` method that creates the node based on the property values, saves it, and returns it. ```language-php public function getPost(): NodeInterface { $post = Node::create([ 'created' => $this->created?->getTimestamp(), 'title' => $this->title, 'type' => 'post', ]); $post->save(); return $post; } ``` Now, refactor the test to use the `PostBuilder`: ```language-php PostBuilder::create() ->setCreatedDate('-1 week') ->setTitle('Post one') ->getPost(); PostBuilder::create() ->setCreatedDate('-8 days') ->setTitle('Post two') ->getPost(); PostBuilder::create() ->setCreatedDate('yesterday') ->setTitle('Post three') ->getPost(); ``` Doing this simplifies the test and makes it easier to extend in the future by adding more methods to `PostBuilder`. ## Creating a custom assertion Finally, for today, let's refactor the assertion that verifies the titles are returned in the correct order. This is the current assertion: ```language-php self::assertSame( ['Post two', 'Post one', 'Post three'], array_map( fn (NodeInterface $node) => $node->label(), $nodes ) ); ``` We create an array of expected titles and compare that to an array created from `array_map`. We can make this more reusable and readable by extracting this into a new custom assertion, which is just another static method. Create a new static function at the bottom of the class with a name that describes what it's asserting: ```language-php /** * @param array $expectedTitles * @param array $nodes */ private static function assertNodeTitlesAreSame( array $expectedTitles, array $nodes, ): void { self::assertSame( $expectedTitles, array_map( fn (NodeInterface $node) => $node->label(), $nodes ) ); } ``` We can add arguments for the arrays of titles and nodes, and be explicit about what they contain by adding a docblock. In this method, we can do the same logic and use `array_map` to create a list of node titles and compare them to the expected titles. The benefits are that this now has a name that describes what we're asserting, and because it's a separate method, it can be reused in the same test or moved to a base class and used elsewhere. Finally, refactor the test to use the new assertion: ```language-php self::assertNodeTitlesAreSame( ['Post two', 'Post one', 'Post three'], $nodes, ); ``` In my opinion, this is a lot better. In tomorrow's lesson, let's add some more tests to the `PostNodeRepository` that we skipped in previous lessons. {% endblock %}