mirror of
https://github.com/cypress-io/cypress.git
synced 2026-05-07 23:40:21 -05:00
ed51bcbdda
Co-authored-by: Brian Mann <brian.mann86@gmail.com>
175 lines
5.2 KiB
JavaScript
175 lines
5.2 KiB
JavaScript
const _ = require('lodash')
|
|
const path = require('path')
|
|
const Promise = require('bluebird')
|
|
const cwd = require('../cwd')
|
|
const glob = require('../util/glob')
|
|
const debug = require('debug')('cypress:server:controllers')
|
|
const { escapeFilenameInUrl } = require('../util/escape_filename')
|
|
const { getCtx } = require('@packages/data-context')
|
|
|
|
module.exports = {
|
|
handleIframe (req, res, config, getRemoteState, extraOptions) {
|
|
const test = req.params[0]
|
|
const iframePath = cwd('lib', 'html', 'iframe.html')
|
|
const specFilter = _.get(extraOptions, 'specFilter')
|
|
|
|
debug('handle iframe %o', { test, specFilter })
|
|
|
|
return this.getSpecs(test, config, extraOptions)
|
|
.then((specs) => {
|
|
return this.getSupportFile(config)
|
|
.then((js) => {
|
|
const allFilesToSend = js.concat(specs)
|
|
|
|
debug('all files to send %o', _.map(allFilesToSend, 'relative'))
|
|
|
|
const iframeOptions = {
|
|
title: this.getTitle(test),
|
|
domain: getRemoteState().domainName,
|
|
scripts: JSON.stringify(allFilesToSend),
|
|
}
|
|
|
|
debug('iframe %s options %o', test, iframeOptions)
|
|
|
|
return res.render(iframePath, iframeOptions)
|
|
})
|
|
})
|
|
},
|
|
|
|
getSpecs (spec, config, extraOptions = {}) {
|
|
// when asking for all specs: spec = "__all"
|
|
// otherwise it is a relative spec filename like "integration/spec.js"
|
|
debug('get specs %o', { spec, extraOptions })
|
|
|
|
const convertSpecPath = (spec) => {
|
|
// get the absolute path to this spec and
|
|
// get the browser url + cache buster
|
|
const convertedSpec = path.join(config.projectRoot, spec)
|
|
|
|
debug('converted %s to %s', spec, convertedSpec)
|
|
|
|
return this.prepareForBrowser(convertedSpec, config.projectRoot, config.namespace)
|
|
}
|
|
|
|
const specFilter = _.get(extraOptions, 'specFilter')
|
|
|
|
debug('specFilter %o', { specFilter })
|
|
const specFilterContains = (spec) => {
|
|
// only makes sense if there is specFilter string
|
|
// the filter should match the logic in
|
|
// desktop-gui/src/specs/specs-store.js
|
|
return spec.relative.toLowerCase().includes(specFilter.toLowerCase())
|
|
}
|
|
const specFilterFn = specFilter ? specFilterContains : () => true
|
|
|
|
const getSpecsHelper = async () => {
|
|
// grab all of the specs if this is ci
|
|
if (spec === '__all') {
|
|
debug('returning all specs')
|
|
|
|
const ctx = getCtx()
|
|
|
|
const [e2ePatterns, componentPatterns] = await Promise.all([
|
|
ctx.project.specPatternsForTestingType(ctx.project.projectRoot, 'e2e'),
|
|
ctx.project.specPatternsForTestingType(ctx.project.projectRoot, 'component'),
|
|
])
|
|
|
|
// It's possible that the E2E pattern matches some component tests, for example
|
|
// e2e.specPattern: src/**/*.cy.ts
|
|
// component.specPattern: src/components/**/*.cy.ts
|
|
// in this case, we want to remove anything that matches
|
|
// - the component.specPattern
|
|
// - the e2e.ignoreSpecPattern
|
|
return ctx.project.findSpecs(config.projectRoot, 'e2e', e2ePatterns.specPattern, e2ePatterns.ignoreSpecPattern, componentPatterns.specPattern)
|
|
.then((specs) => {
|
|
debug('found __all specs %o', specs)
|
|
|
|
return specs
|
|
})
|
|
.filter(specFilterFn)
|
|
.then((specs) => {
|
|
debug('filtered __all specs %o', specs)
|
|
|
|
return specs
|
|
}).map(convertSpecPath)
|
|
}
|
|
|
|
debug('normalizing spec %o', { spec })
|
|
|
|
// normalize by sending in an array of 1
|
|
return [convertSpecPath(spec)]
|
|
}
|
|
|
|
return Promise
|
|
.try(() => {
|
|
return getSpecsHelper()
|
|
})
|
|
},
|
|
|
|
prepareForBrowser (filePath, projectRoot, namespace) {
|
|
const SPEC_URL_PREFIX = `/${namespace}/tests?p`
|
|
|
|
filePath = filePath.replace(SPEC_URL_PREFIX, '__CYPRESS_SPEC_URL_PREFIX__')
|
|
filePath = escapeFilenameInUrl(filePath).replace('__CYPRESS_SPEC_URL_PREFIX__', SPEC_URL_PREFIX)
|
|
const relativeFilePath = path.relative(projectRoot, filePath)
|
|
|
|
return {
|
|
absolute: filePath,
|
|
relative: relativeFilePath,
|
|
relativeUrl: this.getTestUrl(relativeFilePath, namespace),
|
|
}
|
|
},
|
|
|
|
getTestUrl (file, namespace) {
|
|
const url = `/${namespace}/tests?p=${file}`
|
|
|
|
debug('test url for file %o', { file, url })
|
|
|
|
return url
|
|
},
|
|
|
|
getTitle (test) {
|
|
if (test === '__all') {
|
|
return 'All Tests'
|
|
}
|
|
|
|
return test
|
|
},
|
|
|
|
getSupportFile (config) {
|
|
const { projectRoot, supportFile, namespace } = config
|
|
|
|
let files = []
|
|
|
|
if (supportFile !== false) {
|
|
files = [supportFile]
|
|
}
|
|
|
|
// TODO: there shouldn't be any reason
|
|
// why we need to re-map these. its due
|
|
// to the javascripts array but that should
|
|
// probably be mapped during the config
|
|
const paths = _.map(files, (file) => {
|
|
return path.resolve(projectRoot, file)
|
|
})
|
|
|
|
return Promise
|
|
.map(paths, (p) => {
|
|
// is the path a glob?
|
|
if (!glob.hasMagic(p)) {
|
|
return p
|
|
}
|
|
|
|
// handle both relative + absolute paths
|
|
// by simply resolving the path from projectRoot
|
|
p = path.resolve(projectRoot, p)
|
|
|
|
return glob(p, { nodir: true })
|
|
}).then(_.flatten)
|
|
.map((filePath) => {
|
|
return this.prepareForBrowser(filePath, projectRoot, namespace)
|
|
})
|
|
},
|
|
|
|
}
|