Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

chore: test refactor #7262

Merged
merged 1 commit into from
Mar 5, 2024
Merged
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
78 changes: 78 additions & 0 deletions test/lib/arborist-cmd.js
Original file line number Diff line number Diff line change
@@ -138,3 +138,81 @@ t.test('handle getWorkspaces raising an error', async t => {
{ message: 'oopsie' }
)
})

t.test('location detection and audit', async (t) => {
await t.test('audit false without package.json', async t => {
const { npm } = await loadMockNpm(t, {
command: 'install',
prefixDir: {
// no package.json
'readme.txt': 'just a file',
'other-dir': { a: 'a' },
},
})
t.equal(npm.config.get('location'), 'user')
t.equal(npm.config.get('audit'), false)
})

await t.test('audit true with package.json', async t => {
const { npm } = await loadMockNpm(t, {
command: 'install',
prefixDir: {
'package.json': '{ "name": "testpkg", "version": "1.0.0" }',
'readme.txt': 'just a file',
},
})
t.equal(npm.config.get('location'), 'user')
t.equal(npm.config.get('audit'), true)
})

await t.test('audit true without package.json when set', async t => {
const { npm } = await loadMockNpm(t, {
command: 'install',
prefixDir: {
// no package.json
'readme.txt': 'just a file',
'other-dir': { a: 'a' },
},
config: {
audit: true,
},
})
t.equal(npm.config.get('location'), 'user')
t.equal(npm.config.get('audit'), true)
})

await t.test('audit true in root config without package.json', async t => {
const { npm } = await loadMockNpm(t, {
command: 'install',
prefixDir: {
// no package.json
'readme.txt': 'just a file',
'other-dir': { a: 'a' },
},
// change npmRoot to get it to use a builtin rc file
otherDirs: { npmrc: 'audit=true' },
npm: ({ other }) => ({ npmRoot: other }),
})
t.equal(npm.config.get('location'), 'user')
t.equal(npm.config.get('audit'), true)
})

await t.test('test for warning when --global & --audit', async t => {
const { npm, logs } = await loadMockNpm(t, {
command: 'install',
prefixDir: {
// no package.json
'readme.txt': 'just a file',
'other-dir': { a: 'a' },
},
config: {
audit: true,
global: true,
},
})
t.equal(npm.config.get('location'), 'user')
t.equal(npm.config.get('audit'), true)
t.equal(logs.warn[0][0], 'config')
t.equal(logs.warn[0][1], 'includes both --global and --audit, which is currently unsupported.')
})
})
447 changes: 153 additions & 294 deletions test/lib/commands/install.js
Original file line number Diff line number Diff line change
@@ -1,159 +1,136 @@
const t = require('tap')
const tspawk = require('../../fixtures/tspawk')
const MockRegistry = require('@npmcli/mock-registry')
const { load: loadMockNpm } = require('../../fixtures/mock-npm')
const path = require('node:path')

t.test('exec commands', async t => {
await t.test('with args, dev=true', async t => {
const SCRIPTS = []
let ARB_ARGS = null
let REIFY_CALLED = false
let ARB_OBJ = null
const spawk = tspawk(t)

const { npm } = await loadMockNpm(t, {
mocks: {
'@npmcli/run-script': ({ event }) => {
SCRIPTS.push(event)
},
'@npmcli/arborist': function (args) {
ARB_ARGS = args
ARB_OBJ = this
this.reify = () => {
REIFY_CALLED = true
}
},
'{LIB}/utils/reify-finish.js': (_, arb) => {
if (arb !== ARB_OBJ) {
throw new Error('got wrong object passed to reify-finish')
}
},
},
config: {
// This is here because CI calls tests with `--ignore-scripts`, which config
// picks up from argv
'ignore-scripts': false,
'audit-level': 'low',
dev: true,
},
})
const abbrev = {
'package.json': '{"name": "abbrev", "version": "1.0.0"}',
test: 'test file',
}

await npm.exec('install', ['fizzbuzz'])
const packageJson = {
name: '@npmcli/test-package',
version: '1.0.0',
dependencies: {
abbrev: '^1.0.0',
},
}

t.match(
ARB_ARGS,
{ global: false, path: npm.prefix, auditLevel: null },
'Arborist gets correct args and ignores auditLevel'
)
t.equal(REIFY_CALLED, true, 'called reify')
t.strictSame(SCRIPTS, [], 'no scripts when adding dep')
})

await t.test('without args', async t => {
const SCRIPTS = []
let ARB_ARGS = null
let REIFY_CALLED = false
let ARB_OBJ = null
// tspawk calls preventUnmatched which assures that no scripts run if we don't mock any

t.test('exec commands', async t => {
await t.test('with args does not run lifecycle scripts', async t => {
const { npm } = await loadMockNpm(t, {
mocks: {
'@npmcli/run-script': ({ event }) => {
SCRIPTS.push(event)
},
'@npmcli/arborist': function (args) {
ARB_ARGS = args
ARB_OBJ = this
this.reify = () => {
REIFY_CALLED = true
}
},
'{LIB}/utils/reify-finish.js': (_, arb) => {
if (arb !== ARB_OBJ) {
throw new Error('got wrong object passed to reify-finish')
}
},
},
config: {

audit: false,
},
prefixDir: {
'package.json': JSON.stringify({
...packageJson,
scripts: {
install: 'echo install',
},
}),
abbrev,
},
})
const registry = new MockRegistry({
tap: t,
registry: npm.config.get('registry'),
})

await npm.exec('install', [])
t.match(ARB_ARGS, { global: false, path: npm.prefix })
t.equal(REIFY_CALLED, true, 'called reify')
t.strictSame(SCRIPTS, [
const manifest = registry.manifest({ name: 'abbrev' })
await registry.package({ manifest })
await registry.tarball({
manifest: manifest.versions['1.0.0'],
tarball: path.join(npm.prefix, 'abbrev'),
})

await npm.exec('install', ['abbrev'])
})

await t.test('without args runs lifecycle scripts', async t => {
const lifecycleScripts = [
'preinstall',
'install',
'postinstall',
'prepublish',
'preprepare',
'prepare',
'postprepare',
], 'exec scripts when doing local build')
})

await t.test('should ignore scripts with --ignore-scripts', async t => {
const SCRIPTS = []
let REIFY_CALLED = false
]
const scripts = {}
for (const script of lifecycleScripts) {
spawk.spawn(/.*/, a => {
runOrder.push(script)
return a.includes(`${script} lifecycle script`)
})
scripts[script] = `${script} lifecycle script`
}
const { npm } = await loadMockNpm(t, {
mocks: {
'{LIB}/utils/reify-finish.js': async () => {},
'@npmcli/run-script': ({ event }) => {
SCRIPTS.push(event)
},
'@npmcli/arborist': function () {
this.reify = () => {
REIFY_CALLED = true
}
},
},
config: {
'ignore-scripts': true,
audit: false,
},
prefixDir: {
'package.json': JSON.stringify({
...packageJson,
scripts,
}),
abbrev,
},
})
const runOrder = []
const registry = new MockRegistry({
tap: t,
registry: npm.config.get('registry'),
})
const manifest = registry.manifest({ name: 'abbrev' })
await registry.package({ manifest })
await registry.tarball({
manifest: manifest.versions['1.0.0'],
tarball: path.join(npm.prefix, 'abbrev'),
})

await npm.exec('install', [])
t.equal(REIFY_CALLED, true, 'called reify')
t.strictSame(SCRIPTS, [], 'no scripts when adding dep')
await npm.exec('install')
t.strictSame(lifecycleScripts, runOrder, 'all script ran in the correct order')
})

await t.test('should install globally using Arborist', async t => {
const SCRIPTS = []
let ARB_ARGS = null
let REIFY_CALLED
await t.test('should ignore scripts with --ignore-scripts', async t => {
const { npm } = await loadMockNpm(t, {
mocks: {
'@npmcli/run-script': ({ event }) => {
SCRIPTS.push(event)
},
'{LIB}/utils/reify-finish.js': async () => {},
'@npmcli/arborist': function (args) {
ARB_ARGS = args
this.reify = () => {
REIFY_CALLED = true
}
},
},
config: {
global: true,
'ignore-scripts': true,
audit: false,
},
prefixDir: {
'package.json': JSON.stringify({
...packageJson,
scripts: {
install: 'echo install',
},
}),
abbrev,
},
})
await npm.exec('install', [])
t.match(
ARB_ARGS,
{ global: true, path: npm.globalPrefix }
)
t.equal(REIFY_CALLED, true, 'called reify')
t.strictSame(SCRIPTS, [], 'no scripts when installing globally')
t.notOk(npm.config.get('audit', 'cli'))
const registry = new MockRegistry({
tap: t,
registry: npm.config.get('registry'),
})

const manifest = registry.manifest({ name: 'abbrev' })
await registry.package({ manifest })
await registry.tarball({
manifest: manifest.versions['1.0.0'],
tarball: path.join(npm.prefix, 'abbrev'),
})

await npm.exec('install')
})

await t.test('should not install invalid global package name', async t => {
const { npm } = await loadMockNpm(t, {
mocks: {
'@npmcli/run-script': () => {},
'{LIB}/utils/reify-finish.js': async () => {},
'@npmcli/arborist': function (args) {
throw new Error('should not reify')
},
},
config: {
global: true,
},
@@ -167,60 +144,61 @@ t.test('exec commands', async t => {

await t.test('npm i -g npm engines check success', async t => {
const { npm } = await loadMockNpm(t, {
mocks: {
'{LIB}/utils/reify-finish.js': async () => {},
'@npmcli/arborist': function () {
this.reify = () => {}
},
pacote: {
manifest: () => {
return {
version: '100.100.100',
engines: {
node: '>1',
},
}
},
prefixDir: {
npm: {
'package.json': JSON.stringify({ name: 'npm', version: '1.0.0' }),
'index.js': 'console.log("this is npm")',
},
},
config: {
global: true,
},
config: { global: true },
})
const registry = new MockRegistry({
tap: t,
registry: npm.config.get('registry'),
})
const manifest = registry.manifest({
name: 'npm',
packuments: [{ version: '1.0.0', engines: { node: '>1' } }],
})
await registry.package({ manifest, times: 2 })
await registry.tarball({
manifest: manifest.versions['1.0.0'],
tarball: path.join(npm.localPrefix, 'npm'),
})
await npm.exec('install', ['npm'])
t.ok('No exceptions happen')
})

await t.test('npm i -g npm engines check failure', async t => {
const { npm } = await loadMockNpm(t, {
mocks: {
pacote: {
manifest: () => {
return {
_id: 'npm@1.2.3',
version: '100.100.100',
engines: {
node: '>1000',
},
}
},
prefixDir: {
npm: {
'package.json': JSON.stringify({ name: 'npm', version: '1.0.0' }),
'index.js': 'console.log("this is the npm we are installing")',
},
},
config: {
global: true,
},
config: { global: true },
})
const registry = new MockRegistry({
tap: t,
registry: npm.config.get('registry'),
})
const manifest = registry.manifest({
name: 'npm',
packuments: [{ version: '1.0.0', engines: { node: '~1' } }],
})
await registry.package({ manifest })
await t.rejects(
npm.exec('install', ['npm']),
{
message: 'Unsupported engine',
pkgid: 'npm@1.2.3',
pkgid: 'npm@1.0.0',
current: {
node: process.version,
npm: '100.100.100',
npm: '1.0.0',
},
required: {
node: '>1000',
node: '~1',
},
code: 'EBADENGINE',
}
@@ -229,70 +207,29 @@ t.test('exec commands', async t => {

await t.test('npm i -g npm engines check failure forced override', async t => {
const { npm } = await loadMockNpm(t, {
mocks: {
'{LIB}/utils/reify-finish.js': async () => {},
'@npmcli/arborist': function () {
this.reify = () => {}
},
pacote: {
manifest: () => {
return {
_id: 'npm@1.2.3',
version: '100.100.100',
engines: {
node: '>1000',
},
}
},
prefixDir: {
npm: {
'package.json': JSON.stringify({ name: 'npm', version: '1.0.0' }),
'index.js': 'console.log("this is npm")',
},
},
config: {
global: true,
force: true,
},
config: { global: true, force: true },
})
await npm.exec('install', ['npm'])
t.ok('Does not throw')
})

await t.test('npm i -g npm@version engines check failure', async t => {
const { npm } = await loadMockNpm(t, {
mocks: {
'{LIB}/utils/reify-finish.js': async () => {},
'@npmcli/arborist': function () {
this.reify = () => {}
},
pacote: {
manifest: () => {
return {
_id: 'npm@1.2.3',
version: '100.100.100',
engines: {
node: '>1000',
},
}
},
},
},
config: {
global: true,
},
const registry = new MockRegistry({
tap: t,
registry: npm.config.get('registry'),
})
await t.rejects(
npm.exec('install', ['npm@100']),
{
message: 'Unsupported engine',
pkgid: 'npm@1.2.3',
current: {
node: process.version,
npm: '100.100.100',
},
required: {
node: '>1000',
},
code: 'EBADENGINE',
}
)
const manifest = registry.manifest({
name: 'npm',
packuments: [{ version: '1.0.0', engines: { node: '~1' } }],
})
await registry.package({ manifest, times: 2 })
await registry.tarball({
manifest: manifest.versions['1.0.0'],
tarball: path.join(npm.localPrefix, 'npm'),
})
await npm.exec('install', ['npm'])
t.ok('No exceptions happen')
})
})

@@ -351,81 +288,3 @@ t.test('completion', async t => {
t.strictSame(res, [])
})
})

t.test('location detection and audit', async (t) => {
await t.test('audit false without package.json', async t => {
const { npm } = await loadMockNpm(t, {
command: 'install',
prefixDir: {
// no package.json
'readme.txt': 'just a file',
'other-dir': { a: 'a' },
},
})
t.equal(npm.config.get('location'), 'user')
t.equal(npm.config.get('audit'), false)
})

await t.test('audit true with package.json', async t => {
const { npm } = await loadMockNpm(t, {
command: 'install',
prefixDir: {
'package.json': '{ "name": "testpkg", "version": "1.0.0" }',
'readme.txt': 'just a file',
},
})
t.equal(npm.config.get('location'), 'user')
t.equal(npm.config.get('audit'), true)
})

await t.test('audit true without package.json when set', async t => {
const { npm } = await loadMockNpm(t, {
command: 'install',
prefixDir: {
// no package.json
'readme.txt': 'just a file',
'other-dir': { a: 'a' },
},
config: {
audit: true,
},
})
t.equal(npm.config.get('location'), 'user')
t.equal(npm.config.get('audit'), true)
})

await t.test('audit true in root config without package.json', async t => {
const { npm } = await loadMockNpm(t, {
command: 'install',
prefixDir: {
// no package.json
'readme.txt': 'just a file',
'other-dir': { a: 'a' },
},
// change npmRoot to get it to use a builtin rc file
otherDirs: { npmrc: 'audit=true' },
npm: ({ other }) => ({ npmRoot: other }),
})
t.equal(npm.config.get('location'), 'user')
t.equal(npm.config.get('audit'), true)
})

await t.test('test for warning when --global & --audit', async t => {
const { npm, logs } = await loadMockNpm(t, {
command: 'install',
prefixDir: {
// no package.json
'readme.txt': 'just a file',
'other-dir': { a: 'a' },
},
config: {
audit: true,
global: true,
},
})
t.equal(npm.config.get('location'), 'user')
t.equal(npm.config.get('audit'), true)
t.equal(logs.warn[0][0], 'config')
t.equal(logs.warn[0][1], 'includes both --global and --audit, which is currently unsupported.')
})
})