diff --git a/e2e/nx/src/import.test.ts b/e2e/nx/src/import.test.ts new file mode 100644 index 0000000000000..3bb54f7c8add2 --- /dev/null +++ b/e2e/nx/src/import.test.ts @@ -0,0 +1,41 @@ +import { + checkFilesExist, + cleanupProject, + newProject, + runCLI, +} from '@nx/e2e/utils'; + +describe('Nx Import', () => { + let proj: string; + beforeAll( + () => + (proj = newProject({ + packages: ['@nx/vue'], + unsetProjectNameAndRootFormat: false, + })) + ); + afterAll(() => cleanupProject()); + + it('should be able to import a vite app', () => { + const remote = 'https://github.com/FrozenPandaz/nx-examples.git'; + const ref = 'vue'; + const source = '.'; + const directory = 'projects/vue-app'; + runCLI( + `import ${remote} ${directory} --ref ${ref} --source ${source} --no-interactive` + ); + + checkFilesExist( + 'projects/vue-app/.gitignore', + 'projects/vue-app/package.json', + 'projects/vue-app/index.html', + 'projects/vue-app/vite.config.ts', + 'projects/vue-app/e2e/tsconfig.json', + 'projects/vue-app/e2e/vue.spec.ts', + 'projects/vue-app/src/main.ts', + 'projects/vue-app/src/App.vue', + 'projects/vue-app/public/favicon.ico' + ); + runCLI(`build created-vue-app`); + }); +}); diff --git a/packages/nx/src/command-line/import/command-object.ts b/packages/nx/src/command-line/import/command-object.ts new file mode 100644 index 0000000000000..be958ad646bdd --- /dev/null +++ b/packages/nx/src/command-line/import/command-object.ts @@ -0,0 +1,44 @@ +import { Arguments, CommandModule, MiddlewareFunction } from 'yargs'; +import { linkToNxDevAndExamples } from '../yargs-utils/documentation'; +import { withVerbose } from '../yargs-utils/shared-options'; +import { handleErrors } from '../../utils/params'; +import type { ImportOptions } from './import'; + +export const yargsImportCommand: CommandModule = { + command: 'import [sourceRemoteUrl] [destination]', + describe: false, + builder: (yargs) => + linkToNxDevAndExamples( + withVerbose( + yargs + .positional('sourceRemoteUrl', { + type: 'string', + description: 'The remote URL of the source to import', + }) + .positional('destination', { + type: 'string', + description: + 'The directory in the current workspace to import into', + }) + .option('source', { + type: 'string', + description: + 'The directory in the source repository to import from', + }) + .option('ref', { + type: 'string', + description: 'The branch from the source repository to import', + }) + ), + 'import' + ), + handler: async (args) => { + const exitCode = await handleErrors( + (args.verbose as boolean) ?? process.env.NX_VERBOSE_LOGGING === 'true', + async () => { + return (await import('./import')).importHandler(args as any); + } + ); + process.exit(exitCode); + }, +}; diff --git a/packages/nx/src/command-line/import/import.ts b/packages/nx/src/command-line/import/import.ts new file mode 100644 index 0000000000000..10e197d81ccbb --- /dev/null +++ b/packages/nx/src/command-line/import/import.ts @@ -0,0 +1,366 @@ +import { basename, dirname, join, relative } from 'path'; +import { cloneFromUpstream, GitRepository } from '../../utils/git-utils'; +import { copyFile, mkdir, rm } from 'fs'; +import { promisify } from 'util'; +import { tmpdir } from 'tmp'; +import { prompt } from 'enquirer'; +import { output } from '../../utils/output'; +import * as createSpinner from 'ora'; +import { detectPlugins, installPlugins } from '../init/init-v2'; +import { readNxJson } from '../../config/nx-json'; +import { workspaceRoot } from '../../utils/workspace-root'; +import { getPackageManagerCommand } from '../../utils/package-manager'; + +const rmAsync = promisify(rm); +const copyFileAsync = promisify(copyFile); +const mkdirAsync = promisify(mkdir); + +function wait(ms: number) { + return new Promise((resolve) => setTimeout(resolve, ms)); +} + +export interface ImportOptions { + /** + * The remote URL of the repository to import + */ + sourceRemoteUrl: string; + /** + * The branch or reference to import + */ + ref: string; + /** + * The directory in the source repo to import + */ + source: string; + /** + * The directory in the destination repo to import into + */ + destination: string; + + verbose: boolean; + interactive: boolean; +} + +const importRemoteName = '__tmp_nx_import__'; +const tempImportBranch = '__tmp_import_stage__'; +const tempFileDir = '__tmp_import_stage__'; + +async function prepareSourceRepo( + gitClient: GitRepository, + ref: string, + source: string, + destination: string +) { + const spinner = createSpinner().start(`Fetching ${ref}`); + await gitClient.addFetchRemote('origin', ref); + await gitClient.fetch('origin', ref); + spinner.succeed(`Fetched ${ref}`); + spinner.start( + `Checking out a temporary branch, ${tempImportBranch} based on ${ref}` + ); + await gitClient.checkout(tempImportBranch, { + new: true, + base: `origin/${ref}`, + }); + spinner.succeed(`Created a ${tempImportBranch} branch based on ${ref}`); + const relativeSourceDir = relative( + gitClient.root, + join(gitClient.root, source) + ); + + const destinationInSource = join(gitClient.root, destination); + spinner.start(`Moving files and git history to ${destinationInSource}`); + if (relativeSourceDir === '') { + const files = await gitClient.getGitFiles('.'); + try { + await rmAsync(destinationInSource, { + recursive: true, + }); + } catch {} + await mkdirAsync(destinationInSource, { recursive: true }); + const gitignores = new Set(); + for (const file of files) { + if (basename(file) === '.gitignore') { + gitignores.add(file); + continue; + } + spinner.start( + `Moving files and git history to ${destinationInSource}: ${file}` + ); + + const newPath = join(destinationInSource, file); + + await mkdirAsync(dirname(newPath), { recursive: true }); + try { + await gitClient.move(file, newPath); + } catch { + await wait(100); + await gitClient.move(file, newPath); + } + } + + await gitClient.commit('chore(repo): prepare for import'); + + for (const gitignore of gitignores) { + await gitClient.move(gitignore, join(destinationInSource, gitignore)); + } + await gitClient.amendCommit(); + for (const gitignore of gitignores) { + await copyFileAsync( + join(destinationInSource, gitignore), + join(gitClient.root, gitignore) + ); + } + } else { + let needsSquash = false; + const needsMove = destinationInSource !== join(gitClient.root, source); + if (needsMove) { + try { + await rmAsync(destinationInSource, { + recursive: true, + }); + await gitClient.commit('chore(repo): prepare for import'); + needsSquash = true; + } catch {} + + await mkdirAsync(destinationInSource, { recursive: true }); + } + + const files = await gitClient.getGitFiles('.'); + for (const file of files) { + if (file === '.gitignore') { + continue; + } + spinner.start( + `Moving files and git history to ${destinationInSource}: ${file}` + ); + + if (!relative(source, file).startsWith('..')) { + if (needsMove) { + const newPath = join(destinationInSource, file); + + await mkdirAsync(dirname(newPath), { recursive: true }); + try { + await gitClient.move(file, newPath); + } catch { + await wait(100); + await gitClient.move(file, newPath); + } + } + } else { + await rmAsync(join(gitClient.root, file), { + recursive: true, + }); + } + } + await gitClient.commit('chore(repo): prepare for import 2'); + if (needsSquash) { + await gitClient.squashLastTwoCommits(); + } + } + spinner.succeed(`Prepared for import`); +} + +async function confirmOrExitWithAnError(message: string) { + const { confirm } = await prompt<{ confirm: boolean }>([ + { + type: 'select', + name: 'confirm', + choices: ['Yes', 'No'], + message, + }, + ]); + + if (confirm === false) { + throw new Error('Cancelled'); + } +} + +async function mergeRemoteSource( + destinationGitClient: GitRepository, + sourceRemoteUrl: string, + branch: string, + destination: string +) { + const spinner = createSpinner(); + spinner.start(`Merged ${branch} from ${sourceRemoteUrl} into ${destination}`); + + spinner.start(`Adding ${sourceRemoteUrl} as a remote`); + await createTemporarySourceRemote( + destinationGitClient, + sourceRemoteUrl, + importRemoteName + ); + spinner.succeed(`Added ${sourceRemoteUrl} as a remote`); + + spinner.start( + `Merging files and git history from ${branch} from ${sourceRemoteUrl} into ${destination}` + ); + await destinationGitClient.mergeUnrelatedHistories( + `${importRemoteName}/${branch}`, + `feat(repo): merge ${sourceRemoteUrl}` + ); + + spinner.succeed( + `Merged files and git history from ${branch} from ${sourceRemoteUrl} into ${destination}` + ); +} + +async function createTemporarySourceRemote( + destinationGitClient: GitRepository, + sourceRemoteUrl: string, + remoteName: string +) { + try { + await destinationGitClient.deleteGitRemote(remoteName); + } catch {} + await destinationGitClient.addGitRemote(remoteName, sourceRemoteUrl); + await destinationGitClient.fetch(remoteName); +} + +export async function importHandler(options: ImportOptions) { + let { sourceRemoteUrl, ref, source, destination } = options; + + output.log({ + title: + 'Nx will walk you through the process of importing another repository into the workspace:', + bodyLines: [ + `1. Nx will clone another repository into a temporary directory`, + `2. Code to be imported will be moved to the same directory it will be imported into`, + `3. A temporary branch will be pushed to the remote repository`, + `4. The code will be merged into this workspace`, + `5. Nx will recommend plugins to integrate tools used in the import code with Nx`, + '', + `Git history will be preserved`, + ], + }); + + const tempRepoPath = join(tmpdir, 'nx-import'); + + if (!sourceRemoteUrl) { + sourceRemoteUrl = ( + await prompt<{ sourceRemoteUrl: string }>([ + { + type: 'input', + name: 'sourceRemoteUrl', + message: + 'What is the Remote URL of the repository you want to import?', + required: true, + }, + ]) + ).sourceRemoteUrl; + } + + const spinner = createSpinner( + `Cloning ${sourceRemoteUrl} into ${tempRepoPath}` + ).start(); + try { + await rmAsync(tempRepoPath, { recursive: true }); + } catch {} + await mkdirAsync(tempRepoPath, { recursive: true }); + + const sourceGitClient = await cloneFromUpstream( + sourceRemoteUrl, + join(tempRepoPath, 'repo') + ); + spinner.succeed(`Cloned into ${tempRepoPath}`); + + if (!ref) { + const branchChoices = await sourceGitClient.listBranches(); + ref = ( + await prompt<{ ref: string }>([ + { + type: 'autocomplete', + name: 'ref', + message: `Which branch do you want to import?`, + choices: branchChoices, + /** + * Limit the number of choices so that it fits on screen + */ + limit: process.stdout.rows - 3, + required: true, + } as any, + ]) + ).ref; + } + + await wait(100); + + if (!source) { + source = ( + await prompt<{ source: string }>([ + { + type: 'input', + name: 'source', + message: `Which directory do you want to import into this workspace? (leave blank to import the entire repository)`, + }, + ]) + ).source; + } + + if (!destination) { + destination = ( + await prompt<{ destination: string }>([ + { + type: 'input', + name: 'destination', + message: 'Where in this workspace should the code be imported into?', + required: true, + }, + ]) + ).destination; + } + + await prepareSourceRepo(sourceGitClient, ref, source, destination); + + console.log(await sourceGitClient.showStat()); + + output.log({ + title: `${sourceRemoteUrl} has been prepared to be imported into this repo on a temporary branch: ${tempImportBranch}`, + }); + + if (options.interactive) { + await confirmOrExitWithAnError( + `Push ${tempImportBranch} to ${sourceRemoteUrl} (git push -u -f origin ${tempImportBranch})\nAnd then import it from there into ${destination} in this workspace?` + ); + } + await sourceGitClient.push(tempImportBranch); + + // Ready to import + const destinationGitClient = new GitRepository(process.cwd()); + const mergeSpinner = createSpinner().start(`Importing `); + await mergeRemoteSource( + destinationGitClient, + sourceRemoteUrl, + tempImportBranch, + destination + ); + mergeSpinner.succeed( + `Imported ${source} from ${sourceRemoteUrl} into ${destination}!` + ); + + const pmc = getPackageManagerCommand(); + const nxJson = readNxJson(workspaceRoot); + const { plugins, updatePackageScripts } = await detectPlugins( + nxJson, + options.interactive + ); + + if (plugins.length > 0) { + output.log({ title: 'Installing Plugins' }); + installPlugins(workspaceRoot, plugins, pmc, updatePackageScripts); + + await destinationGitClient.amendCommit(); + } + + console.log(await destinationGitClient.showStat()); + + output.log({ + title: `Merging these changes`, + bodyLines: [ + `MERGE these changes when merging`, + `Do NOT squash and do NOT rebase these changes when merging these changes.`, + `If you would like to UNDO these changes, run "git reset HEAD~1 --hard"`, + ], + }); +} diff --git a/packages/nx/src/command-line/init/init-v2.ts b/packages/nx/src/command-line/init/init-v2.ts index 8085bbdfca2d3..c83e169b075fa 100644 --- a/packages/nx/src/command-line/init/init-v2.ts +++ b/packages/nx/src/command-line/init/init-v2.ts @@ -2,7 +2,10 @@ import { existsSync } from 'fs'; import { PackageJson } from '../../utils/package-json'; import { prerelease } from 'semver'; import { output } from '../../utils/output'; -import { getPackageManagerCommand } from '../../utils/package-manager'; +import { + getPackageManagerCommand, + PackageManagerCommands, +} from '../../utils/package-manager'; import { generateDotNxSetup } from './implementation/dot-nx/add-nx-scripts'; import { runNxSync } from '../../utils/child-process'; import { readJsonFile } from '../../utils/fileutils'; @@ -23,6 +26,8 @@ import { globWithWorkspaceContext } from '../../utils/workspace-context'; import { connectExistingRepoToNxCloudPrompt } from '../connect/connect-to-nx-cloud'; import { addNxToNpmRepo } from './implementation/add-nx-to-npm-repo'; import { addNxToMonorepo } from './implementation/add-nx-to-monorepo'; +import { NxJsonConfiguration, readNxJson } from '../../config/nx-json'; +import { getPackageNameFromImportPath } from '../../utils/get-package-name-from-import-path'; export interface InitArgs { interactive: boolean; @@ -31,6 +36,34 @@ export interface InitArgs { integrated?: boolean; // For Angular projects only } +export function installPlugins( + repoRoot: string, + plugins: string[], + pmc: PackageManagerCommands, + updatePackageScripts: boolean +) { + if (plugins.length === 0) { + return; + } + + addDepsToPackageJson(repoRoot, plugins); + + runInstall(repoRoot, pmc); + + output.log({ title: '🔨 Configuring plugins' }); + for (const plugin of plugins) { + execSync( + `${pmc.exec} nx g ${plugin}:init --keepExistingVersions ${ + updatePackageScripts ? '--updatePackageScripts' : '' + } --no-interactive`, + { + stdio: [0, 1, 2], + cwd: repoRoot, + } + ); + } +} + export async function initHandler(options: InitArgs): Promise { process.env.NX_RUNNING_NX_INIT = 'true'; const version = @@ -50,7 +83,8 @@ export async function initHandler(options: InitArgs): Promise { ); } generateDotNxSetup(version); - const { plugins } = await detectPlugins(); + const nxJson = readNxJson(process.cwd()); + const { plugins } = await detectPlugins(nxJson, options.interactive); plugins.forEach((plugin) => { execSync(`./nx add ${plugin}`, { stdio: 'inherit', @@ -75,10 +109,6 @@ export async function initHandler(options: InitArgs): Promise { return; } - output.log({ title: '🧐 Checking dependencies' }); - - const { plugins, updatePackageScripts } = await detectPlugins(); - const packageJson: PackageJson = readJsonFile('package.json'); if (isMonorepo(packageJson)) { await addNxToMonorepo({ @@ -104,26 +134,18 @@ export async function initHandler(options: InitArgs): Promise { createNxJsonFile(repoRoot, [], [], {}); updateGitIgnore(repoRoot); - addDepsToPackageJson(repoRoot, plugins); + const nxJson = readNxJson(repoRoot); - output.log({ title: '📦 Installing Nx' }); + output.log({ title: '🧐 Checking dependencies' }); - runInstall(repoRoot, pmc); + const { plugins, updatePackageScripts } = await detectPlugins( + nxJson, + options.interactive + ); - if (plugins.length > 0) { - output.log({ title: '🔨 Configuring plugins' }); - for (const plugin of plugins) { - execSync( - `${pmc.exec} nx g ${plugin}:init --keepExistingVersions ${ - updatePackageScripts ? '--updatePackageScripts' : '' - } --no-interactive`, - { - stdio: [0, 1, 2], - cwd: repoRoot, - } - ); - } - } + output.log({ title: '📦 Installing Nx' }); + + installPlugins(repoRoot, plugins, pmc, updatePackageScripts); if (useNxCloud) { output.log({ title: '🛠️ Setting up Nx Cloud' }); @@ -157,7 +179,10 @@ const npmPackageToPluginMap: Record = { '@remix-run/dev': '@nx/remix', }; -async function detectPlugins(): Promise<{ +export async function detectPlugins( + nxJson: NxJsonConfiguration, + interactive: boolean +): Promise<{ plugins: string[]; updatePackageScripts: boolean; }> { @@ -165,6 +190,13 @@ async function detectPlugins(): Promise<{ await globWithWorkspaceContext(process.cwd(), ['**/*/package.json']) ); + const currentPlugins = new Set( + (nxJson.plugins ?? []).map((p) => { + const plugin = typeof p === 'string' ? p : p.plugin; + return getPackageNameFromImportPath(plugin); + }) + ); + const detectedPlugins = new Set(); for (const file of files) { if (!existsSync(file)) continue; @@ -192,6 +224,13 @@ async function detectPlugins(): Promise<{ detectedPlugins.add('@nx/gradle'); } + // Remove existing plugins + for (const plugin of detectedPlugins) { + if (currentPlugins.has(plugin)) { + detectedPlugins.delete(plugin); + } + } + const plugins = Array.from(detectedPlugins); if (plugins.length === 0) { @@ -201,6 +240,22 @@ async function detectPlugins(): Promise<{ }; } + console.log({ plugins }); + + if (!interactive) { + output.log({ + title: `Recommended Plugins:`, + bodyLines: [ + `Adding these Nx plugins to integrate with the tools used in your workspace:`, + ...plugins.map((p) => `- ${p}`), + ], + }); + return { + plugins, + updatePackageScripts: true, + }; + } + output.log({ title: `Recommended Plugins:`, bodyLines: [ diff --git a/packages/nx/src/command-line/nx-commands.ts b/packages/nx/src/command-line/nx-commands.ts index 6ba7000c61781..78ac8a484c2c2 100644 --- a/packages/nx/src/command-line/nx-commands.ts +++ b/packages/nx/src/command-line/nx-commands.ts @@ -20,6 +20,7 @@ import { yargsFormatWriteCommand, } from './format/command-object'; import { yargsGenerateCommand } from './generate/command-object'; +import { yargsImportCommand } from './import/command-object'; import { yargsInitCommand } from './init/command-object'; import { yargsListCommand } from './list/command-object'; import { @@ -73,6 +74,7 @@ export const commandsObject = yargs .command(yargsFormatCheckCommand) .command(yargsFormatWriteCommand) .command(yargsGenerateCommand) + .command(yargsImportCommand) .command(yargsInitCommand) .command(yargsInternalMigrateCommand) .command(yargsListCommand) diff --git a/packages/nx/src/command-line/yargs-utils/shared-options.ts b/packages/nx/src/command-line/yargs-utils/shared-options.ts index 7f92ec50fafd0..3e673ab1339b1 100644 --- a/packages/nx/src/command-line/yargs-utils/shared-options.ts +++ b/packages/nx/src/command-line/yargs-utils/shared-options.ts @@ -124,7 +124,7 @@ export function withConfiguration(yargs: Argv) { }); } -export function withVerbose(yargs: Argv) { +export function withVerbose(yargs: Argv) { return yargs .option('verbose', { describe: diff --git a/packages/nx/src/utils/__snapshots__/git-utils.spec.ts.snap b/packages/nx/src/utils/__snapshots__/git-utils.spec.ts.snap new file mode 100644 index 0000000000000..5dc1f3c4715d9 --- /dev/null +++ b/packages/nx/src/utils/__snapshots__/git-utils.spec.ts.snap @@ -0,0 +1,37 @@ +// Jest Snapshot v1, https://goo.gl/fbAQLP + +exports[`git utils tests updateRebaseFile should squash the last 2 commits 1`] = ` +"pick 6a642190 chore(repo): hi +pick 022528d9 chore(repo): prepare for import +fixup 84ef7741 feat(repo): complete import of git@github.com:FrozenPandaz/created-vite-app.git + +# Rebase 3441f39e..84ef7741 onto 3441f39e (3 commands) +# +# Commands: +# p, pick = use commit +# r, reword = use commit, but edit the commit message +# e, edit = use commit, but stop for amending +# s, squash = use commit, but meld into previous commit +# f, fixup [-C | -c] = like "squash" but keep only the previous +# commit's log message, unless -C is used, in which case +# keep only this commit's message; -c is same as -C but +# opens the editor +# x, exec = run command (the rest of the line) using shell +# b, break = stop here (continue rebase later with 'git rebase --continue') +# d, drop = remove commit +# l, label