test_wpas_config.py 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400
  1. # wpa_supplicant config file
  2. # Copyright (c) 2014, Jouni Malinen <j@w1.fi>
  3. #
  4. # This software may be distributed under the terms of the BSD license.
  5. # See README for more details.
  6. import logging
  7. logger = logging.getLogger()
  8. import os
  9. from wpasupplicant import WpaSupplicant
  10. import hostapd
  11. def check_config(config):
  12. with open(config, "r") as f:
  13. data = f.read()
  14. if "update_config=1\n" not in data:
  15. raise Exception("Missing update_config")
  16. if "device_name=name\n" not in data:
  17. raise Exception("Missing device_name")
  18. if "eapol_version=2\n" not in data:
  19. raise Exception("Missing eapol_version")
  20. if "ctrl_interface=DIR=/var/run/wpa_supplicant GROUP=" not in data:
  21. raise Exception("Missing ctrl_interface")
  22. if "blob-base64-foo={" not in data:
  23. raise Exception("Missing blob")
  24. if "cred={" not in data:
  25. raise Exception("Missing cred")
  26. if "network={" not in data:
  27. raise Exception("Missing network")
  28. if "wps_priority=5\n" not in data:
  29. raise Exception("Missing wps_priority")
  30. if "ip_addr_go=192.168.1.1\n" not in data:
  31. raise Exception("Missing ip_addr_go")
  32. if "ip_addr_mask=255.255.255.0\n" not in data:
  33. raise Exception("Missing ip_addr_mask")
  34. if "ip_addr_start=192.168.1.10\n" not in data:
  35. raise Exception("Missing ip_addr_start")
  36. if "ip_addr_end=192.168.1.20\n" not in data:
  37. raise Exception("Missing ip_addr_end")
  38. return data
  39. def test_wpas_config_file(dev, apdev, params):
  40. """wpa_supplicant config file parsing/writing"""
  41. config = os.path.join(params['logdir'], 'wpas_config_file.conf')
  42. if os.path.exists(config):
  43. try:
  44. os.remove(config)
  45. except:
  46. pass
  47. try:
  48. os.rmdir(config)
  49. except:
  50. pass
  51. wpas = WpaSupplicant(global_iface='/tmp/wpas-wlan5')
  52. try:
  53. wpas.interface_add("wlan5", config=config)
  54. initialized = True
  55. except:
  56. initialized = False
  57. if initialized:
  58. raise Exception("Missing config file did not result in an error")
  59. try:
  60. with open(config, "w") as f:
  61. f.write("update_config=1 \t\r\n")
  62. f.write("# foo\n")
  63. f.write("\n")
  64. f.write(" \t\reapol_version=2")
  65. for i in range(0, 100):
  66. f.write(" ")
  67. f.write("foo\n")
  68. f.write("device_name=name#foo\n")
  69. wpas.interface_add("wlan5", config=config)
  70. wpas.request("SET wps_priority 5")
  71. id = wpas.add_network()
  72. wpas.set_network_quoted(id, "ssid", "foo")
  73. wpas.set_network_quoted(id, "psk", "12345678")
  74. wpas.set_network(id, "bssid", "00:11:22:33:44:55")
  75. wpas.set_network(id, "proto", "RSN")
  76. wpas.set_network(id, "key_mgmt", "WPA-PSK-SHA256")
  77. wpas.set_network(id, "pairwise", "CCMP")
  78. wpas.set_network(id, "group", "CCMP")
  79. wpas.set_network(id, "auth_alg", "OPEN")
  80. id = wpas.add_cred()
  81. wpas.set_cred(id, "priority", "3")
  82. wpas.set_cred(id, "sp_priority", "6")
  83. wpas.set_cred(id, "update_identifier", "4")
  84. wpas.set_cred(id, "ocsp", "1")
  85. wpas.set_cred(id, "eap", "TTLS")
  86. wpas.set_cred(id, "req_conn_capab", "6:1234")
  87. wpas.set_cred_quoted(id, "realm", "example.com")
  88. wpas.set_cred_quoted(id, "provisioning_sp", "example.com")
  89. wpas.set_cred_quoted(id, "domain", "example.com")
  90. wpas.set_cred_quoted(id, "domain_suffix_match", "example.com")
  91. wpas.set_cred(id, "roaming_consortium", "112233")
  92. wpas.set_cred(id, "required_roaming_consortium", "112233")
  93. wpas.set_cred_quoted(id, "roaming_partner",
  94. "roaming.example.net,1,127,*")
  95. wpas.set_cred_quoted(id, "ca_cert", "/tmp/ca.pem")
  96. wpas.set_cred_quoted(id, "username", "user")
  97. wpas.set_cred_quoted(id, "password", "secret")
  98. ev = wpas.wait_event(["CRED-MODIFIED 0 password"])
  99. wpas.request("SET blob foo 12345678")
  100. wpas.request("SET ip_addr_go 192.168.1.1")
  101. wpas.request("SET ip_addr_mask 255.255.255.0")
  102. wpas.request("SET ip_addr_start 192.168.1.10")
  103. wpas.request("SET ip_addr_end 192.168.1.20")
  104. if "OK" not in wpas.request("SAVE_CONFIG"):
  105. raise Exception("Failed to save configuration file")
  106. if "OK" not in wpas.global_request("SAVE_CONFIG"):
  107. raise Exception("Failed to save configuration file")
  108. wpas.interface_remove("wlan5")
  109. data1 = check_config(config)
  110. wpas.interface_add("wlan5", config=config)
  111. if len(wpas.list_networks()) != 1:
  112. raise Exception("Unexpected number of networks")
  113. if len(wpas.request("LIST_CREDS").splitlines()) != 2:
  114. raise Exception("Unexpected number of credentials")
  115. if "OK" not in wpas.request("SAVE_CONFIG"):
  116. raise Exception("Failed to save configuration file")
  117. data2 = check_config(config)
  118. if data1 != data2:
  119. logger.debug(data1)
  120. logger.debug(data2)
  121. raise Exception("Unexpected configuration change")
  122. wpas.request("SET update_config 0")
  123. wpas.global_request("SET update_config 0")
  124. if "OK" in wpas.request("SAVE_CONFIG"):
  125. raise Exception("SAVE_CONFIG succeeded unexpectedly")
  126. if "OK" in wpas.global_request("SAVE_CONFIG"):
  127. raise Exception("SAVE_CONFIG (global) succeeded unexpectedly")
  128. # replace the config file with a directory to break writing/renaming
  129. os.remove(config)
  130. os.mkdir(config)
  131. wpas.request("SET update_config 1")
  132. wpas.global_request("SET update_config 1")
  133. if "OK" in wpas.request("SAVE_CONFIG"):
  134. raise Exception("SAVE_CONFIG succeeded unexpectedly")
  135. if "OK" in wpas.global_request("SAVE_CONFIG"):
  136. raise Exception("SAVE_CONFIG (global) succeeded unexpectedly")
  137. finally:
  138. try:
  139. os.rmdir(config)
  140. except:
  141. pass
  142. def test_wpas_config_file_wps(dev, apdev):
  143. """wpa_supplicant config file parsing/writing with WPS"""
  144. config = "/tmp/test_wpas_config_file.conf"
  145. if os.path.exists(config):
  146. os.remove(config)
  147. params = { "ssid": "test-wps", "eap_server": "1", "wps_state": "2",
  148. "skip_cred_build": "1", "extra_cred": "wps-ctrl-cred" }
  149. hapd = hostapd.add_ap(apdev[0]['ifname'], params)
  150. wpas = WpaSupplicant(global_iface='/tmp/wpas-wlan5')
  151. try:
  152. with open(config, "w") as f:
  153. f.write("update_config=1\n")
  154. wpas.interface_add("wlan5", config=config)
  155. hapd.request("WPS_PIN any 12345670")
  156. wpas.scan_for_bss(apdev[0]['bssid'], freq="2412")
  157. wpas.request("WPS_PIN " + apdev[0]['bssid'] + " 12345670")
  158. ev = wpas.wait_event(["WPS-FAIL"], timeout=10)
  159. if ev is None:
  160. raise Exception("WPS-FAIL event timed out")
  161. with open(config, "r") as f:
  162. data = f.read()
  163. logger.info("Configuration file contents: " + data)
  164. if "network=" in data:
  165. raise Exception("Unexpected network block in configuration data")
  166. finally:
  167. try:
  168. os.remove(config)
  169. except:
  170. pass
  171. try:
  172. os.remove(config + ".tmp")
  173. except:
  174. pass
  175. try:
  176. os.rmdir(config)
  177. except:
  178. pass
  179. def test_wpas_config_file_wps2(dev, apdev):
  180. """wpa_supplicant config file parsing/writing with WPS (2)"""
  181. config = "/tmp/test_wpas_config_file.conf"
  182. if os.path.exists(config):
  183. os.remove(config)
  184. params = { "ssid": "test-wps", "eap_server": "1", "wps_state": "2",
  185. "skip_cred_build": "1", "extra_cred": "wps-ctrl-cred2" }
  186. hapd = hostapd.add_ap(apdev[0]['ifname'], params)
  187. wpas = WpaSupplicant(global_iface='/tmp/wpas-wlan5')
  188. try:
  189. with open(config, "w") as f:
  190. f.write("update_config=1\n")
  191. wpas.interface_add("wlan5", config=config)
  192. hapd.request("WPS_PIN any 12345670")
  193. wpas.scan_for_bss(apdev[0]['bssid'], freq="2412")
  194. wpas.request("WPS_PIN " + apdev[0]['bssid'] + " 12345670")
  195. ev = wpas.wait_event(["WPS-SUCCESS"], timeout=10)
  196. if ev is None:
  197. raise Exception("WPS-SUCCESS event timed out")
  198. with open(config, "r") as f:
  199. data = f.read()
  200. logger.info("Configuration file contents: " + data)
  201. with open(config, "r") as f:
  202. data = f.read()
  203. if "network=" not in data:
  204. raise Exception("Missing network block in configuration data")
  205. if "ssid=410a420d430044" not in data:
  206. raise Exception("Unexpected ssid parameter value")
  207. finally:
  208. try:
  209. os.remove(config)
  210. except:
  211. pass
  212. try:
  213. os.remove(config + ".tmp")
  214. except:
  215. pass
  216. try:
  217. os.rmdir(config)
  218. except:
  219. pass
  220. def test_wpas_config_file_set_psk(dev):
  221. """wpa_supplicant config file parsing/writing with arbitrary PSK value"""
  222. config = "/tmp/test_wpas_config_file.conf"
  223. if os.path.exists(config):
  224. os.remove(config)
  225. wpas = WpaSupplicant(global_iface='/tmp/wpas-wlan5')
  226. try:
  227. with open(config, "w") as f:
  228. f.write("update_config=1\n")
  229. wpas.interface_add("wlan5", config=config)
  230. id = wpas.add_network()
  231. wpas.set_network_quoted(id, "ssid", "foo")
  232. if "OK" in wpas.request('SET_NETWORK %d psk "12345678"\n}\nmodel_name=foobar\nnetwork={\n#\"' % id):
  233. raise Exception("Invalid psk value accepted")
  234. if "OK" not in wpas.request("SAVE_CONFIG"):
  235. raise Exception("Failed to save configuration file")
  236. with open(config, "r") as f:
  237. data = f.read()
  238. logger.info("Configuration file contents: " + data)
  239. if "model_name" in data:
  240. raise Exception("Unexpected parameter added to configuration")
  241. wpas.interface_remove("wlan5")
  242. wpas.interface_add("wlan5", config=config)
  243. finally:
  244. try:
  245. os.remove(config)
  246. except:
  247. pass
  248. try:
  249. os.remove(config + ".tmp")
  250. except:
  251. pass
  252. try:
  253. os.rmdir(config)
  254. except:
  255. pass
  256. def test_wpas_config_file_set_cred(dev):
  257. """wpa_supplicant config file parsing/writing with arbitrary cred values"""
  258. config = "/tmp/test_wpas_config_file.conf"
  259. if os.path.exists(config):
  260. os.remove(config)
  261. wpas = WpaSupplicant(global_iface='/tmp/wpas-wlan5')
  262. try:
  263. with open(config, "w") as f:
  264. f.write("update_config=1\n")
  265. wpas.interface_add("wlan5", config=config)
  266. id = wpas.add_cred()
  267. wpas.set_cred_quoted(id, "username", "hello")
  268. fields = [ "username", "milenage", "imsi", "password", "realm",
  269. "phase1", "phase2", "provisioning_sp" ]
  270. for field in fields:
  271. if "FAIL" not in wpas.request('SET_CRED %d %s "hello"\n}\nmodel_name=foobar\ncred={\n#\"' % (id, field)):
  272. raise Exception("Invalid %s value accepted" % field)
  273. if "OK" not in wpas.request("SAVE_CONFIG"):
  274. raise Exception("Failed to save configuration file")
  275. with open(config, "r") as f:
  276. data = f.read()
  277. logger.info("Configuration file contents: " + data)
  278. if "model_name" in data:
  279. raise Exception("Unexpected parameter added to configuration")
  280. wpas.interface_remove("wlan5")
  281. wpas.interface_add("wlan5", config=config)
  282. finally:
  283. try:
  284. os.remove(config)
  285. except:
  286. pass
  287. try:
  288. os.remove(config + ".tmp")
  289. except:
  290. pass
  291. try:
  292. os.rmdir(config)
  293. except:
  294. pass
  295. def test_wpas_config_file_set_global(dev):
  296. """wpa_supplicant config file parsing/writing with arbitrary global values"""
  297. config = "/tmp/test_wpas_config_file.conf"
  298. if os.path.exists(config):
  299. os.remove(config)
  300. wpas = WpaSupplicant(global_iface='/tmp/wpas-wlan5')
  301. try:
  302. with open(config, "w") as f:
  303. f.write("update_config=1\n")
  304. wpas.interface_add("wlan5", config=config)
  305. fields = [ "model_name", "device_name", "ctrl_interface_group",
  306. "opensc_engine_path", "pkcs11_engine_path",
  307. "pkcs11_module_path", "openssl_ciphers", "pcsc_reader",
  308. "pcsc_pin", "driver_param", "manufacturer", "model_name",
  309. "model_number", "serial_number", "config_methods",
  310. "p2p_ssid_postfix", "autoscan", "ext_password_backend",
  311. "osu_dir", "wowlan_triggers", "fst_group_id",
  312. "sched_scan_plans", "non_pref_chan" ]
  313. for field in fields:
  314. if "FAIL" not in wpas.request('SET %s hello\nmodel_name=foobar' % field):
  315. raise Exception("Invalid %s value accepted" % field)
  316. if "OK" not in wpas.request("SAVE_CONFIG"):
  317. raise Exception("Failed to save configuration file")
  318. with open(config, "r") as f:
  319. data = f.read()
  320. logger.info("Configuration file contents: " + data)
  321. if "model_name" in data:
  322. raise Exception("Unexpected parameter added to configuration")
  323. wpas.interface_remove("wlan5")
  324. wpas.interface_add("wlan5", config=config)
  325. finally:
  326. try:
  327. os.remove(config)
  328. except:
  329. pass
  330. try:
  331. os.remove(config + ".tmp")
  332. except:
  333. pass
  334. try:
  335. os.rmdir(config)
  336. except:
  337. pass