@did-btcr2/method 0.29.0 → 0.32.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +13 -5
- package/dist/.tsbuildinfo +1 -1
- package/dist/browser.js +8174 -7157
- package/dist/browser.mjs +8174 -7157
- package/dist/cjs/index.js +1845 -455
- package/dist/esm/core/aggregation/transport/factory.js +15 -6
- package/dist/esm/core/aggregation/transport/factory.js.map +1 -1
- package/dist/esm/core/aggregation/transport/http/client.js +350 -0
- package/dist/esm/core/aggregation/transport/http/client.js.map +1 -0
- package/dist/esm/core/aggregation/transport/http/envelope.js +126 -0
- package/dist/esm/core/aggregation/transport/http/envelope.js.map +1 -0
- package/dist/esm/core/aggregation/transport/http/errors.js +11 -0
- package/dist/esm/core/aggregation/transport/http/errors.js.map +1 -0
- package/dist/esm/core/aggregation/transport/http/inbox-buffer.js +45 -0
- package/dist/esm/core/aggregation/transport/http/inbox-buffer.js.map +1 -0
- package/dist/esm/core/aggregation/transport/http/index.js +12 -0
- package/dist/esm/core/aggregation/transport/http/index.js.map +1 -0
- package/dist/esm/core/aggregation/transport/http/nonce-cache.js +38 -0
- package/dist/esm/core/aggregation/transport/http/nonce-cache.js.map +1 -0
- package/dist/esm/core/aggregation/transport/http/protocol.js +28 -0
- package/dist/esm/core/aggregation/transport/http/protocol.js.map +1 -0
- package/dist/esm/core/aggregation/transport/http/rate-limiter.js +45 -0
- package/dist/esm/core/aggregation/transport/http/rate-limiter.js.map +1 -0
- package/dist/esm/core/aggregation/transport/http/request-auth.js +100 -0
- package/dist/esm/core/aggregation/transport/http/request-auth.js.map +1 -0
- package/dist/esm/core/aggregation/transport/http/server.js +481 -0
- package/dist/esm/core/aggregation/transport/http/server.js.map +1 -0
- package/dist/esm/core/aggregation/transport/http/sse-stream.js +110 -0
- package/dist/esm/core/aggregation/transport/http/sse-stream.js.map +1 -0
- package/dist/esm/core/aggregation/transport/http/sse-writer.js +25 -0
- package/dist/esm/core/aggregation/transport/http/sse-writer.js.map +1 -0
- package/dist/esm/core/aggregation/transport/index.js +1 -0
- package/dist/esm/core/aggregation/transport/index.js.map +1 -1
- package/dist/esm/core/beacon/beacon.js +197 -51
- package/dist/esm/core/beacon/beacon.js.map +1 -1
- package/dist/esm/core/beacon/cas-beacon.js +3 -3
- package/dist/esm/core/beacon/cas-beacon.js.map +1 -1
- package/dist/esm/core/beacon/singleton-beacon.js +3 -3
- package/dist/esm/core/beacon/singleton-beacon.js.map +1 -1
- package/dist/esm/core/beacon/smt-beacon.js +3 -3
- package/dist/esm/core/beacon/smt-beacon.js.map +1 -1
- package/dist/esm/core/updater.js +63 -55
- package/dist/esm/core/updater.js.map +1 -1
- package/dist/types/core/aggregation/transport/factory.d.ts +22 -7
- package/dist/types/core/aggregation/transport/factory.d.ts.map +1 -1
- package/dist/types/core/aggregation/transport/http/client.d.ts +48 -0
- package/dist/types/core/aggregation/transport/http/client.d.ts.map +1 -0
- package/dist/types/core/aggregation/transport/http/envelope.d.ts +64 -0
- package/dist/types/core/aggregation/transport/http/envelope.d.ts.map +1 -0
- package/dist/types/core/aggregation/transport/http/errors.d.ts +9 -0
- package/dist/types/core/aggregation/transport/http/errors.d.ts.map +1 -0
- package/dist/types/core/aggregation/transport/http/inbox-buffer.d.ts +32 -0
- package/dist/types/core/aggregation/transport/http/inbox-buffer.d.ts.map +1 -0
- package/dist/types/core/aggregation/transport/http/index.d.ts +12 -0
- package/dist/types/core/aggregation/transport/http/index.d.ts.map +1 -0
- package/dist/types/core/aggregation/transport/http/nonce-cache.d.ts +26 -0
- package/dist/types/core/aggregation/transport/http/nonce-cache.d.ts.map +1 -0
- package/dist/types/core/aggregation/transport/http/protocol.d.ts +53 -0
- package/dist/types/core/aggregation/transport/http/protocol.d.ts.map +1 -0
- package/dist/types/core/aggregation/transport/http/rate-limiter.d.ts +41 -0
- package/dist/types/core/aggregation/transport/http/rate-limiter.d.ts.map +1 -0
- package/dist/types/core/aggregation/transport/http/request-auth.d.ts +50 -0
- package/dist/types/core/aggregation/transport/http/request-auth.d.ts.map +1 -0
- package/dist/types/core/aggregation/transport/http/server.d.ts +110 -0
- package/dist/types/core/aggregation/transport/http/server.d.ts.map +1 -0
- package/dist/types/core/aggregation/transport/http/sse-stream.d.ts +34 -0
- package/dist/types/core/aggregation/transport/http/sse-stream.d.ts.map +1 -0
- package/dist/types/core/aggregation/transport/http/sse-writer.d.ts +12 -0
- package/dist/types/core/aggregation/transport/http/sse-writer.d.ts.map +1 -0
- package/dist/types/core/aggregation/transport/index.d.ts +1 -0
- package/dist/types/core/aggregation/transport/index.d.ts.map +1 -1
- package/dist/types/core/aggregation/transport/transport.d.ts +1 -1
- package/dist/types/core/aggregation/transport/transport.d.ts.map +1 -1
- package/dist/types/core/beacon/beacon.d.ts +72 -12
- package/dist/types/core/beacon/beacon.d.ts.map +1 -1
- package/dist/types/core/beacon/cas-beacon.d.ts +3 -3
- package/dist/types/core/beacon/cas-beacon.d.ts.map +1 -1
- package/dist/types/core/beacon/singleton-beacon.d.ts +3 -3
- package/dist/types/core/beacon/singleton-beacon.d.ts.map +1 -1
- package/dist/types/core/beacon/smt-beacon.d.ts +3 -3
- package/dist/types/core/beacon/smt-beacon.d.ts.map +1 -1
- package/dist/types/core/updater.d.ts +27 -12
- package/dist/types/core/updater.d.ts.map +1 -1
- package/package.json +5 -5
- package/src/core/aggregation/transport/factory.ts +48 -12
- package/src/core/aggregation/transport/http/client.ts +409 -0
- package/src/core/aggregation/transport/http/envelope.ts +204 -0
- package/src/core/aggregation/transport/http/errors.ts +11 -0
- package/src/core/aggregation/transport/http/inbox-buffer.ts +53 -0
- package/src/core/aggregation/transport/http/index.ts +11 -0
- package/src/core/aggregation/transport/http/nonce-cache.ts +43 -0
- package/src/core/aggregation/transport/http/protocol.ts +57 -0
- package/src/core/aggregation/transport/http/rate-limiter.ts +75 -0
- package/src/core/aggregation/transport/http/request-auth.ts +164 -0
- package/src/core/aggregation/transport/http/server.ts +615 -0
- package/src/core/aggregation/transport/http/sse-stream.ts +121 -0
- package/src/core/aggregation/transport/http/sse-writer.ts +23 -0
- package/src/core/aggregation/transport/index.ts +1 -0
- package/src/core/aggregation/transport/transport.ts +1 -1
- package/src/core/beacon/beacon.ts +255 -64
- package/src/core/beacon/cas-beacon.ts +4 -4
- package/src/core/beacon/singleton-beacon.ts +4 -4
- package/src/core/beacon/smt-beacon.ts +4 -4
- package/src/core/updater.ts +113 -67
|
@@ -0,0 +1,481 @@
|
|
|
1
|
+
import { CompressedSecp256k1PublicKey } from '@did-btcr2/keypair';
|
|
2
|
+
import { Identifier } from '../../../identifier.js';
|
|
3
|
+
import { CONSOLE_LOGGER } from '../../logger.js';
|
|
4
|
+
import { reviveFromWire, signEnvelope, verifyEnvelope } from './envelope.js';
|
|
5
|
+
import { HttpTransportError } from './errors.js';
|
|
6
|
+
import { InboxBuffer } from './inbox-buffer.js';
|
|
7
|
+
import { NonceCache } from './nonce-cache.js';
|
|
8
|
+
import { DEFAULT_CLOCK_SKEW_SEC, HTTP_ENVELOPE_VERSION, HTTP_ROUTE, SSE_EVENT, } from './protocol.js';
|
|
9
|
+
import { RateLimiter } from './rate-limiter.js';
|
|
10
|
+
import { verifyRequestAuth } from './request-auth.js';
|
|
11
|
+
const INBOX_PATH_PREFIX = '/v1/actors/';
|
|
12
|
+
const INBOX_PATH_SUFFIX = '/inbox';
|
|
13
|
+
const DEFAULT_ADVERT_TTL_MS = 5 * 60 * 1000;
|
|
14
|
+
const DEFAULT_HEARTBEAT_MS = 20_000;
|
|
15
|
+
/**
|
|
16
|
+
* Server-side HTTP transport. Sans-I/O — the caller mounts
|
|
17
|
+
* {@link handleRequest} and {@link handleSse} under their HTTP framework of
|
|
18
|
+
* choice; the transport owns only in-memory state (actors, inboxes, advert
|
|
19
|
+
* cache, replay / rate-limit policies).
|
|
20
|
+
*
|
|
21
|
+
* Implements the generic {@link Transport} interface so the aggregation
|
|
22
|
+
* runners can drive it exactly the same way they drive {@link NostrTransport}
|
|
23
|
+
* or {@link HttpClientTransport}.
|
|
24
|
+
*/
|
|
25
|
+
export class HttpServerTransport {
|
|
26
|
+
name = 'http';
|
|
27
|
+
#logger;
|
|
28
|
+
#cors;
|
|
29
|
+
#clockSkewSec;
|
|
30
|
+
#inboxBufferSize;
|
|
31
|
+
#advertTtlMs;
|
|
32
|
+
#heartbeatMs;
|
|
33
|
+
#rateLimiter;
|
|
34
|
+
#nonceCache;
|
|
35
|
+
#now;
|
|
36
|
+
#actors = new Map();
|
|
37
|
+
#peers = new Map();
|
|
38
|
+
#inboxes = new Map();
|
|
39
|
+
#broadcastSubscribers = new Set();
|
|
40
|
+
#currentAdvert;
|
|
41
|
+
#advertSeq = 0;
|
|
42
|
+
constructor(config = {}) {
|
|
43
|
+
this.#logger = config.logger ?? CONSOLE_LOGGER;
|
|
44
|
+
this.#cors = config.cors ?? { mode: 'permissive' };
|
|
45
|
+
this.#clockSkewSec = config.clockSkewSec ?? DEFAULT_CLOCK_SKEW_SEC;
|
|
46
|
+
this.#inboxBufferSize = config.inboxBufferSize ?? 100;
|
|
47
|
+
this.#advertTtlMs = config.advertTtlMs ?? DEFAULT_ADVERT_TTL_MS;
|
|
48
|
+
this.#heartbeatMs = config.heartbeatIntervalMs ?? DEFAULT_HEARTBEAT_MS;
|
|
49
|
+
this.#rateLimiter = config.rateLimiter ?? new RateLimiter();
|
|
50
|
+
this.#nonceCache = config.nonceCache ?? new NonceCache();
|
|
51
|
+
this.#now = config.now ?? (() => Date.now());
|
|
52
|
+
}
|
|
53
|
+
// ----------------------------------------------------------------
|
|
54
|
+
// Transport interface
|
|
55
|
+
// ----------------------------------------------------------------
|
|
56
|
+
start() {
|
|
57
|
+
// No-op: server-side transport has no persistent outbound connections.
|
|
58
|
+
// SSE subscribers connect on demand via handleSse().
|
|
59
|
+
}
|
|
60
|
+
/**
|
|
61
|
+
* Detach the transport: close every open SSE subscription, clear the advert
|
|
62
|
+
* cache, and drop all actor / peer / inbox state. Intended for shutdown and
|
|
63
|
+
* for test teardown.
|
|
64
|
+
*/
|
|
65
|
+
stop() {
|
|
66
|
+
for (const sub of this.#broadcastSubscribers)
|
|
67
|
+
this.#closeBroadcastSubscriber(sub);
|
|
68
|
+
this.#broadcastSubscribers.clear();
|
|
69
|
+
for (const inbox of this.#inboxes.values()) {
|
|
70
|
+
for (const sub of inbox.subscribers)
|
|
71
|
+
this.#closeInboxSubscriber(sub);
|
|
72
|
+
inbox.subscribers.clear();
|
|
73
|
+
}
|
|
74
|
+
this.#inboxes.clear();
|
|
75
|
+
this.#currentAdvert = undefined;
|
|
76
|
+
}
|
|
77
|
+
registerActor(did, keys) {
|
|
78
|
+
this.#actors.set(did, { keys, handlers: new Map() });
|
|
79
|
+
}
|
|
80
|
+
unregisterActor(did) {
|
|
81
|
+
this.#actors.delete(did);
|
|
82
|
+
this.#peers.delete(did);
|
|
83
|
+
}
|
|
84
|
+
getActorPk(did) {
|
|
85
|
+
return this.#actors.get(did)?.keys.publicKey.compressed;
|
|
86
|
+
}
|
|
87
|
+
registerPeer(did, communicationPk) {
|
|
88
|
+
try {
|
|
89
|
+
new CompressedSecp256k1PublicKey(communicationPk);
|
|
90
|
+
}
|
|
91
|
+
catch {
|
|
92
|
+
throw new HttpTransportError(`Invalid peer public key for ${did}`, 'INVALID_PEER_KEY', { did, keyLength: communicationPk.length });
|
|
93
|
+
}
|
|
94
|
+
this.#peers.set(did, communicationPk);
|
|
95
|
+
}
|
|
96
|
+
getPeerPk(did) {
|
|
97
|
+
return this.#peers.get(did);
|
|
98
|
+
}
|
|
99
|
+
registerMessageHandler(actorDid, messageType, handler) {
|
|
100
|
+
const actor = this.#actors.get(actorDid);
|
|
101
|
+
if (!actor) {
|
|
102
|
+
throw new HttpTransportError(`Cannot register handler: actor ${actorDid} not registered`, 'UNKNOWN_ACTOR', { did: actorDid });
|
|
103
|
+
}
|
|
104
|
+
actor.handlers.set(messageType, handler);
|
|
105
|
+
}
|
|
106
|
+
unregisterMessageHandler(actorDid, messageType) {
|
|
107
|
+
this.#actors.get(actorDid)?.handlers.delete(messageType);
|
|
108
|
+
}
|
|
109
|
+
async sendMessage(message, sender, recipient) {
|
|
110
|
+
if (!recipient) {
|
|
111
|
+
throw new HttpTransportError('HttpServerTransport.sendMessage requires a recipient. Use publishRepeating for broadcasts.', 'MISSING_RECIPIENT', { messageType: message.type });
|
|
112
|
+
}
|
|
113
|
+
const actor = this.#actors.get(sender);
|
|
114
|
+
if (!actor) {
|
|
115
|
+
throw new HttpTransportError(`Unknown sender: ${sender}`, 'UNKNOWN_SENDER', { did: sender });
|
|
116
|
+
}
|
|
117
|
+
const envelope = signEnvelope(message, { did: sender, keys: actor.keys }, { to: recipient });
|
|
118
|
+
const dataJson = JSON.stringify(envelope);
|
|
119
|
+
const inbox = this.#getOrCreateInbox(recipient);
|
|
120
|
+
const stored = inbox.buffer.append(SSE_EVENT.MESSAGE, dataJson);
|
|
121
|
+
for (const sub of inbox.subscribers) {
|
|
122
|
+
this.#safeWrite(sub.stream, stored.event, stored.data, stored.id);
|
|
123
|
+
}
|
|
124
|
+
}
|
|
125
|
+
publishRepeating(message, sender, _intervalMs, _recipient) {
|
|
126
|
+
const actor = this.#actors.get(sender);
|
|
127
|
+
if (!actor) {
|
|
128
|
+
throw new HttpTransportError(`Unknown sender: ${sender}`, 'UNKNOWN_SENDER', { did: sender });
|
|
129
|
+
}
|
|
130
|
+
const envelope = signEnvelope(message, { did: sender, keys: actor.keys });
|
|
131
|
+
const dataJson = JSON.stringify(envelope);
|
|
132
|
+
const id = String(++this.#advertSeq);
|
|
133
|
+
const expiresAtMs = this.#now() + this.#advertTtlMs;
|
|
134
|
+
this.#currentAdvert = { dataJson, id, expiresAtMs };
|
|
135
|
+
for (const sub of this.#broadcastSubscribers) {
|
|
136
|
+
this.#safeWrite(sub.stream, SSE_EVENT.ADVERT, dataJson, id);
|
|
137
|
+
}
|
|
138
|
+
return () => {
|
|
139
|
+
if (this.#currentAdvert?.id === id)
|
|
140
|
+
this.#currentAdvert = undefined;
|
|
141
|
+
};
|
|
142
|
+
}
|
|
143
|
+
// ----------------------------------------------------------------
|
|
144
|
+
// Sans-I/O HTTP surface
|
|
145
|
+
// ----------------------------------------------------------------
|
|
146
|
+
/**
|
|
147
|
+
* Handle a POST / GET request (non-SSE). The caller dispatches SSE paths to
|
|
148
|
+
* {@link handleSse} instead. Returns a fully formed response; the caller's
|
|
149
|
+
* adapter turns it into an HTTP write.
|
|
150
|
+
*/
|
|
151
|
+
async handleRequest(req) {
|
|
152
|
+
const method = req.method.toUpperCase();
|
|
153
|
+
if (method === 'OPTIONS')
|
|
154
|
+
return this.#respond(204, '', req);
|
|
155
|
+
const path = extractPath(req.url);
|
|
156
|
+
if (method === 'GET' && path === HTTP_ROUTE.WELL_KNOWN) {
|
|
157
|
+
return this.#respondJson(200, this.#wellKnownMetadata(), req);
|
|
158
|
+
}
|
|
159
|
+
if (method === 'POST' && path === HTTP_ROUTE.MESSAGES) {
|
|
160
|
+
return await this.#handleMessagesPost(req);
|
|
161
|
+
}
|
|
162
|
+
if (method === 'POST' && path === HTTP_ROUTE.ADVERTS) {
|
|
163
|
+
return await this.#handleAdvertsPost(req);
|
|
164
|
+
}
|
|
165
|
+
return this.#respondJson(404, { error: 'not_found' }, req);
|
|
166
|
+
}
|
|
167
|
+
/**
|
|
168
|
+
* Open an SSE stream for a GET request. The caller is responsible for
|
|
169
|
+
* flushing writes and propagating the `onClose` callback when the HTTP
|
|
170
|
+
* connection ends.
|
|
171
|
+
*/
|
|
172
|
+
handleSse(req, stream) {
|
|
173
|
+
if (req.method.toUpperCase() !== 'GET') {
|
|
174
|
+
stream.close();
|
|
175
|
+
return;
|
|
176
|
+
}
|
|
177
|
+
const path = extractPath(req.url);
|
|
178
|
+
if (path === HTTP_ROUTE.ADVERTS) {
|
|
179
|
+
this.#openBroadcastSubscription(stream);
|
|
180
|
+
return;
|
|
181
|
+
}
|
|
182
|
+
const inboxMatch = matchInboxPath(path);
|
|
183
|
+
if (inboxMatch) {
|
|
184
|
+
this.#openInboxSubscription(req, stream, inboxMatch.did, path);
|
|
185
|
+
return;
|
|
186
|
+
}
|
|
187
|
+
stream.close();
|
|
188
|
+
}
|
|
189
|
+
// ----------------------------------------------------------------
|
|
190
|
+
// Request handlers
|
|
191
|
+
// ----------------------------------------------------------------
|
|
192
|
+
async #handleMessagesPost(req) {
|
|
193
|
+
const envelope = parseJsonBody(req.body);
|
|
194
|
+
if (!envelope)
|
|
195
|
+
return this.#respondJson(400, { error: 'invalid_json' }, req);
|
|
196
|
+
const senderPk = this.#resolveSenderPk(envelope.from);
|
|
197
|
+
if (!senderPk) {
|
|
198
|
+
return this.#respondJson(401, { error: 'unknown_sender' }, req);
|
|
199
|
+
}
|
|
200
|
+
try {
|
|
201
|
+
verifyEnvelope(envelope, senderPk, { clockSkewSec: this.#clockSkewSec });
|
|
202
|
+
}
|
|
203
|
+
catch (err) {
|
|
204
|
+
this.#logger.debug('POST /v1/messages: envelope verification failed:', err);
|
|
205
|
+
return this.#respondJson(401, { error: 'invalid_envelope' }, req);
|
|
206
|
+
}
|
|
207
|
+
if (!this.#nonceCache.store(envelope.from, envelope.nonce, envelope.timestamp)) {
|
|
208
|
+
return this.#respondJson(409, { error: 'replay' }, req);
|
|
209
|
+
}
|
|
210
|
+
if (!this.#rateLimiter.consume(envelope.from, this.#now())) {
|
|
211
|
+
return this.#respondJson(429, { error: 'rate_limited' }, req);
|
|
212
|
+
}
|
|
213
|
+
if (!envelope.to) {
|
|
214
|
+
return this.#respondJson(400, { error: 'missing_recipient' }, req);
|
|
215
|
+
}
|
|
216
|
+
const actor = this.#actors.get(envelope.to);
|
|
217
|
+
if (!actor) {
|
|
218
|
+
return this.#respondJson(404, { error: 'unknown_recipient' }, req);
|
|
219
|
+
}
|
|
220
|
+
const revived = reviveFromWire(envelope.message);
|
|
221
|
+
const flat = flattenMessage(revived);
|
|
222
|
+
const messageType = typeof flat.type === 'string' ? flat.type : undefined;
|
|
223
|
+
if (!messageType)
|
|
224
|
+
return this.#respondJson(400, { error: 'missing_message_type' }, req);
|
|
225
|
+
const handler = actor.handlers.get(messageType);
|
|
226
|
+
if (handler) {
|
|
227
|
+
try {
|
|
228
|
+
await handler(flat);
|
|
229
|
+
}
|
|
230
|
+
catch (err) {
|
|
231
|
+
this.#logger.debug(`Handler threw for ${messageType}:`, err);
|
|
232
|
+
}
|
|
233
|
+
}
|
|
234
|
+
return this.#respondJson(202, { ok: true }, req);
|
|
235
|
+
}
|
|
236
|
+
async #handleAdvertsPost(req) {
|
|
237
|
+
const envelope = parseJsonBody(req.body);
|
|
238
|
+
if (!envelope)
|
|
239
|
+
return this.#respondJson(400, { error: 'invalid_json' }, req);
|
|
240
|
+
const senderPk = this.#resolveSenderPk(envelope.from);
|
|
241
|
+
if (!senderPk)
|
|
242
|
+
return this.#respondJson(401, { error: 'unknown_sender' }, req);
|
|
243
|
+
try {
|
|
244
|
+
verifyEnvelope(envelope, senderPk, { clockSkewSec: this.#clockSkewSec });
|
|
245
|
+
}
|
|
246
|
+
catch {
|
|
247
|
+
return this.#respondJson(401, { error: 'invalid_envelope' }, req);
|
|
248
|
+
}
|
|
249
|
+
if (!this.#nonceCache.store(envelope.from, envelope.nonce, envelope.timestamp)) {
|
|
250
|
+
return this.#respondJson(409, { error: 'replay' }, req);
|
|
251
|
+
}
|
|
252
|
+
if (!this.#rateLimiter.consume(envelope.from, this.#now())) {
|
|
253
|
+
return this.#respondJson(429, { error: 'rate_limited' }, req);
|
|
254
|
+
}
|
|
255
|
+
// Only registered actors can publish adverts on this server.
|
|
256
|
+
if (!this.#actors.has(envelope.from)) {
|
|
257
|
+
return this.#respondJson(403, { error: 'not_an_actor' }, req);
|
|
258
|
+
}
|
|
259
|
+
const id = String(++this.#advertSeq);
|
|
260
|
+
this.#currentAdvert = {
|
|
261
|
+
dataJson: JSON.stringify(envelope),
|
|
262
|
+
id,
|
|
263
|
+
expiresAtMs: this.#now() + this.#advertTtlMs,
|
|
264
|
+
};
|
|
265
|
+
for (const sub of this.#broadcastSubscribers) {
|
|
266
|
+
this.#safeWrite(sub.stream, SSE_EVENT.ADVERT, this.#currentAdvert.dataJson, id);
|
|
267
|
+
}
|
|
268
|
+
return this.#respondJson(202, { ok: true }, req);
|
|
269
|
+
}
|
|
270
|
+
#openBroadcastSubscription(stream) {
|
|
271
|
+
const sub = { stream };
|
|
272
|
+
this.#broadcastSubscribers.add(sub);
|
|
273
|
+
stream.onClose(() => {
|
|
274
|
+
this.#closeBroadcastSubscriber(sub);
|
|
275
|
+
this.#broadcastSubscribers.delete(sub);
|
|
276
|
+
});
|
|
277
|
+
// Replay current advert if still within TTL.
|
|
278
|
+
if (this.#currentAdvert && this.#currentAdvert.expiresAtMs > this.#now()) {
|
|
279
|
+
this.#safeWrite(stream, SSE_EVENT.ADVERT, this.#currentAdvert.dataJson, this.#currentAdvert.id);
|
|
280
|
+
}
|
|
281
|
+
if (this.#heartbeatMs > 0) {
|
|
282
|
+
sub.heartbeatTimer = setInterval(() => {
|
|
283
|
+
try {
|
|
284
|
+
stream.writeComment('hb');
|
|
285
|
+
}
|
|
286
|
+
catch { /* caller-owned failure */ }
|
|
287
|
+
}, this.#heartbeatMs);
|
|
288
|
+
}
|
|
289
|
+
}
|
|
290
|
+
#openInboxSubscription(req, stream, did, path) {
|
|
291
|
+
const auth = req.headers.authorization;
|
|
292
|
+
if (!auth) {
|
|
293
|
+
this.#logger.debug(`Inbox subscribe: missing authorization header for ${did}`);
|
|
294
|
+
stream.close();
|
|
295
|
+
return;
|
|
296
|
+
}
|
|
297
|
+
const senderPk = this.#resolveSenderPk(did);
|
|
298
|
+
if (!senderPk) {
|
|
299
|
+
stream.close();
|
|
300
|
+
return;
|
|
301
|
+
}
|
|
302
|
+
let parsedTs = 0;
|
|
303
|
+
let parsedNonce = '';
|
|
304
|
+
try {
|
|
305
|
+
const parsed = verifyRequestAuth(auth, path, senderPk, {
|
|
306
|
+
clockSkewSec: this.#clockSkewSec,
|
|
307
|
+
now: () => this.#now(),
|
|
308
|
+
});
|
|
309
|
+
if (parsed.did !== did) {
|
|
310
|
+
stream.close();
|
|
311
|
+
return;
|
|
312
|
+
}
|
|
313
|
+
parsedTs = parsed.ts;
|
|
314
|
+
parsedNonce = parsed.nonce;
|
|
315
|
+
}
|
|
316
|
+
catch (err) {
|
|
317
|
+
this.#logger.debug(`Inbox subscribe: auth verification failed for ${did}:`, err);
|
|
318
|
+
stream.close();
|
|
319
|
+
return;
|
|
320
|
+
}
|
|
321
|
+
if (!this.#nonceCache.store(did, parsedNonce, parsedTs)) {
|
|
322
|
+
stream.close();
|
|
323
|
+
return;
|
|
324
|
+
}
|
|
325
|
+
if (!this.#rateLimiter.consume(did, this.#now())) {
|
|
326
|
+
stream.close();
|
|
327
|
+
return;
|
|
328
|
+
}
|
|
329
|
+
const inbox = this.#getOrCreateInbox(did);
|
|
330
|
+
const sub = { stream };
|
|
331
|
+
inbox.subscribers.add(sub);
|
|
332
|
+
stream.onClose(() => {
|
|
333
|
+
this.#closeInboxSubscriber(sub);
|
|
334
|
+
inbox.subscribers.delete(sub);
|
|
335
|
+
});
|
|
336
|
+
// Replay buffered events since the client's Last-Event-ID, if any.
|
|
337
|
+
const lastEventId = req.headers['last-event-id'];
|
|
338
|
+
for (const stored of inbox.buffer.since(lastEventId)) {
|
|
339
|
+
this.#safeWrite(stream, stored.event, stored.data, stored.id);
|
|
340
|
+
}
|
|
341
|
+
if (this.#heartbeatMs > 0) {
|
|
342
|
+
sub.heartbeatTimer = setInterval(() => {
|
|
343
|
+
try {
|
|
344
|
+
stream.writeComment('hb');
|
|
345
|
+
}
|
|
346
|
+
catch { /* caller-owned failure */ }
|
|
347
|
+
}, this.#heartbeatMs);
|
|
348
|
+
}
|
|
349
|
+
}
|
|
350
|
+
// ----------------------------------------------------------------
|
|
351
|
+
// Internal helpers
|
|
352
|
+
// ----------------------------------------------------------------
|
|
353
|
+
#getOrCreateInbox(did) {
|
|
354
|
+
let inbox = this.#inboxes.get(did);
|
|
355
|
+
if (!inbox) {
|
|
356
|
+
inbox = { buffer: new InboxBuffer(this.#inboxBufferSize), subscribers: new Set() };
|
|
357
|
+
this.#inboxes.set(did, inbox);
|
|
358
|
+
}
|
|
359
|
+
return inbox;
|
|
360
|
+
}
|
|
361
|
+
#resolveSenderPk(did) {
|
|
362
|
+
const peerBytes = this.#peers.get(did);
|
|
363
|
+
if (peerBytes) {
|
|
364
|
+
try {
|
|
365
|
+
return new CompressedSecp256k1PublicKey(peerBytes);
|
|
366
|
+
}
|
|
367
|
+
catch { /* fall through */ }
|
|
368
|
+
}
|
|
369
|
+
try {
|
|
370
|
+
const components = Identifier.decode(did);
|
|
371
|
+
if (components.idType === 'KEY') {
|
|
372
|
+
return new CompressedSecp256k1PublicKey(components.genesisBytes);
|
|
373
|
+
}
|
|
374
|
+
}
|
|
375
|
+
catch { /* not decodable */ }
|
|
376
|
+
return undefined;
|
|
377
|
+
}
|
|
378
|
+
#safeWrite(stream, event, data, id) {
|
|
379
|
+
try {
|
|
380
|
+
stream.writeEvent(event, data, id);
|
|
381
|
+
}
|
|
382
|
+
catch (err) {
|
|
383
|
+
this.#logger.debug('SSE writeEvent failed:', err);
|
|
384
|
+
}
|
|
385
|
+
}
|
|
386
|
+
#closeBroadcastSubscriber(sub) {
|
|
387
|
+
if (sub.heartbeatTimer)
|
|
388
|
+
clearInterval(sub.heartbeatTimer);
|
|
389
|
+
try {
|
|
390
|
+
sub.stream.close();
|
|
391
|
+
}
|
|
392
|
+
catch { /* already closed */ }
|
|
393
|
+
}
|
|
394
|
+
#closeInboxSubscriber(sub) {
|
|
395
|
+
if (sub.heartbeatTimer)
|
|
396
|
+
clearInterval(sub.heartbeatTimer);
|
|
397
|
+
try {
|
|
398
|
+
sub.stream.close();
|
|
399
|
+
}
|
|
400
|
+
catch { /* already closed */ }
|
|
401
|
+
}
|
|
402
|
+
#respondJson(status, body, req) {
|
|
403
|
+
return {
|
|
404
|
+
status,
|
|
405
|
+
headers: { 'content-type': 'application/json', ...this.#corsHeaders(req) },
|
|
406
|
+
body: JSON.stringify(body),
|
|
407
|
+
};
|
|
408
|
+
}
|
|
409
|
+
#respond(status, body, req) {
|
|
410
|
+
return { status, headers: this.#corsHeaders(req), body };
|
|
411
|
+
}
|
|
412
|
+
#corsHeaders(req) {
|
|
413
|
+
const origin = req.headers.origin;
|
|
414
|
+
if (!origin)
|
|
415
|
+
return {};
|
|
416
|
+
const common = {
|
|
417
|
+
'access-control-allow-methods': 'GET, POST, OPTIONS',
|
|
418
|
+
'access-control-allow-headers': 'authorization, content-type, last-event-id',
|
|
419
|
+
'access-control-max-age': '86400',
|
|
420
|
+
};
|
|
421
|
+
switch (this.#cors.mode) {
|
|
422
|
+
case 'permissive':
|
|
423
|
+
return { 'access-control-allow-origin': '*', ...common };
|
|
424
|
+
case 'allowlist':
|
|
425
|
+
if (this.#cors.origins.includes(origin)) {
|
|
426
|
+
return { 'access-control-allow-origin': origin, vary: 'origin', ...common };
|
|
427
|
+
}
|
|
428
|
+
return {};
|
|
429
|
+
case 'same-origin':
|
|
430
|
+
return {};
|
|
431
|
+
}
|
|
432
|
+
}
|
|
433
|
+
#wellKnownMetadata() {
|
|
434
|
+
return {
|
|
435
|
+
envelopeVersion: HTTP_ENVELOPE_VERSION,
|
|
436
|
+
heartbeatIntervalMs: this.#heartbeatMs,
|
|
437
|
+
inboxBufferSize: this.#inboxBufferSize,
|
|
438
|
+
advertTtlMs: this.#advertTtlMs,
|
|
439
|
+
};
|
|
440
|
+
}
|
|
441
|
+
}
|
|
442
|
+
// ----------------------------------------------------------------
|
|
443
|
+
// Module-local pure helpers
|
|
444
|
+
// ----------------------------------------------------------------
|
|
445
|
+
function extractPath(reqUrl) {
|
|
446
|
+
if (reqUrl.startsWith('http://') || reqUrl.startsWith('https://')) {
|
|
447
|
+
return new URL(reqUrl).pathname;
|
|
448
|
+
}
|
|
449
|
+
const q = reqUrl.indexOf('?');
|
|
450
|
+
return q === -1 ? reqUrl : reqUrl.slice(0, q);
|
|
451
|
+
}
|
|
452
|
+
function matchInboxPath(path) {
|
|
453
|
+
if (!path.startsWith(INBOX_PATH_PREFIX) || !path.endsWith(INBOX_PATH_SUFFIX))
|
|
454
|
+
return undefined;
|
|
455
|
+
const encodedDid = path.slice(INBOX_PATH_PREFIX.length, path.length - INBOX_PATH_SUFFIX.length);
|
|
456
|
+
if (!encodedDid)
|
|
457
|
+
return undefined;
|
|
458
|
+
try {
|
|
459
|
+
return { did: decodeURIComponent(encodedDid) };
|
|
460
|
+
}
|
|
461
|
+
catch {
|
|
462
|
+
return undefined;
|
|
463
|
+
}
|
|
464
|
+
}
|
|
465
|
+
function parseJsonBody(body) {
|
|
466
|
+
if (body === undefined || body === '')
|
|
467
|
+
return undefined;
|
|
468
|
+
try {
|
|
469
|
+
return JSON.parse(body);
|
|
470
|
+
}
|
|
471
|
+
catch {
|
|
472
|
+
return undefined;
|
|
473
|
+
}
|
|
474
|
+
}
|
|
475
|
+
function flattenMessage(msg) {
|
|
476
|
+
if (msg.body && typeof msg.body === 'object') {
|
|
477
|
+
return { ...msg, ...msg.body };
|
|
478
|
+
}
|
|
479
|
+
return msg;
|
|
480
|
+
}
|
|
481
|
+
//# sourceMappingURL=server.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"server.js","sourceRoot":"","sources":["../../../../../../src/core/aggregation/transport/http/server.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,110 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Parse an SSE stream into an async iterable of {@link SseEvent} records.
|
|
3
|
+
*
|
|
4
|
+
* The parser follows the HTML Living Standard ({@link https://html.spec.whatwg.org/multipage/server-sent-events.html})
|
|
5
|
+
* closely enough for our needs: LF and CRLF line terminators, multi-line
|
|
6
|
+
* `data` fields, `event` / `id` / `retry` fields, and `:`-prefixed comments.
|
|
7
|
+
* CR-only line terminators are not supported (every mainstream SSE
|
|
8
|
+
* implementation emits LF or CRLF).
|
|
9
|
+
*
|
|
10
|
+
* Pure, runtime-agnostic — works anywhere `ReadableStream<Uint8Array>` and
|
|
11
|
+
* `TextDecoder` exist (browsers and Node 22+).
|
|
12
|
+
*
|
|
13
|
+
* The caller owns stream lifecycle: cancellation should be effected via an
|
|
14
|
+
* `AbortController` on the producing `fetch`, which propagates as a read
|
|
15
|
+
* error and cleanly unwinds this generator's `finally`.
|
|
16
|
+
*/
|
|
17
|
+
export async function* parseSseStream(readable) {
|
|
18
|
+
const decoder = new TextDecoder('utf-8');
|
|
19
|
+
const reader = readable.getReader();
|
|
20
|
+
let buffer = '';
|
|
21
|
+
let pending = {};
|
|
22
|
+
const dispatchPending = () => {
|
|
23
|
+
if (pending.data === undefined) {
|
|
24
|
+
pending = {};
|
|
25
|
+
return null;
|
|
26
|
+
}
|
|
27
|
+
const ev = { data: pending.data };
|
|
28
|
+
if (pending.event !== undefined)
|
|
29
|
+
ev.event = pending.event;
|
|
30
|
+
if (pending.id !== undefined)
|
|
31
|
+
ev.id = pending.id;
|
|
32
|
+
if (pending.retry !== undefined)
|
|
33
|
+
ev.retry = pending.retry;
|
|
34
|
+
pending = {};
|
|
35
|
+
return ev;
|
|
36
|
+
};
|
|
37
|
+
const processLine = (line) => {
|
|
38
|
+
if (line.startsWith(':'))
|
|
39
|
+
return; // comment
|
|
40
|
+
const colon = line.indexOf(':');
|
|
41
|
+
const field = colon === -1 ? line : line.slice(0, colon);
|
|
42
|
+
let value = colon === -1 ? '' : line.slice(colon + 1);
|
|
43
|
+
if (value.startsWith(' '))
|
|
44
|
+
value = value.slice(1);
|
|
45
|
+
switch (field) {
|
|
46
|
+
case 'data':
|
|
47
|
+
pending.data = pending.data === undefined ? value : `${pending.data}\n${value}`;
|
|
48
|
+
break;
|
|
49
|
+
case 'event':
|
|
50
|
+
pending.event = value;
|
|
51
|
+
break;
|
|
52
|
+
case 'id':
|
|
53
|
+
// Per spec: ignore ids containing NUL.
|
|
54
|
+
if (!value.includes('\0'))
|
|
55
|
+
pending.id = value;
|
|
56
|
+
break;
|
|
57
|
+
case 'retry': {
|
|
58
|
+
const n = Number(value);
|
|
59
|
+
if (Number.isInteger(n) && n >= 0)
|
|
60
|
+
pending.retry = n;
|
|
61
|
+
break;
|
|
62
|
+
}
|
|
63
|
+
// Other fields (including unknown names) are ignored per the spec.
|
|
64
|
+
}
|
|
65
|
+
};
|
|
66
|
+
try {
|
|
67
|
+
for (;;) {
|
|
68
|
+
const { value, done } = await reader.read();
|
|
69
|
+
if (done) {
|
|
70
|
+
// Flush any bytes the decoder is still holding.
|
|
71
|
+
buffer += decoder.decode();
|
|
72
|
+
if (buffer.length > 0) {
|
|
73
|
+
const line = buffer.endsWith('\r') ? buffer.slice(0, -1) : buffer;
|
|
74
|
+
if (line.length > 0)
|
|
75
|
+
processLine(line);
|
|
76
|
+
buffer = '';
|
|
77
|
+
}
|
|
78
|
+
const tail = dispatchPending();
|
|
79
|
+
if (tail)
|
|
80
|
+
yield tail;
|
|
81
|
+
return;
|
|
82
|
+
}
|
|
83
|
+
buffer += decoder.decode(value, { stream: true });
|
|
84
|
+
// Drain as many complete lines as are available.
|
|
85
|
+
let lineEnd = buffer.indexOf('\n');
|
|
86
|
+
while (lineEnd !== -1) {
|
|
87
|
+
let line = buffer.slice(0, lineEnd);
|
|
88
|
+
if (line.endsWith('\r'))
|
|
89
|
+
line = line.slice(0, -1);
|
|
90
|
+
buffer = buffer.slice(lineEnd + 1);
|
|
91
|
+
if (line.length === 0) {
|
|
92
|
+
const ev = dispatchPending();
|
|
93
|
+
if (ev)
|
|
94
|
+
yield ev;
|
|
95
|
+
}
|
|
96
|
+
else {
|
|
97
|
+
processLine(line);
|
|
98
|
+
}
|
|
99
|
+
lineEnd = buffer.indexOf('\n');
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
finally {
|
|
104
|
+
try {
|
|
105
|
+
reader.releaseLock();
|
|
106
|
+
}
|
|
107
|
+
catch { /* already released */ }
|
|
108
|
+
}
|
|
109
|
+
}
|
|
110
|
+
//# sourceMappingURL=sse-stream.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"sse-stream.js","sourceRoot":"","sources":["../../../../../../src/core/aggregation/transport/http/sse-stream.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Format an SSE event frame. Pairs with {@link parseSseStream}.
|
|
3
|
+
*
|
|
4
|
+
* Multi-line `data` is split across multiple `data:` lines per the SSE spec —
|
|
5
|
+
* each embedded `\n` becomes its own line, and the parser rejoins them.
|
|
6
|
+
*
|
|
7
|
+
* The returned string includes a trailing blank line (the dispatch marker).
|
|
8
|
+
*/
|
|
9
|
+
export function formatSseEvent(event, data, id) {
|
|
10
|
+
const lines = [];
|
|
11
|
+
if (id !== undefined)
|
|
12
|
+
lines.push(`id: ${id}`);
|
|
13
|
+
lines.push(`event: ${event}`);
|
|
14
|
+
for (const part of data.split('\n'))
|
|
15
|
+
lines.push(`data: ${part}`);
|
|
16
|
+
lines.push('');
|
|
17
|
+
lines.push('');
|
|
18
|
+
return lines.join('\n');
|
|
19
|
+
}
|
|
20
|
+
/** SSE comment frame (server keepalive). Lines starting with `:` are ignored by compliant parsers. */
|
|
21
|
+
export function formatSseComment(comment) {
|
|
22
|
+
const safe = comment.replace(/\n/g, ' ');
|
|
23
|
+
return `: ${safe}\n\n`;
|
|
24
|
+
}
|
|
25
|
+
//# sourceMappingURL=sse-writer.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"sse-writer.js","sourceRoot":"","sources":["../../../../../../src/core/aggregation/transport/http/sse-writer.ts"],"names":[],"mappings":"AAAA;;;;;;;GAOG;AACH,MAAM,UAAU,cAAc,CAAC,KAAa,EAAE,IAAY,EAAE,EAAW;IACrE,MAAM,KAAK,GAAa,EAAE,CAAC;IAC3B,IAAG,EAAE,KAAK,SAAS;QAAE,KAAK,CAAC,IAAI,CAAC,OAAO,EAAE,EAAE,CAAC,CAAC;IAC7C,KAAK,CAAC,IAAI,CAAC,UAAU,KAAK,EAAE,CAAC,CAAC;IAC9B,KAAI,MAAM,IAAI,IAAI,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC;QAAE,KAAK,CAAC,IAAI,CAAC,SAAS,IAAI,EAAE,CAAC,CAAC;IAChE,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;IACf,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;IACf,OAAO,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;AAC1B,CAAC;AAED,sGAAsG;AACtG,MAAM,UAAU,gBAAgB,CAAC,OAAe;IAC9C,MAAM,IAAI,GAAG,OAAO,CAAC,OAAO,CAAC,KAAK,EAAE,GAAG,CAAC,CAAC;IACzC,OAAO,KAAK,IAAI,MAAM,CAAC;AACzB,CAAC"}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../../../src/core/aggregation/transport/index.ts"],"names":[],"mappings":"AAAA,cAAc,gBAAgB,CAAC;AAC/B,cAAc,YAAY,CAAC;AAC3B,cAAc,cAAc,CAAC;AAC7B,cAAc,YAAY,CAAC;AAC3B,cAAc,cAAc,CAAC"}
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../../../src/core/aggregation/transport/index.ts"],"names":[],"mappings":"AAAA,cAAc,gBAAgB,CAAC;AAC/B,cAAc,YAAY,CAAC;AAC3B,cAAc,cAAc,CAAC;AAC7B,cAAc,YAAY,CAAC;AAC3B,cAAc,cAAc,CAAC;AAC7B,cAAc,iBAAiB,CAAC"}
|