{"id":484337,"date":"2026-06-19T17:29:54","date_gmt":"2026-06-19T17:29:54","guid":{"rendered":"https:\/\/savepearlharbor.com\/?p=484337"},"modified":"-0001-11-30T00:00:00","modified_gmt":"-0001-11-29T21:00:00","slug":"","status":"publish","type":"post","link":"https:\/\/savepearlharbor.com\/?p=484337","title":{"rendered":"\u041c\u043e\u0439 \u043b\u0438\u0447\u043d\u044b\u0439 \u0434\u0436\u0443\u043d\u0438\u043e\u0440. \u0427\u0430\u0441\u0442\u044c 3. \u0423\u0447\u0438\u043c \u0430\u0433\u0435\u043d\u0442\u0430 \u0436\u0434\u0430\u0442\u044c"},"content":{"rendered":"<div xmlns=\"http:\/\/www.w3.org\/1999\/xhtml\">\n<p>\u041f\u0440\u0438\u0432\u0435\u0442, \u0425\u0430\u0431\u0440! \u041c\u0435\u043d\u044f \u0437\u043e\u0432\u0443\u0442 \u0412\u043b\u0430\u0434\u0438\u043c\u0438\u0440 \u0438 \u044d\u0442\u043e \u043f\u0440\u043e\u0434\u043e\u043b\u0436\u0435\u043d\u0438\u0435 \u0441\u0442\u0430\u0442\u044c\u0438 \u043f\u0440\u043e \u0440\u0430\u0437\u0440\u0430\u0431\u043e\u0442\u043a\u0443 \u043b\u043e\u043a\u0430\u043b\u044c\u043d\u043e\u0433\u043e \u043a\u043e\u0434\u0435\u0440-\u0430\u0433\u0435\u043d\u0442\u0430.<\/p>\n<p>\u0412 <a href=\"https:\/\/habr.com\/ru\/articles\/1043348\/\" rel=\"noopener noreferrer nofollow\">\u043f\u0435\u0440\u0432\u043e\u0439 \u0447\u0430\u0441\u0442\u0438<\/a> \u043c\u044b \u0441\u043e\u0437\u0434\u0430\u043b\u0438 \u0438\u043d\u0444\u0440\u0430\u0441\u0442\u0440\u0443\u043a\u0442\u0443\u0440\u0443: \u043f\u043e\u0434\u043d\u044f\u043b\u0438 \u043a\u043e\u043d\u0442\u0435\u0439\u043d\u0435\u0440 \u0441 \u043c\u043e\u0434\u0435\u043b\u044c\u044e, \u043d\u0430\u0441\u0442\u0440\u043e\u0438\u043b\u0438 Langfuse \u0434\u043b\u044f \u0442\u0440\u0430\u0441\u0441\u0438\u0440\u043e\u0432\u043a\u0438 \u0438 \u043d\u0430\u043f\u0438\u0441\u0430\u043b\u0438 \u043f\u0440\u043e\u0441\u0442\u0435\u0439\u0448\u0435\u0433\u043e \u0430\u0433\u0435\u043d\u0442\u0430 \u0441 \u0434\u043e\u0441\u0442\u0443\u043f\u043e\u043c \u043a MCP-\u0438\u043d\u0441\u0442\u0440\u0443\u043c\u0435\u043d\u0442\u0430\u043c.<\/p>\n<p>\u0412\u043e <a href=\"https:\/\/habr.com\/ru\/articles\/1049772\/\" rel=\"noopener noreferrer nofollow\">\u0432\u0442\u043e\u0440\u043e\u0439 \u0447\u0430\u0441\u0442\u0438<\/a> \u043c\u044b \u0434\u043e\u0431\u0430\u0432\u0438\u043b\u0438 \u0430\u0433\u0435\u043d\u0442\u0443 \u043c\u043e\u0437\u0433\u0438: \u043f\u043b\u0430\u043d\u0438\u0440\u043e\u0432\u0449\u0438\u043a, \u043e\u0446\u0435\u043d\u0449\u0438\u043a\u0438, \u0437\u0430\u0449\u0438\u0442\u0443 \u043e\u0442 \u0437\u0430\u0446\u0438\u043a\u043b\u0438\u0432\u0430\u043d\u0438\u044f \u0438 \u0441\u0443\u043c\u043c\u0430\u0440\u0438\u0437\u0430\u0446\u0438\u044e \u043a\u043e\u043d\u0442\u0435\u043a\u0441\u0442\u0430.<\/p>\n<p>\u0421\u0435\u0433\u043e\u0434\u043d\u044f \u043c\u044b \u043f\u0440\u0435\u0432\u0440\u0430\u0442\u0438\u043c \u044d\u0442\u043e\u0442 \u043d\u0430\u0431\u043e\u0440 \u0432 \u0440\u0430\u0431\u043e\u0442\u0430\u044e\u0449\u0438\u0439 \u0433\u0440\u0430\u0444. \u0414\u043e\u0431\u0430\u0432\u0438\u043c \u0447\u0435\u043a\u043f\u043e\u0438\u043d\u0442\u0435\u0440\u044b, \u043f\u0440\u0435\u0440\u044b\u0432\u0430\u043d\u0438\u044f \u0438 \u0434\u043e\u0440\u0430\u0431\u043e\u0442\u0430\u0435\u043c \u0438\u043d\u0442\u0435\u0440\u0444\u0435\u0439\u0441 \u043f\u043e\u0434 \u043e\u0431\u043d\u043e\u0432\u043b\u0451\u043d\u043d\u0443\u044e \u0430\u0440\u0445\u0438\u0442\u0435\u043a\u0442\u0443\u0440\u0443.<\/p>\n<h2>\u0421\u0431\u043e\u0440\u043a\u0430 \u0433\u0440\u0430\u0444\u0430<\/h2>\n<p>\u0414\u043e\u0440\u0430\u0431\u043e\u0442\u043a\u0443 \u0433\u0440\u0430\u0444\u0430 \u043d\u0430\u0447\u043d\u0435\u043c \u0441 \u043a\u043e\u043d\u0441\u0442\u0440\u0443\u043a\u0442\u043e\u0440\u0430 &#8212; \u0432 \u043d\u0435\u0433\u043e \u043d\u0435\u043e\u0431\u0445\u043e\u0434\u0438\u043c\u043e \u0434\u043e\u0431\u0430\u0432\u0438\u0442\u044c \u0447\u0435\u043a\u043f\u043e\u0438\u043d\u0442\u0435\u0440. \u0414\u043e\u043f\u043e\u043b\u043d\u0438\u0442\u0435\u043b\u044c\u043d\u043e, \u0434\u043b\u044f \u0443\u043f\u043e\u0440\u044f\u0434\u043e\u0447\u0438\u0432\u0430\u043d\u0438\u044f \u0442\u0440\u0430\u0441\u0441\u0438\u0440\u043e\u0432\u043e\u043a \u0434\u043e\u0431\u0430\u0432\u0438\u043c <code>session_id<\/code> \u043a LangFuse. \u0415\u0433\u043e \u0431\u0443\u0434\u0435\u043c \u0444\u043e\u0440\u043c\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u043d\u0430 \u043e\u0441\u043d\u043e\u0432\u0435 \u0442\u0435\u043a\u0443\u0449\u0435\u0439 \u0434\u0430\u0442\u044b \u0438 \u0432\u0440\u0435\u043c\u0435\u043d\u0438, \u0434\u043b\u044f \u0447\u0435\u0433\u043e \u0432 \u043a\u043e\u043d\u0441\u0442\u0440\u0443\u043a\u0442\u043e\u0440 \u0434\u043e\u0431\u0430\u0432\u0438\u043c \u0432\u0440\u0435\u043c\u0435\u043d\u043d\u0443\u044e \u043c\u0435\u0442\u043a\u0443 \u0441\u043e\u0437\u0434\u0430\u043d\u0438\u044f \u044d\u043a\u0437\u0435\u043c\u043f\u043b\u044f\u0440\u0430 \u0430\u0433\u0435\u043d\u0442\u0430:<\/p>\n<pre><code class=\"python\">class MCPAgent:    def __init__(self):        self.llm_with_tools: ChatOpenAI | None = None        self.tools: list[BaseTool] | None = None        self.graph = None        self.checkpointer = InMemorySaver()        self.lf_handler = CallbackHandler(public_key=settings.langfuse.public_key)        self.init_time = datetime.now().strftime('%Y%m%d_%H%M%S')<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:87px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u0418\u043d\u0438\u0446\u0438\u0430\u043b\u0438\u0437\u0430\u0446\u0438\u044f \u0433\u0440\u0430\u0444\u0430 \u043d\u0435 \u043c\u0435\u043d\u044f\u043b\u0430\u0441\u044c &#8212; \u0438\u043d\u0438\u0446\u0438\u0430\u043b\u0438\u0437\u0438\u0440\u0443\u0435\u043c \u0438\u043d\u0441\u0442\u0440\u0443\u043c\u0435\u043d\u0442\u044b, \u0431\u0438\u043d\u0434\u0438\u043c \u043d\u0430 LLM \u0438 \u0441\u043e\u0431\u0438\u0440\u0430\u0435\u043c \u0433\u0440\u0430\u0444 (\u043a\u043e\u0434 \u0434\u0430\u043b\u0435\u0435)<\/p>\n<pre><code class=\"python\">class MCPAgent:    # \u041f\u0440\u0435\u0434\u044b\u0434\u0443\u0449\u0438\u0439 \u043a\u043e\u0434    async def init_graph(self):        try:            self.tools = await init_tools()        except Exception as e:            logger.error(f'\u041e\u0448\u0438\u0431\u043a\u0430 \u0438\u043d\u0438\u0446\u0438\u0430\u043b\u0438\u0437\u0430\u0446\u0438\u0438 \u0438\u043d\u0441\u0442\u0440\u0443\u043c\u0435\u043d\u0442\u043e\u0432: {e}')            raise        self.llm_with_tools = settings.llm.chat.llm.bind_tools(self.tools, parallel_tool_calls=False)        self.graph = self._compile_graph()<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u041f\u0435\u0440\u0435\u0439\u0434\u0451\u043c \u043a \u0443\u0441\u043b\u043e\u0432\u043d\u044b\u043c \u0440\u0451\u0431\u0440\u0430\u043c. \u041d\u0430\u0447\u043d\u0451\u043c \u0441 \u0440\u0451\u0431\u0435\u0440 \u043c\u0430\u0440\u0448\u0440\u0443\u0442\u0438\u0440\u0438\u0437\u0430\u0446\u0438\u0438 \u043f\u043e\u0434\u0442\u0432\u0435\u0440\u0436\u0434\u0435\u043d\u0438\u044f \u043f\u043b\u0430\u043d\u0430 \u0438 \u0432\u044b\u043f\u043e\u043b\u043d\u0435\u043d\u0438\u044f \u0448\u0430\u0433\u0430. \u0412 \u043d\u0438\u0445 \u043f\u0440\u043e\u0432\u0435\u0440\u044f\u0435\u043c \u0444\u043b\u0430\u0433 <code>is_approved<\/code>. \u0415\u0441\u043b\u0438 \u043f\u043e\u0434\u0442\u0432\u0435\u0440\u0436\u0434\u0435\u043d\u0438\u0435 \u0435\u0441\u0442\u044c, \u0442\u043e \u043f\u0435\u0440\u0435\u0445\u043e\u0434\u0438\u043c \u043a \u0438\u0441\u043f\u043e\u043b\u043d\u0435\u043d\u0438\u044e \u043f\u043b\u0430\u043d\u0430 \u0438\u043b\u0438 \u0441\u0443\u043c\u043c\u0430\u0440\u0438\u0437\u0430\u0446\u0438\u0438 \u0441\u043e\u043e\u0442\u0432\u0435\u0442\u0441\u0442\u0432\u0435\u043d\u043d\u043e. \u0415\u0441\u043b\u0438 \u043d\u0435\u0442, \u0442\u043e \u0432\u043e\u0437\u0432\u0440\u0430\u0449\u0430\u0435\u043c \u043d\u0430 \u0434\u043e\u0440\u0430\u0431\u043e\u0442\u043a\u0443.<\/p>\n<pre><code class=\"python\">class MCPAgent:    # \u041f\u0440\u0435\u0434\u044b\u0434\u0443\u0449\u0438\u0439 \u043a\u043e\u0434        @staticmethod    def need_adjust_plan_router(state: AgentState) -&gt; Literal['injector', 'planer']:        if state.get('is_approved', False):            return 'injector'        return 'planer'    @staticmethod    def need_modify_step_router(state: AgentState) -&gt; Literal['agent_node', 'compressor']:        if state.get('is_approved', False):            return 'compressor'        return 'agent_node'<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u0422\u0435\u043f\u0435\u0440\u044c \u043c\u0430\u0440\u0448\u0440\u0443\u0442\u0438\u0437\u0430\u0442\u043e\u0440 \u043f\u0435\u0440\u0435\u0445\u043e\u0434\u0430 \u043f\u043e \u0448\u0430\u0433\u0430\u043c. \u0412 \u043d\u0451\u043c \u043c\u044b \u0434\u043e\u043b\u0436\u043d\u044b \u043f\u0440\u043e\u0432\u0435\u0440\u0438\u0442\u044c, \u0447\u0442\u043e \u043f\u043b\u0430\u043d \u0435\u0449\u0451 \u0432 \u043f\u0440\u043e\u0446\u0435\u0441\u0441\u0435 \u0432\u044b\u043f\u043e\u043b\u043d\u0435\u043d\u0438\u044f. \u041f\u0440\u043e\u0432\u0435\u0440\u044f\u0435\u043c \u0441\u0440\u0430\u0432\u043d\u0435\u043d\u0438\u0435\u043c \u043d\u043e\u043c\u0435\u0440\u0430 \u0448\u0430\u0433\u0430 \u0438 \u0434\u043b\u0438\u043d\u044b \u043f\u043b\u0430\u043d\u0430<\/p>\n<pre><code class=\"python\">class MCPAgent:    # \u041f\u0440\u0435\u0434\u044b\u0434\u0443\u0449\u0438\u0439 \u043a\u043e\u0434    @staticmethod    def next_step_router(state: AgentState) -&gt; Literal['injector', 'finalizer']:        if state['current_step'] &lt; len(state['plan']):            return 'injector'        return 'finalizer'<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u0421\u043e\u0431\u0435\u0440\u0451\u043c \u0433\u0440\u0430\u0444. \u0418\u043d\u0438\u0446\u0438\u0430\u043b\u0438\u0437\u0438\u0440\u0443\u0435\u043c \u0443\u0437\u043b\u044b \u0438 \u0441\u043e\u0435\u0434\u0438\u043d\u0438\u043c \u0438\u0445 \u0441\u043e\u0433\u043b\u0430\u0441\u043d\u043e \u0441\u0445\u0435\u043c\u0435 \u0438\u0437 \u0432\u0442\u043e\u0440\u043e\u0439 \u0447\u0430\u0441\u0442\u0438 \u0441\u0442\u0430\u0442\u044c\u0438:<\/p>\n<pre><code class=\"python\">class MCPAgent:    # \u041f\u0440\u0435\u0434\u044b\u0434\u0443\u0449\u0438\u0439 \u043a\u043e\u0434    def _compile_graph(self):        workflow = StateGraph(AgentState)        workflow.add_node('agent_node', AgentNode(llm=self.llm_with_tools).node)        workflow.add_node('compressor', ContextCompressorNode().node)        workflow.add_node('finalizer', FinalizerNode(llm=settings.llm.chat.llm).node)        workflow.add_node('planer', PlanerNode(llm=settings.llm.chat.llm).node)        workflow.add_node('plan_solver', PlanSolverNode().node)        workflow.add_node('agent_solver', AgentSolverNode().node)        workflow.add_node('injector', StepInjectorNode().node)        workflow.add_node('tools', ToolNode(self.tools))        workflow.set_entry_point(key='planer')        workflow.add_edge(start_key='planer', end_key='plan_solver')        workflow.add_conditional_edges(source='plan_solver', path=self.need_adjust_plan_router)        workflow.add_edge(start_key='injector', end_key='agent_node')        workflow.add_conditional_edges(source='agent_node', path=self.agent_router)        workflow.add_edge(start_key='tools', end_key='agent_node')        workflow.add_conditional_edges(source='agent_solver', path=self.need_modify_step_router)        workflow.add_conditional_edges(source='compressor', path=self.next_step_router)        workflow.set_finish_point(key='finalizer')        graph = workflow.compile(            checkpointer=self.checkpointer,            interrupt_after=['planer'],            interrupt_before=['agent_solver'],        )        return graph<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u0412 \u043a\u043e\u043c\u043f\u0438\u043b\u044f\u0442\u043e\u0440 \u043f\u0435\u0440\u0435\u0434\u0430\u0451\u043c \u0434\u0432\u0430 \u043e\u0441\u0442\u0430\u043d\u043e\u0432\u0430 &#8212; \u043f\u043e\u0441\u043b\u0435 \u0444\u043e\u0440\u043c\u0438\u0440\u043e\u0432\u0430\u043d\u0438\u044f \u043f\u043b\u0430\u043d\u0430 \u0438 \u0434\u043e &#171;\u0440\u0435\u0448\u0430\u0442\u0435\u043b\u044f&#187; \u0443\u0437\u043b\u0430-\u0430\u0433\u0435\u043d\u0442\u0430.<\/p>\n<p>\u0414\u043b\u044f \u0442\u043e\u0433\u043e, \u0447\u0442\u043e\u0431\u044b \u0433\u0440\u0430\u0444 \u0440\u0430\u0431\u043e\u0442\u0430\u043b \u0441 \u043e\u0441\u0442\u0430\u043d\u043e\u0432\u0430\u043c\u0438, \u043d\u0430\u043c \u043d\u0430\u0434\u043e \u0434\u0432\u0430 \u043c\u0435\u0442\u043e\u0434\u0430 &#8212; \u043e\u0434\u0438\u043d \u0434\u043b\u044f \u043f\u0435\u0440\u0432\u043e\u043d\u0430\u0447\u0430\u043b\u044c\u043d\u043e\u0433\u043e \u0437\u0430\u043f\u0443\u0441\u043a\u0430 \u0433\u0440\u0430\u0444\u0430, \u0432\u0442\u043e\u0440\u043e\u0439 &#8212; \u0434\u043b\u044f \u043f\u0440\u043e\u0434\u043e\u043b\u0436\u0435\u043d\u0438\u044f \u043e\u0431\u0449\u0435\u043d\u0438\u044f. \u041d\u0430\u0447\u043d\u0435\u043c \u0441 \u0437\u0430\u043f\u0443\u0441\u043a\u0430:<\/p>\n<pre><code class=\"python\">class MCPAgent:    # \u041f\u0440\u0435\u0434\u044b\u0434\u0443\u0449\u0438\u0439 \u043a\u043e\u0434    async def run(self, user_messages: str, request_id: str | None = None) -&gt; dict[str, Any]:        self._check_graph_available()        trace_id = Langfuse.create_trace_id()        return await self._ainvoke_with_tracing(            data={'user_request': user_messages, 'user_input': user_messages, 'trace_id': trace_id},            request_id=request_id, trace_id=trace_id, span_name='agent_run')<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u0421\u043b\u0443\u0436\u0435\u0431\u043d\u044b\u0435 \u043c\u0435\u0442\u043e\u0434\u044b \u0440\u0430\u0437\u0431\u0435\u0440\u0443 \u043f\u043e\u0437\u0436\u0435, \u043f\u043e\u043a\u0430 \u0441\u0430\u043c \u0430\u043b\u0433\u043e\u0440\u0438\u0442\u043c. \u0414\u043b\u044f \u043d\u0430\u0447\u0430\u043b\u0430 \u043f\u0440\u043e\u0432\u0435\u0440\u044f\u0435\u043c, \u0447\u0442\u043e \u043d\u0435 \u0437\u0430\u0431\u044b\u043b\u0438 \u0438\u043d\u0438\u0446\u0438\u0430\u043b\u0438\u0437\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u0433\u0440\u0430\u0444. \u0414\u0430\u043b\u0435\u0435 \u0444\u043e\u0440\u043c\u0438\u0440\u0443\u0435\u043c <code>trace_id<\/code> \u0434\u043b\u044f Langfuse \u0441\u0440\u0435\u0434\u0441\u0442\u0432\u0430\u043c\u0438 Langfuse. \u0411\u0435\u0437 \u043f\u0435\u0440\u0435\u0434\u0430\u0447\u0438 <code>trace_id<\/code> Langfuse \u0431\u0443\u0434\u0435\u0442 \u0441\u043e\u0437\u0434\u0430\u0432\u0430\u0442\u044c \u043e\u0442\u0434\u0435\u043b\u044c\u043d\u044b\u0435 \u0442\u0440\u0435\u0439\u0441\u044b \u0434\u043b\u044f \u043a\u0430\u0436\u0434\u043e\u0433\u043e \u0432\u043e\u0437\u0432\u0440\u0430\u0442\u0430 \u0432 \u0433\u0440\u0430\u0444. \u041d\u0443 \u0438 \u0432\u044b\u0437\u044b\u0432\u0430\u0435\u043c \u0441\u043b\u0443\u0436\u0435\u0431\u043d\u044b\u0439 \u043c\u0435\u0442\u043e\u0434 \u0432\u044b\u0437\u043e\u0432\u0430 LLM.<\/p>\n<p>\u0422\u0435\u043f\u0435\u0440\u044c \u043c\u0435\u0442\u043e\u0434 \u0432\u043e\u0437\u043e\u0431\u043d\u043e\u0432\u043b\u0435\u043d\u0438\u044f \u043e\u0431\u0449\u0435\u043d\u0438\u044f:<\/p>\n<pre><code class=\"python\">class MCPAgent:    # \u041f\u0440\u0435\u0434\u044b\u0434\u0443\u0449\u0438\u0439 \u043a\u043e\u0434    async def resume(self, user_messages: str, request_id: str | None = None) -&gt; dict[str, Any]:        self._check_graph_available()        trace_id = self._get_trace_id(request_id)        return await self._ainvoke_with_tracing(            data=Command(update={'user_input': user_messages}),            request_id=request_id, trace_id=trace_id, span_name='agent_resume')<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u041e\u0441\u043d\u043e\u0432\u043d\u044b\u0435 \u043e\u0442\u043b\u0438\u0447\u0438\u044f &#8212; \u043d\u043e\u043c\u0435\u0440 \u0442\u0440\u0435\u0439\u0441\u0430 \u0432 Langfuse \u043c\u044b \u0431\u0435\u0440\u0451\u043c \u0438\u0437 \u0441\u0442\u0435\u0439\u0442\u0430 \u0438 \u043d\u0430\u0448\u0435 \u0441\u043e\u043e\u0431\u0449\u0435\u043d\u0438\u0435 \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044f \u043e\u0431\u043e\u0440\u0430\u0447\u0438\u0432\u0430\u0435\u043c \u0432 <code>Command<\/code>. \u041f\u0435\u0440\u0435\u0445\u043e\u0434\u0438\u043c \u043a \u0441\u043b\u0443\u0436\u0435\u0431\u043d\u044b\u043c \u043c\u0435\u0442\u043e\u0434\u0430\u043c.<\/p>\n<p>\u041f\u0435\u0440\u0432\u044b\u0439 \u043c\u0435\u0442\u043e\u0434 \u043f\u0440\u043e\u0432\u0435\u0440\u044f\u0435\u0442 \u043d\u0430\u043b\u0438\u0447\u0438\u0435 \u0430\u0433\u0435\u043d\u0442\u0430:<\/p>\n<pre><code class=\"python\">class MCPAgent:    # \u041f\u0440\u0435\u0434\u044b\u0434\u0443\u0449\u0438\u0439 \u043a\u043e\u0434    def _check_graph_available(self):        if self.graph is None:            raise RuntimeError(                '\u0410\u0433\u0435\u043d\u0442 \u043d\u0435 \u0438\u043d\u0438\u0446\u0438\u0430\u043b\u0438\u0437\u0438\u0440\u043e\u0432\u0430\u043d. \u0417\u0430\u043f\u0443\u0441\u0442\u0438\u0442\u0435 `initialize()`.')<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u0412\u0442\u043e\u0440\u043e\u0439 \u043d\u0443\u0436\u0435\u043d \u0434\u043b\u044f \u0438\u0437\u0432\u043b\u0435\u0447\u0435\u043d\u0438\u044f <code>trace_id<\/code> \u0438\u0437 \u0441\u043e\u0445\u0440\u0430\u043d\u0435\u043d\u043d\u043e\u0433\u043e \u0441\u0442\u0435\u0439\u0442\u0430:<\/p>\n<pre><code class=\"python\">class MCPAgent:    # \u041f\u0440\u0435\u0434\u044b\u0434\u0443\u0449\u0438\u0439 \u043a\u043e\u0434    def _get_trace_id(self, request_id: str) -&gt; str | None:        return self.graph.get_state(            {'configurable': {'thread_id': request_id}}        ).values.get('trace_id', None)<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u041f\u0440\u043e <code>thread_id<\/code> \u043f\u0438\u0441\u0430\u043b \u0432 \u0447\u0430\u0441\u0442\u0438 \u043f\u0440\u043e \u0447\u0435\u043a\u043f\u043e\u0438\u043d\u0442\u0435\u0440. \u041e\u043d \u043d\u0443\u0436\u0435\u043d \u0434\u043b\u044f \u0442\u043e\u0433\u043e, \u0447\u0442\u043e\u0431\u044b \u0433\u0440\u0430\u0444 \u0437\u043d\u0430\u043b, \u0441\u0442\u0435\u0439\u0442 \u043a\u0430\u043a\u043e\u0439 \u0438\u043c\u0435\u043d\u043d\u043e \u0441\u0435\u0441\u0441\u0438\u0438 \u043d\u0430\u0434\u043e \u0438\u0437\u0432\u043b\u0435\u043a\u0430\u0442\u044c \u0438\u0437 \u043f\u0430\u043c\u044f\u0442\u0438. \u0424\u043e\u0440\u043c\u0438\u0440\u0443\u0435\u0442\u0441\u044f \u043d\u0430 \u0441\u0442\u043e\u0440\u043e\u043d\u0435 \u0432\u0435\u0431-\u0438\u043d\u0442\u0435\u0440\u0444\u0435\u0439\u0441\u0430. \u0422\u0435\u043f\u0435\u0440\u044c \u043c\u0435\u0442\u043e\u0434 \u0432\u044b\u0437\u043e\u0432\u0430 LLM:<\/p>\n<pre><code class=\"python\">class MCPAgent:    # \u041f\u0440\u0435\u0434\u044b\u0434\u0443\u0449\u0438\u0439 \u043a\u043e\u0434    async def _ainvoke_with_tracing(            self, data: dict[str, Any] | Command, request_id: str, trace_id: str, span_name: str    ) -&gt; dict[str, Any]:        with settings.langfuse.client.start_as_current_observation(                as_type='span',                name=span_name,                trace_context={'trace_id': trace_id},        ) as span:            result = await self.graph.ainvoke(data, config=self._create_config(request_id))            return result['messages']<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u041f\u0435\u0440\u0432\u044b\u043c \u0434\u0435\u043b\u043e\u043c \u0441\u043e\u0437\u0434\u0430\u0451\u043c \u043a\u0430\u0441\u0442\u043e\u043c\u043d\u043e\u0435 \u043d\u0430\u0431\u043b\u044e\u0434\u0435\u043d\u0438\u0435 \u0442\u0438\u043f\u0430 \u0441\u043f\u0430\u043d \u0438 \u043f\u0435\u0440\u0435\u0434\u0430\u0451\u043c \u0432 \u043d\u0435\u0433\u043e \u043f\u0430\u0440\u0430\u043c\u0435\u0442\u0440\u043e\u043c \u043d\u0430\u0448 <code>trace_id<\/code>. Langfuse \u0441\u0430\u043c \u0441\u0433\u0440\u0443\u043f\u043f\u0438\u0440\u0443\u0435\u0442 \u0432\u0441\u0435 \u043d\u0430\u0431\u043b\u044e\u0434\u0435\u043d\u0438\u044f \u043f\u043e <code>trace_id<\/code>. \u0414\u0430\u043b\u0435\u0435 \u0432\u044b\u0437\u044b\u0432\u0430\u0435\u043c \u043c\u0435\u0442\u043e\u0434 <code>ainvoke<\/code> \u0433\u0440\u0430\u0444\u0430, \u043f\u0435\u0440\u0435\u0434\u0430\u0432 \u0432 \u043d\u0435\u0433\u043e \u043a\u043e\u043d\u0444\u0438\u0433:<\/p>\n<pre><code class=\"python\">class MCPAgent:    # \u041f\u0440\u0435\u0434\u044b\u0434\u0443\u0449\u0438\u0439 \u043a\u043e\u0434    def _create_config(self, request_id: str) -&gt; dict[str, Any]:        return {            'callbacks': [self.lf_handler],            'metadata': {                'langfuse_session_id': f'docker_session_{self.init_time}',            },            'configurable': {'thread_id': request_id}        }<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u0412 \u043a\u043e\u043d\u0444\u0438\u0433\u0435 \u043c\u044b \u043f\u0440\u043e\u043f\u0438\u0441\u044b\u0432\u0430\u0435\u043c Langfuse CallbackHandler \u0434\u043b\u044f \u043e\u0440\u0433\u0430\u043d\u0438\u0437\u0430\u0446\u0438\u0438 \u043d\u0430\u0431\u043b\u044e\u0434\u0435\u043d\u0438\u044f, \u0432 \u043c\u0435\u0442\u0430\u0434\u0430\u043d\u043d\u044b\u0445 \u043f\u0435\u0440\u0435\u0434\u0430\u0451\u043c \u043f\u0430\u0440\u0430\u043c\u0435\u0442\u0440 <code>langfuse_session_id<\/code>, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u0432 \u0434\u0430\u043b\u044c\u043d\u0435\u0439\u0448\u0435\u043c \u043c\u043e\u0436\u043d\u043e \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u044c \u0434\u043b\u044f \u0444\u0438\u043b\u044c\u0442\u0440\u0430\u0446\u0438\u0438 \u0442\u0440\u0435\u0439\u0441\u043e\u0432 \u0438 <code>thread_id<\/code> \u0434\u043b\u044f \u0441\u043e\u0445\u0440\u0430\u043d\u0435\u043d\u0438\u044f \u0441\u0442\u0435\u0439\u0442\u0430 \u0432 \u0447\u0435\u043a\u043f\u043e\u0438\u043d\u0442\u0435\u0440\u0435.<\/p>\n<p>\u0413\u0440\u0430\u0444 \u0433\u043e\u0442\u043e\u0432. \u041e\u0441\u0442\u0430\u043b\u043e\u0441\u044c \u0434\u043e\u0440\u0430\u0431\u043e\u0442\u0430\u0442\u044c Gradio \u0438\u043d\u0442\u0435\u0440\u0444\u0435\u0439\u0441 \u043f\u043e\u0434 \u043d\u043e\u0432\u044b\u0435 \u0444\u0443\u043d\u043a\u0446\u0438\u0438 \u0438 \u043c\u043e\u0436\u043d\u043e \u0440\u0435\u043b\u0438\u0437\u0438\u0442\u044c)<\/p>\n<h2>\u0418\u043d\u0442\u0435\u0440\u0444\u0435\u0439\u0441 \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044f<\/h2>\n<p>\u041f\u0440\u0438 \u0438\u0437\u043c\u0435\u043d\u0435\u043d\u0438\u0438 \u0430\u0440\u0445\u0438\u0442\u0435\u043a\u0442\u0443\u0440\u044b \u0430\u0433\u0435\u043d\u0442\u0430 (\u0434\u043e\u0431\u0430\u0432\u043b\u0435\u043d\u0438\u0435 \u043f\u043e\u0434\u0442\u0432\u0435\u0440\u0436\u0434\u0435\u043d\u0438\u044f \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u0435\u043c) \u044f \u043f\u043e\u0448\u0435\u043b \u043d\u0430 \u043e\u0434\u0438\u043d \u043a\u043e\u043c\u043f\u0440\u043e\u043c\u0438\u0441 &#8212; \u043f\u043e\u0434\u0442\u0432\u0435\u0440\u0436\u0434\u0435\u043d\u0438\u0435 \u043f\u0440\u043e\u0441\u0442\u044b\u043c \u0441\u043b\u043e\u0432\u043e\u043c \u201c\u041f\u0440\u043e\u0434\u043e\u043b\u0436\u0438\u0442\u044c\u201d. \u0418 \u0447\u0442\u043e\u0431\u044b \u043a\u0430\u0436\u0434\u044b\u0439 \u0440\u0430\u0437 \u0435\u0433\u043e \u043d\u0435 \u043f\u0438\u0441\u0430\u0442\u044c, \u044f \u0440\u0435\u0448\u0438\u043b \u0434\u043e\u0431\u0430\u0432\u0438\u0442\u044c \u0432 \u0438\u043d\u0442\u0435\u0440\u0444\u0435\u0439\u0441 \u043a\u043d\u043e\u043f\u043a\u0443 \u201c\u041f\u0440\u043e\u0434\u043e\u043b\u0436\u0438\u0442\u044c\u201d. \u0414\u043b\u044f \u044d\u0442\u043e\u0433\u043e \u043f\u0440\u0438\u0448\u043b\u043e\u0441\u044c \u043e\u0442\u043a\u0430\u0437\u0430\u0442\u044c\u0441\u044f \u043e\u0442 <code>gr.ChatInterface<\/code> \u0438 \u043f\u0435\u0440\u0435\u043f\u0438\u0441\u0430\u0442\u044c \u0438\u043d\u0442\u0435\u0440\u0444\u0435\u0439\u0441 \u043d\u0430 <code>gr.Chatbot<\/code>:<\/p>\n<pre><code class=\"python\">class MCPCodingAgentApp:    def build_interface(self):        with gr.Blocks(title='MCP Coding Agent', fill_height=True) as self.demo:            gr.Markdown('# MCP Coding Agent')            gr.Markdown('\u041f\u043e\u043c\u043e\u0449\u043d\u0438\u043a \u0440\u0430\u0437\u0440\u0430\u0431\u043e\u0442\u0447\u0438\u043a\u0430 \u0441 \u0434\u043e\u0441\u0442\u0443\u043f\u043e\u043c \u043a \u0444\u0430\u0439\u043b\u0430\u043c, Git \u0438 \u0434\u043e\u043a\u0443\u043c\u0435\u043d\u0442\u0430\u0446\u0438\u0438')            chatbot = gr.Chatbot(label='\u0427\u0430\u0442 \u0441 \u0430\u0433\u0435\u043d\u0442\u043e\u043c', height=700)            request_id_state = gr.State('')            with gr.Row():                msg = gr.Textbox(                    label='\u0412\u0430\u0448\u0435 \u0441\u043e\u043e\u0431\u0449\u0435\u043d\u0438\u0435',                    placeholder='\u0412\u0432\u0435\u0434\u0438\u0442\u0435 \u0441\u043e\u043e\u0431\u0449\u0435\u043d\u0438\u0435 \u0438\u043b\u0438 \u043d\u0430\u0436\u043c\u0438\u0442\u0435 \"\u041f\u0440\u043e\u0434\u043e\u043b\u0436\u0438\u0442\u044c\"',                    scale=8,                    container=False                )                submit_btn = gr.Button('\u041e\u0442\u043f\u0440\u0430\u0432\u0438\u0442\u044c', variant='primary')                continue_btn = gr.Button('\u25b6 \u041f\u0440\u043e\u0434\u043e\u043b\u0436\u0438\u0442\u044c', variant='secondary')            submit_btn.click(                fn=self._respond,                inputs=[msg, chatbot, request_id_state],                outputs=[chatbot, msg, request_id_state])            msg.submit(                fn=self._respond,                inputs=[msg, chatbot, request_id_state],                outputs=[chatbot, msg, request_id_state])            continue_btn.click(                fn=self._continue,                inputs=[chatbot, request_id_state],                outputs=[chatbot, request_id_state])<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u041f\u043e\u0441\u043b\u0435 \u043f\u043e\u043b\u044f \u0447\u0430\u0442\u0430 \u0441\u043e\u0437\u0434\u0430\u0451\u043c \u0441\u0442\u0440\u043e\u043a\u0443 \u0441 \u043f\u043e\u043b\u0435\u043c \u0434\u043b\u044f \u0432\u0432\u043e\u0434\u0430 \u0441\u043e\u043e\u0431\u0449\u0435\u043d\u0438\u044f \u0438 \u0434\u0432\u0443\u043c\u044f \u043a\u043d\u043e\u043f\u043a\u0430\u043c\u0438 &#8212; \u201c\u041e\u0442\u043f\u0440\u0430\u0432\u0438\u0442\u044c\u201d \u0438 \u201c\u041f\u0440\u043e\u0434\u043e\u043b\u0436\u0438\u0442\u044c\u201d. \u0414\u0430\u043b\u0435\u0435 \u043d\u0430\u0437\u043d\u0430\u0447\u0430\u0435\u043c \u0444\u0443\u043d\u043a\u0446\u0438\u0438-\u043e\u0431\u0440\u0430\u0431\u043e\u0442\u0447\u0438\u043a\u0438 \u0434\u043b\u044f \u043d\u0430\u0448\u0438\u0445 \u044d\u043b\u0435\u043c\u0435\u043d\u0442\u043e\u0432 (\u0434\u043b\u044f \u043f\u043e\u043b\u044f \u0432\u0432\u043e\u0434\u0430 \u0442\u043e\u0436\u0435, \u0447\u0442\u043e\u0431\u044b \u0431\u044b\u043b\u0430 \u043e\u0442\u043f\u0440\u0430\u0432\u043a\u0430 \u043f\u043e \u043d\u0430\u0436\u0430\u0442\u0438\u044e Enter). \u041f\u0430\u0440\u0430\u043c\u0435\u0442\u0440\u044b <code>inputs<\/code> \u0438 <code>outputs<\/code> \u0441\u0432\u044f\u0437\u044b\u0432\u0430\u044e\u0442 \u0432\u0445\u043e\u0434\u044b-\u0432\u044b\u0445\u043e\u0434\u044b \u0444\u0443\u043d\u043a\u0446\u0438\u0438-\u043e\u0431\u0440\u0430\u0431\u043e\u0442\u0447\u0438\u043a\u0430 \u0441 \u043e\u0431\u044a\u0435\u043a\u0442\u0430\u043c\u0438 Gradio.<\/p>\n<p>\u041e\u0431\u044a\u0435\u043a\u0442 <code>gr.State('')<\/code> \u043d\u0443\u0436\u0435\u043d \u0434\u043b\u044f \u0445\u0440\u0430\u043d\u0435\u043d\u0438\u044f request_id \u0432 \u0440\u0430\u043c\u043a\u0430\u0445 \u0441\u0435\u0441\u0441\u0438\u0438.<\/p>\n<p>\u0422\u0430\u043a \u043a\u0430\u043a \u0447\u0430\u0442 \u0443 \u043d\u0430\u0441 \u0442\u0435\u043f\u0435\u0440\u044c \u0441\u0430\u043c\u043e\u0434\u0435\u043b\u044c\u043d\u044b\u0439, \u0442\u043e \u0438 \u0443\u043f\u0440\u0430\u0432\u043b\u044f\u0442\u044c \u0432\u0441\u0435\u043c \u0432 \u043e\u0431\u0440\u0430\u0431\u043e\u0442\u0447\u0438\u043a\u0430\u0445 \u043c\u044b \u0434\u043e\u043b\u0436\u043d\u044b \u0440\u0443\u043a\u0430\u043c\u0438:<\/p>\n<pre><code class=\"python\">class MCPCodingAgentApp:    # \u041f\u0440\u0435\u0434\u044b\u0434\u0443\u0449\u0438\u0439 \u043a\u043e\u0434    async def _respond(self, message: str, history: list, request_id: str):        if not request_id:            request_id = str(uuid.uuid4())        if not message or not message.strip():            return history, '', request_id        phase = self.agent.get_phase(request_id)        if phase is None or phase == 'done':            result = await self.agent.run(user_messages=message, request_id=request_id)        else:            result = await self.agent.resume(user_messages=message, request_id=request_id)        history.append({'role': 'user', 'content': message})        agent_response = result[-1].content if result else '\u041d\u0435\u0442 \u043e\u0442\u0432\u0435\u0442\u0430'        history.append({'role': 'assistant', 'content': agent_response})        return history, '', request_id<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u0412 \u043e\u0431\u0440\u0430\u0431\u043e\u0442\u0447\u0438\u043a\u0435 \u043a\u043d\u043e\u043f\u043a\u0438 \u201c\u041e\u0442\u043f\u0440\u0430\u0432\u0438\u0442\u044c\u201d \u0438 \u043d\u0430\u0436\u0430\u0442\u0438\u044f Enter \u043c\u044b \u0434\u043e\u043b\u0436\u043d\u044b \u043f\u0440\u0435\u0434\u0443\u0441\u043c\u043e\u0442\u0440\u0435\u0442\u044c, \u0447\u0442\u043e \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044c \u0440\u0435\u0448\u0438\u043b \u043d\u0430\u043f\u0435\u0447\u0430\u0442\u0430\u0442\u044c \u0441\u043b\u043e\u0432\u043e \u201c\u041f\u0440\u043e\u0434\u043e\u043b\u0436\u0438\u0442\u044c\u201d \u0432\u043c\u0435\u0441\u0442\u043e \u043d\u0430\u0436\u0430\u0442\u0438\u044f \u043e\u0442\u0434\u0435\u043b\u044c\u043d\u043e\u0439 \u043a\u043d\u043e\u043f\u043a\u0438. \u0414\u043b\u044f \u044d\u0442\u043e\u0433\u043e \u043c\u044b \u0430\u043d\u0430\u043b\u0438\u0437\u0438\u0440\u0443\u0435\u043c \u0444\u0430\u0437\u0443 \u0432\u044b\u043f\u043e\u043b\u043d\u0435\u043d\u0438\u044f \u0430\u0433\u0435\u043d\u0442\u0430 \u0438, \u0432 \u0437\u0430\u0432\u0438\u0441\u0438\u043c\u043e\u0441\u0442\u0438 \u043e\u0442 \u043d\u0435\u0451, \u0432\u044b\u0437\u044b\u0432\u0430\u0435\u043c \u0441\u043e\u043e\u0442\u0432\u0435\u0442\u0441\u0442\u0432\u0443\u044e\u0449\u0438\u0439 \u043c\u0435\u0442\u043e\u0434 \u0430\u0433\u0435\u043d\u0442\u0430. \u0414\u043e\u043f\u043e\u043b\u043d\u0438\u0442\u0435\u043b\u044c\u043d\u043e \u0441\u043e\u0437\u0434\u0430\u0451\u043c \u043f\u0435\u0440\u0432\u0438\u0447\u043d\u044b\u0439 <code>request_id<\/code>, \u0437\u0430\u0449\u0438\u0449\u0430\u0435\u043c\u0441\u044f \u043e\u0442 \u043f\u0443\u0441\u0442\u043e\u0433\u043e \u0441\u043e\u043e\u0431\u0449\u0435\u043d\u0438\u044f (\u043a\u0430\u043a \u0432 \u0437\u0430\u043f\u0440\u043e\u0441\u0435, \u0442\u0430\u043a \u0438 \u0432 \u043e\u0442\u0432\u0435\u0442\u0435) \u0438 \u043d\u0430\u043f\u043e\u043b\u043d\u044f\u0435\u043c \u0438\u0441\u0442\u043e\u0440\u0438\u044e.<\/p>\n<p>\u041e\u0431\u0440\u0430\u0431\u043e\u0442\u0447\u0438\u043a \u043f\u0440\u043e\u0434\u043e\u043b\u0436\u0435\u043d\u0438\u044f \u043e\u0431\u0449\u0435\u043d\u0438\u044f \u043d\u0435\u043c\u043d\u043e\u0433\u043e \u043f\u043e\u043a\u043e\u0440\u043e\u0447\u0435:<\/p>\n<pre><code class=\"python\">class MCPCodingAgentApp:    # \u041f\u0440\u0435\u0434\u044b\u0434\u0443\u0449\u0438\u0439 \u043a\u043e\u0434    async def _continue(self, history: list, request_id: str):        phase = self.agent.get_phase(request_id)        if phase is None or phase == 'done':            history.append({                'role': 'assistant',                'content': '\u041d\u0435\u0442 \u0430\u043a\u0442\u0438\u0432\u043d\u044b\u0445 \u0437\u0430\u0434\u0430\u0447 \u0434\u043b\u044f \u043f\u0440\u043e\u0434\u043e\u043b\u0436\u0435\u043d\u0438\u044f. \u0417\u0430\u0434\u0430\u0439\u0442\u0435 \u043d\u043e\u0432\u044b\u0439 \u0432\u043e\u043f\u0440\u043e\u0441.'            })            return history, request_id        result = await self.agent.resume(user_messages='\u041f\u0440\u043e\u0434\u043e\u043b\u0436\u0438\u0442\u044c', request_id=request_id)        history.append({'role': 'user', 'content': '\u25b6 \u041f\u0440\u043e\u0434\u043e\u043b\u0436\u0438\u0442\u044c'})        agent_response = result[-1].content if result else '\u041d\u0435\u0442 \u043e\u0442\u0432\u0435\u0442\u0430'        history.append({'role': 'assistant', 'content': agent_response})        return history, request_id<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u0423\u0431\u0435\u0436\u0434\u0430\u0435\u043c\u0441\u044f \u043f\u043e \u0444\u0430\u0437\u0435, \u0447\u0442\u043e \u0430\u0433\u0435\u043d\u0442 \u0432 \u043f\u0440\u043e\u0446\u0435\u0441\u0441\u0435 \u0440\u0430\u0431\u043e\u0442\u044b, \u043f\u043e\u0441\u043b\u0435 \u0447\u0435\u0433\u043e \u043d\u0430\u043f\u0440\u0430\u0432\u043b\u044f\u0435\u043c \u0435\u043c\u0443 \u0441\u043e\u043e\u0431\u0449\u0435\u043d\u0438\u0435 \u201c\u041f\u0440\u043e\u0434\u043e\u043b\u0436\u0438\u0442\u044c\u201d, \u0438\u043d\u0430\u0447\u0435 \u0441\u043e\u043e\u0431\u0449\u0430\u0435\u043c \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044e, \u0447\u0442\u043e \u0432\u044b \u043d\u0435 \u0432 \u0446\u0438\u043a\u043b\u0435. \u0422\u0430\u043a\u0436\u0435 \u0441\u043e\u0445\u0440\u0430\u043d\u044f\u0435\u043c \u0438\u0441\u0442\u043e\u0440\u0438\u044e.<\/p>\n<p>\u041a\u043e\u0434 \u0438\u043d\u0438\u0446\u0438\u0430\u043b\u0438\u0437\u0430\u0446\u0438\u0438 \u0438 \u0437\u0430\u043f\u0443\u0441\u043a\u0430 \u043e\u0441\u0442\u0430\u0451\u0442\u0441\u044f \u0431\u0435\u0437 \u0438\u0437\u043c\u0435\u043d\u0435\u043d\u0438\u0439.<\/p>\n<p>\u0421 \u043a\u043e\u0434\u043e\u043c \u0432\u0441\u0451.<\/p>\n<h2>\u041f\u0440\u043e\u0432\u0435\u0440\u043a\u0430 \u0440\u0430\u0431\u043e\u0442\u043e\u0441\u043f\u043e\u0441\u043e\u0431\u043d\u043e\u0441\u0442\u0438<\/h2>\n<p>\u0417\u0430\u043f\u0443\u0441\u043a\u0430\u0435\u043c <code>make up<\/code>.<\/p>\n<blockquote>\n<p>\u0422\u0435\u043f\u0435\u0440\u044c \u043c\u043e\u0436\u043d\u043e \u0440\u0430\u0441\u0441\u043b\u0430\u0431\u0438\u0442\u044c\u0441\u044f \u0438 \u043e\u0442\u043a\u0438\u043d\u0443\u0442\u044c\u0441\u044f \u043d\u0430 \u0441\u043f\u0438\u043d\u043a\u0443 \u043a\u0440\u0435\u0441\u043b\u0430 (\u0435\u0441\u0442\u044c \u0442\u0443\u0442 \u0442\u0435, \u043a\u0442\u043e \u0441\u0442\u0430\u0432\u0438\u043b Windows98?\ud83d\ude04)<\/p>\n<\/blockquote>\n<p>\u0416\u0434\u0451\u043c \u0437\u0430\u043a\u0430\u0447\u043a\u0438 \u043e\u0431\u0440\u0430\u0437\u043e\u0432, \u043c\u043e\u0434\u0435\u043b\u0438. \u0414\u043b\u044f \u0441\u043a\u0430\u0447\u0438\u0432\u0430\u043d\u0438\u044f \u043c\u043e\u0434\u0435\u043b\u0438 \u043c\u043e\u0436\u043d\u043e \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u044c uvx (\u0418\u041c\u0425\u041e \u0432 \u0440\u0430\u0437\u044b \u0431\u044b\u0441\u0442\u0440\u0435\u0435):<\/p>\n<pre><code class=\"bash\">HF_TOKEN=&lt;\u0422\u041e\u041a\u0415\u041d&gt; HF_HOME=&lt;\u041f\u0410\u041f\u041a\u0410 \u0414\u041b\u042f \u041c\u041e\u0414\u0415\u041b\u0418&gt; uvx hf download nvidia\/Qwen3.6-35B-A3B-NVFP4<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u0412 <code>HF_HOME<\/code> \u043d\u0443\u0436\u043d\u043e \u0443\u043a\u0430\u0437\u0430\u0442\u044c \u0442\u0443\u0436\u0435 \u043f\u0430\u043f\u043a\u0443, \u043a\u043e\u0442\u043e\u0440\u0443\u044e \u0443\u043a\u0430\u0437\u044b\u0432\u0430\u0435\u043c \u043a\u0430\u043a \u0442\u043e\u043c \u0434\u043b\u044f vLLM.<\/p>\n<h3>\u041f\u0440\u043e\u0441\u0442\u043e\u0439 \u0432\u043e\u043f\u0440\u043e\u0441<\/h3>\n<p>\u041f\u0440\u043e\u0432\u0435\u0440\u043a\u0443 \u043d\u0430\u0447\u0430\u043b \u0441 \u0437\u0430\u0434\u0430\u043d\u0438\u044f \u043f\u043e\u043f\u0440\u043e\u0449\u0435, \u043d\u043e \u0432 \u043d\u0435\u0441\u043a\u043e\u043b\u044c\u043a\u043e \u0448\u0430\u0433\u043e\u0432:<\/p>\n<blockquote>\n<p>\u041d\u0430\u043f\u0438\u0448\u0438 \u0444\u0443\u043d\u043a\u0446\u0438\u044e \u043f\u0440\u043e\u0432\u0435\u0440\u043a\u0438 \u043d\u0430 \u0447\u0435\u0442\u043d\u043e\u0441\u0442\u044c. \u0421\u043e\u0445\u0440\u0430\u043d\u0438 \u0435\u0451 \u0432 \u0444\u0430\u0439\u043b \u0438 \u0441\u0434\u0435\u043b\u0430\u0439 \u043a\u043e\u043c\u043c\u0438\u0442<\/p>\n<\/blockquote>\n<figure class=\"full-width \"><img decoding=\"async\" src=\"https:\/\/habrastorage.org\/r\/w1560\/getpro\/habr\/upload_files\/37a\/8b3\/f0a\/37a8b3f0a14153613b3ce8c6c327b13e.png\" alt=\"\" title=\"\" width=\"976\" height=\"767\" sizes=\"auto, (max-width: 780px) 100vw, 50vw\" srcset=\"https:\/\/habrastorage.org\/r\/w780\/getpro\/habr\/upload_files\/37a\/8b3\/f0a\/37a8b3f0a14153613b3ce8c6c327b13e.png 780w,&#10;       https:\/\/habrastorage.org\/r\/w1560\/getpro\/habr\/upload_files\/37a\/8b3\/f0a\/37a8b3f0a14153613b3ce8c6c327b13e.png 781w\" loading=\"lazy\" decode=\"async\"\/><\/figure>\n<p>\u041f\u0440\u0438\u0435\u043c\u043b\u0435\u043c\u043e. \u0413\u043b\u044f\u043d\u0435\u043c \u0442\u0440\u0435\u0439\u0441 \u0437\u0430\u0434\u0430\u0447\u0438<\/p>\n<figure class=\"full-width \"><img decoding=\"async\" src=\"https:\/\/habrastorage.org\/r\/w1560\/getpro\/habr\/upload_files\/bf4\/17a\/6ce\/bf417a6ce6f6c3ee96675a0a2a8de268.png\" width=\"1484\" height=\"701\" sizes=\"auto, (max-width: 780px) 100vw, 50vw\" srcset=\"https:\/\/habrastorage.org\/r\/w780\/getpro\/habr\/upload_files\/bf4\/17a\/6ce\/bf417a6ce6f6c3ee96675a0a2a8de268.png 780w,&#10;       https:\/\/habrastorage.org\/r\/w1560\/getpro\/habr\/upload_files\/bf4\/17a\/6ce\/bf417a6ce6f6c3ee96675a0a2a8de268.png 781w\" loading=\"lazy\" decode=\"async\"\/><\/figure>\n<p>\u0412\u0441\u0451 \u043a\u0430\u043a \u043d\u0430\u0434\u043e &#8212; \u0434\u0432\u0430 \u0448\u0430\u0433\u0430 \u0432\u044b\u043f\u043e\u043b\u043d\u0435\u043d\u0438\u044f, \u0442\u0440\u0438 \u0432\u044b\u0437\u043e\u0432\u0430 \u0438\u043d\u0441\u0442\u0440\u0443\u043c\u0435\u043d\u0442\u043e\u0432 (<code>write file<\/code>, <code>git add<\/code>, <code>git commit<\/code>).<\/p>\n<h3>\u0412\u043e\u043f\u0440\u043e\u0441 \u043f\u043e\u0441\u043b\u043e\u0436\u043d\u0435\u0435<\/h3>\n<p>\u0422\u0443\u0442 \u043d\u0430\u0448\u0435\u043b \u0432 \u0438\u043d\u0442\u0435\u0440\u043d\u0435\u0442\u0435 \u043a\u0430\u043a\u0443\u044e-\u0442\u043e \u0437\u0430\u0434\u0430\u0447\u0443 \u043d\u0430 \u0437\u043d\u0430\u043d\u0438\u0435 FastAPI, \u043f\u043e\u043c\u0435\u0447\u0435\u043d\u043d\u0443\u044e \u043a\u0430\u043a \u0441\u0440\u0435\u0434\u043d\u0435\u0439 \u0441\u043b\u043e\u0436\u043d\u043e\u0441\u0442\u0438<\/p>\n<blockquote>\n<p>\u0421\u043e\u0437\u0434\u0430\u0439 \u0430\u0441\u0438\u043d\u0445\u0440\u043e\u043d\u043d\u044b\u0439 dependency get_db, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u043e\u0442\u043a\u0440\u044b\u0432\u0430\u0435\u0442 \u0442\u0440\u0430\u043d\u0437\u0430\u043a\u0446\u0438\u044e.<\/p>\n<p>\u0420\u0435\u0430\u043b\u0438\u0437\u0443\u0439 \u043c\u043e\u0434\u0435\u043b\u044c \u0434\u043b\u044f \u044d\u043d\u0434\u043f\u043e\u0438\u043d\u0442\u0430 User \u0441 id:uuid \u0438 nickname: str<\/p>\n<p>\u0420\u0435\u0430\u043b\u0438\u0437\u0443\u0439 \u044d\u043d\u0434\u043f\u043e\u0438\u043d\u0442 GET \/users\/{user_id}\/posts, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u0432\u043e\u0437\u0432\u0440\u0430\u0449\u0430\u0435\u0442 \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044f \u0438 \u0435\u0433\u043e \u043f\u043e\u0441\u0442\u044b \u0432 \u043e\u0434\u043d\u043e\u043c \u0437\u0430\u043f\u0440\u043e\u0441\u0435 \u043a \u0411\u0414 (\u0431\u0435\u0437 N+1 \u043f\u0440\u043e\u0431\u043b\u0435\u043c\u044b).<\/p>\n<p>\u0421\u043e\u0445\u0440\u0430\u043d\u0438 \u0440\u0435\u0437\u0443\u043b\u044c\u0442\u0430\u0442\u044b \u0432 \u043e\u0442\u0434\u0435\u043b\u044c\u043d\u044b\u0435 \u0444\u0430\u0439\u043b\u044b<\/p>\n<p>\u0421\u0434\u0435\u043b\u0430\u0439 \u043a\u043e\u043c\u043c\u0438\u0442 \u0438\u0437\u043c\u0435\u043d\u0435\u043d\u0438\u0439<\/p>\n<\/blockquote>\n<p>\u041d\u0430 \u0437\u0430\u043f\u0440\u043e\u0441 \u043f\u043e\u043b\u0443\u0447\u0438\u043b \u0441\u043b\u0435\u0434\u0443\u044e\u0449\u0438\u0439 \u043f\u043b\u0430\u043d \u0434\u0435\u0439\u0441\u0442\u0432\u0438\u0439:<\/p>\n<figure class=\"full-width \"><img decoding=\"async\" src=\"https:\/\/habrastorage.org\/r\/w1560\/getpro\/habr\/upload_files\/030\/f57\/681\/030f57681b70c42c306a7f40f7b73e0d.png\" width=\"985\" height=\"862\" sizes=\"auto, (max-width: 780px) 100vw, 50vw\" srcset=\"https:\/\/habrastorage.org\/r\/w780\/getpro\/habr\/upload_files\/030\/f57\/681\/030f57681b70c42c306a7f40f7b73e0d.png 780w,&#10;       https:\/\/habrastorage.org\/r\/w1560\/getpro\/habr\/upload_files\/030\/f57\/681\/030f57681b70c42c306a7f40f7b73e0d.png 781w\" loading=\"lazy\" decode=\"async\"\/><\/figure>\n<p>\u0413\u043b\u044f\u043d\u0435\u043c \u0441\u0445\u0435\u043c\u0443 \u0442\u0440\u0435\u0439\u0441\u0430<\/p>\n<figure class=\"full-width \"><img decoding=\"async\" src=\"https:\/\/habrastorage.org\/r\/w1560\/getpro\/habr\/upload_files\/fb2\/1de\/773\/fb21de7733b34881d8b75f9e8f84ac7f.png\" width=\"766\" height=\"572\" sizes=\"auto, (max-width: 780px) 100vw, 50vw\" srcset=\"https:\/\/habrastorage.org\/r\/w780\/getpro\/habr\/upload_files\/fb2\/1de\/773\/fb21de7733b34881d8b75f9e8f84ac7f.png 780w,&#10;       https:\/\/habrastorage.org\/r\/w1560\/getpro\/habr\/upload_files\/fb2\/1de\/773\/fb21de7733b34881d8b75f9e8f84ac7f.png 781w\" loading=\"lazy\" decode=\"async\"\/><\/figure>\n<p>\u041a\u0440\u0430\u0441\u043e\u0442\u0430) \u041f\u0440\u0438 \u0432\u044b\u043f\u043e\u043b\u043d\u0435\u043d\u0438\u0438 \u0430\u0433\u0435\u043d\u0442 \u0434\u0430\u0436\u0435 \u0432\u044b\u0437\u044b\u0432\u0430\u043b \u0438\u043d\u0441\u0442\u0440\u0443\u043c\u0435\u043d\u0442\u044b \u0434\u043b\u044f \u043f\u0440\u043e\u0441\u043c\u043e\u0442\u0440\u0430 \u0441\u043e\u0434\u0435\u0440\u0436\u0438\u043c\u043e\u0433\u043e \u0434\u0438\u0440\u0435\u043a\u0442\u043e\u0440\u0438\u0438 \u0432 \u043f\u043e\u0438\u0441\u043a\u0430\u0445 \u0444\u0430\u0439\u043b\u0430 <code>model.py<\/code>. \u041d\u0443 \u0438 \u0440\u0435\u0437\u0443\u043b\u044c\u0442\u0430\u0442 \u0440\u0430\u0431\u043e\u0442\u044b \u0430\u0433\u0435\u043d\u0442\u0430 \u0432 \u0440\u0430\u0431\u043e\u0447\u0435\u0439 \u0434\u0438\u0440\u0435\u043a\u0442\u043e\u0440\u0438\u0438:<\/p>\n<figure class=\"full-width \"><img decoding=\"async\" src=\"https:\/\/habrastorage.org\/r\/w1560\/getpro\/habr\/upload_files\/736\/1b9\/85d\/7361b985deaacb4069ba2b10645a6f30.png\" width=\"881\" height=\"492\" sizes=\"auto, (max-width: 780px) 100vw, 50vw\" srcset=\"https:\/\/habrastorage.org\/r\/w780\/getpro\/habr\/upload_files\/736\/1b9\/85d\/7361b985deaacb4069ba2b10645a6f30.png 780w,&#10;       https:\/\/habrastorage.org\/r\/w1560\/getpro\/habr\/upload_files\/736\/1b9\/85d\/7361b985deaacb4069ba2b10645a6f30.png 781w\" loading=\"lazy\" decode=\"async\"\/><\/figure>\n<h2>\u0418\u0442\u043e\u0433\u0438<\/h2>\n<p>\u0424\u0438\u043d\u0430\u043b\u044c\u043d\u044b\u043c \u044d\u043a\u0441\u043f\u0435\u0440\u0438\u043c\u0435\u043d\u0442\u043e\u043c \u044f \u0440\u0435\u0448\u0438\u043b \u043f\u0440\u043e\u0432\u0435\u0440\u0438\u0442\u044c, \u0441\u043c\u043e\u0436\u0435\u0442 \u043b\u0438 \u0430\u0433\u0435\u043d\u0442 \u043d\u0430\u043f\u0438\u0441\u0430\u0442\u044c \u0434\u043e\u043a\u0443\u043c\u0435\u043d\u0442\u0430\u0446\u0438\u044e \u0434\u043b\u044f \u0441\u0430\u043c\u043e\u0433\u043e \u0441\u0435\u0431\u044f. \u0414\u0430\u043b \u0435\u043c\u0443 \u0434\u043e\u0441\u0442\u0443\u043f \u043a \u0441\u043e\u0431\u0441\u0442\u0432\u0435\u043d\u043d\u043e\u0439 \u043f\u0430\u043f\u043a\u0435 \u0438 \u043f\u043e\u043f\u0440\u043e\u0441\u0438\u043b:<\/p>\n<blockquote>\n<p>\u041f\u0440\u043e\u0430\u043d\u0430\u043b\u0438\u0437\u0438\u0440\u0443\u0439 \u0444\u0430\u0439\u043b\u044b \u0432 \u0440\u0430\u0431\u043e\u0447\u0435\u0439 \u043f\u0430\u043f\u043a\u0435. \u041d\u0435\u043e\u0431\u0445\u043e\u0434\u0438\u043c\u043e \u0441\u043e\u0441\u0442\u0430\u0432\u0438\u0442\u044c \u0444\u0430\u0439\u043b readme.md. \u0424\u0430\u0439\u043b \u0434\u043e\u043b\u0436\u0435\u043d \u0441\u043e\u0434\u0435\u0440\u0436\u0430\u0442\u044c \u043a\u0440\u0430\u0442\u043a\u043e\u0435 \u043e\u043f\u0438\u0441\u0430\u043d\u0438\u0435 \u043f\u0440\u043e\u0435\u043a\u0442\u0430 \u0438 \u0441\u043f\u043e\u0441\u043e\u0431 \u0437\u0430\u043f\u0443\u0441\u043a\u0430<\/p>\n<\/blockquote>\n<p>\u041f\u043e\u043b\u0443\u0447\u0438\u043b\u043e\u0441\u044c \u043d\u0435\u043e\u0436\u0438\u0434\u0430\u043d\u043d\u043e \u0445\u043e\u0440\u043e\u0448\u043e &#8212; \u0430\u0433\u0435\u043d\u0442 \u043e\u043f\u0438\u0441\u0430\u043b \u0441\u0442\u0440\u0443\u043a\u0442\u0443\u0440\u0443 \u043f\u0440\u043e\u0435\u043a\u0442\u0430, \u043f\u0435\u0440\u0435\u0447\u0438\u0441\u043b\u0438\u043b \u043e\u0441\u043d\u043e\u0432\u043d\u044b\u0435 \u043a\u043e\u043c\u043f\u043e\u043d\u0435\u043d\u0442\u044b \u0438 \u0434\u043e\u0431\u0430\u0432\u0438\u043b \u043a\u043e\u043c\u0430\u043d\u0434\u044b \u0434\u043b\u044f \u0437\u0430\u043f\u0443\u0441\u043a\u0430 \u0447\u0435\u0440\u0435\u0437 Docker. \u0420\u0435\u0437\u0443\u043b\u044c\u0442\u0430\u0442 \u043c\u043e\u0436\u043d\u043e \u043f\u043e\u0441\u043c\u043e\u0442\u0440\u0435\u0442\u044c \u0432 \u0440\u0435\u043f\u043e\u0437\u0438\u0442\u043e\u0440\u0438\u0438: <a href=\"https:\/\/github.com\/Golden-Gekko\/MCP_Agent\/blob\/master\/README.md\" rel=\"noopener noreferrer nofollow\">\u0441\u0441\u044b\u043b\u043a\u0430 \u043d\u0430 <\/a><a href=\"http:\/\/README.md\" rel=\"noopener noreferrer nofollow\">README.md<\/a>. \u0420\u0443\u0447\u043d\u043e\u0433\u043e \u0432 \u0444\u0430\u0439\u043b\u0435 \u0442\u043e\u043b\u044c\u043a\u043e \u043f\u0440\u043e \u043f\u0430\u0440\u0430\u043c\u0435\u0442\u0440 <code>WS<\/code>.<\/p>\n<p>\u041a\u043e\u0434 \u043f\u0440\u043e\u0435\u043a\u0442\u0430 \u0434\u043e\u0441\u0442\u0443\u043f\u0435\u043d <a href=\"https:\/\/github.com\/Golden-Gekko\/MCP_Agent\" rel=\"noopener noreferrer nofollow\"><strong>\u0442\u0443\u0442<\/strong><\/a><\/p>\n<p>\u041f\u0435\u0440\u0432\u0430\u044f \u0447\u0430\u0441\u0442\u044c \u0441 \u0431\u0430\u0437\u043e\u0439 \u0438 \u043f\u0440\u043e\u0441\u0442\u044b\u043c ReAct \u0430\u0433\u0435\u043d\u0442\u043e\u043c \u0441 MCP \u0438\u043d\u0441\u0442\u0440\u0443\u043c\u0435\u043d\u0442\u0430\u043c\u0438 <a href=\"https:\/\/habr.com\/ru\/articles\/1043348\/\" rel=\"noopener noreferrer nofollow\"><strong>\u0442\u0443\u0442<\/strong><\/a><\/p>\n<p>\u0412\u0442\u043e\u0440\u0430\u044f \u0447\u0430\u0441\u0442\u044c \u0432 \u043a\u043e\u0442\u043e\u0440\u043e\u0439 \u043f\u043e\u043a\u0430\u0437\u0430\u043d\u0430 \u0440\u0435\u0430\u043b\u0438\u0437\u0430\u0446\u0438\u044f \u0443\u0437\u043b\u043e\u0432 <a href=\"https:\/\/habr.com\/ru\/articles\/1049772\/\" rel=\"noopener noreferrer nofollow\"><strong>\u0442\u0443\u0442<\/strong><\/a><\/p>\n<\/div>\n<p>\u0441\u0441\u044b\u043b\u043a\u0430 \u043d\u0430 \u043e\u0440\u0438\u0433\u0438\u043d\u0430\u043b \u0441\u0442\u0430\u0442\u044c\u0438 <a href=\"https:\/\/habr.com\/ru\/articles\/1049774\/\">https:\/\/habr.com\/ru\/articles\/1049774\/<\/a><\/p>\n","protected":false},"excerpt":{"rendered":"<p>\u041f\u0440\u0438\u0432\u0435\u0442, \u0425\u0430\u0431\u0440! \u041c\u0435\u043d\u044f \u0437\u043e\u0432\u0443\u0442 \u0412\u043b\u0430\u0434\u0438\u043c\u0438\u0440 \u0438 \u044d\u0442\u043e \u043f\u0440\u043e\u0434\u043e\u043b\u0436\u0435\u043d\u0438\u0435 \u0441\u0442\u0430\u0442\u044c\u0438 \u043f\u0440\u043e \u0440\u0430\u0437\u0440\u0430\u0431\u043e\u0442\u043a\u0443 \u043b\u043e\u043a\u0430\u043b\u044c\u043d\u043e\u0433\u043e \u043a\u043e\u0434\u0435\u0440-\u0430\u0433\u0435\u043d\u0442\u0430.\u0412 \u043f\u0435\u0440\u0432\u043e\u0439 \u0447\u0430\u0441\u0442\u0438 \u043c\u044b \u0441\u043e\u0437\u0434\u0430\u043b\u0438 \u0438\u043d\u0444\u0440\u0430\u0441\u0442\u0440\u0443\u043a\u0442\u0443\u0440\u0443: \u043f\u043e\u0434\u043d\u044f\u043b\u0438 \u043a\u043e\u043d\u0442\u0435\u0439\u043d\u0435\u0440 \u0441 \u043c\u043e\u0434\u0435\u043b\u044c\u044e, \u043d\u0430\u0441\u0442\u0440\u043e\u0438\u043b\u0438 Langfuse \u0434\u043b\u044f \u0442\u0440\u0430\u0441\u0441\u0438\u0440\u043e\u0432\u043a\u0438 \u0438 \u043d\u0430\u043f\u0438\u0441\u0430\u043b\u0438 \u043f\u0440\u043e\u0441\u0442\u0435\u0439\u0448\u0435\u0433\u043e \u0430\u0433\u0435\u043d\u0442\u0430 \u0441 \u0434\u043e\u0441\u0442\u0443\u043f\u043e\u043c \u043a MCP-\u0438\u043d\u0441\u0442\u0440\u0443\u043c\u0435\u043d\u0442\u0430\u043c.\u0412\u043e \u0432\u0442\u043e\u0440\u043e\u0439 \u0447\u0430\u0441\u0442\u0438 \u043c\u044b \u0434\u043e\u0431\u0430\u0432\u0438\u043b\u0438 \u0430\u0433\u0435\u043d\u0442\u0443 \u043c\u043e\u0437\u0433\u0438: \u043f\u043b\u0430\u043d\u0438\u0440\u043e\u0432\u0449\u0438\u043a, \u043e\u0446\u0435\u043d\u0449\u0438\u043a\u0438, \u0437\u0430\u0449\u0438\u0442\u0443 \u043e\u0442 \u0437\u0430\u0446\u0438\u043a\u043b\u0438\u0432\u0430\u043d\u0438\u044f \u0438 \u0441\u0443\u043c\u043c\u0430\u0440\u0438\u0437\u0430\u0446\u0438\u044e \u043a\u043e\u043d\u0442\u0435\u043a\u0441\u0442\u0430.\u0421\u0435\u0433\u043e\u0434\u043d\u044f \u043c\u044b \u043f\u0440\u0435\u0432\u0440\u0430\u0442\u0438\u043c \u044d\u0442\u043e\u0442 \u043d\u0430\u0431\u043e\u0440 \u0432 \u0440\u0430\u0431\u043e\u0442\u0430\u044e\u0449\u0438\u0439 \u0433\u0440\u0430\u0444. \u0414\u043e\u0431\u0430\u0432\u0438\u043c \u0447\u0435\u043a\u043f\u043e\u0438\u043d\u0442\u0435\u0440\u044b, \u043f\u0440\u0435\u0440\u044b\u0432\u0430\u043d\u0438\u044f \u0438 \u0434\u043e\u0440\u0430\u0431\u043e\u0442\u0430\u0435\u043c \u0438\u043d\u0442\u0435\u0440\u0444\u0435\u0439\u0441 \u043f\u043e\u0434 \u043e\u0431\u043d\u043e\u0432\u043b\u0451\u043d\u043d\u0443\u044e \u0430\u0440\u0445\u0438\u0442\u0435\u043a\u0442\u0443\u0440\u0443.\u0421\u0431\u043e\u0440\u043a\u0430 \u0433\u0440\u0430\u0444\u0430\u0414\u043e\u0440\u0430\u0431\u043e\u0442\u043a\u0443 \u0433\u0440\u0430\u0444\u0430 \u043d\u0430\u0447\u043d\u0435\u043c \u0441 \u043a\u043e\u043d\u0441\u0442\u0440\u0443\u043a\u0442\u043e\u0440\u0430 &#8212; \u0432 \u043d\u0435\u0433\u043e \u043d\u0435\u043e\u0431\u0445\u043e\u0434\u0438\u043c\u043e \u0434\u043e\u0431\u0430\u0432\u0438\u0442\u044c \u0447\u0435\u043a\u043f\u043e\u0438\u043d\u0442\u0435\u0440. \u0414\u043e\u043f\u043e\u043b\u043d\u0438\u0442\u0435\u043b\u044c\u043d\u043e, \u0434\u043b\u044f \u0443\u043f\u043e\u0440\u044f\u0434\u043e\u0447\u0438\u0432\u0430\u043d\u0438\u044f \u0442\u0440\u0430\u0441\u0441\u0438\u0440\u043e\u0432\u043e\u043a \u0434\u043e\u0431\u0430\u0432\u0438\u043c session_id \u043a LangFuse. \u0415\u0433\u043e \u0431\u0443\u0434\u0435\u043c \u0444\u043e\u0440\u043c\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u043d\u0430 \u043e\u0441\u043d\u043e\u0432\u0435 \u0442\u0435\u043a\u0443\u0449\u0435\u0439 \u0434\u0430\u0442\u044b \u0438 \u0432\u0440\u0435\u043c\u0435\u043d\u0438, \u0434\u043b\u044f \u0447\u0435\u0433\u043e \u0432 \u043a\u043e\u043d\u0441\u0442\u0440\u0443\u043a\u0442\u043e\u0440 \u0434\u043e\u0431\u0430\u0432\u0438\u043c \u0432\u0440\u0435\u043c\u0435\u043d\u043d\u0443\u044e \u043c\u0435\u0442\u043a\u0443 \u0441\u043e\u0437\u0434\u0430\u043d\u0438\u044f \u044d\u043a\u0437\u0435\u043c\u043f\u043b\u044f\u0440\u0430 \u0430\u0433\u0435\u043d\u0442\u0430:class MCPAgent:    def __init__(self):        self.llm_with_tools: ChatOpenAI | None = None        self.tools: list[BaseTool] | None = None        self.graph = None        self.checkpointer = InMemorySaver()        self.lf_handler = CallbackHandler(public_key=settings.langfuse.public_key)        self.init_time = datetime.now().strftime(&#8216;%Y%m%d_%H%M%S&#8217;)\u0418\u043d\u0438\u0446\u0438\u0430\u043b\u0438\u0437\u0430\u0446\u0438\u044f \u0433\u0440\u0430\u0444\u0430 \u043d\u0435 \u043c\u0435\u043d\u044f\u043b\u0430\u0441\u044c &#8212; \u0438\u043d\u0438\u0446\u0438\u0430\u043b\u0438\u0437\u0438\u0440\u0443\u0435\u043c \u0438\u043d\u0441\u0442\u0440\u0443\u043c\u0435\u043d\u0442\u044b, \u0431\u0438\u043d\u0434\u0438\u043c \u043d\u0430 LLM \u0438 \u0441\u043e\u0431\u0438\u0440\u0430\u0435\u043c \u0433\u0440\u0430\u0444 (\u043a\u043e\u0434 \u0434\u0430\u043b\u0435\u0435)class MCPAgent:    # \u041f\u0440\u0435\u0434\u044b\u0434\u0443\u0449\u0438\u0439 \u043a\u043e\u0434    async def init_graph(self):        try:            self.tools = await init_tools()        except Exception as e:            logger.error(f&#8217;\u041e\u0448\u0438\u0431\u043a\u0430 \u0438\u043d\u0438\u0446\u0438\u0430\u043b\u0438\u0437\u0430\u0446\u0438\u0438 \u0438\u043d\u0441\u0442\u0440\u0443\u043c\u0435\u043d\u0442\u043e\u0432: {e}&#8217;)            raise        self.llm_with_tools = settings.llm.chat.llm.bind_tools(self.tools, parallel_tool_calls=False)        self.graph = self._compile_graph()\u041f\u0435\u0440\u0435\u0439\u0434\u0451\u043c \u043a \u0443\u0441\u043b\u043e\u0432\u043d\u044b\u043c \u0440\u0451\u0431\u0440\u0430\u043c. \u041d\u0430\u0447\u043d\u0451\u043c \u0441 \u0440\u0451\u0431\u0435\u0440 \u043c\u0430\u0440\u0448\u0440\u0443\u0442\u0438\u0440\u0438\u0437\u0430\u0446\u0438\u0438 \u043f\u043e\u0434\u0442\u0432\u0435\u0440\u0436\u0434\u0435\u043d\u0438\u044f \u043f\u043b\u0430\u043d\u0430 \u0438 \u0432\u044b\u043f\u043e\u043b\u043d\u0435\u043d\u0438\u044f \u0448\u0430\u0433\u0430. \u0412 \u043d\u0438\u0445 \u043f\u0440\u043e\u0432\u0435\u0440\u044f\u0435\u043c \u0444\u043b\u0430\u0433 is_approved. \u0415\u0441\u043b\u0438 \u043f\u043e\u0434\u0442\u0432\u0435\u0440\u0436\u0434\u0435\u043d\u0438\u0435 \u0435\u0441\u0442\u044c, \u0442\u043e \u043f\u0435\u0440\u0435\u0445\u043e\u0434\u0438\u043c \u043a \u0438\u0441\u043f\u043e\u043b\u043d\u0435\u043d\u0438\u044e \u043f\u043b\u0430\u043d\u0430 \u0438\u043b\u0438 \u0441\u0443\u043c\u043c\u0430\u0440\u0438\u0437\u0430\u0446\u0438\u0438 \u0441\u043e\u043e\u0442\u0432\u0435\u0442\u0441\u0442\u0432\u0435\u043d\u043d\u043e. \u0415\u0441\u043b\u0438 \u043d\u0435\u0442, \u0442\u043e \u0432\u043e\u0437\u0432\u0440\u0430\u0449\u0430\u0435\u043c \u043d\u0430 \u0434\u043e\u0440\u0430\u0431\u043e\u0442\u043a\u0443.class MCPAgent:    # \u041f\u0440\u0435\u0434\u044b\u0434\u0443\u0449\u0438\u0439 \u043a\u043e\u0434        @staticmethod    def need_adjust_plan_router(state: AgentState) -&gt; Literal[&#8216;injector&#8217;, &#8216;planer&#8217;]:        if state.get(&#8216;is_approved&#8217;, False):            return &#8216;injector&#8217;        return &#8216;planer&#8217;    @staticmethod    def need_modify_step_router(state: AgentState) -&gt; Literal[&#8216;agent_node&#8217;, &#8216;compressor&#8217;]:        if state.get(&#8216;is_approved&#8217;, False):            return &#8216;compressor&#8217;        return &#8216;agent_node&#8217;\u0422\u0435\u043f\u0435\u0440\u044c \u043c\u0430\u0440\u0448\u0440\u0443\u0442\u0438\u0437\u0430\u0442\u043e\u0440 \u043f\u0435\u0440\u0435\u0445\u043e\u0434\u0430 \u043f\u043e \u0448\u0430\u0433\u0430\u043c. \u0412 \u043d\u0451\u043c \u043c\u044b \u0434\u043e\u043b\u0436\u043d\u044b \u043f\u0440\u043e\u0432\u0435\u0440\u0438\u0442\u044c, \u0447\u0442\u043e \u043f\u043b\u0430\u043d \u0435\u0449\u0451 \u0432 \u043f\u0440\u043e\u0446\u0435\u0441\u0441\u0435 \u0432\u044b\u043f\u043e\u043b\u043d\u0435\u043d\u0438\u044f. \u041f\u0440\u043e\u0432\u0435\u0440\u044f\u0435\u043c \u0441\u0440\u0430\u0432\u043d\u0435\u043d\u0438\u0435\u043c \u043d\u043e\u043c\u0435\u0440\u0430 \u0448\u0430\u0433\u0430 \u0438 \u0434\u043b\u0438\u043d\u044b \u043f\u043b\u0430\u043d\u0430class MCPAgent:    # \u041f\u0440\u0435\u0434\u044b\u0434\u0443\u0449\u0438\u0439 \u043a\u043e\u0434    @staticmethod    def next_step_router(state: AgentState) -&gt; Literal[&#8216;injector&#8217;, &#8216;finalizer&#8217;]:        if state[&#8216;current_step&#8217;] &lt; len(state[&#8216;plan&#8217;]):            return &#8216;injector&#8217;        return &#8216;finalizer&#8217;\u0421\u043e\u0431\u0435\u0440\u0451\u043c \u0433\u0440\u0430\u0444. \u0418\u043d\u0438\u0446\u0438\u0430\u043b\u0438\u0437\u0438\u0440\u0443\u0435\u043c \u0443\u0437\u043b\u044b \u0438 \u0441\u043e\u0435\u0434\u0438\u043d\u0438\u043c \u0438\u0445 \u0441\u043e\u0433\u043b\u0430\u0441\u043d\u043e \u0441\u0445\u0435\u043c\u0435 \u0438\u0437 \u0432\u0442\u043e\u0440\u043e\u0439 \u0447\u0430\u0441\u0442\u0438 \u0441\u0442\u0430\u0442\u044c\u0438:class MCPAgent:    # \u041f\u0440\u0435\u0434\u044b\u0434\u0443\u0449\u0438\u0439 \u043a\u043e\u0434    def _compile_graph(self):        workflow = StateGraph(AgentState)        workflow.add_node(&#8216;agent_node&#8217;, AgentNode(llm=self.llm_with_tools).node)        workflow.add_node(&#8216;compressor&#8217;, ContextCompressorNode().node)        workflow.add_node(&#8216;finalizer&#8217;, FinalizerNode(llm=settings.llm.chat.llm).node)        workflow.add_node(&#8216;planer&#8217;, PlanerNode(llm=settings.llm.chat.llm).node)        workflow.add_node(&#8216;plan_solver&#8217;, PlanSolverNode().node)        workflow.add_node(&#8216;agent_solver&#8217;, AgentSolverNode().node)        workflow.add_node(&#8216;injector&#8217;, StepInjectorNode().node)        workflow.add_node(&#8216;tools&#8217;, ToolNode(self.tools))        workflow.set_entry_point(key=&#8217;planer&#8217;)        workflow.add_edge(start_key=&#8217;planer&#8217;, end_key=&#8217;plan_solver&#8217;)        workflow.add_conditional_edges(source=&#8217;plan_solver&#8217;, path=self.need_adjust_plan_router)        workflow.add_edge(start_key=&#8217;injector&#8217;, end_key=&#8217;agent_node&#8217;)        workflow.add_conditional_edges(source=&#8217;agent_node&#8217;, path=self.agent_router)        workflow.add_edge(start_key=&#8217;tools&#8217;, end_key=&#8217;agent_node&#8217;)        workflow.add_conditional_edges(source=&#8217;agent_solver&#8217;, path=self.need_modify_step_router)        workflow.add_conditional_edges(source=&#8217;compressor&#8217;, path=self.next_step_router)        workflow.set_finish_point(key=&#8217;finalizer&#8217;)        graph = workflow.compile(            checkpointer=self.checkpointer,            interrupt_after=[&#8216;planer&#8217;],            interrupt_before=[&#8216;agent_solver&#8217;],        )        return graph\u0412 \u043a\u043e\u043c\u043f\u0438\u043b\u044f\u0442\u043e\u0440 \u043f\u0435\u0440\u0435\u0434\u0430\u0451\u043c \u0434\u0432\u0430 \u043e\u0441\u0442\u0430\u043d\u043e\u0432\u0430 &#8212; \u043f\u043e\u0441\u043b\u0435 \u0444\u043e\u0440\u043c\u0438\u0440\u043e\u0432\u0430\u043d\u0438\u044f \u043f\u043b\u0430\u043d\u0430 \u0438 \u0434\u043e &#171;\u0440\u0435\u0448\u0430\u0442\u0435\u043b\u044f&#187; \u0443\u0437\u043b\u0430-\u0430\u0433\u0435\u043d\u0442\u0430.\u0414\u043b\u044f \u0442\u043e\u0433\u043e, \u0447\u0442\u043e\u0431\u044b \u0433\u0440\u0430\u0444 \u0440\u0430\u0431\u043e\u0442\u0430\u043b \u0441 \u043e\u0441\u0442\u0430\u043d\u043e\u0432\u0430\u043c\u0438, \u043d\u0430\u043c \u043d\u0430\u0434\u043e \u0434\u0432\u0430 \u043c\u0435\u0442\u043e\u0434\u0430 &#8212; \u043e\u0434\u0438\u043d \u0434\u043b\u044f \u043f\u0435\u0440\u0432\u043e\u043d\u0430\u0447\u0430\u043b\u044c\u043d\u043e\u0433\u043e \u0437\u0430\u043f\u0443\u0441\u043a\u0430 \u0433\u0440\u0430\u0444\u0430, \u0432\u0442\u043e\u0440\u043e\u0439 &#8212; \u0434\u043b\u044f \u043f\u0440\u043e\u0434\u043e\u043b\u0436\u0435\u043d\u0438\u044f \u043e\u0431\u0449\u0435\u043d\u0438\u044f. \u041d\u0430\u0447\u043d\u0435\u043c \u0441 \u0437\u0430\u043f\u0443\u0441\u043a\u0430:class MCPAgent:    # \u041f\u0440\u0435\u0434\u044b\u0434\u0443\u0449\u0438\u0439 \u043a\u043e\u0434    async def run(self, user_messages: str, request_id: str | None = None) -&gt; dict[str, Any]:        self._check_graph_available()        trace_id = Langfuse.create_trace_id()        return await self._ainvoke_with_tracing(            data={&#8216;user_request&#8217;: user_messages, &#8216;user_input&#8217;: user_messages, &#8216;trace_id&#8217;: trace_id},            request_id=request_id, trace_id=trace_id, span_name=&#8217;agent_run&#8217;)\u0421\u043b\u0443\u0436\u0435\u0431\u043d\u044b\u0435 \u043c\u0435\u0442\u043e\u0434\u044b \u0440\u0430\u0437\u0431\u0435\u0440\u0443 \u043f\u043e\u0437\u0436\u0435, \u043f\u043e\u043a\u0430 \u0441\u0430\u043c \u0430\u043b\u0433\u043e\u0440\u0438\u0442\u043c. \u0414\u043b\u044f \u043d\u0430\u0447\u0430\u043b\u0430 \u043f\u0440\u043e\u0432\u0435\u0440\u044f\u0435\u043c, \u0447\u0442\u043e \u043d\u0435 \u0437\u0430\u0431\u044b\u043b\u0438 \u0438\u043d\u0438\u0446\u0438\u0430\u043b\u0438\u0437\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u0433\u0440\u0430\u0444. \u0414\u0430\u043b\u0435\u0435 \u0444\u043e\u0440\u043c\u0438\u0440\u0443\u0435\u043c trace_id \u0434\u043b\u044f Langfuse \u0441\u0440\u0435\u0434\u0441\u0442\u0432\u0430\u043c\u0438 Langfuse. \u0411\u0435\u0437 \u043f\u0435\u0440\u0435\u0434\u0430\u0447\u0438 trace_id Langfuse \u0431\u0443\u0434\u0435\u0442 \u0441\u043e\u0437\u0434\u0430\u0432\u0430\u0442\u044c \u043e\u0442\u0434\u0435\u043b\u044c\u043d\u044b\u0435 \u0442\u0440\u0435\u0439\u0441\u044b \u0434\u043b\u044f \u043a\u0430\u0436\u0434\u043e\u0433\u043e \u0432\u043e\u0437\u0432\u0440\u0430\u0442\u0430 \u0432 \u0433\u0440\u0430\u0444. \u041d\u0443 \u0438 \u0432\u044b\u0437\u044b\u0432\u0430\u0435\u043c \u0441\u043b\u0443\u0436\u0435\u0431\u043d\u044b\u0439 \u043c\u0435\u0442\u043e\u0434 \u0432\u044b\u0437\u043e\u0432\u0430 LLM.\u0422\u0435\u043f\u0435\u0440\u044c \u043c\u0435\u0442\u043e\u0434 \u0432\u043e\u0437\u043e\u0431\u043d\u043e\u0432\u043b\u0435\u043d\u0438\u044f \u043e\u0431\u0449\u0435\u043d\u0438\u044f:class MCPAgent:    # \u041f\u0440\u0435\u0434\u044b\u0434\u0443\u0449\u0438\u0439 \u043a\u043e\u0434    async def resume(self, user_messages: str, request_id: str | None = None) -&gt; dict[str, Any]:        self._check_graph_available()        trace_id = self._get_trace_id(request_id)        return await self._ainvoke_with_tracing(            data=Command(update={&#8216;user_input&#8217;: user_messages}),            request_id=request_id, trace_id=trace_id, span_name=&#8217;agent_resume&#8217;)\u041e\u0441\u043d\u043e\u0432\u043d\u044b\u0435 \u043e\u0442\u043b\u0438\u0447\u0438\u044f &#8212; \u043d\u043e\u043c\u0435\u0440 \u0442\u0440\u0435\u0439\u0441\u0430 \u0432 Langfuse \u043c\u044b \u0431\u0435\u0440\u0451\u043c \u0438\u0437 \u0441\u0442\u0435\u0439\u0442\u0430 \u0438 \u043d\u0430\u0448\u0435 \u0441\u043e\u043e\u0431\u0449\u0435\u043d\u0438\u0435 \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044f \u043e\u0431\u043e\u0440\u0430\u0447\u0438\u0432\u0430\u0435\u043c \u0432 Command. \u041f\u0435\u0440\u0435\u0445\u043e\u0434\u0438\u043c \u043a \u0441\u043b\u0443\u0436\u0435\u0431\u043d\u044b\u043c \u043c\u0435\u0442\u043e\u0434\u0430\u043c.\u041f\u0435\u0440\u0432\u044b\u0439 \u043c\u0435\u0442\u043e\u0434 \u043f\u0440\u043e\u0432\u0435\u0440\u044f\u0435\u0442 \u043d\u0430\u043b\u0438\u0447\u0438\u0435 \u0430\u0433\u0435\u043d\u0442\u0430:class MCPAgent:    # \u041f\u0440\u0435\u0434\u044b\u0434\u0443\u0449\u0438\u0439 \u043a\u043e\u0434    def _check_graph_available(self):        if self.graph is None:            raise RuntimeError(                &#8216;\u0410\u0433\u0435\u043d\u0442 \u043d\u0435 \u0438\u043d\u0438\u0446\u0438\u0430\u043b\u0438\u0437\u0438\u0440\u043e\u0432\u0430\u043d. \u0417\u0430\u043f\u0443\u0441\u0442\u0438\u0442\u0435 `initialize()`.&#8217;)\u0412\u0442\u043e\u0440\u043e\u0439 \u043d\u0443\u0436\u0435\u043d \u0434\u043b\u044f \u0438\u0437\u0432\u043b\u0435\u0447\u0435\u043d\u0438\u044f trace_id \u0438\u0437 \u0441\u043e\u0445\u0440\u0430\u043d\u0435\u043d\u043d\u043e\u0433\u043e \u0441\u0442\u0435\u0439\u0442\u0430:class MCPAgent:    # \u041f\u0440\u0435\u0434\u044b\u0434\u0443\u0449\u0438\u0439 \u043a\u043e\u0434    def _get_trace_id(self, request_id: str) -&gt; str | None:        return self.graph.get_state(            {&#8216;configurable&#8217;: {&#8216;thread_id&#8217;: request_id}}        ).values.get(&#8216;trace_id&#8217;, None)\u041f\u0440\u043e thread_id \u043f\u0438\u0441\u0430\u043b \u0432 \u0447\u0430\u0441\u0442\u0438 \u043f\u0440\u043e \u0447\u0435\u043a\u043f\u043e\u0438\u043d\u0442\u0435\u0440. \u041e\u043d \u043d\u0443\u0436\u0435\u043d \u0434\u043b\u044f \u0442\u043e\u0433\u043e, \u0447\u0442\u043e\u0431\u044b \u0433\u0440\u0430\u0444 \u0437\u043d\u0430\u043b, \u0441\u0442\u0435\u0439\u0442 \u043a\u0430\u043a\u043e\u0439 \u0438\u043c\u0435\u043d\u043d\u043e \u0441\u0435\u0441\u0441\u0438\u0438 \u043d\u0430\u0434\u043e \u0438\u0437\u0432\u043b\u0435\u043a\u0430\u0442\u044c \u0438\u0437 \u043f\u0430\u043c\u044f\u0442\u0438. \u0424\u043e\u0440\u043c\u0438\u0440\u0443\u0435\u0442\u0441\u044f \u043d\u0430 \u0441\u0442\u043e\u0440\u043e\u043d\u0435 \u0432\u0435\u0431-\u0438\u043d\u0442\u0435\u0440\u0444\u0435\u0439\u0441\u0430. \u0422\u0435\u043f\u0435\u0440\u044c \u043c\u0435\u0442\u043e\u0434 \u0432\u044b\u0437\u043e\u0432\u0430 LLM:class MCPAgent:    # \u041f\u0440\u0435\u0434\u044b\u0434\u0443\u0449\u0438\u0439 \u043a\u043e\u0434    async def _ainvoke_with_tracing(            self, data: dict[str, Any] | Command, request_id: str, trace_id: str, span_name: str    ) -&gt; dict[str, Any]:        with settings.langfuse.client.start_as_current_observation(                as_type=&#8217;span&#8217;,                name=span_name,                trace_context={&#8216;trace_id&#8217;: trace_id},        ) as span:            result = await self.graph.ainvoke(data, config=self._create_config(request_id))            return result[&#8216;messages&#8217;]\u041f\u0435\u0440\u0432\u044b\u043c \u0434\u0435\u043b\u043e\u043c \u0441\u043e\u0437\u0434\u0430\u0451\u043c \u043a\u0430\u0441\u0442\u043e\u043c\u043d\u043e\u0435 \u043d\u0430\u0431\u043b\u044e\u0434\u0435\u043d\u0438\u0435 \u0442\u0438\u043f\u0430 \u0441\u043f\u0430\u043d \u0438 \u043f\u0435\u0440\u0435\u0434\u0430\u0451\u043c \u0432 \u043d\u0435\u0433\u043e \u043f\u0430\u0440\u0430\u043c\u0435\u0442\u0440\u043e\u043c \u043d\u0430\u0448 trace_id. Langfuse \u0441\u0430\u043c \u0441\u0433\u0440\u0443\u043f\u043f\u0438\u0440\u0443\u0435\u0442 \u0432\u0441\u0435 \u043d\u0430\u0431\u043b\u044e\u0434\u0435\u043d\u0438\u044f \u043f\u043e trace_id. \u0414\u0430\u043b\u0435\u0435 \u0432\u044b\u0437\u044b\u0432\u0430\u0435\u043c \u043c\u0435\u0442\u043e\u0434 ainvoke \u0433\u0440\u0430\u0444\u0430, \u043f\u0435\u0440\u0435\u0434\u0430\u0432 \u0432 \u043d\u0435\u0433\u043e \u043a\u043e\u043d\u0444\u0438\u0433:class MCPAgent:    # \u041f\u0440\u0435\u0434\u044b\u0434\u0443\u0449\u0438\u0439 \u043a\u043e\u0434    def _create_config(self, request_id: str) -&gt; dict[str, Any]:        return {            &#8216;callbacks&#8217;: [self.lf_handler],            &#8216;metadata&#8217;: {                &#8216;langfuse_session_id&#8217;: f&#8217;docker_session_{self.init_time}&#8217;,            },            &#8216;configurable&#8217;: {&#8216;thread_id&#8217;: request_id}        }\u0412 \u043a\u043e\u043d\u0444\u0438\u0433\u0435 \u043c\u044b \u043f\u0440\u043e\u043f\u0438\u0441\u044b\u0432\u0430\u0435\u043c Langfuse CallbackHandler \u0434\u043b\u044f \u043e\u0440\u0433\u0430\u043d\u0438\u0437\u0430\u0446\u0438\u0438 \u043d\u0430\u0431\u043b\u044e\u0434\u0435\u043d\u0438\u044f, \u0432 \u043c\u0435\u0442\u0430\u0434\u0430\u043d\u043d\u044b\u0445 \u043f\u0435\u0440\u0435\u0434\u0430\u0451\u043c \u043f\u0430\u0440\u0430\u043c\u0435\u0442\u0440 langfuse_session_id, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u0432 \u0434\u0430\u043b\u044c\u043d\u0435\u0439\u0448\u0435\u043c \u043c\u043e\u0436\u043d\u043e \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u044c \u0434\u043b\u044f \u0444\u0438\u043b\u044c\u0442\u0440\u0430\u0446\u0438\u0438 \u0442\u0440\u0435\u0439\u0441\u043e\u0432 \u0438 thread_id \u0434\u043b\u044f \u0441\u043e\u0445\u0440\u0430\u043d\u0435\u043d\u0438\u044f \u0441\u0442\u0435\u0439\u0442\u0430 \u0432 \u0447\u0435\u043a\u043f\u043e\u0438\u043d\u0442\u0435\u0440\u0435.\u0413\u0440\u0430\u0444 \u0433\u043e\u0442\u043e\u0432. \u041e\u0441\u0442\u0430\u043b\u043e\u0441\u044c \u0434\u043e\u0440\u0430\u0431\u043e\u0442\u0430\u0442\u044c Gradio \u0438\u043d\u0442\u0435\u0440\u0444\u0435\u0439\u0441 \u043f\u043e\u0434 \u043d\u043e\u0432\u044b\u0435 \u0444\u0443\u043d\u043a\u0446\u0438\u0438 \u0438 \u043c\u043e\u0436\u043d\u043e \u0440\u0435\u043b\u0438\u0437\u0438\u0442\u044c)\u0418\u043d\u0442\u0435\u0440\u0444\u0435\u0439\u0441 \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044f\u041f\u0440\u0438 \u0438\u0437\u043c\u0435\u043d\u0435\u043d\u0438\u0438 \u0430\u0440\u0445\u0438\u0442\u0435\u043a\u0442\u0443\u0440\u044b \u0430\u0433\u0435\u043d\u0442\u0430 (\u0434\u043e\u0431\u0430\u0432\u043b\u0435\u043d\u0438\u0435 \u043f\u043e\u0434\u0442\u0432\u0435\u0440\u0436\u0434\u0435\u043d\u0438\u044f \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u0435\u043c) \u044f \u043f\u043e\u0448\u0435\u043b \u043d\u0430 \u043e\u0434\u0438\u043d \u043a\u043e\u043c\u043f\u0440\u043e\u043c\u0438\u0441 &#8212; \u043f\u043e\u0434\u0442\u0432\u0435\u0440\u0436\u0434\u0435\u043d\u0438\u0435 \u043f\u0440\u043e\u0441\u0442\u044b\u043c \u0441\u043b\u043e\u0432\u043e\u043c \u201c\u041f\u0440\u043e\u0434\u043e\u043b\u0436\u0438\u0442\u044c\u201d. \u0418 \u0447\u0442\u043e\u0431\u044b \u043a\u0430\u0436\u0434\u044b\u0439 \u0440\u0430\u0437 \u0435\u0433\u043e \u043d\u0435 \u043f\u0438\u0441\u0430\u0442\u044c, \u044f \u0440\u0435\u0448\u0438\u043b \u0434\u043e\u0431\u0430\u0432\u0438\u0442\u044c \u0432 \u0438\u043d\u0442\u0435\u0440\u0444\u0435\u0439\u0441 \u043a\u043d\u043e\u043f\u043a\u0443 \u201c\u041f\u0440\u043e\u0434\u043e\u043b\u0436\u0438\u0442\u044c\u201d. \u0414\u043b\u044f \u044d\u0442\u043e\u0433\u043e \u043f\u0440\u0438\u0448\u043b\u043e\u0441\u044c \u043e\u0442\u043a\u0430\u0437\u0430\u0442\u044c\u0441\u044f \u043e\u0442 gr.ChatInterface \u0438 \u043f\u0435\u0440\u0435\u043f\u0438\u0441\u0430\u0442\u044c \u0438\u043d\u0442\u0435\u0440\u0444\u0435\u0439\u0441 \u043d\u0430 gr.Chatbot:class MCPCodingAgentApp:    def build_interface(self):        with gr.Blocks(title=&#8217;MCP Coding Agent&#8217;, fill_height=True) as self.demo:            gr.Markdown(&#8216;# MCP Coding Agent&#8217;)            gr.Markdown(&#8216;\u041f\u043e\u043c\u043e\u0449\u043d\u0438\u043a \u0440\u0430\u0437\u0440\u0430\u0431\u043e\u0442\u0447\u0438\u043a\u0430 \u0441 \u0434\u043e\u0441\u0442\u0443\u043f\u043e\u043c \u043a \u0444\u0430\u0439\u043b\u0430\u043c, Git \u0438 \u0434\u043e\u043a\u0443\u043c\u0435\u043d\u0442\u0430\u0446\u0438\u0438&#8217;)            chatbot = gr.Chatbot(label=&#8217;\u0427\u0430\u0442 \u0441 \u0430\u0433\u0435\u043d\u0442\u043e\u043c&#8217;, height=700)            request_id_state = gr.State(&#187;)            with gr.Row():                msg = gr.Textbox(                    label=&#8217;\u0412\u0430\u0448\u0435 \u0441\u043e\u043e\u0431\u0449\u0435\u043d\u0438\u0435&#8217;,                    placeholder=&#8217;\u0412\u0432\u0435\u0434\u0438\u0442\u0435 \u0441\u043e\u043e\u0431\u0449\u0435\u043d\u0438\u0435 \u0438\u043b\u0438 \u043d\u0430\u0436\u043c\u0438\u0442\u0435 &#171;\u041f\u0440\u043e\u0434\u043e\u043b\u0436\u0438\u0442\u044c&#187;&#8216;,                    scale=8,                    container=False                )                submit_btn = gr.Button(&#8216;\u041e\u0442\u043f\u0440\u0430\u0432\u0438\u0442\u044c&#8217;, variant=&#8217;primary&#8217;)                continue_btn = gr.Button(&#8216;\u25b6 \u041f\u0440\u043e\u0434\u043e\u043b\u0436\u0438\u0442\u044c&#8217;, variant=&#8217;secondary&#8217;)            submit_btn.click(                fn=self._respond,                inputs=[msg, chatbot, request_id_state],                outputs=[chatbot, msg, request_id_state])            msg.submit(                fn=self._respond,                inputs=[msg, chatbot, request_id_state],                outputs=[chatbot, msg, request_id_state])            continue_btn.click(                fn=self._continue,                inputs=[chatbot, request_id_state],                outputs=[chatbot, request_id_state])\u041f\u043e\u0441\u043b\u0435 \u043f\u043e\u043b\u044f \u0447\u0430\u0442\u0430 \u0441\u043e\u0437\u0434\u0430\u0451\u043c \u0441\u0442\u0440\u043e\u043a\u0443 \u0441 \u043f\u043e\u043b\u0435\u043c \u0434\u043b\u044f \u0432\u0432\u043e\u0434\u0430 \u0441\u043e\u043e\u0431\u0449\u0435\u043d\u0438\u044f \u0438 \u0434\u0432\u0443\u043c\u044f \u043a\u043d\u043e\u043f\u043a\u0430\u043c\u0438 &#8212; \u201c\u041e\u0442\u043f\u0440\u0430\u0432\u0438\u0442\u044c\u201d \u0438 \u201c\u041f\u0440\u043e\u0434\u043e\u043b\u0436\u0438\u0442\u044c\u201d. \u0414\u0430\u043b\u0435\u0435 \u043d\u0430\u0437\u043d\u0430\u0447\u0430\u0435\u043c \u0444\u0443\u043d\u043a\u0446\u0438\u0438-\u043e\u0431\u0440\u0430\u0431\u043e\u0442\u0447\u0438\u043a\u0438 \u0434\u043b\u044f \u043d\u0430\u0448\u0438\u0445 \u044d\u043b\u0435\u043c\u0435\u043d\u0442\u043e\u0432 (\u0434\u043b\u044f \u043f\u043e\u043b\u044f \u0432\u0432\u043e\u0434\u0430 \u0442\u043e\u0436\u0435, \u0447\u0442\u043e\u0431\u044b \u0431\u044b\u043b\u0430 \u043e\u0442\u043f\u0440\u0430\u0432\u043a\u0430 \u043f\u043e \u043d\u0430\u0436\u0430\u0442\u0438\u044e Enter). \u041f\u0430\u0440\u0430\u043c\u0435\u0442\u0440\u044b inputs \u0438 outputs \u0441\u0432\u044f\u0437\u044b\u0432\u0430\u044e\u0442 \u0432\u0445\u043e\u0434\u044b-\u0432\u044b\u0445\u043e\u0434\u044b \u0444\u0443\u043d\u043a\u0446\u0438\u0438-\u043e\u0431\u0440\u0430\u0431\u043e\u0442\u0447\u0438\u043a\u0430 \u0441 \u043e\u0431\u044a\u0435\u043a\u0442\u0430\u043c\u0438 Gradio.\u041e\u0431\u044a\u0435\u043a\u0442 gr.State(&#187;) &#8230;<\/p>\n","protected":false},"author":1,"featured_media":0,"comment_status":"closed","ping_status":"closed","sticky":false,"template":"","format":"standard","meta":{"footnotes":""},"categories":[],"tags":[],"class_list":["post-484337","post","type-post","status-publish","format-standard","hentry"],"_links":{"self":[{"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/posts\/484337","targetHints":{"allow":["GET"]}}],"collection":[{"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/posts"}],"about":[{"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/types\/post"}],"author":[{"embeddable":true,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/users\/1"}],"replies":[{"embeddable":true,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=%2Fwp%2Fv2%2Fcomments&post=484337"}],"version-history":[{"count":0,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/posts\/484337\/revisions"}],"wp:attachment":[{"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=%2Fwp%2Fv2%2Fmedia&parent=484337"}],"wp:term":[{"taxonomy":"category","embeddable":true,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=%2Fwp%2Fv2%2Fcategories&post=484337"},{"taxonomy":"post_tag","embeddable":true,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=%2Fwp%2Fv2%2Ftags&post=484337"}],"curies":[{"name":"wp","href":"https:\/\/api.w.org\/{rel}","templated":true}]}}