@outputai/llm 0.6.1-next.fc6a93e.0 → 0.7.1-next.2a4105c.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 CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@outputai/llm",
3
- "version": "0.6.1-next.fc6a93e.0",
3
+ "version": "0.7.1-next.2a4105c.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.fc6a93e.0"
16
+ "@outputai/core": "0.7.1-next.2a4105c.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 { createAmazonBedrock } from '@ai-sdk/amazon-bedrock';
2
- import { createAnthropic } from '@ai-sdk/anthropic';
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 function loadTextModel( prompt ) {
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 function loadImageModel( prompt ) {
115
- const config = parseModelPrompt( prompt );
116
- const { provider: providerName, model: modelName } = config;
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 Error( `Provider "${providerName}" does not support image models.` );
24
+ throw new ValidationError( `Provider "${providerName}" does not support image models.` );
128
25
  }
129
-
130
- return imageModelFactory( modelName );
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 function loadTools( prompt ) {
140
- const config = parseToolsPrompt( prompt );
141
- const { tools: toolsConfig, provider: providerName } = config;
35
+ export const loadTools = prompt => {
36
+ const { tools: promptTools, provider: providerName } = prompt.config;
142
37
 
143
- if ( !toolsConfig || Object.keys( toolsConfig ).length === 0 ) {
38
+ if ( Object.keys( promptTools ?? {} ).length === 0 ) {
144
39
  return null;
145
40
  }
146
41
 
147
- const provider = providers[providerName];
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 || typeof provider.tools !== 'object' ) {
155
- throw new Error( `Provider "${providerName}" does not support provider-specific tools.` );
44
+ if ( !provider.tools ) {
45
+ throw new ValidationError( `Provider "${providerName}" does not support provider-specific tools.` );
156
46
  }
157
47
 
158
- const tools = {};
48
+ const supportedTools = Object.keys( provider.tools );
49
+ const invalidTools = Object.keys( promptTools ).filter( name => !supportedTools.includes( name ) );
159
50
 
160
- for ( const [ toolName, toolConfig ] of Object.entries( toolsConfig ) ) {
161
- const toolFactory = provider.tools[toolName];
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 tools;
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
+ };