Files
nezha-dash-v1/src/context/websocket-provider.tsx
2024-12-02 02:15:46 +08:00

82 lines
2.0 KiB
TypeScript

import React, { useEffect, useRef, useState } from "react";
import { WebSocketContext, WebSocketContextType } from "./websocket-context";
interface WebSocketProviderProps {
url: string;
children: React.ReactNode;
}
export const WebSocketProvider: React.FC<WebSocketProviderProps> = ({
url,
children,
}) => {
const [lastMessage, setLastMessage] = useState<{ data: string } | null>(null);
const [connected, setConnected] = useState(false);
const ws = useRef<WebSocket | null>(null);
const reconnectTimeout = useRef<NodeJS.Timeout>();
const maxReconnectAttempts = 30;
const reconnectAttempts = useRef(0);
const connect = () => {
try {
const wsUrl = new URL(url, window.location.origin);
wsUrl.protocol = wsUrl.protocol.replace("http", "ws");
ws.current = new WebSocket(wsUrl.toString());
ws.current.onopen = () => {
console.log("WebSocket connected");
setConnected(true);
reconnectAttempts.current = 0;
};
ws.current.onclose = () => {
console.log("WebSocket disconnected");
setConnected(false);
// 重连逻辑
if (reconnectAttempts.current < maxReconnectAttempts) {
reconnectTimeout.current = setTimeout(() => {
reconnectAttempts.current++;
connect();
}, 3000);
}
};
ws.current.onmessage = (event) => {
setLastMessage({ data: event.data });
};
ws.current.onerror = (error) => {
console.error("WebSocket error:", error);
};
} catch (error) {
console.error("WebSocket connection error:", error);
}
};
useEffect(() => {
connect();
return () => {
if (ws.current) {
ws.current.close();
}
if (reconnectTimeout.current) {
clearTimeout(reconnectTimeout.current);
}
};
}, [url]);
const contextValue: WebSocketContextType = {
lastMessage,
connected,
};
return (
<WebSocketContext.Provider value={contextValue}>
{children}
</WebSocketContext.Provider>
);
};