2016-08-27 18:27:58 +00:00
|
|
|
import hashlib
|
2016-08-27 23:53:31 +00:00
|
|
|
from mock import Mock
|
2019-06-19 18:19:37 +00:00
|
|
|
|
|
|
|
from twisted.application.service import (
|
|
|
|
Service,
|
|
|
|
)
|
2014-11-23 05:46:33 +00:00
|
|
|
|
|
|
|
from twisted.trial import unittest
|
2016-04-26 17:44:58 +00:00
|
|
|
from twisted.internet.defer import succeed, inlineCallbacks
|
2016-04-25 23:39:33 +00:00
|
|
|
|
2019-06-19 18:19:37 +00:00
|
|
|
from allmydata.util import base32, yamlutil
|
2014-11-23 05:46:33 +00:00
|
|
|
from allmydata.storage_client import NativeStorageServer
|
2016-07-22 00:23:22 +00:00
|
|
|
from allmydata.storage_client import StorageFarmBroker
|
2014-11-23 05:46:33 +00:00
|
|
|
|
|
|
|
|
|
|
|
class NativeStorageServerWithVersion(NativeStorageServer):
|
2016-05-03 22:09:13 +00:00
|
|
|
def __init__(self, version):
|
|
|
|
# note: these instances won't work for anything other than
|
|
|
|
# get_available_space() because we don't upcall
|
|
|
|
self.version = version
|
2014-11-23 05:46:33 +00:00
|
|
|
def get_version(self):
|
|
|
|
return self.version
|
|
|
|
|
|
|
|
|
|
|
|
class TestNativeStorageServer(unittest.TestCase):
|
|
|
|
def test_get_available_space_new(self):
|
|
|
|
nss = NativeStorageServerWithVersion(
|
|
|
|
{ "http://allmydata.org/tahoe/protocols/storage/v1":
|
|
|
|
{ "maximum-immutable-share-size": 111,
|
|
|
|
"available-space": 222,
|
|
|
|
}
|
|
|
|
})
|
|
|
|
self.failUnlessEqual(nss.get_available_space(), 222)
|
|
|
|
|
|
|
|
def test_get_available_space_old(self):
|
|
|
|
nss = NativeStorageServerWithVersion(
|
|
|
|
{ "http://allmydata.org/tahoe/protocols/storage/v1":
|
|
|
|
{ "maximum-immutable-share-size": 111,
|
|
|
|
}
|
|
|
|
})
|
|
|
|
self.failUnlessEqual(nss.get_available_space(), 111)
|
|
|
|
|
2016-08-27 00:55:52 +00:00
|
|
|
def test_missing_nickname(self):
|
|
|
|
ann = {"anonymous-storage-FURL": "pb://w2hqnbaa25yw4qgcvghl5psa3srpfgw3@tcp:127.0.0.1:51309/vucto2z4fxment3vfxbqecblbf6zyp6x",
|
|
|
|
"permutation-seed-base32": "w2hqnbaa25yw4qgcvghl5psa3srpfgw3",
|
|
|
|
}
|
2016-08-27 23:53:31 +00:00
|
|
|
nss = NativeStorageServer("server_id", ann, None, {})
|
2016-08-27 00:55:52 +00:00
|
|
|
self.assertEqual(nss.get_nickname(), "")
|
2016-04-25 23:39:33 +00:00
|
|
|
|
2019-06-19 18:19:37 +00:00
|
|
|
|
|
|
|
class UnrecognizedAnnouncement(unittest.TestCase):
|
|
|
|
"""
|
|
|
|
Tests for handling of announcements that aren't recognized and don't use
|
|
|
|
*anonymous-storage-FURL*.
|
|
|
|
|
|
|
|
Recognition failure is created by making up something completely novel for
|
|
|
|
these tests. In real use, recognition failure would most likely come from
|
|
|
|
an announcement generated by a storage server plugin which is not loaded
|
|
|
|
in the client.
|
|
|
|
"""
|
|
|
|
ann = {
|
|
|
|
u"name": u"tahoe-lafs-testing-v1",
|
|
|
|
u"any-parameter": 12345,
|
|
|
|
}
|
|
|
|
server_id = b"abc"
|
|
|
|
|
|
|
|
def _tub_maker(self, overrides):
|
|
|
|
return Service()
|
|
|
|
|
|
|
|
def native_storage_server(self):
|
|
|
|
"""
|
|
|
|
Make a ``NativeStorageServer`` out of an unrecognizable announcement.
|
|
|
|
"""
|
|
|
|
return NativeStorageServer(
|
|
|
|
self.server_id,
|
|
|
|
self.ann,
|
|
|
|
self._tub_maker,
|
|
|
|
{},
|
|
|
|
)
|
|
|
|
|
|
|
|
def test_no_exceptions(self):
|
|
|
|
"""
|
|
|
|
``NativeStorageServer`` can be instantiated with an unrecognized
|
|
|
|
announcement.
|
|
|
|
"""
|
|
|
|
self.native_storage_server()
|
|
|
|
|
|
|
|
def test_start_connecting(self):
|
|
|
|
"""
|
|
|
|
``NativeStorageServer.start_connecting`` does not raise an exception.
|
|
|
|
"""
|
|
|
|
server = self.native_storage_server()
|
|
|
|
server.start_connecting(None)
|
|
|
|
|
|
|
|
def test_stop_connecting(self):
|
|
|
|
"""
|
|
|
|
``NativeStorageServer.stop_connecting`` does not raise an exception.
|
|
|
|
"""
|
|
|
|
server = self.native_storage_server()
|
|
|
|
server.start_connecting(None)
|
|
|
|
server.stop_connecting()
|
|
|
|
|
|
|
|
def test_try_to_connect(self):
|
|
|
|
"""
|
|
|
|
``NativeStorageServer.try_to_connect`` does not raise an exception.
|
|
|
|
"""
|
|
|
|
server = self.native_storage_server()
|
|
|
|
server.start_connecting(None)
|
|
|
|
server.try_to_connect()
|
|
|
|
|
2019-06-21 19:35:24 +00:00
|
|
|
def test_various_data_methods(self):
|
|
|
|
"""
|
|
|
|
The data accessors of ``NativeStorageServer`` that depend on the
|
|
|
|
announcement do not raise an exception.
|
|
|
|
"""
|
|
|
|
server = self.native_storage_server()
|
|
|
|
server.get_permutation_seed()
|
|
|
|
server.get_name()
|
|
|
|
server.get_longname()
|
|
|
|
server.get_tubid()
|
|
|
|
server.get_lease_seed()
|
|
|
|
server.get_foolscap_write_enabler_seed()
|
|
|
|
server.get_nickname()
|
|
|
|
|
2019-06-19 18:19:37 +00:00
|
|
|
|
2016-04-25 23:39:33 +00:00
|
|
|
class TestStorageFarmBroker(unittest.TestCase):
|
|
|
|
|
2016-08-27 00:31:02 +00:00
|
|
|
def test_static_servers(self):
|
2016-08-27 23:53:31 +00:00
|
|
|
broker = StorageFarmBroker(True, lambda h: Mock())
|
2016-08-24 21:11:58 +00:00
|
|
|
|
2016-08-28 02:41:20 +00:00
|
|
|
key_s = 'v0-1234-1'
|
|
|
|
servers_yaml = """\
|
|
|
|
storage:
|
|
|
|
v0-1234-1:
|
|
|
|
ann:
|
|
|
|
anonymous-storage-FURL: pb://ge@nowhere/fake
|
|
|
|
permutation-seed-base32: aaaaaaaaaaaaaaaaaaaaaaaa
|
|
|
|
"""
|
|
|
|
servers = yamlutil.safe_load(servers_yaml)
|
2016-08-27 00:31:02 +00:00
|
|
|
permseed = base32.a2b("aaaaaaaaaaaaaaaaaaaaaaaa")
|
2016-08-28 02:41:20 +00:00
|
|
|
broker.set_static_servers(servers["storage"])
|
2016-08-27 00:31:02 +00:00
|
|
|
self.failUnlessEqual(len(broker._static_server_ids), 1)
|
|
|
|
s = broker.servers[key_s]
|
2016-08-28 02:41:20 +00:00
|
|
|
self.failUnlessEqual(s.announcement,
|
|
|
|
servers["storage"]["v0-1234-1"]["ann"])
|
2016-08-27 00:31:02 +00:00
|
|
|
self.failUnlessEqual(s.get_serverid(), key_s)
|
|
|
|
self.assertEqual(s.get_permutation_seed(), permseed)
|
|
|
|
|
|
|
|
# if the Introducer announces the same thing, we're supposed to
|
|
|
|
# ignore it
|
|
|
|
|
|
|
|
ann2 = {
|
|
|
|
"service-name": "storage",
|
|
|
|
"anonymous-storage-FURL": "pb://{}@nowhere/fake2".format(base32.b2a(str(1))),
|
|
|
|
"permutation-seed-base32": "bbbbbbbbbbbbbbbbbbbbbbbb",
|
|
|
|
}
|
|
|
|
broker._got_announcement(key_s, ann2)
|
|
|
|
s2 = broker.servers[key_s]
|
|
|
|
self.assertIdentical(s2, s)
|
|
|
|
self.assertEqual(s2.get_permutation_seed(), permseed)
|
2016-08-24 21:11:58 +00:00
|
|
|
|
2016-08-27 18:27:58 +00:00
|
|
|
def test_static_permutation_seed_pubkey(self):
|
2016-08-27 23:53:31 +00:00
|
|
|
broker = StorageFarmBroker(True, lambda h: Mock())
|
2016-08-27 18:27:58 +00:00
|
|
|
server_id = "v0-4uazse3xb6uu5qpkb7tel2bm6bpea4jhuigdhqcuvvse7hugtsia"
|
|
|
|
k = "4uazse3xb6uu5qpkb7tel2bm6bpea4jhuigdhqcuvvse7hugtsia"
|
|
|
|
ann = {
|
|
|
|
"anonymous-storage-FURL": "pb://abcde@nowhere/fake",
|
|
|
|
}
|
2016-08-28 02:41:20 +00:00
|
|
|
broker.set_static_servers({server_id.decode("ascii"): {"ann": ann}})
|
2016-08-27 18:27:58 +00:00
|
|
|
s = broker.servers[server_id]
|
|
|
|
self.assertEqual(s.get_permutation_seed(), base32.a2b(k))
|
|
|
|
|
|
|
|
def test_static_permutation_seed_explicit(self):
|
2016-08-27 23:53:31 +00:00
|
|
|
broker = StorageFarmBroker(True, lambda h: Mock())
|
2016-08-27 18:27:58 +00:00
|
|
|
server_id = "v0-4uazse3xb6uu5qpkb7tel2bm6bpea4jhuigdhqcuvvse7hugtsia"
|
|
|
|
k = "w5gl5igiexhwmftwzhai5jy2jixn7yx7"
|
|
|
|
ann = {
|
|
|
|
"anonymous-storage-FURL": "pb://abcde@nowhere/fake",
|
|
|
|
"permutation-seed-base32": k,
|
|
|
|
}
|
2016-08-28 02:41:20 +00:00
|
|
|
broker.set_static_servers({server_id.decode("ascii"): {"ann": ann}})
|
2016-08-27 18:27:58 +00:00
|
|
|
s = broker.servers[server_id]
|
|
|
|
self.assertEqual(s.get_permutation_seed(), base32.a2b(k))
|
|
|
|
|
|
|
|
def test_static_permutation_seed_hashed(self):
|
2016-08-27 23:53:31 +00:00
|
|
|
broker = StorageFarmBroker(True, lambda h: Mock())
|
2016-08-27 18:27:58 +00:00
|
|
|
server_id = "unparseable"
|
|
|
|
ann = {
|
|
|
|
"anonymous-storage-FURL": "pb://abcde@nowhere/fake",
|
|
|
|
}
|
2016-08-28 02:41:20 +00:00
|
|
|
broker.set_static_servers({server_id.decode("ascii"): {"ann": ann}})
|
2016-08-27 18:27:58 +00:00
|
|
|
s = broker.servers[server_id]
|
|
|
|
self.assertEqual(s.get_permutation_seed(),
|
|
|
|
hashlib.sha256(server_id).digest())
|
|
|
|
|
2016-04-26 06:32:21 +00:00
|
|
|
@inlineCallbacks
|
2016-04-25 23:39:33 +00:00
|
|
|
def test_threshold_reached(self):
|
|
|
|
introducer = Mock()
|
2016-08-27 23:53:31 +00:00
|
|
|
new_tubs = []
|
|
|
|
def make_tub(*args, **kwargs):
|
|
|
|
return new_tubs.pop()
|
|
|
|
broker = StorageFarmBroker(True, make_tub)
|
2016-07-22 00:23:22 +00:00
|
|
|
done = broker.when_connected_enough(5)
|
2016-04-25 23:39:33 +00:00
|
|
|
broker.use_introducer(introducer)
|
|
|
|
# subscribes to "storage" to learn of new storage nodes
|
|
|
|
subscribe = introducer.mock_calls[0]
|
|
|
|
self.assertEqual(subscribe[0], 'subscribe_to')
|
2016-05-03 21:03:17 +00:00
|
|
|
self.assertEqual(subscribe[1][0], 'storage')
|
|
|
|
got_announcement = subscribe[1][1]
|
2016-04-25 23:39:33 +00:00
|
|
|
|
|
|
|
data = {
|
|
|
|
"service-name": "storage",
|
|
|
|
"anonymous-storage-FURL": None,
|
|
|
|
"permutation-seed-base32": "aaaaaaaaaaaaaaaaaaaaaaaa",
|
|
|
|
}
|
|
|
|
|
|
|
|
def add_one_server(x):
|
|
|
|
data["anonymous-storage-FURL"] = "pb://{}@nowhere/fake".format(base32.b2a(str(x)))
|
2016-05-03 21:49:31 +00:00
|
|
|
tub = Mock()
|
2016-08-27 23:53:31 +00:00
|
|
|
new_tubs.append(tub)
|
|
|
|
got_announcement('v0-1234-{}'.format(x), data)
|
|
|
|
self.assertEqual(tub.mock_calls[-1][0], 'connectTo')
|
|
|
|
got_connection = tub.mock_calls[-1][1][1]
|
2016-04-25 23:39:33 +00:00
|
|
|
rref = Mock()
|
|
|
|
rref.callRemote = Mock(return_value=succeed(1234))
|
|
|
|
got_connection(rref)
|
|
|
|
|
|
|
|
# first 4 shouldn't trigger connected_threashold
|
|
|
|
for x in range(4):
|
|
|
|
add_one_server(x)
|
|
|
|
self.assertFalse(done.called)
|
|
|
|
|
|
|
|
# ...but the 5th *should* trigger the threshold
|
|
|
|
add_one_server(42)
|
2016-04-26 06:32:21 +00:00
|
|
|
|
|
|
|
# so: the OneShotObserverList only notifies via
|
|
|
|
# foolscap.eventually() -- which forces the Deferred call
|
|
|
|
# through the reactor -- so it's no longer synchronous,
|
|
|
|
# meaning that we have to do "real reactor stuff" for the
|
|
|
|
# Deferred from when_connected_enough() to actually fire. (or
|
|
|
|
# @patch() out the reactor in foolscap.eventually to be a
|
|
|
|
# Clock() so we can advance time ourselves, but ... luckily
|
|
|
|
# eventually() uses 0 as the timeout currently)
|
|
|
|
|
|
|
|
yield done
|
2016-04-25 23:39:33 +00:00
|
|
|
self.assertTrue(done.called)
|