---
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 %}