使用不同语言环境测试 API 验证错误

发表于 作者

Testing API Validation Errors With Different Locales image

您是否曾经想过如何在不同的语言环境下提供 API 验证错误和翻译?这篇初学者文章将向您展示如何在无状态的 Laravel API 中设置语言环境以及测试各种语言环境验证消息。

相关的是,如果您想快速将翻译拉入您的项目以进行验证,我们已经介绍了一个名为 Laravel-lang - Laravel 翻译 的软件包。Larvel-lang 包含 68 种语言的翻译,包括身份验证和验证(以及其他)的翻译。

现在,我们将跳过安装此软件包,并从 Laravel 提供的用于语言环境支持的内置功能开始。

入门

首先,我们将创建一个新的 Laravel 项目来开始。本教程不会利用任何语言环境软件包。

laravel new locale-api
cd locale-api

接下来,让我们创建将用于演示无状态 API 中语言环境工作原理的文件。

php artisan make:middleware DetectLocale
php artisan make:controller ProfileController
php artisan make:test LocaleDetectionTest
 
mkdir -p resources/lang/es/
touch resources/lang/es/validation.php

我们将通过中间件设置 API 语言环境,该中间件将尝试设置首选语言环境。我们创建了一个控制器和测试,以便我们可以演示一些可以在 API 中执行的验证测试。

最后,我们正在为西班牙语创建一个新的验证文件,以演示 API 验证错误的本地化。

我们不会安装完整的 Laravel-lang 软件包,但对于本教程,请获取 es/validation.php 的内容并将其添加到我们上面创建的 resources/lang/es/validation.php 文件中。

中间件

虽然我通常使用 TDD,但我们将先创建中间件和控制器,然后创建一个测试作为使用中间件的一种方式。DetectLocale 中间件对于我们的目的很简单,因为 API 是无状态的;我们将依赖用户在请求中发送 Accept-Locale 标头。您也可以检查查询字符串参数以设置语言环境,但我们将保持简单。

<?php
 
namespace App\Http\Middleware;
 
use Closure;
 
class DetectLocale
{
/**
* Handle an incoming request.
*
* @param \Illuminate\Http\Request $request
* @param \Closure $next
* @param mixed ...$locales
* @return mixed
*/
public function handle($request, Closure $next, ...$locales)
{
$locales = $locales ?: ['en'];
 
if ($language = $request->getPreferredLanguage($locales)) {
app()->setLocale($language);
}
 
return $next($request);
}
}

我们的中间件获取一个 $locales 配置,它是一个支持语言的数组。然后,我们调用 Symfony HTTP Foundation 组件提供的超级方便的 Request::getPreferredLanguage() 方法,该方法获取可用的语言环境的排序数组并返回首选语言环境。

如果该方法返回首选语言环境,我们通过 Laravel 的 App::setLocale() 方法设置它。

接下来,我们在 app/Http/Kernel.php 文件中配置中间件。

protected $routeMiddleware = [
// ...
'locale' => \App\Http\Middleware\DetectLocale::class,
];

最后,在同一个文件中,我们将 locale 中间件添加到 api 组,并使用首选语言环境。

protected $middlewareGroups = [
// ...
'api' => [
'throttle:60,1',
'bindings',
'locale:en,es',
],
];

控制器

我们不会编写完整的控制器;我们只将创建一个简单的验证检查来触发一些验证规则。

<?php
 
namespace App\Http\Controllers;
 
use Illuminate\Http\Request;
 
class ProfileController extends Controller
{
public function store(Request $request)
{
$request->validate([
'name' => 'required',
'email' => 'required|email'
]);
 
return response()->json(['status' => 'OK']);
}
}

此控制器将验证名称和电子邮件输入,如果验证失败,将抛出异常,返回 422 Unprocessable Entity 状态代码和 JSON 验证错误。

接下来,我们需要在 routes/api.php 中定义路由。

<?php
 
Route::post('/profile', 'ProfileController@store');

接下来,我们将使用我们在 LocaleDetectionTest 中创建的功能测试中的示例测试用例。

<?php
 
namespace Tests\Feature;
 
use Illuminate\Http\Response;
use Tests\TestCase;
 
class LocaleDetectionTest extends TestCase
{
/**
* A basic feature test example.
*
* @return void
*/
public function testExample()
{
$response = $this
->withHeaders([
'Accept-Language' => 'es',
])
->postJson('/api/profile', [
'email' => 'invalid.email',
]);
 
$response
->assertStatus(Response::HTTP_UNPROCESSABLE_ENTITY)
->assertJsonPath('errors.email', ['correo electrónico no es un correo válido'])
->assertJsonPath('errors.name', ['El campo nombre es obligatorio.']);
}
}

我们的测试向我们的个人资料端点发送 JSON POST 请求,并使用 es(西班牙语)发送 Accept-Language 标头。正如您在运行测试时所看到的,当我们使用在 Laravel 6.0.4 中添加的新的 assertJsonPath() 进行断言时,测试通过了。

如果将 JSON 作为数组转储出来,您可以看到消息是英文的。

dd($response->decodeResponseJson());
..array:2 [
"message" => "The given data was invalid."
"errors" => array:2 [
"name" => array:1 [
0 => "El campo nombre es obligatorio."
]
"email" => array:1 [
0 => "correo electrónico no es un correo válido"
]
]
]

我们不会在本教程中深入探讨解决这个问题,但也许我们会写一篇后续文章,更详细地介绍 API 其他部分的翻译。一个简单的解决方案可能是捕获 ValidationException 消息并使用消息的翻译。

另一件需要考虑的事情是尝试一下当您发送不支持的语言时会发生什么。我给您一个提示:Symonfy 的 getPreferredLanguage() 将返回您传递给它的第一个首选语言(在本例中为 en,它是数组中的第一个)。尝试更改首选语言环境的顺序,并使用不支持的语言环境进行测试。

好了,这就是我们对在 Laravel API 中设置语言环境的简单方法的简要介绍。我要感谢 Yaz Jallad (@ninjaparade),他昨天在和我讨论这个话题。从我们的讨论中,我对如何在 API 中添加验证翻译感到好奇——我以前从未做过——并想在稍微研究之后写一篇简短的文章。

要了解有关 Laravel 中本地化的更多信息,请查看 本地化文档

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

只需 $2500/月,即可为您的项目配备经验丰富的 Laravel 开发人员,拥有 4-6 年的经验。获得 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 Prompts 构建 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 应用程序添加评论

阅读文章