2007-07-11 01:41:52 +00:00
|
|
|
|
2007-07-13 23:58:08 +00:00
|
|
|
# do not import any allmydata modules at this level. Do that from inside
|
|
|
|
# individual functions instead.
|
2008-01-14 20:43:25 +00:00
|
|
|
import sys, struct, time, os
|
2007-07-11 01:41:52 +00:00
|
|
|
from twisted.python import usage
|
|
|
|
|
|
|
|
class DumpOptions(usage.Options):
|
|
|
|
optParameters = [
|
|
|
|
["filename", "f", None, "which file to dump"],
|
|
|
|
]
|
|
|
|
|
|
|
|
def parseArgs(self, filename=None):
|
|
|
|
if filename:
|
|
|
|
self['filename'] = filename
|
|
|
|
|
|
|
|
def postOptions(self):
|
|
|
|
if not self['filename']:
|
|
|
|
raise usage.UsageError("<filename> parameter is required")
|
|
|
|
|
2007-09-02 21:48:20 +00:00
|
|
|
def dump_share(config, out=sys.stdout, err=sys.stderr):
|
2007-07-14 00:25:45 +00:00
|
|
|
from allmydata import uri, storage
|
2007-07-11 01:41:52 +00:00
|
|
|
|
2007-11-07 01:55:55 +00:00
|
|
|
# check the version, to see if we have a mutable or immutable share
|
2008-02-06 20:37:43 +00:00
|
|
|
print >>out, "share filename: %s" % config['filename']
|
2008-02-06 20:19:51 +00:00
|
|
|
|
2007-11-07 01:55:55 +00:00
|
|
|
f = open(config['filename'], "rb")
|
|
|
|
prefix = f.read(32)
|
|
|
|
f.close()
|
|
|
|
if prefix == storage.MutableShareFile.MAGIC:
|
|
|
|
return dump_mutable_share(config, out, err)
|
|
|
|
# otherwise assume it's immutable
|
2007-09-02 21:48:20 +00:00
|
|
|
f = storage.ShareFile(config['filename'])
|
2007-07-13 23:58:08 +00:00
|
|
|
# use a ReadBucketProxy to parse the bucket and find the uri extension
|
2007-07-14 00:25:45 +00:00
|
|
|
bp = storage.ReadBucketProxy(None)
|
2007-09-04 16:00:24 +00:00
|
|
|
offsets = bp._parse_offsets(f.read_share_data(0, 0x24))
|
2007-09-02 21:48:20 +00:00
|
|
|
seek = offsets['uri_extension']
|
|
|
|
length = struct.unpack(">L", f.read_share_data(seek, 4))[0]
|
|
|
|
seek += 4
|
2008-02-06 19:48:19 +00:00
|
|
|
UEB_data = f.read_share_data(seek, length)
|
2007-07-13 23:58:08 +00:00
|
|
|
|
2008-02-06 19:48:19 +00:00
|
|
|
unpacked = uri.unpack_extension_readable(UEB_data)
|
2007-07-11 01:41:52 +00:00
|
|
|
keys1 = ("size", "num_segments", "segment_size",
|
|
|
|
"needed_shares", "total_shares")
|
|
|
|
keys2 = ("codec_name", "codec_params", "tail_codec_params")
|
|
|
|
keys3 = ("plaintext_hash", "plaintext_root_hash",
|
|
|
|
"crypttext_hash", "crypttext_root_hash",
|
2008-02-06 19:48:19 +00:00
|
|
|
"share_root_hash", "UEB_hash")
|
2007-09-26 22:00:59 +00:00
|
|
|
display_keys = {"size": "file_size"}
|
2007-07-11 01:41:52 +00:00
|
|
|
for k in keys1:
|
|
|
|
if k in unpacked:
|
2007-09-26 22:00:59 +00:00
|
|
|
dk = display_keys.get(k, k)
|
2008-02-06 20:37:43 +00:00
|
|
|
print >>out, "%20s: %s" % (dk, unpacked[k])
|
2007-07-11 01:41:52 +00:00
|
|
|
print >>out
|
|
|
|
for k in keys2:
|
|
|
|
if k in unpacked:
|
2007-09-26 22:00:59 +00:00
|
|
|
dk = display_keys.get(k, k)
|
2008-02-06 20:37:43 +00:00
|
|
|
print >>out, "%20s: %s" % (dk, unpacked[k])
|
2007-07-11 01:41:52 +00:00
|
|
|
print >>out
|
|
|
|
for k in keys3:
|
|
|
|
if k in unpacked:
|
2007-09-26 22:00:59 +00:00
|
|
|
dk = display_keys.get(k, k)
|
2008-02-06 20:37:43 +00:00
|
|
|
print >>out, "%20s: %s" % (dk, unpacked[k])
|
2007-07-11 01:41:52 +00:00
|
|
|
|
|
|
|
leftover = set(unpacked.keys()) - set(keys1 + keys2 + keys3)
|
|
|
|
if leftover:
|
|
|
|
print >>out
|
2007-07-13 23:58:08 +00:00
|
|
|
print >>out, "LEFTOVER:"
|
2007-07-11 01:41:52 +00:00
|
|
|
for k in sorted(leftover):
|
2008-02-06 20:37:43 +00:00
|
|
|
print >>out, "%20s: %s" % (k, unpacked[k])
|
2007-07-11 01:41:52 +00:00
|
|
|
|
2007-08-27 06:42:39 +00:00
|
|
|
sizes = {}
|
|
|
|
sizes['data'] = bp._data_size
|
|
|
|
sizes['validation'] = (offsets['uri_extension'] -
|
|
|
|
offsets['plaintext_hash_tree'])
|
2008-02-06 19:48:19 +00:00
|
|
|
sizes['uri-extension'] = len(UEB_data)
|
2007-08-27 06:42:39 +00:00
|
|
|
print >>out
|
2008-02-06 20:37:43 +00:00
|
|
|
print >>out, " Size of data within the share:"
|
2007-08-27 06:42:39 +00:00
|
|
|
for k in sorted(sizes):
|
2008-02-06 20:37:43 +00:00
|
|
|
print >>out, "%20s: %s" % (k, sizes[k])
|
2007-08-27 06:42:39 +00:00
|
|
|
|
2007-09-02 21:48:20 +00:00
|
|
|
# display lease information too
|
|
|
|
leases = list(f.iter_leases())
|
|
|
|
if leases:
|
|
|
|
for i,lease in enumerate(leases):
|
|
|
|
(owner_num, renew_secret, cancel_secret, expiration_time) = lease
|
2007-11-07 01:55:55 +00:00
|
|
|
when = format_expiration_time(expiration_time)
|
2008-02-06 20:37:43 +00:00
|
|
|
print >>out, " Lease #%d: owner=%d, expire in %s" % (i, owner_num,
|
|
|
|
when)
|
2007-09-02 21:48:20 +00:00
|
|
|
else:
|
2008-02-06 20:37:43 +00:00
|
|
|
print >>out, " No leases."
|
2007-09-02 21:48:20 +00:00
|
|
|
|
|
|
|
print >>out
|
|
|
|
return 0
|
|
|
|
|
2007-11-07 01:55:55 +00:00
|
|
|
def format_expiration_time(expiration_time):
|
|
|
|
now = time.time()
|
|
|
|
remains = expiration_time - now
|
|
|
|
when = "%ds" % remains
|
|
|
|
if remains > 24*3600:
|
|
|
|
when += " (%d days)" % (remains / (24*3600))
|
|
|
|
elif remains > 3600:
|
|
|
|
when += " (%d hours)" % (remains / 3600)
|
|
|
|
return when
|
|
|
|
|
|
|
|
|
|
|
|
def dump_mutable_share(config, out, err):
|
|
|
|
from allmydata import storage
|
|
|
|
from allmydata.util import idlib
|
|
|
|
m = storage.MutableShareFile(config['filename'])
|
|
|
|
f = open(config['filename'], "rb")
|
|
|
|
WE, nodeid = m._read_write_enabler_and_nodeid(f)
|
|
|
|
num_extra_leases = m._read_num_extra_leases(f)
|
|
|
|
data_length = m._read_data_length(f)
|
|
|
|
extra_lease_offset = m._read_extra_lease_offset(f)
|
2007-11-07 02:31:22 +00:00
|
|
|
container_size = extra_lease_offset - m.DATA_OFFSET
|
2007-11-07 01:55:55 +00:00
|
|
|
leases = list(m._enumerate_leases(f))
|
2007-11-07 02:46:31 +00:00
|
|
|
|
|
|
|
share_type = "unknown"
|
|
|
|
f.seek(m.DATA_OFFSET)
|
|
|
|
if f.read(1) == "\x00":
|
|
|
|
# this slot contains an SMDF share
|
|
|
|
share_type = "SDMF"
|
2007-11-07 01:55:55 +00:00
|
|
|
f.close()
|
|
|
|
|
|
|
|
print >>out
|
2007-11-07 02:10:49 +00:00
|
|
|
print >>out, "Mutable slot found:"
|
2007-11-07 02:46:31 +00:00
|
|
|
print >>out, " share_type: %s" % share_type
|
2007-11-07 02:10:49 +00:00
|
|
|
print >>out, " write_enabler: %s" % idlib.b2a(WE)
|
|
|
|
print >>out, " WE for nodeid: %s" % idlib.nodeid_b2a(nodeid)
|
|
|
|
print >>out, " num_extra_leases: %d" % num_extra_leases
|
2007-11-07 02:31:22 +00:00
|
|
|
print >>out, " container_size: %d" % container_size
|
2007-11-07 02:10:49 +00:00
|
|
|
print >>out, " data_length: %d" % data_length
|
2007-11-07 01:55:55 +00:00
|
|
|
if leases:
|
|
|
|
for (leasenum, (oid,et,rs,cs,anid)) in leases:
|
2007-11-07 02:31:22 +00:00
|
|
|
print >>out
|
2007-11-07 02:10:49 +00:00
|
|
|
print >>out, " Lease #%d:" % leasenum
|
|
|
|
print >>out, " ownerid: %d" % oid
|
2007-11-07 01:55:55 +00:00
|
|
|
when = format_expiration_time(et)
|
2007-11-07 02:10:49 +00:00
|
|
|
print >>out, " expires in %s" % when
|
|
|
|
print >>out, " renew_secret: %s" % idlib.b2a(rs)
|
|
|
|
print >>out, " cancel_secret: %s" % idlib.b2a(cs)
|
|
|
|
print >>out, " secrets are for nodeid: %s" % idlib.nodeid_b2a(anid)
|
2007-11-07 01:55:55 +00:00
|
|
|
else:
|
|
|
|
print >>out, "No leases."
|
|
|
|
print >>out
|
2007-11-07 02:46:31 +00:00
|
|
|
|
|
|
|
if share_type == "SDMF":
|
|
|
|
dump_SDMF_share(m.DATA_OFFSET, data_length, config, out, err)
|
|
|
|
|
2007-11-07 01:55:55 +00:00
|
|
|
return 0
|
|
|
|
|
2007-11-07 02:46:31 +00:00
|
|
|
def dump_SDMF_share(offset, length, config, out, err):
|
|
|
|
from allmydata import mutable
|
|
|
|
from allmydata.util import idlib
|
|
|
|
|
|
|
|
f = open(config['filename'], "rb")
|
|
|
|
f.seek(offset)
|
|
|
|
data = f.read(min(length, 2000))
|
|
|
|
f.close()
|
|
|
|
|
2007-11-08 00:52:09 +00:00
|
|
|
try:
|
|
|
|
pieces = mutable.unpack_share(data)
|
|
|
|
except mutable.NeedMoreDataError, e:
|
|
|
|
# retry once with the larger size
|
|
|
|
size = e.needed_bytes
|
|
|
|
f = open(config['filename'], "rb")
|
|
|
|
f.seek(offset)
|
|
|
|
data = f.read(min(length, size))
|
|
|
|
f.close()
|
|
|
|
pieces = mutable.unpack_share(data)
|
2007-11-07 02:46:31 +00:00
|
|
|
|
|
|
|
(seqnum, root_hash, IV, k, N, segsize, datalen,
|
|
|
|
pubkey, signature, share_hash_chain, block_hash_tree,
|
|
|
|
share_data, enc_privkey) = pieces
|
|
|
|
|
|
|
|
print >>out, " SDMF contents:"
|
|
|
|
print >>out, " seqnum: %d" % seqnum
|
|
|
|
print >>out, " root_hash: %s" % idlib.b2a(root_hash)
|
|
|
|
print >>out, " IV: %s" % idlib.b2a(IV)
|
|
|
|
print >>out, " required_shares: %d" % k
|
|
|
|
print >>out, " total_shares: %d" % N
|
|
|
|
print >>out, " segsize: %d" % segsize
|
|
|
|
print >>out, " datalen: %d" % datalen
|
2007-11-14 06:08:15 +00:00
|
|
|
share_hash_ids = ",".join(sorted([str(hid)
|
|
|
|
for hid in share_hash_chain.keys()]))
|
2007-11-07 02:46:31 +00:00
|
|
|
print >>out, " share_hash_chain: %s" % share_hash_ids
|
|
|
|
print >>out, " block_hash_tree: %d nodes" % len(block_hash_tree)
|
|
|
|
|
|
|
|
print >>out
|
|
|
|
|
2007-11-07 01:55:55 +00:00
|
|
|
|
2008-01-14 20:43:25 +00:00
|
|
|
|
|
|
|
class DumpCapOptions(usage.Options):
|
|
|
|
optParameters = [
|
|
|
|
["nodeid", "n", None, "storage server nodeid (ascii), to construct WE and secrets."],
|
|
|
|
["client-secret", "c", None, "client's base secret (ascii), to construct secrets"],
|
|
|
|
["client-dir", "d", None, "client's base directory, from which a -c secret will be read"],
|
|
|
|
]
|
|
|
|
def parseArgs(self, cap):
|
|
|
|
self.cap = cap
|
|
|
|
|
|
|
|
def dump_cap(config, out=sys.stdout, err=sys.stderr):
|
|
|
|
from allmydata import uri
|
|
|
|
from allmydata.util.idlib import a2b
|
|
|
|
from base64 import b32decode
|
2008-01-14 21:12:27 +00:00
|
|
|
import urlparse, urllib
|
2008-01-14 20:43:25 +00:00
|
|
|
|
|
|
|
cap = config.cap
|
|
|
|
nodeid = None
|
|
|
|
if config['nodeid']:
|
|
|
|
nodeid = b32decode(config['nodeid'].upper())
|
|
|
|
secret = None
|
|
|
|
if config['client-secret']:
|
|
|
|
secret = a2b(config['client-secret'])
|
|
|
|
elif config['client-dir']:
|
|
|
|
secretfile = os.path.join(config['client-dir'], "private", "secret")
|
|
|
|
try:
|
|
|
|
secret = a2b(open(secretfile, "r").read().strip())
|
|
|
|
except EnvironmentError:
|
|
|
|
pass
|
|
|
|
|
2008-01-14 21:12:27 +00:00
|
|
|
if cap.startswith("http"):
|
|
|
|
scheme, netloc, path, params, query, fragment = urlparse.urlparse(cap)
|
|
|
|
assert path.startswith("/uri/")
|
|
|
|
cap = urllib.unquote(path[len("/uri/"):])
|
|
|
|
|
|
|
|
u = uri.from_string(cap)
|
|
|
|
|
2008-01-14 20:43:25 +00:00
|
|
|
print >>out
|
|
|
|
dump_uri_instance(u, nodeid, secret, out, err)
|
|
|
|
|
|
|
|
def _dump_secrets(storage_index, secret, nodeid, out):
|
|
|
|
from allmydata.util import hashutil
|
|
|
|
from allmydata.util.idlib import b2a
|
|
|
|
|
|
|
|
if secret:
|
|
|
|
crs = hashutil.my_renewal_secret_hash(secret)
|
|
|
|
print >>out, " client renewal secret:", b2a(crs)
|
|
|
|
frs = hashutil.file_renewal_secret_hash(crs, storage_index)
|
|
|
|
print >>out, " file renewal secret:", b2a(frs)
|
|
|
|
if nodeid:
|
|
|
|
renew = hashutil.bucket_renewal_secret_hash(frs, nodeid)
|
|
|
|
print >>out, " lease renewal secret:", b2a(renew)
|
|
|
|
ccs = hashutil.my_cancel_secret_hash(secret)
|
|
|
|
print >>out, " client cancel secret:", b2a(ccs)
|
|
|
|
fcs = hashutil.file_cancel_secret_hash(ccs, storage_index)
|
|
|
|
print >>out, " file cancel secret:", b2a(fcs)
|
|
|
|
if nodeid:
|
|
|
|
cancel = hashutil.bucket_cancel_secret_hash(fcs, nodeid)
|
|
|
|
print >>out, " lease cancel secret:", b2a(cancel)
|
|
|
|
|
|
|
|
def dump_uri_instance(u, nodeid, secret, out, err, show_header=True):
|
|
|
|
from allmydata import uri
|
|
|
|
from allmydata.util.idlib import b2a
|
|
|
|
from allmydata.util import hashutil
|
|
|
|
|
|
|
|
if isinstance(u, uri.CHKFileURI):
|
|
|
|
if show_header:
|
|
|
|
print >>out, "CHK File:"
|
|
|
|
print >>out, " key:", b2a(u.key)
|
|
|
|
print >>out, " UEB hash:", b2a(u.uri_extension_hash)
|
|
|
|
print >>out, " size:", u.size
|
|
|
|
print >>out, " k/N: %d/%d" % (u.needed_shares, u.total_shares)
|
|
|
|
print >>out, " storage index:", b2a(u.storage_index)
|
|
|
|
_dump_secrets(u.storage_index, secret, nodeid, out)
|
|
|
|
elif isinstance(u, uri.CHKFileVerifierURI):
|
|
|
|
if show_header:
|
|
|
|
print >>out, "CHK Verifier URI:"
|
|
|
|
print >>out, " UEB hash:", b2a(u.uri_extension_hash)
|
|
|
|
print >>out, " size:", u.size
|
|
|
|
print >>out, " k/N: %d/%d" % (u.needed_shares, u.total_shares)
|
|
|
|
print >>out, " storage index:", b2a(u.storage_index)
|
|
|
|
|
|
|
|
elif isinstance(u, uri.LiteralFileURI):
|
|
|
|
if show_header:
|
|
|
|
print >>out, "Literal File URI:"
|
|
|
|
print >>out, " data:", u.data
|
|
|
|
|
|
|
|
elif isinstance(u, uri.WriteableSSKFileURI):
|
|
|
|
if show_header:
|
|
|
|
print >>out, "SSK Writeable URI:"
|
|
|
|
print >>out, " writekey:", b2a(u.writekey)
|
|
|
|
print >>out, " readkey:", b2a(u.readkey)
|
|
|
|
print >>out, " storage index:", b2a(u.storage_index)
|
|
|
|
print >>out, " fingerprint:", b2a(u.fingerprint)
|
|
|
|
print >>out
|
|
|
|
if nodeid:
|
|
|
|
we = hashutil.ssk_write_enabler_hash(u.writekey, nodeid)
|
|
|
|
print >>out, " write_enabler:", b2a(we)
|
|
|
|
print >>out
|
|
|
|
_dump_secrets(u.storage_index, secret, nodeid, out)
|
|
|
|
|
|
|
|
elif isinstance(u, uri.ReadonlySSKFileURI):
|
|
|
|
if show_header:
|
|
|
|
print >>out, "SSK Read-only URI:"
|
|
|
|
print >>out, " readkey:", b2a(u.readkey)
|
|
|
|
print >>out, " storage index:", b2a(u.storage_index)
|
|
|
|
print >>out, " fingerprint:", b2a(u.fingerprint)
|
|
|
|
elif isinstance(u, uri.SSKVerifierURI):
|
|
|
|
if show_header:
|
|
|
|
print >>out, "SSK Verifier URI:"
|
|
|
|
print >>out, " storage index:", b2a(u.storage_index)
|
|
|
|
print >>out, " fingerprint:", b2a(u.fingerprint)
|
|
|
|
|
|
|
|
elif isinstance(u, uri.NewDirectoryURI):
|
|
|
|
if show_header:
|
|
|
|
print >>out, "Directory Writeable URI:"
|
|
|
|
dump_uri_instance(u._filenode_uri, nodeid, secret, out, err, False)
|
|
|
|
elif isinstance(u, uri.ReadonlyNewDirectoryURI):
|
|
|
|
if show_header:
|
|
|
|
print >>out, "Directory Read-only URI:"
|
|
|
|
dump_uri_instance(u._filenode_uri, nodeid, secret, out, err, False)
|
|
|
|
elif isinstance(u, uri.NewDirectoryURIVerifier):
|
|
|
|
if show_header:
|
|
|
|
print >>out, "Directory Verifier URI:"
|
|
|
|
dump_uri_instance(u._filenode_uri, nodeid, secret, out, err, False)
|
|
|
|
else:
|
|
|
|
print >>out, "unknown cap type"
|
|
|
|
|
2008-02-06 20:19:51 +00:00
|
|
|
class FindSharesOptions(usage.Options):
|
|
|
|
def parseArgs(self, storage_index_s, *nodedirs):
|
|
|
|
self.si_s = storage_index_s
|
|
|
|
self.nodedirs = nodedirs
|
|
|
|
|
|
|
|
def find_shares(config, out=sys.stdout, err=sys.stderr):
|
|
|
|
"""Given a storage index and a list of node directories, emit a list of
|
|
|
|
all matching shares to stdout, one per line. For example:
|
|
|
|
|
|
|
|
find-shares.py 44kai1tui348689nrw8fjegc8c ~/testnet/node-*
|
|
|
|
|
|
|
|
gives:
|
|
|
|
|
|
|
|
/home/warner/testnet/node-1/storage/shares/44k/44kai1tui348689nrw8fjegc8c/5
|
|
|
|
/home/warner/testnet/node-1/storage/shares/44k/44kai1tui348689nrw8fjegc8c/9
|
|
|
|
/home/warner/testnet/node-2/storage/shares/44k/44kai1tui348689nrw8fjegc8c/2
|
|
|
|
"""
|
|
|
|
from allmydata import storage
|
|
|
|
from allmydata.util import idlib
|
|
|
|
|
|
|
|
sharedir = storage.storage_index_to_dir(idlib.a2b(config.si_s))
|
|
|
|
for d in config.nodedirs:
|
|
|
|
d = os.path.join(os.path.expanduser(d), "storage/shares", sharedir)
|
|
|
|
if os.path.exists(d):
|
|
|
|
for shnum in os.listdir(d):
|
|
|
|
print >>out, os.path.join(d, shnum)
|
|
|
|
|
|
|
|
return 0
|
2008-01-14 20:43:25 +00:00
|
|
|
|
2007-07-11 01:41:52 +00:00
|
|
|
subCommands = [
|
2007-09-02 21:48:20 +00:00
|
|
|
["dump-share", None, DumpOptions,
|
|
|
|
"Unpack and display the contents of a share (uri_extension and leases)."],
|
2008-02-06 20:19:51 +00:00
|
|
|
["dump-cap", None, DumpCapOptions, "Unpack a read-cap or write-cap"],
|
|
|
|
["find-shares", None, FindSharesOptions, "Locate sharefiles in node dirs"],
|
2007-07-11 01:41:52 +00:00
|
|
|
]
|
|
|
|
|
|
|
|
dispatch = {
|
2007-09-02 21:48:20 +00:00
|
|
|
"dump-share": dump_share,
|
2008-01-14 20:43:25 +00:00
|
|
|
"dump-cap": dump_cap,
|
2008-02-06 20:19:51 +00:00
|
|
|
"find-shares": find_shares,
|
2007-07-11 01:41:52 +00:00
|
|
|
}
|