migrating-to-zero-yaml
Migrating to Zero YAML
This skill replaces the old CLI migrator with an LLM-driven mechanical migration. Keep behavior as close as possible to the existing YAML project, then validate and stop. Do not turn this into a broader cleanup unless the user asks.
Critical rule
- Do not tell the user to run
nango migrate-to-zero-yaml. - Perform the migration manually in code.
- Preserve existing behavior first.
- Do not opportunistically migrate
lastSyncDate,syncType,trackDeletes, or validation patterns unless the user asks or compilation forces you to touch them. - If the repo is already Zero YAML, stop and switch to a regular Zero YAML refactor workflow instead of forcing a migration.
Preconditions
- Confirm you are in the Nango project root and
nango.yamlexists. - Make sure there is a rollback path before destructive edits (git branch/commit or external backup).
- Read
nango.yamland inventory:- integrations
- syncs
- actions
- on-event handlers
- models
- Inspect the existing TypeScript files that those YAML entries point to.
- Keep the migration mechanical. Avoid redesigning APIs, renaming scripts, or changing business logic.
End state you are aiming for
The migrated project should have:
package.jsontsconfig.json- root
models.ts - root
index.ts - integration files exporting
createSync(),createAction(), orcreateOnEvent() - helper files importing Nango runtime types from
nango - no
nango.yamlin the final migrated state
Migration workflow
- Read
nango.yamland build a migration inventory. - Create or update Zero YAML scaffolding:
package.jsontsconfig.json
- Generate
models.tsfrom the YAML models. - Rewrite each sync file into
createSync(...). - Rewrite each action file into
createAction(...). - Rewrite each lifecycle handler into
createOnEvent(...). - Generate
index.tsside-effect imports for every migrated script. - Fix helper-file imports and small TypeScript breakages.
- Compile and validate.
- Remove
nango.yamlonly after the replacement project is in place and there is still a rollback path outside the file.
Scaffolding rules
package.json
If the project does not have a package file, create one. If it already exists, preserve existing fields and add the Zero YAML essentials:
type: "module"engines.node: ">=20.0"- scripts:
compile: "nango compile"dev: "nango dev"
devDependencies.nangodevDependencies.zod
Prefer preserving the repo's package manager and existing workspace fields.
tsconfig.json
Use the standard Zero YAML TypeScript config shape:
{
"$schema": "https://json.schemastore.org/tsconfig",
"include": ["index.ts", "**/*.ts"],
"exclude": ["node_modules", "dist", "build", ".nango"],
"compilerOptions": {
"module": "node16",
"target": "esnext",
"strict": true,
"esModuleInterop": true,
"skipLibCheck": true,
"forceConsistentCasingInFileNames": true,
"moduleResolution": "node16",
"exactOptionalPropertyTypes": true,
"noImplicitOverride": true,
"noImplicitReturns": true,
"noPropertyAccessFromIndexSignature": true,
"noUncheckedIndexedAccess": true,
"noUnusedLocals": true,
"noUnusedParameters": true,
"sourceMap": true,
"noEmit": true
}
}
Generate models.ts
Create a root models.ts file that turns YAML model definitions into Zod schemas plus exported inferred types.
Use this pattern:
import * as z from 'zod';
export const Ticket = z.object({
id: z.string(),
title: z.string()
});
export type Ticket = z.infer<typeof Ticket>;
export const models = {
Ticket
};
Conversion rules:
string->z.string()number->z.number()boolean->z.boolean()Date->z.date()any->z.any()orz.unknown()if you need a safer fallback- optional field ->
.optional() - array ->
z.array(...) - nested object -> nested
z.object(...) - union ->
z.union([...]) - model reference -> reuse the referenced schema
- dynamic object fields ->
z.record(...)or.catchall(...)
Practical rule: if the old YAML model is hard to express exactly, preserve migration momentum with the broadest safe schema that still lets the project compile. Tighten later only if needed.
Define models in dependency order. If you hit circular references, use z.lazy(() => ModelName).
Rewrite syncs
For each YAML sync, keep the existing file path when possible and wrap the current default-exported function in createSync(...).
Map YAML config to Zero YAML config like this:
description->descriptionversion->version(default to0.0.1if missing)runs->frequencyauto_start->autoStartsync_type->syncTypetrack_deletes->trackDeletesendpoints->endpointsscopes->scopes- webhook subscriptions ->
webhookSubscriptions - YAML
inputmodel ->metadata - YAML
outputmodels ->models - existing default export function body ->
exec
If no YAML input model exists for the sync, default metadata to z.object({}) for migration parity.
If the file exports onWebhookPayloadReceived, move that function into the createSync(...) config as onWebhook.
Use this shape:
import { createSync } from 'nango';
import * as z from 'zod';
import { Metadata, Ticket } from '../../models.js';
const sync = createSync({
description: 'Sync tickets',
version: '0.0.1',
frequency: 'every hour',
autoStart: true,
syncType: 'full',
trackDeletes: false,
endpoints: [{ method: 'GET', path: '/tickets', group: 'Tickets' }],
metadata: Metadata,
models: { Ticket },
exec: async (nango) => {
// keep existing logic
}
});
export type NangoSyncLocal = Parameters<typeof sync['exec']>[0];
export default sync;
Rewrite actions
For each YAML action, wrap the existing default-exported function in createAction(...).
Map YAML config like this:
description->descriptionversion->version(default0.0.1)endpoint->endpointinput->inputoutput->outputscopes->scopes- existing default export function body ->
exec
If the action has no input or output schema, use z.void().
Use this shape:
import { createAction } from 'nango';
import * as z from 'zod';
import { CreateTicketInput, CreateTicketOutput } from '../../models.js';
const action = createAction({
description: 'Create ticket',
version: '0.0.1',
endpoint: { method: 'POST', path: '/tickets', group: 'Tickets' },
input: CreateTicketInput,
output: CreateTicketOutput,
exec: async (nango, input) => {
// keep existing logic
}
});
export type NangoActionLocal = Parameters<typeof action['exec']>[0];
export default action;
Rewrite on-event handlers
For each YAML lifecycle handler, wrap the existing default export in createOnEvent(...).
Map like this:
- event name ->
event - description -> preserve an existing description if one exists; otherwise use
<event> event handler - existing default export function body ->
exec
Use this shape:
import { createOnEvent } from 'nango';
export default createOnEvent({
event: 'post-connection-creation',
description: 'post-connection-creation event handler',
exec: async (nango) => {
// keep existing logic
}
});
Import and typing fixes
The old migrator also cleaned up imports. Do the same manually.
- Remove
NangoSync,NangoAction,ProxyConfiguration, andActionErrorimports from oldmodelsimports. - Import runtime types from
nangoinstead. - Keep generated model imports pointing to
../../models.jsor the correct relative path. - Use side-effect imports with
.jsinindex.ts.
Important mechanical fixes:
- Replace local
NangoSyncparameter annotations withNangoSyncLocalafter wrapping the file. - Replace local
NangoActionparameter annotations withNangoActionLocalafter wrapping the file. - Remove type arguments from:
nango.batchSave<T>(...)nango.batchUpdate<T>(...)nango.batchDelete<T>(...)nango.getMetadata<T>()
Those generic arguments often compile in the old format but become noisy or unnecessary after migration.
Generate index.ts
Create a root index.ts with side-effect imports for every migrated script. Include the .js extension.
Example:
import './github/syncs/fetch-issues.js';
import './github/actions/create-issue.js';
import './github/on-events/post-connection-creation.js';
Do not use named or default imports here.
Helper-file cleanup
Search the rest of the repo for TypeScript helper files outside the migrated integration entrypoints.
If they import Nango runtime types from old model files, move those imports to nango and leave model imports in place only for actual model schemas/types.
Do not rewrite unrelated business logic in helper files.
Keep the migration parity-first
The manual migration should behave like the old CLI migrator:
- preserve script names
- preserve file locations when possible
- preserve function bodies
- preserve
syncType - preserve
trackDeletes - preserve
nango.lastSyncDateusage if it already exists
Only do a second modernization pass if the user explicitly asks.
Validation loop
After the mechanical migration:
- run
nango compile - fix straightforward TypeScript/import/schema issues
- run
nango dryrun <script-name> <connection-id> --validate -e dev --no-interactive --auto-confirm - for actions, add
--input '{...}'or--input '{}' - if validation passes, run
nango dryrun <script-name> <connection-id> --save -e dev --no-interactive --auto-confirm - run
nango generate:tests && npm test - never hand-edit generated
*.test.json
If a migrated sync/action still needs a true behavioral refactor after this point, stop and treat that as a separate task.
Useful docs
- Zero YAML migration doc: https://www.nango.dev/docs/implementation-guides/platform/migrations/migrate-to-zero-yaml
- Testing: https://www.nango.dev/docs/implementation-guides/platform/functions/testing
- Functions SDK: https://www.nango.dev/docs/reference/functions
- Checkpoints migration: https://www.nango.dev/docs/implementation-guides/platform/migrations/migrate-to-checkpoints
- Deletion detection: https://www.nango.dev/docs/implementation-guides/use-cases/syncs/deletion-detection
More from nangohq/skills
nango-function-builder
Builds Nango Functions (TypeScript createAction/createSync) with checkpoint-first sync patterns, action and sync references, project/root checks, deletion strategies, and a docs-aligned dryrun/test workflow. Use when creating or updating Nango actions or syncs.
120building-nango-functions-locally
Builds Nango Functions in a checked-out Zero YAML TypeScript Nango project using local files, index.ts registration, nango dryrun, generated tests, and optional nango deploy. Use when creating or updating Nango actions or syncs locally.
70migrating-nango-deletion-detection
Migrates Nango syncs from deleteRecordsFromPreviousExecutions()/trackDeletes to trackDeletesStart/trackDeletesEnd for automated deletion detection (including checkpoint-based full refresh). Use when updating existing createSync code.
65building-nango-functions-remotely
Builds Nango Functions without a checked-out Nango project by calling Nango remote compile, dryrun, and deploy endpoints, resolving NANGO_SERVER_URL from env or .env, and using shared action and sync patterns. Use when creating or updating Nango actions or syncs remotely.
28nango-toolbox
Universal gateway for any third-party API or SaaS (Google Calendar, Gmail, Slack, Notion, Linear, HubSpot, etc.). TRIGGER on any request to read or modify data in an external product, even when no matching MCP tool is loaded.
25migrate-nango-deletion-detection
Migrates Nango syncs from deleteRecordsFromPreviousExecutions()/trackDeletes to trackDeletesStart/trackDeletesEnd for automated deletion detection (including checkpoint-based full refresh). Use when updating existing createSync code.
11