Import, merge, or combine repositories into an Nx workspace using nx import. USE WHEN the user asks to adopt Nx across repos, move projects into a monorepo, or bring code/history from another repository.
nx import brings code from a source repository or folder into the current workspace, preserving commit history.22.6.0, nx import responds with .ndjson outputs and follow-up questions. For earlier versions, always run with --no-interactive and specify all flags directly.nx import --help for available options.libs/utils and libs/models; source has libs/ui and libs/data-access — you cannot import libs/ into libs/ directly. Import each source library individually.Primary docs:
Read the nx docs if you have the tools for it.
Subdirectory-at-a-time (nx import <source> apps --source=apps):
imported-apps/), then renameWhole repo (nx import <source> imported --source=.):
imported/nx.json, imported/tsconfig.base.json, etc.)tsconfig.base.json (projects extend it), prefix workspace globs and executor pathslibs/but dest uses packages/? Import into packages/ (nx import <source> packages/foo --source=libs/foo).Before importing, identify whether the source is an application or a library:
next.config.*, vite.config.* with a build entry point, framework-specific app scaffolding (CRA, Angular CLI app, etc.)"exports" field in package.jsonpom.xml with <packaging>jar</packaging> or <packaging>war</packaging> and a main class; Gradle application plugin or mainClass setting.csproj/.fsproj with <OutputType>Exe</OutputType> or <OutputType>WinExe</OutputType>"main"/"exports" in package.json, Maven/Gradle packaging as a library jar, .NET <OutputType>Library</OutputType>, named exports intended for import by other packages.Destination directory rules:
apps/<name>. Check workspace globs (e.g. pnpm-workspace.yaml, workspaces in root package.json) for an existing apps/* entry.
apps/* is not present, add it before importing: update the workspace glob config and commit (or stage) the change.nx import <source> apps/my-app --source=packages/my-apppackages/, libs/, etc.).nx import adds the imported directory itself (e.g. apps) to pnpm-workspace.yaml, NOT glob patterns for packages within it. Cross-package imports will fail with Cannot find module.
Fix: Replace with proper globs from the source config (e.g. apps/*, libs/shared/*), then pnpm install.
nx import does NOT merge from the source's root:
dependencies/devDependencies from package.jsontargetDefaults from nx.json (e.g. "@nx/esbuild:esbuild": { "dependsOn": ["^build"] } — critical for build ordering)namedInputs from nx.json (e.g. production exclusion patterns for test files)nx.jsonFix: Diff source and dest package.json + nx.json. Add missing deps, merge relevant targetDefaults and namedInputs.
After import, run nx sync --yes. If it reports nothing but typecheck still fails, nx reset first, then nx sync --yes again.
Inferred targets (via Nx plugins) resolve config relative to project root — no changes needed. Explicit executor targets (e.g. @nx/esbuild:esbuild) have workspace-root-relative paths (main, outputPath, tsConfig, assets, sourceRoot) that must be prefixed with the import destination directory.
nx import detects and offers to install plugins. Accept them.npx nx add @nx/PLUGIN. Check include/exclude patterns — defaults won't match alternate directories (e.g. apps-beta/).npx nx reset after any plugin config changes.Whole-repo import brings ALL source root files into the dest subdirectory. Clean up:
pnpm-lock.yaml — stale; dest has its own lockfilepnpm-workspace.yaml — source workspace config; conflicts with destnode_modules/ — stale symlinks pointing to source filesystem.gitignore — redundant with dest root .gitignorenx.json — source Nx config; dest has its ownREADME.md — optional; keep or removeDon't blindly delete tsconfig.base.json — imported projects may extend it via relative paths.
Subdirectory import doesn't bring the source's root eslint.config.mjs, but project configs reference ../../eslint.config.mjs.
Fix order:
pnpm add -wD eslint@^9 @nx/eslint-plugin typescript-eslint (plus framework-specific plugins)eslint.config.mjs (copy from source or create with @nx/eslint-plugin base rules)npx nx add @nx/eslint to register the plugin in nx.jsonInstall typescript-eslint explicitly — pnpm's strict hoisting won't auto-resolve this transitive dep of @nx/eslint-plugin.
Pin ESLint to v9 (eslint@^9.0.0). ESLint 10 breaks @nx/eslint and many plugins with cryptic errors like Cannot read properties of undefined (reading 'version').
@nx/eslint may peer-depend on ESLint 8, causing the wrong version to resolve. If lint fails with Cannot read properties of undefined (reading 'allow'), add pnpm.overrides:
{ "pnpm": { "overrides": { "eslint": "^9.0.0" } } }
After import, compare key deps (typescript, eslint, framework-specific). If dest uses newer versions, upgrade imported packages to match (usually safe). If source is newer, may need to upgrade dest first. Use pnpm.overrides to enforce single-version policy if desired.
Imported projects may lack tags. Add tags or update @nx/enforce-module-boundaries rules.
Same name in package.json across source and dest causes MultipleProjectsWithSameNameError. Fix: Rename conflicting names (e.g. @org/api → @org/teama-api), update all dep references and import statements, pnpm install. The root package.json of each imported repo also becomes a project — rename those too.
pnpm install fails during nx import if a "workspace:*" dependency hasn't been imported yet. File operations still succeed. Fix: Import all projects first, then pnpm install --no-frozen-lockfile.
.gitkeep Blocking Subdirectory ImportThe TS preset creates packages/.gitkeep. Remove it and commit before importing.
The TS preset defaults (module: "nodenext", moduleResolution: "nodenext", lib: ["es2022"]) are incompatible with frontend frameworks (React, Next.js, Vue, Vite). After importing frontend projects, verify the dest root tsconfig.base.json:
moduleResolution: Must be "bundler" (not "nodenext")module: Must be "esnext" (not "nodenext")lib: Must include "dom" and "dom.iterable" (frontend projects need these)jsx: "react-jsx" for React-only workspaces, per-project for mixed frameworksFor subdirectory imports, the dest root tsconfig is authoritative — update it. For whole-repo imports, imported projects may extend their own nested tsconfig.base.json, making this less critical.
If the dest also has backend projects needing nodenext, use per-project overrides instead of changing the root.
Gotcha: TypeScript does NOT merge lib arrays — a project-level override replaces the base array entirely. Always include all needed entries (e.g. es2022, dom, dom.iterable) in any project-level lib.
@nx/react Typings for LibrariesReact libraries generated with @nx/react:library reference @nx/react/typings/cssmodule.d.ts and @nx/react/typings/image.d.ts in their tsconfig types. These fail with Cannot find type definition file unless @nx/react is installed in the dest workspace.
Fix: pnpm add -wD @nx/react
Nx presets create jest.preset.js at the workspace root, and project jest configs reference it (e.g. ../../jest.preset.js). Subdirectory import does NOT bring this file.
Fix:
npx nx add @nx/jest — registers @nx/jest/plugin in nx.json and updates namedInputsjest.preset.js at workspace root (see references/JEST.md for content) — nx add only creates this when a generator runs, not on bare nx addpnpm add -wD jest jest-environment-jsdom ts-jest @types/jestreferences/JEST.md)For deeper Jest issues (tsconfig.spec.json, Babel transforms, CI atomization, Jest vs Vitest coexistence), see references/JEST.md.
When importing a project with existing npm scripts (build, dev, start, lint), Nx plugins auto-prefix inferred target names to avoid conflicts: e.g. next:build, vite:build, eslint:lint.
Fix: Remove the Nx-rewritten npm scripts from the imported package.json, then either:
nx run app:next:build)nx.json to use unprefixed namesWhen the source is a plain pnpm/npm workspace without nx.json.
Nx rewrites package.json scripts during init, creating broken commands (e.g. vitest run → nx test run). Fix: Remove all rewritten scripts — Nx plugins infer targets from config files.
noEmit → composite + emitDeclarationOnly (Critical)Plain TS projects use "noEmit": true, incompatible with Nx project references.
Symptoms: "typecheck target is disabled because one or more project references set 'noEmit: true'" or TS6310.
Fix in all imported tsconfigs:
"noEmit": true. If inherited via extends chain, set "noEmit": false explicitly."composite": true, "emitDeclarationOnly": true, "declarationMap": true"outDir": "dist" and "tsBuildInfoFile": "dist/tsconfig.tsbuildinfo""extends": "../../tsconfig.base.json" if missing. Remove settings now inherited from base.nx import may bring node_modules/ (pnpm symlinks pointing to the source filesystem) and pnpm-lock.yaml from the source. Both are stale.
Fix: rm -rf imported/node_modules imported/pnpm-lock.yaml imported/pnpm-workspace.yaml imported/.gitignore, then pnpm install.
.eslintrc.json (ESLint 8): Delete all .eslintrc.*, remove v8 deps, create flat eslint.config.mjs.eslint.config.js): Self-contained configs can often be left as-is.paths AliasesNx uses package.json "exports" + pnpm workspace linking instead of tsconfig "paths". If packages have proper "exports", paths are redundant. Otherwise, update paths for the new directory structure.
Identify technologies in the source repo, then read and apply the matching reference file(s).
Available references:
references/ESLINT.md — ESLint projects: duplicate lint/eslint:lint targets, legacy .eslintrc.* linting generated files, flat config .cjs self-linting, typescript-eslint v7/v9 peer dep conflict, mixed ESLint v8+v9 in one workspace.references/GRADLE.mdreferences/JEST.md — Jest testing: @nx/jest/plugin setup, jest.preset.js, testing deps by framework, tsconfig.spec.json, Jest vs Vitest coexistence, Babel transforms, CI atomization.references/NEXT.md — Next.js projects: @nx/next/plugin targets, withNx, Next.js TS config (noEmit, jsx: "preserve"), auto-installing deps via wrong PM, non-Nx create-next-app imports, mixed Next.js+Vite coexistence.references/TURBOREPO.mdreferences/VITE.md — Vite projects (React, Vue, or both): @nx/vite/plugin typecheck target, resolve.alias/__dirname fixes, framework deps, Vue-specific setup, mixed React+Vue coexistence.