@infuro/cms-core 1.0.16 → 1.0.19

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 (47) hide show
  1. package/README.md +739 -724
  2. package/dist/admin.cjs +1 -1
  3. package/dist/admin.cjs.map +1 -1
  4. package/dist/admin.js +1 -1
  5. package/dist/admin.js.map +1 -1
  6. package/dist/api.cjs +132 -57
  7. package/dist/api.cjs.map +1 -1
  8. package/dist/api.d.cts +1 -1
  9. package/dist/api.d.ts +1 -1
  10. package/dist/api.js +132 -57
  11. package/dist/api.js.map +1 -1
  12. package/dist/auth.cjs.map +1 -1
  13. package/dist/auth.js.map +1 -1
  14. package/dist/cli.cjs +21 -6
  15. package/dist/cli.cjs.map +1 -1
  16. package/dist/cli.js +21 -6
  17. package/dist/cli.js.map +1 -1
  18. package/dist/hooks.cjs.map +1 -1
  19. package/dist/hooks.js.map +1 -1
  20. package/dist/{index-h42MoUNq.d.cts → index-D2C1O9b4.d.cts} +8 -1
  21. package/dist/{index-C85X7cc7.d.ts → index-GMn7-9PX.d.ts} +8 -1
  22. package/dist/index.cjs +139 -57
  23. package/dist/index.cjs.map +1 -1
  24. package/dist/index.d.cts +3 -2
  25. package/dist/index.d.ts +3 -2
  26. package/dist/index.js +139 -57
  27. package/dist/index.js.map +1 -1
  28. package/dist/migrations/1772178563554-InitialSchema.ts +304 -304
  29. package/dist/migrations/1772178563555-ChatAndKnowledgeBase.ts +55 -55
  30. package/dist/migrations/1772178563556-KnowledgeBaseVector.ts +16 -16
  31. package/dist/migrations/1774300000000-RbacSeedGroupsAndPermissionUnique.ts +24 -24
  32. package/dist/migrations/1774300000001-SeedAdministratorUsersPermission.ts +35 -35
  33. package/dist/migrations/1774400000000-CustomerAdminAccessContactUser.ts +37 -37
  34. package/dist/migrations/1774400000001-StorefrontCartWishlist.ts +100 -100
  35. package/dist/migrations/1774400000002-WishlistGuestId.ts +29 -29
  36. package/dist/migrations/1774500000000-ProductCollectionHsn.ts +15 -15
  37. package/dist/migrations/1774600000000-OrderKindParentOrderNumber.ts +36 -36
  38. package/dist/migrations/1774700000000-CollectionVariants.ts +13 -0
  39. package/dist/migrations/1774800000000-OtpChallengesUserPhone.ts +41 -41
  40. package/dist/migrations/1774900000000-MessageTemplates.ts +39 -39
  41. package/dist/migrations/1775000000000-ProductUomTypeOrderItemSnapshots.ts +29 -29
  42. package/dist/migrations/1775200000000-MediaDriveFolders.ts +38 -38
  43. package/dist/migrations/README.md +3 -3
  44. package/dist/theme.cjs.map +1 -1
  45. package/dist/theme.js.map +1 -1
  46. package/package.json +13 -6
  47. package/src/admin/admin.css +72 -72
package/dist/cli.js CHANGED
@@ -55,6 +55,8 @@ const helpers = createAuthHelpers(
55
55
 
56
56
  export const requireAuth = helpers.requireAuth;
57
57
  export const requirePermission = helpers.requirePermission;
58
+ export const requireEntityPermission = helpers.requireEntityPermission;
59
+ export const requireAdminAccess = helpers.requireAdminAccess;
58
60
  export const getAuthenticatedUser = helpers.getAuthenticatedUser;
59
61
  `,
60
62
  "src/lib/cms.ts": `import {
@@ -84,12 +86,23 @@ import { getServerSession } from 'next-auth';
84
86
  import { createCmsApiHandler } from '@infuro/cms-core/api';
85
87
  import { CMS_ENTITY_MAP } from '@infuro/cms-core';
86
88
  import { getDataSourceInitialized } from '@/lib/data-source';
87
- import { requireAuth } from '@/lib/auth-helpers';
89
+ import {
90
+ requireAuth,
91
+ requireAdminAccess,
92
+ requireEntityPermission,
93
+ getAuthenticatedUser,
94
+ } from '@/lib/auth-helpers';
88
95
  import { getCms } from '@/lib/cms';
89
96
  import bcrypt from 'bcryptjs';
90
97
 
91
98
  const baseUrl = process.env.NEXTAUTH_URL || 'http://localhost:3000';
92
99
 
100
+ async function requireAdminApiAuth(req: Request) {
101
+ const a = await requireAuth(req);
102
+ if (a) return a;
103
+ return requireAdminAccess(req);
104
+ }
105
+
93
106
  let handlerPromise: Promise<ReturnType<typeof createCmsApiHandler>> | null = null;
94
107
 
95
108
  async function getHandler() {
@@ -99,7 +112,9 @@ async function getHandler() {
99
112
  createCmsApiHandler({
100
113
  dataSource,
101
114
  entityMap: CMS_ENTITY_MAP,
102
- requireAuth,
115
+ requireAuth: requireAdminApiAuth,
116
+ requireEntityPermission,
117
+ getSessionUser: getAuthenticatedUser,
103
118
  json: NextResponse.json.bind(NextResponse),
104
119
  getCms,
105
120
  userAuth: {
@@ -117,12 +132,12 @@ async function getHandler() {
117
132
  dataSource,
118
133
  entityMap: CMS_ENTITY_MAP,
119
134
  json: NextResponse.json.bind(NextResponse),
120
- requireAuth,
121
- requirePermission: requireAuth,
135
+ requireAuth: requireAdminApiAuth,
136
+ requirePermission: requireAdminApiAuth,
122
137
  },
123
138
  upload: {
124
139
  json: NextResponse.json.bind(NextResponse),
125
- requireAuth,
140
+ requireAuth: requireAdminApiAuth,
126
141
  storage: () => getCms().then((cms) => cms.getPlugin('storage')),
127
142
  localUploadDir: 'public/uploads',
128
143
  },
@@ -142,7 +157,7 @@ async function getHandler() {
142
157
  dataSource,
143
158
  entityMap: CMS_ENTITY_MAP,
144
159
  json: NextResponse.json.bind(NextResponse),
145
- requireAuth,
160
+ requireAuth: requireAdminApiAuth,
146
161
  baseUrl,
147
162
  },
148
163
  })
package/dist/cli.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/cli.ts"],"sourcesContent":["#!/usr/bin/env node\n/**\n * @infuro/cms-core init CLI\n * Usage: npx @infuro/cms-core init [--force] [--dry-run] [--no-deps] [--no-patch-config]\n */\nimport fs from 'fs';\nimport path from 'path';\n\nconst TEMPLATES = {\n 'src/lib/data-source.ts': `import 'reflect-metadata';\nimport path from 'path';\nimport { createRequire } from 'module';\nimport { DataSource } from 'typeorm';\nimport { CMS_ENTITY_MAP } from '@infuro/cms-core';\n\nconst require = createRequire(import.meta.url);\nconst coreDir = path.dirname(require.resolve('@infuro/cms-core'));\n\nlet dataSource: DataSource | null = null;\n\nexport function getDataSource(): DataSource {\n if (!dataSource) {\n dataSource = new DataSource({\n type: 'postgres',\n url: process.env.DATABASE_URL,\n entities: Object.values(CMS_ENTITY_MAP),\n synchronize: false,\n ...(process.env.TYPEORM_CLI && {\n migrations: [\n path.join(coreDir, 'migrations', '*.ts'),\n path.join(process.cwd(), 'src', 'migrations', '*.ts'),\n ],\n }),\n });\n }\n return dataSource;\n}\n\nexport async function getDataSourceInitialized(): Promise<DataSource> {\n const ds = getDataSource();\n if (!ds.isInitialized) await ds.initialize();\n return ds;\n}\n\nexport default getDataSource;\n`,\n\n 'src/lib/auth-helpers.ts': `import { getServerSession } from 'next-auth';\nimport { NextResponse } from 'next/server';\nimport { createAuthHelpers } from '@infuro/cms-core/auth';\n\nconst helpers = createAuthHelpers(\n async () => {\n const s = await getServerSession();\n return s ? { user: s.user } : null;\n },\n NextResponse\n);\n\nexport const requireAuth = helpers.requireAuth;\nexport const requirePermission = helpers.requirePermission;\nexport const getAuthenticatedUser = helpers.getAuthenticatedUser;\n`,\n\n 'src/lib/cms.ts': `import {\n createCmsApp,\n localStoragePlugin,\n type CmsApp,\n} from '@infuro/cms-core';\nimport { getDataSourceInitialized } from './data-source';\n\nlet cmsPromise: Promise<CmsApp> | null = null;\n\nexport async function getCms(): Promise<CmsApp> {\n if (cmsPromise) return cmsPromise;\n const dataSource = await getDataSourceInitialized();\n cmsPromise = createCmsApp({\n dataSource,\n config: process.env as unknown as Record<string, string>,\n plugins: [\n localStoragePlugin({ dir: 'public/uploads' }),\n ],\n });\n return cmsPromise;\n}\n`,\n\n 'src/app/api/[[...path]]/route.ts': `import { NextResponse } from 'next/server';\nimport { getServerSession } from 'next-auth';\nimport { createCmsApiHandler } from '@infuro/cms-core/api';\nimport { CMS_ENTITY_MAP } from '@infuro/cms-core';\nimport { getDataSourceInitialized } from '@/lib/data-source';\nimport { requireAuth } from '@/lib/auth-helpers';\nimport { getCms } from '@/lib/cms';\nimport bcrypt from 'bcryptjs';\n\nconst baseUrl = process.env.NEXTAUTH_URL || 'http://localhost:3000';\n\nlet handlerPromise: Promise<ReturnType<typeof createCmsApiHandler>> | null = null;\n\nasync function getHandler() {\n if (!handlerPromise) {\n const dataSource = await getDataSourceInitialized();\n handlerPromise = Promise.resolve(\n createCmsApiHandler({\n dataSource,\n entityMap: CMS_ENTITY_MAP,\n requireAuth,\n json: NextResponse.json.bind(NextResponse),\n getCms,\n userAuth: {\n dataSource,\n entityMap: CMS_ENTITY_MAP,\n json: NextResponse.json.bind(NextResponse),\n baseUrl,\n hashPassword: (p) => Promise.resolve(bcrypt.hashSync(p, 12)),\n comparePassword: (p, h) => Promise.resolve(bcrypt.compareSync(p, h)),\n resetExpiryHours: 1,\n getSession: () =>\n getServerSession().then((s) => (s ? { user: s.user } : null)),\n },\n dashboard: {\n dataSource,\n entityMap: CMS_ENTITY_MAP,\n json: NextResponse.json.bind(NextResponse),\n requireAuth,\n requirePermission: requireAuth,\n },\n upload: {\n json: NextResponse.json.bind(NextResponse),\n requireAuth,\n storage: () => getCms().then((cms) => cms.getPlugin('storage')),\n localUploadDir: 'public/uploads',\n },\n blogBySlug: {\n dataSource,\n entityMap: CMS_ENTITY_MAP,\n json: NextResponse.json.bind(NextResponse),\n requireAuth: async () => null,\n },\n formBySlug: {\n dataSource,\n entityMap: CMS_ENTITY_MAP,\n json: NextResponse.json.bind(NextResponse),\n requireAuth: async () => null,\n },\n usersApi: {\n dataSource,\n entityMap: CMS_ENTITY_MAP,\n json: NextResponse.json.bind(NextResponse),\n requireAuth,\n baseUrl,\n },\n })\n );\n }\n return handlerPromise;\n}\n\nasync function handle(method: string, req: Request, context: { params: Promise<{ path?: string[] }> }) {\n try {\n const handler = await getHandler();\n const { path: pathSegments = [] } = await context.params;\n return handler.handle(method, pathSegments, req);\n } catch {\n return NextResponse.json({ error: 'Server Error' }, { status: 500 });\n }\n}\n\nexport async function GET(req: Request, ctx: { params: Promise<{ path?: string[] }> }) { return handle('GET', req, ctx); }\nexport async function POST(req: Request, ctx: { params: Promise<{ path?: string[] }> }) { return handle('POST', req, ctx); }\nexport async function PUT(req: Request, ctx: { params: Promise<{ path?: string[] }> }) { return handle('PUT', req, ctx); }\nexport async function PATCH(req: Request, ctx: { params: Promise<{ path?: string[] }> }) { return handle('PATCH', req, ctx); }\nexport async function DELETE(req: Request, ctx: { params: Promise<{ path?: string[] }> }) { return handle('DELETE', req, ctx); }\n`,\n\n 'src/app/api/auth/[...nextauth]/route.ts': `import NextAuth from 'next-auth';\nimport { getNextAuthOptions } from '@infuro/cms-core/auth';\nimport { getDataSourceInitialized } from '@/lib/data-source';\nimport { CMS_ENTITY_MAP } from '@infuro/cms-core';\nimport bcrypt from 'bcryptjs';\n\nasync function getOptions() {\n const dataSource = await getDataSourceInitialized();\n const userRepo = dataSource.getRepository(CMS_ENTITY_MAP.users);\n return getNextAuthOptions({\n getUserByEmail: async (email: string) => {\n return userRepo.findOne({\n where: { email },\n relations: ['group', 'group.permissions'],\n select: ['id', 'email', 'name', 'password', 'blocked', 'deleted', 'groupId', 'adminAccess'],\n }) as any;\n },\n comparePassword: (plain, hash) => Promise.resolve(bcrypt.compareSync(plain, hash)),\n signInPage: '/admin/signin',\n });\n}\n\nlet handler: ReturnType<typeof NextAuth> | null = null;\n\nasync function getHandler() {\n if (!handler) handler = NextAuth(await getOptions());\n return handler;\n}\n\ntype NextAuthContext = { params: Promise<{ nextauth?: string[] }> };\n\nexport async function GET(req: Request, context: NextAuthContext) {\n return (await getHandler())(req, context);\n}\nexport async function POST(req: Request, context: NextAuthContext) {\n return (await getHandler())(req, context);\n}\n`,\n\n 'src/app/admin/layout.tsx': `'use client';\n\nimport '@infuro/cms-core/admin.css';\nimport AdminLayout from '@infuro/cms-core/admin';\n\nexport default function AdminLayoutWrapper({ children }: { children: React.ReactNode }) {\n return (\n <AdminLayout\n customNavItems={[]}\n customNavSections={[]}\n customCrudConfigs={{}}\n >\n {children}\n </AdminLayout>\n );\n}\n`,\n\n 'src/app/admin/[[...slug]]/page.tsx': `import { AdminPageResolver } from '@infuro/cms-core/admin';\n\nexport default async function AdminPage({ params }: { params: Promise<{ slug?: string[] }> }) {\n const { slug } = await params;\n return <AdminPageResolver slug={slug} />;\n}\n`,\n\n 'src/middleware.ts': `import { NextResponse } from 'next/server';\nimport type { NextRequest } from 'next/server';\nimport { createCmsMiddleware } from '@infuro/cms-core/auth';\n\nconst cmsMiddleware = createCmsMiddleware({\n publicApiMethods: {\n '/api/contacts': ['POST'],\n '/api/form-submissions': ['POST'],\n '/api/blogs': ['GET'],\n '/api/forms': ['GET'],\n '/api/auth': ['GET', 'POST'],\n '/api/users/forgot-password': ['POST'],\n '/api/users/set-password': ['POST'],\n '/api/users/invite': ['POST'],\n },\n});\n\nexport function middleware(request: NextRequest) {\n const result = cmsMiddleware({\n nextUrl: request.nextUrl,\n url: request.url,\n method: request.method,\n cookies: request.cookies,\n });\n\n if (result.type === 'next') return NextResponse.next();\n if (result.type === 'redirect') return NextResponse.redirect(result.url);\n if (result.type === 'json') return NextResponse.json(result.body, { status: result.status });\n return NextResponse.next();\n}\n\nexport const config = {\n matcher: ['/admin/:path*', '/api/:path*'],\n};\n`,\n\n 'src/app/providers.tsx': `\"use client\";\n\nimport { ThemeProvider } from \"next-themes\";\nimport { SessionProvider } from \"next-auth/react\";\nimport { Toaster } from \"sonner\";\n\nexport function Providers({ children }: { children: React.ReactNode }) {\n return (\n <SessionProvider>\n <ThemeProvider attribute=\"class\" defaultTheme=\"system\" enableSystem>\n {children}\n <Toaster position=\"top-right\" />\n </ThemeProvider>\n </SessionProvider>\n );\n}\n`,\n\n '.env.example': `DATABASE_URL=postgres://user:password@localhost:5432/mydb\nNEXTAUTH_SECRET=your-random-secret\nNEXTAUTH_URL=http://localhost:3000\n\n# Admin user (for npm run seed)\nADMIN_EMAIL=admin@example.com\nADMIN_PASSWORD=changeme\n`,\n\n 'src/lib/seed.ts': `try { require('dotenv/config'); } catch {}\nimport 'reflect-metadata';\nimport { getDataSourceInitialized } from './data-source';\nimport { CMS_ENTITY_MAP } from '@infuro/cms-core';\nimport bcrypt from 'bcryptjs';\n\nasync function main() {\n const ds = await getDataSourceInitialized();\n const userRepo = ds.getRepository(CMS_ENTITY_MAP.users);\n\n const email = process.env.ADMIN_EMAIL || 'admin@example.com';\n const password = process.env.ADMIN_PASSWORD || 'changeme';\n\n const existing = await userRepo.findOne({ where: { email } });\n if (!existing) {\n const hashedPassword = await bcrypt.hash(password, 10);\n await userRepo.save(userRepo.create({ name: 'Admin', email, password: hashedPassword }));\n console.log('Default admin user created');\n } else {\n console.log('Default admin user already exists');\n }\n\n await ds.destroy();\n}\n\nmain().catch((e) => {\n console.error(e);\n process.exit(1);\n});\n`,\n\n 'scripts/migration-datasource.cjs': `/**\n * Data source for TypeORM CLI (migration:generate). Resolves @infuro/cms-core from project root (works with npm link).\n */\nrequire('reflect-metadata');\nrequire('dotenv/config');\nconst path = require('path');\nconst { DataSource } = require('typeorm');\nconst coreEntry = path.resolve(__dirname, '..', 'node_modules', '@infuro', 'cms-core', 'dist', 'index.cjs');\nconst { CMS_ENTITY_MAP } = require(coreEntry);\n\nmodule.exports = new DataSource({\n type: 'postgres',\n url: process.env.DATABASE_URL,\n entities: Object.values(CMS_ENTITY_MAP),\n synchronize: false,\n migrations: ['src/migrations/*.ts'],\n});\n`,\n\n 'scripts/run-migrations.ts': `/**\n * Run TypeORM migrations. Loads .env so DATABASE_URL is set.\n * Usage: npm run migration:run\n */\ntry { require('dotenv/config'); } catch {}\nimport 'reflect-metadata';\n\nasync function main() {\n process.env.TYPEORM_CLI = '1';\n const { getDataSourceInitialized } = await import('../src/lib/data-source');\n const ds = await getDataSourceInitialized();\n const run = await ds.runMigrations();\n console.log(run.length ? \\`Ran \\${run.length} migration(s).\\` : 'No pending migrations.');\n await ds.destroy();\n}\n\nmain().catch((e) => {\n console.error(e);\n process.exit(1);\n});\n`,\n\n 'src/migrations/README.md': `# TypeORM migrations\n\nGenerate a new migration (after changing entities):\n\n\\`\\`\\`bash\nnpm run migration:generate -- MyMigrationName\n\\`\\`\\`\n\nRun pending migrations:\n\n\\`\\`\\`bash\nnpm run migration:run\n\\`\\`\\`\n`,\n\n 'src/themes/default/index.ts': `import { createTheme } from '@infuro/cms-core/theme';\n\nimport { Container, meta as containerMeta } from './components/Container';\nimport { TextBlock, meta as textBlockMeta } from './components/TextBlock';\n\nimport { Navbar } from './layout/Navbar';\nimport { Footer, footerFields, footerDefaults } from './layout/Footer';\n\nexport default createTheme({\n name: 'default',\n label: 'Default Theme',\n components: [\n { component: Container, meta: containerMeta },\n { component: TextBlock, meta: textBlockMeta },\n ],\n layout: {\n navbar: { component: Navbar },\n footer: { component: Footer, fields: footerFields, defaults: footerDefaults },\n },\n});\n`,\n\n 'src/themes/default/layout/Navbar.tsx': `import type { NavbarConfig, NavItem } from '@infuro/cms-core/theme';\n\nfunction NavLink({ item }: { item: NavItem }) {\n return (\n <li className=\"list-none\">\n <a\n href={item.url}\n target={item.openInNewTab ? '_blank' : undefined}\n rel={item.openInNewTab ? 'noopener noreferrer' : undefined}\n className=\"text-sm font-medium text-gray-700 hover:text-gray-900 px-3 py-2 inline-block\"\n >\n {item.label}\n </a>\n </li>\n );\n}\n\nexport function Navbar({ logo, items, ctaLabel, ctaUrl }: NavbarConfig) {\n return (\n <nav className=\"bg-white sticky top-0 z-40 border-b\">\n <div className=\"max-w-7xl mx-auto px-4 sm:px-6 lg:px-8\">\n <div className=\"flex items-center justify-between h-16\">\n <a href=\"/\" className=\"flex-shrink-0\">\n {logo ? (\n <img src={logo} alt=\"Logo\" className=\"h-8\" />\n ) : (\n <span className=\"text-xl font-bold text-gray-900\">Logo</span>\n )}\n </a>\n <ul className=\"flex items-center gap-1 list-none m-0 p-0\">\n {items.map((item) => (\n <NavLink key={item.id} item={item} />\n ))}\n </ul>\n {ctaLabel && (\n <a href={ctaUrl || '#'} className=\"text-sm font-medium text-gray-900 hover:underline\">\n {ctaLabel}\n </a>\n )}\n </div>\n </div>\n </nav>\n );\n}\n`,\n\n 'src/themes/default/layout/Footer.tsx': `import type { FooterConfig, PropDefinition } from '@infuro/cms-core/theme';\n\nexport const footerFields: PropDefinition[] = [\n { name: 'copyright', label: 'Copyright Text', type: 'text' },\n];\n\nexport const footerDefaults: Record<string, any> = {\n copyright: '© 2025 Your Site. All rights reserved.',\n columns: [],\n socialLinks: [],\n};\n\nexport function Footer({ copyright, columns = [], socialLinks = [] }: FooterConfig) {\n return (\n <footer className=\"bg-gray-900 text-gray-400 py-8 px-8\">\n <div className=\"max-w-7xl mx-auto\">\n {columns.length > 0 && (\n <div className=\"grid grid-cols-2 md:grid-cols-4 gap-8 mb-6\">\n {columns.map((col, i) => (\n <div key={i}>\n <h4 className=\"text-white font-semibold mb-3 text-sm\">{col.title}</h4>\n <ul className=\"space-y-2\">\n {col.links.map((link, j) => (\n <li key={j}>\n <a href={link.url} className=\"text-sm hover:text-white transition-colors\">\n {link.label}\n </a>\n </li>\n ))}\n </ul>\n </div>\n ))}\n </div>\n )}\n <div className=\"border-t border-gray-800 pt-6\">\n <p className=\"text-sm\">{copyright}</p>\n </div>\n </div>\n </footer>\n );\n}\n`,\n\n 'src/themes/default/components/Container.tsx': `import type { ComponentMeta } from '@infuro/cms-core/theme';\n\nexport const meta: ComponentMeta = {\n name: 'Container',\n label: 'Container',\n category: 'layout',\n icon: 'LayoutDashboard',\n description: 'A layout container that holds other components',\n defaultProps: { background: '#ffffff' },\n props: [{ name: 'background', label: 'Background', type: 'color' }],\n canContainChildren: true,\n};\n\nexport function Container({\n background = '#ffffff',\n children,\n}: {\n background?: string;\n children?: React.ReactNode;\n}) {\n return (\n <div style={{ backgroundColor: background, minHeight: '48px', padding: '1rem' }}>\n {children}\n </div>\n );\n}\n`,\n\n 'src/themes/default/components/TextBlock.tsx': `import type { ComponentMeta } from '@infuro/cms-core/theme';\n\nexport const meta: ComponentMeta = {\n name: 'TextBlock',\n label: 'Text Block',\n category: 'content',\n icon: 'Type',\n description: 'Rich text content block',\n defaultProps: { content: '<p>Enter your text here...</p>' },\n props: [{ name: 'content', label: 'Content', type: 'richtext' }],\n};\n\nexport function TextBlock({ content }: { content?: string }) {\n return (\n <div\n className=\"prose prose-gray max-w-none py-4 px-2\"\n dangerouslySetInnerHTML={{ __html: content || '' }}\n />\n );\n}\n`,\n\n 'src/lib/theme-registry.ts': `import defaultTheme from '@/themes/default';\nimport type { ThemeConfig } from '@infuro/cms-core/theme';\n\nexport const defaultThemeConfig = defaultTheme;\n\nexport interface ThemeRegistryItem {\n id: string;\n label: string;\n config: ThemeConfig;\n description?: string;\n}\n\nexport const THEME_REGISTRY: ThemeRegistryItem[] = [\n {\n id: 'default',\n label: 'Default',\n config: defaultTheme,\n description: 'Default theme with standard layout and components.',\n },\n];\n\nexport function getThemeById(id: string | undefined): ThemeConfig {\n const theme = THEME_REGISTRY.find((t) => t.id === (id || '')) ?? THEME_REGISTRY[0];\n return theme?.config ?? defaultTheme;\n}\n`,\n\n 'src/app/page.tsx': `export default function HomePage() {\n return (\n <main className=\"min-h-screen flex flex-col items-center justify-center p-8\">\n <h1 className=\"text-3xl font-bold text-gray-900 mb-4\">Welcome</h1>\n <p className=\"text-gray-600 mb-6\">Your CMS is set up. Manage content at the admin panel.</p>\n <a\n href=\"/admin\"\n className=\"text-white bg-gray-900 hover:bg-gray-800 px-4 py-2 rounded-lg font-medium transition-colors\"\n >\n Open Admin\n </a>\n </main>\n );\n}\n`,\n\n 'src/app/contact/page.tsx': `export default function ContactPage() {\n return (\n <main className=\"min-h-screen p-8 max-w-2xl mx-auto\">\n <h1 className=\"text-3xl font-bold text-gray-900 mb-4\">Contact</h1>\n <p className=\"text-gray-600\">\n Add a contact form or wire this page to your CMS form. Use the admin panel to manage forms and submissions.\n </p>\n </main>\n );\n}\n`,\n};\n\nfunction findRoot(cwd: string): string | null {\n let dir = path.resolve(cwd);\n for (let i = 0; i < 20; i++) {\n const pkgPath = path.join(dir, 'package.json');\n if (fs.existsSync(pkgPath)) {\n try {\n const pkg = JSON.parse(fs.readFileSync(pkgPath, 'utf8'));\n if (pkg.dependencies?.next || pkg.devDependencies?.next) return dir;\n } catch {\n // ignore\n }\n }\n const parent = path.dirname(dir);\n if (parent === dir) break;\n dir = parent;\n }\n return null;\n}\n\nfunction writeFile(\n root: string,\n filePath: string,\n content: string,\n force: boolean,\n dryRun: boolean,\n log: (msg: string) => void\n): boolean {\n const full = path.join(root, filePath);\n if (fs.existsSync(full) && !force) {\n log(` skip (exists): ${filePath}`);\n return false;\n }\n if (dryRun) {\n log(` would create: ${filePath}`);\n return true;\n }\n const dir = path.dirname(full);\n fs.mkdirSync(dir, { recursive: true });\n fs.writeFileSync(full, content, 'utf8');\n log(` created: ${filePath}`);\n return true;\n}\n\nfunction patchNextConfig(root: string, dryRun: boolean, log: (msg: string) => void): boolean {\n const candidates = ['next.config.js', 'next.config.mjs', 'next.config.cjs'];\n let configPath: string | null = null;\n for (const name of candidates) {\n const p = path.join(root, name);\n if (fs.existsSync(p)) {\n configPath = p;\n break;\n }\n }\n if (!configPath) {\n log(' skip next.config: not found');\n return false;\n }\n let content = fs.readFileSync(configPath, 'utf8');\n if (content.includes(\"'@infuro/cms-core'\") || content.includes('\"@infuro/cms-core\"')) {\n log(` skip (already has core): ${path.basename(configPath)}`);\n return false;\n }\n if (content.includes('serverExternalPackages')) {\n content = content.replace(\n /(serverExternalPackages:\\s*\\[)/,\n \"$1'@infuro/cms-core', 'typeorm', \"\n );\n } else {\n content = content.replace(\n /(const nextConfig\\s*=\\s*\\{|module\\.exports\\s*=\\s*\\{)/,\n \"$1\\n serverExternalPackages: ['@infuro/cms-core', 'typeorm'],\"\n );\n }\n if (dryRun) {\n log(` would patch: ${path.basename(configPath)}`);\n return true;\n }\n fs.writeFileSync(configPath, content, 'utf8');\n log(` patched: ${path.basename(configPath)}`);\n return true;\n}\n\nfunction patchTailwind(root: string, dryRun: boolean, log: (msg: string) => void): boolean {\n const candidates = ['tailwind.config.js', 'tailwind.config.mjs', 'tailwind.config.ts'];\n let configPath: string | null = null;\n for (const name of candidates) {\n const p = path.join(root, name);\n if (fs.existsSync(p)) {\n configPath = p;\n break;\n }\n }\n if (!configPath) {\n log(' skip tailwind: config not found');\n return false;\n }\n let content = fs.readFileSync(configPath, 'utf8');\n const coreContent = \"./node_modules/@infuro/cms-core/dist/**/*.{js,cjs}\";\n if (content.includes('@infuro/cms-core')) {\n log(` skip (already has core): ${path.basename(configPath)}`);\n return false;\n }\n if (content.includes('content:')) {\n content = content.replace(\n /(content:\\s*\\[)/,\n `$1\\n \"${coreContent}\",`\n );\n }\n if (dryRun) {\n log(` would patch: ${path.basename(configPath)}`);\n return true;\n }\n fs.writeFileSync(configPath, content, 'utf8');\n log(` patched: ${path.basename(configPath)}`);\n return true;\n}\n\nfunction patchLayout(root: string, dryRun: boolean, log: (msg: string) => void): boolean {\n const layoutPath = path.join(root, 'src/app/layout.tsx');\n if (!fs.existsSync(layoutPath)) {\n log(' skip layout: src/app/layout.tsx not found');\n return false;\n }\n let content = fs.readFileSync(layoutPath, 'utf8');\n if (content.includes('<Providers>')) {\n log(' skip layout: already uses Providers');\n return false;\n }\n const bodyMatch = content.match(/<body([^>]*)>\\s*(\\{children\\})\\s*<\\/body>/s);\n if (!bodyMatch) {\n log(' skip layout: unexpected structure (add <Providers> manually)');\n return false;\n }\n if (dryRun) {\n log(' would patch: src/app/layout.tsx');\n return true;\n }\n const [, bodyAttrs, children] = bodyMatch;\n const newBody = `<body${bodyAttrs}>\\n <Providers>${children}</Providers>\\n </body>`;\n content = content.replace(/<body[^>]*>\\s*\\{children\\}\\s*<\\/body>/s, newBody);\n if (!content.includes(\"from './providers'\") && !content.includes('from \"./providers\"')) {\n const firstImport = content.match(/^import .+ from .+;\\n/m);\n content = firstImport\n ? content.replace(firstImport[0], firstImport[0] + \"import { Providers } from './providers';\\n\")\n : \"import { Providers } from './providers';\\n\" + content;\n }\n if (content.includes('<html') && !content.includes('suppressHydrationWarning')) {\n content = content.replace(/<html(\\s)/, '<html suppressHydrationWarning$1');\n }\n fs.writeFileSync(layoutPath, content, 'utf8');\n log(' patched: src/app/layout.tsx');\n return true;\n}\n\nfunction patchPackageJson(root: string, dryRun: boolean, log: (msg: string) => void): boolean {\n const pkgPath = path.join(root, 'package.json');\n if (!fs.existsSync(pkgPath)) return false;\n const pkg = JSON.parse(fs.readFileSync(pkgPath, 'utf8'));\n const scripts = pkg.scripts || {};\n let changed = false;\n if (!scripts.seed) {\n scripts.seed = 'tsx src/lib/seed.ts';\n changed = true;\n }\n if (!scripts['migration:run']) {\n scripts['migration:run'] = 'tsx scripts/run-migrations.ts';\n changed = true;\n }\n const dev = pkg.devDependencies || {};\n const deps = pkg.dependencies || {};\n if (!deps['@infuro/cms-core']) {\n deps['@infuro/cms-core'] = '^1.0.6';\n changed = true;\n }\n if (!dev.tsx) {\n dev.tsx = '^4.0.0';\n changed = true;\n }\n if (!dev.dotenv) {\n dev.dotenv = '^16.0.0';\n changed = true;\n }\n if (!changed) {\n log(' skip package.json: scripts/devDeps already present');\n return false;\n }\n pkg.scripts = scripts;\n pkg.dependencies = { ...pkg.dependencies, ...deps };\n pkg.devDependencies = { ...pkg.devDependencies, ...dev };\n if (dryRun) {\n log(' would patch: package.json (scripts + devDependencies)');\n return true;\n }\n fs.writeFileSync(pkgPath, JSON.stringify(pkg, null, 2), 'utf8');\n log(' patched: package.json');\n return true;\n}\n\nasync function runNpmInstall(root: string, log: (msg: string) => void): Promise<void> {\n const { spawnSync } = await import('child_process');\n log(' running: npm install (deps)...');\n spawnSync('npm', ['install', '@infuro/cms-core', 'typeorm', 'reflect-metadata', 'bcryptjs', 'next-auth', 'next-themes', 'sonner'], {\n cwd: root,\n stdio: 'inherit',\n shell: true,\n });\n log(' running: npm install -D tsx dotenv @types/node...');\n spawnSync('npm', ['install', '-D', 'tsx', 'dotenv', '@types/node'], {\n cwd: root,\n stdio: 'inherit',\n shell: true,\n });\n}\n\nasync function runInit(opts: {\n force: boolean;\n dryRun: boolean;\n noDeps: boolean;\n noPatchConfig: boolean;\n}) {\n const log = (msg: string) => console.log(msg);\n const cwd = process.cwd();\n const root = findRoot(cwd);\n if (!root) {\n console.error('Not a Next.js project (no package.json with next dependency found from ' + cwd + ')');\n process.exit(1);\n }\n const appDir = path.join(root, 'src/app');\n if (!fs.existsSync(appDir)) {\n console.error('Expected src/app directory not found. Use a Next.js app with src directory (e.g. create-next-app --src-dir).');\n process.exit(1);\n }\n\n log('Infuro CMS init @ ' + root);\n if (opts.dryRun) log('(dry run)');\n\n for (const [filePath, content] of Object.entries(TEMPLATES)) {\n writeFile(root, filePath, content, opts.force, opts.dryRun, log);\n }\n\n if (!opts.noPatchConfig) {\n log('Config patches:');\n patchNextConfig(root, opts.dryRun, log);\n patchTailwind(root, opts.dryRun, log);\n patchLayout(root, opts.dryRun, log);\n patchPackageJson(root, opts.dryRun, log);\n }\n\n if (!opts.noDeps && !opts.dryRun) {\n log('Dependencies:');\n await runNpmInstall(root, log);\n } else if (!opts.noDeps && opts.dryRun) {\n log(' would run: npm install @infuro/cms-core typeorm reflect-metadata bcryptjs next-auth next-themes sonner');\n log(' would run: npm install -D tsx dotenv @types/node');\n }\n\n log('');\n log('Done. Next steps:');\n log(' 1. Copy .env.example to .env and set DATABASE_URL, NEXTAUTH_SECRET, NEXTAUTH_URL, ADMIN_EMAIL, ADMIN_PASSWORD');\n log(' 2. Run npm run migration:run then npm run seed (creates admin from ADMIN_EMAIL/ADMIN_PASSWORD)');\n log(' 3. npm run dev');\n}\n\nconst args = process.argv.slice(2);\nconst force = args.includes('--force');\nconst dryRun = args.includes('--dry-run');\nconst noDeps = args.includes('--no-deps');\nconst noPatchConfig = args.includes('--no-patch-config');\n\nif (args[0] === 'init' || args.includes('--init') || (args.length === 0 && !args.some((a) => a.startsWith('--')))) {\n runInit({ force, dryRun, noDeps, noPatchConfig }).catch((e) => {\n console.error(e);\n process.exit(1);\n });\n} else {\n console.log('Usage: npx @infuro/cms-core init [--force] [--dry-run] [--no-deps] [--no-patch-config]');\n process.exit(args[0] === '--help' || args[0] === '-h' ? 0 : 1);\n}\n"],"mappings":"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","names":["force","dryRun"]}
1
+ {"version":3,"sources":["../src/cli.ts"],"sourcesContent":["#!/usr/bin/env node\r\n/**\r\n * @infuro/cms-core init CLI\r\n * Usage: npx @infuro/cms-core init [--force] [--dry-run] [--no-deps] [--no-patch-config]\r\n */\r\nimport fs from 'fs';\r\nimport path from 'path';\r\n\r\nconst TEMPLATES = {\r\n 'src/lib/data-source.ts': `import 'reflect-metadata';\r\nimport path from 'path';\r\nimport { createRequire } from 'module';\r\nimport { DataSource } from 'typeorm';\r\nimport { CMS_ENTITY_MAP } from '@infuro/cms-core';\r\n\r\nconst require = createRequire(import.meta.url);\r\nconst coreDir = path.dirname(require.resolve('@infuro/cms-core'));\r\n\r\nlet dataSource: DataSource | null = null;\r\n\r\nexport function getDataSource(): DataSource {\r\n if (!dataSource) {\r\n dataSource = new DataSource({\r\n type: 'postgres',\r\n url: process.env.DATABASE_URL,\r\n entities: Object.values(CMS_ENTITY_MAP),\r\n synchronize: false,\r\n ...(process.env.TYPEORM_CLI && {\r\n migrations: [\r\n path.join(coreDir, 'migrations', '*.ts'),\r\n path.join(process.cwd(), 'src', 'migrations', '*.ts'),\r\n ],\r\n }),\r\n });\r\n }\r\n return dataSource;\r\n}\r\n\r\nexport async function getDataSourceInitialized(): Promise<DataSource> {\r\n const ds = getDataSource();\r\n if (!ds.isInitialized) await ds.initialize();\r\n return ds;\r\n}\r\n\r\nexport default getDataSource;\r\n`,\r\n\r\n 'src/lib/auth-helpers.ts': `import { getServerSession } from 'next-auth';\r\nimport { NextResponse } from 'next/server';\r\nimport { createAuthHelpers } from '@infuro/cms-core/auth';\r\n\r\nconst helpers = createAuthHelpers(\r\n async () => {\r\n const s = await getServerSession();\r\n return s ? { user: s.user } : null;\r\n },\r\n NextResponse\r\n);\r\n\r\nexport const requireAuth = helpers.requireAuth;\r\nexport const requirePermission = helpers.requirePermission;\r\nexport const requireEntityPermission = helpers.requireEntityPermission;\r\nexport const requireAdminAccess = helpers.requireAdminAccess;\r\nexport const getAuthenticatedUser = helpers.getAuthenticatedUser;\r\n`,\r\n\r\n 'src/lib/cms.ts': `import {\r\n createCmsApp,\r\n localStoragePlugin,\r\n type CmsApp,\r\n} from '@infuro/cms-core';\r\nimport { getDataSourceInitialized } from './data-source';\r\n\r\nlet cmsPromise: Promise<CmsApp> | null = null;\r\n\r\nexport async function getCms(): Promise<CmsApp> {\r\n if (cmsPromise) return cmsPromise;\r\n const dataSource = await getDataSourceInitialized();\r\n cmsPromise = createCmsApp({\r\n dataSource,\r\n config: process.env as unknown as Record<string, string>,\r\n plugins: [\r\n localStoragePlugin({ dir: 'public/uploads' }),\r\n ],\r\n });\r\n return cmsPromise;\r\n}\r\n`,\r\n\r\n 'src/app/api/[[...path]]/route.ts': `import { NextResponse } from 'next/server';\r\nimport { getServerSession } from 'next-auth';\r\nimport { createCmsApiHandler } from '@infuro/cms-core/api';\r\nimport { CMS_ENTITY_MAP } from '@infuro/cms-core';\r\nimport { getDataSourceInitialized } from '@/lib/data-source';\r\nimport {\r\n requireAuth,\r\n requireAdminAccess,\r\n requireEntityPermission,\r\n getAuthenticatedUser,\r\n} from '@/lib/auth-helpers';\r\nimport { getCms } from '@/lib/cms';\r\nimport bcrypt from 'bcryptjs';\r\n\r\nconst baseUrl = process.env.NEXTAUTH_URL || 'http://localhost:3000';\r\n\r\nasync function requireAdminApiAuth(req: Request) {\r\n const a = await requireAuth(req);\r\n if (a) return a;\r\n return requireAdminAccess(req);\r\n}\r\n\r\nlet handlerPromise: Promise<ReturnType<typeof createCmsApiHandler>> | null = null;\r\n\r\nasync function getHandler() {\r\n if (!handlerPromise) {\r\n const dataSource = await getDataSourceInitialized();\r\n handlerPromise = Promise.resolve(\r\n createCmsApiHandler({\r\n dataSource,\r\n entityMap: CMS_ENTITY_MAP,\r\n requireAuth: requireAdminApiAuth,\r\n requireEntityPermission,\r\n getSessionUser: getAuthenticatedUser,\r\n json: NextResponse.json.bind(NextResponse),\r\n getCms,\r\n userAuth: {\r\n dataSource,\r\n entityMap: CMS_ENTITY_MAP,\r\n json: NextResponse.json.bind(NextResponse),\r\n baseUrl,\r\n hashPassword: (p) => Promise.resolve(bcrypt.hashSync(p, 12)),\r\n comparePassword: (p, h) => Promise.resolve(bcrypt.compareSync(p, h)),\r\n resetExpiryHours: 1,\r\n getSession: () =>\r\n getServerSession().then((s) => (s ? { user: s.user } : null)),\r\n },\r\n dashboard: {\r\n dataSource,\r\n entityMap: CMS_ENTITY_MAP,\r\n json: NextResponse.json.bind(NextResponse),\r\n requireAuth: requireAdminApiAuth,\r\n requirePermission: requireAdminApiAuth,\r\n },\r\n upload: {\r\n json: NextResponse.json.bind(NextResponse),\r\n requireAuth: requireAdminApiAuth,\r\n storage: () => getCms().then((cms) => cms.getPlugin('storage')),\r\n localUploadDir: 'public/uploads',\r\n },\r\n blogBySlug: {\r\n dataSource,\r\n entityMap: CMS_ENTITY_MAP,\r\n json: NextResponse.json.bind(NextResponse),\r\n requireAuth: async () => null,\r\n },\r\n formBySlug: {\r\n dataSource,\r\n entityMap: CMS_ENTITY_MAP,\r\n json: NextResponse.json.bind(NextResponse),\r\n requireAuth: async () => null,\r\n },\r\n usersApi: {\r\n dataSource,\r\n entityMap: CMS_ENTITY_MAP,\r\n json: NextResponse.json.bind(NextResponse),\r\n requireAuth: requireAdminApiAuth,\r\n baseUrl,\r\n },\r\n })\r\n );\r\n }\r\n return handlerPromise;\r\n}\r\n\r\nasync function handle(method: string, req: Request, context: { params: Promise<{ path?: string[] }> }) {\r\n try {\r\n const handler = await getHandler();\r\n const { path: pathSegments = [] } = await context.params;\r\n return handler.handle(method, pathSegments, req);\r\n } catch {\r\n return NextResponse.json({ error: 'Server Error' }, { status: 500 });\r\n }\r\n}\r\n\r\nexport async function GET(req: Request, ctx: { params: Promise<{ path?: string[] }> }) { return handle('GET', req, ctx); }\r\nexport async function POST(req: Request, ctx: { params: Promise<{ path?: string[] }> }) { return handle('POST', req, ctx); }\r\nexport async function PUT(req: Request, ctx: { params: Promise<{ path?: string[] }> }) { return handle('PUT', req, ctx); }\r\nexport async function PATCH(req: Request, ctx: { params: Promise<{ path?: string[] }> }) { return handle('PATCH', req, ctx); }\r\nexport async function DELETE(req: Request, ctx: { params: Promise<{ path?: string[] }> }) { return handle('DELETE', req, ctx); }\r\n`,\r\n\r\n 'src/app/api/auth/[...nextauth]/route.ts': `import NextAuth from 'next-auth';\r\nimport { getNextAuthOptions } from '@infuro/cms-core/auth';\r\nimport { getDataSourceInitialized } from '@/lib/data-source';\r\nimport { CMS_ENTITY_MAP } from '@infuro/cms-core';\r\nimport bcrypt from 'bcryptjs';\r\n\r\nasync function getOptions() {\r\n const dataSource = await getDataSourceInitialized();\r\n const userRepo = dataSource.getRepository(CMS_ENTITY_MAP.users);\r\n return getNextAuthOptions({\r\n getUserByEmail: async (email: string) => {\r\n return userRepo.findOne({\r\n where: { email },\r\n relations: ['group', 'group.permissions'],\r\n select: ['id', 'email', 'name', 'password', 'blocked', 'deleted', 'groupId', 'adminAccess'],\r\n }) as any;\r\n },\r\n comparePassword: (plain, hash) => Promise.resolve(bcrypt.compareSync(plain, hash)),\r\n signInPage: '/admin/signin',\r\n });\r\n}\r\n\r\nlet handler: ReturnType<typeof NextAuth> | null = null;\r\n\r\nasync function getHandler() {\r\n if (!handler) handler = NextAuth(await getOptions());\r\n return handler;\r\n}\r\n\r\ntype NextAuthContext = { params: Promise<{ nextauth?: string[] }> };\r\n\r\nexport async function GET(req: Request, context: NextAuthContext) {\r\n return (await getHandler())(req, context);\r\n}\r\nexport async function POST(req: Request, context: NextAuthContext) {\r\n return (await getHandler())(req, context);\r\n}\r\n`,\r\n\r\n 'src/app/admin/layout.tsx': `'use client';\r\n\r\nimport '@infuro/cms-core/admin.css';\r\nimport AdminLayout from '@infuro/cms-core/admin';\r\n\r\nexport default function AdminLayoutWrapper({ children }: { children: React.ReactNode }) {\r\n return (\r\n <AdminLayout\r\n customNavItems={[]}\r\n customNavSections={[]}\r\n customCrudConfigs={{}}\r\n >\r\n {children}\r\n </AdminLayout>\r\n );\r\n}\r\n`,\r\n\r\n 'src/app/admin/[[...slug]]/page.tsx': `import { AdminPageResolver } from '@infuro/cms-core/admin';\r\n\r\nexport default async function AdminPage({ params }: { params: Promise<{ slug?: string[] }> }) {\r\n const { slug } = await params;\r\n return <AdminPageResolver slug={slug} />;\r\n}\r\n`,\r\n\r\n 'src/middleware.ts': `import { NextResponse } from 'next/server';\r\nimport type { NextRequest } from 'next/server';\r\nimport { createCmsMiddleware } from '@infuro/cms-core/auth';\r\n\r\nconst cmsMiddleware = createCmsMiddleware({\r\n publicApiMethods: {\r\n '/api/contacts': ['POST'],\r\n '/api/form-submissions': ['POST'],\r\n '/api/blogs': ['GET'],\r\n '/api/forms': ['GET'],\r\n '/api/auth': ['GET', 'POST'],\r\n '/api/users/forgot-password': ['POST'],\r\n '/api/users/set-password': ['POST'],\r\n '/api/users/invite': ['POST'],\r\n },\r\n});\r\n\r\nexport function middleware(request: NextRequest) {\r\n const result = cmsMiddleware({\r\n nextUrl: request.nextUrl,\r\n url: request.url,\r\n method: request.method,\r\n cookies: request.cookies,\r\n });\r\n\r\n if (result.type === 'next') return NextResponse.next();\r\n if (result.type === 'redirect') return NextResponse.redirect(result.url);\r\n if (result.type === 'json') return NextResponse.json(result.body, { status: result.status });\r\n return NextResponse.next();\r\n}\r\n\r\nexport const config = {\r\n matcher: ['/admin/:path*', '/api/:path*'],\r\n};\r\n`,\r\n\r\n 'src/app/providers.tsx': `\"use client\";\r\n\r\nimport { ThemeProvider } from \"next-themes\";\r\nimport { SessionProvider } from \"next-auth/react\";\r\nimport { Toaster } from \"sonner\";\r\n\r\nexport function Providers({ children }: { children: React.ReactNode }) {\r\n return (\r\n <SessionProvider>\r\n <ThemeProvider attribute=\"class\" defaultTheme=\"system\" enableSystem>\r\n {children}\r\n <Toaster position=\"top-right\" />\r\n </ThemeProvider>\r\n </SessionProvider>\r\n );\r\n}\r\n`,\r\n\r\n '.env.example': `DATABASE_URL=postgres://user:password@localhost:5432/mydb\r\nNEXTAUTH_SECRET=your-random-secret\r\nNEXTAUTH_URL=http://localhost:3000\r\n\r\n# Admin user (for npm run seed)\r\nADMIN_EMAIL=admin@example.com\r\nADMIN_PASSWORD=changeme\r\n`,\r\n\r\n 'src/lib/seed.ts': `try { require('dotenv/config'); } catch {}\r\nimport 'reflect-metadata';\r\nimport { getDataSourceInitialized } from './data-source';\r\nimport { CMS_ENTITY_MAP } from '@infuro/cms-core';\r\nimport bcrypt from 'bcryptjs';\r\n\r\nasync function main() {\r\n const ds = await getDataSourceInitialized();\r\n const userRepo = ds.getRepository(CMS_ENTITY_MAP.users);\r\n\r\n const email = process.env.ADMIN_EMAIL || 'admin@example.com';\r\n const password = process.env.ADMIN_PASSWORD || 'changeme';\r\n\r\n const existing = await userRepo.findOne({ where: { email } });\r\n if (!existing) {\r\n const hashedPassword = await bcrypt.hash(password, 10);\r\n await userRepo.save(userRepo.create({ name: 'Admin', email, password: hashedPassword }));\r\n console.log('Default admin user created');\r\n } else {\r\n console.log('Default admin user already exists');\r\n }\r\n\r\n await ds.destroy();\r\n}\r\n\r\nmain().catch((e) => {\r\n console.error(e);\r\n process.exit(1);\r\n});\r\n`,\r\n\r\n 'scripts/migration-datasource.cjs': `/**\r\n * Data source for TypeORM CLI (migration:generate). Resolves @infuro/cms-core from project root (works with npm link).\r\n */\r\nrequire('reflect-metadata');\r\nrequire('dotenv/config');\r\nconst path = require('path');\r\nconst { DataSource } = require('typeorm');\r\nconst coreEntry = path.resolve(__dirname, '..', 'node_modules', '@infuro', 'cms-core', 'dist', 'index.cjs');\r\nconst { CMS_ENTITY_MAP } = require(coreEntry);\r\n\r\nmodule.exports = new DataSource({\r\n type: 'postgres',\r\n url: process.env.DATABASE_URL,\r\n entities: Object.values(CMS_ENTITY_MAP),\r\n synchronize: false,\r\n migrations: ['src/migrations/*.ts'],\r\n});\r\n`,\r\n\r\n 'scripts/run-migrations.ts': `/**\r\n * Run TypeORM migrations. Loads .env so DATABASE_URL is set.\r\n * Usage: npm run migration:run\r\n */\r\ntry { require('dotenv/config'); } catch {}\r\nimport 'reflect-metadata';\r\n\r\nasync function main() {\r\n process.env.TYPEORM_CLI = '1';\r\n const { getDataSourceInitialized } = await import('../src/lib/data-source');\r\n const ds = await getDataSourceInitialized();\r\n const run = await ds.runMigrations();\r\n console.log(run.length ? \\`Ran \\${run.length} migration(s).\\` : 'No pending migrations.');\r\n await ds.destroy();\r\n}\r\n\r\nmain().catch((e) => {\r\n console.error(e);\r\n process.exit(1);\r\n});\r\n`,\r\n\r\n 'src/migrations/README.md': `# TypeORM migrations\r\n\r\nGenerate a new migration (after changing entities):\r\n\r\n\\`\\`\\`bash\r\nnpm run migration:generate -- MyMigrationName\r\n\\`\\`\\`\r\n\r\nRun pending migrations:\r\n\r\n\\`\\`\\`bash\r\nnpm run migration:run\r\n\\`\\`\\`\r\n`,\r\n\r\n 'src/themes/default/index.ts': `import { createTheme } from '@infuro/cms-core/theme';\r\n\r\nimport { Container, meta as containerMeta } from './components/Container';\r\nimport { TextBlock, meta as textBlockMeta } from './components/TextBlock';\r\n\r\nimport { Navbar } from './layout/Navbar';\r\nimport { Footer, footerFields, footerDefaults } from './layout/Footer';\r\n\r\nexport default createTheme({\r\n name: 'default',\r\n label: 'Default Theme',\r\n components: [\r\n { component: Container, meta: containerMeta },\r\n { component: TextBlock, meta: textBlockMeta },\r\n ],\r\n layout: {\r\n navbar: { component: Navbar },\r\n footer: { component: Footer, fields: footerFields, defaults: footerDefaults },\r\n },\r\n});\r\n`,\r\n\r\n 'src/themes/default/layout/Navbar.tsx': `import type { NavbarConfig, NavItem } from '@infuro/cms-core/theme';\r\n\r\nfunction NavLink({ item }: { item: NavItem }) {\r\n return (\r\n <li className=\"list-none\">\r\n <a\r\n href={item.url}\r\n target={item.openInNewTab ? '_blank' : undefined}\r\n rel={item.openInNewTab ? 'noopener noreferrer' : undefined}\r\n className=\"text-sm font-medium text-gray-700 hover:text-gray-900 px-3 py-2 inline-block\"\r\n >\r\n {item.label}\r\n </a>\r\n </li>\r\n );\r\n}\r\n\r\nexport function Navbar({ logo, items, ctaLabel, ctaUrl }: NavbarConfig) {\r\n return (\r\n <nav className=\"bg-white sticky top-0 z-40 border-b\">\r\n <div className=\"max-w-7xl mx-auto px-4 sm:px-6 lg:px-8\">\r\n <div className=\"flex items-center justify-between h-16\">\r\n <a href=\"/\" className=\"flex-shrink-0\">\r\n {logo ? (\r\n <img src={logo} alt=\"Logo\" className=\"h-8\" />\r\n ) : (\r\n <span className=\"text-xl font-bold text-gray-900\">Logo</span>\r\n )}\r\n </a>\r\n <ul className=\"flex items-center gap-1 list-none m-0 p-0\">\r\n {items.map((item) => (\r\n <NavLink key={item.id} item={item} />\r\n ))}\r\n </ul>\r\n {ctaLabel && (\r\n <a href={ctaUrl || '#'} className=\"text-sm font-medium text-gray-900 hover:underline\">\r\n {ctaLabel}\r\n </a>\r\n )}\r\n </div>\r\n </div>\r\n </nav>\r\n );\r\n}\r\n`,\r\n\r\n 'src/themes/default/layout/Footer.tsx': `import type { FooterConfig, PropDefinition } from '@infuro/cms-core/theme';\r\n\r\nexport const footerFields: PropDefinition[] = [\r\n { name: 'copyright', label: 'Copyright Text', type: 'text' },\r\n];\r\n\r\nexport const footerDefaults: Record<string, any> = {\r\n copyright: '© 2025 Your Site. All rights reserved.',\r\n columns: [],\r\n socialLinks: [],\r\n};\r\n\r\nexport function Footer({ copyright, columns = [], socialLinks = [] }: FooterConfig) {\r\n return (\r\n <footer className=\"bg-gray-900 text-gray-400 py-8 px-8\">\r\n <div className=\"max-w-7xl mx-auto\">\r\n {columns.length > 0 && (\r\n <div className=\"grid grid-cols-2 md:grid-cols-4 gap-8 mb-6\">\r\n {columns.map((col, i) => (\r\n <div key={i}>\r\n <h4 className=\"text-white font-semibold mb-3 text-sm\">{col.title}</h4>\r\n <ul className=\"space-y-2\">\r\n {col.links.map((link, j) => (\r\n <li key={j}>\r\n <a href={link.url} className=\"text-sm hover:text-white transition-colors\">\r\n {link.label}\r\n </a>\r\n </li>\r\n ))}\r\n </ul>\r\n </div>\r\n ))}\r\n </div>\r\n )}\r\n <div className=\"border-t border-gray-800 pt-6\">\r\n <p className=\"text-sm\">{copyright}</p>\r\n </div>\r\n </div>\r\n </footer>\r\n );\r\n}\r\n`,\r\n\r\n 'src/themes/default/components/Container.tsx': `import type { ComponentMeta } from '@infuro/cms-core/theme';\r\n\r\nexport const meta: ComponentMeta = {\r\n name: 'Container',\r\n label: 'Container',\r\n category: 'layout',\r\n icon: 'LayoutDashboard',\r\n description: 'A layout container that holds other components',\r\n defaultProps: { background: '#ffffff' },\r\n props: [{ name: 'background', label: 'Background', type: 'color' }],\r\n canContainChildren: true,\r\n};\r\n\r\nexport function Container({\r\n background = '#ffffff',\r\n children,\r\n}: {\r\n background?: string;\r\n children?: React.ReactNode;\r\n}) {\r\n return (\r\n <div style={{ backgroundColor: background, minHeight: '48px', padding: '1rem' }}>\r\n {children}\r\n </div>\r\n );\r\n}\r\n`,\r\n\r\n 'src/themes/default/components/TextBlock.tsx': `import type { ComponentMeta } from '@infuro/cms-core/theme';\r\n\r\nexport const meta: ComponentMeta = {\r\n name: 'TextBlock',\r\n label: 'Text Block',\r\n category: 'content',\r\n icon: 'Type',\r\n description: 'Rich text content block',\r\n defaultProps: { content: '<p>Enter your text here...</p>' },\r\n props: [{ name: 'content', label: 'Content', type: 'richtext' }],\r\n};\r\n\r\nexport function TextBlock({ content }: { content?: string }) {\r\n return (\r\n <div\r\n className=\"prose prose-gray max-w-none py-4 px-2\"\r\n dangerouslySetInnerHTML={{ __html: content || '' }}\r\n />\r\n );\r\n}\r\n`,\r\n\r\n 'src/lib/theme-registry.ts': `import defaultTheme from '@/themes/default';\r\nimport type { ThemeConfig } from '@infuro/cms-core/theme';\r\n\r\nexport const defaultThemeConfig = defaultTheme;\r\n\r\nexport interface ThemeRegistryItem {\r\n id: string;\r\n label: string;\r\n config: ThemeConfig;\r\n description?: string;\r\n}\r\n\r\nexport const THEME_REGISTRY: ThemeRegistryItem[] = [\r\n {\r\n id: 'default',\r\n label: 'Default',\r\n config: defaultTheme,\r\n description: 'Default theme with standard layout and components.',\r\n },\r\n];\r\n\r\nexport function getThemeById(id: string | undefined): ThemeConfig {\r\n const theme = THEME_REGISTRY.find((t) => t.id === (id || '')) ?? THEME_REGISTRY[0];\r\n return theme?.config ?? defaultTheme;\r\n}\r\n`,\r\n\r\n 'src/app/page.tsx': `export default function HomePage() {\r\n return (\r\n <main className=\"min-h-screen flex flex-col items-center justify-center p-8\">\r\n <h1 className=\"text-3xl font-bold text-gray-900 mb-4\">Welcome</h1>\r\n <p className=\"text-gray-600 mb-6\">Your CMS is set up. Manage content at the admin panel.</p>\r\n <a\r\n href=\"/admin\"\r\n className=\"text-white bg-gray-900 hover:bg-gray-800 px-4 py-2 rounded-lg font-medium transition-colors\"\r\n >\r\n Open Admin\r\n </a>\r\n </main>\r\n );\r\n}\r\n`,\r\n\r\n 'src/app/contact/page.tsx': `export default function ContactPage() {\r\n return (\r\n <main className=\"min-h-screen p-8 max-w-2xl mx-auto\">\r\n <h1 className=\"text-3xl font-bold text-gray-900 mb-4\">Contact</h1>\r\n <p className=\"text-gray-600\">\r\n Add a contact form or wire this page to your CMS form. Use the admin panel to manage forms and submissions.\r\n </p>\r\n </main>\r\n );\r\n}\r\n`,\r\n};\r\n\r\nfunction findRoot(cwd: string): string | null {\r\n let dir = path.resolve(cwd);\r\n for (let i = 0; i < 20; i++) {\r\n const pkgPath = path.join(dir, 'package.json');\r\n if (fs.existsSync(pkgPath)) {\r\n try {\r\n const pkg = JSON.parse(fs.readFileSync(pkgPath, 'utf8'));\r\n if (pkg.dependencies?.next || pkg.devDependencies?.next) return dir;\r\n } catch {\r\n // ignore\r\n }\r\n }\r\n const parent = path.dirname(dir);\r\n if (parent === dir) break;\r\n dir = parent;\r\n }\r\n return null;\r\n}\r\n\r\nfunction writeFile(\r\n root: string,\r\n filePath: string,\r\n content: string,\r\n force: boolean,\r\n dryRun: boolean,\r\n log: (msg: string) => void\r\n): boolean {\r\n const full = path.join(root, filePath);\r\n if (fs.existsSync(full) && !force) {\r\n log(` skip (exists): ${filePath}`);\r\n return false;\r\n }\r\n if (dryRun) {\r\n log(` would create: ${filePath}`);\r\n return true;\r\n }\r\n const dir = path.dirname(full);\r\n fs.mkdirSync(dir, { recursive: true });\r\n fs.writeFileSync(full, content, 'utf8');\r\n log(` created: ${filePath}`);\r\n return true;\r\n}\r\n\r\nfunction patchNextConfig(root: string, dryRun: boolean, log: (msg: string) => void): boolean {\r\n const candidates = ['next.config.js', 'next.config.mjs', 'next.config.cjs'];\r\n let configPath: string | null = null;\r\n for (const name of candidates) {\r\n const p = path.join(root, name);\r\n if (fs.existsSync(p)) {\r\n configPath = p;\r\n break;\r\n }\r\n }\r\n if (!configPath) {\r\n log(' skip next.config: not found');\r\n return false;\r\n }\r\n let content = fs.readFileSync(configPath, 'utf8');\r\n if (content.includes(\"'@infuro/cms-core'\") || content.includes('\"@infuro/cms-core\"')) {\r\n log(` skip (already has core): ${path.basename(configPath)}`);\r\n return false;\r\n }\r\n if (content.includes('serverExternalPackages')) {\r\n content = content.replace(\r\n /(serverExternalPackages:\\s*\\[)/,\r\n \"$1'@infuro/cms-core', 'typeorm', \"\r\n );\r\n } else {\r\n content = content.replace(\r\n /(const nextConfig\\s*=\\s*\\{|module\\.exports\\s*=\\s*\\{)/,\r\n \"$1\\n serverExternalPackages: ['@infuro/cms-core', 'typeorm'],\"\r\n );\r\n }\r\n if (dryRun) {\r\n log(` would patch: ${path.basename(configPath)}`);\r\n return true;\r\n }\r\n fs.writeFileSync(configPath, content, 'utf8');\r\n log(` patched: ${path.basename(configPath)}`);\r\n return true;\r\n}\r\n\r\nfunction patchTailwind(root: string, dryRun: boolean, log: (msg: string) => void): boolean {\r\n const candidates = ['tailwind.config.js', 'tailwind.config.mjs', 'tailwind.config.ts'];\r\n let configPath: string | null = null;\r\n for (const name of candidates) {\r\n const p = path.join(root, name);\r\n if (fs.existsSync(p)) {\r\n configPath = p;\r\n break;\r\n }\r\n }\r\n if (!configPath) {\r\n log(' skip tailwind: config not found');\r\n return false;\r\n }\r\n let content = fs.readFileSync(configPath, 'utf8');\r\n const coreContent = \"./node_modules/@infuro/cms-core/dist/**/*.{js,cjs}\";\r\n if (content.includes('@infuro/cms-core')) {\r\n log(` skip (already has core): ${path.basename(configPath)}`);\r\n return false;\r\n }\r\n if (content.includes('content:')) {\r\n content = content.replace(\r\n /(content:\\s*\\[)/,\r\n `$1\\n \"${coreContent}\",`\r\n );\r\n }\r\n if (dryRun) {\r\n log(` would patch: ${path.basename(configPath)}`);\r\n return true;\r\n }\r\n fs.writeFileSync(configPath, content, 'utf8');\r\n log(` patched: ${path.basename(configPath)}`);\r\n return true;\r\n}\r\n\r\nfunction patchLayout(root: string, dryRun: boolean, log: (msg: string) => void): boolean {\r\n const layoutPath = path.join(root, 'src/app/layout.tsx');\r\n if (!fs.existsSync(layoutPath)) {\r\n log(' skip layout: src/app/layout.tsx not found');\r\n return false;\r\n }\r\n let content = fs.readFileSync(layoutPath, 'utf8');\r\n if (content.includes('<Providers>')) {\r\n log(' skip layout: already uses Providers');\r\n return false;\r\n }\r\n const bodyMatch = content.match(/<body([^>]*)>\\s*(\\{children\\})\\s*<\\/body>/s);\r\n if (!bodyMatch) {\r\n log(' skip layout: unexpected structure (add <Providers> manually)');\r\n return false;\r\n }\r\n if (dryRun) {\r\n log(' would patch: src/app/layout.tsx');\r\n return true;\r\n }\r\n const [, bodyAttrs, children] = bodyMatch;\r\n const newBody = `<body${bodyAttrs}>\\n <Providers>${children}</Providers>\\n </body>`;\r\n content = content.replace(/<body[^>]*>\\s*\\{children\\}\\s*<\\/body>/s, newBody);\r\n if (!content.includes(\"from './providers'\") && !content.includes('from \"./providers\"')) {\r\n const firstImport = content.match(/^import .+ from .+;\\n/m);\r\n content = firstImport\r\n ? content.replace(firstImport[0], firstImport[0] + \"import { Providers } from './providers';\\n\")\r\n : \"import { Providers } from './providers';\\n\" + content;\r\n }\r\n if (content.includes('<html') && !content.includes('suppressHydrationWarning')) {\r\n content = content.replace(/<html(\\s)/, '<html suppressHydrationWarning$1');\r\n }\r\n fs.writeFileSync(layoutPath, content, 'utf8');\r\n log(' patched: src/app/layout.tsx');\r\n return true;\r\n}\r\n\r\nfunction patchPackageJson(root: string, dryRun: boolean, log: (msg: string) => void): boolean {\r\n const pkgPath = path.join(root, 'package.json');\r\n if (!fs.existsSync(pkgPath)) return false;\r\n const pkg = JSON.parse(fs.readFileSync(pkgPath, 'utf8'));\r\n const scripts = pkg.scripts || {};\r\n let changed = false;\r\n if (!scripts.seed) {\r\n scripts.seed = 'tsx src/lib/seed.ts';\r\n changed = true;\r\n }\r\n if (!scripts['migration:run']) {\r\n scripts['migration:run'] = 'tsx scripts/run-migrations.ts';\r\n changed = true;\r\n }\r\n const dev = pkg.devDependencies || {};\r\n const deps = pkg.dependencies || {};\r\n if (!deps['@infuro/cms-core']) {\r\n deps['@infuro/cms-core'] = '^1.0.6';\r\n changed = true;\r\n }\r\n if (!dev.tsx) {\r\n dev.tsx = '^4.0.0';\r\n changed = true;\r\n }\r\n if (!dev.dotenv) {\r\n dev.dotenv = '^16.0.0';\r\n changed = true;\r\n }\r\n if (!changed) {\r\n log(' skip package.json: scripts/devDeps already present');\r\n return false;\r\n }\r\n pkg.scripts = scripts;\r\n pkg.dependencies = { ...pkg.dependencies, ...deps };\r\n pkg.devDependencies = { ...pkg.devDependencies, ...dev };\r\n if (dryRun) {\r\n log(' would patch: package.json (scripts + devDependencies)');\r\n return true;\r\n }\r\n fs.writeFileSync(pkgPath, JSON.stringify(pkg, null, 2), 'utf8');\r\n log(' patched: package.json');\r\n return true;\r\n}\r\n\r\nasync function runNpmInstall(root: string, log: (msg: string) => void): Promise<void> {\r\n const { spawnSync } = await import('child_process');\r\n log(' running: npm install (deps)...');\r\n spawnSync('npm', ['install', '@infuro/cms-core', 'typeorm', 'reflect-metadata', 'bcryptjs', 'next-auth', 'next-themes', 'sonner'], {\r\n cwd: root,\r\n stdio: 'inherit',\r\n shell: true,\r\n });\r\n log(' running: npm install -D tsx dotenv @types/node...');\r\n spawnSync('npm', ['install', '-D', 'tsx', 'dotenv', '@types/node'], {\r\n cwd: root,\r\n stdio: 'inherit',\r\n shell: true,\r\n });\r\n}\r\n\r\nasync function runInit(opts: {\r\n force: boolean;\r\n dryRun: boolean;\r\n noDeps: boolean;\r\n noPatchConfig: boolean;\r\n}) {\r\n const log = (msg: string) => console.log(msg);\r\n const cwd = process.cwd();\r\n const root = findRoot(cwd);\r\n if (!root) {\r\n console.error('Not a Next.js project (no package.json with next dependency found from ' + cwd + ')');\r\n process.exit(1);\r\n }\r\n const appDir = path.join(root, 'src/app');\r\n if (!fs.existsSync(appDir)) {\r\n console.error('Expected src/app directory not found. Use a Next.js app with src directory (e.g. create-next-app --src-dir).');\r\n process.exit(1);\r\n }\r\n\r\n log('Infuro CMS init @ ' + root);\r\n if (opts.dryRun) log('(dry run)');\r\n\r\n for (const [filePath, content] of Object.entries(TEMPLATES)) {\r\n writeFile(root, filePath, content, opts.force, opts.dryRun, log);\r\n }\r\n\r\n if (!opts.noPatchConfig) {\r\n log('Config patches:');\r\n patchNextConfig(root, opts.dryRun, log);\r\n patchTailwind(root, opts.dryRun, log);\r\n patchLayout(root, opts.dryRun, log);\r\n patchPackageJson(root, opts.dryRun, log);\r\n }\r\n\r\n if (!opts.noDeps && !opts.dryRun) {\r\n log('Dependencies:');\r\n await runNpmInstall(root, log);\r\n } else if (!opts.noDeps && opts.dryRun) {\r\n log(' would run: npm install @infuro/cms-core typeorm reflect-metadata bcryptjs next-auth next-themes sonner');\r\n log(' would run: npm install -D tsx dotenv @types/node');\r\n }\r\n\r\n log('');\r\n log('Done. Next steps:');\r\n log(' 1. Copy .env.example to .env and set DATABASE_URL, NEXTAUTH_SECRET, NEXTAUTH_URL, ADMIN_EMAIL, ADMIN_PASSWORD');\r\n log(' 2. Run npm run migration:run then npm run seed (creates admin from ADMIN_EMAIL/ADMIN_PASSWORD)');\r\n log(' 3. npm run dev');\r\n}\r\n\r\nconst args = process.argv.slice(2);\r\nconst force = args.includes('--force');\r\nconst dryRun = args.includes('--dry-run');\r\nconst noDeps = args.includes('--no-deps');\r\nconst noPatchConfig = args.includes('--no-patch-config');\r\n\r\nif (args[0] === 'init' || args.includes('--init') || (args.length === 0 && !args.some((a) => a.startsWith('--')))) {\r\n runInit({ force, dryRun, noDeps, noPatchConfig }).catch((e) => {\r\n console.error(e);\r\n process.exit(1);\r\n });\r\n} else {\r\n console.log('Usage: npx @infuro/cms-core init [--force] [--dry-run] [--no-deps] [--no-patch-config]');\r\n process.exit(args[0] === '--help' || args[0] === '-h' ? 0 : 1);\r\n}\r\n"],"mappings":"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","names":["force","dryRun"]}
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/hooks/index.ts","../src/hooks/use-mobile.ts","../src/hooks/use-analytics.ts","../src/hooks/use-plugin.ts","../src/components/CaptchaProvider.tsx"],"sourcesContent":["export { useIsMobile } from './use-mobile';\nexport { useAnalytics } from './use-analytics';\nexport { usePlugin, PluginProvider } from './use-plugin';\nexport { CaptchaProvider, useCaptchaPayload } from '../components/CaptchaProvider';\n","import { useState, useEffect } from 'react';\n\nconst MOBILE_BREAKPOINT = 768;\n\nexport function useIsMobile(): boolean {\n const [isMobile, setIsMobile] = useState<boolean | undefined>(undefined);\n\n useEffect(() => {\n const mql = window.matchMedia(`(max-width: ${MOBILE_BREAKPOINT - 1}px)`);\n const onChange = () => setIsMobile(window.innerWidth < MOBILE_BREAKPOINT);\n mql.addEventListener('change', onChange);\n setIsMobile(window.innerWidth < MOBILE_BREAKPOINT);\n return () => mql.removeEventListener('change', onChange);\n }, []);\n\n return !!isMobile;\n}\n","import { useEffect } from 'react';\n\n/**\n * Call from app with pathname from usePathname() (next/navigation).\n * Disables tracking on admin routes when pathname starts with /admin.\n */\nexport function useAnalytics(pathname: string | null): { isAdminRoute: boolean; shouldTrack: boolean } {\n const isAdminRoute = pathname != null && pathname.startsWith('/admin');\n\n useEffect(() => {\n if (isAdminRoute && typeof window !== 'undefined') {\n if (typeof (window as unknown as { gtag?: () => void }).gtag === 'function') {\n (window as unknown as { gtag: () => void }).gtag = () => {};\n }\n if (Array.isArray((window as unknown as { dataLayer?: unknown[] }).dataLayer)) {\n (window as unknown as { dataLayer: unknown[] }).dataLayer = [];\n }\n }\n }, [isAdminRoute]);\n\n return { isAdminRoute, shouldTrack: !isAdminRoute };\n}\n","import React, { createContext, useContext, type ReactNode } from 'react';\n\nconst PluginContext = createContext<{ getPlugin: <T>(name: string) => T | undefined } | null>(null);\n\nexport function PluginProvider({\n getPlugin,\n children,\n}: {\n getPlugin: <T>(name: string) => T | undefined;\n children: ReactNode;\n}) {\n return React.createElement(PluginContext.Provider, { value: { getPlugin } }, children);\n}\n\nexport function usePlugin<T = unknown>(name: string): T | undefined {\n const ctx = useContext(PluginContext);\n if (!ctx) return undefined;\n return ctx.getPlugin<T>(name);\n}\n","'use client';\n\nimport React, {\n createContext,\n useCallback,\n useContext,\n useEffect,\n useMemo,\n useRef,\n useState,\n} from 'react';\nimport type { CaptchaPublicConfig, CaptchaProviderId } from '../plugins/captcha';\n\ndeclare global {\n interface Window {\n turnstile?: {\n render: (el: HTMLElement | string, opts: Record<string, unknown>) => string;\n execute: (id: string) => void;\n reset: (id: string) => void;\n remove: (id: string) => void;\n };\n grecaptcha?: {\n ready: (fn: () => void) => void;\n execute: (siteKey: string, opts: { action: string }) => Promise<string>;\n };\n }\n}\n\ntype CaptchaCtx = {\n config: CaptchaPublicConfig | null;\n ready: boolean;\n selectedProvider: CaptchaProviderId | null;\n setSelectedProvider: (p: CaptchaProviderId) => void;\n getCaptchaPayload: () => Promise<Record<string, string>>;\n};\n\nconst CaptchaReactContext = createContext<CaptchaCtx | null>(null);\n\nfunction loadScript(src: string): Promise<void> {\n return new Promise((resolve, reject) => {\n const base = src.split('?')[0] ?? src;\n const existing = Array.from(document.querySelectorAll('script[src]')).some((el) =>\n (el as HTMLScriptElement).src.startsWith(base)\n );\n if (existing) {\n resolve();\n return;\n }\n const s = document.createElement('script');\n s.src = src;\n s.async = true;\n s.onload = () => resolve();\n s.onerror = () => reject(new Error('Failed to load script'));\n document.head.appendChild(s);\n });\n}\n\nexport function CaptchaProvider({ children }: { children: React.ReactNode }) {\n const [config, setConfig] = useState<CaptchaPublicConfig | null>(null);\n const [selectedProvider, setSelectedProvider] = useState<CaptchaProviderId | null>(null);\n const [ready, setReady] = useState(false);\n const turnstileWidgetIdRef = useRef<string | null>(null);\n const turnstileContainerRef = useRef<HTMLDivElement | null>(null);\n const turnstileTokenRef = useRef<{\n resolve: (t: string) => void;\n reject: (e: Error) => void;\n } | null>(null);\n\n useEffect(() => {\n let cancelled = false;\n fetch('/api/public/captcha-config')\n .then((r) => (r.ok ? r.json() : null))\n .then((c: CaptchaPublicConfig | null) => {\n if (cancelled || !c) return;\n setConfig(c);\n if (c.enabled && c.activeProvider) setSelectedProvider(c.activeProvider);\n })\n .catch(() => {})\n .finally(() => {\n if (!cancelled) setReady(true);\n });\n return () => {\n cancelled = true;\n };\n }, []);\n\n useEffect(() => {\n if (!config?.enabled || !selectedProvider) return;\n const siteKeyFor = config.availableProviders.find((p) => p.id === selectedProvider)?.siteKey;\n if (!siteKeyFor) return;\n\n let cancelled = false;\n (async () => {\n try {\n if (selectedProvider === 'recaptcha_v3') {\n await loadScript(\n `https://www.google.com/recaptcha/api.js?render=${encodeURIComponent(siteKeyFor)}`\n );\n } else {\n await loadScript('https://challenges.cloudflare.com/turnstile/v0/api.js');\n if (cancelled || !turnstileContainerRef.current || !window.turnstile) return;\n if (turnstileWidgetIdRef.current) {\n try {\n window.turnstile.remove(turnstileWidgetIdRef.current);\n } catch {\n /* ignore */\n }\n turnstileWidgetIdRef.current = null;\n }\n const wid = window.turnstile.render(turnstileContainerRef.current, {\n sitekey: siteKeyFor,\n execution: 'execute',\n callback: (token: string) => {\n const p = turnstileTokenRef.current;\n turnstileTokenRef.current = null;\n p?.resolve(token);\n },\n 'error-callback': () => {\n const p = turnstileTokenRef.current;\n turnstileTokenRef.current = null;\n p?.reject(new Error('Turnstile error'));\n },\n 'expired-callback': () => {\n const p = turnstileTokenRef.current;\n turnstileTokenRef.current = null;\n p?.reject(new Error('Turnstile expired'));\n },\n });\n turnstileWidgetIdRef.current = wid;\n }\n } catch {\n /* ignore */\n }\n })();\n return () => {\n cancelled = true;\n };\n }, [config, selectedProvider]);\n\n const getCaptchaPayload = useCallback(async (): Promise<Record<string, string>> => {\n if (!config?.enabled || !selectedProvider) return {};\n const siteKeyFor = config.availableProviders.find((p) => p.id === selectedProvider)?.siteKey;\n if (!siteKeyFor) return {};\n\n if (selectedProvider === 'recaptcha_v3') {\n if (!window.grecaptcha) return {};\n const token = await new Promise<string>((resolve, reject) => {\n window.grecaptcha!.ready(() => {\n window.grecaptcha!.execute(siteKeyFor, { action: 'submit' }).then(resolve).catch(reject);\n });\n });\n return { captchaToken: token, captchaProvider: 'recaptcha_v3' };\n }\n\n const wid = turnstileWidgetIdRef.current;\n if (!wid || !window.turnstile) return {};\n const token = await new Promise<string>((resolve, reject) => {\n turnstileTokenRef.current = { resolve, reject };\n try {\n window.turnstile!.reset(wid);\n window.turnstile!.execute(wid);\n } catch (e) {\n turnstileTokenRef.current = null;\n reject(e instanceof Error ? e : new Error('Turnstile execute failed'));\n }\n });\n return { captchaToken: token, captchaProvider: 'turnstile' };\n }, [config, selectedProvider]);\n\n const ctx = useMemo<CaptchaCtx>(\n () => ({\n config,\n ready,\n selectedProvider,\n setSelectedProvider,\n getCaptchaPayload,\n }),\n [config, ready, selectedProvider, getCaptchaPayload]\n );\n\n return (\n <CaptchaReactContext.Provider value={ctx}>\n {config?.enabled && config.multipleProviders ? (\n <div className=\"mb-2 text-sm text-muted-foreground max-w-md\">\n <label className=\"mr-2\" htmlFor=\"captcha-provider-select\">\n Verification\n </label>\n <select\n id=\"captcha-provider-select\"\n className=\"border rounded px-2 py-1 bg-background\"\n value={selectedProvider ?? ''}\n onChange={(e) => setSelectedProvider(e.target.value as CaptchaProviderId)}\n >\n {config.availableProviders.map((p) => (\n <option key={p.id} value={p.id}>\n {p.id === 'turnstile' ? 'Cloudflare Turnstile' : 'Google reCAPTCHA v3'}\n </option>\n ))}\n </select>\n </div>\n ) : null}\n <div ref={turnstileContainerRef} className=\"hidden\" aria-hidden />\n {children}\n </CaptchaReactContext.Provider>\n );\n}\n\n/** Returns a function that resolves to captcha fields for JSON bodies, or {} if disabled / not ready. */\nexport function useCaptchaPayload(): () => Promise<Record<string, string>> {\n const ctx = useContext(CaptchaReactContext);\n return useCallback(async () => {\n if (!ctx?.config?.enabled || !ctx.ready) return {};\n return ctx.getCaptchaPayload();\n }, [ctx]);\n}\n"],"mappings":"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","names":["import_react","import_react","React","import_react","token"]}
1
+ {"version":3,"sources":["../src/hooks/index.ts","../src/hooks/use-mobile.ts","../src/hooks/use-analytics.ts","../src/hooks/use-plugin.ts","../src/components/CaptchaProvider.tsx"],"sourcesContent":["export { useIsMobile } from './use-mobile';\r\nexport { useAnalytics } from './use-analytics';\r\nexport { usePlugin, PluginProvider } from './use-plugin';\r\nexport { CaptchaProvider, useCaptchaPayload } from '../components/CaptchaProvider';\r\n","import { useState, useEffect } from 'react';\r\n\r\nconst MOBILE_BREAKPOINT = 768;\r\n\r\nexport function useIsMobile(): boolean {\r\n const [isMobile, setIsMobile] = useState<boolean | undefined>(undefined);\r\n\r\n useEffect(() => {\r\n const mql = window.matchMedia(`(max-width: ${MOBILE_BREAKPOINT - 1}px)`);\r\n const onChange = () => setIsMobile(window.innerWidth < MOBILE_BREAKPOINT);\r\n mql.addEventListener('change', onChange);\r\n setIsMobile(window.innerWidth < MOBILE_BREAKPOINT);\r\n return () => mql.removeEventListener('change', onChange);\r\n }, []);\r\n\r\n return !!isMobile;\r\n}\r\n","import { useEffect } from 'react';\r\n\r\n/**\r\n * Call from app with pathname from usePathname() (next/navigation).\r\n * Disables tracking on admin routes when pathname starts with /admin.\r\n */\r\nexport function useAnalytics(pathname: string | null): { isAdminRoute: boolean; shouldTrack: boolean } {\r\n const isAdminRoute = pathname != null && pathname.startsWith('/admin');\r\n\r\n useEffect(() => {\r\n if (isAdminRoute && typeof window !== 'undefined') {\r\n if (typeof (window as unknown as { gtag?: () => void }).gtag === 'function') {\r\n (window as unknown as { gtag: () => void }).gtag = () => {};\r\n }\r\n if (Array.isArray((window as unknown as { dataLayer?: unknown[] }).dataLayer)) {\r\n (window as unknown as { dataLayer: unknown[] }).dataLayer = [];\r\n }\r\n }\r\n }, [isAdminRoute]);\r\n\r\n return { isAdminRoute, shouldTrack: !isAdminRoute };\r\n}\r\n","import React, { createContext, useContext, type ReactNode } from 'react';\r\n\r\nconst PluginContext = createContext<{ getPlugin: <T>(name: string) => T | undefined } | null>(null);\r\n\r\nexport function PluginProvider({\r\n getPlugin,\r\n children,\r\n}: {\r\n getPlugin: <T>(name: string) => T | undefined;\r\n children: ReactNode;\r\n}) {\r\n return React.createElement(PluginContext.Provider, { value: { getPlugin } }, children);\r\n}\r\n\r\nexport function usePlugin<T = unknown>(name: string): T | undefined {\r\n const ctx = useContext(PluginContext);\r\n if (!ctx) return undefined;\r\n return ctx.getPlugin<T>(name);\r\n}\r\n","'use client';\r\n\r\nimport React, {\r\n createContext,\r\n useCallback,\r\n useContext,\r\n useEffect,\r\n useMemo,\r\n useRef,\r\n useState,\r\n} from 'react';\r\nimport type { CaptchaPublicConfig, CaptchaProviderId } from '../plugins/captcha';\r\n\r\ndeclare global {\r\n interface Window {\r\n turnstile?: {\r\n render: (el: HTMLElement | string, opts: Record<string, unknown>) => string;\r\n execute: (id: string) => void;\r\n reset: (id: string) => void;\r\n remove: (id: string) => void;\r\n };\r\n grecaptcha?: {\r\n ready: (fn: () => void) => void;\r\n execute: (siteKey: string, opts: { action: string }) => Promise<string>;\r\n };\r\n }\r\n}\r\n\r\ntype CaptchaCtx = {\r\n config: CaptchaPublicConfig | null;\r\n ready: boolean;\r\n selectedProvider: CaptchaProviderId | null;\r\n setSelectedProvider: (p: CaptchaProviderId) => void;\r\n getCaptchaPayload: () => Promise<Record<string, string>>;\r\n};\r\n\r\nconst CaptchaReactContext = createContext<CaptchaCtx | null>(null);\r\n\r\nfunction loadScript(src: string): Promise<void> {\r\n return new Promise((resolve, reject) => {\r\n const base = src.split('?')[0] ?? src;\r\n const existing = Array.from(document.querySelectorAll('script[src]')).some((el) =>\r\n (el as HTMLScriptElement).src.startsWith(base)\r\n );\r\n if (existing) {\r\n resolve();\r\n return;\r\n }\r\n const s = document.createElement('script');\r\n s.src = src;\r\n s.async = true;\r\n s.onload = () => resolve();\r\n s.onerror = () => reject(new Error('Failed to load script'));\r\n document.head.appendChild(s);\r\n });\r\n}\r\n\r\nexport function CaptchaProvider({ children }: { children: React.ReactNode }) {\r\n const [config, setConfig] = useState<CaptchaPublicConfig | null>(null);\r\n const [selectedProvider, setSelectedProvider] = useState<CaptchaProviderId | null>(null);\r\n const [ready, setReady] = useState(false);\r\n const turnstileWidgetIdRef = useRef<string | null>(null);\r\n const turnstileContainerRef = useRef<HTMLDivElement | null>(null);\r\n const turnstileTokenRef = useRef<{\r\n resolve: (t: string) => void;\r\n reject: (e: Error) => void;\r\n } | null>(null);\r\n\r\n useEffect(() => {\r\n let cancelled = false;\r\n fetch('/api/public/captcha-config')\r\n .then((r) => (r.ok ? r.json() : null))\r\n .then((c: CaptchaPublicConfig | null) => {\r\n if (cancelled || !c) return;\r\n setConfig(c);\r\n if (c.enabled && c.activeProvider) setSelectedProvider(c.activeProvider);\r\n })\r\n .catch(() => {})\r\n .finally(() => {\r\n if (!cancelled) setReady(true);\r\n });\r\n return () => {\r\n cancelled = true;\r\n };\r\n }, []);\r\n\r\n useEffect(() => {\r\n if (!config?.enabled || !selectedProvider) return;\r\n const siteKeyFor = config.availableProviders.find((p) => p.id === selectedProvider)?.siteKey;\r\n if (!siteKeyFor) return;\r\n\r\n let cancelled = false;\r\n (async () => {\r\n try {\r\n if (selectedProvider === 'recaptcha_v3') {\r\n await loadScript(\r\n `https://www.google.com/recaptcha/api.js?render=${encodeURIComponent(siteKeyFor)}`\r\n );\r\n } else {\r\n await loadScript('https://challenges.cloudflare.com/turnstile/v0/api.js');\r\n if (cancelled || !turnstileContainerRef.current || !window.turnstile) return;\r\n if (turnstileWidgetIdRef.current) {\r\n try {\r\n window.turnstile.remove(turnstileWidgetIdRef.current);\r\n } catch {\r\n /* ignore */\r\n }\r\n turnstileWidgetIdRef.current = null;\r\n }\r\n const wid = window.turnstile.render(turnstileContainerRef.current, {\r\n sitekey: siteKeyFor,\r\n execution: 'execute',\r\n callback: (token: string) => {\r\n const p = turnstileTokenRef.current;\r\n turnstileTokenRef.current = null;\r\n p?.resolve(token);\r\n },\r\n 'error-callback': () => {\r\n const p = turnstileTokenRef.current;\r\n turnstileTokenRef.current = null;\r\n p?.reject(new Error('Turnstile error'));\r\n },\r\n 'expired-callback': () => {\r\n const p = turnstileTokenRef.current;\r\n turnstileTokenRef.current = null;\r\n p?.reject(new Error('Turnstile expired'));\r\n },\r\n });\r\n turnstileWidgetIdRef.current = wid;\r\n }\r\n } catch {\r\n /* ignore */\r\n }\r\n })();\r\n return () => {\r\n cancelled = true;\r\n };\r\n }, [config, selectedProvider]);\r\n\r\n const getCaptchaPayload = useCallback(async (): Promise<Record<string, string>> => {\r\n if (!config?.enabled || !selectedProvider) return {};\r\n const siteKeyFor = config.availableProviders.find((p) => p.id === selectedProvider)?.siteKey;\r\n if (!siteKeyFor) return {};\r\n\r\n if (selectedProvider === 'recaptcha_v3') {\r\n if (!window.grecaptcha) return {};\r\n const token = await new Promise<string>((resolve, reject) => {\r\n window.grecaptcha!.ready(() => {\r\n window.grecaptcha!.execute(siteKeyFor, { action: 'submit' }).then(resolve).catch(reject);\r\n });\r\n });\r\n return { captchaToken: token, captchaProvider: 'recaptcha_v3' };\r\n }\r\n\r\n const wid = turnstileWidgetIdRef.current;\r\n if (!wid || !window.turnstile) return {};\r\n const token = await new Promise<string>((resolve, reject) => {\r\n turnstileTokenRef.current = { resolve, reject };\r\n try {\r\n window.turnstile!.reset(wid);\r\n window.turnstile!.execute(wid);\r\n } catch (e) {\r\n turnstileTokenRef.current = null;\r\n reject(e instanceof Error ? e : new Error('Turnstile execute failed'));\r\n }\r\n });\r\n return { captchaToken: token, captchaProvider: 'turnstile' };\r\n }, [config, selectedProvider]);\r\n\r\n const ctx = useMemo<CaptchaCtx>(\r\n () => ({\r\n config,\r\n ready,\r\n selectedProvider,\r\n setSelectedProvider,\r\n getCaptchaPayload,\r\n }),\r\n [config, ready, selectedProvider, getCaptchaPayload]\r\n );\r\n\r\n return (\r\n <CaptchaReactContext.Provider value={ctx}>\r\n {config?.enabled && config.multipleProviders ? (\r\n <div className=\"mb-2 text-sm text-muted-foreground max-w-md\">\r\n <label className=\"mr-2\" htmlFor=\"captcha-provider-select\">\r\n Verification\r\n </label>\r\n <select\r\n id=\"captcha-provider-select\"\r\n className=\"border rounded px-2 py-1 bg-background\"\r\n value={selectedProvider ?? ''}\r\n onChange={(e) => setSelectedProvider(e.target.value as CaptchaProviderId)}\r\n >\r\n {config.availableProviders.map((p) => (\r\n <option key={p.id} value={p.id}>\r\n {p.id === 'turnstile' ? 'Cloudflare Turnstile' : 'Google reCAPTCHA v3'}\r\n </option>\r\n ))}\r\n </select>\r\n </div>\r\n ) : null}\r\n <div ref={turnstileContainerRef} className=\"hidden\" aria-hidden />\r\n {children}\r\n </CaptchaReactContext.Provider>\r\n );\r\n}\r\n\r\n/** Returns a function that resolves to captcha fields for JSON bodies, or {} if disabled / not ready. */\r\nexport function useCaptchaPayload(): () => Promise<Record<string, string>> {\r\n const ctx = useContext(CaptchaReactContext);\r\n return useCallback(async () => {\r\n if (!ctx?.config?.enabled || !ctx.ready) return {};\r\n return ctx.getCaptchaPayload();\r\n }, [ctx]);\r\n}\r\n"],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;;;ACAA,mBAAoC;AAEpC,IAAM,oBAAoB;AAEnB,SAAS,cAAuB;AACrC,QAAM,CAAC,UAAU,WAAW,QAAI,uBAA8B,MAAS;AAEvE,8BAAU,MAAM;AACd,UAAM,MAAM,OAAO,WAAW,eAAe,oBAAoB,CAAC,KAAK;AACvE,UAAM,WAAW,MAAM,YAAY,OAAO,aAAa,iBAAiB;AACxE,QAAI,iBAAiB,UAAU,QAAQ;AACvC,gBAAY,OAAO,aAAa,iBAAiB;AACjD,WAAO,MAAM,IAAI,oBAAoB,UAAU,QAAQ;AAAA,EACzD,GAAG,CAAC,CAAC;AAEL,SAAO,CAAC,CAAC;AACX;;;AChBA,IAAAA,gBAA0B;AAMnB,SAAS,aAAa,UAA0E;AACrG,QAAM,eAAe,YAAY,QAAQ,SAAS,WAAW,QAAQ;AAErE,+BAAU,MAAM;AACd,QAAI,gBAAgB,OAAO,WAAW,aAAa;AACjD,UAAI,OAAQ,OAA4C,SAAS,YAAY;AAC3E,QAAC,OAA2C,OAAO,MAAM;AAAA,QAAC;AAAA,MAC5D;AACA,UAAI,MAAM,QAAS,OAAgD,SAAS,GAAG;AAC7E,QAAC,OAA+C,YAAY,CAAC;AAAA,MAC/D;AAAA,IACF;AAAA,EACF,GAAG,CAAC,YAAY,CAAC;AAEjB,SAAO,EAAE,cAAc,aAAa,CAAC,aAAa;AACpD;;;ACrBA,IAAAC,gBAAiE;AAEjE,IAAM,oBAAgB,6BAAwE,IAAI;AAE3F,SAAS,eAAe;AAAA,EAC7B;AAAA,EACA;AACF,GAGG;AACD,SAAO,cAAAC,QAAM,cAAc,cAAc,UAAU,EAAE,OAAO,EAAE,UAAU,EAAE,GAAG,QAAQ;AACvF;AAEO,SAAS,UAAuB,MAA6B;AAClE,QAAM,UAAM,0BAAW,aAAa;AACpC,MAAI,CAAC,IAAK,QAAO;AACjB,SAAO,IAAI,UAAa,IAAI;AAC9B;;;AChBA,IAAAC,gBAQO;AA6KC;AAnJR,IAAM,0BAAsB,6BAAiC,IAAI;AAEjE,SAAS,WAAW,KAA4B;AAC9C,SAAO,IAAI,QAAQ,CAAC,SAAS,WAAW;AACtC,UAAM,OAAO,IAAI,MAAM,GAAG,EAAE,CAAC,KAAK;AAClC,UAAM,WAAW,MAAM,KAAK,SAAS,iBAAiB,aAAa,CAAC,EAAE;AAAA,MAAK,CAAC,OACzE,GAAyB,IAAI,WAAW,IAAI;AAAA,IAC/C;AACA,QAAI,UAAU;AACZ,cAAQ;AACR;AAAA,IACF;AACA,UAAM,IAAI,SAAS,cAAc,QAAQ;AACzC,MAAE,MAAM;AACR,MAAE,QAAQ;AACV,MAAE,SAAS,MAAM,QAAQ;AACzB,MAAE,UAAU,MAAM,OAAO,IAAI,MAAM,uBAAuB,CAAC;AAC3D,aAAS,KAAK,YAAY,CAAC;AAAA,EAC7B,CAAC;AACH;AAEO,SAAS,gBAAgB,EAAE,SAAS,GAAkC;AAC3E,QAAM,CAAC,QAAQ,SAAS,QAAI,wBAAqC,IAAI;AACrE,QAAM,CAAC,kBAAkB,mBAAmB,QAAI,wBAAmC,IAAI;AACvF,QAAM,CAAC,OAAO,QAAQ,QAAI,wBAAS,KAAK;AACxC,QAAM,2BAAuB,sBAAsB,IAAI;AACvD,QAAM,4BAAwB,sBAA8B,IAAI;AAChE,QAAM,wBAAoB,sBAGhB,IAAI;AAEd,+BAAU,MAAM;AACd,QAAI,YAAY;AAChB,UAAM,4BAA4B,EAC/B,KAAK,CAAC,MAAO,EAAE,KAAK,EAAE,KAAK,IAAI,IAAK,EACpC,KAAK,CAAC,MAAkC;AACvC,UAAI,aAAa,CAAC,EAAG;AACrB,gBAAU,CAAC;AACX,UAAI,EAAE,WAAW,EAAE,eAAgB,qBAAoB,EAAE,cAAc;AAAA,IACzE,CAAC,EACA,MAAM,MAAM;AAAA,IAAC,CAAC,EACd,QAAQ,MAAM;AACb,UAAI,CAAC,UAAW,UAAS,IAAI;AAAA,IAC/B,CAAC;AACH,WAAO,MAAM;AACX,kBAAY;AAAA,IACd;AAAA,EACF,GAAG,CAAC,CAAC;AAEL,+BAAU,MAAM;AACd,QAAI,CAAC,QAAQ,WAAW,CAAC,iBAAkB;AAC3C,UAAM,aAAa,OAAO,mBAAmB,KAAK,CAAC,MAAM,EAAE,OAAO,gBAAgB,GAAG;AACrF,QAAI,CAAC,WAAY;AAEjB,QAAI,YAAY;AAChB,KAAC,YAAY;AACX,UAAI;AACF,YAAI,qBAAqB,gBAAgB;AACvC,gBAAM;AAAA,YACJ,kDAAkD,mBAAmB,UAAU,CAAC;AAAA,UAClF;AAAA,QACF,OAAO;AACL,gBAAM,WAAW,uDAAuD;AACxE,cAAI,aAAa,CAAC,sBAAsB,WAAW,CAAC,OAAO,UAAW;AACtE,cAAI,qBAAqB,SAAS;AAChC,gBAAI;AACF,qBAAO,UAAU,OAAO,qBAAqB,OAAO;AAAA,YACtD,QAAQ;AAAA,YAER;AACA,iCAAqB,UAAU;AAAA,UACjC;AACA,gBAAM,MAAM,OAAO,UAAU,OAAO,sBAAsB,SAAS;AAAA,YACjE,SAAS;AAAA,YACT,WAAW;AAAA,YACX,UAAU,CAAC,UAAkB;AAC3B,oBAAM,IAAI,kBAAkB;AAC5B,gCAAkB,UAAU;AAC5B,iBAAG,QAAQ,KAAK;AAAA,YAClB;AAAA,YACA,kBAAkB,MAAM;AACtB,oBAAM,IAAI,kBAAkB;AAC5B,gCAAkB,UAAU;AAC5B,iBAAG,OAAO,IAAI,MAAM,iBAAiB,CAAC;AAAA,YACxC;AAAA,YACA,oBAAoB,MAAM;AACxB,oBAAM,IAAI,kBAAkB;AAC5B,gCAAkB,UAAU;AAC5B,iBAAG,OAAO,IAAI,MAAM,mBAAmB,CAAC;AAAA,YAC1C;AAAA,UACF,CAAC;AACD,+BAAqB,UAAU;AAAA,QACjC;AAAA,MACF,QAAQ;AAAA,MAER;AAAA,IACF,GAAG;AACH,WAAO,MAAM;AACX,kBAAY;AAAA,IACd;AAAA,EACF,GAAG,CAAC,QAAQ,gBAAgB,CAAC;AAE7B,QAAM,wBAAoB,2BAAY,YAA6C;AACjF,QAAI,CAAC,QAAQ,WAAW,CAAC,iBAAkB,QAAO,CAAC;AACnD,UAAM,aAAa,OAAO,mBAAmB,KAAK,CAAC,MAAM,EAAE,OAAO,gBAAgB,GAAG;AACrF,QAAI,CAAC,WAAY,QAAO,CAAC;AAEzB,QAAI,qBAAqB,gBAAgB;AACvC,UAAI,CAAC,OAAO,WAAY,QAAO,CAAC;AAChC,YAAMC,SAAQ,MAAM,IAAI,QAAgB,CAAC,SAAS,WAAW;AAC3D,eAAO,WAAY,MAAM,MAAM;AAC7B,iBAAO,WAAY,QAAQ,YAAY,EAAE,QAAQ,SAAS,CAAC,EAAE,KAAK,OAAO,EAAE,MAAM,MAAM;AAAA,QACzF,CAAC;AAAA,MACH,CAAC;AACD,aAAO,EAAE,cAAcA,QAAO,iBAAiB,eAAe;AAAA,IAChE;AAEA,UAAM,MAAM,qBAAqB;AACjC,QAAI,CAAC,OAAO,CAAC,OAAO,UAAW,QAAO,CAAC;AACvC,UAAM,QAAQ,MAAM,IAAI,QAAgB,CAAC,SAAS,WAAW;AAC3D,wBAAkB,UAAU,EAAE,SAAS,OAAO;AAC9C,UAAI;AACF,eAAO,UAAW,MAAM,GAAG;AAC3B,eAAO,UAAW,QAAQ,GAAG;AAAA,MAC/B,SAAS,GAAG;AACV,0BAAkB,UAAU;AAC5B,eAAO,aAAa,QAAQ,IAAI,IAAI,MAAM,0BAA0B,CAAC;AAAA,MACvE;AAAA,IACF,CAAC;AACD,WAAO,EAAE,cAAc,OAAO,iBAAiB,YAAY;AAAA,EAC7D,GAAG,CAAC,QAAQ,gBAAgB,CAAC;AAE7B,QAAM,UAAM;AAAA,IACV,OAAO;AAAA,MACL;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,IACF;AAAA,IACA,CAAC,QAAQ,OAAO,kBAAkB,iBAAiB;AAAA,EACrD;AAEA,SACE,6CAAC,oBAAoB,UAApB,EAA6B,OAAO,KAClC;AAAA,YAAQ,WAAW,OAAO,oBACzB,6CAAC,SAAI,WAAU,+CACb;AAAA,kDAAC,WAAM,WAAU,QAAO,SAAQ,2BAA0B,0BAE1D;AAAA,MACA;AAAA,QAAC;AAAA;AAAA,UACC,IAAG;AAAA,UACH,WAAU;AAAA,UACV,OAAO,oBAAoB;AAAA,UAC3B,UAAU,CAAC,MAAM,oBAAoB,EAAE,OAAO,KAA0B;AAAA,UAEvE,iBAAO,mBAAmB,IAAI,CAAC,MAC9B,4CAAC,YAAkB,OAAO,EAAE,IACzB,YAAE,OAAO,cAAc,yBAAyB,yBADtC,EAAE,EAEf,CACD;AAAA;AAAA,MACH;AAAA,OACF,IACE;AAAA,IACJ,4CAAC,SAAI,KAAK,uBAAuB,WAAU,UAAS,eAAW,MAAC;AAAA,IAC/D;AAAA,KACH;AAEJ;AAGO,SAAS,oBAA2D;AACzE,QAAM,UAAM,0BAAW,mBAAmB;AAC1C,aAAO,2BAAY,YAAY;AAC7B,QAAI,CAAC,KAAK,QAAQ,WAAW,CAAC,IAAI,MAAO,QAAO,CAAC;AACjD,WAAO,IAAI,kBAAkB;AAAA,EAC/B,GAAG,CAAC,GAAG,CAAC;AACV;","names":["import_react","import_react","React","import_react","token"]}
package/dist/hooks.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/hooks/use-mobile.ts","../src/hooks/use-analytics.ts","../src/hooks/use-plugin.ts","../src/components/CaptchaProvider.tsx"],"sourcesContent":["import { useState, useEffect } from 'react';\n\nconst MOBILE_BREAKPOINT = 768;\n\nexport function useIsMobile(): boolean {\n const [isMobile, setIsMobile] = useState<boolean | undefined>(undefined);\n\n useEffect(() => {\n const mql = window.matchMedia(`(max-width: ${MOBILE_BREAKPOINT - 1}px)`);\n const onChange = () => setIsMobile(window.innerWidth < MOBILE_BREAKPOINT);\n mql.addEventListener('change', onChange);\n setIsMobile(window.innerWidth < MOBILE_BREAKPOINT);\n return () => mql.removeEventListener('change', onChange);\n }, []);\n\n return !!isMobile;\n}\n","import { useEffect } from 'react';\n\n/**\n * Call from app with pathname from usePathname() (next/navigation).\n * Disables tracking on admin routes when pathname starts with /admin.\n */\nexport function useAnalytics(pathname: string | null): { isAdminRoute: boolean; shouldTrack: boolean } {\n const isAdminRoute = pathname != null && pathname.startsWith('/admin');\n\n useEffect(() => {\n if (isAdminRoute && typeof window !== 'undefined') {\n if (typeof (window as unknown as { gtag?: () => void }).gtag === 'function') {\n (window as unknown as { gtag: () => void }).gtag = () => {};\n }\n if (Array.isArray((window as unknown as { dataLayer?: unknown[] }).dataLayer)) {\n (window as unknown as { dataLayer: unknown[] }).dataLayer = [];\n }\n }\n }, [isAdminRoute]);\n\n return { isAdminRoute, shouldTrack: !isAdminRoute };\n}\n","import React, { createContext, useContext, type ReactNode } from 'react';\n\nconst PluginContext = createContext<{ getPlugin: <T>(name: string) => T | undefined } | null>(null);\n\nexport function PluginProvider({\n getPlugin,\n children,\n}: {\n getPlugin: <T>(name: string) => T | undefined;\n children: ReactNode;\n}) {\n return React.createElement(PluginContext.Provider, { value: { getPlugin } }, children);\n}\n\nexport function usePlugin<T = unknown>(name: string): T | undefined {\n const ctx = useContext(PluginContext);\n if (!ctx) return undefined;\n return ctx.getPlugin<T>(name);\n}\n","'use client';\n\nimport React, {\n createContext,\n useCallback,\n useContext,\n useEffect,\n useMemo,\n useRef,\n useState,\n} from 'react';\nimport type { CaptchaPublicConfig, CaptchaProviderId } from '../plugins/captcha';\n\ndeclare global {\n interface Window {\n turnstile?: {\n render: (el: HTMLElement | string, opts: Record<string, unknown>) => string;\n execute: (id: string) => void;\n reset: (id: string) => void;\n remove: (id: string) => void;\n };\n grecaptcha?: {\n ready: (fn: () => void) => void;\n execute: (siteKey: string, opts: { action: string }) => Promise<string>;\n };\n }\n}\n\ntype CaptchaCtx = {\n config: CaptchaPublicConfig | null;\n ready: boolean;\n selectedProvider: CaptchaProviderId | null;\n setSelectedProvider: (p: CaptchaProviderId) => void;\n getCaptchaPayload: () => Promise<Record<string, string>>;\n};\n\nconst CaptchaReactContext = createContext<CaptchaCtx | null>(null);\n\nfunction loadScript(src: string): Promise<void> {\n return new Promise((resolve, reject) => {\n const base = src.split('?')[0] ?? src;\n const existing = Array.from(document.querySelectorAll('script[src]')).some((el) =>\n (el as HTMLScriptElement).src.startsWith(base)\n );\n if (existing) {\n resolve();\n return;\n }\n const s = document.createElement('script');\n s.src = src;\n s.async = true;\n s.onload = () => resolve();\n s.onerror = () => reject(new Error('Failed to load script'));\n document.head.appendChild(s);\n });\n}\n\nexport function CaptchaProvider({ children }: { children: React.ReactNode }) {\n const [config, setConfig] = useState<CaptchaPublicConfig | null>(null);\n const [selectedProvider, setSelectedProvider] = useState<CaptchaProviderId | null>(null);\n const [ready, setReady] = useState(false);\n const turnstileWidgetIdRef = useRef<string | null>(null);\n const turnstileContainerRef = useRef<HTMLDivElement | null>(null);\n const turnstileTokenRef = useRef<{\n resolve: (t: string) => void;\n reject: (e: Error) => void;\n } | null>(null);\n\n useEffect(() => {\n let cancelled = false;\n fetch('/api/public/captcha-config')\n .then((r) => (r.ok ? r.json() : null))\n .then((c: CaptchaPublicConfig | null) => {\n if (cancelled || !c) return;\n setConfig(c);\n if (c.enabled && c.activeProvider) setSelectedProvider(c.activeProvider);\n })\n .catch(() => {})\n .finally(() => {\n if (!cancelled) setReady(true);\n });\n return () => {\n cancelled = true;\n };\n }, []);\n\n useEffect(() => {\n if (!config?.enabled || !selectedProvider) return;\n const siteKeyFor = config.availableProviders.find((p) => p.id === selectedProvider)?.siteKey;\n if (!siteKeyFor) return;\n\n let cancelled = false;\n (async () => {\n try {\n if (selectedProvider === 'recaptcha_v3') {\n await loadScript(\n `https://www.google.com/recaptcha/api.js?render=${encodeURIComponent(siteKeyFor)}`\n );\n } else {\n await loadScript('https://challenges.cloudflare.com/turnstile/v0/api.js');\n if (cancelled || !turnstileContainerRef.current || !window.turnstile) return;\n if (turnstileWidgetIdRef.current) {\n try {\n window.turnstile.remove(turnstileWidgetIdRef.current);\n } catch {\n /* ignore */\n }\n turnstileWidgetIdRef.current = null;\n }\n const wid = window.turnstile.render(turnstileContainerRef.current, {\n sitekey: siteKeyFor,\n execution: 'execute',\n callback: (token: string) => {\n const p = turnstileTokenRef.current;\n turnstileTokenRef.current = null;\n p?.resolve(token);\n },\n 'error-callback': () => {\n const p = turnstileTokenRef.current;\n turnstileTokenRef.current = null;\n p?.reject(new Error('Turnstile error'));\n },\n 'expired-callback': () => {\n const p = turnstileTokenRef.current;\n turnstileTokenRef.current = null;\n p?.reject(new Error('Turnstile expired'));\n },\n });\n turnstileWidgetIdRef.current = wid;\n }\n } catch {\n /* ignore */\n }\n })();\n return () => {\n cancelled = true;\n };\n }, [config, selectedProvider]);\n\n const getCaptchaPayload = useCallback(async (): Promise<Record<string, string>> => {\n if (!config?.enabled || !selectedProvider) return {};\n const siteKeyFor = config.availableProviders.find((p) => p.id === selectedProvider)?.siteKey;\n if (!siteKeyFor) return {};\n\n if (selectedProvider === 'recaptcha_v3') {\n if (!window.grecaptcha) return {};\n const token = await new Promise<string>((resolve, reject) => {\n window.grecaptcha!.ready(() => {\n window.grecaptcha!.execute(siteKeyFor, { action: 'submit' }).then(resolve).catch(reject);\n });\n });\n return { captchaToken: token, captchaProvider: 'recaptcha_v3' };\n }\n\n const wid = turnstileWidgetIdRef.current;\n if (!wid || !window.turnstile) return {};\n const token = await new Promise<string>((resolve, reject) => {\n turnstileTokenRef.current = { resolve, reject };\n try {\n window.turnstile!.reset(wid);\n window.turnstile!.execute(wid);\n } catch (e) {\n turnstileTokenRef.current = null;\n reject(e instanceof Error ? e : new Error('Turnstile execute failed'));\n }\n });\n return { captchaToken: token, captchaProvider: 'turnstile' };\n }, [config, selectedProvider]);\n\n const ctx = useMemo<CaptchaCtx>(\n () => ({\n config,\n ready,\n selectedProvider,\n setSelectedProvider,\n getCaptchaPayload,\n }),\n [config, ready, selectedProvider, getCaptchaPayload]\n );\n\n return (\n <CaptchaReactContext.Provider value={ctx}>\n {config?.enabled && config.multipleProviders ? (\n <div className=\"mb-2 text-sm text-muted-foreground max-w-md\">\n <label className=\"mr-2\" htmlFor=\"captcha-provider-select\">\n Verification\n </label>\n <select\n id=\"captcha-provider-select\"\n className=\"border rounded px-2 py-1 bg-background\"\n value={selectedProvider ?? ''}\n onChange={(e) => setSelectedProvider(e.target.value as CaptchaProviderId)}\n >\n {config.availableProviders.map((p) => (\n <option key={p.id} value={p.id}>\n {p.id === 'turnstile' ? 'Cloudflare Turnstile' : 'Google reCAPTCHA v3'}\n </option>\n ))}\n </select>\n </div>\n ) : null}\n <div ref={turnstileContainerRef} className=\"hidden\" aria-hidden />\n {children}\n </CaptchaReactContext.Provider>\n );\n}\n\n/** Returns a function that resolves to captcha fields for JSON bodies, or {} if disabled / not ready. */\nexport function useCaptchaPayload(): () => Promise<Record<string, string>> {\n const ctx = useContext(CaptchaReactContext);\n return useCallback(async () => {\n if (!ctx?.config?.enabled || !ctx.ready) return {};\n return ctx.getCaptchaPayload();\n }, [ctx]);\n}\n"],"mappings":";AAAA,SAAS,UAAU,iBAAiB;AAEpC,IAAM,oBAAoB;AAEnB,SAAS,cAAuB;AACrC,QAAM,CAAC,UAAU,WAAW,IAAI,SAA8B,MAAS;AAEvE,YAAU,MAAM;AACd,UAAM,MAAM,OAAO,WAAW,eAAe,oBAAoB,CAAC,KAAK;AACvE,UAAM,WAAW,MAAM,YAAY,OAAO,aAAa,iBAAiB;AACxE,QAAI,iBAAiB,UAAU,QAAQ;AACvC,gBAAY,OAAO,aAAa,iBAAiB;AACjD,WAAO,MAAM,IAAI,oBAAoB,UAAU,QAAQ;AAAA,EACzD,GAAG,CAAC,CAAC;AAEL,SAAO,CAAC,CAAC;AACX;;;AChBA,SAAS,aAAAA,kBAAiB;AAMnB,SAAS,aAAa,UAA0E;AACrG,QAAM,eAAe,YAAY,QAAQ,SAAS,WAAW,QAAQ;AAErE,EAAAA,WAAU,MAAM;AACd,QAAI,gBAAgB,OAAO,WAAW,aAAa;AACjD,UAAI,OAAQ,OAA4C,SAAS,YAAY;AAC3E,QAAC,OAA2C,OAAO,MAAM;AAAA,QAAC;AAAA,MAC5D;AACA,UAAI,MAAM,QAAS,OAAgD,SAAS,GAAG;AAC7E,QAAC,OAA+C,YAAY,CAAC;AAAA,MAC/D;AAAA,IACF;AAAA,EACF,GAAG,CAAC,YAAY,CAAC;AAEjB,SAAO,EAAE,cAAc,aAAa,CAAC,aAAa;AACpD;;;ACrBA,OAAO,SAAS,eAAe,kBAAkC;AAEjE,IAAM,gBAAgB,cAAwE,IAAI;AAE3F,SAAS,eAAe;AAAA,EAC7B;AAAA,EACA;AACF,GAGG;AACD,SAAO,MAAM,cAAc,cAAc,UAAU,EAAE,OAAO,EAAE,UAAU,EAAE,GAAG,QAAQ;AACvF;AAEO,SAAS,UAAuB,MAA6B;AAClE,QAAM,MAAM,WAAW,aAAa;AACpC,MAAI,CAAC,IAAK,QAAO;AACjB,SAAO,IAAI,UAAa,IAAI;AAC9B;;;AChBA;AAAA,EACE,iBAAAC;AAAA,EACA;AAAA,EACA,cAAAC;AAAA,EACA,aAAAC;AAAA,EACA;AAAA,EACA;AAAA,EACA,YAAAC;AAAA,OACK;AA6KC,SACE,KADF;AAnJR,IAAM,sBAAsBH,eAAiC,IAAI;AAEjE,SAAS,WAAW,KAA4B;AAC9C,SAAO,IAAI,QAAQ,CAAC,SAAS,WAAW;AACtC,UAAM,OAAO,IAAI,MAAM,GAAG,EAAE,CAAC,KAAK;AAClC,UAAM,WAAW,MAAM,KAAK,SAAS,iBAAiB,aAAa,CAAC,EAAE;AAAA,MAAK,CAAC,OACzE,GAAyB,IAAI,WAAW,IAAI;AAAA,IAC/C;AACA,QAAI,UAAU;AACZ,cAAQ;AACR;AAAA,IACF;AACA,UAAM,IAAI,SAAS,cAAc,QAAQ;AACzC,MAAE,MAAM;AACR,MAAE,QAAQ;AACV,MAAE,SAAS,MAAM,QAAQ;AACzB,MAAE,UAAU,MAAM,OAAO,IAAI,MAAM,uBAAuB,CAAC;AAC3D,aAAS,KAAK,YAAY,CAAC;AAAA,EAC7B,CAAC;AACH;AAEO,SAAS,gBAAgB,EAAE,SAAS,GAAkC;AAC3E,QAAM,CAAC,QAAQ,SAAS,IAAIG,UAAqC,IAAI;AACrE,QAAM,CAAC,kBAAkB,mBAAmB,IAAIA,UAAmC,IAAI;AACvF,QAAM,CAAC,OAAO,QAAQ,IAAIA,UAAS,KAAK;AACxC,QAAM,uBAAuB,OAAsB,IAAI;AACvD,QAAM,wBAAwB,OAA8B,IAAI;AAChE,QAAM,oBAAoB,OAGhB,IAAI;AAEd,EAAAD,WAAU,MAAM;AACd,QAAI,YAAY;AAChB,UAAM,4BAA4B,EAC/B,KAAK,CAAC,MAAO,EAAE,KAAK,EAAE,KAAK,IAAI,IAAK,EACpC,KAAK,CAAC,MAAkC;AACvC,UAAI,aAAa,CAAC,EAAG;AACrB,gBAAU,CAAC;AACX,UAAI,EAAE,WAAW,EAAE,eAAgB,qBAAoB,EAAE,cAAc;AAAA,IACzE,CAAC,EACA,MAAM,MAAM;AAAA,IAAC,CAAC,EACd,QAAQ,MAAM;AACb,UAAI,CAAC,UAAW,UAAS,IAAI;AAAA,IAC/B,CAAC;AACH,WAAO,MAAM;AACX,kBAAY;AAAA,IACd;AAAA,EACF,GAAG,CAAC,CAAC;AAEL,EAAAA,WAAU,MAAM;AACd,QAAI,CAAC,QAAQ,WAAW,CAAC,iBAAkB;AAC3C,UAAM,aAAa,OAAO,mBAAmB,KAAK,CAAC,MAAM,EAAE,OAAO,gBAAgB,GAAG;AACrF,QAAI,CAAC,WAAY;AAEjB,QAAI,YAAY;AAChB,KAAC,YAAY;AACX,UAAI;AACF,YAAI,qBAAqB,gBAAgB;AACvC,gBAAM;AAAA,YACJ,kDAAkD,mBAAmB,UAAU,CAAC;AAAA,UAClF;AAAA,QACF,OAAO;AACL,gBAAM,WAAW,uDAAuD;AACxE,cAAI,aAAa,CAAC,sBAAsB,WAAW,CAAC,OAAO,UAAW;AACtE,cAAI,qBAAqB,SAAS;AAChC,gBAAI;AACF,qBAAO,UAAU,OAAO,qBAAqB,OAAO;AAAA,YACtD,QAAQ;AAAA,YAER;AACA,iCAAqB,UAAU;AAAA,UACjC;AACA,gBAAM,MAAM,OAAO,UAAU,OAAO,sBAAsB,SAAS;AAAA,YACjE,SAAS;AAAA,YACT,WAAW;AAAA,YACX,UAAU,CAAC,UAAkB;AAC3B,oBAAM,IAAI,kBAAkB;AAC5B,gCAAkB,UAAU;AAC5B,iBAAG,QAAQ,KAAK;AAAA,YAClB;AAAA,YACA,kBAAkB,MAAM;AACtB,oBAAM,IAAI,kBAAkB;AAC5B,gCAAkB,UAAU;AAC5B,iBAAG,OAAO,IAAI,MAAM,iBAAiB,CAAC;AAAA,YACxC;AAAA,YACA,oBAAoB,MAAM;AACxB,oBAAM,IAAI,kBAAkB;AAC5B,gCAAkB,UAAU;AAC5B,iBAAG,OAAO,IAAI,MAAM,mBAAmB,CAAC;AAAA,YAC1C;AAAA,UACF,CAAC;AACD,+BAAqB,UAAU;AAAA,QACjC;AAAA,MACF,QAAQ;AAAA,MAER;AAAA,IACF,GAAG;AACH,WAAO,MAAM;AACX,kBAAY;AAAA,IACd;AAAA,EACF,GAAG,CAAC,QAAQ,gBAAgB,CAAC;AAE7B,QAAM,oBAAoB,YAAY,YAA6C;AACjF,QAAI,CAAC,QAAQ,WAAW,CAAC,iBAAkB,QAAO,CAAC;AACnD,UAAM,aAAa,OAAO,mBAAmB,KAAK,CAAC,MAAM,EAAE,OAAO,gBAAgB,GAAG;AACrF,QAAI,CAAC,WAAY,QAAO,CAAC;AAEzB,QAAI,qBAAqB,gBAAgB;AACvC,UAAI,CAAC,OAAO,WAAY,QAAO,CAAC;AAChC,YAAME,SAAQ,MAAM,IAAI,QAAgB,CAAC,SAAS,WAAW;AAC3D,eAAO,WAAY,MAAM,MAAM;AAC7B,iBAAO,WAAY,QAAQ,YAAY,EAAE,QAAQ,SAAS,CAAC,EAAE,KAAK,OAAO,EAAE,MAAM,MAAM;AAAA,QACzF,CAAC;AAAA,MACH,CAAC;AACD,aAAO,EAAE,cAAcA,QAAO,iBAAiB,eAAe;AAAA,IAChE;AAEA,UAAM,MAAM,qBAAqB;AACjC,QAAI,CAAC,OAAO,CAAC,OAAO,UAAW,QAAO,CAAC;AACvC,UAAM,QAAQ,MAAM,IAAI,QAAgB,CAAC,SAAS,WAAW;AAC3D,wBAAkB,UAAU,EAAE,SAAS,OAAO;AAC9C,UAAI;AACF,eAAO,UAAW,MAAM,GAAG;AAC3B,eAAO,UAAW,QAAQ,GAAG;AAAA,MAC/B,SAAS,GAAG;AACV,0BAAkB,UAAU;AAC5B,eAAO,aAAa,QAAQ,IAAI,IAAI,MAAM,0BAA0B,CAAC;AAAA,MACvE;AAAA,IACF,CAAC;AACD,WAAO,EAAE,cAAc,OAAO,iBAAiB,YAAY;AAAA,EAC7D,GAAG,CAAC,QAAQ,gBAAgB,CAAC;AAE7B,QAAM,MAAM;AAAA,IACV,OAAO;AAAA,MACL;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,IACF;AAAA,IACA,CAAC,QAAQ,OAAO,kBAAkB,iBAAiB;AAAA,EACrD;AAEA,SACE,qBAAC,oBAAoB,UAApB,EAA6B,OAAO,KAClC;AAAA,YAAQ,WAAW,OAAO,oBACzB,qBAAC,SAAI,WAAU,+CACb;AAAA,0BAAC,WAAM,WAAU,QAAO,SAAQ,2BAA0B,0BAE1D;AAAA,MACA;AAAA,QAAC;AAAA;AAAA,UACC,IAAG;AAAA,UACH,WAAU;AAAA,UACV,OAAO,oBAAoB;AAAA,UAC3B,UAAU,CAAC,MAAM,oBAAoB,EAAE,OAAO,KAA0B;AAAA,UAEvE,iBAAO,mBAAmB,IAAI,CAAC,MAC9B,oBAAC,YAAkB,OAAO,EAAE,IACzB,YAAE,OAAO,cAAc,yBAAyB,yBADtC,EAAE,EAEf,CACD;AAAA;AAAA,MACH;AAAA,OACF,IACE;AAAA,IACJ,oBAAC,SAAI,KAAK,uBAAuB,WAAU,UAAS,eAAW,MAAC;AAAA,IAC/D;AAAA,KACH;AAEJ;AAGO,SAAS,oBAA2D;AACzE,QAAM,MAAMH,YAAW,mBAAmB;AAC1C,SAAO,YAAY,YAAY;AAC7B,QAAI,CAAC,KAAK,QAAQ,WAAW,CAAC,IAAI,MAAO,QAAO,CAAC;AACjD,WAAO,IAAI,kBAAkB;AAAA,EAC/B,GAAG,CAAC,GAAG,CAAC;AACV;","names":["useEffect","createContext","useContext","useEffect","useState","token"]}
1
+ {"version":3,"sources":["../src/hooks/use-mobile.ts","../src/hooks/use-analytics.ts","../src/hooks/use-plugin.ts","../src/components/CaptchaProvider.tsx"],"sourcesContent":["import { useState, useEffect } from 'react';\r\n\r\nconst MOBILE_BREAKPOINT = 768;\r\n\r\nexport function useIsMobile(): boolean {\r\n const [isMobile, setIsMobile] = useState<boolean | undefined>(undefined);\r\n\r\n useEffect(() => {\r\n const mql = window.matchMedia(`(max-width: ${MOBILE_BREAKPOINT - 1}px)`);\r\n const onChange = () => setIsMobile(window.innerWidth < MOBILE_BREAKPOINT);\r\n mql.addEventListener('change', onChange);\r\n setIsMobile(window.innerWidth < MOBILE_BREAKPOINT);\r\n return () => mql.removeEventListener('change', onChange);\r\n }, []);\r\n\r\n return !!isMobile;\r\n}\r\n","import { useEffect } from 'react';\r\n\r\n/**\r\n * Call from app with pathname from usePathname() (next/navigation).\r\n * Disables tracking on admin routes when pathname starts with /admin.\r\n */\r\nexport function useAnalytics(pathname: string | null): { isAdminRoute: boolean; shouldTrack: boolean } {\r\n const isAdminRoute = pathname != null && pathname.startsWith('/admin');\r\n\r\n useEffect(() => {\r\n if (isAdminRoute && typeof window !== 'undefined') {\r\n if (typeof (window as unknown as { gtag?: () => void }).gtag === 'function') {\r\n (window as unknown as { gtag: () => void }).gtag = () => {};\r\n }\r\n if (Array.isArray((window as unknown as { dataLayer?: unknown[] }).dataLayer)) {\r\n (window as unknown as { dataLayer: unknown[] }).dataLayer = [];\r\n }\r\n }\r\n }, [isAdminRoute]);\r\n\r\n return { isAdminRoute, shouldTrack: !isAdminRoute };\r\n}\r\n","import React, { createContext, useContext, type ReactNode } from 'react';\r\n\r\nconst PluginContext = createContext<{ getPlugin: <T>(name: string) => T | undefined } | null>(null);\r\n\r\nexport function PluginProvider({\r\n getPlugin,\r\n children,\r\n}: {\r\n getPlugin: <T>(name: string) => T | undefined;\r\n children: ReactNode;\r\n}) {\r\n return React.createElement(PluginContext.Provider, { value: { getPlugin } }, children);\r\n}\r\n\r\nexport function usePlugin<T = unknown>(name: string): T | undefined {\r\n const ctx = useContext(PluginContext);\r\n if (!ctx) return undefined;\r\n return ctx.getPlugin<T>(name);\r\n}\r\n","'use client';\r\n\r\nimport React, {\r\n createContext,\r\n useCallback,\r\n useContext,\r\n useEffect,\r\n useMemo,\r\n useRef,\r\n useState,\r\n} from 'react';\r\nimport type { CaptchaPublicConfig, CaptchaProviderId } from '../plugins/captcha';\r\n\r\ndeclare global {\r\n interface Window {\r\n turnstile?: {\r\n render: (el: HTMLElement | string, opts: Record<string, unknown>) => string;\r\n execute: (id: string) => void;\r\n reset: (id: string) => void;\r\n remove: (id: string) => void;\r\n };\r\n grecaptcha?: {\r\n ready: (fn: () => void) => void;\r\n execute: (siteKey: string, opts: { action: string }) => Promise<string>;\r\n };\r\n }\r\n}\r\n\r\ntype CaptchaCtx = {\r\n config: CaptchaPublicConfig | null;\r\n ready: boolean;\r\n selectedProvider: CaptchaProviderId | null;\r\n setSelectedProvider: (p: CaptchaProviderId) => void;\r\n getCaptchaPayload: () => Promise<Record<string, string>>;\r\n};\r\n\r\nconst CaptchaReactContext = createContext<CaptchaCtx | null>(null);\r\n\r\nfunction loadScript(src: string): Promise<void> {\r\n return new Promise((resolve, reject) => {\r\n const base = src.split('?')[0] ?? src;\r\n const existing = Array.from(document.querySelectorAll('script[src]')).some((el) =>\r\n (el as HTMLScriptElement).src.startsWith(base)\r\n );\r\n if (existing) {\r\n resolve();\r\n return;\r\n }\r\n const s = document.createElement('script');\r\n s.src = src;\r\n s.async = true;\r\n s.onload = () => resolve();\r\n s.onerror = () => reject(new Error('Failed to load script'));\r\n document.head.appendChild(s);\r\n });\r\n}\r\n\r\nexport function CaptchaProvider({ children }: { children: React.ReactNode }) {\r\n const [config, setConfig] = useState<CaptchaPublicConfig | null>(null);\r\n const [selectedProvider, setSelectedProvider] = useState<CaptchaProviderId | null>(null);\r\n const [ready, setReady] = useState(false);\r\n const turnstileWidgetIdRef = useRef<string | null>(null);\r\n const turnstileContainerRef = useRef<HTMLDivElement | null>(null);\r\n const turnstileTokenRef = useRef<{\r\n resolve: (t: string) => void;\r\n reject: (e: Error) => void;\r\n } | null>(null);\r\n\r\n useEffect(() => {\r\n let cancelled = false;\r\n fetch('/api/public/captcha-config')\r\n .then((r) => (r.ok ? r.json() : null))\r\n .then((c: CaptchaPublicConfig | null) => {\r\n if (cancelled || !c) return;\r\n setConfig(c);\r\n if (c.enabled && c.activeProvider) setSelectedProvider(c.activeProvider);\r\n })\r\n .catch(() => {})\r\n .finally(() => {\r\n if (!cancelled) setReady(true);\r\n });\r\n return () => {\r\n cancelled = true;\r\n };\r\n }, []);\r\n\r\n useEffect(() => {\r\n if (!config?.enabled || !selectedProvider) return;\r\n const siteKeyFor = config.availableProviders.find((p) => p.id === selectedProvider)?.siteKey;\r\n if (!siteKeyFor) return;\r\n\r\n let cancelled = false;\r\n (async () => {\r\n try {\r\n if (selectedProvider === 'recaptcha_v3') {\r\n await loadScript(\r\n `https://www.google.com/recaptcha/api.js?render=${encodeURIComponent(siteKeyFor)}`\r\n );\r\n } else {\r\n await loadScript('https://challenges.cloudflare.com/turnstile/v0/api.js');\r\n if (cancelled || !turnstileContainerRef.current || !window.turnstile) return;\r\n if (turnstileWidgetIdRef.current) {\r\n try {\r\n window.turnstile.remove(turnstileWidgetIdRef.current);\r\n } catch {\r\n /* ignore */\r\n }\r\n turnstileWidgetIdRef.current = null;\r\n }\r\n const wid = window.turnstile.render(turnstileContainerRef.current, {\r\n sitekey: siteKeyFor,\r\n execution: 'execute',\r\n callback: (token: string) => {\r\n const p = turnstileTokenRef.current;\r\n turnstileTokenRef.current = null;\r\n p?.resolve(token);\r\n },\r\n 'error-callback': () => {\r\n const p = turnstileTokenRef.current;\r\n turnstileTokenRef.current = null;\r\n p?.reject(new Error('Turnstile error'));\r\n },\r\n 'expired-callback': () => {\r\n const p = turnstileTokenRef.current;\r\n turnstileTokenRef.current = null;\r\n p?.reject(new Error('Turnstile expired'));\r\n },\r\n });\r\n turnstileWidgetIdRef.current = wid;\r\n }\r\n } catch {\r\n /* ignore */\r\n }\r\n })();\r\n return () => {\r\n cancelled = true;\r\n };\r\n }, [config, selectedProvider]);\r\n\r\n const getCaptchaPayload = useCallback(async (): Promise<Record<string, string>> => {\r\n if (!config?.enabled || !selectedProvider) return {};\r\n const siteKeyFor = config.availableProviders.find((p) => p.id === selectedProvider)?.siteKey;\r\n if (!siteKeyFor) return {};\r\n\r\n if (selectedProvider === 'recaptcha_v3') {\r\n if (!window.grecaptcha) return {};\r\n const token = await new Promise<string>((resolve, reject) => {\r\n window.grecaptcha!.ready(() => {\r\n window.grecaptcha!.execute(siteKeyFor, { action: 'submit' }).then(resolve).catch(reject);\r\n });\r\n });\r\n return { captchaToken: token, captchaProvider: 'recaptcha_v3' };\r\n }\r\n\r\n const wid = turnstileWidgetIdRef.current;\r\n if (!wid || !window.turnstile) return {};\r\n const token = await new Promise<string>((resolve, reject) => {\r\n turnstileTokenRef.current = { resolve, reject };\r\n try {\r\n window.turnstile!.reset(wid);\r\n window.turnstile!.execute(wid);\r\n } catch (e) {\r\n turnstileTokenRef.current = null;\r\n reject(e instanceof Error ? e : new Error('Turnstile execute failed'));\r\n }\r\n });\r\n return { captchaToken: token, captchaProvider: 'turnstile' };\r\n }, [config, selectedProvider]);\r\n\r\n const ctx = useMemo<CaptchaCtx>(\r\n () => ({\r\n config,\r\n ready,\r\n selectedProvider,\r\n setSelectedProvider,\r\n getCaptchaPayload,\r\n }),\r\n [config, ready, selectedProvider, getCaptchaPayload]\r\n );\r\n\r\n return (\r\n <CaptchaReactContext.Provider value={ctx}>\r\n {config?.enabled && config.multipleProviders ? (\r\n <div className=\"mb-2 text-sm text-muted-foreground max-w-md\">\r\n <label className=\"mr-2\" htmlFor=\"captcha-provider-select\">\r\n Verification\r\n </label>\r\n <select\r\n id=\"captcha-provider-select\"\r\n className=\"border rounded px-2 py-1 bg-background\"\r\n value={selectedProvider ?? ''}\r\n onChange={(e) => setSelectedProvider(e.target.value as CaptchaProviderId)}\r\n >\r\n {config.availableProviders.map((p) => (\r\n <option key={p.id} value={p.id}>\r\n {p.id === 'turnstile' ? 'Cloudflare Turnstile' : 'Google reCAPTCHA v3'}\r\n </option>\r\n ))}\r\n </select>\r\n </div>\r\n ) : null}\r\n <div ref={turnstileContainerRef} className=\"hidden\" aria-hidden />\r\n {children}\r\n </CaptchaReactContext.Provider>\r\n );\r\n}\r\n\r\n/** Returns a function that resolves to captcha fields for JSON bodies, or {} if disabled / not ready. */\r\nexport function useCaptchaPayload(): () => Promise<Record<string, string>> {\r\n const ctx = useContext(CaptchaReactContext);\r\n return useCallback(async () => {\r\n if (!ctx?.config?.enabled || !ctx.ready) return {};\r\n return ctx.getCaptchaPayload();\r\n }, [ctx]);\r\n}\r\n"],"mappings":"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","names":["useEffect","createContext","useContext","useEffect","useState","token"]}
@@ -61,6 +61,8 @@ interface CrudHandlerOptions {
61
61
  getCms?: () => Promise<{
62
62
  getPlugin: (name: string) => unknown;
63
63
  }>;
64
+ /** When set, soft-delete sets `deletedBy` from the current admin user. */
65
+ getDeletedByUserId?: (req: Request) => Promise<number | null>;
64
66
  }
65
67
  declare function createCrudHandler(dataSource: DataSource, entityMap: EntityMap, options: CrudHandlerOptions): {
66
68
  GET(req: Request, resource: string): Promise<Response>;
@@ -244,6 +246,8 @@ interface UsersApiConfig extends CmsHandlersBase {
244
246
  getPlugin: (name: string) => unknown;
245
247
  }>;
246
248
  getCompanyDetails?: () => Promise<CompanyDetails>;
249
+ /** When set, soft-delete sets `deletedBy` on the user row. */
250
+ getSessionUser?: () => Promise<SessionUser | null>;
247
251
  }
248
252
  declare function createUsersApiHandlers(config: UsersApiConfig): {
249
253
  list(req: Request): Promise<Response>;
@@ -360,7 +364,10 @@ interface CmsApiHandlerConfig {
360
364
  settings?: SettingsApiConfig;
361
365
  /** POST /api/chat/identify, GET /api/chat/conversations/:id/messages, POST /api/chat/messages */
362
366
  chat?: ChatApiConfig;
363
- /** When set, CRUD and admin routes enforce entity-level permissions from session */
367
+ /**
368
+ * Entity-level RBAC (session `entityPerms` / Administrator). When omitted, admin routes that need it respond with 403
369
+ * (`entity_rbac_required`) so CRUD cannot run as auth-only. Pass `createAuthHelpers(...).requireEntityPermission` from the app.
370
+ */
364
371
  requireEntityPermission?: (req: Request, entity: string, action: EntityCrudAction) => Promise<Response | null>;
365
372
  /** Required for GET/POST/PATCH/DELETE /api/admin/roles */
366
373
  getSessionUser?: () => Promise<SessionUser | null>;
@@ -61,6 +61,8 @@ interface CrudHandlerOptions {
61
61
  getCms?: () => Promise<{
62
62
  getPlugin: (name: string) => unknown;
63
63
  }>;
64
+ /** When set, soft-delete sets `deletedBy` from the current admin user. */
65
+ getDeletedByUserId?: (req: Request) => Promise<number | null>;
64
66
  }
65
67
  declare function createCrudHandler(dataSource: DataSource, entityMap: EntityMap, options: CrudHandlerOptions): {
66
68
  GET(req: Request, resource: string): Promise<Response>;
@@ -244,6 +246,8 @@ interface UsersApiConfig extends CmsHandlersBase {
244
246
  getPlugin: (name: string) => unknown;
245
247
  }>;
246
248
  getCompanyDetails?: () => Promise<CompanyDetails>;
249
+ /** When set, soft-delete sets `deletedBy` on the user row. */
250
+ getSessionUser?: () => Promise<SessionUser | null>;
247
251
  }
248
252
  declare function createUsersApiHandlers(config: UsersApiConfig): {
249
253
  list(req: Request): Promise<Response>;
@@ -360,7 +364,10 @@ interface CmsApiHandlerConfig {
360
364
  settings?: SettingsApiConfig;
361
365
  /** POST /api/chat/identify, GET /api/chat/conversations/:id/messages, POST /api/chat/messages */
362
366
  chat?: ChatApiConfig;
363
- /** When set, CRUD and admin routes enforce entity-level permissions from session */
367
+ /**
368
+ * Entity-level RBAC (session `entityPerms` / Administrator). When omitted, admin routes that need it respond with 403
369
+ * (`entity_rbac_required`) so CRUD cannot run as auth-only. Pass `createAuthHelpers(...).requireEntityPermission` from the app.
370
+ */
364
371
  requireEntityPermission?: (req: Request, entity: string, action: EntityCrudAction) => Promise<Response | null>;
365
372
  /** Required for GET/POST/PATCH/DELETE /api/admin/roles */
366
373
  getSessionUser?: () => Promise<SessionUser | null>;