2007-11-01 22:15:29 +00:00
|
|
|
|
2008-04-11 01:44:06 +00:00
|
|
|
import struct
|
2008-03-10 23:14:08 +00:00
|
|
|
from cStringIO import StringIO
|
2007-11-01 22:15:29 +00:00
|
|
|
from twisted.trial import unittest
|
2008-03-11 05:15:43 +00:00
|
|
|
from twisted.internet import defer, reactor
|
|
|
|
from twisted.python import failure
|
2008-04-11 21:31:16 +00:00
|
|
|
from allmydata import uri, download
|
2008-04-05 00:09:26 +00:00
|
|
|
from allmydata.util import base32
|
2008-03-11 06:16:28 +00:00
|
|
|
from allmydata.util.idlib import shortnodeid_b2a
|
2007-12-04 21:32:04 +00:00
|
|
|
from allmydata.util.hashutil import tagged_hash
|
2008-04-15 23:08:32 +00:00
|
|
|
from allmydata.encode import NotEnoughSharesError
|
2008-04-11 01:44:06 +00:00
|
|
|
from allmydata.interfaces import IURI, IMutableFileURI, IUploadable
|
2008-04-05 00:09:26 +00:00
|
|
|
from foolscap.eventual import eventually, fireEventually
|
2008-03-11 05:15:43 +00:00
|
|
|
from foolscap.logging import log
|
2007-11-05 07:38:07 +00:00
|
|
|
import sha
|
2007-11-01 22:15:29 +00:00
|
|
|
|
2008-04-11 21:31:16 +00:00
|
|
|
from allmydata.mutable.node import MutableFileNode
|
|
|
|
from allmydata.mutable.common import DictOfSets, ResponseCache, \
|
2008-04-16 22:22:30 +00:00
|
|
|
MODE_CHECK, MODE_ANYTHING, MODE_WRITE, MODE_READ, UnrecoverableFileError
|
2008-04-11 21:31:16 +00:00
|
|
|
from allmydata.mutable.retrieve import Retrieve
|
|
|
|
from allmydata.mutable.publish import Publish
|
|
|
|
from allmydata.mutable.servermap import ServerMap, ServermapUpdater
|
|
|
|
from allmydata.mutable.layout import unpack_header, unpack_share
|
|
|
|
|
2008-04-11 01:44:06 +00:00
|
|
|
# this "FastMutableFileNode" exists solely to speed up tests by using smaller
|
|
|
|
# public/private keys. Once we switch to fast DSA-based keys, we can get rid
|
|
|
|
# of this.
|
2007-12-05 06:01:37 +00:00
|
|
|
|
2008-04-11 21:31:16 +00:00
|
|
|
class FastMutableFileNode(MutableFileNode):
|
2008-04-11 01:44:06 +00:00
|
|
|
SIGNATURE_KEY_SIZE = 522
|
2007-12-03 21:52:42 +00:00
|
|
|
|
2008-04-11 01:44:06 +00:00
|
|
|
# this "FakeStorage" exists to put the share data in RAM and avoid using real
|
|
|
|
# network connections, both to speed up the tests and to reduce the amount of
|
|
|
|
# non-mutable.py code being exercised.
|
2007-11-01 22:15:29 +00:00
|
|
|
|
2008-03-10 23:14:08 +00:00
|
|
|
class FakeStorage:
|
|
|
|
# this class replaces the collection of storage servers, allowing the
|
|
|
|
# tests to examine and manipulate the published shares. It also lets us
|
|
|
|
# control the order in which read queries are answered, to exercise more
|
2008-04-11 21:31:16 +00:00
|
|
|
# of the error-handling code in Retrieve .
|
2008-03-11 00:46:52 +00:00
|
|
|
#
|
|
|
|
# Note that we ignore the storage index: this FakeStorage instance can
|
|
|
|
# only be used for a single storage index.
|
|
|
|
|
2008-03-10 23:14:08 +00:00
|
|
|
|
|
|
|
def __init__(self):
|
|
|
|
self._peers = {}
|
2008-03-11 05:15:43 +00:00
|
|
|
# _sequence is used to cause the responses to occur in a specific
|
|
|
|
# order. If it is in use, then we will defer queries instead of
|
|
|
|
# answering them right away, accumulating the Deferreds in a dict. We
|
|
|
|
# don't know exactly how many queries we'll get, so exactly one
|
|
|
|
# second after the first query arrives, we will release them all (in
|
|
|
|
# order).
|
|
|
|
self._sequence = None
|
|
|
|
self._pending = {}
|
2008-03-10 23:14:08 +00:00
|
|
|
|
|
|
|
def read(self, peerid, storage_index):
|
|
|
|
shares = self._peers.get(peerid, {})
|
2008-03-11 05:15:43 +00:00
|
|
|
if self._sequence is None:
|
2008-04-11 01:44:06 +00:00
|
|
|
return defer.succeed(shares)
|
2008-03-11 05:15:43 +00:00
|
|
|
d = defer.Deferred()
|
|
|
|
if not self._pending:
|
|
|
|
reactor.callLater(1.0, self._fire_readers)
|
|
|
|
self._pending[peerid] = (d, shares)
|
|
|
|
return d
|
|
|
|
|
|
|
|
def _fire_readers(self):
|
|
|
|
pending = self._pending
|
|
|
|
self._pending = {}
|
|
|
|
extra = []
|
|
|
|
for peerid in self._sequence:
|
|
|
|
if peerid in pending:
|
|
|
|
d, shares = pending.pop(peerid)
|
|
|
|
eventually(d.callback, shares)
|
2008-03-11 06:16:28 +00:00
|
|
|
for (d, shares) in pending.values():
|
2008-03-11 05:15:43 +00:00
|
|
|
eventually(d.callback, shares)
|
2008-03-10 23:14:08 +00:00
|
|
|
|
|
|
|
def write(self, peerid, storage_index, shnum, offset, data):
|
|
|
|
if peerid not in self._peers:
|
|
|
|
self._peers[peerid] = {}
|
|
|
|
shares = self._peers[peerid]
|
|
|
|
f = StringIO()
|
|
|
|
f.write(shares.get(shnum, ""))
|
|
|
|
f.seek(offset)
|
|
|
|
f.write(data)
|
|
|
|
shares[shnum] = f.getvalue()
|
|
|
|
|
|
|
|
|
2008-04-11 01:44:06 +00:00
|
|
|
class FakeStorageServer:
|
|
|
|
def __init__(self, peerid, storage):
|
|
|
|
self.peerid = peerid
|
|
|
|
self.storage = storage
|
|
|
|
self.queries = 0
|
|
|
|
def callRemote(self, methname, *args, **kwargs):
|
|
|
|
def _call():
|
|
|
|
meth = getattr(self, methname)
|
|
|
|
return meth(*args, **kwargs)
|
2008-04-05 00:09:26 +00:00
|
|
|
d = fireEventually()
|
2008-04-11 01:44:06 +00:00
|
|
|
d.addCallback(lambda res: _call())
|
|
|
|
return d
|
|
|
|
|
|
|
|
def slot_readv(self, storage_index, shnums, readv):
|
|
|
|
d = self.storage.read(self.peerid, storage_index)
|
|
|
|
def _read(shares):
|
|
|
|
response = {}
|
|
|
|
for shnum in shares:
|
|
|
|
if shnums and shnum not in shnums:
|
|
|
|
continue
|
|
|
|
vector = response[shnum] = []
|
|
|
|
for (offset, length) in readv:
|
|
|
|
assert isinstance(offset, (int, long)), offset
|
|
|
|
assert isinstance(length, (int, long)), length
|
|
|
|
vector.append(shares[shnum][offset:offset+length])
|
|
|
|
return response
|
|
|
|
d.addCallback(_read)
|
2008-04-05 00:09:26 +00:00
|
|
|
return d
|
2007-11-05 07:38:07 +00:00
|
|
|
|
2008-04-11 01:44:06 +00:00
|
|
|
def slot_testv_and_readv_and_writev(self, storage_index, secrets,
|
|
|
|
tw_vectors, read_vector):
|
2007-11-06 04:29:47 +00:00
|
|
|
# always-pass: parrot the test vectors back to them.
|
|
|
|
readv = {}
|
2008-03-10 23:14:08 +00:00
|
|
|
for shnum, (testv, writev, new_length) in tw_vectors.items():
|
2007-11-06 04:29:47 +00:00
|
|
|
for (offset, length, op, specimen) in testv:
|
|
|
|
assert op in ("le", "eq", "ge")
|
2008-04-11 01:44:06 +00:00
|
|
|
# TODO: this isn't right, the read is controlled by read_vector,
|
|
|
|
# not by testv
|
2007-11-06 04:29:47 +00:00
|
|
|
readv[shnum] = [ specimen
|
|
|
|
for (offset, length, op, specimen)
|
|
|
|
in testv ]
|
2008-03-10 23:14:08 +00:00
|
|
|
for (offset, data) in writev:
|
2008-04-11 01:44:06 +00:00
|
|
|
self.storage.write(self.peerid, storage_index, shnum,
|
|
|
|
offset, data)
|
2007-11-06 04:29:47 +00:00
|
|
|
answer = (True, readv)
|
2008-04-11 01:44:06 +00:00
|
|
|
return fireEventually(answer)
|
2007-11-05 07:38:07 +00:00
|
|
|
|
2008-03-10 23:14:08 +00:00
|
|
|
|
2008-04-11 01:44:06 +00:00
|
|
|
# our "FakeClient" has just enough functionality of the real Client to let
|
|
|
|
# the tests run.
|
2007-11-01 22:15:29 +00:00
|
|
|
|
2007-12-03 21:52:42 +00:00
|
|
|
class FakeClient:
|
2008-04-11 01:44:06 +00:00
|
|
|
mutable_file_node_class = FastMutableFileNode
|
|
|
|
|
2007-11-05 07:38:07 +00:00
|
|
|
def __init__(self, num_peers=10):
|
2008-04-11 01:44:06 +00:00
|
|
|
self._storage = FakeStorage()
|
2007-11-05 07:38:07 +00:00
|
|
|
self._num_peers = num_peers
|
2007-11-07 01:49:59 +00:00
|
|
|
self._peerids = [tagged_hash("peerid", "%d" % i)[:20]
|
2007-11-05 07:38:07 +00:00
|
|
|
for i in range(self._num_peers)]
|
2008-04-11 01:44:06 +00:00
|
|
|
self._connections = dict([(peerid, FakeStorageServer(peerid,
|
|
|
|
self._storage))
|
|
|
|
for peerid in self._peerids])
|
2007-12-19 07:06:53 +00:00
|
|
|
self.nodeid = "fakenodeid"
|
|
|
|
|
2007-12-03 22:25:14 +00:00
|
|
|
def log(self, msg, **kw):
|
|
|
|
return log.msg(msg, **kw)
|
2007-11-01 22:15:29 +00:00
|
|
|
|
2007-11-06 04:51:08 +00:00
|
|
|
def get_renewal_secret(self):
|
|
|
|
return "I hereby permit you to renew my files"
|
|
|
|
def get_cancel_secret(self):
|
|
|
|
return "I hereby permit you to cancel my leases"
|
|
|
|
|
2008-01-14 21:55:59 +00:00
|
|
|
def create_mutable_file(self, contents=""):
|
2008-04-11 01:44:06 +00:00
|
|
|
n = self.mutable_file_node_class(self)
|
2008-01-14 21:55:59 +00:00
|
|
|
d = n.create(contents)
|
2007-11-01 22:15:29 +00:00
|
|
|
d.addCallback(lambda res: n)
|
|
|
|
return d
|
2007-11-09 09:54:51 +00:00
|
|
|
|
2008-03-13 01:00:43 +00:00
|
|
|
def notify_retrieve(self, r):
|
|
|
|
pass
|
2008-04-17 01:06:54 +00:00
|
|
|
def notify_publish(self, p):
|
|
|
|
pass
|
2008-04-17 02:05:41 +00:00
|
|
|
def notify_mapupdate(self, u):
|
|
|
|
pass
|
2008-03-13 01:00:43 +00:00
|
|
|
|
2007-11-09 09:54:51 +00:00
|
|
|
def create_node_from_uri(self, u):
|
|
|
|
u = IURI(u)
|
2007-12-04 04:37:54 +00:00
|
|
|
assert IMutableFileURI.providedBy(u), u
|
2008-04-11 01:44:06 +00:00
|
|
|
res = self.mutable_file_node_class(self).init_from_uri(u)
|
2007-12-03 21:52:42 +00:00
|
|
|
return res
|
2007-11-01 22:15:29 +00:00
|
|
|
|
2008-02-05 20:05:13 +00:00
|
|
|
def get_permuted_peers(self, service_name, key):
|
2007-11-05 07:38:07 +00:00
|
|
|
"""
|
2008-02-05 20:05:13 +00:00
|
|
|
@return: list of (peerid, connection,)
|
2007-11-05 07:38:07 +00:00
|
|
|
"""
|
|
|
|
results = []
|
2008-04-11 01:44:06 +00:00
|
|
|
for (peerid, connection) in self._connections.items():
|
2007-11-05 07:38:07 +00:00
|
|
|
assert isinstance(peerid, str)
|
2007-12-04 01:10:01 +00:00
|
|
|
permuted = sha.new(key + peerid).digest()
|
2007-11-05 07:38:07 +00:00
|
|
|
results.append((permuted, peerid, connection))
|
|
|
|
results.sort()
|
2008-02-05 20:05:13 +00:00
|
|
|
results = [ (r[1],r[2]) for r in results]
|
2007-11-05 07:38:07 +00:00
|
|
|
return results
|
2007-11-01 22:15:29 +00:00
|
|
|
|
2008-01-14 21:55:59 +00:00
|
|
|
def upload(self, uploadable):
|
2007-12-04 04:37:54 +00:00
|
|
|
assert IUploadable.providedBy(uploadable)
|
|
|
|
d = uploadable.get_size()
|
|
|
|
d.addCallback(lambda length: uploadable.read(length))
|
|
|
|
#d.addCallback(self.create_mutable_file)
|
|
|
|
def _got_data(datav):
|
|
|
|
data = "".join(datav)
|
2008-04-11 01:44:06 +00:00
|
|
|
#newnode = FastMutableFileNode(self)
|
2007-12-04 04:37:54 +00:00
|
|
|
return uri.LiteralFileURI(data)
|
|
|
|
d.addCallback(_got_data)
|
|
|
|
return d
|
|
|
|
|
2008-03-10 22:44:05 +00:00
|
|
|
|
2008-04-16 21:49:47 +00:00
|
|
|
def flip_bit(original, byte_offset):
|
|
|
|
return (original[:byte_offset] +
|
|
|
|
chr(ord(original[byte_offset]) ^ 0x01) +
|
|
|
|
original[byte_offset+1:])
|
|
|
|
|
|
|
|
def corrupt(res, s, offset, shnums_to_corrupt=None):
|
|
|
|
# if shnums_to_corrupt is None, corrupt all shares. Otherwise it is a
|
|
|
|
# list of shnums to corrupt.
|
|
|
|
for peerid in s._peers:
|
|
|
|
shares = s._peers[peerid]
|
|
|
|
for shnum in shares:
|
|
|
|
if (shnums_to_corrupt is not None
|
|
|
|
and shnum not in shnums_to_corrupt):
|
|
|
|
continue
|
|
|
|
data = shares[shnum]
|
|
|
|
(version,
|
|
|
|
seqnum,
|
|
|
|
root_hash,
|
|
|
|
IV,
|
|
|
|
k, N, segsize, datalen,
|
|
|
|
o) = unpack_header(data)
|
|
|
|
if isinstance(offset, tuple):
|
|
|
|
offset1, offset2 = offset
|
|
|
|
else:
|
|
|
|
offset1 = offset
|
|
|
|
offset2 = 0
|
|
|
|
if offset1 == "pubkey":
|
|
|
|
real_offset = 107
|
|
|
|
elif offset1 in o:
|
|
|
|
real_offset = o[offset1]
|
|
|
|
else:
|
|
|
|
real_offset = offset1
|
|
|
|
real_offset = int(real_offset) + offset2
|
|
|
|
assert isinstance(real_offset, int), offset
|
|
|
|
shares[shnum] = flip_bit(data, real_offset)
|
|
|
|
return res
|
|
|
|
|
2007-11-01 22:15:29 +00:00
|
|
|
class Filenode(unittest.TestCase):
|
|
|
|
def setUp(self):
|
2007-12-03 21:52:42 +00:00
|
|
|
self.client = FakeClient()
|
2007-11-01 22:15:29 +00:00
|
|
|
|
|
|
|
def test_create(self):
|
2008-01-14 21:55:59 +00:00
|
|
|
d = self.client.create_mutable_file()
|
2007-11-01 22:15:29 +00:00
|
|
|
def _created(n):
|
2008-04-11 01:44:06 +00:00
|
|
|
self.failUnless(isinstance(n, FastMutableFileNode))
|
|
|
|
peer0 = self.client._peerids[0]
|
|
|
|
shnums = self.client._storage._peers[peer0].keys()
|
|
|
|
self.failUnlessEqual(len(shnums), 1)
|
|
|
|
d.addCallback(_created)
|
|
|
|
return d
|
|
|
|
|
|
|
|
def test_upload_and_download(self):
|
|
|
|
d = self.client.create_mutable_file()
|
|
|
|
def _created(n):
|
|
|
|
d = defer.succeed(None)
|
|
|
|
d.addCallback(lambda res: n.update_servermap())
|
|
|
|
d.addCallback(lambda smap: smap.dump(StringIO()))
|
|
|
|
d.addCallback(lambda sio:
|
|
|
|
self.failUnless("3-of-10" in sio.getvalue()))
|
|
|
|
d.addCallback(lambda res: n.overwrite("contents 1"))
|
2007-11-01 22:15:29 +00:00
|
|
|
d.addCallback(lambda res: self.failUnlessIdentical(res, None))
|
|
|
|
d.addCallback(lambda res: n.download_to_data())
|
|
|
|
d.addCallback(lambda res: self.failUnlessEqual(res, "contents 1"))
|
2008-03-13 01:00:43 +00:00
|
|
|
d.addCallback(lambda res: n.overwrite("contents 2"))
|
2007-11-01 22:15:29 +00:00
|
|
|
d.addCallback(lambda res: n.download_to_data())
|
|
|
|
d.addCallback(lambda res: self.failUnlessEqual(res, "contents 2"))
|
2008-03-04 22:11:40 +00:00
|
|
|
d.addCallback(lambda res: n.download(download.Data()))
|
|
|
|
d.addCallback(lambda res: self.failUnlessEqual(res, "contents 2"))
|
2008-03-13 01:00:43 +00:00
|
|
|
d.addCallback(lambda res: n.update("contents 3"))
|
|
|
|
d.addCallback(lambda res: n.download_to_data())
|
|
|
|
d.addCallback(lambda res: self.failUnlessEqual(res, "contents 3"))
|
2007-11-01 22:15:29 +00:00
|
|
|
return d
|
|
|
|
d.addCallback(_created)
|
|
|
|
return d
|
|
|
|
|
|
|
|
def test_create_with_initial_contents(self):
|
|
|
|
d = self.client.create_mutable_file("contents 1")
|
|
|
|
def _created(n):
|
|
|
|
d = n.download_to_data()
|
|
|
|
d.addCallback(lambda res: self.failUnlessEqual(res, "contents 1"))
|
2008-03-13 01:00:43 +00:00
|
|
|
d.addCallback(lambda res: n.overwrite("contents 2"))
|
2007-11-01 22:15:29 +00:00
|
|
|
d.addCallback(lambda res: n.download_to_data())
|
|
|
|
d.addCallback(lambda res: self.failUnlessEqual(res, "contents 2"))
|
|
|
|
return d
|
|
|
|
d.addCallback(_created)
|
|
|
|
return d
|
|
|
|
|
2008-04-11 01:44:06 +00:00
|
|
|
def test_upload_and_download_full_size_keys(self):
|
2008-04-11 21:31:16 +00:00
|
|
|
self.client.mutable_file_node_class = MutableFileNode
|
2008-04-11 01:44:06 +00:00
|
|
|
d = self.client.create_mutable_file()
|
|
|
|
def _created(n):
|
|
|
|
d = defer.succeed(None)
|
|
|
|
d.addCallback(lambda res: n.update_servermap())
|
|
|
|
d.addCallback(lambda smap: smap.dump(StringIO()))
|
|
|
|
d.addCallback(lambda sio:
|
|
|
|
self.failUnless("3-of-10" in sio.getvalue()))
|
|
|
|
d.addCallback(lambda res: n.overwrite("contents 1"))
|
|
|
|
d.addCallback(lambda res: self.failUnlessIdentical(res, None))
|
|
|
|
d.addCallback(lambda res: n.download_to_data())
|
|
|
|
d.addCallback(lambda res: self.failUnlessEqual(res, "contents 1"))
|
|
|
|
d.addCallback(lambda res: n.overwrite("contents 2"))
|
|
|
|
d.addCallback(lambda res: n.download_to_data())
|
|
|
|
d.addCallback(lambda res: self.failUnlessEqual(res, "contents 2"))
|
|
|
|
d.addCallback(lambda res: n.download(download.Data()))
|
|
|
|
d.addCallback(lambda res: self.failUnlessEqual(res, "contents 2"))
|
|
|
|
d.addCallback(lambda res: n.update("contents 3"))
|
|
|
|
d.addCallback(lambda res: n.download_to_data())
|
|
|
|
d.addCallback(lambda res: self.failUnlessEqual(res, "contents 3"))
|
|
|
|
return d
|
|
|
|
d.addCallback(_created)
|
|
|
|
return d
|
|
|
|
|
2007-11-07 01:56:39 +00:00
|
|
|
|
2008-04-11 21:31:16 +00:00
|
|
|
class MakeShares(unittest.TestCase):
|
2007-11-03 03:51:39 +00:00
|
|
|
def test_encrypt(self):
|
2007-12-03 21:52:42 +00:00
|
|
|
c = FakeClient()
|
2008-04-11 01:44:06 +00:00
|
|
|
fn = FastMutableFileNode(c)
|
2007-11-03 03:51:39 +00:00
|
|
|
CONTENTS = "some initial contents"
|
2008-04-11 01:44:06 +00:00
|
|
|
d = fn.create(CONTENTS)
|
|
|
|
def _created(res):
|
2008-04-11 21:31:16 +00:00
|
|
|
p = Publish(fn, None)
|
2008-04-11 01:44:06 +00:00
|
|
|
p.salt = "SALT" * 4
|
|
|
|
p.readkey = "\x00" * 16
|
|
|
|
p.newdata = CONTENTS
|
|
|
|
p.required_shares = 3
|
|
|
|
p.total_shares = 10
|
|
|
|
p.setup_encoding_parameters()
|
|
|
|
return p._encrypt_and_encode()
|
|
|
|
d.addCallback(_created)
|
|
|
|
def _done(shares_and_shareids):
|
|
|
|
(shares, share_ids) = shares_and_shareids
|
2007-11-03 03:51:39 +00:00
|
|
|
self.failUnlessEqual(len(shares), 10)
|
|
|
|
for sh in shares:
|
|
|
|
self.failUnless(isinstance(sh, str))
|
|
|
|
self.failUnlessEqual(len(sh), 7)
|
|
|
|
self.failUnlessEqual(len(share_ids), 10)
|
|
|
|
d.addCallback(_done)
|
|
|
|
return d
|
|
|
|
|
|
|
|
def test_generate(self):
|
2007-12-03 21:52:42 +00:00
|
|
|
c = FakeClient()
|
2008-04-11 01:44:06 +00:00
|
|
|
fn = FastMutableFileNode(c)
|
2007-11-03 03:51:39 +00:00
|
|
|
CONTENTS = "some initial contents"
|
2008-01-14 21:55:59 +00:00
|
|
|
fn.create(CONTENTS)
|
2007-11-03 03:51:39 +00:00
|
|
|
p = mutable.Publish(fn)
|
2008-04-05 00:09:26 +00:00
|
|
|
r = mutable.Retrieve(fn)
|
2007-11-03 03:51:39 +00:00
|
|
|
# make some fake shares
|
|
|
|
shares_and_ids = ( ["%07d" % i for i in range(10)], range(10) )
|
2007-11-08 09:46:27 +00:00
|
|
|
target_info = None
|
|
|
|
p._privkey = FakePrivKey(0)
|
|
|
|
p._encprivkey = "encprivkey"
|
|
|
|
p._pubkey = FakePubKey(0)
|
2007-11-03 03:51:39 +00:00
|
|
|
d = defer.maybeDeferred(p._generate_shares,
|
|
|
|
(shares_and_ids,
|
|
|
|
3, 10,
|
|
|
|
21, # segsize
|
|
|
|
len(CONTENTS),
|
2007-11-08 09:46:27 +00:00
|
|
|
target_info),
|
2007-11-03 03:51:39 +00:00
|
|
|
3, # seqnum
|
2007-11-08 09:46:27 +00:00
|
|
|
"IV"*8)
|
|
|
|
def _done( (seqnum, root_hash, final_shares, target_info2) ):
|
2007-11-03 03:51:39 +00:00
|
|
|
self.failUnlessEqual(seqnum, 3)
|
|
|
|
self.failUnlessEqual(len(root_hash), 32)
|
|
|
|
self.failUnless(isinstance(final_shares, dict))
|
|
|
|
self.failUnlessEqual(len(final_shares), 10)
|
|
|
|
self.failUnlessEqual(sorted(final_shares.keys()), range(10))
|
|
|
|
for i,sh in final_shares.items():
|
|
|
|
self.failUnless(isinstance(sh, str))
|
2007-11-03 05:28:31 +00:00
|
|
|
# feed the share through the unpacker as a sanity-check
|
2008-04-11 21:31:16 +00:00
|
|
|
pieces = unpack_share(sh)
|
2007-11-06 22:04:46 +00:00
|
|
|
(u_seqnum, u_root_hash, IV, k, N, segsize, datalen,
|
2007-11-03 05:28:31 +00:00
|
|
|
pubkey, signature, share_hash_chain, block_hash_tree,
|
2007-11-06 22:04:46 +00:00
|
|
|
share_data, enc_privkey) = pieces
|
2007-11-03 05:28:31 +00:00
|
|
|
self.failUnlessEqual(u_seqnum, 3)
|
|
|
|
self.failUnlessEqual(u_root_hash, root_hash)
|
|
|
|
self.failUnlessEqual(k, 3)
|
|
|
|
self.failUnlessEqual(N, 10)
|
|
|
|
self.failUnlessEqual(segsize, 21)
|
|
|
|
self.failUnlessEqual(datalen, len(CONTENTS))
|
2008-04-11 01:44:06 +00:00
|
|
|
self.failUnlessEqual(pubkey, p._pubkey.serialize())
|
2007-11-06 22:04:46 +00:00
|
|
|
sig_material = struct.pack(">BQ32s16s BBQQ",
|
2008-04-11 01:44:06 +00:00
|
|
|
0, p._new_seqnum, root_hash, IV,
|
2007-11-03 05:28:31 +00:00
|
|
|
k, N, segsize, datalen)
|
2008-04-11 01:44:06 +00:00
|
|
|
self.failUnless(p._pubkey.verify(sig_material, signature))
|
|
|
|
#self.failUnlessEqual(signature, p._privkey.sign(sig_material))
|
2007-11-14 06:08:15 +00:00
|
|
|
self.failUnless(isinstance(share_hash_chain, dict))
|
2007-11-03 05:28:31 +00:00
|
|
|
self.failUnlessEqual(len(share_hash_chain), 4) # ln2(10)++
|
2007-11-14 06:08:15 +00:00
|
|
|
for shnum,share_hash in share_hash_chain.items():
|
|
|
|
self.failUnless(isinstance(shnum, int))
|
|
|
|
self.failUnless(isinstance(share_hash, str))
|
|
|
|
self.failUnlessEqual(len(share_hash), 32)
|
2007-11-03 05:28:31 +00:00
|
|
|
self.failUnless(isinstance(block_hash_tree, list))
|
|
|
|
self.failUnlessEqual(len(block_hash_tree), 1) # very small tree
|
2008-04-11 01:44:06 +00:00
|
|
|
self.failUnlessEqual(IV, "SALT"*4)
|
2007-11-03 05:28:31 +00:00
|
|
|
self.failUnlessEqual(len(share_data), len("%07d" % 1))
|
2008-04-11 01:44:06 +00:00
|
|
|
self.failUnlessEqual(enc_privkey, fn.get_encprivkey())
|
|
|
|
d.addCallback(_generated)
|
2007-11-05 07:38:07 +00:00
|
|
|
return d
|
|
|
|
|
2008-04-11 01:44:06 +00:00
|
|
|
# TODO: when we publish to 20 peers, we should get one share per peer on 10
|
|
|
|
# when we publish to 3 peers, we should get either 3 or 4 shares per peer
|
2008-04-15 23:08:32 +00:00
|
|
|
# when we publish to zero peers, we should get a NotEnoughSharesError
|
2008-03-11 00:46:52 +00:00
|
|
|
|
2008-04-11 01:44:06 +00:00
|
|
|
class Servermap(unittest.TestCase):
|
2008-04-05 00:09:26 +00:00
|
|
|
def setUp(self):
|
|
|
|
# publish a file and create shares, which can then be manipulated
|
|
|
|
# later.
|
|
|
|
num_peers = 20
|
|
|
|
self._client = FakeClient(num_peers)
|
2008-04-11 01:44:06 +00:00
|
|
|
self._storage = self._client._storage
|
|
|
|
d = self._client.create_mutable_file("New contents go here")
|
|
|
|
def _created(node):
|
|
|
|
self._fn = node
|
2008-04-05 00:09:26 +00:00
|
|
|
d.addCallback(_created)
|
|
|
|
return d
|
|
|
|
|
2008-04-11 21:31:16 +00:00
|
|
|
def make_servermap(self, mode=MODE_CHECK):
|
|
|
|
smu = ServermapUpdater(self._fn, ServerMap(), mode)
|
2008-04-05 00:09:26 +00:00
|
|
|
d = smu.update()
|
|
|
|
return d
|
|
|
|
|
2008-04-11 21:31:16 +00:00
|
|
|
def update_servermap(self, oldmap, mode=MODE_CHECK):
|
|
|
|
smu = ServermapUpdater(self._fn, oldmap, mode)
|
2008-04-05 00:09:26 +00:00
|
|
|
d = smu.update()
|
|
|
|
return d
|
|
|
|
|
|
|
|
def failUnlessOneRecoverable(self, sm, num_shares):
|
|
|
|
self.failUnlessEqual(len(sm.recoverable_versions()), 1)
|
|
|
|
self.failUnlessEqual(len(sm.unrecoverable_versions()), 0)
|
|
|
|
best = sm.best_recoverable_version()
|
|
|
|
self.failIfEqual(best, None)
|
|
|
|
self.failUnlessEqual(sm.recoverable_versions(), set([best]))
|
|
|
|
self.failUnlessEqual(len(sm.shares_available()), 1)
|
|
|
|
self.failUnlessEqual(sm.shares_available()[best], (num_shares, 3))
|
|
|
|
return sm
|
|
|
|
|
|
|
|
def test_basic(self):
|
|
|
|
d = defer.succeed(None)
|
|
|
|
ms = self.make_servermap
|
|
|
|
us = self.update_servermap
|
|
|
|
|
2008-04-11 21:31:16 +00:00
|
|
|
d.addCallback(lambda res: ms(mode=MODE_CHECK))
|
2008-04-05 00:09:26 +00:00
|
|
|
d.addCallback(lambda sm: self.failUnlessOneRecoverable(sm, 10))
|
2008-04-11 21:31:16 +00:00
|
|
|
d.addCallback(lambda res: ms(mode=MODE_WRITE))
|
2008-04-05 00:09:26 +00:00
|
|
|
d.addCallback(lambda sm: self.failUnlessOneRecoverable(sm, 10))
|
2008-04-16 22:22:30 +00:00
|
|
|
d.addCallback(lambda res: ms(mode=MODE_READ))
|
2008-04-05 00:09:26 +00:00
|
|
|
# this more stops at k+epsilon, and epsilon=k, so 6 shares
|
|
|
|
d.addCallback(lambda sm: self.failUnlessOneRecoverable(sm, 6))
|
2008-04-11 21:31:16 +00:00
|
|
|
d.addCallback(lambda res: ms(mode=MODE_ANYTHING))
|
2008-04-05 00:09:26 +00:00
|
|
|
# this mode stops at 'k' shares
|
|
|
|
d.addCallback(lambda sm: self.failUnlessOneRecoverable(sm, 3))
|
|
|
|
|
|
|
|
# and can we re-use the same servermap? Note that these are sorted in
|
|
|
|
# increasing order of number of servers queried, since once a server
|
|
|
|
# gets into the servermap, we'll always ask it for an update.
|
|
|
|
d.addCallback(lambda sm: self.failUnlessOneRecoverable(sm, 3))
|
2008-04-16 22:22:30 +00:00
|
|
|
d.addCallback(lambda sm: us(sm, mode=MODE_READ))
|
2008-04-05 00:09:26 +00:00
|
|
|
d.addCallback(lambda sm: self.failUnlessOneRecoverable(sm, 6))
|
2008-04-11 21:31:16 +00:00
|
|
|
d.addCallback(lambda sm: us(sm, mode=MODE_WRITE))
|
|
|
|
d.addCallback(lambda sm: us(sm, mode=MODE_CHECK))
|
2008-04-05 00:09:26 +00:00
|
|
|
d.addCallback(lambda sm: self.failUnlessOneRecoverable(sm, 10))
|
2008-04-11 21:31:16 +00:00
|
|
|
d.addCallback(lambda sm: us(sm, mode=MODE_ANYTHING))
|
2008-04-05 00:09:26 +00:00
|
|
|
d.addCallback(lambda sm: self.failUnlessOneRecoverable(sm, 10))
|
|
|
|
|
|
|
|
return d
|
|
|
|
|
2008-04-16 21:49:47 +00:00
|
|
|
def test_mark_bad(self):
|
|
|
|
d = defer.succeed(None)
|
|
|
|
ms = self.make_servermap
|
|
|
|
us = self.update_servermap
|
|
|
|
|
2008-04-16 22:22:30 +00:00
|
|
|
d.addCallback(lambda res: ms(mode=MODE_READ))
|
2008-04-16 21:49:47 +00:00
|
|
|
d.addCallback(lambda sm: self.failUnlessOneRecoverable(sm, 6))
|
|
|
|
def _made_map(sm):
|
|
|
|
v = sm.best_recoverable_version()
|
|
|
|
vm = sm.make_versionmap()
|
|
|
|
shares = list(vm[v])
|
|
|
|
self.failUnlessEqual(len(shares), 6)
|
|
|
|
self._corrupted = set()
|
|
|
|
# mark the first 5 shares as corrupt, then update the servermap.
|
|
|
|
# The map should not have the marked shares it in any more, and
|
|
|
|
# new shares should be found to replace the missing ones.
|
|
|
|
for (shnum, peerid, timestamp) in shares:
|
|
|
|
if shnum < 5:
|
|
|
|
self._corrupted.add( (peerid, shnum) )
|
|
|
|
sm.mark_bad_share(peerid, shnum)
|
|
|
|
return self.update_servermap(sm, MODE_WRITE)
|
|
|
|
d.addCallback(_made_map)
|
|
|
|
def _check_map(sm):
|
|
|
|
# this should find all 5 shares that weren't marked bad
|
|
|
|
v = sm.best_recoverable_version()
|
|
|
|
vm = sm.make_versionmap()
|
|
|
|
shares = list(vm[v])
|
|
|
|
for (peerid, shnum) in self._corrupted:
|
|
|
|
peer_shares = sm.shares_on_peer(peerid)
|
|
|
|
self.failIf(shnum in peer_shares,
|
|
|
|
"%d was in %s" % (shnum, peer_shares))
|
|
|
|
self.failUnlessEqual(len(shares), 5)
|
|
|
|
d.addCallback(_check_map)
|
|
|
|
return d
|
|
|
|
|
2008-04-05 00:09:26 +00:00
|
|
|
def failUnlessNoneRecoverable(self, sm):
|
|
|
|
self.failUnlessEqual(len(sm.recoverable_versions()), 0)
|
|
|
|
self.failUnlessEqual(len(sm.unrecoverable_versions()), 0)
|
|
|
|
best = sm.best_recoverable_version()
|
|
|
|
self.failUnlessEqual(best, None)
|
|
|
|
self.failUnlessEqual(len(sm.shares_available()), 0)
|
|
|
|
|
|
|
|
def test_no_shares(self):
|
2008-04-11 01:44:06 +00:00
|
|
|
self._client._storage._peers = {} # delete all shares
|
2008-04-05 00:09:26 +00:00
|
|
|
ms = self.make_servermap
|
|
|
|
d = defer.succeed(None)
|
|
|
|
|
2008-04-11 21:31:16 +00:00
|
|
|
d.addCallback(lambda res: ms(mode=MODE_CHECK))
|
2008-04-05 00:09:26 +00:00
|
|
|
d.addCallback(lambda sm: self.failUnlessNoneRecoverable(sm))
|
|
|
|
|
2008-04-11 21:31:16 +00:00
|
|
|
d.addCallback(lambda res: ms(mode=MODE_ANYTHING))
|
2008-04-05 00:09:26 +00:00
|
|
|
d.addCallback(lambda sm: self.failUnlessNoneRecoverable(sm))
|
|
|
|
|
2008-04-11 21:31:16 +00:00
|
|
|
d.addCallback(lambda res: ms(mode=MODE_WRITE))
|
2008-04-05 00:09:26 +00:00
|
|
|
d.addCallback(lambda sm: self.failUnlessNoneRecoverable(sm))
|
|
|
|
|
2008-04-16 22:22:30 +00:00
|
|
|
d.addCallback(lambda res: ms(mode=MODE_READ))
|
2008-04-05 00:09:26 +00:00
|
|
|
d.addCallback(lambda sm: self.failUnlessNoneRecoverable(sm))
|
|
|
|
|
|
|
|
return d
|
|
|
|
|
|
|
|
def failUnlessNotQuiteEnough(self, sm):
|
|
|
|
self.failUnlessEqual(len(sm.recoverable_versions()), 0)
|
|
|
|
self.failUnlessEqual(len(sm.unrecoverable_versions()), 1)
|
|
|
|
best = sm.best_recoverable_version()
|
|
|
|
self.failUnlessEqual(best, None)
|
|
|
|
self.failUnlessEqual(len(sm.shares_available()), 1)
|
|
|
|
self.failUnlessEqual(sm.shares_available().values()[0], (2,3) )
|
|
|
|
|
|
|
|
def test_not_quite_enough_shares(self):
|
2008-04-11 01:44:06 +00:00
|
|
|
s = self._client._storage
|
2008-04-05 00:09:26 +00:00
|
|
|
ms = self.make_servermap
|
|
|
|
num_shares = len(s._peers)
|
|
|
|
for peerid in s._peers:
|
|
|
|
s._peers[peerid] = {}
|
|
|
|
num_shares -= 1
|
|
|
|
if num_shares == 2:
|
|
|
|
break
|
|
|
|
# now there ought to be only two shares left
|
|
|
|
assert len([peerid for peerid in s._peers if s._peers[peerid]]) == 2
|
|
|
|
|
|
|
|
d = defer.succeed(None)
|
|
|
|
|
2008-04-11 21:31:16 +00:00
|
|
|
d.addCallback(lambda res: ms(mode=MODE_CHECK))
|
2008-04-05 00:09:26 +00:00
|
|
|
d.addCallback(lambda sm: self.failUnlessNotQuiteEnough(sm))
|
2008-04-11 21:31:16 +00:00
|
|
|
d.addCallback(lambda res: ms(mode=MODE_ANYTHING))
|
2008-04-05 00:09:26 +00:00
|
|
|
d.addCallback(lambda sm: self.failUnlessNotQuiteEnough(sm))
|
2008-04-11 21:31:16 +00:00
|
|
|
d.addCallback(lambda res: ms(mode=MODE_WRITE))
|
2008-04-05 00:09:26 +00:00
|
|
|
d.addCallback(lambda sm: self.failUnlessNotQuiteEnough(sm))
|
2008-04-16 22:22:30 +00:00
|
|
|
d.addCallback(lambda res: ms(mode=MODE_READ))
|
2008-04-05 00:09:26 +00:00
|
|
|
d.addCallback(lambda sm: self.failUnlessNotQuiteEnough(sm))
|
|
|
|
|
|
|
|
return d
|
|
|
|
|
|
|
|
|
2008-03-11 00:46:52 +00:00
|
|
|
|
2008-03-10 23:14:08 +00:00
|
|
|
class Roundtrip(unittest.TestCase):
|
2008-04-05 00:09:26 +00:00
|
|
|
def setUp(self):
|
|
|
|
# publish a file and create shares, which can then be manipulated
|
|
|
|
# later.
|
|
|
|
self.CONTENTS = "New contents go here"
|
|
|
|
num_peers = 20
|
|
|
|
self._client = FakeClient(num_peers)
|
2008-04-11 01:44:06 +00:00
|
|
|
self._storage = self._client._storage
|
|
|
|
d = self._client.create_mutable_file(self.CONTENTS)
|
|
|
|
def _created(node):
|
|
|
|
self._fn = node
|
2008-04-05 00:09:26 +00:00
|
|
|
d.addCallback(_created)
|
|
|
|
return d
|
|
|
|
|
2008-04-16 22:22:30 +00:00
|
|
|
def make_servermap(self, mode=MODE_READ, oldmap=None):
|
2008-04-05 00:09:26 +00:00
|
|
|
if oldmap is None:
|
2008-04-11 21:31:16 +00:00
|
|
|
oldmap = ServerMap()
|
|
|
|
smu = ServermapUpdater(self._fn, oldmap, mode)
|
2008-04-05 00:09:26 +00:00
|
|
|
d = smu.update()
|
|
|
|
return d
|
|
|
|
|
|
|
|
def abbrev_verinfo(self, verinfo):
|
|
|
|
if verinfo is None:
|
|
|
|
return None
|
|
|
|
(seqnum, root_hash, IV, segsize, datalength, k, N, prefix,
|
|
|
|
offsets_tuple) = verinfo
|
|
|
|
return "%d-%s" % (seqnum, base32.b2a(root_hash)[:4])
|
|
|
|
|
|
|
|
def abbrev_verinfo_dict(self, verinfo_d):
|
|
|
|
output = {}
|
|
|
|
for verinfo,value in verinfo_d.items():
|
|
|
|
(seqnum, root_hash, IV, segsize, datalength, k, N, prefix,
|
|
|
|
offsets_tuple) = verinfo
|
|
|
|
output["%d-%s" % (seqnum, base32.b2a(root_hash)[:4])] = value
|
|
|
|
return output
|
|
|
|
|
|
|
|
def dump_servermap(self, servermap):
|
|
|
|
print "SERVERMAP", servermap
|
|
|
|
print "RECOVERABLE", [self.abbrev_verinfo(v)
|
|
|
|
for v in servermap.recoverable_versions()]
|
|
|
|
print "BEST", self.abbrev_verinfo(servermap.best_recoverable_version())
|
|
|
|
print "available", self.abbrev_verinfo_dict(servermap.shares_available())
|
|
|
|
|
|
|
|
def do_download(self, servermap, version=None):
|
|
|
|
if version is None:
|
|
|
|
version = servermap.best_recoverable_version()
|
2008-04-11 21:31:16 +00:00
|
|
|
r = Retrieve(self._fn, servermap, version)
|
2008-04-05 00:09:26 +00:00
|
|
|
return r.download()
|
2007-11-06 05:38:43 +00:00
|
|
|
|
2008-03-10 23:14:08 +00:00
|
|
|
def test_basic(self):
|
2008-04-05 00:09:26 +00:00
|
|
|
d = self.make_servermap()
|
|
|
|
def _do_retrieve(servermap):
|
|
|
|
self._smap = servermap
|
|
|
|
#self.dump_servermap(servermap)
|
|
|
|
self.failUnlessEqual(len(servermap.recoverable_versions()), 1)
|
|
|
|
return self.do_download(servermap)
|
|
|
|
d.addCallback(_do_retrieve)
|
2008-03-10 23:14:08 +00:00
|
|
|
def _retrieved(new_contents):
|
2008-04-05 00:09:26 +00:00
|
|
|
self.failUnlessEqual(new_contents, self.CONTENTS)
|
|
|
|
d.addCallback(_retrieved)
|
|
|
|
# we should be able to re-use the same servermap, both with and
|
|
|
|
# without updating it.
|
|
|
|
d.addCallback(lambda res: self.do_download(self._smap))
|
|
|
|
d.addCallback(_retrieved)
|
|
|
|
d.addCallback(lambda res: self.make_servermap(oldmap=self._smap))
|
|
|
|
d.addCallback(lambda res: self.do_download(self._smap))
|
|
|
|
d.addCallback(_retrieved)
|
|
|
|
# clobbering the pubkey should make the servermap updater re-fetch it
|
|
|
|
def _clobber_pubkey(res):
|
|
|
|
self._fn._pubkey = None
|
|
|
|
d.addCallback(_clobber_pubkey)
|
|
|
|
d.addCallback(lambda res: self.make_servermap(oldmap=self._smap))
|
|
|
|
d.addCallback(lambda res: self.do_download(self._smap))
|
2008-03-10 23:14:08 +00:00
|
|
|
d.addCallback(_retrieved)
|
2007-11-06 05:38:43 +00:00
|
|
|
return d
|
2008-03-10 23:14:08 +00:00
|
|
|
|
2008-03-11 00:46:52 +00:00
|
|
|
|
|
|
|
def shouldFail(self, expected_failure, which, substring,
|
|
|
|
callable, *args, **kwargs):
|
|
|
|
assert substring is None or isinstance(substring, str)
|
|
|
|
d = defer.maybeDeferred(callable, *args, **kwargs)
|
|
|
|
def done(res):
|
|
|
|
if isinstance(res, failure.Failure):
|
|
|
|
res.trap(expected_failure)
|
|
|
|
if substring:
|
|
|
|
self.failUnless(substring in str(res),
|
|
|
|
"substring '%s' not in '%s'"
|
|
|
|
% (substring, str(res)))
|
|
|
|
else:
|
|
|
|
self.fail("%s was supposed to raise %s, not get '%s'" %
|
|
|
|
(which, expected_failure, res))
|
|
|
|
d.addBoth(done)
|
|
|
|
return d
|
|
|
|
|
2008-04-05 00:09:26 +00:00
|
|
|
def _test_corrupt_all(self, offset, substring,
|
|
|
|
should_succeed=False, corrupt_early=True):
|
|
|
|
d = defer.succeed(None)
|
|
|
|
if corrupt_early:
|
2008-04-16 21:49:47 +00:00
|
|
|
d.addCallback(corrupt, self._storage, offset)
|
2008-04-05 00:09:26 +00:00
|
|
|
d.addCallback(lambda res: self.make_servermap())
|
|
|
|
if not corrupt_early:
|
2008-04-16 21:49:47 +00:00
|
|
|
d.addCallback(corrupt, self._storage, offset)
|
2008-04-05 00:09:26 +00:00
|
|
|
def _do_retrieve(servermap):
|
|
|
|
ver = servermap.best_recoverable_version()
|
|
|
|
if ver is None and not should_succeed:
|
|
|
|
# no recoverable versions == not succeeding. The problem
|
|
|
|
# should be noted in the servermap's list of problems.
|
|
|
|
if substring:
|
|
|
|
allproblems = [str(f) for f in servermap.problems]
|
|
|
|
self.failUnless(substring in "".join(allproblems))
|
|
|
|
return
|
|
|
|
if should_succeed:
|
2008-04-16 21:49:47 +00:00
|
|
|
d1 = self._fn.download_to_data()
|
2008-04-05 00:09:26 +00:00
|
|
|
d1.addCallback(lambda new_contents:
|
|
|
|
self.failUnlessEqual(new_contents, self.CONTENTS))
|
|
|
|
return d1
|
|
|
|
else:
|
2008-04-15 23:08:32 +00:00
|
|
|
return self.shouldFail(NotEnoughSharesError,
|
2008-04-05 00:09:26 +00:00
|
|
|
"_corrupt_all(offset=%s)" % (offset,),
|
|
|
|
substring,
|
2008-04-16 21:49:47 +00:00
|
|
|
self._fn.download_to_data)
|
2008-04-05 00:09:26 +00:00
|
|
|
d.addCallback(_do_retrieve)
|
2008-03-11 00:46:52 +00:00
|
|
|
return d
|
|
|
|
|
|
|
|
def test_corrupt_all_verbyte(self):
|
|
|
|
# when the version byte is not 0, we hit an assertion error in
|
|
|
|
# unpack_share().
|
2008-04-05 00:09:26 +00:00
|
|
|
return self._test_corrupt_all(0, "AssertionError")
|
2008-03-11 00:46:52 +00:00
|
|
|
|
|
|
|
def test_corrupt_all_seqnum(self):
|
|
|
|
# a corrupt sequence number will trigger a bad signature
|
2008-04-05 00:09:26 +00:00
|
|
|
return self._test_corrupt_all(1, "signature is invalid")
|
2008-03-11 00:46:52 +00:00
|
|
|
|
|
|
|
def test_corrupt_all_R(self):
|
|
|
|
# a corrupt root hash will trigger a bad signature
|
2008-04-05 00:09:26 +00:00
|
|
|
return self._test_corrupt_all(9, "signature is invalid")
|
2008-03-11 00:46:52 +00:00
|
|
|
|
|
|
|
def test_corrupt_all_IV(self):
|
|
|
|
# a corrupt salt/IV will trigger a bad signature
|
2008-04-05 00:09:26 +00:00
|
|
|
return self._test_corrupt_all(41, "signature is invalid")
|
2008-03-11 00:46:52 +00:00
|
|
|
|
|
|
|
def test_corrupt_all_k(self):
|
|
|
|
# a corrupt 'k' will trigger a bad signature
|
2008-04-05 00:09:26 +00:00
|
|
|
return self._test_corrupt_all(57, "signature is invalid")
|
2008-03-11 00:46:52 +00:00
|
|
|
|
|
|
|
def test_corrupt_all_N(self):
|
|
|
|
# a corrupt 'N' will trigger a bad signature
|
2008-04-05 00:09:26 +00:00
|
|
|
return self._test_corrupt_all(58, "signature is invalid")
|
2008-03-11 00:46:52 +00:00
|
|
|
|
|
|
|
def test_corrupt_all_segsize(self):
|
|
|
|
# a corrupt segsize will trigger a bad signature
|
2008-04-05 00:09:26 +00:00
|
|
|
return self._test_corrupt_all(59, "signature is invalid")
|
2008-03-11 00:46:52 +00:00
|
|
|
|
|
|
|
def test_corrupt_all_datalen(self):
|
|
|
|
# a corrupt data length will trigger a bad signature
|
2008-04-05 00:09:26 +00:00
|
|
|
return self._test_corrupt_all(67, "signature is invalid")
|
2008-03-11 00:46:52 +00:00
|
|
|
|
|
|
|
def test_corrupt_all_pubkey(self):
|
2008-04-05 00:09:26 +00:00
|
|
|
# a corrupt pubkey won't match the URI's fingerprint. We need to
|
|
|
|
# remove the pubkey from the filenode, or else it won't bother trying
|
|
|
|
# to update it.
|
|
|
|
self._fn._pubkey = None
|
|
|
|
return self._test_corrupt_all("pubkey",
|
|
|
|
"pubkey doesn't match fingerprint")
|
2008-03-11 00:46:52 +00:00
|
|
|
|
|
|
|
def test_corrupt_all_sig(self):
|
|
|
|
# a corrupt signature is a bad one
|
|
|
|
# the signature runs from about [543:799], depending upon the length
|
|
|
|
# of the pubkey
|
2008-04-05 00:09:26 +00:00
|
|
|
return self._test_corrupt_all("signature", "signature is invalid")
|
2008-03-11 00:46:52 +00:00
|
|
|
|
|
|
|
def test_corrupt_all_share_hash_chain_number(self):
|
|
|
|
# a corrupt share hash chain entry will show up as a bad hash. If we
|
|
|
|
# mangle the first byte, that will look like a bad hash number,
|
|
|
|
# causing an IndexError
|
2008-04-05 00:09:26 +00:00
|
|
|
return self._test_corrupt_all("share_hash_chain", "corrupt hashes")
|
2008-03-11 00:46:52 +00:00
|
|
|
|
|
|
|
def test_corrupt_all_share_hash_chain_hash(self):
|
|
|
|
# a corrupt share hash chain entry will show up as a bad hash. If we
|
|
|
|
# mangle a few bytes in, that will look like a bad hash.
|
2008-04-05 00:09:26 +00:00
|
|
|
return self._test_corrupt_all(("share_hash_chain",4), "corrupt hashes")
|
2008-03-11 00:46:52 +00:00
|
|
|
|
|
|
|
def test_corrupt_all_block_hash_tree(self):
|
2008-04-05 00:09:26 +00:00
|
|
|
return self._test_corrupt_all("block_hash_tree",
|
|
|
|
"block hash tree failure")
|
2008-03-11 00:46:52 +00:00
|
|
|
|
|
|
|
def test_corrupt_all_block(self):
|
2008-04-05 00:09:26 +00:00
|
|
|
return self._test_corrupt_all("share_data", "block hash tree failure")
|
2008-03-11 00:46:52 +00:00
|
|
|
|
|
|
|
def test_corrupt_all_encprivkey(self):
|
2008-04-05 00:09:26 +00:00
|
|
|
# a corrupted privkey won't even be noticed by the reader, only by a
|
|
|
|
# writer.
|
|
|
|
return self._test_corrupt_all("enc_privkey", None, should_succeed=True)
|
2008-03-11 05:15:43 +00:00
|
|
|
|
|
|
|
def test_basic_pubkey_at_end(self):
|
|
|
|
# we corrupt the pubkey in all but the last 'k' shares, allowing the
|
|
|
|
# download to succeed but forcing a bunch of retries first. Note that
|
|
|
|
# this is rather pessimistic: our Retrieve process will throw away
|
|
|
|
# the whole share if the pubkey is bad, even though the rest of the
|
|
|
|
# share might be good.
|
2008-04-05 00:09:26 +00:00
|
|
|
|
|
|
|
self._fn._pubkey = None
|
|
|
|
k = self._fn.get_required_shares()
|
|
|
|
N = self._fn.get_total_shares()
|
|
|
|
d = defer.succeed(None)
|
2008-04-16 21:49:47 +00:00
|
|
|
d.addCallback(corrupt, self._storage, "pubkey",
|
2008-04-05 00:09:26 +00:00
|
|
|
shnums_to_corrupt=range(0, N-k))
|
|
|
|
d.addCallback(lambda res: self.make_servermap())
|
|
|
|
def _do_retrieve(servermap):
|
|
|
|
self.failUnless(servermap.problems)
|
|
|
|
self.failUnless("pubkey doesn't match fingerprint"
|
|
|
|
in str(servermap.problems[0]))
|
|
|
|
ver = servermap.best_recoverable_version()
|
2008-04-11 21:31:16 +00:00
|
|
|
r = Retrieve(self._fn, servermap, ver)
|
2008-04-05 00:09:26 +00:00
|
|
|
return r.download()
|
|
|
|
d.addCallback(_do_retrieve)
|
|
|
|
d.addCallback(lambda new_contents:
|
|
|
|
self.failUnlessEqual(new_contents, self.CONTENTS))
|
2008-03-11 05:15:43 +00:00
|
|
|
return d
|
|
|
|
|
2008-04-16 21:49:47 +00:00
|
|
|
def test_corrupt_some(self):
|
|
|
|
# corrupt the data of first five shares (so the servermap thinks
|
|
|
|
# they're good but retrieve marks them as bad), so that the
|
2008-04-16 22:22:30 +00:00
|
|
|
# MODE_READ set of 6 will be insufficient, forcing node.download to
|
2008-04-16 21:49:47 +00:00
|
|
|
# retry with more servers.
|
|
|
|
corrupt(None, self._storage, "share_data", range(5))
|
|
|
|
d = self.make_servermap()
|
|
|
|
def _do_retrieve(servermap):
|
|
|
|
ver = servermap.best_recoverable_version()
|
|
|
|
self.failUnless(ver)
|
|
|
|
return self._fn.download_to_data()
|
|
|
|
d.addCallback(_do_retrieve)
|
|
|
|
d.addCallback(lambda new_contents:
|
|
|
|
self.failUnlessEqual(new_contents, self.CONTENTS))
|
|
|
|
return d
|
|
|
|
|
|
|
|
def test_download_fails(self):
|
|
|
|
corrupt(None, self._storage, "signature")
|
|
|
|
d = self.shouldFail(UnrecoverableFileError, "test_download_anyway",
|
|
|
|
"no recoverable versions",
|
|
|
|
self._fn.download_to_data)
|
|
|
|
return d
|
|
|
|
|
2008-04-11 01:44:06 +00:00
|
|
|
|
|
|
|
class MultipleEncodings(unittest.TestCase):
|
|
|
|
def setUp(self):
|
|
|
|
self.CONTENTS = "New contents go here"
|
|
|
|
num_peers = 20
|
|
|
|
self._client = FakeClient(num_peers)
|
|
|
|
self._storage = self._client._storage
|
|
|
|
d = self._client.create_mutable_file(self.CONTENTS)
|
|
|
|
def _created(node):
|
|
|
|
self._fn = node
|
|
|
|
d.addCallback(_created)
|
|
|
|
return d
|
|
|
|
|
|
|
|
def _encode(self, k, n, data):
|
2008-03-11 06:47:35 +00:00
|
|
|
# encode 'data' into a peerid->shares dict.
|
2008-03-11 05:15:43 +00:00
|
|
|
|
2008-04-11 01:44:06 +00:00
|
|
|
fn2 = FastMutableFileNode(self._client)
|
2008-03-11 05:15:43 +00:00
|
|
|
# init_from_uri populates _uri, _writekey, _readkey, _storage_index,
|
|
|
|
# and _fingerprint
|
2008-04-11 01:44:06 +00:00
|
|
|
fn = self._fn
|
2008-03-11 06:47:35 +00:00
|
|
|
fn2.init_from_uri(fn.get_uri())
|
2008-03-11 05:15:43 +00:00
|
|
|
# then we copy over other fields that are normally fetched from the
|
|
|
|
# existing shares
|
2008-03-11 06:47:35 +00:00
|
|
|
fn2._pubkey = fn._pubkey
|
|
|
|
fn2._privkey = fn._privkey
|
|
|
|
fn2._encprivkey = fn._encprivkey
|
2008-03-11 05:15:43 +00:00
|
|
|
fn2._current_seqnum = 0
|
|
|
|
fn2._current_roothash = "\x00" * 32
|
|
|
|
# and set the encoding parameters to something completely different
|
2008-03-11 06:47:35 +00:00
|
|
|
fn2._required_shares = k
|
|
|
|
fn2._total_shares = n
|
2008-03-11 05:15:43 +00:00
|
|
|
|
2008-04-11 01:44:06 +00:00
|
|
|
s = self._client._storage
|
|
|
|
s._peers = {} # clear existing storage
|
2008-04-11 21:31:16 +00:00
|
|
|
p2 = Publish(fn2, None)
|
2008-03-11 06:47:35 +00:00
|
|
|
d = p2.publish(data)
|
|
|
|
def _published(res):
|
|
|
|
shares = s._peers
|
|
|
|
s._peers = {}
|
|
|
|
return shares
|
|
|
|
d.addCallback(_published)
|
|
|
|
return d
|
|
|
|
|
2008-04-16 22:22:30 +00:00
|
|
|
def make_servermap(self, mode=MODE_READ, oldmap=None):
|
2008-04-05 00:09:26 +00:00
|
|
|
if oldmap is None:
|
2008-04-11 21:31:16 +00:00
|
|
|
oldmap = ServerMap()
|
|
|
|
smu = ServermapUpdater(self._fn, oldmap, mode)
|
2008-04-05 00:09:26 +00:00
|
|
|
d = smu.update()
|
|
|
|
return d
|
|
|
|
|
2008-03-11 06:47:35 +00:00
|
|
|
def test_multiple_encodings(self):
|
|
|
|
# we encode the same file in two different ways (3-of-10 and 4-of-9),
|
|
|
|
# then mix up the shares, to make sure that download survives seeing
|
|
|
|
# a variety of encodings. This is actually kind of tricky to set up.
|
|
|
|
|
|
|
|
contents1 = "Contents for encoding 1 (3-of-10) go here"
|
|
|
|
contents2 = "Contents for encoding 2 (4-of-9) go here"
|
|
|
|
contents3 = "Contents for encoding 3 (4-of-7) go here"
|
2008-03-11 05:15:43 +00:00
|
|
|
|
|
|
|
# we make a retrieval object that doesn't know what encoding
|
|
|
|
# parameters to use
|
2008-04-11 01:44:06 +00:00
|
|
|
fn3 = FastMutableFileNode(self._client)
|
|
|
|
fn3.init_from_uri(self._fn.get_uri())
|
2008-03-11 05:15:43 +00:00
|
|
|
|
|
|
|
# now we upload a file through fn1, and grab its shares
|
2008-04-11 01:44:06 +00:00
|
|
|
d = self._encode(3, 10, contents1)
|
2008-03-11 06:47:35 +00:00
|
|
|
def _encoded_1(shares):
|
|
|
|
self._shares1 = shares
|
|
|
|
d.addCallback(_encoded_1)
|
2008-04-11 01:44:06 +00:00
|
|
|
d.addCallback(lambda res: self._encode(4, 9, contents2))
|
2008-03-11 06:47:35 +00:00
|
|
|
def _encoded_2(shares):
|
|
|
|
self._shares2 = shares
|
|
|
|
d.addCallback(_encoded_2)
|
2008-04-11 01:44:06 +00:00
|
|
|
d.addCallback(lambda res: self._encode(4, 7, contents3))
|
2008-03-11 06:47:35 +00:00
|
|
|
def _encoded_3(shares):
|
|
|
|
self._shares3 = shares
|
|
|
|
d.addCallback(_encoded_3)
|
|
|
|
|
2008-03-11 05:15:43 +00:00
|
|
|
def _merge(res):
|
|
|
|
log.msg("merging sharelists")
|
2008-03-11 06:16:28 +00:00
|
|
|
# we merge the shares from the two sets, leaving each shnum in
|
|
|
|
# its original location, but using a share from set1 or set2
|
|
|
|
# according to the following sequence:
|
|
|
|
#
|
2008-03-11 05:15:43 +00:00
|
|
|
# 4-of-9 a s2
|
|
|
|
# 4-of-9 b s2
|
2008-03-11 06:47:35 +00:00
|
|
|
# 4-of-7 c s3
|
|
|
|
# 4-of-9 d s2
|
2008-03-11 05:15:43 +00:00
|
|
|
# 3-of-9 e s1
|
2008-03-11 06:16:28 +00:00
|
|
|
# 3-of-9 f s1
|
2008-03-11 06:47:35 +00:00
|
|
|
# 3-of-9 g s1
|
|
|
|
# 4-of-9 h s2
|
2008-03-11 06:16:28 +00:00
|
|
|
#
|
|
|
|
# so that neither form can be recovered until fetch [f], at which
|
|
|
|
# point version-s1 (the 3-of-10 form) should be recoverable. If
|
|
|
|
# the implementation latches on to the first version it sees,
|
|
|
|
# then s2 will be recoverable at fetch [g].
|
|
|
|
|
|
|
|
# Later, when we implement code that handles multiple versions,
|
|
|
|
# we can use this framework to assert that all recoverable
|
|
|
|
# versions are retrieved, and test that 'epsilon' does its job
|
|
|
|
|
2008-03-11 06:47:35 +00:00
|
|
|
places = [2, 2, 3, 2, 1, 1, 1, 2]
|
2008-03-11 06:16:28 +00:00
|
|
|
|
|
|
|
sharemap = {}
|
|
|
|
|
2008-04-11 01:44:06 +00:00
|
|
|
for i,peerid in enumerate(self._client._peerids):
|
2008-03-11 06:16:28 +00:00
|
|
|
peerid_s = shortnodeid_b2a(peerid)
|
|
|
|
for shnum in self._shares1.get(peerid, {}):
|
|
|
|
if shnum < len(places):
|
|
|
|
which = places[shnum]
|
|
|
|
else:
|
|
|
|
which = "x"
|
2008-04-11 01:44:06 +00:00
|
|
|
self._client._storage._peers[peerid] = peers = {}
|
2008-03-11 06:16:28 +00:00
|
|
|
in_1 = shnum in self._shares1[peerid]
|
|
|
|
in_2 = shnum in self._shares2.get(peerid, {})
|
2008-03-11 06:47:35 +00:00
|
|
|
in_3 = shnum in self._shares3.get(peerid, {})
|
|
|
|
#print peerid_s, shnum, which, in_1, in_2, in_3
|
2008-03-11 06:16:28 +00:00
|
|
|
if which == 1:
|
|
|
|
if in_1:
|
|
|
|
peers[shnum] = self._shares1[peerid][shnum]
|
2008-03-11 06:47:35 +00:00
|
|
|
sharemap[shnum] = peerid
|
2008-03-11 06:16:28 +00:00
|
|
|
elif which == 2:
|
|
|
|
if in_2:
|
|
|
|
peers[shnum] = self._shares2[peerid][shnum]
|
2008-03-11 06:47:35 +00:00
|
|
|
sharemap[shnum] = peerid
|
|
|
|
elif which == 3:
|
|
|
|
if in_3:
|
|
|
|
peers[shnum] = self._shares3[peerid][shnum]
|
|
|
|
sharemap[shnum] = peerid
|
2008-03-11 06:16:28 +00:00
|
|
|
|
2008-03-11 05:15:43 +00:00
|
|
|
# we don't bother placing any other shares
|
2008-03-11 06:16:28 +00:00
|
|
|
# now sort the sequence so that share 0 is returned first
|
|
|
|
new_sequence = [sharemap[shnum]
|
|
|
|
for shnum in sorted(sharemap.keys())]
|
2008-04-11 01:44:06 +00:00
|
|
|
self._client._storage._sequence = new_sequence
|
2008-03-11 05:15:43 +00:00
|
|
|
log.msg("merge done")
|
|
|
|
d.addCallback(_merge)
|
2008-04-11 01:44:06 +00:00
|
|
|
d.addCallback(lambda res: fn3.download_to_data())
|
2008-03-11 05:15:43 +00:00
|
|
|
def _retrieved(new_contents):
|
2008-03-11 07:26:00 +00:00
|
|
|
# the current specified behavior is "first version recoverable"
|
|
|
|
self.failUnlessEqual(new_contents, contents1)
|
2008-03-11 05:15:43 +00:00
|
|
|
d.addCallback(_retrieved)
|
|
|
|
return d
|
|
|
|
|
2008-04-05 00:09:26 +00:00
|
|
|
|
|
|
|
class Utils(unittest.TestCase):
|
|
|
|
def test_dict_of_sets(self):
|
2008-04-11 21:31:16 +00:00
|
|
|
ds = DictOfSets()
|
2008-04-05 00:09:26 +00:00
|
|
|
ds.add(1, "a")
|
|
|
|
ds.add(2, "b")
|
|
|
|
ds.add(2, "b")
|
|
|
|
ds.add(2, "c")
|
|
|
|
self.failUnlessEqual(ds[1], set(["a"]))
|
|
|
|
self.failUnlessEqual(ds[2], set(["b", "c"]))
|
|
|
|
ds.discard(3, "d") # should not raise an exception
|
|
|
|
ds.discard(2, "b")
|
|
|
|
self.failUnlessEqual(ds[2], set(["c"]))
|
|
|
|
ds.discard(2, "c")
|
|
|
|
self.failIf(2 in ds)
|
|
|
|
|
|
|
|
def _do_inside(self, c, x_start, x_length, y_start, y_length):
|
|
|
|
# we compare this against sets of integers
|
|
|
|
x = set(range(x_start, x_start+x_length))
|
|
|
|
y = set(range(y_start, y_start+y_length))
|
|
|
|
should_be_inside = x.issubset(y)
|
|
|
|
self.failUnlessEqual(should_be_inside, c._inside(x_start, x_length,
|
|
|
|
y_start, y_length),
|
|
|
|
str((x_start, x_length, y_start, y_length)))
|
|
|
|
|
|
|
|
def test_cache_inside(self):
|
2008-04-11 21:31:16 +00:00
|
|
|
c = ResponseCache()
|
2008-04-05 00:09:26 +00:00
|
|
|
x_start = 10
|
|
|
|
x_length = 5
|
|
|
|
for y_start in range(8, 17):
|
|
|
|
for y_length in range(8):
|
|
|
|
self._do_inside(c, x_start, x_length, y_start, y_length)
|
|
|
|
|
|
|
|
def _do_overlap(self, c, x_start, x_length, y_start, y_length):
|
|
|
|
# we compare this against sets of integers
|
|
|
|
x = set(range(x_start, x_start+x_length))
|
|
|
|
y = set(range(y_start, y_start+y_length))
|
|
|
|
overlap = bool(x.intersection(y))
|
|
|
|
self.failUnlessEqual(overlap, c._does_overlap(x_start, x_length,
|
|
|
|
y_start, y_length),
|
|
|
|
str((x_start, x_length, y_start, y_length)))
|
|
|
|
|
|
|
|
def test_cache_overlap(self):
|
2008-04-11 21:31:16 +00:00
|
|
|
c = ResponseCache()
|
2008-04-05 00:09:26 +00:00
|
|
|
x_start = 10
|
|
|
|
x_length = 5
|
|
|
|
for y_start in range(8, 17):
|
|
|
|
for y_length in range(8):
|
|
|
|
self._do_overlap(c, x_start, x_length, y_start, y_length)
|
|
|
|
|
|
|
|
def test_cache(self):
|
2008-04-11 21:31:16 +00:00
|
|
|
c = ResponseCache()
|
2008-04-05 00:09:26 +00:00
|
|
|
# xdata = base62.b2a(os.urandom(100))[:100]
|
|
|
|
xdata = "1Ex4mdMaDyOl9YnGBM3I4xaBF97j8OQAg1K3RBR01F2PwTP4HohB3XpACuku8Xj4aTQjqJIR1f36mEj3BCNjXaJmPBEZnnHL0U9l"
|
|
|
|
ydata = "4DCUQXvkEPnnr9Lufikq5t21JsnzZKhzxKBhLhrBB6iIcBOWRuT4UweDhjuKJUre8A4wOObJnl3Kiqmlj4vjSLSqUGAkUD87Y3vs"
|
|
|
|
nope = (None, None)
|
|
|
|
c.add("v1", 1, 0, xdata, "time0")
|
|
|
|
c.add("v1", 1, 2000, ydata, "time1")
|
|
|
|
self.failUnlessEqual(c.read("v2", 1, 10, 11), nope)
|
|
|
|
self.failUnlessEqual(c.read("v1", 2, 10, 11), nope)
|
|
|
|
self.failUnlessEqual(c.read("v1", 1, 0, 10), (xdata[:10], "time0"))
|
|
|
|
self.failUnlessEqual(c.read("v1", 1, 90, 10), (xdata[90:], "time0"))
|
|
|
|
self.failUnlessEqual(c.read("v1", 1, 300, 10), nope)
|
|
|
|
self.failUnlessEqual(c.read("v1", 1, 2050, 5), (ydata[50:55], "time1"))
|
|
|
|
self.failUnlessEqual(c.read("v1", 1, 0, 101), nope)
|
|
|
|
self.failUnlessEqual(c.read("v1", 1, 99, 1), (xdata[99:100], "time0"))
|
|
|
|
self.failUnlessEqual(c.read("v1", 1, 100, 1), nope)
|
|
|
|
self.failUnlessEqual(c.read("v1", 1, 1990, 9), nope)
|
|
|
|
self.failUnlessEqual(c.read("v1", 1, 1990, 10), nope)
|
|
|
|
self.failUnlessEqual(c.read("v1", 1, 1990, 11), nope)
|
|
|
|
self.failUnlessEqual(c.read("v1", 1, 1990, 15), nope)
|
|
|
|
self.failUnlessEqual(c.read("v1", 1, 1990, 19), nope)
|
|
|
|
self.failUnlessEqual(c.read("v1", 1, 1990, 20), nope)
|
|
|
|
self.failUnlessEqual(c.read("v1", 1, 1990, 21), nope)
|
|
|
|
self.failUnlessEqual(c.read("v1", 1, 1990, 25), nope)
|
|
|
|
self.failUnlessEqual(c.read("v1", 1, 1999, 25), nope)
|
|
|
|
|
|
|
|
# optional: join fragments
|
2008-04-11 21:31:16 +00:00
|
|
|
c = ResponseCache()
|
2008-04-05 00:09:26 +00:00
|
|
|
c.add("v1", 1, 0, xdata[:10], "time0")
|
|
|
|
c.add("v1", 1, 10, xdata[10:20], "time1")
|
|
|
|
#self.failUnlessEqual(c.read("v1", 1, 0, 20), (xdata[:20], "time0"))
|
|
|
|
|