- agent_runner: local directory + cloud agent orchestration via Redis - 5 domain agents: filesystem, task, note, project, timeline - integrations: Gmail, MS Graph (Outlook + Teams) - journey: guided chatbot conversation to build prompt_template - routes: REST endpoints (catalog, can-create, trigger) - redis_consumer: subscribes to batch:request:* pattern - ws_context: Redis-based execute_on_client for tool round-trip - Dockerfile with 300s timeout for long-running batch jobs
111 lines
3.1 KiB
Python
111 lines
3.1 KiB
Python
"""Note agent — Markdown note management.
|
|
|
|
Adapted for Batch Agent Service: import from app.ws_context and app.llm.
|
|
"""
|
|
|
|
from __future__ import annotations
|
|
|
|
import re
|
|
from typing import Any
|
|
|
|
from langchain_core.tools import tool
|
|
|
|
from app.llm import embed
|
|
from app.ws_context import execute_on_client
|
|
|
|
_UUID_RE = re.compile(
|
|
r"^[0-9a-fA-F]{8}-[0-9a-fA-F]{4}-[1-5][0-9a-fA-F]{3}-[89abAB][0-9a-fA-F]{3}-[0-9a-fA-F]{12}$"
|
|
)
|
|
|
|
|
|
def _is_uuid(value: str) -> bool:
|
|
return bool(_UUID_RE.match(value))
|
|
|
|
|
|
@tool
|
|
async def list_notes(project_id: str = "") -> str:
|
|
"""List notes, optionally scoped to a project by project_id."""
|
|
normalized_project_id = project_id if (project_id and _is_uuid(project_id)) else ""
|
|
result = await execute_on_client(
|
|
action="select",
|
|
table="notes",
|
|
filters={"projectId": normalized_project_id or None},
|
|
)
|
|
rows = result.get("rows", [])
|
|
if not rows:
|
|
return "No notes found."
|
|
lines = [f"- {r['title']} (id: {r['id']})" for r in rows]
|
|
return f"Found {len(rows)} note(s):\n" + "\n".join(lines)
|
|
|
|
|
|
@tool
|
|
async def get_note(note_id: str) -> str:
|
|
"""Fetch a single note by its UUID to read its full Markdown content."""
|
|
result = await execute_on_client(action="get", table="notes", data={"id": note_id})
|
|
row = result.get("row")
|
|
if not row:
|
|
return f"Note {note_id} not found."
|
|
return f"Note '{row['title']}' (id: {row['id']}):\n\n{row['content']}"
|
|
|
|
|
|
@tool
|
|
async def create_note(title: str, content: str, project_id: str = "") -> str:
|
|
"""Create a new note."""
|
|
result = await execute_on_client(
|
|
action="insert",
|
|
table="notes",
|
|
data={
|
|
"title": title,
|
|
"content": content,
|
|
"projectId": project_id or None,
|
|
},
|
|
)
|
|
row = result["row"]
|
|
vector = await embed(content)
|
|
await execute_on_client(
|
|
action="vector_upsert",
|
|
data={"id": row["id"], "projectId": row.get("projectId"), "content": content},
|
|
vector=vector,
|
|
)
|
|
return f"Note created: '{row['title']}' (id: {row['id']})."
|
|
|
|
|
|
@tool
|
|
async def update_note(note_id: str, title: str = "", content: str = "") -> str:
|
|
"""Update an existing note. Only pass fields that should change."""
|
|
updates: dict[str, Any] = {}
|
|
if title:
|
|
updates["title"] = title
|
|
if content:
|
|
updates["content"] = content
|
|
result = await execute_on_client(
|
|
action="update",
|
|
table="notes",
|
|
data={"id": note_id, "updates": updates},
|
|
)
|
|
row = result["row"]
|
|
if content:
|
|
vector = await embed(content)
|
|
await execute_on_client(
|
|
action="vector_upsert",
|
|
data={"id": note_id, "projectId": row.get("projectId"), "content": content},
|
|
vector=vector,
|
|
)
|
|
return f"Note updated: '{row['title']}' (id: {row['id']})."
|
|
|
|
|
|
@tool
|
|
async def delete_note(note_id: str) -> str:
|
|
"""Delete a note permanently by its UUID."""
|
|
await execute_on_client(action="delete", table="notes", data={"id": note_id})
|
|
return f"Note {note_id} deleted."
|
|
|
|
|
|
NOTE_TOOLS: list[Any] = [
|
|
list_notes,
|
|
get_note,
|
|
create_note,
|
|
update_note,
|
|
delete_note,
|
|
]
|