const express = require('express'); const { chromium } = require('playwright'); const admin = require('firebase-admin'); const cors = require('cors'); // --- CONFIGURACIÓN FIREBASE --- try { if (process.env.FIREBASE_PRIVATE_KEY) { if (!admin.apps.length) { admin.initializeApp({ credential: admin.credential.cert({ projectId: process.env.FIREBASE_PROJECT_ID, clientEmail: process.env.FIREBASE_CLIENT_EMAIL, privateKey: process.env.FIREBASE_PRIVATE_KEY.replace(/\\n/g, '\n'), }), }); } console.log("✅ Firebase inicializado."); } } catch (e) { console.error("❌ Error Firebase:", e.message); } const db = admin.apps.length ? admin.firestore() : null; const APPOINTMENTS_COL = "appointments"; const app = express(); app.use(cors({ origin: '*' })); app.use(express.json()); // --- ENDPOINT UNIFICADO --- app.post('/api/robot-cobros', async (req, res) => { const { action, urls } = req.body; // action: 'scan' o 'process' console.log(`🔔 Orden recibida: ${action.toUpperCase()}`); try { if (action === 'scan') { const lista = await runScanner(); res.json({ success: true, mode: 'scan', data: lista }); } else if (action === 'process') { if (!urls || urls.length === 0) throw new Error("No has seleccionado ninguna fecha."); const procesados = await runProcessor(urls); res.json({ success: true, mode: 'process', count: procesados }); } else { throw new Error("Acción no válida"); } } catch (err) { console.error("❌ Error:", err.message); res.status(500).json({ success: false, message: err.message }); } }); // --- FUNCIÓN 1: ESCANEAR FECHAS DISPONIBLES --- async function runScanner() { let browser = null; try { const { browser: b, page } = await loginAndGoToLiquidaciones(); browser = b; console.log("🔍 Buscando lista de liquidaciones..."); // Extraemos todos los enlaces de fechas const liquidaciones = await page.evaluate(() => { const links = Array.from(document.querySelectorAll('a')); const results = []; links.forEach(l => { const txt = l.innerText.trim(); // Filtramos por formato fecha (DD/MM/YYYY) o similar if (/\d{2}\/\d{2}\/\d{4}/.test(txt)) { // Guardamos la URL absoluta para navegar directo luego results.push({ fecha: txt, url: l.href }); } }); return results; }); console.log(`✅ Encontradas ${liquidaciones.length} fechas.`); return liquidaciones; } catch (e) { throw e; } finally { if(browser) await browser.close(); } } // --- FUNCIÓN 2: PROCESAR SELECCIONADAS --- async function runProcessor(urlsAProcesar) { let browser = null; let totalActualizados = 0; try { const { browser: b, page } = await loginAndGoToLiquidaciones(); // Login inicial browser = b; for (const targetUrl of urlsAProcesar) { console.log(`➡️ Procesando URL: ${targetUrl}`); // 1. Navegar a la fecha específica await page.goto(targetUrl, { timeout: 60000 }); await page.waitForTimeout(2000); // 2. Buscar botón "Desglose" y pulsar si existe const hayDesglose = await page.evaluate(() => { const btn = Array.from(document.querySelectorAll('input[type="button"], button, a')) .find(el => (el.value || el.innerText || "").toLowerCase().includes('desglose')); if(btn) { btn.click(); return true; } return false; }); if(hayDesglose) { console.log(" 🔘 Entrando al Desglose..."); await page.waitForTimeout(3000); } // 3. LEER LA TABLA (EXTRAER SALDO) const datosTabla = await page.evaluate(() => { const filas = Array.from(document.querySelectorAll('tr')); const datos = []; filas.forEach(tr => { const tds = tr.querySelectorAll('td'); // Según tu saldo.html: // Col 0: Servicio | Col 5 (Sexta columna): Saldo if (tds.length >= 6) { const servicio = tds[0].innerText.trim(); const saldoRaw = tds[5].innerText.trim(); // <--- COLUMNA SALDO // Validar que sea un servicio numérico if (/^\d{5,}$/.test(servicio)) { datos.push({ servicio, saldoRaw }); } } }); return datos; }); console.log(` 💰 Leídos ${datosTabla.length} servicios. Guardando en Firebase...`); // 4. GUARDAR EN FIREBASE if (db && datosTabla.length > 0) { const batch = db.batch(); let batchCount = 0; const nowISO = new Date().toISOString(); for (const item of datosTabla) { // Limpieza: "-33.48" -> 33.48 (Positivo) let cleanSaldo = item.saldoRaw.replace(/[^\d.-]/g, ''); // Deja solo numeros, punto y menos let importe = parseFloat(cleanSaldo); if (isNaN(importe)) continue; // IMPORTANTE: Convertimos a positivo para paidAmount // Si prefieres negativo, quita Math.abs() importe = Math.abs(importe); // Buscar el documento en appointments const q = await db.collection(APPOINTMENTS_COL).where("serviceNumber", "==", item.servicio).get(); q.forEach(doc => { batch.update(doc.ref, { paidAmount: importe, // <--- GUARDAMOS EL SALDO AQUÍ status: 'completed', // Asumimos completado paymentDate: nowISO, homeservePaymentStatus: 'paid_saldo', lastUpdatedByRobot: nowISO }); batchCount++; }); } if (batchCount > 0) { await batch.commit(); totalActualizados += batchCount; } } } return totalActualizados; } catch (e) { throw e; } finally { if(browser) await browser.close(); } } // --- AUXILIAR LOGIN --- async function loginAndGoToLiquidaciones() { let user = "SIN_USER", pass = "SIN_PASS"; if (db) { const doc = await db.collection("providerCredentials").doc("homeserve").get(); if (doc.exists) { user = doc.data().user; pass = doc.data().pass; } } const browser = await chromium.launch({ headless: true, args: ['--no-sandbox'] }); const page = await browser.newPage(); console.log("🌍 Login..."); await page.goto('https://www.clientes.homeserve.es/cgi-bin/fccgi.exe?w3exec=PROF_PASS', { timeout: 60000 }); if (await page.isVisible('input[name="CODIGO"]')) { await page.fill('input[name="CODIGO"]', user); await page.fill('input[type="password"]', pass); await page.keyboard.press('Enter'); await page.waitForTimeout(4000); } console.log("📂 Sección Liquidaciones..."); await page.goto('https://www.clientes.homeserve.es/cgi-bin/fccgi.exe?w3exec=CONSULTALIQ_WEB'); await page.waitForTimeout(2000); return { browser, page }; } const PORT = process.env.PORT || 3000; app.listen(PORT, () => console.log(`🚀 Server on port ${PORT}`));