Refactor: decouple task queues, add JS injection support, and simplify stapp rendering

This commit is contained in:
Liang Jiaqing
2026-02-09 09:24:53 +08:00
parent 9bc79db442
commit 8cfafe119d
3 changed files with 32 additions and 40 deletions

View File

@@ -61,16 +61,15 @@ class GeneraticAgent:
self.handler.code_stop_signal.append(1)
def put_task(self, query, source="user"):
while self.display_queue.qsize() > 0:
try: self.display_queue.get_nowait()
except queue.Empty: break
self.task_queue.put({"query": query, "source": source})
display_queue = queue.Queue()
self.task_queue.put({"query": query, "source": source, "output": display_queue})
return display_queue
def run(self):
while True:
task = self.task_queue.get()
self.is_running = True
raw_query, source = task["query"], task["source"]
raw_query, source, display_queue = task["query"], task["source"], task["output"]
self.current_source = source
self.last_active_time = time.time()
@@ -90,15 +89,15 @@ class GeneraticAgent:
if self.stop_sig: break
full_response += chunk
if len(full_response) - last_pos > 50:
self.display_queue.put({'next': f'{full_response}', 'source': source})
display_queue.put({'next': f'{full_response}', 'source': source})
last_pos = len(full_response)
if '</summary>' in full_response: full_response = full_response.replace('</summary>', '</summary>\n\n')
if '</file_content>' in full_response: full_response = re.sub(r'<file_content>\s*(.*?)\s*</file_content>', r'\n````\n<file_content>\n\1\n</file_content>\n````', full_response, flags=re.DOTALL)
self.display_queue.put({'done': full_response, 'source': source})
display_queue.put({'done': full_response, 'source': source})
self.history = handler.history_info
except Exception as e:
print(f"Backend Error: {format_error(e)}")
self.display_queue.put({'done': full_response + f'\n```\n{format_error(e)}\n```', 'source': source})
display_queue.put({'done': full_response + f'\n```\n{format_error(e)}\n```', 'source': source})
finally:
self.is_running = False
self.stop_sig = False

View File

@@ -12,12 +12,9 @@ TOP_PADDING = 300 # 离屏幕上边缘的距离
def get_screen_width():
try:
# GetSystemMetrics(0) 获取主屏幕宽度
user32 = ctypes.windll.user32
return user32.GetSystemMetrics(0)
except:
# 如果不是 Windows 或者出错了,返回一个兜底值 (比如 1920)
return 1920
except: return 1920
def start_streamlit(port):
global proc
@@ -30,6 +27,18 @@ def start_streamlit(port):
proc = subprocess.Popen(cmd)
atexit.register(proc.kill)
def inject(text):
"""注入输入到 Streamlit"""
window.evaluate_js(f"""
const input = document.querySelector('input[data-testid="stChatInputTextInput"]');
if (input) {{
input.value = {repr(text)};
input.dispatchEvent(new Event('input', {{bubbles: true}}));
input.dispatchEvent(new KeyboardEvent('keydown', {{key: 'Enter', keyCode: 13, bubbles: true}}));
}}
""")
if __name__ == '__main__':
port = sys.argv[1] if len(sys.argv) > 1 else "8501"
t = threading.Thread(target=start_streamlit, args=(port,), daemon=True)
@@ -39,7 +48,7 @@ if __name__ == '__main__':
x_pos = screen_width - WINDOW_WIDTH - RIGHT_PADDING
else: x_pos = 100
time.sleep(2)
webview.create_window(
window = webview.create_window(
title='GenericAgent',
url=f'http://localhost:{port}',
width=WINDOW_WIDTH,

View File

@@ -23,12 +23,6 @@ agent = init()
st.title("🖥️ Cowork")
if "idle_buf" not in st.session_state: st.session_state.idle_buf = ""
if "messages" not in st.session_state: st.session_state.messages = []
for msg in st.session_state.messages:
with st.chat_message(msg["role"]): st.markdown(msg["content"])
@st.fragment
def render_llm_switcher():
current_idx = agent.llm_no
@@ -44,36 +38,22 @@ def render_llm_switcher():
st.toast("下次将重新注入System Prompt")
with st.sidebar: render_llm_switcher()
@st.fragment(run_every="1s")
def global_queue_listener():
if agent.current_source == 'auto':
while not agent.display_queue.empty():
item = agent.display_queue.get()
if item.get('source') == 'auto':
if 'next' in item: st.session_state.idle_buf = item['next']
if 'done' in item:
st.session_state.messages.append({"role": "assistant", "content": f"🤖 {item['done']}"})
st.session_state.idle_buf = ""; st.rerun()
if st.session_state.get("idle_buf"):
with st.chat_message("assistant"):
st.write(st.session_state.idle_buf + "")
else:
st.caption("🟢 Agent Listener Active", help=f"Last sync: {int(time.time())}")
st.session_state.idle_buf = ""
global_queue_listener()
def agent_backend_stream(prompt):
agent.put_task(prompt, source="user")
display_queue = agent.put_task(prompt, source="user")
try:
while True:
item = agent.display_queue.get()
item = display_queue.get()
if 'next' in item: yield item['next']
if 'done' in item:
yield item['done']; break
finally:
agent.abort()
if "messages" not in st.session_state: st.session_state.messages = []
for msg in st.session_state.messages:
with st.chat_message(msg["role"]): st.markdown(msg["content"])
if prompt := st.chat_input("请输入指令"):
st.session_state.messages.append({"role": "user", "content": prompt})
with st.chat_message("user"): st.markdown(prompt)
@@ -85,3 +65,7 @@ if prompt := st.chat_input("请输入指令"):
message_placeholder.markdown(response + "")
message_placeholder.markdown(response)
st.session_state.messages.append({"role": "assistant", "content": response})
st.session_state.last_reply_time = int(time.time())
st.markdown(f"""<div id="last-reply-time" style="display:none">{st.session_state.get('last_reply_time', int(time.time()))}</div>""", unsafe_allow_html=True)