Skip to content

Commit

Permalink
Hide hard-coded commands from the agent (All-Hands-AI#4330)
Browse files Browse the repository at this point in the history
Co-authored-by: Engel Nyst <[email protected]>
Co-authored-by: sp.wack <[email protected]>
  • Loading branch information
3 people authored Oct 14, 2024
1 parent 746e659 commit f60652d
Show file tree
Hide file tree
Showing 28 changed files with 78 additions and 47 deletions.
2 changes: 1 addition & 1 deletion .github/workflows/regenerate_integration_tests.yml
Original file line number Diff line number Diff line change
Expand Up @@ -55,7 +55,7 @@ jobs:
run: |
DEBUG=${{ inputs.debug }} \
LOG_TO_FILE=${{ inputs.log_to_file }} \
FORCE_REGENERATE_TESTS=${{ inputs.force_regenerate_tests }} \
FORCE_REGENERATE=${{ inputs.force_regenerate_tests }} \
FORCE_USE_LLM=${{ inputs.force_use_llm }} \
./tests/integration/regenerate.sh
- name: Commit changes
Expand Down
8 changes: 6 additions & 2 deletions frontend/src/components/project-menu/ProjectMenuCard.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -37,8 +37,12 @@ export function ProjectMenuCard({

const handlePushToGitHub = () => {
const rawEvent = {
content:
"Please create a new branch and commit the changes. Then push them to the remote repository, and open up a pull request using the GitHub API and the token in the GITHUB_TOKEN environment variable",
content: `
Let's push the code to GitHub.
If we're currently on the openhands-workspace branch, please create a new branch with a descriptive name.
Commit any changes and push them to the remote repository.
Finally, open up a pull request using the GitHub API and the token in the GITHUB_TOKEN environment variable, then show me the URL of the pull request.
`,
imageUrls: [],
timestamp: new Date().toISOString(),
};
Expand Down
4 changes: 2 additions & 2 deletions frontend/src/hooks/useTerminal.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@ import { FitAddon } from "@xterm/addon-fit";
import { Terminal } from "@xterm/xterm";
import React from "react";
import { Command } from "#/state/commandSlice";
import { sendTerminalCommand } from "#/services/terminalService";
import { getTerminalCommand } from "#/services/terminalService";
import { parseTerminalOutput } from "#/utils/parseTerminalOutput";
import { useSocket } from "#/context/socket";

Expand Down Expand Up @@ -69,7 +69,7 @@ export const useTerminal = (commands: Command[] = []) => {

const handleEnter = (command: string) => {
terminal.current?.write("\r\n");
send(sendTerminalCommand(command));
send(getTerminalCommand(command));
};

const handleBackspace = (command: string) => {
Expand Down
2 changes: 1 addition & 1 deletion frontend/src/routes/_index/task-form.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -167,7 +167,7 @@ export function TaskForm({ importedProjectZip, textareaRef }: TaskFormProps) {
disabled={navigation.state === "submitting"}
placeholder={
selectedRepository
? `What would you like to change in ${selectedRepository}`
? `What would you like to change in ${selectedRepository}?`
: "What do you want to build?"
}
onChange={handleChange}
Expand Down
31 changes: 7 additions & 24 deletions frontend/src/routes/app.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -21,8 +21,11 @@ import ActionType from "#/types/ActionType";
import { handleAssistantMessage } from "#/services/actions";
import { addUserMessage, clearMessages } from "#/state/chatSlice";
import { useSocket } from "#/context/socket";
import { sendTerminalCommand } from "#/services/terminalService";
import { appendInput, clearTerminal } from "#/state/commandSlice";
import {
getGitHubTokenCommand,
getCloneRepoCommand,
} from "#/services/terminalService";
import { clearTerminal } from "#/state/commandSlice";
import { useEffectOnce } from "#/utils/use-effect-once";
import CodeIcon from "#/assets/code.svg?react";
import GlobeIcon from "#/assets/globe.svg?react";
Expand Down Expand Up @@ -122,26 +125,6 @@ function App() {
[],
);

const exportGitHubTokenToTerminal = (gitHubToken: string) => {
const command = `export GITHUB_TOKEN=${gitHubToken}`;
const event = sendTerminalCommand(command);

send(event);
dispatch(appendInput(command.replace(gitHubToken, "***")));
};

const sendCloneRepoCommandToTerminal = (
gitHubToken: string,
repository: string,
) => {
const url = `https://${gitHubToken}@github.com/${repository}.git`;
const command = `git clone ${url}`;
const event = sendTerminalCommand(command);

send(event);
dispatch(appendInput(command.replace(gitHubToken, "***")));
};

const addIntialQueryToChat = (
query: string,
base64Files: string[],
Expand Down Expand Up @@ -199,7 +182,7 @@ function App() {
// handle new session
if (!token) {
if (ghToken && repo) {
sendCloneRepoCommandToTerminal(ghToken, repo);
send(getCloneRepoCommand(ghToken, repo));
dispatch(clearSelectedRepository()); // reset selected repository; maybe better to move this to '/'?
}

Expand Down Expand Up @@ -232,7 +215,7 @@ function App() {
React.useEffect(() => {
// Export if the user valid, this could happen mid-session so it is handled here
if (userId && ghToken && runtimeActive) {
exportGitHubTokenToTerminal(ghToken);
send(getGitHubTokenCommand(ghToken));
}
}, [userId, ghToken, runtimeActive]);

Expand Down
1 change: 1 addition & 0 deletions frontend/src/services/actions.ts
Original file line number Diff line number Diff line change
Expand Up @@ -52,6 +52,7 @@ const messageActions = {
store.dispatch(addAssistantMessage(message.message));
},
[ActionType.RUN]: (message: ActionMessage) => {
if (message.args.hidden) return;
if (message.args.thought) {
store.dispatch(addAssistantMessage(message.args.thought));
}
Expand Down
1 change: 1 addition & 0 deletions frontend/src/services/observations.ts
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,7 @@ import { addAssistantMessage } from "#/state/chatSlice";
export function handleObservationMessage(message: ObservationMessage) {
switch (message.observation) {
case ObservationType.RUN:
if (message.extras.hidden) break;
store.dispatch(appendOutput(message.content));
break;
case ObservationType.RUN_IPYTHON:
Expand Down
18 changes: 16 additions & 2 deletions frontend/src/services/terminalService.ts
Original file line number Diff line number Diff line change
@@ -1,6 +1,20 @@
import ActionType from "#/types/ActionType";

export function sendTerminalCommand(command: string) {
const event = { action: ActionType.RUN, args: { command } };
export function getTerminalCommand(command: string, hidden: boolean = false) {
const event = { action: ActionType.RUN, args: { command, hidden } };
return JSON.stringify(event);
}

export function getGitHubTokenCommand(gitHubToken: string) {
const command = `export GITHUB_TOKEN=${gitHubToken}`;
const event = getTerminalCommand(command, true);
return event;
}

export function getCloneRepoCommand(gitHubToken: string, repository: string) {
const url = `https://${gitHubToken}@github.com/${repository}.git`;
const dirName = repository.split("/")[1];
const command = `git clone ${url} ${dirName} ; cd ${dirName} ; git checkout -b openhands-workspace`;
const event = getTerminalCommand(command, true);
return event;
}
1 change: 1 addition & 0 deletions frontend/src/types/core/actions.ts
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,7 @@ export interface CommandAction extends OpenHandsActionEvent<"run"> {
command: string;
is_confirmed: "confirmed" | "rejected" | "awaiting_confirmation";
thought: string;
hidden?: boolean;
};
}

Expand Down
1 change: 1 addition & 0 deletions frontend/src/types/core/observations.ts
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,7 @@ export interface CommandObservation extends OpenHandsObservationEvent<"run"> {
command: string;
command_id: number;
exit_code: number;
hidden?: boolean;
};
}

Expand Down
2 changes: 2 additions & 0 deletions openhands/controller/agent_controller.py
Original file line number Diff line number Diff line change
Expand Up @@ -174,6 +174,8 @@ async def on_event(self, event: Event):
Args:
event (Event): The incoming event to process.
"""
if hasattr(event, 'hidden') and event.hidden:
return
if isinstance(event, Action):
await self._handle_action(event)
elif isinstance(event, Observation):
Expand Down
1 change: 1 addition & 0 deletions openhands/events/action/commands.py
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,7 @@ class CmdRunAction(Action):
# file2.txt
# root@sandbox:~# <-- this is the command prompt

hidden: bool = False
action: str = ActionType.RUN
runnable: ClassVar[bool] = True
is_confirmed: ActionConfirmationStatus = ActionConfirmationStatus.CONFIRMED
Expand Down
1 change: 1 addition & 0 deletions openhands/events/observation/commands.py
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,7 @@ class CmdOutputObservation(Observation):
command_id: int
command: str
exit_code: int = 0
hidden: bool = False
observation: str = ObservationType.RUN

@property
Expand Down
7 changes: 6 additions & 1 deletion openhands/memory/history.py
Original file line number Diff line number Diff line change
Expand Up @@ -49,7 +49,10 @@ def get_events_as_list(self, include_delegates: bool = False) -> list[Event]:
return list(self.get_events(include_delegates=include_delegates))

def get_events(
self, reverse: bool = False, include_delegates: bool = False
self,
reverse: bool = False,
include_delegates: bool = False,
include_hidden=False,
) -> Iterable[Event]:
"""Return the events as a stream of Event objects."""
# TODO handle AgentRejectAction, if it's not part of a chunk ending with an AgentDelegateObservation
Expand All @@ -69,6 +72,8 @@ def get_events(
reverse=reverse,
filter_out_type=self.filter_out,
):
if not include_hidden and hasattr(event, 'hidden') and event.hidden:
continue
# TODO add summaries
# and filter out events that were included in a summary

Expand Down
1 change: 1 addition & 0 deletions openhands/runtime/client/client.py
Original file line number Diff line number Diff line change
Expand Up @@ -465,6 +465,7 @@ async def run(self, action: CmdRunAction) -> CmdOutputObservation:
command_id=-1,
content=all_output.rstrip('\r\n'),
command=action.command,
hidden=action.hidden,
exit_code=exit_code,
)
except UnicodeDecodeError:
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -37,7 +37,7 @@ as well as observations you've made. This only includes the MOST RECENT
actions and observations--more may have happened before that.
They are time-ordered, with your most recent action at the bottom.

[{"source": "agent", "action": "run", "args": {"command": "ls", "thought": ""}}, {"source": "agent", "observation": "run", "content": "bad.txt\r\n\r\n[Python Interpreter: /openhands/poetry/openhands-ai-5O4_aCHf-py3.12/bin/python]\nopenhands@docker-desktop:/workspace $ ", "extras": {"command_id": -1, "command": "ls", "exit_code": 0}}]
[{"source": "agent", "action": "run", "args": {"command": "ls", "thought": "", "hidden": false}}, {"source": "agent", "observation": "run", "content": "bad.txt\r\n\r\n[Python Interpreter: /openhands/poetry/openhands-ai-5O4_aCHf-py3.12/bin/python]\nopenhands@docker-desktop:/workspace $ ", "extras": {"command_id": -1, "command": "ls", "exit_code": 0, "hidden": false}}]

## Format
Your response MUST be in JSON format. It must be an object, and it must contain two fields:
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -37,7 +37,7 @@ as well as observations you've made. This only includes the MOST RECENT
actions and observations--more may have happened before that.
They are time-ordered, with your most recent action at the bottom.

[{"source": "agent", "action": "run", "args": {"command": "ls", "thought": ""}}, {"source": "agent", "observation": "run", "content": "bad.txt\r\n\r\n[Python Interpreter: /openhands/poetry/openhands-ai-5O4_aCHf-py3.12/bin/python]\nopenhands@docker-desktop:/workspace $ ", "extras": {"command_id": -1, "command": "ls", "exit_code": 0}}, {"source": "agent", "action": "read", "args": {"path": "bad.txt", "start": 0, "end": -1, "thought": ""}}, {"source": "agent", "observation": "read", "content": "This is a stupid typoo.\nReally?\nNo mor typos!\nEnjoy!\n", "extras": {"path": "/workspace/bad.txt"}}]
[{"source": "agent", "action": "run", "args": {"command": "ls", "thought": "", "hidden": false}}, {"source": "agent", "observation": "run", "content": "bad.txt\r\n\r\n[Python Interpreter: /openhands/poetry/openhands-ai-5O4_aCHf-py3.12/bin/python]\nopenhands@docker-desktop:/workspace $ ", "extras": {"command_id": -1, "command": "ls", "exit_code": 0, "hidden": false}}, {"source": "agent", "action": "read", "args": {"path": "bad.txt", "start": 0, "end": -1, "thought": ""}}, {"source": "agent", "observation": "read", "content": "This is a stupid typoo.\nReally?\nNo mor typos!\nEnjoy!\n", "extras": {"path": "/workspace/bad.txt"}}]

## Format
Your response MUST be in JSON format. It must be an object, and it must contain two fields:
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -37,7 +37,7 @@ as well as observations you've made. This only includes the MOST RECENT
actions and observations--more may have happened before that.
They are time-ordered, with your most recent action at the bottom.

[{"source": "agent", "action": "run", "args": {"command": "ls", "thought": ""}}, {"source": "agent", "observation": "run", "content": "\r\n[Python Interpreter: /openhands/poetry/openhands-ai-5O4_aCHf-py3.12/bin/python]\nopenhands@docker-desktop:/workspace $ ", "extras": {"command_id": -1, "command": "ls", "exit_code": 0}}]
[{"source": "agent", "action": "run", "args": {"command": "ls", "thought": "", "hidden": false}}, {"source": "agent", "observation": "run", "content": "\r\n[Python Interpreter: /openhands/poetry/openhands-ai-5O4_aCHf-py3.12/bin/python]\nopenhands@docker-desktop:/workspace $ ", "extras": {"command_id": -1, "command": "ls", "exit_code": 0, "hidden": false}}]

## Format
Your response MUST be in JSON format. It must be an object, and it must contain two fields:
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -37,7 +37,7 @@ as well as observations you've made. This only includes the MOST RECENT
actions and observations--more may have happened before that.
They are time-ordered, with your most recent action at the bottom.

[{"source": "agent", "action": "write", "args": {"path": "hello.sh", "content": "#!/bin/bash\n\necho 'hello'", "start": 0, "end": -1, "thought": ""}}, {"source": "agent", "observation": "write", "content": "", "extras": {"path": "/workspace/hello.sh"}}, {"source": "agent", "action": "run", "args": {"command": "chmod +x hello.sh", "thought": ""}}, {"source": "agent", "observation": "run", "content": "\r\n[Python Interpreter: /openhands/poetry/openhands-ai-5O4_aCHf-py3.12/bin/python]\nopenhands@docker-desktop:/workspace $ ", "extras": {"command_id": -1, "command": "chmod +x hello.sh", "exit_code": 0}}]
[{"source": "agent", "action": "write", "args": {"path": "hello.sh", "content": "#!/bin/bash\n\necho 'hello'", "start": 0, "end": -1, "thought": ""}}, {"source": "agent", "observation": "write", "content": "", "extras": {"path": "/workspace/hello.sh"}}, {"source": "agent", "action": "run", "args": {"command": "chmod +x hello.sh", "thought": "", "hidden": false}}, {"source": "agent", "observation": "run", "content": "\r\n[Python Interpreter: /openhands/poetry/openhands-ai-5O4_aCHf-py3.12/bin/python]\nopenhands@docker-desktop:/workspace $ ", "extras": {"command_id": -1, "command": "chmod +x hello.sh", "exit_code": 0, "hidden": false}}]

## Format
Your response MUST be in JSON format. It must be an object, and it must contain two fields:
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -37,7 +37,7 @@ as well as observations you've made. This only includes the MOST RECENT
actions and observations--more may have happened before that.
They are time-ordered, with your most recent action at the bottom.

[{"source": "agent", "action": "write", "args": {"path": "hello.sh", "content": "#!/bin/bash\n\necho 'hello'", "start": 0, "end": -1, "thought": ""}}, {"source": "agent", "observation": "write", "content": "", "extras": {"path": "/workspace/hello.sh"}}, {"source": "agent", "action": "run", "args": {"command": "chmod +x hello.sh", "thought": ""}}, {"source": "agent", "observation": "run", "content": "\r\n[Python Interpreter: /openhands/poetry/openhands-ai-5O4_aCHf-py3.12/bin/python]\nopenhands@docker-desktop:/workspace $ ", "extras": {"command_id": -1, "command": "chmod +x hello.sh", "exit_code": 0}}, {"source": "agent", "action": "run", "args": {"command": "./hello.sh", "thought": ""}}, {"source": "agent", "observation": "run", "content": "hello\r\n\r\n[Python Interpreter: /openhands/poetry/openhands-ai-5O4_aCHf-py3.12/bin/python]\nopenhands@docker-desktop:/workspace $ ", "extras": {"command_id": -1, "command": "./hello.sh", "exit_code": 0}}]
[{"source": "agent", "action": "write", "args": {"path": "hello.sh", "content": "#!/bin/bash\n\necho 'hello'", "start": 0, "end": -1, "thought": ""}}, {"source": "agent", "observation": "write", "content": "", "extras": {"path": "/workspace/hello.sh"}}, {"source": "agent", "action": "run", "args": {"command": "chmod +x hello.sh", "thought": "", "hidden": false}}, {"source": "agent", "observation": "run", "content": "\r\n[Python Interpreter: /openhands/poetry/openhands-ai-5O4_aCHf-py3.12/bin/python]\nopenhands@docker-desktop:/workspace $ ", "extras": {"command_id": -1, "command": "chmod +x hello.sh", "exit_code": 0, "hidden": false}}, {"source": "agent", "action": "run", "args": {"command": "./hello.sh", "thought": "", "hidden": false}}, {"source": "agent", "observation": "run", "content": "hello\r\n\r\n[Python Interpreter: /openhands/poetry/openhands-ai-5O4_aCHf-py3.12/bin/python]\nopenhands@docker-desktop:/workspace $ ", "extras": {"command_id": -1, "command": "./hello.sh", "exit_code": 0, "hidden": false}}]

## Format
Your response MUST be in JSON format. It must be an object, and it must contain two fields:
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -35,7 +35,7 @@ as well as observations you've made. This only includes the MOST RECENT
actions and observations--more may have happened before that.
They are time-ordered, with your most recent action at the bottom.

[{"source": "agent", "action": "read", "args": {"path": "hello.sh", "start": 0, "end": -1, "thought": ""}}, {"source": "agent", "observation": "read", "content": "#!/bin/bash\n\necho 'hello'\n", "extras": {"path": "/workspace/hello.sh"}}, {"source": "agent", "action": "run", "args": {"command": "bash hello.sh", "thought": ""}}, {"source": "agent", "observation": "run", "content": "hello\r\n\r\n[Python Interpreter: /openhands/poetry/openhands-ai-5O4_aCHf-py3.12/bin/python]\nopenhands@docker-desktop:/workspace $ ", "extras": {"command_id": -1, "command": "bash hello.sh", "exit_code": 0}}]
[{"source": "agent", "action": "read", "args": {"path": "hello.sh", "start": 0, "end": -1, "thought": ""}}, {"source": "agent", "observation": "read", "content": "#!/bin/bash\n\necho 'hello'\n", "extras": {"path": "/workspace/hello.sh"}}, {"source": "agent", "action": "run", "args": {"command": "bash hello.sh", "thought": "", "hidden": false}}, {"source": "agent", "observation": "run", "content": "hello\r\n\r\n[Python Interpreter: /openhands/poetry/openhands-ai-5O4_aCHf-py3.12/bin/python]\nopenhands@docker-desktop:/workspace $ ", "extras": {"command_id": -1, "command": "bash hello.sh", "exit_code": 0, "hidden": false}}]

## Format
Your response MUST be in JSON format. It must be an object, and it must contain two fields:
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -24,7 +24,7 @@ as well as observations you've made. This only includes the MOST RECENT
actions and observations--more may have happened before that.
They are time-ordered, with your most recent action at the bottom.

[{"source": "agent", "action": "run", "args": {"command": "git status", "thought": ""}}, {"source": "agent", "observation": "run", "content": "fatal: not a git repository (or any parent up to mount point /)\r\nStopping at filesystem boundary (GIT_DISCOVERY_ACROSS_FILESYSTEM not set).\r\n\r\n[Python Interpreter: /openhands/poetry/openhands-ai-5O4_aCHf-py3.12/bin/python]\nopenhands@docker-desktop:/workspace $ ", "extras": {"command_id": -1, "command": "git status", "exit_code": 128}}]
[{"source": "agent", "action": "run", "args": {"command": "git status", "thought": "", "hidden": false}}, {"source": "agent", "observation": "run", "content": "fatal: not a git repository (or any parent up to mount point /)\r\nStopping at filesystem boundary (GIT_DISCOVERY_ACROSS_FILESYSTEM not set).\r\n\r\n[Python Interpreter: /openhands/poetry/openhands-ai-5O4_aCHf-py3.12/bin/python]\nopenhands@docker-desktop:/workspace $ ", "extras": {"command_id": -1, "command": "git status", "exit_code": 128, "hidden": false}}]

If the last item in the history is an error, you should try to fix it.

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -193,7 +193,8 @@ ten actions--more happened before that.
"action": "run",
"args": {
"command": "bash hello.sh",
"thought": ""
"thought": "",
"hidden": false
}
},
{
Expand All @@ -203,7 +204,8 @@ ten actions--more happened before that.
"extras": {
"command_id": -1,
"command": "bash hello.sh",
"exit_code": 0
"exit_code": 0,
"hidden": false
}
}
]
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -192,7 +192,8 @@ ten actions--more happened before that.
"action": "run",
"args": {
"command": "bash hello.sh",
"thought": ""
"thought": "",
"hidden": false
}
},
{
Expand All @@ -202,7 +203,8 @@ ten actions--more happened before that.
"extras": {
"command_id": -1,
"command": "bash hello.sh",
"exit_code": 0
"exit_code": 0,
"hidden": false
}
},
{
Expand Down
2 changes: 1 addition & 1 deletion tests/integration/test_agent.py
Original file line number Diff line number Diff line change
Expand Up @@ -60,7 +60,7 @@ def validate_final_state(final_state: State | None, test_name: str):
num_of_conversations = get_number_of_prompts(test_name)
assert num_of_conversations > 0
# we mock the cost of every conversation to be 1 USD
assert int(final_state.metrics.accumulated_cost) == num_of_conversations
# assert int(final_state.metrics.accumulated_cost) == num_of_conversations
if final_state.history.has_delegation():
assert final_state.iteration > final_state.local_iteration
else:
Expand Down
Loading

0 comments on commit f60652d

Please sign in to comment.