import json import os import sys import subprocess import platform import hashlib import re from pathlib import Path from modules import paths_internal, timer, shared_cmd_options, errors, launch_utils checksum_token = "DontStealMyGamePlz__WINNERS_DONT_USE_DRUGS__DONT_COPY_THAT_FLOPPY" environment_whitelist = { "GIT", "INDEX_URL", "WEBUI_LAUNCH_LIVE_OUTPUT", "GRADIO_ANALYTICS_ENABLED", "PYTHONPATH", "TORCH_INDEX_URL", "TORCH_COMMAND", "REQS_FILE", "XFORMERS_PACKAGE", "CLIP_PACKAGE", "OPENCLIP_PACKAGE", "ASSETS_REPO", "STABLE_DIFFUSION_REPO", "K_DIFFUSION_REPO", "BLIP_REPO", "ASSETS_COMMIT_HASH", "STABLE_DIFFUSION_COMMIT_HASH", "K_DIFFUSION_COMMIT_HASH", "BLIP_COMMIT_HASH", "COMMANDLINE_ARGS", "IGNORE_CMD_ARGS_ERRORS", } def pretty_bytes(num, suffix="B"): for unit in ["", "K", "M", "G", "T", "P", "E", "Z", "Y"]: if abs(num) < 1024 or unit == 'Y': return f"{num:.0f}{unit}{suffix}" num /= 1024 def get(): res = get_dict() text = json.dumps(res, ensure_ascii=False, indent=4) h = hashlib.sha256(text.encode("utf8")) text = text.replace(checksum_token, h.hexdigest()) return text re_checksum = re.compile(r'"Checksum": "([0-9a-fA-F]{64})"') def check(x): m = re.search(re_checksum, x) if not m: return False replaced = re.sub(re_checksum, f'"Checksum": "{checksum_token}"', x) h = hashlib.sha256(replaced.encode("utf8")) return h.hexdigest() == m.group(1) def get_cpu_info(): cpu_info = {"model": platform.processor()} try: import psutil cpu_info["count logical"] = psutil.cpu_count(logical=True) cpu_info["count physical"] = psutil.cpu_count(logical=False) except Exception as e: cpu_info["error"] = str(e) return cpu_info def get_ram_info(): try: import psutil ram = psutil.virtual_memory() return {x: pretty_bytes(getattr(ram, x, 0)) for x in ["total", "used", "free", "active", "inactive", "buffers", "cached", "shared"] if getattr(ram, x, 0) != 0} except Exception as e: return str(e) def get_packages(): try: return subprocess.check_output([sys.executable, '-m', 'pip', 'freeze', '--all']).decode("utf8").splitlines() except Exception as pip_error: try: import importlib.metadata packages = importlib.metadata.distributions() return sorted([f"{package.metadata['Name']}=={package.version}" for package in packages]) except Exception as e2: return {'error pip': pip_error, 'error importlib': str(e2)} def get_dict(): config = get_config() res = { "Platform": platform.platform(), "Python": platform.python_version(), "Version": launch_utils.git_tag(), "Commit": launch_utils.commit_hash(), "Git status": git_status(paths_internal.script_path), "Script path": paths_internal.script_path, "Data path": paths_internal.data_path, "Extensions dir": paths_internal.extensions_dir, "Checksum": checksum_token, "Commandline": get_argv(), "Torch env info": get_torch_sysinfo(), "Exceptions": errors.get_exceptions(), "CPU": get_cpu_info(), "RAM": get_ram_info(), "Extensions": get_extensions(enabled=True, fallback_disabled_extensions=config.get('disabled_extensions', [])), "Inactive extensions": get_extensions(enabled=False, fallback_disabled_extensions=config.get('disabled_extensions', [])), "Environment": get_environment(), "Config": config, "Startup": timer.startup_record, "Packages": get_packages(), } return res def get_environment(): return {k: os.environ[k] for k in sorted(os.environ) if k in environment_whitelist} def get_argv(): res = [] for v in sys.argv: if shared_cmd_options.cmd_opts.gradio_auth and shared_cmd_options.cmd_opts.gradio_auth == v: res.append("") continue if shared_cmd_options.cmd_opts.api_auth and shared_cmd_options.cmd_opts.api_auth == v: res.append("") continue res.append(v) return res re_newline = re.compile(r"\r*\n") def get_torch_sysinfo(): try: import torch.utils.collect_env info = torch.utils.collect_env.get_env_info()._asdict() return {k: re.split(re_newline, str(v)) if "\n" in str(v) else v for k, v in info.items()} except Exception as e: return str(e) def run_git(path, *args): try: return subprocess.check_output([launch_utils.git, '-C', path, *args], shell=False, encoding='utf8').strip() except Exception as e: return str(e) def git_status(path): if (Path(path) / '.git').is_dir(): return run_git(paths_internal.script_path, 'status') def get_info_from_repo_path(path: Path): is_repo = (path / '.git').is_dir() return { 'name': path.name, 'path': str(path), 'commit': run_git(path, 'rev-parse', 'HEAD') if is_repo else None, 'branch': run_git(path, 'branch', '--show-current') if is_repo else None, 'remote': run_git(path, 'remote', 'get-url', 'origin') if is_repo else None, } def get_extensions(*, enabled, fallback_disabled_extensions=None): try: from modules import extensions if extensions.extensions: def to_json(x: extensions.Extension): return { "name": x.name, "path": x.path, "commit": x.commit_hash, "branch": x.branch, "remote": x.remote, } return [to_json(x) for x in extensions.extensions if not x.is_builtin and x.enabled == enabled] else: return [get_info_from_repo_path(d) for d in Path(paths_internal.extensions_dir).iterdir() if d.is_dir() and enabled != (str(d.name) in fallback_disabled_extensions)] except Exception as e: return str(e) def get_config(): try: from modules import shared return shared.opts.data except Exception as _: try: with open(shared_cmd_options.cmd_opts.ui_settings_file, 'r') as f: return json.load(f) except Exception as e: return str(e) def download_sysinfo(attachment=False): from fastapi.responses import PlainTextResponse from datetime import datetime text = get() filename = f"sysinfo-{datetime.utcnow().strftime('%Y-%m-%d-%H-%M')}.json" return PlainTextResponse(text, headers={'Content-Disposition': f'{"attachment" if attachment else "inline"}; filename="{filename}"'})