Initial commit: Port Nimara CRM (Layers 0-4)
Full CRM rebuild with Next.js 15, TypeScript, Tailwind, Drizzle ORM,
PostgreSQL, Redis, BullMQ, MinIO, and Socket.io. Includes 461 source
files covering clients, berths, interests/pipeline, documents/EOI,
expenses/invoices, email, notifications, dashboard, admin, and
client portal. CI/CD via Gitea Actions with Docker builds.
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-26 11:52:51 +01:00
|
|
|
import type { Metadata } from 'next';
|
|
|
|
|
|
|
|
|
|
import { getPortalSession } from '@/lib/portal/auth';
|
|
|
|
|
import { getPortalDashboard } from '@/lib/services/portal.service';
|
2026-05-11 14:47:46 +02:00
|
|
|
import { isPortalDisabledGlobally } from '@/lib/services/portal-auth.service';
|
Initial commit: Port Nimara CRM (Layers 0-4)
Full CRM rebuild with Next.js 15, TypeScript, Tailwind, Drizzle ORM,
PostgreSQL, Redis, BullMQ, MinIO, and Socket.io. Includes 461 source
files covering clients, berths, interests/pipeline, documents/EOI,
expenses/invoices, email, notifications, dashboard, admin, and
client portal. CI/CD via Gitea Actions with Docker builds.
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-26 11:52:51 +01:00
|
|
|
import { PortalHeader } from '@/components/portal/portal-header';
|
|
|
|
|
import { PortalNav } from '@/components/portal/portal-nav';
|
|
|
|
|
|
|
|
|
|
export const metadata: Metadata = {
|
|
|
|
|
title: {
|
|
|
|
|
default: 'Client Portal',
|
|
|
|
|
template: '%s | Client Portal',
|
|
|
|
|
},
|
|
|
|
|
};
|
|
|
|
|
|
2026-05-04 22:57:01 +02:00
|
|
|
export default async function PortalLayout({ children }: { children: React.ReactNode }) {
|
2026-05-11 14:47:46 +02:00
|
|
|
// Route-level kill switch. When every port has client_portal_enabled=false,
|
|
|
|
|
// surface a clean "Portal not available" notice instead of letting the
|
|
|
|
|
// login form render (it would just reject every submit with a confusing
|
|
|
|
|
// ConflictError). Single-port deployments effectively get a global toggle
|
|
|
|
|
// out of the admin System Settings UI.
|
|
|
|
|
if (await isPortalDisabledGlobally()) {
|
|
|
|
|
return (
|
|
|
|
|
<div className="min-h-screen bg-gray-50 flex items-center justify-center px-4">
|
|
|
|
|
<div className="max-w-md text-center space-y-3">
|
|
|
|
|
<h1 className="text-2xl font-semibold text-gray-900">Client portal unavailable</h1>
|
|
|
|
|
<p className="text-sm text-gray-600">
|
|
|
|
|
The client portal isn't currently enabled for this site. If you were expecting to
|
|
|
|
|
sign in here, please contact your account manager.
|
|
|
|
|
</p>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
Initial commit: Port Nimara CRM (Layers 0-4)
Full CRM rebuild with Next.js 15, TypeScript, Tailwind, Drizzle ORM,
PostgreSQL, Redis, BullMQ, MinIO, and Socket.io. Includes 461 source
files covering clients, berths, interests/pipeline, documents/EOI,
expenses/invoices, email, notifications, dashboard, admin, and
client portal. CI/CD via Gitea Actions with Docker builds.
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-26 11:52:51 +01:00
|
|
|
// This layout wraps all portal routes including login/verify
|
|
|
|
|
// We can't easily check pathname in a server layout, so we attempt
|
2026-05-04 22:57:01 +02:00
|
|
|
// to get the session and pass it down - login/verify pages handle their own
|
Initial commit: Port Nimara CRM (Layers 0-4)
Full CRM rebuild with Next.js 15, TypeScript, Tailwind, Drizzle ORM,
PostgreSQL, Redis, BullMQ, MinIO, and Socket.io. Includes 461 source
files covering clients, berths, interests/pipeline, documents/EOI,
expenses/invoices, email, notifications, dashboard, admin, and
client portal. CI/CD via Gitea Actions with Docker builds.
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-26 11:52:51 +01:00
|
|
|
// redirect logic independently.
|
|
|
|
|
const session = await getPortalSession().catch(() => null);
|
|
|
|
|
|
|
|
|
|
// For authenticated routes we need client info for the header.
|
|
|
|
|
// If session is absent, children (login/verify pages) handle their own redirect.
|
|
|
|
|
let clientName = '';
|
|
|
|
|
let portName = 'Client Portal';
|
|
|
|
|
let portLogoUrl: string | null = null;
|
|
|
|
|
|
|
|
|
|
if (session) {
|
|
|
|
|
const dashboard = await getPortalDashboard(session.clientId, session.portId).catch(() => null);
|
|
|
|
|
if (dashboard) {
|
|
|
|
|
clientName = dashboard.client.fullName;
|
|
|
|
|
portName = dashboard.port.name;
|
|
|
|
|
portLogoUrl = dashboard.port.logoUrl;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return (
|
|
|
|
|
<div className="min-h-screen bg-gray-50">
|
|
|
|
|
{session && (
|
|
|
|
|
<>
|
2026-05-04 22:57:01 +02:00
|
|
|
<PortalHeader portName={portName} portLogoUrl={portLogoUrl} clientName={clientName} />
|
Initial commit: Port Nimara CRM (Layers 0-4)
Full CRM rebuild with Next.js 15, TypeScript, Tailwind, Drizzle ORM,
PostgreSQL, Redis, BullMQ, MinIO, and Socket.io. Includes 461 source
files covering clients, berths, interests/pipeline, documents/EOI,
expenses/invoices, email, notifications, dashboard, admin, and
client portal. CI/CD via Gitea Actions with Docker builds.
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-26 11:52:51 +01:00
|
|
|
<PortalNav />
|
|
|
|
|
</>
|
|
|
|
|
)}
|
2026-05-04 22:57:01 +02:00
|
|
|
<main className={session ? 'max-w-5xl mx-auto px-4 sm:px-6 py-8' : ''}>{children}</main>
|
Initial commit: Port Nimara CRM (Layers 0-4)
Full CRM rebuild with Next.js 15, TypeScript, Tailwind, Drizzle ORM,
PostgreSQL, Redis, BullMQ, MinIO, and Socket.io. Includes 461 source
files covering clients, berths, interests/pipeline, documents/EOI,
expenses/invoices, email, notifications, dashboard, admin, and
client portal. CI/CD via Gitea Actions with Docker builds.
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-26 11:52:51 +01:00
|
|
|
</div>
|
|
|
|
|
);
|
|
|
|
|
}
|