الانتقال إلى المحتوى الرئيسي
التطبيقات حاليًا في مرحلة الاختبار الألفا. الميزة تعمل لكنها لا تزال قيد التطور.

ما هي التطبيقات؟

تتيح لك التطبيقات إنشاء وإدارة تخصيصات Twenty ككود. بدلًا من تكوين كل شيء عبر واجهة المستخدم، تُعرِّف نموذج بياناتك ووظائف المنطق في الكود — مما يجعل البناء والصيانة والنشر إلى مساحات عمل متعددة أسرع. ما الذي يمكنك فعله اليوم:
  • عرِّف كائنات وحقولًا مخصصة على شكل كود (نموذج بيانات مُدار)
  • أنشئ وظائف منطقية مع مشغلات مخصصة
  • حدد المهارات لوكلاء الذكاء الاصطناعي
  • انشر التطبيق نفسه عبر مساحات عمل متعددة

المتطلبات الأساسية

البدء

أنشئ تطبيقًا جديدًا باستخدام المُهيئ الرسمي، ثم قم بالمصادقة وابدأ التطوير:
# إنشاء تطبيق جديد (يتضمن جميع الأمثلة افتراضيًا)
npx create-twenty-app@latest my-twenty-app
cd my-twenty-app

# إذا كنت لا تستخدم yarn@4
corepack enable
yarn install

# قم بالمصادقة باستخدام مفتاح واجهة برمجة التطبيقات الخاص بك (سيُطلب منك ذلك)
yarn twenty auth:login

# ابدأ وضع التطوير: يُزامن التغييرات المحلية تلقائيًا مع مساحة العمل الخاصة بك
yarn twenty app:dev
يدعم المُنشئ ثلاثة أوضاع للتحكم في ملفات الأمثلة التي سيتم تضمينها:
# الافتراضي (شامل): جميع الأمثلة (كائن، حقل، دالة منطقية، مكوّن الواجهة الأمامية، عرض، عنصر قائمة التنقل، مهارة)
npx create-twenty-app@latest my-app

# الأدنى: الملفات الأساسية فقط (application-config.ts و default-role.ts)
npx create-twenty-app@latest my-app --minimal

# التفاعلي: اختر الأمثلة التي تريد تضمينها
npx create-twenty-app@latest my-app --interactive
من هنا يمكنك:
# 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
راجع أيضًا: صفحات مرجع CLI لـ create-twenty-app وtwenty-sdk CLI.

هيكل المشروع (مُنشأ بالقالب)

عند تشغيل npx create-twenty-app@latest my-twenty-app، يقوم المُهيئ بما يلي:
  • ينسخ تطبيقًا أساسيًا مصغّرًا إلى my-twenty-app/
  • يضيف اعتمادًا محليًا twenty-sdk وتهيئة Yarn 4
  • ينشئ ملفات ضبط ونصوصًا مرتبطة بـ twenty CLI
  • يُنشئ الملفات الأساسية (تهيئة التطبيق، دور الدالة الافتراضي، دالة ما بعد التثبيت) بالإضافة إلى ملفات الأمثلة بحسب وضع الإنشاء
يبدو التطبيق المُنشأ حديثًا باستخدام الوضع الافتراضي --exhaustive كما يلي:
my-twenty-app/
  package.json
  yarn.lock
  .gitignore
  .nvmrc
  .yarnrc.yml
  .yarn/
    install-state.gz
  eslint.config.mjs
  tsconfig.json
  README.md
  public/                           # مجلد الأصول العامة (صور، خطوط، إلخ)
  src/
  ├── application-config.ts           # مطلوب - إعدادات التطبيق الرئيسية
  ├── roles/
  │   └── default-role.ts               # الدور الافتراضي للدوال المنطقية
  ├── objects/
  │   └── example-object.ts             # تعريف كائن مخصص — مثال
  ├── fields/
  │   └── example-field.ts              # تعريف حقل مستقل — مثال
  ├── logic-functions/
  │   ├── hello-world.ts                # دالة منطقية — مثال
  │   └── post-install.ts               # دالة منطقية لما بعد التثبيت
  ├── front-components/
  │   └── hello-world.tsx               # مكوّن واجهة أمامية — مثال
  ├── views/
  │   └── example-view.ts                # تعريف عرض محفوظ — مثال
  ├── navigation-menu-items/
  │   └── example-navigation-menu-item.ts # رابط تنقّل في الشريط الجانبي — مثال
  └── skills/
      └── example-skill.ts                # تعريف مهارة لوكيل الذكاء الاصطناعي — مثال
مع --minimal، سيتم إنشاء الملفات الأساسية فقط (application-config.ts وroles/default-role.ts وlogic-functions/post-install.ts). مع --interactive، تختار ملفات الأمثلة التي تريد تضمينها. بشكل عام:
  • package.json: يصرّح باسم التطبيق والإصدار والمحرّكات (Node 24+، Yarn 4)، ويضيف twenty-sdk بالإضافة إلى نص برمجي twenty يفوِّض إلى twenty CLI المحلي. شغِّل yarn twenty help لعرض جميع الأوامر المتاحة.
  • .gitignore: يتجاهل العناصر الشائعة مثل node_modules و.yarn وgenerated/ (عميل مضبوط الأنواع) وdist/ وbuild/ ومجلدات التغطية وملفات السجلات وملفات .env*.
  • yarn.lock، .yarnrc.yml، .yarn/: تقوم بقفل وتكوين حزمة أدوات Yarn 4 المستخدمة في المشروع.
  • .nvmrc: يثبّت إصدار Node.js المتوقع للمشروع.
  • eslint.config.mjs وtsconfig.json: يقدّمان إعدادات الفحص والتهيئة لـ TypeScript لمصادر TypeScript في تطبيقك.
  • README.md: ملف README قصير في جذر التطبيق يتضمن تعليمات أساسية.
  • public/: مجلد لتخزين الأصول العامة (صور، خطوط، ملفات ثابتة) التي سيتم تقديمها مع تطبيقك. الملفات الموضوعة هنا تُرفع أثناء المزامنة وتكون متاحة أثناء وقت التشغيل.
  • src/: المكان الرئيسي حيث تعرّف تطبيقك ككود

اكتشاف الكيانات

يكتشف SDK الكيانات عبر تحليل ملفات TypeScript الخاصة بك بحثًا عن استدعاءات export default define<Entity>({...}). يحتوي كل نوع كيان على دالة مساعدة مقابلة يتم تصديرها من twenty-sdk:
دالة مساعدةنوع الكيان
defineObject()تعريفات كائنات مخصصة
defineLogicFunction()تعريفات الوظائف المنطقية
defineFrontComponent()Front component definitions
defineRole()تعريفات الأدوار
defineField()امتدادات الحقول للكائنات الموجودة
defineView()تعريفات العروض المحفوظة
defineNavigationMenuItem()تعريفات عناصر قائمة التنقل
defineSkill()تعريفات مهارات وكيل الذكاء الاصطناعي
تسمية الملفات مرنة. يعتمد اكتشاف الكيانات على بنية الشجرة المجردة (AST) — إذ يقوم SDK بفحص ملفات المصدر لديك بحثًا عن النمط export default define<Entity>({...}). يمكنك تنظيم ملفاتك ومجلداتك كيفما تشاء. التجميع حسب نوع الكيان (مثلًا، logic-functions/ وroles/) هو مجرد عرف لتنظيم الشيفرة، وليس مطلبًا إلزاميًا.
مثال على كيان تم اكتشافه:
// 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
});
ستضيف الأوامر اللاحقة مزيدًا من الملفات والمجلدات:
  • yarn twenty app:dev سيولّد تلقائيًا عميل API مضبوط الأنواع في node_modules/twenty-sdk/generated (عميل Twenty مضبوط الأنواع + أنواع مساحة العمل).
  • yarn twenty entity:add سيضيف ملفات تعريف الكيانات ضمن src/ لكائناتك المخصّصة، والوظائف، ومكوّنات الواجهة الأمامية، والأدوار، والمهارات، وغير ذلك.

المصادقة

في المرة الأولى التي تشغّل فيها yarn twenty auth:login، سيُطلب منك إدخال:
  • عنوان URL لواجهة برمجة التطبيقات (الافتراضي http://localhost:3000 أو ملف تعريف مساحة العمل الحالية لديك)
  • مفتاح واجهة برمجة التطبيقات
تُخزَّن بيانات اعتمادك لكل مستخدم في ~/.twenty/config.json. You can maintain multiple profiles and switch between them.

Managing workspaces

# تسجيل الدخول تفاعليًا (مُوصى به)
yarn twenty auth:login

# تسجيل الدخول إلى ملف تعريف لمساحة عمل محددة
yarn twenty auth:login --workspace my-custom-workspace

# عرض جميع مساحات العمل المُكوَّنة
yarn twenty auth:list

# تبديل مساحة العمل الافتراضية (تفاعليًا)
yarn twenty auth:switch

# التبديل إلى مساحة عمل محددة
yarn twenty auth:switch production

# التحقق من حالة المصادقة الحالية
yarn twenty auth:status
بمجرد أن تقوم بالتبديل بين مساحات العمل باستخدام yarn twenty auth:switch، ستستخدم جميع الأوامر اللاحقة تلك المساحة افتراضيًا. You can still override it temporarily with --workspace <name>.

استخدم موارد SDK (الأنواع والتكوين)

يوفّر twenty-sdk كتلَ بناءٍ مضبوطة الأنواع ودوال مساعدة تستخدمها داخل تطبيقك. فيما يلي الأجزاء الأساسية التي ستتعامل معها غالبًا.

دوال مساعدة

يوفّر SDK دوالًا مساعدة لتعريف كيانات تطبيقك. كما هو موضح في اكتشاف الكيانات، يجب استخدام export default define<Entity>({...}) كي يتم اكتشاف كياناتك:
دالةالغرض
defineApplication()تهيئة بيانات التعريف للتطبيق (مطلوب، واحد لكل تطبيق)
defineObject()تعريف كائنات مخصصة مع حقول
defineLogicFunction()تعريف وظائف منطقية مع معالجات
defineFrontComponent()عرِّف مكوّنات أمامية لواجهة مستخدم مخصّصة
defineRole()تهيئة صلاحيات الدور والوصول إلى الكائنات
defineField()وسّع الكائنات الموجودة بحقول إضافية
defineView()تعريف العروض المحفوظة للكائنات
defineNavigationMenuItem()تعريف روابط التنقل في الشريط الجانبي
defineSkill()عرّف مهارات وكيل الذكاء الاصطناعي
تتحقق هذه الدوال من تكوينك وقت البناء وتوفّر إكمالًا تلقائيًا في بيئة التطوير وأمان الأنواع.

تعريف الكائنات

تصف الكائنات المخصصة كلًا من المخطط والسلوك للسجلات في مساحة عملك. استخدم defineObject() لتعريف كائنات مع تحقق مدمج:
// 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,
    },
  ],
});
النقاط الرئيسية:
  • استخدم defineObject() للحصول على تحقق مدمج ودعم أفضل من IDE.
  • universalIdentifier يجب أن يكون فريدًا وثابتًا عبر عمليات النشر.
  • يتطلب كل حقل name وtype وlabel ومعرّف universalIdentifier ثابتًا خاصًا به.
  • المصفوفة fields اختيارية — يمكنك تعريف كائنات بدون حقول مخصصة.
  • يمكنك إنشاء كائنات جديدة باستخدام yarn twenty entity:add، والذي يرشدك خلال التسمية والحقول والعلاقات.
يتم إنشاء الحقول الأساسية تلقائيًا. عند تعريف كائن مخصص، يضيف Twenty تلقائيًا حقولًا قياسية مثل id وname وcreatedAt وupdatedAt وcreatedBy وupdatedBy وdeletedAt. لا تحتاج إلى تعريف هذه في مصفوفة fields — أضف فقط حقولك المخصصة. يمكنك تجاوز الحقول الافتراضية من خلال تعريف حقل بالاسم نفسه في مصفوفة fields الخاصة بك، لكن هذا غير مستحسن.

تكوين التطبيق (application-config.ts)

كل تطبيق لديه ملف واحد application-config.ts يصف:
  • هوية التطبيق: المعرفات، اسم العرض، والوصف.
  • كيفية تشغيل وظائفه: الدور الذي تستخدمه للأذونات.
  • متغيرات (اختياري): أزواج مفتاح-قيمة تُعرض لوظائفك كمتغيرات بيئة.
  • (Optional) post-install function: a logic function that runs after the app is installed.
Use defineApplication() to define your application configuration:
// 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,
});
الملاحظات:
  • حقول universalIdentifier هي معرّفات حتمية تخصك؛ أنشئها مرة واحدة واحتفظ بها ثابتة عبر عمليات المزامنة.
  • applicationVariables تصبح متغيرات بيئة لوظائفك (على سبيل المثال، DEFAULT_RECIPIENT_NAME متاح كـ process.env.DEFAULT_RECIPIENT_NAME).
  • defaultRoleUniversalIdentifier يجب أن يطابق ملف الدور (انظر أدناه).
  • postInstallLogicFunctionUniversalIdentifier (optional) points to a logic function that runs automatically after the app is installed. See Post-install functions.

الأدوار والصلاحيات

يمكن للتطبيقات تعريف أدوار تُغلّف الصلاحيات على كائنات وإجراءات مساحة العمل لديك. يعين الحقل defaultRoleUniversalIdentifier في application-config.ts الدور الافتراضي الذي تستخدمه وظائف المنطق في تطبيقك.
  • مفتاح واجهة البرمجة في وقت التشغيل المحقون باسم TWENTY_API_KEY مستمد من دور الوظيفة الافتراضي هذا.
  • سيُقيَّد العميل مضبوط الأنواع بالأذونات الممنوحة لذلك الدور.
  • اتبع مبدأ أقل الامتياز: أنشئ دورًا مخصصًا بالأذونات التي تحتاجها وظائفك فقط، ثم أشِر إلى معرّفه الشامل.
الدور الافتراضي للوظيفة (*.role.ts)
عند توليد تطبيق جديد بالقالب، ينشئ CLI أيضًا ملف دور افتراضي. استخدم defineRole() لتعريف أدوار مع تحقق مدمج:
// 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],
});
يُشار بعد ذلك إلى universalIdentifier لهذا الدور في application-config.ts باسم defaultRoleUniversalIdentifier. بعبارة أخرى:
  • \*.role.ts يحدد ما يمكن أن يفعله الدور الافتراضي للوظيفة.
  • application-config.ts يشير إلى ذلك الدور بحيث ترث وظائفك أذوناته.
الملاحظات:
  • ابدأ من الدور المُنشأ بالقالب، ثم قيّده تدريجيًا باتباع مبدأ أقل الامتياز.
  • استبدل objectPermissions وfieldPermissions بالكائنات/الحقول التي تحتاجها وظائفك.
  • permissionFlags تتحكم في الوصول إلى القدرات على مستوى المنصة. اجعلها في الحد الأدنى؛ أضف فقط ما تحتاجه.
  • اطّلع على مثال عملي في تطبيق Hello World: packages/twenty-apps/hello-world/src/roles/function-role.ts.

تكوين الوظيفة المنطقية ونقطة الدخول

كل ملف وظيفة يستخدم defineLogicFunction() لتصدير تكوين مع معالج ومشغّلات اختيارية.
// 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'],
    // },
  ],
});
أنواع المشغلات الشائعة:
  • route: يعرِض وظيفتك على مسار وطريقة HTTP تحت نقطة النهاية /s/:
مثال: path: '/post-card/create', -> الاستدعاء على <APP_URL>/s/post-card/create
  • cron: يشغّل وظيفتك على جدول باستخدام تعبير CRON.
  • databaseEvent: يعمل على أحداث دورة حياة كائنات مساحة العمل. عندما تكون عملية الحدث هي updated، يمكن تحديد الحقول المحددة المراد الاستماع إليها في مصفوفة updatedFields. إذا تُركت غير معرّفة أو فارغة، فسيؤدي أي تحديث إلى تشغيل الدالة.
مثال: person.updated
الملاحظات:
  • المصفوفة triggers اختيارية. يمكن استخدام الوظائف بدون مشغلات كوظائف مساعدة تُستدعى بواسطة وظائف أخرى.
  • يمكنك مزج أنواع متعددة من المشغلات في وظيفة واحدة.

Post-install functions

A post-install function is a logic function that runs automatically after your app is installed on a workspace. This is useful for one-time setup tasks such as seeding default data, creating initial records, or configuring workspace settings. عند إنشاء هيكل تطبيق جديد باستخدام create-twenty-app، يتم إنشاء دالة ما بعد التثبيت لك في src/logic-functions/post-install.ts:
// 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,
});
يتم ربط الدالة بتطبيقك من خلال الإشارة إلى المعرِّف العالمي الخاص بها في application-config.ts:
import { POST_INSTALL_UNIVERSAL_IDENTIFIER } from 'src/logic-functions/post-install';

export default defineApplication({
  // ...
  postInstallLogicFunctionUniversalIdentifier: POST_INSTALL_UNIVERSAL_IDENTIFIER,
});
يمكنك أيضًا تنفيذ دالة ما بعد التثبيت يدويًا في أي وقت باستخدام CLI:
yarn twenty function:execute --postInstall
النقاط الرئيسية:
  • دوال ما بعد التثبيت هي دوال منطقية قياسية — فهي تستخدم defineLogicFunction() مثل أي دالة أخرى.
  • حقل postInstallLogicFunctionUniversalIdentifier في defineApplication() اختياري. إذا تم تجاهله، لن يتم تشغيل أي دالة بعد التثبيت.
  • تم تعيين مهلة افتراضية إلى 300 ثانية (5 دقائق) للسماح بمهام الإعداد الأطول مثل تهيئة البيانات.
  • لا تحتاج دوال ما بعد التثبيت إلى مُشغِّلات — حيث يستدعيها النظام الأساسي أثناء التثبيت أو يدويًا عبر function:execute --postInstall.

حمولة مشغل المسار

تغيير غير متوافق (v1.16، يناير 2026): لقد تغير تنسيق حمولة مشغل المسار. قبل v1.16، كانت معلمات الاستعلام، ومعلمات المسار، وجسم الطلب تُرسل مباشرةً كحمولة. بدءًا من v1.16، أصبحت متداخلة داخل كائن منظَّم RoutePayload.قبل v1.16:
const handler = async (params) => {
  const { param1, param2 } = params; // Direct access
};
بعد v1.16:
const handler = async (event: RoutePayload) => {
  const { param1, param2 } = event.body; // Access via .body
  const { queryParam } = event.queryStringParameters;
  const { id } = event.pathParameters;
};
لترحيل الدوال الحالية: حدّث المعالج لديك لفكّ البنية من event.body أو event.queryStringParameters أو event.pathParameters بدلاً من القراءة مباشرةً من كائن params.
عندما يستدعي مشغّل المسار وظيفتك المنطقية، يتلقى كائنًا من النوع RoutePayload يتبع تنسيق AWS HTTP API v2. استورد النوع من twenty-sdk:
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 على البنية التالية:
الخاصيةالنوعالوصف
headersRecord<string, string | undefined>رؤوس HTTP (فقط تلك المدرجة في forwardedRequestHeaders)
queryStringParametersRecord<string, string | undefined>معلمات سلسلة الاستعلام (تُضمّ القيم المتعددة باستخدام فواصل)
pathParametersRecord<string, string | undefined>معلمات المسار المستخرجة من نمط المسار (على سبيل المثال، /users/:id{ id: '123' })
المحتوىobject | nullجسم الطلب المُحلَّل (JSON)
isBase64Encodedقيمة منطقيةما إذا كان جسم الطلب مُرمَّزًا بترميز base64
requestContext.http.methodstringطريقة HTTP (GET, POST, PUT, PATCH, DELETE)
requestContext.http.pathstringالمسار الخام للطلب

تمرير رؤوس HTTP

افتراضيًا، لا تُمرَّر رؤوس HTTP من الطلبات الواردة إلى وظيفتك المنطقية لأسباب أمنية. للوصول إلى رؤوس محددة، قم بإدراجها صراحةً في مصفوفة forwardedRequestHeaders:
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'],
    },
  ],
});
في المعالج الخاص بك، يمكنك حينها الوصول إلى هذه الرؤوس:
const handler = async (event: RoutePayload) => {
  const signature = event.headers['x-webhook-signature'];
  const contentType = event.headers['content-type'];

  // Validate webhook signature...
  return { received: true };
};
تُحوَّل أسماء الرؤوس إلى أحرف صغيرة. يمكنك الوصول إليها باستخدام مفاتيح بأحرف صغيرة (على سبيل المثال، event.headers['content-type']).
يمكنك إنشاء وظائف جديدة بطريقتين:
  • مُنشأ بالقالب: شغّل yarn twenty entity:add واختر خيار إضافة وظيفة منطقية جديدة. يُولّد هذا ملفًا مبدئيًا مع معالج وتكوين.
  • يدوي: أنشئ ملفًا جديدًا *.logic-function.ts واستخدم defineLogicFunction() مع اتباع النمط نفسه.

تمييز دالة منطقية كأداة

يمكن إتاحة الدوال المنطقية بوصفها أدوات لوكلاء الذكاء الاصطناعي وسير العمل. عندما يتم تمييز دالة كأداة، تصبح قابلة للاكتشاف بواسطة ميزات الذكاء الاصطناعي الخاصة بـ Twenty ويمكن اختيارها كخطوة في أتمتة سير العمل. لتمييز دالة منطقية كأداة، عيّن isTool: true وقدّم toolInputSchema يصف معاملات الإدخال المتوقعة باستخدام مخطط JSON:
// 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'],
  },
});
النقاط الرئيسية:
  • isTool (boolean, الافتراضي: false): عند ضبطه على true، يتم تسجيل الدالة كأداة وتصبح متاحة لوكلاء الذكاء الاصطناعي ولأتمتة سير العمل.
  • toolInputSchema (object, اختياري): كائن JSON Schema يصف المعلمات التي تقبلها دالتك. يستخدم وكلاء الذكاء الاصطناعي هذا المخطط لفهم المدخلات التي تتوقعها الأداة وللتحقق من صحة الاستدعاءات. إذا تم إغفاله، فالقيمة الافتراضية للمخطط هي { type: 'object', properties: {} } (من دون معلمات).
  • الدوال التي لديها isTool: false (أو غير معيَّنة) غير معروضة كأدوات. لا يزال بالإمكان تنفيذها مباشرةً أو استدعاؤها بواسطة دوال أخرى، لكنها لن تظهر في اكتشاف الأدوات.
  • تسمية الأداة: عند كشفها كأداة، يتم تطبيع اسم الدالة تلقائيًا إلى logic_function_<name> (تحويله إلى أحرف صغيرة، واستبدال المحارف غير الأبجدية الرقمية بشرطات سفلية). على سبيل المثال، enrich-company تصبح logic_function_enrich_company.
  • يمكنك دمج isTool مع المشغِّلات — إذ يمكن للدالة أن تكون أداة (قابلة للاستدعاء من قِبل وكلاء الذكاء الاصطناعي) وأن تُشغَّل بواسطة أحداث (cron، وأحداث قاعدة البيانات، والمسارات) في الوقت نفسه.
اكتب description جيدًا. يعتمد وكلاء الذكاء الاصطناعي على حقل description الخاص بالدالة لتحديد وقت استخدام الأداة. كن محددًا بشأن ما تفعله الأداة ومتى ينبغي استدعاؤها.

المكوّنات الأمامية

تتيح لك المكوّنات الأمامية إنشاء مكوّنات React مخصّصة تُعرَض داخل واجهة مستخدم Twenty. استخدم defineFrontComponent() لتعريف مكوّنات مع تحقّق مدمج:
// 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,
});
النقاط الرئيسية:
  • المكوّنات الأمامية هي مكوّنات React تُعرَض ضمن سياقات معزولة داخل Twenty.
  • استخدم لاحقة الملف *.front-component.tsx للاكتشاف التلقائي.
  • يشير الحقل component إلى مكوّن React الخاص بك.
  • يتم بناء المكوّنات ومزامنتها تلقائيًا أثناء yarn twenty app:dev.
يمكنك إنشاء مكوّنات أمامية جديدة بطريقتين:
  • مُنشأ بالقالب: شغّل yarn twenty entity:add واختر خيار إضافة مكوّن أمامي جديد.
  • يدوي: أنشئ ملفًا جديدًا *.front-component.tsx واستخدم defineFrontComponent().

المهارات

تُحدِّد المهارات تعليمات وإمكانات قابلة لإعادة الاستخدام يمكن لوكلاء الذكاء الاصطناعي استخدامها داخل مساحة العمل لديك. استخدم defineSkill() لتعريف مهارات مع تحقّق مدمج:
// src/skills/example-skill.ts
import { defineSkill } from 'twenty-sdk';

export default defineSkill({
  universalIdentifier: 'a1b2c3d4-e5f6-7890-abcd-ef1234567890',
  name: 'sales-outreach',
  label: 'التواصل البيعي',
  description: 'يرشد وكيل الذكاء الاصطناعي خلال عملية منظّمة للتواصل البيعي',
  icon: 'IconBrain',
  content: `أنت مساعد للتواصل البيعي. عند التواصل مع عميل محتمل:
1. ابحث عن الشركة وآخر الأخبار
2. حدِّد دور العميل المحتمل ونقاط الألم المرجّحة
3. صِغ رسالة مخصّصة تشير إلى تفاصيل محدّدة
4. حافظ على نبرة احترافية ولكن حوارية`,
});
النقاط الرئيسية:
  • name هي سلسلة معرّف فريدة للمهارة (يُنصَح باستخدام kebab-case).
  • label هو اسم العرض المقروء للبشر الظاهر في واجهة المستخدم.
  • content يحتوي على تعليمات المهارة — وهو النص الذي يستخدمه وكيل الذكاء الاصطناعي.
  • icon (اختياري) يحدّد الأيقونة المعروضة في واجهة المستخدم.
  • description (اختياري) يوفّر سياقًا إضافيًا حول غرض المهارة.
يمكنك إنشاء مهارات جديدة بطريقتين:
  • مُنشأ بالقالب: شغِّل yarn twenty entity:add واختر خيار إضافة مهارة جديدة.
  • يدوي: أنشئ ملفًا جديدًا واستخدم defineSkill() مع اتباع النمط نفسه.

عميل مُولَّد مضبوط الأنواع

يُولَّد العميل مضبوط الأنواع تلقائيًا بواسطة yarn twenty app:dev ويُخزَّن في node_modules/twenty-sdk/generated استنادًا إلى مخطط مساحة العمل لديك. استخدمه في وظائفك:
import Twenty from '~/generated';

const client = new Twenty();
const { me } = await client.query({ me: { id: true, displayName: true } });
يُعاد توليد العميل تلقائيًا بواسطة yarn twenty app:dev كلما تغيّرت كائناتك أو حقولك.

بيانات الاعتماد وقت التشغيل في الوظائف المنطقية

عندما تعمل وظيفتك على Twenty، يقوم النظام الأساسي بحقن بيانات الاعتماد كمتغيرات بيئة قبل تنفيذ كودك:
  • TWENTY_API_URL: عنوان URL الأساسي لواجهة Twenty البرمجية التي يستهدفها تطبيقك.
  • TWENTY_API_KEY: مفتاح قصير العمر ذو نطاق يقتصر على الدور الافتراضي لوظيفة تطبيقك.
الملاحظات:
  • لا تحتاج إلى تمرير عنوان URL أو مفتاح واجهة برمجة التطبيقات إلى العميل المُولَّد. يقوم بقراءة TWENTY_API_URL وTWENTY_API_KEY من process.env وقت التشغيل.
  • تُحدَّد أذونات مفتاح واجهة برمجة التطبيقات بواسطة الدور المشار إليه في application-config.ts عبر defaultRoleUniversalIdentifier. هذا هو الدور الافتراضي الذي تستخدمه الوظائف المنطقية في تطبيقك.
  • يمكن للتطبيقات تعريف أدوار لاتباع مبدأ أقل الامتياز. امنح فقط الأذونات التي تحتاجها وظائفك، ثم وجّه defaultRoleUniversalIdentifier إلى المعرّف الشامل لذلك الدور.

مثال Hello World

استكشف مثالًا بسيطًا شاملًا من البداية إلى النهاية يوضح الكائنات والوظائف المنطقية والمكوّنات الأمامية ومشغّلات متعددة هنا:

إعداد يدوي (بدون المهيئ)

بينما نوصي باستخدام create-twenty-app للحصول على أفضل تجربة للبدء، يمكنك أيضًا إعداد مشروع يدويًا. لا تثبّت CLI عالميًا. بدل ذلك، أضف twenty-sdk كاعتماد محلي واربط سكربتًا واحدًا في ملف package.json لديك:
yarn add -D twenty-sdk
ثم أضف سكربتًا باسم twenty:
{
  "scripts": {
    "twenty": "twenty"
  }
}
الآن يمكنك تشغيل جميع الأوامر عبر yarn twenty <command>، مثلًا: yarn twenty app:dev، yarn twenty help، إلخ.

استكشاف الأخطاء وإصلاحها

  • أخطاء المصادقة: شغّل yarn twenty auth:login وتأكد من أن مفتاح واجهة برمجة التطبيقات لديك يمتلك الأذونات المطلوبة.
  • يتعذّر الاتصال بالخادم: تحقق من عنوان URL لواجهة البرمجة وأن خادم Twenty قابل للوصول.
  • الأنواع أو العميل مفقود/قديم: أعد تشغيل yarn twenty app:dev — فهو ينشئ العميل مضبوط الأنواع بشكل تلقائي.
  • وضع التطوير لا يزامن: تأكد من أن yarn twenty app:dev قيد التشغيل وأن التغييرات ليست متجاهلة من بيئتك.
قناة المساعدة على Discord: https://discord.com/channels/1130383047699738754/1130386664812982322