Session 守卫

Session 守卫

Session 守卫使用 @adonisjs/session 包在 HTTP 请求期间登录和验证用户。

会话和 cookie 在互联网上已存在很长时间,并且适用于大多数应用程序。因此,我们建议对服务器渲染的应用程序或同一顶级域名下的 SPA 网页客户端使用 session 守卫。

配置 guard

身份验证守卫在 config/auth.ts 文件中定义。你可以在此文件的 guards 对象下配置多个守卫。

config/auth.ts
import { defineConfig } from '@adonisjs/auth'
import { sessionGuard, sessionUserProvider } from '@adonisjs/auth/session'
const authConfig = defineConfig({
default: 'web',
guards: {
web: sessionGuard({
useRememberMeTokens: false,
provider: sessionUserProvider({
model: () => import('#models/user'),
}),
})
},
})
export default authConfig

sessionGuard 方法创建 SessionGuard 类的实例。它接受一个用户提供者,该提供者可在身份验证期间用于查找用户,并且可选地接受一个配置对象来配置记住令牌的行为。

sessionUserProvider 方法创建 SessionLucidUserProvider 类的实例。它接受一个用于身份验证的模型引用。

执行登录

你可以使用守卫的 login 方法登录用户。该方法接受一个 User 模型的实例,并为用户创建一个登录会话。

在以下示例中:

  • 我们使用 AuthFinder mixin 中的 verifyCredentials 通过电子邮件和密码查找用户。

  • auth.use('web') 返回在 config/auth.ts 文件中配置的 SessionGuard 的实例(web 是你在配置中定义的守卫名称)。

  • 接下来,我们调用 auth.use('web').login(user) 方法为用户创建登录会话。

  • 最后,我们将用户重定向到 /dashboard 端点。你可以根据需要自定义重定向端点。

import User from '#models/user'
import { HttpContext } from '@adonisjs/core/http'
export default class SessionController {
async store({ request, auth, response }: HttpContext) {
/**
* 步骤 1:从请求体中获取凭据
*/
const { email, password } = request.only(['email', 'password'])
/**
* 步骤 2:验证凭据
*/
const user = await User.verifyCredentials(email, password)
/**
* 步骤 3:登录用户
*/
await auth.use('web').login(user)
/**
* 步骤 4:将他们发送到受保护的路由
*/
response.redirect('/dashboard')
}
}

保护路由

你可以使用 auth 中间件保护路由,防止未验证的用户访问。该中间件在 start/kernel.ts 文件中注册,位于命名中间件集合下。

import router from '@adonisjs/core/services/router'
export const middleware = router.named({
auth: () => import('#middleware/auth_middleware')
})

auth 中间件应用于你想要保护、防止未验证用户访问的路由。

import { middleware } from '#start/kernel'
import router from '@adonisjs/core/services/router'
router
.get('dashboard', () => {})
.use(middleware.auth())

默认情况下,auth 中间件将根据 default 守卫(如在配置文件中定义)对用户进行身份验证。但是,在分配 auth 中间件时,你可以指定一个守卫数组。

在以下示例中,auth 中间件将尝试使用 webapi 守卫对请求进行身份验证。

import { middleware } from '#start/kernel'
import router from '@adonisjs/core/services/router'
router
.get('dashboard', () => {})
.use(
middleware.auth({
guards: ['web', 'api']
})
)

处理身份验证异常

如果用户未通过身份验证,auth 中间件将抛出 E_UNAUTHORIZED_ACCESS 异常。该异常将根据以下内容协商规则自动处理。

  • 带有 Accept=application/json 头的请求将收到一个包含 message 属性的错误数组。

  • 带有 Accept=application/vnd.api+json 头的请求将根据 JSON API 规范收到一个错误数组。

  • 对于服务器端渲染的应用程序,用户将被重定向到 /login 页面。你可以在 auth 中间件类中配置重定向端点。

访问已登录用户

你可以使用 auth.user 属性访问已登录用户的实例。仅在使用 authsilent_auth 中间件,或者手动调用 auth.authenticateauth.check 方法时,该值才可用。

使用 auth 中间件
import { middleware } from '#start/kernel'
import router from '@adonisjs/core/services/router'
router
.get('dashboard', async ({ auth }) => {
await auth.user!.getAllMetrics()
})
.use(middleware.auth())
手动调用 authenticate 方法
import { middleware } from '#start/kernel'
import router from '@adonisjs/core/services/router'
router
.get('dashboard', async ({ auth }) => {
/**
* 首先,对用户进行身份验证
*/
await auth.authenticate()
/**
* 然后访问用户对象
*/
await auth.user!.getAllMetrics()
})

静默身份验证中间件

silent_auth 中间件与 auth 中间件类似,但当用户未通过身份验证时,它不会抛出异常。相反,请求将正常继续。

当你希望始终对用户进行身份验证以执行某些操作,但不希望在用户未通过身份验证时阻止请求时,此中间件非常有用。

如果你计划使用此中间件,则必须在 router 中间件 列表中注册它。

start/kernel.ts
import router from '@adonisjs/core/services/router'
router.use([
// ...
() => import('#middleware/silent_auth_middleware')
])

检查请求是否已验证

你可以使用 auth.isAuthenticated 标志检查请求是否已通过身份验证。对于已验证的请求,auth.user 的值将始终被定义。

import { middleware } from '#start/kernel'
import router from '@adonisjs/core/services/router'
router
.get('dashboard', async ({ auth }) => {
if (auth.isAuthenticated) {
await auth.user!.getAllMetrics()
}
})
.use(middleware.auth())

获取已认证用户或失败

如果你不喜欢在 auth.user 属性上使用非空断言运算符,你可以使用 auth.getUserOrFail 方法。此方法将返回用户对象或抛出E_UNAUTHORIZED_ACCESS 异常。

import { middleware } from '#start/kernel'
import router from '@adonisjs/core/services/router'
router
.get('dashboard', async ({ auth }) => {
const user = auth.getUserOrFail()
await user.getAllMetrics()
})
.use(middleware.auth())

在 Edge 模板中访问用户

InitializeAuthMiddleware 还会与 Edge 模板共享 ctx.auth 属性。因此,你可以通过 auth.user 属性访问当前登录的用户。

@if(auth.isAuthenticated)
<p> Hello {{ auth.user.email }} </p>
@end

如果你想在未受保护的路由上获取已登录用户的信息,可以使用 auth.check 方法来检查用户是否已登录,然后访问 auth.user 属性。一个很好的用例是在公共页面的网站标题中显示已登录用户的信息。

{{--
这是一个公共页面,因此不受 auth 中间件的保护。
但是,我们仍然希望在网站的标题中显示已登录用户的信息。
为此,我们使用 `auth.check` 方法来静默检查用户是否已登录,
然后在标题中显示他们的电子邮件。
你明白了吧!
--}}
@eval(await auth.check())
<header>
@if(auth.isAuthenticated)
<p> Hello {{ auth.user.email }} </p>
@end
</header>

执行注销

你可以使用 guard.logout 方法注销用户。在注销期间,用户状态将从会话存储中删除。当前有效的“记住我”令牌也将被删除(如果使用“记住我”令牌的话)。

import { middleware } from '#start/kernel'
import router from '@adonisjs/core/services/router'
router
.post('logout', async ({ auth, response }) => {
await auth.use('web').logout()
return response.redirect('/login')
})
.use(middleware.auth())

使用“记住我”功能

“记住我”功能可以在用户会话过期后自动登录用户。这是通过生成一个加密安全的令牌并将其作为 cookie 保存在用户浏览器中实现的。

用户会话过期后,AdonisJS 将使用“记住我”cookie,验证令牌的有效性,并自动为用户重新创建已登录的会话。

创建“记住我”令牌表

“记住我”令牌保存在数据库中,因此你必须创建一个新的迁移来创建 remember_me_tokens 表。

node ace make:migration remember_me_tokens
import { BaseSchema } from '@adonisjs/lucid/schema'
export default class extends BaseSchema {
protected tableName = 'remember_me_tokens'
async up() {
this.schema.createTable(this.tableName, (table) => {
table.increments()
table
.integer('tokenable_id')
.notNullable()
.unsigned()
.references('id')
.inTable('users')
.onDelete('CASCADE')
table.string('hash').notNullable().unique()
table.timestamp('created_at').notNullable()
table.timestamp('updated_at').notNullable()
table.timestamp('expires_at').notNullable()
})
}
async down() {
this.schema.dropTable(this.tableName)
}
}

配置令牌提供者

为了读写令牌,你需要将DbRememberMeTokensProvider 分配给 User 模型。

import { BaseModel } from '@adonisjs/lucid/orm'
import { DbRememberMeTokensProvider } from '@adonisjs/auth/session'
export default class User extends BaseModel {
// ...模型的其他属性
static rememberMeTokens = DbRememberMeTokensProvider.forModel(User)
}

在配置中启用“记住我”令牌

最后,让我们在 config/auth.ts 文件中启用会话守卫配置中的 useRememberTokens 标志。

import { defineConfig } from '@adonisjs/auth'
import { sessionGuard, sessionUserProvider } from '@adonisjs/auth/session'
const authConfig = defineConfig({
default: 'web',
guards: {
web: sessionGuard({
useRememberMeTokens: true,
rememberMeTokensAge: '2 years',
provider: sessionUserProvider({
model: () => import('#models/user'),
}),
})
},
})
export default authConfig

登录时记住用户

设置完成后,你可以使用 guard.login 方法生成“记住我”令牌和 cookie,如下所示。

import User from '#models/user'
import { HttpContext } from '@adonisjs/core/http'
export default class SessionController {
async store({ request, auth, response }: HttpContext) {
const { email, password } = request.only(['email', 'password'])
const user = await User.verifyCredentials(email, password)
await auth.use('web').login(
user,
/**
* 当存在“remember_me”输入时生成令牌
*/
!!request.input('remember_me')
)
response.redirect('/dashboard')
}
}

使用访客中间件

auth 包附带了一个访客中间件,你可以使用它来重定向已登录用户,防止其访问 /login 页面。这样做是为了避免在同一设备的同一用户上创建多个会话。

import router from '@adonisjs/core/services/router'
import { middleware } from '#start/kernel'
router
.get('/login', () => {})
.use(middleware.guest())

默认情况下,访客中间件将使用 default 守卫(如在配置文件中定义)检查用户的登录状态。但是,在分配 guest 中间件时,你可以指定一个守卫数组。

router
.get('/login', () => {})
.use(middleware.guest({
guards: ['web', 'admin_web']
}))

最后,你可以在 ./app/middleware/guest_middleware.ts 文件中为已登录用户配置重定向路由。

事件

请查阅事件参考指南,以查看 Auth 包发出的可用事件列表。