@lssm/integration.providers-impls 0.0.0-canary-20251217083314 → 0.0.0-canary-20251220002821
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +21 -0
- package/dist/_virtual/rolldown_runtime.js +1 -7
- package/dist/calendar.d.ts +3 -2
- package/dist/calendar.d.ts.map +1 -0
- package/dist/email.d.ts +3 -2
- package/dist/email.d.ts.map +1 -0
- package/dist/embedding.d.ts +3 -2
- package/dist/embedding.d.ts.map +1 -0
- package/dist/impls/elevenlabs-voice.d.ts +2 -1
- package/dist/impls/elevenlabs-voice.d.ts.map +1 -0
- package/dist/impls/elevenlabs-voice.js +8 -10
- package/dist/impls/elevenlabs-voice.js.map +1 -0
- package/dist/impls/gcs-storage.d.ts +2 -1
- package/dist/impls/gcs-storage.d.ts.map +1 -0
- package/dist/impls/gcs-storage.js +11 -9
- package/dist/impls/gcs-storage.js.map +1 -0
- package/dist/impls/gmail-inbound.d.ts +2 -1
- package/dist/impls/gmail-inbound.d.ts.map +1 -0
- package/dist/impls/gmail-inbound.js +3 -2
- package/dist/impls/gmail-inbound.js.map +1 -0
- package/dist/impls/gmail-outbound.d.ts +2 -1
- package/dist/impls/gmail-outbound.d.ts.map +1 -0
- package/dist/impls/gmail-outbound.js +6 -5
- package/dist/impls/gmail-outbound.js.map +1 -0
- package/dist/impls/google-calendar.d.ts +2 -1
- package/dist/impls/google-calendar.d.ts.map +1 -0
- package/dist/impls/google-calendar.js +2 -1
- package/dist/impls/google-calendar.js.map +1 -0
- package/dist/impls/mistral-embedding.d.ts +2 -1
- package/dist/impls/mistral-embedding.d.ts.map +1 -0
- package/dist/impls/mistral-embedding.js +2 -1
- package/dist/impls/mistral-embedding.js.map +1 -0
- package/dist/impls/mistral-llm.d.ts +3 -2
- package/dist/impls/mistral-llm.d.ts.map +1 -0
- package/dist/impls/mistral-llm.js +3 -2
- package/dist/impls/mistral-llm.js.map +1 -0
- package/dist/impls/postmark-email.d.ts +2 -1
- package/dist/impls/postmark-email.d.ts.map +1 -0
- package/dist/impls/postmark-email.js +2 -1
- package/dist/impls/postmark-email.js.map +1 -0
- package/dist/impls/powens-client.d.ts +2 -1
- package/dist/impls/powens-client.d.ts.map +1 -0
- package/dist/impls/powens-client.js +2 -1
- package/dist/impls/powens-client.js.map +1 -0
- package/dist/impls/powens-openbanking.d.ts +3 -2
- package/dist/impls/powens-openbanking.d.ts.map +1 -0
- package/dist/impls/powens-openbanking.js +3 -2
- package/dist/impls/powens-openbanking.js.map +1 -0
- package/dist/impls/provider-factory.d.ts +2 -1
- package/dist/impls/provider-factory.d.ts.map +1 -0
- package/dist/impls/provider-factory.js +3 -2
- package/dist/impls/provider-factory.js.map +1 -0
- package/dist/impls/qdrant-vector.d.ts +2 -1
- package/dist/impls/qdrant-vector.d.ts.map +1 -0
- package/dist/impls/qdrant-vector.js +7 -4
- package/dist/impls/qdrant-vector.js.map +1 -0
- package/dist/impls/stripe-payments.d.ts +2 -1
- package/dist/impls/stripe-payments.d.ts.map +1 -0
- package/dist/impls/stripe-payments.js +2 -1
- package/dist/impls/stripe-payments.js.map +1 -0
- package/dist/impls/twilio-sms.d.ts +2 -1
- package/dist/impls/twilio-sms.d.ts.map +1 -0
- package/dist/impls/twilio-sms.js +2 -1
- package/dist/impls/twilio-sms.js.map +1 -0
- package/dist/index.d.ts +2 -1
- package/dist/index.d.ts.map +1 -0
- package/dist/llm.d.ts +3 -2
- package/dist/llm.d.ts.map +1 -0
- package/dist/openbanking.d.ts +3 -2
- package/dist/openbanking.d.ts.map +1 -0
- package/dist/payments.d.ts +3 -2
- package/dist/payments.d.ts.map +1 -0
- package/dist/runtime/dist/secrets/provider.js +2 -1
- package/dist/runtime/dist/secrets/provider.js.map +1 -0
- package/dist/sms.d.ts +3 -2
- package/dist/sms.d.ts.map +1 -0
- package/dist/storage.d.ts +3 -2
- package/dist/storage.d.ts.map +1 -0
- package/dist/vector-store.d.ts +3 -2
- package/dist/vector-store.d.ts.map +1 -0
- package/dist/voice.d.ts +3 -2
- package/dist/voice.d.ts.map +1 -0
- package/package.json +16 -9
package/LICENSE
ADDED
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
MIT License
|
|
2
|
+
|
|
3
|
+
Copyright (c) 2025 Chaman Ventures, SASU
|
|
4
|
+
|
|
5
|
+
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
6
|
+
of this software and associated documentation files (the "Software"), to deal
|
|
7
|
+
in the Software without restriction, including without limitation the rights
|
|
8
|
+
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
9
|
+
copies of the Software, and to permit persons to whom the Software is
|
|
10
|
+
furnished to do so, subject to the following conditions:
|
|
11
|
+
|
|
12
|
+
The above copyright notice and this permission notice shall be included in all
|
|
13
|
+
copies or substantial portions of the Software.
|
|
14
|
+
|
|
15
|
+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
16
|
+
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
17
|
+
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
18
|
+
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
19
|
+
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
20
|
+
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
21
|
+
SOFTWARE.
|
|
@@ -30,13 +30,7 @@ var __copyProps = (to, from, except, desc) => {
|
|
|
30
30
|
}
|
|
31
31
|
return to;
|
|
32
32
|
};
|
|
33
|
-
var __reExport = (target, mod, secondTarget,
|
|
34
|
-
if (symbols) {
|
|
35
|
-
__defProp(target, Symbol.toStringTag, { value: "Module" });
|
|
36
|
-
secondTarget && __defProp(secondTarget, Symbol.toStringTag, { value: "Module" });
|
|
37
|
-
}
|
|
38
|
-
__copyProps(target, mod, "default"), secondTarget && __copyProps(secondTarget, mod, "default");
|
|
39
|
-
};
|
|
33
|
+
var __reExport = (target, mod, secondTarget) => (__copyProps(target, mod, "default"), secondTarget && __copyProps(secondTarget, mod, "default"));
|
|
40
34
|
|
|
41
35
|
//#endregion
|
|
42
36
|
export { __export, __reExport };
|
package/dist/calendar.d.ts
CHANGED
|
@@ -2,6 +2,7 @@ export * from "@lssm/lib.contracts/integrations/providers/calendar";
|
|
|
2
2
|
|
|
3
3
|
//#region src/calendar.d.ts
|
|
4
4
|
|
|
5
|
-
import * as
|
|
5
|
+
import * as import__lssm_lib_contracts_integrations_providers_calendar from "@lssm/lib.contracts/integrations/providers/calendar";
|
|
6
6
|
//#endregion
|
|
7
|
-
export {
|
|
7
|
+
export { import__lssm_lib_contracts_integrations_providers_calendar as calendar_d_exports };
|
|
8
|
+
//# sourceMappingURL=calendar.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"calendar.d.ts","names":[],"sources":["../src/calendar.ts"],"sourcesContent":[],"mappings":""}
|
package/dist/email.d.ts
CHANGED
|
@@ -2,6 +2,7 @@ export * from "@lssm/lib.contracts/integrations/providers/email";
|
|
|
2
2
|
|
|
3
3
|
//#region src/email.d.ts
|
|
4
4
|
|
|
5
|
-
import * as
|
|
5
|
+
import * as import__lssm_lib_contracts_integrations_providers_email from "@lssm/lib.contracts/integrations/providers/email";
|
|
6
6
|
//#endregion
|
|
7
|
-
export {
|
|
7
|
+
export { import__lssm_lib_contracts_integrations_providers_email as email_d_exports };
|
|
8
|
+
//# sourceMappingURL=email.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"email.d.ts","names":[],"sources":["../src/email.ts"],"sourcesContent":[],"mappings":""}
|
package/dist/embedding.d.ts
CHANGED
|
@@ -2,6 +2,7 @@ export * from "@lssm/lib.contracts/integrations/providers/embedding";
|
|
|
2
2
|
|
|
3
3
|
//#region src/embedding.d.ts
|
|
4
4
|
|
|
5
|
-
import * as
|
|
5
|
+
import * as import__lssm_lib_contracts_integrations_providers_embedding from "@lssm/lib.contracts/integrations/providers/embedding";
|
|
6
6
|
//#endregion
|
|
7
|
-
export {
|
|
7
|
+
export { import__lssm_lib_contracts_integrations_providers_embedding as embedding_d_exports };
|
|
8
|
+
//# sourceMappingURL=embedding.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"embedding.d.ts","names":[],"sources":["../src/embedding.ts"],"sourcesContent":[],"mappings":""}
|
|
@@ -17,4 +17,5 @@ declare class ElevenLabsVoiceProvider implements voice_d_exports.VoiceProvider {
|
|
|
17
17
|
synthesize(input: voice_d_exports.VoiceSynthesisInput): Promise<voice_d_exports.VoiceSynthesisResult>;
|
|
18
18
|
}
|
|
19
19
|
//#endregion
|
|
20
|
-
export { ElevenLabsVoiceProvider, ElevenLabsVoiceProviderOptions };
|
|
20
|
+
export { ElevenLabsVoiceProvider, ElevenLabsVoiceProviderOptions };
|
|
21
|
+
//# sourceMappingURL=elevenlabs-voice.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"elevenlabs-voice.d.ts","names":[],"sources":["../../src/impls/elevenlabs-voice.ts"],"sourcesContent":[],"mappings":";;;;UAUiB,8BAAA;;EAAA,cAAA,CAAA,EAAA,MAAA;EAiCJ,OAAA,CAAA,EAAA,MAAA;EAKU,MAAA,CAAA,EAlCZ,gBAkCY;;AAUD,cAfT,uBAAA,YAAmC,eAAA,CAAA,aAe1B,CAAA;EAkBI,iBAAA,MAAA;EAA8B,iBAAA,cAAA;EAAR,iBAAA,OAAA;EAjCA,WAAA,CAAA,OAAA,EAKzB,8BALyB;EAAa,UAAA,CAAA,CAAA,EAevC,OAfuC,CAe/B,eAAA,CAAA,KAf+B,EAAA,CAAA;oBAiCnC,eAAA,CAAA,sBAAsB,QAAQ,eAAA,CAAA"}
|
|
@@ -32,14 +32,16 @@ var ElevenLabsVoiceProvider = class {
|
|
|
32
32
|
async listVoices() {
|
|
33
33
|
return ((await this.client.voices.getAll()).voices ?? []).map((voice) => ({
|
|
34
34
|
id: voice.voiceId ?? "",
|
|
35
|
-
name: voice.name ??
|
|
35
|
+
name: voice.name ?? "",
|
|
36
36
|
description: voice.description ?? void 0,
|
|
37
37
|
language: voice.labels?.language ?? void 0,
|
|
38
|
-
gender: normalizeGender(voice.labels?.gender),
|
|
39
|
-
previewUrl: voice.previewUrl ?? void 0,
|
|
40
38
|
metadata: {
|
|
41
39
|
category: voice.category ?? "",
|
|
42
|
-
|
|
40
|
+
previewUrl: voice.previewUrl ?? void 0,
|
|
41
|
+
...(() => {
|
|
42
|
+
const { language, ...rest } = voice.labels ?? {};
|
|
43
|
+
return rest;
|
|
44
|
+
})()
|
|
43
45
|
}
|
|
44
46
|
}));
|
|
45
47
|
}
|
|
@@ -68,11 +70,6 @@ var ElevenLabsVoiceProvider = class {
|
|
|
68
70
|
};
|
|
69
71
|
}
|
|
70
72
|
};
|
|
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
73
|
async function readWebStream(stream) {
|
|
77
74
|
const reader = stream.getReader();
|
|
78
75
|
const chunks = [];
|
|
@@ -92,4 +89,5 @@ async function readWebStream(stream) {
|
|
|
92
89
|
}
|
|
93
90
|
|
|
94
91
|
//#endregion
|
|
95
|
-
export { ElevenLabsVoiceProvider };
|
|
92
|
+
export { ElevenLabsVoiceProvider };
|
|
93
|
+
//# sourceMappingURL=elevenlabs-voice.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"elevenlabs-voice.js","names":["FORMAT_MAP: Record<\n NonNullable<VoiceSynthesisInput['format']>,\n ElevenLabs.TextToSpeechConvertRequestOutputFormat\n>","SAMPLE_RATE: Partial<\n Record<ElevenLabs.TextToSpeechConvertRequestOutputFormat, number>\n>","chunks: Uint8Array[]"],"sources":["../../src/impls/elevenlabs-voice.ts"],"sourcesContent":["import { ElevenLabsClient } from '@elevenlabs/elevenlabs-js';\nimport type { ElevenLabs } from '@elevenlabs/elevenlabs-js';\n\nimport type {\n Voice,\n VoiceProvider,\n VoiceSynthesisInput,\n VoiceSynthesisResult,\n} from '../voice';\n\nexport interface ElevenLabsVoiceProviderOptions {\n apiKey: string;\n defaultVoiceId?: string;\n modelId?: string;\n client?: ElevenLabsClient;\n}\n\nconst FORMAT_MAP: Record<\n NonNullable<VoiceSynthesisInput['format']>,\n ElevenLabs.TextToSpeechConvertRequestOutputFormat\n> = {\n mp3: 'mp3_44100_128',\n wav: 'pcm_44100',\n ogg: 'mp3_44100_128',\n pcm: 'pcm_16000',\n};\n\nconst SAMPLE_RATE: Partial<\n Record<ElevenLabs.TextToSpeechConvertRequestOutputFormat, number>\n> = {\n mp3_22050_32: 22050,\n mp3_44100_32: 44100,\n mp3_44100_64: 44100,\n mp3_44100_96: 44100,\n mp3_44100_128: 44100,\n mp3_44100_192: 44100,\n pcm_16000: 16000,\n pcm_22050: 22050,\n pcm_24000: 24000,\n pcm_44100: 44100,\n ulaw_8000: 8000,\n};\n\nexport class ElevenLabsVoiceProvider implements VoiceProvider {\n private readonly client: ElevenLabsClient;\n private readonly defaultVoiceId?: string;\n private readonly modelId?: string;\n\n constructor(options: ElevenLabsVoiceProviderOptions) {\n this.client =\n options.client ??\n new ElevenLabsClient({\n apiKey: options.apiKey,\n });\n this.defaultVoiceId = options.defaultVoiceId;\n this.modelId = options.modelId;\n }\n\n async listVoices(): Promise<Voice[]> {\n const response = await this.client.voices.getAll();\n return (response.voices ?? []).map((voice) => ({\n id: voice.voiceId ?? '',\n name: voice.name ?? '',\n description: voice.description ?? undefined,\n language: voice.labels?.language ?? undefined,\n metadata: {\n category: voice.category ?? '',\n previewUrl: voice.previewUrl ?? undefined,\n ...(() => {\n const { language, ...rest } = voice.labels ?? {};\n return rest;\n })(),\n },\n })) as any[];\n }\n\n async synthesize(input: VoiceSynthesisInput): Promise<VoiceSynthesisResult> {\n const voiceId = input.voiceId ?? this.defaultVoiceId;\n if (!voiceId) {\n throw new Error('Voice ID is required for ElevenLabs synthesis.');\n }\n\n const formatKey = input.format ?? 'mp3';\n const outputFormat = FORMAT_MAP[formatKey] ?? FORMAT_MAP.mp3;\n const sampleRate =\n input.sampleRateHz ??\n SAMPLE_RATE[outputFormat] ??\n SAMPLE_RATE.mp3_44100_128 ??\n 44100;\n\n const voiceSettings =\n input.stability != null ||\n input.similarityBoost != null ||\n input.style != null\n ? {\n ...(input.stability != null ? { stability: input.stability } : {}),\n ...(input.similarityBoost != null\n ? { similarityBoost: input.similarityBoost }\n : {}),\n ...(input.style != null ? { style: input.style } : {}),\n }\n : undefined;\n\n const stream = await this.client.textToSpeech.convert(voiceId, {\n text: input.text,\n modelId: this.modelId,\n outputFormat,\n voiceSettings,\n });\n\n const audio = await readWebStream(stream);\n\n return {\n audio,\n format: formatKey,\n sampleRateHz: sampleRate,\n durationSeconds: undefined,\n url: undefined,\n };\n }\n}\n\nasync function readWebStream(\n stream: ReadableStream<Uint8Array>\n): Promise<Uint8Array> {\n const reader = stream.getReader();\n const chunks: Uint8Array[] = [];\n\n while (true) {\n const { done, value } = await reader.read();\n if (done) break;\n if (value) {\n chunks.push(value);\n }\n }\n\n const length = chunks.reduce((total, chunk) => total + chunk.length, 0);\n const result = new Uint8Array(length);\n let offset = 0;\n for (const chunk of chunks) {\n result.set(chunk, offset);\n offset += chunk.length;\n }\n return result;\n}\n"],"mappings":";;;AAiBA,MAAMA,aAGF;CACF,KAAK;CACL,KAAK;CACL,KAAK;CACL,KAAK;CACN;AAED,MAAMC,cAEF;CACF,cAAc;CACd,cAAc;CACd,cAAc;CACd,cAAc;CACd,eAAe;CACf,eAAe;CACf,WAAW;CACX,WAAW;CACX,WAAW;CACX,WAAW;CACX,WAAW;CACZ;AAED,IAAa,0BAAb,MAA8D;CAC5D,AAAiB;CACjB,AAAiB;CACjB,AAAiB;CAEjB,YAAY,SAAyC;AACnD,OAAK,SACH,QAAQ,UACR,IAAI,iBAAiB,EACnB,QAAQ,QAAQ,QACjB,CAAC;AACJ,OAAK,iBAAiB,QAAQ;AAC9B,OAAK,UAAU,QAAQ;;CAGzB,MAAM,aAA+B;AAEnC,WADiB,MAAM,KAAK,OAAO,OAAO,QAAQ,EACjC,UAAU,EAAE,EAAE,KAAK,WAAW;GAC7C,IAAI,MAAM,WAAW;GACrB,MAAM,MAAM,QAAQ;GACpB,aAAa,MAAM,eAAe;GAClC,UAAU,MAAM,QAAQ,YAAY;GACpC,UAAU;IACR,UAAU,MAAM,YAAY;IAC5B,YAAY,MAAM,cAAc;IAChC,UAAU;KACR,MAAM,EAAE,UAAU,GAAG,SAAS,MAAM,UAAU,EAAE;AAChD,YAAO;QACL;IACL;GACF,EAAE;;CAGL,MAAM,WAAW,OAA2D;EAC1E,MAAM,UAAU,MAAM,WAAW,KAAK;AACtC,MAAI,CAAC,QACH,OAAM,IAAI,MAAM,iDAAiD;EAGnE,MAAM,YAAY,MAAM,UAAU;EAClC,MAAM,eAAe,WAAW,cAAc,WAAW;EACzD,MAAM,aACJ,MAAM,gBACN,YAAY,iBACZ,YAAY,iBACZ;EAEF,MAAM,gBACJ,MAAM,aAAa,QACnB,MAAM,mBAAmB,QACzB,MAAM,SAAS,OACX;GACE,GAAI,MAAM,aAAa,OAAO,EAAE,WAAW,MAAM,WAAW,GAAG,EAAE;GACjE,GAAI,MAAM,mBAAmB,OACzB,EAAE,iBAAiB,MAAM,iBAAiB,GAC1C,EAAE;GACN,GAAI,MAAM,SAAS,OAAO,EAAE,OAAO,MAAM,OAAO,GAAG,EAAE;GACtD,GACD;AAWN,SAAO;GACL,OAHY,MAAM,cAPL,MAAM,KAAK,OAAO,aAAa,QAAQ,SAAS;IAC7D,MAAM,MAAM;IACZ,SAAS,KAAK;IACd;IACA;IACD,CAAC,CAEuC;GAIvC,QAAQ;GACR,cAAc;GACd,iBAAiB;GACjB,KAAK;GACN;;;AAIL,eAAe,cACb,QACqB;CACrB,MAAM,SAAS,OAAO,WAAW;CACjC,MAAMC,SAAuB,EAAE;AAE/B,QAAO,MAAM;EACX,MAAM,EAAE,MAAM,UAAU,MAAM,OAAO,MAAM;AAC3C,MAAI,KAAM;AACV,MAAI,MACF,QAAO,KAAK,MAAM;;CAItB,MAAM,SAAS,OAAO,QAAQ,OAAO,UAAU,QAAQ,MAAM,QAAQ,EAAE;CACvE,MAAM,SAAS,IAAI,WAAW,OAAO;CACrC,IAAI,SAAS;AACb,MAAK,MAAM,SAAS,QAAQ;AAC1B,SAAO,IAAI,OAAO,OAAO;AACzB,YAAU,MAAM;;AAElB,QAAO"}
|
|
@@ -21,4 +21,5 @@ declare class GoogleCloudStorageProvider implements storage_d_exports.ObjectStor
|
|
|
21
21
|
listObjects(query: storage_d_exports.ListObjectsQuery): Promise<storage_d_exports.ListObjectsResult>;
|
|
22
22
|
}
|
|
23
23
|
//#endregion
|
|
24
|
-
export { GoogleCloudStorageProvider, GoogleCloudStorageProviderOptions };
|
|
24
|
+
export { GoogleCloudStorageProvider, GoogleCloudStorageProviderOptions };
|
|
25
|
+
//# sourceMappingURL=gcs-storage.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"gcs-storage.d.ts","names":[],"sources":["../../src/impls/gcs-storage.ts"],"sourcesContent":[],"mappings":";;;;UAaiB,iCAAA;;EAAA,OAAA,CAAA,EAEL,OAFK;EAMJ,aAAA,CAAA,EAHK,cAGsB;;AAUf,cAVZ,0BAAA,YAAsC,iBAAA,CAAA,qBAU1B,CAAA;EAAyB,iBAAA,OAAA;EAAR,iBAAA,UAAA;EAiBjB,WAAA,CAAA,OAAA,EAvBF,iCAuBE;EAA4B,SAAA,CAAA,KAAA,EAjB5B,iBAAA,CAAA,cAiB4B,CAAA,EAjBX,OAiBW,CAjBH,iBAAA,CAAA,qBAiBG,CAAA;EAAR,SAAA,CAAA,KAAA,EAApB,iBAAA,CAAA,iBAAoB,CAAA,EAAA,OAAA,CAAQ,iBAAA,CAAA,eAAR,GAAA,IAAA,CAAA;EAcjB,YAAA,CAAA,KAAA,EAAA,iBAAA,CAAA,iBAAA,CAAA,EAAoB,OAApB,CAAA,IAAA,CAAA;EAAoB,iBAAA,CAAA,OAAA,EAOb,iBAAA,CAAA,gBAPa,CAAA,EAOM,OAPN,CAAA;IAOb,GAAA,EAAA,MAAA;IAEpB,SAAA,EAAA,IAAA;EAFuC,CAAA,CAAA;EAiB3B,WAAA,CAAA,KAAA,EAAA,iBAAA,CAAA,gBAAA,CAAA,EAAmB,OAAnB,CAA2B,iBAAA,CAAA,iBAA3B,CAAA"}
|
|
@@ -72,17 +72,19 @@ function toBuffer(data) {
|
|
|
72
72
|
return Buffer.from(data);
|
|
73
73
|
}
|
|
74
74
|
function toMetadata(metadata) {
|
|
75
|
+
const meta = metadata;
|
|
75
76
|
return {
|
|
76
|
-
bucket:
|
|
77
|
-
key:
|
|
78
|
-
sizeBytes:
|
|
79
|
-
contentType:
|
|
80
|
-
etag:
|
|
81
|
-
checksum:
|
|
82
|
-
lastModified:
|
|
83
|
-
metadata:
|
|
77
|
+
bucket: String(meta.bucket ?? ""),
|
|
78
|
+
key: String(meta.name ?? ""),
|
|
79
|
+
sizeBytes: meta.size ? Number(meta.size) : void 0,
|
|
80
|
+
contentType: meta.contentType ? String(meta.contentType) : void 0,
|
|
81
|
+
etag: meta.etag ? String(meta.etag) : void 0,
|
|
82
|
+
checksum: meta.md5Hash ? String(meta.md5Hash) : void 0,
|
|
83
|
+
lastModified: meta.updated ? new Date(String(meta.updated)) : void 0,
|
|
84
|
+
metadata: meta.metadata
|
|
84
85
|
};
|
|
85
86
|
}
|
|
86
87
|
|
|
87
88
|
//#endregion
|
|
88
|
-
export { GoogleCloudStorageProvider };
|
|
89
|
+
export { GoogleCloudStorageProvider };
|
|
90
|
+
//# sourceMappingURL=gcs-storage.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"gcs-storage.js","names":[],"sources":["../../src/impls/gcs-storage.ts"],"sourcesContent":["import { Storage, type StorageOptions } from '@google-cloud/storage';\n\nimport type {\n ObjectStorageProvider,\n PutObjectInput,\n GetObjectResult,\n StorageObjectMetadata,\n ListObjectsQuery,\n ListObjectsResult,\n SignedUrlOptions,\n DeleteObjectInput,\n} from '../storage';\n\nexport interface GoogleCloudStorageProviderOptions {\n bucket: string;\n storage?: Storage;\n clientOptions?: StorageOptions;\n}\n\nexport class GoogleCloudStorageProvider implements ObjectStorageProvider {\n private readonly storage: Storage;\n private readonly bucketName: string;\n\n constructor(options: GoogleCloudStorageProviderOptions) {\n this.storage =\n options.storage ?? new Storage(options.clientOptions ?? undefined);\n this.bucketName = options.bucket;\n }\n\n async putObject(input: PutObjectInput): Promise<StorageObjectMetadata> {\n const bucketName = input.bucket ?? this.bucketName;\n const bucket = this.storage.bucket(bucketName);\n const file = bucket.file(input.key);\n const buffer = toBuffer(input.data);\n await file.save(buffer, {\n resumable: false,\n contentType: input.contentType,\n metadata: input.metadata,\n });\n if (input.makePublic) {\n await file.makePublic();\n }\n const [metadata] = await file.getMetadata();\n return toMetadata(metadata);\n }\n\n async getObject(input: DeleteObjectInput): Promise<GetObjectResult | null> {\n const bucketName = input.bucket ?? this.bucketName;\n const bucket = this.storage.bucket(bucketName);\n const file = bucket.file(input.key);\n const [exists] = await file.exists();\n if (!exists) return null;\n const [contents] = await file.download();\n const [metadata] = await file.getMetadata();\n return {\n ...toMetadata(metadata),\n data: new Uint8Array(contents),\n };\n }\n\n async deleteObject(input: DeleteObjectInput): Promise<void> {\n const bucketName = input.bucket ?? this.bucketName;\n const bucket = this.storage.bucket(bucketName);\n const file = bucket.file(input.key);\n await file.delete({ ignoreNotFound: true });\n }\n\n async generateSignedUrl(options: SignedUrlOptions): Promise<{\n url: string;\n expiresAt: Date;\n }> {\n const bucketName = options.bucket ?? this.bucketName;\n const bucket = this.storage.bucket(bucketName);\n const file = bucket.file(options.key);\n const action = options.method === 'PUT' ? 'write' : 'read';\n const expires = Date.now() + options.expiresInSeconds * 1000;\n const [url] = await file.getSignedUrl({\n action,\n expires,\n contentType: options.contentType,\n });\n return { url, expiresAt: new Date(expires) };\n }\n\n async listObjects(query: ListObjectsQuery): Promise<ListObjectsResult> {\n const bucketName = query.bucket ?? this.bucketName;\n const bucket = this.storage.bucket(bucketName);\n const [files, nextQuery, response] = await bucket.getFiles({\n prefix: query.prefix,\n maxResults: query.maxResults,\n pageToken: query.pageToken,\n });\n const nextTokenFromQuery =\n typeof nextQuery === 'object' &&\n nextQuery !== null &&\n 'pageToken' in nextQuery\n ? (nextQuery as { pageToken?: string }).pageToken\n : undefined;\n const nextTokenFromResponse =\n response && typeof response === 'object' && 'nextPageToken' in response\n ? (response as { nextPageToken?: string }).nextPageToken\n : undefined;\n return {\n objects: files.map((file) => toMetadata(file.metadata)),\n nextPageToken: nextTokenFromQuery ?? nextTokenFromResponse ?? undefined,\n };\n }\n}\n\nfunction toBuffer(data: Uint8Array | ArrayBuffer): Buffer {\n if (data instanceof Uint8Array) {\n return Buffer.from(data);\n }\n return Buffer.from(data);\n}\n\nfunction toMetadata(metadata: unknown): StorageObjectMetadata {\n const meta = metadata as Record<string, unknown>;\n return {\n bucket: String(meta.bucket ?? ''),\n key: String(meta.name ?? ''),\n sizeBytes: meta.size ? Number(meta.size) : undefined,\n contentType: meta.contentType ? String(meta.contentType) : undefined,\n etag: meta.etag ? String(meta.etag) : undefined,\n checksum: meta.md5Hash ? String(meta.md5Hash) : undefined,\n lastModified: meta.updated ? new Date(String(meta.updated)) : undefined,\n metadata: meta.metadata as Record<string, string> | undefined,\n };\n}\n"],"mappings":";;;AAmBA,IAAa,6BAAb,MAAyE;CACvE,AAAiB;CACjB,AAAiB;CAEjB,YAAY,SAA4C;AACtD,OAAK,UACH,QAAQ,WAAW,IAAI,QAAQ,QAAQ,iBAAiB,OAAU;AACpE,OAAK,aAAa,QAAQ;;CAG5B,MAAM,UAAU,OAAuD;EACrE,MAAM,aAAa,MAAM,UAAU,KAAK;EAExC,MAAM,OADS,KAAK,QAAQ,OAAO,WAAW,CAC1B,KAAK,MAAM,IAAI;EACnC,MAAM,SAAS,SAAS,MAAM,KAAK;AACnC,QAAM,KAAK,KAAK,QAAQ;GACtB,WAAW;GACX,aAAa,MAAM;GACnB,UAAU,MAAM;GACjB,CAAC;AACF,MAAI,MAAM,WACR,OAAM,KAAK,YAAY;EAEzB,MAAM,CAAC,YAAY,MAAM,KAAK,aAAa;AAC3C,SAAO,WAAW,SAAS;;CAG7B,MAAM,UAAU,OAA2D;EACzE,MAAM,aAAa,MAAM,UAAU,KAAK;EAExC,MAAM,OADS,KAAK,QAAQ,OAAO,WAAW,CAC1B,KAAK,MAAM,IAAI;EACnC,MAAM,CAAC,UAAU,MAAM,KAAK,QAAQ;AACpC,MAAI,CAAC,OAAQ,QAAO;EACpB,MAAM,CAAC,YAAY,MAAM,KAAK,UAAU;EACxC,MAAM,CAAC,YAAY,MAAM,KAAK,aAAa;AAC3C,SAAO;GACL,GAAG,WAAW,SAAS;GACvB,MAAM,IAAI,WAAW,SAAS;GAC/B;;CAGH,MAAM,aAAa,OAAyC;EAC1D,MAAM,aAAa,MAAM,UAAU,KAAK;AAGxC,QAFe,KAAK,QAAQ,OAAO,WAAW,CAC1B,KAAK,MAAM,IAAI,CACxB,OAAO,EAAE,gBAAgB,MAAM,CAAC;;CAG7C,MAAM,kBAAkB,SAGrB;EACD,MAAM,aAAa,QAAQ,UAAU,KAAK;EAE1C,MAAM,OADS,KAAK,QAAQ,OAAO,WAAW,CAC1B,KAAK,QAAQ,IAAI;EACrC,MAAM,SAAS,QAAQ,WAAW,QAAQ,UAAU;EACpD,MAAM,UAAU,KAAK,KAAK,GAAG,QAAQ,mBAAmB;EACxD,MAAM,CAAC,OAAO,MAAM,KAAK,aAAa;GACpC;GACA;GACA,aAAa,QAAQ;GACtB,CAAC;AACF,SAAO;GAAE;GAAK,WAAW,IAAI,KAAK,QAAQ;GAAE;;CAG9C,MAAM,YAAY,OAAqD;EACrE,MAAM,aAAa,MAAM,UAAU,KAAK;EAExC,MAAM,CAAC,OAAO,WAAW,YAAY,MADtB,KAAK,QAAQ,OAAO,WAAW,CACI,SAAS;GACzD,QAAQ,MAAM;GACd,YAAY,MAAM;GAClB,WAAW,MAAM;GAClB,CAAC;EACF,MAAM,qBACJ,OAAO,cAAc,YACrB,cAAc,QACd,eAAe,YACV,UAAqC,YACtC;EACN,MAAM,wBACJ,YAAY,OAAO,aAAa,YAAY,mBAAmB,WAC1D,SAAwC,gBACzC;AACN,SAAO;GACL,SAAS,MAAM,KAAK,SAAS,WAAW,KAAK,SAAS,CAAC;GACvD,eAAe,sBAAsB,yBAAyB;GAC/D;;;AAIL,SAAS,SAAS,MAAwC;AACxD,KAAI,gBAAgB,WAClB,QAAO,OAAO,KAAK,KAAK;AAE1B,QAAO,OAAO,KAAK,KAAK;;AAG1B,SAAS,WAAW,UAA0C;CAC5D,MAAM,OAAO;AACb,QAAO;EACL,QAAQ,OAAO,KAAK,UAAU,GAAG;EACjC,KAAK,OAAO,KAAK,QAAQ,GAAG;EAC5B,WAAW,KAAK,OAAO,OAAO,KAAK,KAAK,GAAG;EAC3C,aAAa,KAAK,cAAc,OAAO,KAAK,YAAY,GAAG;EAC3D,MAAM,KAAK,OAAO,OAAO,KAAK,KAAK,GAAG;EACtC,UAAU,KAAK,UAAU,OAAO,KAAK,QAAQ,GAAG;EAChD,cAAc,KAAK,UAAU,IAAI,KAAK,OAAO,KAAK,QAAQ,CAAC,GAAG;EAC9D,UAAU,KAAK;EAChB"}
|
|
@@ -23,4 +23,5 @@ declare class GmailInboundProvider implements email_d_exports.EmailInboundProvid
|
|
|
23
23
|
private transformMessage;
|
|
24
24
|
}
|
|
25
25
|
//#endregion
|
|
26
|
-
export { GmailInboundProvider, GmailInboundProviderOptions };
|
|
26
|
+
export { GmailInboundProvider, GmailInboundProviderOptions };
|
|
27
|
+
//# sourceMappingURL=gmail-inbound.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"gmail-inbound.d.ts","names":[],"sources":["../../src/impls/gmail-inbound.ts"],"sourcesContent":[],"mappings":";;;;UAUiB,2BAAA;QACT,QAAA,CAAS;EADA,MAAA,CAAA,EAAA,MAAA;EAOJ,KAAA,CAAA,EAJH,QAAA,CAAS,KAIN;EAMU,gBAAA,CAAA,EAAA,OAAA;;AAYoC,cAlB9C,oBAAA,YAAgC,eAAA,CAAA,oBAkBc,CAAA;EAAR,iBAAA,KAAA;EAqBN,iBAAA,MAAA;EAAR,iBAAA,gBAAA;EAsDJ,iBAAA,IAAA;uBAvFV;EAuFiC,WAAA,CAAA,KAAA,CAAA,EA3E5B,eAAA,CAAA,oBA2E4B,CAAA,EA3EL,OA2EK,CA3EG,eAAA,CAAA,WA2EH,EAAA,CAAA;EA7FX,SAAA,CAAA,QAAA,EAAA,MAAA,CAAA,EAuCR,OAvCQ,CAuCA,eAAA,CAAA,WAvCA,GAAA,IAAA,CAAA;EAAoB,iBAAA,CAAA,KAAA,EA6FhC,eAAA,CAAA,uBA7FgC,CAAA,EA6FT,OA7FS,CAAA"}
|
|
@@ -131,7 +131,7 @@ function headerValue(headers, name) {
|
|
|
131
131
|
function parseAddress(header) {
|
|
132
132
|
const addresses = parseAddressList(header);
|
|
133
133
|
if (addresses.length === 0) return null;
|
|
134
|
-
return addresses[0];
|
|
134
|
+
return addresses[0] || null;
|
|
135
135
|
}
|
|
136
136
|
function inferFallbackAddress(field, messageId) {
|
|
137
137
|
return { email: `${field}-${messageId ? messageId.replace(/[^\w]/g, "").slice(-8) || "unknown" : "unknown"}@mail.local` };
|
|
@@ -197,4 +197,5 @@ function decodeBase64Url(data) {
|
|
|
197
197
|
}
|
|
198
198
|
|
|
199
199
|
//#endregion
|
|
200
|
-
export { GmailInboundProvider };
|
|
200
|
+
export { GmailInboundProvider };
|
|
201
|
+
//# sourceMappingURL=gmail-inbound.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"gmail-inbound.js","names":["q: string[]","metadata: Record<string, string>","attachments: NonNullable<EmailMessage['attachments']>"],"sources":["../../src/impls/gmail-inbound.ts"],"sourcesContent":["import { google, type gmail_v1 } from 'googleapis';\n\nimport type {\n EmailInboundProvider,\n EmailMessagesSinceQuery,\n EmailMessage,\n EmailThread,\n EmailThreadListQuery,\n} from '../email';\n\nexport interface GmailInboundProviderOptions {\n auth: gmail_v1.Options['auth'];\n userId?: string;\n gmail?: gmail_v1.Gmail;\n includeSpamTrash?: boolean;\n}\n\nexport class GmailInboundProvider implements EmailInboundProvider {\n private readonly gmail: gmail_v1.Gmail;\n private readonly userId: string;\n private readonly includeSpamTrash: boolean;\n private readonly auth: gmail_v1.Options['auth'];\n\n constructor(options: GmailInboundProviderOptions) {\n this.auth = options.auth;\n this.gmail =\n options.gmail ??\n google.gmail({\n version: 'v1',\n auth: options.auth,\n });\n this.userId = options.userId ?? 'me';\n this.includeSpamTrash = options.includeSpamTrash ?? false;\n }\n\n async listThreads(query?: EmailThreadListQuery): Promise<EmailThread[]> {\n const response = await this.gmail.users.threads.list({\n userId: this.userId,\n maxResults: query?.pageSize,\n pageToken: query?.pageToken,\n q: query?.query,\n labelIds: query?.label ? [query.label] : undefined,\n includeSpamTrash: this.includeSpamTrash,\n auth: this.auth,\n });\n\n const threads = await Promise.all(\n (response.data.threads ?? []).map(async (thread) => {\n if (!thread.id) return null;\n return this.getThread(thread.id);\n })\n );\n\n return threads.filter((thread): thread is EmailThread => thread !== null);\n }\n\n async getThread(threadId: string): Promise<EmailThread | null> {\n const response = await this.gmail.users.threads.get({\n id: threadId,\n userId: this.userId,\n format: 'full',\n auth: this.auth,\n });\n const thread = response.data;\n if (!thread) return null;\n\n const messages =\n thread.messages?.map((message) => this.transformMessage(message)) ?? [];\n\n const participants = dedupeAddresses(\n messages.flatMap((message) => [\n message.from,\n ...message.to,\n ...(message.cc ?? []),\n ])\n );\n\n const firstMessage = messages[0];\n const lastMessage = messages[messages.length - 1];\n const updatedAt =\n lastMessage?.receivedAt ??\n lastMessage?.sentAt ??\n firstMessage?.receivedAt ??\n firstMessage?.sentAt ??\n new Date();\n\n const labels = Array.from(\n new Set(\n messages\n .flatMap((message) => {\n const labelField = message.metadata?.labelIds;\n if (!labelField) return [];\n return labelField.split(',').map((label) => label.trim());\n })\n .filter((label): label is string => Boolean(label))\n )\n );\n\n return {\n id: thread.id ?? threadId,\n subject: messages[0]?.subject,\n snippet: thread.snippet ?? '',\n participants,\n messages,\n updatedAt,\n labels,\n metadata: thread.historyId ? { historyId: thread.historyId } : undefined,\n };\n }\n\n async listMessagesSince(query: EmailMessagesSinceQuery) {\n const after = query.since\n ? Math.floor(query.since.getTime() / 1000)\n : undefined;\n const q: string[] = [];\n if (after) {\n q.push(`after:${after}`);\n }\n const response = await this.gmail.users.messages.list({\n userId: this.userId,\n maxResults: query.pageSize,\n pageToken: query.pageToken,\n labelIds: query.label ? [query.label] : undefined,\n q: q.join(' '),\n includeSpamTrash: this.includeSpamTrash,\n auth: this.auth,\n });\n\n const messages = await Promise.all(\n (response.data.messages ?? []).map(async (item) => {\n if (!item.id) return null;\n const full = await this.gmail.users.messages.get({\n userId: this.userId,\n id: item.id,\n format: 'full',\n auth: this.auth,\n });\n if (!full.data) return null;\n return this.transformMessage(full.data);\n })\n );\n\n return {\n messages: messages.filter(\n (message): message is EmailMessage => message !== null\n ),\n nextPageToken: response.data.nextPageToken ?? undefined,\n };\n }\n\n private transformMessage(message: gmail_v1.Schema$Message): EmailMessage {\n const headers = message.payload?.headers ?? [];\n const subject = headerValue(headers, 'Subject') ?? '';\n const from =\n parseAddress(headerValue(headers, 'From')) ??\n inferFallbackAddress('from', message.id);\n const to = parseAddressList(headerValue(headers, 'To'));\n const cc = parseAddressList(headerValue(headers, 'Cc'));\n const bcc = parseAddressList(headerValue(headers, 'Bcc'));\n const replyTo = parseAddress(headerValue(headers, 'Reply-To'));\n\n const { text, html, attachments } = extractContent(message.payload);\n const timestamp = message.internalDate\n ? new Date(Number(message.internalDate))\n : new Date();\n\n const metadata: Record<string, string> = {\n ...(message.labelIds?.length\n ? { labelIds: message.labelIds.join(',') }\n : {}),\n ...(message.historyId ? { historyId: message.historyId } : {}),\n };\n\n return {\n id: message.id ?? '',\n threadId: message.threadId ?? '',\n subject,\n from,\n to,\n cc,\n bcc,\n replyTo: replyTo ?? undefined,\n sentAt: timestamp,\n receivedAt: timestamp,\n textBody: text ?? undefined,\n htmlBody: html ?? undefined,\n attachments,\n headers: Object.fromEntries(\n headers.map((header) => [header.name ?? '', header.value ?? ''])\n ),\n metadata: Object.keys(metadata).length > 0 ? metadata : undefined,\n };\n }\n}\n\nfunction headerValue(\n headers: gmail_v1.Schema$MessagePartHeader[],\n name: string\n): string | undefined {\n const header = headers.find(\n (candidate) => candidate.name?.toLowerCase() === name.toLowerCase()\n );\n const value = header?.value;\n return typeof value === 'string' ? value : undefined;\n}\n\nfunction parseAddress(\n header?: string\n): { email: string; name?: string } | null {\n const addresses = parseAddressList(header);\n if (addresses.length === 0) {\n return null;\n }\n const firstAddress = addresses[0];\n return firstAddress || null;\n}\n\nfunction inferFallbackAddress(\n field: string,\n messageId?: string | null\n): { email: string; name?: string } {\n const suffix = messageId\n ? messageId.replace(/[^\\w]/g, '').slice(-8) || 'unknown'\n : 'unknown';\n return {\n email: `${field}-${suffix}@mail.local`,\n };\n}\n\nfunction parseAddressList(header?: string): { email: string; name?: string }[] {\n if (!header) return [];\n return header\n .split(',')\n .map((part) => part.trim())\n .filter(Boolean)\n .map((value) => {\n const match = value.match(/^(?:\"?([^\"]*)\"?\\s)?<?([^<>]+)>?$/);\n if (!match) {\n return { email: value };\n }\n const name = match[1]?.trim();\n const email = match[2]?.trim();\n if (!email) {\n return { email: value };\n }\n return name ? { email, name } : { email };\n });\n}\n\nfunction dedupeAddresses(\n addresses: (\n | ReturnType<typeof parseAddress>\n | { email: string; name?: string }\n )[]\n) {\n const map = new Map<string, { email: string; name?: string }>();\n for (const address of addresses) {\n if (!address) continue;\n map.set(address.email.toLowerCase(), address);\n }\n return Array.from(map.values());\n}\n\nfunction extractContent(payload?: gmail_v1.Schema$MessagePart): {\n text?: string;\n html?: string;\n attachments: EmailMessage['attachments'];\n} {\n if (!payload) {\n return { attachments: [] };\n }\n const attachments: NonNullable<EmailMessage['attachments']> = [];\n const visit = (\n part?: gmail_v1.Schema$MessagePart\n ): {\n text?: string;\n html?: string;\n } => {\n if (!part) return {};\n if (part.filename && part.body?.attachmentId) {\n attachments.push({\n id: part.body.attachmentId,\n filename: part.filename,\n contentType: part.mimeType ?? 'application/octet-stream',\n sizeBytes: part.body.size ?? undefined,\n });\n }\n const mimeType = part.mimeType ?? '';\n const data = part.body?.data;\n if (mimeType === 'text/plain' && data) {\n return { text: decodeBase64Url(data) };\n }\n if (mimeType === 'text/html' && data) {\n return { html: decodeBase64Url(data) };\n }\n if (part.parts?.length) {\n return part.parts.reduce<{ text?: string; html?: string }>(\n (acc, nested) => {\n const value = visit(nested);\n return {\n text: value.text ?? acc.text,\n html: value.html ?? acc.html,\n };\n },\n {}\n );\n }\n return {};\n };\n\n const { text, html } = visit(payload);\n return { text, html, attachments };\n}\n\nfunction decodeBase64Url(data: string): string {\n const normalized = data.replace(/-/g, '+').replace(/_/g, '/');\n const padding = normalized.length % 4;\n const padded =\n padding === 0 ? normalized : normalized + '='.repeat(4 - padding);\n return Buffer.from(padded, 'base64').toString('utf-8');\n}\n"],"mappings":"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"}
|
|
@@ -15,4 +15,5 @@ declare class GmailOutboundProvider implements email_d_exports.EmailOutboundProv
|
|
|
15
15
|
sendEmail(message: email_d_exports.EmailOutboundMessage): Promise<email_d_exports.EmailOutboundResult>;
|
|
16
16
|
}
|
|
17
17
|
//#endregion
|
|
18
|
-
export { GmailOutboundProvider, GmailOutboundProviderOptions };
|
|
18
|
+
export { GmailOutboundProvider, GmailOutboundProviderOptions };
|
|
19
|
+
//# sourceMappingURL=gmail-outbound.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"gmail-outbound.d.ts","names":[],"sources":["../../src/impls/gmail-outbound.ts"],"sourcesContent":[],"mappings":";;;;UASiB,4BAAA;QACT,QAAA,CAAS;EADA,MAAA,CAAA,EAAA,MAAA;EAMJ,KAAA,CAAA,EAHH,QAAA,CAAS,KAGN;;AAgBc,cAhBd,qBAAA,YAAiC,eAAA,CAAA,qBAgBnB,CAAA;EAA+B,iBAAA,KAAA;EAAR,iBAAA,MAAA;EAhBJ,iBAAA,IAAA;EAAqB,WAAA,CAAA,OAAA,EAK5C,4BAL4C;qBAgBxC,eAAA,CAAA,uBAAuB,QAAQ,eAAA,CAAA"}
|
|
@@ -56,9 +56,9 @@ function encodeMessage(message) {
|
|
|
56
56
|
} else if (hasText && hasHtml) {
|
|
57
57
|
headers.push(`Content-Type: multipart/alternative; boundary="${boundaryAlt}"`);
|
|
58
58
|
body += `\r\n--${boundaryAlt}\r\n`;
|
|
59
|
-
body += buildTextPart("text/plain; charset=\"utf-8\"", message.textBody);
|
|
59
|
+
body += buildTextPart("text/plain; charset=\"utf-8\"", message.textBody || "");
|
|
60
60
|
body += `\r\n--${boundaryAlt}\r\n`;
|
|
61
|
-
body += buildTextPart("text/html; charset=\"utf-8\"", message.htmlBody);
|
|
61
|
+
body += buildTextPart("text/html; charset=\"utf-8\"", message.htmlBody || "");
|
|
62
62
|
body += `\r\n--${boundaryAlt}--`;
|
|
63
63
|
} else if (hasHtml) {
|
|
64
64
|
headers.push("Content-Type: text/html; charset=\"utf-8\"");
|
|
@@ -76,11 +76,11 @@ function buildAlternativePart(hasText, hasHtml, boundary, message) {
|
|
|
76
76
|
content += "\r\n";
|
|
77
77
|
if (hasText) {
|
|
78
78
|
content += `--${boundary}\r\n`;
|
|
79
|
-
content += buildTextPart("text/plain; charset=\"utf-8\"", message.textBody);
|
|
79
|
+
content += buildTextPart("text/plain; charset=\"utf-8\"", message.textBody || "");
|
|
80
80
|
}
|
|
81
81
|
if (hasHtml) {
|
|
82
82
|
content += `\r\n--${boundary}\r\n`;
|
|
83
|
-
content += buildTextPart("text/html; charset=\"utf-8\"", message.htmlBody);
|
|
83
|
+
content += buildTextPart("text/html; charset=\"utf-8\"", message.htmlBody || "");
|
|
84
84
|
}
|
|
85
85
|
content += `\r\n--${boundary}--`;
|
|
86
86
|
return content;
|
|
@@ -102,4 +102,5 @@ function formatAddress(address) {
|
|
|
102
102
|
}
|
|
103
103
|
|
|
104
104
|
//#endregion
|
|
105
|
-
export { GmailOutboundProvider };
|
|
105
|
+
export { GmailOutboundProvider };
|
|
106
|
+
//# sourceMappingURL=gmail-outbound.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"gmail-outbound.js","names":["headers: string[]"],"sources":["../../src/impls/gmail-outbound.ts"],"sourcesContent":["import { google, type gmail_v1 } from 'googleapis';\n\nimport type {\n EmailAttachment,\n EmailOutboundMessage,\n EmailOutboundProvider,\n EmailOutboundResult,\n} from '../email';\n\nexport interface GmailOutboundProviderOptions {\n auth: gmail_v1.Options['auth'];\n userId?: string;\n gmail?: gmail_v1.Gmail;\n}\n\nexport class GmailOutboundProvider implements EmailOutboundProvider {\n private readonly gmail: gmail_v1.Gmail;\n private readonly userId: string;\n private readonly auth: gmail_v1.Options['auth'];\n\n constructor(options: GmailOutboundProviderOptions) {\n this.auth = options.auth;\n this.gmail =\n options.gmail ??\n google.gmail({\n version: 'v1',\n auth: options.auth,\n });\n this.userId = options.userId ?? 'me';\n }\n\n async sendEmail(message: EmailOutboundMessage): Promise<EmailOutboundResult> {\n const raw = encodeMessage(message);\n const response = await this.gmail.users.messages.send({\n userId: this.userId,\n requestBody: {\n raw,\n },\n auth: this.auth,\n });\n\n const id = response.data.id ?? '';\n return {\n id,\n providerMessageId: response.data.id ?? undefined,\n queuedAt: new Date(),\n };\n }\n}\n\nfunction encodeMessage(message: EmailOutboundMessage): string {\n const headers: string[] = [\n `From: ${formatAddress(message.from)}`,\n `To: ${message.to.map(formatAddress).join(', ')}`,\n `Subject: ${message.subject}`,\n 'MIME-Version: 1.0',\n ];\n if (message.cc?.length) {\n headers.push(`Cc: ${message.cc.map(formatAddress).join(', ')}`);\n }\n if (message.replyTo) {\n headers.push(`Reply-To: ${formatAddress(message.replyTo)}`);\n }\n Object.entries(message.headers ?? {}).forEach(([key, value]) => {\n headers.push(`${key}: ${value}`);\n });\n\n const attachments = message.attachments ?? [];\n const hasHtml = Boolean(message.htmlBody);\n const hasText = Boolean(message.textBody);\n const boundaryMain = `mixed_${Date.now()}`;\n const boundaryAlt = `alt_${Date.now()}`;\n\n let body = '';\n if (attachments.length > 0) {\n headers.push(`Content-Type: multipart/mixed; boundary=\"${boundaryMain}\"`);\n body += `\\r\\n--${boundaryMain}\\r\\n`;\n body += buildAlternativePart(hasText, hasHtml, boundaryAlt, message);\n attachments.forEach((attachment) => {\n body += buildAttachmentPart(boundaryMain, attachment);\n });\n body += `\\r\\n--${boundaryMain}--`;\n } else if (hasText && hasHtml) {\n headers.push(\n `Content-Type: multipart/alternative; boundary=\"${boundaryAlt}\"`\n );\n body += `\\r\\n--${boundaryAlt}\\r\\n`;\n body += buildTextPart(\n 'text/plain; charset=\"utf-8\"',\n message.textBody || ''\n );\n body += `\\r\\n--${boundaryAlt}\\r\\n`;\n body += buildTextPart('text/html; charset=\"utf-8\"', message.htmlBody || '');\n body += `\\r\\n--${boundaryAlt}--`;\n } else if (hasHtml) {\n headers.push('Content-Type: text/html; charset=\"utf-8\"');\n body += `\\r\\n\\r\\n${message.htmlBody}`;\n } else {\n headers.push('Content-Type: text/plain; charset=\"utf-8\"');\n body += `\\r\\n\\r\\n${message.textBody ?? ''}`;\n }\n\n const mime = `${headers.join('\\r\\n')}${body}`;\n return Buffer.from(mime)\n .toString('base64')\n .replace(/\\+/g, '-')\n .replace(/\\//g, '_')\n .replace(/=+$/, '');\n}\n\nfunction buildAlternativePart(\n hasText: boolean,\n hasHtml: boolean,\n boundary: string,\n message: EmailOutboundMessage\n) {\n let content = '';\n content += `Content-Type: multipart/alternative; boundary=\"${boundary}\"\\r\\n`;\n content += '\\r\\n';\n if (hasText) {\n content += `--${boundary}\\r\\n`;\n content += buildTextPart(\n 'text/plain; charset=\"utf-8\"',\n message.textBody || ''\n );\n }\n if (hasHtml) {\n content += `\\r\\n--${boundary}\\r\\n`;\n content += buildTextPart(\n 'text/html; charset=\"utf-8\"',\n message.htmlBody || ''\n );\n }\n content += `\\r\\n--${boundary}--`;\n return content;\n}\n\nfunction buildTextPart(contentType: string, content: string) {\n return (\n `Content-Type: ${contentType}\\r\\n` +\n 'Content-Transfer-Encoding: 7bit\\r\\n\\r\\n' +\n content\n );\n}\n\nfunction buildAttachmentPart(\n boundary: string,\n attachment: EmailAttachment\n): string {\n const data = attachment.data ?? new Uint8Array();\n const encoded =\n data.byteLength > 0 ? Buffer.from(data).toString('base64') : '';\n return (\n `\\r\\n--${boundary}\\r\\n` +\n `Content-Type: ${attachment.contentType}; name=\"${attachment.filename}\"\\r\\n` +\n 'Content-Transfer-Encoding: base64\\r\\n' +\n `Content-Disposition: attachment; filename=\"${attachment.filename}\"\\r\\n\\r\\n` +\n encoded\n );\n}\n\nfunction formatAddress(address: { email: string; name?: string }) {\n if (address.name) {\n return `\"${address.name}\" <${address.email}>`;\n }\n return address.email;\n}\n"],"mappings":"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"}
|
|
@@ -20,4 +20,5 @@ declare class GoogleCalendarProvider implements calendar_d_exports.CalendarProvi
|
|
|
20
20
|
private toGoogleEvent;
|
|
21
21
|
}
|
|
22
22
|
//#endregion
|
|
23
|
-
export { GoogleCalendarProvider, GoogleCalendarProviderOptions };
|
|
23
|
+
export { GoogleCalendarProvider, GoogleCalendarProviderOptions };
|
|
24
|
+
//# sourceMappingURL=google-calendar.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"google-calendar.d.ts","names":[],"sources":["../../src/impls/google-calendar.ts"],"sourcesContent":[],"mappings":";;;;UAWiB,6BAAA;QACT,WAAA,CAAY;EADH,QAAA,CAAA,EAEJ,WAAA,CAAY,QAFR;EAMJ,UAAA,CAAA,EAAA,MAAA;;AAiBF,cAjBE,sBAAA,YAAkC,kBAAA,CAAA,gBAiBpC,CAAA;EACE,iBAAA,QAAA;EAAR,iBAAA,iBAAA;EAuBsB,iBAAA,IAAA;EAA6B,WAAA,CAAA,OAAA,EApCjC,6BAoCiC;EAAR,UAAA,CAAA,KAAA,EAxBrC,kBAAA,CAAA,uBAwBqC,CAAA,EAvB3C,OAuB2C,CAvBnC,kBAAA,CAAA,wBAuBmC,CAAA;EAcrC,WAAA,CAAA,KAAA,EAdgB,kBAAA,CAAA,kBAchB,CAAA,EAdqC,OAcrC,CAd6C,kBAAA,CAAA,aAc7C,CAAA;EACE,WAAA,CAAA,UAAA,EAAA,MAAA,EAAA,OAAA,EAAA,MAAA,EAAA,KAAA,EADF,kBAAA,CAAA,wBACE,CAAA,EAAR,OAAQ,CAAA,kBAAA,CAAA,aAAA,CAAA;EAAR,WAAA,CAAA,UAAA,EAAA,MAAA,EAAA,OAAA,EAAA,MAAA,CAAA,EAWqD,OAXrD,CAAA,IAAA,CAAA;EAWqD,QAAA,eAAA;EAnEX,QAAA,aAAA"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"google-calendar.js","names":["event: calendar_v3.Schema$Event","metadata: Record<string, string>"],"sources":["../../src/impls/google-calendar.ts"],"sourcesContent":["import { google, type calendar_v3 } from 'googleapis';\n\nimport type {\n CalendarEvent,\n CalendarEventInput,\n CalendarEventUpdateInput,\n CalendarListEventsQuery,\n CalendarListEventsResult,\n CalendarProvider,\n} from '../calendar';\n\nexport interface GoogleCalendarProviderOptions {\n auth: calendar_v3.Options['auth'];\n calendar?: calendar_v3.Calendar;\n calendarId?: string;\n}\n\nexport class GoogleCalendarProvider implements CalendarProvider {\n private readonly calendar: calendar_v3.Calendar;\n private readonly defaultCalendarId: string;\n private readonly auth: calendar_v3.Options['auth'];\n\n constructor(options: GoogleCalendarProviderOptions) {\n this.auth = options.auth;\n this.calendar =\n options.calendar ??\n google.calendar({\n version: 'v3',\n auth: options.auth,\n });\n this.defaultCalendarId = options.calendarId ?? 'primary';\n }\n\n async listEvents(\n query: CalendarListEventsQuery\n ): Promise<CalendarListEventsResult> {\n const response = await this.calendar.events.list({\n calendarId: query.calendarId ?? this.defaultCalendarId,\n timeMin: query.timeMin?.toISOString(),\n timeMax: query.timeMax?.toISOString(),\n maxResults: query.maxResults,\n pageToken: query.pageToken,\n singleEvents: true,\n orderBy: 'startTime',\n auth: this.auth,\n });\n\n const events =\n response.data.items?.map((item) =>\n this.fromGoogleEvent(query.calendarId ?? this.defaultCalendarId, item)\n ) ?? [];\n\n return {\n events,\n nextPageToken: response.data.nextPageToken ?? undefined,\n };\n }\n\n async createEvent(input: CalendarEventInput): Promise<CalendarEvent> {\n const calendarId = input.calendarId ?? this.defaultCalendarId;\n const response = await this.calendar.events.insert({\n calendarId,\n requestBody: this.toGoogleEvent(input),\n conferenceDataVersion: input.conference?.create ? 1 : undefined,\n auth: this.auth,\n });\n return this.fromGoogleEvent(calendarId, response.data);\n }\n\n async updateEvent(\n calendarId: string,\n eventId: string,\n input: CalendarEventUpdateInput\n ): Promise<CalendarEvent> {\n const response = await this.calendar.events.patch({\n calendarId: calendarId ?? this.defaultCalendarId,\n eventId,\n requestBody: this.toGoogleEvent(input),\n conferenceDataVersion: input.conference?.create ? 1 : undefined,\n auth: this.auth,\n });\n return this.fromGoogleEvent(calendarId, response.data);\n }\n\n async deleteEvent(calendarId: string, eventId: string): Promise<void> {\n await this.calendar.events.delete({\n calendarId: calendarId ?? this.defaultCalendarId,\n eventId,\n auth: this.auth,\n });\n }\n\n private fromGoogleEvent(\n calendarId: string,\n event: calendar_v3.Schema$Event\n ): CalendarEvent {\n const start = parseDateTime(event.start);\n const end = parseDateTime(event.end);\n const attendees =\n event.attendees?.map((attendee) => ({\n email: attendee.email ?? '',\n name: attendee.displayName ?? undefined,\n optional: attendee.optional ?? undefined,\n responseStatus: normalizeResponseStatus(attendee.responseStatus),\n })) ?? [];\n const reminders =\n event.reminders?.overrides?.map((reminder) => ({\n method: (reminder.method as 'email' | 'popup') ?? 'popup',\n minutesBeforeStart: reminder.minutes ?? 0,\n })) ?? [];\n const metadata = buildMetadata(event);\n return {\n id: event.id ?? '',\n calendarId,\n title: event.summary ?? '',\n description: event.description ?? undefined,\n location: event.location ?? undefined,\n start,\n end,\n allDay: event.start?.date ? true : undefined,\n attendees,\n reminders,\n conferenceLink:\n event.hangoutLink ??\n event.conferenceData?.entryPoints?.find((entry) => entry.uri)?.uri ??\n undefined,\n metadata,\n createdAt: event.created ? new Date(event.created) : undefined,\n updatedAt: event.updated ? new Date(event.updated) : undefined,\n };\n }\n\n private toGoogleEvent(\n input: CalendarEventInput | CalendarEventUpdateInput\n ): calendar_v3.Schema$Event {\n const event: calendar_v3.Schema$Event = {};\n if ('title' in input && input.title) event.summary = input.title;\n if (input.description !== undefined) event.description = input.description;\n if (input.location !== undefined) event.location = input.location;\n if (input.start) {\n event.start = formatDateTime(input.start, input.allDay);\n }\n if (input.end) {\n event.end = formatDateTime(input.end, input.allDay);\n }\n if (input.attendees) {\n event.attendees = input.attendees.map((attendee) => ({\n email: attendee.email,\n displayName: attendee.name,\n optional: attendee.optional,\n responseStatus: attendee.responseStatus,\n }));\n }\n if (input.reminders) {\n event.reminders = {\n useDefault: false,\n overrides: input.reminders.map((reminder) => ({\n method: reminder.method,\n minutes: reminder.minutesBeforeStart,\n })),\n };\n }\n if (input.conference?.create) {\n event.conferenceData = {\n createRequest: {\n requestId: `conf-${Date.now()}`,\n },\n };\n }\n if (input.metadata) {\n event.extendedProperties = {\n ...(event.extendedProperties ?? {}),\n private: {\n ...(event.extendedProperties?.private ?? {}),\n ...input.metadata,\n },\n };\n }\n return event;\n }\n}\n\nfunction parseDateTime(time?: calendar_v3.Schema$EventDateTime | null): Date {\n if (!time) return new Date();\n if (time.dateTime) return new Date(time.dateTime);\n if (time.date) return new Date(`${time.date}T00:00:00`);\n return new Date();\n}\n\nfunction formatDateTime(\n date: Date,\n allDay?: boolean\n): calendar_v3.Schema$EventDateTime {\n if (allDay) {\n return { date: date.toISOString().slice(0, 10) };\n }\n return { dateTime: date.toISOString() };\n}\n\ntype CalendarResponseStatus =\n | 'needsAction'\n | 'declined'\n | 'tentative'\n | 'accepted';\n\nfunction normalizeResponseStatus(\n status?: string | null\n): CalendarResponseStatus | undefined {\n if (!status) return undefined;\n const allowed: CalendarResponseStatus[] = [\n 'needsAction',\n 'declined',\n 'tentative',\n 'accepted',\n ];\n return allowed.includes(status as CalendarResponseStatus)\n ? (status as CalendarResponseStatus)\n : undefined;\n}\n\nfunction buildMetadata(\n event: calendar_v3.Schema$Event\n): Record<string, string> | undefined {\n const metadata: Record<string, string> = {};\n if (event.status) metadata.status = event.status;\n if (event.htmlLink) metadata.htmlLink = event.htmlLink;\n if (event.iCalUID) metadata.iCalUID = event.iCalUID;\n if (event.etag) metadata.etag = event.etag;\n if (event.conferenceData?.conferenceSolution?.name) {\n metadata.conferenceSolution = event.conferenceData.conferenceSolution.name;\n }\n if (event.extendedProperties?.private) {\n Object.entries(event.extendedProperties.private).forEach(([key, value]) => {\n if (typeof value === 'string') {\n metadata[`extended.${key}`] = value;\n }\n });\n }\n return Object.keys(metadata).length > 0 ? metadata : undefined;\n}\n"],"mappings":"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"}
|
|
@@ -20,4 +20,5 @@ declare class MistralEmbeddingProvider implements embedding_d_exports.EmbeddingP
|
|
|
20
20
|
}): Promise<embedding_d_exports.EmbeddingResult>;
|
|
21
21
|
}
|
|
22
22
|
//#endregion
|
|
23
|
-
export { MistralEmbeddingProvider, MistralEmbeddingProviderOptions };
|
|
23
|
+
export { MistralEmbeddingProvider, MistralEmbeddingProviderOptions };
|
|
24
|
+
//# sourceMappingURL=mistral-embedding.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"mistral-embedding.d.ts","names":[],"sources":["../../src/impls/mistral-embedding.ts"],"sourcesContent":[],"mappings":";;;;UAQiB,+BAAA;;EAAA,YAAA,CAAA,EAAA,MAAA;EAOJ,SAAA,CAAA,EAAA,MAAA;EAIU,MAAA,CAAA,EAPZ,OAOY;;AAiBV,cArBA,wBAAA,YAAoC,mBAAA,CAAA,iBAqBpC,CAAA;EAAR,iBAAA,MAAA;EA4BQ,iBAAA,YAAA;EAAR,WAAA,CAAA,OAAA,EA7CkB,+BA6ClB;EAjD4C,cAAA,CAAA,SAAA,EAmBlC,mBAAA,CAAA,iBAnBkC,EAAA,EAAA,QAAA,EAAA;IAAiB,KAAA,CAAA,EAAA,MAAA;MAqB7D,QAAQ,mBAAA,CAAA;;;MA4BR,QAAQ,mBAAA,CAAA"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"mistral-embedding.js","names":[],"sources":["../../src/impls/mistral-embedding.ts"],"sourcesContent":["import { Mistral } from '@mistralai/mistralai';\n\nimport type {\n EmbeddingDocument,\n EmbeddingProvider,\n EmbeddingResult,\n} from '../embedding';\n\nexport interface MistralEmbeddingProviderOptions {\n apiKey: string;\n defaultModel?: string;\n serverURL?: string;\n client?: Mistral;\n}\n\nexport class MistralEmbeddingProvider implements EmbeddingProvider {\n private readonly client: Mistral;\n private readonly defaultModel: string;\n\n constructor(options: MistralEmbeddingProviderOptions) {\n if (!options.apiKey) {\n throw new Error('MistralEmbeddingProvider requires an apiKey');\n }\n\n this.client =\n options.client ??\n new Mistral({\n apiKey: options.apiKey,\n serverURL: options.serverURL,\n });\n this.defaultModel = options.defaultModel ?? 'mistral-embed';\n }\n\n async embedDocuments(\n documents: EmbeddingDocument[],\n options?: { model?: string }\n ): Promise<EmbeddingResult[]> {\n if (documents.length === 0) return [];\n const model = options?.model ?? this.defaultModel;\n const response = await this.client.embeddings.create({\n model,\n inputs: documents.map((doc) => doc.text),\n });\n\n return response.data.map((item, index) => ({\n id:\n documents[index]?.id ??\n (item.index != null ? `embedding-${item.index}` : `embedding-${index}`),\n vector: item.embedding ?? [],\n dimensions: item.embedding?.length ?? 0,\n model: response.model,\n metadata: documents[index]?.metadata\n ? Object.fromEntries(\n Object.entries(documents[index]?.metadata ?? {}).map(\n ([key, value]) => [key, String(value)]\n )\n )\n : undefined,\n }));\n }\n\n async embedQuery(\n query: string,\n options?: { model?: string }\n ): Promise<EmbeddingResult> {\n const [result] = await this.embedDocuments(\n [{ id: 'query', text: query }],\n options\n );\n if (!result) {\n throw new Error('Failed to compute embedding for query');\n }\n return result;\n }\n}\n"],"mappings":";;;AAeA,IAAa,2BAAb,MAAmE;CACjE,AAAiB;CACjB,AAAiB;CAEjB,YAAY,SAA0C;AACpD,MAAI,CAAC,QAAQ,OACX,OAAM,IAAI,MAAM,8CAA8C;AAGhE,OAAK,SACH,QAAQ,UACR,IAAI,QAAQ;GACV,QAAQ,QAAQ;GAChB,WAAW,QAAQ;GACpB,CAAC;AACJ,OAAK,eAAe,QAAQ,gBAAgB;;CAG9C,MAAM,eACJ,WACA,SAC4B;AAC5B,MAAI,UAAU,WAAW,EAAG,QAAO,EAAE;EACrC,MAAM,QAAQ,SAAS,SAAS,KAAK;EACrC,MAAM,WAAW,MAAM,KAAK,OAAO,WAAW,OAAO;GACnD;GACA,QAAQ,UAAU,KAAK,QAAQ,IAAI,KAAK;GACzC,CAAC;AAEF,SAAO,SAAS,KAAK,KAAK,MAAM,WAAW;GACzC,IACE,UAAU,QAAQ,OACjB,KAAK,SAAS,OAAO,aAAa,KAAK,UAAU,aAAa;GACjE,QAAQ,KAAK,aAAa,EAAE;GAC5B,YAAY,KAAK,WAAW,UAAU;GACtC,OAAO,SAAS;GAChB,UAAU,UAAU,QAAQ,WACxB,OAAO,YACL,OAAO,QAAQ,UAAU,QAAQ,YAAY,EAAE,CAAC,CAAC,KAC9C,CAAC,KAAK,WAAW,CAAC,KAAK,OAAO,MAAM,CAAC,CACvC,CACF,GACD;GACL,EAAE;;CAGL,MAAM,WACJ,OACA,SAC0B;EAC1B,MAAM,CAAC,UAAU,MAAM,KAAK,eAC1B,CAAC;GAAE,IAAI;GAAS,MAAM;GAAO,CAAC,EAC9B,QACD;AACD,MAAI,CAAC,OACH,OAAM,IAAI,MAAM,wCAAwC;AAE1D,SAAO"}
|
|
@@ -15,7 +15,7 @@ declare class MistralLLMProvider implements llm_d_exports.LLMProvider {
|
|
|
15
15
|
constructor(options: MistralLLMProviderOptions);
|
|
16
16
|
chat(messages: llm_d_exports.LLMMessage[], options?: llm_d_exports.LLMChatOptions): Promise<llm_d_exports.LLMResponse>;
|
|
17
17
|
stream(messages: llm_d_exports.LLMMessage[], options?: llm_d_exports.LLMChatOptions): AsyncIterable<llm_d_exports.LLMStreamChunk>;
|
|
18
|
-
countTokens(
|
|
18
|
+
countTokens(_messages: llm_d_exports.LLMMessage[]): Promise<{
|
|
19
19
|
promptTokens: number;
|
|
20
20
|
}>;
|
|
21
21
|
private buildChatRequest;
|
|
@@ -28,4 +28,5 @@ declare class MistralLLMProvider implements llm_d_exports.LLMProvider {
|
|
|
28
28
|
private extractToolCalls;
|
|
29
29
|
}
|
|
30
30
|
//#endregion
|
|
31
|
-
export { MistralLLMProvider, MistralLLMProviderOptions };
|
|
31
|
+
export { MistralLLMProvider, MistralLLMProviderOptions };
|
|
32
|
+
//# sourceMappingURL=mistral-llm.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"mistral-llm.d.ts","names":[],"sources":["../../src/impls/mistral-llm.ts"],"sourcesContent":[],"mappings":";;;;UAciB,yBAAA;;EAAA,YAAA,CAAA,EAAA,MAAA;EAQJ,SAAA,CAAA,EAAA,MAAA;EAIU,MAAA,CAAA,EARZ,OAQY;EAkBT,eAAA,CAAA,EAAA,MAAA;;AAED,cAxBA,kBAAA,YAA8B,aAAA,CAAA,WAwB9B,CAAA;EAAR,iBAAA,MAAA;EAOS,iBAAA,YAAA;EACD,WAAA,CAAA,OAAA,EA5BU,yBA4BV;EACM,IAAA,CAAA,QAAA,EAXL,aAAA,CAAA,UAWK,EAAA,EAAA,OAAA,CAAA,EAVN,aAAA,CAAA,cAUM,CAAA,EATd,OASc,CATN,aAAA,CAAA,WASM,CAAA;EAAd,MAAA,CAAA,QAAA,EAFS,aAAA,CAAA,UAET,EAAA,EAAA,OAAA,CAAA,EADQ,aAAA,CAAA,cACR,CAAA,EAAA,aAAA,CAAc,aAAA,CAAA,cAAd,CAAA;EAuFU,WAAA,CAAA,SAAA,EAAA,aAAA,CAAA,UAAA,EAAA,CAAA,EACV,OADU,CAAA;IACV,YAAA,EAAA,MAAA;EAzHsC,CAAA,CAAA;EAAW,QAAA,gBAAA"}
|
|
@@ -103,7 +103,7 @@ var MistralLLMProvider = class {
|
|
|
103
103
|
}
|
|
104
104
|
};
|
|
105
105
|
}
|
|
106
|
-
async countTokens(
|
|
106
|
+
async countTokens(_messages) {
|
|
107
107
|
throw new Error("Mistral API does not currently support token counting");
|
|
108
108
|
}
|
|
109
109
|
buildChatRequest(messages, options) {
|
|
@@ -244,4 +244,5 @@ function mapFinishReason(reason) {
|
|
|
244
244
|
}
|
|
245
245
|
|
|
246
246
|
//#endregion
|
|
247
|
-
export { MistralLLMProvider };
|
|
247
|
+
export { MistralLLMProvider };
|
|
248
|
+
//# sourceMappingURL=mistral-llm.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"mistral-llm.js","names":["aggregatedParts: LLMContentPart[]","aggregatedToolCalls: LLMToolCallPart[]","usage: LLMTokenUsage | undefined","finishReason: string | undefined","message: LLMMessage","request: components.ChatCompletionRequest","parts: LLMContentPart[]"],"sources":["../../src/impls/mistral-llm.ts"],"sourcesContent":["import { Mistral } from '@mistralai/mistralai';\nimport type * as components from '@mistralai/mistralai/models/components/index.js';\n\nimport type {\n LLMChatOptions,\n LLMContentPart,\n LLMMessage,\n LLMProvider,\n LLMResponse,\n LLMStreamChunk,\n LLMToolCallPart,\n LLMTokenUsage,\n} from '../llm';\n\nexport interface MistralLLMProviderOptions {\n apiKey: string;\n defaultModel?: string;\n serverURL?: string;\n client?: Mistral;\n userAgentSuffix?: string;\n}\n\nexport class MistralLLMProvider implements LLMProvider {\n private readonly client: Mistral;\n private readonly defaultModel: string;\n\n constructor(options: MistralLLMProviderOptions) {\n if (!options.apiKey) {\n throw new Error('MistralLLMProvider requires an apiKey');\n }\n\n this.client =\n options.client ??\n new Mistral({\n apiKey: options.apiKey,\n serverURL: options.serverURL,\n userAgent: options.userAgentSuffix\n ? `${options.userAgentSuffix}`\n : undefined,\n });\n this.defaultModel = options.defaultModel ?? 'mistral-large-latest';\n }\n\n async chat(\n messages: LLMMessage[],\n options: LLMChatOptions = {}\n ): Promise<LLMResponse> {\n const request = this.buildChatRequest(messages, options);\n const response = await this.client.chat.complete(request);\n return this.buildLLMResponse(response);\n }\n\n async *stream(\n messages: LLMMessage[],\n options: LLMChatOptions = {}\n ): AsyncIterable<LLMStreamChunk> {\n const request = this.buildChatRequest(messages, options);\n request.stream = true;\n const stream = await this.client.chat.stream(request);\n\n const aggregatedParts: LLMContentPart[] = [];\n const aggregatedToolCalls: LLMToolCallPart[] = [];\n let usage: LLMTokenUsage | undefined;\n let finishReason: string | undefined;\n\n for await (const event of stream) {\n for (const choice of event.data.choices) {\n const delta = choice.delta;\n if (typeof delta.content === 'string') {\n if (delta.content.length > 0) {\n aggregatedParts.push({ type: 'text', text: delta.content });\n yield {\n type: 'message_delta',\n delta: { type: 'text', text: delta.content },\n index: choice.index,\n };\n }\n } else if (Array.isArray(delta.content)) {\n for (const chunk of delta.content) {\n if (chunk.type === 'text' && 'text' in chunk) {\n aggregatedParts.push({ type: 'text', text: chunk.text });\n yield {\n type: 'message_delta',\n delta: { type: 'text', text: chunk.text },\n index: choice.index,\n };\n }\n }\n }\n\n if (delta.toolCalls) {\n let localIndex = 0;\n for (const call of delta.toolCalls) {\n const toolCall = this.fromMistralToolCall(call, localIndex);\n aggregatedToolCalls.push(toolCall);\n yield {\n type: 'tool_call',\n call: toolCall,\n index: choice.index,\n };\n localIndex += 1;\n }\n }\n\n if (choice.finishReason && choice.finishReason !== 'null') {\n finishReason = choice.finishReason;\n }\n }\n\n if (event.data.usage) {\n const usageEntry = this.fromUsage(event.data.usage);\n if (usageEntry) {\n usage = usageEntry;\n yield { type: 'usage', usage: usageEntry };\n }\n }\n }\n\n const message: LLMMessage = {\n role: 'assistant',\n content: aggregatedParts.length\n ? aggregatedParts\n : [{ type: 'text', text: '' }],\n };\n if (aggregatedToolCalls.length > 0) {\n message.content = [\n ...aggregatedToolCalls,\n ...(aggregatedParts.length ? aggregatedParts : []),\n ];\n }\n\n yield {\n type: 'end',\n response: {\n message,\n usage,\n finishReason: mapFinishReason(finishReason),\n },\n };\n }\n\n async countTokens(\n _messages: LLMMessage[]\n ): Promise<{ promptTokens: number }> {\n throw new Error('Mistral API does not currently support token counting');\n }\n\n private buildChatRequest(\n messages: LLMMessage[],\n options: LLMChatOptions\n ): components.ChatCompletionRequest {\n const model = options.model ?? this.defaultModel;\n const mappedMessages = messages.map((message) =>\n this.toMistralMessage(message)\n );\n\n const request: components.ChatCompletionRequest = {\n model,\n messages: mappedMessages,\n };\n\n if (options.temperature != null) {\n request.temperature = options.temperature;\n }\n if (options.topP != null) {\n request.topP = options.topP;\n }\n if (options.maxOutputTokens != null) {\n request.maxTokens = options.maxOutputTokens;\n }\n if (options.stopSequences?.length) {\n request.stop =\n options.stopSequences.length === 1\n ? options.stopSequences[0]\n : options.stopSequences;\n }\n if (options.tools?.length) {\n request.tools = options.tools.map((tool) => ({\n type: 'function',\n function: {\n name: tool.name,\n description: tool.description,\n parameters:\n typeof tool.inputSchema === 'object' && tool.inputSchema !== null\n ? tool.inputSchema\n : {},\n },\n }));\n }\n if (options.responseFormat === 'json') {\n request.responseFormat = { type: 'json_object' };\n }\n\n return request;\n }\n\n private buildLLMResponse(\n response: components.ChatCompletionResponse\n ): LLMResponse {\n const firstChoice = response.choices[0];\n if (!firstChoice) {\n return {\n message: {\n role: 'assistant',\n content: [{ type: 'text', text: '' }],\n },\n usage: this.fromUsage(response.usage),\n raw: response,\n };\n }\n const message = this.fromAssistantMessage(firstChoice.message);\n return {\n message,\n usage: this.fromUsage(response.usage),\n finishReason: mapFinishReason(firstChoice.finishReason),\n raw: response,\n };\n }\n\n private fromUsage(\n usage: components.UsageInfo | undefined\n ): LLMTokenUsage | undefined {\n if (!usage) return undefined;\n return {\n promptTokens: usage.promptTokens ?? 0,\n completionTokens: usage.completionTokens ?? 0,\n totalTokens: usage.totalTokens ?? 0,\n };\n }\n\n private fromAssistantMessage(\n message: components.AssistantMessage\n ): LLMMessage {\n const parts: LLMContentPart[] = [];\n if (typeof message.content === 'string') {\n parts.push({ type: 'text', text: message.content });\n } else if (Array.isArray(message.content)) {\n message.content.forEach((chunk) => {\n if (chunk.type === 'text' && 'text' in chunk) {\n parts.push({ type: 'text', text: chunk.text });\n }\n });\n }\n\n const toolCalls =\n message.toolCalls?.map((call, index) =>\n this.fromMistralToolCall(call, index)\n ) ?? [];\n\n if (toolCalls.length > 0) {\n parts.splice(0, 0, ...toolCalls);\n }\n\n if (parts.length === 0) {\n parts.push({ type: 'text', text: '' });\n }\n\n return {\n role: 'assistant',\n content: parts,\n };\n }\n\n private fromMistralToolCall(\n call: components.ToolCall,\n index: number\n ): LLMToolCallPart {\n const args =\n typeof call.function.arguments === 'string'\n ? call.function.arguments\n : JSON.stringify(call.function.arguments);\n return {\n type: 'tool-call',\n id: call.id ?? `tool-call-${index}`,\n name: call.function.name,\n arguments: args,\n };\n }\n\n private toMistralMessage(message: LLMMessage): components.Messages {\n const textContent = this.extractText(message.content);\n const toolCalls = this.extractToolCalls(message);\n\n switch (message.role) {\n case 'system':\n return {\n role: 'system',\n content: textContent ?? '',\n };\n case 'user':\n return {\n role: 'user',\n content: textContent ?? '',\n };\n case 'assistant':\n return {\n role: 'assistant',\n content: toolCalls.length > 0 ? null : (textContent ?? ''),\n toolCalls: toolCalls.length > 0 ? toolCalls : undefined,\n };\n case 'tool':\n return {\n role: 'tool',\n content: textContent ?? '',\n toolCallId: message.toolCallId ?? toolCalls[0]?.id,\n };\n default:\n return {\n role: 'user',\n content: textContent ?? '',\n };\n }\n }\n\n private extractText(parts: LLMContentPart[]): string | null {\n const textParts = parts\n .filter((part) => part.type === 'text')\n .map((part) => (part as Extract<LLMContentPart, { type: 'text' }>).text);\n if (textParts.length === 0) return null;\n return textParts.join('');\n }\n\n private extractToolCalls(message: LLMMessage): components.ToolCall[] {\n const toolCallParts = message.content.filter(\n (part): part is LLMToolCallPart => part.type === 'tool-call'\n );\n return toolCallParts.map((call, index) => ({\n id: call.id ?? `call_${index}`,\n type: 'function',\n index,\n function: {\n name: call.name,\n arguments: call.arguments,\n },\n }));\n }\n}\n\nfunction mapFinishReason(\n reason?: string | null\n): LLMResponse['finishReason'] | undefined {\n if (!reason) return undefined;\n const normalized = reason.toLowerCase();\n switch (normalized) {\n case 'stop':\n return 'stop';\n case 'length':\n return 'length';\n case 'tool_call':\n case 'tool_calls':\n return 'tool_call';\n case 'content_filter':\n return 'content_filter';\n default:\n return undefined;\n }\n}\n"],"mappings":";;;AAsBA,IAAa,qBAAb,MAAuD;CACrD,AAAiB;CACjB,AAAiB;CAEjB,YAAY,SAAoC;AAC9C,MAAI,CAAC,QAAQ,OACX,OAAM,IAAI,MAAM,wCAAwC;AAG1D,OAAK,SACH,QAAQ,UACR,IAAI,QAAQ;GACV,QAAQ,QAAQ;GAChB,WAAW,QAAQ;GACnB,WAAW,QAAQ,kBACf,GAAG,QAAQ,oBACX;GACL,CAAC;AACJ,OAAK,eAAe,QAAQ,gBAAgB;;CAG9C,MAAM,KACJ,UACA,UAA0B,EAAE,EACN;EACtB,MAAM,UAAU,KAAK,iBAAiB,UAAU,QAAQ;EACxD,MAAM,WAAW,MAAM,KAAK,OAAO,KAAK,SAAS,QAAQ;AACzD,SAAO,KAAK,iBAAiB,SAAS;;CAGxC,OAAO,OACL,UACA,UAA0B,EAAE,EACG;EAC/B,MAAM,UAAU,KAAK,iBAAiB,UAAU,QAAQ;AACxD,UAAQ,SAAS;EACjB,MAAM,SAAS,MAAM,KAAK,OAAO,KAAK,OAAO,QAAQ;EAErD,MAAMA,kBAAoC,EAAE;EAC5C,MAAMC,sBAAyC,EAAE;EACjD,IAAIC;EACJ,IAAIC;AAEJ,aAAW,MAAM,SAAS,QAAQ;AAChC,QAAK,MAAM,UAAU,MAAM,KAAK,SAAS;IACvC,MAAM,QAAQ,OAAO;AACrB,QAAI,OAAO,MAAM,YAAY,UAC3B;SAAI,MAAM,QAAQ,SAAS,GAAG;AAC5B,sBAAgB,KAAK;OAAE,MAAM;OAAQ,MAAM,MAAM;OAAS,CAAC;AAC3D,YAAM;OACJ,MAAM;OACN,OAAO;QAAE,MAAM;QAAQ,MAAM,MAAM;QAAS;OAC5C,OAAO,OAAO;OACf;;eAEM,MAAM,QAAQ,MAAM,QAAQ,EACrC;UAAK,MAAM,SAAS,MAAM,QACxB,KAAI,MAAM,SAAS,UAAU,UAAU,OAAO;AAC5C,sBAAgB,KAAK;OAAE,MAAM;OAAQ,MAAM,MAAM;OAAM,CAAC;AACxD,YAAM;OACJ,MAAM;OACN,OAAO;QAAE,MAAM;QAAQ,MAAM,MAAM;QAAM;OACzC,OAAO,OAAO;OACf;;;AAKP,QAAI,MAAM,WAAW;KACnB,IAAI,aAAa;AACjB,UAAK,MAAM,QAAQ,MAAM,WAAW;MAClC,MAAM,WAAW,KAAK,oBAAoB,MAAM,WAAW;AAC3D,0BAAoB,KAAK,SAAS;AAClC,YAAM;OACJ,MAAM;OACN,MAAM;OACN,OAAO,OAAO;OACf;AACD,oBAAc;;;AAIlB,QAAI,OAAO,gBAAgB,OAAO,iBAAiB,OACjD,gBAAe,OAAO;;AAI1B,OAAI,MAAM,KAAK,OAAO;IACpB,MAAM,aAAa,KAAK,UAAU,MAAM,KAAK,MAAM;AACnD,QAAI,YAAY;AACd,aAAQ;AACR,WAAM;MAAE,MAAM;MAAS,OAAO;MAAY;;;;EAKhD,MAAMC,UAAsB;GAC1B,MAAM;GACN,SAAS,gBAAgB,SACrB,kBACA,CAAC;IAAE,MAAM;IAAQ,MAAM;IAAI,CAAC;GACjC;AACD,MAAI,oBAAoB,SAAS,EAC/B,SAAQ,UAAU,CAChB,GAAG,qBACH,GAAI,gBAAgB,SAAS,kBAAkB,EAAE,CAClD;AAGH,QAAM;GACJ,MAAM;GACN,UAAU;IACR;IACA;IACA,cAAc,gBAAgB,aAAa;IAC5C;GACF;;CAGH,MAAM,YACJ,WACmC;AACnC,QAAM,IAAI,MAAM,wDAAwD;;CAG1E,AAAQ,iBACN,UACA,SACkC;EAMlC,MAAMC,UAA4C;GAChD,OANY,QAAQ,SAAS,KAAK;GAOlC,UANqB,SAAS,KAAK,YACnC,KAAK,iBAAiB,QAAQ,CAC/B;GAKA;AAED,MAAI,QAAQ,eAAe,KACzB,SAAQ,cAAc,QAAQ;AAEhC,MAAI,QAAQ,QAAQ,KAClB,SAAQ,OAAO,QAAQ;AAEzB,MAAI,QAAQ,mBAAmB,KAC7B,SAAQ,YAAY,QAAQ;AAE9B,MAAI,QAAQ,eAAe,OACzB,SAAQ,OACN,QAAQ,cAAc,WAAW,IAC7B,QAAQ,cAAc,KACtB,QAAQ;AAEhB,MAAI,QAAQ,OAAO,OACjB,SAAQ,QAAQ,QAAQ,MAAM,KAAK,UAAU;GAC3C,MAAM;GACN,UAAU;IACR,MAAM,KAAK;IACX,aAAa,KAAK;IAClB,YACE,OAAO,KAAK,gBAAgB,YAAY,KAAK,gBAAgB,OACzD,KAAK,cACL,EAAE;IACT;GACF,EAAE;AAEL,MAAI,QAAQ,mBAAmB,OAC7B,SAAQ,iBAAiB,EAAE,MAAM,eAAe;AAGlD,SAAO;;CAGT,AAAQ,iBACN,UACa;EACb,MAAM,cAAc,SAAS,QAAQ;AACrC,MAAI,CAAC,YACH,QAAO;GACL,SAAS;IACP,MAAM;IACN,SAAS,CAAC;KAAE,MAAM;KAAQ,MAAM;KAAI,CAAC;IACtC;GACD,OAAO,KAAK,UAAU,SAAS,MAAM;GACrC,KAAK;GACN;AAGH,SAAO;GACL,SAFc,KAAK,qBAAqB,YAAY,QAAQ;GAG5D,OAAO,KAAK,UAAU,SAAS,MAAM;GACrC,cAAc,gBAAgB,YAAY,aAAa;GACvD,KAAK;GACN;;CAGH,AAAQ,UACN,OAC2B;AAC3B,MAAI,CAAC,MAAO,QAAO;AACnB,SAAO;GACL,cAAc,MAAM,gBAAgB;GACpC,kBAAkB,MAAM,oBAAoB;GAC5C,aAAa,MAAM,eAAe;GACnC;;CAGH,AAAQ,qBACN,SACY;EACZ,MAAMC,QAA0B,EAAE;AAClC,MAAI,OAAO,QAAQ,YAAY,SAC7B,OAAM,KAAK;GAAE,MAAM;GAAQ,MAAM,QAAQ;GAAS,CAAC;WAC1C,MAAM,QAAQ,QAAQ,QAAQ,CACvC,SAAQ,QAAQ,SAAS,UAAU;AACjC,OAAI,MAAM,SAAS,UAAU,UAAU,MACrC,OAAM,KAAK;IAAE,MAAM;IAAQ,MAAM,MAAM;IAAM,CAAC;IAEhD;EAGJ,MAAM,YACJ,QAAQ,WAAW,KAAK,MAAM,UAC5B,KAAK,oBAAoB,MAAM,MAAM,CACtC,IAAI,EAAE;AAET,MAAI,UAAU,SAAS,EACrB,OAAM,OAAO,GAAG,GAAG,GAAG,UAAU;AAGlC,MAAI,MAAM,WAAW,EACnB,OAAM,KAAK;GAAE,MAAM;GAAQ,MAAM;GAAI,CAAC;AAGxC,SAAO;GACL,MAAM;GACN,SAAS;GACV;;CAGH,AAAQ,oBACN,MACA,OACiB;EACjB,MAAM,OACJ,OAAO,KAAK,SAAS,cAAc,WAC/B,KAAK,SAAS,YACd,KAAK,UAAU,KAAK,SAAS,UAAU;AAC7C,SAAO;GACL,MAAM;GACN,IAAI,KAAK,MAAM,aAAa;GAC5B,MAAM,KAAK,SAAS;GACpB,WAAW;GACZ;;CAGH,AAAQ,iBAAiB,SAA0C;EACjE,MAAM,cAAc,KAAK,YAAY,QAAQ,QAAQ;EACrD,MAAM,YAAY,KAAK,iBAAiB,QAAQ;AAEhD,UAAQ,QAAQ,MAAhB;GACE,KAAK,SACH,QAAO;IACL,MAAM;IACN,SAAS,eAAe;IACzB;GACH,KAAK,OACH,QAAO;IACL,MAAM;IACN,SAAS,eAAe;IACzB;GACH,KAAK,YACH,QAAO;IACL,MAAM;IACN,SAAS,UAAU,SAAS,IAAI,OAAQ,eAAe;IACvD,WAAW,UAAU,SAAS,IAAI,YAAY;IAC/C;GACH,KAAK,OACH,QAAO;IACL,MAAM;IACN,SAAS,eAAe;IACxB,YAAY,QAAQ,cAAc,UAAU,IAAI;IACjD;GACH,QACE,QAAO;IACL,MAAM;IACN,SAAS,eAAe;IACzB;;;CAIP,AAAQ,YAAY,OAAwC;EAC1D,MAAM,YAAY,MACf,QAAQ,SAAS,KAAK,SAAS,OAAO,CACtC,KAAK,SAAU,KAAmD,KAAK;AAC1E,MAAI,UAAU,WAAW,EAAG,QAAO;AACnC,SAAO,UAAU,KAAK,GAAG;;CAG3B,AAAQ,iBAAiB,SAA4C;AAInE,SAHsB,QAAQ,QAAQ,QACnC,SAAkC,KAAK,SAAS,YAClD,CACoB,KAAK,MAAM,WAAW;GACzC,IAAI,KAAK,MAAM,QAAQ;GACvB,MAAM;GACN;GACA,UAAU;IACR,MAAM,KAAK;IACX,WAAW,KAAK;IACjB;GACF,EAAE;;;AAIP,SAAS,gBACP,QACyC;AACzC,KAAI,CAAC,OAAQ,QAAO;AAEpB,SADmB,OAAO,aAAa,EACvC;EACE,KAAK,OACH,QAAO;EACT,KAAK,SACH,QAAO;EACT,KAAK;EACL,KAAK,aACH,QAAO;EACT,KAAK,iBACH,QAAO;EACT,QACE"}
|
|
@@ -16,4 +16,5 @@ declare class PostmarkEmailProvider implements email_d_exports.EmailOutboundProv
|
|
|
16
16
|
sendEmail(message: email_d_exports.EmailOutboundMessage): Promise<email_d_exports.EmailOutboundResult>;
|
|
17
17
|
}
|
|
18
18
|
//#endregion
|
|
19
|
-
export { PostmarkEmailProvider, PostmarkEmailProviderOptions };
|
|
19
|
+
export { PostmarkEmailProvider, PostmarkEmailProviderOptions };
|
|
20
|
+
//# sourceMappingURL=postmark-email.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"postmark-email.d.ts","names":[],"sources":["../../src/impls/postmark-email.ts"],"sourcesContent":[],"mappings":";;;;UAQiB,4BAAA;;EAAA,gBAAA,CAAA,EAAA,MAAA;EAOJ,aAAA,CAAA,EAAA,MAAA;EAKU,MAAA,CAAA,EARZ,YAQY;;AAUmC,cAf7C,qBAAA,YAAiC,eAAA,CAAA,qBAeY,CAAA;EAAR,iBAAA,MAAA;EAfJ,iBAAA,gBAAA;EAAqB,iBAAA,aAAA;uBAK5C;qBAUI,eAAA,CAAA,uBAAuB,QAAQ,eAAA,CAAA"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"postmark-email.js","names":["request: PostmarkModels.Message"],"sources":["../../src/impls/postmark-email.ts"],"sourcesContent":["import { ServerClient, type Models as PostmarkModels } from 'postmark';\n\nimport type {\n EmailOutboundMessage,\n EmailOutboundProvider,\n EmailOutboundResult,\n} from '../email';\n\nexport interface PostmarkEmailProviderOptions {\n serverToken: string;\n defaultFromEmail?: string;\n messageStream?: string;\n client?: ServerClient;\n}\n\nexport class PostmarkEmailProvider implements EmailOutboundProvider {\n private readonly client: ServerClient;\n private readonly defaultFromEmail?: string;\n private readonly messageStream?: string;\n\n constructor(options: PostmarkEmailProviderOptions) {\n this.client =\n options.client ??\n new ServerClient(options.serverToken, {\n useHttps: true,\n });\n this.defaultFromEmail = options.defaultFromEmail;\n this.messageStream = options.messageStream;\n }\n\n async sendEmail(message: EmailOutboundMessage): Promise<EmailOutboundResult> {\n const request: PostmarkModels.Message = {\n From: formatAddress(message.from) ?? this.defaultFromEmail,\n To: message.to.map((addr) => formatAddress(addr)).join(', '),\n Cc:\n message.cc?.map((addr) => formatAddress(addr)).join(', ') || undefined,\n Bcc:\n message.bcc?.map((addr) => formatAddress(addr)).join(', ') || undefined,\n ReplyTo: message.replyTo ? formatAddress(message.replyTo) : undefined,\n Subject: message.subject,\n TextBody: message.textBody,\n HtmlBody: message.htmlBody,\n Headers: message.headers\n ? Object.entries(message.headers).map(([name, value]) => ({\n Name: name,\n Value: value,\n }))\n : undefined,\n MessageStream: this.messageStream,\n Attachments: buildAttachments(message),\n };\n\n const response = await this.client.sendEmail(request);\n return {\n id: response.MessageID,\n providerMessageId: response.MessageID,\n queuedAt: new Date(response.SubmittedAt ?? new Date().toISOString()),\n };\n }\n}\n\nfunction formatAddress(address: { email: string; name?: string }): string {\n if (address.name) {\n return `\"${address.name}\" <${address.email}>`;\n }\n return address.email;\n}\n\nfunction buildAttachments(\n message: EmailOutboundMessage\n): PostmarkModels.Attachment[] | undefined {\n if (!message.attachments?.length) return undefined;\n return message.attachments\n .filter((attachment) => attachment.data)\n .map((attachment) => ({\n Name: attachment.filename,\n Content: Buffer.from(attachment.data ?? new Uint8Array()).toString(\n 'base64'\n ),\n ContentType: attachment.contentType,\n ContentID: null,\n ContentLength: attachment.sizeBytes,\n Disposition: 'attachment',\n }));\n}\n"],"mappings":";;;AAeA,IAAa,wBAAb,MAAoE;CAClE,AAAiB;CACjB,AAAiB;CACjB,AAAiB;CAEjB,YAAY,SAAuC;AACjD,OAAK,SACH,QAAQ,UACR,IAAI,aAAa,QAAQ,aAAa,EACpC,UAAU,MACX,CAAC;AACJ,OAAK,mBAAmB,QAAQ;AAChC,OAAK,gBAAgB,QAAQ;;CAG/B,MAAM,UAAU,SAA6D;EAC3E,MAAMA,UAAkC;GACtC,MAAM,cAAc,QAAQ,KAAK,IAAI,KAAK;GAC1C,IAAI,QAAQ,GAAG,KAAK,SAAS,cAAc,KAAK,CAAC,CAAC,KAAK,KAAK;GAC5D,IACE,QAAQ,IAAI,KAAK,SAAS,cAAc,KAAK,CAAC,CAAC,KAAK,KAAK,IAAI;GAC/D,KACE,QAAQ,KAAK,KAAK,SAAS,cAAc,KAAK,CAAC,CAAC,KAAK,KAAK,IAAI;GAChE,SAAS,QAAQ,UAAU,cAAc,QAAQ,QAAQ,GAAG;GAC5D,SAAS,QAAQ;GACjB,UAAU,QAAQ;GAClB,UAAU,QAAQ;GAClB,SAAS,QAAQ,UACb,OAAO,QAAQ,QAAQ,QAAQ,CAAC,KAAK,CAAC,MAAM,YAAY;IACtD,MAAM;IACN,OAAO;IACR,EAAE,GACH;GACJ,eAAe,KAAK;GACpB,aAAa,iBAAiB,QAAQ;GACvC;EAED,MAAM,WAAW,MAAM,KAAK,OAAO,UAAU,QAAQ;AACrD,SAAO;GACL,IAAI,SAAS;GACb,mBAAmB,SAAS;GAC5B,UAAU,IAAI,KAAK,SAAS,gCAAe,IAAI,MAAM,EAAC,aAAa,CAAC;GACrE;;;AAIL,SAAS,cAAc,SAAmD;AACxE,KAAI,QAAQ,KACV,QAAO,IAAI,QAAQ,KAAK,KAAK,QAAQ,MAAM;AAE7C,QAAO,QAAQ;;AAGjB,SAAS,iBACP,SACyC;AACzC,KAAI,CAAC,QAAQ,aAAa,OAAQ,QAAO;AACzC,QAAO,QAAQ,YACZ,QAAQ,eAAe,WAAW,KAAK,CACvC,KAAK,gBAAgB;EACpB,MAAM,WAAW;EACjB,SAAS,OAAO,KAAK,WAAW,QAAQ,IAAI,YAAY,CAAC,CAAC,SACxD,SACD;EACD,aAAa,WAAW;EACxB,WAAW;EACX,eAAe,WAAW;EAC1B,aAAa;EACd,EAAE"}
|
|
@@ -121,4 +121,5 @@ declare class PowensClient {
|
|
|
121
121
|
private fetchAccessToken;
|
|
122
122
|
}
|
|
123
123
|
//#endregion
|
|
124
|
-
export { PowensAccount, PowensAccountListResponse, PowensBalance, PowensClient, PowensClientError, PowensClientOptions, PowensConnectionStatusResponse, PowensEnvironment, PowensTransaction, PowensTransactionListResponse };
|
|
124
|
+
export { PowensAccount, PowensAccountListResponse, PowensBalance, PowensClient, PowensClientError, PowensClientOptions, PowensConnectionStatusResponse, PowensEnvironment, PowensTransaction, PowensTransactionListResponse };
|
|
125
|
+
//# sourceMappingURL=powens-client.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"powens-client.d.ts","names":[],"sources":["../../src/impls/powens-client.ts"],"sourcesContent":[],"mappings":";KAEY,iBAAA;AAAA,UAOK,mBAAA,CAPY;EAOZ,QAAA,EAAA,MAAA;EA4BA,YAAA,EAAA,MAAa;EAoBb,MAAA,CAAA,EAAA,MAAA;EAQA,WAAA,EApDF,iBAoDmB;EAmBjB,OAAA,CAAA,EAAA,MAAA;EAQA,SAAA,CAAA,EAAA,OA7EI,KA6ES;EASb,gBAAA,CAAA,EAAA,MAAA;EASJ,MAAA,CAAA,EAAA;IAgCA,KAAA,CAAA,EAAA,CAAA,GAAY,IAAA,EAAA,OAAA,EAAA,EAAA,GAAA,IAAA;IAUF,IAAA,CAAA,EAAA,CAAA,GAAA,IAAA,EAAA,OAAA,EAAA,EAAA,GAAA,IAAA;IAmBT,KAAA,CAAA,EAAA,CAAA,GAAA,IAAA,EAAA,OAAA,EAAA,EAAA,GAAA,IAAA;EAAR,CAAA;;AAgBmC,UAtJxB,aAAA,CAsJwB;EAc3B,IAAA,EAAA,MAAA;EAAR,SAAA,EAAA,MAAA;EAgB4C,QAAA,EAAA,MAAA;EAAR,WAAA,EAAA;IAS7B,EAAA,EAAA,MAAA;IAAR,IAAA,EAAA,MAAA;IAAO,OAAA,CAAA,EAAA,MAAA;;;;;;;;;;aA5KC;;UAGI,yBAAA;YACL;;;;;;UAOK,iBAAA;;;;;;;;;;;;;;;;aAgBJ;;UAGI,6BAAA;gBACD;;;;;;UAOC,aAAA;;;;;;aAMJ;;UAGI,8BAAA;;;;;;aAMJ;;cAGA,iBAAA,SAA0B,KAAA;;;;;;;cAgC1B,YAAA;;;;;;;;;uBAUU;;;;;;;MAmBjB,QAAQ;mCAgB2B,QAAQ;;;;;;;;MAc3C,QAAQ;oCAgB4B,QAAQ;+CAS7C,QAAQ"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"powens-client.js","names":["POWENS_BASE_URL: Record<PowensEnvironment, string>","searchParams: Record<string, string | number | boolean | undefined>","headers: Record<string, string>","requestId: string | undefined","errorBody: unknown","message: string"],"sources":["../../src/impls/powens-client.ts"],"sourcesContent":["import { URL } from 'node:url';\n\nexport type PowensEnvironment = 'sandbox' | 'production';\n\nconst POWENS_BASE_URL: Record<PowensEnvironment, string> = {\n sandbox: 'https://api-sandbox.powens.com/v2',\n production: 'https://api.powens.com/v2',\n};\n\nexport interface PowensClientOptions {\n clientId: string;\n clientSecret: string;\n apiKey?: string;\n environment: PowensEnvironment;\n baseUrl?: string;\n fetchImpl?: typeof fetch;\n defaultTimeoutMs?: number;\n logger?: {\n debug?: (...args: unknown[]) => void;\n warn?: (...args: unknown[]) => void;\n error?: (...args: unknown[]) => void;\n };\n}\n\ninterface PowensOAuthTokenResponse {\n access_token: string;\n expires_in: number;\n scope?: string;\n token_type: string;\n}\n\ninterface PowensOAuthToken {\n accessToken: string;\n expiresAt: number;\n scope?: string;\n}\n\nexport interface PowensAccount {\n uuid: string;\n reference: string;\n userUuid: string;\n institution: {\n id: string;\n name: string;\n logoUrl?: string;\n };\n name: string;\n iban?: string;\n bic?: string;\n currency?: string;\n balance?: number;\n availableBalance?: number;\n type?: string;\n status?: string;\n metadata?: Record<string, unknown>;\n}\n\nexport interface PowensAccountListResponse {\n accounts: PowensAccount[];\n pagination?: {\n nextCursor?: string;\n hasMore?: boolean;\n };\n}\n\nexport interface PowensTransaction {\n uuid: string;\n accountUuid: string;\n amount: number;\n currency: string;\n direction: 'credit' | 'debit';\n description?: string;\n category?: string;\n rawLabel?: string;\n bookingDate?: string;\n valueDate?: string;\n status?: string;\n merchantName?: string;\n merchantCategoryCode?: string;\n counterpartyName?: string;\n counterpartyAccount?: string;\n metadata?: Record<string, unknown>;\n}\n\nexport interface PowensTransactionListResponse {\n transactions: PowensTransaction[];\n pagination?: {\n nextCursor?: string;\n hasMore?: boolean;\n };\n}\n\nexport interface PowensBalance {\n accountUuid: string;\n type: string;\n amount: number;\n currency: string;\n updatedAt: string;\n metadata?: Record<string, unknown>;\n}\n\nexport interface PowensConnectionStatusResponse {\n connectionUuid: string;\n status: 'healthy' | 'error' | 'revoked' | 'pending';\n lastAttemptAt?: string;\n errorCode?: string;\n errorMessage?: string;\n metadata?: Record<string, unknown>;\n}\n\nexport class PowensClientError extends Error {\n readonly status: number;\n readonly code?: string;\n readonly requestId?: string;\n readonly response?: unknown;\n\n constructor(\n message: string,\n status: number,\n code?: string,\n requestId?: string,\n response?: unknown\n ) {\n super(message);\n this.name = 'PowensClientError';\n this.status = status;\n this.code = code;\n this.requestId = requestId;\n this.response = response;\n }\n}\n\ninterface RequestOptions {\n method: string;\n path: string;\n searchParams?: Record<string, string | number | boolean | undefined>;\n body?: Record<string, unknown> | undefined;\n headers?: Record<string, string | undefined>;\n timeoutMs?: number;\n skipAuth?: boolean;\n}\n\nexport class PowensClient {\n private readonly clientId: string;\n private readonly clientSecret: string;\n private readonly apiKey?: string;\n private readonly fetchImpl: typeof fetch;\n private readonly logger?: PowensClientOptions['logger'];\n private readonly defaultTimeoutMs: number;\n private token?: PowensOAuthToken;\n private readonly baseUrl: string;\n\n constructor(options: PowensClientOptions) {\n this.clientId = options.clientId;\n this.clientSecret = options.clientSecret;\n this.apiKey = options.apiKey;\n this.fetchImpl = options.fetchImpl ?? fetch;\n this.logger = options.logger;\n this.defaultTimeoutMs = options.defaultTimeoutMs ?? 15000;\n this.baseUrl =\n options.baseUrl ??\n POWENS_BASE_URL[options.environment] ??\n POWENS_BASE_URL.production;\n }\n\n async listAccounts(params: {\n userUuid: string;\n cursor?: string;\n limit?: number;\n includeBalances?: boolean;\n institutionUuid?: string;\n }): Promise<PowensAccountListResponse> {\n const searchParams: Record<string, string | number | boolean | undefined> =\n {\n cursor: params.cursor,\n limit: params.limit,\n include_balances: params.includeBalances,\n institution_uuid: params.institutionUuid,\n };\n const response = await this.request<PowensAccountListResponse>({\n method: 'GET',\n path: `/users/${encodeURIComponent(params.userUuid)}/accounts`,\n searchParams,\n });\n return response;\n }\n\n async getAccount(accountUuid: string): Promise<PowensAccount> {\n return this.request<PowensAccount>({\n method: 'GET',\n path: `/accounts/${encodeURIComponent(accountUuid)}`,\n });\n }\n\n async listTransactions(params: {\n accountUuid: string;\n cursor?: string;\n limit?: number;\n from?: string;\n to?: string;\n includePending?: boolean;\n }): Promise<PowensTransactionListResponse> {\n const searchParams: Record<string, string | number | boolean | undefined> =\n {\n cursor: params.cursor,\n limit: params.limit,\n from: params.from,\n to: params.to,\n include_pending: params.includePending,\n };\n return this.request<PowensTransactionListResponse>({\n method: 'GET',\n path: `/accounts/${encodeURIComponent(params.accountUuid)}/transactions`,\n searchParams,\n });\n }\n\n async getBalances(accountUuid: string): Promise<PowensBalance[]> {\n return this.request<PowensBalance[]>({\n method: 'GET',\n path: `/accounts/${encodeURIComponent(accountUuid)}/balances`,\n });\n }\n\n async getConnectionStatus(\n connectionUuid: string\n ): Promise<PowensConnectionStatusResponse> {\n return this.request<PowensConnectionStatusResponse>({\n method: 'GET',\n path: `/connections/${encodeURIComponent(connectionUuid)}`,\n });\n }\n\n private async request<T>(options: RequestOptions): Promise<T> {\n const url = new URL(options.path, this.baseUrl);\n if (options.searchParams) {\n for (const [key, value] of Object.entries(options.searchParams)) {\n if (value === undefined || value === null) continue;\n url.searchParams.set(key, String(value));\n }\n }\n\n const headers: Record<string, string> = {\n Accept: 'application/json',\n 'Content-Type': 'application/json',\n ...options.headers,\n };\n\n if (this.apiKey) {\n headers['x-api-key'] = this.apiKey;\n }\n\n if (!options.skipAuth) {\n const token = await this.ensureAccessToken();\n headers.Authorization = `Bearer ${token}`;\n }\n\n const controller = new AbortController();\n const timeout = setTimeout(\n () => controller.abort(),\n options.timeoutMs ?? this.defaultTimeoutMs\n );\n\n try {\n const response = await this.fetchImpl(url, {\n method: options.method,\n headers,\n body: options.body ? JSON.stringify(options.body) : undefined,\n signal: controller.signal,\n });\n\n const requestId: string | undefined =\n response.headers.get('x-request-id') ?? undefined;\n\n if (!response.ok) {\n let errorBody: unknown;\n try {\n errorBody = await response.json();\n } catch {\n // ignore parsing errors\n }\n const errorObject =\n typeof errorBody === 'object' && errorBody !== null\n ? (errorBody as Record<string, unknown>)\n : undefined;\n const message: string =\n typeof errorObject?.message === 'string'\n ? errorObject.message\n : `Powens API request failed with status ${response.status}`;\n const code =\n typeof errorObject?.code === 'string' ? errorObject.code : undefined;\n throw new PowensClientError(\n message,\n response.status,\n code,\n requestId,\n errorBody\n );\n }\n\n if (response.status === 204) {\n return undefined as T;\n }\n\n try {\n return (await response.json()) as T;\n } catch (error) {\n this.logger?.error?.(\n '[PowensClient] Failed to parse JSON response',\n error\n );\n throw new PowensClientError(\n 'Failed to parse Powens response payload as JSON',\n response.status,\n undefined,\n requestId\n );\n }\n } catch (error) {\n if (error instanceof PowensClientError) {\n throw error;\n }\n if ((error as Error).name === 'AbortError') {\n throw new PowensClientError(\n `Powens API request timed out after ${options.timeoutMs ?? this.defaultTimeoutMs}ms`,\n 408\n );\n }\n this.logger?.error?.('[PowensClient] Request failed', error);\n throw new PowensClientError(\n (error as Error).message ?? 'Powens API request failed',\n 500\n );\n } finally {\n clearTimeout(timeout);\n }\n }\n\n private async ensureAccessToken(): Promise<string> {\n if (this.token && Date.now() < this.token.expiresAt - 5000) {\n return this.token.accessToken;\n }\n this.token = await this.fetchAccessToken();\n return this.token.accessToken;\n }\n\n private async fetchAccessToken(): Promise<PowensOAuthToken> {\n const url = new URL('/oauth/token', this.baseUrl);\n const basicAuth = Buffer.from(\n `${this.clientId}:${this.clientSecret}`,\n 'utf-8'\n ).toString('base64');\n\n const response = await this.fetchImpl(url, {\n method: 'POST',\n headers: {\n Authorization: `Basic ${basicAuth}`,\n 'Content-Type': 'application/x-www-form-urlencoded',\n Accept: 'application/json',\n },\n body: new URLSearchParams({\n grant_type: 'client_credentials',\n }).toString(),\n });\n\n if (!response.ok) {\n let errorBody: unknown;\n try {\n errorBody = await response.json();\n } catch {\n // ignore\n }\n const errorObject =\n typeof errorBody === 'object' && errorBody !== null\n ? (errorBody as Record<string, unknown>)\n : undefined;\n const message: string =\n typeof errorObject?.error_description === 'string'\n ? errorObject.error_description\n : 'Failed to obtain Powens access token';\n throw new PowensClientError(\n message,\n response.status,\n undefined,\n undefined,\n errorBody\n );\n }\n\n const payload = (await response.json()) as PowensOAuthTokenResponse;\n const expiresAt = Date.now() + payload.expires_in * 1000;\n this.logger?.debug?.('[PowensClient] Received access token', {\n expiresIn: payload.expires_in,\n });\n return {\n accessToken: payload.access_token,\n expiresAt,\n scope: payload.scope,\n };\n }\n}\n"],"mappings":"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"}
|