wps-nfc.py 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508
  1. #!/usr/bin/python
  2. #
  3. # Example nfcpy to wpa_supplicant wrapper for WPS NFC operations
  4. # Copyright (c) 2012-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 sys
  10. import time
  11. import random
  12. import threading
  13. import argparse
  14. import nfc
  15. import nfc.ndef
  16. import nfc.llcp
  17. import nfc.handover
  18. import logging
  19. import wpaspy
  20. wpas_ctrl = '/var/run/wpa_supplicant'
  21. srv = None
  22. continue_loop = True
  23. terminate_now = False
  24. summary_file = None
  25. success_file = None
  26. def summary(txt):
  27. print txt
  28. if summary_file:
  29. with open(summary_file, 'a') as f:
  30. f.write(txt + "\n")
  31. def success_report(txt):
  32. summary(txt)
  33. if success_file:
  34. with open(success_file, 'a') as f:
  35. f.write(txt + "\n")
  36. def wpas_connect():
  37. ifaces = []
  38. if os.path.isdir(wpas_ctrl):
  39. try:
  40. ifaces = [os.path.join(wpas_ctrl, i) for i in os.listdir(wpas_ctrl)]
  41. except OSError, error:
  42. print "Could not find wpa_supplicant: ", error
  43. return None
  44. if len(ifaces) < 1:
  45. print "No wpa_supplicant control interface found"
  46. return None
  47. for ctrl in ifaces:
  48. try:
  49. wpas = wpaspy.Ctrl(ctrl)
  50. return wpas
  51. except Exception, e:
  52. pass
  53. return None
  54. def wpas_tag_read(message):
  55. wpas = wpas_connect()
  56. if (wpas == None):
  57. return False
  58. if "FAIL" in wpas.request("WPS_NFC_TAG_READ " + str(message).encode("hex")):
  59. return False
  60. return True
  61. def wpas_get_config_token(id=None):
  62. wpas = wpas_connect()
  63. if (wpas == None):
  64. return None
  65. if id:
  66. ret = wpas.request("WPS_NFC_CONFIG_TOKEN NDEF " + id)
  67. else:
  68. ret = wpas.request("WPS_NFC_CONFIG_TOKEN NDEF")
  69. if "FAIL" in ret:
  70. return None
  71. return ret.rstrip().decode("hex")
  72. def wpas_get_er_config_token(uuid):
  73. wpas = wpas_connect()
  74. if (wpas == None):
  75. return None
  76. ret = wpas.request("WPS_ER_NFC_CONFIG_TOKEN NDEF " + uuid)
  77. if "FAIL" in ret:
  78. return None
  79. return ret.rstrip().decode("hex")
  80. def wpas_get_password_token():
  81. wpas = wpas_connect()
  82. if (wpas == None):
  83. return None
  84. ret = wpas.request("WPS_NFC_TOKEN NDEF")
  85. if "FAIL" in ret:
  86. return None
  87. return ret.rstrip().decode("hex")
  88. def wpas_get_handover_req():
  89. wpas = wpas_connect()
  90. if (wpas == None):
  91. return None
  92. ret = wpas.request("NFC_GET_HANDOVER_REQ NDEF WPS-CR")
  93. if "FAIL" in ret:
  94. return None
  95. return ret.rstrip().decode("hex")
  96. def wpas_get_handover_sel(uuid):
  97. wpas = wpas_connect()
  98. if (wpas == None):
  99. return None
  100. if uuid is None:
  101. res = wpas.request("NFC_GET_HANDOVER_SEL NDEF WPS-CR").rstrip()
  102. else:
  103. res = wpas.request("NFC_GET_HANDOVER_SEL NDEF WPS-CR " + uuid).rstrip()
  104. if "FAIL" in res:
  105. return None
  106. return res.decode("hex")
  107. def wpas_report_handover(req, sel, type):
  108. wpas = wpas_connect()
  109. if (wpas == None):
  110. return None
  111. return wpas.request("NFC_REPORT_HANDOVER " + type + " WPS " +
  112. str(req).encode("hex") + " " +
  113. str(sel).encode("hex"))
  114. class HandoverServer(nfc.handover.HandoverServer):
  115. def __init__(self, llc):
  116. super(HandoverServer, self).__init__(llc)
  117. self.sent_carrier = None
  118. self.ho_server_processing = False
  119. self.success = False
  120. def process_request(self, request):
  121. self.ho_server_processing = True
  122. summary("HandoverServer - request received")
  123. try:
  124. print "Parsed handover request: " + request.pretty()
  125. except Exception, e:
  126. print e
  127. sel = nfc.ndef.HandoverSelectMessage(version="1.2")
  128. for carrier in request.carriers:
  129. print "Remote carrier type: " + carrier.type
  130. if carrier.type == "application/vnd.wfa.wsc":
  131. summary("WPS carrier type match - add WPS carrier record")
  132. data = wpas_get_handover_sel(self.uuid)
  133. if data is None:
  134. summary("Could not get handover select carrier record from wpa_supplicant")
  135. continue
  136. print "Handover select carrier record from wpa_supplicant:"
  137. print data.encode("hex")
  138. self.sent_carrier = data
  139. if "OK" in wpas_report_handover(carrier.record, self.sent_carrier, "RESP"):
  140. success_report("Handover reported successfully (responder)")
  141. else:
  142. summary("Handover report rejected (responder)")
  143. message = nfc.ndef.Message(data);
  144. sel.add_carrier(message[0], "active", message[1:])
  145. print "Handover select:"
  146. try:
  147. print sel.pretty()
  148. except Exception, e:
  149. print e
  150. print str(sel).encode("hex")
  151. summary("Sending handover select")
  152. self.success = True
  153. return sel
  154. def wps_handover_init(llc):
  155. summary("Trying to initiate WPS handover")
  156. data = wpas_get_handover_req()
  157. if (data == None):
  158. summary("Could not get handover request carrier record from wpa_supplicant")
  159. return
  160. print "Handover request carrier record from wpa_supplicant: " + data.encode("hex")
  161. message = nfc.ndef.HandoverRequestMessage(version="1.2")
  162. message.nonce = random.randint(0, 0xffff)
  163. datamsg = nfc.ndef.Message(data)
  164. message.add_carrier(datamsg[0], "active", datamsg[1:])
  165. print "Handover request:"
  166. try:
  167. print message.pretty()
  168. except Exception, e:
  169. print e
  170. print str(message).encode("hex")
  171. client = nfc.handover.HandoverClient(llc)
  172. try:
  173. summary("Trying to initiate NFC connection handover")
  174. client.connect()
  175. summary("Connected for handover")
  176. except nfc.llcp.ConnectRefused:
  177. summary("Handover connection refused")
  178. client.close()
  179. return
  180. except Exception, e:
  181. summary("Other exception: " + str(e))
  182. client.close()
  183. return
  184. summary("Sending handover request")
  185. if not client.send(message):
  186. summary("Failed to send handover request")
  187. client.close()
  188. return
  189. summary("Receiving handover response")
  190. message = client._recv()
  191. if message is None:
  192. summary("No response received")
  193. client.close()
  194. return
  195. if message.type != "urn:nfc:wkt:Hs":
  196. summary("Response was not Hs - received: " + message.type)
  197. client.close()
  198. return
  199. print "Received message"
  200. try:
  201. print message.pretty()
  202. except Exception, e:
  203. print e
  204. print str(message).encode("hex")
  205. message = nfc.ndef.HandoverSelectMessage(message)
  206. summary("Handover select received")
  207. try:
  208. print message.pretty()
  209. except Exception, e:
  210. print e
  211. for carrier in message.carriers:
  212. print "Remote carrier type: " + carrier.type
  213. if carrier.type == "application/vnd.wfa.wsc":
  214. print "WPS carrier type match - send to wpa_supplicant"
  215. if "OK" in wpas_report_handover(data, carrier.record, "INIT"):
  216. success_report("Handover reported successfully (initiator)")
  217. else:
  218. summary("Handover report rejected (initiator)")
  219. # nfcpy does not support the new format..
  220. #wifi = nfc.ndef.WifiConfigRecord(carrier.record)
  221. #print wifi.pretty()
  222. print "Remove peer"
  223. client.close()
  224. print "Done with handover"
  225. global only_one
  226. if only_one:
  227. global continue_loop
  228. continue_loop = False
  229. global no_wait
  230. if no_wait:
  231. print "Trying to exit.."
  232. global terminate_now
  233. terminate_now = True
  234. def wps_tag_read(tag, wait_remove=True):
  235. success = False
  236. if len(tag.ndef.message):
  237. for record in tag.ndef.message:
  238. print "record type " + record.type
  239. if record.type == "application/vnd.wfa.wsc":
  240. summary("WPS tag - send to wpa_supplicant")
  241. success = wpas_tag_read(tag.ndef.message)
  242. break
  243. else:
  244. summary("Empty tag")
  245. if success:
  246. success_report("Tag read succeeded")
  247. if wait_remove:
  248. print "Remove tag"
  249. while tag.is_present:
  250. time.sleep(0.1)
  251. return success
  252. def rdwr_connected_write(tag):
  253. summary("Tag found - writing - " + str(tag))
  254. global write_data
  255. tag.ndef.message = str(write_data)
  256. success_report("Tag write succeeded")
  257. print "Done - remove tag"
  258. global only_one
  259. if only_one:
  260. global continue_loop
  261. continue_loop = False
  262. global write_wait_remove
  263. while write_wait_remove and tag.is_present:
  264. time.sleep(0.1)
  265. def wps_write_config_tag(clf, id=None, wait_remove=True):
  266. print "Write WPS config token"
  267. global write_data, write_wait_remove
  268. write_wait_remove = wait_remove
  269. write_data = wpas_get_config_token(id)
  270. if write_data == None:
  271. print "Could not get WPS config token from wpa_supplicant"
  272. sys.exit(1)
  273. return
  274. print "Touch an NFC tag"
  275. clf.connect(rdwr={'on-connect': rdwr_connected_write})
  276. def wps_write_er_config_tag(clf, uuid, wait_remove=True):
  277. print "Write WPS ER config token"
  278. global write_data, write_wait_remove
  279. write_wait_remove = wait_remove
  280. write_data = wpas_get_er_config_token(uuid)
  281. if write_data == None:
  282. print "Could not get WPS config token from wpa_supplicant"
  283. return
  284. print "Touch an NFC tag"
  285. clf.connect(rdwr={'on-connect': rdwr_connected_write})
  286. def wps_write_password_tag(clf, wait_remove=True):
  287. print "Write WPS password token"
  288. global write_data, write_wait_remove
  289. write_wait_remove = wait_remove
  290. write_data = wpas_get_password_token()
  291. if write_data == None:
  292. print "Could not get WPS password token from wpa_supplicant"
  293. return
  294. print "Touch an NFC tag"
  295. clf.connect(rdwr={'on-connect': rdwr_connected_write})
  296. def rdwr_connected(tag):
  297. global only_one, no_wait
  298. summary("Tag connected: " + str(tag))
  299. if tag.ndef:
  300. print "NDEF tag: " + tag.type
  301. try:
  302. print tag.ndef.message.pretty()
  303. except Exception, e:
  304. print e
  305. success = wps_tag_read(tag, not only_one)
  306. if only_one and success:
  307. global continue_loop
  308. continue_loop = False
  309. else:
  310. summary("Not an NDEF tag - remove tag")
  311. return True
  312. return not no_wait
  313. def llcp_worker(llc):
  314. global arg_uuid
  315. if arg_uuid is None:
  316. wps_handover_init(llc)
  317. print "Exiting llcp_worker thread"
  318. return
  319. global srv
  320. global wait_connection
  321. while not wait_connection and srv.sent_carrier is None:
  322. if srv.ho_server_processing:
  323. time.sleep(0.025)
  324. def llcp_startup(clf, llc):
  325. global arg_uuid
  326. if arg_uuid:
  327. print "Start LLCP server"
  328. global srv
  329. srv = HandoverServer(llc)
  330. if arg_uuid is "ap":
  331. print "Trying to handle WPS handover"
  332. srv.uuid = None
  333. else:
  334. print "Trying to handle WPS handover with AP " + arg_uuid
  335. srv.uuid = arg_uuid
  336. return llc
  337. def llcp_connected(llc):
  338. print "P2P LLCP connected"
  339. global wait_connection
  340. wait_connection = False
  341. global arg_uuid
  342. if arg_uuid:
  343. global srv
  344. srv.start()
  345. else:
  346. threading.Thread(target=llcp_worker, args=(llc,)).start()
  347. print "llcp_connected returning"
  348. return True
  349. def terminate_loop():
  350. global terminate_now
  351. return terminate_now
  352. def main():
  353. clf = nfc.ContactlessFrontend()
  354. parser = argparse.ArgumentParser(description='nfcpy to wpa_supplicant integration for WPS NFC operations')
  355. parser.add_argument('-d', const=logging.DEBUG, default=logging.INFO,
  356. action='store_const', dest='loglevel',
  357. help='verbose debug output')
  358. parser.add_argument('-q', const=logging.WARNING, action='store_const',
  359. dest='loglevel', help='be quiet')
  360. parser.add_argument('--only-one', '-1', action='store_true',
  361. help='run only one operation and exit')
  362. parser.add_argument('--no-wait', action='store_true',
  363. help='do not wait for tag to be removed before exiting')
  364. parser.add_argument('--uuid',
  365. help='UUID of an AP (used for WPS ER operations)')
  366. parser.add_argument('--id',
  367. help='network id (used for WPS ER operations)')
  368. parser.add_argument('--summary',
  369. help='summary file for writing status updates')
  370. parser.add_argument('--success',
  371. help='success file for writing success update')
  372. parser.add_argument('command', choices=['write-config',
  373. 'write-er-config',
  374. 'write-password'],
  375. nargs='?')
  376. args = parser.parse_args()
  377. global arg_uuid
  378. arg_uuid = args.uuid
  379. global only_one
  380. only_one = args.only_one
  381. global no_wait
  382. no_wait = args.no_wait
  383. if args.summary:
  384. global summary_file
  385. summary_file = args.summary
  386. if args.success:
  387. global success_file
  388. success_file = args.success
  389. logging.basicConfig(level=args.loglevel)
  390. try:
  391. if not clf.open("usb"):
  392. print "Could not open connection with an NFC device"
  393. raise SystemExit
  394. if args.command == "write-config":
  395. wps_write_config_tag(clf, id=args.id, wait_remove=not args.no_wait)
  396. raise SystemExit
  397. if args.command == "write-er-config":
  398. wps_write_er_config_tag(clf, args.uuid, wait_remove=not args.no_wait)
  399. raise SystemExit
  400. if args.command == "write-password":
  401. wps_write_password_tag(clf, wait_remove=not args.no_wait)
  402. raise SystemExit
  403. global continue_loop
  404. while continue_loop:
  405. print "Waiting for a tag or peer to be touched"
  406. wait_connection = True
  407. try:
  408. if not clf.connect(rdwr={'on-connect': rdwr_connected},
  409. llcp={'on-startup': llcp_startup,
  410. 'on-connect': llcp_connected},
  411. terminate=terminate_loop):
  412. break
  413. except Exception, e:
  414. print "clf.connect failed"
  415. global srv
  416. if only_one and srv and srv.success:
  417. raise SystemExit
  418. except KeyboardInterrupt:
  419. raise SystemExit
  420. finally:
  421. clf.close()
  422. raise SystemExit
  423. if __name__ == '__main__':
  424. main()