mirror of
https://github.com/pnpm/pnpm.git
synced 2026-02-01 10:42:28 -05:00
162 lines
4.8 KiB
TypeScript
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 })
|
|
}
|