run-tests.py 7.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249
  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. import sqlite3
  13. from datetime import datetime
  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('" + test.replace('test_', '', 1) + "', '" + result + "', " + str(run) + ", " + str(time.time()) + ", " + str(diff.total_seconds()) + ", '" + build + "', '" + commit + "')"
  36. try:
  37. conn.execute(sql)
  38. conn.commit()
  39. except Exception, e:
  40. print "sqlite: " + str(e)
  41. print "sql: " + sql
  42. def main():
  43. test_file = None
  44. error_file = None
  45. log_file = None
  46. results_file = None
  47. conn = None
  48. run = None
  49. build = None
  50. commit = None
  51. idx = 1
  52. print_res = False
  53. if len(sys.argv) > 1 and sys.argv[1] == '-d':
  54. logging.basicConfig(level=logging.DEBUG)
  55. idx = idx + 1
  56. elif len(sys.argv) > 1 and sys.argv[1] == '-q':
  57. logging.basicConfig(level=logging.WARNING)
  58. print_res = True
  59. idx = idx + 1
  60. elif len(sys.argv) > 2 and sys.argv[1] == '-l':
  61. log_file = sys.argv[2]
  62. logging.basicConfig(filename=log_file,level=logging.DEBUG)
  63. idx = idx + 2
  64. else:
  65. logging.basicConfig(level=logging.INFO)
  66. while len(sys.argv) > idx:
  67. if len(sys.argv) > idx + 1 and sys.argv[idx] == '-e':
  68. error_file = sys.argv[idx + 1]
  69. idx = idx + 2
  70. elif len(sys.argv) > idx + 1 and sys.argv[idx] == '-r':
  71. results_file = sys.argv[idx + 1]
  72. idx = idx + 2
  73. elif len(sys.argv) > idx + 1 and sys.argv[idx] == '-f':
  74. test_file = sys.argv[idx + 1]
  75. idx = idx + 2
  76. elif len(sys.argv) > idx + 1 and sys.argv[idx] == '-S':
  77. conn = sqlite3.connect(sys.argv[idx + 1])
  78. idx = idx + 2
  79. elif len(sys.argv) > idx + 1 and sys.argv[idx] == '-b':
  80. build = sys.argv[idx + 1]
  81. idx = idx + 2
  82. else:
  83. break
  84. if conn:
  85. run = str(int(time.time()))
  86. try:
  87. with open("commit") as f:
  88. val = f.readlines()
  89. if len(val) > 0:
  90. commit = val[0].rstrip()
  91. except IOError:
  92. pass
  93. tests = []
  94. for t in os.listdir("."):
  95. m = re.match(r'(test_.*)\.py$', t)
  96. if m:
  97. if test_file and test_file not in t:
  98. continue
  99. logger.debug("Import test cases from " + t)
  100. mod = __import__(m.group(1))
  101. for s in dir(mod):
  102. if s.startswith("test_"):
  103. func = mod.__dict__.get(s)
  104. tests.append(func)
  105. if len(sys.argv) > idx and sys.argv[idx] == '-L':
  106. for t in tests:
  107. print t.__name__ + " - " + t.__doc__
  108. if conn:
  109. sql = 'INSERT OR REPLACE INTO tests(test,description) VALUES ("' + t.__name__.replace('test_', '', 1) + '", "' + t.__doc__ + '")';
  110. try:
  111. conn.execute(sql)
  112. except Exception, e:
  113. print "sqlite: " + str(e)
  114. print "sql: " + sql
  115. if conn:
  116. conn.commit()
  117. conn.close()
  118. sys.exit(0)
  119. if len(sys.argv) > idx:
  120. test_filter = sys.argv[idx]
  121. else:
  122. test_filter = None
  123. dev0 = WpaSupplicant('wlan0', '/tmp/wpas-wlan0')
  124. dev1 = WpaSupplicant('wlan1', '/tmp/wpas-wlan1')
  125. dev2 = WpaSupplicant('wlan2', '/tmp/wpas-wlan2')
  126. dev = [ dev0, dev1, dev2 ]
  127. apdev = [ ]
  128. apdev.append({"ifname": 'wlan3', "bssid": "02:00:00:00:03:00"})
  129. apdev.append({"ifname": 'wlan4', "bssid": "02:00:00:00:04:00"})
  130. for d in dev:
  131. if not d.ping():
  132. logger.info(d.ifname + ": No response from wpa_supplicant")
  133. return
  134. logger.info("DEV: " + d.ifname + ": " + d.p2p_dev_addr())
  135. for ap in apdev:
  136. logger.info("APDEV: " + ap['ifname'])
  137. passed = []
  138. skipped = []
  139. failed = []
  140. for t in tests:
  141. if test_filter:
  142. if test_filter != t.__name__:
  143. continue
  144. reset_devs(dev, apdev)
  145. logger.info("START " + t.__name__)
  146. if log_file:
  147. print "START " + t.__name__
  148. sys.stdout.flush()
  149. if t.__doc__:
  150. logger.info("Test: " + t.__doc__)
  151. start = datetime.now()
  152. for d in dev:
  153. try:
  154. d.request("NOTE TEST-START " + t.__name__)
  155. except Exception, e:
  156. logger.info("Failed to issue TEST-START before " + t.__name__ + " for " + d.ifname)
  157. logger.info(e)
  158. print "FAIL " + t.__name__ + " - could not start test"
  159. if conn:
  160. conn.close()
  161. conn = None
  162. sys.exit(1)
  163. try:
  164. if t.func_code.co_argcount > 1:
  165. res = t(dev, apdev)
  166. else:
  167. res = t(dev)
  168. end = datetime.now()
  169. diff = end - start
  170. if res == "skip":
  171. skipped.append(t.__name__)
  172. result = "SKIP"
  173. else:
  174. passed.append(t.__name__)
  175. result = "PASS"
  176. report(conn, build, commit, run, t.__name__, result, diff)
  177. result = result + " " + t.__name__ + " "
  178. result = result + str(diff.total_seconds()) + " " + str(end)
  179. logger.info(result)
  180. if log_file or print_res:
  181. print result
  182. sys.stdout.flush()
  183. if results_file:
  184. f = open(results_file, 'a')
  185. f.write(result + "\n")
  186. f.close()
  187. except Exception, e:
  188. end = datetime.now()
  189. diff = end - start
  190. logger.info(e)
  191. failed.append(t.__name__)
  192. report(conn, build, commit, run, t.__name__, "FAIL", diff)
  193. result = "FAIL " + t.__name__ + " " + str(diff.total_seconds()) + " " + str(end)
  194. logger.info(result)
  195. if log_file:
  196. print result
  197. sys.stdout.flush()
  198. if results_file:
  199. f = open(results_file, 'a')
  200. f.write(result + "\n")
  201. f.close()
  202. for d in dev:
  203. try:
  204. d.request("NOTE TEST-STOP " + t.__name__)
  205. except Exception, e:
  206. logger.info("Failed to issue TEST-STOP after " + t.__name__ + " for " + d.ifname)
  207. logger.info(e)
  208. if not test_filter:
  209. reset_devs(dev, apdev)
  210. if conn:
  211. conn.close()
  212. if len(failed):
  213. logger.info("passed " + str(len(passed)) + " test case(s)")
  214. logger.info("skipped " + str(len(skipped)) + " test case(s)")
  215. logger.info("failed tests: " + str(failed))
  216. if error_file:
  217. f = open(error_file, 'w')
  218. f.write(str(failed) + '\n')
  219. f.close()
  220. sys.exit(1)
  221. logger.info("passed all " + str(len(passed)) + " test case(s)")
  222. if len(skipped):
  223. logger.info("skipped " + str(len(skipped)) + " test case(s)")
  224. if log_file:
  225. print "passed all " + str(len(passed)) + " test case(s)"
  226. if len(skipped):
  227. print "skipped " + str(len(skipped)) + " test case(s)"
  228. if __name__ == "__main__":
  229. main()