Files
cypress/packages/server/lib/experiments.ts
Chris Breiding 41512c416a feat: Bundle cy.origin() dependencies at runtime (#25626)
Co-authored-by: cypress-bot[bot] <2f0651858c6e38e0+cypress-bot[bot]@users.noreply.github.com>
Co-authored-by: Ryan Manuel <ryanm@cypress.io>
2023-02-13 11:59:29 -05:00

162 lines
6.0 KiB
TypeScript

import { get } from 'lodash'
/**
* Returns a single string with human-readable experiments.
```
const experimental = getExperimentsFromResolved(config.resolved)
const enabledExperiments = _.pickBy(experimental, (experiment) => experiment.enabled)
formatExperiments(enabledExperiments)
// "componentsTesting=true,featureB=false"
```
*/
export const formatExperiments = (exp: CypressExperiments) => {
return Object.keys(exp).map((name) => `${name}=${exp[name].value}`).join(',')
}
type CypressProject = unknown
/**
* Single experimental feature. Experiment is enabled
* if its value is different from the default value (coming from the config).
*/
interface CypressExperiment {
enabled: boolean // is the experiment enabled
value: unknown // current value
key: string // usually the config key used to control the experiment
name: string // short name of the experiment
summary: string // one or two line experiment summary
}
/**
* Collection of Cypress experiments
*/
interface CypressExperiments {
[key: string]: CypressExperiment
}
interface StringValues {
[key: string]: string
}
/**
* Keeps summaries of experiments. Each summary is 1 - 2 sentences
* describing the purpose of the experiment.
* When adding an experiment, add its summary text here.
*
* @example
```
{
experimentalFetchPolyfill: 'Polyfills `window.fetch` to enable Network spying and stubbing.'
}
```
*/
const _summaries: StringValues = {
experimentalFetchPolyfill: 'Polyfills `window.fetch` to enable Network spying and stubbing.',
experimentalInteractiveRunEvents: 'Allows listening to the `before:run`, `after:run`, `before:spec`, and `after:spec` events in the plugins file during interactive mode.',
experimentalModifyObstructiveThirdPartyCode: 'Applies `modifyObstructiveCode` to third party `.html` and `.js`, removes subresource integrity, and modifies the user agent in Electron.',
experimentalSkipDomainInjection: 'Disables setting document.domain to the document\'s super domain on injection.',
experimentalSourceRewriting: 'Enables AST-based JS/HTML rewriting. This may fix issues caused by the existing regex-based JS/HTML replacement algorithm.',
experimentalSingleTabRunMode: 'Runs all component specs in a single tab, trading spec isolation for faster run mode execution.',
experimentalStudio: 'Generate and save commands directly to your test suite by interacting with your app as an end user would.',
experimentalWebKitSupport: 'Adds support for testing in the WebKit browser engine used by Safari. See https://on.cypress.io/webkit-experiment for more information.',
experimentalRunAllSpecs: 'Enables the "Run All Specs" UI feature, allowing the execution of multiple specs sequentially',
experimentalOriginDependencies: 'Enables support for `Cypress.require()` for including dependencies within the `cy.origin()` callback.',
experimentalMemoryManagement: 'Enables support for improved memory management within Chromium-based browsers.',
}
/**
* Keeps short names for experiments. When adding new experiments, add a short name.
* The name and summary will be shown in the Settings tab of the Desktop GUI.
* @example
```
{
experimentalFetchPolyfill: 'Fetch polyfill'
}
```
*/
const _names: StringValues = {
experimentalFetchPolyfill: 'Fetch Polyfill',
experimentalInteractiveRunEvents: 'Interactive Mode Run Events',
experimentalModifyObstructiveThirdPartyCode: 'Modify Obstructive Third Party Code',
experimentalSkipDomainInjection: 'Use Default document.domain',
experimentalSingleTabRunMode: 'Single Tab Run Mode',
experimentalSourceRewriting: 'Improved Source Rewriting',
experimentalStudio: 'Studio',
experimentalWebKitSupport: 'WebKit Support',
experimentalRunAllSpecs: 'Run All Specs',
experimentalOriginDependencies: 'Origin Dependencies',
experimentalMemoryManagement: 'Memory Management',
}
/**
* Export this object for easy stubbing from end-to-end tests.
* If you cannot easily pass "names" and "summaries" arguments
* to "getExperimentsFromResolved" function, then use this
* object to change "experiments.names" and "experimental.summaries" objects.
*/
export const experimental = {
names: _names,
summaries: _summaries,
}
export const getExperimentsFromResolved = (resolvedConfig, names = experimental.names, summaries = experimental.summaries): CypressExperiments => {
const experiments: CypressExperiments = {}
if (!resolvedConfig) {
// no config - no experiments
// this is likely to happen during unit testing
return experiments
}
const isExperimentKey = (key) => key.startsWith('experimental')
const experimentalKeys = Object.keys(resolvedConfig).filter(isExperimentKey)
experimentalKeys.forEach((key) => {
const name = get(names, key)
if (!name) {
// ignore unknown experiments
return
}
const summary = get(summaries, key, 'top secret')
// it would be nice to have default value in the resolved config
experiments[key] = {
key,
value: resolvedConfig[key].value,
enabled: resolvedConfig[key].from !== 'default',
name,
summary,
}
})
return experiments
}
/**
* Looks at the resolved config, finds all keys that start with "experimental" prefix
* and have non-default values and returns a simple object with {key: {value, enabled}}
* where "on" is set to true if the value is different from default..
*/
export const getExperiments = (project: CypressProject, names = experimental.names, summaries = experimental.summaries): CypressExperiments => {
const resolvedEnv = get(project, 'resolvedConfig', {})
return getExperimentsFromResolved(resolvedEnv, names, summaries)
}
/**
* Allow known experiments here to avoid accidentally showing
* any config key that starts with "experimental" prefix
*/
// @ts-ignore
export const isKnownExperiment = (experiment, key) => {
return Object.keys(experimental.names).includes(key)
}
// exporting a single default object with methods
// helps make it is to stub and to test
export default {
getExperiments,
}