-
Notifications
You must be signed in to change notification settings - Fork 29
/
Copy pathcli.py
141 lines (103 loc) · 4.42 KB
/
cli.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
# Copyright (c) Microsoft Corporation.
# Licensed under the MIT License.
"""AIOpsLab CLI client."""
import asyncio
from prompt_toolkit import PromptSession
from prompt_toolkit.styles import Style
from prompt_toolkit.patch_stdout import patch_stdout
from rich.console import Console
from rich.markdown import Markdown
from rich.panel import Panel
from prompt_toolkit.completion import WordCompleter
from aiopslab.orchestrator import Orchestrator
WELCOME = """
# AIOpsLab
- Type your commands or actions below.
- Use `exit` to quit the application.
- Use `start <problem_id>` to begin a new problem.
"""
TASK_MESSAGE = """{prob_desc}
You are provided with the following APIs to interact with the service:
{telemetry_apis}
You are also provided an API to a secure terminal to the service where you can run commands:
{shell_api}
Finally, you will submit your solution for this task using the following API:
{submit_api}
At each turn think step-by-step and respond with your action.
"""
class HumanAgent:
def __init__(self, orchestrator):
self.session = PromptSession()
self.console = Console(force_terminal=True, color_system="auto")
self.orchestrator = orchestrator
self.pids = self.orchestrator.probs.get_problem_ids()
self.completer = WordCompleter(self.pids, ignore_case=True, match_middle=True)
def display_welcome_message(self):
self.console.print(Markdown(WELCOME), justify="center")
self.console.print()
def display_context(self, problem_desc, apis):
self.shell_api = self._filter_dict(apis, lambda k, _: "exec_shell" in k)
self.submit_api = self._filter_dict(apis, lambda k, _: "submit" in k)
self.telemetry_apis = self._filter_dict(
apis, lambda k, _: "exec_shell" not in k and "submit" not in k
)
stringify_apis = lambda apis: "\n\n".join(
[f"{k}\n{v}" for k, v in apis.items()]
)
self.task_message = TASK_MESSAGE.format(
prob_desc=problem_desc,
telemetry_apis=stringify_apis(self.telemetry_apis),
shell_api=stringify_apis(self.shell_api),
submit_api=stringify_apis(self.submit_api),
)
self.console.print(Markdown(self.task_message))
def display_env_message(self, env_input):
self.console.print(Panel(env_input, title="Environment", style="white on blue"))
self.console.print()
async def set_problem(self):
user_input = await self.get_user_input(completer=self.completer)
if user_input.startswith("start"):
try:
_, problem_id = user_input.split(maxsplit=1)
except ValueError:
self.console.print("Invalid command. Please use `start <problem_id>`")
return
self.init_problem(problem_id.strip())
else:
self.console.print("Invalid command. Please use `start <problem_id>`")
async def get_action(self, env_input):
self.display_env_message(env_input)
user_input = await self.get_user_input()
template = "Action:```\n{}\n```"
return template.format(user_input)
def init_problem(self, problem_id="misconfig-mitigation-1"):
problem_desc, _, apis = self.orchestrator.init_problem(problem_id)
self.display_context(problem_desc, apis)
async def get_user_input(self, completer=None):
loop = asyncio.get_running_loop()
style = Style.from_dict({"prompt": "ansigreen bold"})
prompt_text = [("class:prompt", "aiopslab> ")]
with patch_stdout():
try:
input = await loop.run_in_executor(
None,
lambda: self.session.prompt(
prompt_text, style=style, completer=completer
),
)
if input.lower() == "exit":
raise SystemExit
return input
except (SystemExit, KeyboardInterrupt, EOFError):
raise SystemExit from None
def _filter_dict(self, dictionary, filter_func):
return {k: v for k, v in dictionary.items() if filter_func(k, v)}
async def main():
orchestrator = Orchestrator()
agent = HumanAgent(orchestrator)
orchestrator.register_agent(agent, name="human")
agent.display_welcome_message()
await agent.set_problem()
await orchestrator.start_problem(max_steps=30)
if __name__ == "__main__":
asyncio.run(main())