Spaces:
Sleeping
Sleeping
Upload 6 files
Browse files- aworld/cmd/web_legacy/chat.py +126 -0
- aworld/cmd/web_legacy/main.py +18 -0
- aworld/cmd/web_legacy/trace.py +50 -0
- aworld/cmd/web_legacy/trace_net.py +243 -0
- aworld/cmd/web_legacy/utils.py +32 -0
- aworld/cmd/web_legacy/web_server.py +9 -0
aworld/cmd/web_legacy/chat.py
ADDED
@@ -0,0 +1,126 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import asyncio
|
2 |
+
import sys
|
3 |
+
import streamlit as st
|
4 |
+
from dotenv import load_dotenv
|
5 |
+
import logging
|
6 |
+
import os
|
7 |
+
import traceback
|
8 |
+
import importlib.util
|
9 |
+
import utils
|
10 |
+
import aworld.trace as trace
|
11 |
+
from trace_net import generate_trace_graph_full
|
12 |
+
from aworld.trace.base import get_tracer_provider
|
13 |
+
|
14 |
+
load_dotenv(os.path.join(os.getcwd(), ".env"))
|
15 |
+
|
16 |
+
logging.basicConfig(level=logging.INFO)
|
17 |
+
logger = logging.getLogger(__name__)
|
18 |
+
|
19 |
+
sys.path.insert(0, os.getcwd())
|
20 |
+
|
21 |
+
|
22 |
+
def agent_page():
|
23 |
+
st.set_page_config(
|
24 |
+
page_title="AWorld Agent",
|
25 |
+
page_icon=":robot_face:",
|
26 |
+
layout="wide",
|
27 |
+
)
|
28 |
+
|
29 |
+
st.markdown(
|
30 |
+
"""\
|
31 |
+
<style>
|
32 |
+
.stAppHeader { display: none; }
|
33 |
+
|
34 |
+
div[data-testid="stMarkdownContainer"] pre {
|
35 |
+
max-height: 300px;
|
36 |
+
overflow-y: auto;
|
37 |
+
}
|
38 |
+
div[data-testid="stMarkdownContainer"] img {
|
39 |
+
max-height: 500px;
|
40 |
+
}
|
41 |
+
</style>""",
|
42 |
+
unsafe_allow_html=True,
|
43 |
+
)
|
44 |
+
|
45 |
+
query_params = st.query_params
|
46 |
+
selected_agent_from_url = query_params.get("agent", None)
|
47 |
+
|
48 |
+
if "selected_agent" not in st.session_state:
|
49 |
+
st.session_state.selected_agent = selected_agent_from_url
|
50 |
+
logger.info(f"Initialized selected_agent from URL: {selected_agent_from_url}")
|
51 |
+
|
52 |
+
if selected_agent_from_url != st.session_state.selected_agent:
|
53 |
+
st.session_state.selected_agent = selected_agent_from_url
|
54 |
+
|
55 |
+
with st.sidebar:
|
56 |
+
st.title("AWorld Agents List")
|
57 |
+
for agent in utils.list_agents():
|
58 |
+
if st.button(agent):
|
59 |
+
st.query_params["agent"] = agent
|
60 |
+
st.session_state.selected_agent = agent
|
61 |
+
logger.info(f"selected_agent={st.session_state.selected_agent}")
|
62 |
+
|
63 |
+
if st.session_state.selected_agent:
|
64 |
+
agent_name = st.session_state.selected_agent
|
65 |
+
st.title(f"AWorld Agent: {agent_name}")
|
66 |
+
|
67 |
+
if prompt := st.chat_input("Input message here~"):
|
68 |
+
|
69 |
+
with st.chat_message("user"):
|
70 |
+
st.markdown(prompt)
|
71 |
+
|
72 |
+
with st.chat_message("assistant"):
|
73 |
+
agent_name = st.session_state.selected_agent
|
74 |
+
agent_package_path = utils.get_agent_package_path(agent_name)
|
75 |
+
agent_module_file = os.path.join(agent_package_path, "agent.py")
|
76 |
+
try:
|
77 |
+
spec = importlib.util.spec_from_file_location(
|
78 |
+
agent_name, agent_module_file
|
79 |
+
)
|
80 |
+
|
81 |
+
if spec is None or spec.loader is None:
|
82 |
+
logger.error(
|
83 |
+
f"Could not load spec for agent {agent_name} from {agent_module_file}"
|
84 |
+
)
|
85 |
+
st.error(f"Error: Could not load agent! {agent_name}")
|
86 |
+
return
|
87 |
+
|
88 |
+
agent_module = importlib.util.module_from_spec(spec)
|
89 |
+
spec.loader.exec_module(agent_module)
|
90 |
+
except Exception as e:
|
91 |
+
logger.error(
|
92 |
+
f"Error loading agent {agent_name}, cwd:{os.getcwd()}, sys.path:{sys.path}: {traceback.format_exc()}"
|
93 |
+
)
|
94 |
+
st.error(f"Error: Could not load agent! {agent_name}")
|
95 |
+
return
|
96 |
+
|
97 |
+
agent = agent_module.AWorldAgent()
|
98 |
+
|
99 |
+
async def markdown_generator():
|
100 |
+
trace_id = None
|
101 |
+
async with trace.span("start") as span:
|
102 |
+
trace_id = span.get_trace_id()
|
103 |
+
async for line in agent.run(prompt):
|
104 |
+
st.write(line)
|
105 |
+
await asyncio.sleep(0.1)
|
106 |
+
|
107 |
+
get_tracer_provider().force_flush(5000)
|
108 |
+
file_name = f"graph.{trace_id}.html"
|
109 |
+
folder_name = "trace_data"
|
110 |
+
generate_trace_graph_full(
|
111 |
+
trace_id, folder_name=folder_name, file_name=file_name
|
112 |
+
)
|
113 |
+
view_page_url = f"/trace?trace_id={trace_id}"
|
114 |
+
st.write(f"\n---\n[View Trace]({view_page_url})\n")
|
115 |
+
|
116 |
+
asyncio.run(markdown_generator())
|
117 |
+
else:
|
118 |
+
st.title("AWorld Agent Chat Assistant")
|
119 |
+
st.info("Please select an Agent from the left sidebar to start")
|
120 |
+
|
121 |
+
|
122 |
+
try:
|
123 |
+
agent_page()
|
124 |
+
except Exception as e:
|
125 |
+
logger.error(f">>> Error: {traceback.format_exc()}")
|
126 |
+
st.error(f"Error: {str(e)}")
|
aworld/cmd/web_legacy/main.py
ADDED
@@ -0,0 +1,18 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import streamlit as st
|
2 |
+
from dotenv import load_dotenv
|
3 |
+
import logging
|
4 |
+
import os
|
5 |
+
import aworld.trace as trace
|
6 |
+
|
7 |
+
load_dotenv(os.path.join(os.getcwd(), ".env"))
|
8 |
+
|
9 |
+
logging.basicConfig(level=logging.INFO)
|
10 |
+
logger = logging.getLogger(__name__)
|
11 |
+
|
12 |
+
trace.configure()
|
13 |
+
|
14 |
+
chat = st.Page("chat.py", title="Chat", icon=":material/message:")
|
15 |
+
trace = st.Page("trace.py", title="Trace")
|
16 |
+
|
17 |
+
pg = st.navigation([chat, trace], position="hidden")
|
18 |
+
pg.run()
|
aworld/cmd/web_legacy/trace.py
ADDED
@@ -0,0 +1,50 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import logging
|
2 |
+
import traceback
|
3 |
+
import streamlit as st
|
4 |
+
import os
|
5 |
+
|
6 |
+
logger = logging.getLogger(__name__)
|
7 |
+
|
8 |
+
def view_page():
|
9 |
+
st.set_page_config(
|
10 |
+
page_title="HTML Viewer",
|
11 |
+
page_icon=":robot_face:",
|
12 |
+
layout="wide",
|
13 |
+
)
|
14 |
+
|
15 |
+
st.markdown(
|
16 |
+
"<style> .stAppHeader { display: none !important;} .stMainBlockContainer { padding: 5px 10px !important; } </style>",
|
17 |
+
unsafe_allow_html=True,
|
18 |
+
)
|
19 |
+
|
20 |
+
query_params = st.query_params
|
21 |
+
trace_id = query_params.get("trace_id", None)
|
22 |
+
|
23 |
+
side, main = st.columns([2, 8])
|
24 |
+
|
25 |
+
with side:
|
26 |
+
if st.button("Back To Chat"):
|
27 |
+
st.switch_page("chat.py")
|
28 |
+
|
29 |
+
with main:
|
30 |
+
if trace_id:
|
31 |
+
try:
|
32 |
+
st.header(f"Chat Trace Graph: {trace_id}")
|
33 |
+
folder_name = "trace_data"
|
34 |
+
file_name = f"graph.{trace_id}.html"
|
35 |
+
html_file_path = os.path.join(
|
36 |
+
os.getcwd(), folder_name, file_name
|
37 |
+
)
|
38 |
+
with open(html_file_path, "r") as file:
|
39 |
+
html_content = file.read()
|
40 |
+
|
41 |
+
st.components.v1.html(html_content, height=600, scrolling=True)
|
42 |
+
except Exception as e:
|
43 |
+
logger.error(f"Error: {traceback.format_exc()}")
|
44 |
+
st.write(f"Error: {traceback.format_exc()}")
|
45 |
+
else:
|
46 |
+
st.write("Parameter error!")
|
47 |
+
|
48 |
+
|
49 |
+
if __name__ == "__main__":
|
50 |
+
view_page()
|
aworld/cmd/web_legacy/trace_net.py
ADDED
@@ -0,0 +1,243 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import requests
|
2 |
+
import json
|
3 |
+
import os
|
4 |
+
from pyvis.network import Network
|
5 |
+
from aworld.logs.util import logger
|
6 |
+
from aworld.trace.base import get_tracer_provider_silent
|
7 |
+
|
8 |
+
run_type_colors = {
|
9 |
+
"LLM": "#E6E6FA",
|
10 |
+
"TOOL": "#99FF99", # green
|
11 |
+
"MCP": "#99FF99",
|
12 |
+
"AGENT": "#9999FF", # blue
|
13 |
+
"OTHER": "#FFFF99" # yellow
|
14 |
+
}
|
15 |
+
|
16 |
+
|
17 |
+
def get_span_color(span):
|
18 |
+
run_type = span.get('run_type', 'OTHER')
|
19 |
+
return run_type_colors.get(run_type, run_type_colors['OTHER'])
|
20 |
+
|
21 |
+
|
22 |
+
def fetch_trace_data(trace_id=None):
|
23 |
+
try:
|
24 |
+
if trace_id:
|
25 |
+
response = requests.get(
|
26 |
+
f'http://localhost:7079/api/traces/{trace_id}')
|
27 |
+
response.raise_for_status()
|
28 |
+
return response.json() or {"root_span": []}
|
29 |
+
except requests.exceptions.RequestException as e:
|
30 |
+
logger.error(f"Error fetching trace data: {e}")
|
31 |
+
return {"root_span": []}
|
32 |
+
|
33 |
+
|
34 |
+
def flatten_spans(node, result=None):
|
35 |
+
if result is None:
|
36 |
+
result = []
|
37 |
+
result.append(node)
|
38 |
+
for child in node.get('children', []):
|
39 |
+
flatten_spans(child, result)
|
40 |
+
return result
|
41 |
+
|
42 |
+
|
43 |
+
def generate_trace_graph(trace_id=None):
|
44 |
+
tracer_provider = get_tracer_provider_silent()
|
45 |
+
if tracer_provider:
|
46 |
+
tracer_provider.force_flush(5000)
|
47 |
+
trace_data = fetch_trace_data(trace_id)
|
48 |
+
net = Network(height="200px", width="100%",
|
49 |
+
notebook=False, cdn_resources='in_line')
|
50 |
+
|
51 |
+
net.set_options("""
|
52 |
+
{
|
53 |
+
"nodes": {
|
54 |
+
"font": {"size": 8}
|
55 |
+
},
|
56 |
+
"physics": {
|
57 |
+
"enabled": true,
|
58 |
+
"hierarchicalRepulsion": {
|
59 |
+
"centralGravity": 0.5,
|
60 |
+
"springLength": 150,
|
61 |
+
"nodeDistance": 120
|
62 |
+
}
|
63 |
+
},
|
64 |
+
"layout": {
|
65 |
+
"hierarchical": {
|
66 |
+
"enabled": true,
|
67 |
+
"direction": "LR",
|
68 |
+
"sortMethod": "directed",
|
69 |
+
"nodeSpacing": 50,
|
70 |
+
"levelSeparation": 100,
|
71 |
+
"blockShifting": true,
|
72 |
+
"edgeMinimization": true
|
73 |
+
}
|
74 |
+
}
|
75 |
+
}
|
76 |
+
""")
|
77 |
+
root_spans = trace_data['root_span']
|
78 |
+
all_spans = []
|
79 |
+
for span in root_spans:
|
80 |
+
all_spans.extend(flatten_spans(span))
|
81 |
+
|
82 |
+
spans = {span['span_id']: span for span in all_spans}
|
83 |
+
for span_id, span in spans.items():
|
84 |
+
title = f"{span['name']}\n({span['run_type']})"
|
85 |
+
net.add_node(span_id, label=title, title=title,
|
86 |
+
shape='box', color=get_span_color(span))
|
87 |
+
if span.get('parent_id') and span['parent_id'] in spans:
|
88 |
+
net.add_edge(span['parent_id'], span_id, arrows='to')
|
89 |
+
|
90 |
+
net.show('trace_graph.html', notebook=False)
|
91 |
+
with open('trace_graph.html', 'a') as f:
|
92 |
+
f.write(f"""
|
93 |
+
<style>
|
94 |
+
#fullscreenBtn {{
|
95 |
+
position: absolute;
|
96 |
+
top: 10px;
|
97 |
+
right: 10px;
|
98 |
+
padding: 8px 12px;
|
99 |
+
background: #4CAF50;
|
100 |
+
color: white;
|
101 |
+
border: none;
|
102 |
+
border-radius: 4px;
|
103 |
+
cursor: pointer;
|
104 |
+
z-index: 1000;
|
105 |
+
}}
|
106 |
+
</style>
|
107 |
+
<div style="margin-top: 20px; padding: 20px; border-top: 1px solid #eee;">
|
108 |
+
<h3>Span Details</h3>
|
109 |
+
<pre id="spanDetails" style="max-height: 600px; overflow-y: auto;">Click on a node to view details</pre>
|
110 |
+
</div>
|
111 |
+
<script>
|
112 |
+
function moveNodesOnce() {{
|
113 |
+
var nodes = network.body.nodes;
|
114 |
+
var offsetX = 100;
|
115 |
+
var offsetY = 80;
|
116 |
+
for (var nodeId in nodes) {{
|
117 |
+
if (nodes.hasOwnProperty(nodeId)) {{
|
118 |
+
var node = nodes[nodeId];
|
119 |
+
network.moveNode(nodeId, node.x + offsetX, node.y + offsetY);
|
120 |
+
}}
|
121 |
+
}}
|
122 |
+
network.off("afterDrawing", moveNodesOnce);
|
123 |
+
}}
|
124 |
+
network.on("afterDrawing", moveNodesOnce);
|
125 |
+
|
126 |
+
network.on("click", function(params) {{
|
127 |
+
var nodeId = params.nodes[0];
|
128 |
+
var spanData = {json.dumps(spans)}[nodeId];
|
129 |
+
var displayData = {{}};
|
130 |
+
for (var key in spanData) {{
|
131 |
+
if (key !== 'children') {{
|
132 |
+
displayData[key] = spanData[key];
|
133 |
+
}}
|
134 |
+
}}
|
135 |
+
document.getElementById("spanDetails").innerHTML =
|
136 |
+
JSON.stringify(displayData, null, 2);
|
137 |
+
}});
|
138 |
+
</script>
|
139 |
+
""")
|
140 |
+
|
141 |
+
|
142 |
+
def generate_trace_graph_full(trace_id=None, folder_name="", file_name="trace_graph_full.html"):
|
143 |
+
trace_data = fetch_trace_data(trace_id)
|
144 |
+
net = Network(height="100%", width="100%",
|
145 |
+
notebook=False, cdn_resources='in_line')
|
146 |
+
|
147 |
+
net.set_options("""
|
148 |
+
{
|
149 |
+
"nodes": {
|
150 |
+
"font": {"size": 8}
|
151 |
+
},
|
152 |
+
"physics": {
|
153 |
+
"enabled": true,
|
154 |
+
"hierarchicalRepulsion": {
|
155 |
+
"centralGravity": 0.5,
|
156 |
+
"springLength": 150,
|
157 |
+
"nodeDistance": 120
|
158 |
+
}
|
159 |
+
},
|
160 |
+
"layout": {
|
161 |
+
"hierarchical": {
|
162 |
+
"enabled": true,
|
163 |
+
"direction": "LR",
|
164 |
+
"sortMethod": "directed",
|
165 |
+
"nodeSpacing": 50,
|
166 |
+
"levelSeparation": 100,
|
167 |
+
"blockShifting": true,
|
168 |
+
"edgeMinimization": true
|
169 |
+
}
|
170 |
+
}
|
171 |
+
}
|
172 |
+
""")
|
173 |
+
|
174 |
+
root_spans = trace_data['root_span']
|
175 |
+
all_spans = []
|
176 |
+
for span in root_spans:
|
177 |
+
all_spans.extend(flatten_spans(span))
|
178 |
+
spans = {span['span_id']: span for span in all_spans}
|
179 |
+
for span_id, span in spans.items():
|
180 |
+
title = f"{span['name']}\n({span['run_type']})"
|
181 |
+
net.add_node(span_id, label=title, title=title,
|
182 |
+
shape='box', color=get_span_color(span))
|
183 |
+
if span.get('parent_id') and span['parent_id'] in spans:
|
184 |
+
net.add_edge(span['parent_id'], span_id, arrows='to')
|
185 |
+
|
186 |
+
folder_path = os.path.join(os.getcwd(), folder_name)
|
187 |
+
os.makedirs(folder_path, exist_ok=True)
|
188 |
+
file_path = os.path.join(folder_path, file_name)
|
189 |
+
net.show(file_path, notebook=False)
|
190 |
+
|
191 |
+
with open(file_path, 'a') as f:
|
192 |
+
f.write(f"""
|
193 |
+
<style>
|
194 |
+
body {{ margin: 0; padding: 0; }}
|
195 |
+
#mynetwork {{
|
196 |
+
width: 100vw;
|
197 |
+
height: 100vh;
|
198 |
+
}}
|
199 |
+
#spanDetails {{
|
200 |
+
position: absolute;
|
201 |
+
right: 0;
|
202 |
+
top: 0;
|
203 |
+
width: 30%;
|
204 |
+
height: 100%;
|
205 |
+
padding: 20px;
|
206 |
+
background: white;
|
207 |
+
border-left: 1px solid #eee;
|
208 |
+
overflow-y: auto;
|
209 |
+
z-index: 1000;
|
210 |
+
}}
|
211 |
+
</style>
|
212 |
+
<div>
|
213 |
+
<h3>Span Details</h3>
|
214 |
+
<pre id="spanDetails" style="max-height: 100%; overflow-y: auto;">Click on a node to view details</pre>
|
215 |
+
</div>
|
216 |
+
<script>
|
217 |
+
function moveNodesOnce() {{
|
218 |
+
var nodes = network.body.nodes;
|
219 |
+
var offsetX = -250;
|
220 |
+
var offsetY = 0;
|
221 |
+
for (var nodeId in nodes) {{
|
222 |
+
if (nodes.hasOwnProperty(nodeId)) {{
|
223 |
+
var node = nodes[nodeId];
|
224 |
+
network.moveNode(nodeId, node.x + offsetX, node.y + offsetY);
|
225 |
+
}}
|
226 |
+
}}
|
227 |
+
network.off("afterDrawing", moveNodesOnce);
|
228 |
+
}}
|
229 |
+
network.on("afterDrawing", moveNodesOnce);
|
230 |
+
network.on("click", function(params) {{
|
231 |
+
var nodeId = params.nodes[0];
|
232 |
+
var spanData = {json.dumps(spans)}[nodeId];
|
233 |
+
var displayData = {{}};
|
234 |
+
for (var key in spanData) {{
|
235 |
+
if (key !== 'children') {{
|
236 |
+
displayData[key] = spanData[key];
|
237 |
+
}}
|
238 |
+
}}
|
239 |
+
document.getElementById("spanDetails").innerHTML =
|
240 |
+
JSON.stringify(displayData, null, 2);
|
241 |
+
}});
|
242 |
+
</script>
|
243 |
+
""")
|
aworld/cmd/web_legacy/utils.py
ADDED
@@ -0,0 +1,32 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import os
|
2 |
+
|
3 |
+
|
4 |
+
def list_agents():
|
5 |
+
agents_dir = os.path.join(os.getcwd(), "agent_deploy")
|
6 |
+
|
7 |
+
if not os.path.exists(agents_dir):
|
8 |
+
return []
|
9 |
+
|
10 |
+
try:
|
11 |
+
# 列出agents_dir下的所有目录
|
12 |
+
agents = []
|
13 |
+
for item in os.listdir(agents_dir):
|
14 |
+
item_path = os.path.join(agents_dir, item)
|
15 |
+
if os.path.isdir(item_path):
|
16 |
+
# 检查是否包含agent.py文件
|
17 |
+
agent_file = os.path.join(item_path, "agent.py")
|
18 |
+
if os.path.exists(agent_file):
|
19 |
+
agents.append(item)
|
20 |
+
return agents
|
21 |
+
except OSError as e:
|
22 |
+
# 处理权限错误或其他文件系统错误
|
23 |
+
print(f"Error listing agents: {e}")
|
24 |
+
return []
|
25 |
+
|
26 |
+
|
27 |
+
def get_agent_package_path(agent_name):
|
28 |
+
return os.path.join(
|
29 |
+
os.getcwd(),
|
30 |
+
"agent_deploy",
|
31 |
+
agent_name,
|
32 |
+
)
|
aworld/cmd/web_legacy/web_server.py
ADDED
@@ -0,0 +1,9 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import os
|
2 |
+
import streamlit.web.bootstrap as bootstrap
|
3 |
+
|
4 |
+
|
5 |
+
def run_web_server(port, args=None, **kwargs):
|
6 |
+
script = os.path.join(os.path.dirname(os.path.abspath(__file__)), "main.py")
|
7 |
+
kwargs = {**kwargs, "server.port": port}
|
8 |
+
bootstrap.load_config_options(flag_options=kwargs)
|
9 |
+
bootstrap.run(script, False, args, flag_options=kwargs)
|