llms-py 3.0.10__py3-none-any.whl → 3.0.18__py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- llms/extensions/app/__init__.py +0 -1
- llms/extensions/app/db.py +7 -3
- llms/extensions/app/ui/threadStore.mjs +10 -3
- llms/extensions/computer/README.md +96 -0
- llms/extensions/computer/__init__.py +59 -0
- llms/extensions/computer/base.py +80 -0
- llms/extensions/computer/bash.py +185 -0
- llms/extensions/computer/computer.py +523 -0
- llms/extensions/computer/edit.py +299 -0
- llms/extensions/computer/filesystem.py +542 -0
- llms/extensions/computer/platform.py +461 -0
- llms/extensions/computer/run.py +37 -0
- llms/extensions/core_tools/__init__.py +0 -38
- llms/extensions/providers/anthropic.py +28 -1
- llms/extensions/providers/cerebras.py +0 -1
- llms/extensions/providers/google.py +112 -34
- llms/extensions/skills/LICENSE +202 -0
- llms/extensions/skills/__init__.py +130 -0
- llms/extensions/skills/errors.py +25 -0
- llms/extensions/skills/models.py +39 -0
- llms/extensions/skills/parser.py +178 -0
- llms/extensions/skills/ui/index.mjs +376 -0
- llms/extensions/skills/ui/skills/create-plan/SKILL.md +74 -0
- llms/extensions/skills/validator.py +177 -0
- llms/extensions/system_prompts/ui/index.mjs +6 -10
- llms/extensions/tools/__init__.py +5 -82
- llms/extensions/tools/ui/index.mjs +194 -63
- llms/main.py +502 -146
- llms/ui/ai.mjs +1 -1
- llms/ui/app.css +530 -0
- llms/ui/ctx.mjs +53 -6
- llms/ui/modules/chat/ChatBody.mjs +200 -20
- llms/ui/modules/chat/index.mjs +108 -104
- llms/ui/tailwind.input.css +10 -0
- llms/ui/utils.mjs +25 -1
- {llms_py-3.0.10.dist-info → llms_py-3.0.18.dist-info}/METADATA +2 -2
- {llms_py-3.0.10.dist-info → llms_py-3.0.18.dist-info}/RECORD +41 -24
- {llms_py-3.0.10.dist-info → llms_py-3.0.18.dist-info}/WHEEL +1 -1
- {llms_py-3.0.10.dist-info → llms_py-3.0.18.dist-info}/entry_points.txt +0 -0
- {llms_py-3.0.10.dist-info → llms_py-3.0.18.dist-info}/licenses/LICENSE +0 -0
- {llms_py-3.0.10.dist-info → llms_py-3.0.18.dist-info}/top_level.txt +0 -0
llms/extensions/app/__init__.py
CHANGED
|
@@ -199,7 +199,6 @@ def install(ctx):
|
|
|
199
199
|
"model": thread.get("model"),
|
|
200
200
|
"messages": thread.get("messages"),
|
|
201
201
|
"modalities": thread.get("modalities"),
|
|
202
|
-
"systemPrompt": thread.get("systemPrompt"),
|
|
203
202
|
"tools": thread.get("tools"), # tools request
|
|
204
203
|
"metadata": metadata,
|
|
205
204
|
}
|
llms/extensions/app/db.py
CHANGED
|
@@ -1,5 +1,6 @@
|
|
|
1
1
|
import json
|
|
2
2
|
import os
|
|
3
|
+
import time
|
|
3
4
|
from datetime import datetime, timedelta
|
|
4
5
|
from typing import Any, Dict
|
|
5
6
|
|
|
@@ -344,9 +345,12 @@ class AppDB:
|
|
|
344
345
|
else:
|
|
345
346
|
thread["createdAt"] = now
|
|
346
347
|
thread["updatedAt"] = now
|
|
348
|
+
initial_timestamp = int(time.time() * 1000) + 1
|
|
347
349
|
if "messages" in thread:
|
|
348
|
-
for m in thread["messages"]:
|
|
350
|
+
for idx, m in enumerate(thread["messages"]):
|
|
349
351
|
self.ctx.cache_message_inline_data(m)
|
|
352
|
+
if "timestamp" not in m:
|
|
353
|
+
m["timestamp"] = initial_timestamp + idx
|
|
350
354
|
return with_user(thread, user=user)
|
|
351
355
|
|
|
352
356
|
def create_thread(self, thread: Dict[str, Any], user=None):
|
|
@@ -528,9 +532,9 @@ class AppDB:
|
|
|
528
532
|
with self.db.create_writer_connection() as conn:
|
|
529
533
|
conn.execute(
|
|
530
534
|
"UPDATE thread SET completedAt = :completedAt, error = :error WHERE completedAt IS NULL",
|
|
531
|
-
{"completedAt": datetime.now(), "error": "Server Shutdown"},
|
|
535
|
+
{"completedAt": datetime.now().isoformat(" "), "error": "Server Shutdown"},
|
|
532
536
|
)
|
|
533
537
|
conn.execute(
|
|
534
538
|
"UPDATE request SET completedAt = :completedAt, error = :error WHERE completedAt IS NULL",
|
|
535
|
-
{"completedAt": datetime.now(), "error": "Server Shutdown"},
|
|
539
|
+
{"completedAt": datetime.now().isoformat(" "), "error": "Server Shutdown"},
|
|
536
540
|
)
|
|
@@ -354,11 +354,11 @@ async function startNewThread({ title, model, tools, redirect }) {
|
|
|
354
354
|
async function queueChat(ctxRequest, options = {}) {
|
|
355
355
|
if (!ctxRequest.request) return ctx.createErrorResult({ message: 'No request provided' })
|
|
356
356
|
if (!ctxRequest.thread) return ctx.createErrorResult({ message: 'No thread provided' })
|
|
357
|
-
|
|
358
|
-
ctxRequest.request.metadata = {}
|
|
359
|
-
}
|
|
357
|
+
ctxRequest = ctx.createChatContext(ctxRequest)
|
|
360
358
|
ctx.chatRequestFilters.forEach(f => f(ctxRequest))
|
|
361
359
|
const { thread, request } = ctxRequest
|
|
360
|
+
ctx.completeChatContext(ctxRequest)
|
|
361
|
+
|
|
362
362
|
const api = await ctx.postJson(`/ext/app/threads/${thread.id}/chat`, {
|
|
363
363
|
...options,
|
|
364
364
|
body: typeof request == 'string'
|
|
@@ -381,6 +381,12 @@ async function loadThreadDetails(id, opt = null) {
|
|
|
381
381
|
return threadDetails.value[id]
|
|
382
382
|
}
|
|
383
383
|
|
|
384
|
+
function getCurrentThreadSystemPrompt() {
|
|
385
|
+
return currentThread.value?.systemPrompt
|
|
386
|
+
?? currentThread.value?.messages?.find(m => m.role == 'system')?.content
|
|
387
|
+
?? ''
|
|
388
|
+
}
|
|
389
|
+
|
|
384
390
|
// Export the store
|
|
385
391
|
export function useThreadStore() {
|
|
386
392
|
return {
|
|
@@ -391,6 +397,7 @@ export function useThreadStore() {
|
|
|
391
397
|
groupedThreads,
|
|
392
398
|
|
|
393
399
|
// Actions
|
|
400
|
+
getCurrentThreadSystemPrompt,
|
|
394
401
|
query,
|
|
395
402
|
createThread,
|
|
396
403
|
updateThread,
|
|
@@ -0,0 +1,96 @@
|
|
|
1
|
+
# Computer Use Tools
|
|
2
|
+
|
|
3
|
+
This extension provides a set of tools that allow an Agent to interact with a computer environment in a way similar to a human user. It includes capabilities for screen interaction (mouse/keyboard), shell execution, and file editing. Based on [Anthropic's computer use tools](https://github.com/anthropics/anthropic-quickstarts/tree/main/computer-use-demo).
|
|
4
|
+
|
|
5
|
+
## Available Tools
|
|
6
|
+
|
|
7
|
+
### 1. Computer Tool (`computer`)
|
|
8
|
+
Allows interaction with the screen, keyboard, and mouse.
|
|
9
|
+
|
|
10
|
+
**Capabilities:**
|
|
11
|
+
- **Mouse Interaction**: Move cursor, click (left, right, middle, double, triple), click & drag.
|
|
12
|
+
- **Keyboard Interaction**: Type text, press specific keys or key combinations.
|
|
13
|
+
- **Screen**: Take screenshots, get cursor position.
|
|
14
|
+
- **Zooming**: Zoom into specific regions of the screen (Action: `zoom`).
|
|
15
|
+
|
|
16
|
+
**Key Parameters:**
|
|
17
|
+
- `action`: The action to perform (e.g., `mouse_move`, `left_click`, `type`, `screenshot`, `zoom`).
|
|
18
|
+
- `coordinate`: `(x, y)` coordinates for mouse actions.
|
|
19
|
+
- `text`: Text to type.
|
|
20
|
+
- `key`: Key sequence to press (e.g., `Return`, `Control+c`).
|
|
21
|
+
- `region`: `(x0, y0, x1, y1)` region for zooming.
|
|
22
|
+
|
|
23
|
+
### 2. Bash Tool (`bash`)
|
|
24
|
+
Provides a persistent shell session to execute command-line instructions.
|
|
25
|
+
|
|
26
|
+
**Capabilities:**
|
|
27
|
+
- **Execute Commands**: Run any bash command.
|
|
28
|
+
- **Persistent Session**: State (like environment variables, working directory) is preserved between calls within the same session.
|
|
29
|
+
- **Process Management**: Can restart the session if needed.
|
|
30
|
+
- **Open Files/URLs**: Helper function `open` allows opening files or URLs using the system's default handler (`xdg-open`, `open`, or `start`).
|
|
31
|
+
|
|
32
|
+
**Key Parameters:**
|
|
33
|
+
- `command`: The bash command to execute.
|
|
34
|
+
- `restart`: Boolean to restart the session.
|
|
35
|
+
|
|
36
|
+
### 3. Edit Tool (`str_replace_editor`)
|
|
37
|
+
A filesystem editor for viewing and modifying files.
|
|
38
|
+
|
|
39
|
+
**Capabilities:**
|
|
40
|
+
- **View**: Read file contents or list directories.
|
|
41
|
+
- **Create**: Create new files with content.
|
|
42
|
+
- **String Replace**: Replace unique strings in a file (robust for LLM editing).
|
|
43
|
+
- **Insert**: Insert text at specific line numbers.
|
|
44
|
+
- **Undo**: Undo the last edit to a file.
|
|
45
|
+
|
|
46
|
+
**Key Parameters:**
|
|
47
|
+
- `command`: The edit command (`view`, `create`, `str_replace`, `insert`, `undo_edit`).
|
|
48
|
+
- `path`: Absolute path to the file or directory.
|
|
49
|
+
- `file_text`: Content for file creation.
|
|
50
|
+
- `old_str` / `new_str`: Strings for replacement.
|
|
51
|
+
|
|
52
|
+
---
|
|
53
|
+
|
|
54
|
+
## Capabilities & Workflows
|
|
55
|
+
|
|
56
|
+
These tools are designed to work together to enable complex end-to-end tasks. An Agent can act as a developer, tester, or general user.
|
|
57
|
+
|
|
58
|
+
### Example: "Build a Tetris web app in a tetris folder, open it then take a screenshot"
|
|
59
|
+
|
|
60
|
+
To achieve this high-level task, the Agent would sequence the tools as follows:
|
|
61
|
+
|
|
62
|
+
1. **Create the Project Structure**
|
|
63
|
+
* **Tool**: `bash`
|
|
64
|
+
* **Command**: `mkdir -p tetris`
|
|
65
|
+
* *Result*: Creates the folder.
|
|
66
|
+
|
|
67
|
+
2. **Create the Application Files**
|
|
68
|
+
* **Tool**: `edit` (command: `create`)
|
|
69
|
+
* **Path**: `/path/to/tetris/index.html`
|
|
70
|
+
* **Content**: (HTML code for Tetris game)
|
|
71
|
+
* *Result*: Writes the HTML file.
|
|
72
|
+
|
|
73
|
+
3. **Open the Application**
|
|
74
|
+
* **Tool**: `bash` (via helper `open`) or `bash` directly.
|
|
75
|
+
* **Command**: `xdg-open /path/to/tetris/index.html` (Linux) or just `python -m http.server` and open localhost.
|
|
76
|
+
* *Result*: Opens the file in the default web browser.
|
|
77
|
+
|
|
78
|
+
4. **Wait & Verify**
|
|
79
|
+
* **Tool**: `computer`
|
|
80
|
+
* **Action**: `wait` or `screenshot` to see if it loaded.
|
|
81
|
+
|
|
82
|
+
5. **Take a Screenshot**
|
|
83
|
+
* **Tool**: `computer`
|
|
84
|
+
* **Action**: `screenshot`
|
|
85
|
+
* *Result*: Captures the visual state of the running Tetris app for the user to see.
|
|
86
|
+
|
|
87
|
+
### How it handles the "Build a Tetris..." request:
|
|
88
|
+
When a user gives the command:
|
|
89
|
+
> "Build a Tetris web app in a tetris folder, open it then take a screenshot"
|
|
90
|
+
|
|
91
|
+
The Agent decomposes this into:
|
|
92
|
+
1. **"Build... in a tetris folder"** -> Uses `bash` to make the directory and `edit` to write the `index.html` / `style.css` / `script.js` files.
|
|
93
|
+
2. **"Open it"** -> Uses `bash` to run a server or open the file in a browser.
|
|
94
|
+
3. **"Take a screenshot"** -> Uses `computer` to verify the visual output.
|
|
95
|
+
|
|
96
|
+
This combination allows the Agent to not just generate code, but **verify** it visually and interactively, closing the loop on development tasks.
|
|
@@ -0,0 +1,59 @@
|
|
|
1
|
+
"""
|
|
2
|
+
Anthropic's Computer Use Tools
|
|
3
|
+
https://github.com/anthropics/claude-quickstarts/tree/main/computer-use-demo
|
|
4
|
+
"""
|
|
5
|
+
|
|
6
|
+
import os
|
|
7
|
+
|
|
8
|
+
from .bash import open, run_bash
|
|
9
|
+
from .computer import computer
|
|
10
|
+
from .edit import edit
|
|
11
|
+
from .filesystem import (
|
|
12
|
+
create_directory,
|
|
13
|
+
directory_tree,
|
|
14
|
+
edit_file,
|
|
15
|
+
filesystem_init,
|
|
16
|
+
get_file_info,
|
|
17
|
+
list_allowed_directories,
|
|
18
|
+
list_directory,
|
|
19
|
+
list_directory_with_sizes,
|
|
20
|
+
move_file,
|
|
21
|
+
read_media_file,
|
|
22
|
+
read_multiple_files,
|
|
23
|
+
read_text_file,
|
|
24
|
+
search_files,
|
|
25
|
+
write_file,
|
|
26
|
+
)
|
|
27
|
+
from .platform import get_display_num, get_screen_resolution
|
|
28
|
+
|
|
29
|
+
width, height = get_screen_resolution()
|
|
30
|
+
# set enviroment variables
|
|
31
|
+
os.environ["WIDTH"] = str(width)
|
|
32
|
+
os.environ["HEIGHT"] = str(height)
|
|
33
|
+
os.environ["DISPLAY_NUM"] = str(get_display_num())
|
|
34
|
+
|
|
35
|
+
|
|
36
|
+
def install(ctx):
|
|
37
|
+
filesystem_init(ctx)
|
|
38
|
+
|
|
39
|
+
ctx.register_tool(run_bash, group="computer")
|
|
40
|
+
ctx.register_tool(open, group="computer")
|
|
41
|
+
ctx.register_tool(edit, group="computer")
|
|
42
|
+
ctx.register_tool(computer, group="computer")
|
|
43
|
+
|
|
44
|
+
ctx.register_tool(read_text_file, group="filesystem")
|
|
45
|
+
ctx.register_tool(read_media_file, group="filesystem")
|
|
46
|
+
ctx.register_tool(read_multiple_files, group="filesystem")
|
|
47
|
+
ctx.register_tool(write_file, group="filesystem")
|
|
48
|
+
ctx.register_tool(edit_file, group="filesystem")
|
|
49
|
+
ctx.register_tool(create_directory, group="filesystem")
|
|
50
|
+
ctx.register_tool(list_directory, group="filesystem")
|
|
51
|
+
ctx.register_tool(list_directory_with_sizes, group="filesystem")
|
|
52
|
+
ctx.register_tool(directory_tree, group="filesystem")
|
|
53
|
+
ctx.register_tool(move_file, group="filesystem")
|
|
54
|
+
ctx.register_tool(search_files, group="filesystem")
|
|
55
|
+
ctx.register_tool(get_file_info, group="filesystem")
|
|
56
|
+
ctx.register_tool(list_allowed_directories, group="filesystem")
|
|
57
|
+
|
|
58
|
+
|
|
59
|
+
__install__ = install
|
|
@@ -0,0 +1,80 @@
|
|
|
1
|
+
from abc import ABCMeta, abstractmethod
|
|
2
|
+
from dataclasses import dataclass, fields, replace
|
|
3
|
+
from typing import Any
|
|
4
|
+
|
|
5
|
+
|
|
6
|
+
class BaseTool(metaclass=ABCMeta):
|
|
7
|
+
"""Abstract base class"""
|
|
8
|
+
|
|
9
|
+
@abstractmethod
|
|
10
|
+
def __call__(self, **kwargs) -> Any:
|
|
11
|
+
"""Executes the tool with the given arguments."""
|
|
12
|
+
...
|
|
13
|
+
|
|
14
|
+
@abstractmethod
|
|
15
|
+
def to_params(
|
|
16
|
+
self,
|
|
17
|
+
) -> Any:
|
|
18
|
+
raise NotImplementedError
|
|
19
|
+
|
|
20
|
+
|
|
21
|
+
@dataclass(kw_only=True, frozen=True)
|
|
22
|
+
class ToolResult:
|
|
23
|
+
"""Represents the result of a tool execution."""
|
|
24
|
+
|
|
25
|
+
output: str | None = None
|
|
26
|
+
error: str | None = None
|
|
27
|
+
base64_image: str | None = None
|
|
28
|
+
system: str | None = None
|
|
29
|
+
|
|
30
|
+
def __bool__(self):
|
|
31
|
+
return any(getattr(self, field.name) for field in fields(self))
|
|
32
|
+
|
|
33
|
+
def __add__(self, other: "ToolResult"):
|
|
34
|
+
def combine_fields(field: str | None, other_field: str | None, concatenate: bool = True):
|
|
35
|
+
if field and other_field:
|
|
36
|
+
if concatenate:
|
|
37
|
+
return field + other_field
|
|
38
|
+
raise ValueError("Cannot combine tool results")
|
|
39
|
+
return field or other_field
|
|
40
|
+
|
|
41
|
+
return ToolResult(
|
|
42
|
+
output=combine_fields(self.output, other.output),
|
|
43
|
+
error=combine_fields(self.error, other.error),
|
|
44
|
+
base64_image=combine_fields(self.base64_image, other.base64_image, False),
|
|
45
|
+
system=combine_fields(self.system, other.system),
|
|
46
|
+
)
|
|
47
|
+
|
|
48
|
+
def replace(self, **kwargs):
|
|
49
|
+
"""Returns a new ToolResult with the given fields replaced."""
|
|
50
|
+
return replace(self, **kwargs)
|
|
51
|
+
|
|
52
|
+
def to_tool_results(self) -> list[dict[str, Any]]:
|
|
53
|
+
text = ""
|
|
54
|
+
if self.output:
|
|
55
|
+
text += f"{self.output}\n"
|
|
56
|
+
if self.error:
|
|
57
|
+
text += f"stderr: {self.error}\n"
|
|
58
|
+
if self.system:
|
|
59
|
+
text += f"system: {self.system}\n"
|
|
60
|
+
ret = []
|
|
61
|
+
if text:
|
|
62
|
+
ret.append({"type": "text", "text": text})
|
|
63
|
+
if self.base64_image:
|
|
64
|
+
ret.append({"type": "image", "format": "png", "data": self.base64_image})
|
|
65
|
+
return ret
|
|
66
|
+
|
|
67
|
+
|
|
68
|
+
class CLIResult(ToolResult):
|
|
69
|
+
"""A ToolResult that can be rendered as a CLI output."""
|
|
70
|
+
|
|
71
|
+
|
|
72
|
+
class ToolFailure(ToolResult):
|
|
73
|
+
"""A ToolResult that represents a failure."""
|
|
74
|
+
|
|
75
|
+
|
|
76
|
+
class ToolError(Exception):
|
|
77
|
+
"""Raised when a tool encounters an error."""
|
|
78
|
+
|
|
79
|
+
def __init__(self, message):
|
|
80
|
+
self.message = message
|
|
@@ -0,0 +1,185 @@
|
|
|
1
|
+
import asyncio
|
|
2
|
+
import os
|
|
3
|
+
import sys
|
|
4
|
+
from typing import Annotated, Any, Literal
|
|
5
|
+
|
|
6
|
+
from .base import BaseTool, CLIResult, ToolError, ToolResult
|
|
7
|
+
|
|
8
|
+
|
|
9
|
+
class _BashSession:
|
|
10
|
+
"""A session of a bash shell."""
|
|
11
|
+
|
|
12
|
+
_started: bool
|
|
13
|
+
_process: asyncio.subprocess.Process
|
|
14
|
+
|
|
15
|
+
command: str = "/bin/bash"
|
|
16
|
+
_output_delay: float = 0.2 # seconds
|
|
17
|
+
_timeout: float = 120.0 # seconds
|
|
18
|
+
_sentinel: str = "<<exit>>"
|
|
19
|
+
|
|
20
|
+
def __init__(self):
|
|
21
|
+
self._started = False
|
|
22
|
+
self._timed_out = False
|
|
23
|
+
|
|
24
|
+
async def start(self):
|
|
25
|
+
if self._started:
|
|
26
|
+
return
|
|
27
|
+
|
|
28
|
+
self._process = await asyncio.create_subprocess_shell(
|
|
29
|
+
self.command,
|
|
30
|
+
preexec_fn=os.setsid,
|
|
31
|
+
shell=True,
|
|
32
|
+
bufsize=0,
|
|
33
|
+
stdin=asyncio.subprocess.PIPE,
|
|
34
|
+
stdout=asyncio.subprocess.PIPE,
|
|
35
|
+
stderr=asyncio.subprocess.PIPE,
|
|
36
|
+
)
|
|
37
|
+
|
|
38
|
+
self._started = True
|
|
39
|
+
|
|
40
|
+
def stop(self):
|
|
41
|
+
"""Terminate the bash shell."""
|
|
42
|
+
if not self._started:
|
|
43
|
+
raise ToolError("Session has not started.")
|
|
44
|
+
if self._process.returncode is not None:
|
|
45
|
+
return
|
|
46
|
+
self._process.terminate()
|
|
47
|
+
|
|
48
|
+
async def run(self, command: str):
|
|
49
|
+
"""Execute a command in the bash shell."""
|
|
50
|
+
if not self._started:
|
|
51
|
+
raise ToolError("Session has not started.")
|
|
52
|
+
if self._process.returncode is not None:
|
|
53
|
+
return ToolResult(
|
|
54
|
+
system="tool must be restarted",
|
|
55
|
+
error=f"bash has exited with returncode {self._process.returncode}",
|
|
56
|
+
)
|
|
57
|
+
if self._timed_out:
|
|
58
|
+
raise ToolError(
|
|
59
|
+
f"timed out: bash has not returned in {self._timeout} seconds and must be restarted",
|
|
60
|
+
)
|
|
61
|
+
|
|
62
|
+
# we know these are not None because we created the process with PIPEs
|
|
63
|
+
assert self._process.stdin
|
|
64
|
+
assert self._process.stdout
|
|
65
|
+
assert self._process.stderr
|
|
66
|
+
|
|
67
|
+
# send command to the process
|
|
68
|
+
self._process.stdin.write(command.encode() + f"; echo '{self._sentinel}'\n".encode())
|
|
69
|
+
await self._process.stdin.drain()
|
|
70
|
+
|
|
71
|
+
# read output from the process, until the sentinel is found
|
|
72
|
+
try:
|
|
73
|
+
async with asyncio.timeout(self._timeout):
|
|
74
|
+
while True:
|
|
75
|
+
await asyncio.sleep(self._output_delay)
|
|
76
|
+
# if we read directly from stdout/stderr, it will wait forever for
|
|
77
|
+
# EOF. use the StreamReader buffer directly instead.
|
|
78
|
+
output = self._process.stdout._buffer.decode() # pyright: ignore[reportAttributeAccessIssue]
|
|
79
|
+
if self._sentinel in output:
|
|
80
|
+
# strip the sentinel and break
|
|
81
|
+
output = output[: output.index(self._sentinel)]
|
|
82
|
+
break
|
|
83
|
+
except asyncio.TimeoutError:
|
|
84
|
+
self._timed_out = True
|
|
85
|
+
raise ToolError(
|
|
86
|
+
f"timed out: bash has not returned in {self._timeout} seconds and must be restarted",
|
|
87
|
+
) from None
|
|
88
|
+
|
|
89
|
+
if output.endswith("\n"):
|
|
90
|
+
output = output[:-1]
|
|
91
|
+
|
|
92
|
+
error = self._process.stderr._buffer.decode() # pyright: ignore[reportAttributeAccessIssue]
|
|
93
|
+
if error.endswith("\n"):
|
|
94
|
+
error = error[:-1]
|
|
95
|
+
|
|
96
|
+
# clear the buffers so that the next output can be read correctly
|
|
97
|
+
self._process.stdout._buffer.clear() # pyright: ignore[reportAttributeAccessIssue]
|
|
98
|
+
self._process.stderr._buffer.clear() # pyright: ignore[reportAttributeAccessIssue]
|
|
99
|
+
|
|
100
|
+
return CLIResult(output=output, error=error)
|
|
101
|
+
|
|
102
|
+
|
|
103
|
+
class BashTool20250124(BaseTool):
|
|
104
|
+
"""
|
|
105
|
+
A tool that allows the agent to run bash commands.
|
|
106
|
+
The tool parameters are defined by Anthropic and are not editable.
|
|
107
|
+
"""
|
|
108
|
+
|
|
109
|
+
_session: _BashSession | None
|
|
110
|
+
|
|
111
|
+
api_type: Literal["bash_20250124"] = "bash_20250124"
|
|
112
|
+
name: Literal["bash"] = "bash"
|
|
113
|
+
|
|
114
|
+
def __init__(self):
|
|
115
|
+
self._session = None
|
|
116
|
+
super().__init__()
|
|
117
|
+
|
|
118
|
+
def to_params(self) -> Any:
|
|
119
|
+
return {
|
|
120
|
+
"type": self.api_type,
|
|
121
|
+
"name": self.name,
|
|
122
|
+
}
|
|
123
|
+
|
|
124
|
+
async def __call__(self, command: str | None = None, restart: bool = False, **kwargs):
|
|
125
|
+
if restart:
|
|
126
|
+
if self._session:
|
|
127
|
+
self._session.stop()
|
|
128
|
+
self._session = _BashSession()
|
|
129
|
+
await self._session.start()
|
|
130
|
+
|
|
131
|
+
return ToolResult(system="tool has been restarted.")
|
|
132
|
+
|
|
133
|
+
if self._session is None:
|
|
134
|
+
self._session = _BashSession()
|
|
135
|
+
await self._session.start()
|
|
136
|
+
|
|
137
|
+
if command is not None:
|
|
138
|
+
return await self._session.run(command)
|
|
139
|
+
|
|
140
|
+
raise ToolError("no command provided.")
|
|
141
|
+
|
|
142
|
+
|
|
143
|
+
class BashTool20241022(BashTool20250124):
|
|
144
|
+
api_type: Literal["bash_20250124"] = "bash_20250124" # pyright: ignore[reportIncompatibleVariableOverride]
|
|
145
|
+
|
|
146
|
+
|
|
147
|
+
g_tool = None
|
|
148
|
+
|
|
149
|
+
|
|
150
|
+
async def run_bash(
|
|
151
|
+
command: Annotated[str | None, "Command to run"],
|
|
152
|
+
restart: Annotated[bool, "Restart the bash session"] = False,
|
|
153
|
+
) -> list[dict[str, Any]]:
|
|
154
|
+
"""
|
|
155
|
+
A tool that allows the agent to run bash commands.
|
|
156
|
+
"""
|
|
157
|
+
global g_tool
|
|
158
|
+
if g_tool is None:
|
|
159
|
+
g_tool = BashTool20241022()
|
|
160
|
+
|
|
161
|
+
result = await g_tool(command=command, restart=restart)
|
|
162
|
+
if isinstance(result, Exception):
|
|
163
|
+
raise result
|
|
164
|
+
else:
|
|
165
|
+
return result.to_tool_results()
|
|
166
|
+
|
|
167
|
+
|
|
168
|
+
async def open(target: Annotated[str, "URL or file path to open"]) -> list[dict[str, Any]]:
|
|
169
|
+
"""
|
|
170
|
+
Open a URL or file using the appropriate system opener, uses `xdg-open` on Linux, `open` on macOS, and `start` on Windows.
|
|
171
|
+
"""
|
|
172
|
+
target = target.strip()
|
|
173
|
+
if not target:
|
|
174
|
+
raise ValueError("No target specified")
|
|
175
|
+
|
|
176
|
+
platform = sys.platform
|
|
177
|
+
|
|
178
|
+
if platform == "darwin":
|
|
179
|
+
cmd = ["open", target]
|
|
180
|
+
elif platform == "win32":
|
|
181
|
+
cmd = ["cmd", "/c", "start", "", target]
|
|
182
|
+
else: # Linux and other Unix-like
|
|
183
|
+
cmd = ["xdg-open", target]
|
|
184
|
+
|
|
185
|
+
return await run_bash(command=" ".join(cmd))
|