signalrProvider.tsx 3.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112
  1. 'use client';
  2. import { createContext, useContext, useEffect, useState, useRef } from 'react';
  3. import * as signalR from '@microsoft/signalr';
  4. import { fetchApi } from '@/lib/utils/client';
  5. const SignalRContext = createContext<{
  6. chatConnection: signalR.HubConnection | null;
  7. chatConnected: boolean;
  8. stopConnections: () => Promise<void>;
  9. reconnectChat: (accessToken?: string | null) => Promise<void>;
  10. }>({
  11. chatConnection: null,
  12. chatConnected: false,
  13. stopConnections: async () => {},
  14. reconnectChat: async () => {}
  15. });
  16. type Props = {
  17. children: React.ReactNode;
  18. accessToken: string|null;
  19. signalRChatUrl: string;
  20. }
  21. export function SignalRProvider({ children, accessToken, signalRChatUrl }: Props) {
  22. const chatConnectionRef = useRef<signalR.HubConnection|null>(null);
  23. const [chatConnection, setChatConnection] = useState<signalR.HubConnection|null>(null);
  24. const [chatConnected, setChatConnected] = useState<boolean>(false);
  25. // 초기 렌더 시에만 전달됨. 토큰 갱신 시에는 reconnectChat()을 통해 수동으로 재연결 처리
  26. useEffect(() => {
  27. initChatConnection(accessToken);
  28. return () => {
  29. stopConnections();
  30. };
  31. }, []);
  32. useEffect(() => {
  33. if (chatConnected) {
  34. console.info('SignalR Chat Connected');
  35. }
  36. }, [chatConnected]);
  37. const initChatConnection = async (accessToken?: string|null) => {
  38. try {
  39. if (chatConnectionRef.current && chatConnectionRef.current.state !== signalR.HubConnectionState.Disconnected) {
  40. await chatConnectionRef.current.stop();
  41. }
  42. const connectionOptions = accessToken ? { accessTokenFactory: async () => accessToken, withCredentials: true } : {};
  43. const conn = new signalR.HubConnectionBuilder().withUrl(signalRChatUrl, connectionOptions).withAutomaticReconnect().build();
  44. await conn.start();
  45. conn.on('Connected', (message) => {
  46. console.info(message);
  47. });
  48. conn.on('Logout', (message) => {
  49. console.info(message);
  50. });
  51. conn.on('Kick', async () => {
  52. fetchApi('/api/auth/logout', {
  53. method: 'POST'
  54. }).then(() => {
  55. alert('관리자에 의해 강제 종료되었습니다.');
  56. localStorage.setItem('rememberMe', "false");
  57. localStorage.removeItem('member');
  58. location.replace('/');
  59. });
  60. });
  61. chatConnectionRef.current = conn;
  62. setChatConnection(conn);
  63. setChatConnected(true);
  64. } catch (error) {
  65. console.error('SignalR Chat Connect Failed:', error);
  66. }
  67. };
  68. const stopConnections = async () => {
  69. if (chatConnectionRef.current && chatConnectionRef.current.state === signalR.HubConnectionState.Connected) {
  70. try {
  71. await chatConnectionRef.current.invoke('Logout');
  72. setChatConnected(false);
  73. } catch (error) {
  74. console.error('SignalR Chat Disconnect Failed:', error);
  75. }
  76. }
  77. };
  78. const reconnectChat = async (token?: string | null) => {
  79. await initChatConnection(token);
  80. };
  81. return (
  82. <SignalRContext.Provider value={{
  83. chatConnection,
  84. chatConnected,
  85. stopConnections,
  86. reconnectChat
  87. }}>
  88. {children}
  89. </SignalRContext.Provider>
  90. )
  91. }
  92. export function useSignalRContext() {
  93. return useContext(SignalRContext);
  94. }