From 473164109b1205f21ba8a9832f5137059547da35 Mon Sep 17 00:00:00 2001 From: Simon Vocella Date: Tue, 2 May 2017 11:30:20 +0200 Subject: [PATCH] Add tests for global command (#3238) * add test for global bin command * add test for global add command * add test for global remove * add test for global ls command and fix issue #3142 * add test for global upgrade * move tests with side-effects under isCI branch; the other tests now are side-effects free * the global folder in win32 platform is prefix env/parameter without bin folder inside, do assert only on global folders for yarn global v0.23.3 Done in 0.26s. --- __tests__/commands/_helpers.js | 3 +- __tests__/commands/global.js | 92 +++++++++++++++--- .../-/react-native-cli-2.0.0.tgz.bin | Bin 0 -> 6602 bytes src/cli/commands/global.js | 4 +- src/cli/index.js | 4 +- 5 files changed, 84 insertions(+), 19 deletions(-) create mode 100644 __tests__/fixtures/request-cache/GET/registry.yarnpkg.com/react-native-cli/-/react-native-cli-2.0.0.tgz.bin diff --git a/__tests__/commands/_helpers.js b/__tests__/commands/_helpers.js index 1a1caacb50..cb52648172 100644 --- a/__tests__/commands/_helpers.js +++ b/__tests__/commands/_helpers.js @@ -122,9 +122,10 @@ export async function run( const config = await Config.create({ binLinks: !!flags.binLinks, cwd, - globalFolder: path.join(cwd, '.yarn-global'), + globalFolder: flags.globalFolder || path.join(cwd, '.yarn-global'), cacheFolder: flags.cacheFolder || path.join(cwd, '.yarn-cache'), linkFolder: flags.linkFolder || path.join(cwd, '.yarn-link'), + prefix: flags.prefix, production: flags.production, }, reporter); diff --git a/__tests__/commands/global.js b/__tests__/commands/global.js index 624cc0a57c..669f867896 100644 --- a/__tests__/commands/global.js +++ b/__tests__/commands/global.js @@ -5,6 +5,7 @@ import {ConsoleReporter} from '../../src/reporters/index.js'; import {run as buildRun} from './_helpers.js'; import {run as global} from '../../src/cli/commands/global.js'; import * as fs from '../../src/util/fs.js'; +import mkdir from '../_temp.js'; const isCI = require('is-ci'); jasmine.DEFAULT_TIMEOUT_INTERVAL = 90000; @@ -34,7 +35,16 @@ function getTempGlobalFolder(): string { return path.join(os.tmpdir(), `yarn-global-${Math.random()}`); } -// this test has global folder side effects, run it only in CI +async function createTempGlobalFolder(): Promise { + return await mkdir('yarn-global'); +} + +async function createTempPrefixFolder(): Promise { + const prefixFolder = await mkdir('yarn-prefix'); + return path.join(prefixFolder, 'bin'); +} + +// these tests have global folder side or prefix folder effects, run it only in CI if (isCI) { test.concurrent('add without flag', (): Promise => { return runGlobal(['add', 'react-native-cli'], {}, 'add-without-flag', async (config) => { @@ -42,23 +52,79 @@ if (isCI) { expect(await fs.exists(path.join(config.globalFolder, 'node_modules', '.bin', 'react-native'))).toEqual(true); }); }); + + test.concurrent('add with prefix flag', (): Promise => { + const tmpGlobalFolder = getTempGlobalFolder(); + return runGlobal(['add', 'react-native-cli'], {prefix: tmpGlobalFolder}, 'add-with-prefix-flag', async (config) => { + expect(await fs.exists(getGlobalPath(tmpGlobalFolder, 'react-native'))).toEqual(true); + }); + }); + + // don't run this test in `concurrent`, it will affect other tests + test('add with PREFIX enviroment variable', (): Promise => { + const tmpGlobalFolder = getTempGlobalFolder(); + const envPrefix = process.env.PREFIX; + process.env.PREFIX = tmpGlobalFolder; + return runGlobal(['add', 'react-native-cli'], {}, 'add-with-prefix-env', async (config) => { + expect(await fs.exists(getGlobalPath(tmpGlobalFolder, 'react-native'))).toEqual(true); + // restore env + process.env.PREFIX = envPrefix; + }); + }); } -test.concurrent('add with prefix flag', (): Promise => { +test.concurrent('bin', (): Promise => { const tmpGlobalFolder = getTempGlobalFolder(); - return runGlobal(['add', 'react-native-cli'], {prefix: tmpGlobalFolder}, 'add-with-prefix-flag', async (config) => { - expect(await fs.exists(getGlobalPath(tmpGlobalFolder, 'react-native'))).toEqual(true); + return runGlobal(['bin'], {prefix: tmpGlobalFolder}, 'add-with-prefix-flag', + (config, reporter, install, getStdout) => { + expect(getStdout()).toContain(tmpGlobalFolder); }); }); -// don't run this test in `concurrent`, it will affect other tests -test('add with PREFIX enviroment variable', (): Promise => { - const tmpGlobalFolder = getTempGlobalFolder(); - const envPrefix = process.env.PREFIX; - process.env.PREFIX = tmpGlobalFolder; - return runGlobal(['add', 'react-native-cli'], {}, 'add-with-prefix-env', async (config) => { - expect(await fs.exists(getGlobalPath(tmpGlobalFolder, 'react-native'))).toEqual(true); - // restore env - process.env.PREFIX = envPrefix; +test.concurrent('add', async (): Promise => { + const tmpGlobalFolder = await createTempGlobalFolder(); + const tmpPrefixFolder = await createTempPrefixFolder(); + const flags = {globalFolder: tmpGlobalFolder, prefix: tmpPrefixFolder}; + return runGlobal(['add', 'react-native-cli'], flags, 'add-with-prefix-flag', + async (config) => { + expect(await fs.exists(path.join(tmpGlobalFolder, 'node_modules', 'react-native-cli'))).toEqual(true); + }); +}); + +test.concurrent('remove', async (): Promise => { + const tmpGlobalFolder = await createTempGlobalFolder(); + const tmpPrefixFolder = await createTempPrefixFolder(); + const flags = {globalFolder: tmpGlobalFolder, prefix: tmpPrefixFolder}; + return runGlobal(['add', 'react-native-cli'], flags, 'add-with-prefix-flag', () => {}) + .then(() => { + return runGlobal(['remove', 'react-native-cli'], flags, 'add-with-prefix-flag', async (config) => { + expect(await fs.exists(path.join(tmpGlobalFolder, 'node_modules', 'react-native-cli'))).toEqual(false); + }); + }); +}); + +test.concurrent('ls', async (): Promise => { + const tmpGlobalFolder = await createTempGlobalFolder(); + const tmpPrefixFolder = await createTempPrefixFolder(); + const flags = {globalFolder: tmpGlobalFolder, prefix: tmpPrefixFolder}; + return runGlobal(['add', 'react-native-cli'], flags, 'add-with-prefix-flag', () => {}) + .then(() => { + return runGlobal(['ls'], flags, 'add-with-prefix-flag', (config, reporter, install, getStdout) => { + expect(getStdout()).toContain('react-native-cli'); + }); + }); +}); + +test.concurrent('upgrade', async (): Promise => { + const tmpGlobalFolder = await createTempGlobalFolder(); + const tmpPrefixFolder = await createTempPrefixFolder(); + const flags = {globalFolder: tmpGlobalFolder, prefix: tmpPrefixFolder}; + return runGlobal(['add', 'react-native-cli@2.0.0'], flags, 'add-with-prefix-flag', () => {}) + .then(() => { + return runGlobal(['upgrade', 'react-native-cli'], flags, 'add-with-prefix-flag', + (config, reporter, install, getStdout) => { + expect(getStdout()).toContain('react-native-cli'); + expect(getStdout()).not.toContain('react-native-cli@2.0.0'); + }); }); }); diff --git a/__tests__/fixtures/request-cache/GET/registry.yarnpkg.com/react-native-cli/-/react-native-cli-2.0.0.tgz.bin b/__tests__/fixtures/request-cache/GET/registry.yarnpkg.com/react-native-cli/-/react-native-cli-2.0.0.tgz.bin new file mode 100644 index 0000000000000000000000000000000000000000..c2beaf817c9fe84b755ae4c02695c5030e4f5316 GIT binary patch literal 6602 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 Y&J|v8C{gGCI=Dcor8AV`Sm5CN7uQNP!2kdN literal 0 HcmV?d00001 diff --git a/src/cli/commands/global.js b/src/cli/commands/global.js index 05d4e6e980..7697d239bb 100644 --- a/src/cli/commands/global.js +++ b/src/cli/commands/global.js @@ -201,7 +201,7 @@ const {run, setFlags: _setFlags} = buildSubCommands('global', { flags: Object, args: Array, ) { - console.log(getBinFolder(config, flags)); + reporter.log(getBinFolder(config, flags)); }, async ls( @@ -214,7 +214,7 @@ const {run, setFlags: _setFlags} = buildSubCommands('global', { // install so we get hard file paths const lockfile = await Lockfile.fromDirectory(config.cwd); - const install = new Install({skipIntegrity: true}, config, new NoopReporter(), lockfile); + const install = new Install({skipIntegrityCheck: true}, config, new NoopReporter(), lockfile); const patterns = await install.init(); // dump global modules diff --git a/src/cli/index.js b/src/cli/index.js index d56fd91ffb..f5c5ce9484 100644 --- a/src/cli/index.js +++ b/src/cli/index.js @@ -51,7 +51,7 @@ commander.option('--no-lockfile', "don't read or generate a lockfile"); commander.option('--pure-lockfile', "don't generate a lockfile"); commander.option('--frozen-lockfile', "don't generate a lockfile and fail if an update is needed"); commander.option('--link-duplicates', 'create hardlinks to the repeated modules in node_modules'); -commander.option('--global-folder ', ''); +commander.option('--global-folder ', 'specify a custom folder to store global packages'); commander.option( '--modules-folder ', 'rather than installing modules into the node_modules folder relative to the cwd, output them here', @@ -299,7 +299,6 @@ function writeErrorReport(log) : ?string { return errorReportLoc; } -// config.init({ binLinks: commander.binLinks, modulesFolder: commander.modulesFolder, @@ -319,7 +318,6 @@ config.init({ nonInteractive: commander.nonInteractive, commandName: commandName === 'run' ? commander.args[0] : commandName, }).then(() => { - // option "no-progress" stored in yarn config const noProgressConfig = config.registries.yarn.getOption('no-progress');