name: delete-initial-resource description: Remove one or more of the initial CRM resources (contacts, companies, deals, tags, tasks) from the codebase. Use when the user asks to delete, remove, or strip out one or several of these built-in resources. Runs the delete-initial-resource.ts script to drop each resource's own folder, then guides cleanup of every file that references them.
delete-initial-resource
Overview
Removes one or more of the five initial Atomic CRM resources (contacts, companies, deals, tags, tasks) and every reference to them. Irreversible confirm the target(s) with the user first; rely on git to recover. Exit criterion: make typecheck && make lint is clean and no live reference to the deleted resource(s) survives in src/ or supabase/.
When to Use
- The user asks to delete, remove, or strip out one or several built-in CRM resources.
- Not for custom entities added during setup only the five built-ins above.
For the backend migration mechanics, see Skill({skill: "backend-dev"}) and Skill({skill: "writing-migrations"}).
Steps
- Confirm the target(s), then read each one's file before editing:
contacts→contacts.md— spine; confirm cascade scope firstcompanies→companies.md— spine/link; confirm cascade scope firstdeals→deals.mdtags→tags.mdtasks→tasks.md
- Run the script (below) — deletes each
src/components/atomic-crm/<resource>/folder and prints the dependent files to clean. - Clean each dependent file using the "Shapes" patterns below + the per-resource file.
- Verify (below).
Deleting several at once: clean from the most-coupled resource outward; clean a shared file once for both; and reconcile interacting union guidance — deleting both contacts and deals removes the notes/activity-log reference entirely rather than each narrowing it, so don't half-apply each file's "narrow the union" note.
The script
node .claude/skills/delete-initial-resource/delete-initial-resource.ts <resource> [<resource> ...]
Validates the names, deletes each resource's folder, and prints the merged, de-duplicated dependent-file list (files inside a just-deleted folder are dropped). It does not edit those files — that's your job. Each entry in the script's dependentFiles map has a // comment saying what to remove; read it alongside the printed list, and update the map if the codebase grows new references.
Shapes a resource takes
A resource is rarely just a folder. The later shapes are invisible to a \b<resource>\b grep:
- Standalone — imports, JSX, routes, menu/nav, dashboard widgets, the
<Resource>entry inroot/CRM.tsx. Wrapper components outside the folder that only render the resource: delete outright, then clean importers. - Field/column on another record — fans out far wider: the
types.tstype,providers/commons/mergeContacts.ts, CSV/JSON import, i18n, stories/tests,StoryWrapperbuilders, and the sample CSVs (header and every row). For a link column, decide per-link: orphaned shell (typed, defaulted) or drop — remove the UI either way. - Aggregated
nb_<resource>(from the_summaryviews) lands on a record type, read across shows/lists/filters/generators. Invisible to a\b<resource>\bgrep (the_is a word char) — grepnb_<resource>and the denormalizedcompany_nameseparately. - Shared subsystems — notes (
reference: "contacts" | "deals") and the activity log ("company" | "contact" | "deal" | "all"union). Narrow the union, delete the per-type render components (ActivityLog*Created.tsx) + dead branches, prune theproviders/commons/activity.tsfetcher +consts.ts/types.tsmembers. Surviving=== "<other>"checks stay valid — only the deleted member's branches go. - Config props (
taskTypes,dealStages,companySectors) live inroot/defaultConfiguration.ts,ConfigurationContext.tsx,CRM.tsx(default + jsdoc + store seed),App.tsx, andSettingsPage.tsx(the<Card>and the section-list entry). Not always named after the resource — grep the settings<Card>for other keys (Deals ownedcurrency) and flag borderline-general ones rather than dropping them silently. - Custom dataProvider methods live in both providers;
CrmDataProviderderives from the supabase one, so removing them there drops them from the type. Also clean the lifecycle-callback blocks in both providers, thesalesbeforeDeletereassignment, the supabase<resource> -> <resource>_summarygetListrouting, and the FakeRest generators (which drive record creation). - JSON importer (
misc/useImportFromJson.ts) treatscontacts/companies/notes/tasksas top-level types (import<X>fn,$.<x>.*path, aTYPESarm + switch, guards,stats/failedImports/idsMapskeys). Mirror inImportPage.tsx+import-sample.json; the CSV importer (useContactImport.tsx) has parallel mappings. - Check
useCallback/useMemodeps — a stalegetTags/getCompaniesthere is invisible untiltsc.
Backend (the script only touches frontend)
See the backend-dev skill for the migration workflow. In supabase/schemas/:
- Drop/adjust the table + view, generate a migration (
npx supabase db diff --local -f remove_<resource>); updatecontacts_summary/companies_summaryif they aggregated it. activity_logis aUNION ALLwith the same column set per branch — remove the resource's columns from every branch (keep them aligned), delete its dedicated branch(es), and drop the matching snake→camel rename in the supabasegetList("activity_log").06_grants.sqlgrants the table and its<resource>_id_seq(drop both).05_policies.sqlhas an RLS-enable line andcreate policystatements (drop both).- A resource used inside a SQL function (
merge_contacts): drop those lines + the unused local var. Editing02_functions.sqlregenerates the whole function (expected) — renumber leftover step comments. - Orphan check: deleting triggers can orphan helpers — but verify shared callers first (
get_domain_faviconsurvives acompaniesdelete because contact avatars also call it). - Edge functions are separate and easy to miss:
merge_contacts/,_shared/db.ts(Kysely types),postmark/(inbound email), andmcp/(an AI subsystem whose depth varies per resource). When a surviving function loses a capability, flag it to the user.
Rationalizations
| Rationalization | Reality |
|---|---|
| "I deleted the folder, the resource is gone." | A resource is also fields on other records, nb_<resource> aggregates, shared subsystems, config props, dataProvider methods, and SQL — the folder is the easy part. |
"A \b<resource>\b grep came back clean, so it's clean." |
nb_<resource> and denormalized company_name are invisible to that grep (_ is a word char). Grep them separately. |
| "I'll remove the i18n key from English only." | The French catalog is type-checked against English — a one-sided removal is a tsc error. Remove from both or neither. |
| "The script edits the dependent files for me." | The script only deletes folders and prints the list. Editing every dependent file is your job. |
| "Typecheck passes, so the deletion is complete." | tsc misses broken views/functions/grants. Reset the DB and grep for live references too. |
Red Flags
- Editing files inside
supabase/migrations/(never — they are append-only history). - A one-sided i18n key removal (English without French, or vice versa).
- Relying on a single
\b<resource>\bgrep and skippingnb_<resource>/company_name. - Deleting a shared SQL helper or edge-function capability without checking surviving callers.
- Deleting several resources but half-applying each file's "narrow the union" note.
- Skipping the
db resetcheck when Supabase is running.
Verification
make typecheck && make lint
Resolve whatever tsc surfaces (the dependent-file list is a guide, not a guarantee). Then grep -rniE "\b<resource>\b" over src/+supabase/, plus separate greps for nb_<resource>, company_name, and config-prop names (<resource>Types, companySectors, currency). Watch for benign/substring false positives (each resource file lists its own); never edit supabase/migrations/.
-
make typecheck && make lintis clean. -
\b<resource>\b,nb_<resource>,company_name, and config-prop greps return only benign matches. - i18n keys removed from both catalogs, or neither.
- Backend: table/view/policies/grants/sequences and any function references handled in
supabase/schemas/(notmigrations/). - If Supabase is running,
npx supabase db reset --localreplays cleanly. - Any lost edge-function capability flagged to the user.
i18n is all-or-nothing: frenchCrmMessages.ts is type-checked against the type derived from englishCrmMessages.ts, so remove a key from both catalogs or neither (a one-sided removal is a tsc error). Dead resources.<x>.* keys are harmless — flag rather than force. frenchCrmMessages.ts uses literal … escapes; if an exact-string Edit fails on a block spanning one, fall back to sed -i '<from>,<to>d'.
If Supabase is running, npx supabase db reset --local replays the chain + seed.sql and catches a broken view/function/grant db diff misses (benign index "…_pkey" does not exist NOTICEs expected). On Node 22 (see .nvmrc), make test (vitest) is also available.