cobros-hs/robot_cobros.js

214 lines
8.0 KiB
JavaScript

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}`));