From 075d996084c95be5a5b1ce1cc42fef5960aac1be Mon Sep 17 00:00:00 2001 From: Joshua Fehler Date: Fri, 23 Feb 2024 09:41:55 -0500 Subject: [PATCH] Start using a generic retry function --- splinter/driver/webdriver/__init__.py | 43 +++++++-------------------- splinter/retry.py | 33 +++++++++++++------- 2 files changed, 33 insertions(+), 43 deletions(-) diff --git a/splinter/driver/webdriver/__init__.py b/splinter/driver/webdriver/__init__.py index 07fd2ed1a..ff4b53557 100644 --- a/splinter/driver/webdriver/__init__.py +++ b/splinter/driver/webdriver/__init__.py @@ -200,7 +200,7 @@ def __repr__(self): ) -def _find(self, finder, finder_kwargs=None): +def _safe_find(finder, finder_kwargs=None): """Search for elements. Returns a list of results. Arguments: @@ -209,12 +209,10 @@ def _find(self, finder, finder_kwargs=None): Returns: list - """ finder_kwargs = finder_kwargs or {} - elements = None - elem_list = [] + elements = [] try: elements = finder(**finder_kwargs) @@ -225,14 +223,10 @@ def _find(self, finder, finder_kwargs=None): NoSuchElementException, StaleElementReferenceException, ): - # This exception is sometimes thrown if the page changes - # quickly + # Exception which can be thrown if the page isn't ready. pass - if elements: - elem_list = [self.element_class(element, self, finder_kwargs) for element in elements] - - return elem_list + return elements def find_by( @@ -249,26 +243,18 @@ def find_by( Returns: ElementList - """ - elem_list = [] - find_by = original_find or finder_kwargs["by"] query = original_query or finder_kwargs.get("value") - # Zero second wait time means only check once - if wait_time == 0: - elem_list = _find(self, finder, finder_kwargs) - else: - wait_time = wait_time or self.wait_time - end_time = time.time() + wait_time - - while time.time() < end_time: - elem_list = _find(self, finder, finder_kwargs) - - if elem_list: - break + elements = _retry( + _safe_find, + [finder], + {"finder_kwargs": finder_kwargs}, + timeout=self.wait_time, + ) + elem_list = [self.element_class(elem, self, finder_kwargs) for elem in elements] return ElementList(elem_list, find_by=find_by, query=query) @@ -466,7 +452,6 @@ def find_by_css(self, css_selector, wait_time=None): self.driver.find_elements, finder_kwargs={"by": By.CSS_SELECTOR, "value": css_selector}, original_find="css", - original_query=css_selector, wait_time=wait_time, ) @@ -490,7 +475,6 @@ def find_by_name(self, name, wait_time=None): return self.find_by( self.driver.find_elements, finder_kwargs={"by": By.NAME, "value": name}, - original_find="name", wait_time=wait_time, ) @@ -498,7 +482,6 @@ def find_by_tag(self, tag, wait_time=None): return self.find_by( self.driver.find_elements, finder_kwargs={"by": By.TAG_NAME, "value": tag}, - original_find="tag_name", wait_time=wait_time, ) @@ -526,7 +509,6 @@ def find_by_id(self, id, wait_time=None): # NOQA: A002 return self.find_by( self.driver.find_element, finder_kwargs={"by": By.ID, "value": id}, - original_find="id", wait_time=wait_time, ) @@ -951,7 +933,6 @@ def find_by_name(self, selector, wait_time=None): return self.find_by( self._element.find_elements, finder_kwargs={"by": By.NAME, "value": selector}, - original_find="name", wait_time=wait_time, ) @@ -959,7 +940,6 @@ def find_by_tag(self, selector, wait_time=None): return self.find_by( self._element.find_elements, finder_kwargs={"by": By.TAG_NAME, "value": selector}, - original_find="tag", wait_time=wait_time, ) @@ -990,7 +970,6 @@ def find_by_id(self, selector, wait_time=None): return self.find_by( self._element.find_elements, finder_kwargs={"by": By.ID, "value": selector}, - original_find="id", wait_time=wait_time, ) diff --git a/splinter/retry.py b/splinter/retry.py index 546611a44..a2057129b 100644 --- a/splinter/retry.py +++ b/splinter/retry.py @@ -9,24 +9,35 @@ def _retry( fn_args: Optional[list] = None, fn_kwargs: Optional[dict] = None, timeout: int = 0, -) -> bool: - """Retry a function that should return a truthy value until a timeout is hit. +) -> Any: + """Retry a function until it returns a non-falsey result or timeout is hit. + + If timeout is set to 0, the function will only be run once. + + This will not wrap Exceptions, only falsey values. Arguments: - fn: Function to retry - timeout: Number of seconds to retry. + fn: A function to retry. + timeout: How long, in seconds, to retry the function. Returns: - bool - True if the function returns a truthy value before the timeout, else False. - + The final return value of func. """ fn_args = fn_args or [] fn_kwargs = fn_kwargs or {} - end_time = time.time() + timeout + result = None - while time.time() < end_time: + # Zero second wait time means only check once + if timeout == 0: result = fn(*fn_args, **fn_kwargs) - if result: - return True - return False + else: + end_time = time.time() + timeout + + while time.time() < end_time: + result = fn(*fn_args, **fn_kwargs) + + if result: + break + + return result