Version 4.2.1 — Démarrage série 4.2
This commit is contained in:
+195
-59
@@ -7,8 +7,8 @@
|
||||
// - fetchPlanning : fetch le XML du planning pour une date (1 requête = tout)
|
||||
// - fetchXhr2 : fetch un texte d'action détaillé (utilisé en lazy-load au survol)
|
||||
// - fetchFiche : fetch une fiche individuelle (HTML) pour statut + commentaire tech
|
||||
// 3. Programmer les alarmes de refresh auto (12h, 15h)
|
||||
// 4. Nettoyer les vieux caches (>7 jours)
|
||||
// 3. Nettoyer les vieux caches (>7 jours)
|
||||
// (v4.2 : l'auto-refresh 12h/15h a été retiré)
|
||||
//
|
||||
// v4 : suppression de fetchTimeline (pu utilisé). Le calendar_block contient
|
||||
// directement ref/contact/lieu/catégorie dans ses attributs attr1/attr2/attr3,
|
||||
@@ -87,12 +87,30 @@ async function fetchPlanningXml(origin, phpsessid, unixDate) {
|
||||
console.log("[bg] fetchPlanningXml →", url.substring(0, 140));
|
||||
const r = await fetch(url, { credentials: "include" });
|
||||
console.log("[bg] status =", r.status);
|
||||
if (!r.ok) throw new Error("HTTP " + r.status);
|
||||
if (!r.ok) {
|
||||
// v4.2 : classifier l'erreur HTTP pour que le viewer affiche le bon
|
||||
// écran (session expirée vs EV inaccessible).
|
||||
const err = new Error("HTTP " + r.status);
|
||||
err.kind = classifyHttpStatus(r.status);
|
||||
err.status = r.status;
|
||||
throw err;
|
||||
}
|
||||
const xml = await r.text();
|
||||
console.log("[bg] taille XML =", xml.length);
|
||||
return xml;
|
||||
}
|
||||
|
||||
/**
|
||||
* v4.2 : classifie un statut HTTP comme "session_expired" ou "ev_unreachable".
|
||||
* - 401, 403, 404 → session_expired (EV renvoie souvent 404 au lieu de rediriger
|
||||
* vers la page de login quand PHPSESSID n'est plus valide)
|
||||
* - 5xx, autres → ev_unreachable (service down, surcharge, etc.)
|
||||
*/
|
||||
function classifyHttpStatus(status) {
|
||||
if (status === 401 || status === 403 || status === 404) return "session_expired";
|
||||
return "ev_unreachable";
|
||||
}
|
||||
|
||||
/**
|
||||
* Fetch planning_xhr_2.php?id=ACTIONID pour UNE intervention.
|
||||
* Retourne ~400 octets au format custom :
|
||||
@@ -101,7 +119,12 @@ async function fetchPlanningXml(origin, phpsessid, unixDate) {
|
||||
async function fetchXhr2(origin, phpsessid, actionId) {
|
||||
const url = `${origin}/planning_xhr_2.php?PHPSESSID=${encodeURIComponent(phpsessid)}&id=${encodeURIComponent(actionId)}`;
|
||||
const r = await fetch(url, { credentials: "include" });
|
||||
if (!r.ok) throw new Error("HTTP " + r.status);
|
||||
if (!r.ok) {
|
||||
const err = new Error("HTTP " + r.status);
|
||||
err.kind = classifyHttpStatus(r.status);
|
||||
err.status = r.status;
|
||||
throw err;
|
||||
}
|
||||
return await r.text();
|
||||
}
|
||||
|
||||
@@ -109,7 +132,12 @@ async function fetchFicheHtml(origin, phpsessid, formLink) {
|
||||
const url = `${origin}/index.php?${formLink}&PHPSESSID=${encodeURIComponent(phpsessid)}`;
|
||||
console.log("[bg] fetchFicheHtml →", url.substring(0, 120));
|
||||
const r = await fetch(url, { credentials: "include" });
|
||||
if (!r.ok) throw new Error("HTTP " + r.status);
|
||||
if (!r.ok) {
|
||||
const err = new Error("HTTP " + r.status);
|
||||
err.kind = classifyHttpStatus(r.status);
|
||||
err.status = r.status;
|
||||
throw err;
|
||||
}
|
||||
const html = await r.text();
|
||||
console.log("[bg] fiche status =", r.status, "| taille =", html.length);
|
||||
return html;
|
||||
@@ -136,7 +164,12 @@ async function fetchTimelineApi(origin, phpsessid, guid, formId, formChecksum) {
|
||||
`&type=todo§ionId=1&navigator=&nbRecord=0` +
|
||||
`&PHPSESSID=${encodeURIComponent(phpsessid)}`;
|
||||
const r = await fetch(url, { credentials: "include" });
|
||||
if (!r.ok) throw new Error("HTTP " + r.status);
|
||||
if (!r.ok) {
|
||||
const err = new Error("HTTP " + r.status);
|
||||
err.kind = classifyHttpStatus(r.status);
|
||||
err.status = r.status;
|
||||
throw err;
|
||||
}
|
||||
return await r.text();
|
||||
}
|
||||
|
||||
@@ -149,6 +182,93 @@ function looksLikeLoginPage(text) {
|
||||
return /customer_login|my\.policy/i.test((text || "").substring(0, 3000));
|
||||
}
|
||||
|
||||
// ============================================================================
|
||||
// v4.2 : récupération de l'utilisateur connecté
|
||||
// ============================================================================
|
||||
|
||||
/**
|
||||
* Essaie de récupérer le nom de l'utilisateur EasyVista connecté en fetchant
|
||||
* la page d'accueil avec la session active. EasyVista n'exposant pas
|
||||
* d'endpoint public simple, on cherche des patterns typiques dans le HTML :
|
||||
* - <title>...Nom, Prénom...</title>
|
||||
* - éléments avec data-user-name, data-user-login
|
||||
* - balises cachées ou variables JS EV.User.name
|
||||
* - champ "Bienvenue Nom Prénom"
|
||||
* Retourne { name: "Nom Prénom" | null, login: "..." | null } ou null si
|
||||
* tout a échoué.
|
||||
*/
|
||||
async function fetchCurrentUser(origin, phpsessid) {
|
||||
const url = `${origin}/index.php?PHPSESSID=${encodeURIComponent(phpsessid)}`;
|
||||
const resp = await fetch(url, {
|
||||
method: "GET",
|
||||
credentials: "include",
|
||||
headers: { "Accept": "text/html,*/*" }
|
||||
});
|
||||
// v4.2 : cette fonction est lancée en tâche de fond au démarrage. Si la
|
||||
// session est expirée ou EV inaccessible, on retourne juste null — le
|
||||
// planning lui-même déclenchera l'écran d'erreur approprié.
|
||||
if (!resp.ok) return null;
|
||||
const html = await resp.text();
|
||||
if (looksLikeLoginPage(html)) return null;
|
||||
|
||||
// Essais de patterns (du plus spécifique au plus générique)
|
||||
const patterns = [
|
||||
// Attribut data-user-name (si EasyVista l'expose)
|
||||
/data-user-name\s*=\s*["']([^"']+)["']/i,
|
||||
/data-username\s*=\s*["']([^"']+)["']/i,
|
||||
/data-user-fullname\s*=\s*["']([^"']+)["']/i,
|
||||
// Variable JS typique EasyVista
|
||||
/EV\.User\.name\s*=\s*["']([^"']+)["']/,
|
||||
/EV\.User\.fullname\s*=\s*["']([^"']+)["']/,
|
||||
/userFullName\s*[:=]\s*["']([^"']+)["']/,
|
||||
/currentUser(?:Name)?\s*[:=]\s*["']([^"']+)["']/,
|
||||
// Balises cachées ou spans avec classe "user"
|
||||
/<(?:span|div)[^>]*class=["'][^"']*(?:user[_-]?(?:name|full|display))[^"']*["'][^>]*>([^<]{2,80})<\/(?:span|div)>/i,
|
||||
// "Bienvenue" / "Welcome"
|
||||
/(?:Bienvenue|Welcome)[,\s]+(?:M\.?\s+|Mme\s+)?([A-ZÀÁÂÄÇÉÈÊËÍÎÏÓÔÖÚÛÜÑ][\wÀ-ÿ'.\-]+(?:\s*,?\s+[A-ZÀÁÂÄÇÉÈÊËÍÎÏÓÔÖÚÛÜÑ][\wÀ-ÿ'.\-]+){0,3})/,
|
||||
// Title de la page (souvent "EasyVista - Nom Prénom")
|
||||
/<title>([^<]*)<\/title>/i
|
||||
];
|
||||
|
||||
let name = null;
|
||||
for (const rx of patterns) {
|
||||
const m = html.match(rx);
|
||||
if (m && m[1]) {
|
||||
const candidate = m[1].trim()
|
||||
.replace(/\s+/g, " ")
|
||||
// Enlever des éléments du <title> type "EasyVista" / "Planning" / etc.
|
||||
.replace(/^(?:EasyVista|EV|Accueil|Home|Planning|ITSMA)[\s\-|•]+/i, "")
|
||||
.replace(/[\s\-|•]+(?:EasyVista|EV|ITSMA)$/i, "")
|
||||
.trim();
|
||||
if (candidate && candidate.length >= 3 && candidate.length <= 80
|
||||
&& /[A-Za-zÀ-ÿ]/.test(candidate)
|
||||
&& !/\b(login|connexion|sign\s*in|easyvista|ITSMA)\b/i.test(candidate)) {
|
||||
name = candidate;
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Chercher aussi le login (ID court) — utile comme fallback secondaire
|
||||
let login = null;
|
||||
const loginPatterns = [
|
||||
/data-user-login\s*=\s*["']([^"']+)["']/i,
|
||||
/data-login\s*=\s*["']([^"']+)["']/i,
|
||||
/EV\.User\.login\s*=\s*["']([^"']+)["']/,
|
||||
/userLogin\s*[:=]\s*["']([^"']+)["']/
|
||||
];
|
||||
for (const rx of loginPatterns) {
|
||||
const m = html.match(rx);
|
||||
if (m && m[1]) {
|
||||
login = m[1].trim();
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
if (!name && !login) return null;
|
||||
return { name, login };
|
||||
}
|
||||
|
||||
// ============================================================================
|
||||
// Messages du viewer
|
||||
// ============================================================================
|
||||
@@ -168,13 +288,21 @@ chrome.runtime.onMessage.addListener((msg, sender, sendResponse) => {
|
||||
sendResponse({ ok: false, error: "no_session" });
|
||||
return;
|
||||
}
|
||||
// Fetch XML calendar_block du planning (rapide ~40 ko)
|
||||
const xml = await fetchPlanningXml(session.origin, session.phpsessid, msg.unixDate);
|
||||
if (looksLikeLoginPage(xml)) {
|
||||
sendResponse({ ok: false, error: "session_expired" });
|
||||
return;
|
||||
try {
|
||||
// Fetch XML calendar_block du planning (rapide ~40 ko)
|
||||
const xml = await fetchPlanningXml(session.origin, session.phpsessid, msg.unixDate);
|
||||
if (looksLikeLoginPage(xml)) {
|
||||
sendResponse({ ok: false, error: "session_expired" });
|
||||
return;
|
||||
}
|
||||
sendResponse({ ok: true, xml, session });
|
||||
} catch (err) {
|
||||
// v4.2 : classification de l'erreur pour afficher le bon écran
|
||||
const errorCode = err.kind || (
|
||||
/network|fetch|typeerror/i.test(err.message) ? "ev_unreachable" : "ev_unreachable"
|
||||
);
|
||||
sendResponse({ ok: false, error: errorCode, httpStatus: err.status, detail: err.message });
|
||||
}
|
||||
sendResponse({ ok: true, xml, session });
|
||||
return;
|
||||
}
|
||||
|
||||
@@ -188,7 +316,12 @@ chrome.runtime.onMessage.addListener((msg, sender, sendResponse) => {
|
||||
const body = await fetchXhr2(session.origin, session.phpsessid, msg.actionId);
|
||||
sendResponse({ ok: true, body });
|
||||
} catch (err) {
|
||||
sendResponse({ ok: false, error: String(err) });
|
||||
sendResponse({
|
||||
ok: false,
|
||||
error: err.kind || "fetch_failed",
|
||||
httpStatus: err.status,
|
||||
detail: err.message || String(err)
|
||||
});
|
||||
}
|
||||
return;
|
||||
}
|
||||
@@ -199,12 +332,21 @@ chrome.runtime.onMessage.addListener((msg, sender, sendResponse) => {
|
||||
sendResponse({ ok: false, error: "no_session" });
|
||||
return;
|
||||
}
|
||||
const html = await fetchFicheHtml(session.origin, session.phpsessid, msg.formLink);
|
||||
if (looksLikeLoginPage(html)) {
|
||||
sendResponse({ ok: false, error: "session_expired" });
|
||||
return;
|
||||
try {
|
||||
const html = await fetchFicheHtml(session.origin, session.phpsessid, msg.formLink);
|
||||
if (looksLikeLoginPage(html)) {
|
||||
sendResponse({ ok: false, error: "session_expired" });
|
||||
return;
|
||||
}
|
||||
sendResponse({ ok: true, html, session });
|
||||
} catch (err) {
|
||||
sendResponse({
|
||||
ok: false,
|
||||
error: err.kind || "fetch_failed",
|
||||
httpStatus: err.status,
|
||||
detail: err.message || String(err)
|
||||
});
|
||||
}
|
||||
sendResponse({ ok: true, html, session });
|
||||
return;
|
||||
}
|
||||
|
||||
@@ -225,14 +367,32 @@ chrome.runtime.onMessage.addListener((msg, sender, sendResponse) => {
|
||||
}
|
||||
sendResponse({ ok: true, body });
|
||||
} catch (err) {
|
||||
sendResponse({ ok: false, error: String(err) });
|
||||
sendResponse({
|
||||
ok: false,
|
||||
error: err.kind || "fetch_failed",
|
||||
httpStatus: err.status,
|
||||
detail: err.message || String(err)
|
||||
});
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
if (msg.type === "scheduleAutoRefresh") {
|
||||
scheduleAutoRefreshAlarms();
|
||||
sendResponse({ ok: true });
|
||||
if (msg.type === "fetchCurrentUser") {
|
||||
// v4.2 : essaie d'identifier l'utilisateur EasyVista connecté en
|
||||
// fetchant la page d'accueil et en cherchant dans le HTML un champ
|
||||
// contenant son nom. Si on trouve rien, on renvoie { ok: true,
|
||||
// user: null } pour que l'UI sache qu'on n'a pas pu.
|
||||
const session = await findEasyVistaSession();
|
||||
if (!session) {
|
||||
sendResponse({ ok: false, error: "no_session" });
|
||||
return;
|
||||
}
|
||||
try {
|
||||
const user = await fetchCurrentUser(session.origin, session.phpsessid);
|
||||
sendResponse({ ok: true, user });
|
||||
} catch (err) {
|
||||
sendResponse({ ok: false, error: String(err) });
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
@@ -254,45 +414,21 @@ chrome.runtime.onMessage.addListener((msg, sender, sendResponse) => {
|
||||
});
|
||||
|
||||
// ============================================================================
|
||||
// Alarmes : refresh auto 12h / 15h
|
||||
// v4.2 : les alarmes d'auto-refresh 12h/15h ont été supprimées. Seul le
|
||||
// nettoyage quotidien des caches > 7 jours reste.
|
||||
// On supprime aussi activement les anciennes alarmes créées par les
|
||||
// versions précédentes pour éviter qu'elles restent programmées.
|
||||
// ============================================================================
|
||||
|
||||
function scheduleAutoRefreshAlarms() {
|
||||
// Calculer le prochain 12h et 15h à partir de maintenant
|
||||
const now = new Date();
|
||||
|
||||
function nextAt(hour, minute) {
|
||||
const d = new Date();
|
||||
d.setHours(hour, minute, 0, 0);
|
||||
if (d <= now) d.setDate(d.getDate() + 1);
|
||||
return d.getTime();
|
||||
async function clearLegacyRefreshAlarms() {
|
||||
try {
|
||||
await chrome.alarms.clear("refresh_12h");
|
||||
await chrome.alarms.clear("refresh_15h");
|
||||
} catch (e) {
|
||||
console.warn("clearLegacyRefreshAlarms:", e);
|
||||
}
|
||||
|
||||
chrome.alarms.create("refresh_12h", {
|
||||
when: nextAt(12, 0),
|
||||
periodInMinutes: 24 * 60 // tous les jours
|
||||
});
|
||||
chrome.alarms.create("refresh_15h", {
|
||||
when: nextAt(15, 0),
|
||||
periodInMinutes: 24 * 60
|
||||
});
|
||||
}
|
||||
|
||||
chrome.alarms.onAlarm.addListener(async (alarm) => {
|
||||
if (alarm.name === "refresh_12h" || alarm.name === "refresh_15h") {
|
||||
// Envoyer un message à tous les viewers ouverts pour qu'ils se rafraîchissent
|
||||
const viewerUrl = chrome.runtime.getURL("viewer.html");
|
||||
const tabs = await chrome.tabs.query({ url: viewerUrl + "*" });
|
||||
for (const tab of tabs) {
|
||||
try {
|
||||
await chrome.tabs.sendMessage(tab.id, { type: "autoRefresh" });
|
||||
} catch {
|
||||
// Onglet fermé ou pas réactif, on ignore
|
||||
}
|
||||
}
|
||||
}
|
||||
});
|
||||
|
||||
// ============================================================================
|
||||
// Nettoyage caches > 7 jours
|
||||
// ============================================================================
|
||||
@@ -317,13 +453,13 @@ async function cleanupOldCaches(daysToKeep) {
|
||||
return toRemove.length;
|
||||
}
|
||||
|
||||
// Au démarrage, programmer les alarmes et nettoyer
|
||||
// Au démarrage, nettoyer les anciennes alarmes et les anciens caches
|
||||
chrome.runtime.onInstalled.addListener(() => {
|
||||
scheduleAutoRefreshAlarms();
|
||||
clearLegacyRefreshAlarms();
|
||||
cleanupOldCaches(7).catch(err => console.warn("cleanup:", err));
|
||||
});
|
||||
|
||||
chrome.runtime.onStartup.addListener(() => {
|
||||
scheduleAutoRefreshAlarms();
|
||||
clearLegacyRefreshAlarms();
|
||||
cleanupOldCaches(7).catch(err => console.warn("cleanup:", err));
|
||||
});
|
||||
|
||||
Reference in New Issue
Block a user