2007-01-17 04:29:59 +00:00
|
|
|
|
2007-06-12 01:25:18 +00:00
|
|
|
import re
|
2007-07-21 22:40:36 +00:00
|
|
|
from zope.interface import implements
|
|
|
|
from twisted.python.components import registerAdapter
|
2007-06-12 01:25:18 +00:00
|
|
|
from allmydata.util import idlib, hashutil
|
2007-07-21 22:40:36 +00:00
|
|
|
from allmydata.interfaces import IURI, IDirnodeURI, IFileURI
|
2007-07-12 23:17:49 +00:00
|
|
|
|
2007-01-17 04:29:59 +00:00
|
|
|
# the URI shall be an ascii representation of the file. It shall contain
|
|
|
|
# enough information to retrieve and validate the contents. It shall be
|
|
|
|
# expressed in a limited character set (namely [TODO]).
|
|
|
|
|
2007-06-02 01:48:01 +00:00
|
|
|
|
2007-07-21 22:40:36 +00:00
|
|
|
class _BaseURI:
|
|
|
|
def __hash__(self):
|
|
|
|
return hash((self.__class__, self.to_string()))
|
|
|
|
def __cmp__(self, them):
|
|
|
|
if cmp(type(self), type(them)):
|
|
|
|
return cmp(type(self), type(them))
|
|
|
|
if cmp(self.__class__, them.__class__):
|
|
|
|
return cmp(self.__class__, them.__class__)
|
|
|
|
return cmp(self.to_string(), them.to_string())
|
|
|
|
|
|
|
|
class CHKFileURI(_BaseURI):
|
|
|
|
implements(IURI, IFileURI)
|
|
|
|
|
|
|
|
def __init__(self, **kwargs):
|
|
|
|
# construct me with kwargs, since there are so many of them
|
|
|
|
if not kwargs:
|
|
|
|
return
|
2007-07-22 01:23:15 +00:00
|
|
|
keys = ("key", "uri_extension_hash",
|
|
|
|
"needed_shares", "total_shares", "size")
|
|
|
|
for name in kwargs:
|
|
|
|
if name in keys:
|
|
|
|
value = kwargs[name]
|
|
|
|
setattr(self, name, value)
|
|
|
|
else:
|
|
|
|
raise TypeError("CHKFileURI does not accept '%s=' argument"
|
|
|
|
% name)
|
|
|
|
self.storage_index = hashutil.storage_index_chk_hash(self.key)
|
2007-07-21 22:40:36 +00:00
|
|
|
|
|
|
|
def init_from_string(self, uri):
|
|
|
|
assert uri.startswith("URI:CHK:"), uri
|
|
|
|
d = {}
|
|
|
|
(header_uri, header_chk,
|
2007-07-22 01:23:15 +00:00
|
|
|
key_s, uri_extension_hash_s,
|
2007-07-21 22:40:36 +00:00
|
|
|
needed_shares_s, total_shares_s, size_s) = uri.split(":")
|
|
|
|
assert header_uri == "URI"
|
|
|
|
assert header_chk == "CHK"
|
2007-07-22 01:23:15 +00:00
|
|
|
|
2007-07-21 22:40:36 +00:00
|
|
|
self.key = idlib.a2b(key_s)
|
2007-07-22 01:23:15 +00:00
|
|
|
assert isinstance(self.key, str)
|
|
|
|
assert len(self.key) == 16 # AES-128
|
2007-07-21 22:40:36 +00:00
|
|
|
|
2007-07-22 01:23:15 +00:00
|
|
|
self.storage_index = hashutil.storage_index_chk_hash(self.key)
|
2007-07-21 22:40:36 +00:00
|
|
|
assert isinstance(self.storage_index, str)
|
|
|
|
assert len(self.storage_index) == 32 # sha256 hash
|
|
|
|
|
2007-07-22 01:23:15 +00:00
|
|
|
self.uri_extension_hash = idlib.a2b(uri_extension_hash_s)
|
2007-07-21 22:40:36 +00:00
|
|
|
assert isinstance(self.uri_extension_hash, str)
|
|
|
|
assert len(self.uri_extension_hash) == 32 # sha56 hash
|
|
|
|
|
2007-07-22 01:23:15 +00:00
|
|
|
self.needed_shares = int(needed_shares_s)
|
|
|
|
self.total_shares = int(total_shares_s)
|
|
|
|
self.size = int(size_s)
|
|
|
|
return self
|
|
|
|
|
|
|
|
def to_string(self):
|
2007-07-21 22:40:36 +00:00
|
|
|
assert isinstance(self.needed_shares, int)
|
|
|
|
assert isinstance(self.total_shares, int)
|
|
|
|
assert isinstance(self.size, (int,long))
|
|
|
|
|
2007-07-22 01:23:15 +00:00
|
|
|
return ("URI:CHK:%s:%s:%d:%d:%d" %
|
|
|
|
(idlib.b2a(self.key),
|
2007-07-21 22:40:36 +00:00
|
|
|
idlib.b2a(self.uri_extension_hash),
|
|
|
|
self.needed_shares,
|
|
|
|
self.total_shares,
|
|
|
|
self.size))
|
|
|
|
|
|
|
|
def is_readonly(self):
|
|
|
|
return True
|
|
|
|
def is_mutable(self):
|
|
|
|
return False
|
|
|
|
def get_readonly(self):
|
|
|
|
return self
|
|
|
|
|
|
|
|
def get_size(self):
|
|
|
|
return self.size
|
|
|
|
|
|
|
|
class LiteralFileURI(_BaseURI):
|
|
|
|
implements(IURI, IFileURI)
|
|
|
|
|
|
|
|
def __init__(self, data=None):
|
|
|
|
if data is not None:
|
|
|
|
self.data = data
|
|
|
|
|
|
|
|
def init_from_string(self, uri):
|
|
|
|
assert uri.startswith("URI:LIT:")
|
|
|
|
data_s = uri[len("URI:LIT:"):]
|
|
|
|
self.data = idlib.a2b(data_s)
|
|
|
|
return self
|
|
|
|
|
|
|
|
def to_string(self):
|
|
|
|
return "URI:LIT:%s" % idlib.b2a(self.data)
|
|
|
|
|
|
|
|
def is_readonly(self):
|
|
|
|
return True
|
|
|
|
def is_mutable(self):
|
|
|
|
return False
|
|
|
|
def get_readonly(self):
|
|
|
|
return self
|
|
|
|
|
|
|
|
def get_size(self):
|
|
|
|
return len(self.data)
|
|
|
|
|
|
|
|
class DirnodeURI(_BaseURI):
|
|
|
|
implements(IURI, IDirnodeURI)
|
|
|
|
|
|
|
|
def __init__(self, furl=None, writekey=None):
|
|
|
|
if furl is not None or writekey is not None:
|
|
|
|
assert furl is not None
|
|
|
|
assert writekey is not None
|
|
|
|
self.furl = furl
|
|
|
|
self.writekey = writekey
|
2007-07-22 00:45:00 +00:00
|
|
|
self._derive_values()
|
2007-07-21 22:40:36 +00:00
|
|
|
|
|
|
|
def init_from_string(self, uri):
|
|
|
|
# URI:DIR:furl:key
|
|
|
|
# but note that the furl contains colons
|
|
|
|
prefix = "URI:DIR:"
|
|
|
|
assert uri.startswith(prefix)
|
|
|
|
uri = uri[len(prefix):]
|
|
|
|
colon = uri.rindex(":")
|
|
|
|
self.furl = uri[:colon]
|
|
|
|
self.writekey = idlib.a2b(uri[colon+1:])
|
2007-07-22 00:45:00 +00:00
|
|
|
self._derive_values()
|
2007-07-21 22:40:36 +00:00
|
|
|
return self
|
|
|
|
|
2007-07-22 00:45:00 +00:00
|
|
|
def _derive_values(self):
|
|
|
|
wk, we, rk, index = \
|
|
|
|
hashutil.generate_dirnode_keys_from_writekey(self.writekey)
|
|
|
|
self.write_enabler = we
|
|
|
|
self.readkey = rk
|
|
|
|
self.storage_index = index
|
|
|
|
|
2007-07-21 22:40:36 +00:00
|
|
|
def to_string(self):
|
|
|
|
return "URI:DIR:%s:%s" % (self.furl, idlib.b2a(self.writekey))
|
|
|
|
|
|
|
|
def is_readonly(self):
|
|
|
|
return False
|
|
|
|
def is_mutable(self):
|
|
|
|
return True
|
|
|
|
def get_readonly(self):
|
2007-07-22 00:45:00 +00:00
|
|
|
return ReadOnlyDirnodeURI(self.furl, self.readkey)
|
2007-07-21 22:40:36 +00:00
|
|
|
|
|
|
|
class ReadOnlyDirnodeURI(_BaseURI):
|
|
|
|
implements(IURI, IDirnodeURI)
|
|
|
|
|
|
|
|
def __init__(self, furl=None, readkey=None):
|
|
|
|
if furl is not None or readkey is not None:
|
|
|
|
assert furl is not None
|
|
|
|
assert readkey is not None
|
|
|
|
self.furl = furl
|
|
|
|
self.readkey = readkey
|
2007-07-22 00:45:00 +00:00
|
|
|
self._derive_values()
|
2007-07-21 22:40:36 +00:00
|
|
|
|
|
|
|
def init_from_string(self, uri):
|
|
|
|
# URI:DIR-RO:furl:key
|
|
|
|
# but note that the furl contains colons
|
|
|
|
prefix = "URI:DIR-RO:"
|
|
|
|
assert uri.startswith(prefix)
|
|
|
|
uri = uri[len(prefix):]
|
|
|
|
colon = uri.rindex(":")
|
|
|
|
self.furl = uri[:colon]
|
|
|
|
self.readkey = idlib.a2b(uri[colon+1:])
|
2007-07-22 00:45:00 +00:00
|
|
|
self._derive_values()
|
2007-07-21 22:40:36 +00:00
|
|
|
return self
|
|
|
|
|
2007-07-22 00:45:00 +00:00
|
|
|
def _derive_values(self):
|
|
|
|
wk, we, rk, index = \
|
|
|
|
hashutil.generate_dirnode_keys_from_readkey(self.readkey)
|
|
|
|
self.writekey = wk # None
|
|
|
|
self.write_enabler = we # None
|
|
|
|
self.storage_index = index
|
|
|
|
|
2007-07-21 22:40:36 +00:00
|
|
|
def to_string(self):
|
|
|
|
return "URI:DIR-RO:%s:%s" % (self.furl, idlib.b2a(self.readkey))
|
|
|
|
|
|
|
|
def is_readonly(self):
|
|
|
|
return True
|
|
|
|
def is_mutable(self):
|
|
|
|
return True
|
|
|
|
def get_readonly(self):
|
|
|
|
return self
|
|
|
|
|
|
|
|
def from_string(s):
|
|
|
|
if s.startswith("URI:CHK:"):
|
|
|
|
return CHKFileURI().init_from_string(s)
|
|
|
|
elif s.startswith("URI:LIT:"):
|
|
|
|
return LiteralFileURI().init_from_string(s)
|
|
|
|
elif s.startswith("URI:DIR:"):
|
|
|
|
return DirnodeURI().init_from_string(s)
|
|
|
|
elif s.startswith("URI:DIR-RO:"):
|
|
|
|
return ReadOnlyDirnodeURI().init_from_string(s)
|
|
|
|
else:
|
|
|
|
raise RuntimeError("unknown URI type: %s.." % s[:10])
|
2007-06-02 01:48:01 +00:00
|
|
|
|
2007-07-21 22:40:36 +00:00
|
|
|
registerAdapter(from_string, str, IURI)
|
2007-06-02 01:48:01 +00:00
|
|
|
|
2007-07-21 22:40:36 +00:00
|
|
|
def from_string_dirnode(s):
|
|
|
|
u = from_string(s)
|
|
|
|
assert IDirnodeURI.providedBy(u)
|
|
|
|
return u
|
2007-01-17 04:29:59 +00:00
|
|
|
|
2007-07-21 22:40:36 +00:00
|
|
|
registerAdapter(from_string_dirnode, str, IDirnodeURI)
|
2007-01-17 04:29:59 +00:00
|
|
|
|
2007-07-21 22:40:36 +00:00
|
|
|
def from_string_filenode(s):
|
|
|
|
u = from_string(s)
|
|
|
|
assert IFileURI.providedBy(u)
|
|
|
|
return u
|
|
|
|
|
|
|
|
registerAdapter(from_string_filenode, str, IFileURI)
|
2007-01-17 04:29:59 +00:00
|
|
|
|
|
|
|
|
2007-06-12 01:25:18 +00:00
|
|
|
def pack_extension(data):
|
|
|
|
pieces = []
|
|
|
|
for k in sorted(data.keys()):
|
|
|
|
value = data[k]
|
|
|
|
if isinstance(value, (int, long)):
|
|
|
|
value = "%d" % value
|
|
|
|
assert isinstance(value, str), k
|
|
|
|
assert re.match(r'^[a-zA-Z_\-]+$', k)
|
|
|
|
pieces.append(k + ":" + hashutil.netstring(value))
|
|
|
|
uri_extension = "".join(pieces)
|
|
|
|
return uri_extension
|
|
|
|
|
|
|
|
def unpack_extension(data):
|
|
|
|
d = {}
|
|
|
|
while data:
|
|
|
|
colon = data.index(":")
|
|
|
|
key = data[:colon]
|
|
|
|
data = data[colon+1:]
|
|
|
|
|
|
|
|
colon = data.index(":")
|
|
|
|
number = data[:colon]
|
|
|
|
length = int(number)
|
|
|
|
data = data[colon+1:]
|
|
|
|
|
|
|
|
value = data[:length]
|
|
|
|
assert data[length] == ","
|
|
|
|
data = data[length+1:]
|
|
|
|
|
|
|
|
d[key] = value
|
|
|
|
|
|
|
|
# convert certain things to numbers
|
|
|
|
for intkey in ("size", "segment_size", "num_segments",
|
|
|
|
"needed_shares", "total_shares"):
|
|
|
|
if intkey in d:
|
|
|
|
d[intkey] = int(d[intkey])
|
|
|
|
return d
|
|
|
|
|
2007-06-12 01:38:21 +00:00
|
|
|
|
|
|
|
def unpack_extension_readable(data):
|
|
|
|
unpacked = unpack_extension(data)
|
|
|
|
for k in sorted(unpacked.keys()):
|
|
|
|
if "hash" in k:
|
|
|
|
unpacked[k] = idlib.b2a(unpacked[k])
|
|
|
|
return unpacked
|
2007-06-25 20:23:51 +00:00
|
|
|
|