1
0

knode.py 2.8 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283
  1. # The contents of this file are subject to the BitTorrent Open Source License
  2. # Version 1.1 (the License). You may not copy or use this file, in either
  3. # source code or executable form, except in compliance with the License. You
  4. # may obtain a copy of the License at http://www.bittorrent.com/license/.
  5. #
  6. # Software distributed under the License is distributed on an AS IS basis,
  7. # WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
  8. # for the specific language governing rights and limitations under the
  9. # License.
  10. from node import Node
  11. from defer import Deferred
  12. from const import NULL_ID
  13. from krpc import KRPCProtocolError
  14. class IDChecker:
  15. def __init__(self, id):
  16. self.id = id
  17. class KNodeBase(Node):
  18. __slots__= ('cfa', 'table')
  19. def __init__(self, cfa):
  20. Node.__init__(self)
  21. self.cfa = cfa
  22. def conn(self):
  23. return self.cfa((self.host, self.port))
  24. def checkSender(self, dict):
  25. try:
  26. senderid = dict['rsp']['id']
  27. except KeyError:
  28. raise KRPCProtocolError, "No peer id in response."
  29. else:
  30. if self.id != NULL_ID and senderid != self.id:
  31. self.table.table.invalidateNode(self)
  32. else:
  33. if self.id == NULL_ID:
  34. self.id = senderid
  35. self.table.insertNode(self, contacted=1)
  36. return dict
  37. def errBack(self, err):
  38. self.table.table.nodeFailed(self)
  39. return err[0]
  40. def ping(self, id):
  41. df = self.conn().sendRequest('ping', {"id":id})
  42. self.conn().pinging = True
  43. def endping(x):
  44. self.conn().pinging = False
  45. return x
  46. df.addCallbacks(endping, endping)
  47. df.addCallbacks(self.checkSender, self.errBack)
  48. return df
  49. def findNode(self, target, id):
  50. df = self.conn().sendRequest('find_node', {"target" : target, "id": id})
  51. df.addErrback(self.errBack)
  52. df.addCallback(self.checkSender)
  53. return df
  54. def inPing(self):
  55. return self.conn().pinging
  56. class KNodeRead(KNodeBase):
  57. def findValue(self, key, id):
  58. df = self.conn().sendRequest('find_value', {"key" : key, "id" : id})
  59. df.addErrback(self.errBack)
  60. df.addCallback(self.checkSender)
  61. return df
  62. class KNodeWrite(KNodeRead):
  63. def storeValue(self, key, value, id):
  64. df = self.conn().sendRequest('store_value', {"key" : key, "value" : value, "id": id})
  65. df.addErrback(self.errBack)
  66. df.addCallback(self.checkSender)
  67. return df
  68. def storeValues(self, key, value, id):
  69. df = self.conn().sendRequest('store_values', {"key" : key, "values" : value, "id": id})
  70. df.addErrback(self.errBack)
  71. df.addCallback(self.checkSender)
  72. return df