@contractspec/lib.contracts-integrations 2.0.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 (240) hide show
  1. package/README.md +3 -0
  2. package/dist/index.d.ts +11 -0
  3. package/dist/index.js +3676 -0
  4. package/dist/integrations/binding.d.ts +14 -0
  5. package/dist/integrations/binding.js +1 -0
  6. package/dist/integrations/connection.d.ts +47 -0
  7. package/dist/integrations/connection.js +1 -0
  8. package/dist/integrations/docs/integrations.docblock.d.ts +2 -0
  9. package/dist/integrations/docs/integrations.docblock.js +110 -0
  10. package/dist/integrations/health.d.ts +17 -0
  11. package/dist/integrations/health.js +73 -0
  12. package/dist/integrations/health.test.d.ts +1 -0
  13. package/dist/integrations/index.d.ts +11 -0
  14. package/dist/integrations/index.js +3264 -0
  15. package/dist/integrations/integrations.capability.d.ts +1 -0
  16. package/dist/integrations/integrations.capability.js +18 -0
  17. package/dist/integrations/integrations.feature.d.ts +5 -0
  18. package/dist/integrations/integrations.feature.js +33 -0
  19. package/dist/integrations/meeting-recorder/contracts/index.d.ts +7 -0
  20. package/dist/integrations/meeting-recorder/contracts/index.js +474 -0
  21. package/dist/integrations/meeting-recorder/contracts/meetings.d.ts +451 -0
  22. package/dist/integrations/meeting-recorder/contracts/meetings.js +219 -0
  23. package/dist/integrations/meeting-recorder/contracts/transcripts.d.ts +166 -0
  24. package/dist/integrations/meeting-recorder/contracts/transcripts.js +287 -0
  25. package/dist/integrations/meeting-recorder/contracts/webhooks.d.ts +85 -0
  26. package/dist/integrations/meeting-recorder/contracts/webhooks.js +172 -0
  27. package/dist/integrations/meeting-recorder/meeting-recorder.capability.d.ts +1 -0
  28. package/dist/integrations/meeting-recorder/meeting-recorder.capability.js +18 -0
  29. package/dist/integrations/meeting-recorder/meeting-recorder.feature.d.ts +5 -0
  30. package/dist/integrations/meeting-recorder/meeting-recorder.feature.js +33 -0
  31. package/dist/integrations/meeting-recorder/models.d.ts +402 -0
  32. package/dist/integrations/meeting-recorder/models.js +122 -0
  33. package/dist/integrations/meeting-recorder/telemetry.d.ts +13 -0
  34. package/dist/integrations/meeting-recorder/telemetry.js +54 -0
  35. package/dist/integrations/openbanking/contracts/accounts.d.ts +282 -0
  36. package/dist/integrations/openbanking/contracts/accounts.js +328 -0
  37. package/dist/integrations/openbanking/contracts/balances.d.ts +158 -0
  38. package/dist/integrations/openbanking/contracts/balances.js +292 -0
  39. package/dist/integrations/openbanking/contracts/index.d.ts +7 -0
  40. package/dist/integrations/openbanking/contracts/index.js +644 -0
  41. package/dist/integrations/openbanking/contracts/transactions.d.ts +206 -0
  42. package/dist/integrations/openbanking/contracts/transactions.js +298 -0
  43. package/dist/integrations/openbanking/guards.d.ts +8 -0
  44. package/dist/integrations/openbanking/guards.js +42 -0
  45. package/dist/integrations/openbanking/guards.test.d.ts +1 -0
  46. package/dist/integrations/openbanking/models.d.ts +223 -0
  47. package/dist/integrations/openbanking/models.js +110 -0
  48. package/dist/integrations/openbanking/openbanking.capability.d.ts +1 -0
  49. package/dist/integrations/openbanking/openbanking.capability.js +18 -0
  50. package/dist/integrations/openbanking/openbanking.feature.d.ts +5 -0
  51. package/dist/integrations/openbanking/openbanking.feature.js +35 -0
  52. package/dist/integrations/openbanking/telemetry.d.ts +12 -0
  53. package/dist/integrations/openbanking/telemetry.js +51 -0
  54. package/dist/integrations/operations.d.ts +430 -0
  55. package/dist/integrations/operations.js +297 -0
  56. package/dist/integrations/operations.test.d.ts +1 -0
  57. package/dist/integrations/providers/analytics-reader.d.ts +103 -0
  58. package/dist/integrations/providers/analytics-reader.js +1 -0
  59. package/dist/integrations/providers/analytics-writer.d.ts +6 -0
  60. package/dist/integrations/providers/analytics-writer.js +1 -0
  61. package/dist/integrations/providers/analytics.d.ts +47 -0
  62. package/dist/integrations/providers/analytics.js +1 -0
  63. package/dist/integrations/providers/calendar.d.ts +75 -0
  64. package/dist/integrations/providers/calendar.js +1 -0
  65. package/dist/integrations/providers/database.d.ts +12 -0
  66. package/dist/integrations/providers/database.js +1 -0
  67. package/dist/integrations/providers/elevenlabs.d.ts +3 -0
  68. package/dist/integrations/providers/elevenlabs.js +86 -0
  69. package/dist/integrations/providers/email.d.ts +83 -0
  70. package/dist/integrations/providers/email.js +1 -0
  71. package/dist/integrations/providers/embedding.d.ts +21 -0
  72. package/dist/integrations/providers/embedding.js +1 -0
  73. package/dist/integrations/providers/fal.d.ts +3 -0
  74. package/dist/integrations/providers/fal.js +112 -0
  75. package/dist/integrations/providers/fathom.d.ts +3 -0
  76. package/dist/integrations/providers/fathom.js +126 -0
  77. package/dist/integrations/providers/fireflies.d.ts +3 -0
  78. package/dist/integrations/providers/fireflies.js +106 -0
  79. package/dist/integrations/providers/gcs-storage.d.ts +3 -0
  80. package/dist/integrations/providers/gcs-storage.js +97 -0
  81. package/dist/integrations/providers/gmail.d.ts +3 -0
  82. package/dist/integrations/providers/gmail.js +109 -0
  83. package/dist/integrations/providers/google-calendar.d.ts +3 -0
  84. package/dist/integrations/providers/google-calendar.js +92 -0
  85. package/dist/integrations/providers/gradium.d.ts +3 -0
  86. package/dist/integrations/providers/gradium.js +110 -0
  87. package/dist/integrations/providers/granola.d.ts +3 -0
  88. package/dist/integrations/providers/granola.js +107 -0
  89. package/dist/integrations/providers/index.d.ts +38 -0
  90. package/dist/integrations/providers/index.js +2094 -0
  91. package/dist/integrations/providers/jira.d.ts +3 -0
  92. package/dist/integrations/providers/jira.js +108 -0
  93. package/dist/integrations/providers/linear.d.ts +3 -0
  94. package/dist/integrations/providers/linear.js +107 -0
  95. package/dist/integrations/providers/llm.d.ts +79 -0
  96. package/dist/integrations/providers/llm.js +1 -0
  97. package/dist/integrations/providers/meeting-recorder.d.ts +129 -0
  98. package/dist/integrations/providers/meeting-recorder.js +1 -0
  99. package/dist/integrations/providers/mistral.d.ts +3 -0
  100. package/dist/integrations/providers/mistral.js +94 -0
  101. package/dist/integrations/providers/notion.d.ts +3 -0
  102. package/dist/integrations/providers/notion.js +113 -0
  103. package/dist/integrations/providers/openbanking.d.ts +125 -0
  104. package/dist/integrations/providers/openbanking.js +1 -0
  105. package/dist/integrations/providers/payments.d.ts +106 -0
  106. package/dist/integrations/providers/payments.js +1 -0
  107. package/dist/integrations/providers/posthog-llm-telemetry.d.ts +51 -0
  108. package/dist/integrations/providers/posthog-llm-telemetry.js +176 -0
  109. package/dist/integrations/providers/posthog.d.ts +3 -0
  110. package/dist/integrations/providers/posthog.js +106 -0
  111. package/dist/integrations/providers/postmark.d.ts +3 -0
  112. package/dist/integrations/providers/postmark.js +98 -0
  113. package/dist/integrations/providers/powens.d.ts +3 -0
  114. package/dist/integrations/providers/powens.js +124 -0
  115. package/dist/integrations/providers/project-management.d.ts +32 -0
  116. package/dist/integrations/providers/project-management.js +1 -0
  117. package/dist/integrations/providers/providers.test.d.ts +1 -0
  118. package/dist/integrations/providers/qdrant.d.ts +3 -0
  119. package/dist/integrations/providers/qdrant.js +101 -0
  120. package/dist/integrations/providers/registry.d.ts +6 -0
  121. package/dist/integrations/providers/registry.js +1878 -0
  122. package/dist/integrations/providers/sms.d.ts +31 -0
  123. package/dist/integrations/providers/sms.js +1 -0
  124. package/dist/integrations/providers/storage.d.ts +57 -0
  125. package/dist/integrations/providers/storage.js +1 -0
  126. package/dist/integrations/providers/stripe.d.ts +3 -0
  127. package/dist/integrations/providers/stripe.js +105 -0
  128. package/dist/integrations/providers/supabase-postgres.d.ts +3 -0
  129. package/dist/integrations/providers/supabase-postgres.js +87 -0
  130. package/dist/integrations/providers/supabase-vector.d.ts +3 -0
  131. package/dist/integrations/providers/supabase-vector.js +107 -0
  132. package/dist/integrations/providers/tldv.d.ts +3 -0
  133. package/dist/integrations/providers/tldv.js +106 -0
  134. package/dist/integrations/providers/twilio-sms.d.ts +3 -0
  135. package/dist/integrations/providers/twilio-sms.js +91 -0
  136. package/dist/integrations/providers/vector-store.d.ts +39 -0
  137. package/dist/integrations/providers/vector-store.js +1 -0
  138. package/dist/integrations/providers/voice.d.ts +31 -0
  139. package/dist/integrations/providers/voice.js +1 -0
  140. package/dist/integrations/runtime.d.ts +95 -0
  141. package/dist/integrations/runtime.js +209 -0
  142. package/dist/integrations/runtime.test.d.ts +1 -0
  143. package/dist/integrations/secrets/aws-secret-manager.d.ts +28 -0
  144. package/dist/integrations/secrets/aws-secret-manager.js +346 -0
  145. package/dist/integrations/secrets/env-secret-provider.d.ts +28 -0
  146. package/dist/integrations/secrets/env-secret-provider.js +159 -0
  147. package/dist/integrations/secrets/gcp-secret-manager.d.ts +29 -0
  148. package/dist/integrations/secrets/gcp-secret-manager.js +347 -0
  149. package/dist/integrations/secrets/index.d.ts +6 -0
  150. package/dist/integrations/secrets/index.js +1129 -0
  151. package/dist/integrations/secrets/manager.d.ts +44 -0
  152. package/dist/integrations/secrets/manager.js +183 -0
  153. package/dist/integrations/secrets/provider.d.ts +49 -0
  154. package/dist/integrations/secrets/provider.js +74 -0
  155. package/dist/integrations/secrets/provider.test.d.ts +1 -0
  156. package/dist/integrations/secrets/scaleway-secret-manager.d.ts +35 -0
  157. package/dist/integrations/secrets/scaleway-secret-manager.js +375 -0
  158. package/dist/integrations/secrets-types.d.ts +14 -0
  159. package/dist/integrations/secrets-types.js +1 -0
  160. package/dist/integrations/spec.d.ts +72 -0
  161. package/dist/integrations/spec.js +22 -0
  162. package/dist/integrations/spec.test.d.ts +1 -0
  163. package/dist/node/index.js +3675 -0
  164. package/dist/node/integrations/binding.js +0 -0
  165. package/dist/node/integrations/connection.js +0 -0
  166. package/dist/node/integrations/docs/integrations.docblock.js +109 -0
  167. package/dist/node/integrations/health.js +72 -0
  168. package/dist/node/integrations/index.js +3263 -0
  169. package/dist/node/integrations/integrations.capability.js +17 -0
  170. package/dist/node/integrations/integrations.feature.js +32 -0
  171. package/dist/node/integrations/meeting-recorder/contracts/index.js +473 -0
  172. package/dist/node/integrations/meeting-recorder/contracts/meetings.js +218 -0
  173. package/dist/node/integrations/meeting-recorder/contracts/transcripts.js +286 -0
  174. package/dist/node/integrations/meeting-recorder/contracts/webhooks.js +171 -0
  175. package/dist/node/integrations/meeting-recorder/meeting-recorder.capability.js +17 -0
  176. package/dist/node/integrations/meeting-recorder/meeting-recorder.feature.js +32 -0
  177. package/dist/node/integrations/meeting-recorder/models.js +121 -0
  178. package/dist/node/integrations/meeting-recorder/telemetry.js +53 -0
  179. package/dist/node/integrations/openbanking/contracts/accounts.js +327 -0
  180. package/dist/node/integrations/openbanking/contracts/balances.js +291 -0
  181. package/dist/node/integrations/openbanking/contracts/index.js +643 -0
  182. package/dist/node/integrations/openbanking/contracts/transactions.js +297 -0
  183. package/dist/node/integrations/openbanking/guards.js +41 -0
  184. package/dist/node/integrations/openbanking/models.js +109 -0
  185. package/dist/node/integrations/openbanking/openbanking.capability.js +17 -0
  186. package/dist/node/integrations/openbanking/openbanking.feature.js +34 -0
  187. package/dist/node/integrations/openbanking/telemetry.js +50 -0
  188. package/dist/node/integrations/operations.js +296 -0
  189. package/dist/node/integrations/providers/analytics-reader.js +0 -0
  190. package/dist/node/integrations/providers/analytics-writer.js +0 -0
  191. package/dist/node/integrations/providers/analytics.js +0 -0
  192. package/dist/node/integrations/providers/calendar.js +0 -0
  193. package/dist/node/integrations/providers/database.js +0 -0
  194. package/dist/node/integrations/providers/elevenlabs.js +85 -0
  195. package/dist/node/integrations/providers/email.js +0 -0
  196. package/dist/node/integrations/providers/embedding.js +0 -0
  197. package/dist/node/integrations/providers/fal.js +111 -0
  198. package/dist/node/integrations/providers/fathom.js +125 -0
  199. package/dist/node/integrations/providers/fireflies.js +105 -0
  200. package/dist/node/integrations/providers/gcs-storage.js +96 -0
  201. package/dist/node/integrations/providers/gmail.js +108 -0
  202. package/dist/node/integrations/providers/google-calendar.js +91 -0
  203. package/dist/node/integrations/providers/gradium.js +109 -0
  204. package/dist/node/integrations/providers/granola.js +106 -0
  205. package/dist/node/integrations/providers/index.js +2093 -0
  206. package/dist/node/integrations/providers/jira.js +107 -0
  207. package/dist/node/integrations/providers/linear.js +106 -0
  208. package/dist/node/integrations/providers/llm.js +0 -0
  209. package/dist/node/integrations/providers/meeting-recorder.js +0 -0
  210. package/dist/node/integrations/providers/mistral.js +93 -0
  211. package/dist/node/integrations/providers/notion.js +112 -0
  212. package/dist/node/integrations/providers/openbanking.js +0 -0
  213. package/dist/node/integrations/providers/payments.js +0 -0
  214. package/dist/node/integrations/providers/posthog-llm-telemetry.js +175 -0
  215. package/dist/node/integrations/providers/posthog.js +105 -0
  216. package/dist/node/integrations/providers/postmark.js +97 -0
  217. package/dist/node/integrations/providers/powens.js +123 -0
  218. package/dist/node/integrations/providers/project-management.js +0 -0
  219. package/dist/node/integrations/providers/qdrant.js +100 -0
  220. package/dist/node/integrations/providers/registry.js +1877 -0
  221. package/dist/node/integrations/providers/sms.js +0 -0
  222. package/dist/node/integrations/providers/storage.js +0 -0
  223. package/dist/node/integrations/providers/stripe.js +104 -0
  224. package/dist/node/integrations/providers/supabase-postgres.js +86 -0
  225. package/dist/node/integrations/providers/supabase-vector.js +106 -0
  226. package/dist/node/integrations/providers/tldv.js +105 -0
  227. package/dist/node/integrations/providers/twilio-sms.js +90 -0
  228. package/dist/node/integrations/providers/vector-store.js +0 -0
  229. package/dist/node/integrations/providers/voice.js +0 -0
  230. package/dist/node/integrations/runtime.js +208 -0
  231. package/dist/node/integrations/secrets/aws-secret-manager.js +345 -0
  232. package/dist/node/integrations/secrets/env-secret-provider.js +158 -0
  233. package/dist/node/integrations/secrets/gcp-secret-manager.js +346 -0
  234. package/dist/node/integrations/secrets/index.js +1128 -0
  235. package/dist/node/integrations/secrets/manager.js +182 -0
  236. package/dist/node/integrations/secrets/provider.js +73 -0
  237. package/dist/node/integrations/secrets/scaleway-secret-manager.js +374 -0
  238. package/dist/node/integrations/secrets-types.js +0 -0
  239. package/dist/node/integrations/spec.js +21 -0
  240. package/package.json +1029 -0
@@ -0,0 +1,28 @@
1
+ import type { SecretProvider, SecretReference, SecretRotationResult, SecretValue, SecretWritePayload } from './provider';
2
+ interface EnvSecretProviderOptions {
3
+ /**
4
+ * Optional map to alias secret references to environment variable names.
5
+ * Useful when referencing secrets from other providers (e.g. gcp://...)
6
+ * while still allowing local overrides.
7
+ */
8
+ aliases?: Record<string, string>;
9
+ }
10
+ /**
11
+ * Environment-variable backed secret provider. Read-only by design.
12
+ * Allows overriding other secret providers by deriving environment variable
13
+ * names from secret references (or by using explicit aliases).
14
+ */
15
+ export declare class EnvSecretProvider implements SecretProvider {
16
+ readonly id = "env";
17
+ private readonly aliases;
18
+ constructor(options?: EnvSecretProviderOptions);
19
+ canHandle(reference: SecretReference): boolean;
20
+ getSecret(reference: SecretReference): Promise<SecretValue>;
21
+ setSecret(reference: SecretReference, _payload: SecretWritePayload): Promise<SecretRotationResult>;
22
+ rotateSecret(reference: SecretReference, _payload: SecretWritePayload): Promise<SecretRotationResult>;
23
+ deleteSecret(reference: SecretReference): Promise<void>;
24
+ private resolveEnvKey;
25
+ private deriveEnvKey;
26
+ private forbiddenError;
27
+ }
28
+ export {};
@@ -0,0 +1,159 @@
1
+ // @bun
2
+ // src/integrations/secrets/provider.ts
3
+ import { Buffer as Buffer2 } from "buffer";
4
+
5
+ class SecretProviderError extends Error {
6
+ provider;
7
+ reference;
8
+ code;
9
+ cause;
10
+ constructor(params) {
11
+ super(params.message);
12
+ this.name = "SecretProviderError";
13
+ this.provider = params.provider;
14
+ this.reference = params.reference;
15
+ this.code = params.code ?? "UNKNOWN";
16
+ this.cause = params.cause;
17
+ }
18
+ }
19
+ function parseSecretUri(reference) {
20
+ if (!reference) {
21
+ throw new SecretProviderError({
22
+ message: "Secret reference cannot be empty",
23
+ provider: "unknown",
24
+ reference,
25
+ code: "INVALID"
26
+ });
27
+ }
28
+ const [scheme, rest] = reference.split("://");
29
+ if (!scheme || !rest) {
30
+ throw new SecretProviderError({
31
+ message: `Invalid secret reference: ${reference}`,
32
+ provider: "unknown",
33
+ reference,
34
+ code: "INVALID"
35
+ });
36
+ }
37
+ const queryIndex = rest.indexOf("?");
38
+ if (queryIndex === -1) {
39
+ return {
40
+ provider: scheme,
41
+ path: rest
42
+ };
43
+ }
44
+ const path = rest.slice(0, queryIndex);
45
+ const query = rest.slice(queryIndex + 1);
46
+ const extras = Object.fromEntries(query.split("&").filter(Boolean).map((pair) => {
47
+ const [keyRaw, valueRaw] = pair.split("=");
48
+ const key = keyRaw ?? "";
49
+ const value = valueRaw ?? "";
50
+ return [decodeURIComponent(key), decodeURIComponent(value)];
51
+ }));
52
+ return {
53
+ provider: scheme,
54
+ path,
55
+ extras
56
+ };
57
+ }
58
+ function normalizeSecretPayload(payload) {
59
+ if (payload.data instanceof Uint8Array) {
60
+ return payload.data;
61
+ }
62
+ if (payload.encoding === "base64") {
63
+ return Buffer2.from(payload.data, "base64");
64
+ }
65
+ if (payload.encoding === "binary") {
66
+ return Buffer2.from(payload.data, "binary");
67
+ }
68
+ return Buffer2.from(payload.data, "utf-8");
69
+ }
70
+
71
+ // src/integrations/secrets/env-secret-provider.ts
72
+ class EnvSecretProvider {
73
+ id = "env";
74
+ aliases;
75
+ constructor(options = {}) {
76
+ this.aliases = options.aliases ?? {};
77
+ }
78
+ canHandle(reference) {
79
+ const envKey = this.resolveEnvKey(reference);
80
+ return envKey !== undefined && process.env[envKey] !== undefined;
81
+ }
82
+ async getSecret(reference) {
83
+ const envKey = this.resolveEnvKey(reference);
84
+ if (!envKey) {
85
+ throw new SecretProviderError({
86
+ message: `Unable to resolve environment variable for reference "${reference}".`,
87
+ provider: this.id,
88
+ reference,
89
+ code: "INVALID"
90
+ });
91
+ }
92
+ const value = process.env[envKey];
93
+ if (value === undefined) {
94
+ throw new SecretProviderError({
95
+ message: `Environment variable "${envKey}" not found for reference "${reference}".`,
96
+ provider: this.id,
97
+ reference,
98
+ code: "NOT_FOUND"
99
+ });
100
+ }
101
+ return {
102
+ data: Buffer.from(value, "utf-8"),
103
+ version: "current",
104
+ metadata: {
105
+ source: "env",
106
+ envKey
107
+ },
108
+ retrievedAt: new Date
109
+ };
110
+ }
111
+ async setSecret(reference, _payload) {
112
+ throw this.forbiddenError("setSecret", reference);
113
+ }
114
+ async rotateSecret(reference, _payload) {
115
+ throw this.forbiddenError("rotateSecret", reference);
116
+ }
117
+ async deleteSecret(reference) {
118
+ throw this.forbiddenError("deleteSecret", reference);
119
+ }
120
+ resolveEnvKey(reference) {
121
+ if (!reference) {
122
+ return;
123
+ }
124
+ if (this.aliases[reference]) {
125
+ return this.aliases[reference];
126
+ }
127
+ if (!reference.includes("://")) {
128
+ return reference;
129
+ }
130
+ try {
131
+ const parsed = parseSecretUri(reference);
132
+ if (parsed.provider === "env") {
133
+ return parsed.path;
134
+ }
135
+ if (parsed.extras?.env) {
136
+ return parsed.extras.env;
137
+ }
138
+ return this.deriveEnvKey(parsed.path);
139
+ } catch {
140
+ return reference;
141
+ }
142
+ }
143
+ deriveEnvKey(path) {
144
+ if (!path)
145
+ return;
146
+ return path.split(/[\/:\-\.]/).filter(Boolean).map((segment) => segment.replace(/[^a-zA-Z0-9]/g, "_").replace(/_{2,}/g, "_").toUpperCase()).join("_");
147
+ }
148
+ forbiddenError(operation, reference) {
149
+ return new SecretProviderError({
150
+ message: `EnvSecretProvider is read-only. "${operation}" is not allowed for ${reference}.`,
151
+ provider: this.id,
152
+ reference,
153
+ code: "FORBIDDEN"
154
+ });
155
+ }
156
+ }
157
+ export {
158
+ EnvSecretProvider
159
+ };
@@ -0,0 +1,29 @@
1
+ import { protos, SecretManagerServiceClient } from '@google-cloud/secret-manager';
2
+ import type { CallOptions } from 'google-gax';
3
+ import type { SecretProvider, SecretReference, SecretRotationResult, SecretValue, SecretWritePayload } from './provider';
4
+ type SecretManagerClient = SecretManagerServiceClient;
5
+ interface GcpSecretManagerProviderOptions {
6
+ projectId?: string;
7
+ client?: SecretManagerClient;
8
+ clientOptions?: ConstructorParameters<typeof SecretManagerServiceClient>[0];
9
+ defaultReplication?: protos.google.cloud.secretmanager.v1.IReplication;
10
+ }
11
+ export declare class GcpSecretManagerProvider implements SecretProvider {
12
+ readonly id = "gcp-secret-manager";
13
+ private readonly client;
14
+ private readonly explicitProjectId?;
15
+ private readonly replication;
16
+ constructor(options?: GcpSecretManagerProviderOptions);
17
+ canHandle(reference: SecretReference): boolean;
18
+ getSecret(reference: SecretReference, options?: {
19
+ version?: string;
20
+ }, callOptions?: CallOptions): Promise<SecretValue>;
21
+ setSecret(reference: SecretReference, payload: SecretWritePayload): Promise<SecretRotationResult>;
22
+ rotateSecret(reference: SecretReference, payload: SecretWritePayload): Promise<SecretRotationResult>;
23
+ deleteSecret(reference: SecretReference): Promise<void>;
24
+ private parseReference;
25
+ private buildNames;
26
+ private buildVersionName;
27
+ private ensureSecretExists;
28
+ }
29
+ export {};
@@ -0,0 +1,347 @@
1
+ // @bun
2
+ // src/integrations/secrets/provider.ts
3
+ import { Buffer } from "buffer";
4
+
5
+ class SecretProviderError extends Error {
6
+ provider;
7
+ reference;
8
+ code;
9
+ cause;
10
+ constructor(params) {
11
+ super(params.message);
12
+ this.name = "SecretProviderError";
13
+ this.provider = params.provider;
14
+ this.reference = params.reference;
15
+ this.code = params.code ?? "UNKNOWN";
16
+ this.cause = params.cause;
17
+ }
18
+ }
19
+ function parseSecretUri(reference) {
20
+ if (!reference) {
21
+ throw new SecretProviderError({
22
+ message: "Secret reference cannot be empty",
23
+ provider: "unknown",
24
+ reference,
25
+ code: "INVALID"
26
+ });
27
+ }
28
+ const [scheme, rest] = reference.split("://");
29
+ if (!scheme || !rest) {
30
+ throw new SecretProviderError({
31
+ message: `Invalid secret reference: ${reference}`,
32
+ provider: "unknown",
33
+ reference,
34
+ code: "INVALID"
35
+ });
36
+ }
37
+ const queryIndex = rest.indexOf("?");
38
+ if (queryIndex === -1) {
39
+ return {
40
+ provider: scheme,
41
+ path: rest
42
+ };
43
+ }
44
+ const path = rest.slice(0, queryIndex);
45
+ const query = rest.slice(queryIndex + 1);
46
+ const extras = Object.fromEntries(query.split("&").filter(Boolean).map((pair) => {
47
+ const [keyRaw, valueRaw] = pair.split("=");
48
+ const key = keyRaw ?? "";
49
+ const value = valueRaw ?? "";
50
+ return [decodeURIComponent(key), decodeURIComponent(value)];
51
+ }));
52
+ return {
53
+ provider: scheme,
54
+ path,
55
+ extras
56
+ };
57
+ }
58
+ function normalizeSecretPayload(payload) {
59
+ if (payload.data instanceof Uint8Array) {
60
+ return payload.data;
61
+ }
62
+ if (payload.encoding === "base64") {
63
+ return Buffer.from(payload.data, "base64");
64
+ }
65
+ if (payload.encoding === "binary") {
66
+ return Buffer.from(payload.data, "binary");
67
+ }
68
+ return Buffer.from(payload.data, "utf-8");
69
+ }
70
+
71
+ // src/integrations/secrets/gcp-secret-manager.ts
72
+ import {
73
+ SecretManagerServiceClient
74
+ } from "@google-cloud/secret-manager";
75
+ var DEFAULT_REPLICATION = {
76
+ automatic: {}
77
+ };
78
+
79
+ class GcpSecretManagerProvider {
80
+ id = "gcp-secret-manager";
81
+ client;
82
+ explicitProjectId;
83
+ replication;
84
+ constructor(options = {}) {
85
+ this.client = options.client ?? new SecretManagerServiceClient(options.clientOptions ?? {});
86
+ this.explicitProjectId = options.projectId;
87
+ this.replication = options.defaultReplication ?? DEFAULT_REPLICATION;
88
+ }
89
+ canHandle(reference) {
90
+ try {
91
+ const parsed = parseSecretUri(reference);
92
+ return parsed.provider === "gcp";
93
+ } catch {
94
+ return false;
95
+ }
96
+ }
97
+ async getSecret(reference, options, callOptions) {
98
+ const location = this.parseReference(reference);
99
+ const secretVersionName = this.buildVersionName(location, options?.version);
100
+ try {
101
+ const response = await this.client.accessSecretVersion({
102
+ name: secretVersionName
103
+ }, callOptions ?? {});
104
+ const [result] = response;
105
+ const payload = result.payload;
106
+ if (!payload?.data) {
107
+ throw new SecretProviderError({
108
+ message: `Secret payload empty for ${secretVersionName}`,
109
+ provider: this.id,
110
+ reference,
111
+ code: "UNKNOWN"
112
+ });
113
+ }
114
+ const version = extractVersionFromName(result.name ?? secretVersionName);
115
+ return {
116
+ data: payload.data,
117
+ version,
118
+ metadata: payload.dataCrc32c ? { crc32c: payload.dataCrc32c.toString() } : undefined,
119
+ retrievedAt: new Date
120
+ };
121
+ } catch (error) {
122
+ throw toSecretProviderError({
123
+ error,
124
+ provider: this.id,
125
+ reference,
126
+ operation: "access"
127
+ });
128
+ }
129
+ }
130
+ async setSecret(reference, payload) {
131
+ const location = this.parseReference(reference);
132
+ const { secretName } = this.buildNames(location);
133
+ const data = normalizeSecretPayload(payload);
134
+ await this.ensureSecretExists(location, payload);
135
+ try {
136
+ const response = await this.client.addSecretVersion({
137
+ parent: secretName,
138
+ payload: {
139
+ data
140
+ }
141
+ });
142
+ if (!response) {
143
+ throw new SecretProviderError({
144
+ message: `No version returned when adding secret version for ${secretName}`,
145
+ provider: this.id,
146
+ reference,
147
+ code: "UNKNOWN"
148
+ });
149
+ }
150
+ const [version] = response;
151
+ const versionName = version?.name ?? `${secretName}/versions/latest`;
152
+ return {
153
+ reference: `gcp://${versionName}`,
154
+ version: extractVersionFromName(versionName) ?? "latest"
155
+ };
156
+ } catch (error) {
157
+ throw toSecretProviderError({
158
+ error,
159
+ provider: this.id,
160
+ reference,
161
+ operation: "addSecretVersion"
162
+ });
163
+ }
164
+ }
165
+ async rotateSecret(reference, payload) {
166
+ return this.setSecret(reference, payload);
167
+ }
168
+ async deleteSecret(reference) {
169
+ const location = this.parseReference(reference);
170
+ const { secretName } = this.buildNames(location);
171
+ try {
172
+ await this.client.deleteSecret({
173
+ name: secretName
174
+ });
175
+ } catch (error) {
176
+ throw toSecretProviderError({
177
+ error,
178
+ provider: this.id,
179
+ reference,
180
+ operation: "delete"
181
+ });
182
+ }
183
+ }
184
+ parseReference(reference) {
185
+ const parsed = parseSecretUri(reference);
186
+ if (parsed.provider !== "gcp") {
187
+ throw new SecretProviderError({
188
+ message: `Unsupported secret provider: ${parsed.provider}`,
189
+ provider: this.id,
190
+ reference,
191
+ code: "INVALID"
192
+ });
193
+ }
194
+ const segments = parsed.path.split("/").filter(Boolean);
195
+ if (segments.length < 4 || segments[0] !== "projects") {
196
+ throw new SecretProviderError({
197
+ message: `Expected secret reference format gcp://projects/{project}/secrets/{secret}[(/versions/{version})] but received "${parsed.path}"`,
198
+ provider: this.id,
199
+ reference,
200
+ code: "INVALID"
201
+ });
202
+ }
203
+ const projectIdCandidate = segments[1] ?? this.explicitProjectId;
204
+ if (!projectIdCandidate) {
205
+ throw new SecretProviderError({
206
+ message: `Unable to resolve project or secret from reference "${parsed.path}"`,
207
+ provider: this.id,
208
+ reference,
209
+ code: "INVALID"
210
+ });
211
+ }
212
+ const indexOfSecrets = segments.indexOf("secrets");
213
+ if (indexOfSecrets === -1 || indexOfSecrets + 1 >= segments.length) {
214
+ throw new SecretProviderError({
215
+ message: `Unable to resolve project or secret from reference "${parsed.path}"`,
216
+ provider: this.id,
217
+ reference,
218
+ code: "INVALID"
219
+ });
220
+ }
221
+ const resolvedProjectId = projectIdCandidate;
222
+ const secretIdCandidate = segments[indexOfSecrets + 1];
223
+ if (!secretIdCandidate) {
224
+ throw new SecretProviderError({
225
+ message: `Unable to resolve secret ID from reference "${parsed.path}"`,
226
+ provider: this.id,
227
+ reference,
228
+ code: "INVALID"
229
+ });
230
+ }
231
+ const secretId = secretIdCandidate;
232
+ const indexOfVersions = segments.indexOf("versions");
233
+ const version = parsed.extras?.version ?? (indexOfVersions !== -1 && indexOfVersions + 1 < segments.length ? segments[indexOfVersions + 1] : undefined);
234
+ return {
235
+ projectId: resolvedProjectId,
236
+ secretId,
237
+ version
238
+ };
239
+ }
240
+ buildNames(location) {
241
+ const projectId = location.projectId ?? this.explicitProjectId;
242
+ if (!projectId) {
243
+ throw new SecretProviderError({
244
+ message: "Project ID must be provided either in reference or provider configuration",
245
+ provider: this.id,
246
+ reference: `gcp://projects//secrets/${location.secretId}`,
247
+ code: "INVALID"
248
+ });
249
+ }
250
+ const projectParent = `projects/${projectId}`;
251
+ const secretName = `${projectParent}/secrets/${location.secretId}`;
252
+ return {
253
+ projectParent,
254
+ secretName
255
+ };
256
+ }
257
+ buildVersionName(location, explicitVersion) {
258
+ const { secretName } = this.buildNames(location);
259
+ const version = explicitVersion ?? location.version ?? "latest";
260
+ return `${secretName}/versions/${version}`;
261
+ }
262
+ async ensureSecretExists(location, payload) {
263
+ const { secretName, projectParent } = this.buildNames(location);
264
+ try {
265
+ await this.client.getSecret({ name: secretName });
266
+ } catch (error) {
267
+ const providerError = toSecretProviderError({
268
+ error,
269
+ provider: this.id,
270
+ reference: `gcp://${secretName}`,
271
+ operation: "getSecret",
272
+ suppressThrow: true
273
+ });
274
+ if (!providerError || providerError.code !== "NOT_FOUND") {
275
+ if (providerError) {
276
+ throw providerError;
277
+ }
278
+ throw error;
279
+ }
280
+ try {
281
+ await this.client.createSecret({
282
+ parent: projectParent,
283
+ secretId: location.secretId,
284
+ secret: {
285
+ replication: this.replication,
286
+ labels: payload.labels
287
+ }
288
+ });
289
+ } catch (creationError) {
290
+ const creationProviderError = toSecretProviderError({
291
+ error: creationError,
292
+ provider: this.id,
293
+ reference: `gcp://${secretName}`,
294
+ operation: "createSecret"
295
+ });
296
+ throw creationProviderError;
297
+ }
298
+ }
299
+ }
300
+ }
301
+ function extractVersionFromName(name) {
302
+ const segments = name.split("/").filter(Boolean);
303
+ const index = segments.indexOf("versions");
304
+ if (index === -1 || index + 1 >= segments.length) {
305
+ return;
306
+ }
307
+ return segments[index + 1];
308
+ }
309
+ function toSecretProviderError(params) {
310
+ const { error, provider, reference, operation, suppressThrow } = params;
311
+ if (error instanceof SecretProviderError) {
312
+ return error;
313
+ }
314
+ const code = deriveErrorCode(error);
315
+ const message = error instanceof Error ? error.message : `Unknown error during ${operation}`;
316
+ const providerError = new SecretProviderError({
317
+ message,
318
+ provider,
319
+ reference,
320
+ code,
321
+ cause: error
322
+ });
323
+ if (suppressThrow) {
324
+ return providerError;
325
+ }
326
+ throw providerError;
327
+ }
328
+ function deriveErrorCode(error) {
329
+ if (typeof error !== "object" || error === null) {
330
+ return "UNKNOWN";
331
+ }
332
+ const errorAny = error;
333
+ const code = errorAny.code;
334
+ if (code === 5 || code === "NOT_FOUND")
335
+ return "NOT_FOUND";
336
+ if (code === 6 || code === "ALREADY_EXISTS")
337
+ return "INVALID";
338
+ if (code === 7 || code === "PERMISSION_DENIED" || code === 403) {
339
+ return "FORBIDDEN";
340
+ }
341
+ if (code === 3 || code === "INVALID_ARGUMENT")
342
+ return "INVALID";
343
+ return "UNKNOWN";
344
+ }
345
+ export {
346
+ GcpSecretManagerProvider
347
+ };
@@ -0,0 +1,6 @@
1
+ export * from './provider';
2
+ export * from './gcp-secret-manager';
3
+ export * from './aws-secret-manager';
4
+ export * from './scaleway-secret-manager';
5
+ export * from './env-secret-provider';
6
+ export * from './manager';