mirror of
https://github.com/bytedance/deer-flow.git
synced 2026-05-02 06:48:21 +00:00
* fix(harness): resolve runtime paths from project root * docs(config): update * fix(config): address runtime path review feedback * test(config): fix skills path e2e root * test(config): cover legacy config fallback when project root lacks config files Verifies that when DEER_FLOW_PROJECT_ROOT is unset and cwd has no config.yaml/extensions_config.json, AppConfig and ExtensionsConfig fall back to the legacy backend/repo-root candidates — the backward-compat path requested in PR #2642 review. --------- Co-authored-by: Willem Jiang <willem.jiang@gmail.com>
42 lines
1.4 KiB
Python
42 lines
1.4 KiB
Python
"""Runtime path resolution for standalone harness usage."""
|
|
|
|
import os
|
|
from pathlib import Path
|
|
|
|
|
|
def project_root() -> Path:
|
|
"""Return the caller project root for runtime-owned files."""
|
|
if env_root := os.getenv("DEER_FLOW_PROJECT_ROOT"):
|
|
root = Path(env_root).resolve()
|
|
if not root.exists():
|
|
raise ValueError(f"DEER_FLOW_PROJECT_ROOT is set to '{env_root}', but the resolved path '{root}' does not exist.")
|
|
if not root.is_dir():
|
|
raise ValueError(f"DEER_FLOW_PROJECT_ROOT is set to '{env_root}', but the resolved path '{root}' is not a directory.")
|
|
return root
|
|
return Path.cwd().resolve()
|
|
|
|
|
|
def runtime_home() -> Path:
|
|
"""Return the writable DeerFlow state directory."""
|
|
if env_home := os.getenv("DEER_FLOW_HOME"):
|
|
return Path(env_home).resolve()
|
|
return project_root() / ".deer-flow"
|
|
|
|
|
|
def resolve_path(value: str | os.PathLike[str], *, base: Path | None = None) -> Path:
|
|
"""Resolve absolute paths as-is and relative paths against the project root."""
|
|
path = Path(value)
|
|
if not path.is_absolute():
|
|
path = (base or project_root()) / path
|
|
return path.resolve()
|
|
|
|
|
|
def existing_project_file(names: tuple[str, ...]) -> Path | None:
|
|
"""Return the first existing named file under the project root."""
|
|
root = project_root()
|
|
for name in names:
|
|
candidate = root / name
|
|
if candidate.is_file():
|
|
return candidate
|
|
return None
|