diff --git a/.eslintignore b/.eslintignore index f22775658380..45d0e83020f6 100644 --- a/.eslintignore +++ b/.eslintignore @@ -9,6 +9,10 @@ **/node_modules **/support/fixtures/* !**/support/fixtures/projects +**/support/fixtures/projects/**/_fixtures/* +**/support/fixtures/projects/**/*.jsx +**/support/fixtures/projects/**/jquery.js +**/support/fixtures/projects/**/fail.js **/test/fixtures **/vendor diff --git a/cli/__snapshots__/cli_spec.js b/cli/__snapshots__/cli_spec.js index d471f77b5da0..5465cc92f7d9 100644 --- a/cli/__snapshots__/cli_spec.js +++ b/cli/__snapshots__/cli_spec.js @@ -354,3 +354,32 @@ exports['shows help for run --foo 1'] = ` ------- ` + +exports['cli CYPRESS_ENV allows staging environment 1'] = ` + code: 0 + stderr: + ------- + + ------- + +` + +exports['cli CYPRESS_ENV catches environment "foo" 1'] = ` + code: 11 + stderr: + ------- + The environment variable with the reserved name "CYPRESS_ENV" is set. + + Unset the "CYPRESS_ENV" environment variable and run Cypress again. + + ---------- + + CYPRESS_ENV=foo + + ---------- + + Platform: xxx + Cypress Version: 1.2.3 + ------- + +` diff --git a/cli/__snapshots__/errors_spec.js b/cli/__snapshots__/errors_spec.js index f8333d37acba..44b1a631d786 100644 --- a/cli/__snapshots__/errors_spec.js +++ b/cli/__snapshots__/errors_spec.js @@ -32,6 +32,7 @@ exports['errors individual has the following errors 1'] = [ "failedDownload", "failedUnzip", "invalidCacheDirectory", + "invalidCypressEnv", "invalidSmokeTestDisplayError", "missingApp", "missingDependency", diff --git a/cli/index.js b/cli/index.js index 7da84f9ec87b..250d3d4b52b1 100644 --- a/cli/index.js +++ b/cli/index.js @@ -23,6 +23,6 @@ switch (args.exec) { break default: - // export our node module interface + debug('exporting Cypress module interface') module.exports = require('./lib/cypress') } diff --git a/cli/lib/cli.js b/cli/lib/cli.js index dc6a6e82e7c8..fb94e25eebc6 100644 --- a/cli/lib/cli.js +++ b/cli/lib/cli.js @@ -5,6 +5,7 @@ const logSymbols = require('log-symbols') const debug = require('debug')('cypress:cli') const util = require('./util') const logger = require('./logger') +const errors = require('./errors') const cache = require('./tasks/cache') // patch "commander" method called when a user passed an unknown option @@ -27,7 +28,9 @@ const coerceFalse = (arg) => { const spaceDelimitedSpecsMsg = (files) => { logger.log() logger.warn(stripIndent` - ${logSymbols.warning} Warning: It looks like you're passing --spec a space-separated list of files: + ${ + logSymbols.warning +} Warning: It looks like you're passing --spec a space-separated list of files: "${files.join(' ')}" @@ -54,7 +57,8 @@ const parseVariableOpts = (fnArgs, args) => { const nextOptOffset = _.findIndex(_.slice(args, argIndex), (arg) => { return _.startsWith(arg, '--') }) - const endIndex = nextOptOffset !== -1 ? argIndex + nextOptOffset : args.length + const endIndex = + nextOptOffset !== -1 ? argIndex + nextOptOffset : args.length const maybeSpecs = _.slice(args, argIndex, endIndex) const extraSpecs = _.intersection(maybeSpecs, fnArgs) @@ -70,11 +74,34 @@ const parseVariableOpts = (fnArgs, args) => { } const parseOpts = (opts) => { - opts = _.pick(opts, - 'project', 'spec', 'reporter', 'reporterOptions', 'path', 'destination', - 'port', 'env', 'cypressVersion', 'config', 'record', 'key', - 'browser', 'detached', 'headed', 'global', 'dev', 'force', 'exit', - 'cachePath', 'cacheList', 'cacheClear', 'parallel', 'group', 'ciBuildId') + opts = _.pick( + opts, + 'project', + 'spec', + 'reporter', + 'reporterOptions', + 'path', + 'destination', + 'port', + 'env', + 'cypressVersion', + 'config', + 'record', + 'key', + 'browser', + 'detached', + 'headed', + 'global', + 'dev', + 'force', + 'exit', + 'cachePath', + 'cacheList', + 'cacheClear', + 'parallel', + 'group', + 'ciBuildId' + ) if (opts.exit) { opts = _.omit(opts, 'exit') @@ -86,16 +113,23 @@ const parseOpts = (opts) => { } const descriptions = { - record: 'records the run. sends test results, screenshots and videos to your Cypress Dashboard.', - key: 'your secret Record Key. you can omit this if you set a CYPRESS_RECORD_KEY environment variable.', + record: + 'records the run. sends test results, screenshots and videos to your Cypress Dashboard.', + key: + 'your secret Record Key. you can omit this if you set a CYPRESS_RECORD_KEY environment variable.', spec: 'runs a specific spec file. defaults to "all"', - reporter: 'runs a specific mocha reporter. pass a path to use a custom reporter. defaults to "spec"', + reporter: + 'runs a specific mocha reporter. pass a path to use a custom reporter. defaults to "spec"', reporterOptions: 'options for the mocha reporter. defaults to "null"', port: 'runs Cypress on a specific port. overrides any value in cypress.json.', - env: 'sets environment variables. separate multiple values with a comma. overrides any value in cypress.json or cypress.env.json', - config: 'sets configuration values. separate multiple values with a comma. overrides any value in cypress.json.', - browserRunMode: 'runs Cypress in the browser with the given name. if a filesystem path is supplied, Cypress will attempt to use the browser at that path.', - browserOpenMode: 'path to a custom browser to be added to the list of available browsers in Cypress', + env: + 'sets environment variables. separate multiple values with a comma. overrides any value in cypress.json or cypress.env.json', + config: + 'sets configuration values. separate multiple values with a comma. overrides any value in cypress.json.', + browserRunMode: + 'runs Cypress in the browser with the given name. if a filesystem path is supplied, Cypress will attempt to use the browser at that path.', + browserOpenMode: + 'path to a custom browser to be added to the list of available browsers in Cypress', detached: 'runs Cypress application in detached mode', project: 'path to the project', global: 'force Cypress into global mode as if its globally installed', @@ -108,11 +142,25 @@ const descriptions = { cacheList: 'list cached binary versions', cacheClear: 'delete all cached binaries', group: 'a named group for recorded runs in the Cypress dashboard', - parallel: 'enables concurrent runs and automatic load balancing of specs across multiple machines or processes', - ciBuildId: 'the unique identifier for a run on your CI provider. typically a "BUILD_ID" env var. this value is automatically detected for most CI providers', + parallel: + 'enables concurrent runs and automatic load balancing of specs across multiple machines or processes', + ciBuildId: + 'the unique identifier for a run on your CI provider. typically a "BUILD_ID" env var. this value is automatically detected for most CI providers', } -const knownCommands = ['version', 'run', 'open', 'install', 'verify', '-v', '--version', 'help', '-h', '--help', 'cache'] +const knownCommands = [ + 'version', + 'run', + 'open', + 'install', + 'verify', + '-v', + '--version', + 'help', + '-h', + '--help', + 'cache', +] const text = (description) => { if (!descriptions[description]) { @@ -123,9 +171,11 @@ const text = (description) => { } function includesVersion (args) { - return _.includes(args, 'version') || + return ( + _.includes(args, 'version') || _.includes(args, '--version') || _.includes(args, '-v') + ) } function showVersions () { @@ -147,6 +197,14 @@ module.exports = { args = process.argv } + if (!util.isValidCypressEnvValue(process.env.CYPRESS_ENV)) { + debug('invalid CYPRESS_ENV value', process.env.CYPRESS_ENV) + + return errors.exitWithError(errors.errors.invalidCypressEnv)( + `CYPRESS_ENV=${process.env.CYPRESS_ENV}` + ) + } + const program = new commander.Command() // bug in commaner not printing name @@ -177,7 +235,10 @@ module.exports = { .option('-k, --key ', text('key')) .option('-s, --spec ', text('spec')) .option('-r, --reporter ', text('reporter')) - .option('-o, --reporter-options ', text('reporterOptions')) + .option( + '-o, --reporter-options ', + text('reporterOptions') + ) .option('-p, --port ', text('port')) .option('-e, --env ', text('env')) .option('-c, --config ', text('config')) @@ -218,7 +279,9 @@ module.exports = { program .command('install') .usage('[options]') - .description('Installs the Cypress executable matching this package\'s version') + .description( + 'Installs the Cypress executable matching this package\'s version' + ) .option('-f, --force', text('forceInstall')) .action((opts) => { require('./tasks/install') @@ -229,7 +292,9 @@ module.exports = { program .command('verify') .usage('[options]') - .description('Verifies that Cypress is installed correctly and executable') + .description( + 'Verifies that Cypress is installed correctly and executable' + ) .option('--dev', text('dev'), coerceFalse) .action((opts) => { const defaultOpts = { force: true, welcomeMessage: false } diff --git a/cli/lib/errors.js b/cli/lib/errors.js index 1a83b96ce434..ee45a1c21c4a 100644 --- a/cli/lib/errors.js +++ b/cli/lib/errors.js @@ -36,7 +36,9 @@ const failedUnzip = { const missingApp = (binaryDir) => { return { - description: `No version of Cypress is installed in: ${chalk.cyan(binaryDir)}`, + description: `No version of Cypress is installed in: ${chalk.cyan( + binaryDir + )}`, solution: stripIndent` \nPlease reinstall Cypress by running: ${chalk.cyan('cypress install')} `, @@ -59,7 +61,8 @@ const binaryNotExecutable = (executable) => { const notInstalledCI = (executable) => { return { - description: 'The cypress npm package is installed, but the Cypress binary is missing.', + description: + 'The cypress npm package is installed, but the Cypress binary is missing.', solution: stripIndent`\n We expected the binary to be installed here: ${chalk.cyan(executable)} @@ -114,7 +117,7 @@ const smokeTestFailure = (smokeTestCommand, timedOut) => { const invalidSmokeTestDisplayError = { code: 'INVALID_SMOKE_TEST_DISPLAY_ERROR', description: 'Cypress verification failed.', - solution (msg) { + solution (msg) { return stripIndent` Cypress failed to start after spawning a new Xvfb server. @@ -152,7 +155,8 @@ const missingDependency = { } const invalidCacheDirectory = { - description: 'Cypress cannot write to the cache directory due to file permissions', + description: + 'Cypress cannot write to the cache directory due to file permissions', solution: stripIndent` See discussion and possible solutions at ${chalk.blue(util.getGitHubIssueUrl(1281))} @@ -165,7 +169,8 @@ const versionMismatch = { } const unexpected = { - description: 'An unexpected error occurred while verifying the Cypress executable.', + description: + 'An unexpected error occurred while verifying the Cypress executable.', solution: stripIndent` Please search Cypress documentation for possible solutions: @@ -179,10 +184,19 @@ const unexpected = { `, } +const invalidCypressEnv = { + description: + chalk.red('The environment variable with the reserved name "CYPRESS_ENV" is set.'), + solution: chalk.red('Unset the "CYPRESS_ENV" environment variable and run Cypress again.'), + exitCode: 11, +} + const removed = { CYPRESS_BINARY_VERSION: { description: stripIndent` - The environment variable CYPRESS_BINARY_VERSION has been renamed to CYPRESS_INSTALL_BINARY as of version ${chalk.green('3.0.0')} + The environment variable CYPRESS_BINARY_VERSION has been renamed to CYPRESS_INSTALL_BINARY as of version ${chalk.green( + '3.0.0' + )} `, solution: stripIndent` You should set CYPRESS_INSTALL_BINARY instead. @@ -190,7 +204,9 @@ const removed = { }, CYPRESS_SKIP_BINARY_INSTALL: { description: stripIndent` - The environment variable CYPRESS_SKIP_BINARY_INSTALL has been removed as of version ${chalk.green('3.0.0')} + The environment variable CYPRESS_SKIP_BINARY_INSTALL has been removed as of version ${chalk.green( + '3.0.0' + )} `, solution: stripIndent` To skip the binary install, set CYPRESS_INSTALL_BINARY=0 @@ -210,8 +226,7 @@ const CYPRESS_RUN_BINARY = { } function getPlatformInfo () { - return util.getOsVersionAsync() - .then((version) => { + return util.getOsVersionAsync().then((version) => { return stripIndent` Platform: ${os.platform()} (${version}) Cypress Version: ${util.pkgVersion()} @@ -220,8 +235,7 @@ function getPlatformInfo () { } function addPlatformInformation (info) { - return getPlatformInfo() - .then((platform) => { + return getPlatformInfo().then((platform) => { return merge(info, { platform }) }) } @@ -231,18 +245,18 @@ function addPlatformInformation (info) { * and if possible a way to solve it. Resolves with a string. */ function formErrorText (info, msg, prevMessage) { - return addPlatformInformation(info) - .then((obj) => { + return addPlatformInformation(info).then((obj) => { const formatted = [] function add (msg) { - formatted.push( - stripIndents(msg) - ) + formatted.push(stripIndents(msg)) } - la(is.unemptyString(obj.description), - 'expected error description to be text', obj.description) + la( + is.unemptyString(obj.description), + 'expected error description to be text', + obj.description + ) // assuming that if there the solution is a function it will handle // error message and (optional previous error message) @@ -258,8 +272,11 @@ function formErrorText (info, msg, prevMessage) { `) } else { - la(is.unemptyString(obj.solution), - 'expected error solution to be text', obj.solution) + la( + is.unemptyString(obj.solution), + 'expected error solution to be text', + obj.solution + ) add(` ${obj.description} @@ -312,13 +329,30 @@ const raise = (info) => { const throwFormErrorText = (info) => { return (msg, prevMessage) => { - return formErrorText(info, msg, prevMessage) - .then(raise(info)) + return formErrorText(info, msg, prevMessage).then(raise(info)) + } +} + +/** + * Forms full error message with error and OS details, prints to the error output + * and then exits the process. + * @param {ErrorInformation} info Error information {description, solution} + * @example return exitWithError(errors.invalidCypressEnv)('foo') + */ +const exitWithError = (info) => { + return (msg) => { + return formErrorText(info, msg).then((text) => { + // eslint-disable-next-line no-console + console.error(text) + process.exit(info.exitCode || 1) + }) } } module.exports = { raise, + exitWithError, + // formError, formErrorText, throwFormErrorText, hr, @@ -334,6 +368,7 @@ module.exports = { unexpected, failedDownload, failedUnzip, + invalidCypressEnv, invalidCacheDirectory, removed, CYPRESS_RUN_BINARY, diff --git a/cli/lib/util.js b/cli/lib/util.js index 9d0044835675..029c0bb1893c 100644 --- a/cli/lib/util.js +++ b/cli/lib/util.js @@ -119,6 +119,25 @@ function stdoutLineMatches (expectedLine, stdout) { return lines.some(lineMatches) } +/** + * Confirms if given value is a valid CYPRESS_ENV value. Undefined values + * are valid, because the system can set the default one. + * + * @param {string} value + * @example util.isValidCypressEnvValue(process.env.CYPRESS_ENV) + */ +function isValidCypressEnvValue (value) { + if (_.isUndefined(value)) { + // will get default value + return true + } + + // names of config environments, see "packages/server/config/app.yml" + const names = ['development', 'test', 'staging', 'production'] + + return _.includes(names, value) +} + /** * Prints NODE_OPTIONS using debug() module, but only * if DEBUG=cypress... is set @@ -158,7 +177,7 @@ const dequote = (str) => { const util = { normalizeModuleOptions, - + isValidCypressEnvValue, printNodeOptions, isCi () { diff --git a/cli/package.json b/cli/package.json index 30e1e8fa1780..37ff7f30be79 100644 --- a/cli/package.json +++ b/cli/package.json @@ -86,7 +86,8 @@ "shelljs": "0.8.3", "sinon": "7.2.2", "snap-shot-it": "7.8.0", - "spawn-mock": "1.0.0" + "spawn-mock": "1.0.0", + "strip-ansi": "4.0.0" }, "files": [ "bin", diff --git a/cli/test/lib/cli_spec.js b/cli/test/lib/cli_spec.js index 952441ad1132..42ed40e53a41 100644 --- a/cli/test/lib/cli_spec.js +++ b/cli/test/lib/cli_spec.js @@ -73,6 +73,58 @@ describe('cli', () => { }) }) + context('CYPRESS_ENV', () => { + /** + * Replaces line "Platform: ..." with "Platform: xxx" + * @param {string} s + */ + const replacePlatform = (s) => { + return s.replace(/Platform: .+/, 'Platform: xxx') + } + + /** + * Replaces line "Cypress Version: ..." with "Cypress Version: 1.2.3" + * @param {string} s + */ + const replaceCypressVersion = (s) => { + return s.replace(/Cypress Version: .+/, 'Cypress Version: 1.2.3') + } + + const sanitizePlatform = (text) => { + return text + .split(os.eol) + .map(replacePlatform) + .map(replaceCypressVersion) + .join(os.eol) + } + + it('allows staging environment', () => { + const options = { + env: { + CYPRESS_ENV: 'staging', + }, + // we are only interested in the exit code + filter: ['code', 'stderr'], + } + + return execa('bin/cypress', ['help'], options).then(snapshot) + }) + + it('catches environment "foo"', () => { + const options = { + env: { + CYPRESS_ENV: 'foo', + }, + // we are only interested in the exit code + filter: ['code', 'stderr'], + } + + return execa('bin/cypress', ['help'], options) + .then(sanitizePlatform) + .then(snapshot) + }) + }) + context('cypress version', () => { const binaryDir = '/binary/dir' diff --git a/packages/driver/src/cy/commands/querying.coffee b/packages/driver/src/cy/commands/querying.coffee index 934edb0b584a..25ddb2fcf4fd 100644 --- a/packages/driver/src/cy/commands/querying.coffee +++ b/packages/driver/src/cy/commands/querying.coffee @@ -68,7 +68,10 @@ module.exports = (Commands, Cypress, cy, state, config) -> get: (selector, options = {}) -> ctx = @ - + + if options is null or Array.isArray(options) or typeof options isnt 'object' then return $utils.throwErrByPath "get.invalid_options", { + args: { options } + } _.defaults(options, { retry: true withinSubject: cy.state("withinSubject") @@ -78,7 +81,6 @@ module.exports = (Commands, Cypress, cy, state, config) -> }) consoleProps = {} - start = (aliasType) -> return if options.log is false @@ -467,4 +469,4 @@ module.exports = (Commands, Cypress, cy, state, config) -> cy.state("withinSubject", null) return subject - }) + }) \ No newline at end of file diff --git a/packages/driver/src/cy/commands/window.coffee b/packages/driver/src/cy/commands/window.coffee index e6b3ec5a1375..0d6ca733cd3b 100644 --- a/packages/driver/src/cy/commands/window.coffee +++ b/packages/driver/src/cy/commands/window.coffee @@ -159,7 +159,7 @@ module.exports = (Commands, Cypress, cy, state, config) -> widthAndHeightAreWithinBounds = (width, height) -> _.every [width, height], (val) -> - val >= 20 and val <= 3000 + val >= 20 and val <= 4000 switch when _.isString(presetOrWidth) and _.isBlank(presetOrWidth) diff --git a/packages/driver/src/cypress/error_messages.coffee b/packages/driver/src/cypress/error_messages.coffee index 58740aed0964..54d6801baaba 100644 --- a/packages/driver/src/cypress/error_messages.coffee +++ b/packages/driver/src/cypress/error_messages.coffee @@ -292,6 +292,7 @@ module.exports = { get: alias_invalid: "'{{prop}}' is not a valid alias property. Only 'numbers' or 'all' is permitted." alias_zero: "'0' is not a valid alias property. Are you trying to ask for the first response? If so write @{{alias}}.1" + invalid_options: "#{cmd('get')} only accepts an options object for its second argument. You passed {{options}}" getCookie: invalid_argument: "#{cmd('getCookie')} must be passed a string argument for name." @@ -975,7 +976,7 @@ module.exports = { viewport: bad_args: "#{cmd('viewport')} can only accept a string preset or a width and height as numbers." - dimensions_out_of_range: "#{cmd('viewport')} width and height must be between 20px and 3000px." + dimensions_out_of_range: "#{cmd('viewport')} width and height must be between 20px and 4000px." empty_string: "#{cmd('viewport')} cannot be passed an empty string." invalid_orientation: "#{cmd('viewport')} can only accept '{{all}}' as valid orientations. Your orientation was: '{{orientation}}'" missing_preset: "#{cmd('viewport')} could not find a preset for: '{{preset}}'. Available presets are: {{presets}}" diff --git a/packages/driver/test/cypress/integration/commands/querying_spec.coffee b/packages/driver/test/cypress/integration/commands/querying_spec.coffee index 11c79a85dd82..a6303e9c22f4 100644 --- a/packages/driver/test/cypress/integration/commands/querying_spec.coffee +++ b/packages/driver/test/cypress/integration/commands/querying_spec.coffee @@ -1173,7 +1173,14 @@ describe "src/cy/commands/querying", -> .server() .route(/users/, {}).as("getUsers") .get("@getUsers.all ") + _.each ["", "foo", [], 1, null ], (value) => + it "throws when options property is not an object. Such as: #{value}", (done) -> + cy.on "fail", (err) -> + expect(err.message).to.include "only accepts an options object for its second argument. You passed #{value}" + done() + cy.get("foobar", value) + it "logs out $el when existing $el is found even on failure", (done) -> button = cy.$$("#button").hide() diff --git a/packages/driver/test/cypress/integration/commands/window_spec.coffee b/packages/driver/test/cypress/integration/commands/window_spec.coffee index ca012cf72d49..fc6ff91e4bf7 100644 --- a/packages/driver/test/cypress/integration/commands/window_spec.coffee +++ b/packages/driver/test/cypress/integration/commands/window_spec.coffee @@ -640,7 +640,7 @@ describe "src/cy/commands/window", -> it "throws when passed negative numbers", (done) -> cy.on "fail", (err) => expect(@logs.length).to.eq(1) - expect(err.message).to.eq "cy.viewport() width and height must be between 20px and 3000px." + expect(err.message).to.eq "cy.viewport() width and height must be between 20px and 4000px." done() cy.viewport(800, -600) @@ -648,7 +648,7 @@ describe "src/cy/commands/window", -> it "throws when passed width less than 20", (done) -> cy.on "fail", (err) => expect(@logs.length).to.eq(1) - expect(err.message).to.eq "cy.viewport() width and height must be between 20px and 3000px." + expect(err.message).to.eq "cy.viewport() width and height must be between 20px and 4000px." done() cy.viewport(19, 600) @@ -656,16 +656,16 @@ describe "src/cy/commands/window", -> it "does not throw when passed width equal to 20", -> cy.viewport(20, 600) - it "throws when passed height greater than than 3000", (done) -> + it "throws when passed height greater than than 4000", (done) -> cy.on "fail", (err) => expect(@logs.length).to.eq(1) - expect(err.message).to.eq "cy.viewport() width and height must be between 20px and 3000px." + expect(err.message).to.eq "cy.viewport() width and height must be between 20px and 4000px." done() - cy.viewport(1000, 3001) + cy.viewport(1000, 4001) - it "does not throw when passed width equal to 3000", -> - cy.viewport(200, 3000) + it "does not throw when passed width equal to 4000", -> + cy.viewport(200, 4000) it "throws when passed an empty string as width", (done) -> cy.on "fail", (err) => diff --git a/packages/server/__snapshots__/5_screenshots_spec.coffee.js b/packages/server/__snapshots__/5_screenshots_spec.coffee.js index 37055a86e1ac..93f119dd717e 100644 --- a/packages/server/__snapshots__/5_screenshots_spec.coffee.js +++ b/packages/server/__snapshots__/5_screenshots_spec.coffee.js @@ -7,14 +7,14 @@ exports['e2e screenshots passes 1'] = ` ┌────────────────────────────────────────────────────────────────────────────────────────────────┐ │ Cypress: 1.2.3 │ │ Browser: FooBrowser 88 │ - │ Specs: 1 found (screenshots_spec.coffee) │ - │ Searched: cypress/integration/screenshots_spec.coffee │ + │ Specs: 1 found (screenshots_spec.js) │ + │ Searched: cypress/integration/screenshots_spec.js │ └────────────────────────────────────────────────────────────────────────────────────────────────┘ ──────────────────────────────────────────────────────────────────────────────────────────────────── - Running: screenshots_spec.coffee... (1 of 1) + Running: screenshots_spec.js... (1 of 1) taking screenshots @@ -31,7 +31,7 @@ exports['e2e screenshots passes 1'] = ` ✓ retries each screenshot for up to XX:XX ✓ ensures unique paths for non-named screenshots 2) ensures unique paths when there's a non-named screenshot and a failure - - doesn't take a screenshot for a pending test + - does not take a screenshot for a pending test clipping ✓ can clip app screenshots ✓ can clip runner screenshots @@ -82,46 +82,46 @@ Because this error occurred during a 'after each' hook we are skipping the remai (Results) - ┌───────────────────────────────────────┐ - │ Tests: 22 │ - │ Passing: 17 │ - │ Failing: 4 │ - │ Pending: 1 │ - │ Skipped: 0 │ - │ Screenshots: 25 │ - │ Video: true │ - │ Duration: X seconds │ - │ Spec Ran: screenshots_spec.coffee │ - └───────────────────────────────────────┘ + ┌───────────────────────────────────┐ + │ Tests: 22 │ + │ Passing: 17 │ + │ Failing: 4 │ + │ Pending: 1 │ + │ Skipped: 0 │ + │ Screenshots: 25 │ + │ Video: true │ + │ Duration: X seconds │ + │ Spec Ran: screenshots_spec.js │ + └───────────────────────────────────┘ (Screenshots) - - /foo/bar/.projects/e2e/cypress/screenshots/screenshots_spec.coffee/black.png (1280x720) - - /foo/bar/.projects/e2e/cypress/screenshots/screenshots_spec.coffee/red.png (1280x720) - - /foo/bar/.projects/e2e/cypress/screenshots/screenshots_spec.coffee/foo/bar/baz.png (1280x720) - - /foo/bar/.projects/e2e/cypress/screenshots/screenshots_spec.coffee/taking screenshots -- generates pngs on failure (failed).png (1280x720) - - /foo/bar/.projects/e2e/cypress/screenshots/screenshots_spec.coffee/color-check.png (1280x720) - - /foo/bar/.projects/e2e/cypress/screenshots/screenshots_spec.coffee/crop-check.png (600x400) - - /foo/bar/.projects/e2e/cypress/screenshots/screenshots_spec.coffee/fullPage.png (600x500) - - /foo/bar/.projects/e2e/cypress/screenshots/screenshots_spec.coffee/fullPage-same.png (600x500) - - /foo/bar/.projects/e2e/cypress/screenshots/screenshots_spec.coffee/pathological.png (1280x720) - - /foo/bar/.projects/e2e/cypress/screenshots/screenshots_spec.coffee/element.png (400x300) - - /foo/bar/.projects/e2e/cypress/screenshots/screenshots_spec.coffee/taking screenshots -- retries each screenshot for up to XX:XX.png (200x1300) - - /foo/bar/.projects/e2e/cypress/screenshots/screenshots_spec.coffee/taking screenshots -- ensures unique paths for non-named screenshots.png (1280x720) - - /foo/bar/.projects/e2e/cypress/screenshots/screenshots_spec.coffee/taking screenshots -- ensures unique paths for non-named screenshots (1).png (1280x720) - - /foo/bar/.projects/e2e/cypress/screenshots/screenshots_spec.coffee/taking screenshots -- ensures unique paths for non-named screenshots (2).png (1280x720) - - /foo/bar/.projects/e2e/cypress/screenshots/screenshots_spec.coffee/taking screenshots -- ensures unique paths when there's a non-named screenshot and a failure.png (1000x660) - - /foo/bar/.projects/e2e/cypress/screenshots/screenshots_spec.coffee/taking screenshots -- ensures unique paths when there's a non-named screenshot and a failure (failed).png (1280x720) - - /foo/bar/.projects/e2e/cypress/screenshots/screenshots_spec.coffee/app-clip.png (100x50) - - /foo/bar/.projects/e2e/cypress/screenshots/screenshots_spec.coffee/runner-clip.png (120x60) - - /foo/bar/.projects/e2e/cypress/screenshots/screenshots_spec.coffee/fullPage-clip.png (140x70) - - /foo/bar/.projects/e2e/cypress/screenshots/screenshots_spec.coffee/element-clip.png (160x80) - - /foo/bar/.projects/e2e/cypress/screenshots/screenshots_spec.coffee/taking screenshots -- before hooks -- empty test 1 -- before all hook (failed).png (1280x720) - - /foo/bar/.projects/e2e/cypress/screenshots/screenshots_spec.coffee/taking screenshots -- each hooks -- empty test 2 -- before each hook (failed).png (1280x720) - - /foo/bar/.projects/e2e/cypress/screenshots/screenshots_spec.coffee/taking screenshots -- each hooks -- empty test 2 -- after each hook (failed).png (1280x720) - - /foo/bar/.projects/e2e/cypress/screenshots/screenshots_spec.coffee/taking screenshots -- really long test title aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa.png (1000x660) - - /foo/bar/.projects/e2e/cypress/screenshots/screenshots_spec.coffee/taking screenshots -- really long test title aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa (1).png (1000x660) + - /foo/bar/.projects/e2e/cypress/screenshots/screenshots_spec.js/black.png (1280x720) + - /foo/bar/.projects/e2e/cypress/screenshots/screenshots_spec.js/red.png (1280x720) + - /foo/bar/.projects/e2e/cypress/screenshots/screenshots_spec.js/foo/bar/baz.png (1280x720) + - /foo/bar/.projects/e2e/cypress/screenshots/screenshots_spec.js/taking screenshots -- generates pngs on failure (failed).png (1280x720) + - /foo/bar/.projects/e2e/cypress/screenshots/screenshots_spec.js/color-check.png (1280x720) + - /foo/bar/.projects/e2e/cypress/screenshots/screenshots_spec.js/crop-check.png (600x400) + - /foo/bar/.projects/e2e/cypress/screenshots/screenshots_spec.js/fullPage.png (600x500) + - /foo/bar/.projects/e2e/cypress/screenshots/screenshots_spec.js/fullPage-same.png (600x500) + - /foo/bar/.projects/e2e/cypress/screenshots/screenshots_spec.js/pathological.png (1280x720) + - /foo/bar/.projects/e2e/cypress/screenshots/screenshots_spec.js/element.png (400x300) + - /foo/bar/.projects/e2e/cypress/screenshots/screenshots_spec.js/taking screenshots -- retries each screenshot for up to XX:XX.png (200x1300) + - /foo/bar/.projects/e2e/cypress/screenshots/screenshots_spec.js/taking screenshots -- ensures unique paths for non-named screenshots.png (1280x720) + - /foo/bar/.projects/e2e/cypress/screenshots/screenshots_spec.js/taking screenshots -- ensures unique paths for non-named screenshots (1).png (1280x720) + - /foo/bar/.projects/e2e/cypress/screenshots/screenshots_spec.js/taking screenshots -- ensures unique paths for non-named screenshots (2).png (1280x720) + - /foo/bar/.projects/e2e/cypress/screenshots/screenshots_spec.js/taking screenshots -- ensures unique paths when there's a non-named screenshot and a failure.png (1000x660) + - /foo/bar/.projects/e2e/cypress/screenshots/screenshots_spec.js/taking screenshots -- ensures unique paths when there's a non-named screenshot and a failure (failed).png (1280x720) + - /foo/bar/.projects/e2e/cypress/screenshots/screenshots_spec.js/app-clip.png (100x50) + - /foo/bar/.projects/e2e/cypress/screenshots/screenshots_spec.js/runner-clip.png (120x60) + - /foo/bar/.projects/e2e/cypress/screenshots/screenshots_spec.js/fullPage-clip.png (140x70) + - /foo/bar/.projects/e2e/cypress/screenshots/screenshots_spec.js/element-clip.png (160x80) + - /foo/bar/.projects/e2e/cypress/screenshots/screenshots_spec.js/taking screenshots -- before hooks -- empty test 1 -- before all hook (failed).png (1280x720) + - /foo/bar/.projects/e2e/cypress/screenshots/screenshots_spec.js/taking screenshots -- each hooks -- empty test 2 -- before each hook (failed).png (1280x720) + - /foo/bar/.projects/e2e/cypress/screenshots/screenshots_spec.js/taking screenshots -- each hooks -- empty test 2 -- after each hook (failed).png (1280x720) + - /foo/bar/.projects/e2e/cypress/screenshots/screenshots_spec.js/taking screenshots -- really long test title aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa.png (1000x660) + - /foo/bar/.projects/e2e/cypress/screenshots/screenshots_spec.js/taking screenshots -- really long test title aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa (1).png (1000x660) (Video) @@ -137,7 +137,7 @@ Because this error occurred during a 'after each' hook we are skipping the remai Spec Tests Passing Failing Pending Skipped ┌────────────────────────────────────────────────────────────────────────────────────────────────┐ - │ ✖ screenshots_spec.coffee XX:XX 22 17 4 1 - │ + │ ✖ screenshots_spec.js XX:XX 22 17 4 1 - │ └────────────────────────────────────────────────────────────────────────────────────────────────┘ 1 of 1 failed (100%) XX:XX 22 17 4 1 - diff --git a/packages/server/lib/config.coffee b/packages/server/lib/config.coffee index 7777795fec20..96f9a36bd1a8 100644 --- a/packages/server/lib/config.coffee +++ b/packages/server/lib/config.coffee @@ -10,7 +10,7 @@ origin = require("./util/origin") coerce = require("./util/coerce") settings = require("./util/settings") v = require("./util/validation") -debug = require("debug")("cypress:server:config") +debug = require("debug")("cypress:server:config") pathHelpers = require("./util/path_helpers") CYPRESS_ENV_PREFIX = "CYPRESS_" @@ -207,6 +207,11 @@ hideSpecialVals = (val, key) -> module.exports = { getConfigKeys: -> configKeys + isValidCypressEnvValue: (value) -> + # names of config environments, see "config/app.yml" + names = ["development", "test", "staging", "production"] + _.includes(names, value) + whitelist: (obj = {}) -> _.pick(obj, configKeys.concat(breakingConfigKeys)) @@ -265,7 +270,12 @@ module.exports = { ## split out our own app wide env from user env variables ## and delete envFile config.env = @parseEnv(config, options.env, resolved) + config.cypressEnv = process.env["CYPRESS_ENV"] + debug("using CYPRESS_ENV %s", config.cypressEnv) + if not @isValidCypressEnvValue(config.cypressEnv) + errors.throw("INVALID_CYPRESS_ENV", config.cypressEnv) + delete config.envFile ## when headless diff --git a/packages/server/lib/errors.coffee b/packages/server/lib/errors.coffee index 02ff1114092f..b5c329493b0c 100644 --- a/packages/server/lib/errors.coffee +++ b/packages/server/lib/errors.coffee @@ -826,6 +826,14 @@ getMsgByType = (type, arg1 = {}, arg2) -> """ Cypress detected policy settings on your computer that may cause issues with using this browser. For more information, see https://on.cypress.io/bad-browser-policy """ + when "INVALID_CYPRESS_ENV" + """ + We have detected unknown or unsupported CYPRESS_ENV value + + #{chalk.yellow(arg1)} + + Please do not modify CYPRESS_ENV value. + """ get = (type, arg1, arg2) -> msg = getMsgByType(type, arg1, arg2) diff --git a/packages/server/test/e2e/5_screenshots_spec.coffee b/packages/server/test/e2e/5_screenshots_spec.coffee index 0869e906666f..08c7798dc9a2 100644 --- a/packages/server/test/e2e/5_screenshots_spec.coffee +++ b/packages/server/test/e2e/5_screenshots_spec.coffee @@ -62,14 +62,14 @@ describe "e2e screenshots", -> ## the test title as the file name e2e.exec(@, { - spec: "screenshots_spec.coffee" + spec: "screenshots_spec.js" expectedExitCode: 4 snapshot: true timeout: 180000 }) .then -> screenshot = (paths...) -> - path.join(e2ePath, "cypress", "screenshots", "screenshots_spec.coffee", paths...) + path.join(e2ePath, "cypress", "screenshots", "screenshots_spec.js", paths...) screenshot1 = screenshot("black.png") screenshot2 = screenshot("red.png") diff --git a/packages/server/test/support/fixtures/projects/busted-support-file/cypress/support/index.js b/packages/server/test/support/fixtures/projects/busted-support-file/cypress/support/index.js index b634e9380711..fbab6df357e4 100644 --- a/packages/server/test/support/fixtures/projects/busted-support-file/cypress/support/index.js +++ b/packages/server/test/support/fixtures/projects/busted-support-file/cypress/support/index.js @@ -1 +1 @@ -import "./does/not/exist" \ No newline at end of file +import './does/not/exist' diff --git a/packages/server/test/support/fixtures/projects/default-layout/cypress/integration/default_layout_spec.js b/packages/server/test/support/fixtures/projects/default-layout/cypress/integration/default_layout_spec.js index e2fbbbe9ab71..7e5e612bae4e 100644 --- a/packages/server/test/support/fixtures/projects/default-layout/cypress/integration/default_layout_spec.js +++ b/packages/server/test/support/fixtures/projects/default-layout/cypress/integration/default_layout_spec.js @@ -1 +1,2 @@ +/* eslint-disable mocha/no-global-tests */ it('works', () => {}) diff --git a/packages/server/test/support/fixtures/projects/e2e/cypress/integration/browserify_babel_es2015_failing_spec.js b/packages/server/test/support/fixtures/projects/e2e/cypress/integration/browserify_babel_es2015_failing_spec.js index 16a6f89e5e57..8bcc6576e58b 100644 --- a/packages/server/test/support/fixtures/projects/e2e/cypress/integration/browserify_babel_es2015_failing_spec.js +++ b/packages/server/test/support/fixtures/projects/e2e/cypress/integration/browserify_babel_es2015_failing_spec.js @@ -1 +1 @@ -import "../../lib/fail" \ No newline at end of file +import '../../lib/fail' diff --git a/packages/server/test/support/fixtures/projects/e2e/cypress/integration/https_passthru_spec.js b/packages/server/test/support/fixtures/projects/e2e/cypress/integration/https_passthru_spec.js index ae3a83c6480d..b7315852265f 100644 --- a/packages/server/test/support/fixtures/projects/e2e/cypress/integration/https_passthru_spec.js +++ b/packages/server/test/support/fixtures/projects/e2e/cypress/integration/https_passthru_spec.js @@ -23,6 +23,7 @@ describe('https passthru retries', () => { img.onload = () => { reject(new Error('onload event fired, but should not have. expected onerror to fire.')) } + img.onerror = resolve }) }) diff --git a/packages/server/test/support/fixtures/projects/e2e/cypress/integration/network_error_handling_spec.js b/packages/server/test/support/fixtures/projects/e2e/cypress/integration/network_error_handling_spec.js index b62f31b30c8c..318ddc13ae52 100644 --- a/packages/server/test/support/fixtures/projects/e2e/cypress/integration/network_error_handling_spec.js +++ b/packages/server/test/support/fixtures/projects/e2e/cypress/integration/network_error_handling_spec.js @@ -27,8 +27,10 @@ describe('network error handling', function () { }) .get('input[type=text]') .type('bar') + cy.get('input[type=submit]') .click() + cy.contains('{"foo":"bar"}') }) }) diff --git a/packages/server/test/support/fixtures/projects/e2e/cypress/integration/screenshots_spec.js b/packages/server/test/support/fixtures/projects/e2e/cypress/integration/screenshots_spec.js index 199b3352a160..8cbb8f339633 100644 --- a/packages/server/test/support/fixtures/projects/e2e/cypress/integration/screenshots_spec.js +++ b/packages/server/test/support/fixtures/projects/e2e/cypress/integration/screenshots_spec.js @@ -1,283 +1,277 @@ -/* - * decaffeinate suggestions: - * DS102: Remove unnecessary code created because of implicit returns - * Full docs: https://github.com/decaffeinate/decaffeinate/blob/master/docs/suggestions.md - */ -const { devicePixelRatio } = window; +const { devicePixelRatio } = window +const path = require('path') -describe("taking screenshots", function() { - let failureTestRan = false; - const onAfterScreenshotResults = []; +describe('taking screenshots', () => { + let failureTestRan = false + const onAfterScreenshotResults = [] Cypress.Screenshot.defaults({ - onAfterScreenshot($el, results) { - return onAfterScreenshotResults.push(results); - } - }); - - it("manually generates pngs", () => - cy - .visit("http://localhost:3322/color/black") - .screenshot("black", { capture: "runner" }) - .wait(1500) - .visit("http://localhost:3322/color/red") - .screenshot("red", { capture: "runner" }) - ); - - it("can nest screenshots in folders", () => - cy - .visit("http://localhost:3322/color/white") - .screenshot("foo/bar/baz", { capture: "runner" }) - ); - - it("generates pngs on failure", function() { - failureTestRan = true; - - return cy - .visit("http://localhost:3322/color/yellow") - .wait(1500) - .then(function() { - //# failure 1 - throw new Error("fail whale"); - }); - }); - - it("calls onAfterScreenshot with results of failed tests", function() { - //# this test will only pass if the previous test ran + onAfterScreenshot ($el, results) { + onAfterScreenshotResults.push(results) + }, + }) + + it('manually generates pngs', () => { + cy.visit('http://localhost:3322/color/black') + cy.screenshot('black', { capture: 'runner' }) + cy.wait(1500) + cy.visit('http://localhost:3322/color/red') + cy.screenshot('red', { capture: 'runner' }) + }) + + it('can nest screenshots in folders', () => { + cy.visit('http://localhost:3322/color/white') + cy.screenshot('foo/bar/baz', { capture: 'runner' }) + }) + + it('generates pngs on failure', () => { + failureTestRan = true + + cy.visit('http://localhost:3322/color/yellow') + cy.wait(1500) + .then(() => { + // failure 1 + throw new Error('fail whale') + }) + }) + + it('calls onAfterScreenshot with results of failed tests', () => { + // this test will only pass if the previous test ran if (!failureTestRan) { - throw new Error("this test can only pass if the previous test ran"); + throw new Error('this test can only pass if the previous test ran') } const testFailure = Cypress._.find(onAfterScreenshotResults, { - testFailure: true - }); - - expect(testFailure).to.exist; - - return expect(Cypress._.map(onAfterScreenshotResults, "name")).to.deep.eq([ - "black", "red", "foo/bar/baz", undefined - ]); - }); - - it("handles devicePixelRatio correctly on headless electron", () => - //# this checks to see if the topLeftRight pixel (1, 0) is - //# currently white. when electron runs offscreen it upscales - //# images incorrectly on retina screens and the algorithm - //# blurs this pixel into gray. - cy - .screenshot("color-check", { capture: "runner" }) - .task("ensure:pixel:color", { - devicePixelRatio, - coords: [1, 0], - color: [255, 255, 255], //# white - name: "screenshots_spec.coffee/color-check" + testFailure: true, + }) + + expect(testFailure).to.exist + + expect(Cypress._.map(onAfterScreenshotResults, 'name')).to.deep.eq([ + 'black', 'red', 'foo/bar/baz', undefined, + ]) + }) + + it('handles devicePixelRatio correctly on headless electron', () => { + // this checks to see if the topLeftRight pixel (1, 0) is + // currently white. when electron runs offscreen it upscales + // images incorrectly on retina screens and the algorithm + // blurs this pixel into gray. + cy.screenshot('color-check', { capture: 'runner' }) + cy.task('ensure:pixel:color', { + devicePixelRatio, + coords: [1, 0], + color: [255, 255, 255], // white + name: `${path.basename(__filename)}/color-check`, + }) + + cy.task('ensure:pixel:color', { + devicePixelRatio, + coords: [0, 1], + color: [255, 255, 255], // white + name: `${path.basename(__filename)}/color-check`, + }) + }) + + it('crops app captures to just app size', () => { + cy.viewport(600, 400) + cy.visit('http://localhost:3322/color/yellow') + cy.screenshot('crop-check', { capture: 'viewport' }) + cy.task('check:screenshot:size', { + name: `${path.basename(__filename)}/crop-check.png`, + width: 600, + height: 400, + devicePixelRatio, + }) + }) + + it('can capture fullPage screenshots', () => { + cy.viewport(600, 200) + cy.visit('http://localhost:3322/fullPage') + cy.screenshot('fullPage', { capture: 'fullPage' }) + cy.task('check:screenshot:size', { + name: `${path.basename(__filename)}/fullPage.png`, + width: 600, + height: 500, + devicePixelRatio, + }) + }) + + it('accepts subsequent same captures after multiple tries', () => { + cy.viewport(600, 200) + cy.visit('http://localhost:3322/fullPage-same') + cy.screenshot('fullPage-same', { capture: 'fullPage' }) + cy.task('check:screenshot:size', { + name: `${path.basename(__filename)}/fullPage-same.png`, + width: 600, + height: 500, + devicePixelRatio, + }) + }) + + it('accepts screenshot after multiple tries if somehow app has pixels that match helper pixels', () => { + cy.viewport(1280, 720) + cy.visit('http://localhost:3322/pathological') + cy.screenshot('pathological', { capture: 'viewport' }) + }) + + it('can capture element screenshots', () => { + cy.viewport(600, 200) + cy.visit('http://localhost:3322/element') + cy.get('.element') + .screenshot('element') + + cy.task('check:screenshot:size', { + name: `${path.basename(__filename)}/element.png`, + width: 400, + height: 300, + devicePixelRatio, + }) + }) + + it('retries each screenshot for up to 1500ms', () => { + cy.viewport(400, 400) + cy.visit('http://localhost:3322/identical') + cy.get('div:first').should('have.css', 'height', '1300px') + .screenshot({ + onAfterScreenshot ($el, results) { + let fourth; let third + + expect($el).to.match('div') + + const { duration } = results + + // there should be 4 screenshots taken + // because the height is 1700px. + // the 1st will resolve super fast since it + // won't match any other screenshots. + // the 2nd/3rd will take up to their 1500ms + // because they will be identical to the first. + // the 4th will also go quickly because it will not + // match the 3rd + const first = (fourth = 250) + const second = (third = 1500) + const total = first + second + third + fourth + const padding = 2000 // account for slower machines + + expect(duration).to.be.within(total, total + padding) + }, + }) + }) + + it('ensures unique paths for non-named screenshots', () => { + cy.screenshot({ capture: 'runner' }) + cy.screenshot({ capture: 'runner' }) + cy.screenshot({ capture: 'runner' }) + cy.readFile(`cypress/screenshots/${path.basename(__filename)}/taking screenshots -- ensures unique paths for non-named screenshots.png`, 'base64') + cy.readFile(`cypress/screenshots/${path.basename(__filename)}/taking screenshots -- ensures unique paths for non-named screenshots (1).png`, 'base64') + + cy.readFile(`cypress/screenshots/${path.basename(__filename)}/taking screenshots -- ensures unique paths for non-named screenshots (2).png`, 'base64') + }) + + it('ensures unique paths when there\'s a non-named screenshot and a failure', () => { + cy.screenshot({ capture: 'viewport' }).then(() => { + throw new Error('failing on purpose') + }) + }) + + describe('clipping', () => { + it('can clip app screenshots', () => { + cy.viewport(600, 200) + cy.visit('http://localhost:3322/color/yellow') + cy.screenshot('app-clip', { + capture: 'viewport', clip: { x: 10, y: 10, width: 100, height: 50 }, }) - .task("ensure:pixel:color", { + + cy.task('check:screenshot:size', { + name: `${path.basename(__filename)}/app-clip.png`, + width: 100, + height: 50, devicePixelRatio, - coords: [0, 1], - color: [255, 255, 255], //# white - name: "screenshots_spec.coffee/color-check" }) - ); - - it("crops app captures to just app size", () => - cy - .viewport(600, 400) - .visit("http://localhost:3322/color/yellow") - .screenshot("crop-check", { capture: "viewport" }) - .task("check:screenshot:size", { - name: "screenshots_spec.coffee/crop-check.png", - width: 600, - height: 400, - devicePixelRatio + }) + + it('can clip runner screenshots', () => { + cy.viewport(600, 200) + cy.visit('http://localhost:3322/color/yellow') + cy.screenshot('runner-clip', { + capture: 'runner', clip: { x: 15, y: 15, width: 120, height: 60 }, }) - ); - - it("can capture fullPage screenshots", () => - cy - .viewport(600, 200) - .visit("http://localhost:3322/fullPage") - .screenshot("fullPage", { capture: "fullPage" }) - .task("check:screenshot:size", { - name: "screenshots_spec.coffee/fullPage.png", - width: 600, - height: 500, - devicePixelRatio + + cy.task('check:screenshot:size', { + name: `${path.basename(__filename)}/runner-clip.png`, + width: 120, + height: 60, + devicePixelRatio, }) - ); - - it("accepts subsequent same captures after multiple tries", () => - cy - .viewport(600, 200) - .visit("http://localhost:3322/fullPage-same") - .screenshot("fullPage-same", { capture: "fullPage" }) - .task("check:screenshot:size", { - name: "screenshots_spec.coffee/fullPage-same.png", - width: 600, - height: 500, - devicePixelRatio + }) + + it('can clip fullPage screenshots', () => { + cy.viewport(600, 200) + cy.visit('http://localhost:3322/fullPage') + cy.screenshot('fullPage-clip', { + capture: 'fullPage', clip: { x: 20, y: 20, width: 140, height: 70 }, }) - ); - - it("accepts screenshot after multiple tries if somehow app has pixels that match helper pixels", () => - cy - .viewport(1280, 720) - .visit("http://localhost:3322/pathological") - .screenshot("pathological", { capture: "viewport" }) - ); - - it("can capture element screenshots", () => - cy - .viewport(600, 200) - .visit("http://localhost:3322/element") - .get(".element") - .screenshot("element") - .task("check:screenshot:size", { - name: "screenshots_spec.coffee/element.png", - width: 400, - height: 300, - devicePixelRatio + + cy.task('check:screenshot:size', { + name: `${path.basename(__filename)}/fullPage-clip.png`, + width: 140, + height: 70, + devicePixelRatio, }) - ); - - it("retries each screenshot for up to 1500ms", () => - cy - .viewport(400, 400) - .visit("http://localhost:3322/identical") - .get("div:first").should("have.css", "height", "1300px") - .screenshot({ - onAfterScreenshot($el, results) { - let fourth, third; - expect($el).to.match("div"); - - const { duration } = results; - - //# there should be 4 screenshots taken - //# because the height is 1700px. - //# the 1st will resolve super fast since it - //# won't match any other screenshots. - //# the 2nd/3rd will take up to their 1500ms - //# because they will be identical to the first. - //# the 4th will also go quickly because it will not - //# match the 3rd - const first = (fourth = 250); - const second = (third = 1500); - const total = first + second + third + fourth; - const padding = 2000; //# account for slower machines - - return expect(duration).to.be.within(total, total + padding); - } + }) + + it('can clip element screenshots', () => { + cy.viewport(600, 200) + cy.visit('http://localhost:3322/element') + cy.get('.element') + .screenshot('element-clip', { + clip: { x: 25, y: 25, width: 160, height: 80 }, }) - ); - - it("ensures unique paths for non-named screenshots", function() { - cy.screenshot({ capture: "runner" }); - cy.screenshot({ capture: "runner" }); - cy.screenshot({ capture: "runner" }); - cy.readFile("cypress/screenshots/screenshots_spec.coffee/taking screenshots -- ensures unique paths for non-named screenshots.png", "base64"); - cy.readFile("cypress/screenshots/screenshots_spec.coffee/taking screenshots -- ensures unique paths for non-named screenshots (1).png", "base64"); - return cy.readFile("cypress/screenshots/screenshots_spec.coffee/taking screenshots -- ensures unique paths for non-named screenshots (2).png", "base64"); - }); - - it("ensures unique paths when there's a non-named screenshot and a failure", () => - cy.screenshot({ capture: "viewport" }).then(function() { - throw new Error("failing on purpose"); + + cy.task('check:screenshot:size', { + name: `${path.basename(__filename)}/element-clip.png`, + width: 160, + height: 80, + devicePixelRatio, + }) + }) + }) + + it('does not take a screenshot for a pending test', function () { + this.skip() + }) + + context('before hooks', () => { + before(() => { + // failure 2 + throw new Error('before hook failing') + }) + + it('empty test 1', () => {}) + }) + + context('each hooks', () => { + beforeEach(() => { + // failure 3 + throw new Error('before each hook failed') + }) + + afterEach(() => { + // failure 3 still (since associated only to a single test) + throw new Error('after each hook failed') + }) + + it('empty test 2', () => {}) + }) + + context(`really long test title ${Cypress._.repeat('a', 255)}`, () => { + it('takes a screenshot', () => { + cy.screenshot() + }) + + it('takes another screenshot', () => { + cy.screenshot() }) - ); - - describe("clipping", function() { - it("can clip app screenshots", () => - cy - .viewport(600, 200) - .visit("http://localhost:3322/color/yellow") - .screenshot("app-clip", { - capture: "viewport", clip: { x: 10, y: 10, width: 100, height: 50 } - }) - .task("check:screenshot:size", { - name: "screenshots_spec.coffee/app-clip.png", - width: 100, - height: 50, - devicePixelRatio - }) - ); - - it("can clip runner screenshots", () => - cy - .viewport(600, 200) - .visit("http://localhost:3322/color/yellow") - .screenshot("runner-clip", { - capture: "runner", clip: { x: 15, y: 15, width: 120, height: 60 } - }) - .task("check:screenshot:size", { - name: "screenshots_spec.coffee/runner-clip.png", - width: 120, - height: 60, - devicePixelRatio - }) - ); - - it("can clip fullPage screenshots", () => - cy - .viewport(600, 200) - .visit("http://localhost:3322/fullPage") - .screenshot("fullPage-clip", { - capture: "fullPage", clip: { x: 20, y: 20, width: 140, height: 70 } - }) - .task("check:screenshot:size", { - name: "screenshots_spec.coffee/fullPage-clip.png", - width: 140, - height: 70, - devicePixelRatio - }) - ); - - return it("can clip element screenshots", () => - cy - .viewport(600, 200) - .visit("http://localhost:3322/element") - .get(".element") - .screenshot("element-clip", { - clip: { x: 25, y: 25, width: 160, height: 80 } - }) - .task("check:screenshot:size", { - name: "screenshots_spec.coffee/element-clip.png", - width: 160, - height: 80, - devicePixelRatio - }) - ); - }); - - it("doesn't take a screenshot for a pending test", function() { - return this.skip(); - }); - - context("before hooks", function() { - before(function() { - //# failure 2 - throw new Error("before hook failing"); - }); - - return it("empty test 1", function() {}); - }); - - context("each hooks", function() { - beforeEach(function() { - //# failure 3 - throw new Error("before each hook failed"); - }); - - afterEach(function() { - //# failure 3 still (since associated only to a single test) - throw new Error("after each hook failed"); - }); - - return it("empty test 2", function() {}); - }); - - return context(`really long test title ${Cypress._.repeat('a', 255)}`, function() { - it("takes a screenshot", () => cy.screenshot()); - - return it("takes another screenshot", () => cy.screenshot()); - }); -}); + }) +}) diff --git a/packages/server/test/support/fixtures/projects/e2e/cypress/plugins/index.js b/packages/server/test/support/fixtures/projects/e2e/cypress/plugins/index.js index ed471c9b69e3..2d920c90a544 100644 --- a/packages/server/test/support/fixtures/projects/e2e/cypress/plugins/index.js +++ b/packages/server/test/support/fixtures/projects/e2e/cypress/plugins/index.js @@ -101,6 +101,7 @@ module.exports = (on) => { 'record:fast_visit_spec' ({ percentiles, url, browser, currentRetry }) { percentiles.forEach(([percent, percentile]) => { + // eslint-disable-next-line no-console console.log(`${percent}%\t of visits to ${url} finished in less than ${percentile}ms`) }) @@ -110,8 +111,9 @@ module.exports = (on) => { currentRetry, ...percentiles.reduce((acc, pair) => { acc[pair[0]] = pair[1] + return acc - }, {}) + }, {}), } return performance.track('fast_visit_spec percentiles', data) diff --git a/packages/server/test/support/fixtures/projects/e2e/cypress/support/foo/bar.js b/packages/server/test/support/fixtures/projects/e2e/cypress/support/foo/bar.js index be19a41deb62..80446ed767db 100644 --- a/packages/server/test/support/fixtures/projects/e2e/cypress/support/foo/bar.js +++ b/packages/server/test/support/fixtures/projects/e2e/cypress/support/foo/bar.js @@ -1 +1,2 @@ -console.log("bar") \ No newline at end of file +/* eslint-disable no-console */ +console.log('bar') diff --git a/packages/server/test/support/fixtures/projects/e2e/lib/bar.js b/packages/server/test/support/fixtures/projects/e2e/lib/bar.js index b6a1e32eb015..fec0747a32df 100644 --- a/packages/server/test/support/fixtures/projects/e2e/lib/bar.js +++ b/packages/server/test/support/fixtures/projects/e2e/lib/bar.js @@ -1,3 +1,3 @@ -import baz from "./baz" +import baz from './baz' -export default baz \ No newline at end of file +export default baz diff --git a/packages/server/test/support/fixtures/projects/e2e/lib/baz.js b/packages/server/test/support/fixtures/projects/e2e/lib/baz.js index 1ddc68846578..72796455a560 100644 --- a/packages/server/test/support/fixtures/projects/e2e/lib/baz.js +++ b/packages/server/test/support/fixtures/projects/e2e/lib/baz.js @@ -1,3 +1,3 @@ export default () => { - return "baz" -} \ No newline at end of file + return 'baz' +} diff --git a/packages/server/test/support/fixtures/projects/e2e/reporters/custom.js b/packages/server/test/support/fixtures/projects/e2e/reporters/custom.js index b8577e357f5e..af6bdba91c33 100644 --- a/packages/server/test/support/fixtures/projects/e2e/reporters/custom.js +++ b/packages/server/test/support/fixtures/projects/e2e/reporters/custom.js @@ -1,3 +1,4 @@ +/* eslint-disable no-console */ module.exports = function Reporter (runner) { runner.on('test end', function (test) { console.log(test.title) diff --git a/packages/server/test/support/fixtures/projects/ids/cypress/integration/bar.js b/packages/server/test/support/fixtures/projects/ids/cypress/integration/bar.js index 62dbdba0395b..1b64ff8bcc23 100644 --- a/packages/server/test/support/fixtures/projects/ids/cypress/integration/bar.js +++ b/packages/server/test/support/fixtures/projects/ids/cypress/integration/bar.js @@ -1,3 +1,3 @@ -context("some context[i9w]", function(){ +context('some context[i9w]', function () { it('tests[abc]') -}) \ No newline at end of file +}) diff --git a/packages/server/test/support/fixtures/projects/ids/cypress/integration/baz.js b/packages/server/test/support/fixtures/projects/ids/cypress/integration/baz.js index fb5b944d2aac..fd32a7c442d5 100644 --- a/packages/server/test/support/fixtures/projects/ids/cypress/integration/baz.js +++ b/packages/server/test/support/fixtures/projects/ids/cypress/integration/baz.js @@ -1 +1 @@ -import "./dom.jsx" \ No newline at end of file +import './dom.jsx' diff --git a/packages/server/test/support/fixtures/projects/multiple-task-registrations/cypress/plugins/index.js b/packages/server/test/support/fixtures/projects/multiple-task-registrations/cypress/plugins/index.js index 2979d06e965d..8c66f332da02 100644 --- a/packages/server/test/support/fixtures/projects/multiple-task-registrations/cypress/plugins/index.js +++ b/packages/server/test/support/fixtures/projects/multiple-task-registrations/cypress/plugins/index.js @@ -1,11 +1,19 @@ module.exports = (on) => { on('task', { - 'one' () { return 'one' }, - 'two' () { return 'two' }, + 'one' () { + return 'one' + }, + 'two' () { + return 'two' + }, }) on('task', { - 'two' () { return 'two again' }, - 'three' () { return 'three' }, + 'two' () { + return 'two again' + }, + 'three' () { + return 'three' + }, }) } diff --git a/packages/server/test/support/fixtures/projects/no-server/helpers/includes.js b/packages/server/test/support/fixtures/projects/no-server/helpers/includes.js index 03f300c4fa91..81969706f702 100644 --- a/packages/server/test/support/fixtures/projects/no-server/helpers/includes.js +++ b/packages/server/test/support/fixtures/projects/no-server/helpers/includes.js @@ -1,3 +1,3 @@ -beforeEach(function(){ +beforeEach(function () { -}); \ No newline at end of file +}) diff --git a/packages/server/test/support/fixtures/projects/no-server/my-tests/test1.js b/packages/server/test/support/fixtures/projects/no-server/my-tests/test1.js index d995e2b550f8..e3a5395eee36 100644 --- a/packages/server/test/support/fixtures/projects/no-server/my-tests/test1.js +++ b/packages/server/test/support/fixtures/projects/no-server/my-tests/test1.js @@ -1,3 +1,4 @@ -it("tests without a server", function(){ +/* eslint-disable mocha/no-global-tests */ +it('tests without a server', function () { -}); \ No newline at end of file +}) diff --git a/packages/server/test/support/fixtures/projects/plugin-extension/ext/background.js b/packages/server/test/support/fixtures/projects/plugin-extension/ext/background.js index aa1a7bbebb23..c8bd7e7d9306 100644 --- a/packages/server/test/support/fixtures/projects/plugin-extension/ext/background.js +++ b/packages/server/test/support/fixtures/projects/plugin-extension/ext/background.js @@ -1,5 +1,3 @@ -/* global document */ - const el = document.getElementById('extension') if (el) { diff --git a/packages/server/test/support/fixtures/projects/plugin-extension/ext/manifest.json b/packages/server/test/support/fixtures/projects/plugin-extension/ext/manifest.json index f99b8dac309a..68a6c0163a9c 100644 --- a/packages/server/test/support/fixtures/projects/plugin-extension/ext/manifest.json +++ b/packages/server/test/support/fixtures/projects/plugin-extension/ext/manifest.json @@ -3,16 +3,22 @@ "version": "0", "description": "tests adding user extension into Cypress", "permissions": [ - "tabs", "webNavigation", "" + "tabs", + "webNavigation", + "" ], "content_scripts": [ { - "matches": [""], + "matches": [ + "" + ], "exclude_matches": [ "*://*/__cypress/*", "*://*/__/*" ], - "js": ["background.js"], + "js": [ + "background.js" + ], "run_at": "document_end", "all_frames": true } diff --git a/packages/server/test/support/fixtures/projects/plugins-async-error/cypress/plugins/index.js b/packages/server/test/support/fixtures/projects/plugins-async-error/cypress/plugins/index.js index 30ae36270881..9689fd8b504b 100644 --- a/packages/server/test/support/fixtures/projects/plugins-async-error/cypress/plugins/index.js +++ b/packages/server/test/support/fixtures/projects/plugins-async-error/cypress/plugins/index.js @@ -1,5 +1,3 @@ -/* global Promise */ - module.exports = (on) => { on('file:preprocessor', () => { return new Promise(() => { diff --git a/packages/server/test/support/fixtures/projects/todos/tests/test1.js b/packages/server/test/support/fixtures/projects/todos/tests/test1.js index 06281e2ddb99..e5b9547cd65a 100644 --- a/packages/server/test/support/fixtures/projects/todos/tests/test1.js +++ b/packages/server/test/support/fixtures/projects/todos/tests/test1.js @@ -1,3 +1,4 @@ -it("is truthy", function(){ +/* eslint-disable mocha/no-global-tests */ +it('is truthy', function () { expect(true).to.be.true -}) \ No newline at end of file +}) diff --git a/packages/server/test/support/fixtures/projects/working-preprocessor/cypress/integration/another_spec.js b/packages/server/test/support/fixtures/projects/working-preprocessor/cypress/integration/another_spec.js index e67642893051..7bf33f67833f 100644 --- a/packages/server/test/support/fixtures/projects/working-preprocessor/cypress/integration/another_spec.js +++ b/packages/server/test/support/fixtures/projects/working-preprocessor/cypress/integration/another_spec.js @@ -1,5 +1,4 @@ -/* global it, expect */ - +/* eslint-disable mocha/no-global-tests */ it('is another spec', () => { expect(false).to.be.false }) diff --git a/packages/server/test/unit/config_spec.coffee b/packages/server/test/unit/config_spec.coffee index 8880304be789..57e4f20e47a6 100644 --- a/packages/server/test/unit/config_spec.coffee +++ b/packages/server/test/unit/config_spec.coffee @@ -3,10 +3,11 @@ require("../spec_helper") _ = require("lodash") path = require("path") R = require("ramda") -config = require("#{root}lib/config") -configUtil = require("#{root}lib/util/config") -scaffold = require("#{root}lib/scaffold") -settings = require("#{root}lib/util/settings") +config = require("#{root}lib/config") +errors = require("#{root}lib/errors") +configUtil = require("#{root}lib/util/config") +scaffold = require("#{root}lib/scaffold") +settings = require("#{root}lib/util/settings") describe "lib/config", -> beforeEach -> @@ -17,6 +18,27 @@ describe "lib/config", -> afterEach -> process.env = @env + context "environment name check", -> + it "throws an error for unknown CYPRESS_ENV", -> + sinon.stub(errors, "throw").withArgs("INVALID_CYPRESS_ENV", "foo-bar") + process.env.CYPRESS_ENV = "foo-bar" + cfg = { + projectRoot: "/foo/bar/" + } + options = {} + config.mergeDefaults(cfg, options) + expect(errors.throw).have.been.calledOnce + + it "allows known CYPRESS_ENV", -> + sinon.stub(errors, "throw") + process.env.CYPRESS_ENV = "test" + cfg = { + projectRoot: "/foo/bar/" + } + options = {} + config.mergeDefaults(cfg, options) + expect(errors.throw).not.to.be.called + context ".get", -> beforeEach -> @projectRoot = "/_test-output/path/to/project"