harness.py 7.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235
  1. # SPDX-License-Identifier: Apache-2.0
  2. import re
  3. import os
  4. import subprocess
  5. from collections import OrderedDict
  6. import xml.etree.ElementTree as ET
  7. result_re = re.compile(".*(PASS|FAIL|SKIP) - (test_)?(.*) in")
  8. class Harness:
  9. GCOV_START = "GCOV_COVERAGE_DUMP_START"
  10. GCOV_END = "GCOV_COVERAGE_DUMP_END"
  11. FAULT = "ZEPHYR FATAL ERROR"
  12. RUN_PASSED = "PROJECT EXECUTION SUCCESSFUL"
  13. RUN_FAILED = "PROJECT EXECUTION FAILED"
  14. def __init__(self):
  15. self.state = None
  16. self.type = None
  17. self.regex = []
  18. self.matches = OrderedDict()
  19. self.ordered = True
  20. self.repeat = 1
  21. self.tests = {}
  22. self.id = None
  23. self.fail_on_fault = True
  24. self.fault = False
  25. self.capture_coverage = False
  26. self.next_pattern = 0
  27. self.record = None
  28. self.recording = []
  29. self.fieldnames = []
  30. self.ztest = False
  31. self.is_pytest = False
  32. def configure(self, instance):
  33. config = instance.testcase.harness_config
  34. self.id = instance.testcase.id
  35. if "ignore_faults" in instance.testcase.tags:
  36. self.fail_on_fault = False
  37. if config:
  38. self.type = config.get('type', None)
  39. self.regex = config.get('regex', [])
  40. self.repeat = config.get('repeat', 1)
  41. self.ordered = config.get('ordered', True)
  42. self.record = config.get('record', {})
  43. def process_test(self, line):
  44. if self.RUN_PASSED in line:
  45. if self.fault:
  46. self.state = "failed"
  47. else:
  48. self.state = "passed"
  49. if self.RUN_FAILED in line:
  50. self.state = "failed"
  51. if self.fail_on_fault:
  52. if self.FAULT == line:
  53. self.fault = True
  54. if self.GCOV_START in line:
  55. self.capture_coverage = True
  56. elif self.GCOV_END in line:
  57. self.capture_coverage = False
  58. class Console(Harness):
  59. def configure(self, instance):
  60. super(Console, self).configure(instance)
  61. if self.type == "one_line":
  62. self.pattern = re.compile(self.regex[0])
  63. elif self.type == "multi_line":
  64. self.patterns = []
  65. for r in self.regex:
  66. self.patterns.append(re.compile(r))
  67. def handle(self, line):
  68. if self.type == "one_line":
  69. if self.pattern.search(line):
  70. self.state = "passed"
  71. elif self.type == "multi_line" and self.ordered:
  72. if (self.next_pattern < len(self.patterns) and
  73. self.patterns[self.next_pattern].search(line)):
  74. self.next_pattern += 1
  75. if self.next_pattern >= len(self.patterns):
  76. self.state = "passed"
  77. elif self.type == "multi_line" and not self.ordered:
  78. for i, pattern in enumerate(self.patterns):
  79. r = self.regex[i]
  80. if pattern.search(line) and not r in self.matches:
  81. self.matches[r] = line
  82. if len(self.matches) == len(self.regex):
  83. self.state = "passed"
  84. if self.fail_on_fault:
  85. if self.FAULT in line:
  86. self.fault = True
  87. if self.GCOV_START in line:
  88. self.capture_coverage = True
  89. elif self.GCOV_END in line:
  90. self.capture_coverage = False
  91. if self.record:
  92. pattern = re.compile(self.record.get("regex", ""))
  93. match = pattern.search(line)
  94. if match:
  95. csv = []
  96. if not self.fieldnames:
  97. for k,v in match.groupdict().items():
  98. self.fieldnames.append(k)
  99. for k,v in match.groupdict().items():
  100. csv.append(v.strip())
  101. self.recording.append(csv)
  102. self.process_test(line)
  103. if self.state == "passed":
  104. self.tests[self.id] = "PASS"
  105. else:
  106. self.tests[self.id] = "FAIL"
  107. class Pytest(Harness):
  108. def configure(self, instance):
  109. super(Pytest, self).configure(instance)
  110. self.running_dir = instance.build_dir
  111. self.source_dir = instance.testcase.source_dir
  112. self.pytest_root = 'pytest'
  113. self.is_pytest = True
  114. config = instance.testcase.harness_config
  115. if config:
  116. self.pytest_root = config.get('pytest_root', 'pytest')
  117. def handle(self, line):
  118. ''' Test cases that make use of pytest more care about results given
  119. by pytest tool which is called in pytest_run(), so works of this
  120. handle is trying to give a PASS or FAIL to avoid timeout, nothing
  121. is writen into handler.log
  122. '''
  123. self.state = "passed"
  124. self.tests[self.id] = "PASS"
  125. def pytest_run(self, log_file):
  126. ''' To keep artifacts of pytest in self.running_dir, pass this directory
  127. by "--cmdopt". On pytest end, add a command line option and provide
  128. the cmdopt through a fixture function
  129. If pytest harness report failure, twister will direct user to see
  130. handler.log, this method writes test result in handler.log
  131. '''
  132. cmd = [
  133. 'pytest',
  134. '-s',
  135. os.path.join(self.source_dir, self.pytest_root),
  136. '--cmdopt',
  137. self.running_dir,
  138. '--junit-xml',
  139. os.path.join(self.running_dir, 'report.xml'),
  140. '-q'
  141. ]
  142. log = open(log_file, "a")
  143. outs = []
  144. errs = []
  145. with subprocess.Popen(cmd,
  146. stdout = subprocess.PIPE,
  147. stderr = subprocess.PIPE) as proc:
  148. try:
  149. outs, errs = proc.communicate()
  150. tree = ET.parse(os.path.join(self.running_dir, "report.xml"))
  151. root = tree.getroot()
  152. for child in root:
  153. if child.tag == 'testsuite':
  154. if child.attrib['failures'] != '0':
  155. self.state = "failed"
  156. elif child.attrib['skipped'] != '0':
  157. self.state = "skipped"
  158. elif child.attrib['errors'] != '0':
  159. self.state = "errors"
  160. else:
  161. self.state = "passed"
  162. except subprocess.TimeoutExpired:
  163. proc.kill()
  164. self.state = "failed"
  165. except ET.ParseError:
  166. self.state = "failed"
  167. except IOError:
  168. log.write("Can't access report.xml\n")
  169. self.state = "failed"
  170. if self.state == "passed":
  171. self.tests[self.id] = "PASS"
  172. log.write("Pytest cases passed\n")
  173. elif self.state == "skipped":
  174. self.tests[self.id] = "SKIP"
  175. log.write("Pytest cases skipped\n")
  176. log.write("Please refer report.xml for detail")
  177. else:
  178. self.tests[self.id] = "FAIL"
  179. log.write("Pytest cases failed\n")
  180. log.write("\nOutput from pytest:\n")
  181. log.write(outs.decode('UTF-8'))
  182. log.write(errs.decode('UTF-8'))
  183. log.close()
  184. class Test(Harness):
  185. RUN_PASSED = "PROJECT EXECUTION SUCCESSFUL"
  186. RUN_FAILED = "PROJECT EXECUTION FAILED"
  187. def handle(self, line):
  188. match = result_re.match(line)
  189. if match and match.group(2):
  190. name = "{}.{}".format(self.id, match.group(3))
  191. self.tests[name] = match.group(1)
  192. self.ztest = True
  193. self.process_test(line)
  194. if not self.ztest and self.state:
  195. if self.state == "passed":
  196. self.tests[self.id] = "PASS"
  197. else:
  198. self.tests[self.id] = "FAIL"
  199. class Ztest(Test):
  200. pass