✨ Add Dify.AI block (#1183)
Implemented [Dify.AI](https://dify.ai) Block - Dify Features: -- Can Create Multiple Chat Bots -- Assign Knowledge base/vector database to chat bots -- Variables send by client to be used in the prompt -- Options of custom and cloud AI LLMs to be changed with on click - Dify API Function/Action Implemented -- Create Chat Message (Takes in input variables, query, conversation id, user id and returns answer, usage metadata and conversation id) - Future Implantations with this block -- Streaming response -- File Upload for GPT Vision -- Speech to text action <!-- This is an auto-generated comment: release notes by coderabbit.ai --> ## Summary by CodeRabbit - **New Features** - Introduced the ability to create chat messages within the Dify platform. - Added secure authentication for Dify.AI accounts. - Implemented a new Dify.AI block with integrated chat message creation and logo display. - Enabled Dify.AI block in the repository for user access. - **Enhancements** - Enhanced security for API key input by changing it to a password field. - **Documentation** - Included new types to support Dify AI service responses. - **Refactor** - Updated schema imports and array listings to include the new Dify.AI block. <!-- end of auto-generated comment: release notes by coderabbit.ai --> --------- Co-authored-by: Baptiste Arnaud <baptiste.arnaud95@gmail.com>
This commit is contained in:
committed by
GitHub
parent
cf101d6cf6
commit
0817fbaebb
80
packages/forge/blocks/difyAi/actions/createChatMessage.ts
Normal file
80
packages/forge/blocks/difyAi/actions/createChatMessage.ts
Normal file
@ -0,0 +1,80 @@
|
||||
import { createAction, option } from '@typebot.io/forge'
|
||||
import { isDefined, isEmpty } from '@typebot.io/lib'
|
||||
import { got } from 'got'
|
||||
import { auth } from '../auth'
|
||||
import { DifyResponse } from '../types'
|
||||
import { defaultBaseUrl } from '../constants'
|
||||
|
||||
export const createChatMessage = createAction({
|
||||
auth,
|
||||
name: 'Create Chat Message',
|
||||
options: option.object({
|
||||
query: option.string.layout({
|
||||
label: 'Query',
|
||||
placeholder: 'User input/question content',
|
||||
inputType: 'textarea',
|
||||
isRequired: true,
|
||||
}),
|
||||
conversation_id: option.string.layout({
|
||||
label: 'Conversation ID',
|
||||
moreInfoTooltip:
|
||||
'Used to remember the conversation with the user. If empty, a new conversation id is created.',
|
||||
}),
|
||||
user: option.string.layout({
|
||||
label: 'User',
|
||||
moreInfoTooltip:
|
||||
'The user identifier, defined by the developer, must ensure uniqueness within the app.',
|
||||
}),
|
||||
inputs: option.keyValueList.layout({
|
||||
accordion: 'Inputs',
|
||||
}),
|
||||
responseMapping: option
|
||||
.saveResponseArray(['Answer', 'Conversation ID', 'Total Tokens'])
|
||||
.layout({
|
||||
accordion: 'Save response',
|
||||
}),
|
||||
}),
|
||||
getSetVariableIds: ({ responseMapping }) =>
|
||||
responseMapping?.map((r) => r.variableId).filter(isDefined) ?? [],
|
||||
run: {
|
||||
server: async ({
|
||||
credentials: { apiEndpoint, apiKey },
|
||||
options: { conversation_id, query, user, inputs, responseMapping },
|
||||
variables,
|
||||
}) => {
|
||||
const res: DifyResponse = await got
|
||||
.post((apiEndpoint ?? defaultBaseUrl) + '/v1/chat-messages', {
|
||||
headers: {
|
||||
Authorization: `Bearer ${apiKey}`,
|
||||
},
|
||||
json: {
|
||||
inputs: inputs?.reduce((acc, { key, value }) => {
|
||||
if (isEmpty(key) || isEmpty(value)) return acc
|
||||
return {
|
||||
...acc,
|
||||
[key]: value,
|
||||
}
|
||||
}, {}),
|
||||
query,
|
||||
response_mode: 'blocking',
|
||||
conversation_id,
|
||||
user,
|
||||
},
|
||||
})
|
||||
.json()
|
||||
|
||||
responseMapping?.forEach((mapping) => {
|
||||
if (!mapping.variableId) return
|
||||
|
||||
const item = mapping.item ?? 'Answer'
|
||||
if (item === 'Answer') variables.set(mapping.variableId, res.answer)
|
||||
|
||||
if (item === 'Conversation ID')
|
||||
variables.set(mapping.variableId, res.conversation_id)
|
||||
|
||||
if (item === 'Total Tokens')
|
||||
variables.set(mapping.variableId, res.metadata.usage.total_tokens)
|
||||
})
|
||||
},
|
||||
},
|
||||
})
|
23
packages/forge/blocks/difyAi/auth.ts
Normal file
23
packages/forge/blocks/difyAi/auth.ts
Normal file
@ -0,0 +1,23 @@
|
||||
import { option, AuthDefinition } from '@typebot.io/forge'
|
||||
import { defaultBaseUrl } from './constants'
|
||||
|
||||
export const auth = {
|
||||
type: 'encryptedCredentials',
|
||||
name: 'Dify.AI account',
|
||||
schema: option.object({
|
||||
apiEndpoint: option.string.layout({
|
||||
label: 'API Endpoint',
|
||||
isRequired: true,
|
||||
helperText: 'URI where the Service API is hosted.',
|
||||
withVariableButton: false,
|
||||
defaultValue: defaultBaseUrl,
|
||||
}),
|
||||
apiKey: option.string.layout({
|
||||
label: 'App API key',
|
||||
isRequired: true,
|
||||
helperText: 'API Secret Key for your Dify App.',
|
||||
inputType: 'password',
|
||||
withVariableButton: false,
|
||||
}),
|
||||
}),
|
||||
} satisfies AuthDefinition
|
1
packages/forge/blocks/difyAi/constants.ts
Normal file
1
packages/forge/blocks/difyAi/constants.ts
Normal file
@ -0,0 +1 @@
|
||||
export const defaultBaseUrl = 'https://api.dify.ai'
|
13
packages/forge/blocks/difyAi/index.ts
Normal file
13
packages/forge/blocks/difyAi/index.ts
Normal file
@ -0,0 +1,13 @@
|
||||
import { createBlock } from '@typebot.io/forge'
|
||||
import { DifyAiLogo } from './logo'
|
||||
import { auth } from './auth'
|
||||
import { createChatMessage } from './actions/createChatMessage'
|
||||
|
||||
export const difyAi = createBlock({
|
||||
id: 'dify-ai',
|
||||
name: 'Dify.AI',
|
||||
tags: ['dify', 'ai', 'documents', 'files', 'knowledge base'],
|
||||
LightLogo: DifyAiLogo,
|
||||
auth,
|
||||
actions: [createChatMessage],
|
||||
})
|
11
packages/forge/blocks/difyAi/logo.tsx
Normal file
11
packages/forge/blocks/difyAi/logo.tsx
Normal file
@ -0,0 +1,11 @@
|
||||
import React from 'react'
|
||||
|
||||
export const DifyAiLogo = (props: React.SVGProps<SVGSVGElement>) => (
|
||||
<svg viewBox="0 0 25 25" xmlns="http://www.w3.org/2000/svg" {...props}>
|
||||
<image
|
||||
href="https://framerusercontent.com/images/xRJ6vNo9mUYeVNxt0KITXCXEuSk.png"
|
||||
height="25"
|
||||
width="25"
|
||||
/>
|
||||
</svg>
|
||||
)
|
16
packages/forge/blocks/difyAi/package.json
Normal file
16
packages/forge/blocks/difyAi/package.json
Normal file
@ -0,0 +1,16 @@
|
||||
{
|
||||
"name": "@typebot.io/dify-ai-block",
|
||||
"version": "1.0.0",
|
||||
"description": "",
|
||||
"main": "index.ts",
|
||||
"keywords": [],
|
||||
"license": "ISC",
|
||||
"devDependencies": {
|
||||
"@typebot.io/forge": "workspace:*",
|
||||
"@typebot.io/lib": "workspace:*",
|
||||
"@typebot.io/tsconfig": "workspace:*",
|
||||
"@types/react": "18.2.15",
|
||||
"got": "12.6.0",
|
||||
"typescript": "5.3.2"
|
||||
}
|
||||
}
|
10
packages/forge/blocks/difyAi/tsconfig.json
Normal file
10
packages/forge/blocks/difyAi/tsconfig.json
Normal file
@ -0,0 +1,10 @@
|
||||
{
|
||||
"extends": "@typebot.io/tsconfig/base.json",
|
||||
"include": ["**/*.ts", "**/*.tsx"],
|
||||
"exclude": ["node_modules"],
|
||||
"compilerOptions": {
|
||||
"lib": ["ESNext", "DOM"],
|
||||
"noEmit": true,
|
||||
"jsx": "react"
|
||||
}
|
||||
}
|
9
packages/forge/blocks/difyAi/types.ts
Normal file
9
packages/forge/blocks/difyAi/types.ts
Normal file
@ -0,0 +1,9 @@
|
||||
export type DifyResponse = {
|
||||
answer: string
|
||||
metadata: {
|
||||
usage: {
|
||||
total_tokens: number
|
||||
}
|
||||
}
|
||||
conversation_id: string
|
||||
}
|
Reference in New Issue
Block a user