Files
Shade/packages/shade-sdk/tests/sdk.test.ts
Sterister dbb3a090d8 release(v4.4.0): public accessor for device identity public key
Expose the local device's 32-byte Ed25519 identity public key on Shade
so apps can hand it to their own backend at enrollment time for
signature verification, key pinning or per-device safety-number
computation. Closes the gap that forced consumers to ship placeholder
random bytes their backend could store but never verify against.

- @shade/sdk Shade.identityPublicKey: Promise<Uint8Array> — getter
  mirrors the existing fingerprint accessor. Throws pre-init,
  reflects the current key after rotate(), retired key preserved in
  retired-identities storage per existing grace-period contract.
  Private key remains unreachable.
- Test in shade-sdk/tests/sdk.test.ts: round-trip match against the
  underlying storage's signingPublicKey, plus value updates after
  rotate().
- Lockstep version bump 4.3.0 → 4.4.0 across all 25 packages.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-05 17:58:45 +02:00

221 lines
7.0 KiB
TypeScript

import { describe, test, expect, beforeEach, afterEach } from 'bun:test';
import { createShade, type Shade } from '../src/index.js';
import {
createPrekeyServer,
MemoryPrekeyStore,
PrekeyServerEvents,
} from '@shade/server';
import { SubtleCryptoProvider, MemoryStorage } from '@shade/crypto-web';
const crypto = new SubtleCryptoProvider();
/**
* Spin up a real prekey server on a random port and return its URL
* + a teardown function.
*/
async function startPrekeyServer(): Promise<{
url: string;
stop: () => void;
events: PrekeyServerEvents;
}> {
const events = new PrekeyServerEvents();
const server = createPrekeyServer({
crypto,
store: new MemoryPrekeyStore(),
disableRateLimit: true,
events,
});
const port = 19500 + Math.floor(Math.random() * 500);
const handle = Bun.serve({ port, fetch: server.fetch });
return {
url: `http://localhost:${port}`,
stop: () => handle.stop(),
events,
};
}
describe('createShade — happy path', () => {
let server: Awaited<ReturnType<typeof startPrekeyServer>>;
let alice: Shade;
let bob: Shade;
beforeEach(async () => {
server = await startPrekeyServer();
});
afterEach(async () => {
await alice?.shutdown();
await bob?.shutdown();
server.stop();
});
test('one-liner creation and initialization', async () => {
alice = await createShade({
prekeyServer: server.url,
address: 'alice',
});
expect(alice.myAddress).toBe('alice');
const fp = await alice.fingerprint;
expect(fp.split(' ').length).toBe(12);
});
test('auto-publishes bundle on init', async () => {
const registered: string[] = [];
server.events.on((e) => {
if (e.name === 'server.identity_registered') {
registered.push(e.data.address);
}
});
alice = await createShade({
prekeyServer: server.url,
address: 'alice',
});
expect(registered).toContain('alice');
});
test('two-process conversation: Alice ↔ Bob via SDK only', async () => {
alice = await createShade({ prekeyServer: server.url, address: 'alice' });
bob = await createShade({ prekeyServer: server.url, address: 'bob' });
// Alice sends to Bob — SDK auto-establishes session
const env1 = await alice.send('bob', 'hello Bob');
const plain1 = await bob.receive('alice', env1);
expect(plain1).toBe('hello Bob');
// Bob replies (DH ratchet triggers)
const env2 = await bob.send('alice', 'hi Alice');
const plain2 = await alice.receive('bob', env2);
expect(plain2).toBe('hi Alice');
});
test('onMessage handler fires on receive', async () => {
alice = await createShade({ prekeyServer: server.url, address: 'alice' });
bob = await createShade({ prekeyServer: server.url, address: 'bob' });
const received: Array<{ from: string; msg: string }> = [];
bob.onMessage((from, msg) => received.push({ from, msg }));
const env = await alice.send('bob', 'callback test');
await bob.receive('alice', env);
expect(received.length).toBe(1);
expect(received[0]!.from).toBe('alice');
expect(received[0]!.msg).toBe('callback test');
});
test('concurrent sends to same new peer establish exactly one session', async () => {
alice = await createShade({ prekeyServer: server.url, address: 'alice' });
bob = await createShade({ prekeyServer: server.url, address: 'bob' });
// Fire 3 parallel sends to Bob
const results = await Promise.all([
alice.send('bob', 'msg1'),
alice.send('bob', 'msg2'),
alice.send('bob', 'msg3'),
]);
// All 3 should succeed and be decryptable in order
expect(results.length).toBe(3);
const decrypted: string[] = [];
for (const env of results) {
decrypted.push(await bob.receive('alice', env));
}
expect(decrypted.sort()).toEqual(['msg1', 'msg2', 'msg3']);
});
test('send to unknown address throws clear error', async () => {
alice = await createShade({ prekeyServer: server.url, address: 'alice' });
await expect(alice.send('nobody', 'ghost')).rejects.toThrow();
});
test('verify fingerprint matches pinned identity', async () => {
alice = await createShade({ prekeyServer: server.url, address: 'alice' });
bob = await createShade({ prekeyServer: server.url, address: 'bob' });
// Establish session
const env = await alice.send('bob', 'init');
await bob.receive('alice', env);
const bobFp = await bob.fingerprint;
// Alice knows Bob via session; but remote fingerprint is derived from
// stored DH key only (not full identity), so we just check it's returned
const remoteFp = await alice.getFingerprintFor('bob');
expect(remoteFp.split(' ').length).toBe(12);
});
test('shutdown clears background timers and closes storage', async () => {
alice = await createShade({
prekeyServer: server.url,
address: 'alice',
autoReplenish: { min: 5, target: 20, intervalMs: 100 },
});
await alice.shutdown();
// If background timer wasn't cleared, this test would hang after the
// final afterEach. We rely on bun test's cleanup to catch that.
});
test('manual replenish is callable', async () => {
alice = await createShade({ prekeyServer: server.url, address: 'alice' });
// Initially has 20 prekeys, so replenish is a no-op
const n = await alice.replenish();
expect(n).toBe(0);
});
test('auto-replenish is disabled when set to false', async () => {
alice = await createShade({
prekeyServer: server.url,
address: 'alice',
autoReplenish: false,
});
expect(alice.myAddress).toBe('alice');
});
test('rotate regenerates identity', async () => {
alice = await createShade({ prekeyServer: server.url, address: 'alice' });
const oldFp = await alice.fingerprint;
await alice.rotate();
const newFp = await alice.fingerprint;
expect(newFp).not.toBe(oldFp);
});
test('identityPublicKey exposes the device Ed25519 key and tracks rotation', async () => {
const storage = new MemoryStorage();
alice = await createShade({ prekeyServer: server.url, address: 'alice', storage });
const pk = await alice.identityPublicKey;
expect(pk).toBeInstanceOf(Uint8Array);
expect(pk.length).toBe(32);
// Matches what the underlying storage holds
const stored = await storage.getIdentityKeyPair();
expect(stored).not.toBeNull();
expect(pk).toEqual(stored!.signingPublicKey);
// Reflects the new key after rotate (acceptance criteria #3)
await alice.rotate();
const pkAfter = await alice.identityPublicKey;
expect(pkAfter.length).toBe(32);
expect(pkAfter).not.toEqual(pk);
});
});
describe('createShade — validation', () => {
test('throws when prekeyServer is missing', async () => {
await expect(createShade({} as any)).rejects.toThrow(/prekeyServer is required/);
});
test('throws when observer token is too short', async () => {
await expect(
createShade({
prekeyServer: 'http://localhost:9999',
observer: { token: 'short' },
}),
).rejects.toThrow(/at least 16/);
});
});