2
0
Files
bot/packages/forge/blocks/anthropic/actions/createChatMessage.tsx
Baptiste Arnaud 043f0054b0 ⬆️ Upgrade AI SDK (#1641)
2024-07-15 14:32:42 +02:00

166 lines
4.9 KiB
TypeScript

import { createAction, option } from '@typebot.io/forge'
import { auth } from '../auth'
import {
anthropicLegacyModels,
anthropicModelLabels,
anthropicModels,
defaultAnthropicOptions,
maxToolRoundtrips,
} from '../constants'
import { isDefined } from '@typebot.io/lib'
import { createAnthropic } from '@ai-sdk/anthropic'
import { generateText } from 'ai'
import { runChatCompletionStream } from '../helpers/runChatCompletionStream'
import { toolsSchema } from '@typebot.io/ai/schemas'
import { parseTools } from '@typebot.io/ai/parseTools'
import { parseChatCompletionMessages } from '@typebot.io/ai/parseChatCompletionMessages'
import { isModelCompatibleWithVision } from '../helpers/isModelCompatibleWithVision'
const nativeMessageContentSchema = {
content: option.string.layout({
inputType: 'textarea',
placeholder: 'Content',
}),
}
const userMessageItemSchema = option
.object({
role: option.literal('user'),
})
.extend(nativeMessageContentSchema)
const assistantMessageItemSchema = option
.object({
role: option.literal('assistant'),
})
.extend(nativeMessageContentSchema)
const dialogueMessageItemSchema = option.object({
role: option.literal('Dialogue'),
dialogueVariableId: option.string.layout({
inputType: 'variableDropdown',
placeholder: 'Dialogue variable',
}),
startsBy: option.enum(['user', 'assistant']).layout({
label: 'starts by',
direction: 'row',
defaultValue: 'user',
}),
})
export const options = option.object({
model: option.enum(anthropicModels).layout({
toLabels: (val) =>
val
? anthropicModelLabels[val as (typeof anthropicModels)[number]]
: undefined,
hiddenItems: anthropicLegacyModels,
}),
messages: option
.array(
option.discriminatedUnion('role', [
userMessageItemSchema,
assistantMessageItemSchema,
dialogueMessageItemSchema,
])
)
.layout({ accordion: 'Messages', itemLabel: 'message', isOrdered: true }),
tools: toolsSchema,
systemMessage: option.string.layout({
accordion: 'Advanced Settings',
label: 'System prompt',
direction: 'row',
inputType: 'textarea',
}),
temperature: option.number.layout({
accordion: 'Advanced Settings',
label: 'Temperature',
direction: 'row',
defaultValue: defaultAnthropicOptions.temperature,
}),
maxTokens: option.number.layout({
accordion: 'Advanced Settings',
label: 'Max Tokens',
direction: 'row',
defaultValue: defaultAnthropicOptions.maxTokens,
}),
responseMapping: option
.saveResponseArray(['Message Content'] as const)
.layout({
accordion: 'Save Response',
}),
})
const transformToChatCompletionOptions = (
options: any,
resetModel = false
) => ({
...options,
model: resetModel ? undefined : options.model,
action: 'Create chat completion',
responseMapping: options.responseMapping?.map((res: any) =>
res.item === 'Message Content' ? { ...res, item: 'Message content' } : res
),
})
export const createChatMessage = createAction({
name: 'Create Chat Message',
auth,
options,
turnableInto: [
{
blockId: 'mistral',
transform: (opts) => transformToChatCompletionOptions(opts, true),
},
{
blockId: 'openai',
transform: (opts) => transformToChatCompletionOptions(opts, true),
},
{ blockId: 'open-router', transform: transformToChatCompletionOptions },
{ blockId: 'together-ai', transform: transformToChatCompletionOptions },
],
getSetVariableIds: ({ responseMapping }) =>
responseMapping?.map((res) => res.variableId).filter(isDefined) ?? [],
run: {
server: async ({ credentials: { apiKey }, options, variables, logs }) => {
const modelName = options.model ?? defaultAnthropicOptions.model
const model = createAnthropic({
apiKey,
})(modelName)
const { text } = await generateText({
model,
temperature: options.temperature
? Number(options.temperature)
: undefined,
messages: await parseChatCompletionMessages({
messages: options.messages,
isVisionEnabled: isModelCompatibleWithVision(modelName),
shouldDownloadImages: true,
variables,
}),
tools: parseTools({ tools: options.tools, variables }),
maxToolRoundtrips: maxToolRoundtrips,
})
options.responseMapping?.forEach((mapping) => {
if (!mapping.variableId) return
if (!mapping.item || mapping.item === 'Message Content')
variables.set(mapping.variableId, text)
})
},
stream: {
getStreamVariableId: (options) =>
options.responseMapping?.find(
(res) => res.item === 'Message Content' || !res.item
)?.variableId,
run: async ({ credentials: { apiKey }, options, variables }) =>
runChatCompletionStream({
credentials: { apiKey },
options,
variables,
}),
},
},
})