Skip to content

Instantly share code, notes, and snippets.

@mildsunrise
Created July 15, 2019 23:40
Show Gist options
  • Star 1 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save mildsunrise/dc02bc28c7e897a7672d03fe85631694 to your computer and use it in GitHub Desktop.
Save mildsunrise/dc02bc28c7e897a7672d03fe85631694 to your computer and use it in GitHub Desktop.
Quick & dirty uility for diffing filesystems
import os, re
# Ignores differences on folder mode / ownership
# Collapses "all contents are new" into "whole folder is new"
# Forces no type differences (i.e. link vs dir, etc.)
# Keeps in mind loaded passwd / group for comparison of UID or GID
with open("/etc/passwd") as f:
new_uid_table = { int(x.split(":")[2]): x.split(":")[0] for x in f.read().strip().splitlines() }
with open("/other/etc/passwd") as f:
old_uid_table = { int(x.split(":")[2]): x.split(":")[0] for x in f.read().strip().splitlines() }
with open("/etc/group") as f:
new_gid_table = { int(x.split(":")[2]): x.split(":")[0] for x in f.read().strip().splitlines() }
with open("/other/etc/group") as f:
old_gid_table = { int(x.split(":")[2]): x.split(":")[0] for x in f.read().strip().splitlines() }
with open("root-tree.json", "r") as f:
new = json.loads(f.read())
with open("other-tree.json", "r") as f:
old = json.loads(f.read())
EXCLUDE = set('''
/etc/resolv.conf
/usr/bin/dockerd
/usr/share/doc/libglib2.0-bin
/usr/share/doc/libglib2.0-dev
/usr/share/libgda-5.0/gda_trml2pdf
/etc/cups
/lib/modules
/usr/share/doc
/usr/share/icons
/usr/share/locale
/usr/share/man
/usr/local/lib/python2.7
/usr/local/lib/python2.7
/usr/lib/python3/dist-packages
/usr/lib/python2.7/dist-packages
/cdrom
/initrd.img.old
/lost+found
/media
/mnt
/srv
/home
'''[1:].splitlines())
EXCLUDE_PATTERNS = list(map(re.compile,
[ ".*/__pycache__", ".*\\.pyc", "/usr/src/linux.*", "/etc/rc.\\.d" ]))
def match_attrs(new, old):
return new["mode"] == old["mode"] and new_uid_table[new["uid"]] == old_uid_table[old["uid"]] and new_gid_table[new["gid"]] == old_gid_table[old["gid"]]
def accumulate(seen):
if not seen: return "equal"
if "multiple" in seen or "delete" in seen: return "multiple"
if len(seen) == 1: return next(iter(seen))
return "multiple"
output = open("changes.txt", "w")
def process_tree(new, old, relpath="/"):
if relpath in EXCLUDE or any(re.fullmatch(x, relpath) for x in EXCLUDE_PATTERNS): return
if not old: return { "diff": "add" }
if not new: return { "diff": "delete" }
assert new["type"] == old["type"]
amatch = match_attrs(new["attrs"], old["attrs"])
if new["type"] == 'dir':
files = set(new["files"]) | set(old["files"])
results = []
seen = set()
for f in sorted(files):
result = process_tree(new["files"].get(f), old["files"].get(f), os.path.join(relpath, f))
if result is None: continue
seen.add(result["diff"])
results.append((os.path.join(relpath, f), result["diff"]))
gdiff = accumulate(seen)
if gdiff == "multiple":
for f, r in results:
if r not in {"multiple", "equal", "delete"}:
print({"add": "+", "change": ":", "delete": "-", "equal": " "}[r], f, file=output)
return { "diff": gdiff }
if new["type"] == "link":
return { "diff": "equal" if amatch and new["link"] == old["link"] else "change" }
if new["type"] == "file":
return { "diff": "equal" if amatch and new["hash"] == old["hash"] else "change" }
process_tree(new, old)
output.close()
# Generates "root-tree.json" or "other-tree.json"
import os
import hashlib
import json
import sys
EXCLUDE= {
"/vmlinuz",
"/vmlinuz.old",
"/tmp",
"/dev",
"/run",
"/var",
"/home",
}
def process_file(path, relpath="/"):
stat = os.lstat(path)
attrs = { "mode": stat.st_mode, "uid": stat.st_uid, "gid": stat.st_gid }
if (relpath != "/" and os.path.ismount(path)) or relpath in EXCLUDE:
return
if os.path.islink(path):
return { "type": "link", "link": os.readlink(path) }
if os.path.isdir(path):
files = {}
for k in os.listdir(path):
v = process_file(os.path.join(path, k), os.path.join(relpath, k))
if not (v is None): files[k] = v
return { "type": "dir", "attrs": attrs, "files": files }
if os.path.isfile(path):
return { "type": "file", "attrs": attrs, "hash": hash_file(path) }
print("WARNING: Can't figure out type of: {}".format(path), file=sys.stderr)
def hash_file(path):
h = hashlib.md5()
b = bytearray(128*1024)
mv = memoryview(b)
with open(path, 'rb', buffering=0) as f:
for n in iter(lambda : f.readinto(mv), 0):
h.update(mv[:n])
return h.hexdigest()
argv = sys.argv[1:]
if len(argv) != 2:
print("Usage: inspect.py <root folder> <output file>", file=sys.stderr)
exit(1)
print("Inspecting...")
my_tree = process_file(argv[0])
print("Writing...")
with open(argv[1], "w") as f:
f.write(json.dumps(my_tree) + "\n")
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment