driver.py 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179
  1. #!/usr/bin/env python
  2. import argparse
  3. import contextlib
  4. import logging
  5. import urllib.parse
  6. import shlex
  7. import sys
  8. import time
  9. import pathlib
  10. from typing import Optional
  11. from selenium import webdriver
  12. import selenium.common.exceptions
  13. from selenium.webdriver.common.by import By
  14. from selenium.webdriver.support.ui import WebDriverWait
  15. logger = logging.getLogger(__name__)
  16. class SDLSeleniumTestDriver:
  17. def __init__(self, server: str, test: str, arguments: list[str], browser: str, firefox_binary: Optional[str]=None, chrome_binary: Optional[str]=None):
  18. self. server = server
  19. self.test = test
  20. self.arguments = arguments
  21. self.chrome_binary = chrome_binary
  22. self.firefox_binary = firefox_binary
  23. self.driver = None
  24. self.stdout_printed = False
  25. self.failed_messages: list[str] = []
  26. self.return_code = None
  27. driver_contructor = None
  28. match browser:
  29. case "firefox":
  30. driver_contructor = webdriver.Firefox
  31. driver_options = webdriver.FirefoxOptions()
  32. if self.firefox_binary:
  33. driver_options.binary_location = self.firefox_binary
  34. case "chrome":
  35. driver_contructor = webdriver.Chrome
  36. driver_options = webdriver.ChromeOptions()
  37. if self.chrome_binary:
  38. driver_options.binary_location = self.chrome_binary
  39. if driver_contructor is None:
  40. raise ValueError(f"Invalid {browser=}")
  41. options = [
  42. "--headless",
  43. ]
  44. for o in options:
  45. driver_options.add_argument(o)
  46. logger.debug("About to create driver")
  47. self.driver = driver_contructor(options=driver_options)
  48. @property
  49. def finished(self):
  50. return len(self.failed_messages) > 0 or self.return_code is not None
  51. def __del__(self):
  52. if self.driver:
  53. self.driver.quit()
  54. @property
  55. def url(self):
  56. req = {
  57. "loghtml": "1",
  58. "SDL_ASSERT": "abort",
  59. }
  60. req.update({f"arg_{i}": a for i, a in enumerate(self.arguments, 1) })
  61. req_str = urllib.parse.urlencode(req)
  62. return f"{self.server}/{self.test}.html?{req_str}"
  63. @contextlib.contextmanager
  64. def _selenium_catcher(self):
  65. try:
  66. yield
  67. success = True
  68. except selenium.common.exceptions.UnexpectedAlertPresentException as e:
  69. # FIXME: switch context, verify text of dialog and answer "a" for abort
  70. wait = WebDriverWait(self.driver, timeout=2)
  71. try:
  72. alert = wait.until(lambda d: d.switch_to.alert)
  73. except selenium.common.exceptions.NoAlertPresentException:
  74. self.failed_messages.append(e.msg)
  75. return False
  76. self.failed_messages.append(alert)
  77. if "Assertion failure" in e.msg and "[ariA]" in e.msg:
  78. alert.send_keys("a")
  79. alert.accept()
  80. else:
  81. self.failed_messages.append(e.msg)
  82. success = False
  83. return success
  84. def get_stdout_and_print(self):
  85. if self.stdout_printed:
  86. return
  87. with self._selenium_catcher():
  88. div_terminal = self.driver.find_element(by=By.ID, value="terminal")
  89. assert div_terminal
  90. text = div_terminal.text
  91. print(text)
  92. self.stdout_printed = True
  93. def update_return_code(self):
  94. with self._selenium_catcher():
  95. div_process_quit = self.driver.find_element(by=By.ID, value="process-quit")
  96. if not div_process_quit:
  97. return
  98. if div_process_quit.text != "":
  99. try:
  100. self.return_code = int(div_process_quit.text)
  101. except ValueError:
  102. raise ValueError(f"process-quit element contains invalid data: {div_process_quit.text:r}")
  103. def loop(self):
  104. print(f"Connecting to \"{self.url}\"", file=sys.stderr)
  105. self.driver.get(url=self.url)
  106. self.driver.implicitly_wait(0.2)
  107. while True:
  108. self.update_return_code()
  109. if self.finished:
  110. break
  111. time.sleep(0.1)
  112. self.get_stdout_and_print()
  113. if not self.stdout_printed:
  114. self.failed_messages.append("Failed to get stdout/stderr")
  115. def main() -> int:
  116. parser = argparse.ArgumentParser(allow_abbrev=False, description="Selenium SDL test driver")
  117. parser.add_argument("--browser", default="firefox", choices=["firefox", "chrome"], help="browser")
  118. parser.add_argument("--server", default="http://localhost:8080", help="Server where SDL tests live")
  119. parser.add_argument("--verbose", action="store_true", help="Verbose logging")
  120. parser.add_argument("--chrome-binary", help="Chrome binary")
  121. parser.add_argument("--firefox-binary", help="Firefox binary")
  122. index_double_dash = sys.argv.index("--")
  123. if index_double_dash < 0:
  124. parser.error("Missing test arguments. Need -- <FILENAME> <ARGUMENTS>")
  125. driver_arguments = sys.argv[1:index_double_dash]
  126. test = pathlib.Path(sys.argv[index_double_dash+1]).name
  127. test_arguments = sys.argv[index_double_dash+2:]
  128. args = parser.parse_args(args=driver_arguments)
  129. logging.basicConfig(level=logging.DEBUG if args.verbose else logging.INFO)
  130. logger.debug("driver_arguments=%r test=%r test_arguments=%r", driver_arguments, test, test_arguments)
  131. sdl_test_driver = SDLSeleniumTestDriver(
  132. server=args.server,
  133. test=test,
  134. arguments=test_arguments,
  135. browser=args.browser,
  136. chrome_binary=args.chrome_binary,
  137. firefox_binary=args.firefox_binary,
  138. )
  139. sdl_test_driver.loop()
  140. rc = sdl_test_driver.return_code
  141. if sdl_test_driver.failed_messages:
  142. for msg in sdl_test_driver.failed_messages:
  143. print(f"FAILURE MESSAGE: {msg}", file=sys.stderr)
  144. if rc == 0:
  145. print(f"Test signaled success (rc=0) but a failure happened", file=sys.stderr)
  146. rc = 1
  147. sys.stdout.flush()
  148. logger.info("Exit code = %d", rc)
  149. return rc
  150. if __name__ == "__main__":
  151. raise SystemExit(main())