最近社内で新規サービス立ち上げPJに関わる機会があり、その際にユーザのログイン/ログアウト状態の保持について触れる機会がありました。ただ、想像以上にこの辺りをまったく理解していなかったことに気が付いたので、自分で調べたことをまとめてみました。もし間違っているところなどあれば、コメントいただけると嬉しいです。特にステートフルとステートフルや、cookie vs web storageの比較はよく見ますが、これらをビッグピクチャーで考えたことがなかったので、全然理解していませんでした…
ステートフルとステータス
Http通信はそもそもステートレス
です。これは元々ドキュメント公開の方法としてのみウェブサイトなどが使われ始めていたことによるものですが、現代ではオンラインショップやアプリなど、状態をページを跨いでやり取りしたりする際にステートが必要になり、ステートフルというものが使われだしました。
・ステートフルについて
どう行うのか
ステートフルセッションはまずバックエンドサイド
で作成されます。そしてこのセッション参照IDをクライアント側に送ります。クライアント側でこのセッション参照IDを保持し、サーバーサイドにあるこの参照IDを元にユーザが認証済みであるかなどを判断します。このセッションIDを無効にしたい際は、サーバーサイドでこのセッションIDを削除するだけで実現可能です。
メリット
- セッションをいつでも無効にすることが出来る
- 実装や、1セッションサーバーシナリオの管理が簡単
- セッションデータをその後変更可能
デメリット
- サーバーの管理が大変
セッション管理はRedisなどのデータベースを用意する必要があり、たくさんのログインが集中したりすると、このサーバーのスケーラビリティが問題になってきます。
・ステートレスについて
どう行うのか
ステートレスはステートフルの問題点を解決するべく使われることが多い方法です。よく使われるのはOpenID ConnectやJWTですが、これらの方法はバックエンド側でセッションを管理するDBなどの仕組みが必要がない為、その点の面倒さを解決することは出来ます。
ステートレスの場合は、ユーザのセッション情報をクライアントサイド(ブラウザ)
に保存します。そのためサーバーサイドではペイロードと署名がマッチするかどうかだけを見て有効性チェックを行います。
メリット
- サーバー管理が不要
- スケールアウトが楽
デメリット
- セッションをいつでも無効にできるわけではない
セッション情報がクライアント側にあるため、サーバー側でセッション削除をすることが出来ません。 - セッション情報の変更はセッションの有効期限が切れてからでないと行えない
参考:
・Stateful and stateless authentication
Session vs Token認証
Session
Sessionは前述のステートレスの説明にもあった通り、ユーザーがログインした際にバックエンドでセッションIDを発行し、それをcookieに保存します。そしてリクエストの度にこのcookieに保存された値がサーバに送られ、サーバ側でこのセッションIDを確認します。
Token認証
こちらは大抵localStorageにTokenが保存されて、リクエストの度にヘッダーにこれをつけて送ります。サーバ側ではこれをチェックして、ユーザー情報を取得します。主な二つの違いはToken認証の場合ユーザーの状態はサーバで管理せず、トークンに含まれており、クライアントサイドでのみ保持されます。
ただ、ここで注意ですが、JWTをセッション管理に使う事例をネットではかなりよく見るものの、 推奨はされていません。
なぜかというと、上のステートレス/ステートフルでも書きましたが、JWTはサーバーサイドでの管理が必要なくなる反面、セッションを無効にしたいときに自由にそれが出来ないからです。(JWTのトークンの有効期限が切れるのを待つしかない。)
これが何を引き起こすかというと、例えばログイン/ログアウトという機能があってもユーザがログアウトした際に実際はログアウトしていない、とか何らかの理由があってユーザをシステムからブロックしても実際にはブロックされていないのでシステム内に入って情報を見ることが出来てしまう。管理者から外す、ということをしても、トークンが有効な間は管理者オンリーの機能が使えてしまう、などなど。
これを防ぎつつJWTを使うには、レフレッシュトークンを発行する前提でJWTのトークンの有効期限をなるべく短く設定し、セッションが切れたらリフレッシュトークンを使う。そしてリフレッシュトークンはユーザと紐づけ管理する、というchatworkさんの事例がありました。
まあ便利な仕組みはだからと言って何にでも使えるわけではなく、使い方はよく考えないといけないですね。。
参考:
・ WebアプリケーションでJWTをセッションに使う際の保存先は(自分なりに説明できれば)どちらでもよいと思います
・JWTでセッション管理してはいけない
・そもそもJWTに関する私の理解は完全に間違っていた!
・JSON Web Tokens (JWT) are Dangerous for User Sessions—Here’s a Solution
Cookie
次にcookieについてです。cookieは主にセッション管理やパーソナライゼーション(ターゲティング広告とか)、追跡に使われます。
cookieのセキュリティ面ですが、データ自体はプレーンなテキストの為、漏れて困る情報をそのまま保存するのには向いていません。また、SSLを使っていない無料wifiなどを使っている場合はサーバとのやり取り時に通信から読み取られてしまう可能性もあります。
ただcookieには一押しHttpOnlyフラグを立てることである程度セキュリティレベルを上げることができます。このHttpOnlyフラグがHttpレスポンスヘッダーに含まれていた場合、cookieはクライアントサイドスクリプトにアクセスすることができません。(ブラウザがサポートしていた場合だけですが)
参考:
・What is the difference between localStorage, sessionStorage, session and cookies?
・Everything You Need to Know About Cookies for Web Development
・牧歌的 Cookie の終焉
とまあ、色々な目線からユーザー認証について、その方法や仕組みについてまとめてみました。
お付き合いいただき、ありがとうございました。