hostapd.py 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562
  1. # Python class for controlling hostapd
  2. # Copyright (c) 2013-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 os
  7. import time
  8. import logging
  9. import binascii
  10. import struct
  11. import wpaspy
  12. import remotehost
  13. import utils
  14. logger = logging.getLogger()
  15. hapd_ctrl = '/var/run/hostapd'
  16. hapd_global = '/var/run/hostapd-global'
  17. def mac2tuple(mac):
  18. return struct.unpack('6B', binascii.unhexlify(mac.replace(':','')))
  19. class HostapdGlobal:
  20. def __init__(self, apdev=None):
  21. try:
  22. hostname = apdev['hostname']
  23. port = apdev['port']
  24. except:
  25. hostname = None
  26. port = 8878
  27. self.host = remotehost.Host(hostname)
  28. self.hostname = hostname
  29. self.port = port
  30. if hostname is None:
  31. self.ctrl = wpaspy.Ctrl(hapd_global)
  32. self.mon = wpaspy.Ctrl(hapd_global)
  33. self.dbg = ""
  34. else:
  35. self.ctrl = wpaspy.Ctrl(hostname, port)
  36. self.mon = wpaspy.Ctrl(hostname, port)
  37. self.dbg = hostname + "/" + str(port)
  38. self.mon.attach()
  39. def request(self, cmd, timeout=10):
  40. logger.debug(self.dbg + ": CTRL(global): " + cmd)
  41. return self.ctrl.request(cmd, timeout)
  42. def wait_event(self, events, timeout):
  43. start = os.times()[4]
  44. while True:
  45. while self.mon.pending():
  46. ev = self.mon.recv()
  47. logger.debug(self.dbg + "(global): " + ev)
  48. for event in events:
  49. if event in ev:
  50. return ev
  51. now = os.times()[4]
  52. remaining = start + timeout - now
  53. if remaining <= 0:
  54. break
  55. if not self.mon.pending(timeout=remaining):
  56. break
  57. return None
  58. def add(self, ifname, driver=None):
  59. cmd = "ADD " + ifname + " " + hapd_ctrl
  60. if driver:
  61. cmd += " " + driver
  62. res = self.request(cmd)
  63. if not "OK" in res:
  64. raise Exception("Could not add hostapd interface " + ifname)
  65. def add_iface(self, ifname, confname):
  66. res = self.request("ADD " + ifname + " config=" + confname)
  67. if not "OK" in res:
  68. raise Exception("Could not add hostapd interface")
  69. def add_bss(self, phy, confname, ignore_error=False):
  70. res = self.request("ADD bss_config=" + phy + ":" + confname)
  71. if not "OK" in res:
  72. if not ignore_error:
  73. raise Exception("Could not add hostapd BSS")
  74. def remove(self, ifname):
  75. self.request("REMOVE " + ifname, timeout=30)
  76. def relog(self):
  77. self.request("RELOG")
  78. def flush(self):
  79. self.request("FLUSH")
  80. def get_ctrl_iface_port(self, ifname):
  81. if self.hostname is None:
  82. return None
  83. res = self.request("INTERFACES ctrl")
  84. lines = res.splitlines()
  85. found = False
  86. for line in lines:
  87. words = line.split()
  88. if words[0] == ifname:
  89. found = True
  90. break
  91. if not found:
  92. raise Exception("Could not find UDP port for " + ifname)
  93. res = line.find("ctrl_iface=udp:")
  94. if res == -1:
  95. raise Exception("Wrong ctrl_interface format")
  96. words = line.split(":")
  97. return int(words[1])
  98. def terminate(self):
  99. self.mon.detach()
  100. self.mon.close()
  101. self.mon = None
  102. self.ctrl.terminate()
  103. self.ctrl = None
  104. class Hostapd:
  105. def __init__(self, ifname, bssidx=0, hostname=None, port=8877):
  106. self.host = remotehost.Host(hostname, ifname)
  107. self.ifname = ifname
  108. if hostname is None:
  109. self.ctrl = wpaspy.Ctrl(os.path.join(hapd_ctrl, ifname))
  110. self.mon = wpaspy.Ctrl(os.path.join(hapd_ctrl, ifname))
  111. self.dbg = ifname
  112. else:
  113. self.ctrl = wpaspy.Ctrl(hostname, port)
  114. self.mon = wpaspy.Ctrl(hostname, port)
  115. self.dbg = hostname + "/" + ifname
  116. self.mon.attach()
  117. self.bssid = None
  118. self.bssidx = bssidx
  119. def close_ctrl(self):
  120. if self.mon is not None:
  121. self.mon.detach()
  122. self.mon.close()
  123. self.mon = None
  124. self.ctrl.close()
  125. self.ctrl = None
  126. def own_addr(self):
  127. if self.bssid is None:
  128. self.bssid = self.get_status_field('bssid[%d]' % self.bssidx)
  129. return self.bssid
  130. def request(self, cmd):
  131. logger.debug(self.dbg + ": CTRL: " + cmd)
  132. return self.ctrl.request(cmd)
  133. def ping(self):
  134. return "PONG" in self.request("PING")
  135. def set(self, field, value):
  136. if not "OK" in self.request("SET " + field + " " + value):
  137. raise Exception("Failed to set hostapd parameter " + field)
  138. def set_defaults(self):
  139. self.set("driver", "nl80211")
  140. self.set("hw_mode", "g")
  141. self.set("channel", "1")
  142. self.set("ieee80211n", "1")
  143. self.set("logger_stdout", "-1")
  144. self.set("logger_stdout_level", "0")
  145. def set_open(self, ssid):
  146. self.set_defaults()
  147. self.set("ssid", ssid)
  148. def set_wpa2_psk(self, ssid, passphrase):
  149. self.set_defaults()
  150. self.set("ssid", ssid)
  151. self.set("wpa_passphrase", passphrase)
  152. self.set("wpa", "2")
  153. self.set("wpa_key_mgmt", "WPA-PSK")
  154. self.set("rsn_pairwise", "CCMP")
  155. def set_wpa_psk(self, ssid, passphrase):
  156. self.set_defaults()
  157. self.set("ssid", ssid)
  158. self.set("wpa_passphrase", passphrase)
  159. self.set("wpa", "1")
  160. self.set("wpa_key_mgmt", "WPA-PSK")
  161. self.set("wpa_pairwise", "TKIP")
  162. def set_wpa_psk_mixed(self, ssid, passphrase):
  163. self.set_defaults()
  164. self.set("ssid", ssid)
  165. self.set("wpa_passphrase", passphrase)
  166. self.set("wpa", "3")
  167. self.set("wpa_key_mgmt", "WPA-PSK")
  168. self.set("wpa_pairwise", "TKIP")
  169. self.set("rsn_pairwise", "CCMP")
  170. def set_wep(self, ssid, key):
  171. self.set_defaults()
  172. self.set("ssid", ssid)
  173. self.set("wep_key0", key)
  174. def enable(self):
  175. if not "OK" in self.request("ENABLE"):
  176. raise Exception("Failed to enable hostapd interface " + self.ifname)
  177. def disable(self):
  178. if not "OK" in self.request("DISABLE"):
  179. raise Exception("Failed to disable hostapd interface " + self.ifname)
  180. def dump_monitor(self):
  181. while self.mon.pending():
  182. ev = self.mon.recv()
  183. logger.debug(self.dbg + ": " + ev)
  184. def wait_event(self, events, timeout):
  185. start = os.times()[4]
  186. while True:
  187. while self.mon.pending():
  188. ev = self.mon.recv()
  189. logger.debug(self.dbg + ": " + ev)
  190. for event in events:
  191. if event in ev:
  192. return ev
  193. now = os.times()[4]
  194. remaining = start + timeout - now
  195. if remaining <= 0:
  196. break
  197. if not self.mon.pending(timeout=remaining):
  198. break
  199. return None
  200. def get_status(self):
  201. res = self.request("STATUS")
  202. lines = res.splitlines()
  203. vals = dict()
  204. for l in lines:
  205. [name,value] = l.split('=', 1)
  206. vals[name] = value
  207. return vals
  208. def get_status_field(self, field):
  209. vals = self.get_status()
  210. if field in vals:
  211. return vals[field]
  212. return None
  213. def get_driver_status(self):
  214. res = self.request("STATUS-DRIVER")
  215. lines = res.splitlines()
  216. vals = dict()
  217. for l in lines:
  218. [name,value] = l.split('=', 1)
  219. vals[name] = value
  220. return vals
  221. def get_driver_status_field(self, field):
  222. vals = self.get_driver_status()
  223. if field in vals:
  224. return vals[field]
  225. return None
  226. def get_config(self):
  227. res = self.request("GET_CONFIG")
  228. lines = res.splitlines()
  229. vals = dict()
  230. for l in lines:
  231. [name,value] = l.split('=', 1)
  232. vals[name] = value
  233. return vals
  234. def mgmt_rx(self, timeout=5):
  235. ev = self.wait_event(["MGMT-RX"], timeout=timeout)
  236. if ev is None:
  237. return None
  238. msg = {}
  239. frame = binascii.unhexlify(ev.split(' ')[1])
  240. msg['frame'] = frame
  241. hdr = struct.unpack('<HH6B6B6BH', frame[0:24])
  242. msg['fc'] = hdr[0]
  243. msg['subtype'] = (hdr[0] >> 4) & 0xf
  244. hdr = hdr[1:]
  245. msg['duration'] = hdr[0]
  246. hdr = hdr[1:]
  247. msg['da'] = "%02x:%02x:%02x:%02x:%02x:%02x" % hdr[0:6]
  248. hdr = hdr[6:]
  249. msg['sa'] = "%02x:%02x:%02x:%02x:%02x:%02x" % hdr[0:6]
  250. hdr = hdr[6:]
  251. msg['bssid'] = "%02x:%02x:%02x:%02x:%02x:%02x" % hdr[0:6]
  252. hdr = hdr[6:]
  253. msg['seq_ctrl'] = hdr[0]
  254. msg['payload'] = frame[24:]
  255. return msg
  256. def mgmt_tx(self, msg):
  257. t = (msg['fc'], 0) + mac2tuple(msg['da']) + mac2tuple(msg['sa']) + mac2tuple(msg['bssid']) + (0,)
  258. hdr = struct.pack('<HH6B6B6BH', *t)
  259. self.request("MGMT_TX " + binascii.hexlify(hdr + msg['payload']))
  260. def get_sta(self, addr, info=None, next=False):
  261. cmd = "STA-NEXT " if next else "STA "
  262. if addr is None:
  263. res = self.request("STA-FIRST")
  264. elif info:
  265. res = self.request(cmd + addr + " " + info)
  266. else:
  267. res = self.request(cmd + addr)
  268. lines = res.splitlines()
  269. vals = dict()
  270. first = True
  271. for l in lines:
  272. if first and '=' not in l:
  273. vals['addr'] = l
  274. first = False
  275. else:
  276. [name,value] = l.split('=', 1)
  277. vals[name] = value
  278. return vals
  279. def get_mib(self, param=None):
  280. if param:
  281. res = self.request("MIB " + param)
  282. else:
  283. res = self.request("MIB")
  284. lines = res.splitlines()
  285. vals = dict()
  286. for l in lines:
  287. name_val = l.split('=', 1)
  288. if len(name_val) > 1:
  289. vals[name_val[0]] = name_val[1]
  290. return vals
  291. def get_pmksa(self, addr):
  292. res = self.request("PMKSA")
  293. lines = res.splitlines()
  294. for l in lines:
  295. if addr not in l:
  296. continue
  297. vals = dict()
  298. [index,aa,pmkid,expiration,opportunistic] = l.split(' ')
  299. vals['index'] = index
  300. vals['pmkid'] = pmkid
  301. vals['expiration'] = expiration
  302. vals['opportunistic'] = opportunistic
  303. return vals
  304. return None
  305. def add_ap(apdev, params, wait_enabled=True, no_enable=False, timeout=30):
  306. if isinstance(apdev, dict):
  307. ifname = apdev['ifname']
  308. try:
  309. hostname = apdev['hostname']
  310. port = apdev['port']
  311. logger.info("Starting AP " + hostname + "/" + port + " " + ifname)
  312. except:
  313. logger.info("Starting AP " + ifname)
  314. hostname = None
  315. port = 8878
  316. else:
  317. ifname = apdev
  318. logger.info("Starting AP " + ifname + " (old add_ap argument type)")
  319. hostname = None
  320. port = 8878
  321. hapd_global = HostapdGlobal(apdev)
  322. hapd_global.remove(ifname)
  323. hapd_global.add(ifname)
  324. port = hapd_global.get_ctrl_iface_port(ifname)
  325. hapd = Hostapd(ifname, hostname=hostname, port=port)
  326. if not hapd.ping():
  327. raise Exception("Could not ping hostapd")
  328. hapd.set_defaults()
  329. fields = [ "ssid", "wpa_passphrase", "nas_identifier", "wpa_key_mgmt",
  330. "wpa",
  331. "wpa_pairwise", "rsn_pairwise", "auth_server_addr",
  332. "acct_server_addr", "osu_server_uri" ]
  333. for field in fields:
  334. if field in params:
  335. hapd.set(field, params[field])
  336. for f,v in params.items():
  337. if f in fields:
  338. continue
  339. if isinstance(v, list):
  340. for val in v:
  341. hapd.set(f, val)
  342. else:
  343. hapd.set(f, v)
  344. if no_enable:
  345. return hapd
  346. hapd.enable()
  347. if wait_enabled:
  348. ev = hapd.wait_event(["AP-ENABLED", "AP-DISABLED"], timeout=timeout)
  349. if ev is None:
  350. raise Exception("AP startup timed out")
  351. if "AP-ENABLED" not in ev:
  352. raise Exception("AP startup failed")
  353. return hapd
  354. def add_bss(apdev, ifname, confname, ignore_error=False):
  355. phy = utils.get_phy(apdev)
  356. try:
  357. hostname = apdev['hostname']
  358. port = apdev['port']
  359. logger.info("Starting BSS " + hostname + "/" + port + " phy=" + phy + " ifname=" + ifname)
  360. except:
  361. logger.info("Starting BSS phy=" + phy + " ifname=" + ifname)
  362. hostname = None
  363. port = 8878
  364. hapd_global = HostapdGlobal(apdev)
  365. hapd_global.add_bss(phy, confname, ignore_error)
  366. port = hapd_global.get_ctrl_iface_port(ifname)
  367. hapd = Hostapd(ifname, hostname=hostname, port=port)
  368. if not hapd.ping():
  369. raise Exception("Could not ping hostapd")
  370. return hapd
  371. def add_iface(apdev, confname):
  372. ifname = apdev['ifname']
  373. try:
  374. hostname = apdev['hostname']
  375. port = apdev['port']
  376. logger.info("Starting interface " + hostname + "/" + port + " " + ifname)
  377. except:
  378. logger.info("Starting interface " + ifname)
  379. hostname = None
  380. port = 8878
  381. hapd_global = HostapdGlobal(apdev)
  382. hapd_global.add_iface(ifname, confname)
  383. port = hapd_global.get_ctrl_iface_port(ifname)
  384. hapd = Hostapd(ifname, hostname=hostname, port=port)
  385. if not hapd.ping():
  386. raise Exception("Could not ping hostapd")
  387. return hapd
  388. def remove_bss(apdev, ifname=None):
  389. if ifname == None:
  390. ifname = apdev['ifname']
  391. try:
  392. hostname = apdev['hostname']
  393. port = apdev['port']
  394. logger.info("Removing BSS " + hostname + "/" + port + " " + ifname)
  395. except:
  396. logger.info("Removing BSS " + ifname)
  397. hapd_global = HostapdGlobal(apdev)
  398. hapd_global.remove(ifname)
  399. def terminate(apdev):
  400. try:
  401. hostname = apdev['hostname']
  402. port = apdev['port']
  403. logger.info("Terminating hostapd " + hostname + "/" + port)
  404. except:
  405. logger.info("Terminating hostapd")
  406. hapd_global = HostapdGlobal(apdev)
  407. hapd_global.terminate()
  408. def wpa2_params(ssid=None, passphrase=None):
  409. params = { "wpa": "2",
  410. "wpa_key_mgmt": "WPA-PSK",
  411. "rsn_pairwise": "CCMP" }
  412. if ssid:
  413. params["ssid"] = ssid
  414. if passphrase:
  415. params["wpa_passphrase"] = passphrase
  416. return params
  417. def wpa_params(ssid=None, passphrase=None):
  418. params = { "wpa": "1",
  419. "wpa_key_mgmt": "WPA-PSK",
  420. "wpa_pairwise": "TKIP" }
  421. if ssid:
  422. params["ssid"] = ssid
  423. if passphrase:
  424. params["wpa_passphrase"] = passphrase
  425. return params
  426. def wpa_mixed_params(ssid=None, passphrase=None):
  427. params = { "wpa": "3",
  428. "wpa_key_mgmt": "WPA-PSK",
  429. "wpa_pairwise": "TKIP",
  430. "rsn_pairwise": "CCMP" }
  431. if ssid:
  432. params["ssid"] = ssid
  433. if passphrase:
  434. params["wpa_passphrase"] = passphrase
  435. return params
  436. def radius_params():
  437. params = { "auth_server_addr": "127.0.0.1",
  438. "auth_server_port": "1812",
  439. "auth_server_shared_secret": "radius",
  440. "nas_identifier": "nas.w1.fi" }
  441. return params
  442. def wpa_eap_params(ssid=None):
  443. params = radius_params()
  444. params["wpa"] = "1"
  445. params["wpa_key_mgmt"] = "WPA-EAP"
  446. params["wpa_pairwise"] = "TKIP"
  447. params["ieee8021x"] = "1"
  448. if ssid:
  449. params["ssid"] = ssid
  450. return params
  451. def wpa2_eap_params(ssid=None):
  452. params = radius_params()
  453. params["wpa"] = "2"
  454. params["wpa_key_mgmt"] = "WPA-EAP"
  455. params["rsn_pairwise"] = "CCMP"
  456. params["ieee8021x"] = "1"
  457. if ssid:
  458. params["ssid"] = ssid
  459. return params
  460. def b_only_params(channel="1", ssid=None, country=None):
  461. params = { "hw_mode" : "b",
  462. "channel" : channel }
  463. if ssid:
  464. params["ssid"] = ssid
  465. if country:
  466. params["country_code"] = country
  467. return params
  468. def g_only_params(channel="1", ssid=None, country=None):
  469. params = { "hw_mode" : "g",
  470. "channel" : channel }
  471. if ssid:
  472. params["ssid"] = ssid
  473. if country:
  474. params["country_code"] = country
  475. return params
  476. def a_only_params(channel="36", ssid=None, country=None):
  477. params = { "hw_mode" : "a",
  478. "channel" : channel }
  479. if ssid:
  480. params["ssid"] = ssid
  481. if country:
  482. params["country_code"] = country
  483. return params
  484. def ht20_params(channel="1", ssid=None, country=None):
  485. params = { "ieee80211n" : "1",
  486. "channel" : channel,
  487. "hw_mode" : "g" }
  488. if int(channel) > 14:
  489. params["hw_mode"] = "a"
  490. if ssid:
  491. params["ssid"] = ssid
  492. if country:
  493. params["country_code"] = country
  494. return params
  495. def ht40_plus_params(channel="1", ssid=None, country=None):
  496. params = ht20_params(channel, ssid, country)
  497. params['ht_capab'] = "[HT40+]"
  498. return params
  499. def ht40_minus_params(channel="1", ssid=None, country=None):
  500. params = ht20_params(channel, ssid, country)
  501. params['ht_capab'] = "[HT40-]"
  502. return params