157 lines
4.6 KiB
JavaScript
157 lines
4.6 KiB
JavaScript
|
const path = require('path')
|
||
|
const util = require('util')
|
||
|
const npa = require('npm-package-arg')
|
||
|
const libaccess = require('libnpmaccess')
|
||
|
const npmFetch = require('npm-registry-fetch')
|
||
|
const libunpub = require('libnpmpublish').unpublish
|
||
|
const readJson = util.promisify(require('read-package-json'))
|
||
|
const log = require('../utils/log-shim')
|
||
|
const otplease = require('../utils/otplease.js')
|
||
|
const getIdentity = require('../utils/get-identity.js')
|
||
|
|
||
|
const LAST_REMAINING_VERSION_ERROR = 'Refusing to delete the last version of the package. ' +
|
||
|
'It will block from republishing a new version for 24 hours.\n' +
|
||
|
'Run with --force to do this.'
|
||
|
|
||
|
const BaseCommand = require('../base-command.js')
|
||
|
class Unpublish extends BaseCommand {
|
||
|
static description = 'Remove a package from the registry'
|
||
|
static name = 'unpublish'
|
||
|
static params = ['dry-run', 'force', 'workspace', 'workspaces']
|
||
|
static usage = ['[<@scope>/]<pkg>[@<version>]']
|
||
|
|
||
|
async getKeysOfVersions (name, opts) {
|
||
|
const json = await npmFetch.json(npa(name).escapedName, opts)
|
||
|
return Object.keys(json.versions)
|
||
|
}
|
||
|
|
||
|
async completion (args) {
|
||
|
const { partialWord, conf } = args
|
||
|
|
||
|
if (conf.argv.remain.length >= 3) {
|
||
|
return []
|
||
|
}
|
||
|
|
||
|
const opts = { ...this.npm.flatOptions, log }
|
||
|
const username = await getIdentity(this.npm, { ...opts }).catch(() => null)
|
||
|
if (!username) {
|
||
|
return []
|
||
|
}
|
||
|
|
||
|
const access = await libaccess.lsPackages(username, opts)
|
||
|
// do a bit of filtering at this point, so that we don't need
|
||
|
// to fetch versions for more than one thing, but also don't
|
||
|
// accidentally unpublish a whole project
|
||
|
let pkgs = Object.keys(access || {})
|
||
|
if (!partialWord || !pkgs.length) {
|
||
|
return pkgs
|
||
|
}
|
||
|
|
||
|
const pp = npa(partialWord).name
|
||
|
pkgs = pkgs.filter(p => !p.indexOf(pp))
|
||
|
if (pkgs.length > 1) {
|
||
|
return pkgs
|
||
|
}
|
||
|
|
||
|
const versions = await this.getKeysOfVersions(pkgs[0], opts)
|
||
|
if (!versions.length) {
|
||
|
return pkgs
|
||
|
} else {
|
||
|
return versions.map(v => `${pkgs[0]}@${v}`)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
async exec (args) {
|
||
|
if (args.length > 1) {
|
||
|
throw this.usageError()
|
||
|
}
|
||
|
|
||
|
const spec = args.length && npa(args[0])
|
||
|
const force = this.npm.config.get('force')
|
||
|
const loglevel = this.npm.config.get('loglevel')
|
||
|
const silent = loglevel === 'silent'
|
||
|
const dryRun = this.npm.config.get('dry-run')
|
||
|
let pkgName
|
||
|
let pkgVersion
|
||
|
|
||
|
log.silly('unpublish', 'args[0]', args[0])
|
||
|
log.silly('unpublish', 'spec', spec)
|
||
|
|
||
|
if ((!spec || !spec.rawSpec) && !force) {
|
||
|
throw this.usageError(
|
||
|
'Refusing to delete entire project.\n' +
|
||
|
'Run with --force to do this.'
|
||
|
)
|
||
|
}
|
||
|
|
||
|
const opts = { ...this.npm.flatOptions, log }
|
||
|
if (!spec || path.resolve(spec.name) === this.npm.localPrefix) {
|
||
|
// if there's a package.json in the current folder, then
|
||
|
// read the package name and version out of that.
|
||
|
const pkgJson = path.join(this.npm.localPrefix, 'package.json')
|
||
|
let manifest
|
||
|
try {
|
||
|
manifest = await readJson(pkgJson)
|
||
|
} catch (err) {
|
||
|
if (err && err.code !== 'ENOENT' && err.code !== 'ENOTDIR') {
|
||
|
throw err
|
||
|
} else {
|
||
|
throw this.usageError()
|
||
|
}
|
||
|
}
|
||
|
|
||
|
log.verbose('unpublish', manifest)
|
||
|
|
||
|
const { name, version, publishConfig } = manifest
|
||
|
const pkgJsonSpec = npa.resolve(name, version)
|
||
|
const optsWithPub = { ...opts, publishConfig }
|
||
|
|
||
|
const versions = await this.getKeysOfVersions(name, optsWithPub)
|
||
|
if (versions.length === 1 && !force) {
|
||
|
throw this.usageError(
|
||
|
LAST_REMAINING_VERSION_ERROR
|
||
|
)
|
||
|
}
|
||
|
|
||
|
if (!dryRun) {
|
||
|
await otplease(opts, opts => libunpub(pkgJsonSpec, optsWithPub))
|
||
|
}
|
||
|
pkgName = name
|
||
|
pkgVersion = version ? `@${version}` : ''
|
||
|
} else {
|
||
|
const versions = await this.getKeysOfVersions(spec.name, opts)
|
||
|
if (versions.length === 1 && !force) {
|
||
|
throw this.usageError(
|
||
|
LAST_REMAINING_VERSION_ERROR
|
||
|
)
|
||
|
}
|
||
|
if (!dryRun) {
|
||
|
await otplease(opts, opts => libunpub(spec, opts))
|
||
|
}
|
||
|
pkgName = spec.name
|
||
|
pkgVersion = spec.type === 'version' ? `@${spec.rawSpec}` : ''
|
||
|
}
|
||
|
|
||
|
if (!silent) {
|
||
|
this.npm.output(`- ${pkgName}${pkgVersion}`)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
async execWorkspaces (args, filters) {
|
||
|
await this.setWorkspaces(filters)
|
||
|
|
||
|
const force = this.npm.config.get('force')
|
||
|
if (!force) {
|
||
|
throw this.usageError(
|
||
|
'Refusing to delete entire project(s).\n' +
|
||
|
'Run with --force to do this.'
|
||
|
)
|
||
|
}
|
||
|
|
||
|
for (const name of this.workspaceNames) {
|
||
|
await this.exec([name])
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
module.exports = Unpublish
|