Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Documents changes to cy.origin, configuration for document.domain inj… #6058

Merged
merged 12 commits into from
Jan 3, 2025
Merged
47 changes: 31 additions & 16 deletions docs/api/commands/origin.mdx
Original file line number Diff line number Diff line change
Expand Up @@ -3,8 +3,8 @@ title: origin
e2eSpecific: true
---

Visit multiple domains of different
[origin](https://developer.mozilla.org/en-US/docs/Web/Security/Same-origin_policy#definition_of_an_origin)
Visit multiple different
[origins](https://developer.mozilla.org/en-US/docs/Web/Security/Same-origin_policy#definition_of_an_origin)
in a single test.

In normal use, a single Cypress test may only run commands in a single origin, a
Expand All @@ -28,6 +28,18 @@ doc.

:::

:::info

<strong>Changes in Cypress [v14.0.0](/app/references/changelog#14-0-0)</strong>

Cypress no longer injects `document.domain` by default, which means `cy.origin()`
must now be used to navigate between any two origins in the same test, even if
the two origins are in the same superdomain. This behavior can be disabled by setting
the `injectDocumentDomain` configuration option to `true`, to allow a smooth transition
of tests to the new behavior. In Cypress 15, this configuration option will be removed.

:::

## Syntax

```js
Expand Down Expand Up @@ -63,26 +75,28 @@ cy.get('h1').contains('My cool site under test')
const hits = getHits()
cy.visit('https://example.cypress.io/history/founder')
// To interact with cross-origin content, move this inside cy.origin() callback
cy.get('h1').contains('About our Founder')
// Domain must be a precise match including subdomain, i.e. example.cypress.io
cy.origin('cypress.io', () => {
cy.visit('/history/founder')
cy.get('h1').contains('About our Founder')
// Fails because hits is not passed in via args
cy.get('#hitcounter').contains(hits)
cy.get('h1').contains('Kitchen Sink')
// Origin must be a precise match including scheme, subdomain and port, i.e. https://www.cypress.io
cy.origin('https://www.cypress.io', () => {
cy.visit('/about-us')
cy.get('h1').contains('About us')
// Fails because doanloads is not passed in via args
cy.contains(downloads)
})
// Won't work because still on cypress.io
cy.get('h1').contains('My cool site under test')
// Won't work because still on www.cypress.io
cy.get('h1').contains('Kitchen Sink')
```

### Arguments

<Icon name="angle-right" /> **url _(String)_**
<Icon name="angle-right" /> **origin _(String)_**

A URL specifying the secondary origin in which the callback is to be executed.
This should at the very least contain a hostname, and may also include the
protocol, port number & path. The hostname must precisely match that of the
secondary origin, including all subdomains. Query params are not supported.
A string specifying the origin in which the callback is to be executed.
This should at the very least contain a hostname. It may also include the
scheme and port number. The path may be included, but is not necessary.
The hostname must precisely match that of the secondary origin, including
all subdomains. Query params are not supported. If no scheme is provided,
the scheme defaults to `https`.

This argument will be used in two ways:

Expand Down Expand Up @@ -542,6 +556,7 @@ inclusion in a future version of Cypress.

| Version | Changes |
| ------------------------------------------- | ----------------------------------------------------------------------------------------------------------- |
| [14.0.0](/app/references/changelog#14-0-0) | `cy.origin()` is now required when navigating between origins in the same test, rather than superdomains. |
| [12.6.0](/app/references/changelog#10-7-0) | Support for `Cypress.require()` added and support for CommonJS `require()` and ES module `import()` removed |
| [10.11.0](/app/references/changelog#10-7-0) | Support for CommonJS `require()` and ES module `import()` added and support for `Cypress.require()` removed |
| [10.7.0](/app/references/changelog#10-7-0) | Support for `Cypress.require()` added |
Expand Down
2 changes: 1 addition & 1 deletion docs/app/faq.mdx
Original file line number Diff line number Diff line change
Expand Up @@ -452,7 +452,7 @@ a remote page and does not resolve until all of the external resources complete
their loading phase. Because we expect your applications to observe differing
load times, this command's default timeout is set to 60000ms. If you visit an
invalid url or a
[second unique domain](/app/guides/cross-origin-testing#Different-superdomain-per-test-requires-cyorigin),
[second unique domain](/app/guides/cross-origin-testing#Different-origins-per-test-require-cyorigin),
Cypress will log a verbose yet friendly error message.

**_In CI, how do I make sure my server has started?_**
Expand Down
97 changes: 38 additions & 59 deletions docs/app/guides/cross-origin-testing.mdx
Original file line number Diff line number Diff line change
Expand Up @@ -31,9 +31,6 @@ and we are _mostly_ able to do this.

### What Cypress does under the hood

- Injects
[`document.domain`](https://developer.mozilla.org/en-US/docs/Web/API/Document/domain)
into `text/html` pages.
- Proxies all HTTP / HTTPS traffic.
- Changes the hosted URL to match that of the application under test.
- Uses the browser's internal APIs for network level traffic.
Expand All @@ -55,7 +52,7 @@ Cypress must assign and manage browser certificates to be able to
modify the traffic in real time. You'll notice Chrome display a warning that the 'SSL certificate does not
match'. This is expected behavior. Under the hood we act as our own CA
authority and issue certificates dynamically in order to intercept requests
otherwise impossible to access. We only do this for the superdomain currently
otherwise impossible to access. We only do this for the origin currently
under test, and bypass other traffic.

Note, that Cypress allows you to optionally specify CA / client certificate
Expand All @@ -72,39 +69,37 @@ It's important to note that although we do our **very best** to ensure your
application works normally inside of Cypress, there _are_ some limitations you
need to be aware of.

- [Different superdomain per test requires cy.origin command](#Different-superdomain-per-test-requires-cyorigin)
- [Different origins per test require cy.origin()](#Different-origins-per-test-require-cyorigin)
- [Cross-origin iframes are not supported](#Cross-origin-iframes)
- [Navigating from HTTPS to HTTP will error](#Insecure-Content)
- [Cypress requires that the URLs navigated to have the same port](#Same-port-per-test)

### Different superdomain per test requires cy.origin
### Different origins per test require `cy.origin()`

Cypress changes its own host URL to match that of your applications. With the
exception of `cy.origin`, Cypress requires that the URLs navigated to have the
[same superdomain](/app/guides/cross-origin-testing#Parts-of-a-URL) for the
[origin](https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Origin) for the
entirety of a single test.

If you attempt to visit two different superdomains, the `cy.origin` command must
If you attempt to visit two different origins, the `cy.origin` command must
be used to wrap Cypress commands of the second visited domain. Otherwise,
Cypress commands will timeout after the navigation and will eventually error.
This is because the commands that were expected to run on the second domain are
actually being run on the first domain.

Without `cy.origin`, you can visit different superdomains in _different_ tests,
but not in the _same_ test. Please read our
[Cross Origin Testing Guide](/app/guides/cross-origin-testing) for more
information.
Without `cy.origin()`, you can visit different origins in _different_ tests,
but not in the _same_ test.

#### Examples of test cases that will error without the use of `cy.origin`

1. [`.click()`](/api/commands/click) an `<a>` with an `href` to a different
superdomain with subsequent Cypress commands being run.
origin with subsequent Cypress commands being run.
2. [`.submit()`](/api/commands/submit) a `<form>` that causes your web server to
redirect you a different superdomain where additional Cypress commands are
redirect you to a different origin where additional Cypress commands are
run.
3. Issue a JavaScript redirect in your application, such as
`window.location.href = '...'`, to a different superdomain where additional
Cypress commands are run.
`window.location.href = '...'`, which navigates to a different origin
where additional Cypress commands are run.

In each of these situations, Cypress will lose the ability to automate your
application and will error via command timeout unless the `cy.origin` command is
Expand All @@ -113,17 +108,6 @@ used.
Read on to learn about
[working around these common problems](#Common-Workarounds).

#### What is a superdomain?

But what is same superdomain? It's actually very similar to that of same
origin! Two URLs have the same origin if the **protocol**, **port** (if specified), and
**host** match. Cypress automatically handles hosts of the same superdomain by
injecting the
[`document.domain`](https://developer.mozilla.org/en-US/docs/Web/API/Document/domain)
property into the visited `text/html` pages. This is why navigations without the
use of the [`cy.origin()`](/api/commands/origin) command are solely scoped to the
same superdomain.

<DocumentDomainWorkaround />

#### Parts of a URL
Expand All @@ -136,38 +120,33 @@ chart to help clarify the differences!
alt="Diagram showing the parts of a URL including href broken into protocol, host, pathname, hash and smaller components of hostname, port, path, and search with and example URL"
/>

Given the URLs below, all have the same superdomain compared to
`https://www.cypress.io`.

- `https://cypress.io`
- `https://docs.cypress.io`
- `https://example.cypress.io/commands/querying`
##### Origin

The URLs below, however, will have different superdomains/origins compared to
`https://www.cypress.io`.
An `origin` is comprised of a URL's `scheme`, `hostname`, _and_ `port`. Given the URLs
below, all have the same origin compared to `https://www.cypress.io`:

- `http://www.cypress.io` (Different protocol)
- `https://docs.cypress.io:81` (Different port)
- `https://www.auth0.com/` (Different host of different superdomain)
- `https://www.cypress.io/cloud`
- `https://www.cypress.io/app`

The `http://localhost` URLs differ if their ports are different. For example,
the `http://localhost:3000` URL is considered to be a different origin from the
`http://localhost:8080` URL.
But the following have different origins compared to `https://www.cypress.io`:

The rules are:
- `http://www.cypress.io` (different `scheme`)
- `https://docs.cypress.io` (different `hostname` due to the subdomain)
- `https://www.auth0.com` (different `hostname`)
- `https://www.cypress.io:81` (different `port`)

:::danger

<Icon name="exclamation-triangle" color="red" /> You **cannot** [visit](/api/commands/visit)
two domains of different superdomains in the same test and continue to interact with
the page without the use of the [`cy.origin()`](/api/commands/origin) command.
two different origins in the same test and continue to interact with the page without
the use of the [`cy.origin()`](/api/commands/origin) command.

:::

:::tip

<Icon name="check-circle" color="green" /> You **can** [visit](/api/commands/visit)
two or more domains of different origin in **different** tests without needing [`cy.origin()`](/api/commands/origin).
two or more origins in **different** tests without needing [`cy.origin()`](/api/commands/origin).
:::

For practical purposes, this means the following:
Expand All @@ -176,16 +155,16 @@ For practical purposes, this means the following:
// This test will run without error
it('navigates', () => {
cy.visit('https://www.cypress.io')
cy.visit('https://docs.cypress.io')
cy.visit('https://www.cypress.io/app')
cy.get('selector') // yup all good
})
```

```javascript
// this will error because cypress-dx.com doesn't match the cypress.io superdomain
// this will error because the origin https://docs.cypress.io doesn't match the origin https://www.cypress.io
it('navigates', () => {
cy.visit('https://www.cypress.io')
cy.visit('https://www.cypress-dx.com')
cy.visit('https://docs.cypress.io')
cy.get('selector')
})
```
Expand All @@ -196,8 +175,8 @@ the sequential command should run against:
```javascript
it('navigates', () => {
cy.visit('https://example.cypress.io')
cy.visit('https://www.cypress-dx.com')
cy.origin('https://www.cypress-dx.com', () => {
cy.visit('https://docs.cypress.io')
cy.origin('https://docs.cypress.io', () => {
cy.get('selector') // yup all good
})
})
Expand Down Expand Up @@ -306,7 +285,7 @@ and break down how to work around them in Cypress.
### External Navigation

The most common situation where you might encounter this error is when you click
on an `<a>` that navigates to another superdomain.
on an `<a>` that navigates to another origin.

```html title="index.html"
<html>
Expand All @@ -322,14 +301,14 @@ cy.get('selector').should('exist') // Cypress errors

:::warning

We don't recommend visiting a superdomain that you don't control in your tests
We don't recommend visiting an origin that you don't control in your tests
which you can read more about
[here](/app/core-concepts/best-practices#Visiting-External-Sites)

:::

If you control this superdomain, either by owning the hosted instance
or by other means, we recommend testing this superdomain with `cy.origin`.
If you control this origin, either by owning the hosted instance
or by other means, we recommend testing this origin with `cy.origin`.

```javascript title="test.cy.js"
cy.visit('http://localhost:8080') // where your web server + HTML is hosted
Expand All @@ -340,7 +319,7 @@ cy.origin('https://example.cypress.io', () => {
})
```

If you're not in control of this superdomain, we recommend you test that the `href`
If you're not in control of this origin, we recommend you test that the `href`
property is correct instead of performing the navigation. This will help lead to
more deterministic tests.

Expand Down Expand Up @@ -385,7 +364,7 @@ cy.get('form').submit() // submit the form!
```

If your back end server handling the `/submit` route does a `30x` redirect to a
different superdomain, you'll need to use the `cy.origin` command if running
different origin, you'll need to use the `cy.origin` command if running
additional Cypress commands after submitting the form.

```javascript title="routes.js"
Expand Down Expand Up @@ -533,7 +512,7 @@ Setting `chromeWebSecurity` to `false` in Chrome-based browsers allows you to do
the following:

- Display insecure content
- Navigate to any superdomain without cross-origin errors with or without
- Navigate to any origin without cross-origin errors with or without
`cy.origin`
- Access cross-origin iframes that are embedded in your application

Expand Down Expand Up @@ -586,7 +565,7 @@ Want to enable `experimentalModifyObstructiveThirdPartyCode`? Let's do it!

## Other workarounds

There are other ways of testing the interaction between two superdomains. The
There are other ways of testing the interaction between two origins. The
browser has a natural security barrier called `origin policy` this means that
state like `localStorage`, `cookies`, `service workers` and many other APIs are
not shared between them anyways. Cypress does offer APIs around `localStorage`,
Expand All @@ -596,7 +575,7 @@ As a best practice, you should not visit or interact with any website not under
your control.

If your organization uses Single Sign On (SSO) or OAuth then you might choose to
test a 3rd party service other than your superdomain, which can be tested with
test a 3rd party service other than your origin, which can be tested with
[`cy.origin()`](/api/commands/origin).

We've written several other guides specifically about handling this situation.
Expand Down
6 changes: 4 additions & 2 deletions docs/app/references/changelog.mdx
Original file line number Diff line number Diff line change
Expand Up @@ -136,6 +136,8 @@ _Released 11/19/2024_

## 13.15.2

## 13.15.2

_Released 11/5/2024_

**Bugfixes:**
Expand Down Expand Up @@ -1542,11 +1544,11 @@ _Released 1/24/2023_
will help us make decisions to improve memory issues. Addresses
[#23391](https://github.com/cypress-io/cypress/issues/23391).
- Added new
[`experimentalSkipDomainInjection`](/app/references/experiments#Experimental-Skip-Domain-Injection)
`experimentalSkipDomainInjection`
configuration option to disable Cypress from setting `document.domain` on
injection, allowing users to test Salesforce domains. If you believe you are
having `document.domain` issues, please see the
[`experimentalSkipDomainInjection`](/app/references/experiments#Experimental-Skip-Domain-Injection)
`experimentalSkipDomainInjection`
guide. This config option is end-to-end only. Addresses
[#2367](https://github.com/cypress-io/cypress/issues/2367),
[#23958](https://github.com/cypress-io/cypress/issues/23958),
Expand Down
Loading