Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Frontend/react-client does not resume session after backend connection loss #828

Closed
jbeckerdm opened this issue Mar 19, 2024 · 8 comments · Fixed by #996
Closed

Frontend/react-client does not resume session after backend connection loss #828

jbeckerdm opened this issue Mar 19, 2024 · 8 comments · Fixed by #996
Labels
bug Something isn't working

Comments

@jbeckerdm
Copy link

jbeckerdm commented Mar 19, 2024

Describe the bug
We use a chainlit setup with chat_persistence and on_chat_resume.
If we start a chat with one message, then cause a websocket disconnect by, for example, restarting the backend,
and then write another message once the reconnect happened, both messages will be persisted in two different sessions.
This can be reproduced with the integrated chainlit frontend.

To Reproduce
Steps to reproduce the behavior:

  1. Start a chainlit backend on version ^1.0.301
  2. Open the frontend on localhost:8000
  3. Start a new chat by writing a message, say "message 1"
  4. Stop and restart the backend process
  5. Switch back to the frontend and wait for the "Could not reach the server" message to disappear.
  6. Write a new message, say "message 2"
  7. Reload the page and notice how there are two chats with one message each.

Expected behavior
There should be one persisted chat session with both messages in it

Screenshots
Screenshot 2024-03-19 at 15 35 24

Screenshot 2024-03-19 at 15 35 32

Desktop (please complete the following information):

  • OS: MacOS
  • Browser: Chrome
  • Version: 122
@jbeckerdm jbeckerdm added the bug Something isn't working label Mar 19, 2024
@qtangs
Copy link
Contributor

qtangs commented May 7, 2024

I face this error too. This happens frequently for Cloud-hosted servers with auto-scaling.

I've traced the root cause to this:

  • When server restarts, frontend requests a new websocket session with X-Chainlit-Thread-Id header (code below). But this header is empty because idToResume is never set for new threads.
    'X-Chainlit-Thread-Id': idToResume || '',

    idToResume is only set when a thread is resumed:
    setIdToResume(threadId!);
  • Due to this, backend creates a new thread and ignore the existing messages completely (because the old thread is not loaded).

Possible solutions:

@tpatel, what do you think? I can open a new PR for this fix.

@qvalentin
Copy link
Contributor

@qtangs, my analysis of the problem found the same root cause. However, I don't think your suggested solutions will work.
The problem is that the X-Chainlit-Thread-Id value is given to the socket.io session when the session is created and cannot be updated later. So even if you use setIdToResume(threadId!); the session will still have the same values for the headers.
When the server restarts, socket.io will retry the connection using the existing headers.

A possible solution would be to listen on the reconnect or reconnect_attempt event of socket.io and create a new session, if idToResume is not set.

@qtangs
Copy link
Contributor

qtangs commented May 14, 2024

@qvalentin you're right. Using currentThreadId like that won't work as the initial value is undefined and the connection headers are not updated after the initial socket initialization.

I've tested this addition of the update after useChatSession.ts#L68 and verified that it works:

  const idToResume = useRecoilValue(threadIdToResumeState);
  const setCurrentThreadId = useSetRecoilState(currentThreadIdState);

  // Use currentThreadId as thread id in websocket header if it's set
  const currentThreadId = useRecoilValue(currentThreadIdState);

  useEffect(() => {
    if (session?.socket && currentThreadId) {
        session.socket.io.opts.extraHeaders!['X-Chainlit-Thread-Id'] = currentThreadId;
    }
  }, [currentThreadId]);

@qvalentin
Copy link
Contributor

Great, will you open a PR?

@qtangs
Copy link
Contributor

qtangs commented May 14, 2024

Yeah, I'm planning to do that when time permits. Will need to add some test cases too

@willydouhard
Copy link
Collaborator

Please submit a PR!

qtangs added a commit to qtangs/chainlit that referenced this issue May 15, 2024
…th currentThreadId to ensure session continuation after backend restart
@qtangs
Copy link
Contributor

qtangs commented May 15, 2024

Please submit a PR!

PR is created, pls review.

dokterbob pushed a commit that referenced this issue Aug 28, 2024
Update websocket's thread id header with currentThreadId to ensure session continuation after backend restart.
@github-project-automation github-project-automation bot moved this from Todo to Done in Chainlit Community Aug 28, 2024
@deepakjyadav01
Copy link

Hey, just wanna know. is it fixed? I was facing a similar issue related to X-Chainlit-Thread-id being not set properly even after using clear(), using setIdtoResume() while resuming to the old chat?. What will be the best way of doing so. The values are getting updated if i refresh it but not when i set it before making reconnection. Any help??

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
bug Something isn't working
Projects
5 participants