'use strict' /* eslint-disable standard/no-callback-literal */ const BB = require('bluebird') const figgyPudding = require('figgy-pudding') const libaccess = require('libnpm/access') const npmConfig = require('./config/figgy-config.js') const output = require('./utils/output.js') const otplease = require('./utils/otplease.js') const path = require('path') const prefix = require('./npm.js').prefix const readPackageJson = BB.promisify(require('read-package-json')) const usage = require('./utils/usage.js') const whoami = require('./whoami.js') module.exports = access access.usage = usage( 'npm access', 'npm access public [<package>]\n' + 'npm access restricted [<package>]\n' + 'npm access grant <read-only|read-write> <scope:team> [<package>]\n' + 'npm access revoke <scope:team> [<package>]\n' + 'npm access 2fa-required [<package>]\n' + 'npm access 2fa-not-required [<package>]\n' + 'npm access ls-packages [<user>|<scope>|<scope:team>]\n' + 'npm access ls-collaborators [<package> [<user>]]\n' + 'npm access edit [<package>]' ) access.subcommands = [ 'public', 'restricted', 'grant', 'revoke', 'ls-packages', 'ls-collaborators', 'edit', '2fa-required', '2fa-not-required' ] const AccessConfig = figgyPudding({ json: {} }) function UsageError (msg = '') { throw Object.assign(new Error( (msg ? `\nUsage: ${msg}\n\n` : '') + access.usage ), {code: 'EUSAGE'}) } access.completion = function (opts, cb) { var argv = opts.conf.argv.remain if (argv.length === 2) { return cb(null, access.subcommands) } switch (argv[2]) { case 'grant': if (argv.length === 3) { return cb(null, ['read-only', 'read-write']) } else { return cb(null, []) } case 'public': case 'restricted': case 'ls-packages': case 'ls-collaborators': case 'edit': case '2fa-required': case '2fa-not-required': return cb(null, []) case 'revoke': return cb(null, []) default: return cb(new Error(argv[2] + ' not recognized')) } } function access ([cmd, ...args], cb) { return BB.try(() => { const fn = access.subcommands.includes(cmd) && access[cmd] if (!cmd) { UsageError('Subcommand is required.') } if (!fn) { UsageError(`${cmd} is not a recognized subcommand.`) } return fn(args, AccessConfig(npmConfig())) }).then( x => cb(null, x), err => err.code === 'EUSAGE' ? cb(err.message) : cb(err) ) } access.public = ([pkg], opts) => { return modifyPackage(pkg, opts, libaccess.public) } access.restricted = ([pkg], opts) => { return modifyPackage(pkg, opts, libaccess.restricted) } access.grant = ([perms, scopeteam, pkg], opts) => { return BB.try(() => { if (!perms || (perms !== 'read-only' && perms !== 'read-write')) { UsageError('First argument must be either `read-only` or `read-write.`') } if (!scopeteam) { UsageError('`<scope:team>` argument is required.') } const [, scope, team] = scopeteam.match(/^@?([^:]+):(.*)$/) || [] if (!scope && !team) { UsageError( 'Second argument used incorrect format.\n' + 'Example: @example:developers' ) } return modifyPackage(pkg, opts, (pkgName, opts) => { return libaccess.grant(pkgName, scopeteam, perms, opts) }, false) }) } access.revoke = ([scopeteam, pkg], opts) => { return BB.try(() => { if (!scopeteam) { UsageError('`<scope:team>` argument is required.') } const [, scope, team] = scopeteam.match(/^@?([^:]+):(.*)$/) || [] if (!scope || !team) { UsageError( 'First argument used incorrect format.\n' + 'Example: @example:developers' ) } return modifyPackage(pkg, opts, (pkgName, opts) => { return libaccess.revoke(pkgName, scopeteam, opts) }) }) } access['2fa-required'] = access.tfaRequired = ([pkg], opts) => { return modifyPackage(pkg, opts, libaccess.tfaRequired, false) } access['2fa-not-required'] = access.tfaNotRequired = ([pkg], opts) => { return modifyPackage(pkg, opts, libaccess.tfaNotRequired, false) } access['ls-packages'] = access.lsPackages = ([owner], opts) => { return ( owner ? BB.resolve(owner) : BB.fromNode(cb => whoami([], true, cb)) ).then(owner => { return libaccess.lsPackages(owner, opts) }).then(pkgs => { // TODO - print these out nicely (breaking change) output(JSON.stringify(pkgs, null, 2)) }) } access['ls-collaborators'] = access.lsCollaborators = ([pkg, usr], opts) => { return getPackage(pkg, false).then(pkgName => libaccess.lsCollaborators(pkgName, usr, opts) ).then(collabs => { // TODO - print these out nicely (breaking change) output(JSON.stringify(collabs, null, 2)) }) } access['edit'] = () => BB.reject(new Error('edit subcommand is not implemented yet')) function modifyPackage (pkg, opts, fn, requireScope = true) { return getPackage(pkg, requireScope).then(pkgName => otplease(opts, opts => fn(pkgName, opts)) ) } function getPackage (name, requireScope = true) { return BB.try(() => { if (name && name.trim()) { return name.trim() } else { return readPackageJson( path.resolve(prefix, 'package.json') ).then( data => data.name, err => { if (err.code === 'ENOENT') { throw new Error('no package name passed to command and no package.json found') } else { throw err } } ) } }).then(name => { if (requireScope && !name.match(/^@[^/]+\/.*$/)) { UsageError('This command is only available for scoped packages.') } else { return name } }) }
Name | Type | Size | Permission | Actions |
---|---|---|---|---|
auth | Folder | 0755 |
|
|
config | Folder | 0755 |
|
|
doctor | Folder | 0755 |
|
|
install | Folder | 0755 |
|
|
search | Folder | 0755 |
|
|
utils | Folder | 0755 |
|
|
access.js | File | 5.54 KB | 0644 |
|
adduser.js | File | 1.31 KB | 0644 |
|
audit.js | File | 10.56 KB | 0644 |
|
bin.js | File | 515 B | 0644 |
|
bugs.js | File | 864 B | 0644 |
|
build.js | File | 4.44 KB | 0644 |
|
cache.js | File | 4.66 KB | 0644 |
|
ci.js | File | 1.31 KB | 0644 |
|
completion.js | File | 7.11 KB | 0644 |
|
config.js | File | 7.43 KB | 0644 |
|
dedupe.js | File | 4.88 KB | 0644 |
|
deprecate.js | File | 2.11 KB | 0644 |
|
dist-tag.js | File | 4.11 KB | 0644 |
|
docs.js | File | 1.04 KB | 0644 |
|
doctor.js | File | 3.98 KB | 0644 |
|
edit.js | File | 1.37 KB | 0644 |
|
explore.js | File | 1.67 KB | 0644 |
|
fetch-package-metadata.js | File | 3.97 KB | 0644 |
|
fetch-package-metadata.md | File | 1.77 KB | 0644 |
|
fund.js | File | 4.91 KB | 0644 |
|
get.js | File | 235 B | 0644 |
|
help-search.js | File | 5.64 KB | 0644 |
|
help.js | File | 6.35 KB | 0644 |
|
hook.js | File | 4.62 KB | 0644 |
|
init.js | File | 2.74 KB | 0644 |
|
install-ci-test.js | File | 486 B | 0644 |
|
install-test.js | File | 507 B | 0644 |
|
install.js | File | 36.47 KB | 0644 |
|
link.js | File | 5.6 KB | 0644 |
|
logout.js | File | 1.26 KB | 0644 |
|
ls.js | File | 16.09 KB | 0644 |
|
npm.js | File | 14.37 KB | 0644 |
|
org.js | File | 4.18 KB | 0644 |
|
outdated.js | File | 12.28 KB | 0644 |
|
owner.js | File | 6.6 KB | 0644 |
|
pack.js | File | 11.79 KB | 0644 |
|
ping.js | File | 1.11 KB | 0644 |
|
prefix.js | File | 330 B | 0644 |
|
profile.js | File | 11.13 KB | 0644 |
|
prune.js | File | 2.23 KB | 0644 |
|
publish.js | File | 5.14 KB | 0644 |
|
rebuild.js | File | 2.09 KB | 0644 |
|
repo.js | File | 1.44 KB | 0644 |
|
restart.js | File | 64 B | 0644 |
|
root.js | File | 320 B | 0644 |
|
run-script.js | File | 5.41 KB | 0644 |
|
search.js | File | 3.36 KB | 0644 |
|
set.js | File | 276 B | 0644 |
|
shrinkwrap.js | File | 9.82 KB | 0644 |
|
star.js | File | 2.11 KB | 0644 |
|
stars.js | File | 1.03 KB | 0644 |
|
start.js | File | 62 B | 0644 |
|
stop.js | File | 61 B | 0644 |
|
substack.js | File | 509 B | 0644 |
|
team.js | File | 4.61 KB | 0644 |
|
test.js | File | 374 B | 0644 |
|
token.js | File | 6.66 KB | 0644 |
|
unbuild.js | File | 4.27 KB | 0644 |
|
uninstall.js | File | 2.21 KB | 0644 |
|
unpublish.js | File | 3.51 KB | 0644 |
|
update.js | File | 2.16 KB | 0644 |
|
version.js | File | 9.79 KB | 0644 |
|
view.js | File | 15.11 KB | 0644 |
|
visnup.js | File | 4.01 KB | 0644 |
|
whoami.js | File | 1.77 KB | 0644 |
|
xmas.js | File | 1.62 KB | 0644 |
|