blog.euxn.me

これから NestJS をはじめようとしている方へ

2019-12-31 Tue.

この記事は NestJS アドベントカレンダー 2019 の 25 日目の枠です。

はじめに

NestJS の知名度は 2019 年を通して上昇したように感じられますが、まだまだ日本語情報が少ないという思いからこのアドベントカレンダーを開始しました。

まず、 NestJS とは何か、どういうユースケースにマッチするか、という紹介については以下のスライドにまとまっています。

What is NestJS? / @potato4d https://speakerdeck.com/potato4d/what-is-nestjs-number-nestjs-meetup

次に 1 日目 〜 7 日目では、 NestJS を触ってみてアプリケーションを作るまでの具体的な作法を、コードを交えて紹介しています。 8 日目以降は NestJS を使っていて当たりがちなつまづきポイントなど、1 記事完結の tips をテーマとして構成しました。

以下では、 NestJS に対して思われがちと考えられる疑問について回答をします。私の主観も入っている箇所はありますが、 NestJS を選定する際の参考にしてみてください。

NestJS って Angular っぽくない? / Angular やってないと分からなくない?

NestJS が Angular の影響を受けている側面が大きいことは否定しません、 Angular 経験者の方が馴染み易いとは思います。 しかし、 Angular の経験者向け/経験者でないと難しい、ということは一切ありません。 特に Angular に似ている部分としては Module と DI 、 Decorator かと思いますが、このうち DI と Decorator に関しては TypeScript や Java 等のバックエンドフレームワークでの開発を行なった方であれば、経験している方も多いかと思います。 Module についても、 JavaScript の Module システムのスコープが広がりすぎてしまうものを厳密に定義し、 DI に役立てている程度のものだという理解をしておけば基本的には問題ありません。 Module と DI については NestJS の Module と DI を理解する を読んでいただければ、理解の助けになるのではないかと思います。

NestJS ってデカくない?

Express 比では相当に大きく感じますが、上記のスライドにある通り、 Express のみで効率的な開発を維持するには現実的には制約が多いです。 得に、 Express には定められたアーキテクチャが存在しないため、アーキテクトとなる開発者がチームを離脱した際に路頭に迷うという話も聞きます。 上記で Angular の経験が無くても良いという断りをした上での引用ですが、 Angular での開発を快適に進めるために知っておきたいこと100 点か 50 点か、それより全員 80 点を と仰っているのは NestJS にも当てはまる考え方であり、フレームワークのもたらす恩恵というものが自分たちにとって有益であるかどうか、ということを軸に採用を判断するのが良いと思います。

また、バックエンドフレームワークとしては現代の多様化するフロントエンドやデータベースに適する形であり、 UI 層と永続化層を内包していないため、任せられる責務は実は限定的であります。

Swagger でのドキュメント生成って本当にいいの?

手でドキュメントをメンテナンスされるのが正しい情報が保たれるとは思いますが、メンテナンスされないドキュメントが残るよりは、自動生成で拙くてもコードに距離が近い場所でメンテナンスされるドキュメントがある方が健全なプロジェクトだとは思います。 Swagger については NestJS の @nestjs/swagger でコントローラーから Open API(Swagger) の定義書を生成する と、以下のスライドが参考になると思います。

NestJS アプリケーションから Swagger を自動生成する / @euxn23

https://speakerdeck.com/euxn23/nestjs-meetup-tokyo-01

NestJS って遅くない?

まず、バックエンドアプリケーションの遅さの性質としては大きく分けて 2 つあります。

  • アプリケーションの起動速度が遅い
  • レスポンス速度が遅い

Express をベースとしたアプリケーションと比較した際に、アプリケーションの起動速度の遅さは目立つかもしれません。それゆえ、 firebase functions や lambda で使用するのには向いているとは言えませんが、そもそも FaaS を使用する場合は NestJS は選択肢に上がらないと思います。 レスポンス速度については、 Express 比でわずかに遅いところはありますが、そもそも Express がいわゆる LL としては十分に速すぎるので、実用上ほぼほぼ問題のない速度でレスポンスが返ります。 どちらかというと永続化層の方がネックになるかと思われますが、それでもアプリケーション層に不安が残る場合は想定されるリクエスト数に対してのパフォーマンステストを行なって判断するのが良いと思います。

NestJS と ClassValidator / TypeORM って一緒に使わないといけないの?

TypeScript であり、 Decorator をベースとした記述が可能という点を除いて、特に関連性のあるものではありません。 必要に応じて採用したりしなかったりするのが良いと思います。

Decorator が不安

まず Decorator が内部で何を行なっているかについては TypeScript の Decorator と継承 で説明しています。 Decorator の仕様自体が不安定な中で先行実装があることについて不安はあるかと思いますが、 TypeScript のビルドが後方互換を保つ限りは動作しなくなるということはないでしょう。 Parameter Decorator で副作用を起こして動作する Class Validator 等のライブラリへの不安はあるかと思いますが、 NestJS を通常の範囲で使う上では、 Parameter Decorator は (Swagger 等の Production に不要なものを除いて) 不要なので、ここは懸念する必要はありません。 また、 Decorator を用いるライブラリを組み合わせる場合に Decorator Hell になることへの懸念については、 TypeScript の Decorator Hell を解消する で 1 つの解決へのアプローチを書いているので、ご参考にしてください。

サーバサイド TypeScript に不安がある

フロントエンドでの webpack を使用したビルドとは異なり不安になる箇所もあるかと思いますが、 NestJS ではプロダクションのビルドまでサポートしています。 また、 tsconfig-paths がバックエンドで使えない、という経験があるかと思いますが tsconfig の path alias 解決に tsconfig-paths/register を node で使う方法と TS 依存の分離方法 で解決方法を紹介していますのでご参考ください。

まとめ

以上で NestJS の概要および疑問点のご説明をしました。他にも NestJS についても疑問点がある場合は、 NestJS Japan Users Group の Discord でお気軽にお聞きください。 また、現在 日本語ドキュメントについても準備を進めている段階です。 本家の公式ドキュメントの情報量が膨大で、基本的なことはそちらにも書いてあるため、日本語化することによってより多くの方に届けられればと思っています。

謝辞

この度 NestJS Japan Users Group としても活動を開始して、当初私と @potato4d の 2 名だけで書く予定だったこのアドベントカレンダーも、何件も書いて頂きました。ご協力頂いたみなさまありがとうございました。 今後も NestJS が必要な人に正しく届くよう、情報発信を継続して行なっていきたいと思いますので、是非ご協力頂ければと思います。

Other Works
2024-12-01 Sun.
OpenAPI Spec を出力できる DSL、TypeSpec の実践例
- ドワンゴ教育サービス開発者ブログ

2024-11-16 Sat.
型付き API リクエストを実現するいくつかの手法とその選択
- TSKaigi Kansai 2024

2024-09-10 Tue.
corepack が標準同梱じゃなくなる未来、 mise でパッケージマネージャを管理する
- Zenn

2024-09-10 Tue.
言語環境の管理は *env や *vm を超えて、 mise へ
- Zenn

2024-06-28 Fri.
TypeSpec を使い倒してる
- Kyoto.js 22

2024-05-11 Sat.
Powerfully Typed TypeScript
- TSKaigi 2024

2024-05-10 Fri.
pnpm の node_modules を探検して理解しよう
- ドワンゴ教育サービス開発者ブログ

2024-03-17 Sun.
neverthrow で局所的に Result 型を使い、 try-catch より安全に記述する
- Zenn

2023-12-20 Wed.
レガシーブラウザ向けのビルドオプションを剪定する
- ドワンゴ教育サービス開発者ブログ

2023-05-26 Fri.
Next.js で dynamic import を使い Client だけで動かす Component を実現する
- Zenn

2023-05-02 Tue.
Node.js でファイル名から拡張子を取り除く/取り出すために path.parse を使う
- Zenn

2023-02-27 Mon.
WSL2 で外部からアクセス可能にするために bridge mode を有効にする
- Zenn

2023-01-26 Thu.
init.vim & dein から init.lua & lazy.nvim へ、シンプル設定で移行した
- Zenn

2023-01-13 Fri.
kindle の本をブクログ形式の csv でエクスポートする@2023初春
- Zenn

2023-01-10 Tue.
自宅サーバの移設に際して docker から nerdctl に移行した
- Zenn

2023-01-10 Tue.
自宅サーバを rootless に移行した際のトラブル対応
- Zenn

2021-11-11 Thu.
並列実行した Promise で throw されても全てハンドルしたいときの方法(allSettled, finally, etc...)
- Zenn