Files
pnpm/patching/plugin-commands-patching/src/patch.ts
btea da3fa97c8d fix: should not delete state.json info after apply patch (#8401)
* fix: should not delete state.json info after apply patch

* Update .changeset/yellow-dodos-press.md

Co-authored-by: Khải <hvksmr1996@gmail.com>

* Update .changeset/yellow-dodos-press.md

Co-authored-by: Khải <hvksmr1996@gmail.com>

* test: patch-commit multiple times

* refactor: remove unused deleteEditDirState

* fix: read patch state from correct directory

---------

Co-authored-by: Khải <hvksmr1996@gmail.com>
Co-authored-by: Zoltan Kochan <z@kochan.io>
2024-08-15 02:09:49 +02:00

164 lines
4.9 KiB
TypeScript

import fs from 'fs'
import path from 'path'
import { applyPatchToDir } from '@pnpm/patching.apply-patch'
import { docsUrl } from '@pnpm/cli-utils'
import { type Config, types as allTypes } from '@pnpm/config'
import { type LogBase } from '@pnpm/logger'
import {
type CreateStoreControllerOptions,
} from '@pnpm/store-connection-manager'
import pick from 'ramda/src/pick'
import renderHelp from 'render-help'
import chalk from 'chalk'
import terminalLink from 'terminal-link'
import { PnpmError } from '@pnpm/error'
import { type ParseWantedDependencyResult } from '@pnpm/parse-wanted-dependency'
import { writePackage } from './writePackage'
import { getEditDirPath } from './getEditDirPath'
import { getPatchedDependency } from './getPatchedDependency'
import { writeEditDirState } from './stateFile'
import { tryReadProjectManifest } from '@pnpm/read-project-manifest'
export function rcOptionsTypes (): Record<string, unknown> {
return pick([], allTypes)
}
export function cliOptionsTypes (): Record<string, unknown> {
return { ...rcOptionsTypes(), 'edit-dir': String, 'ignore-existing': Boolean }
}
export const shorthands = {
d: '--edit-dir',
}
export const commandNames = ['patch']
export function help (): string {
return renderHelp({
description: 'Prepare a package for patching',
descriptionLists: [{
title: 'Options',
list: [
{
description: 'The package that needs to be modified will be extracted to this directory',
name: '--edit-dir',
},
{
description: 'Ignore existing patch files when patching',
name: '--ignore-existing',
},
],
}],
url: docsUrl('patch'),
usages: ['pnpm patch <pkg name>@<version>'],
})
}
export type PatchCommandOptions = Pick<Config,
| 'dir'
| 'registries'
| 'tag'
| 'storeDir'
| 'rootProjectManifest'
| 'lockfileDir'
| 'modulesDir'
| 'virtualStoreDir'
| 'sharedWorkspaceLockfile'
> & CreateStoreControllerOptions & {
editDir?: string
reporter?: (logObj: LogBase) => void
ignoreExisting?: boolean
}
export async function handler (opts: PatchCommandOptions, params: string[]): Promise<string> {
if (opts.editDir && fs.existsSync(opts.editDir) && fs.readdirSync(opts.editDir).length > 0) {
throw new PnpmError('PATCH_EDIT_DIR_EXISTS', `The target directory already exists: '${opts.editDir}'`)
}
if (!params[0]) {
throw new PnpmError('MISSING_PACKAGE_NAME', '`pnpm patch` requires the package name')
}
const lockfileDir = opts.lockfileDir ?? opts.dir ?? process.cwd()
const patchedDep = await getPatchedDependency(params[0], {
lockfileDir,
modulesDir: opts.modulesDir,
virtualStoreDir: opts.virtualStoreDir,
})
const modulesDir = path.join(lockfileDir, opts.modulesDir ?? 'node_modules')
const editDir = opts.editDir
? path.resolve(opts.dir, opts.editDir)
: getEditDirPath(params[0], patchedDep, { modulesDir })
if (fs.existsSync(editDir) && fs.readdirSync(editDir).length !== 0) {
throw new PnpmError('EDIT_DIR_NOT_EMPTY', `The directory ${editDir} is not empty`, {
hint: 'Either run `pnpm patch-commit` to commit or delete it then run `pnpm patch` to recreate it',
})
}
await writePackage(patchedDep, editDir, opts)
writeEditDirState({
editDir,
modulesDir: path.join(opts.dir, opts.modulesDir ?? 'node_modules'),
patchedPkg: params[0],
applyToAll: patchedDep.applyToAll,
})
if (!opts.ignoreExisting) {
let rootProjectManifest = opts.rootProjectManifest
if (!opts.sharedWorkspaceLockfile) {
const { manifest } = await tryReadProjectManifest(lockfileDir)
if (manifest) {
rootProjectManifest = manifest
}
}
if (rootProjectManifest?.pnpm?.patchedDependencies) {
tryPatchWithExistingPatchFile({
allowFailure: patchedDep.applyToAll,
patchedDep,
patchedDir: editDir,
patchedDependencies: rootProjectManifest.pnpm.patchedDependencies,
lockfileDir,
})
}
}
return `Patch: You can now edit the package at:
${terminalLink(chalk.blue(editDir), 'file://' + editDir)}
To commit your changes, run:
${chalk.green(`pnpm patch-commit '${editDir}'`)}
`
}
function tryPatchWithExistingPatchFile (
{
allowFailure,
patchedDep,
patchedDir,
patchedDependencies,
lockfileDir,
}: {
allowFailure: boolean
patchedDep: ParseWantedDependencyResult
patchedDir: string
patchedDependencies: Record<string, string>
lockfileDir: string
}
): void {
if (!patchedDep.alias || !patchedDep.pref) {
return
}
const existingPatchFile = patchedDependencies[`${patchedDep.alias}@${patchedDep.pref}`]
if (!existingPatchFile) {
return
}
const existingPatchFilePath = path.resolve(lockfileDir, existingPatchFile)
if (!fs.existsSync(existingPatchFilePath)) {
throw new PnpmError('PATCH_FILE_NOT_FOUND', `Unable to find patch file ${existingPatchFilePath}`)
}
applyPatchToDir({ patchedDir, patchFilePath: existingPatchFilePath, allowFailure })
}