跳转到主要内容
技能和智能体目前处于 Alpha 阶段。 该功能可用,但仍在演进中。
应用可以定义存在于工作区内的 AI 能力——可复用的技能指令以及具有自定义系统提示词的智能体。
技能定义了可复用的指令和能力,AI 智能体可在你的工作区中使用。 使用 defineSkill() 定义带内置校验的技能:
src/skills/example-skill.ts
import { defineSkill } from 'twenty-sdk/define';

export default defineSkill({
  universalIdentifier: 'a1b2c3d4-e5f6-7890-abcd-ef1234567890',
  name: 'sales-outreach',
  label: 'Sales Outreach',
  description: 'Guides the AI agent through a structured sales outreach process',
  icon: 'IconBrain',
  content: `You are a sales outreach assistant. When reaching out to a prospect:
1. Research the company and recent news
2. Identify the prospect's role and likely pain points
3. Draft a personalized message referencing specific details
4. Keep the tone professional but conversational`,
});
关键点:
  • name 是该技能的唯一标识字符串(推荐使用 kebab-case)。
  • label 是在 UI 中显示的人类可读名称。
  • content 包含技能指令——这是 AI 智能体使用的文本。
  • icon(可选)设置在 UI 中显示的图标。
  • description(可选)提供有关技能用途的更多上下文。
智能体是在你的工作区内驻留的 AI 助手。 使用 defineAgent() 来创建带有自定义系统提示词的智能体:
src/agents/example-agent.ts
import { defineAgent } from 'twenty-sdk/define';

export default defineAgent({
  universalIdentifier: 'b3c4d5e6-f7a8-9012-bcde-f34567890123',
  name: 'sales-assistant',
  label: 'Sales Assistant',
  description: 'Helps the sales team draft outreach emails and research prospects',
  icon: 'IconRobot',
  prompt: 'You are a helpful sales assistant. Help users with their questions and tasks.',
});
关键点:
  • name 是该智能体的唯一标识字符串(推荐使用 kebab-case)。
  • label 是在 UI 中显示的名称。
  • prompt 是定义智能体行为的系统提示词。
  • description(可选)提供有关智能体功能的上下文。
  • icon(可选)设置在 UI 中显示的图标。
  • modelId(可选)会覆盖该智能体使用的默认 AI 模型。
  • responseFormat(可选)控制代理输出的结构形式。 对于自由格式文本,默认值为 { type: 'text' }。 使用 { type: 'json', schema } 来强制生成结构化 JSON 输出。
默认情况下,代理返回自由格式文本。 要获取结构化输出,将 responseFormat 设置为 { type: 'json' },并提供一个 schema
src/agents/structured-agent.ts
import { defineAgent } from 'twenty-sdk/define';

export default defineAgent({
  universalIdentifier: 'c4d5e6f7-a8b9-0123-cdef-456789012345',
  name: 'lead-scorer',
  label: 'Lead Scorer',
  prompt: 'Score the lead and explain your reasoning.',
  responseFormat: {
    type: 'json',
    schema: {
      type: 'object',
      properties: {
        score: { type: 'number', description: 'Lead score from 0 to 100' },
        summary: { type: 'string', description: 'Short reasoning for the score' },
      },
      required: ['score', 'summary'],
      additionalProperties: false,
    },
  },
});
架构说明:
  • 该架构是一个扁平对象:每个属性的 type 必须是原始类型(stringnumberboolean)。 不支持嵌套对象和数组。
  • 每个属性上的 description(可选)用于引导模型应在此处填入什么内容。
  • required(可选)列出模型必须始终返回的属性。
  • additionalProperties: false(可选)禁止任何未在 properties 中声明的属性。
runAgent() 允许逻辑函数运行你的应用的某个代理(及其技能和工具)。 通过你传递给 defineAgent()universalIdentifier 来标识该代理:
src/logic-functions/run-enricher.ts
import { runAgent } from 'twenty-sdk/logic-function';

const { result, error, success } = await runAgent({
  agentUniversalIdentifier: 'b3c4d5e6-f7a8-9012-bcde-f34567890123',
  prompt: 'Enrich House Ad <recordId>: fill empty fields from its listing URL.',
});
关键点:
  • 代理以同步方式运行,并且可以通过其自身的工具读取/更新记录 —— runAgent() 会在运行完成后才 resolve。
  • 一个应用只能运行它自己的代理。
  • 应用的默认角色必须授予 AI 权限标记 —— 在其 permissionFlagUniversalIdentifiers 中添加 SystemPermissionFlag.AI(或设置 canAccessAllTools: true)。 否则,runAgent() 会因权限错误而失败。
  • 在逻辑函数上设置较大的 timeoutSeconds 值 —— 代理运行可能需要数秒时间。
  • 当运行完成时,successtrueresult 为非空;失败时,successfalseresultnull,并且 error 保存失败原因(例如,当工作区在运行过程中耗尽 AI 额度时)。
src/roles/default-role.ts
import { defineApplicationRole, SystemPermissionFlag } from 'twenty-sdk/define';

export default defineApplicationRole({
  universalIdentifier: 'b648f87b-1d26-4961-b974-0908fd991061',
  label: 'Default function role',
  // runAgent() requires the AI permission flag on the app's default role.
  permissionFlagUniversalIdentifiers: [SystemPermissionFlag.AI],
});
**避免循环:**如果你从 *.updated 数据库事件触发器中调用 runAgent(),而代理又会更新同一条记录,请将触发器的 updatedFields 限定为代理永远不会写入的字段(例如来源 URL),或者在调用 runAgent() 之前检查任何目标字段是否仍为空。