driver.py 6.2 KB

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