""" Given a StorageIndex, count how many shares we can find. This does no verification of the shares whatsoever. If the peer claims to have the share, we believe them. """ from twisted.internet import defer from twisted.python import log from allmydata import storage from allmydata.checker_results import CheckerResults from allmydata.immutable import download from allmydata.uri import CHKFileURI from allmydata.util import hashutil from allmydata.util.assertutil import _assert, precondition class SimpleCHKFileChecker: """Return a list of (needed, total, found, sharemap), where sharemap maps share number to a list of (binary) nodeids of the shareholders.""" def __init__(self, client, uri, storage_index, needed_shares, total_shares): self.peer_getter = client.get_permuted_peers self.needed_shares = needed_shares self.total_shares = total_shares self.found_shares = set() self.uri = uri self.storage_index = storage_index self.sharemap = {} self.responded = set() ''' def check_synchronously(self, si): # this is how we would write this class if we were using synchronous # messages (or if we used promises). found = set() for (pmpeerid, peerid, connection) in self.peer_getter(storage_index): buckets = connection.get_buckets(si) found.update(buckets.keys()) return len(found) ''' def start(self): d = self._get_all_shareholders(self.storage_index) d.addCallback(self._done) return d def _get_all_shareholders(self, storage_index): dl = [] for (peerid, ss) in self.peer_getter("storage", storage_index): d = ss.callRemote("get_buckets", storage_index) d.addCallbacks(self._got_response, self._got_error, callbackArgs=(peerid,)) dl.append(d) return defer.DeferredList(dl) def _got_response(self, buckets, peerid): # buckets is a dict: maps shum to an rref of the server who holds it self.found_shares.update(buckets.keys()) for k in buckets: if k not in self.sharemap: self.sharemap[k] = [] self.sharemap[k].append(peerid) self.responded.add(peerid) def _got_error(self, f): if f.check(KeyError): pass log.err(f) pass def _done(self, res): r = CheckerResults(self.uri.to_string(), self.storage_index) report = [] healthy = bool(len(self.found_shares) >= self.total_shares) r.set_healthy(healthy) recoverable = bool(len(self.found_shares) >= self.needed_shares) r.set_recoverable(recoverable) data = {"count-shares-good": len(self.found_shares), "count-shares-needed": self.needed_shares, "count-shares-expected": self.total_shares, "count-wrong-shares": 0, } if recoverable: data["count-recoverable-versions"] = 1 data["count-unrecoverable-versions"] = 0 else: data["count-recoverable-versions"] = 0 data["count-unrecoverable-versions"] = 1 data["count-corrupt-shares"] = 0 # non-verifier doesn't see corruption data["list-corrupt-shares"] = [] hosts = set() sharemap = {} for (shnum,nodeids) in self.sharemap.items(): hosts.update(nodeids) sharemap[shnum] = nodeids data["count-good-share-hosts"] = len(hosts) data["servers-responding"] = list(self.responded) data["sharemap"] = sharemap r.set_data(data) r.set_needs_rebalancing(bool( len(self.found_shares) > len(hosts) )) #r.stuff = (self.needed_shares, self.total_shares, # len(self.found_shares), self.sharemap) if len(self.found_shares) < self.total_shares: wanted = set(range(self.total_shares)) missing = wanted - self.found_shares report.append("Missing shares: %s" % ",".join(["sh%d" % shnum for shnum in sorted(missing)])) r.set_report(report) if healthy: r.set_summary("Healthy") else: r.set_summary("Not Healthy") # TODO: more detail return r class VerifyingOutput: def __init__(self, total_length, results): self._crypttext_hasher = hashutil.crypttext_hasher() self.length = 0 self.total_length = total_length self._segment_number = 0 self._crypttext_hash_tree = None self._opened = False self._results = results results.set_healthy(False) results.set_recoverable(False) results.set_summary("Not Healthy") def got_crypttext_hash_tree(self, crypttext_hashtree): self._crypttext_hash_tree = crypttext_hashtree def write_segment(self, crypttext): self.length += len(crypttext) self._crypttext_hasher.update(crypttext) if self._crypttext_hash_tree: ch = hashutil.crypttext_segment_hasher() ch.update(crypttext) crypttext_leaves = {self._segment_number: ch.digest()} self._crypttext_hash_tree.set_hashes(leaves=crypttext_leaves) self._segment_number += 1 def close(self): self.crypttext_hash = self._crypttext_hasher.digest() def finish(self): self._results.set_healthy(True) self._results.set_recoverable(True) self._results.set_summary("Healthy") # the return value of finish() is passed out of FileDownloader._done, # but SimpleCHKFileVerifier overrides this with the CheckerResults # instance instead. class SimpleCHKFileVerifier(download.FileDownloader): # this reconstructs the crypttext, which verifies that at least 'k' of # the shareholders are around and have valid data. It does not check the # remaining shareholders, and it cannot verify the plaintext. check_plaintext_hash = False def __init__(self, client, u, storage_index, k, N, size, ueb_hash): precondition(isinstance(u, CHKFileURI), u) self._client = client self._uri = u self._storage_index = storage_index self._uri_extension_hash = ueb_hash self._total_shares = N self._size = size self._num_needed_shares = k self._si_s = storage.si_b2a(self._storage_index) self.init_logging() self._check_results = r = CheckerResults(self._uri.to_string(), self._storage_index) r.set_data({"count-shares-needed": k, "count-shares-expected": N, }) self._output = VerifyingOutput(self._size, r) self._paused = False self._stopped = False self._results = None self.active_buckets = {} # k: shnum, v: bucket self._share_buckets = [] # list of (sharenum, bucket) tuples self._share_vbuckets = {} # k: shnum, v: set of ValidatedBuckets self._uri_extension_sources = [] self._uri_extension_data = None self._fetch_failures = {"uri_extension": 0, "plaintext_hashroot": 0, "plaintext_hashtree": 0, "crypttext_hashroot": 0, "crypttext_hashtree": 0, } def init_logging(self): self._log_prefix = prefix = storage.si_b2a(self._storage_index)[:5] num = self._client.log("SimpleCHKFileVerifier(%s): starting" % prefix) self._log_number = num def log(self, *args, **kwargs): if not "parent" in kwargs: kwargs['parent'] = self._log_number # add a prefix to the message, regardless of how it is expressed prefix = "SimpleCHKFileVerifier(%s): " % self._log_prefix if "format" in kwargs: kwargs["format"] = prefix + kwargs["format"] elif "message" in kwargs: kwargs["message"] = prefix + kwargs["message"] elif args: m = prefix + args[0] args = (m,) + args[1:] return self._client.log(*args, **kwargs) def start(self): log.msg("starting download [%s]" % storage.si_b2a(self._storage_index)[:5]) # first step: who should we download from? d = defer.maybeDeferred(self._get_all_shareholders) d.addCallback(self._got_all_shareholders) # now get the uri_extension block from somebody and validate it d.addCallback(self._obtain_uri_extension) d.addCallback(self._get_crypttext_hash_tree) # once we know that, we can download blocks from everybody d.addCallback(self._download_all_segments) d.addCallback(self._done) d.addCallbacks(self._verify_done, self._verify_failed) return d def _verify_done(self, ignored): # TODO: The following results are just stubs, and need to be replaced # with actual values. These exist to make things like deep-check not # fail. self._check_results.set_needs_rebalancing(False) N = self._total_shares data = { "count-shares-good": N, "count-good-share-hosts": N, "count-corrupt-shares": 0, "list-corrupt-shares": [], "servers-responding": [], "sharemap": {}, "count-wrong-shares": 0, "count-recoverable-versions": 1, "count-unrecoverable-versions": 0, } self._check_results.set_data(data) return self._check_results def _verify_failed(self, ignored): # TODO: The following results are just stubs, and need to be replaced # with actual values. These exist to make things like deep-check not # fail. self._check_results.set_needs_rebalancing(False) N = self._total_shares data = { "count-shares-good": 0, "count-good-share-hosts": 0, "count-corrupt-shares": 0, "list-corrupt-shares": [], "servers-responding": [], "sharemap": {}, "count-wrong-shares": 0, "count-recoverable-versions": 0, "count-unrecoverable-versions": 1, } self._check_results.set_data(data) return self._check_results