John6666's picture
Upload 3 files
d83c4c9 verified
import argparse
from pathlib import Path
import json
import re
import gc
from safetensors.torch import load_file, save_file
import torch
SDXL_KEYS_FILE = "keys/sdxl_keys.txt"
def list_uniq(l):
return sorted(set(l), key=l.index)
def read_safetensors_metadata(path: str):
with open(path, 'rb') as f:
header_size = int.from_bytes(f.read(8), 'little')
header_json = f.read(header_size).decode('utf-8')
header = json.loads(header_json)
metadata = header.get('__metadata__', {})
return metadata
def keys_from_file(path: str):
keys = []
try:
with open(str(Path(path)), encoding='utf-8', mode='r') as f:
lines = f.readlines()
for line in lines:
keys.append(line.strip())
except Exception as e:
print(e)
finally:
return keys
def validate_keys(keys: list[str], rfile: str=SDXL_KEYS_FILE):
missing = []
added = []
try:
rkeys = keys_from_file(rfile)
all_keys = list_uniq(keys + rkeys)
for key in all_keys:
if key in set(rkeys) and key not in set(keys): missing.append(key)
if key in set(keys) and key not in set(rkeys): added.append(key)
except Exception as e:
print(e)
finally:
return missing, added
def read_safetensors_key(path: str):
try:
keys = []
state_dict = load_file(str(Path(path)))
for k in list(state_dict.keys()):
keys.append(k)
state_dict.pop(k)
except Exception as e:
print(e)
finally:
del state_dict
torch.cuda.empty_cache()
gc.collect()
return keys
def write_safetensors_key(keys: list[str], path: str, is_validate: bool=True, rpath: str=SDXL_KEYS_FILE):
if len(keys) == 0: return False
try:
with open(str(Path(path)), encoding='utf-8', mode='w') as f:
f.write("\n".join(keys))
if is_validate:
missing, added = validate_keys(keys, rpath)
with open(str(Path(path).stem + "_missing.txt"), encoding='utf-8', mode='w') as f:
f.write("\n".join(missing))
with open(str(Path(path).stem + "_added.txt"), encoding='utf-8', mode='w') as f:
f.write("\n".join(added))
return True
except Exception as e:
print(e)
return False
def stkey(input: str, out_filename: str="", is_validate: bool=True, rfile: str=SDXL_KEYS_FILE):
keys = read_safetensors_key(input)
if len(keys) != 0 and out_filename: write_safetensors_key(keys, out_filename, is_validate, rfile)
if len(keys) != 0:
print("Metadata:")
print(read_safetensors_metadata(input))
print("\nKeys:")
print("\n".join(keys))
if is_validate:
missing, added = validate_keys(keys, rfile)
print("\nMissing Keys:")
print("\n".join(missing))
print("\nAdded Keys:")
print("\n".join(added))
if __name__ == "__main__":
parser = argparse.ArgumentParser()
parser.add_argument("input", type=str, help="Input safetensors file.")
parser.add_argument("-s", "--save", action="store_true", default=False, help="Output to text file.")
parser.add_argument("-o", "--output", default="", type=str, help="Output to specific text file.")
parser.add_argument("-v", "--val", action="store_false", default=True, help="Disable key validation.")
parser.add_argument("-r", "--rfile", default=SDXL_KEYS_FILE, type=str, help="Specify reference file to validate keys.")
args = parser.parse_args()
if args.save: out_filename = Path(args.input).stem + ".txt"
out_filename = args.output if args.output else out_filename
stkey(args.input, out_filename, args.val, args.rfile)
# Usage:
# python stkey.py sd_xl_base_1.0_0.9vae.safetensors
# python stkey.py sd_xl_base_1.0_0.9vae.safetensors -s
# python stkey.py sd_xl_base_1.0_0.9vae.safetensors -o key.txt