POST だけ?こんな馬鹿げた API デザインの議論を終わらせよう
"POST だけ" API の神話を打ち破り、なぜそれが API デザイン原則の誤解に基づいているのかを説明し、RESTful と RPC のアーキテクチャスタイルの適切な使用事例を明確にします。
最近、"POST だけ" を使って API を設計するべきかどうかの議論が私の注意を引きました。この議論を掘り下げてみると、人々が争っている問題が無意味であるばかりでなく、多くの開発者が API デザインの本質を誤解していることも明らかになりました。今日は、API デザインの核心的な考え方に深く迫り、この議論が最初から存在するべきではない理由を見ていきましょう。
"POST だけ" の誤解
"POST だけ" を使って RESTful API 仕様を置き換えることを支持する開発者は、明らかに API デザインの最も重要な点を理解していません。彼らの主張は通常、以下のようなものです:
- 設計の簡素化: 1 つのメソッドですべてを処理できる
- セキュリティ: POST パラメータは URL に表示されない
- 柔軟性: POST は任意のデータ構造を送信できる
一見すると、これらの主張には一理あるように見えます。しかし、実際には、これは HTTP メソッドの選択と API デザインスタイルの混同、すなわち異なるレベルの問題を混同しているに過ぎません。POST は HTTP プロトコルの 1 つのメソッドに過ぎませんが、REST は API デザインのスタイルです。
API デザインの本質
具体的な API スタイルについて議論する前に、まず API デザインの核心的な目的が何であるかを理解する必要があります。優れた API は次のようであるべきです:
- 明確で理解しやすい: 他の開発者(将来の自分自身も含む)が、各エンドポイントの目的を直感的に理解できる
- 一貫性がある: 一定の仕様に従い、学習コストを削減する
- 拡張性がある: バージョン管理や機能の拡張を簡単に行える
- 効率的である: パフォーマンスやリソースの利用効率を考慮する
RESTful API: HTTP メソッドの選択だけではない
RESTful API は多くの API デザインスタイルの中の 1 つであり、リソースとリソースに対する操作に焦点を当てています。簡単なブログシステムを例にして、RESTful APIがどのように設計されているかを見てみましょう:
-
すべての記事を取得:
-
特定の記事を取得:
-
新しい記事を作成:
-
記事を更新:
-
記事を削除:
この例では次のことがわかります:
- API は "記事" リソースを基盤に設計されています
- 異なる操作を表すために異なる HTTP メソッドが使用されています
- URL 構造は操作されているリソースを示しており、明確です
この設計アプローチによって、API はより直感的で自己説明的になり、開発者が各エンドポイントの機能を理解しやすくなります。
RPC: "POST だけ" の背後にある API スタイルを理解する
RPC (Remote Procedure Call) スタイルの API デザインの目的は、リモートサービス呼び出しをローカル機能の呼び出しのようにシンプルに見せることです。
興味深いことに、"POST だけ" を支持する人々は、実際には RPC スタイルを説明していることに気づいていないかもしれません。
RESTful API に比べて、RPC はリソースよりも操作そのものに重点を置いています。これが、RPC スタイルの API が通常 "動詞 + 名詞" 形式を使用する理由です。例えば getProduct(productId)
や createUser(userData)
です。
多くの RPC 実装では、すべての操作が通常同じエンドポイントに POST リクエストとして送信され、リクエストボディ内で特定の操作とパラメータが指定されます。これが "POST だけ" の考えが実際には RPC に近い理由です。
例えば、HTTP ベースの RPC スタイルでの "商品の取得" リクエストは次のようになります:
現代の RPC フレームワーク(例えば gRPC)は、より強力で効率的な実装を提供 しています。これを例にして RPC スタイルを説明しましょう:
まず、サービスとメッセージ形式を定義します(Protocol Buffers を使用):
次に、Node.js クライアントでこのサービスを使用するのは、ローカル関数を呼び出すように簡単です:
この RPC スタイルの例では、次のことがわかります:
- サービス定義が利用可能なすべての操作を明確にリストしています(この簡略化された例では
GetArticle
とCreateArticle
)。 - 各操作には明確に定義されたリクエストとレスポンスのタイプがあります。
- クライアントコードは、非同期のローカル関数を呼び出すように見えます。
await
を使用して結果を待機し、ネットワーク通信の複雑さをさらに隠しています。 - 手動で HTTP リクエストを構築したり、JSON レスポンスを解析したりする必要はありません。
潜在的に HTTP/2 をトランスポートプロトコルとして使用している場合がありますが、RPC フレームワーク(例えば gRPC)は、リモート呼び出しをローカル関数呼び出しのように見せる抽象化層を提供します。
したがって、"POST だけ" と RESTful API に関する議論の多くは、実質的にこれら 2 種類の API スタイル、すなわち REST と RPC に関する議論であることがわかります。しかし、重要なのは、これら 2 種類のスタイルがそれぞれの適用シナリオを持ち、選択はプロジェクトの具体的なニーズに基づいて行われるべきだということです。個人的な好みで選んではいけません。