wps-ap-nfc.py 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251
  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 nfc
  12. import nfc.ndef
  13. import nfc.llcp
  14. import nfc.handover
  15. import logging
  16. logging.basicConfig()
  17. import wpactrl
  18. wpas_ctrl = '/var/run/hostapd'
  19. def wpas_connect():
  20. ifaces = []
  21. if os.path.isdir(wpas_ctrl):
  22. try:
  23. ifaces = [os.path.join(wpas_ctrl, i) for i in os.listdir(wpas_ctrl)]
  24. except OSError, error:
  25. print "Could not find hostapd: ", error
  26. return None
  27. if len(ifaces) < 1:
  28. print "No hostapd control interface found"
  29. return None
  30. for ctrl in ifaces:
  31. try:
  32. wpas = wpactrl.WPACtrl(ctrl)
  33. return wpas
  34. except wpactrl.error, error:
  35. print "Error: ", error
  36. pass
  37. return None
  38. def wpas_tag_read(message):
  39. wpas = wpas_connect()
  40. if (wpas == None):
  41. return
  42. print wpas.request("WPS_NFC_TAG_READ " + message.encode("hex"))
  43. def wpas_get_config_token():
  44. wpas = wpas_connect()
  45. if (wpas == None):
  46. return None
  47. return wpas.request("WPS_NFC_CONFIG_TOKEN NDEF").rstrip().decode("hex")
  48. def wpas_get_password_token():
  49. wpas = wpas_connect()
  50. if (wpas == None):
  51. return None
  52. return wpas.request("WPS_NFC_TOKEN NDEF").rstrip().decode("hex")
  53. def wpas_get_handover_req():
  54. wpas = wpas_connect()
  55. if (wpas == None):
  56. return None
  57. return wpas.request("NFC_GET_HANDOVER_REQ NDEF WPS").rstrip().decode("hex")
  58. def wpas_put_handover_sel(message):
  59. wpas = wpas_connect()
  60. if (wpas == None):
  61. return
  62. print wpas.request("NFC_RX_HANDOVER_SEL " + str(message).encode("hex"))
  63. def wps_handover_init(peer):
  64. print "Trying to initiate WPS handover"
  65. data = wpas_get_handover_req()
  66. if (data == None):
  67. print "Could not get handover request message from hostapd"
  68. return
  69. print "Handover request from hostapd: " + data.encode("hex")
  70. message = nfc.ndef.Message(data)
  71. print "Parsed handover request: " + message.pretty()
  72. nfc.llcp.activate(peer);
  73. client = nfc.handover.HandoverClient()
  74. try:
  75. print "Trying handover";
  76. client.connect()
  77. print "Connected for handover"
  78. except nfc.llcp.ConnectRefused:
  79. print "Handover connection refused"
  80. nfc.llcp.shutdown()
  81. client.close()
  82. return
  83. print "Sending handover request"
  84. if not client.send(message):
  85. print "Failed to send handover request"
  86. print "Receiving handover response"
  87. message = client._recv()
  88. print "Handover select received"
  89. print message.pretty()
  90. wpas_put_handover_sel(message)
  91. print "Remove peer"
  92. nfc.llcp.shutdown()
  93. client.close()
  94. print "Done with handover"
  95. def wps_tag_read(tag):
  96. if len(tag.ndef.message):
  97. message = nfc.ndef.Message(tag.ndef.message)
  98. print "message type " + message.type
  99. for record in message:
  100. print "record type " + record.type
  101. if record.type == "application/vnd.wfa.wsc":
  102. print "WPS tag - send to hostapd"
  103. wpas_tag_read(tag.ndef.message)
  104. break
  105. else:
  106. print "Empty tag"
  107. print "Remove tag"
  108. while tag.is_present:
  109. time.sleep(0.1)
  110. def wps_write_config_tag(clf):
  111. print "Write WPS config token"
  112. data = wpas_get_config_token()
  113. if (data == None):
  114. print "Could not get WPS config token from hostapd"
  115. return
  116. print "Touch an NFC tag"
  117. while True:
  118. tag = clf.poll()
  119. if tag == None:
  120. time.sleep(0.1)
  121. continue
  122. break
  123. print "Tag found - writing"
  124. tag.ndef.message = data
  125. print "Done - remove tag"
  126. while tag.is_present:
  127. time.sleep(0.1)
  128. def wps_write_password_tag(clf):
  129. print "Write WPS password token"
  130. data = wpas_get_password_token()
  131. if (data == None):
  132. print "Could not get WPS password token from hostapd"
  133. return
  134. print "Touch an NFC tag"
  135. while True:
  136. tag = clf.poll()
  137. if tag == None:
  138. time.sleep(0.1)
  139. continue
  140. break
  141. print "Tag found - writing"
  142. tag.ndef.message = data
  143. print "Done - remove tag"
  144. while tag.is_present:
  145. time.sleep(0.1)
  146. def find_peer(clf):
  147. while True:
  148. if nfc.llcp.connected():
  149. print "LLCP connected"
  150. general_bytes = nfc.llcp.startup({})
  151. peer = clf.listen(ord(os.urandom(1)) + 200, general_bytes)
  152. if isinstance(peer, nfc.DEP):
  153. print "listen -> DEP";
  154. if peer.general_bytes.startswith("Ffm"):
  155. print "Found DEP"
  156. return peer
  157. print "mismatch in general_bytes"
  158. print peer.general_bytes
  159. peer = clf.poll(general_bytes)
  160. if isinstance(peer, nfc.DEP):
  161. print "poll -> DEP";
  162. if peer.general_bytes.startswith("Ffm"):
  163. print "Found DEP"
  164. return peer
  165. print "mismatch in general_bytes"
  166. print peer.general_bytes
  167. if peer:
  168. print "Found tag"
  169. return peer
  170. def main():
  171. clf = nfc.ContactlessFrontend()
  172. try:
  173. if len(sys.argv) > 1 and sys.argv[1] == "write-config":
  174. wps_write_config_tag(clf)
  175. raise SystemExit
  176. if len(sys.argv) > 1 and sys.argv[1] == "write-password":
  177. wps_write_password_tag(clf)
  178. raise SystemExit
  179. while True:
  180. print "Waiting for a tag or peer to be touched"
  181. tag = find_peer(clf)
  182. if isinstance(tag, nfc.DEP):
  183. wps_handover_init(tag)
  184. continue
  185. if tag.ndef:
  186. wps_tag_read(tag)
  187. continue
  188. print "Not an NDEF tag - remove tag"
  189. while tag.is_present:
  190. time.sleep(0.1)
  191. except KeyboardInterrupt:
  192. raise SystemExit
  193. finally:
  194. clf.close()
  195. raise SystemExit
  196. if __name__ == '__main__':
  197. main()