123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258 |
- """Integration tests for forms."""
- import functools
- import time
- from collections.abc import Generator
- import pytest
- from selenium.webdriver.common.by import By
- from selenium.webdriver.common.keys import Keys
- from reflex.testing import AppHarness
- from reflex.utils import format
- def FormSubmit(form_component):
- """App with a form using on_submit.
- Args:
- form_component: The str name of the form component to use.
- """
- import reflex as rx
- class FormState(rx.State):
- form_data: dict = {}
- var_options: list[str] = ["option3", "option4"]
- def form_submit(self, form_data: dict):
- self.form_data = form_data
- app = rx.App()
- @app.add_page
- def index():
- return rx.vstack(
- rx.input(
- value=FormState.router.session.client_token,
- is_read_only=True,
- id="token",
- ),
- eval(form_component)(
- rx.vstack(
- rx.input(id="name_input"),
- rx.checkbox(id="bool_input"),
- rx.switch(id="bool_input2"),
- rx.checkbox(id="bool_input3"),
- rx.switch(id="bool_input4"),
- rx.slider(id="slider_input", default_value=[50], width="100%"),
- rx.radio(["option1", "option2"], id="radio_input"),
- rx.radio(FormState.var_options, id="radio_input_var"),
- rx.select(
- ["option1", "option2"],
- name="select_input",
- default_value="option1",
- ),
- rx.select(FormState.var_options, id="select_input_var"),
- rx.text_area(id="text_area_input"),
- rx.input(
- id="debounce_input",
- debounce_timeout=0,
- on_change=rx.console_log,
- ),
- rx.button("Submit", type_="submit"),
- ),
- on_submit=FormState.form_submit,
- custom_attrs={"action": "/invalid"},
- ),
- rx.spacer(),
- height="100vh",
- )
- def FormSubmitName(form_component):
- """App with a form using on_submit.
- Args:
- form_component: The str name of the form component to use.
- """
- import reflex as rx
- class FormState(rx.State):
- form_data: dict = {}
- val: str = "foo"
- options: list[str] = ["option1", "option2"]
- def form_submit(self, form_data: dict):
- self.form_data = form_data
- app = rx.App()
- @app.add_page
- def index():
- return rx.vstack(
- rx.input(
- value=FormState.router.session.client_token,
- is_read_only=True,
- id="token",
- ),
- eval(form_component)(
- rx.vstack(
- rx.input(name="name_input"),
- rx.checkbox(name="bool_input"),
- rx.switch(name="bool_input2"),
- rx.checkbox(name="bool_input3"),
- rx.switch(name="bool_input4"),
- rx.slider(name="slider_input", default_value=[50], width="100%"),
- rx.radio(FormState.options, name="radio_input"),
- rx.select(
- FormState.options,
- name="select_input",
- default_value=FormState.options[0],
- ),
- rx.text_area(name="text_area_input"),
- rx.input(
- name="debounce_input",
- debounce_timeout=0,
- on_change=rx.console_log,
- ),
- rx.button("Submit", type_="submit"),
- rx.icon_button(rx.icon(tag="plus")),
- ),
- on_submit=FormState.form_submit,
- custom_attrs={"action": "/invalid"},
- ),
- rx.spacer(),
- height="100vh",
- )
- @pytest.fixture(
- scope="module",
- params=[
- functools.partial(FormSubmit, form_component="rx.form.root"),
- functools.partial(FormSubmitName, form_component="rx.form.root"),
- functools.partial(FormSubmit, form_component="rx.el.form"),
- functools.partial(FormSubmitName, form_component="rx.el.form"),
- ],
- ids=[
- "id-radix",
- "name-radix",
- "id-html",
- "name-html",
- ],
- )
- def form_submit(request, tmp_path_factory) -> Generator[AppHarness, None, None]:
- """Start FormSubmit app at tmp_path via AppHarness.
- Args:
- request: pytest request fixture
- tmp_path_factory: pytest tmp_path_factory fixture
- Yields:
- running AppHarness instance
- """
- param_id = request._pyfuncitem.callspec.id.replace("-", "_")
- with AppHarness.create(
- root=tmp_path_factory.mktemp("form_submit"),
- app_source=request.param,
- app_name=request.param.func.__name__ + f"_{param_id}",
- ) as harness:
- assert harness.app_instance is not None, "app is not running"
- yield harness
- @pytest.fixture
- def driver(form_submit: AppHarness):
- """GEt an instance of the browser open to the form_submit app.
- Args:
- form_submit: harness for ServerSideEvent app
- Yields:
- WebDriver instance.
- """
- driver = form_submit.frontend()
- try:
- yield driver
- finally:
- driver.quit()
- @pytest.mark.asyncio
- async def test_submit(driver, form_submit: AppHarness):
- """Fill a form with various different output, submit it to backend and verify
- the output.
- Args:
- driver: selenium WebDriver open to the app
- form_submit: harness for FormSubmit app
- """
- assert form_submit.app_instance is not None, "app is not running"
- by = By.ID if form_submit.app_source is FormSubmit else By.NAME
- # get a reference to the connected client
- token_input = driver.find_element(By.ID, "token")
- assert token_input
- # wait for the backend connection to send the token
- token = form_submit.poll_for_value(token_input)
- assert token
- name_input = driver.find_element(by, "name_input")
- name_input.send_keys("foo")
- checkbox_input = driver.find_element(By.XPATH, "//button[@role='checkbox']")
- checkbox_input.click()
- switch_input = driver.find_element(By.XPATH, "//button[@role='switch']")
- switch_input.click()
- radio_buttons = driver.find_elements(By.XPATH, "//button[@role='radio']")
- radio_buttons[1].click()
- textarea_input = driver.find_element(By.TAG_NAME, "textarea")
- textarea_input.send_keys("Some", Keys.ENTER, "Text")
- debounce_input = driver.find_element(by, "debounce_input")
- debounce_input.send_keys("bar baz")
- time.sleep(1)
- prev_url = driver.current_url
- submit_input = driver.find_element(By.CLASS_NAME, "rt-Button")
- submit_input.click()
- state_name = form_submit.get_state_name("_form_state")
- full_state_name = form_submit.get_full_state_name(["_form_state"])
- async def get_form_data():
- return (
- (await form_submit.get_state(f"{token}_{full_state_name}"))
- .substates[state_name]
- .form_data
- )
- # wait for the form data to arrive at the backend
- form_data = await AppHarness._poll_for_async(get_form_data)
- assert isinstance(form_data, dict)
- form_data = format.collect_form_dict_names(form_data)
- print(form_data)
- assert form_data["name_input"] == "foo"
- assert form_data["bool_input"]
- assert form_data["bool_input2"]
- assert not form_data.get("bool_input3", False)
- assert not form_data.get("bool_input4", False)
- assert form_data["slider_input"] == "50"
- assert form_data["radio_input"] == "option2"
- assert form_data["select_input"] == "option1"
- assert form_data["text_area_input"] == "Some\nText"
- assert form_data["debounce_input"] == "bar baz"
- # submitting the form should NOT change the url (preventDefault on_submit event)
- assert driver.current_url == prev_url
|