2007-11-01 22:15:29 +00:00
|
|
|
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +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
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
from allmydata import uri, client
|
|
|
|
from allmydata.nodemaker import NodeMaker
|
|
|
|
from allmydata.util import base32
|
|
|
|
from allmydata.util.hashutil import tagged_hash, ssk_writekey_hash, \
|
|
|
|
ssk_pubkey_fingerprint_hash
|
|
|
|
from allmydata.interfaces import IRepairResults, ICheckAndRepairResults, \
|
|
|
|
NotEnoughSharesError
|
2008-10-22 08:38:18 +00:00
|
|
|
from allmydata.monitor import Monitor
|
2008-08-06 19:06:07 +00:00
|
|
|
from allmydata.test.common import ShouldFailMixin
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
from allmydata.test.no_network import GridTestMixin
|
2009-05-22 00:38:23 +00:00
|
|
|
from foolscap.api import eventually, fireEventually
|
2008-03-11 05:15:43 +00:00
|
|
|
from foolscap.logging import log
|
2009-06-01 21:06:04 +00:00
|
|
|
from allmydata.storage_client import StorageFarmBroker
|
2007-11-01 22:15:29 +00:00
|
|
|
|
2008-12-07 15:20:08 +00:00
|
|
|
from allmydata.mutable.filenode import MutableFileNode, BackoffAgent
|
2009-02-16 03:32:10 +00:00
|
|
|
from allmydata.mutable.common import ResponseCache, \
|
2008-04-18 02:12:42 +00:00
|
|
|
MODE_CHECK, MODE_ANYTHING, MODE_WRITE, MODE_READ, \
|
2008-04-23 01:53:20 +00:00
|
|
|
NeedMoreDataError, UnrecoverableFileError, UncoordinatedWriteError, \
|
2008-07-08 00:36:00 +00:00
|
|
|
NotEnoughServersError, CorruptShareError
|
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-12-07 15:20:08 +00:00
|
|
|
from allmydata.mutable.repairer import MustForceRepairError
|
2008-04-11 21:31:16 +00:00
|
|
|
|
2010-02-26 08:14:33 +00:00
|
|
|
import allmydata.test.common_util as testutil
|
2008-10-29 04:28:31 +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-04-22 23:47:52 +00:00
|
|
|
self._pending_timer = None
|
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:
|
2008-04-22 23:47:52 +00:00
|
|
|
self._pending_timer = reactor.callLater(1.0, self._fire_readers)
|
2008-03-11 05:15:43 +00:00
|
|
|
self._pending[peerid] = (d, shares)
|
|
|
|
return d
|
|
|
|
|
|
|
|
def _fire_readers(self):
|
2008-04-22 23:47:52 +00:00
|
|
|
self._pending_timer = None
|
2008-03-11 05:15:43 +00:00
|
|
|
pending = self._pending
|
|
|
|
self._pending = {}
|
|
|
|
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
|
2008-10-24 20:21:28 +00:00
|
|
|
def callRemoteOnly(self, methname, *args, **kwargs):
|
|
|
|
d = self.callRemote(methname, *args, **kwargs)
|
|
|
|
d.addBoth(lambda ignore: None)
|
|
|
|
pass
|
|
|
|
|
|
|
|
def advise_corrupt_share(self, share_type, storage_index, shnum, reason):
|
|
|
|
pass
|
2008-04-11 01:44:06 +00:00
|
|
|
|
|
|
|
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-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:])
|
|
|
|
|
2008-07-08 00:36:00 +00:00
|
|
|
def corrupt(res, s, offset, shnums_to_corrupt=None, offset_offset=0):
|
2008-04-16 21:49:47 +00:00
|
|
|
# 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
|
2008-07-08 00:36:00 +00:00
|
|
|
real_offset = int(real_offset) + offset2 + offset_offset
|
2008-04-16 21:49:47 +00:00
|
|
|
assert isinstance(real_offset, int), offset
|
|
|
|
shares[shnum] = flip_bit(data, real_offset)
|
|
|
|
return res
|
|
|
|
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
def make_storagebroker(s=None, num_peers=10):
|
|
|
|
if not s:
|
|
|
|
s = FakeStorage()
|
|
|
|
peerids = [tagged_hash("peerid", "%d" % i)[:20]
|
|
|
|
for i in range(num_peers)]
|
|
|
|
storage_broker = StorageFarmBroker(None, True)
|
|
|
|
for peerid in peerids:
|
|
|
|
fss = FakeStorageServer(peerid, s)
|
|
|
|
storage_broker.test_add_server(peerid, fss)
|
|
|
|
return storage_broker
|
|
|
|
|
|
|
|
def make_nodemaker(s=None, num_peers=10):
|
|
|
|
storage_broker = make_storagebroker(s, num_peers)
|
2009-11-18 01:54:44 +00:00
|
|
|
sh = client.SecretHolder("lease secret", "convergence secret")
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
keygen = client.KeyGenerator()
|
|
|
|
keygen.set_default_keysize(522)
|
|
|
|
nodemaker = NodeMaker(storage_broker, sh, None,
|
2010-08-04 07:27:10 +00:00
|
|
|
None, None,
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
{"k": 3, "n": 10}, keygen)
|
|
|
|
return nodemaker
|
|
|
|
|
2008-04-18 02:55:51 +00:00
|
|
|
class Filenode(unittest.TestCase, testutil.ShouldFailMixin):
|
2009-06-29 20:03:58 +00:00
|
|
|
# this used to be in Publish, but we removed the limit. Some of
|
|
|
|
# these tests test whether the new code correctly allows files
|
2009-06-20 21:28:22 +00:00
|
|
|
# larger than the limit.
|
|
|
|
OLD_MAX_SEGMENT_SIZE = 3500000
|
2007-11-01 22:15:29 +00:00
|
|
|
def setUp(self):
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
self._storage = s = FakeStorage()
|
|
|
|
self.nodemaker = make_nodemaker(s)
|
2007-11-01 22:15:29 +00:00
|
|
|
|
|
|
|
def test_create(self):
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
d = self.nodemaker.create_mutable_file()
|
2007-11-01 22:15:29 +00:00
|
|
|
def _created(n):
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
self.failUnless(isinstance(n, MutableFileNode))
|
2008-08-12 23:14:07 +00:00
|
|
|
self.failUnlessEqual(n.get_storage_index(), n._storage_index)
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
sb = self.nodemaker.storage_broker
|
2009-06-02 03:07:50 +00:00
|
|
|
peer0 = sorted(sb.get_all_serverids())[0]
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
shnums = self._storage._peers[peer0].keys()
|
2008-04-11 01:44:06 +00:00
|
|
|
self.failUnlessEqual(len(shnums), 1)
|
|
|
|
d.addCallback(_created)
|
|
|
|
return d
|
|
|
|
|
2008-04-18 02:12:42 +00:00
|
|
|
def test_serialize(self):
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
n = MutableFileNode(None, None, {"k": 3, "n": 10}, None)
|
2008-04-18 02:12:42 +00:00
|
|
|
calls = []
|
|
|
|
def _callback(*args, **kwargs):
|
|
|
|
self.failUnlessEqual(args, (4,) )
|
|
|
|
self.failUnlessEqual(kwargs, {"foo": 5})
|
|
|
|
calls.append(1)
|
|
|
|
return 6
|
|
|
|
d = n._do_serialized(_callback, 4, foo=5)
|
|
|
|
def _check_callback(res):
|
|
|
|
self.failUnlessEqual(res, 6)
|
|
|
|
self.failUnlessEqual(calls, [1])
|
|
|
|
d.addCallback(_check_callback)
|
|
|
|
|
|
|
|
def _errback():
|
|
|
|
raise ValueError("heya")
|
|
|
|
d.addCallback(lambda res:
|
|
|
|
self.shouldFail(ValueError, "_check_errback", "heya",
|
|
|
|
n._do_serialized, _errback))
|
|
|
|
return d
|
|
|
|
|
2008-04-11 01:44:06 +00:00
|
|
|
def test_upload_and_download(self):
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
d = self.nodemaker.create_mutable_file()
|
2008-04-11 01:44:06 +00:00
|
|
|
def _created(n):
|
|
|
|
d = defer.succeed(None)
|
2008-04-18 00:51:38 +00:00
|
|
|
d.addCallback(lambda res: n.get_servermap(MODE_READ))
|
2008-04-11 01:44:06 +00:00
|
|
|
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))
|
2008-04-18 00:51:38 +00:00
|
|
|
d.addCallback(lambda res: n.download_best_version())
|
2007-11-01 22:15:29 +00:00
|
|
|
d.addCallback(lambda res: self.failUnlessEqual(res, "contents 1"))
|
2008-08-13 02:02:52 +00:00
|
|
|
d.addCallback(lambda res: n.get_size_of_best_version())
|
|
|
|
d.addCallback(lambda size:
|
|
|
|
self.failUnlessEqual(size, len("contents 1")))
|
2008-03-13 01:00:43 +00:00
|
|
|
d.addCallback(lambda res: n.overwrite("contents 2"))
|
2008-04-18 00:51:38 +00:00
|
|
|
d.addCallback(lambda res: n.download_best_version())
|
2007-11-01 22:15:29 +00:00
|
|
|
d.addCallback(lambda res: self.failUnlessEqual(res, "contents 2"))
|
2008-04-18 00:51:38 +00:00
|
|
|
d.addCallback(lambda res: n.get_servermap(MODE_WRITE))
|
|
|
|
d.addCallback(lambda smap: n.upload("contents 3", smap))
|
|
|
|
d.addCallback(lambda res: n.download_best_version())
|
|
|
|
d.addCallback(lambda res: self.failUnlessEqual(res, "contents 3"))
|
|
|
|
d.addCallback(lambda res: n.get_servermap(MODE_ANYTHING))
|
|
|
|
d.addCallback(lambda smap:
|
|
|
|
n.download_version(smap,
|
|
|
|
smap.best_recoverable_version()))
|
2008-03-13 01:00:43 +00:00
|
|
|
d.addCallback(lambda res: self.failUnlessEqual(res, "contents 3"))
|
2008-04-22 00:50:50 +00:00
|
|
|
# test a file that is large enough to overcome the
|
|
|
|
# mapupdate-to-retrieve data caching (i.e. make the shares larger
|
|
|
|
# than the default readsize, which is 2000 bytes). A 15kB file
|
|
|
|
# will have 5kB shares.
|
|
|
|
d.addCallback(lambda res: n.overwrite("large size file" * 1000))
|
|
|
|
d.addCallback(lambda res: n.download_best_version())
|
|
|
|
d.addCallback(lambda res:
|
|
|
|
self.failUnlessEqual(res, "large size file" * 1000))
|
2007-11-01 22:15:29 +00:00
|
|
|
return d
|
|
|
|
d.addCallback(_created)
|
|
|
|
return d
|
|
|
|
|
|
|
|
def test_create_with_initial_contents(self):
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
d = self.nodemaker.create_mutable_file("contents 1")
|
2007-11-01 22:15:29 +00:00
|
|
|
def _created(n):
|
2008-04-18 00:51:38 +00:00
|
|
|
d = n.download_best_version()
|
2007-11-01 22:15:29 +00:00
|
|
|
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"))
|
2008-04-18 00:51:38 +00:00
|
|
|
d.addCallback(lambda res: n.download_best_version())
|
2007-11-01 22:15:29 +00:00
|
|
|
d.addCallback(lambda res: self.failUnlessEqual(res, "contents 2"))
|
|
|
|
return d
|
|
|
|
d.addCallback(_created)
|
|
|
|
return d
|
|
|
|
|
2010-10-24 19:34:09 +00:00
|
|
|
def test_response_cache_memory_leak(self):
|
|
|
|
d = self.nodemaker.create_mutable_file("contents")
|
|
|
|
def _created(n):
|
|
|
|
d = n.download_best_version()
|
|
|
|
d.addCallback(lambda res: self.failUnlessEqual(res, "contents"))
|
|
|
|
d.addCallback(lambda ign: self.failUnless(isinstance(n._cache, ResponseCache)))
|
|
|
|
|
2010-10-27 04:33:02 +00:00
|
|
|
def _check_cache(expected):
|
|
|
|
# The total size of cache entries should not increase on the second download;
|
|
|
|
# in fact the cache contents should be identical.
|
2010-10-24 19:34:09 +00:00
|
|
|
d2 = n.download_best_version()
|
2010-10-27 04:33:02 +00:00
|
|
|
d2.addCallback(lambda rep: self.failUnlessEqual(repr(n._cache.cache), expected))
|
2010-10-24 19:34:09 +00:00
|
|
|
return d2
|
2010-10-27 04:33:02 +00:00
|
|
|
d.addCallback(lambda ign: _check_cache(repr(n._cache.cache)))
|
2010-10-24 19:34:09 +00:00
|
|
|
return d
|
|
|
|
d.addCallback(_created)
|
|
|
|
return d
|
|
|
|
|
2009-10-13 03:12:32 +00:00
|
|
|
def test_create_with_initial_contents_function(self):
|
|
|
|
data = "initial contents"
|
|
|
|
def _make_contents(n):
|
|
|
|
self.failUnless(isinstance(n, MutableFileNode))
|
|
|
|
key = n.get_writekey()
|
|
|
|
self.failUnless(isinstance(key, str), key)
|
|
|
|
self.failUnlessEqual(len(key), 16) # AES key size
|
|
|
|
return data
|
|
|
|
d = self.nodemaker.create_mutable_file(_make_contents)
|
|
|
|
def _created(n):
|
|
|
|
return n.download_best_version()
|
|
|
|
d.addCallback(_created)
|
|
|
|
d.addCallback(lambda data2: self.failUnlessEqual(data2, data))
|
|
|
|
return d
|
|
|
|
|
2008-06-03 07:02:10 +00:00
|
|
|
def test_create_with_too_large_contents(self):
|
2009-06-20 21:28:22 +00:00
|
|
|
BIG = "a" * (self.OLD_MAX_SEGMENT_SIZE + 1)
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
d = self.nodemaker.create_mutable_file(BIG)
|
2008-06-03 07:02:10 +00:00
|
|
|
def _created(n):
|
2009-06-20 21:28:22 +00:00
|
|
|
d = n.overwrite(BIG)
|
|
|
|
return d
|
2008-06-03 07:02:10 +00:00
|
|
|
d.addCallback(_created)
|
|
|
|
return d
|
|
|
|
|
2008-12-06 05:49:23 +00:00
|
|
|
def failUnlessCurrentSeqnumIs(self, n, expected_seqnum, which):
|
2008-04-18 02:12:42 +00:00
|
|
|
d = n.get_servermap(MODE_READ)
|
|
|
|
d.addCallback(lambda servermap: servermap.best_recoverable_version())
|
|
|
|
d.addCallback(lambda verinfo:
|
2008-12-06 05:49:23 +00:00
|
|
|
self.failUnlessEqual(verinfo[0], expected_seqnum, which))
|
2008-04-18 02:12:42 +00:00
|
|
|
return d
|
|
|
|
|
|
|
|
def test_modify(self):
|
2008-12-06 05:07:10 +00:00
|
|
|
def _modifier(old_contents, servermap, first_time):
|
2008-04-18 02:12:42 +00:00
|
|
|
return old_contents + "line2"
|
2008-12-06 05:07:10 +00:00
|
|
|
def _non_modifier(old_contents, servermap, first_time):
|
2008-04-18 02:12:42 +00:00
|
|
|
return old_contents
|
2008-12-06 05:07:10 +00:00
|
|
|
def _none_modifier(old_contents, servermap, first_time):
|
2008-04-18 02:12:42 +00:00
|
|
|
return None
|
2008-12-06 05:07:10 +00:00
|
|
|
def _error_modifier(old_contents, servermap, first_time):
|
2008-04-18 02:55:51 +00:00
|
|
|
raise ValueError("oops")
|
2008-12-06 05:07:10 +00:00
|
|
|
def _toobig_modifier(old_contents, servermap, first_time):
|
2009-06-20 21:28:22 +00:00
|
|
|
return "b" * (self.OLD_MAX_SEGMENT_SIZE+1)
|
2008-04-18 02:12:42 +00:00
|
|
|
calls = []
|
2008-12-06 05:07:10 +00:00
|
|
|
def _ucw_error_modifier(old_contents, servermap, first_time):
|
2008-04-18 02:12:42 +00:00
|
|
|
# simulate an UncoordinatedWriteError once
|
|
|
|
calls.append(1)
|
|
|
|
if len(calls) <= 1:
|
|
|
|
raise UncoordinatedWriteError("simulated")
|
|
|
|
return old_contents + "line3"
|
2008-12-06 05:49:23 +00:00
|
|
|
def _ucw_error_non_modifier(old_contents, servermap, first_time):
|
|
|
|
# simulate an UncoordinatedWriteError once, and don't actually
|
|
|
|
# modify the contents on subsequent invocations
|
|
|
|
calls.append(1)
|
|
|
|
if len(calls) <= 1:
|
|
|
|
raise UncoordinatedWriteError("simulated")
|
|
|
|
return old_contents
|
2008-04-18 02:12:42 +00:00
|
|
|
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
d = self.nodemaker.create_mutable_file("line1")
|
2008-04-18 02:12:42 +00:00
|
|
|
def _created(n):
|
|
|
|
d = n.modify(_modifier)
|
|
|
|
d.addCallback(lambda res: n.download_best_version())
|
|
|
|
d.addCallback(lambda res: self.failUnlessEqual(res, "line1line2"))
|
2008-12-06 05:49:23 +00:00
|
|
|
d.addCallback(lambda res: self.failUnlessCurrentSeqnumIs(n, 2, "m"))
|
2008-04-18 02:12:42 +00:00
|
|
|
|
|
|
|
d.addCallback(lambda res: n.modify(_non_modifier))
|
|
|
|
d.addCallback(lambda res: n.download_best_version())
|
|
|
|
d.addCallback(lambda res: self.failUnlessEqual(res, "line1line2"))
|
2008-12-06 05:49:23 +00:00
|
|
|
d.addCallback(lambda res: self.failUnlessCurrentSeqnumIs(n, 2, "non"))
|
2008-04-18 02:12:42 +00:00
|
|
|
|
|
|
|
d.addCallback(lambda res: n.modify(_none_modifier))
|
|
|
|
d.addCallback(lambda res: n.download_best_version())
|
|
|
|
d.addCallback(lambda res: self.failUnlessEqual(res, "line1line2"))
|
2008-12-06 05:49:23 +00:00
|
|
|
d.addCallback(lambda res: self.failUnlessCurrentSeqnumIs(n, 2, "none"))
|
2008-04-18 02:12:42 +00:00
|
|
|
|
|
|
|
d.addCallback(lambda res:
|
|
|
|
self.shouldFail(ValueError, "error_modifier", None,
|
|
|
|
n.modify, _error_modifier))
|
|
|
|
d.addCallback(lambda res: n.download_best_version())
|
|
|
|
d.addCallback(lambda res: self.failUnlessEqual(res, "line1line2"))
|
2008-12-06 05:49:23 +00:00
|
|
|
d.addCallback(lambda res: self.failUnlessCurrentSeqnumIs(n, 2, "err"))
|
2008-04-18 02:12:42 +00:00
|
|
|
|
2009-06-20 21:28:22 +00:00
|
|
|
|
2008-06-03 07:02:10 +00:00
|
|
|
d.addCallback(lambda res: n.download_best_version())
|
|
|
|
d.addCallback(lambda res: self.failUnlessEqual(res, "line1line2"))
|
2008-12-06 05:49:23 +00:00
|
|
|
d.addCallback(lambda res: self.failUnlessCurrentSeqnumIs(n, 2, "big"))
|
2008-06-03 07:02:10 +00:00
|
|
|
|
2008-04-18 02:12:42 +00:00
|
|
|
d.addCallback(lambda res: n.modify(_ucw_error_modifier))
|
|
|
|
d.addCallback(lambda res: self.failUnlessEqual(len(calls), 2))
|
|
|
|
d.addCallback(lambda res: n.download_best_version())
|
|
|
|
d.addCallback(lambda res: self.failUnlessEqual(res,
|
|
|
|
"line1line2line3"))
|
2008-12-06 05:49:23 +00:00
|
|
|
d.addCallback(lambda res: self.failUnlessCurrentSeqnumIs(n, 3, "ucw"))
|
|
|
|
|
|
|
|
def _reset_ucw_error_modifier(res):
|
|
|
|
calls[:] = []
|
|
|
|
return res
|
|
|
|
d.addCallback(_reset_ucw_error_modifier)
|
|
|
|
|
|
|
|
# in practice, this n.modify call should publish twice: the first
|
|
|
|
# one gets a UCWE, the second does not. But our test jig (in
|
|
|
|
# which the modifier raises the UCWE) skips over the first one,
|
|
|
|
# so in this test there will be only one publish, and the seqnum
|
|
|
|
# will only be one larger than the previous test, not two (i.e. 4
|
|
|
|
# instead of 5).
|
|
|
|
d.addCallback(lambda res: n.modify(_ucw_error_non_modifier))
|
|
|
|
d.addCallback(lambda res: self.failUnlessEqual(len(calls), 2))
|
|
|
|
d.addCallback(lambda res: n.download_best_version())
|
|
|
|
d.addCallback(lambda res: self.failUnlessEqual(res,
|
|
|
|
"line1line2line3"))
|
|
|
|
d.addCallback(lambda res: self.failUnlessCurrentSeqnumIs(n, 4, "ucw"))
|
2009-06-20 21:28:22 +00:00
|
|
|
d.addCallback(lambda res: n.modify(_toobig_modifier))
|
2008-04-18 02:12:42 +00:00
|
|
|
return d
|
|
|
|
d.addCallback(_created)
|
|
|
|
return d
|
|
|
|
|
|
|
|
def test_modify_backoffer(self):
|
2008-12-06 05:07:10 +00:00
|
|
|
def _modifier(old_contents, servermap, first_time):
|
2008-04-18 02:12:42 +00:00
|
|
|
return old_contents + "line2"
|
|
|
|
calls = []
|
2008-12-06 05:07:10 +00:00
|
|
|
def _ucw_error_modifier(old_contents, servermap, first_time):
|
2008-04-18 02:12:42 +00:00
|
|
|
# simulate an UncoordinatedWriteError once
|
|
|
|
calls.append(1)
|
|
|
|
if len(calls) <= 1:
|
|
|
|
raise UncoordinatedWriteError("simulated")
|
|
|
|
return old_contents + "line3"
|
2008-12-06 05:07:10 +00:00
|
|
|
def _always_ucw_error_modifier(old_contents, servermap, first_time):
|
2008-04-18 02:12:42 +00:00
|
|
|
raise UncoordinatedWriteError("simulated")
|
|
|
|
def _backoff_stopper(node, f):
|
|
|
|
return f
|
|
|
|
def _backoff_pauser(node, f):
|
|
|
|
d = defer.Deferred()
|
|
|
|
reactor.callLater(0.5, d.callback, None)
|
|
|
|
return d
|
|
|
|
|
|
|
|
# the give-up-er will hit its maximum retry count quickly
|
|
|
|
giveuper = BackoffAgent()
|
|
|
|
giveuper._delay = 0.1
|
|
|
|
giveuper.factor = 1
|
|
|
|
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
d = self.nodemaker.create_mutable_file("line1")
|
2008-04-18 02:12:42 +00:00
|
|
|
def _created(n):
|
|
|
|
d = n.modify(_modifier)
|
|
|
|
d.addCallback(lambda res: n.download_best_version())
|
|
|
|
d.addCallback(lambda res: self.failUnlessEqual(res, "line1line2"))
|
2008-12-06 05:49:23 +00:00
|
|
|
d.addCallback(lambda res: self.failUnlessCurrentSeqnumIs(n, 2, "m"))
|
2008-04-18 02:12:42 +00:00
|
|
|
|
|
|
|
d.addCallback(lambda res:
|
|
|
|
self.shouldFail(UncoordinatedWriteError,
|
|
|
|
"_backoff_stopper", None,
|
|
|
|
n.modify, _ucw_error_modifier,
|
|
|
|
_backoff_stopper))
|
|
|
|
d.addCallback(lambda res: n.download_best_version())
|
|
|
|
d.addCallback(lambda res: self.failUnlessEqual(res, "line1line2"))
|
2008-12-06 05:49:23 +00:00
|
|
|
d.addCallback(lambda res: self.failUnlessCurrentSeqnumIs(n, 2, "stop"))
|
2008-04-18 02:12:42 +00:00
|
|
|
|
|
|
|
def _reset_ucw_error_modifier(res):
|
|
|
|
calls[:] = []
|
|
|
|
return res
|
|
|
|
d.addCallback(_reset_ucw_error_modifier)
|
|
|
|
d.addCallback(lambda res: n.modify(_ucw_error_modifier,
|
|
|
|
_backoff_pauser))
|
|
|
|
d.addCallback(lambda res: n.download_best_version())
|
|
|
|
d.addCallback(lambda res: self.failUnlessEqual(res,
|
|
|
|
"line1line2line3"))
|
2008-12-06 05:49:23 +00:00
|
|
|
d.addCallback(lambda res: self.failUnlessCurrentSeqnumIs(n, 3, "pause"))
|
2008-04-18 02:12:42 +00:00
|
|
|
|
|
|
|
d.addCallback(lambda res:
|
|
|
|
self.shouldFail(UncoordinatedWriteError,
|
|
|
|
"giveuper", None,
|
|
|
|
n.modify, _always_ucw_error_modifier,
|
|
|
|
giveuper.delay))
|
|
|
|
d.addCallback(lambda res: n.download_best_version())
|
|
|
|
d.addCallback(lambda res: self.failUnlessEqual(res,
|
|
|
|
"line1line2line3"))
|
2008-12-06 05:49:23 +00:00
|
|
|
d.addCallback(lambda res: self.failUnlessCurrentSeqnumIs(n, 3, "giveup"))
|
2008-04-18 02:12:42 +00:00
|
|
|
|
|
|
|
return d
|
|
|
|
d.addCallback(_created)
|
|
|
|
return d
|
|
|
|
|
2008-04-11 01:44:06 +00:00
|
|
|
def test_upload_and_download_full_size_keys(self):
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
self.nodemaker.key_generator = client.KeyGenerator()
|
|
|
|
d = self.nodemaker.create_mutable_file()
|
2008-04-11 01:44:06 +00:00
|
|
|
def _created(n):
|
|
|
|
d = defer.succeed(None)
|
2008-04-18 00:51:38 +00:00
|
|
|
d.addCallback(lambda res: n.get_servermap(MODE_READ))
|
2008-04-11 01:44:06 +00:00
|
|
|
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))
|
2008-04-18 00:51:38 +00:00
|
|
|
d.addCallback(lambda res: n.download_best_version())
|
2008-04-11 01:44:06 +00:00
|
|
|
d.addCallback(lambda res: self.failUnlessEqual(res, "contents 1"))
|
|
|
|
d.addCallback(lambda res: n.overwrite("contents 2"))
|
2008-04-18 00:51:38 +00:00
|
|
|
d.addCallback(lambda res: n.download_best_version())
|
2008-04-11 01:44:06 +00:00
|
|
|
d.addCallback(lambda res: self.failUnlessEqual(res, "contents 2"))
|
2008-04-18 00:51:38 +00:00
|
|
|
d.addCallback(lambda res: n.get_servermap(MODE_WRITE))
|
|
|
|
d.addCallback(lambda smap: n.upload("contents 3", smap))
|
|
|
|
d.addCallback(lambda res: n.download_best_version())
|
|
|
|
d.addCallback(lambda res: self.failUnlessEqual(res, "contents 3"))
|
|
|
|
d.addCallback(lambda res: n.get_servermap(MODE_ANYTHING))
|
|
|
|
d.addCallback(lambda smap:
|
|
|
|
n.download_version(smap,
|
|
|
|
smap.best_recoverable_version()))
|
2008-04-11 01:44:06 +00:00
|
|
|
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):
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
nm = make_nodemaker()
|
2007-11-03 03:51:39 +00:00
|
|
|
CONTENTS = "some initial contents"
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
d = nm.create_mutable_file(CONTENTS)
|
|
|
|
def _created(fn):
|
|
|
|
p = Publish(fn, nm.storage_broker, 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):
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
nm = make_nodemaker()
|
2007-11-03 03:51:39 +00:00
|
|
|
CONTENTS = "some initial contents"
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
d = nm.create_mutable_file(CONTENTS)
|
|
|
|
def _created(fn):
|
|
|
|
self._fn = fn
|
|
|
|
p = Publish(fn, nm.storage_broker, None)
|
2008-04-17 20:09:22 +00:00
|
|
|
self._p = p
|
|
|
|
p.newdata = CONTENTS
|
|
|
|
p.required_shares = 3
|
|
|
|
p.total_shares = 10
|
|
|
|
p.setup_encoding_parameters()
|
|
|
|
p._new_seqnum = 3
|
|
|
|
p.salt = "SALT" * 4
|
|
|
|
# make some fake shares
|
|
|
|
shares_and_ids = ( ["%07d" % i for i in range(10)], range(10) )
|
|
|
|
p._privkey = fn.get_privkey()
|
|
|
|
p._encprivkey = fn.get_encprivkey()
|
|
|
|
p._pubkey = fn.get_pubkey()
|
|
|
|
return p._generate_shares(shares_and_ids)
|
|
|
|
d.addCallback(_created)
|
|
|
|
def _generated(res):
|
|
|
|
p = self._p
|
|
|
|
final_shares = p.shares
|
|
|
|
root_hash = p.root_hash
|
2007-11-03 03:51:39 +00:00
|
|
|
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))
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
self.failUnlessEqual(enc_privkey, self._fn.get_encprivkey())
|
2008-04-11 01:44:06 +00:00
|
|
|
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-08-06 17:38:04 +00:00
|
|
|
class PublishMixin:
|
|
|
|
def publish_one(self):
|
2008-04-05 00:09:26 +00:00
|
|
|
# publish a file and create shares, which can then be manipulated
|
|
|
|
# later.
|
2008-08-06 17:38:04 +00:00
|
|
|
self.CONTENTS = "New contents go here" * 1000
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
self._storage = FakeStorage()
|
|
|
|
self._nodemaker = make_nodemaker(self._storage)
|
|
|
|
self._storage_broker = self._nodemaker.storage_broker
|
|
|
|
d = self._nodemaker.create_mutable_file(self.CONTENTS)
|
2008-04-11 01:44:06 +00:00
|
|
|
def _created(node):
|
|
|
|
self._fn = node
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
self._fn2 = self._nodemaker.create_from_cap(node.get_uri())
|
2008-04-05 00:09:26 +00:00
|
|
|
d.addCallback(_created)
|
|
|
|
return d
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
|
2008-08-06 17:38:04 +00:00
|
|
|
def publish_multiple(self):
|
|
|
|
self.CONTENTS = ["Contents 0",
|
|
|
|
"Contents 1",
|
|
|
|
"Contents 2",
|
|
|
|
"Contents 3a",
|
|
|
|
"Contents 3b"]
|
|
|
|
self._copied_shares = {}
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
self._storage = FakeStorage()
|
|
|
|
self._nodemaker = make_nodemaker(self._storage)
|
|
|
|
d = self._nodemaker.create_mutable_file(self.CONTENTS[0]) # seqnum=1
|
2008-08-06 17:38:04 +00:00
|
|
|
def _created(node):
|
|
|
|
self._fn = node
|
|
|
|
# now create multiple versions of the same file, and accumulate
|
|
|
|
# their shares, so we can mix and match them later.
|
|
|
|
d = defer.succeed(None)
|
|
|
|
d.addCallback(self._copy_shares, 0)
|
|
|
|
d.addCallback(lambda res: node.overwrite(self.CONTENTS[1])) #s2
|
|
|
|
d.addCallback(self._copy_shares, 1)
|
|
|
|
d.addCallback(lambda res: node.overwrite(self.CONTENTS[2])) #s3
|
|
|
|
d.addCallback(self._copy_shares, 2)
|
|
|
|
d.addCallback(lambda res: node.overwrite(self.CONTENTS[3])) #s4a
|
|
|
|
d.addCallback(self._copy_shares, 3)
|
|
|
|
# now we replace all the shares with version s3, and upload a new
|
|
|
|
# version to get s4b.
|
|
|
|
rollback = dict([(i,2) for i in range(10)])
|
|
|
|
d.addCallback(lambda res: self._set_versions(rollback))
|
|
|
|
d.addCallback(lambda res: node.overwrite(self.CONTENTS[4])) #s4b
|
|
|
|
d.addCallback(self._copy_shares, 4)
|
|
|
|
# we leave the storage in state 4
|
|
|
|
return d
|
|
|
|
d.addCallback(_created)
|
|
|
|
return d
|
|
|
|
|
|
|
|
def _copy_shares(self, ignored, index):
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
shares = self._storage._peers
|
2008-08-06 17:38:04 +00:00
|
|
|
# we need a deep copy
|
|
|
|
new_shares = {}
|
|
|
|
for peerid in shares:
|
|
|
|
new_shares[peerid] = {}
|
|
|
|
for shnum in shares[peerid]:
|
|
|
|
new_shares[peerid][shnum] = shares[peerid][shnum]
|
|
|
|
self._copied_shares[index] = new_shares
|
|
|
|
|
|
|
|
def _set_versions(self, versionmap):
|
|
|
|
# versionmap maps shnums to which version (0,1,2,3,4) we want the
|
|
|
|
# share to be at. Any shnum which is left out of the map will stay at
|
|
|
|
# its current version.
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
shares = self._storage._peers
|
2008-08-06 17:38:04 +00:00
|
|
|
oldshares = self._copied_shares
|
|
|
|
for peerid in shares:
|
|
|
|
for shnum in shares[peerid]:
|
|
|
|
if shnum in versionmap:
|
|
|
|
index = versionmap[shnum]
|
|
|
|
shares[peerid][shnum] = oldshares[index][peerid][shnum]
|
|
|
|
|
|
|
|
|
|
|
|
class Servermap(unittest.TestCase, PublishMixin):
|
|
|
|
def setUp(self):
|
|
|
|
return self.publish_one()
|
2008-04-05 00:09:26 +00:00
|
|
|
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
def make_servermap(self, mode=MODE_CHECK, fn=None, sb=None):
|
2008-04-19 02:55:12 +00:00
|
|
|
if fn is None:
|
|
|
|
fn = self._fn
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
if sb is None:
|
|
|
|
sb = self._storage_broker
|
|
|
|
smu = ServermapUpdater(fn, sb, Monitor(),
|
|
|
|
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):
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
smu = ServermapUpdater(self._fn, self._storage_broker, Monitor(),
|
|
|
|
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)
|
2008-07-07 22:06:15 +00:00
|
|
|
self.failUnlessEqual(sm.shares_available()[best], (num_shares, 3, 10))
|
2008-04-19 03:13:28 +00:00
|
|
|
shnum, peerids = sm.make_sharemap().items()[0]
|
|
|
|
peerid = list(peerids)[0]
|
|
|
|
self.failUnlessEqual(sm.version_on_peer(peerid, shnum), best)
|
|
|
|
self.failUnlessEqual(sm.version_on_peer(peerid, 666), None)
|
2008-04-05 00:09:26 +00:00
|
|
|
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))
|
2009-12-08 17:26:08 +00:00
|
|
|
# this mode stops at k+epsilon, and epsilon=k, so 6 shares
|
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 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))
|
2008-04-19 02:55:12 +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_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-19 02:55:12 +00:00
|
|
|
def test_fetch_privkey(self):
|
|
|
|
d = defer.succeed(None)
|
|
|
|
# use the sibling filenode (which hasn't been used yet), and make
|
|
|
|
# sure it can fetch the privkey. The file is small, so the privkey
|
|
|
|
# will be fetched on the first (query) pass.
|
|
|
|
d.addCallback(lambda res: self.make_servermap(MODE_WRITE, self._fn2))
|
|
|
|
d.addCallback(lambda sm: self.failUnlessOneRecoverable(sm, 10))
|
|
|
|
|
|
|
|
# create a new file, which is large enough to knock the privkey out
|
2008-04-22 23:47:52 +00:00
|
|
|
# of the early part of the file
|
2008-04-19 02:55:12 +00:00
|
|
|
LARGE = "These are Larger contents" * 200 # about 5KB
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
d.addCallback(lambda res: self._nodemaker.create_mutable_file(LARGE))
|
2008-04-19 02:55:12 +00:00
|
|
|
def _created(large_fn):
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
large_fn2 = self._nodemaker.create_from_cap(large_fn.get_uri())
|
2008-04-19 02:55:12 +00:00
|
|
|
return self.make_servermap(MODE_WRITE, large_fn2)
|
|
|
|
d.addCallback(_created)
|
|
|
|
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
|
|
|
|
|
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) )
|
2008-07-18 04:09:23 +00:00
|
|
|
sm.mark_bad_share(peerid, shnum, "")
|
2008-04-16 21:49:47 +00:00
|
|
|
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):
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
self._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)
|
2008-07-07 22:06:15 +00:00
|
|
|
self.failUnlessEqual(sm.shares_available().values()[0], (2,3,10) )
|
2008-04-19 02:55:12 +00:00
|
|
|
return sm
|
2008-04-05 00:09:26 +00:00
|
|
|
|
|
|
|
def test_not_quite_enough_shares(self):
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
s = self._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-19 02:55:12 +00:00
|
|
|
d.addCallback(lambda sm:
|
|
|
|
self.failUnlessEqual(len(sm.make_sharemap()), 2))
|
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-08-06 17:38:04 +00:00
|
|
|
class Roundtrip(unittest.TestCase, testutil.ShouldFailMixin, PublishMixin):
|
2008-04-05 00:09:26 +00:00
|
|
|
def setUp(self):
|
2008-08-06 17:38:04 +00:00
|
|
|
return self.publish_one()
|
2008-04-05 00:09:26 +00:00
|
|
|
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
def make_servermap(self, mode=MODE_READ, oldmap=None, sb=None):
|
2008-04-05 00:09:26 +00:00
|
|
|
if oldmap is None:
|
2008-04-11 21:31:16 +00:00
|
|
|
oldmap = ServerMap()
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
if sb is None:
|
|
|
|
sb = self._storage_broker
|
|
|
|
smu = ServermapUpdater(self._fn, sb, Monitor(), 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-07-07 19:18:10 +00:00
|
|
|
def test_all_shares_vanished(self):
|
|
|
|
d = self.make_servermap()
|
|
|
|
def _remove_shares(servermap):
|
|
|
|
for shares in self._storage._peers.values():
|
|
|
|
shares.clear()
|
|
|
|
d1 = self.shouldFail(NotEnoughSharesError,
|
|
|
|
"test_all_shares_vanished",
|
|
|
|
"ran out of peers",
|
|
|
|
self.do_download, servermap)
|
|
|
|
return d1
|
|
|
|
d.addCallback(_remove_shares)
|
|
|
|
return d
|
|
|
|
|
|
|
|
def test_no_servers(self):
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
sb2 = make_storagebroker(num_peers=0)
|
2008-07-07 19:18:10 +00:00
|
|
|
# if there are no servers, then a MODE_READ servermap should come
|
|
|
|
# back empty
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
d = self.make_servermap(sb=sb2)
|
2008-07-07 19:18:10 +00:00
|
|
|
def _check_servermap(servermap):
|
|
|
|
self.failUnlessEqual(servermap.best_recoverable_version(), None)
|
|
|
|
self.failIf(servermap.recoverable_versions())
|
|
|
|
self.failIf(servermap.unrecoverable_versions())
|
|
|
|
self.failIf(servermap.all_peers())
|
|
|
|
d.addCallback(_check_servermap)
|
|
|
|
return d
|
|
|
|
test_no_servers.timeout = 15
|
|
|
|
|
|
|
|
def test_no_servers_download(self):
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
sb2 = make_storagebroker(num_peers=0)
|
|
|
|
self._fn._storage_broker = sb2
|
2008-07-07 19:18:10 +00:00
|
|
|
d = self.shouldFail(UnrecoverableFileError,
|
|
|
|
"test_no_servers_download",
|
|
|
|
"no recoverable versions",
|
|
|
|
self._fn.download_best_version)
|
|
|
|
def _restore(res):
|
|
|
|
# a failed download that occurs while we aren't connected to
|
|
|
|
# anybody should not prevent a subsequent download from working.
|
|
|
|
# This isn't quite the webapi-driven test that #463 wants, but it
|
|
|
|
# should be close enough.
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
self._fn._storage_broker = self._storage_broker
|
2008-07-07 19:18:10 +00:00
|
|
|
return self._fn.download_best_version()
|
|
|
|
def _retrieved(new_contents):
|
|
|
|
self.failUnlessEqual(new_contents, self.CONTENTS)
|
|
|
|
d.addCallback(_restore)
|
|
|
|
d.addCallback(_retrieved)
|
|
|
|
return d
|
|
|
|
test_no_servers_download.timeout = 15
|
2008-03-11 00:46:52 +00:00
|
|
|
|
2008-04-05 00:09:26 +00:00
|
|
|
def _test_corrupt_all(self, offset, substring,
|
2008-06-24 18:08:10 +00:00
|
|
|
should_succeed=False, corrupt_early=True,
|
|
|
|
failure_checker=None):
|
2008-04-05 00:09:26 +00:00
|
|
|
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]
|
2009-07-15 06:45:10 +00:00
|
|
|
self.failUnlessIn(substring, "".join(allproblems))
|
2008-04-19 02:55:12 +00:00
|
|
|
return servermap
|
2008-04-05 00:09:26 +00:00
|
|
|
if should_succeed:
|
2008-06-24 18:08:10 +00:00
|
|
|
d1 = self._fn.download_version(servermap, ver)
|
2008-04-05 00:09:26 +00:00
|
|
|
d1.addCallback(lambda new_contents:
|
|
|
|
self.failUnlessEqual(new_contents, self.CONTENTS))
|
|
|
|
else:
|
2008-04-19 02:55:12 +00:00
|
|
|
d1 = self.shouldFail(NotEnoughSharesError,
|
|
|
|
"_corrupt_all(offset=%s)" % (offset,),
|
|
|
|
substring,
|
2008-06-24 18:08:10 +00:00
|
|
|
self._fn.download_version, servermap, ver)
|
|
|
|
if failure_checker:
|
|
|
|
d1.addCallback(failure_checker)
|
2008-04-19 02:55:12 +00:00
|
|
|
d1.addCallback(lambda res: servermap)
|
|
|
|
return d1
|
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):
|
2009-07-15 06:45:10 +00:00
|
|
|
# when the version byte is not 0, we hit an UnknownVersionError error
|
|
|
|
# in unpack_share().
|
|
|
|
d = self._test_corrupt_all(0, "UnknownVersionError")
|
2008-04-19 02:55:12 +00:00
|
|
|
def _check_servermap(servermap):
|
|
|
|
# and the dump should mention the problems
|
|
|
|
s = StringIO()
|
|
|
|
dump = servermap.dump(s).getvalue()
|
|
|
|
self.failUnless("10 PROBLEMS" in dump, dump)
|
|
|
|
d.addCallback(_check_servermap)
|
|
|
|
return d
|
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
|
|
|
|
2008-06-24 18:08:10 +00:00
|
|
|
|
|
|
|
def test_corrupt_all_seqnum_late(self):
|
|
|
|
# corrupting the seqnum between mapupdate and retrieve should result
|
|
|
|
# in NotEnoughSharesError, since each share will look invalid
|
|
|
|
def _check(res):
|
|
|
|
f = res[0]
|
|
|
|
self.failUnless(f.check(NotEnoughSharesError))
|
|
|
|
self.failUnless("someone wrote to the data since we read the servermap" in str(f))
|
|
|
|
return self._test_corrupt_all(1, "ran out of peers",
|
|
|
|
corrupt_early=False,
|
|
|
|
failure_checker=_check)
|
|
|
|
|
|
|
|
def test_corrupt_all_block_hash_tree_late(self):
|
|
|
|
def _check(res):
|
|
|
|
f = res[0]
|
|
|
|
self.failUnless(f.check(NotEnoughSharesError))
|
|
|
|
return self._test_corrupt_all("block_hash_tree",
|
|
|
|
"block hash tree failure",
|
|
|
|
corrupt_early=False,
|
|
|
|
failure_checker=_check)
|
|
|
|
|
|
|
|
|
|
|
|
def test_corrupt_all_block_late(self):
|
|
|
|
def _check(res):
|
|
|
|
f = res[0]
|
|
|
|
self.failUnless(f.check(NotEnoughSharesError))
|
|
|
|
return self._test_corrupt_all("share_data", "block hash tree failure",
|
|
|
|
corrupt_early=False,
|
|
|
|
failure_checker=_check)
|
|
|
|
|
|
|
|
|
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)
|
2008-04-18 00:51:38 +00:00
|
|
|
return self._fn.download_best_version()
|
2008-04-16 21:49:47 +00:00
|
|
|
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",
|
2008-04-18 00:51:38 +00:00
|
|
|
self._fn.download_best_version)
|
2008-04-16 21:49:47 +00:00
|
|
|
return d
|
|
|
|
|
2008-04-11 01:44:06 +00:00
|
|
|
|
2008-07-08 00:36:00 +00:00
|
|
|
class CheckerMixin:
|
|
|
|
def check_good(self, r, where):
|
2008-09-07 19:44:56 +00:00
|
|
|
self.failUnless(r.is_healthy(), where)
|
2008-07-08 00:36:00 +00:00
|
|
|
return r
|
|
|
|
|
|
|
|
def check_bad(self, r, where):
|
2008-09-07 19:44:56 +00:00
|
|
|
self.failIf(r.is_healthy(), where)
|
2008-07-08 00:36:00 +00:00
|
|
|
return r
|
|
|
|
|
|
|
|
def check_expected_failure(self, r, expected_exception, substring, where):
|
2008-07-16 00:23:25 +00:00
|
|
|
for (peerid, storage_index, shnum, f) in r.problems:
|
2008-07-08 00:36:00 +00:00
|
|
|
if f.check(expected_exception):
|
|
|
|
self.failUnless(substring in str(f),
|
|
|
|
"%s: substring '%s' not in '%s'" %
|
|
|
|
(where, substring, str(f)))
|
|
|
|
return
|
|
|
|
self.fail("%s: didn't see expected exception %s in problems %s" %
|
2008-07-16 00:23:25 +00:00
|
|
|
(where, expected_exception, r.problems))
|
2008-07-08 00:36:00 +00:00
|
|
|
|
|
|
|
|
2008-08-06 17:38:04 +00:00
|
|
|
class Checker(unittest.TestCase, CheckerMixin, PublishMixin):
|
2008-07-08 00:36:00 +00:00
|
|
|
def setUp(self):
|
2008-08-06 17:38:04 +00:00
|
|
|
return self.publish_one()
|
2008-07-08 00:36:00 +00:00
|
|
|
|
|
|
|
|
|
|
|
def test_check_good(self):
|
2008-10-22 08:38:18 +00:00
|
|
|
d = self._fn.check(Monitor())
|
2008-07-08 00:36:00 +00:00
|
|
|
d.addCallback(self.check_good, "test_check_good")
|
|
|
|
return d
|
|
|
|
|
|
|
|
def test_check_no_shares(self):
|
|
|
|
for shares in self._storage._peers.values():
|
|
|
|
shares.clear()
|
2008-10-22 08:38:18 +00:00
|
|
|
d = self._fn.check(Monitor())
|
2008-07-08 00:36:00 +00:00
|
|
|
d.addCallback(self.check_bad, "test_check_no_shares")
|
|
|
|
return d
|
|
|
|
|
|
|
|
def test_check_not_enough_shares(self):
|
|
|
|
for shares in self._storage._peers.values():
|
|
|
|
for shnum in shares.keys():
|
|
|
|
if shnum > 0:
|
|
|
|
del shares[shnum]
|
2008-10-22 08:38:18 +00:00
|
|
|
d = self._fn.check(Monitor())
|
2008-07-08 00:36:00 +00:00
|
|
|
d.addCallback(self.check_bad, "test_check_not_enough_shares")
|
|
|
|
return d
|
|
|
|
|
|
|
|
def test_check_all_bad_sig(self):
|
|
|
|
corrupt(None, self._storage, 1) # bad sig
|
2008-10-22 08:38:18 +00:00
|
|
|
d = self._fn.check(Monitor())
|
2008-07-08 00:36:00 +00:00
|
|
|
d.addCallback(self.check_bad, "test_check_all_bad_sig")
|
|
|
|
return d
|
|
|
|
|
|
|
|
def test_check_all_bad_blocks(self):
|
|
|
|
corrupt(None, self._storage, "share_data", [9]) # bad blocks
|
|
|
|
# the Checker won't notice this.. it doesn't look at actual data
|
2008-10-22 08:38:18 +00:00
|
|
|
d = self._fn.check(Monitor())
|
2008-07-08 00:36:00 +00:00
|
|
|
d.addCallback(self.check_good, "test_check_all_bad_blocks")
|
|
|
|
return d
|
|
|
|
|
|
|
|
def test_verify_good(self):
|
2008-10-22 08:38:18 +00:00
|
|
|
d = self._fn.check(Monitor(), verify=True)
|
2008-07-08 00:36:00 +00:00
|
|
|
d.addCallback(self.check_good, "test_verify_good")
|
|
|
|
return d
|
|
|
|
|
|
|
|
def test_verify_all_bad_sig(self):
|
|
|
|
corrupt(None, self._storage, 1) # bad sig
|
2008-10-22 08:38:18 +00:00
|
|
|
d = self._fn.check(Monitor(), verify=True)
|
2008-07-08 00:36:00 +00:00
|
|
|
d.addCallback(self.check_bad, "test_verify_all_bad_sig")
|
|
|
|
return d
|
|
|
|
|
|
|
|
def test_verify_one_bad_sig(self):
|
|
|
|
corrupt(None, self._storage, 1, [9]) # bad sig
|
2008-10-22 08:38:18 +00:00
|
|
|
d = self._fn.check(Monitor(), verify=True)
|
2008-07-08 00:36:00 +00:00
|
|
|
d.addCallback(self.check_bad, "test_verify_one_bad_sig")
|
|
|
|
return d
|
|
|
|
|
|
|
|
def test_verify_one_bad_block(self):
|
|
|
|
corrupt(None, self._storage, "share_data", [9]) # bad blocks
|
|
|
|
# the Verifier *will* notice this, since it examines every byte
|
2008-10-22 08:38:18 +00:00
|
|
|
d = self._fn.check(Monitor(), verify=True)
|
2008-07-08 00:36:00 +00:00
|
|
|
d.addCallback(self.check_bad, "test_verify_one_bad_block")
|
|
|
|
d.addCallback(self.check_expected_failure,
|
|
|
|
CorruptShareError, "block hash tree failure",
|
|
|
|
"test_verify_one_bad_block")
|
|
|
|
return d
|
|
|
|
|
|
|
|
def test_verify_one_bad_sharehash(self):
|
|
|
|
corrupt(None, self._storage, "share_hash_chain", [9], 5)
|
2008-10-22 08:38:18 +00:00
|
|
|
d = self._fn.check(Monitor(), verify=True)
|
2008-07-08 00:36:00 +00:00
|
|
|
d.addCallback(self.check_bad, "test_verify_one_bad_sharehash")
|
|
|
|
d.addCallback(self.check_expected_failure,
|
|
|
|
CorruptShareError, "corrupt hashes",
|
|
|
|
"test_verify_one_bad_sharehash")
|
|
|
|
return d
|
|
|
|
|
|
|
|
def test_verify_one_bad_encprivkey(self):
|
|
|
|
corrupt(None, self._storage, "enc_privkey", [9]) # bad privkey
|
2008-10-22 08:38:18 +00:00
|
|
|
d = self._fn.check(Monitor(), verify=True)
|
2008-07-08 00:36:00 +00:00
|
|
|
d.addCallback(self.check_bad, "test_verify_one_bad_encprivkey")
|
|
|
|
d.addCallback(self.check_expected_failure,
|
|
|
|
CorruptShareError, "invalid privkey",
|
|
|
|
"test_verify_one_bad_encprivkey")
|
|
|
|
return d
|
|
|
|
|
|
|
|
def test_verify_one_bad_encprivkey_uncheckable(self):
|
|
|
|
corrupt(None, self._storage, "enc_privkey", [9]) # bad privkey
|
|
|
|
readonly_fn = self._fn.get_readonly()
|
|
|
|
# a read-only node has no way to validate the privkey
|
2008-10-22 08:38:18 +00:00
|
|
|
d = readonly_fn.check(Monitor(), verify=True)
|
2008-07-08 00:36:00 +00:00
|
|
|
d.addCallback(self.check_good,
|
|
|
|
"test_verify_one_bad_encprivkey_uncheckable")
|
|
|
|
return d
|
|
|
|
|
2008-08-06 19:06:07 +00:00
|
|
|
class Repair(unittest.TestCase, PublishMixin, ShouldFailMixin):
|
2008-08-06 06:12:39 +00:00
|
|
|
|
|
|
|
def get_shares(self, s):
|
|
|
|
all_shares = {} # maps (peerid, shnum) to share data
|
|
|
|
for peerid in s._peers:
|
|
|
|
shares = s._peers[peerid]
|
|
|
|
for shnum in shares:
|
|
|
|
data = shares[shnum]
|
|
|
|
all_shares[ (peerid, shnum) ] = data
|
|
|
|
return all_shares
|
|
|
|
|
2008-08-06 19:06:07 +00:00
|
|
|
def copy_shares(self, ignored=None):
|
|
|
|
self.old_shares.append(self.get_shares(self._storage))
|
|
|
|
|
2008-08-06 06:12:39 +00:00
|
|
|
def test_repair_nop(self):
|
2008-08-06 19:06:07 +00:00
|
|
|
self.old_shares = []
|
2008-08-06 17:38:04 +00:00
|
|
|
d = self.publish_one()
|
2008-08-06 19:06:07 +00:00
|
|
|
d.addCallback(self.copy_shares)
|
2008-10-22 08:38:18 +00:00
|
|
|
d.addCallback(lambda res: self._fn.check(Monitor()))
|
2008-08-06 17:38:04 +00:00
|
|
|
d.addCallback(lambda check_results: self._fn.repair(check_results))
|
2008-08-06 06:12:39 +00:00
|
|
|
def _check_results(rres):
|
|
|
|
self.failUnless(IRepairResults.providedBy(rres))
|
2009-12-29 23:37:46 +00:00
|
|
|
self.failUnless(rres.get_successful())
|
2008-08-06 06:12:39 +00:00
|
|
|
# TODO: examine results
|
|
|
|
|
2008-08-06 19:06:07 +00:00
|
|
|
self.copy_shares()
|
|
|
|
|
|
|
|
initial_shares = self.old_shares[0]
|
|
|
|
new_shares = self.old_shares[1]
|
2008-08-06 17:38:50 +00:00
|
|
|
# TODO: this really shouldn't change anything. When we implement
|
|
|
|
# a "minimal-bandwidth" repairer", change this test to assert:
|
|
|
|
#self.failUnlessEqual(new_shares, initial_shares)
|
|
|
|
|
2008-08-06 06:12:39 +00:00
|
|
|
# all shares should be in the same place as before
|
2008-08-06 19:06:07 +00:00
|
|
|
self.failUnlessEqual(set(initial_shares.keys()),
|
2008-08-06 06:12:39 +00:00
|
|
|
set(new_shares.keys()))
|
|
|
|
# but they should all be at a newer seqnum. The IV will be
|
|
|
|
# different, so the roothash will be too.
|
2008-08-06 19:06:07 +00:00
|
|
|
for key in initial_shares:
|
2008-08-06 06:12:39 +00:00
|
|
|
(version0,
|
|
|
|
seqnum0,
|
|
|
|
root_hash0,
|
|
|
|
IV0,
|
|
|
|
k0, N0, segsize0, datalen0,
|
2008-08-06 19:06:07 +00:00
|
|
|
o0) = unpack_header(initial_shares[key])
|
2008-08-06 06:12:39 +00:00
|
|
|
(version1,
|
|
|
|
seqnum1,
|
|
|
|
root_hash1,
|
|
|
|
IV1,
|
|
|
|
k1, N1, segsize1, datalen1,
|
|
|
|
o1) = unpack_header(new_shares[key])
|
|
|
|
self.failUnlessEqual(version0, version1)
|
|
|
|
self.failUnlessEqual(seqnum0+1, seqnum1)
|
|
|
|
self.failUnlessEqual(k0, k1)
|
|
|
|
self.failUnlessEqual(N0, N1)
|
|
|
|
self.failUnlessEqual(segsize0, segsize1)
|
|
|
|
self.failUnlessEqual(datalen0, datalen1)
|
|
|
|
d.addCallback(_check_results)
|
|
|
|
return d
|
|
|
|
|
2008-08-06 19:06:07 +00:00
|
|
|
def failIfSharesChanged(self, ignored=None):
|
|
|
|
old_shares = self.old_shares[-2]
|
|
|
|
current_shares = self.old_shares[-1]
|
|
|
|
self.failUnlessEqual(old_shares, current_shares)
|
|
|
|
|
2009-12-29 23:37:46 +00:00
|
|
|
def test_unrepairable_0shares(self):
|
|
|
|
d = self.publish_one()
|
|
|
|
def _delete_all_shares(ign):
|
|
|
|
shares = self._storage._peers
|
|
|
|
for peerid in shares:
|
|
|
|
shares[peerid] = {}
|
|
|
|
d.addCallback(_delete_all_shares)
|
|
|
|
d.addCallback(lambda ign: self._fn.check(Monitor()))
|
|
|
|
d.addCallback(lambda check_results: self._fn.repair(check_results))
|
|
|
|
def _check(crr):
|
|
|
|
self.failUnlessEqual(crr.get_successful(), False)
|
|
|
|
d.addCallback(_check)
|
|
|
|
return d
|
|
|
|
|
|
|
|
def test_unrepairable_1share(self):
|
|
|
|
d = self.publish_one()
|
|
|
|
def _delete_all_shares(ign):
|
|
|
|
shares = self._storage._peers
|
|
|
|
for peerid in shares:
|
|
|
|
for shnum in list(shares[peerid]):
|
|
|
|
if shnum > 0:
|
|
|
|
del shares[peerid][shnum]
|
|
|
|
d.addCallback(_delete_all_shares)
|
|
|
|
d.addCallback(lambda ign: self._fn.check(Monitor()))
|
|
|
|
d.addCallback(lambda check_results: self._fn.repair(check_results))
|
|
|
|
def _check(crr):
|
|
|
|
self.failUnlessEqual(crr.get_successful(), False)
|
|
|
|
d.addCallback(_check)
|
|
|
|
return d
|
|
|
|
|
2008-08-06 19:06:07 +00:00
|
|
|
def test_merge(self):
|
|
|
|
self.old_shares = []
|
|
|
|
d = self.publish_multiple()
|
|
|
|
# repair will refuse to merge multiple highest seqnums unless you
|
|
|
|
# pass force=True
|
|
|
|
d.addCallback(lambda res:
|
|
|
|
self._set_versions({0:3,2:3,4:3,6:3,8:3,
|
|
|
|
1:4,3:4,5:4,7:4,9:4}))
|
|
|
|
d.addCallback(self.copy_shares)
|
2008-10-22 08:38:18 +00:00
|
|
|
d.addCallback(lambda res: self._fn.check(Monitor()))
|
2008-08-06 19:06:07 +00:00
|
|
|
def _try_repair(check_results):
|
|
|
|
ex = "There were multiple recoverable versions with identical seqnums, so force=True must be passed to the repair() operation"
|
|
|
|
d2 = self.shouldFail(MustForceRepairError, "test_merge", ex,
|
|
|
|
self._fn.repair, check_results)
|
|
|
|
d2.addCallback(self.copy_shares)
|
|
|
|
d2.addCallback(self.failIfSharesChanged)
|
|
|
|
d2.addCallback(lambda res: check_results)
|
|
|
|
return d2
|
|
|
|
d.addCallback(_try_repair)
|
|
|
|
d.addCallback(lambda check_results:
|
|
|
|
self._fn.repair(check_results, force=True))
|
|
|
|
# this should give us 10 shares of the highest roothash
|
|
|
|
def _check_repair_results(rres):
|
2009-12-29 23:37:46 +00:00
|
|
|
self.failUnless(rres.get_successful())
|
2008-08-06 19:06:07 +00:00
|
|
|
pass # TODO
|
|
|
|
d.addCallback(_check_repair_results)
|
|
|
|
d.addCallback(lambda res: self._fn.get_servermap(MODE_CHECK))
|
|
|
|
def _check_smap(smap):
|
|
|
|
self.failUnlessEqual(len(smap.recoverable_versions()), 1)
|
|
|
|
self.failIf(smap.unrecoverable_versions())
|
|
|
|
# now, which should have won?
|
|
|
|
roothash_s4a = self.get_roothash_for(3)
|
|
|
|
roothash_s4b = self.get_roothash_for(4)
|
|
|
|
if roothash_s4b > roothash_s4a:
|
|
|
|
expected_contents = self.CONTENTS[4]
|
|
|
|
else:
|
|
|
|
expected_contents = self.CONTENTS[3]
|
|
|
|
new_versionid = smap.best_recoverable_version()
|
|
|
|
self.failUnlessEqual(new_versionid[0], 5) # seqnum 5
|
|
|
|
d2 = self._fn.download_version(smap, new_versionid)
|
|
|
|
d2.addCallback(self.failUnlessEqual, expected_contents)
|
|
|
|
return d2
|
|
|
|
d.addCallback(_check_smap)
|
|
|
|
return d
|
|
|
|
|
2008-10-24 04:00:24 +00:00
|
|
|
def test_non_merge(self):
|
|
|
|
self.old_shares = []
|
|
|
|
d = self.publish_multiple()
|
|
|
|
# repair should not refuse a repair that doesn't need to merge. In
|
|
|
|
# this case, we combine v2 with v3. The repair should ignore v2 and
|
|
|
|
# copy v3 into a new v5.
|
|
|
|
d.addCallback(lambda res:
|
|
|
|
self._set_versions({0:2,2:2,4:2,6:2,8:2,
|
|
|
|
1:3,3:3,5:3,7:3,9:3}))
|
|
|
|
d.addCallback(lambda res: self._fn.check(Monitor()))
|
|
|
|
d.addCallback(lambda check_results: self._fn.repair(check_results))
|
|
|
|
# this should give us 10 shares of v3
|
|
|
|
def _check_repair_results(rres):
|
2009-12-29 23:37:46 +00:00
|
|
|
self.failUnless(rres.get_successful())
|
2008-10-24 04:00:24 +00:00
|
|
|
pass # TODO
|
|
|
|
d.addCallback(_check_repair_results)
|
|
|
|
d.addCallback(lambda res: self._fn.get_servermap(MODE_CHECK))
|
|
|
|
def _check_smap(smap):
|
|
|
|
self.failUnlessEqual(len(smap.recoverable_versions()), 1)
|
|
|
|
self.failIf(smap.unrecoverable_versions())
|
|
|
|
# now, which should have won?
|
|
|
|
expected_contents = self.CONTENTS[3]
|
|
|
|
new_versionid = smap.best_recoverable_version()
|
|
|
|
self.failUnlessEqual(new_versionid[0], 5) # seqnum 5
|
|
|
|
d2 = self._fn.download_version(smap, new_versionid)
|
|
|
|
d2.addCallback(self.failUnlessEqual, expected_contents)
|
|
|
|
return d2
|
|
|
|
d.addCallback(_check_smap)
|
|
|
|
return d
|
|
|
|
|
2008-08-06 19:06:07 +00:00
|
|
|
def get_roothash_for(self, index):
|
|
|
|
# return the roothash for the first share we see in the saved set
|
|
|
|
shares = self._copied_shares[index]
|
|
|
|
for peerid in shares:
|
|
|
|
for shnum in shares[peerid]:
|
|
|
|
share = shares[peerid][shnum]
|
|
|
|
(version, seqnum, root_hash, IV, k, N, segsize, datalen, o) = \
|
|
|
|
unpack_header(share)
|
|
|
|
return root_hash
|
2008-07-08 00:36:00 +00:00
|
|
|
|
2009-07-01 01:13:43 +00:00
|
|
|
def test_check_and_repair_readcap(self):
|
|
|
|
# we can't currently repair from a mutable readcap: #625
|
|
|
|
self.old_shares = []
|
|
|
|
d = self.publish_one()
|
|
|
|
d.addCallback(self.copy_shares)
|
|
|
|
def _get_readcap(res):
|
|
|
|
self._fn3 = self._fn.get_readonly()
|
|
|
|
# also delete some shares
|
|
|
|
for peerid,shares in self._storage._peers.items():
|
|
|
|
shares.pop(0, None)
|
|
|
|
d.addCallback(_get_readcap)
|
|
|
|
d.addCallback(lambda res: self._fn3.check_and_repair(Monitor()))
|
|
|
|
def _check_results(crr):
|
|
|
|
self.failUnless(ICheckAndRepairResults.providedBy(crr))
|
|
|
|
# we should detect the unhealthy, but skip over mutable-readcap
|
|
|
|
# repairs until #625 is fixed
|
|
|
|
self.failIf(crr.get_pre_repair_results().is_healthy())
|
|
|
|
self.failIf(crr.get_repair_attempted())
|
|
|
|
self.failIf(crr.get_post_repair_results().is_healthy())
|
|
|
|
d.addCallback(_check_results)
|
|
|
|
return d
|
|
|
|
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
class DevNullDictionary(dict):
|
|
|
|
def __setitem__(self, key, value):
|
|
|
|
return
|
|
|
|
|
2008-04-11 01:44:06 +00:00
|
|
|
class MultipleEncodings(unittest.TestCase):
|
|
|
|
def setUp(self):
|
|
|
|
self.CONTENTS = "New contents go here"
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
self._storage = FakeStorage()
|
|
|
|
self._nodemaker = make_nodemaker(self._storage, num_peers=20)
|
|
|
|
self._storage_broker = self._nodemaker.storage_broker
|
|
|
|
d = self._nodemaker.create_mutable_file(self.CONTENTS)
|
2008-04-11 01:44:06 +00:00
|
|
|
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
|
|
|
fn = self._fn
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
# disable the nodecache, since for these tests we explicitly need
|
|
|
|
# multiple nodes pointing at the same file
|
|
|
|
self._nodemaker._node_cache = DevNullDictionary()
|
|
|
|
fn2 = self._nodemaker.create_from_cap(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
|
|
|
# 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
|
|
|
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
s = self._storage
|
2008-04-11 01:44:06 +00:00
|
|
|
s._peers = {} # clear existing storage
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
p2 = Publish(fn2, self._storage_broker, 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()
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
smu = ServermapUpdater(self._fn, self._storage_broker, Monitor(),
|
|
|
|
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
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
fn3 = self._nodemaker.create_from_cap(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 = {}
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
sb = self._storage_broker
|
2008-03-11 06:16:28 +00:00
|
|
|
|
2009-06-23 02:10:47 +00:00
|
|
|
for peerid in sorted(sb.get_all_serverids()):
|
2008-03-11 06:16:28 +00:00
|
|
|
for shnum in self._shares1.get(peerid, {}):
|
|
|
|
if shnum < len(places):
|
|
|
|
which = places[shnum]
|
|
|
|
else:
|
|
|
|
which = "x"
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
self._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, {})
|
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())]
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
self._storage._sequence = new_sequence
|
2008-03-11 05:15:43 +00:00
|
|
|
log.msg("merge done")
|
|
|
|
d.addCallback(_merge)
|
2008-04-18 00:51:38 +00:00
|
|
|
d.addCallback(lambda res: fn3.download_best_version())
|
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-19 02:55:12 +00:00
|
|
|
|
2008-08-06 17:38:04 +00:00
|
|
|
class MultipleVersions(unittest.TestCase, PublishMixin, CheckerMixin):
|
2008-04-19 02:55:12 +00:00
|
|
|
|
2008-08-06 17:38:04 +00:00
|
|
|
def setUp(self):
|
|
|
|
return self.publish_multiple()
|
2008-04-19 02:55:12 +00:00
|
|
|
|
|
|
|
def test_multiple_versions(self):
|
|
|
|
# if we see a mix of versions in the grid, download_best_version
|
|
|
|
# should get the latest one
|
|
|
|
self._set_versions(dict([(i,2) for i in (0,2,4,6,8)]))
|
|
|
|
d = self._fn.download_best_version()
|
|
|
|
d.addCallback(lambda res: self.failUnlessEqual(res, self.CONTENTS[4]))
|
2008-07-08 00:36:00 +00:00
|
|
|
# and the checker should report problems
|
2008-10-22 08:38:18 +00:00
|
|
|
d.addCallback(lambda res: self._fn.check(Monitor()))
|
2008-07-08 00:36:00 +00:00
|
|
|
d.addCallback(self.check_bad, "test_multiple_versions")
|
|
|
|
|
2008-04-19 02:55:12 +00:00
|
|
|
# but if everything is at version 2, that's what we should download
|
|
|
|
d.addCallback(lambda res:
|
|
|
|
self._set_versions(dict([(i,2) for i in range(10)])))
|
|
|
|
d.addCallback(lambda res: self._fn.download_best_version())
|
|
|
|
d.addCallback(lambda res: self.failUnlessEqual(res, self.CONTENTS[2]))
|
|
|
|
# if exactly one share is at version 3, we should still get v2
|
|
|
|
d.addCallback(lambda res:
|
|
|
|
self._set_versions({0:3}))
|
|
|
|
d.addCallback(lambda res: self._fn.download_best_version())
|
|
|
|
d.addCallback(lambda res: self.failUnlessEqual(res, self.CONTENTS[2]))
|
|
|
|
# but the servermap should see the unrecoverable version. This
|
|
|
|
# depends upon the single newer share being queried early.
|
|
|
|
d.addCallback(lambda res: self._fn.get_servermap(MODE_READ))
|
|
|
|
def _check_smap(smap):
|
|
|
|
self.failUnlessEqual(len(smap.unrecoverable_versions()), 1)
|
|
|
|
newer = smap.unrecoverable_newer_versions()
|
|
|
|
self.failUnlessEqual(len(newer), 1)
|
|
|
|
verinfo, health = newer.items()[0]
|
|
|
|
self.failUnlessEqual(verinfo[0], 4)
|
|
|
|
self.failUnlessEqual(health, (1,3))
|
|
|
|
self.failIf(smap.needs_merge())
|
|
|
|
d.addCallback(_check_smap)
|
|
|
|
# if we have a mix of two parallel versions (s4a and s4b), we could
|
|
|
|
# recover either
|
|
|
|
d.addCallback(lambda res:
|
|
|
|
self._set_versions({0:3,2:3,4:3,6:3,8:3,
|
|
|
|
1:4,3:4,5:4,7:4,9:4}))
|
|
|
|
d.addCallback(lambda res: self._fn.get_servermap(MODE_READ))
|
|
|
|
def _check_smap_mixed(smap):
|
|
|
|
self.failUnlessEqual(len(smap.unrecoverable_versions()), 0)
|
|
|
|
newer = smap.unrecoverable_newer_versions()
|
|
|
|
self.failUnlessEqual(len(newer), 0)
|
|
|
|
self.failUnless(smap.needs_merge())
|
|
|
|
d.addCallback(_check_smap_mixed)
|
|
|
|
d.addCallback(lambda res: self._fn.download_best_version())
|
|
|
|
d.addCallback(lambda res: self.failUnless(res == self.CONTENTS[3] or
|
|
|
|
res == self.CONTENTS[4]))
|
|
|
|
return d
|
|
|
|
|
2008-05-14 20:10:41 +00:00
|
|
|
def test_replace(self):
|
|
|
|
# if we see a mix of versions in the grid, we should be able to
|
|
|
|
# replace them all with a newer version
|
|
|
|
|
|
|
|
# if exactly one share is at version 3, we should download (and
|
|
|
|
# replace) v2, and the result should be v4. Note that the index we
|
|
|
|
# give to _set_versions is different than the sequence number.
|
|
|
|
target = dict([(i,2) for i in range(10)]) # seqnum3
|
|
|
|
target[0] = 3 # seqnum4
|
|
|
|
self._set_versions(target)
|
|
|
|
|
2008-12-06 05:07:10 +00:00
|
|
|
def _modify(oldversion, servermap, first_time):
|
2008-05-14 20:10:41 +00:00
|
|
|
return oldversion + " modified"
|
|
|
|
d = self._fn.modify(_modify)
|
|
|
|
d.addCallback(lambda res: self._fn.download_best_version())
|
|
|
|
expected = self.CONTENTS[2] + " modified"
|
|
|
|
d.addCallback(lambda res: self.failUnlessEqual(res, expected))
|
|
|
|
# and the servermap should indicate that the outlier was replaced too
|
|
|
|
d.addCallback(lambda res: self._fn.get_servermap(MODE_CHECK))
|
|
|
|
def _check_smap(smap):
|
|
|
|
self.failUnlessEqual(smap.highest_seqnum(), 5)
|
|
|
|
self.failUnlessEqual(len(smap.unrecoverable_versions()), 0)
|
|
|
|
self.failUnlessEqual(len(smap.recoverable_versions()), 1)
|
|
|
|
d.addCallback(_check_smap)
|
|
|
|
return d
|
|
|
|
|
2008-04-05 00:09:26 +00:00
|
|
|
|
|
|
|
class Utils(unittest.TestCase):
|
|
|
|
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"
|
2010-10-27 04:33:02 +00:00
|
|
|
c.add("v1", 1, 0, xdata)
|
|
|
|
c.add("v1", 1, 2000, ydata)
|
|
|
|
self.failUnlessEqual(c.read("v2", 1, 10, 11), None)
|
|
|
|
self.failUnlessEqual(c.read("v1", 2, 10, 11), None)
|
|
|
|
self.failUnlessEqual(c.read("v1", 1, 0, 10), xdata[:10])
|
|
|
|
self.failUnlessEqual(c.read("v1", 1, 90, 10), xdata[90:])
|
|
|
|
self.failUnlessEqual(c.read("v1", 1, 300, 10), None)
|
|
|
|
self.failUnlessEqual(c.read("v1", 1, 2050, 5), ydata[50:55])
|
|
|
|
self.failUnlessEqual(c.read("v1", 1, 0, 101), None)
|
|
|
|
self.failUnlessEqual(c.read("v1", 1, 99, 1), xdata[99:100])
|
|
|
|
self.failUnlessEqual(c.read("v1", 1, 100, 1), None)
|
|
|
|
self.failUnlessEqual(c.read("v1", 1, 1990, 9), None)
|
|
|
|
self.failUnlessEqual(c.read("v1", 1, 1990, 10), None)
|
|
|
|
self.failUnlessEqual(c.read("v1", 1, 1990, 11), None)
|
|
|
|
self.failUnlessEqual(c.read("v1", 1, 1990, 15), None)
|
|
|
|
self.failUnlessEqual(c.read("v1", 1, 1990, 19), None)
|
|
|
|
self.failUnlessEqual(c.read("v1", 1, 1990, 20), None)
|
|
|
|
self.failUnlessEqual(c.read("v1", 1, 1990, 21), None)
|
|
|
|
self.failUnlessEqual(c.read("v1", 1, 1990, 25), None)
|
|
|
|
self.failUnlessEqual(c.read("v1", 1, 1999, 25), None)
|
|
|
|
|
|
|
|
# test joining fragments
|
2008-04-11 21:31:16 +00:00
|
|
|
c = ResponseCache()
|
2010-10-27 04:33:02 +00:00
|
|
|
c.add("v1", 1, 0, xdata[:10])
|
|
|
|
c.add("v1", 1, 10, xdata[10:20])
|
|
|
|
self.failUnlessEqual(c.read("v1", 1, 0, 20), xdata[:20])
|
2008-04-05 00:09:26 +00:00
|
|
|
|
2008-04-21 22:10:50 +00:00
|
|
|
class Exceptions(unittest.TestCase):
|
|
|
|
def test_repr(self):
|
|
|
|
nmde = NeedMoreDataError(100, 50, 100)
|
|
|
|
self.failUnless("NeedMoreDataError" in repr(nmde), repr(nmde))
|
|
|
|
ucwe = UncoordinatedWriteError()
|
|
|
|
self.failUnless("UncoordinatedWriteError" in repr(ucwe), repr(ucwe))
|
|
|
|
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
class SameKeyGenerator:
|
|
|
|
def __init__(self, pubkey, privkey):
|
|
|
|
self.pubkey = pubkey
|
|
|
|
self.privkey = privkey
|
|
|
|
def generate(self, keysize=None):
|
|
|
|
return defer.succeed( (self.pubkey, self.privkey) )
|
|
|
|
|
|
|
|
class FirstServerGetsKilled:
|
|
|
|
done = False
|
|
|
|
def notify(self, retval, wrapper, methname):
|
|
|
|
if not self.done:
|
|
|
|
wrapper.broken = True
|
|
|
|
self.done = True
|
|
|
|
return retval
|
|
|
|
|
|
|
|
class FirstServerGetsDeleted:
|
|
|
|
def __init__(self):
|
|
|
|
self.done = False
|
|
|
|
self.silenced = None
|
|
|
|
def notify(self, retval, wrapper, methname):
|
|
|
|
if not self.done:
|
|
|
|
# this query will work, but later queries should think the share
|
|
|
|
# has been deleted
|
|
|
|
self.done = True
|
|
|
|
self.silenced = wrapper
|
|
|
|
return retval
|
|
|
|
if wrapper == self.silenced:
|
|
|
|
assert methname == "slot_testv_and_readv_and_writev"
|
|
|
|
return (True, {})
|
|
|
|
return retval
|
|
|
|
|
|
|
|
class Problems(GridTestMixin, unittest.TestCase, testutil.ShouldFailMixin):
|
2008-04-23 00:25:14 +00:00
|
|
|
def test_publish_surprise(self):
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
self.basedir = "mutable/Problems/test_publish_surprise"
|
|
|
|
self.set_up_grid()
|
|
|
|
nm = self.g.clients[0].nodemaker
|
|
|
|
d = nm.create_mutable_file("contents 1")
|
2008-04-22 18:49:53 +00:00
|
|
|
def _created(n):
|
|
|
|
d = defer.succeed(None)
|
|
|
|
d.addCallback(lambda res: n.get_servermap(MODE_WRITE))
|
|
|
|
def _got_smap1(smap):
|
|
|
|
# stash the old state of the file
|
|
|
|
self.old_map = smap
|
|
|
|
d.addCallback(_got_smap1)
|
|
|
|
# then modify the file, leaving the old map untouched
|
|
|
|
d.addCallback(lambda res: log.msg("starting winning write"))
|
|
|
|
d.addCallback(lambda res: n.overwrite("contents 2"))
|
|
|
|
# now attempt to modify the file with the old servermap. This
|
|
|
|
# will look just like an uncoordinated write, in which every
|
|
|
|
# single share got updated between our mapupdate and our publish
|
|
|
|
d.addCallback(lambda res: log.msg("starting doomed write"))
|
|
|
|
d.addCallback(lambda res:
|
|
|
|
self.shouldFail(UncoordinatedWriteError,
|
2008-04-23 00:25:14 +00:00
|
|
|
"test_publish_surprise", None,
|
2008-04-22 18:49:53 +00:00
|
|
|
n.upload,
|
|
|
|
"contents 2a", self.old_map))
|
|
|
|
return d
|
|
|
|
d.addCallback(_created)
|
|
|
|
return d
|
|
|
|
|
2008-04-23 00:25:14 +00:00
|
|
|
def test_retrieve_surprise(self):
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
self.basedir = "mutable/Problems/test_retrieve_surprise"
|
|
|
|
self.set_up_grid()
|
|
|
|
nm = self.g.clients[0].nodemaker
|
|
|
|
d = nm.create_mutable_file("contents 1")
|
2008-04-23 00:25:14 +00:00
|
|
|
def _created(n):
|
|
|
|
d = defer.succeed(None)
|
|
|
|
d.addCallback(lambda res: n.get_servermap(MODE_READ))
|
|
|
|
def _got_smap1(smap):
|
|
|
|
# stash the old state of the file
|
|
|
|
self.old_map = smap
|
|
|
|
d.addCallback(_got_smap1)
|
|
|
|
# then modify the file, leaving the old map untouched
|
|
|
|
d.addCallback(lambda res: log.msg("starting winning write"))
|
|
|
|
d.addCallback(lambda res: n.overwrite("contents 2"))
|
|
|
|
# now attempt to retrieve the old version with the old servermap.
|
|
|
|
# This will look like someone has changed the file since we
|
|
|
|
# updated the servermap.
|
|
|
|
d.addCallback(lambda res: n._cache._clear())
|
|
|
|
d.addCallback(lambda res: log.msg("starting doomed read"))
|
|
|
|
d.addCallback(lambda res:
|
|
|
|
self.shouldFail(NotEnoughSharesError,
|
|
|
|
"test_retrieve_surprise",
|
|
|
|
"ran out of peers: have 0 shares (k=3)",
|
|
|
|
n.download_version,
|
|
|
|
self.old_map,
|
|
|
|
self.old_map.best_recoverable_version(),
|
|
|
|
))
|
|
|
|
return d
|
|
|
|
d.addCallback(_created)
|
|
|
|
return d
|
|
|
|
|
2008-04-22 18:49:53 +00:00
|
|
|
def test_unexpected_shares(self):
|
|
|
|
# upload the file, take a servermap, shut down one of the servers,
|
|
|
|
# upload it again (causing shares to appear on a new server), then
|
|
|
|
# upload using the old servermap. The last upload should fail with an
|
|
|
|
# UncoordinatedWriteError, because of the shares that didn't appear
|
|
|
|
# in the servermap.
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
self.basedir = "mutable/Problems/test_unexpected_shares"
|
|
|
|
self.set_up_grid()
|
|
|
|
nm = self.g.clients[0].nodemaker
|
|
|
|
d = nm.create_mutable_file("contents 1")
|
2008-04-22 18:49:53 +00:00
|
|
|
def _created(n):
|
|
|
|
d = defer.succeed(None)
|
|
|
|
d.addCallback(lambda res: n.get_servermap(MODE_WRITE))
|
|
|
|
def _got_smap1(smap):
|
|
|
|
# stash the old state of the file
|
|
|
|
self.old_map = smap
|
|
|
|
# now shut down one of the servers
|
|
|
|
peer0 = list(smap.make_sharemap()[0])[0]
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
self.g.remove_server(peer0)
|
2008-04-22 18:49:53 +00:00
|
|
|
# then modify the file, leaving the old map untouched
|
|
|
|
log.msg("starting winning write")
|
|
|
|
return n.overwrite("contents 2")
|
|
|
|
d.addCallback(_got_smap1)
|
|
|
|
# now attempt to modify the file with the old servermap. This
|
|
|
|
# will look just like an uncoordinated write, in which every
|
|
|
|
# single share got updated between our mapupdate and our publish
|
|
|
|
d.addCallback(lambda res: log.msg("starting doomed write"))
|
|
|
|
d.addCallback(lambda res:
|
|
|
|
self.shouldFail(UncoordinatedWriteError,
|
|
|
|
"test_surprise", None,
|
|
|
|
n.upload,
|
|
|
|
"contents 2a", self.old_map))
|
|
|
|
return d
|
|
|
|
d.addCallback(_created)
|
|
|
|
return d
|
|
|
|
|
|
|
|
def test_bad_server(self):
|
|
|
|
# Break one server, then create the file: the initial publish should
|
|
|
|
# complete with an alternate server. Breaking a second server should
|
|
|
|
# not prevent an update from succeeding either.
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
self.basedir = "mutable/Problems/test_bad_server"
|
|
|
|
self.set_up_grid()
|
|
|
|
nm = self.g.clients[0].nodemaker
|
|
|
|
|
2008-04-22 18:49:53 +00:00
|
|
|
# to make sure that one of the initial peers is broken, we have to
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
# get creative. We create an RSA key and compute its storage-index.
|
|
|
|
# Then we make a KeyGenerator that always returns that one key, and
|
|
|
|
# use it to create the mutable file. This will get easier when we can
|
|
|
|
# use #467 static-server-selection to disable permutation and force
|
|
|
|
# the choice of server for share[0].
|
|
|
|
|
|
|
|
d = nm.key_generator.generate(522)
|
|
|
|
def _got_key( (pubkey, privkey) ):
|
|
|
|
nm.key_generator = SameKeyGenerator(pubkey, privkey)
|
|
|
|
pubkey_s = pubkey.serialize()
|
|
|
|
privkey_s = privkey.serialize()
|
|
|
|
u = uri.WriteableSSKFileURI(ssk_writekey_hash(privkey_s),
|
|
|
|
ssk_pubkey_fingerprint_hash(pubkey_s))
|
2010-02-22 02:45:04 +00:00
|
|
|
self._storage_index = u.get_storage_index()
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
d.addCallback(_got_key)
|
2008-04-22 18:49:53 +00:00
|
|
|
def _break_peer0(res):
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
si = self._storage_index
|
2011-02-21 01:58:04 +00:00
|
|
|
servers = nm.storage_broker.get_servers_for_psi(si)
|
|
|
|
self.g.break_server(servers[0].get_serverid())
|
|
|
|
self.server1 = servers[1]
|
2008-04-22 18:49:53 +00:00
|
|
|
d.addCallback(_break_peer0)
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
# now "create" the file, using the pre-established key, and let the
|
|
|
|
# initial publish finally happen
|
|
|
|
d.addCallback(lambda res: nm.create_mutable_file("contents 1"))
|
2008-04-22 18:49:53 +00:00
|
|
|
# that ought to work
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
def _got_node(n):
|
|
|
|
d = n.download_best_version()
|
|
|
|
d.addCallback(lambda res: self.failUnlessEqual(res, "contents 1"))
|
|
|
|
# now break the second peer
|
|
|
|
def _break_peer1(res):
|
2011-02-21 01:58:04 +00:00
|
|
|
self.g.break_server(self.server1.get_serverid())
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
d.addCallback(_break_peer1)
|
|
|
|
d.addCallback(lambda res: n.overwrite("contents 2"))
|
|
|
|
# that ought to work too
|
|
|
|
d.addCallback(lambda res: n.download_best_version())
|
|
|
|
d.addCallback(lambda res: self.failUnlessEqual(res, "contents 2"))
|
|
|
|
def _explain_error(f):
|
|
|
|
print f
|
|
|
|
if f.check(NotEnoughServersError):
|
|
|
|
print "first_error:", f.value.first_error
|
|
|
|
return f
|
|
|
|
d.addErrback(_explain_error)
|
|
|
|
return d
|
|
|
|
d.addCallback(_got_node)
|
2008-04-22 18:49:53 +00:00
|
|
|
return d
|
|
|
|
|
2008-12-10 05:44:49 +00:00
|
|
|
def test_bad_server_overlap(self):
|
|
|
|
# like test_bad_server, but with no extra unused servers to fall back
|
|
|
|
# upon. This means that we must re-use a server which we've already
|
|
|
|
# used. If we don't remember the fact that we sent them one share
|
|
|
|
# already, we'll mistakenly think we're experiencing an
|
|
|
|
# UncoordinatedWriteError.
|
|
|
|
|
|
|
|
# Break one server, then create the file: the initial publish should
|
|
|
|
# complete with an alternate server. Breaking a second server should
|
|
|
|
# not prevent an update from succeeding either.
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
self.basedir = "mutable/Problems/test_bad_server_overlap"
|
|
|
|
self.set_up_grid()
|
|
|
|
nm = self.g.clients[0].nodemaker
|
|
|
|
sb = nm.storage_broker
|
2008-12-10 05:44:49 +00:00
|
|
|
|
2011-02-21 01:58:04 +00:00
|
|
|
peerids = [s.get_serverid() for s in sb.get_connected_servers()]
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
self.g.break_server(peerids[0])
|
2008-12-10 05:44:49 +00:00
|
|
|
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
d = nm.create_mutable_file("contents 1")
|
2008-12-10 05:44:49 +00:00
|
|
|
def _created(n):
|
|
|
|
d = n.download_best_version()
|
|
|
|
d.addCallback(lambda res: self.failUnlessEqual(res, "contents 1"))
|
|
|
|
# now break one of the remaining servers
|
|
|
|
def _break_second_server(res):
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
self.g.break_server(peerids[1])
|
2008-12-10 05:44:49 +00:00
|
|
|
d.addCallback(_break_second_server)
|
|
|
|
d.addCallback(lambda res: n.overwrite("contents 2"))
|
|
|
|
# that ought to work too
|
|
|
|
d.addCallback(lambda res: n.download_best_version())
|
|
|
|
d.addCallback(lambda res: self.failUnlessEqual(res, "contents 2"))
|
|
|
|
return d
|
|
|
|
d.addCallback(_created)
|
|
|
|
return d
|
|
|
|
|
2008-04-23 01:53:20 +00:00
|
|
|
def test_publish_all_servers_bad(self):
|
|
|
|
# Break all servers: the publish should fail
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
self.basedir = "mutable/Problems/test_publish_all_servers_bad"
|
|
|
|
self.set_up_grid()
|
|
|
|
nm = self.g.clients[0].nodemaker
|
2011-02-21 01:58:04 +00:00
|
|
|
for s in nm.storage_broker.get_connected_servers():
|
|
|
|
s.get_rref().broken = True
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
|
2008-04-23 01:53:20 +00:00
|
|
|
d = self.shouldFail(NotEnoughServersError,
|
|
|
|
"test_publish_all_servers_bad",
|
|
|
|
"Ran out of non-bad servers",
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
nm.create_mutable_file, "contents")
|
2008-04-23 01:53:20 +00:00
|
|
|
return d
|
|
|
|
|
2008-07-07 19:18:10 +00:00
|
|
|
def test_publish_no_servers(self):
|
|
|
|
# no servers at all: the publish should fail
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
self.basedir = "mutable/Problems/test_publish_no_servers"
|
|
|
|
self.set_up_grid(num_servers=0)
|
|
|
|
nm = self.g.clients[0].nodemaker
|
|
|
|
|
2008-07-07 19:18:10 +00:00
|
|
|
d = self.shouldFail(NotEnoughServersError,
|
|
|
|
"test_publish_no_servers",
|
|
|
|
"Ran out of non-bad servers",
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
nm.create_mutable_file, "contents")
|
2008-07-07 19:18:10 +00:00
|
|
|
return d
|
|
|
|
test_publish_no_servers.timeout = 30
|
|
|
|
|
|
|
|
|
2008-04-22 23:47:52 +00:00
|
|
|
def test_privkey_query_error(self):
|
|
|
|
# when a servermap is updated with MODE_WRITE, it tries to get the
|
|
|
|
# privkey. Something might go wrong during this query attempt.
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
# Exercise the code in _privkey_query_failed which tries to handle
|
|
|
|
# such an error.
|
|
|
|
self.basedir = "mutable/Problems/test_privkey_query_error"
|
|
|
|
self.set_up_grid(num_servers=20)
|
|
|
|
nm = self.g.clients[0].nodemaker
|
|
|
|
nm._node_cache = DevNullDictionary() # disable the nodecache
|
|
|
|
|
2008-04-22 23:47:52 +00:00
|
|
|
# we need some contents that are large enough to push the privkey out
|
|
|
|
# of the early part of the file
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
LARGE = "These are Larger contents" * 2000 # about 50KB
|
|
|
|
d = nm.create_mutable_file(LARGE)
|
2008-04-22 23:47:52 +00:00
|
|
|
def _created(n):
|
|
|
|
self.uri = n.get_uri()
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
self.n2 = nm.create_from_cap(self.uri)
|
|
|
|
|
|
|
|
# When a mapupdate is performed on a node that doesn't yet know
|
|
|
|
# the privkey, a short read is sent to a batch of servers, to get
|
|
|
|
# the verinfo and (hopefully, if the file is short enough) the
|
|
|
|
# encprivkey. Our file is too large to let this first read
|
|
|
|
# contain the encprivkey. Each non-encprivkey-bearing response
|
|
|
|
# that arrives (until the node gets the encprivkey) will trigger
|
|
|
|
# a second read to specifically read the encprivkey.
|
|
|
|
#
|
|
|
|
# So, to exercise this case:
|
|
|
|
# 1. notice which server gets a read() call first
|
|
|
|
# 2. tell that server to start throwing errors
|
|
|
|
killer = FirstServerGetsKilled()
|
2011-02-21 01:58:04 +00:00
|
|
|
for s in nm.storage_broker.get_connected_servers():
|
|
|
|
s.get_rref().post_call_notifier = killer.notify
|
2008-04-22 23:47:52 +00:00
|
|
|
d.addCallback(_created)
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
|
2008-04-22 23:47:52 +00:00
|
|
|
# now we update a servermap from a new node (which doesn't have the
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
# privkey yet, forcing it to use a separate privkey query). Note that
|
|
|
|
# the map-update will succeed, since we'll just get a copy from one
|
|
|
|
# of the other shares.
|
2008-04-22 23:47:52 +00:00
|
|
|
d.addCallback(lambda res: self.n2.get_servermap(MODE_WRITE))
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
|
2008-04-22 23:47:52 +00:00
|
|
|
return d
|
|
|
|
|
|
|
|
def test_privkey_query_missing(self):
|
|
|
|
# like test_privkey_query_error, but the shares are deleted by the
|
|
|
|
# second query, instead of raising an exception.
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
self.basedir = "mutable/Problems/test_privkey_query_missing"
|
|
|
|
self.set_up_grid(num_servers=20)
|
|
|
|
nm = self.g.clients[0].nodemaker
|
|
|
|
LARGE = "These are Larger contents" * 2000 # about 50KB
|
|
|
|
nm._node_cache = DevNullDictionary() # disable the nodecache
|
|
|
|
|
|
|
|
d = nm.create_mutable_file(LARGE)
|
2008-04-22 23:47:52 +00:00
|
|
|
def _created(n):
|
|
|
|
self.uri = n.get_uri()
|
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
|
|
|
self.n2 = nm.create_from_cap(self.uri)
|
|
|
|
deleter = FirstServerGetsDeleted()
|
2011-02-21 01:58:04 +00:00
|
|
|
for s in nm.storage_broker.get_connected_servers():
|
|
|
|
s.get_rref().post_call_notifier = deleter.notify
|
2008-04-22 23:47:52 +00:00
|
|
|
d.addCallback(_created)
|
|
|
|
d.addCallback(lambda res: self.n2.get_servermap(MODE_WRITE))
|
|
|
|
return d
|