Skip to content

セキュリティ思考と攻防の基礎

はじめに

あなたのウェブサイトは安全ですか? 多くの開発者は「セキュリティはセキュリティチームの仕事」と考えていますが、自分のプロジェクトが攻撃され、ユーザーデータが漏洩して初めてその重要性に気づきます。セキュリティはオプションではなく、すべての開発者にとっての基本スキルです。

この章では、セキュリティのマインドセットを構築し、最も一般的なウェブセキュリティの脅威と防御方法を理解します。

この記事で何を学ぶか?

内容主要概念
第 1 章セキュリティ思考モデル攻撃者のように考える
第 2 章一般的なウェブ攻撃XSS、SQLインジェクション、CSRF
第 3 章防御戦略入力検証、出力エンコーディング、アクセス制御
第 4 章セキュリティチェックリストローンチ前のセキュリティ自己監査

この章を終えると、基本的なセキュリティ意識を持ち、最も一般的なウェブセキュリティの脅威を識別し防御できるようになります。


0. 全体像:なぜ開発者はセキュリティを理解する必要があるのか?

家を建てたと想像してください。機能は完璧で、内装も美しい——しかし、鍵を取り付けるのを忘れています。セキュリティ脆弱性は、コードの世界における「付け忘れた鍵」です。

セキュリティの核心原則

  • 最小権限:必要な権限だけを付与し、一切多く与えない
  • 多層防御:単一の防御線に依存せず、複数の層で防御する
  • 入力を信頼しない:外部からのすべてのデータは悪意のある可能性がある
  • デフォルトで安全:デフォルト設定は便利さではなく、安全性を優先する

1. 一般的なウェブ攻撃

下のインタラクティブコンポーネントで、3つの最も一般的なウェブ攻撃の原理を理解しましょう(教育目的のみ):

Web security vulnerability demo (educational) - click to switch vulnerability type
Attack flow
1Attacker submits malicious script in an input field
2Server stores it without filtering
3Script runs when another user opens the page
4User cookies or data are stolen
❌ Vulnerable code
// Directly inserting user input (dangerous!)
el.innerHTML = userInput
// If userInput = '<scr' + 'ipt>steal(cookie)</scr' + 'ipt>'
// the script will execute!
✅ Fixed code
// Insert safely with textContent
el.textContent = userInput
// Or use framework escaping
// Vue: {{ userInput }}  escaped automatically
// React: {userInput}    escaped automatically
Defense:Never trust user input. Use framework escaping, avoid innerHTML, and encode output.

1.1 XSS(クロスサイトスクリプティング)

攻撃者がウェブページに悪意のあるスクリプトを注入します。他のユーザーがそのページを訪問すると、スクリプトがブラウザで実行されます。

javascript
// 危険:ユーザー入力を直接HTMLに挿入
element.innerHTML = userInput
// userInputが <script>悪意のあるコード</script> の場合、実行される

// 安全:textContent またはエスケープを使用
element.textContent = userInput
// またはフレームワークの自動エスケープ(Vueの{{ }}、ReactのJSX)を使用

防御のポイント

  • 出力時にHTMLの特殊文字(<>&"')をエスケープする
  • モダンフレームワークの自動エスケープ機構を使用する
  • Content-Security-Policy HTTPヘッダーを設定する

1.2 SQLインジェクション

攻撃者が特別な入力を構成して、SQLクエリのロジックを改変します。

javascript
// 危険:文字列連結によるSQL
const query = `SELECT * FROM users WHERE name = '${userInput}'`
// userInputが ' OR '1'='1 の場合、すべてのユーザーが返される

// 安全:パラメータ化されたクエリを使用
const query = 'SELECT * FROM users WHERE name = ?'
db.execute(query, [userInput])

防御のポイント

  • 常にパラメータ化されたクエリ / プリペアドステートメントを使用する
  • ORMフレームワーク(Prisma、Sequelizeなど)を使用する
  • データベースアカウントの権限を制限する

1.3 CSRF(クロスサイトリクエストフォージェリ)

攻撃者がログイン済みのユーザーを悪意のあるページに誘導し、ユーザーのログイン状態を利用してリクエストを送信させます。

防御のポイント

  • CSRFトークンを使用する
  • Referer / Origin ヘッダーを確認する
  • 重要な操作にはGETではなくPOSTを使用する
  • Cookieに SameSite 属性を設定する

2. 防御戦略

2.1 入力検証

javascript
// ホワイトリスト検証:期待される形式のみを許可
function isValidEmail(email) {
  return /^[^\s@]+@[^\s@]+\.[^\s@]+$/.test(email)
}

// 長さ制限
function isValidUsername(name) {
  return name.length >= 2 && name.length <= 50
}

2.2 機密データの保護

データタイプ保護対策
パスワードbcrypt/argon2によるハッシュ化、平文保存は絶対にしない
APIキー環境変数、コードリポジトリにコミットしない
ユーザーデータHTTPS転送、暗号化保存
セッショントークンHttpOnly + Secure + SameSite Cookie

2.3 HTTP セキュリティヘッダー

Content-Security-Policy: default-src 'self'
X-Content-Type-Options: nosniff
X-Frame-Options: DENY
Strict-Transport-Security: max-age=31536000

3. セキュリティチェックリスト

ローンチ前に、下のインタラクティブコンポーネントでプロジェクトのセキュリティ状況を確認しましょう:

Project Security Checklist
Check completed safeguards and view the project security score
Security score
0 pts
Danger
🔍Input validation0/3
Validate all user input on the server
Use allowlists instead of blocklists
Limit uploaded file type and size
🔐Authentication and authorization0/4
🛡️Data protection0/3
🌐Communication security0/3

3.1 開発フェーズ

  • [ ] すべてのユーザー入力が検証およびエスケープされている
  • [ ] パラメータ化されたクエリを使用し、SQL連結がない
  • [ ] パスワードがbcryptなどのアルゴリズムでハッシュ化されている
  • [ ] 機密設定が環境変数で管理されている
  • [ ] .env ファイルが .gitignore に追加されている

3.2 デプロイメントフェーズ

  • [ ] HTTPSが有効になっている
  • [ ] セキュリティHTTPヘッダーが設定されている
  • [ ] デバッグモードと詳細なエラーメッセージが無効になっている
  • [ ] データベースが最小権限アカウントを使用している
  • [ ] 依存関係が定期的に更新されている(npm audit

4. AI 活用:大規模言語モデルでセキュリティを強化

LLMは「セキュリティコンサルタント」として機能し、コードの脆弱性監査やセキュリティソリューションの生成を支援できます。

4.1 コードセキュリティ監査

プロンプト

以下のコードのセキュリティ監査を行い、以下を確認してください:
- XSS 脆弱性(エスケープされていないユーザー入力)
- SQL インジェクション(文字列連結によるクエリ)
- CSRF リスク(トークン検証の欠如)
- 機密データの漏洩(ハードコードされたキー、平文パスワード)
各問題について、リスクレベル、具体的な場所、修正案を提示してください。

[コードを貼り付け]

4.2 セキュリティ設定の生成

プロンプト

プロジェクトは Express.js + PostgreSQL を使用しており、まもなくローンチします。
完全なセキュリティ設定チェックリストを生成してください。含める内容:
- HTTP セキュリティヘッダーの設定コード
- CORS 設定
- 安全なデータベース接続設定
- 環境変数管理ソリューション
すぐに使えるコードスニペットを提供してください。

4.3 脆弱性の原理を説明する

プロンプト

具体例を使って、CSRF攻撃の完全な流れを説明してください:
1. 攻撃者が悪意のあるページをどのように構成するか
2. なぜブラウザが自動的にCookieを送信するのか
3. サーバー側でCSRFトークンを使ってどう防御するか
コードで攻撃と防御の完全なプロセスをデモしてください。

AI 活用のアドバイス

AIのセキュリティ監査は、専門的なセキュリティテストに代わるものではありません。第一段階のスクリーニングとして活用し、重要なシステムは専門のセキュリティチームによる監査を受けてください。


5. まとめ

  1. セキュリティ思考:外部入力を信頼しない、最小権限、多層防御
  2. 一般的な攻撃:XSS、SQLインジェクション、CSRFは最も頻度の高いウェブセキュリティの脅威
  3. 防御戦略:入力検証、出力エンコーディング、パラメータ化クエリ、セキュリティHTTPヘッダー
  4. セキュリティ習慣:ローンチ前にセキュリティチェックリストを実行し、依存関係を定期的に監査

最後に

セキュリティは一度きりの作業ではなく、開発プロセス全体を貫く習慣です。車の運転でシートベルトをするようなもの——事故を予期しているからではなく、基本的な安全意識だからです。すべてのコード行を書く際、自分に問いかけてください:もしこの入力が悪意のあるものだったら、何が起こるか?


さらに学ぶために

  • OWASP Top 10:ウェブアプリケーションのセキュリティリスク上位10。すべての開発者が知っておくべき。
  • 実践ツールnpm audit で依存関係の脆弱性をチェックし、ESLintセキュリティプラグインでコードを検査する。
  • 深い学習:HTTPSの原理、JWTのセキュリティプラクティス、OAuth 2.0のセキュリティ考慮事項を学ぶ。
  • セキュリティコミュニティ:セキュリティ勧告に注目し、既知の脆弱性を迅速に修正する。