wps-ap-nfc.py 6.7 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 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_sel():
  54. wpas = wpas_connect()
  55. if (wpas == None):
  56. return None
  57. return wpas.request("NFC_GET_HANDOVER_SEL NDEF WPS-CR").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. class HandoverServer(nfc.handover.HandoverServer):
  64. def __init__(self):
  65. super(HandoverServer, self).__init__()
  66. def process_request(self, request):
  67. print "HandoverServer - request received"
  68. print "Parsed handover request: " + request.pretty()
  69. sel = nfc.ndef.HandoverSelectMessage(version="1.2")
  70. for carrier in request.carriers:
  71. print "Remote carrier type: " + carrier.type
  72. if carrier.type == "application/vnd.wfa.wsc":
  73. print "WPS carrier type match - add WPS carrier record"
  74. data = wpas_get_handover_sel()
  75. if data is None:
  76. print "Could not get handover select carrier record from hostapd"
  77. continue
  78. print "Handover select carrier record from hostapd:"
  79. print data.encode("hex")
  80. message = nfc.ndef.Message(data);
  81. sel.add_carrier(message[0], "active", message[1:])
  82. print "Handover select:"
  83. print sel.pretty()
  84. print str(sel).encode("hex")
  85. print "Sending handover select"
  86. return sel
  87. def wps_handover_resp(peer):
  88. print "Trying to handle WPS handover"
  89. srv = HandoverServer()
  90. nfc.llcp.activate(peer);
  91. try:
  92. print "Trying handover";
  93. srv.start()
  94. print "Wait for disconnect"
  95. while nfc.llcp.connected():
  96. time.sleep(0.1)
  97. print "Disconnected after handover"
  98. except nfc.llcp.ConnectRefused:
  99. print "Handover connection refused"
  100. nfc.llcp.shutdown()
  101. return
  102. print "Remove peer"
  103. nfc.llcp.shutdown()
  104. print "Done with handover"
  105. def wps_tag_read(tag):
  106. if len(tag.ndef.message):
  107. message = nfc.ndef.Message(tag.ndef.message)
  108. print "message type " + message.type
  109. for record in message:
  110. print "record type " + record.type
  111. if record.type == "application/vnd.wfa.wsc":
  112. print "WPS tag - send to hostapd"
  113. wpas_tag_read(tag.ndef.message)
  114. break
  115. else:
  116. print "Empty tag"
  117. print "Remove tag"
  118. while tag.is_present:
  119. time.sleep(0.1)
  120. def wps_write_config_tag(clf):
  121. print "Write WPS config token"
  122. data = wpas_get_config_token()
  123. if (data == None):
  124. print "Could not get WPS config token from hostapd"
  125. return
  126. print "Touch an NFC tag"
  127. while True:
  128. tag = clf.poll()
  129. if tag == None:
  130. time.sleep(0.1)
  131. continue
  132. break
  133. print "Tag found - writing"
  134. tag.ndef.message = data
  135. print "Done - remove tag"
  136. while tag.is_present:
  137. time.sleep(0.1)
  138. def wps_write_password_tag(clf):
  139. print "Write WPS password token"
  140. data = wpas_get_password_token()
  141. if (data == None):
  142. print "Could not get WPS password token from hostapd"
  143. return
  144. print "Touch an NFC tag"
  145. while True:
  146. tag = clf.poll()
  147. if tag == None:
  148. time.sleep(0.1)
  149. continue
  150. break
  151. print "Tag found - writing"
  152. tag.ndef.message = data
  153. print "Done - remove tag"
  154. while tag.is_present:
  155. time.sleep(0.1)
  156. def find_peer(clf):
  157. while True:
  158. if nfc.llcp.connected():
  159. print "LLCP connected"
  160. general_bytes = nfc.llcp.startup({})
  161. peer = clf.listen(ord(os.urandom(1)) + 200, general_bytes)
  162. if isinstance(peer, nfc.DEP):
  163. print "listen -> DEP";
  164. if peer.general_bytes.startswith("Ffm"):
  165. print "Found DEP"
  166. return peer
  167. print "mismatch in general_bytes"
  168. print peer.general_bytes
  169. peer = clf.poll(general_bytes)
  170. if isinstance(peer, nfc.DEP):
  171. print "poll -> DEP";
  172. if peer.general_bytes.startswith("Ffm"):
  173. print "Found DEP"
  174. return peer
  175. print "mismatch in general_bytes"
  176. print peer.general_bytes
  177. if peer:
  178. print "Found tag"
  179. return peer
  180. def main():
  181. clf = nfc.ContactlessFrontend()
  182. try:
  183. if len(sys.argv) > 1 and sys.argv[1] == "write-config":
  184. wps_write_config_tag(clf)
  185. raise SystemExit
  186. if len(sys.argv) > 1 and sys.argv[1] == "write-password":
  187. wps_write_password_tag(clf)
  188. raise SystemExit
  189. while True:
  190. print "Waiting for a tag or peer to be touched"
  191. tag = find_peer(clf)
  192. if isinstance(tag, nfc.DEP):
  193. wps_handover_resp(tag)
  194. continue
  195. if tag.ndef:
  196. wps_tag_read(tag)
  197. continue
  198. print "Not an NDEF tag - remove tag"
  199. while tag.is_present:
  200. time.sleep(0.1)
  201. except KeyboardInterrupt:
  202. raise SystemExit
  203. finally:
  204. clf.close()
  205. raise SystemExit
  206. if __name__ == '__main__':
  207. main()