TypeScript implementation of the Marmot protocol — end-to-end encrypted group messaging on Nostr using MLS (Messaging Layer Security).
This library is in Alpha and under heavy development. The API is subject to breaking changes without notice. It relies on ts-mls for MLS cryptographic guarantees. Do not use in production yet.
GenericKeyValueStore backend (LocalForage, IndexedDB, in-memory, …)marmot-ts currently supports the following Marmot Improvement Proposals (MIPs):
| MIP | Description | Status |
|---|---|---|
| MIP-00 | Introduction and Basic Operations | ✅ Supported |
| MIP-01 | Network Transport & Relay Communication | ✅ Supported |
| MIP-02 | Identities and Keys | ✅ Supported |
| MIP-03 | Group State & Memberships | ✅ Supported |
npm install @internet-privacy/marmot-ts
# or
pnpm add @internet-privacy/marmot-ts
A MarmotClient needs four things to operate:
EventSigner) — signs Nostr events on behalf of the user.NostrNetworkInterface) — publishes, requests, and subscribes to events on relays.Both stores share a single interface: GenericKeyValueStore<T>.
interface GenericKeyValueStore<T> {
getItem(key: string): Promise<T | null>;
setItem(key: string, value: T): Promise<T>;
removeItem(key: string): Promise<void>;
clear(): Promise<void>;
keys(): Promise<string[]>;
}
Any backend that matches this shape works. LocalForage instances satisfy it directly:
import localforage from "localforage";
const groupStateStore = localforage.createInstance({ name: "marmot-groups" });
const keyPackageStore = localforage.createInstance({ name: "marmot-keys" });
For tests or short-lived processes, the library ships an in-memory implementation:
import { InMemoryKeyValueStore } from "@internet-privacy/marmot-ts";
const groupStateStore = new InMemoryKeyValueStore();
const keyPackageStore = new InMemoryKeyValueStore();
import { MarmotClient } from "@internet-privacy/marmot-ts";
const client = new MarmotClient({
signer, // your EventSigner (e.g. from applesauce-core)
network, // your NostrNetworkInterface implementation
groupStateStore, // GenericKeyValueStore<SerializedClientState>
keyPackageStore, // GenericKeyValueStore<StoredKeyPackage>
clientId: "my-app-desktop", // stable d-tag for kind 30443 key packages
});
Other users invite you by referencing a key package you've published to relays.
await client.keyPackages.create({
relays: ["wss://relay.example.com"],
});
const group = await client.groups.create("My Secret Group", {
description: "A private discussion",
relays: ["wss://relay.example.com"],
});
await group.sendChatMessage("Hello, world!");
Look up their key package event on a relay, then invite by event:
const [keyPackageEvent] = await client.network.request(
["wss://relay.example.com"],
[{ kinds: [30443], authors: [memberPubkey], limit: 1 }],
);
if (keyPackageEvent) {
await group.inviteByKeyPackageEvent(keyPackageEvent);
}
When you receive a kind 1059 gift wrap, decrypt it to a kind 444 rumor and pass it to joinGroupFromWelcome:
const { group } = await client.joinGroupFromWelcome({ welcomeRumor });
Subscribe to the group's relays for kind 445 events and feed them to group.ingest:
import { bytesToHex } from "@noble/hashes/utils.js";
const subscription = client.network.subscription(group.relays, [
{ kinds: [445], "#h": [bytesToHex(group.groupData.nostrGroupId)] },
]);
subscription.subscribe({
next: async (event) => {
for await (const result of group.ingest([event])) {
if (result.kind === "applicationMessage") {
console.log(result.message);
}
}
},
});
Full documentation is in docs/ and served via VitePress. Run pnpm docs:dev to browse locally.
MarmotClient, MarmotGroup, storage, network, UI integrationpnpm install # Install dependencies
pnpm build # Compile TypeScript
pnpm test # Run tests (watch mode)
pnpm format # Format code with Prettier
pnpm docs:dev # Serve documentation locally
pnpm docs:build # Build documentation