mirror of
https://github.com/comfyanonymous/ComfyUI.git
synced 2026-03-30 13:33:42 +08:00
feat: switch SubgraphManager to index.json-driven discovery with distribution filtering
Amp-Thread-ID: https://ampcode.com/threads/T-019c30d2-a605-708d-824f-35e8f3a0c2f5
This commit is contained in:
parent
2b70ab9ad0
commit
6797e20bcf
@ -1,6 +1,8 @@
|
|||||||
from __future__ import annotations
|
from __future__ import annotations
|
||||||
|
|
||||||
from typing import TypedDict
|
from typing import TypedDict
|
||||||
|
import json
|
||||||
|
import logging
|
||||||
import os
|
import os
|
||||||
import folder_paths
|
import folder_paths
|
||||||
import glob
|
import glob
|
||||||
@ -40,6 +42,7 @@ class SubgraphManager:
|
|||||||
def __init__(self):
|
def __init__(self):
|
||||||
self.cached_custom_node_subgraphs: dict[SubgraphEntry] | None = None
|
self.cached_custom_node_subgraphs: dict[SubgraphEntry] | None = None
|
||||||
self.cached_blueprint_subgraphs: dict[SubgraphEntry] | None = None
|
self.cached_blueprint_subgraphs: dict[SubgraphEntry] | None = None
|
||||||
|
self.distribution = os.environ.get("DISTRIBUTION", "localhost")
|
||||||
|
|
||||||
def _create_entry(self, file: str, source: str, node_pack: str) -> tuple[str, SubgraphEntry]:
|
def _create_entry(self, file: str, source: str, node_pack: str) -> tuple[str, SubgraphEntry]:
|
||||||
"""Create a subgraph entry from a file path. Expects normalized path (forward slashes)."""
|
"""Create a subgraph entry from a file path. Expects normalized path (forward slashes)."""
|
||||||
@ -90,15 +93,56 @@ class SubgraphManager:
|
|||||||
return subgraphs_dict
|
return subgraphs_dict
|
||||||
|
|
||||||
async def get_blueprint_subgraphs(self, force_reload=False):
|
async def get_blueprint_subgraphs(self, force_reload=False):
|
||||||
"""Load subgraphs from the blueprints directory."""
|
"""Load subgraphs from the blueprints directory using index.json for discovery."""
|
||||||
if not force_reload and self.cached_blueprint_subgraphs is not None:
|
if not force_reload and self.cached_blueprint_subgraphs is not None:
|
||||||
return self.cached_blueprint_subgraphs
|
return self.cached_blueprint_subgraphs
|
||||||
|
|
||||||
subgraphs_dict: dict[SubgraphEntry] = {}
|
subgraphs_dict: dict[SubgraphEntry] = {}
|
||||||
blueprints_dir = os.path.join(os.path.dirname(os.path.dirname(__file__)), 'blueprints')
|
blueprints_dir = os.path.join(os.path.dirname(os.path.dirname(__file__)), 'blueprints')
|
||||||
|
|
||||||
if os.path.exists(blueprints_dir):
|
index_path = os.path.join(blueprints_dir, "index.json")
|
||||||
|
if os.path.isfile(index_path):
|
||||||
|
try:
|
||||||
|
with open(index_path, "r", encoding="utf-8") as f:
|
||||||
|
categories = json.load(f)
|
||||||
|
except (json.JSONDecodeError, OSError) as e:
|
||||||
|
logging.error("Failed to load blueprint index %s: %s", index_path, e)
|
||||||
|
categories = []
|
||||||
|
|
||||||
|
if not isinstance(categories, list):
|
||||||
|
logging.error("Blueprint index.json is not a list: %s", index_path)
|
||||||
|
categories = []
|
||||||
|
|
||||||
|
for category in categories:
|
||||||
|
module_name = category.get("moduleName", "default")
|
||||||
|
for blueprint in category.get("blueprints", []):
|
||||||
|
name = blueprint.get("name")
|
||||||
|
if not name:
|
||||||
|
logging.warning("Blueprint entry missing 'name' in category '%s', skipping", module_name)
|
||||||
|
continue
|
||||||
|
|
||||||
|
include_on = blueprint.get("includeOnDistributions")
|
||||||
|
if include_on is not None and self.distribution not in include_on:
|
||||||
|
continue
|
||||||
|
|
||||||
|
file_by_dist = blueprint.get("fileByDistribution", {})
|
||||||
|
filename = file_by_dist.get(self.distribution, f"{name}.json")
|
||||||
|
filepath = os.path.realpath(os.path.join(blueprints_dir, filename))
|
||||||
|
if not filepath.startswith(os.path.realpath(blueprints_dir) + os.sep):
|
||||||
|
logging.warning("Blueprint path escapes blueprints directory: %s", filepath)
|
||||||
|
continue
|
||||||
|
|
||||||
|
if not os.path.isfile(filepath):
|
||||||
|
logging.warning("Blueprint file not found: %s", filepath)
|
||||||
|
continue
|
||||||
|
|
||||||
|
entry_id, entry = self._create_entry(filepath, Source.templates, module_name)
|
||||||
|
subgraphs_dict[entry_id] = entry
|
||||||
|
elif os.path.exists(blueprints_dir):
|
||||||
|
logging.warning("No blueprint index.json found at %s, falling back to glob", index_path)
|
||||||
for file in glob.glob(os.path.join(blueprints_dir, "*.json")):
|
for file in glob.glob(os.path.join(blueprints_dir, "*.json")):
|
||||||
|
if os.path.basename(file) == "index.json":
|
||||||
|
continue
|
||||||
file = file.replace('\\', '/')
|
file = file.replace('\\', '/')
|
||||||
entry_id, entry = self._create_entry(file, Source.templates, "comfyui")
|
entry_id, entry = self._create_entry(file, Source.templates, "comfyui")
|
||||||
subgraphs_dict[entry_id] = entry
|
subgraphs_dict[entry_id] = entry
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user