Files
pnpm/patching/plugin-commands-patching/src/patch.ts
2025-09-24 10:51:53 +02:00

162 lines
4.8 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'
import renderHelp from 'render-help'
import chalk from 'chalk'
import terminalLink from 'terminal-link'
import { PnpmError } from '@pnpm/error'
import { writePackage } from './writePackage.js'
import { getEditDirPath } from './getEditDirPath.js'
import { type GetPatchedDependencyResult, getPatchedDependency } from './getPatchedDependency.js'
import { writeEditDirState } from './stateFile.js'
import isWindows from 'is-windows'
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'
| 'patchedDependencies'
| '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 quote = isWindows() ? '"' : "'"
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 ' + quote + editDir + quote + '` to commit or delete it then run `pnpm patch` to recreate it',
})
}
await writePackage(patchedDep, editDir, opts)
writeEditDirState({
editDir,
modulesDir,
patchedPkg: params[0],
applyToAll: patchedDep.applyToAll,
})
if (!opts.ignoreExisting && opts.patchedDependencies) {
tryPatchWithExistingPatchFile({
allowFailure: patchedDep.applyToAll,
patchedDep,
patchedDir: editDir,
patchedDependencies: opts.patchedDependencies,
lockfileDir,
})
}
return `Patch: You can now edit the package at:
${terminalLink(chalk.blue(editDir), 'file://' + editDir, { fallback: false })}
To commit your changes, run:
${chalk.green(`pnpm patch-commit ${quote}${editDir}${quote}`)}
`
}
function tryPatchWithExistingPatchFile (
{
allowFailure,
patchedDep: { applyToAll, alias, bareSpecifier },
patchedDir,
patchedDependencies,
lockfileDir,
}: {
allowFailure: boolean
patchedDep: GetPatchedDependencyResult
patchedDir: string
patchedDependencies: Record<string, string>
lockfileDir: string
}
): void {
if (!alias) return
let existingPatchFile: string | undefined
if (bareSpecifier) {
existingPatchFile = patchedDependencies[`${alias}@${bareSpecifier}`]
}
if (!existingPatchFile && applyToAll) {
existingPatchFile = patchedDependencies[alias]
}
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 })
}