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);
}
}
};
|