mirror of
				https://github.com/9001/copyparty.git
				synced 2025-10-24 16:43:55 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			199 lines
		
	
	
		
			5.6 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			199 lines
		
	
	
		
			5.6 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable File
		
	
	
	
	
| #!/usr/bin/env python3
 | |
| 
 | |
| import os
 | |
| import sys
 | |
| import sqlite3
 | |
| import argparse
 | |
| 
 | |
| DB_VER = 3
 | |
| 
 | |
| 
 | |
| def die(msg):
 | |
|     print("\033[31m\n" + msg + "\n\033[0m")
 | |
|     sys.exit(1)
 | |
| 
 | |
| 
 | |
| def read_ver(db):
 | |
|     for tab in ["ki", "kv"]:
 | |
|         try:
 | |
|             c = db.execute(r"select v from {} where k = 'sver'".format(tab))
 | |
|         except:
 | |
|             continue
 | |
| 
 | |
|         rows = c.fetchall()
 | |
|         if rows:
 | |
|             return int(rows[0][0])
 | |
| 
 | |
|     return "corrupt"
 | |
| 
 | |
| 
 | |
| def ls(db):
 | |
|     nfiles = next(db.execute("select count(w) from up"))[0]
 | |
|     ntags = next(db.execute("select count(w) from mt"))[0]
 | |
|     print(f"{nfiles} files")
 | |
|     print(f"{ntags} tags\n")
 | |
| 
 | |
|     print("number of occurences for each tag,")
 | |
|     print(" 'x' = file has no tags")
 | |
|     print(" 't:mtp' = the mtp flag (file not mtp processed yet)")
 | |
|     print()
 | |
|     for k, nk in db.execute("select k, count(k) from mt group by k order by k"):
 | |
|         print(f"{nk:9} {k}")
 | |
| 
 | |
| 
 | |
| def compare(n1, d1, n2, d2, verbose):
 | |
|     nt = next(d1.execute("select count(w) from up"))[0]
 | |
|     n = 0
 | |
|     miss = 0
 | |
|     for w, rd, fn in d1.execute("select w, rd, fn from up"):
 | |
|         n += 1
 | |
|         if n % 25_000 == 0:
 | |
|             m = f"\033[36mchecked {n:,} of {nt:,} files in {n1} against {n2}\033[0m"
 | |
|             print(m)
 | |
| 
 | |
|         q = "select w from up where substr(w,1,16) = ?"
 | |
|         hit = d2.execute(q, (w[:16],)).fetchone()
 | |
|         if not hit:
 | |
|             miss += 1
 | |
|             if verbose:
 | |
|                 print(f"file in {n1} missing in {n2}: [{w}] {rd}/{fn}")
 | |
| 
 | |
|     print(f" {miss} files in {n1} missing in {n2}\n")
 | |
| 
 | |
|     nt = next(d1.execute("select count(w) from mt"))[0]
 | |
|     n = 0
 | |
|     miss = {}
 | |
|     nmiss = 0
 | |
|     for w, k, v in d1.execute("select * from mt"):
 | |
|         n += 1
 | |
|         if n % 100_000 == 0:
 | |
|             m = f"\033[36mchecked {n:,} of {nt:,} tags in {n1} against {n2}, so far {nmiss} missing tags\033[0m"
 | |
|             print(m)
 | |
| 
 | |
|         v2 = d2.execute("select v from mt where w = ? and +k = ?", (w, k)).fetchone()
 | |
|         if v2:
 | |
|             v2 = v2[0]
 | |
| 
 | |
|         # if v != v2 and v2 and k in [".bpm", "key"] and n2 == "src":
 | |
|         #    print(f"{w} [{rd}/{fn}] {k} = [{v}] / [{v2}]")
 | |
| 
 | |
|         if v2 is not None:
 | |
|             if k.startswith("."):
 | |
|                 try:
 | |
|                     diff = abs(float(v) - float(v2))
 | |
|                     if diff > float(v) / 0.9:
 | |
|                         v2 = None
 | |
|                     else:
 | |
|                         v2 = v
 | |
|                 except:
 | |
|                     pass
 | |
| 
 | |
|             if v != v2:
 | |
|                 v2 = None
 | |
| 
 | |
|         if v2 is None:
 | |
|             nmiss += 1
 | |
|             try:
 | |
|                 miss[k] += 1
 | |
|             except:
 | |
|                 miss[k] = 1
 | |
| 
 | |
|             if verbose:
 | |
|                 q = "select rd, fn from up where substr(w,1,16) = ?"
 | |
|                 rd, fn = d1.execute(q, (w,)).fetchone()
 | |
|                 print(f"missing in {n2}: [{w}] [{rd}/{fn}] {k} = {v}")
 | |
| 
 | |
|     for k, v in sorted(miss.items()):
 | |
|         if v:
 | |
|             print(f"{n1} has {v:6} more {k:<6} tags than {n2}")
 | |
| 
 | |
|     print(f"in total, {nmiss} missing tags in {n2}\n")
 | |
| 
 | |
| 
 | |
| def copy_mtp(d1, d2, tag, rm):
 | |
|     nt = next(d1.execute("select count(w) from mt where k = ?", (tag,)))[0]
 | |
|     n = 0
 | |
|     ndone = 0
 | |
|     for w, k, v in d1.execute("select * from mt where k = ?", (tag,)):
 | |
|         n += 1
 | |
|         if n % 25_000 == 0:
 | |
|             m = f"\033[36m{n:,} of {nt:,} tags checked, so far {ndone} copied\033[0m"
 | |
|             print(m)
 | |
| 
 | |
|         hit = d2.execute("select v from mt where w = ? and +k = ?", (w, k)).fetchone()
 | |
|         if hit:
 | |
|             hit = hit[0]
 | |
| 
 | |
|         if hit != v:
 | |
|             ndone += 1
 | |
|             if hit is not None:
 | |
|                 d2.execute("delete from mt where w = ? and +k = ?", (w, k))
 | |
| 
 | |
|             d2.execute("insert into mt values (?,?,?)", (w, k, v))
 | |
|             if rm:
 | |
|                 d2.execute("delete from mt where w = ? and +k = 't:mtp'", (w,))
 | |
| 
 | |
|     d2.commit()
 | |
|     print(f"copied {ndone} {tag} tags over")
 | |
| 
 | |
| 
 | |
| def main():
 | |
|     os.system("")
 | |
|     print()
 | |
|     
 | |
|     ap = argparse.ArgumentParser()
 | |
|     ap.add_argument("db", help="database to work on")
 | |
|     ap.add_argument("-src", metavar="DB", type=str, help="database to copy from")
 | |
| 
 | |
|     ap2 = ap.add_argument_group("informational / read-only stuff")
 | |
|     ap2.add_argument("-v", action="store_true", help="verbose")
 | |
|     ap2.add_argument("-ls", action="store_true", help="list summary for db")
 | |
|     ap2.add_argument("-cmp", action="store_true", help="compare databases")
 | |
| 
 | |
|     ap2 = ap.add_argument_group("options which modify target db")
 | |
|     ap2.add_argument("-copy", metavar="TAG", type=str, help="mtp tag to copy over")
 | |
|     ap2.add_argument(
 | |
|         "-rm-mtp-flag",
 | |
|         action="store_true",
 | |
|         help="when an mtp tag is copied over, also mark that as done, so copyparty won't run mtp on it",
 | |
|     )
 | |
|     ap2.add_argument("-vac", action="store_true", help="optimize DB")
 | |
| 
 | |
|     ar = ap.parse_args()
 | |
| 
 | |
|     for v in [ar.db, ar.src]:
 | |
|         if v and not os.path.exists(v):
 | |
|             die("database must exist")
 | |
| 
 | |
|     db = sqlite3.connect(ar.db)
 | |
|     ds = sqlite3.connect(ar.src) if ar.src else None
 | |
| 
 | |
|     for d, n in [[ds, "src"], [db, "dst"]]:
 | |
|         if not d:
 | |
|             continue
 | |
| 
 | |
|         ver = read_ver(d)
 | |
|         if ver == "corrupt":
 | |
|             die("{} database appears to be corrupt, sorry")
 | |
| 
 | |
|         if ver != DB_VER:
 | |
|             m = f"{n} db is version {ver}, this tool only supports version {DB_VER}, please upgrade it with copyparty first"
 | |
|             die(m)
 | |
| 
 | |
|     if ar.ls:
 | |
|         ls(db)
 | |
| 
 | |
|     if ar.cmp:
 | |
|         if not ds:
 | |
|             die("need src db to compare against")
 | |
| 
 | |
|         compare("src", ds, "dst", db, ar.v)
 | |
|         compare("dst", db, "src", ds, ar.v)
 | |
| 
 | |
|     if ar.copy:
 | |
|         copy_mtp(ds, db, ar.copy, ar.rm_mtp_flag)
 | |
| 
 | |
| 
 | |
| if __name__ == "__main__":
 | |
|     main()
 |