mirror of https://github.com/nodejs/corepack.git
1116 lines
38 KiB
TypeScript
1116 lines
38 KiB
TypeScript
import {beforeEach, describe, expect, it} from '@jest/globals';
|
|
import {Filename, ppath, xfs, npath, PortablePath} from '@yarnpkg/fslib';
|
|
import os from 'node:os';
|
|
import process from 'node:process';
|
|
|
|
import config from '../config.json';
|
|
import * as folderUtils from '../sources/folderUtils';
|
|
import {SupportedPackageManagerSet} from '../sources/types';
|
|
|
|
import {runCli} from './_runCli';
|
|
|
|
|
|
beforeEach(async () => {
|
|
// `process.env` is reset after each tests in setupTests.js.
|
|
process.env.COREPACK_HOME = npath.fromPortablePath(await xfs.mktempPromise());
|
|
process.env.COREPACK_DEFAULT_TO_LATEST = `0`;
|
|
});
|
|
|
|
it(`should refuse to download a package manager if the hash doesn't match`, async () => {
|
|
await xfs.mktempPromise(async cwd => {
|
|
await xfs.writeJsonPromise(ppath.join(cwd, `package.json` as Filename), {
|
|
packageManager: `yarn@1.22.4+sha1.deadbeef`,
|
|
});
|
|
|
|
await expect(runCli(cwd, [`yarn`, `--version`])).resolves.toMatchObject({
|
|
exitCode: 1,
|
|
stderr: ``,
|
|
stdout: /Mismatch hashes/,
|
|
});
|
|
});
|
|
});
|
|
|
|
it(`should refuse to download a known package manager from a URL`, async () => {
|
|
await xfs.mktempPromise(async cwd => {
|
|
// Package managers known by Corepack cannot be loaded from a URL.
|
|
await expect(runCli(cwd, [`yarn@https://registry.npmjs.com/yarn/-/yarn-1.22.21.tgz`, `--version`])).resolves.toMatchObject({
|
|
exitCode: 1,
|
|
stderr: ``,
|
|
stdout: /Illegal use of URL for known package manager/,
|
|
});
|
|
|
|
// Unknown package managers can be loaded from a URL.
|
|
await expect(runCli(cwd, [`corepack@https://registry.npmjs.com/corepack/-/corepack-0.24.1.tgz`, `--version`])).resolves.toMatchObject({
|
|
exitCode: 0,
|
|
stderr: ``,
|
|
stdout: `0.24.1\n`,
|
|
});
|
|
});
|
|
});
|
|
|
|
it.failing(`should refuse to download a known package manager from a URL in package.json`, async () => {
|
|
await xfs.mktempPromise(async cwd => {
|
|
// Package managers known by Corepack cannot be loaded from a URL.
|
|
await xfs.writeJsonPromise(ppath.join(cwd, `package.json` as Filename), {
|
|
packageManager: `yarn@https://registry.npmjs.com/yarn/-/yarn-1.22.21.tgz`,
|
|
});
|
|
|
|
await expect(runCli(cwd, [`yarn`, `--version`])).resolves.toMatchObject({
|
|
exitCode: 1,
|
|
stderr: ``,
|
|
stdout: /Illegal use of URL for known package manager/,
|
|
});
|
|
|
|
// Unknown package managers can be loaded from a URL.
|
|
await xfs.writeJsonPromise(ppath.join(cwd, `package.json` as Filename), {
|
|
packageManager: `corepack@https://registry.npmjs.com/corepack/-/corepack-0.24.1.tgz`,
|
|
});
|
|
|
|
await expect(runCli(cwd, [`corepack`, `--version`])).resolves.toMatchObject({
|
|
exitCode: 0,
|
|
stderr: ``,
|
|
stdout: `0.24.1\n`,
|
|
});
|
|
});
|
|
});
|
|
|
|
it(`should require a version to be specified`, async () => {
|
|
await xfs.mktempPromise(async cwd => {
|
|
await xfs.writeJsonPromise(ppath.join(cwd, `package.json` as Filename), {
|
|
packageManager: `yarn`,
|
|
});
|
|
|
|
await expect(runCli(cwd, [`yarn`, `--version`])).resolves.toMatchObject({
|
|
exitCode: 1,
|
|
stderr: ``,
|
|
stdout: /expected a semver version/,
|
|
});
|
|
|
|
await xfs.writeJsonPromise(ppath.join(cwd, `package.json` as Filename), {
|
|
packageManager: `yarn@stable`,
|
|
});
|
|
|
|
await expect(runCli(cwd, [`yarn`, `--version`])).resolves.toMatchObject({
|
|
exitCode: 1,
|
|
stderr: ``,
|
|
stdout: /expected a semver version/,
|
|
});
|
|
|
|
await xfs.writeJsonPromise(ppath.join(cwd, `package.json` as Filename), {
|
|
packageManager: `yarn@^1.0.0`,
|
|
});
|
|
|
|
await expect(runCli(cwd, [`yarn`, `--version`])).resolves.toMatchObject({
|
|
exitCode: 1,
|
|
stderr: ``,
|
|
stdout: /expected a semver version/,
|
|
});
|
|
});
|
|
});
|
|
|
|
const testedPackageManagers: Array<[string, string] | [string, string, string]> = [
|
|
[`yarn`, `1.22.4`],
|
|
[`yarn`, `1.22.4+sha1.01c1197ca5b27f21edc8bc472cd4c8ce0e5a470e`],
|
|
[`yarn`, `1.22.4+sha224.0d6eecaf4d82ec12566fdd97143794d0f0c317e0d652bd4d1b305430`],
|
|
[`yarn`, `https://registry.npmjs.com/yarn/-/yarn-1.22.21.tgz`, `1.22.21`],
|
|
[`yarn`, `https://registry.npmjs.com/yarn/-/yarn-1.22.21.tgz#sha1.1959a18351b811cdeedbd484a8f86c3cc3bbaf72`, `1.22.21`],
|
|
[`yarn`, `2.0.0-rc.30`],
|
|
[`yarn`, `2.0.0-rc.30+sha1.4f0423b01bcb57f8e390b4e0f1990831f92dd1da`],
|
|
[`yarn`, `2.0.0-rc.30+sha224.0e7a64468c358596db21c401ffeb11b6534fce7367afd3ae640eadf1`],
|
|
[`yarn`, `3.0.0-rc.2`],
|
|
[`yarn`, `3.0.0-rc.2+sha1.694bdad81703169e203febd57f9dc97d3be867bd`],
|
|
[`yarn`, `3.0.0-rc.2+sha224.f83f6d1cbfac10ba6b516a62ccd2a72ccd857aa6c514d1cd7185ec60`],
|
|
[`pnpm`, `4.11.6`],
|
|
[`pnpm`, `4.11.6+sha1.7cffc04295f4db4740225c6c37cc345eb923c06a`],
|
|
[`pnpm`, `4.11.6+sha224.7783c4b01916b7a69e6ff05d328df6f83cb7f127e9c96be88739386d`],
|
|
[`pnpm`, `6.6.2`],
|
|
[`pnpm`, `6.6.2+sha1.7b4d6b176c1b93b5670ed94c24babb7d80c13854`],
|
|
[`pnpm`, `6.6.2+sha224.eb5c0acad3b0f40ecdaa2db9aa5a73134ad256e17e22d1419a2ab073`],
|
|
[`npm`, `6.14.2`],
|
|
[`npm`, `6.14.2+sha1.f057d35cd4792c4c511bb1fa332edb43143d07b0`],
|
|
[`npm`, `6.14.2+sha224.50512c1eb404900ee78586faa6d756b8d867ff46a328e6fb4cdf3a87`],
|
|
];
|
|
|
|
for (const [name, version, expectedVersion = version.split(`+`, 1)[0]] of testedPackageManagers) {
|
|
it(`should use the right package manager version for a given project (${name}@${version})`, async () => {
|
|
process.env.COREPACK_ENABLE_UNSAFE_CUSTOM_URLS = `1`;
|
|
await xfs.mktempPromise(async cwd => {
|
|
await expect(runCli(cwd, [`${name}@${version}`, `--version`])).resolves.toMatchObject({
|
|
exitCode: 0,
|
|
stderr: ``,
|
|
stdout: `${expectedVersion}\n`,
|
|
});
|
|
|
|
await xfs.writeJsonPromise(ppath.join(cwd, `package.json` as Filename), {
|
|
packageManager: `${name}@${version}`,
|
|
});
|
|
|
|
await expect(runCli(cwd, [name, `--version`])).resolves.toMatchObject({
|
|
exitCode: 0,
|
|
stderr: ``,
|
|
stdout: `${expectedVersion}\n`,
|
|
});
|
|
});
|
|
});
|
|
}
|
|
|
|
it(`should update the Known Good Release only when the major matches`, async () => {
|
|
await xfs.writeJsonPromise(ppath.join(npath.toPortablePath(folderUtils.getCorepackHomeFolder()), `lastKnownGood.json`), {
|
|
yarn: `1.0.0`,
|
|
});
|
|
|
|
process.env.COREPACK_DEFAULT_TO_LATEST = `1`;
|
|
|
|
await xfs.mktempPromise(async cwd => {
|
|
await xfs.writeJsonPromise(ppath.join(cwd, `package.json` as Filename), {
|
|
packageManager: `yarn@1.22.4+sha224.0d6eecaf4d82ec12566fdd97143794d0f0c317e0d652bd4d1b305430`,
|
|
});
|
|
|
|
await expect(runCli(cwd, [`yarn`, `--version`])).resolves.toMatchObject({
|
|
exitCode: 0,
|
|
stderr: ``,
|
|
stdout: `1.22.4\n`,
|
|
});
|
|
|
|
await xfs.removePromise(ppath.join(cwd, `package.json` as Filename));
|
|
|
|
await expect(runCli(cwd, [`yarn`, `--version`])).resolves.toMatchObject({
|
|
exitCode: 0,
|
|
stderr: ``,
|
|
stdout: `1.22.4\n`,
|
|
});
|
|
|
|
await xfs.writeJsonPromise(ppath.join(cwd, `package.json` as Filename), {
|
|
packageManager: `yarn@2.2.2`,
|
|
});
|
|
|
|
await expect(runCli(cwd, [`yarn`, `--version`])).resolves.toMatchObject({
|
|
exitCode: 0,
|
|
stderr: ``,
|
|
stdout: `2.2.2\n`,
|
|
});
|
|
|
|
await xfs.removePromise(ppath.join(cwd, `package.json` as Filename));
|
|
|
|
await expect(runCli(cwd, [`yarn`, `--version`])).resolves.toMatchObject({
|
|
exitCode: 0,
|
|
stderr: ``,
|
|
stdout: `1.22.4\n`,
|
|
});
|
|
});
|
|
});
|
|
|
|
it(`should ignore the packageManager field when found within a node_modules vendor`, async () => {
|
|
await xfs.mktempPromise(async cwd => {
|
|
await xfs.mkdirPromise(ppath.join(cwd, `node_modules/foo` as PortablePath), {recursive: true});
|
|
await xfs.mkdirPromise(ppath.join(cwd, `node_modules/@foo/bar` as PortablePath), {recursive: true});
|
|
|
|
await xfs.writeJsonPromise(ppath.join(cwd, `package.json` as PortablePath), {
|
|
packageManager: `yarn@1.22.4`,
|
|
});
|
|
|
|
await xfs.writeJsonPromise(ppath.join(cwd, `node_modules/foo/package.json` as PortablePath), {
|
|
packageManager: `npm@6.14.2`,
|
|
});
|
|
|
|
await xfs.writeJsonPromise(ppath.join(cwd, `node_modules/@foo/bar/package.json` as PortablePath), {
|
|
packageManager: `npm@6.14.2`,
|
|
});
|
|
|
|
await expect(runCli(ppath.join(cwd, `node_modules/foo` as PortablePath), [`yarn`, `--version`])).resolves.toMatchObject({
|
|
exitCode: 0,
|
|
stderr: ``,
|
|
stdout: `1.22.4\n`,
|
|
});
|
|
|
|
await expect(runCli(ppath.join(cwd, `node_modules/@foo/bar` as PortablePath), [`yarn`, `--version`])).resolves.toMatchObject({
|
|
exitCode: 0,
|
|
stderr: ``,
|
|
stdout: `1.22.4\n`,
|
|
});
|
|
});
|
|
});
|
|
|
|
it(`should use the closest matching packageManager field`, async () => {
|
|
await xfs.mktempPromise(async cwd => {
|
|
await xfs.mkdirPromise(ppath.join(cwd, `foo` as PortablePath), {recursive: true});
|
|
|
|
await xfs.writeJsonPromise(ppath.join(cwd, `package.json` as PortablePath), {
|
|
packageManager: `yarn@1.22.4`,
|
|
});
|
|
|
|
await xfs.writeJsonPromise(ppath.join(cwd, `foo/package.json` as PortablePath), {
|
|
packageManager: `npm@6.14.2`,
|
|
});
|
|
|
|
await expect(runCli(ppath.join(cwd, `foo` as PortablePath), [`npm`, `--version`])).resolves.toMatchObject({
|
|
exitCode: 0,
|
|
stderr: ``,
|
|
stdout: `6.14.2\n`,
|
|
});
|
|
});
|
|
});
|
|
|
|
it(`should expose its root to spawned processes`, async () => {
|
|
await xfs.mktempPromise(async cwd => {
|
|
await xfs.writeJsonPromise(ppath.join(cwd, `package.json` as Filename), {
|
|
packageManager: `npm@6.14.2`,
|
|
});
|
|
|
|
await expect(runCli(cwd, [`npm`, `run`, `env`])).resolves.toMatchObject({
|
|
exitCode: 0,
|
|
stdout: expect.stringContaining(`COREPACK_ROOT=${npath.dirname(__dirname)}`),
|
|
});
|
|
});
|
|
});
|
|
|
|
it(`shouldn't allow using regular Yarn commands on npm-configured projects`, async () => {
|
|
await xfs.mktempPromise(async cwd => {
|
|
await xfs.writeJsonPromise(ppath.join(cwd, `package.json` as Filename), {
|
|
packageManager: `npm@6.14.2`,
|
|
});
|
|
|
|
await expect(runCli(cwd, [`yarn`, `--version`])).resolves.toMatchObject({
|
|
exitCode: 1,
|
|
stderr: ``,
|
|
});
|
|
});
|
|
});
|
|
|
|
it(`should allow using transparent commands on npm-configured projects`, async () => {
|
|
await xfs.mktempPromise(async cwd => {
|
|
await xfs.writeJsonPromise(ppath.join(cwd, `package.json` as Filename), {
|
|
packageManager: `npm@6.14.2`,
|
|
});
|
|
|
|
await expect(runCli(cwd, [`yarn`, `dlx`, `--help`])).resolves.toMatchObject({
|
|
exitCode: 0,
|
|
stderr: ``,
|
|
});
|
|
});
|
|
});
|
|
|
|
it(`should transparently use the preconfigured version when there is no local project`, async () => {
|
|
await xfs.mktempPromise(async cwd => {
|
|
await expect(runCli(cwd, [`yarn`, `--version`])).resolves.toMatchObject({
|
|
exitCode: 0,
|
|
stderr: ``,
|
|
});
|
|
});
|
|
});
|
|
|
|
// Note that we don't prevent using any package manager. This ensures that
|
|
// projects will receive as little disruption as possible (for example, we
|
|
// don't prompt to set the packageManager field).
|
|
|
|
for (const name of SupportedPackageManagerSet) {
|
|
it(`should use the pinned version when local projects don't list any spec (${name})`, async () => {
|
|
await xfs.mktempPromise(async cwd => {
|
|
await xfs.writeJsonPromise(ppath.join(cwd, `package.json` as Filename), {
|
|
// empty package.json file
|
|
});
|
|
|
|
await expect(runCli(cwd, [name, `--version`])).resolves.toMatchObject({
|
|
stdout: `${config.definitions[name].default.split(`+`, 1)[0]}\n`,
|
|
exitCode: 0,
|
|
});
|
|
});
|
|
});
|
|
}
|
|
|
|
it(`should configure the project when calling a package manager on it for the first time`, async () => {
|
|
await xfs.mktempPromise(async cwd => {
|
|
await xfs.writeJsonPromise(ppath.join(cwd, `package.json` as Filename), {
|
|
// empty package.json file
|
|
});
|
|
|
|
await runCli(cwd, [`yarn`]);
|
|
|
|
const data = await xfs.readJsonPromise(ppath.join(cwd, `package.json` as Filename));
|
|
|
|
expect(data).toMatchObject({
|
|
packageManager: `yarn@${config.definitions.yarn.default}`,
|
|
});
|
|
});
|
|
});
|
|
|
|
it(`should allow updating the pinned version using the "corepack install -g" command`, async () => {
|
|
await xfs.mktempPromise(async cwd => {
|
|
await expect(runCli(cwd, [`install`, `-g`, `yarn@1.0.0`])).resolves.toMatchObject({
|
|
exitCode: 0,
|
|
stderr: ``,
|
|
});
|
|
|
|
await xfs.writeJsonPromise(ppath.join(cwd, `package.json` as Filename), {
|
|
// empty package.json file
|
|
});
|
|
|
|
await expect(runCli(cwd, [`yarn`, `--version`])).resolves.toMatchObject({
|
|
stdout: `1.0.0\n`,
|
|
exitCode: 0,
|
|
});
|
|
});
|
|
});
|
|
|
|
it(`should allow to call "corepack install -g" with a tag`, async () => {
|
|
await xfs.mktempPromise(async cwd => {
|
|
await expect(runCli(cwd, [`install`, `-g`, `npm@latest-7`])).resolves.toMatchObject({
|
|
exitCode: 0,
|
|
stderr: ``,
|
|
});
|
|
|
|
await xfs.writeJsonPromise(ppath.join(cwd, `package.json` as Filename), {
|
|
// empty package.json file
|
|
});
|
|
|
|
await expect(runCli(cwd, [`npm`, `--version`])).resolves.toMatchObject({
|
|
stdout: expect.stringMatching(/^7\./),
|
|
exitCode: 0,
|
|
});
|
|
});
|
|
});
|
|
|
|
it(`should allow to call "corepack install -g" without any range`, async () => {
|
|
await xfs.mktempPromise(async cwd => {
|
|
await expect(runCli(cwd, [`install`, `-g`, `yarn`])).resolves.toMatchObject({
|
|
exitCode: 0,
|
|
stderr: ``,
|
|
});
|
|
|
|
await xfs.writeJsonPromise(ppath.join(cwd, `package.json` as Filename), {
|
|
// empty package.json file
|
|
});
|
|
|
|
await expect(runCli(cwd, [`yarn`, `--version`])).resolves.toMatchObject({
|
|
stdout: expect.not.stringMatching(/^[123]\./),
|
|
exitCode: 0,
|
|
});
|
|
});
|
|
});
|
|
|
|
it(`should allow to call "corepack install" without arguments within a configured project`, async () => {
|
|
await xfs.mktempPromise(async cwd => {
|
|
await xfs.writeJsonPromise(ppath.join(cwd, `package.json` as Filename), {
|
|
packageManager: `yarn@1.0.0`,
|
|
});
|
|
|
|
await expect(runCli(cwd, [`install`])).resolves.toMatchObject({
|
|
exitCode: 0,
|
|
stderr: ``,
|
|
});
|
|
|
|
// Disable the network to make sure we don't succeed by accident
|
|
process.env.COREPACK_ENABLE_NETWORK = `0`;
|
|
|
|
await expect(runCli(cwd, [`yarn`, `--version`])).resolves.toMatchObject({
|
|
stdout: `1.0.0\n`,
|
|
stderr: ``,
|
|
exitCode: 0,
|
|
});
|
|
});
|
|
});
|
|
|
|
it(`should refuse to run a different package manager within a configured project`, async () => {
|
|
await xfs.mktempPromise(async cwd => {
|
|
await xfs.writeJsonPromise(ppath.join(cwd, `package.json` as Filename), {
|
|
packageManager: `yarn@1.0.0`,
|
|
});
|
|
|
|
process.env.FORCE_COLOR = `0`;
|
|
|
|
await expect(runCli(cwd, [`pnpm`, `--version`])).resolves.toMatchObject({
|
|
stdout: `Usage Error: This project is configured to use yarn because ${
|
|
npath.fromPortablePath(ppath.join(cwd, `package.json` as Filename))
|
|
} has a "packageManager" field\n\n$ pnpm ...\n`,
|
|
exitCode: 1,
|
|
});
|
|
|
|
// Disable strict checking to workaround the UsageError.
|
|
process.env.COREPACK_ENABLE_STRICT = `0`;
|
|
|
|
await expect(runCli(cwd, [`yarn`, `--version`])).resolves.toMatchObject({
|
|
stdout: `1.0.0\n`,
|
|
stderr: ``,
|
|
exitCode: 0,
|
|
});
|
|
await expect(runCli(cwd, [`pnpm`, `--version`])).resolves.toMatchObject({
|
|
stdout: `${config.definitions.pnpm.default.split(`+`, 1)[0]}\n`,
|
|
stderr: ``,
|
|
exitCode: 0,
|
|
});
|
|
});
|
|
});
|
|
|
|
|
|
it(`should always use fallback version when project spec env is disabled`, async () => {
|
|
await xfs.mktempPromise(async cwd => {
|
|
await xfs.writeJsonPromise(ppath.join(cwd, `package.json` as Filename), {
|
|
packageManager: `yarn@1.0.0`,
|
|
});
|
|
process.env.COREPACK_ENABLE_PROJECT_SPEC = `0`;
|
|
|
|
await expect(runCli(cwd, [`yarn`, `--version`])).resolves.toMatchObject({
|
|
stdout: `${config.definitions.yarn.default.split(`+`, 1)[0]}\n`,
|
|
stderr: ``,
|
|
exitCode: 0,
|
|
});
|
|
await expect(runCli(cwd, [`pnpm`, `--version`])).resolves.toMatchObject({
|
|
stdout: `${config.definitions.pnpm.default.split(`+`, 1)[0]}\n`,
|
|
stderr: ``,
|
|
exitCode: 0,
|
|
});
|
|
});
|
|
});
|
|
|
|
it(`should support disabling the network accesses from the environment`, async () => {
|
|
process.env.COREPACK_ENABLE_NETWORK = `0`;
|
|
|
|
await xfs.mktempPromise(async cwd => {
|
|
await xfs.writeJsonPromise(ppath.join(cwd, `package.json` as Filename), {
|
|
packageManager: `yarn@2.2.2`,
|
|
});
|
|
|
|
await expect(runCli(cwd, [`yarn`, `--version`])).resolves.toMatchObject({
|
|
stdout: expect.stringContaining(`Network access disabled by the environment`),
|
|
stderr: ``,
|
|
exitCode: 1,
|
|
});
|
|
});
|
|
});
|
|
|
|
describe(`read-only and offline environment`, () => {
|
|
it(`should support running in project scope`, async () => {
|
|
await xfs.mktempPromise(async cwd => {
|
|
// Reset to default
|
|
delete process.env.COREPACK_DEFAULT_TO_LATEST;
|
|
|
|
// Prepare fake project
|
|
await xfs.writeJsonPromise(ppath.join(cwd, `package.json` as Filename), {
|
|
packageManager: `yarn@2.2.2`,
|
|
});
|
|
|
|
// $ corepack install
|
|
await expect(runCli(cwd, [`install`])).resolves.toMatchObject({
|
|
stdout: `Adding yarn@2.2.2 to the cache...\n`,
|
|
stderr: ``,
|
|
exitCode: 0,
|
|
});
|
|
|
|
// Let corepack discover the latest yarn version.
|
|
// BUG: This should not be necessary with a fully specified version in package.json plus populated corepack cache.
|
|
// Engine.executePackageManagerRequest needs to defer the fallback work. This requires a big refactoring.
|
|
await expect(runCli(cwd, [`yarn`, `--version`])).resolves.toMatchObject({
|
|
exitCode: 0,
|
|
});
|
|
|
|
// Make COREPACK_HOME ro
|
|
const home = npath.toPortablePath(folderUtils.getCorepackHomeFolder());
|
|
await xfs.chmodPromise(ppath.join(home, `lastKnownGood.json`), 0o444);
|
|
await xfs.chmodPromise(home, 0o555);
|
|
|
|
// Use fake proxies to simulate offline mode
|
|
process.env.HTTP_PROXY = `0.0.0.0`;
|
|
process.env.HTTPS_PROXY = `0.0.0.0`;
|
|
|
|
// $ corepack yarn --version
|
|
await expect(runCli(cwd, [`yarn`, `--version`])).resolves.toMatchObject({
|
|
stdout: `2.2.2\n`,
|
|
stderr: ``,
|
|
exitCode: 0,
|
|
});
|
|
});
|
|
});
|
|
|
|
it(`should support running globally`, async () => {
|
|
await xfs.mktempPromise(async installDir => {
|
|
// Reset to default
|
|
delete process.env.COREPACK_DEFAULT_TO_LATEST;
|
|
|
|
await expect(runCli(installDir, [`enable`, `--install-directory`, npath.fromPortablePath(installDir), `yarn`])).resolves.toMatchObject({
|
|
stdout: ``,
|
|
stderr: ``,
|
|
exitCode: 0,
|
|
});
|
|
|
|
await expect(runCli(installDir, [`install`, `--global`, `yarn@2.2.2`])).resolves.toMatchObject({
|
|
stdout: `Installing yarn@2.2.2...\n`,
|
|
stderr: ``,
|
|
exitCode: 0,
|
|
});
|
|
|
|
// Make COREPACK_HOME ro
|
|
const home = npath.toPortablePath(folderUtils.getCorepackHomeFolder());
|
|
await xfs.chmodPromise(ppath.join(home, `lastKnownGood.json`), 0o444);
|
|
await xfs.chmodPromise(home, 0o555);
|
|
|
|
// Use fake proxies to simulate offline mode
|
|
process.env.HTTP_PROXY = `0.0.0.0`;
|
|
process.env.HTTPS_PROXY = `0.0.0.0`;
|
|
|
|
await expect(runCli(installDir, [`yarn`, `--version`])).resolves.toMatchObject({
|
|
stdout: `2.2.2\n`,
|
|
stderr: ``,
|
|
exitCode: 0,
|
|
});
|
|
});
|
|
});
|
|
});
|
|
|
|
it(`should support hydrating package managers from cached archives`, async () => {
|
|
await xfs.mktempPromise(async cwd => {
|
|
await expect(runCli(cwd, [`pack`, `yarn@2.2.2`])).resolves.toMatchObject({
|
|
stderr: ``,
|
|
exitCode: 0,
|
|
});
|
|
|
|
// Use a new cache
|
|
process.env.COREPACK_HOME = npath.fromPortablePath(await xfs.mktempPromise());
|
|
|
|
// Disable the network to make sure we don't succeed by accident
|
|
process.env.COREPACK_ENABLE_NETWORK = `0`;
|
|
|
|
await expect(runCli(cwd, [`install`, `-g`, `corepack.tgz`])).resolves.toMatchObject({
|
|
stderr: ``,
|
|
exitCode: 0,
|
|
});
|
|
|
|
await xfs.writeJsonPromise(ppath.join(cwd, `package.json` as Filename), {
|
|
packageManager: `yarn@2.2.2`,
|
|
});
|
|
|
|
await expect(runCli(cwd, [`yarn`, `--version`])).resolves.toMatchObject({
|
|
stdout: `2.2.2\n`,
|
|
stderr: ``,
|
|
exitCode: 0,
|
|
});
|
|
});
|
|
});
|
|
|
|
it(`should support hydrating package managers if cache folder was removed`, async () => {
|
|
await xfs.mktempPromise(async cwd => {
|
|
await expect(runCli(cwd, [`pack`, `yarn@2.2.2`])).resolves.toMatchObject({
|
|
exitCode: 0,
|
|
stderr: ``,
|
|
});
|
|
|
|
// Use a new cache
|
|
process.env.COREPACK_HOME = npath.fromPortablePath(await xfs.mktempPromise());
|
|
|
|
// Simulate cache removal
|
|
await xfs.removePromise(npath.toPortablePath(process.env.COREPACK_HOME));
|
|
|
|
// Disable the network to make sure we don't succeed by accident
|
|
process.env.COREPACK_ENABLE_NETWORK = `0`;
|
|
|
|
await expect(runCli(cwd, [`install`, `-g`, `corepack.tgz`])).resolves.toMatchObject({
|
|
stderr: ``,
|
|
exitCode: 0,
|
|
});
|
|
|
|
await xfs.writeJsonPromise(ppath.join(cwd, `package.json` as Filename), {
|
|
packageManager: `yarn@2.2.2`,
|
|
});
|
|
|
|
await expect(runCli(cwd, [`yarn`, `--version`])).resolves.toMatchObject({
|
|
stdout: `2.2.2\n`,
|
|
stderr: ``,
|
|
exitCode: 0,
|
|
});
|
|
});
|
|
});
|
|
|
|
it(`should support hydrating multiple package managers from cached archives`, async () => {
|
|
await xfs.mktempPromise(async cwd => {
|
|
await expect(runCli(cwd, [`pack`, `yarn@2.2.2`, `pnpm@5.8.0`])).resolves.toMatchObject({
|
|
exitCode: 0,
|
|
stderr: ``,
|
|
});
|
|
|
|
// Use a new cache
|
|
process.env.COREPACK_HOME = npath.fromPortablePath(await xfs.mktempPromise());
|
|
|
|
// Disable the network to make sure we don't succeed by accident
|
|
process.env.COREPACK_ENABLE_NETWORK = `0`;
|
|
|
|
await expect(runCli(cwd, [`install`, `-g`, `corepack.tgz`])).resolves.toMatchObject({
|
|
stderr: ``,
|
|
exitCode: 0,
|
|
});
|
|
|
|
await xfs.writeJsonPromise(ppath.join(cwd, `package.json` as Filename), {
|
|
packageManager: `yarn@2.2.2`,
|
|
});
|
|
|
|
await expect(runCli(cwd, [`yarn`, `--version`])).resolves.toMatchObject({
|
|
stdout: `2.2.2\n`,
|
|
stderr: ``,
|
|
exitCode: 0,
|
|
});
|
|
|
|
await xfs.writeJsonPromise(ppath.join(cwd, `package.json` as Filename), {
|
|
packageManager: `pnpm@5.8.0`,
|
|
});
|
|
|
|
await expect(runCli(cwd, [`pnpm`, `--version`])).resolves.toMatchObject({
|
|
stdout: `5.8.0\n`,
|
|
stderr: ``,
|
|
exitCode: 0,
|
|
});
|
|
});
|
|
}, 180_000);
|
|
|
|
it(`should support running package managers with bin array`, async () => {
|
|
await xfs.mktempPromise(async cwd => {
|
|
await xfs.writeJsonPromise(ppath.join(cwd, `package.json` as Filename), {
|
|
packageManager: `yarn@2.2.2`,
|
|
});
|
|
|
|
await expect(runCli(cwd, [`yarnpkg`, `--version`])).resolves.toMatchObject({
|
|
stdout: `2.2.2\n`,
|
|
stderr: ``,
|
|
exitCode: 0,
|
|
});
|
|
|
|
await expect(runCli(cwd, [`yarn`, `--version`])).resolves.toMatchObject({
|
|
stdout: `2.2.2\n`,
|
|
stderr: ``,
|
|
exitCode: 0,
|
|
});
|
|
});
|
|
});
|
|
|
|
it(`should handle parallel installs`, async () => {
|
|
await xfs.mktempPromise(async cwd => {
|
|
await xfs.writeJsonPromise(ppath.join(cwd, `package.json` as Filename), {
|
|
packageManager: `yarn@2.2.2`,
|
|
});
|
|
|
|
await expect(Promise.all([
|
|
runCli(cwd, [`yarn`, `--version`]),
|
|
runCli(cwd, [`yarn`, `--version`]),
|
|
runCli(cwd, [`yarn`, `--version`]),
|
|
])).resolves.toMatchObject([
|
|
{
|
|
stdout: `2.2.2\n`,
|
|
stderr: ``,
|
|
exitCode: 0,
|
|
},
|
|
{
|
|
stdout: `2.2.2\n`,
|
|
stderr: ``,
|
|
exitCode: 0,
|
|
},
|
|
{
|
|
stdout: `2.2.2\n`,
|
|
stderr: ``,
|
|
exitCode: 0,
|
|
},
|
|
]);
|
|
});
|
|
});
|
|
|
|
it(`should not override the package manager exit code`, async () => {
|
|
await xfs.mktempPromise(async cwd => {
|
|
await xfs.writeJsonPromise(ppath.join(cwd, `package.json` as Filename), {
|
|
packageManager: `yarn@2.2.2`,
|
|
});
|
|
|
|
const yarnFolder = ppath.join(npath.toPortablePath(folderUtils.getInstallFolder()), `yarn/2.2.2`);
|
|
await xfs.mkdirPromise(yarnFolder, {recursive: true});
|
|
await xfs.writeJsonPromise(ppath.join(yarnFolder, `.corepack`), {});
|
|
|
|
await xfs.writeFilePromise(ppath.join(yarnFolder, `yarn.js`), `
|
|
process.exitCode = 42;
|
|
`);
|
|
|
|
await expect(runCli(cwd, [`yarn`, `--version`])).resolves.toMatchObject({
|
|
exitCode: 42,
|
|
stdout: ``,
|
|
stderr: ``,
|
|
});
|
|
});
|
|
});
|
|
|
|
it(`should not preserve the process.exitCode when a package manager throws`, async () => {
|
|
// Node.js doesn't preserve process.exitCode when an exception is thrown
|
|
// so we need to make sure we don't break this behaviour.
|
|
|
|
await xfs.mktempPromise(async cwd => {
|
|
await xfs.writeJsonPromise(ppath.join(cwd, `package.json` as Filename), {
|
|
packageManager: `yarn@2.2.2`,
|
|
});
|
|
|
|
const yarnFolder = ppath.join(npath.toPortablePath(folderUtils.getInstallFolder()), `yarn/2.2.2`);
|
|
await xfs.mkdirPromise(yarnFolder, {recursive: true});
|
|
await xfs.writeJsonPromise(ppath.join(yarnFolder, `.corepack`), {});
|
|
|
|
await xfs.writeFilePromise(ppath.join(yarnFolder, `yarn.js`), `
|
|
process.exitCode = 42;
|
|
throw new Error('foo');
|
|
`);
|
|
|
|
await expect(runCli(cwd, [`yarn`, `--version`])).resolves.toMatchObject({
|
|
exitCode: 1,
|
|
stdout: ``,
|
|
stderr: expect.stringContaining(`foo`),
|
|
});
|
|
});
|
|
});
|
|
|
|
it(`should not set the exit code after successfully launching the package manager`, async () => {
|
|
await xfs.mktempPromise(async cwd => {
|
|
await xfs.writeJsonPromise(ppath.join(cwd, `package.json` as Filename), {
|
|
packageManager: `yarn@2.2.2`,
|
|
});
|
|
|
|
const yarnFolder = ppath.join(npath.toPortablePath(folderUtils.getInstallFolder()), `yarn/2.2.2`);
|
|
await xfs.mkdirPromise(yarnFolder, {recursive: true});
|
|
await xfs.writeJsonPromise(ppath.join(yarnFolder, `.corepack`), {});
|
|
|
|
await xfs.writeFilePromise(ppath.join(yarnFolder, `yarn.js`), `
|
|
process.once('beforeExit', () => {
|
|
if (process.exitCode === undefined) {
|
|
process.exitCode = 42;
|
|
}
|
|
});
|
|
`);
|
|
|
|
await expect(runCli(cwd, [`yarn`, `--version`])).resolves.toMatchObject({
|
|
exitCode: 42,
|
|
stdout: ``,
|
|
stderr: ``,
|
|
});
|
|
});
|
|
});
|
|
|
|
it(`should support package managers in ESM format`, async () => {
|
|
await xfs.mktempPromise(async cwd => {
|
|
await xfs.writeJsonPromise(ppath.join(cwd, `package.json` as Filename), {
|
|
packageManager: `yarn@2.2.2`,
|
|
});
|
|
|
|
const yarnFolder = ppath.join(npath.toPortablePath(folderUtils.getInstallFolder()), `yarn/2.2.2`);
|
|
await xfs.mkdirPromise(yarnFolder, {recursive: true});
|
|
await xfs.writeJsonPromise(ppath.join(yarnFolder, `.corepack`), {});
|
|
|
|
await xfs.writeFilePromise(ppath.join(yarnFolder, `yarn.js`), `
|
|
import 'fs';
|
|
console.log(42);
|
|
`);
|
|
|
|
await xfs.writeJsonPromise(ppath.join(yarnFolder, `package.json`), {
|
|
type: `module`,
|
|
});
|
|
|
|
await expect(runCli(cwd, [`yarn`, `--version`])).resolves.toMatchObject({
|
|
exitCode: 0,
|
|
stdout: `42\n`,
|
|
stderr: ``,
|
|
});
|
|
});
|
|
});
|
|
|
|
it(`should show a warning on stderr before downloading when enable`, async() => {
|
|
await xfs.mktempPromise(async cwd => {
|
|
process.env.COREPACK_ENABLE_DOWNLOAD_PROMPT = `1`;
|
|
await xfs.writeJsonPromise(ppath.join(cwd, `package.json` as Filename), {
|
|
packageManager: `yarn@3.0.0`,
|
|
});
|
|
await expect(runCli(cwd, [`yarn`, `--version`])).resolves.toMatchObject({
|
|
exitCode: 0,
|
|
stdout: `3.0.0\n`,
|
|
stderr: `! Corepack is about to download https://repo.yarnpkg.com/3.0.0/packages/yarnpkg-cli/bin/yarn.js\n`,
|
|
});
|
|
});
|
|
});
|
|
|
|
it(`should be able to show the latest version`, async () => {
|
|
process.env.COREPACK_DEFAULT_TO_LATEST = `1`;
|
|
await xfs.mktempPromise(async cwd => {
|
|
await expect(runCli(cwd, [`yarn`, `--version`])).resolves.toMatchObject({
|
|
exitCode: 0,
|
|
stdout: /^1\.\d+\.\d+\r?\n$/,
|
|
stderr: ``,
|
|
});
|
|
|
|
// Should keep working if the home folder is removed
|
|
await xfs.rmdirPromise(process.env.COREPACK_HOME as any, {recursive: true});
|
|
await expect(runCli(cwd, [`yarn`, `--version`])).resolves.toMatchObject({
|
|
exitCode: 0,
|
|
stdout: /^1\.\d+\.\d+\r?\n$/,
|
|
stderr: ``,
|
|
});
|
|
});
|
|
});
|
|
|
|
it(`should download yarn classic from custom registry`, async () => {
|
|
await xfs.mktempPromise(async cwd => {
|
|
process.env.COREPACK_NPM_REGISTRY = `https://registry.npmmirror.com`;
|
|
process.env.COREPACK_ENABLE_DOWNLOAD_PROMPT = `1`;
|
|
await expect(runCli(cwd, [`yarn`, `--version`])).resolves.toMatchObject({
|
|
exitCode: 0,
|
|
stdout: /^1\.\d+\.\d+\r?\n$/,
|
|
stderr: /^! Corepack is about to download https:\/\/registry\.npmmirror\.com\/yarn\/-\/yarn-1\.\d+\.\d+\.tgz\r?\n$/,
|
|
});
|
|
|
|
// Should keep working with cache
|
|
await expect(runCli(cwd, [`yarn`, `--version`])).resolves.toMatchObject({
|
|
exitCode: 0,
|
|
stdout: /^1\.\d+\.\d+\r?\n$/,
|
|
stderr: ``,
|
|
});
|
|
});
|
|
});
|
|
|
|
it(`should download yarn berry from custom registry`, async () => {
|
|
await xfs.mktempPromise(async cwd => {
|
|
process.env.COREPACK_NPM_REGISTRY = `https://registry.npmmirror.com`;
|
|
process.env.COREPACK_ENABLE_DOWNLOAD_PROMPT = `1`;
|
|
|
|
await xfs.writeJsonPromise(ppath.join(cwd, `package.json` as Filename), {
|
|
packageManager: `yarn@3.0.0-rc.2+sha224.f83f6d1cbfac10ba6b516a62ccd2a72ccd857aa6c514d1cd7185ec60`,
|
|
});
|
|
|
|
await expect(runCli(cwd, [`yarn`, `--version`])).resolves.toMatchObject({
|
|
exitCode: 0,
|
|
stdout: `3.0.0-rc.2\n`,
|
|
stderr: `! Corepack is about to download https://registry.npmmirror.com/@yarnpkg/cli-dist/-/cli-dist-3.0.0-rc.2.tgz\n`,
|
|
});
|
|
|
|
// Should keep working with cache
|
|
await expect(runCli(cwd, [`yarn`, `--version`])).resolves.toMatchObject({
|
|
exitCode: 0,
|
|
stdout: `3.0.0-rc.2\n`,
|
|
stderr: ``,
|
|
});
|
|
});
|
|
});
|
|
|
|
for (const authType of [`COREPACK_NPM_REGISTRY`, `COREPACK_NPM_TOKEN`, `COREPACK_NPM_PASSWORD`, `PROXY`]) {
|
|
describe(`custom registry with auth ${authType}`, () => {
|
|
beforeEach(() => {
|
|
process.env.AUTH_TYPE = authType; // See `_registryServer.mjs`
|
|
process.env.COREPACK_INTEGRITY_KEYS = ``;
|
|
});
|
|
|
|
it(`should download yarn classic`, async () => {
|
|
await xfs.mktempPromise(async cwd => {
|
|
await expect(runCli(cwd, [`yarn@1.x`, `--version`], true)).resolves.toMatchObject({
|
|
exitCode: 0,
|
|
stdout: `yarn: Hello from custom registry\n`,
|
|
stderr: ``,
|
|
});
|
|
});
|
|
});
|
|
|
|
it(`should download yarn berry`, async () => {
|
|
await xfs.mktempPromise(async cwd => {
|
|
await xfs.writeJsonPromise(ppath.join(cwd, `package.json` as Filename), {
|
|
packageManager: `yarn@3.0.0`,
|
|
});
|
|
|
|
await expect(runCli(cwd, [`yarn@5.x`, `--version`], true)).resolves.toMatchObject({
|
|
exitCode: 0,
|
|
stdout: `yarn: Hello from custom registry\n`,
|
|
stderr: ``,
|
|
});
|
|
});
|
|
});
|
|
|
|
it(`should download pnpm`, async () => {
|
|
await xfs.mktempPromise(async cwd => {
|
|
await expect(runCli(cwd, [`pnpm@1.x`, `--version`], true)).resolves.toMatchObject({
|
|
exitCode: 0,
|
|
stdout: `pnpm: Hello from custom registry\n`,
|
|
stderr: ``,
|
|
});
|
|
});
|
|
});
|
|
|
|
it(`should download custom package manager`, async () => {
|
|
await xfs.mktempPromise(async cwd => {
|
|
await expect(runCli(cwd, [`customPkgManager@https://registry.npmjs.org/customPkgManager/-/customPkgManager-1.0.0.tgz`, `--version`], true)).resolves.toMatchObject({
|
|
exitCode: 0,
|
|
stdout: `customPkgManager: Hello from custom registry\n`,
|
|
stderr: ``,
|
|
});
|
|
});
|
|
});
|
|
});
|
|
}
|
|
|
|
describe(`handle integrity checks`, () => {
|
|
beforeEach(() => {
|
|
process.env.AUTH_TYPE = `COREPACK_NPM_TOKEN`; // See `_registryServer.mjs`
|
|
process.env.COREPACK_DEFAULT_TO_LATEST = `1`;
|
|
});
|
|
|
|
it(`should return no error when signature matches`, async () => {
|
|
process.env.TEST_INTEGRITY = `valid`; // See `_registryServer.mjs`
|
|
|
|
await xfs.mktempPromise(async cwd => {
|
|
await Promise.all([
|
|
expect(runCli(cwd, [`pnpm`, `--version`], true)).resolves.toMatchObject({
|
|
exitCode: 0,
|
|
stdout: `pnpm: Hello from custom registry\n`,
|
|
stderr: ``,
|
|
}),
|
|
expect(runCli(cwd, [`yarn@1.x`, `--version`], true)).resolves.toMatchObject({
|
|
exitCode: 0,
|
|
stdout: `yarn: Hello from custom registry\n`,
|
|
stderr: ``,
|
|
}),
|
|
expect(runCli(cwd, [`yarn@5.x`, `--version`], true)).resolves.toMatchObject({
|
|
exitCode: 0,
|
|
stdout: `yarn: Hello from custom registry\n`,
|
|
stderr: ``,
|
|
}),
|
|
]);
|
|
|
|
// Skip rest of the test on Windows & Node.js 18.x as it inevitably times out otherwise.
|
|
if (process.version.startsWith(`v18.`) && os.platform() === `win32`) return;
|
|
|
|
// Removing home directory to force the "re-download"
|
|
await xfs.rmdirPromise(process.env.COREPACK_HOME as any, {recursive: true});
|
|
|
|
await Promise.all([
|
|
expect(runCli(cwd, [`use`, `pnpm`], true)).resolves.toMatchObject({
|
|
exitCode: 0,
|
|
stdout: `Installing pnpm@1.9998.9999 in the project...\n\npnpm: Hello from custom registry\n`,
|
|
stderr: ``,
|
|
}),
|
|
expect(runCli(cwd, [`use`, `yarn@1.x`], true)).resolves.toMatchObject({
|
|
exitCode: 0,
|
|
stdout: `Installing yarn@1.9998.9999 in the project...\n\nyarn: Hello from custom registry\n`,
|
|
stderr: ``,
|
|
}),
|
|
expect(runCli(cwd, [`use`, `yarn@latest`], true)).resolves.toMatchObject({
|
|
exitCode: 0,
|
|
stdout: `Installing yarn@5.9999.9999 in the project...\n\nyarn: Hello from custom registry\n`,
|
|
stderr: ``,
|
|
}),
|
|
]);
|
|
});
|
|
});
|
|
it(`should return an error when signature does not match with a tag`, async () => {
|
|
process.env.TEST_INTEGRITY = `invalid_signature`; // See `_registryServer.mjs`
|
|
|
|
await xfs.mktempPromise(async cwd => {
|
|
await expect(runCli(cwd, [`pnpm@1.x`, `--version`], true)).resolves.toMatchObject({
|
|
exitCode: 1,
|
|
stdout: /Signature does not match/,
|
|
stderr: ``,
|
|
});
|
|
await expect(runCli(cwd, [`yarn@stable`, `--version`], true)).resolves.toMatchObject({
|
|
exitCode: 1,
|
|
stdout: /Signature does not match/,
|
|
stderr: ``,
|
|
});
|
|
});
|
|
});
|
|
it(`should return an error when hash does not match without a tag`, async () => {
|
|
process.env.TEST_INTEGRITY = `invalid_integrity`; // See `_registryServer.mjs`
|
|
|
|
await xfs.mktempPromise(async cwd => {
|
|
await expect(runCli(cwd, [`pnpm`, `--version`], true)).resolves.toMatchObject({
|
|
exitCode: 1,
|
|
stdout: /Mismatch hashes. Expected [a-f0-9]{128}, got [a-f0-9]{128}/,
|
|
stderr: ``,
|
|
});
|
|
// A second time to validate the invalid version was not cached.
|
|
await expect(runCli(cwd, [`pnpm`, `--version`], true)).resolves.toMatchObject({
|
|
exitCode: 1,
|
|
stdout: /Mismatch hashes. Expected [a-f0-9]{128}, got [a-f0-9]{128}/,
|
|
stderr: ``,
|
|
});
|
|
await expect(runCli(cwd, [`yarn`, `--version`], true)).resolves.toMatchObject({
|
|
exitCode: 1,
|
|
stdout: /Mismatch hashes. Expected [a-f0-9]{128}, got [a-f0-9]{128}/,
|
|
stderr: ``,
|
|
});
|
|
await expect(runCli(cwd, [`use`, `pnpm`], true)).resolves.toMatchObject({
|
|
exitCode: 1,
|
|
stdout: /Mismatch hashes. Expected [a-f0-9]{128}, got [a-f0-9]{128}/,
|
|
stderr: ``,
|
|
});
|
|
});
|
|
});
|
|
it(`should return an error when signature does not match without a tag`, async () => {
|
|
process.env.TEST_INTEGRITY = `invalid_signature`; // See `_registryServer.mjs`
|
|
|
|
await xfs.mktempPromise(async cwd => {
|
|
await expect(runCli(cwd, [`pnpm`, `--version`], true)).resolves.toMatchObject({
|
|
exitCode: 1,
|
|
stdout: /Signature does not match/,
|
|
stderr: ``,
|
|
});
|
|
// A second time to validate the invalid version was not cached.
|
|
await expect(runCli(cwd, [`pnpm`, `--version`], true)).resolves.toMatchObject({
|
|
exitCode: 1,
|
|
stdout: /Signature does not match/,
|
|
stderr: ``,
|
|
});
|
|
await expect(runCli(cwd, [`yarn`, `--version`], true)).resolves.toMatchObject({
|
|
exitCode: 1,
|
|
stdout: /Signature does not match/,
|
|
stderr: ``,
|
|
});
|
|
await expect(runCli(cwd, [`use`, `pnpm`], true)).resolves.toMatchObject({
|
|
exitCode: 1,
|
|
stdout: /Signature does not match/,
|
|
stderr: ``,
|
|
});
|
|
});
|
|
});
|
|
it(`should return an error when signature does not match when version is provided`, async () => {
|
|
process.env.TEST_INTEGRITY = `invalid_signature`; // See `_registryServer.mjs`
|
|
|
|
await xfs.mktempPromise(async cwd => {
|
|
await expect(runCli(cwd, [`yarn@1.9998.9999`, `--version`], true)).resolves.toMatchObject({
|
|
exitCode: 1,
|
|
stdout: /Signature does not match/,
|
|
stderr: ``,
|
|
});
|
|
await expect(runCli(cwd, [`use`, `yarn@1.9998.9999`], true)).resolves.toMatchObject({
|
|
exitCode: 1,
|
|
stdout: /Signature does not match/,
|
|
stderr: ``,
|
|
});
|
|
});
|
|
});
|
|
it(`should return an error when hash does not match`, async () => {
|
|
process.env.TEST_INTEGRITY = `invalid_integrity`; // See `_registryServer.mjs`
|
|
|
|
await xfs.mktempPromise(async cwd => {
|
|
await expect(runCli(cwd, [`yarn@1.9998.9999`, `--version`], true)).resolves.toMatchObject({
|
|
exitCode: 1,
|
|
stdout: /Mismatch hashes. Expected [a-f0-9]{128}, got [a-f0-9]{128}/,
|
|
stderr: ``,
|
|
});
|
|
await expect(runCli(cwd, [`use`, `yarn@1.9998.9999`], true)).resolves.toMatchObject({
|
|
exitCode: 1,
|
|
stdout: /Mismatch hashes. Expected [a-f0-9]{128}, got [a-f0-9]{128}/,
|
|
stderr: ``,
|
|
});
|
|
});
|
|
});
|
|
it(`should return no error when signature does not match when hash is provided`, async () => {
|
|
process.env.TEST_INTEGRITY = `invalid_signature`; // See `_registryServer.mjs`
|
|
|
|
await xfs.mktempPromise(async cwd => {
|
|
const result = await runCli(cwd, [`yarn@1.9998.9999+sha1.deadbeef`, `--version`], true);
|
|
expect(result).toMatchObject({
|
|
exitCode: 1,
|
|
stderr: ``,
|
|
});
|
|
const match = /Mismatch hashes. Expected deadbeef, got ([a-f0-9]{40})/.exec(result.stdout);
|
|
if (match == null) throw new Error(`Invalid output`, {cause: result.stdout});
|
|
await expect(runCli(cwd, [`yarn@1.9998.9999+sha1.${match[1]}`, `--version`], true)).resolves.toMatchObject({
|
|
exitCode: 0,
|
|
stdout: `yarn: Hello from custom registry\n`,
|
|
stderr: ``,
|
|
});
|
|
});
|
|
});
|
|
});
|