- Add /dentaquest-eligibility endpoint in Python agent (Tufts SCO uses providers.dentaquest.com) - Add backend route, processor, and service client for Tufts SCO (separate from UnitedSCO/DentalHub) - Fix Tufts SCO button to post to new tuftssco route instead of unitedsco - Fix credential field names: dentaquestUsername/Password (was tuftsscoUsername/Password) - Fix socket event: listen for selenium:dentaquest_session_started (was unitedsco) - Fix error visibility: keep session alive 30s on error so backend reads real message - Replace free-text Site Key field with dropdown to prevent key mismatches Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
109 lines
3.2 KiB
TypeScript
109 lines
3.2 KiB
TypeScript
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<any> => {
|
|
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<any> => {
|
|
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;
|