0
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

Nuxt3のserver middlewareで追加したevent contextに型を付けたい

Posted at

導入

今日の話題はNuxt3のserver middlewareで追加したevent contextにTypeScriptの型を付けたいという内容です。
(Nuxt3のバージョンは3.6で確認)

※ 私が株式会社愛宕 Advent Calendar 2023に書く記事は主に社内向けに共有しておきたいけど勉強会をするまでもないちょっとしたTipsにしたいと思います。

Nuxt3ではサーバーサイドの実装でミドルウェアを作成しリクエストごとに共通的に呼び出される処理を記述することができます。
さらに、このミドルウェアの内部でevent contextに何らかの値を代入すると各APIで利用できるようになって便利です。
例えば、認証済かどうかチェックしてセッションオブジェクトを構築しcontextに紐づけておくなどの使い方が一般的でしょう。

server/middleware/auth.ts
export default defineEventHandler(async (event) => {
  const sessionId = getCookie(event, "cookie-name")
  // 取得したデータを使って何かの構築をしたり判定をしたり
  // ...中略...
  event.context.hoge = new Hoge(sessionId) // 各APIで共通的に使えるように代入しておく
})

ここで、event.contextのインターフェースにはhogeはないですが、extends Record<string, any>で宣言されているため一応hogeというプロパティは使用できます。
ただし、型がanyになってしまい、各APIで使用するときに不便です。
正しくHoge型を付けるにはどうすればいいんでしょうか?

解決

TypeScriptでは既存の型を拡張することができます。event.contextはh3モジュールのH3EventContextというインターフェースなので次のようにすれば型を付けることができます。

server/middleware/auth.ts
declare module 'h3' {
  // 普通にここにインターフェースを書けばマージしてくれる
  interface H3EventContext {
    hoge: Hoge
  }
}

同様に、サーバーサイドのプラグインでnitroAppにデータを埋め込んで各APIでuseNitroApp()を呼び出して再利用することもできます。ただし、こちらはextends Record<string, any>ではないのでこういう使い方は非推奨なのかもしれません。

server/plugins/dependencies.ts
export default defineNitroPlugin((nitroApp) => {
  // ...省略...
  nitroApp.dependencies.push(hogeClient)
})

nitroAppはnitropackモジュールのNitroAppというインターフェースなので次のようにすれば型を付けることができます。

server/plugins/dependencies.ts
declare module 'nitropack' {
  // 普通にここにインターフェースを書けばマージしてくれる
  interface NitroApp {
    dependencies: Dependency[]
  }
}
0
0
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
0
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?