2007-04-18 03:03:44 +00:00
|
|
|
|
|
|
|
from twisted.trial import unittest
|
|
|
|
|
2007-04-18 22:42:34 +00:00
|
|
|
from twisted.application import service
|
2007-07-13 21:04:49 +00:00
|
|
|
from twisted.internet import defer
|
2007-04-18 22:42:34 +00:00
|
|
|
from foolscap import Referenceable
|
2007-04-18 03:03:44 +00:00
|
|
|
import os.path
|
2007-08-28 06:41:40 +00:00
|
|
|
import itertools
|
2007-07-13 22:09:01 +00:00
|
|
|
from allmydata import interfaces
|
2007-07-13 21:04:49 +00:00
|
|
|
from allmydata.util import fileutil, hashutil
|
2007-07-14 00:25:45 +00:00
|
|
|
from allmydata.storage import BucketWriter, BucketReader, \
|
2007-07-13 22:09:01 +00:00
|
|
|
WriteBucketProxy, ReadBucketProxy, StorageServer
|
2007-04-18 03:03:44 +00:00
|
|
|
|
2007-04-18 22:42:34 +00:00
|
|
|
class Bucket(unittest.TestCase):
|
2007-04-18 03:03:44 +00:00
|
|
|
def make_workdir(self, name):
|
2007-07-13 21:04:49 +00:00
|
|
|
basedir = os.path.join("storage", "Bucket", name)
|
2007-04-18 03:21:05 +00:00
|
|
|
incoming = os.path.join(basedir, "tmp", "bucket")
|
|
|
|
final = os.path.join(basedir, "bucket")
|
2007-04-18 03:03:44 +00:00
|
|
|
fileutil.make_dirs(basedir)
|
2007-07-13 21:04:49 +00:00
|
|
|
fileutil.make_dirs(os.path.join(basedir, "tmp"))
|
2007-04-18 03:21:05 +00:00
|
|
|
return incoming, final
|
2007-04-18 03:03:44 +00:00
|
|
|
|
2007-07-04 00:38:49 +00:00
|
|
|
def bucket_writer_closed(self, bw, consumed):
|
2007-07-04 00:08:02 +00:00
|
|
|
pass
|
|
|
|
|
2007-04-18 03:03:44 +00:00
|
|
|
def test_create(self):
|
2007-04-18 03:21:05 +00:00
|
|
|
incoming, final = self.make_workdir("test_create")
|
2007-07-13 22:09:01 +00:00
|
|
|
bw = BucketWriter(self, incoming, final, 200)
|
2007-07-13 21:04:49 +00:00
|
|
|
bw.remote_write(0, "a"*25)
|
|
|
|
bw.remote_write(25, "b"*25)
|
|
|
|
bw.remote_write(50, "c"*25)
|
|
|
|
bw.remote_write(75, "d"*7)
|
2007-04-18 03:03:44 +00:00
|
|
|
bw.remote_close()
|
|
|
|
|
|
|
|
def test_readwrite(self):
|
2007-04-18 03:21:05 +00:00
|
|
|
incoming, final = self.make_workdir("test_readwrite")
|
2007-07-13 22:09:01 +00:00
|
|
|
bw = BucketWriter(self, incoming, final, 200)
|
2007-07-13 21:04:49 +00:00
|
|
|
bw.remote_write(0, "a"*25)
|
|
|
|
bw.remote_write(25, "b"*25)
|
|
|
|
bw.remote_write(50, "c"*7) # last block may be short
|
2007-04-18 03:03:44 +00:00
|
|
|
bw.remote_close()
|
|
|
|
|
|
|
|
# now read from it
|
2007-07-13 22:09:01 +00:00
|
|
|
br = BucketReader(final)
|
2007-07-13 21:04:49 +00:00
|
|
|
self.failUnlessEqual(br.remote_read(0, 25), "a"*25)
|
|
|
|
self.failUnlessEqual(br.remote_read(25, 25), "b"*25)
|
|
|
|
self.failUnlessEqual(br.remote_read(50, 7), "c"*7)
|
|
|
|
|
|
|
|
class RemoteBucket:
|
|
|
|
|
|
|
|
def callRemote(self, methname, *args, **kwargs):
|
|
|
|
def _call():
|
|
|
|
meth = getattr(self.target, "remote_" + methname)
|
|
|
|
return meth(*args, **kwargs)
|
|
|
|
return defer.maybeDeferred(_call)
|
|
|
|
|
|
|
|
class BucketProxy(unittest.TestCase):
|
|
|
|
def make_bucket(self, name, size):
|
|
|
|
basedir = os.path.join("storage", "BucketProxy", name)
|
|
|
|
incoming = os.path.join(basedir, "tmp", "bucket")
|
|
|
|
final = os.path.join(basedir, "bucket")
|
|
|
|
fileutil.make_dirs(basedir)
|
|
|
|
fileutil.make_dirs(os.path.join(basedir, "tmp"))
|
2007-07-13 22:09:01 +00:00
|
|
|
bw = BucketWriter(self, incoming, final, size)
|
2007-07-13 21:04:49 +00:00
|
|
|
rb = RemoteBucket()
|
|
|
|
rb.target = bw
|
|
|
|
return bw, rb, final
|
|
|
|
|
|
|
|
def bucket_writer_closed(self, bw, consumed):
|
|
|
|
pass
|
|
|
|
|
|
|
|
def test_create(self):
|
|
|
|
bw, rb, final = self.make_bucket("test_create", 500)
|
2007-07-13 22:09:01 +00:00
|
|
|
bp = WriteBucketProxy(rb,
|
|
|
|
data_size=300,
|
|
|
|
segment_size=10,
|
|
|
|
num_segments=5,
|
|
|
|
num_share_hashes=3,
|
|
|
|
uri_extension_size=500)
|
2007-07-13 21:04:49 +00:00
|
|
|
self.failUnless(interfaces.IStorageBucketWriter.providedBy(bp))
|
|
|
|
|
|
|
|
def test_readwrite(self):
|
|
|
|
# Let's pretend each share has 100 bytes of data, and that there are
|
|
|
|
# 4 segments (25 bytes each), and 8 shares total. So the three
|
|
|
|
# per-segment merkle trees (plaintext_hash_tree, crypttext_hash_tree,
|
|
|
|
# block_hashes) will have 4 leaves and 7 nodes each. The per-share
|
|
|
|
# merkle tree (share_hashes) has 8 leaves and 15 nodes, and we need 3
|
|
|
|
# nodes. Furthermore, let's assume the uri_extension is 500 bytes
|
|
|
|
# long. That should make the whole share:
|
|
|
|
#
|
|
|
|
# 0x1c + 100 + 7*32 + 7*32 + 7*32 + 3*(2+32) + 4+500 = 1406 bytes long
|
|
|
|
|
|
|
|
plaintext_hashes = [hashutil.tagged_hash("plain", "bar%d" % i)
|
|
|
|
for i in range(7)]
|
|
|
|
crypttext_hashes = [hashutil.tagged_hash("crypt", "bar%d" % i)
|
|
|
|
for i in range(7)]
|
|
|
|
block_hashes = [hashutil.tagged_hash("block", "bar%d" % i)
|
|
|
|
for i in range(7)]
|
|
|
|
share_hashes = [(i, hashutil.tagged_hash("share", "bar%d" % i))
|
|
|
|
for i in (1,9,13)]
|
|
|
|
uri_extension = "s" + "E"*498 + "e"
|
|
|
|
|
|
|
|
bw, rb, final = self.make_bucket("test_readwrite", 1406)
|
2007-07-13 22:09:01 +00:00
|
|
|
bp = WriteBucketProxy(rb,
|
2007-07-13 23:38:25 +00:00
|
|
|
data_size=95,
|
2007-07-13 22:09:01 +00:00
|
|
|
segment_size=25,
|
|
|
|
num_segments=4,
|
|
|
|
num_share_hashes=3,
|
|
|
|
uri_extension_size=len(uri_extension))
|
2007-07-13 21:04:49 +00:00
|
|
|
|
|
|
|
d = bp.start()
|
|
|
|
d.addCallback(lambda res: bp.put_block(0, "a"*25))
|
|
|
|
d.addCallback(lambda res: bp.put_block(1, "b"*25))
|
|
|
|
d.addCallback(lambda res: bp.put_block(2, "c"*25))
|
2007-07-13 23:38:25 +00:00
|
|
|
d.addCallback(lambda res: bp.put_block(3, "d"*20))
|
2007-07-13 21:04:49 +00:00
|
|
|
d.addCallback(lambda res: bp.put_plaintext_hashes(plaintext_hashes))
|
|
|
|
d.addCallback(lambda res: bp.put_crypttext_hashes(crypttext_hashes))
|
|
|
|
d.addCallback(lambda res: bp.put_block_hashes(block_hashes))
|
|
|
|
d.addCallback(lambda res: bp.put_share_hashes(share_hashes))
|
|
|
|
d.addCallback(lambda res: bp.put_uri_extension(uri_extension))
|
|
|
|
d.addCallback(lambda res: bp.close())
|
|
|
|
|
|
|
|
# now read everything back
|
|
|
|
def _start_reading(res):
|
2007-07-13 22:09:01 +00:00
|
|
|
br = BucketReader(final)
|
2007-07-13 21:04:49 +00:00
|
|
|
rb = RemoteBucket()
|
|
|
|
rb.target = br
|
2007-07-13 22:09:01 +00:00
|
|
|
rbp = ReadBucketProxy(rb)
|
2007-07-13 21:04:49 +00:00
|
|
|
self.failUnless(interfaces.IStorageBucketReader.providedBy(rbp))
|
|
|
|
|
2007-07-13 22:09:01 +00:00
|
|
|
d1 = rbp.startIfNecessary()
|
2007-07-13 21:04:49 +00:00
|
|
|
d1.addCallback(lambda res: rbp.get_block(0))
|
|
|
|
d1.addCallback(lambda res: self.failUnlessEqual(res, "a"*25))
|
|
|
|
d1.addCallback(lambda res: rbp.get_block(1))
|
|
|
|
d1.addCallback(lambda res: self.failUnlessEqual(res, "b"*25))
|
|
|
|
d1.addCallback(lambda res: rbp.get_block(2))
|
|
|
|
d1.addCallback(lambda res: self.failUnlessEqual(res, "c"*25))
|
|
|
|
d1.addCallback(lambda res: rbp.get_block(3))
|
2007-07-13 23:38:25 +00:00
|
|
|
d1.addCallback(lambda res: self.failUnlessEqual(res, "d"*20))
|
2007-07-13 21:04:49 +00:00
|
|
|
|
|
|
|
d1.addCallback(lambda res: rbp.get_plaintext_hashes())
|
|
|
|
d1.addCallback(lambda res:
|
|
|
|
self.failUnlessEqual(res, plaintext_hashes))
|
|
|
|
d1.addCallback(lambda res: rbp.get_crypttext_hashes())
|
|
|
|
d1.addCallback(lambda res:
|
|
|
|
self.failUnlessEqual(res, crypttext_hashes))
|
|
|
|
d1.addCallback(lambda res: rbp.get_block_hashes())
|
|
|
|
d1.addCallback(lambda res: self.failUnlessEqual(res, block_hashes))
|
|
|
|
d1.addCallback(lambda res: rbp.get_share_hashes())
|
|
|
|
d1.addCallback(lambda res: self.failUnlessEqual(res, share_hashes))
|
|
|
|
d1.addCallback(lambda res: rbp.get_uri_extension())
|
|
|
|
d1.addCallback(lambda res:
|
|
|
|
self.failUnlessEqual(res, uri_extension))
|
|
|
|
|
|
|
|
return d1
|
|
|
|
|
|
|
|
d.addCallback(_start_reading)
|
|
|
|
|
|
|
|
return d
|
|
|
|
|
|
|
|
|
2007-04-18 03:03:44 +00:00
|
|
|
|
2007-04-18 22:42:34 +00:00
|
|
|
class Server(unittest.TestCase):
|
|
|
|
|
|
|
|
def setUp(self):
|
|
|
|
self.sparent = service.MultiService()
|
2007-08-28 06:41:40 +00:00
|
|
|
self._secret = itertools.count()
|
2007-04-18 22:42:34 +00:00
|
|
|
def tearDown(self):
|
|
|
|
return self.sparent.stopService()
|
|
|
|
|
|
|
|
def workdir(self, name):
|
2007-07-04 00:08:02 +00:00
|
|
|
basedir = os.path.join("storage", "Server", name)
|
2007-04-18 22:42:34 +00:00
|
|
|
return basedir
|
|
|
|
|
2007-07-04 00:08:02 +00:00
|
|
|
def create(self, name, sizelimit=None):
|
2007-04-18 22:42:34 +00:00
|
|
|
workdir = self.workdir(name)
|
2007-07-13 22:09:01 +00:00
|
|
|
ss = StorageServer(workdir, sizelimit)
|
2007-04-18 22:42:34 +00:00
|
|
|
ss.setServiceParent(self.sparent)
|
|
|
|
return ss
|
|
|
|
|
|
|
|
def test_create(self):
|
|
|
|
ss = self.create("test_create")
|
|
|
|
|
2007-08-28 06:41:40 +00:00
|
|
|
def allocate(self, ss, storage_index, sharenums, size):
|
|
|
|
renew_secret = hashutil.tagged_hash("blah", "%d" % self._secret.next())
|
|
|
|
cancel_secret = hashutil.tagged_hash("blah", "%d" % self._secret.next())
|
|
|
|
return ss.remote_allocate_buckets(storage_index,
|
|
|
|
renew_secret, cancel_secret,
|
|
|
|
sharenums, size, Referenceable())
|
|
|
|
|
2007-04-18 22:42:34 +00:00
|
|
|
def test_allocate(self):
|
|
|
|
ss = self.create("test_allocate")
|
|
|
|
|
|
|
|
self.failUnlessEqual(ss.remote_get_buckets("vid"), {})
|
|
|
|
|
|
|
|
canary = Referenceable()
|
2007-08-28 06:41:40 +00:00
|
|
|
already,writers = self.allocate(ss, "vid", [0,1,2], 75)
|
2007-04-18 22:42:34 +00:00
|
|
|
self.failUnlessEqual(already, set())
|
|
|
|
self.failUnlessEqual(set(writers.keys()), set([0,1,2]))
|
|
|
|
|
|
|
|
# while the buckets are open, they should not count as readable
|
|
|
|
self.failUnlessEqual(ss.remote_get_buckets("vid"), {})
|
|
|
|
|
|
|
|
for i,wb in writers.items():
|
2007-07-13 21:04:49 +00:00
|
|
|
wb.remote_write(0, "%25d" % i)
|
2007-04-18 22:42:34 +00:00
|
|
|
wb.remote_close()
|
|
|
|
|
|
|
|
# now they should be readable
|
|
|
|
b = ss.remote_get_buckets("vid")
|
|
|
|
self.failUnlessEqual(set(b.keys()), set([0,1,2]))
|
2007-07-13 21:04:49 +00:00
|
|
|
self.failUnlessEqual(b[0].remote_read(0, 25), "%25d" % 0)
|
2007-04-18 22:42:34 +00:00
|
|
|
|
|
|
|
# now if we about writing again, the server should offer those three
|
|
|
|
# buckets as already present
|
2007-08-28 06:41:40 +00:00
|
|
|
already,writers = self.allocate(ss, "vid", [0,1,2,3,4], 75)
|
2007-04-18 22:42:34 +00:00
|
|
|
self.failUnlessEqual(already, set([0,1,2]))
|
|
|
|
self.failUnlessEqual(set(writers.keys()), set([3,4]))
|
|
|
|
|
|
|
|
# while those two buckets are open for writing, the server should
|
|
|
|
# tell new uploaders that they already exist (so that we don't try to
|
|
|
|
# upload into them a second time)
|
|
|
|
|
2007-08-28 06:41:40 +00:00
|
|
|
already,writers = self.allocate(ss, "vid", [2,3,4,5], 75)
|
2007-04-18 22:42:34 +00:00
|
|
|
self.failUnlessEqual(already, set([2,3,4]))
|
|
|
|
self.failUnlessEqual(set(writers.keys()), set([5]))
|
|
|
|
|
2007-07-04 00:08:02 +00:00
|
|
|
def test_sizelimits(self):
|
|
|
|
ss = self.create("test_sizelimits", 100)
|
|
|
|
canary = Referenceable()
|
|
|
|
|
2007-08-28 06:41:40 +00:00
|
|
|
already,writers = self.allocate(ss, "vid1", [0,1,2], 25)
|
2007-07-04 00:08:02 +00:00
|
|
|
self.failUnlessEqual(len(writers), 3)
|
|
|
|
# now the StorageServer should have 75 bytes provisionally allocated,
|
|
|
|
# allowing only 25 more to be claimed
|
2007-07-13 21:04:49 +00:00
|
|
|
self.failUnlessEqual(len(ss._active_writers), 3)
|
2007-07-04 00:08:02 +00:00
|
|
|
|
2007-08-28 06:41:40 +00:00
|
|
|
already2,writers2 = self.allocate(ss, "vid2", [0,1,2], 25)
|
2007-07-04 00:08:02 +00:00
|
|
|
self.failUnlessEqual(len(writers2), 1)
|
2007-07-13 21:04:49 +00:00
|
|
|
self.failUnlessEqual(len(ss._active_writers), 4)
|
2007-07-04 00:08:02 +00:00
|
|
|
|
|
|
|
# we abandon the first set, so their provisional allocation should be
|
|
|
|
# returned
|
|
|
|
del already
|
|
|
|
del writers
|
2007-07-13 21:04:49 +00:00
|
|
|
self.failUnlessEqual(len(ss._active_writers), 1)
|
2007-07-04 00:08:02 +00:00
|
|
|
|
|
|
|
# and we close the second set, so their provisional allocation should
|
|
|
|
# become real, long-term allocation
|
|
|
|
for bw in writers2.values():
|
2007-07-13 21:04:49 +00:00
|
|
|
bw.remote_write(0, "a"*25)
|
2007-07-04 00:08:02 +00:00
|
|
|
bw.remote_close()
|
|
|
|
del already2
|
|
|
|
del writers2
|
|
|
|
del bw
|
2007-07-13 21:04:49 +00:00
|
|
|
self.failUnlessEqual(len(ss._active_writers), 0)
|
2007-07-04 00:08:02 +00:00
|
|
|
|
|
|
|
# now there should be 25 bytes allocated, and 75 free
|
2007-08-28 06:41:40 +00:00
|
|
|
already3,writers3 = self.allocate(ss,"vid3", [0,1,2,3], 25)
|
2007-07-04 00:08:02 +00:00
|
|
|
self.failUnlessEqual(len(writers3), 3)
|
2007-07-13 21:04:49 +00:00
|
|
|
self.failUnlessEqual(len(ss._active_writers), 3)
|
2007-07-04 00:08:02 +00:00
|
|
|
|
|
|
|
del already3
|
|
|
|
del writers3
|
2007-07-13 21:04:49 +00:00
|
|
|
self.failUnlessEqual(len(ss._active_writers), 0)
|
2007-07-04 00:08:02 +00:00
|
|
|
ss.disownServiceParent()
|
|
|
|
del ss
|
|
|
|
|
|
|
|
# creating a new StorageServer in the same directory should see the
|
|
|
|
# same usage. note that metadata will be counted at startup but not
|
|
|
|
# during runtime, so if we were creating any metadata, the allocation
|
|
|
|
# would be more than 25 bytes and this test would need to be changed.
|
|
|
|
ss = self.create("test_sizelimits", 100)
|
2007-08-28 06:41:40 +00:00
|
|
|
already4,writers4 = self.allocate(ss, "vid4", [0,1,2,3], 25)
|
2007-07-04 00:08:02 +00:00
|
|
|
self.failUnlessEqual(len(writers4), 3)
|
2007-07-13 21:04:49 +00:00
|
|
|
self.failUnlessEqual(len(ss._active_writers), 3)
|
2007-08-28 06:41:40 +00:00
|
|
|
|
|
|
|
def test_leases(self):
|
|
|
|
ss = self.create("test_leases")
|
|
|
|
canary = Referenceable()
|
|
|
|
sharenums = range(5)
|
|
|
|
size = 100
|
|
|
|
|
|
|
|
rs0,cs0 = (hashutil.tagged_hash("blah", "%d" % self._secret.next()),
|
|
|
|
hashutil.tagged_hash("blah", "%d" % self._secret.next()))
|
|
|
|
already,writers = ss.remote_allocate_buckets("si0", rs0, cs0,
|
|
|
|
sharenums, size, canary)
|
|
|
|
self.failUnlessEqual(len(already), 0)
|
|
|
|
self.failUnlessEqual(len(writers), 5)
|
|
|
|
for wb in writers.values():
|
|
|
|
wb.remote_close()
|
|
|
|
|
|
|
|
rs1,cs1 = (hashutil.tagged_hash("blah", "%d" % self._secret.next()),
|
|
|
|
hashutil.tagged_hash("blah", "%d" % self._secret.next()))
|
|
|
|
already,writers = ss.remote_allocate_buckets("si1", rs1, cs1,
|
|
|
|
sharenums, size, canary)
|
|
|
|
for wb in writers.values():
|
|
|
|
wb.remote_close()
|
|
|
|
|
|
|
|
# take out a second lease on si1
|
|
|
|
rs2,cs2 = (hashutil.tagged_hash("blah", "%d" % self._secret.next()),
|
|
|
|
hashutil.tagged_hash("blah", "%d" % self._secret.next()))
|
|
|
|
already,writers = ss.remote_allocate_buckets("si1", rs2, cs2,
|
|
|
|
sharenums, size, canary)
|
|
|
|
self.failUnlessEqual(len(already), 5)
|
|
|
|
self.failUnlessEqual(len(writers), 0)
|
|
|
|
|
|
|
|
# check that si0 is readable
|
|
|
|
readers = ss.remote_get_buckets("si0")
|
|
|
|
self.failUnlessEqual(len(readers), 5)
|
|
|
|
|
|
|
|
# renew the first lease. Only the proper renew_secret should work
|
|
|
|
ss.remote_renew_lease("si0", rs0)
|
|
|
|
self.failUnlessRaises(IndexError, ss.remote_renew_lease, "si0", cs0)
|
|
|
|
self.failUnlessRaises(IndexError, ss.remote_renew_lease, "si0", rs1)
|
|
|
|
|
|
|
|
# check that si0 is still readable
|
|
|
|
readers = ss.remote_get_buckets("si0")
|
|
|
|
self.failUnlessEqual(len(readers), 5)
|
|
|
|
|
|
|
|
# now cancel it
|
|
|
|
self.failUnlessRaises(IndexError, ss.remote_cancel_lease, "si0", rs0)
|
|
|
|
self.failUnlessRaises(IndexError, ss.remote_cancel_lease, "si0", cs1)
|
|
|
|
ss.remote_cancel_lease("si0", cs0)
|
|
|
|
|
|
|
|
# si0 should now be gone
|
|
|
|
readers = ss.remote_get_buckets("si0")
|
|
|
|
self.failUnlessEqual(len(readers), 0)
|
|
|
|
# and the renew should no longer work
|
|
|
|
self.failUnlessRaises(IndexError, ss.remote_renew_lease, "si0", rs0)
|
|
|
|
|
|
|
|
|
|
|
|
# cancel the first lease on si1, leaving the second in place
|
|
|
|
ss.remote_cancel_lease("si1", cs1)
|
|
|
|
readers = ss.remote_get_buckets("si1")
|
|
|
|
self.failUnlessEqual(len(readers), 5)
|
|
|
|
# the corresponding renew should no longer work
|
|
|
|
self.failUnlessRaises(IndexError, ss.remote_renew_lease, "si1", rs1)
|
|
|
|
|
|
|
|
ss.remote_renew_lease("si1", rs2)
|
|
|
|
# cancelling the second should make it go away
|
|
|
|
ss.remote_cancel_lease("si1", cs2)
|
|
|
|
readers = ss.remote_get_buckets("si1")
|
|
|
|
self.failUnlessEqual(len(readers), 0)
|
|
|
|
self.failUnlessRaises(IndexError, ss.remote_renew_lease, "si1", rs1)
|
|
|
|
self.failUnlessRaises(IndexError, ss.remote_renew_lease, "si1", rs2)
|
|
|
|
|