cookie-session vs express-mysql-session vs express-session vs koa-session
Node.js Webフレームワークにおけるセッション管理ライブラリの比較
cookie-sessionexpress-mysql-sessionexpress-sessionkoa-session類似パッケージ:

Node.js Webフレームワークにおけるセッション管理ライブラリの比較

cookie-sessionexpress-mysql-sessionexpress-sessionkoa-session は、Node.js の Web アプリケーションでユーザーのセッション状態を管理するための npm パッケージです。これらのパッケージは、Express や Koa といった人気フレームワークと連携して、HTTP リクエスト間でユーザー情報を保持する仕組みを提供します。cookie-session はセッションデータをクッキー内に保存するのに対し、express-session はサーバー側(メモリまたは外部ストア)に保存します。express-mysql-sessionexpress-session の拡張として MySQL データベースにセッションを永続化し、koa-session は Koa 専用のセッションミドルウェアとして、メモリ保存やクッキー保存、外部ストア対応を柔軟にサポートします。

npmのダウンロードトレンド

3 年

GitHub Starsランキング

統計詳細

パッケージ
ダウンロード数
Stars
サイズ
Issues
公開日時
ライセンス
cookie-session419,9811,14524 kB1610ヶ月前MIT
express-mysql-session031131.1 kB172年前MIT
express-session06,36277.2 kB1104ヶ月前MIT
koa-session0909163 kB531年前MIT

Express と Koa 向けセッション管理ライブラリの技術比較

Node.js の Web アプリケーションでユーザー認証や状態管理を行う際、セッションは欠かせない要素です。cookie-sessionexpress-mysql-sessionexpress-sessionkoa-session はそれぞれ異なるアーキテクチャやユースケースに最適化されたパッケージですが、選択を誤るとセキュリティリスクやスケーラビリティの問題を引き起こす可能性があります。この記事では、実際の開発現場での判断材料となるような、技術的な違いと使い分けポイントを詳しく解説します。

🍪 セッションの保存先:クッキー vs サーバー側ストア

cookie-sessionすべてのセッションデータをクッキー内に保存します。つまり、サーバー側には永続的な状態を持ちません。

// cookie-session (Express)
const express = require('express');
const cookieSession = require('cookie-session');

const app = express();
app.use(cookieSession({
  name: 'session',
  keys: ['key1', 'key2'], // 署名用キー(複数可)
  maxAge: 24 * 60 * 60 * 1000 // 1日
}));

app.get('/', (req, res) => {
  req.session.views = (req.session.views || 0) + 1;
  res.send(`Views: ${req.session.views}`);
});

一方、express-sessionデフォルトでメモリ内にセッションを保存し、本番環境では外部ストア(例:Redis、MongoDB、MySQL)との連携が前提です。

// express-session (Express)
const session = require('express-session');

app.use(session({
  secret: 'keyboard cat',
  resave: false,
  saveUninitialized: false,
  cookie: { secure: false }
}));

// セッションIDのみクッキーに保存され、実データはサーバー側に格納

express-mysql-sessionexpress-sessionストレージ拡張として動作し、セッションデータを MySQL データベースに永続化します。

// express-mysql-session
const MySQLStore = require('express-mysql-session')(session);

const mysqlSessionStore = new MySQLStore({
  host: 'localhost',
  port: 3306,
  user: 'dbuser',
  password: 'password',
  database: 'dbname'
});

app.use(session({
  store: mysqlSessionStore,
  secret: 'secret',
  resave: false,
  saveUninitialized: false
}));

koa-session は Koa 専用で、デフォルトではメモリ保存ですが、externalKey を使うことで外部ストアにも対応可能です。また、設定次第で クッキー保存モードにもできます。

// koa-session (Koa)
const Koa = require('koa');
const session = require('koa-session');

const app = new Koa();
app.keys = ['some secret'];

const CONFIG = {
  key: 'koa:sess',
  maxAge: 86400000,
  overwrite: true,
  httpOnly: true,
  signed: true,
  rolling: false,
  renew: false,
  // store: myCustomStore // 外部ストアを指定可能
};

app.use(session(CONFIG, app));

app.use(ctx => {
  ctx.session.views = (ctx.session.views || 0) + 1;
  ctx.body = `Views: ${ctx.session.views}`;
});

💡 注意:cookie-sessionセッションデータがすべてクライアントに送信されるため、機密情報(例:ユーザーロール、トークン)を含めることは絶対に避けてください。署名付きクッキーでも内容は Base64 でエンコードされるだけで、暗号化はされません。

🔒 セキュリティとデータサイズの制約

クッキーのサイズ制限

HTTP クッキーは 4KB という厳しい制限があります。cookie-session を使う場合、セッションデータがこの制限を超えるとエラーになります。たとえば、OAuth トークンやプロフィール画像のバイナリデータなどを保存しようとすると簡単に超えてしまいます。

セッション固定攻撃への対策

express-sessionkoa-session は、ログイン成功時に req.session.regenerate()(Express)や ctx.session = null(Koa)を使ってセッションIDを再生成することで、セッション固定攻撃を防げます。

// Express: ログイン時のセッション再生成
app.post('/login', (req, res) => {
  authenticate(req.body, (err, user) => {
    if (user) {
      req.session.regenerate((err) => {
        req.session.userId = user.id;
        res.redirect('/');
      });
    }
  });
});

一方、cookie-session は毎回新しい署名付きクッキーを発行するため、自動的にセッションIDが変わるという特性があります。これは一見安全に見えますが、古いクッキーが有効なまま残る可能性があるため、明示的な無効化ロジックが必要です。

🗄️ スケーラビリティとクラスタ対応

アプリケーションが複数のサーバーインスタンスで動作する場合(例:Docker クラスタ、ロードバランサー下)、メモリ保存型のセッションは使用できません。なぜなら、各インスタンスが独立したメモリ空間を持つため、ユーザーが別のインスタンスにアクセスすると「ログアウト」してしまうからです。

  • express-session(デフォルト)→ ❌ クラスタ非対応(メモリ保存)
  • express-session + express-mysql-session → ✅ クラスタ対応(共有DB)
  • koa-session(メモリ)→ ❌ クラスタ非対応
  • koa-session(外部ストア)→ ✅ クラスタ対応
  • cookie-session → ✅ クラスタ対応(状態レス)

ただし、cookie-session のクラスタ対応は「状態を持たない」ことが理由であり、前述のデータサイズやセキュリティ制約を理解した上で使う必要があります。

⚙️ フレームワーク統合とミドルウェア設計

Express vs Koa のアーキテクチャ差

Express は コールバック中心のミドルウェアチェーンを採用しています。そのため、express-sessioncookie-sessionreq.session というプロパティを直接拡張します。

// Express: req.session に直接アクセス
app.use((req, res, next) => {
  req.session.counter = (req.session.counter || 0) + 1;
  next();
});

一方、Koa は async/await とコンテキストオブジェクトctx)を活用します。koa-sessionctx.session という getter/setter を提供し、遅延読み込み(lazy loading)によりパフォーマンスを最適化しています。

// Koa: ctx.session はプロキシオブジェクト
app.use(async (ctx, next) => {
  ctx.session.counter = (ctx.session.counter || 0) + 1;
  await next();
});

この違いにより、Koa の方が非同期処理との相性が良く、エラーハンドリングも直感的です。

🧪 実際の選定シナリオ

シナリオ1:シンプルな個人プロジェクト(ステートレス優先)

  • 要件:ユーザー認証あり、機密データなし、小規模
  • 推奨: cookie-session
  • 理由:外部DB不要、セットアップが簡単、クラスタ対応

シナリオ2:企業向けWebアプリ(高セキュリティ・大規模)

  • 要件:機密情報保存、複数サーバー、監査ログ必要
  • 推奨: express-session + express-mysql-session(または Redis ストア)
  • 理由:セッションデータを安全に隔離、柔軟なTTL制御、監査対応

シナリオ3:Koa ベースのモダンAPIサーバー

  • 要件:軽量、非同期重視、将来的に外部ストア移行予定
  • 推奨: koa-session
  • 理由:Koa の非同期モデルと完全互換、外部ストアへの移行も容易

📌 まとめ:各パッケージの特徴と注意点

パッケージ保存先フレームワーククラスタ対応機密データ可主な用途
cookie-sessionクッキーExpress小規模アプリ、ステートレス設計
express-sessionメモリ(デフォルト)Express単一サーバー開発環境
express-mysql-sessionMySQLExpress本番環境向け永続化
koa-sessionメモリ or クッキー or 外部Koa✅(外部時)Koa アプリ全般

💡 最終アドバイス

  • 機密情報を扱うなら、絶対に cookie-session を使わないでください。代わりに express-sessionkoa-session と信頼できる外部ストア(Redis が一般的)を組み合わせましょう。
  • Koa を使うなら、koa-session 一択です。Express 向けのセッションミドルウェアは Koa と互換性がありません。
  • MySQL にこだわる必要はありませんexpress-mysql-session は特定のDBに依存するため、将来的にPostgreSQLやMongoDBに移行したい場合は、より汎用的な connect-redis のようなパッケージを検討しましょう。

セッション管理は「動けばいい」と思われがちですが、適切な選択はアプリケーションのセキュリティ、スケーラビリティ、保守性に大きく影響します。今回の比較を参考に、プロジェクトの要件に合ったライブラリを選んでください。

選び方: cookie-session vs express-mysql-session vs express-session vs koa-session

  • cookie-session:

    cookie-session は、セッションデータをすべてクッキー内に保存する軽量なソリューションです。機密情報が含まれず、データサイズが小さい(4KB未満)シンプルなアプリケーションや、ステートレスなアーキテクチャを採用したい場合に適しています。ただし、クッキー内のデータは署名されても暗号化されないため、パスワードやトークンなどの機密情報を保存してはいけません。クラスタ環境でも動作しますが、セキュリティ要件が高い本番環境では避けるべきです。

  • express-mysql-session:

    express-mysql-session は、express-session のセッションストアとして MySQL データベースを利用する拡張パッケージです。既に MySQL を利用しており、セッションデータを永続化・監査可能な形で管理したい場合に最適です。ただし、MySQL に強く依存するため、将来的に他のデータベース(例:Redis、PostgreSQL)に移行する可能性がある場合は、より汎用的なストア選択を検討してください。また、データベース接続の負荷や可用性も考慮する必要があります。

  • express-session:

    express-session は Express 向けの標準的なセッションミドルウェアで、デフォルトではメモリにセッションを保存します。開発環境や単一サーバーでの小規模アプリケーションに適していますが、本番環境では必ず外部ストア(例:Redis、MongoDB)と組み合わせて使用する必要があります。クラスタ環境ではメモリ保存が機能しないため、スケーラビリティを考慮した設計が必須です。機密情報を安全に扱える点が最大のメリットです。

  • koa-session:

    koa-session は Koa 専用のセッションミドルウェアで、メモリ保存、クッキー保存、外部ストア対応を柔軟に切り替えられます。Koa アプリケーションを構築している場合、これが唯一の自然な選択肢です。非同期処理との相性が良く、コンテキストベースの設計によりコードが簡潔になります。将来的に外部ストアに移行する可能性がある場合でも、初期段階でメモリ保存から始められるため、開発の柔軟性が高いです。

cookie-session のREADME

cookie-session

NPM Version NPM Downloads Build Status Test Coverage

Simple cookie-based session middleware.

A user session can be stored in two main ways with cookies: on the server or on the client. This module stores the session data on the client within a cookie, while a module like express-session stores only a session identifier on the client within a cookie and stores the session data on the server, typically in a database.

The following points can help you choose which to use:

  • cookie-session does not require any database / resources on the server side, though the total session data cannot exceed the browser's max cookie size.
  • cookie-session can simplify certain load-balanced scenarios.
  • cookie-session can be used to store a "light" session and include an identifier to look up a database-backed secondary store to reduce database lookups.

NOTE This module does not encrypt the session contents in the cookie, only provides signing to prevent tampering. The client will be able to read the session data by examining the cookie's value. Secret data should not be set in req.session without encrypting it, or use a server-side session instead.

NOTE This module does not prevent session replay, as the expiration set is that of the cookie only; if that is a concern of your application, you can store an expiration date in req.session object and validate it on the server, and implement any other logic to extend the session as your application needs.

Install

This is a Node.js module available through the npm registry. Installation is done using the npm install command:

$ npm install cookie-session

API

var cookieSession = require('cookie-session')
var express = require('express')

var app = express()

app.use(cookieSession({
  name: 'session',
  keys: [/* secret keys */],

  // Cookie Options
  maxAge: 24 * 60 * 60 * 1000 // 24 hours
}))

cookieSession(options)

Create a new cookie session middleware with the provided options. This middleware will attach the property session to req, which provides an object representing the loaded session. This session is either a new session if no valid session was provided in the request, or a loaded session from the request.

The middleware will automatically add a Set-Cookie header to the response if the contents of req.session were altered. Note that no Set-Cookie header will be in the response (and thus no session created for a specific user) unless there are contents in the session, so be sure to add something to req.session as soon as you have identifying information to store for the session.

Options

Cookie session accepts these properties in the options object.

name

The name of the cookie to set, defaults to session.

keys

The list of keys to use to sign & verify cookie values, or a configured Keygrip instance. Set cookies are always signed with keys[0], while the other keys are valid for verification, allowing for key rotation. If a Keygrip instance is provided, it can be used to change signature parameters like the algorithm of the signature.

secret

A string which will be used as single key if keys is not provided.

Cookie Options

Other options are passed to cookies.get() and cookies.set() allowing you to control security, domain, path, and signing among other settings.

The options can also contain any of the following (for the full list, see cookies module documentation:

  • maxAge: a number representing the milliseconds from Date.now() for expiry
  • expires: a Date object indicating the cookie's expiration date (expires at the end of session by default).
  • path: a string indicating the path of the cookie (/ by default).
  • domain: a string indicating the domain of the cookie (no default).
  • partitioned: a boolean indicating whether to partition the cookie in Chrome for the CHIPS Update (false by default). If this is true, Cookies from embedded sites will be partitioned and only readable from the same top level site from which it was created.
  • priority: a string indicating the cookie priority. This can be set to 'low', 'medium', or 'high'.
  • sameSite: a boolean or string indicating whether the cookie is a "same site" cookie (false by default). This can be set to 'strict', 'lax', 'none', or true (which maps to 'strict').
  • secure: a boolean indicating whether the cookie is only to be sent over HTTPS (false by default for HTTP, true by default for HTTPS). If this is set to true and Node.js is not directly over a TLS connection, be sure to read how to setup Express behind proxies or the cookie may not ever set correctly.
  • httpOnly: a boolean indicating whether the cookie is only to be sent over HTTP(S), and not made available to client JavaScript (true by default).
  • signed: a boolean indicating whether the cookie is to be signed (true by default).
  • overwrite: a boolean indicating whether to overwrite previously set cookies of the same name (true by default).

req.session

Represents the session for the given request.

.isChanged

Is true if the session has been changed during the request.

.isNew

Is true if the session is new.

.isPopulated

Determine if the session has been populated with data or is empty.

req.sessionOptions

Represents the session options for the current request. These options are a shallow clone of what was provided at middleware construction and can be altered to change cookie setting behavior on a per-request basis.

Destroying a session

To destroy a session simply set it to null:

req.session = null

Saving a session

Since the entire contents of the session is kept in a client-side cookie, the session is "saved" by writing a cookie out in a Set-Cookie response header. This is done automatically if there has been a change made to the session when the Node.js response headers are being written to the client and the session was not destroyed.

Examples

Simple view counter example

var cookieSession = require('cookie-session')
var express = require('express')

var app = express()

app.set('trust proxy', 1) // trust first proxy

app.use(cookieSession({
  name: 'session',
  keys: ['key1', 'key2']
}))

app.get('/', function (req, res, next) {
  // Update views
  req.session.views = (req.session.views || 0) + 1

  // Write response
  res.end(req.session.views + ' views')
})

app.listen(3000)

Per-user sticky max age

var cookieSession = require('cookie-session')
var express = require('express')

var app = express()

app.set('trust proxy', 1) // trust first proxy

app.use(cookieSession({
  name: 'session',
  keys: ['key1', 'key2']
}))

// This allows you to set req.session.maxAge to let certain sessions
// have a different value than the default.
app.use(function (req, res, next) {
  req.sessionOptions.maxAge = req.session.maxAge || req.sessionOptions.maxAge
  next()
})

// ... your logic here ...

Extending the session expiration

This module does not send a Set-Cookie header if the contents of the session have not changed. This means that to extend the expiration of a session in the user's browser (in response to user activity, for example) some kind of modification to the session needs be made.

var cookieSession = require('cookie-session')
var express = require('express')

var app = express()

app.use(cookieSession({
  name: 'session',
  keys: ['key1', 'key2']
}))

// Update a value in the cookie so that the set-cookie will be sent.
// Only changes every minute so that it's not sent with every request.
app.use(function (req, res, next) {
  req.session.nowInMinutes = Math.floor(Date.now() / 60e3)
  next()
})

// ... your logic here ...

Using a custom signature algorithm

This example shows creating a custom Keygrip instance as the keys option to provide keys and additional signature configuration.

var cookieSession = require('cookie-session')
var express = require('express')
var Keygrip = require('keygrip')

var app = express()

app.use(cookieSession({
  name: 'session',
  keys: new Keygrip(['key1', 'key2'], 'SHA384', 'base64')
}))

// ... your logic here ...

Usage Limitations

Max Cookie Size

Because the entire session object is encoded and stored in a cookie, it is possible to exceed the maximum cookie size limits on different browsers. The RFC6265 specification recommends that a browser SHOULD allow

At least 4096 bytes per cookie (as measured by the sum of the length of the cookie's name, value, and attributes)

In practice this limit differs slightly across browsers. See a list of browser limits here. As a rule of thumb don't exceed 4093 bytes per domain.

If your session object is large enough to exceed a browser limit when encoded, in most cases the browser will refuse to store the cookie. This will cause the following requests from the browser to either a) not have any session information or b) use old session information that was small enough to not exceed the cookie limit.

If you find your session object is hitting these limits, it is best to consider if data in your session should be loaded from a database on the server instead of transmitted to/from the browser with every request. Or move to an alternative session strategy

License

MIT