Laravel Fortify 不响应 HTTP 代码而是响应实际路由,即使我将注册请求作为 XHR 发送

Posted

技术标签:

【中文标题】Laravel Fortify 不响应 HTTP 代码而是响应实际路由,即使我将注册请求作为 XHR 发送【英文标题】:Laravel Fortify doesn't respond with HTTP code but with actual routes, even if I send the registration request as an XHR one 【发布时间】:2021-06-08 07:06:08 【问题描述】:

上下文

使用 Postman,为了在数据库中注册用户,我将以下字段发送到 URL http://mywebsite/register

电子邮件

密码

密码确认

名字

根据文档https://laravel.com/docs/8.x/fortify#registration(我从中找到了上述字段),Fortify 已经定义了路由register,所以我不需要自己定义。

根据文档和我的需要,我不需要创建注册表单:我直接使用 Postman 将这些注册数据发送到 Laravel Fortify 的路由 /register 作为 XHR POST 请求数据。此外,我不需要 Fortify 来返回视图,所以我禁用了它们 (https://laravel.com/docs/8.x/fortify#disabling-views)。事实上,我只是等待我将在 Postman 的返回数据中看到的 HTTP 代码响应(见下文)。

Fortify 应该返回什么

因为我已经禁用了视图并且因为我发送了一个 XHR POST 请求(在 Postman 中,我将这个 HTTP 标头与注册请求一起发送:X-Requested-With = XMLHttpRequest):

如果注册尝试成功,Fortify 会将用户重定向到通过应用程序的 fortify 配置文件中的 home 配置选项配置的 URI。 如果登录请求是 XHR 请求,则会返回 200 HTTP 响应。

如果请求不成功,用户将被重定向回注册屏幕,验证错误将通过共享的 $errors Blade 模板变量提供给您。 或者,对于 XHR 请求,验证错误将返回 422 HTTP 响应。

(https://laravel.com/docs/8.x/fortify#registration)

Fortify 实际返回的内容

它正确地注册了用户。但是我在 Postman 中看到的结果是 Symfony\\Component\\HttpKernel\\Exception\\NotFoundHttpException。事实上,Fortify 正在尝试访问主视图。

问题

为什么 Fortify 似乎仍然试图访问返回的视图路由(未定义,因为我不需要它们),因为我的请求是正确的 XHR POST 请求并且我已经禁用了 Fortify 配置文件中的视图?

【问题讨论】:

【参考方案1】:

我想我在 Fortify 文档 (https://laravel.com/docs/8.x/fortify#registration) 中发现了一些“有趣”的东西。那里有些东西是不完整的。我想我已经找到了文档中缺少的内容!

我解释一下。

在我的例子中,一个手机应用会显示一个注册表单。它将 Fortify 的注册必填字段(如 email, password, confirmation_password, ...)发送到由 Fortify 定义的 URL <LaravelSite>/register。 Fortify 成功注册用户并将他重定向到主路由。 但是,我没有定义任何主路由,因为正如您所理解的,这个 Laravel 站点是一个 API。 结果 => 我有这个错误:Symfony\\Component\\HttpKernel\\Exception\\NotFoundHttpException

我已关注文档:我已禁用视图(在 Fortify 配置文件中)。我已正确指定电话应用程序(实际上是 Postman)发送的请求是 XHR 请求(实际上,我已发送标头 X-Requested-With = XMLHttpRequest)。 所以 Fortify 应该返回给我一个 HTTP 代码(这里是200),而不是试图返回我的主页 URL。

文档中缺少什么? =====>>>> 中间件RedirectIfAuthenticated 被执行。在其句柄方法中有一个重定向:return redirect(RouteServiceProvider::HOME);。为了使 Fortify 正常工作,必须通过注释禁用此行。

Fortify 文档应该包含与上述句子类似的内容。

应在其 Github 存储库中创建 PR 或问题。

编辑:我们还必须发送这个 HTTP 头:Accept: application/json(否则在注册成功的情况下,它仍然会显示 404)

【讨论】:

【参考方案2】:

将配置变量views设置为false将不起作用,因为它只会禁用渲染fortify的视图文件,如登录页面、注册页面、重置密码页面等。

Fortify 路由默认使用中间件 [web],您需要在将视图设置为 false 的同一配置文件中将此中间件值设置为 [api]。

【讨论】:

如果我使用api 而不是web,我有这个错误:“参数 1 传递给 Laravel\\Fortify\\Http\\Controllers\\RegisteredUserController::__construct() 必须是Illuminate\\Contracts\\Auth\\StatefulGuard 的实例,Illuminate\\Auth\\TokenGuard 的实例给定"【参考方案3】:

顺便说一句,我遇到了完全相同的问题。经过 30 分钟的在线搜索,我最终检查了我信任的 Laracast 数据库,并且确定我找到了一个 5 分钟的视频,向我展示了如何(以某种方式)解决问题。

https://laracasts.com/series/laravel-authentication-options/episodes/15

简单地说,您需要做的就是将register 路由添加到您的routes/api.php 文件中。路由的代码可以从 Fortify 供应商目录中复制: vendor/laravel/fortify/routes/routes.php。复制第 72/73 行的注册路线。将其粘贴到api.php。立即将您的请求发布到api/register,它会像在视频中一样工作。

希望有帮助

【讨论】:

【参考方案4】:

问题在于 @JarsOfJam-Scheduler 所述的 RedirectIfAuthenticated 中间件。您可以通过添加检查在 RedirectIfAuthenticated 中间件中禁用 xhr 请求的重定向。

if(!$request->wantsJson())
    return redirect(RouteServiceProvider::HOME);

别忘了给 axios 添加标题

'Accept': 'application/json',
'Content-Type': 'application/json'

【讨论】:

以上是关于Laravel Fortify 不响应 HTTP 代码而是响应实际路由,即使我将注册请求作为 XHR 发送的主要内容,如果未能解决你的问题,请参考以下文章

Laravel Fortify 登录后重定向,但不登录用户

混合清单不存在 - 在 Hostgator 共享主机中部署 laravel 时 Laravel Fortify Jetstream 登录/注册问题

使用 Laravel Livewire 和 Laravel Fortify 进行登录

Laravel 8 Fortify 登录显示 429 太多请求

Laravel Fortify 自定义身份验证重定向

如何在同一个网站的多个地方使用 laravel fortify 进行身份验证