Added withToken func to remove the duplicated code
This commit is contained in:
parent
740a29688d
commit
7310f49eb8
@ -6,7 +6,11 @@ import { splitPrivateData, isValid32ByteHex } from '../utils/service.utils';
|
|||||||
import { MerkleProofResult } from '../../pkg/sdk_client';
|
import { MerkleProofResult } from '../../pkg/sdk_client';
|
||||||
|
|
||||||
export class IframeController {
|
export class IframeController {
|
||||||
|
private static isInitialized = false; // <--- VERROU
|
||||||
|
|
||||||
static async init() {
|
static async init() {
|
||||||
|
if (this.isInitialized) return; // On sort si déjà lancé
|
||||||
|
|
||||||
// On ne lance l'écoute que si on est dans une iframe
|
// On ne lance l'écoute que si on est dans une iframe
|
||||||
if (window.self !== window.top) {
|
if (window.self !== window.top) {
|
||||||
console.log('[IframeController] 📡 Mode Iframe détecté. Démarrage des listeners API...');
|
console.log('[IframeController] 📡 Mode Iframe détecté. Démarrage des listeners API...');
|
||||||
@ -36,6 +40,17 @@ export class IframeController {
|
|||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// Helper pour vérifier le token avant chaque action sensible
|
||||||
|
const withToken = async (event: MessageEvent, action: () => Promise<void>) => {
|
||||||
|
const { accessToken } = event.data;
|
||||||
|
// On vérifie si le token est présent ET valide pour l'origine de l'iframe
|
||||||
|
if (!accessToken || !(await tokenService.validateToken(accessToken, event.origin))) {
|
||||||
|
throw new Error('Invalid or expired session token');
|
||||||
|
}
|
||||||
|
// Si tout est bon, on exécute l'action
|
||||||
|
await action();
|
||||||
|
};
|
||||||
|
|
||||||
// --- Définitions des gestionnaires (Handlers) ---
|
// --- Définitions des gestionnaires (Handlers) ---
|
||||||
|
|
||||||
const handleRequestLink = async (event: MessageEvent) => {
|
const handleRequestLink = async (event: MessageEvent) => {
|
||||||
@ -95,11 +110,7 @@ export class IframeController {
|
|||||||
throw new Error('Device already paired — ignoring CREATE_PAIRING request');
|
throw new Error('Device already paired — ignoring CREATE_PAIRING request');
|
||||||
}
|
}
|
||||||
|
|
||||||
const { accessToken } = event.data;
|
await withToken(event, async () => {
|
||||||
if (!accessToken || !(await tokenService.validateToken(accessToken, event.origin))) {
|
|
||||||
throw new Error('Invalid or expired session token');
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log("[Router:API] 🚀 Démarrage du processus d'appairage...");
|
console.log("[Router:API] 🚀 Démarrage du processus d'appairage...");
|
||||||
|
|
||||||
const myAddress = services.getDeviceAddress();
|
const myAddress = services.getDeviceAddress();
|
||||||
@ -138,17 +149,14 @@ export class IframeController {
|
|||||||
messageId: event.data.messageId,
|
messageId: event.data.messageId,
|
||||||
};
|
};
|
||||||
window.parent.postMessage(successMsg, event.origin);
|
window.parent.postMessage(successMsg, event.origin);
|
||||||
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
const handleGetMyProcesses = async (event: MessageEvent) => {
|
const handleGetMyProcesses = async (event: MessageEvent) => {
|
||||||
console.log(`[Router:API] 📨 Message ${MessageType.GET_MY_PROCESSES} reçu`);
|
console.log(`[Router:API] 📨 Message ${MessageType.GET_MY_PROCESSES} reçu`);
|
||||||
if (!services.isPaired()) throw new Error('Device not paired');
|
if (!services.isPaired()) throw new Error('Device not paired');
|
||||||
|
|
||||||
const { accessToken } = event.data;
|
await withToken(event, async () => {
|
||||||
if (!accessToken || !(await tokenService.validateToken(accessToken, event.origin))) {
|
|
||||||
throw new Error('Invalid or expired session token');
|
|
||||||
}
|
|
||||||
|
|
||||||
const myProcesses = await services.getMyProcesses();
|
const myProcesses = await services.getMyProcesses();
|
||||||
|
|
||||||
window.parent.postMessage(
|
window.parent.postMessage(
|
||||||
@ -159,17 +167,14 @@ export class IframeController {
|
|||||||
},
|
},
|
||||||
event.origin,
|
event.origin,
|
||||||
);
|
);
|
||||||
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
const handleGetProcesses = async (event: MessageEvent) => {
|
const handleGetProcesses = async (event: MessageEvent) => {
|
||||||
console.log(`[Router:API] 📨 Message ${MessageType.GET_PROCESSES} reçu`);
|
console.log(`[Router:API] 📨 Message ${MessageType.GET_PROCESSES} reçu`);
|
||||||
if (!services.isPaired()) throw new Error('Device not paired');
|
if (!services.isPaired()) throw new Error('Device not paired');
|
||||||
|
|
||||||
const { accessToken } = event.data;
|
await withToken(event, async () => {
|
||||||
if (!accessToken || !(await tokenService.validateToken(accessToken, event.origin))) {
|
|
||||||
throw new Error('Invalid or expired session token');
|
|
||||||
}
|
|
||||||
|
|
||||||
const processes = await services.getProcesses();
|
const processes = await services.getProcesses();
|
||||||
|
|
||||||
window.parent.postMessage(
|
window.parent.postMessage(
|
||||||
@ -180,17 +185,16 @@ export class IframeController {
|
|||||||
},
|
},
|
||||||
event.origin,
|
event.origin,
|
||||||
);
|
);
|
||||||
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
const handleDecryptState = async (event: MessageEvent) => {
|
const handleDecryptState = async (event: MessageEvent) => {
|
||||||
console.log(`[Router:API] 📨 Message ${MessageType.RETRIEVE_DATA} reçu`);
|
console.log(`[Router:API] 📨 Message ${MessageType.RETRIEVE_DATA} reçu`);
|
||||||
if (!services.isPaired()) throw new Error('Device not paired');
|
if (!services.isPaired()) throw new Error('Device not paired');
|
||||||
|
|
||||||
const { processId, stateId, accessToken } = event.data;
|
const { processId, stateId } = event.data;
|
||||||
if (!accessToken || !(await tokenService.validateToken(accessToken, event.origin))) {
|
|
||||||
throw new Error('Invalid or expired session token');
|
|
||||||
}
|
|
||||||
|
|
||||||
|
await withToken(event, async () => {
|
||||||
const process = await services.getProcess(processId);
|
const process = await services.getProcess(processId);
|
||||||
if (!process) throw new Error("Can't find process");
|
if (!process) throw new Error("Can't find process");
|
||||||
|
|
||||||
@ -220,6 +224,7 @@ export class IframeController {
|
|||||||
},
|
},
|
||||||
event.origin,
|
event.origin,
|
||||||
);
|
);
|
||||||
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
const handleValidateToken = async (event: MessageEvent) => {
|
const handleValidateToken = async (event: MessageEvent) => {
|
||||||
@ -295,11 +300,7 @@ export class IframeController {
|
|||||||
throw new Error('Device not paired');
|
throw new Error('Device not paired');
|
||||||
}
|
}
|
||||||
|
|
||||||
const { accessToken } = event.data;
|
await withToken(event, async () => {
|
||||||
if (!accessToken || !(await tokenService.validateToken(accessToken, event.origin))) {
|
|
||||||
throw new Error('Invalid or expired session token');
|
|
||||||
}
|
|
||||||
|
|
||||||
window.parent.postMessage(
|
window.parent.postMessage(
|
||||||
{
|
{
|
||||||
type: MessageType.GET_PAIRING_ID,
|
type: MessageType.GET_PAIRING_ID,
|
||||||
@ -308,17 +309,16 @@ export class IframeController {
|
|||||||
},
|
},
|
||||||
event.origin,
|
event.origin,
|
||||||
);
|
);
|
||||||
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
const handleCreateProcess = async (event: MessageEvent) => {
|
const handleCreateProcess = async (event: MessageEvent) => {
|
||||||
console.log(`[Router:API] 📨 Message ${MessageType.CREATE_PROCESS} reçu`);
|
console.log(`[Router:API] 📨 Message ${MessageType.CREATE_PROCESS} reçu`);
|
||||||
if (!services.isPaired()) throw new Error('Device not paired');
|
if (!services.isPaired()) throw new Error('Device not paired');
|
||||||
|
|
||||||
const { processData, privateFields, roles, accessToken } = event.data;
|
const { processData, privateFields, roles } = event.data;
|
||||||
if (!accessToken || !(await tokenService.validateToken(accessToken, event.origin))) {
|
|
||||||
throw new Error('Invalid or expired session token');
|
|
||||||
}
|
|
||||||
|
|
||||||
|
await withToken(event, async () => {
|
||||||
console.log('[Router:API] 🚀 Démarrage de la création de processus standard...');
|
console.log('[Router:API] 🚀 Démarrage de la création de processus standard...');
|
||||||
const { privateData, publicData } = splitPrivateData(processData, privateFields);
|
const { privateData, publicData } = splitPrivateData(processData, privateFields);
|
||||||
|
|
||||||
@ -350,16 +350,16 @@ export class IframeController {
|
|||||||
},
|
},
|
||||||
event.origin,
|
event.origin,
|
||||||
);
|
);
|
||||||
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
const handleNotifyUpdate = async (event: MessageEvent) => {
|
const handleNotifyUpdate = async (event: MessageEvent) => {
|
||||||
console.log(`[Router:API] 📨 Message ${MessageType.NOTIFY_UPDATE} reçu`);
|
console.log(`[Router:API] 📨 Message ${MessageType.NOTIFY_UPDATE} reçu`);
|
||||||
if (!services.isPaired()) throw new Error('Device not paired');
|
if (!services.isPaired()) throw new Error('Device not paired');
|
||||||
|
|
||||||
const { processId, stateId, accessToken } = event.data;
|
const { processId, stateId } = event.data;
|
||||||
if (!accessToken || !(await tokenService.validateToken(accessToken, event.origin))) {
|
|
||||||
throw new Error('Invalid or expired session token');
|
await withToken(event, async () => {
|
||||||
}
|
|
||||||
if (!isValid32ByteHex(stateId)) throw new Error('Invalid state id');
|
if (!isValid32ByteHex(stateId)) throw new Error('Invalid state id');
|
||||||
|
|
||||||
const res = await services.createPrdUpdate(processId, stateId);
|
const res = await services.createPrdUpdate(processId, stateId);
|
||||||
@ -372,17 +372,16 @@ export class IframeController {
|
|||||||
},
|
},
|
||||||
event.origin,
|
event.origin,
|
||||||
);
|
);
|
||||||
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
const handleValidateState = async (event: MessageEvent) => {
|
const handleValidateState = async (event: MessageEvent) => {
|
||||||
console.log(`[Router:API] 📨 Message ${MessageType.VALIDATE_STATE} reçu`);
|
console.log(`[Router:API] 📨 Message ${MessageType.VALIDATE_STATE} reçu`);
|
||||||
if (!services.isPaired()) throw new Error('Device not paired');
|
if (!services.isPaired()) throw new Error('Device not paired');
|
||||||
|
|
||||||
const { processId, stateId, accessToken } = event.data;
|
const { processId, stateId } = event.data;
|
||||||
if (!accessToken || !(await tokenService.validateToken(accessToken, event.origin))) {
|
|
||||||
throw new Error('Invalid or expired session token');
|
|
||||||
}
|
|
||||||
|
|
||||||
|
await withToken(event, async () => {
|
||||||
const res = await services.approveChange(processId, stateId);
|
const res = await services.approveChange(processId, stateId);
|
||||||
await services.handleApiReturn(res);
|
await services.handleApiReturn(res);
|
||||||
|
|
||||||
@ -394,17 +393,16 @@ export class IframeController {
|
|||||||
},
|
},
|
||||||
event.origin,
|
event.origin,
|
||||||
);
|
);
|
||||||
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
const handleUpdateProcess = async (event: MessageEvent) => {
|
const handleUpdateProcess = async (event: MessageEvent) => {
|
||||||
console.log(`[Router:API] 📨 Message ${MessageType.UPDATE_PROCESS} reçu`);
|
console.log(`[Router:API] 📨 Message ${MessageType.UPDATE_PROCESS} reçu`);
|
||||||
if (!services.isPaired()) throw new Error('Device not paired');
|
if (!services.isPaired()) throw new Error('Device not paired');
|
||||||
|
|
||||||
const { processId, newData, privateFields, roles, accessToken } = event.data;
|
const { processId, newData, privateFields, roles } = event.data;
|
||||||
if (!accessToken || !(await tokenService.validateToken(accessToken, event.origin))) {
|
|
||||||
throw new Error('Invalid or expired session token');
|
|
||||||
}
|
|
||||||
|
|
||||||
|
await withToken(event, async () => {
|
||||||
console.log(`[Router:API] 🔄 Transfert de la mise à jour de ${processId} au service...`);
|
console.log(`[Router:API] 🔄 Transfert de la mise à jour de ${processId} au service...`);
|
||||||
|
|
||||||
// Le service gère maintenant tout : récupération, réparation d'état, et mise à jour.
|
// Le service gère maintenant tout : récupération, réparation d'état, et mise à jour.
|
||||||
@ -422,17 +420,16 @@ export class IframeController {
|
|||||||
},
|
},
|
||||||
event.origin,
|
event.origin,
|
||||||
);
|
);
|
||||||
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
const handleDecodePublicData = async (event: MessageEvent) => {
|
const handleDecodePublicData = async (event: MessageEvent) => {
|
||||||
console.log(`[Router:API] 📨 Message ${MessageType.DECODE_PUBLIC_DATA} reçu`);
|
console.log(`[Router:API] 📨 Message ${MessageType.DECODE_PUBLIC_DATA} reçu`);
|
||||||
if (!services.isPaired()) throw new Error('Device not paired');
|
if (!services.isPaired()) throw new Error('Device not paired');
|
||||||
|
|
||||||
const { accessToken, encodedData } = event.data;
|
const { encodedData } = event.data;
|
||||||
if (!accessToken || !(await tokenService.validateToken(accessToken, event.origin))) {
|
|
||||||
throw new Error('Invalid or expired session token');
|
|
||||||
}
|
|
||||||
|
|
||||||
|
await withToken(event, async () => {
|
||||||
const decodedData = services.decodeValue(encodedData);
|
const decodedData = services.decodeValue(encodedData);
|
||||||
window.parent.postMessage(
|
window.parent.postMessage(
|
||||||
{
|
{
|
||||||
@ -442,15 +439,14 @@ export class IframeController {
|
|||||||
},
|
},
|
||||||
event.origin,
|
event.origin,
|
||||||
);
|
);
|
||||||
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
const handleHashValue = async (event: MessageEvent) => {
|
const handleHashValue = async (event: MessageEvent) => {
|
||||||
console.log(`[Router:API] 📨 Message ${MessageType.HASH_VALUE} reçu`);
|
console.log(`[Router:API] 📨 Message ${MessageType.HASH_VALUE} reçu`);
|
||||||
const { accessToken, commitedIn, label, fileBlob } = event.data;
|
const { commitedIn, label, fileBlob } = event.data;
|
||||||
if (!accessToken || !(await tokenService.validateToken(accessToken, event.origin))) {
|
|
||||||
throw new Error('Invalid or expired session token');
|
|
||||||
}
|
|
||||||
|
|
||||||
|
await withToken(event, async () => {
|
||||||
const hash = services.getHashForFile(commitedIn, label, fileBlob);
|
const hash = services.getHashForFile(commitedIn, label, fileBlob);
|
||||||
window.parent.postMessage(
|
window.parent.postMessage(
|
||||||
{
|
{
|
||||||
@ -460,15 +456,14 @@ export class IframeController {
|
|||||||
},
|
},
|
||||||
event.origin,
|
event.origin,
|
||||||
);
|
);
|
||||||
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
const handleGetMerkleProof = async (event: MessageEvent) => {
|
const handleGetMerkleProof = async (event: MessageEvent) => {
|
||||||
console.log(`[Router:API] 📨 Message ${MessageType.GET_MERKLE_PROOF} reçu`);
|
console.log(`[Router:API] 📨 Message ${MessageType.GET_MERKLE_PROOF} reçu`);
|
||||||
const { accessToken, processState, attributeName } = event.data;
|
const { processState, attributeName } = event.data;
|
||||||
if (!accessToken || !(await tokenService.validateToken(accessToken, event.origin))) {
|
|
||||||
throw new Error('Invalid or expired session token');
|
|
||||||
}
|
|
||||||
|
|
||||||
|
await withToken(event, async () => {
|
||||||
const proof = services.getMerkleProofForFile(processState, attributeName);
|
const proof = services.getMerkleProofForFile(processState, attributeName);
|
||||||
window.parent.postMessage(
|
window.parent.postMessage(
|
||||||
{
|
{
|
||||||
@ -478,15 +473,14 @@ export class IframeController {
|
|||||||
},
|
},
|
||||||
event.origin,
|
event.origin,
|
||||||
);
|
);
|
||||||
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
const handleValidateMerkleProof = async (event: MessageEvent) => {
|
const handleValidateMerkleProof = async (event: MessageEvent) => {
|
||||||
console.log(`[Router:API] 📨 Message ${MessageType.VALIDATE_MERKLE_PROOF} reçu`);
|
console.log(`[Router:API] 📨 Message ${MessageType.VALIDATE_MERKLE_PROOF} reçu`);
|
||||||
const { accessToken, merkleProof, documentHash } = event.data;
|
const { merkleProof, documentHash } = event.data;
|
||||||
if (!accessToken || !(await tokenService.validateToken(accessToken, event.origin))) {
|
|
||||||
throw new Error('Invalid or expired session token');
|
|
||||||
}
|
|
||||||
|
|
||||||
|
await withToken(event, async () => {
|
||||||
let parsedMerkleProof: MerkleProofResult;
|
let parsedMerkleProof: MerkleProofResult;
|
||||||
try {
|
try {
|
||||||
parsedMerkleProof = JSON.parse(merkleProof);
|
parsedMerkleProof = JSON.parse(merkleProof);
|
||||||
@ -503,6 +497,7 @@ export class IframeController {
|
|||||||
},
|
},
|
||||||
event.origin,
|
event.origin,
|
||||||
);
|
);
|
||||||
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
// --- Le "Switchyard" : il reçoit tous les messages et les dispatche ---
|
// --- Le "Switchyard" : il reçoit tous les messages et les dispatche ---
|
||||||
|
|||||||
@ -1,36 +1,53 @@
|
|||||||
import { AnkFlag } from 'pkg/sdk_client';
|
import { AnkFlag } from '../../pkg/sdk_client'; // Vérifie le chemin vers pkg
|
||||||
import Services from './service';
|
import Services from './service';
|
||||||
|
|
||||||
let ws: WebSocket;
|
let ws: WebSocket | null = null;
|
||||||
let messageQueue: string[] = [];
|
let messageQueue: string[] = [];
|
||||||
|
let reconnectInterval = 1000; // Délai initial de 1s avant reconnexion
|
||||||
|
const MAX_RECONNECT_INTERVAL = 30000; // Max 30s
|
||||||
|
let isConnecting = false;
|
||||||
|
let urlReference: string = '';
|
||||||
|
let pingIntervalId: any = null;
|
||||||
|
|
||||||
export async function initWebsocket(url: string) {
|
export async function initWebsocket(url: string) {
|
||||||
ws = new WebSocket(url);
|
urlReference = url;
|
||||||
|
connect();
|
||||||
|
}
|
||||||
|
|
||||||
if (ws !== null) {
|
function connect() {
|
||||||
ws.onopen = async (event) => {
|
if (isConnecting || (ws && ws.readyState === WebSocket.OPEN)) return;
|
||||||
console.log('WebSocket connection established');
|
isConnecting = true;
|
||||||
|
|
||||||
|
console.log(`[WS] 🔌 Tentative de connexion à ${urlReference}...`);
|
||||||
|
ws = new WebSocket(urlReference);
|
||||||
|
|
||||||
|
ws.onopen = async () => {
|
||||||
|
console.log('[WS] ✅ Connexion établie !');
|
||||||
|
isConnecting = false;
|
||||||
|
reconnectInterval = 1000; // Reset du délai
|
||||||
|
|
||||||
|
// Démarrer le Heartbeat (Ping pour garder la connexion vivante)
|
||||||
|
startHeartbeat();
|
||||||
|
|
||||||
|
// Vider la file d'attente (messages envoyés pendant la coupure)
|
||||||
while (messageQueue.length > 0) {
|
while (messageQueue.length > 0) {
|
||||||
const message = messageQueue.shift();
|
const message = messageQueue.shift();
|
||||||
if (message) {
|
if (message) ws?.send(message);
|
||||||
ws.send(message);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
// Listen for messages
|
|
||||||
ws.onmessage = (event) => {
|
ws.onmessage = (event) => {
|
||||||
const msgData = event.data;
|
const msgData = event.data;
|
||||||
|
|
||||||
// console.log("Received text message: ", msgData);
|
|
||||||
(async () => {
|
|
||||||
if (typeof msgData === 'string') {
|
if (typeof msgData === 'string') {
|
||||||
|
(async () => {
|
||||||
try {
|
try {
|
||||||
const parsedMessage = JSON.parse(msgData);
|
const parsedMessage = JSON.parse(msgData);
|
||||||
const services = await Services.getInstance();
|
const services = await Services.getInstance();
|
||||||
|
|
||||||
|
// Gestion des messages
|
||||||
switch (parsedMessage.flag) {
|
switch (parsedMessage.flag) {
|
||||||
case 'Handshake':
|
case 'Handshake':
|
||||||
await services.handleHandshakeMsg(url, parsedMessage.content);
|
await services.handleHandshakeMsg(urlReference, parsedMessage.content);
|
||||||
break;
|
break;
|
||||||
case 'NewTx':
|
case 'NewTx':
|
||||||
await services.parseNewTx(parsedMessage.content);
|
await services.parseNewTx(parsedMessage.content);
|
||||||
@ -39,51 +56,80 @@ export async function initWebsocket(url: string) {
|
|||||||
await services.parseCipher(parsedMessage.content);
|
await services.parseCipher(parsedMessage.content);
|
||||||
break;
|
break;
|
||||||
case 'Commit':
|
case 'Commit':
|
||||||
// Basically if we see this it means we have an error
|
|
||||||
await services.handleCommitError(parsedMessage.content);
|
await services.handleCommitError(parsedMessage.content);
|
||||||
break;
|
break;
|
||||||
|
// Ajoute d'autres cas si nécessaire
|
||||||
|
default:
|
||||||
|
// console.log('[WS] Message reçu:', parsedMessage.flag);
|
||||||
}
|
}
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error('Received an invalid message:', error);
|
console.error('[WS] Erreur traitement message:', error);
|
||||||
}
|
|
||||||
} else {
|
|
||||||
console.error('Received a non-string message');
|
|
||||||
}
|
}
|
||||||
})();
|
})();
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
// Listen for possible errors
|
|
||||||
ws.onerror = (event) => {
|
ws.onerror = (event) => {
|
||||||
console.error('WebSocket error:', event);
|
console.error('[WS] 💥 Erreur:', event);
|
||||||
|
// Pas besoin de reconnecter ici, onclose sera appelé juste après
|
||||||
};
|
};
|
||||||
|
|
||||||
// Listen for when the connection is closed
|
|
||||||
ws.onclose = (event) => {
|
ws.onclose = (event) => {
|
||||||
console.log('WebSocket is closed now.');
|
isConnecting = false;
|
||||||
|
stopHeartbeat();
|
||||||
|
console.warn(`[WS] ⚠️ Déconnecté (Code: ${event.code}). Reconnexion dans ${reconnectInterval / 1000}s...`);
|
||||||
|
|
||||||
|
// Reconnexion exponentielle (1s, 1.5s, 2.25s...)
|
||||||
|
setTimeout(() => {
|
||||||
|
connect();
|
||||||
|
reconnectInterval = Math.min(reconnectInterval * 1.5, MAX_RECONNECT_INTERVAL);
|
||||||
|
}, reconnectInterval);
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function startHeartbeat() {
|
||||||
|
stopHeartbeat();
|
||||||
|
// Envoie un ping toutes les 30 secondes pour éviter que le serveur ou le navigateur ne coupe la connexion
|
||||||
|
pingIntervalId = setInterval(() => {
|
||||||
|
if (ws && ws.readyState === WebSocket.OPEN) {
|
||||||
|
// Adapter selon ce que ton serveur attend comme Ping, ou envoyer un message vide
|
||||||
|
// ws.send(JSON.stringify({ flag: 'Ping', content: '' }));
|
||||||
|
}
|
||||||
|
}, 30000);
|
||||||
|
}
|
||||||
|
|
||||||
|
function stopHeartbeat() {
|
||||||
|
if (pingIntervalId) clearInterval(pingIntervalId);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Method to send messages
|
|
||||||
export function sendMessage(flag: AnkFlag, message: string): void {
|
export function sendMessage(flag: AnkFlag, message: string): void {
|
||||||
if (ws.readyState === WebSocket.OPEN) {
|
if (ws && ws.readyState === WebSocket.OPEN) {
|
||||||
const networkMessage = {
|
const networkMessage = {
|
||||||
flag: flag,
|
flag: flag,
|
||||||
content: message,
|
content: message,
|
||||||
};
|
};
|
||||||
console.log('Sending message of type:', flag);
|
|
||||||
ws.send(JSON.stringify(networkMessage));
|
ws.send(JSON.stringify(networkMessage));
|
||||||
} else {
|
} else {
|
||||||
console.error('WebSocket is not open. ReadyState:', ws.readyState);
|
console.warn(`[WS] Pas connecté. Message '${flag}' mis en file d'attente.`);
|
||||||
messageQueue.push(message);
|
const networkMessage = {
|
||||||
|
flag: flag,
|
||||||
|
content: message,
|
||||||
|
};
|
||||||
|
messageQueue.push(JSON.stringify(networkMessage));
|
||||||
|
|
||||||
|
// Si on n'est pas déjà en train de se connecter, on force une tentative
|
||||||
|
if (!isConnecting) connect();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export function getUrl(): string {
|
export function getUrl(): string {
|
||||||
return ws.url;
|
return urlReference;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Method to close the WebSocket connection
|
|
||||||
export function close(): void {
|
export function close(): void {
|
||||||
|
if (ws) {
|
||||||
|
ws.onclose = null; // On évite la reconnexion auto si fermeture volontaire
|
||||||
|
stopHeartbeat();
|
||||||
ws.close();
|
ws.close();
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user