Skip to content

認可コード + PKCE

最も一般的な OIDC フローです。Web アプリ、モバイルアプリ、SPA、デスクトップアプリで人がログインするとき、ほぼすべてがこれです。本ライブラリでは PKCE (Proof Key for Code Exchange、RFC 7636) は必須です — OAuth 2.0 BCP (RFC 9700) と FAPI 2.0 が要求するため、また 2026 年現在 PKCE を無効にする合理的な構成は存在しないためです。

このページで触れる仕様
  • RFC 6749 — OAuth 2.0 Authorization Framework(§5.2 エラーコード)
  • RFC 7636 — Proof Key for Code Exchange (PKCE)
  • RFC 9700 — OAuth 2.0 Security Best Current Practice
  • RFC 9126 — Pushed Authorization Requests (PAR)
  • OpenID Connect Core 1.0 — §3.1(Authorization Code Flow)
用語の補足
  • 認可コード(authorization code) — OP がブラウザリダイレクト経由で RP に渡す 1 回限りの opaque 文字列。RP はこれを /token でトークンに交換します。
  • PKCE(「ピクシー」と読む) — code_verifiercode_challenge を使った小さな手続きで、「このコードを引き換えに来たクライアントは、フローを始めたクライアントと同一」と OP に証明させる仕組み。リダイレクトされたコードを悪意あるアプリに横取りされるのを防ぎます。詳しい解説は後述。
  • state — RP が authorize 要求に乗せ、コールバックで検査するランダム opaque 値。リダイレクトの CSRF 防御。
  • nonce — ID トークンに紐づくランダム opaque 値。RP 側でのリプレイ防御。

完全なシーケンス

パラメータ用語集

パラメータ送信タイミング目的
response_type=code/authorize認可コード grant を要求。
client_id/authorize/token登録済み RP を識別。
redirect_uri/authorize/token でも echo)OP がユーザを戻す先。登録リストとの 完全一致
scope/authorize要求権限。OIDC では openid を含むこと。
state/authorizeRP がコールバックで echo する不透明値。redirect の CSRF 防御。
nonce/authorizeID トークンの nonce claim にバインドされるランダム値。replay 防御。
code_challenge/authorizeBASE64URL(SHA256(code_verifier))
code_challenge_method/authorizeS256(本ライブラリが受理する唯一の値)。
code/authorize レスポンス単発。spec 上 max-age 60 秒、本ライブラリも同値。
code_verifier/tokencode_challenge の原像。OP が SHA-256 を再計算。
grant_type=authorization_code/tokenこの grant を選択。
Client auth/tokenclient_secret_basic / client_secret_post / private_key_jwt / tls_client_auth / self_signed_tls_client_auth / none(PKCE のみ)のいずれか。
statenonce の違い

両方ともランダムな opaque 値で、両方とも replay 系攻撃を防ぎますが、守る経路が違います。

  • stateフロントチャネル(ブラウザのクエリ文字列)を流れる値です。RP はリダイレクト前にユーザのセッションに保存し、コールバック時に突き合わせます。守るのは リダイレクト 経路の CSRF — 攻撃者が /callback に偽コールバックを投げてアプリに受理させる、を阻止します。
  • nonceID トークンの claim に乗ります。RP はリダイレクト前にユーザセッションに保存し、トークン交換後に突き合わせます。守るのは ID トークン のリプレイ — 盗まれた ID トークンを別 RP で、あるいは同じ RP の別ログイン試行で再利用させない、を阻止します。

両方使ってください。本ライブラリは confidential client で state 欠落のリクエストを拒否しますし、OIDC は response_type=code id_tokenid_token を使うときには nonce 必須です。

code_verifier / code_challenge / S256 とは

code_verifier は RP が生成して 自分だけが持っている 高エントロピのランダム文字列です。RFC 7636 §4.1 で 43〜128 文字の URL セーフ文字と定められています。

code_challenge は RP が /authorize で OP に送る値です。code_challenge_method=S256 の場合、BASE64URL(SHA-256(code_verifier)) — つまり一方向ハッシュです。OP には逆算できず、後で /token に verifier 本体を送ったときに RP の所有を証明できる形になります。

S256 は SHA-256 ベースの変換で、本ライブラリが受理する唯一の code_challenge_method です。古い plain 方式(challenge と verifier が同じ)は URL を読める攻撃者には何の保護にもならないので、RFC 9700 が新規構築での使用を禁止しています。

redirect_uri — 完全一致が必須な理由

/authorizeredirect_uri は、クライアントの登録リストに対して バイト単位 で照合されます — 末尾スラッシュの正規化なし、パスプレフィックス一致なし、ワイルドカードなし。この厳格さは意図的です: open-redirect バグや「https://app.example.com/ のサブパスならどれでも」式の登録は、コードを攻撃者が制御する URL に漏らす経路として何度も悪用されてきました。RFC 9700 §2.1 が完全一致を要求しており、本ライブラリも強制します。/token では RP が /authorize で送ったのと 同じ redirect_uri を再送する必要があり、ずれると redirect_uri_mismatch を返します。

response_type=code とは

response_type=code認可コードフロー を要求します — OP がリダイレクトで短寿命の code を返し、RP がそれを /token で実トークンに交換するパターンです。代替の tokenid_token tokencode id_token などは古い hybrid / implicit フローで、OAuth 2.0 BCP(RFC 9700)が非推奨としています。本ライブラリは code を正規経路として実装し、hybrid 形式は互換性のためのオプトイン表面にとどめています — 新規構築向けではありません。

PAR とは何か、いつ必要になるか

PAR(Pushed Authorization Requests、RFC 9126)は、RP が authorize パラメータを 先に サーバ側の /par エンドポイントに POST し、短寿命の request_uri を受け取って、ブラウザは ?client_id=...&request_uri=... だけでリダイレクトする方式です。利点:

  • リクエスト全体がブラウザ履歴・サーバログ・referrer に残らない。
  • ユーザエージェント境界での改竄が無効化される — そこに露出するのは request_uri だけ。
  • FAPI 2.0 Baseline では必須。それ以外でもオプトインする価値あり。

op.WithFeature(feature.PAR) で有効化すると、discovery 文書の pushed_authorization_request_endpoint に出ます。

PKCE が防ぐもの

解説: PKCE が防ぐ攻撃

PKCE が無いと、同一デバイスで myapp:// の URI ハンドラを乗っ取った悪意あるアプリが認可コードのリダイレクトを傍受できます:

  1. ユーザが正規 RP でログイン。OP が myapp://callbackcode=abc を発行。
  2. 悪意アプリがリダイレクトを傍受し(race condition または universal-link なりすまし)、code=abc を読む。
  3. 悪意アプリが code=abc/token に投げ、トークンを得る。

PKCE はコードを 正規 RP のみが知る秘密 にバインドします:

  1. 正規 RP がランダム code_verifier を生成し、/authorize には SHA256(code_verifier)code_challenge)のみを送る。
  2. OP は発行コードと一緒に code_challenge を保存。
  3. /token で OP は code_verifier を要求し、SHA-256 を再計算。
  4. 悪意アプリは code を見ても verifier を見ていない — /token 呼び出しは失敗。

これは RP がクライアントシークレットを保管できないケース(SPA / ネイティブ)でも機能します。

本ライブラリでの強制方法

挙動場所
code_challenge_method=plain拒否S256 のみ受理)。internal/pkce
クライアントの RequiresPKCE が true(public client のデフォルト、FAPI 2.0 では強制)の場合、code_challenge 無しは拒否。internal/authorize
code_verifier の長さ / 文字集合は RFC 7636 §4.1 に対して検証。internal/pkce
ミスマッチは /token で RFC 6749 §5.2 の invalid_grant を返す(/authorize ではなく)。internal/tokenendpoint/authcode.go

よくあるエラーと意味

エラー原因対処
invalid_request code_challenge_methodクライアントが plain を送ったS256 を送る
invalid_request_uriPAR request_uri の期限切れ / 消費済み新しい PAR を発行
invalid_grant/tokencode_verifier 不一致、または code が使用済 / 期限切れcode を再利用しない、再生成
redirect_uri_mismatch/tokenredirect_uri/authorize と異なるバイト単位で同一にする

自分でフローを動かす

examples/03-fapi2 は PAR + JAR + DPoP + PKCE をひとつの構成にまとめた FAPI 2.0 Baseline OP を起動します。OFCS conformance suite は同じシーケンスを 2 つの FAPI plan にわたって約 129 module 実行します。OFCS 適合状況 に内訳があります。

次に読むもの