test_dht_storage.py 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130
  1. import time
  2. from hivemind.dht.storage import DHTID, DHTLocalStorage, DictionaryDHTValue
  3. from hivemind.utils.serializer import MSGPackSerializer
  4. from hivemind.utils.timed_storage import get_dht_time
  5. def test_store():
  6. d = DHTLocalStorage()
  7. d.store(DHTID.generate("key"), b"val", get_dht_time() + 0.5)
  8. assert d.get(DHTID.generate("key"))[0] == b"val", "Wrong value"
  9. def test_get_expired():
  10. d = DHTLocalStorage()
  11. d.store(DHTID.generate("key"), b"val", get_dht_time() + 0.1)
  12. time.sleep(0.5)
  13. assert d.get(DHTID.generate("key")) is None, "Expired value must be deleted"
  14. def test_get_empty():
  15. d = DHTLocalStorage()
  16. assert d.get(DHTID.generate(source="key")) is None, "DHTLocalStorage returned non-existent value"
  17. def test_change_expiration_time():
  18. d = DHTLocalStorage()
  19. d.store(DHTID.generate("key"), b"val1", get_dht_time() + 1)
  20. assert d.get(DHTID.generate("key"))[0] == b"val1", "Wrong value"
  21. d.store(DHTID.generate("key"), b"val2", get_dht_time() + 200)
  22. time.sleep(1)
  23. assert d.get(DHTID.generate("key"))[0] == b"val2", "Value must be changed, but still kept in table"
  24. def test_maxsize_cache():
  25. d = DHTLocalStorage(maxsize=2)
  26. d.store(DHTID.generate("key1a"), b"val1a", get_dht_time() + 1)
  27. d.store(DHTID.generate("key1b"), b"val1b", get_dht_time() + 1)
  28. d.store(DHTID.generate("key1a"), b"val1a2", get_dht_time() + 2)
  29. d.store(DHTID.generate("key2"), b"val2", get_dht_time() + 200)
  30. assert d.get(DHTID.generate("key2"))[0] == b"val2", "Value with bigger exp. time must be kept"
  31. assert d.get(DHTID.generate("key1a"))[0] == b"val1a2", "Value with bigger exp. time must be kept"
  32. assert d.get(DHTID.generate("key1b")) is None, "Value with less exp time, must be deleted"
  33. def test_localstorage_top():
  34. d = DHTLocalStorage(maxsize=3)
  35. d.store(DHTID.generate("key1"), b"val1", get_dht_time() + 1)
  36. d.store(DHTID.generate("key2"), b"val2", get_dht_time() + 2)
  37. d.store(DHTID.generate("key3"), b"val3", get_dht_time() + 4)
  38. assert d.top()[0] == DHTID.generate("key1") and d.top()[1].value == b"val1"
  39. d.store(DHTID.generate("key1"), b"val1_new", get_dht_time() + 3)
  40. assert d.top()[0] == DHTID.generate("key2") and d.top()[1].value == b"val2"
  41. del d[DHTID.generate("key2")]
  42. assert d.top()[0] == DHTID.generate("key1") and d.top()[1].value == b"val1_new"
  43. d.store(DHTID.generate("key2"), b"val2_new", get_dht_time() + 5)
  44. d.store(DHTID.generate("key4"), b"val4", get_dht_time() + 6) # key4 will push out key1 due to maxsize
  45. assert d.top()[0] == DHTID.generate("key3") and d.top()[1].value == b"val3"
  46. def test_localstorage_nested():
  47. time = get_dht_time()
  48. d1 = DHTLocalStorage()
  49. d2 = DictionaryDHTValue()
  50. d2.store("subkey1", b"value1", time + 2)
  51. d2.store("subkey2", b"value2", time + 3)
  52. d2.store("subkey3", b"value3", time + 1)
  53. assert d2.latest_expiration_time == time + 3
  54. for subkey, (subvalue, subexpiration) in d2.items():
  55. assert d1.store_subkey(DHTID.generate("foo"), subkey, subvalue, subexpiration)
  56. assert d1.store(DHTID.generate("bar"), b"456", time + 2)
  57. assert d1.get(DHTID.generate("foo"))[0].data == d2.data
  58. assert d1.get(DHTID.generate("foo"))[1] == d2.latest_expiration_time
  59. assert d1.get(DHTID.generate("foo"))[0].get("subkey1") == (b"value1", time + 2)
  60. assert len(d1.get(DHTID.generate("foo"))[0]) == 3
  61. assert d1.store_subkey(DHTID.generate("foo"), "subkey4", b"value4", time + 4)
  62. assert len(d1.get(DHTID.generate("foo"))[0]) == 4
  63. assert (
  64. d1.store_subkey(DHTID.generate("bar"), "subkeyA", b"valueA", time + 1) is False
  65. ) # prev has better expiration
  66. assert d1.store_subkey(DHTID.generate("bar"), "subkeyA", b"valueA", time + 3) # new value has better expiration
  67. assert d1.store_subkey(DHTID.generate("bar"), "subkeyB", b"valueB", time + 4) # new value has better expiration
  68. assert d1.store_subkey(DHTID.generate("bar"), "subkeyA", b"valueA+", time + 5) # overwrite subkeyA under key bar
  69. assert all(subkey in d1.get(DHTID.generate("bar"))[0] for subkey in ("subkeyA", "subkeyB"))
  70. assert len(d1.get(DHTID.generate("bar"))[0]) == 2 and d1.get(DHTID.generate("bar"))[1] == time + 5
  71. assert d1.store(DHTID.generate("foo"), b"nothing", time + 3.5) is False # previous value has better expiration
  72. assert d1.get(DHTID.generate("foo"))[0].get("subkey2") == (b"value2", time + 3)
  73. assert d1.store(DHTID.generate("foo"), b"nothing", time + 5) is True # new value has better expiraiton
  74. assert d1.get(DHTID.generate("foo")) == (b"nothing", time + 5) # value should be replaced
  75. def test_localstorage_freeze():
  76. d = DHTLocalStorage(maxsize=2)
  77. with d.freeze():
  78. d.store(DHTID.generate("key1"), b"val1", get_dht_time() + 0.01)
  79. assert DHTID.generate("key1") in d
  80. time.sleep(0.03)
  81. assert DHTID.generate("key1") in d
  82. assert DHTID.generate("key1") not in d
  83. with d.freeze():
  84. d.store(DHTID.generate("key1"), b"val1", get_dht_time() + 1)
  85. d.store(DHTID.generate("key2"), b"val2", get_dht_time() + 2)
  86. d.store(DHTID.generate("key3"), b"val3", get_dht_time() + 3) # key3 will push key1 out due to maxsize
  87. assert DHTID.generate("key1") in d
  88. assert DHTID.generate("key1") not in d
  89. def test_localstorage_serialize():
  90. d1 = DictionaryDHTValue()
  91. d2 = DictionaryDHTValue()
  92. now = get_dht_time()
  93. d1.store("key1", b"ololo", now + 1)
  94. d2.store("key2", b"pysh", now + 1)
  95. d2.store("key3", b"pyshpysh", now + 2)
  96. data = MSGPackSerializer.dumps([d1, d2, 123321])
  97. assert isinstance(data, bytes)
  98. new_d1, new_d2, new_value = MSGPackSerializer.loads(data)
  99. assert isinstance(new_d1, DictionaryDHTValue) and isinstance(new_d2, DictionaryDHTValue) and new_value == 123321
  100. assert "key1" in new_d1 and len(new_d1) == 1
  101. assert "key1" not in new_d2 and len(new_d2) == 2 and new_d2.get("key3") == (b"pyshpysh", now + 2)