Merge branch 'fix/chore-fix' into dev/plugin-deploy
commit
ff81d12426
@ -0,0 +1,42 @@
|
|||||||
|
from typing import Optional
|
||||||
|
|
||||||
|
from pydantic import BaseModel, ConfigDict, Field, ValidationInfo, field_validator
|
||||||
|
|
||||||
|
from core.tools.entities.common_entities import I18nObject
|
||||||
|
from core.tools.entities.tool_entities import ToolIdentity, ToolParameter, ToolProviderIdentity
|
||||||
|
|
||||||
|
|
||||||
|
class AgentProviderIdentity(ToolProviderIdentity):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
class AgentParameter(ToolParameter):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
class AgentProviderEntity(BaseModel):
|
||||||
|
identity: AgentProviderIdentity
|
||||||
|
plugin_id: Optional[str] = Field(None, description="The id of the plugin")
|
||||||
|
|
||||||
|
|
||||||
|
class AgentIdentity(ToolIdentity):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
class AgentStrategyEntity(BaseModel):
|
||||||
|
identity: AgentIdentity
|
||||||
|
parameters: list[AgentParameter] = Field(default_factory=list)
|
||||||
|
description: I18nObject = Field(..., description="The description of the agent strategy")
|
||||||
|
output_schema: Optional[dict] = None
|
||||||
|
|
||||||
|
# pydantic configs
|
||||||
|
model_config = ConfigDict(protected_namespaces=())
|
||||||
|
|
||||||
|
@field_validator("parameters", mode="before")
|
||||||
|
@classmethod
|
||||||
|
def set_parameters(cls, v, validation_info: ValidationInfo) -> list[AgentParameter]:
|
||||||
|
return v or []
|
||||||
|
|
||||||
|
|
||||||
|
class AgentProviderEntityWithPlugin(AgentProviderEntity):
|
||||||
|
strategies: list[AgentStrategyEntity] = Field(default_factory=list)
|
||||||
@ -0,0 +1,41 @@
|
|||||||
|
from abc import ABC, abstractmethod
|
||||||
|
from typing import Any, Generator, Optional, Sequence
|
||||||
|
|
||||||
|
from core.agent.entities import AgentInvokeMessage
|
||||||
|
from core.agent.plugin_entities import AgentParameter
|
||||||
|
|
||||||
|
|
||||||
|
class BaseAgentStrategy(ABC):
|
||||||
|
"""
|
||||||
|
Agent Strategy
|
||||||
|
"""
|
||||||
|
|
||||||
|
def invoke(
|
||||||
|
self,
|
||||||
|
params: dict[str, Any],
|
||||||
|
user_id: str,
|
||||||
|
conversation_id: Optional[str] = None,
|
||||||
|
app_id: Optional[str] = None,
|
||||||
|
message_id: Optional[str] = None,
|
||||||
|
) -> Generator[AgentInvokeMessage, None, None]:
|
||||||
|
"""
|
||||||
|
Invoke the agent strategy.
|
||||||
|
"""
|
||||||
|
yield from self._invoke(params, user_id, conversation_id, app_id, message_id)
|
||||||
|
|
||||||
|
def get_parameters(self) -> Sequence[AgentParameter]:
|
||||||
|
"""
|
||||||
|
Get the parameters for the agent strategy.
|
||||||
|
"""
|
||||||
|
return []
|
||||||
|
|
||||||
|
@abstractmethod
|
||||||
|
def _invoke(
|
||||||
|
self,
|
||||||
|
params: dict[str, Any],
|
||||||
|
user_id: str,
|
||||||
|
conversation_id: Optional[str] = None,
|
||||||
|
app_id: Optional[str] = None,
|
||||||
|
message_id: Optional[str] = None,
|
||||||
|
) -> Generator[AgentInvokeMessage, None, None]:
|
||||||
|
pass
|
||||||
@ -0,0 +1,52 @@
|
|||||||
|
from typing import Any, Generator, Optional, Sequence
|
||||||
|
|
||||||
|
from core.agent.entities import AgentInvokeMessage
|
||||||
|
from core.agent.plugin_entities import AgentParameter, AgentStrategyEntity
|
||||||
|
from core.agent.strategy.base import BaseAgentStrategy
|
||||||
|
from core.plugin.manager.agent import PluginAgentManager
|
||||||
|
from core.tools.plugin_tool.tool import PluginTool
|
||||||
|
|
||||||
|
|
||||||
|
class PluginAgentStrategy(BaseAgentStrategy):
|
||||||
|
"""
|
||||||
|
Agent Strategy
|
||||||
|
"""
|
||||||
|
|
||||||
|
tenant_id: str
|
||||||
|
plugin_unique_identifier: str
|
||||||
|
declaration: AgentStrategyEntity
|
||||||
|
|
||||||
|
def __init__(self, tenant_id: str, plugin_unique_identifier: str, declaration: AgentStrategyEntity):
|
||||||
|
self.tenant_id = tenant_id
|
||||||
|
self.plugin_unique_identifier = plugin_unique_identifier
|
||||||
|
self.declaration = declaration
|
||||||
|
|
||||||
|
def get_parameters(self) -> Sequence[AgentParameter]:
|
||||||
|
return self.declaration.parameters
|
||||||
|
|
||||||
|
def _invoke(
|
||||||
|
self,
|
||||||
|
params: dict[str, Any],
|
||||||
|
user_id: str,
|
||||||
|
conversation_id: Optional[str] = None,
|
||||||
|
app_id: Optional[str] = None,
|
||||||
|
message_id: Optional[str] = None,
|
||||||
|
) -> Generator[AgentInvokeMessage, None, None]:
|
||||||
|
"""
|
||||||
|
Invoke the agent strategy.
|
||||||
|
"""
|
||||||
|
manager = PluginAgentManager()
|
||||||
|
|
||||||
|
# convert agent parameters with File type to PluginFileEntity
|
||||||
|
params = PluginTool._transform_image_parameters(params)
|
||||||
|
|
||||||
|
yield from manager.invoke(
|
||||||
|
tenant_id=self.tenant_id,
|
||||||
|
user_id=user_id,
|
||||||
|
agent_provider=self.declaration.identity.provider,
|
||||||
|
agent_strategy=self.declaration.identity.name,
|
||||||
|
agent_params=params,
|
||||||
|
conversation_id=conversation_id,
|
||||||
|
app_id=app_id,
|
||||||
|
message_id=message_id,
|
||||||
|
)
|
||||||
@ -0,0 +1,109 @@
|
|||||||
|
from collections.abc import Generator
|
||||||
|
from typing import Any, Optional
|
||||||
|
|
||||||
|
from core.agent.entities import AgentInvokeMessage
|
||||||
|
from core.plugin.entities.plugin import GenericProviderID
|
||||||
|
from core.plugin.entities.plugin_daemon import (
|
||||||
|
PluginAgentProviderEntity,
|
||||||
|
)
|
||||||
|
from core.plugin.manager.base import BasePluginManager
|
||||||
|
|
||||||
|
|
||||||
|
class PluginAgentManager(BasePluginManager):
|
||||||
|
def fetch_agent_providers(self, tenant_id: str) -> list[PluginAgentProviderEntity]:
|
||||||
|
"""
|
||||||
|
Fetch agent providers for the given tenant.
|
||||||
|
"""
|
||||||
|
|
||||||
|
def transformer(json_response: dict[str, Any]) -> dict:
|
||||||
|
for provider in json_response.get("data", []):
|
||||||
|
declaration = provider.get("declaration", {}) or {}
|
||||||
|
provider_name = declaration.get("identity", {}).get("name")
|
||||||
|
for tool in declaration.get("tools", []):
|
||||||
|
tool["identity"]["provider"] = provider_name
|
||||||
|
|
||||||
|
return json_response
|
||||||
|
|
||||||
|
response = self._request_with_plugin_daemon_response(
|
||||||
|
"GET",
|
||||||
|
f"plugin/{tenant_id}/management/agents",
|
||||||
|
list[PluginAgentProviderEntity],
|
||||||
|
params={"page": 1, "page_size": 256},
|
||||||
|
transformer=transformer,
|
||||||
|
)
|
||||||
|
|
||||||
|
for provider in response:
|
||||||
|
provider.declaration.identity.name = f"{provider.plugin_id}/{provider.declaration.identity.name}"
|
||||||
|
|
||||||
|
# override the provider name for each tool to plugin_id/provider_name
|
||||||
|
for strategy in provider.declaration.strategies:
|
||||||
|
strategy.identity.provider = provider.declaration.identity.name
|
||||||
|
|
||||||
|
return response
|
||||||
|
|
||||||
|
def fetch_agent_provider(self, tenant_id: str, provider: str) -> PluginAgentProviderEntity:
|
||||||
|
"""
|
||||||
|
Fetch tool provider for the given tenant and plugin.
|
||||||
|
"""
|
||||||
|
agent_provider_id = GenericProviderID(provider)
|
||||||
|
|
||||||
|
def transformer(json_response: dict[str, Any]) -> dict:
|
||||||
|
for strategy in json_response.get("data", {}).get("declaration", {}).get("strategies", []):
|
||||||
|
strategy["identity"]["provider"] = agent_provider_id.provider_name
|
||||||
|
|
||||||
|
return json_response
|
||||||
|
|
||||||
|
response = self._request_with_plugin_daemon_response(
|
||||||
|
"GET",
|
||||||
|
f"plugin/{tenant_id}/management/agent",
|
||||||
|
PluginAgentProviderEntity,
|
||||||
|
params={"provider": agent_provider_id.provider_name, "plugin_id": agent_provider_id.plugin_id},
|
||||||
|
transformer=transformer,
|
||||||
|
)
|
||||||
|
|
||||||
|
response.declaration.identity.name = f"{response.plugin_id}/{response.declaration.identity.name}"
|
||||||
|
|
||||||
|
# override the provider name for each tool to plugin_id/provider_name
|
||||||
|
for strategy in response.declaration.strategies:
|
||||||
|
strategy.identity.provider = response.declaration.identity.name
|
||||||
|
|
||||||
|
return response
|
||||||
|
|
||||||
|
def invoke(
|
||||||
|
self,
|
||||||
|
tenant_id: str,
|
||||||
|
user_id: str,
|
||||||
|
agent_provider: str,
|
||||||
|
agent_strategy: str,
|
||||||
|
agent_params: dict[str, Any],
|
||||||
|
conversation_id: Optional[str] = None,
|
||||||
|
app_id: Optional[str] = None,
|
||||||
|
message_id: Optional[str] = None,
|
||||||
|
) -> Generator[AgentInvokeMessage, None, None]:
|
||||||
|
"""
|
||||||
|
Invoke the agent with the given tenant, user, plugin, provider, name and parameters.
|
||||||
|
"""
|
||||||
|
|
||||||
|
agent_provider_id = GenericProviderID(agent_provider)
|
||||||
|
|
||||||
|
response = self._request_with_plugin_daemon_response_stream(
|
||||||
|
"POST",
|
||||||
|
f"plugin/{tenant_id}/dispatch/agent/invoke",
|
||||||
|
AgentInvokeMessage,
|
||||||
|
data={
|
||||||
|
"user_id": user_id,
|
||||||
|
"conversation_id": conversation_id,
|
||||||
|
"app_id": app_id,
|
||||||
|
"message_id": message_id,
|
||||||
|
"data": {
|
||||||
|
"provider": agent_provider_id.provider_name,
|
||||||
|
"strategy": agent_strategy,
|
||||||
|
"agent_params": agent_params,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
headers={
|
||||||
|
"X-Plugin-ID": agent_provider_id.plugin_id,
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
},
|
||||||
|
)
|
||||||
|
return response
|
||||||
@ -0,0 +1,3 @@
|
|||||||
|
from .agent_node import AgentNode
|
||||||
|
|
||||||
|
__all__ = ["AgentNode"]
|
||||||
@ -0,0 +1,86 @@
|
|||||||
|
from collections.abc import Generator
|
||||||
|
from typing import cast
|
||||||
|
|
||||||
|
from core.plugin.manager.exc import PluginDaemonClientSideError
|
||||||
|
from core.workflow.entities.node_entities import NodeRunResult
|
||||||
|
from core.workflow.nodes.agent.entities import AgentNodeData
|
||||||
|
from core.workflow.nodes.enums import NodeType
|
||||||
|
from core.workflow.nodes.event.event import RunCompletedEvent
|
||||||
|
from core.workflow.nodes.tool.tool_node import ToolNode
|
||||||
|
from factories.agent_factory import get_plugin_agent_strategy
|
||||||
|
from models.workflow import WorkflowNodeExecutionStatus
|
||||||
|
|
||||||
|
|
||||||
|
class AgentNode(ToolNode):
|
||||||
|
"""
|
||||||
|
Agent Node
|
||||||
|
"""
|
||||||
|
|
||||||
|
_node_data_cls = AgentNodeData
|
||||||
|
_node_type = NodeType.AGENT
|
||||||
|
|
||||||
|
def _run(self) -> Generator:
|
||||||
|
"""
|
||||||
|
Run the agent node
|
||||||
|
"""
|
||||||
|
node_data = cast(AgentNodeData, self.node_data)
|
||||||
|
|
||||||
|
try:
|
||||||
|
strategy = get_plugin_agent_strategy(
|
||||||
|
tenant_id=self.tenant_id,
|
||||||
|
plugin_unique_identifier=node_data.plugin_unique_identifier,
|
||||||
|
agent_strategy_provider_name=node_data.agent_strategy_provider_name,
|
||||||
|
agent_strategy_name=node_data.agent_strategy_name,
|
||||||
|
)
|
||||||
|
except Exception as e:
|
||||||
|
yield RunCompletedEvent(
|
||||||
|
run_result=NodeRunResult(
|
||||||
|
status=WorkflowNodeExecutionStatus.FAILED,
|
||||||
|
inputs={},
|
||||||
|
error=f"Failed to get agent strategy: {str(e)}",
|
||||||
|
)
|
||||||
|
)
|
||||||
|
return
|
||||||
|
|
||||||
|
agent_parameters = strategy.get_parameters()
|
||||||
|
|
||||||
|
# get parameters
|
||||||
|
parameters = self._generate_parameters(
|
||||||
|
tool_parameters=agent_parameters,
|
||||||
|
variable_pool=self.graph_runtime_state.variable_pool,
|
||||||
|
node_data=self.node_data,
|
||||||
|
)
|
||||||
|
parameters_for_log = self._generate_parameters(
|
||||||
|
tool_parameters=agent_parameters,
|
||||||
|
variable_pool=self.graph_runtime_state.variable_pool,
|
||||||
|
node_data=self.node_data,
|
||||||
|
for_log=True,
|
||||||
|
)
|
||||||
|
|
||||||
|
try:
|
||||||
|
message_stream = strategy.invoke(
|
||||||
|
params=parameters,
|
||||||
|
user_id=self.user_id,
|
||||||
|
app_id=self.app_id,
|
||||||
|
# TODO: conversation id and message id
|
||||||
|
)
|
||||||
|
except Exception as e:
|
||||||
|
yield RunCompletedEvent(
|
||||||
|
run_result=NodeRunResult(
|
||||||
|
status=WorkflowNodeExecutionStatus.FAILED,
|
||||||
|
inputs=parameters_for_log,
|
||||||
|
error=f"Failed to invoke agent: {str(e)}",
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
try:
|
||||||
|
# convert tool messages
|
||||||
|
yield from self._transform_message(message_stream, {}, parameters_for_log)
|
||||||
|
except PluginDaemonClientSideError as e:
|
||||||
|
yield RunCompletedEvent(
|
||||||
|
run_result=NodeRunResult(
|
||||||
|
status=WorkflowNodeExecutionStatus.FAILED,
|
||||||
|
inputs=parameters_for_log,
|
||||||
|
error=f"Failed to transform agent message: {str(e)}",
|
||||||
|
)
|
||||||
|
)
|
||||||
@ -0,0 +1,52 @@
|
|||||||
|
from typing import Any, Literal, Union
|
||||||
|
|
||||||
|
from pydantic import BaseModel, ValidationInfo, field_validator
|
||||||
|
|
||||||
|
from core.workflow.nodes.base.entities import BaseNodeData
|
||||||
|
|
||||||
|
|
||||||
|
class AgentEntity(BaseModel):
|
||||||
|
agent_strategy_provider_name: str # redundancy
|
||||||
|
agent_strategy_name: str
|
||||||
|
agent_strategy_label: str # redundancy
|
||||||
|
agent_configurations: dict[str, Any]
|
||||||
|
plugin_unique_identifier: str
|
||||||
|
|
||||||
|
@field_validator("agent_configurations", mode="before")
|
||||||
|
@classmethod
|
||||||
|
def validate_agent_configurations(cls, value, values: ValidationInfo):
|
||||||
|
if not isinstance(value, dict):
|
||||||
|
raise ValueError("agent_configurations must be a dictionary")
|
||||||
|
|
||||||
|
for key in values.data.get("agent_configurations", {}):
|
||||||
|
value = values.data.get("agent_configurations", {}).get(key)
|
||||||
|
if not isinstance(value, str | int | float | bool):
|
||||||
|
raise ValueError(f"{key} must be a string")
|
||||||
|
|
||||||
|
return value
|
||||||
|
|
||||||
|
|
||||||
|
class AgentNodeData(BaseNodeData, AgentEntity):
|
||||||
|
class AgentInput(BaseModel):
|
||||||
|
# TODO: check this type
|
||||||
|
value: Union[Any, list[str]]
|
||||||
|
type: Literal["mixed", "variable", "constant"]
|
||||||
|
|
||||||
|
@field_validator("type", mode="before")
|
||||||
|
@classmethod
|
||||||
|
def check_type(cls, value, validation_info: ValidationInfo):
|
||||||
|
typ = value
|
||||||
|
value = validation_info.data.get("value")
|
||||||
|
if typ == "mixed" and not isinstance(value, str):
|
||||||
|
raise ValueError("value must be a string")
|
||||||
|
elif typ == "variable":
|
||||||
|
if not isinstance(value, list):
|
||||||
|
raise ValueError("value must be a list")
|
||||||
|
for val in value:
|
||||||
|
if not isinstance(val, str):
|
||||||
|
raise ValueError("value must be a list of strings")
|
||||||
|
elif typ == "constant" and not isinstance(value, str | int | float | bool):
|
||||||
|
raise ValueError("value must be a string, int, float, or bool")
|
||||||
|
return typ
|
||||||
|
|
||||||
|
agent_parameters: dict[str, AgentInput]
|
||||||
@ -0,0 +1,15 @@
|
|||||||
|
from core.agent.strategy.plugin import PluginAgentStrategy
|
||||||
|
from core.plugin.manager.agent import PluginAgentManager
|
||||||
|
|
||||||
|
|
||||||
|
def get_plugin_agent_strategy(
|
||||||
|
tenant_id: str, plugin_unique_identifier: str, agent_strategy_provider_name: str, agent_strategy_name: str
|
||||||
|
) -> PluginAgentStrategy:
|
||||||
|
# TODO: use contexts to cache the agent provider
|
||||||
|
manager = PluginAgentManager()
|
||||||
|
agent_provider = manager.fetch_agent_provider(tenant_id, agent_strategy_provider_name)
|
||||||
|
for agent_strategy in agent_provider.declaration.strategies:
|
||||||
|
if agent_strategy.identity.name == agent_strategy_name:
|
||||||
|
return PluginAgentStrategy(tenant_id, plugin_unique_identifier, agent_strategy)
|
||||||
|
|
||||||
|
raise ValueError(f"Agent strategy {agent_strategy_name} not found")
|
||||||
Loading…
Reference in New Issue