@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.
Files changed (156) hide show
  1. package/README.md +3 -3
  2. package/lib/cjs/adobe/firefly.js +119 -0
  3. package/lib/cjs/adobe/firefly.js.map +1 -0
  4. package/lib/cjs/bedrock/converse.js +177 -0
  5. package/lib/cjs/bedrock/converse.js.map +1 -0
  6. package/lib/cjs/bedrock/index.js +329 -228
  7. package/lib/cjs/bedrock/index.js.map +1 -1
  8. package/lib/cjs/bedrock/nova-image-payload.js +207 -0
  9. package/lib/cjs/bedrock/nova-image-payload.js.map +1 -0
  10. package/lib/cjs/groq/index.js +34 -9
  11. package/lib/cjs/groq/index.js.map +1 -1
  12. package/lib/cjs/huggingface_ie.js +28 -12
  13. package/lib/cjs/huggingface_ie.js.map +1 -1
  14. package/lib/cjs/index.js +1 -0
  15. package/lib/cjs/index.js.map +1 -1
  16. package/lib/cjs/mistral/index.js +31 -12
  17. package/lib/cjs/mistral/index.js.map +1 -1
  18. package/lib/cjs/mistral/types.js.map +1 -1
  19. package/lib/cjs/openai/index.js +149 -27
  20. package/lib/cjs/openai/index.js.map +1 -1
  21. package/lib/cjs/replicate.js +16 -18
  22. package/lib/cjs/replicate.js.map +1 -1
  23. package/lib/cjs/test/TestValidationErrorCompletionStream.js.map +1 -1
  24. package/lib/cjs/test/index.js.map +1 -1
  25. package/lib/cjs/togetherai/index.js +40 -10
  26. package/lib/cjs/togetherai/index.js.map +1 -1
  27. package/lib/cjs/vertexai/embeddings/embeddings-image.js +26 -0
  28. package/lib/cjs/vertexai/embeddings/embeddings-image.js.map +1 -0
  29. package/lib/cjs/vertexai/embeddings/embeddings-text.js +1 -1
  30. package/lib/cjs/vertexai/embeddings/embeddings-text.js.map +1 -1
  31. package/lib/cjs/vertexai/index.js +88 -24
  32. package/lib/cjs/vertexai/index.js.map +1 -1
  33. package/lib/cjs/vertexai/models/claude.js +252 -0
  34. package/lib/cjs/vertexai/models/claude.js.map +1 -0
  35. package/lib/cjs/vertexai/models/gemini.js +147 -22
  36. package/lib/cjs/vertexai/models/gemini.js.map +1 -1
  37. package/lib/cjs/vertexai/models/imagen.js +317 -0
  38. package/lib/cjs/vertexai/models/imagen.js.map +1 -0
  39. package/lib/cjs/vertexai/models.js +12 -64
  40. package/lib/cjs/vertexai/models.js.map +1 -1
  41. package/lib/cjs/watsonx/index.js +39 -8
  42. package/lib/cjs/watsonx/index.js.map +1 -1
  43. package/lib/cjs/xai/index.js +71 -0
  44. package/lib/cjs/xai/index.js.map +1 -0
  45. package/lib/esm/adobe/firefly.js +115 -0
  46. package/lib/esm/adobe/firefly.js.map +1 -0
  47. package/lib/esm/bedrock/converse.js +171 -0
  48. package/lib/esm/bedrock/converse.js.map +1 -0
  49. package/lib/esm/bedrock/index.js +331 -230
  50. package/lib/esm/bedrock/index.js.map +1 -1
  51. package/lib/esm/bedrock/nova-image-payload.js +203 -0
  52. package/lib/esm/bedrock/nova-image-payload.js.map +1 -0
  53. package/lib/esm/groq/index.js +34 -9
  54. package/lib/esm/groq/index.js.map +1 -1
  55. package/lib/esm/huggingface_ie.js +29 -13
  56. package/lib/esm/huggingface_ie.js.map +1 -1
  57. package/lib/esm/index.js +1 -0
  58. package/lib/esm/index.js.map +1 -1
  59. package/lib/esm/mistral/index.js +31 -12
  60. package/lib/esm/mistral/index.js.map +1 -1
  61. package/lib/esm/mistral/types.js.map +1 -1
  62. package/lib/esm/openai/index.js +150 -28
  63. package/lib/esm/openai/index.js.map +1 -1
  64. package/lib/esm/replicate.js +17 -19
  65. package/lib/esm/replicate.js.map +1 -1
  66. package/lib/esm/test/TestValidationErrorCompletionStream.js.map +1 -1
  67. package/lib/esm/test/index.js.map +1 -1
  68. package/lib/esm/togetherai/index.js +40 -10
  69. package/lib/esm/togetherai/index.js.map +1 -1
  70. package/lib/esm/vertexai/embeddings/embeddings-image.js +23 -0
  71. package/lib/esm/vertexai/embeddings/embeddings-image.js.map +1 -0
  72. package/lib/esm/vertexai/embeddings/embeddings-text.js +1 -1
  73. package/lib/esm/vertexai/embeddings/embeddings-text.js.map +1 -1
  74. package/lib/esm/vertexai/index.js +89 -26
  75. package/lib/esm/vertexai/index.js.map +1 -1
  76. package/lib/esm/vertexai/models/claude.js +247 -0
  77. package/lib/esm/vertexai/models/claude.js.map +1 -0
  78. package/lib/esm/vertexai/models/gemini.js +148 -23
  79. package/lib/esm/vertexai/models/gemini.js.map +1 -1
  80. package/lib/esm/vertexai/models/imagen.js +310 -0
  81. package/lib/esm/vertexai/models/imagen.js.map +1 -0
  82. package/lib/esm/vertexai/models.js +12 -61
  83. package/lib/esm/vertexai/models.js.map +1 -1
  84. package/lib/esm/watsonx/index.js +39 -8
  85. package/lib/esm/watsonx/index.js.map +1 -1
  86. package/lib/esm/xai/index.js +64 -0
  87. package/lib/esm/xai/index.js.map +1 -0
  88. package/lib/types/adobe/firefly.d.ts +30 -0
  89. package/lib/types/adobe/firefly.d.ts.map +1 -0
  90. package/lib/types/bedrock/converse.d.ts +8 -0
  91. package/lib/types/bedrock/converse.d.ts.map +1 -0
  92. package/lib/types/bedrock/index.d.ts +26 -11
  93. package/lib/types/bedrock/index.d.ts.map +1 -1
  94. package/lib/types/bedrock/nova-image-payload.d.ts +74 -0
  95. package/lib/types/bedrock/nova-image-payload.d.ts.map +1 -0
  96. package/lib/types/bedrock/payloads.d.ts +9 -65
  97. package/lib/types/bedrock/payloads.d.ts.map +1 -1
  98. package/lib/types/groq/index.d.ts +3 -3
  99. package/lib/types/groq/index.d.ts.map +1 -1
  100. package/lib/types/huggingface_ie.d.ts +5 -7
  101. package/lib/types/huggingface_ie.d.ts.map +1 -1
  102. package/lib/types/index.d.ts +1 -0
  103. package/lib/types/index.d.ts.map +1 -1
  104. package/lib/types/mistral/index.d.ts +4 -4
  105. package/lib/types/mistral/index.d.ts.map +1 -1
  106. package/lib/types/mistral/types.d.ts +1 -0
  107. package/lib/types/mistral/types.d.ts.map +1 -1
  108. package/lib/types/openai/index.d.ts +5 -4
  109. package/lib/types/openai/index.d.ts.map +1 -1
  110. package/lib/types/replicate.d.ts +4 -9
  111. package/lib/types/replicate.d.ts.map +1 -1
  112. package/lib/types/test/index.d.ts +2 -2
  113. package/lib/types/test/index.d.ts.map +1 -1
  114. package/lib/types/togetherai/index.d.ts +4 -4
  115. package/lib/types/togetherai/index.d.ts.map +1 -1
  116. package/lib/types/vertexai/embeddings/embeddings-image.d.ts +11 -0
  117. package/lib/types/vertexai/embeddings/embeddings-image.d.ts.map +1 -0
  118. package/lib/types/vertexai/index.d.ts +16 -7
  119. package/lib/types/vertexai/index.d.ts.map +1 -1
  120. package/lib/types/vertexai/models/claude.d.ts +20 -0
  121. package/lib/types/vertexai/models/claude.d.ts.map +1 -0
  122. package/lib/types/vertexai/models/gemini.d.ts +4 -4
  123. package/lib/types/vertexai/models/gemini.d.ts.map +1 -1
  124. package/lib/types/vertexai/models/imagen.d.ts +75 -0
  125. package/lib/types/vertexai/models/imagen.d.ts.map +1 -0
  126. package/lib/types/vertexai/models.d.ts +3 -6
  127. package/lib/types/vertexai/models.d.ts.map +1 -1
  128. package/lib/types/watsonx/index.d.ts +3 -3
  129. package/lib/types/watsonx/index.d.ts.map +1 -1
  130. package/lib/types/xai/index.d.ts +19 -0
  131. package/lib/types/xai/index.d.ts.map +1 -0
  132. package/package.json +17 -15
  133. package/src/adobe/firefly.ts +207 -0
  134. package/src/bedrock/converse.ts +194 -0
  135. package/src/bedrock/index.ts +349 -237
  136. package/src/bedrock/nova-image-payload.ts +309 -0
  137. package/src/bedrock/payloads.ts +12 -66
  138. package/src/groq/index.ts +35 -12
  139. package/src/huggingface_ie.ts +34 -13
  140. package/src/index.ts +1 -0
  141. package/src/mistral/index.ts +34 -12
  142. package/src/mistral/types.ts +2 -1
  143. package/src/openai/index.ts +167 -33
  144. package/src/replicate.ts +21 -20
  145. package/src/test/TestValidationErrorCompletionStream.ts +2 -2
  146. package/src/test/index.ts +3 -2
  147. package/src/togetherai/index.ts +44 -12
  148. package/src/vertexai/embeddings/embeddings-image.ts +50 -0
  149. package/src/vertexai/embeddings/embeddings-text.ts +1 -1
  150. package/src/vertexai/index.ts +106 -32
  151. package/src/vertexai/models/claude.ts +281 -0
  152. package/src/vertexai/models/gemini.ts +159 -26
  153. package/src/vertexai/models/imagen.ts +401 -0
  154. package/src/vertexai/models.ts +16 -78
  155. package/src/watsonx/index.ts +42 -10
  156. 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
+ }