tree compare
This commit is contained in:
		| @ -44,11 +44,17 @@ class BlockHasher(): | |||||||
|         checked=0 |         checked=0 | ||||||
|         with open(fname, "rb") as f: |         with open(fname, "rb") as f: | ||||||
|             for ( chunk_nr, hexdigest ) in generator: |             for ( chunk_nr, hexdigest ) in generator: | ||||||
|  |                 print ("comparing {} {} {}".format(fname, chunk_nr, hexdigest)) | ||||||
|  |  | ||||||
|                 checked=checked+1 |                 checked=checked+1 | ||||||
|                 hash = self.hash_class() |                 hash = self.hash_class() | ||||||
|                 f.seek(chunk_nr * self.bs * self.count) |                 f.seek(chunk_nr * self.bs * self.count) | ||||||
|                 for block_nr in range(0,self.count): |                 block_nr=0 | ||||||
|                     hash.update(f.read(self.bs)) |                 for block in iter(lambda: f.read(self.bs), b""): | ||||||
|  |                     hash.update(block) | ||||||
|  |                     block_nr=block_nr+1 | ||||||
|  |                     if block_nr == self.count: | ||||||
|  |                         break | ||||||
|  |  | ||||||
|                 if (hash.hexdigest()!=hexdigest): |                 if (hash.hexdigest()!=hexdigest): | ||||||
|                     raise Exception("Block {} mismatched! Hash is {}, but should be {}".format(chunk_nr, hash.hexdigest(), hexdigest)) |                     raise Exception("Block {} mismatched! Hash is {}, but should be {}".format(chunk_nr, hash.hexdigest(), hexdigest)) | ||||||
|  | |||||||
| @ -5,6 +5,10 @@ class TreeHasher(): | |||||||
|     """uses BlockHasher recursively on a directory tree""" |     """uses BlockHasher recursively on a directory tree""" | ||||||
|  |  | ||||||
|     def __init__(self, block_hasher): |     def __init__(self, block_hasher): | ||||||
|  |         """ | ||||||
|  |  | ||||||
|  |         :type block_hasher: BlockHasher | ||||||
|  |         """ | ||||||
|         self.block_hasher=block_hasher |         self.block_hasher=block_hasher | ||||||
|  |  | ||||||
|     def generate(self, start_path): |     def generate(self, start_path): | ||||||
| @ -31,3 +35,37 @@ class TreeHasher(): | |||||||
|         finally: |         finally: | ||||||
|             os.chdir(cwd) |             os.chdir(cwd) | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     def compare(self, start_path, generator): | ||||||
|  |         """reads from generator and compares blocks, raises exception on error | ||||||
|  |         """ | ||||||
|  |  | ||||||
|  |         cwd=os.getcwd() | ||||||
|  |         os.chdir(start_path) | ||||||
|  |  | ||||||
|  |         try: | ||||||
|  |             current= [None] | ||||||
|  |             def per_file_generator(): | ||||||
|  |  | ||||||
|  |                 (current_file_path, chunk_nr, hash)=current[0] | ||||||
|  |                 yield ( chunk_nr, hash) | ||||||
|  |  | ||||||
|  |                 for (file_path, chunk_nr, hash) in generator: | ||||||
|  |                     if file_path==current_file_path: | ||||||
|  |                         yield ( chunk_nr, hash) | ||||||
|  |                     else: | ||||||
|  |                         current[0] = (file_path, chunk_nr, hash) | ||||||
|  |                         return | ||||||
|  |  | ||||||
|  |                 current[0]=None | ||||||
|  |  | ||||||
|  |  | ||||||
|  |             current[0] = generator.next() | ||||||
|  |             while current[0] is not None: | ||||||
|  |                 self.block_hasher.compare(current[0][0], per_file_generator()) | ||||||
|  |  | ||||||
|  |         finally: | ||||||
|  |             os.chdir(cwd) | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | |||||||
| @ -68,6 +68,11 @@ class ZfsCheck(CliBase): | |||||||
|  |  | ||||||
|             self.debug("Hashing tree: {}".format(mnt)) |             self.debug("Hashing tree: {}".format(mnt)) | ||||||
|             if not self.args.test: |             if not self.args.test: | ||||||
|  |  | ||||||
|  |                 # generator=tree_hasher.generate(mnt) | ||||||
|  |                 # tree_hasher.compare(mnt, generator) | ||||||
|  |  | ||||||
|  |  | ||||||
|                 for (file, block, hash) in tree_hasher.generate(mnt): |                 for (file, block, hash) in tree_hasher.generate(mnt): | ||||||
|                     print("{}\t{}\t{}".format(file, block, hash)) |                     print("{}\t{}\t{}".format(file, block, hash)) | ||||||
|                     sys.stdout.flush() #important, to generate SIGPIPES on ssh disconnect |                     sys.stdout.flush() #important, to generate SIGPIPES on ssh disconnect | ||||||
| @ -158,6 +163,21 @@ def cli(): | |||||||
|  |  | ||||||
|     sys.exit(ZfsCheck(sys.argv[1:], False).run()) |     sys.exit(ZfsCheck(sys.argv[1:], False).run()) | ||||||
|  |  | ||||||
|  |     # block_hasher=BlockHasher() | ||||||
|  |  | ||||||
|  |     # if sys.argv[1]=="s": | ||||||
|  |     #     for ( fname, nr, hash ) in TreeHasher(block_hasher).generate("/usr/src/linux-headers-5.14.14-051414"): | ||||||
|  |     #         print("{}\t{}\t{}".format(fname, nr, hash)) | ||||||
|  |     # | ||||||
|  |     # if sys.argv[1]=="r": | ||||||
|  |     # | ||||||
|  |     #     def gen(): | ||||||
|  |     #         for line in sys.stdin: | ||||||
|  |     #             ( fname, nr, hash)=line.rstrip().split('\t') | ||||||
|  |     #             yield (fname, int(nr), hash) | ||||||
|  |     # | ||||||
|  |     #     TreeHasher(block_hasher).compare("/usr/src/linux-headers-5.14.14-051414", gen()) | ||||||
|  |  | ||||||
|  |  | ||||||
| if __name__ == "__main__": | if __name__ == "__main__": | ||||||
|  |  | ||||||
|  | |||||||
		Reference in New Issue
	
	Block a user