Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Improved console testing #37

Merged
merged 7 commits into from
Oct 8, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
3 changes: 2 additions & 1 deletion .github/workflows/docs.yml
Original file line number Diff line number Diff line change
Expand Up @@ -41,10 +41,11 @@ jobs:
php src/bin/generate-docs.php src/factories/Asset.php docs/factories/asset.md
php src/bin/generate-docs.php src/dom/NodeList.php docs/dom/node-list.md
php src/bin/generate-docs.php src/dom/Form.php docs/dom/forms.md
php src/bin/generate-docs.php src/behaviors/TestableResponseBehavior.php docs/assertions/response.md
php src/bin/generate-docs.php src/behaviors/TestableResponseBehavior.php docs/assertions/http-response.md
php src/bin/generate-docs.php src/behaviors/TestableElementBehavior.php docs/assertions/element.md
php src/bin/generate-docs.php src/test/DatabaseAssertions.php docs/assertions/database.md
php src/bin/generate-docs.php src/test/RequestBuilders.php docs/making-requests.md
php src/bin/generate-docs.php src/console/TestableResponse.php docs/assertions/console-response.md
php src/bin/generate-docs.php src/web/BenchmarkResult.php docs/assertions/benchmark.md
php src/bin/generate-docs.php src/test/CookieState.php docs/cookies.md
php src/bin/generate-docs.php src/test/ActingAs.php docs/logging-in.md
Expand Down
39 changes: 39 additions & 0 deletions docs/assertions/console-response.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,39 @@
# Console Response Assertions

A testable response is returned when running a console command action. This class provides a fluent interface for
asserting on the response.

## assertSuccesful()
Assert that the console command exited successfully (with a zero exit code).

```php
$this->command(ConsoleController::class, 'actionName')->assertSuccessful();
```

## assertFailed()
Assert that the console command failed (with a non-zero exit code).

```php
$this->command(ConsoleController::class, 'actionName')->assertFailed();
```

## assertExitCode(int $exitCode)
Assert the integer exit code

```php
$this->command(ConsoleController::class, 'actionName')->assertExitCode(1337);
```

## assertSee(string $text)
Assert that the command contains the passed text in stdout or stderr

```php
$this->command(ConsoleController::class, 'actionName')->assertSee('text output');
```

## assertDontSee(string $text)
Assert that the command does not contain the passed text in stdout or stderr

```php
$this->command(ConsoleController::class, 'actionName')->assertDontSee('text output');
```
329 changes: 329 additions & 0 deletions docs/assertions/http-response.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,329 @@
# HTTP Response Assertions

A testable response is returned whenever you perform a HTTP request
with Pest. It is an extension of Craft's native Response with a
number of convience methods added for testing. For example, most
tests will perform a `get()` and want to check that the response did
not return an error. You may use `->assertOk()` to check that the
status code was 200.

## getRequest()
Get the requesr that triggered this reaponse.

## getJsonContent()
Fetch the response body as a JSON array. This can be run through the expectation API
as well for a fluent chain,

```php
$response->expect()->jsonContent->toBe(['foo' => 'bar']);
```

## querySelector(string $selector)
If the response returns HTML you can `querySelector()` to inspect the
HTML for specific content. The `querySelector()` method takes a
CSS selector to look for (just like in Javascript).

The return from `querySelector()` is always a `NodeList` containing zero
or more nodes. You can interact with the `NodeList` regardless of the return
and you will get back a scalar value or a collection of values.

```php
$response->querySelector('h1')->text; // returns the string contents of the h1 element
$response->querySelector('li')->text; // returns a collection containing the text of all list items
```

## form(?string $selector = NULL)
The entry point for interactions with forms. This returns a testable
implementaion of the [Symfony DomCrawler's Form](#) class.

If a response only has one form you may call `->form()` without any parameters
to get the only form in the response. If the response contains more than
one form then you must pass in a selector matching a specific form.

To submit the form use `->submit()` or `->click('.button-selector')`.

## expectSelector(string $selector)
Runs the same `querySelector()` against the response's HTML but instead
of returning a `NodeList` it returns an expectation against the `NodeList`.
This allows you to use Pest's expectation API against the found nodes.

```php
$response->expectSelector('h1')->text->toBe('Hello World!');
```

## assertCacheTag(string $tags)
Check that a response contains the given cache tag header. This is commonly used on
edge-side CDNs to "tag" pages with a unique identifier so that they can be purged
from the cache by that unique tag later. In Craft this will usually be something like
the element ID of any entries being rendered.

```php
$response->assertCacheTag('my-tag');
$response->assertCacheTag('el1234');
```

## assertCookie(string $name, ?string $value = NULL)
Checks that the response contains the given cookie. When not passed a value
the assertion only checks the presence of the cookie. When passed a value the
value will be checked for strict equality.

```php
$response->assertCookie('cookieName'); // checks presence, with any value
$response->assertCookie('cookieName', 'cookie value'); // checks that the values match
```

## assertCookieExpired(string $name)
Checks that the given cookie has an expiration in the past. Cookies are sent in headers and if left
unset a cookie will persist from request to request. Therefore, the only way to "remove" a cookie
is to set its expiration to a date in the past (negative number). This is common when logging people out.

```php
$response->assertCookieExpired('cookieName');
```

## assertCookieNotExpired(string $name)
Checks that the given cookie has an expiration in the future.

```php
$response->assertCookieNotExpired('cookieName');
```

## assertCookieMissing(string $name)
Checks that the given cookie is not present in the response

```php
$response->assertCookieMissing('cookieName');
```

## assertCreated()
Checks that the response has a 201 Created status code

```php
$response->assertCreated();
```

## assertDontSee(string $text)
Checks that the given string does not appear in thr response.

```php
$response->assertDontSee('text that should not be in the response');
```

## assertDontSeeText(string $text)
Checks that the given string does not appear in the response after first stripping all non-text elements (like HTML) from the response.
For example, if the response contains `foo <em>bar</em>` you could check against the text `foo bar` because the `<em>` will be stripped.

```php
$response->assertDontSeeText('foo bar');
```

## assertDownload(?string $filename = NULL)
Checks that the response contains a file download, optionally checking that the filename of the download
matches the given filename.

```php
$response->assertDownload(); // checks that any download is returned
$response->assertDownload('file.jpg'); // checks that a download with the name `file.jpg` is returned
```

## assertExactJson(array $data)
Checks that the given JSON exactly matches the returned JSON using PHPUnit's "canonicalizing" logic to
validate the objects.

```php
$response->assertExactJson(['foo' => 'bar']);
```

## assertForbidden()
Checks that the response has a 403 Forbidden status code

```php
$response->assertForbidden();
```

## assertHeader(string $name, ?string $expected = NULL)
Checks that the given header is present in the response and, if provided, that the value of the
header matches the given value.

```php
$response->assertHeader('x-foo'); // checks for presence of header, with any value
$response->assertHeader('x-foo', 'bar'); // checks for header with matching value
```

## assertHeaderMissing(string $name)
Checks that the response headers do not contain the given header.

```php
$response->assertHeaderMissing('x-foo');
```

## assertLocation(string $location, ?array $checkParts = NULL)
Checks that the location header matches the given location

```php
$response->assertLocation('/foo/bar');
```

By default the full location will be checked including the path,
host, port, etc... If you would like to check only a portion of
the location you can pass in an array of keys in the second
parameter. The keys take their names from PHP's [`parse_url`](https://www.php.net/parse_url)
function.

```php
$response->assertLocation('/foo', ['host', 'path']);
```

## assertLocationPath(string $uri)
Assert that the given path marches the path of the returned
`location` header. The other parts of the location, like the
host name, are ignored.

```php
$response->assertLocationPath('/foo');
```

## assertFlash(?string $message = NULL, ?string $key = NULL)
Check that the given message/key is present in the flashed data.

```php
$response->assertFlash('The title is required');
$response->assertFlash('Field is required', 'title');
```

## assertNoContent($status = 204)
Check that the response has the given status code and no content.

```php
$response->assertNoContent();
```

## assertNotFound()
Check that the response returns a 404 Not Found status code

```php
$response->assertNotFound();
```

## assertOk()
Check that the response returns a 200 OK status code

```php
$response->assertOk();
```

## assertRedirect()
Check that the response returns a 300 status code

```php
$response->assertRedirect();
```

## assertRedirectTo(string $location)
A sugar method that checks the status code as well as the location of the redirect.

```php
$response->assertRedirectTo('/foo/bar');
```

## followRedirect()
For a 300 class response with a `Location` header, trigger a new
request for the redirected page.

```php
$response->assertRedirect()->followRedirect()->assertOk();
```

## followRedirects()
For a 300 class response with a `Location` header, trigger a new
request for the redirected page. If the redirected page also contains
a redirect, follow the resulting redirects until you reach a non-300
response code.

```php
$response->assertRedirect()->followRedirects()->assertOk();
```

## assertSee(string $text)
Checks that the response contains the given text

```php
$response->assertSee('foo bar');
```

## assertSeeInOrder(array $texts)
Checks that the response contains the given text, in successive order

```php
$response->assertSee(['first', 'second', 'third']);
```

## assertSeeText(string $text)
Checks that the response contains the given text stripping tags. This would
pass against source code of `<b>foo</b> bar`

```php
$response->assertSeeText('foo bar');
```

## assertSeeTextInOrder(array $texts)
Checks that the response contains the given text, in successive order
while stripping tags.

```php
$response->assertSeeTextInOrder(['first', 'second', 'third']);
```

## assertStatus($code)
Asserts the given status code matches the response status code.

```php
$response->assertStatus(404);
```

## assertSuccessful()
Asserts a successfull (200-class) response code.

## assertTitle(string $title)
Assert the given title matches the title of the page.

```php
$response->assertTitle('The Title');
```

## assertUnauthorized()
Asserts that the response's status code is 401

## beginBenchmark()
Benchmarks are started on your test case by calling `->beginBenchmark()`. You are
free to start as many benchmarks as needed, however, note that starting a new
benchmark will clear out any existing benchmarks already in progress.

> **Warning**
> In order to use a benchmark you must enable Craft's `devMode` (which
will enable the Yii Debug Bar).

## endBenchmark()
Ending a benchmark returns a testable Benchmark class. You can end a benchmark
by calling `->endBenchmark()` on the test case or on a response. Either of the
following will work,

```php
it('ends on the test case', function () {
$this->beginBenchmark();
$this->get('/');
$benchmark = $this->endBenchmark();
});
```

```php
it('ends on the response', function () {
$this->beginBenchmark()
->get('/')
->endBenchmark();
});
```

> **Note**
> Unlike the traditional Craft request/response lifecycle you are
free to make multiple requests in a single benchmark.
2 changes: 1 addition & 1 deletion src/behaviors/TestableResponseBehavior.php
Original file line number Diff line number Diff line change
Expand Up @@ -17,7 +17,7 @@
use yii\base\Behavior;

/**
* # Response Assertions
* # HTTP Response Assertions
*
* A testable response is returned whenever you perform a HTTP request
* with Pest. It is an extension of Craft's native Response with a
Expand Down
Loading
Loading