扩展框架

扩展框架

AdonisJS 的架构使得扩展框架变得非常容易。我们使用框架的核心 API 来构建一个第一方包生态系统。

在本指南中,我们将探讨不同的 API,你可以使用这些 API 通过包或在你的应用程序代码库中扩展框架。

宏和 getter

宏和 getter 提供了一种 API,用于向类的原型添加属性。你可以将它们视为 Object.defineProperty 的语法糖。在底层,我们使用 macroable 包,你可以参考其 README 以获取深入的技术解释。

由于宏和 getter 是在运行时添加的,因此你需要使用 声明合并 向 TypeScript 告知添加属性的类型信息。

你可以将添加宏的代码写入一个专用文件(如 extensions.ts),并在服务提供者的 boot 方法中导入它。

providers/app_provider.ts
export default class AppProvider {
async boot() {
await import('../src/extensions.js')
}
}

在下面的示例中,我们将 wantsJSON 方法添加到 Request 类,并同时定义其类型。

src/extensions.ts
import { Request } from '@adonisjs/core/http'
Request.macro('wantsJSON', function (this: Request) {
const firstType = this.types()[0]
if (!firstType) {
return false
}
return firstType.includes('/json') || firstType.includes('+json')
})
src/extensions.ts
declare module '@adonisjs/core/http' {
interface Request {
wantsJSON(): boolean
}
}
  • declare module 调用期间,模块路径必须与用于导入类的路径相同。
  • interface 名称必须与添加宏或 getter 的类名称相同。

Getter

Getter 是添加到类的惰性求值属性。你可以使用 Class.getter 方法添加 getter。第一个参数是 getter 名称,第二个参数是用于计算属性值的回调函数。

Getter 回调函数不能是异步的,因为 JavaScript 中的 getter 不能是异步的。

import { Request } from '@adonisjs/core/http'
Request.getter('hasRequestId', function (this: Request) {
return this.header('x-request-id')
})
// 你可以如下使用此属性。
if (ctx.request.hasRequestId) {
}

Getter 可以是单例,这意味着计算 getter 值的函数将被调用一次,并且返回值将为类的实例缓存。

const isSingleton = true
Request.getter('hasRequestId', function (this: Request) {
return this.header('x-request-id')
}, isSingleton)

可宏扩展的类

以下是可以使用宏和 getter 扩展的类的列表。

导入路径
Application@adonisjs/core/app
Request@adonisjs/core/http
Response@adonisjs/core/http
HttpContext@adonisjs/core/http
Route@adonisjs/core/http
RouteGroup@adonisjs/core/http
RouteResource@adonisjs/core/http
BriskRoute@adonisjs/core/http
ExceptionHandler@adonisjs/core/http
MultipartFile@adonisjs/core/bodyparser

扩展模块

大多数 AdonisJS 模块提供了可扩展的 API,以注册自定义实现。以下是聚合列表。