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 🙏

© 2026 – Pkg Stats / Ryan Hefner

@yoda.digital/mycelium

v0.1.1

Published

End-to-end encrypted messaging between Claude Code instances. Curve25519 PFS, Ed25519 TOFU, replay protection, stateless relay.

Downloads

36

Readme

E2E encrypted messaging between Claude Code instances. ~2100 lines of TypeScript, runs on Bun.

Listed on Yoda Digital Open Source npm License: MIT Bun


Overview

Mycelium is a zero-trust messaging layer for Claude Code. The relay is a stateless router — it never sees plaintext. Even a fully compromised relay cannot read, forge, or silently drop messages.

Threat model

| If the relay is compromised, can it... | Answer | |---|---| | Read messages? | No. Ephemeral Curve25519 keys, shared secret never touches the wire. | | MITM key exchange? | No. Ed25519 TOFU + STS mutual authentication. Even first contact is protected. | | Forge messages? | No. Canonical Ed25519 signatures. Bad sig = hard block, not a warning. | | Replay old messages with new IDs? | No. msg_id and seq are inside the signature. Tamper = sig fails. | | Forge permission approvals? | No. Permission messages use the same E2E envelope as everything else. | | Hijack a peer name? | No. Names are bound to identity keys. Different key claiming same name = rejected. | | Drop messages silently? | No. E2E encrypted delivery acks. 30s timeout, then you're told. | | Decrypt old sessions? | No. Ephemeral keys live in memory only. New session = new keys. |

Crypto stack

  • Identity: Ed25519 signing keypair, persisted to ~/.mycelium-keys.json
  • Session encryption: Curve25519 ephemeral keypair, generated per connection (PFS)
  • Authenticated encryption: NaCl crypto_box (XSalsa20-Poly1305) via libsodium WASM (constant-time)
  • Signatures: Ed25519 detached, over canonical JSON (sorted keys, 11 fields including msg_id + seq + request_id)
  • Identity pinning: TOFU fail-closed + STS mutual authentication (eliminates first-contact MITM)
  • Relay authentication: Ed25519 challenge-response. Token is a one-time invite; known peers auth via cryptographic identity.
  • Relay identity: Relay has its own Ed25519 keypair. Peers verify fingerprint before sending credentials.
  • Replay protection: msg_id dedup (write-ahead log) + reorder buffer + 30min time-based expiry
  • Broadcast: N x unicast (each peer gets independently encrypted copy)

Setup

Start the relay:

bun install
RELAY_TOKEN=$(openssl rand -hex 32) bun run relay.ts

Add to your Claude Code MCP config:

{
  "mcpServers": {
    "mycelium": {
      "command": "bun",
      "args": ["/path/to/peer-channel.ts"],
      "env": {
        "MYC_RELAY": "wss://relay1.example.com,wss://relay2.example.com",
        "MYC_TOKEN": "your-token",
        "MYC_PEER": "unique-name",
        "MYC_RELAY_FINGERPRINT": "a1b2:c3d4:e5f6:..."
      }
    }
  }
}

Then tell Claude Code to load it:

claude --dangerously-load-development-channels server:mycelium

Each peer needs a unique MYC_PEER name and the same MYC_TOKEN. If running multiple peers on the same machine, each must have its own MYC_KEY_FILE — otherwise they share one identity keypair and cause TOFU violations:

"MYC_KEY_FILE": "~/.mycelium-keys-alice.json"

Environment variables

Relay (relay.ts)

| Variable | Default | What it does | |---|---|---| | RELAY_TOKEN | required | Shared auth token (one-time invite for new peers) | | RELAY_PORT | 9900 | Listen port | | RELAY_MAX_PEERS | 50 | Max peers per room | | RELAY_MAX_MSG_BYTES | 65536 | Max message size | | RELAY_RATE_LIMIT | 300 | Messages/minute per peer | | RELAY_QUEUE_MAX_MSGS | 50 | Offline queue depth | | RELAY_QUEUE_TTL_S | 300 | Offline message TTL | | RELAY_REQUIRE_TLS | false | Refuse non-TLS connections | | RELAY_TRUSTED_PROXY | false | Trust X-Forwarded-For | | RELAY_MAX_IP_CONNS | 10 | Max connections per IP | | RELAY_KEY_FILE | ~/.mycelium-relay-keys.json | Relay Ed25519 identity keypair | | RELAY_ALLOW_FILE | ~/.mycelium-relay-allow.json | Per-room peer allow-list | | RELAY_REQUIRE_CHALLENGE | false | Require challenge-response (reject token-only) |

Peer (peer-channel.ts)

| Variable | Default | What it does | |---|---|---| | MYC_RELAY | required | WebSocket URL(s), comma-separated for failover | | MYC_TOKEN | required | Auth token (one-time invite; not needed after first auth) | | MYC_PEER | required | This peer's name | | MYC_ROOM | default | Room to join | | MYC_KEY_FILE | ~/.mycelium-keys.json | Ed25519 identity keypair | | MYC_TOFU_FILE | ~/.mycelium-known-peers.json | TOFU pinned keys | | MYC_REPLAY_FILE | ~/.mycelium-replay-state.json | Replay protection state | | MYC_RELAY_FINGERPRINT | (none) | Expected relay fingerprint (e.g., a1b2:c3d4:...) |

Use per-project MYC_TOFU_FILE and MYC_REPLAY_FILE paths if you don't want TOFU state leaking across projects.

Set MYC_RELAY_FINGERPRINT to pin relay identity — the peer will refuse to send credentials to an unverified relay.

MCP tools

| Tool | Description | |---|---| | myc_send | Send encrypted message to a specific peer (supports request_id for correlation) | | myc_broadcast | Send encrypted message to all peers (N x unicast) | | myc_peers | List connected peers with TOFU and encryption status | | myc_trust | Override a TOFU block after out-of-band fingerprint verification |

Security

Mycelium addresses the full spectrum of relay-trust vulnerabilities:

| Attack vector | Mitigation | |---|---| | Shared-token auth | Ed25519 challenge-response. Token is a one-time invite; known peers auth via cryptographic identity. | | Timing side-channels | libsodium WASM with audited constant-time operations. | | Message reordering | Request-ID correlation + sequence-based reorder buffer. | | Single point of failure | Multi-relay client failover via comma-separated MYC_RELAY URLs. | | First-contact MITM | STS (Station-to-Station) mutual authentication post-handshake. | | Relay impersonation | Relay Ed25519 identity verification + sealed (encrypted) auth tokens. |

Full audit trail and findings are documented in docs/source/README.md.

Testing

bun run test.ts

75 tests covering relay infrastructure, crypto protocol, and E2E integration scenarios.

Architecture

| File | Purpose | |---|---| | relay.ts | WebSocket relay server. Challenge-response auth, Ed25519 identity, allow-list, offline queues. | | peer-channel.ts | MCP server. E2E encryption, STS mutual auth, TOFU, reorder buffer, multi-relay failover. | | test.ts | Test suite covering infrastructure, crypto protocol, and integration. | | package.json | Dependencies: libsodium-wrappers-sumo, @modelcontextprotocol/sdk, zod. |