wps-ap-nfc.py 7.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266
  1. #!/usr/bin/python
  2. #
  3. # Example nfcpy to hostapd 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 argparse
  12. import nfc
  13. import nfc.ndef
  14. import nfc.llcp
  15. import nfc.handover
  16. import logging
  17. logging.basicConfig()
  18. import wpaspy
  19. wpas_ctrl = '/var/run/hostapd'
  20. continue_loop = True
  21. def wpas_connect():
  22. ifaces = []
  23. if os.path.isdir(wpas_ctrl):
  24. try:
  25. ifaces = [os.path.join(wpas_ctrl, i) for i in os.listdir(wpas_ctrl)]
  26. except OSError, error:
  27. print "Could not find hostapd: ", error
  28. return None
  29. if len(ifaces) < 1:
  30. print "No hostapd control interface found"
  31. return None
  32. for ctrl in ifaces:
  33. try:
  34. wpas = wpaspy.Ctrl(ctrl)
  35. return wpas
  36. except Exception, e:
  37. pass
  38. return None
  39. def wpas_tag_read(message):
  40. wpas = wpas_connect()
  41. if (wpas == None):
  42. return
  43. if "FAIL" in wpas.request("WPS_NFC_TAG_READ " + str(message).encode("hex")):
  44. return False
  45. return True
  46. def wpas_get_config_token():
  47. wpas = wpas_connect()
  48. if (wpas == None):
  49. return None
  50. return wpas.request("WPS_NFC_CONFIG_TOKEN NDEF").rstrip().decode("hex")
  51. def wpas_get_password_token():
  52. wpas = wpas_connect()
  53. if (wpas == None):
  54. return None
  55. return wpas.request("WPS_NFC_TOKEN NDEF").rstrip().decode("hex")
  56. def wpas_get_handover_sel():
  57. wpas = wpas_connect()
  58. if (wpas == None):
  59. return None
  60. return wpas.request("NFC_GET_HANDOVER_SEL NDEF WPS-CR").rstrip().decode("hex")
  61. def wpas_report_handover(req, sel):
  62. wpas = wpas_connect()
  63. if (wpas == None):
  64. return None
  65. return wpas.request("NFC_REPORT_HANDOVER RESP WPS " +
  66. str(req).encode("hex") + " " +
  67. str(sel).encode("hex"))
  68. class HandoverServer(nfc.handover.HandoverServer):
  69. def __init__(self, llc):
  70. super(HandoverServer, self).__init__(llc)
  71. self.ho_server_processing = False
  72. self.success = False
  73. def process_request(self, request):
  74. print "HandoverServer - request received"
  75. print "Parsed handover request: " + request.pretty()
  76. sel = nfc.ndef.HandoverSelectMessage(version="1.2")
  77. for carrier in request.carriers:
  78. print "Remote carrier type: " + carrier.type
  79. if carrier.type == "application/vnd.wfa.wsc":
  80. print "WPS carrier type match - add WPS carrier record"
  81. data = wpas_get_handover_sel()
  82. if data is None:
  83. print "Could not get handover select carrier record from hostapd"
  84. continue
  85. print "Handover select carrier record from hostapd:"
  86. print data.encode("hex")
  87. wpas_report_handover(carrier.record, data)
  88. message = nfc.ndef.Message(data);
  89. sel.add_carrier(message[0], "active", message[1:])
  90. print "Handover select:"
  91. print sel.pretty()
  92. print str(sel).encode("hex")
  93. print "Sending handover select"
  94. self.success = True
  95. return sel
  96. def wps_tag_read(tag):
  97. success = False
  98. if len(tag.ndef.message):
  99. for record in tag.ndef.message:
  100. print "record type " + record.type
  101. if record.type == "application/vnd.wfa.wsc":
  102. print "WPS tag - send to hostapd"
  103. success = wpas_tag_read(tag.ndef.message)
  104. break
  105. else:
  106. print "Empty tag"
  107. return success
  108. def rdwr_connected_write(tag):
  109. print "Tag found - writing"
  110. global write_data
  111. tag.ndef.message = str(write_data)
  112. print "Done - remove tag"
  113. global only_one
  114. if only_one:
  115. global continue_loop
  116. continue_loop = False
  117. global write_wait_remove
  118. while write_wait_remove and tag.is_present:
  119. time.sleep(0.1)
  120. def wps_write_config_tag(clf):
  121. print "Write WPS config token"
  122. global write_data, write_wait_remove
  123. write_wait_remove = True
  124. write_data = wpas_get_config_token()
  125. if write_data == None:
  126. print "Could not get WPS config token from hostapd"
  127. return
  128. print "Touch an NFC tag"
  129. clf.connect(rdwr={'on-connect': rdwr_connected_write})
  130. def wps_write_password_tag(clf):
  131. print "Write WPS password token"
  132. global write_data, write_wait_remove
  133. write_wait_remove = True
  134. write_data = wpas_get_password_token()
  135. if write_data == None:
  136. print "Could not get WPS password token from hostapd"
  137. return
  138. print "Touch an NFC tag"
  139. clf.connect(rdwr={'on-connect': rdwr_connected_write})
  140. def rdwr_connected(tag):
  141. global only_one
  142. print "Tag connected: " + str(tag)
  143. if tag.ndef:
  144. print "NDEF tag: " + tag.type
  145. try:
  146. print tag.ndef.message.pretty()
  147. except Exception, e:
  148. print e
  149. success = wps_tag_read(tag)
  150. if only_one and success:
  151. global continue_loop
  152. continue_loop = False
  153. else:
  154. print "Not an NDEF tag - remove tag"
  155. while tag.is_present:
  156. time.sleep(0.1)
  157. return not no_wait
  158. def llcp_startup(clf, llc):
  159. print "Start LLCP server"
  160. global srv
  161. srv = HandoverServer(llc)
  162. return llc
  163. def llcp_connected(llc):
  164. print "P2P LLCP connected"
  165. global wait_connection
  166. wait_connection = False
  167. global srv
  168. srv.start()
  169. return True
  170. def main():
  171. clf = nfc.ContactlessFrontend()
  172. parser = argparse.ArgumentParser(description='nfcpy to hostapd integration for WPS NFC operations')
  173. parser.add_argument('--only-one', '-1', action='store_true',
  174. help='run only one operation and exit')
  175. parser.add_argument('command', choices=['write-config',
  176. 'write-password'],
  177. nargs='?')
  178. args = parser.parse_args()
  179. global only_one
  180. only_one = args.only_one
  181. try:
  182. if not clf.open("usb"):
  183. print "Could not open connection with an NFC device"
  184. raise SystemExit
  185. if args.command == "write-config":
  186. wps_write_config_tag(clf)
  187. raise SystemExit
  188. if args.command == "write-password":
  189. wps_write_password_tag(clf)
  190. raise SystemExit
  191. global continue_loop
  192. while continue_loop:
  193. print "Waiting for a tag or peer to be touched"
  194. wait_connection = True
  195. try:
  196. if not clf.connect(rdwr={'on-connect': rdwr_connected},
  197. llcp={'on-startup': llcp_startup,
  198. 'on-connect': llcp_connected}):
  199. break
  200. except Exception, e:
  201. print "clf.connect failed"
  202. global srv
  203. if only_one and srv and srv.success:
  204. raise SystemExit
  205. except KeyboardInterrupt:
  206. raise SystemExit
  207. finally:
  208. clf.close()
  209. raise SystemExit
  210. if __name__ == '__main__':
  211. main()