Test framework for the Hologram web framework.
The entry point is visit/2, which initializes a page and expands its
template into a fully-resolved DOM that tests can make assertions against.
Summary
Functions
Trigger a blur event on an element.
Checks a checkbox by its associated label text.
Selects a radio button by its associated label.
Trigger a $click event on the element matching the given CSS selector.
Click on a button by its text.
Click on a link by its text.
Fill in an input or textarea by its label.
Trigger a focus event on an element.
Mount a component in isolation for testing.
Opens the current page HTML in the default browser.
"Reloads" the current page by revisiting it with the current params.
Selects an option in a <select> box by its label.
Triggers a $select event on a text input or textarea by its label selecting
the text given to option_text.
Unchecks a checkbox by its associated label.
Entry point to create a session.
Scopes all operations within the given block to descendants of the element
matching selector.
Scopes to the <article> whose first heading (h1 - h6) matches header.
Scopes to the <fieldset> whose <legend> matches legend.
Scopes to the <section> whose first heading (h1 - h6) matches header.
Assertions
Asserts that the session's DOM contains exactly one element matching the given CSS selector (and optional filters).
Asserts that we are on a specific page. Useful after redirect.
Asserts that the session's DOM does not contain any element matching the given CSS selector (and optional filters).
Functions
Trigger a blur event on an element.
Accepts the same options as Mirage.click/3.
@spec check(Mirage.Session.t(), String.t(), keyword()) :: Mirage.Session.t()
Checks a checkbox by its associated label text.
Triggers the input's $change as well as its form's $change event (if it
has one).
Matches exactly by default; pass exact: false to match substrings.
Raises if no matching radio button is found, or if more than one matches.
@spec choose(Mirage.Session.t(), String.t(), keyword()) :: Mirage.Session.t()
Selects a radio button by its associated label.
Triggers the input's $change event as well as its form's $change event (if
there is one).
Labels may wrap the input or reference it via a for/id pair.
Matches exactly by default; pass exact: false to match substrings.
Raises if no matching radio button is found, or if more than one matches.
Example
Profile
|> visit()
|> choose("robot")
|> assert_has("p", "Your gender is 'robot'")
@spec click(Mirage.Session.t(), String.t(), String.t() | keyword()) :: Mirage.Session.t()
Trigger a $click event on the element matching the given CSS selector.
Any actions or commands will be run. If the click triggers a page navigation, the new page will be loaded into the session.
Raises if no matching element with a $click handler is found, or if more
than one matches.
Examples
SignUpPage
|> visit()
|> fill_in("Name", with: "Bender")
|> fill_in("Password", with: "killallhumans")
|> click("button", "Submit")
|> assert_page(WelcomePage)
HomePage
|> visit()
|> click("button", "Log out")Options
:text- Match on the element's inner text.:exact- Set tofalseto match on a substring of an element's text. Default istruemeaning you must provide an exact match.
@spec click_button(Mirage.Session.t(), String.t(), keyword(any())) :: Mirage.Session.t()
Click on a button by its text.
If it's a button inside a form, it will trigger the form's $submit event.
This is otherwise short-hand for Mirage.click/3 with "button" as its selector.
@spec click_link(Mirage.Session.t(), String.t(), keyword(any())) :: Mirage.Session.t()
Click on a link by its text.
This is simply a short-hand for Mirage.click/3 with "a" as its selector.
@spec fill_in(Mirage.Session.t(), String.t(), keyword()) :: Mirage.Session.t()
Fill in an input or textarea by its label.
Finds an input by its associated label and triggers the input's $change
and as well as its form's $change event (if it has one).
Labels may be associated with their input either by wrapping the input
(<label>Name <input/></label>) or via a for attribute matching the
input's id.
Matches exactly by default; pass exact: false to match substrings.
Raises if no matching label is found, or if more than one matches.
Trigger a focus event on an element.
Accepts the same options as Mirage.click/3.
@spec mount(module(), props: map(), context: map()) :: Mirage.Session.t()
Mount a component in isolation for testing.
Takes a Hologram.Component module and an optional keyword list. Returns a
session that can be used with the rest of the Mirage API just like one
created by visit/2, but without a page or layout wrapper.
Options
:props— a map of props to pass to the component:context— a map of context values; props declared withfrom_contextwill be populated from this map
Example
MyApp.Counter
|> mount(props: %{cid: "counter"})
|> click("button", "Increment")
|> assert_has("span", "1")
@spec open_browser(Mirage.Session.t()) :: Mirage.Session.t()
Opens the current page HTML in the default browser.
session
|> fill_in("Name", with: "Philip")
|> open_browser()
|> assert_has("Philip")When using with a component (via Mirage.mount/2), the output will be wrapped
in a thin layout bringing in your app's styles.
@spec reload(Mirage.Session.t()) :: Mirage.Session.t()
"Reloads" the current page by revisiting it with the current params.
All client-side state (component state, checked radios, etc.) is reset, just like a real browser reload.
session
|> fill_in("Name", with: "Leela")
|> reload()
|> refute_has("input", value: "Leela")
@spec select(Mirage.Session.t(), String.t(), String.t(), keyword()) :: Mirage.Session.t()
Selects an option in a <select> box by its label.
Triggers the select's $change event with the option's value
attribute (defaulting to the option's inner text when no value attribute
is present).
Labels may wrap the select element or reference it via a for/id pair.
Works with multi-selects.
Matches exactly by default; pass exact: false to match substrings.
Raises if no matching label or option is found, or if more than one matches.
Example
EditProfilePage
|> visit()
|> select("Company", "Planet Express")
|> assert_has("p", "You work for 'Planet Express'")
@spec select_text(Mirage.Session.t(), String.t(), String.t() | keyword()) :: Mirage.Session.t()
Triggers a $select event on a text input or textarea by its label selecting
the text given to option_text.
When text is omitted, all text in the input is selected.
Raises if the label does not point to an input that accepts text (i.e. raises for checkboxes, radios, selects, and non-text input types).
Examples
session
|> fill_in("Bio", with: "I'm a bending unit. I bend girders.")
|> select_text("Bio", "girder")
session
|> fill_in("Bio", with: "My hobbies include smoking cigars, drinking, and killing all humans")
|> select_text("Bio")
@spec uncheck(Mirage.Session.t(), String.t(), keyword()) :: Mirage.Session.t()
Unchecks a checkbox by its associated label.
Trigger's the input's $change event as well as its form's $change event
(if it has one).
Matches exactly by default; pass exact: false to match substrings.
Raises if no matching radio button is found, or if more than one matches.
@spec visit(module(), map()) :: Mirage.Session.t()
Entry point to create a session.
Takes a Hologram.Page and, optional, any params. It returns a session which
the rest of Mirage can use.
@spec within(Mirage.Session.t(), String.t(), (Mirage.Session.t() -> Mirage.Session.t())) :: Mirage.Session.t()
Scopes all operations within the given block to descendants of the element
matching selector.
session
|> within(".sidebar", fn session ->
session
|> assert_has("a", "Home")
|> click_link("Home")
end)
@spec within_article(Mirage.Session.t(), String.t(), (Mirage.Session.t() -> Mirage.Session.t())) :: Mirage.Session.t()
Scopes to the <article> whose first heading (h1 - h6) matches header.
session
|> within_article("Blog Post", fn session ->
assert_has(session, "p", "Post content")
end)
@spec within_fieldset(Mirage.Session.t(), String.t(), (Mirage.Session.t() -> Mirage.Session.t())) :: Mirage.Session.t()
Scopes to the <fieldset> whose <legend> matches legend.
session
|> within_fieldset("Account", fn session ->
assert_has(session, "input#username")
end)
@spec within_section( Mirage.Session.t(), String.t(), String.t(), (Mirage.Session.t() -> Mirage.Session.t()) ) :: Mirage.Session.t()
Scopes to the <section> whose first heading (h1 - h6) matches header.
session
|> within_section("Settings", fn session ->
assert_has(session, "Send me update", "No")
end)This can also be used more generally when given a CSS selector as the second argument.
session
|> within_section("div[role=article]", "My header", fn session ->
assert_has(session, "p", "content")
end)
Assertions
Asserts that the session's DOM contains exactly one element matching the given CSS selector (and optional filters).
Raises if no element matches or if more than one element matches.
session
|> assert_has("button")
|> assert_has("h1", "Welcome")
|> assert_has("input#email", value: "bender@planetexpress.com")Options
:text— also require the element's inner text (trimmed) to equal this value:value— also require the element'svalueattribute to equal this value:at— match only the element at this 1-based position among all nodes matching the selector
@spec assert_page(Mirage.Session.t(), module()) :: Mirage.Session.t() | no_return()
Asserts that we are on a specific page. Useful after redirect.
Asserts that the session's DOM does not contain any element matching the given CSS selector (and optional filters).
session
|> refute_has(".error")
|> refute_has("p", text: "Deleted")Accepts the same options as assert_has/3.