Next.jsの型を厳密に定義しなおしてロジックのミスを発見する
これは、株式会社ゆめみ Advent Calendar 2024 13日目の記事です。
Next.js (Pages Router) のAPI Routesのhandlerにおいて、ValibotやZodで値の検証をせずにリクエストボディの値を使用してしまう不具合を、型検査レベルで防ぐことを考えます。
例えば、次のようなコードで、処理の順番やロジックの誤りを型エラーを出して気づきたいといったものです。
import { NextApiHandler } from "next";
import * as v from "valibot";
const RequestBodySchema = v.object({
someValue: v.string(),
});
const handler: NextApiHandler = async (req, res) => {
// バリデーション前に参照! 型エラーが出てほしい
console.log(req.body.someValue);
// if文の条件が逆になっている
if (v.is(RequestBodySchema, req.body)) {
res.status(400).send({ message: "bad request" });
return; // ここの早期 return も忘れるかも……
}
// 逆になったif文のせいで、バリデーション前に参照してしまっている! 型エラーが出てほしい
// 前の早期 return が忘れてしまっていた場合も、バリデーション前に参照してしまうことに…
console.log(req.body.someValue);
};
export default handler;
そのままでは、 Next.jsの型付けにより req.body
(NextApiRequest["body"]
) の型は any
になるため、req.body.someValue
などのように検証なしで参照しても型エラーは出ません。
そこで、req.body
の型を unknown
として上書きすることで、先述のような検証なしでの req.body
の参照が型エラーとする方法を考えます。
より厳密な型を定義する
Next.jsが提供する型 NextApiRequest
を上書きした型 StrictNextApiRequest
と、それを用いて NextApiHandler
と同様に StrictNextApiHandler
を作成します。
import { type NextApiRequest } from 'next';
export type StrictNextApiRequest = Omit<NextApiRequest, 'body'> & {
body: unknown;
};
import { type NextApiResponse } from 'next';
import { type StrictNextApiRequest } from './StrictNextApiRequest';
export type StrictNextApiHandler = (req: StrictNextApiRequest, res: NextApiResponse) => void | Promise<void>;
そして、ハンドラーの定義にStrictNextApiHandler
を用いるようにします。
import * as v from "valibot";
import { type StrictNextApiHandler } from "@/utils/next/StrictNextApiHandler"
const RequestBodySchema = v.object({
someValue: v.string(),
});
const handler: StrictNextApiHandler = async (req, res) => {
// バリデーション前に参照! 型エラーが発生
console.log(req.body.someValue);
// if文の条件が逆になっている
if (v.is(RequestBodySchema, req.body)) {
res.status(400).send({ message: "bad request" });
return; // ここの早期 return も忘れるかも……
}
// 逆になったif文のせいで、バリデーション前に参照してしまっている! 型エラーが発生
console.log(req.body.someValue);
};
export default handler;
これで、型エラーが起こるようになりました。
ESLint を使って元の型の使用を防ぐ
これで StrictNextApiHandler
を使用している限りは検証なしの参照を防ぐことができました。しかし、ミスやこの情報を知らない開発者によって、Next.jsからインポートした NextApiHandler
を使用した場合は、依然としてこのチェックをすり抜けてしまいます。
そこで、NextApiHandler
や NextApiRequest
を使用した時に、自動で StrictNextApiHandler
や StrictNextApiRequest
を使うよう警告するようにする方法を考えます。
ここでは、ESLintの no-restricted-syntaxルール を使用します。このルールでは、抽象構文木を表す AST selectors を指定することにより、特定の構文の使用を禁止できます。
この場合は、NextApiHandler
やNextApiRequest
といったname
を持ったIdentifier
を禁止するため、次のように記述します。
{
"rules": {
"no-restricted-syntax": [
"error",
{
"selector": "Identifier[name=NextApiHandler]",
"message": "`NextApiHandler` is forbidden. Use `StrictNextApiHandler` instead."
},
{
"selector": "Identifier[name=NextApiRequest]",
"message": "`NextApiRequest` is forbidden. Use `StrictNextApiRequest` instead."
}
]
}
}
この状態でエディタ上やCLIでESLintを実行すると、`NextApiHandler` is forbidden. Use `StrictNextApiHandler` instead.
などのエラーになります。
ちなみに、狙ったSelectorを決めるためにはtypescript-eslintのPlaygroundが便利でした。コードを入力しながら「ESTree」タブであたりをつけて「ESQuery filter」に入力して狙った箇所が選択できているかを確認できます。
「ESTree」タブでNodeのtypeとattributeを調べている。1つめのimport文のNextApiHandler
型にフォーカスが当たっており、その箇所に対応する AST Node がハイライトされている。
Selectorを入力して、該当の要素が選択できているかを確認している。
余談
リクエストボディから値を参照する際、直接 req.body
を参照するのではなく、スキーマと元のデータをもとに新たに値を組み立て、必ずその値を使用するようにして、不正な値の使用を防ぐこともできます。この考え方をParse, don’t validate(和訳記事)といい、ValibotやZodではこれに基づいた parse
や safeParse
といったAPIを提供しています。
しかし、パースによって得られた値を使わずに直接 req.body
を参照してしまうとエラーは起こらないため、型を定義しなおす対応は別途必要になります。
import { type NextApiHandler } from "next";
import * as v from "valibot";
const RequestBodySchema = v.object({
someValue: v.string(),
});
const handler: NextApiHandler = async (req, res) => {
const reqestBodyParseResult = v.safeParse(RequestBodySchema, req.body);
// if文の条件が逆になっている
if (reqestBodyParseResult.success) {
res.status(400).send({ message: "bad request" });
return;
}
// reqestBodyParseResult.output.someValue はunknownなのでエラー!
console.log(reqestBodyParseResult.output.someValue);
// 直接 req.body を参照してしまうと、エラーにならない
console.log(req.body.someValue);
};
export default handler;
コードの構造や安全性の面でもパースを使用することは有用なのですが、ここで扱う問題への完全な解決策ではないため、余談とします。
まとめ
この記事では、Next.jsが提供する型をより厳密にしたものを新たに定義し、その型を使用することを強制する方法を紹介しました。今回の問題やフレームワークに限らず、この方法が有用な機会はあると考えています。今回紹介した内容があなたの開発の一助となれば幸いです。
Discussion
Next の型を拡張して以下のような型定義を tsconfig.json に読ませてあげると import 先を変更せずに型安全にできそうです。公式の型を上書きするのは良し悪しあるのでこちらが絶対に良いとは言えませんが、テクニックのひとつとして参考までに