@contractspec/integration.runtime 1.57.0 → 1.58.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.
Files changed (39) hide show
  1. package/dist/health.d.ts +14 -18
  2. package/dist/health.d.ts.map +1 -1
  3. package/dist/health.js +71 -68
  4. package/dist/index.d.ts +4 -8
  5. package/dist/index.d.ts.map +1 -0
  6. package/dist/index.js +828 -9
  7. package/dist/node/health.js +72 -0
  8. package/dist/node/index.js +827 -0
  9. package/dist/node/runtime.js +208 -0
  10. package/dist/node/secrets/env-secret-provider.js +158 -0
  11. package/dist/node/secrets/gcp-secret-manager.js +346 -0
  12. package/dist/node/secrets/index.js +549 -0
  13. package/dist/node/secrets/manager.js +182 -0
  14. package/dist/node/secrets/provider.js +73 -0
  15. package/dist/runtime.d.ts +86 -90
  16. package/dist/runtime.d.ts.map +1 -1
  17. package/dist/runtime.js +204 -181
  18. package/dist/secrets/env-secret-provider.d.ts +20 -23
  19. package/dist/secrets/env-secret-provider.d.ts.map +1 -1
  20. package/dist/secrets/env-secret-provider.js +157 -80
  21. package/dist/secrets/gcp-secret-manager.d.ts +25 -28
  22. package/dist/secrets/gcp-secret-manager.d.ts.map +1 -1
  23. package/dist/secrets/gcp-secret-manager.js +339 -222
  24. package/dist/secrets/index.d.ts +5 -5
  25. package/dist/secrets/index.d.ts.map +1 -0
  26. package/dist/secrets/index.js +549 -5
  27. package/dist/secrets/manager.d.ts +32 -35
  28. package/dist/secrets/manager.d.ts.map +1 -1
  29. package/dist/secrets/manager.js +180 -101
  30. package/dist/secrets/provider.d.ts +42 -45
  31. package/dist/secrets/provider.d.ts.map +1 -1
  32. package/dist/secrets/provider.js +69 -54
  33. package/package.json +76 -30
  34. package/dist/health.js.map +0 -1
  35. package/dist/runtime.js.map +0 -1
  36. package/dist/secrets/env-secret-provider.js.map +0 -1
  37. package/dist/secrets/gcp-secret-manager.js.map +0 -1
  38. package/dist/secrets/manager.js.map +0 -1
  39. package/dist/secrets/provider.js.map +0 -1
package/dist/index.js CHANGED
@@ -1,9 +1,828 @@
1
- import { IntegrationHealthService } from "./health.js";
2
- import { IntegrationCallGuard, connectionStatusLabel, ensureConnectionReady } from "./runtime.js";
3
- import { SecretProviderError, normalizeSecretPayload, parseSecretUri } from "./secrets/provider.js";
4
- import { GcpSecretManagerProvider } from "./secrets/gcp-secret-manager.js";
5
- import { EnvSecretProvider } from "./secrets/env-secret-provider.js";
6
- import { SecretProviderManager } from "./secrets/manager.js";
7
- import "./secrets/index.js";
8
-
9
- export { EnvSecretProvider, GcpSecretManagerProvider, IntegrationCallGuard, IntegrationHealthService, SecretProviderError, SecretProviderManager, connectionStatusLabel, ensureConnectionReady, normalizeSecretPayload, parseSecretUri };
1
+ // @bun
2
+ // src/health.ts
3
+ class IntegrationHealthService {
4
+ telemetry;
5
+ nowFn;
6
+ constructor(options = {}) {
7
+ this.telemetry = options.telemetry;
8
+ this.nowFn = options.now ?? (() => new Date);
9
+ }
10
+ async check(context, executor) {
11
+ const start = this.nowFn();
12
+ try {
13
+ await executor(context);
14
+ const end = this.nowFn();
15
+ const result = {
16
+ status: "connected",
17
+ checkedAt: end,
18
+ latencyMs: end.getTime() - start.getTime()
19
+ };
20
+ this.emitTelemetry(context, result, "success");
21
+ return result;
22
+ } catch (error) {
23
+ const end = this.nowFn();
24
+ const message = error instanceof Error ? error.message : "Unknown error";
25
+ const code = extractErrorCode(error);
26
+ const result = {
27
+ status: "error",
28
+ checkedAt: end,
29
+ latencyMs: end.getTime() - start.getTime(),
30
+ errorMessage: message,
31
+ errorCode: code
32
+ };
33
+ this.emitTelemetry(context, result, "error", code, message);
34
+ return result;
35
+ }
36
+ }
37
+ emitTelemetry(context, result, status, errorCode, errorMessage) {
38
+ if (!this.telemetry)
39
+ return;
40
+ this.telemetry.record({
41
+ tenantId: context.tenantId,
42
+ appId: context.appId,
43
+ environment: context.environment,
44
+ slotId: context.slotId,
45
+ integrationKey: context.spec.meta.key,
46
+ integrationVersion: context.spec.meta.version,
47
+ connectionId: context.connection.meta.id,
48
+ status,
49
+ durationMs: result.latencyMs,
50
+ errorCode,
51
+ errorMessage,
52
+ occurredAt: result.checkedAt ?? this.nowFn(),
53
+ metadata: {
54
+ ...context.trace ? {
55
+ blueprint: `${context.trace.blueprintName}.v${context.trace.blueprintVersion}`,
56
+ configVersion: context.trace.configVersion
57
+ } : {},
58
+ status: result.status
59
+ }
60
+ });
61
+ }
62
+ }
63
+ function extractErrorCode(error) {
64
+ if (!error || typeof error !== "object")
65
+ return;
66
+ const candidate = error;
67
+ if (candidate.code == null)
68
+ return;
69
+ return String(candidate.code);
70
+ }
71
+
72
+ // src/runtime.ts
73
+ import { performance } from "perf_hooks";
74
+ var DEFAULT_MAX_ATTEMPTS = 3;
75
+ var DEFAULT_BACKOFF_MS = 250;
76
+
77
+ class IntegrationCallGuard {
78
+ secretProvider;
79
+ telemetry;
80
+ maxAttempts;
81
+ backoffMs;
82
+ shouldRetry;
83
+ sleep;
84
+ now;
85
+ constructor(secretProvider, options = {}) {
86
+ this.secretProvider = secretProvider;
87
+ this.telemetry = options.telemetry;
88
+ this.maxAttempts = Math.max(1, options.maxAttempts ?? DEFAULT_MAX_ATTEMPTS);
89
+ this.backoffMs = options.backoffMs ?? DEFAULT_BACKOFF_MS;
90
+ this.shouldRetry = options.shouldRetry ?? ((error) => typeof error === "object" && error !== null && ("retryable" in error) && Boolean(error.retryable));
91
+ this.sleep = options.sleep ?? ((ms) => ms <= 0 ? Promise.resolve() : new Promise((resolve) => setTimeout(resolve, ms)));
92
+ this.now = options.now ?? (() => new Date);
93
+ }
94
+ async executeWithGuards(slotId, operation, _input, resolvedConfig, executor) {
95
+ const integration = this.findIntegration(slotId, resolvedConfig);
96
+ if (!integration) {
97
+ return this.failure({
98
+ tenantId: resolvedConfig.tenantId,
99
+ appId: resolvedConfig.appId,
100
+ environment: resolvedConfig.environment,
101
+ blueprintName: resolvedConfig.blueprintName,
102
+ blueprintVersion: resolvedConfig.blueprintVersion,
103
+ configVersion: resolvedConfig.configVersion,
104
+ slotId,
105
+ operation
106
+ }, undefined, {
107
+ code: "SLOT_NOT_BOUND",
108
+ message: `Integration slot "${slotId}" is not bound for tenant "${resolvedConfig.tenantId}".`,
109
+ retryable: false
110
+ }, 0);
111
+ }
112
+ const status = integration.connection.status;
113
+ if (status === "disconnected" || status === "error") {
114
+ return this.failure(this.makeContext(slotId, operation, resolvedConfig), integration, {
115
+ code: "CONNECTION_NOT_READY",
116
+ message: `Integration connection "${integration.connection.meta.label}" is in status "${status}".`,
117
+ retryable: false
118
+ }, 0);
119
+ }
120
+ const secrets = await this.fetchSecrets(integration.connection);
121
+ let attempt = 0;
122
+ const started = performance.now();
123
+ while (attempt < this.maxAttempts) {
124
+ attempt += 1;
125
+ try {
126
+ const data = await executor(integration.connection, secrets);
127
+ const duration = performance.now() - started;
128
+ this.emitTelemetry(this.makeContext(slotId, operation, resolvedConfig), integration, "success", duration);
129
+ return {
130
+ success: true,
131
+ data,
132
+ metadata: {
133
+ latencyMs: duration,
134
+ connectionId: integration.connection.meta.id,
135
+ ownershipMode: integration.connection.ownershipMode,
136
+ attempts: attempt
137
+ }
138
+ };
139
+ } catch (error) {
140
+ const duration = performance.now() - started;
141
+ this.emitTelemetry(this.makeContext(slotId, operation, resolvedConfig), integration, "error", duration, this.errorCodeFor(error), error instanceof Error ? error.message : String(error));
142
+ const retryable = this.shouldRetry(error, attempt);
143
+ if (!retryable || attempt >= this.maxAttempts) {
144
+ return {
145
+ success: false,
146
+ error: {
147
+ code: this.errorCodeFor(error),
148
+ message: error instanceof Error ? error.message : String(error),
149
+ retryable,
150
+ cause: error
151
+ },
152
+ metadata: {
153
+ latencyMs: duration,
154
+ connectionId: integration.connection.meta.id,
155
+ ownershipMode: integration.connection.ownershipMode,
156
+ attempts: attempt
157
+ }
158
+ };
159
+ }
160
+ await this.sleep(this.backoffMs);
161
+ }
162
+ }
163
+ return {
164
+ success: false,
165
+ error: {
166
+ code: "UNKNOWN_ERROR",
167
+ message: "Integration call failed after retries.",
168
+ retryable: false
169
+ },
170
+ metadata: {
171
+ latencyMs: performance.now() - started,
172
+ connectionId: integration.connection.meta.id,
173
+ ownershipMode: integration.connection.ownershipMode,
174
+ attempts: this.maxAttempts
175
+ }
176
+ };
177
+ }
178
+ findIntegration(slotId, config) {
179
+ return config.integrations.find((integration) => integration.slot.slotId === slotId);
180
+ }
181
+ async fetchSecrets(connection) {
182
+ if (!this.secretProvider.canHandle(connection.secretRef)) {
183
+ throw new Error(`Secret provider "${this.secretProvider.id}" cannot handle reference "${connection.secretRef}".`);
184
+ }
185
+ const secret = await this.secretProvider.getSecret(connection.secretRef);
186
+ return this.parseSecret(secret);
187
+ }
188
+ parseSecret(secret) {
189
+ const text = new TextDecoder().decode(secret.data);
190
+ try {
191
+ const parsed = JSON.parse(text);
192
+ if (parsed && typeof parsed === "object" && !Array.isArray(parsed)) {
193
+ const entries = Object.entries(parsed).filter(([, value]) => typeof value === "string" || typeof value === "number" || typeof value === "boolean");
194
+ return Object.fromEntries(entries.map(([key, value]) => [key, String(value)]));
195
+ }
196
+ } catch {}
197
+ return { secret: text };
198
+ }
199
+ emitTelemetry(context, integration, status, durationMs, errorCode, errorMessage) {
200
+ if (!this.telemetry || !integration)
201
+ return;
202
+ this.telemetry.record({
203
+ tenantId: context.tenantId,
204
+ appId: context.appId,
205
+ environment: context.environment,
206
+ slotId: context.slotId,
207
+ integrationKey: integration.connection.meta.integrationKey,
208
+ integrationVersion: integration.connection.meta.integrationVersion,
209
+ connectionId: integration.connection.meta.id,
210
+ status,
211
+ durationMs,
212
+ errorCode,
213
+ errorMessage,
214
+ occurredAt: this.now(),
215
+ metadata: {
216
+ blueprint: `${context.blueprintName}.v${context.blueprintVersion}`,
217
+ configVersion: context.configVersion,
218
+ operation: context.operation
219
+ }
220
+ });
221
+ }
222
+ failure(context, integration, error, attempts) {
223
+ if (integration) {
224
+ this.emitTelemetry(context, integration, "error", 0, error.code, error.message);
225
+ }
226
+ return {
227
+ success: false,
228
+ error,
229
+ metadata: {
230
+ latencyMs: 0,
231
+ connectionId: integration?.connection.meta.id ?? "unknown",
232
+ ownershipMode: integration?.connection.ownershipMode ?? "managed",
233
+ attempts
234
+ }
235
+ };
236
+ }
237
+ makeContext(slotId, operation, config) {
238
+ return {
239
+ tenantId: config.tenantId,
240
+ appId: config.appId,
241
+ environment: config.environment,
242
+ blueprintName: config.blueprintName,
243
+ blueprintVersion: config.blueprintVersion,
244
+ configVersion: config.configVersion,
245
+ slotId,
246
+ operation
247
+ };
248
+ }
249
+ errorCodeFor(error) {
250
+ if (typeof error === "object" && error !== null && "code" in error && typeof error.code === "string") {
251
+ return error.code;
252
+ }
253
+ return "PROVIDER_ERROR";
254
+ }
255
+ }
256
+ function ensureConnectionReady(integration) {
257
+ const status = integration.connection.status;
258
+ if (status === "disconnected" || status === "error") {
259
+ throw new Error(`Integration connection "${integration.connection.meta.label}" is in status "${status}".`);
260
+ }
261
+ }
262
+ function connectionStatusLabel(status) {
263
+ switch (status) {
264
+ case "connected":
265
+ return "connected";
266
+ case "disconnected":
267
+ return "disconnected";
268
+ case "error":
269
+ return "error";
270
+ case "unknown":
271
+ default:
272
+ return "unknown";
273
+ }
274
+ }
275
+
276
+ // src/secrets/provider.ts
277
+ import { Buffer as Buffer2 } from "buffer";
278
+
279
+ class SecretProviderError extends Error {
280
+ provider;
281
+ reference;
282
+ code;
283
+ cause;
284
+ constructor(params) {
285
+ super(params.message);
286
+ this.name = "SecretProviderError";
287
+ this.provider = params.provider;
288
+ this.reference = params.reference;
289
+ this.code = params.code ?? "UNKNOWN";
290
+ this.cause = params.cause;
291
+ }
292
+ }
293
+ function parseSecretUri(reference) {
294
+ if (!reference) {
295
+ throw new SecretProviderError({
296
+ message: "Secret reference cannot be empty",
297
+ provider: "unknown",
298
+ reference,
299
+ code: "INVALID"
300
+ });
301
+ }
302
+ const [scheme, rest] = reference.split("://");
303
+ if (!scheme || !rest) {
304
+ throw new SecretProviderError({
305
+ message: `Invalid secret reference: ${reference}`,
306
+ provider: "unknown",
307
+ reference,
308
+ code: "INVALID"
309
+ });
310
+ }
311
+ const queryIndex = rest.indexOf("?");
312
+ if (queryIndex === -1) {
313
+ return {
314
+ provider: scheme,
315
+ path: rest
316
+ };
317
+ }
318
+ const path = rest.slice(0, queryIndex);
319
+ const query = rest.slice(queryIndex + 1);
320
+ const extras = Object.fromEntries(query.split("&").filter(Boolean).map((pair) => {
321
+ const [keyRaw, valueRaw] = pair.split("=");
322
+ const key = keyRaw ?? "";
323
+ const value = valueRaw ?? "";
324
+ return [decodeURIComponent(key), decodeURIComponent(value)];
325
+ }));
326
+ return {
327
+ provider: scheme,
328
+ path,
329
+ extras
330
+ };
331
+ }
332
+ function normalizeSecretPayload(payload) {
333
+ if (payload.data instanceof Uint8Array) {
334
+ return payload.data;
335
+ }
336
+ if (payload.encoding === "base64") {
337
+ return Buffer2.from(payload.data, "base64");
338
+ }
339
+ if (payload.encoding === "binary") {
340
+ return Buffer2.from(payload.data, "binary");
341
+ }
342
+ return Buffer2.from(payload.data, "utf-8");
343
+ }
344
+
345
+ // src/secrets/gcp-secret-manager.ts
346
+ import {
347
+ SecretManagerServiceClient
348
+ } from "@google-cloud/secret-manager";
349
+ var DEFAULT_REPLICATION = {
350
+ automatic: {}
351
+ };
352
+
353
+ class GcpSecretManagerProvider {
354
+ id = "gcp-secret-manager";
355
+ client;
356
+ explicitProjectId;
357
+ replication;
358
+ constructor(options = {}) {
359
+ this.client = options.client ?? new SecretManagerServiceClient(options.clientOptions ?? {});
360
+ this.explicitProjectId = options.projectId;
361
+ this.replication = options.defaultReplication ?? DEFAULT_REPLICATION;
362
+ }
363
+ canHandle(reference) {
364
+ try {
365
+ const parsed = parseSecretUri(reference);
366
+ return parsed.provider === "gcp";
367
+ } catch {
368
+ return false;
369
+ }
370
+ }
371
+ async getSecret(reference, options, callOptions) {
372
+ const location = this.parseReference(reference);
373
+ const secretVersionName = this.buildVersionName(location, options?.version);
374
+ try {
375
+ const response = await this.client.accessSecretVersion({
376
+ name: secretVersionName
377
+ }, callOptions ?? {});
378
+ const [result] = response;
379
+ const payload = result.payload;
380
+ if (!payload?.data) {
381
+ throw new SecretProviderError({
382
+ message: `Secret payload empty for ${secretVersionName}`,
383
+ provider: this.id,
384
+ reference,
385
+ code: "UNKNOWN"
386
+ });
387
+ }
388
+ const version = extractVersionFromName(result.name ?? secretVersionName);
389
+ return {
390
+ data: payload.data,
391
+ version,
392
+ metadata: payload.dataCrc32c ? { crc32c: payload.dataCrc32c.toString() } : undefined,
393
+ retrievedAt: new Date
394
+ };
395
+ } catch (error) {
396
+ throw toSecretProviderError({
397
+ error,
398
+ provider: this.id,
399
+ reference,
400
+ operation: "access"
401
+ });
402
+ }
403
+ }
404
+ async setSecret(reference, payload) {
405
+ const location = this.parseReference(reference);
406
+ const { secretName } = this.buildNames(location);
407
+ const data = normalizeSecretPayload(payload);
408
+ await this.ensureSecretExists(location, payload);
409
+ try {
410
+ const response = await this.client.addSecretVersion({
411
+ parent: secretName,
412
+ payload: {
413
+ data
414
+ }
415
+ });
416
+ if (!response) {
417
+ throw new SecretProviderError({
418
+ message: `No version returned when adding secret version for ${secretName}`,
419
+ provider: this.id,
420
+ reference,
421
+ code: "UNKNOWN"
422
+ });
423
+ }
424
+ const [version] = response;
425
+ const versionName = version?.name ?? `${secretName}/versions/latest`;
426
+ return {
427
+ reference: `gcp://${versionName}`,
428
+ version: extractVersionFromName(versionName) ?? "latest"
429
+ };
430
+ } catch (error) {
431
+ throw toSecretProviderError({
432
+ error,
433
+ provider: this.id,
434
+ reference,
435
+ operation: "addSecretVersion"
436
+ });
437
+ }
438
+ }
439
+ async rotateSecret(reference, payload) {
440
+ return this.setSecret(reference, payload);
441
+ }
442
+ async deleteSecret(reference) {
443
+ const location = this.parseReference(reference);
444
+ const { secretName } = this.buildNames(location);
445
+ try {
446
+ await this.client.deleteSecret({
447
+ name: secretName
448
+ });
449
+ } catch (error) {
450
+ throw toSecretProviderError({
451
+ error,
452
+ provider: this.id,
453
+ reference,
454
+ operation: "delete"
455
+ });
456
+ }
457
+ }
458
+ parseReference(reference) {
459
+ const parsed = parseSecretUri(reference);
460
+ if (parsed.provider !== "gcp") {
461
+ throw new SecretProviderError({
462
+ message: `Unsupported secret provider: ${parsed.provider}`,
463
+ provider: this.id,
464
+ reference,
465
+ code: "INVALID"
466
+ });
467
+ }
468
+ const segments = parsed.path.split("/").filter(Boolean);
469
+ if (segments.length < 4 || segments[0] !== "projects") {
470
+ throw new SecretProviderError({
471
+ message: `Expected secret reference format gcp://projects/{project}/secrets/{secret}[(/versions/{version})] but received "${parsed.path}"`,
472
+ provider: this.id,
473
+ reference,
474
+ code: "INVALID"
475
+ });
476
+ }
477
+ const projectIdCandidate = segments[1] ?? this.explicitProjectId;
478
+ if (!projectIdCandidate) {
479
+ throw new SecretProviderError({
480
+ message: `Unable to resolve project or secret from reference "${parsed.path}"`,
481
+ provider: this.id,
482
+ reference,
483
+ code: "INVALID"
484
+ });
485
+ }
486
+ const indexOfSecrets = segments.indexOf("secrets");
487
+ if (indexOfSecrets === -1 || indexOfSecrets + 1 >= segments.length) {
488
+ throw new SecretProviderError({
489
+ message: `Unable to resolve project or secret from reference "${parsed.path}"`,
490
+ provider: this.id,
491
+ reference,
492
+ code: "INVALID"
493
+ });
494
+ }
495
+ const resolvedProjectId = projectIdCandidate;
496
+ const secretIdCandidate = segments[indexOfSecrets + 1];
497
+ if (!secretIdCandidate) {
498
+ throw new SecretProviderError({
499
+ message: `Unable to resolve secret ID from reference "${parsed.path}"`,
500
+ provider: this.id,
501
+ reference,
502
+ code: "INVALID"
503
+ });
504
+ }
505
+ const secretId = secretIdCandidate;
506
+ const indexOfVersions = segments.indexOf("versions");
507
+ const version = parsed.extras?.version ?? (indexOfVersions !== -1 && indexOfVersions + 1 < segments.length ? segments[indexOfVersions + 1] : undefined);
508
+ return {
509
+ projectId: resolvedProjectId,
510
+ secretId,
511
+ version
512
+ };
513
+ }
514
+ buildNames(location) {
515
+ const projectId = location.projectId ?? this.explicitProjectId;
516
+ if (!projectId) {
517
+ throw new SecretProviderError({
518
+ message: "Project ID must be provided either in reference or provider configuration",
519
+ provider: this.id,
520
+ reference: `gcp://projects//secrets/${location.secretId}`,
521
+ code: "INVALID"
522
+ });
523
+ }
524
+ const projectParent = `projects/${projectId}`;
525
+ const secretName = `${projectParent}/secrets/${location.secretId}`;
526
+ return {
527
+ projectParent,
528
+ secretName
529
+ };
530
+ }
531
+ buildVersionName(location, explicitVersion) {
532
+ const { secretName } = this.buildNames(location);
533
+ const version = explicitVersion ?? location.version ?? "latest";
534
+ return `${secretName}/versions/${version}`;
535
+ }
536
+ async ensureSecretExists(location, payload) {
537
+ const { secretName, projectParent } = this.buildNames(location);
538
+ try {
539
+ await this.client.getSecret({ name: secretName });
540
+ } catch (error) {
541
+ const providerError = toSecretProviderError({
542
+ error,
543
+ provider: this.id,
544
+ reference: `gcp://${secretName}`,
545
+ operation: "getSecret",
546
+ suppressThrow: true
547
+ });
548
+ if (!providerError || providerError.code !== "NOT_FOUND") {
549
+ if (providerError) {
550
+ throw providerError;
551
+ }
552
+ throw error;
553
+ }
554
+ try {
555
+ await this.client.createSecret({
556
+ parent: projectParent,
557
+ secretId: location.secretId,
558
+ secret: {
559
+ replication: this.replication,
560
+ labels: payload.labels
561
+ }
562
+ });
563
+ } catch (creationError) {
564
+ const creationProviderError = toSecretProviderError({
565
+ error: creationError,
566
+ provider: this.id,
567
+ reference: `gcp://${secretName}`,
568
+ operation: "createSecret"
569
+ });
570
+ throw creationProviderError;
571
+ }
572
+ }
573
+ }
574
+ }
575
+ function extractVersionFromName(name) {
576
+ const segments = name.split("/").filter(Boolean);
577
+ const index = segments.indexOf("versions");
578
+ if (index === -1 || index + 1 >= segments.length) {
579
+ return;
580
+ }
581
+ return segments[index + 1];
582
+ }
583
+ function toSecretProviderError(params) {
584
+ const { error, provider, reference, operation, suppressThrow } = params;
585
+ if (error instanceof SecretProviderError) {
586
+ return error;
587
+ }
588
+ const code = deriveErrorCode(error);
589
+ const message = error instanceof Error ? error.message : `Unknown error during ${operation}`;
590
+ const providerError = new SecretProviderError({
591
+ message,
592
+ provider,
593
+ reference,
594
+ code,
595
+ cause: error
596
+ });
597
+ if (suppressThrow) {
598
+ return providerError;
599
+ }
600
+ throw providerError;
601
+ }
602
+ function deriveErrorCode(error) {
603
+ if (typeof error !== "object" || error === null) {
604
+ return "UNKNOWN";
605
+ }
606
+ const errorAny = error;
607
+ const code = errorAny.code;
608
+ if (code === 5 || code === "NOT_FOUND")
609
+ return "NOT_FOUND";
610
+ if (code === 6 || code === "ALREADY_EXISTS")
611
+ return "INVALID";
612
+ if (code === 7 || code === "PERMISSION_DENIED" || code === 403) {
613
+ return "FORBIDDEN";
614
+ }
615
+ if (code === 3 || code === "INVALID_ARGUMENT")
616
+ return "INVALID";
617
+ return "UNKNOWN";
618
+ }
619
+
620
+ // src/secrets/env-secret-provider.ts
621
+ class EnvSecretProvider {
622
+ id = "env";
623
+ aliases;
624
+ constructor(options = {}) {
625
+ this.aliases = options.aliases ?? {};
626
+ }
627
+ canHandle(reference) {
628
+ const envKey = this.resolveEnvKey(reference);
629
+ return envKey !== undefined && process.env[envKey] !== undefined;
630
+ }
631
+ async getSecret(reference) {
632
+ const envKey = this.resolveEnvKey(reference);
633
+ if (!envKey) {
634
+ throw new SecretProviderError({
635
+ message: `Unable to resolve environment variable for reference "${reference}".`,
636
+ provider: this.id,
637
+ reference,
638
+ code: "INVALID"
639
+ });
640
+ }
641
+ const value = process.env[envKey];
642
+ if (value === undefined) {
643
+ throw new SecretProviderError({
644
+ message: `Environment variable "${envKey}" not found for reference "${reference}".`,
645
+ provider: this.id,
646
+ reference,
647
+ code: "NOT_FOUND"
648
+ });
649
+ }
650
+ return {
651
+ data: Buffer.from(value, "utf-8"),
652
+ version: "current",
653
+ metadata: {
654
+ source: "env",
655
+ envKey
656
+ },
657
+ retrievedAt: new Date
658
+ };
659
+ }
660
+ async setSecret(reference, _payload) {
661
+ throw this.forbiddenError("setSecret", reference);
662
+ }
663
+ async rotateSecret(reference, _payload) {
664
+ throw this.forbiddenError("rotateSecret", reference);
665
+ }
666
+ async deleteSecret(reference) {
667
+ throw this.forbiddenError("deleteSecret", reference);
668
+ }
669
+ resolveEnvKey(reference) {
670
+ if (!reference) {
671
+ return;
672
+ }
673
+ if (this.aliases[reference]) {
674
+ return this.aliases[reference];
675
+ }
676
+ if (!reference.includes("://")) {
677
+ return reference;
678
+ }
679
+ try {
680
+ const parsed = parseSecretUri(reference);
681
+ if (parsed.provider === "env") {
682
+ return parsed.path;
683
+ }
684
+ if (parsed.extras?.env) {
685
+ return parsed.extras.env;
686
+ }
687
+ return this.deriveEnvKey(parsed.path);
688
+ } catch {
689
+ return reference;
690
+ }
691
+ }
692
+ deriveEnvKey(path) {
693
+ if (!path)
694
+ return;
695
+ return path.split(/[/:\-.]/).filter(Boolean).map((segment) => segment.replace(/[^a-zA-Z0-9]/g, "_").replace(/_{2,}/g, "_").toUpperCase()).join("_");
696
+ }
697
+ forbiddenError(operation, reference) {
698
+ return new SecretProviderError({
699
+ message: `EnvSecretProvider is read-only. "${operation}" is not allowed for ${reference}.`,
700
+ provider: this.id,
701
+ reference,
702
+ code: "FORBIDDEN"
703
+ });
704
+ }
705
+ }
706
+
707
+ // src/secrets/manager.ts
708
+ class SecretProviderManager {
709
+ id;
710
+ providers = [];
711
+ registrationCounter = 0;
712
+ constructor(options = {}) {
713
+ this.id = options.id ?? "secret-provider-manager";
714
+ const initialProviders = options.providers ?? [];
715
+ for (const entry of initialProviders) {
716
+ this.register(entry.provider, { priority: entry.priority });
717
+ }
718
+ }
719
+ register(provider, options = {}) {
720
+ this.providers.push({
721
+ provider,
722
+ priority: options.priority ?? 0,
723
+ order: this.registrationCounter++
724
+ });
725
+ this.providers.sort((a, b) => {
726
+ if (a.priority !== b.priority) {
727
+ return b.priority - a.priority;
728
+ }
729
+ return a.order - b.order;
730
+ });
731
+ return this;
732
+ }
733
+ canHandle(reference) {
734
+ return this.providers.some(({ provider }) => safeCanHandle(provider, reference));
735
+ }
736
+ async getSecret(reference, options) {
737
+ const errors = [];
738
+ for (const { provider } of this.providers) {
739
+ if (!safeCanHandle(provider, reference)) {
740
+ continue;
741
+ }
742
+ try {
743
+ return await provider.getSecret(reference, options);
744
+ } catch (error) {
745
+ if (error instanceof SecretProviderError) {
746
+ errors.push(error);
747
+ if (error.code !== "NOT_FOUND") {
748
+ break;
749
+ }
750
+ continue;
751
+ }
752
+ throw error;
753
+ }
754
+ }
755
+ throw this.composeError("getSecret", reference, errors, options?.version);
756
+ }
757
+ async setSecret(reference, payload) {
758
+ return this.delegateToFirst("setSecret", reference, (provider) => provider.setSecret(reference, payload));
759
+ }
760
+ async rotateSecret(reference, payload) {
761
+ return this.delegateToFirst("rotateSecret", reference, (provider) => provider.rotateSecret(reference, payload));
762
+ }
763
+ async deleteSecret(reference) {
764
+ await this.delegateToFirst("deleteSecret", reference, (provider) => provider.deleteSecret(reference));
765
+ }
766
+ async delegateToFirst(operation, reference, invoker) {
767
+ const errors = [];
768
+ for (const { provider } of this.providers) {
769
+ if (!safeCanHandle(provider, reference)) {
770
+ continue;
771
+ }
772
+ try {
773
+ return await invoker(provider);
774
+ } catch (error) {
775
+ if (error instanceof SecretProviderError) {
776
+ errors.push(error);
777
+ continue;
778
+ }
779
+ throw error;
780
+ }
781
+ }
782
+ throw this.composeError(operation, reference, errors);
783
+ }
784
+ composeError(operation, reference, errors, version) {
785
+ if (errors.length === 1) {
786
+ const [singleError] = errors;
787
+ if (singleError) {
788
+ return singleError;
789
+ }
790
+ }
791
+ const messageParts = [
792
+ `No registered secret provider could ${operation}`,
793
+ `reference "${reference}"`
794
+ ];
795
+ if (version) {
796
+ messageParts.push(`(version: ${version})`);
797
+ }
798
+ if (errors.length > 1) {
799
+ messageParts.push(`Attempts: ${errors.map((error) => `${error.provider}:${error.code}`).join(", ")}`);
800
+ }
801
+ return new SecretProviderError({
802
+ message: messageParts.join(" "),
803
+ provider: this.id,
804
+ reference,
805
+ code: errors.length > 0 ? errors[errors.length - 1].code : "UNKNOWN",
806
+ cause: errors
807
+ });
808
+ }
809
+ }
810
+ function safeCanHandle(provider, reference) {
811
+ try {
812
+ return provider.canHandle(reference);
813
+ } catch {
814
+ return false;
815
+ }
816
+ }
817
+ export {
818
+ parseSecretUri,
819
+ normalizeSecretPayload,
820
+ ensureConnectionReady,
821
+ connectionStatusLabel,
822
+ SecretProviderManager,
823
+ SecretProviderError,
824
+ IntegrationHealthService,
825
+ IntegrationCallGuard,
826
+ GcpSecretManagerProvider,
827
+ EnvSecretProvider
828
+ };