|
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): |
|
@classmethod |
|
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() |
|
|