run-tests.py 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329
  1. #!/usr/bin/python
  2. #
  3. # AP tests
  4. # Copyright (c) 2013, Jouni Malinen <j@w1.fi>
  5. #
  6. # This software may be distributed under the terms of the BSD license.
  7. # See README for more details.
  8. import os
  9. import re
  10. import sys
  11. import time
  12. from datetime import datetime
  13. import argparse
  14. import subprocess
  15. import logging
  16. logger = logging.getLogger()
  17. sys.path.append('../../wpaspy')
  18. from wpasupplicant import WpaSupplicant
  19. from hostapd import HostapdGlobal
  20. def reset_devs(dev, apdev):
  21. hapd = HostapdGlobal()
  22. for d in dev:
  23. try:
  24. d.reset()
  25. except Exception, e:
  26. logger.info("Failed to reset device " + d.ifname)
  27. print str(e)
  28. for ap in apdev:
  29. hapd.remove(ap['ifname'])
  30. def report(conn, build, commit, run, test, result, diff):
  31. if conn:
  32. if not build:
  33. build = ''
  34. if not commit:
  35. commit = ''
  36. sql = "INSERT INTO results(test,result,run,time,duration,build,commitid) VALUES(?, ?, ?, ?, ?, ?, ?)"
  37. params = (test, result, run, time.time(), diff.total_seconds(), build, commit)
  38. try:
  39. conn.execute(sql, params)
  40. conn.commit()
  41. except Exception, e:
  42. print "sqlite: " + str(e)
  43. print "sql: %r" % (params, )
  44. class DataCollector(object):
  45. def __init__(self, logdir, testname, tracing, dmesg):
  46. self._logdir = logdir
  47. self._testname = testname
  48. self._tracing = tracing
  49. self._dmesg = dmesg
  50. def __enter__(self):
  51. if self._tracing:
  52. output = os.path.join(self._logdir, '%s.dat' % (self._testname, ))
  53. self._trace_cmd = subprocess.Popen(['sudo', 'trace-cmd', 'record', '-o', output, '-e', 'mac80211', '-e', 'cfg80211', 'sh', '-c', 'echo STARTED ; read l'],
  54. stdin=subprocess.PIPE,
  55. stdout=subprocess.PIPE,
  56. stderr=open('/dev/null', 'w'),
  57. cwd=self._logdir)
  58. l = self._trace_cmd.stdout.read(7)
  59. while not 'STARTED' in l:
  60. l += self._trace_cmd.stdout.read(1)
  61. def __exit__(self, type, value, traceback):
  62. if self._tracing:
  63. self._trace_cmd.stdin.write('DONE\n')
  64. self._trace_cmd.wait()
  65. if self._dmesg:
  66. output = os.path.join(self._logdir, '%s.dmesg' % (self._testname, ))
  67. subprocess.call(['sudo', 'dmesg', '-c'], stdout=open(output, 'w'))
  68. def main():
  69. tests = []
  70. test_modules = []
  71. for t in os.listdir("."):
  72. m = re.match(r'(test_.*)\.py$', t)
  73. if m:
  74. logger.debug("Import test cases from " + t)
  75. mod = __import__(m.group(1))
  76. test_modules.append(mod.__name__.replace('test_', '', 1))
  77. for s in dir(mod):
  78. if s.startswith("test_"):
  79. func = mod.__dict__.get(s)
  80. tests.append(func)
  81. test_names = list(set([t.__name__.replace('test_', '', 1) for t in tests]))
  82. run = None
  83. print_res = False
  84. parser = argparse.ArgumentParser(description='hwsim test runner')
  85. parser.add_argument('--logdir', metavar='<directory>', default='logs',
  86. help='log output directory for all other options, ' +
  87. 'must be given if other log options are used')
  88. group = parser.add_mutually_exclusive_group()
  89. group.add_argument('-d', const=logging.DEBUG, action='store_const',
  90. dest='loglevel', default=logging.INFO,
  91. help="verbose debug output")
  92. group.add_argument('-q', const=logging.WARNING, action='store_const',
  93. dest='loglevel', help="be quiet")
  94. group.add_argument('-l', action='store_true', dest='logfile',
  95. help='store debug log to a file (in log directory)')
  96. parser.add_argument('-e', metavar="<filename>", dest='errorfile',
  97. nargs='?', const="failed",
  98. help='error filename (in log directory)')
  99. parser.add_argument('-r', metavar="<filename>", dest='resultsfile',
  100. nargs='?', const="results.txt",
  101. help='results filename (in log directory)')
  102. parser.add_argument('-S', metavar='<sqlite3 db>', dest='database',
  103. help='database to write results to')
  104. parser.add_argument('--commit', metavar='<commit id>',
  105. help='commit ID, only for database')
  106. parser.add_argument('-b', metavar='<build>', dest='build', help='build ID')
  107. parser.add_argument('-L', action='store_true', dest='update_tests_db',
  108. help='List tests (and update descriptions in DB)')
  109. parser.add_argument('-T', action='store_true', dest='tracing',
  110. help='collect tracing per test case (in log directory)')
  111. parser.add_argument('-D', action='store_true', dest='dmesg',
  112. help='collect dmesg per test case (in log directory)')
  113. parser.add_argument('-f', dest='testmodules', metavar='<test module>',
  114. help='execute only tests from these test modules',
  115. type=str, choices=[[]] + test_modules, nargs='+')
  116. parser.add_argument('tests', metavar='<test>', nargs='*', type=str,
  117. help='tests to run (only valid without -f)',
  118. choices=[[]] + test_names)
  119. args = parser.parse_args()
  120. if args.tests and args.testmodules:
  121. print 'Invalid arguments - both test module and tests given'
  122. sys.exit(2)
  123. if (args.logfile or args.errorfile or
  124. args.resultsfile or args.tracing):
  125. if not args.logdir:
  126. print 'Need --logdir for the given options'
  127. sys.exit(2)
  128. if args.logfile:
  129. logger.setLevel(logging.DEBUG)
  130. file_name = os.path.join(args.logdir, 'run-tests.log')
  131. log_handler = logging.FileHandler(file_name)
  132. fmt = "%(asctime)s %(levelname)s %(message)s"
  133. log_formatter = logging.Formatter(fmt)
  134. log_handler.setFormatter(log_formatter)
  135. logger.addHandler(log_handler)
  136. log_to_file = True
  137. else:
  138. logging.basicConfig(level=args.loglevel)
  139. log_handler = None
  140. log_to_file = False
  141. if args.loglevel == logging.WARNING:
  142. print_res = True
  143. error_file = args.errorfile and os.path.join(args.logdir, args.errorfile)
  144. results_file = args.resultsfile and os.path.join(args.logdir, args.resultsfile)
  145. if args.database:
  146. import sqlite3
  147. conn = sqlite3.connect(args.database)
  148. else:
  149. conn = None
  150. if conn:
  151. run = int(time.time())
  152. if args.update_tests_db:
  153. for t in tests:
  154. name = t.__name__.replace('test_', '', 1)
  155. print name + " - " + t.__doc__
  156. if conn:
  157. sql = 'INSERT OR REPLACE INTO tests(test,description) VALUES (?, ?)'
  158. params = (name, t.__doc__)
  159. try:
  160. conn.execute(sql, params)
  161. except Exception, e:
  162. print "sqlite: " + str(e)
  163. print "sql: %r" % (params,)
  164. if conn:
  165. conn.commit()
  166. conn.close()
  167. sys.exit(0)
  168. dev0 = WpaSupplicant('wlan0', '/tmp/wpas-wlan0')
  169. dev1 = WpaSupplicant('wlan1', '/tmp/wpas-wlan1')
  170. dev2 = WpaSupplicant('wlan2', '/tmp/wpas-wlan2')
  171. dev = [ dev0, dev1, dev2 ]
  172. apdev = [ ]
  173. apdev.append({"ifname": 'wlan3', "bssid": "02:00:00:00:03:00"})
  174. apdev.append({"ifname": 'wlan4', "bssid": "02:00:00:00:04:00"})
  175. for d in dev:
  176. if not d.ping():
  177. logger.info(d.ifname + ": No response from wpa_supplicant")
  178. return
  179. logger.info("DEV: " + d.ifname + ": " + d.p2p_dev_addr())
  180. for ap in apdev:
  181. logger.info("APDEV: " + ap['ifname'])
  182. passed = []
  183. skipped = []
  184. failed = []
  185. # make sure nothing is left over from previous runs
  186. # (if there were any other manual runs or we crashed)
  187. reset_devs(dev, apdev)
  188. if args.dmesg:
  189. subprocess.call(['sudo', 'dmesg', '-c'], stdout=open('/dev/null', 'w'))
  190. for t in tests:
  191. name = t.__name__.replace('test_', '', 1)
  192. if args.tests:
  193. if not name in args.tests:
  194. continue
  195. if args.testmodules:
  196. if not t.__module__.replace('test_', '', 1) in args.testmodules:
  197. continue
  198. if log_handler:
  199. log_handler.stream.close()
  200. logger.removeHandler(log_handler)
  201. file_name = os.path.join(args.logdir, name + '.log')
  202. log_handler = logging.FileHandler(file_name)
  203. log_handler.setFormatter(log_formatter)
  204. logger.addHandler(log_handler)
  205. with DataCollector(args.logdir, name, args.tracing, args.dmesg):
  206. logger.info("START " + name)
  207. if log_to_file:
  208. print "START " + name
  209. sys.stdout.flush()
  210. if t.__doc__:
  211. logger.info("Test: " + t.__doc__)
  212. start = datetime.now()
  213. for d in dev:
  214. try:
  215. d.request("NOTE TEST-START " + name)
  216. except Exception, e:
  217. logger.info("Failed to issue TEST-START before " + name + " for " + d.ifname)
  218. logger.info(e)
  219. print "FAIL " + name + " - could not start test"
  220. if conn:
  221. conn.close()
  222. conn = None
  223. sys.exit(1)
  224. try:
  225. if t.func_code.co_argcount > 1:
  226. res = t(dev, apdev)
  227. else:
  228. res = t(dev)
  229. end = datetime.now()
  230. diff = end - start
  231. if res == "skip":
  232. skipped.append(name)
  233. result = "SKIP"
  234. else:
  235. passed.append(name)
  236. result = "PASS"
  237. report(conn, args.build, args.commit, run, name, result, diff)
  238. result = result + " " + name + " "
  239. result = result + str(diff.total_seconds()) + " " + str(end)
  240. logger.info(result)
  241. if log_to_file or print_res:
  242. print result
  243. sys.stdout.flush()
  244. if results_file:
  245. f = open(results_file, 'a')
  246. f.write(result + "\n")
  247. f.close()
  248. except Exception, e:
  249. end = datetime.now()
  250. diff = end - start
  251. logger.info(e)
  252. failed.append(name)
  253. report(conn, args.build, args.commit, run, name, "FAIL", diff)
  254. result = "FAIL " + name + " " + str(diff.total_seconds()) + " " + str(end)
  255. logger.info(result)
  256. if log_to_file:
  257. print result
  258. sys.stdout.flush()
  259. if results_file:
  260. f = open(results_file, 'a')
  261. f.write(result + "\n")
  262. f.close()
  263. for d in dev:
  264. try:
  265. d.request("NOTE TEST-STOP " + name)
  266. except Exception, e:
  267. logger.info("Failed to issue TEST-STOP after " + name + " for " + d.ifname)
  268. logger.info(e)
  269. reset_devs(dev, apdev)
  270. if log_handler:
  271. log_handler.stream.close()
  272. logger.removeHandler(log_handler)
  273. file_name = os.path.join(args.logdir, 'run-tests.log')
  274. log_handler = logging.FileHandler(file_name)
  275. log_handler.setFormatter(log_formatter)
  276. logger.addHandler(log_handler)
  277. if conn:
  278. conn.close()
  279. if len(failed):
  280. logger.info("passed " + str(len(passed)) + " test case(s)")
  281. logger.info("skipped " + str(len(skipped)) + " test case(s)")
  282. logger.info("failed tests: " + str(failed))
  283. if error_file:
  284. f = open(error_file, 'w')
  285. f.write(str(failed) + '\n')
  286. f.close()
  287. sys.exit(1)
  288. logger.info("passed all " + str(len(passed)) + " test case(s)")
  289. if len(skipped):
  290. logger.info("skipped " + str(len(skipped)) + " test case(s)")
  291. if log_to_file:
  292. print "passed all " + str(len(passed)) + " test case(s)"
  293. if len(skipped):
  294. print "skipped " + str(len(skipped)) + " test case(s)"
  295. if __name__ == "__main__":
  296. main()