feat: deploy to Hostinger VPS, switch to global_healthx Ozonetel account

- Add helix.svg and PNG favicon (generated via nano-banana)
- Update page title to "Helix Engage" with proper meta tags
- Make seed scripts configurable via SEED_GQL/SEED_ORIGIN env vars
- Support remote workspace member IDs in seed-data.ts
- Dynamic doctor-to-clinic linking in seed-new-entities.ts (fetch IDs from platform)
- Remove deprecated branchClinic field from seed data
- Fix TypeScript errors: callNotes null vs undefined, Lead type casting

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
This commit is contained in:
2026-03-19 10:27:25 +05:30
parent 526ad18159
commit 44bd221108
13 changed files with 43 additions and 49 deletions

View File

@@ -2,16 +2,17 @@
<html lang="en">
<head>
<meta charset="UTF-8" />
<link rel="icon" type="image/svg+xml" href="/vite.svg" />
<link rel="icon" type="image/png" href="/favicon-32.png" />
<link rel="apple-touch-icon" href="/helix-512.png" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<meta name="description" content="Generated by Untitled UI CLI" />
<meta name="theme-color" content="#7f56d9" />
<meta name="description" content="Helix Engage — Hospital Lead Management & Call Center" />
<meta name="theme-color" content="#2563eb" />
<link rel="preconnect" href="https://fonts.googleapis.com" />
<link rel="preconnect" href="https://fonts.gstatic.com" crossorigin />
<link href="https://fonts.googleapis.com/css2?family=Inter:ital,opsz,wght@0,14..32,400..700;1,14..32,400..700&display=swap" rel="stylesheet" />
<title>Starter kit — Untitled UI</title>
<title>Helix Engage</title>
</head>
<body class="size-full bg-primary antialiased">

Binary file not shown.

After

Width:  |  Height:  |  Size: 918 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 895 KiB

BIN
public/favicon-32.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.5 KiB

BIN
public/helix-512.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 918 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 918 KiB

4
public/helix.svg Normal file
View File

@@ -0,0 +1,4 @@
<svg xmlns="http://www.w3.org/2000/svg" width="32" height="32" viewBox="0 0 32 32">
<rect width="32" height="32" rx="8" fill="#2563eb"/>
<text x="16" y="22" font-family="Inter, system-ui, sans-serif" font-size="20" font-weight="700" fill="white" text-anchor="middle">H</text>
</svg>

After

Width:  |  Height:  |  Size: 287 B

View File

@@ -16,9 +16,9 @@
* NOTE: callNotes/visitNotes/clinicalNotes are RICH_TEXT — read-only, cannot seed
*/
const GQL = 'http://localhost:4000/graphql';
const GQL = process.env.SEED_GQL ?? 'http://localhost:4000/graphql';
const SUB = 'fortytwo-dev';
const ORIGIN = 'http://fortytwo-dev.localhost:4010';
const ORIGIN = process.env.SEED_ORIGIN ?? 'http://fortytwo-dev.localhost:4010';
let token = '';
@@ -56,8 +56,14 @@ async function main() {
await auth();
console.log('✅ Auth OK\n');
// Workspace member IDs (from platform — set names + roles in UI before running)
const WM = {
// Workspace member IDs — switch based on target platform
const WM = GQL.includes('srv1477139') ? {
drSharma: '107efa70-fd32-4819-8936-994197c6ada1',
drPatel: '7e1fe368-1f23-4a10-8c2f-3e9c3846b209',
drKumar: 'b86ff7d3-57de-44e5-aa13-e5da848a960c',
drReddy: 'b82693b6-701c-4783-8d02-cc137c9c306b',
drSingh: 'b2a00dd2-5bb5-4c29-8fb1-70a681193a4c',
} : {
drSharma: '251e9b32-3a83-4f3c-a904-fad7e8b840c3',
drPatel: '2b1bbf20-3838-434f-9fe9-b98436362230',
drKumar: '16109622-9b13-4682-b327-eb611ffa8338',
@@ -76,7 +82,6 @@ async function main() {
specialty: 'Interventional Cardiology',
qualifications: 'MBBS, MD (Medicine), DM (Cardiology), FACC',
yearsOfExperience: 18,
branchClinic: 'Koramangala, Whitefield',
visitingHours: 'Mon/Wed/Fri 10:00 AM 1:00 PM',
consultationFeeNew: { amountMicros: 800_000_000, currencyCode: 'INR' },
consultationFeeFollowUp: { amountMicros: 500_000_000, currencyCode: 'INR' },
@@ -95,7 +100,6 @@ async function main() {
specialty: 'Reproductive Medicine & IVF',
qualifications: 'MBBS, MS (OBG), Fellowship in Reproductive Medicine',
yearsOfExperience: 15,
branchClinic: 'Indiranagar, Koramangala',
visitingHours: 'Tue/Thu/Sat 9:00 AM 12:00 PM',
consultationFeeNew: { amountMicros: 700_000_000, currencyCode: 'INR' },
consultationFeeFollowUp: { amountMicros: 400_000_000, currencyCode: 'INR' },
@@ -114,7 +118,6 @@ async function main() {
specialty: 'Joint Replacement & Sports Medicine',
qualifications: 'MBBS, MS (Ortho), Fellowship in Arthroplasty',
yearsOfExperience: 12,
branchClinic: 'Whitefield',
visitingHours: 'MonFri 2:00 PM 5:00 PM',
consultationFeeNew: { amountMicros: 600_000_000, currencyCode: 'INR' },
consultationFeeFollowUp: { amountMicros: 400_000_000, currencyCode: 'INR' },
@@ -133,7 +136,6 @@ async function main() {
specialty: 'Internal Medicine & Preventive Health',
qualifications: 'MBBS, MD (General Medicine)',
yearsOfExperience: 20,
branchClinic: 'Koramangala, Indiranagar, Whitefield',
visitingHours: 'MonSat 9:00 AM 6:00 PM',
consultationFeeNew: { amountMicros: 500_000_000, currencyCode: 'INR' },
consultationFeeFollowUp: { amountMicros: 300_000_000, currencyCode: 'INR' },
@@ -152,7 +154,6 @@ async function main() {
specialty: 'Otorhinolaryngology & Head/Neck Surgery',
qualifications: 'MBBS, MS (ENT), DNB',
yearsOfExperience: 10,
branchClinic: 'Indiranagar',
visitingHours: 'Mon/Wed/Fri 11:00 AM 3:00 PM',
consultationFeeNew: { amountMicros: 600_000_000, currencyCode: 'INR' },
consultationFeeFollowUp: { amountMicros: 400_000_000, currencyCode: 'INR' },

View File

@@ -10,9 +10,9 @@
* PackageTest: position→order, isMandatory→mandatory
*/
const GQL = 'http://localhost:4000/graphql';
const GQL = process.env.SEED_GQL ?? 'http://localhost:4000/graphql';
const SUB = 'fortytwo-dev';
const ORIGIN = 'http://fortytwo-dev.localhost:4010';
const ORIGIN = process.env.SEED_ORIGIN ?? 'http://fortytwo-dev.localhost:4010';
let token = '';

View File

@@ -14,9 +14,9 @@
* InsurancePartner: planTypes→planTypesAccepted
*/
const GQL = 'http://localhost:4000/graphql';
const GQL = process.env.SEED_GQL ?? 'http://localhost:4000/graphql';
const SUB = 'fortytwo-dev';
const ORIGIN = 'http://fortytwo-dev.localhost:4010';
const ORIGIN = process.env.SEED_ORIGIN ?? 'http://fortytwo-dev.localhost:4010';
let token = '';
@@ -143,23 +143,23 @@ async function main() {
// LINK DOCTORS TO CLINICS
// ═══════════════════════════════════════════
console.log('🔗 Linking doctors to clinics');
const doctors: Record<string, string> = {
'da5678f3-6b52-492e-87d3-c4707d105938': 'Dr. Sharma', // Koramangala
'b080cdf0-4527-46c7-b723-47f2eee623e4': 'Dr. Patel', // Indiranagar
'd780976a-7ddb-4a00-9a56-e7e3a77fa416': 'Dr. Kumar', // Whitefield
'bf77c148-438f-4b6f-9e5d-b1c1ff2e10f8': 'Dr. Reddy', // Koramangala
'e71c2c59-574f-4e81-b8cd-2d7b4b5da8e5': 'Dr. Singh', // Indiranagar
// Fetch doctor IDs dynamically from platform
const docData = await gql(`{ doctors(first: 10) { edges { node { id name } } } }`);
const allDocs = docData.doctors.edges.map((e: any) => e.node);
const clinicAssignment: Record<string, string> = {
'Sharma': koramangala,
'Patel': indiranagar,
'Kumar': whitefield,
'Reddy': koramangala,
'Singh': indiranagar,
};
const doctorClinicMap: Record<string, string> = {
'da5678f3-6b52-492e-87d3-c4707d105938': koramangala,
'b080cdf0-4527-46c7-b723-47f2eee623e4': indiranagar,
'd780976a-7ddb-4a00-9a56-e7e3a77fa416': whitefield,
'bf77c148-438f-4b6f-9e5d-b1c1ff2e10f8': koramangala,
'e71c2c59-574f-4e81-b8cd-2d7b4b5da8e5': indiranagar,
};
for (const [docId, clinicId] of Object.entries(doctorClinicMap)) {
await update('doctor', docId, { clinicId });
console.log(` ${doctors[docId]}${clinicId === koramangala ? 'Koramangala' : clinicId === whitefield ? 'Whitefield' : 'Indiranagar'}`);
for (const doc of allDocs) {
const lastName = Object.keys(clinicAssignment).find(n => doc.name?.includes(n));
if (lastName && clinicAssignment[lastName]) {
await update('doctor', doc.id, { clinicId: clinicAssignment[lastName] });
const clinicName = clinicAssignment[lastName] === koramangala ? 'Koramangala' : clinicAssignment[lastName] === whitefield ? 'Whitefield' : 'Indiranagar';
console.log(` ${doc.name}${clinicName}`);
}
}
console.log('');

View File

@@ -5,9 +5,9 @@
* Run: cd helix-engage && npx tsx scripts/test-ai-flow.ts
*/
const GQL = 'http://localhost:4000/graphql';
const GQL = process.env.SEED_GQL ?? 'http://localhost:4000/graphql';
const SUB = 'fortytwo-dev';
const ORIGIN = 'http://fortytwo-dev.localhost:4010';
const ORIGIN = process.env.SEED_ORIGIN ?? 'http://fortytwo-dev.localhost:4010';
// Rekha's credentials
const AGENT_EMAIL = 'rekha.cc@globalhospital.com';

View File

@@ -144,7 +144,7 @@ export function transformCalls(data: any): Call[] {
durationSeconds: n.durationSec ?? 0,
recordingUrl: n.recordingUrl,
disposition: n.disposition,
callNotes: undefined,
callNotes: null,
patientId: n.patientId,
appointmentId: n.appointmentId,
leadId: n.leadId,

View File

@@ -39,19 +39,7 @@ export const CallDeskPage = () => {
const activeLead = isInCall ? (callerLead ?? selectedLead) : selectedLead;
// Convert worklist lead to full Lead type for components that need it
const activeLeadFull = activeLead ? {
...activeLead,
updatedAt: activeLead.createdAt,
contactPhone: activeLead.contactPhone ?? [],
contactEmail: (activeLead as any).contactEmail ?? [],
priority: 'NORMAL' as const,
utmSource: null, utmMedium: null, utmCampaign: null, utmContent: null, utmTerm: null,
landingPageUrl: null, referrerUrl: null,
spamScore: 0, isSpam: false, isDuplicate: false, duplicateOfLeadId: null,
firstContactedAt: null, lastContactedAt: null, contactAttempts: 0,
convertedAt: null, patientId: null, campaignId: null, adId: null,
assignedAgent: null, leadScore: null,
} : null;
const activeLeadFull = activeLead as any;
return (
<div className="flex flex-1 flex-col overflow-hidden">