run-tests.py 8.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263
  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 logging
  15. logger = logging.getLogger(__name__)
  16. sys.path.append('../../wpaspy')
  17. from wpasupplicant import WpaSupplicant
  18. from hostapd import HostapdGlobal
  19. def reset_devs(dev, apdev):
  20. hapd = HostapdGlobal()
  21. for d in dev:
  22. try:
  23. d.reset()
  24. except Exception, e:
  25. logger.info("Failed to reset device " + d.ifname)
  26. print str(e)
  27. for ap in apdev:
  28. hapd.remove(ap['ifname'])
  29. def report(conn, build, commit, run, test, result, diff):
  30. if conn:
  31. if not build:
  32. build = ''
  33. if not commit:
  34. commit = ''
  35. sql = "INSERT INTO results(test,result,run,time,duration,build,commitid) VALUES(?, ?, ?, ?, ?, ?, ?)"
  36. params = (test.replace('test_', '', 1), result, run, time.time(), diff.total_seconds(), build, commit)
  37. try:
  38. conn.execute(sql, params)
  39. conn.commit()
  40. except Exception, e:
  41. print "sqlite: " + str(e)
  42. print "sql: %r" % (params, )
  43. def main():
  44. tests = []
  45. test_modules = []
  46. for t in os.listdir("."):
  47. m = re.match(r'(test_.*)\.py$', t)
  48. if m:
  49. logger.debug("Import test cases from " + t)
  50. mod = __import__(m.group(1))
  51. test_modules.append(mod.__name__)
  52. for s in dir(mod):
  53. if s.startswith("test_"):
  54. func = mod.__dict__.get(s)
  55. tests.append(func)
  56. test_names = list(set([t.__name__ for t in tests]))
  57. run = None
  58. commit = None
  59. print_res = False
  60. parser = argparse.ArgumentParser(description='hwsim test runner')
  61. group = parser.add_mutually_exclusive_group()
  62. group.add_argument('-d', const=logging.DEBUG, action='store_const',
  63. dest='loglevel', default=logging.INFO,
  64. help="verbose debug output")
  65. group.add_argument('-q', const=logging.WARNING, action='store_const',
  66. dest='loglevel', help="be quiet")
  67. group.add_argument('-l', metavar='<filename>', dest='logfile',
  68. help='debug log filename')
  69. parser.add_argument('-e', metavar="<filename>", dest='errorfile',
  70. help='error filename')
  71. parser.add_argument('-r', metavar="<filename>", dest='resultsfile',
  72. help='results filename')
  73. parser.add_argument('-S', metavar='<sqlite3 db>', dest='database',
  74. help='database to write results to')
  75. parser.add_argument('-b', metavar='<build>', dest='build', help='build ID')
  76. parser.add_argument('-L', action='store_true', dest='update_tests_db',
  77. help='List tests (and update descriptions in DB)')
  78. parser.add_argument('-f', dest='testmodule', metavar='<test module>',
  79. help='execute only tests from this test module',
  80. type=str, choices=[[]] + test_modules)
  81. parser.add_argument('tests', metavar='<test>', nargs='*', type=str,
  82. help='tests to run (only valid without -f)',
  83. choices=[[]] + test_names)
  84. args = parser.parse_args()
  85. if args.tests and args.testmodule:
  86. print 'Invalid arguments - both test module and tests given'
  87. sys.exit(2)
  88. if args.logfile:
  89. logging.basicConfig(filename=args.logfile,
  90. level=logging.DEBUG)
  91. log_to_file = True
  92. else:
  93. logging.basicConfig(level=args.loglevel)
  94. log_to_file = False
  95. if args.loglevel == logging.WARNING:
  96. print_res = True
  97. error_file = args.errorfile
  98. results_file = args.resultsfile
  99. if args.database:
  100. import sqlite3
  101. conn = sqlite3.connect(args.database)
  102. else:
  103. conn = None
  104. if conn:
  105. run = str(int(time.time()))
  106. try:
  107. with open("commit") as f:
  108. val = f.readlines()
  109. if len(val) > 0:
  110. commit = val[0].rstrip()
  111. except IOError:
  112. pass
  113. if args.update_tests_db:
  114. for t in tests:
  115. print t.__name__ + " - " + t.__doc__
  116. if conn:
  117. sql = 'INSERT OR REPLACE INTO tests(test,description) VALUES (?, ?)'
  118. params = (t.__name__.replace('test_', '', 1), t.__doc__)
  119. try:
  120. conn.execute(sql, params)
  121. except Exception, e:
  122. print "sqlite: " + str(e)
  123. print "sql: %r" % (params,)
  124. if conn:
  125. conn.commit()
  126. conn.close()
  127. sys.exit(0)
  128. dev0 = WpaSupplicant('wlan0', '/tmp/wpas-wlan0')
  129. dev1 = WpaSupplicant('wlan1', '/tmp/wpas-wlan1')
  130. dev2 = WpaSupplicant('wlan2', '/tmp/wpas-wlan2')
  131. dev = [ dev0, dev1, dev2 ]
  132. apdev = [ ]
  133. apdev.append({"ifname": 'wlan3', "bssid": "02:00:00:00:03:00"})
  134. apdev.append({"ifname": 'wlan4', "bssid": "02:00:00:00:04:00"})
  135. for d in dev:
  136. if not d.ping():
  137. logger.info(d.ifname + ": No response from wpa_supplicant")
  138. return
  139. logger.info("DEV: " + d.ifname + ": " + d.p2p_dev_addr())
  140. for ap in apdev:
  141. logger.info("APDEV: " + ap['ifname'])
  142. passed = []
  143. skipped = []
  144. failed = []
  145. for t in tests:
  146. if args.tests:
  147. if not t.__name__ in args.tests:
  148. continue
  149. if args.testmodule:
  150. if not t.__module__ == args.testmodule:
  151. continue
  152. reset_devs(dev, apdev)
  153. logger.info("START " + t.__name__)
  154. if log_to_file:
  155. print "START " + t.__name__
  156. sys.stdout.flush()
  157. if t.__doc__:
  158. logger.info("Test: " + t.__doc__)
  159. start = datetime.now()
  160. for d in dev:
  161. try:
  162. d.request("NOTE TEST-START " + t.__name__)
  163. except Exception, e:
  164. logger.info("Failed to issue TEST-START before " + t.__name__ + " for " + d.ifname)
  165. logger.info(e)
  166. print "FAIL " + t.__name__ + " - could not start test"
  167. if conn:
  168. conn.close()
  169. conn = None
  170. sys.exit(1)
  171. try:
  172. if t.func_code.co_argcount > 1:
  173. res = t(dev, apdev)
  174. else:
  175. res = t(dev)
  176. end = datetime.now()
  177. diff = end - start
  178. if res == "skip":
  179. skipped.append(t.__name__)
  180. result = "SKIP"
  181. else:
  182. passed.append(t.__name__)
  183. result = "PASS"
  184. report(conn, args.build, commit, run, t.__name__, result, diff)
  185. result = result + " " + t.__name__ + " "
  186. result = result + str(diff.total_seconds()) + " " + str(end)
  187. logger.info(result)
  188. if log_to_file or print_res:
  189. print result
  190. sys.stdout.flush()
  191. if results_file:
  192. f = open(results_file, 'a')
  193. f.write(result + "\n")
  194. f.close()
  195. except Exception, e:
  196. end = datetime.now()
  197. diff = end - start
  198. logger.info(e)
  199. failed.append(t.__name__)
  200. report(conn, args.build, commit, run, t.__name__, "FAIL", diff)
  201. result = "FAIL " + t.__name__ + " " + str(diff.total_seconds()) + " " + str(end)
  202. logger.info(result)
  203. if log_to_file:
  204. print result
  205. sys.stdout.flush()
  206. if results_file:
  207. f = open(results_file, 'a')
  208. f.write(result + "\n")
  209. f.close()
  210. for d in dev:
  211. try:
  212. d.request("NOTE TEST-STOP " + t.__name__)
  213. except Exception, e:
  214. logger.info("Failed to issue TEST-STOP after " + t.__name__ + " for " + d.ifname)
  215. logger.info(e)
  216. if not args.tests:
  217. reset_devs(dev, apdev)
  218. if conn:
  219. conn.close()
  220. if len(failed):
  221. logger.info("passed " + str(len(passed)) + " test case(s)")
  222. logger.info("skipped " + str(len(skipped)) + " test case(s)")
  223. logger.info("failed tests: " + str(failed))
  224. if error_file:
  225. f = open(error_file, 'w')
  226. f.write(str(failed) + '\n')
  227. f.close()
  228. sys.exit(1)
  229. logger.info("passed all " + str(len(passed)) + " test case(s)")
  230. if len(skipped):
  231. logger.info("skipped " + str(len(skipped)) + " test case(s)")
  232. if log_to_file:
  233. print "passed all " + str(len(passed)) + " test case(s)"
  234. if len(skipped):
  235. print "skipped " + str(len(skipped)) + " test case(s)"
  236. if __name__ == "__main__":
  237. main()