Skip to content

Koa 适配器

@inferdi/koa 是 Koa v3 中间件。它创建一个请求作用域,将其暴露为 ctx.state.di,并在 Node 响应 finish 或 close 之后释放它。

安装

bash
pnpm add @inferdi/inferdi @inferdi/koa koa
pnpm add -D @types/koa
ts
import Koa from 'koa'
import { inferdiKoa, type InferdiScopeOf } from '@inferdi/koa'

请求作用域

ts
const root = buildRootContainer()

declare module 'koa' {
  interface DefaultState {
    di: InferdiScopeOf<typeof root>
  }
}

const app = new Koa()

app.use(inferdiKoa({
  container: root,
  setupScope: (scope, ctx) => {
    const request = scope.get('request')
    request.requestId = crypto.randomUUID()
    request.userId = ctx.get('x-user-id') || undefined
    request.ip = ctx.ip
  },
}))

app.use(async (ctx) => {
  const id = ctx.path.split('/').pop() ?? ''
  ctx.body = await ctx.state.di.get('users').profile(id)
})

自定义状态键

ts
import type { DefaultState, ParameterizedContext } from 'koa'
import { type InferdiKoaState, type InferdiScopeOf } from '@inferdi/koa'

type AppState =
  & DefaultState
  & InferdiKoaState<InferdiScopeOf<typeof root>, 'container'>

type AppContext = ParameterizedContext<AppState>

app.use(inferdiKoa({ container: root, key: 'container' }))

app.use(async (ctx: AppContext) => {
  ctx.body = await ctx.state.container.get('users').profile('42')
})

选项

选项默认值描述
container必填根容器。该中间件从不释放它。
key'di'Koa 状态键。
createScoperoot.createScope()自定义请求作用域创建。
setupScope在下游中间件之前填充作用域。
disposeScopescope.dispose()自定义释放。
autoDisposetruefalse 或返回 false 的谓词会转移所有权。
onDisposeErrorctx.app.emit('error')清理失败的接收端。

流式

普通的 Koa 流式响应体不需要跳过。适配器会等待 finishclose

仅当应用代码有意将作用域保留到 HTTP 响应边界之外时(例如后台工作),才使用 skipInferdiDispose(ctx)

ts
import { skipInferdiDispose } from '@inferdi/koa'

app.use(async (ctx) => {
  skipInferdiDispose(ctx)
  const scope = ctx.state.di

  queue.add(async () => {
    try {
      await scope.get('jobs').run()
    } finally {
      await scope.dispose()
    }
  })

  ctx.body = { status: 'queued' }
})

下游错误总是会释放作用域;成功且被跳过的请求则归应用所有。