diff --git a/robot_cobros.js b/robot_cobros.js index db261d5..7028fe7 100644 --- a/robot_cobros.js +++ b/robot_cobros.js @@ -59,7 +59,7 @@ app.post('/api/robot-cobros', async (req, res) => { }); // ================================================================== -// 🤖 LÓGICA MULTIASISTENCIA V15 (SOPORTE FRAMES) +// 🤖 LÓGICA MULTIASISTENCIA V16 (HÍBRIDO LISTA TEXTO + PRECIO FRAMES) // ================================================================== async function runMultiStream(mes, anio, send) { @@ -68,7 +68,7 @@ async function runMultiStream(mes, anio, send) { const valD2 = `${anio}`; try { - send('LOG', `🚀 (MULTI) Iniciando para: ${valD1}`); + send('LOG', `🚀 (MULTI V16) Iniciando para: ${valD1}`); const { browser: b, page } = await loginMulti(); browser = b; @@ -80,26 +80,32 @@ async function runMultiStream(mes, anio, send) { await page.selectOption('select[name="D1"]', valD1); await page.selectOption('select[name="D2"]', valD2); await page.click('input[name="continuar"]'); - await page.waitForTimeout(2500); + await page.waitForTimeout(2000); } - // --- FASE 1: RECOLECTAR IDs --- + // --- FASE 1: RECOLECTAR IDs (LÓGICA V13 QUE SÍ FUNCIONABA) --- let idsServicios = []; let tieneSiguiente = true; let pagActual = 1; while (tieneSiguiente && pagActual <= 10) { send('LOG', `📄 Leyendo página ${pagActual}...`); + + // BUSCAR TEXTO EN CELDAS (NO SOLO ENLACES) const nuevosIds = await page.evaluate(() => { - const links = Array.from(document.querySelectorAll('a[href*="reparacion="]')); - return links.map(a => { - const m = a.href.match(/reparacion=(\d+)/); - return m ? m[1] : null; - }).filter(id => id && /^\d{8}$/.test(id)); + const celdas = Array.from(document.querySelectorAll('td.tdet')); + const lista = []; + celdas.forEach(td => { + const txt = td.innerText.trim(); + // Buscamos cualquier número de 8 cifras + if (/^\d{8}$/.test(txt)) lista.push(txt); + }); + return lista; }); - idsServicios.push(...[...new Set(nuevosIds)]); - send('LOG', ` -> Encontrados ${nuevosIds.length} servicios.`); + const unicosPag = [...new Set(nuevosIds)]; + idsServicios.push(...unicosPag); + send('LOG', ` -> Encontrados ${unicosPag.length} servicios.`); const haySiguiente = await page.$('a:has-text("Página siguiente")'); if (haySiguiente) { await haySiguiente.click(); await page.waitForTimeout(3000); pagActual++; } @@ -109,7 +115,9 @@ async function runMultiStream(mes, anio, send) { idsServicios = [...new Set(idsServicios)]; send('LOG', `🔍 Total a revisar: ${idsServicios.length}`); - // --- FASE 2: LEER DETALLES (CON FRAMES) --- + if (idsServicios.length === 0) return { encontrados: [], noEncontrados: [] }; + + // --- FASE 2: LEER DETALLES (CON FRAMES V15) --- for (const [index, idServicio] of idsServicios.entries()) { send('PROGRESS', { current: index + 1, total: idsServicios.length }); @@ -117,40 +125,48 @@ async function runMultiStream(mes, anio, send) { try { await page.goto(urlPresupuesto, { timeout: 60000, waitUntil: 'domcontentloaded' }); - await page.waitForTimeout(1000); // Esperar renderizado base + await page.waitForTimeout(1500); // Esperar renderizado // === BÚSQUEDA PROFUNDA EN FRAMES === let totalStr = ""; let importe = 0; - // 1. Iterar sobre todos los frames (marcos) de la web + // 1. Iterar sobre todos los frames const frames = page.frames(); for (const frame of frames) { try { - // Buscamos el elemento exacto que me pasaste const text = await frame.evaluate(() => { - const el = document.querySelector('td[data-label="TOTAL REPARACION:"]'); // Selector exacto + // Selector flexible: busca data-label que contenga "TOTAL" + const el = document.querySelector('td[data-label*="TOTAL"]'); return el ? el.innerText : null; }); if (text) { - totalStr = text; // "53,66 €" - break; // ¡Encontrado! Dejamos de buscar + totalStr = text; + break; } - } catch(e) { /* Error en frame, seguimos al siguiente */ } + } catch(e) { } } - // 2. Limpieza del precio + // 2. Si no encuentra en frames, busca en la página principal (por si acaso) + if (!totalStr) { + const infoPage = await page.evaluate(() => { + const el = document.querySelector('td[data-label*="TOTAL"]'); + return el ? el.innerText : ""; + }); + totalStr = infoPage; + } + + // 3. Limpieza precio if (totalStr) { - // "53,66 €" -> "53.66" const cleanMoney = totalStr.replace(/[^\d,]/g, '').replace(',', '.'); importe = parseFloat(cleanMoney) || 0; } - // 3. CONSULTAR BASE DE DATOS Y FORZAR NOMBRE + // 4. CONSULTAR BASE DE DATOS let docId = null; let enBD = false; - let clienteFinal = "Desconocido"; // Fallback por defecto + let clienteFinal = "Desconocido"; let direccionFinal = "Sin dirección"; if (db) { @@ -159,16 +175,13 @@ async function runMultiStream(mes, anio, send) { const doc = q.docs[0]; docId = doc.id; enBD = true; - - const datosBD = doc.data(); - - // PRIORIDAD ABSOLUTA A LOS DATOS DE FIREBASE - if (datosBD.clientName) clienteFinal = datosBD.clientName; - if (datosBD.address) direccionFinal = datosBD.address; + const d = doc.data(); + if (d.clientName) clienteFinal = d.clientName; + if (d.address) direccionFinal = d.address; } } - // ENVIAR (Verde o Amarillo) + // ENVIAR send('ITEM_FOUND', { servicio: idServicio, direccion: direccionFinal, @@ -215,7 +228,6 @@ async function loginMulti() { return { browser, page }; } -// ... RESTO DE FUNCIONES (HS, Saver) NO CAMBIAN ... async function runHSScanner() { let browser = null; try {