2026-01-14 07:16:07 +08:00
|
|
|
from importlib import import_module
|
|
|
|
|
|
2026-03-03 14:56:54 +08:00
|
|
|
MODULE_TO_PACKAGE_HINTS = {
|
|
|
|
|
"langchain_google_genai": "langchain-google-genai",
|
|
|
|
|
"langchain_anthropic": "langchain-anthropic",
|
|
|
|
|
"langchain_openai": "langchain-openai",
|
|
|
|
|
"langchain_deepseek": "langchain-deepseek",
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def _build_missing_dependency_hint(module_path: str, err: ImportError) -> str:
|
|
|
|
|
"""Build an actionable hint when module import fails."""
|
|
|
|
|
module_root = module_path.split(".", 1)[0]
|
|
|
|
|
missing_module = getattr(err, "name", None) or module_root
|
|
|
|
|
|
|
|
|
|
# Prefer provider package hints for known integrations, even when the import
|
|
|
|
|
# error is triggered by a transitive dependency (e.g. `google`).
|
|
|
|
|
package_name = MODULE_TO_PACKAGE_HINTS.get(module_root)
|
|
|
|
|
if package_name is None:
|
|
|
|
|
package_name = MODULE_TO_PACKAGE_HINTS.get(missing_module, missing_module.replace("_", "-"))
|
|
|
|
|
|
feat: add IM channels for Feishu, Slack, and Telegram (#1010)
* feat: add IM channels system for Feishu, Slack, and Telegram integration
Bridge external messaging platforms to DeerFlow via LangGraph Server with
async message bus, thread management, and per-channel configuration.
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
* fix: address review comments on IM channels system
Fix topic_id handling in store remove/list_entries and manager commands,
correct Telegram reply threading, remove unused imports/variables, update
docstrings and docs to match implementation, and prevent config mutation.
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
* update skill creator
* fix im reply text
* fix comments
---------
Co-authored-by: Claude Opus 4.6 <noreply@anthropic.com>
2026-03-08 15:21:18 +08:00
|
|
|
return f"Missing dependency '{missing_module}'. Install it with `uv add {package_name}` (or `pip install {package_name}`), then restart DeerFlow."
|
2026-03-03 14:56:54 +08:00
|
|
|
|
2026-01-14 07:16:07 +08:00
|
|
|
|
|
|
|
|
def resolve_variable[T](
|
|
|
|
|
variable_path: str,
|
2026-01-14 09:08:20 +08:00
|
|
|
expected_type: type[T] | tuple[type, ...] | None = None,
|
2026-01-14 07:16:07 +08:00
|
|
|
) -> T:
|
|
|
|
|
"""Resolve a variable from a path.
|
|
|
|
|
|
|
|
|
|
Args:
|
|
|
|
|
variable_path: The path to the variable (e.g. "parent_package_name.sub_package_name.module_name:variable_name").
|
|
|
|
|
expected_type: Optional type or tuple of types to validate the resolved variable against.
|
|
|
|
|
If provided, uses isinstance() to check if the variable is an instance of the expected type(s).
|
|
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
|
The resolved variable.
|
|
|
|
|
|
|
|
|
|
Raises:
|
|
|
|
|
ImportError: If the module path is invalid or the attribute doesn't exist.
|
|
|
|
|
ValueError: If the resolved variable doesn't pass the validation checks.
|
|
|
|
|
"""
|
|
|
|
|
try:
|
|
|
|
|
module_path, variable_name = variable_path.rsplit(":", 1)
|
|
|
|
|
except ValueError as err:
|
2026-01-14 09:08:20 +08:00
|
|
|
raise ImportError(f"{variable_path} doesn't look like a variable path. Example: parent_package_name.sub_package_name.module_name:variable_name") from err
|
2026-01-14 07:16:07 +08:00
|
|
|
|
|
|
|
|
try:
|
|
|
|
|
module = import_module(module_path)
|
|
|
|
|
except ImportError as err:
|
2026-03-03 14:56:54 +08:00
|
|
|
module_root = module_path.split(".", 1)[0]
|
|
|
|
|
err_name = getattr(err, "name", None)
|
|
|
|
|
if isinstance(err, ModuleNotFoundError) or err_name == module_root:
|
|
|
|
|
hint = _build_missing_dependency_hint(module_path, err)
|
|
|
|
|
raise ImportError(f"Could not import module {module_path}. {hint}") from err
|
|
|
|
|
# Preserve the original ImportError message for non-missing-module failures.
|
|
|
|
|
raise ImportError(f"Error importing module {module_path}: {err}") from err
|
2026-01-14 07:16:07 +08:00
|
|
|
|
|
|
|
|
try:
|
|
|
|
|
variable = getattr(module, variable_name)
|
|
|
|
|
except AttributeError as err:
|
2026-01-14 09:08:20 +08:00
|
|
|
raise ImportError(f"Module {module_path} does not define a {variable_name} attribute/class") from err
|
2026-01-14 07:16:07 +08:00
|
|
|
|
|
|
|
|
# Type validation
|
|
|
|
|
if expected_type is not None:
|
|
|
|
|
if not isinstance(variable, expected_type):
|
2026-01-14 09:08:20 +08:00
|
|
|
type_name = expected_type.__name__ if isinstance(expected_type, type) else " or ".join(t.__name__ for t in expected_type)
|
|
|
|
|
raise ValueError(f"{variable_path} is not an instance of {type_name}, got {type(variable).__name__}")
|
2026-01-14 07:16:07 +08:00
|
|
|
|
|
|
|
|
return variable
|
|
|
|
|
|
|
|
|
|
|
2026-01-14 09:08:20 +08:00
|
|
|
def resolve_class[T](class_path: str, base_class: type[T] | None = None) -> type[T]:
|
2026-01-14 07:16:07 +08:00
|
|
|
"""Resolve a class from a module path and class name.
|
|
|
|
|
|
|
|
|
|
Args:
|
|
|
|
|
class_path: The path to the class (e.g. "langchain_openai:ChatOpenAI").
|
|
|
|
|
base_class: The base class to check if the resolved class is a subclass of.
|
|
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
|
The resolved class.
|
|
|
|
|
|
|
|
|
|
Raises:
|
|
|
|
|
ImportError: If the module path is invalid or the attribute doesn't exist.
|
|
|
|
|
ValueError: If the resolved object is not a class or not a subclass of base_class.
|
|
|
|
|
"""
|
|
|
|
|
model_class = resolve_variable(class_path, expected_type=type)
|
|
|
|
|
|
|
|
|
|
if not isinstance(model_class, type):
|
|
|
|
|
raise ValueError(f"{class_path} is not a valid class")
|
|
|
|
|
|
|
|
|
|
if base_class is not None and not issubclass(model_class, base_class):
|
|
|
|
|
raise ValueError(f"{class_path} is not a subclass of {base_class.__name__}")
|
|
|
|
|
|
|
|
|
|
return model_class
|