better resume
Some checks failed
CI/CD Pipeline / build (push) Failing after 4s

This commit is contained in:
2026-03-20 12:16:02 -06:00
parent 6138d71d29
commit 85eb8bcefa
4 changed files with 42 additions and 3 deletions

View File

@@ -167,12 +167,13 @@ defmodule ElixirAi.ConversationManager do
# Returns the full conversation state using the pid directly, bypassing the
# Horde registry (which may not have synced yet on the calling node).
# Also includes the runner pid so the caller can make further direct calls.
defp reply_with_conversation(name, state) do
case start_and_subscribe(name, state) do
{:ok, pid, new_subscriptions, new_runners} ->
new_state = %{state | subscriptions: new_subscriptions, runners: new_runners}
conversation = GenServer.call(pid, :get_conversation)
{:reply, {:ok, conversation}, new_state}
{:reply, {:ok, Map.put(conversation, :runner_pid, pid)}, new_state}
{:error, _reason} = error ->
{:reply, error, state}

View File

@@ -14,11 +14,13 @@ defmodule ElixirAiWeb.ChatLive do
if connected?(socket) do
Phoenix.PubSub.subscribe(ElixirAi.PubSub, chat_topic(name))
:pg.join(ElixirAi.LiveViewPG, {:liveview, __MODULE__}, self())
send(self(), :sync_streaming)
end
{:ok,
socket
|> assign(conversation_name: name)
|> assign(runner_pid: Map.get(conversation, :runner_pid))
|> assign(user_input: "")
|> assign(messages: conversation.messages)
|> assign(streaming_response: conversation.streaming_response)
@@ -128,6 +130,35 @@ defmodule ElixirAiWeb.ChatLive do
{:noreply, assign(socket, streaming_response: nil, ai_error: nil)}
end
# Fetches the authoritative streaming snapshot directly from the runner pid,
# bypassing the Horde registry. Sent to self immediately after subscribing on
# connect so it is the first message processed — before any PubSub chunks.
def handle_info(:sync_streaming, %{assigns: %{runner_pid: pid}} = socket)
when is_pid(pid) do
case GenServer.call(pid, :get_streaming_response) do
nil ->
{:noreply, assign(socket, streaming_response: nil)}
%{content: content, reasoning_content: reasoning_content} = snapshot ->
socket =
socket
|> assign(streaming_response: snapshot)
|> then(fn s ->
if content != "", do: push_event(s, "md_chunk", %{chunk: content}), else: s
end)
|> then(fn s ->
if reasoning_content != "",
do: push_event(s, "reasoning_chunk", %{chunk: reasoning_content}),
else: s
end)
|> push_event("scroll_to_bottom", %{})
{:noreply, socket}
end
end
def handle_info(:sync_streaming, socket), do: {:noreply, socket}
def handle_info({:user_chat_message, message}, socket) do
{:noreply,
socket
@@ -235,6 +266,13 @@ defmodule ElixirAiWeb.ChatLive do
{:noreply, assign(socket, background_color: color)}
end
defp get_snapshot(%{assigns: %{runner_pid: pid}} = _socket) when is_pid(pid) do
case GenServer.call(pid, :get_streaming_response) do
nil -> %{id: nil, content: "", reasoning_content: "", tool_calls: []}
snapshot -> snapshot
end
end
defp get_snapshot(socket) do
ChatRunner.get_streaming_response(socket.assigns.conversation_name)
|> case do