Spaces:
Running
on
Zero
Running
on
Zero
import os | |
from pathlib import Path | |
from typing import Optional | |
from pydantic_settings import PydanticBaseSettingsSource, TomlConfigSettingsSource | |
from comfy_config.types import ( | |
ComfyConfig, | |
ProjectConfig, | |
PyProjectConfig, | |
PyProjectSettings | |
) | |
def validate_and_extract_os_classifiers(classifiers: list) -> list: | |
os_classifiers = [c for c in classifiers if c.startswith("Operating System :: ")] | |
if not os_classifiers: | |
return [] | |
os_values = [c[len("Operating System :: ") :] for c in os_classifiers] | |
valid_os_prefixes = {"Microsoft", "POSIX", "MacOS", "OS Independent"} | |
for os_value in os_values: | |
if not any(os_value.startswith(prefix) for prefix in valid_os_prefixes): | |
return [] | |
return os_values | |
def validate_and_extract_accelerator_classifiers(classifiers: list) -> list: | |
accelerator_classifiers = [c for c in classifiers if c.startswith("Environment ::")] | |
if not accelerator_classifiers: | |
return [] | |
accelerator_values = [c[len("Environment :: ") :] for c in accelerator_classifiers] | |
valid_accelerators = { | |
"GPU :: NVIDIA CUDA", | |
"GPU :: AMD ROCm", | |
"GPU :: Intel Arc", | |
"NPU :: Huawei Ascend", | |
"GPU :: Apple Metal", | |
} | |
for accelerator_value in accelerator_values: | |
if accelerator_value not in valid_accelerators: | |
return [] | |
return accelerator_values | |
""" | |
Extract configuration from a custom node directory's pyproject.toml file or a Python file. | |
This function reads and parses the pyproject.toml file in the specified directory | |
to extract project and ComfyUI-specific configuration information. If no | |
pyproject.toml file is found, it creates a minimal configuration using the | |
folder name as the project name. If a Python file is provided, it uses the | |
file name (without extension) as the project name. | |
Args: | |
path (str): Path to the directory containing the pyproject.toml file, or | |
path to a .py file. If pyproject.toml doesn't exist in a directory, | |
the folder name will be used as the default project name. If a .py | |
file is provided, the filename (without .py extension) will be used | |
as the project name. | |
Returns: | |
Optional[PyProjectConfig]: A PyProjectConfig object containing: | |
- project: Basic project information (name, version, dependencies, etc.) | |
- tool_comfy: ComfyUI-specific configuration (publisher_id, models, etc.) | |
Returns None if configuration extraction fails or if the provided file | |
is not a Python file. | |
Notes: | |
- If pyproject.toml is missing in a directory, creates a default config with folder name | |
- If a .py file is provided, creates a default config with filename (without extension) | |
- Returns None for non-Python files | |
Example: | |
>>> from comfy_config import config_parser | |
>>> # For directory | |
>>> custom_node_dir = os.path.dirname(os.path.realpath(__file__)) | |
>>> project_config = config_parser.extract_node_configuration(custom_node_dir) | |
>>> print(project_config.project.name) # "my_custom_node" or name from pyproject.toml | |
>>> | |
>>> # For single-file Python node file | |
>>> py_file_path = os.path.realpath(__file__) # "/path/to/my_node.py" | |
>>> project_config = config_parser.extract_node_configuration(py_file_path) | |
>>> print(project_config.project.name) # "my_node" | |
""" | |
def extract_node_configuration(path) -> Optional[PyProjectConfig]: | |
if os.path.isfile(path): | |
file_path = Path(path) | |
if file_path.suffix.lower() != '.py': | |
return None | |
project_name = file_path.stem | |
project = ProjectConfig(name=project_name) | |
comfy = ComfyConfig() | |
return PyProjectConfig(project=project, tool_comfy=comfy) | |
folder_name = os.path.basename(path) | |
toml_path = Path(path) / "pyproject.toml" | |
if not toml_path.exists(): | |
project = ProjectConfig(name=folder_name) | |
comfy = ComfyConfig() | |
return PyProjectConfig(project=project, tool_comfy=comfy) | |
raw_settings = load_pyproject_settings(toml_path) | |
project_data = raw_settings.project | |
tool_data = raw_settings.tool | |
comfy_data = tool_data.get("comfy", {}) if tool_data else {} | |
dependencies = project_data.get("dependencies", []) | |
supported_comfyui_frontend_version = "" | |
for dep in dependencies: | |
if isinstance(dep, str) and dep.startswith("comfyui-frontend-package"): | |
supported_comfyui_frontend_version = dep.removeprefix("comfyui-frontend-package") | |
break | |
supported_comfyui_version = comfy_data.get("requires-comfyui", "") | |
classifiers = project_data.get('classifiers', []) | |
supported_os = validate_and_extract_os_classifiers(classifiers) | |
supported_accelerators = validate_and_extract_accelerator_classifiers(classifiers) | |
project_data['supported_os'] = supported_os | |
project_data['supported_accelerators'] = supported_accelerators | |
project_data['supported_comfyui_frontend_version'] = supported_comfyui_frontend_version | |
project_data['supported_comfyui_version'] = supported_comfyui_version | |
return PyProjectConfig(project=project_data, tool_comfy=comfy_data) | |
def load_pyproject_settings(toml_path: Path) -> PyProjectSettings: | |
class PyProjectLoader(PyProjectSettings): | |
def settings_customise_sources( | |
cls, | |
settings_cls, | |
init_settings: PydanticBaseSettingsSource, | |
env_settings: PydanticBaseSettingsSource, | |
dotenv_settings: PydanticBaseSettingsSource, | |
file_secret_settings: PydanticBaseSettingsSource, | |
): | |
return (TomlConfigSettingsSource(settings_cls, toml_path),) | |
return PyProjectLoader() | |