@digilogiclabs/create-saas-app 2.1.0 → 2.2.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +7 -0
- package/README.md +975 -891
- package/bin/index.js +2 -0
- package/dist/.tsbuildinfo +1 -1
- package/dist/cli/commands/create.d.ts +2 -2
- package/dist/cli/commands/create.d.ts.map +1 -1
- package/dist/cli/commands/create.js.map +1 -1
- package/dist/cli/prompts/project-setup.d.ts.map +1 -1
- package/dist/cli/prompts/project-setup.js +13 -5
- package/dist/cli/prompts/project-setup.js.map +1 -1
- package/dist/generators/template-generator.d.ts +11 -0
- package/dist/generators/template-generator.d.ts.map +1 -1
- package/dist/generators/template-generator.js +360 -16
- package/dist/generators/template-generator.js.map +1 -1
- package/dist/templates/shared/admin/web/app/admin/layout.tsx +34 -0
- package/dist/templates/shared/admin/web/components/admin-nav.tsx +48 -0
- package/dist/templates/shared/audit/web/lib/audit.ts +24 -0
- package/dist/templates/shared/auth/keycloak/web/app/api/auth/federated-logout/route.ts +173 -0
- package/dist/templates/shared/auth/keycloak/web/auth.config.ts +84 -0
- package/dist/templates/shared/auth/keycloak/web/auth.ts +26 -0
- package/dist/templates/shared/beta/web/app/api/beta-settings/route.ts +25 -0
- package/dist/templates/shared/beta/web/app/api/validate-beta-code/route.ts +67 -0
- package/dist/templates/shared/beta/web/lib/beta/settings.ts +31 -0
- package/dist/templates/shared/cache/web/lib/cache.ts +44 -0
- package/dist/templates/shared/config/web/lib/config.ts +112 -0
- package/dist/templates/shared/config/web/next.config.mjs +62 -0
- package/dist/templates/shared/contact/web/app/api/contact/route.ts +113 -0
- package/dist/templates/shared/contact/web/app/contact/page.tsx +195 -0
- package/dist/templates/shared/cookie-consent/web/components/cookie-consent.tsx +54 -0
- package/dist/templates/shared/database/postgresql/web/drizzle.config.ts +16 -0
- package/dist/templates/shared/database/postgresql/web/lib/db/drizzle.ts +39 -0
- package/dist/templates/shared/database/postgresql/web/lib/db/schema.ts +33 -0
- package/dist/templates/shared/database/supabase/web/lib/supabase/client.ts +12 -0
- package/dist/templates/shared/database/supabase/web/lib/supabase/server.ts +31 -0
- package/dist/templates/shared/database/supabase/web/lib/supabase/service.ts +15 -0
- package/dist/templates/shared/email/web/lib/email/branding.ts +18 -0
- package/dist/templates/shared/email/web/lib/email/client.ts +96 -0
- package/dist/templates/shared/error-pages/web/app/error.tsx +70 -0
- package/dist/templates/shared/error-pages/web/app/global-error.tsx +102 -0
- package/dist/templates/shared/error-pages/web/app/not-found.tsx +39 -0
- package/dist/templates/shared/health/web/app/api/health/route.ts +68 -0
- package/dist/templates/shared/legal/web/app/(legal)/privacy/page.tsx +205 -0
- package/dist/templates/shared/legal/web/app/(legal)/terms/page.tsx +154 -0
- package/dist/templates/shared/legal/web/lib/legal-config.ts +50 -0
- package/dist/templates/shared/loading/web/app/loading.tsx +5 -0
- package/dist/templates/shared/loading/web/components/skeleton.tsx +95 -0
- package/dist/templates/shared/middleware/web/middleware.ts +68 -0
- package/dist/templates/shared/observability/web/lib/observability.ts +135 -0
- package/dist/templates/shared/payments/web/app/api/webhooks/stripe/route.ts +109 -0
- package/dist/templates/shared/platform/web/lib/platform.ts +37 -0
- package/dist/templates/shared/redis/web/lib/rate-limit-store.ts +18 -0
- package/dist/templates/shared/redis/web/lib/redis.ts +48 -0
- package/dist/templates/shared/security/web/lib/api-security.ts +318 -0
- package/dist/templates/shared/seo/web/app/api/og/route.tsx +97 -0
- package/dist/templates/shared/seo/web/app/robots.ts +53 -0
- package/dist/templates/shared/seo/web/app/sitemap.ts +53 -0
- package/dist/templates/shared/utils/web/lib/api-response.ts +71 -0
- package/dist/templates/shared/utils/web/lib/utils.ts +85 -0
- package/package.json +5 -4
- package/src/templates/shared/admin/web/app/admin/layout.tsx +34 -0
- package/src/templates/shared/admin/web/components/admin-nav.tsx +48 -0
- package/src/templates/shared/audit/web/lib/audit.ts +24 -0
- package/src/templates/shared/auth/keycloak/web/app/api/auth/federated-logout/route.ts +173 -0
- package/src/templates/shared/auth/keycloak/web/auth.config.ts +84 -0
- package/src/templates/shared/auth/keycloak/web/auth.ts +26 -0
- package/src/templates/shared/beta/web/app/api/beta-settings/route.ts +25 -0
- package/src/templates/shared/beta/web/app/api/validate-beta-code/route.ts +67 -0
- package/src/templates/shared/beta/web/lib/beta/settings.ts +31 -0
- package/src/templates/shared/cache/web/lib/cache.ts +44 -0
- package/src/templates/shared/config/web/lib/config.ts +112 -0
- package/src/templates/shared/config/web/next.config.mjs +62 -0
- package/src/templates/shared/contact/web/app/api/contact/route.ts +113 -0
- package/src/templates/shared/contact/web/app/contact/page.tsx +195 -0
- package/src/templates/shared/cookie-consent/web/components/cookie-consent.tsx +54 -0
- package/src/templates/shared/database/postgresql/web/drizzle.config.ts +16 -0
- package/src/templates/shared/database/postgresql/web/lib/db/drizzle.ts +39 -0
- package/src/templates/shared/database/postgresql/web/lib/db/schema.ts +33 -0
- package/src/templates/shared/database/supabase/web/lib/supabase/client.ts +12 -0
- package/src/templates/shared/database/supabase/web/lib/supabase/server.ts +31 -0
- package/src/templates/shared/database/supabase/web/lib/supabase/service.ts +15 -0
- package/src/templates/shared/email/web/lib/email/branding.ts +18 -0
- package/src/templates/shared/email/web/lib/email/client.ts +96 -0
- package/src/templates/shared/error-pages/web/app/error.tsx +70 -0
- package/src/templates/shared/error-pages/web/app/global-error.tsx +102 -0
- package/src/templates/shared/error-pages/web/app/not-found.tsx +39 -0
- package/src/templates/shared/health/web/app/api/health/route.ts +68 -0
- package/src/templates/shared/legal/web/app/(legal)/privacy/page.tsx +205 -0
- package/src/templates/shared/legal/web/app/(legal)/terms/page.tsx +154 -0
- package/src/templates/shared/legal/web/lib/legal-config.ts +50 -0
- package/src/templates/shared/loading/web/app/loading.tsx +5 -0
- package/src/templates/shared/loading/web/components/skeleton.tsx +95 -0
- package/src/templates/shared/middleware/web/middleware.ts +68 -0
- package/src/templates/shared/observability/web/lib/observability.ts +135 -0
- package/src/templates/shared/payments/web/app/api/webhooks/stripe/route.ts +109 -0
- package/src/templates/shared/platform/web/lib/platform.ts +37 -0
- package/src/templates/shared/redis/web/lib/rate-limit-store.ts +18 -0
- package/src/templates/shared/redis/web/lib/redis.ts +48 -0
- package/src/templates/shared/security/web/lib/api-security.ts +318 -0
- package/src/templates/shared/seo/web/app/api/og/route.tsx +97 -0
- package/src/templates/shared/seo/web/app/robots.ts +53 -0
- package/src/templates/shared/seo/web/app/sitemap.ts +53 -0
- package/src/templates/shared/utils/web/lib/api-response.ts +71 -0
- package/src/templates/shared/utils/web/lib/utils.ts +85 -0
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
import type { EmailBranding } from '@digilogiclabs/platform-core/email-templates';
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* App branding for shared email templates.
|
|
5
|
+
* Used by platform-core's email template functions (welcomeEmail, notificationEmail, etc.)
|
|
6
|
+
*
|
|
7
|
+
* Update these values to match your app's branding.
|
|
8
|
+
*/
|
|
9
|
+
export const APP_BRANDING: EmailBranding = {
|
|
10
|
+
appName: 'My App',
|
|
11
|
+
primaryColor: '#3b82f6',
|
|
12
|
+
gradientFrom: '#3b82f6',
|
|
13
|
+
gradientTo: '#8b5cf6',
|
|
14
|
+
fromEmail: 'noreply@example.com',
|
|
15
|
+
supportEmail: 'support@example.com',
|
|
16
|
+
baseUrl: process.env.NEXT_PUBLIC_APP_URL || 'http://localhost:3000',
|
|
17
|
+
footerText: 'Built with Digi Logic Labs platform',
|
|
18
|
+
};
|
|
@@ -0,0 +1,96 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Email client singleton.
|
|
3
|
+
*
|
|
4
|
+
* Lazy-initializes a Resend client when first needed.
|
|
5
|
+
* Returns null gracefully if RESEND_API_KEY is not configured.
|
|
6
|
+
*
|
|
7
|
+
* Usage:
|
|
8
|
+
* import { sendEmail } from '@/lib/email/client';
|
|
9
|
+
* await sendEmail({ to: 'user@example.com', subject: 'Hello', html: '<p>Hi</p>' });
|
|
10
|
+
*/
|
|
11
|
+
|
|
12
|
+
let resendClient: Resend | null = null;
|
|
13
|
+
let initialized = false;
|
|
14
|
+
|
|
15
|
+
type Resend = {
|
|
16
|
+
emails: {
|
|
17
|
+
send: (params: {
|
|
18
|
+
from: string;
|
|
19
|
+
to: string | string[];
|
|
20
|
+
subject: string;
|
|
21
|
+
html: string;
|
|
22
|
+
text?: string;
|
|
23
|
+
replyTo?: string;
|
|
24
|
+
}) => Promise<{ data: { id: string } | null; error: { message: string } | null }>;
|
|
25
|
+
};
|
|
26
|
+
};
|
|
27
|
+
|
|
28
|
+
function getResend(): Resend | null {
|
|
29
|
+
if (initialized) return resendClient;
|
|
30
|
+
initialized = true;
|
|
31
|
+
|
|
32
|
+
const apiKey = process.env.RESEND_API_KEY;
|
|
33
|
+
if (!apiKey) {
|
|
34
|
+
console.warn('[Email] RESEND_API_KEY not configured — emails will be logged to console');
|
|
35
|
+
return null;
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
try {
|
|
39
|
+
// Dynamic import to avoid requiring resend in all environments
|
|
40
|
+
// eslint-disable-next-line @typescript-eslint/no-require-imports
|
|
41
|
+
const { Resend } = require('resend');
|
|
42
|
+
resendClient = new Resend(apiKey);
|
|
43
|
+
return resendClient;
|
|
44
|
+
} catch {
|
|
45
|
+
console.warn('[Email] resend package not installed — emails will be logged to console');
|
|
46
|
+
return null;
|
|
47
|
+
}
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
const DEFAULT_FROM = process.env.EMAIL_FROM || 'noreply@example.com';
|
|
51
|
+
|
|
52
|
+
interface EmailParams {
|
|
53
|
+
to: string | string[];
|
|
54
|
+
subject: string;
|
|
55
|
+
html: string;
|
|
56
|
+
text?: string;
|
|
57
|
+
from?: string;
|
|
58
|
+
replyTo?: string;
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
/**
|
|
62
|
+
* Send an email via Resend. Returns true on success, false on failure.
|
|
63
|
+
* Falls back to console logging in development or when Resend is not configured.
|
|
64
|
+
*/
|
|
65
|
+
export async function sendEmail(params: EmailParams): Promise<boolean> {
|
|
66
|
+
const from = params.from || DEFAULT_FROM;
|
|
67
|
+
const resend = getResend();
|
|
68
|
+
|
|
69
|
+
if (!resend) {
|
|
70
|
+
if (process.env.NODE_ENV === 'development') {
|
|
71
|
+
console.log('[Email] Would send:', { to: params.to, subject: params.subject, from });
|
|
72
|
+
}
|
|
73
|
+
return false;
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
try {
|
|
77
|
+
const { error } = await resend.emails.send({
|
|
78
|
+
from,
|
|
79
|
+
to: params.to,
|
|
80
|
+
subject: params.subject,
|
|
81
|
+
html: params.html,
|
|
82
|
+
text: params.text,
|
|
83
|
+
replyTo: params.replyTo,
|
|
84
|
+
});
|
|
85
|
+
|
|
86
|
+
if (error) {
|
|
87
|
+
console.error('[Email] Send failed:', error.message);
|
|
88
|
+
return false;
|
|
89
|
+
}
|
|
90
|
+
|
|
91
|
+
return true;
|
|
92
|
+
} catch (err) {
|
|
93
|
+
console.error('[Email] Send error:', err);
|
|
94
|
+
return false;
|
|
95
|
+
}
|
|
96
|
+
}
|
|
@@ -0,0 +1,70 @@
|
|
|
1
|
+
'use client';
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* Error boundary page.
|
|
5
|
+
*
|
|
6
|
+
* Catches runtime errors in the app and shows a recovery UI.
|
|
7
|
+
* Shows error details in development mode only.
|
|
8
|
+
*/
|
|
9
|
+
export default function Error({
|
|
10
|
+
error,
|
|
11
|
+
reset,
|
|
12
|
+
}: {
|
|
13
|
+
error: Error & { digest?: string };
|
|
14
|
+
reset: () => void;
|
|
15
|
+
}) {
|
|
16
|
+
return (
|
|
17
|
+
<main className="flex min-h-screen items-center justify-center bg-white px-4 dark:bg-gray-950">
|
|
18
|
+
<div className="text-center">
|
|
19
|
+
<div className="mx-auto mb-4 flex h-16 w-16 items-center justify-center rounded-full bg-red-100 dark:bg-red-900/30">
|
|
20
|
+
<svg
|
|
21
|
+
className="h-8 w-8 text-red-600 dark:text-red-400"
|
|
22
|
+
fill="none"
|
|
23
|
+
viewBox="0 0 24 24"
|
|
24
|
+
strokeWidth="1.5"
|
|
25
|
+
stroke="currentColor"
|
|
26
|
+
>
|
|
27
|
+
<path
|
|
28
|
+
strokeLinecap="round"
|
|
29
|
+
strokeLinejoin="round"
|
|
30
|
+
d="M12 9v3.75m-9.303 3.376c-.866 1.5.217 3.374 1.948 3.374h14.71c1.73 0 2.813-1.874 1.948-3.374L13.949 3.378c-.866-1.5-3.032-1.5-3.898 0L2.697 16.126ZM12 15.75h.007v.008H12v-.008Z"
|
|
31
|
+
/>
|
|
32
|
+
</svg>
|
|
33
|
+
</div>
|
|
34
|
+
<h1 className="text-2xl font-semibold text-gray-900 dark:text-white">
|
|
35
|
+
Something went wrong
|
|
36
|
+
</h1>
|
|
37
|
+
<p className="mt-2 max-w-md text-gray-600 dark:text-gray-400">
|
|
38
|
+
An unexpected error occurred. Please try again.
|
|
39
|
+
</p>
|
|
40
|
+
{error.digest && (
|
|
41
|
+
<p className="mt-1 text-xs text-gray-400 dark:text-gray-500">
|
|
42
|
+
Error ID: {error.digest}
|
|
43
|
+
</p>
|
|
44
|
+
)}
|
|
45
|
+
|
|
46
|
+
{process.env.NODE_ENV === 'development' && (
|
|
47
|
+
<pre className="mx-auto mt-4 max-w-lg overflow-auto rounded-lg bg-gray-100 p-4 text-left text-xs text-red-600 dark:bg-gray-900 dark:text-red-400">
|
|
48
|
+
{error.message}
|
|
49
|
+
{error.stack && `\n\n${error.stack}`}
|
|
50
|
+
</pre>
|
|
51
|
+
)}
|
|
52
|
+
|
|
53
|
+
<div className="mt-8 flex items-center justify-center gap-4">
|
|
54
|
+
<button
|
|
55
|
+
onClick={reset}
|
|
56
|
+
className="rounded-lg bg-blue-600 px-5 py-2.5 text-sm font-medium text-white transition-colors hover:bg-blue-700"
|
|
57
|
+
>
|
|
58
|
+
Try Again
|
|
59
|
+
</button>
|
|
60
|
+
<a
|
|
61
|
+
href="/"
|
|
62
|
+
className="rounded-lg border border-gray-300 px-5 py-2.5 text-sm font-medium text-gray-700 transition-colors hover:bg-gray-50 dark:border-gray-700 dark:text-gray-300 dark:hover:bg-gray-800"
|
|
63
|
+
>
|
|
64
|
+
Go Home
|
|
65
|
+
</a>
|
|
66
|
+
</div>
|
|
67
|
+
</div>
|
|
68
|
+
</main>
|
|
69
|
+
);
|
|
70
|
+
}
|
|
@@ -0,0 +1,102 @@
|
|
|
1
|
+
'use client';
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* Root error boundary.
|
|
5
|
+
*
|
|
6
|
+
* This is the last-resort error handler that catches errors in the root layout.
|
|
7
|
+
* It must render its own <html> and <body> tags since it replaces the entire page.
|
|
8
|
+
* Uses inline styles only — no framework CSS is available at this level.
|
|
9
|
+
*/
|
|
10
|
+
export default function GlobalError({
|
|
11
|
+
error,
|
|
12
|
+
reset,
|
|
13
|
+
}: {
|
|
14
|
+
error: Error & { digest?: string };
|
|
15
|
+
reset: () => void;
|
|
16
|
+
}) {
|
|
17
|
+
return (
|
|
18
|
+
<html lang="en">
|
|
19
|
+
<body
|
|
20
|
+
style={{
|
|
21
|
+
margin: 0,
|
|
22
|
+
minHeight: '100vh',
|
|
23
|
+
display: 'flex',
|
|
24
|
+
alignItems: 'center',
|
|
25
|
+
justifyContent: 'center',
|
|
26
|
+
fontFamily:
|
|
27
|
+
'-apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, sans-serif',
|
|
28
|
+
backgroundColor: '#fafafa',
|
|
29
|
+
color: '#111',
|
|
30
|
+
}}
|
|
31
|
+
>
|
|
32
|
+
<div style={{ textAlign: 'center', padding: '2rem' }}>
|
|
33
|
+
<div
|
|
34
|
+
style={{
|
|
35
|
+
width: 64,
|
|
36
|
+
height: 64,
|
|
37
|
+
margin: '0 auto 1rem',
|
|
38
|
+
borderRadius: '50%',
|
|
39
|
+
backgroundColor: '#fee2e2',
|
|
40
|
+
display: 'flex',
|
|
41
|
+
alignItems: 'center',
|
|
42
|
+
justifyContent: 'center',
|
|
43
|
+
fontSize: 32,
|
|
44
|
+
}}
|
|
45
|
+
>
|
|
46
|
+
!
|
|
47
|
+
</div>
|
|
48
|
+
<h1 style={{ fontSize: '1.5rem', fontWeight: 600, margin: 0 }}>
|
|
49
|
+
Something went wrong
|
|
50
|
+
</h1>
|
|
51
|
+
<p style={{ color: '#666', marginTop: '0.5rem' }}>
|
|
52
|
+
A critical error occurred. Please try reloading the page.
|
|
53
|
+
</p>
|
|
54
|
+
{error.digest && (
|
|
55
|
+
<p style={{ color: '#999', fontSize: '0.75rem', marginTop: '0.25rem' }}>
|
|
56
|
+
Error ID: {error.digest}
|
|
57
|
+
</p>
|
|
58
|
+
)}
|
|
59
|
+
<div
|
|
60
|
+
style={{
|
|
61
|
+
marginTop: '2rem',
|
|
62
|
+
display: 'flex',
|
|
63
|
+
gap: '1rem',
|
|
64
|
+
justifyContent: 'center',
|
|
65
|
+
}}
|
|
66
|
+
>
|
|
67
|
+
<button
|
|
68
|
+
onClick={reset}
|
|
69
|
+
style={{
|
|
70
|
+
padding: '0.625rem 1.25rem',
|
|
71
|
+
backgroundColor: '#2563eb',
|
|
72
|
+
color: 'white',
|
|
73
|
+
border: 'none',
|
|
74
|
+
borderRadius: '0.5rem',
|
|
75
|
+
cursor: 'pointer',
|
|
76
|
+
fontSize: '0.875rem',
|
|
77
|
+
fontWeight: 500,
|
|
78
|
+
}}
|
|
79
|
+
>
|
|
80
|
+
Try Again
|
|
81
|
+
</button>
|
|
82
|
+
<a
|
|
83
|
+
href="/"
|
|
84
|
+
style={{
|
|
85
|
+
padding: '0.625rem 1.25rem',
|
|
86
|
+
backgroundColor: 'white',
|
|
87
|
+
color: '#374151',
|
|
88
|
+
border: '1px solid #d1d5db',
|
|
89
|
+
borderRadius: '0.5rem',
|
|
90
|
+
textDecoration: 'none',
|
|
91
|
+
fontSize: '0.875rem',
|
|
92
|
+
fontWeight: 500,
|
|
93
|
+
}}
|
|
94
|
+
>
|
|
95
|
+
Go Home
|
|
96
|
+
</a>
|
|
97
|
+
</div>
|
|
98
|
+
</div>
|
|
99
|
+
</body>
|
|
100
|
+
</html>
|
|
101
|
+
);
|
|
102
|
+
}
|
|
@@ -0,0 +1,39 @@
|
|
|
1
|
+
import Link from 'next/link';
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* Custom 404 page.
|
|
5
|
+
*
|
|
6
|
+
* Shown when a route is not found. Uses Tailwind CSS for styling.
|
|
7
|
+
* Customize the links and messaging to match your app.
|
|
8
|
+
*/
|
|
9
|
+
export default function NotFound() {
|
|
10
|
+
return (
|
|
11
|
+
<main className="flex min-h-screen items-center justify-center bg-white px-4 dark:bg-gray-950">
|
|
12
|
+
<div className="text-center">
|
|
13
|
+
<p className="bg-gradient-to-r from-blue-500 to-purple-600 bg-clip-text text-8xl font-bold text-transparent">
|
|
14
|
+
404
|
|
15
|
+
</p>
|
|
16
|
+
<h1 className="mt-4 text-2xl font-semibold text-gray-900 dark:text-white">
|
|
17
|
+
Page not found
|
|
18
|
+
</h1>
|
|
19
|
+
<p className="mt-2 max-w-md text-gray-600 dark:text-gray-400">
|
|
20
|
+
The page you're looking for doesn't exist or has been moved.
|
|
21
|
+
</p>
|
|
22
|
+
<div className="mt-8 flex items-center justify-center gap-4">
|
|
23
|
+
<Link
|
|
24
|
+
href="/"
|
|
25
|
+
className="rounded-lg bg-blue-600 px-5 py-2.5 text-sm font-medium text-white transition-colors hover:bg-blue-700"
|
|
26
|
+
>
|
|
27
|
+
Go Home
|
|
28
|
+
</Link>
|
|
29
|
+
<Link
|
|
30
|
+
href="/dashboard"
|
|
31
|
+
className="rounded-lg border border-gray-300 px-5 py-2.5 text-sm font-medium text-gray-700 transition-colors hover:bg-gray-50 dark:border-gray-700 dark:text-gray-300 dark:hover:bg-gray-800"
|
|
32
|
+
>
|
|
33
|
+
Dashboard
|
|
34
|
+
</Link>
|
|
35
|
+
</div>
|
|
36
|
+
</div>
|
|
37
|
+
</main>
|
|
38
|
+
);
|
|
39
|
+
}
|
|
@@ -0,0 +1,68 @@
|
|
|
1
|
+
import { NextResponse } from 'next/server';
|
|
2
|
+
|
|
3
|
+
export const dynamic = 'force-dynamic';
|
|
4
|
+
|
|
5
|
+
interface ServiceHealth {
|
|
6
|
+
status: 'up' | 'down' | 'unknown';
|
|
7
|
+
responseTime?: number;
|
|
8
|
+
error?: string;
|
|
9
|
+
}
|
|
10
|
+
|
|
11
|
+
/**
|
|
12
|
+
* GET /api/health
|
|
13
|
+
*
|
|
14
|
+
* Non-destructive health check for monitoring (Uptime Kuma, Coolify, K8s probes).
|
|
15
|
+
* Returns 200 if healthy, 503 if degraded/unhealthy.
|
|
16
|
+
*/
|
|
17
|
+
export async function GET() {
|
|
18
|
+
const services: Record<string, ServiceHealth> = {};
|
|
19
|
+
const details: string[] = [];
|
|
20
|
+
|
|
21
|
+
// Check database connectivity
|
|
22
|
+
try {
|
|
23
|
+
const start = Date.now();
|
|
24
|
+
// Replace with your database check — e.g. a simple SELECT 1
|
|
25
|
+
const dbUrl = process.env.DATABASE_URL;
|
|
26
|
+
services.database = dbUrl
|
|
27
|
+
? { status: 'up', responseTime: Date.now() - start }
|
|
28
|
+
: { status: 'down', error: 'DATABASE_URL not configured' };
|
|
29
|
+
} catch {
|
|
30
|
+
services.database = { status: 'down', error: 'Database connection failed' };
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
// Check Redis connectivity
|
|
34
|
+
try {
|
|
35
|
+
const start = Date.now();
|
|
36
|
+
const redisUrl = process.env.REDIS_URL;
|
|
37
|
+
services.cache = redisUrl
|
|
38
|
+
? { status: 'up', responseTime: Date.now() - start }
|
|
39
|
+
: { status: 'unknown', error: 'REDIS_URL not configured (optional)' };
|
|
40
|
+
} catch {
|
|
41
|
+
services.cache = { status: 'down', error: 'Cache connection failed' };
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
// Check email service
|
|
45
|
+
services.email = process.env.RESEND_API_KEY
|
|
46
|
+
? { status: 'up' }
|
|
47
|
+
: { status: 'unknown', error: 'RESEND_API_KEY not configured' };
|
|
48
|
+
|
|
49
|
+
// Determine overall status
|
|
50
|
+
const dbDown = services.database?.status === 'down';
|
|
51
|
+
let status: 'healthy' | 'degraded' | 'unhealthy';
|
|
52
|
+
|
|
53
|
+
if (dbDown) {
|
|
54
|
+
status = 'unhealthy';
|
|
55
|
+
details.push('Database is down — critical service unavailable');
|
|
56
|
+
} else if (Object.values(services).some((s) => s.status === 'down')) {
|
|
57
|
+
status = 'degraded';
|
|
58
|
+
details.push('Some services degraded');
|
|
59
|
+
} else {
|
|
60
|
+
status = 'healthy';
|
|
61
|
+
details.push('All systems operational');
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
return NextResponse.json(
|
|
65
|
+
{ status, timestamp: new Date().toISOString(), services, details },
|
|
66
|
+
{ status: status === 'healthy' ? 200 : 503 }
|
|
67
|
+
);
|
|
68
|
+
}
|
|
@@ -0,0 +1,205 @@
|
|
|
1
|
+
import { Metadata } from 'next';
|
|
2
|
+
import { LEGAL_CONFIG } from '@/lib/legal-config';
|
|
3
|
+
|
|
4
|
+
export const metadata: Metadata = {
|
|
5
|
+
title: `Privacy Policy | ${LEGAL_CONFIG.appName}`,
|
|
6
|
+
description: `Privacy Policy for ${LEGAL_CONFIG.appName}. Learn how we collect, use, and protect your data.`,
|
|
7
|
+
alternates: {
|
|
8
|
+
canonical: `${LEGAL_CONFIG.appUrl}/privacy`,
|
|
9
|
+
},
|
|
10
|
+
};
|
|
11
|
+
|
|
12
|
+
export default function PrivacyPage() {
|
|
13
|
+
const {
|
|
14
|
+
appName,
|
|
15
|
+
companyName,
|
|
16
|
+
appUrl,
|
|
17
|
+
supportEmail,
|
|
18
|
+
privacyEmail,
|
|
19
|
+
effectiveDate,
|
|
20
|
+
thirdPartyServices,
|
|
21
|
+
usesAnalyticsCookies,
|
|
22
|
+
dataRetentionPeriod,
|
|
23
|
+
} = LEGAL_CONFIG;
|
|
24
|
+
|
|
25
|
+
return (
|
|
26
|
+
<main className="min-h-screen bg-white dark:bg-gray-950">
|
|
27
|
+
<div className="mx-auto max-w-3xl px-4 py-16 sm:px-6 lg:px-8">
|
|
28
|
+
<h1 className="mb-2 text-3xl font-bold tracking-tight text-gray-900 dark:text-white">
|
|
29
|
+
Privacy Policy
|
|
30
|
+
</h1>
|
|
31
|
+
<p className="mb-8 text-sm text-gray-500 dark:text-gray-400">
|
|
32
|
+
Effective: {effectiveDate}
|
|
33
|
+
</p>
|
|
34
|
+
|
|
35
|
+
<div className="prose prose-gray dark:prose-invert max-w-none">
|
|
36
|
+
<section>
|
|
37
|
+
<h2>1. Information We Collect</h2>
|
|
38
|
+
<p>We collect information you provide directly:</p>
|
|
39
|
+
<ul>
|
|
40
|
+
<li>
|
|
41
|
+
<strong>Account data:</strong> name, email, profile information
|
|
42
|
+
</li>
|
|
43
|
+
<li>
|
|
44
|
+
<strong>Usage data:</strong> pages visited, features used,
|
|
45
|
+
interactions
|
|
46
|
+
</li>
|
|
47
|
+
<li>
|
|
48
|
+
<strong>Device data:</strong> browser type, IP address, operating
|
|
49
|
+
system
|
|
50
|
+
</li>
|
|
51
|
+
</ul>
|
|
52
|
+
</section>
|
|
53
|
+
|
|
54
|
+
<section>
|
|
55
|
+
<h2>2. How We Use Your Information</h2>
|
|
56
|
+
<ul>
|
|
57
|
+
<li>Provide and improve the service</li>
|
|
58
|
+
<li>Send transactional emails (account, security, updates)</li>
|
|
59
|
+
<li>Analyze usage patterns to improve user experience</li>
|
|
60
|
+
<li>Prevent fraud and enforce our terms</li>
|
|
61
|
+
<li>Comply with legal obligations</li>
|
|
62
|
+
</ul>
|
|
63
|
+
</section>
|
|
64
|
+
|
|
65
|
+
<section>
|
|
66
|
+
<h2>3. Information Sharing</h2>
|
|
67
|
+
<p>
|
|
68
|
+
We do not sell your personal data. We may share information with:
|
|
69
|
+
</p>
|
|
70
|
+
<ul>
|
|
71
|
+
<li>
|
|
72
|
+
<strong>Service providers:</strong> who help us operate{' '}
|
|
73
|
+
{appName}
|
|
74
|
+
</li>
|
|
75
|
+
<li>
|
|
76
|
+
<strong>Legal authorities:</strong> when required by law or to
|
|
77
|
+
protect rights
|
|
78
|
+
</li>
|
|
79
|
+
</ul>
|
|
80
|
+
{thirdPartyServices.length > 0 && (
|
|
81
|
+
<>
|
|
82
|
+
<p>Third-party services we use:</p>
|
|
83
|
+
<ul>
|
|
84
|
+
{thirdPartyServices.map((service) => (
|
|
85
|
+
<li key={service}>{service}</li>
|
|
86
|
+
))}
|
|
87
|
+
</ul>
|
|
88
|
+
</>
|
|
89
|
+
)}
|
|
90
|
+
</section>
|
|
91
|
+
|
|
92
|
+
<section>
|
|
93
|
+
<h2>4. Cookies</h2>
|
|
94
|
+
<p>We use cookies for:</p>
|
|
95
|
+
<ul>
|
|
96
|
+
<li>
|
|
97
|
+
<strong>Essential cookies:</strong> authentication, security,
|
|
98
|
+
preferences
|
|
99
|
+
</li>
|
|
100
|
+
{usesAnalyticsCookies && (
|
|
101
|
+
<li>
|
|
102
|
+
<strong>Analytics cookies:</strong> understanding how users
|
|
103
|
+
interact with {appName}
|
|
104
|
+
</li>
|
|
105
|
+
)}
|
|
106
|
+
</ul>
|
|
107
|
+
<p>
|
|
108
|
+
You can control cookies through your browser settings. Disabling
|
|
109
|
+
essential cookies may limit functionality.
|
|
110
|
+
</p>
|
|
111
|
+
</section>
|
|
112
|
+
|
|
113
|
+
<section>
|
|
114
|
+
<h2>5. Your Rights</h2>
|
|
115
|
+
<p>You have the right to:</p>
|
|
116
|
+
<ul>
|
|
117
|
+
<li>
|
|
118
|
+
<strong>Access:</strong> request a copy of your personal data
|
|
119
|
+
</li>
|
|
120
|
+
<li>
|
|
121
|
+
<strong>Rectification:</strong> correct inaccurate data
|
|
122
|
+
</li>
|
|
123
|
+
<li>
|
|
124
|
+
<strong>Erasure:</strong> request deletion of your data
|
|
125
|
+
</li>
|
|
126
|
+
<li>
|
|
127
|
+
<strong>Portability:</strong> receive your data in a portable
|
|
128
|
+
format
|
|
129
|
+
</li>
|
|
130
|
+
<li>
|
|
131
|
+
<strong>Objection:</strong> object to certain processing of your
|
|
132
|
+
data
|
|
133
|
+
</li>
|
|
134
|
+
</ul>
|
|
135
|
+
<p>
|
|
136
|
+
To exercise these rights, contact{' '}
|
|
137
|
+
<a href={`mailto:${privacyEmail}`}>{privacyEmail}</a>.
|
|
138
|
+
</p>
|
|
139
|
+
</section>
|
|
140
|
+
|
|
141
|
+
<section>
|
|
142
|
+
<h2>6. Data Security</h2>
|
|
143
|
+
<p>
|
|
144
|
+
We implement industry-standard security measures including
|
|
145
|
+
encryption in transit (TLS), encrypted storage, and access
|
|
146
|
+
controls. No method of transmission is 100% secure, but we strive
|
|
147
|
+
to protect your data.
|
|
148
|
+
</p>
|
|
149
|
+
</section>
|
|
150
|
+
|
|
151
|
+
<section>
|
|
152
|
+
<h2>7. Data Retention</h2>
|
|
153
|
+
<p>
|
|
154
|
+
We retain your data for as long as your account is active. After
|
|
155
|
+
deletion, data is retained for {dataRetentionPeriod} for legal and
|
|
156
|
+
operational purposes, then permanently deleted.
|
|
157
|
+
</p>
|
|
158
|
+
</section>
|
|
159
|
+
|
|
160
|
+
<section>
|
|
161
|
+
<h2>8. Children's Privacy</h2>
|
|
162
|
+
<p>
|
|
163
|
+
{appName} is not directed at children under 13. We do not
|
|
164
|
+
knowingly collect data from children. If you believe a child has
|
|
165
|
+
provided us data, contact us for removal.
|
|
166
|
+
</p>
|
|
167
|
+
</section>
|
|
168
|
+
|
|
169
|
+
<section>
|
|
170
|
+
<h2>9. International Transfers</h2>
|
|
171
|
+
<p>
|
|
172
|
+
Your data may be processed in countries outside your own. We
|
|
173
|
+
ensure appropriate safeguards are in place for such transfers in
|
|
174
|
+
compliance with applicable data protection laws.
|
|
175
|
+
</p>
|
|
176
|
+
</section>
|
|
177
|
+
|
|
178
|
+
<section>
|
|
179
|
+
<h2>10. Changes to This Policy</h2>
|
|
180
|
+
<p>
|
|
181
|
+
We may update this policy periodically. We will notify you of
|
|
182
|
+
material changes via email or prominent notice on {appName}.
|
|
183
|
+
Continued use after changes constitutes acceptance.
|
|
184
|
+
</p>
|
|
185
|
+
</section>
|
|
186
|
+
|
|
187
|
+
<section>
|
|
188
|
+
<h2>11. Contact</h2>
|
|
189
|
+
<p>
|
|
190
|
+
For privacy questions or data requests, contact us at{' '}
|
|
191
|
+
<a href={`mailto:${privacyEmail}`}>{privacyEmail}</a>.
|
|
192
|
+
</p>
|
|
193
|
+
<p>
|
|
194
|
+
{companyName}
|
|
195
|
+
<br />
|
|
196
|
+
{appUrl}
|
|
197
|
+
<br />
|
|
198
|
+
{supportEmail}
|
|
199
|
+
</p>
|
|
200
|
+
</section>
|
|
201
|
+
</div>
|
|
202
|
+
</div>
|
|
203
|
+
</main>
|
|
204
|
+
);
|
|
205
|
+
}
|