fix: respect frozen-lockfile flag when migrating config dependencies (#11067)

* fix: respect frozen-lockfile flag when migrating config dependencies

* fix: throw FROZEN_LOCKFILE_WITH_OUTDATED_LOCKFILE when installing config deps with --frozen-lockfile

* fix: correct changeset package name and clean up minor issues

- Fix changeset referencing non-existent @pnpm/config.deps-installer
  (should be @pnpm/installing.env-installer)
- Fix merge artifact in AGENTS.md
- Revert unnecessary Promise.all refactoring in migrateConfigDeps.ts
- Remove extra blank line in test file

* fix: move frozenLockfile check to call site and add missing tests

Move the frozenLockfile check from migrateConfigDepsToLockfile() to
normalizeForInstall() to minimize the number of check points.

Add unit tests for all frozenLockfile code paths:
- installConfigDeps: migration fails with frozenLockfile
- resolveAndInstallConfigDeps: old-format migration, new-format
  resolution, and up-to-date lockfile success
- resolveConfigDeps: fails with frozenLockfile

* refactor: consolidate duplicate frozenLockfile checks in resolveAndInstallConfigDeps

Merge two identical frozenLockfile throw statements into a single check
covering both lockfileChanged and depsToResolve conditions.

* Delete respect-frozen-lockfile.md

* refactor: order fields

---------

Co-authored-by: Zoltan Kochan <z@kochan.io>
This commit is contained in:
Alessio Attilio
2026-03-28 18:17:52 +01:00
committed by GitHub
parent 64393a3148
commit d8be9706d9
11 changed files with 159 additions and 7 deletions

View File

@@ -0,0 +1,6 @@
---
"@pnpm/installing.env-installer": minor
"pnpm": minor
---
Throws `FROZEN_LOCKFILE_WITH_OUTDATED_LOCKFILE` when attempting to install configuration dependencies with `--frozen-lockfile` active and the env lockfile is missing or out-of-date. Previously, the operation would silently rewrite the workspace file or resolve in-memory.

View File

@@ -109,7 +109,7 @@ Example:
```
---
"@pnpm/installing.deps-installer"lling.deps-installer": minor
"@pnpm/installing.deps-installer": minor
"pnpm": minor
---

View File

@@ -18,6 +18,7 @@ import { migrateConfigDepsToLockfile } from './migrateConfigDeps.js'
import type { NormalizedConfigDep } from './parseIntegrity.js'
export interface InstallConfigDepsOpts {
frozenLockfile?: boolean
registries: Registries
rootDir: string
store: StoreController
@@ -105,6 +106,9 @@ async function normalizeForInstall (
}
// No env lockfile yet — migrate from old inline integrity format
if (opts.frozenLockfile) {
throw new PnpmError('FROZEN_LOCKFILE_WITH_OUTDATED_LOCKFILE', 'Cannot migrate configDependencies with "frozen-lockfile" because the lockfile is not up to date')
}
return migrateConfigDepsToLockfile(configDepsOrLockfile, opts)
}

View File

@@ -85,6 +85,10 @@ export async function resolveAndInstallConfigDeps (
depsToResolve.push({ name, specifier })
}
if (opts.frozenLockfile && (lockfileChanged || depsToResolve.length > 0)) {
throw new PnpmError('FROZEN_LOCKFILE_WITH_OUTDATED_LOCKFILE', 'Cannot update configDependencies with "frozen-lockfile" because the lockfile is not up to date')
}
if (depsToResolve.length === 0) {
if (lockfileChanged) {
await writeEnvLockfile(opts.rootDir, envLockfile)

View File

@@ -23,6 +23,10 @@ export type ResolveConfigDepsOpts = CreateFetchFromRegistryOptions & ResolverFac
}
export async function resolveConfigDeps (configDeps: string[], opts: ResolveConfigDepsOpts): Promise<void> {
if (opts.frozenLockfile) {
throw new PnpmError('FROZEN_LOCKFILE_WITH_OUTDATED_LOCKFILE', 'Cannot resolve configDependencies with "frozen-lockfile" because the lockfile is not up to date')
}
const fetch = createFetchFromRegistry(opts)
const getAuthHeader = createGetAuthHeaderByURI({ allSettings: opts.userConfig!, userSettings: opts.userConfig })
const { resolveFromNpm } = createNpmResolver(fetch, getAuthHeader, opts)

View File

@@ -139,6 +139,25 @@ test('migration: installs from old inline integrity format and creates env lockf
expect((envLockfile.packages['@pnpm.e2e/foo@100.0.0'].resolution as { integrity: string }).integrity).toBe(integrity)
})
test('migration fails with frozenLockfile when no env lockfile exists', async () => {
prepareEmpty()
const { storeController, storeDir } = createTempStore()
const integrity = getIntegrity('@pnpm.e2e/foo', '100.0.0')
const configDeps: Record<string, string> = {
'@pnpm.e2e/foo': `100.0.0+${integrity}`,
}
await expect(installConfigDeps(configDeps, {
registries: {
default: registry,
},
rootDir: process.cwd(),
store: storeController,
storeDir,
frozenLockfile: true,
})).rejects.toThrow('Cannot migrate configDependencies with "frozen-lockfile"')
})
test('installation fails if the config dependency does not have a checksum (old format)', async () => {
prepareEmpty()
const { storeController, storeDir } = createTempStore({

View File

@@ -201,3 +201,47 @@ test('handles mixed old-format and new-format config deps together', async () =>
expect(envLockfile!.packages['@pnpm.e2e/foo@100.0.0']).toBeDefined()
expect(envLockfile!.packages['@pnpm.e2e/bar@100.0.0']).toBeDefined()
})
test('fails with frozenLockfile when old-format deps need migration', async () => {
prepareEmpty()
const opts = createOpts()
const integrity = getIntegrity('@pnpm.e2e/foo', '100.0.0')
await expect(resolveAndInstallConfigDeps({
'@pnpm.e2e/foo': `100.0.0+${integrity}`,
}, { ...opts, frozenLockfile: true })).rejects.toThrow('Cannot update configDependencies with "frozen-lockfile"')
})
test('fails with frozenLockfile when new-format deps need resolution', async () => {
prepareEmpty()
const opts = createOpts()
await expect(resolveAndInstallConfigDeps({
'@pnpm.e2e/foo': '100.0.0',
}, { ...opts, frozenLockfile: true })).rejects.toThrow('Cannot update configDependencies with "frozen-lockfile"')
})
test('succeeds with frozenLockfile when env lockfile is up-to-date', async () => {
prepareEmpty()
const opts = createOpts()
// Pre-create complete env lockfile
const lockfile = createEnvLockfile()
lockfile.importers['.'].configDependencies['@pnpm.e2e/foo'] = {
specifier: '100.0.0',
version: '100.0.0',
}
lockfile.packages['@pnpm.e2e/foo@100.0.0'] = {
resolution: { integrity: getIntegrity('@pnpm.e2e/foo', '100.0.0') },
}
lockfile.snapshots['@pnpm.e2e/foo@100.0.0'] = {}
await writeEnvLockfile(process.cwd(), lockfile)
await resolveAndInstallConfigDeps({
'@pnpm.e2e/foo': '100.0.0',
}, { ...opts, frozenLockfile: true })
const manifest = loadJsonFileSync<{ name: string, version: string }>('node_modules/.pnpm-config/@pnpm.e2e/foo/package.json')
expect(manifest.name).toBe('@pnpm.e2e/foo')
expect(manifest.version).toBe('100.0.0')
})

View File

@@ -44,3 +44,20 @@ test('configuration dependency is resolved', async () => {
})
expect(envLockfile!.snapshots['@pnpm.e2e/foo@100.0.0']).toStrictEqual({})
})
test('fails with frozenLockfile', async () => {
prepareEmpty()
const { storeController, storeDir } = createTempStore()
await expect(resolveConfigDeps(['@pnpm.e2e/foo@100.0.0'], {
registries: {
default: registry,
},
rootDir: process.cwd(),
cacheDir: path.resolve('cache'),
userConfig: {},
store: storeController,
storeDir,
frozenLockfile: true,
})).rejects.toThrow('Cannot resolve configDependencies with "frozen-lockfile"')
})

View File

@@ -47,12 +47,17 @@ export async function getConfig (
export async function installConfigDepsAndLoadHooks (config: Config): Promise<Config> {
if (config.configDependencies) {
const store = await createStoreController(config)
await resolveAndInstallConfigDeps(config.configDependencies, {
...config,
store: store.ctrl,
storeDir: store.dir,
rootDir: config.lockfileDir ?? config.rootProjectManifestDir,
})
try {
await resolveAndInstallConfigDeps(config.configDependencies, {
...config,
store: store.ctrl,
storeDir: store.dir,
rootDir: config.lockfileDir ?? config.rootProjectManifestDir,
frozenLockfile: config.frozenLockfile,
})
} finally {
await store.ctrl.close()
}
}
if (!config.ignorePnpmfile) {
config.tryLoadDefaultPnpmfile = config.pnpmfile == null

View File

@@ -152,6 +152,7 @@ export async function main (inputArgv: string[]): Promise<void> {
const hint = err['hint'] ? err['hint'] : `For help, run: pnpm help${cmd ? ` ${cmd}` : ''}`
printError(err.message, hint)
process.exitCode = 1
await finishWorkers()
return
}
if (cmd == null && cliOptions.version) {

View File

@@ -0,0 +1,48 @@
import { readEnvLockfile } from '@pnpm/lockfile.fs'
import { prepare } from '@pnpm/prepare'
import { getIntegrity } from '@pnpm/registry-mock'
import { writeYamlFileSync } from 'write-yaml-file'
import { execPnpm, execPnpmSync } from '../utils/index.js'
test('installing configDependencies migrating to env lockfile', async () => {
prepare()
writeYamlFileSync('pnpm-workspace.yaml', {
configDependencies: {
'@pnpm.e2e/foo': '100.0.0+' + getIntegrity('@pnpm.e2e/foo', '100.0.0'),
},
})
await execPnpm(['install'])
const envLockfile = await readEnvLockfile(process.cwd())
expect(envLockfile?.importers['.'].configDependencies['@pnpm.e2e/foo'].version).toBe('100.0.0')
})
test('installing configDependencies fails with --frozen-lockfile if env lockfile is missing', async () => {
prepare()
writeYamlFileSync('pnpm-workspace.yaml', {
configDependencies: {
'@pnpm.e2e/foo': '100.0.0+' + getIntegrity('@pnpm.e2e/foo', '100.0.0'),
},
})
const result = execPnpmSync(['install', '--frozen-lockfile'])
expect(result.status).toBe(1)
expect(result.stderr.toString()).toContain('Cannot update configDependencies with "frozen-lockfile" because the lockfile is not up to date')
})
test('installing configDependencies succeeds with --frozen-lockfile if env lockfile is present and up-to-date', async () => {
prepare()
writeYamlFileSync('pnpm-workspace.yaml', {
configDependencies: {
'@pnpm.e2e/foo': '100.0.0+' + getIntegrity('@pnpm.e2e/foo', '100.0.0'),
},
})
// First install to generate the env lockfile
await execPnpm(['install'])
// Second install with frozen-lockfile should succeed
await execPnpm(['install', '--frozen-lockfile'])
})