O que são aplicativos?
Os aplicativos permitem criar e gerenciar personalizações do Twenty como código. Em vez de configurar tudo pela UI, você define seu modelo de dados e funções de lógica em código — tornando mais rápido criar, manter e distribuir para vários workspaces. O que você pode fazer hoje:- Defina objetos e campos personalizados como código (modelo de dados gerenciado)
- Crie funções de lógica com gatilhos personalizados
- Definir habilidades para agentes de IA
- Implemente o mesmo aplicativo em vários espaços de trabalho
Pré-requisitos
- Node.js 24+ e Yarn 4
- Um espaço de trabalho do Twenty e uma chave de API (crie uma em https://app.twenty.com/settings/api-webhooks)
Primeiros passos
Crie um novo aplicativo usando o gerador oficial, depois autentique-se e comece a desenvolver:Estrutura do projeto (com scaffold)
Ao executarnpx create-twenty-app@latest my-twenty-app, o gerador:
- Copia um aplicativo base mínimo para
my-twenty-app/ - Adiciona uma dependência local
twenty-sdke a configuração do Yarn 4 - Cria arquivos de configuração e scripts conectados à CLI
twenty - Gera arquivos principais (configuração da aplicação, papel padrão para funções de lógica, função de pós-instalação) além de arquivos de exemplo com base no modo de geração de estrutura
--exhaustive fica assim:
--minimal, apenas os arquivos principais são criados (application-config.ts, roles/default-role.ts e logic-functions/post-install.ts). Com --interactive, você escolhe quais arquivos de exemplo incluir.
Em alto nível:
- package.json: Declara o nome do app, versão, engines (Node 24+, Yarn 4), e adiciona
twenty-sdkalém de um scripttwentyque delega para a CLItwentylocal. Executeyarn twenty helppara listar todos os comandos disponíveis. - .gitignore: Ignora artefatos comuns como
node_modules,.yarn,generated/(cliente tipado),dist/,build/, pastas de cobertura, arquivos de log e arquivos.env*. - yarn.lock, .yarnrc.yml, .yarn/: Bloqueiam e configuram a ferramenta Yarn 4 usada pelo projeto.
- .nvmrc: Fixa a versão do Node.js esperada pelo projeto.
- eslint.config.mjs e tsconfig.json: Fornecem lint e configuração do TypeScript para os fontes TypeScript do seu aplicativo.
- README.md: Um README curto na raiz do aplicativo com instruções básicas.
- public/: Uma pasta para armazenar recursos públicos (imagens, fontes, arquivos estáticos) que serão servidos com sua aplicação. Os arquivos colocados aqui são enviados durante a sincronização e ficam acessíveis em tempo de execução.
- src/: O local principal onde você define seu aplicativo como código
Detecção de entidades
O SDK detecta entidades analisando seus arquivos TypeScript em busca de chamadasexport default define<Entity>({...}). Cada tipo de entidade tem uma função utilitária correspondente exportada de twenty-sdk:
| Função utilitária | Tipo de entidade |
|---|---|
defineObject() | Definições de objetos personalizados |
defineLogicFunction() | Definições de funções de lógica |
defineFrontComponent() | Definições de componentes de front-end |
defineRole() | Definições de papéis |
defineField() | Extensões de campos para objetos existentes |
defineView() | Definições de visualizações salvas |
defineNavigationMenuItem() | Definições de itens do menu de navegação |
defineSkill() | AI agent skill definitions |
A nomeação de arquivos é flexível. A detecção de entidades é baseada em AST — o SDK varre seus arquivos fonte em busca do padrão
export default define<Entity>({...}). Você pode organizar seus arquivos e pastas como quiser. Agrupar por tipo de entidade (por exemplo, logic-functions/, roles/) é apenas uma convenção para organização do código, não um requisito.yarn twenty app:devvai gerar automaticamente um cliente de API tipado emnode_modules/twenty-sdk/generated(cliente Twenty tipado + tipos do espaço de trabalho).yarn twenty entity:addwill add entity definition files undersrc/for your custom objects, functions, front components, roles, skills, and more.
Autenticação
Na primeira vez que você executaryarn twenty auth:login, será solicitado o seguinte:
- URL da API (padrão: http://localhost:3000 ou o perfil do seu espaço de trabalho atual)
- Chave de API
~/.twenty/config.json. Você pode manter vários perfis e alternar entre eles.
Gerenciando espaços de trabalho
yarn twenty auth:switch, todos os comandos subsequentes usarão esse espaço de trabalho por padrão. Você ainda pode substituí-lo temporariamente com --workspace <name>.
Use os recursos do SDK (tipos e configuração)
O twenty-sdk fornece blocos de construção tipados e funções utilitárias que você usa dentro do seu aplicativo. A seguir estão as partes principais que você usará com mais frequência.Funções utilitárias
O SDK fornece funções utilitárias para definir as entidades do seu app. Conforme descrito em Detecção de entidades, você deve usarexport default define<Entity>({...}) para que suas entidades sejam detectadas:
| Função | Finalidade |
|---|---|
defineApplication() | Configurar metadados do aplicativo (obrigatório, um por app) |
defineObject() | Define objetos personalizados com campos |
defineLogicFunction() | Defina funções de lógica com handlers |
defineFrontComponent() | Definir componentes de front-end para UI personalizada |
defineRole() | Configura permissões de papéis e acesso a objetos |
defineField() | Estender objetos existentes com campos adicionais |
defineView() | Define visualizações salvas para objetos |
defineNavigationMenuItem() | Define links de navegação da barra lateral |
defineSkill() | Define AI agent skills |
Definindo objetos
Objetos personalizados descrevem tanto o esquema quanto o comportamento de registros no seu espaço de trabalho. UsedefineObject() para definir objetos com validação integrada:
- Use
defineObject()para validação integrada e melhor suporte na IDE. - O
universalIdentifierdeve ser exclusivo e estável entre implantações. - Cada campo requer
name,type,labele seu própriouniversalIdentifierestável. - O array
fieldsé opcional — você pode definir objetos sem campos personalizados. - Você pode criar novos objetos usando
yarn twenty entity:add, que orienta você sobre nomeação, campos e relacionamentos.
Os campos base são criados automaticamente. Quando você define um objeto personalizado, o Twenty adiciona automaticamente campos padrão
como
id, name, createdAt, updatedAt, createdBy, updatedBy e deletedAt.
Você não precisa definir esses no seu array fields — adicione apenas seus campos personalizados.
Você pode substituir os campos padrão definindo um campo com o mesmo nome no seu array fields,
mas isso não é recomendado.Configuração do aplicativo (application-config.ts)
Todo aplicativo tem um único arquivoapplication-config.ts que descreve:
- O que é o aplicativo: identificadores, nome de exibição e descrição.
- Como suas funções são executadas: qual papel usam para permissões.
- Variáveis (opcional): pares chave–valor expostos às suas funções como variáveis de ambiente.
- (Opcional) função de pós-instalação: uma função de lógica que é executada após a instalação da aplicação.
defineApplication() to define your application configuration:
universalIdentifiersão IDs determinísticos que você controla; gere-os uma vez e mantenha-os estáveis entre sincronizações.applicationVariablestornam-se variáveis de ambiente para suas funções (por exemplo,DEFAULT_RECIPIENT_NAMEfica disponível comoprocess.env.DEFAULT_RECIPIENT_NAME).defaultRoleUniversalIdentifierdeve corresponder ao arquivo do papel (veja abaixo).postInstallLogicFunctionUniversalIdentifier(opcional) aponta para uma função de lógica que é executada automaticamente após a instalação da aplicação. Consulte Funções de pós-instalação.
Papéis e permissões
Os aplicativos podem definir papéis que encapsulam permissões sobre os objetos e ações do seu espaço de trabalho. O campodefaultRoleUniversalIdentifier em application-config.ts designa o papel padrão usado pelas funções de lógica do seu app.
- A chave de API em tempo de execução, injetada como
TWENTY_API_KEY, é derivada desse papel padrão de função. - O cliente tipado ficará restrito às permissões concedidas a esse papel.
- Siga o princípio do menor privilégio: crie um papel dedicado com apenas as permissões de que suas funções precisam e, em seguida, faça referência ao seu identificador universal.
Papel de função padrão (*.role.ts)
Ao criar um novo aplicativo com o scaffold, a CLI também cria um arquivo de papel padrão. UsedefineRole() para definir papéis com validação integrada:
universalIdentifier desse papel é então referenciado em application-config.ts como defaultRoleUniversalIdentifier. Em outras palavras:
- *.role.ts define o que o papel de função padrão pode fazer.
- application-config.ts aponta para esse papel para que suas funções herdem suas permissões.
- Comece pelo papel gerado pelo scaffold e depois restrinja-o progressivamente seguindo o princípio do menor privilégio.
- Substitua
objectPermissionsefieldPermissionspelos objetos/campos de que suas funções precisam. permissionFlagscontrolam o acesso a recursos em nível de plataforma. Mantenha-os mínimos; adicione apenas o que for necessário.- Veja um exemplo funcional no app Hello World:
packages/twenty-apps/hello-world/src/roles/function-role.ts.
Configuração de função de lógica e ponto de entrada
Cada arquivo de função usadefineLogicFunction() para exportar uma configuração com um handler e gatilhos opcionais.
- route: Expõe sua função em um caminho e método HTTP no endpoint
/s/:
por exemplo,path: '/post-card/create',-> chamar em<APP_URL>/s/post-card/create
- cron: Executa sua função em um agendamento usando uma expressão CRON.
- databaseEvent: Executa em eventos do ciclo de vida de objetos do espaço de trabalho. Quando a operação do evento é
updated, campos específicos a serem observados podem ser especificados no arrayupdatedFields. Se deixar indefinido ou vazio, qualquer atualização acionará a função.
por exemplo, person.updated
Notas:
- O array
triggersé opcional. Funções sem gatilhos podem ser usadas como funções utilitárias chamadas por outras funções. - Você pode misturar vários tipos de gatilho em uma única função.
Funções de pós-instalação
Uma função de pós-instalação é uma função de lógica que é executada automaticamente após a sua aplicação ser instalada em um espaço de trabalho. Isso é útil para tarefas de configuração únicas, como preencher dados padrão, criar registros iniciais ou configurar as configurações do espaço de trabalho. Ao criar a estrutura de um novo app comcreate-twenty-app, uma função de pós-instalação é gerada para você em src/logic-functions/post-install.ts:
application-config.ts:
- As funções de pós-instalação são funções de lógica padrão — elas usam
defineLogicFunction()como qualquer outra função. - O campo
postInstallLogicFunctionUniversalIdentifieremdefineApplication()é opcional. Se omitido, nenhuma função é executada após a instalação. - O tempo limite padrão é definido como 300 segundos (5 minutos) para permitir tarefas de configuração mais longas, como o pré-carregamento de dados.
- As funções de pós-instalação não precisam de gatilhos — elas são invocadas pela plataforma durante a instalação ou manualmente via
function:execute --postInstall.
Payload de gatilho de rota
Quando um gatilho de rota invoca sua função de lógica, ela recebe um objetoRoutePayload que segue o formato do AWS HTTP API v2. Importe o tipo de twenty-sdk:
RoutePayload tem a seguinte estrutura:
| Propriedade | Tipo | Descrição |
|---|---|---|
headers | Record<string, string | undefined> | Cabeçalhos HTTP (apenas aqueles listados em forwardedRequestHeaders) |
queryStringParameters | Record<string, string | undefined> | Parâmetros de query string (valores múltiplos unidos por vírgulas) |
pathParameters | Record<string, string | undefined> | Parâmetros de caminho extraídos do padrão de rota (por exemplo, /users/:id → { id: '123' }) |
corpo | object | null | Corpo da requisição analisado (JSON) |
isBase64Encoded | booleano | Se o corpo está codificado em base64 |
requestContext.http.method | string | Método HTTP (GET, POST, PUT, PATCH, DELETE) |
requestContext.http.path | string | Caminho bruto da requisição |
Encaminhamento de cabeçalhos HTTP
Por padrão, os cabeçalhos HTTP das requisições recebidas não são repassados para sua função de lógica por motivos de segurança. Para acessar cabeçalhos específicos, liste-os explicitamente no arrayforwardedRequestHeaders:
Os nomes dos cabeçalhos são normalizados para minúsculas. Acesse-os usando chaves em minúsculas (por exemplo,
event.headers['content-type']).- Gerado automaticamente: Execute
yarn twenty entity:adde escolha a opção para adicionar uma nova função de lógica. Isso gera um arquivo inicial com um handler e configuração. - Manual: Crie um novo arquivo
*.logic-function.tse usedefineLogicFunction(), seguindo o mesmo padrão.
Marcar uma função lógica como ferramenta
Funções lógicas podem ser expostas como ferramentas para agentes de IA e fluxos de trabalho. Quando uma função é marcada como ferramenta, ela fica disponível para os recursos de IA do Twenty e pode ser selecionada como uma etapa em automações de fluxos de trabalho. Para marcar uma função lógica como ferramenta, definaisTool: true e forneça um toolInputSchema descrevendo os parâmetros de entrada esperados usando JSON Schema:
isTool(boolean, padrão:false): Quando definido comotrue, a função é registrada como uma ferramenta e fica disponível para agentes de IA e automações de fluxos de trabalho.toolInputSchema(object, opcional): Um objeto JSON Schema que descreve os parâmetros que sua função aceita. Os agentes de IA usam esse esquema para entender quais entradas a ferramenta espera e para validar as chamadas. Se omitido, o esquema tem como padrão{ type: 'object', properties: {} }(sem parâmetros).- Funções com
isTool: false(ou não definido) não são expostas como ferramentas. Elas ainda podem ser executadas diretamente ou chamadas por outras funções, mas não aparecerão na descoberta de ferramentas. - Nomenclatura de ferramentas: Quando exposta como uma ferramenta, o nome da função é automaticamente normalizado para
logic_function_<name>(em minúsculas, caracteres não alfanuméricos substituídos por sublinhados). Por exemplo,enrich-companytorna-selogic_function_enrich_company. - Você pode combinar
isToolcom gatilhos — uma função pode ser ao mesmo tempo uma ferramenta (chamável por agentes de IA) e acionada por eventos (cron, eventos de banco de dados, rotas) simultaneamente.
Escreva uma boa
description. Os agentes de IA dependem do campo description da função para decidir quando usar a ferramenta. Seja específico sobre o que a ferramenta faz e quando ela deve ser chamada.Componentes de front-end
Componentes de front-end permitem criar componentes React personalizados que são renderizados na UI do Twenty. UsedefineFrontComponent() para definir componentes com validação integrada:
- Componentes de front-end são componentes React que renderizam em contextos isolados dentro do Twenty.
- Use o sufixo de arquivo
*.front-component.tsxpara detecção automática. - O campo
componentfaz referência ao seu componente React. - Os componentes são compilados e sincronizados automaticamente durante
yarn twenty app:dev.
- Gerado automaticamente: Execute
yarn twenty entity:adde escolha a opção para adicionar um novo componente de front-end. - Manual: Crie um novo arquivo
*.front-component.tsxe usedefineFrontComponent().
Habilidades
Skills define reusable instructions and capabilities that AI agents can use within your workspace. UsedefineSkill() to define skills with built-in validation:
nameis a unique identifier string for the skill (kebab-case recommended).labelis the human-readable display name shown in the UI.contentcontains the skill instructions — this is the text the AI agent uses.icon(optional) sets the icon displayed in the UI.description(optional) provides additional context about the skill’s purpose.
- Scaffolded: Run
yarn twenty entity:addand choose the option to add a new skill. - Manual: Create a new file and use
defineSkill(), following the same pattern.
Cliente tipado gerado
O cliente tipado é gerado automaticamente peloyarn twenty app:dev e armazenado em node_modules/twenty-sdk/generated com base no esquema do seu espaço de trabalho. Use-o em suas funções:
yarn twenty app:dev sempre que seus objetos ou campos forem alterados.
Credenciais em tempo de execução em funções de lógica
Quando sua função é executada no Twenty, a plataforma injeta credenciais como variáveis de ambiente antes da execução do seu código:TWENTY_API_URL: URL base da API do Twenty que seu aplicativo usa como alvo.TWENTY_API_KEY: Chave de curta duração com escopo para o papel de função padrão do seu aplicativo.
- Você não precisa passar a URL ou a chave de API para o cliente gerado. Ele lê
TWENTY_API_URLeTWENTY_API_KEYde process.env em tempo de execução. - As permissões da chave de API são determinadas pelo papel referenciado no seu
application-config.tsviadefaultRoleUniversalIdentifier. Este é o papel padrão usado pelas funções de lógica do seu app. - Os aplicativos podem definir papéis para seguir o princípio do menor privilégio. Conceda apenas as permissões de que suas funções precisam e, em seguida, aponte
defaultRoleUniversalIdentifierpara o identificador universal desse papel.
Exemplo Hello World
Explore um exemplo mínimo de ponta a ponta que demonstra objetos, funções de lógica, componentes de front-end e vários gatilhos aqui:Configuração manual (sem o gerador)
Embora recomendemos usarcreate-twenty-app para a melhor experiência inicial, você também pode configurar um projeto manualmente. Não instale a CLI globalmente. Em vez disso, adicione twenty-sdk como uma dependência local e configure um único script no seu package.json:
twenty:
yarn twenty <command>, por exemplo, yarn twenty app:dev, yarn twenty help, etc.
Resolução de Problemas
- Erros de autenticação: execute
yarn twenty auth:logine certifique-se de que sua chave de API tenha as permissões necessárias. - Não é possível conectar ao servidor: verifique a URL da API e se o servidor do Twenty está acessível.
- Tipos ou cliente ausentes/desatualizados: reinicie
yarn twenty app:dev— ele gera automaticamente o cliente tipado. - Modo de desenvolvimento não sincronizando: certifique-se de que
yarn twenty app:devesteja em execução e de que as alterações não estejam sendo ignoradas pelo seu ambiente.