メインコンテンツまでスキップ

第08章:CORSとセキュリティの最初の壁🧱🔒

この章は「React(ブラウザ)からHTTP関数を呼んだ瞬間に出る、あの謎エラー」を倒す回です😇🌪️ そして大事な本音:CORSは“セキュリティ”じゃなくて、ブラウザの交通整理です🚦(ここを勘違いすると事故ります)


この章でできるようになること🎯

  • ✅ CORSエラーの正体を“一言で”説明できる🗣️
  • ✅ Functions(HTTP / onRequest)に許可Originを絞ったCORS設定を入れられる🔧
  • ✅ 「CORSを直した=安全になった」じゃない理由がわかる🧠
  • ✅ AI(Genkit / Firebase AI Logic)を呼び出す前に、課金事故を防ぐ入口の守りを作れる🛡️💸

キーワード📚

  • CORS / Origin / Same-Origin Policy
  • Preflight(事前通信)/ OPTIONS
  • Allowlist(許可リスト)
  • Auth(IDトークン)/ App Check(アプリ正当性)
  • “CORSはブラウザ限定の壁”🧱

1) まず結論:CORSって何?🤔

CORS Traffic Controller

CORSは、ざっくり言うと—— ブラウザが「そのWebページは、そのAPIを呼んでいい?」をチェックする仕組みです🌐✅

  • ブラウザは、勝手に他ドメインのAPIを呼ぶのを嫌がります(ユーザー保護)🧯
  • だからサーバー側(Functions)が「このOriginならOKだよ」と返す必要があります🧩

Firebase公式のHTTP関数ドキュメントにも、HTTP関数はデフォルトでCORS未設定で、クロスオリジンだとエラーになる、と明記されています。(Firebase)


2) つまずきポイント:Preflight(OPTIONS)って何👀?

Preflight (OPTIONS) Flow

「GETなら動くのに、POSTしたら死ぬ😇」みたいな時、だいたいこれです。

ブラウザは、条件によって本番のPOSTの前に「この送信していい?」って確認の OPTIONS を投げます(= preflight)🧪 HTTP関数は OPTIONS もサポート対象に含まれます。(Firebase)

なので、CORS設定がない or 許可条件が合わないと、ブラウザ側がブロックします🚫


3) 一番ラクで堅い解決:onRequest の cors オプションを使う🛠️✨

ここが2026時点の“正攻法”です。 Firebase公式は、HTTP関数(onRequest)で cors オプションを提供していて、

  • true:全Origin許可(公開API向け)
  • string / regex / array:許可したいOriginだけ指定(アプリ向け)
  • デフォルトは false(CORSなし)

…と説明しています。(Firebase)


✅ ハンズオン:許可Originを“限定”してCORSを通す(推奨)🔒

例:自分のフロントが

  • https://YOUR-PROJECT.web.app
  • https://example.com(独自ドメイン)

だけから呼べるようにする✋

import { onRequest } from "firebase-functions/v2/https";

const ALLOWED_ORIGINS = [
"https://YOUR-PROJECT.web.app",
"https://example.com",
];

export const apiEcho = onRequest(
{ cors: ALLOWED_ORIGINS },
async (req, res) => {
// JSONを返すだけの超ミニAPI
res.status(200).json({
ok: true,
method: req.method,
body: req.body ?? null,
});
}
);

✅ これで preflight(OPTIONS)も含めてブラウザが通しやすくなります🧠✨ (自前で Access-Control-Allow-* を書くより事故が減ります)


⚠️ あるある注意:Originは「スキーム+ドメイン+ポート」🎯

Origin Components

  • http://localhost:5173http://localhost:3000別Originです😵‍💫
  • https://http:// も別です
  • “プレビューURL”も別Originになります(プレビュー運用するなら、そこも許可リストに入れる)🧪

4) 超重要:CORSは“セキュリティ”ではない🙅‍♂️🔐

CORS is Not Security

ここ、テストに出ます(マジで)📣

  • CORSは ブラウザだけが守る壁🧱
  • curl / Postman / 悪いスクリプトは、CORS関係なく叩けます🔫(ブラウザじゃないので)

つまり、CORSを直しただけだと…

「“誰でも”叩けるAPIが、ブラウザからも叩けるようになった」

になりがちです😇


5) “本当の守り”の基本セット:Auth + App Check 🛡️✨

Three Layers of Defense

AI(Genkit / Firebase AI Logic)を裏側で呼ぶなら、ここ超大事💸🔥 守らないと、勝手に叩かれて課金が溶ける系の事故が起きます。

5-1) Auth(ログインした人だけ)🔐

Firebase Admin SDKで IDトークン検証できます。公式に「検証・デコードできる」説明があります。(Firebase)

5-2) App Check(“本物のアプリ”からだけ)🧿

App Checkは「あなたのアプリから来た通信か」を見る仕組みで、Authと相互補完って公式も言っています。(Firebase)


6) ここが落とし穴:onRequest は enforceAppCheck が使えない😇

Callable(onCall)なら enforceAppCheck: true が公式手順としてあります。(Firebase) でも onRequest(HTTP)には同じノリで付けられません

実際、2nd genの HttpsOptions(onRequest用オプション)は enforceAppCheck を含まない形で定義されています。(Firebase)

👉 なのでHTTP関数でApp Checkをやるなら、自分でトークン検証が基本になります(次節)🧠


7) ハンズオン:HTTP関数で「CORS + Auth + App Check」を最低限いれる🧰🛡️

Manual Token Verification

守りの設計(この章のゴール形)🏁

  • ✅ CORS:許可Originだけ通す
  • ✅ Auth:Authorization: Bearer <ID_TOKEN> を検証
  • ✅ App Check:X-Firebase-AppCheck を検証

App Checkのトークン検証は、公式が「X-Firebase-AppCheck ヘッダを取り、Admin SDKで verify する」例を出しています。(Firebase)


✅ 例コード:守り入りのHTTP関数テンプレ🧱🔒

import { onRequest } from "firebase-functions/v2/https";
import { initializeApp } from "firebase-admin/app";
import { getAuth } from "firebase-admin/auth";
import { getAppCheck } from "firebase-admin/app-check";

initializeApp();

const ALLOWED_ORIGINS = [
"https://YOUR-PROJECT.web.app",
"https://example.com",
];

export const apiSecureEcho = onRequest(
{ cors: ALLOWED_ORIGINS },
async (req, res) => {
try {
// 1) Auth(IDトークン)チェック 🔐
const authHeader = req.header("Authorization") || "";
const match = authHeader.match(/^Bearer (.+)$/);
if (!match) {
return res.status(401).json({ ok: false, error: "Missing Bearer token" });
}
const idToken = match[1];
const decoded = await getAuth().verifyIdToken(idToken);

// 2) App Checkチェック 🧿
const appCheckToken = req.header("X-Firebase-AppCheck") || "";
if (!appCheckToken) {
return res.status(401).json({ ok: false, error: "Missing App Check token" });
}
await getAppCheck().verifyToken(appCheckToken);

// 3) ここから先が“本処理”(AI呼び出し等)🤖✨
return res.status(200).json({
ok: true,
uid: decoded.uid,
method: req.method,
body: req.body ?? null,
});
} catch (e) {
return res.status(401).json({ ok: false, error: "Unauthorized" });
}
}
);

📌 この形にしておくと、次の章以降で

  • GenkitでAI処理
  • Slack通知
  • Firestore書き込み みたいな“お金が動く処理”を安全に足せます💸🧯

8) React側はどう呼ぶ?(超ミニ例)⚛️📡

イメージだけ掴めればOKです🙂

  • AuthのIDトークンを取って Authorization に入れる
  • App Checkトークンを取って X-Firebase-AppCheck に入れる
  • それで fetch() する

(App Checkトークンをカスタムバックエンドに送って検証する流れ自体が公式の前提です)(Firebase)


9) AIでデバッグを爆速にする(Antigravity / Gemini CLI)🛸🤖

AI Debugging CORS

CORSって、正直「1文字違い」で沼ります😇 そんな時は Gemini CLI + Firebase拡張に“症状から原因を当てさせる”のが強いです💪

Firebase拡張は、Gemini CLIにFirebase向けの能力を足して、MCPサーバー設定なども面倒を見てくれます。(Firebase)

使い方の雰囲気🧰

  • 「ブラウザのNetworkタブのスクショ or エラーメッセージ」を貼る📎
  • 「許可したいOriginはこれ」「本番とプレビューはこれ」って添える
  • cors の許可リストを最小で提案して」って頼む🤖

(拡張の導入・更新コマンドも公式に載ってます)(Firebase)


10) ミニ課題🎓✨(10〜15分)

課題A:CORSだけ直す🧱

  1. apiEcho を作る
  2. 許可Originを 1つだけにしてデプロイ
  3. Reactから叩いて成功させる🎉

✅ チェック

  • 「CORSはデフォルト無効で、許可しないとブラウザが止める」と言える(Firebase)

課題B:守りを入れて“安全な入口”にする🔐🧿

  1. apiSecureEcho を作る
  2. Authorization が無い時に 401 になる
  3. X-Firebase-AppCheck が無い時に 401 になる

✅ チェック

  • 「HTTP関数はCallableみたいに enforceAppCheck で一発、ではない」と言える(Firebase)

まとめ🎁

  • CORSは「ブラウザの壁」であって「サーバーの防御」じゃない🧱
  • onRequest({ cors: [...] }) で、許可Originを絞るのが最短ルート🚀 (Firebase)
  • 本当に守るなら Auth + App Check をセットで考える🔐🧿 (Firebase)
  • AI(Genkit / Firebase AI Logic)を裏で呼ぶほど、入口の守りが重要になる🤖💸
  • Gemini CLI拡張+MCPで、沼(CORS)を早めに脱出しよう🛸✨ (Firebase)

次の第9章(Callable / onCall)は、この章の地獄(CORS)をかなり回避しつつ、Auth/App Checkを“楽に”やる王道に入っていきます🔐✨