@outputai/llm 0.6.1-next.f67f4b9.0 → 0.6.1-next.f8d698e.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/package.json +20 -12
- package/src/ai_model.js +25 -146
- package/src/ai_model.spec.js +185 -762
- package/src/ai_provider.js +86 -0
- package/src/ai_provider.spec.js +147 -0
- package/src/index.d.ts +51 -16
- package/src/index.js +1 -4
- package/src/prompt/validations.js +4 -1
- package/src/prompt/validations.spec.js +65 -0
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@outputai/llm",
|
|
3
|
-
"version": "0.6.1-next.
|
|
3
|
+
"version": "0.6.1-next.f8d698e.0",
|
|
4
4
|
"description": "Framework abstraction to interact with LLM models",
|
|
5
5
|
"type": "module",
|
|
6
6
|
"main": "src/index.js",
|
|
@@ -9,21 +9,29 @@
|
|
|
9
9
|
"./src"
|
|
10
10
|
],
|
|
11
11
|
"dependencies": {
|
|
12
|
-
"@ai-sdk/amazon-bedrock": "4.0.96",
|
|
13
|
-
"@ai-sdk/anthropic": "3.0.71",
|
|
14
|
-
"@ai-sdk/azure": "3.0.54",
|
|
15
|
-
"@ai-sdk/google-vertex": "4.0.112",
|
|
16
|
-
"@ai-sdk/openai": "3.0.53",
|
|
17
|
-
"@ai-sdk/perplexity": "3.0.29",
|
|
18
|
-
"@exalabs/ai-sdk": "2.0.1",
|
|
19
|
-
"@perplexity-ai/ai-sdk": "0.1.3",
|
|
20
|
-
"@tavily/ai-sdk": "0.4.1",
|
|
21
|
-
"ai": "6.0.168",
|
|
22
12
|
"entities": "8.0.0",
|
|
23
13
|
"gray-matter": "4.0.3",
|
|
24
14
|
"liquidjs": "10.25.7",
|
|
25
15
|
"undici": "8.1.0",
|
|
26
|
-
"@outputai/core": "0.6.1-next.
|
|
16
|
+
"@outputai/core": "0.6.1-next.f8d698e.0"
|
|
17
|
+
},
|
|
18
|
+
"devDependencies": {
|
|
19
|
+
"ai": "6.0.168",
|
|
20
|
+
"@ai-sdk/amazon-bedrock": "4.0.111",
|
|
21
|
+
"@ai-sdk/anthropic": "3.0.81",
|
|
22
|
+
"@ai-sdk/azure": "3.0.68",
|
|
23
|
+
"@ai-sdk/google-vertex": "4.0.140",
|
|
24
|
+
"@ai-sdk/openai": "3.0.67",
|
|
25
|
+
"@ai-sdk/perplexity": "3.0.33"
|
|
26
|
+
},
|
|
27
|
+
"peerDependencies": {
|
|
28
|
+
"ai": ">=6 <7",
|
|
29
|
+
"@ai-sdk/amazon-bedrock": ">=4 <5",
|
|
30
|
+
"@ai-sdk/anthropic": ">=3 <4",
|
|
31
|
+
"@ai-sdk/azure": ">=3 <4",
|
|
32
|
+
"@ai-sdk/google-vertex": ">=4 <5",
|
|
33
|
+
"@ai-sdk/openai": ">=3 <4",
|
|
34
|
+
"@ai-sdk/perplexity": ">=3 <4"
|
|
27
35
|
},
|
|
28
36
|
"license": "Apache-2.0",
|
|
29
37
|
"publishConfig": {
|
package/src/ai_model.js
CHANGED
|
@@ -1,89 +1,5 @@
|
|
|
1
|
-
import {
|
|
2
|
-
import {
|
|
3
|
-
import { createAzure } from '@ai-sdk/azure';
|
|
4
|
-
import { createVertex } from '@ai-sdk/google-vertex';
|
|
5
|
-
import { createOpenAI } from '@ai-sdk/openai';
|
|
6
|
-
import { createPerplexity } from '@ai-sdk/perplexity';
|
|
7
|
-
import { ValidationError, z } from '@outputai/core';
|
|
8
|
-
import { Agent, fetch } from 'undici';
|
|
9
|
-
|
|
10
|
-
const dispatcher = new Agent( {
|
|
11
|
-
headersTimeout: 15 * 60 * 1000, // 15 min
|
|
12
|
-
bodyTimeout: 15 * 60 * 1000
|
|
13
|
-
} );
|
|
14
|
-
|
|
15
|
-
const customFetch = ( input, init ) => fetch( input, { dispatcher, ...init } );
|
|
16
|
-
const initProvider = factory => factory( { fetch: customFetch } );
|
|
17
|
-
|
|
18
|
-
export const builtInProviders = {
|
|
19
|
-
azure: initProvider( createAzure ),
|
|
20
|
-
anthropic: initProvider( createAnthropic ),
|
|
21
|
-
openai: initProvider( createOpenAI ),
|
|
22
|
-
vertex: initProvider( createVertex ),
|
|
23
|
-
bedrock: initProvider( createAmazonBedrock ),
|
|
24
|
-
perplexity: initProvider( createPerplexity )
|
|
25
|
-
};
|
|
26
|
-
|
|
27
|
-
export const providers = { ...builtInProviders };
|
|
28
|
-
|
|
29
|
-
const registerProviderSchema = z.object( {
|
|
30
|
-
name: z.string().min( 1, 'Provider name must be a non-empty string' ),
|
|
31
|
-
providerFn: z.function()
|
|
32
|
-
} );
|
|
33
|
-
|
|
34
|
-
const modelPromptSchema = z.object( {
|
|
35
|
-
config: z.object( {
|
|
36
|
-
provider: z.string().min( 1 ),
|
|
37
|
-
model: z.string().min( 1 )
|
|
38
|
-
} ).loose()
|
|
39
|
-
} ).loose();
|
|
40
|
-
|
|
41
|
-
const toolsPromptSchema = z.object( {
|
|
42
|
-
config: z.object( {
|
|
43
|
-
provider: z.string().min( 1 ),
|
|
44
|
-
tools: z.record( z.string(), z.record( z.string(), z.unknown() ) ).optional()
|
|
45
|
-
} ).loose()
|
|
46
|
-
} ).loose();
|
|
47
|
-
|
|
48
|
-
const toolConfigSchema = z.record( z.string(), z.unknown() );
|
|
49
|
-
|
|
50
|
-
const parseModelPrompt = prompt => {
|
|
51
|
-
const result = modelPromptSchema.safeParse( prompt );
|
|
52
|
-
if ( !result.success ) {
|
|
53
|
-
throw new ValidationError( `Invalid model prompt config: ${z.prettifyError( result.error )}` );
|
|
54
|
-
}
|
|
55
|
-
return result.data.config;
|
|
56
|
-
};
|
|
57
|
-
|
|
58
|
-
const parseToolsPrompt = prompt => {
|
|
59
|
-
const result = toolsPromptSchema.safeParse( prompt );
|
|
60
|
-
if ( !result.success ) {
|
|
61
|
-
throw new ValidationError( `Invalid tools prompt config: ${z.prettifyError( result.error )}` );
|
|
62
|
-
}
|
|
63
|
-
return result.data.config;
|
|
64
|
-
};
|
|
65
|
-
|
|
66
|
-
/**
|
|
67
|
-
* Register or override an AI SDK provider factory by name.
|
|
68
|
-
*
|
|
69
|
-
* @param {string} name - Provider name used in prompt frontmatter
|
|
70
|
-
* @param {Function} providerFn - Factory function that receives a model id
|
|
71
|
-
* @returns {void}
|
|
72
|
-
*/
|
|
73
|
-
export function registerProvider( name, providerFn ) {
|
|
74
|
-
const result = registerProviderSchema.safeParse( { name, providerFn } );
|
|
75
|
-
if ( !result.success ) {
|
|
76
|
-
throw new ValidationError( `Invalid provider registration: ${z.prettifyError( result.error )}` );
|
|
77
|
-
}
|
|
78
|
-
providers[name] = providerFn;
|
|
79
|
-
}
|
|
80
|
-
|
|
81
|
-
/**
|
|
82
|
-
* List all currently registered provider names.
|
|
83
|
-
*
|
|
84
|
-
* @returns {string[]} Provider names
|
|
85
|
-
*/
|
|
86
|
-
export const getRegisteredProviders = () => Object.keys( providers );
|
|
1
|
+
import { ValidationError } from '@outputai/core';
|
|
2
|
+
import { getProvider } from './ai_provider.js';
|
|
87
3
|
|
|
88
4
|
/**
|
|
89
5
|
* Load a text model from a loaded prompt config.
|
|
@@ -91,19 +7,7 @@ export const getRegisteredProviders = () => Object.keys( providers );
|
|
|
91
7
|
* @param {object} prompt - Loaded prompt object with `config.provider` and `config.model`
|
|
92
8
|
* @returns {unknown} AI SDK language model
|
|
93
9
|
*/
|
|
94
|
-
export
|
|
95
|
-
const config = parseModelPrompt( prompt );
|
|
96
|
-
const { provider: providerName, model: modelName } = config;
|
|
97
|
-
|
|
98
|
-
const provider = providers[providerName];
|
|
99
|
-
|
|
100
|
-
if ( !provider ) {
|
|
101
|
-
const availableProviders = Object.keys( providers ).join( ', ' );
|
|
102
|
-
throw new Error( `Invalid provider "${providerName}". Valid providers: ${availableProviders}` );
|
|
103
|
-
}
|
|
104
|
-
|
|
105
|
-
return provider( modelName );
|
|
106
|
-
}
|
|
10
|
+
export const loadTextModel = prompt => getProvider( prompt.config.provider )( prompt.config.model );
|
|
107
11
|
|
|
108
12
|
/**
|
|
109
13
|
* Load an image model from a loaded prompt config.
|
|
@@ -111,24 +15,16 @@ export function loadTextModel( prompt ) {
|
|
|
111
15
|
* @param {object} prompt - Loaded prompt object with `config.provider` and `config.model`
|
|
112
16
|
* @returns {unknown} AI SDK image model
|
|
113
17
|
*/
|
|
114
|
-
export
|
|
115
|
-
const
|
|
116
|
-
const
|
|
117
|
-
|
|
118
|
-
const provider = providers[providerName];
|
|
119
|
-
|
|
120
|
-
if ( !provider ) {
|
|
121
|
-
const availableProviders = Object.keys( providers ).join( ', ' );
|
|
122
|
-
throw new Error( `Invalid provider "${providerName}". Valid providers: ${availableProviders}` );
|
|
123
|
-
}
|
|
18
|
+
export const loadImageModel = prompt => {
|
|
19
|
+
const { provider: providerName, model } = prompt.config;
|
|
20
|
+
const provider = getProvider( prompt.config.provider );
|
|
124
21
|
|
|
125
22
|
const imageModelFactory = provider.image ?? provider.imageModel;
|
|
126
23
|
if ( typeof imageModelFactory !== 'function' ) {
|
|
127
|
-
throw new
|
|
24
|
+
throw new ValidationError( `Provider "${providerName}" does not support image models.` );
|
|
128
25
|
}
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
}
|
|
26
|
+
return imageModelFactory( model );
|
|
27
|
+
};
|
|
132
28
|
|
|
133
29
|
/**
|
|
134
30
|
* Load provider-specific tools configured in a prompt.
|
|
@@ -136,46 +32,29 @@ export function loadImageModel( prompt ) {
|
|
|
136
32
|
* @param {object} prompt - Loaded prompt object with `config.provider` and optional `config.tools`
|
|
137
33
|
* @returns {Record<string, unknown> | null} AI SDK tools, or null when none are configured
|
|
138
34
|
*/
|
|
139
|
-
export
|
|
140
|
-
const
|
|
141
|
-
const { tools: toolsConfig, provider: providerName } = config;
|
|
35
|
+
export const loadTools = prompt => {
|
|
36
|
+
const { tools: promptTools, provider: providerName } = prompt.config;
|
|
142
37
|
|
|
143
|
-
if (
|
|
38
|
+
if ( Object.keys( promptTools ?? {} ).length === 0 ) {
|
|
144
39
|
return null;
|
|
145
40
|
}
|
|
146
41
|
|
|
147
|
-
const provider =
|
|
148
|
-
|
|
149
|
-
if ( !provider ) {
|
|
150
|
-
const availableProviders = Object.keys( providers ).join( ', ' );
|
|
151
|
-
throw new Error( `Invalid provider "${providerName}". Valid providers: ${availableProviders}` );
|
|
152
|
-
}
|
|
42
|
+
const provider = getProvider( providerName );
|
|
153
43
|
|
|
154
|
-
if ( !provider.tools
|
|
155
|
-
throw new
|
|
44
|
+
if ( !provider.tools ) {
|
|
45
|
+
throw new ValidationError( `Provider "${providerName}" does not support provider-specific tools.` );
|
|
156
46
|
}
|
|
157
47
|
|
|
158
|
-
const
|
|
48
|
+
const supportedTools = Object.keys( provider.tools );
|
|
49
|
+
const invalidTools = Object.keys( promptTools ).filter( name => !supportedTools.includes( name ) );
|
|
159
50
|
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
|
|
163
|
-
if ( !toolFactory || typeof toolFactory !== 'function' ) {
|
|
164
|
-
const availableTools = Object.keys( provider.tools )
|
|
165
|
-
.filter( key => typeof provider.tools[key] === 'function' )
|
|
166
|
-
.join( ', ' );
|
|
167
|
-
const toolsMessage = availableTools ? `Available tools: ${availableTools}` : 'No tools are available';
|
|
168
|
-
|
|
169
|
-
throw new Error( `Unknown tool "${toolName}" for provider "${providerName}". ${toolsMessage}` );
|
|
170
|
-
}
|
|
171
|
-
|
|
172
|
-
const result = toolConfigSchema.safeParse( toolConfig );
|
|
173
|
-
if ( !result.success ) {
|
|
174
|
-
throw new ValidationError( `Invalid config for tool "${toolName}": ${z.prettifyError( result.error )}` );
|
|
175
|
-
}
|
|
176
|
-
|
|
177
|
-
tools[toolName] = toolFactory( toolConfig );
|
|
51
|
+
if ( invalidTools.length > 0 ) {
|
|
52
|
+
throw new ValidationError( `Invalid tool(s) ${invalidTools.join( ', ' )} for provider "${providerName}". \
|
|
53
|
+
Available: ${supportedTools.join( ', ' )}.` );
|
|
178
54
|
}
|
|
179
55
|
|
|
180
|
-
return
|
|
181
|
-
|
|
56
|
+
// load all tools and return in an object
|
|
57
|
+
return Object.fromEntries(
|
|
58
|
+
Object.entries( promptTools ).map( ( [ name, args ] ) => [ name, provider.tools[name]( args ) ] )
|
|
59
|
+
);
|
|
60
|
+
};
|