wps-nfc.py 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219
  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 StringIO
  13. import nfc
  14. import nfc.ndef
  15. import nfc.llcp
  16. import nfc.handover
  17. import logging
  18. logging.basicConfig()
  19. import wpactrl
  20. wpas_ctrl = '/var/run/wpa_supplicant'
  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 wpa_supplicant: ", error
  28. return None
  29. if len(ifaces) < 1:
  30. print "No wpa_supplicant control interface found"
  31. return None
  32. for ctrl in ifaces:
  33. try:
  34. wpas = wpactrl.WPACtrl(ctrl)
  35. return wpas
  36. except wpactrl.error, error:
  37. print "Error: ", error
  38. pass
  39. return None
  40. def wpas_tag_read(message):
  41. wpas = wpas_connect()
  42. if (wpas == None):
  43. return
  44. print wpas.request("WPS_NFC_TAG_READ " + message.encode("hex"))
  45. def wpas_get_handover_req():
  46. wpas = wpas_connect()
  47. if (wpas == None):
  48. return None
  49. return wpas.request("NFC_GET_HANDOVER_REQ NDEF WPS-CR").rstrip().decode("hex")
  50. def wpas_put_handover_sel(message):
  51. wpas = wpas_connect()
  52. if (wpas == None):
  53. return
  54. print wpas.request("NFC_RX_HANDOVER_SEL " + str(message).encode("hex"))
  55. def wps_handover_init(peer):
  56. print "Trying to initiate WPS handover"
  57. data = wpas_get_handover_req()
  58. if (data == None):
  59. print "Could not get handover request carrier record from wpa_supplicant"
  60. return
  61. print "Handover request carrier record from wpa_supplicant: " + data.encode("hex")
  62. record = nfc.ndef.Record()
  63. f = StringIO.StringIO(data)
  64. record._read(f)
  65. record = nfc.ndef.HandoverCarrierRecord(record)
  66. print "Parsed handover request carrier record:"
  67. print record.pretty()
  68. message = nfc.ndef.HandoverRequestMessage(version="1.2")
  69. message.nonce = random.randint(0, 0xffff)
  70. message.add_carrier(record, "active")
  71. print "Handover request:"
  72. print message.pretty()
  73. nfc.llcp.activate(peer);
  74. client = nfc.handover.HandoverClient()
  75. try:
  76. print "Trying handover";
  77. client.connect()
  78. print "Connected for handover"
  79. except nfc.llcp.ConnectRefused:
  80. print "Handover connection refused"
  81. nfc.llcp.shutdown()
  82. client.close()
  83. return
  84. print "Sending handover request"
  85. if not client.send(message):
  86. print "Failed to send handover request"
  87. print "Receiving handover response"
  88. message = client._recv()
  89. if message is None:
  90. print "No response received"
  91. nfc.llcp.shutdown()
  92. client.close()
  93. return
  94. if message.type != "urn:nfc:wkt:Hs":
  95. print "Response was not Hs - received: " + message.type
  96. nfc.llcp.shutdown()
  97. client.close()
  98. return
  99. print "Received message"
  100. print message.pretty()
  101. message = nfc.ndef.HandoverSelectMessage(message)
  102. print "Handover select received"
  103. print message.pretty()
  104. for carrier in message.carriers:
  105. print "Remote carrier type: " + carrier.type
  106. if carrier.type == "application/vnd.wfa.wsc":
  107. print "WPS carrier type match - send to wpa_supplicant"
  108. wpas_put_handover_sel(carrier.record)
  109. wifi = nfc.ndef.WifiConfigRecord(carrier.record)
  110. print wifi.pretty()
  111. print "Remove peer"
  112. nfc.llcp.shutdown()
  113. client.close()
  114. print "Done with handover"
  115. def wps_tag_read(tag):
  116. if len(tag.ndef.message):
  117. message = nfc.ndef.Message(tag.ndef.message)
  118. print "message type " + message.type
  119. for record in message:
  120. print "record type " + record.type
  121. if record.type == "application/vnd.wfa.wsc":
  122. print "WPS tag - send to wpa_supplicant"
  123. wpas_tag_read(tag.ndef.message)
  124. break
  125. else:
  126. print "Empty tag"
  127. print "Remove tag"
  128. while tag.is_present:
  129. time.sleep(0.1)
  130. def find_peer(clf):
  131. while True:
  132. if nfc.llcp.connected():
  133. print "LLCP connected"
  134. general_bytes = nfc.llcp.startup({})
  135. peer = clf.listen(ord(os.urandom(1)) + 250, general_bytes)
  136. if isinstance(peer, nfc.DEP):
  137. print "listen -> DEP";
  138. if peer.general_bytes.startswith("Ffm"):
  139. print "Found DEP"
  140. return peer
  141. print "mismatch in general_bytes"
  142. print peer.general_bytes
  143. peer = clf.poll(general_bytes)
  144. if isinstance(peer, nfc.DEP):
  145. print "poll -> DEP";
  146. if peer.general_bytes.startswith("Ffm"):
  147. print "Found DEP"
  148. return peer
  149. print "mismatch in general_bytes"
  150. print peer.general_bytes
  151. if peer:
  152. print "Found tag"
  153. return peer
  154. def main():
  155. clf = nfc.ContactlessFrontend()
  156. try:
  157. while True:
  158. print "Waiting for a tag or peer to be touched"
  159. tag = find_peer(clf)
  160. if isinstance(tag, nfc.DEP):
  161. wps_handover_init(tag)
  162. continue
  163. if tag.ndef:
  164. wps_tag_read(tag)
  165. continue
  166. print "Not an NDEF tag - remove tag"
  167. while tag.is_present:
  168. time.sleep(0.1)
  169. except KeyboardInterrupt:
  170. raise SystemExit
  171. finally:
  172. clf.close()
  173. raise SystemExit
  174. if __name__ == '__main__':
  175. main()