2023-08-09 15:25:35 +08:00
|
|
|
import os
|
|
|
|
import re
|
|
|
|
|
|
|
|
from modules import shared
|
2023-12-28 05:22:51 +08:00
|
|
|
from modules.paths_internal import script_path, cwd
|
2023-08-09 15:25:35 +08:00
|
|
|
|
|
|
|
|
|
|
|
def natural_sort_key(s, regex=re.compile('([0-9]+)')):
|
|
|
|
return [int(text) if text.isdigit() else text.lower() for text in regex.split(s)]
|
|
|
|
|
|
|
|
|
|
|
|
def listfiles(dirname):
|
|
|
|
filenames = [os.path.join(dirname, x) for x in sorted(os.listdir(dirname), key=natural_sort_key) if not x.startswith(".")]
|
|
|
|
return [file for file in filenames if os.path.isfile(file)]
|
|
|
|
|
|
|
|
|
|
|
|
def html_path(filename):
|
|
|
|
return os.path.join(script_path, "html", filename)
|
|
|
|
|
|
|
|
|
|
|
|
def html(filename):
|
|
|
|
path = html_path(filename)
|
|
|
|
|
2024-01-04 06:16:58 +08:00
|
|
|
try:
|
2023-08-09 15:25:35 +08:00
|
|
|
with open(path, encoding="utf8") as file:
|
|
|
|
return file.read()
|
2024-01-04 06:16:58 +08:00
|
|
|
except OSError:
|
|
|
|
return ""
|
2023-08-09 15:25:35 +08:00
|
|
|
|
|
|
|
|
|
|
|
def walk_files(path, allowed_extensions=None):
|
|
|
|
if not os.path.exists(path):
|
|
|
|
return
|
|
|
|
|
|
|
|
if allowed_extensions is not None:
|
|
|
|
allowed_extensions = set(allowed_extensions)
|
|
|
|
|
|
|
|
items = list(os.walk(path, followlinks=True))
|
|
|
|
items = sorted(items, key=lambda x: natural_sort_key(x[0]))
|
|
|
|
|
|
|
|
for root, _, files in items:
|
|
|
|
for filename in sorted(files, key=natural_sort_key):
|
|
|
|
if allowed_extensions is not None:
|
|
|
|
_, ext = os.path.splitext(filename)
|
2024-02-10 04:39:01 +08:00
|
|
|
if ext.lower() not in allowed_extensions:
|
2023-08-09 15:25:35 +08:00
|
|
|
continue
|
|
|
|
|
|
|
|
if not shared.opts.list_hidden_files and ("/." in root or "\\." in root):
|
|
|
|
continue
|
|
|
|
|
|
|
|
yield os.path.join(root, filename)
|
|
|
|
|
|
|
|
|
|
|
|
def ldm_print(*args, **kwargs):
|
|
|
|
if shared.opts.hide_ldm_prints:
|
|
|
|
return
|
|
|
|
|
|
|
|
print(*args, **kwargs)
|
2023-12-28 05:22:51 +08:00
|
|
|
|
|
|
|
|
|
|
|
def truncate_path(target_path, base_path=cwd):
|
|
|
|
abs_target, abs_base = os.path.abspath(target_path), os.path.abspath(base_path)
|
|
|
|
try:
|
|
|
|
if os.path.commonpath([abs_target, abs_base]) == abs_base:
|
|
|
|
return os.path.relpath(abs_target, abs_base)
|
|
|
|
except ValueError:
|
|
|
|
pass
|
|
|
|
return abs_target
|
2024-01-04 07:28:05 +08:00
|
|
|
|
|
|
|
|
|
|
|
class MassFileListerCachedDir:
|
|
|
|
"""A class that caches file metadata for a specific directory."""
|
|
|
|
|
|
|
|
def __init__(self, dirname):
|
|
|
|
self.files = None
|
|
|
|
self.files_cased = None
|
|
|
|
self.dirname = dirname
|
|
|
|
|
|
|
|
stats = ((x.name, x.stat(follow_symlinks=False)) for x in os.scandir(self.dirname))
|
|
|
|
files = [(n, s.st_mtime, s.st_ctime) for n, s in stats]
|
|
|
|
self.files = {x[0].lower(): x for x in files}
|
|
|
|
self.files_cased = {x[0]: x for x in files}
|
|
|
|
|
2024-03-10 05:07:16 +08:00
|
|
|
def update_entry(self, filename):
|
|
|
|
"""Add a file to the cache"""
|
|
|
|
file_path = os.path.join(self.dirname, filename)
|
|
|
|
try:
|
|
|
|
stat = os.stat(file_path)
|
|
|
|
entry = (filename, stat.st_mtime, stat.st_ctime)
|
|
|
|
self.files[filename.lower()] = entry
|
|
|
|
self.files_cased[filename] = entry
|
|
|
|
except FileNotFoundError as e:
|
|
|
|
print(f'MassFileListerCachedDir.add_entry: "{file_path}" {e}')
|
|
|
|
|
2024-01-04 07:28:05 +08:00
|
|
|
|
|
|
|
class MassFileLister:
|
|
|
|
"""A class that provides a way to check for the existence and mtime/ctile of files without doing more than one stat call per file."""
|
|
|
|
|
|
|
|
def __init__(self):
|
|
|
|
self.cached_dirs = {}
|
|
|
|
|
|
|
|
def find(self, path):
|
|
|
|
"""
|
|
|
|
Find the metadata for a file at the given path.
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
tuple or None: A tuple of (name, mtime, ctime) if the file exists, or None if it does not.
|
|
|
|
"""
|
|
|
|
|
|
|
|
dirname, filename = os.path.split(path)
|
|
|
|
|
|
|
|
cached_dir = self.cached_dirs.get(dirname)
|
|
|
|
if cached_dir is None:
|
|
|
|
cached_dir = MassFileListerCachedDir(dirname)
|
|
|
|
self.cached_dirs[dirname] = cached_dir
|
|
|
|
|
|
|
|
stats = cached_dir.files_cased.get(filename)
|
|
|
|
if stats is not None:
|
|
|
|
return stats
|
|
|
|
|
|
|
|
stats = cached_dir.files.get(filename.lower())
|
|
|
|
if stats is None:
|
|
|
|
return None
|
|
|
|
|
|
|
|
try:
|
|
|
|
os_stats = os.stat(path, follow_symlinks=False)
|
|
|
|
return filename, os_stats.st_mtime, os_stats.st_ctime
|
|
|
|
except Exception:
|
|
|
|
return None
|
|
|
|
|
|
|
|
def exists(self, path):
|
|
|
|
"""Check if a file exists at the given path."""
|
|
|
|
|
|
|
|
return self.find(path) is not None
|
|
|
|
|
|
|
|
def mctime(self, path):
|
|
|
|
"""
|
|
|
|
Get the modification and creation times for a file at the given path.
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
tuple: A tuple of (mtime, ctime) if the file exists, or (0, 0) if it does not.
|
|
|
|
"""
|
|
|
|
|
|
|
|
stats = self.find(path)
|
|
|
|
return (0, 0) if stats is None else stats[1:3]
|
|
|
|
|
|
|
|
def reset(self):
|
|
|
|
"""Clear the cache of all directories."""
|
|
|
|
self.cached_dirs.clear()
|
2024-04-05 14:51:42 +08:00
|
|
|
|
2024-04-06 14:56:57 +08:00
|
|
|
def update_file_entry(self, path):
|
|
|
|
"""Update the cache for a specific directory."""
|
|
|
|
dirname, filename = os.path.split(path)
|
|
|
|
if cached_dir := self.cached_dirs.get(dirname):
|
|
|
|
cached_dir.update_entry(filename)
|
2024-03-10 20:14:04 +08:00
|
|
|
|
|
|
|
def topological_sort(dependencies):
|
|
|
|
"""Accepts a dictionary mapping name to its dependencies, returns a list of names ordered according to dependencies.
|
2024-06-21 09:52:02 +08:00
|
|
|
Ignores errors relating to missing dependencies or circular dependencies
|
2024-03-10 20:14:04 +08:00
|
|
|
"""
|
|
|
|
|
|
|
|
visited = {}
|
|
|
|
result = []
|
|
|
|
|
|
|
|
def inner(name):
|
|
|
|
visited[name] = True
|
|
|
|
|
|
|
|
for dep in dependencies.get(name, []):
|
|
|
|
if dep in dependencies and dep not in visited:
|
|
|
|
inner(dep)
|
|
|
|
|
|
|
|
result.append(name)
|
|
|
|
|
|
|
|
for depname in dependencies:
|
|
|
|
if depname not in visited:
|
|
|
|
inner(depname)
|
|
|
|
|
|
|
|
return result
|
2024-04-05 15:14:38 +08:00
|
|
|
|
|
|
|
|
2024-04-05 14:51:42 +08:00
|
|
|
def open_folder(path):
|
2024-04-05 15:19:20 +08:00
|
|
|
"""Open a folder in the file manager of the respect OS."""
|
2024-04-05 14:51:42 +08:00
|
|
|
# import at function level to avoid potential issues
|
|
|
|
import gradio as gr
|
|
|
|
import platform
|
|
|
|
import sys
|
|
|
|
import subprocess
|
|
|
|
|
|
|
|
if not os.path.exists(path):
|
|
|
|
msg = f'Folder "{path}" does not exist. after you save an image, the folder will be created.'
|
|
|
|
print(msg)
|
|
|
|
gr.Info(msg)
|
|
|
|
return
|
|
|
|
elif not os.path.isdir(path):
|
|
|
|
msg = f"""
|
|
|
|
WARNING
|
|
|
|
An open_folder request was made with an path that is not a folder.
|
|
|
|
This could be an error or a malicious attempt to run code on your computer.
|
|
|
|
Requested path was: {path}
|
|
|
|
"""
|
|
|
|
print(msg, file=sys.stderr)
|
|
|
|
gr.Warning(msg)
|
|
|
|
return
|
|
|
|
|
|
|
|
path = os.path.normpath(path)
|
|
|
|
if platform.system() == "Windows":
|
|
|
|
os.startfile(path)
|
|
|
|
elif platform.system() == "Darwin":
|
|
|
|
subprocess.Popen(["open", path])
|
|
|
|
elif "microsoft-standard-WSL2" in platform.uname().release:
|
2024-06-04 01:23:43 +08:00
|
|
|
subprocess.Popen(["explorer.exe", subprocess.check_output(["wslpath", "-w", path])])
|
2024-04-05 14:51:42 +08:00
|
|
|
else:
|
|
|
|
subprocess.Popen(["xdg-open", path])
|
2024-10-24 21:05:51 +08:00
|
|
|
|
|
|
|
|
|
|
|
def load_file_from_url(
|
|
|
|
url: str,
|
|
|
|
*,
|
|
|
|
model_dir: str,
|
|
|
|
progress: bool = True,
|
|
|
|
file_name: str | None = None,
|
|
|
|
hash_prefix: str | None = None,
|
|
|
|
re_download: bool = False,
|
|
|
|
) -> str:
|
|
|
|
"""Download a file from `url` into `model_dir`, using the file present if possible.
|
|
|
|
Returns the path to the downloaded file.
|
|
|
|
|
|
|
|
file_name: if specified, it will be used as the filename, otherwise the filename will be extracted from the url.
|
|
|
|
file is downloaded to {file_name}.tmp then moved to the final location after download is complete.
|
|
|
|
hash_prefix: sha256 hex string, if provided, the hash of the downloaded file will be checked against this prefix.
|
|
|
|
if the hash does not match, the temporary file is deleted and a ValueError is raised.
|
|
|
|
re_download: forcibly re-download the file even if it already exists.
|
|
|
|
"""
|
|
|
|
from urllib.parse import urlparse
|
|
|
|
import requests
|
|
|
|
try:
|
|
|
|
from tqdm import tqdm
|
|
|
|
except ImportError:
|
|
|
|
class tqdm:
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
|
|
pass
|
|
|
|
|
|
|
|
def update(self, n=1, *args, **kwargs):
|
|
|
|
pass
|
|
|
|
|
|
|
|
def __enter__(self):
|
|
|
|
return self
|
|
|
|
|
|
|
|
def __exit__(self, exc_type, exc_val, exc_tb):
|
|
|
|
pass
|
|
|
|
|
|
|
|
if not file_name:
|
|
|
|
parts = urlparse(url)
|
|
|
|
file_name = os.path.basename(parts.path)
|
|
|
|
|
|
|
|
cached_file = os.path.abspath(os.path.join(model_dir, file_name))
|
|
|
|
|
|
|
|
if re_download or not os.path.exists(cached_file):
|
|
|
|
os.makedirs(model_dir, exist_ok=True)
|
|
|
|
temp_file = os.path.join(model_dir, f"{file_name}.tmp")
|
|
|
|
print(f'\nDownloading: "{url}" to {cached_file}')
|
|
|
|
response = requests.get(url, stream=True)
|
|
|
|
response.raise_for_status()
|
|
|
|
total_size = int(response.headers.get('content-length', 0))
|
|
|
|
with tqdm(total=total_size, unit='B', unit_scale=True, desc=file_name, disable=not progress) as progress_bar:
|
|
|
|
with open(temp_file, 'wb') as file:
|
|
|
|
for chunk in response.iter_content(chunk_size=1024):
|
|
|
|
if chunk:
|
|
|
|
file.write(chunk)
|
|
|
|
progress_bar.update(len(chunk))
|
|
|
|
|
|
|
|
if hash_prefix and not compare_sha256(temp_file, hash_prefix):
|
|
|
|
print(f"Hash mismatch for {temp_file}. Deleting the temporary file.")
|
|
|
|
os.remove(temp_file)
|
|
|
|
raise ValueError(f"File hash does not match the expected hash prefix {hash_prefix}!")
|
|
|
|
|
|
|
|
os.rename(temp_file, cached_file)
|
|
|
|
return cached_file
|
|
|
|
|
|
|
|
|
|
|
|
def compare_sha256(file_path: str, hash_prefix: str) -> bool:
|
|
|
|
"""Check if the SHA256 hash of the file matches the given prefix."""
|
|
|
|
import hashlib
|
|
|
|
hash_sha256 = hashlib.sha256()
|
|
|
|
blksize = 1024 * 1024
|
|
|
|
|
|
|
|
with open(file_path, "rb") as f:
|
|
|
|
for chunk in iter(lambda: f.read(blksize), b""):
|
|
|
|
hash_sha256.update(chunk)
|
|
|
|
return hash_sha256.hexdigest().startswith(hash_prefix.strip().lower())
|