driver.py 6.3 KB

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