Ana içeriğe atla
Uygulamalar şu anda alfa testinde. Özellik işlevsel ancak hâlâ gelişmekte.

Uygulamalar Nedir?

Uygulamalar, Twenty özelleştirmelerini kod olarak oluşturup yönetmenizi sağlar. Her şeyi UI üzerinden yapılandırmak yerine, veri modelinizi ve mantık fonksiyonlarınızı kodla tanımlarsınız — bu da oluşturmayı, bakımı ve birden çok çalışma alanına dağıtmayı hızlandırır. Bugün Yapabilecekleriniz:
  • Özel nesneleri ve alanları kod olarak tanımlayın (yönetilen veri modeli)
  • Özel tetikleyicilerle mantık fonksiyonları oluşturun
  • Yapay zekâ ajanları için becerileri tanımlayın
  • Aynı uygulamayı birden çok çalışma alanına dağıtın

Ön Gereksinimler

Başlarken

Resmi scaffolder aracını kullanarak yeni bir uygulama oluşturun, ardından kimlik doğrulaması yapıp geliştirmeye başlayın:
# Scaffold a new app (includes all examples by default)
npx create-twenty-app@latest my-twenty-app
cd my-twenty-app

# If you don't use yarn@4
corepack enable
yarn install

# Authenticate using your API key (you'll be prompted)
yarn twenty auth:login

# Start dev mode: automatically syncs local changes to your workspace
yarn twenty app:dev
İskelet oluşturucu, hangi örnek dosyaların dahil edileceğini kontrol etmek için üç modu destekler:
# Default (exhaustive): all examples (object, field, logic function, front component, view, navigation menu item, skill)
npx create-twenty-app@latest my-app

# Minimal: only core files (application-config.ts and default-role.ts)
npx create-twenty-app@latest my-app --minimal

# Interactive: select which examples to include
npx create-twenty-app@latest my-app --interactive
Buradan şunları yapabilirsiniz:
# Add a new entity to your application (guided)
yarn twenty entity:add

# Watch your application's function logs
yarn twenty function:logs

# Execute a function by name
yarn twenty function:execute -n my-function -p '{"name": "test"}'

# Execute the post-install function
yarn twenty function:execute --postInstall

# Uninstall the application from the current workspace
yarn twenty app:uninstall

# Display commands' help
yarn twenty help
Ayrıca bkz.: create-twenty-app ve twenty-sdk CLI için CLI başvuru sayfaları.

Proje yapısı (şablondan oluşturulmuş)

npx create-twenty-app@latest my-twenty-app komutunu çalıştırdığınızda scaffolder şunları yapar:
  • Minimal bir temel uygulamayı my-twenty-app/ içine kopyalar
  • Yerel bir twenty-sdk bağımlılığı ve Yarn 4 yapılandırması ekler
  • twenty CLI ile bağlantılı yapılandırma dosyaları ve betikler oluşturur
  • İskelet oluşturma moduna bağlı olarak çekirdek dosyaları (uygulama yapılandırması, varsayılan işlev rolü, kurulum sonrası işlev) ile örnek dosyaları üretir
Varsayılan --exhaustive moduyla yeni oluşturulmuş bir uygulama şu şekilde görünür:
my-twenty-app/
  package.json
  yarn.lock
  .gitignore
  .nvmrc
  .yarnrc.yml
  .yarn/
    install-state.gz
  eslint.config.mjs
  tsconfig.json
  README.md
  public/                           # Public assets folder (images, fonts, etc.)
  src/
  ├── application-config.ts           # Required - main application configuration
  ├── roles/
  │   └── default-role.ts               # Default role for logic functions
  ├── objects/
  │   └── example-object.ts             # Example custom object definition
  ├── fields/
  │   └── example-field.ts              # Example standalone field definition
  ├── logic-functions/
  │   ├── hello-world.ts                # Example logic function
  │   └── post-install.ts               # Post-install logic function
  ├── front-components/
  │   └── hello-world.tsx               # Example front component
  ├── views/
  │   └── example-view.ts                # Example saved view definition
  ├── navigation-menu-items/
  │   └── example-navigation-menu-item.ts # Example sidebar navigation link
  └── skills/
      └── example-skill.ts                # Example AI agent skill definition
--minimal ile yalnızca çekirdek dosyalar oluşturulur (application-config.ts, roles/default-role.ts ve logic-functions/post-install.ts). --interactive ile hangi örnek dosyaların dahil edileceğini siz seçersiniz. Genel hatlarıyla:
  • package.json: Uygulama adını, sürümünü, motorları (Node 24+, Yarn 4) bildirir ve twenty-sdk ile yerel twenty CLI’sine yetki devreden bir twenty betiği ekler. Tüm mevcut komutları listelemek için yarn twenty help komutunu çalıştırın.
  • .gitignore: node_modules, .yarn, generated/ (türlendirilmiş istemci), dist/, build/, kapsam klasörleri, günlük dosyaları ve .env* dosyaları gibi yaygın artifaktları yok sayar.
  • yarn.lock, .yarnrc.yml, .yarn/: Proje tarafından kullanılan Yarn 4 araç zincirini kilitler ve yapılandırır.
  • .nvmrc: Projenin beklediği Node.js sürümünü sabitler.
  • eslint.config.mjs ve tsconfig.json: Uygulamanızın TypeScript kaynakları için linting ve TypeScript yapılandırması sağlar.
  • README.md: Uygulama kökünde temel talimatların yer aldığı kısa bir README.
  • public/: Uygulamanızla birlikte sunulacak genel varlıkları (görseller, yazı tipleri, statik dosyalar) depolamak için bir klasör. Buraya yerleştirilen dosyalar senkronizasyon sırasında yüklenir ve çalışma zamanında erişilebilir olur.
  • src/: Uygulamanızı kod olarak tanımladığınız ana yer

Varlık algılama

SDK, TypeScript dosyalarınızı export default define<Entity>({...}) çağrılarını arayarak ayrıştırıp varlıkları algılar. Her varlık türünün, twenty-sdk tarafından dışa aktarılan karşılık gelen bir yardımcı fonksiyonu vardır:
Yardımcı fonksiyonVarlık türü
defineObject()Özel nesne tanımları
defineLogicFunction()Mantık fonksiyon tanımları
defineFrontComponent()Front component definitions
defineRole()Rol tanımları
defineField()Mevcut nesneler için alan genişletmeleri
defineView()Kaydedilmiş görünüm tanımları
defineNavigationMenuItem()Gezinme menüsü öğesi tanımları
defineSkill()AI agent skill definitions
Dosya adlandırma esnektir. Varlık algılama AST tabanlıdır — SDK, kaynak dosyalarınızı export default define<Entity>({...}) desenini bulmak için tarar. Dosyalarınızı ve klasörlerinizi dilediğiniz gibi düzenleyebilirsiniz. Varlık türüne göre gruplama (örn. logic-functions/, roles/) bir gereklilik değil, yalnızca kod organizasyonu için bir gelenektir.
Algılanan bir varlığa örnek:
// This file can be named anything and placed anywhere in src/
import { defineObject, FieldType } from 'twenty-sdk';

export default defineObject({
  universalIdentifier: '...',
  nameSingular: 'postCard',
  // ... rest of config
});
İlerideki komutlar daha fazla dosya ve klasör ekleyecektir:
  • yarn twenty app:dev, node_modules/twenty-sdk/generated içinde tipli bir API istemcisini otomatik olarak oluşturur (tipli Twenty istemcisi + çalışma alanı türleri).
  • yarn twenty entity:add will add entity definition files under src/ for your custom objects, functions, front components, roles, skills, and more.

Kimlik Doğrulama

yarn twenty auth:login komutunu ilk kez çalıştırdığınızda, sizden şunlar istenir:
  • API URL’si (varsayılan: http://localhost:3000 veya mevcut çalışma alanı profiliniz)
  • API anahtarı
Kimlik bilgileriniz kullanıcı başına ~/.twenty/config.json içinde saklanır. You can maintain multiple profiles and switch between them.

Managing workspaces

# Etkileşimli giriş yapın (önerilir)
yarn twenty auth:login

# Belirli bir çalışma alanı profiline giriş yapın
yarn twenty auth:login --workspace my-custom-workspace

# Yapılandırılmış tüm çalışma alanlarını listeleyin
yarn twenty auth:list

# Varsayılan çalışma alanını değiştirin (etkileşimli)
yarn twenty auth:switch

# Belirli bir çalışma alanına geçin
yarn twenty auth:switch production

# Mevcut kimlik doğrulama durumunu kontrol edin
yarn twenty auth:status
yarn twenty auth:switch ile çalışma alanlarını değiştirdikten sonra, sonraki tüm komutlar varsayılan olarak o çalışma alanını kullanacaktır. You can still override it temporarily with --workspace <name>.

SDK kaynaklarını kullanın (türler ve yapılandırma)

twenty-sdk, uygulamanız içinde kullandığınız türlendirilmiş yapı taşları ve yardımcı fonksiyonlar sağlar. Aşağıda en sık dokunacağınız başlıca parçalar yer alıyor.

Yardımcı fonksiyonlar

SDK, uygulama varlıklarınızı tanımlamak için yardımcı fonksiyonlar sağlar. Varlık algılama bölümünde açıklandığı gibi, varlıklarınızın algılanması için export default define<Entity>({...}) kullanmalısınız:
FonksiyonAmaç
defineApplication()Uygulama meta verilerini yapılandırın (zorunlu, uygulama başına bir adet)
defineObject()Alanlara sahip özel nesneler tanımlayın
defineLogicFunction()İşleyicilerle mantık fonksiyonları tanımlayın
defineFrontComponent()Özel kullanıcı arayüzü için ön uç bileşenlerini tanımlayın
defineRole()Rol izinlerini ve nesne erişimini yapılandırın
defineField()Mevcut nesneleri ek alanlarla genişletin
defineView()Nesneler için kaydedilmiş görünümler tanımlayın
defineNavigationMenuItem()Kenar çubuğu gezinme bağlantılarını tanımlayın
defineSkill()Define AI agent skills
Bu fonksiyonlar, derleme zamanında yapılandırmanızı doğrular ve IDE otomatik tamamlama ile tür güvenliği sağlar.

Nesnelerin tanımlanması

Özel nesneler, çalışma alanınızdaki kayıtlar için hem şemayı hem de davranışı tanımlar. Yerleşik doğrulamayla nesneler tanımlamak için defineObject() kullanın:
// src/app/postCard.object.ts
import { defineObject, FieldType } from 'twenty-sdk';

enum PostCardStatus {
  DRAFT = 'DRAFT',
  SENT = 'SENT',
  DELIVERED = 'DELIVERED',
  RETURNED = 'RETURNED',
}

export default defineObject({
  universalIdentifier: '54b589ca-eeed-4950-a176-358418b85c05',
  nameSingular: 'postCard',
  namePlural: 'postCards',
  labelSingular: 'Post Card',
  labelPlural: 'Post Cards',
  description: 'A post card object',
  icon: 'IconMail',
  fields: [
    {
      universalIdentifier: '58a0a314-d7ea-4865-9850-7fb84e72f30b',
      name: 'content',
      type: FieldType.TEXT,
      label: 'Content',
      description: "Postcard's content",
      icon: 'IconAbc',
    },
    {
      universalIdentifier: 'c6aa31f3-da76-4ac6-889f-475e226009ac',
      name: 'recipientName',
      type: FieldType.FULL_NAME,
      label: 'Recipient name',
      icon: 'IconUser',
    },
    {
      universalIdentifier: '95045777-a0ad-49ec-98f9-22f9fc0c8266',
      name: 'recipientAddress',
      type: FieldType.ADDRESS,
      label: 'Recipient address',
      icon: 'IconHome',
    },
    {
      universalIdentifier: '87b675b8-dd8c-4448-b4ca-20e5a2234a1e',
      name: 'status',
      type: FieldType.SELECT,
      label: 'Status',
      icon: 'IconSend',
      defaultValue: `'${PostCardStatus.DRAFT}'`,
      options: [
        { value: PostCardStatus.DRAFT, label: 'Draft', position: 0, color: 'gray' },
        { value: PostCardStatus.SENT, label: 'Sent', position: 1, color: 'orange' },
        { value: PostCardStatus.DELIVERED, label: 'Delivered', position: 2, color: 'green' },
        { value: PostCardStatus.RETURNED, label: 'Returned', position: 3, color: 'orange' },
      ],
    },
    {
      universalIdentifier: 'e06abe72-5b44-4e7f-93be-afc185a3c433',
      name: 'deliveredAt',
      type: FieldType.DATE_TIME,
      label: 'Delivered at',
      icon: 'IconCheck',
      isNullable: true,
      defaultValue: null,
    },
  ],
});
Önemli noktalar:
  • Yerleşik doğrulama ve daha iyi IDE desteği için defineObject() kullanın.
  • universalIdentifier dağıtımlar arasında benzersiz ve kararlı olmalıdır.
  • Her alan bir name, type, label ve kendi kararlı universalIdentifier değerini gerektirir.
  • fields dizisi isteğe bağlıdır — özel alanlar olmadan da nesneler tanımlayabilirsiniz.
  • yarn twenty entity:add kullanarak, adlandırma, alanlar ve ilişkiler konusunda sizi yönlendirerek yeni nesneler oluşturabilirsiniz.
Temel alanlar otomatik olarak oluşturulur. Özel bir nesne tanımladığınızda Twenty, standart alanları otomatik olarak ekler örneğin id, name, createdAt, updatedAt, createdBy, updatedBy ve deletedAt. Bunları fields dizinizde tanımlamanız gerekmez — yalnızca özel alanlarınızı ekleyin. fields dizinizde aynı ada sahip bir alan tanımlayarak varsayılan alanları geçersiz kılabilirsiniz, ancak bu önerilmez.

Uygulama yapılandırması (application-config.ts)

Her uygulamanın aşağıdakileri açıklayan tek bir application-config.ts dosyası vardır:
  • Uygulamanın kim olduğu: tanımlayıcılar, görünen ad ve açıklama.
  • Fonksiyonlarının nasıl çalıştığı: izinler için hangi rolü kullandıkları.
  • (İsteğe bağlı) değişkenler: fonksiyonlarınıza ortam değişkenleri olarak sunulan anahtar–değer çiftleri.
  • (İsteğe bağlı) kurulum sonrası işlev: uygulama yüklendikten sonra çalışan bir mantık işlevi.
Uygulama yapılandırmanızı tanımlamak için defineApplication() kullanın:
// src/application-config.ts
import { defineApplication } from 'twenty-sdk';
import { DEFAULT_ROLE_UNIVERSAL_IDENTIFIER } from 'src/roles/default-role';
import { POST_INSTALL_UNIVERSAL_IDENTIFIER } from 'src/logic-functions/post-install';

export default defineApplication({
  universalIdentifier: '4ec0391d-18d5-411c-b2f3-266ddc1c3ef7',
  displayName: 'My Twenty App',
  description: 'My first Twenty app',
  icon: 'IconWorld',
  applicationVariables: {
    DEFAULT_RECIPIENT_NAME: {
      universalIdentifier: '19e94e59-d4fe-4251-8981-b96d0a9f74de',
      description: 'Default recipient name for postcards',
      value: 'Jane Doe',
      isSecret: false,
    },
  },
  defaultRoleUniversalIdentifier: DEFAULT_ROLE_UNIVERSAL_IDENTIFIER,
  postInstallLogicFunctionUniversalIdentifier: POST_INSTALL_UNIVERSAL_IDENTIFIER,
});
Notlar:
  • universalIdentifier alanları size ait belirleyici kimliklerdir; bunları bir kez oluşturun ve eşitlemeler boyunca kararlı tutun.
  • applicationVariables, fonksiyonlarınız için ortam değişkenlerine dönüşür (örneğin, DEFAULT_RECIPIENT_NAME değeri process.env.DEFAULT_RECIPIENT_NAME olarak kullanılabilir).
  • defaultRoleUniversalIdentifier, rol dosyasıyla eşleşmelidir (aşağıya bakın).
  • postInstallLogicFunctionUniversalIdentifier (isteğe bağlı), uygulama yüklendikten sonra otomatik olarak çalışan bir mantık işlevine işaret eder. Bkz. Kurulum sonrası işlevler.

Roller ve izinler

Uygulamalar, çalışma alanınızdaki nesneler ve eylemler üzerindeki izinleri kapsülleyen roller tanımlayabilir. application-config.ts içindeki defaultRoleUniversalIdentifier alanı, uygulamanızın mantık fonksiyonlarının kullandığı varsayılan rolü belirtir.
  • TWENTY_API_KEY olarak enjekte edilen çalışma zamanı API anahtarı bu varsayılan fonksiyon rolünden türetilir.
  • Türlendirilmiş istemci, o role tanınan izinlerle sınırlandırılır.
  • En az ayrıcalık ilkesini izleyin: Yalnızca fonksiyonlarınızın ihtiyaç duyduğu izinlere sahip özel bir rol oluşturun ve ardından evrensel tanımlayıcısına referans verin.
Varsayılan fonksiyon rolü (*.role.ts)
Yeni bir uygulama oluşturduğunuzda CLI ayrıca varsayılan bir rol dosyası da oluşturur. Yerleşik doğrulamayla roller tanımlamak için defineRole() kullanın:
// src/roles/default-role.ts
import { defineRole, PermissionFlag } from 'twenty-sdk';

export const DEFAULT_ROLE_UNIVERSAL_IDENTIFIER =
  'b648f87b-1d26-4961-b974-0908fd991061';

export default defineRole({
  universalIdentifier: DEFAULT_ROLE_UNIVERSAL_IDENTIFIER,
  label: 'Default function role',
  description: 'Default role for function Twenty client',
  canReadAllObjectRecords: false,
  canUpdateAllObjectRecords: false,
  canSoftDeleteAllObjectRecords: false,
  canDestroyAllObjectRecords: false,
  canUpdateAllSettings: false,
  canBeAssignedToAgents: false,
  canBeAssignedToUsers: false,
  canBeAssignedToApiKeys: false,
  objectPermissions: [
    {
      objectUniversalIdentifier: '9f9882af-170c-4879-b013-f9628b77c050',
      canReadObjectRecords: true,
      canUpdateObjectRecords: true,
      canSoftDeleteObjectRecords: false,
      canDestroyObjectRecords: false,
    },
  ],
  fieldPermissions: [
    {
      objectUniversalIdentifier: '9f9882af-170c-4879-b013-f9628b77c050',
      fieldUniversalIdentifier: 'b2c37dc0-8ae7-470e-96cd-1476b47dfaff',
      canReadFieldValue: false,
      canUpdateFieldValue: false,
    },
  ],
  permissionFlags: [PermissionFlag.APPLICATIONS],
});
Bu rolün universalIdentifier değeri daha sonra application-config.ts içinde defaultRoleUniversalIdentifier olarak referans verilir. Başka bir deyişle:
  • *.role.ts, varsayılan fonksiyon rolünün neler yapabileceğini tanımlar.
  • application-config.ts, fonksiyonlarınızın izinlerini devralması için bu role işaret eder.
Notlar:
  • Oluşturulan rol ile başlayın ve en az ayrıcalık ilkesini izleyerek aşamalı olarak kısıtlayın.
  • objectPermissions ve fieldPermissions değerlerini, fonksiyonlarınızın ihtiyaç duyduğu nesneler/alanlarla değiştirin.
  • permissionFlags, platform düzeyindeki yeteneklere erişimi kontrol eder. Minimumda tutun; yalnızca ihtiyacınız olanları ekleyin.
  • Çalışan bir örneği Hello World uygulamasında görün: packages/twenty-apps/hello-world/src/roles/function-role.ts.

Mantık fonksiyon yapılandırması ve giriş noktası

Her fonksiyon dosyası, bir işleyici ve isteğe bağlı tetikleyiciler içeren bir yapılandırmayı dışa aktarmak için defineLogicFunction() kullanır.
// src/app/createPostCard.logic-function.ts
import { defineLogicFunction } from 'twenty-sdk';
import type { DatabaseEventPayload, ObjectRecordCreateEvent, CronPayload, RoutePayload } from 'twenty-sdk';
import Twenty, { type Person } from '~/generated';

const handler = async (params: RoutePayload) => {
  const client = new Twenty(); // generated typed client
  const name = 'name' in params.queryStringParameters
    ? params.queryStringParameters.name ?? process.env.DEFAULT_RECIPIENT_NAME ?? 'Hello world'
    : 'Hello world';

  const result = await client.mutation({
    createPostCard: {
      __args: { data: { name } },
      id: true,
      name: true,
    },
  });
  return result;
};

export default defineLogicFunction({
  universalIdentifier: 'e56d363b-0bdc-4d8a-a393-6f0d1c75bdcf',
  name: 'create-new-post-card',
  timeoutSeconds: 2,
  handler,
  triggers: [
    // Public HTTP route trigger '/s/post-card/create'
    {
      universalIdentifier: 'c9f84c8d-b26d-40d1-95dd-4f834ae5a2c6',
      type: 'route',
      path: '/post-card/create',
      httpMethod: 'GET',
      isAuthRequired: false,
    },
    // Cron trigger (CRON pattern)
    // {
    //   universalIdentifier: 'dd802808-0695-49e1-98c9-d5c9e2704ce2',
    //   type: 'cron',
    //   pattern: '0 0 1 1 *',
    // },
    // Database event trigger
    // {
    //   universalIdentifier: '203f1df3-4a82-4d06-a001-b8cf22a31156',
    //   type: 'databaseEvent',
    //   eventName: 'person.updated',
    //   updatedFields: ['name'],
    // },
  ],
});
Yaygın tetikleyici türleri:
  • route: Fonksiyonunuzu bir HTTP yolu ve yöntemiyle /s/ uç noktası altında sunar:
örn. path: '/post-card/create', -> <APP_URL>/s/post-card/create üzerinden çağırın
  • cron: Bir CRON ifadesi kullanarak fonksiyonunuzu bir zamanlamayla çalıştırır.
  • databaseEvent: Çalışma alanı nesnesi yaşam döngüsü olaylarında çalışır. Olay işlemi updated olduğunda, dinlenecek belirli alanlar updatedFields dizisinde belirtilebilir. Tanımsız veya boş bırakılırsa, herhangi bir güncelleme fonksiyonu tetikler.
örn. person.updated
Notlar:
  • triggers dizisi isteğe bağlıdır. Tetikleyicisi olmayan fonksiyonlar, diğer fonksiyonlar tarafından çağrılan yardımcı fonksiyonlar olarak kullanılabilir.
  • Tek bir fonksiyonda birden çok tetikleyici türünü birleştirebilirsiniz.

Kurulum sonrası işlevler

Kurulum sonrası işlev, uygulamanız bir çalışma alanına yüklendikten sonra otomatik olarak çalışan bir mantık işlevidir. Bu, varsayılan verileri tohumlama, ilk kayıtları oluşturma veya çalışma alanı ayarlarını yapılandırma gibi tek seferlik kurulum görevleri için yararlıdır. create-twenty-app ile yeni bir uygulama iskeleti oluşturduğunuzda, src/logic-functions/post-install.ts konumunda sizin için bir kurulum sonrası işlevi oluşturulur:
// src/logic-functions/post-install.ts
import { defineLogicFunction } from 'twenty-sdk';

export const POST_INSTALL_UNIVERSAL_IDENTIFIER = '<generated-uuid>';

const handler = async (): Promise<void> => {
  console.log('Post install logic function executed successfully!');
};

export default defineLogicFunction({
  universalIdentifier: POST_INSTALL_UNIVERSAL_IDENTIFIER,
  name: 'post-install',
  description: 'Runs after installation to set up the application.',
  timeoutSeconds: 300,
  handler,
});
İşlev, application-config.ts içinde evrensel tanımlayıcısına başvurularak uygulamanıza bağlanır:
import { POST_INSTALL_UNIVERSAL_IDENTIFIER } from 'src/logic-functions/post-install';

export default defineApplication({
  // ...
  postInstallLogicFunctionUniversalIdentifier: POST_INSTALL_UNIVERSAL_IDENTIFIER,
});
Ayrıca kurulum sonrası işlevi istediğiniz zaman CLI kullanarak manuel olarak çalıştırabilirsiniz:
yarn twenty function:execute --postInstall
Önemli noktalar:
  • Kurulum sonrası işlevleri standart mantık işlevleridir — diğer herhangi bir işlev gibi defineLogicFunction() kullanırlar.
  • defineApplication() içindeki postInstallLogicFunctionUniversalIdentifier alanı isteğe bağlıdır. Atlanırsa, kurulumdan sonra hiçbir işlev çalıştırılmaz.
  • Varsayılan zaman aşımı, veri tohumlama gibi daha uzun kurulum görevlerine izin vermek için 300 saniye (5 dakika) olarak ayarlanmıştır.
  • Kurulum sonrası işlevlerin tetikleyicilere ihtiyacı yoktur — kurulum sırasında platform tarafından veya function:execute --postInstall aracılığıyla manuel olarak çağrılırlar.

Rota tetikleyicisi yükü

Kırıcı değişiklik (v1.16, Ocak 2026): Rota tetikleyicisi yük formatı değişti. v1.16’dan önce, sorgu parametreleri, yol parametreleri ve gövde doğrudan payload olarak gönderiliyordu. v1.16 itibarıyla, yapılandırılmış bir RoutePayload nesnesinin içine yerleştiriliyorlar.v1.16’dan önce:
const handler = async (params) => {
  const { param1, param2 } = params; // Direct access
};
v1.16’dan sonra:
const handler = async (event: RoutePayload) => {
  const { param1, param2 } = event.body; // Access via .body
  const { queryParam } = event.queryStringParameters;
  const { id } = event.pathParameters;
};
Mevcut fonksiyonları taşımak için: İşleyicinizi, parametreler nesnesinden doğrudan ayırmak yerine event.body, event.queryStringParameters veya event.pathParameters üzerinden ayrıştıracak şekilde güncelleyin.
Bir rota tetikleyicisi mantık fonksiyonunuzu çağırdığında, AWS HTTP API v2 formatını izleyen bir RoutePayload nesnesi alır. Türü twenty-sdk içinden içe aktarın:
import { defineLogicFunction, type RoutePayload } from 'twenty-sdk';

const handler = async (event: RoutePayload) => {
  // Access request data
  const { headers, queryStringParameters, pathParameters, body } = event;

  // HTTP method and path are available in requestContext
  const { method, path } = event.requestContext.http;

  return { message: 'Success' };
};
RoutePayload türünün yapısı şu şekildedir:
ÖzellikTürAçıklama
headersRecord<string, string | undefined>HTTP başlıkları (forwardedRequestHeaders içinde listelenenlerle sınırlı)
queryStringParametersRecord<string, string | undefined>Sorgu dizesi parametreleri (birden çok değer virgülle birleştirilir)
pathParametersRecord<string, string | undefined>Rota deseninden çıkarılan yol parametreleri (örn., /users/:id{ id: '123' })
gövdeobject | nullAyrıştırılmış istek gövdesi (JSON)
isBase64EncodedbooleanGövdenin base64 ile kodlanıp kodlanmadığı
requestContext.http.methodstringHTTP yöntemi (GET, POST, PUT, PATCH, DELETE)
requestContext.http.pathstringHam istek yolu

HTTP başlıklarını iletme

Varsayılan olarak, güvenlik nedenleriyle gelen isteklerden HTTP başlıkları mantık fonksiyonunuza aktarılmaz. Belirli başlıklara erişmek için bunları açıkça forwardedRequestHeaders dizisinde listeleyin:
export default defineLogicFunction({
  universalIdentifier: 'e56d363b-0bdc-4d8a-a393-6f0d1c75bdcf',
  name: 'webhook-handler',
  handler,
  triggers: [
    {
      universalIdentifier: 'c9f84c8d-b26d-40d1-95dd-4f834ae5a2c6',
      type: 'route',
      path: '/webhook',
      httpMethod: 'POST',
      isAuthRequired: false,
      forwardedRequestHeaders: ['x-webhook-signature', 'content-type'],
    },
  ],
});
Daha sonra işleyicinizde bu başlıklara erişebilirsiniz:
const handler = async (event: RoutePayload) => {
  const signature = event.headers['x-webhook-signature'];
  const contentType = event.headers['content-type'];

  // Validate webhook signature...
  return { received: true };
};
Başlık adları küçük harfe normalize edilir. Onlara küçük harfli anahtarlarla erişin (örneğin, event.headers['content-type']).
Yeni fonksiyonları iki şekilde oluşturabilirsiniz:
  • Şablondan: yarn twenty entity:add çalıştırın ve yeni bir mantık fonksiyonu ekleme seçeneğini seçin. Bu, bir işleyici ve yapılandırma içeren bir başlangıç dosyası oluşturur.
  • Manuel: Yeni bir *.logic-function.ts dosyası oluşturun ve aynı deseni izleyerek defineLogicFunction() kullanın.

Bir mantık işlevini araç olarak işaretleme

Mantık işlevleri, yapay zeka ajanları ve iş akışları için araçlar olarak sunulabilir. Bir işlev bir araç olarak işaretlendiğinde, Twenty’nin yapay zeka özellikleri tarafından keşfedilebilir hâle gelir ve iş akışı otomasyonlarında bir adım olarak seçilebilir. Bir mantık işlevini bir araç olarak işaretlemek için isTool: true olarak ayarlayın ve beklenen giriş parametrelerini açıklayan bir toolInputSchemaJSON Şeması kullanarak sağlayın:
// src/logic-functions/enrich-company.logic-function.ts
import { defineLogicFunction } from 'twenty-sdk';
import Twenty from '~/generated';

const handler = async (params: { companyName: string; domain?: string }) => {
  const client = new Twenty();

  const result = await client.mutation({
    createTask: {
      __args: {
        data: {
          title: `Enrich data for ${params.companyName}`,
          body: `Domain: ${params.domain ?? 'unknown'}`,
        },
      },
      id: true,
    },
  });

  return { taskId: result.createTask.id };
};

export default defineLogicFunction({
  universalIdentifier: 'f47ac10b-58cc-4372-a567-0e02b2c3d479',
  name: 'enrich-company',
  description: 'Enrich a company record with external data',
  timeoutSeconds: 10,
  handler,
  isTool: true,
  toolInputSchema: {
    type: 'object',
    properties: {
      companyName: {
        type: 'string',
        description: 'The name of the company to enrich',
      },
      domain: {
        type: 'string',
        description: 'The company website domain (optional)',
      },
    },
    required: ['companyName'],
  },
});
Önemli noktalar:
  • isTool (boolean, varsayılan: false): true olarak ayarlandığında, işlev bir araç olarak kaydedilir ve AI ajanları ile iş akışı otomasyonları tarafından kullanılabilir hale gelir.
  • toolInputSchema (object, isteğe bağlı): İşlevinizin kabul ettiği parametreleri tanımlayan bir JSON Schema nesnesi. AI ajanları, aracın hangi girdileri beklediğini anlamak ve çağrıları doğrulamak için bu şemayı kullanır. Atlanırsa, şema varsayılan olarak { type: 'object', properties: {} } olur (parametre yok).
  • isTool: false (veya ayarlanmamış) olan işlevler araç olarak sunulmaz. Yine de doğrudan yürütülebilir veya diğer işlevler tarafından çağrılabilirler, ancak araç keşfinde görünmezler.
  • Araç adlandırma: Bir araç olarak sunulduğunda, işlev adı otomatik olarak logic_function_<name> biçimine dönüştürülür (küçük harfe çevrilir, alfasayısal olmayan karakterler alt çizgi ile değiştirilir). Örneğin, enrich-company logic_function_enrich_company haline gelir.
  • isTool özelliğini tetikleyicilerle birleştirebilirsiniz — bir işlev aynı anda hem bir araç (AI ajanları tarafından çağrılabilir) olabilir hem de olaylar tarafından tetiklenebilir (cron, veritabanı olayları, routes).
İyi bir description yazın. AI ajanları, aracı ne zaman kullanacaklarına karar vermek için işlevin description alanına güvenir. Aracın ne yaptığını ve ne zaman çağrılması gerektiğini açıkça belirtin.

Ön uç bileşenleri

Ön uç bileşenleri, Twenty’nin kullanıcı arayüzünde görüntülenen özel React bileşenleri oluşturmanıza olanak tanır. Yerleşik doğrulamayla bileşenleri tanımlamak için defineFrontComponent() kullanın:
// src/my-widget.front-component.tsx
import { defineFrontComponent } from 'twenty-sdk';

const MyWidget = () => {
  return (
    <div style={{ padding: '20px', fontFamily: 'sans-serif' }}>
      <h1>My Custom Widget</h1>
      <p>This is a custom front component for Twenty.</p>
    </div>
  );
};

export default defineFrontComponent({
  universalIdentifier: 'a1b2c3d4-e5f6-7890-abcd-ef1234567890',
  name: 'my-widget',
  description: 'A custom widget component',
  component: MyWidget,
});
Önemli noktalar:
  • Ön uç bileşenleri, Twenty içinde yalıtılmış bağlamlarda görüntülenen React bileşenleridir.
  • Otomatik algılama için *.front-component.tsx dosya soneğini kullanın.
  • component alanı, React bileşeninize referans verir.
  • Bileşenler, yarn twenty app:dev sırasında otomatik olarak oluşturulur ve senkronize edilir.
Yeni ön uç bileşenlerini iki şekilde oluşturabilirsiniz:
  • Şablondan: yarn twenty entity:add çalıştırın ve yeni bir ön uç bileşeni ekleme seçeneğini seçin.
  • Manuel: Yeni bir *.front-component.tsx dosyası oluşturun ve defineFrontComponent() kullanın.

Beceriler

Skills define reusable instructions and capabilities that AI agents can use within your workspace. Use defineSkill() to define skills with built-in validation:
// src/skills/example-skill.ts
import { defineSkill } from 'twenty-sdk';

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`,
});
Önemli noktalar:
  • name is a unique identifier string for the skill (kebab-case recommended).
  • label is the human-readable display name shown in the UI.
  • content contains 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.
You can create new skills in two ways:
  • Scaffolded: Run yarn twenty entity:add and choose the option to add a new skill.
  • Manual: Create a new file and use defineSkill(), following the same pattern.

Oluşturulmuş türlendirilmiş istemci

Tipli istemci, yarn twenty app:dev tarafından otomatik olarak oluşturulur ve çalışma alanı şemanıza göre node_modules/twenty-sdk/generated içine kaydedilir. Fonksiyonlarınızda kullanın:
import Twenty from '~/generated';

const client = new Twenty();
const { me } = await client.query({ me: { id: true, displayName: true } });
Nesneleriniz veya alanlarınız değiştiğinde, istemci yarn twenty app:dev tarafından otomatik olarak yeniden oluşturulur.

Mantık fonksiyonlarında çalışma zamanı kimlik bilgileri

Fonksiyonunuz Twenty üzerinde çalıştığında, platform kodunuz yürütülmeden önce kimlik bilgilerini ortam değişkenleri olarak enjekte eder:
  • TWENTY_API_URL: Uygulamanızın hedeflediği Twenty API’nin temel URL’si.
  • TWENTY_API_KEY: Uygulamanızın varsayılan fonksiyon rolü kapsamına sahip kısa ömürlü anahtar.
Notlar:
  • Oluşturulan istemciye URL veya API anahtarı geçirmeniz gerekmez. Çalışma zamanında TWENTY_API_URL ve TWENTY_API_KEY değerlerini process.env üzerinden okur.
  • API anahtarının izinleri, application-config.ts içinde defaultRoleUniversalIdentifier aracılığıyla referans verilen role göre belirlenir. Bu, uygulamanızın mantık fonksiyonları tarafından kullanılan varsayılan roldür.
  • Uygulamalar, en az ayrıcalık ilkesini izlemek için roller tanımlayabilir. Yalnızca fonksiyonlarınızın ihtiyaç duyduğu izinleri verin ve ardından defaultRoleUniversalIdentifier değerini o rolün evrensel tanımlayıcısına yönlendirin.

Hello World örneği

Nesneleri, mantık fonksiyonlarını, ön uç bileşenlerini ve birden çok tetikleyiciyi gösteren minimal, uçtan uca bir örneği buradan inceleyin:

Manuel kurulum (scaffolder olmadan)

En iyi başlangıç deneyimi için create-twenty-app kullanmanızı önersek de, bir projeyi manuel olarak da kurabilirsiniz. CLI’yi global olarak kurmayın. Bunun yerine twenty-sdk’yi yerel bir bağımlılık olarak ekleyin ve package.json içinde tek bir betik tanımlayın:
yarn add -D twenty-sdk
Ardından bir twenty betiği ekleyin:
{
  "scripts": {
    "twenty": "twenty"
  }
}
Artık tüm komutları yarn twenty <command> üzerinden çalıştırabilirsiniz; örn. yarn twenty app:dev, yarn twenty help vb.

Sorun Giderme

  • Kimlik doğrulama hataları: yarn twenty auth:login çalıştırın ve API anahtarınızın gerekli izinlere sahip olduğundan emin olun.
  • Sunucuya bağlanılamıyor: API URL’sini ve Twenty sunucusunun erişilebilir olduğunu doğrulayın.
  • Türler veya istemci eksik/eski: yarn twenty app:dev komutunu yeniden çalıştırın — tip tanımlı istemciyi otomatik olarak oluşturur.
  • Geliştirme modu eşitlenmiyor: yarn twenty app:dev’in çalıştığından ve değişikliklerin ortamınız tarafından yok sayılmadığından emin olun.
Discord Yardım Kanalı: https://discord.com/channels/1130383047699738754/1130386664812982322