NocoBase Application Builder (DSL path)
Before you use this skill
This is the
opt-in DSL path. Default is
.
Stay on this skill only when the user explicitly wants YAML files they
can commit +
. If you arrived here from a generic "build me a
NocoBase app" request without the user naming DSL/YAML/git, switch to
instead — it's the default entry point.
Golden rule
is a
read-only reference library. When you're
unsure how a specific field, block, or popup is shaped, open the
closest CRM example, read it, then write your own adapted version in
your workspace. Per-scenario pointers live in the workflow sections below.
Never copy CRM files wholesale into your workspace. Do not
to get started, do not duplicate
collections/nb_crm_*.yaml
, do not base your
on CRM's.
Bulk-copying drags unrelated leads/opportunities/orders state and
workflows into your project — you then spend the whole session fighting
hundreds of irrelevant validator errors instead of building your module.
The pre-deploy spec validator catches most structural mistakes with a
clear error message.
Trust the validator: when it errors, fix what
it says rather than guessing — don't grep through
.
When NOT to use this skill
Hand off to the matching skill when the user's request is orthogonal:
| User asks for… | Skill |
|---|
| One-off live-UI tweak on an already-running page (move / reorder / reconfigure a single block, field, action) — no DSL commit wanted | |
| ACL / role permissions / route permissions | |
| Workflow create / update / revision / execution | (this skill only wires the trigger button; authoring the graph goes there) |
| Collection / field / relation authoring outside a DSL project | |
| Plugin development ( components, server code) | nocobase-plugin-development
|
| Install / enable plugin | |
| Environment setup / app install / upgrade | |
Any change that should live as a committed YAML file under
— stays here.
Environment
bash
cd <skill-dir>/src
export NB_USER=admin@nocobase.com NB_PASSWORD=admin123 NB_URL=http://localhost:14000
Quick start — new build
Default path for "build me a NocoBase app": copy the starter and
modify it. Do not hand-write the skeleton; do not study CRM first.
bash
cd <skill-dir>/src
cp -r ../templates/starter ../workspaces/<name>
# First push: --copy bypasses validation rules that only matter once
# popups exist (m2o popup binding, clickToOpen file presence). The
# starter ships with its own popups so the first push is actually fully
# valid — --copy is for your *extensions* before popups are wired.
npx tsx cli/cli.ts push <name> --force
The starter is a complete minimal CRUD — 1 collection (Projects),
1 Dashboard page (4 KPI tiles + 2 charts), 1 list page with
filterForm / table / addNew popup / detail popup / 2 updateRecord
recordActions. Push as-is → visible in NB → then edit.
Customizing the starter (the agile loop)
Iterate one concern at a time, push between each:
- Rename identifiers to match the user's domain.
collections/nb_starter_projects.yaml
→ your collection name
(match the convention)
- : change "Starter" / "Projects" titles
- directory name if you want (match
key)
- Find-replace across all files
- Adjust the field list in your collection YAML to match the
user's entity. Update and the
popup templates' accordingly.
- Add a 2nd entity only when the first one works end-to-end.
Create a new + dir,
copy
pages/starter/projects/layout.yaml
as a starting point.
- Extend incrementally: add a tab, a chart, a workflow trigger.
Push after every change. See "Advanced patterns" below for which
CRM file matches which pattern.
Never write the whole module in one shot. For customer-facing
builds — land the skeleton, show the user, gather feedback, iterate.
The starter push takes minutes; a hand-built module takes hours.
Fast-track: when helps
Pass
when the workspace has
no popup files yet (early
stage — validator would fire errors about "m2o field X has no popup
binding" that the user will fix in the next push). The reconciler
auto-bypasses spec errors in this state. Once any
exists, drop
and let validation run.
Incremental edits — existing workspace
- Add a block / field / action / popup → write the DSL → push.
- Remove from DSL → push. The reconciler destroys the matching
live model on the NB side and cleans . Manual NB-UI
authored elements (not tracked in ) are left alone.
- Rename → not supported automatically. Delete + re-add.
Targeted pushes:
- scopes to one menu subtree
- scopes to one page
- skips pages whose DSL hasn't changed since last push
For pure live-UI tweaks without a DSL commit, hand off to
instead (see the routing table above).
Advanced workflow — when the starter isn't enough
Triggers for going beyond the agile starter loop:
- More than ~3 collections with cross-relations (m2m, tree structures)
- Dedicated workflow / approval / permission logic that the user wants
designed up-front
- Multi-tab pages, sub-tables, or cross-module navigation
- Dashboard with bespoke KPIs mapped to the user's domain language
Progression: Round 0 design → Round 1 scaffold → Round 2 fill →
Round 2' seed → Round 3 JS. Each round is a deployable state.
Round 0: System architecture — confirm with user
Write a
(markdown, not YAML) covering:
- Collections — every table, its fields, and its relations.
See skill for field-interface reference.
- Page list — every page, one-line purpose each, grouped by menu.
- Navigation wiring — which m2o fields open which popup
templates; which pages link to each other.
Wait for user confirmation before writing YAML. A single design pass
saves 3× redesigns.
Skip Round 0 if the user's ask fits the starter shape (single entity,
basic CRUD).
Example:
markdown
## Collections
- nb_lib_books (title, author, isbn, category, status, loans: o2m → nb_lib_loans)
- nb_lib_members (name, email, phone, join_date, loans: o2m → nb_lib_loans)
- nb_lib_loans (loan_no, book: m2o, member: m2o, borrowed_at, due_date, returned_at, status)
## Pages (under menu "Library")
- Books list — browse + search books, add new
- Members list — browse members, their loan history
- Loans list — active/overdue loans, return action
- Dashboard — KPIs + charts
## Navigation
- books.table.title → books detail popup (shared template)
- loans.table.book → books detail popup (shared via defaults.yaml)
- loans.table.member → members detail popup (shared via defaults.yaml)
Round 0.5: Sub-agent CWD (only when spawning)
If launching a sub-agent (kimi TUI, Claude Code subprocess), its
CWD becomes the default write target. Set it before launch:
bash
mkdir -p <user-workdir>
cd <user-workdir>
kimi --yolo # or claude, codex
Skip the
and the agent writes to the parent project root.
Round 1: Scaffold — still start from the starter
Even in the advanced path, don't hand-write
+
from scratch. Copy the starter, then grow:
| Step | What to do |
|---|
| Base | cp -r templates/starter workspaces/<name>
— push once |
| Add collection | Write collections/<next_coll>.yaml
(format matches starter's). Match convention. |
| Add page | mkdir pages/<module>/<page>/
+ ; mirror starter's projects layout. |
| Update routes | Add entry under the existing group in . |
| Deploy | after each collection/page addition |
CRM references (consult only when stuck on structure):
templates/crm/collections/nb_crm_leads.yaml
— collection format
templates/crm/routes.yaml
— multi-group routes (shape only)
Round 2: Fill content — blocks, popups, templates
For each page beyond the starter basics:
| Building | CRM reference |
|---|
| Main list table + filter | templates/crm/pages/main/leads/layout.yaml
|
| Multi-tab page | templates/crm/pages/main/customers/
( + ) |
| Create-form with inline sub-table for o2m children | templates/crm/templates/block/form_add_new_opportunities_quotations_quotations.yaml
— is an o2m field listed in and rendered as an inline editable sub-table. Also templates/crm/pages/main/products/
for master/child UX. |
| Detail-popup template | templates/crm/templates/popup/activity_view.yaml
|
| m2o auto-popup bindings | templates/crm/defaults.yaml
|
| Parent-detail + child-list popup | templates/crm/pages/main/customers/tab_customers/popups/
|
| addNew + field click-popup pattern | templates/crm/pages/main/leads/popups/
|
Rules for copying from CRM:
- Copy 10–30 lines, adapt names. Never copy whole files.
- Don't copy / / — deployer assigns fresh.
- For every m2o field displayed in a table, either set
clickToOpen: templates/popup/popup_detail_<target>.yaml
OR add
in . Validator errors otherwise.
Per-row actions ()
By default a table's row-action column is empty — NB won't render
any action buttons unless
lists them. "Just edit
and delete" is a common but weak default: for most list tables the
user actually wants a
one-click state change (Mark Done,
Approve, Archive) sitting next to edit.
Decision order:
- Is there a boolean / enum status field? → add
with to show the matching button only when the
record is in the right state. One button per state transition
(Mark Done hidden when already done; Reopen hidden when not done).
- Does the record need a second detail/form view different from
the default edit popup? → with .
- Tree/hierarchy collection? → add .
- Need to navigate elsewhere with this record's id/filter? →
with
url: /admin/...?filter={{ctx.record.id}}
.
- Want to let the user clone a complex record? → .
- Need to start a workflow manually? → .
Prefer updateRecord + linkageRules
over custom JS buttons for
state changes. linkageRules covers 80% of row-level UX (conditional
show/hide, field-based gating, role checks via
) without
touching JS. JS actions are only needed for multi-step logic
(query, then update, then navigate) and are not currently
deployable by this reconciler.
Full per-row action palette (declared in
):
| DSL | Purpose | CRM reference |
|---|
| Edit popup (default shape) | many |
| Read-only detail popup | templates/crm/templates/popup/opportunity_view.yaml
|
| Single-row delete with confirm | many |
| Assign fields + optional linkageRules — the state-change workhorse | templates/crm/pages/main/overview/layout.yaml
(Done / Undone pair) |
| Open a custom popup with (form/detail different from edit) | templates/crm/pages/main/leads/popups/table.name.yaml
|
| Navigate to another admin page, carrying record context in URL | templates/crm/pages/lookup/layout.yaml
|
| Tree collection: add a child node under this row | templates/crm/pages/main/products/tab_categories/layout.yaml
|
| Clone the record into a new form | — |
| Manually trigger a workflow on this record | (toolbar in templates/crm/pages/main/customers/tab_customers/layout.yaml
— same shape works per-row) |
| / | Record-history plugin inline expand | — |
| AI employee button (tasks_file + employee) | templates/crm/pages/main/leads/layout.yaml
(toolbar — same shape per-row) |
Toolbar actions (declared in block-level
) use the same
type names plus
,
,
,
,
,
. Not every type makes sense in both contexts —
as a toolbar action would apply to
no specific row,
so put it in
.
Goal of Round 2: all pages have working CRUD — add / edit / view
popups wired correctly, row-action columns reflect real per-record
operations (not just edit+delete). Validator clean, NB UI shows no
"Collection may have been deleted" banners.
Round 2': Test data (parallel with Round 2)
Can run concurrently with page-filling. Insert data via API:
bash
TOKEN=$(curl -sS -X POST $NB_URL/api/auth:signIn \
-H 'Content-Type: application/json' -H 'X-Authenticator: basic' \
-d '{"account":"'$NB_USER'","password":"'$NB_PASSWORD'"}' \
| python3 -c 'import json,sys;print(json.load(sys.stdin)["data"]["token"])')
# Always GET existing record IDs first — they're snowflake integers
# (e.g. 359571523764224), NEVER 1/2/3.
curl -sS -X POST $NB_URL/api/<collection>:create -H "Authorization: Bearer $TOKEN" \
-H 'Content-Type: application/json' -d '{...fields..., "owner":{"id": <real-user-id>}}'
Parent tables first; fill every FK on children (leaving it null
orphans the row). Then:
bash
npx tsx cli/cli.ts verify-data <name> # FK & completeness check
Why parallel: Round 3 JS (charts, KPIs) needs data to render
anything. Start the seed once you have collections (end of Round 1);
by the time Round 2 finishes pages, you have records to test against.
Round 3: JS — where CRM uses it, you probably need it
Now that CRUD + data work, audit where JavaScript adds value. Walk the
CRM template and ask "does the CRM have JS here?" for each spot in your
project. Three typical JS opportunities:
| Spot | CRM has JS? | Your project likely needs JS if... |
|---|
| Field renderer / column (e.g. color-coded status tag, days-until-due badge) | ✅ in most list tables | Any field whose display depends on a derived value (date math, status-to-color, multi-field composite) |
| Block (KPI card, custom widget inside a form) | ✅ overview, analytics, per-form tips | A summary widget, inline chart, or "helper panel" that reads from multiple collections |
| Dashboard page (whole page of charts + KPIs) | ✅ analytics page | Module has ≥3 measurable metrics users care about; validator requires ≥5 charts on pages titled "Dashboard" / "Analytics" |
Start by grepping the matching CRM page and its
+
folders to confirm where the CRM adds JS. Then write YOUR JS file
adapted from the single-file table below.
Dashboards specifically look bad when designed freehand — mirror the
CRM shape. Open the reference layout first, copy its
block count,
ordering, and grid widths into your own
; then fill in
leaf files with your content.
| Reference layout | What to mirror |
|---|
templates/crm/pages/main/overview/layout.yaml
| Overview: 1 jsBlock hero row, 2 small tables underneath. Use for a landing page with a few KPIs. |
templates/crm/pages/main/analytics/layout.yaml
| Full dashboard: filterForm row → 4 KPI jsBlocks in one row → 5 charts in a grid. Use when you want ≥5 charts (validator requires this when the page title contains or ). |
Procedure:
- Open the reference . Note the block keys / types / widths.
- Write YOUR with the SAME shape — same number of blocks,
same grid widths in the section — but your own block
keys and your own collection names.
- For each block's leaf JS/SQL file, copy from the single-file table
below. Copy files individually; do not the folder.
| Leaf file to copy | Reference |
|---|
| KPI card jsBlock | templates/crm/pages/main/overview/js/overview_jsBlock.js
|
| Filtered summary jsBlock | templates/crm/pages/main/analytics/js/analytics_jsBlock.js
|
| Chart SQL (grouped counts) | templates/crm/pages/main/analytics/charts/analytics_chart_2.sql
|
| Chart render (echarts bar/pie) | templates/crm/pages/main/analytics/charts/analytics_chart_2_render.js
|
| Filter stat buttons on filterForm | templates/crm/pages/main/customers/tab_customers/js/customers_customers_filterForm_customer_stats_filter_block.js
|
| Full-page custom UI (wizard / multi-step / custom flow) | templates/crm/pages/main/customers/tab_merge/js/customers_merge_jsBlock.js
— whole page is one , ~580 lines React |
After copying each leaf file:
- Rename in place and retarget SQL/collection/field names.
- Remove i18n wrappers unless your module has i18n.
- Simplify filter var references if your page's
filterForm uses different field keys.
SQL charts: save + run as a two-step pattern —
then
.
Core concepts
Two identifiers: and
= lower_snake_ascii identity — drives directory names under
and entries in
. Always write it explicitly when the title
isn't pure ASCII (Chinese/spaces slugify to gibberish).
= display text as the user wants it shown.
yaml
- key: it_ops
title: IT 运维
type: group
children:
- key: tickets
title: 工单
Two popup modes: vs bare
— popup block is a
reference to the template.
Editing the template updates every popup that references it. Use for any
shared Add/Edit form.
yaml
blocks:
- ref: templates/block/form_add_new_tickets.yaml
key: reference # REQUIRED for shared refs
Bare (no
) — template content is
inlined
per popup; each copy is independent. Use only to factor a bulky block
out of the page file.
After deploy, a shared template's
should be ≥ 1. If it
stays at 0,
was forgotten.
Auto-created columns — do NOT declare them
NocoBase auto-creates these; declaring them causes silent filtering or
type conflicts:
- System columns: , , , ,
- m2o / o2m FK columns: declaring auto-creates
; don't add a second row
- m2m join tables: is auto-created; don't write a
collection YAML for it
Table vs sub-table — two different things, don't confuse
| Table | Sub-table |
|---|
| What it is | Full CRUD block (filter + list + add/edit popups) for child records of a parent | Inline editable grid for child rows, lives INSIDE a parent form |
| DSL | + resource_binding.sourceId + associationName
| { field: tasks, type: subTable, columns: [...] }
inside a createForm/editForm's |
| Where used | Detail popup, tab page, standalone-list popup | Inside createForm / editForm |
| Use when | Children browsed separately (customer detail → orders list) | Children entered alongside parent (invoice + line items) |
Bare
in a form is the third option: a
RecordSelect picker
("pick existing record"). Rarely what you want — validator warns.
Canonical CRM examples:
- Table (standalone CRUD block):
templates/crm/pages/main/customers/tab_customers/popups/table.name.yaml
- Sub-table (inline editable grid):
templates/crm/templates/block/form_add_new_opportunities_quotations_quotations.yaml
()
flips meaning
On
m2o,
names the FK column on the
current table
(
owner: m2o, foreignKey: owner_id
→
on SELF).
On
o2m,
names the FK column on the
target table
(
tasks: o2m → nb_pm_tasks, foreignKey: project_id
→
on
).
Command reference
bash
cd <skill-dir>/src
export NB_USER=... NB_PASSWORD=... NB_URL=...
npx tsx cli/cli.ts push <name> --force # deploy DSL → NocoBase
npx tsx cli/cli.ts push <name> --group <key> # only one subtree
npx tsx cli/cli.ts push <name> --incremental # skip unchanged (git diff)
npx tsx cli/cli.ts pull <name> # NocoBase → DSL (full round-trip)
npx tsx cli/cli.ts diff <left> <right> # compare two DSL trees
npx tsx cli/cli.ts duplicate-project <src> <dst> --key-suffix _v2
npx tsx cli/cli.ts verify-data <name> # FK / completeness check
push and pull are both one-way. Round-tripping = push + pull + git diff.
Common errors
| Error | Fix |
|---|
| Field names don't match the collection YAML |
| Set or add a / field |
| Only some pages deployed | mismatch with a directory |
| on create | / declared in YAML — remove |
| Chart SQL failed | Seed data first; quote field names like |
| m2o link 400 in UI | Missing binding for target collection |
Collection X not found in data source main
| used a short name — use the full collection name (, not ). See templates/crm/pages/main/customers/tab_customers/popups/
|
| Per-row column shows only edit+delete | missing — see "Per-row actions" below for what to add. Removing an action/field/column from the DSL now does destroy it on the NB side on next push. |
If any of the above contradicts what you observe at runtime, the manual
is stale — note what was missing and tell the user.