Compare commits
218 Commits
2.51.1
...
refactor/p
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
34a48fbae4 | ||
|
|
76b073c366 | ||
|
|
fa357479ef | ||
|
|
19a9c24485 | ||
|
|
05ecca1f4d | ||
|
|
58847298be | ||
|
|
a9f8cecaec | ||
|
|
441b4d2797 | ||
|
|
1808bc3027 | ||
|
|
d4db1a51d2 | ||
|
|
f06ac47557 | ||
|
|
d8fb8ce606 | ||
|
|
26b7816552 | ||
|
|
8576f9c97f | ||
|
|
c4eecca6b6 | ||
|
|
02f1788261 | ||
|
|
4a908d970a | ||
|
|
0278ee2515 | ||
|
|
eabde97d17 | ||
|
|
eb88e06ab1 | ||
|
|
11ba0ed696 | ||
|
|
73f477941d | ||
|
|
2de4b7e4c4 | ||
|
|
9552ce6210 | ||
|
|
688bf25be9 | ||
|
|
14f567c031 | ||
|
|
1ab304bfbc | ||
|
|
2e2a7cee79 | ||
|
|
a3c0ed8113 | ||
|
|
642a3673af | ||
|
|
971653f117 | ||
|
|
ff8fb087e0 | ||
|
|
8be3cce9a9 | ||
|
|
c7b2ec7502 | ||
|
|
f2da376df1 | ||
|
|
90035ded39 | ||
|
|
8c593fe3f6 | ||
|
|
96a4b73bb4 | ||
|
|
83fdaaf409 | ||
|
|
f17dd82d8e | ||
|
|
81929b9ec1 | ||
|
|
b8dfdfa31d | ||
|
|
4021be531e | ||
|
|
f985c02ce7 | ||
|
|
602eacf0ed | ||
|
|
bcca781896 | ||
|
|
b6a8e6ba81 | ||
|
|
4768c46151 | ||
|
|
fef46c71a0 | ||
|
|
acfbc8ef04 | ||
|
|
e72cd14d9d | ||
|
|
fecd317275 | ||
|
|
d848dcf56c | ||
|
|
df849b4ca8 | ||
|
|
0b3e627de6 | ||
|
|
4fc1fed44a | ||
|
|
bf2ffe8dfa | ||
|
|
e73e3b7726 | ||
|
|
e519c9c5f3 | ||
|
|
76aa2e9354 | ||
|
|
67e13d7249 | ||
|
|
60de0ffb3a | ||
|
|
d4fa00ab96 | ||
|
|
25aebd030a | ||
|
|
7ce5b8f5fb | ||
|
|
af298b86d7 | ||
|
|
cfdb687db3 | ||
|
|
6f9feb00ee | ||
|
|
7214c58b4b | ||
|
|
47aeefe395 | ||
|
|
a9708513aa | ||
|
|
26c1db4091 | ||
|
|
37e07338ca | ||
|
|
c4879aef3d | ||
|
|
ac9b41fdae | ||
|
|
137d998e46 | ||
|
|
b0eda6166d | ||
|
|
d0c48ad350 | ||
|
|
f20c5b66b9 | ||
|
|
536c27233c | ||
|
|
92a8a98000 | ||
|
|
40a4631eba | ||
|
|
cd71e87d0e | ||
|
|
8612bb1dea | ||
|
|
f1a0dabf15 | ||
|
|
17e256544c | ||
|
|
a401c51364 | ||
|
|
a68eb721d5 | ||
|
|
cd98b90c35 | ||
|
|
b56f3fb3a8 | ||
|
|
4e7e66ac9b | ||
|
|
039fdc0384 | ||
|
|
cbd19c4b48 | ||
|
|
e166ba5c24 | ||
|
|
8c45a5ee84 | ||
|
|
81513888e3 | ||
|
|
b1ed3eff49 | ||
|
|
f8d9425c7f | ||
|
|
c7ea960718 | ||
|
|
ca4c09ceca | ||
|
|
16176d759a | ||
|
|
8b11764b08 | ||
|
|
a9dabbdd20 | ||
|
|
294244b99f | ||
|
|
0a9a8e418b | ||
|
|
b04840ffa3 | ||
|
|
2fe812084a | ||
|
|
c4b640f073 | ||
|
|
51a5209218 | ||
|
|
988cda9b76 | ||
|
|
1c0797d990 | ||
|
|
90d9b5e39a | ||
|
|
6f2f97ec06 | ||
|
|
9905c2a3da | ||
|
|
008d19732a | ||
|
|
93f8614070 | ||
|
|
cbf5cfa4d3 | ||
|
|
c365c3388e | ||
|
|
77496b30fd | ||
|
|
52568da4a8 | ||
|
|
588efab2c7 | ||
|
|
1afd444238 | ||
|
|
9bd335f7d3 | ||
|
|
12cb8dfcaf | ||
|
|
7f3a70bf6b | ||
|
|
967dc4e4dd | ||
|
|
d6e30e48d9 | ||
|
|
055101d1fd | ||
|
|
d56d64985e | ||
|
|
3e2915d531 | ||
|
|
65caaaf7a8 | ||
|
|
58f3332dd6 | ||
|
|
4c09464838 | ||
|
|
97aafacd40 | ||
|
|
ade30470cc | ||
|
|
4242dc0f94 | ||
|
|
9f4054e7bf | ||
|
|
0012b8cd6e | ||
|
|
c41aaca444 | ||
|
|
290de33bbd | ||
|
|
9be2572a82 | ||
|
|
5aba0c509c | ||
|
|
6225159434 | ||
|
|
0cac32d0d5 | ||
|
|
75d57e4335 | ||
|
|
9e95e3162a | ||
|
|
5dc31e9fde | ||
|
|
ba678cb92b | ||
|
|
e8bb7ccda7 | ||
|
|
b139bfabf0 | ||
|
|
e9a5137889 | ||
|
|
690b7a7d05 | ||
|
|
5c8292d804 | ||
|
|
c88266f76c | ||
|
|
d8da2a8c50 | ||
|
|
4d09f6345e | ||
|
|
8b1be9d35a | ||
|
|
a1110c22b0 | ||
|
|
405e83b79c | ||
|
|
e3ab7e86ce | ||
|
|
1dc683e872 | ||
|
|
3fb3f07644 | ||
|
|
7f937aeaae | ||
|
|
ff93a16548 | ||
|
|
bec91ae52d | ||
|
|
bc8a624dbd | ||
|
|
ca9f8dad5c | ||
|
|
496ca64008 | ||
|
|
2383e134de | ||
|
|
38d1bd613f | ||
|
|
3315807a37 | ||
|
|
8dd84bb0ec | ||
|
|
e9eaff7f7e | ||
|
|
0469cad3bc | ||
|
|
d289dd523a | ||
|
|
1cb6adb3dc | ||
|
|
e93d0ab0f2 | ||
|
|
61537d985d | ||
|
|
c999886fe0 | ||
|
|
0e6e391caf | ||
|
|
26573ce489 | ||
|
|
c2869c8c67 | ||
|
|
792a836121 | ||
|
|
72ef927b32 | ||
|
|
bb67901e92 | ||
|
|
f87d4ea150 | ||
|
|
7ec376774f | ||
|
|
21ed4bd420 | ||
|
|
93b8c39ad7 | ||
|
|
6ea771781f | ||
|
|
7fc4343dac | ||
|
|
b12bd3849c | ||
|
|
78d9cbe659 | ||
|
|
3e3800bc59 | ||
|
|
9adbff6e16 | ||
|
|
f7a2e3f874 | ||
|
|
5b78ce0bff | ||
|
|
2f2ac86513 | ||
|
|
fbb1cd60d6 | ||
|
|
51d3f3829c | ||
|
|
53c8fc7751 | ||
|
|
22fa2825f8 | ||
|
|
752c17df3c | ||
|
|
66542292c3 | ||
|
|
4745597c0a | ||
|
|
1e7d101063 | ||
|
|
4dc9ee9363 | ||
|
|
07f9a34036 | ||
|
|
97b55a0da6 | ||
|
|
cb8bcd7edf | ||
|
|
eb607a6b41 | ||
|
|
32430968b8 | ||
|
|
f3e9856107 | ||
|
|
d9ec870598 | ||
|
|
a182e526d7 | ||
|
|
a8d338a997 | ||
|
|
9589f343df | ||
|
|
71c602be95 |
2
.github/workflows/publish.yml
vendored
2
.github/workflows/publish.yml
vendored
@@ -3,7 +3,7 @@ on:
|
|||||||
workflow_dispatch:
|
workflow_dispatch:
|
||||||
push:
|
push:
|
||||||
branches:
|
branches:
|
||||||
- main
|
- main-blocked
|
||||||
paths:
|
paths:
|
||||||
- "pyproject.toml"
|
- "pyproject.toml"
|
||||||
|
|
||||||
|
|||||||
@@ -363,10 +363,10 @@ When you run the `scan.sh` script:
|
|||||||
* `high` level risky features
|
* `high` level risky features
|
||||||
* `Install via git url`, `pip install`
|
* `Install via git url`, `pip install`
|
||||||
* Installation of custom nodes registered not in the `default channel`.
|
* Installation of custom nodes registered not in the `default channel`.
|
||||||
* Display terminal log
|
* Fix custom nodes
|
||||||
|
|
||||||
* `middle` level risky features
|
* `middle` level risky features
|
||||||
* Uninstall/Update/Fix custom nodes
|
* Uninstall/Update
|
||||||
* Installation of custom nodes registered in the `default channel`.
|
* Installation of custom nodes registered in the `default channel`.
|
||||||
* Restore/Remove Snapshot
|
* Restore/Remove Snapshot
|
||||||
* Restart
|
* Restart
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
6666
github-stats.json
6666
github-stats.json
File diff suppressed because it is too large
Load Diff
0
glob/__init__.py
Normal file
0
glob/__init__.py
Normal file
174
glob/git_wrapper.py
Normal file
174
glob/git_wrapper.py
Normal file
@@ -0,0 +1,174 @@
|
|||||||
|
import pygit2
|
||||||
|
import os
|
||||||
|
from tqdm import tqdm
|
||||||
|
import traceback
|
||||||
|
|
||||||
|
class GitProgress(pygit2.RemoteCallbacks):
|
||||||
|
def __init__(self):
|
||||||
|
super().__init__()
|
||||||
|
self.pbar = None
|
||||||
|
|
||||||
|
def transfer_progress(self, stats):
|
||||||
|
if self.pbar is None:
|
||||||
|
self.pbar = tqdm(total=stats.total_objects, unit="obj", desc="Fetching objects")
|
||||||
|
self.pbar.n = stats.received_objects
|
||||||
|
self.pbar.refresh()
|
||||||
|
if stats.received_objects == stats.total_objects:
|
||||||
|
self.pbar.close()
|
||||||
|
self.pbar = None
|
||||||
|
|
||||||
|
|
||||||
|
class Remote:
|
||||||
|
def __init__(self, repo, remote):
|
||||||
|
self.repo = repo
|
||||||
|
self.remote = remote
|
||||||
|
|
||||||
|
def get_default_branch(self, remote_name='origin'):
|
||||||
|
remote = self.repo.remotes[remote_name]
|
||||||
|
remote.fetch() # Fetch latest data from the remote
|
||||||
|
|
||||||
|
# Look for the remote HEAD reference
|
||||||
|
head_ref = f'refs/remotes/{remote_name}/HEAD'
|
||||||
|
if head_ref in self.repo.references:
|
||||||
|
# Resolve the symbolic reference to get the actual branch
|
||||||
|
target_ref = self.repo.references[head_ref].resolve().name
|
||||||
|
return target_ref.replace(f'refs/remotes/{remote_name}/', '')
|
||||||
|
else:
|
||||||
|
raise ValueError(f"Could not determine the default branch for remote '{remote_name}'")
|
||||||
|
|
||||||
|
|
||||||
|
def pull(self, remote_name='origin'):
|
||||||
|
try:
|
||||||
|
# Detect if we are in detached HEAD state
|
||||||
|
if self.repo.head_is_detached:
|
||||||
|
# Find the default branch
|
||||||
|
branch_name = self.get_default_branch(remote_name)
|
||||||
|
|
||||||
|
# Checkout the branch if exists, or create it
|
||||||
|
branch_ref = f"refs/heads/{branch_name}"
|
||||||
|
if branch_ref in self.repo.references:
|
||||||
|
self.repo.checkout(branch_ref)
|
||||||
|
else:
|
||||||
|
# Create and checkout the branch
|
||||||
|
target_commit = self.repo.lookup_reference(f"refs/remotes/{remote_name}/{branch_name}").target
|
||||||
|
self.repo.create_branch(branch_name, self.repo[target_commit])
|
||||||
|
self.repo.checkout(branch_ref)
|
||||||
|
|
||||||
|
# Get the current branch
|
||||||
|
current_branch = self.repo.head.shorthand
|
||||||
|
|
||||||
|
# Fetch from the remote
|
||||||
|
remote = self.repo.remotes[remote_name]
|
||||||
|
remote.fetch()
|
||||||
|
|
||||||
|
# Merge changes from the remote
|
||||||
|
remote_branch_ref = f"refs/remotes/{remote_name}/{current_branch}"
|
||||||
|
remote_branch = self.repo.lookup_reference(remote_branch_ref).target
|
||||||
|
|
||||||
|
self.repo.merge(remote_branch)
|
||||||
|
|
||||||
|
# Check for merge conflicts
|
||||||
|
if self.repo.index.conflicts is not None:
|
||||||
|
print("Merge conflicts detected!")
|
||||||
|
for conflict in self.repo.index.conflicts:
|
||||||
|
print(f"Conflict: {conflict}")
|
||||||
|
return
|
||||||
|
|
||||||
|
# Commit the merge
|
||||||
|
user = self.repo.default_signature
|
||||||
|
merge_commit = self.repo.create_commit(
|
||||||
|
'HEAD',
|
||||||
|
user,
|
||||||
|
user,
|
||||||
|
f"Merge branch '{current_branch}' from {remote_name}",
|
||||||
|
self.repo.index.write_tree(),
|
||||||
|
[self.repo.head.target, remote_branch]
|
||||||
|
)
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
traceback.print_exc()
|
||||||
|
print(f"An error occurred: {e}")
|
||||||
|
self.repo.state_cleanup() # Clean up the merge state if necessary
|
||||||
|
|
||||||
|
|
||||||
|
class Repo:
|
||||||
|
def __init__(self, repo_path):
|
||||||
|
self.repo = pygit2.Repository(repo_path)
|
||||||
|
|
||||||
|
def remote(self, name="origin"):
|
||||||
|
return Remote(self.repo, self.repo.remotes[name])
|
||||||
|
|
||||||
|
def update_recursive(self):
|
||||||
|
update_submodules(self.repo)
|
||||||
|
|
||||||
|
|
||||||
|
def resolve_repository_state(repo):
|
||||||
|
if repo.is_empty:
|
||||||
|
raise ValueError("Repository is empty. Cannot proceed with submodule update.")
|
||||||
|
|
||||||
|
try:
|
||||||
|
state = repo.state() # Call the state method
|
||||||
|
except Exception as e:
|
||||||
|
print(f"Error retrieving repository state: {e}")
|
||||||
|
raise
|
||||||
|
|
||||||
|
if state != pygit2.GIT_REPOSITORY_STATE_NONE:
|
||||||
|
if state in (pygit2.GIT_REPOSITORY_STATE_MERGE, pygit2.GIT_REPOSITORY_STATE_REVERT):
|
||||||
|
print(f"Conflict detected. Cleaning up repository state... {repo.path} / {state}")
|
||||||
|
repo.state_cleanup()
|
||||||
|
print("Repository state cleaned up.")
|
||||||
|
else:
|
||||||
|
raise RuntimeError(f"Unsupported repository state: {state}")
|
||||||
|
|
||||||
|
|
||||||
|
def update_submodules(repo):
|
||||||
|
try:
|
||||||
|
resolve_repository_state(repo)
|
||||||
|
except Exception as e:
|
||||||
|
print(f"Error resolving repository state: {e}")
|
||||||
|
return
|
||||||
|
|
||||||
|
gitmodules_path = os.path.join(repo.workdir, ".gitmodules")
|
||||||
|
if not os.path.exists(gitmodules_path):
|
||||||
|
return
|
||||||
|
|
||||||
|
with open(gitmodules_path, "r") as f:
|
||||||
|
lines = f.readlines()
|
||||||
|
|
||||||
|
submodules = []
|
||||||
|
submodule_path = None
|
||||||
|
submodule_url = None
|
||||||
|
|
||||||
|
for line in lines:
|
||||||
|
if line.strip().startswith("[submodule"):
|
||||||
|
if submodule_path and submodule_url:
|
||||||
|
submodules.append((submodule_path, submodule_url))
|
||||||
|
submodule_path = None
|
||||||
|
submodule_url = None
|
||||||
|
elif line.strip().startswith("path ="):
|
||||||
|
submodule_path = line.strip().split("=", 1)[1].strip()
|
||||||
|
elif line.strip().startswith("url ="):
|
||||||
|
submodule_url = line.strip().split("=", 1)[1].strip()
|
||||||
|
|
||||||
|
if submodule_path and submodule_url:
|
||||||
|
submodules.append((submodule_path, submodule_url))
|
||||||
|
|
||||||
|
for path, url in submodules:
|
||||||
|
submodule_repo_path = os.path.join(repo.workdir, path)
|
||||||
|
|
||||||
|
print(f"submodule_repo_path: {submodule_repo_path}")
|
||||||
|
|
||||||
|
if not os.path.exists(submodule_repo_path):
|
||||||
|
print(f"Cloning submodule {path}...")
|
||||||
|
pygit2.clone_repository(url, submodule_repo_path, callbacks=GitProgress())
|
||||||
|
else:
|
||||||
|
print(f"Updating submodule {path}...")
|
||||||
|
submodule_repo = Repo(submodule_repo_path)
|
||||||
|
submodule_repo.remote("origin").pull()
|
||||||
|
|
||||||
|
update_submodules(submodule_repo)
|
||||||
|
|
||||||
|
|
||||||
|
def clone_from(git_url, repo_dir, recursive=True):
|
||||||
|
pygit2.clone_repository(git_url, repo_dir, callbacks=GitProgress())
|
||||||
|
Repo(repo_dir).update_recursive()
|
||||||
@@ -23,13 +23,35 @@ sys.path.append(glob_path)
|
|||||||
import cm_global
|
import cm_global
|
||||||
from manager_util import *
|
from manager_util import *
|
||||||
|
|
||||||
version = [2, 51, 1]
|
version = [2, 53]
|
||||||
version_str = f"V{version[0]}.{version[1]}" + (f'.{version[2]}' if len(version) > 2 else '')
|
version_str = f"V{version[0]}.{version[1]}" + (f'.{version[2]}' if len(version) > 2 else '')
|
||||||
|
|
||||||
|
|
||||||
comfyui_manager_path = os.path.abspath(os.path.join(os.path.dirname(__file__), '..'))
|
comfyui_manager_path = os.path.abspath(os.path.join(os.path.dirname(__file__), '..'))
|
||||||
custom_nodes_path = os.path.abspath(os.path.join(comfyui_manager_path, '..'))
|
custom_nodes_path = os.path.abspath(os.path.join(comfyui_manager_path, '..'))
|
||||||
|
|
||||||
|
default_custom_nodes_path = None
|
||||||
|
|
||||||
|
def get_default_custom_nodes_path():
|
||||||
|
global default_custom_nodes_path
|
||||||
|
if default_custom_nodes_path is None:
|
||||||
|
try:
|
||||||
|
import folder_paths
|
||||||
|
default_custom_nodes_path = folder_paths.get_folder_paths("custom_nodes")[0]
|
||||||
|
except:
|
||||||
|
default_custom_nodes_path = custom_nodes_path
|
||||||
|
|
||||||
|
return default_custom_nodes_path
|
||||||
|
|
||||||
|
|
||||||
|
def get_custom_nodes_paths():
|
||||||
|
try:
|
||||||
|
import folder_paths
|
||||||
|
return folder_paths.get_folder_paths("custom_nodes")
|
||||||
|
except:
|
||||||
|
return [custom_nodes_path]
|
||||||
|
|
||||||
|
|
||||||
comfy_path = os.environ.get('COMFYUI_PATH')
|
comfy_path = os.environ.get('COMFYUI_PATH')
|
||||||
if comfy_path is None:
|
if comfy_path is None:
|
||||||
comfy_path = os.path.abspath(os.path.join(custom_nodes_path, '..'))
|
comfy_path = os.path.abspath(os.path.join(custom_nodes_path, '..'))
|
||||||
@@ -340,7 +362,7 @@ def __win_check_git_update(path, do_fetch=False, do_update=False):
|
|||||||
|
|
||||||
new_env = os.environ.copy()
|
new_env = os.environ.copy()
|
||||||
new_env["COMFYUI_PATH"] = comfy_path
|
new_env["COMFYUI_PATH"] = comfy_path
|
||||||
process = subprocess.Popen(command, stdout=subprocess.PIPE, stderr=subprocess.PIPE, cwd=custom_nodes_path)
|
process = subprocess.Popen(command, stdout=subprocess.PIPE, stderr=subprocess.PIPE, cwd=get_default_custom_nodes_path())
|
||||||
output, _ = process.communicate()
|
output, _ = process.communicate()
|
||||||
output = output.decode('utf-8').strip()
|
output = output.decode('utf-8').strip()
|
||||||
|
|
||||||
@@ -367,13 +389,13 @@ def __win_check_git_update(path, do_fetch=False, do_update=False):
|
|||||||
if do_update:
|
if do_update:
|
||||||
if "CUSTOM NODE PULL: Success" in output:
|
if "CUSTOM NODE PULL: Success" in output:
|
||||||
process.wait()
|
process.wait()
|
||||||
print(f"\rUpdated: {path}")
|
print(f"\x1b[2K\rUpdated: {path}")
|
||||||
return True, True # updated
|
return True, True # updated
|
||||||
elif "CUSTOM NODE PULL: None" in output:
|
elif "CUSTOM NODE PULL: None" in output:
|
||||||
process.wait()
|
process.wait()
|
||||||
return False, True # there is no update
|
return False, True # there is no update
|
||||||
else:
|
else:
|
||||||
print(f"\rUpdate error: {path}")
|
print(f"\x1b[2K\rUpdate error: {path}")
|
||||||
process.wait()
|
process.wait()
|
||||||
return False, False # update failed
|
return False, False # update failed
|
||||||
else:
|
else:
|
||||||
@@ -384,7 +406,7 @@ def __win_check_git_update(path, do_fetch=False, do_update=False):
|
|||||||
process.wait()
|
process.wait()
|
||||||
return False, True
|
return False, True
|
||||||
else:
|
else:
|
||||||
print(f"\rFetch error: {path}")
|
print(f"\x1b[2K\rFetch error: {path}")
|
||||||
print(f"\n{output}\n")
|
print(f"\n{output}\n")
|
||||||
process.wait()
|
process.wait()
|
||||||
return False, True
|
return False, True
|
||||||
@@ -394,7 +416,7 @@ def __win_check_git_pull(path):
|
|||||||
new_env = os.environ.copy()
|
new_env = os.environ.copy()
|
||||||
new_env["COMFYUI_PATH"] = comfy_path
|
new_env["COMFYUI_PATH"] = comfy_path
|
||||||
command = [sys.executable, git_script_path, "--pull", path]
|
command = [sys.executable, git_script_path, "--pull", path]
|
||||||
process = subprocess.Popen(command, env=new_env, cwd=custom_nodes_path)
|
process = subprocess.Popen(command, env=new_env, cwd=get_default_custom_nodes_path())
|
||||||
process.wait()
|
process.wait()
|
||||||
|
|
||||||
|
|
||||||
@@ -408,6 +430,7 @@ def execute_install_script(url, repo_path, lazy_mode=False, instant_execution=Fa
|
|||||||
else:
|
else:
|
||||||
if os.path.exists(requirements_path):
|
if os.path.exists(requirements_path):
|
||||||
print("Install: pip packages")
|
print("Install: pip packages")
|
||||||
|
pip_fixer = PIPFixer(get_installed_packages())
|
||||||
with open(requirements_path, "r") as requirements_file:
|
with open(requirements_path, "r") as requirements_file:
|
||||||
for line in requirements_file:
|
for line in requirements_file:
|
||||||
#handle comments
|
#handle comments
|
||||||
@@ -430,6 +453,8 @@ def execute_install_script(url, repo_path, lazy_mode=False, instant_execution=Fa
|
|||||||
if package_name.strip() != "" and not package_name.startswith('#'):
|
if package_name.strip() != "" and not package_name.startswith('#'):
|
||||||
try_install_script(url, repo_path, install_cmd, instant_execution=instant_execution)
|
try_install_script(url, repo_path, install_cmd, instant_execution=instant_execution)
|
||||||
|
|
||||||
|
pip_fixer.fix_broken()
|
||||||
|
|
||||||
if os.path.exists(install_script_path):
|
if os.path.exists(install_script_path):
|
||||||
print(f"Install: install script")
|
print(f"Install: install script")
|
||||||
install_cmd = [sys.executable, "install.py"]
|
install_cmd = [sys.executable, "install.py"]
|
||||||
@@ -559,11 +584,11 @@ def gitclone_install(files, instant_execution=False, msg_prefix=''):
|
|||||||
try:
|
try:
|
||||||
print(f"Download: git clone '{url}'")
|
print(f"Download: git clone '{url}'")
|
||||||
repo_name = os.path.splitext(os.path.basename(url))[0]
|
repo_name = os.path.splitext(os.path.basename(url))[0]
|
||||||
repo_path = os.path.join(custom_nodes_path, repo_name)
|
repo_path = os.path.join(get_default_custom_nodes_path(), repo_name)
|
||||||
|
|
||||||
# Clone the repository from the remote URL
|
# Clone the repository from the remote URL
|
||||||
if not instant_execution and platform.system() == 'Windows':
|
if not instant_execution and platform.system() == 'Windows':
|
||||||
res = manager_funcs.run_script([sys.executable, git_script_path, "--clone", custom_nodes_path, url], cwd=custom_nodes_path)
|
res = manager_funcs.run_script([sys.executable, git_script_path, "--clone", get_default_custom_nodes_path(), url], cwd=get_default_custom_nodes_path())
|
||||||
if res != 0:
|
if res != 0:
|
||||||
return False
|
return False
|
||||||
else:
|
else:
|
||||||
@@ -687,6 +712,22 @@ async def get_data_by_mode(mode, filename, channel_url=None):
|
|||||||
return json_obj
|
return json_obj
|
||||||
|
|
||||||
|
|
||||||
|
def lookup_installed_custom_nodes(repo_name):
|
||||||
|
try:
|
||||||
|
import folder_paths
|
||||||
|
base_paths = folder_paths.get_folder_paths("custom_nodes")
|
||||||
|
except:
|
||||||
|
base_paths = [custom_nodes_path]
|
||||||
|
|
||||||
|
for base_path in base_paths:
|
||||||
|
repo_path = os.path.join(base_path, repo_name)
|
||||||
|
if os.path.exists(repo_path):
|
||||||
|
return True, repo_path
|
||||||
|
elif os.path.exists(repo_path+'.disabled'):
|
||||||
|
return False, repo_path
|
||||||
|
|
||||||
|
return None
|
||||||
|
|
||||||
def gitclone_fix(files, instant_execution=False):
|
def gitclone_fix(files, instant_execution=False):
|
||||||
print(f"Try fixing: {files}")
|
print(f"Try fixing: {files}")
|
||||||
for url in files:
|
for url in files:
|
||||||
@@ -698,13 +739,15 @@ def gitclone_fix(files, instant_execution=False):
|
|||||||
url = url[:-1]
|
url = url[:-1]
|
||||||
try:
|
try:
|
||||||
repo_name = os.path.splitext(os.path.basename(url))[0]
|
repo_name = os.path.splitext(os.path.basename(url))[0]
|
||||||
repo_path = os.path.join(custom_nodes_path, repo_name)
|
repo_path = lookup_installed_custom_nodes(repo_name)
|
||||||
|
|
||||||
if os.path.exists(repo_path+'.disabled'):
|
if repo_path is not None:
|
||||||
repo_path = repo_path+'.disabled'
|
repo_path = repo_path[1]
|
||||||
|
|
||||||
if not execute_install_script(url, repo_path, instant_execution=instant_execution):
|
if not execute_install_script(url, repo_path, instant_execution=instant_execution):
|
||||||
return False
|
return False
|
||||||
|
else:
|
||||||
|
print(f"Custom node not found: {repo_name}")
|
||||||
|
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
print(f"Install(git-clone) error: {url} / {e}", file=sys.stderr)
|
print(f"Install(git-clone) error: {url} / {e}", file=sys.stderr)
|
||||||
@@ -751,12 +794,12 @@ def gitclone_uninstall(files):
|
|||||||
url = url[:-1]
|
url = url[:-1]
|
||||||
try:
|
try:
|
||||||
dir_name = os.path.splitext(os.path.basename(url))[0].replace(".git", "")
|
dir_name = os.path.splitext(os.path.basename(url))[0].replace(".git", "")
|
||||||
dir_path = os.path.join(custom_nodes_path, dir_name)
|
repo_path = lookup_installed_custom_nodes(dir_name)
|
||||||
|
|
||||||
# safety check
|
if repo_path is None:
|
||||||
if dir_path == '/' or dir_path[1:] == ":/" or dir_path == '':
|
continue
|
||||||
print(f"Uninstall(git-clone) error: invalid path '{dir_path}' for '{url}'")
|
|
||||||
return False
|
dir_path = repo_path[1]
|
||||||
|
|
||||||
install_script_path = os.path.join(dir_path, "uninstall.py")
|
install_script_path = os.path.join(dir_path, "uninstall.py")
|
||||||
disable_script_path = os.path.join(dir_path, "disable.py")
|
disable_script_path = os.path.join(dir_path, "disable.py")
|
||||||
@@ -772,10 +815,7 @@ def gitclone_uninstall(files):
|
|||||||
if code != 0:
|
if code != 0:
|
||||||
print(f"An error occurred during the execution of the disable.py script. Only the '{dir_path}' will be deleted.")
|
print(f"An error occurred during the execution of the disable.py script. Only the '{dir_path}' will be deleted.")
|
||||||
|
|
||||||
if os.path.exists(dir_path):
|
rmtree(dir_path)
|
||||||
rmtree(dir_path)
|
|
||||||
elif os.path.exists(dir_path + ".disabled"):
|
|
||||||
rmtree(dir_path + ".disabled")
|
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
print(f"Uninstall(git-clone) error: {url} / {e}", file=sys.stderr)
|
print(f"Uninstall(git-clone) error: {url} / {e}", file=sys.stderr)
|
||||||
return False
|
return False
|
||||||
@@ -798,13 +838,13 @@ def gitclone_set_active(files, is_disable):
|
|||||||
url = url[:-1]
|
url = url[:-1]
|
||||||
try:
|
try:
|
||||||
dir_name = os.path.splitext(os.path.basename(url))[0].replace(".git", "")
|
dir_name = os.path.splitext(os.path.basename(url))[0].replace(".git", "")
|
||||||
dir_path = os.path.join(custom_nodes_path, dir_name)
|
repo_path = lookup_installed_custom_nodes(dir_name)
|
||||||
|
|
||||||
# safety check
|
if repo_path is None:
|
||||||
if dir_path == '/' or dir_path[1:] == ":/" or dir_path == '':
|
continue
|
||||||
print(f"{action_name}(git-clone) error: invalid path '{dir_path}' for '{url}'")
|
|
||||||
return False
|
|
||||||
|
|
||||||
|
dir_path = repo_path[1]
|
||||||
|
|
||||||
if is_disable:
|
if is_disable:
|
||||||
current_path = dir_path
|
current_path = dir_path
|
||||||
new_path = dir_path + ".disabled"
|
new_path = dir_path + ".disabled"
|
||||||
@@ -840,11 +880,13 @@ def gitclone_update(files, instant_execution=False, skip_script=False, msg_prefi
|
|||||||
url = url[:-1]
|
url = url[:-1]
|
||||||
try:
|
try:
|
||||||
repo_name = os.path.splitext(os.path.basename(url))[0].replace(".git", "")
|
repo_name = os.path.splitext(os.path.basename(url))[0].replace(".git", "")
|
||||||
repo_path = os.path.join(custom_nodes_path, repo_name)
|
repo_path = lookup_installed_custom_nodes(repo_name)
|
||||||
|
|
||||||
if os.path.exists(repo_path+'.disabled'):
|
if repo_path is None:
|
||||||
repo_path = repo_path+'.disabled'
|
continue
|
||||||
|
|
||||||
|
repo_path = repo_path[1]
|
||||||
|
|
||||||
git_pull(repo_path)
|
git_pull(repo_path)
|
||||||
|
|
||||||
if not skip_script:
|
if not skip_script:
|
||||||
@@ -914,10 +956,14 @@ def lookup_customnode_by_url(data, target):
|
|||||||
for x in data['custom_nodes']:
|
for x in data['custom_nodes']:
|
||||||
if target in x['files']:
|
if target in x['files']:
|
||||||
dir_name = os.path.splitext(os.path.basename(target))[0].replace(".git", "")
|
dir_name = os.path.splitext(os.path.basename(target))[0].replace(".git", "")
|
||||||
dir_path = os.path.join(custom_nodes_path, dir_name)
|
repo_path = lookup_installed_custom_nodes(dir_name)
|
||||||
if os.path.exists(dir_path):
|
|
||||||
|
if repo_path is None:
|
||||||
|
continue
|
||||||
|
|
||||||
|
if repo_path[0]:
|
||||||
x['installed'] = 'True'
|
x['installed'] = 'True'
|
||||||
elif os.path.exists(dir_path + ".disabled"):
|
else:
|
||||||
x['installed'] = 'Disabled'
|
x['installed'] = 'Disabled'
|
||||||
return x
|
return x
|
||||||
|
|
||||||
@@ -926,13 +972,15 @@ def lookup_customnode_by_url(data, target):
|
|||||||
|
|
||||||
def simple_check_custom_node(url):
|
def simple_check_custom_node(url):
|
||||||
dir_name = os.path.splitext(os.path.basename(url))[0].replace(".git", "")
|
dir_name = os.path.splitext(os.path.basename(url))[0].replace(".git", "")
|
||||||
dir_path = os.path.join(custom_nodes_path, dir_name)
|
repo_path = lookup_installed_custom_nodes(dir_name)
|
||||||
if os.path.exists(dir_path):
|
|
||||||
return 'installed'
|
|
||||||
elif os.path.exists(dir_path+'.disabled'):
|
|
||||||
return 'disabled'
|
|
||||||
|
|
||||||
return 'not-installed'
|
if repo_path is None:
|
||||||
|
return 'not-installed'
|
||||||
|
|
||||||
|
if repo_path[0]:
|
||||||
|
return 'installed'
|
||||||
|
else:
|
||||||
|
return 'disabled'
|
||||||
|
|
||||||
|
|
||||||
def check_a_custom_node_installed(item, do_fetch=False, do_update_check=True, do_update=False):
|
def check_a_custom_node_installed(item, do_fetch=False, do_update_check=True, do_update=False):
|
||||||
@@ -945,8 +993,12 @@ def check_a_custom_node_installed(item, do_fetch=False, do_update_check=True, do
|
|||||||
url = url[:-1]
|
url = url[:-1]
|
||||||
|
|
||||||
dir_name = os.path.splitext(os.path.basename(url))[0].replace(".git", "")
|
dir_name = os.path.splitext(os.path.basename(url))[0].replace(".git", "")
|
||||||
dir_path = os.path.join(custom_nodes_path, dir_name)
|
repo_path = lookup_installed_custom_nodes(dir_name)
|
||||||
if os.path.exists(dir_path):
|
|
||||||
|
if repo_path is None:
|
||||||
|
item['installed'] = 'False'
|
||||||
|
elif repo_path[0]:
|
||||||
|
dir_path = repo_path[1]
|
||||||
try:
|
try:
|
||||||
item['installed'] = 'True' # default
|
item['installed'] = 'True' # default
|
||||||
|
|
||||||
@@ -965,17 +1017,23 @@ def check_a_custom_node_installed(item, do_fetch=False, do_update_check=True, do
|
|||||||
else:
|
else:
|
||||||
item['installed'] = 'True'
|
item['installed'] = 'True'
|
||||||
|
|
||||||
elif os.path.exists(dir_path + ".disabled"):
|
|
||||||
item['installed'] = 'Disabled'
|
|
||||||
|
|
||||||
else:
|
else:
|
||||||
item['installed'] = 'False'
|
item['installed'] = 'Disabled'
|
||||||
|
|
||||||
elif item['install_type'] == 'copy' and len(item['files']) == 1:
|
elif item['install_type'] == 'copy' and len(item['files']) == 1:
|
||||||
dir_name = os.path.basename(item['files'][0])
|
dir_name = os.path.basename(item['files'][0])
|
||||||
|
|
||||||
if item['files'][0].endswith('.py'):
|
if item['files'][0].endswith('.py'):
|
||||||
base_path = custom_nodes_path
|
base_path = lookup_installed_custom_nodes(item['files'][0])
|
||||||
|
if base_path is None:
|
||||||
|
item['installed'] = 'False'
|
||||||
|
return
|
||||||
|
elif base_path[0]:
|
||||||
|
item['installed'] = 'True'
|
||||||
|
else:
|
||||||
|
item['installed'] = 'Disabled'
|
||||||
|
|
||||||
|
return
|
||||||
elif 'js_path' in item:
|
elif 'js_path' in item:
|
||||||
base_path = os.path.join(js_path, item['js_path'])
|
base_path = os.path.join(js_path, item['js_path'])
|
||||||
else:
|
else:
|
||||||
@@ -987,8 +1045,6 @@ def check_a_custom_node_installed(item, do_fetch=False, do_update_check=True, do
|
|||||||
item['installed'] = 'Fail'
|
item['installed'] = 'Fail'
|
||||||
else:
|
else:
|
||||||
item['installed'] = 'True'
|
item['installed'] = 'True'
|
||||||
elif os.path.exists(file_path + ".disabled"):
|
|
||||||
item['installed'] = 'Disabled'
|
|
||||||
else:
|
else:
|
||||||
item['installed'] = 'False'
|
item['installed'] = 'False'
|
||||||
|
|
||||||
@@ -1025,39 +1081,46 @@ def get_current_snapshot():
|
|||||||
git_custom_nodes = {}
|
git_custom_nodes = {}
|
||||||
file_custom_nodes = []
|
file_custom_nodes = []
|
||||||
|
|
||||||
|
try:
|
||||||
|
import folder_paths
|
||||||
|
base_paths = folder_paths.get_folder_paths("custom_nodes")
|
||||||
|
except:
|
||||||
|
base_paths = [custom_nodes_path]
|
||||||
|
|
||||||
# Get custom nodes hash
|
# Get custom nodes hash
|
||||||
for path in os.listdir(custom_nodes_path):
|
for base_path in base_paths:
|
||||||
fullpath = os.path.join(custom_nodes_path, path)
|
for path in os.listdir(base_path):
|
||||||
|
fullpath = os.path.join(base_path, path)
|
||||||
|
|
||||||
if os.path.isdir(fullpath):
|
if os.path.isdir(fullpath):
|
||||||
is_disabled = path.endswith(".disabled")
|
is_disabled = path.endswith(".disabled")
|
||||||
|
|
||||||
try:
|
try:
|
||||||
git_dir = os.path.join(fullpath, '.git')
|
git_dir = os.path.join(fullpath, '.git')
|
||||||
|
|
||||||
if not os.path.exists(git_dir):
|
if not os.path.exists(git_dir):
|
||||||
continue
|
continue
|
||||||
|
|
||||||
repo = git.Repo(fullpath)
|
repo = git.Repo(fullpath)
|
||||||
commit_hash = repo.head.commit.hexsha
|
commit_hash = repo.head.commit.hexsha
|
||||||
url = repo.remotes.origin.url
|
url = repo.remotes.origin.url
|
||||||
git_custom_nodes[url] = {
|
git_custom_nodes[url] = {
|
||||||
'hash': commit_hash,
|
'hash': commit_hash,
|
||||||
|
'disabled': is_disabled
|
||||||
|
}
|
||||||
|
|
||||||
|
except:
|
||||||
|
print(f"Failed to extract snapshots for the custom node '{path}'.")
|
||||||
|
|
||||||
|
elif path.endswith('.py'):
|
||||||
|
is_disabled = path.endswith(".py.disabled")
|
||||||
|
filename = os.path.basename(path)
|
||||||
|
item = {
|
||||||
|
'filename': filename,
|
||||||
'disabled': is_disabled
|
'disabled': is_disabled
|
||||||
}
|
}
|
||||||
|
|
||||||
except:
|
file_custom_nodes.append(item)
|
||||||
print(f"Failed to extract snapshots for the custom node '{path}'.")
|
|
||||||
|
|
||||||
elif path.endswith('.py'):
|
|
||||||
is_disabled = path.endswith(".py.disabled")
|
|
||||||
filename = os.path.basename(path)
|
|
||||||
item = {
|
|
||||||
'filename': filename,
|
|
||||||
'disabled': is_disabled
|
|
||||||
}
|
|
||||||
|
|
||||||
file_custom_nodes.append(item)
|
|
||||||
|
|
||||||
pip_packages = get_installed_pip_packages()
|
pip_packages = get_installed_pip_packages()
|
||||||
|
|
||||||
@@ -1135,7 +1198,7 @@ async def extract_nodes_from_workflow(filepath, mode='local', channel_url='defau
|
|||||||
if node_name in ['Reroute', 'Note']:
|
if node_name in ['Reroute', 'Note']:
|
||||||
continue
|
continue
|
||||||
|
|
||||||
if node_name is not None and not node_name.startswith('workflow/'):
|
if node_name is not None and not (node_name.startswith('workflow/') or node_name.startswith('workflow>')):
|
||||||
used_nodes.add(node_name)
|
used_nodes.add(node_name)
|
||||||
|
|
||||||
if 'nodes' in workflow:
|
if 'nodes' in workflow:
|
||||||
|
|||||||
@@ -62,7 +62,7 @@ def is_allowed_security_level(level):
|
|||||||
|
|
||||||
async def get_risky_level(files, pip_packages):
|
async def get_risky_level(files, pip_packages):
|
||||||
json_data1 = await core.get_data_by_mode('local', 'custom-node-list.json')
|
json_data1 = await core.get_data_by_mode('local', 'custom-node-list.json')
|
||||||
json_data2 = await core.get_data_by_mode('cache', 'custom-node-list.json', channel_url='https://github.com/ltdrdata/ComfyUI-Manager/raw/main')
|
json_data2 = await core.get_data_by_mode('cache', 'custom-node-list.json', channel_url='https://raw.githubusercontent.com/ltdrdata/ComfyUI-Manager/main')
|
||||||
|
|
||||||
all_urls = set()
|
all_urls = set()
|
||||||
for x in json_data1['custom_nodes'] + json_data2['custom_nodes']:
|
for x in json_data1['custom_nodes'] + json_data2['custom_nodes']:
|
||||||
@@ -820,7 +820,14 @@ async def install_custom_node(request):
|
|||||||
res = unzip_install(json_data['files'])
|
res = unzip_install(json_data['files'])
|
||||||
|
|
||||||
if install_type == "copy":
|
if install_type == "copy":
|
||||||
js_path_name = json_data['js_path'] if 'js_path' in json_data else '.'
|
if 'js_path' in json_data:
|
||||||
|
if '.' in json_data['js_path'] or ':' in json_data['js_path'] or json_data['js_path'].startswith('/'):
|
||||||
|
print(f"[ComfyUI Manager] An abnormal JS path has been transmitted. This could be the result of a security attack.\n{json_data['js_path']}")
|
||||||
|
return web.Response(status=400)
|
||||||
|
else:
|
||||||
|
js_path_name = json_data['js_path']
|
||||||
|
else:
|
||||||
|
js_path_name = '.'
|
||||||
res = copy_install(json_data['files'], js_path_name)
|
res = copy_install(json_data['files'], js_path_name)
|
||||||
|
|
||||||
elif install_type == "git-clone":
|
elif install_type == "git-clone":
|
||||||
@@ -844,7 +851,7 @@ async def install_custom_node(request):
|
|||||||
@PromptServer.instance.routes.post("/customnode/fix")
|
@PromptServer.instance.routes.post("/customnode/fix")
|
||||||
async def fix_custom_node(request):
|
async def fix_custom_node(request):
|
||||||
if not is_allowed_security_level('middle'):
|
if not is_allowed_security_level('middle'):
|
||||||
print(SECURITY_MESSAGE_MIDDLE_OR_BELOW)
|
print(SECURITY_MESSAGE_GENERAL)
|
||||||
return web.Response(status=403)
|
return web.Response(status=403)
|
||||||
|
|
||||||
json_data = await request.json()
|
json_data = await request.json()
|
||||||
@@ -864,6 +871,10 @@ async def fix_custom_node(request):
|
|||||||
return web.Response(status=400)
|
return web.Response(status=400)
|
||||||
|
|
||||||
if 'pip' in json_data:
|
if 'pip' in json_data:
|
||||||
|
if not is_allowed_security_level('high'):
|
||||||
|
print(SECURITY_MESSAGE_GENERAL)
|
||||||
|
return web.Response(status=403)
|
||||||
|
|
||||||
for pname in json_data['pip']:
|
for pname in json_data['pip']:
|
||||||
install_cmd = [sys.executable, "-m", "pip", "install", '-U', pname]
|
install_cmd = [sys.executable, "-m", "pip", "install", '-U', pname]
|
||||||
core.try_install_script(json_data['files'][0], ".", install_cmd)
|
core.try_install_script(json_data['files'][0], ".", install_cmd)
|
||||||
@@ -1059,32 +1070,6 @@ async def install_model(request):
|
|||||||
return web.Response(status=400)
|
return web.Response(status=400)
|
||||||
|
|
||||||
|
|
||||||
class ManagerTerminalHook:
|
|
||||||
def write_stderr(self, msg):
|
|
||||||
PromptServer.instance.send_sync("manager-terminal-feedback", {"data": msg})
|
|
||||||
|
|
||||||
def write_stdout(self, msg):
|
|
||||||
PromptServer.instance.send_sync("manager-terminal-feedback", {"data": msg})
|
|
||||||
|
|
||||||
|
|
||||||
manager_terminal_hook = ManagerTerminalHook()
|
|
||||||
|
|
||||||
|
|
||||||
@PromptServer.instance.routes.get("/manager/terminal")
|
|
||||||
async def terminal_mode(request):
|
|
||||||
if not is_allowed_security_level('high'):
|
|
||||||
print(SECURITY_MESSAGE_NORMAL_MINUS)
|
|
||||||
return web.Response(status=403)
|
|
||||||
|
|
||||||
if "mode" in request.rel_url.query:
|
|
||||||
if request.rel_url.query['mode'] == 'true':
|
|
||||||
sys.__comfyui_manager_terminal_hook.add_hook('cm', manager_terminal_hook)
|
|
||||||
else:
|
|
||||||
sys.__comfyui_manager_terminal_hook.remove_hook('cm')
|
|
||||||
|
|
||||||
return web.Response(status=200)
|
|
||||||
|
|
||||||
|
|
||||||
@PromptServer.instance.routes.get("/manager/preview_method")
|
@PromptServer.instance.routes.get("/manager/preview_method")
|
||||||
async def preview_method(request):
|
async def preview_method(request):
|
||||||
if "value" in request.rel_url.query:
|
if "value" in request.rel_url.query:
|
||||||
@@ -1231,6 +1216,11 @@ def restart(self):
|
|||||||
exit(0)
|
exit(0)
|
||||||
|
|
||||||
print(f"\nRestarting... [Legacy Mode]\n\n")
|
print(f"\nRestarting... [Legacy Mode]\n\n")
|
||||||
|
|
||||||
|
sys_argv = sys.argv.copy()
|
||||||
|
if '--windows-standalone-build' in sys_argv:
|
||||||
|
sys_argv.remove('--windows-standalone-build')
|
||||||
|
|
||||||
if sys.platform.startswith('win32'):
|
if sys.platform.startswith('win32'):
|
||||||
return os.execv(sys.executable, ['"' + sys.executable + '"', '"' + sys.argv[0] + '"'] + sys.argv[1:])
|
return os.execv(sys.executable, ['"' + sys.executable + '"', '"' + sys.argv[0] + '"'] + sys.argv[1:])
|
||||||
else:
|
else:
|
||||||
@@ -1361,6 +1351,6 @@ if not os.path.exists(core.config_path):
|
|||||||
cm_global.register_extension('ComfyUI-Manager',
|
cm_global.register_extension('ComfyUI-Manager',
|
||||||
{'version': core.version,
|
{'version': core.version,
|
||||||
'name': 'ComfyUI Manager',
|
'name': 'ComfyUI Manager',
|
||||||
'nodes': {'Terminal Log //CM'},
|
'nodes': {},
|
||||||
'description': 'It provides the ability to manage custom nodes in ComfyUI.', })
|
'description': 'It provides the ability to manage custom nodes in ComfyUI.', })
|
||||||
|
|
||||||
|
|||||||
@@ -1,3 +1,6 @@
|
|||||||
|
import subprocess
|
||||||
|
import sys
|
||||||
|
|
||||||
# DON'T USE StrictVersion - cannot handle pre_release version
|
# DON'T USE StrictVersion - cannot handle pre_release version
|
||||||
# try:
|
# try:
|
||||||
# from distutils.version import StrictVersion
|
# from distutils.version import StrictVersion
|
||||||
@@ -62,3 +65,150 @@ class StrictVersion:
|
|||||||
def __ne__(self, other):
|
def __ne__(self, other):
|
||||||
return not self == other
|
return not self == other
|
||||||
|
|
||||||
|
|
||||||
|
pip_map = None
|
||||||
|
|
||||||
|
def get_installed_packages(renew=False):
|
||||||
|
global pip_map
|
||||||
|
|
||||||
|
if renew or pip_map is None:
|
||||||
|
try:
|
||||||
|
result = subprocess.check_output([sys.executable, '-m', 'pip', 'list'], universal_newlines=True)
|
||||||
|
|
||||||
|
pip_map = {}
|
||||||
|
for line in result.split('\n'):
|
||||||
|
x = line.strip()
|
||||||
|
if x:
|
||||||
|
y = line.split()
|
||||||
|
if y[0] == 'Package' or y[0].startswith('-'):
|
||||||
|
continue
|
||||||
|
|
||||||
|
pip_map[y[0]] = y[1]
|
||||||
|
except subprocess.CalledProcessError as e:
|
||||||
|
print(f"[ComfyUI-Manager] Failed to retrieve the information of installed pip packages.")
|
||||||
|
return set()
|
||||||
|
|
||||||
|
return pip_map
|
||||||
|
|
||||||
|
|
||||||
|
def clear_pip_cache():
|
||||||
|
global pip_map
|
||||||
|
pip_map = None
|
||||||
|
|
||||||
|
|
||||||
|
torch_torchvision_version_map = {
|
||||||
|
'2.5.1': '0.20.1',
|
||||||
|
'2.5.0': '0.20.0',
|
||||||
|
'2.4.1': '0.19.1',
|
||||||
|
'2.4.0': '0.19.0',
|
||||||
|
'2.3.1': '0.18.1',
|
||||||
|
'2.3.0': '0.18.0',
|
||||||
|
'2.2.2': '0.17.2',
|
||||||
|
'2.2.1': '0.17.1',
|
||||||
|
'2.2.0': '0.17.0',
|
||||||
|
'2.1.2': '0.16.2',
|
||||||
|
'2.1.1': '0.16.1',
|
||||||
|
'2.1.0': '0.16.0',
|
||||||
|
'2.0.1': '0.15.2',
|
||||||
|
'2.0.0': '0.15.1',
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
class PIPFixer:
|
||||||
|
def __init__(self, prev_pip_versions):
|
||||||
|
self.prev_pip_versions = { **prev_pip_versions }
|
||||||
|
|
||||||
|
def torch_rollback(self):
|
||||||
|
spec = self.prev_pip_versions['torch'].split('+')
|
||||||
|
if len(spec) > 0:
|
||||||
|
platform = spec[1]
|
||||||
|
else:
|
||||||
|
cmd = [sys.executable, '-m', 'pip', 'install', '--force', 'torch', 'torchvision', 'torchaudio']
|
||||||
|
subprocess.check_output(cmd, universal_newlines=True)
|
||||||
|
print(cmd)
|
||||||
|
return
|
||||||
|
|
||||||
|
torch_ver = StrictVersion(spec[0])
|
||||||
|
torch_ver = f"{torch_ver.major}.{torch_ver.minor}.{torch_ver.patch}"
|
||||||
|
torchvision_ver = torch_torchvision_version_map.get(torch_ver)
|
||||||
|
|
||||||
|
if torchvision_ver is None:
|
||||||
|
cmd = [sys.executable, '-m', 'pip', 'install', '--pre',
|
||||||
|
'torch', 'torchvision', 'torchaudio',
|
||||||
|
'--index-url', f"https://download.pytorch.org/whl/nightly/{platform}"]
|
||||||
|
print("[manager-core] restore PyTorch to nightly version")
|
||||||
|
else:
|
||||||
|
cmd = [sys.executable, '-m', 'pip', 'install',
|
||||||
|
f'torch=={torch_ver}', f'torchvision=={torchvision_ver}', f"torchaudio=={torch_ver}",
|
||||||
|
'--index-url', f"https://download.pytorch.org/whl/{platform}"]
|
||||||
|
print(f"[manager-core] restore PyTorch to {torch_ver}+{platform}")
|
||||||
|
|
||||||
|
subprocess.check_output(cmd, universal_newlines=True)
|
||||||
|
|
||||||
|
def fix_broken(self):
|
||||||
|
new_pip_versions = get_installed_packages(True)
|
||||||
|
|
||||||
|
# remove `comfy` python package
|
||||||
|
try:
|
||||||
|
if 'comfy' in new_pip_versions:
|
||||||
|
cmd = [sys.executable, '-m', 'pip', 'uninstall', 'comfy']
|
||||||
|
subprocess.check_output(cmd, universal_newlines=True)
|
||||||
|
|
||||||
|
print(f"[manager-core] 'comfy' python package is uninstalled.\nWARN: The 'comfy' package is completely unrelated to ComfyUI and should never be installed as it causes conflicts with ComfyUI.")
|
||||||
|
except Exception as e:
|
||||||
|
print(f"[manager-core] Failed to uninstall `comfy` python package")
|
||||||
|
print(e)
|
||||||
|
|
||||||
|
# fix torch - reinstall torch packages if version is changed
|
||||||
|
try:
|
||||||
|
if self.prev_pip_versions['torch'] != new_pip_versions['torch'] \
|
||||||
|
or self.prev_pip_versions['torchvision'] != new_pip_versions['torchvision'] \
|
||||||
|
or self.prev_pip_versions['torchaudio'] != new_pip_versions['torchaudio']:
|
||||||
|
self.torch_rollback()
|
||||||
|
except Exception as e:
|
||||||
|
print(f"[manager-core] Failed to restore PyTorch")
|
||||||
|
print(e)
|
||||||
|
|
||||||
|
# fix opencv
|
||||||
|
try:
|
||||||
|
ocp = new_pip_versions.get('opencv-contrib-python')
|
||||||
|
ocph = new_pip_versions.get('opencv-contrib-python-headless')
|
||||||
|
op = new_pip_versions.get('opencv-python')
|
||||||
|
oph = new_pip_versions.get('opencv-python-headless')
|
||||||
|
|
||||||
|
versions = [ocp, ocph, op, oph]
|
||||||
|
versions = [StrictVersion(x) for x in versions if x is not None]
|
||||||
|
versions.sort(reverse=True)
|
||||||
|
|
||||||
|
if len(versions) > 0:
|
||||||
|
# upgrade to maximum version
|
||||||
|
targets = []
|
||||||
|
cur = versions[0]
|
||||||
|
if ocp is not None and StrictVersion(ocp) != cur:
|
||||||
|
targets.append('opencv-contrib-python')
|
||||||
|
if ocph is not None and StrictVersion(ocph) != cur:
|
||||||
|
targets.append('opencv-contrib-python-headless')
|
||||||
|
if op is not None and StrictVersion(op) != cur:
|
||||||
|
targets.append('opencv-python')
|
||||||
|
if oph is not None and StrictVersion(oph) != cur:
|
||||||
|
targets.append('opencv-python-headless')
|
||||||
|
|
||||||
|
if len(targets) > 0:
|
||||||
|
for x in targets:
|
||||||
|
cmd = [sys.executable, '-m', 'pip', 'install', f"{x}=={versions[0].version_string}"]
|
||||||
|
subprocess.check_output(cmd, universal_newlines=True)
|
||||||
|
|
||||||
|
print(f"[manager-core] 'opencv' dependencies were fixed: {targets}")
|
||||||
|
except Exception as e:
|
||||||
|
print(f"[manager-core] Failed to restore opencv")
|
||||||
|
print(e)
|
||||||
|
|
||||||
|
# fix numpy
|
||||||
|
try:
|
||||||
|
np = new_pip_versions.get('numpy')
|
||||||
|
if np is not None:
|
||||||
|
if StrictVersion(np) >= StrictVersion('2'):
|
||||||
|
subprocess.check_output([sys.executable, '-m', 'pip', 'install', f"numpy<2"], universal_newlines=True)
|
||||||
|
except Exception as e:
|
||||||
|
print(f"[manager-core] Failed to restore numpy")
|
||||||
|
print(e)
|
||||||
|
|||||||
@@ -982,6 +982,7 @@ class ManagerMenuDialog extends ComfyDialog {
|
|||||||
dbl_click_policy_combo.className = "cm-menu-combo";
|
dbl_click_policy_combo.className = "cm-menu-combo";
|
||||||
dbl_click_policy_combo.appendChild($el('option', { value: 'none', text: 'Double-Click: None' }, []));
|
dbl_click_policy_combo.appendChild($el('option', { value: 'none', text: 'Double-Click: None' }, []));
|
||||||
dbl_click_policy_combo.appendChild($el('option', { value: 'copy-all', text: 'Double-Click: Copy All Connections' }, []));
|
dbl_click_policy_combo.appendChild($el('option', { value: 'copy-all', text: 'Double-Click: Copy All Connections' }, []));
|
||||||
|
dbl_click_policy_combo.appendChild($el('option', { value: 'copy-full', text: 'Double-Click: Copy All Connections and shape' }, []));
|
||||||
dbl_click_policy_combo.appendChild($el('option', { value: 'copy-input', text: 'Double-Click: Copy Input Connections' }, []));
|
dbl_click_policy_combo.appendChild($el('option', { value: 'copy-input', text: 'Double-Click: Copy Input Connections' }, []));
|
||||||
dbl_click_policy_combo.appendChild($el('option', { value: 'possible-input', text: 'Double-Click: Possible Input Connections' }, []));
|
dbl_click_policy_combo.appendChild($el('option', { value: 'possible-input', text: 'Double-Click: Possible Input Connections' }, []));
|
||||||
dbl_click_policy_combo.appendChild($el('option', { value: 'dual', text: 'Double-Click: Possible(left) + Copy(right)' }, []));
|
dbl_click_policy_combo.appendChild($el('option', { value: 'dual', text: 'Double-Click: Possible(left) + Copy(right)' }, []));
|
||||||
@@ -1081,6 +1082,10 @@ class ManagerMenuDialog extends ComfyDialog {
|
|||||||
LiteGraph.closeAllContextMenus();
|
LiteGraph.closeAllContextMenus();
|
||||||
const menu = new LiteGraph.ContextMenu(
|
const menu = new LiteGraph.ContextMenu(
|
||||||
[
|
[
|
||||||
|
{
|
||||||
|
title: "ComfyUI Docs",
|
||||||
|
callback: () => { window.open("https://docs.comfy.org/", "comfyui-official-manual"); },
|
||||||
|
},
|
||||||
{
|
{
|
||||||
title: "Comfy Custom Node How To",
|
title: "Comfy Custom Node How To",
|
||||||
callback: () => { window.open("https://github.com/chrisgoringe/Comfy-Custom-Node-How-To/wiki/aaa_index", "comfyui-community-manual1"); },
|
callback: () => { window.open("https://github.com/chrisgoringe/Comfy-Custom-Node-How-To/wiki/aaa_index", "comfyui-community-manual1"); },
|
||||||
@@ -1271,15 +1276,6 @@ class ManagerMenuDialog extends ComfyDialog {
|
|||||||
modifyButtonStyle(url);
|
modifyButtonStyle(url);
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
|
||||||
title: "Open 'flowt.ai'",
|
|
||||||
callback: () => {
|
|
||||||
const url = "https://flowt.ai/";
|
|
||||||
localStorage.setItem("wg_last_visited", url);
|
|
||||||
window.open(url, url);
|
|
||||||
modifyButtonStyle(url);
|
|
||||||
},
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
title: "Open 'esheep'",
|
title: "Open 'esheep'",
|
||||||
callback: () => {
|
callback: () => {
|
||||||
@@ -1479,7 +1475,7 @@ app.registerExtension({
|
|||||||
node.prototype.getExtraMenuOptions = function (_, options) {
|
node.prototype.getExtraMenuOptions = function (_, options) {
|
||||||
origGetExtraMenuOptions?.apply?.(this, arguments);
|
origGetExtraMenuOptions?.apply?.(this, arguments);
|
||||||
|
|
||||||
if (node.category.startsWith('group nodes/')) {
|
if (node.category.startsWith('group nodes>')) {
|
||||||
options.push({
|
options.push({
|
||||||
content: "Save As Component",
|
content: "Save As Component",
|
||||||
callback: (obj) => {
|
callback: (obj) => {
|
||||||
|
|||||||
@@ -12,6 +12,10 @@ export async function sleep(ms) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export function rebootAPI() {
|
export function rebootAPI() {
|
||||||
|
if ('electronAPI' in window) {
|
||||||
|
window.electronAPI.restartApp();
|
||||||
|
return true;
|
||||||
|
}
|
||||||
if (confirm("Are you sure you'd like to reboot the server?")) {
|
if (confirm("Are you sure you'd like to reboot the server?")) {
|
||||||
try {
|
try {
|
||||||
api.fetchApi("/manager/reboot");
|
api.fetchApi("/manager/reboot");
|
||||||
@@ -237,4 +241,4 @@ export const icons = {
|
|||||||
conflicts: '<svg viewBox="0 0 400 400" width="100%" height="100%" pointer-events="none" xmlns="http://www.w3.org/2000/svg"><path fill="currentColor" d="m397.2 350.4.2-.2-180-320-.2.2C213.8 24.2 207.4 20 200 20s-13.8 4.2-17.2 10.4l-.2-.2-180 320 .2.2c-1.6 2.8-2.8 6-2.8 9.6 0 11 9 20 20 20h360c11 0 20-9 20-20 0-3.6-1.2-6.8-2.8-9.6M220 340h-40v-40h40zm0-60h-40V120h40z"/></svg>',
|
conflicts: '<svg viewBox="0 0 400 400" width="100%" height="100%" pointer-events="none" xmlns="http://www.w3.org/2000/svg"><path fill="currentColor" d="m397.2 350.4.2-.2-180-320-.2.2C213.8 24.2 207.4 20 200 20s-13.8 4.2-17.2 10.4l-.2-.2-180 320 .2.2c-1.6 2.8-2.8 6-2.8 9.6 0 11 9 20 20 20h360c11 0 20-9 20-20 0-3.6-1.2-6.8-2.8-9.6M220 340h-40v-40h40zm0-60h-40V120h40z"/></svg>',
|
||||||
passed: '<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 426.667 426.667"><path fill="#6AC259" d="M213.333,0C95.518,0,0,95.514,0,213.333s95.518,213.333,213.333,213.333c117.828,0,213.333-95.514,213.333-213.333S331.157,0,213.333,0z M174.199,322.918l-93.935-93.931l31.309-31.309l62.626,62.622l140.894-140.898l31.309,31.309L174.199,322.918z"/></svg>',
|
passed: '<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 426.667 426.667"><path fill="#6AC259" d="M213.333,0C95.518,0,0,95.514,0,213.333s95.518,213.333,213.333,213.333c117.828,0,213.333-95.514,213.333-213.333S331.157,0,213.333,0z M174.199,322.918l-93.935-93.931l31.309-31.309l62.626,62.622l140.894-140.898l31.309,31.309L174.199,322.918z"/></svg>',
|
||||||
download: '<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" aria-hidden="true" width="100%" height="100%" viewBox="0 0 32 32"><path fill="currentColor" d="M26 24v4H6v-4H4v4a2 2 0 0 0 2 2h20a2 2 0 0 0 2-2v-4zm0-10l-1.41-1.41L17 20.17V2h-2v18.17l-7.59-7.58L6 14l10 10l10-10z"></path></svg>'
|
download: '<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" aria-hidden="true" width="100%" height="100%" viewBox="0 0 32 32"><path fill="currentColor" d="M26 24v4H6v-4H4v4a2 2 0 0 0 2 2h20a2 2 0 0 0 2-2v-4zm0-10l-1.41-1.41L17 20.17V2h-2v18.17l-7.59-7.58L6 14l10 10l10-10z"></path></svg>'
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,6 +4,8 @@ import { sleep, show_message } from "./common.js";
|
|||||||
import { GroupNodeConfig, GroupNodeHandler } from "../../extensions/core/groupNode.js";
|
import { GroupNodeConfig, GroupNodeHandler } from "../../extensions/core/groupNode.js";
|
||||||
import { ComfyDialog, $el } from "../../scripts/ui.js";
|
import { ComfyDialog, $el } from "../../scripts/ui.js";
|
||||||
|
|
||||||
|
const SEPARATOR = ">"
|
||||||
|
|
||||||
let pack_map = {};
|
let pack_map = {};
|
||||||
let rpack_map = {};
|
let rpack_map = {};
|
||||||
|
|
||||||
@@ -20,7 +22,7 @@ export function getPureName(node) {
|
|||||||
let purename = node.comfyClass.substring(category.length+1);
|
let purename = node.comfyClass.substring(category.length+1);
|
||||||
return purename;
|
return purename;
|
||||||
}
|
}
|
||||||
else if(node.comfyClass.startsWith('workflow/')) {
|
else if(node.comfyClass.startsWith('workflow/') || node.comfyClass.startsWith(`workflow${SEPARATOR}`)) {
|
||||||
return node.comfyClass.substring(9);
|
return node.comfyClass.substring(9);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
@@ -76,7 +78,7 @@ export async function load_components() {
|
|||||||
|
|
||||||
let category = data.packname;
|
let category = data.packname;
|
||||||
if(data.category) {
|
if(data.category) {
|
||||||
category += "/" + data.category;
|
category += SEPARATOR + data.category;
|
||||||
}
|
}
|
||||||
if(category == '') {
|
if(category == '') {
|
||||||
category = 'components';
|
category = 'components';
|
||||||
@@ -100,7 +102,7 @@ export async function load_components() {
|
|||||||
try {
|
try {
|
||||||
let category = nodeData.packname;
|
let category = nodeData.packname;
|
||||||
if(nodeData.category) {
|
if(nodeData.category) {
|
||||||
category += "/" + nodeData.category;
|
category += SEPARATOR + nodeData.category;
|
||||||
}
|
}
|
||||||
if(category == '') {
|
if(category == '') {
|
||||||
category = 'components';
|
category = 'components';
|
||||||
@@ -139,7 +141,7 @@ export async function load_components() {
|
|||||||
try {
|
try {
|
||||||
let category = nodeData.packname;
|
let category = nodeData.packname;
|
||||||
if(nodeData.workflow.category) {
|
if(nodeData.workflow.category) {
|
||||||
category += "/" + nodeData.category;
|
category += SEPARATOR + nodeData.category;
|
||||||
}
|
}
|
||||||
if(category == '') {
|
if(category == '') {
|
||||||
category = 'components';
|
category = 'components';
|
||||||
@@ -174,7 +176,7 @@ export async function load_components() {
|
|||||||
try {
|
try {
|
||||||
let category = nodeData.workflow.packname;
|
let category = nodeData.workflow.packname;
|
||||||
if(nodeData.workflow.category) {
|
if(nodeData.workflow.category) {
|
||||||
category += "/" + nodeData.category;
|
category += SEPARATOR + nodeData.category;
|
||||||
}
|
}
|
||||||
if(category == '') {
|
if(category == '') {
|
||||||
category = 'components';
|
category = 'components';
|
||||||
@@ -234,7 +236,7 @@ async function save_as_component(node, version, author, prefix, nodename, packna
|
|||||||
|
|
||||||
let category = body.workflow.packname;
|
let category = body.workflow.packname;
|
||||||
if(body.workflow.category) {
|
if(body.workflow.category) {
|
||||||
category += "/" + body.workflow.category;
|
category += SEPARATOR + body.workflow.category;
|
||||||
}
|
}
|
||||||
if(category == '') {
|
if(category == '') {
|
||||||
category = 'components';
|
category = 'components';
|
||||||
@@ -266,7 +268,7 @@ async function import_component(component_name, component, mode) {
|
|||||||
|
|
||||||
let category = component.packname;
|
let category = component.packname;
|
||||||
if(component.category) {
|
if(component.category) {
|
||||||
category += "/" + component.category;
|
category += SEPARATOR + component.category;
|
||||||
}
|
}
|
||||||
if(category == '') {
|
if(category == '') {
|
||||||
category = 'components';
|
category = 'components';
|
||||||
@@ -403,7 +405,7 @@ function handle_import_components(components) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if(cnt == 1 && last_name) {
|
if(cnt == 1 && last_name) {
|
||||||
const node = LiteGraph.createNode(`workflow/${last_name}`);
|
const node = LiteGraph.createNode(`workflow${SEPARATOR}${last_name}`);
|
||||||
node.pos = [app.canvas.graph_mouse[0], app.canvas.graph_mouse[1]];
|
node.pos = [app.canvas.graph_mouse[0], app.canvas.graph_mouse[1]];
|
||||||
app.canvas.graph.add(node, false);
|
app.canvas.graph.add(node, false);
|
||||||
}
|
}
|
||||||
@@ -786,7 +788,7 @@ app.graphToPrompt = async function () {
|
|||||||
// get used group nodes
|
// get used group nodes
|
||||||
let used_group_nodes = new Set();
|
let used_group_nodes = new Set();
|
||||||
for(let node of p.workflow.nodes) {
|
for(let node of p.workflow.nodes) {
|
||||||
if(node.type.startsWith('workflow/')) {
|
if(node.type.startsWith(`workflow/`) || node.type.startsWith(`workflow${SEPARATOR}`)) {
|
||||||
used_group_nodes.add(node.type.substring(9));
|
used_group_nodes.add(node.type.substring(9));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1106,7 +1106,7 @@ export class CustomNodesManager {
|
|||||||
|
|
||||||
for (let i in nodes) {
|
for (let i in nodes) {
|
||||||
const node_type = nodes[i].type;
|
const node_type = nodes[i].type;
|
||||||
if(node_type.startsWith('workflow/'))
|
if(node_type.startsWith('workflow/') || node_type.startsWith('workflow>'))
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
if (!registered_nodes.has(node_type)) {
|
if (!registered_nodes.has(node_type)) {
|
||||||
|
|||||||
@@ -101,7 +101,7 @@ function connect_inputs(nearest_inputs, node) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function node_info_copy(src, dest, connect_both) {
|
function node_info_copy(src, dest, connect_both, copy_shape) {
|
||||||
// copy input connections
|
// copy input connections
|
||||||
for(let i in src.inputs) {
|
for(let i in src.inputs) {
|
||||||
let input = src.inputs[i];
|
let input = src.inputs[i];
|
||||||
@@ -142,9 +142,11 @@ function node_info_copy(src, dest, connect_both) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
dest.color = src.color;
|
if(copy_shape) {
|
||||||
dest.bgcolor = src.bgcolor;
|
dest.color = src.color;
|
||||||
dest.size = src.size;
|
dest.bgcolor = src.bgcolor;
|
||||||
|
dest.size = max(src.size, dest.size);
|
||||||
|
}
|
||||||
|
|
||||||
app.graph.afterChange();
|
app.graph.afterChange();
|
||||||
}
|
}
|
||||||
@@ -162,6 +164,7 @@ app.registerExtension({
|
|||||||
|
|
||||||
switch(double_click_policy) {
|
switch(double_click_policy) {
|
||||||
case "copy-all":
|
case "copy-all":
|
||||||
|
case "copy-full":
|
||||||
case "copy-input":
|
case "copy-input":
|
||||||
{
|
{
|
||||||
if(node.inputs?.some(x => x.link != null) || node.outputs?.some(x => x.links != null && x.links.length > 0) )
|
if(node.inputs?.some(x => x.link != null) || node.outputs?.some(x => x.links != null && x.links.length > 0) )
|
||||||
@@ -169,7 +172,11 @@ app.registerExtension({
|
|||||||
|
|
||||||
let src_node = lookup_nearest_nodes(node);
|
let src_node = lookup_nearest_nodes(node);
|
||||||
if(src_node)
|
if(src_node)
|
||||||
node_info_copy(src_node, node, double_click_policy == "copy-all");
|
{
|
||||||
|
let both_connection = double_click_policy != "copy-input";
|
||||||
|
let copy_shape = double_click_policy == "copy-full";
|
||||||
|
node_info_copy(src_node, node, both_connection, copy_shape);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case "possible-input":
|
case "possible-input":
|
||||||
|
|||||||
@@ -1,83 +0,0 @@
|
|||||||
import {app} from "../../scripts/app.js";
|
|
||||||
import {ComfyWidgets} from "../../scripts/widgets.js";
|
|
||||||
// Node that add notes to your project
|
|
||||||
|
|
||||||
let terminal_node;
|
|
||||||
let log_mode = false;
|
|
||||||
|
|
||||||
app.registerExtension({
|
|
||||||
name: "Comfy.Manager.Terminal",
|
|
||||||
|
|
||||||
registerCustomNodes() {
|
|
||||||
class TerminalNode extends LiteGraph.LGraphNode {
|
|
||||||
color = "#222222";
|
|
||||||
bgcolor = "#000000";
|
|
||||||
groupcolor = LGraphCanvas.node_colors.black.groupcolor;
|
|
||||||
constructor() {
|
|
||||||
super();
|
|
||||||
this.title = "Terminal Log (Manager)";
|
|
||||||
this.logs = [];
|
|
||||||
|
|
||||||
if (!this.properties) {
|
|
||||||
this.properties = {};
|
|
||||||
this.properties.text="";
|
|
||||||
}
|
|
||||||
|
|
||||||
ComfyWidgets.STRING(this, "", ["", {default:this.properties.text, multiline: true}], app)
|
|
||||||
ComfyWidgets.BOOLEAN(this, "mode", ["", {default:true, label_on:'Logging', label_off:'Stop'}], app)
|
|
||||||
ComfyWidgets.INT(this, "lines", ["", {default:500, min:10, max:10000, steps:1}], app)
|
|
||||||
|
|
||||||
let self = this;
|
|
||||||
Object.defineProperty(this.widgets[1], 'value', {
|
|
||||||
set: (v) => {
|
|
||||||
api.fetchApi(`/manager/terminal?mode=${v}`, {});
|
|
||||||
log_mode = v;
|
|
||||||
},
|
|
||||||
get: () => {
|
|
||||||
return log_mode;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
this.serialize_widgets = false;
|
|
||||||
this.isVirtualNode = true;
|
|
||||||
|
|
||||||
if(terminal_node) {
|
|
||||||
try {
|
|
||||||
terminal_node.widgets[0].value = 'The output of this node is disabled because another terminal node has appeared.';
|
|
||||||
}
|
|
||||||
catch {}
|
|
||||||
}
|
|
||||||
terminal_node = this;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Load default visibility
|
|
||||||
LiteGraph.registerNodeType(
|
|
||||||
"Terminal Log //CM",
|
|
||||||
Object.assign(TerminalNode, {
|
|
||||||
title_mode: LiteGraph.NORMAL_TITLE,
|
|
||||||
title: "Terminal Log (Manager)",
|
|
||||||
collapsable: true,
|
|
||||||
})
|
|
||||||
);
|
|
||||||
|
|
||||||
TerminalNode.category = "utils";
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
|
|
||||||
import { api } from "../../scripts/api.js";
|
|
||||||
|
|
||||||
function terminalFeedback(event) {
|
|
||||||
if(terminal_node) {
|
|
||||||
terminal_node.logs.push(event.detail.data);
|
|
||||||
if(terminal_node.logs.length > terminal_node.widgets[2].value) {
|
|
||||||
terminal_node.logs.shift();
|
|
||||||
if(terminal_node.logs[0] == '' || terminal_node.logs[0] == '\n')
|
|
||||||
terminal_node.logs.shift();
|
|
||||||
}
|
|
||||||
terminal_node.widgets[0].value = [...terminal_node.logs].reverse().join('').trim();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
api.addEventListener("manager-terminal-feedback", terminalFeedback);
|
|
||||||
@@ -3,10 +3,12 @@ import argparse
|
|||||||
|
|
||||||
def check_json_syntax(file_path):
|
def check_json_syntax(file_path):
|
||||||
try:
|
try:
|
||||||
with open(file_path, 'r') as file:
|
with open(file_path, 'r', encoding='utf-8') as file:
|
||||||
json_str = file.read()
|
json_str = file.read()
|
||||||
json.loads(json_str)
|
json.loads(json_str)
|
||||||
print(f"[ OK ] {file_path}")
|
print(f"[ OK ] {file_path}")
|
||||||
|
except UnicodeDecodeError as e:
|
||||||
|
print(f"Unicode decode error: {e}")
|
||||||
except json.JSONDecodeError as e:
|
except json.JSONDecodeError as e:
|
||||||
print(f"[FAIL] {file_path}\n\n {e}\n")
|
print(f"[FAIL] {file_path}\n\n {e}\n")
|
||||||
except FileNotFoundError:
|
except FileNotFoundError:
|
||||||
|
|||||||
565
model-list.json
565
model-list.json
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
@@ -1,8 +1,48 @@
|
|||||||
{
|
{
|
||||||
"custom_nodes": [
|
"custom_nodes": [
|
||||||
|
{
|
||||||
|
"author": "BlenderNeko",
|
||||||
|
"title": "ltdrdata/ComfyUI_TiledKSampler",
|
||||||
|
"reference": "https://github.com/ltdrdata/ComfyUI_TiledKSampler",
|
||||||
|
"files": [
|
||||||
|
"https://github.com/ltdrdata/ComfyUI_TiledKSampler"
|
||||||
|
],
|
||||||
|
"install_type": "git-clone",
|
||||||
|
"description": "PR for [a/https://github.com/BlenderNeko/ComfyUI_TiledKSampler/pull/59](https://github.com/BlenderNeko/ComfyUI_TiledKSampler/pull/59)"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"author": "leeooo001",
|
||||||
|
"title": "ComfyUI-leo-Hamer",
|
||||||
|
"reference": "https://github.com/leeooo001/ComfyUI-leo-Hamer",
|
||||||
|
"files": [
|
||||||
|
"https://github.com/leeooo001/ComfyUI-leo-Hamer"
|
||||||
|
],
|
||||||
|
"install_type": "git-clone",
|
||||||
|
"description": "Unoffice Hamer-ComfyUI by leo\nNOTE:base on [a/hamer](https://github.com/geopavlakos/hamer)"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"author": "leeooo001",
|
||||||
|
"title": "ComfyUI-leo-GVHMR",
|
||||||
|
"reference": "https://github.com/leeooo001/ComfyUI-leo-GVHMR",
|
||||||
|
"files": [
|
||||||
|
"https://github.com/leeooo001/ComfyUI-leo-GVHMR"
|
||||||
|
],
|
||||||
|
"install_type": "git-clone",
|
||||||
|
"description": "Unoffice Hamer-ComfyUI by leo\nNOTE:base on [a/GVHMR](https://github.com/zju3dv/GVHMR)"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"author": "leeooo001",
|
||||||
|
"title": "RealisDance-ComfyUI",
|
||||||
|
"reference": "https://github.com/leeooo001/ComfyUI-leo-RealisDance",
|
||||||
|
"files": [
|
||||||
|
"https://github.com/leeooo001/ComfyUI-leo-RealisDance"
|
||||||
|
],
|
||||||
|
"install_type": "git-clone",
|
||||||
|
"description": "Unoffice RealisDance-ComfyUI by leo\nNOTE:base on [a/RealisDance](https://github.com/damo-cv/RealisDance), modified on [a/RealisDanceComfyui](https://github.com/AIFSH/RealisDance-ComfyUI)"
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"author": "jags111",
|
"author": "jags111",
|
||||||
"title": "NyaamZ/ComfyUI-Long-CLIP",
|
"title": "NyaamZ/efficiency-nodes-ED",
|
||||||
"reference": "https://github.com/NyaamZ/efficiency-nodes-ED",
|
"reference": "https://github.com/NyaamZ/efficiency-nodes-ED",
|
||||||
"files": [
|
"files": [
|
||||||
"https://github.com/NyaamZ/efficiency-nodes-ED"
|
"https://github.com/NyaamZ/efficiency-nodes-ED"
|
||||||
|
|||||||
@@ -9,8 +9,215 @@
|
|||||||
"description": "If you see this message, your ComfyUI-Manager is outdated.\nLegacy channel provides only the list of the deprecated nodes. If you want to find the complete node list, please go to the Default channel."
|
"description": "If you see this message, your ComfyUI-Manager is outdated.\nLegacy channel provides only the list of the deprecated nodes. If you want to find the complete node list, please go to the Default channel."
|
||||||
},
|
},
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
{
|
||||||
|
"author": "huangyangke",
|
||||||
|
"title": "ComfyUI-Kolors-IpadapterFaceId [DEPRECATED]",
|
||||||
|
"reference": "https://github.com/huangyangke/ComfyUI-Kolors-IpadapterFaceId",
|
||||||
|
"files": [
|
||||||
|
"https://github.com/huangyangke/ComfyUI-Kolors-IpadapterFaceId"
|
||||||
|
],
|
||||||
|
"install_type": "git-clone",
|
||||||
|
"description": "NODES:kolors_ipadapter_faceid\nNOTE: The files in the repo are not organized."
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"author": "zmwv823",
|
||||||
|
"title": "ComfyUI_Ctrlora [DEPRECATED]",
|
||||||
|
"reference": "https://github.com/zmwv823/ComfyUI_Ctrlora",
|
||||||
|
"files": [
|
||||||
|
"https://github.com/zmwv823/ComfyUI_Ctrlora"
|
||||||
|
],
|
||||||
|
"install_type": "git-clone",
|
||||||
|
"description": "Unofficial custom_node for [a/xyfJASON/ctrlora](https://github.com/xyfJASON/ctrlora)."
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"author": "Fannovel16",
|
||||||
|
"title": "ComfyUI Loopchain [DEPRECATED]",
|
||||||
|
"id": "loopchain",
|
||||||
|
"reference": "https://github.com/Fannovel16/ComfyUI-Loopchain",
|
||||||
|
"files": [
|
||||||
|
"https://github.com/Fannovel16/ComfyUI-Loopchain"
|
||||||
|
],
|
||||||
|
"install_type": "git-clone",
|
||||||
|
"description": "A collection of nodes which can be useful for animation in ComfyUI. The main focus of this extension is implementing a mechanism called loopchain. A loopchain in this case is the chain of nodes only executed repeatly in the workflow. If a node chain contains a loop node from this extension, it will become a loop chain."
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"author": "DonBaronFactory",
|
||||||
|
"title": "ComfyUI-Cre8it-Nodes [DEPRECATED]",
|
||||||
|
"reference": "https://github.com/DonBaronFactory/ComfyUI-Cre8it-Nodes",
|
||||||
|
"files": [
|
||||||
|
"https://github.com/DonBaronFactory/ComfyUI-Cre8it-Nodes"
|
||||||
|
],
|
||||||
|
"install_type": "git-clone",
|
||||||
|
"description": "Nodes:CRE8IT Serial Prompter, CRE8IT Apply Serial Prompter, CRE8IT Image Sizer. A few simple nodes to facilitate working wiht ComfyUI Workflows"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"author": "thecooltechguy",
|
||||||
|
"title": "ComfyUI-ComfyRun [DEPRECATED/UNSAFE]",
|
||||||
|
"reference": "https://github.com/thecooltechguy/ComfyUI-ComfyRun",
|
||||||
|
"files": [
|
||||||
|
"https://github.com/thecooltechguy/ComfyUI-ComfyRun"
|
||||||
|
],
|
||||||
|
"install_type": "git-clone",
|
||||||
|
"description": "The easiest way to run & share any ComfyUI workflow [a/https://comfyrun.com](https://comfyrun.com)\nNOTE: Vulnerability discovered. Not being managed."
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"author": "Cardoso-topdev",
|
||||||
|
"title": "comfyui_meshanything_v1 [REMOVED]",
|
||||||
|
"reference": "https://github.com/Cardoso-topdev/comfyui_meshanything_v1",
|
||||||
|
"files": [
|
||||||
|
"https://github.com/Cardoso-topdev/comfyui_meshanything_v1"
|
||||||
|
],
|
||||||
|
"install_type": "git-clone",
|
||||||
|
"description": "MeshAnything V2: Artist-Created Mesh Generation With Adjacent Mesh Tokenization"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"author": "palant",
|
||||||
|
"title": "Extended Save Image for ComfyUI [DEPRECATED]",
|
||||||
|
"reference": "https://github.com/palant/extended-saveimage-comfyui",
|
||||||
|
"files": [
|
||||||
|
"https://github.com/palant/extended-saveimage-comfyui"
|
||||||
|
],
|
||||||
|
"install_type": "git-clone",
|
||||||
|
"description": "This custom node is largely identical to the usual Save Image but allows saving images also in JPEG and WEBP formats, the latter with both lossless and lossy compression. Metadata is embedded in the images as usual, and the resulting images can be used to load a workflow."
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"author": "1038lab",
|
||||||
|
"title": "ComfyUI-GPT2P [REMOVED]",
|
||||||
|
"id": "gpt2p",
|
||||||
|
"reference": "https://github.com/1038lab/ComfyUI-GPT2P",
|
||||||
|
"files": [
|
||||||
|
"https://github.com/1038lab/ComfyUI-GPT2P"
|
||||||
|
],
|
||||||
|
"install_type": "git-clone",
|
||||||
|
"description": "ComfyUI Node - Hugging Face repositories GTP2 Prompt"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"author": "yushan777",
|
||||||
|
"title": "Y7 Nodes for ComfyUI [REMOVED]",
|
||||||
|
"id": "y7nodes",
|
||||||
|
"reference": "https://github.com/yushan777/ComfyUI-Y7Nodes",
|
||||||
|
"files": [
|
||||||
|
"https://github.com/yushan777/ComfyUI-Y7Nodes"
|
||||||
|
],
|
||||||
|
"install_type": "git-clone",
|
||||||
|
"description": "Nodes:Count_Tokens_(Y7)"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"author": "city96",
|
||||||
|
"title": "SD-Advanced-Noise [DEPRECATED]",
|
||||||
|
"id": "adv-noise",
|
||||||
|
"reference": "https://github.com/city96/SD-Advanced-Noise",
|
||||||
|
"files": [
|
||||||
|
"https://github.com/city96/SD-Advanced-Noise"
|
||||||
|
],
|
||||||
|
"install_type": "git-clone",
|
||||||
|
"description": "Nodes: LatentGaussianNoise, MathEncode. An experimental custom node that generates latent noise directly by utilizing the linear characteristics of the latent space."
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"author": "shockz0rz",
|
||||||
|
"title": "InterpolateEverything [DEPRECATED]",
|
||||||
|
"id": "interpolate-everything",
|
||||||
|
"reference": "https://github.com/shockz0rz/ComfyUI_InterpolateEverything",
|
||||||
|
"files": [
|
||||||
|
"https://github.com/shockz0rz/ComfyUI_InterpolateEverything"
|
||||||
|
],
|
||||||
|
"install_type": "git-clone",
|
||||||
|
"description": "Nodes: Interpolate Poses, Interpolate Lineart, ... Custom nodes for interpolating between, well, everything in the Stable Diffusion ComfyUI."
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"author": "svdC1",
|
||||||
|
"title": "LoRa Dataset Tools [REMOVED]",
|
||||||
|
"reference": "https://github.com/svdC1/comfy-ui-lora-dataset-tools",
|
||||||
|
"files": [
|
||||||
|
"https://github.com/svdC1/comfy-ui-lora-dataset-tools"
|
||||||
|
],
|
||||||
|
"install_type": "git-clone",
|
||||||
|
"description": "NODES:Directory Loader, Filter Images Without Faces, Detect Faces and Draw Detection Box"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"author": "MiddleKD",
|
||||||
|
"title": "ComfyUI-default-workflow-setter [REMOVED]",
|
||||||
|
"reference": "https://github.com/MiddleKD/ComfyUI-default-workflow-setter",
|
||||||
|
"files": [
|
||||||
|
"https://github.com/MiddleKD/ComfyUI-default-workflow-setter"
|
||||||
|
],
|
||||||
|
"install_type": "git-clone",
|
||||||
|
"description": "Default workflow setter"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"author": "Firetheft",
|
||||||
|
"title": "ComfyUI-Flux-Prompt-Tools [REMOVED]",
|
||||||
|
"reference": "https://github.com/Firetheft/ComfyUI-Flux-Prompt-Tools",
|
||||||
|
"files": [
|
||||||
|
"https://github.com/Firetheft/ComfyUI-Flux-Prompt-Tools"
|
||||||
|
],
|
||||||
|
"install_type": "git-clone",
|
||||||
|
"description": "NODES:Flux Prompt Enhance, Flux Prompt Gemini Flash, Flux Prompt Generator, MiniCPM V2.6 Int4"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"author": "jtydhr88",
|
||||||
|
"title": "ComfyUI Unique3D [DEPRECATED]",
|
||||||
|
"id": "unique3d",
|
||||||
|
"reference": "https://github.com/jtydhr88/ComfyUI-Unique3D",
|
||||||
|
"files": [
|
||||||
|
"https://github.com/jtydhr88/ComfyUI-Unique3D"
|
||||||
|
],
|
||||||
|
"install_type": "git-clone",
|
||||||
|
"description": "ComfyUI Unique3D is custom nodes that running AiuniAI/Unique3D into ComfyUI[w/Please follow readme to install with ComfyUI embedded python.]"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"author": "mpiquero7164",
|
||||||
|
"title": "SaveImgPrompt [DEPRECATED]",
|
||||||
|
"id": "save-imgprompt",
|
||||||
|
"reference": "https://github.com/mpiquero7164/ComfyUI-SaveImgPrompt",
|
||||||
|
"files": [
|
||||||
|
"https://github.com/mpiquero7164/ComfyUI-SaveImgPrompt"
|
||||||
|
],
|
||||||
|
"install_type": "git-clone",
|
||||||
|
"description": "Save a png or jpeg and option to save prompt/workflow in a text or json file for each image in Comfy + Workflow loading."
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"author": "guoyk93",
|
||||||
|
"title": "y.k.'s ComfyUI node suite [DEPRECATED]",
|
||||||
|
"id": "yks",
|
||||||
|
"reference": "https://github.com/yankeguo-deprecated/yk-node-suite-comfyui",
|
||||||
|
"files": [
|
||||||
|
"https://github.com/yankeguo-deprecated/yk-node-suite-comfyui"
|
||||||
|
],
|
||||||
|
"install_type": "git-clone",
|
||||||
|
"description": "Nodes: YKImagePadForOutpaint, YKMaskToImage"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"author": "adityathiru",
|
||||||
|
"title": "ComfyUI LLMs [REMOVED]",
|
||||||
|
"reference": "https://github.com/adityathiru/ComfyUI-LLMs",
|
||||||
|
"files": [
|
||||||
|
"https://github.com/adityathiru/ComfyUI-LLMs"
|
||||||
|
],
|
||||||
|
"install_type": "git-clone",
|
||||||
|
"description": "Goal: To enable folks to rapidly build complex workflows with LLMs\nNOTE:☠️ This is experimental and not recommended to use in a production environment (yet!)"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"author": "DannyStone1999",
|
||||||
|
"title": "ComfyUI-Depth2Mask [REMOVED]",
|
||||||
|
"reference": "https://github.com/DannyStone1999/ComfyUI-Depth2Mask",
|
||||||
|
"files": [
|
||||||
|
"https://github.com/DannyStone1999/ComfyUI-Depth2Mask/raw/main/Depth2Mask.py"
|
||||||
|
],
|
||||||
|
"install_type": "copy",
|
||||||
|
"description": "Nodes:Depth2Mask"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"author": "syaofox",
|
||||||
|
"title": "ComfyUI_FoxTools [REMOVED]",
|
||||||
|
"reference": "https://github.com/syaofox/ComfyUI_FoxTools",
|
||||||
|
"files": [
|
||||||
|
"https://github.com/syaofox/ComfyUI_FoxTools"
|
||||||
|
],
|
||||||
|
"install_type": "git-clone",
|
||||||
|
"description": "Nodes:BatchImageFromList, Load Face Occlusion Model, Create Face Mask, Simple FaceAlign, Cacul FaceAlign, Gen Blurbord, Face Align, Face Rotate, ImageAdd, LoadImageList, SaveImage Plus, RegTextFind"
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"author": "AIFSH",
|
"author": "AIFSH",
|
||||||
"title": "SeedVC-ComfyUI [REMOVED]",
|
"title": "SeedVC-ComfyUI [REMOVED]",
|
||||||
@@ -84,16 +291,6 @@
|
|||||||
"install_type": "git-clone",
|
"install_type": "git-clone",
|
||||||
"description": "ComfyUI-Talking-Head"
|
"description": "ComfyUI-Talking-Head"
|
||||||
},
|
},
|
||||||
{
|
|
||||||
"author": "yvann-ba",
|
|
||||||
"title": "ComfyUI_Yvann-Nodes [REMOVED]",
|
|
||||||
"reference": "https://github.com/yvann-ba/ComfyUI_Yvann-Nodes",
|
|
||||||
"files": [
|
|
||||||
"https://github.com/yvann-ba/ComfyUI_Yvann-Nodes"
|
|
||||||
],
|
|
||||||
"install_type": "git-clone",
|
|
||||||
"description": "Audio reactivity nodes for AI animations 🔊 Analyze audio, extract drums and vocals. Generate reactive masks and weights. Create audio-driven visuals. Produce weight graphs and audio masks. Compatible with IPAdapter, ControlNets and more. Features audio scheduling and waveform analysis. Tutorials to use this pack: [a/Yvann Youtube](https://www.youtube.com/@yvann.mp4)"
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
"author": "jh-leon-kim",
|
"author": "jh-leon-kim",
|
||||||
"title": "ComfyUI-JHK-utils [REMOVED]",
|
"title": "ComfyUI-JHK-utils [REMOVED]",
|
||||||
@@ -145,16 +342,6 @@
|
|||||||
"install_type": "git-clone",
|
"install_type": "git-clone",
|
||||||
"description": "This custom node for ComfyUI provides full-body animation capabilities, including facial rigging, various lighting styles, and green screen output."
|
"description": "This custom node for ComfyUI provides full-body animation capabilities, including facial rigging, various lighting styles, and green screen output."
|
||||||
},
|
},
|
||||||
{
|
|
||||||
"author": "m-ai-studio",
|
|
||||||
"title": "mai-prompt-progress [REMOVED]",
|
|
||||||
"reference": "https://github.com/m-ai-studio/mai-prompt-progress",
|
|
||||||
"files": [
|
|
||||||
"https://github.com/m-ai-studio/mai-prompt-progress"
|
|
||||||
],
|
|
||||||
"install_type": "git-clone",
|
|
||||||
"description": "mai-prompt-progress"
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
"author": "ZHO-ZHO-ZHO",
|
"author": "ZHO-ZHO-ZHO",
|
||||||
"title": "ComfyUI-AnyText [NOT MAINTAINED]",
|
"title": "ComfyUI-AnyText [NOT MAINTAINED]",
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
@@ -1,5 +1,336 @@
|
|||||||
{
|
{
|
||||||
"models": [
|
"models": [
|
||||||
|
{
|
||||||
|
"name": "stabilityai/SD3.5-Large-Controlnet-Blur",
|
||||||
|
"type": "controlnet",
|
||||||
|
"base": "SD3.5",
|
||||||
|
"save_path": "controlnet/SD3.5",
|
||||||
|
"description": "Blur Controlnet model for SD3.5 Large",
|
||||||
|
"reference": "https://huggingface.co/stabilityai/stable-diffusion-3.5-controlnets",
|
||||||
|
"filename": "sd3.5_large_controlnet_blur.safetensors",
|
||||||
|
"url": "https://huggingface.co/stabilityai/stable-diffusion-3.5-controlnets/resolve/main/sd3.5_large_controlnet_blur.safetensors",
|
||||||
|
"size": "8.65GB"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "stabilityai/SD3.5-Large-Controlnet-Canny",
|
||||||
|
"type": "controlnet",
|
||||||
|
"base": "SD3.5",
|
||||||
|
"save_path": "controlnet/SD3.5",
|
||||||
|
"description": "Canny Controlnet model for SD3.5 Large",
|
||||||
|
"reference": "https://huggingface.co/stabilityai/stable-diffusion-3.5-controlnets",
|
||||||
|
"filename": "sd3.5_large_controlnet_canny.safetensors",
|
||||||
|
"url": "https://huggingface.co/stabilityai/stable-diffusion-3.5-controlnets/resolve/main/sd3.5_large_controlnet_canny.safetensors",
|
||||||
|
"size": "8.65GB"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "stabilityai/SD3.5-Large-Controlnet-Depth",
|
||||||
|
"type": "controlnet",
|
||||||
|
"base": "SD3.5",
|
||||||
|
"save_path": "controlnet/SD3.5",
|
||||||
|
"description": "Depth Controlnet model for SD3.5 Large",
|
||||||
|
"reference": "https://huggingface.co/stabilityai/stable-diffusion-3.5-controlnets",
|
||||||
|
"filename": "sd3.5_large_controlnet_depth.safetensors",
|
||||||
|
"url": "https://huggingface.co/stabilityai/stable-diffusion-3.5-controlnets/resolve/main/sd3.5_large_controlnet_depth.safetensors",
|
||||||
|
"size": "8.65GB"
|
||||||
|
},
|
||||||
|
|
||||||
|
{
|
||||||
|
"name": "LTX-Video 2B v0.9 Checkpoint",
|
||||||
|
"type": "checkpoint",
|
||||||
|
"base": "LTX-Video",
|
||||||
|
"save_path": "checkpoints/LTXV",
|
||||||
|
"description": "LTX-Video is the first DiT-based video generation model capable of generating high-quality videos in real-time. It produces 24 FPS videos at a 768x512 resolution faster than they can be watched. Trained on a large-scale dataset of diverse videos, the model generates high-resolution videos with realistic and varied content.",
|
||||||
|
"reference": "https://huggingface.co/Lightricks/LTX-Video",
|
||||||
|
"filename": "ltx-video-2b-v0.9.safetensors",
|
||||||
|
"url": "https://huggingface.co/Lightricks/LTX-Video/resolve/main/ltx-video-2b-v0.9.safetensors",
|
||||||
|
"size": "9.37GB"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "InstantX/FLUX.1-dev-IP-Adapter",
|
||||||
|
"type": "IP-Adapter",
|
||||||
|
"base": "FLUX.1",
|
||||||
|
"save_path": "ipadapter-flux",
|
||||||
|
"description": "FLUX.1-dev-IP-Adapter",
|
||||||
|
"reference": "https://huggingface.co/InstantX/FLUX.1-dev-IP-Adapter",
|
||||||
|
"filename": "ip-adapter.bin",
|
||||||
|
"url": "https://huggingface.co/InstantX/FLUX.1-dev-IP-Adapter/resolve/main/ip-adapter.bin",
|
||||||
|
"size": "5.29GB"
|
||||||
|
},
|
||||||
|
|
||||||
|
{
|
||||||
|
"name": "Comfy-Org/sigclip_vision_384 (patch14_384)",
|
||||||
|
"type": "clip_vision",
|
||||||
|
"base": "sigclip",
|
||||||
|
"save_path": "clip_vision",
|
||||||
|
"description": "This clip vision model is required for FLUX.1 Redux.",
|
||||||
|
"reference": "https://huggingface.co/Comfy-Org/sigclip_vision_384/tree/main",
|
||||||
|
"filename": "sigclip_vision_patch14_384.safetensors",
|
||||||
|
"url": "https://huggingface.co/Comfy-Org/sigclip_vision_384/resolve/main/sigclip_vision_patch14_384.safetensors",
|
||||||
|
"size": "857MB"
|
||||||
|
},
|
||||||
|
|
||||||
|
{
|
||||||
|
"name": "comfyanonymous/flux_text_encoders - t5xxl (fp16)",
|
||||||
|
"type": "clip",
|
||||||
|
"base": "t5",
|
||||||
|
"save_path": "text_encoders/t5",
|
||||||
|
"description": "Text Encoders for FLUX (fp16)",
|
||||||
|
"reference": "https://huggingface.co/comfyanonymous/flux_text_encoders",
|
||||||
|
"filename": "t5xxl_fp16.safetensors",
|
||||||
|
"url": "https://huggingface.co/comfyanonymous/flux_text_encoders/resolve/main/t5xxl_fp16.safetensors",
|
||||||
|
"size": "9.79GB"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "comfyanonymous/flux_text_encoders - t5xxl (fp8_e4m3fn)",
|
||||||
|
"type": "clip",
|
||||||
|
"base": "t5",
|
||||||
|
"save_path": "text_encoders/t5",
|
||||||
|
"description": "Text Encoders for FLUX (fp8_e4m3fn)",
|
||||||
|
"reference": "https://huggingface.co/comfyanonymous/flux_text_encoders",
|
||||||
|
"filename": "t5xxl_fp8_e4m3fn.safetensors",
|
||||||
|
"url": "https://huggingface.co/comfyanonymous/flux_text_encoders/resolve/main/t5xxl_fp8_e4m3fn.safetensors",
|
||||||
|
"size": "4.89GB"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "comfyanonymous/flux_text_encoders - t5xxl (fp8_e4m3fn_scaled)",
|
||||||
|
"type": "clip",
|
||||||
|
"base": "t5",
|
||||||
|
"save_path": "text_encoders/t5",
|
||||||
|
"description": "Text Encoders for FLUX (fp16)",
|
||||||
|
"reference": "https://huggingface.co/comfyanonymous/flux_text_encoders",
|
||||||
|
"filename": "t5xxl_fp8_e4m3fn_scaled.safetensors",
|
||||||
|
"url": "https://huggingface.co/comfyanonymous/flux_text_encoders/resolve/main/t5xxl_fp8_e4m3fn_scaled.safetensors",
|
||||||
|
"size": "5.16GB"
|
||||||
|
},
|
||||||
|
|
||||||
|
{
|
||||||
|
"name": "FLUX.1 [Dev] Diffusion model (scaled fp8)",
|
||||||
|
"type": "diffusion_model",
|
||||||
|
"base": "FLUX.1",
|
||||||
|
"save_path": "diffusion_models/FLUX1",
|
||||||
|
"description": "FLUX.1 [Dev] Diffusion model (scaled fp8)[w/Due to the large size of the model, it is recommended to download it through a browser if possible.]",
|
||||||
|
"reference": "https://huggingface.co/comfyanonymous/flux_dev_scaled_fp8_test",
|
||||||
|
"filename": "flux_dev_fp8_scaled_diffusion_model.safetensors",
|
||||||
|
"url": "https://huggingface.co/comfyanonymous/flux_dev_scaled_fp8_test/resolve/main/flux_dev_fp8_scaled_diffusion_model.safetensors",
|
||||||
|
"size": "11.9GB"
|
||||||
|
},
|
||||||
|
|
||||||
|
{
|
||||||
|
"name": "kijai/MoGe_ViT_L_fp16.safetensors",
|
||||||
|
"type": "MoGe",
|
||||||
|
"base": "MoGe",
|
||||||
|
"save_path": "MoGe",
|
||||||
|
"description": "Safetensors versions of [a/https://github.com/microsoft/MoGe](https://github.com/microsoft/MoGe)",
|
||||||
|
"reference": "https://huggingface.co/Kijai/MoGe_safetensors",
|
||||||
|
"filename": "MoGe_ViT_L_fp16.safetensors",
|
||||||
|
"url": "https://huggingface.co/Kijai/MoGe_safetensors/resolve/main/MoGe_ViT_L_fp16.safetensors",
|
||||||
|
"size": "628MB"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "kijai/MoGe_ViT_L_fp16.safetensors",
|
||||||
|
"type": "MoGe",
|
||||||
|
"base": "MoGe",
|
||||||
|
"save_path": "MoGe",
|
||||||
|
"description": "Safetensors versions of [a/https://github.com/microsoft/MoGe](https://github.com/microsoft/MoGe)",
|
||||||
|
"reference": "https://huggingface.co/Kijai/MoGe_safetensors",
|
||||||
|
"filename": "MoGe_ViT_L_fp16.safetensors",
|
||||||
|
"url": "https://huggingface.co/Kijai/MoGe_safetensors/resolve/main/MoGe_ViT_L_fp16.safetensors",
|
||||||
|
"size": "1.26GB"
|
||||||
|
},
|
||||||
|
|
||||||
|
{
|
||||||
|
"name": "pulid_flux_v0.9.1.safetensors",
|
||||||
|
"type": "PuLID",
|
||||||
|
"base": "FLUX",
|
||||||
|
"save_path": "pulid",
|
||||||
|
"description": "This is required for PuLID (FLUX)",
|
||||||
|
"reference": "https://huggingface.co/guozinan/PuLID",
|
||||||
|
"filename": "pulid_flux_v0.9.1.safetensors",
|
||||||
|
"url": "https://huggingface.co/guozinan/PuLID/resolve/main/pulid_flux_v0.9.1.safetensors",
|
||||||
|
"size": "1.14GB"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "pulid_v1.1.safetensors",
|
||||||
|
"type": "PuLID",
|
||||||
|
"base": "SDXL",
|
||||||
|
"save_path": "pulid",
|
||||||
|
"description": "This is required for PuLID (SDXL)",
|
||||||
|
"reference": "https://huggingface.co/guozinan/PuLID",
|
||||||
|
"filename": "pulid_v1.1.safetensors",
|
||||||
|
"url": "https://huggingface.co/guozinan/PuLID/resolve/main/pulid_v1.1.safetensors",
|
||||||
|
"size": "984MB"
|
||||||
|
},
|
||||||
|
|
||||||
|
{
|
||||||
|
"name": "Kolors-IP-Adapter-Plus.bin (Kwai-Kolors/Kolors-IP-Adapter-Plus)",
|
||||||
|
"type": "IP-Adapter",
|
||||||
|
"base": "Kolors",
|
||||||
|
"save_path": "ipadapter",
|
||||||
|
"description": "You can use this model in the [a/ComfyUI IPAdapter plus](https://github.com/cubiq/ComfyUI_IPAdapter_plus) extension.",
|
||||||
|
"reference": "https://huggingface.co/Kwai-Kolors/Kolors-IP-Adapter-Plus",
|
||||||
|
"filename": "Kolors-IP-Adapter-Plus.bin",
|
||||||
|
"url": "https://huggingface.co/Kwai-Kolors/Kolors-IP-Adapter-Plus/resolve/main/ip_adapter_plus_general.bin",
|
||||||
|
"size": "1.01GB"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "Kolors-IP-Adapter-FaceID-Plus.bin (Kwai-Kolors/Kolors-IP-Adapter-Plus)",
|
||||||
|
"type": "IP-Adapter",
|
||||||
|
"base": "Kolors",
|
||||||
|
"save_path": "ipadapter",
|
||||||
|
"description": "You can use this model in the [a/ComfyUI IPAdapter plus](https://github.com/cubiq/ComfyUI_IPAdapter_plus) extension.",
|
||||||
|
"reference": "https://huggingface.co/Kwai-Kolors/Kolors-IP-Adapter-FaceID-Plus",
|
||||||
|
"filename": "Kolors-IP-Adapter-FaceID-Plus.bin",
|
||||||
|
"url": "https://huggingface.co/Kwai-Kolors/Kolors-IP-Adapter-FaceID-Plus/resolve/main/ipa-faceid-plus.bin",
|
||||||
|
"size": "2.39GB"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "CLIPVision model (Kwai-Kolors/Kolors-IP-Adapter-Plus/clip-vit-large)",
|
||||||
|
"type": "clip_vision",
|
||||||
|
"base": "ViT-L",
|
||||||
|
"save_path": "clip_vision",
|
||||||
|
"description": "CLIPVision model (This is required in cubiq/ComfyUI_IPAdapter_plus)",
|
||||||
|
"reference": "https://huggingface.co/Kwai-Kolors/Kolors-IP-Adapter-Plus",
|
||||||
|
"filename": "clip-vit-large-patch14-336.bin",
|
||||||
|
"url": "https://huggingface.co/Kwai-Kolors/Kolors-IP-Adapter-Plus/resolve/main/image_encoder/pytorch_model.bin",
|
||||||
|
"size": "1.71GB"
|
||||||
|
},
|
||||||
|
|
||||||
|
{
|
||||||
|
"name": "kijai/lotus depth d model v1.1 (fp16)",
|
||||||
|
"type": "diffusion_model",
|
||||||
|
"base": "lotus",
|
||||||
|
"save_path": "diffusion_models",
|
||||||
|
"description": "lotus depth d model v1.1 (fp16). This model can be used in ComfyUI-Lotus custom nodes.",
|
||||||
|
"reference": "https://huggingface.co/Kijai/lotus-comfyui",
|
||||||
|
"filename": "lotus-depth-d-v-1-1-fp16.safetensors",
|
||||||
|
"url": "https://huggingface.co/Kijai/lotus-comfyui/resolve/main/lotus-depth-d-v-1-1-fp16.safetensors",
|
||||||
|
"size": "1.74GB"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "kijai/lotus depth g model v1.0 (fp16)",
|
||||||
|
"type": "diffusion_model",
|
||||||
|
"base": "lotus",
|
||||||
|
"save_path": "diffusion_models",
|
||||||
|
"description": "lotus depth g model v1.0 (fp16). This model can be used in ComfyUI-Lotus custom nodes.",
|
||||||
|
"reference": "https://huggingface.co/Kijai/lotus-comfyui",
|
||||||
|
"filename": "lotus-depth-g-v1-0-fp16.safetensors",
|
||||||
|
"url": "https://huggingface.co/Kijai/lotus-comfyui/resolve/main/lotus-depth-g-v1-0-fp16.safetensors",
|
||||||
|
"size": "1.74GB"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "kijai/lotus depth g model v1.0",
|
||||||
|
"type": "diffusion_model",
|
||||||
|
"base": "lotus",
|
||||||
|
"save_path": "diffusion_models",
|
||||||
|
"description": "lotus depth g model v1.0. This model can be used in ComfyUI-Lotus custom nodes.",
|
||||||
|
"reference": "https://huggingface.co/Kijai/lotus-comfyui",
|
||||||
|
"filename": "lotus-depth-g-v1-0.safetensors",
|
||||||
|
"url": "https://huggingface.co/Kijai/lotus-comfyui/resolve/main/lotus-depth-g-v1-0.safetensors",
|
||||||
|
"size": "3.47GB"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "kijai/lotus normal d model v1.0 (fp16)",
|
||||||
|
"type": "diffusion_model",
|
||||||
|
"base": "lotus",
|
||||||
|
"save_path": "diffusion_models",
|
||||||
|
"description": "lotus normal d model v1.0 (fp16). This model can be used in ComfyUI-Lotus custom nodes.",
|
||||||
|
"reference": "https://huggingface.co/Kijai/lotus-comfyui",
|
||||||
|
"filename": "lotus-normal-d-v1-0-fp16.safetensors",
|
||||||
|
"url": "https://huggingface.co/Kijai/lotus-comfyui/resolve/main/lotus-normal-d-v1-0-fp16.safetensors",
|
||||||
|
"size": "1.74GB"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "kijai/lotus normal d model v1.0",
|
||||||
|
"type": "diffusion_model",
|
||||||
|
"base": "lotus",
|
||||||
|
"save_path": "diffusion_models",
|
||||||
|
"description": "lotus normal d model v1.0. This model can be used in ComfyUI-Lotus custom nodes.",
|
||||||
|
"reference": "https://huggingface.co/Kijai/lotus-comfyui",
|
||||||
|
"filename": "lotus-normal-d-v1-0.safetensors",
|
||||||
|
"url": "https://huggingface.co/Kijai/lotus-comfyui/resolve/main/lotus-normal-d-v1-0.safetensors",
|
||||||
|
"size": "3.47GB"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "kijai/lotus normal g model v1.0 (fp16)",
|
||||||
|
"type": "diffusion_model",
|
||||||
|
"base": "lotus",
|
||||||
|
"save_path": "diffusion_models",
|
||||||
|
"description": "lotus normal g model v1.0 (fp16). This model can be used in ComfyUI-Lotus custom nodes.",
|
||||||
|
"reference": "https://huggingface.co/Kijai/lotus-comfyui",
|
||||||
|
"filename": "lotus-normal-g-v1-0-fp16.safetensors",
|
||||||
|
"url": "https://huggingface.co/Kijai/lotus-comfyui/resolve/main/lotus-normal-g-v1-0-fp16.safetensors",
|
||||||
|
"size": "1.74GB"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "kijai/lotus normal g model v1.0",
|
||||||
|
"type": "diffusion_model",
|
||||||
|
"base": "lotus",
|
||||||
|
"save_path": "diffusion_models",
|
||||||
|
"description": "lotus normal g model v1.0. This model can be used in ComfyUI-Lotus custom nodes.",
|
||||||
|
"reference": "https://huggingface.co/Kijai/lotus-comfyui",
|
||||||
|
"filename": "lotus-normal-g-v1-0.safetensors",
|
||||||
|
"url": "https://huggingface.co/Kijai/lotus-comfyui/resolve/main/lotus-normal-g-v1-0.safetensors",
|
||||||
|
"size": "3.47GB"
|
||||||
|
},
|
||||||
|
|
||||||
|
{
|
||||||
|
"name": "Depth Pro model",
|
||||||
|
"type": "depth-pro",
|
||||||
|
"base": "depth-pro",
|
||||||
|
"save_path": "depth/ml-depth-pro",
|
||||||
|
"description": "Depth pro model for [a/ComfyUI-Depth-Pro](https://github.com/spacepxl/ComfyUI-Depth-Pro)",
|
||||||
|
"reference": "https://huggingface.co/spacepxl/ml-depth-pro",
|
||||||
|
"filename": "depth_pro.fp16.safetensors",
|
||||||
|
"url": "https://huggingface.co/spacepxl/ml-depth-pro/resolve/main/depth_pro.fp16.safetensors",
|
||||||
|
"size": "1.9GB"
|
||||||
|
},
|
||||||
|
|
||||||
|
{
|
||||||
|
"name": "jasperai/FLUX.1-dev-Controlnet-Upscaler",
|
||||||
|
"type": "controlnet",
|
||||||
|
"base": "FLUX.1",
|
||||||
|
"save_path": "controlnet/FLUX.1/jasperai-dev-Upscaler",
|
||||||
|
"description": "This is Flux.1-dev ControlNet for low resolution images developed by Jasper research team.",
|
||||||
|
"reference": "https://huggingface.co/jasperai/Flux.1-dev-Controlnet-Upscaler",
|
||||||
|
"filename": "diffusion_pytorch_model.safetensors",
|
||||||
|
"url": "https://huggingface.co/jasperai/Flux.1-dev-Controlnet-Upscaler/resolve/main/diffusion_pytorch_model.safetensors",
|
||||||
|
"size": "3.58GB"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "jasperai/FLUX.1-dev-Controlnet-Depth",
|
||||||
|
"type": "controlnet",
|
||||||
|
"base": "FLUX.1",
|
||||||
|
"save_path": "controlnet/FLUX.1/jasperai-dev-Depth",
|
||||||
|
"description": "This is Flux.1-dev ControlNet for Depth map developed by Jasper research team.",
|
||||||
|
"reference": "https://huggingface.co/jasperai/Flux.1-dev-Controlnet-Depth",
|
||||||
|
"filename": "diffusion_pytorch_model.safetensors",
|
||||||
|
"url": "https://huggingface.co/jasperai/Flux.1-dev-Controlnet-Depth/resolve/main/diffusion_pytorch_model.safetensors",
|
||||||
|
"size": "3.58GB"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "jasperai/Flux.1-dev-Controlnet-Surface-Normals",
|
||||||
|
"type": "controlnet",
|
||||||
|
"base": "FLUX.1",
|
||||||
|
"save_path": "controlnet/FLUX.1/jasperai-dev-Surface-Normals",
|
||||||
|
"description": "This is Flux.1-dev ControlNet for Surface Normals map developed by Jasper research team.",
|
||||||
|
"reference": "https://huggingface.co/jasperai/Flux.1-dev-Controlnet-Surface-Normals",
|
||||||
|
"filename": "diffusion_pytorch_model.safetensors",
|
||||||
|
"url": "https://huggingface.co/jasperai/Flux.1-dev-Controlnet-Surface-Normals/resolve/main/diffusion_pytorch_model.safetensors",
|
||||||
|
"size": "3.58GB"
|
||||||
|
},
|
||||||
|
|
||||||
|
{
|
||||||
|
"name": "Shakker-Labs/FLUX.1-dev-ControlNet-Union-Pro (fp8_e4m3fn) by Kijai",
|
||||||
|
"type": "controlnet",
|
||||||
|
"base": "FLUX.1",
|
||||||
|
"save_path": "controlnet/FLUX.1",
|
||||||
|
"description": "FLUX.1 [Dev] Union Controlnet. Supports Canny, Tile, Depth, Blur, Pose, Gray, Low Quality\nVersion quantized to fp8_e4m3fn by Kijai",
|
||||||
|
"reference": "https://huggingface.co/Kijai/flux-fp8",
|
||||||
|
"filename": "flux_shakker_labs_union_pro-fp8_e4m3fn.safetensors",
|
||||||
|
"url": "https://huggingface.co/Kijai/flux-fp8/resolve/main/flux_shakker_labs_union_pro-fp8_e4m3fn.safetensors",
|
||||||
|
"size": "3.3GB"
|
||||||
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
"name": "ViT-L-14-TEXT-detail-improved-hiT-GmP-HF.safetensors [Long CLIP L]",
|
"name": "ViT-L-14-TEXT-detail-improved-hiT-GmP-HF.safetensors [Long CLIP L]",
|
||||||
"type": "clip",
|
"type": "clip",
|
||||||
@@ -349,369 +680,6 @@
|
|||||||
"filename": "flux1-dev-Q5_0.gguf",
|
"filename": "flux1-dev-Q5_0.gguf",
|
||||||
"url": "https://huggingface.co/city96/FLUX.1-dev-gguf/resolve/main/flux1-dev-Q5_0.gguf",
|
"url": "https://huggingface.co/city96/FLUX.1-dev-gguf/resolve/main/flux1-dev-Q5_0.gguf",
|
||||||
"size": "8.27GB"
|
"size": "8.27GB"
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "city96/flux1-dev-Q5_1.gguf",
|
|
||||||
"type": "diffusion_model",
|
|
||||||
"base": "FLUX.1",
|
|
||||||
"save_path": "diffusion_models/FLUX1",
|
|
||||||
"description": "FLUX.1 [Dev] Diffusion model (Q5_1/.gguf)",
|
|
||||||
"reference": "https://huggingface.co/city96/FLUX.1-dev-gguf",
|
|
||||||
"filename": "flux1-dev-Q5_1.gguf",
|
|
||||||
"url": "https://huggingface.co/city96/FLUX.1-dev-gguf/resolve/main/flux1-dev-Q5_1.gguf",
|
|
||||||
"size": "9.01GB"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "city96/flux1-dev-Q5_K_S.gguf",
|
|
||||||
"type": "diffusion_model",
|
|
||||||
"base": "FLUX.1",
|
|
||||||
"save_path": "diffusion_models/FLUX1",
|
|
||||||
"description": "FLUX.1 [Dev] Diffusion model (Q5_K_S/.gguf)",
|
|
||||||
"reference": "https://huggingface.co/city96/FLUX.1-dev-gguf",
|
|
||||||
"filename": "flux1-dev-Q5_K_S.gguf",
|
|
||||||
"url": "https://huggingface.co/city96/FLUX.1-dev-gguf/resolve/main/flux1-dev-Q5_K_S.gguf",
|
|
||||||
"size": "8.29GB"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "city96/flux1-dev-Q6_K.gguf",
|
|
||||||
"type": "diffusion_model",
|
|
||||||
"base": "FLUX.1",
|
|
||||||
"save_path": "diffusion_models/FLUX1",
|
|
||||||
"description": "FLUX.1 [Dev] Diffusion model (Q6_K/.gguf)",
|
|
||||||
"reference": "https://huggingface.co/city96/FLUX.1-dev-gguf",
|
|
||||||
"filename": "flux1-dev-Q6_K.gguf",
|
|
||||||
"url": "https://huggingface.co/city96/FLUX.1-dev-gguf/resolve/main/flux1-dev-Q6_K.gguf",
|
|
||||||
"size": "9.86GB"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "city96/flux1-dev-Q8_0.gguf",
|
|
||||||
"type": "diffusion_model",
|
|
||||||
"base": "FLUX.1",
|
|
||||||
"save_path": "diffusion_models/FLUX1",
|
|
||||||
"description": "FLUX.1 [Dev] Diffusion model (Q8_0/.gguf)",
|
|
||||||
"reference": "https://huggingface.co/city96/FLUX.1-dev-gguf",
|
|
||||||
"filename": "flux1-dev-Q8_0.gguf",
|
|
||||||
"url": "https://huggingface.co/city96/FLUX.1-dev-gguf/resolve/main/flux1-dev-Q8_0.gguf",
|
|
||||||
"size": "12.7GB"
|
|
||||||
},
|
|
||||||
|
|
||||||
{
|
|
||||||
"name": "city96/flux1-schnell-F16.gguf",
|
|
||||||
"type": "diffusion_model",
|
|
||||||
"base": "FLUX.1",
|
|
||||||
"save_path": "diffusion_models/FLUX1",
|
|
||||||
"description": "FLUX.1 [Dev] Diffusion model (f16/.gguf)",
|
|
||||||
"reference": "https://huggingface.co/city96/FLUX.1-schnell-gguf",
|
|
||||||
"filename": "flux1-schnell-F16.gguf",
|
|
||||||
"url": "https://huggingface.co/city96/FLUX.1-schnell-gguf/resolve/main/flux1-schnell-F16.gguf",
|
|
||||||
"size": "23.8GB"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "city96/flux1-schnell-Q2_K.gguf",
|
|
||||||
"type": "diffusion_model",
|
|
||||||
"base": "FLUX.1",
|
|
||||||
"save_path": "diffusion_models/FLUX1",
|
|
||||||
"description": "FLUX.1 [Dev] Diffusion model (Q2_K/.gguf)",
|
|
||||||
"reference": "https://huggingface.co/city96/FLUX.1-schnell-gguf",
|
|
||||||
"filename": "flux1-schnell-Q2_K.gguf",
|
|
||||||
"url": "https://huggingface.co/city96/FLUX.1-schnell-gguf/resolve/main/flux1-schnell-Q2_K.gguf",
|
|
||||||
"size": "4.01GB"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "city96/flux1-schnell-Q3_K_S.gguf",
|
|
||||||
"type": "diffusion_model",
|
|
||||||
"base": "FLUX.1",
|
|
||||||
"save_path": "diffusion_models/FLUX1",
|
|
||||||
"description": "FLUX.1 [Dev] Diffusion model (Q3_K_S/.gguf)",
|
|
||||||
"reference": "https://huggingface.co/city96/FLUX.1-schnell-gguf",
|
|
||||||
"filename": "flux1-schnell-Q3_K_S.gguf",
|
|
||||||
"url": "https://huggingface.co/city96/FLUX.1-schnell-gguf/resolve/main/flux1-schnell-Q3_K_S.gguf",
|
|
||||||
"size": "5.21GB"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "city96/flux1-schnell-Q4_0.gguf",
|
|
||||||
"type": "diffusion_model",
|
|
||||||
"base": "FLUX.1",
|
|
||||||
"save_path": "diffusion_models/FLUX1",
|
|
||||||
"description": "FLUX.1 [Dev] Diffusion model (Q4_0/.gguf)",
|
|
||||||
"reference": "https://huggingface.co/city96/FLUX.1-schnell-gguf",
|
|
||||||
"filename": "flux1-schnell-Q4_0.gguf",
|
|
||||||
"url": "https://huggingface.co/city96/FLUX.1-schnell-gguf/resolve/main/flux1-schnell-Q4_0.gguf",
|
|
||||||
"size": "6.77GB"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "city96/flux1-schnell-Q4_1.gguf",
|
|
||||||
"type": "diffusion_model",
|
|
||||||
"base": "FLUX.1",
|
|
||||||
"save_path": "diffusion_models/FLUX1",
|
|
||||||
"description": "FLUX.1 [Dev] Diffusion model (Q4_1/.gguf)",
|
|
||||||
"reference": "https://huggingface.co/city96/FLUX.1-schnell-gguf",
|
|
||||||
"filename": "flux1-schnell-Q4_1.gguf",
|
|
||||||
"url": "https://huggingface.co/city96/FLUX.1-schnell-gguf/resolve/main/flux1-schnell-Q4_1.gguf",
|
|
||||||
"size": "7.51GB"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "city96/flux1-schnell-Q4_K_S.gguf",
|
|
||||||
"type": "diffusion_model",
|
|
||||||
"base": "FLUX.1",
|
|
||||||
"save_path": "diffusion_models/FLUX1",
|
|
||||||
"description": "FLUX.1 [Dev] Diffusion model (Q4_K_S/.gguf)",
|
|
||||||
"reference": "https://huggingface.co/city96/FLUX.1-schnell-gguf",
|
|
||||||
"filename": "flux1-schnell-Q4_K_S.gguf",
|
|
||||||
"url": "https://huggingface.co/city96/FLUX.1-schnell-gguf/resolve/main/flux1-schnell-Q4_K_S.gguf",
|
|
||||||
"size": "6.78GB"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "city96/flux1-schnell-Q5_0.gguf",
|
|
||||||
"type": "diffusion_model",
|
|
||||||
"base": "FLUX.1",
|
|
||||||
"save_path": "diffusion_models/FLUX1",
|
|
||||||
"description": "FLUX.1 [Dev] Diffusion model (Q5_0/.gguf)",
|
|
||||||
"reference": "https://huggingface.co/city96/FLUX.1-schnell-gguf",
|
|
||||||
"filename": "flux1-schnell-Q5_0.gguf",
|
|
||||||
"url": "https://huggingface.co/city96/FLUX.1-schnell-gguf/resolve/main/flux1-schnell-Q5_0.gguf",
|
|
||||||
"size": "8.25GB"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "city96/flux1-schnell-Q5_1.gguf",
|
|
||||||
"type": "diffusion_model",
|
|
||||||
"base": "FLUX.1",
|
|
||||||
"save_path": "diffusion_models/FLUX1",
|
|
||||||
"description": "FLUX.1 [Dev] Diffusion model (Q5_1/.gguf)",
|
|
||||||
"reference": "https://huggingface.co/city96/FLUX.1-schnell-gguf",
|
|
||||||
"filename": "flux1-schnell-Q5_1.gguf",
|
|
||||||
"url": "https://huggingface.co/city96/FLUX.1-schnell-gguf/resolve/main/flux1-schnell-Q5_1.gguf",
|
|
||||||
"size": "8.99GB"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "city96/flux1-schnell-Q5_K_S.gguf",
|
|
||||||
"type": "diffusion_model",
|
|
||||||
"base": "FLUX.1",
|
|
||||||
"save_path": "diffusion_models/FLUX1",
|
|
||||||
"description": "FLUX.1 [Dev] Diffusion model (Q5_K_S/.gguf)",
|
|
||||||
"reference": "https://huggingface.co/city96/FLUX.1-schnell-gguf",
|
|
||||||
"filename": "flux1-schnell-Q5_K_S.gguf",
|
|
||||||
"url": "https://huggingface.co/city96/FLUX.1-schnell-gguf/resolve/main/flux1-schnell-Q5_K_S.gguf",
|
|
||||||
"size": "8.26GB"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "city96/flux1-schnell-Q6_K.gguf",
|
|
||||||
"type": "diffusion_model",
|
|
||||||
"base": "FLUX.1",
|
|
||||||
"save_path": "diffusion_models/FLUX1",
|
|
||||||
"description": "FLUX.1 [Dev] Diffusion model (Q6_K/.gguf)",
|
|
||||||
"reference": "https://huggingface.co/city96/FLUX.1-schnell-gguf",
|
|
||||||
"filename": "flux1-schnell-Q6_K.gguf",
|
|
||||||
"url": "https://huggingface.co/city96/FLUX.1-schnell-gguf/resolve/main/flux1-schnell-Q6_K.gguf",
|
|
||||||
"size": "9.83GB"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "city96/flux1-schnell-Q8_0.gguf",
|
|
||||||
"type": "diffusion_model",
|
|
||||||
"base": "FLUX.1",
|
|
||||||
"save_path": "diffusion_models/FLUX1",
|
|
||||||
"description": "FLUX.1 [Dev] Diffusion model (Q8_0/.gguf)",
|
|
||||||
"reference": "https://huggingface.co/city96/FLUX.1-schnell-gguf",
|
|
||||||
"filename": "flux1-schnell-Q8_0.gguf",
|
|
||||||
"url": "https://huggingface.co/city96/FLUX.1-schnell-gguf/resolve/main/flux1-schnell-Q8_0.gguf",
|
|
||||||
"size": "12.7GB"
|
|
||||||
},
|
|
||||||
|
|
||||||
{
|
|
||||||
"name": "TAEF1 Decoder",
|
|
||||||
"type": "TAESD",
|
|
||||||
"base": "FLUX.1",
|
|
||||||
"save_path": "vae_approx",
|
|
||||||
"description": "(FLUX.1 Verison) To view the preview in high quality while running samples in ComfyUI, you will need this model.",
|
|
||||||
"reference": "https://github.com/madebyollin/taesd",
|
|
||||||
"filename": "taef1_decoder.pth",
|
|
||||||
"url": "https://github.com/madebyollin/taesd/raw/main/taef1_decoder.pth",
|
|
||||||
"size": "4.71MB"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "TAEF1 Encoder",
|
|
||||||
"type": "TAESD",
|
|
||||||
"base": "FLUX.1",
|
|
||||||
"save_path": "vae_approx",
|
|
||||||
"description": "(FLUX.1 Verison) To view the preview in high quality while running samples in ComfyUI, you will need this model.",
|
|
||||||
"reference": "https://github.com/madebyollin/taesd",
|
|
||||||
"filename": "taef1_encoder.pth",
|
|
||||||
"url": "https://github.com/madebyollin/taesd/raw/main/taef1_encoder.pth",
|
|
||||||
"size": "4.71MB"
|
|
||||||
},
|
|
||||||
|
|
||||||
{
|
|
||||||
"name": "comfyanonymous/clip_l",
|
|
||||||
"type": "clip",
|
|
||||||
"base": "clip",
|
|
||||||
"save_path": "default",
|
|
||||||
"description": "clip_l model",
|
|
||||||
"reference": "https://huggingface.co/comfyanonymous/flux_text_encoders/tree/main",
|
|
||||||
"filename": "clip_l.safetensors",
|
|
||||||
"url": "https://huggingface.co/comfyanonymous/flux_text_encoders/resolve/main/clip_l.safetensors",
|
|
||||||
"size": "246MB"
|
|
||||||
},
|
|
||||||
|
|
||||||
{
|
|
||||||
"name": "Comfy Org/FLUX.1 [dev] Checkpoint model (fp8)",
|
|
||||||
"type": "checkpoint",
|
|
||||||
"base": "FLUX.1",
|
|
||||||
"save_path": "checkpoints/FLUX1",
|
|
||||||
"description": "FLUX.1 [dev] Checkpoint model (fp8)",
|
|
||||||
"reference": "https://huggingface.co/Comfy-Org/flux1-dev/tree/main",
|
|
||||||
"filename": "flux1-dev-fp8.safetensors",
|
|
||||||
"url": "https://huggingface.co/Comfy-Org/flux1-dev/resolve/main/flux1-dev-fp8.safetensors",
|
|
||||||
"size": "17.2GB"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "Comfy Org/FLUX.1 [schnell] Checkpoint model (fp8)",
|
|
||||||
"type": "checkpoint",
|
|
||||||
"base": "FLUX.1",
|
|
||||||
"save_path": "checkpoints/FLUX1",
|
|
||||||
"description": "FLUX.1 [schnell] Checkpoint model (fp8)",
|
|
||||||
"reference": "https://huggingface.co/Comfy-Org/flux1-dev/tree/main",
|
|
||||||
"filename": "flux1-schnell-fp8.safetensors",
|
|
||||||
"url": "https://huggingface.co/Comfy-Org/flux1-schnell/resolve/main/flux1-schnell-fp8.safetensors",
|
|
||||||
"size": "17.2GB"
|
|
||||||
},
|
|
||||||
|
|
||||||
{
|
|
||||||
"name": "google-t5/t5-v1_1-xxl_encoderonly-fp16",
|
|
||||||
"type": "clip",
|
|
||||||
"base": "t5",
|
|
||||||
"save_path": "clip/t5",
|
|
||||||
"description": "The encoder part of https://huggingface.co/google/t5-v1_1-xxl, used with SD3 and Flux1",
|
|
||||||
"reference": "https://huggingface.co/mcmonkey/google_t5-v1_1-xxl_encoderonly",
|
|
||||||
"filename": "google_t5-v1_1-xxl_encoderonly-fp16.safetensors",
|
|
||||||
"url": "https://huggingface.co/mcmonkey/google_t5-v1_1-xxl_encoderonly/resolve/main/pytorch_model.safetensors",
|
|
||||||
"size": "10.1GB"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "google-t5/t5-v1_1-xxl_encoderonly-fp8_e4m3fn",
|
|
||||||
"type": "clip",
|
|
||||||
"base": "t5",
|
|
||||||
"save_path": "clip/t5",
|
|
||||||
"description": "The encoder part of https://huggingface.co/google/t5-v1_1-xxl, used with SD3 and Flux1",
|
|
||||||
"reference": "https://huggingface.co/mcmonkey/google_t5-v1_1-xxl_encoderonly",
|
|
||||||
"filename": "google_t5-v1_1-xxl_encoderonly-fp8_e4m3fn.safetensors",
|
|
||||||
"url": "https://huggingface.co/mcmonkey/google_t5-v1_1-xxl_encoderonly/resolve/main/t5xxl_fp8_e4m3fn.safetensors",
|
|
||||||
"size": "4.89GB"
|
|
||||||
},
|
|
||||||
|
|
||||||
{
|
|
||||||
"name": "FLUX.1 [schnell] Diffusion model",
|
|
||||||
"type": "unet",
|
|
||||||
"base": "FLUX.1",
|
|
||||||
"save_path": "unet/FLUX1",
|
|
||||||
"description": "FLUX.1 [Schnell] Diffusion model (a.k.a. FLUX.1 turbo model)[w/Due to the large size of the model, it is recommended to download it through a browser if possible.]",
|
|
||||||
"reference": "https://huggingface.co/black-forest-labs/FLUX.1-schnell",
|
|
||||||
"filename": "flux1-schnell.safetensors",
|
|
||||||
"url": "https://huggingface.co/black-forest-labs/FLUX.1-schnell/resolve/main/flux1-schnell.safetensors",
|
|
||||||
"size": "23.8GB"
|
|
||||||
},
|
|
||||||
|
|
||||||
{
|
|
||||||
"name": "FLUX.1 VAE model",
|
|
||||||
"type": "VAE",
|
|
||||||
"base": "FLUX.1",
|
|
||||||
"save_path": "vae/FLUX1",
|
|
||||||
"description": "FLUX.1 VAE model",
|
|
||||||
"reference": "https://huggingface.co/black-forest-labs/FLUX.1-schnell",
|
|
||||||
"filename": "ae.sft",
|
|
||||||
"url": "https://huggingface.co/black-forest-labs/FLUX.1-schnell/resolve/main/ae.safetensors",
|
|
||||||
"size": "335MB"
|
|
||||||
},
|
|
||||||
|
|
||||||
{
|
|
||||||
"name": "kijai/FLUX.1 [schnell] Diffusion model (float8_e4m3fn)",
|
|
||||||
"type": "unet",
|
|
||||||
"base": "FLUX.1",
|
|
||||||
"save_path": "unet/FLUX1",
|
|
||||||
"description": "FLUX.1 [Schnell] Diffusion model (float8_e4m3fn)",
|
|
||||||
"reference": "https://huggingface.co/Kijai/flux-fp8",
|
|
||||||
"filename": "flux1-schnell-fp8.safetensors",
|
|
||||||
"url": "https://huggingface.co/Kijai/flux-fp8/resolve/main/flux1-schnell-fp8.safetensors",
|
|
||||||
"size": "11.9GB"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "kijai/FLUX.1 [dev] Diffusion model (float8_e4m3fn)",
|
|
||||||
"type": "unet",
|
|
||||||
"base": "FLUX.1",
|
|
||||||
"save_path": "unet/FLUX1",
|
|
||||||
"description": "FLUX.1 [dev] Diffusion model (float8_e4m3fn)",
|
|
||||||
"reference": "https://huggingface.co/Kijai/flux-fp8",
|
|
||||||
"filename": "flux1-dev-fp8.safetensors",
|
|
||||||
"url": "https://huggingface.co/Kijai/flux-fp8/resolve/main/flux1-dev-fp8.safetensors",
|
|
||||||
"size": "11.9GB"
|
|
||||||
},
|
|
||||||
|
|
||||||
{
|
|
||||||
"name": "photomaker-v2.bin",
|
|
||||||
"type": "photomaker",
|
|
||||||
"base": "SDXL",
|
|
||||||
"save_path": "photomaker",
|
|
||||||
"description": "PhotoMaker model. This model is compatible with SDXL.",
|
|
||||||
"reference": "https://huggingface.co/TencentARC/PhotoMaker-V2",
|
|
||||||
"filename": "photomaker-v2.bin",
|
|
||||||
"url": "https://huggingface.co/TencentARC/PhotoMaker-V2/resolve/main/photomaker-v2.bin",
|
|
||||||
"size": "1.8GB"
|
|
||||||
},
|
|
||||||
|
|
||||||
{
|
|
||||||
"name": "hunyuan_dit_1.2.safetensors",
|
|
||||||
"type": "checkpoint",
|
|
||||||
"base": "Hunyuan-DiT",
|
|
||||||
"save_path": "checkpoints/hunyuan_dit_comfyui",
|
|
||||||
"description": "Different versions of HunyuanDIT packaged for ComfyUI use.",
|
|
||||||
"reference": "https://huggingface.co/comfyanonymous/hunyuan_dit_comfyui",
|
|
||||||
"filename": "hunyuan_dit_1.2.safetensors",
|
|
||||||
"url": "https://huggingface.co/comfyanonymous/hunyuan_dit_comfyui/resolve/main/hunyuan_dit_1.2.safetensors",
|
|
||||||
"size": "8.24GB"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "hunyuan_dit_1.1.safetensors",
|
|
||||||
"type": "checkpoint",
|
|
||||||
"base": "Hunyuan-DiT",
|
|
||||||
"save_path": "checkpoints/hunyuan_dit_comfyui",
|
|
||||||
"description": "Different versions of HunyuanDIT packaged for ComfyUI use.",
|
|
||||||
"reference": "https://huggingface.co/comfyanonymous/hunyuan_dit_comfyui",
|
|
||||||
"filename": "hunyuan_dit_1.1.safetensors",
|
|
||||||
"url": "https://huggingface.co/comfyanonymous/hunyuan_dit_comfyui/resolve/main/hunyuan_dit_1.1.safetensors",
|
|
||||||
"size": "8.24GB"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "hunyuan_dit_1.0.safetensors",
|
|
||||||
"type": "checkpoint",
|
|
||||||
"base": "Hunyuan-DiT",
|
|
||||||
"save_path": "checkpoints/hunyuan_dit_comfyui",
|
|
||||||
"description": "Different versions of HunyuanDIT packaged for ComfyUI use.",
|
|
||||||
"reference": "https://huggingface.co/comfyanonymous/hunyuan_dit_comfyui",
|
|
||||||
"filename": "hunyuan_dit_1.0.safetensors",
|
|
||||||
"url": "https://huggingface.co/comfyanonymous/hunyuan_dit_comfyui/resolve/main/hunyuan_dit_1.0.safetensors",
|
|
||||||
"size": "8.24GB"
|
|
||||||
},
|
|
||||||
|
|
||||||
{
|
|
||||||
"name": "xinsir/ControlNet++: All-in-one ControlNet (ProMax model)",
|
|
||||||
"type": "controlnet",
|
|
||||||
"base": "SDXL",
|
|
||||||
"save_path": "controlnet/SDXL/controlnet-union-sdxl-1.0",
|
|
||||||
"description": "All-in-one ControlNet for image generations and editing! (ProMax model)",
|
|
||||||
"reference": "https://huggingface.co/xinsir/controlnet-union-sdxl-1.0",
|
|
||||||
"filename": "diffusion_pytorch_model_promax.safetensors",
|
|
||||||
"url": "https://huggingface.co/xinsir/controlnet-union-sdxl-1.0/resolve/main/diffusion_pytorch_model_promax.safetensors",
|
|
||||||
"size": "2.50GB"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "xinsir/ControlNet++: All-in-one ControlNet",
|
|
||||||
"type": "controlnet",
|
|
||||||
"base": "SDXL",
|
|
||||||
"save_path": "controlnet/SDXL/controlnet-union-sdxl-1.0",
|
|
||||||
"description": "All-in-one ControlNet for image generations and editing!",
|
|
||||||
"reference": "https://huggingface.co/xinsir/controlnet-union-sdxl-1.0",
|
|
||||||
"filename": "diffusion_pytorch_model.safetensors",
|
|
||||||
"url": "https://huggingface.co/xinsir/controlnet-union-sdxl-1.0/resolve/main/diffusion_pytorch_model.safetensors",
|
|
||||||
"size": "2.50GB"
|
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -90,16 +90,6 @@
|
|||||||
"install_type": "git-clone",
|
"install_type": "git-clone",
|
||||||
"description": "Nodes:WW_ImageResize"
|
"description": "Nodes:WW_ImageResize"
|
||||||
},
|
},
|
||||||
{
|
|
||||||
"author": "bmz55",
|
|
||||||
"title": "bmz nodes",
|
|
||||||
"reference": "https://github.com/bmz55/comfyui-bmz-nodes",
|
|
||||||
"files": [
|
|
||||||
"https://github.com/bmz55/comfyui-bmz-nodes"
|
|
||||||
],
|
|
||||||
"install_type": "git-clone",
|
|
||||||
"description": "Nodes:Load Images From Dir With Name (Inspire - BMZ), Count Images In Dir (BMZ), Get Level Text (BMZ), Get Level Float (BMZ)"
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
"author": "azure-dragon-ai",
|
"author": "azure-dragon-ai",
|
||||||
"title": "ComfyUI-HPSv2-Nodes",
|
"title": "ComfyUI-HPSv2-Nodes",
|
||||||
@@ -150,36 +140,6 @@
|
|||||||
"install_type": "git-clone",
|
"install_type": "git-clone",
|
||||||
"description": "This module provides an annotation @ComfyFunc to streamline adding custom node types in ComfyUI. It processes your function's signature to create a wrapped function and custom node definition required for ComfyUI, eliminating all the boilerplate code. In most cases you can just add a @ComfyFunc(\"category\") annotation to your existing function."
|
"description": "This module provides an annotation @ComfyFunc to streamline adding custom node types in ComfyUI. It processes your function's signature to create a wrapped function and custom node definition required for ComfyUI, eliminating all the boilerplate code. In most cases you can just add a @ComfyFunc(\"category\") annotation to your existing function."
|
||||||
},
|
},
|
||||||
{
|
|
||||||
"author": "MokkaBoss1",
|
|
||||||
"title": "Woman_in_a_dress",
|
|
||||||
"reference": "https://github.com/MokkaBoss1/Woman_in_a_dress",
|
|
||||||
"files": [
|
|
||||||
"https://github.com/MokkaBoss1/Woman_in_a_dress/raw/main/Woman_In_A_Dress.py"
|
|
||||||
],
|
|
||||||
"install_type": "git-clone",
|
|
||||||
"description": "Nodes:Woman_in_a_dress"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"author": "shinich39",
|
|
||||||
"title": "comfyui-concat-text-39",
|
|
||||||
"reference": "https://github.com/shinich39/comfyui-concat-text-39",
|
|
||||||
"files": [
|
|
||||||
"https://github.com/shinich39/comfyui-concat-text-39"
|
|
||||||
],
|
|
||||||
"install_type": "git-clone",
|
|
||||||
"description": "Nodes:Concatenate multiple text nodes."
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"author": "nilor-corp",
|
|
||||||
"title": "nilor-nodes",
|
|
||||||
"reference": "https://github.com/nilor-corp/nilor-nodes",
|
|
||||||
"files": [
|
|
||||||
"https://github.com/nilor-corp/nilor-nodes"
|
|
||||||
],
|
|
||||||
"install_type": "git-clone",
|
|
||||||
"description": "Nodes:Nilor Floats, Nilor Int To List Of Bools, Nilor Bool From List Of Bools, Nilor Int From List Of Ints, Nilor List of Ints, Nilor Count Images In Directory"
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
"author": "OuticNZ",
|
"author": "OuticNZ",
|
||||||
"title": "ComfyUI-Simple-Of-Complex",
|
"title": "ComfyUI-Simple-Of-Complex",
|
||||||
@@ -260,6 +220,26 @@
|
|||||||
],
|
],
|
||||||
"install_type": "git-clone",
|
"install_type": "git-clone",
|
||||||
"description": "The primitive node and dummy input are required because comfy doesn't accept requests with identical graphs. You'll also need a show text node. I like the one from ComfyUI-Custom-Scripts. I got the generic tetris remake from claude so it may or may not be ripped from somewhere else."
|
"description": "The primitive node and dummy input are required because comfy doesn't accept requests with identical graphs. You'll also need a show text node. I like the one from ComfyUI-Custom-Scripts. I got the generic tetris remake from claude so it may or may not be ripped from somewhere else."
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"author": "FlyMyAI",
|
||||||
|
"title": "ComfyUI-ExampleNode",
|
||||||
|
"reference": "https://github.com/FlyMyAI/ComfyUI-ExampleNode",
|
||||||
|
"files": [
|
||||||
|
"https://github.com/FlyMyAI/ComfyUI-ExampleNode"
|
||||||
|
],
|
||||||
|
"install_type": "git-clone",
|
||||||
|
"description": "Node to provide convenient ComfyUI standard, supported by flymy_comfyui."
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"author": "Wanghanying",
|
||||||
|
"title": "ComfyUI_RAGDemo",
|
||||||
|
"reference": "https://github.com/Wanghanying/ComfyUI_RAGDemo",
|
||||||
|
"files": [
|
||||||
|
"https://github.com/Wanghanying/ComfyUI_RAGDemo"
|
||||||
|
],
|
||||||
|
"install_type": "git-clone",
|
||||||
|
"description": "RAG Demo for LLM"
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
@@ -1 +1,546 @@
|
|||||||
{}
|
{
|
||||||
|
"https://github.com/BadCafeCode/execution-inversion-demo-comfyui": [
|
||||||
|
[
|
||||||
|
"AccumulateNode",
|
||||||
|
"AccumulationGetItemNode",
|
||||||
|
"AccumulationGetLengthNode",
|
||||||
|
"AccumulationHeadNode",
|
||||||
|
"AccumulationSetItemNode",
|
||||||
|
"AccumulationTailNode",
|
||||||
|
"AccumulationToListNode",
|
||||||
|
"BoolOperationNode",
|
||||||
|
"ComponentInput",
|
||||||
|
"ComponentMetadata",
|
||||||
|
"ComponentOutput",
|
||||||
|
"DebugPrint",
|
||||||
|
"ExecutionBlocker",
|
||||||
|
"FloatConditions",
|
||||||
|
"ForLoopClose",
|
||||||
|
"ForLoopOpen",
|
||||||
|
"IntConditions",
|
||||||
|
"IntMathOperation",
|
||||||
|
"InversionDemoAdvancedPromptNode",
|
||||||
|
"InversionDemoLazyConditional",
|
||||||
|
"InversionDemoLazyIndexSwitch",
|
||||||
|
"InversionDemoLazyMixImages",
|
||||||
|
"InversionDemoLazySwitch",
|
||||||
|
"ListToAccumulationNode",
|
||||||
|
"MakeListNode",
|
||||||
|
"StringConditions",
|
||||||
|
"ToBoolNode",
|
||||||
|
"WhileLoopClose",
|
||||||
|
"WhileLoopOpen"
|
||||||
|
],
|
||||||
|
{
|
||||||
|
"title_aux": "execution-inversion-demo-comfyui"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"https://github.com/BetaDoggo/ComfyUI-Tetris": [
|
||||||
|
[
|
||||||
|
"Tetris"
|
||||||
|
],
|
||||||
|
{
|
||||||
|
"title_aux": "ComfyUI Tetris"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"https://github.com/BoosterCore/ComfyUI-BC-Experimental": [
|
||||||
|
[
|
||||||
|
"ClipTextEncodeBC",
|
||||||
|
"ClipTextEncodeBCA",
|
||||||
|
"FluxEmptyLatentSize",
|
||||||
|
"LoraWithTriggerWord",
|
||||||
|
"SaveAnyText",
|
||||||
|
"SimpleText"
|
||||||
|
],
|
||||||
|
{
|
||||||
|
"title_aux": "ComfyUI-BC-Experimental"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"https://github.com/FlyMyAI/ComfyUI-ExampleNode": [
|
||||||
|
[
|
||||||
|
"ExampleT2IFMANode"
|
||||||
|
],
|
||||||
|
{
|
||||||
|
"title_aux": "ComfyUI-ExampleNode"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"https://github.com/IvanRybakov/comfyui-node-int-to-string-convertor": [
|
||||||
|
[
|
||||||
|
"Int To String"
|
||||||
|
],
|
||||||
|
{
|
||||||
|
"title_aux": "comfyui-node-int-to-string-convertor"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"https://github.com/LarryJane491/Custom-Node-Base": [
|
||||||
|
[
|
||||||
|
"My First Node"
|
||||||
|
],
|
||||||
|
{
|
||||||
|
"title_aux": "Custom-Node-Base"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"https://github.com/OuticNZ/ComfyUI-Simple-Of-Complex": [
|
||||||
|
[
|
||||||
|
"Pipe From Parameters",
|
||||||
|
"Pipe To Parameters",
|
||||||
|
"Prompt Tidy",
|
||||||
|
"Text Switch 2 Way",
|
||||||
|
"Text With Context"
|
||||||
|
],
|
||||||
|
{
|
||||||
|
"title_aux": "ComfyUI-Simple-Of-Complex"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"https://github.com/Suzie1/ComfyUI_Guide_To_Making_Custom_Nodes": [
|
||||||
|
[
|
||||||
|
"Concatenate Hello World",
|
||||||
|
"Hello World Overlay Text",
|
||||||
|
"Print Hello World"
|
||||||
|
],
|
||||||
|
{
|
||||||
|
"title_aux": "Guide To Making Custom Nodes in ComfyUI"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"https://github.com/Wanghanying/ComfyUI_RAGDemo": [
|
||||||
|
[
|
||||||
|
"testRAG"
|
||||||
|
],
|
||||||
|
{
|
||||||
|
"title_aux": "ComfyUI_RAGDemo"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"https://github.com/azure-dragon-ai/ComfyUI-HPSv2-Nodes": [
|
||||||
|
[
|
||||||
|
"GetImageSize",
|
||||||
|
"HaojihuiHPSv2ImageProcessor",
|
||||||
|
"HaojihuiHPSv2ImageScore",
|
||||||
|
"HaojihuiHPSv2ImageScores",
|
||||||
|
"HaojihuiHPSv2Loader",
|
||||||
|
"HaojihuiHPSv2SaveAnimatedWEBP",
|
||||||
|
"HaojihuiHPSv2SaveImage",
|
||||||
|
"HaojihuiHPSv2SaveWEBP",
|
||||||
|
"HaojihuiHPSv2SaveWebpImage",
|
||||||
|
"HaojihuiHPSv2TextProcessor",
|
||||||
|
"SaveImageWebp",
|
||||||
|
"ScaleShort"
|
||||||
|
],
|
||||||
|
{
|
||||||
|
"title_aux": "ComfyUI-HPSv2-Nodes"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"https://github.com/bamboodia/BAM_Nodes": [
|
||||||
|
[
|
||||||
|
"BAM Crop To Ratio",
|
||||||
|
"BAM Empty Latent By Ratio",
|
||||||
|
"BAM Get Shortest Side",
|
||||||
|
"BAM OnOff INT",
|
||||||
|
"BAM Random Float",
|
||||||
|
"BAM Random Image From Folder"
|
||||||
|
],
|
||||||
|
{
|
||||||
|
"title_aux": "BAM Nodes"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"https://github.com/boricuapab/ComfyUI_BoricuapabWFNodePack": [
|
||||||
|
[
|
||||||
|
"BoricuapabWF Concatenate Hello World",
|
||||||
|
"BoricuapabWF Integer",
|
||||||
|
"BoricuapabWF Print Hello Puerto Rican World",
|
||||||
|
"BoricuapabWF Print Puerto Rican"
|
||||||
|
],
|
||||||
|
{
|
||||||
|
"title_aux": "ComfyUI_BoricuapabWFNodePack"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"https://github.com/comfyanonymous/ComfyUI": [
|
||||||
|
[
|
||||||
|
"AddNoise",
|
||||||
|
"AlignYourStepsScheduler",
|
||||||
|
"BasicGuider",
|
||||||
|
"BasicScheduler",
|
||||||
|
"BetaSamplingScheduler",
|
||||||
|
"CFGGuider",
|
||||||
|
"CLIPAttentionMultiply",
|
||||||
|
"CLIPLoader",
|
||||||
|
"CLIPMergeAdd",
|
||||||
|
"CLIPMergeSimple",
|
||||||
|
"CLIPMergeSubtract",
|
||||||
|
"CLIPSave",
|
||||||
|
"CLIPSetLastLayer",
|
||||||
|
"CLIPTextEncode",
|
||||||
|
"CLIPTextEncodeControlnet",
|
||||||
|
"CLIPTextEncodeFlux",
|
||||||
|
"CLIPTextEncodeHunyuanDiT",
|
||||||
|
"CLIPTextEncodeSD3",
|
||||||
|
"CLIPTextEncodeSDXL",
|
||||||
|
"CLIPTextEncodeSDXLRefiner",
|
||||||
|
"CLIPVisionEncode",
|
||||||
|
"CLIPVisionLoader",
|
||||||
|
"Canny",
|
||||||
|
"CheckpointLoader",
|
||||||
|
"CheckpointLoaderSimple",
|
||||||
|
"CheckpointSave",
|
||||||
|
"ConditioningAverage",
|
||||||
|
"ConditioningCombine",
|
||||||
|
"ConditioningConcat",
|
||||||
|
"ConditioningSetArea",
|
||||||
|
"ConditioningSetAreaPercentage",
|
||||||
|
"ConditioningSetAreaStrength",
|
||||||
|
"ConditioningSetMask",
|
||||||
|
"ConditioningSetTimestepRange",
|
||||||
|
"ConditioningZeroOut",
|
||||||
|
"ControlNetApply",
|
||||||
|
"ControlNetApplyAdvanced",
|
||||||
|
"ControlNetApplySD3",
|
||||||
|
"ControlNetInpaintingAliMamaApply",
|
||||||
|
"ControlNetLoader",
|
||||||
|
"CropMask",
|
||||||
|
"DiffControlNetLoader",
|
||||||
|
"DifferentialDiffusion",
|
||||||
|
"DiffusersLoader",
|
||||||
|
"DisableNoise",
|
||||||
|
"DualCFGGuider",
|
||||||
|
"DualCLIPLoader",
|
||||||
|
"EmptyImage",
|
||||||
|
"EmptyLatentAudio",
|
||||||
|
"EmptyLatentImage",
|
||||||
|
"EmptyMochiLatentVideo",
|
||||||
|
"EmptySD3LatentImage",
|
||||||
|
"ExponentialScheduler",
|
||||||
|
"FeatherMask",
|
||||||
|
"FlipSigmas",
|
||||||
|
"FluxGuidance",
|
||||||
|
"FreeU",
|
||||||
|
"FreeU_V2",
|
||||||
|
"GITSScheduler",
|
||||||
|
"GLIGENLoader",
|
||||||
|
"GLIGENTextBoxApply",
|
||||||
|
"GrowMask",
|
||||||
|
"HyperTile",
|
||||||
|
"HypernetworkLoader",
|
||||||
|
"ImageBatch",
|
||||||
|
"ImageBlend",
|
||||||
|
"ImageBlur",
|
||||||
|
"ImageColorToMask",
|
||||||
|
"ImageCompositeMasked",
|
||||||
|
"ImageCrop",
|
||||||
|
"ImageFromBatch",
|
||||||
|
"ImageInvert",
|
||||||
|
"ImageOnlyCheckpointLoader",
|
||||||
|
"ImageOnlyCheckpointSave",
|
||||||
|
"ImagePadForOutpaint",
|
||||||
|
"ImageQuantize",
|
||||||
|
"ImageScale",
|
||||||
|
"ImageScaleBy",
|
||||||
|
"ImageScaleToTotalPixels",
|
||||||
|
"ImageSharpen",
|
||||||
|
"ImageToMask",
|
||||||
|
"ImageUpscaleWithModel",
|
||||||
|
"InpaintModelConditioning",
|
||||||
|
"InstructPixToPixConditioning",
|
||||||
|
"InvertMask",
|
||||||
|
"JoinImageWithAlpha",
|
||||||
|
"KSampler",
|
||||||
|
"KSamplerAdvanced",
|
||||||
|
"KSamplerSelect",
|
||||||
|
"KarrasScheduler",
|
||||||
|
"LaplaceScheduler",
|
||||||
|
"LatentAdd",
|
||||||
|
"LatentApplyOperation",
|
||||||
|
"LatentApplyOperationCFG",
|
||||||
|
"LatentBatch",
|
||||||
|
"LatentBatchSeedBehavior",
|
||||||
|
"LatentBlend",
|
||||||
|
"LatentComposite",
|
||||||
|
"LatentCompositeMasked",
|
||||||
|
"LatentCrop",
|
||||||
|
"LatentFlip",
|
||||||
|
"LatentFromBatch",
|
||||||
|
"LatentInterpolate",
|
||||||
|
"LatentMultiply",
|
||||||
|
"LatentOperationSharpen",
|
||||||
|
"LatentOperationTonemapReinhard",
|
||||||
|
"LatentRotate",
|
||||||
|
"LatentSubtract",
|
||||||
|
"LatentUpscale",
|
||||||
|
"LatentUpscaleBy",
|
||||||
|
"LoadAudio",
|
||||||
|
"LoadImage",
|
||||||
|
"LoadImageMask",
|
||||||
|
"LoadLatent",
|
||||||
|
"LoraLoader",
|
||||||
|
"LoraLoaderModelOnly",
|
||||||
|
"LoraSave",
|
||||||
|
"MaskComposite",
|
||||||
|
"MaskToImage",
|
||||||
|
"ModelMergeAdd",
|
||||||
|
"ModelMergeBlocks",
|
||||||
|
"ModelMergeFlux1",
|
||||||
|
"ModelMergeSD1",
|
||||||
|
"ModelMergeSD2",
|
||||||
|
"ModelMergeSD35_Large",
|
||||||
|
"ModelMergeSD3_2B",
|
||||||
|
"ModelMergeSDXL",
|
||||||
|
"ModelMergeSimple",
|
||||||
|
"ModelMergeSubtract",
|
||||||
|
"ModelSamplingAuraFlow",
|
||||||
|
"ModelSamplingContinuousEDM",
|
||||||
|
"ModelSamplingContinuousV",
|
||||||
|
"ModelSamplingDiscrete",
|
||||||
|
"ModelSamplingFlux",
|
||||||
|
"ModelSamplingSD3",
|
||||||
|
"ModelSamplingStableCascade",
|
||||||
|
"ModelSave",
|
||||||
|
"Morphology",
|
||||||
|
"PatchModelAddDownscale",
|
||||||
|
"PerpNeg",
|
||||||
|
"PerpNegGuider",
|
||||||
|
"PerturbedAttentionGuidance",
|
||||||
|
"PhotoMakerEncode",
|
||||||
|
"PhotoMakerLoader",
|
||||||
|
"PolyexponentialScheduler",
|
||||||
|
"PorterDuffImageComposite",
|
||||||
|
"PreviewAudio",
|
||||||
|
"PreviewImage",
|
||||||
|
"RandomNoise",
|
||||||
|
"RebatchImages",
|
||||||
|
"RebatchLatents",
|
||||||
|
"RepeatImageBatch",
|
||||||
|
"RepeatLatentBatch",
|
||||||
|
"RescaleCFG",
|
||||||
|
"SDTurboScheduler",
|
||||||
|
"SD_4XUpscale_Conditioning",
|
||||||
|
"SV3D_Conditioning",
|
||||||
|
"SVD_img2vid_Conditioning",
|
||||||
|
"SamplerCustom",
|
||||||
|
"SamplerCustomAdvanced",
|
||||||
|
"SamplerDPMAdaptative",
|
||||||
|
"SamplerDPMPP_2M_SDE",
|
||||||
|
"SamplerDPMPP_2S_Ancestral",
|
||||||
|
"SamplerDPMPP_3M_SDE",
|
||||||
|
"SamplerDPMPP_SDE",
|
||||||
|
"SamplerEulerAncestral",
|
||||||
|
"SamplerEulerAncestralCFGPP",
|
||||||
|
"SamplerEulerCFGpp",
|
||||||
|
"SamplerLCMUpscale",
|
||||||
|
"SamplerLMS",
|
||||||
|
"SaveAnimatedPNG",
|
||||||
|
"SaveAnimatedWEBP",
|
||||||
|
"SaveAudio",
|
||||||
|
"SaveImage",
|
||||||
|
"SaveImageWebsocket",
|
||||||
|
"SaveLatent",
|
||||||
|
"SelfAttentionGuidance",
|
||||||
|
"SetLatentNoiseMask",
|
||||||
|
"SetUnionControlNetType",
|
||||||
|
"SkipLayerGuidanceSD3",
|
||||||
|
"SolidMask",
|
||||||
|
"SplitImageWithAlpha",
|
||||||
|
"SplitSigmas",
|
||||||
|
"SplitSigmasDenoise",
|
||||||
|
"StableCascade_EmptyLatentImage",
|
||||||
|
"StableCascade_StageB_Conditioning",
|
||||||
|
"StableCascade_StageC_VAEEncode",
|
||||||
|
"StableCascade_SuperResolutionControlnet",
|
||||||
|
"StableZero123_Conditioning",
|
||||||
|
"StableZero123_Conditioning_Batched",
|
||||||
|
"StubConstantImage",
|
||||||
|
"StubFloat",
|
||||||
|
"StubImage",
|
||||||
|
"StubInt",
|
||||||
|
"StubMask",
|
||||||
|
"StyleModelApply",
|
||||||
|
"StyleModelLoader",
|
||||||
|
"TestAccumulateNode",
|
||||||
|
"TestAccumulationGetItemNode",
|
||||||
|
"TestAccumulationGetLengthNode",
|
||||||
|
"TestAccumulationHeadNode",
|
||||||
|
"TestAccumulationSetItemNode",
|
||||||
|
"TestAccumulationTailNode",
|
||||||
|
"TestAccumulationToListNode",
|
||||||
|
"TestBoolOperationNode",
|
||||||
|
"TestCustomIsChanged",
|
||||||
|
"TestCustomValidation1",
|
||||||
|
"TestCustomValidation2",
|
||||||
|
"TestCustomValidation3",
|
||||||
|
"TestCustomValidation4",
|
||||||
|
"TestCustomValidation5",
|
||||||
|
"TestDynamicDependencyCycle",
|
||||||
|
"TestExecutionBlocker",
|
||||||
|
"TestFloatConditions",
|
||||||
|
"TestForLoopClose",
|
||||||
|
"TestForLoopOpen",
|
||||||
|
"TestIntConditions",
|
||||||
|
"TestIntMathOperation",
|
||||||
|
"TestIsChangedWithConstants",
|
||||||
|
"TestLazyMixImages",
|
||||||
|
"TestListToAccumulationNode",
|
||||||
|
"TestMakeListNode",
|
||||||
|
"TestMixedExpansionReturns",
|
||||||
|
"TestStringConditions",
|
||||||
|
"TestToBoolNode",
|
||||||
|
"TestVariadicAverage",
|
||||||
|
"TestWhileLoopClose",
|
||||||
|
"TestWhileLoopOpen",
|
||||||
|
"ThresholdMask",
|
||||||
|
"TomePatchModel",
|
||||||
|
"TorchCompileModel",
|
||||||
|
"TripleCLIPLoader",
|
||||||
|
"UNETLoader",
|
||||||
|
"UNetCrossAttentionMultiply",
|
||||||
|
"UNetSelfAttentionMultiply",
|
||||||
|
"UNetTemporalAttentionMultiply",
|
||||||
|
"UpscaleModelLoader",
|
||||||
|
"VAEDecode",
|
||||||
|
"VAEDecodeAudio",
|
||||||
|
"VAEDecodeTiled",
|
||||||
|
"VAEEncode",
|
||||||
|
"VAEEncodeAudio",
|
||||||
|
"VAEEncodeForInpaint",
|
||||||
|
"VAEEncodeTiled",
|
||||||
|
"VAELoader",
|
||||||
|
"VAESave",
|
||||||
|
"VPScheduler",
|
||||||
|
"VideoLinearCFGGuidance",
|
||||||
|
"VideoTriangleCFGGuidance",
|
||||||
|
"WebcamCapture",
|
||||||
|
"unCLIPCheckpointLoader",
|
||||||
|
"unCLIPConditioning"
|
||||||
|
],
|
||||||
|
{
|
||||||
|
"title_aux": "ComfyUI"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"https://github.com/dynamixar/Atluris": [
|
||||||
|
[
|
||||||
|
"RandomLine"
|
||||||
|
],
|
||||||
|
{
|
||||||
|
"title_aux": "Atluris"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"https://github.com/ecjojo/ecjojo-example-nodes": [
|
||||||
|
[
|
||||||
|
"BiggerNote_Example",
|
||||||
|
"DisplayTextNode_Example",
|
||||||
|
"EmptyNode_Example",
|
||||||
|
"ExampleNode_Example",
|
||||||
|
"FilePrefixNode_Example",
|
||||||
|
"HelloWorldNode_Example",
|
||||||
|
"RandomSizeNode_Example",
|
||||||
|
"StringNode_Example",
|
||||||
|
"TextOverlayNode_Example"
|
||||||
|
],
|
||||||
|
{
|
||||||
|
"title_aux": "ecjojo_example_nodes"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"https://github.com/et118/ComfyUI-ElGogh-Nodes": [
|
||||||
|
[
|
||||||
|
"ElGoghCLIPSetLastLayer",
|
||||||
|
"ElGoghCheckpointLoaderSimple",
|
||||||
|
"ElGoghEmptyLatentImage",
|
||||||
|
"ElGoghKSamplerAdvanced",
|
||||||
|
"ElGoghNegativePrompt",
|
||||||
|
"ElGoghPositivePrompt",
|
||||||
|
"ElGoghPrimaryLoraLoader",
|
||||||
|
"ElGoghSecondaryLoraLoader",
|
||||||
|
"ElGoghSendWebsocketNSFWBool",
|
||||||
|
"ElGoghTertiaryLoraLoader",
|
||||||
|
"ElGoghVAELoader"
|
||||||
|
],
|
||||||
|
{
|
||||||
|
"title_aux": "ComfyUI-ElGogh-Nodes"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"https://github.com/foxtrot-roger/comfyui-custom-nodes": [
|
||||||
|
[
|
||||||
|
"RF_Tutorial"
|
||||||
|
],
|
||||||
|
{
|
||||||
|
"title_aux": "comfyui-custom-nodes"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"https://github.com/jtong/comfyui-jtong-workflow": [
|
||||||
|
[
|
||||||
|
"Example",
|
||||||
|
"high_workflow_caller",
|
||||||
|
"jtong.Highend",
|
||||||
|
"jtong.Highway"
|
||||||
|
],
|
||||||
|
{
|
||||||
|
"author": "Trung0246",
|
||||||
|
"description": "Random nodes for ComfyUI I made to solve my struggle with ComfyUI (ex: pipe, process). Have varying quality.",
|
||||||
|
"nickname": "ComfyUI-0246",
|
||||||
|
"title": "ComfyUI-0246",
|
||||||
|
"title_aux": "comfyui-jtong-workflow"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"https://github.com/kappa54m/ComfyUI_Usability": [
|
||||||
|
[
|
||||||
|
"KLoadImageByPath",
|
||||||
|
"KLoadImageByPathAdvanced",
|
||||||
|
"KLoadImageDedup"
|
||||||
|
],
|
||||||
|
{
|
||||||
|
"title_aux": "ComfyUI-HPSv2-Nodes"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"https://github.com/mira-6/mira-wildcard-node": [
|
||||||
|
[
|
||||||
|
"MiraWildcard"
|
||||||
|
],
|
||||||
|
{
|
||||||
|
"author": "mira-6",
|
||||||
|
"description": "Single-node wildcard implementation.",
|
||||||
|
"nickname": "mira-wildcard-node",
|
||||||
|
"title": "mira-wildcard-node",
|
||||||
|
"title_aux": "mira-wildcard-node"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"https://github.com/sonyeon-sj/ComfyUI-easy_ImageSize_Selecter": [
|
||||||
|
[
|
||||||
|
"ImageSizer",
|
||||||
|
"promptSelecter"
|
||||||
|
],
|
||||||
|
{
|
||||||
|
"title_aux": "ComfyUI-easy_ImageSize_Selecter"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"https://github.com/thinkthinking/ComfyUI-Ye": [
|
||||||
|
[
|
||||||
|
"CheckpointLoader|Ye",
|
||||||
|
"OllamaVision|Ye",
|
||||||
|
"PrintHelloWorld|Ye",
|
||||||
|
"Signature|Ye"
|
||||||
|
],
|
||||||
|
{
|
||||||
|
"title_aux": "ComfyUI-Ye"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"https://github.com/wailovet/ComfyUI-WW": [
|
||||||
|
[
|
||||||
|
"WW_AccumulationPreviewImages",
|
||||||
|
"WW_AppendString",
|
||||||
|
"WW_CurrentPreviewImages",
|
||||||
|
"WW_ImageResize",
|
||||||
|
"WW_PreviewTextNode",
|
||||||
|
"WW_RandString"
|
||||||
|
],
|
||||||
|
{
|
||||||
|
"title_aux": "ComfyUI-WW"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"https://github.com/yowipr/ComfyUI-Manual": [
|
||||||
|
[
|
||||||
|
"EXAMPLE",
|
||||||
|
"M_Layer",
|
||||||
|
"M_Output",
|
||||||
|
"M_RenderArea"
|
||||||
|
],
|
||||||
|
{
|
||||||
|
"title_aux": "ComfyUI-Manual"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
@@ -1,4 +1,3 @@
|
|||||||
#!/bin/bash
|
#!/bin/bash
|
||||||
source ../../../../venv/bin/activate
|
rm ~/.tmp/dev/*.py > /dev/null 2>&1
|
||||||
rm .tmp/*.py > /dev/null
|
python ../../scanner.py ~/.tmp/tutorial
|
||||||
python ../../scanner.py
|
|
||||||
|
|||||||
@@ -98,36 +98,6 @@ def remap_pip_package(pkg):
|
|||||||
std_log_lock = threading.Lock()
|
std_log_lock = threading.Lock()
|
||||||
|
|
||||||
|
|
||||||
class TerminalHook:
|
|
||||||
def __init__(self):
|
|
||||||
self.hooks = {}
|
|
||||||
|
|
||||||
def add_hook(self, k, v):
|
|
||||||
self.hooks[k] = v
|
|
||||||
|
|
||||||
def remove_hook(self, k):
|
|
||||||
if k in self.hooks:
|
|
||||||
del self.hooks[k]
|
|
||||||
|
|
||||||
def write_stderr(self, msg):
|
|
||||||
for v in self.hooks.values():
|
|
||||||
try:
|
|
||||||
v.write_stderr(msg)
|
|
||||||
except Exception:
|
|
||||||
pass
|
|
||||||
|
|
||||||
def write_stdout(self, msg):
|
|
||||||
for v in self.hooks.values():
|
|
||||||
try:
|
|
||||||
v.write_stdout(msg)
|
|
||||||
except Exception:
|
|
||||||
pass
|
|
||||||
|
|
||||||
|
|
||||||
terminal_hook = TerminalHook()
|
|
||||||
sys.__comfyui_manager_terminal_hook = terminal_hook
|
|
||||||
|
|
||||||
|
|
||||||
def handle_stream(stream, prefix):
|
def handle_stream(stream, prefix):
|
||||||
stream.reconfigure(encoding=locale.getpreferredencoding(), errors='replace')
|
stream.reconfigure(encoding=locale.getpreferredencoding(), errors='replace')
|
||||||
for msg in stream:
|
for msg in stream:
|
||||||
@@ -257,7 +227,7 @@ try:
|
|||||||
|
|
||||||
def sync_write(self, message, file_only=False):
|
def sync_write(self, message, file_only=False):
|
||||||
with log_lock:
|
with log_lock:
|
||||||
timestamp = datetime.datetime.now().strftime('%Y-%m-%d %H:%M:%S')[:-3]
|
timestamp = datetime.datetime.now().strftime('%Y-%m-%d %H:%M:%S.%f')[:-3]
|
||||||
if self.last_char != '\n':
|
if self.last_char != '\n':
|
||||||
log_file.write(message)
|
log_file.write(message)
|
||||||
else:
|
else:
|
||||||
@@ -270,11 +240,9 @@ try:
|
|||||||
if self.is_stdout:
|
if self.is_stdout:
|
||||||
write_stdout(message)
|
write_stdout(message)
|
||||||
original_stdout.flush()
|
original_stdout.flush()
|
||||||
terminal_hook.write_stderr(message)
|
|
||||||
else:
|
else:
|
||||||
write_stderr(message)
|
write_stderr(message)
|
||||||
original_stderr.flush()
|
original_stderr.flush()
|
||||||
terminal_hook.write_stdout(message)
|
|
||||||
|
|
||||||
def flush(self):
|
def flush(self):
|
||||||
log_file.flush()
|
log_file.flush()
|
||||||
@@ -417,30 +385,7 @@ check_bypass_ssl()
|
|||||||
# Perform install
|
# Perform install
|
||||||
processed_install = set()
|
processed_install = set()
|
||||||
script_list_path = os.path.join(os.path.dirname(os.path.realpath(__file__)), "startup-scripts", "install-scripts.txt")
|
script_list_path = os.path.join(os.path.dirname(os.path.realpath(__file__)), "startup-scripts", "install-scripts.txt")
|
||||||
pip_map = None
|
pip_fixer = PIPFixer(get_installed_packages())
|
||||||
|
|
||||||
|
|
||||||
def get_installed_packages():
|
|
||||||
global pip_map
|
|
||||||
|
|
||||||
if pip_map is None:
|
|
||||||
try:
|
|
||||||
result = subprocess.check_output([sys.executable, '-m', 'pip', 'list'], universal_newlines=True)
|
|
||||||
|
|
||||||
pip_map = {}
|
|
||||||
for line in result.split('\n'):
|
|
||||||
x = line.strip()
|
|
||||||
if x:
|
|
||||||
y = line.split()
|
|
||||||
if y[0] == 'Package' or y[0].startswith('-'):
|
|
||||||
continue
|
|
||||||
|
|
||||||
pip_map[y[0]] = y[1]
|
|
||||||
except subprocess.CalledProcessError as e:
|
|
||||||
print(f"[ComfyUI-Manager] Failed to retrieve the information of installed pip packages.")
|
|
||||||
return set()
|
|
||||||
|
|
||||||
return pip_map
|
|
||||||
|
|
||||||
|
|
||||||
def is_installed(name):
|
def is_installed(name):
|
||||||
@@ -652,8 +597,11 @@ if os.path.exists(script_list_path):
|
|||||||
print("\n[ComfyUI-Manager] Startup script completed.")
|
print("\n[ComfyUI-Manager] Startup script completed.")
|
||||||
print("#######################################################################\n")
|
print("#######################################################################\n")
|
||||||
|
|
||||||
|
pip_fixer.fix_broken()
|
||||||
|
|
||||||
del processed_install
|
del processed_install
|
||||||
del pip_map
|
del pip_fixer
|
||||||
|
clear_pip_cache()
|
||||||
|
|
||||||
|
|
||||||
def check_windows_event_loop_policy():
|
def check_windows_event_loop_policy():
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
[project]
|
[project]
|
||||||
name = "comfyui-manager"
|
name = "comfyui-manager"
|
||||||
description = "ComfyUI-Manager provides features to install and manage custom nodes for ComfyUI, as well as various functionalities to assist with ComfyUI."
|
description = "ComfyUI-Manager provides features to install and manage custom nodes for ComfyUI, as well as various functionalities to assist with ComfyUI."
|
||||||
version = "2.51.1"
|
version = "2.53"
|
||||||
license = { file = "LICENSE.txt" }
|
license = { file = "LICENSE.txt" }
|
||||||
dependencies = ["GitPython", "PyGithub", "matrix-client==0.4.0", "transformers", "huggingface-hub>0.20", "typer", "rich", "typing-extensions"]
|
dependencies = ["GitPython", "PyGithub", "matrix-client==0.4.0", "transformers", "huggingface-hub>0.20", "typer", "rich", "typing-extensions"]
|
||||||
|
|
||||||
|
|||||||
@@ -1,3 +1,4 @@
|
|||||||
|
pygit2
|
||||||
GitPython
|
GitPython
|
||||||
PyGithub
|
PyGithub
|
||||||
matrix-client==0.4.0
|
matrix-client==0.4.0
|
||||||
@@ -5,4 +6,4 @@ transformers
|
|||||||
huggingface-hub>0.20
|
huggingface-hub>0.20
|
||||||
typer
|
typer
|
||||||
rich
|
rich
|
||||||
typing-extensions
|
typing-extensions
|
||||||
|
|||||||
2
scan.sh
2
scan.sh
@@ -1,6 +1,6 @@
|
|||||||
#!/bin/bash
|
#!/bin/bash
|
||||||
rm ~/.tmp/default/*.py > /dev/null 2>&1
|
rm ~/.tmp/default/*.py > /dev/null 2>&1
|
||||||
python scanner.py ~/.tmp/default $*
|
python -m scanner ~/.tmp/default $*
|
||||||
cp extension-node-map.json node_db/new/.
|
cp extension-node-map.json node_db/new/.
|
||||||
|
|
||||||
echo "Integrity check"
|
echo "Integrity check"
|
||||||
|
|||||||
13
scanner.py
13
scanner.py
@@ -2,7 +2,8 @@ import ast
|
|||||||
import re
|
import re
|
||||||
import os
|
import os
|
||||||
import json
|
import json
|
||||||
from git import Repo
|
import sys
|
||||||
|
from glob import git_wrapper
|
||||||
import concurrent
|
import concurrent
|
||||||
import datetime
|
import datetime
|
||||||
import concurrent.futures
|
import concurrent.futures
|
||||||
@@ -243,25 +244,27 @@ def get_py_urls_from_json(json_file):
|
|||||||
|
|
||||||
return py_files
|
return py_files
|
||||||
|
|
||||||
|
import traceback
|
||||||
def clone_or_pull_git_repository(git_url):
|
def clone_or_pull_git_repository(git_url):
|
||||||
repo_name = git_url.split("/")[-1].split(".")[0]
|
repo_name = git_url.split("/")[-1].split(".")[0]
|
||||||
repo_dir = os.path.join(temp_dir, repo_name)
|
repo_dir = os.path.join(temp_dir, repo_name)
|
||||||
|
|
||||||
if os.path.exists(repo_dir):
|
if os.path.exists(repo_dir):
|
||||||
try:
|
try:
|
||||||
repo = Repo(repo_dir)
|
repo = git_wrapper.Repo(repo_dir)
|
||||||
origin = repo.remote(name="origin")
|
origin = repo.remote(name="origin")
|
||||||
origin.pull()
|
origin.pull()
|
||||||
repo.git.submodule('update', '--init', '--recursive')
|
repo.update_recursive()
|
||||||
print(f"Pulling {repo_name}...")
|
print(f"Pulling {repo_name}...")
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
|
traceback.print_exc()
|
||||||
print(f"Pulling {repo_name} failed: {e}")
|
print(f"Pulling {repo_name} failed: {e}")
|
||||||
else:
|
else:
|
||||||
try:
|
try:
|
||||||
Repo.clone_from(git_url, repo_dir, recursive=True)
|
git_wrapper.clone_from(git_url, repo_dir, recursive=True)
|
||||||
print(f"Cloning {repo_name}...")
|
print(f"Cloning {repo_name}...")
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
|
traceback.print_exc()
|
||||||
print(f"Cloning {repo_name} failed: {e}")
|
print(f"Cloning {repo_name} failed: {e}")
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user