Symfony 的 DomCrawler 与 Laravel HTTP 测试

发布时间 作者

Symfony's DomCrawler with Laravel HTTP Tests image

您是否曾经需要在 Laravel 的 HTTP 测试 中断言 HTML 响应的一部分?我最近需要验证响应的一部分以确认重要的内容是否已渲染。例如,假设您有一个关键的 JavaScript 文件,您希望确保它包含在 DOM 中?

以下是我们将构建的测试 API

$node = $response->get('/')
->crawl()
->filter('a')
->reduce(function (Crawler $node): bool {
return $node->attr('href') === 'https://news.laravel.net.cn';
});
 
$this->assertCount(1, $node);

在 PHP 中解析和遍历 DOM 有几种选择,例如 PHP 的 DOMDocument、PHPUnit 的 DOM 断言(已弃用)、Symfony 的 DOMCrawler 组件以及其他各种组件。我个人比较喜欢 Symfony 的 DOMCrawler 组件,它具有非常强大的过滤、遍历等功能。

让我们看看如何快速将 DOMCrawler 组件集成到我们的 Laravel HTTP 测试中!

示例

使用 DOMCrawler 的关键是使用传递给构造函数的 HTML 内容创建一个新的 Crawler 实例

use Symfony\Component\DomCrawler\Crawler;
 
$crawler = new Crawler($response->getContent());
 
foreach ($crawler as $domElement) {
var_dump($domElement->nodeName);
}

直接使用 Crawler 实例可以工作,但会很重复。鉴于 Laravel 强大的宏功能,我们可以快速定义一个宏来爬取 TestResponse

以下是我的宏的使用方式

$response = $this->get('/');
 
$crawler = $response->crawl();
 
// DOM traversal
// Assertions

如果要尝试一下,请在服务提供者的 boot() 方法中定义以下宏

use Illuminate\Support\ServiceProvider;
use Illuminate\Testing\TestResponse;
use Symfony\Component\DomCrawler\Crawler;
use PHPUnit\Framework\Assert as PHPUnit;
 
// ...
 
public function boot(): void
{
TestResponse::macro('crawl', function(?callable $callback = null): Crawler {
if (empty($content = $this->getContent())) {
PHPUnit::fail('The HTTP response is empty.');
}
 
$callback ??= fn ($c): Crawler => $c;
 
return call_user_func($callback, new Crawler($content));
});
}

我们的宏执行以下操作

  1. 获取响应内容,如果内容为空,则立即使测试失败
  2. 如果用户没有使用空合并赋值运算符提供回调,则创建一个直通回调
  3. 通过回调传递一个新的 Crawler 实例,最终应返回一个 Crawler 实例

使用空合并赋值运算符,我们可以通过提供一个默认的直通回调来避免任何 if 检查,即使用户没有传递回调。

callable 的想法是,用户可以遍历 DOM,执行一些断言,并最终通过 crawl() 返回 DOM 的子集,如果只需要 DOM 的一部分。

// Return the full content
$crawler = $response->crawl();
 
// Filter and return the filtered crawler instance
$card = $response->crawl(function (Crawler $c) {
return $c->filter('a')
->reduce(function (Crawler $node) {
return $node->attr('href') === 'https://news.laravel.net.cn';
});
});

也许这个例子有点过头了,但让我们验证一下 Laravel 默认安装中包含在 welcome.blade.php 中的 Laravel 新闻 HTML

use Symfony\Component\DomCrawler\Crawler;
 
// ...
 
/**
* A basic test example.
*/
public function test_the_application_returns_a_successful_response(): void
{
$response = $this->get('/');
 
$response->assertStatus(200);
 
// Find the Laravel News node
$card = $response->crawl()
->filter('a')
->reduce(function (Crawler $node) {
return $node->attr('href') === 'https://news.laravel.net.cn';
});
 
$this->assertNotEmpty($card, 'The Laravel News homepage card was not found!');
 
// Validate that the $card node has an H2 with `Laravel News`
$this->assertEquals(
'Laravel News',
$card->filter('h2')->first()->text()
);
 
// Validate that the page shows the blurb
$blurb = $card
->filter('p')
->reduce(function (Crawler $n) {
return str($n->text())->startsWith('Laravel News is a community driven portal');
});
 
$this->assertCount(1, $blurb);
}

奖励

您可能想要编写更多便利助手来验证测试中的 DOM 元素。例如,以下是一个验证节点是否存在的基本断言

$response
->assertStatus(200)
->assertNodeExists('a[href="https://news.laravel.net.cn"]');

以下是我定义的宏,它还允许与 TestResponse 实例链式调用

TestResponse::macro('assertNodeExists', function(string $selector): static {
$node = $this->crawl()->filter($selector);
$message = "Failed asserting the node exists with selector \"{$selector}\".";
 
PHPUnit::assertGreaterThan(0, $node->count(), $message);
 
return $this;
});

如果节点不存在,以下是宏在测试输出中的示例

1) Tests\Feature\ExampleTest::test_the_application_returns_a_successful_response
Failed asserting the node exists with selector "a[href="https://laravelnews.com"]".
Failed asserting that 0 is greater than 0.

我建议您查看 Symphony DOMCrawler 组件 的功能。它包括强大的 XPath 和 CSS 选择器、节点遍历等等!

Paul Redmond photo

Laravel 新闻的专栏作家。全栈 Web 开发人员和作家。

Cube

Laravel 新闻通讯

加入 40,000 多名其他开发人员,绝不错过新的提示、教程等。

Laravel Forge logo

Laravel Forge

轻松创建和管理您的服务器,并在几秒钟内部署您的 Laravel 应用程序。

Laravel Forge
Tinkerwell logo

Tinkerwell

Laravel 开发人员必备的代码运行器。使用 AI、自动完成和对本地和生产环境的即时反馈进行测试。

Tinkerwell
No Compromises logo

无妥协

Joel 和 Aaron 是来自无妥协播客的两名经验丰富的开发人员,现在可以为您的 Laravel 项目聘用。 ⬧ 固定费率 7500 美元/月。 ⬧ 无冗长的销售流程。 ⬧ 无合同。 ⬧ 100% 退款保证。

无妥协
Kirschbaum logo

Kirschbaum

提供创新和稳定性,确保您的 Web 应用程序取得成功。

Kirschbaum
Shift logo

Shift

运行旧版本的 Laravel?即时自动 Laravel 升级和代码现代化,让您的应用程序保持新鲜。

Shift
Bacancy logo

Bacancy

使用经验丰富的 Laravel 开发人员为您的项目加油,他们拥有 4-6 年的经验,每月只需 2500 美元。获取 160 小时的专业知识和 15 天无风险试用。立即安排通话!

Bacancy
Lucky Media logo

Lucky Media

立即获得好运 - Laravel 开发的理想选择,拥有十多年的经验!

Lucky Media
Lunar: Laravel E-Commerce logo

Lunar:Laravel 电子商务

Laravel 的电子商务。一个开源软件包,将现代无头电子商务功能的力量带到 Laravel。

Lunar:Laravel 电子商务
LaraJobs logo

LaraJobs

官方 Laravel 职位招聘网站

LaraJobs
SaaSykit: Laravel SaaS Starter Kit logo

SaaSykit:Laravel SaaS 启动工具包

SaaSykit 是一个 Laravel SaaS 启动工具包,它包含运行现代 SaaS 所需的所有功能。付款、精美结账、管理面板、用户仪表板、身份验证、即用型组件、统计信息、博客、文档等等。

SaaSykit:Laravel SaaS 启动工具包
Rector logo

Rector

您无缝 Laravel 升级的合作伙伴,降低成本,加速创新,为成功企业提供助力

Rector
MongoDB logo

MongoDB

通过 MongoDB 和 Laravel 的强大集成增强您的 PHP 应用程序,使开发人员能够轻松高效地构建应用程序。支持事务、搜索、分析和移动用例,同时使用熟悉的 Eloquent API。了解 MongoDB 的灵活、现代数据库如何改变您的 Laravel 应用程序。

MongoDB
Maska is a Simple Zero-dependency Input Mask Library image

Maska 是一个简单的无依赖项输入掩码库

阅读文章
Add Swagger UI to Your Laravel Application image

将 Swagger UI 添加到您的 Laravel 应用程序

阅读文章
Assert the Exact JSON Structure of a Response in Laravel 11.19 image

在 Laravel 11.19 中断言响应的精确 JSON 结构

阅读文章
Build SSH Apps with PHP and Laravel Prompts image

使用 PHP 和 Laravel 提示构建 SSH 应用程序

阅读文章
Building fast, fuzzy site search with Laravel and Typesense image

使用 Laravel 和 Typesense 构建快速模糊站点搜索

阅读文章
Add Comments to your Laravel Application with the Commenter Package image

使用 Commenter 软件包在您的 Laravel 应用程序中添加评论

阅读文章