@bananalink-test/agent 1.0.1

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.
@@ -0,0 +1,175 @@
1
+ import { Agent, Agent as Agent$1, EthSendTransactionParams, EthSendTransactionParams as EthSendTransactionParams$1, EthSendTransactionResult, TransportHandle } from "@bananalink-test/sdk-core";
2
+
3
+ //#region src/types/ConnectOpts.d.ts
4
+ type ConnectOpts = {
5
+ accessToken?: string;
6
+ };
7
+ //#endregion
8
+ //#region src/types/JwtPayload.d.ts
9
+ type JwtPayload = {
10
+ agentId: string;
11
+ jwt: string;
12
+ };
13
+ //#endregion
14
+ //#region src/types/RequestMap.d.ts
15
+ type RequestMap = {
16
+ eth_sendTransaction: {
17
+ params: [EthSendTransactionParams$1];
18
+ result: string;
19
+ };
20
+ };
21
+ //#endregion
22
+ //#region src/types/Request.d.ts
23
+ type Request = keyof RequestMap;
24
+ //#endregion
25
+ //#region src/types/RequestParams.d.ts
26
+ type RequestParams<T extends keyof RequestMap> = RequestMap[T]['params'];
27
+ //#endregion
28
+ //#region src/types/RequestResult.d.ts
29
+ type RequestResult<T extends keyof RequestMap> = RequestMap[T]['result'];
30
+ //#endregion
31
+ //#region src/types/SessionClaims.d.ts
32
+ type SessionClaims = {
33
+ accessToken: string;
34
+ };
35
+ //#endregion
36
+ //#region src/core/BananalinkSession.d.ts
37
+ declare class BananalinkSession {
38
+ readonly sessionClaims: SessionClaims;
39
+ private readonly ws;
40
+ private readonly apiUrl;
41
+ private static readonly REQUEST_TIMEOUT_MS;
42
+ private static readonly CREATE_DAPP_MESSAGE_MAX_ATTEMPTS;
43
+ private static readonly CREATE_DAPP_MESSAGE_BASE_DELAY_MS;
44
+ private readonly pendingRequests;
45
+ private readonly stopListening;
46
+ private readonly stopListeningOnClose;
47
+ private closed;
48
+ private static readonly RESULT_HANDLERS;
49
+ constructor(sessionClaims: SessionClaims, ws: TransportHandle, apiUrl: string);
50
+ request<T extends Request>({
51
+ method,
52
+ params,
53
+ timeoutMs
54
+ }: {
55
+ method: T;
56
+ params?: RequestParams<T>;
57
+ timeoutMs?: number;
58
+ }): Promise<RequestResult<T>>;
59
+ close(): void;
60
+ private handleMessageResponse;
61
+ private cleanup;
62
+ private rejectAllPendingRequests;
63
+ }
64
+ //#endregion
65
+ //#region src/core/BananalinkConnection.d.ts
66
+ type BananalinkAgentJwt = {
67
+ jwtPayload: JwtPayload;
68
+ accessToken: string;
69
+ } | undefined;
70
+ declare class BananalinkConnection {
71
+ private static readonly AUTH_TIMEOUT_MILLIS;
72
+ private static readonly BIND_RESPONSE_TIMEOUT_MILLIS;
73
+ private readonly apiUrl;
74
+ private readonly wsUrl;
75
+ private readonly jwt;
76
+ private readonly agentPairingCode;
77
+ private pendingSessionPromise;
78
+ private pendingSessionAbortController;
79
+ constructor(opts: {
80
+ apiUrl: string;
81
+ wsUrl: string;
82
+ jwt: BananalinkAgentJwt;
83
+ agentPairingCode: string | null;
84
+ });
85
+ get connectionUrl(): string;
86
+ abortPendingSession(): void;
87
+ getSession(): Promise<BananalinkSession>;
88
+ private openSession;
89
+ private resumeSession;
90
+ private waitForAuthorization;
91
+ private getTransportHandle;
92
+ private bind;
93
+ }
94
+ //#endregion
95
+ //#region src/core/BananalinkAgent.d.ts
96
+ declare class BananalinkAgent {
97
+ private readonly apiUrl;
98
+ private readonly wsUrl;
99
+ private readonly jwks;
100
+ private readonly agent;
101
+ private static readonly DEFAULT_BANANALINK_API_URL;
102
+ private static readonly DEFAULT_BANANALINK_WS_URL;
103
+ constructor(config: {
104
+ apiUrl?: string;
105
+ wsUrl?: string;
106
+ agent: Agent$1;
107
+ });
108
+ connect(opts?: ConnectOpts): Promise<BananalinkConnection>;
109
+ private getJwtPayload;
110
+ private pairAgent;
111
+ }
112
+ //#endregion
113
+ //#region src/errors/BananalinkError.d.ts
114
+ declare class BananalinkError extends Error {
115
+ constructor(message: string, options?: ErrorOptions);
116
+ }
117
+ //#endregion
118
+ //#region src/errors/AgentApiError.d.ts
119
+ declare class AgentApiError extends BananalinkError {
120
+ readonly statusCode: number;
121
+ constructor(message: string, statusCode: number);
122
+ }
123
+ //#endregion
124
+ //#region src/errors/ConnectionRejectedError.d.ts
125
+ declare class ConnectionRejectedError extends BananalinkError {
126
+ constructor();
127
+ }
128
+ //#endregion
129
+ //#region src/errors/ConnectionTimeoutError.d.ts
130
+ declare class ConnectionTimeoutError extends BananalinkError {
131
+ constructor();
132
+ }
133
+ //#endregion
134
+ //#region src/errors/InvalidConfigError.d.ts
135
+ declare class InvalidConfigError extends BananalinkError {
136
+ constructor(message: string);
137
+ }
138
+ //#endregion
139
+ //#region src/errors/InvalidConnectionStateError.d.ts
140
+ declare class InvalidConnectionStateError extends BananalinkError {
141
+ constructor(message: string);
142
+ }
143
+ //#endregion
144
+ //#region src/errors/InvalidJwtError.d.ts
145
+ declare class InvalidJwtError extends BananalinkError {
146
+ constructor();
147
+ }
148
+ //#endregion
149
+ //#region src/errors/PendingSessionAbortedError.d.ts
150
+ declare class PendingSessionAbortedError extends BananalinkError {
151
+ constructor();
152
+ }
153
+ //#endregion
154
+ //#region src/errors/RequestRejectedError.d.ts
155
+ declare class RequestRejectedError extends BananalinkError {
156
+ readonly messageId: string;
157
+ constructor(messageId: string);
158
+ }
159
+ //#endregion
160
+ //#region src/errors/RequestTimeoutError.d.ts
161
+ declare class RequestTimeoutError extends BananalinkError {
162
+ readonly method: Request;
163
+ constructor(method: Request);
164
+ }
165
+ //#endregion
166
+ //#region src/errors/SessionClosedError.d.ts
167
+ declare class SessionClosedError extends BananalinkError {
168
+ constructor();
169
+ }
170
+ //#endregion
171
+ //#region src/utils/displayBananalinkQR.d.ts
172
+ declare function displayBananalinkQR(url: string): Promise<string>;
173
+ //#endregion
174
+ export { type Agent, AgentApiError, BananalinkAgent, BananalinkConnection, BananalinkError, BananalinkSession, type ConnectOpts, ConnectionRejectedError, ConnectionTimeoutError, type EthSendTransactionParams, type EthSendTransactionResult, InvalidConfigError, InvalidConnectionStateError, InvalidJwtError, PendingSessionAbortedError, type Request, type RequestMap, type RequestParams, RequestRejectedError, type RequestResult, RequestTimeoutError, SessionClosedError, displayBananalinkQR };
175
+ //# sourceMappingURL=index.d.mts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.mts","names":[],"sources":["../src/types/ConnectOpts.ts","../src/types/JwtPayload.ts","../src/types/RequestMap.ts","../src/types/Request.ts","../src/types/RequestParams.ts","../src/types/RequestResult.ts","../src/types/SessionClaims.ts","../src/core/BananalinkSession.ts","../src/core/BananalinkConnection.ts","../src/core/BananalinkAgent.ts","../src/errors/BananalinkError.ts","../src/errors/AgentApiError.ts","../src/errors/ConnectionRejectedError.ts","../src/errors/ConnectionTimeoutError.ts","../src/errors/InvalidConfigError.ts","../src/errors/InvalidConnectionStateError.ts","../src/errors/InvalidJwtError.ts","../src/errors/PendingSessionAbortedError.ts","../src/errors/RequestRejectedError.ts","../src/errors/RequestTimeoutError.ts","../src/errors/SessionClosedError.ts","../src/utils/displayBananalinkQR.ts"],"mappings":"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"}
package/dist/index.mjs ADDED
@@ -0,0 +1,439 @@
1
+ import { createRemoteJWKSet, jwtVerify } from "jose";
2
+ import { BindTransportError, bindTransport, connectWebSocket, createReconnectingTransport, isAuthorizedMessage, isMessageResponseMessage, isRejectedMessage, waitForTransportEvent, withBackoff } from "@bananalink-test/sdk-core";
3
+ import QRCode from "qrcode";
4
+
5
+ //#region src/errors/BananalinkError.ts
6
+ var BananalinkError = class extends Error {
7
+ constructor(message, options) {
8
+ super(message, options);
9
+ this.name = "BananalinkError";
10
+ }
11
+ };
12
+
13
+ //#endregion
14
+ //#region src/errors/AgentApiError.ts
15
+ var AgentApiError = class extends BananalinkError {
16
+ constructor(message, statusCode) {
17
+ super(message);
18
+ this.statusCode = statusCode;
19
+ this.name = "AgentApiError";
20
+ }
21
+ };
22
+
23
+ //#endregion
24
+ //#region src/api/createAgentPairingRequest.ts
25
+ function isCreateAgentPairingResponse(body) {
26
+ return typeof body === "object" && body !== null && "pairingCode" in body;
27
+ }
28
+ async function createAgentPairingRequest(apiUrl, params) {
29
+ const response = await fetch(`${apiUrl}/agents/pairings`, {
30
+ headers: { "Content-Type": "application/json" },
31
+ method: "POST",
32
+ body: JSON.stringify(params)
33
+ });
34
+ if (response.ok) {
35
+ const body = await response.json().catch(() => {
36
+ throw new AgentApiError("Invalid JSON response from bananalink API", response.status);
37
+ });
38
+ if (!isCreateAgentPairingResponse(body)) throw new AgentApiError("Unexpected response shape from bananalink API", response.status);
39
+ return body;
40
+ }
41
+ throw new AgentApiError("Failed to create agent pairing request", response.status);
42
+ }
43
+
44
+ //#endregion
45
+ //#region src/errors/InvalidConfigError.ts
46
+ var InvalidConfigError = class extends BananalinkError {
47
+ constructor(message) {
48
+ super(message);
49
+ this.name = "InvalidConfigError";
50
+ }
51
+ };
52
+
53
+ //#endregion
54
+ //#region src/errors/InvalidJwtError.ts
55
+ var InvalidJwtError = class extends BananalinkError {
56
+ constructor() {
57
+ super("Invalid Bananalink agent jwt");
58
+ }
59
+ };
60
+
61
+ //#endregion
62
+ //#region src/jwt/createBananalinkJwks.ts
63
+ function createBananalinkJwks(apiUrl) {
64
+ return createRemoteJWKSet(new URL(`${apiUrl}/.well-known/jwks.json`));
65
+ }
66
+
67
+ //#endregion
68
+ //#region src/jwt/verifyJwt.ts
69
+ async function verifyJwt(jwt, jwks) {
70
+ try {
71
+ const { payload } = await jwtVerify(jwt, jwks, {
72
+ algorithms: ["ES256"],
73
+ issuer: "bananalink",
74
+ requiredClaims: ["sub", "aud"]
75
+ });
76
+ return {
77
+ agentId: Array.isArray(payload.aud) ? payload.aud[0] : payload.aud,
78
+ jwt
79
+ };
80
+ } catch {
81
+ return null;
82
+ }
83
+ }
84
+
85
+ //#endregion
86
+ //#region src/utils/isValidUrl.ts
87
+ function isValidUrl(url, protocols) {
88
+ try {
89
+ const parsed = new URL(url);
90
+ return protocols.includes(parsed.protocol.replace(":", ""));
91
+ } catch {
92
+ return false;
93
+ }
94
+ }
95
+
96
+ //#endregion
97
+ //#region src/errors/ConnectionRejectedError.ts
98
+ var ConnectionRejectedError = class extends BananalinkError {
99
+ constructor() {
100
+ super("Agent connection attempt was rejected by the hub");
101
+ this.name = "ConnectionRejectedError";
102
+ }
103
+ };
104
+
105
+ //#endregion
106
+ //#region src/errors/ConnectionTimeoutError.ts
107
+ var ConnectionTimeoutError = class extends BananalinkError {
108
+ constructor() {
109
+ super("Agent connection attempt timed out");
110
+ this.name = "ConnectionTimeoutError";
111
+ }
112
+ };
113
+
114
+ //#endregion
115
+ //#region src/errors/InvalidConnectionStateError.ts
116
+ var InvalidConnectionStateError = class extends BananalinkError {
117
+ constructor(message) {
118
+ super(message);
119
+ this.name = "InvalidConnectionStateError";
120
+ }
121
+ };
122
+
123
+ //#endregion
124
+ //#region src/errors/PendingSessionAbortedError.ts
125
+ var PendingSessionAbortedError = class extends BananalinkError {
126
+ constructor() {
127
+ super("Pending session was intentionally aborted");
128
+ this.name = "PendingSessionAbortedError";
129
+ }
130
+ };
131
+
132
+ //#endregion
133
+ //#region src/api/createAgentMessage.ts
134
+ function isCreateAgentMessageResponse(body) {
135
+ return typeof body === "object" && body !== null && "messageId" in body && typeof body.messageId === "string";
136
+ }
137
+ async function createAgentMessage(apiUrl, accessToken, params) {
138
+ const response = await fetch(`${apiUrl}/agents/messages`, {
139
+ method: "POST",
140
+ headers: {
141
+ Authorization: `Bearer ${accessToken}`,
142
+ "Content-Type": "application/json"
143
+ },
144
+ body: JSON.stringify(params)
145
+ });
146
+ if (response.ok) {
147
+ const body = await response.json().catch(() => {
148
+ throw new AgentApiError("Invalid JSON response from bananalink API", response.status);
149
+ });
150
+ if (!isCreateAgentMessageResponse(body)) throw new AgentApiError("Unexpected response shape from bananalink API", response.status);
151
+ return body;
152
+ }
153
+ throw new AgentApiError("Failed to create agent message", response.status);
154
+ }
155
+
156
+ //#endregion
157
+ //#region src/errors/RequestRejectedError.ts
158
+ var RequestRejectedError = class extends BananalinkError {
159
+ constructor(messageId) {
160
+ super(`Request rejected by hub (messageId: ${messageId})`);
161
+ this.messageId = messageId;
162
+ this.name = "RequestRejectedError";
163
+ }
164
+ };
165
+
166
+ //#endregion
167
+ //#region src/errors/RequestTimeoutError.ts
168
+ var RequestTimeoutError = class extends BananalinkError {
169
+ constructor(method) {
170
+ super(`Timeout reached for request ${method}`);
171
+ this.method = method;
172
+ }
173
+ };
174
+
175
+ //#endregion
176
+ //#region src/errors/SessionClosedError.ts
177
+ var SessionClosedError = class extends BananalinkError {
178
+ constructor() {
179
+ super("Session is closed");
180
+ }
181
+ };
182
+
183
+ //#endregion
184
+ //#region src/core/BananalinkSession.ts
185
+ var BananalinkSession = class BananalinkSession {
186
+ static REQUEST_TIMEOUT_MS = 600 * 1e3;
187
+ static CREATE_DAPP_MESSAGE_MAX_ATTEMPTS = 3;
188
+ static CREATE_DAPP_MESSAGE_BASE_DELAY_MS = 1e3;
189
+ pendingRequests = /* @__PURE__ */ new Map();
190
+ stopListening;
191
+ stopListeningOnClose;
192
+ closed;
193
+ static RESULT_HANDLERS = { eth_sendTransaction: (result) => {
194
+ if (typeof result === "object" && result !== null && "txHash" in result && typeof result.txHash === "string") return result.txHash;
195
+ throw new BananalinkError(`Unexpected eth_sendTransaction result: ${result}`);
196
+ } };
197
+ constructor(sessionClaims, ws, apiUrl) {
198
+ this.sessionClaims = sessionClaims;
199
+ this.ws = ws;
200
+ this.apiUrl = apiUrl;
201
+ this.stopListening = this.ws.onMessage((payload) => {
202
+ if (isMessageResponseMessage(payload)) this.handleMessageResponse(payload);
203
+ });
204
+ this.stopListeningOnClose = this.ws.onClose(() => {
205
+ this.cleanup();
206
+ this.closed = true;
207
+ });
208
+ this.closed = false;
209
+ }
210
+ async request({ method, params, timeoutMs = BananalinkSession.REQUEST_TIMEOUT_MS }) {
211
+ if (this.closed) throw new SessionClosedError();
212
+ const { messageId } = await withBackoff(async () => await createAgentMessage(this.apiUrl, this.sessionClaims.accessToken, {
213
+ method,
214
+ payload: params
215
+ }), BananalinkSession.CREATE_DAPP_MESSAGE_MAX_ATTEMPTS, BananalinkSession.CREATE_DAPP_MESSAGE_BASE_DELAY_MS);
216
+ if (this.closed) throw new SessionClosedError();
217
+ return new Promise((resolve, reject) => {
218
+ let timeout;
219
+ if (timeoutMs > 0) timeout = setTimeout(() => {
220
+ this.pendingRequests.delete(messageId);
221
+ reject(new RequestTimeoutError(method));
222
+ }, timeoutMs);
223
+ this.pendingRequests.set(messageId, {
224
+ timeout,
225
+ handle: (messageResponse) => {
226
+ if (messageResponse.status === "rejected") {
227
+ reject(new RequestRejectedError(messageId));
228
+ return;
229
+ }
230
+ const resultHandler = BananalinkSession.RESULT_HANDLERS[method];
231
+ try {
232
+ resolve(resultHandler(messageResponse.payloadResponse));
233
+ } catch (error) {
234
+ reject(new BananalinkError(`Failed resolving request response, ${error}`));
235
+ }
236
+ },
237
+ reject
238
+ });
239
+ });
240
+ }
241
+ close() {
242
+ this.cleanup();
243
+ if (!this.ws.closed) this.ws.close();
244
+ this.closed = true;
245
+ }
246
+ handleMessageResponse(messageResponseMessage) {
247
+ const pending = this.pendingRequests.get(messageResponseMessage.msgId);
248
+ if (!pending) return;
249
+ this.pendingRequests.delete(messageResponseMessage.msgId);
250
+ clearTimeout(pending.timeout);
251
+ pending.handle(messageResponseMessage);
252
+ }
253
+ cleanup() {
254
+ this.rejectAllPendingRequests();
255
+ this.stopListening();
256
+ this.stopListeningOnClose();
257
+ }
258
+ rejectAllPendingRequests() {
259
+ for (const [, pending] of this.pendingRequests) {
260
+ clearTimeout(pending.timeout);
261
+ pending.reject(/* @__PURE__ */ new Error("Pending requests rejected"));
262
+ }
263
+ this.pendingRequests.clear();
264
+ }
265
+ };
266
+
267
+ //#endregion
268
+ //#region src/core/BananalinkConnection.ts
269
+ var BananalinkConnection = class BananalinkConnection {
270
+ static AUTH_TIMEOUT_MILLIS = 600 * 1e3;
271
+ static BIND_RESPONSE_TIMEOUT_MILLIS = 20 * 1e3;
272
+ apiUrl;
273
+ wsUrl;
274
+ jwt;
275
+ agentPairingCode;
276
+ pendingSessionPromise = null;
277
+ pendingSessionAbortController = null;
278
+ constructor(opts) {
279
+ this.apiUrl = opts.apiUrl;
280
+ this.wsUrl = opts.wsUrl;
281
+ this.jwt = opts.jwt;
282
+ this.agentPairingCode = opts.agentPairingCode;
283
+ if (!this.agentPairingCode && !this.jwt) throw new InvalidConnectionStateError("Cannot get session without authorizing agent or providing a valid JWT");
284
+ }
285
+ get connectionUrl() {
286
+ if (!this.agentPairingCode) throw new InvalidConnectionStateError("Cannot get connection URL without a pairing code");
287
+ return `bananalink://?code=${encodeURIComponent(this.agentPairingCode)}`;
288
+ }
289
+ abortPendingSession() {
290
+ this.pendingSessionAbortController?.abort();
291
+ }
292
+ async getSession() {
293
+ if (this.pendingSessionPromise) return this.pendingSessionPromise;
294
+ this.pendingSessionAbortController = new AbortController();
295
+ this.pendingSessionPromise = (this.jwt ? this.resumeSession(this.jwt, this.pendingSessionAbortController.signal) : this.waitForAuthorization(this.pendingSessionAbortController.signal)).finally(() => {
296
+ this.pendingSessionPromise = null;
297
+ this.pendingSessionAbortController = null;
298
+ });
299
+ return this.pendingSessionPromise;
300
+ }
301
+ async openSession(getSessionClaims, abortSignal) {
302
+ const transportHandle = await this.getTransportHandle(this.agentPairingCode);
303
+ const onAbort = () => transportHandle.close();
304
+ abortSignal.addEventListener("abort", onAbort, { once: true });
305
+ if (abortSignal.aborted) {
306
+ transportHandle.close();
307
+ throw new PendingSessionAbortedError();
308
+ }
309
+ try {
310
+ const sessionClaims = await getSessionClaims(transportHandle);
311
+ await this.bind(transportHandle, sessionClaims.accessToken, abortSignal);
312
+ return new BananalinkSession(sessionClaims, createReconnectingTransport(transportHandle, {
313
+ reconnectTransport: async () => this.getTransportHandle(null),
314
+ onReconnect: async (th) => this.bind(th, sessionClaims.accessToken),
315
+ maxReconnectAttempts: 5,
316
+ baseDelayReconnectMs: 1e3
317
+ }), this.apiUrl);
318
+ } catch (error) {
319
+ transportHandle.close();
320
+ throw error;
321
+ } finally {
322
+ abortSignal.removeEventListener("abort", onAbort);
323
+ }
324
+ }
325
+ async resumeSession(jwt, abortSignal) {
326
+ return await this.openSession(async () => ({
327
+ ...jwt.jwtPayload,
328
+ accessToken: jwt.accessToken
329
+ }), abortSignal);
330
+ }
331
+ async waitForAuthorization(abortSignal) {
332
+ return await this.openSession(async (transportHandle) => await waitForTransportEvent(transportHandle, {
333
+ onMessage: (payload, resolve, reject) => {
334
+ if (isAuthorizedMessage(payload)) resolve(payload);
335
+ if (isRejectedMessage(payload)) reject(new ConnectionRejectedError());
336
+ },
337
+ timeoutMs: BananalinkConnection.AUTH_TIMEOUT_MILLIS,
338
+ createTimeoutError: () => new ConnectionTimeoutError(),
339
+ createClosedError: () => new BananalinkError("Connection closed unexpectedly"),
340
+ abortSignal,
341
+ createAbortError: () => new PendingSessionAbortedError()
342
+ }), abortSignal);
343
+ }
344
+ async getTransportHandle(agentPairingCode) {
345
+ const url = `${this.wsUrl}${agentPairingCode != null ? `?code=${encodeURIComponent(agentPairingCode)}` : ""}`;
346
+ try {
347
+ return await connectWebSocket({
348
+ url,
349
+ pingIntervalMs: 3e4,
350
+ pongTimeoutMs: 5e3,
351
+ maxReconnectAttempts: 0
352
+ });
353
+ } catch (error) {
354
+ throw new BananalinkError("Unable to establish agent websocket connection", { cause: error });
355
+ }
356
+ }
357
+ async bind(ws, accessToken, abortSignal) {
358
+ try {
359
+ await bindTransport(ws, accessToken, {
360
+ timeoutMs: BananalinkConnection.BIND_RESPONSE_TIMEOUT_MILLIS,
361
+ createTimeoutError: () => new ConnectionTimeoutError(),
362
+ createClosedError: () => new BananalinkError("Connection closed unexpectedly"),
363
+ abortSignal,
364
+ createAbortError: () => new PendingSessionAbortedError()
365
+ });
366
+ } catch (error) {
367
+ if (error instanceof BindTransportError) throw new InvalidJwtError();
368
+ throw error;
369
+ }
370
+ }
371
+ };
372
+
373
+ //#endregion
374
+ //#region src/core/BananalinkAgent.ts
375
+ var BananalinkAgent = class BananalinkAgent {
376
+ apiUrl;
377
+ wsUrl;
378
+ jwks;
379
+ agent;
380
+ static DEFAULT_BANANALINK_API_URL = "https://api.dev.banana.link";
381
+ static DEFAULT_BANANALINK_WS_URL = "wss://ws.dev.banana.link";
382
+ constructor(config) {
383
+ this.apiUrl = config.apiUrl ?? BananalinkAgent.DEFAULT_BANANALINK_API_URL;
384
+ this.wsUrl = config.wsUrl ?? BananalinkAgent.DEFAULT_BANANALINK_WS_URL;
385
+ if (!isValidUrl(this.apiUrl, ["http", "https"])) throw new InvalidConfigError(`Invalid apiUrl: "${this.apiUrl}". Must use http or https.`);
386
+ if (!isValidUrl(this.wsUrl, ["ws", "wss"])) throw new InvalidConfigError(`Invalid wsUrl: "${this.wsUrl}". Must use ws or wss.`);
387
+ this.agent = config.agent;
388
+ this.jwks = createBananalinkJwks(this.apiUrl);
389
+ }
390
+ async connect(opts) {
391
+ const accessToken = opts?.accessToken;
392
+ if (!accessToken) {
393
+ const { pairingCode } = await this.pairAgent();
394
+ return new BananalinkConnection({
395
+ jwt: void 0,
396
+ agentPairingCode: pairingCode,
397
+ apiUrl: this.apiUrl,
398
+ wsUrl: this.wsUrl
399
+ });
400
+ }
401
+ const jwtPayload = await this.getJwtPayload(accessToken);
402
+ if (!jwtPayload) throw new InvalidJwtError();
403
+ return new BananalinkConnection({
404
+ jwt: {
405
+ jwtPayload,
406
+ accessToken
407
+ },
408
+ agentPairingCode: null,
409
+ apiUrl: this.apiUrl,
410
+ wsUrl: this.wsUrl
411
+ });
412
+ }
413
+ async getJwtPayload(accessToken) {
414
+ return await verifyJwt(accessToken, this.jwks);
415
+ }
416
+ async pairAgent() {
417
+ return await createAgentPairingRequest(this.apiUrl, this.agent);
418
+ }
419
+ };
420
+
421
+ //#endregion
422
+ //#region src/errors/InvalidUrlError.ts
423
+ var InvalidUrlError = class extends BananalinkError {
424
+ constructor(url) {
425
+ super(`${url} is not a valid url`);
426
+ this.url = url;
427
+ }
428
+ };
429
+
430
+ //#endregion
431
+ //#region src/utils/displayBananalinkQR.ts
432
+ async function displayBananalinkQR(url) {
433
+ if (!isValidUrl(url, ["bananalink"])) throw new InvalidUrlError(url);
434
+ return QRCode.toDataURL(url);
435
+ }
436
+
437
+ //#endregion
438
+ export { AgentApiError, BananalinkAgent, BananalinkConnection, BananalinkError, BananalinkSession, ConnectionRejectedError, ConnectionTimeoutError, InvalidConfigError, InvalidConnectionStateError, InvalidJwtError, PendingSessionAbortedError, RequestRejectedError, RequestTimeoutError, SessionClosedError, displayBananalinkQR };
439
+ //# sourceMappingURL=index.mjs.map