- Augment log search with app type filtering to enable keyword searching within workflow_executions. - Introduce execution sequence markers to ensure logs are displayed in the correct chronological order. - Ameliorate error handling to capture successful node outputs alongside failure details. - Rectify the processing of empty JSON bodies in HTTP request nodes.
479 lines
16 KiB
Python
479 lines
16 KiB
Python
import uuid
|
||
from typing import Optional
|
||
|
||
from sqlalchemy import select, desc, func, or_, cast, Text
|
||
from sqlalchemy.orm import Session
|
||
|
||
from app.core.exceptions import ResourceNotFoundException
|
||
from app.core.logging_config import get_db_logger
|
||
from app.models import Conversation, Message
|
||
from app.models.app_model import AppType
|
||
from app.models.conversation_model import ConversationDetail
|
||
from app.models.workflow_model import WorkflowExecution
|
||
|
||
logger = get_db_logger()
|
||
|
||
|
||
class ConversationRepository:
|
||
"""Repository for Conversation entity, encapsulating CRUD operations."""
|
||
|
||
def __init__(self, db: Session):
|
||
self.db = db
|
||
|
||
def create_conversation(
|
||
self,
|
||
app_id: uuid.UUID,
|
||
workspace_id: uuid.UUID,
|
||
user_id: Optional[str] = None,
|
||
title: Optional[str] = None,
|
||
is_draft: bool = False,
|
||
config_snapshot: Optional[dict] = None
|
||
) -> Conversation:
|
||
"""
|
||
Create a new conversation record.
|
||
|
||
Args:
|
||
app_id: Application ID the conversation belongs to.
|
||
workspace_id: Workspace ID where the conversation is created.
|
||
user_id: Optional user ID associated with the conversation.
|
||
title: Optional conversation title. Defaults to "New Conversation".
|
||
is_draft: Whether the conversation is a draft.
|
||
config_snapshot: Optional configuration snapshot.
|
||
|
||
Returns:
|
||
Conversation: Newly created Conversation instance.
|
||
"""
|
||
conversation = Conversation(
|
||
app_id=app_id,
|
||
workspace_id=workspace_id,
|
||
user_id=user_id,
|
||
title=title or "New Conversation",
|
||
is_draft=is_draft,
|
||
config_snapshot=config_snapshot
|
||
)
|
||
self.db.add(conversation)
|
||
return conversation
|
||
|
||
def get_conversation_by_conversation_id(
|
||
self,
|
||
conversation_id: uuid.UUID,
|
||
workspace_id: Optional[uuid.UUID] = None
|
||
) -> Conversation:
|
||
"""
|
||
Retrieve a conversation by its ID, optionally filtered by workspace.
|
||
|
||
Args:
|
||
conversation_id: The UUID of the conversation.
|
||
workspace_id: Optional workspace UUID to filter the conversation.
|
||
|
||
Raises:
|
||
ResourceNotFoundException: If conversation does not exist.
|
||
|
||
Returns:
|
||
Conversation: The matching Conversation instance.
|
||
"""
|
||
logger.info(f"Fetching conversation: {conversation_id}")
|
||
|
||
stmt = select(Conversation).where(Conversation.id == conversation_id)
|
||
|
||
if workspace_id:
|
||
stmt = stmt.where(Conversation.workspace_id == workspace_id)
|
||
|
||
conversation = self.db.scalars(stmt).first()
|
||
|
||
if not conversation:
|
||
logger.warning(f"Conversation not found: {conversation_id}")
|
||
raise ResourceNotFoundException("Conversation", str(conversation_id))
|
||
|
||
logger.info(f"Conversation fetched successfully: {conversation_id}")
|
||
return conversation
|
||
|
||
def get_conversation_by_user_id(
|
||
self,
|
||
user_id: uuid.UUID,
|
||
workspace_id: uuid.UUID = None,
|
||
is_activate: bool = True,
|
||
page: int = 1,
|
||
page_size: int = 20
|
||
) -> tuple[list[Conversation], int]:
|
||
"""
|
||
Retrieve recent conversations for a specific user with pagination.
|
||
|
||
This method queries conversations associated with the given user ID,
|
||
optionally scoped to a specific workspace. Results are ordered by the
|
||
most recently updated conversations.
|
||
|
||
Args:
|
||
user_id (uuid.UUID): Unique identifier of the user.
|
||
workspace_id (uuid.UUID, optional): Workspace scope for the query.
|
||
If provided, only conversations under this workspace will be returned.
|
||
is_activate (bool): Conversation State limit.
|
||
page (int): Page number (1-based). Defaults to 1.
|
||
page_size (int): Number of items per page. Defaults to 20.
|
||
|
||
Returns:
|
||
tuple[list[Conversation], int]: A list of conversation entities and total count.
|
||
"""
|
||
logger.info(f"Fetching conversation by user_id: {user_id}")
|
||
|
||
stmt = select(Conversation).where(
|
||
Conversation.user_id == str(user_id),
|
||
Conversation.is_active.is_(is_activate)
|
||
)
|
||
|
||
if workspace_id:
|
||
stmt = stmt.where(Conversation.workspace_id == workspace_id)
|
||
|
||
# Calculate total count
|
||
total = int(self.db.execute(
|
||
select(func.count()).select_from(stmt.subquery())
|
||
).scalar_one())
|
||
|
||
# Apply ordering and pagination
|
||
stmt = stmt.order_by(desc(Conversation.updated_at))
|
||
stmt = stmt.offset((page - 1) * page_size).limit(page_size)
|
||
|
||
conversations = list(self.db.scalars(stmt).all())
|
||
logger.info(
|
||
"Conversation fetched successfully",
|
||
extra={
|
||
"user_id": str(user_id),
|
||
"workspace_id": str(workspace_id),
|
||
"total": total,
|
||
}
|
||
)
|
||
return conversations, total
|
||
|
||
def list_conversations(
|
||
self,
|
||
app_id: uuid.UUID,
|
||
workspace_id: uuid.UUID,
|
||
user_id: Optional[str] = None,
|
||
is_draft: Optional[bool] = None,
|
||
page: int = 1,
|
||
pagesize: int = 20
|
||
) -> tuple[list[Conversation], int]:
|
||
"""
|
||
List conversations with optional filters and pagination.
|
||
|
||
Args:
|
||
app_id: Application ID filter.
|
||
workspace_id: Workspace ID filter.
|
||
user_id: Optional user ID filter.
|
||
is_draft: Optional draft status filter.
|
||
page: Page number (1-based).
|
||
pagesize: Number of items per page.
|
||
|
||
Returns:
|
||
Tuple[List[Conversation], int]: List of Conversation instances and total count.
|
||
"""
|
||
stmt = select(Conversation).where(
|
||
Conversation.app_id == app_id,
|
||
Conversation.workspace_id == workspace_id,
|
||
Conversation.is_active.is_(True)
|
||
)
|
||
|
||
if user_id:
|
||
stmt = stmt.where(Conversation.user_id == str(user_id))
|
||
|
||
if is_draft is not None:
|
||
stmt = stmt.where(Conversation.is_draft == is_draft)
|
||
|
||
# Calculate total number of records
|
||
total = int(self.db.execute(
|
||
select(func.count()).select_from(stmt.subquery())
|
||
).scalar_one())
|
||
|
||
# Apply pagination
|
||
stmt = stmt.order_by(desc(Conversation.updated_at))
|
||
stmt = stmt.offset((page - 1) * pagesize).limit(pagesize)
|
||
|
||
conversations = list(self.db.scalars(stmt).all())
|
||
|
||
logger.info(
|
||
"Listed conversations successfully",
|
||
extra={
|
||
"app_id": str(app_id),
|
||
"workspace_id": str(workspace_id),
|
||
"returned": len(conversations),
|
||
"total": total
|
||
}
|
||
)
|
||
return conversations, total
|
||
|
||
def list_app_conversations(
|
||
self,
|
||
app_id: uuid.UUID,
|
||
workspace_id: uuid.UUID,
|
||
is_draft: Optional[bool] = None,
|
||
keyword: Optional[str] = None,
|
||
page: int = 1,
|
||
pagesize: int = 20,
|
||
app_type: Optional[str] = None,
|
||
) -> tuple[list[Conversation], int]:
|
||
"""
|
||
查询应用日志会话列表(带分页和过滤)
|
||
|
||
Args:
|
||
app_id: 应用 ID
|
||
workspace_id: 工作空间 ID
|
||
is_draft: 是否草稿会话(None表示返回全部)
|
||
keyword: 搜索关键词(匹配消息内容)
|
||
page: 页码(从 1 开始)
|
||
pagesize: 每页数量
|
||
app_type: 应用类型。WORKFLOW 类型改用 workflow_executions 的
|
||
input_data/output_data 做关键词过滤(因为失败的工作流不会写入 messages 表);
|
||
其他类型仍走 messages 表。
|
||
|
||
Returns:
|
||
Tuple[List[Conversation], int]: (会话列表,总数)
|
||
"""
|
||
base_conditions = [
|
||
Conversation.app_id == app_id,
|
||
Conversation.workspace_id == workspace_id,
|
||
Conversation.is_active.is_(True),
|
||
]
|
||
if is_draft is not None:
|
||
base_conditions.append(Conversation.is_draft == is_draft)
|
||
|
||
base_stmt = select(Conversation).where(*base_conditions)
|
||
|
||
# 如果有关键词搜索,通过子查询过滤包含该关键词的 conversation
|
||
if keyword:
|
||
kw_pattern = f"%{keyword}%"
|
||
if app_type == AppType.WORKFLOW:
|
||
# 工作流:从 workflow_executions 的 input_data / output_data 匹配
|
||
# (messages 表只存开场白 assistant 消息,失败的工作流也不会写入)
|
||
keyword_stmt = (
|
||
select(WorkflowExecution.conversation_id)
|
||
.where(
|
||
WorkflowExecution.conversation_id.is_not(None),
|
||
or_(
|
||
cast(WorkflowExecution.input_data, Text).ilike(kw_pattern),
|
||
cast(WorkflowExecution.output_data, Text).ilike(kw_pattern),
|
||
),
|
||
)
|
||
.distinct()
|
||
)
|
||
else:
|
||
# Agent 等其他类型:仍走 messages 表(user + assistant 内容)
|
||
keyword_stmt = (
|
||
select(Message.conversation_id)
|
||
.where(Message.content.ilike(kw_pattern))
|
||
.distinct()
|
||
)
|
||
base_stmt = base_stmt.where(Conversation.id.in_(keyword_stmt))
|
||
|
||
# Calculate total number of records
|
||
total = int(self.db.execute(
|
||
select(func.count()).select_from(base_stmt.subquery())
|
||
).scalar_one())
|
||
|
||
# Apply pagination
|
||
stmt = base_stmt.order_by(desc(Conversation.updated_at))
|
||
stmt = stmt.offset((page - 1) * pagesize).limit(pagesize)
|
||
|
||
conversations = list(self.db.scalars(stmt).all())
|
||
|
||
logger.info(
|
||
"Listed app conversations successfully",
|
||
extra={
|
||
"app_id": str(app_id),
|
||
"workspace_id": str(workspace_id),
|
||
"keyword": keyword,
|
||
"returned": len(conversations),
|
||
"total": total
|
||
}
|
||
)
|
||
return conversations, total
|
||
|
||
def get_conversation_for_app_log(
|
||
self,
|
||
conversation_id: uuid.UUID,
|
||
app_id: uuid.UUID,
|
||
workspace_id: uuid.UUID
|
||
) -> Conversation:
|
||
"""
|
||
查询应用日志的会话详情
|
||
|
||
Args:
|
||
conversation_id: 会话 ID
|
||
app_id: 应用 ID
|
||
workspace_id: 工作空间 ID
|
||
|
||
Returns:
|
||
Conversation: 会话对象
|
||
|
||
Raises:
|
||
ResourceNotFoundException: 当会话不存在时
|
||
"""
|
||
logger.info(f"Fetching conversation for app log: {conversation_id}")
|
||
|
||
stmt = select(Conversation).where(
|
||
Conversation.id == conversation_id,
|
||
Conversation.app_id == app_id,
|
||
Conversation.workspace_id == workspace_id,
|
||
Conversation.is_active.is_(True)
|
||
)
|
||
|
||
conversation = self.db.scalars(stmt).first()
|
||
|
||
if not conversation:
|
||
logger.warning(f"Conversation not found: {conversation_id}")
|
||
raise ResourceNotFoundException("会话", str(conversation_id))
|
||
|
||
logger.info(f"Conversation fetched successfully: {conversation_id}")
|
||
return conversation
|
||
|
||
def soft_delete_conversation_by_conversation_id(
|
||
self,
|
||
conversation_id: uuid.UUID,
|
||
workspace_id: uuid.UUID,
|
||
):
|
||
"""
|
||
Soft delete a conversation by setting is_active to False.
|
||
|
||
Args:
|
||
conversation_id: The UUID of the conversation.
|
||
workspace_id: Workspace ID for verification.
|
||
"""
|
||
conversation = self.get_conversation_by_conversation_id(
|
||
conversation_id,
|
||
workspace_id
|
||
)
|
||
conversation.is_active = False
|
||
|
||
def get_conversation_detail(
|
||
self,
|
||
conversation_id: uuid.UUID
|
||
) -> ConversationDetail | None:
|
||
"""
|
||
Retrieve the detail of a conversation by its ID.
|
||
|
||
Args:
|
||
conversation_id (UUID): The unique identifier of the conversation.
|
||
|
||
Returns:
|
||
ConversationDetail or None: The conversation detail object if found,
|
||
otherwise None.
|
||
|
||
Notes:
|
||
- This method queries the database but does not modify it.
|
||
- The caller is responsible for handling the case where None is returned.
|
||
"""
|
||
stmt = select(ConversationDetail).where(
|
||
ConversationDetail.conversation_id == conversation_id
|
||
)
|
||
detail = self.db.scalars(stmt).first()
|
||
return detail
|
||
|
||
def add_conversation_detail(
|
||
self,
|
||
conversation_detail: ConversationDetail,
|
||
):
|
||
"""
|
||
Add a new conversation detail record to the database session.
|
||
|
||
Args:
|
||
conversation_detail (ConversationDetail): The ORM object representing
|
||
the conversation detail to add.
|
||
|
||
Returns:
|
||
ConversationDetail: The same object added to the session.
|
||
|
||
Notes:
|
||
- This method only adds the object to the current session.
|
||
- It does not commit the transaction; commit/rollback is handled
|
||
by the caller.
|
||
- Useful for batch operations or transactional control.
|
||
"""
|
||
self.db.add(conversation_detail)
|
||
return conversation_detail
|
||
|
||
|
||
class MessageRepository:
|
||
"""Repository for Message entity, encapsulating CRUD operations."""
|
||
|
||
def __init__(self, db: Session):
|
||
self.db = db
|
||
|
||
def add_message(self, message: Message) -> Message:
|
||
"""
|
||
Add a new message record to the conversation.
|
||
|
||
Args:
|
||
message (Message): The Message ORM object to be added.
|
||
|
||
Returns:
|
||
Message: The same message object added to the conversation.
|
||
|
||
Notes:
|
||
- This method only adds the object to the current conversation.
|
||
- It does not commit the transaction; commit/rollback should be handled
|
||
by the caller.
|
||
- Useful for transactional control or batch operations.
|
||
"""
|
||
self.db.add(message)
|
||
return message
|
||
|
||
def get_messages_by_conversation(
|
||
self,
|
||
conversation_id: uuid.UUID
|
||
) -> list[Message]:
|
||
"""
|
||
查询会话的所有消息(按时间正序)
|
||
|
||
Args:
|
||
conversation_id: 会话 ID
|
||
|
||
Returns:
|
||
List[Message]: 消息列表
|
||
"""
|
||
stmt = select(Message).where(
|
||
Message.conversation_id == conversation_id
|
||
).order_by(Message.created_at)
|
||
|
||
messages = list(self.db.scalars(stmt).all())
|
||
|
||
logger.info(
|
||
"Fetched messages for conversation",
|
||
extra={
|
||
"conversation_id": str(conversation_id),
|
||
"message_count": len(messages)
|
||
}
|
||
)
|
||
return messages
|
||
|
||
def get_message_by_conversation_id(
|
||
self,
|
||
conversation_id: uuid.UUID,
|
||
limit: Optional[int] = None
|
||
) -> list[Message]:
|
||
"""
|
||
Retrieve messages by conversation ID.
|
||
|
||
Args:
|
||
conversation_id: The UUID of the conversation.
|
||
limit: Optional limit on the number of messages returned.
|
||
|
||
Returns:
|
||
List[Message]: List of Message instances.
|
||
"""
|
||
stmt = select(Message).where(
|
||
Message.conversation_id == conversation_id
|
||
).order_by(Message.created_at)
|
||
|
||
if limit:
|
||
stmt = stmt.limit(limit)
|
||
|
||
messages = list(self.db.scalars(stmt).all())
|
||
|
||
logger.info(
|
||
"Fetched messages successfully",
|
||
extra={
|
||
"conversation_id": str(conversation_id),
|
||
"returned": len(messages)
|
||
}
|
||
)
|
||
return messages
|