Files
pnpm/resolving/npm-resolver/test/optionalDependencies.test.ts
Zoltan Kochan 6c480a4375 perf: replace node-fetch with undici (#10537)
Replace node-fetch with native undici for HTTP requests throughout pnpm.

Key changes:
- Replace node-fetch with undici's fetch() and dispatcher system
- Replace @pnpm/network.agent with a new dispatcher module in @pnpm/network.fetch
- Cache dispatchers via LRU cache keyed by connection parameters
- Handle proxies via undici ProxyAgent instead of http/https-proxy-agent
- Convert test mocking from nock to undici MockAgent where applicable
- Add minimatch@9 override to fix ESM incompatibility with brace-expansion
2026-03-29 12:44:00 +02:00

146 lines
4.7 KiB
TypeScript

/// <reference path="../../../__typings__/index.d.ts"/>
import { createFetchFromRegistry } from '@pnpm/network.fetch'
import { createNpmResolver } from '@pnpm/resolving.npm-resolver'
import type { Registries } from '@pnpm/types'
import { temporaryDirectory } from 'tempy'
import { getMockAgent, setupMockAgent, teardownMockAgent } from './utils/index.js'
const registries = {
default: 'https://registry.npmjs.org/',
} satisfies Registries
const fetch = createFetchFromRegistry({})
const getAuthHeader = () => undefined
const createResolveFromNpm = createNpmResolver.bind(null, fetch, getAuthHeader)
afterEach(async () => {
await teardownMockAgent()
})
beforeEach(async () => {
await setupMockAgent()
})
describe('optional dependencies', () => {
test('optional dependencies receive full metadata with libc field', async () => {
// This test verifies the fix for https://github.com/pnpm/pnpm/issues/9950
// Optional dependencies need full metadata to get the libc field for platform compatibility checks.
const packageMeta = {
name: 'platform-pkg',
'dist-tags': { latest: '1.0.0' },
versions: {
'1.0.0': {
name: 'platform-pkg',
version: '1.0.0',
os: ['linux'],
cpu: ['x64'],
libc: ['glibc'],
dist: {
tarball: 'https://registry.npmjs.org/platform-pkg/-/platform-pkg-1.0.0.tgz',
integrity: 'sha512-test1234567890123456789012345678901234567890123456789012345678',
},
},
},
}
// Mock the full metadata request for optional dependency
getMockAgent().get(registries.default.replace(/\/$/, ''))
.intercept({ path: '/platform-pkg', method: 'GET' })
.reply(200, packageMeta)
const { resolveFromNpm } = createResolveFromNpm({
storeDir: temporaryDirectory(),
cacheDir: temporaryDirectory(),
registries,
})
const result = await resolveFromNpm(
{
alias: 'platform-pkg',
bareSpecifier: '1.0.0',
optional: true,
},
{}
)
expect(result!.manifest!.libc).toEqual(['glibc'])
expect(result!.manifest!.os).toEqual(['linux'])
expect(result!.manifest!.cpu).toEqual(['x64'])
})
test('abbreviated and full metadata are cached separately', async () => {
// Abbreviated metadata doesn't include scripts, full metadata does.
// When resolving the same package first as regular, then as optional,
// we should get different metadata from each request.
const abbreviatedMeta = {
name: 'cache-test',
'dist-tags': { latest: '1.0.0' },
versions: {
'1.0.0': {
name: 'cache-test',
version: '1.0.0',
dist: {
tarball: 'https://registry.npmjs.org/cache-test/-/cache-test-1.0.0.tgz',
integrity: 'sha512-test1234567890123456789012345678901234567890123456789012345678',
},
},
},
}
const fullMeta = {
name: 'cache-test',
'dist-tags': { latest: '1.0.0' },
versions: {
'1.0.0': {
name: 'cache-test',
version: '1.0.0',
scripts: {
test: 'jest',
build: 'tsc',
},
dist: {
tarball: 'https://registry.npmjs.org/cache-test/-/cache-test-1.0.0.tgz',
integrity: 'sha512-test1234567890123456789012345678901234567890123456789012345678',
},
},
},
}
const mockPool = getMockAgent().get(registries.default.replace(/\/$/, ''))
// First request: abbreviated metadata for regular dependency (accept header prefers abbreviated)
mockPool.intercept({
path: '/cache-test',
method: 'GET',
headers: { accept: /application\/vnd\.npm\.install-v1\+json/ },
}).reply(200, abbreviatedMeta)
// Second request: full metadata for optional dependency (accept header prefers full JSON)
mockPool.intercept({
path: '/cache-test',
method: 'GET',
headers: { accept: /application\/json/ },
}).reply(200, fullMeta)
const cacheDir = temporaryDirectory()
const { resolveFromNpm } = createResolveFromNpm({
storeDir: temporaryDirectory(),
cacheDir,
registries,
})
// Resolve as regular dependency - should get abbreviated metadata
const regularResult = await resolveFromNpm(
{ alias: 'cache-test', bareSpecifier: '1.0.0' },
{}
)
expect(regularResult!.manifest!.scripts).toBeUndefined()
// Resolve as optional dependency - should get full metadata (separate cache entry)
const optionalResult = await resolveFromNpm(
{ alias: 'cache-test', bareSpecifier: '1.0.0', optional: true },
{}
)
expect(optionalResult!.manifest!.scripts).toEqual({ test: 'jest', build: 'tsc' })
})
})