2008-04-11 14:31:16 -07:00
|
|
|
|
|
|
|
from allmydata.util import idlib
|
2010-10-26 21:33:02 -07:00
|
|
|
from allmydata.util.spans import DataSpans
|
2008-04-11 14:31:16 -07:00
|
|
|
|
2008-04-16 15:22:30 -07:00
|
|
|
MODE_CHECK = "MODE_CHECK" # query all peers
|
|
|
|
MODE_ANYTHING = "MODE_ANYTHING" # one recoverable version
|
|
|
|
MODE_WRITE = "MODE_WRITE" # replace all shares, probably.. not for initial
|
|
|
|
# creation
|
|
|
|
MODE_READ = "MODE_READ"
|
2008-04-11 14:31:16 -07:00
|
|
|
|
2010-01-26 22:44:30 -08:00
|
|
|
class NotWriteableError(Exception):
|
2008-04-11 14:31:16 -07:00
|
|
|
pass
|
|
|
|
|
|
|
|
class NeedMoreDataError(Exception):
|
|
|
|
def __init__(self, needed_bytes, encprivkey_offset, encprivkey_length):
|
|
|
|
Exception.__init__(self)
|
|
|
|
self.needed_bytes = needed_bytes # up through EOF
|
|
|
|
self.encprivkey_offset = encprivkey_offset
|
|
|
|
self.encprivkey_length = encprivkey_length
|
2008-04-21 15:10:50 -07:00
|
|
|
def __repr__(self):
|
2008-04-11 14:31:16 -07:00
|
|
|
return "<NeedMoreDataError (%d bytes)>" % self.needed_bytes
|
|
|
|
|
|
|
|
class UncoordinatedWriteError(Exception):
|
|
|
|
def __repr__(self):
|
2008-04-21 15:10:50 -07:00
|
|
|
return ("<%s -- You, oh user, tried to change a file or directory "
|
|
|
|
"at the same time as another process was trying to change it. "
|
|
|
|
" To avoid data loss, don't do this. Please see "
|
2011-05-10 12:16:50 -07:00
|
|
|
"docs/write_coordination.rst for details.>" %
|
2008-04-21 15:10:50 -07:00
|
|
|
(self.__class__.__name__,))
|
2008-04-11 14:31:16 -07:00
|
|
|
|
|
|
|
class UnrecoverableFileError(Exception):
|
|
|
|
pass
|
|
|
|
|
2008-04-22 11:49:53 -07:00
|
|
|
class NotEnoughServersError(Exception):
|
|
|
|
"""There were not enough functioning servers available to place shares
|
2009-06-01 14:04:07 -07:00
|
|
|
upon. This might result from all servers being full or having an error, a
|
|
|
|
local bug which causes all server requests to fail in the same way, or
|
|
|
|
from there being zero servers. The first error received (if any) is
|
|
|
|
stored in my .first_error attribute."""
|
|
|
|
def __init__(self, why, first_error=None):
|
|
|
|
Exception.__init__(self, why, first_error)
|
|
|
|
self.first_error = first_error
|
2008-04-22 11:49:53 -07:00
|
|
|
|
2008-04-11 14:31:16 -07:00
|
|
|
class CorruptShareError(Exception):
|
|
|
|
def __init__(self, peerid, shnum, reason):
|
|
|
|
self.args = (peerid, shnum, reason)
|
|
|
|
self.peerid = peerid
|
|
|
|
self.shnum = shnum
|
|
|
|
self.reason = reason
|
|
|
|
def __str__(self):
|
|
|
|
short_peerid = idlib.nodeid_b2a(self.peerid)[:8]
|
|
|
|
return "<CorruptShareError peerid=%s shnum[%d]: %s" % (short_peerid,
|
|
|
|
self.shnum,
|
|
|
|
self.reason)
|
|
|
|
|
2009-07-14 23:45:10 -07:00
|
|
|
class UnknownVersionError(Exception):
|
|
|
|
"""The share we received was of a version we don't recognize."""
|
2008-04-11 14:31:16 -07:00
|
|
|
|
|
|
|
class ResponseCache:
|
|
|
|
"""I cache share data, to reduce the number of round trips used during
|
|
|
|
mutable file operations. All of the data in my cache is for a single
|
2010-10-26 21:33:02 -07:00
|
|
|
storage index, but I will keep information on multiple shares for
|
|
|
|
that storage index.
|
|
|
|
|
|
|
|
I maintain a highest-seen sequence number, and will flush all entries
|
|
|
|
each time this number increases (this doesn't necessarily imply that
|
|
|
|
all entries have the same sequence number).
|
2008-04-11 14:31:16 -07:00
|
|
|
|
|
|
|
My cache is indexed by a (verinfo, shnum) tuple.
|
|
|
|
|
2010-10-26 21:33:02 -07:00
|
|
|
My cache entries are DataSpans instances, each representing a set of
|
|
|
|
non-overlapping byteranges.
|
2008-04-11 14:31:16 -07:00
|
|
|
"""
|
|
|
|
|
|
|
|
def __init__(self):
|
2010-10-26 21:33:02 -07:00
|
|
|
self.cache = {}
|
|
|
|
self.seqnum = None
|
2008-04-11 14:31:16 -07:00
|
|
|
|
2008-04-22 17:25:14 -07:00
|
|
|
def _clear(self):
|
2010-10-26 21:33:02 -07:00
|
|
|
# also used by unit tests
|
|
|
|
self.cache = {}
|
|
|
|
|
|
|
|
def add(self, verinfo, shnum, offset, data):
|
|
|
|
seqnum = verinfo[0]
|
|
|
|
if seqnum > self.seqnum:
|
|
|
|
self._clear()
|
|
|
|
self.seqnum = seqnum
|
|
|
|
|
2008-04-11 14:31:16 -07:00
|
|
|
index = (verinfo, shnum)
|
2010-10-26 21:33:02 -07:00
|
|
|
if index in self.cache:
|
|
|
|
self.cache[index].add(offset, data)
|
|
|
|
else:
|
|
|
|
spans = DataSpans()
|
|
|
|
spans.add(offset, data)
|
|
|
|
self.cache[index] = spans
|
2008-04-11 14:31:16 -07:00
|
|
|
|
|
|
|
def read(self, verinfo, shnum, offset, length):
|
|
|
|
"""Try to satisfy a read request from cache.
|
2010-10-26 21:33:02 -07:00
|
|
|
Returns data, or None if the cache did not hold the entire requested span.
|
2008-04-11 14:31:16 -07:00
|
|
|
"""
|
|
|
|
|
2010-10-26 21:33:02 -07:00
|
|
|
# TODO: perhaps return a DataSpans object representing the fragments
|
|
|
|
# that we have, instead of only returning a hit if we can satisfy the
|
|
|
|
# whole request from cache.
|
2008-04-11 14:31:16 -07:00
|
|
|
|
|
|
|
index = (verinfo, shnum)
|
2010-10-26 21:33:02 -07:00
|
|
|
if index in self.cache:
|
|
|
|
return self.cache[index].get(offset, length)
|
|
|
|
else:
|
|
|
|
return None
|