From 65e22c3cb511b4ce2abec5f51f7880dbbb6ba8e8 Mon Sep 17 00:00:00 2001 From: Benjamin Maggi Date: Wed, 18 Aug 2021 18:01:37 -0300 Subject: [PATCH 01/15] fix: use npm instead of http --- package.json | 1 + src/lib/installationVerification.ts | 222 ++++++------ test/lib/installationVerification.test.ts | 390 +++++++++------------- yarn.lock | 2 +- 4 files changed, 265 insertions(+), 350 deletions(-) diff --git a/package.json b/package.json index ce3f77dd..2f700848 100644 --- a/package.json +++ b/package.json @@ -8,6 +8,7 @@ "@oclif/config": "^1.17.0", "@salesforce/command": "^3.0.5", "@salesforce/core": "^2.23.2", + "shelljs": "^0.8.4", "tslib": "^2" }, "devDependencies": { diff --git a/src/lib/installationVerification.ts b/src/lib/installationVerification.ts index 5930cb8b..e3649076 100644 --- a/src/lib/installationVerification.ts +++ b/src/lib/installationVerification.ts @@ -15,6 +15,7 @@ import { Readable } from 'stream'; import { parse as parseUrl, URL, UrlWithStringQuery } from 'url'; import { promisify as utilPromisify } from 'util'; import * as crypto from 'crypto'; +import * as shelljs from 'shelljs'; import { Logger, fs, SfdxError } from '@salesforce/core'; import { get } from '@salesforce/ts-types'; import * as request from 'request'; @@ -23,13 +24,8 @@ import { NpmName } from './NpmName'; const CRYPTO_LEVEL = 'RSA-SHA256'; export const ALLOW_LIST_FILENAME = 'unsignedPluginAllowList.json'; export const DEFAULT_REGISTRY = 'https://registry.npmjs.org/'; -export type IRequest = (url: string, cb?: request.RequestCallback) => Readable; -type Version = { - sfdx: NpmMeta; - dist: { - tarball: string; - }; -}; +export type IRequest = (url: string, cb?: request.RequestCallback) => void; + export interface ConfigContext { configDir?: string; cacheDir?: string; @@ -39,6 +35,21 @@ export interface Verifier { verify(): Promise; isAllowListed(): Promise; } + +export type NpmShowResults = { + versions: string[]; + 'dist-tags': { + [name: string]: string; + }; + sfdx?: { + publicKeyUrl: string; + signatureUrl: string; + }; + dist?: { + [name: string]: string; + }; +}; + export class CodeVerifierInfo { private signature: Readable; private publicKey: Readable; @@ -149,6 +160,9 @@ export class NpmMeta { public publicKeyUrl: string; public tarballLocalPath: string; public verified: boolean; + public moduleName: string; + public version: string; + public tarballFilename: string; } /** @@ -310,8 +324,11 @@ export class InstallationVerification implements Verifier { public async streamTagGz(): Promise { const logger = await this.getLogger(); const npmMeta = await this.retrieveNpmMeta(); + const npmCmd = `npm pack ${npmMeta.moduleName}@${npmMeta.version} --registry ${getNpmRegistry().href}`; + logger.debug(`npmCmd: ${npmCmd}`); const urlObject: URL = new URL(npmMeta.tarballUrl); const urlPathsAsArray = urlObject.pathname.split('/'); + npmMeta.tarballFilename = npmMeta.moduleName.replace(/@/g, ''); logger.debug(`streamTagGz | urlPathsAsArray: ${urlPathsAsArray.join(',')}`); const fileNameStr: string = urlPathsAsArray[urlPathsAsArray.length - 1]; @@ -320,27 +337,18 @@ export class InstallationVerification implements Verifier { // Make sure the cache path exists. try { await fs.mkdirp(this.getCachePath()); + shelljs.pushd(this.getCachePath()); + shelljs.exec(npmCmd, { silent: true, fatal: true }); + const tarBallFile = fs + .readdirSync(this.getCachePath(), { withFileTypes: true }) + .find((entry) => entry.isFile() && entry.name.includes(npmMeta.version)); + npmMeta.tarballLocalPath = `${this.getCachePath()}${path.sep}${tarBallFile.name}`; + shelljs.popd(); } catch (err) { logger.debug(err); } - return new Promise((resolve, reject) => { - const cacheFilePath = path.join(this.getCachePath(), fileNameStr); - logger.debug(`streamTagGz | cacheFilePath: ${cacheFilePath}`); - - const writeStream = this.fsImpl.createWriteStream(cacheFilePath, { encoding: 'binary' }); - this.requestImpl(npmMeta.tarballUrl) - .on('end', () => { - logger.debug('streamTagGz | Finished writing tgz file'); - npmMeta.tarballLocalPath = cacheFilePath; - return resolve(npmMeta); - }) - .on('error', (err) => { - logger.debug(err); - return reject(err); - }) - .pipe(writeStream); - }); + return npmMeta; } // this is generally $HOME/.config/sfdx @@ -366,108 +374,96 @@ export class InstallationVerification implements Verifier { logger.debug(`retrieveNpmMeta | this.pluginNpmName.scope: ${this.pluginNpmName.scope}`); logger.debug(`retrieveNpmMeta | this.pluginNpmName.tag: ${this.pluginNpmName.tag}`); - if (this.pluginNpmName.scope) { - npmRegistry.pathname = path.join( - npmRegistry.pathname, - `@${this.pluginNpmName.scope}%2f${this.pluginNpmName.name}` + const npmShowModule = this.pluginNpmName.scope + ? `@${this.pluginNpmName.scope}/${this.pluginNpmName.name}` + : this.pluginNpmName.name; + + // run npm show to get metadata + const npmCmd = `npm show ${npmShowModule} --registry ${npmRegistry.href} --json`; + let npmMetadata: NpmShowResults; + try { + const npmShowResult = shelljs.exec(npmCmd, { silent: true, fatal: true }); + npmMetadata = JSON.parse(npmShowResult.stdout); + } catch (error) { + return reject(new SfdxError(error, 'ShellExecError')); + } + const meta: NpmMeta = new NpmMeta(); + meta.moduleName = npmShowModule; + logger.debug('retrieveNpmMeta | Found npm meta information.'); + if (!npmMetadata.versions) { + return reject( + new SfdxError( + `The npm metadata for plugin ${this.pluginNpmName.name} is missing the versions attribute.`, + 'InvalidNpmMetadata' + ) ); - } else { - npmRegistry.pathname = path.join(npmRegistry.pathname, this.pluginNpmName.name); } - logger.debug(`retrieveNpmMeta | npmRegistry.pathname: ${npmRegistry.pathname}`); - this.requestImpl(npmRegistry.href, (err, response, body) => { - if (err) { - return reject(err); - } - if (response && response.statusCode === 200) { - logger.debug('retrieveNpmMeta | Found npm meta information. Parsing.'); - const responseObj = JSON.parse(body); + // Assume the tag is version tag. + let versionNumber = npmMetadata.versions.find((version) => version === this.pluginNpmName.tag); + + logger.debug(`retrieveNpmMeta | versionObject: ${JSON.stringify(versionNumber)}`); + + // If the assumption was not correct the tag must be a non-versioned dist-tag or not specified. + if (!versionNumber) { + // Assume dist-tag; + const distTags = npmMetadata['dist-tags']; + logger.debug(`retrieveNpmMeta | distTags: ${JSON.stringify(distTags)}`); + if (distTags) { + const tagVersionStr: string = get(distTags, this.pluginNpmName.tag) as string; + logger.debug(`retrieveNpmMeta | tagVersionStr: ${tagVersionStr}`); - // Make sure the response has a version attribute - if (!responseObj.versions) { + // if we got a dist tag hit look up the version object + if (tagVersionStr && tagVersionStr.length > 0 && tagVersionStr.includes('.')) { + versionNumber = npmMetadata.versions.find((version) => version === tagVersionStr); + logger.debug(`retrieveNpmMeta | versionObject: ${versionNumber}`); + } else { return reject( new SfdxError( - `The npm metadata for plugin ${this.pluginNpmName.name} is missing the versions attribute.`, - 'InvalidNpmMetadata' + `The dist tag ${this.pluginNpmName.tag} was not found for plugin: ${this.pluginNpmName.name}`, + 'NpmTagNotFound' ) ); } + } else { + throw new SfdxError('The deployed NPM is missing dist-tags.', 'UnexpectedNpmFormat'); + } + } - // Assume the tag is version tag. - let versionObject: Version = responseObj.versions[this.pluginNpmName.tag]; - - logger.debug(`retrieveNpmMeta | versionObject: ${JSON.stringify(versionObject)}`); - - // If the assumption was not correct the tag must be a non-versioned dist-tag or not specified. - if (!versionObject) { - // Assume dist-tag; - const distTags: string = get(responseObj, 'dist-tags') as string; - logger.debug(`retrieveNpmMeta | distTags: ${distTags}`); - if (distTags) { - const tagVersionStr: string = get(distTags, this.pluginNpmName.tag) as string; - logger.debug(`retrieveNpmMeta | tagVersionStr: ${tagVersionStr}`); - - // if we got a dist tag hit look up the version object - if (tagVersionStr && tagVersionStr.length > 0 && tagVersionStr.includes('.')) { - versionObject = responseObj.versions[tagVersionStr]; - logger.debug(`retrieveNpmMeta | versionObject: ${JSON.stringify(versionObject)}`); - } else { - return reject( - new SfdxError( - `The dist tag ${this.pluginNpmName.tag} was not found for plugin: ${this.pluginNpmName.name}`, - 'NpmTagNotFound' - ) - ); - } - } else { - return reject(new SfdxError('The deployed NPM is missing dist-tags.', 'UnexpectedNpmFormat')); - } - } + meta.version = versionNumber; - if (!(versionObject && versionObject.sfdx)) { - return reject(new SfdxError('This plugin is not signed by Salesforce.com, Inc.', 'NotSigned')); - } else { - const meta: NpmMeta = new NpmMeta(); - if (!validSalesforceHostname(versionObject.sfdx.publicKeyUrl)) { - throw new SfdxError( - `The host is not allowed to provide signing information. [${versionObject.sfdx.publicKeyUrl}]`, - 'UnexpectedHost' - ); - } else { - logger.debug(`retrieveNpmMeta | versionObject.sfdx.publicKeyUrl: ${versionObject.sfdx.publicKeyUrl}`); - meta.publicKeyUrl = versionObject.sfdx.publicKeyUrl; - } - - if (!validSalesforceHostname(versionObject.sfdx.signatureUrl)) { - throw new SfdxError( - `The host is not allowed to provide signing information. [${versionObject.sfdx.signatureUrl}]`, - 'UnexpectedHost' - ); - } else { - logger.debug(`retrieveNpmMeta | versionObject.sfdx.signatureUrl: ${versionObject.sfdx.signatureUrl}`); - meta.signatureUrl = versionObject.sfdx.signatureUrl; - } - - meta.tarballUrl = versionObject.dist.tarball; - logger.debug(`retrieveNpmMeta | meta.tarballUrl: ${meta.tarballUrl}`); - - return resolve(meta); - } + if (!npmMetadata.sfdx) { + throw new SfdxError('This plugin is not signed by Salesforce.com, Inc.', 'NotSigned'); + } else { + if (!validSalesforceHostname(npmMetadata.sfdx.publicKeyUrl)) { + return reject( + new SfdxError( + `The host is not allowed to provide signing information. [${npmMetadata.sfdx.publicKeyUrl}]`, + 'UnexpectedHost' + ) + ); } else { - switch (response.statusCode) { - case 403: - throw new SfdxError(`Access to the plugin was denied. url: ${npmRegistry.href}`, 'PluginAccessDenied'); - case 404: - throw new SfdxError(`The plugin requested was not found. url: ${npmRegistry.href}.`, 'PluginNotFound'); - default: - throw new SfdxError( - `The url request returned ${response.statusCode as string} - ${npmRegistry.href}`, - 'UrlRetrieve' - ); - } + logger.debug(`retrieveNpmMeta | versionObject.sfdx.publicKeyUrl: ${npmMetadata.sfdx.publicKeyUrl}`); + meta.publicKeyUrl = npmMetadata.sfdx.publicKeyUrl; } - }); + + if (!validSalesforceHostname(npmMetadata.sfdx.signatureUrl)) { + return reject( + new SfdxError( + `The host is not allowed to provide signing information. [${npmMetadata.sfdx.signatureUrl}]`, + 'UnexpectedHost' + ) + ); + } else { + logger.debug(`retrieveNpmMeta | versionObject.sfdx.signatureUrl: ${npmMetadata.sfdx.signatureUrl}`); + meta.signatureUrl = npmMetadata.sfdx.signatureUrl; + } + + meta.tarballUrl = npmMetadata.dist.tarball; + logger.debug(`retrieveNpmMeta | meta.tarballUrl: ${meta.tarballUrl}`); + + return resolve(meta); + } }); } diff --git a/test/lib/installationVerification.test.ts b/test/lib/installationVerification.test.ts index ff841849..e4ac3b53 100644 --- a/test/lib/installationVerification.test.ts +++ b/test/lib/installationVerification.test.ts @@ -7,9 +7,10 @@ import { Readable, Writable } from 'stream'; import { expect } from 'chai'; import * as request from 'request'; - +import * as shelljs from 'shelljs'; +import { stubMethod } from '@salesforce/ts-sinon'; import { SfdxError } from '@salesforce/core'; - +import Sinon = require('sinon'); import { ConfigContext, DEFAULT_REGISTRY, @@ -20,12 +21,30 @@ import { NpmMeta, VerificationConfig, Verifier, + NpmShowResults, } from '../../src/lib/installationVerification'; import { NpmName } from '../../src/lib/NpmName'; import { CERTIFICATE, TEST_DATA, TEST_DATA_SIGNATURE } from '../testCert'; const BLANK_PLUGIN = { plugin: '', tag: '' }; +const getShelljsExecStub = (sandbox: sinon.SinonSandbox, npmMetadata: NpmShowResults): Sinon.SinonStub => { + return stubMethod(sandbox, shelljs, 'exec').callsFake((cmd: string) => { + expect(cmd).to.be.a('string').and.not.to.be.empty; + if (cmd.includes('show')) { + return { + stdout: JSON.stringify(npmMetadata), + }; + } else if (cmd.includes('pack')) { + return { + stdout: undefined, + }; + } else { + throw new Error(`Unexpected test cmd - ${cmd}`); + } + }); +}; + describe('getNpmRegistry', () => { const currentRegistry = process.env.SFDX_NPM_REGISTRY; after(() => { @@ -60,11 +79,20 @@ describe('InstallationVerification Tests', () => { }; const currentRegistry = process.env.SFDX_NPM_REGISTRY; let plugin: NpmName; + let sandbox: sinon.SinonSandbox; + let shelljsExecStub: Sinon.SinonStub; beforeEach(() => { + sandbox = Sinon.createSandbox(); plugin = NpmName.parse('foo'); }); + afterEach(() => { + if (shelljsExecStub) { + shelljsExecStub.restore(); + } + }); + after(() => { if (currentRegistry) { process.env.SFDX_NPM_REGISTRY = currentRegistry; @@ -78,40 +106,27 @@ describe('InstallationVerification Tests', () => { }); it('Steel thread test', async () => { - const iRequest: IRequest = (url: string, cb?: request.RequestCallback): Readable => { - if (url.includes('foo.tgz')) { - const reader = new Readable({ - read() {}, - }); - process.nextTick(() => { - reader.emit('end'); - }); - return reader; - } else if (url.includes('sig')) { + const npmMetadata: NpmShowResults = { + versions: ['1.0.0'], + 'dist-tags': { + latest: '1.0.0', + }, + sfdx: { + publicKeyUrl: 'https://developer.salesforce.com/crt', + signatureUrl: 'https://developer.salesforce.com/sig', + }, + dist: { + tarball: 'https://registry.example.com/foo.tgz', + }, + }; + + shelljsExecStub = getShelljsExecStub(sandbox, npmMetadata); + + const iRequest: IRequest = (url: string, cb?: request.RequestCallback): void => { + if (url.includes('sig')) { cb(null, { statusCode: 200 } as request.Response, TEST_DATA_SIGNATURE); - } else if (url.includes('key')) { + } else if (url.includes('crt')) { cb(null, { statusCode: 200 } as request.Response, CERTIFICATE); - } else if (url.endsWith(plugin.name)) { - cb( - null, - { statusCode: 200 } as request.Response, - JSON.stringify({ - versions: { - '1.2.3': { - sfdx: { - publicKeyUrl: 'https://developer.salesforce.com/key', - signatureUrl: 'https://developer.salesforce.com/sig', - }, - dist: { - tarball: 'https://registry.example.com/foo.tgz', - }, - }, - }, - 'dist-tags': { - latest: '1.2.3', - }, - }) - ); } else { throw new Error(`Unexpected test url - ${url}`); } @@ -145,6 +160,22 @@ describe('InstallationVerification Tests', () => { }); it('Steel thread version - version number', async () => { + const npmMetadata: NpmShowResults = { + versions: ['1.0.0', '1.0.1'], + 'dist-tags': { + latest: '1.0.1', + }, + sfdx: { + publicKeyUrl: 'https://developer.salesforce.com/crt', + signatureUrl: 'https://developer.salesforce.com/sig', + }, + dist: { + tarball: 'https://registry.example.com/foo.tgz', + }, + }; + + shelljsExecStub = getShelljsExecStub(sandbox, npmMetadata); + const iRequest: IRequest = (url: string, cb?: request.RequestCallback): Readable => { if (url.includes('foo.tgz')) { const reader = new Readable({ @@ -156,38 +187,8 @@ describe('InstallationVerification Tests', () => { return reader; } else if (url.includes('sig')) { cb(null, { statusCode: 200 } as request.Response, TEST_DATA_SIGNATURE); - } else if (url.includes('key')) { + } else if (url.includes('crt')) { cb(null, { statusCode: 200 } as request.Response, CERTIFICATE); - } else if (url.endsWith(plugin.name)) { - cb( - null, - { statusCode: 200 } as request.Response, - JSON.stringify({ - versions: { - '1.2.3': { - sfdx: { - publicKeyUrl: 'https://developer.salesforce.com/key', - signatureUrl: 'https://developer.salesforce.com/sig', - }, - dist: { - tarball: 'https://registry.example.com/foo.tgz', - }, - }, - '1.2.4': { - sfdx: { - publicKeyUrl: 'https://developer.salesforce.com/key1', - signatureUrl: 'https://developer.salesforce.com/sig1', - }, - dist: { - tarball: 'https://registry.example.com/foo1.tgz', - }, - }, - }, - 'dist-tags': { - latest: '1.2.4', - }, - }) - ); } else { throw new Error(`Unexpected test url - ${url}`); } @@ -213,7 +214,7 @@ describe('InstallationVerification Tests', () => { }, }; - plugin.tag = '1.2.3'; + plugin.tag = '1.0.0'; const verification = new InstallationVerification(iRequest, fs).setPluginNpmName(plugin).setConfig(config); return verification.verify().then((meta: NpmMeta) => { @@ -222,50 +223,27 @@ describe('InstallationVerification Tests', () => { }); it('Steel thread version - tag name', async () => { - const iRequest: IRequest = (url: string, cb?: request.RequestCallback): Readable => { - if (url.includes('foo.tgz')) { - const reader = new Readable({ - read() {}, - }); - process.nextTick(() => { - reader.emit('end'); - }); - return reader; - } else if (url.includes('sig.weaver')) { + const npmMetadata: NpmShowResults = { + versions: ['1.0.0', '1.0.1'], + 'dist-tags': { + latest: '1.0.1', + gozer: '1.0.0', + }, + sfdx: { + publicKeyUrl: 'https://developer.salesforce.com/crt.master', + signatureUrl: 'https://developer.salesforce.com/sig.weaver', + }, + dist: { + tarball: 'https://registry.example.com/foo.tgz', + }, + }; + + shelljsExecStub = getShelljsExecStub(sandbox, npmMetadata); + const iRequest: IRequest = (url: string, cb?: request.RequestCallback) => { + if (url.includes('sig.weaver')) { cb(null, { statusCode: 200 } as request.Response, TEST_DATA_SIGNATURE); - } else if (url.includes('key.master')) { + } else if (url.includes('crt.master')) { cb(null, { statusCode: 200 } as request.Response, CERTIFICATE); - } else if (url.endsWith(plugin.name)) { - cb( - null, - { statusCode: 200 } as request.Response, - JSON.stringify({ - versions: { - '1.2.3': { - sfdx: { - publicKeyUrl: 'https://developer.salesforce.com/key.master', - signatureUrl: 'https://developer.salesforce.com/sig.weaver', - }, - dist: { - tarball: 'https://registry.example.com/foo.tgz', - }, - }, - '1.2.4': { - sfdx: { - publicKeyUrl: 'https://developer.salesforce.com/key1', - signatureUrl: 'https://developer.salesforce.com/sig1', - }, - dist: { - tarball: 'https://registry.example.com/foo1.tgz', - }, - }, - }, - 'dist-tags': { - latest: '1.2.4', - gozer: '1.2.3', - }, - }) - ); } else { throw new Error(`Unexpected test url - ${url}`); } @@ -304,51 +282,28 @@ describe('InstallationVerification Tests', () => { const TEST_REG = 'https://example.com/registry'; process.env.SFDX_NPM_REGISTRY = TEST_REG; - const iRequest: IRequest = (url: string, cb?: request.RequestCallback): Readable => { - if (url.includes('foo.tgz')) { - const reader = new Readable({ - read() {}, - }); - process.nextTick(() => { - reader.emit('end'); - }); - return reader; - } else if (url.includes('sig.weaver')) { + const npmMetadata: NpmShowResults = { + versions: ['1.0.0', '1.0.1'], + 'dist-tags': { + latest: '1.0.1', + gozer: '1.0.0', + }, + sfdx: { + publicKeyUrl: 'https://developer.salesforce.com/crt.master', + signatureUrl: 'https://developer.salesforce.com/sig.weaver', + }, + dist: { + tarball: 'https://example.com/registry/foo.tgz', + }, + }; + + shelljsExecStub = getShelljsExecStub(sandbox, npmMetadata); + + const iRequest: IRequest = (url: string, cb?: request.RequestCallback) => { + if (url.includes('sig.weaver')) { cb(null, { statusCode: 200 } as request.Response, TEST_DATA_SIGNATURE); - } else if (url.includes('key.master')) { + } else if (url.includes('crt.master')) { cb(null, { statusCode: 200 } as request.Response, CERTIFICATE); - } else if (url.endsWith(plugin.name)) { - expect(url).to.include(TEST_REG); - cb( - null, - { statusCode: 200 } as request.Response, - JSON.stringify({ - versions: { - '1.2.3': { - sfdx: { - publicKeyUrl: 'https://developer.salesforce.com/key.master', - signatureUrl: 'https://developer.salesforce.com/sig.weaver', - }, - dist: { - tarball: 'https://example.com/registry/foo.tgz', - }, - }, - '1.2.4': { - sfdx: { - publicKeyUrl: 'https://developer.salesforce.com/key1', - signatureUrl: 'https://developer.salesforce.com/sig1', - }, - dist: { - tarball: 'https://example.com/registry/foo1.tgz', - }, - }, - }, - 'dist-tags': { - latest: '1.2.4', - gozer: '1.2.3', - }, - }) - ); } else { throw new Error(`Unexpected test url - ${url}`); } @@ -384,21 +339,15 @@ describe('InstallationVerification Tests', () => { }); it('InvalidNpmMetadata', async () => { - const iRequest: IRequest = (url: string, cb?: request.RequestCallback): Readable => { - if (url.includes('foo.tgz')) { - const reader = new Readable({ - read() {}, - }); - process.nextTick(() => { - reader.emit('end'); - }); - return reader; - } else if (url.includes('sig')) { + const npmMetadata = {} as unknown as NpmShowResults; + + shelljsExecStub = getShelljsExecStub(sandbox, npmMetadata); + + const iRequest: IRequest = (url: string, cb?: request.RequestCallback) => { + if (url.includes('sig')) { cb(null, { statusCode: 200 } as request.Response, TEST_DATA_SIGNATURE); } else if (url.includes('key')) { cb(null, { statusCode: 200 } as request.Response, CERTIFICATE); - } else if (url.endsWith(plugin.name)) { - cb(null, { statusCode: 200 } as request.Response, JSON.stringify({})); } else { throw new Error(`Unexpected test url - ${url}`); } @@ -422,36 +371,24 @@ describe('InstallationVerification Tests', () => { }); it('Not Signed', async () => { - const iRequest: IRequest = (url: string, cb?: request.RequestCallback): Readable => { - if (url.includes('foo.tgz')) { - const reader = new Readable({ - read() {}, - }); - process.nextTick(() => { - reader.emit('end'); - }); - return reader; - } else if (url.includes('sig')) { + const npmMetadata: NpmShowResults = { + versions: ['1.0.0', '1.0.1'], + 'dist-tags': { + latest: '1.0.1', + gozer: '1.0.0', + }, + dist: { + tarball: 'https://example.com/registry/foo.tgz', + }, + }; + + shelljsExecStub = getShelljsExecStub(sandbox, npmMetadata); + + const iRequest: IRequest = (url: string, cb?: request.RequestCallback) => { + if (url.includes('sig')) { cb(null, { statusCode: 200 } as request.Response, TEST_DATA_SIGNATURE); } else if (url.includes('key')) { cb(null, { statusCode: 200 } as request.Response, CERTIFICATE); - } else if (url.endsWith(plugin.name)) { - cb( - null, - { statusCode: 200 } as request.Response, - JSON.stringify({ - versions: { - '1.2.3': { - dist: { - tarball: 'https://registry.example.com/foo.tgz', - }, - }, - }, - 'dist-tags': { - latest: '1.2.3', - }, - }) - ); } else { throw new Error(`Unexpected test url - ${url}`); } @@ -474,17 +411,11 @@ describe('InstallationVerification Tests', () => { }); }); - it('Npm Meta Request Error', async () => { - const iRequest: IRequest = (url: string, cb?: request.RequestCallback): Readable => { - if (url.includes('foo.tgz')) { - const reader = new Readable({ - read() {}, - }); - process.nextTick(() => { - reader.emit('end'); - }); - return reader; - } else if (url.includes('sig')) { + it('NpmCommand Meta Request Error', async () => { + shelljsExecStub = stubMethod(sandbox, shelljs, 'exec').throws('SomeError'); + + const iRequest: IRequest = (url: string, cb?: request.RequestCallback) => { + if (url.includes('sig')) { cb(null, { statusCode: 200 } as request.Response, TEST_DATA_SIGNATURE); } else if (url.includes('key')) { cb(null, { statusCode: 200 } as request.Response, CERTIFICATE); @@ -510,11 +441,11 @@ describe('InstallationVerification Tests', () => { throw new Error("This shouldn't happen. Failure expected"); }) .catch((err: Error) => { - expect(err).to.have.property('name', 'NPMMetaError'); + expect(err).to.have.property('name', 'ShellExecError'); }); }); - it('server error', async () => { + it.skip('server error', async () => { let returnCode = 404; const iRequest: IRequest = (url: string, cb?: request.RequestCallback): Readable => { if (url.includes('foo.tgz')) { @@ -557,7 +488,7 @@ describe('InstallationVerification Tests', () => { } }); - it('Read tarball stream failed', () => { + it.skip('Read tarball stream failed', () => { const ERROR = 'Ok, who brought the dog? - Louis Tully'; const iRequest: IRequest = (url: string, cb?: request.RequestCallback): Readable => { @@ -618,40 +549,27 @@ describe('InstallationVerification Tests', () => { }); it('404 for public key', () => { - const iRequest: IRequest = (url: string, cb?: request.RequestCallback): Readable => { - if (url.includes('foo.tgz')) { - const reader = new Readable({ - read() {}, - }); - process.nextTick(() => { - reader.emit('end'); - }); - return reader; - } else if (url.includes('sig')) { + const npmMetadata: NpmShowResults = { + versions: ['1.0.0', '1.0.1'], + 'dist-tags': { + latest: '1.0.1', + }, + sfdx: { + publicKeyUrl: 'https://developer.salesforce.com/crt', + signatureUrl: 'https://developer.salesforce.com/sig', + }, + dist: { + tarball: 'https://example.com/registry/foo.tgz', + }, + }; + + shelljsExecStub = getShelljsExecStub(sandbox, npmMetadata); + + const iRequest: IRequest = (url: string, cb?: request.RequestCallback) => { + if (url.includes('sig')) { cb(null, { statusCode: 200 } as request.Response, TEST_DATA_SIGNATURE); - } else if (url.includes('key')) { + } else if (url.includes('crt')) { cb(null, { statusCode: 404 } as request.Response, {}); - } else if (url.endsWith(plugin.name)) { - cb( - null, - { statusCode: 200 } as request.Response, - JSON.stringify({ - versions: { - '1.2.3': { - sfdx: { - publicKeyUrl: 'https://developer.salesforce.com/key', - signatureUrl: 'https://developer.salesforce.com/sig', - }, - dist: { - tarball: 'https://registry.example.com/foo.tgz', - }, - }, - }, - 'dist-tags': { - latest: '1.2.3', - }, - }) - ); } else { throw new Error(`Unexpected test url - ${url}`); } diff --git a/yarn.lock b/yarn.lock index 6fd60b3b..3f6a51bb 100644 --- a/yarn.lock +++ b/yarn.lock @@ -6123,7 +6123,7 @@ shebang-regex@^3.0.0: shelljs@^0.8.4, shelljs@~0.8.4: version "0.8.4" - resolved "https://registry.npmjs.org/shelljs/-/shelljs-0.8.4.tgz#de7684feeb767f8716b326078a8a00875890e3c2" + resolved "https://nexus-proxy-prd.soma.salesforce.com/nexus/content/groups/npm-all/shelljs/-/shelljs-0.8.4.tgz#de7684feeb767f8716b326078a8a00875890e3c2" integrity sha512-7gk3UZ9kOfPLIAbslLzyWeGiEqx9e3rxwZM0KE6EL8GlGwjym9Mrlx5/p33bWTu9YG6vcS4MBxYZDHYr5lr8BQ== dependencies: glob "^7.0.0" From 4147cb975ab8f28f2e5d6bb39dd509bab729e1e6 Mon Sep 17 00:00:00 2001 From: Benjamin Maggi Date: Wed, 18 Aug 2021 18:14:50 -0300 Subject: [PATCH 02/15] fix: use public registry --- yarn.lock | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/yarn.lock b/yarn.lock index 3f6a51bb..11cf11e5 100644 --- a/yarn.lock +++ b/yarn.lock @@ -6123,7 +6123,7 @@ shebang-regex@^3.0.0: shelljs@^0.8.4, shelljs@~0.8.4: version "0.8.4" - resolved "https://nexus-proxy-prd.soma.salesforce.com/nexus/content/groups/npm-all/shelljs/-/shelljs-0.8.4.tgz#de7684feeb767f8716b326078a8a00875890e3c2" + resolved "https://registry.yarnpkg.com/shelljs/-/shelljs-0.8.4.tgz#de7684feeb767f8716b326078a8a00875890e3c2" integrity sha512-7gk3UZ9kOfPLIAbslLzyWeGiEqx9e3rxwZM0KE6EL8GlGwjym9Mrlx5/p33bWTu9YG6vcS4MBxYZDHYr5lr8BQ== dependencies: glob "^7.0.0" From b60b367992bdc00b008ded8eecefa819f41e0011 Mon Sep 17 00:00:00 2001 From: Benjamin Maggi Date: Thu, 19 Aug 2021 23:43:31 -0300 Subject: [PATCH 03/15] chore: use wrapper class --- package.json | 2 + src/lib/installationVerification.ts | 166 ++- src/lib/npmCommand.ts | 100 ++ test/lib/installationVerification.test.ts | 104 +- test/lib/npmCommand.test.ts | 198 ++++ yarn.lock | 1272 ++++++++++++++++++++- 6 files changed, 1687 insertions(+), 155 deletions(-) create mode 100644 src/lib/npmCommand.ts create mode 100644 test/lib/npmCommand.test.ts diff --git a/package.json b/package.json index 2f700848..85bd53cb 100644 --- a/package.json +++ b/package.json @@ -8,6 +8,7 @@ "@oclif/config": "^1.17.0", "@salesforce/command": "^3.0.5", "@salesforce/core": "^2.23.2", + "npm": "^7.21.0", "shelljs": "^0.8.4", "tslib": "^2" }, @@ -20,6 +21,7 @@ "@salesforce/plugin-command-reference": "^1.3.0", "@salesforce/prettier-config": "^0.0.2", "@salesforce/ts-sinon": "1.3.21", + "@types/shelljs": "^0.8.9", "@typescript-eslint/eslint-plugin": "^4.2.0", "@typescript-eslint/parser": "^4.2.0", "chai": "^4.2.0", diff --git a/src/lib/installationVerification.ts b/src/lib/installationVerification.ts index e3649076..a6126783 100644 --- a/src/lib/installationVerification.ts +++ b/src/lib/installationVerification.ts @@ -15,10 +15,9 @@ import { Readable } from 'stream'; import { parse as parseUrl, URL, UrlWithStringQuery } from 'url'; import { promisify as utilPromisify } from 'util'; import * as crypto from 'crypto'; -import * as shelljs from 'shelljs'; import { Logger, fs, SfdxError } from '@salesforce/core'; -import { get } from '@salesforce/ts-types'; import * as request from 'request'; +import { NpmModule } from '../lib/npmCommand'; import { NpmName } from './NpmName'; const CRYPTO_LEVEL = 'RSA-SHA256'; @@ -324,8 +323,6 @@ export class InstallationVerification implements Verifier { public async streamTagGz(): Promise { const logger = await this.getLogger(); const npmMeta = await this.retrieveNpmMeta(); - const npmCmd = `npm pack ${npmMeta.moduleName}@${npmMeta.version} --registry ${getNpmRegistry().href}`; - logger.debug(`npmCmd: ${npmCmd}`); const urlObject: URL = new URL(npmMeta.tarballUrl); const urlPathsAsArray = urlObject.pathname.split('/'); npmMeta.tarballFilename = npmMeta.moduleName.replace(/@/g, ''); @@ -337,15 +334,14 @@ export class InstallationVerification implements Verifier { // Make sure the cache path exists. try { await fs.mkdirp(this.getCachePath()); - shelljs.pushd(this.getCachePath()); - shelljs.exec(npmCmd, { silent: true, fatal: true }); + new NpmModule(npmMeta.moduleName, npmMeta.version).pack(getNpmRegistry().href, { cwd: this.getCachePath() }); const tarBallFile = fs .readdirSync(this.getCachePath(), { withFileTypes: true }) .find((entry) => entry.isFile() && entry.name.includes(npmMeta.version)); - npmMeta.tarballLocalPath = `${this.getCachePath()}${path.sep}${tarBallFile.name}`; - shelljs.popd(); + npmMeta.tarballLocalPath = path.join(this.getCachePath(), tarBallFile.name); } catch (err) { logger.debug(err); + throw new SfdxError(err, 'ShellExecError'); } return npmMeta; @@ -366,105 +362,87 @@ export class InstallationVerification implements Verifier { */ private async retrieveNpmMeta(): Promise { const logger = await this.getLogger(); - return new Promise((resolve, reject) => { - const npmRegistry = getNpmRegistry(); - - logger.debug(`retrieveNpmMeta | npmRegistry: ${npmRegistry.href}`); - logger.debug(`retrieveNpmMeta | this.pluginNpmName.name: ${this.pluginNpmName.name}`); - logger.debug(`retrieveNpmMeta | this.pluginNpmName.scope: ${this.pluginNpmName.scope}`); - logger.debug(`retrieveNpmMeta | this.pluginNpmName.tag: ${this.pluginNpmName.tag}`); - - const npmShowModule = this.pluginNpmName.scope - ? `@${this.pluginNpmName.scope}/${this.pluginNpmName.name}` - : this.pluginNpmName.name; - - // run npm show to get metadata - const npmCmd = `npm show ${npmShowModule} --registry ${npmRegistry.href} --json`; - let npmMetadata: NpmShowResults; - try { - const npmShowResult = shelljs.exec(npmCmd, { silent: true, fatal: true }); - npmMetadata = JSON.parse(npmShowResult.stdout); - } catch (error) { - return reject(new SfdxError(error, 'ShellExecError')); - } - const meta: NpmMeta = new NpmMeta(); - meta.moduleName = npmShowModule; - logger.debug('retrieveNpmMeta | Found npm meta information.'); - if (!npmMetadata.versions) { - return reject( - new SfdxError( - `The npm metadata for plugin ${this.pluginNpmName.name} is missing the versions attribute.`, - 'InvalidNpmMetadata' - ) - ); - } + const npmRegistry = getNpmRegistry(); + + logger.debug(`retrieveNpmMeta | npmRegistry: ${npmRegistry.href}`); + logger.debug(`retrieveNpmMeta | this.pluginNpmName.name: ${this.pluginNpmName.name}`); + logger.debug(`retrieveNpmMeta | this.pluginNpmName.scope: ${this.pluginNpmName.scope}`); + logger.debug(`retrieveNpmMeta | this.pluginNpmName.tag: ${this.pluginNpmName.tag}`); + + const npmShowModule = this.pluginNpmName.scope + ? `@${this.pluginNpmName.scope}/${this.pluginNpmName.name}` + : this.pluginNpmName.name; + + const npmMetadata = new NpmModule(npmShowModule).show(npmRegistry.href); + const meta: NpmMeta = new NpmMeta(); + meta.moduleName = npmShowModule; + logger.debug('retrieveNpmMeta | Found npm meta information.'); + if (!npmMetadata.versions) { + throw new SfdxError( + `The npm metadata for plugin ${this.pluginNpmName.name} is missing the versions attribute.`, + 'InvalidNpmMetadata' + ); + } - // Assume the tag is version tag. - let versionNumber = npmMetadata.versions.find((version) => version === this.pluginNpmName.tag); + // Assume the tag is version tag. + let versionNumber = npmMetadata.versions.find((version) => version === this.pluginNpmName.tag); - logger.debug(`retrieveNpmMeta | versionObject: ${JSON.stringify(versionNumber)}`); + logger.debug(`retrieveNpmMeta | versionObject: ${JSON.stringify(versionNumber)}`); - // If the assumption was not correct the tag must be a non-versioned dist-tag or not specified. - if (!versionNumber) { - // Assume dist-tag; - const distTags = npmMetadata['dist-tags']; - logger.debug(`retrieveNpmMeta | distTags: ${JSON.stringify(distTags)}`); - if (distTags) { - const tagVersionStr: string = get(distTags, this.pluginNpmName.tag) as string; - logger.debug(`retrieveNpmMeta | tagVersionStr: ${tagVersionStr}`); + // If the assumption was not correct the tag must be a non-versioned dist-tag or not specified. + if (!versionNumber) { + // Assume dist-tag; + const distTags = npmMetadata['dist-tags']; + logger.debug(`retrieveNpmMeta | distTags: ${JSON.stringify(distTags)}`); + if (distTags) { + const tagVersionStr: string = distTags[this.pluginNpmName.tag]; + logger.debug(`retrieveNpmMeta | tagVersionStr: ${tagVersionStr}`); - // if we got a dist tag hit look up the version object - if (tagVersionStr && tagVersionStr.length > 0 && tagVersionStr.includes('.')) { - versionNumber = npmMetadata.versions.find((version) => version === tagVersionStr); - logger.debug(`retrieveNpmMeta | versionObject: ${versionNumber}`); - } else { - return reject( - new SfdxError( - `The dist tag ${this.pluginNpmName.tag} was not found for plugin: ${this.pluginNpmName.name}`, - 'NpmTagNotFound' - ) - ); - } + // if we got a dist tag hit look up the version object + if (tagVersionStr && tagVersionStr.length > 0 && tagVersionStr.includes('.')) { + versionNumber = npmMetadata.versions.find((version) => version === tagVersionStr); + logger.debug(`retrieveNpmMeta | versionObject: ${versionNumber}`); } else { - throw new SfdxError('The deployed NPM is missing dist-tags.', 'UnexpectedNpmFormat'); + throw new SfdxError( + `The dist tag ${this.pluginNpmName.tag} was not found for plugin: ${this.pluginNpmName.name}`, + 'NpmTagNotFound' + ); } + } else { + throw new SfdxError('The deployed NPM is missing dist-tags.', 'UnexpectedNpmFormat'); } + } - meta.version = versionNumber; + meta.version = versionNumber; - if (!npmMetadata.sfdx) { - throw new SfdxError('This plugin is not signed by Salesforce.com, Inc.', 'NotSigned'); + if (!npmMetadata.sfdx) { + throw new SfdxError('This plugin is not signed by Salesforce.com, Inc.', 'NotSigned'); + } else { + if (!validSalesforceHostname(npmMetadata.sfdx.publicKeyUrl)) { + throw new SfdxError( + `The host is not allowed to provide signing information. [${npmMetadata.sfdx.publicKeyUrl}]`, + 'UnexpectedHost' + ); } else { - if (!validSalesforceHostname(npmMetadata.sfdx.publicKeyUrl)) { - return reject( - new SfdxError( - `The host is not allowed to provide signing information. [${npmMetadata.sfdx.publicKeyUrl}]`, - 'UnexpectedHost' - ) - ); - } else { - logger.debug(`retrieveNpmMeta | versionObject.sfdx.publicKeyUrl: ${npmMetadata.sfdx.publicKeyUrl}`); - meta.publicKeyUrl = npmMetadata.sfdx.publicKeyUrl; - } + logger.debug(`retrieveNpmMeta | versionObject.sfdx.publicKeyUrl: ${npmMetadata.sfdx.publicKeyUrl}`); + meta.publicKeyUrl = npmMetadata.sfdx.publicKeyUrl; + } - if (!validSalesforceHostname(npmMetadata.sfdx.signatureUrl)) { - return reject( - new SfdxError( - `The host is not allowed to provide signing information. [${npmMetadata.sfdx.signatureUrl}]`, - 'UnexpectedHost' - ) - ); - } else { - logger.debug(`retrieveNpmMeta | versionObject.sfdx.signatureUrl: ${npmMetadata.sfdx.signatureUrl}`); - meta.signatureUrl = npmMetadata.sfdx.signatureUrl; - } + if (!validSalesforceHostname(npmMetadata.sfdx.signatureUrl)) { + throw new SfdxError( + `The host is not allowed to provide signing information. [${npmMetadata.sfdx.signatureUrl}]`, + 'UnexpectedHost' + ); + } else { + logger.debug(`retrieveNpmMeta | versionObject.sfdx.signatureUrl: ${npmMetadata.sfdx.signatureUrl}`); + meta.signatureUrl = npmMetadata.sfdx.signatureUrl; + } - meta.tarballUrl = npmMetadata.dist.tarball; - logger.debug(`retrieveNpmMeta | meta.tarballUrl: ${meta.tarballUrl}`); + meta.tarballUrl = npmMetadata.dist.tarball; + logger.debug(`retrieveNpmMeta | meta.tarballUrl: ${meta.tarballUrl}`); - return resolve(meta); - } - }); + return meta; + } } private async getLogger(): Promise { diff --git a/src/lib/npmCommand.ts b/src/lib/npmCommand.ts new file mode 100644 index 00000000..d66fd3d1 --- /dev/null +++ b/src/lib/npmCommand.ts @@ -0,0 +1,100 @@ +/* + * Copyright (c) 2020, salesforce.com, inc. + * All rights reserved. + * Licensed under the BSD 3-Clause license. + * For full license text, see LICENSE.txt file in the repo root or https://opensource.org/licenses/BSD-3-Clause + */ +/* eslint-disable @typescript-eslint/no-unused-vars */ + +import * as path from 'path'; +import * as shelljs from 'shelljs'; +import { SfdxError, fs } from '@salesforce/core'; + +type DistTags = { + [name: string]: string; +}; + +type NpmShowResults = { + versions: string[]; + 'dist-tags': DistTags; + sfdx?: { + publicKeyUrl: string; + signatureUrl: string; + }; + dist?: { + integrity: string; + shasum: string; + tarball: string; + fileCount: number; + unpackedSize: number; + 'npm-signature': string; + }; +}; + +type NpmCommandOptions = shelljs.ExecOptions & { + json?: boolean; + registry?: string; +}; + +type NpmCommandResult = NpmShowResults & { + [name: string]: string; +}; + +type NpmPackage = { + bin: { + [name: string]: string; + }; +}; + +export class NpmCommand { + private static npmPkgPath = require.resolve('npm/package.json'); + + public static runNpmCmd(cmd: string, options = {} as NpmCommandOptions): NpmCommandResult { + const npmCli = NpmCommand.npmCli(); + const exec = `/bin/bash -c "node ${npmCli} ${cmd} --registry=${options.registry} --json"`; + const npmShowResult = shelljs.exec(exec, { + ...options, + silent: true, + fatal: true, + async: false, + }); + if (npmShowResult.code !== 0) { + throw new SfdxError(npmShowResult.stderr, 'ShellExecError'); + } + try { + return JSON.parse(npmShowResult.stdout) as NpmCommandResult; + } catch (error) { + throw new SfdxError(error, 'ShellParseError'); + } + } + + private static npmPackagePath(): string { + return this.npmPkgPath; + } + + private static npmCli(): string { + const pkgPath = NpmCommand.npmPackagePath(); + const pkgJson = fs.readJsonSync(pkgPath) as NpmPackage; + const prjPath = pkgPath.substring(0, pkgPath.lastIndexOf(path.sep)); + return path.join(prjPath, pkgJson.bin['npm']); + } +} + +export class NpmModule { + private version: string; + private module: string; + + public constructor(module: string, version = 'latest') { + this.module = module; + this.version = version; + } + + public show(registry: string): NpmShowResults { + return NpmCommand.runNpmCmd(`show ${this.module}@${this.version}`, { registry }); + } + + public pack(registry: string, options?: shelljs.ExecOptions): void { + NpmCommand.runNpmCmd(`pack ${this.module}@${this.version}`, { ...options, registry }); + return; + } +} diff --git a/test/lib/installationVerification.test.ts b/test/lib/installationVerification.test.ts index e4ac3b53..e8f1f554 100644 --- a/test/lib/installationVerification.test.ts +++ b/test/lib/installationVerification.test.ts @@ -9,7 +9,7 @@ import { expect } from 'chai'; import * as request from 'request'; import * as shelljs from 'shelljs'; import { stubMethod } from '@salesforce/ts-sinon'; -import { SfdxError } from '@salesforce/core'; +import { SfdxError, fs } from '@salesforce/core'; import Sinon = require('sinon'); import { ConfigContext, @@ -27,17 +27,47 @@ import { NpmName } from '../../src/lib/NpmName'; import { CERTIFICATE, TEST_DATA, TEST_DATA_SIGNATURE } from '../testCert'; const BLANK_PLUGIN = { plugin: '', tag: '' }; - -const getShelljsExecStub = (sandbox: sinon.SinonSandbox, npmMetadata: NpmShowResults): Sinon.SinonStub => { +const MODULE_NAME = '@salesforce/plugin-source'; +const MODULE_VERSION = '1.0.0'; +const PACK_RESULT = [ + { + id: `${MODULE_NAME}@${MODULE_VERSION}`, + name: MODULE_NAME, + version: MODULE_VERSION, + size: 1024, + unpackedSize: 4096, + shasum: 'DEADBEEF', + integrity: 'sha512-L5/ABCDE/ABCDEFGHIJKLMNOPQRSTUVWXYZ==', + filename: `${MODULE_NAME}-${MODULE_VERSION}.tgz`, + files: [ + { + path: 'README.md', + size: 512, + mode: 444, + }, + ], + }, +]; + +const getShelljsExecStub = ( + sandbox: sinon.SinonSandbox, + npmMetadata: NpmShowResults, + code = 0 as number, + stderr?: string +): Sinon.SinonStub => { return stubMethod(sandbox, shelljs, 'exec').callsFake((cmd: string) => { expect(cmd).to.be.a('string').and.not.to.be.empty; if (cmd.includes('show')) { return { + code, + stderr, stdout: JSON.stringify(npmMetadata), }; } else if (cmd.includes('pack')) { return { - stdout: undefined, + code, + stderr, + stdout: JSON.stringify(PACK_RESULT), }; } else { throw new Error(`Unexpected test cmd - ${cmd}`); @@ -81,13 +111,23 @@ describe('InstallationVerification Tests', () => { let plugin: NpmName; let sandbox: sinon.SinonSandbox; let shelljsExecStub: Sinon.SinonStub; + let fsReaddirSyncStub: Sinon.SinonStub; beforeEach(() => { sandbox = Sinon.createSandbox(); + fsReaddirSyncStub = stubMethod(sandbox, fs, 'readdirSync').returns([ + { + name: 'foo-1.0.0.tgz', + isFile() { + return true; + }, + }, + ]); plugin = NpmName.parse('foo'); }); afterEach(() => { + fsReaddirSyncStub.restore(); if (shelljsExecStub) { shelljsExecStub.restore(); } @@ -132,7 +172,7 @@ describe('InstallationVerification Tests', () => { } }; - const fs = { + const fsImpl = { readFile() {}, createWriteStream() { return new Writable({ @@ -152,7 +192,7 @@ describe('InstallationVerification Tests', () => { }, }; - const verification = new InstallationVerification(iRequest, fs).setPluginNpmName(plugin).setConfig(config); + const verification = new InstallationVerification(iRequest, fsImpl).setPluginNpmName(plugin).setConfig(config); return verification.verify().then((meta: NpmMeta) => { expect(meta).to.have.property('verified', true); @@ -194,7 +234,7 @@ describe('InstallationVerification Tests', () => { } }; - const fs = { + const fsImpl = { readFile() {}, createWriteStream() { return new Writable({ @@ -215,7 +255,7 @@ describe('InstallationVerification Tests', () => { }; plugin.tag = '1.0.0'; - const verification = new InstallationVerification(iRequest, fs).setPluginNpmName(plugin).setConfig(config); + const verification = new InstallationVerification(iRequest, fsImpl).setPluginNpmName(plugin).setConfig(config); return verification.verify().then((meta: NpmMeta) => { expect(meta).to.have.property('verified', true); @@ -249,7 +289,7 @@ describe('InstallationVerification Tests', () => { } }; - const fs = { + const fsImpl = { readFile() {}, createWriteStream() { return new Writable({ @@ -271,7 +311,7 @@ describe('InstallationVerification Tests', () => { plugin.tag = 'gozer'; // For the key and signature to line up gozer must map to 1.2.3 - const verification = new InstallationVerification(iRequest, fs).setPluginNpmName(plugin).setConfig(config); + const verification = new InstallationVerification(iRequest, fsImpl).setPluginNpmName(plugin).setConfig(config); return verification.verify().then((meta: NpmMeta) => { expect(meta).to.have.property('verified', true); @@ -309,7 +349,7 @@ describe('InstallationVerification Tests', () => { } }; - const fs = { + const fsImpl = { readFile() {}, createWriteStream() { return new Writable({ @@ -331,7 +371,7 @@ describe('InstallationVerification Tests', () => { plugin.tag = 'gozer'; // For the key and signature to line up gozer must map to 1.2.3 - const verification = new InstallationVerification(iRequest, fs).setPluginNpmName(plugin).setConfig(config); + const verification = new InstallationVerification(iRequest, fsImpl).setPluginNpmName(plugin).setConfig(config); return verification.verify().then((meta: NpmMeta) => { expect(meta).to.have.property('verified', true); @@ -353,12 +393,12 @@ describe('InstallationVerification Tests', () => { } }; - const fs = { + const fsImpl = { readFile() {}, unlink() {}, }; - const verification = new InstallationVerification(iRequest, fs).setPluginNpmName(plugin).setConfig(config); + const verification = new InstallationVerification(iRequest, fsImpl).setPluginNpmName(plugin).setConfig(config); return verification .verify() @@ -394,12 +434,12 @@ describe('InstallationVerification Tests', () => { } }; - const fs = { + const fsImpl = { readFile() {}, unlink() {}, }; - const verification = new InstallationVerification(iRequest, fs).setPluginNpmName(plugin).setConfig(config); + const verification = new InstallationVerification(iRequest, fsImpl).setPluginNpmName(plugin).setConfig(config); return verification .verify() @@ -412,7 +452,7 @@ describe('InstallationVerification Tests', () => { }); it('NpmCommand Meta Request Error', async () => { - shelljsExecStub = stubMethod(sandbox, shelljs, 'exec').throws('SomeError'); + shelljsExecStub = getShelljsExecStub(sandbox, {} as NpmShowResults, 1, 'command execution error'); const iRequest: IRequest = (url: string, cb?: request.RequestCallback) => { if (url.includes('sig')) { @@ -428,12 +468,12 @@ describe('InstallationVerification Tests', () => { } }; - const fs = { + const fsImpl = { readFile() {}, unlink() {}, }; - const verification = new InstallationVerification(iRequest, fs).setPluginNpmName(plugin).setConfig(config); + const verification = new InstallationVerification(iRequest, fsImpl).setPluginNpmName(plugin).setConfig(config); return verification .verify() @@ -467,7 +507,7 @@ describe('InstallationVerification Tests', () => { } }; - const fs = { + const fsImpl = { readFile() {}, unlink() {}, }; @@ -479,7 +519,7 @@ describe('InstallationVerification Tests', () => { for (const testMeta of results) { returnCode = testMeta.code; - const verification = new InstallationVerification(iRequest, fs).setPluginNpmName(plugin).setConfig(config); + const verification = new InstallationVerification(iRequest, fsImpl).setPluginNpmName(plugin).setConfig(config); try { await verification.verify(); } catch (error) { @@ -526,7 +566,7 @@ describe('InstallationVerification Tests', () => { } }; - const fs = { + const fsImpl = { readFile() {}, unlink() {}, createWriteStream() { @@ -536,7 +576,7 @@ describe('InstallationVerification Tests', () => { }, }; - const verification = new InstallationVerification(iRequest, fs).setPluginNpmName(plugin).setConfig(config); + const verification = new InstallationVerification(iRequest, fsImpl).setPluginNpmName(plugin).setConfig(config); return verification .verify() @@ -552,7 +592,7 @@ describe('InstallationVerification Tests', () => { const npmMetadata: NpmShowResults = { versions: ['1.0.0', '1.0.1'], 'dist-tags': { - latest: '1.0.1', + latest: '1.0.0', }, sfdx: { publicKeyUrl: 'https://developer.salesforce.com/crt', @@ -575,7 +615,7 @@ describe('InstallationVerification Tests', () => { } }; - const fs = { + const fsImpl = { readFile() {}, unlink() {}, createWriteStream() { @@ -593,7 +633,7 @@ describe('InstallationVerification Tests', () => { }, }; - const verification = new InstallationVerification(iRequest, fs).setPluginNpmName(plugin).setConfig(config); + const verification = new InstallationVerification(iRequest, fsImpl).setPluginNpmName(plugin).setConfig(config); return verification .verify() @@ -609,13 +649,13 @@ describe('InstallationVerification Tests', () => { describe('isAllowListed', () => { it('steel thread with scope', async () => { const TEST_VALUE1 = '@salesforce/FOO'; - const fs = { + const fsImpl = { readFile(path, cb) { cb(null, `["${TEST_VALUE1}"]`); }, unlink() {}, }; - const verification1 = new InstallationVerification(null, fs) + const verification1 = new InstallationVerification(null, fsImpl) .setPluginNpmName(NpmName.parse(TEST_VALUE1)) .setConfig(config); expect(await verification1.isAllowListed()).to.be.equal(true); @@ -623,20 +663,20 @@ describe('InstallationVerification Tests', () => { it('steel thread without scope', async () => { const TEST_VALUE2 = 'FOO'; - const fs = { + const fsImpl = { readFile(path, cb) { cb(null, `["${TEST_VALUE2}"]`); }, unlink() {}, }; - const verification2 = new InstallationVerification(null, fs) + const verification2 = new InstallationVerification(null, fsImpl) .setPluginNpmName(NpmName.parse(TEST_VALUE2)) .setConfig(config); expect(await verification2.isAllowListed()).to.be.equal(true); }); it("file doesn't exist", async () => { - const fs = { + const fsImpl = { readFile(path, cb) { const error = new SfdxError('ENOENT', 'ENOENT'); error['code'] = 'ENOENT'; @@ -645,7 +685,7 @@ describe('InstallationVerification Tests', () => { unlink() {}, }; - const verification = new InstallationVerification(null, fs) + const verification = new InstallationVerification(null, fsImpl) .setPluginNpmName(NpmName.parse('BAR')) .setConfig(config); expect(await verification.isAllowListed()).to.be.equal(false); diff --git a/test/lib/npmCommand.test.ts b/test/lib/npmCommand.test.ts new file mode 100644 index 00000000..4ec7df56 --- /dev/null +++ b/test/lib/npmCommand.test.ts @@ -0,0 +1,198 @@ +/* + * Copyright (c) 2020, salesforce.com, inc. + * All rights reserved. + * Licensed under the BSD 3-Clause license. + * For full license text, see LICENSE.txt file in the repo root or https://opensource.org/licenses/BSD-3-Clause + */ + +import { fail } from 'assert'; +import { expect } from 'chai'; +import Sinon = require('sinon'); +import * as shelljs from 'shelljs'; +import { stubMethod } from '@salesforce/ts-sinon'; +import { NpmModule } from '../../src/lib/npmCommand'; + +const DEFAULT_REGISTRY = 'https://registry.npmjs.org/'; +const MODULE_NAME = '@salesforce/plugin-source'; +const MODULE_VERSION = '1.0.0'; +const CACHE_PATH = '/tmp'; +const SHOW_RESULT = { + versions: ['1.0.0'], + 'dist-tags': { + latest: '1.0.0', + }, + sfdx: { + publicKeyUrl: 'https://developer.salesforce.com/crt', + signatureUrl: 'https://developer.salesforce.com/sig', + }, + dist: { + tarball: 'https://registry.example.com/foo.tgz', + }, +}; +const PACK_RESULT = [ + { + id: `${MODULE_NAME}@${MODULE_VERSION}`, + name: MODULE_NAME, + version: MODULE_VERSION, + size: 1024, + unpackedSize: 4096, + shasum: 'DEADBEEF', + integrity: 'sha512-L5/ABCDE/ABCDEFGHIJKLMNOPQRSTUVWXYZ==', + filename: `${MODULE_NAME}-${MODULE_VERSION}.tgz`, + files: [ + { + path: 'README.md', + size: 512, + mode: 444, + }, + ], + }, +]; + +describe('should run npm commands', () => { + let sandbox: sinon.SinonSandbox; + let shelljsExecStub: Sinon.SinonStub; + + beforeEach(() => { + sandbox = Sinon.createSandbox(); + shelljsExecStub = stubMethod(sandbox, shelljs, 'exec').callsFake((cmd: string) => { + expect(cmd).to.be.a('string').and.not.to.be.empty; + if (cmd.includes('show')) { + return { + code: 0, + stdout: JSON.stringify(SHOW_RESULT), + }; + } else if (cmd.includes('pack')) { + return { + code: 0, + stdout: JSON.stringify(PACK_RESULT), + }; + } else { + throw new Error(`Unexpected test cmd - ${cmd}`); + } + }); + }); + + afterEach(() => { + sandbox.restore(); + }); + + it('Runs the show command', () => { + const npmMetadata = new NpmModule(MODULE_NAME).show(DEFAULT_REGISTRY); + expect(shelljsExecStub.callCount).to.equal(1); + expect(shelljsExecStub.firstCall.args[0]).to.include(`show ${MODULE_NAME}@latest`); + expect(shelljsExecStub.firstCall.args[0]).to.include(`--registry=${DEFAULT_REGISTRY}`); + expect(npmMetadata).to.deep.equal(SHOW_RESULT); + }); + + it('Runs the show command with specified version', () => { + const npmMetadata = new NpmModule(MODULE_NAME, MODULE_VERSION).show(DEFAULT_REGISTRY); + expect(shelljsExecStub.callCount).to.equal(1); + expect(shelljsExecStub.firstCall.args[0]).to.include(`show ${MODULE_NAME}@${MODULE_VERSION}`); + expect(shelljsExecStub.firstCall.args[0]).to.include(`--registry=${DEFAULT_REGISTRY}`); + expect(npmMetadata).to.deep.equal(SHOW_RESULT); + }); + + it('Runs the pack command', () => { + new NpmModule(MODULE_NAME, MODULE_VERSION).pack(DEFAULT_REGISTRY, { cwd: CACHE_PATH }); + expect(shelljsExecStub.callCount).to.equal(1); + expect(shelljsExecStub.firstCall.args[0]).to.include(`pack ${MODULE_NAME}@${MODULE_VERSION}`); + expect(shelljsExecStub.firstCall.args[0]).to.include(`--registry=${DEFAULT_REGISTRY}`); + }); +}); + +describe('should run npm commands with execution errors', () => { + let sandbox: sinon.SinonSandbox; + + beforeEach(() => { + sandbox = Sinon.createSandbox(); + stubMethod(sandbox, shelljs, 'exec').callsFake((cmd: string) => { + expect(cmd).to.be.a('string').and.not.to.be.empty; + if (cmd.includes('show')) { + return { + code: 1, + stderr: 'command execution error', + stdout: '', + }; + } else if (cmd.includes('pack')) { + return { + code: 1, + stderr: 'command execution error', + stdout: '', + }; + } else { + throw new Error(`Unexpected test cmd - ${cmd}`); + } + }); + }); + + afterEach(() => { + sandbox.restore(); + }); + + it('show command throws error', () => { + try { + const npmMetadata = new NpmModule(MODULE_NAME).show(DEFAULT_REGISTRY); + expect(npmMetadata).to.be.undefined; + fail('Error'); + } catch (error) { + expect(error.code).to.equal('ShellExecError'); + } + }); + + it('Runs the pack command', () => { + try { + new NpmModule(MODULE_NAME, MODULE_VERSION).pack(DEFAULT_REGISTRY, { cwd: CACHE_PATH }); + fail('Error'); + } catch (error) { + expect(error.code).to.equal('ShellExecError'); + } + }); +}); + +describe('should run npm commands with parse errors', () => { + let sandbox: sinon.SinonSandbox; + + beforeEach(() => { + sandbox = Sinon.createSandbox(); + stubMethod(sandbox, shelljs, 'exec').callsFake((cmd: string) => { + expect(cmd).to.be.a('string').and.not.to.be.empty; + if (cmd.includes('show')) { + return { + code: 0, + stdout: 'not a json string', + }; + } else if (cmd.includes('pack')) { + return { + code: 0, + stdout: 'not a json string', + }; + } else { + throw new Error(`Unexpected test cmd - ${cmd}`); + } + }); + }); + + afterEach(() => { + sandbox.restore(); + }); + + it('show command throws error', () => { + try { + const npmMetadata = new NpmModule(MODULE_NAME).show(DEFAULT_REGISTRY); + expect(npmMetadata).to.be.undefined; + fail('Error'); + } catch (error) { + expect(error.code).to.equal('ShellParseError'); + } + }); + + it('Runs the pack command', () => { + try { + new NpmModule(MODULE_NAME, MODULE_VERSION).pack(DEFAULT_REGISTRY, { cwd: CACHE_PATH }); + fail('Error'); + } catch (error) { + expect(error.code).to.equal('ShellParseError'); + } + }); +}); diff --git a/yarn.lock b/yarn.lock index 11cf11e5..51aab7a7 100644 --- a/yarn.lock +++ b/yarn.lock @@ -394,6 +394,149 @@ "@nodelib/fs.scandir" "2.1.3" fastq "^1.6.0" +"@npmcli/arborist@^2.3.0", "@npmcli/arborist@^2.5.0", "@npmcli/arborist@^2.8.2": + version "2.8.2" + resolved "https://registry.yarnpkg.com/@npmcli/arborist/-/arborist-2.8.2.tgz#643f8c8a26ffbaa579983972f67a60cb6217e86a" + integrity sha512-6E1XJ0YXBaI9J+25gcTF110MGNx3jv6npr4Rz1U0UAqkuVV7bbDznVJvNqi6F0p8vgrE+Smf9jDTn1DR+7uBjQ== + dependencies: + "@npmcli/installed-package-contents" "^1.0.7" + "@npmcli/map-workspaces" "^1.0.2" + "@npmcli/metavuln-calculator" "^1.1.0" + "@npmcli/move-file" "^1.1.0" + "@npmcli/name-from-folder" "^1.0.1" + "@npmcli/node-gyp" "^1.0.1" + "@npmcli/package-json" "^1.0.1" + "@npmcli/run-script" "^1.8.2" + bin-links "^2.2.1" + cacache "^15.0.3" + common-ancestor-path "^1.0.1" + json-parse-even-better-errors "^2.3.1" + json-stringify-nice "^1.1.4" + mkdirp "^1.0.4" + mkdirp-infer-owner "^2.0.0" + npm-install-checks "^4.0.0" + npm-package-arg "^8.1.5" + npm-pick-manifest "^6.1.0" + npm-registry-fetch "^11.0.0" + pacote "^11.3.5" + parse-conflict-json "^1.1.1" + proc-log "^1.0.0" + promise-all-reject-late "^1.0.0" + promise-call-limit "^1.0.1" + read-package-json-fast "^2.0.2" + readdir-scoped-modules "^1.1.0" + rimraf "^3.0.2" + semver "^7.3.5" + ssri "^8.0.1" + treeverse "^1.0.4" + walk-up-path "^1.0.0" + +"@npmcli/ci-detect@^1.2.0", "@npmcli/ci-detect@^1.3.0": + version "1.3.0" + resolved "https://registry.yarnpkg.com/@npmcli/ci-detect/-/ci-detect-1.3.0.tgz#6c1d2c625fb6ef1b9dea85ad0a5afcbef85ef22a" + integrity sha512-oN3y7FAROHhrAt7Rr7PnTSwrHrZVRTS2ZbyxeQwSSYD0ifwM3YNgQqbaRmjcWoPyq77MjchusjJDspbzMmip1Q== + +"@npmcli/config@^2.2.0": + version "2.2.0" + resolved "https://registry.yarnpkg.com/@npmcli/config/-/config-2.2.0.tgz#c3f6cb76e74691d1ae746cda482b7df751ed2124" + integrity sha512-y0V3F7RCWXy8kBOvKvKSRUNKRobLB6vL/UNchy/6+IUNIqu+UyrY3Z7jvj1ZA/AkYc/0WkCUtppCo+bPhMU8Aw== + dependencies: + ini "^2.0.0" + mkdirp-infer-owner "^2.0.0" + nopt "^5.0.0" + semver "^7.3.4" + walk-up-path "^1.0.0" + +"@npmcli/disparity-colors@^1.0.1": + version "1.0.1" + resolved "https://registry.yarnpkg.com/@npmcli/disparity-colors/-/disparity-colors-1.0.1.tgz#b23c864c9658f9f0318d5aa6d17986619989535c" + integrity sha512-kQ1aCTTU45mPXN+pdAaRxlxr3OunkyztjbbxDY/aIcPS5CnCUrx+1+NvA6pTcYR7wmLZe37+Mi5v3nfbwPxq3A== + dependencies: + ansi-styles "^4.3.0" + +"@npmcli/git@^2.0.7", "@npmcli/git@^2.1.0": + version "2.1.0" + resolved "https://registry.yarnpkg.com/@npmcli/git/-/git-2.1.0.tgz#2fbd77e147530247d37f325930d457b3ebe894f6" + integrity sha512-/hBFX/QG1b+N7PZBFs0bi+evgRZcK9nWBxQKZkGoXUT5hJSwl5c4d7y8/hm+NQZRPhQ67RzFaj5UM9YeyKoryw== + dependencies: + "@npmcli/promise-spawn" "^1.3.2" + lru-cache "^6.0.0" + mkdirp "^1.0.4" + npm-pick-manifest "^6.1.1" + promise-inflight "^1.0.1" + promise-retry "^2.0.1" + semver "^7.3.5" + which "^2.0.2" + +"@npmcli/installed-package-contents@^1.0.6", "@npmcli/installed-package-contents@^1.0.7": + version "1.0.7" + resolved "https://registry.yarnpkg.com/@npmcli/installed-package-contents/-/installed-package-contents-1.0.7.tgz#ab7408c6147911b970a8abe261ce512232a3f4fa" + integrity sha512-9rufe0wnJusCQoLpV9ZPKIVP55itrM5BxOXs10DmdbRfgWtHy1LDyskbwRnBghuB0PrF7pNPOqREVtpz4HqzKw== + dependencies: + npm-bundled "^1.1.1" + npm-normalize-package-bin "^1.0.1" + +"@npmcli/map-workspaces@^1.0.2", "@npmcli/map-workspaces@^1.0.4": + version "1.0.4" + resolved "https://registry.yarnpkg.com/@npmcli/map-workspaces/-/map-workspaces-1.0.4.tgz#915708b55afa25e20bc2c14a766c124c2c5d4cab" + integrity sha512-wVR8QxhyXsFcD/cORtJwGQodeeaDf0OxcHie8ema4VgFeqwYkFsDPnSrIRSytX8xR6nKPAH89WnwTcaU608b/Q== + dependencies: + "@npmcli/name-from-folder" "^1.0.1" + glob "^7.1.6" + minimatch "^3.0.4" + read-package-json-fast "^2.0.1" + +"@npmcli/metavuln-calculator@^1.1.0": + version "1.1.1" + resolved "https://registry.yarnpkg.com/@npmcli/metavuln-calculator/-/metavuln-calculator-1.1.1.tgz#2f95ff3c6d88b366dd70de1c3f304267c631b458" + integrity sha512-9xe+ZZ1iGVaUovBVFI9h3qW+UuECUzhvZPxK9RaEA2mjU26o5D0JloGYWwLYvQELJNmBdQB6rrpuN8jni6LwzQ== + dependencies: + cacache "^15.0.5" + pacote "^11.1.11" + semver "^7.3.2" + +"@npmcli/move-file@^1.0.1", "@npmcli/move-file@^1.1.0": + version "1.1.2" + resolved "https://registry.yarnpkg.com/@npmcli/move-file/-/move-file-1.1.2.tgz#1a82c3e372f7cae9253eb66d72543d6b8685c674" + integrity sha512-1SUf/Cg2GzGDyaf15aR9St9TWlb+XvbZXWpDx8YKs7MLzMH/BCeopv+y9vzrzgkfykCGuWOlSu3mZhj2+FQcrg== + dependencies: + mkdirp "^1.0.4" + rimraf "^3.0.2" + +"@npmcli/name-from-folder@^1.0.1": + version "1.0.1" + resolved "https://registry.yarnpkg.com/@npmcli/name-from-folder/-/name-from-folder-1.0.1.tgz#77ecd0a4fcb772ba6fe927e2e2e155fbec2e6b1a" + integrity sha512-qq3oEfcLFwNfEYOQ8HLimRGKlD8WSeGEdtUa7hmzpR8Sa7haL1KVQrvgO6wqMjhWFFVjgtrh1gIxDz+P8sjUaA== + +"@npmcli/node-gyp@^1.0.1", "@npmcli/node-gyp@^1.0.2": + version "1.0.2" + resolved "https://registry.yarnpkg.com/@npmcli/node-gyp/-/node-gyp-1.0.2.tgz#3cdc1f30e9736dbc417373ed803b42b1a0a29ede" + integrity sha512-yrJUe6reVMpktcvagumoqD9r08fH1iRo01gn1u0zoCApa9lnZGEigVKUd2hzsCId4gdtkZZIVscLhNxMECKgRg== + +"@npmcli/package-json@^1.0.1": + version "1.0.1" + resolved "https://registry.yarnpkg.com/@npmcli/package-json/-/package-json-1.0.1.tgz#1ed42f00febe5293c3502fd0ef785647355f6e89" + integrity sha512-y6jnu76E9C23osz8gEMBayZmaZ69vFOIk8vR1FJL/wbEJ54+9aVG9rLTjQKSXfgYZEr50nw1txBBFfBZZe+bYg== + dependencies: + json-parse-even-better-errors "^2.3.1" + +"@npmcli/promise-spawn@^1.2.0", "@npmcli/promise-spawn@^1.3.2": + version "1.3.2" + resolved "https://registry.yarnpkg.com/@npmcli/promise-spawn/-/promise-spawn-1.3.2.tgz#42d4e56a8e9274fba180dabc0aea6e38f29274f5" + integrity sha512-QyAGYo/Fbj4MXeGdJcFzZ+FkDkomfRBrPM+9QYJSg+PxgAUL+LU3FneQk37rKR2/zjqkCV1BLHccX98wRXG3Sg== + dependencies: + infer-owner "^1.0.4" + +"@npmcli/run-script@^1.8.2", "@npmcli/run-script@^1.8.3", "@npmcli/run-script@^1.8.4", "@npmcli/run-script@^1.8.6": + version "1.8.6" + resolved "https://registry.yarnpkg.com/@npmcli/run-script/-/run-script-1.8.6.tgz#18314802a6660b0d4baa4c3afe7f1ad39d8c28b7" + integrity sha512-e42bVZnC6VluBZBAFEr3YrdqSspG3bgilyg4nSLBJ7TRGNCzxHa92XAHxQBLYg0BmgwO4b2mf3h/l5EkEWRn3g== + dependencies: + "@npmcli/node-gyp" "^1.0.2" + "@npmcli/promise-spawn" "^1.3.2" + node-gyp "^7.1.0" + read-package-json-fast "^2.0.1" + "@oclif/command@^1.5.13", "@oclif/command@^1.5.17", "@oclif/command@^1.5.19", "@oclif/command@^1.5.20", "@oclif/command@^1.6.0", "@oclif/command@^1.8.0": version "1.8.0" resolved "https://registry.npmjs.org/@oclif/command/-/command-1.8.0.tgz#c1a499b10d26e9d1a611190a81005589accbb339" @@ -743,6 +886,11 @@ resolved "https://registry.npmjs.org/@sinonjs/text-encoding/-/text-encoding-0.7.1.tgz#8da5c6530915653f3a1f38fd5f101d8c3f8079c5" integrity sha512-+iTbntw2IZPb/anVDbypzfQa+ay64MW0Zo8aJ8gZPWMMK6/OubMVb6lUPMagqjOPnmtauXnFCACVl3O7ogjeqQ== +"@tootallnate/once@1": + version "1.1.2" + resolved "https://registry.yarnpkg.com/@tootallnate/once/-/once-1.1.2.tgz#ccb91445360179a04e7fe6aff78c00ffc1eeaf82" + integrity sha512-RbzJvlNzmRq5c3O09UipeuXno4tA1FE6ikOjxZK0tuxVv3412l64l5t1W5pj4+rJq9vpkm/kwiR07aZXnsKPxw== + "@types/chai@*", "@types/chai@^4.2.11": version "4.2.12" resolved "https://registry.npmjs.org/@types/chai/-/chai-4.2.12.tgz#6160ae454cd89dae05adc3bb97997f488b608201" @@ -753,6 +901,14 @@ resolved "https://registry.npmjs.org/@types/color-name/-/color-name-1.1.1.tgz#1c1261bbeaa10a8055bbc5d8ab84b7b2afc846a0" integrity sha512-rr+OQyAjxze7GgWrSaJwydHStIhHq2lvY3BOC2Mj7KnzI7XK0Uw1TOOdI9lDoajEbSWLiYgoo4f1R51erQfhPQ== +"@types/glob@*": + version "7.1.4" + resolved "https://registry.yarnpkg.com/@types/glob/-/glob-7.1.4.tgz#ea59e21d2ee5c517914cb4bc8e4153b99e566672" + integrity sha512-w+LsMxKyYQm347Otw+IfBXOv9UWVjpHpCDdbBMt8Kz/xbvCYNjP+0qPh91Km3iKfSRLBB0P7fAMf0KHrPu+MyA== + dependencies: + "@types/minimatch" "*" + "@types/node" "*" + "@types/glob@^7.1.1": version "7.1.3" resolved "https://registry.npmjs.org/@types/glob/-/glob-7.1.3.tgz#e6ba80f36b7daad2c685acd9266382e68985c183" @@ -832,6 +988,14 @@ resolved "https://registry.npmjs.org/@types/parse-json/-/parse-json-4.0.0.tgz#2f8bb441434d163b35fb8ffdccd7138927ffb8c0" integrity sha512-//oorEZjL6sbPcKUaCdIGlIUeH26mgzimjBB77G6XRgnDl/L5wOnpyBGRe/Mmf5CVW3PwEBE1NjiMZ/ssFh4wA== +"@types/shelljs@^0.8.9": + version "0.8.9" + resolved "https://registry.yarnpkg.com/@types/shelljs/-/shelljs-0.8.9.tgz#45dd8501aa9882976ca3610517dac3831c2fbbf4" + integrity sha512-flVe1dvlrCyQJN/SGrnBxqHG+RzXrVKsmjD8WS/qYHpq5UPjfq7UWFBENP0ZuOl0g6OpAlL6iBoLSvKYUUmyQw== + dependencies: + "@types/glob" "*" + "@types/node" "*" + "@types/sinon@*", "@types/sinon@^9.0.4": version "9.0.4" resolved "https://registry.npmjs.org/@types/sinon/-/sinon-9.0.4.tgz#e934f904606632287a6e7f7ab0ce3f08a0dad4b1" @@ -926,6 +1090,11 @@ JSONStream@^1.0.4: jsonparse "^1.2.0" through ">=2.2.7 <3" +abbrev@1, abbrev@~1.1.1: + version "1.1.1" + resolved "https://registry.yarnpkg.com/abbrev/-/abbrev-1.1.1.tgz#f8f2c887ad10bf67f634f005b6987fed3179aac8" + integrity sha512-nne9/IiQ/hzIhY6pdDnbBtz7DjPTKrY00P/zvPSm5pOFkl6xuGrGnXn/VtTNNfNtAfZ9/1RtehkszU9qcTii0Q== + acorn-jsx@^5.2.0: version "5.2.0" resolved "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-5.2.0.tgz#4c66069173d6fdd68ed85239fc256226182b2ebe" @@ -946,6 +1115,22 @@ acorn@^7.4.0: resolved "https://registry.yarnpkg.com/acorn/-/acorn-7.4.1.tgz#feaed255973d2e77555b83dbc08851a6c63520fa" integrity sha512-nQyp0o1/mNdbTO1PO6kHkwSrmgZ0MT/jCCpNiwbUjGoRN4dlBhqJtoQuCnEOKzgTVwg0ZWiCoQy6SxMebQVh8A== +agent-base@6, agent-base@^6.0.2: + version "6.0.2" + resolved "https://registry.yarnpkg.com/agent-base/-/agent-base-6.0.2.tgz#49fff58577cfee3f37176feab4c22e00f86d7f77" + integrity sha512-RZNwNclF7+MS/8bDg70amg32dyeZGZxiDuQmZxKLAlQjr3jGyLx+4Kkk58UO7D2QdgFIQCovuSuZESne6RG6XQ== + dependencies: + debug "4" + +agentkeepalive@^4.1.3: + version "4.1.4" + resolved "https://registry.yarnpkg.com/agentkeepalive/-/agentkeepalive-4.1.4.tgz#d928028a4862cb11718e55227872e842a44c945b" + integrity sha512-+V/rGa3EuU74H6wR04plBb7Ks10FbtUQgRj/FQOG7uUIEuaINI+AiqJR1k6t3SVNs7o7ZjIdus6706qqzVq8jQ== + dependencies: + debug "^4.1.0" + depd "^1.1.2" + humanize-ms "^1.2.1" + aggregate-error@^3.0.0: version "3.0.1" resolved "https://registry.npmjs.org/aggregate-error/-/aggregate-error-3.0.1.tgz#db2fe7246e536f40d9b5442a39e117d7dd6a24e0" @@ -991,6 +1176,11 @@ ansi-escapes@^4.2.1, ansi-escapes@^4.3.0: dependencies: type-fest "^0.11.0" +ansi-regex@^2.0.0: + version "2.1.1" + resolved "https://registry.yarnpkg.com/ansi-regex/-/ansi-regex-2.1.1.tgz#c3b33ab5ee360d86e0e628f0468ae7ef27d654df" + integrity sha1-w7M6te42DYbg5ijwRorn7yfWVN8= + ansi-regex@^3.0.0: version "3.0.0" resolved "https://registry.npmjs.org/ansi-regex/-/ansi-regex-3.0.0.tgz#ed0317c322064f79466c02966bddb605ab37d998" @@ -1021,11 +1211,23 @@ ansi-styles@^4.0.0, ansi-styles@^4.1.0, ansi-styles@^4.2.0: "@types/color-name" "^1.1.1" color-convert "^2.0.1" +ansi-styles@^4.3.0: + version "4.3.0" + resolved "https://registry.yarnpkg.com/ansi-styles/-/ansi-styles-4.3.0.tgz#edd803628ae71c04c85ae7a0906edad34b648937" + integrity sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg== + dependencies: + color-convert "^2.0.1" + ansicolors@~0.3.2: version "0.3.2" resolved "https://registry.npmjs.org/ansicolors/-/ansicolors-0.3.2.tgz#665597de86a9ffe3aa9bfbe6cae5c6ea426b4979" integrity sha1-ZlWX3oap/+Oqm/vmyuXG6kJrSXk= +ansistyles@~0.1.3: + version "0.1.3" + resolved "https://registry.yarnpkg.com/ansistyles/-/ansistyles-0.1.3.tgz#5de60415bda071bb37127854c864f41b23254539" + integrity sha1-XeYEFb2gcbs3EnhUyGT0GyMlRTk= + anymatch@~3.1.1: version "3.1.1" resolved "https://registry.npmjs.org/anymatch/-/anymatch-3.1.1.tgz#c55ecf02185e2469259399310c173ce31233b142" @@ -1041,6 +1243,16 @@ append-transform@^2.0.0: dependencies: default-require-extensions "^3.0.0" +aproba@^1.0.3: + version "1.2.0" + resolved "https://registry.yarnpkg.com/aproba/-/aproba-1.2.0.tgz#6802e6264efd18c790a1b0d517f0f2627bf2c94a" + integrity sha512-Y9J6ZjXtoYh8RnXVCMOU/ttDmk1aBjunq9vO0ta5x85WDQiQfUF9sIPBITdbiiIVcBo03Hi3jMxigBtsddlXRw== + +"aproba@^1.0.3 || ^2.0.0", aproba@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/aproba/-/aproba-2.0.0.tgz#52520b8ae5b569215b354efc0caa3fe1e45a8adc" + integrity sha512-lYe4Gx7QT+MKGbDsA+Z+he/Wtef0BiwDOlK/XkBrdfsh9J/jPPXbX0tE9x9cl27Tmu5gg3QUbUrQYa/y+KOHPQ== + archiver-utils@^2.1.0: version "2.1.0" resolved "https://registry.yarnpkg.com/archiver-utils/-/archiver-utils-2.1.0.tgz#e8a460e94b693c3e3da182a098ca6285ba9249e2" @@ -1070,11 +1282,19 @@ archiver@^5.2.0: tar-stream "^2.1.4" zip-stream "^4.0.4" -archy@^1.0.0: +archy@^1.0.0, archy@~1.0.0: version "1.0.0" - resolved "https://registry.npmjs.org/archy/-/archy-1.0.0.tgz#f9c8c13757cc1dd7bc379ac77b2c62a5c2868c40" + resolved "https://registry.yarnpkg.com/archy/-/archy-1.0.0.tgz#f9c8c13757cc1dd7bc379ac77b2c62a5c2868c40" integrity sha1-+cjBN1fMHde8N5rHeyxipcKGjEA= +are-we-there-yet@^1.1.5, are-we-there-yet@~1.1.2: + version "1.1.5" + resolved "https://registry.yarnpkg.com/are-we-there-yet/-/are-we-there-yet-1.1.5.tgz#4b35c2944f062a8bfcda66410760350fe9ddfc21" + integrity sha512-5hYdAkZlcG8tOLujVDTgCT+uPX0VnpAH28gWsLfzpXYm7wP6mp5Q/gYyR7YQ0cKVJcXJnl3j2kpBan13PtQf6w== + dependencies: + delegates "^1.0.0" + readable-stream "^2.0.6" + arg@^4.1.0: version "4.1.3" resolved "https://registry.npmjs.org/arg/-/arg-4.1.3.tgz#269fc7ad5b8e42cb63c896d5666017261c144089" @@ -1184,7 +1404,7 @@ arrify@^2.0.1: resolved "https://registry.npmjs.org/arrify/-/arrify-2.0.1.tgz#c9655e9331e0abcd588d2a7cad7e9956f66701fa" integrity sha512-3duEwti880xqi4eAMN8AyR4a0ByT90zoYdLlevfrvU43vb0YZwZVfxOgxWrLXXXpyugL0hNZc9G6BiB5B3nUug== -asap@*, asap@~2.0.3, asap@~2.0.6: +asap@*, asap@^2.0.0, asap@~2.0.3, asap@~2.0.6: version "2.0.6" resolved "https://registry.npmjs.org/asap/-/asap-2.0.6.tgz#e50347611d7e690943208bbdafebcbc2fb866d46" integrity sha1-5QNHYR1+aQlDIIu9r+vLwvuGbUY= @@ -1286,11 +1506,28 @@ bcrypt-pbkdf@^1.0.0: dependencies: tweetnacl "^0.14.3" +bin-links@^2.2.1: + version "2.2.1" + resolved "https://registry.yarnpkg.com/bin-links/-/bin-links-2.2.1.tgz#347d9dbb48f7d60e6c11fe68b77a424bee14d61b" + integrity sha512-wFzVTqavpgCCYAh8SVBdnZdiQMxTkGR+T3b14CNpBXIBe2neJWaMGAZ55XWWHELJJ89dscuq0VCBqcVaIOgCMg== + dependencies: + cmd-shim "^4.0.1" + mkdirp "^1.0.3" + npm-normalize-package-bin "^1.0.0" + read-cmd-shim "^2.0.0" + rimraf "^3.0.0" + write-file-atomic "^3.0.3" + binary-extensions@^2.0.0: version "2.1.0" resolved "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.1.0.tgz#30fa40c9e7fe07dbc895678cd287024dea241dd9" integrity sha512-1Yj8h9Q+QDF5FzhMs/c9+6UntbD5MkRfRwac8DoEm9ZfUBZ7tZ55YcGVAzEe4bXsdQHEk+s9S5wsOKVdZrw0tQ== +binary-extensions@^2.2.0: + version "2.2.0" + resolved "https://registry.yarnpkg.com/binary-extensions/-/binary-extensions-2.2.0.tgz#75f502eeaf9ffde42fc98829645be4ea76bd9e2d" + integrity sha512-jDctJ/IVQbZoJykoeHbhXpOlNBqGNcwXJKJog42E5HDPUwQTSdjCHdihjj0DlnheQ7blbT6dHOafNAiS8ooQKA== + bl@^4.0.1: version "4.0.3" resolved "https://registry.yarnpkg.com/bl/-/bl-4.0.3.tgz#12d6287adc29080e22a705e5764b2a9522cdc489" @@ -1368,6 +1605,34 @@ buffer@^5.5.0: base64-js "^1.3.1" ieee754 "^1.1.13" +builtins@^1.0.3: + version "1.0.3" + resolved "https://registry.yarnpkg.com/builtins/-/builtins-1.0.3.tgz#cb94faeb61c8696451db36534e1422f94f0aee88" + integrity sha1-y5T662HIaWRR2zZTThQi+U8K7og= + +cacache@^15.0.3, cacache@^15.0.5, cacache@^15.2.0: + version "15.2.0" + resolved "https://registry.yarnpkg.com/cacache/-/cacache-15.2.0.tgz#73af75f77c58e72d8c630a7a2858cb18ef523389" + integrity sha512-uKoJSHmnrqXgthDFx/IU6ED/5xd+NNGe+Bb+kLZy7Ku4P+BaiWEUflAKPZ7eAzsYGcsAGASJZsybXp+quEcHTw== + dependencies: + "@npmcli/move-file" "^1.0.1" + chownr "^2.0.0" + fs-minipass "^2.0.0" + glob "^7.1.4" + infer-owner "^1.0.4" + lru-cache "^6.0.0" + minipass "^3.1.1" + minipass-collect "^1.0.2" + minipass-flush "^1.0.5" + minipass-pipeline "^1.2.2" + mkdirp "^1.0.3" + p-map "^4.0.0" + promise-inflight "^1.0.1" + rimraf "^3.0.2" + ssri "^8.0.1" + tar "^6.0.2" + unique-filename "^1.1.1" + cache-base@^1.0.1: version "1.0.1" resolved "https://registry.npmjs.org/cache-base/-/cache-base-1.0.1.tgz#0a7f46416831c8b662ee36fe4e7c59d76f666ab2" @@ -1501,6 +1766,14 @@ chalk@^4.0.0, chalk@^4.1.0, chalk@^4.1.1: ansi-styles "^4.1.0" supports-color "^7.1.0" +chalk@^4.1.2: + version "4.1.2" + resolved "https://registry.yarnpkg.com/chalk/-/chalk-4.1.2.tgz#aac4e2b7734a740867aeb16bf02aad556a1e7a01" + integrity sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA== + dependencies: + ansi-styles "^4.1.0" + supports-color "^7.1.0" + chardet@^0.7.0: version "0.7.0" resolved "https://registry.npmjs.org/chardet/-/chardet-0.7.0.tgz#90094849f0937f2eedc2425d0d28a9e5f0cbad9e" @@ -1531,11 +1804,23 @@ chownr@^1.1.1: resolved "https://registry.npmjs.org/chownr/-/chownr-1.1.4.tgz#6fc9d7b42d32a583596337666e7d08084da2cc6b" integrity sha512-jJ0bqzaylmJtVnNgzTeSOs8DPavpbYgEr/b0YL8/2GO3xJEhInFmhKMUnEJQjZumK7KXGFhUy89PrsJWlakBVg== +chownr@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/chownr/-/chownr-2.0.0.tgz#15bfbe53d2eab4cf70f18a8cd68ebe5b3cb1dece" + integrity sha512-bIomtDF5KGpdogkLd9VspvFzk9KfpyyGlS8YFVZl7TGPBHL5snIOnxeshwVgPteQ9b4Eydl+pVbIyE1DcvCWgQ== + ci-info@^2.0.0: version "2.0.0" resolved "https://registry.npmjs.org/ci-info/-/ci-info-2.0.0.tgz#67a9e964be31a51e15e5010d58e6f12834002f46" integrity sha512-5tK7EtrZ0N+OLFMthtqOj4fI2Jeb88C4CAZPu25LDVUgXJ0A3Js4PMGqrn0JU1W0Mh1/Z8wZzYPxqUrXeBboCQ== +cidr-regex@^3.1.1: + version "3.1.1" + resolved "https://registry.yarnpkg.com/cidr-regex/-/cidr-regex-3.1.1.tgz#ba1972c57c66f61875f18fd7dd487469770b571d" + integrity sha512-RBqYd32aDwbCMFJRL6wHOlDNYJsPNTt8vC82ErHF5vKt8QQzxm1FrkW8s/R5pVrXMf17sba09Uoy91PKiddAsw== + dependencies: + ip-regex "^4.1.0" + class-utils@^0.3.5: version "0.3.6" resolved "https://registry.npmjs.org/class-utils/-/class-utils-0.3.6.tgz#f93369ae8b9a7ce02fd41faad0ca83033190c463" @@ -1558,6 +1843,14 @@ clean-stack@^3.0.0: dependencies: escape-string-regexp "4.0.0" +cli-columns@^3.1.2: + version "3.1.2" + resolved "https://registry.yarnpkg.com/cli-columns/-/cli-columns-3.1.2.tgz#6732d972979efc2ae444a1f08e08fa139c96a18e" + integrity sha1-ZzLZcpee/CrkRKHwjgj6E5yWoY4= + dependencies: + string-width "^2.0.0" + strip-ansi "^3.0.1" + cli-cursor@^2.1.0: version "2.1.0" resolved "https://registry.npmjs.org/cli-cursor/-/cli-cursor-2.1.0.tgz#b35dac376479facc3e94747d41d0d0f5238ffcb5" @@ -1580,6 +1873,16 @@ cli-progress@^3.4.0: colors "^1.1.2" string-width "^4.2.0" +cli-table3@^0.6.0: + version "0.6.0" + resolved "https://registry.yarnpkg.com/cli-table3/-/cli-table3-0.6.0.tgz#b7b1bc65ca8e7b5cef9124e13dc2b21e2ce4faee" + integrity sha512-gnB85c3MGC7Nm9I/FkiasNBOKjOiO1RNuXXarQms37q4QMpWdlbBgD/VnOStA2faG1dpXMv31RFApjX1/QdgWQ== + dependencies: + object-assign "^4.1.0" + string-width "^4.2.0" + optionalDependencies: + colors "^1.1.2" + cli-truncate@^2.1.0: version "2.1.0" resolved "https://registry.npmjs.org/cli-truncate/-/cli-truncate-2.1.0.tgz#c39e28bf05edcde5be3b98992a22deed5a2b93c7" @@ -1675,6 +1978,18 @@ cliui@^7.0.2: strip-ansi "^6.0.0" wrap-ansi "^7.0.0" +clone@^1.0.2: + version "1.0.4" + resolved "https://registry.yarnpkg.com/clone/-/clone-1.0.4.tgz#da309cc263df15994c688ca902179ca3c7cd7c7e" + integrity sha1-2jCcwmPfFZlMaIypAheco8fNfH4= + +cmd-shim@^4.0.1: + version "4.1.0" + resolved "https://registry.yarnpkg.com/cmd-shim/-/cmd-shim-4.1.0.tgz#b3a904a6743e9fede4148c6f3800bf2a08135bdd" + integrity sha512-lb9L7EM4I/ZRVuljLPEtUJOP+xiQVknZ4ZMpMgEp4JzNldPb27HU03hi6K1/6CoIuit/Zm/LQXySErFeXxDprw== + dependencies: + mkdirp-infer-owner "^2.0.0" + co-prompt@^1.0.0: version "1.0.0" resolved "https://registry.npmjs.org/co-prompt/-/co-prompt-1.0.0.tgz#fb370e9edac48576b27a732fe5d7f21d9fc6e6f6" @@ -1682,6 +1997,11 @@ co-prompt@^1.0.0: dependencies: keypress "~0.2.1" +code-point-at@^1.0.0: + version "1.1.0" + resolved "https://registry.yarnpkg.com/code-point-at/-/code-point-at-1.1.0.tgz#0d070b4d043a5bea33a2f1a40e2edb3d9a4ccf77" + integrity sha1-DQcLTQQ6W+ozovGkDi7bPZpMz3c= + coffeescript@^1.10.0: version "1.12.7" resolved "https://registry.npmjs.org/coffeescript/-/coffeescript-1.12.7.tgz#e57ee4c4867cf7f606bfc4a0f2d550c0981ddd27" @@ -1719,6 +2039,11 @@ color-name@~1.1.4: resolved "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz#c2a09a87acbde69543de6f63fa3995c826c536a2" integrity sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA== +color-support@^1.1.2: + version "1.1.3" + resolved "https://registry.yarnpkg.com/color-support/-/color-support-1.1.3.tgz#93834379a1cc9a0c61f82f52f0d04322251bd5a2" + integrity sha512-qiBjkpbMLO/HL68y+lh4q0/O1MZFj2RX6X/KmMa3+gJD3z+WwI1ZzDHysvqHGS3mP6mznPckpXmw1nI9cJjyRg== + colorette@^1.2.2: version "1.2.2" resolved "https://registry.yarnpkg.com/colorette/-/colorette-1.2.2.tgz#cbcc79d5e99caea2dbf10eb3a26fd8b3e6acfa94" @@ -1729,6 +2054,14 @@ colors@^1.1.2: resolved "https://registry.npmjs.org/colors/-/colors-1.4.0.tgz#c50491479d4c1bdaed2c9ced32cf7c7dc2360f78" integrity sha512-a+UqTh4kgZg/SlGvfbzDHpgRu7AAQOmmqRHJnxhRZICKFUT91brVhNNt58CMWU9PsBbv3PDCZUHbVxuDiH2mtA== +columnify@~1.5.4: + version "1.5.4" + resolved "https://registry.yarnpkg.com/columnify/-/columnify-1.5.4.tgz#4737ddf1c7b69a8a7c340570782e947eec8e78bb" + integrity sha1-Rzfd8ce2mop8NAVweC6UfuyOeLs= + dependencies: + strip-ansi "^3.0.0" + wcwidth "^1.0.0" + combined-stream@^1.0.6, combined-stream@~1.0.6: version "1.0.8" resolved "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz#c3d45a8b34fd730631a110a8a2520682b31d5a7f" @@ -1776,6 +2109,11 @@ commitizen@^4.0.3: strip-bom "4.0.0" strip-json-comments "3.0.1" +common-ancestor-path@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/common-ancestor-path/-/common-ancestor-path-1.0.1.tgz#4f7d2d1394d91b7abdf51871c62f71eadb0182a7" + integrity sha512-L3sHRo1pXXEqX8VU28kfgUY+YGsk09hPqZiZmLacNib6XNTCM8ubYeT7ryXQw8asB1sKgcU5lkB7ONug08aB8w== + commondir@^1.0.1: version "1.0.1" resolved "https://registry.npmjs.org/commondir/-/commondir-1.0.1.tgz#ddd800da0c66127393cca5950ea968a3aaf1253b" @@ -1814,6 +2152,11 @@ concat-map@0.0.1: resolved "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz#d8a96bd77fd68df7793a73036a3ba0d5405d477b" integrity sha1-2Klr13/Wjfd5OnMDajug1UBdR3s= +console-control-strings@^1.0.0, console-control-strings@^1.1.0, console-control-strings@~1.1.0: + version "1.1.0" + resolved "https://registry.yarnpkg.com/console-control-strings/-/console-control-strings-1.1.0.tgz#3d7cf4464db6446ea644bf4b39507f9851008e8e" + integrity sha1-PXz0Rk22RG6mRL9LOVB/mFEAjo4= + contains-path@^0.1.0: version "0.1.0" resolved "https://registry.npmjs.org/contains-path/-/contains-path-0.1.0.tgz#fe8cf184ff6670b6baef01a9d4861a5cbec4120a" @@ -2023,6 +2366,13 @@ dayjs@^1.8.16: resolved "https://registry.npmjs.org/dayjs/-/dayjs-1.8.33.tgz#18bc4a2b6c1c6f4d67b4c4f2536c0b97e5b766f7" integrity sha512-881TDLZCdpJFKbraWRHcUG8zfMLLX400ENf9rFZDuWc5zYMss6xifo2PhlDX0ftOmR2NRmaIY47bAa4gKQfXqw== +debug@4, debug@^4.0.1, debug@^4.1.0, debug@^4.1.1, debug@^4.3.1, debug@^4.3.2: + version "4.3.2" + resolved "https://registry.yarnpkg.com/debug/-/debug-4.3.2.tgz#f0a49c18ac8779e31d4a0c6029dfb76873c7428b" + integrity sha512-mOp8wKcvj7XxC78zLgw/ZA+6TSgkoE2C/ienthhRD298T7UNwAg9diBpLRxC0mOezLl4B0xV7M0cCO6P/O0Xhw== + dependencies: + ms "2.1.2" + debug@4.3.1: version "4.3.1" resolved "https://registry.npmjs.org/debug/-/debug-4.3.1.tgz#f0d229c505e0c6d8c49ac553d1b13dc183f6b2ee" @@ -2044,12 +2394,10 @@ debug@^3.1.0, debug@^3.2.7: dependencies: ms "^2.1.1" -debug@^4.0.1, debug@^4.1.0, debug@^4.1.1, debug@^4.3.1, debug@^4.3.2: - version "4.3.2" - resolved "https://registry.yarnpkg.com/debug/-/debug-4.3.2.tgz#f0a49c18ac8779e31d4a0c6029dfb76873c7428b" - integrity sha512-mOp8wKcvj7XxC78zLgw/ZA+6TSgkoE2C/ienthhRD298T7UNwAg9diBpLRxC0mOezLl4B0xV7M0cCO6P/O0Xhw== - dependencies: - ms "2.1.2" +debuglog@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/debuglog/-/debuglog-1.0.1.tgz#aa24ffb9ac3df9a2351837cfb2d279360cd78492" + integrity sha1-qiT/uaw9+aI1GDfPstJ5NgzXhJI= decamelize-keys@^1.1.0: version "1.1.0" @@ -2098,6 +2446,13 @@ default-require-extensions@^3.0.0: dependencies: strip-bom "^4.0.0" +defaults@^1.0.3: + version "1.0.3" + resolved "https://registry.yarnpkg.com/defaults/-/defaults-1.0.3.tgz#c656051e9817d9ff08ed881477f3fe4019f3ef7d" + integrity sha1-xlYFHpgX2f8I7YgUd/P+QBnz730= + dependencies: + clone "^1.0.2" + define-properties@^1.1.2, define-properties@^1.1.3: version "1.1.3" resolved "https://registry.npmjs.org/define-properties/-/define-properties-1.1.3.tgz#cf88da6cbee26fe6db7094f61d870cbd84cee9f1" @@ -2132,6 +2487,16 @@ delayed-stream@~1.0.0: resolved "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz#df3ae199acadfb7d440aaae0b29e2272b24ec619" integrity sha1-3zrhmayt+31ECqrgsp4icrJOxhk= +delegates@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/delegates/-/delegates-1.0.0.tgz#84c6e159b81904fdca59a0ef44cd870d31250f9a" + integrity sha1-hMbhWbgZBP3KWaDvRM2HDTElD5o= + +depd@^1.1.2: + version "1.1.2" + resolved "https://registry.yarnpkg.com/depd/-/depd-1.1.2.tgz#9bcd52e14c097763e749b274c4346ed2e560b5a9" + integrity sha1-m81S4UwJd2PnSbJ0xDRu0uVgtak= + detect-file@^1.0.0: version "1.0.0" resolved "https://registry.npmjs.org/detect-file/-/detect-file-1.0.0.tgz#f0d66d03672a825cb1b73bdb3fe62310c8e552b7" @@ -2142,6 +2507,14 @@ detect-indent@6.0.0, detect-indent@^6.0.0: resolved "https://registry.npmjs.org/detect-indent/-/detect-indent-6.0.0.tgz#0abd0f549f69fc6659a254fe96786186b6f528fd" integrity sha512-oSyFlqaTHCItVRGK5RmrmjB+CmaMOW7IaNA/kdxqhoa6d17j/5ce9O9eWXmV/KEdRwqpQA+Vqe8a8Bsybu4YnA== +dezalgo@^1.0.0: + version "1.0.3" + resolved "https://registry.yarnpkg.com/dezalgo/-/dezalgo-1.0.3.tgz#7f742de066fc748bc8db820569dddce49bf0d456" + integrity sha1-f3Qt4Gb8dIvI24IFad3c5Jvw1FY= + dependencies: + asap "^2.0.0" + wrappy "1" + diff@5.0.0, diff@^5.0.0: version "5.0.0" resolved "https://registry.yarnpkg.com/diff/-/diff-5.0.0.tgz#7ed6ad76d859d030787ec35855f5b1daf31d852b" @@ -2230,6 +2603,13 @@ emoji-regex@^8.0.0: resolved "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz#e818fd69ce5ccfcb404594f842963bf53164cc37" integrity sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A== +encoding@^0.1.12: + version "0.1.13" + resolved "https://registry.yarnpkg.com/encoding/-/encoding-0.1.13.tgz#56574afdd791f54a8e9b2785c0582a2d26210fa9" + integrity sha512-ETBauow1T35Y/WZMkio9jiM0Z5xjHHmJ4XmjZOq1l/dXz3lr2sRn87nJy20RupqSh1F2m3HHPSp8ShIPQJrJ3A== + dependencies: + iconv-lite "^0.6.2" + end-of-stream@^1.1.0, end-of-stream@^1.4.1: version "1.4.4" resolved "https://registry.npmjs.org/end-of-stream/-/end-of-stream-1.4.4.tgz#5ae64a5f45057baf3626ec14da0ca5e4b2431eb0" @@ -2244,6 +2624,16 @@ enquirer@^2.3.5, enquirer@^2.3.6: dependencies: ansi-colors "^4.1.1" +env-paths@^2.2.0: + version "2.2.1" + resolved "https://registry.yarnpkg.com/env-paths/-/env-paths-2.2.1.tgz#420399d416ce1fbe9bc0a07c62fa68d67fd0f8f2" + integrity sha512-+h1lkLKhZMTYjog1VEpJNG7NZJWcuc2DDk/qsqSTRRCOXiLjeQ1d1/udrUGhqMxUgAlwKNZ0cf2uqan5GLuS2A== + +err-code@^2.0.2: + version "2.0.3" + resolved "https://registry.yarnpkg.com/err-code/-/err-code-2.0.3.tgz#23c2f3b756ffdfc608d30e27c9a941024807e7f9" + integrity sha512-2bmlRpNKBxT/CRmPOlyISQpNj+qSeYvcym/uT0Jx2bMOlKLtSy1ZmLuVxSEKKyor/N5yhvp/ZiG1oE3DEYMSFA== + error-ex@^1.2.0, error-ex@^1.3.1: version "1.3.2" resolved "https://registry.npmjs.org/error-ex/-/error-ex-1.3.2.tgz#b4ac40648107fdcdcfae242f428bea8a14d4f1bf" @@ -2823,6 +3213,11 @@ fast-levenshtein@^2.0.6, fast-levenshtein@~2.0.6: resolved "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz#3d8a5c66883a16a30ca8643e851f19baa7797917" integrity sha1-PYpcZog6FqMMqGQ+hR8Zuqd5eRc= +fastest-levenshtein@^1.0.12: + version "1.0.12" + resolved "https://registry.yarnpkg.com/fastest-levenshtein/-/fastest-levenshtein-1.0.12.tgz#9990f7d3a88cc5a9ffd1f1745745251700d497e2" + integrity sha512-On2N+BpYJ15xIC974QNVuYGMOlEVt4s0EOI3wwMqOmK1fdDY+FN/zltPV8vosq4ad4c/gJ1KHScUn/6AWIgiow== + fastq@^1.6.0: version "1.8.0" resolved "https://registry.npmjs.org/fastq/-/fastq-1.8.0.tgz#550e1f9f59bbc65fe185cb6a9b4d95357107f481" @@ -3091,6 +3486,13 @@ fs-extra@^9.0.1: jsonfile "^6.0.1" universalify "^1.0.0" +fs-minipass@^2.0.0, fs-minipass@^2.1.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/fs-minipass/-/fs-minipass-2.1.0.tgz#7f5036fdbf12c63c169190cbe4199c852271f9fb" + integrity sha512-V/JgOLFCS+R6Vcq0slCuaeWEdNC3ouDlJMNIsacH2VtALiu9mV4LPrHc5cDl8k5aw6J8jwgWWpiTo5RYhmIzvg== + dependencies: + minipass "^3.0.0" + fs.realpath@^1.0.0: version "1.0.0" resolved "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz#1504ad2523158caa40db4a2787cb01411994ea4f" @@ -3111,6 +3513,35 @@ functional-red-black-tree@^1.0.1: resolved "https://registry.npmjs.org/functional-red-black-tree/-/functional-red-black-tree-1.0.1.tgz#1b0ab3bd553b2a0d6399d29c0e3ea0b252078327" integrity sha1-GwqzvVU7Kg1jmdKcDj6gslIHgyc= +gauge@^3.0.0: + version "3.0.1" + resolved "https://registry.yarnpkg.com/gauge/-/gauge-3.0.1.tgz#4bea07bcde3782f06dced8950e51307aa0f4a346" + integrity sha512-6STz6KdQgxO4S/ko+AbjlFGGdGcknluoqU+79GOFCDqqyYj5OanQf9AjxwN0jCidtT+ziPMmPSt9E4hfQ0CwIQ== + dependencies: + aproba "^1.0.3 || ^2.0.0" + color-support "^1.1.2" + console-control-strings "^1.0.0" + has-unicode "^2.0.1" + object-assign "^4.1.1" + signal-exit "^3.0.0" + string-width "^1.0.1 || ^2.0.0" + strip-ansi "^3.0.1 || ^4.0.0" + wide-align "^1.1.2" + +gauge@~2.7.3: + version "2.7.4" + resolved "https://registry.yarnpkg.com/gauge/-/gauge-2.7.4.tgz#2c03405c7538c39d7eb37b317022e325fb018bf7" + integrity sha1-LANAXHU4w51+s3sxcCLjJfsBi/c= + dependencies: + aproba "^1.0.3" + console-control-strings "^1.0.0" + has-unicode "^2.0.0" + object-assign "^4.1.0" + signal-exit "^3.0.0" + string-width "^1.0.1" + strip-ansi "^3.0.1" + wide-align "^1.1.0" + gensync@^1.0.0-beta.1: version "1.0.0-beta.1" resolved "https://registry.npmjs.org/gensync/-/gensync-1.0.0-beta.1.tgz#58f4361ff987e5ff6e1e7a210827aa371eaac269" @@ -3249,7 +3680,7 @@ glob@^6.0.1: once "^1.3.0" path-is-absolute "^1.0.0" -glob@^7.1.7: +glob@^7.1.1, glob@^7.1.7: version "7.1.7" resolved "https://registry.yarnpkg.com/glob/-/glob-7.1.7.tgz#3b193e9233f01d42d0b3f78294bbeeb418f94a90" integrity sha512-OvD9ENzPLbegENnYP5UUfJIirTg4+XwMWGaQfQTY0JenxNvvIKP3U3/tAQSPIu/lHxXYSZmpXlUHeqAIdKzBLQ== @@ -3350,6 +3781,11 @@ graceful-fs@^4.1.15, graceful-fs@^4.1.2, graceful-fs@^4.1.6, graceful-fs@^4.2.0, resolved "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.4.tgz#2256bde14d3632958c465ebc96dc467ca07a29fb" integrity sha512-WjKPNJF79dtJAVniUlGGWHYGz2jWxT6VhN/4m1NdkbZ2nOsEF+cI1Edgql5zCRhs/VsQYRvrXctxktVXZUkixw== +graceful-fs@^4.2.3, graceful-fs@^4.2.8: + version "4.2.8" + resolved "https://registry.yarnpkg.com/graceful-fs/-/graceful-fs-4.2.8.tgz#e412b8d33f5e006593cbd3cee6df9f2cebbe802a" + integrity sha512-qkIilPUYcNhJpd33n0GBXTB1MMPp14TxEsEs0pTrsSVucApsYzW5V+Q8Qxhik6KU3evy+qkAAowTByymK0avdg== + growl@1.10.5: version "1.10.5" resolved "https://registry.npmjs.org/growl/-/growl-1.10.5.tgz#f2735dc2283674fa67478b10181059355c369e5e" @@ -3415,6 +3851,11 @@ has-symbols@^1.0.2: resolved "https://registry.yarnpkg.com/has-symbols/-/has-symbols-1.0.2.tgz#165d3070c00309752a1236a479331e3ac56f1423" integrity sha512-chXa79rL/UC2KlX17jo3vRGz0azaWEx5tGqZg5pO3NUyEJVB17dMruQlzCCOfUvElghKcm5194+BCRvi2Rv/Gw== +has-unicode@^2.0.0, has-unicode@^2.0.1: + version "2.0.1" + resolved "https://registry.yarnpkg.com/has-unicode/-/has-unicode-2.0.1.tgz#e0e6fe6a28cf51138855e086d1691e771de2a8b9" + integrity sha1-4Ob+aijPUROIVeCG0Wkedx3iqLk= + has-value@^0.3.1: version "0.3.1" resolved "https://registry.npmjs.org/has-value/-/has-value-0.3.1.tgz#7b1f58bada62ca827ec0a2078025654845995e1f" @@ -3490,11 +3931,23 @@ hosted-git-info@^3.0.6: dependencies: lru-cache "^6.0.0" +hosted-git-info@^4.0.1, hosted-git-info@^4.0.2: + version "4.0.2" + resolved "https://registry.yarnpkg.com/hosted-git-info/-/hosted-git-info-4.0.2.tgz#5e425507eede4fea846b7262f0838456c4209961" + integrity sha512-c9OGXbZ3guC/xOlCg1Ci/VgWlwsqDv1yMQL1CWqXDL0hDjXuNcq0zuR4xqPSuasI3kqFDhqSyTjREz5gzq0fXg== + dependencies: + lru-cache "^6.0.0" + html-escaper@^2.0.0: version "2.0.2" resolved "https://registry.npmjs.org/html-escaper/-/html-escaper-2.0.2.tgz#dfd60027da36a36dfcbe236262c00a5822681453" integrity sha512-H2iMtd0I4Mt5eYiapRdIDjp+XzelXQ0tFE4JS7YFwFevXXMmOp9myNrUvCg0D6ws8iqkRPBfKHgbwig1SmlLfg== +http-cache-semantics@^4.1.0: + version "4.1.0" + resolved "https://registry.yarnpkg.com/http-cache-semantics/-/http-cache-semantics-4.1.0.tgz#49e91c5cbf36c9b94bcfcd71c23d5249ec74e390" + integrity sha512-carPklcUh7ROWRK7Cv27RPtdhYhUsela/ue5/jKzjegVvXDqM2ILE9Q2BGn9JZJh1g87cp56su/FgQSzcWS8cQ== + http-call@^5.1.2: version "5.3.0" resolved "https://registry.npmjs.org/http-call/-/http-call-5.3.0.tgz#4ded815b13f423de176eb0942d69c43b25b148db" @@ -3512,6 +3965,15 @@ http-parser-js@>=0.5.1: resolved "https://registry.npmjs.org/http-parser-js/-/http-parser-js-0.5.2.tgz#da2e31d237b393aae72ace43882dd7e270a8ff77" integrity sha512-opCO9ASqg5Wy2FNo7A0sxy71yGbbkJJXLdgMK04Tcypw9jr2MgWbyubb0+WdmDmGnFflO7fRbqbaihh/ENDlRQ== +http-proxy-agent@^4.0.1: + version "4.0.1" + resolved "https://registry.yarnpkg.com/http-proxy-agent/-/http-proxy-agent-4.0.1.tgz#8a8c8ef7f5932ccf953c296ca8291b95aa74aa3a" + integrity sha512-k0zdNgqWTGA6aeIRVpvfVob4fL52dTfaehylg0Y4UvSySvOq/Y+BOyPrgpUrA7HylqvU8vIZGsRuXmspskV0Tg== + dependencies: + "@tootallnate/once" "1" + agent-base "6" + debug "4" + http-signature@~1.2.0: version "1.2.0" resolved "https://registry.npmjs.org/http-signature/-/http-signature-1.2.0.tgz#9aecd925114772f3d95b65a60abb8f7c18fbace1" @@ -3521,11 +3983,26 @@ http-signature@~1.2.0: jsprim "^1.2.2" sshpk "^1.7.0" +https-proxy-agent@^5.0.0: + version "5.0.0" + resolved "https://registry.yarnpkg.com/https-proxy-agent/-/https-proxy-agent-5.0.0.tgz#e2a90542abb68a762e0a0850f6c9edadfd8506b2" + integrity sha512-EkYm5BcKUGiduxzSt3Eppko+PiNWNEpa4ySk9vTC6wDsQJW9rHSa+UhGNJoRYp7bz6Ht1eaRIa6QaJqO5rCFbA== + dependencies: + agent-base "6" + debug "4" + human-signals@^2.1.0: version "2.1.0" resolved "https://registry.yarnpkg.com/human-signals/-/human-signals-2.1.0.tgz#dc91fcba42e4d06e4abaed33b3e7a3c02f514ea0" integrity sha512-B4FFZ6q/T2jhhksgkbEW3HBvWIfDW85snkQgawt07S7J5QXTk6BkNV+0yAeZrM5QpMAdYlocGoljn0sJ/WQkFw== +humanize-ms@^1.2.1: + version "1.2.1" + resolved "https://registry.yarnpkg.com/humanize-ms/-/humanize-ms-1.2.1.tgz#c46e3159a293f6b896da29316d8b6fe8bb79bbed" + integrity sha1-xG4xWaKT9riW2ikxbYtv6Lt5u+0= + dependencies: + ms "^2.0.0" + husky@^4.2.5, husky@^4.3.8: version "4.3.8" resolved "https://registry.yarnpkg.com/husky/-/husky-4.3.8.tgz#31144060be963fd6850e5cc8f019a1dfe194296d" @@ -3554,11 +4031,25 @@ iconv-lite@^0.4.24: dependencies: safer-buffer ">= 2.1.2 < 3" +iconv-lite@^0.6.2: + version "0.6.3" + resolved "https://registry.yarnpkg.com/iconv-lite/-/iconv-lite-0.6.3.tgz#a52f80bf38da1952eb5c681790719871a1a72501" + integrity sha512-4fCk79wshMdzMp2rH06qWrJE4iolqLhCUH+OiuIgU++RB0+94NlDL81atO7GX55uUKueo0txHNtvEyI6D7WdMw== + dependencies: + safer-buffer ">= 2.1.2 < 3.0.0" + ieee754@^1.1.13: version "1.2.1" resolved "https://registry.yarnpkg.com/ieee754/-/ieee754-1.2.1.tgz#8eb7a10a63fff25d15a57b001586d177d1b0d352" integrity sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA== +ignore-walk@^3.0.3: + version "3.0.4" + resolved "https://registry.yarnpkg.com/ignore-walk/-/ignore-walk-3.0.4.tgz#c9a09f69b7c7b479a5d74ac1a3c0d4236d2a6335" + integrity sha512-PY6Ii8o1jMRA1z4F2hRkH/xN59ox43DavKvD3oDpfurRlOJyAHpifIwpbdv1n4jt4ov0jSpw3kQ4GhJnpBL6WQ== + dependencies: + minimatch "^3.0.4" + ignore@^4.0.6: version "4.0.6" resolved "https://registry.npmjs.org/ignore/-/ignore-4.0.6.tgz#750e3db5862087b4737ebac8207ffd1ef27b25fc" @@ -3599,6 +4090,11 @@ indent-string@^4.0.0: resolved "https://registry.npmjs.org/indent-string/-/indent-string-4.0.0.tgz#624f8f4497d619b2d9768531d58f4122854d7251" integrity sha512-EdDDZu4A2OyIK7Lr/2zG+w5jmbuk1DVBnEwREQvBzspBJkCEbRa8GxU1lghYcaGJCnRWibjDXlq779X1/y5xwg== +infer-owner@^1.0.4: + version "1.0.4" + resolved "https://registry.yarnpkg.com/infer-owner/-/infer-owner-1.0.4.tgz#c4cefcaa8e51051c2a40ba2ce8a3d27295af9467" + integrity sha512-IClj+Xz94+d7irH5qRyfJonOdfTzuDaifE6ZPWfx0N0+/ATZCbuTPq2prFl526urkQd90WyUKIh1DfBQ2hMz9A== + inflight@^1.0.4: version "1.0.6" resolved "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz#49bd6331d7d02d0c09bc910a1075ba8165b56df9" @@ -3617,6 +4113,25 @@ ini@^1.3.4: resolved "https://registry.yarnpkg.com/ini/-/ini-1.3.8.tgz#a29da425b48806f34767a4efce397269af28432c" integrity sha512-JV/yugV2uzW5iMRSiZAyDtQd+nxtUnjeLt0acNdw98kKLrvuRVyB80tsREOE7yvGVgalhZ6RNXCmEHkUKBKxew== +ini@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/ini/-/ini-2.0.0.tgz#e5fd556ecdd5726be978fa1001862eacb0a94bc5" + integrity sha512-7PnF4oN3CvZF23ADhA5wRaYEQpJ8qygSkbtTXWBeXWXmEVRXK+1ITciHWwHhsjv1TmW0MgacIv6hEi5pX5NQdA== + +init-package-json@^2.0.4: + version "2.0.4" + resolved "https://registry.yarnpkg.com/init-package-json/-/init-package-json-2.0.4.tgz#9f9f66cd5934e6d5f645150e15013d384d0b90d2" + integrity sha512-gUACSdZYka+VvnF90TsQorC+1joAVWNI724vBNj3RD0LLMeDss2IuzaeiQs0T4YzKs76BPHtrp/z3sn2p+KDTw== + dependencies: + glob "^7.1.1" + npm-package-arg "^8.1.2" + promzard "^0.3.0" + read "~1.0.1" + read-package-json "^4.0.0" + semver "^7.3.5" + validate-npm-package-license "^3.0.4" + validate-npm-package-name "^3.0.0" + inquirer@6.5.0: version "6.5.0" resolved "https://registry.npmjs.org/inquirer/-/inquirer-6.5.0.tgz#2303317efc9a4ea7ec2e2df6f86569b734accf42" @@ -3660,6 +4175,16 @@ interpret@^1.0.0: resolved "https://registry.npmjs.org/interpret/-/interpret-1.4.0.tgz#665ab8bc4da27a774a40584e812e3e0fa45b1a1e" integrity sha512-agE4QfB2Lkp9uICn7BAqoscw4SZP9kTE2hxiFI3jBPmXJfdqiahTbUuKGsMoN2GtqL9AxhYioAcVvgsb1HvRbA== +ip-regex@^4.1.0: + version "4.3.0" + resolved "https://registry.yarnpkg.com/ip-regex/-/ip-regex-4.3.0.tgz#687275ab0f57fa76978ff8f4dddc8a23d5990db5" + integrity sha512-B9ZWJxHHOHUhUjCPrMpLD4xEq35bUTClHM1S6CBU5ixQnkZmwipwgc96vAd7AAGM9TGHvJR+Uss+/Ak6UphK+Q== + +ip@^1.1.5: + version "1.1.5" + resolved "https://registry.yarnpkg.com/ip/-/ip-1.1.5.tgz#bdded70114290828c0a039e72ef25f5aaec4354a" + integrity sha1-vd7XARQpCCjAoDnnLvJfWq7ENUo= + is-accessor-descriptor@^0.1.6: version "0.1.6" resolved "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-0.1.6.tgz#a9e12cb3ae8d876727eeef3843f8a0897b5c98d6" @@ -3713,6 +4238,13 @@ is-callable@^1.2.3: resolved "https://registry.yarnpkg.com/is-callable/-/is-callable-1.2.3.tgz#8b1e0500b73a1d76c70487636f368e519de8db8e" integrity sha512-J1DcMe8UYTBSrKezuIUTUwjXsho29693unXM2YhJUTR2txK/eG47bvNa/wipPFmZFgr/N6f1GA66dv0mEyTIyQ== +is-cidr@^4.0.2: + version "4.0.2" + resolved "https://registry.yarnpkg.com/is-cidr/-/is-cidr-4.0.2.tgz#94c7585e4c6c77ceabf920f8cde51b8c0fda8814" + integrity sha512-z4a1ENUajDbEl/Q6/pVBpTR1nBjjEE1X7qb7bmWYanNnPoKAvUCPFKeXV6Fe4mgTkWKBqiHIcwsI3SndiO5FeA== + dependencies: + cidr-regex "^3.1.1" + is-core-module@^2.2.0, is-core-module@^2.4.0: version "2.4.0" resolved "https://registry.yarnpkg.com/is-core-module/-/is-core-module-2.4.0.tgz#8e9fc8e15027b011418026e98f0e6f4d86305cc1" @@ -3720,6 +4252,13 @@ is-core-module@^2.2.0, is-core-module@^2.4.0: dependencies: has "^1.0.3" +is-core-module@^2.5.0: + version "2.6.0" + resolved "https://registry.yarnpkg.com/is-core-module/-/is-core-module-2.6.0.tgz#d7553b2526fe59b92ba3e40c8df757ec8a709e19" + integrity sha512-wShG8vs60jKfPWpF2KZRaAtvt3a20OAn7+IJ6hLPECpSABLcKtFKTTI4ZtH5QcBruBHlq+WsdHWyz0BCZW7svQ== + dependencies: + has "^1.0.3" + is-data-descriptor@^0.1.4: version "0.1.4" resolved "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-0.1.4.tgz#0b5ee648388e2c860282e793f1856fec3f301b56" @@ -3784,6 +4323,13 @@ is-finite@^1.0.0: resolved "https://registry.npmjs.org/is-finite/-/is-finite-1.1.0.tgz#904135c77fb42c0641d6aa1bcdbc4daa8da082f3" integrity sha512-cdyMtqX/BOqqNBBiKlIVkytNHm49MtMlYyn1zxzvJKWmFMlGzm+ry5BBfYyeY9YmNKbRSo/o7OX9w9ale0wg3w== +is-fullwidth-code-point@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/is-fullwidth-code-point/-/is-fullwidth-code-point-1.0.0.tgz#ef9e31386f031a7f0d643af82fde50c457ef00cb" + integrity sha1-754xOG8DGn8NZDr4L95QxFfvAMs= + dependencies: + number-is-nan "^1.0.0" + is-fullwidth-code-point@^2.0.0: version "2.0.0" resolved "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz#a3b30a5c4f199183167aaab93beefae3ddfb654f" @@ -3801,6 +4347,11 @@ is-glob@^4.0.0, is-glob@^4.0.1, is-glob@~4.0.1: dependencies: is-extglob "^2.1.1" +is-lambda@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/is-lambda/-/is-lambda-1.0.1.tgz#3d9877899e6a53efc0160504cde15f82e6f061d5" + integrity sha1-PZh3iZ5qU+/AFgUEzeFfgubwYdU= + is-negative-zero@^2.0.1: version "2.0.1" resolved "https://registry.yarnpkg.com/is-negative-zero/-/is-negative-zero-2.0.1.tgz#3de746c18dda2319241a53675908d8f766f11c24" @@ -4112,6 +4663,11 @@ json-parse-better-errors@^1.0.1: resolved "https://registry.npmjs.org/json-parse-better-errors/-/json-parse-better-errors-1.0.2.tgz#bb867cfb3450e69107c131d1c514bab3dc8bcaa9" integrity sha512-mrqyZKfX5EhL7hvqcV6WG1yYjnjeuYDzDhhcAAUrq8Po85NBQBJP+ZDUT75qZQ98IkUoBqdkExkukOU7Ts2wrw== +json-parse-even-better-errors@^2.3.0, json-parse-even-better-errors@^2.3.1: + version "2.3.1" + resolved "https://registry.yarnpkg.com/json-parse-even-better-errors/-/json-parse-even-better-errors-2.3.1.tgz#7c47805a94319928e05777405dc12e1f7a4ee02d" + integrity sha512-xyFwyhro/JEof6Ghe2iz2NcXoj2sloNsWr/XsERDK/oiPCfaNhl5ONfp+jQdAZRQQ0IJWNzH9zIZF7li91kh2w== + json-schema-traverse@^0.4.1: version "0.4.1" resolved "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz#69f6a87d9513ab8bb8fe63bdb0979c448e684660" @@ -4139,6 +4695,11 @@ json-stable-stringify@^1.0.1: dependencies: jsonify "~0.0.0" +json-stringify-nice@^1.1.4: + version "1.1.4" + resolved "https://registry.yarnpkg.com/json-stringify-nice/-/json-stringify-nice-1.1.4.tgz#2c937962b80181d3f317dd39aa323e14f5a60a67" + integrity sha512-5Z5RFW63yxReJ7vANgW6eZFGWaQvnPE3WNmZoOJrSkGju2etKA2L5rrOa1sm877TVTFt57A80BH1bArcmlLfPw== + json-stringify-safe@~5.0.1: version "5.0.1" resolved "https://registry.npmjs.org/json-stringify-safe/-/json-stringify-safe-5.0.1.tgz#1296a2d58fd45f19a0f6ce01d65701e2c735b6eb" @@ -4179,7 +4740,7 @@ jsonify@~0.0.0: resolved "https://registry.yarnpkg.com/jsonify/-/jsonify-0.0.0.tgz#2c74b6ee41d93ca51b7b5aaee8f503631d252a73" integrity sha1-LHS27kHZPKUbe1qu6PUDYx0lKnM= -jsonparse@^1.2.0: +jsonparse@^1.2.0, jsonparse@^1.3.1: version "1.3.1" resolved "https://registry.npmjs.org/jsonparse/-/jsonparse-1.3.1.tgz#3f4dae4a91fac315f71062f8521cc239f1366280" integrity sha1-P02uSpH6wxX3EGL4UhzCOfE2YoA= @@ -4210,7 +4771,12 @@ jsprim@^1.2.2: json-schema "0.2.3" verror "1.10.0" -just-diff@^3.1.1: +just-diff-apply@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/just-diff-apply/-/just-diff-apply-3.0.0.tgz#a77348d24f0694e378b57293dceb65bdf5a91c4f" + integrity sha512-K2MLc+ZC2DVxX4V61bIKPeMUUfj1YYZ3h0myhchDXOW1cKoPZMnjIoNCqv9bF2n5Oob1PFxuR2gVJxkxz4e58w== + +just-diff@^3.0.1, just-diff@^3.1.1: version "3.1.1" resolved "https://registry.yarnpkg.com/just-diff/-/just-diff-3.1.1.tgz#d50c597c6fd4776495308c63bdee1b6839082647" integrity sha512-sdMWKjRq8qWZEjDcVA6llnUT8RDEBIfOiGpYFPYa9u+2c39JCsejktSP7mj5eRid5EIvTzIpQ2kDOCw1Nq9BjQ== @@ -4289,6 +4855,116 @@ levn@^0.4.1: prelude-ls "^1.2.1" type-check "~0.4.0" +libnpmaccess@^4.0.2: + version "4.0.3" + resolved "https://registry.yarnpkg.com/libnpmaccess/-/libnpmaccess-4.0.3.tgz#dfb0e5b0a53c315a2610d300e46b4ddeb66e7eec" + integrity sha512-sPeTSNImksm8O2b6/pf3ikv4N567ERYEpeKRPSmqlNt1dTZbvgpJIzg5vAhXHpw2ISBsELFRelk0jEahj1c6nQ== + dependencies: + aproba "^2.0.0" + minipass "^3.1.1" + npm-package-arg "^8.1.2" + npm-registry-fetch "^11.0.0" + +libnpmdiff@^2.0.4: + version "2.0.4" + resolved "https://registry.yarnpkg.com/libnpmdiff/-/libnpmdiff-2.0.4.tgz#bb1687992b1a97a8ea4a32f58ad7c7f92de53b74" + integrity sha512-q3zWePOJLHwsLEUjZw3Kyu/MJMYfl4tWCg78Vl6QGSfm4aXBUSVzMzjJ6jGiyarsT4d+1NH4B1gxfs62/+y9iQ== + dependencies: + "@npmcli/disparity-colors" "^1.0.1" + "@npmcli/installed-package-contents" "^1.0.7" + binary-extensions "^2.2.0" + diff "^5.0.0" + minimatch "^3.0.4" + npm-package-arg "^8.1.1" + pacote "^11.3.0" + tar "^6.1.0" + +libnpmexec@^2.0.1: + version "2.0.1" + resolved "https://registry.yarnpkg.com/libnpmexec/-/libnpmexec-2.0.1.tgz#729ae3e15a3ba225964ccf248117a75d311eeb73" + integrity sha512-4SqBB7eJvJWmUKNF42Q5qTOn20DRjEE4TgvEh2yneKlAiRlwlhuS9MNR45juWwmoURJlf2K43bozlVt7OZiIOw== + dependencies: + "@npmcli/arborist" "^2.3.0" + "@npmcli/ci-detect" "^1.3.0" + "@npmcli/run-script" "^1.8.4" + chalk "^4.1.0" + mkdirp-infer-owner "^2.0.0" + npm-package-arg "^8.1.2" + pacote "^11.3.1" + proc-log "^1.0.0" + read "^1.0.7" + read-package-json-fast "^2.0.2" + walk-up-path "^1.0.0" + +libnpmfund@^1.1.0: + version "1.1.0" + resolved "https://registry.yarnpkg.com/libnpmfund/-/libnpmfund-1.1.0.tgz#ee91313905b3194b900530efa339bc3f9fc4e5c4" + integrity sha512-Kfmh3pLS5/RGKG5WXEig8mjahPVOxkik6lsbH4iX0si1xxNi6eeUh/+nF1MD+2cgalsQif3O5qyr6mNz2ryJrQ== + dependencies: + "@npmcli/arborist" "^2.5.0" + +libnpmhook@^6.0.2: + version "6.0.3" + resolved "https://registry.yarnpkg.com/libnpmhook/-/libnpmhook-6.0.3.tgz#1d7f0d7e6a7932fbf7ce0881fdb0ed8bf8748a30" + integrity sha512-3fmkZJibIybzmAvxJ65PeV3NzRc0m4xmYt6scui5msocThbEp4sKFT80FhgrCERYDjlUuFahU6zFNbJDHbQ++g== + dependencies: + aproba "^2.0.0" + npm-registry-fetch "^11.0.0" + +libnpmorg@^2.0.2: + version "2.0.3" + resolved "https://registry.yarnpkg.com/libnpmorg/-/libnpmorg-2.0.3.tgz#4e605d4113dfa16792d75343824a0625c76703bc" + integrity sha512-JSGl3HFeiRFUZOUlGdiNcUZOsUqkSYrg6KMzvPZ1WVZ478i47OnKSS0vkPmX45Pai5mTKuwIqBMcGWG7O8HfdA== + dependencies: + aproba "^2.0.0" + npm-registry-fetch "^11.0.0" + +libnpmpack@^2.0.1: + version "2.0.1" + resolved "https://registry.yarnpkg.com/libnpmpack/-/libnpmpack-2.0.1.tgz#d3eac25cc8612f4e7cdeed4730eee339ba51c643" + integrity sha512-He4/jxOwlaQ7YG7sIC1+yNeXeUDQt8RLBvpI68R3RzPMZPa4/VpxhlDo8GtBOBDYoU8eq6v1wKL38sq58u4ibQ== + dependencies: + "@npmcli/run-script" "^1.8.3" + npm-package-arg "^8.1.0" + pacote "^11.2.6" + +libnpmpublish@^4.0.1: + version "4.0.2" + resolved "https://registry.yarnpkg.com/libnpmpublish/-/libnpmpublish-4.0.2.tgz#be77e8bf5956131bcb45e3caa6b96a842dec0794" + integrity sha512-+AD7A2zbVeGRCFI2aO//oUmapCwy7GHqPXFJh3qpToSRNU+tXKJ2YFUgjt04LPPAf2dlEH95s6EhIHM1J7bmOw== + dependencies: + normalize-package-data "^3.0.2" + npm-package-arg "^8.1.2" + npm-registry-fetch "^11.0.0" + semver "^7.1.3" + ssri "^8.0.1" + +libnpmsearch@^3.1.1: + version "3.1.2" + resolved "https://registry.yarnpkg.com/libnpmsearch/-/libnpmsearch-3.1.2.tgz#aee81b9e4768750d842b627a3051abc89fdc15f3" + integrity sha512-BaQHBjMNnsPYk3Bl6AiOeVuFgp72jviShNBw5aHaHNKWqZxNi38iVNoXbo6bG/Ccc/m1To8s0GtMdtn6xZ1HAw== + dependencies: + npm-registry-fetch "^11.0.0" + +libnpmteam@^2.0.3: + version "2.0.4" + resolved "https://registry.yarnpkg.com/libnpmteam/-/libnpmteam-2.0.4.tgz#9dbe2e18ae3cb97551ec07d2a2daf9944f3edc4c" + integrity sha512-FPrVJWv820FZFXaflAEVTLRWZrerCvfe7ZHSMzJ/62EBlho2KFlYKjyNEsPW3JiV7TLSXi3vo8u0gMwIkXSMTw== + dependencies: + aproba "^2.0.0" + npm-registry-fetch "^11.0.0" + +libnpmversion@^1.2.1: + version "1.2.1" + resolved "https://registry.yarnpkg.com/libnpmversion/-/libnpmversion-1.2.1.tgz#689aa7fe0159939b3cbbf323741d34976f4289e9" + integrity sha512-AA7x5CFgBFN+L4/JWobnY5t4OAHjQuPbAwUYJ7/NtHuyLut5meb+ne/aj0n7PWNiTGCJcRw/W6Zd2LoLT7EZuQ== + dependencies: + "@npmcli/git" "^2.0.7" + "@npmcli/run-script" "^1.8.4" + json-parse-even-better-errors "^2.3.1" + semver "^7.3.5" + stringify-package "^1.0.1" + lines-and-columns@^1.1.6: version "1.1.6" resolved "https://registry.npmjs.org/lines-and-columns/-/lines-and-columns-1.1.6.tgz#1c00c743b433cd0a4e80758f7b64a57440d9ff00" @@ -4580,6 +5256,28 @@ make-error@^1.1.1: resolved "https://registry.npmjs.org/make-error/-/make-error-1.3.6.tgz#2eb2e37ea9b67c4891f684a1394799af484cf7a2" integrity sha512-s8UhlNe7vPKomQhC1qFelMokr/Sc3AgNbso3n74mVPA5LTZwkB9NlXf4XPamLxJE8h0gh73rM94xvwRT2CVInw== +make-fetch-happen@^9.0.1, make-fetch-happen@^9.0.5: + version "9.0.5" + resolved "https://registry.yarnpkg.com/make-fetch-happen/-/make-fetch-happen-9.0.5.tgz#e7819afd9c8605f1452df4c1c6dc5c502ca18459" + integrity sha512-XN0i/VqHsql30Oq7179spk6vu3IuaPL1jaivNYhBrJtK7tkOuJwMK2IlROiOnJ40b9SvmOo2G86FZyI6LD2EsQ== + dependencies: + agentkeepalive "^4.1.3" + cacache "^15.2.0" + http-cache-semantics "^4.1.0" + http-proxy-agent "^4.0.1" + https-proxy-agent "^5.0.0" + is-lambda "^1.0.1" + lru-cache "^6.0.0" + minipass "^3.1.3" + minipass-collect "^1.0.2" + minipass-fetch "^1.3.2" + minipass-flush "^1.0.5" + minipass-pipeline "^1.2.4" + negotiator "^0.6.2" + promise-retry "^2.0.1" + socks-proxy-agent "^6.0.0" + ssri "^8.0.0" + map-cache@^0.2.2: version "0.2.2" resolved "https://registry.npmjs.org/map-cache/-/map-cache-0.2.2.tgz#c32abd0bd6525d9b051645bb4f26ac5dc98a0dbf" @@ -4730,6 +5428,68 @@ minimist@1.2.5, minimist@^1.1.3, minimist@^1.2.0, minimist@^1.2.3, minimist@^1.2 resolved "https://registry.yarnpkg.com/minimist/-/minimist-1.2.5.tgz#67d66014b66a6a8aaa0c083c5fd58df4e4e97602" integrity sha512-FM9nNUYrRBAELZQT3xeZQ7fmMOBg6nWNmJKTcgsJeaLstP/UODVpGsr5OhXhhXg6f+qtJ8uiZ+PUxkDWcgIXLw== +minipass-collect@^1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/minipass-collect/-/minipass-collect-1.0.2.tgz#22b813bf745dc6edba2576b940022ad6edc8c617" + integrity sha512-6T6lH0H8OG9kITm/Jm6tdooIbogG9e0tLgpY6mphXSm/A9u8Nq1ryBG+Qspiub9LjWlBPsPS3tWQ/Botq4FdxA== + dependencies: + minipass "^3.0.0" + +minipass-fetch@^1.3.0, minipass-fetch@^1.3.2: + version "1.3.4" + resolved "https://registry.yarnpkg.com/minipass-fetch/-/minipass-fetch-1.3.4.tgz#63f5af868a38746ca7b33b03393ddf8c291244fe" + integrity sha512-TielGogIzbUEtd1LsjZFs47RWuHHfhl6TiCx1InVxApBAmQ8bL0dL5ilkLGcRvuyW/A9nE+Lvn855Ewz8S0PnQ== + dependencies: + minipass "^3.1.0" + minipass-sized "^1.0.3" + minizlib "^2.0.0" + optionalDependencies: + encoding "^0.1.12" + +minipass-flush@^1.0.5: + version "1.0.5" + resolved "https://registry.yarnpkg.com/minipass-flush/-/minipass-flush-1.0.5.tgz#82e7135d7e89a50ffe64610a787953c4c4cbb373" + integrity sha512-JmQSYYpPUqX5Jyn1mXaRwOda1uQ8HP5KAT/oDSLCzt1BYRhQU0/hDtsB1ufZfEEzMZ9aAVmsBw8+FWsIXlClWw== + dependencies: + minipass "^3.0.0" + +minipass-json-stream@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/minipass-json-stream/-/minipass-json-stream-1.0.1.tgz#7edbb92588fbfc2ff1db2fc10397acb7b6b44aa7" + integrity sha512-ODqY18UZt/I8k+b7rl2AENgbWE8IDYam+undIJONvigAz8KR5GWblsFTEfQs0WODsjbSXWlm+JHEv8Gr6Tfdbg== + dependencies: + jsonparse "^1.3.1" + minipass "^3.0.0" + +minipass-pipeline@^1.2.2, minipass-pipeline@^1.2.4: + version "1.2.4" + resolved "https://registry.yarnpkg.com/minipass-pipeline/-/minipass-pipeline-1.2.4.tgz#68472f79711c084657c067c5c6ad93cddea8214c" + integrity sha512-xuIq7cIOt09RPRJ19gdi4b+RiNvDFYe5JH+ggNvBqGqpQXcru3PcRmOZuHBKWK1Txf9+cQ+HMVN4d6z46LZP7A== + dependencies: + minipass "^3.0.0" + +minipass-sized@^1.0.3: + version "1.0.3" + resolved "https://registry.yarnpkg.com/minipass-sized/-/minipass-sized-1.0.3.tgz#70ee5a7c5052070afacfbc22977ea79def353b70" + integrity sha512-MbkQQ2CTiBMlA2Dm/5cY+9SWFEN8pzzOXi6rlM5Xxq0Yqbda5ZQy9sU75a673FE9ZK0Zsbr6Y5iP6u9nktfg2g== + dependencies: + minipass "^3.0.0" + +minipass@^3.0.0, minipass@^3.1.0, minipass@^3.1.1, minipass@^3.1.3: + version "3.1.3" + resolved "https://registry.yarnpkg.com/minipass/-/minipass-3.1.3.tgz#7d42ff1f39635482e15f9cdb53184deebd5815fd" + integrity sha512-Mgd2GdMVzY+x3IJ+oHnVM+KG3lA5c8tnabyJKmHSaG2kAGpudxuOf8ToDkhumF7UzME7DecbQE9uOZhNm7PuJg== + dependencies: + yallist "^4.0.0" + +minizlib@^2.0.0, minizlib@^2.1.1: + version "2.1.2" + resolved "https://registry.yarnpkg.com/minizlib/-/minizlib-2.1.2.tgz#e90d3466ba209b932451508a11ce3d3632145931" + integrity sha512-bAxsR8BVfj60DWXHE3u30oHzfl4G7khkSuPW+qvpd7jFRHm7dLxOjUk1EHACJ/hxLY8phGJ0YhYHZo7jil7Qdg== + dependencies: + minipass "^3.0.0" + yallist "^4.0.0" + mixin-deep@^1.2.0: version "1.3.2" resolved "https://registry.npmjs.org/mixin-deep/-/mixin-deep-1.3.2.tgz#1120b43dc359a785dce65b55b82e257ccf479566" @@ -4743,7 +5503,16 @@ mkdirp-classic@^0.5.2: resolved "https://registry.npmjs.org/mkdirp-classic/-/mkdirp-classic-0.5.3.tgz#fa10c9115cc6d8865be221ba47ee9bed78601113" integrity sha512-gKLcREMhtuZRwRAfqP3RFW+TK4JqApVBtOIftVgjuABpAtpxhPGaDcfvbhNvD0B8iD1oUr/txX35NjcaY6Ns/A== -mkdirp@1.0.4: +mkdirp-infer-owner@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/mkdirp-infer-owner/-/mkdirp-infer-owner-2.0.0.tgz#55d3b368e7d89065c38f32fd38e638f0ab61d316" + integrity sha512-sdqtiFt3lkOaYvTXSRIUjkIdPTcxgv5+fgqYE/5qgwdw12cOrAuzzgzvVExIkH/ul1oeHN3bCLOWSG3XOqbKKw== + dependencies: + chownr "^2.0.0" + infer-owner "^1.0.4" + mkdirp "^1.0.3" + +mkdirp@1.0.4, mkdirp@^1.0.3, mkdirp@^1.0.4: version "1.0.4" resolved "https://registry.npmjs.org/mkdirp/-/mkdirp-1.0.4.tgz#3eb5ed62622756d79a5f0e2a221dfebad75c2f7e" integrity sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw== @@ -4837,7 +5606,7 @@ ms@2.1.2: resolved "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz#d09d1f357b443f493382a8eb3ccd183872ae6009" integrity sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w== -ms@2.1.3, ms@^2.1.1: +ms@2.1.3, ms@^2.0.0, ms@^2.1.1, ms@^2.1.2: version "2.1.3" resolved "https://registry.yarnpkg.com/ms/-/ms-2.1.3.tgz#574c8138ce1d2b5861f0b44579dbadd60c6615b2" integrity sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA== @@ -4866,7 +5635,7 @@ mute-stream@0.0.7: resolved "https://registry.npmjs.org/mute-stream/-/mute-stream-0.0.7.tgz#3075ce93bc21b8fab43e1bc4da7e8115ed1e7bab" integrity sha1-MHXOk7whuPq0PhvE2n6BFe0ee6s= -mute-stream@0.0.8: +mute-stream@0.0.8, mute-stream@~0.0.4: version "0.0.8" resolved "https://registry.npmjs.org/mute-stream/-/mute-stream-0.0.8.tgz#1630c42b2251ff81e2a283de96a5497ea92e5e0d" integrity sha512-nnbWWOkoWyUsTjKrhgD0dcz22mdkSnpYqbEjIm2nhwhuxlSkpywJmBo8h0ZqJdkp73mb90SssHkN4rsRaBAfAA== @@ -4922,6 +5691,11 @@ ncp@~2.0.0: resolved "https://registry.npmjs.org/ncp/-/ncp-2.0.0.tgz#195a21d6c46e361d2fb1281ba38b91e9df7bdbb3" integrity sha1-GVoh1sRuNh0vsSgbo4uR6d9727M= +negotiator@^0.6.2: + version "0.6.2" + resolved "https://registry.yarnpkg.com/negotiator/-/negotiator-0.6.2.tgz#feacf7ccf525a77ae9634436a64883ffeca346fb" + integrity sha512-hZXc7K2e+PgeI1eDBe/10Ard4ekbfrrqG8Ep+8Jmf4JID2bNg7NvCPOZN+kfF574pFQI7mum2AUqDidoKqcTOw== + neo-async@^2.6.0: version "2.6.2" resolved "https://registry.yarnpkg.com/neo-async/-/neo-async-2.6.2.tgz#b4aafb93e3aeb2d8174ca53cf163ab7d7308305f" @@ -4976,6 +5750,22 @@ nise@^5.1.0: just-extend "^4.0.2" path-to-regexp "^1.7.0" +node-gyp@^7.1.0, node-gyp@^7.1.2: + version "7.1.2" + resolved "https://registry.yarnpkg.com/node-gyp/-/node-gyp-7.1.2.tgz#21a810aebb187120251c3bcec979af1587b188ae" + integrity sha512-CbpcIo7C3eMu3dL1c3d0xw449fHIGALIJsRP4DDPHpyiW8vcriNY7ubh9TE4zEKfSxscY7PjeFnshE7h75ynjQ== + dependencies: + env-paths "^2.2.0" + glob "^7.1.4" + graceful-fs "^4.2.3" + nopt "^5.0.0" + npmlog "^4.1.2" + request "^2.88.2" + rimraf "^3.0.2" + semver "^7.3.2" + tar "^6.0.2" + which "^2.0.2" + node-preload@^0.2.1: version "0.2.1" resolved "https://registry.npmjs.org/node-preload/-/node-preload-0.2.1.tgz#c03043bb327f417a18fee7ab7ee57b408a144301" @@ -4983,6 +5773,13 @@ node-preload@^0.2.1: dependencies: process-on-spawn "^1.0.0" +nopt@^5.0.0: + version "5.0.0" + resolved "https://registry.yarnpkg.com/nopt/-/nopt-5.0.0.tgz#530942bb58a512fccafe53fe210f13a25355dc88" + integrity sha512-Tbj67rffqceeLpcRXrT7vKAN8CwfPeIBgM7E6iBkmKLV7bEMwpGgYLGv0jACUsECaa/vuxP0IjEont6umdMgtQ== + dependencies: + abbrev "1" + normalize-package-data@^2.3.2, normalize-package-data@^2.3.4, normalize-package-data@^2.5.0: version "2.5.0" resolved "https://registry.npmjs.org/normalize-package-data/-/normalize-package-data-2.5.0.tgz#e66db1838b200c1dfc233225d12cb36520e234a8" @@ -5003,11 +5800,95 @@ normalize-package-data@^3.0.0: semver "^7.3.2" validate-npm-package-license "^3.0.1" +normalize-package-data@^3.0.2: + version "3.0.3" + resolved "https://registry.yarnpkg.com/normalize-package-data/-/normalize-package-data-3.0.3.tgz#dbcc3e2da59509a0983422884cd172eefdfa525e" + integrity sha512-p2W1sgqij3zMMyRC067Dg16bfzVH+w7hyegmpIvZ4JNjqtGOVAIvLmjBx3yP7YTe9vKJgkoNOPjwQGogDoMXFA== + dependencies: + hosted-git-info "^4.0.1" + is-core-module "^2.5.0" + semver "^7.3.4" + validate-npm-package-license "^3.0.1" + normalize-path@^3.0.0, normalize-path@~3.0.0: version "3.0.0" resolved "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz#0dcd69ff23a1c9b11fd0978316644a0388216a65" integrity sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA== +npm-audit-report@^2.1.5: + version "2.1.5" + resolved "https://registry.yarnpkg.com/npm-audit-report/-/npm-audit-report-2.1.5.tgz#a5b8850abe2e8452fce976c8960dd432981737b5" + integrity sha512-YB8qOoEmBhUH1UJgh1xFAv7Jg1d+xoNhsDYiFQlEFThEBui0W1vIz2ZK6FVg4WZjwEdl7uBQlm1jy3MUfyHeEw== + dependencies: + chalk "^4.0.0" + +npm-bundled@^1.1.1: + version "1.1.2" + resolved "https://registry.yarnpkg.com/npm-bundled/-/npm-bundled-1.1.2.tgz#944c78789bd739035b70baa2ca5cc32b8d860bc1" + integrity sha512-x5DHup0SuyQcmL3s7Rx/YQ8sbw/Hzg0rj48eN0dV7hf5cmQq5PXIeioroH3raV1QC1yh3uTYuMThvEQF3iKgGQ== + dependencies: + npm-normalize-package-bin "^1.0.1" + +npm-install-checks@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/npm-install-checks/-/npm-install-checks-4.0.0.tgz#a37facc763a2fde0497ef2c6d0ac7c3fbe00d7b4" + integrity sha512-09OmyDkNLYwqKPOnbI8exiOZU2GVVmQp7tgez2BPi5OZC8M82elDAps7sxC4l//uSUtotWqoEIDwjRvWH4qz8w== + dependencies: + semver "^7.1.1" + +npm-normalize-package-bin@^1.0.0, npm-normalize-package-bin@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/npm-normalize-package-bin/-/npm-normalize-package-bin-1.0.1.tgz#6e79a41f23fd235c0623218228da7d9c23b8f6e2" + integrity sha512-EPfafl6JL5/rU+ot6P3gRSCpPDW5VmIzX959Ob1+ySFUuuYHWHekXpwdUZcKP5C+DS4GEtdJluwBjnsNDl+fSA== + +npm-package-arg@^8.0.0, npm-package-arg@^8.0.1, npm-package-arg@^8.1.0, npm-package-arg@^8.1.1, npm-package-arg@^8.1.2, npm-package-arg@^8.1.5: + version "8.1.5" + resolved "https://registry.yarnpkg.com/npm-package-arg/-/npm-package-arg-8.1.5.tgz#3369b2d5fe8fdc674baa7f1786514ddc15466e44" + integrity sha512-LhgZrg0n0VgvzVdSm1oiZworPbTxYHUJCgtsJW8mGvlDpxTM1vSJc3m5QZeUkhAHIzbz3VCHd/R4osi1L1Tg/Q== + dependencies: + hosted-git-info "^4.0.1" + semver "^7.3.4" + validate-npm-package-name "^3.0.0" + +npm-packlist@^2.1.4: + version "2.2.2" + resolved "https://registry.yarnpkg.com/npm-packlist/-/npm-packlist-2.2.2.tgz#076b97293fa620f632833186a7a8f65aaa6148c8" + integrity sha512-Jt01acDvJRhJGthnUJVF/w6gumWOZxO7IkpY/lsX9//zqQgnF7OJaxgQXcerd4uQOLu7W5bkb4mChL9mdfm+Zg== + dependencies: + glob "^7.1.6" + ignore-walk "^3.0.3" + npm-bundled "^1.1.1" + npm-normalize-package-bin "^1.0.1" + +npm-pick-manifest@^6.0.0, npm-pick-manifest@^6.1.0, npm-pick-manifest@^6.1.1: + version "6.1.1" + resolved "https://registry.yarnpkg.com/npm-pick-manifest/-/npm-pick-manifest-6.1.1.tgz#7b5484ca2c908565f43b7f27644f36bb816f5148" + integrity sha512-dBsdBtORT84S8V8UTad1WlUyKIY9iMsAmqxHbLdeEeBNMLQDlDWWra3wYUx9EBEIiG/YwAy0XyNHDd2goAsfuA== + dependencies: + npm-install-checks "^4.0.0" + npm-normalize-package-bin "^1.0.1" + npm-package-arg "^8.1.2" + semver "^7.3.4" + +npm-profile@^5.0.3: + version "5.0.4" + resolved "https://registry.yarnpkg.com/npm-profile/-/npm-profile-5.0.4.tgz#73e5bd1d808edc2c382d7139049cc367ac43161b" + integrity sha512-OKtU7yoAEBOnc8zJ+/uo5E4ugPp09sopo+6y1njPp+W99P8DvQon3BJYmpvyK2Bf1+3YV5LN1bvgXRoZ1LUJBA== + dependencies: + npm-registry-fetch "^11.0.0" + +npm-registry-fetch@^11.0.0: + version "11.0.0" + resolved "https://registry.yarnpkg.com/npm-registry-fetch/-/npm-registry-fetch-11.0.0.tgz#68c1bb810c46542760d62a6a965f85a702d43a76" + integrity sha512-jmlgSxoDNuhAtxUIG6pVwwtz840i994dL14FoNVZisrmZW5kWd63IUTNv1m/hyRSGSqWjCUp/YZlS1BJyNp9XA== + dependencies: + make-fetch-happen "^9.0.1" + minipass "^3.1.3" + minipass-fetch "^1.3.0" + minipass-json-stream "^1.0.1" + minizlib "^2.0.0" + npm-package-arg "^8.0.0" + npm-run-path@^2.0.0: version "2.0.2" resolved "https://registry.npmjs.org/npm-run-path/-/npm-run-path-2.0.2.tgz#35a9232dfa35d7067b4cb2ddf2357b1871536c5f" @@ -5029,6 +5910,110 @@ npm-run-path@^4.0.1: dependencies: path-key "^3.0.0" +npm-user-validate@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/npm-user-validate/-/npm-user-validate-1.0.1.tgz#31428fc5475fe8416023f178c0ab47935ad8c561" + integrity sha512-uQwcd/tY+h1jnEaze6cdX/LrhWhoBxfSknxentoqmIuStxUExxjWd3ULMLFPiFUrZKbOVMowH6Jq2FRWfmhcEw== + +npm@^7.21.0: + version "7.21.0" + resolved "https://registry.yarnpkg.com/npm/-/npm-7.21.0.tgz#12af61f27ab6ece10af2b20ffb355fb2ae227fb6" + integrity sha512-OYSQykXItCDXYGb9U8o85Snhmbe0k/nwVK6CmUNmgtOcfPevVB5ZXwA44eWOCvM+WdWYQsJAJoA7eCHKImQt8g== + dependencies: + "@npmcli/arborist" "^2.8.2" + "@npmcli/ci-detect" "^1.2.0" + "@npmcli/config" "^2.2.0" + "@npmcli/map-workspaces" "^1.0.4" + "@npmcli/package-json" "^1.0.1" + "@npmcli/run-script" "^1.8.6" + abbrev "~1.1.1" + ansicolors "~0.3.2" + ansistyles "~0.1.3" + archy "~1.0.0" + cacache "^15.2.0" + chalk "^4.1.2" + chownr "^2.0.0" + cli-columns "^3.1.2" + cli-table3 "^0.6.0" + columnify "~1.5.4" + fastest-levenshtein "^1.0.12" + glob "^7.1.7" + graceful-fs "^4.2.8" + hosted-git-info "^4.0.2" + ini "^2.0.0" + init-package-json "^2.0.4" + is-cidr "^4.0.2" + json-parse-even-better-errors "^2.3.1" + libnpmaccess "^4.0.2" + libnpmdiff "^2.0.4" + libnpmexec "^2.0.1" + libnpmfund "^1.1.0" + libnpmhook "^6.0.2" + libnpmorg "^2.0.2" + libnpmpack "^2.0.1" + libnpmpublish "^4.0.1" + libnpmsearch "^3.1.1" + libnpmteam "^2.0.3" + libnpmversion "^1.2.1" + make-fetch-happen "^9.0.5" + minipass "^3.1.3" + minipass-pipeline "^1.2.4" + mkdirp "^1.0.4" + mkdirp-infer-owner "^2.0.0" + ms "^2.1.2" + node-gyp "^7.1.2" + nopt "^5.0.0" + npm-audit-report "^2.1.5" + npm-package-arg "^8.1.5" + npm-pick-manifest "^6.1.1" + npm-profile "^5.0.3" + npm-registry-fetch "^11.0.0" + npm-user-validate "^1.0.1" + npmlog "^5.0.0" + opener "^1.5.2" + pacote "^11.3.5" + parse-conflict-json "^1.1.1" + qrcode-terminal "^0.12.0" + read "~1.0.7" + read-package-json "^4.0.0" + read-package-json-fast "^2.0.3" + readdir-scoped-modules "^1.1.0" + rimraf "^3.0.2" + semver "^7.3.5" + ssri "^8.0.1" + tar "^6.1.10" + text-table "~0.2.0" + tiny-relative-date "^1.3.0" + treeverse "^1.0.4" + validate-npm-package-name "~3.0.0" + which "^2.0.2" + write-file-atomic "^3.0.3" + +npmlog@^4.1.2: + version "4.1.2" + resolved "https://registry.yarnpkg.com/npmlog/-/npmlog-4.1.2.tgz#08a7f2a8bf734604779a9efa4ad5cc717abb954b" + integrity sha512-2uUqazuKlTaSI/dC8AzicUck7+IrEaOnN/e0jd3Xtt1KcGpwx30v50mL7oPyr/h9bL3E4aZccVwpwP+5W9Vjkg== + dependencies: + are-we-there-yet "~1.1.2" + console-control-strings "~1.1.0" + gauge "~2.7.3" + set-blocking "~2.0.0" + +npmlog@^5.0.0: + version "5.0.0" + resolved "https://registry.yarnpkg.com/npmlog/-/npmlog-5.0.0.tgz#e6a41b556e9b34cb29ea132294676c07acb30efb" + integrity sha512-ftpIiLjerL2tUg3dCqN8pOSoB90gqZlzv/gaZoxHaKjeLClrfJIEQ1Pdxi6qSzflz916Bljdy8dTWQ4J7hAFSQ== + dependencies: + are-we-there-yet "^1.1.5" + console-control-strings "^1.1.0" + gauge "^3.0.0" + set-blocking "^2.0.0" + +number-is-nan@^1.0.0: + version "1.0.1" + resolved "https://registry.yarnpkg.com/number-is-nan/-/number-is-nan-1.0.1.tgz#097b602b53422a522c1afb8790318336941a011d" + integrity sha1-CXtgK1NCKlIsGvuHkDGDNpQaAR0= + nyc@^15.1.0: version "15.1.0" resolved "https://registry.npmjs.org/nyc/-/nyc-15.1.0.tgz#1335dae12ddc87b6e249d5a1994ca4bdaea75f02" @@ -5067,7 +6052,7 @@ oauth-sign@~0.9.0: resolved "https://registry.npmjs.org/oauth-sign/-/oauth-sign-0.9.0.tgz#47a7b016baa68b5fa0ecf3dee08a85c679ac6455" integrity sha512-fexhUFFPTGV8ybAtSIGbV6gOkSv8UtRbDBnAyLQw4QPKkgNlsH2ByPGtMUqdWkos6YCRmAqViwgZrJc/mRDzZQ== -object-assign@^4.0.1: +object-assign@^4.0.1, object-assign@^4.1.0, object-assign@^4.1.1: version "4.1.1" resolved "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz#2109adc7965887cfc05cbbd442cac8bfbb360863" integrity sha1-IQmtx5ZYh8/AXLvUQsrIv7s2CGM= @@ -5180,6 +6165,11 @@ opencollective-postinstall@^2.0.2: resolved "https://registry.npmjs.org/opencollective-postinstall/-/opencollective-postinstall-2.0.3.tgz#7a0fff978f6dbfa4d006238fbac98ed4198c3259" integrity sha512-8AV/sCtuzUeTo8gQK5qDZzARrulB3egtLzFgteqB2tcT4Mw7B8Kt7JcDHmltjz6FOAHsvTevk70gZEbhM4ZS9Q== +opener@^1.5.2: + version "1.5.2" + resolved "https://registry.yarnpkg.com/opener/-/opener-1.5.2.tgz#5d37e1f35077b9dcac4301372271afdeb2a13598" + integrity sha512-ur5UIdyw5Y7yEj9wLzhqXiy6GZ3Mwx0yGI+5sMn2r0N0v3cKJvUmFH5yPP+WXh9e0xfyzyJX95D8l088DNFj7A== + opn@^5.3.0: version "5.5.0" resolved "https://registry.npmjs.org/opn/-/opn-5.5.0.tgz#fc7164fab56d235904c51c3b27da6758ca3b9bfc" @@ -5302,6 +6292,31 @@ package-hash@^4.0.0: lodash.flattendeep "^4.4.0" release-zalgo "^1.0.0" +pacote@^11.1.11, pacote@^11.2.6, pacote@^11.3.0, pacote@^11.3.1, pacote@^11.3.5: + version "11.3.5" + resolved "https://registry.yarnpkg.com/pacote/-/pacote-11.3.5.tgz#73cf1fc3772b533f575e39efa96c50be8c3dc9d2" + integrity sha512-fT375Yczn4zi+6Hkk2TBe1x1sP8FgFsEIZ2/iWaXY2r/NkhDJfxbcn5paz1+RTFCyNf+dPnaoBDJoAxXSU8Bkg== + dependencies: + "@npmcli/git" "^2.1.0" + "@npmcli/installed-package-contents" "^1.0.6" + "@npmcli/promise-spawn" "^1.2.0" + "@npmcli/run-script" "^1.8.2" + cacache "^15.0.5" + chownr "^2.0.0" + fs-minipass "^2.1.0" + infer-owner "^1.0.4" + minipass "^3.1.3" + mkdirp "^1.0.3" + npm-package-arg "^8.0.1" + npm-packlist "^2.1.4" + npm-pick-manifest "^6.0.0" + npm-registry-fetch "^11.0.0" + promise-retry "^2.0.1" + read-package-json-fast "^2.0.1" + rimraf "^3.0.2" + ssri "^8.0.1" + tar "^6.1.0" + parent-module@^1.0.0: version "1.0.1" resolved "https://registry.npmjs.org/parent-module/-/parent-module-1.0.1.tgz#691d2709e78c79fae3a156622452d00762caaaa2" @@ -5309,6 +6324,15 @@ parent-module@^1.0.0: dependencies: callsites "^3.0.0" +parse-conflict-json@^1.1.1: + version "1.1.1" + resolved "https://registry.yarnpkg.com/parse-conflict-json/-/parse-conflict-json-1.1.1.tgz#54ec175bde0f2d70abf6be79e0e042290b86701b" + integrity sha512-4gySviBiW5TRl7XHvp1agcS7SOe0KZOjC//71dzZVWJrY9hCrgtvl5v3SyIxCZ4fZF47TxD9nfzmxcx76xmbUw== + dependencies: + json-parse-even-better-errors "^2.3.0" + just-diff "^3.0.1" + just-diff-apply "^3.0.0" + parse-json@^2.2.0: version "2.2.0" resolved "https://registry.npmjs.org/parse-json/-/parse-json-2.2.0.tgz#f480f40434ef80741f8469099f8dea18f55a4dc9" @@ -5545,6 +6569,11 @@ printj@~1.1.0: resolved "https://registry.yarnpkg.com/printj/-/printj-1.1.2.tgz#d90deb2975a8b9f600fb3a1c94e3f4c53c78a222" integrity sha512-zA2SmoLaxZyArQTOPj5LXecR+RagfPSU5Kw1qP+jkWeNlrq+eJZyY2oS68SU1Z/7/myXM4lo9716laOFAVStCQ== +proc-log@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/proc-log/-/proc-log-1.0.0.tgz#0d927307401f69ed79341e83a0b2c9a13395eb77" + integrity sha512-aCk8AO51s+4JyuYGg3Q/a6gnrlDO09NpVWePtjp7xwphcoQ04x5WAfCyugcsbLooWcMJ87CLkD4+604IckEdhg== + process-nextick-args@~2.0.0: version "2.0.1" resolved "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-2.0.1.tgz#7820d9b16120cc55ca9ae7792680ae7dba6d7fe2" @@ -5562,6 +6591,29 @@ progress@^2.0.0, progress@^2.0.3: resolved "https://registry.npmjs.org/progress/-/progress-2.0.3.tgz#7e8cf8d8f5b8f239c1bc68beb4eb78567d572ef8" integrity sha512-7PiHtLll5LdnKIMw100I+8xJXR5gW2QwWYkT6iJva0bXitZKa/XMrSbdmg3r2Xnaidz9Qumd0VPaMrZlF9V9sA== +promise-all-reject-late@^1.0.0: + version "1.0.1" + resolved "https://registry.yarnpkg.com/promise-all-reject-late/-/promise-all-reject-late-1.0.1.tgz#f8ebf13483e5ca91ad809ccc2fcf25f26f8643c2" + integrity sha512-vuf0Lf0lOxyQREH7GDIOUMLS7kz+gs8i6B+Yi8dC68a2sychGrHTJYghMBD6k7eUcH0H5P73EckCA48xijWqXw== + +promise-call-limit@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/promise-call-limit/-/promise-call-limit-1.0.1.tgz#4bdee03aeb85674385ca934da7114e9bcd3c6e24" + integrity sha512-3+hgaa19jzCGLuSCbieeRsu5C2joKfYn8pY6JAuXFRVfF4IO+L7UPpFWNTeWT9pM7uhskvbPPd/oEOktCn317Q== + +promise-inflight@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/promise-inflight/-/promise-inflight-1.0.1.tgz#98472870bf228132fcbdd868129bad12c3c029e3" + integrity sha1-mEcocL8igTL8vdhoEputEsPAKeM= + +promise-retry@^2.0.1: + version "2.0.1" + resolved "https://registry.yarnpkg.com/promise-retry/-/promise-retry-2.0.1.tgz#ff747a13620ab57ba688f5fc67855410c370da22" + integrity sha512-y+WKFlBR8BGXnsNlIHFGPZmyDf3DFMoLhaflAnyZgV6rG6xu+JwesTo2Q9R6XwYmtmwAFCkAk3e35jEdoeh/3g== + dependencies: + err-code "^2.0.2" + retry "^0.12.0" + promise@^7.1.1: version "7.3.1" resolved "https://registry.npmjs.org/promise/-/promise-7.3.1.tgz#064b72602b18f90f29192b8b1bc418ffd1ebd3bf" @@ -5569,6 +6621,13 @@ promise@^7.1.1: dependencies: asap "~2.0.3" +promzard@^0.3.0: + version "0.3.0" + resolved "https://registry.yarnpkg.com/promzard/-/promzard-0.3.0.tgz#26a5d6ee8c7dee4cb12208305acfb93ba382a9ee" + integrity sha1-JqXW7ox97kyxIggwWs+5O6OCqe4= + dependencies: + read "1" + psl@^1.1.24, psl@^1.1.28, psl@^1.1.33: version "1.8.0" resolved "https://registry.npmjs.org/psl/-/psl-1.8.0.tgz#9326f8bcfb013adcc005fdff056acce020e51c24" @@ -5616,6 +6675,11 @@ qqjs@^0.3.10: tmp "^0.1.0" write-json-file "^4.1.1" +qrcode-terminal@^0.12.0: + version "0.12.0" + resolved "https://registry.yarnpkg.com/qrcode-terminal/-/qrcode-terminal-0.12.0.tgz#bb5b699ef7f9f0505092a3748be4464fe71b5819" + integrity sha512-EXtzRZmC+YGmGlDFbXKxQiMZNwCLEO6BANKXG4iCtSIM0yqc/pappSx3RIKr4r0uh5JsBckOXeKrB3Iz7mdQpQ== + qs@~6.5.2: version "6.5.2" resolved "https://registry.npmjs.org/qs/-/qs-6.5.2.tgz#cb3ae806e8740444584ef154ce8ee98d403f3e36" @@ -5633,6 +6697,29 @@ randombytes@^2.1.0: dependencies: safe-buffer "^5.1.0" +read-cmd-shim@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/read-cmd-shim/-/read-cmd-shim-2.0.0.tgz#4a50a71d6f0965364938e9038476f7eede3928d9" + integrity sha512-HJpV9bQpkl6KwjxlJcBoqu9Ba0PQg8TqSNIOrulGt54a0uup0HtevreFHzYzkm0lpnleRdNBzXznKrgxglEHQw== + +read-package-json-fast@^2.0.1, read-package-json-fast@^2.0.2, read-package-json-fast@^2.0.3: + version "2.0.3" + resolved "https://registry.yarnpkg.com/read-package-json-fast/-/read-package-json-fast-2.0.3.tgz#323ca529630da82cb34b36cc0b996693c98c2b83" + integrity sha512-W/BKtbL+dUjTuRL2vziuYhp76s5HZ9qQhd/dKfWIZveD0O40453QNyZhC0e63lqZrAQ4jiOapVoeJ7JrszenQQ== + dependencies: + json-parse-even-better-errors "^2.3.0" + npm-normalize-package-bin "^1.0.1" + +read-package-json@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/read-package-json/-/read-package-json-4.0.0.tgz#b555a9f749bf5eb9b8f053806b32f17001914e90" + integrity sha512-EBQiek1udd0JKvUzaViAWHYVQRuQZ0IP0LWUOqVCJaZIX92ZO86dOpvsTOO3esRIQGgl7JhFBaGqW41VI57KvQ== + dependencies: + glob "^7.1.1" + json-parse-even-better-errors "^2.3.0" + normalize-package-data "^3.0.0" + npm-normalize-package-bin "^1.0.0" + read-pkg-up@^1.0.1: version "1.0.1" resolved "https://registry.npmjs.org/read-pkg-up/-/read-pkg-up-1.0.1.tgz#9d63c13276c065918d57f002a57f40a1b643fb02" @@ -5703,6 +6790,13 @@ read-pkg@^5.2.0: parse-json "^5.0.0" type-fest "^0.6.0" +read@1, read@^1.0.7, read@~1.0.1, read@~1.0.7: + version "1.0.7" + resolved "https://registry.yarnpkg.com/read/-/read-1.0.7.tgz#b3da19bd052431a97671d44a42634adf710b40c4" + integrity sha1-s9oZvQUkMal2cdRKQmNK33ELQMQ= + dependencies: + mute-stream "~0.0.4" + "readable-stream@2 || 3", readable-stream@^3.1.1, readable-stream@^3.4.0, readable-stream@^3.6.0: version "3.6.0" resolved "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.0.tgz#337bbda3adc0706bd3e024426a286d4b4b2c9198" @@ -5712,7 +6806,7 @@ read-pkg@^5.2.0: string_decoder "^1.1.1" util-deprecate "^1.0.1" -readable-stream@^2.0.0, readable-stream@^2.0.5, readable-stream@^2.1.0, readable-stream@~2.3.6: +readable-stream@^2.0.0, readable-stream@^2.0.5, readable-stream@^2.0.6, readable-stream@^2.1.0, readable-stream@~2.3.6: version "2.3.7" resolved "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.7.tgz#1eca1cf711aef814c04f62252a36a62f6cb23b57" integrity sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw== @@ -5732,6 +6826,16 @@ readdir-glob@^1.0.0: dependencies: minimatch "^3.0.4" +readdir-scoped-modules@^1.1.0: + version "1.1.0" + resolved "https://registry.yarnpkg.com/readdir-scoped-modules/-/readdir-scoped-modules-1.1.0.tgz#8d45407b4f870a0dcaebc0e28670d18e74514309" + integrity sha512-asaikDeqAQg7JifRsZn1NJZXo9E+VwlyCfbkZhwyISinqk5zNS6266HS5kah6P0SaQKGF6SkNnZVHUzHFYxYDw== + dependencies: + debuglog "^1.0.1" + dezalgo "^1.0.0" + graceful-fs "^4.1.2" + once "^1.3.0" + readdirp@~3.5.0: version "3.5.0" resolved "https://registry.yarnpkg.com/readdirp/-/readdirp-3.5.0.tgz#9ba74c019b15d365278d2e91bb8c48d7b4d42c9e" @@ -5826,7 +6930,7 @@ repeating@^2.0.0: dependencies: is-finite "^1.0.0" -request@^2.72.0: +request@^2.72.0, request@^2.88.2: version "2.88.2" resolved "https://registry.npmjs.org/request/-/request-2.88.2.tgz#d73c918731cb5a87da047e207234146f664d12b3" integrity sha512-MsvtOrfG9ZcrOwAW+Qi+F6HbD0CWXEh9ou77uOb7FM2WPhwT7smM833PzanhJLsgXjN89Ir6V2PczXNnMpwKhw== @@ -5933,6 +7037,11 @@ ret@~0.1.10: resolved "https://registry.npmjs.org/ret/-/ret-0.1.15.tgz#b8a4825d5bdb1fc3f6f53c2bc33f81388681c7bc" integrity sha512-TTlYpa+OL+vMMNG24xSlQGEJ3B/RzEfUlLct7b5G/ytav+wPrplCpVMFuwzXbkecJrb6IYo1iFb0S9v37754mg== +retry@^0.12.0: + version "0.12.0" + resolved "https://registry.yarnpkg.com/retry/-/retry-0.12.0.tgz#1b42a6266a21f07421d1b0b54b7dc167b01c013b" + integrity sha1-G0KmJmoh8HQh0bC1S33BZ7AcATs= + reusify@^1.0.4: version "1.0.4" resolved "https://registry.npmjs.org/reusify/-/reusify-1.0.4.tgz#90da382b1e126efc02146e90845a88db12925d76" @@ -6012,7 +7121,7 @@ safe-regex@^1.1.0: dependencies: ret "~0.1.10" -"safer-buffer@>= 2.1.2 < 3", safer-buffer@^2.0.2, safer-buffer@^2.1.0, safer-buffer@~2.1.0: +"safer-buffer@>= 2.1.2 < 3", "safer-buffer@>= 2.1.2 < 3.0.0", safer-buffer@^2.0.2, safer-buffer@^2.1.0, safer-buffer@~2.1.0: version "2.1.2" resolved "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz#44fa161b0187b9549dd84bb91802f9bd8385cd6a" integrity sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg== @@ -6052,7 +7161,7 @@ semver@^6.0.0, semver@^6.1.2, semver@^6.3.0: resolved "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz#ee0a64c8af5e8ceea67687b133761e1becbd1d3d" integrity sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw== -semver@^7.1.1, semver@^7.2.1, semver@^7.3.2, semver@^7.3.5: +semver@^7.1.1, semver@^7.1.3, semver@^7.2.1, semver@^7.3.2, semver@^7.3.4, semver@^7.3.5: version "7.3.5" resolved "https://registry.yarnpkg.com/semver/-/semver-7.3.5.tgz#0b621c879348d8998e4b0e4be94b3f12e6018ef7" integrity sha512-PoeGJYh8HK4BTO/a9Tf6ZG3veo/A7ZVsYrSA6J8ny9nb3B1VrpkuN+z9OE5wfE5p6H4LchYZsegiQgbJD94ZFQ== @@ -6071,7 +7180,7 @@ serialize-javascript@5.0.1: dependencies: randombytes "^2.1.0" -set-blocking@^2.0.0: +set-blocking@^2.0.0, set-blocking@~2.0.0: version "2.0.0" resolved "https://registry.npmjs.org/set-blocking/-/set-blocking-2.0.0.tgz#045f9782d011ae9a6803ddd382b24392b3d890f7" integrity sha1-BF+XgtARrppoA93TgrJDkrPYkPc= @@ -6224,6 +7333,11 @@ slice-ansi@^4.0.0: astral-regex "^2.0.0" is-fullwidth-code-point "^3.0.0" +smart-buffer@^4.1.0: + version "4.2.0" + resolved "https://registry.yarnpkg.com/smart-buffer/-/smart-buffer-4.2.0.tgz#6e1d71fa4f18c05f7d0ff216dd16a481d0e8d9ae" + integrity sha512-94hK0Hh8rPqQl2xXc3HsaBoOXKV20MToPkcXvwbISWLEs+64sBq5kFgn2kJDHb1Pry9yrP0dxrCI9RRci7RXKg== + snapdragon-node@^2.0.1: version "2.1.1" resolved "https://registry.npmjs.org/snapdragon-node/-/snapdragon-node-2.1.1.tgz#6c175f86ff14bdb0724563e8f3c1b021a286853b" @@ -6254,6 +7368,23 @@ snapdragon@^0.8.1: source-map-resolve "^0.5.0" use "^3.1.0" +socks-proxy-agent@^6.0.0: + version "6.0.0" + resolved "https://registry.yarnpkg.com/socks-proxy-agent/-/socks-proxy-agent-6.0.0.tgz#9f8749cdc05976505fa9f9a958b1818d0e60573b" + integrity sha512-FIgZbQWlnjVEQvMkylz64/rUggGtrKstPnx8OZyYFG0tAFR8CSBtpXxSwbFLHyeXFn/cunFL7MpuSOvDSOPo9g== + dependencies: + agent-base "^6.0.2" + debug "^4.3.1" + socks "^2.6.1" + +socks@^2.6.1: + version "2.6.1" + resolved "https://registry.yarnpkg.com/socks/-/socks-2.6.1.tgz#989e6534a07cf337deb1b1c94aaa44296520d30e" + integrity sha512-kLQ9N5ucj8uIcxrDwjm0Jsqk06xdpBjGNQtpXy4Q8/QY2k+fY7nZH8CARy+hkbG+SGAovmzzuauCpBlb8FrnBA== + dependencies: + ip "^1.1.5" + smart-buffer "^4.1.0" + sort-keys@^4.0.0: version "4.0.0" resolved "https://registry.npmjs.org/sort-keys/-/sort-keys-4.0.0.tgz#56dc5e256637bfe3fec8db0dc57c08b1a2be22d6" @@ -6367,6 +7498,13 @@ sshpk@^1.7.0: safer-buffer "^2.0.2" tweetnacl "~0.14.0" +ssri@^8.0.0, ssri@^8.0.1: + version "8.0.1" + resolved "https://registry.yarnpkg.com/ssri/-/ssri-8.0.1.tgz#638e4e439e2ffbd2cd289776d5ca457c4f51a2af" + integrity sha512-97qShzy1AiyxvPNIkLWoGua7xoQzzPjQ0HAH4B0rWKo7SZ6USuPcrUiAFrws0UH8RrbWmgq3LMTObhPIHbbBeQ== + dependencies: + minipass "^3.1.1" + static-extend@^0.1.1: version "0.1.2" resolved "https://registry.npmjs.org/static-extend/-/static-extend-0.1.2.tgz#60809c39cbff55337226fd5e0b520f341f1fb5c6" @@ -6388,7 +7526,16 @@ string-argv@0.3.1: resolved "https://registry.npmjs.org/string-argv/-/string-argv-0.3.1.tgz#95e2fbec0427ae19184935f816d74aaa4c5c19da" integrity sha512-a1uQGz7IyVy9YwhqjZIZu1c8JO8dNIe20xBmSS6qu9kv++k3JGzCVmprbNN5Kn+BgzD5E7YYwg1CcjuJMRNsvg== -"string-width@^1.0.2 || 2", string-width@^2.1.0, string-width@^2.1.1: +string-width@^1.0.1: + version "1.0.2" + resolved "https://registry.yarnpkg.com/string-width/-/string-width-1.0.2.tgz#118bdf5b8cdc51a2a7e70d211e07e2b0b9b107d3" + integrity sha1-EYvfW4zcUaKn5w0hHgfisLmxB9M= + dependencies: + code-point-at "^1.0.0" + is-fullwidth-code-point "^1.0.0" + strip-ansi "^3.0.0" + +"string-width@^1.0.1 || ^2.0.0", "string-width@^1.0.2 || 2", string-width@^2.0.0, string-width@^2.1.0, string-width@^2.1.1: version "2.1.1" resolved "https://registry.npmjs.org/string-width/-/string-width-2.1.1.tgz#ab93f27a8dc13d28cac815c462143a6d9012ae9e" integrity sha512-nOqH59deCq9SRHlxq1Aw85Jnt4w6KvLKqWVik6oA9ZklXLNIOlqg4F2yrT1MVaTjAqvVwdfeZ7w7aCvJD7ugkw== @@ -6469,6 +7616,11 @@ stringify-object@^3.3.0: is-obj "^1.0.1" is-regexp "^1.0.0" +stringify-package@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/stringify-package/-/stringify-package-1.0.1.tgz#e5aa3643e7f74d0f28628b72f3dad5cecfc3ba85" + integrity sha512-sa4DUQsYciMP1xhKWGuFM04fB0LG/9DlluZoSVywUMRNvzid6XucHK0/90xGxRoHrAaROrcHK1aPKaijCtSrhg== + strip-ansi@6.0.0, strip-ansi@^6.0.0: version "6.0.0" resolved "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.0.tgz#0b1571dd7669ccd4f3e06e14ef1eed26225ae532" @@ -6476,7 +7628,14 @@ strip-ansi@6.0.0, strip-ansi@^6.0.0: dependencies: ansi-regex "^5.0.0" -strip-ansi@^4.0.0: +strip-ansi@^3.0.0, strip-ansi@^3.0.1: + version "3.0.1" + resolved "https://registry.yarnpkg.com/strip-ansi/-/strip-ansi-3.0.1.tgz#6a385fb8853d952d5ff05d0e8aaf94278dc63dcf" + integrity sha1-ajhfuIU9lS1f8F0Oiq+UJ43GPc8= + dependencies: + ansi-regex "^2.0.0" + +"strip-ansi@^3.0.1 || ^4.0.0", strip-ansi@^4.0.0: version "4.0.0" resolved "https://registry.npmjs.org/strip-ansi/-/strip-ansi-4.0.0.tgz#a8479022eb1ac368a871389b635262c505ee368f" integrity sha1-qEeQIusaw2iocTibY1JixQXuNo8= @@ -6632,6 +7791,18 @@ tar-stream@^2.1.4: inherits "^2.0.3" readable-stream "^3.1.1" +tar@^6.0.2, tar@^6.1.0, tar@^6.1.10: + version "6.1.10" + resolved "https://registry.yarnpkg.com/tar/-/tar-6.1.10.tgz#8a320a74475fba54398fa136cd9883aa8ad11175" + integrity sha512-kvvfiVvjGMxeUNB6MyYv5z7vhfFRwbwCXJAeL0/lnbrttBVqcMOnpHUf0X42LrPMR8mMpgapkJMchFH4FSHzNA== + dependencies: + chownr "^2.0.0" + fs-minipass "^2.0.0" + minipass "^3.0.0" + minizlib "^2.1.1" + mkdirp "^1.0.3" + yallist "^4.0.0" + test-exclude@^6.0.0: version "6.0.0" resolved "https://registry.npmjs.org/test-exclude/-/test-exclude-6.0.0.tgz#04a8698661d805ea6fa293b6cb9e63ac044ef15e" @@ -6646,9 +7817,9 @@ text-extensions@^1.0.0: resolved "https://registry.npmjs.org/text-extensions/-/text-extensions-1.9.0.tgz#1853e45fee39c945ce6f6c36b2d659b5aabc2a26" integrity sha512-wiBrwC1EhBelW12Zy26JeOUkQ5mRu+5o8rpsJk5+2t+Y5vE7e842qtZDQ2g1NpX/29HdyFeJ4nSIhI47ENSxlQ== -text-table@^0.2.0: +text-table@^0.2.0, text-table@~0.2.0: version "0.2.0" - resolved "https://registry.npmjs.org/text-table/-/text-table-0.2.0.tgz#7f5ee823ae805207c00af2df4a84ec3fcfa570b4" + resolved "https://registry.yarnpkg.com/text-table/-/text-table-0.2.0.tgz#7f5ee823ae805207c00af2df4a84ec3fcfa570b4" integrity sha1-f17oI66AUgfACvLfSoTsP8+lcLQ= through2@^2.0.2: @@ -6672,6 +7843,11 @@ through2@^3.0.0: resolved "https://registry.npmjs.org/through/-/through-2.3.8.tgz#0dd4c9ffaabc357960b1b724115d7e0e86a2e1f5" integrity sha1-DdTJ/6q8NXlgsbckEV1+Doai4fU= +tiny-relative-date@^1.3.0: + version "1.3.0" + resolved "https://registry.yarnpkg.com/tiny-relative-date/-/tiny-relative-date-1.3.0.tgz#fa08aad501ed730f31cc043181d995c39a935e07" + integrity sha512-MOQHpzllWxDCHHaDno30hhLfbouoYlOI8YlMNtvKe1zXbjEVhbcEovQxvZrPvtiYW630GQDoMMarCnjfyfHA+A== + tmp@^0.0.33: version "0.0.33" resolved "https://registry.npmjs.org/tmp/-/tmp-0.0.33.tgz#6d34335889768d21b2bcda0aa277ced3b1bfadf9" @@ -6753,6 +7929,11 @@ treeify@^1.1.0: resolved "https://registry.npmjs.org/treeify/-/treeify-1.1.0.tgz#4e31c6a463accd0943879f30667c4fdaff411bb8" integrity sha512-1m4RA7xVAJrSGrrXGs0L3YTwyvBs2S8PbRHaLZAkFw7JR8oIFwYtysxlBZhYIa7xSyiYJKZ3iGrrk55cGA3i9A== +treeverse@^1.0.4: + version "1.0.4" + resolved "https://registry.yarnpkg.com/treeverse/-/treeverse-1.0.4.tgz#a6b0ebf98a1bca6846ddc7ecbc900df08cb9cd5f" + integrity sha512-whw60l7r+8ZU8Tu/Uc2yxtc4ZTZbR/PF3u1IPNKGQ6p8EICLb3Z2lAgoqw9bqYd8IkgnsaOcLzYHFckjqNsf0g== + trim-newlines@^1.0.0: version "1.0.0" resolved "https://registry.npmjs.org/trim-newlines/-/trim-newlines-1.0.0.tgz#5887966bb582a4503a41eb524f7d35011815a613" @@ -6952,6 +8133,20 @@ union-value@^1.0.0: is-extendable "^0.1.1" set-value "^2.0.1" +unique-filename@^1.1.1: + version "1.1.1" + resolved "https://registry.yarnpkg.com/unique-filename/-/unique-filename-1.1.1.tgz#1d69769369ada0583103a1e6ae87681b56573230" + integrity sha512-Vmp0jIp2ln35UTXuryvjzkjGdRyf9b2lTXuSYUiPmzRcl3FDtYqAwOnTJkAngD9SWhnoJzDbTKwaOrZ+STtxNQ== + dependencies: + unique-slug "^2.0.0" + +unique-slug@^2.0.0: + version "2.0.2" + resolved "https://registry.yarnpkg.com/unique-slug/-/unique-slug-2.0.2.tgz#baabce91083fc64e945b0f3ad613e264f7cd4e6c" + integrity sha512-zoWr9ObaxALD3DOPfjPSqxt4fnZiWblxHIgeWqW8x7UqDzEtHEQLzji2cuJYQFCU6KmoJikOYAZlrTHHebjx2w== + dependencies: + imurmurhash "^0.1.4" + universalify@^0.1.0, universalify@^0.1.2: version "0.1.2" resolved "https://registry.npmjs.org/universalify/-/universalify-0.1.2.tgz#b646f69be3942dabcecc9d6639c80dc105efaa66" @@ -7002,7 +8197,7 @@ v8-compile-cache@^2.0.3: resolved "https://registry.npmjs.org/v8-compile-cache/-/v8-compile-cache-2.1.1.tgz#54bc3cdd43317bca91e35dcaf305b1a7237de745" integrity sha512-8OQ9CL+VWyt3JStj7HX7/ciTL2V3Rl1Wf5OL+SNTm0yK1KvtReVulksyeRnCANHHuUxHlQig+JJDlUhBt1NQDQ== -validate-npm-package-license@^3.0.1: +validate-npm-package-license@^3.0.1, validate-npm-package-license@^3.0.4: version "3.0.4" resolved "https://registry.npmjs.org/validate-npm-package-license/-/validate-npm-package-license-3.0.4.tgz#fc91f6b9c7ba15c857f4cb2c5defeec39d4f410a" integrity sha512-DpKm2Ui/xN7/HQKCtpZxoRWBhZ9Z0kqtygG8XCgNQ8ZlDnxuQmWhj566j8fN4Cu3/JmbhsDo7fcAJq4s9h27Ew== @@ -7010,6 +8205,13 @@ validate-npm-package-license@^3.0.1: spdx-correct "^3.0.0" spdx-expression-parse "^3.0.0" +validate-npm-package-name@^3.0.0, validate-npm-package-name@~3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/validate-npm-package-name/-/validate-npm-package-name-3.0.0.tgz#5fa912d81eb7d0c74afc140de7317f0ca7df437e" + integrity sha1-X6kS2B630MdK/BQN5zF/DKffQ34= + dependencies: + builtins "^1.0.3" + verror@1.10.0: version "1.10.0" resolved "https://registry.npmjs.org/verror/-/verror-1.10.0.tgz#3a105ca17053af55d6e270c1f8288682e18da400" @@ -7019,6 +8221,18 @@ verror@1.10.0: core-util-is "1.0.2" extsprintf "^1.2.0" +walk-up-path@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/walk-up-path/-/walk-up-path-1.0.0.tgz#d4745e893dd5fd0dbb58dd0a4c6a33d9c9fec53e" + integrity sha512-hwj/qMDUEjCU5h0xr90KGCf0tg0/LgJbmOWgrWKYlcJZM7XvquvUJZ0G/HMGr7F7OQMOUuPHWP9JpriinkAlkg== + +wcwidth@^1.0.0: + version "1.0.1" + resolved "https://registry.yarnpkg.com/wcwidth/-/wcwidth-1.0.1.tgz#f0b0dcf915bc5ff1528afadb2c0e17b532da2fe8" + integrity sha1-8LDc+RW8X/FSivrbLA4XtTLaL+g= + dependencies: + defaults "^1.0.3" + websocket-driver@>=0.5.1: version "0.7.4" resolved "https://registry.npmjs.org/websocket-driver/-/websocket-driver-0.7.4.tgz#89ad5295bbf64b480abcba31e4953aca706f5760" @@ -7054,7 +8268,7 @@ which-pm-runs@^1.0.0: resolved "https://registry.npmjs.org/which-pm-runs/-/which-pm-runs-1.0.0.tgz#670b3afbc552e0b55df6b7780ca74615f23ad1cb" integrity sha1-Zws6+8VS4LVd9rd4DKdGFfI60cs= -which@2.0.2, which@^2.0.1: +which@2.0.2, which@^2.0.1, which@^2.0.2: version "2.0.2" resolved "https://registry.npmjs.org/which/-/which-2.0.2.tgz#7c6a8dd0a636a0327e10b59c9286eee93f3f51b1" integrity sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA== @@ -7068,7 +8282,7 @@ which@^1.2.14, which@^1.2.9: dependencies: isexe "^2.0.0" -wide-align@1.1.3: +wide-align@1.1.3, wide-align@^1.1.0, wide-align@^1.1.2: version "1.1.3" resolved "https://registry.npmjs.org/wide-align/-/wide-align-1.1.3.tgz#ae074e6bdc0c14a431e804e624549c633b000457" integrity sha512-QGkOQc8XL6Bt5PwnsExKBPuMKBxnGxWWW3fU55Xt4feHozMUhdUMaBCk290qpm/wG5u/RSKzwdAC4i51YigihA== @@ -7136,7 +8350,7 @@ wrappy@1: resolved "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz#b5243d8f3ec1aa35f1364605bc0d1036e30ab69f" integrity sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8= -write-file-atomic@^3.0.0: +write-file-atomic@^3.0.0, write-file-atomic@^3.0.3: version "3.0.3" resolved "https://registry.npmjs.org/write-file-atomic/-/write-file-atomic-3.0.3.tgz#56bd5c5a5c70481cd19c571bd39ab965a5de56e8" integrity sha512-AvHcyZ5JnSfq3ioSyjrBkH9yW4m7Ayk8/9My/DD9onKeu/94fwrMocemO2QAJFAlnnDN+ZDS+ZjAR5ua1/PV/Q== From aeccd87820b2b06704a70059dfe20b2d9893ca15 Mon Sep 17 00:00:00 2001 From: Benjamin Maggi Date: Fri, 20 Aug 2021 09:34:14 -0300 Subject: [PATCH 04/15] fix: use node from env --- package.json | 1 + src/lib/npmCommand.ts | 4 +++- 2 files changed, 4 insertions(+), 1 deletion(-) diff --git a/package.json b/package.json index 85bd53cb..cd3488e9 100644 --- a/package.json +++ b/package.json @@ -9,6 +9,7 @@ "@salesforce/command": "^3.0.5", "@salesforce/core": "^2.23.2", "npm": "^7.21.0", + "npm-run-path": "^4.0.1", "shelljs": "^0.8.4", "tslib": "^2" }, diff --git a/src/lib/npmCommand.ts b/src/lib/npmCommand.ts index d66fd3d1..f259a569 100644 --- a/src/lib/npmCommand.ts +++ b/src/lib/npmCommand.ts @@ -8,6 +8,7 @@ import * as path from 'path'; import * as shelljs from 'shelljs'; +import npmRunPath from 'npm-run-path'; import { SfdxError, fs } from '@salesforce/core'; type DistTags = { @@ -51,12 +52,13 @@ export class NpmCommand { public static runNpmCmd(cmd: string, options = {} as NpmCommandOptions): NpmCommandResult { const npmCli = NpmCommand.npmCli(); - const exec = `/bin/bash -c "node ${npmCli} ${cmd} --registry=${options.registry} --json"`; + const exec = `${npmCli} ${cmd} --registry=${options.registry} --json`; const npmShowResult = shelljs.exec(exec, { ...options, silent: true, fatal: true, async: false, + env: npmRunPath.env({ env: process.env }), }); if (npmShowResult.code !== 0) { throw new SfdxError(npmShowResult.stderr, 'ShellExecError'); From 5e68d8e66c19b8b2cc41e0a9bfccc1b337285019 Mon Sep 17 00:00:00 2001 From: Benjamin Maggi Date: Mon, 30 Aug 2021 09:08:18 -0300 Subject: [PATCH 05/15] chore: test if basic nuts run without timeout --- test/nuts/plugin-install.nut.ts | 123 ++++++++++++++++++++++++++++++++ 1 file changed, 123 insertions(+) create mode 100644 test/nuts/plugin-install.nut.ts diff --git a/test/nuts/plugin-install.nut.ts b/test/nuts/plugin-install.nut.ts new file mode 100644 index 00000000..7685fe25 --- /dev/null +++ b/test/nuts/plugin-install.nut.ts @@ -0,0 +1,123 @@ +/* + * Copyright (c) 2020, salesforce.com, inc. + * All rights reserved. + * Licensed under the BSD 3-Clause license. + * For full license text, see LICENSE.txt file in the repo root or https://opensource.org/licenses/BSD-3-Clause + */ + +import * as path from 'path'; +// import { /* existsSync, */ promises as fs } from 'fs'; +// import { copy, remove } from 'fs-extra'; +import { expect } from 'chai'; +// import { expect } from '@salesforce/command/lib/test'; +import { TestSession, execCmd } from '@salesforce/cli-plugins-testkit'; +import { fs } from '@salesforce/core'; + +const SIGNED_MODULE_NAME = '@salesforce/plugin-user'; +const UNSIGNED_MODULE_NAME = 'sfdx-jayree'; +// const PLUGIN_TRUST_REPO = 'git@github.com:salesforcecli/plugin-trust.git'; +let session: TestSession; + +describe.skip('plugins:install commands', () => { + before(async () => { + session = await TestSession.create(); + // await Promise.all([ + // fs.mkdir(path.join(session.homeDir, 'SC_OUTPUT')), + // fs.mkdir(path.join(session.homeDir, 'MC_OUTPUT')), + // ]); + }); + + after(async () => { + await session?.zip(undefined, 'artifacts'); + await session?.clean(); + }); + + it.skip('plugins:install signed plugin', () => { + process.env.TESTKIT_EXECUTABLE_PATH = 'sfdx'; + const result = execCmd(`plugins:install ${SIGNED_MODULE_NAME}`, { + ensureExitCode: 0, + }); + // eslint-disable-next-line no-console + console.log('result', result); + expect(result.shellOutput.stdout).to.contain(`Successfully validated digital signature for ${SIGNED_MODULE_NAME}`); + }); + + it('plugins:install unsigned plugin', () => { + process.env.TESTKIT_EXECUTABLE_PATH = 'sfdx'; + const result = execCmd(`plugins:install ${UNSIGNED_MODULE_NAME}`, { + ensureExitCode: 0, + }); + // eslint-disable-next-line no-console + console.log('result', result); + expect(result.shellOutput.stdout).to.contain( + 'This plugin is not digitally signed and its authenticity cannot be verified. Continue installation y/n?:' + ); + }); + + // it('plugins:trust:verify', () => { + // execCmd('plugins:trust:verify --npm @salesforce/plugin-user', { + // ensureExitCode: 0, + // }); + // }); + + // it('plugins:trust:verify', () => { + // execCmd('plugins:trust:verify --npm sfdx-jayree', { + // ensureExitCode: 0, + // }); + // }); +}); + +describe('plugins:install commands', () => { + before(async () => { + session = await TestSession.create(); + const configDir = path.join(session.homeDir, '.config', 'sfdx'); + fs.mkdirSync(configDir, { recursive: true }); + fs.writeJsonSync(path.join(configDir, 'unsignedPluginAllowList.json'), [UNSIGNED_MODULE_NAME]); + // process.env.TESTKIT_EXECUTABLE_PATH = 'sfdx'; + // const pluginTrustRealPath = path.dirname(session.dir).split(path.sep).pop(); + const pluginTrustRealPath = path.dirname(session.dir); + // eslint-disable-next-line no-console + console.log('pluginTrustRealPath', pluginTrustRealPath); + const pluginTrustPath = path.join(session.dir, 'plugin-trust'); + + // fs.mkdirSync(pluginTrustPath, { recursive: true }); + // fs.copyFileSync(pluginTrustRealPath, pluginTrustPath); + + // eslint-disable-next-line no-console + console.log('session.dir', session.dir); + // process.env.TESTKIT_EXECUTABLE_PATH = 'git'; + // execCmd(`clone ${PLUGIN_TRUST_REPO}`, { + // cwd: session.dir, + // ensureExitCode: 0, + // }); + // eslint-disable-next-line no-console + console.log('pluginTrustPath', pluginTrustPath); + // process.env.TESTKIT_EXECUTABLE_PATH = 'yarn'; + // execCmd('install', { + // cwd: pluginTrustPath, + // ensureExitCode: 0, + // }); + // eslint-disable-next-line no-console + console.log('plugins:link', pluginTrustPath); + process.env.TESTKIT_EXECUTABLE_PATH = 'sfdx'; + execCmd('plugins:link . --dev-debug', { + cwd: pluginTrustRealPath, + ensureExitCode: 0, + }); + }); + + after(async () => { + await session?.zip(undefined, 'artifacts'); + await session?.clean(); + }); + + it('plugins:install unsigned plugin', () => { + process.env.TESTKIT_EXECUTABLE_PATH = 'sfdx'; + const result = execCmd(`plugins:install ${UNSIGNED_MODULE_NAME}`, { + ensureExitCode: 0, + }); + // eslint-disable-next-line no-console + console.log('result', result); + // expect(result.shellOutput.stdout).to.contain(); + }); +}); From cfe98fae043e1ce677ffc35e6efce71141fb0698 Mon Sep 17 00:00:00 2001 From: Benjamin Maggi Date: Mon, 30 Aug 2021 10:38:02 -0300 Subject: [PATCH 06/15] chore: test with new orb --- .circleci/config.yml | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/.circleci/config.yml b/.circleci/config.yml index 5633c499..ed214118 100644 --- a/.circleci/config.yml +++ b/.circleci/config.yml @@ -1,6 +1,6 @@ version: 2.1 orbs: - release-management: salesforce/npm-release-management@4 + release-management: salesforce/npm-release-management@dev:ecc0179 parameters: run-auto-workflows: description: > @@ -29,7 +29,7 @@ parameters: default: '' type: string repo_tag: - description: 'The tag of the module repo to checkout, '''' defaults to branch/PR' + description: "The tag of the module repo to checkout, '' defaults to branch/PR" default: '' type: string npm_module_name: From adebfa1114db24c2c18a40c1790f627ad3be4496 Mon Sep 17 00:00:00 2001 From: Benjamin Maggi Date: Mon, 30 Aug 2021 10:51:38 -0300 Subject: [PATCH 07/15] chore: test with new orb --- .circleci/config.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.circleci/config.yml b/.circleci/config.yml index ed214118..7cd4ba19 100644 --- a/.circleci/config.yml +++ b/.circleci/config.yml @@ -1,6 +1,6 @@ version: 2.1 orbs: - release-management: salesforce/npm-release-management@dev:ecc0179 + release-management: salesforce/npm-release-management@dev:2cb4d86 parameters: run-auto-workflows: description: > From 239d273e25002dace67a0246747a20ce6ce3a443 Mon Sep 17 00:00:00 2001 From: Benjamin Maggi Date: Mon, 30 Aug 2021 11:03:32 -0300 Subject: [PATCH 08/15] chore: test with new orb --- .circleci/config.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.circleci/config.yml b/.circleci/config.yml index 7cd4ba19..e35e26e1 100644 --- a/.circleci/config.yml +++ b/.circleci/config.yml @@ -1,6 +1,6 @@ version: 2.1 orbs: - release-management: salesforce/npm-release-management@dev:2cb4d86 + release-management: salesforce/npm-release-management@dev:fa5e1cc parameters: run-auto-workflows: description: > From a514dc97ef986b39c464212ce37dc277be5f0938 Mon Sep 17 00:00:00 2001 From: Benjamin Maggi Date: Mon, 30 Aug 2021 14:02:21 -0300 Subject: [PATCH 09/15] chore: unskip test --- test/nuts/plugin-install.nut.ts | 70 +++------------------------------ 1 file changed, 6 insertions(+), 64 deletions(-) diff --git a/test/nuts/plugin-install.nut.ts b/test/nuts/plugin-install.nut.ts index 7685fe25..68d9f81e 100644 --- a/test/nuts/plugin-install.nut.ts +++ b/test/nuts/plugin-install.nut.ts @@ -6,25 +6,17 @@ */ import * as path from 'path'; -// import { /* existsSync, */ promises as fs } from 'fs'; -// import { copy, remove } from 'fs-extra'; import { expect } from 'chai'; -// import { expect } from '@salesforce/command/lib/test'; import { TestSession, execCmd } from '@salesforce/cli-plugins-testkit'; import { fs } from '@salesforce/core'; const SIGNED_MODULE_NAME = '@salesforce/plugin-user'; const UNSIGNED_MODULE_NAME = 'sfdx-jayree'; -// const PLUGIN_TRUST_REPO = 'git@github.com:salesforcecli/plugin-trust.git'; let session: TestSession; -describe.skip('plugins:install commands', () => { +describe('plugins:install commands', () => { before(async () => { session = await TestSession.create(); - // await Promise.all([ - // fs.mkdir(path.join(session.homeDir, 'SC_OUTPUT')), - // fs.mkdir(path.join(session.homeDir, 'MC_OUTPUT')), - // ]); }); after(async () => { @@ -32,39 +24,13 @@ describe.skip('plugins:install commands', () => { await session?.clean(); }); - it.skip('plugins:install signed plugin', () => { + it('plugins:install signed plugin', () => { process.env.TESTKIT_EXECUTABLE_PATH = 'sfdx'; const result = execCmd(`plugins:install ${SIGNED_MODULE_NAME}`, { ensureExitCode: 0, }); - // eslint-disable-next-line no-console - console.log('result', result); expect(result.shellOutput.stdout).to.contain(`Successfully validated digital signature for ${SIGNED_MODULE_NAME}`); }); - - it('plugins:install unsigned plugin', () => { - process.env.TESTKIT_EXECUTABLE_PATH = 'sfdx'; - const result = execCmd(`plugins:install ${UNSIGNED_MODULE_NAME}`, { - ensureExitCode: 0, - }); - // eslint-disable-next-line no-console - console.log('result', result); - expect(result.shellOutput.stdout).to.contain( - 'This plugin is not digitally signed and its authenticity cannot be verified. Continue installation y/n?:' - ); - }); - - // it('plugins:trust:verify', () => { - // execCmd('plugins:trust:verify --npm @salesforce/plugin-user', { - // ensureExitCode: 0, - // }); - // }); - - // it('plugins:trust:verify', () => { - // execCmd('plugins:trust:verify --npm sfdx-jayree', { - // ensureExitCode: 0, - // }); - // }); }); describe('plugins:install commands', () => { @@ -73,35 +39,9 @@ describe('plugins:install commands', () => { const configDir = path.join(session.homeDir, '.config', 'sfdx'); fs.mkdirSync(configDir, { recursive: true }); fs.writeJsonSync(path.join(configDir, 'unsignedPluginAllowList.json'), [UNSIGNED_MODULE_NAME]); - // process.env.TESTKIT_EXECUTABLE_PATH = 'sfdx'; - // const pluginTrustRealPath = path.dirname(session.dir).split(path.sep).pop(); - const pluginTrustRealPath = path.dirname(session.dir); - // eslint-disable-next-line no-console - console.log('pluginTrustRealPath', pluginTrustRealPath); - const pluginTrustPath = path.join(session.dir, 'plugin-trust'); - - // fs.mkdirSync(pluginTrustPath, { recursive: true }); - // fs.copyFileSync(pluginTrustRealPath, pluginTrustPath); - - // eslint-disable-next-line no-console - console.log('session.dir', session.dir); - // process.env.TESTKIT_EXECUTABLE_PATH = 'git'; - // execCmd(`clone ${PLUGIN_TRUST_REPO}`, { - // cwd: session.dir, - // ensureExitCode: 0, - // }); - // eslint-disable-next-line no-console - console.log('pluginTrustPath', pluginTrustPath); - // process.env.TESTKIT_EXECUTABLE_PATH = 'yarn'; - // execCmd('install', { - // cwd: pluginTrustPath, - // ensureExitCode: 0, - // }); - // eslint-disable-next-line no-console - console.log('plugins:link', pluginTrustPath); process.env.TESTKIT_EXECUTABLE_PATH = 'sfdx'; execCmd('plugins:link . --dev-debug', { - cwd: pluginTrustRealPath, + cwd: path.dirname(session.dir), ensureExitCode: 0, }); }); @@ -118,6 +58,8 @@ describe('plugins:install commands', () => { }); // eslint-disable-next-line no-console console.log('result', result); - // expect(result.shellOutput.stdout).to.contain(); + expect(result.shellOutput.stdout).to.contain( + `The plugin [${UNSIGNED_MODULE_NAME}] is not digitally signed but it is allow-listed.` + ); }); }); From fae95a04a97642b17fbc311e7d7356bfbe9fbba6 Mon Sep 17 00:00:00 2001 From: Benjamin Maggi Date: Mon, 30 Aug 2021 14:48:45 -0300 Subject: [PATCH 10/15] chore: test answers for prompt --- package.json | 2 +- test/nuts/plugin-install.nut.ts | 15 ++++++++++++++- yarn.lock | 8 ++++---- 3 files changed, 19 insertions(+), 6 deletions(-) diff --git a/package.json b/package.json index cd3488e9..85ca746f 100644 --- a/package.json +++ b/package.json @@ -16,7 +16,7 @@ "devDependencies": { "@oclif/dev-cli": "^1", "@oclif/plugin-command-snapshot": "^2.0.0", - "@salesforce/cli-plugins-testkit": "^1.1.5", + "@salesforce/cli-plugins-testkit": "^1.3.0", "@salesforce/dev-config": "^2.1.2", "@salesforce/dev-scripts": "0.9.11", "@salesforce/plugin-command-reference": "^1.3.0", diff --git a/test/nuts/plugin-install.nut.ts b/test/nuts/plugin-install.nut.ts index 68d9f81e..c9975b95 100644 --- a/test/nuts/plugin-install.nut.ts +++ b/test/nuts/plugin-install.nut.ts @@ -51,13 +51,26 @@ describe('plugins:install commands', () => { await session?.clean(); }); + it('plugins:install fails on unsigned plugin', () => { + process.env.TESTKIT_EXECUTABLE_PATH = 'sfdx'; + const result = execCmd(`plugins:install ${UNSIGNED_MODULE_NAME}`, { + ensureExitCode: 1, + answers: ['N'], + }); + // eslint-disable-next-line no-console + console.log('result', result); + // expect(result.shellOutput.stdout).to.contain( + // `The plugin [${UNSIGNED_MODULE_NAME}] is not digitally signed but it is allow-listed.` + // ); + }); + it('plugins:install unsigned plugin', () => { process.env.TESTKIT_EXECUTABLE_PATH = 'sfdx'; const result = execCmd(`plugins:install ${UNSIGNED_MODULE_NAME}`, { ensureExitCode: 0, }); // eslint-disable-next-line no-console - console.log('result', result); + // console.log('result', result); expect(result.shellOutput.stdout).to.contain( `The plugin [${UNSIGNED_MODULE_NAME}] is not digitally signed but it is allow-listed.` ); diff --git a/yarn.lock b/yarn.lock index 51aab7a7..e190a29a 100644 --- a/yarn.lock +++ b/yarn.lock @@ -674,10 +674,10 @@ mv "~2" safe-json-stringify "~1" -"@salesforce/cli-plugins-testkit@^1.1.5": - version "1.2.2" - resolved "https://registry.yarnpkg.com/@salesforce/cli-plugins-testkit/-/cli-plugins-testkit-1.2.2.tgz#a61aa0b44f46d5432d4fed48b59985edf2f94800" - integrity sha512-o2UyppeU7ZZKWLHmvkxq1KNTS4ThrCUvOI5DAA1qujetDR0uHbv08UxSGEYlhllL9nIj6HZWcEczjbShc9tWMw== +"@salesforce/cli-plugins-testkit@^1.3.0": + version "1.3.0" + resolved "https://registry.yarnpkg.com/@salesforce/cli-plugins-testkit/-/cli-plugins-testkit-1.3.0.tgz#19466352285412065a3f4d13ff3e7a1739d6410b" + integrity sha512-8YLfvRZZQBDborBrrWz/QL8G5Fhx/UHcgex1kWgLNLmYnm2gaDbZBpP/TWvw8jQvyd8EKxKzivDdMBC8JIXBog== dependencies: "@salesforce/core" "^2.24.0" "@salesforce/kit" "^1.5.13" From 962e09ecf91a80728e9a1803ee895dd066db7bd0 Mon Sep 17 00:00:00 2001 From: Benjamin Maggi Date: Mon, 30 Aug 2021 15:10:44 -0300 Subject: [PATCH 11/15] chore: test install fail --- test/nuts/plugin-install.nut.ts | 26 +++++++++++++------------- 1 file changed, 13 insertions(+), 13 deletions(-) diff --git a/test/nuts/plugin-install.nut.ts b/test/nuts/plugin-install.nut.ts index c9975b95..af832dd0 100644 --- a/test/nuts/plugin-install.nut.ts +++ b/test/nuts/plugin-install.nut.ts @@ -31,6 +31,19 @@ describe('plugins:install commands', () => { }); expect(result.shellOutput.stdout).to.contain(`Successfully validated digital signature for ${SIGNED_MODULE_NAME}`); }); + + it('plugins:install fails on unsigned plugin', () => { + process.env.TESTKIT_EXECUTABLE_PATH = 'sfdx'; + const result = execCmd(`plugins:install ${UNSIGNED_MODULE_NAME}`, { + ensureExitCode: 1, + answers: ['N'], + }); + // eslint-disable-next-line no-console + console.log('result', result); + // expect(result.shellOutput.stdout).to.contain( + // `The plugin [${UNSIGNED_MODULE_NAME}] is not digitally signed but it is allow-listed.` + // ); + }); }); describe('plugins:install commands', () => { @@ -51,19 +64,6 @@ describe('plugins:install commands', () => { await session?.clean(); }); - it('plugins:install fails on unsigned plugin', () => { - process.env.TESTKIT_EXECUTABLE_PATH = 'sfdx'; - const result = execCmd(`plugins:install ${UNSIGNED_MODULE_NAME}`, { - ensureExitCode: 1, - answers: ['N'], - }); - // eslint-disable-next-line no-console - console.log('result', result); - // expect(result.shellOutput.stdout).to.contain( - // `The plugin [${UNSIGNED_MODULE_NAME}] is not digitally signed but it is allow-listed.` - // ); - }); - it('plugins:install unsigned plugin', () => { process.env.TESTKIT_EXECUTABLE_PATH = 'sfdx'; const result = execCmd(`plugins:install ${UNSIGNED_MODULE_NAME}`, { From 39a017f53ef64c7ac573cc11d857bb88f934d50f Mon Sep 17 00:00:00 2001 From: Benjamin Maggi Date: Mon, 30 Aug 2021 15:21:25 -0300 Subject: [PATCH 12/15] chore: test install fail and accept --- test/nuts/plugin-install.nut.ts | 19 ++++++++++++++++--- 1 file changed, 16 insertions(+), 3 deletions(-) diff --git a/test/nuts/plugin-install.nut.ts b/test/nuts/plugin-install.nut.ts index af832dd0..617f576b 100644 --- a/test/nuts/plugin-install.nut.ts +++ b/test/nuts/plugin-install.nut.ts @@ -32,16 +32,29 @@ describe('plugins:install commands', () => { expect(result.shellOutput.stdout).to.contain(`Successfully validated digital signature for ${SIGNED_MODULE_NAME}`); }); - it('plugins:install fails on unsigned plugin', () => { + it('plugins:install prompts on unsigned plugin (denies)', () => { process.env.TESTKIT_EXECUTABLE_PATH = 'sfdx'; const result = execCmd(`plugins:install ${UNSIGNED_MODULE_NAME}`, { - ensureExitCode: 1, + ensureExitCode: 2, // This is the output code for the NO answer answers: ['N'], }); // eslint-disable-next-line no-console console.log('result', result); + expect(result.shellOutput.stdout).to.contain( + 'This plugin is not digitally signed and its authenticity cannot be verified. Continue installation y/n?:.' + ); + }); + + it('plugins:install prompts on unsigned plugin (accepts)', () => { + process.env.TESTKIT_EXECUTABLE_PATH = 'sfdx'; + const result = execCmd(`plugins:install ${UNSIGNED_MODULE_NAME}`, { + ensureExitCode: 0, // This is the output code for the NO answer + answers: ['Y'], + }); + // eslint-disable-next-line no-console + console.log('result', result); // expect(result.shellOutput.stdout).to.contain( - // `The plugin [${UNSIGNED_MODULE_NAME}] is not digitally signed but it is allow-listed.` + // 'This plugin is not digitally signed and its authenticity cannot be verified. Continue installation y/n?:.' // ); }); }); From 280f1b9cbaa0362e7215c494fde21a87cd5a896d Mon Sep 17 00:00:00 2001 From: Benjamin Maggi Date: Mon, 30 Aug 2021 15:39:39 -0300 Subject: [PATCH 13/15] chore: check Y and N --- test/nuts/plugin-install.nut.ts | 16 ++++++++-------- 1 file changed, 8 insertions(+), 8 deletions(-) diff --git a/test/nuts/plugin-install.nut.ts b/test/nuts/plugin-install.nut.ts index 617f576b..ae9b4763 100644 --- a/test/nuts/plugin-install.nut.ts +++ b/test/nuts/plugin-install.nut.ts @@ -40,9 +40,10 @@ describe('plugins:install commands', () => { }); // eslint-disable-next-line no-console console.log('result', result); - expect(result.shellOutput.stdout).to.contain( - 'This plugin is not digitally signed and its authenticity cannot be verified. Continue installation y/n?:.' + expect(result.shellOutput.stderr).to.contain( + 'This plugin is not digitally signed and its authenticity cannot be verified. Continue installation y/n?:' ); + expect(result.shellOutput.stderr).to.contain('The user canceled the plugin installation'); }); it('plugins:install prompts on unsigned plugin (accepts)', () => { @@ -53,9 +54,10 @@ describe('plugins:install commands', () => { }); // eslint-disable-next-line no-console console.log('result', result); - // expect(result.shellOutput.stdout).to.contain( - // 'This plugin is not digitally signed and its authenticity cannot be verified. Continue installation y/n?:.' - // ); + expect(result.shellOutput.stderr).to.contain( + 'This plugin is not digitally signed and its authenticity cannot be verified. Continue installation y/n?:' + ); + expect(result.shellOutput.stdout).to.contain('Finished digital signature check'); }); }); @@ -77,13 +79,11 @@ describe('plugins:install commands', () => { await session?.clean(); }); - it('plugins:install unsigned plugin', () => { + it('plugins:install unsigned plugin in the allow list', () => { process.env.TESTKIT_EXECUTABLE_PATH = 'sfdx'; const result = execCmd(`plugins:install ${UNSIGNED_MODULE_NAME}`, { ensureExitCode: 0, }); - // eslint-disable-next-line no-console - // console.log('result', result); expect(result.shellOutput.stdout).to.contain( `The plugin [${UNSIGNED_MODULE_NAME}] is not digitally signed but it is allow-listed.` ); From d4ad850a5a679895358c4776bc7ebd3f5983da82 Mon Sep 17 00:00:00 2001 From: Benjamin Maggi Date: Mon, 30 Aug 2021 16:40:47 -0300 Subject: [PATCH 14/15] chore: consolidate class --- src/lib/installationVerification.ts | 47 +++++------------------ src/lib/npmCommand.ts | 28 +++++++++----- test/lib/installationVerification.test.ts | 3 +- test/nuts/plugin-install.nut.ts | 8 +--- 4 files changed, 30 insertions(+), 56 deletions(-) diff --git a/src/lib/installationVerification.ts b/src/lib/installationVerification.ts index a6126783..65de42eb 100644 --- a/src/lib/installationVerification.ts +++ b/src/lib/installationVerification.ts @@ -17,7 +17,7 @@ import { promisify as utilPromisify } from 'util'; import * as crypto from 'crypto'; import { Logger, fs, SfdxError } from '@salesforce/core'; import * as request from 'request'; -import { NpmModule } from '../lib/npmCommand'; +import { NpmModule, NpmMeta } from '../lib/npmCommand'; import { NpmName } from './NpmName'; const CRYPTO_LEVEL = 'RSA-SHA256'; @@ -35,20 +35,6 @@ export interface Verifier { isAllowListed(): Promise; } -export type NpmShowResults = { - versions: string[]; - 'dist-tags': { - [name: string]: string; - }; - sfdx?: { - publicKeyUrl: string; - signatureUrl: string; - }; - dist?: { - [name: string]: string; - }; -}; - export class CodeVerifierInfo { private signature: Readable; private publicKey: Readable; @@ -150,20 +136,6 @@ export const getNpmRegistry = (): URL => { return new URL(process.env.SFDX_NPM_REGISTRY || DEFAULT_REGISTRY); }; -/** - * simple data structure representing the discovered meta information needed for signing, - */ -export class NpmMeta { - public tarballUrl: string; - public signatureUrl: string; - public publicKeyUrl: string; - public tarballLocalPath: string; - public verified: boolean; - public moduleName: string; - public version: string; - public tarballFilename: string; -} - /** * class for verifying a digital signature pack of an npm */ @@ -373,9 +345,8 @@ export class InstallationVerification implements Verifier { ? `@${this.pluginNpmName.scope}/${this.pluginNpmName.name}` : this.pluginNpmName.name; - const npmMetadata = new NpmModule(npmShowModule).show(npmRegistry.href); - const meta: NpmMeta = new NpmMeta(); - meta.moduleName = npmShowModule; + const npmModule = new NpmModule(npmShowModule); + const npmMetadata = npmModule.show(npmRegistry.href); logger.debug('retrieveNpmMeta | Found npm meta information.'); if (!npmMetadata.versions) { throw new SfdxError( @@ -413,7 +384,7 @@ export class InstallationVerification implements Verifier { } } - meta.version = versionNumber; + npmModule.npmMeta.version = versionNumber; if (!npmMetadata.sfdx) { throw new SfdxError('This plugin is not signed by Salesforce.com, Inc.', 'NotSigned'); @@ -425,7 +396,7 @@ export class InstallationVerification implements Verifier { ); } else { logger.debug(`retrieveNpmMeta | versionObject.sfdx.publicKeyUrl: ${npmMetadata.sfdx.publicKeyUrl}`); - meta.publicKeyUrl = npmMetadata.sfdx.publicKeyUrl; + npmModule.npmMeta.publicKeyUrl = npmMetadata.sfdx.publicKeyUrl; } if (!validSalesforceHostname(npmMetadata.sfdx.signatureUrl)) { @@ -435,13 +406,13 @@ export class InstallationVerification implements Verifier { ); } else { logger.debug(`retrieveNpmMeta | versionObject.sfdx.signatureUrl: ${npmMetadata.sfdx.signatureUrl}`); - meta.signatureUrl = npmMetadata.sfdx.signatureUrl; + npmModule.npmMeta.signatureUrl = npmMetadata.sfdx.signatureUrl; } - meta.tarballUrl = npmMetadata.dist.tarball; - logger.debug(`retrieveNpmMeta | meta.tarballUrl: ${meta.tarballUrl}`); + npmModule.npmMeta.tarballUrl = npmMetadata.dist.tarball; + logger.debug(`retrieveNpmMeta | meta.tarballUrl: ${npmModule.npmMeta.tarballUrl}`); - return meta; + return npmModule.npmMeta; } } diff --git a/src/lib/npmCommand.ts b/src/lib/npmCommand.ts index f259a569..0ca3c190 100644 --- a/src/lib/npmCommand.ts +++ b/src/lib/npmCommand.ts @@ -11,24 +11,28 @@ import * as shelljs from 'shelljs'; import npmRunPath from 'npm-run-path'; import { SfdxError, fs } from '@salesforce/core'; -type DistTags = { - [name: string]: string; +export type NpmMeta = { + tarballUrl?: string; + signatureUrl?: string; + publicKeyUrl?: string; + tarballLocalPath?: string; + verified?: boolean; + moduleName?: string; + version?: string; + tarballFilename?: string; }; -type NpmShowResults = { +export type NpmShowResults = { versions: string[]; - 'dist-tags': DistTags; + 'dist-tags': { + [name: string]: string; + }; sfdx?: { publicKeyUrl: string; signatureUrl: string; }; dist?: { - integrity: string; - shasum: string; - tarball: string; - fileCount: number; - unpackedSize: number; - 'npm-signature': string; + [name: string]: string; }; }; @@ -83,12 +87,16 @@ export class NpmCommand { } export class NpmModule { + public npmMeta: NpmMeta; private version: string; private module: string; public constructor(module: string, version = 'latest') { this.module = module; this.version = version; + this.npmMeta = { + moduleName: module, + }; } public show(registry: string): NpmShowResults { diff --git a/test/lib/installationVerification.test.ts b/test/lib/installationVerification.test.ts index e8f1f554..7a020826 100644 --- a/test/lib/installationVerification.test.ts +++ b/test/lib/installationVerification.test.ts @@ -18,11 +18,10 @@ import { getNpmRegistry, InstallationVerification, IRequest, - NpmMeta, VerificationConfig, Verifier, - NpmShowResults, } from '../../src/lib/installationVerification'; +import { NpmMeta, NpmShowResults } from '../../src/lib/npmCommand'; import { NpmName } from '../../src/lib/NpmName'; import { CERTIFICATE, TEST_DATA, TEST_DATA_SIGNATURE } from '../testCert'; diff --git a/test/nuts/plugin-install.nut.ts b/test/nuts/plugin-install.nut.ts index ae9b4763..b08f72c9 100644 --- a/test/nuts/plugin-install.nut.ts +++ b/test/nuts/plugin-install.nut.ts @@ -35,11 +35,9 @@ describe('plugins:install commands', () => { it('plugins:install prompts on unsigned plugin (denies)', () => { process.env.TESTKIT_EXECUTABLE_PATH = 'sfdx'; const result = execCmd(`plugins:install ${UNSIGNED_MODULE_NAME}`, { - ensureExitCode: 2, // This is the output code for the NO answer + ensureExitCode: 2, // code 2 is the output code for the NO answer answers: ['N'], }); - // eslint-disable-next-line no-console - console.log('result', result); expect(result.shellOutput.stderr).to.contain( 'This plugin is not digitally signed and its authenticity cannot be verified. Continue installation y/n?:' ); @@ -49,11 +47,9 @@ describe('plugins:install commands', () => { it('plugins:install prompts on unsigned plugin (accepts)', () => { process.env.TESTKIT_EXECUTABLE_PATH = 'sfdx'; const result = execCmd(`plugins:install ${UNSIGNED_MODULE_NAME}`, { - ensureExitCode: 0, // This is the output code for the NO answer + ensureExitCode: 0, answers: ['Y'], }); - // eslint-disable-next-line no-console - console.log('result', result); expect(result.shellOutput.stderr).to.contain( 'This plugin is not digitally signed and its authenticity cannot be verified. Continue installation y/n?:' ); From f044746a1c799d8be86396b79584ed031bb1ccd8 Mon Sep 17 00:00:00 2001 From: Benjamin Maggi Date: Wed, 1 Sep 2021 11:12:04 -0300 Subject: [PATCH 15/15] chore: use prod orb + classy constructor --- .circleci/config.yml | 2 +- src/lib/npmCommand.ts | 7 +------ 2 files changed, 2 insertions(+), 7 deletions(-) diff --git a/.circleci/config.yml b/.circleci/config.yml index e35e26e1..8729ec07 100644 --- a/.circleci/config.yml +++ b/.circleci/config.yml @@ -1,6 +1,6 @@ version: 2.1 orbs: - release-management: salesforce/npm-release-management@dev:fa5e1cc + release-management: salesforce/npm-release-management@4 parameters: run-auto-workflows: description: > diff --git a/src/lib/npmCommand.ts b/src/lib/npmCommand.ts index 0ca3c190..33bf0389 100644 --- a/src/lib/npmCommand.ts +++ b/src/lib/npmCommand.ts @@ -88,12 +88,7 @@ export class NpmCommand { export class NpmModule { public npmMeta: NpmMeta; - private version: string; - private module: string; - - public constructor(module: string, version = 'latest') { - this.module = module; - this.version = version; + public constructor(private module: string, private version: string = 'latest') { this.npmMeta = { moduleName: module, };