mirror of
https://github.com/MODSetter/SurfSense.git
synced 2026-06-06 20:15:17 +02:00
refactor(agents): colocate onedrive connector tools into subagent slice
Repoint the dead tools/__init__ shim at the live local impls and delete the dead shared/tools/onedrive twin (subagent already ran its local copies via tools/index.py). No runtime behavior change.
This commit is contained in:
parent
1a778883b3
commit
97ec27c786
4 changed files with 2 additions and 596 deletions
|
|
@ -1,9 +1,5 @@
|
||||||
from app.agents.shared.tools.onedrive.create_file import (
|
from .create_file import create_create_onedrive_file_tool
|
||||||
create_create_onedrive_file_tool,
|
from .trash_file import create_delete_onedrive_file_tool
|
||||||
)
|
|
||||||
from app.agents.shared.tools.onedrive.trash_file import (
|
|
||||||
create_delete_onedrive_file_tool,
|
|
||||||
)
|
|
||||||
|
|
||||||
__all__ = [
|
__all__ = [
|
||||||
"create_create_onedrive_file_tool",
|
"create_create_onedrive_file_tool",
|
||||||
|
|
|
||||||
|
|
@ -1,11 +0,0 @@
|
||||||
from app.agents.shared.tools.onedrive.create_file import (
|
|
||||||
create_create_onedrive_file_tool,
|
|
||||||
)
|
|
||||||
from app.agents.shared.tools.onedrive.trash_file import (
|
|
||||||
create_delete_onedrive_file_tool,
|
|
||||||
)
|
|
||||||
|
|
||||||
__all__ = [
|
|
||||||
"create_create_onedrive_file_tool",
|
|
||||||
"create_delete_onedrive_file_tool",
|
|
||||||
]
|
|
||||||
|
|
@ -1,274 +0,0 @@
|
||||||
import logging
|
|
||||||
import os
|
|
||||||
import tempfile
|
|
||||||
from pathlib import Path
|
|
||||||
from typing import Any
|
|
||||||
|
|
||||||
from langchain_core.tools import tool
|
|
||||||
from sqlalchemy.ext.asyncio import AsyncSession
|
|
||||||
from sqlalchemy.future import select
|
|
||||||
|
|
||||||
from app.agents.shared.tools.hitl import request_approval
|
|
||||||
from app.connectors.onedrive.client import OneDriveClient
|
|
||||||
from app.db import SearchSourceConnector, SearchSourceConnectorType, async_session_maker
|
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
|
||||||
|
|
||||||
DOCX_MIME = "application/vnd.openxmlformats-officedocument.wordprocessingml.document"
|
|
||||||
|
|
||||||
|
|
||||||
def _ensure_docx_extension(name: str) -> str:
|
|
||||||
"""Strip any existing extension and append .docx."""
|
|
||||||
stem = Path(name).stem
|
|
||||||
return f"{stem}.docx"
|
|
||||||
|
|
||||||
|
|
||||||
def _markdown_to_docx(markdown_text: str) -> bytes:
|
|
||||||
"""Convert a markdown string to DOCX bytes using pypandoc."""
|
|
||||||
import pypandoc
|
|
||||||
|
|
||||||
fd, tmp_path = tempfile.mkstemp(suffix=".docx")
|
|
||||||
os.close(fd)
|
|
||||||
try:
|
|
||||||
pypandoc.convert_text(
|
|
||||||
markdown_text,
|
|
||||||
"docx",
|
|
||||||
format="gfm",
|
|
||||||
extra_args=["--standalone"],
|
|
||||||
outputfile=tmp_path,
|
|
||||||
)
|
|
||||||
with open(tmp_path, "rb") as f:
|
|
||||||
return f.read()
|
|
||||||
finally:
|
|
||||||
os.unlink(tmp_path)
|
|
||||||
|
|
||||||
|
|
||||||
def create_create_onedrive_file_tool(
|
|
||||||
db_session: AsyncSession | None = None,
|
|
||||||
search_space_id: int | None = None,
|
|
||||||
user_id: str | None = None,
|
|
||||||
):
|
|
||||||
"""
|
|
||||||
Factory function to create the create_onedrive_file tool.
|
|
||||||
|
|
||||||
The tool acquires its own short-lived ``AsyncSession`` per call via
|
|
||||||
:data:`async_session_maker` so the closure is safe to share across
|
|
||||||
HTTP requests by the compiled-agent cache. Capturing a per-request
|
|
||||||
session here would surface stale/closed sessions on cache hits.
|
|
||||||
|
|
||||||
Args:
|
|
||||||
db_session: Reserved for registry compatibility. Per-call sessions
|
|
||||||
are opened via :data:`async_session_maker` inside the tool body.
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
Configured create_onedrive_file tool
|
|
||||||
"""
|
|
||||||
del db_session # per-call session — see docstring
|
|
||||||
|
|
||||||
@tool
|
|
||||||
async def create_onedrive_file(
|
|
||||||
name: str,
|
|
||||||
content: str | None = None,
|
|
||||||
) -> dict[str, Any]:
|
|
||||||
"""Create a new Word document (.docx) in Microsoft OneDrive.
|
|
||||||
|
|
||||||
Use this tool when the user explicitly asks to create a new document
|
|
||||||
in OneDrive. The user MUST specify a topic before you call this tool.
|
|
||||||
|
|
||||||
The file is always saved as a .docx Word document. Provide content as
|
|
||||||
markdown and it will be automatically converted to a formatted Word file.
|
|
||||||
|
|
||||||
Args:
|
|
||||||
name: The document title (without extension). Extension will be set to .docx automatically.
|
|
||||||
content: Optional initial content as markdown. Will be converted to a formatted Word document.
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
Dictionary with status, file_id, name, web_url, and message.
|
|
||||||
"""
|
|
||||||
logger.info(f"create_onedrive_file called: name='{name}'")
|
|
||||||
|
|
||||||
if search_space_id is None or user_id is None:
|
|
||||||
return {
|
|
||||||
"status": "error",
|
|
||||||
"message": "OneDrive tool not properly configured.",
|
|
||||||
}
|
|
||||||
|
|
||||||
try:
|
|
||||||
async with async_session_maker() as db_session:
|
|
||||||
result = await db_session.execute(
|
|
||||||
select(SearchSourceConnector).filter(
|
|
||||||
SearchSourceConnector.search_space_id == search_space_id,
|
|
||||||
SearchSourceConnector.user_id == user_id,
|
|
||||||
SearchSourceConnector.connector_type
|
|
||||||
== SearchSourceConnectorType.ONEDRIVE_CONNECTOR,
|
|
||||||
)
|
|
||||||
)
|
|
||||||
connectors = result.scalars().all()
|
|
||||||
|
|
||||||
if not connectors:
|
|
||||||
return {
|
|
||||||
"status": "error",
|
|
||||||
"message": "No OneDrive connector found. Please connect OneDrive in your workspace settings.",
|
|
||||||
}
|
|
||||||
|
|
||||||
accounts = []
|
|
||||||
for c in connectors:
|
|
||||||
cfg = c.config or {}
|
|
||||||
accounts.append(
|
|
||||||
{
|
|
||||||
"id": c.id,
|
|
||||||
"name": c.name,
|
|
||||||
"user_email": cfg.get("user_email"),
|
|
||||||
"auth_expired": cfg.get("auth_expired", False),
|
|
||||||
}
|
|
||||||
)
|
|
||||||
|
|
||||||
if all(a.get("auth_expired") for a in accounts):
|
|
||||||
return {
|
|
||||||
"status": "auth_error",
|
|
||||||
"message": "All connected OneDrive accounts need re-authentication.",
|
|
||||||
"connector_type": "onedrive",
|
|
||||||
}
|
|
||||||
|
|
||||||
parent_folders: dict[int, list[dict[str, str]]] = {}
|
|
||||||
for acc in accounts:
|
|
||||||
cid = acc["id"]
|
|
||||||
if acc.get("auth_expired"):
|
|
||||||
parent_folders[cid] = []
|
|
||||||
continue
|
|
||||||
try:
|
|
||||||
client = OneDriveClient(session=db_session, connector_id=cid)
|
|
||||||
items, err = await client.list_children("root")
|
|
||||||
if err:
|
|
||||||
logger.warning(
|
|
||||||
"Failed to list folders for connector %s: %s", cid, err
|
|
||||||
)
|
|
||||||
parent_folders[cid] = []
|
|
||||||
else:
|
|
||||||
parent_folders[cid] = [
|
|
||||||
{"folder_id": item["id"], "name": item["name"]}
|
|
||||||
for item in items
|
|
||||||
if item.get("folder") is not None
|
|
||||||
and item.get("id")
|
|
||||||
and item.get("name")
|
|
||||||
]
|
|
||||||
except Exception:
|
|
||||||
logger.warning(
|
|
||||||
"Error fetching folders for connector %s",
|
|
||||||
cid,
|
|
||||||
exc_info=True,
|
|
||||||
)
|
|
||||||
parent_folders[cid] = []
|
|
||||||
|
|
||||||
context: dict[str, Any] = {
|
|
||||||
"accounts": accounts,
|
|
||||||
"parent_folders": parent_folders,
|
|
||||||
}
|
|
||||||
|
|
||||||
result = request_approval(
|
|
||||||
action_type="onedrive_file_creation",
|
|
||||||
tool_name="create_onedrive_file",
|
|
||||||
params={
|
|
||||||
"name": name,
|
|
||||||
"content": content,
|
|
||||||
"connector_id": None,
|
|
||||||
"parent_folder_id": None,
|
|
||||||
},
|
|
||||||
context=context,
|
|
||||||
)
|
|
||||||
|
|
||||||
if result.rejected:
|
|
||||||
return {
|
|
||||||
"status": "rejected",
|
|
||||||
"message": "User declined. Do not retry or suggest alternatives.",
|
|
||||||
}
|
|
||||||
|
|
||||||
final_name = result.params.get("name", name)
|
|
||||||
final_content = result.params.get("content", content)
|
|
||||||
final_connector_id = result.params.get("connector_id")
|
|
||||||
final_parent_folder_id = result.params.get("parent_folder_id")
|
|
||||||
|
|
||||||
if not final_name or not final_name.strip():
|
|
||||||
return {"status": "error", "message": "File name cannot be empty."}
|
|
||||||
|
|
||||||
final_name = _ensure_docx_extension(final_name)
|
|
||||||
|
|
||||||
if final_connector_id is not None:
|
|
||||||
result = await db_session.execute(
|
|
||||||
select(SearchSourceConnector).filter(
|
|
||||||
SearchSourceConnector.id == final_connector_id,
|
|
||||||
SearchSourceConnector.search_space_id == search_space_id,
|
|
||||||
SearchSourceConnector.user_id == user_id,
|
|
||||||
SearchSourceConnector.connector_type
|
|
||||||
== SearchSourceConnectorType.ONEDRIVE_CONNECTOR,
|
|
||||||
)
|
|
||||||
)
|
|
||||||
connector = result.scalars().first()
|
|
||||||
else:
|
|
||||||
connector = connectors[0]
|
|
||||||
|
|
||||||
if not connector:
|
|
||||||
return {
|
|
||||||
"status": "error",
|
|
||||||
"message": "Selected OneDrive connector is invalid.",
|
|
||||||
}
|
|
||||||
|
|
||||||
docx_bytes = _markdown_to_docx(final_content or "")
|
|
||||||
|
|
||||||
client = OneDriveClient(session=db_session, connector_id=connector.id)
|
|
||||||
created = await client.create_file(
|
|
||||||
name=final_name,
|
|
||||||
parent_id=final_parent_folder_id,
|
|
||||||
content=docx_bytes,
|
|
||||||
mime_type=DOCX_MIME,
|
|
||||||
)
|
|
||||||
|
|
||||||
logger.info(
|
|
||||||
f"OneDrive file created: id={created.get('id')}, name={created.get('name')}"
|
|
||||||
)
|
|
||||||
|
|
||||||
kb_message_suffix = ""
|
|
||||||
try:
|
|
||||||
from app.services.onedrive import OneDriveKBSyncService
|
|
||||||
|
|
||||||
kb_service = OneDriveKBSyncService(db_session)
|
|
||||||
kb_result = await kb_service.sync_after_create(
|
|
||||||
file_id=created.get("id"),
|
|
||||||
file_name=created.get("name", final_name),
|
|
||||||
mime_type=DOCX_MIME,
|
|
||||||
web_url=created.get("webUrl"),
|
|
||||||
content=final_content,
|
|
||||||
connector_id=connector.id,
|
|
||||||
search_space_id=search_space_id,
|
|
||||||
user_id=user_id,
|
|
||||||
)
|
|
||||||
if kb_result["status"] == "success":
|
|
||||||
kb_message_suffix = (
|
|
||||||
" Your knowledge base has also been updated."
|
|
||||||
)
|
|
||||||
else:
|
|
||||||
kb_message_suffix = " This file will be added to your knowledge base in the next scheduled sync."
|
|
||||||
except Exception as kb_err:
|
|
||||||
logger.warning(f"KB sync after create failed: {kb_err}")
|
|
||||||
kb_message_suffix = " This file will be added to your knowledge base in the next scheduled sync."
|
|
||||||
|
|
||||||
return {
|
|
||||||
"status": "success",
|
|
||||||
"file_id": created.get("id"),
|
|
||||||
"name": created.get("name"),
|
|
||||||
"web_url": created.get("webUrl"),
|
|
||||||
"message": f"Successfully created '{created.get('name')}' in OneDrive.{kb_message_suffix}",
|
|
||||||
}
|
|
||||||
|
|
||||||
except Exception as e:
|
|
||||||
from langgraph.errors import GraphInterrupt
|
|
||||||
|
|
||||||
if isinstance(e, GraphInterrupt):
|
|
||||||
raise
|
|
||||||
logger.error(f"Error creating OneDrive file: {e}", exc_info=True)
|
|
||||||
return {
|
|
||||||
"status": "error",
|
|
||||||
"message": "Something went wrong while creating the file. Please try again.",
|
|
||||||
}
|
|
||||||
|
|
||||||
return create_onedrive_file
|
|
||||||
|
|
@ -1,305 +0,0 @@
|
||||||
import logging
|
|
||||||
from typing import Any
|
|
||||||
|
|
||||||
from langchain_core.tools import tool
|
|
||||||
from sqlalchemy import String, and_, cast, func
|
|
||||||
from sqlalchemy.ext.asyncio import AsyncSession
|
|
||||||
from sqlalchemy.future import select
|
|
||||||
|
|
||||||
from app.agents.shared.tools.hitl import request_approval
|
|
||||||
from app.connectors.onedrive.client import OneDriveClient
|
|
||||||
from app.db import (
|
|
||||||
Document,
|
|
||||||
DocumentType,
|
|
||||||
SearchSourceConnector,
|
|
||||||
SearchSourceConnectorType,
|
|
||||||
async_session_maker,
|
|
||||||
)
|
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
|
||||||
|
|
||||||
|
|
||||||
def create_delete_onedrive_file_tool(
|
|
||||||
db_session: AsyncSession | None = None,
|
|
||||||
search_space_id: int | None = None,
|
|
||||||
user_id: str | None = None,
|
|
||||||
):
|
|
||||||
"""
|
|
||||||
Factory function to create the delete_onedrive_file tool.
|
|
||||||
|
|
||||||
The tool acquires its own short-lived ``AsyncSession`` per call via
|
|
||||||
:data:`async_session_maker` so the closure is safe to share across
|
|
||||||
HTTP requests by the compiled-agent cache. Capturing a per-request
|
|
||||||
session here would surface stale/closed sessions on cache hits.
|
|
||||||
|
|
||||||
Args:
|
|
||||||
db_session: Reserved for registry compatibility. Per-call sessions
|
|
||||||
are opened via :data:`async_session_maker` inside the tool body.
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
Configured delete_onedrive_file tool
|
|
||||||
"""
|
|
||||||
del db_session # per-call session — see docstring
|
|
||||||
|
|
||||||
@tool
|
|
||||||
async def delete_onedrive_file(
|
|
||||||
file_name: str,
|
|
||||||
delete_from_kb: bool = False,
|
|
||||||
) -> dict[str, Any]:
|
|
||||||
"""Move a OneDrive file to the recycle bin.
|
|
||||||
|
|
||||||
Use this tool when the user explicitly asks to delete, remove, or trash
|
|
||||||
a file in OneDrive.
|
|
||||||
|
|
||||||
Args:
|
|
||||||
file_name: The exact name of the file to trash.
|
|
||||||
delete_from_kb: Whether to also remove the file from the knowledge base.
|
|
||||||
Default is False.
|
|
||||||
Set to True to remove from both OneDrive and knowledge base.
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
Dictionary with:
|
|
||||||
- status: "success", "rejected", "not_found", or "error"
|
|
||||||
- file_id: OneDrive file ID (if success)
|
|
||||||
- deleted_from_kb: whether the document was removed from the knowledge base
|
|
||||||
- message: Result message
|
|
||||||
|
|
||||||
IMPORTANT:
|
|
||||||
- If status is "rejected", the user explicitly declined. Respond with a brief
|
|
||||||
acknowledgment and do NOT retry or suggest alternatives.
|
|
||||||
- If status is "not_found", relay the exact message to the user and ask them
|
|
||||||
to verify the file name or check if it has been indexed.
|
|
||||||
"""
|
|
||||||
logger.info(
|
|
||||||
f"delete_onedrive_file called: file_name='{file_name}', delete_from_kb={delete_from_kb}"
|
|
||||||
)
|
|
||||||
|
|
||||||
if search_space_id is None or user_id is None:
|
|
||||||
return {
|
|
||||||
"status": "error",
|
|
||||||
"message": "OneDrive tool not properly configured.",
|
|
||||||
}
|
|
||||||
|
|
||||||
try:
|
|
||||||
async with async_session_maker() as db_session:
|
|
||||||
doc_result = await db_session.execute(
|
|
||||||
select(Document)
|
|
||||||
.join(
|
|
||||||
SearchSourceConnector,
|
|
||||||
Document.connector_id == SearchSourceConnector.id,
|
|
||||||
)
|
|
||||||
.filter(
|
|
||||||
and_(
|
|
||||||
Document.search_space_id == search_space_id,
|
|
||||||
Document.document_type == DocumentType.ONEDRIVE_FILE,
|
|
||||||
func.lower(Document.title) == func.lower(file_name),
|
|
||||||
SearchSourceConnector.user_id == user_id,
|
|
||||||
)
|
|
||||||
)
|
|
||||||
.order_by(Document.updated_at.desc().nullslast())
|
|
||||||
.limit(1)
|
|
||||||
)
|
|
||||||
document = doc_result.scalars().first()
|
|
||||||
|
|
||||||
if not document:
|
|
||||||
doc_result = await db_session.execute(
|
|
||||||
select(Document)
|
|
||||||
.join(
|
|
||||||
SearchSourceConnector,
|
|
||||||
Document.connector_id == SearchSourceConnector.id,
|
|
||||||
)
|
|
||||||
.filter(
|
|
||||||
and_(
|
|
||||||
Document.search_space_id == search_space_id,
|
|
||||||
Document.document_type == DocumentType.ONEDRIVE_FILE,
|
|
||||||
func.lower(
|
|
||||||
cast(
|
|
||||||
Document.document_metadata[
|
|
||||||
"onedrive_file_name"
|
|
||||||
],
|
|
||||||
String,
|
|
||||||
)
|
|
||||||
)
|
|
||||||
== func.lower(file_name),
|
|
||||||
SearchSourceConnector.user_id == user_id,
|
|
||||||
)
|
|
||||||
)
|
|
||||||
.order_by(Document.updated_at.desc().nullslast())
|
|
||||||
.limit(1)
|
|
||||||
)
|
|
||||||
document = doc_result.scalars().first()
|
|
||||||
|
|
||||||
if not document:
|
|
||||||
return {
|
|
||||||
"status": "not_found",
|
|
||||||
"message": (
|
|
||||||
f"File '{file_name}' not found in your indexed OneDrive files. "
|
|
||||||
"This could mean: (1) the file doesn't exist, (2) it hasn't been indexed yet, "
|
|
||||||
"or (3) the file name is different."
|
|
||||||
),
|
|
||||||
}
|
|
||||||
|
|
||||||
if not document.connector_id:
|
|
||||||
return {
|
|
||||||
"status": "error",
|
|
||||||
"message": "Document has no associated connector.",
|
|
||||||
}
|
|
||||||
|
|
||||||
meta = document.document_metadata or {}
|
|
||||||
file_id = meta.get("onedrive_file_id")
|
|
||||||
document_id = document.id
|
|
||||||
|
|
||||||
if not file_id:
|
|
||||||
return {
|
|
||||||
"status": "error",
|
|
||||||
"message": "File ID is missing. Please re-index the file.",
|
|
||||||
}
|
|
||||||
|
|
||||||
conn_result = await db_session.execute(
|
|
||||||
select(SearchSourceConnector).filter(
|
|
||||||
and_(
|
|
||||||
SearchSourceConnector.id == document.connector_id,
|
|
||||||
SearchSourceConnector.search_space_id == search_space_id,
|
|
||||||
SearchSourceConnector.user_id == user_id,
|
|
||||||
SearchSourceConnector.connector_type
|
|
||||||
== SearchSourceConnectorType.ONEDRIVE_CONNECTOR,
|
|
||||||
)
|
|
||||||
)
|
|
||||||
)
|
|
||||||
connector = conn_result.scalars().first()
|
|
||||||
if not connector:
|
|
||||||
return {
|
|
||||||
"status": "error",
|
|
||||||
"message": "OneDrive connector not found or access denied.",
|
|
||||||
}
|
|
||||||
|
|
||||||
cfg = connector.config or {}
|
|
||||||
if cfg.get("auth_expired"):
|
|
||||||
return {
|
|
||||||
"status": "auth_error",
|
|
||||||
"message": "OneDrive account needs re-authentication. Please re-authenticate in your connector settings.",
|
|
||||||
"connector_type": "onedrive",
|
|
||||||
}
|
|
||||||
|
|
||||||
context = {
|
|
||||||
"file": {
|
|
||||||
"file_id": file_id,
|
|
||||||
"name": file_name,
|
|
||||||
"document_id": document_id,
|
|
||||||
"web_url": meta.get("web_url"),
|
|
||||||
},
|
|
||||||
"account": {
|
|
||||||
"id": connector.id,
|
|
||||||
"name": connector.name,
|
|
||||||
"user_email": cfg.get("user_email"),
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
result = request_approval(
|
|
||||||
action_type="onedrive_file_trash",
|
|
||||||
tool_name="delete_onedrive_file",
|
|
||||||
params={
|
|
||||||
"file_id": file_id,
|
|
||||||
"connector_id": connector.id,
|
|
||||||
"delete_from_kb": delete_from_kb,
|
|
||||||
},
|
|
||||||
context=context,
|
|
||||||
)
|
|
||||||
|
|
||||||
if result.rejected:
|
|
||||||
return {
|
|
||||||
"status": "rejected",
|
|
||||||
"message": "User declined. Do not retry or suggest alternatives.",
|
|
||||||
}
|
|
||||||
|
|
||||||
final_file_id = result.params.get("file_id", file_id)
|
|
||||||
final_connector_id = result.params.get("connector_id", connector.id)
|
|
||||||
final_delete_from_kb = result.params.get(
|
|
||||||
"delete_from_kb", delete_from_kb
|
|
||||||
)
|
|
||||||
|
|
||||||
if final_connector_id != connector.id:
|
|
||||||
result = await db_session.execute(
|
|
||||||
select(SearchSourceConnector).filter(
|
|
||||||
and_(
|
|
||||||
SearchSourceConnector.id == final_connector_id,
|
|
||||||
SearchSourceConnector.search_space_id
|
|
||||||
== search_space_id,
|
|
||||||
SearchSourceConnector.user_id == user_id,
|
|
||||||
SearchSourceConnector.connector_type
|
|
||||||
== SearchSourceConnectorType.ONEDRIVE_CONNECTOR,
|
|
||||||
)
|
|
||||||
)
|
|
||||||
)
|
|
||||||
validated_connector = result.scalars().first()
|
|
||||||
if not validated_connector:
|
|
||||||
return {
|
|
||||||
"status": "error",
|
|
||||||
"message": "Selected OneDrive connector is invalid or has been disconnected.",
|
|
||||||
}
|
|
||||||
actual_connector_id = validated_connector.id
|
|
||||||
else:
|
|
||||||
actual_connector_id = connector.id
|
|
||||||
|
|
||||||
logger.info(
|
|
||||||
f"Deleting OneDrive file: file_id='{final_file_id}', connector={actual_connector_id}"
|
|
||||||
)
|
|
||||||
|
|
||||||
client = OneDriveClient(
|
|
||||||
session=db_session, connector_id=actual_connector_id
|
|
||||||
)
|
|
||||||
await client.trash_file(final_file_id)
|
|
||||||
|
|
||||||
logger.info(
|
|
||||||
f"OneDrive file deleted (moved to recycle bin): file_id={final_file_id}"
|
|
||||||
)
|
|
||||||
|
|
||||||
trash_result: dict[str, Any] = {
|
|
||||||
"status": "success",
|
|
||||||
"file_id": final_file_id,
|
|
||||||
"message": f"Successfully moved '{file_name}' to the recycle bin.",
|
|
||||||
}
|
|
||||||
|
|
||||||
deleted_from_kb = False
|
|
||||||
if final_delete_from_kb and document_id:
|
|
||||||
try:
|
|
||||||
doc_result = await db_session.execute(
|
|
||||||
select(Document).filter(Document.id == document_id)
|
|
||||||
)
|
|
||||||
doc = doc_result.scalars().first()
|
|
||||||
if doc:
|
|
||||||
await db_session.delete(doc)
|
|
||||||
await db_session.commit()
|
|
||||||
deleted_from_kb = True
|
|
||||||
logger.info(
|
|
||||||
f"Deleted document {document_id} from knowledge base"
|
|
||||||
)
|
|
||||||
else:
|
|
||||||
logger.warning(f"Document {document_id} not found in KB")
|
|
||||||
except Exception as e:
|
|
||||||
logger.error(f"Failed to delete document from KB: {e}")
|
|
||||||
await db_session.rollback()
|
|
||||||
trash_result["warning"] = (
|
|
||||||
f"File moved to recycle bin, but failed to remove from knowledge base: {e!s}"
|
|
||||||
)
|
|
||||||
|
|
||||||
trash_result["deleted_from_kb"] = deleted_from_kb
|
|
||||||
if deleted_from_kb:
|
|
||||||
trash_result["message"] = (
|
|
||||||
f"{trash_result.get('message', '')} (also removed from knowledge base)"
|
|
||||||
)
|
|
||||||
|
|
||||||
return trash_result
|
|
||||||
|
|
||||||
except Exception as e:
|
|
||||||
from langgraph.errors import GraphInterrupt
|
|
||||||
|
|
||||||
if isinstance(e, GraphInterrupt):
|
|
||||||
raise
|
|
||||||
logger.error(f"Error deleting OneDrive file: {e}", exc_info=True)
|
|
||||||
return {
|
|
||||||
"status": "error",
|
|
||||||
"message": "Something went wrong while trashing the file. Please try again.",
|
|
||||||
}
|
|
||||||
|
|
||||||
return delete_onedrive_file
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue