npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2024 – Pkg Stats / Ryan Hefner

hono-kv-session

v1.0.0

Published

Stateful session middleware for Hono. Works with Cloudflare KV or node-redis.

Downloads

23

Readme

hono-kv-session

Stateful session middleware for Hono. Works on Cloudflare Workers, Node.js, Bun, Deno, etc.

ステートフルなセッションを提供するHonoのミドルウェアです。Cloudflare WorkersやNode.js、Bunなどの環境で動作します。

日本語版ドキュメント

ToDo

  • Support MongoDB
  • Support Fastly KV store
  • Support AWS DynamoDB

Supported Key-Value stores

  • Cloudflare KV
  • Redis (with node-redis)
  • Deno KV
  • New! Cloudflare D1 (sqlite)

Supported Runtimes

| Supported | Runtime | Tested | | --- | --- | --- | | ✔️ | Bun | ✔️ | | ✔️ | Cloudflare Workers | ✔️ | | ✔️ | Cloudflare Pages (Functions) | | | ✔️ | Node.js | ✔️ | | ✔️ | Deno (with Redis) | ✔️ | | ✔️ | Deno KV | ✔️ |

Installation

npm install hono-kv-session

Setup KV

  • Cloudflare Workers
    1. Create KV namespace the bind name SESSION.
      $ wrangler kv:namespace create SESSION
    2. Set the UUID of the KV namespace generated in 1. to wrangler.toml.
      Like, { binding = "SESSION", id = "b80d8fc5924d43ba85b56aa6b6dbb1c3" }
  • Bun, Node.js, Deno, etc.
    1. Just start the Redis server.
      For systemd: # systemctl start redis-server
  • Deno KV
    Deno KV is currently experimental
    Executing Deno programs with --unstable flag, as below:
    $ deno run --allow-net --watch --unstable app.ts
  • Cloudflare D1
    1. Create D1 Database. $ wrangler d1 create session-db
    2. Update D1's database_id to wrangler.toml.
      [[ d1_databases ]]
      binding = "SESSION_DB"
      database_name = "session-db"
      database_id = "<Unique ID for Your Database Here>"
      preview_database_id = "local"
    3. Run $ npm run d1:init

Usage

You can see the sample code in the ./dev directory in Github.

Mount kv client

  • Cloudflare Workers, Cloudflare Pages

    import { kvClient } from 'hono-kv-session/cloudflare';
    app.use('*', kvClient());
  • Node.js, Bun, Deno (with Redis)

    import { kvClient } from 'hono-kv-session/redis';
    app.use('*', kvClient());
    
    // or You can set any node-redis's createClient() options
    app.use('*', kvClient({
      url: 'redis://alice:[email protected]:6380'
    }));
  • Deno KV

    import { kvClient } from 'https://deno.land/x/hono_kv_session/kv/denokv.js';
    app.use('*', kvClient());
  • Cloudflare D1

    import { kvClient } from 'hono-kv-session/d1';
    app.use('*', kvClient());

Use SessionManager

  • Set SessionManager() middleware.

    import { SessionManager, createSession, deleteSession } from 'hono-kv-session' // If you are using Deno, replace module name to "npm:hono-kv-session"
      
    app.use('*', SessionManager({
      // Cookie's id
      name: 'session_cookie' // Default: 'id'
      
      // Secret for Hono's signed cookies
      secret: 'Strong_Secret_123' // Default: null
    
      // Session TTL. Set for both KV and cookies. Minimum 60.
      ttl: 60, // Default: 604800 (1 week)
    
      // Update session TTL for each access.
      renew: true, // Default: true
    
      // Update session ID for each access.
      regenerate: true, // Default: false
    }))
    • secret is secret of Hono's signed cookies (This feature has untested).
      See Hono's Cookie Helper documentation for details.
  • Get session data

    app.get('/', async (c) => {
      const { value, key, name, status } = c.session;
      return c.json({
        username: value,
        session_id: key, // Default: crypto.randomUUID()'s uuid
        cookie_id: name,
        status,
      })
    })
  • Deny Access If you don’t use the denyAccess() middleware, unauthorized sessions will not be denied and will be able to access the system. By referencing c.session.status = true|false, you can restrict access to specific routes or HTTP methods.

    import { denyAccess } from 'hono-kv-session';
    
    // If JSON
    app.use('*', denyAccess({
      type: 'json', // 'json' or 'html' or 'text'
      status: 401, // status code
      response: { status: false, message: 'Invalid session' }
    }))
    
    // If HTML
    app.use('*', denyAccess({
      type: 'html', // 'json' or 'html' or 'text'
      status: 401, // status code
      response: '<p>Invalid session</p>'
    }));
  • Create session

    app.post('/login', async (c) => {
      // Extract client's username from FormData
      const { user } = await c.req.parseBody()
      
      // Create session
      await createSession(c, user, {
        secret: 'Strong_Secret_123' // If you are using signed cookie
      })
      
      return c.redirect('/')
    })
  • Renewal session

    app.get('/renew', async (c) => {
      await renewSession(c)
      return c.redirect('/')
    })
  • Delete session

    app.post('/logout', async (c) => {
      await deleteSession(c)
      return c.redirect('/')
    })

Session Format

  • in Key-Value store: session:<hostname>:<uuid> and value
    Key: session:www.example.com:49b0b962-5b95-43c6-9e00-94ce1313d0ed
    Value: user01
  • in Cookie: id=49b0b962-5b95-43c6-9e00-94ce1313d0ed
  • in c.session
    c.session = {
      session: 'user01' // KV value
      key: `49b0b962-5b95-43c6-9e00-94ce1313d0ed` // KV key
      name: 'id' // Cookie name
    }

Dependecies

License

MIT