@llumiverse/drivers 0.15.0 → 0.16.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +3 -3
- package/lib/cjs/adobe/firefly.js +119 -0
- package/lib/cjs/adobe/firefly.js.map +1 -0
- package/lib/cjs/bedrock/converse.js +177 -0
- package/lib/cjs/bedrock/converse.js.map +1 -0
- package/lib/cjs/bedrock/index.js +329 -228
- package/lib/cjs/bedrock/index.js.map +1 -1
- package/lib/cjs/bedrock/nova-image-payload.js +207 -0
- package/lib/cjs/bedrock/nova-image-payload.js.map +1 -0
- package/lib/cjs/groq/index.js +34 -9
- package/lib/cjs/groq/index.js.map +1 -1
- package/lib/cjs/huggingface_ie.js +28 -12
- package/lib/cjs/huggingface_ie.js.map +1 -1
- package/lib/cjs/index.js +1 -0
- package/lib/cjs/index.js.map +1 -1
- package/lib/cjs/mistral/index.js +31 -12
- package/lib/cjs/mistral/index.js.map +1 -1
- package/lib/cjs/mistral/types.js.map +1 -1
- package/lib/cjs/openai/index.js +149 -27
- package/lib/cjs/openai/index.js.map +1 -1
- package/lib/cjs/replicate.js +16 -18
- package/lib/cjs/replicate.js.map +1 -1
- package/lib/cjs/test/TestValidationErrorCompletionStream.js.map +1 -1
- package/lib/cjs/test/index.js.map +1 -1
- package/lib/cjs/togetherai/index.js +40 -10
- package/lib/cjs/togetherai/index.js.map +1 -1
- package/lib/cjs/vertexai/embeddings/embeddings-image.js +26 -0
- package/lib/cjs/vertexai/embeddings/embeddings-image.js.map +1 -0
- package/lib/cjs/vertexai/embeddings/embeddings-text.js +1 -1
- package/lib/cjs/vertexai/embeddings/embeddings-text.js.map +1 -1
- package/lib/cjs/vertexai/index.js +88 -24
- package/lib/cjs/vertexai/index.js.map +1 -1
- package/lib/cjs/vertexai/models/claude.js +252 -0
- package/lib/cjs/vertexai/models/claude.js.map +1 -0
- package/lib/cjs/vertexai/models/gemini.js +147 -22
- package/lib/cjs/vertexai/models/gemini.js.map +1 -1
- package/lib/cjs/vertexai/models/imagen.js +317 -0
- package/lib/cjs/vertexai/models/imagen.js.map +1 -0
- package/lib/cjs/vertexai/models.js +12 -64
- package/lib/cjs/vertexai/models.js.map +1 -1
- package/lib/cjs/watsonx/index.js +39 -8
- package/lib/cjs/watsonx/index.js.map +1 -1
- package/lib/cjs/xai/index.js +71 -0
- package/lib/cjs/xai/index.js.map +1 -0
- package/lib/esm/adobe/firefly.js +115 -0
- package/lib/esm/adobe/firefly.js.map +1 -0
- package/lib/esm/bedrock/converse.js +171 -0
- package/lib/esm/bedrock/converse.js.map +1 -0
- package/lib/esm/bedrock/index.js +331 -230
- package/lib/esm/bedrock/index.js.map +1 -1
- package/lib/esm/bedrock/nova-image-payload.js +203 -0
- package/lib/esm/bedrock/nova-image-payload.js.map +1 -0
- package/lib/esm/groq/index.js +34 -9
- package/lib/esm/groq/index.js.map +1 -1
- package/lib/esm/huggingface_ie.js +29 -13
- package/lib/esm/huggingface_ie.js.map +1 -1
- package/lib/esm/index.js +1 -0
- package/lib/esm/index.js.map +1 -1
- package/lib/esm/mistral/index.js +31 -12
- package/lib/esm/mistral/index.js.map +1 -1
- package/lib/esm/mistral/types.js.map +1 -1
- package/lib/esm/openai/index.js +150 -28
- package/lib/esm/openai/index.js.map +1 -1
- package/lib/esm/replicate.js +17 -19
- package/lib/esm/replicate.js.map +1 -1
- package/lib/esm/test/TestValidationErrorCompletionStream.js.map +1 -1
- package/lib/esm/test/index.js.map +1 -1
- package/lib/esm/togetherai/index.js +40 -10
- package/lib/esm/togetherai/index.js.map +1 -1
- package/lib/esm/vertexai/embeddings/embeddings-image.js +23 -0
- package/lib/esm/vertexai/embeddings/embeddings-image.js.map +1 -0
- package/lib/esm/vertexai/embeddings/embeddings-text.js +1 -1
- package/lib/esm/vertexai/embeddings/embeddings-text.js.map +1 -1
- package/lib/esm/vertexai/index.js +89 -26
- package/lib/esm/vertexai/index.js.map +1 -1
- package/lib/esm/vertexai/models/claude.js +247 -0
- package/lib/esm/vertexai/models/claude.js.map +1 -0
- package/lib/esm/vertexai/models/gemini.js +148 -23
- package/lib/esm/vertexai/models/gemini.js.map +1 -1
- package/lib/esm/vertexai/models/imagen.js +310 -0
- package/lib/esm/vertexai/models/imagen.js.map +1 -0
- package/lib/esm/vertexai/models.js +12 -61
- package/lib/esm/vertexai/models.js.map +1 -1
- package/lib/esm/watsonx/index.js +39 -8
- package/lib/esm/watsonx/index.js.map +1 -1
- package/lib/esm/xai/index.js +64 -0
- package/lib/esm/xai/index.js.map +1 -0
- package/lib/types/adobe/firefly.d.ts +30 -0
- package/lib/types/adobe/firefly.d.ts.map +1 -0
- package/lib/types/bedrock/converse.d.ts +8 -0
- package/lib/types/bedrock/converse.d.ts.map +1 -0
- package/lib/types/bedrock/index.d.ts +26 -11
- package/lib/types/bedrock/index.d.ts.map +1 -1
- package/lib/types/bedrock/nova-image-payload.d.ts +74 -0
- package/lib/types/bedrock/nova-image-payload.d.ts.map +1 -0
- package/lib/types/bedrock/payloads.d.ts +9 -65
- package/lib/types/bedrock/payloads.d.ts.map +1 -1
- package/lib/types/groq/index.d.ts +3 -3
- package/lib/types/groq/index.d.ts.map +1 -1
- package/lib/types/huggingface_ie.d.ts +5 -7
- package/lib/types/huggingface_ie.d.ts.map +1 -1
- package/lib/types/index.d.ts +1 -0
- package/lib/types/index.d.ts.map +1 -1
- package/lib/types/mistral/index.d.ts +4 -4
- package/lib/types/mistral/index.d.ts.map +1 -1
- package/lib/types/mistral/types.d.ts +1 -0
- package/lib/types/mistral/types.d.ts.map +1 -1
- package/lib/types/openai/index.d.ts +5 -4
- package/lib/types/openai/index.d.ts.map +1 -1
- package/lib/types/replicate.d.ts +4 -9
- package/lib/types/replicate.d.ts.map +1 -1
- package/lib/types/test/index.d.ts +2 -2
- package/lib/types/test/index.d.ts.map +1 -1
- package/lib/types/togetherai/index.d.ts +4 -4
- package/lib/types/togetherai/index.d.ts.map +1 -1
- package/lib/types/vertexai/embeddings/embeddings-image.d.ts +11 -0
- package/lib/types/vertexai/embeddings/embeddings-image.d.ts.map +1 -0
- package/lib/types/vertexai/index.d.ts +16 -7
- package/lib/types/vertexai/index.d.ts.map +1 -1
- package/lib/types/vertexai/models/claude.d.ts +20 -0
- package/lib/types/vertexai/models/claude.d.ts.map +1 -0
- package/lib/types/vertexai/models/gemini.d.ts +4 -4
- package/lib/types/vertexai/models/gemini.d.ts.map +1 -1
- package/lib/types/vertexai/models/imagen.d.ts +75 -0
- package/lib/types/vertexai/models/imagen.d.ts.map +1 -0
- package/lib/types/vertexai/models.d.ts +3 -6
- package/lib/types/vertexai/models.d.ts.map +1 -1
- package/lib/types/watsonx/index.d.ts +3 -3
- package/lib/types/watsonx/index.d.ts.map +1 -1
- package/lib/types/xai/index.d.ts +19 -0
- package/lib/types/xai/index.d.ts.map +1 -0
- package/package.json +17 -15
- package/src/adobe/firefly.ts +207 -0
- package/src/bedrock/converse.ts +194 -0
- package/src/bedrock/index.ts +349 -237
- package/src/bedrock/nova-image-payload.ts +309 -0
- package/src/bedrock/payloads.ts +12 -66
- package/src/groq/index.ts +35 -12
- package/src/huggingface_ie.ts +34 -13
- package/src/index.ts +1 -0
- package/src/mistral/index.ts +34 -12
- package/src/mistral/types.ts +2 -1
- package/src/openai/index.ts +167 -33
- package/src/replicate.ts +21 -20
- package/src/test/TestValidationErrorCompletionStream.ts +2 -2
- package/src/test/index.ts +3 -2
- package/src/togetherai/index.ts +44 -12
- package/src/vertexai/embeddings/embeddings-image.ts +50 -0
- package/src/vertexai/embeddings/embeddings-text.ts +1 -1
- package/src/vertexai/index.ts +106 -32
- package/src/vertexai/models/claude.ts +281 -0
- package/src/vertexai/models/gemini.ts +159 -26
- package/src/vertexai/models/imagen.ts +401 -0
- package/src/vertexai/models.ts +16 -78
- package/src/watsonx/index.ts +42 -10
- package/src/xai/index.ts +110 -0
|
@@ -0,0 +1,207 @@
|
|
|
1
|
+
import {
|
|
2
|
+
AbstractDriver, AIModel,
|
|
3
|
+
Completion,
|
|
4
|
+
CompletionChunk, DriverOptions, EmbeddingsOptions,
|
|
5
|
+
EmbeddingsResult,
|
|
6
|
+
ExecutionOptions,
|
|
7
|
+
ImageGeneration,
|
|
8
|
+
ModelSearchPayload,
|
|
9
|
+
PromptSegment
|
|
10
|
+
} from "@llumiverse/core";
|
|
11
|
+
|
|
12
|
+
interface FireflyImageSource {
|
|
13
|
+
url?: string;
|
|
14
|
+
uploadId?: string;
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
interface FireflyImageReference {
|
|
18
|
+
source: FireflyImageSource;
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
interface FireflyStyle {
|
|
22
|
+
presets?: string[];
|
|
23
|
+
strength?: number;
|
|
24
|
+
imageReference?: FireflyImageReference;
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
interface FireflyStructure {
|
|
28
|
+
strength: number;
|
|
29
|
+
imageReference: FireflyImageReference;
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
interface FireflySize {
|
|
33
|
+
width: number;
|
|
34
|
+
height: number;
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
interface FireflyGenerateRequest {
|
|
38
|
+
numVariations?: number;
|
|
39
|
+
seeds?: number[];
|
|
40
|
+
size?: FireflySize;
|
|
41
|
+
prompt: string;
|
|
42
|
+
negativePrompt?: string;
|
|
43
|
+
contentClass?: 'photo' | 'art' | 'graphic';
|
|
44
|
+
visualIntensity?: number;
|
|
45
|
+
style?: FireflyStyle;
|
|
46
|
+
promptBiasingLocaleCode?: string;
|
|
47
|
+
tileable?: boolean;
|
|
48
|
+
structure?: FireflyStructure;
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
interface FireflyOutput {
|
|
52
|
+
seed: number;
|
|
53
|
+
image: {
|
|
54
|
+
url: string;
|
|
55
|
+
};
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
interface FireflyGenerateResponse {
|
|
59
|
+
size: FireflySize;
|
|
60
|
+
outputs: FireflyOutput[];
|
|
61
|
+
promptHasDeniedWords?: boolean;
|
|
62
|
+
promptHasBlockedArtists?: boolean;
|
|
63
|
+
contentClass?: string;
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
export interface FireflyDriverOptions extends DriverOptions {
|
|
67
|
+
/**
|
|
68
|
+
* Adobe Firefly API key
|
|
69
|
+
*/
|
|
70
|
+
apiKey: string;
|
|
71
|
+
|
|
72
|
+
/**
|
|
73
|
+
* Optional API endpoint override
|
|
74
|
+
*/
|
|
75
|
+
endpoint?: string;
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
export class FireflyDriver extends AbstractDriver<FireflyDriverOptions> {
|
|
79
|
+
static PROVIDER = "firefly";
|
|
80
|
+
provider = FireflyDriver.PROVIDER;
|
|
81
|
+
|
|
82
|
+
private readonly endpoint: string;
|
|
83
|
+
|
|
84
|
+
constructor(options: FireflyDriverOptions) {
|
|
85
|
+
super(options);
|
|
86
|
+
|
|
87
|
+
if (!options.apiKey) {
|
|
88
|
+
throw new Error("No API key provided for Firefly driver");
|
|
89
|
+
}
|
|
90
|
+
|
|
91
|
+
this.endpoint = options.endpoint || "https://firefly-api.adobe.io/v3";
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
async requestTextCompletion(_prompt: string, _options: ExecutionOptions): Promise<Completion> {
|
|
95
|
+
throw new Error("Text completion not supported by Firefly");
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
async requestTextCompletionStream(_prompt: string, _options: ExecutionOptions): Promise<AsyncIterable<CompletionChunk>> {
|
|
99
|
+
throw new Error("Text completion streaming not supported by Firefly");
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
async requestImageGeneration(segments: PromptSegment[], options: ExecutionOptions): Promise<Completion<ImageGeneration>> {
|
|
103
|
+
this.logger.debug(`[${this.provider}] Generating image with model ${options.model}`);
|
|
104
|
+
const prompt = segments.map(s => s.content).join("\n\n");
|
|
105
|
+
|
|
106
|
+
|
|
107
|
+
try {
|
|
108
|
+
const payload: FireflyGenerateRequest = {
|
|
109
|
+
prompt: prompt as string,
|
|
110
|
+
};
|
|
111
|
+
|
|
112
|
+
const response = await fetch(`${this.endpoint}/images/generate`, {
|
|
113
|
+
method: 'POST',
|
|
114
|
+
headers: {
|
|
115
|
+
'Content-Type': 'application/json',
|
|
116
|
+
'x-api-key': this.options.apiKey,
|
|
117
|
+
},
|
|
118
|
+
body: JSON.stringify(payload)
|
|
119
|
+
});
|
|
120
|
+
|
|
121
|
+
if (!response.ok) {
|
|
122
|
+
const error = await response.json();
|
|
123
|
+
throw new Error(`Firefly API error: ${error.message || response.statusText}`);
|
|
124
|
+
}
|
|
125
|
+
|
|
126
|
+
const result: FireflyGenerateResponse = await response.json();
|
|
127
|
+
|
|
128
|
+
if (result.promptHasDeniedWords || result.promptHasBlockedArtists) {
|
|
129
|
+
return {
|
|
130
|
+
result: {},
|
|
131
|
+
error: {
|
|
132
|
+
message: "Prompt contains denied words or blocked artists",
|
|
133
|
+
code: "content_policy_violation"
|
|
134
|
+
}
|
|
135
|
+
};
|
|
136
|
+
}
|
|
137
|
+
|
|
138
|
+
return {
|
|
139
|
+
result: {
|
|
140
|
+
images: result.outputs.map(output => output.image.url)
|
|
141
|
+
}
|
|
142
|
+
};
|
|
143
|
+
|
|
144
|
+
} catch (error: any) {
|
|
145
|
+
this.logger.error("[Firefly] Image generation failed", { error });
|
|
146
|
+
return {
|
|
147
|
+
result: {},
|
|
148
|
+
error: {
|
|
149
|
+
message: error.message,
|
|
150
|
+
code: error.code || 'GENERATION_FAILED'
|
|
151
|
+
}
|
|
152
|
+
};
|
|
153
|
+
}
|
|
154
|
+
}
|
|
155
|
+
|
|
156
|
+
mapSize(size?: string): FireflySize {
|
|
157
|
+
// Default to 1024x1024 if no size specified
|
|
158
|
+
if (!size) return { width: 1024, height: 1024 };
|
|
159
|
+
|
|
160
|
+
const [width, height] = size.split('x').map(Number);
|
|
161
|
+
return { width, height };
|
|
162
|
+
}
|
|
163
|
+
|
|
164
|
+
async listModels(_params?: ModelSearchPayload): Promise<AIModel[]> {
|
|
165
|
+
return [
|
|
166
|
+
{
|
|
167
|
+
id: "firefly-v3-text-to-image",
|
|
168
|
+
name: "Firefly v3 Text to Image",
|
|
169
|
+
provider: this.provider,
|
|
170
|
+
description: "Adobe Firefly v3 text to image generation model",
|
|
171
|
+
tags: ["image-generation"]
|
|
172
|
+
},
|
|
173
|
+
{
|
|
174
|
+
id: "firefly-v3-image-to-image",
|
|
175
|
+
name: "Firefly v3 Image to Image",
|
|
176
|
+
provider: this.provider,
|
|
177
|
+
description: "Adobe Firefly v3 image to image generation model",
|
|
178
|
+
tags: ["image-generation"]
|
|
179
|
+
},
|
|
180
|
+
{
|
|
181
|
+
id: "firefly-v3-inpainting",
|
|
182
|
+
name: "Firefly v3 Inpainting",
|
|
183
|
+
provider: this.provider,
|
|
184
|
+
description: "Adobe Firefly v3 inpainting model",
|
|
185
|
+
tags: ["image-generation"]
|
|
186
|
+
}
|
|
187
|
+
];
|
|
188
|
+
}
|
|
189
|
+
|
|
190
|
+
async validateConnection(): Promise<boolean> {
|
|
191
|
+
try {
|
|
192
|
+
const response = await fetch(`${this.endpoint}/auth/validate`, {
|
|
193
|
+
headers: {
|
|
194
|
+
'x-api-key': this.options.apiKey
|
|
195
|
+
}
|
|
196
|
+
});
|
|
197
|
+
return response.ok;
|
|
198
|
+
} catch (error) {
|
|
199
|
+
this.logger.error("[Firefly] Connection validation failed", { error });
|
|
200
|
+
return false;
|
|
201
|
+
}
|
|
202
|
+
}
|
|
203
|
+
|
|
204
|
+
async generateEmbeddings(_options: EmbeddingsOptions): Promise<EmbeddingsResult> {
|
|
205
|
+
throw new Error("Embeddings not supported by Firefly");
|
|
206
|
+
}
|
|
207
|
+
}
|
|
@@ -0,0 +1,194 @@
|
|
|
1
|
+
import { JSONSchema4 } from "json-schema";
|
|
2
|
+
import { PromptSegment, PromptRole } from "@llumiverse/core";
|
|
3
|
+
import {
|
|
4
|
+
ConversationRole,
|
|
5
|
+
ConverseRequest,
|
|
6
|
+
Message,
|
|
7
|
+
SystemContentBlock,
|
|
8
|
+
ContentBlock,
|
|
9
|
+
} from "@aws-sdk/client-bedrock-runtime";
|
|
10
|
+
|
|
11
|
+
function getJSONSafetyNotice(schema: JSONSchema4) {
|
|
12
|
+
return "The answer must be a JSON object using the following JSON Schema:\n" + JSON.stringify(schema, undefined, 2);
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
function roleConversion(role: PromptRole): ConversationRole {
|
|
16
|
+
return role === PromptRole.assistant ? ConversationRole.ASSISTANT : ConversationRole.USER;
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
function mimeToImageType(mime: string): "png" | "jpeg" | "gif" | "webp" {
|
|
20
|
+
if (mime.startsWith("image/")) {
|
|
21
|
+
return mime.split("/")[1] as "png" | "jpeg" | "gif" | "webp";
|
|
22
|
+
}
|
|
23
|
+
return "png";
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
function mimeToDocType(mime: string): "pdf" | "csv" | "doc" | "docx" | "xls" | "xlsx" | "html" | "txt" | "md" {
|
|
27
|
+
if (mime.startsWith("application/") || mime.startsWith("text/")) {
|
|
28
|
+
return mime.split("/")[1] as "pdf" | "csv" | "doc" | "docx" | "xls" | "xlsx" | "html" | "txt" | "md";
|
|
29
|
+
}
|
|
30
|
+
return "txt";
|
|
31
|
+
}
|
|
32
|
+
function mimeToVideoType(mime: string): "mov" | "mkv" | "mp4" | "webm" | "flv" | "mpeg" | "mpg" | "wmv" | "three_gp" {
|
|
33
|
+
if (mime.startsWith("video/")) {
|
|
34
|
+
return mime.split("/")[1] as "mov" | "mkv" | "mp4" | "webm" | "flv" | "mpeg" | "mpg" | "wmv" | "three_gp";
|
|
35
|
+
}
|
|
36
|
+
return "mp4";
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
async function readStreamAsString(stream: ReadableStream): Promise<string> {
|
|
40
|
+
const out: Buffer[] = [];
|
|
41
|
+
for await (const chunk of stream as any) {
|
|
42
|
+
out.push(Buffer.from(chunk));
|
|
43
|
+
}
|
|
44
|
+
return Buffer.concat(out).toString();
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
async function readStreamAsUint8Array(stream: ReadableStream): Promise<Uint8Array> {
|
|
48
|
+
const out: Buffer[] = [];
|
|
49
|
+
for await (const chunk of stream as any) {
|
|
50
|
+
out.push(Buffer.from(chunk));
|
|
51
|
+
}
|
|
52
|
+
return Buffer.concat(out);
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
export function converseConcatMessages(messages: Message[] | undefined): Message[] {
|
|
56
|
+
if (!messages) return [];
|
|
57
|
+
//Concatenate messages of the same role. Required to have alternative user and assistant roles
|
|
58
|
+
for (let i = 0; i < messages.length - 1; i++) {
|
|
59
|
+
if (messages[i].role === messages[i + 1].role) {
|
|
60
|
+
messages[i].content = messages[i].content?.concat(...(messages[i + 1].content || []));
|
|
61
|
+
messages.splice(i + 1, 1);
|
|
62
|
+
i--;
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
return messages;
|
|
66
|
+
}
|
|
67
|
+
|
|
68
|
+
export function converseSystemToMessages(system: SystemContentBlock[]): Message {
|
|
69
|
+
return {
|
|
70
|
+
content: [{text: system.map(system => system.text).join('\n').trim()}],
|
|
71
|
+
role: ConversationRole.USER
|
|
72
|
+
};
|
|
73
|
+
}
|
|
74
|
+
|
|
75
|
+
export function converseRemoveJSONprefill(messages: Message[] | undefined): Message[] {
|
|
76
|
+
//Remove the "```json" stop message
|
|
77
|
+
if (messages && messages.length > 0) {
|
|
78
|
+
if (messages[messages.length - 1].content?.[0].text === "```json") {
|
|
79
|
+
messages.pop();
|
|
80
|
+
}
|
|
81
|
+
}
|
|
82
|
+
return messages ?? [];
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
export async function fortmatConversePrompt(segments: PromptSegment[], schema?: JSONSchema4): Promise<ConverseRequest> {
|
|
86
|
+
//Non-const for concat
|
|
87
|
+
let system: SystemContentBlock[] = [];
|
|
88
|
+
const safety: SystemContentBlock[] = [];
|
|
89
|
+
let messages: Message[] = [];
|
|
90
|
+
|
|
91
|
+
for (const segment of segments) {
|
|
92
|
+
const parts: Message[] = [];
|
|
93
|
+
|
|
94
|
+
//File segments
|
|
95
|
+
if (segment.files)
|
|
96
|
+
for (const f of segment.files) {
|
|
97
|
+
const source = await f.getStream();
|
|
98
|
+
let content: ContentBlock[];
|
|
99
|
+
|
|
100
|
+
//Image file - "png" | "jpeg" | "gif" | "webp"
|
|
101
|
+
if (f.mime_type && f.mime_type.startsWith("image")) {
|
|
102
|
+
content = [
|
|
103
|
+
{
|
|
104
|
+
image: {
|
|
105
|
+
format: mimeToImageType(f.mime_type),
|
|
106
|
+
source: { bytes: await readStreamAsUint8Array(source) },
|
|
107
|
+
},
|
|
108
|
+
},
|
|
109
|
+
];
|
|
110
|
+
|
|
111
|
+
//Document file - "pdf | csv | doc | docx | xls | xlsx | html | txt | md"
|
|
112
|
+
} else if (f.mime_type && (f.mime_type.startsWith("text") || f.mime_type?.startsWith("application"))) {
|
|
113
|
+
content = [
|
|
114
|
+
{ text: f.name },
|
|
115
|
+
{
|
|
116
|
+
document: {
|
|
117
|
+
format: mimeToDocType(f.mime_type),
|
|
118
|
+
name: f.name,
|
|
119
|
+
source: { bytes: await readStreamAsUint8Array(source) },
|
|
120
|
+
},
|
|
121
|
+
},
|
|
122
|
+
];
|
|
123
|
+
|
|
124
|
+
//Video file - "mov | mkv | mp4 | webm | flv | mpeg | mpg | wmv | three_gp"
|
|
125
|
+
} else if (f.mime_type && f.mime_type.startsWith("video")) {
|
|
126
|
+
content = [
|
|
127
|
+
{
|
|
128
|
+
video: {
|
|
129
|
+
format: mimeToVideoType(f.mime_type),
|
|
130
|
+
source: { bytes: await readStreamAsUint8Array(source) },
|
|
131
|
+
},
|
|
132
|
+
},
|
|
133
|
+
];
|
|
134
|
+
|
|
135
|
+
//Fallback, send string
|
|
136
|
+
} else {
|
|
137
|
+
content = [{ text: await readStreamAsString(source) }];
|
|
138
|
+
}
|
|
139
|
+
|
|
140
|
+
parts.push({
|
|
141
|
+
content: content,
|
|
142
|
+
role: roleConversion(segment.role),
|
|
143
|
+
});
|
|
144
|
+
}
|
|
145
|
+
|
|
146
|
+
//Text segments
|
|
147
|
+
if (segment.content) {
|
|
148
|
+
parts.push({
|
|
149
|
+
content: [{ text: segment.content }],
|
|
150
|
+
role: roleConversion(segment.role),
|
|
151
|
+
});
|
|
152
|
+
}
|
|
153
|
+
|
|
154
|
+
if (segment.role === PromptRole.system) {
|
|
155
|
+
system.push({ text: segment.content });
|
|
156
|
+
} else if (segment.role === PromptRole.safety) {
|
|
157
|
+
safety.push({ text: segment.content });
|
|
158
|
+
} else if (segment.role !== PromptRole.negative && segment.role !== PromptRole.mask) {
|
|
159
|
+
//User or Assistant
|
|
160
|
+
messages = messages.concat(parts);
|
|
161
|
+
}
|
|
162
|
+
}
|
|
163
|
+
|
|
164
|
+
//Conversations must start with a user message
|
|
165
|
+
//Use the system messages if none are provided
|
|
166
|
+
if (messages.length === 0) {
|
|
167
|
+
const systemMessage = converseSystemToMessages(system);
|
|
168
|
+
if (systemMessage?.content && systemMessage.content[0].text) {
|
|
169
|
+
messages.push(systemMessage);
|
|
170
|
+
} else {
|
|
171
|
+
throw new Error("Prompt must contain at least one message");
|
|
172
|
+
}
|
|
173
|
+
system = [];
|
|
174
|
+
}
|
|
175
|
+
|
|
176
|
+
if (schema) {
|
|
177
|
+
safety.push({ text: "IMPORTANT: " + getJSONSafetyNotice(schema) });
|
|
178
|
+
system = system.concat(safety);
|
|
179
|
+
|
|
180
|
+
//prefill the json
|
|
181
|
+
messages.push({
|
|
182
|
+
content: [{ text: "```json" }],
|
|
183
|
+
role: ConversationRole.ASSISTANT,
|
|
184
|
+
});
|
|
185
|
+
}
|
|
186
|
+
|
|
187
|
+
messages = converseConcatMessages(messages);
|
|
188
|
+
|
|
189
|
+
return {
|
|
190
|
+
modelId: undefined, //required property, but allowed to be undefined
|
|
191
|
+
messages: messages,
|
|
192
|
+
system: system,
|
|
193
|
+
};
|
|
194
|
+
}
|