@emmanuel-nike/ark-notify-js 0.1.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/LICENSE +21 -0
- package/README.md +295 -0
- package/dist/index.cjs +564 -0
- package/dist/index.cjs.map +1 -0
- package/dist/index.d.cts +16 -0
- package/dist/index.d.ts +16 -0
- package/dist/index.js +552 -0
- package/dist/index.js.map +1 -0
- package/dist/react/index.cjs +963 -0
- package/dist/react/index.cjs.map +1 -0
- package/dist/react/index.d.cts +109 -0
- package/dist/react/index.d.ts +109 -0
- package/dist/react/index.js +951 -0
- package/dist/react/index.js.map +1 -0
- package/dist/utils-Cw2SnD6p.d.cts +407 -0
- package/dist/utils-Cw2SnD6p.d.ts +407 -0
- package/package.json +120 -0
package/dist/index.cjs
ADDED
|
@@ -0,0 +1,564 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
// src/config.ts
|
|
4
|
+
var DEFAULT_BASE_URL = "https://ark-notify-933303906015.europe-north1.run.app";
|
|
5
|
+
var configuredBaseUrl;
|
|
6
|
+
function configureArkNotify(config) {
|
|
7
|
+
if (config.baseUrl !== void 0) {
|
|
8
|
+
configuredBaseUrl = config.baseUrl;
|
|
9
|
+
}
|
|
10
|
+
}
|
|
11
|
+
function resolveBaseUrl(baseUrl) {
|
|
12
|
+
return (baseUrl ?? configuredBaseUrl ?? DEFAULT_BASE_URL).replace(/\/$/, "");
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
// src/utils.ts
|
|
16
|
+
var ArkNotifyError = class extends Error {
|
|
17
|
+
constructor(status, body) {
|
|
18
|
+
super(body.message);
|
|
19
|
+
this.name = "ArkNotifyError";
|
|
20
|
+
this.status = status;
|
|
21
|
+
this.code = body.error;
|
|
22
|
+
this.retryAfterSec = body.retryAfterSec;
|
|
23
|
+
this.reason = body.reason;
|
|
24
|
+
}
|
|
25
|
+
};
|
|
26
|
+
function toWebSocketUrl(baseUrl, path) {
|
|
27
|
+
const url = new URL(path, baseUrl);
|
|
28
|
+
url.protocol = url.protocol === "https:" ? "wss:" : "ws:";
|
|
29
|
+
return url.toString();
|
|
30
|
+
}
|
|
31
|
+
function resolveValue(value) {
|
|
32
|
+
return typeof value === "function" ? value() : value;
|
|
33
|
+
}
|
|
34
|
+
function isPrivateChannel(channel) {
|
|
35
|
+
return channel.startsWith("private-");
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
// src/client.ts
|
|
39
|
+
var ArkNotifyClient = class {
|
|
40
|
+
constructor(config) {
|
|
41
|
+
this.baseUrl = resolveBaseUrl(config.baseUrl);
|
|
42
|
+
this.token = config.token;
|
|
43
|
+
this.fetchFn = config.fetch ?? globalThis.fetch.bind(globalThis);
|
|
44
|
+
}
|
|
45
|
+
setToken(token) {
|
|
46
|
+
this.token = token ?? void 0;
|
|
47
|
+
}
|
|
48
|
+
getAuthHeader() {
|
|
49
|
+
const token = resolveValue(this.token);
|
|
50
|
+
return token ? `Bearer ${token}` : void 0;
|
|
51
|
+
}
|
|
52
|
+
async request(path, options = {}) {
|
|
53
|
+
const headers = {
|
|
54
|
+
"Content-Type": "application/json",
|
|
55
|
+
...options.headers
|
|
56
|
+
};
|
|
57
|
+
const auth = this.getAuthHeader();
|
|
58
|
+
if (auth) {
|
|
59
|
+
headers.Authorization = auth;
|
|
60
|
+
}
|
|
61
|
+
if (options.credentials) {
|
|
62
|
+
headers["X-App-Key"] = options.credentials.appKey;
|
|
63
|
+
headers["X-App-Secret"] = options.credentials.secret;
|
|
64
|
+
}
|
|
65
|
+
const response = await this.fetchFn(`${this.baseUrl}${path}`, {
|
|
66
|
+
method: options.method ?? "GET",
|
|
67
|
+
headers,
|
|
68
|
+
body: options.body !== void 0 ? JSON.stringify(options.body) : void 0
|
|
69
|
+
});
|
|
70
|
+
if (!response.ok) {
|
|
71
|
+
let body;
|
|
72
|
+
try {
|
|
73
|
+
body = await response.json();
|
|
74
|
+
} catch {
|
|
75
|
+
body = { error: "request_failed", message: response.statusText };
|
|
76
|
+
}
|
|
77
|
+
throw new ArkNotifyError(response.status, body);
|
|
78
|
+
}
|
|
79
|
+
if (response.status === 204) {
|
|
80
|
+
return void 0;
|
|
81
|
+
}
|
|
82
|
+
return response.json();
|
|
83
|
+
}
|
|
84
|
+
// ── Health ──────────────────────────────────────────────────────────────
|
|
85
|
+
health() {
|
|
86
|
+
return this.request("/health");
|
|
87
|
+
}
|
|
88
|
+
// ── Platform auth ───────────────────────────────────────────────────────
|
|
89
|
+
login(input) {
|
|
90
|
+
return this.request("/api/v1/auth/login", { method: "POST", body: input });
|
|
91
|
+
}
|
|
92
|
+
me() {
|
|
93
|
+
return this.request("/api/v1/auth/me");
|
|
94
|
+
}
|
|
95
|
+
// ── Applications ────────────────────────────────────────────────────────
|
|
96
|
+
listApplications() {
|
|
97
|
+
return this.request("/api/v1/applications");
|
|
98
|
+
}
|
|
99
|
+
createApplication(input) {
|
|
100
|
+
return this.request("/api/v1/applications", { method: "POST", body: input });
|
|
101
|
+
}
|
|
102
|
+
getApplication(id) {
|
|
103
|
+
return this.request(`/api/v1/applications/${id}`);
|
|
104
|
+
}
|
|
105
|
+
updateApplication(id, input) {
|
|
106
|
+
return this.request(`/api/v1/applications/${id}`, { method: "PUT", body: input });
|
|
107
|
+
}
|
|
108
|
+
deleteApplication(id) {
|
|
109
|
+
return this.request(`/api/v1/applications/${id}`, { method: "DELETE" });
|
|
110
|
+
}
|
|
111
|
+
regenerateSecret(id) {
|
|
112
|
+
return this.request(`/api/v1/applications/${id}/regenerate-secret`, {
|
|
113
|
+
method: "POST"
|
|
114
|
+
});
|
|
115
|
+
}
|
|
116
|
+
// ── System admin ────────────────────────────────────────────────────────
|
|
117
|
+
adminChannels() {
|
|
118
|
+
return this.request("/api/v1/admin/channels");
|
|
119
|
+
}
|
|
120
|
+
// ── Data plane ──────────────────────────────────────────────────────────
|
|
121
|
+
publishEvent(appKey, credentials, input) {
|
|
122
|
+
return this.request(`/api/v1/apps/${appKey}/events`, {
|
|
123
|
+
method: "POST",
|
|
124
|
+
body: input,
|
|
125
|
+
credentials
|
|
126
|
+
});
|
|
127
|
+
}
|
|
128
|
+
authorizeChannel(appKey, credentials, input) {
|
|
129
|
+
return this.request(`/api/v1/apps/${appKey}/auth`, {
|
|
130
|
+
method: "POST",
|
|
131
|
+
body: input,
|
|
132
|
+
credentials
|
|
133
|
+
});
|
|
134
|
+
}
|
|
135
|
+
issueConnectionToken(appKey, credentials, input) {
|
|
136
|
+
return this.request(`/api/v1/apps/${appKey}/connection-token`, {
|
|
137
|
+
method: "POST",
|
|
138
|
+
body: input,
|
|
139
|
+
credentials
|
|
140
|
+
});
|
|
141
|
+
}
|
|
142
|
+
};
|
|
143
|
+
|
|
144
|
+
// src/connection-token.ts
|
|
145
|
+
async function fetchConnectionToken(options) {
|
|
146
|
+
const {
|
|
147
|
+
baseUrl,
|
|
148
|
+
appKey,
|
|
149
|
+
credentials,
|
|
150
|
+
fetch: fetchFn = globalThis.fetch.bind(globalThis),
|
|
151
|
+
client_id,
|
|
152
|
+
clientId,
|
|
153
|
+
user_data,
|
|
154
|
+
userData,
|
|
155
|
+
ttl,
|
|
156
|
+
capabilities,
|
|
157
|
+
serverAuthUrl
|
|
158
|
+
} = options;
|
|
159
|
+
const resolvedClientId = client_id ?? clientId;
|
|
160
|
+
if (!resolvedClientId) {
|
|
161
|
+
throw new Error("client_id is required to fetch a connection token");
|
|
162
|
+
}
|
|
163
|
+
const body = {
|
|
164
|
+
client_id: resolvedClientId,
|
|
165
|
+
user_data: user_data ?? userData,
|
|
166
|
+
ttl,
|
|
167
|
+
capabilities,
|
|
168
|
+
serverAuthUrl
|
|
169
|
+
};
|
|
170
|
+
const url = `${resolveBaseUrl(baseUrl)}/api/v1/apps/${appKey}/connection-token`;
|
|
171
|
+
const response = await fetchFn(url, {
|
|
172
|
+
method: "POST",
|
|
173
|
+
headers: {
|
|
174
|
+
"Content-Type": "application/json",
|
|
175
|
+
"X-App-Key": credentials.appKey,
|
|
176
|
+
"X-App-Secret": credentials.secret
|
|
177
|
+
},
|
|
178
|
+
body: JSON.stringify(body)
|
|
179
|
+
});
|
|
180
|
+
if (!response.ok) {
|
|
181
|
+
let errorBody;
|
|
182
|
+
try {
|
|
183
|
+
errorBody = await response.json();
|
|
184
|
+
} catch {
|
|
185
|
+
errorBody = { error: "request_failed", message: response.statusText };
|
|
186
|
+
}
|
|
187
|
+
throw new ArkNotifyError(response.status, errorBody);
|
|
188
|
+
}
|
|
189
|
+
return response.json();
|
|
190
|
+
}
|
|
191
|
+
|
|
192
|
+
// src/connection.ts
|
|
193
|
+
var ArkNotifyConnection = class {
|
|
194
|
+
constructor(config) {
|
|
195
|
+
this.ws = null;
|
|
196
|
+
this.state = "disconnected";
|
|
197
|
+
this.connectionId = null;
|
|
198
|
+
this.clientId = null;
|
|
199
|
+
this.authenticated = false;
|
|
200
|
+
this.subscribedChannels = /* @__PURE__ */ new Set();
|
|
201
|
+
this.pendingSubscriptions = /* @__PURE__ */ new Map();
|
|
202
|
+
this.listeners = /* @__PURE__ */ new Map();
|
|
203
|
+
this.reconnectAttempt = 0;
|
|
204
|
+
this.reconnectTimer = null;
|
|
205
|
+
this.intentionalClose = false;
|
|
206
|
+
this.connectPromise = null;
|
|
207
|
+
this.config = {
|
|
208
|
+
autoReconnect: true,
|
|
209
|
+
reconnectDelayMs: 1e3,
|
|
210
|
+
maxReconnectDelayMs: 3e4,
|
|
211
|
+
...config,
|
|
212
|
+
baseUrl: resolveBaseUrl(config.baseUrl)
|
|
213
|
+
};
|
|
214
|
+
this.WebSocketCtor = config.WebSocket ?? globalThis.WebSocket;
|
|
215
|
+
}
|
|
216
|
+
getConnectionState() {
|
|
217
|
+
return this.state;
|
|
218
|
+
}
|
|
219
|
+
getConnectionId() {
|
|
220
|
+
return this.connectionId;
|
|
221
|
+
}
|
|
222
|
+
getClientId() {
|
|
223
|
+
return this.clientId;
|
|
224
|
+
}
|
|
225
|
+
isAuthenticated() {
|
|
226
|
+
return this.authenticated;
|
|
227
|
+
}
|
|
228
|
+
getSubscribedChannels() {
|
|
229
|
+
return [...this.subscribedChannels];
|
|
230
|
+
}
|
|
231
|
+
on(event, handler) {
|
|
232
|
+
if (!this.listeners.has(event)) {
|
|
233
|
+
this.listeners.set(event, /* @__PURE__ */ new Set());
|
|
234
|
+
}
|
|
235
|
+
this.listeners.get(event).add(handler);
|
|
236
|
+
return () => this.off(event, handler);
|
|
237
|
+
}
|
|
238
|
+
off(event, handler) {
|
|
239
|
+
this.listeners.get(event)?.delete(handler);
|
|
240
|
+
}
|
|
241
|
+
emit(event, ...args) {
|
|
242
|
+
for (const handler of this.listeners.get(event) ?? []) {
|
|
243
|
+
handler(...args);
|
|
244
|
+
}
|
|
245
|
+
}
|
|
246
|
+
setState(state) {
|
|
247
|
+
if (this.state !== state) {
|
|
248
|
+
this.state = state;
|
|
249
|
+
this.emit("state", state);
|
|
250
|
+
}
|
|
251
|
+
}
|
|
252
|
+
connect() {
|
|
253
|
+
if (this.ws?.readyState === WebSocket.OPEN || this.ws?.readyState === WebSocket.CONNECTING) {
|
|
254
|
+
return Promise.resolve();
|
|
255
|
+
}
|
|
256
|
+
if (this.connectPromise) {
|
|
257
|
+
return this.connectPromise;
|
|
258
|
+
}
|
|
259
|
+
this.connectPromise = this.doConnect().finally(() => {
|
|
260
|
+
this.connectPromise = null;
|
|
261
|
+
});
|
|
262
|
+
return this.connectPromise;
|
|
263
|
+
}
|
|
264
|
+
async doConnect() {
|
|
265
|
+
this.intentionalClose = false;
|
|
266
|
+
this.clearReconnectTimer();
|
|
267
|
+
this.setState(this.reconnectAttempt > 0 ? "reconnecting" : "connecting");
|
|
268
|
+
const url = new URL(
|
|
269
|
+
toWebSocketUrl(this.config.baseUrl, `/app/${this.config.appKey}`)
|
|
270
|
+
);
|
|
271
|
+
let token = resolveValue(this.config.token);
|
|
272
|
+
if (!token && this.config.clientId && this.config.credentials) {
|
|
273
|
+
const result = await fetchConnectionToken({
|
|
274
|
+
baseUrl: this.config.baseUrl,
|
|
275
|
+
appKey: this.config.appKey,
|
|
276
|
+
credentials: this.config.credentials,
|
|
277
|
+
client_id: this.config.clientId,
|
|
278
|
+
user_data: this.config.user_data,
|
|
279
|
+
fetch: this.config.fetch
|
|
280
|
+
});
|
|
281
|
+
token = result.token;
|
|
282
|
+
}
|
|
283
|
+
if (token) {
|
|
284
|
+
url.searchParams.set("token", token);
|
|
285
|
+
} else if (this.config.clientId) {
|
|
286
|
+
url.searchParams.set("clientId", this.config.clientId);
|
|
287
|
+
}
|
|
288
|
+
this.ws = new this.WebSocketCtor(url.toString());
|
|
289
|
+
this.ws.onopen = () => {
|
|
290
|
+
this.reconnectAttempt = 0;
|
|
291
|
+
};
|
|
292
|
+
this.ws.onmessage = (event) => {
|
|
293
|
+
this.handleMessage(event.data);
|
|
294
|
+
};
|
|
295
|
+
this.ws.onerror = () => {
|
|
296
|
+
this.emit("error", { code: "websocket_error", message: "WebSocket error" });
|
|
297
|
+
};
|
|
298
|
+
this.ws.onclose = (event) => {
|
|
299
|
+
this.ws = null;
|
|
300
|
+
this.connectionId = null;
|
|
301
|
+
this.emit("close", { code: event.code, reason: event.reason });
|
|
302
|
+
if (!this.intentionalClose && this.config.autoReconnect) {
|
|
303
|
+
this.scheduleReconnect();
|
|
304
|
+
} else {
|
|
305
|
+
this.setState("failed");
|
|
306
|
+
}
|
|
307
|
+
};
|
|
308
|
+
}
|
|
309
|
+
disconnect() {
|
|
310
|
+
this.intentionalClose = true;
|
|
311
|
+
this.clearReconnectTimer();
|
|
312
|
+
this.subscribedChannels.clear();
|
|
313
|
+
this.pendingSubscriptions.clear();
|
|
314
|
+
if (this.ws) {
|
|
315
|
+
this.ws.close();
|
|
316
|
+
this.ws = null;
|
|
317
|
+
}
|
|
318
|
+
this.setState("disconnected");
|
|
319
|
+
}
|
|
320
|
+
scheduleReconnect() {
|
|
321
|
+
this.setState("reconnecting");
|
|
322
|
+
const delay = Math.min(
|
|
323
|
+
this.config.reconnectDelayMs * 2 ** this.reconnectAttempt,
|
|
324
|
+
this.config.maxReconnectDelayMs
|
|
325
|
+
);
|
|
326
|
+
this.reconnectAttempt++;
|
|
327
|
+
this.reconnectTimer = setTimeout(() => void this.connect(), delay);
|
|
328
|
+
}
|
|
329
|
+
clearReconnectTimer() {
|
|
330
|
+
if (this.reconnectTimer) {
|
|
331
|
+
clearTimeout(this.reconnectTimer);
|
|
332
|
+
this.reconnectTimer = null;
|
|
333
|
+
}
|
|
334
|
+
}
|
|
335
|
+
send(payload) {
|
|
336
|
+
if (this.ws?.readyState !== WebSocket.OPEN) {
|
|
337
|
+
throw new Error("WebSocket is not connected");
|
|
338
|
+
}
|
|
339
|
+
this.ws.send(JSON.stringify(payload));
|
|
340
|
+
}
|
|
341
|
+
handleMessage(raw) {
|
|
342
|
+
let message;
|
|
343
|
+
try {
|
|
344
|
+
message = JSON.parse(raw);
|
|
345
|
+
} catch {
|
|
346
|
+
this.emit("error", { code: "invalid_json", message: "Invalid JSON from server" });
|
|
347
|
+
return;
|
|
348
|
+
}
|
|
349
|
+
this.emit("message", message);
|
|
350
|
+
switch (message.type) {
|
|
351
|
+
case "connected": {
|
|
352
|
+
this.connectionId = message.connection_id;
|
|
353
|
+
this.clientId = message.client_id;
|
|
354
|
+
this.authenticated = message.authenticated;
|
|
355
|
+
this.setState("connected");
|
|
356
|
+
this.emit("connected", message);
|
|
357
|
+
this.resubscribeAll();
|
|
358
|
+
break;
|
|
359
|
+
}
|
|
360
|
+
case "event":
|
|
361
|
+
this.emit("event", message);
|
|
362
|
+
break;
|
|
363
|
+
case "presence":
|
|
364
|
+
this.emit("presence", message);
|
|
365
|
+
break;
|
|
366
|
+
case "subscribed":
|
|
367
|
+
this.subscribedChannels.add(message.channel);
|
|
368
|
+
break;
|
|
369
|
+
case "unsubscribed":
|
|
370
|
+
this.subscribedChannels.delete(message.channel);
|
|
371
|
+
break;
|
|
372
|
+
case "error":
|
|
373
|
+
this.emit("error", { code: message.code, message: message.message });
|
|
374
|
+
break;
|
|
375
|
+
case "ping":
|
|
376
|
+
this.send({ action: "ping" });
|
|
377
|
+
break;
|
|
378
|
+
}
|
|
379
|
+
}
|
|
380
|
+
resubscribeAll() {
|
|
381
|
+
for (const [channel, options] of this.pendingSubscriptions) {
|
|
382
|
+
void this.subscribe(channel, options);
|
|
383
|
+
}
|
|
384
|
+
}
|
|
385
|
+
async subscribe(channel, options = {}) {
|
|
386
|
+
this.pendingSubscriptions.set(channel, options);
|
|
387
|
+
if (this.state !== "connected" || !this.connectionId) {
|
|
388
|
+
return;
|
|
389
|
+
}
|
|
390
|
+
const payload = {
|
|
391
|
+
action: "subscribe",
|
|
392
|
+
channel
|
|
393
|
+
};
|
|
394
|
+
if (options.history) payload.history = true;
|
|
395
|
+
if (options.presence) {
|
|
396
|
+
payload.presence = true;
|
|
397
|
+
if (options.presence_data) payload.presence_data = options.presence_data;
|
|
398
|
+
}
|
|
399
|
+
if (isPrivateChannel(channel)) {
|
|
400
|
+
if (options.auth) {
|
|
401
|
+
payload.auth = options.auth;
|
|
402
|
+
} else if (this.config.onPrivateChannelAuth) {
|
|
403
|
+
payload.auth = await this.config.onPrivateChannelAuth(channel, this.connectionId);
|
|
404
|
+
} else {
|
|
405
|
+
throw new Error(
|
|
406
|
+
`Private channel "${channel}" requires auth. Provide options.auth or onPrivateChannelAuth.`
|
|
407
|
+
);
|
|
408
|
+
}
|
|
409
|
+
}
|
|
410
|
+
this.send(payload);
|
|
411
|
+
}
|
|
412
|
+
unsubscribe(channel) {
|
|
413
|
+
this.pendingSubscriptions.delete(channel);
|
|
414
|
+
this.subscribedChannels.delete(channel);
|
|
415
|
+
if (this.state === "connected") {
|
|
416
|
+
this.send({ action: "unsubscribe", channel });
|
|
417
|
+
}
|
|
418
|
+
}
|
|
419
|
+
publish(channel, event, data) {
|
|
420
|
+
this.send({ action: "publish", channel, event, data });
|
|
421
|
+
}
|
|
422
|
+
presenceEnter(channel, data) {
|
|
423
|
+
this.send({ action: "presence_enter", channel, data });
|
|
424
|
+
}
|
|
425
|
+
presenceUpdate(channel, data) {
|
|
426
|
+
this.send({ action: "presence_update", channel, data });
|
|
427
|
+
}
|
|
428
|
+
presenceLeave(channel) {
|
|
429
|
+
this.send({ action: "presence_leave", channel });
|
|
430
|
+
}
|
|
431
|
+
presenceSync(channel) {
|
|
432
|
+
this.send({ action: "presence_sync", channel });
|
|
433
|
+
}
|
|
434
|
+
ping() {
|
|
435
|
+
this.send({ action: "ping" });
|
|
436
|
+
}
|
|
437
|
+
bind(channel, event, handler) {
|
|
438
|
+
const listener = (message) => {
|
|
439
|
+
if (message.channel === channel && message.event === event) {
|
|
440
|
+
handler(message.data, message);
|
|
441
|
+
}
|
|
442
|
+
};
|
|
443
|
+
return this.on("event", listener);
|
|
444
|
+
}
|
|
445
|
+
bindAll(channel, handler) {
|
|
446
|
+
const listener = (message) => {
|
|
447
|
+
if (message.channel === channel) {
|
|
448
|
+
handler(message.data, message);
|
|
449
|
+
}
|
|
450
|
+
};
|
|
451
|
+
return this.on("event", listener);
|
|
452
|
+
}
|
|
453
|
+
};
|
|
454
|
+
|
|
455
|
+
// src/sse.ts
|
|
456
|
+
var ArkNotifySSE = class {
|
|
457
|
+
constructor(config) {
|
|
458
|
+
this.es = null;
|
|
459
|
+
this.connectionId = null;
|
|
460
|
+
this.listeners = /* @__PURE__ */ new Map();
|
|
461
|
+
this.config = { ...config, baseUrl: resolveBaseUrl(config.baseUrl) };
|
|
462
|
+
this.EventSourceCtor = config.EventSource ?? globalThis.EventSource;
|
|
463
|
+
}
|
|
464
|
+
getConnectionId() {
|
|
465
|
+
return this.connectionId;
|
|
466
|
+
}
|
|
467
|
+
on(event, handler) {
|
|
468
|
+
if (!this.listeners.has(event)) {
|
|
469
|
+
this.listeners.set(event, /* @__PURE__ */ new Set());
|
|
470
|
+
}
|
|
471
|
+
this.listeners.get(event).add(handler);
|
|
472
|
+
return () => this.off(event, handler);
|
|
473
|
+
}
|
|
474
|
+
off(event, handler) {
|
|
475
|
+
this.listeners.get(event)?.delete(handler);
|
|
476
|
+
}
|
|
477
|
+
emit(event, ...args) {
|
|
478
|
+
for (const handler of this.listeners.get(event) ?? []) {
|
|
479
|
+
handler(...args);
|
|
480
|
+
}
|
|
481
|
+
}
|
|
482
|
+
async connect() {
|
|
483
|
+
if (this.es) return;
|
|
484
|
+
const base = this.config.baseUrl.replace(/\/$/, "");
|
|
485
|
+
const url = new URL(`${base}/app/${this.config.appKey}/stream`);
|
|
486
|
+
url.searchParams.set("channels", this.config.channels.join(","));
|
|
487
|
+
const token = resolveValue(this.config.token);
|
|
488
|
+
if (token) {
|
|
489
|
+
url.searchParams.set("token", token);
|
|
490
|
+
} else if (this.config.clientId) {
|
|
491
|
+
url.searchParams.set("clientId", this.config.clientId);
|
|
492
|
+
}
|
|
493
|
+
if (this.config.history) {
|
|
494
|
+
url.searchParams.set("history", "true");
|
|
495
|
+
}
|
|
496
|
+
if (this.config.user_data) {
|
|
497
|
+
url.searchParams.set("user_data", JSON.stringify(this.config.user_data));
|
|
498
|
+
}
|
|
499
|
+
let authMap = this.config.auth ? { ...this.config.auth } : {};
|
|
500
|
+
const privateChannels = this.config.channels.filter(isPrivateChannel);
|
|
501
|
+
if (privateChannels.length > 0 && this.config.onPrivateChannelAuth && !this.config.auth) ;
|
|
502
|
+
if (Object.keys(authMap).length > 0) {
|
|
503
|
+
url.searchParams.set("auth", JSON.stringify(authMap));
|
|
504
|
+
}
|
|
505
|
+
this.es = new this.EventSourceCtor(url.toString());
|
|
506
|
+
this.es.addEventListener("connected", (e) => {
|
|
507
|
+
const message = JSON.parse(e.data);
|
|
508
|
+
this.connectionId = message.connection_id;
|
|
509
|
+
this.emit("connected", message);
|
|
510
|
+
this.emit("message", message);
|
|
511
|
+
});
|
|
512
|
+
this.es.addEventListener("event", (e) => {
|
|
513
|
+
const message = JSON.parse(e.data);
|
|
514
|
+
this.emit("event", message);
|
|
515
|
+
this.emit("message", message);
|
|
516
|
+
});
|
|
517
|
+
this.es.addEventListener("presence", (e) => {
|
|
518
|
+
const message = JSON.parse(e.data);
|
|
519
|
+
this.emit("presence", message);
|
|
520
|
+
this.emit("message", message);
|
|
521
|
+
});
|
|
522
|
+
this.es.onerror = () => {
|
|
523
|
+
this.emit("error", new Error("SSE connection error"));
|
|
524
|
+
};
|
|
525
|
+
}
|
|
526
|
+
disconnect() {
|
|
527
|
+
if (this.es) {
|
|
528
|
+
this.es.close();
|
|
529
|
+
this.es = null;
|
|
530
|
+
this.connectionId = null;
|
|
531
|
+
this.emit("close");
|
|
532
|
+
}
|
|
533
|
+
}
|
|
534
|
+
bind(channel, event, handler) {
|
|
535
|
+
const listener = (message) => {
|
|
536
|
+
if (message.channel === channel && message.event === event) {
|
|
537
|
+
handler(message.data, message);
|
|
538
|
+
}
|
|
539
|
+
};
|
|
540
|
+
return this.on("event", listener);
|
|
541
|
+
}
|
|
542
|
+
bindAll(channel, handler) {
|
|
543
|
+
const listener = (message) => {
|
|
544
|
+
if (message.channel === channel) {
|
|
545
|
+
handler(message.data, message);
|
|
546
|
+
}
|
|
547
|
+
};
|
|
548
|
+
return this.on("event", listener);
|
|
549
|
+
}
|
|
550
|
+
};
|
|
551
|
+
|
|
552
|
+
exports.ArkNotifyClient = ArkNotifyClient;
|
|
553
|
+
exports.ArkNotifyConnection = ArkNotifyConnection;
|
|
554
|
+
exports.ArkNotifyError = ArkNotifyError;
|
|
555
|
+
exports.ArkNotifySSE = ArkNotifySSE;
|
|
556
|
+
exports.DEFAULT_BASE_URL = DEFAULT_BASE_URL;
|
|
557
|
+
exports.configureArkNotify = configureArkNotify;
|
|
558
|
+
exports.fetchConnectionToken = fetchConnectionToken;
|
|
559
|
+
exports.isPrivateChannel = isPrivateChannel;
|
|
560
|
+
exports.resolveBaseUrl = resolveBaseUrl;
|
|
561
|
+
exports.resolveValue = resolveValue;
|
|
562
|
+
exports.toWebSocketUrl = toWebSocketUrl;
|
|
563
|
+
//# sourceMappingURL=index.cjs.map
|
|
564
|
+
//# sourceMappingURL=index.cjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/config.ts","../src/utils.ts","../src/client.ts","../src/connection-token.ts","../src/connection.ts","../src/sse.ts"],"names":[],"mappings":"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","file":"index.cjs","sourcesContent":["export const DEFAULT_BASE_URL = 'https://ark-notify-933303906015.europe-north1.run.app'\n\nlet configuredBaseUrl: string | undefined\n\nexport interface ArkNotifyGlobalConfig {\n baseUrl?: string\n}\n\n/** Set the default base URL once at application startup. */\nexport function configureArkNotify(config: ArkNotifyGlobalConfig): void {\n if (config.baseUrl !== undefined) {\n configuredBaseUrl = config.baseUrl\n }\n}\n\nexport function resolveBaseUrl(baseUrl?: string): string {\n return (baseUrl ?? configuredBaseUrl ?? DEFAULT_BASE_URL).replace(/\\/$/, '')\n}\n","import type { ApiError } from './types'\n\nexport class ArkNotifyError extends Error {\n readonly status: number\n readonly code: string\n readonly retryAfterSec?: number\n readonly reason?: string\n\n constructor(status: number, body: ApiError) {\n super(body.message)\n this.name = 'ArkNotifyError'\n this.status = status\n this.code = body.error\n this.retryAfterSec = body.retryAfterSec\n this.reason = body.reason\n }\n}\n\nexport function toWebSocketUrl(baseUrl: string, path: string): string {\n const url = new URL(path, baseUrl)\n url.protocol = url.protocol === 'https:' ? 'wss:' : 'ws:'\n return url.toString()\n}\n\nexport function resolveValue<T>(value: T | (() => T)): T {\n return typeof value === 'function' ? (value as () => T)() : value\n}\n\nexport function isPrivateChannel(channel: string): boolean {\n return channel.startsWith('private-')\n}\n","import { resolveBaseUrl } from './config'\nimport { ArkNotifyError } from './utils'\nimport type {\n AdminChannelsResponse,\n Application,\n ArkNotifyClientConfig,\n AppCredentials,\n AuthResponse,\n ChannelAuthInput,\n ChannelAuthResponse,\n ConnectionTokenInput,\n ConnectionTokenResponse,\n CreateApplicationInput,\n HealthResponse,\n LoginInput,\n PublishEventInput,\n PublishEventResponse,\n UpdateApplicationInput,\n User,\n} from './types'\nimport { resolveValue } from './utils'\n\ntype RequestOptions = {\n method?: string\n body?: unknown\n headers?: Record<string, string>\n credentials?: AppCredentials\n}\n\nexport class ArkNotifyClient {\n private readonly baseUrl: string\n private readonly fetchFn: typeof fetch\n private token?: string | (() => string | null | undefined)\n\n constructor(config: ArkNotifyClientConfig) {\n this.baseUrl = resolveBaseUrl(config.baseUrl)\n this.token = config.token\n this.fetchFn = config.fetch ?? globalThis.fetch.bind(globalThis)\n }\n\n setToken(token: string | null): void {\n this.token = token ?? undefined\n }\n\n private getAuthHeader(): string | undefined {\n const token = resolveValue(this.token)\n return token ? `Bearer ${token}` : undefined\n }\n\n private async request<T>(path: string, options: RequestOptions = {}): Promise<T> {\n const headers: Record<string, string> = {\n 'Content-Type': 'application/json',\n ...options.headers,\n }\n\n const auth = this.getAuthHeader()\n if (auth) {\n headers.Authorization = auth\n }\n\n if (options.credentials) {\n headers['X-App-Key'] = options.credentials.appKey\n headers['X-App-Secret'] = options.credentials.secret\n }\n\n const response = await this.fetchFn(`${this.baseUrl}${path}`, {\n method: options.method ?? 'GET',\n headers,\n body: options.body !== undefined ? JSON.stringify(options.body) : undefined,\n })\n\n if (!response.ok) {\n let body: { error: string; message: string; reason?: string; retryAfterSec?: number }\n try {\n body = await response.json()\n } catch {\n body = { error: 'request_failed', message: response.statusText }\n }\n throw new ArkNotifyError(response.status, body)\n }\n\n if (response.status === 204) {\n return undefined as T\n }\n\n return response.json() as Promise<T>\n }\n\n // ── Health ──────────────────────────────────────────────────────────────\n\n health(): Promise<HealthResponse> {\n return this.request('/health')\n }\n\n // ── Platform auth ───────────────────────────────────────────────────────\n\n login(input: LoginInput): Promise<AuthResponse> {\n return this.request('/api/v1/auth/login', { method: 'POST', body: input })\n }\n\n me(): Promise<{ user: User }> {\n return this.request('/api/v1/auth/me')\n }\n\n // ── Applications ────────────────────────────────────────────────────────\n\n listApplications(): Promise<{ apps: Application[] }> {\n return this.request('/api/v1/applications')\n }\n\n createApplication(input: CreateApplicationInput): Promise<{ app: Application }> {\n return this.request('/api/v1/applications', { method: 'POST', body: input })\n }\n\n getApplication(id: string): Promise<{ app: Application }> {\n return this.request(`/api/v1/applications/${id}`)\n }\n\n updateApplication(\n id: string,\n input: UpdateApplicationInput\n ): Promise<{ app: Application }> {\n return this.request(`/api/v1/applications/${id}`, { method: 'PUT', body: input })\n }\n\n deleteApplication(id: string): Promise<{ deleted: boolean; id: string }> {\n return this.request(`/api/v1/applications/${id}`, { method: 'DELETE' })\n }\n\n regenerateSecret(id: string): Promise<{ app: Application }> {\n return this.request(`/api/v1/applications/${id}/regenerate-secret`, {\n method: 'POST',\n })\n }\n\n // ── System admin ────────────────────────────────────────────────────────\n\n adminChannels(): Promise<AdminChannelsResponse> {\n return this.request('/api/v1/admin/channels')\n }\n\n // ── Data plane ──────────────────────────────────────────────────────────\n\n publishEvent(\n appKey: string,\n credentials: AppCredentials,\n input: PublishEventInput\n ): Promise<PublishEventResponse> {\n return this.request(`/api/v1/apps/${appKey}/events`, {\n method: 'POST',\n body: input,\n credentials,\n })\n }\n\n authorizeChannel(\n appKey: string,\n credentials: AppCredentials,\n input: ChannelAuthInput\n ): Promise<ChannelAuthResponse> {\n return this.request(`/api/v1/apps/${appKey}/auth`, {\n method: 'POST',\n body: input,\n credentials,\n })\n }\n\n issueConnectionToken(\n appKey: string,\n credentials: AppCredentials,\n input: ConnectionTokenInput\n ): Promise<ConnectionTokenResponse> {\n return this.request(`/api/v1/apps/${appKey}/connection-token`, {\n method: 'POST',\n body: input,\n credentials,\n })\n }\n}\n","import { resolveBaseUrl } from './config'\nimport type { AppCredentials, ConnectionTokenInput, ConnectionTokenResponse } from './types'\nimport { ArkNotifyError } from './utils'\n\nexport interface FetchConnectionTokenOptions extends ConnectionTokenInput {\n baseUrl?: string\n appKey: string\n credentials: AppCredentials\n fetch?: typeof fetch\n}\n\n/**\n * Request a signed connection token from the Ark Notify API.\n * Requires app credentials — use from your backend, not in browser code.\n */\nexport async function fetchConnectionToken(\n options: FetchConnectionTokenOptions\n): Promise<ConnectionTokenResponse> {\n const {\n baseUrl,\n appKey,\n credentials,\n fetch: fetchFn = globalThis.fetch.bind(globalThis),\n client_id,\n clientId,\n user_data,\n userData,\n ttl,\n capabilities,\n serverAuthUrl,\n } = options\n\n const resolvedClientId = client_id ?? clientId\n if (!resolvedClientId) {\n throw new Error('client_id is required to fetch a connection token')\n }\n\n const body: ConnectionTokenInput = {\n client_id: resolvedClientId,\n user_data: user_data ?? userData,\n ttl,\n capabilities,\n serverAuthUrl,\n }\n\n const url = `${resolveBaseUrl(baseUrl)}/api/v1/apps/${appKey}/connection-token`\n const response = await fetchFn(url, {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n 'X-App-Key': credentials.appKey,\n 'X-App-Secret': credentials.secret,\n },\n body: JSON.stringify(body),\n })\n\n if (!response.ok) {\n let errorBody: { error: string; message: string; reason?: string; retryAfterSec?: number }\n try {\n errorBody = await response.json()\n } catch {\n errorBody = { error: 'request_failed', message: response.statusText }\n }\n throw new ArkNotifyError(response.status, errorBody)\n }\n\n return response.json() as Promise<ConnectionTokenResponse>\n}\n","import { resolveBaseUrl } from './config'\nimport { fetchConnectionToken } from './connection-token'\nimport type {\n ArkNotifyConnectionConfig,\n ChannelEventHandler,\n ConnectedMessage,\n ConnectionState,\n EventMessage,\n PresenceMessage,\n ServerMessage,\n SubscribeOptions,\n} from './types'\nimport { isPrivateChannel, resolveValue, toWebSocketUrl } from './utils'\n\ntype EventMap = {\n state: (state: ConnectionState) => void\n connected: (message: ConnectedMessage) => void\n event: (message: EventMessage) => void\n presence: (message: PresenceMessage) => void\n message: (message: ServerMessage) => void\n error: (error: { code: string; message: string }) => void\n close: (event: { code: number; reason: string }) => void\n}\n\ntype EventName = keyof EventMap\n\nexport class ArkNotifyConnection {\n private readonly config: Required<\n Pick<\n ArkNotifyConnectionConfig,\n 'baseUrl' | 'appKey' | 'autoReconnect' | 'reconnectDelayMs' | 'maxReconnectDelayMs'\n >\n > &\n ArkNotifyConnectionConfig\n\n private ws: WebSocket | null = null\n private state: ConnectionState = 'disconnected'\n private connectionId: string | null = null\n private clientId: string | null = null\n private authenticated = false\n private subscribedChannels = new Set<string>()\n private pendingSubscriptions = new Map<string, SubscribeOptions>()\n private listeners = new Map<EventName, Set<EventMap[EventName]>>()\n private reconnectAttempt = 0\n private reconnectTimer: ReturnType<typeof setTimeout> | null = null\n private intentionalClose = false\n private connectPromise: Promise<void> | null = null\n private readonly WebSocketCtor: typeof WebSocket\n\n constructor(config: ArkNotifyConnectionConfig) {\n this.config = {\n autoReconnect: true,\n reconnectDelayMs: 1000,\n maxReconnectDelayMs: 30000,\n ...config,\n baseUrl: resolveBaseUrl(config.baseUrl),\n }\n this.WebSocketCtor = config.WebSocket ?? globalThis.WebSocket\n }\n\n getConnectionState(): ConnectionState {\n return this.state\n }\n\n getConnectionId(): string | null {\n return this.connectionId\n }\n\n getClientId(): string | null {\n return this.clientId\n }\n\n isAuthenticated(): boolean {\n return this.authenticated\n }\n\n getSubscribedChannels(): string[] {\n return [...this.subscribedChannels]\n }\n\n on<E extends EventName>(event: E, handler: EventMap[E]): () => void {\n if (!this.listeners.has(event)) {\n this.listeners.set(event, new Set())\n }\n this.listeners.get(event)!.add(handler as EventMap[EventName])\n return () => this.off(event, handler)\n }\n\n off<E extends EventName>(event: E, handler: EventMap[E]): void {\n this.listeners.get(event)?.delete(handler as EventMap[EventName])\n }\n\n private emit<E extends EventName>(event: E, ...args: Parameters<EventMap[E]>): void {\n for (const handler of this.listeners.get(event) ?? []) {\n ;(handler as (...a: Parameters<EventMap[E]>) => void)(...args)\n }\n }\n\n private setState(state: ConnectionState): void {\n if (this.state !== state) {\n this.state = state\n this.emit('state', state)\n }\n }\n\n connect(): Promise<void> {\n if (this.ws?.readyState === WebSocket.OPEN || this.ws?.readyState === WebSocket.CONNECTING) {\n return Promise.resolve()\n }\n\n if (this.connectPromise) {\n return this.connectPromise\n }\n\n this.connectPromise = this.doConnect().finally(() => {\n this.connectPromise = null\n })\n\n return this.connectPromise\n }\n\n private async doConnect(): Promise<void> {\n this.intentionalClose = false\n this.clearReconnectTimer()\n this.setState(this.reconnectAttempt > 0 ? 'reconnecting' : 'connecting')\n\n const url = new URL(\n toWebSocketUrl(this.config.baseUrl, `/app/${this.config.appKey}`)\n )\n\n let token = resolveValue(this.config.token)\n if (!token && this.config.clientId && this.config.credentials) {\n const result = await fetchConnectionToken({\n baseUrl: this.config.baseUrl,\n appKey: this.config.appKey,\n credentials: this.config.credentials,\n client_id: this.config.clientId,\n user_data: this.config.user_data,\n fetch: this.config.fetch,\n })\n token = result.token\n }\n\n if (token) {\n url.searchParams.set('token', token)\n } else if (this.config.clientId) {\n url.searchParams.set('clientId', this.config.clientId)\n }\n\n this.ws = new this.WebSocketCtor(url.toString())\n\n this.ws.onopen = () => {\n this.reconnectAttempt = 0\n }\n\n this.ws.onmessage = (event) => {\n this.handleMessage(event.data as string)\n }\n\n this.ws.onerror = () => {\n this.emit('error', { code: 'websocket_error', message: 'WebSocket error' })\n }\n\n this.ws.onclose = (event) => {\n this.ws = null\n this.connectionId = null\n this.emit('close', { code: event.code, reason: event.reason })\n\n if (!this.intentionalClose && this.config.autoReconnect) {\n this.scheduleReconnect()\n } else {\n this.setState('failed')\n }\n }\n }\n\n disconnect(): void {\n this.intentionalClose = true\n this.clearReconnectTimer()\n this.subscribedChannels.clear()\n this.pendingSubscriptions.clear()\n if (this.ws) {\n this.ws.close()\n this.ws = null\n }\n this.setState('disconnected')\n }\n\n private scheduleReconnect(): void {\n this.setState('reconnecting')\n const delay = Math.min(\n this.config.reconnectDelayMs * 2 ** this.reconnectAttempt,\n this.config.maxReconnectDelayMs\n )\n this.reconnectAttempt++\n this.reconnectTimer = setTimeout(() => void this.connect(), delay)\n }\n\n private clearReconnectTimer(): void {\n if (this.reconnectTimer) {\n clearTimeout(this.reconnectTimer)\n this.reconnectTimer = null\n }\n }\n\n private send(payload: Record<string, unknown>): void {\n if (this.ws?.readyState !== WebSocket.OPEN) {\n throw new Error('WebSocket is not connected')\n }\n this.ws.send(JSON.stringify(payload))\n }\n\n private handleMessage(raw: string): void {\n let message: ServerMessage\n try {\n message = JSON.parse(raw) as ServerMessage\n } catch {\n this.emit('error', { code: 'invalid_json', message: 'Invalid JSON from server' })\n return\n }\n\n this.emit('message', message)\n\n switch (message.type) {\n case 'connected': {\n this.connectionId = message.connection_id\n this.clientId = message.client_id\n this.authenticated = message.authenticated\n this.setState('connected')\n this.emit('connected', message)\n this.resubscribeAll()\n break\n }\n case 'event':\n this.emit('event', message)\n break\n case 'presence':\n this.emit('presence', message)\n break\n case 'subscribed':\n this.subscribedChannels.add(message.channel)\n break\n case 'unsubscribed':\n this.subscribedChannels.delete(message.channel)\n break\n case 'error':\n this.emit('error', { code: message.code, message: message.message })\n break\n case 'ping':\n this.send({ action: 'ping' })\n break\n default:\n break\n }\n }\n\n private resubscribeAll(): void {\n for (const [channel, options] of this.pendingSubscriptions) {\n void this.subscribe(channel, options)\n }\n }\n\n async subscribe(channel: string, options: SubscribeOptions = {}): Promise<void> {\n this.pendingSubscriptions.set(channel, options)\n\n if (this.state !== 'connected' || !this.connectionId) {\n return\n }\n\n const payload: Record<string, unknown> = {\n action: 'subscribe',\n channel,\n }\n\n if (options.history) payload.history = true\n if (options.presence) {\n payload.presence = true\n if (options.presence_data) payload.presence_data = options.presence_data\n }\n\n if (isPrivateChannel(channel)) {\n if (options.auth) {\n payload.auth = options.auth\n } else if (this.config.onPrivateChannelAuth) {\n payload.auth = await this.config.onPrivateChannelAuth(channel, this.connectionId)\n } else {\n throw new Error(\n `Private channel \"${channel}\" requires auth. Provide options.auth or onPrivateChannelAuth.`\n )\n }\n }\n\n this.send(payload)\n }\n\n unsubscribe(channel: string): void {\n this.pendingSubscriptions.delete(channel)\n this.subscribedChannels.delete(channel)\n if (this.state === 'connected') {\n this.send({ action: 'unsubscribe', channel })\n }\n }\n\n publish(channel: string, event: string, data?: unknown): void {\n this.send({ action: 'publish', channel, event, data })\n }\n\n presenceEnter(channel: string, data: Record<string, unknown>): void {\n this.send({ action: 'presence_enter', channel, data })\n }\n\n presenceUpdate(channel: string, data: Record<string, unknown>): void {\n this.send({ action: 'presence_update', channel, data })\n }\n\n presenceLeave(channel: string): void {\n this.send({ action: 'presence_leave', channel })\n }\n\n presenceSync(channel: string): void {\n this.send({ action: 'presence_sync', channel })\n }\n\n ping(): void {\n this.send({ action: 'ping' })\n }\n\n bind(channel: string, event: string, handler: ChannelEventHandler): () => void {\n const listener = (message: EventMessage) => {\n if (message.channel === channel && message.event === event) {\n handler(message.data, message)\n }\n }\n return this.on('event', listener)\n }\n\n bindAll(channel: string, handler: ChannelEventHandler): () => void {\n const listener = (message: EventMessage) => {\n if (message.channel === channel) {\n handler(message.data, message)\n }\n }\n return this.on('event', listener)\n }\n}\n","import { resolveBaseUrl } from './config'\nimport type {\n ArkNotifySSEConfig,\n ConnectedMessage,\n EventMessage,\n PresenceMessage,\n ServerMessage,\n} from './types'\nimport { isPrivateChannel, resolveValue } from './utils'\n\ntype SSEEventMap = {\n connected: (message: ConnectedMessage) => void\n event: (message: EventMessage) => void\n presence: (message: PresenceMessage) => void\n message: (message: ServerMessage) => void\n error: (error: Error) => void\n close: () => void\n}\n\ntype SSEEventName = keyof SSEEventMap\n\nexport class ArkNotifySSE {\n private readonly config: ArkNotifySSEConfig & { baseUrl: string }\n private es: EventSource | null = null\n private connectionId: string | null = null\n private listeners = new Map<SSEEventName, Set<SSEEventMap[SSEEventName]>>()\n private readonly EventSourceCtor: typeof EventSource\n\n constructor(config: ArkNotifySSEConfig) {\n this.config = { ...config, baseUrl: resolveBaseUrl(config.baseUrl) }\n this.EventSourceCtor = config.EventSource ?? globalThis.EventSource\n }\n\n getConnectionId(): string | null {\n return this.connectionId\n }\n\n on<E extends SSEEventName>(event: E, handler: SSEEventMap[E]): () => void {\n if (!this.listeners.has(event)) {\n this.listeners.set(event, new Set())\n }\n this.listeners.get(event)!.add(handler as SSEEventMap[SSEEventName])\n return () => this.off(event, handler)\n }\n\n off<E extends SSEEventName>(event: E, handler: SSEEventMap[E]): void {\n this.listeners.get(event)?.delete(handler as SSEEventMap[SSEEventName])\n }\n\n private emit<E extends SSEEventName>(event: E, ...args: Parameters<SSEEventMap[E]>): void {\n for (const handler of this.listeners.get(event) ?? []) {\n ;(handler as (...a: Parameters<SSEEventMap[E]>) => void)(...args)\n }\n }\n\n async connect(): Promise<void> {\n if (this.es) return\n\n const base = this.config.baseUrl.replace(/\\/$/, '')\n const url = new URL(`${base}/app/${this.config.appKey}/stream`)\n url.searchParams.set('channels', this.config.channels.join(','))\n\n const token = resolveValue(this.config.token)\n if (token) {\n url.searchParams.set('token', token)\n } else if (this.config.clientId) {\n url.searchParams.set('clientId', this.config.clientId)\n }\n\n if (this.config.history) {\n url.searchParams.set('history', 'true')\n }\n\n if (this.config.user_data) {\n url.searchParams.set('user_data', JSON.stringify(this.config.user_data))\n }\n\n let authMap = this.config.auth ? { ...this.config.auth } : {}\n\n const privateChannels = this.config.channels.filter(isPrivateChannel)\n if (privateChannels.length > 0 && this.config.onPrivateChannelAuth && !this.config.auth) {\n // Private SSE channels need auth tokens in the query before connect.\n // Fetch tokens server-side and pass them via the `auth` option, or use WebSocket instead.\n }\n\n if (Object.keys(authMap).length > 0) {\n url.searchParams.set('auth', JSON.stringify(authMap))\n }\n\n this.es = new this.EventSourceCtor(url.toString())\n\n this.es.addEventListener('connected', (e) => {\n const message = JSON.parse((e as MessageEvent).data) as ConnectedMessage\n this.connectionId = message.connection_id\n this.emit('connected', message)\n this.emit('message', message)\n })\n\n this.es.addEventListener('event', (e) => {\n const message = JSON.parse((e as MessageEvent).data) as EventMessage\n this.emit('event', message)\n this.emit('message', message)\n })\n\n this.es.addEventListener('presence', (e) => {\n const message = JSON.parse((e as MessageEvent).data) as PresenceMessage\n this.emit('presence', message)\n this.emit('message', message)\n })\n\n this.es.onerror = () => {\n this.emit('error', new Error('SSE connection error'))\n }\n }\n\n disconnect(): void {\n if (this.es) {\n this.es.close()\n this.es = null\n this.connectionId = null\n this.emit('close')\n }\n }\n\n bind(channel: string, event: string, handler: (data: unknown, message: EventMessage) => void): () => void {\n const listener = (message: EventMessage) => {\n if (message.channel === channel && message.event === event) {\n handler(message.data, message)\n }\n }\n return this.on('event', listener)\n }\n\n bindAll(channel: string, handler: (data: unknown, message: EventMessage) => void): () => void {\n const listener = (message: EventMessage) => {\n if (message.channel === channel) {\n handler(message.data, message)\n }\n }\n return this.on('event', listener)\n }\n}\n"]}
|
package/dist/index.d.cts
ADDED
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
import { C as ConnectionTokenInput, A as AppCredentials, a as ConnectionTokenResponse } from './utils-Cw2SnD6p.cjs';
|
|
2
|
+
export { b as AdminChannel, c as AdminChannelsResponse, d as AdminConnection, e as ApiError, f as Application, g as ArkNotifyClient, h as ArkNotifyClientConfig, i as ArkNotifyConnection, j as ArkNotifyConnectionConfig, k as ArkNotifyError, l as ArkNotifyGlobalConfig, m as ArkNotifySSE, n as ArkNotifySSEConfig, o as AuthResponse, p as ChannelAuthInput, q as ChannelAuthResponse, r as ChannelEventHandler, s as ChannelHandlers, t as ConnectedMessage, u as ConnectionCapabilities, v as ConnectionState, w as CreateApplicationInput, D as DEFAULT_BASE_URL, E as EventMessage, H as HealthResponse, L as LoginInput, P as PingMessage, x as PongMessage, y as PresenceAction, z as PresenceLeftMessage, B as PresenceMember, F as PresenceMemberInfo, G as PresenceMessage, I as PresenceUpdatedMessage, J as PrivateChannelAuthHandler, K as PublishEventInput, M as PublishEventResponse, N as PublishedMessage, S as ServerErrorMessage, O as ServerMessage, Q as SubscribeOptions, R as SubscribedMessage, U as UnsubscribedMessage, T as UpdateApplicationInput, V as User, W as UserRole, X as configureArkNotify, Y as isPrivateChannel, Z as resolveBaseUrl, _ as resolveValue, $ as toWebSocketUrl } from './utils-Cw2SnD6p.cjs';
|
|
3
|
+
|
|
4
|
+
interface FetchConnectionTokenOptions extends ConnectionTokenInput {
|
|
5
|
+
baseUrl?: string;
|
|
6
|
+
appKey: string;
|
|
7
|
+
credentials: AppCredentials;
|
|
8
|
+
fetch?: typeof fetch;
|
|
9
|
+
}
|
|
10
|
+
/**
|
|
11
|
+
* Request a signed connection token from the Ark Notify API.
|
|
12
|
+
* Requires app credentials — use from your backend, not in browser code.
|
|
13
|
+
*/
|
|
14
|
+
declare function fetchConnectionToken(options: FetchConnectionTokenOptions): Promise<ConnectionTokenResponse>;
|
|
15
|
+
|
|
16
|
+
export { AppCredentials, ConnectionTokenInput, ConnectionTokenResponse, type FetchConnectionTokenOptions, fetchConnectionToken };
|