@lssm/lib.contracts 0.0.0-canary-20251217062943 → 0.0.0-canary-20251217072406

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 (237) hide show
  1. package/dist/app-config/app-config.feature.js +53 -1
  2. package/dist/app-config/contracts.d.ts +50 -50
  3. package/dist/app-config/contracts.js +396 -1
  4. package/dist/app-config/docs/app-config.docblock.js +22 -220
  5. package/dist/app-config/events.js +168 -1
  6. package/dist/app-config/index.js +8 -1
  7. package/dist/app-config/lifecycle-contracts.js +441 -1
  8. package/dist/app-config/runtime.js +617 -1
  9. package/dist/app-config/spec.js +36 -1
  10. package/dist/app-config/validation.js +538 -1
  11. package/dist/capabilities/docs/capabilities.docblock.js +22 -1
  12. package/dist/capabilities/openbanking.js +92 -1
  13. package/dist/capabilities.js +50 -1
  14. package/dist/client/index.js +9 -1
  15. package/dist/client/react/drivers/rn-reusables.js +21 -1
  16. package/dist/client/react/drivers/shadcn.js +11 -1
  17. package/dist/client/react/feature-render.js +43 -1
  18. package/dist/client/react/form-render.js +298 -1
  19. package/dist/client/react/index.js +8 -1
  20. package/dist/contract-registry/index.js +3 -1
  21. package/dist/contract-registry/schemas.js +61 -1
  22. package/dist/contracts-adapter-hydration.js +41 -1
  23. package/dist/contracts-adapter-input.js +77 -1
  24. package/dist/data-views/docs/data-views.docblock.js +22 -1
  25. package/dist/data-views/query-generator.js +48 -1
  26. package/dist/data-views/runtime.js +39 -1
  27. package/dist/data-views.js +35 -1
  28. package/dist/docs/PUBLISHING.docblock.js +17 -76
  29. package/dist/docs/accessibility_wcag_compliance_specs.docblock.js +17 -350
  30. package/dist/docs/index.js +33 -1
  31. package/dist/docs/meta.docs.js +15 -2
  32. package/dist/docs/presentations.js +77 -1
  33. package/dist/docs/registry.js +51 -1
  34. package/dist/docs/tech/PHASE_1_QUICKSTART.docblock.js +17 -383
  35. package/dist/docs/tech/PHASE_2_AI_NATIVE_OPERATIONS.docblock.js +17 -68
  36. package/dist/docs/tech/PHASE_3_AUTO_EVOLUTION.docblock.js +17 -140
  37. package/dist/docs/tech/PHASE_4_PERSONALIZATION_ENGINE.docblock.js +17 -86
  38. package/dist/docs/tech/PHASE_5_ZERO_TOUCH_OPERATIONS.docblock.js +17 -1
  39. package/dist/docs/tech/auth/better-auth-nextjs.docblock.js +25 -2
  40. package/dist/docs/tech/contracts/README.docblock.js +21 -1
  41. package/dist/docs/tech/contracts/create-subscription.docblock.js +21 -1
  42. package/dist/docs/tech/contracts/graphql-typed-outputs.docblock.js +21 -180
  43. package/dist/docs/tech/contracts/migrations.docblock.js +21 -1
  44. package/dist/docs/tech/contracts/openapi-export.docblock.js +22 -2
  45. package/dist/docs/tech/contracts/ops-to-presentation-linking.docblock.js +19 -60
  46. package/dist/docs/tech/contracts/overlays.docblock.js +21 -68
  47. package/dist/docs/tech/contracts/tests.docblock.js +21 -132
  48. package/dist/docs/tech/contracts/themes.docblock.js +21 -1
  49. package/dist/docs/tech/contracts/vertical-pocket-family-office.docblock.js +21 -106
  50. package/dist/docs/tech/lifecycle-stage-system.docblock.js +17 -213
  51. package/dist/docs/tech/llm/llm-integration.docblock.js +74 -5
  52. package/dist/docs/tech/mcp-endpoints.docblock.js +38 -1
  53. package/dist/docs/tech/presentation-runtime.docblock.js +17 -1
  54. package/dist/docs/tech/schema/README.docblock.js +21 -262
  55. package/dist/docs/tech/studio/learning-events.docblock.js +49 -1
  56. package/dist/docs/tech/studio/learning-journeys.docblock.js +25 -2
  57. package/dist/docs/tech/studio/platform-admin-panel.docblock.js +24 -2
  58. package/dist/docs/tech/studio/project-access-teams.docblock.js +26 -16
  59. package/dist/docs/tech/studio/project-routing.docblock.js +68 -1
  60. package/dist/docs/tech/studio/sandbox-unlogged.docblock.js +23 -2
  61. package/dist/docs/tech/studio/team-invitations.docblock.js +41 -36
  62. package/dist/docs/tech/studio/workspace-ops.docblock.js +48 -1
  63. package/dist/docs/tech/studio/workspaces.docblock.js +24 -2
  64. package/dist/docs/tech/telemetry-ingest.docblock.js +37 -3
  65. package/dist/docs/tech/templates/runtime.docblock.js +21 -1
  66. package/dist/docs/tech/vscode-extension.docblock.js +37 -3
  67. package/dist/docs/tech/workflows/overview.docblock.js +21 -1
  68. package/dist/docs/tech-contracts.docs.js +19 -2
  69. package/dist/events.js +12 -1
  70. package/dist/experiments/docs/experiments.docblock.js +22 -128
  71. package/dist/experiments/evaluator.js +101 -1
  72. package/dist/experiments/spec.js +33 -1
  73. package/dist/features.js +68 -1
  74. package/dist/forms/docs/forms.docblock.js +22 -1
  75. package/dist/forms.js +119 -1
  76. package/dist/index.js +107 -1
  77. package/dist/install.js +40 -1
  78. package/dist/integrations/contracts.d.ts +102 -102
  79. package/dist/integrations/contracts.js +388 -1
  80. package/dist/integrations/docs/integrations.docblock.js +95 -1
  81. package/dist/integrations/health.js +69 -1
  82. package/dist/integrations/index.js +23 -1
  83. package/dist/integrations/openbanking/contracts/accounts.d.ts +66 -66
  84. package/dist/integrations/openbanking/contracts/accounts.js +237 -1
  85. package/dist/integrations/openbanking/contracts/balances.d.ts +34 -34
  86. package/dist/integrations/openbanking/contracts/balances.js +167 -1
  87. package/dist/integrations/openbanking/contracts/index.js +12 -1
  88. package/dist/integrations/openbanking/contracts/transactions.d.ts +48 -48
  89. package/dist/integrations/openbanking/contracts/transactions.js +218 -1
  90. package/dist/integrations/openbanking/guards.js +32 -1
  91. package/dist/integrations/openbanking/models.d.ts +55 -55
  92. package/dist/integrations/openbanking/models.js +242 -1
  93. package/dist/integrations/openbanking/openbanking.feature.js +68 -1
  94. package/dist/integrations/openbanking/telemetry.js +39 -1
  95. package/dist/integrations/providers/elevenlabs.js +56 -1
  96. package/dist/integrations/providers/gcs-storage.js +79 -1
  97. package/dist/integrations/providers/gmail.js +91 -1
  98. package/dist/integrations/providers/google-calendar.js +70 -1
  99. package/dist/integrations/providers/impls/elevenlabs-voice.js +95 -1
  100. package/dist/integrations/providers/impls/gcs-storage.js +88 -1
  101. package/dist/integrations/providers/impls/gmail-inbound.js +200 -1
  102. package/dist/integrations/providers/impls/gmail-outbound.js +104 -5
  103. package/dist/integrations/providers/impls/google-calendar.js +154 -1
  104. package/dist/integrations/providers/impls/index.js +16 -1
  105. package/dist/integrations/providers/impls/mistral-embedding.js +41 -1
  106. package/dist/integrations/providers/impls/mistral-llm.js +247 -1
  107. package/dist/integrations/providers/impls/postmark-email.js +55 -1
  108. package/dist/integrations/providers/impls/powens-client.js +171 -1
  109. package/dist/integrations/providers/impls/powens-openbanking.js +218 -1
  110. package/dist/integrations/providers/impls/provider-factory.js +142 -1
  111. package/dist/integrations/providers/impls/qdrant-vector.js +69 -1
  112. package/dist/integrations/providers/impls/stripe-payments.js +202 -1
  113. package/dist/integrations/providers/impls/twilio-sms.js +58 -1
  114. package/dist/integrations/providers/index.js +13 -1
  115. package/dist/integrations/providers/mistral.js +72 -1
  116. package/dist/integrations/providers/postmark.js +72 -1
  117. package/dist/integrations/providers/powens.js +120 -1
  118. package/dist/integrations/providers/qdrant.js +77 -1
  119. package/dist/integrations/providers/registry.js +34 -1
  120. package/dist/integrations/providers/stripe.js +87 -1
  121. package/dist/integrations/providers/twilio-sms.js +65 -1
  122. package/dist/integrations/runtime.js +186 -1
  123. package/dist/integrations/secrets/aws-secret-manager.js +231 -1
  124. package/dist/integrations/secrets/env-secret-provider.js +81 -1
  125. package/dist/integrations/secrets/gcp-secret-manager.js +229 -1
  126. package/dist/integrations/secrets/index.js +8 -1
  127. package/dist/integrations/secrets/manager.js +103 -1
  128. package/dist/integrations/secrets/provider.js +58 -1
  129. package/dist/integrations/secrets/scaleway-secret-manager.js +247 -1
  130. package/dist/integrations/spec.js +39 -1
  131. package/dist/jobs/define-job.js +16 -1
  132. package/dist/jobs/gcp-cloud-tasks.js +53 -1
  133. package/dist/jobs/gcp-pubsub.js +39 -1
  134. package/dist/jobs/handlers/gmail-sync-handler.js +9 -1
  135. package/dist/jobs/handlers/index.js +12 -1
  136. package/dist/jobs/handlers/ping-handler.js +15 -1
  137. package/dist/jobs/handlers/storage-document-handler.js +14 -1
  138. package/dist/jobs/index.js +4 -1
  139. package/dist/jobs/memory-queue.js +71 -1
  140. package/dist/jobs/queue.js +33 -1
  141. package/dist/jobs/scaleway-sqs-queue.js +153 -1
  142. package/dist/jsonschema.d.ts +3 -3
  143. package/dist/jsonschema.js +32 -1
  144. package/dist/knowledge/contracts.d.ts +66 -66
  145. package/dist/knowledge/contracts.js +317 -1
  146. package/dist/knowledge/docs/knowledge.docblock.js +22 -138
  147. package/dist/knowledge/index.js +10 -1
  148. package/dist/knowledge/ingestion/document-processor.js +54 -1
  149. package/dist/knowledge/ingestion/embedding-service.js +25 -1
  150. package/dist/knowledge/ingestion/gmail-adapter.js +50 -5
  151. package/dist/knowledge/ingestion/index.js +7 -1
  152. package/dist/knowledge/ingestion/storage-adapter.js +26 -1
  153. package/dist/knowledge/ingestion/vector-indexer.js +32 -1
  154. package/dist/knowledge/query/index.js +3 -1
  155. package/dist/knowledge/query/service.js +64 -2
  156. package/dist/knowledge/runtime.js +49 -1
  157. package/dist/knowledge/spaces/email-threads.js +38 -1
  158. package/dist/knowledge/spaces/financial-docs.js +38 -1
  159. package/dist/knowledge/spaces/financial-overview.js +42 -1
  160. package/dist/knowledge/spaces/index.js +8 -1
  161. package/dist/knowledge/spaces/product-canon.js +38 -1
  162. package/dist/knowledge/spaces/support-faq.js +41 -1
  163. package/dist/knowledge/spaces/uploaded-docs.js +38 -1
  164. package/dist/knowledge/spec.js +39 -1
  165. package/dist/llm/exporters.js +541 -8
  166. package/dist/llm/index.js +4 -1
  167. package/dist/llm/prompts.js +246 -56
  168. package/dist/markdown.js +116 -3
  169. package/dist/migrations.js +33 -1
  170. package/dist/onboarding-base.d.ts +29 -29
  171. package/dist/onboarding-base.js +196 -1
  172. package/dist/openapi.js +75 -1
  173. package/dist/openbanking/docs/openbanking.docblock.js +22 -109
  174. package/dist/ownership.js +40 -1
  175. package/dist/policy/docs/policy.docblock.js +22 -1
  176. package/dist/policy/engine.js +223 -1
  177. package/dist/policy/opa-adapter.js +71 -1
  178. package/dist/policy/spec.js +33 -1
  179. package/dist/presentations/docs/presentations-conventions.docblock.js +21 -7
  180. package/dist/presentations.backcompat.js +47 -1
  181. package/dist/presentations.d.ts +3 -3
  182. package/dist/presentations.js +66 -1
  183. package/dist/presentations.v2.js +278 -6
  184. package/dist/prompt.js +10 -1
  185. package/dist/promptRegistry.js +34 -1
  186. package/dist/regenerator/docs/regenerator.docblock.js +22 -184
  187. package/dist/regenerator/executor.js +86 -1
  188. package/dist/regenerator/index.js +6 -1
  189. package/dist/regenerator/service.js +92 -1
  190. package/dist/regenerator/sinks.js +32 -1
  191. package/dist/regenerator/utils.js +51 -1
  192. package/dist/registry.js +208 -1
  193. package/dist/resources.js +47 -1
  194. package/dist/schema/dist/EnumType.js +2 -1
  195. package/dist/schema/dist/FieldType.js +49 -1
  196. package/dist/schema/dist/ScalarTypeEnum.js +236 -1
  197. package/dist/schema/dist/SchemaModel.js +39 -1
  198. package/dist/schema/dist/entity/defineEntity.js +1 -1
  199. package/dist/schema/dist/entity/index.js +2 -1
  200. package/dist/schema/dist/entity/types.js +1 -1
  201. package/dist/schema/dist/index.js +6 -1
  202. package/dist/schema-to-markdown.js +214 -10
  203. package/dist/server/graphql-pothos.js +128 -1
  204. package/dist/server/index.js +10 -1
  205. package/dist/server/mcp/createMcpServer.js +28 -1
  206. package/dist/server/mcp/registerPresentations.js +151 -1
  207. package/dist/server/mcp/registerPrompts.js +36 -2
  208. package/dist/server/mcp/registerResources.js +35 -1
  209. package/dist/server/mcp/registerTools.js +22 -1
  210. package/dist/server/provider-mcp.js +3 -1
  211. package/dist/server/rest-elysia.js +20 -1
  212. package/dist/server/rest-express.js +39 -1
  213. package/dist/server/rest-generic.js +125 -1
  214. package/dist/server/rest-next-app.js +38 -1
  215. package/dist/server/rest-next-mcp.js +45 -1
  216. package/dist/server/rest-next-pages.js +25 -1
  217. package/dist/spec.js +35 -1
  218. package/dist/telemetry/anomaly.js +48 -1
  219. package/dist/telemetry/docs/telemetry.docblock.js +22 -139
  220. package/dist/telemetry/index.js +5 -1
  221. package/dist/telemetry/spec.js +69 -1
  222. package/dist/telemetry/tracker.js +76 -1
  223. package/dist/tests/index.js +4 -1
  224. package/dist/tests/runner.js +150 -1
  225. package/dist/tests/spec.js +33 -1
  226. package/dist/themes.js +39 -1
  227. package/dist/workflow/adapters/db-adapter.js +83 -1
  228. package/dist/workflow/adapters/file-adapter.js +11 -1
  229. package/dist/workflow/adapters/index.js +5 -1
  230. package/dist/workflow/adapters/memory-store.js +58 -1
  231. package/dist/workflow/expression.js +98 -1
  232. package/dist/workflow/index.js +9 -1
  233. package/dist/workflow/runner.js +337 -1
  234. package/dist/workflow/sla-monitor.js +47 -1
  235. package/dist/workflow/spec.js +32 -1
  236. package/dist/workflow/validation.js +175 -1
  237. package/package.json +11 -4
@@ -1 +1,95 @@
1
- import{ElevenLabsClient as e}from"@elevenlabs/elevenlabs-js";const t={mp3:`mp3_44100_128`,wav:`pcm_44100`,ogg:`mp3_44100_128`,pcm:`pcm_16000`},n={mp3_22050_32:22050,mp3_44100_32:44100,mp3_44100_64:44100,mp3_44100_96:44100,mp3_44100_128:44100,mp3_44100_192:44100,pcm_16000:16e3,pcm_22050:22050,pcm_24000:24e3,pcm_44100:44100,ulaw_8000:8e3};var r=class{client;defaultVoiceId;modelId;constructor(t){this.client=t.client??new e({apiKey:t.apiKey}),this.defaultVoiceId=t.defaultVoiceId,this.modelId=t.modelId}async listVoices(){return((await this.client.voices.getAll()).voices??[]).map(e=>({id:e.voiceId??``,name:e.name??e.voiceId??``,description:e.description??void 0,language:e.labels?.language??void 0,gender:i(e.labels?.gender),previewUrl:e.previewUrl??void 0,metadata:{category:e.category??``,...e.labels}}))}async synthesize(e){let r=e.voiceId??this.defaultVoiceId;if(!r)throw Error(`Voice ID is required for ElevenLabs synthesis.`);let i=e.format??`mp3`,o=t[i]??t.mp3,s=e.sampleRateHz??n[o]??n.mp3_44100_128??44100,c=e.stability!=null||e.similarityBoost!=null||e.style!=null?{...e.stability==null?{}:{stability:e.stability},...e.similarityBoost==null?{}:{similarityBoost:e.similarityBoost},...e.style==null?{}:{style:e.style}}:void 0;return{audio:await a(await this.client.textToSpeech.convert(r,{text:e.text,modelId:this.modelId,outputFormat:o,voiceSettings:c})),format:i,sampleRateHz:s,durationSeconds:void 0,url:void 0}}};function i(e){if(!e)return;let t=e.toLowerCase();if(t===`male`||t===`female`||t===`neutral`)return t}async function a(e){let t=e.getReader(),n=[];for(;;){let{done:e,value:r}=await t.read();if(e)break;r&&n.push(r)}let r=n.reduce((e,t)=>e+t.length,0),i=new Uint8Array(r),a=0;for(let e of n)i.set(e,a),a+=e.length;return i}export{r as ElevenLabsVoiceProvider};
1
+ import { ElevenLabsClient } from "@elevenlabs/elevenlabs-js";
2
+
3
+ //#region src/integrations/providers/impls/elevenlabs-voice.ts
4
+ const FORMAT_MAP = {
5
+ mp3: "mp3_44100_128",
6
+ wav: "pcm_44100",
7
+ ogg: "mp3_44100_128",
8
+ pcm: "pcm_16000"
9
+ };
10
+ const SAMPLE_RATE = {
11
+ mp3_22050_32: 22050,
12
+ mp3_44100_32: 44100,
13
+ mp3_44100_64: 44100,
14
+ mp3_44100_96: 44100,
15
+ mp3_44100_128: 44100,
16
+ mp3_44100_192: 44100,
17
+ pcm_16000: 16e3,
18
+ pcm_22050: 22050,
19
+ pcm_24000: 24e3,
20
+ pcm_44100: 44100,
21
+ ulaw_8000: 8e3
22
+ };
23
+ var ElevenLabsVoiceProvider = class {
24
+ client;
25
+ defaultVoiceId;
26
+ modelId;
27
+ constructor(options) {
28
+ this.client = options.client ?? new ElevenLabsClient({ apiKey: options.apiKey });
29
+ this.defaultVoiceId = options.defaultVoiceId;
30
+ this.modelId = options.modelId;
31
+ }
32
+ async listVoices() {
33
+ return ((await this.client.voices.getAll()).voices ?? []).map((voice) => ({
34
+ id: voice.voiceId ?? "",
35
+ name: voice.name ?? voice.voiceId ?? "",
36
+ description: voice.description ?? void 0,
37
+ language: voice.labels?.language ?? void 0,
38
+ gender: normalizeGender(voice.labels?.gender),
39
+ previewUrl: voice.previewUrl ?? void 0,
40
+ metadata: {
41
+ category: voice.category ?? "",
42
+ ...voice.labels
43
+ }
44
+ }));
45
+ }
46
+ async synthesize(input) {
47
+ const voiceId = input.voiceId ?? this.defaultVoiceId;
48
+ if (!voiceId) throw new Error("Voice ID is required for ElevenLabs synthesis.");
49
+ const formatKey = input.format ?? "mp3";
50
+ const outputFormat = FORMAT_MAP[formatKey] ?? FORMAT_MAP.mp3;
51
+ const sampleRate = input.sampleRateHz ?? SAMPLE_RATE[outputFormat] ?? SAMPLE_RATE.mp3_44100_128 ?? 44100;
52
+ const voiceSettings = input.stability != null || input.similarityBoost != null || input.style != null ? {
53
+ ...input.stability != null ? { stability: input.stability } : {},
54
+ ...input.similarityBoost != null ? { similarityBoost: input.similarityBoost } : {},
55
+ ...input.style != null ? { style: input.style } : {}
56
+ } : void 0;
57
+ return {
58
+ audio: await readWebStream(await this.client.textToSpeech.convert(voiceId, {
59
+ text: input.text,
60
+ modelId: this.modelId,
61
+ outputFormat,
62
+ voiceSettings
63
+ })),
64
+ format: formatKey,
65
+ sampleRateHz: sampleRate,
66
+ durationSeconds: void 0,
67
+ url: void 0
68
+ };
69
+ }
70
+ };
71
+ function normalizeGender(value) {
72
+ if (!value) return void 0;
73
+ const normalized = value.toLowerCase();
74
+ if (normalized === "male" || normalized === "female" || normalized === "neutral") return normalized;
75
+ }
76
+ async function readWebStream(stream) {
77
+ const reader = stream.getReader();
78
+ const chunks = [];
79
+ while (true) {
80
+ const { done, value } = await reader.read();
81
+ if (done) break;
82
+ if (value) chunks.push(value);
83
+ }
84
+ const length = chunks.reduce((total, chunk) => total + chunk.length, 0);
85
+ const result = new Uint8Array(length);
86
+ let offset = 0;
87
+ for (const chunk of chunks) {
88
+ result.set(chunk, offset);
89
+ offset += chunk.length;
90
+ }
91
+ return result;
92
+ }
93
+
94
+ //#endregion
95
+ export { ElevenLabsVoiceProvider };
@@ -1 +1,88 @@
1
- import{Storage as e}from"@google-cloud/storage";var t=class{storage;bucketName;constructor(t){this.storage=t.storage??new e(t.clientOptions??void 0),this.bucketName=t.bucket}async putObject(e){let t=e.bucket??this.bucketName,i=this.storage.bucket(t).file(e.key),a=n(e.data);await i.save(a,{resumable:!1,contentType:e.contentType,metadata:e.metadata}),e.makePublic&&await i.makePublic();let[o]=await i.getMetadata();return r(o)}async getObject(e){let t=e.bucket??this.bucketName,n=this.storage.bucket(t).file(e.key),[i]=await n.exists();if(!i)return null;let[a]=await n.download(),[o]=await n.getMetadata();return{...r(o),data:new Uint8Array(a)}}async deleteObject(e){let t=e.bucket??this.bucketName;await this.storage.bucket(t).file(e.key).delete({ignoreNotFound:!0})}async generateSignedUrl(e){let t=e.bucket??this.bucketName,n=this.storage.bucket(t).file(e.key),r=e.method===`PUT`?`write`:`read`,i=Date.now()+e.expiresInSeconds*1e3,[a]=await n.getSignedUrl({action:r,expires:i,contentType:e.contentType});return{url:a,expiresAt:new Date(i)}}async listObjects(e){let t=e.bucket??this.bucketName,[n,i,a]=await this.storage.bucket(t).getFiles({prefix:e.prefix,maxResults:e.maxResults,pageToken:e.pageToken}),o=typeof i==`object`&&i&&`pageToken`in i?i.pageToken:void 0,s=a&&typeof a==`object`&&`nextPageToken`in a?a.nextPageToken:void 0;return{objects:n.map(e=>r(e.metadata)),nextPageToken:o??s??void 0}}};function n(e){return e instanceof Uint8Array,Buffer.from(e)}function r(e){return{bucket:e.bucket??``,key:e.name??``,sizeBytes:e.size?Number(e.size):void 0,contentType:e.contentType??void 0,etag:e.etag??void 0,checksum:e.md5Hash??void 0,lastModified:e.updated?new Date(e.updated):void 0,metadata:e.metadata}}export{t as GoogleCloudStorageProvider};
1
+ import { Storage } from "@google-cloud/storage";
2
+
3
+ //#region src/integrations/providers/impls/gcs-storage.ts
4
+ var GoogleCloudStorageProvider = class {
5
+ storage;
6
+ bucketName;
7
+ constructor(options) {
8
+ this.storage = options.storage ?? new Storage(options.clientOptions ?? void 0);
9
+ this.bucketName = options.bucket;
10
+ }
11
+ async putObject(input) {
12
+ const bucketName = input.bucket ?? this.bucketName;
13
+ const file = this.storage.bucket(bucketName).file(input.key);
14
+ const buffer = toBuffer(input.data);
15
+ await file.save(buffer, {
16
+ resumable: false,
17
+ contentType: input.contentType,
18
+ metadata: input.metadata
19
+ });
20
+ if (input.makePublic) await file.makePublic();
21
+ const [metadata] = await file.getMetadata();
22
+ return toMetadata(metadata);
23
+ }
24
+ async getObject(input) {
25
+ const bucketName = input.bucket ?? this.bucketName;
26
+ const file = this.storage.bucket(bucketName).file(input.key);
27
+ const [exists] = await file.exists();
28
+ if (!exists) return null;
29
+ const [contents] = await file.download();
30
+ const [metadata] = await file.getMetadata();
31
+ return {
32
+ ...toMetadata(metadata),
33
+ data: new Uint8Array(contents)
34
+ };
35
+ }
36
+ async deleteObject(input) {
37
+ const bucketName = input.bucket ?? this.bucketName;
38
+ await this.storage.bucket(bucketName).file(input.key).delete({ ignoreNotFound: true });
39
+ }
40
+ async generateSignedUrl(options) {
41
+ const bucketName = options.bucket ?? this.bucketName;
42
+ const file = this.storage.bucket(bucketName).file(options.key);
43
+ const action = options.method === "PUT" ? "write" : "read";
44
+ const expires = Date.now() + options.expiresInSeconds * 1e3;
45
+ const [url] = await file.getSignedUrl({
46
+ action,
47
+ expires,
48
+ contentType: options.contentType
49
+ });
50
+ return {
51
+ url,
52
+ expiresAt: new Date(expires)
53
+ };
54
+ }
55
+ async listObjects(query) {
56
+ const bucketName = query.bucket ?? this.bucketName;
57
+ const [files, nextQuery, response] = await this.storage.bucket(bucketName).getFiles({
58
+ prefix: query.prefix,
59
+ maxResults: query.maxResults,
60
+ pageToken: query.pageToken
61
+ });
62
+ const nextTokenFromQuery = typeof nextQuery === "object" && nextQuery !== null && "pageToken" in nextQuery ? nextQuery.pageToken : void 0;
63
+ const nextTokenFromResponse = response && typeof response === "object" && "nextPageToken" in response ? response.nextPageToken : void 0;
64
+ return {
65
+ objects: files.map((file) => toMetadata(file.metadata)),
66
+ nextPageToken: nextTokenFromQuery ?? nextTokenFromResponse ?? void 0
67
+ };
68
+ }
69
+ };
70
+ function toBuffer(data) {
71
+ if (data instanceof Uint8Array) return Buffer.from(data);
72
+ return Buffer.from(data);
73
+ }
74
+ function toMetadata(metadata) {
75
+ return {
76
+ bucket: metadata.bucket ?? "",
77
+ key: metadata.name ?? "",
78
+ sizeBytes: metadata.size ? Number(metadata.size) : void 0,
79
+ contentType: metadata.contentType ?? void 0,
80
+ etag: metadata.etag ?? void 0,
81
+ checksum: metadata.md5Hash ?? void 0,
82
+ lastModified: metadata.updated ? new Date(metadata.updated) : void 0,
83
+ metadata: metadata.metadata
84
+ };
85
+ }
86
+
87
+ //#endregion
88
+ export { GoogleCloudStorageProvider };
@@ -1 +1,200 @@
1
- import{google as e}from"googleapis";var t=class{gmail;userId;includeSpamTrash;auth;constructor(t){this.auth=t.auth,this.gmail=t.gmail??e.gmail({version:`v1`,auth:t.auth}),this.userId=t.userId??`me`,this.includeSpamTrash=t.includeSpamTrash??!1}async listThreads(e){let t=await this.gmail.users.threads.list({userId:this.userId,maxResults:e?.pageSize,pageToken:e?.pageToken,q:e?.query,labelIds:e?.label?[e.label]:void 0,includeSpamTrash:this.includeSpamTrash,auth:this.auth});return(await Promise.all((t.data.threads??[]).map(async e=>e.id?this.getThread(e.id):null))).filter(e=>e!==null)}async getThread(e){let t=(await this.gmail.users.threads.get({id:e,userId:this.userId,format:`full`,auth:this.auth})).data;if(!t)return null;let n=t.messages?.map(e=>this.transformMessage(e))??[],r=o(n.flatMap(e=>[e.from,...e.to,...e.cc??[]])),i=n[0],a=n[n.length-1],s=a?.receivedAt??a?.sentAt??i?.receivedAt??i?.sentAt??new Date,c=Array.from(new Set(n.flatMap(e=>{let t=e.metadata?.labelIds;return t?t.split(`,`).map(e=>e.trim()):[]}).filter(e=>!!e)));return{id:t.id??e,subject:n[0]?.subject,snippet:t.snippet??``,participants:r,messages:n,updatedAt:s,labels:c,metadata:t.historyId?{historyId:t.historyId}:void 0}}async listMessagesSince(e){let t=e.since?Math.floor(e.since.getTime()/1e3):void 0,n=[];t&&n.push(`after:${t}`);let r=await this.gmail.users.messages.list({userId:this.userId,maxResults:e.pageSize,pageToken:e.pageToken,labelIds:e.label?[e.label]:void 0,q:n.join(` `),includeSpamTrash:this.includeSpamTrash,auth:this.auth});return{messages:(await Promise.all((r.data.messages??[]).map(async e=>{if(!e.id)return null;let t=await this.gmail.users.messages.get({userId:this.userId,id:e.id,format:`full`,auth:this.auth});return t.data?this.transformMessage(t.data):null}))).filter(e=>e!==null),nextPageToken:r.data.nextPageToken??void 0}}transformMessage(e){let t=e.payload?.headers??[],o=n(t,`Subject`)??``,c=r(n(t,`From`))??i(`from`,e.id),l=a(n(t,`To`)),u=a(n(t,`Cc`)),d=a(n(t,`Bcc`)),f=r(n(t,`Reply-To`)),{text:p,html:m,attachments:h}=s(e.payload),g=e.internalDate?new Date(Number(e.internalDate)):new Date,_={...e.labelIds?.length?{labelIds:e.labelIds.join(`,`)}:{},...e.historyId?{historyId:e.historyId}:{}};return{id:e.id??``,threadId:e.threadId??``,subject:o,from:c,to:l,cc:u,bcc:d,replyTo:f??void 0,sentAt:g,receivedAt:g,textBody:p??void 0,htmlBody:m??void 0,attachments:h,headers:Object.fromEntries(t.map(e=>[e.name??``,e.value??``])),metadata:Object.keys(_).length>0?_:void 0}}};function n(e,t){let n=e.find(e=>e.name?.toLowerCase()===t.toLowerCase())?.value;return typeof n==`string`?n:void 0}function r(e){let t=a(e);return t.length===0?null:t[0]}function i(e,t){return{email:`${e}-${t&&t.replace(/[^\w]/g,``).slice(-8)||`unknown`}@mail.local`}}function a(e){return e?e.split(`,`).map(e=>e.trim()).filter(Boolean).map(e=>{let t=e.match(/^(?:"?([^"]*)"?\s)?<?([^<>]+)>?$/);if(!t)return{email:e};let n=t[1]?.trim(),r=t[2]?.trim();return r?n?{email:r,name:n}:{email:r}:{email:e}}):[]}function o(e){let t=new Map;for(let n of e)n&&t.set(n.email.toLowerCase(),n);return Array.from(t.values())}function s(e){if(!e)return{attachments:[]};let t=[],n=e=>{if(!e)return{};e.filename&&e.body?.attachmentId&&t.push({id:e.body.attachmentId,filename:e.filename,contentType:e.mimeType??`application/octet-stream`,sizeBytes:e.body.size??void 0});let r=e.mimeType??``,i=e.body?.data;return r===`text/plain`&&i?{text:c(i)}:r===`text/html`&&i?{html:c(i)}:e.parts?.length?e.parts.reduce((e,t)=>{let r=n(t);return{text:r.text??e.text,html:r.html??e.html}},{}):{}},{text:r,html:i}=n(e);return{text:r,html:i,attachments:t}}function c(e){let t=e.replace(/-/g,`+`).replace(/_/g,`/`),n=t.length%4,r=n===0?t:t+`=`.repeat(4-n);return Buffer.from(r,`base64`).toString(`utf-8`)}export{t as GmailInboundProvider};
1
+ import { google } from "googleapis";
2
+
3
+ //#region src/integrations/providers/impls/gmail-inbound.ts
4
+ var GmailInboundProvider = class {
5
+ gmail;
6
+ userId;
7
+ includeSpamTrash;
8
+ auth;
9
+ constructor(options) {
10
+ this.auth = options.auth;
11
+ this.gmail = options.gmail ?? google.gmail({
12
+ version: "v1",
13
+ auth: options.auth
14
+ });
15
+ this.userId = options.userId ?? "me";
16
+ this.includeSpamTrash = options.includeSpamTrash ?? false;
17
+ }
18
+ async listThreads(query) {
19
+ const response = await this.gmail.users.threads.list({
20
+ userId: this.userId,
21
+ maxResults: query?.pageSize,
22
+ pageToken: query?.pageToken,
23
+ q: query?.query,
24
+ labelIds: query?.label ? [query.label] : void 0,
25
+ includeSpamTrash: this.includeSpamTrash,
26
+ auth: this.auth
27
+ });
28
+ return (await Promise.all((response.data.threads ?? []).map(async (thread) => {
29
+ if (!thread.id) return null;
30
+ return this.getThread(thread.id);
31
+ }))).filter((thread) => thread !== null);
32
+ }
33
+ async getThread(threadId) {
34
+ const thread = (await this.gmail.users.threads.get({
35
+ id: threadId,
36
+ userId: this.userId,
37
+ format: "full",
38
+ auth: this.auth
39
+ })).data;
40
+ if (!thread) return null;
41
+ const messages = thread.messages?.map((message) => this.transformMessage(message)) ?? [];
42
+ const participants = dedupeAddresses(messages.flatMap((message) => [
43
+ message.from,
44
+ ...message.to,
45
+ ...message.cc ?? []
46
+ ]));
47
+ const firstMessage = messages[0];
48
+ const lastMessage = messages[messages.length - 1];
49
+ const updatedAt = lastMessage?.receivedAt ?? lastMessage?.sentAt ?? firstMessage?.receivedAt ?? firstMessage?.sentAt ?? /* @__PURE__ */ new Date();
50
+ const labels = Array.from(new Set(messages.flatMap((message) => {
51
+ const labelField = message.metadata?.labelIds;
52
+ if (!labelField) return [];
53
+ return labelField.split(",").map((label) => label.trim());
54
+ }).filter((label) => Boolean(label))));
55
+ return {
56
+ id: thread.id ?? threadId,
57
+ subject: messages[0]?.subject,
58
+ snippet: thread.snippet ?? "",
59
+ participants,
60
+ messages,
61
+ updatedAt,
62
+ labels,
63
+ metadata: thread.historyId ? { historyId: thread.historyId } : void 0
64
+ };
65
+ }
66
+ async listMessagesSince(query) {
67
+ const after = query.since ? Math.floor(query.since.getTime() / 1e3) : void 0;
68
+ const q = [];
69
+ if (after) q.push(`after:${after}`);
70
+ const response = await this.gmail.users.messages.list({
71
+ userId: this.userId,
72
+ maxResults: query.pageSize,
73
+ pageToken: query.pageToken,
74
+ labelIds: query.label ? [query.label] : void 0,
75
+ q: q.join(" "),
76
+ includeSpamTrash: this.includeSpamTrash,
77
+ auth: this.auth
78
+ });
79
+ return {
80
+ messages: (await Promise.all((response.data.messages ?? []).map(async (item) => {
81
+ if (!item.id) return null;
82
+ const full = await this.gmail.users.messages.get({
83
+ userId: this.userId,
84
+ id: item.id,
85
+ format: "full",
86
+ auth: this.auth
87
+ });
88
+ if (!full.data) return null;
89
+ return this.transformMessage(full.data);
90
+ }))).filter((message) => message !== null),
91
+ nextPageToken: response.data.nextPageToken ?? void 0
92
+ };
93
+ }
94
+ transformMessage(message) {
95
+ const headers = message.payload?.headers ?? [];
96
+ const subject = headerValue(headers, "Subject") ?? "";
97
+ const from = parseAddress(headerValue(headers, "From")) ?? inferFallbackAddress("from", message.id);
98
+ const to = parseAddressList(headerValue(headers, "To"));
99
+ const cc = parseAddressList(headerValue(headers, "Cc"));
100
+ const bcc = parseAddressList(headerValue(headers, "Bcc"));
101
+ const replyTo = parseAddress(headerValue(headers, "Reply-To"));
102
+ const { text, html, attachments } = extractContent(message.payload);
103
+ const timestamp = message.internalDate ? new Date(Number(message.internalDate)) : /* @__PURE__ */ new Date();
104
+ const metadata = {
105
+ ...message.labelIds?.length ? { labelIds: message.labelIds.join(",") } : {},
106
+ ...message.historyId ? { historyId: message.historyId } : {}
107
+ };
108
+ return {
109
+ id: message.id ?? "",
110
+ threadId: message.threadId ?? "",
111
+ subject,
112
+ from,
113
+ to,
114
+ cc,
115
+ bcc,
116
+ replyTo: replyTo ?? void 0,
117
+ sentAt: timestamp,
118
+ receivedAt: timestamp,
119
+ textBody: text ?? void 0,
120
+ htmlBody: html ?? void 0,
121
+ attachments,
122
+ headers: Object.fromEntries(headers.map((header) => [header.name ?? "", header.value ?? ""])),
123
+ metadata: Object.keys(metadata).length > 0 ? metadata : void 0
124
+ };
125
+ }
126
+ };
127
+ function headerValue(headers, name) {
128
+ const value = headers.find((candidate) => candidate.name?.toLowerCase() === name.toLowerCase())?.value;
129
+ return typeof value === "string" ? value : void 0;
130
+ }
131
+ function parseAddress(header) {
132
+ const addresses = parseAddressList(header);
133
+ if (addresses.length === 0) return null;
134
+ return addresses[0];
135
+ }
136
+ function inferFallbackAddress(field, messageId) {
137
+ return { email: `${field}-${messageId ? messageId.replace(/[^\w]/g, "").slice(-8) || "unknown" : "unknown"}@mail.local` };
138
+ }
139
+ function parseAddressList(header) {
140
+ if (!header) return [];
141
+ return header.split(",").map((part) => part.trim()).filter(Boolean).map((value) => {
142
+ const match = value.match(/^(?:"?([^"]*)"?\s)?<?([^<>]+)>?$/);
143
+ if (!match) return { email: value };
144
+ const name = match[1]?.trim();
145
+ const email = match[2]?.trim();
146
+ if (!email) return { email: value };
147
+ return name ? {
148
+ email,
149
+ name
150
+ } : { email };
151
+ });
152
+ }
153
+ function dedupeAddresses(addresses) {
154
+ const map = /* @__PURE__ */ new Map();
155
+ for (const address of addresses) {
156
+ if (!address) continue;
157
+ map.set(address.email.toLowerCase(), address);
158
+ }
159
+ return Array.from(map.values());
160
+ }
161
+ function extractContent(payload) {
162
+ if (!payload) return { attachments: [] };
163
+ const attachments = [];
164
+ const visit = (part) => {
165
+ if (!part) return {};
166
+ if (part.filename && part.body?.attachmentId) attachments.push({
167
+ id: part.body.attachmentId,
168
+ filename: part.filename,
169
+ contentType: part.mimeType ?? "application/octet-stream",
170
+ sizeBytes: part.body.size ?? void 0
171
+ });
172
+ const mimeType = part.mimeType ?? "";
173
+ const data = part.body?.data;
174
+ if (mimeType === "text/plain" && data) return { text: decodeBase64Url(data) };
175
+ if (mimeType === "text/html" && data) return { html: decodeBase64Url(data) };
176
+ if (part.parts?.length) return part.parts.reduce((acc, nested) => {
177
+ const value = visit(nested);
178
+ return {
179
+ text: value.text ?? acc.text,
180
+ html: value.html ?? acc.html
181
+ };
182
+ }, {});
183
+ return {};
184
+ };
185
+ const { text, html } = visit(payload);
186
+ return {
187
+ text,
188
+ html,
189
+ attachments
190
+ };
191
+ }
192
+ function decodeBase64Url(data) {
193
+ const normalized = data.replace(/-/g, "+").replace(/_/g, "/");
194
+ const padding = normalized.length % 4;
195
+ const padded = padding === 0 ? normalized : normalized + "=".repeat(4 - padding);
196
+ return Buffer.from(padded, "base64").toString("utf-8");
197
+ }
198
+
199
+ //#endregion
200
+ export { GmailInboundProvider };
@@ -1,6 +1,105 @@
1
- import{google as e}from"googleapis";var t=class{gmail;userId;auth;constructor(t){this.auth=t.auth,this.gmail=t.gmail??e.gmail({version:`v1`,auth:t.auth}),this.userId=t.userId??`me`}async sendEmail(e){let t=n(e),r=await this.gmail.users.messages.send({userId:this.userId,requestBody:{raw:t},auth:this.auth});return{id:r.data.id??``,providerMessageId:r.data.id??void 0,queuedAt:new Date}}};function n(e){let t=[`From: ${o(e.from)}`,`To: ${e.to.map(o).join(`, `)}`,`Subject: ${e.subject}`,`MIME-Version: 1.0`];e.cc?.length&&t.push(`Cc: ${e.cc.map(o).join(`, `)}`),e.replyTo&&t.push(`Reply-To: ${o(e.replyTo)}`),Object.entries(e.headers??{}).forEach(([e,n])=>{t.push(`${e}: ${n}`)});let n=e.attachments??[],s=!!e.htmlBody,c=!!e.textBody,l=`mixed_${Date.now()}`,u=`alt_${Date.now()}`,d=``;n.length>0?(t.push(`Content-Type: multipart/mixed; boundary="${l}"`),d+=`\r\n--${l}\r\n`,d+=r(c,s,u,e),n.forEach(e=>{d+=a(l,e)}),d+=`\r\n--${l}--`):c&&s?(t.push(`Content-Type: multipart/alternative; boundary="${u}"`),d+=`\r\n--${u}\r\n`,d+=i(`text/plain; charset="utf-8"`,e.textBody),d+=`\r\n--${u}\r\n`,d+=i(`text/html; charset="utf-8"`,e.htmlBody),d+=`\r\n--${u}--`):s?(t.push(`Content-Type: text/html; charset="utf-8"`),d+=`\r\n\r\n${e.htmlBody}`):(t.push(`Content-Type: text/plain; charset="utf-8"`),d+=`\r\n\r\n${e.textBody??``}`);let f=`${t.join(`\r
2
- `)}${d}`;return Buffer.from(f).toString(`base64`).replace(/\+/g,`-`).replace(/\//g,`_`).replace(/=+$/,``)}function r(e,t,n,r){let a=``;return a+=`Content-Type: multipart/alternative; boundary="${n}"\r\n`,a+=`\r
3
- `,e&&(a+=`--${n}\r\n`,a+=i(`text/plain; charset="utf-8"`,r.textBody)),t&&(a+=`\r\n--${n}\r\n`,a+=i(`text/html; charset="utf-8"`,r.htmlBody)),a+=`\r\n--${n}--`,a}function i(e,t){return`Content-Type: ${e}\r\nContent-Transfer-Encoding: 7bit\r
1
+ import { google } from "googleapis";
2
+
3
+ //#region src/integrations/providers/impls/gmail-outbound.ts
4
+ var GmailOutboundProvider = class {
5
+ gmail;
6
+ userId;
7
+ auth;
8
+ constructor(options) {
9
+ this.auth = options.auth;
10
+ this.gmail = options.gmail ?? google.gmail({
11
+ version: "v1",
12
+ auth: options.auth
13
+ });
14
+ this.userId = options.userId ?? "me";
15
+ }
16
+ async sendEmail(message) {
17
+ const raw = encodeMessage(message);
18
+ const response = await this.gmail.users.messages.send({
19
+ userId: this.userId,
20
+ requestBody: { raw },
21
+ auth: this.auth
22
+ });
23
+ return {
24
+ id: response.data.id ?? "",
25
+ providerMessageId: response.data.id ?? void 0,
26
+ queuedAt: /* @__PURE__ */ new Date()
27
+ };
28
+ }
29
+ };
30
+ function encodeMessage(message) {
31
+ const headers = [
32
+ `From: ${formatAddress(message.from)}`,
33
+ `To: ${message.to.map(formatAddress).join(", ")}`,
34
+ `Subject: ${message.subject}`,
35
+ "MIME-Version: 1.0"
36
+ ];
37
+ if (message.cc?.length) headers.push(`Cc: ${message.cc.map(formatAddress).join(", ")}`);
38
+ if (message.replyTo) headers.push(`Reply-To: ${formatAddress(message.replyTo)}`);
39
+ Object.entries(message.headers ?? {}).forEach(([key, value]) => {
40
+ headers.push(`${key}: ${value}`);
41
+ });
42
+ const attachments = message.attachments ?? [];
43
+ const hasHtml = Boolean(message.htmlBody);
44
+ const hasText = Boolean(message.textBody);
45
+ const boundaryMain = `mixed_${Date.now()}`;
46
+ const boundaryAlt = `alt_${Date.now()}`;
47
+ let body = "";
48
+ if (attachments.length > 0) {
49
+ headers.push(`Content-Type: multipart/mixed; boundary="${boundaryMain}"`);
50
+ body += `\r\n--${boundaryMain}\r\n`;
51
+ body += buildAlternativePart(hasText, hasHtml, boundaryAlt, message);
52
+ attachments.forEach((attachment) => {
53
+ body += buildAttachmentPart(boundaryMain, attachment);
54
+ });
55
+ body += `\r\n--${boundaryMain}--`;
56
+ } else if (hasText && hasHtml) {
57
+ headers.push(`Content-Type: multipart/alternative; boundary="${boundaryAlt}"`);
58
+ body += `\r\n--${boundaryAlt}\r\n`;
59
+ body += buildTextPart("text/plain; charset=\"utf-8\"", message.textBody);
60
+ body += `\r\n--${boundaryAlt}\r\n`;
61
+ body += buildTextPart("text/html; charset=\"utf-8\"", message.htmlBody);
62
+ body += `\r\n--${boundaryAlt}--`;
63
+ } else if (hasHtml) {
64
+ headers.push("Content-Type: text/html; charset=\"utf-8\"");
65
+ body += `\r\n\r\n${message.htmlBody}`;
66
+ } else {
67
+ headers.push("Content-Type: text/plain; charset=\"utf-8\"");
68
+ body += `\r\n\r\n${message.textBody ?? ""}`;
69
+ }
70
+ const mime = `${headers.join("\r\n")}${body}`;
71
+ return Buffer.from(mime).toString("base64").replace(/\+/g, "-").replace(/\//g, "_").replace(/=+$/, "");
72
+ }
73
+ function buildAlternativePart(hasText, hasHtml, boundary, message) {
74
+ let content = "";
75
+ content += `Content-Type: multipart/alternative; boundary="${boundary}"\r\n`;
76
+ content += "\r\n";
77
+ if (hasText) {
78
+ content += `--${boundary}\r\n`;
79
+ content += buildTextPart("text/plain; charset=\"utf-8\"", message.textBody);
80
+ }
81
+ if (hasHtml) {
82
+ content += `\r\n--${boundary}\r\n`;
83
+ content += buildTextPart("text/html; charset=\"utf-8\"", message.htmlBody);
84
+ }
85
+ content += `\r\n--${boundary}--`;
86
+ return content;
87
+ }
88
+ function buildTextPart(contentType, content) {
89
+ return `Content-Type: ${contentType}\r\nContent-Transfer-Encoding: 7bit\r
4
90
  \r
5
- `+t}function a(e,t){let n=t.data??new Uint8Array,r=n.byteLength>0?Buffer.from(n).toString(`base64`):``;return`\r\n--${e}\r\nContent-Type: ${t.contentType}; name="${t.filename}"\r\nContent-Transfer-Encoding: base64\r
6
- Content-Disposition: attachment; filename="${t.filename}"\r\n\r\n`+r}function o(e){return e.name?`"${e.name}" <${e.email}>`:e.email}export{t as GmailOutboundProvider};
91
+ ` + content;
92
+ }
93
+ function buildAttachmentPart(boundary, attachment) {
94
+ const data = attachment.data ?? new Uint8Array();
95
+ const encoded = data.byteLength > 0 ? Buffer.from(data).toString("base64") : "";
96
+ return `\r\n--${boundary}\r\nContent-Type: ${attachment.contentType}; name="${attachment.filename}"\r\nContent-Transfer-Encoding: base64\r
97
+ Content-Disposition: attachment; filename="${attachment.filename}"\r\n\r\n` + encoded;
98
+ }
99
+ function formatAddress(address) {
100
+ if (address.name) return `"${address.name}" <${address.email}>`;
101
+ return address.email;
102
+ }
103
+
104
+ //#endregion
105
+ export { GmailOutboundProvider };
@@ -1 +1,154 @@
1
- import{google as e}from"googleapis";var t=class{calendar;defaultCalendarId;auth;constructor(t){this.auth=t.auth,this.calendar=t.calendar??e.calendar({version:`v3`,auth:t.auth}),this.defaultCalendarId=t.calendarId??`primary`}async listEvents(e){let t=await this.calendar.events.list({calendarId:e.calendarId??this.defaultCalendarId,timeMin:e.timeMin?.toISOString(),timeMax:e.timeMax?.toISOString(),maxResults:e.maxResults,pageToken:e.pageToken,singleEvents:!0,orderBy:`startTime`,auth:this.auth});return{events:t.data.items?.map(t=>this.fromGoogleEvent(e.calendarId??this.defaultCalendarId,t))??[],nextPageToken:t.data.nextPageToken??void 0}}async createEvent(e){let t=e.calendarId??this.defaultCalendarId,n=await this.calendar.events.insert({calendarId:t,requestBody:this.toGoogleEvent(e),conferenceDataVersion:e.conference?.create?1:void 0,auth:this.auth});return this.fromGoogleEvent(t,n.data)}async updateEvent(e,t,n){let r=await this.calendar.events.patch({calendarId:e??this.defaultCalendarId,eventId:t,requestBody:this.toGoogleEvent(n),conferenceDataVersion:n.conference?.create?1:void 0,auth:this.auth});return this.fromGoogleEvent(e,r.data)}async deleteEvent(e,t){await this.calendar.events.delete({calendarId:e??this.defaultCalendarId,eventId:t,auth:this.auth})}fromGoogleEvent(e,t){let r=n(t.start),o=n(t.end),s=t.attendees?.map(e=>({email:e.email??``,name:e.displayName??void 0,optional:e.optional??void 0,responseStatus:i(e.responseStatus)}))??[],c=t.reminders?.overrides?.map(e=>({method:e.method??`popup`,minutesBeforeStart:e.minutes??0}))??[],l=a(t);return{id:t.id??``,calendarId:e,title:t.summary??``,description:t.description??void 0,location:t.location??void 0,start:r,end:o,allDay:t.start?.date?!0:void 0,attendees:s,reminders:c,conferenceLink:t.hangoutLink??t.conferenceData?.entryPoints?.find(e=>e.uri)?.uri??void 0,metadata:l,createdAt:t.created?new Date(t.created):void 0,updatedAt:t.updated?new Date(t.updated):void 0}}toGoogleEvent(e){let t={};return`title`in e&&e.title&&(t.summary=e.title),e.description!==void 0&&(t.description=e.description),e.location!==void 0&&(t.location=e.location),e.start&&(t.start=r(e.start,e.allDay)),e.end&&(t.end=r(e.end,e.allDay)),e.attendees&&(t.attendees=e.attendees.map(e=>({email:e.email,displayName:e.name,optional:e.optional,responseStatus:e.responseStatus}))),e.reminders&&(t.reminders={useDefault:!1,overrides:e.reminders.map(e=>({method:e.method,minutes:e.minutesBeforeStart}))}),e.conference?.create&&(t.conferenceData={createRequest:{requestId:`conf-${Date.now()}`}}),e.metadata&&(t.extendedProperties={...t.extendedProperties??{},private:{...t.extendedProperties?.private??{},...e.metadata}}),t}};function n(e){return e?e.dateTime?new Date(e.dateTime):e.date?new Date(`${e.date}T00:00:00`):new Date:new Date}function r(e,t){return t?{date:e.toISOString().slice(0,10)}:{dateTime:e.toISOString()}}function i(e){if(e)return[`needsAction`,`declined`,`tentative`,`accepted`].includes(e)?e:void 0}function a(e){let t={};return e.status&&(t.status=e.status),e.htmlLink&&(t.htmlLink=e.htmlLink),e.iCalUID&&(t.iCalUID=e.iCalUID),e.etag&&(t.etag=e.etag),e.conferenceData?.conferenceSolution?.name&&(t.conferenceSolution=e.conferenceData.conferenceSolution.name),e.extendedProperties?.private&&Object.entries(e.extendedProperties.private).forEach(([e,n])=>{typeof n==`string`&&(t[`extended.${e}`]=n)}),Object.keys(t).length>0?t:void 0}export{t as GoogleCalendarProvider};
1
+ import { google } from "googleapis";
2
+
3
+ //#region src/integrations/providers/impls/google-calendar.ts
4
+ var GoogleCalendarProvider = class {
5
+ calendar;
6
+ defaultCalendarId;
7
+ auth;
8
+ constructor(options) {
9
+ this.auth = options.auth;
10
+ this.calendar = options.calendar ?? google.calendar({
11
+ version: "v3",
12
+ auth: options.auth
13
+ });
14
+ this.defaultCalendarId = options.calendarId ?? "primary";
15
+ }
16
+ async listEvents(query) {
17
+ const response = await this.calendar.events.list({
18
+ calendarId: query.calendarId ?? this.defaultCalendarId,
19
+ timeMin: query.timeMin?.toISOString(),
20
+ timeMax: query.timeMax?.toISOString(),
21
+ maxResults: query.maxResults,
22
+ pageToken: query.pageToken,
23
+ singleEvents: true,
24
+ orderBy: "startTime",
25
+ auth: this.auth
26
+ });
27
+ return {
28
+ events: response.data.items?.map((item) => this.fromGoogleEvent(query.calendarId ?? this.defaultCalendarId, item)) ?? [],
29
+ nextPageToken: response.data.nextPageToken ?? void 0
30
+ };
31
+ }
32
+ async createEvent(input) {
33
+ const calendarId = input.calendarId ?? this.defaultCalendarId;
34
+ const response = await this.calendar.events.insert({
35
+ calendarId,
36
+ requestBody: this.toGoogleEvent(input),
37
+ conferenceDataVersion: input.conference?.create ? 1 : void 0,
38
+ auth: this.auth
39
+ });
40
+ return this.fromGoogleEvent(calendarId, response.data);
41
+ }
42
+ async updateEvent(calendarId, eventId, input) {
43
+ const response = await this.calendar.events.patch({
44
+ calendarId: calendarId ?? this.defaultCalendarId,
45
+ eventId,
46
+ requestBody: this.toGoogleEvent(input),
47
+ conferenceDataVersion: input.conference?.create ? 1 : void 0,
48
+ auth: this.auth
49
+ });
50
+ return this.fromGoogleEvent(calendarId, response.data);
51
+ }
52
+ async deleteEvent(calendarId, eventId) {
53
+ await this.calendar.events.delete({
54
+ calendarId: calendarId ?? this.defaultCalendarId,
55
+ eventId,
56
+ auth: this.auth
57
+ });
58
+ }
59
+ fromGoogleEvent(calendarId, event) {
60
+ const start = parseDateTime(event.start);
61
+ const end = parseDateTime(event.end);
62
+ const attendees = event.attendees?.map((attendee) => ({
63
+ email: attendee.email ?? "",
64
+ name: attendee.displayName ?? void 0,
65
+ optional: attendee.optional ?? void 0,
66
+ responseStatus: normalizeResponseStatus(attendee.responseStatus)
67
+ })) ?? [];
68
+ const reminders = event.reminders?.overrides?.map((reminder) => ({
69
+ method: reminder.method ?? "popup",
70
+ minutesBeforeStart: reminder.minutes ?? 0
71
+ })) ?? [];
72
+ const metadata = buildMetadata(event);
73
+ return {
74
+ id: event.id ?? "",
75
+ calendarId,
76
+ title: event.summary ?? "",
77
+ description: event.description ?? void 0,
78
+ location: event.location ?? void 0,
79
+ start,
80
+ end,
81
+ allDay: event.start?.date ? true : void 0,
82
+ attendees,
83
+ reminders,
84
+ conferenceLink: event.hangoutLink ?? event.conferenceData?.entryPoints?.find((entry) => entry.uri)?.uri ?? void 0,
85
+ metadata,
86
+ createdAt: event.created ? new Date(event.created) : void 0,
87
+ updatedAt: event.updated ? new Date(event.updated) : void 0
88
+ };
89
+ }
90
+ toGoogleEvent(input) {
91
+ const event = {};
92
+ if ("title" in input && input.title) event.summary = input.title;
93
+ if (input.description !== void 0) event.description = input.description;
94
+ if (input.location !== void 0) event.location = input.location;
95
+ if (input.start) event.start = formatDateTime(input.start, input.allDay);
96
+ if (input.end) event.end = formatDateTime(input.end, input.allDay);
97
+ if (input.attendees) event.attendees = input.attendees.map((attendee) => ({
98
+ email: attendee.email,
99
+ displayName: attendee.name,
100
+ optional: attendee.optional,
101
+ responseStatus: attendee.responseStatus
102
+ }));
103
+ if (input.reminders) event.reminders = {
104
+ useDefault: false,
105
+ overrides: input.reminders.map((reminder) => ({
106
+ method: reminder.method,
107
+ minutes: reminder.minutesBeforeStart
108
+ }))
109
+ };
110
+ if (input.conference?.create) event.conferenceData = { createRequest: { requestId: `conf-${Date.now()}` } };
111
+ if (input.metadata) event.extendedProperties = {
112
+ ...event.extendedProperties ?? {},
113
+ private: {
114
+ ...event.extendedProperties?.private ?? {},
115
+ ...input.metadata
116
+ }
117
+ };
118
+ return event;
119
+ }
120
+ };
121
+ function parseDateTime(time) {
122
+ if (!time) return /* @__PURE__ */ new Date();
123
+ if (time.dateTime) return new Date(time.dateTime);
124
+ if (time.date) return /* @__PURE__ */ new Date(`${time.date}T00:00:00`);
125
+ return /* @__PURE__ */ new Date();
126
+ }
127
+ function formatDateTime(date, allDay) {
128
+ if (allDay) return { date: date.toISOString().slice(0, 10) };
129
+ return { dateTime: date.toISOString() };
130
+ }
131
+ function normalizeResponseStatus(status) {
132
+ if (!status) return void 0;
133
+ return [
134
+ "needsAction",
135
+ "declined",
136
+ "tentative",
137
+ "accepted"
138
+ ].includes(status) ? status : void 0;
139
+ }
140
+ function buildMetadata(event) {
141
+ const metadata = {};
142
+ if (event.status) metadata.status = event.status;
143
+ if (event.htmlLink) metadata.htmlLink = event.htmlLink;
144
+ if (event.iCalUID) metadata.iCalUID = event.iCalUID;
145
+ if (event.etag) metadata.etag = event.etag;
146
+ if (event.conferenceData?.conferenceSolution?.name) metadata.conferenceSolution = event.conferenceData.conferenceSolution.name;
147
+ if (event.extendedProperties?.private) Object.entries(event.extendedProperties.private).forEach(([key, value]) => {
148
+ if (typeof value === "string") metadata[`extended.${key}`] = value;
149
+ });
150
+ return Object.keys(metadata).length > 0 ? metadata : void 0;
151
+ }
152
+
153
+ //#endregion
154
+ export { GoogleCalendarProvider };