ما هي التطبيقات؟
تتيح لك التطبيقات إنشاء وإدارة تخصيصات Twenty ككود. بدلًا من تكوين كل شيء عبر واجهة المستخدم، تُعرِّف نموذج بياناتك ووظائف المنطق في الكود — مما يجعل البناء والصيانة والنشر إلى مساحات عمل متعددة أسرع. ما الذي يمكنك فعله اليوم:- عرِّف كائنات وحقولًا مخصصة على شكل كود (نموذج بيانات مُدار)
- أنشئ وظائف منطقية مع مشغلات مخصصة
- انشر التطبيق نفسه عبر مساحات عمل متعددة
المتطلبات الأساسية
- Node.js 24+ وYarn 4
- مساحة عمل Twenty ومفتاح واجهة برمجة التطبيقات (أنشئ واحدًا على https://app.twenty.com/settings/api-webhooks)
البدء
أنشئ تطبيقًا جديدًا باستخدام المُهيئ الرسمي، ثم قم بالمصادقة وابدأ التطوير:هيكل المشروع (مُنشأ بالقالب)
عند تشغيلnpx create-twenty-app@latest my-twenty-app، يقوم المُهيئ بما يلي:
- ينسخ تطبيقًا أساسيًا مصغّرًا إلى
my-twenty-app/ - يضيف اعتمادًا محليًا
twenty-sdkوتهيئة Yarn 4 - ينشئ ملفات ضبط ونصوصًا مرتبطة بـ
twentyCLI - يُولّد ضبطًا افتراضيًا للتطبيق ودورًا افتراضيًا للوظيفة
- package.json: Declares the app name, version, engines (Node 24+, Yarn 4), and adds
twenty-sdkplus atwentyscript that delegates to the localtwentyCLI. Runyarn twenty helpto list all available commands. - .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() | امتدادات الحقول للكائنات الموجودة |
تسمية الملفات مرنة. يعتمد اكتشاف الكيانات على بنية الشجرة المجردة (AST) — إذ يقوم SDK بفحص ملفات المصدر لديك بحثًا عن النمط
export default define<Entity>({...}). يمكنك تنظيم ملفاتك ومجلداتك كيفما تشاء. التجميع حسب نوع الكيان (مثلًا، logic-functions/ وroles/) هو مجرد عرف لتنظيم الشيفرة، وليس مطلبًا إلزاميًا.yarn twenty app:generatewill create agenerated/folder (typed Twenty client + workspace types).yarn twenty entity:addwill add entity definition files undersrc/for your custom objects, functions, front components, or roles.
المصادقة
The first time you runyarn twenty auth:login, you’ll be prompted for:
- عنوان URL لواجهة برمجة التطبيقات (الافتراضي http://localhost:3000 أو ملف تعريف مساحة العمل الحالية لديك)
- مفتاح واجهة برمجة التطبيقات
~/.twenty/config.json. You can maintain multiple profiles and switch between them.
Managing workspaces
yarn twenty auth:switch, all subsequent commands will use that workspace by default. You can still override it temporarily with --workspace <name>.
استخدم موارد SDK (الأنواع والتكوين)
يوفّر twenty-sdk كتلَ بناءٍ مضبوطة الأنواع ودوال مساعدة تستخدمها داخل تطبيقك. فيما يلي الأجزاء الأساسية التي ستتعامل معها غالبًا.دوال مساعدة
يوفّر SDK دوالًا مساعدة لتعريف كيانات تطبيقك. كما هو موضح في اكتشاف الكيانات، يجب استخدامexport default define<Entity>({...}) كي يتم اكتشاف كياناتك:
| دالة | الغرض |
|---|---|
defineApplication() | تهيئة بيانات التعريف للتطبيق (مطلوب، واحد لكل تطبيق) |
defineObject() | تعريف كائنات مخصصة مع حقول |
defineLogicFunction() | تعريف وظائف منطقية مع معالجات |
defineFrontComponent() | عرِّف مكوّنات أمامية لواجهة مستخدم مخصّصة |
defineRole() | تهيئة صلاحيات الدور والوصول إلى الكائنات |
defineField() | وسّع الكائنات الموجودة بحقول إضافية |
تعريف الكائنات
تصف الكائنات المخصصة كلًا من المخطط والسلوك للسجلات في مساحة عملك. استخدمdefineObject() لتعريف كائنات مع تحقق مدمج:
- استخدم
defineObject()للحصول على تحقق مدمج ودعم أفضل من IDE. universalIdentifierيجب أن يكون فريدًا وثابتًا عبر عمليات النشر.- يتطلب كل حقل
nameوtypeوlabelومعرّفuniversalIdentifierثابتًا خاصًا به. - المصفوفة
fieldsاختيارية — يمكنك تعريف كائنات بدون حقول مخصصة. - You can scaffold new objects using
yarn twenty entity:add, which guides you through naming, fields, and relationships.
يتم إنشاء الحقول الأساسية تلقائيًا. عند تعريف كائن مخصص، يضيف Twenty تلقائيًا حقولًا قياسية مثل
name وcreatedAt وupdatedAt وcreatedBy وposition وdeletedAt. لا تحتاج إلى تعريف هذه في مصفوفة fields — أضف فقط حقولك المخصصة.تكوين التطبيق (application-config.ts)
كل تطبيق لديه ملف واحدapplication-config.ts يصف:
- هوية التطبيق: المعرفات، اسم العرض، والوصف.
- كيفية تشغيل وظائفه: الدور الذي تستخدمه للأذونات.
- متغيرات (اختياري): أزواج مفتاح-قيمة تُعرض لوظائفك كمتغيرات بيئة.
defineApplication() to define your application configuration:
- حقول
universalIdentifierهي معرّفات حتمية تخصك؛ أنشئها مرة واحدة واحتفظ بها ثابتة عبر عمليات المزامنة. applicationVariablesتصبح متغيرات بيئة لوظائفك (على سبيل المثال،DEFAULT_RECIPIENT_NAMEمتاح كـprocess.env.DEFAULT_RECIPIENT_NAME).defaultRoleUniversalIdentifierيجب أن يطابق ملف الدور (انظر أدناه).
الأدوار والصلاحيات
يمكن للتطبيقات تعريف أدوار تُغلّف الصلاحيات على كائنات وإجراءات مساحة العمل لديك. يعين الحقلdefaultRoleUniversalIdentifier في application-config.ts الدور الافتراضي الذي تستخدمه وظائف المنطق في تطبيقك.
- مفتاح واجهة البرمجة في وقت التشغيل المحقون باسم
TWENTY_API_KEYمستمد من دور الوظيفة الافتراضي هذا. - سيُقيَّد العميل مضبوط الأنواع بالأذونات الممنوحة لذلك الدور.
- اتبع مبدأ أقل الامتياز: أنشئ دورًا مخصصًا بالأذونات التي تحتاجها وظائفك فقط، ثم أشِر إلى معرّفه الشامل.
الدور الافتراضي للوظيفة (*.role.ts)
عند توليد تطبيق جديد بالقالب، ينشئ CLI أيضًا ملف دور افتراضي. استخدمdefineRole() لتعريف أدوار مع تحقق مدمج:
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() لتصدير تكوين مع معالج ومشغّلات اختيارية.
- route: يعرِض وظيفتك على مسار وطريقة HTTP تحت نقطة النهاية
/s/:
مثال:path: '/post-card/create',-> الاستدعاء على<APP_URL>/s/post-card/create
- cron: يشغّل وظيفتك على جدول باستخدام تعبير CRON.
- databaseEvent: يعمل على أحداث دورة حياة كائنات مساحة العمل. عندما تكون عملية الحدث هي
updated، يمكن تحديد الحقول المحددة المراد الاستماع إليها في مصفوفةupdatedFields. إذا تُركت غير معرّفة أو فارغة، فسيؤدي أي تحديث إلى تشغيل الدالة.
مثال: person.updated
الملاحظات:
- المصفوفة
triggersاختيارية. يمكن استخدام الوظائف بدون مشغلات كوظائف مساعدة تُستدعى بواسطة وظائف أخرى. - يمكنك مزج أنواع متعددة من المشغلات في وظيفة واحدة.
حمولة مشغل المسار
عندما يستدعي مشغّل المسار وظيفتك المنطقية، يتلقى كائنًا من النوعRoutePayload يتبع تنسيق AWS HTTP API v2. استورد النوع من twenty-sdk:
RoutePayload على البنية التالية:
| الخاصية | النوع | الوصف |
|---|---|---|
headers | Record<string, string | undefined> | رؤوس HTTP (فقط تلك المدرجة في forwardedRequestHeaders) |
queryStringParameters | Record<string, string | undefined> | معلمات سلسلة الاستعلام (تُضمّ القيم المتعددة باستخدام فواصل) |
pathParameters | Record<string, string | undefined> | معلمات المسار المستخرجة من نمط المسار (على سبيل المثال، /users/:id → { id: '123' }) |
المحتوى | object | null | جسم الطلب المُحلَّل (JSON) |
isBase64Encoded | قيمة منطقية | ما إذا كان جسم الطلب مُرمَّزًا بترميز base64 |
requestContext.http.method | string | طريقة HTTP (GET, POST, PUT, PATCH, DELETE) |
requestContext.http.path | string | المسار الخام للطلب |
تمرير رؤوس HTTP
افتراضيًا، لا تُمرَّر رؤوس HTTP من الطلبات الواردة إلى وظيفتك المنطقية لأسباب أمنية. للوصول إلى رؤوس محددة، قم بإدراجها صراحةً في مصفوفةforwardedRequestHeaders:
تُحوَّل أسماء الرؤوس إلى أحرف صغيرة. يمكنك الوصول إليها باستخدام مفاتيح بأحرف صغيرة (على سبيل المثال،
event.headers['content-type']).- Scaffolded: Run
yarn twenty entity:addand choose the option to add a new logic function. يُولّد هذا ملفًا مبدئيًا مع معالج وتكوين. - يدوي: أنشئ ملفًا جديدًا
*.logic-function.tsواستخدمdefineLogicFunction()مع اتباع النمط نفسه.
المكوّنات الأمامية
تتيح لك المكوّنات الأمامية إنشاء مكوّنات React مخصّصة تُعرَض داخل واجهة مستخدم Twenty. استخدمdefineFrontComponent() لتعريف مكوّنات مع تحقّق مدمج:
- المكوّنات الأمامية هي مكوّنات React تُعرَض ضمن سياقات معزولة داخل Twenty.
- استخدم لاحقة الملف
*.front-component.tsxللاكتشاف التلقائي. - يشير الحقل
componentإلى مكوّن React الخاص بك. - Components are built and synced automatically during
yarn twenty app:dev.
- Scaffolded: Run
yarn twenty entity:addand choose the option to add a new front component. - يدوي: أنشئ ملفًا جديدًا
*.front-component.tsxواستخدمdefineFrontComponent().
عميل مُولَّد مضبوط الأنواع
Runyarn twenty app:generate to create a local typed client in generated/ based on your workspace schema. استخدمه في وظائفك:
yarn twenty app:generate. أعِد التشغيل بعد تغيير كائناتك أو عند الانضمام إلى مساحة عمل جديدة.
بيانات الاعتماد وقت التشغيل في الوظائف المنطقية
عندما تعمل وظيفتك على 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 عالميًا. Instead, add twenty-sdk as a local dependency and wire a single script in your package.json:
twenty script:
yarn twenty <command>, e.g. yarn twenty app:dev, yarn twenty app:generate, yarn twenty help, etc.
استكشاف الأخطاء وإصلاحها
- Authentication errors: run
yarn twenty auth:loginand ensure your API key has the required permissions. - يتعذّر الاتصال بالخادم: تحقق من عنوان URL لواجهة البرمجة وأن خادم Twenty قابل للوصول.
- Types or client missing/outdated: run
yarn twenty app:generate. - Dev mode not syncing: ensure
yarn twenty app:devis running and that changes are not ignored by your environment.