File size: 7,249 Bytes
9a8789a
ff9325e
 
fe66ec6
 
9a8789a
fe66ec6
 
9a8789a
 
ff9325e
 
3207814
ff9325e
3207814
 
ff9325e
9a8789a
ff9325e
9a8789a
 
 
 
 
 
 
 
 
 
 
 
 
 
ff9325e
fe66ec6
 
 
9a8789a
 
 
fe66ec6
 
 
 
d6fedfa
9a8789a
 
 
 
 
fe66ec6
9a8789a
 
 
 
 
 
 
 
 
 
 
 
fe66ec6
9a8789a
fe66ec6
 
9a8789a
 
 
 
 
 
 
 
 
 
 
bea2d0b
9a8789a
 
fe66ec6
9a8789a
fe66ec6
9a8789a
 
 
 
 
fe66ec6
9a8789a
fe66ec6
9a8789a
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
fe66ec6
 
 
9a8789a
 
3207814
fe66ec6
 
 
 
 
9a8789a
 
 
 
 
 
 
fe66ec6
9a8789a
 
 
 
 
 
 
 
 
 
fe66ec6
9a8789a
 
 
 
fe66ec6
 
9a8789a
 
 
 
 
 
 
 
 
 
 
 
 
fe66ec6
 
9a8789a
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
fe66ec6
 
 
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
import { get, writable } from 'svelte/store';

export enum LCMLiveStatus {
  CONNECTED = 'connected',
  DISCONNECTED = 'disconnected',
  CONNECTING = 'connecting',
  WAIT = 'wait',
  SEND_FRAME = 'send_frame',
  TIMEOUT = 'timeout',
  ERROR = 'error'
}

const initStatus: LCMLiveStatus = LCMLiveStatus.DISCONNECTED;

export const lcmLiveStatus = writable<LCMLiveStatus>(initStatus);
export const streamId = writable<string | null>(null);

// WebSocket connection
let websocket: WebSocket | null = null;
// Flag to track intentional connection closure
let intentionalClosure = false;

// Register browser unload event listener to properly close WebSockets
if (typeof window !== 'undefined') {
  window.addEventListener('beforeunload', () => {
    // Mark any closure during page unload as intentional
    intentionalClosure = true;
    // Close the WebSocket properly if it exists
    if (websocket && websocket.readyState === WebSocket.OPEN) {
      websocket.close(1000, 'Page unload');
    }
  });
}
export const lcmLiveActions = {
  async start(getSreamdata: () => any[]) {
    return new Promise((resolve, reject) => {
      try {
        // Set connecting status immediately
        lcmLiveStatus.set(LCMLiveStatus.CONNECTING);

        const userId = crypto.randomUUID();
        const websocketURL = `${
          window.location.protocol === 'https:' ? 'wss' : 'ws'
        }:${window.location.host}/api/ws/${userId}`;

        // Close any existing connection first
        if (websocket && websocket.readyState !== WebSocket.CLOSED) {
          websocket.close();
        }

        websocket = new WebSocket(websocketURL);

        // Set a connection timeout
        const connectionTimeout = setTimeout(() => {
          if (websocket && websocket.readyState !== WebSocket.OPEN) {
            console.error('WebSocket connection timeout');
            lcmLiveStatus.set(LCMLiveStatus.ERROR);
            streamId.set(null);
            reject(new Error('Connection timeout. Please try again.'));
            websocket.close();
          }
        }, 10000); // 10 second timeout

        websocket.onopen = () => {
          clearTimeout(connectionTimeout);
          console.log('Connected to websocket');
        };

        websocket.onclose = (event) => {
          clearTimeout(connectionTimeout);
          console.log(`Disconnected from websocket: ${event.code} ${event.reason}`);

          // Only change status if we're not in ERROR state (which would mean we already handled the error)
          if (get(lcmLiveStatus) !== LCMLiveStatus.ERROR) {
            lcmLiveStatus.set(LCMLiveStatus.DISCONNECTED);
          }

          // If connection was never established (close without open)
          if (event.code === 1006 && get(streamId) === null) {
            reject(new Error('Cannot connect to server. Please try again later.'));
          }
        };

        websocket.onerror = (err) => {
          clearTimeout(connectionTimeout);
          console.error('WebSocket error:', err);
          lcmLiveStatus.set(LCMLiveStatus.ERROR);
          streamId.set(null);
          reject(new Error('Connection error. Please try again.'));
        };

        websocket.onmessage = (event) => {
          try {
            const data = JSON.parse(event.data);
            switch (data.status) {
              case 'connected':
                lcmLiveStatus.set(LCMLiveStatus.CONNECTED);
                streamId.set(userId);
                resolve({ status: 'connected', userId });
                break;
              case 'send_frame':
                lcmLiveStatus.set(LCMLiveStatus.SEND_FRAME);
                try {
                  const streamData = getSreamdata();
                  // Send as an object, not a string, to use the proper handling in the send method
                  this.send({ status: 'next_frame' });
                  for (const d of streamData) {
                    this.send(d);
                  }
                } catch (error) {
                  console.error('Error sending frame data:', error);
                }
                break;
              case 'wait':
                lcmLiveStatus.set(LCMLiveStatus.WAIT);
                break;
              case 'timeout':
                console.log('Session timeout');
                lcmLiveStatus.set(LCMLiveStatus.TIMEOUT);
                streamId.set(null);
                reject(new Error('Session timeout. Please restart.'));
                break;
              case 'error':
                console.error('Server error:', data.message);
                lcmLiveStatus.set(LCMLiveStatus.ERROR);
                streamId.set(null);
                reject(new Error(data.message || 'Server error occurred'));
                break;
              default:
                console.log('Unknown message status:', data.status);
            }
          } catch (error) {
            console.error('Error handling websocket message:', error);
          }
        };
      } catch (err) {
        console.error('Error initializing websocket:', err);
        lcmLiveStatus.set(LCMLiveStatus.ERROR);
        streamId.set(null);
        reject(err);
      }
    });
  },
  send(data: Blob | { [key: string]: any }) {
    try {
      if (websocket && websocket.readyState === WebSocket.OPEN) {
        if (data instanceof Blob) {
          websocket.send(data);
        } else {
          websocket.send(JSON.stringify(data));
        }
      } else {
        const readyStateText = websocket
          ? ['CONNECTING', 'OPEN', 'CLOSING', 'CLOSED'][websocket.readyState]
          : 'null';
        console.warn(`WebSocket not ready for sending: ${readyStateText}`);

        // If WebSocket is closed unexpectedly, set status to disconnected
        if (!websocket || websocket.readyState === WebSocket.CLOSED) {
          lcmLiveStatus.set(LCMLiveStatus.DISCONNECTED);
          streamId.set(null);
        }
      }
    } catch (error) {
      console.error('Error sending data through WebSocket:', error);
      // Handle WebSocket error by forcing disconnection
      this.stop();
    }
  },

  async reconnect(getSreamdata: () => any[]) {
    try {
      await this.stop();
      // Small delay to ensure clean disconnection before reconnecting
      await new Promise((resolve) => setTimeout(resolve, 500));
      return await this.start(getSreamdata);
    } catch (error) {
      console.error('Reconnection failed:', error);
      throw error;
    }
  },

  async stop() {
    lcmLiveStatus.set(LCMLiveStatus.DISCONNECTED);
    try {
      if (websocket) {
        // Only attempt to close if not already closed
        if (websocket.readyState !== WebSocket.CLOSED) {
          // Set up onclose handler to clean up only
          websocket.onclose = () => {
            console.log('WebSocket closed cleanly during stop()');
          };

          // Set up onerror to be silent during intentional closure
          websocket.onerror = () => {};

          websocket.close(1000, 'Client initiated disconnect');
        }
      }
    } catch (error) {
      console.error('Error during WebSocket closure:', error);
    } finally {
      // Always clean up references
      websocket = null;
      streamId.set(null);
    }
  }
};