フューチャー技術ブログ

RFC 7807 から RFC 9457 で Web API のエラー表現で変わったこと

はじめに

TIG(Technology Innovation Group)の真野です。

HTTP API のエラーレスポンスを標準化する仕様、RFC 7807 “Problem Details for HTTP APIs” は 2016 年 3 月に公開され、以降 Web API のエラー表現のデファクトとして広く参照されてきました。その RFC 7807 ですが、約 7 年後の 2023 年 7 月に RFC 9457 に置き換えられていたのはご存知でしょうか。

RFC 9457 は後方互換で、差分は思ったより少ないのです。それでも「何が変わって、なぜ変わったのか」を押さえておくと、これから Web API のエラー設計をする際に判断しやすくなるでしょう。本記事では両 RFC の概要と差分、そして今後の設計指針を整理します。

RFC 7807: Problem Details の標準化 (2016年)

RFC 7807 は 2016年3月に公開された Proposed Standard です。

API ごとにエラーの表現方法がバラバラで、クライアント側が毎回パース方法を変える必要があるという悩みを解決するために作られました。RFC 7807 は以下の 5 つのメンバと、application/problem+json というメディアタイプを定義しています。

メンバ 役割
type 問題の種類を識別する URI
title 人間向けの短い要約
status HTTP ステータスコード
detail 問題の詳細説明
instance 特定の発生インスタンスを示す URI

サンプルとして、2 歳になる我が子に「早くお風呂に入ろう」とリクエストしたところ、「今はおもちゃで遊びたい」と 400 エラーで返された状況を problem details で表現してみます。なお、RFC 原文のサンプルは「残高不足」を扱っています。

HTTP/1.1 400 Bad Request
Content-Type: application/problem+json
Content-Language: ja

{
"type": "https://example.com/probs/busy-with-toys",
"title": "今はお風呂に入れません。",
"detail": "今はおもちゃで遊びたい気分なので、もう少し待ってください。",
"instance": "/children/myson/bath-requests/20260420-0730",
"status": 400
}

この「共通フォーマットで機械可読なエラーを返す」というアイデアは普及し、多くのフレームワーク・ライブラリが対応しました。

RFC 9457: 7807 を置き換える改訂版 (2023年)

RFC 9457 は 2023年7月に公開された Proposed Standard です。RFC 7807 を obsolete(廃止)する位置づけになっています。

Abstract にはこう書かれています。

This document defines a “problem detail” to carry machine-readable details of errors in HTTP response content to avoid the need to define new error response formats for HTTP APIs. This document obsoletes RFC 7807.
(Google翻訳)
この文書は、HTTP API 用の新しいエラー応答フォーマットを定義する必要性を回避するために、HTTP レスポンスコンテンツにエラーの機械可読な詳細情報を含めるための「問題の詳細」を定義します。この文書は RFC 7807 を廃止します。

「新しいフォーマットを定義するものではない」と明記されており、つまりフォーマットは 7807 と同じです。 完全な後方互換性が保たれている ため、7807 に準拠して実装したレスポンスはそのまま 9457 準拠として有効です。

変更点

RFC 9457 の Appendix D “Changes from RFC 7807” が挙げる変更点は3点です。

  1. Section 4.2 で、共通の問題型 URI を登録する IANA レジストリを新設した
  2. Section 3 で、複数の問題をどう扱うべきかを明確化した
  3. Section 3.1.1 で、デリファレンスできない type URI の使用に関するガイダンスを追加した

加えて Section 5 の Security Considerations もやや強化されています。新しいフィールドの追加や、既存フィールドの削除は一切ありません。

改善点①: type URI のガイダンス明確化

7807 時代の課題

RFC 7807 は「type URI はデリファレンス可能である必要はない」と書いていました。しかし実装者からすると、

  • 相対 URI を使ってよいのか?
  • 存在しない URI を書いてよいのか?
  • 人間向けのドキュメントはどこに置けばいいのか?

といった具体的な判断材料が不足しており、迷いやすい状態でした。結果として、各社で運用がバラけたわけです。

9457 での改善

RFC 9457 の Section 3.1.1 は以下の点を明確化しました。

  • type はあくまで識別子であり、HTTP で取得できなくてもよい
  • 可能ならその URI に人間向けの説明(HTML)を置くことが望ましい
  • 型固有の拡張メンバの仕様は、別途ドキュメント化すべき

これで、社内向け API で相対 URI や URN を type に使う設計が、公式にお墨付きを得た形です。

具体例

子どもが夕食前に「ゼリーを食べたい」とリクエストしたものの、「夕食前のおやつはダメ」ルールで却下された、というケースを例に見てみます。

❌ Before (7807 時代): 絶対 URI でないと不安で、存在しない URL をわざわざ書く

{
"type": "https://api.example.com/problems/snack-before-dinner",
"title": "夕食前のおやつは食べられません",
"detail": "夕食まであと 30 分です。ゼリーは夕食の後に食べましょう。",
"status": 422
}

この URL を開いても 404 なのですが、それでも「絶対 URI にしておけば安全かな」と書きがちでした。

✅ After (9457): 相対 URI や URN を堂々と使える

{
"type": "/problems/snack-before-dinner",
"title": "夕食前のおやつは食べられません",
"detail": "夕食まであと 30 分です。ゼリーは夕食の後に食べましょう。",
"status": 422
}

URN でも OK です。

{
"type": "urn:example:problems:snack-before-dinner",
"title": "夕食前のおやつは食べられません",
"detail": "夕食まであと 30 分です。ゼリーは夕食の後に食べましょう。",
"status": 422
}

備考

「どんなドキュメントを、どこに、どの形式で置くか」まで踏み込んだ指針は RFC には書かれていません。そこは各組織の設計判断に委ねられています。

改善点②: 複数エラーの扱い

7807 時代の課題

フォームバリデーションのように「複数の問題を同時に返したい」場面に対する指針が、RFC 7807 にはありませんでした。結果、各フレームワーク・各社が独自拡張を持つことになりました。

  • Spring の invalid-params
  • JSON API 系の errors
  • 独自の violations / issues など

命名も構造もバラバラで、相互運用性が弱かったわけです。

9457 での改善

RFC 9457 の Section 3 が、まず次の方針を示しました。

When a problem in an HTTP response has multiple problems, it is RECOMMENDED that a single problem be chosen to represent the response, preferring the most relevant or urgent problem.
(Google翻訳)
HTTPレスポンスに複数の問題が含まれている場合、レスポンスを代表する問題として、最も関連性が高い、または緊急性の高い問題を優先して選択することが推奨されます

レスポンス全体としては単一の problem details を選ぶのが推奨、という方針です。そのうえで、関連する複数の問題を伝えたい場合は拡張メンバに含めればよいとし、Section 3.2 で errors 配列を拡張メンバのとして具体的に示しています。

朝の幼稚園準備で、検温忘れ・水筒が空・連絡帳未記入、名札がない靴下の利用という 4つのミスが一度に検出された、というケースを例にしてみます。

✅️具体例
HTTP/1.1 400 Bad Request
Content-Type: application/problem+json
Content-Language: ja

{
"type": "/problems/kindergarten-preparation-failed",
"title": "幼稚園の持ち物に不備があります",
"errors": [
{ "detail": "朝の検温がまだ終わっていません", "pointer": "#/healthCheck/temperature" },
{ "detail": "水筒が空のままです", "pointer": "#/items/waterBottle" },
{ "detail": "連絡帳の家庭欄が未記入です", "pointer": "#/items/communicationNotebook" },
{ "detail": "名札がついていない靴下を履いています", "pointer": "#/clothing/socks" }
]
}

備考

ただし、errors 配列は規範化されているわけではなく、あくまで 例示です。位置指定の手段(JSON Pointer か JSONPath か独自キー名か)も自由で、依然として実装者の裁量は広く残っています。

改善点③: 共通型レジストリの新設

7807 時代の課題

「Rate Limit 超過」「Quota 超過」のような、どの API でも共通で使えそうなエラー型でさえ、各社が独自の URI を定義していました。再利用と収束のための仕組みが仕様側になかったわけです。

9457 での改善

RFC 9457 の Section 4.2 で IANA に HTTP Problem Types レジストリ が新設されました。登録テンプレートには以下の項目を含めます。

  • Type URI
  • Title
  • Recommended HTTP status code
  • Reference

登録方法

登録ポリシーは Specification Required で、RFC 8126 Section 4.6 に基づきます。簡単に言うと以下です。

  • 登録したい問題型の仕様書(公開された安定したドキュメント)を用意する
  • 上記テンプレートの項目を埋めて IANA に申請する
  • Designated Expert がレビューして承認・却下を判断する
  • 審査では「定義の明確性」「RFC 9457 の要件適合性」「コミュニティからのフィードバック」が考慮される

ベンダー固有・アプリケーション固有・デプロイ固有の値は登録不可とされており、広く再利用できる共通型だけが対象です。

プレフィックスとして https://iana.org/assignments/http-problem-types# を使うことも可能ですが、RFC 自身も「この URI は解決可能でない場合がある」と書いており、あくまで識別子として使う位置づけです。

例示

子どもが 1 分間に 20 回「おかし食べたい」と連投してくる状況を想定します。API としては HTTP 共通の 429 Too Many Requests を返したいところですが、同じ 429 エラーでも実装ごとに type URI の付け方はバラバラでした。

❌ Before (7807 時代): 共通のエラーでも実装ごとに独自 URI を定義していた

{
"type": "https://a-family.example/problems/too-many-snack-requests",
"title": "Too Many Requests",
"status": 429
}
{
"type": "https://b-family.example/problems/snack-request-cooldown",
"title": "Too Many Requests",
"status": 429
}
{
"type": "urn:c-family:problems:snack-quota-exceeded",
"title": "Too Many Requests",
"status": 429
}

✅ After (9457): 共通型を IANA レジストリで共有できる仕組みが整った

現状は about:blank のみ登録済みで、型に特別な意味を持たせない場合の標準値として使います。

{
"type": "about:blank",
"title": "Too Many Requests",
"status": 429,
"detail": "おやつの要求が多すぎます。30 分後にまた声をかけてください。"
}

将来的に Rate Limit 超過のような汎用エラー型が IANA に登録されれば、複数の API 間で type を共通化できるようになるというのがレジストリ新設の狙いになります。

備考

2026年4月時点のレジストリに登録されているのは 3 件です。about:blank に加えて、https://iana.org/assignments/http-problem-types#date#ohttp-key がありますが、後者 2 つはいずれも RFC 9458 Oblivious HTTP 由来で登録されたものです。汎用的な業務エラー型(Rate Limit 超過、バリデーションエラー等)が登録されているわけではありません。

改善点④: Security Considerations の補強

7807 時代の課題

RFC 7807 にもセキュリティ上の注意書きはありましたが、具体性に欠けていました。結果、detail フィールドにスタックトレースや内部パスがそのまま載る実装が散見される状態でした。

9457 での改善

RFC 9457 の Section 5 はより踏み込んだ記述になっています。

  • スタックダンプ・内部実装詳細の露出を避ける
  • instance に不透明でない(デバッグ性のある)URL を置くと、情報漏洩経路になりうる
  • クライアントは信頼できないソースから来た problem details を、安易にそのまま表示しない(XSS 観点)

具体例

冒頭のお風呂サンプルを引きずって、「子どもがお風呂に持ち込むおもちゃが電池式だったので業務ルールで弾いた」ケースをイメージしてみます。

❌ Before (悪い例): detail にスタックトレースをそのまま載せる

{
"type": "/errors/bath-preparation-failed",
"title": "Bath preparation failed",
"detail": "com.example.bathtime.exception.ToyNotWaterproofException: Battery-powered toy cannot be brought into the bathroom\n at com.example.bathtime.validator.ToyWaterSafetyValidator.validate(ToyWaterSafetyValidator.java:58)\n at com.example.bathtime.service.BathPreparationService.validateItems(BathPreparationService.java:34)\n at com.example.bathtime.service.BathPreparationService.prepare(BathPreparationService.java:21)\n at com.example.bathtime.controller.BathController.startBath(BathController.java:19)",
"status": 422
}

パッケージ構成・例外クラス名・内部バリデータ構造まで透けて見えるため、攻撃者の下調べに使われかねません。

✅ After (良い例): 利用者向けは業務的な説明にとどめ、相関 ID で運用側から追跡可能にする

{
"type": "/errors/toy-not-waterproof",
"title": "このおもちゃはお風呂に持ち込めません",
"detail": "電池で動くおもちゃはお風呂場に持ち込めません。別のおもちゃを選んでください。",
"instance": "/errors/trace/a1b2c3d4e5f6",
"status": 422
}

instance には不透明な相関 ID(トレース ID 等)だけを入れ、サーバ側のログと突き合わせればデバッグできる運用にしておくのが定石です。

備考

「どこまで出してよいか」の線引きは、公開 API(B2C)か社内 API かで大きく変わります。最終的には設計者が文脈を踏まえて判断するしかなく、RFC としては「気をつけろ」以上のことは書けないものと思います。

フューチャーの Web API 設計ガイドラインでの扱い

私がフューチャーの Web API 設計ガイドライン の執筆に関わった関係で、ガイドライン側との対応関係にも触れておきます。

ガイドラインではエラーレスポンスを RFC 9457 準拠で推奨しており、主な整合点は以下の通りです。

  • type は相対パス URI を推奨(9457 Section 3.1.1 の「デリファレンス不可でよい」ガイダンスに相当)
  • 複数エラーは errors 配列で表現し、件数が 1 件でも配列形式で統一
  • errors[].detail を必須、pointer を任意とする
  • エラー時の Content-Type は application/problem+json を推奨
  • B2C 公開 API では詳細情報を控え、内部 API では詳細を許容(9457 Section 5 Security Considerations と整合)

興味ある方は、ぜひ ガイドライン本体 も読んでみてください。

まとめ

RFC 7807 から RFC 9457 で変わったことは、フォーマットとしてはゼロに等しく、本質的には「実務で迷いやすかった部分のガイダンスが整理された」という位置づけのアップデートです。

新規の Web API 設計では、まず RFC 9457 を読み、application/problem+json を返すところから始めれば良いと思いました。差分が少ないこと自体が、この仕様がもう十分に枯れていることを示しているようで、個人的には読んでいて心地よい RFC の更新でした。