mirror of
https://github.com/Frooodle/Stirling-PDF.git
synced 2026-05-10 23:10:08 +02:00
# Description of Changes Redesign AI engine so that it autogenerates the `tool_models.py` file from the OpenAPI spec so the Python has access to the Java API parameters and the full list of Java tools that it can run. CI ensures that whenever someone modifies a tool endpoint that the AI enigne tool models get updated as well (the dev gets told to run `task engine:tool-models`). There's loads of advantages to having the Java be the one that actually executes the tools, rather than the frontend as it was previously set up to theoretically use: - The AI gets much better descriptions of the params from the API docs - It'll be usable headless in the future so a Java daemon could run to execute ops on files in a folder without the need for the UI to run - The Java already has all the logic it needs to execute the tools - We don't need to parse the TypeScript to find the API (which is hard because the TS wasn't designed to be computer-read to extract the API) I've also hooked up the prototype frontend to ensure it's working properly, and have built it in a way that all the tool names can be translated properly, which was always an issue with previous prototypes of this. --------- Co-authored-by: Anthony Stirling <77850077+Frooodle@users.noreply.github.com> Co-authored-by: EthanHealy01 <80844253+EthanHealy01@users.noreply.github.com>
153 lines
5.0 KiB
Python
153 lines
5.0 KiB
Python
from __future__ import annotations
|
|
|
|
import pytest
|
|
from pydantic import ValidationError
|
|
|
|
from stirling.agents import UserSpecAgent
|
|
from stirling.contracts import (
|
|
AgentDraft,
|
|
AgentDraftRequest,
|
|
AgentRevisionRequest,
|
|
ConversationMessage,
|
|
EditClarificationRequest,
|
|
EditPlanResponse,
|
|
ToolOperationStep,
|
|
)
|
|
from stirling.models.tool_models import Angle, FlattenParams, RotatePdfParams, ToolEndpoint
|
|
from stirling.services.runtime import AppRuntime
|
|
|
|
|
|
class StubUserSpecAgent(UserSpecAgent):
|
|
def __init__(self, runtime: AppRuntime, draft_result: AgentDraft, revision_result: AgentDraft) -> None:
|
|
super().__init__(runtime)
|
|
self.draft_result = draft_result
|
|
self.revision_result = revision_result
|
|
self.edit_plan = EditPlanResponse(
|
|
summary="Rotate the document.",
|
|
steps=[
|
|
ToolOperationStep(
|
|
tool=ToolEndpoint.ROTATE_PDF,
|
|
parameters=RotatePdfParams(angle=Angle(90)),
|
|
)
|
|
],
|
|
)
|
|
|
|
async def _build_edit_plan(self, user_message: str) -> EditPlanResponse:
|
|
return self.edit_plan
|
|
|
|
async def _run_draft_agent(self, request: AgentDraftRequest, edit_plan: EditPlanResponse) -> AgentDraft:
|
|
return self.draft_result
|
|
|
|
async def _run_revision_agent(self, request: AgentRevisionRequest, edit_plan: EditPlanResponse) -> AgentDraft:
|
|
return self.revision_result
|
|
|
|
|
|
class ClarifyingUserSpecAgent(UserSpecAgent):
|
|
def __init__(self, runtime: AppRuntime) -> None:
|
|
super().__init__(runtime)
|
|
|
|
async def _build_edit_plan(self, user_message: str) -> EditClarificationRequest:
|
|
return EditClarificationRequest(
|
|
question="Which pages should be changed?",
|
|
reason="The request does not specify the target pages.",
|
|
)
|
|
|
|
|
|
@pytest.mark.anyio
|
|
async def test_user_spec_agent_drafts_agent_spec(runtime: AppRuntime) -> None:
|
|
agent = StubUserSpecAgent(
|
|
runtime,
|
|
AgentDraft(
|
|
name="Invoice Cleanup",
|
|
description="Prepare invoices for review.",
|
|
objective="Normalize invoices before accounting review.",
|
|
steps=[
|
|
ToolOperationStep(
|
|
tool=ToolEndpoint.ROTATE_PDF,
|
|
parameters=RotatePdfParams(angle=Angle(90)),
|
|
)
|
|
],
|
|
),
|
|
revision_result=AgentDraft(
|
|
name="Unused",
|
|
description="Unused",
|
|
objective="Unused",
|
|
steps=[],
|
|
),
|
|
)
|
|
|
|
response = await agent.draft(
|
|
AgentDraftRequest(
|
|
user_message="Build me an invoice cleanup agent.",
|
|
conversation_history=[
|
|
ConversationMessage(role="user", content="It should handle scanned PDFs."),
|
|
],
|
|
)
|
|
)
|
|
|
|
assert response.outcome == "draft"
|
|
assert response.draft.name == "Invoice Cleanup"
|
|
assert response.draft.steps[0].kind == "tool"
|
|
|
|
|
|
@pytest.mark.anyio
|
|
async def test_user_spec_agent_revises_existing_draft(runtime: AppRuntime) -> None:
|
|
current_draft = AgentDraft(
|
|
name="Invoice Cleanup",
|
|
description="Prepare invoices for review.",
|
|
objective="Normalize invoices before accounting review.",
|
|
steps=[
|
|
ToolOperationStep(
|
|
tool=ToolEndpoint.ROTATE_PDF,
|
|
parameters=RotatePdfParams(angle=Angle(90)),
|
|
)
|
|
],
|
|
)
|
|
agent = StubUserSpecAgent(
|
|
runtime,
|
|
draft_result=current_draft,
|
|
revision_result=AgentDraft(
|
|
name="Invoice Cleanup",
|
|
description="Prepare invoices for review and reduce file size.",
|
|
objective="Normalize invoices before accounting review.",
|
|
steps=[
|
|
ToolOperationStep(
|
|
tool=ToolEndpoint.ROTATE_PDF,
|
|
parameters=RotatePdfParams(angle=Angle(90)),
|
|
),
|
|
ToolOperationStep(
|
|
tool=ToolEndpoint.FLATTEN,
|
|
parameters=FlattenParams(flatten_only_forms=False, render_dpi=None),
|
|
),
|
|
],
|
|
),
|
|
)
|
|
|
|
response = await agent.revise(
|
|
AgentRevisionRequest(
|
|
user_message="Also compress the files before upload.",
|
|
current_draft=current_draft,
|
|
)
|
|
)
|
|
|
|
assert response.outcome == "draft"
|
|
assert len(response.draft.steps) == 2
|
|
assert response.draft.steps[1].kind == "tool"
|
|
|
|
|
|
def test_tool_operation_step_rejects_mismatched_parameters() -> None:
|
|
with pytest.raises(ValidationError):
|
|
ToolOperationStep(
|
|
tool=ToolEndpoint.ROTATE_PDF,
|
|
parameters=FlattenParams(flatten_only_forms=False, render_dpi=None),
|
|
)
|
|
|
|
|
|
@pytest.mark.anyio
|
|
async def test_user_spec_agent_propagates_edit_clarification(runtime: AppRuntime) -> None:
|
|
agent = ClarifyingUserSpecAgent(runtime)
|
|
|
|
response = await agent.draft(AgentDraftRequest(user_message="Build an agent to rotate some pages."))
|
|
|
|
assert isinstance(response, EditClarificationRequest)
|