@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.
Files changed (103) hide show
  1. package/CHANGELOG.md +7 -0
  2. package/README.md +975 -891
  3. package/bin/index.js +2 -0
  4. package/dist/.tsbuildinfo +1 -1
  5. package/dist/cli/commands/create.d.ts +2 -2
  6. package/dist/cli/commands/create.d.ts.map +1 -1
  7. package/dist/cli/commands/create.js.map +1 -1
  8. package/dist/cli/prompts/project-setup.d.ts.map +1 -1
  9. package/dist/cli/prompts/project-setup.js +13 -5
  10. package/dist/cli/prompts/project-setup.js.map +1 -1
  11. package/dist/generators/template-generator.d.ts +11 -0
  12. package/dist/generators/template-generator.d.ts.map +1 -1
  13. package/dist/generators/template-generator.js +360 -16
  14. package/dist/generators/template-generator.js.map +1 -1
  15. package/dist/templates/shared/admin/web/app/admin/layout.tsx +34 -0
  16. package/dist/templates/shared/admin/web/components/admin-nav.tsx +48 -0
  17. package/dist/templates/shared/audit/web/lib/audit.ts +24 -0
  18. package/dist/templates/shared/auth/keycloak/web/app/api/auth/federated-logout/route.ts +173 -0
  19. package/dist/templates/shared/auth/keycloak/web/auth.config.ts +84 -0
  20. package/dist/templates/shared/auth/keycloak/web/auth.ts +26 -0
  21. package/dist/templates/shared/beta/web/app/api/beta-settings/route.ts +25 -0
  22. package/dist/templates/shared/beta/web/app/api/validate-beta-code/route.ts +67 -0
  23. package/dist/templates/shared/beta/web/lib/beta/settings.ts +31 -0
  24. package/dist/templates/shared/cache/web/lib/cache.ts +44 -0
  25. package/dist/templates/shared/config/web/lib/config.ts +112 -0
  26. package/dist/templates/shared/config/web/next.config.mjs +62 -0
  27. package/dist/templates/shared/contact/web/app/api/contact/route.ts +113 -0
  28. package/dist/templates/shared/contact/web/app/contact/page.tsx +195 -0
  29. package/dist/templates/shared/cookie-consent/web/components/cookie-consent.tsx +54 -0
  30. package/dist/templates/shared/database/postgresql/web/drizzle.config.ts +16 -0
  31. package/dist/templates/shared/database/postgresql/web/lib/db/drizzle.ts +39 -0
  32. package/dist/templates/shared/database/postgresql/web/lib/db/schema.ts +33 -0
  33. package/dist/templates/shared/database/supabase/web/lib/supabase/client.ts +12 -0
  34. package/dist/templates/shared/database/supabase/web/lib/supabase/server.ts +31 -0
  35. package/dist/templates/shared/database/supabase/web/lib/supabase/service.ts +15 -0
  36. package/dist/templates/shared/email/web/lib/email/branding.ts +18 -0
  37. package/dist/templates/shared/email/web/lib/email/client.ts +96 -0
  38. package/dist/templates/shared/error-pages/web/app/error.tsx +70 -0
  39. package/dist/templates/shared/error-pages/web/app/global-error.tsx +102 -0
  40. package/dist/templates/shared/error-pages/web/app/not-found.tsx +39 -0
  41. package/dist/templates/shared/health/web/app/api/health/route.ts +68 -0
  42. package/dist/templates/shared/legal/web/app/(legal)/privacy/page.tsx +205 -0
  43. package/dist/templates/shared/legal/web/app/(legal)/terms/page.tsx +154 -0
  44. package/dist/templates/shared/legal/web/lib/legal-config.ts +50 -0
  45. package/dist/templates/shared/loading/web/app/loading.tsx +5 -0
  46. package/dist/templates/shared/loading/web/components/skeleton.tsx +95 -0
  47. package/dist/templates/shared/middleware/web/middleware.ts +68 -0
  48. package/dist/templates/shared/observability/web/lib/observability.ts +135 -0
  49. package/dist/templates/shared/payments/web/app/api/webhooks/stripe/route.ts +109 -0
  50. package/dist/templates/shared/platform/web/lib/platform.ts +37 -0
  51. package/dist/templates/shared/redis/web/lib/rate-limit-store.ts +18 -0
  52. package/dist/templates/shared/redis/web/lib/redis.ts +48 -0
  53. package/dist/templates/shared/security/web/lib/api-security.ts +318 -0
  54. package/dist/templates/shared/seo/web/app/api/og/route.tsx +97 -0
  55. package/dist/templates/shared/seo/web/app/robots.ts +53 -0
  56. package/dist/templates/shared/seo/web/app/sitemap.ts +53 -0
  57. package/dist/templates/shared/utils/web/lib/api-response.ts +71 -0
  58. package/dist/templates/shared/utils/web/lib/utils.ts +85 -0
  59. package/package.json +5 -4
  60. package/src/templates/shared/admin/web/app/admin/layout.tsx +34 -0
  61. package/src/templates/shared/admin/web/components/admin-nav.tsx +48 -0
  62. package/src/templates/shared/audit/web/lib/audit.ts +24 -0
  63. package/src/templates/shared/auth/keycloak/web/app/api/auth/federated-logout/route.ts +173 -0
  64. package/src/templates/shared/auth/keycloak/web/auth.config.ts +84 -0
  65. package/src/templates/shared/auth/keycloak/web/auth.ts +26 -0
  66. package/src/templates/shared/beta/web/app/api/beta-settings/route.ts +25 -0
  67. package/src/templates/shared/beta/web/app/api/validate-beta-code/route.ts +67 -0
  68. package/src/templates/shared/beta/web/lib/beta/settings.ts +31 -0
  69. package/src/templates/shared/cache/web/lib/cache.ts +44 -0
  70. package/src/templates/shared/config/web/lib/config.ts +112 -0
  71. package/src/templates/shared/config/web/next.config.mjs +62 -0
  72. package/src/templates/shared/contact/web/app/api/contact/route.ts +113 -0
  73. package/src/templates/shared/contact/web/app/contact/page.tsx +195 -0
  74. package/src/templates/shared/cookie-consent/web/components/cookie-consent.tsx +54 -0
  75. package/src/templates/shared/database/postgresql/web/drizzle.config.ts +16 -0
  76. package/src/templates/shared/database/postgresql/web/lib/db/drizzle.ts +39 -0
  77. package/src/templates/shared/database/postgresql/web/lib/db/schema.ts +33 -0
  78. package/src/templates/shared/database/supabase/web/lib/supabase/client.ts +12 -0
  79. package/src/templates/shared/database/supabase/web/lib/supabase/server.ts +31 -0
  80. package/src/templates/shared/database/supabase/web/lib/supabase/service.ts +15 -0
  81. package/src/templates/shared/email/web/lib/email/branding.ts +18 -0
  82. package/src/templates/shared/email/web/lib/email/client.ts +96 -0
  83. package/src/templates/shared/error-pages/web/app/error.tsx +70 -0
  84. package/src/templates/shared/error-pages/web/app/global-error.tsx +102 -0
  85. package/src/templates/shared/error-pages/web/app/not-found.tsx +39 -0
  86. package/src/templates/shared/health/web/app/api/health/route.ts +68 -0
  87. package/src/templates/shared/legal/web/app/(legal)/privacy/page.tsx +205 -0
  88. package/src/templates/shared/legal/web/app/(legal)/terms/page.tsx +154 -0
  89. package/src/templates/shared/legal/web/lib/legal-config.ts +50 -0
  90. package/src/templates/shared/loading/web/app/loading.tsx +5 -0
  91. package/src/templates/shared/loading/web/components/skeleton.tsx +95 -0
  92. package/src/templates/shared/middleware/web/middleware.ts +68 -0
  93. package/src/templates/shared/observability/web/lib/observability.ts +135 -0
  94. package/src/templates/shared/payments/web/app/api/webhooks/stripe/route.ts +109 -0
  95. package/src/templates/shared/platform/web/lib/platform.ts +37 -0
  96. package/src/templates/shared/redis/web/lib/rate-limit-store.ts +18 -0
  97. package/src/templates/shared/redis/web/lib/redis.ts +48 -0
  98. package/src/templates/shared/security/web/lib/api-security.ts +318 -0
  99. package/src/templates/shared/seo/web/app/api/og/route.tsx +97 -0
  100. package/src/templates/shared/seo/web/app/robots.ts +53 -0
  101. package/src/templates/shared/seo/web/app/sitemap.ts +53 -0
  102. package/src/templates/shared/utils/web/lib/api-response.ts +71 -0
  103. 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&apos;re looking for doesn&apos;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&apos;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
+ }