import { Router, Request, Response } from "express"; import { storage } from "../storage"; import { forwardOtpToSeleniumDentaQuestAgent } from "../services/seleniumDentaQuestEligibilityClient"; import { io } from "../socket"; import { enqueueSeleniumJob } from "../queue/jobRunner"; const router = Router(); function log(tag: string, msg: string, ctx?: any) { console.log(`${new Date().toISOString()} [${tag}] ${msg}`, ctx ?? ""); } function emitSafe(socketId: string | undefined, event: string, payload: any) { if (!socketId || !io) return; try { const socket = io.sockets.sockets.get(socketId); if (socket) socket.emit(event, payload); } catch (err: any) { log("socket", "emit failed", { socketId, event, err: err?.message }); } } router.post( "/tuftssco-eligibility", async (req: Request, res: Response): Promise => { if (!req.body.data) { return res.status(400).json({ error: "Missing eligibility data for selenium" }); } if (!req.user?.id) { return res.status(401).json({ error: "Unauthorized: user info missing" }); } try { const rawData = typeof req.body.data === "string" ? JSON.parse(req.body.data) : req.body.data; const credentials = await storage.getInsuranceCredentialByUserAndSiteKey( req.user.id, rawData.insuranceSiteKey ); if (!credentials) { return res.status(404).json({ error: "No credentials found for Tufts SCO. Please add them on the Settings page.", }); } const enrichedData = { ...rawData, dentaquestUsername: credentials.username, dentaquestPassword: credentials.password, }; const socketId: string | undefined = req.body.socketId; const jobId = enqueueSeleniumJob({ jobType: "tuftssco-eligibility-check", userId: req.user.id, socketId, enrichedPayload: enrichedData, insuranceId: String(rawData.memberId ?? "").trim(), formFirstName: rawData.firstName, formLastName: rawData.lastName, formDob: rawData.dateOfBirth, }); log("tuftssco-route", "job enqueued", { jobId, insuranceId: rawData.memberId }); return res.json({ status: "queued", jobId }); } catch (err: any) { console.error("[tuftssco-route] enqueue failed:", err); return res.status(500).json({ error: err.message || "Failed to enqueue Tufts SCO selenium job", }); } } ); router.post( "/selenium/submit-otp", async (req: Request, res: Response): Promise => { const { session_id: sessionId, otp, socketId } = req.body; if (!sessionId || !otp) { return res.status(400).json({ error: "session_id and otp are required" }); } try { const r = await forwardOtpToSeleniumDentaQuestAgent(sessionId, otp); emitSafe(socketId, "selenium:otp_submitted", { session_id: sessionId, result: r, }); return res.json(r); } catch (err: any) { console.error( "[tuftssco-route] submit-otp failed:", err?.response?.data || err?.message || err ); return res.status(500).json({ error: "Failed to forward OTP to selenium agent", detail: err?.message || err, }); } } ); export default router;