Skip to content

Commit 62af3a1

Browse files
authoredMay 3, 2022
feat: make npm owner workspace aware (#4835)
1 parent 2bd5d7b commit 62af3a1

File tree

6 files changed

+277
-15
lines changed

6 files changed

+277
-15
lines changed
 

‎docs/content/commands/npm-owner.md

+46
Original file line numberDiff line numberDiff line change
@@ -73,6 +73,52 @@ password, npm will prompt on the command line for one.
7373
<!-- automatically generated, do not edit manually -->
7474
<!-- see lib/utils/config/definitions.js -->
7575

76+
#### `workspace`
77+
78+
* Default:
79+
* Type: String (can be set multiple times)
80+
81+
Enable running a command in the context of the configured workspaces of the
82+
current project while filtering by running only the workspaces defined by
83+
this configuration option.
84+
85+
Valid values for the `workspace` config are either:
86+
87+
* Workspace names
88+
* Path to a workspace directory
89+
* Path to a parent workspace directory (will result in selecting all
90+
workspaces within that folder)
91+
92+
When set for the `npm init` command, this may be set to the folder of a
93+
workspace which does not yet exist, to create the folder and set it up as a
94+
brand new workspace within the project.
95+
96+
This value is not exported to the environment for child processes.
97+
98+
<!-- automatically generated, do not edit manually -->
99+
<!-- see lib/utils/config/definitions.js -->
100+
101+
#### `workspaces`
102+
103+
* Default: null
104+
* Type: null or Boolean
105+
106+
Set to true to run the command in the context of **all** configured
107+
workspaces.
108+
109+
Explicitly setting this to false will cause commands like `install` to
110+
ignore workspaces altogether. When not set explicitly:
111+
112+
- Commands that operate on the `node_modules` tree (install, update, etc.)
113+
will link workspaces into the `node_modules` folder. - Commands that do
114+
other things (test, exec, publish, etc.) will operate on the root project,
115+
_unless_ one or more workspaces are specified in the `workspace` config.
116+
117+
This value is not exported to the environment for child processes.
118+
119+
<!-- automatically generated, do not edit manually -->
120+
<!-- see lib/utils/config/definitions.js -->
121+
76122
<!-- AUTOGENERATED CONFIG DESCRIPTIONS END -->
77123

78124
### See Also

‎lib/commands/owner.js

+37-14
Original file line numberDiff line numberDiff line change
@@ -22,6 +22,8 @@ class Owner extends BaseCommand {
2222
static params = [
2323
'registry',
2424
'otp',
25+
'workspace',
26+
'workspaces',
2527
]
2628

2729
static usage = [
@@ -69,22 +71,43 @@ class Owner extends BaseCommand {
6971
}
7072

7173
async exec ([action, ...args]) {
72-
switch (action) {
73-
case 'ls':
74-
case 'list':
75-
return this.ls(args[0])
76-
case 'add':
77-
return this.changeOwners(args[0], args[1], 'add')
78-
case 'rm':
79-
case 'remove':
80-
return this.changeOwners(args[0], args[1], 'rm')
81-
default:
74+
if (action === 'ls' || action === 'list') {
75+
await this.ls(args[0])
76+
} else if (action === 'add') {
77+
await this.changeOwners(args[0], args[1], 'add')
78+
} else if (action === 'rm' || action === 'remove') {
79+
await this.changeOwners(args[0], args[1], 'rm')
80+
} else {
81+
throw this.usageError()
82+
}
83+
}
84+
85+
async execWorkspaces ([action, ...args], filters) {
86+
await this.setWorkspaces(filters)
87+
// ls pkg or owner add/rm package
88+
if ((action === 'ls' && args.length > 0) || args.length > 1) {
89+
const implicitWorkspaces = this.npm.config.get('workspace', 'default')
90+
if (implicitWorkspaces.length === 0) {
91+
log.warn(`Ignoring specified workspace(s)`)
92+
}
93+
return this.exec([action, ...args])
94+
}
95+
96+
for (const [name] of this.workspaces) {
97+
if (action === 'ls' || action === 'list') {
98+
await this.ls(name)
99+
} else if (action === 'add') {
100+
await this.changeOwners(args[0], name, 'add')
101+
} else if (action === 'rm' || action === 'remove') {
102+
await this.changeOwners(args[0], name, 'rm')
103+
} else {
82104
throw this.usageError()
105+
}
83106
}
84107
}
85108

86109
async ls (pkg) {
87-
pkg = await this.getPkg(pkg)
110+
pkg = await this.getPkg(this.npm.prefix, pkg)
88111
const spec = npa(pkg)
89112

90113
try {
@@ -101,12 +124,12 @@ class Owner extends BaseCommand {
101124
}
102125
}
103126

104-
async getPkg (pkg) {
127+
async getPkg (prefix, pkg) {
105128
if (!pkg) {
106129
if (this.npm.config.get('global')) {
107130
throw this.usageError()
108131
}
109-
const { name } = await readJson(resolve(this.npm.prefix, 'package.json'))
132+
const { name } = await readJson(resolve(prefix, 'package.json'))
110133
if (!name) {
111134
throw this.usageError()
112135
}
@@ -121,7 +144,7 @@ class Owner extends BaseCommand {
121144
throw this.usageError()
122145
}
123146

124-
pkg = await this.getPkg(pkg)
147+
pkg = await this.getPkg(this.npm.prefix, pkg)
125148
log.verbose(`owner ${addOrRm}`, '%s to %s', user, pkg)
126149

127150
const spec = npa(pkg)

‎tap-snapshots/test/lib/load-all-commands.js.test.cjs

+2
Original file line numberDiff line numberDiff line change
@@ -596,6 +596,8 @@ npm owner ls [<@scope>/]<pkg>
596596
597597
Options:
598598
[--registry <registry>] [--otp <otp>]
599+
[-w|--workspace <workspace-name> [-w|--workspace <workspace-name> ...]]
600+
[-ws|--workspaces]
599601
600602
alias: author
601603

‎tap-snapshots/test/lib/utils/npm-usage.js.test.cjs

+2
Original file line numberDiff line numberDiff line change
@@ -664,6 +664,8 @@ All commands:
664664
665665
Options:
666666
[--registry <registry>] [--otp <otp>]
667+
[-w|--workspace <workspace-name> [-w|--workspace <workspace-name> ...]]
668+
[-ws|--workspaces]
667669
668670
alias: author
669671

‎test/lib/commands/owner.js

+189
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@ const t = require('tap')
22
const { load: loadMockNpm } = require('../../fixtures/mock-npm.js')
33
const MockRegistry = require('../../fixtures/mock-registry.js')
44

5+
const path = require('path')
56
const npa = require('npm-package-arg')
67
const packageName = '@npmcli/test-package'
78
const spec = npa(packageName)
@@ -12,6 +13,42 @@ const maintainers = [
1213
{ email: 'test-user-b@npmjs.org', name: 'test-user-b' },
1314
]
1415

16+
const workspaceFixture = {
17+
'package.json': JSON.stringify({
18+
name: packageName,
19+
version: '1.2.3-test',
20+
workspaces: ['workspace-a', 'workspace-b', 'workspace-c'],
21+
}),
22+
'workspace-a': {
23+
'package.json': JSON.stringify({
24+
name: 'workspace-a',
25+
version: '1.2.3-a',
26+
}),
27+
},
28+
'workspace-b': {
29+
'package.json': JSON.stringify({
30+
name: 'workspace-b',
31+
version: '1.2.3-n',
32+
}),
33+
},
34+
'workspace-c': JSON.stringify({
35+
'package.json': {
36+
name: 'workspace-n',
37+
version: '1.2.3-n',
38+
},
39+
}),
40+
}
41+
42+
function registryPackage (t, registry, name) {
43+
const mockRegistry = new MockRegistry({ tap: t, registry })
44+
45+
const manifest = mockRegistry.manifest({
46+
name,
47+
packuments: [{ maintainers, version: '1.0.0' }],
48+
})
49+
mockRegistry.package({ manifest })
50+
}
51+
1552
t.test('owner no args', async t => {
1653
const { npm } = await loadMockNpm(t)
1754
await t.rejects(
@@ -429,6 +466,158 @@ t.test('owner rm <user> no cwd package', async t => {
429466
)
430467
})
431468

469+
t.test('workspaces', async t => {
470+
t.test('owner no args --workspace', async t => {
471+
const { npm } = await loadMockNpm(t, {
472+
prefixDir: workspaceFixture,
473+
})
474+
npm.config.set('workspace', ['workspace-a'])
475+
await t.rejects(
476+
npm.exec('owner', []),
477+
{ code: 'EUSAGE' },
478+
'rejects with usage'
479+
)
480+
})
481+
482+
t.test('owner ls implicit workspace', async t => {
483+
const { npm, joinedOutput } = await loadMockNpm(t, {
484+
prefixDir: workspaceFixture,
485+
globals: ({ prefix }) => ({
486+
'process.cwd': () => path.join(prefix, 'workspace-a'),
487+
}),
488+
})
489+
registryPackage(t, npm.config.get('registry'), 'workspace-a')
490+
await npm.exec('owner', ['ls'])
491+
t.match(joinedOutput(), maintainers.map(m => `${m.name} <${m.email}>`).join('\n'))
492+
})
493+
494+
t.test('owner ls explicit workspace', async t => {
495+
const { npm, joinedOutput } = await loadMockNpm(t, {
496+
prefixDir: workspaceFixture,
497+
globals: ({ prefix }) => ({
498+
'process.cwd': () => prefix,
499+
}),
500+
})
501+
npm.config.set('workspace', ['workspace-a'])
502+
registryPackage(t, npm.config.get('registry'), 'workspace-a')
503+
await npm.exec('owner', ['ls'])
504+
t.match(joinedOutput(), maintainers.map(m => `${m.name} <${m.email}>`).join('\n'))
505+
})
506+
507+
t.test('owner ls <pkg> implicit workspace', async t => {
508+
const { npm, joinedOutput } = await loadMockNpm(t, {
509+
prefixDir: workspaceFixture,
510+
globals: ({ prefix }) => ({
511+
'process.cwd': () => path.join(prefix, 'workspace-a'),
512+
}),
513+
})
514+
registryPackage(t, npm.config.get('registry'), packageName)
515+
await npm.exec('owner', ['ls', packageName])
516+
t.match(joinedOutput(), maintainers.map(m => `${m.name} <${m.email}>`).join('\n'))
517+
})
518+
519+
t.test('owner ls <pkg> explicit workspace', async t => {
520+
const { npm, joinedOutput } = await loadMockNpm(t, {
521+
prefixDir: workspaceFixture,
522+
globals: ({ prefix }) => ({
523+
'process.cwd': () => prefix,
524+
}),
525+
})
526+
npm.config.set('workspace', ['workspace-a'])
527+
registryPackage(t, npm.config.get('registry'), packageName)
528+
await npm.exec('owner', ['ls', packageName])
529+
t.match(joinedOutput(), maintainers.map(m => `${m.name} <${m.email}>`).join('\n'))
530+
})
531+
532+
t.test('owner add implicit workspace', async t => {
533+
const { npm, joinedOutput } = await loadMockNpm(t, {
534+
prefixDir: workspaceFixture,
535+
globals: ({ prefix }) => ({
536+
'process.cwd': () => path.join(prefix, 'workspace-a'),
537+
}),
538+
})
539+
const username = 'foo'
540+
const registry = new MockRegistry({ tap: t, registry: npm.config.get('registry') })
541+
542+
const manifest = registry.manifest({
543+
name: 'workspace-a',
544+
packuments: [{ maintainers, version: '1.0.0' }],
545+
})
546+
registry.package({ manifest })
547+
registry.couchuser({ username })
548+
registry.nock.put(`/workspace-a/-rev/${manifest._rev}`, body => {
549+
t.match(body, {
550+
_id: manifest._id,
551+
_rev: manifest._rev,
552+
maintainers: [
553+
...manifest.maintainers,
554+
{ name: username, email: '' },
555+
],
556+
})
557+
return true
558+
}).reply(200, {})
559+
await npm.exec('owner', ['add', username])
560+
t.equal(joinedOutput(), `+ ${username} (workspace-a)`)
561+
})
562+
563+
t.test('owner add --workspace', async t => {
564+
const { npm, joinedOutput } = await loadMockNpm(t, {
565+
prefixDir: workspaceFixture,
566+
})
567+
npm.config.set('workspace', ['workspace-a'])
568+
const username = 'foo'
569+
const registry = new MockRegistry({ tap: t, registry: npm.config.get('registry') })
570+
571+
const manifest = registry.manifest({
572+
name: 'workspace-a',
573+
packuments: [{ maintainers, version: '1.0.0' }],
574+
})
575+
registry.package({ manifest })
576+
registry.couchuser({ username })
577+
registry.nock.put(`/workspace-a/-rev/${manifest._rev}`, body => {
578+
t.match(body, {
579+
_id: manifest._id,
580+
_rev: manifest._rev,
581+
maintainers: [
582+
...manifest.maintainers,
583+
{ name: username, email: '' },
584+
],
585+
})
586+
return true
587+
}).reply(200, {})
588+
await npm.exec('owner', ['add', username])
589+
t.equal(joinedOutput(), `+ ${username} (workspace-a)`)
590+
})
591+
592+
t.test('owner rm --workspace', async t => {
593+
const { npm, joinedOutput } = await loadMockNpm(t, {
594+
prefixDir: workspaceFixture,
595+
globals: ({ prefix }) => ({
596+
'process.cwd': () => path.join(prefix, 'workspace-a'),
597+
}),
598+
})
599+
const registry = new MockRegistry({ tap: t, registry: npm.config.get('registry') })
600+
601+
const username = maintainers[0].name
602+
const manifest = registry.manifest({
603+
name: 'workspace-a',
604+
packuments: [{ maintainers, version: '1.0.0' }],
605+
})
606+
registry.package({ manifest })
607+
registry.couchuser({ username })
608+
registry.nock.put(`/workspace-a/-rev/${manifest._rev}`, body => {
609+
t.match(body, {
610+
_id: manifest._id,
611+
_rev: manifest._rev,
612+
maintainers: maintainers.slice(1),
613+
})
614+
return true
615+
}).reply(200, {})
616+
await npm.exec('owner', ['rm', username])
617+
t.equal(joinedOutput(), `- ${username} (workspace-a)`)
618+
})
619+
})
620+
432621
t.test('completion', async t => {
433622
t.test('basic commands', async t => {
434623
const { npm } = await loadMockNpm(t)

‎test/lib/npm.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -620,7 +620,7 @@ t.test('implicit workspace rejection', async t => {
620620
}),
621621
})
622622
await t.rejects(
623-
mock.npm.exec('owner', []),
623+
mock.npm.exec('team', []),
624624
/This command does not support workspaces/
625625
)
626626
})

0 commit comments

Comments
 (0)
Please sign in to comment.