const fetch = require('npm-registry-fetch') const otplease = require('./utils/otplease.js') const npa = require('npm-package-arg') const semver = require('semver') const getIdentity = require('./utils/get-identity.js') const libaccess = require('libnpmaccess') const BaseCommand = require('./base-command.js') class Deprecate extends BaseCommand { static get description () { return 'Deprecate a version of a package' } /* istanbul ignore next - see test/lib/load-all-commands.js */ static get name () { return 'deprecate' } /* istanbul ignore next - see test/lib/load-all-commands.js */ static get usage () { return ['[@] '] } async completion (opts) { if (opts.conf.argv.remain.length > 1) return [] const username = await getIdentity(this.npm, this.npm.flatOptions) const packages = await libaccess.lsPackages(username, this.npm.flatOptions) return Object.keys(packages) .filter((name) => packages[name] === 'write' && (opts.conf.argv.remain.length === 0 || name.startsWith(opts.conf.argv.remain[0]))) } exec (args, cb) { this.deprecate(args) .then(() => cb()) .catch(err => cb(err.code === 'EUSAGE' ? err.message : err)) } async deprecate ([pkg, msg]) { if (!pkg || !msg) throw this.usageError() // fetch the data and make sure it exists. const p = npa(pkg) // npa makes the default spec "latest", but for deprecation // "*" is the appropriate default. const spec = p.rawSpec === '' ? '*' : p.fetchSpec if (semver.validRange(spec, true) === null) throw new Error(`invalid version range: ${spec}`) const uri = '/' + p.escapedName const packument = await fetch.json(uri, { ...this.npm.flatOptions, spec: p, query: { write: true }, }) Object.keys(packument.versions) .filter(v => semver.satisfies(v, spec, { includePrerelease: true })) .forEach(v => { packument.versions[v].deprecated = msg }) return otplease(this.npm.flatOptions, opts => fetch(uri, { ...opts, spec: p, method: 'PUT', body: packument, ignoreBody: true, })) } } module.exports = Deprecate