tahoe-lafs/src/allmydata/test/test_storage.py

2967 lines
122 KiB
Python
Raw Normal View History

2020-08-31 17:20:57 +00:00
from __future__ import absolute_import
from __future__ import division
from __future__ import print_function
from __future__ import unicode_literals
from future.utils import native_str, PY2, PY3, bytes_to_native_str
if PY2:
from future.builtins import filter, map, zip, ascii, chr, hex, input, next, oct, open, pow, round, super, bytes, dict, list, object, range, str, max, min # noqa: F401
2020-08-27 19:49:04 +00:00
2020-05-05 19:56:59 +00:00
import time
import os.path
import platform
import stat
import struct
import shutil
2020-05-05 19:57:08 +00:00
import gc
from twisted.trial import unittest
from twisted.internet import defer
import itertools
2007-07-13 22:09:01 +00:00
from allmydata import interfaces
2020-08-18 17:03:36 +00:00
from allmydata.util import fileutil, hashutil, base32
from allmydata.storage.server import StorageServer
from allmydata.storage.mutable import MutableShareFile
from allmydata.storage.immutable import BucketWriter, BucketReader
from allmydata.storage.common import DataTooLargeError, storage_index_to_dir, \
UnknownMutableContainerVersionError, UnknownImmutableContainerVersionError
from allmydata.storage.lease import LeaseInfo
from allmydata.immutable.layout import WriteBucketProxy, WriteBucketProxy_v2, \
ReadBucketProxy
from allmydata.mutable.layout import MDMFSlotWriteProxy, MDMFSlotReadProxy, \
LayoutInvalid, MDMFSIGNABLEHEADER, \
SIGNED_PREFIX, MDMFHEADER, \
MDMFOFFSETS, SDMFSlotWriteProxy, \
PRIVATE_KEY_SIZE, \
SIGNATURE_SIZE, \
VERIFICATION_KEY_SIZE, \
SHARE_HASH_CHAIN_SIZE
from allmydata.interfaces import BadWriteEnablerError
from allmydata.test.no_network import NoNetworkServer
from allmydata.storage_client import (
_StorageServer,
)
from .common_py3 import FakeCanary, LoggingServiceParent, ShouldFailMixin
2008-06-18 00:01:42 +00:00
2019-05-15 06:17:44 +00:00
class FakeStatsProvider(object):
2008-06-18 00:01:42 +00:00
def count(self, name, delta=1):
pass
def register_producer(self, producer):
pass
class Bucket(unittest.TestCase):
def make_workdir(self, name):
basedir = os.path.join("storage", "Bucket", 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"))
return incoming, final
def bucket_writer_closed(self, bw, consumed):
pass
def add_latency(self, category, latency):
pass
def count(self, name, delta=1):
pass
def make_lease(self):
owner_num = 0
renew_secret = os.urandom(32)
cancel_secret = os.urandom(32)
expiration_time = time.time() + 5000
return LeaseInfo(owner_num, renew_secret, cancel_secret,
2020-08-27 19:36:54 +00:00
expiration_time, b"\x00" * 20)
def test_create(self):
incoming, final = self.make_workdir("test_create")
bw = BucketWriter(self, incoming, final, 200, self.make_lease(),
FakeCanary())
2020-08-27 19:36:54 +00:00
bw.remote_write(0, b"a"*25)
bw.remote_write(25, b"b"*25)
bw.remote_write(50, b"c"*25)
bw.remote_write(75, b"d"*7)
bw.remote_close()
def test_readwrite(self):
incoming, final = self.make_workdir("test_readwrite")
bw = BucketWriter(self, incoming, final, 200, self.make_lease(),
FakeCanary())
2020-08-27 19:36:54 +00:00
bw.remote_write(0, b"a"*25)
bw.remote_write(25, b"b"*25)
bw.remote_write(50, b"c"*7) # last block may be short
bw.remote_close()
# now read from it
br = BucketReader(self, bw.finalhome)
2020-08-27 19:36:54 +00:00
self.failUnlessEqual(br.remote_read(0, 25), b"a"*25)
self.failUnlessEqual(br.remote_read(25, 25), b"b"*25)
self.failUnlessEqual(br.remote_read(50, 7), b"c"*7)
def test_read_past_end_of_share_data(self):
# test vector for immutable files (hard-coded contents of an immutable share
# file):
# The following immutable share file content is identical to that
# generated with storage.immutable.ShareFile from Tahoe-LAFS v1.8.2
# with share data == 'a'. The total size of this content is 85
# bytes.
containerdata = struct.pack('>LLL', 1, 1, 1)
# A Tahoe-LAFS storage client would send as the share_data a
# complicated string involving hash trees and a URI Extension Block
# -- see allmydata/immutable/layout.py . This test, which is
# simulating a client, just sends 'a'.
2020-08-27 19:36:54 +00:00
share_data = b'a'
ownernumber = struct.pack('>L', 0)
2020-08-27 19:36:54 +00:00
renewsecret = b'THIS LETS ME RENEW YOUR FILE....'
assert len(renewsecret) == 32
2020-08-27 19:36:54 +00:00
cancelsecret = b'THIS LETS ME KILL YOUR FILE HAHA'
assert len(cancelsecret) == 32
expirationtime = struct.pack('>L', 60*60*24*31) # 31 days in seconds
lease_data = ownernumber + renewsecret + cancelsecret + expirationtime
share_file_data = containerdata + share_data + lease_data
incoming, final = self.make_workdir("test_read_past_end_of_share_data")
fileutil.write(final, share_file_data)
class MockStorageServer(object):
def add_latency(self, category, latency):
pass
def count(self, name, delta=1):
pass
mockstorageserver = MockStorageServer()
# Now read from it.
br = BucketReader(mockstorageserver, final)
self.failUnlessEqual(br.remote_read(0, len(share_data)), share_data)
# Read past the end of share data to get the cancel secret.
read_length = len(share_data) + len(ownernumber) + len(renewsecret) + len(cancelsecret)
result_of_read = br.remote_read(0, read_length)
self.failUnlessEqual(result_of_read, share_data)
result_of_read = br.remote_read(0, len(share_data)+1)
self.failUnlessEqual(result_of_read, share_data)
2019-05-15 06:17:44 +00:00
class RemoteBucket(object):
def __init__(self, target):
self.target = target
self.read_count = 0
self.write_count = 0
def callRemote(self, methname, *args, **kwargs):
def _call():
meth = getattr(self.target, "remote_" + methname)
return meth(*args, **kwargs)
if methname == "slot_readv":
self.read_count += 1
if "writev" in methname:
self.write_count += 1
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"))
bw = BucketWriter(self, incoming, final, size, self.make_lease(),
FakeCanary())
rb = RemoteBucket(bw)
return bw, rb, final
def make_lease(self):
owner_num = 0
renew_secret = os.urandom(32)
cancel_secret = os.urandom(32)
expiration_time = time.time() + 5000
return LeaseInfo(owner_num, renew_secret, cancel_secret,
2020-08-27 19:36:54 +00:00
expiration_time, b"\x00" * 20)
def bucket_writer_closed(self, bw, consumed):
pass
def add_latency(self, category, latency):
pass
def count(self, name, delta=1):
pass
def test_create(self):
bw, rb, sharefname = self.make_bucket("test_create", 500)
bp = WriteBucketProxy(rb, None,
2007-07-13 22:09:01 +00:00
data_size=300,
immutable: refactor downloader to be more reusable for checker/verifier/repairer (and better) The code for validating the share hash tree and the block hash tree has been rewritten to make sure it handles all cases, to share metadata about the file (such as the share hash tree, block hash trees, and UEB) among different share downloads, and not to require hashes to be stored on the server unnecessarily, such as the roots of the block hash trees (not needed since they are also the leaves of the share hash tree), and the root of the share hash tree (not needed since it is also included in the UEB). It also passes the latest tests including handling corrupted shares well. ValidatedReadBucketProxy takes a share_hash_tree argument to its constructor, which is a reference to a share hash tree shared by all ValidatedReadBucketProxies for that immutable file download. ValidatedReadBucketProxy requires the block_size and share_size to be provided in its constructor, and it then uses those to compute the offsets and lengths of blocks when it needs them, instead of reading those values out of the share. The user of ValidatedReadBucketProxy therefore has to have first used a ValidatedExtendedURIProxy to compute those two values from the validated contents of the URI. This is pleasingly simplifies safety analysis: the client knows which span of bytes corresponds to a given block from the validated URI data, rather than from the unvalidated data stored on the storage server. It also simplifies unit testing of verifier/repairer, because now it doesn't care about the contents of the "share size" and "block size" fields in the share. It does not relieve the need for share data v2 layout, because we still need to store and retrieve the offsets of the fields which come after the share data, therefore we still need to use share data v2 with its 8-byte fields if we want to store share data larger than about 2^32. Specify which subset of the block hashes and share hashes you need while downloading a particular share. In the future this will hopefully be used to fetch only a subset, for network efficiency, but currently all of them are fetched, regardless of which subset you specify. ReadBucketProxy hides the question of whether it has "started" or not (sent a request to the server to get metadata) from its user. Download is optimized to do as few roundtrips and as few requests as possible, hopefully speeding up download a bit.
2009-01-05 16:51:45 +00:00
block_size=10,
2007-07-13 22:09:01 +00:00
num_segments=5,
num_share_hashes=3,
uri_extension_size_max=500)
self.failUnless(interfaces.IStorageBucketWriter.providedBy(bp), bp)
def _do_test_readwrite(self, name, header_size, wbp_class, rbp_class):
# 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 two
# per-segment merkle trees (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:
#
# 0x24 + 100 + 7*32 + 7*32 + 7*32 + 3*(2+32) + 4+500 = 1414 bytes long
# 0x44 + 100 + 7*32 + 7*32 + 7*32 + 3*(2+32) + 4+500 = 1446 bytes long
sharesize = header_size + 100 + 7*32 + 7*32 + 7*32 + 3*(2+32) + 4+500
2020-08-27 19:36:54 +00:00
crypttext_hashes = [hashutil.tagged_hash(b"crypt", b"bar%d" % i)
for i in range(7)]
2020-08-27 19:36:54 +00:00
block_hashes = [hashutil.tagged_hash(b"block", b"bar%d" % i)
for i in range(7)]
2020-08-27 19:36:54 +00:00
share_hashes = [(i, hashutil.tagged_hash(b"share", b"bar%d" % i))
for i in (1,9,13)]
2020-08-27 19:36:54 +00:00
uri_extension = b"s" + b"E"*498 + b"e"
bw, rb, sharefname = self.make_bucket(name, sharesize)
bp = wbp_class(rb, None,
data_size=95,
immutable: refactor downloader to be more reusable for checker/verifier/repairer (and better) The code for validating the share hash tree and the block hash tree has been rewritten to make sure it handles all cases, to share metadata about the file (such as the share hash tree, block hash trees, and UEB) among different share downloads, and not to require hashes to be stored on the server unnecessarily, such as the roots of the block hash trees (not needed since they are also the leaves of the share hash tree), and the root of the share hash tree (not needed since it is also included in the UEB). It also passes the latest tests including handling corrupted shares well. ValidatedReadBucketProxy takes a share_hash_tree argument to its constructor, which is a reference to a share hash tree shared by all ValidatedReadBucketProxies for that immutable file download. ValidatedReadBucketProxy requires the block_size and share_size to be provided in its constructor, and it then uses those to compute the offsets and lengths of blocks when it needs them, instead of reading those values out of the share. The user of ValidatedReadBucketProxy therefore has to have first used a ValidatedExtendedURIProxy to compute those two values from the validated contents of the URI. This is pleasingly simplifies safety analysis: the client knows which span of bytes corresponds to a given block from the validated URI data, rather than from the unvalidated data stored on the storage server. It also simplifies unit testing of verifier/repairer, because now it doesn't care about the contents of the "share size" and "block size" fields in the share. It does not relieve the need for share data v2 layout, because we still need to store and retrieve the offsets of the fields which come after the share data, therefore we still need to use share data v2 with its 8-byte fields if we want to store share data larger than about 2^32. Specify which subset of the block hashes and share hashes you need while downloading a particular share. In the future this will hopefully be used to fetch only a subset, for network efficiency, but currently all of them are fetched, regardless of which subset you specify. ReadBucketProxy hides the question of whether it has "started" or not (sent a request to the server to get metadata) from its user. Download is optimized to do as few roundtrips and as few requests as possible, hopefully speeding up download a bit.
2009-01-05 16:51:45 +00:00
block_size=25,
num_segments=4,
num_share_hashes=3,
uri_extension_size_max=len(uri_extension))
immutable: refactor downloader to be more reusable for checker/verifier/repairer (and better) The code for validating the share hash tree and the block hash tree has been rewritten to make sure it handles all cases, to share metadata about the file (such as the share hash tree, block hash trees, and UEB) among different share downloads, and not to require hashes to be stored on the server unnecessarily, such as the roots of the block hash trees (not needed since they are also the leaves of the share hash tree), and the root of the share hash tree (not needed since it is also included in the UEB). It also passes the latest tests including handling corrupted shares well. ValidatedReadBucketProxy takes a share_hash_tree argument to its constructor, which is a reference to a share hash tree shared by all ValidatedReadBucketProxies for that immutable file download. ValidatedReadBucketProxy requires the block_size and share_size to be provided in its constructor, and it then uses those to compute the offsets and lengths of blocks when it needs them, instead of reading those values out of the share. The user of ValidatedReadBucketProxy therefore has to have first used a ValidatedExtendedURIProxy to compute those two values from the validated contents of the URI. This is pleasingly simplifies safety analysis: the client knows which span of bytes corresponds to a given block from the validated URI data, rather than from the unvalidated data stored on the storage server. It also simplifies unit testing of verifier/repairer, because now it doesn't care about the contents of the "share size" and "block size" fields in the share. It does not relieve the need for share data v2 layout, because we still need to store and retrieve the offsets of the fields which come after the share data, therefore we still need to use share data v2 with its 8-byte fields if we want to store share data larger than about 2^32. Specify which subset of the block hashes and share hashes you need while downloading a particular share. In the future this will hopefully be used to fetch only a subset, for network efficiency, but currently all of them are fetched, regardless of which subset you specify. ReadBucketProxy hides the question of whether it has "started" or not (sent a request to the server to get metadata) from its user. Download is optimized to do as few roundtrips and as few requests as possible, hopefully speeding up download a bit.
2009-01-05 16:51:45 +00:00
d = bp.put_header()
2020-08-27 19:36:54 +00:00
d.addCallback(lambda res: bp.put_block(0, b"a"*25))
d.addCallback(lambda res: bp.put_block(1, b"b"*25))
d.addCallback(lambda res: bp.put_block(2, b"c"*25))
d.addCallback(lambda res: bp.put_block(3, b"d"*20))
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):
br = BucketReader(self, sharefname)
rb = RemoteBucket(br)
2020-08-27 19:36:54 +00:00
server = NoNetworkServer(b"abc", None)
rbp = rbp_class(rb, server, storage_index=b"")
self.failUnlessIn("to peer", repr(rbp))
self.failUnless(interfaces.IStorageBucketReader.providedBy(rbp), rbp)
immutable: refactor downloader to be more reusable for checker/verifier/repairer (and better) The code for validating the share hash tree and the block hash tree has been rewritten to make sure it handles all cases, to share metadata about the file (such as the share hash tree, block hash trees, and UEB) among different share downloads, and not to require hashes to be stored on the server unnecessarily, such as the roots of the block hash trees (not needed since they are also the leaves of the share hash tree), and the root of the share hash tree (not needed since it is also included in the UEB). It also passes the latest tests including handling corrupted shares well. ValidatedReadBucketProxy takes a share_hash_tree argument to its constructor, which is a reference to a share hash tree shared by all ValidatedReadBucketProxies for that immutable file download. ValidatedReadBucketProxy requires the block_size and share_size to be provided in its constructor, and it then uses those to compute the offsets and lengths of blocks when it needs them, instead of reading those values out of the share. The user of ValidatedReadBucketProxy therefore has to have first used a ValidatedExtendedURIProxy to compute those two values from the validated contents of the URI. This is pleasingly simplifies safety analysis: the client knows which span of bytes corresponds to a given block from the validated URI data, rather than from the unvalidated data stored on the storage server. It also simplifies unit testing of verifier/repairer, because now it doesn't care about the contents of the "share size" and "block size" fields in the share. It does not relieve the need for share data v2 layout, because we still need to store and retrieve the offsets of the fields which come after the share data, therefore we still need to use share data v2 with its 8-byte fields if we want to store share data larger than about 2^32. Specify which subset of the block hashes and share hashes you need while downloading a particular share. In the future this will hopefully be used to fetch only a subset, for network efficiency, but currently all of them are fetched, regardless of which subset you specify. ReadBucketProxy hides the question of whether it has "started" or not (sent a request to the server to get metadata) from its user. Download is optimized to do as few roundtrips and as few requests as possible, hopefully speeding up download a bit.
2009-01-05 16:51:45 +00:00
d1 = rbp.get_block_data(0, 25, 25)
2020-08-27 19:36:54 +00:00
d1.addCallback(lambda res: self.failUnlessEqual(res, b"a"*25))
immutable: refactor downloader to be more reusable for checker/verifier/repairer (and better) The code for validating the share hash tree and the block hash tree has been rewritten to make sure it handles all cases, to share metadata about the file (such as the share hash tree, block hash trees, and UEB) among different share downloads, and not to require hashes to be stored on the server unnecessarily, such as the roots of the block hash trees (not needed since they are also the leaves of the share hash tree), and the root of the share hash tree (not needed since it is also included in the UEB). It also passes the latest tests including handling corrupted shares well. ValidatedReadBucketProxy takes a share_hash_tree argument to its constructor, which is a reference to a share hash tree shared by all ValidatedReadBucketProxies for that immutable file download. ValidatedReadBucketProxy requires the block_size and share_size to be provided in its constructor, and it then uses those to compute the offsets and lengths of blocks when it needs them, instead of reading those values out of the share. The user of ValidatedReadBucketProxy therefore has to have first used a ValidatedExtendedURIProxy to compute those two values from the validated contents of the URI. This is pleasingly simplifies safety analysis: the client knows which span of bytes corresponds to a given block from the validated URI data, rather than from the unvalidated data stored on the storage server. It also simplifies unit testing of verifier/repairer, because now it doesn't care about the contents of the "share size" and "block size" fields in the share. It does not relieve the need for share data v2 layout, because we still need to store and retrieve the offsets of the fields which come after the share data, therefore we still need to use share data v2 with its 8-byte fields if we want to store share data larger than about 2^32. Specify which subset of the block hashes and share hashes you need while downloading a particular share. In the future this will hopefully be used to fetch only a subset, for network efficiency, but currently all of them are fetched, regardless of which subset you specify. ReadBucketProxy hides the question of whether it has "started" or not (sent a request to the server to get metadata) from its user. Download is optimized to do as few roundtrips and as few requests as possible, hopefully speeding up download a bit.
2009-01-05 16:51:45 +00:00
d1.addCallback(lambda res: rbp.get_block_data(1, 25, 25))
2020-08-27 19:36:54 +00:00
d1.addCallback(lambda res: self.failUnlessEqual(res, b"b"*25))
immutable: refactor downloader to be more reusable for checker/verifier/repairer (and better) The code for validating the share hash tree and the block hash tree has been rewritten to make sure it handles all cases, to share metadata about the file (such as the share hash tree, block hash trees, and UEB) among different share downloads, and not to require hashes to be stored on the server unnecessarily, such as the roots of the block hash trees (not needed since they are also the leaves of the share hash tree), and the root of the share hash tree (not needed since it is also included in the UEB). It also passes the latest tests including handling corrupted shares well. ValidatedReadBucketProxy takes a share_hash_tree argument to its constructor, which is a reference to a share hash tree shared by all ValidatedReadBucketProxies for that immutable file download. ValidatedReadBucketProxy requires the block_size and share_size to be provided in its constructor, and it then uses those to compute the offsets and lengths of blocks when it needs them, instead of reading those values out of the share. The user of ValidatedReadBucketProxy therefore has to have first used a ValidatedExtendedURIProxy to compute those two values from the validated contents of the URI. This is pleasingly simplifies safety analysis: the client knows which span of bytes corresponds to a given block from the validated URI data, rather than from the unvalidated data stored on the storage server. It also simplifies unit testing of verifier/repairer, because now it doesn't care about the contents of the "share size" and "block size" fields in the share. It does not relieve the need for share data v2 layout, because we still need to store and retrieve the offsets of the fields which come after the share data, therefore we still need to use share data v2 with its 8-byte fields if we want to store share data larger than about 2^32. Specify which subset of the block hashes and share hashes you need while downloading a particular share. In the future this will hopefully be used to fetch only a subset, for network efficiency, but currently all of them are fetched, regardless of which subset you specify. ReadBucketProxy hides the question of whether it has "started" or not (sent a request to the server to get metadata) from its user. Download is optimized to do as few roundtrips and as few requests as possible, hopefully speeding up download a bit.
2009-01-05 16:51:45 +00:00
d1.addCallback(lambda res: rbp.get_block_data(2, 25, 25))
2020-08-27 19:36:54 +00:00
d1.addCallback(lambda res: self.failUnlessEqual(res, b"c"*25))
immutable: refactor downloader to be more reusable for checker/verifier/repairer (and better) The code for validating the share hash tree and the block hash tree has been rewritten to make sure it handles all cases, to share metadata about the file (such as the share hash tree, block hash trees, and UEB) among different share downloads, and not to require hashes to be stored on the server unnecessarily, such as the roots of the block hash trees (not needed since they are also the leaves of the share hash tree), and the root of the share hash tree (not needed since it is also included in the UEB). It also passes the latest tests including handling corrupted shares well. ValidatedReadBucketProxy takes a share_hash_tree argument to its constructor, which is a reference to a share hash tree shared by all ValidatedReadBucketProxies for that immutable file download. ValidatedReadBucketProxy requires the block_size and share_size to be provided in its constructor, and it then uses those to compute the offsets and lengths of blocks when it needs them, instead of reading those values out of the share. The user of ValidatedReadBucketProxy therefore has to have first used a ValidatedExtendedURIProxy to compute those two values from the validated contents of the URI. This is pleasingly simplifies safety analysis: the client knows which span of bytes corresponds to a given block from the validated URI data, rather than from the unvalidated data stored on the storage server. It also simplifies unit testing of verifier/repairer, because now it doesn't care about the contents of the "share size" and "block size" fields in the share. It does not relieve the need for share data v2 layout, because we still need to store and retrieve the offsets of the fields which come after the share data, therefore we still need to use share data v2 with its 8-byte fields if we want to store share data larger than about 2^32. Specify which subset of the block hashes and share hashes you need while downloading a particular share. In the future this will hopefully be used to fetch only a subset, for network efficiency, but currently all of them are fetched, regardless of which subset you specify. ReadBucketProxy hides the question of whether it has "started" or not (sent a request to the server to get metadata) from its user. Download is optimized to do as few roundtrips and as few requests as possible, hopefully speeding up download a bit.
2009-01-05 16:51:45 +00:00
d1.addCallback(lambda res: rbp.get_block_data(3, 25, 20))
2020-08-27 19:36:54 +00:00
d1.addCallback(lambda res: self.failUnlessEqual(res, b"d"*20))
d1.addCallback(lambda res: rbp.get_crypttext_hashes())
d1.addCallback(lambda res:
self.failUnlessEqual(res, crypttext_hashes))
immutable: refactor downloader to be more reusable for checker/verifier/repairer (and better) The code for validating the share hash tree and the block hash tree has been rewritten to make sure it handles all cases, to share metadata about the file (such as the share hash tree, block hash trees, and UEB) among different share downloads, and not to require hashes to be stored on the server unnecessarily, such as the roots of the block hash trees (not needed since they are also the leaves of the share hash tree), and the root of the share hash tree (not needed since it is also included in the UEB). It also passes the latest tests including handling corrupted shares well. ValidatedReadBucketProxy takes a share_hash_tree argument to its constructor, which is a reference to a share hash tree shared by all ValidatedReadBucketProxies for that immutable file download. ValidatedReadBucketProxy requires the block_size and share_size to be provided in its constructor, and it then uses those to compute the offsets and lengths of blocks when it needs them, instead of reading those values out of the share. The user of ValidatedReadBucketProxy therefore has to have first used a ValidatedExtendedURIProxy to compute those two values from the validated contents of the URI. This is pleasingly simplifies safety analysis: the client knows which span of bytes corresponds to a given block from the validated URI data, rather than from the unvalidated data stored on the storage server. It also simplifies unit testing of verifier/repairer, because now it doesn't care about the contents of the "share size" and "block size" fields in the share. It does not relieve the need for share data v2 layout, because we still need to store and retrieve the offsets of the fields which come after the share data, therefore we still need to use share data v2 with its 8-byte fields if we want to store share data larger than about 2^32. Specify which subset of the block hashes and share hashes you need while downloading a particular share. In the future this will hopefully be used to fetch only a subset, for network efficiency, but currently all of them are fetched, regardless of which subset you specify. ReadBucketProxy hides the question of whether it has "started" or not (sent a request to the server to get metadata) from its user. Download is optimized to do as few roundtrips and as few requests as possible, hopefully speeding up download a bit.
2009-01-05 16:51:45 +00:00
d1.addCallback(lambda res: rbp.get_block_hashes(set(range(4))))
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
def test_readwrite_v1(self):
return self._do_test_readwrite("test_readwrite_v1",
0x24, WriteBucketProxy, ReadBucketProxy)
def test_readwrite_v2(self):
return self._do_test_readwrite("test_readwrite_v2",
0x44, WriteBucketProxy_v2, ReadBucketProxy)
class Server(unittest.TestCase):
def setUp(self):
2008-01-14 18:58:58 +00:00
self.sparent = LoggingServiceParent()
self.sparent.startService()
self._lease_secret = itertools.count()
def tearDown(self):
return self.sparent.stopService()
def workdir(self, name):
basedir = os.path.join("storage", "Server", name)
return basedir
def create(self, name, reserved_space=0, klass=StorageServer):
workdir = self.workdir(name)
2020-08-27 19:36:54 +00:00
ss = klass(workdir, b"\x00" * 20, reserved_space=reserved_space,
stats_provider=FakeStatsProvider())
ss.setServiceParent(self.sparent)
return ss
def test_create(self):
self.create("test_create")
def test_declares_fixed_1528(self):
ss = self.create("test_declares_fixed_1528")
ver = ss.remote_get_version()
sv1 = ver['http://allmydata.org/tahoe/protocols/storage/v1']
self.failUnless(sv1.get('prevents-read-past-end-of-share-data'), sv1)
def test_declares_maximum_share_sizes(self):
ss = self.create("test_declares_maximum_share_sizes")
ver = ss.remote_get_version()
sv1 = ver['http://allmydata.org/tahoe/protocols/storage/v1']
self.failUnlessIn('maximum-immutable-share-size', sv1)
self.failUnlessIn('maximum-mutable-share-size', sv1)
def test_declares_available_space(self):
ss = self.create("test_declares_available_space")
ver = ss.remote_get_version()
sv1 = ver['http://allmydata.org/tahoe/protocols/storage/v1']
self.failUnlessIn('available-space', sv1)
2008-06-18 00:01:42 +00:00
def allocate(self, ss, storage_index, sharenums, size, canary=None):
2020-08-27 19:36:54 +00:00
renew_secret = hashutil.tagged_hash(b"blah", b"%d" % next(self._lease_secret))
cancel_secret = hashutil.tagged_hash(b"blah", b"%d" % next(self._lease_secret))
2008-06-18 00:01:42 +00:00
if not canary:
canary = FakeCanary()
return ss.remote_allocate_buckets(storage_index,
renew_secret, cancel_secret,
2008-06-18 00:01:42 +00:00
sharenums, size, canary)
immutable: storage servers accept any size shares now Nathan Wilcox observed that the storage server can rely on the size of the share file combined with the count of leases to unambiguously identify the location of the leases. This means that it can hold any size share data, even though the field nominally used to hold the size of the share data is only 32 bits wide. With this patch, the storage server still writes the "size of the share data" field (just in case the server gets downgraded to an earlier version which requires that field, or the share file gets moved to another server which is of an earlier vintage), but it doesn't use it. Also, with this patch, the server no longer rejects requests to write shares which are >= 2^32 bytes in size, and it no longer rejects attempts to read such shares. This fixes http://allmydata.org/trac/tahoe/ticket/346 (increase share-size field to 8 bytes, remove 12GiB filesize limit), although there remains open a question of how clients know that a given server can handle large shares (by using the new versioning scheme, probably). Note that share size is also limited by another factor -- how big of a file we can store on the local filesystem on the server. Currently allmydata.com typically uses ext3 and I think we typically have block size = 4 KiB, which means that the largest file is about 2 TiB. Also, the hard drives themselves are only 1 TB, so the largest share is definitely slightly less than 1 TB, which means (when K == 3), the largest file is less than 3 TB. This patch also refactors the creation of new sharefiles so that only a single fopen() is used. This patch also helps with the unit-testing of repairer, since formerly it was unclear what repairer should expect to find if the "share data size" field was corrupted (some corruptions would have no effect, others would cause failure to download). Now it is clear that repairer is not required to notice if this field is corrupted since it has no effect on download. :-)
2008-12-31 22:42:26 +00:00
def test_large_share(self):
syslow = platform.system().lower()
if 'cygwin' in syslow or 'windows' in syslow or 'darwin' in syslow:
raise unittest.SkipTest("If your filesystem doesn't support efficient sparse files then it is very expensive (Mac OS X and Windows don't support efficient sparse files).")
avail = fileutil.get_available_space('.', 512*2**20)
if avail <= 4*2**30:
raise unittest.SkipTest("This test will spuriously fail if you have less than 4 GiB free on your filesystem.")
immutable: storage servers accept any size shares now Nathan Wilcox observed that the storage server can rely on the size of the share file combined with the count of leases to unambiguously identify the location of the leases. This means that it can hold any size share data, even though the field nominally used to hold the size of the share data is only 32 bits wide. With this patch, the storage server still writes the "size of the share data" field (just in case the server gets downgraded to an earlier version which requires that field, or the share file gets moved to another server which is of an earlier vintage), but it doesn't use it. Also, with this patch, the server no longer rejects requests to write shares which are >= 2^32 bytes in size, and it no longer rejects attempts to read such shares. This fixes http://allmydata.org/trac/tahoe/ticket/346 (increase share-size field to 8 bytes, remove 12GiB filesize limit), although there remains open a question of how clients know that a given server can handle large shares (by using the new versioning scheme, probably). Note that share size is also limited by another factor -- how big of a file we can store on the local filesystem on the server. Currently allmydata.com typically uses ext3 and I think we typically have block size = 4 KiB, which means that the largest file is about 2 TiB. Also, the hard drives themselves are only 1 TB, so the largest share is definitely slightly less than 1 TB, which means (when K == 3), the largest file is less than 3 TB. This patch also refactors the creation of new sharefiles so that only a single fopen() is used. This patch also helps with the unit-testing of repairer, since formerly it was unclear what repairer should expect to find if the "share data size" field was corrupted (some corruptions would have no effect, others would cause failure to download). Now it is clear that repairer is not required to notice if this field is corrupted since it has no effect on download. :-)
2008-12-31 22:42:26 +00:00
ss = self.create("test_large_share")
2020-08-27 19:49:04 +00:00
already,writers = self.allocate(ss, b"allocate", [0], 2**32+2)
immutable: storage servers accept any size shares now Nathan Wilcox observed that the storage server can rely on the size of the share file combined with the count of leases to unambiguously identify the location of the leases. This means that it can hold any size share data, even though the field nominally used to hold the size of the share data is only 32 bits wide. With this patch, the storage server still writes the "size of the share data" field (just in case the server gets downgraded to an earlier version which requires that field, or the share file gets moved to another server which is of an earlier vintage), but it doesn't use it. Also, with this patch, the server no longer rejects requests to write shares which are >= 2^32 bytes in size, and it no longer rejects attempts to read such shares. This fixes http://allmydata.org/trac/tahoe/ticket/346 (increase share-size field to 8 bytes, remove 12GiB filesize limit), although there remains open a question of how clients know that a given server can handle large shares (by using the new versioning scheme, probably). Note that share size is also limited by another factor -- how big of a file we can store on the local filesystem on the server. Currently allmydata.com typically uses ext3 and I think we typically have block size = 4 KiB, which means that the largest file is about 2 TiB. Also, the hard drives themselves are only 1 TB, so the largest share is definitely slightly less than 1 TB, which means (when K == 3), the largest file is less than 3 TB. This patch also refactors the creation of new sharefiles so that only a single fopen() is used. This patch also helps with the unit-testing of repairer, since formerly it was unclear what repairer should expect to find if the "share data size" field was corrupted (some corruptions would have no effect, others would cause failure to download). Now it is clear that repairer is not required to notice if this field is corrupted since it has no effect on download. :-)
2008-12-31 22:42:26 +00:00
self.failUnlessEqual(already, set())
self.failUnlessEqual(set(writers.keys()), set([0]))
immutable: storage servers accept any size shares now Nathan Wilcox observed that the storage server can rely on the size of the share file combined with the count of leases to unambiguously identify the location of the leases. This means that it can hold any size share data, even though the field nominally used to hold the size of the share data is only 32 bits wide. With this patch, the storage server still writes the "size of the share data" field (just in case the server gets downgraded to an earlier version which requires that field, or the share file gets moved to another server which is of an earlier vintage), but it doesn't use it. Also, with this patch, the server no longer rejects requests to write shares which are >= 2^32 bytes in size, and it no longer rejects attempts to read such shares. This fixes http://allmydata.org/trac/tahoe/ticket/346 (increase share-size field to 8 bytes, remove 12GiB filesize limit), although there remains open a question of how clients know that a given server can handle large shares (by using the new versioning scheme, probably). Note that share size is also limited by another factor -- how big of a file we can store on the local filesystem on the server. Currently allmydata.com typically uses ext3 and I think we typically have block size = 4 KiB, which means that the largest file is about 2 TiB. Also, the hard drives themselves are only 1 TB, so the largest share is definitely slightly less than 1 TB, which means (when K == 3), the largest file is less than 3 TB. This patch also refactors the creation of new sharefiles so that only a single fopen() is used. This patch also helps with the unit-testing of repairer, since formerly it was unclear what repairer should expect to find if the "share data size" field was corrupted (some corruptions would have no effect, others would cause failure to download). Now it is clear that repairer is not required to notice if this field is corrupted since it has no effect on download. :-)
2008-12-31 22:42:26 +00:00
2020-08-27 19:49:04 +00:00
shnum, bucket = list(writers.items())[0]
immutable: storage servers accept any size shares now Nathan Wilcox observed that the storage server can rely on the size of the share file combined with the count of leases to unambiguously identify the location of the leases. This means that it can hold any size share data, even though the field nominally used to hold the size of the share data is only 32 bits wide. With this patch, the storage server still writes the "size of the share data" field (just in case the server gets downgraded to an earlier version which requires that field, or the share file gets moved to another server which is of an earlier vintage), but it doesn't use it. Also, with this patch, the server no longer rejects requests to write shares which are >= 2^32 bytes in size, and it no longer rejects attempts to read such shares. This fixes http://allmydata.org/trac/tahoe/ticket/346 (increase share-size field to 8 bytes, remove 12GiB filesize limit), although there remains open a question of how clients know that a given server can handle large shares (by using the new versioning scheme, probably). Note that share size is also limited by another factor -- how big of a file we can store on the local filesystem on the server. Currently allmydata.com typically uses ext3 and I think we typically have block size = 4 KiB, which means that the largest file is about 2 TiB. Also, the hard drives themselves are only 1 TB, so the largest share is definitely slightly less than 1 TB, which means (when K == 3), the largest file is less than 3 TB. This patch also refactors the creation of new sharefiles so that only a single fopen() is used. This patch also helps with the unit-testing of repairer, since formerly it was unclear what repairer should expect to find if the "share data size" field was corrupted (some corruptions would have no effect, others would cause failure to download). Now it is clear that repairer is not required to notice if this field is corrupted since it has no effect on download. :-)
2008-12-31 22:42:26 +00:00
# This test is going to hammer your filesystem if it doesn't make a sparse file for this. :-(
2020-08-27 19:49:04 +00:00
bucket.remote_write(2**32, b"ab")
immutable: storage servers accept any size shares now Nathan Wilcox observed that the storage server can rely on the size of the share file combined with the count of leases to unambiguously identify the location of the leases. This means that it can hold any size share data, even though the field nominally used to hold the size of the share data is only 32 bits wide. With this patch, the storage server still writes the "size of the share data" field (just in case the server gets downgraded to an earlier version which requires that field, or the share file gets moved to another server which is of an earlier vintage), but it doesn't use it. Also, with this patch, the server no longer rejects requests to write shares which are >= 2^32 bytes in size, and it no longer rejects attempts to read such shares. This fixes http://allmydata.org/trac/tahoe/ticket/346 (increase share-size field to 8 bytes, remove 12GiB filesize limit), although there remains open a question of how clients know that a given server can handle large shares (by using the new versioning scheme, probably). Note that share size is also limited by another factor -- how big of a file we can store on the local filesystem on the server. Currently allmydata.com typically uses ext3 and I think we typically have block size = 4 KiB, which means that the largest file is about 2 TiB. Also, the hard drives themselves are only 1 TB, so the largest share is definitely slightly less than 1 TB, which means (when K == 3), the largest file is less than 3 TB. This patch also refactors the creation of new sharefiles so that only a single fopen() is used. This patch also helps with the unit-testing of repairer, since formerly it was unclear what repairer should expect to find if the "share data size" field was corrupted (some corruptions would have no effect, others would cause failure to download). Now it is clear that repairer is not required to notice if this field is corrupted since it has no effect on download. :-)
2008-12-31 22:42:26 +00:00
bucket.remote_close()
2020-08-27 19:49:04 +00:00
readers = ss.remote_get_buckets(b"allocate")
immutable: storage servers accept any size shares now Nathan Wilcox observed that the storage server can rely on the size of the share file combined with the count of leases to unambiguously identify the location of the leases. This means that it can hold any size share data, even though the field nominally used to hold the size of the share data is only 32 bits wide. With this patch, the storage server still writes the "size of the share data" field (just in case the server gets downgraded to an earlier version which requires that field, or the share file gets moved to another server which is of an earlier vintage), but it doesn't use it. Also, with this patch, the server no longer rejects requests to write shares which are >= 2^32 bytes in size, and it no longer rejects attempts to read such shares. This fixes http://allmydata.org/trac/tahoe/ticket/346 (increase share-size field to 8 bytes, remove 12GiB filesize limit), although there remains open a question of how clients know that a given server can handle large shares (by using the new versioning scheme, probably). Note that share size is also limited by another factor -- how big of a file we can store on the local filesystem on the server. Currently allmydata.com typically uses ext3 and I think we typically have block size = 4 KiB, which means that the largest file is about 2 TiB. Also, the hard drives themselves are only 1 TB, so the largest share is definitely slightly less than 1 TB, which means (when K == 3), the largest file is less than 3 TB. This patch also refactors the creation of new sharefiles so that only a single fopen() is used. This patch also helps with the unit-testing of repairer, since formerly it was unclear what repairer should expect to find if the "share data size" field was corrupted (some corruptions would have no effect, others would cause failure to download). Now it is clear that repairer is not required to notice if this field is corrupted since it has no effect on download. :-)
2008-12-31 22:42:26 +00:00
reader = readers[shnum]
2020-08-27 19:49:04 +00:00
self.failUnlessEqual(reader.remote_read(2**32, 2), b"ab")
immutable: storage servers accept any size shares now Nathan Wilcox observed that the storage server can rely on the size of the share file combined with the count of leases to unambiguously identify the location of the leases. This means that it can hold any size share data, even though the field nominally used to hold the size of the share data is only 32 bits wide. With this patch, the storage server still writes the "size of the share data" field (just in case the server gets downgraded to an earlier version which requires that field, or the share file gets moved to another server which is of an earlier vintage), but it doesn't use it. Also, with this patch, the server no longer rejects requests to write shares which are >= 2^32 bytes in size, and it no longer rejects attempts to read such shares. This fixes http://allmydata.org/trac/tahoe/ticket/346 (increase share-size field to 8 bytes, remove 12GiB filesize limit), although there remains open a question of how clients know that a given server can handle large shares (by using the new versioning scheme, probably). Note that share size is also limited by another factor -- how big of a file we can store on the local filesystem on the server. Currently allmydata.com typically uses ext3 and I think we typically have block size = 4 KiB, which means that the largest file is about 2 TiB. Also, the hard drives themselves are only 1 TB, so the largest share is definitely slightly less than 1 TB, which means (when K == 3), the largest file is less than 3 TB. This patch also refactors the creation of new sharefiles so that only a single fopen() is used. This patch also helps with the unit-testing of repairer, since formerly it was unclear what repairer should expect to find if the "share data size" field was corrupted (some corruptions would have no effect, others would cause failure to download). Now it is clear that repairer is not required to notice if this field is corrupted since it has no effect on download. :-)
2008-12-31 22:42:26 +00:00
def test_dont_overfill_dirs(self):
"""
This test asserts that if you add a second share whose storage index
share lots of leading bits with an extant share (but isn't the exact
same storage index), this won't add an entry to the share directory.
"""
ss = self.create("test_dont_overfill_dirs")
2020-08-27 19:49:04 +00:00
already, writers = self.allocate(ss, b"storageindex", [0], 10)
for i, wb in writers.items():
2020-08-27 19:49:04 +00:00
wb.remote_write(0, b"%10d" % i)
wb.remote_close()
storedir = os.path.join(self.workdir("test_dont_overfill_dirs"),
"shares")
children_of_storedir = set(os.listdir(storedir))
# Now store another one under another storageindex that has leading
# chars the same as the first storageindex.
2020-08-27 19:49:04 +00:00
already, writers = self.allocate(ss, b"storageindey", [0], 10)
for i, wb in writers.items():
2020-08-27 19:49:04 +00:00
wb.remote_write(0, b"%10d" % i)
wb.remote_close()
storedir = os.path.join(self.workdir("test_dont_overfill_dirs"),
"shares")
new_children_of_storedir = set(os.listdir(storedir))
self.failUnlessEqual(children_of_storedir, new_children_of_storedir)
def test_remove_incoming(self):
ss = self.create("test_remove_incoming")
2020-08-31 17:20:57 +00:00
already, writers = self.allocate(ss, b"vid", list(range(3)), 10)
for i,wb in writers.items():
2020-08-27 19:49:04 +00:00
wb.remote_write(0, b"%10d" % i)
wb.remote_close()
incoming_share_dir = wb.incominghome
incoming_bucket_dir = os.path.dirname(incoming_share_dir)
incoming_prefix_dir = os.path.dirname(incoming_bucket_dir)
incoming_dir = os.path.dirname(incoming_prefix_dir)
self.failIf(os.path.exists(incoming_bucket_dir), incoming_bucket_dir)
self.failIf(os.path.exists(incoming_prefix_dir), incoming_prefix_dir)
self.failUnless(os.path.exists(incoming_dir), incoming_dir)
def test_abort(self):
# remote_abort, when called on a writer, should make sure that
# the allocated size of the bucket is not counted by the storage
# server when accounting for space.
ss = self.create("test_abort")
2020-08-27 19:49:04 +00:00
already, writers = self.allocate(ss, b"allocate", [0, 1, 2], 150)
self.failIfEqual(ss.allocated_size(), 0)
# Now abort the writers.
2020-08-27 19:49:04 +00:00
for writer in writers.values():
writer.remote_abort()
self.failUnlessEqual(ss.allocated_size(), 0)
def test_allocate(self):
ss = self.create("test_allocate")
2020-08-27 19:49:04 +00:00
self.failUnlessEqual(ss.remote_get_buckets(b"allocate"), {})
2020-08-27 19:49:04 +00:00
already,writers = self.allocate(ss, b"allocate", [0,1,2], 75)
self.failUnlessEqual(already, set())
self.failUnlessEqual(set(writers.keys()), set([0,1,2]))
# while the buckets are open, they should not count as readable
2020-08-27 19:49:04 +00:00
self.failUnlessEqual(ss.remote_get_buckets(b"allocate"), {})
# close the buckets
for i,wb in writers.items():
2020-08-27 19:49:04 +00:00
wb.remote_write(0, b"%25d" % i)
wb.remote_close()
2008-06-18 00:01:42 +00:00
# aborting a bucket that was already closed is a no-op
wb.remote_abort()
# now they should be readable
2020-08-27 19:49:04 +00:00
b = ss.remote_get_buckets(b"allocate")
self.failUnlessEqual(set(b.keys()), set([0,1,2]))
2020-08-27 19:49:04 +00:00
self.failUnlessEqual(b[0].remote_read(0, 25), b"%25d" % 0)
b_str = str(b[0])
self.failUnlessIn("BucketReader", b_str)
self.failUnlessIn("mfwgy33dmf2g 0", b_str)
2008-06-18 00:01:42 +00:00
# now if we ask about writing again, the server should offer those
# three buckets as already present. It should offer them even if we
# don't ask about those specific ones.
2020-08-27 19:49:04 +00:00
already,writers = self.allocate(ss, b"allocate", [2,3,4], 75)
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
# refuse to offer them to uploaders
2020-08-27 19:49:04 +00:00
already2,writers2 = self.allocate(ss, b"allocate", [2,3,4,5], 75)
2008-06-18 00:01:42 +00:00
self.failUnlessEqual(already2, set([0,1,2]))
self.failUnlessEqual(set(writers2.keys()), set([5]))
# aborting the writes should remove the tempfiles
for i,wb in writers2.items():
wb.remote_abort()
2020-08-27 19:49:04 +00:00
already2,writers2 = self.allocate(ss, b"allocate", [2,3,4,5], 75)
2008-06-18 00:01:42 +00:00
self.failUnlessEqual(already2, set([0,1,2]))
self.failUnlessEqual(set(writers2.keys()), set([5]))
for i,wb in writers2.items():
wb.remote_abort()
for i,wb in writers.items():
wb.remote_abort()
def test_bad_container_version(self):
ss = self.create("test_bad_container_version")
2020-08-27 19:49:04 +00:00
a,w = self.allocate(ss, b"si1", [0], 10)
w[0].remote_write(0, b"\xff"*10)
w[0].remote_close()
2020-08-27 19:49:04 +00:00
fn = os.path.join(ss.sharedir, storage_index_to_dir(b"si1"), "0")
f = open(fn, "rb+")
f.seek(0)
f.write(struct.pack(">L", 0)) # this is invalid: minimum used is v1
f.close()
2020-08-27 19:49:04 +00:00
ss.remote_get_buckets(b"allocate")
e = self.failUnlessRaises(UnknownImmutableContainerVersionError,
2020-08-27 19:49:04 +00:00
ss.remote_get_buckets, b"si1")
self.failUnlessIn(" had version 0 but we wanted 1", str(e))
2008-06-18 00:01:42 +00:00
def test_disconnect(self):
# simulate a disconnection
ss = self.create("test_disconnect")
canary = FakeCanary()
2020-08-27 19:49:04 +00:00
already,writers = self.allocate(ss, b"disconnect", [0,1,2], 75, canary)
2008-06-18 00:01:42 +00:00
self.failUnlessEqual(already, set())
self.failUnlessEqual(set(writers.keys()), set([0,1,2]))
for (f,args,kwargs) in canary.disconnectors.values():
f(*args, **kwargs)
del already
del writers
# that ought to delete the incoming shares
2020-08-27 19:49:04 +00:00
already,writers = self.allocate(ss, b"disconnect", [0,1,2], 75)
2008-06-18 00:01:42 +00:00
self.failUnlessEqual(already, set())
self.failUnlessEqual(set(writers.keys()), set([0,1,2]))
def test_reserved_space(self):
reserved = 10000
allocated = 0
def call_get_disk_stats(whichdir, reserved_space=0):
self.failUnlessEqual(reserved_space, reserved)
return {
'free_for_nonroot': 15000 - allocated,
'avail': max(15000 - allocated - reserved_space, 0),
}
self.patch(fileutil, 'get_disk_stats', call_get_disk_stats)
ss = self.create("test_reserved_space", reserved_space=reserved)
# 15k available, 10k reserved, leaves 5k for shares
# a newly created and filled share incurs this much overhead, beyond
# the size we request.
OVERHEAD = 3*4
LEASE_SIZE = 4+32+32+4
2008-06-18 00:01:42 +00:00
canary = FakeCanary(True)
2020-08-27 19:36:54 +00:00
already, writers = self.allocate(ss, b"vid1", [0,1,2], 1000, canary)
self.failUnlessEqual(len(writers), 3)
# now the StorageServer should have 3000 bytes provisionally
# allocated, allowing only 2000 more to be claimed
self.failUnlessEqual(len(ss._active_writers), 3)
# allocating 1001-byte shares only leaves room for one
2020-08-27 19:36:54 +00:00
already2, writers2 = self.allocate(ss, b"vid2", [0,1,2], 1001, canary)
self.failUnlessEqual(len(writers2), 1)
self.failUnlessEqual(len(ss._active_writers), 4)
# we abandon the first set, so their provisional allocation should be
# returned
2019-12-07 10:20:01 +00:00
2020-05-05 19:57:08 +00:00
del already
del writers
gc.collect()
self.failUnlessEqual(len(ss._active_writers), 1)
# now we have a provisional allocation of 1001 bytes
# and we close the second set, so their provisional allocation should
# become real, long-term allocation, and grows to include the
# overhead.
for bw in writers2.values():
2020-08-27 19:49:04 +00:00
bw.remote_write(0, b"a"*25)
bw.remote_close()
del already2
del writers2
del bw
self.failUnlessEqual(len(ss._active_writers), 0)
# this also changes the amount reported as available by call_get_disk_stats
allocated = 1001 + OVERHEAD + LEASE_SIZE
# now there should be ALLOCATED=1001+12+72=1085 bytes allocated, and
# 5000-1085=3915 free, therefore we can fit 39 100byte shares
2020-08-31 17:20:57 +00:00
already3, writers3 = self.allocate(ss, b"vid3", list(range(100)), 100, canary)
self.failUnlessEqual(len(writers3), 39)
self.failUnlessEqual(len(ss._active_writers), 39)
2020-05-05 19:57:08 +00:00
del already3
del writers3
gc.collect()
self.failUnlessEqual(len(ss._active_writers), 0)
ss.disownServiceParent()
del ss
def test_seek(self):
basedir = self.workdir("test_seek_behavior")
fileutil.make_dirs(basedir)
filename = os.path.join(basedir, "testfile")
f = open(filename, "wb")
2020-08-27 19:36:54 +00:00
f.write(b"start")
f.close()
# mode="w" allows seeking-to-create-holes, but truncates pre-existing
# files. mode="a" preserves previous contents but does not allow
# seeking-to-create-holes. mode="r+" allows both.
f = open(filename, "rb+")
f.seek(100)
2020-08-27 19:36:54 +00:00
f.write(b"100")
f.close()
filelen = os.stat(filename)[stat.ST_SIZE]
self.failUnlessEqual(filelen, 100+3)
f2 = open(filename, "rb")
2020-08-27 19:36:54 +00:00
self.failUnlessEqual(f2.read(5), b"start")
def test_leases(self):
ss = self.create("test_leases")
canary = FakeCanary()
2020-08-31 17:20:57 +00:00
sharenums = list(range(5))
size = 100
2020-08-27 19:49:04 +00:00
rs0,cs0 = (hashutil.tagged_hash(b"blah", b"%d" % next(self._lease_secret)),
hashutil.tagged_hash(b"blah", b"%d" % next(self._lease_secret)))
already,writers = ss.remote_allocate_buckets(b"si0", rs0, cs0,
sharenums, size, canary)
self.failUnlessEqual(len(already), 0)
self.failUnlessEqual(len(writers), 5)
for wb in writers.values():
wb.remote_close()
2020-08-27 19:49:04 +00:00
leases = list(ss.get_leases(b"si0"))
self.failUnlessEqual(len(leases), 1)
self.failUnlessEqual(set([l.renew_secret for l in leases]), set([rs0]))
2020-08-27 19:49:04 +00:00
rs1,cs1 = (hashutil.tagged_hash(b"blah", b"%d" % next(self._lease_secret)),
hashutil.tagged_hash(b"blah", b"%d" % next(self._lease_secret)))
already,writers = ss.remote_allocate_buckets(b"si1", rs1, cs1,
sharenums, size, canary)
for wb in writers.values():
wb.remote_close()
# take out a second lease on si1
2020-08-27 19:49:04 +00:00
rs2,cs2 = (hashutil.tagged_hash(b"blah", b"%d" % next(self._lease_secret)),
hashutil.tagged_hash(b"blah", b"%d" % next(self._lease_secret)))
already,writers = ss.remote_allocate_buckets(b"si1", rs2, cs2,
sharenums, size, canary)
self.failUnlessEqual(len(already), 5)
self.failUnlessEqual(len(writers), 0)
2020-08-27 19:49:04 +00:00
leases = list(ss.get_leases(b"si1"))
self.failUnlessEqual(len(leases), 2)
self.failUnlessEqual(set([l.renew_secret for l in leases]), set([rs1, rs2]))
# and a third lease, using add-lease
2020-08-27 19:49:04 +00:00
rs2a,cs2a = (hashutil.tagged_hash(b"blah", b"%d" % next(self._lease_secret)),
hashutil.tagged_hash(b"blah", b"%d" % next(self._lease_secret)))
ss.remote_add_lease(b"si1", rs2a, cs2a)
leases = list(ss.get_leases(b"si1"))
self.failUnlessEqual(len(leases), 3)
self.failUnlessEqual(set([l.renew_secret for l in leases]), set([rs1, rs2, rs2a]))
# add-lease on a missing storage index is silently ignored
2020-08-27 19:49:04 +00:00
self.failUnlessEqual(ss.remote_add_lease(b"si18", b"", b""), None)
# check that si0 is readable
2020-08-27 19:49:04 +00:00
readers = ss.remote_get_buckets(b"si0")
self.failUnlessEqual(len(readers), 5)
# renew the first lease. Only the proper renew_secret should work
2020-08-27 19:49:04 +00:00
ss.remote_renew_lease(b"si0", rs0)
self.failUnlessRaises(IndexError, ss.remote_renew_lease, b"si0", cs0)
self.failUnlessRaises(IndexError, ss.remote_renew_lease, b"si0", rs1)
# check that si0 is still readable
2020-08-27 19:49:04 +00:00
readers = ss.remote_get_buckets(b"si0")
self.failUnlessEqual(len(readers), 5)
# There is no such method as remote_cancel_lease for now -- see
# ticket #1528.
self.failIf(hasattr(ss, 'remote_cancel_lease'), \
"ss should not have a 'remote_cancel_lease' method/attribute")
# test overlapping uploads
2020-08-27 19:49:04 +00:00
rs3,cs3 = (hashutil.tagged_hash(b"blah", b"%d" % next(self._lease_secret)),
hashutil.tagged_hash(b"blah", b"%d" % next(self._lease_secret)))
rs4,cs4 = (hashutil.tagged_hash(b"blah", b"%d" % next(self._lease_secret)),
hashutil.tagged_hash(b"blah", b"%d" % next(self._lease_secret)))
already,writers = ss.remote_allocate_buckets(b"si3", rs3, cs3,
sharenums, size, canary)
self.failUnlessEqual(len(already), 0)
self.failUnlessEqual(len(writers), 5)
2020-08-27 19:49:04 +00:00
already2,writers2 = ss.remote_allocate_buckets(b"si3", rs4, cs4,
sharenums, size, canary)
self.failUnlessEqual(len(already2), 0)
self.failUnlessEqual(len(writers2), 0)
for wb in writers.values():
wb.remote_close()
2020-08-27 19:49:04 +00:00
leases = list(ss.get_leases(b"si3"))
self.failUnlessEqual(len(leases), 1)
2020-08-27 19:49:04 +00:00
already3,writers3 = ss.remote_allocate_buckets(b"si3", rs4, cs4,
sharenums, size, canary)
self.failUnlessEqual(len(already3), 5)
self.failUnlessEqual(len(writers3), 0)
2020-08-27 19:49:04 +00:00
leases = list(ss.get_leases(b"si3"))
self.failUnlessEqual(len(leases), 2)
def test_readonly(self):
workdir = self.workdir("test_readonly")
2020-08-27 19:36:54 +00:00
ss = StorageServer(workdir, b"\x00" * 20, readonly_storage=True)
ss.setServiceParent(self.sparent)
2020-08-27 19:36:54 +00:00
already,writers = self.allocate(ss, b"vid", [0,1,2], 75)
self.failUnlessEqual(already, set())
self.failUnlessEqual(writers, {})
stats = ss.get_stats()
self.failUnlessEqual(stats["storage_server.accepting_immutable_shares"], 0)
if "storage_server.disk_avail" in stats:
# Some platforms may not have an API to get disk stats.
# But if there are stats, readonly_storage means disk_avail=0
self.failUnlessEqual(stats["storage_server.disk_avail"], 0)
def test_discard(self):
# discard is really only used for other tests, but we test it anyways
workdir = self.workdir("test_discard")
2020-08-27 19:36:54 +00:00
ss = StorageServer(workdir, b"\x00" * 20, discard_storage=True)
ss.setServiceParent(self.sparent)
2020-08-27 19:36:54 +00:00
already,writers = self.allocate(ss, b"vid", [0,1,2], 75)
self.failUnlessEqual(already, set())
self.failUnlessEqual(set(writers.keys()), set([0,1,2]))
for i,wb in writers.items():
2020-08-27 19:36:54 +00:00
wb.remote_write(0, b"%25d" % i)
wb.remote_close()
# since we discard the data, the shares should be present but sparse.
# Since we write with some seeks, the data we read back will be all
# zeros.
2020-08-27 19:36:54 +00:00
b = ss.remote_get_buckets(b"vid")
self.failUnlessEqual(set(b.keys()), set([0,1,2]))
2020-08-27 19:36:54 +00:00
self.failUnlessEqual(b[0].remote_read(0, 25), b"\x00" * 25)
def test_advise_corruption(self):
workdir = self.workdir("test_advise_corruption")
2020-08-27 19:36:54 +00:00
ss = StorageServer(workdir, b"\x00" * 20, discard_storage=True)
ss.setServiceParent(self.sparent)
2020-08-27 19:49:04 +00:00
si0_s = base32.b2a(b"si0")
ss.remote_advise_corrupt_share(b"immutable", b"si0", 0,
b"This share smells funny.\n")
reportdir = os.path.join(workdir, "corruption-advisories")
reports = os.listdir(reportdir)
self.failUnlessEqual(len(reports), 1)
report_si0 = reports[0]
2020-08-27 19:49:04 +00:00
self.failUnlessIn(native_str(si0_s), report_si0)
f = open(os.path.join(reportdir, report_si0), "rb")
report = f.read()
f.close()
2020-08-27 19:49:04 +00:00
self.failUnlessIn(b"type: immutable", report)
self.failUnlessIn(b"storage_index: %s" % si0_s, report)
self.failUnlessIn(b"share_number: 0", report)
self.failUnlessIn(b"This share smells funny.", report)
# test the RIBucketWriter version too
2020-08-27 19:49:04 +00:00
si1_s = base32.b2a(b"si1")
already,writers = self.allocate(ss, b"si1", [1], 75)
self.failUnlessEqual(already, set())
self.failUnlessEqual(set(writers.keys()), set([1]))
2020-08-27 19:49:04 +00:00
writers[1].remote_write(0, b"data")
writers[1].remote_close()
2020-08-27 19:49:04 +00:00
b = ss.remote_get_buckets(b"si1")
self.failUnlessEqual(set(b.keys()), set([1]))
b[1].remote_advise_corrupt_share(b"This share tastes like dust.\n")
reports = os.listdir(reportdir)
self.failUnlessEqual(len(reports), 2)
report_si1 = [r for r in reports if bytes_to_native_str(si1_s) in r][0]
2020-08-27 19:49:04 +00:00
f = open(os.path.join(reportdir, report_si1), "rb")
report = f.read()
f.close()
2020-08-27 19:49:04 +00:00
self.failUnlessIn(b"type: immutable", report)
self.failUnlessIn(b"storage_index: %s" % si1_s, report)
self.failUnlessIn(b"share_number: 1", report)
self.failUnlessIn(b"This share tastes like dust.", report)
class MutableServer(unittest.TestCase):
def setUp(self):
2008-01-14 18:58:58 +00:00
self.sparent = LoggingServiceParent()
self._lease_secret = itertools.count()
def tearDown(self):
return self.sparent.stopService()
def workdir(self, name):
basedir = os.path.join("storage", "MutableServer", name)
return basedir
def create(self, name):
workdir = self.workdir(name)
2020-08-27 19:36:54 +00:00
ss = StorageServer(workdir, b"\x00" * 20)
ss.setServiceParent(self.sparent)
return ss
def test_create(self):
self.create("test_create")
def write_enabler(self, we_tag):
2020-08-27 19:58:03 +00:00
return hashutil.tagged_hash(b"we_blah", we_tag)
def renew_secret(self, tag):
2020-08-27 19:58:03 +00:00
if isinstance(tag, int):
tag = b"%d" % (tag,)
assert isinstance(tag, bytes)
return hashutil.tagged_hash(b"renew_blah", tag)
def cancel_secret(self, tag):
2020-08-27 19:58:03 +00:00
if isinstance(tag, int):
tag = b"%d" % (tag,)
assert isinstance(tag, bytes)
return hashutil.tagged_hash(b"cancel_blah", tag)
def allocate(self, ss, storage_index, we_tag, lease_tag, sharenums, size):
write_enabler = self.write_enabler(we_tag)
renew_secret = self.renew_secret(lease_tag)
cancel_secret = self.cancel_secret(lease_tag)
rstaraw = ss.remote_slot_testv_and_readv_and_writev
testandwritev = dict( [ (shnum, ([], [], None) )
for shnum in sharenums ] )
readv = []
rc = rstaraw(storage_index,
(write_enabler, renew_secret, cancel_secret),
testandwritev,
readv)
(did_write, readv_data) = rc
self.failUnless(did_write)
self.failUnless(isinstance(readv_data, dict))
self.failUnlessEqual(len(readv_data), 0)
def test_bad_magic(self):
ss = self.create("test_bad_magic")
2020-08-27 19:58:03 +00:00
self.allocate(ss, b"si1", b"we1", next(self._lease_secret), set([0]), 10)
fn = os.path.join(ss.sharedir, storage_index_to_dir(b"si1"), "0")
f = open(fn, "rb+")
f.seek(0)
2020-08-27 19:58:03 +00:00
f.write(b"BAD MAGIC")
f.close()
read = ss.remote_slot_readv
e = self.failUnlessRaises(UnknownMutableContainerVersionError,
2020-08-27 19:58:03 +00:00
read, b"si1", [0], [(0,10)])
self.failUnlessIn(" had magic ", str(e))
self.failUnlessIn(" but we wanted ", str(e))
2008-06-18 00:01:42 +00:00
def test_container_size(self):
ss = self.create("test_container_size")
2020-08-27 19:58:03 +00:00
self.allocate(ss, b"si1", b"we1", next(self._lease_secret),
2008-06-18 00:01:42 +00:00
set([0,1,2]), 100)
read = ss.remote_slot_readv
2008-06-18 00:01:42 +00:00
rstaraw = ss.remote_slot_testv_and_readv_and_writev
2020-08-27 19:58:03 +00:00
secrets = ( self.write_enabler(b"we1"),
self.renew_secret(b"we1"),
self.cancel_secret(b"we1") )
data = b"".join([ (b"%d" % i) * 10 for i in range(10) ])
answer = rstaraw(b"si1", secrets,
2008-06-18 00:01:42 +00:00
{0: ([], [(0,data)], len(data)+12)},
[])
self.failUnlessEqual(answer, (True, {0:[],1:[],2:[]}) )
# Trying to make the container too large (by sending a write vector
# whose offset is too high) will raise an exception.
2008-06-18 00:01:42 +00:00
TOOBIG = MutableShareFile.MAX_SIZE + 10
self.failUnlessRaises(DataTooLargeError,
2020-08-27 19:58:03 +00:00
rstaraw, b"si1", secrets,
{0: ([], [(TOOBIG,data)], None)},
2008-06-18 00:01:42 +00:00
[])
2020-08-27 19:58:03 +00:00
answer = rstaraw(b"si1", secrets,
{0: ([], [(0,data)], None)},
2008-06-18 00:01:42 +00:00
[])
self.failUnlessEqual(answer, (True, {0:[],1:[],2:[]}) )
2020-08-27 19:58:03 +00:00
read_answer = read(b"si1", [0], [(0,10)])
self.failUnlessEqual(read_answer, {0: [data[:10]]})
# Sending a new_length shorter than the current length truncates the
# data.
2020-08-27 19:58:03 +00:00
answer = rstaraw(b"si1", secrets,
{0: ([], [], 9)},
[])
2020-08-27 19:58:03 +00:00
read_answer = read(b"si1", [0], [(0,10)])
self.failUnlessEqual(read_answer, {0: [data[:9]]})
# Sending a new_length longer than the current length doesn't change
# the data.
2020-08-27 19:58:03 +00:00
answer = rstaraw(b"si1", secrets,
{0: ([], [], 20)},
[])
assert answer == (True, {0:[],1:[],2:[]})
2020-08-27 19:58:03 +00:00
read_answer = read(b"si1", [0], [(0, 20)])
self.failUnlessEqual(read_answer, {0: [data[:9]]})
# Sending a write vector whose start is after the end of the current
# data doesn't reveal "whatever was there last time" (palimpsest),
# but instead fills with zeroes.
# To test this, we fill the data area with a recognizable pattern.
2020-08-27 19:58:03 +00:00
pattern = u''.join([chr(i) for i in range(100)]).encode("utf-8")
answer = rstaraw(b"si1", secrets,
{0: ([], [(0, pattern)], None)},
[])
assert answer == (True, {0:[],1:[],2:[]})
# Then truncate the data...
2020-08-27 19:58:03 +00:00
answer = rstaraw(b"si1", secrets,
{0: ([], [], 20)},
[])
assert answer == (True, {0:[],1:[],2:[]})
# Just confirm that you get an empty string if you try to read from
# past the (new) endpoint now.
2020-08-27 19:58:03 +00:00
answer = rstaraw(b"si1", secrets,
{0: ([], [], None)},
[(20, 1980)])
2020-08-27 19:58:03 +00:00
self.failUnlessEqual(answer, (True, {0:[b''],1:[b''],2:[b'']}))
# Then the extend the file by writing a vector which starts out past
# the end...
2020-08-27 19:58:03 +00:00
answer = rstaraw(b"si1", secrets,
{0: ([], [(50, b'hellothere')], None)},
[])
assert answer == (True, {0:[],1:[],2:[]})
# Now if you read the stuff between 20 (where we earlier truncated)
# and 50, it had better be all zeroes.
2020-08-27 19:58:03 +00:00
answer = rstaraw(b"si1", secrets,
{0: ([], [], None)},
[(20, 30)])
2020-08-27 19:58:03 +00:00
self.failUnlessEqual(answer, (True, {0:[b'\x00'*30],1:[b''],2:[b'']}))
# Also see if the server explicitly declares that it supports this
# feature.
ver = ss.remote_get_version()
storage_v1_ver = ver["http://allmydata.org/tahoe/protocols/storage/v1"]
self.failUnless(storage_v1_ver.get("fills-holes-with-zero-bytes"))
# If the size is dropped to zero the share is deleted.
2020-08-27 19:58:03 +00:00
answer = rstaraw(b"si1", secrets,
{0: ([], [(0,data)], 0)},
[])
self.failUnlessEqual(answer, (True, {0:[],1:[],2:[]}) )
2020-08-27 19:58:03 +00:00
read_answer = read(b"si1", [0], [(0,10)])
self.failUnlessEqual(read_answer, {})
def test_allocate(self):
ss = self.create("test_allocate")
2020-08-28 14:53:52 +00:00
self.allocate(ss, b"si1", b"we1", next(self._lease_secret),
2008-06-18 00:01:42 +00:00
set([0,1,2]), 100)
read = ss.remote_slot_readv
2020-08-28 14:53:52 +00:00
self.failUnlessEqual(read(b"si1", [0], [(0, 10)]),
{0: [b""]})
self.failUnlessEqual(read(b"si1", [], [(0, 10)]),
{0: [b""], 1: [b""], 2: [b""]})
self.failUnlessEqual(read(b"si1", [0], [(100, 10)]),
{0: [b""]})
# try writing to one
2020-08-28 14:53:52 +00:00
secrets = ( self.write_enabler(b"we1"),
self.renew_secret(b"we1"),
self.cancel_secret(b"we1") )
data = b"".join([ (b"%d" % i) * 10 for i in range(10) ])
write = ss.remote_slot_testv_and_readv_and_writev
2020-08-28 14:53:52 +00:00
answer = write(b"si1", secrets,
{0: ([], [(0,data)], None)},
[])
self.failUnlessEqual(answer, (True, {0:[],1:[],2:[]}) )
2020-08-28 14:53:52 +00:00
self.failUnlessEqual(read(b"si1", [0], [(0,20)]),
{0: [b"00000000001111111111"]})
self.failUnlessEqual(read(b"si1", [0], [(95,10)]),
{0: [b"99999"]})
#self.failUnlessEqual(s0.remote_get_length(), 100)
2020-08-28 14:53:52 +00:00
bad_secrets = (b"bad write enabler", secrets[1], secrets[2])
f = self.failUnlessRaises(BadWriteEnablerError,
2020-08-28 14:53:52 +00:00
write, b"si1", bad_secrets,
{}, [])
2020-08-28 14:53:52 +00:00
self.failUnlessIn("The write enabler was recorded by nodeid 'aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa'.", str(f))
# this testv should fail
2020-08-28 14:53:52 +00:00
answer = write(b"si1", secrets,
{0: ([(0, 12, b"eq", b"444444444444"),
(20, 5, b"eq", b"22222"),
],
2020-08-28 14:53:52 +00:00
[(0, b"x"*100)],
None),
},
[(0,12), (20,5)],
)
self.failUnlessEqual(answer, (False,
2020-08-28 14:53:52 +00:00
{0: [b"000000000011", b"22222"],
1: [b"", b""],
2: [b"", b""],
}))
2020-08-28 14:53:52 +00:00
self.failUnlessEqual(read(b"si1", [0], [(0,100)]), {0: [data]})
# as should this one
2020-08-28 14:53:52 +00:00
answer = write(b"si1", secrets,
{0: ([(10, 5, b"lt", b"11111"),
],
2020-08-28 14:53:52 +00:00
[(0, b"x"*100)],
None),
},
[(10,5)],
)
self.failUnlessEqual(answer, (False,
2020-08-28 14:53:52 +00:00
{0: [b"11111"],
1: [b""],
2: [b""]},
))
2020-08-28 14:53:52 +00:00
self.failUnlessEqual(read(b"si1", [0], [(0,100)]), {0: [data]})
def test_operators(self):
# test operators, the data we're comparing is '11111' in all cases.
# test both fail+pass, reset data after each one.
ss = self.create("test_operators")
2020-08-28 14:53:52 +00:00
secrets = ( self.write_enabler(b"we1"),
self.renew_secret(b"we1"),
self.cancel_secret(b"we1") )
data = b"".join([ (b"%d" % i) * 10 for i in range(10) ])
write = ss.remote_slot_testv_and_readv_and_writev
read = ss.remote_slot_readv
def reset():
2020-08-28 14:53:52 +00:00
write(b"si1", secrets,
{0: ([], [(0,data)], None)},
[])
reset()
# lt
2020-08-28 14:53:52 +00:00
answer = write(b"si1", secrets, {0: ([(10, 5, b"lt", b"11110"),
],
2020-08-28 14:53:52 +00:00
[(0, b"x"*100)],
None,
)}, [(10,5)])
2020-08-28 14:53:52 +00:00
self.failUnlessEqual(answer, (False, {0: [b"11111"]}))
self.failUnlessEqual(read(b"si1", [0], [(0,100)]), {0: [data]})
self.failUnlessEqual(read(b"si1", [], [(0,100)]), {0: [data]})
reset()
2020-08-28 14:53:52 +00:00
answer = write(b"si1", secrets, {0: ([(10, 5, b"lt", b"11111"),
],
2020-08-28 14:53:52 +00:00
[(0, b"x"*100)],
None,
)}, [(10,5)])
2020-08-28 14:53:52 +00:00
self.failUnlessEqual(answer, (False, {0: [b"11111"]}))
self.failUnlessEqual(read(b"si1", [0], [(0,100)]), {0: [data]})
reset()
2020-08-28 14:53:52 +00:00
answer = write(b"si1", secrets, {0: ([(10, 5, b"lt", b"11112"),
],
2020-08-28 14:53:52 +00:00
[(0, b"y"*100)],
None,
)}, [(10,5)])
2020-08-28 14:53:52 +00:00
self.failUnlessEqual(answer, (True, {0: [b"11111"]}))
self.failUnlessEqual(read(b"si1", [0], [(0,100)]), {0: [b"y"*100]})
reset()
# le
2020-08-28 14:53:52 +00:00
answer = write(b"si1", secrets, {0: ([(10, 5, b"le", b"11110"),
],
2020-08-28 14:53:52 +00:00
[(0, b"x"*100)],
None,
)}, [(10,5)])
2020-08-28 14:53:52 +00:00
self.failUnlessEqual(answer, (False, {0: [b"11111"]}))
self.failUnlessEqual(read(b"si1", [0], [(0,100)]), {0: [data]})
reset()
2020-08-28 14:53:52 +00:00
answer = write(b"si1", secrets, {0: ([(10, 5, b"le", b"11111"),
],
2020-08-28 14:53:52 +00:00
[(0, b"y"*100)],
None,
)}, [(10,5)])
2020-08-28 14:53:52 +00:00
self.failUnlessEqual(answer, (True, {0: [b"11111"]}))
self.failUnlessEqual(read(b"si1", [0], [(0,100)]), {0: [b"y"*100]})
reset()
2020-08-28 14:53:52 +00:00
answer = write(b"si1", secrets, {0: ([(10, 5, b"le", b"11112"),
],
2020-08-28 14:53:52 +00:00
[(0, b"y"*100)],
None,
)}, [(10,5)])
2020-08-28 14:53:52 +00:00
self.failUnlessEqual(answer, (True, {0: [b"11111"]}))
self.failUnlessEqual(read(b"si1", [0], [(0,100)]), {0: [b"y"*100]})
reset()
# eq
2020-08-28 14:53:52 +00:00
answer = write(b"si1", secrets, {0: ([(10, 5, b"eq", b"11112"),
],
2020-08-28 14:53:52 +00:00
[(0, b"x"*100)],
None,
)}, [(10,5)])
2020-08-28 14:53:52 +00:00
self.failUnlessEqual(answer, (False, {0: [b"11111"]}))
self.failUnlessEqual(read(b"si1", [0], [(0,100)]), {0: [data]})
reset()
2020-08-28 14:53:52 +00:00
answer = write(b"si1", secrets, {0: ([(10, 5, b"eq", b"11111"),
],
2020-08-28 14:53:52 +00:00
[(0, b"y"*100)],
None,
)}, [(10,5)])
2020-08-28 14:53:52 +00:00
self.failUnlessEqual(answer, (True, {0: [b"11111"]}))
self.failUnlessEqual(read(b"si1", [0], [(0,100)]), {0: [b"y"*100]})
reset()
# ne
2020-08-28 14:53:52 +00:00
answer = write(b"si1", secrets, {0: ([(10, 5, b"ne", b"11111"),
],
2020-08-28 14:53:52 +00:00
[(0, b"x"*100)],
None,
)}, [(10,5)])
2020-08-28 14:53:52 +00:00
self.failUnlessEqual(answer, (False, {0: [b"11111"]}))
self.failUnlessEqual(read(b"si1", [0], [(0,100)]), {0: [data]})
reset()
2020-08-28 14:53:52 +00:00
answer = write(b"si1", secrets, {0: ([(10, 5, b"ne", b"11112"),
],
2020-08-28 14:53:52 +00:00
[(0, b"y"*100)],
None,
)}, [(10,5)])
2020-08-28 14:53:52 +00:00
self.failUnlessEqual(answer, (True, {0: [b"11111"]}))
self.failUnlessEqual(read(b"si1", [0], [(0,100)]), {0: [b"y"*100]})
reset()
# ge
2020-08-28 14:53:52 +00:00
answer = write(b"si1", secrets, {0: ([(10, 5, b"ge", b"11110"),
],
2020-08-28 14:53:52 +00:00
[(0, b"y"*100)],
None,
)}, [(10,5)])
2020-08-28 14:53:52 +00:00
self.failUnlessEqual(answer, (True, {0: [b"11111"]}))
self.failUnlessEqual(read(b"si1", [0], [(0,100)]), {0: [b"y"*100]})
reset()
2020-08-28 14:53:52 +00:00
answer = write(b"si1", secrets, {0: ([(10, 5, b"ge", b"11111"),
],
2020-08-28 14:53:52 +00:00
[(0, b"y"*100)],
None,
)}, [(10,5)])
2020-08-28 14:53:52 +00:00
self.failUnlessEqual(answer, (True, {0: [b"11111"]}))
self.failUnlessEqual(read(b"si1", [0], [(0,100)]), {0: [b"y"*100]})
reset()
2020-08-28 14:53:52 +00:00
answer = write(b"si1", secrets, {0: ([(10, 5, b"ge", b"11112"),
],
2020-08-28 14:53:52 +00:00
[(0, b"y"*100)],
None,
)}, [(10,5)])
2020-08-28 14:53:52 +00:00
self.failUnlessEqual(answer, (False, {0: [b"11111"]}))
self.failUnlessEqual(read(b"si1", [0], [(0,100)]), {0: [data]})
reset()
# gt
2020-08-28 14:53:52 +00:00
answer = write(b"si1", secrets, {0: ([(10, 5, b"gt", b"11110"),
],
2020-08-28 14:53:52 +00:00
[(0, b"y"*100)],
None,
)}, [(10,5)])
2020-08-28 14:53:52 +00:00
self.failUnlessEqual(answer, (True, {0: [b"11111"]}))
self.failUnlessEqual(read(b"si1", [0], [(0,100)]), {0: [b"y"*100]})
reset()
2020-08-28 14:53:52 +00:00
answer = write(b"si1", secrets, {0: ([(10, 5, b"gt", b"11111"),
],
2020-08-28 14:53:52 +00:00
[(0, b"x"*100)],
None,
)}, [(10,5)])
2020-08-28 14:53:52 +00:00
self.failUnlessEqual(answer, (False, {0: [b"11111"]}))
self.failUnlessEqual(read(b"si1", [0], [(0,100)]), {0: [data]})
reset()
2020-08-28 14:53:52 +00:00
answer = write(b"si1", secrets, {0: ([(10, 5, b"gt", b"11112"),
],
2020-08-28 14:53:52 +00:00
[(0, b"x"*100)],
None,
)}, [(10,5)])
2020-08-28 14:53:52 +00:00
self.failUnlessEqual(answer, (False, {0: [b"11111"]}))
self.failUnlessEqual(read(b"si1", [0], [(0,100)]), {0: [data]})
reset()
2008-06-18 00:44:10 +00:00
# finally, test some operators against empty shares
2020-08-28 14:53:52 +00:00
answer = write(b"si1", secrets, {1: ([(10, 5, b"eq", b"11112"),
2008-06-18 00:44:10 +00:00
],
2020-08-28 14:53:52 +00:00
[(0, b"x"*100)],
2008-06-18 00:44:10 +00:00
None,
)}, [(10,5)])
2020-08-28 14:53:52 +00:00
self.failUnlessEqual(answer, (False, {0: [b"11111"]}))
self.failUnlessEqual(read(b"si1", [0], [(0,100)]), {0: [data]})
2008-06-18 00:44:10 +00:00
reset()
def test_readv(self):
ss = self.create("test_readv")
2020-08-28 14:53:52 +00:00
secrets = ( self.write_enabler(b"we1"),
self.renew_secret(b"we1"),
self.cancel_secret(b"we1") )
data = b"".join([ (b"%d" % i) * 10 for i in range(10) ])
write = ss.remote_slot_testv_and_readv_and_writev
read = ss.remote_slot_readv
2020-08-28 14:53:52 +00:00
data = [(b"%d" % i) * 100 for i in range(3)]
rc = write(b"si1", secrets,
{0: ([], [(0,data[0])], None),
1: ([], [(0,data[1])], None),
2: ([], [(0,data[2])], None),
}, [])
self.failUnlessEqual(rc, (True, {}))
2020-08-28 14:53:52 +00:00
answer = read(b"si1", [], [(0, 10)])
self.failUnlessEqual(answer, {0: [b"0"*10],
1: [b"1"*10],
2: [b"2"*10]})
def compare_leases_without_timestamps(self, leases_a, leases_b):
self.failUnlessEqual(len(leases_a), len(leases_b))
for i in range(len(leases_a)):
a = leases_a[i]
b = leases_b[i]
self.failUnlessEqual(a.owner_num, b.owner_num)
self.failUnlessEqual(a.renew_secret, b.renew_secret)
self.failUnlessEqual(a.cancel_secret, b.cancel_secret)
self.failUnlessEqual(a.nodeid, b.nodeid)
def compare_leases(self, leases_a, leases_b):
self.failUnlessEqual(len(leases_a), len(leases_b))
for i in range(len(leases_a)):
a = leases_a[i]
b = leases_b[i]
self.failUnlessEqual(a.owner_num, b.owner_num)
self.failUnlessEqual(a.renew_secret, b.renew_secret)
self.failUnlessEqual(a.cancel_secret, b.cancel_secret)
self.failUnlessEqual(a.nodeid, b.nodeid)
self.failUnlessEqual(a.expiration_time, b.expiration_time)
def test_leases(self):
ss = self.create("test_leases")
def secrets(n):
2020-08-28 14:53:52 +00:00
return ( self.write_enabler(b"we1"),
self.renew_secret(b"we1-%d" % n),
self.cancel_secret(b"we1-%d" % n) )
data = b"".join([ (b"%d" % i) * 10 for i in range(10) ])
write = ss.remote_slot_testv_and_readv_and_writev
read = ss.remote_slot_readv
2020-08-28 14:53:52 +00:00
rc = write(b"si1", secrets(0), {0: ([], [(0,data)], None)}, [])
self.failUnlessEqual(rc, (True, {}))
# create a random non-numeric file in the bucket directory, to
# exercise the code that's supposed to ignore those.
bucket_dir = os.path.join(self.workdir("test_leases"),
2020-08-28 14:53:52 +00:00
"shares", storage_index_to_dir(b"si1"))
f = open(os.path.join(bucket_dir, "ignore_me.txt"), "w")
f.write("you ought to be ignoring me\n")
f.close()
s0 = MutableShareFile(os.path.join(bucket_dir, "0"))
self.failUnlessEqual(len(list(s0.get_leases())), 1)
# add-lease on a missing storage index is silently ignored
2020-08-28 14:53:52 +00:00
self.failUnlessEqual(ss.remote_add_lease(b"si18", b"", b""), None)
# re-allocate the slots and use the same secrets, that should update
# the lease
2020-08-28 14:53:52 +00:00
write(b"si1", secrets(0), {0: ([], [(0,data)], None)}, [])
self.failUnlessEqual(len(list(s0.get_leases())), 1)
# renew it directly
2020-08-28 14:53:52 +00:00
ss.remote_renew_lease(b"si1", secrets(0)[1])
self.failUnlessEqual(len(list(s0.get_leases())), 1)
# now allocate them with a bunch of different secrets, to trigger the
# extended lease code. Use add_lease for one of them.
2020-08-28 14:53:52 +00:00
write(b"si1", secrets(1), {0: ([], [(0,data)], None)}, [])
self.failUnlessEqual(len(list(s0.get_leases())), 2)
secrets2 = secrets(2)
2020-08-28 14:53:52 +00:00
ss.remote_add_lease(b"si1", secrets2[1], secrets2[2])
self.failUnlessEqual(len(list(s0.get_leases())), 3)
2020-08-28 14:53:52 +00:00
write(b"si1", secrets(3), {0: ([], [(0,data)], None)}, [])
write(b"si1", secrets(4), {0: ([], [(0,data)], None)}, [])
write(b"si1", secrets(5), {0: ([], [(0,data)], None)}, [])
self.failUnlessEqual(len(list(s0.get_leases())), 6)
all_leases = list(s0.get_leases())
# and write enough data to expand the container, forcing the server
# to move the leases
2020-08-28 14:53:52 +00:00
write(b"si1", secrets(0),
{0: ([], [(0,data)], 200), },
[])
# read back the leases, make sure they're still intact.
self.compare_leases_without_timestamps(all_leases, list(s0.get_leases()))
2020-08-28 14:53:52 +00:00
ss.remote_renew_lease(b"si1", secrets(0)[1])
ss.remote_renew_lease(b"si1", secrets(1)[1])
ss.remote_renew_lease(b"si1", secrets(2)[1])
ss.remote_renew_lease(b"si1", secrets(3)[1])
ss.remote_renew_lease(b"si1", secrets(4)[1])
self.compare_leases_without_timestamps(all_leases, list(s0.get_leases()))
# get a new copy of the leases, with the current timestamps. Reading
# data and failing to renew/cancel leases should leave the timestamps
# alone.
all_leases = list(s0.get_leases())
# renewing with a bogus token should prompt an error message
# examine the exception thus raised, make sure the old nodeid is
# present, to provide for share migration
e = self.failUnlessRaises(IndexError,
2020-08-28 14:53:52 +00:00
ss.remote_renew_lease, b"si1",
secrets(20)[1])
e_s = str(e)
self.failUnlessIn("Unable to renew non-existent lease", e_s)
self.failUnlessIn("I have leases accepted by nodeids:", e_s)
self.failUnlessIn("nodeids: 'aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa' .", e_s)
self.compare_leases(all_leases, list(s0.get_leases()))
# reading shares should not modify the timestamp
2020-08-28 14:53:52 +00:00
read(b"si1", [], [(0,200)])
self.compare_leases(all_leases, list(s0.get_leases()))
2020-08-28 14:53:52 +00:00
write(b"si1", secrets(0),
{0: ([], [(200, b"make me bigger")], None)}, [])
self.compare_leases_without_timestamps(all_leases, list(s0.get_leases()))
2020-08-28 14:53:52 +00:00
write(b"si1", secrets(0),
{0: ([], [(500, b"make me really bigger")], None)}, [])
self.compare_leases_without_timestamps(all_leases, list(s0.get_leases()))
def test_remove(self):
ss = self.create("test_remove")
2020-08-28 14:53:52 +00:00
self.allocate(ss, b"si1", b"we1", next(self._lease_secret),
set([0,1,2]), 100)
readv = ss.remote_slot_readv
writev = ss.remote_slot_testv_and_readv_and_writev
2020-08-28 14:53:52 +00:00
secrets = ( self.write_enabler(b"we1"),
self.renew_secret(b"we1"),
self.cancel_secret(b"we1") )
# delete sh0 by setting its size to zero
2020-08-28 14:53:52 +00:00
answer = writev(b"si1", secrets,
{0: ([], [], 0)},
[])
# the answer should mention all the shares that existed before the
# write
self.failUnlessEqual(answer, (True, {0:[],1:[],2:[]}) )
# but a new read should show only sh1 and sh2
2020-08-28 14:53:52 +00:00
self.failUnlessEqual(readv(b"si1", [], [(0,10)]),
{1: [b""], 2: [b""]})
# delete sh1 by setting its size to zero
2020-08-28 14:53:52 +00:00
answer = writev(b"si1", secrets,
{1: ([], [], 0)},
[])
self.failUnlessEqual(answer, (True, {1:[],2:[]}) )
2020-08-28 14:53:52 +00:00
self.failUnlessEqual(readv(b"si1", [], [(0,10)]),
{2: [b""]})
# delete sh2 by setting its size to zero
2020-08-28 14:53:52 +00:00
answer = writev(b"si1", secrets,
{2: ([], [], 0)},
[])
self.failUnlessEqual(answer, (True, {2:[]}) )
2020-08-28 14:53:52 +00:00
self.failUnlessEqual(readv(b"si1", [], [(0,10)]),
{})
# and the bucket directory should now be gone
2020-08-28 14:53:52 +00:00
si = base32.b2a(b"si1")
# note: this is a detail of the storage server implementation, and
# may change in the future
2020-08-28 14:53:52 +00:00
if PY3:
si = si.decode("utf-8")
prefix = si[:2]
prefixdir = os.path.join(self.workdir("test_remove"), "shares", prefix)
bucketdir = os.path.join(prefixdir, si)
self.failUnless(os.path.exists(prefixdir), prefixdir)
self.failIf(os.path.exists(bucketdir), bucketdir)
2008-06-18 00:01:42 +00:00
def test_writev_without_renew_lease(self):
"""
The helper method ``slot_testv_and_readv_and_writev`` does not renew
leases if ``False`` is passed for the ``renew_leases`` parameter.
"""
ss = self.create("test_writev_without_renew_lease")
2020-08-28 14:53:52 +00:00
storage_index = b"si2"
secrets = (
self.write_enabler(storage_index),
self.renew_secret(storage_index),
self.cancel_secret(storage_index),
)
sharenum = 3
datav = [(0, b"Hello, world")]
ss.slot_testv_and_readv_and_writev(
storage_index=storage_index,
secrets=secrets,
test_and_write_vectors={
sharenum: ([], datav, None),
},
read_vector=[],
renew_leases=False,
)
leases = list(ss.get_slot_leases(storage_index))
self.assertEqual([], leases)
def test_get_slot_leases_empty_slot(self):
"""
When ``get_slot_leases`` is called for a slot for which the server has no
shares, it returns an empty iterable.
"""
2020-08-28 14:53:52 +00:00
ss = self.create("test_get_slot_leases_empty_slot")
self.assertEqual(
list(ss.get_slot_leases(b"si1")),
[],
)
def test_remove_non_present(self):
"""
2019-08-23 12:11:38 +00:00
A write vector which would remove a share completely is applied as a no-op
by a server which does not have the share.
"""
ss = self.create("test_remove_non_present")
2020-08-28 14:53:52 +00:00
storage_index = b"si1"
secrets = (
self.write_enabler(storage_index),
self.renew_secret(storage_index),
self.cancel_secret(storage_index),
)
sharenum = 3
testv = []
datav = []
new_length = 0
read_vector = []
# We don't even need to create any shares to exercise this
# functionality. Just go straight to sending a truncate-to-zero
# write.
testv_is_good, read_data = ss.remote_slot_testv_and_readv_and_writev(
storage_index=storage_index,
secrets=secrets,
test_and_write_vectors={
sharenum: (testv, datav, new_length),
},
read_vector=read_vector,
)
self.assertTrue(testv_is_good)
self.assertEqual({}, read_data)
class MDMFProxies(unittest.TestCase, ShouldFailMixin):
def setUp(self):
self.sparent = LoggingServiceParent()
self._lease_secret = itertools.count()
self.ss = self.create("MDMFProxies storage test server")
self.rref = RemoteBucket(self.ss)
self.storage_server = _StorageServer(lambda: self.rref)
2020-08-28 16:41:19 +00:00
self.secrets = (self.write_enabler(b"we_secret"),
self.renew_secret(b"renew_secret"),
self.cancel_secret(b"cancel_secret"))
self.segment = b"aaaaaa"
self.block = b"aa"
self.salt = b"a" * 16
self.block_hash = b"a" * 32
self.block_hash_tree = [self.block_hash for i in range(6)]
self.share_hash = self.block_hash
2020-08-28 16:41:19 +00:00
self.share_hash_chain = dict([(i, self.share_hash) for i in range(6)])
self.signature = b"foobarbaz"
self.verification_key = b"vvvvvv"
self.encprivkey = b"private"
self.root_hash = self.block_hash
self.salt_hash = self.root_hash
2020-08-28 16:41:19 +00:00
self.salt_hash_tree = [self.salt_hash for i in range(6)]
self.block_hash_tree_s = self.serialize_blockhashes(self.block_hash_tree)
self.share_hash_chain_s = self.serialize_sharehashes(self.share_hash_chain)
# blockhashes and salt hashes are serialized in the same way,
# only we lop off the first element and store that in the
# header.
self.salt_hash_tree_s = self.serialize_blockhashes(self.salt_hash_tree[1:])
def tearDown(self):
self.sparent.stopService()
shutil.rmtree(self.workdir("MDMFProxies storage test server"))
def write_enabler(self, we_tag):
2020-08-28 16:41:19 +00:00
return hashutil.tagged_hash(b"we_blah", we_tag)
def renew_secret(self, tag):
2020-08-28 16:41:19 +00:00
if isinstance(tag, int):
tag = b"%d" % tag
return hashutil.tagged_hash(b"renew_blah", tag)
def cancel_secret(self, tag):
2020-08-28 16:41:19 +00:00
if isinstance(tag, int):
tag = b"%d" % tag
return hashutil.tagged_hash(b"cancel_blah", tag)
def workdir(self, name):
basedir = os.path.join("storage", "MutableServer", name)
return basedir
def create(self, name):
workdir = self.workdir(name)
2020-08-27 19:36:54 +00:00
ss = StorageServer(workdir, b"\x00" * 20)
ss.setServiceParent(self.sparent)
return ss
def build_test_mdmf_share(self, tail_segment=False, empty=False):
# Start with the checkstring
data = struct.pack(">BQ32s",
1,
0,
self.root_hash)
self.checkstring = data
# Next, the encoding parameters
if tail_segment:
data += struct.pack(">BBQQ",
3,
10,
6,
33)
elif empty:
data += struct.pack(">BBQQ",
3,
10,
0,
0)
else:
data += struct.pack(">BBQQ",
3,
10,
6,
36)
# Now we'll build the offsets.
2020-08-28 16:41:19 +00:00
sharedata = b""
if not tail_segment and not empty:
2020-08-28 16:41:19 +00:00
for i in range(6):
sharedata += self.salt + self.block
elif tail_segment:
2020-08-28 16:41:19 +00:00
for i in range(5):
sharedata += self.salt + self.block
2020-08-28 16:41:19 +00:00
sharedata += self.salt + b"a"
# The encrypted private key comes after the shares + salts
offset_size = struct.calcsize(MDMFOFFSETS)
encrypted_private_key_offset = len(data) + offset_size
# The share has chain comes after the private key
sharehashes_offset = encrypted_private_key_offset + \
len(self.encprivkey)
# The signature comes after the share hash chain.
signature_offset = sharehashes_offset + len(self.share_hash_chain_s)
verification_key_offset = signature_offset + len(self.signature)
verification_key_end = verification_key_offset + \
len(self.verification_key)
share_data_offset = offset_size
share_data_offset += PRIVATE_KEY_SIZE
share_data_offset += SIGNATURE_SIZE
share_data_offset += VERIFICATION_KEY_SIZE
share_data_offset += SHARE_HASH_CHAIN_SIZE
blockhashes_offset = share_data_offset + len(sharedata)
eof_offset = blockhashes_offset + len(self.block_hash_tree_s)
data += struct.pack(MDMFOFFSETS,
encrypted_private_key_offset,
sharehashes_offset,
signature_offset,
verification_key_offset,
verification_key_end,
share_data_offset,
blockhashes_offset,
eof_offset)
self.offsets = {}
self.offsets['enc_privkey'] = encrypted_private_key_offset
self.offsets['block_hash_tree'] = blockhashes_offset
self.offsets['share_hash_chain'] = sharehashes_offset
self.offsets['signature'] = signature_offset
self.offsets['verification_key'] = verification_key_offset
self.offsets['share_data'] = share_data_offset
self.offsets['verification_key_end'] = verification_key_end
self.offsets['EOF'] = eof_offset
# the private key,
data += self.encprivkey
# the sharehashes
data += self.share_hash_chain_s
# the signature,
data += self.signature
# and the verification key
data += self.verification_key
# Then we'll add in gibberish until we get to the right point.
2020-08-28 16:41:19 +00:00
nulls = b"".join([b" " for i in range(len(data), share_data_offset)])
data += nulls
# Then the share data
data += sharedata
# the blockhashes
data += self.block_hash_tree_s
return data
def write_test_share_to_server(self,
storage_index,
tail_segment=False,
empty=False):
"""
I write some data for the read tests to read to self.ss
If tail_segment=True, then I will write a share that has a
smaller tail segment than other segments.
"""
write = self.ss.remote_slot_testv_and_readv_and_writev
data = self.build_test_mdmf_share(tail_segment, empty)
# Finally, we write the whole thing to the storage server in one
# pass.
2020-08-28 14:53:52 +00:00
testvs = [(0, 1, b"eq", b"")]
tws = {}
tws[0] = (testvs, [(0, data)], None)
readv = [(0, 1)]
results = write(storage_index, self.secrets, tws, readv)
self.failUnless(results[0])
def build_test_sdmf_share(self, empty=False):
if empty:
2020-08-28 16:41:19 +00:00
sharedata = b""
else:
sharedata = self.segment * 6
self.sharedata = sharedata
2020-08-28 16:41:19 +00:00
blocksize = len(sharedata) // 3
block = sharedata[:blocksize]
self.blockdata = block
prefix = struct.pack(">BQ32s16s BBQQ",
0, # version,
0,
self.root_hash,
self.salt,
3,
10,
len(sharedata),
len(sharedata),
)
post_offset = struct.calcsize(">BQ32s16sBBQQLLLLQQ")
signature_offset = post_offset + len(self.verification_key)
sharehashes_offset = signature_offset + len(self.signature)
blockhashes_offset = sharehashes_offset + len(self.share_hash_chain_s)
sharedata_offset = blockhashes_offset + len(self.block_hash_tree_s)
encprivkey_offset = sharedata_offset + len(block)
eof_offset = encprivkey_offset + len(self.encprivkey)
offsets = struct.pack(">LLLLQQ",
signature_offset,
sharehashes_offset,
blockhashes_offset,
sharedata_offset,
encprivkey_offset,
eof_offset)
2020-08-28 14:53:52 +00:00
final_share = b"".join([prefix,
2020-08-31 15:59:34 +00:00
offsets,
self.verification_key,
self.signature,
self.share_hash_chain_s,
self.block_hash_tree_s,
block,
self.encprivkey])
self.offsets = {}
self.offsets['signature'] = signature_offset
self.offsets['share_hash_chain'] = sharehashes_offset
self.offsets['block_hash_tree'] = blockhashes_offset
self.offsets['share_data'] = sharedata_offset
self.offsets['enc_privkey'] = encprivkey_offset
self.offsets['EOF'] = eof_offset
return final_share
def write_sdmf_share_to_server(self,
storage_index,
empty=False):
# Some tests need SDMF shares to verify that we can still
# read them. This method writes one, which resembles but is not
write = self.ss.remote_slot_testv_and_readv_and_writev
share = self.build_test_sdmf_share(empty)
2020-08-28 14:53:52 +00:00
testvs = [(0, 1, b"eq", b"")]
tws = {}
tws[0] = (testvs, [(0, share)], None)
readv = []
results = write(storage_index, self.secrets, tws, readv)
self.failUnless(results[0])
def test_read(self):
2020-08-28 16:41:19 +00:00
self.write_test_share_to_server(b"si1")
2020-08-28 14:53:52 +00:00
mr = MDMFSlotReadProxy(self.storage_server, b"si1", 0)
# Check that every method equals what we expect it to.
d = defer.succeed(None)
def _check_block_and_salt(block_and_salt):
(block, salt) = block_and_salt
self.failUnlessEqual(block, self.block)
self.failUnlessEqual(salt, self.salt)
2020-08-28 16:41:19 +00:00
for i in range(6):
d.addCallback(lambda ignored, i=i:
mr.get_block_and_salt(i))
d.addCallback(_check_block_and_salt)
d.addCallback(lambda ignored:
mr.get_encprivkey())
d.addCallback(lambda encprivkey:
self.failUnlessEqual(self.encprivkey, encprivkey))
d.addCallback(lambda ignored:
mr.get_blockhashes())
d.addCallback(lambda blockhashes:
self.failUnlessEqual(self.block_hash_tree, blockhashes))
d.addCallback(lambda ignored:
mr.get_sharehashes())
d.addCallback(lambda sharehashes:
self.failUnlessEqual(self.share_hash_chain, sharehashes))
d.addCallback(lambda ignored:
mr.get_signature())
d.addCallback(lambda signature:
self.failUnlessEqual(signature, self.signature))
d.addCallback(lambda ignored:
mr.get_verification_key())
d.addCallback(lambda verification_key:
self.failUnlessEqual(verification_key, self.verification_key))
d.addCallback(lambda ignored:
mr.get_seqnum())
d.addCallback(lambda seqnum:
self.failUnlessEqual(seqnum, 0))
d.addCallback(lambda ignored:
mr.get_root_hash())
d.addCallback(lambda root_hash:
self.failUnlessEqual(self.root_hash, root_hash))
d.addCallback(lambda ignored:
mr.get_seqnum())
d.addCallback(lambda seqnum:
self.failUnlessEqual(0, seqnum))
d.addCallback(lambda ignored:
mr.get_encoding_parameters())
def _check_encoding_parameters(args):
(k, n, segsize, datalen) = args
self.failUnlessEqual(k, 3)
self.failUnlessEqual(n, 10)
self.failUnlessEqual(segsize, 6)
self.failUnlessEqual(datalen, 36)
d.addCallback(_check_encoding_parameters)
d.addCallback(lambda ignored:
mr.get_checkstring())
d.addCallback(lambda checkstring:
self.failUnlessEqual(checkstring, checkstring))
return d
def test_read_with_different_tail_segment_size(self):
2020-08-28 16:41:19 +00:00
self.write_test_share_to_server(b"si1", tail_segment=True)
2020-08-28 14:53:52 +00:00
mr = MDMFSlotReadProxy(self.storage_server, b"si1", 0)
d = mr.get_block_and_salt(5)
def _check_tail_segment(results):
block, salt = results
self.failUnlessEqual(len(block), 1)
2020-08-28 14:53:52 +00:00
self.failUnlessEqual(block, b"a")
d.addCallback(_check_tail_segment)
return d
def test_get_block_with_invalid_segnum(self):
2020-08-28 16:41:19 +00:00
self.write_test_share_to_server(b"si1")
2020-08-28 14:53:52 +00:00
mr = MDMFSlotReadProxy(self.storage_server, b"si1", 0)
d = defer.succeed(None)
d.addCallback(lambda ignored:
self.shouldFail(LayoutInvalid, "test invalid segnum",
None,
mr.get_block_and_salt, 7))
return d
def test_get_encoding_parameters_first(self):
2020-08-28 16:41:19 +00:00
self.write_test_share_to_server(b"si1")
2020-08-28 14:53:52 +00:00
mr = MDMFSlotReadProxy(self.storage_server, b"si1", 0)
d = mr.get_encoding_parameters()
def _check_encoding_parameters(args):
(k, n, segment_size, datalen) = args
self.failUnlessEqual(k, 3)
self.failUnlessEqual(n, 10)
self.failUnlessEqual(segment_size, 6)
self.failUnlessEqual(datalen, 36)
d.addCallback(_check_encoding_parameters)
return d
def test_get_seqnum_first(self):
2020-08-28 16:41:19 +00:00
self.write_test_share_to_server(b"si1")
2020-08-28 14:53:52 +00:00
mr = MDMFSlotReadProxy(self.storage_server, b"si1", 0)
d = mr.get_seqnum()
d.addCallback(lambda seqnum:
self.failUnlessEqual(seqnum, 0))
return d
def test_get_root_hash_first(self):
2020-08-28 16:41:19 +00:00
self.write_test_share_to_server(b"si1")
2020-08-28 14:53:52 +00:00
mr = MDMFSlotReadProxy(self.storage_server, b"si1", 0)
d = mr.get_root_hash()
d.addCallback(lambda root_hash:
self.failUnlessEqual(root_hash, self.root_hash))
return d
def test_get_checkstring_first(self):
2020-08-28 16:41:19 +00:00
self.write_test_share_to_server(b"si1")
2020-08-28 14:53:52 +00:00
mr = MDMFSlotReadProxy(self.storage_server, b"si1", 0)
d = mr.get_checkstring()
d.addCallback(lambda checkstring:
self.failUnlessEqual(checkstring, self.checkstring))
return d
def test_write_read_vectors(self):
# When writing for us, the storage server will return to us a
# read vector, along with its result. If a write fails because
# the test vectors failed, this read vector can help us to
# diagnose the problem. This test ensures that the read vector
# is working appropriately.
2020-08-28 16:41:19 +00:00
mw = self._make_new_mw(b"si1", 0)
2020-08-28 16:41:19 +00:00
for i in range(6):
mw.put_block(self.block, i, self.salt)
mw.put_encprivkey(self.encprivkey)
mw.put_blockhashes(self.block_hash_tree)
mw.put_sharehashes(self.share_hash_chain)
mw.put_root_hash(self.root_hash)
mw.put_signature(self.signature)
mw.put_verification_key(self.verification_key)
d = mw.finish_publishing()
def _then(results):
self.failUnless(len(results), 2)
result, readv = results
self.failUnless(result)
self.failIf(readv)
self.old_checkstring = mw.get_checkstring()
2020-08-28 16:41:19 +00:00
mw.set_checkstring(b"")
d.addCallback(_then)
d.addCallback(lambda ignored:
mw.finish_publishing())
def _then_again(results):
self.failUnlessEqual(len(results), 2)
result, readvs = results
self.failIf(result)
self.failUnlessIn(0, readvs)
readv = readvs[0][0]
self.failUnlessEqual(readv, self.old_checkstring)
d.addCallback(_then_again)
# The checkstring remains the same for the rest of the process.
return d
def test_private_key_after_share_hash_chain(self):
2020-08-28 16:41:19 +00:00
mw = self._make_new_mw(b"si1", 0)
d = defer.succeed(None)
2020-08-28 16:41:19 +00:00
for i in range(6):
d.addCallback(lambda ignored, i=i:
mw.put_block(self.block, i, self.salt))
d.addCallback(lambda ignored:
mw.put_encprivkey(self.encprivkey))
d.addCallback(lambda ignored:
mw.put_sharehashes(self.share_hash_chain))
# Now try to put the private key again.
d.addCallback(lambda ignored:
self.shouldFail(LayoutInvalid, "test repeat private key",
None,
mw.put_encprivkey, self.encprivkey))
return d
def test_signature_after_verification_key(self):
2020-08-28 16:41:19 +00:00
mw = self._make_new_mw(b"si1", 0)
d = defer.succeed(None)
# Put everything up to and including the verification key.
2020-08-28 16:41:19 +00:00
for i in range(6):
d.addCallback(lambda ignored, i=i:
mw.put_block(self.block, i, self.salt))
d.addCallback(lambda ignored:
mw.put_encprivkey(self.encprivkey))
d.addCallback(lambda ignored:
mw.put_blockhashes(self.block_hash_tree))
d.addCallback(lambda ignored:
mw.put_sharehashes(self.share_hash_chain))
d.addCallback(lambda ignored:
mw.put_root_hash(self.root_hash))
d.addCallback(lambda ignored:
mw.put_signature(self.signature))
d.addCallback(lambda ignored:
mw.put_verification_key(self.verification_key))
# Now try to put the signature again. This should fail
d.addCallback(lambda ignored:
self.shouldFail(LayoutInvalid, "signature after verification",
None,
mw.put_signature, self.signature))
return d
def test_uncoordinated_write(self):
# Make two mutable writers, both pointing to the same storage
# server, both at the same storage index, and try writing to the
# same share.
2020-08-28 16:41:19 +00:00
mw1 = self._make_new_mw(b"si1", 0)
mw2 = self._make_new_mw(b"si1", 0)
def _check_success(results):
result, readvs = results
self.failUnless(result)
def _check_failure(results):
result, readvs = results
self.failIf(result)
def _write_share(mw):
2020-08-28 16:41:19 +00:00
for i in range(6):
mw.put_block(self.block, i, self.salt)
mw.put_encprivkey(self.encprivkey)
mw.put_blockhashes(self.block_hash_tree)
mw.put_sharehashes(self.share_hash_chain)
mw.put_root_hash(self.root_hash)
mw.put_signature(self.signature)
mw.put_verification_key(self.verification_key)
return mw.finish_publishing()
d = _write_share(mw1)
d.addCallback(_check_success)
d.addCallback(lambda ignored:
_write_share(mw2))
d.addCallback(_check_failure)
return d
def test_invalid_salt_size(self):
# Salts need to be 16 bytes in size. Writes that attempt to
# write more or less than this should be rejected.
2020-08-28 16:41:19 +00:00
mw = self._make_new_mw(b"si1", 0)
invalid_salt = b"a" * 17 # 17 bytes
another_invalid_salt = b"b" * 15 # 15 bytes
d = defer.succeed(None)
d.addCallback(lambda ignored:
self.shouldFail(LayoutInvalid, "salt too big",
None,
mw.put_block, self.block, 0, invalid_salt))
d.addCallback(lambda ignored:
self.shouldFail(LayoutInvalid, "salt too small",
None,
mw.put_block, self.block, 0,
another_invalid_salt))
return d
def test_write_test_vectors(self):
# If we give the write proxy a bogus test vector at
# any point during the process, it should fail to write when we
# tell it to write.
def _check_failure(results):
self.failUnlessEqual(len(results), 2)
res, d = results
self.failIf(res)
def _check_success(results):
self.failUnlessEqual(len(results), 2)
res, d = results
self.failUnless(results)
2020-08-28 16:41:19 +00:00
mw = self._make_new_mw(b"si1", 0)
mw.set_checkstring(b"this is a lie")
for i in range(6):
mw.put_block(self.block, i, self.salt)
mw.put_encprivkey(self.encprivkey)
mw.put_blockhashes(self.block_hash_tree)
mw.put_sharehashes(self.share_hash_chain)
mw.put_root_hash(self.root_hash)
mw.put_signature(self.signature)
mw.put_verification_key(self.verification_key)
d = mw.finish_publishing()
d.addCallback(_check_failure)
d.addCallback(lambda ignored:
2020-08-28 16:41:19 +00:00
mw.set_checkstring(b""))
d.addCallback(lambda ignored:
mw.finish_publishing())
d.addCallback(_check_success)
return d
def serialize_blockhashes(self, blockhashes):
2020-08-28 14:53:52 +00:00
return b"".join(blockhashes)
def serialize_sharehashes(self, sharehashes):
2020-08-28 14:53:52 +00:00
ret = b"".join([struct.pack(">H32s", i, sharehashes[i])
for i in sorted(sharehashes.keys())])
return ret
def test_write(self):
# This translates to a file with 6 6-byte segments, and with 2-byte
# blocks.
2020-08-28 16:41:19 +00:00
mw = self._make_new_mw(b"si1", 0)
# Test writing some blocks.
read = self.ss.remote_slot_readv
expected_private_key_offset = struct.calcsize(MDMFHEADER)
expected_sharedata_offset = struct.calcsize(MDMFHEADER) + \
PRIVATE_KEY_SIZE + \
SIGNATURE_SIZE + \
VERIFICATION_KEY_SIZE + \
SHARE_HASH_CHAIN_SIZE
written_block_size = 2 + len(self.salt)
written_block = self.block + self.salt
2020-08-28 16:41:19 +00:00
for i in range(6):
mw.put_block(self.block, i, self.salt)
mw.put_encprivkey(self.encprivkey)
mw.put_blockhashes(self.block_hash_tree)
mw.put_sharehashes(self.share_hash_chain)
mw.put_root_hash(self.root_hash)
mw.put_signature(self.signature)
mw.put_verification_key(self.verification_key)
d = mw.finish_publishing()
def _check_publish(results):
self.failUnlessEqual(len(results), 2)
result, ign = results
self.failUnless(result, "publish failed")
2020-08-28 16:41:19 +00:00
for i in range(6):
self.failUnlessEqual(read(b"si1", [0], [(expected_sharedata_offset + (i * written_block_size), written_block_size)]),
{0: [written_block]})
self.failUnlessEqual(len(self.encprivkey), 7)
2020-08-28 16:41:19 +00:00
self.failUnlessEqual(read(b"si1", [0], [(expected_private_key_offset, 7)]),
{0: [self.encprivkey]})
expected_block_hash_offset = expected_sharedata_offset + \
(6 * written_block_size)
self.failUnlessEqual(len(self.block_hash_tree_s), 32 * 6)
2020-08-28 16:41:19 +00:00
self.failUnlessEqual(read(b"si1", [0], [(expected_block_hash_offset, 32 * 6)]),
{0: [self.block_hash_tree_s]})
expected_share_hash_offset = expected_private_key_offset + len(self.encprivkey)
2020-08-28 16:41:19 +00:00
self.failUnlessEqual(read(b"si1", [0],[(expected_share_hash_offset, (32 + 2) * 6)]),
{0: [self.share_hash_chain_s]})
2020-08-28 16:41:19 +00:00
self.failUnlessEqual(read(b"si1", [0], [(9, 32)]),
{0: [self.root_hash]})
expected_signature_offset = expected_share_hash_offset + \
len(self.share_hash_chain_s)
self.failUnlessEqual(len(self.signature), 9)
2020-08-28 16:41:19 +00:00
self.failUnlessEqual(read(b"si1", [0], [(expected_signature_offset, 9)]),
{0: [self.signature]})
expected_verification_key_offset = expected_signature_offset + len(self.signature)
self.failUnlessEqual(len(self.verification_key), 6)
2020-08-28 16:41:19 +00:00
self.failUnlessEqual(read(b"si1", [0], [(expected_verification_key_offset, 6)]),
{0: [self.verification_key]})
signable = mw.get_signable()
verno, seq, roothash, k, n, segsize, datalen = \
struct.unpack(">BQ32sBBQQ",
signable)
self.failUnlessEqual(verno, 1)
self.failUnlessEqual(seq, 0)
self.failUnlessEqual(roothash, self.root_hash)
self.failUnlessEqual(k, 3)
self.failUnlessEqual(n, 10)
self.failUnlessEqual(segsize, 6)
self.failUnlessEqual(datalen, 36)
expected_eof_offset = expected_block_hash_offset + \
len(self.block_hash_tree_s)
# Check the version number to make sure that it is correct.
expected_version_number = struct.pack(">B", 1)
2020-08-28 16:41:19 +00:00
self.failUnlessEqual(read(b"si1", [0], [(0, 1)]),
{0: [expected_version_number]})
# Check the sequence number to make sure that it is correct
expected_sequence_number = struct.pack(">Q", 0)
2020-08-28 16:41:19 +00:00
self.failUnlessEqual(read(b"si1", [0], [(1, 8)]),
{0: [expected_sequence_number]})
# Check that the encoding parameters (k, N, segement size, data
# length) are what they should be. These are 3, 10, 6, 36
expected_k = struct.pack(">B", 3)
2020-08-28 16:41:19 +00:00
self.failUnlessEqual(read(b"si1", [0], [(41, 1)]),
{0: [expected_k]})
expected_n = struct.pack(">B", 10)
2020-08-28 16:41:19 +00:00
self.failUnlessEqual(read(b"si1", [0], [(42, 1)]),
{0: [expected_n]})
expected_segment_size = struct.pack(">Q", 6)
2020-08-28 16:41:19 +00:00
self.failUnlessEqual(read(b"si1", [0], [(43, 8)]),
{0: [expected_segment_size]})
expected_data_length = struct.pack(">Q", 36)
2020-08-28 16:41:19 +00:00
self.failUnlessEqual(read(b"si1", [0], [(51, 8)]),
{0: [expected_data_length]})
expected_offset = struct.pack(">Q", expected_private_key_offset)
2020-08-28 16:41:19 +00:00
self.failUnlessEqual(read(b"si1", [0], [(59, 8)]),
{0: [expected_offset]})
expected_offset = struct.pack(">Q", expected_share_hash_offset)
2020-08-28 16:41:19 +00:00
self.failUnlessEqual(read(b"si1", [0], [(67, 8)]),
{0: [expected_offset]})
expected_offset = struct.pack(">Q", expected_signature_offset)
2020-08-28 16:41:19 +00:00
self.failUnlessEqual(read(b"si1", [0], [(75, 8)]),
{0: [expected_offset]})
expected_offset = struct.pack(">Q", expected_verification_key_offset)
2020-08-28 16:41:19 +00:00
self.failUnlessEqual(read(b"si1", [0], [(83, 8)]),
{0: [expected_offset]})
expected_offset = struct.pack(">Q", expected_verification_key_offset + len(self.verification_key))
2020-08-28 16:41:19 +00:00
self.failUnlessEqual(read(b"si1", [0], [(91, 8)]),
{0: [expected_offset]})
expected_offset = struct.pack(">Q", expected_sharedata_offset)
2020-08-28 16:41:19 +00:00
self.failUnlessEqual(read(b"si1", [0], [(99, 8)]),
{0: [expected_offset]})
expected_offset = struct.pack(">Q", expected_block_hash_offset)
2020-08-28 16:41:19 +00:00
self.failUnlessEqual(read(b"si1", [0], [(107, 8)]),
{0: [expected_offset]})
expected_offset = struct.pack(">Q", expected_eof_offset)
2020-08-28 16:41:19 +00:00
self.failUnlessEqual(read(b"si1", [0], [(115, 8)]),
{0: [expected_offset]})
d.addCallback(_check_publish)
return d
def _make_new_mw(self, si, share, datalength=36):
# This is a file of size 36 bytes. Since it has a segment
# size of 6, we know that it has 6 byte segments, which will
# be split into blocks of 2 bytes because our FEC k
# parameter is 3.
mw = MDMFSlotWriteProxy(share, self.storage_server, si, self.secrets, 0, 3, 10,
6, datalength)
return mw
def test_write_rejected_with_too_many_blocks(self):
2020-08-28 16:41:19 +00:00
mw = self._make_new_mw(b"si0", 0)
# Try writing too many blocks. We should not be able to write
# more than 6
# blocks into each share.
d = defer.succeed(None)
2020-08-28 16:41:19 +00:00
for i in range(6):
d.addCallback(lambda ignored, i=i:
mw.put_block(self.block, i, self.salt))
d.addCallback(lambda ignored:
self.shouldFail(LayoutInvalid, "too many blocks",
None,
mw.put_block, self.block, 7, self.salt))
return d
def test_write_rejected_with_invalid_salt(self):
# Try writing an invalid salt. Salts are 16 bytes -- any more or
# less should cause an error.
2020-08-28 16:41:19 +00:00
mw = self._make_new_mw(b"si1", 0)
bad_salt = b"a" * 17 # 17 bytes
d = defer.succeed(None)
d.addCallback(lambda ignored:
self.shouldFail(LayoutInvalid, "test_invalid_salt",
None, mw.put_block, self.block, 7, bad_salt))
return d
def test_write_rejected_with_invalid_root_hash(self):
# Try writing an invalid root hash. This should be SHA256d, and
# 32 bytes long as a result.
2020-08-28 16:41:19 +00:00
mw = self._make_new_mw(b"si2", 0)
# 17 bytes != 32 bytes
2020-08-28 16:41:19 +00:00
invalid_root_hash = b"a" * 17
d = defer.succeed(None)
# Before this test can work, we need to put some blocks + salts,
# a block hash tree, and a share hash tree. Otherwise, we'll see
# failures that match what we are looking for, but are caused by
# the constraints imposed on operation ordering.
2020-08-28 16:41:19 +00:00
for i in range(6):
d.addCallback(lambda ignored, i=i:
mw.put_block(self.block, i, self.salt))
d.addCallback(lambda ignored:
mw.put_encprivkey(self.encprivkey))
d.addCallback(lambda ignored:
mw.put_blockhashes(self.block_hash_tree))
d.addCallback(lambda ignored:
mw.put_sharehashes(self.share_hash_chain))
d.addCallback(lambda ignored:
self.shouldFail(LayoutInvalid, "invalid root hash",
None, mw.put_root_hash, invalid_root_hash))
return d
def test_write_rejected_with_invalid_blocksize(self):
# The blocksize implied by the writer that we get from
# _make_new_mw is 2bytes -- any more or any less than this
# should be cause for failure, unless it is the tail segment, in
# which case it may not be failure.
2020-08-28 16:41:19 +00:00
invalid_block = b"a"
mw = self._make_new_mw(b"si3", 0, 33) # implies a tail segment with
# one byte blocks
# 1 bytes != 2 bytes
d = defer.succeed(None)
d.addCallback(lambda ignored, invalid_block=invalid_block:
self.shouldFail(LayoutInvalid, "test blocksize too small",
None, mw.put_block, invalid_block, 0,
self.salt))
invalid_block = invalid_block * 3
# 3 bytes != 2 bytes
d.addCallback(lambda ignored:
self.shouldFail(LayoutInvalid, "test blocksize too large",
None,
mw.put_block, invalid_block, 0, self.salt))
2020-08-28 16:41:19 +00:00
for i in range(5):
d.addCallback(lambda ignored, i=i:
mw.put_block(self.block, i, self.salt))
# Try to put an invalid tail segment
d.addCallback(lambda ignored:
self.shouldFail(LayoutInvalid, "test invalid tail segment",
None,
mw.put_block, self.block, 5, self.salt))
2020-08-28 16:41:19 +00:00
valid_block = b"a"
d.addCallback(lambda ignored:
mw.put_block(valid_block, 5, self.salt))
return d
def test_write_enforces_order_constraints(self):
# We require that the MDMFSlotWriteProxy be interacted with in a
# specific way.
# That way is:
# 0: __init__
# 1: write blocks and salts
# 2: Write the encrypted private key
# 3: Write the block hashes
# 4: Write the share hashes
# 5: Write the root hash and salt hash
# 6: Write the signature and verification key
# 7: Write the file.
#
# Some of these can be performed out-of-order, and some can't.
# The dependencies that I want to test here are:
# - Private key before block hashes
# - share hashes and block hashes before root hash
# - root hash before signature
# - signature before verification key
2020-08-28 16:41:19 +00:00
mw0 = self._make_new_mw(b"si0", 0)
# Write some shares
d = defer.succeed(None)
2020-08-28 16:41:19 +00:00
for i in range(6):
d.addCallback(lambda ignored, i=i:
mw0.put_block(self.block, i, self.salt))
# Try to write the share hash chain without writing the
# encrypted private key
d.addCallback(lambda ignored:
self.shouldFail(LayoutInvalid, "share hash chain before "
"private key",
None,
mw0.put_sharehashes, self.share_hash_chain))
# Write the private key.
d.addCallback(lambda ignored:
mw0.put_encprivkey(self.encprivkey))
# Now write the block hashes and try again
d.addCallback(lambda ignored:
mw0.put_blockhashes(self.block_hash_tree))
# We haven't yet put the root hash on the share, so we shouldn't
# be able to sign it.
d.addCallback(lambda ignored:
self.shouldFail(LayoutInvalid, "signature before root hash",
None, mw0.put_signature, self.signature))
d.addCallback(lambda ignored:
self.failUnlessRaises(LayoutInvalid, mw0.get_signable))
# ..and, since that fails, we also shouldn't be able to put the
# verification key.
d.addCallback(lambda ignored:
self.shouldFail(LayoutInvalid, "key before signature",
None, mw0.put_verification_key,
self.verification_key))
# Now write the share hashes.
d.addCallback(lambda ignored:
mw0.put_sharehashes(self.share_hash_chain))
# We should be able to write the root hash now too
d.addCallback(lambda ignored:
mw0.put_root_hash(self.root_hash))
# We should still be unable to put the verification key
d.addCallback(lambda ignored:
self.shouldFail(LayoutInvalid, "key before signature",
None, mw0.put_verification_key,
self.verification_key))
d.addCallback(lambda ignored:
mw0.put_signature(self.signature))
# We shouldn't be able to write the offsets to the remote server
# until the offset table is finished; IOW, until we have written
# the verification key.
d.addCallback(lambda ignored:
self.shouldFail(LayoutInvalid, "offsets before verification key",
None,
mw0.finish_publishing))
d.addCallback(lambda ignored:
mw0.put_verification_key(self.verification_key))
return d
def test_end_to_end(self):
2020-08-28 16:41:19 +00:00
mw = self._make_new_mw(b"si1", 0)
# Write a share using the mutable writer, and make sure that the
# reader knows how to read everything back to us.
d = defer.succeed(None)
2020-08-28 16:41:19 +00:00
for i in range(6):
d.addCallback(lambda ignored, i=i:
mw.put_block(self.block, i, self.salt))
d.addCallback(lambda ignored:
mw.put_encprivkey(self.encprivkey))
d.addCallback(lambda ignored:
mw.put_blockhashes(self.block_hash_tree))
d.addCallback(lambda ignored:
mw.put_sharehashes(self.share_hash_chain))
d.addCallback(lambda ignored:
mw.put_root_hash(self.root_hash))
d.addCallback(lambda ignored:
mw.put_signature(self.signature))
d.addCallback(lambda ignored:
mw.put_verification_key(self.verification_key))
d.addCallback(lambda ignored:
mw.finish_publishing())
2020-08-28 16:41:19 +00:00
mr = MDMFSlotReadProxy(self.storage_server, b"si1", 0)
def _check_block_and_salt(block_and_salt):
(block, salt) = block_and_salt
self.failUnlessEqual(block, self.block)
self.failUnlessEqual(salt, self.salt)
2020-08-28 16:41:19 +00:00
for i in range(6):
d.addCallback(lambda ignored, i=i:
mr.get_block_and_salt(i))
d.addCallback(_check_block_and_salt)
d.addCallback(lambda ignored:
mr.get_encprivkey())
d.addCallback(lambda encprivkey:
self.failUnlessEqual(self.encprivkey, encprivkey))
d.addCallback(lambda ignored:
mr.get_blockhashes())
d.addCallback(lambda blockhashes:
self.failUnlessEqual(self.block_hash_tree, blockhashes))
d.addCallback(lambda ignored:
mr.get_sharehashes())
d.addCallback(lambda sharehashes:
self.failUnlessEqual(self.share_hash_chain, sharehashes))
d.addCallback(lambda ignored:
mr.get_signature())
d.addCallback(lambda signature:
self.failUnlessEqual(signature, self.signature))
d.addCallback(lambda ignored:
mr.get_verification_key())
d.addCallback(lambda verification_key:
self.failUnlessEqual(verification_key, self.verification_key))
d.addCallback(lambda ignored:
mr.get_seqnum())
d.addCallback(lambda seqnum:
self.failUnlessEqual(seqnum, 0))
d.addCallback(lambda ignored:
mr.get_root_hash())
d.addCallback(lambda root_hash:
self.failUnlessEqual(self.root_hash, root_hash))
d.addCallback(lambda ignored:
mr.get_encoding_parameters())
def _check_encoding_parameters(args):
(k, n, segsize, datalen) = args
self.failUnlessEqual(k, 3)
self.failUnlessEqual(n, 10)
self.failUnlessEqual(segsize, 6)
self.failUnlessEqual(datalen, 36)
d.addCallback(_check_encoding_parameters)
d.addCallback(lambda ignored:
mr.get_checkstring())
d.addCallback(lambda checkstring:
self.failUnlessEqual(checkstring, mw.get_checkstring()))
return d
def test_is_sdmf(self):
# The MDMFSlotReadProxy should also know how to read SDMF files,
# since it will encounter them on the grid. Callers use the
# is_sdmf method to test this.
2020-08-28 16:41:19 +00:00
self.write_sdmf_share_to_server(b"si1")
2020-08-28 16:59:03 +00:00
mr = MDMFSlotReadProxy(self.storage_server, b"si1", 0)
d = mr.is_sdmf()
d.addCallback(lambda issdmf:
self.failUnless(issdmf))
return d
def test_reads_sdmf(self):
# The slot read proxy should, naturally, know how to tell us
# about data in the SDMF format
2020-08-28 16:41:19 +00:00
self.write_sdmf_share_to_server(b"si1")
2020-08-28 16:59:03 +00:00
mr = MDMFSlotReadProxy(self.storage_server, b"si1", 0)
d = defer.succeed(None)
d.addCallback(lambda ignored:
mr.is_sdmf())
d.addCallback(lambda issdmf:
self.failUnless(issdmf))
# What do we need to read?
# - The sharedata
# - The salt
d.addCallback(lambda ignored:
mr.get_block_and_salt(0))
def _check_block_and_salt(results):
block, salt = results
# Our original file is 36 bytes long. Then each share is 12
# bytes in size. The share is composed entirely of the
# letter a. self.block contains 2 as, so 6 * self.block is
# what we are looking for.
self.failUnlessEqual(block, self.block * 6)
self.failUnlessEqual(salt, self.salt)
d.addCallback(_check_block_and_salt)
# - The blockhashes
d.addCallback(lambda ignored:
mr.get_blockhashes())
d.addCallback(lambda blockhashes:
self.failUnlessEqual(self.block_hash_tree,
blockhashes,
blockhashes))
# - The sharehashes
d.addCallback(lambda ignored:
mr.get_sharehashes())
d.addCallback(lambda sharehashes:
self.failUnlessEqual(self.share_hash_chain,
sharehashes))
# - The keys
d.addCallback(lambda ignored:
mr.get_encprivkey())
d.addCallback(lambda encprivkey:
self.failUnlessEqual(encprivkey, self.encprivkey, encprivkey))
d.addCallback(lambda ignored:
mr.get_verification_key())
d.addCallback(lambda verification_key:
self.failUnlessEqual(verification_key,
self.verification_key,
verification_key))
# - The signature
d.addCallback(lambda ignored:
mr.get_signature())
d.addCallback(lambda signature:
self.failUnlessEqual(signature, self.signature, signature))
# - The sequence number
d.addCallback(lambda ignored:
mr.get_seqnum())
d.addCallback(lambda seqnum:
self.failUnlessEqual(seqnum, 0, seqnum))
# - The root hash
d.addCallback(lambda ignored:
mr.get_root_hash())
d.addCallback(lambda root_hash:
self.failUnlessEqual(root_hash, self.root_hash, root_hash))
return d
def test_only_reads_one_segment_sdmf(self):
# SDMF shares have only one segment, so it doesn't make sense to
# read more segments than that. The reader should know this and
# complain if we try to do that.
2020-08-28 16:41:19 +00:00
self.write_sdmf_share_to_server(b"si1")
2020-08-28 16:59:03 +00:00
mr = MDMFSlotReadProxy(self.storage_server, b"si1", 0)
d = defer.succeed(None)
d.addCallback(lambda ignored:
mr.is_sdmf())
d.addCallback(lambda issdmf:
self.failUnless(issdmf))
d.addCallback(lambda ignored:
self.shouldFail(LayoutInvalid, "test bad segment",
None,
mr.get_block_and_salt, 1))
return d
def test_read_with_prefetched_mdmf_data(self):
# The MDMFSlotReadProxy will prefill certain fields if you pass
# it data that you have already fetched. This is useful for
# cases like the Servermap, which prefetches ~2kb of data while
# finding out which shares are on the remote peer so that it
# doesn't waste round trips.
mdmf_data = self.build_test_mdmf_share()
2020-08-28 16:41:19 +00:00
self.write_test_share_to_server(b"si1")
def _make_mr(ignored, length):
2020-08-28 16:59:03 +00:00
mr = MDMFSlotReadProxy(self.storage_server, b"si1", 0, mdmf_data[:length])
return mr
d = defer.succeed(None)
# This should be enough to fill in both the encoding parameters
# and the table of offsets, which will complete the version
# information tuple.
d.addCallback(_make_mr, 123)
d.addCallback(lambda mr:
mr.get_verinfo())
def _check_verinfo(verinfo):
self.failUnless(verinfo)
self.failUnlessEqual(len(verinfo), 9)
(seqnum,
root_hash,
salt_hash,
segsize,
datalen,
k,
n,
prefix,
offsets) = verinfo
self.failUnlessEqual(seqnum, 0)
self.failUnlessEqual(root_hash, self.root_hash)
self.failUnlessEqual(segsize, 6)
self.failUnlessEqual(datalen, 36)
self.failUnlessEqual(k, 3)
self.failUnlessEqual(n, 10)
expected_prefix = struct.pack(MDMFSIGNABLEHEADER,
1,
seqnum,
root_hash,
k,
n,
segsize,
datalen)
self.failUnlessEqual(expected_prefix, prefix)
self.failUnlessEqual(self.rref.read_count, 0)
d.addCallback(_check_verinfo)
# This is not enough data to read a block and a share, so the
# wrapper should attempt to read this from the remote server.
d.addCallback(_make_mr, 123)
d.addCallback(lambda mr:
mr.get_block_and_salt(0))
def _check_block_and_salt(block_and_salt):
(block, salt) = block_and_salt
self.failUnlessEqual(block, self.block)
self.failUnlessEqual(salt, self.salt)
self.failUnlessEqual(self.rref.read_count, 1)
# This should be enough data to read one block.
d.addCallback(_make_mr, 123 + PRIVATE_KEY_SIZE + SIGNATURE_SIZE + VERIFICATION_KEY_SIZE + SHARE_HASH_CHAIN_SIZE + 140)
d.addCallback(lambda mr:
mr.get_block_and_salt(0))
d.addCallback(_check_block_and_salt)
return d
def test_read_with_prefetched_sdmf_data(self):
sdmf_data = self.build_test_sdmf_share()
2020-08-28 16:41:19 +00:00
self.write_sdmf_share_to_server(b"si1")
def _make_mr(ignored, length):
2020-08-28 16:59:03 +00:00
mr = MDMFSlotReadProxy(self.storage_server, b"si1", 0, sdmf_data[:length])
return mr
d = defer.succeed(None)
# This should be enough to get us the encoding parameters,
# offset table, and everything else we need to build a verinfo
# string.
d.addCallback(_make_mr, 123)
d.addCallback(lambda mr:
mr.get_verinfo())
def _check_verinfo(verinfo):
self.failUnless(verinfo)
self.failUnlessEqual(len(verinfo), 9)
(seqnum,
root_hash,
salt,
segsize,
datalen,
k,
n,
prefix,
offsets) = verinfo
self.failUnlessEqual(seqnum, 0)
self.failUnlessEqual(root_hash, self.root_hash)
self.failUnlessEqual(salt, self.salt)
self.failUnlessEqual(segsize, 36)
self.failUnlessEqual(datalen, 36)
self.failUnlessEqual(k, 3)
self.failUnlessEqual(n, 10)
expected_prefix = struct.pack(SIGNED_PREFIX,
0,
seqnum,
root_hash,
salt,
k,
n,
segsize,
datalen)
self.failUnlessEqual(expected_prefix, prefix)
self.failUnlessEqual(self.rref.read_count, 0)
d.addCallback(_check_verinfo)
# This shouldn't be enough to read any share data.
d.addCallback(_make_mr, 123)
d.addCallback(lambda mr:
mr.get_block_and_salt(0))
def _check_block_and_salt(block_and_salt):
(block, salt) = block_and_salt
self.failUnlessEqual(block, self.block * 6)
self.failUnlessEqual(salt, self.salt)
# TODO: Fix the read routine so that it reads only the data
# that it has cached if it can't read all of it.
self.failUnlessEqual(self.rref.read_count, 2)
# This should be enough to read share data.
d.addCallback(_make_mr, self.offsets['share_data'])
d.addCallback(lambda mr:
mr.get_block_and_salt(0))
d.addCallback(_check_block_and_salt)
return d
def test_read_with_empty_mdmf_file(self):
# Some tests upload a file with no contents to test things
# unrelated to the actual handling of the content of the file.
# The reader should behave intelligently in these cases.
2020-08-28 16:41:19 +00:00
self.write_test_share_to_server(b"si1", empty=True)
2020-08-28 16:59:03 +00:00
mr = MDMFSlotReadProxy(self.storage_server, b"si1", 0)
# We should be able to get the encoding parameters, and they
# should be correct.
d = defer.succeed(None)
d.addCallback(lambda ignored:
mr.get_encoding_parameters())
def _check_encoding_parameters(params):
self.failUnlessEqual(len(params), 4)
k, n, segsize, datalen = params
self.failUnlessEqual(k, 3)
self.failUnlessEqual(n, 10)
self.failUnlessEqual(segsize, 0)
self.failUnlessEqual(datalen, 0)
d.addCallback(_check_encoding_parameters)
# We should not be able to fetch a block, since there are no
# blocks to fetch
d.addCallback(lambda ignored:
self.shouldFail(LayoutInvalid, "get block on empty file",
None,
mr.get_block_and_salt, 0))
return d
def test_read_with_empty_sdmf_file(self):
2020-08-28 16:41:19 +00:00
self.write_sdmf_share_to_server(b"si1", empty=True)
2020-08-28 16:59:03 +00:00
mr = MDMFSlotReadProxy(self.storage_server, b"si1", 0)
# We should be able to get the encoding parameters, and they
# should be correct
d = defer.succeed(None)
d.addCallback(lambda ignored:
mr.get_encoding_parameters())
def _check_encoding_parameters(params):
self.failUnlessEqual(len(params), 4)
k, n, segsize, datalen = params
self.failUnlessEqual(k, 3)
self.failUnlessEqual(n, 10)
self.failUnlessEqual(segsize, 0)
self.failUnlessEqual(datalen, 0)
d.addCallback(_check_encoding_parameters)
# It does not make sense to get a block in this format, so we
# should not be able to.
d.addCallback(lambda ignored:
self.shouldFail(LayoutInvalid, "get block on an empty file",
None,
mr.get_block_and_salt, 0))
return d
def test_verinfo_with_sdmf_file(self):
2020-08-28 16:41:19 +00:00
self.write_sdmf_share_to_server(b"si1")
2020-08-28 16:59:03 +00:00
mr = MDMFSlotReadProxy(self.storage_server, b"si1", 0)
# We should be able to get the version information.
d = defer.succeed(None)
d.addCallback(lambda ignored:
mr.get_verinfo())
def _check_verinfo(verinfo):
self.failUnless(verinfo)
self.failUnlessEqual(len(verinfo), 9)
(seqnum,
root_hash,
salt,
segsize,
datalen,
k,
n,
prefix,
offsets) = verinfo
self.failUnlessEqual(seqnum, 0)
self.failUnlessEqual(root_hash, self.root_hash)
self.failUnlessEqual(salt, self.salt)
self.failUnlessEqual(segsize, 36)
self.failUnlessEqual(datalen, 36)
self.failUnlessEqual(k, 3)
self.failUnlessEqual(n, 10)
expected_prefix = struct.pack(">BQ32s16s BBQQ",
0,
seqnum,
root_hash,
salt,
k,
n,
segsize,
datalen)
self.failUnlessEqual(prefix, expected_prefix)
self.failUnlessEqual(offsets, self.offsets)
d.addCallback(_check_verinfo)
return d
def test_verinfo_with_mdmf_file(self):
2020-08-28 16:41:19 +00:00
self.write_test_share_to_server(b"si1")
2020-08-28 16:59:03 +00:00
mr = MDMFSlotReadProxy(self.storage_server, b"si1", 0)
d = defer.succeed(None)
d.addCallback(lambda ignored:
mr.get_verinfo())
def _check_verinfo(verinfo):
self.failUnless(verinfo)
self.failUnlessEqual(len(verinfo), 9)
(seqnum,
root_hash,
IV,
segsize,
datalen,
k,
n,
prefix,
offsets) = verinfo
self.failUnlessEqual(seqnum, 0)
self.failUnlessEqual(root_hash, self.root_hash)
self.failIf(IV)
self.failUnlessEqual(segsize, 6)
self.failUnlessEqual(datalen, 36)
self.failUnlessEqual(k, 3)
self.failUnlessEqual(n, 10)
expected_prefix = struct.pack(">BQ32s BBQQ",
1,
seqnum,
root_hash,
k,
n,
segsize,
datalen)
self.failUnlessEqual(prefix, expected_prefix)
self.failUnlessEqual(offsets, self.offsets)
d.addCallback(_check_verinfo)
return d
def test_sdmf_writer(self):
# Go through the motions of writing an SDMF share to the storage
# server. Then read the storage server to see that the share got
# written in the way that we think it should have.
# We do this first so that the necessary instance variables get
# set the way we want them for the tests below.
data = self.build_test_sdmf_share()
sdmfr = SDMFSlotWriteProxy(0,
self.storage_server,
2020-08-28 16:59:03 +00:00
b"si1",
self.secrets,
0, 3, 10, 36, 36)
# Put the block and salt.
sdmfr.put_block(self.blockdata, 0, self.salt)
# Put the encprivkey
sdmfr.put_encprivkey(self.encprivkey)
# Put the block and share hash chains
sdmfr.put_blockhashes(self.block_hash_tree)
sdmfr.put_sharehashes(self.share_hash_chain)
sdmfr.put_root_hash(self.root_hash)
# Put the signature
sdmfr.put_signature(self.signature)
# Put the verification key
sdmfr.put_verification_key(self.verification_key)
# Now check to make sure that nothing has been written yet.
self.failUnlessEqual(self.rref.write_count, 0)
# Now finish publishing
d = sdmfr.finish_publishing()
def _then(ignored):
self.failUnlessEqual(self.rref.write_count, 1)
read = self.ss.remote_slot_readv
2020-08-28 16:41:19 +00:00
self.failUnlessEqual(read(b"si1", [0], [(0, len(data))]),
{0: [data]})
d.addCallback(_then)
return d
def test_sdmf_writer_preexisting_share(self):
data = self.build_test_sdmf_share()
2020-08-28 16:41:19 +00:00
self.write_sdmf_share_to_server(b"si1")
# Now there is a share on the storage server. To successfully
# write, we need to set the checkstring correctly. When we
# don't, no write should occur.
sdmfw = SDMFSlotWriteProxy(0,
self.storage_server,
2020-08-28 16:59:03 +00:00
b"si1",
self.secrets,
1, 3, 10, 36, 36)
sdmfw.put_block(self.blockdata, 0, self.salt)
# Put the encprivkey
sdmfw.put_encprivkey(self.encprivkey)
# Put the block and share hash chains
sdmfw.put_blockhashes(self.block_hash_tree)
sdmfw.put_sharehashes(self.share_hash_chain)
# Put the root hash
sdmfw.put_root_hash(self.root_hash)
# Put the signature
sdmfw.put_signature(self.signature)
# Put the verification key
sdmfw.put_verification_key(self.verification_key)
# We shouldn't have a checkstring yet
2020-08-28 16:59:03 +00:00
self.failUnlessEqual(sdmfw.get_checkstring(), b"")
d = sdmfw.finish_publishing()
def _then(results):
self.failIf(results[0])
# this is the correct checkstring
self._expected_checkstring = results[1][0][0]
return self._expected_checkstring
d.addCallback(_then)
d.addCallback(sdmfw.set_checkstring)
d.addCallback(lambda ignored:
sdmfw.get_checkstring())
d.addCallback(lambda checkstring:
self.failUnlessEqual(checkstring, self._expected_checkstring))
d.addCallback(lambda ignored:
sdmfw.finish_publishing())
def _then_again(results):
self.failUnless(results[0])
read = self.ss.remote_slot_readv
2020-08-28 16:41:19 +00:00
self.failUnlessEqual(read(b"si1", [0], [(1, 8)]),
{0: [struct.pack(">Q", 1)]})
2020-08-28 16:41:19 +00:00
self.failUnlessEqual(read(b"si1", [0], [(9, len(data) - 9)]),
{0: [data[9:]]})
d.addCallback(_then_again)
return d
class Stats(unittest.TestCase):
def setUp(self):
self.sparent = LoggingServiceParent()
self._lease_secret = itertools.count()
def tearDown(self):
return self.sparent.stopService()
def workdir(self, name):
basedir = os.path.join("storage", "Server", name)
return basedir
def create(self, name):
workdir = self.workdir(name)
2020-08-27 19:36:54 +00:00
ss = StorageServer(workdir, b"\x00" * 20)
ss.setServiceParent(self.sparent)
return ss
def test_latencies(self):
ss = self.create("test_latencies")
for i in range(10000):
ss.add_latency("allocate", 1.0 * i)
for i in range(1000):
ss.add_latency("renew", 1.0 * i)
for i in range(20):
ss.add_latency("write", 1.0 * i)
for i in range(10):
ss.add_latency("cancel", 2.0 * i)
ss.add_latency("get", 5.0)
output = ss.get_latencies()
self.failUnlessEqual(sorted(output.keys()),
sorted(["allocate", "renew", "cancel", "write", "get"]))
self.failUnlessEqual(len(ss.latencies["allocate"]), 1000)
self.failUnless(abs(output["allocate"]["mean"] - 9500) < 1, output)
self.failUnless(abs(output["allocate"]["01_0_percentile"] - 9010) < 1, output)
self.failUnless(abs(output["allocate"]["10_0_percentile"] - 9100) < 1, output)
self.failUnless(abs(output["allocate"]["50_0_percentile"] - 9500) < 1, output)
self.failUnless(abs(output["allocate"]["90_0_percentile"] - 9900) < 1, output)
self.failUnless(abs(output["allocate"]["95_0_percentile"] - 9950) < 1, output)
self.failUnless(abs(output["allocate"]["99_0_percentile"] - 9990) < 1, output)
self.failUnless(abs(output["allocate"]["99_9_percentile"] - 9999) < 1, output)
self.failUnlessEqual(len(ss.latencies["renew"]), 1000)
self.failUnless(abs(output["renew"]["mean"] - 500) < 1, output)
self.failUnless(abs(output["renew"]["01_0_percentile"] - 10) < 1, output)
self.failUnless(abs(output["renew"]["10_0_percentile"] - 100) < 1, output)
self.failUnless(abs(output["renew"]["50_0_percentile"] - 500) < 1, output)
self.failUnless(abs(output["renew"]["90_0_percentile"] - 900) < 1, output)
self.failUnless(abs(output["renew"]["95_0_percentile"] - 950) < 1, output)
self.failUnless(abs(output["renew"]["99_0_percentile"] - 990) < 1, output)
self.failUnless(abs(output["renew"]["99_9_percentile"] - 999) < 1, output)
self.failUnlessEqual(len(ss.latencies["write"]), 20)
self.failUnless(abs(output["write"]["mean"] - 9) < 1, output)
self.failUnless(output["write"]["01_0_percentile"] is None, output)
self.failUnless(abs(output["write"]["10_0_percentile"] - 2) < 1, output)
self.failUnless(abs(output["write"]["50_0_percentile"] - 10) < 1, output)
self.failUnless(abs(output["write"]["90_0_percentile"] - 18) < 1, output)
self.failUnless(abs(output["write"]["95_0_percentile"] - 19) < 1, output)
self.failUnless(output["write"]["99_0_percentile"] is None, output)
self.failUnless(output["write"]["99_9_percentile"] is None, output)
self.failUnlessEqual(len(ss.latencies["cancel"]), 10)
self.failUnless(abs(output["cancel"]["mean"] - 9) < 1, output)
self.failUnless(output["cancel"]["01_0_percentile"] is None, output)
self.failUnless(abs(output["cancel"]["10_0_percentile"] - 2) < 1, output)
self.failUnless(abs(output["cancel"]["50_0_percentile"] - 10) < 1, output)
self.failUnless(abs(output["cancel"]["90_0_percentile"] - 18) < 1, output)
self.failUnless(output["cancel"]["95_0_percentile"] is None, output)
self.failUnless(output["cancel"]["99_0_percentile"] is None, output)
self.failUnless(output["cancel"]["99_9_percentile"] is None, output)
self.failUnlessEqual(len(ss.latencies["get"]), 1)
self.failUnless(output["get"]["mean"] is None, output)
self.failUnless(output["get"]["01_0_percentile"] is None, output)
self.failUnless(output["get"]["10_0_percentile"] is None, output)
self.failUnless(output["get"]["50_0_percentile"] is None, output)
self.failUnless(output["get"]["90_0_percentile"] is None, output)
self.failUnless(output["get"]["95_0_percentile"] is None, output)
self.failUnless(output["get"]["99_0_percentile"] is None, output)
self.failUnless(output["get"]["99_9_percentile"] is None, output)