Assertions
Cypress bundles the popular Chai assertion library, as well as helpful extensions for Sinon and jQuery, bringing you dozens of powerful assertions for free.
This document is only a reference to every assertion Cypress supports.
If you're looking to understand how to use these assertions please read about assertions in our Introduction to Cypress guide.
Chai
https://github.com/chaijs/chaiThese chainers are available for BDD assertions (expect
/should
). Aliases
listed can be used interchangeably with their original chainer. You can see the
entire list of available BDD Chai assertions here.
Chainer | Example |
---|---|
not | .should('not.equal', 'Jane') expect(name).to.not.equal('Jane') |
deep | .should('deep.equal', { name: 'Jane' }) expect(obj).to.deep.equal({ name: 'Jane' }) |
nested | .should('have.nested.property', 'a.b[1]') .should('nested.include', {'a.b[1]': 'y'}) expect({a: {b: 'x'}}).to.have.nested.property('a.b') expect({a: {b: 'x'}}).to.nested.include({'a.b': 'x'}) |
ordered | .should('have.ordered.members', [1, 2]) expect([1, 2]).to.have.ordered.members([1, 2]) expect([1, 2]).not.to.have.ordered.members([2, 1]) |
any | .should('have.any.keys', 'age') expect(arr).to.have.any.keys('age') |
all | .should('have.all.keys', 'name', 'age') expect(arr).to.have.all.keys('name', 'age') |
a(type) Aliases: an | .should('be.a', 'string') expect('test').to.be.a('string') |
include(value) Aliases: contain, includes, contains | .should('include', 2) expect([1,2,3]).to.include(2) |
ok | .should('not.be.ok') expect(undefined).to.not.be.ok |
true | .should('be.true') expect(true).to.be.true |
false | .should('be.false') expect(false).to.be.false |
null | .should('be.null') expect(null).to.be.null |
undefined | .should('be.undefined') expect(undefined).to.be.undefined |
exist | .should('exist') expect(myVar).to.exist |
empty | .should('be.empty') expect([]).to.be.empty |
arguments Aliases: Arguments | .should('be.arguments') expect(arguments).to.be.arguments |
equal(value) Aliases: equals, eq | .should('equal', 42) expect(42).to.equal(42) |
deep.equal(value) | .should('deep.equal', { name: 'Jane' }) expect({ name: 'Jane' }).to.deep.equal({ name: 'Jane' }) |
eql(value) Aliases: eqls | .should('eql', { name: 'Jane' }) expect({ name: 'Jane' }).to.eql({ name: 'Jane' }) |
greaterThan(value) Aliases: gt, above | .should('be.greaterThan', 5) expect(10).to.be.greaterThan(5) |
least(value) Aliases: gte | .should('be.at.least', 10) expect(10).to.be.at.least(10) |
lessThan(value) Aliases: lt, below | .should('be.lessThan', 10) expect(5).to.be.lessThan(10) |
most(value) Aliases: lte | .should('have.length.of.at.most', 4) expect('test').to.have.length.of.at.most(4) |
within(start, finish) | .should('be.within', 5, 10) expect(7).to.be.within(5, 10) |
instanceOf(constructor) Aliases: instanceof | .should('be.instanceOf', Array) expect([1, 2, 3]).to.be.instanceOf(Array) |
property(name, [value]) | .should('have.property', 'name') expect(obj).to.have.property('name') |
deep.property(name, [value]) | .should('have.deep.property', 'tests[1]', 'e2e') expect(deepObj).to.have.deep.property('tests[1]', 'e2e') |
ownProperty(name) Aliases: haveOwnProperty, own.property | .should('have.ownProperty', 'length') expect('test').to.have.ownProperty('length') |
ownPropertyDescriptor(name) Aliases: haveOwnPropertyDescriptor | .should('have.ownPropertyDescriptor', 'a') expect({a: 1}).to.have.ownPropertyDescriptor('a') |
lengthOf(value) | .should('have.lengthOf', 4) expect('test').to.have.lengthOf(4) |
match(RegExp) Aliases: matches | .should('to.match', /^test/) expect('testing').to.match(/^test/) |
string(string) | .should('have.string', 'test') expect('testing').to.have.string('test') |
keys(key1, [key2], [...]) Aliases: key | .should('have.keys', 'pass', 'fail') expect({ pass: 1, fail: 2 }).to.have.keys('pass', 'fail') |
throw(constructor) Aliases: throws, Throw | .should('throw', Error) expect(fn).to.throw(Error) |
respondTo(method) Aliases: respondsTo | .should('respondTo', 'getName') expect(obj).to.respondTo('getName') |
itself | .should('itself.respondTo', 'getName') expect(Foo).itself.to.respondTo('bar') |
satisfy(method) Aliases: satisfies | .should('satisfy', (num) => num > 0) expect(1).to.satisfy((num) => num > 0) |
closeTo(expected, delta) Aliases: approximately | .should('be.closeTo', 1, 0.5) expect(1.5).to.be.closeTo(1, 0.5) |
members(set) | .should('include.members', [3, 2]) expect([1, 2, 3]).to.include.members([3, 2]) |
oneOf(values) | .should('be.oneOf', [1, 2, 3]) expect(2).to.be.oneOf([1,2,3]) |
change(function) Aliases: changes | .should('change', obj, 'val') expect(fn).to.change(obj, 'val') |
increase(function) Aliases: increases | .should('increase', obj, 'val') expect(fn).to.increase(obj, 'val') |
decrease(function) Aliases: decreases | .should('decrease', obj, 'val') expect(fn).to.decrease(obj, 'val') |
These getters are also available for BDD assertions. They don't actually do anything, but they enable you to write clear, english sentences.
Chainable getters |
---|
to , be , been , is , that , which , and , has , have , with , at , of , same |
Chai-jQuery
https://github.com/chaijs/chai-jqueryThese chainers are available when asserting about a DOM object.
You will commonly use these chainers after using DOM commands like:
cy.get()
, cy.contains()
, etc.
Chainers | Assertion |
---|---|
attr(name, [value]) | .should('have.attr', 'bar') expect($el).to.have.attr('foo', 'bar') |
prop(name, [value]) | .should('have.prop', 'disabled', false) expect($el).to.have.prop('disabled', false) |
css(name, [value]) | .should('have.css', 'background-color', 'rgb(0, 0, 0)') expect($el).to.have.css('background-color', 'rgb(0, 0, 0)') |
data(name, [value]) | .should('have.data', 'foo', 'bar') expect($el).to.have.data('foo', 'bar') |
class(className) | .should('have.class', 'foo') expect($el).to.have.class('foo') |
id(id) | .should('have.id', 'foo') expect($el).to.have.id('foo') |
html(html) | .should('have.html', 'I love testing') expect($el).to.have.html('with Cypress') |
text(text) | .should('have.text', 'I love testing') expect($el).to.have.text('with Cypress') |
value(value) | .should('have.value', 'test@dev.com') expect($el).to.have.value('test@dev.com') |
visible | .should('be.visible') expect($el).to.be.visible |
hidden | .should('be.hidden') expect($el).to.be.hidden |
selected | .should('be.selected') expect($option).not.to.be.selected |
checked | .should('be.checked') expect($input).not.to.be.checked |
focus[ed] | .should('have.focus') expect($input).not.to.be.focused expect($input).to.have.focus |
enabled | .should('be.enabled') expect($input).to.be.enabled |
disabled | .should('be.disabled') expect($input).to.be.disabled |
empty | .should('be.empty') expect($el).not.to.be.empty |
exist | .should('exist') expect($nonexistent).not.to.exist |
match(selector) | .should('match', ':empty') expect($emptyEl).to.match(':empty') |
contain(text) | .should('contain', 'text') expect($el).to.contain('text') |
descendants(selector) | .should('have.descendants', 'div') expect($el).to.have.descendants('div') |
Sinon-Chai
https://github.com/domenic/sinon-chaiThese chainers are used on assertions with cy.stub()
and
cy.spy()
.
Sinon.JS property/method | Assertion |
---|---|
called | .should('have.been.called') expect(spy).to.be.called |
callCount | .should('have.callCount', 3) expect(spy).to.have.callCount(n) |
calledOnce | .should('have.been.calledOnce') expect(spy).to.be.calledOnce |
calledTwice | .should('have.been.calledTwice') expect(spy).to.be.calledTwice |
calledThrice | .should('have.been.calledThrice') expect(spy).to.be.calledThrice |
calledBefore | .should('have.been.calledBefore', spy2) expect(spy1).to.be.calledBefore(spy2) |
calledAfter | .should('have.been.calledAfter', spy2) expect(spy1).to.be.calledAfter(spy2) |
calledWithNew | .should('have.been.calledWithNew') expect(spy).to.be.calledWithNew |
alwaysCalledWithNew | .should('have.always.been.calledWithNew') expect(spy).to.always.be.calledWithNew |
calledOn | .should('have.been.calledOn', context) expect(spy).to.be.calledOn(context) |
alwaysCalledOn | .should('have.always.been.calledOn', context) expect(spy).to.always.be.calledOn(context) |
calledWith | .should('have.been.calledWith', ...args) expect(spy).to.be.calledWith(...args) |
alwaysCalledWith | .should('have.always.been.calledWith', ...args) expect(spy).to.always.be.calledWith(...args) |
calledOnceWith | .should('have.been.calledOnceWith', ...args) expect(spy).to.be.calledOnceWith(...args) |
calledWithExactly | .should('have.been.calledWithExactly', ...args) expect(spy).to.be.calledWithExactly(...args) |
alwaysCalledWithExactly | .should('have.always.been.calledWithExactly', ...args) expect(spy).to.always.be.calledWithExactly(...args) |
calledOnceWithExactly | .should('have.been.calledOnceWithExactly', ...args) expect(spy).to.be.calledOnceWithExactly(...args) |
calledWithMatch | .should('have.been.calledWithMatch',...args) expect(spy).to.be.calledWithMatch(...args) |
alwaysCalledWithMatch | .should('have.always.been.calledWithMatch',...args) expect(spy).to.always.be.calledWithMatch(...args) |
returned | .should('have.returned', 'foo') expect(spy).to.have.returned(returnVal) |
alwaysReturned | .should('have.always.returned', 'foo') expect(spy).to.have.always.returned(returnVal) |
threw | .should('have.thrown', TypeError) expect(spy).to.have.thrown(errorObjOrErrorTypeStringOrNothing) |
alwaysThrew | .should('have.always.thrown', 'TypeError') expect(spy).to.have.always.thrown(errorObjOrErrorTypeStringOrNothing) |
Adding New Assertions
Because we are using chai
, that means you can extend it however you'd like.
Cypress will "just work" with new assertions added to chai
. You can:
- Write your own
chai
assertions as documented here. - npm install any existing
chai
library and import into your test file or support file.
Common Assertions
Here is a list of common element assertions. Notice how we use these assertions
(listed above) with .should()
. You may also want to
read about how Cypress retries
assertions.
Length
// retry until we find 3 matching <li.selected>
cy.get('li.selected').should('have.length', 3)
Class
// retry until this input does not have class disabled
cy.get('form').find('input').should('not.have.class', 'disabled')
Value
// retry until this textarea has the correct value
cy.get('textarea').should('have.value', 'foo bar baz')
Text Content
// assert the element's text content is exactly the given text
cy.get('[data-testid="user-name"]').should('have.text', 'Joe Smith')
// assert the element's text includes the given substring
cy.get('[data-testid="address"]').should('include.text', 'Atlanta')
// retry until this span does not contain 'click me'
cy.get('a').parent('span.help').should('not.contain', 'click me')
// the element's text should start with "Hello"
cy.get('[data-testid="greeting"]')
.invoke('text')
.should('match', /^Hello/)
// use cy.contains to find an element with its text
// matching the given regular expression
cy.contains('[data-testid="greeting"]', /^Hello/)
Tip: read about assertions against text with non-breaking space entities in How do I get an element's text contents?
Visibility
// retry until the element with
// data-testid "form-submit" is visible
cy.get('[data-testid="form-submit"]').should('be.visible')
// retry until the list item with
// text "write tests" is visible
cy.contains('[data-testid="todo"] li', 'write tests').should('be.visible')
Note: if there are multiple elements, the assertions be.visible
and
not.be.visible
act differently:
// retry until SOME elements are visible
cy.get('li').should('be.visible')
// retry until EVERY element is invisible
cy.get('li.hidden').should('not.be.visible')
Watch the short video "Multiple elements and should('be.visible') assertion" that shows how to correctly check the visibility of elements.
Existence
// retry until loading spinner no longer exists
cy.get('[data-testid="loading"]').should('not.exist')
State
// retry until our radio is checked
cy.get(':radio').should('be.checked')
CSS
// retry until element has matching css
cy.get('[data-testid="completed"]').should(
'have.css',
'text-decoration',
'line-through'
)
// retry while accordion css has the
// "display: none" property
cy.get('[data-testid="accordion"]').should('not.have.css', 'display', 'none')
Disabled property
<input type="text" data-testid="example-input" disabled />
cy.get('[data-testid="example-input"]')
.should('be.disabled')
// let's enable this element from the test
.invoke('prop', 'disabled', false)
cy.get('[data-testid="example-input"]')
// we can use "enabled" assertion
.should('be.enabled')
// or negate the "disabled" assertion
.and('not.be.disabled')
Negative assertions
There are positive and negative assertions. Examples of positive assertions are:
cy.get('[data-testid="todo-item"]')
.should('have.length', 2)
.and('have.class', 'completed')
The negative assertions have the "not" chainer prefixed to the assertion. Examples of negative assertions are:
cy.contains('first todo').should('not.have.class', 'completed')
cy.get('[data-testid="loading"]').should('not.be.visible')
False passing tests
Negative assertions may pass for reasons you weren't expecting. Let's say we want to test that a Todo list app adds a new Todo item after typing the Todo and pressing enter.
Positive assertions
When adding an element to the list and using a positive assertion, the test asserts a specific number of Todo items in our application.
The test below may still falsely pass if the application behaves unexpectedly, like adding a blank Todo, instead of adding the new Todo with the text "Write tests".
cy.get('[data-testid="todos"]').should('have.length', 2)
cy.get('[data-testid="new-todo"]').type('Write tests{enter}')
// using a positive assertion to check the
// exact number of items
cy.get('[data-testid="todos"]').should('have.length', 3)
Negative assertions
But when using a negative assertion in the test below, the test can falsely pass when the application behaves in multiple unexpected ways:
- The app deletes the entire list of Todo items instead of inserting the 3rd Todo
- The app deletes a Todo instead of adding a new Todo
- The app adds a blank Todo
cy.get('[data-testid="todos"]').should('have.length', 2)
cy.get('[data-testid="new-todo"]').type('Write tests{enter}')
// using negative assertion to check it's
// not a number of items
cy.get('[data-testid="todos"]').should('not.have.length', 2)
For more examples, please read the blog post Be Careful With Negative Assertions.
Should callback
If built-in assertions are not enough, you can write your own assertion function
and pass it as a callback to the .should()
command. Cypress will automatically
retry the callback function until it
passes or the command times out. See the
.should()
documentation.
<div class="main-abc123 heading-xyz987">Introduction</div>
cy.get('div').should(($div) => {
expect($div).to.have.length(1)
const className = $div[0].className
// className will be a string like "main-abc123 heading-xyz987"
expect(className).to.match(/heading-/)
})
Multiple assertions
You can attach multiple assertions to the same command.
<a
data-testid="assertions-link"
class="active"
href="https://on.cypress.io"
target="_blank"
>
Cypress Docs
</a>
cy.get('[data-testid="assertions-link"]')
.should('have.class', 'active')
.and('have.attr', 'href')
.and('include', 'cypress.io')
Note that all chained assertions will use the same reference to the original subject. For example, if you wanted to test a loading element that first appears and then disappears, the following WILL NOT WORK because the same element cannot be visible and invisible at the same time:
// ⛔️ DOES NOT WORK
cy.get('[data-testid="loading"]').should('be.visible').and('not.be.visible')
Instead you should split the assertions and re-query the element:
// ✅ THE CORRECT WAY
cy.get('[data-testid="loading"]').should('be.visible')
cy.get('[data-testid="loading"]').should('not.be.visible')