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()