Skip to content

Commit

Permalink
fix: show clearer error when script-shell is a .bat/.cmd file
Browse files Browse the repository at this point in the history
  • Loading branch information
gluxon committed Apr 16, 2024
1 parent 2786b54 commit 9791c45
Show file tree
Hide file tree
Showing 5 changed files with 65 additions and 2 deletions.
6 changes: 6 additions & 0 deletions cli/default-reporter/src/reportError.ts
Original file line number Diff line number Diff line change
Expand Up @@ -84,6 +84,8 @@ function getErrorInfo (logObj: Log, config?: Config, peerDependencyRules?: PeerD
case 'ERR_PNPM_FETCH_401':
case 'ERR_PNPM_FETCH_403':
return reportAuthError(err, logObj as any, config) // eslint-disable-line @typescript-eslint/no-explicit-any
case 'ERR_PNPM_INVALID_SCRIPT_SHELL_WINDOWS':
return reportInvalidScriptShellWindowsError(err)
default: {
// Errors with unknown error codes are printed with stack trace
if (!err.code?.startsWith?.('ERR_PNPM_')) {
Expand Down Expand Up @@ -441,3 +443,7 @@ Run ${chalk.yellow('pnpm dedupe')} to apply the changes above.
`,
}
}

function reportInvalidScriptShellWindowsError (_err: unknown): ErrorInfo {
return { title: 'Invalid script shell', body: 'The script-shell .npmrc option was configured to a .bat or .cmd file. These cannot be executed used as a script shell. Please try a .exe file.' }
}
2 changes: 2 additions & 0 deletions exec/lifecycle/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -43,13 +43,15 @@
"@pnpm/read-package-json": "workspace:*",
"@pnpm/store-controller-types": "workspace:*",
"@pnpm/types": "workspace:*",
"is-windows": "^1.0.2",
"path-exists": "^4.0.0",
"run-groups": "^3.0.1"
},
"devDependencies": {
"@pnpm/lifecycle": "workspace:*",
"@pnpm/test-fixtures": "workspace:*",
"@pnpm/test-ipc-server": "workspace:*",
"@types/is-windows": "^1.0.2",
"@types/rimraf": "^3.0.2",
"@zkochan/rimraf": "^2.1.3",
"load-json-file": "^6.2.0"
Expand Down
13 changes: 13 additions & 0 deletions exec/lifecycle/src/runLifecycleHook.ts
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@ import lifecycle from '@pnpm/npm-lifecycle'
import { type DependencyManifest, type ProjectManifest } from '@pnpm/types'
import { PnpmError } from '@pnpm/error'
import { existsSync } from 'fs'
import isWindows from 'is-windows'

function noop () {} // eslint-disable-line:no-empty

Expand Down Expand Up @@ -32,6 +33,10 @@ export async function runLifecycleHook (
): Promise<void> {
const optional = opts.optional === true

if (opts.scriptShell != null && isWindowsBatchFile(opts.scriptShell)) {
throw new PnpmError('ERR_PNPM_INVALID_SCRIPT_SHELL_WINDOWS', 'Cannot spawn .bat or .cmd as a script shell.')
}

const m = { _id: getId(manifest), ...manifest }
m.scripts = { ...m.scripts }

Expand Down Expand Up @@ -126,3 +131,11 @@ export async function runLifecycleHook (
function getId (manifest: ProjectManifest | DependencyManifest): string {
return `${manifest.name ?? ''}@${manifest.version ?? ''}`
}

function isWindowsBatchFile (scriptShell: string) {
// Node.js performs a similar check to determine whether it should throw
// EINVAL when spawning a .cmd/.bat file.
//
// https://github.com/nodejs/node/commit/6627222409#diff-1e725bfa950eda4d4b5c0c00a2bb6be3e5b83d819872a1adf2ef87c658273903
return isWindows() && (scriptShell.toLowerCase().endsWith('.cmd') || scriptShell.toLowerCase().endsWith('.bat'))
}
40 changes: 38 additions & 2 deletions exec/plugin-commands-script-runners/test/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,9 @@ import { DEFAULT_OPTS, REGISTRY_URL } from './utils'

const pnpmBin = path.join(__dirname, '../../../pnpm/bin/pnpm.cjs')

const skipOnWindows = isWindows() ? test.skip : test
const onlyOnWindows = !isWindows() ? test.skip : test

test('pnpm run: returns correct exit code', async () => {
prepare({
scripts: {
Expand Down Expand Up @@ -437,7 +440,7 @@ test('scripts work with PnP', async () => {
expect(fooOutput).toStrictEqual(helloWorldJsBinOutput)
})

test('pnpm run with custom shell', async () => {
skipOnWindows('pnpm run with custom shell', async () => {
prepare({
scripts: {
build: 'foo bar',
Expand All @@ -459,12 +462,45 @@ test('pnpm run with custom shell', async () => {
extraBinPaths: [],
extraEnv: {},
rawConfig: {},
scriptShell: path.resolve(`node_modules/.bin/shell-mock${isWindows() ? '.cmd' : ''}`),
scriptShell: path.resolve('node_modules/.bin/shell-mock'),
}, ['build'])

expect((await import(path.resolve('shell-input.json'))).default).toStrictEqual(['-c', 'foo bar'])
})

onlyOnWindows('pnpm shows error if script-shell is .cmd', async () => {
prepare({
scripts: {
build: 'foo bar',
},
dependencies: {
'@pnpm.e2e/shell-mock': '0.0.0',
},
})

await execa(pnpmBin, [
'install',
`--registry=${REGISTRY_URL}`,
'--store-dir',
path.resolve(DEFAULT_OPTS.storeDir),
])

async function runScript () {
await run.handler({
dir: process.cwd(),
extraBinPaths: [],
extraEnv: {},
rawConfig: {},
scriptShell: path.resolve('node_modules/.bin/shell-mock.cmd'),
}, ['build'])
}

await expect(runScript).rejects.toEqual(expect.objectContaining({
code: 'ERR_PNPM_INVALID_SCRIPT_SHELL_WINDOWS',
hint: 'Cannot spawn .bat or .cmd as a script shell.',
}))
})

test('pnpm run with RegExp script selector should work', async () => {
prepare({
scripts: {
Expand Down
6 changes: 6 additions & 0 deletions pnpm-lock.yaml

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

0 comments on commit 9791c45

Please sign in to comment.