import os, sys, threading, queue, time, json, re, random if sys.stdout is None: sys.stdout = open(os.devnull, "w") elif hasattr(sys.stdout, 'reconfigure'): sys.stdout.reconfigure(errors='replace') if sys.stderr is None: sys.stderr = open(os.devnull, "w") elif hasattr(sys.stderr, 'reconfigure'): sys.stderr.reconfigure(errors='replace') sys.path.append(os.path.abspath(os.path.join(os.path.dirname(__file__), '..'))) from llmcore import SiderLLMSession, LLMSession, ToolClient, ClaudeSession, MixinSession, NativeToolClient, NativeClaudeSession, NativeOAISession from agent_loop import agent_runner_loop from ga import GenericAgentHandler, smart_format, get_global_memory, format_error, consume_file script_dir = os.path.dirname(os.path.abspath(__file__)) def load_tool_schema(suffix=''): global TOOLS_SCHEMA TS = open(os.path.join(script_dir, f'assets/tools_schema{suffix}.json'), 'r', encoding='utf-8').read() TOOLS_SCHEMA = json.loads(TS if os.name == 'nt' else TS.replace('powershell', 'bash')) load_tool_schema() mem_dir = os.path.join(script_dir, 'memory') if not os.path.exists(mem_dir): os.makedirs(mem_dir) mem_txt = os.path.join(mem_dir, 'global_mem.txt') if not os.path.exists(mem_txt): open(mem_txt, 'w', encoding='utf-8').write('') mem_insight = os.path.join(mem_dir, 'global_mem_insight.txt') if not os.path.exists(mem_insight): t = os.path.join(script_dir, 'assets/global_mem_insight_template.txt') open(mem_insight, 'w', encoding='utf-8').write(open(t, encoding='utf-8').read() if os.path.exists(t) else '') cdp_cfg = os.path.join(script_dir, 'assets/tmwd_cdp_bridge/config.js') if not os.path.exists(cdp_cfg): try: os.makedirs(os.path.dirname(cdp_cfg), exist_ok=True) open(cdp_cfg, 'w', encoding='utf-8').write(f"const TID = '__ljq_{hex(random.randint(0, 99999999))[2:8]}';") except Exception as e: print(f'[WARN] CDP config init failed: {e} — advanced web features (tmwebdriver) will be unavailable.') def get_system_prompt(): with open(os.path.join(script_dir, 'assets/sys_prompt.txt'), 'r', encoding='utf-8') as f: prompt = f.read() prompt += f"\nToday: {time.strftime('%Y-%m-%d %a')}\n" prompt += get_global_memory() return prompt class GeneraticAgent: def __init__(self): script_dir = os.path.dirname(os.path.abspath(__file__)) os.makedirs(os.path.join(script_dir, 'temp'), exist_ok=True) from llmcore import mykeys llm_sessions = [] for k, cfg in mykeys.items(): if not any(x in k for x in ['api', 'config', 'cookie']): continue try: if 'native' in k and 'claude' in k: llm_sessions += [NativeToolClient(NativeClaudeSession(cfg=cfg))] elif 'native' in k and 'oai' in k: llm_sessions += [NativeToolClient(NativeOAISession(cfg=cfg))] elif 'claude' in k: llm_sessions += [ToolClient(ClaudeSession(cfg=cfg))] elif 'oai' in k: llm_sessions += [ToolClient(LLMSession(cfg=cfg))] elif 'sider' in k: llm_sessions += [ToolClient(SiderLLMSession(cfg={'apikey': cfg, 'model': x})) for x in \ ["gemini-3.0-flash", "gpt-5.4"]] elif 'mixin' in k: llm_sessions += [{'mixin_cfg': cfg}] except: pass for i, s in enumerate(llm_sessions): if isinstance(s, dict) and 'mixin_cfg' in s: try: mixin = MixinSession(llm_sessions, s['mixin_cfg']) if isinstance(mixin._sessions[0], (NativeClaudeSession, NativeOAISession)): llm_sessions[i] = NativeToolClient(mixin) else: llm_sessions[i] = ToolClient(mixin) except Exception as e: print(f'[WARN] Failed to init MixinSession with cfg {s["mixin_cfg"]}: {e}') self.llmclients = llm_sessions self.lock = threading.Lock() self.task_dir = None self.history = [] self.task_queue = queue.Queue() self.is_running = False; self.stop_sig = False self.llm_no = 0; self.inc_out = False self.handler = None; self.verbose = True self.llmclient = self.llmclients[self.llm_no] def next_llm(self, n=-1): self.llm_no = ((self.llm_no + 1) if n < 0 else n) % len(self.llmclients) self.llmclient = self.llmclients[self.llm_no] self.llmclient.last_tools = '' name = self.get_llm_name() if 'glm' in name or 'minimax' in name or 'kimi' in name: load_tool_schema('_cn') else: load_tool_schema() def list_llms(self): return [(i, f"{type(b.backend).__name__}/{b.backend.name}", i == self.llm_no) for i, b in enumerate(self.llmclients)] def get_llm_name(self): b = self.llmclient return f"{type(b.backend).__name__}/{b.backend.name}" def abort(self): if not self.is_running: return print('Abort current task...') self.stop_sig = True if self.handler is not None: self.handler.code_stop_signal.append(1) def put_task(self, query, source="user", images=None): display_queue = queue.Queue() self.task_queue.put({"query": query, "source": source, "images": images or [], "output": display_queue}) return display_queue def run(self): while True: task = self.task_queue.get() raw_query, source, images, display_queue = task["query"], task["source"], task.get("images") or [], task["output"] if raw_query.startswith('/'): if _sm := re.match(r'/session\.(\w+)=(.*)', raw_query.strip()): k, v = _sm.group(1), _sm.group(2) try: v = int(v) except ValueError: try: v = float(v) except ValueError: pass if k == 'history': v = json.loads(open(v, encoding='utf-8').read()) if os.path.exists(v) else [] setattr(self.llmclient.backend, k, v) display_queue.put({'done': f"✅ session.{k} = {v!r}"}) self.task_queue.task_done(); continue self.is_running = True rquery = smart_format(raw_query.replace('\n', ' '), max_str_len=200) self.history.append(f"[USER]: {rquery}") sys_prompt = get_system_prompt() script_dir = os.path.dirname(os.path.abspath(__file__)) handler = GenericAgentHandler(self, self.history, os.path.join(script_dir, 'temp')) if self.handler and 'key_info' in self.handler.working: ki = re.sub(r'\n\[SYSTEM\] 此为.*?工作记忆[。\n]*', '', self.handler.working['key_info']) # 去旧 handler.working['key_info'] = ki handler.working['passed_sessions'] = ps = self.handler.working.get('passed_sessions', 0) + 1 if ps > 0: handler.working['key_info'] += f'\n[SYSTEM] 此为 {ps} 个对话前设置的key_info,若已在新任务,先更新或清除工作记忆。\n' self.handler = handler user_input = raw_query if source == 'feishu' and len(self.history) > 1: # 如果有历史记录且来自飞书,注入到首轮 user_input 中(支持/restore恢复上下文) user_input = handler._get_anchor_prompt() + f"\n\n### 用户当前消息\n{raw_query}" initial_user_content = None # although new handler, the **full** history is in llmclient, so it is full history! gen = agent_runner_loop(self.llmclient, sys_prompt, user_input, handler, TOOLS_SCHEMA, max_turns=40, verbose=self.verbose, initial_user_content=initial_user_content) try: full_resp = ""; last_pos = 0 for chunk in gen: if consume_file(self.task_dir, '_stop'): self.abort() if self.stop_sig: break full_resp += chunk if len(full_resp) - last_pos > 50 or 'LLM Running' in chunk: display_queue.put({'next': full_resp[last_pos:] if self.inc_out else full_resp, 'source': source}) last_pos = len(full_resp) if self.inc_out and last_pos < len(full_resp): display_queue.put({'next': full_resp[last_pos:], 'source': source}) if '' in full_resp: full_resp = full_resp.replace('', '\n\n') if '' in full_resp: full_resp = re.sub(r'\s*(.*?)\s*', r'\n````\n\n\1\n\n````', full_resp, flags=re.DOTALL) display_queue.put({'done': full_resp, 'source': source}) self.history = handler.history_info except Exception as e: print(f"Backend Error: {format_error(e)}") display_queue.put({'done': full_resp + f'\n```\n{format_error(e)}\n```', 'source': source}) finally: if self.stop_sig: print('User aborted the task.') #with self.task_queue.mutex: self.task_queue.queue.clear() self.is_running = self.stop_sig = False self.task_queue.task_done() if self.handler is not None: self.handler.code_stop_signal.append(1) if __name__ == '__main__': import argparse from datetime import datetime parser = argparse.ArgumentParser() parser.add_argument('--task', metavar='IODIR', help='一次性任务模式(文件IO)') parser.add_argument('--reflect', metavar='SCRIPT', help='反射模式:加载监控脚本,check()触发时发任务') parser.add_argument('--input', help='任务内容') parser.add_argument('--llm_no', type=int, default=0, help='LLM编号') parser.add_argument('--verbose', action='store_true', help='输出包含工具执行结果(监察模式用)') parser.add_argument('--bg', action='store_true', help='后台自举: spawn自身去掉--bg, print PID, exit') args = parser.parse_args() if args.bg: import subprocess, platform cmd = [sys.executable, os.path.abspath(__file__)] + [a for a in sys.argv[1:] if a != '--bg'] d = os.path.join(script_dir, f'temp/{args.task}'); os.makedirs(d, exist_ok=True) p = subprocess.Popen(cmd, cwd=script_dir, creationflags=0x08000000 if platform.system() == 'Windows' else 0, stdout=open(os.path.join(d, 'stdout.log'), 'w', encoding='utf-8'), stderr=open(os.path.join(d, 'stderr.log'), 'w', encoding='utf-8')) print(p.pid); sys.exit(0) agent = GeneraticAgent() agent.next_llm(args.llm_no) agent.verbose = args.verbose threading.Thread(target=agent.run, daemon=True).start() if args.task: agent.task_dir = d = os.path.join(script_dir, f'temp/{args.task}'); nround = '' infile = os.path.join(d, 'input.txt') if args.input: os.makedirs(d, exist_ok=True) import glob; [os.remove(f) for f in glob.glob(os.path.join(d, 'output*.txt'))] with open(infile, 'w', encoding='utf-8') as f: f.write(args.input) with open(infile, encoding='utf-8') as f: raw = f.read() while True: dq = agent.put_task(raw, source='task') while 'done' not in (item := dq.get(timeout=120)): if 'next' in item and random.random() < 0.95: # 概率写一次中间结果 with open(f'{d}/output{nround}.txt', 'w', encoding='utf-8') as f: f.write(item.get('next', '')) with open(f'{d}/output{nround}.txt', 'w', encoding='utf-8') as f: f.write(item['done'] + '\n\n[ROUND END]\n') consume_file(d, '_stop') # 已经成功停下来了,避免打断下次reply for _ in range(300): # 等reply.txt,10分钟超时 time.sleep(2) if (raw := consume_file(d, 'reply.txt')): break else: break nround = nround + 1 if isinstance(nround, int) else 1 elif args.reflect: import importlib.util spec = importlib.util.spec_from_file_location('reflect_script', args.reflect) mod = importlib.util.module_from_spec(spec); spec.loader.exec_module(mod) _mt = os.path.getmtime(args.reflect) print(f'[Reflect] loaded {args.reflect}') while True: if os.path.getmtime(args.reflect) != _mt: try: spec.loader.exec_module(mod); _mt = os.path.getmtime(args.reflect); print('[Reflect] reloaded') except Exception as e: print(f'[Reflect] reload error: {e}') time.sleep(getattr(mod, 'INTERVAL', 5)) try: task = mod.check() except Exception as e: print(f'[Reflect] check() error: {e}'); continue if task is None: continue print(f'[Reflect] triggered: {task[:80]}') dq = agent.put_task(task, source='reflect') try: while 'done' not in (item := dq.get(timeout=120)): pass result = item['done'] print(result) except Exception as e: if getattr(mod, 'ONCE', False): raise print(f'[Reflect] drain error: {e}'); result = f'[ERROR] {e}' log_dir = os.path.join(script_dir, 'temp/reflect_logs'); os.makedirs(log_dir, exist_ok=True) script_name = os.path.splitext(os.path.basename(args.reflect))[0] open(os.path.join(log_dir, f'{script_name}_{datetime.now():%Y-%m-%d}.log'), 'a', encoding='utf-8').write(f'[{datetime.now():%m-%d %H:%M}]\n{result}\n\n') if (on_done := getattr(mod, 'on_done', None)): try: on_done(result) except Exception as e: print(f'[Reflect] on_done error: {e}') if getattr(mod, 'ONCE', False): print('[Reflect] ONCE=True, exiting.'); break else: agent.inc_out = True while True: q = input('> ').strip() if not q: continue try: dq = agent.put_task(q, source='user') while True: item = dq.get() if 'next' in item: print(item['next'], end='', flush=True) if 'done' in item: print(); break except KeyboardInterrupt: agent.abort() print('\n[Interrupted]')