Compare commits

..

45 Commits
3.21.1 ... 3.26

Author SHA1 Message Date
Dr.Lt.Data
4fd17b0bf5 improved: advanced missing node detection based on embedded info
https://github.com/ltdrdata/ComfyUI-Manager/issues/1445

feat: Custom Nodes In Workflow
https://github.com/ltdrdata/ComfyUI-Manager/issues/990
https://github.com/ltdrdata/ComfyUI-Manager/issues/127

improved: show version on main dialog
modified: aux_id - use github_id if possible
removed: `fetch updates` button
2025-02-24 21:18:42 +09:00
Dr.Lt.Data
76d2206058 update DB 2025-02-24 21:00:14 +09:00
LAOGOU
51e8b608dc Update custom-node-list.json (#1575)
* Add Comfyui-Transform and LG_HotReload custom nodes

* Update custom-node-list.json
2025-02-24 20:31:45 +09:00
Dr.Lt.Data
a68330fb8f rollback wip code 2025-02-23 11:25:30 +09:00
Dr.Lt.Data
2449ad5c69 update DB 2025-02-23 11:08:07 +09:00
Dr.Lt.Data
064c812df3 update DB 2025-02-22 20:04:13 +09:00
bymyself
48d5ec9e66 Retain workflow versions when serializing node_versions (#1563)
* retain initial node_versions on serialize

* give precedence to workflow version

* set version info on node

* move patch to setup hook

* switch to nodeCreated
2025-02-22 17:42:36 +09:00
Dr.Lt.Data
914419fd1e update DB 2025-02-22 17:37:06 +09:00
The Dave
f005fc8ca0 added daves_nodes to custom node list for pull request (#1574) 2025-02-22 16:58:37 +09:00
RiceRound
43b7960de2 Add RiceRound Cloud Node (#1572) 2025-02-22 11:11:59 +09:00
Blueprint Coding
2ed1e58032 Update custom-node-list.json to match my node ID to Comfy registry ID (#1570) 2025-02-22 11:11:35 +09:00
Dr.Lt.Data
c63b212700 update DB 2025-02-20 12:33:06 +09:00
Dr.Lt.Data
e9df78c0e7 improved: When user do Switch ComfyUI, update the policy accordingly. 2025-02-20 12:20:04 +09:00
Dr.Lt.Data
b0daf81185 update dependencies in pyproject.toml 2025-02-19 22:09:30 +09:00
Dr.Lt.Data
cee4fdcbb0 fixed: apply ConfigParser(strict=False) to other callsites
https://github.com/ltdrdata/ComfyUI-Manager/pull/1561
2025-02-19 22:07:47 +09:00
Vanisper
df3cdfccb0 fix(git_utils): allow duplicate vscode-merge-base sections with strict=False (#1561)
- Set ConfigParser strict mode to False
- Resolves issue #1529 by permitting section duplicates
- Allow `vscode-merge-base` to appear multiple times in `.git/config`
2025-02-19 22:05:04 +09:00
Dr.Lt.Data
894042cd0e update DB 2025-02-19 22:02:10 +09:00
jmjoy
8123287952 Update model-list.json (#1564) 2025-02-19 21:40:20 +09:00
puke
bc677705d8 Update custom-node-list.json (#1562) 2025-02-19 21:39:41 +09:00
Dr.Lt.Data
5dd8ea8aab feat: update policy for updating ComfyUI
https://github.com/ltdrdata/ComfyUI-Manager/issues/1552

fixed: comfyui versions should be based on commit date
https://github.com/ltdrdata/ComfyUI-Manager/issues/1566

fixed: invalid identifying of nightly node packs which has `git@github.com:...` url
fixed: switch comfyui should be based on `master` branch instead of `main` branch
fixed: switch_to_default_branch - more robust switching
refactor: endpoints for policies
2025-02-19 21:34:13 +09:00
Dr.Lt.Data
41172be796 modified: don't show outdated ComfyUI message if desktop mode
modified: use __COMFYUI_DESKTOP_VERSION__ in notice board if desktop mode
2025-02-19 07:41:54 +09:00
Dr.Lt.Data
ad1b4a9a86 feat: reverse proxy
https://github.com/ltdrdata/ComfyUI-Manager/pull/795/files
2025-02-18 23:41:44 +09:00
Dr.Lt.Data
e0e3ec02b3 update DB 2025-02-18 21:08:19 +09:00
Dr.Lt.Data
a6cc392473 fix typo 2025-02-17 22:34:16 +09:00
Dr.Lt.Data
36f48b8656 feat: custom pip_blacklist
https://github.com/ltdrdata/ComfyUI-Manager/issues/1560
2025-02-17 22:32:26 +09:00
Dr.Lt.Data
3d883ca37d update DB 2025-02-17 22:06:07 +09:00
Dr.Lt.Data
34ed81ca64 update DB 2025-02-17 21:40:48 +09:00
mohseni-mr
a9e0880572 Added ComfyUI Mohseni Kit to ComfyUI Manager (#1559) 2025-02-17 21:39:48 +09:00
Dr.Lt.Data
9500e1c3c4 update DB 2025-02-17 21:39:30 +09:00
Blueprint Coding
d81aa9cbbc Update custom-node-list.json (#1557)
Added my custom nodes: "The AI Doctors Clinical Tools"
description: "MultiInt and MultiText nodes. The MultiInt node allows management of multiple int values with configurable steps, +/- buttons, drag change, & customized labels. The MultiText node offers similar functionality for string values."
2025-02-17 21:38:37 +09:00
Dr.Lt.Data
21d4b25c2d update DB 2025-02-17 21:38:02 +09:00
CY-CHENYUE
0080783a11 Update custom-node-list.json (#1555) 2025-02-17 21:37:08 +09:00
Dr.Lt.Data
2c3f44a3f8 fixed: cm-cli.py - missing 'utils' module if COMYUI_PatH is set
https://github.com/ltdrdata/ComfyUI-Manager/issues/1556
2025-02-17 07:43:35 +09:00
Dr.Lt.Data
3ddf414097 fixed: Modify the import of chardet to be lazy.
- "Prevent `chardet` from being imported in `manager_util` before automatic dependency installation."**

https://github.com/ltdrdata/ComfyUI-Manager/issues/1554
2025-02-16 20:29:29 +09:00
Dr.Lt.Data
59fb63f1f7 ruff fix 2025-02-16 14:42:58 +09:00
Dr.Lt.Data
fa1b514440 improved: Update All - Show link on the result board
fixed: Update All - Updates for unknown nodes were not being applied
fixed: corner case crash whilte install/updating

https://github.com/ltdrdata/ComfyUI-Manager/issues/1168
2025-02-16 14:25:57 +09:00
Dr.Lt.Data
1579c58876 fixed: ensure chardet dependency
https://github.com/ltdrdata/ComfyUI-Manager/discussions/1553
2025-02-16 13:04:56 +09:00
Dr.Lt.Data
153d044331 update DB 2025-02-16 10:30:18 +09:00
wirytiox
f2496f7054 Update custom-node-list.json (#1551) 2025-02-16 10:11:11 +09:00
Sssnap
99022f4f3d Update custom-node-list.json (#1549)
* Update custom-node-list.json

* Update custom-node-list.json

---------

Co-authored-by: Dr.Lt.Data <128333288+ltdrdata@users.noreply.github.com>
2025-02-16 10:10:57 +09:00
Dr.Lt.Data
60a5e4f261 fixed: address abnormal encoding of 'requirements.txt'
improved: better error message

https://github.com/ltdrdata/ComfyUI-Manager/issues/1513
2025-02-16 10:05:29 +09:00
Dr.Lt.Data
661586d3b6 update DB 2025-02-15 17:42:39 +09:00
Dr.Lt.Data
abc26cf906 fixed: pre_startup - restart if script is executed
fixed: normalize cnr versions via StrictVersion
- 2.5 and 2.5.0 were regarded as different version
2025-02-15 17:27:09 +09:00
Dr.Lt.Data
12351bada7 improved: is_local_mode - use ipaddress module instead of string match
refactor: get_config() - ensure lowercase option when returning dict

https://github.com/ltdrdata/ComfyUI-Manager/issues/1546
2025-02-15 10:02:25 +09:00
Dr.Lt.Data
a6816d53d7 update DB 2025-02-15 09:47:42 +09:00
31 changed files with 7531 additions and 4691 deletions

View File

@@ -149,6 +149,7 @@ In `ComfyUI-Manager` V3.0 and later, configuration files and dynamically generat
* Basic config files: `<USER_DIRECTORY>/default/ComfyUI-Manager/config.ini`
* Configurable channel lists: `<USER_DIRECTORY>/default/ComfyUI-Manager/channels.ini`
* Configurable pip overrides: `<USER_DIRECTORY>/default/ComfyUI-Manager/pip_overrides.json`
* Configurable pip blacklist: `<USER_DIRECTORY>/default/ComfyUI-Manager/pip_blacklist.list`
* Saved snapshot files: `<USER_DIRECTORY>/default/ComfyUI-Manager/snapshots`
* Startup script files: `<USER_DIRECTORY>/default/ComfyUI-Manager/startup-scripts`
* Component files: `<USER_DIRECTORY>/default/ComfyUI-Manager/components`
@@ -301,7 +302,10 @@ The following settings are applied based on the section marked as `is_default`.
* Custom pip mapping
* When you create the `pip_overrides.json` file, it changes the installation of specific pip packages to installations defined by the user.
* Please refer to the `pip_overrides.json.template` file.
* Prevent the installation of specific pip packages
* List the package names one per line in the `pip_blacklist.list` file.
* Use `aria2` as downloader
* [howto](docs/en/use_aria2.md)
@@ -309,6 +313,29 @@ The following settings are applied based on the section marked as `is_default`.
* This option can be used if performance issues occur in a Colab+GDrive environment.
## Environment Variables
The following features can be configured using environment variables:
* **COMFYUI_PATH**: The installation path of ComfyUI
* **GITHUB_ENDPOINT**: Reverse proxy configuration for environments with limited access to GitHub
* **HF_ENDPOINT**: Reverse proxy configuration for environments with limited access to Hugging Face
### Example 1:
Redirecting `https://github.com/ltdrdata/ComfyUI-Impact-Pack` to `https://mirror.ghproxy.com/https://github.com/ltdrdata/ComfyUI-Impact-Pack`
```
GITHUB_ENDPOINT=https://mirror.ghproxy.com/https://github.com
```
#### Example 2:
Changing `https://huggingface.co/path/to/somewhere` to `https://some-hf-mirror.com/path/to/somewhere`
```
HF_ENDPOINT=https://some-hf-mirror.com
```
## Scanner
When you run the `scan.sh` script:

View File

@@ -32,6 +32,7 @@ if comfy_path is None:
print("\n[bold yellow]WARN: The `COMFYUI_PATH` environment variable is not set. Assuming `custom_nodes/ComfyUI-Manager/../../` as the ComfyUI path.[/bold yellow]", file=sys.stderr)
comfy_path = os.path.abspath(os.path.join(manager_util.comfyui_manager_path, '..', '..'))
# This should be placed here
sys.path.append(comfy_path)
import utils.extra_config
@@ -42,7 +43,7 @@ import cnr_utils
comfyui_manager_path = os.path.abspath(os.path.dirname(__file__))
cm_global.pip_blacklist = ['torch', 'torchsde', 'torchvision']
cm_global.pip_blacklist = {'torch', 'torchsde', 'torchvision'}
cm_global.pip_downgrade_blacklist = ['torch', 'torchsde', 'torchvision', 'transformers', 'safetensors', 'kornia']
cm_global.pip_overrides = {'numpy': 'numpy<2'}
@@ -51,6 +52,14 @@ if os.path.exists(os.path.join(manager_util.comfyui_manager_path, "pip_overrides
cm_global.pip_overrides = json.load(json_file)
if os.path.exists(os.path.join(manager_util.comfyui_manager_path, "pip_blacklist.list")):
with open(os.path.join(manager_util.comfyui_manager_path, "pip_blacklist.list"), 'r', encoding="UTF-8", errors="ignore") as f:
for x in f.readlines():
y = x.strip()
if y != '':
cm_global.pip_blacklist.add(y)
def check_comfyui_hash():
repo = git.Repo(comfy_path)
core.comfy_ui_revision = len(list(repo.iter_commits('HEAD')))
@@ -67,7 +76,7 @@ core.check_invalid_nodes()
def read_downgrade_blacklist():
try:
import configparser
config = configparser.ConfigParser()
config = configparser.ConfigParser(strict=False)
config.read(core.manager_config.path)
default_conf = config['default']
@@ -136,6 +145,13 @@ class Ctx:
cm_global.pip_overrides = json.load(json_file)
cm_global.pip_overrides = {'numpy': 'numpy<2'}
if os.path.exists(core.manager_pip_blacklist_path):
with open(core.manager_pip_blacklist_path, 'r', encoding="UTF-8", errors="ignore") as f:
for x in f.readlines():
y = x.strip()
if y != '':
cm_global.pip_blacklist.add(y)
@staticmethod
def get_startup_scripts_path():
return os.path.join(core.manager_startup_script_path, "install-scripts.txt")

View File

@@ -3748,17 +3748,6 @@
"install_type": "git-clone",
"description": "Nodes: Noxin Complete Chime, Noxin Scaled Resolutions, Load from Noxin Prompt Library, Save to Noxin Prompt Library"
},
{
"author": "apesplat",
"title": "ezXY scripts and nodes",
"id": "ezxy",
"reference": "https://github.com/GMapeSplat/ComfyUI_ezXY",
"files": [
"https://github.com/GMapeSplat/ComfyUI_ezXY"
],
"install_type": "git-clone",
"description": "Extensions/Patches: Enables linking float and integer inputs and ouputs. Values are automatically cast to the correct type and clamped to the correct range. Works with both builtin and custom nodes.[w/NOTE: This repo patches ComfyUI's validate_inputs and map_node_over_list functions while running. May break depending on your version of ComfyUI. Can be deactivated in config.yaml.]Nodes: A collection of nodes for facilitating the generation of XY plots. Capable of plotting changes over most primitive values.[w/Does not work with current version of Comfyui]"
},
{
"author": "kinfolk0117",
"title": "SimpleTiles",
@@ -5454,7 +5443,7 @@
"https://github.com/lldacing/ComfyUI_Patches_ll"
],
"install_type": "git-clone",
"description": "Some patches for Flux|HunYuanVideo etc, support TeaCache, PuLID."
"description": "Some patches for Flux|HunYuanVideo|LTXVideo etc, support TeaCache, PuLID, First Block Cache."
},
{
"author": "lldacing",
@@ -6444,6 +6433,16 @@
"install_type": "git-clone",
"description": "Make it possible to edit the prompt using the Monaco Editor, an editor implementation used in VSCode.\nNOTE: This extension supports both ComfyUI and A1111 simultaneously."
},
{
"author": "Taremin",
"title": "comfyui-keep-multiple-tabs",
"reference": "https://github.com/Taremin/comfyui-keep-multiple-tabs",
"files": [
"https://github.com/Taremin/comfyui-keep-multiple-tabs"
],
"install_type": "git-clone",
"description": "This is an extension for ComfyUI. It retains multiple workflow tabs so that they are not lost when reloading or restarting."
},
{
"author": "foxtrot-roger",
"title": "RF Nodes",
@@ -10079,6 +10078,16 @@
"install_type": "git-clone",
"description": "a custom node for [a/echomimic_v2](https://github.com/antgroup/echomimic_v2)"
},
{
"author": "AIFSH",
"title": "SemiChat-ComfyUI",
"reference": "https://github.com/AIFSH/SemiChat-ComfyUI",
"files": [
"https://github.com/AIFSH/SemiChat-ComfyUI"
],
"install_type": "git-clone",
"description": "A ComfyUI chat node based on SemiUI."
},
{
"author": "Koishi-Star",
"title": "Euler-Smea-Dyn-Sampler",
@@ -10484,6 +10493,26 @@
"install_type": "git-clone",
"description": "[a/CSD_MT](https://github.com/Snowfallingplum/CSD-MT) is a method about 'Content-Style Decoupling for Unsupervised Makeup Transfer without Generating Pseudo Ground Truth', you can use it in comfyUI."
},
{
"author": "smthemex",
"title": "ComfyUI_Light_A_Video",
"reference": "https://github.com/smthemex/ComfyUI_Light_A_Video",
"files": [
"https://github.com/smthemex/ComfyUI_Light_A_Video"
],
"install_type": "git-clone",
"description": "Light-A-Video: Training-free Video Relighting via Progressive Light Fusion,you can use it in comfyUI"
},
{
"author": "smthemex",
"title": "ComfyUI_YuE",
"reference": "https://github.com/smthemex/ComfyUI_YuE",
"files": [
"https://github.com/smthemex/ComfyUI_YuE"
],
"install_type": "git-clone",
"description": "[a/YuE](https://github.com/multimodal-art-projection/YuE) is a groundbreaking series of open-source foundation models designed for music generation, specifically for transforming lyrics into full songs (lyrics2song). you can use it in comfyUI"
},
{
"author": "choey",
"title": "Comfy-Topaz",
@@ -10634,6 +10663,26 @@
"install_type": "git-clone",
"description": "These are just some nodes I wanted and couldn't find where anyone else had made them yet."
},
{
"author": "quadmoon",
"title": "ComfyUI-Riffusion",
"reference": "https://github.com/traugdor/ComfyUI-Riffusion",
"files": [
"https://github.com/traugdor/ComfyUI-Riffusion"
],
"install_type": "git-clone",
"description": "A ComfyUI extension for Riffusion audio generation."
},
{
"author": "quadmoon",
"title": "ComfyUI-UltimateSDUpscale-GGUF",
"reference": "https://github.com/traugdor/ComfyUI-UltimateSDUpscale-GGUF",
"files": [
"https://github.com/traugdor/ComfyUI-UltimateSDUpscale-GGUF"
],
"install_type": "git-clone",
"description": "GGUF implementation for the ComfyUI Ultimate SD Upscale node."
},
{
"author": "quadme7macoon",
"title": "ComfyUI-ShadertoyGL",
@@ -11206,6 +11255,16 @@
"install_type": "git-clone",
"description": "Nodes:FastImageListToImageBatch"
},
{
"author": "jax-explorer",
"title": "comfyui-model-dynamic-loader",
"reference": "https://github.com/jax-explorer/comfyui-model-dynamic-loader",
"files": [
"https://github.com/jax-explorer/comfyui-model-dynamic-loader"
],
"install_type": "git-clone",
"description": "for comfyonline dynamic loader\ncomfyonline is comfyui cloud website"
},
{
"author": "sugarkwork",
"title": "comfyui_cohere",
@@ -11250,6 +11309,17 @@
"install_type": "git-clone",
"description": "This extension contains a custom node for ComfyUI. The node, called 'Bounding Box Crop', is designed to compute the top-left coordinates of a cropped bounding box based on input coordinates and dimensions of the final cropped image. It does so computing the center of the cropping area and then computing where the top-left coordinates would be."
},
{
"author": "alessandrozonta",
"title": "Comfyui-LoopLoader",
"id": "Comfyui-LoopLoader",
"reference": "https://github.com/alessandrozonta/Comfyui-LoopLoader",
"files": [
"hhttps://github.com/alessandrozonta/Comfyui-LoopLoader"
],
"install_type": "git-clone",
"description": "A ComfyUI custom node for loading images sequentially from a directory. Loops back to the first image when reaching the end"
},
{
"author": "curiousjp",
"title": "ComfyUI-MaskBatchPermutations",
@@ -12970,7 +13040,7 @@
"https://github.com/licyk/ComfyUI-Restart-Sampler"
],
"install_type": "git-clone",
"description": "This extension is a node that directly expands the functionality of KSampler, adding the TCD sampling to KSampler."
"description": "This extension is a node that directly expands the functionality of KSampler, rather than being in the form of a custom node. [w/Workflows created using this feature are not compatible with other users.]"
},
{
"author": "licyk",
@@ -13337,6 +13407,16 @@
"install_type": "git-clone",
"description": "This node for ComfyUI allows saving images with an optional alpha channel (transparency). It supports saving images in formats like PNG, JPEG, and WebP."
},
{
"author": "APZmedia",
"title": "APZmedia Together Image Generator for ComfyUI",
"reference": "https://github.com/APZmedia/APZmedia-comfy-together-lora",
"files": [
"https://github.com/APZmedia/APZmedia-comfy-together-lora"
],
"install_type": "git-clone",
"description": "A ComfyUI node to implement Together AI API image generation"
},
{
"author": "N3rd00d",
"title": "ComfyUI-Paint3D-Nodes",
@@ -13895,6 +13975,16 @@
"install_type": "git-clone",
"description": "This node provides advanced text-to-speech functionality powered by KokoroTTS. Follow the instructions below to install, configure, and use the node within your portable ComfyUI installation."
},
{
"author": "MushroomFleet",
"title": "DJZ-Pedalboard",
"reference": "https://github.com/MushroomFleet/DJZ-Pedalboard",
"files": [
"https://github.com/MushroomFleet/DJZ-Pedalboard"
],
"install_type": "git-clone",
"description": "This project provides a collection of custom nodes designed for enhanced audio effects in ComfyUI. With an intuitive pedalboard interface, users can easily integrate and manipulate various audio effects within their workflows."
},
{
"author": "var1ableX",
"title": "ComfyUI_Accessories",
@@ -14193,17 +14283,6 @@
"install_type": "git-clone",
"description": "multiline text node that strips c-style comments (i.e.'//' and '/* ... */') before passing output string downstream"
},
{
"author": "noarche",
"title": "noarche/Color Enhance",
"id": "color-enhance",
"reference": "https://github.com/noarche/sd-webui-color-enhance",
"files": [
"https://github.com/noarche/sd-webui-color-enhance"
],
"install_type": "git-clone",
"description": "Script for [a/AUTOMATIC1111/stable-diffusion-webui](https://github.com/AUTOMATIC1111/stable-diffusion-webui) and node for ComfyUI to enhance colors.\nThis is the same algorithm GIMP/GEGL uses for color enhancement. The gist of this implementation is that it converts the color space to [CIELCh(ab) and normalizes the chroma (or '[a/colorfulness](https://en.wikipedia.org/wiki/Colorfulness)') component. Original source can be found in the link below."
},
{
"author": "emojiiii",
"title": "ComfyUI_Emojiiii_Custom_Nodes",
@@ -14739,6 +14818,16 @@
"install_type": "git-clone",
"description": "A flexible and customizable prompt generator for generating detailed and creative prompts for image generation models for ComfyUI"
},
{
"author": "fairy-root",
"title": "ComfyUI-Show-Text",
"reference": "https://github.com/fairy-root/ComfyUI-Show-Text",
"files": [
"https://github.com/fairy-root/ComfyUI-Show-Text"
],
"install_type": "git-clone",
"description": "A simple but powerful node for ComfyUI that displays text input in a readable format. Perfect for viewing outputs from text generation nodes, prompt builders, interrogators, and more."
},
{
"author": "ryanontheinside",
"title": "RyanOnTheInside",
@@ -15256,6 +15345,26 @@
"install_type": "git-clone",
"description": "Simple Node to make panoramic images"
},
{
"author": "RodrigoSKohl",
"title": "Interior Design for Comfyui",
"reference": "https://github.com/RodrigoSKohl/InteriorDesign-for-ComfyUI",
"files": [
"https://github.com/RodrigoSKohl/InteriorDesign-for-ComfyUI"
],
"install_type": "git-clone",
"description": "This node is based on MykolaL/StableDesign"
},
{
"author": "RodrigoSKohl",
"title": "TryOff Anyone",
"reference": "https://github.com/RodrigoSKohl/comfyui-tryoff-anyone",
"files": [
"https://github.com/RodrigoSKohl/comfyui-tryoff-anyone"
],
"install_type": "git-clone",
"description": "Node to tryoff clothes"
},
{
"author": "nicehero",
"title": "comfyui-SegGPT",
@@ -15443,6 +15552,17 @@
"install_type": "git-clone",
"description": "This is a custom node to convert only the Diffusion model part or CLIP model part to fp8 in ComfyUI.\nVAE fp8 conversion is not supported.\nThe advantage of this node is that you do not need to separate unet/clip/vae in advance when converting to fp8, but can use the safetenros files that ComfyUI provides."
},
{
"author": "Shiba-2-shiba",
"title": "ComfyUI_FreeU_V2_timestepadd",
"id": "ComfyUI_FreeU_V2_timestepadd",
"reference": "https://github.com/Shiba-2-shiba/ComfyUI_FreeU_V2_timestepadd",
"files": [
"https://github.com/Shiba-2-shiba/ComfyUI_FreeU_V2_timestepadd"
],
"install_type": "git-clone",
"description": "This is a custom node to add timestep for FreeU V2."
},
{
"author": "Bao Pham",
"title": "ComfyUI-LyraVSIH",
@@ -15766,7 +15886,7 @@
"https://github.com/Cyber-BCat/ComfyUI_Auto_Caption"
],
"install_type": "git-clone",
"description": "This report contains a 'load many images' node which is going to load the image set by the number of file names from smallest to largest, and the images will no longer be loaded in the wrong order! Setting index=0 makes it load from the first small value (image flie name) image, and index=2 will load them from the second image. Another node 'load images & resize' can resize the image by the first loaded image."
"description": "Load images in order(All other nodes are in the wrong order)! Using LLM and Joy tag pipeline to tag your image(s folder), it's suitable for train FLUX LoRA and also sdxl."
},
{
"author": "cr7Por",
@@ -16158,6 +16278,17 @@
"install_type": "git-clone",
"description": "ComfyUI nodes for Janus-Pro, a unified multimodal understanding and generation framework."
},
{
"author": "CY-CHENYUE",
"title": "ComfyUI-Free-GPU",
"id": "ComfyUI-Free-GPU",
"reference": "https://github.com/CY-CHENYUE/ComfyUI-Free-GPU",
"files": [
"https://github.com/CY-CHENYUE/ComfyUI-Free-GPU"
],
"description": "ComfyUI-Free-GPU provides a node for releasing RAM and VRAM in ComfyUI.",
"install_type": "git-clone"
},
{
"author": "codecringebinge",
"title": "ComfyUI-Arrow-Key-Canvas-Navigation",
@@ -16444,6 +16575,16 @@
"install_type": "git-clone",
"description": "preprocessing images, presented in a visual way. It also calculates the corresponding image area."
},
{
"author": "SSsnap",
"title": "ComfyUI-LBW_flux",
"reference": "https://github.com/SS-snap/ComfyUI-LBW_flux",
"files": [
"https://github.com/SS-snap/ComfyUI-LBW_flux"
],
"install_type": "git-clone",
"description": "Through this node, you can more easily test the impact of different blocks in flux_lora on the final result."
},
{
"author": "RiceRound",
"title": "ComfyUI Compression and Encryption Node",
@@ -16487,7 +16628,7 @@
"description": "This repository contains a custom ComfyUI node for overlaying media using ffmpeg."
},
{
"author": "laogou666",
"author": "LAOGOU-666",
"title": "ComfyUI_LG_FFT",
"reference": "https://github.com/LAOGOU-666/ComfyUI_LG_FFT",
"files": [
@@ -16497,7 +16638,7 @@
"description": "Implementation of Fast Fourier Transform in COMFYUI"
},
{
"author": "laogou666",
"author": "LAOGOU-666",
"title": "Comfyui-LG_Relight",
"reference": "https://github.com/LAOGOU-666/Comfyui-LG_Relight",
"files": [
@@ -16506,6 +16647,17 @@
"install_type": "git-clone",
"description": "A simple implementation of real-time 3D lighting in ComfyUI. It's an open-source node, have fun playing around!"
},
{
"author": "LAOGOU-666",
"title": "ComfyUI-LG_HotReload",
"id": "ComfyUI-LG_HotReload",
"reference": "https://github.com/LAOGOU-666/Comfyui-LG_HotReload",
"files": [
"https://github.com/LAOGOU-666/Comfyui-LG_HotReload"
],
"install_type": "git-clone",
"description": "An extension for ComfyUI that allows hot reloading. Once installed, you can preview changes in real-time while developing custom nodes or installing plugins without restarting ComfyUI."
},
{
"author": "VertexStudio",
"title": "roblox-comfyui-nodes",
@@ -17083,7 +17235,7 @@
"https://github.com/mfg637/ComfyUI-ScheduledGuider-Ext"
],
"install_type": "git-clone",
"description": "NODES:ScheduledCFGGuider, PerpNegScheduledCFGGuider, CosineScheduler, Add zSNR Sigma max, InvertSigmas, ConcatSigmas, OffsetSigmas"
"description": "This extension contains various nodes for CFG scheduling and more. NODES:ScheduledCFGGuider, PerpNegScheduledCFGGuider, CosineScheduler, GaussianScheduler, LogNormalScheduler, InvertSigmas, ConcatSigmas, OffsetSigmas, SplitSigmasByValue"
},
{
"author": "changwook987",
@@ -17874,6 +18026,16 @@
"install_type": "git-clone",
"description": "A collection of image processing extension nodes for ComfyUI."
},
{
"author": "yichengup",
"title": "ComfyUI-VideoBlender",
"reference": "https://github.com/yichengup/ComfyUI-VideoBlender",
"files": [
"https://github.com/yichengup/ComfyUI-VideoBlender"
],
"install_type": "git-clone",
"description": "Video clip mixing"
},
{
"author": "Horizon Team",
"title": "ComfyUI_FluxMod",
@@ -18466,16 +18628,6 @@
"install_type": "git-clone",
"description": "quickly use the prompt word tool in ComfyUI"
},
{
"author": "jax-explorer",
"title": "comfyui-model-dynamic-loader",
"reference": "https://github.com/jax-explorer/comfyui-model-dynamic-loader",
"files": [
"https://github.com/jax-explorer/comfyui-model-dynamic-loader"
],
"install_type": "git-clone",
"description": "for comfyonline dynamic loader\ncomfyonline is comfyui cloud website"
},
{
"author": "LucipherDev",
"title": "ComfyUI-Golden-Noise",
@@ -18872,6 +19024,16 @@
"install_type": "git-clone",
"description": "A beautiful theme extension for ComfyUI that adds festive touches with dynamic backgrounds, snowfall effects, and animated node connections"
},
{
"author": "AEmotionStudio",
"title": "ComfyUI-EnhancedLinksandNodes 🎨✨",
"reference": "https://github.com/AEmotionStudio/ComfyUI-EnhancedLinksandNodes",
"files": [
"https://github.com/AEmotionStudio/ComfyUI-EnhancedLinksandNodes"
],
"install_type": "git-clone",
"description": "A visually stunning extension for ComfyUI that adds beautiful, customizable animations to both links and nodes in your workflow, with a focus on performance and customization. Includes an end-of-render animation and a text visibility tool for nodes. No extra packages are required, works with the latest version of ComfyUI, and should be compatible with most workflows. Larger workflows may experience performance issues, especially if you have a lot of nodes and are using a lower end system."
},
{
"author": "xfgexo",
"title": "EXO Custom ComfyUI Nodes",
@@ -18923,6 +19085,16 @@
"install_type": "git-clone",
"description": "A ComfyUI node that detects the skin tone of a person in an image and matches it to the standard emoji skin tone palette."
},
{
"author": "kevinmcmahondev",
"title": "KMCDev Nodes",
"reference": "https://github.com/kevinmcmahondev/comfyui-kmcdev-image-filter-adjustments",
"files": [
"https://github.com/kevinmcmahondev/comfyui-kmcdev-image-filter-adjustments"
],
"install_type": "git-clone",
"description": "A ComfyUI node that provides advanced image adjustment filters and controls for image manipulation"
},
{
"author": "mahdi",
"title": "seamless-clone-comfyui",
@@ -19374,6 +19546,16 @@
"install_type": "git-clone",
"description": "ComfyUI_OneButtonPrompt_Flux is a Flux prompt generation node. The subject can be 'human,' 'other' or a combination of both. For human, pose settings can be enabled. Additionally, various styles can be applied. Finally, combine it with 'Prompt Enhancement' to seamlessly automate image generation, eliminating the hassle of designing prompts."
},
{
"author": "billwuhao",
"title": "ComfyUI_StepAudioTTS",
"reference": "https://github.com/billwuhao/ComfyUI_StepAudioTTS",
"files": [
"https://github.com/billwuhao/ComfyUI_StepAudioTTS"
],
"install_type": "git-clone",
"description": "A Text To Speech node using Step-Audio-TTS in ComfyUI. Can speak, rap, sing, or clone voice."
},
{
"author": "pandaer119",
"title": "ComfyUI_pandai",
@@ -19527,6 +19709,16 @@
"install_type": "git-clone",
"description": "NODES: ComfyUI-ImageFx, ComfyUI-Whisk"
},
{
"author": "ainewsto",
"title": "Comfyui_Comfly",
"reference": "https://github.com/ainewsto/Comfyui_Comfly",
"files": [
"https://github.com/ainewsto/Comfyui_Comfly"
],
"install_type": "git-clone",
"description": "NODES: Comfly_Mj, Comfly_mjstyle, Comfly_upload, Comfly_Mju, Comfly_Mjv, Comfly_kling_videoPreview"
},
{
"author": "gremlation",
"title": "ComfyUI-ViewData",
@@ -20433,16 +20625,6 @@
"install_type": "git-clone",
"description": "ComfyUI-SendToDiscord is a custom node for ComfyUI that simplifies sending preview images to Discord via webhooks. It supports both single-image uploads and batch mode, making it an efficient tool for sharing your generated images directly with your Discord server."
},
{
"author": "gmorks",
"title": "ComfyUI Animagine prompt",
"reference": "https://github.com/gmorks/ComfyUI-Animagine-Prompt",
"files": [
"https://github.com/gmorks/ComfyUI-Animagine-Prompt"
],
"install_type": "git-clone",
"description": "Comfy UI node to prompt build for [a/https://huggingface.co/cagliostrolab/animagine-xl-4.0](https://huggingface.co/cagliostrolab/animagine-xl-4.0) model"
},
{
"author": "jinanlongen",
"title": "ComfyUI Prompt Expander Node",
@@ -20576,6 +20758,16 @@
"install_type": "git-clone",
"description": "This plugin integrates the Janus-Pro multi-modal model into ComfyUI, enabling advanced image understanding and text-to-image generation capabilities. It supports both image analysis and creative image generation workflows."
},
{
"author": "greengerong",
"title": "ComfyUI-Lumina-Video",
"reference": "https://github.com/greengerong/ComfyUI-Lumina-Video",
"files": [
"https://github.com/greengerong/ComfyUI-Lumina-Video"
],
"install_type": "git-clone",
"description": "This is a video generation plugin implementation for ComfyUI based on the Lumina Video model."
},
{
"author": "raindrop313",
"title": "ComfyUI_SD3_Flowedit",
@@ -20799,17 +20991,17 @@
"https://github.com/mediocreatmybest/ComfyUI-Transformers-Pipeline"
],
"install_type": "git-clone",
"description": "Some additional ComfyUI nodes allowing tasks via the Huggingface Transformers Pipeline."
"description": "Additional ComfyUI nodes to utilise the Transformers pipeline in a simple and modular way."
},
{
"author": "iris-Neko",
"title": "ComfyUI_ascii_art",
"reference": "https://github.com/iris-Neko/ComfyUI_ascii_art",
"author": "IrisRainbowNeko",
"title": "ascii-art-comfyui",
"reference": "https://github.com/Deep-Neko/ComfyUI_ascii_art",
"files": [
"https://github.com/iris-Neko/ComfyUI_ascii_art"
"https://github.com/Deep-Neko/ComfyUI_ascii_art"
],
"install_type": "git-clone",
"description": "ComfyUI node for [a/ASCII art controlnet](https://civitai.com/models/986392)"
"description": "ascii art preprocessors in ComfyUI"
},
{
"author": "mie",
@@ -20823,13 +21015,13 @@
},
{
"author": "mie",
"title": "ComfyUI_JanusProCaption",
"reference": "https://github.com/MieMieeeee/ComfyUI-JanusProCaption",
"title": "ComfyUI_CaptionThis",
"reference": "https://github.com/MieMieeeee/ComfyUI-CaptionThis",
"files": [
"https://github.com/MieMieeeee/ComfyUI-JanusProCaption"
"https://github.com/MieMieeeee/ComfyUI-CaptionThis"
],
"install_type": "git-clone",
"description": "Describe image or create caption files using Janus Pro Model"
"description": "Describe a single image or all images in a directory using models such as Janus Pro, Florence2 (coming soon), or JoyCaption (coming soon), with a particular focus on building datasets for training LoRA."
},
{
"author": "lum3on",
@@ -20872,6 +21064,16 @@
"install_type": "git-clone",
"description": "This node calculates a contrasting complementary color based on an input RGB color. The goal is to ensure visibility and contrast when overlaying text, UI elements, or graphical components against a given background color."
},
{
"author": "dasilva333",
"title": "ComfyUI_MarkdownImage",
"reference": "https://github.com/dasilva333/ComfyUI_MarkdownImage",
"files": [
"https://github.com/dasilva333/ComfyUI_MarkdownImage"
],
"install_type": "git-clone",
"description": "This project generates an image from Markdown text using imgkit and wkhtmltoimage. It automatically scales the text to fit within the specified image dimensions."
},
{
"author": "moon7star9",
"title": "ComfyUI_BiRefNet_Universal",
@@ -20882,109 +21084,301 @@
"install_type": "git-clone",
"description": "A comprehensive node package that seamlessly integrates all BiRefNet series models into ComfyUI"
},
{
"author": "wirytiox",
"title": "ComfyUI-SelectStringFromListWithIndex",
"id": "ComfyUI-SelectStringFromListWithIndex",
"reference": "https://github.com/wirytiox/ComfyUI-SelectStringFromListWithIndex",
"files": [
"https://github.com/wirytiox/ComfyUI-SelectStringFromListWithIndex"
],
"install_type": "git-clone",
"description": "This node is a node made by GMapeSplat/ComfyUI_ezXY that i copied while his node doesn't work"
},
{
"author": "TheAIDoctor",
"title": "The AI Doctors Clinical Tools",
"id": "The-AI-Doctors-Clinical-Tools",
"reference": "https://github.com/BlueprintCoding/ComfyUI_AIDocsClinicalTools",
"files": [
"https://github.com/BlueprintCoding/ComfyUI_AIDocsClinicalTools"
],
"install_type": "git-clone",
"description": "Nodes: Multi Int and Multi Text; allows for the creation of multiple int and multiple string storage and output from a single node. Multi Float coming soon."
},
{
"author": "Mohammadreza Mohseni",
"title": "ComfyUI Mohseni Kit",
"id": "mohseni-kit",
"reference": "https://github.com/mohseni-mr/ComfyUI-Mohseni-Kit",
"files": [
"https://github.com/mohseni-mr/ComfyUI-Mohseni-Kit"
],
"install_type": "git-clone",
"description": "A collection of useful nodes for ComfyUI, including Float Preview for live image visualization."
},
{
"author": "BuffMcBigHuge",
"title": "ComfyUI-Zonos",
"reference": "https://github.com/BuffMcBigHuge/ComfyUI-Zonos",
"files": [
"https://github.com/BuffMcBigHuge/ComfyUI-Zonos"
],
"install_type": "git-clone",
"description": "TTS with Zyphra Zonos"
},
{
"author": "BahaC",
"title": "ComfyUI Zonos TTS Node",
"reference": "https://github.com/BahaC/ComfyUI-ZonosTTS",
"files": [
"https://github.com/BahaC/ComfyUI-ZonosTTS"
],
"install_type": "git-clone",
"description": "A ComfyUI custom node that brings Zonos Text-to-Speech capabilities to your workflows, featuring high-quality speech synthesis and voice cloning."
},
{
"author": "dzqdzq",
"title": "ComfyUI-crop-alpha",
"reference": "https://github.com/dzqdzq/ComfyUI-crop-alpha",
"files": [
"https://github.com/dzqdzq/ComfyUI-crop-alpha"
],
"install_type": "git-clone",
"description": "Automatic cropping of transparent areas to prevent images from being too large, while also supporting resizing to prevent image dimensions from being too large."
},
{
"author": "bbtaivi",
"title": "AIV ComfyUI Node",
"reference": "https://github.com/bbtaivi/ComfyUI-Aiv-Param",
"files": [
"https://github.com/bbtaivi/ComfyUI-Aiv-Param"
],
"install_type": "git-clone",
"description": "Used to convert workflow node settings into AIV mini-program parameters."
},
{
"author": "PrunaAI",
"title": "Pruna nodes for ComfyUI",
"reference": "https://github.com/PrunaAI/ComfyUI_pruna",
"files": [
"https://github.com/PrunaAI/ComfyUI_pruna"
],
"install_type": "git-clone",
"description": "This repository explains how to accelerate image generation in ComfyUI using Pruna, an inference optimization engine that makes AI models faster, smaller, cheaper, and greener. ComfyUI is a popular node-based GUI for image generation models, for which we provide a custom compilation node that accelerates Stable Diffusion (SD) and Flux inference, while preserving output quality."
},
{
"author": "Hellfiredragon",
"title": "comfyui-image-manipulation",
"reference": "https://github.com/Hellfiredragon/comfyui-image-manipulation",
"files": [
"https://github.com/Hellfiredragon/comfyui-image-manipulation"
],
"install_type": "git-clone",
"description": "Custom nodes to manipulate images in ComfyUI"
},
{
"author": "lunarring",
"title": "bitalino_comfy",
"reference": "https://github.com/lunarring/bitalino_comfy",
"files": [
"https://github.com/lunarring/bitalino_comfy"
],
"install_type": "git-clone",
"description": "A package implementing a Bitalino device ComfyUI custom node."
},
{
"author": "AIDC-AI",
"title": "ComfyUI-Copilot",
"id": "ComfyUI-Copilot",
"reference": "https://github.com/AIDC-AI/ComfyUI-Copilot",
"files": [
"https://github.com/AIDC-AI/ComfyUI-Copilot"
],
"install_type": "git-clone",
"description": "Your Intelligent Assistant for Comfy-UI."
},
{
"author": "attashe",
"title": "ComfyUI-FluxRegionAttention",
"reference": "https://github.com/attashe/ComfyUI-FluxRegionAttention",
"files": [
"https://github.com/attashe/ComfyUI-FluxRegionAttention"
],
"install_type": "git-clone",
"description": "Implement Region Attention for Flux model. Add node RegionAttention that takes a regions - mask + condition, mask could be set from comfyui masks or bbox in FluxRegionBBOX node.\nThis code is not optimized and has a memory leak. If you caught a OOM just try run a query againg - works on my RTX3080. For generation it uses a usual prompt that have influence to all picture and a regions that have their own prompts.\nBase prompt good for setup background and style of image. This is train-free technique and results not always stable - sometimes need to try several seeds or change prompt."
},
{
"author": "RodrigoSKohl",
"title": "Interior Design for Comfyui",
"reference": "https://github.com/RodrigoSKohl/StableDesign-for-ComfyUI",
"files": [
"https://github.com/RodrigoSKohl/StableDesign-for-ComfyUI"
],
"install_type": "git-clone",
"description": "This node is based on MykolaL/StableDesign"
},
{
"author": "yas-ponotech",
"title": "ComfyUI-Stability-AI-API",
"reference": "https://github.com/yhayano-ponotech/comfyui-stability-ai-api",
"files": [
"https://github.com/yhayano-ponotech/comfyui-stability-ai-api"
],
"install_type": "git-clone",
"description": "A collection of custom nodes for using the Stability AI API in ComfyUI."
},
{
"author": "HJH-AILab",
"title": "ComfyUI_StableAnimator",
"reference": "https://github.com/HJH-AILab/ComfyUI_StableAnimator",
"files": [
"https://github.com/HJH-AILab/ComfyUI_StableAnimator"
],
"install_type": "git-clone",
"description": "ComfyUI nodes for StableAnimator"
},
{
"author": "Easymode-ai",
"title": "ComfyUI-ShadowR",
"reference": "https://github.com/Easymode-ai/ComfyUI-ShadowR",
"files": [
"https://github.com/Easymode-ai/ComfyUI-ShadowR"
],
"install_type": "git-clone",
"description": "ComfyUI [a/movingforward100/Shadow_R](https://github.com/movingforward100/Shadow_R) Wrapper"
},
{
"author": "GamingDaveUk",
"title": "Daves Nodes",
"id": "davesnodes",
"reference": "https://github.com/GamingDaveUk/daves_nodes",
"files": [
"https://github.com/GamingDaveUk/daves_nodes"
],
"install_type": "git-clone",
"description": "Nodes that I needed but couldnt find, so ended up making."
},
{
"author": "chenlongming",
"title": "ComfyUI_Spectral",
"reference": "https://github.com/chenlongming/ComfyUI_Spectral",
"files": [
"https://github.com/chenlongming/ComfyUI_Spectral"
],
"install_type": "git-clone",
"description": "ComfyUI Spectral is a ComfyUI custom nodes library based on the spectral, mainly used for visual processing of spectral files"
},
{
"author": "Chengym2023",
"title": "ComfyUI-DeepSeek_Online",
"reference": "https://github.com/Chengym2023/ComfyUI-DeepSeek_Online",
"files": [
"https://github.com/Chengym2023/ComfyUI-DeepSeek_Online"
],
"install_type": "git-clone",
"description": "NODES: SiliconCloudReasoning, DeepSeekOnline"
},
{
"author": "gitmylo",
"title": "Audio nodes",
"reference": "https://github.com/gitmylo/ComfyUI-audio-nodes",
"files": [
"https://github.com/gitmylo/ComfyUI-audio-nodes"
],
"install_type": "git-clone",
"description": "Various nodes related to audio."
},
{
"author": "aicuai",
"title": "aicu-comfyui-stability-ai-api",
"reference": "https://github.com/aicuai/aicu-comfyui-stability-ai-api",
"files": [
"https://github.com/aicuai/aicu-comfyui-stability-ai-api"
],
"install_type": "git-clone",
"description": "This repository contains custom nodes for Stability AI API which supports SD3.0 and 3.5."
},
{
"author": "benda1989",
"title": "CosyVoice2 for ComfyUI",
"reference": "https://github.com/benda1989/CosyVoice2_ComfyUI",
"files": [
"https://github.com/benda1989/CosyVoice2_ComfyUI"
],
"install_type": "git-clone",
"description": "A plugin of ComfyUI for CosyVoice2, one component for text to Sonic Video"
},
{
"author": "benda1989",
"title": "GKK·Sonic",
"reference": "https://github.com/benda1989/Sonic_ComfyUI",
"files": [
"https://github.com/benda1989/Sonic_ComfyUI"
],
"install_type": "git-clone",
"description": "a plugin of ComfyUI for Long Sonic"
},
{
"author": "morgan55555",
"title": "ComfyUI Lock Mode",
"reference": "https://github.com/morgan55555/comfyui-lock-mode",
"files": [
"https://github.com/morgan55555/comfyui-lock-mode"
],
"install_type": "git-clone",
"description": "Lock Mode feature for ComfyUI. Make simple no-code UI easily."
},
{
"author": "pathway8-sudo",
"title": "ComfyUI-Pathway-CutPNG-Node",
"reference": "https://github.com/pathway8-sudo/ComfyUI-Pathway-CutPNG-Node",
"files": [
"https://github.com/pathway8-sudo/ComfyUI-Pathway-CutPNG-Node"
],
"install_type": "git-clone",
"description": "Custom ComfyUI node that uses BRIA RMBG v1.4 for background removal and PNG cutting."
},
{
"author": "crave33",
"title": "RenesStuffDanboruTagGet",
"reference": "https://github.com/crave33/RenesStuffDanboruTagGet",
"files": [
"https://github.com/crave33/RenesStuffDanboruTagGet"
],
"install_type": "git-clone",
"description": "generate tags / prompt from danboru image_id input"
},
{
"author": "MeeeyoAI",
"title": "ComfyUI_StringOps",
"reference": "https://github.com/MeeeyoAI/ComfyUI_StringOps",
"files": [
"https://github.com/MeeeyoAI/ComfyUI_StringOps"
],
"install_type": "git-clone",
"description": "StringOps is a versatile text processing toolkit built for ComfyUI's node-based workflows"
},
{
"author": "svetozarov",
"title": "AS_GeminiCaptioning Node",
"reference": "https://github.com/svetozarov/AS_GeminiCaptioning",
"files": [
"https://github.com/svetozarov/AS_GeminiCaptioning"
],
"install_type": "git-clone",
"description": "A ComfyUI node that combines an image with simple text parameters to create a prompt, sends it to the Google Gemini API via the google-generativeai SDK, and returns the generated text response along with the original prompt and an execution log"
},
{
"author": "Pablerdo",
"title": "ComfyUI-MultiCutAndDrag",
"reference": "https://github.com/Pablerdo/ComfyUI-MultiCutAndDrag",
"files": [
"https://github.com/Pablerdo/ComfyUI-MultiCutAndDrag"
],
"install_type": "git-clone",
"description": "Cut and and drag that allows you to cut and drag multiple images on a path"
},
@@ -21139,7 +21533,7 @@
"https://raw.githubusercontent.com/tudal/Hakkun-ComfyUI-nodes/main/hakkun_nodes.py"
],
"install_type": "copy",
"description": "Nodes: Prompt parser. ComfyUI extra nodes. Mostly prompt parsing."
"description": "Mainly its prompt generating by custom syntax. Prompt Parser, Prompt tags, Random Line, Calculate Upscale, Image size to string, Type Converter, Image Resize To Height/Width, Load Random Image, Load Text"
},
{
"author": "SadaleNet",

View File

File diff suppressed because it is too large Load Diff

View File

@@ -154,14 +154,27 @@ def switch_to_default_branch(repo):
repo.git.checkout(default_branch)
return True
except:
# try checkout master
# try checkout main if failed
try:
repo.git.checkout(repo.heads.master)
return True
except:
try:
if remote_name is not None:
repo.git.checkout('-b', 'master', f'{remote_name}/master')
return True
except:
pass
try:
repo.git.checkout(repo.heads.main)
return True
except:
try:
if remote_name is not None:
repo.git.checkout('-b', 'main', f'{remote_name}/main')
return True
except:
pass
print("[ComfyUI Manager] Failed to switch to the default branch")
return False

View File

File diff suppressed because it is too large Load Diff

View File

@@ -173,7 +173,10 @@ def read_cnr_info(fullpath):
project = data.get('project', {})
name = project.get('name').strip().lower()
version = project.get('version')
# normalize version
# for example: 2.5 -> 2.5.0
version = str(manager_util.StrictVersion(project.get('version')))
urls = project.get('urls', {})
repository = urls.get('Repository')

View File

@@ -2,6 +2,9 @@ import os
import configparser
GITHUB_ENDPOINT = os.getenv('GITHUB_ENDPOINT')
def is_git_repo(path: str) -> bool:
""" Check if the path is a git repository. """
# NOTE: Checking it through `git.Repo` must be avoided.
@@ -37,7 +40,8 @@ def git_url(fullpath):
if not os.path.exists(git_config_path):
return None
config = configparser.ConfigParser()
# Set `strict=False` to allow duplicate `vscode-merge-base` sections, addressing <https://github.com/ltdrdata/ComfyUI-Manager/issues/1529>
config = configparser.ConfigParser(strict=False)
config.read(git_config_path)
for k, v in config.items():
@@ -46,16 +50,36 @@ def git_url(fullpath):
return None
def normalize_url(url) -> str:
url = url.replace("git@github.com:", "https://github.com/")
if url.endswith('.git'):
url = url[:-4]
github_id = normalize_to_github_id(url)
if github_id is not None:
url = f"https://github.com/{github_id}"
return url
def normalize_url_http(url) -> str:
url = url.replace("https://github.com/", "git@github.com:")
if url.endswith('.git'):
url = url[:-4]
return url
def normalize_to_github_id(url) -> str:
if 'github' in url or (GITHUB_ENDPOINT is not None and GITHUB_ENDPOINT in url):
author = os.path.basename(os.path.dirname(url))
if author.startswith('git@github.com:'):
author = author.split(':')[1]
repo_name = os.path.basename(url)
if repo_name.endswith('.git'):
repo_name = repo_name[:-4]
return f"{author}/{repo_name}"
return None
def get_url_for_clone(url):
url = normalize_url(url)
if GITHUB_ENDPOINT is not None and url.startswith('https://github.com/'):
url = GITHUB_ENDPOINT + url[18:] # url[18:] -> remove `https://github.com`
return url

View File

@@ -42,7 +42,7 @@ import manager_downloader
from node_package import InstalledNodePackage
version_code = [3, 21, 1]
version_code = [3, 26]
version_str = f"V{version_code[0]}.{version_code[1]}" + (f'.{version_code[2]}' if len(version_code) > 2 else '')
@@ -177,6 +177,7 @@ manager_channel_list_path = None
manager_startup_script_path:str = None
manager_snapshot_path = None
manager_pip_overrides_path = None
manager_pip_blacklist_path = None
manager_components_path = None
def update_user_directory(user_dir):
@@ -186,6 +187,7 @@ def update_user_directory(user_dir):
global manager_startup_script_path
global manager_snapshot_path
global manager_pip_overrides_path
global manager_pip_blacklist_path
global manager_components_path
manager_files_path = os.path.abspath(os.path.join(user_dir, 'default', 'ComfyUI-Manager'))
@@ -203,6 +205,7 @@ def update_user_directory(user_dir):
manager_config_path = os.path.join(manager_files_path, 'config.ini')
manager_channel_list_path = os.path.join(manager_files_path, 'channels.list')
manager_pip_overrides_path = os.path.join(manager_files_path, "pip_overrides.json")
manager_pip_blacklist_path = os.path.join(manager_files_path, "pip_blacklist.list")
manager_components_path = os.path.join(manager_files_path, "components")
manager_util.cache_dir = os.path.join(manager_files_path, "cache")
@@ -345,6 +348,7 @@ class ManagedResult:
self.msg = None
self.target = None
self.postinstall = lambda: True
self.ver = None
def append(self, item):
self.items.append(item)
@@ -366,6 +370,10 @@ class ManagedResult:
self.postinstall = postinstall
return self
def with_ver(self, ver):
self.ver = ver
return self
class UnifiedManager:
def __init__(self):
@@ -497,6 +505,8 @@ class UnifiedManager:
def resolve_from_path(self, fullpath):
url = git_utils.git_url(fullpath)
if url:
url = git_utils.normalize_url(url)
cnr = self.get_cnr_by_repo(url)
commit_hash = git_utils.get_commit_hash(fullpath)
if cnr:
@@ -513,7 +523,10 @@ class UnifiedManager:
if info:
cnr = self.cnr_map.get(info['id'])
if cnr:
return {'id': cnr['id'], 'cnr': cnr, 'ver': info['version']}
# normalize version
# for example: 2.5 -> 2.5.0
ver = str(manager_util.StrictVersion(info['version']))
return {'id': cnr['id'], 'cnr': cnr, 'ver': ver}
else:
return None
else:
@@ -525,6 +538,8 @@ class UnifiedManager:
if node_package.is_disabled and node_package.is_unknown:
url = git_utils.git_url(node_package.fullpath)
if url is not None:
url = git_utils.normalize_url(url)
self.unknown_inactive_nodes[node_package.id] = (url, node_package.fullpath)
if node_package.is_disabled and node_package.is_nightly:
@@ -535,6 +550,8 @@ class UnifiedManager:
if node_package.is_enabled and node_package.is_unknown:
url = git_utils.git_url(node_package.fullpath)
if url is not None:
url = git_utils.normalize_url(url)
self.unknown_active_nodes[node_package.id] = (url, node_package.fullpath)
if node_package.is_from_cnr and node_package.is_disabled:
@@ -786,6 +803,7 @@ class UnifiedManager:
node_id = v['id']
else:
node_id = v['files'][0].split('/')[-1]
v['repository'] = v['files'][0]
res[node_id] = v
elif len(v['files']) > 1:
res[v['files'][0]] = v # A custom node composed of multiple url is treated as a single repository with one representative path
@@ -812,14 +830,14 @@ class UnifiedManager:
print("Install: pip packages")
pip_fixer = manager_util.PIPFixer(manager_util.get_installed_packages())
res = True
with open(requirements_path, "r") as requirements_file:
for line in requirements_file:
package_name = remap_pip_package(line.strip())
if package_name and not package_name.startswith('#') and package_name not in self.processed_install:
self.processed_install.add(package_name)
install_cmd = manager_util.make_pip_cmd(["install", package_name])
if package_name.strip() != "" and not package_name.startswith('#'):
res = res and try_install_script(url, repo_path, install_cmd, instant_execution=instant_execution)
lines = manager_util.robust_readlines(requirements_path)
for line in lines:
package_name = remap_pip_package(line.strip())
if package_name and not package_name.startswith('#') and package_name not in self.processed_install:
self.processed_install.add(package_name)
install_cmd = manager_util.make_pip_cmd(["install", package_name])
if package_name.strip() != "" and not package_name.startswith('#'):
res = res and try_install_script(url, repo_path, install_cmd, instant_execution=instant_execution)
pip_fixer.fix_broken()
return res
@@ -1045,8 +1063,8 @@ class UnifiedManager:
# update cache
if version_spec == 'unknown':
self.unknown_active_nodes[node_id] = self.unknown_inactive_nodes[node_id][0], to_path
del self.unknown_inactive_nodes[node_id]
self.unknown_active_nodes[node_id] = to_path
return result.with_target(to_path)
elif version_spec == 'nightly':
del self.nightly_inactive_nodes[node_id]
@@ -1227,15 +1245,16 @@ class UnifiedManager:
if url.endswith("/"):
url = url[:-1]
try:
print(f"Download: git clone '{url}'")
# Clone the repository from the remote URL
clone_url = git_utils.get_url_for_clone(url)
print(f"Download: git clone '{clone_url}'")
if not instant_execution and platform.system() == 'Windows':
res = manager_funcs.run_script([sys.executable, git_script_path, "--clone", get_default_custom_nodes_path(), url, repo_path], cwd=get_default_custom_nodes_path())
res = manager_funcs.run_script([sys.executable, git_script_path, "--clone", get_default_custom_nodes_path(), clone_url, repo_path], cwd=get_default_custom_nodes_path())
if res != 0:
return result.fail(f"Failed to clone repo: {url}")
return result.fail(f"Failed to clone repo: {clone_url}")
else:
repo = git.Repo.clone_from(url, repo_path, recursive=True, progress=GitProgress())
repo = git.Repo.clone_from(clone_url, repo_path, recursive=True, progress=GitProgress())
repo.git.clear_cache()
repo.close()
@@ -1249,7 +1268,8 @@ class UnifiedManager:
return result.fail(f"Failed to execute install script: {url}")
except Exception as e:
return result.fail(f"Install(git-clone) error: {url} / {e}")
traceback.print_exc()
return result.fail(f"Install(git-clone) error[2]: {url} / {e}")
print("Installation was successful.")
return result
@@ -1330,14 +1350,14 @@ class UnifiedManager:
version_spec = self.resolve_unspecified_version(node_id, guess_mode='active')
if version_spec is None:
return ManagedResult('update').fail(f'Update not available: {node_id}@{version_spec}')
return ManagedResult('update').fail(f'Update not available: {node_id}@{version_spec}').with_ver(version_spec)
if version_spec == 'nightly':
return self.repo_update(self.active_nodes[node_id][1], instant_execution=instant_execution, no_deps=no_deps, return_postinstall=return_postinstall).with_target('nightly')
return self.repo_update(self.active_nodes[node_id][1], instant_execution=instant_execution, no_deps=no_deps, return_postinstall=return_postinstall).with_target('nightly').with_ver('nightly')
elif version_spec == 'unknown':
return self.repo_update(self.unknown_active_nodes[node_id][1], instant_execution=instant_execution, no_deps=no_deps, return_postinstall=return_postinstall).with_target('unknown')
return self.repo_update(self.unknown_active_nodes[node_id][1], instant_execution=instant_execution, no_deps=no_deps, return_postinstall=return_postinstall).with_target('unknown').with_ver('unknown')
else:
return self.cnr_switch_version(node_id, instant_execution=instant_execution, no_deps=no_deps, return_postinstall=return_postinstall)
return self.cnr_switch_version(node_id, instant_execution=instant_execution, no_deps=no_deps, return_postinstall=return_postinstall).with_ver('cnr')
async def install_by_id(self, node_id, version_spec=None, channel=None, mode=None, instant_execution=False, no_deps=False, return_postinstall=False):
"""
@@ -1385,7 +1405,7 @@ class UnifiedManager:
res = self.repo_install(repo_url, to_path, instant_execution=instant_execution, no_deps=no_deps, return_postinstall=return_postinstall)
if res.result:
if version_spec == 'unknown':
self.unknown_active_nodes[node_id] = to_path
self.unknown_active_nodes[node_id] = repo_url, to_path
elif version_spec == 'nightly':
cnr_utils.generate_cnr_id(to_path, node_id)
self.active_nodes[node_id] = 'nightly', to_path
@@ -1452,7 +1472,7 @@ def identify_node_pack_from_path(fullpath):
# cnr
cnr = cnr_utils.read_cnr_info(fullpath)
if cnr is not None:
return module_name, cnr['version'], cnr['id']
return module_name, cnr['version'], cnr['id'], None
return None
else:
@@ -1460,10 +1480,18 @@ def identify_node_pack_from_path(fullpath):
cnr_id = cnr_utils.read_cnr_id(fullpath)
commit_hash = git_utils.get_commit_hash(fullpath)
github_id = git_utils.normalize_to_github_id(repo_url)
if github_id is None:
try:
github_id = os.path.basename(repo_url)
except:
logging.warning(f"[ComfyUI-Manager] unexpected repo url: {repo_url}")
github_id = module_name
if cnr_id is not None:
return module_name, commit_hash, cnr_id
return module_name, commit_hash, cnr_id, github_id
else:
return module_name, commit_hash, ''
return module_name, commit_hash, '', github_id
def get_installed_node_packs():
@@ -1481,7 +1509,7 @@ def get_installed_node_packs():
is_disabled = not y.endswith('.disabled')
res[info[0]] = { 'ver': info[1], 'cnr_id': info[2], 'enabled': is_disabled }
res[info[0]] = { 'ver': info[1], 'cnr_id': info[2], 'aux_id': info[3], 'enabled': is_disabled }
disabled_dirs = os.path.join(x, '.disabled')
if os.path.exists(disabled_dirs):
@@ -1494,7 +1522,7 @@ def get_installed_node_packs():
if info is None:
continue
res[info[0]] = { 'ver': info[1], 'cnr_id': info[2], 'enabled': False }
res[info[0]] = { 'ver': info[1], 'cnr_id': info[2], 'aux_id': info[3], 'enabled': False }
return res
@@ -1550,7 +1578,7 @@ manager_funcs = ManagerFuncs()
def write_config():
config = configparser.ConfigParser()
config = configparser.ConfigParser(strict=False)
config['default'] = {
'preview_method': manager_funcs.get_current_preview_method(),
@@ -1561,6 +1589,7 @@ def write_config():
'bypass_ssl': get_config()['bypass_ssl'],
"file_logging": get_config()['file_logging'],
'component_policy': get_config()['component_policy'],
'update_policy': get_config()['update_policy'],
'windows_selector_event_loop_policy': get_config()['windows_selector_event_loop_policy'],
'model_download_by_agent': get_config()['model_download_by_agent'],
'downgrade_blacklist': get_config()['downgrade_blacklist'],
@@ -1580,19 +1609,9 @@ def write_config():
def read_config():
try:
config = configparser.ConfigParser()
config = configparser.ConfigParser(strict=False)
config.read(manager_config_path)
default_conf = config['default']
# policy migration: disable_unsecure_features -> security_level
if 'disable_unsecure_features' in default_conf:
if default_conf['disable_unsecure_features'].lower() == 'true':
security_level = 'strong'
else:
security_level = 'normal'
else:
security_level = default_conf['security_level'] if 'security_level' in default_conf else 'normal'
manager_util.use_uv = default_conf['use_uv'].lower() == 'true' if 'use_uv' in default_conf else False
def get_bool(key, default_value):
@@ -1600,22 +1619,23 @@ def read_config():
return {
'http_channel_enabled': get_bool('http_channel_enabled', False),
'preview_method': default_conf.get('preview_method', manager_funcs.get_current_preview_method()),
'preview_method': default_conf.get('preview_method', manager_funcs.get_current_preview_method()).lower(),
'git_exe': default_conf.get('git_exe', ''),
'use_uv': get_bool('use_uv', False),
'channel_url': default_conf.get('channel_url', DEFAULT_CHANNEL),
'default_cache_as_channel_url': get_bool('default_cache_as_channel_url', False),
'share_option': default_conf.get('share_option', 'all'),
'share_option': default_conf.get('share_option', 'all').lower(),
'bypass_ssl': get_bool('bypass_ssl', False),
'file_logging': get_bool('file_logging', True),
'component_policy': default_conf.get('component_policy', 'workflow'),
'component_policy': default_conf.get('component_policy', 'workflow').lower(),
'update_policy': default_conf.get('update_policy', 'stable-comfyui').lower(),
'windows_selector_event_loop_policy': get_bool('windows_selector_event_loop_policy', False),
'model_download_by_agent': get_bool('model_download_by_agent', False),
'downgrade_blacklist': default_conf.get('downgrade_blacklist', ''),
'downgrade_blacklist': default_conf.get('downgrade_blacklist', '').lower(),
'skip_migration_check': get_bool('skip_migration_check', False),
'always_lazy_install': get_bool('always_lazy_install', False),
'network_mode': default_conf.get('network_mode', 'public'),
'security_level': security_level,
'network_mode': default_conf.get('network_mode', 'public').lower(),
'security_level': default_conf.get('security_level', 'normal').lower(),
}
except Exception:
@@ -1631,6 +1651,7 @@ def read_config():
'bypass_ssl': False,
'file_logging': True,
'component_policy': 'workflow',
'update_policy': 'stable-comfyui',
'windows_selector_event_loop_policy': False,
'model_download_by_agent': False,
'downgrade_blacklist': '',
@@ -1682,14 +1703,27 @@ def switch_to_default_branch(repo):
repo.git.checkout(default_branch)
return True
except:
# try checkout master
# try checkout main if failed
try:
repo.git.checkout(repo.heads.master)
return True
except:
try:
if remote_name is not None:
repo.git.checkout('-b', 'master', f'{remote_name}/master')
return True
except:
pass
try:
repo.git.checkout(repo.heads.main)
return True
except:
try:
if remote_name is not None:
repo.git.checkout('-b', 'main', f'{remote_name}/main')
return True
except:
pass
print("[ComfyUI Manager] Failed to switch to the default branch")
return False
@@ -1723,7 +1757,7 @@ def try_install_script(url, repo_path, install_cmd, instant_execution=False):
if platform.system() != "Windows":
try:
if comfy_ui_commit_datetime.date() < comfy_ui_required_commit_datetime.date():
if not os.environ.get('__COMFYUI_DESKTOP_VERSION__') and comfy_ui_commit_datetime.date() < comfy_ui_required_commit_datetime.date():
print("\n\n###################################################################")
print(f"[WARN] ComfyUI-Manager: Your ComfyUI version ({comfy_ui_revision})[{comfy_ui_commit_datetime.date()}] is too old. Please update to the latest version.")
print("[WARN] The extension installation feature may not work properly in the current installed ComfyUI version on Windows environment.")
@@ -2040,12 +2074,14 @@ async def gitclone_install(url, instant_execution=False, msg_prefix='', no_deps=
print(f"CLONE into '{repo_path}'")
# Clone the repository from the remote URL
clone_url = git_utils.get_url_for_clone(url)
if not instant_execution and platform.system() == 'Windows':
res = manager_funcs.run_script([sys.executable, git_script_path, "--clone", get_default_custom_nodes_path(), url, repo_path], cwd=get_default_custom_nodes_path())
res = manager_funcs.run_script([sys.executable, git_script_path, "--clone", get_default_custom_nodes_path(), clone_url, repo_path], cwd=get_default_custom_nodes_path())
if res != 0:
return result.fail(f"Failed to clone '{url}' into '{repo_path}'")
return result.fail(f"Failed to clone '{clone_url}' into '{repo_path}'")
else:
repo = git.Repo.clone_from(url, repo_path, recursive=True, progress=GitProgress())
repo = git.Repo.clone_from(clone_url, repo_path, recursive=True, progress=GitProgress())
repo.git.clear_cache()
repo.close()
@@ -2055,8 +2091,8 @@ async def gitclone_install(url, instant_execution=False, msg_prefix='', no_deps=
except Exception as e:
traceback.print_exc()
print(f"Install(git-clone) error: {url} / {e}", file=sys.stderr)
return result.fail(f"Install(git-clone) error: {url} / {e}")
print(f"Install(git-clone) error[1]: {url} / {e}", file=sys.stderr)
return result.fail(f"Install(git-clone)[1] error: {url} / {e}")
def git_pull(path):
@@ -2155,7 +2191,7 @@ def gitclone_fix(files, instant_execution=False, no_deps=False):
return False
except Exception as e:
print(f"Install(git-clone) error: {url} / {e}", file=sys.stderr)
print(f"Fix(git-clone) error: {url} / {e}", file=sys.stderr)
return False
print(f"Attempt to fixing '{files}' is done.")
@@ -2339,6 +2375,32 @@ def gitclone_update(files, instant_execution=False, skip_script=False, msg_prefi
return True
def update_to_stable_comfyui(repo_path):
try:
repo = git.Repo(repo_path)
repo.git.checkout(repo.heads.master)
versions, current_tag, _ = get_comfyui_versions(repo)
if len(versions) == 0 or (len(versions) == 1 and versions[0] == 'nightly'):
logging.info("[ComfyUI-Manager] Unable to update to the stable ComfyUI version.")
return "fail", None
if versions[0] == 'nightly':
latest_tag = versions[1]
else:
latest_tag = versions[0]
if current_tag == latest_tag:
return "skip", None
else:
logging.info(f"[ComfyUI-Manager] Updating ComfyUI: {current_tag} -> {latest_tag}")
repo.git.checkout(latest_tag)
return 'updated', latest_tag
except:
traceback.print_exc()
return "fail", None
def update_path(repo_path, instant_execution=False, no_deps=False):
if not os.path.exists(os.path.join(repo_path, '.git')):
return "fail"
@@ -2346,9 +2408,12 @@ def update_path(repo_path, instant_execution=False, no_deps=False):
# version check
repo = git.Repo(repo_path)
is_switched = False
if repo.head.is_detached:
if not switch_to_default_branch(repo):
return "fail"
else:
is_switched = True
current_branch = repo.active_branch
branch_name = current_branch.name
@@ -2387,6 +2452,8 @@ def update_path(repo_path, instant_execution=False, no_deps=False):
git_pull(repo_path)
execute_install_script("ComfyUI", repo_path, instant_execution=instant_execution, no_deps=no_deps)
return "updated"
elif is_switched:
return "updated"
else:
return "skipped"
@@ -2697,9 +2764,6 @@ async def extract_nodes_from_workflow(filepath, mode='local', channel_url='defau
if ext == 'https://github.com/comfyanonymous/ComfyUI':
pass
elif ext is not None:
if 'Fooocus' in ext:
print(f">> {node_name}")
used_exts.add(ext)
else:
unknown_nodes.add(node_name)
@@ -2970,7 +3034,14 @@ async def restore_snapshot(snapshot_path, git_helper_extras=None):
print("cm-cli: unexpected [0001]")
# for nightly restore
git_info = info.get('git_custom_nodes')
_git_info = info.get('git_custom_nodes')
git_info = {}
# normalize github repo
for k, v in _git_info.items():
norm_k = git_utils.normalize_url(k)
git_info[norm_k] = v
if git_info is not None:
todo_disable = []
todo_enable = []
@@ -2983,20 +3054,13 @@ async def restore_snapshot(snapshot_path, git_helper_extras=None):
if v[0] == 'nightly' and cnr_repo_map.get(k):
repo_url = cnr_repo_map.get(k)
normalized_url = git_utils.normalize_url(repo_url)
normalized_url1 = git_utils.normalize_url(repo_url)
normalized_url2 = git_utils.normalize_url_http(repo_url)
if normalized_url1 not in git_info and normalized_url2 not in git_info:
if normalized_url not in git_info:
todo_disable.append(k)
else:
if normalized_url1 in git_info:
commit_hash = git_info[normalized_url1]['hash']
todo_checkout.append((v[1], commit_hash))
if normalized_url2 in git_info:
commit_hash = git_info[normalized_url2]['hash']
todo_checkout.append((v[1], commit_hash))
commit_hash = git_info[normalized_url]['hash']
todo_checkout.append((v[1], commit_hash))
for k, v in unified_manager.nightly_inactive_nodes.items():
if 'comfyui-manager' in k:
@@ -3004,18 +3068,12 @@ async def restore_snapshot(snapshot_path, git_helper_extras=None):
if cnr_repo_map.get(k):
repo_url = cnr_repo_map.get(k)
normalized_url1 = git_utils.normalize_url(repo_url)
normalized_url2 = git_utils.normalize_url_http(repo_url)
normalized_url = git_utils.normalize_url(repo_url)
if normalized_url1 in git_info:
commit_hash = git_info[normalized_url1]['hash']
if normalized_url in git_info:
commit_hash = git_info[normalized_url]['hash']
todo_enable.append((k, commit_hash))
processed_urls.append(normalized_url1)
if normalized_url2 in git_info:
commit_hash = git_info[normalized_url2]['hash']
todo_enable.append((k, commit_hash))
processed_urls.append(normalized_url2)
processed_urls.append(normalized_url)
for x in todo_disable:
unified_manager.unified_disable(x, False)
@@ -3068,21 +3126,14 @@ async def restore_snapshot(snapshot_path, git_helper_extras=None):
if repo_url is None:
continue
normalized_url1 = git_utils.normalize_url(repo_url)
normalized_url2 = git_utils.normalize_url_http(repo_url)
normalized_url = git_utils.normalize_url(repo_url)
if normalized_url1 not in git_info and normalized_url2 not in git_info:
if normalized_url not in git_info:
todo_disable.append(k2)
else:
if normalized_url1 in git_info:
commit_hash = git_info[normalized_url1]['hash']
todo_checkout.append((k2, commit_hash))
processed_urls.append(normalized_url1)
if normalized_url2 in git_info:
commit_hash = git_info[normalized_url2]['hash']
todo_checkout.append((k2, commit_hash))
processed_urls.append(normalized_url2)
commit_hash = git_info[normalized_url]['hash']
todo_checkout.append((k2, commit_hash))
processed_urls.append(normalized_url)
for k2, v2 in unified_manager.unknown_inactive_nodes.items():
repo_url = resolve_giturl_from_path(v2[1])
@@ -3090,18 +3141,12 @@ async def restore_snapshot(snapshot_path, git_helper_extras=None):
if repo_url is None:
continue
normalized_url1 = git_utils.normalize_url(repo_url)
normalized_url2 = git_utils.normalize_url_http(repo_url)
normalized_url = git_utils.normalize_url(repo_url)
if normalized_url1 in git_info:
commit_hash = git_info[normalized_url1]['hash']
if normalized_url in git_info:
commit_hash = git_info[normalized_url]['hash']
todo_enable.append((k2, commit_hash))
processed_urls.append(normalized_url1)
if normalized_url2 in git_info:
commit_hash = git_info[normalized_url2]['hash']
todo_enable.append((k2, commit_hash))
processed_urls.append(normalized_url2)
processed_urls.append(normalized_url)
for x in todo_disable:
unified_manager.unified_disable(x, True)
@@ -3187,17 +3232,26 @@ async def check_need_to_migrate():
need_to_migrate = True
def get_comfyui_versions():
repo = git.Repo(comfy_path)
versions = [x.name for x in repo.tags if x.name.startswith('v')]
versions.reverse() # nearest tag
def get_comfyui_versions(repo=None):
if repo is None:
repo = git.Repo(comfy_path)
try:
remote = get_remote_name(repo)
repo.remotes[remote].fetch()
except:
logging.error("[ComfyUI-Manager] Failed to fetch ComfyUI")
versions = [x.name for x in repo.tags if x.name.startswith('v')]
# nearest tag
versions = sorted(versions, key=lambda v: repo.git.log('-1', '--format=%ct', v), reverse=True)
versions = versions[:4]
current_tag = repo.git.describe('--tags')
if current_tag not in versions:
versions = sorted(versions + [current_tag], reverse=True)
versions = sorted(versions + [current_tag], key=lambda v: repo.git.log('-1', '--format=%ct', v), reverse=True)
versions = versions[:4]
main_branch = repo.heads.master
@@ -3210,16 +3264,18 @@ def get_comfyui_versions():
versions[0] = 'nightly'
current_tag = 'nightly'
return versions, current_tag
return versions, current_tag, latest_tag
def switch_comfyui(tag):
repo = git.Repo(comfy_path)
if tag == 'nightly':
repo.git.checkout('main')
repo.remotes.origin.pull()
print("[ComfyUI-Manager] ComfyUI version is switched to the latest 'main' version")
repo.git.checkout('master')
tracking_branch = repo.active_branch.tracking_branch()
remote_name = tracking_branch.remote_name
repo.remotes[remote_name].pull()
print("[ComfyUI-Manager] ComfyUI version is switched to the latest 'master' version")
else:
repo.git.checkout(tag)
print(f"[ComfyUI-Manager] ComfyUI version is switched to '{tag}'")
@@ -3234,7 +3290,7 @@ def resolve_giturl_from_path(fullpath):
if not os.path.exists(git_config_path):
return "unknown"
config = configparser.ConfigParser()
config = configparser.ConfigParser(strict=False)
config.read(git_config_path)
for k, v in config.items():

View File

@@ -11,6 +11,7 @@ from tqdm.auto import tqdm
aria2 = os.getenv('COMFYUI_MANAGER_ARIA2_SERVER')
HF_ENDPOINT = os.getenv('HF_ENDPOINT')
if aria2 is not None:
secret = os.getenv('COMFYUI_MANAGER_ARIA2_SECRET')
url = urlparse(aria2)

View File

@@ -55,8 +55,14 @@ def handle_stream(stream, prefix):
from comfy.cli_args import args
import latent_preview
def is_loopback(address):
import ipaddress
try:
return ipaddress.ip_address(address).is_loopback
except ValueError:
return False
is_local_mode = args.listen.startswith('127.') or args.listen.startswith('local.')
is_local_mode = is_loopback(args.listen)
model_dir_name_map = {
@@ -85,11 +91,11 @@ def is_allowed_security_level(level):
return False
elif level == 'high':
if is_local_mode:
return core.get_config()['security_level'].lower() in ['weak', 'normal-']
return core.get_config()['security_level'] in ['weak', 'normal-']
else:
return core.get_config()['security_level'].lower() == 'weak'
return core.get_config()['security_level'] == 'weak'
elif level == 'middle':
return core.get_config()['security_level'].lower() in ['weak', 'normal', 'normal-']
return core.get_config()['security_level'] in ['weak', 'normal', 'normal-']
else:
return True
@@ -181,6 +187,8 @@ set_preview_method(core.get_config()['preview_method'])
def set_component_policy(mode):
core.get_config()['component_policy'] = mode
def set_update_policy(mode):
core.get_config()['update_policy'] = mode
def print_comfyui_version():
global comfy_ui_hash
@@ -203,7 +211,7 @@ def print_comfyui_version():
comfyui_tag = core.get_comfyui_tag()
try:
if core.comfy_ui_commit_datetime.date() < core.comfy_ui_required_commit_datetime.date():
if not os.environ.get('__COMFYUI_DESKTOP_VERSION__') and core.comfy_ui_commit_datetime.date() < core.comfy_ui_required_commit_datetime.date():
logging.warning(f"\n\n## [WARN] ComfyUI-Manager: Your ComfyUI version ({core.comfy_ui_revision})[{core.comfy_ui_commit_datetime.date()}] is too old. Please update to the latest version. ##\n\n")
except:
pass
@@ -410,40 +418,65 @@ async def task_worker():
traceback.print_exc()
return f"Installation failed:\n{node_spec_str}"
async def do_update(item) -> str:
async def do_update(item):
ui_id, node_name, node_ver = item
try:
res = core.unified_manager.unified_update(node_name, node_ver)
if res.ver == 'unknown':
url = core.unified_manager.unknown_active_nodes[node_name][0]
title = os.path.basename(url)
else:
url = core.unified_manager.cnr_map[node_name].get('repository')
title = core.unified_manager.cnr_map[node_name]['name']
manager_util.clear_pip_cache()
if url is not None:
base_res = {'url': url, 'title': title}
else:
base_res = {'title': title}
if res.result:
if res.action == 'skip':
return 'skip'
base_res['msg'] = 'skip'
return base_res
else:
return 'success'
base_res['msg'] = 'success'
return base_res
base_res['msg'] = f"An error occurred while updating '{node_name}'."
logging.error(f"\nERROR: An error occurred while updating '{node_name}'.")
return base_res
except Exception:
traceback.print_exc()
return f"An error occurred while updating '{node_name}'."
return {'msg':f"An error occurred while updating '{node_name}'."}
async def do_update_comfyui() -> str:
async def do_update_comfyui(is_stable) -> str:
try:
repo_path = os.path.dirname(folder_paths.__file__)
res = core.update_path(repo_path)
latest_tag = None
if is_stable:
res, latest_tag = core.update_to_stable_comfyui(repo_path)
else:
res = core.update_path(repo_path)
if res == "fail":
logging.error("ComfyUI update fail: The installed ComfyUI does not have a Git repository.")
return "The installed ComfyUI does not have a Git repository."
elif res == "updated":
logging.info("ComfyUI is updated.")
return "success"
if is_stable:
logging.info("ComfyUI is updated to latest stable version.")
return "success-stable-"+latest_tag
else:
logging.info("ComfyUI is updated to latest nightly version.")
return "success-nightly"
else: # skipped
logging.info("ComfyUI is up-to-date.")
return "skip"
except Exception:
traceback.print_exc()
@@ -575,7 +608,7 @@ async def task_worker():
elif kind == 'update-main':
msg = await do_update(item)
elif kind == 'update-comfyui':
msg = await do_update_comfyui()
msg = await do_update_comfyui(item[1])
elif kind == 'fix':
msg = await do_fix(item)
elif kind == 'uninstall':
@@ -601,6 +634,9 @@ async def task_worker():
elif kind == 'update-comfyui':
nodepack_result['comfyui'] = msg
ui_target = "main"
elif kind == 'update':
nodepack_result[ui_id] = msg['msg']
ui_target = "nodepack_manager"
else:
nodepack_result[ui_id] = msg
ui_target = "nodepack_manager"
@@ -705,6 +741,15 @@ async def update_all(request):
update_item = k, k, v[0]
task_queue.put(("update-main", update_item))
for k, v in core.unified_manager.unknown_active_nodes.items():
if k == 'comfyui-manager':
# skip updating comfyui-manager if desktop version
if os.environ.get('__COMFYUI_DESKTOP_VERSION__'):
continue
update_item = k, k, 'unknown'
task_queue.put(("update-main", update_item))
return web.Response(status=200)
@@ -1303,14 +1348,15 @@ async def update_custom_node(request):
@routes.get("/manager/queue/update_comfyui")
async def update_comfyui(request):
task_queue.put(("update-comfyui", ('comfyui',)))
is_stable = core.get_config()['update_policy'] != 'nightly-comfyui'
task_queue.put(("update-comfyui", ('comfyui', is_stable)))
return web.Response(status=200)
@routes.get("/comfyui_manager/comfyui_versions")
async def comfyui_versions(request):
try:
res, current = core.get_comfyui_versions()
res, current, latest = core.get_comfyui_versions()
return web.json_response({'versions': res, 'current': current}, status=200, content_type='application/json')
except Exception as e:
logging.error(f"ComfyUI update fail: {e}", file=sys.stderr)
@@ -1401,7 +1447,7 @@ async def preview_method(request):
return web.Response(status=200)
@routes.get("/manager/component/policy")
@routes.get("/manager/policy/component")
async def component_policy(request):
if "value" in request.rel_url.query:
set_component_policy(request.rel_url.query['value'])
@@ -1412,6 +1458,17 @@ async def component_policy(request):
return web.Response(status=200)
@routes.get("/manager/policy/update")
async def update_policy(request):
if "value" in request.rel_url.query:
set_update_policy(request.rel_url.query['value'])
core.write_config()
else:
return web.Response(text=core.get_config()['update_policy'], status=200)
return web.Response(status=200)
@routes.get("/manager/channel_url_list")
async def channel_url_list(request):
channels = core.get_channel_dict()
@@ -1467,7 +1524,11 @@ async def get_notice(request):
markdown_content = match.group(1)
version_tag = core.get_comfyui_tag()
if version_tag is None:
markdown_content += f"<HR>ComfyUI: {core.comfy_ui_revision}[{comfy_ui_hash[:6]}]({core.comfy_ui_commit_datetime.date()})"
version_tag = os.environ.get('__COMFYUI_DESKTOP_VERSION__')
if version_tag is not None:
markdown_content += f"<HR>ComfyUI: {version_tag} [Desktop]"
else:
markdown_content += f"<HR>ComfyUI: {core.comfy_ui_revision}[{comfy_ui_hash[:6]}]({core.comfy_ui_commit_datetime.date()})"
else:
markdown_content += (f"<HR>ComfyUI: {version_tag}<BR>"
f"&nbsp; &nbsp; &nbsp; &nbsp; &nbsp;({core.comfy_ui_commit_datetime.date()})")

View File

@@ -373,3 +373,23 @@ def sanitize(data):
def sanitize_filename(input_string):
result_string = re.sub(r'[^a-zA-Z0-9_]', '_', input_string)
return result_string
def robust_readlines(fullpath):
import chardet
try:
with open(fullpath, "r") as f:
return f.readlines()
except:
encoding = None
with open(fullpath, "rb") as f:
raw_data = f.read()
result = chardet.detect(raw_data)
encoding = result['encoding']
if encoding is not None:
with open(fullpath, "r", encoding=encoding) as f:
return f.readlines()
print(f"[ComfyUI-Manager] Failed to recognize encoding for: {fullpath}")
return []

View File

@@ -21,6 +21,8 @@ import { CustomNodesManager } from "./custom-nodes-manager.js";
import { ModelManager } from "./model-manager.js";
import { SnapshotManager } from "./snapshot.js";
let manager_version = await getVersion();
var docStyle = document.createElement('style');
docStyle.innerHTML = `
.comfy-toast {
@@ -42,7 +44,7 @@ docStyle.innerHTML = `
#cm-manager-dialog {
width: 1000px;
height: 450px;
height: 455px;
box-sizing: content-box;
z-index: 1000;
overflow-y: auto;
@@ -139,7 +141,7 @@ docStyle.innerHTML = `
.cm-notice-board {
width: 290px;
height: 210px;
height: 230px;
overflow: auto;
color: var(--input-text);
border: 1px solid var(--descrip-text);
@@ -228,9 +230,10 @@ var switch_comfyui_button = null;
var fetch_updates_button = null;
var update_all_button = null;
var restart_stop_button = null;
var update_policy_combo = null;
let share_option = 'all';
var is_updating_all = false;
var is_updating = false;
// copied style from https://github.com/pythongosssss/ComfyUI-Custom-Scripts
@@ -477,6 +480,8 @@ async function updateComfyUI() {
const response = await api.fetchApi('/manager/queue/update_comfyui');
showTerminal();
is_updating = true;
await api.fetchApi('/manager/queue/start');
}
@@ -605,8 +610,14 @@ function showVersionSelectorDialog(versions, current, onSelect) {
}
async function switchComfyUI() {
switch_comfyui_button.disabled = true;
switch_comfyui_button.style.backgroundColor = "gray";
let res = await api.fetchApi(`/comfyui_manager/comfyui_versions`, { cache: "no-store" });
switch_comfyui_button.disabled = false;
switch_comfyui_button.style.backgroundColor = "";
if(res.status == 200) {
let obj = await res.json();
@@ -619,6 +630,15 @@ async function switchComfyUI() {
}
showVersionSelectorDialog(versions, obj.current, async (selected_version) => {
if(selected_version == 'nightly') {
update_policy_combo.value = 'nightly-comfyui';
api.fetchApi('/manager/policy/update?value=nightly-comfyui');
}
else {
update_policy_combo.value = 'stable-comfyui';
api.fetchApi('/manager/policy/update?value=stable-comfyui');
}
let response = await api.fetchApi(`/comfyui_manager/comfyui_switch_version?ver=${selected_version}`, { cache: "no-store" });
if (response.status == 200) {
infoToast(`ComfyUI version is switched to ${selected_version}`);
@@ -694,11 +714,11 @@ async function onQueueStatus(event) {
else if(event.detail.status == 'done') {
reset_action_buttons();
if(!is_updating_all) {
if(!is_updating) {
return;
}
is_updating_all = false;
is_updating = false;
let success_list = [];
let failed_list = [];
@@ -707,44 +727,56 @@ async function onQueueStatus(event) {
for(let k in event.detail.nodepack_result){
let v = event.detail.nodepack_result[k];
if(v == 'success') {
if(k == 'comfyui')
comfyui_state = 'success';
else
success_list.push(k);
if(k == 'comfyui') {
comfyui_state = v;
continue;
}
else if(v == 'skip') {
if(k == 'comfyui')
comfyui_state = 'skip';
if(v.msg == 'success') {
success_list.push(k);
}
else
else if(v.msg != 'skip')
failed_list.push(k);
}
let msg = "";
if(success_list.length == 0 && comfyui_state != 'success') {
if(success_list.length == 0 && !comfyui_state.startsWith('success')) {
if(failed_list.length == 0) {
msg += "All custom nodes are already up to date.";
msg += "You are already up to date.";
}
}
else {
msg = "To apply the updates, you need to <button class='cm-small-button' id='cm-reboot-button5'>RESTART</button> ComfyUI.<hr>";
if(comfyui_state == 'success') {
msg += "ComfyUI is updated.<BR><BR>";
if(comfyui_state == 'success-nightly') {
msg += "ComfyUI has been updated to latest nightly version.<BR><BR>";
infoToast("ComfyUI has been updated to the latest nightly version.");
}
else if(comfyui_state.startsWith('success-stable')) {
const ver = comfyui_state.split("-").pop();
msg += `ComfyUI has been updated to ${ver}.<BR><BR>`;
infoToast(`ComfyUI has been updated to ${ver}`);
}
else if(comfyui_state == 'skip') {
msg += "ComfyUI is already up-to-date.<BR><BR>"
msg += "ComfyUI is already up to date.<BR><BR>"
}
else if(comfyui_state != null) {
msg += "Failed to update ComfyUI.<BR><BR>"
}
if(success_list.length > 0) {
msg += "The following custom nodes have been updated:<ul>";
for(let x in success_list) {
if(success_list[x] == 'comfyui')
continue;
msg += '<li>'+success_list[x]+'</li>';
let k = success_list[x];
let url = event.detail.nodepack_result[k].url;
let title = event.detail.nodepack_result[k].title;
if(url) {
msg += `<li><a href='${url}' target='_blank'>${title}</a></li>`;
}
else {
msg += `<li>${k}</li>`;
}
}
msg += "</ul>";
}
@@ -755,7 +787,15 @@ async function onQueueStatus(event) {
if(failed_list.length > 0) {
msg += '<br>The update for the following custom nodes has failed:<ul>';
for(let x in failed_list) {
msg += '<li>'+failed_list[x]+'</li>';
let k = failed_list[x];
let url = event.detail.nodepack_result[k].url;
let title = event.detail.nodepack_result[k].title;
if(url) {
msg += `<li><a href='${url}' target='_blank'>${title}</a></li>`;
}
else {
msg += `<li>${k}</li>`;
}
}
msg += '</ul>'
@@ -797,7 +837,7 @@ async function updateAll(update_comfyui, manager_dialog) {
customAlert('Another task is already in progress. Please stop the ongoing task first.');
}
else if(response.status == 200) {
is_updating_all = true;
is_updating = true;
await api.fetchApi('/manager/queue/start');
}
}
@@ -920,7 +960,19 @@ class ManagerMenuDialog extends ComfyDialog {
}
}),
$el("button.cm-button", {
type: "button",
textContent: "Custom Nodes In Workflow",
onclick:
() => {
if(!CustomNodesManager.instance) {
CustomNodesManager.instance = new CustomNodesManager(app, self);
}
CustomNodesManager.instance.show(CustomNodesManager.ShowMode.IN_WORKFLOW);
}
}),
$el("br", {}, []),
$el("button.cm-button", {
type: "button",
textContent: "Model Manager",
@@ -949,7 +1001,7 @@ class ManagerMenuDialog extends ComfyDialog {
update_all_button,
update_comfyui_button,
switch_comfyui_button,
fetch_updates_button,
// fetch_updates_button,
$el("br", {}, []),
restart_stop_button,
@@ -981,6 +1033,8 @@ class ManagerMenuDialog extends ComfyDialog {
}
createControlsLeft() {
const isElectron = 'electronAPI' in window;
let self = this;
this.update_check_checkbox = $el("input",{type:'checkbox', id:"skip_update_check"},[])
@@ -1059,25 +1113,6 @@ class ManagerMenuDialog extends ComfyDialog {
share_combo.appendChild($el('option', { value: option[0], text: `Share: ${option[1]}` }, []));
}
// default ui state
let component_policy_combo = document.createElement("select");
component_policy_combo.setAttribute("title", "When loading the workflow, configure which version of the component to use.");
component_policy_combo.className = "cm-menu-combo";
component_policy_combo.appendChild($el('option', { value: 'workflow', text: 'Component: Use workflow version' }, []));
component_policy_combo.appendChild($el('option', { value: 'higher', text: 'Component: Use higher version' }, []));
component_policy_combo.appendChild($el('option', { value: 'mine', text: 'Component: Use my version' }, []));
api.fetchApi('/manager/component/policy')
.then(response => response.text())
.then(data => {
component_policy_combo.value = data;
set_component_policy(data);
});
component_policy_combo.addEventListener('change', function (event) {
api.fetchApi(`/manager/component/policy?value=${event.target.value}`);
set_component_policy(event.target.value);
});
api.fetchApi('/manager/share_option')
.then(response => response.text())
.then(data => {
@@ -1097,6 +1132,43 @@ class ManagerMenuDialog extends ComfyDialog {
}
});
let component_policy_combo = document.createElement("select");
component_policy_combo.setAttribute("title", "When loading the workflow, configure which version of the component to use.");
component_policy_combo.className = "cm-menu-combo";
component_policy_combo.appendChild($el('option', { value: 'workflow', text: 'Component: Use workflow version' }, []));
component_policy_combo.appendChild($el('option', { value: 'higher', text: 'Component: Use higher version' }, []));
component_policy_combo.appendChild($el('option', { value: 'mine', text: 'Component: Use my version' }, []));
api.fetchApi('/manager/policy/component')
.then(response => response.text())
.then(data => {
component_policy_combo.value = data;
set_component_policy(data);
});
component_policy_combo.addEventListener('change', function (event) {
api.fetchApi(`/manager/policy/component?value=${event.target.value}`);
set_component_policy(event.target.value);
});
update_policy_combo = document.createElement("select");
if(isElectron)
update_policy_combo.style.display = 'none';
update_policy_combo.setAttribute("title", "Sets the policy to be applied when performing an update.");
update_policy_combo.className = "cm-menu-combo";
update_policy_combo.appendChild($el('option', { value: 'stable-comfyui', text: 'Update: ComfyUI Stable Version' }, []));
update_policy_combo.appendChild($el('option', { value: 'nightly-comfyui', text: 'Update: ComfyUI Nightly Version' }, []));
api.fetchApi('/manager/policy/update')
.then(response => response.text())
.then(data => {
update_policy_combo.value = data;
});
update_policy_combo.addEventListener('change', function (event) {
api.fetchApi(`/manager/policy/update?value=${event.target.value}`);
});
return [
$el("div", {}, [this.update_check_checkbox, uc_checkbox_text]),
$el("br", {}, []),
@@ -1105,6 +1177,7 @@ class ManagerMenuDialog extends ComfyDialog {
preview_combo,
share_combo,
component_policy_combo,
update_policy_combo,
$el("br", {}, []),
$el("br", {}, []),
@@ -1131,11 +1204,6 @@ class ManagerMenuDialog extends ComfyDialog {
install_pip(url, self);
}
}
}),
$el("button.cm-experimental-button", {
type: "button",
textContent: "Unload models",
onclick: () => { free_models(); }
})
]),
];
@@ -1264,7 +1332,7 @@ class ManagerMenuDialog extends ComfyDialog {
$el("div.comfy-modal-content",
[
$el("tr.cm-title", {}, [
$el("font", {size:6, color:"white"}, [`ComfyUI Manager Menu`])]
$el("font", {size:6, color:"white"}, [`ComfyUI Manager ${manager_version}`])]
),
$el("br", {}, []),
$el("div.cm-menu-container",
@@ -1406,13 +1474,12 @@ async function getVersion() {
return await version.text();
}
app.registerExtension({
name: "Comfy.ManagerMenu",
aboutPageBadges: [
{
label: `ComfyUI-Manager ${await getVersion()}`,
label: `ComfyUI-Manager ${manager_version}`,
url: 'https://github.com/ltdrdata/ComfyUI-Manager',
icon: 'pi pi-th-large'
}

View File

@@ -709,7 +709,7 @@ app.handleFile = handleFile;
let current_component_policy = 'workflow';
try {
api.fetchApi('/manager/component/policy')
api.fetchApi('/manager/policy/component')
.then(response => response.text())
.then(data => { current_component_policy = data; });
}

View File

@@ -363,6 +363,7 @@ const pageHtml = `
<button class="cn-manager-restart">Restart</button>
<button class="cn-manager-stop">Stop</button>
<div class="cn-flex-auto"></div>
<button class="cn-manager-used-in-workflow">Used In Workflow</button>
<button class="cn-manager-check-update">Check Update</button>
<button class="cn-manager-check-missing">Check Missing</button>
<button class="cn-manager-install-url">Install via Git URL</button>
@@ -374,7 +375,8 @@ const ShowMode = {
UPDATE: "Update",
MISSING: "Missing",
FAVORITES: "Favorites",
ALTERNATIVES: "Alternatives"
ALTERNATIVES: "Alternatives",
IN_WORKFLOW: "In Workflow",
};
export class CustomNodesManager {
@@ -586,6 +588,10 @@ export class CustomNodesManager {
label: "Update",
value: ShowMode.UPDATE,
hasData: false
}, {
label: "In Workflow",
value: ShowMode.IN_WORKFLOW,
hasData: false
}, {
label: "Missing",
value: ShowMode.MISSING,
@@ -726,7 +732,7 @@ export class CustomNodesManager {
const value = e.target.value
this.filter = value;
const item = this.getFilterItem(value);
if (item && !item.hasData) {
if (item && (!item.hasData)) {
this.loadData(value);
return;
}
@@ -779,6 +785,14 @@ export class CustomNodesManager {
}
},
".cn-manager-used-in-workflow": {
click: (e) => {
e.target.classList.add("cn-btn-loading");
this.setFilter(ShowMode.IN_WORKFLOW);
this.loadData(ShowMode.IN_WORKFLOW);
}
},
".cn-manager-check-update": {
click: (e) => {
e.target.classList.add("cn-btn-loading");
@@ -1529,7 +1543,110 @@ export class CustomNodesManager {
return extension_mappings;
}
getNodesInWorkflow() {
let usedGroupNodes = new Set();
let allUsedNodes = {};
for(let k in app.graph._nodes) {
let node = app.graph._nodes[k];
if(node.type.startsWith('workflow>')) {
usedGroupNodes.add(node.type.slice(9));
continue;
}
allUsedNodes[node.type] = node;
}
for(let k of usedGroupNodes) {
let subnodes = app.graph.extra.groupNodes[k]?.nodes;
if(subnodes) {
for(let k2 in subnodes) {
let node = subnodes[k2];
allUsedNodes[node.type] = node;
}
}
}
return allUsedNodes;
}
async getMissingNodes() {
let unresolved_missing_nodes = new Set();
let hashMap = {};
let allUsedNodes = this.getNodesInWorkflow();
const registered_nodes = new Set();
for (let i in LiteGraph.registered_node_types) {
registered_nodes.add(LiteGraph.registered_node_types[i].type);
}
let unresolved_aux_ids = {};
let outdated_comfyui = false;
for(let k in allUsedNodes) {
let node = allUsedNodes[k];
if(!registered_nodes.has(node.type)) {
// missing node
if(node.properties.cnr_id) {
if(node.properties.cnr_id == 'comfy-core') {
outdated_comfyui = true;
}
let item = this.custom_nodes[node.properties.cnr_id];
hashMap[item.hash] = true;
}
else if(node.properties.aux_id) {
unresolved_aux_ids[node.properties.aux_id] = node.type;
}
else {
unresolved_missing_nodes.add(node.type);
}
}
}
if(outdated_comfyui) {
customAlert('ComfyUI is outdated, so some built-in nodes cannot be used.');
}
if(Object.keys(unresolved_aux_ids).length > 0) {
// building aux_id to nodepack map
let aux_id_to_pack = {};
for(let k in this.custom_nodes) {
let nodepack = this.custom_nodes[k];
let aux_id;
if(nodepack.repository?.startsWith('https://github.com')) {
aux_id = nodepack.repository.split('/').slice(-2).join('/');
aux_id_to_pack[aux_id] = nodepack;
}
else if(nodepack.repository) {
aux_id = nodepack.repository.split('/').slice(-1);
aux_id_to_pack[aux_id] = nodepack;
}
}
// resolving aux_id
for(let k in unresolved_aux_ids) {
let nodepack = aux_id_to_pack[k];
if(nodepack) {
hashMap[nodepack.hash] = true;
}
else {
unresolved_missing_nodes.add(unresolved_aux_ids[k]);
}
}
}
if(unresolved_missing_nodes.size > 0) {
await this.getMissingNodesLegacy(hashMap, unresolved_missing_nodes, registered_nodes);
}
return hashMap;
}
async getMissingNodesLegacy(hashMap, missing_nodes, registered_nodes) {
const mode = manager_instance.datasrc_combo.value;
this.showStatus(`Loading missing nodes (${mode}) ...`);
const res = await fetchData(`/customnode/getmappings?mode=${mode}`);
@@ -1568,23 +1685,8 @@ export class CustomNodesManager {
}
}
const registered_nodes = new Set();
for (let i in LiteGraph.registered_node_types) {
registered_nodes.add(LiteGraph.registered_node_types[i].type);
}
const missing_nodes = new Set();
const workflow = app.graph.serialize();
const group_nodes = workflow.extra && workflow.extra.groupNodes ? workflow.extra.groupNodes : [];
let nodes = workflow.nodes;
for (let i in group_nodes) {
let group_node = group_nodes[i];
nodes = nodes.concat(group_node.nodes);
}
for (let i in nodes) {
const node_type = nodes[i].type;
let unresolved_missing_nodes = new Set();
for (let node_type of missing_nodes) {
if(node_type.startsWith('workflow/') || node_type.startsWith('workflow>'))
continue;
@@ -1592,26 +1694,25 @@ export class CustomNodesManager {
const packs = name_to_packs[node_type.trim()];
if(packs)
packs.forEach(url => {
missing_nodes.add(url);
unresolved_missing_nodes.add(url);
});
else {
for(let j in regex_to_pack) {
if(regex_to_pack[j].regex.test(node_type)) {
missing_nodes.add(regex_to_pack[j].url);
unresolved_missing_nodes.add(regex_to_pack[j].url);
}
}
}
}
}
const hashMap = {};
for(let k in this.custom_nodes) {
let item = this.custom_nodes[k];
if(missing_nodes.has(item.id)) {
if(unresolved_missing_nodes.has(item.id)) {
hashMap[item.hash] = true;
}
else if (item.files?.some(file => missing_nodes.has(file))) {
else if (item.files?.some(file => unresolved_missing_nodes.has(file))) {
hashMap[item.hash] = true;
}
}
@@ -1630,6 +1731,41 @@ export class CustomNodesManager {
return hashMap;
}
async getNodepackInWorkflow() {
let allUsedNodes = this.getNodesInWorkflow();
// building aux_id to nodepack map
let aux_id_to_pack = {};
for(let k in this.custom_nodes) {
let nodepack = this.custom_nodes[k];
let aux_id;
if(nodepack.repository?.startsWith('https://github.com')) {
aux_id = nodepack.repository.split('/').slice(-2).join('/');
aux_id_to_pack[aux_id] = nodepack;
}
else if(nodepack.repository) {
aux_id = nodepack.repository.split('/').slice(-1);
aux_id_to_pack[aux_id] = nodepack;
}
}
const hashMap = {};
for(let k in allUsedNodes) {
var item;
if(allUsedNodes[k].properties.cnr_id) {
item = this.custom_nodes[allUsedNodes[k].properties.cnr_id];
}
else if(allUsedNodes[k].properties.aux_id) {
item = aux_id_to_pack[allUsedNodes[k].properties.aux_id];
}
if(item)
hashMap[item.hash] = true;
}
return hashMap;
}
async getAlternatives() {
const mode = manager_instance.datasrc_combo.value;
this.showStatus(`Loading alternatives (${mode}) ...`);
@@ -1725,9 +1861,14 @@ export class CustomNodesManager {
hashMap = await this.getAlternatives();
} else if(this.show_mode == ShowMode.FAVORITES) {
hashMap = await this.getFavorites();
} else if(this.show_mode == ShowMode.IN_WORKFLOW) {
hashMap = await this.getNodepackInWorkflow();
}
filterItem.hashMap = hashMap;
filterItem.hasData = true;
if(this.show_mode != ShowMode.IN_WORKFLOW) {
filterItem.hasData = true;
}
}
for(let k in node_packs) {
@@ -1779,7 +1920,6 @@ export class CustomNodesManager {
case "disabled":
filterTypes.add("installed");
break;
case "not-installed":
filterTypes.add("not-installed");
break;

View File

@@ -3,12 +3,21 @@
* - custom node pack version to all custom nodes used in the workflow
*
* Example metadata:
"extra": {
"node_versions": {
"comfy-core": "v0.3.8-4-g0b2eb7f",
"comfyui-easy-use": "1.2.5"
}
},
* "nodes": {
* "1": {
* type: "CheckpointLoaderSimple",
* ...
* properties: {
* cnr_id: "comfy-core",
* version: "0.3.8",
* },
* },
* }
*
* @typedef {Object} NodeInfo
* @property {string} ver - Version (git hash or semantic version)
* @property {string} cnr_id - ComfyRegistry node ID
* @property {boolean} enabled - Whether the node is enabled
*/
import { app } from "../../scripts/app.js";
@@ -23,7 +32,7 @@ class WorkflowMetadataExtension {
/**
* Get the installed nodes info
* @returns {Promise<Record<string, {ver: string, cnr_id: string, enabled: boolean}>>} The mapping from node name to its info.
* @returns {Promise<Record<string, NodeInfo>>} The mapping from node name to its info.
* ver can either be a git commit hash or a semantic version such as "1.0.0"
* cnr_id is the id of the node in the ComfyRegistry
* enabled is true if the node is enabled, false if it is disabled
@@ -33,61 +42,42 @@ class WorkflowMetadataExtension {
return await res.json();
}
/**
* Get the node versions for the given graph
* @param {LGraph} graph The graph to get the node versions for
* @returns {Promise<Record<string, string>>} The mapping from node name to version
*/
getGraphNodeVersions(graph) {
const nodeVersions = {};
for (const node of graph.nodes) {
const nodeData = node.constructor.nodeData;
// Frontend only nodes don't have nodeData
if (!nodeData) {
continue;
}
const modules = nodeData.python_module.split(".");
if (modules[0] === "custom_nodes") {
const nodePackageName = modules[1];
const nodeInfo =
this.installedNodes[nodePackageName] ??
this.installedNodes[nodePackageName.toLowerCase()];
if (nodeInfo) {
nodeVersions[nodePackageName] = nodeInfo.ver;
}
} else if (["nodes", "comfy_extras"].includes(modules[0])) {
nodeVersions["comfy-core"] = this.comfyCoreVersion;
} else {
console.warn(`Unknown node source: ${nodeData.python_module}`);
}
}
return nodeVersions;
}
async init() {
const extension = this;
this.installedNodes = await this.getInstalledNodes();
this.comfyCoreVersion = (await api.getSystemStats()).system.comfyui_version;
}
// Attach metadata when app.graphToPrompt is called.
const originalSerialize = LGraph.prototype.serialize;
LGraph.prototype.serialize = function () {
const workflow = originalSerialize.apply(this, arguments);
/**
* Called when any node is created
* @param {LGraphNode} node The newly created node
*/
nodeCreated(node) {
try {
// nodeData doesn't exist if node is missing or node is frontend only node
if (!node?.constructor?.nodeData?.python_module) return;
// Add metadata to the workflow
if (!workflow.extra) {
workflow.extra = {};
const nodeProperties = (node.properties ??= {});
const modules = node.constructor.nodeData.python_module.split(".");
const moduleType = modules[0];
if (moduleType === "custom_nodes") {
const nodePackageName = modules[1];
const { cnr_id, aux_id, ver } =
this.installedNodes[nodePackageName] ??
this.installedNodes[nodePackageName.toLowerCase()] ??
{};
if (cnr_id === "comfy-core") return; // don't allow hijacking comfy-core name
if (cnr_id) nodeProperties.cnr_id = cnr_id;
else nodeProperties.aux_id = aux_id;
if (ver) nodeProperties.ver = ver;
} else if (["nodes", "comfy_extras"].includes(moduleType)) {
nodeProperties.cnr_id = "comfy-core";
nodeProperties.ver = this.comfyCoreVersion;
}
const graph = this;
try {
workflow.extra["node_versions"] = extension.getGraphNodeVersions(graph);
} catch (e) {
console.error(e);
}
return workflow;
};
} catch (e) {
console.error(e);
}
}
}

View File

@@ -4685,6 +4685,28 @@
"filename": "<huggingface>",
"url": "deepseek-ai/Janus-Pro-7B",
"size": "14.85GB"
},
{
"name": "kolors/vae/diffusion_pytorch_model.fp16.safetensors",
"type": "VAE",
"base": "Kolors",
"save_path": "vae/kolors",
"description": "Kolors VAE",
"reference": "https://huggingface.co/Kwai-Kolors/Kolors",
"filename": "diffusion_pytorch_model.fp16.safetensors",
"url": "https://huggingface.co/Kwai-Kolors/Kolors/resolve/main/vae/diffusion_pytorch_model.fp16.safetensors",
"size": "167MB"
},
{
"name": "kolors/vae/diffusion_pytorch_model.safetensors",
"type": "VAE",
"base": "Kolors",
"save_path": "vae/kolors",
"description": "Kolors VAE",
"reference": "https://huggingface.co/Kwai-Kolors/Kolors",
"filename": "diffusion_pytorch_model.safetensors",
"url": "https://huggingface.co/Kwai-Kolors/Kolors/resolve/main/vae/diffusion_pytorch_model.safetensors",
"size": "335MB"
}
]
}

View File

@@ -13,6 +13,248 @@
{
"author": "IfnotFr",
"title": "⚡ ComfyUI Connect [WIP]",
"reference": "https://github.com/IfnotFr/ComfyUI-Connect",
"files": [
"https://github.com/IfnotFr/ComfyUI-Connect"
],
"install_type": "git-clone",
"description": "Transform your ComfyUI into a powerful API, exposing all your saved workflows as ready-to-use HTTP endpoints."
},
{
"author": "muvich3n",
"title": "ComfyUI-Crop-Border",
"reference": "https://github.com/muvich3n/ComfyUI-Crop-Border",
"files": [
"https://github.com/muvich3n/ComfyUI-Crop-Border"
],
"install_type": "git-clone",
"description": "NODES: Crop Image Borders"
},
{
"author": "masmullin2000",
"title": "ComfyUI-MMYolo",
"reference": "https://github.com/masmullin2000/ComfyUI-MMYolo",
"files": [
"https://github.com/masmullin2000/ComfyUI-MMYolo"
],
"install_type": "git-clone",
"description": "A comfy node to find faces and output a mask"
},
{
"author": "RiceRound",
"title": "RiceRound Cloud Node [UNSAFE]",
"id": "riceround",
"reference": "https://github.com/RiceRound/ComfyUI_RiceRound",
"files": [
"https://github.com/RiceRound/ComfyUI_RiceRound"
],
"install_type": "git-clone",
"description": "This is an imaginative project that allows for one-click deployment, providing both an online page and a ComfyUI cloud node.[w/This custom node is vulnerable because it can dynamically download and execute nodes.]"
},
{
"author": "Yeonri",
"title": "ComfyUI_LLM_Are_You_Listening [WIP]",
"reference": "https://github.com/Yeonri/ComfyUI_LLM_Are_You_Listening",
"files": [
"https://github.com/Yeonri/ComfyUI_LLM_Are_You_Listening"
],
"install_type": "git-clone",
"description": "NODES: AYL_Node, AYL_GGUF_Node, AYL_API_Node\nNOTE: The files in the repo are not organized."
},
{
"author": "altkeyproject",
"title": "Dream Painter [WIP]",
"reference": "https://github.com/alt-key-project/comfyui-dream-painter",
"files": [
"https://github.com/alt-key-project/comfyui-dream-painter"
],
"install_type": "git-clone",
"description": "Provide utilities for 2D image generation and processing."
},
{
"author": "kimara-ai",
"title": "ComfyUI-Kimara-AI-Image-From-URL [WIP]",
"reference": "https://github.com/kimara-ai/ComfyUI-Kimara-AI-Image-From-URL",
"files": [
"https://github.com/kimara-ai/ComfyUI-Kimara-AI-Image-From-URL"
],
"install_type": "git-clone",
"description": "Load image from URL and downscale to desired megapixels. Set megapixels to 0 for no downscaling."
},
{
"author": "krisshen2021",
"title": "comfyui_OpenRouterNodes [WIP]",
"reference": "https://github.com/krisshen2021/comfyui_OpenRouterNodes",
"files": [
"https://github.com/krisshen2021/comfyui_OpenRouterNodes"
],
"install_type": "git-clone",
"description": "LLM custom nodes for comfyui\nNOTE: The files in the repo are not organized."
},
{
"author": "Velour-Fog",
"title": "comfy-latent-nodes [UNSAFE]",
"reference": "https://github.com/Velour-Fog/comfy-latent-nodes",
"files": [
"https://github.com/Velour-Fog/comfy-latent-nodes"
],
"install_type": "git-clone",
"description": "ComfyUI nodes to save and load a latent to a specified directory. Saves time for doing operations on a latent such as upscaling without having to re-trigger the creation of the original latent.[w/This node can write files to an arbitrary path.]"
},
{
"author": "jgbyte",
"title": "ComfyUI-RandomCube [WIP]",
"reference": "https://github.com/jgbyte/ComfyUI-RandomCube",
"files": [
"https://github.com/jgbyte/ComfyUI-RandomCube"
],
"install_type": "git-clone",
"description": "NODES: RandomCubeGrid"
},
{
"author": "thot-experiment",
"title": "comfy-live-preview [WIP]",
"reference": "https://github.com/thot-experiment/comfy-live-preview",
"files": [
"https://github.com/thot-experiment/comfy-live-preview"
],
"install_type": "git-clone",
"description": "external live preview plugin for ComfyUI"
},
{
"author": "AhBumm",
"title": "ComfyUI-Upscayl",
"reference": "https://github.com/AhBumm/ComfyUI-Upscayl",
"files": [
"https://github.com/AhBumm/ComfyUI-Upscayl"
],
"nodename_pattern": "\\(BillBum\\)$",
"install_type": "git-clone",
"description": "NODES: Upscayl Upscaler"
},
{
"author": "NEZHA625",
"title": "ComfyUI-tools-by-dong [UNSAFE]",
"reference": "https://github.com/NEZHA625/ComfyUI-tools-by-dong",
"files": [
"https://github.com/NEZHA625/ComfyUI-tools-by-dong"
],
"install_type": "git-clone",
"description": "NODES: HuggingFaceUploadNode, ImageDownloader, LoraIterator, FileMoveNode, InputDetectionNode, ...\nNOTE: The files in the repo are not organized.[w/This node pack includes nodes that can modify arbitrary files.]"
},
{
"author": "if-ai",
"title": "ComfyUI-IF_Zonos [WIP]",
"reference": "https://github.com/if-ai/ComfyUI-IF_Zonos",
"files": [
"https://github.com/if-ai/ComfyUI-IF_Zonos"
],
"install_type": "git-clone",
"description": "Zonos for ComfyUI"
},
{
"author": "grinlau18",
"title": "Xiser_Nodes [WIP]",
"reference": "https://github.com/grinlau18/ComfyUI_XISER_Nodes",
"files": [
"https://github.com/grinlau18/ComfyUI_XISER_Nodes"
],
"install_type": "git-clone",
"description": "A collection of custom nodes for ComfyUI\nNOTE: The files in the repo are not organized."
},
{
"author": "LAOGOU-666",
"title": "Comfyui_StartPatch [UNSAFE]",
"reference": "https://github.com/LAOGOU-666/Comfyui_StartPatch",
"files": [
"https://github.com/LAOGOU-666/Comfyui_StartPatch"
],
"install_type": "git-clone",
"description": "This patch plugin optimizes the node information processing mechanism of the ComfyUI server, significantly improving server performance and response speed. It greatly reduces the browser page initialization waiting time. [w/Since this patch modifies key functions of ComfyUI, it is highly likely to cause compatibility issues.]"
},
{
"author": "badmike",
"title": "Prompt Factory [CONFLICT]",
"reference": "https://github.com/badmike/comfyui-prompt-factory",
"files": [
"https://github.com/badmike/comfyui-prompt-factory"
],
"install_type": "git-clone",
"description": "A modular system that adds randomness to prompt generation [w/This node pack is causing a name conflict with https://github.com/satche/comfyui-prompt-factory]"
},
{
"author": "owengillett",
"title": "ComfyUI-tilefusion",
"reference": "https://github.com/owengillett/ComfyUI-tilefusion",
"files": [
"https://github.com/owengillett/ComfyUI-tilefusion"
],
"install_type": "git-clone",
"description": "Helper nodes for generating seamless tiles."
},
{
"author": "Scaryplasmon",
"title": "ComfTrellis [WIP]",
"reference": "https://github.com/Scaryplasmon/ComfTrellis",
"files": [
"https://github.com/Scaryplasmon/ComfTrellis"
],
"install_type": "git-clone",
"description": "1 click install to run Trellis in ComfyUI\nNOTE: The files in the repo are not organized."
},
{
"author": "fangziheng2321",
"title": "comfyuinode_chopmask [WIP]",
"reference": "https://github.com/fangziheng2321/comfyuinode_chopmask",
"files": [
"https://github.com/fangziheng2321/comfyuinode_chopmask"
],
"install_type": "git-clone",
"description": "a custom comfyui node for '/fooocusinpaint_upload'\nNOTE: The files in the repo are not organized."
},
{
"author": "D1-3105",
"title": "ComfyUI-VideoStream",
"reference": "https://github.com/D1-3105/ComfyUI-VideoStream",
"files": [
"https://github.com/D1-3105/ComfyUI-VideoStream"
],
"install_type": "git-clone",
"description": "NODES: FloWWeaverExportSingleFrameGRPC"
},
{
"author": "gmorks",
"title": "ComfyUI Animagine prompt [WIP]",
"reference": "https://github.com/gmorks/ComfyUI-Animagine-Prompt",
"files": [
"https://github.com/gmorks/ComfyUI-Animagine-Prompt"
],
"install_type": "git-clone",
"description": "Comfy UI node to prompt build for [a/https://huggingface.co/cagliostrolab/animagine-xl-4.0](https://huggingface.co/cagliostrolab/animagine-xl-4.0) model\nNOTE: The files in the repo are not organized."
},
{
"author": "wirytiox",
"title": "ComfyUI-Qwen [CONFLICT]",
"reference": "https://github.com/mr-krak3n/ComfyUI-Qwen",
"files": [
"https://github.com/mr-krak3n/ComfyUI-Qwen"
],
"install_type": "git-clone",
"description": "This repository contains custom nodes for ComfyUI, designed to facilitate working with language models such as Qwen2.5 and DeepSeek. [w/This node pack is causing a name conflict with https://github.com/ZHO-ZHO-ZHO/ComfyUI-Qwen]"
},
{
"author": "hiusdev",
"title": "ComfyUI_Lah_Toffee",
"reference": "https://github.com/hiusdev/ComfyUI_Lah_Toffee",
"files": [
"https://github.com/hiusdev/ComfyUI_Lah_Toffee"
],
"install_type": "git-clone",
"description": "NODES: Lah LoadVideoRandom"
},
{
"author": "hdfhssg",
"title": "ComfyUI_pxtool [WIP]",
@@ -23,16 +265,6 @@
"install_type": "git-clone",
"description": "This is a custom plugin node for ComfyUI that modifies and extends some features from existing projects. The main implementations include:\n* Reproducing some features of the [a/Stable-Diffusion-Webui-Civitai-Helper](https://github.com/zixaphir/Stable-Diffusion-Webui-Civitai-Helper) project within ComfyUI\n* Implementing a feature to randomly generate related prompt words by referencing the [a/noob-wiki dataset](https://huggingface.co/datasets/Laxhar/noob-wiki/tree/main)\nNOTE: The files in the repo are not organized."
},
{
"author": "dasilva333",
"title": "ComfyUI_MarkdownImage [WIP]",
"reference": "https://github.com/dasilva333/ComfyUI_MarkdownImage",
"files": [
"https://github.com/dasilva333/ComfyUI_MarkdownImage"
],
"install_type": "git-clone",
"description": "Create an image using html and markdown in ComfyUI\nNOTE: The files in the repo are not organized."
},
{
"author": "franky519",
"title": "comfyui-redux-style",
@@ -83,16 +315,6 @@
"install_type": "git-clone",
"description": "NODES: Load TIFF"
},
{
"author": "greengerong",
"title": "ComfyUI-Lumina-Video [WIP]",
"reference": "https://github.com/greengerong/ComfyUI-Lumina-Video",
"files": [
"https://github.com/greengerong/ComfyUI-Lumina-Video"
],
"install_type": "git-clone",
"description": "This is a video generation plugin implementation for ComfyUI based on the Lumina Video model."
},
{
"author": "tc888",
"title": "ComfyUI_Save_Flux_Image",
@@ -722,7 +944,7 @@
"https://github.com/yanhuifair/ComfyUI-FairLab"
],
"install_type": "git-clone",
"description": "NODES: CLIP Text Encode Translated, Translate String, Load Image From Folder, Save String To Folder, Fix UTF-8 String, String Combine, String Field, Download Image, Save Images To Folder, Save Image To Folder, Image Resize"
"description": "NODES: CLIP Text Encode Translated, Translate String, Load Image From Folder, Save String To Folder, Fix UTF-8 String, String Combine, String Field, Download Image, Save Images To Folder, Save Image To Folder, Image Resize, ..."
},
{
"author": "nomcycle",
@@ -1621,16 +1843,6 @@
"install_type": "git-clone",
"description": "To use stepfun's library, you need an official api that supports multimodal inputs such as video and pictures [a/https://platform.stepfun.com/request-restriction](https://platform.stepfun.com/request-restriction)"
},
{
"author": "attashe",
"title": "ComfyUI-FluxRegionAttention [WIP]",
"reference": "https://github.com/attashe/ComfyUI-FluxRegionAttention",
"files": [
"https://github.com/attashe/ComfyUI-FluxRegionAttention"
],
"install_type": "git-clone",
"description": "Implement Region Attention for Flux model"
},
{
"author": "aria1th",
"title": "ComfyUI-SkipCFGSigmas",

View File

@@ -178,7 +178,6 @@
"https://github.com/807502278/ComfyUI_TensorRT_Merge": [
[
"BiRefNet2_tensort",
"BiRefNet_ModelLoader_TRT",
"BiRefNet_TRT",
"Building_TRT",
"Custom_Building_TRT",
@@ -189,6 +188,7 @@
"UpscalerTensorrt",
"YoloNasPoseTensorrt",
"load_BiRefNet2_tensort",
"load_BiRefNet_TRT",
"load_DepthAnything_Tensorrt",
"load_Dwpos_Tensorrt"
],
@@ -370,6 +370,15 @@
"title_aux": "comfyui-textools [WIP]"
}
],
"https://github.com/AhBumm/ComfyUI-Upscayl": [
[
"Upscayl Upscaler"
],
{
"nodename_pattern": "\\(BillBum\\)$",
"title_aux": "ComfyUI-Upscayl"
}
],
"https://github.com/AlexXi19/ComfyUI-OpenAINode": [
[
"ImageWithPrompt",
@@ -658,6 +667,14 @@
"title_aux": "ComfyUI OpenAI Nodes"
}
],
"https://github.com/D1-3105/ComfyUI-VideoStream": [
[
"FloWWeaverExportSingleFrameGRPC"
],
{
"title_aux": "ComfyUI-VideoStream"
}
],
"https://github.com/DataCTE/ComfyUI-DataVoid-nodes": [
[
"IPAAdapterFaceIDBatch",
@@ -733,7 +750,6 @@
"https://github.com/DraconicDragon/ComfyUI_e621_booru_toolkit": [
[
"GetBooruPost",
"TagEncode",
"TagWikiFetch"
],
{
@@ -940,9 +956,11 @@
"DeepSeekImageGeneration",
"DeepSeekImageUnderstanding",
"DeepSeekModelLoader",
"GoogleDriveUpload",
"ImagePreprocessor",
"LLM_Loader",
"OpenAICompatibleLoader"
"OpenAICompatibleLoader",
"VideoFileUploader"
],
{
"title_aux": "ComfyUI-DeepSeek_Toolkit [WIP]"
@@ -1011,6 +1029,7 @@
"ImNodeTitleOverride",
"ImSetActionKeywordMapping",
"MergeNode",
"Molmo7BDbnbBatch",
"MuteNode",
"NewNode",
"Node2String",
@@ -1316,6 +1335,38 @@
"title_aux": "ComfyUI-APG_ImYourCFGNow"
}
],
"https://github.com/NEZHA625/ComfyUI-tools-by-dong": [
[
"A1111_FLUX_DATA_NODE",
"CategorizeNode",
"FileMoveNode",
"FolderIteratorNODE",
"Get_cookies_Node",
"Get_json_value_Node",
"HashCalculationsNode",
"HuggingFaceUploadNode",
"Image2GIFNode",
"ImageDownloader",
"InputDetectionNode",
"LLM_Node",
"LibLib_upload_Node",
"LogicToolsNode",
"LoraIterator",
"RandomNumbersNode",
"RenameNode",
"ResolutionNode",
"SaveTXTNode",
"SetAppidNode",
"TextToJsonNode",
"TranslateAPINode",
"ZIPwith7zNode",
"path_join_Node",
"set_api_Node"
],
{
"title_aux": "ComfyUI-tools-by-dong [UNSAFE]"
}
],
"https://github.com/Northerner1/ComfyUI_North_Noise": [
[
"North_Unsampler"
@@ -1415,6 +1466,41 @@
"title_aux": "comfyui-promptbymood [WIP]"
}
],
"https://github.com/RiceRound/ComfyUI_RiceRound": [
[
"RiceRoundAdvancedChoiceNode",
"RiceRoundBooleanNode",
"RiceRoundDecryptNode",
"RiceRoundDownloadImageAndMaskNode",
"RiceRoundDownloadImageNode",
"RiceRoundDownloadMaskNode",
"RiceRoundEncryptNode",
"RiceRoundFloatNode",
"RiceRoundImageBridgeNode",
"RiceRoundImageNode",
"RiceRoundImageUrlNode",
"RiceRoundInputTextNode",
"RiceRoundIntNode",
"RiceRoundMaskBridgeNode",
"RiceRoundOutputBooleanNode",
"RiceRoundOutputFloatNode",
"RiceRoundOutputImageBridgeNode",
"RiceRoundOutputImageNode",
"RiceRoundOutputIntNode",
"RiceRoundOutputMaskBridgeNode",
"RiceRoundOutputTextNode",
"RiceRoundRandomSeedNode",
"RiceRoundSimpleChoiceNode",
"RiceRoundSimpleImageNode",
"RiceRoundStrToBooleanNode",
"RiceRoundStrToFloatNode",
"RiceRoundStrToIntNode",
"RiceRoundUploadImageNode"
],
{
"title_aux": "RiceRound Cloud Node [UNSAFE]"
}
],
"https://github.com/RicherdLee/comfyui-oss-image-save": [
[
"SaveImageOSS"
@@ -1425,9 +1511,12 @@
],
"https://github.com/RobeSantoro/ComfyUI-RobeNodes": [
[
"Boolean Primitive \ud83d\udc24",
"Image Input Switch \ud83d\udc24",
"List Image Path \ud83d\udc24",
"List Model Path \ud83d\udc24",
"List Video Path \ud83d\udc24"
"List Video Path \ud83d\udc24",
"Peaks Weights Generator \ud83d\udc24"
],
{
"title_aux": "Comfy UI Robe Nodes [UNSAFE]"
@@ -1477,6 +1566,18 @@
"title_aux": "ComfyUI_Save2Discord"
}
],
"https://github.com/Scaryplasmon/ComfTrellis": [
[
"LoadTrellisModel",
"RembgSquare",
"SaveGLBFile",
"TrellisGrid",
"TrellisInference"
],
{
"title_aux": "ComfTrellis [WIP]"
}
],
"https://github.com/SeedV/ComfyUI-SeedV-Nodes": [
[
"ALL_Model_UnLoader(SEEDV)",
@@ -1684,6 +1785,15 @@
"title_aux": "plugin-utils-nodes"
}
],
"https://github.com/Velour-Fog/comfy-latent-nodes": [
[
"CustomLoadLatent",
"CustomSaveLatent"
],
{
"title_aux": "comfy-latent-nodes [UNSAFE]"
}
],
"https://github.com/Video3DGenResearch/comfyui-batch-input-node": [
[
"BatchImageAndPrompt",
@@ -1737,6 +1847,16 @@
"title_aux": "visuallabs_comfyui_nodes"
}
],
"https://github.com/Yeonri/ComfyUI_LLM_Are_You_Listening": [
[
"AYL_API_Node",
"AYL_GGUF_Node",
"AYL_Node"
],
{
"title_aux": "ComfyUI_LLM_Are_You_Listening [WIP]"
}
],
"https://github.com/ZHO-ZHO-ZHO/ComfyUI-AuraSR-ZHO": [
[
"AuraSR_Lterative_Zho",
@@ -1842,6 +1962,39 @@
"title_aux": "alexisrolland/ComfyUI-AuraSR"
}
],
"https://github.com/alt-key-project/comfyui-dream-painter": [
[
"Bitmap AND [DPaint]",
"Bitmap Crop Center [DPaint]",
"Bitmap Dimensions [DPaint]",
"Bitmap Edge Detect [DPaint]",
"Bitmap Expand Canvas [DPaint]",
"Bitmap Invert [DPaint]",
"Bitmap OR [DPaint]",
"Bitmap Resize [DPaint]",
"Bitmap Rotate [DPaint]",
"Bitmap To Image & Mask [DPaint]",
"Bitmap XOR [DPaint]",
"Draw Shape As Bitmap [DPaint]",
"Image To Bitmap [DPaint]",
"Random Number Generator [DPaint]",
"Shape Center & Fit [DPaint]",
"Shape Combiner [DPaint]",
"Shape Copycat Tool [DPaint]",
"Shape Find Bounds [DPaint]",
"Shape Flip [DPaint]",
"Shape Grid [DPaint]",
"Shape Resize [DPaint]",
"Shape Rotate [DPaint]",
"Shape of Circular Rays [DPaint]",
"Shape of N-Polygon [DPaint]",
"Shape of Rectangle [DPaint]",
"Shape of Star [DPaint]"
],
{
"title_aux": "Dream Painter [WIP]"
}
],
"https://github.com/alt-key-project/comfyui-dream-video-batches": [
[
"Blended Transition [DVB]",
@@ -1934,20 +2087,6 @@
"title_aux": "comfyui_segformer_b2_sleeves"
}
],
"https://github.com/attashe/ComfyUI-FluxRegionAttention": [
[
"BBoxToMaskNode",
"BoundingBoxNode",
"CLIPDebug",
"FluxRegionBBOX",
"FluxRegionMask",
"RegionAttention",
"VisualizeBBoxesNode"
],
{
"title_aux": "ComfyUI-FluxRegionAttention [WIP]"
}
],
"https://github.com/backearth1/Comfyui-MiniMax-Video": [
[
"ImageToPrompt",
@@ -2348,6 +2487,7 @@
"CLIPTextEncodeControlnet",
"CLIPTextEncodeFlux",
"CLIPTextEncodeHunyuanDiT",
"CLIPTextEncodeLumina2",
"CLIPTextEncodePixArtAlpha",
"CLIPTextEncodeSD3",
"CLIPTextEncodeSDXL",
@@ -2456,6 +2596,7 @@
"LoadAudio",
"LoadImage",
"LoadImageMask",
"LoadImageOutput",
"LoadLatent",
"LoraLoader",
"LoraLoaderModelOnly",
@@ -2463,6 +2604,7 @@
"Mahiro",
"MaskComposite",
"MaskToImage",
"ModelComputeDtype",
"ModelMergeAdd",
"ModelMergeAuraflow",
"ModelMergeBlocks",
@@ -2503,6 +2645,7 @@
"RandomNoise",
"RebatchImages",
"RebatchLatents",
"RenormCFG",
"RepeatImageBatch",
"RepeatLatentBatch",
"RescaleCFG",
@@ -2528,6 +2671,7 @@
"SaveImage",
"SaveImageWebsocket",
"SaveLatent",
"SaveWEBM",
"SelfAttentionGuidance",
"SetFirstSigma",
"SetLatentNoiseMask",
@@ -2670,15 +2814,6 @@
"title_aux": "VoidCustomNodes"
}
],
"https://github.com/dasilva333/ComfyUI_MarkdownImage": [
[
"CreateDialogImage",
"CreateMarkdownImage"
],
{
"title_aux": "ComfyUI_MarkdownImage [WIP]"
}
],
"https://github.com/denislov/Comfyui_AutoSurvey": [
[
"AddDoc2Knowledge",
@@ -2889,6 +3024,14 @@
"title_aux": "ComfyUI-Showrunner-Utils"
}
],
"https://github.com/fangziheng2321/comfyuinode_chopmask": [
[
"cus_chopmask"
],
{
"title_aux": "comfyuinode_chopmask [WIP]"
}
],
"https://github.com/flowtyone/comfyui-flowty-lcm": [
[
"LCMSampler"
@@ -2972,6 +3115,16 @@
"title_aux": "comfyui_median_filter"
}
],
"https://github.com/gmorks/ComfyUI-Animagine-Prompt": [
[
"AnimaginePrompt",
"MultilineTextInput",
"TextFileLoader"
],
{
"title_aux": "ComfyUI Animagine prompt [WIP]"
}
],
"https://github.com/go-package-lab/ComfyUI-Tools-Video-Combine": [
[
"Tools:CopyFile",
@@ -2998,16 +3151,6 @@
"title_aux": "loki-comfyui-node"
}
],
"https://github.com/greengerong/ComfyUI-Lumina-Video": [
[
"LuminaVideoModelLoader",
"LuminaVideoSampler",
"LuminaVideoVAEDecode"
],
{
"title_aux": "ComfyUI-Lumina-Video [WIP]"
}
],
"https://github.com/grimli333/ComfyUI_Grim": [
[
"GenerateFileName",
@@ -3017,6 +3160,17 @@
"title_aux": "ComfyUI_Grim"
}
],
"https://github.com/grinlau18/ComfyUI_XISER_Nodes": [
[
"XIS_Float_Slider",
"XIS_INT_Slider",
"XIS_PromptsWithSwitches",
"XIS_ResizeImageOrMask"
],
{
"title_aux": "Xiser_Nodes [WIP]"
}
],
"https://github.com/haodman/ComfyUI_Rain": [
[
"Rain_ImageSize",
@@ -3055,6 +3209,7 @@
[
"ACE_AnyInputSwitchBool",
"ACE_AnyInputToAny",
"ACE_AudioCrop",
"ACE_AudioLoad",
"ACE_AudioPlay",
"ACE_AudioSave",
@@ -3065,6 +3220,7 @@
"ACE_ImageFaceCrop",
"ACE_ImageGetSize",
"ACE_ImageLoadFromCloud",
"ACE_ImageMakeSlieshow",
"ACE_ImagePixelate",
"ACE_ImageQA",
"ACE_ImageRemoveBackground",
@@ -3083,6 +3239,7 @@
"ACE_TextSelector",
"ACE_TextToResolution",
"ACE_TextTranslate",
"ACE_VideoConcat",
"ACE_VideoLoad",
"ACE_VideoPreview"
],
@@ -3092,10 +3249,12 @@
],
"https://github.com/hdfhssg/ComfyUI_pxtool": [
[
"ArtistLoader",
"CivitaiHelper",
"DanbooruCharacterTag",
"E621CharacterTag",
"NegativeTag",
"PX_Seed",
"QualityTag",
"RandomArtists",
"RandomArtistsAdvanced",
@@ -3105,6 +3264,14 @@
"title_aux": "ComfyUI_pxtool [WIP]"
}
],
"https://github.com/hiusdev/ComfyUI_Lah_Toffee": [
[
"LoadVideoRandom"
],
{
"title_aux": "ComfyUI_Lah_Toffee"
}
],
"https://github.com/horidream/ComfyUI-Horidream": [
[
"PassThroughWithSound"
@@ -3237,6 +3404,14 @@
"title_aux": "comfyui-hydit"
}
],
"https://github.com/if-ai/ComfyUI-IF_Zonos": [
[
"IF_ZonosTTS"
],
{
"title_aux": "ComfyUI-IF_Zonos [WIP]"
}
],
"https://github.com/ilovejohnwhite/Tracer": [
[
"BillyGoatNode",
@@ -3314,6 +3489,14 @@
"title_aux": "ComfyUI-ComfyFluxSize [WIP]"
}
],
"https://github.com/jgbyte/ComfyUI-RandomCube": [
[
"RandomCubeGrid"
],
{
"title_aux": "ComfyUI-RandomCube [WIP]"
}
],
"https://github.com/jimmm-ai/TimeUi-a-ComfyUi-Timeline-Node": [
[
"jimmm.ai.TimelineUI"
@@ -3421,6 +3604,7 @@
],
"https://github.com/jonnydolake/ComfyUI-AIR-Nodes": [
[
"ExtractBlackLines",
"ForceMinimumBatchSize",
"ImageCompositeChained",
"LineDetection",
@@ -3488,6 +3672,8 @@
],
"https://github.com/kandy/ComfyUI-KAndy": [
[
"KAndyBatch2Index",
"KAndyBatchIndex",
"KAndyCivitImagesAPI",
"KAndyCivitPromptAPI",
"KAndyImageSave",
@@ -3593,6 +3779,7 @@
"DownloadAndLoadHy3DDelightModel",
"DownloadAndLoadHy3DPaintModel",
"Hy3DApplyTexture",
"Hy3DBPT",
"Hy3DBakeFromMultiview",
"Hy3DCameraConfig",
"Hy3DDelightImage",
@@ -3732,7 +3919,8 @@
"Custom_Save_Image",
"Display_Any",
"Image_Size_Extractor",
"Strong_Prompt"
"Strong_Prompt",
"Tencent_Translater"
],
{
"title_aux": "KayTool"
@@ -3765,6 +3953,17 @@
"title_aux": "ComfyUI Flow Control [UNSTABLE]"
}
],
"https://github.com/krisshen2021/comfyui_OpenRouterNodes": [
[
"OpenRouterOAINode_Infer",
"OpenRouterOAINode_Models",
"OpenRouterOAINode_hunyuanPrompt",
"OpenRouterOAINode_txt2imgPrompt"
],
{
"title_aux": "comfyui_OpenRouterNodes [WIP]"
}
],
"https://github.com/kuschanow/ComfyUI-SD-Slicer": [
[
"SdSlicer"
@@ -3936,6 +4135,9 @@
"HYCreateRegionalCond",
"HYFetaEnhance",
"HYFlowEditGuider",
"HYFlowEditGuiderAdv",
"HYFlowEditGuiderCFG",
"HYFlowEditGuiderCFGAdv",
"HYFlowEditSampler",
"HYForwardODESampler",
"HYInverseModelSamplingPred",
@@ -4028,7 +4230,9 @@
],
"https://github.com/lum3on/comfyui_LLM_Polymath": [
[
"SaveAbsolute",
"Helper",
"UCEEraserNode",
"polymath_SaveAbsolute",
"polymath_chat",
"polymath_scraper"
],
@@ -4100,6 +4304,14 @@
"title_aux": "ComfyUI mashb1t nodes"
}
],
"https://github.com/masmullin2000/ComfyUI-MMYolo": [
[
"MMFace_Finder"
],
{
"title_aux": "ComfyUI-MMYolo"
}
],
"https://github.com/mehbebe/ComfyLoraGallery": [
[
"LoraGallery"
@@ -4237,6 +4449,16 @@
"title_aux": "ComfyUI-Simple-Image-Tools [WIP]"
}
],
"https://github.com/mr-krak3n/ComfyUI-Qwen": [
[
"DeepSeekResponseParser",
"QwenLoader",
"QwenSampler"
],
{
"title_aux": "ComfyUI-Qwen [CONFLICT]"
}
],
"https://github.com/mut-ex/comfyui-gligengui-node": [
[
"GLIGEN_GUI"
@@ -4253,6 +4475,14 @@
"title_aux": "ComfyUI-Claude-I2T"
}
],
"https://github.com/muvich3n/ComfyUI-Crop-Border": [
[
"CropImageBorder"
],
{
"title_aux": "ComfyUI-Crop-Border"
}
],
"https://github.com/myAiLemon/MagicAutomaticPicture": [
[
"EditableStringNode",
@@ -4444,6 +4674,14 @@
"title_aux": "comfyui-keshigom_custom"
}
],
"https://github.com/owengillett/ComfyUI-tilefusion": [
[
"VideoGridCombine"
],
{
"title_aux": "ComfyUI-tilefusion"
}
],
"https://github.com/oyvindg/ComfyUI-TrollSuite": [
[
"BinaryImageMask",
@@ -5212,6 +5450,7 @@
[
"CLIPTranslatedNode",
"DownloadImageNode",
"FillAlphaNode",
"FixUTF8StringNode",
"ImageResizeNode",
"ImageToVideoNode",

View File

File diff suppressed because it is too large Load Diff

View File

@@ -129,6 +129,16 @@
],
"install_type": "git-clone",
"description": "A forked version of ComfyUI_ExtraModels. (modified by Efficient-Large-Model)"
},
{
"author": "Pablerdo",
"title": "ComfyUI-PSNodes",
"reference": "https://github.com/Pablerdo/ComfyUI-PSNodes",
"files": [
"https://github.com/Pablerdo/ComfyUI-PSNodes"
],
"install_type": "git-clone",
"description": "A fork of KJNodes for ComfyUI.\nVarious quality of life -nodes for ComfyUI, mostly just visual stuff to improve usability"
}
]
}

View File

@@ -10,6 +10,58 @@
},
{
"author": "thanhduong0213929",
"title": "ComfyUI-DeepUnlock [REMOVED]",
"reference": "https://github.com/thanhduong0213929/ComfyUI-DeepUnlock",
"files": [
"https://github.com/thanhduong0213929/ComfyUI-DeepUnlock"
],
"install_type": "git-clone",
"description": "DeepFuze is a state-of-the-art deep learning tool that seamlessly integrates with ComfyUI to revolutionize facial transformations, lipsyncing, video generation, voice cloning, face swapping, and lipsync translation. Leveraging advanced algorithms, DeepFuze enables users to combine audio and video with unparalleled realism, ensuring perfectly synchronized facial movements. This innovative solution is ideal for content creators, animators, developers, and anyone seeking to elevate their video editing projects with sophisticated AI-driven features."
},
{
"author": "pathway8-sudo",
"title": "RMBG [REMOVED]",
"reference": "https://github.com/pathway8-sudo/RMBG",
"files": [
"https://github.com/pathway8-sudo/RMBG"
],
"install_type": "git-clone",
"description": "This repository provides a custom node for ComfyUI, leveraging the BriaRMBG model to remove backgrounds from images and output a transparent PNG."
},
{
"author": "iris-Neko",
"title": "ComfyUI_ascii_art [REMOVED]",
"reference": "https://github.com/iris-Neko/ComfyUI_ascii_art",
"files": [
"https://github.com/iris-Neko/ComfyUI_ascii_art"
],
"install_type": "git-clone",
"description": "ComfyUI node for [a/ASCII art controlnet](https://civitai.com/models/986392)"
},
{
"author": "apesplat",
"title": "ezXY scripts and nodes [NOT MAINTAINED]",
"id": "ezxy",
"reference": "https://github.com/GMapeSplat/ComfyUI_ezXY",
"files": [
"https://github.com/GMapeSplat/ComfyUI_ezXY"
],
"install_type": "git-clone",
"description": "Extensions/Patches: Enables linking float and integer inputs and ouputs. Values are automatically cast to the correct type and clamped to the correct range. Works with both builtin and custom nodes.[w/NOTE: This repo patches ComfyUI's validate_inputs and map_node_over_list functions while running. May break depending on your version of ComfyUI. Can be deactivated in config.yaml.]Nodes: A collection of nodes for facilitating the generation of XY plots. Capable of plotting changes over most primitive values.[w/Does not work with current version of Comfyui]"
},
{
"author": "mie",
"title": "ComfyUI_JanusProCaption [REMOVED]",
"reference": "https://github.com/MieMieeeee/ComfyUI-JanusProCaption",
"files": [
"https://github.com/MieMieeeee/ComfyUI-JanusProCaption"
],
"install_type": "git-clone",
"description": "Describe image or create caption files using Janus Pro Model"
},
{
"author": "Njbx",
"title": "ComfyUI-blockswap [REMOVED]",

View File

File diff suppressed because it is too large Load Diff

View File

File diff suppressed because it is too large Load Diff

View File

@@ -1,5 +1,28 @@
{
"models": [
{
"name": "kolors/vae/diffusion_pytorch_model.fp16.safetensors",
"type": "VAE",
"base": "Kolors",
"save_path": "vae/kolors",
"description": "Kolors VAE",
"reference": "https://huggingface.co/Kwai-Kolors/Kolors",
"filename": "diffusion_pytorch_model.fp16.safetensors",
"url": "https://huggingface.co/Kwai-Kolors/Kolors/resolve/main/vae/diffusion_pytorch_model.fp16.safetensors",
"size": "167MB"
},
{
"name": "kolors/vae/diffusion_pytorch_model.safetensors",
"type": "VAE",
"base": "Kolors",
"save_path": "vae/kolors",
"description": "Kolors VAE",
"reference": "https://huggingface.co/Kwai-Kolors/Kolors",
"filename": "diffusion_pytorch_model.safetensors",
"url": "https://huggingface.co/Kwai-Kolors/Kolors/resolve/main/vae/diffusion_pytorch_model.safetensors",
"size": "335MB"
},
{
"name": "deepseek-ai/Janus-Pro-1B",
"type": "Janus-Pro",

View File

@@ -291,6 +291,26 @@
],
"install_type": "git-clone",
"description": "Example of using ComfyUI Toolbar to Toggle ComfyUI links on/off"
},
{
"author": "xhiroga",
"title": "ComfyUI-TypeScript-CustomNode",
"reference": "https://github.com/xhiroga/ComfyUI-TypeScript-CustomNode",
"files": [
"https://github.com/xhiroga/ComfyUI-TypeScript-CustomNode"
],
"install_type": "git-clone",
"description": "This project is generated from xhiroga/ComfyUI-TypeScript-CustomNode"
},
{
"author": "zentrocdot",
"title": "ComfyUI-Turtle_Graphics_Demos",
"reference": "https://github.com/zentrocdot/ComfyUI-Turtle_Graphics_Demo",
"files": [
"https://github.com/zentrocdot/ComfyUI-Turtle_Graphics_Demo"
],
"description": "ComfyUI node for creating some Turtle Graphic demos.",
"install_type": "git-clone"
}
]
}

View File

@@ -34,7 +34,7 @@ else:
security_check.security_check()
cm_global.pip_blacklist = ['torch', 'torchsde', 'torchvision']
cm_global.pip_blacklist = {'torch', 'torchsde', 'torchvision'}
cm_global.pip_downgrade_blacklist = ['torch', 'torchsde', 'torchvision', 'transformers', 'safetensors', 'kornia']
@@ -82,6 +82,7 @@ comfyui_manager_path = os.path.abspath(os.path.dirname(__file__))
custom_nodes_base_path = folder_paths.get_folder_paths('custom_nodes')[0]
manager_files_path = os.path.abspath(os.path.join(folder_paths.get_user_directory(), 'default', 'ComfyUI-Manager'))
manager_pip_overrides_path = os.path.join(manager_files_path, "pip_overrides.json")
manager_pip_blacklist_path = os.path.join(manager_files_path, "pip_blacklist.list")
restore_snapshot_path = os.path.join(manager_files_path, "startup-scripts", "restore-snapshot.json")
manager_config_path = os.path.join(manager_files_path, 'config.ini')
@@ -94,7 +95,7 @@ def read_config():
global default_conf
try:
import configparser
config = configparser.ConfigParser()
config = configparser.ConfigParser(strict=False)
config.read(manager_config_path)
default_conf = config['default']
except Exception:
@@ -122,6 +123,14 @@ if os.path.exists(manager_pip_overrides_path):
cm_global.pip_overrides['ultralytics'] = 'ultralytics==8.3.40' # for security
if os.path.exists(manager_pip_blacklist_path):
with open(manager_pip_blacklist_path, 'r', encoding="UTF-8", errors="ignore") as f:
for x in f.readlines():
y = x.strip()
if y != '':
cm_global.pip_blacklist.add(y)
def remap_pip_package(pkg):
if pkg in cm_global.pip_overrides:
res = cm_global.pip_overrides[pkg]
@@ -421,29 +430,33 @@ except Exception as e:
print(f"[ComfyUI-Manager] Logging failed: {e}")
try:
import git # noqa: F401
import toml # noqa: F401
import rich # noqa: F401
except ModuleNotFoundError:
my_path = os.path.dirname(__file__)
requirements_path = os.path.join(my_path, "requirements.txt")
print("## ComfyUI-Manager: installing dependencies. (GitPython)")
def ensure_dependencies():
try:
result = subprocess.check_output(manager_util.make_pip_cmd(['install', '-r', requirements_path]))
except subprocess.CalledProcessError:
print("## [ERROR] ComfyUI-Manager: Attempting to reinstall dependencies using an alternative method.")
try:
result = subprocess.check_output(manager_util.make_pip_cmd(['install', '--user', '-r', requirements_path]))
except subprocess.CalledProcessError:
print("## [ERROR] ComfyUI-Manager: Failed to install the GitPython package in the correct Python environment. Please install it manually in the appropriate environment. (You can seek help at https://app.element.io/#/room/%23comfyui_space%3Amatrix.org)")
import git # noqa: F401
import toml # noqa: F401
import rich # noqa: F401
import chardet # noqa: F401
except ModuleNotFoundError:
my_path = os.path.dirname(__file__)
requirements_path = os.path.join(my_path, "requirements.txt")
try:
print("## ComfyUI-Manager: installing dependencies done.")
except:
# maybe we should sys.exit() here? there is at least two screens worth of error messages still being pumped after our error messages
print("## [ERROR] ComfyUI-Manager: GitPython package seems to be installed, but failed to load somehow. Make sure you have a working git client installed")
print("## ComfyUI-Manager: installing dependencies. (GitPython)")
try:
subprocess.check_output(manager_util.make_pip_cmd(['install', '-r', requirements_path]))
except subprocess.CalledProcessError:
print("## [ERROR] ComfyUI-Manager: Attempting to reinstall dependencies using an alternative method.")
try:
subprocess.check_output(manager_util.make_pip_cmd(['install', '--user', '-r', requirements_path]))
except subprocess.CalledProcessError:
print("## [ERROR] ComfyUI-Manager: Failed to install the GitPython package in the correct Python environment. Please install it manually in the appropriate environment. (You can seek help at https://app.element.io/#/room/%23comfyui_space%3Amatrix.org)")
try:
print("## ComfyUI-Manager: installing dependencies done.")
except:
# maybe we should sys.exit() here? there is at least two screens worth of error messages still being pumped after our error messages
print("## [ERROR] ComfyUI-Manager: GitPython package seems to be installed, but failed to load somehow. Make sure you have a working git client installed")
ensure_dependencies()
print("** ComfyUI startup time:", current_timestamp())
@@ -598,17 +611,18 @@ def execute_lazy_install_script(repo_path, executable):
if os.path.exists(requirements_path):
print(f"Install: pip packages for '{repo_path}'")
with open(requirements_path, "r") as requirements_file:
for line in requirements_file:
package_name = remap_pip_package(line.strip())
if package_name and not is_installed(package_name):
if '--index-url' in package_name:
s = package_name.split('--index-url')
install_cmd = manager_util.make_pip_cmd(["install", s[0].strip(), '--index-url', s[1].strip()])
else:
install_cmd = manager_util.make_pip_cmd(["install", package_name])
process_wrap(install_cmd, repo_path)
lines = manager_util.robust_readlines(requirements_path)
for line in lines:
package_name = remap_pip_package(line.strip())
if package_name and not is_installed(package_name):
if '--index-url' in package_name:
s = package_name.split('--index-url')
install_cmd = manager_util.make_pip_cmd(["install", s[0].strip(), '--index-url', s[1].strip()])
else:
install_cmd = manager_util.make_pip_cmd(["install", package_name])
process_wrap(install_cmd, repo_path)
if os.path.exists(install_script_path) and f'{repo_path}/install.py' not in processed_install:
processed_install.add(f'{repo_path}/install.py')
@@ -679,6 +693,7 @@ def execute_migration(moves):
shutil.move(x[0], x[1])
print(f"[ComfyUI-Manager] MIGRATION: '{x[0]}' -> '{x[1]}'")
script_executed = False
# Check if script_list_path exists
if os.path.exists(script_list_path):
@@ -733,6 +748,7 @@ if os.path.exists(script_list_path):
# Remove the script_list_path file
if os.path.exists(script_list_path):
script_executed = True
os.remove(script_list_path)
print("\n[ComfyUI-Manager] Startup script completed.")
@@ -744,11 +760,34 @@ del processed_install
del pip_fixer
manager_util.clear_pip_cache()
if script_executed:
# Restart
print("[ComfyUI-Manager] Restarting to reapply dependency installation.")
if '__COMFY_CLI_SESSION__' in os.environ:
with open(os.path.join(os.environ['__COMFY_CLI_SESSION__'] + '.reboot'), 'w'):
pass
print("--------------------------------------------------------------------------\n")
exit(0)
else:
sys_argv = sys.argv.copy()
if sys.platform.startswith('win32'):
cmds = ['"' + sys.executable + '"', '"' + sys_argv[0] + '"'] + sys_argv[1:]
else:
cmds = [sys.executable] + sys_argv
print(f"Command: {cmds}", flush=True)
print("--------------------------------------------------------------------------\n")
os.execv(sys.executable, cmds)
def check_windows_event_loop_policy():
try:
import configparser
config = configparser.ConfigParser()
config = configparser.ConfigParser(strict=False)
config.read(manager_config_path)
default_conf = config['default']

View File

@@ -1,9 +1,9 @@
[project]
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."
version = "3.21.1"
version = "3.26"
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", "toml", "uv", "chardet"]
[project.urls]
Repository = "https://github.com/ltdrdata/ComfyUI-Manager"

View File

@@ -8,3 +8,4 @@ rich
typing-extensions
toml
uv
chardet

View File

@@ -6,7 +6,7 @@ python -m venv venv
call venv/Scripts/activate
python -m pip install torch torchvision torchaudio --extra-index-url https://download.pytorch.org/whl/cu121
python -m pip install -r requirements.txt
python -m pip install -r custom_nodes/ComfyUI-Manager/requirements.txt
python -m pip install -r custom_nodes/comfyui-manager/requirements.txt
cd ..
echo "cd ComfyUI" >> run_gpu.bat
echo "call venv/Scripts/activate" >> run_gpu.bat

View File

@@ -1,2 +1,3 @@
.\python_embeded\python.exe -s -m pip install gitpython
.\python_embeded\python.exe -c "import git; git.Repo.clone_from('https://github.com/ltdrdata/ComfyUI-Manager', './ComfyUI/custom_nodes/comfyui-manager')"
.\python_embeded\python.exe -m pip install -r ./ComfyUI/custom_nodes/comfyui-manager/requirements.txt