@infuro/cms-core 1.0.6 → 1.0.7

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.
@@ -0,0 +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'],\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":";;;;;;;;;;;;;;;;;;;;;;;;;;AAKA,gBAAe;AACf,kBAAiB;AAEjB,IAAM,YAAY;AAAA,EAChB,0BAA0B;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAsC1B,2BAA2B;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAiB3B,kBAAkB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAuBlB,oCAAoC;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAyFpC,2CAA2C;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAuC3C,4BAA4B;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAkB5B,sCAAsC;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAQtC,qBAAqB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAoCrB,yBAAyB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAkBzB,gBAAgB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAShB,mBAAmB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EA+BnB,oCAAoC;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAmBpC,6BAA6B;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAsB7B,4BAA4B;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAe5B,+BAA+B;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAsB/B,wCAAwC;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EA8CxC,wCAAwC;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EA2CxC,+CAA+C;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EA4B/C,+CAA+C;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAsB/C,6BAA6B;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EA2B7B,oBAAoB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAgBpB,4BAA4B;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAW9B;AAEA,SAAS,SAAS,KAA4B;AAC5C,MAAI,MAAM,YAAAA,QAAK,QAAQ,GAAG;AAC1B,WAAS,IAAI,GAAG,IAAI,IAAI,KAAK;AAC3B,UAAM,UAAU,YAAAA,QAAK,KAAK,KAAK,cAAc;AAC7C,QAAI,UAAAC,QAAG,WAAW,OAAO,GAAG;AAC1B,UAAI;AACF,cAAM,MAAM,KAAK,MAAM,UAAAA,QAAG,aAAa,SAAS,MAAM,CAAC;AACvD,YAAI,IAAI,cAAc,QAAQ,IAAI,iBAAiB,KAAM,QAAO;AAAA,MAClE,QAAQ;AAAA,MAER;AAAA,IACF;AACA,UAAM,SAAS,YAAAD,QAAK,QAAQ,GAAG;AAC/B,QAAI,WAAW,IAAK;AACpB,UAAM;AAAA,EACR;AACA,SAAO;AACT;AAEA,SAAS,UACP,MACA,UACA,SACAE,QACAC,SACA,KACS;AACT,QAAM,OAAO,YAAAH,QAAK,KAAK,MAAM,QAAQ;AACrC,MAAI,UAAAC,QAAG,WAAW,IAAI,KAAK,CAACC,QAAO;AACjC,QAAI,oBAAoB,QAAQ,EAAE;AAClC,WAAO;AAAA,EACT;AACA,MAAIC,SAAQ;AACV,QAAI,mBAAmB,QAAQ,EAAE;AACjC,WAAO;AAAA,EACT;AACA,QAAM,MAAM,YAAAH,QAAK,QAAQ,IAAI;AAC7B,YAAAC,QAAG,UAAU,KAAK,EAAE,WAAW,KAAK,CAAC;AACrC,YAAAA,QAAG,cAAc,MAAM,SAAS,MAAM;AACtC,MAAI,cAAc,QAAQ,EAAE;AAC5B,SAAO;AACT;AAEA,SAAS,gBAAgB,MAAcE,SAAiB,KAAqC;AAC3F,QAAM,aAAa,CAAC,kBAAkB,mBAAmB,iBAAiB;AAC1E,MAAI,aAA4B;AAChC,aAAW,QAAQ,YAAY;AAC7B,UAAM,IAAI,YAAAH,QAAK,KAAK,MAAM,IAAI;AAC9B,QAAI,UAAAC,QAAG,WAAW,CAAC,GAAG;AACpB,mBAAa;AACb;AAAA,IACF;AAAA,EACF;AACA,MAAI,CAAC,YAAY;AACf,QAAI,+BAA+B;AACnC,WAAO;AAAA,EACT;AACA,MAAI,UAAU,UAAAA,QAAG,aAAa,YAAY,MAAM;AAChD,MAAI,QAAQ,SAAS,oBAAoB,KAAK,QAAQ,SAAS,oBAAoB,GAAG;AACpF,QAAI,8BAA8B,YAAAD,QAAK,SAAS,UAAU,CAAC,EAAE;AAC7D,WAAO;AAAA,EACT;AACA,MAAI,QAAQ,SAAS,wBAAwB,GAAG;AAC9C,cAAU,QAAQ;AAAA,MAChB;AAAA,MACA;AAAA,IACF;AAAA,EACF,OAAO;AACL,cAAU,QAAQ;AAAA,MAChB;AAAA,MACA;AAAA,IACF;AAAA,EACF;AACA,MAAIG,SAAQ;AACV,QAAI,kBAAkB,YAAAH,QAAK,SAAS,UAAU,CAAC,EAAE;AACjD,WAAO;AAAA,EACT;AACA,YAAAC,QAAG,cAAc,YAAY,SAAS,MAAM;AAC5C,MAAI,cAAc,YAAAD,QAAK,SAAS,UAAU,CAAC,EAAE;AAC7C,SAAO;AACT;AAEA,SAAS,cAAc,MAAcG,SAAiB,KAAqC;AACzF,QAAM,aAAa,CAAC,sBAAsB,uBAAuB,oBAAoB;AACrF,MAAI,aAA4B;AAChC,aAAW,QAAQ,YAAY;AAC7B,UAAM,IAAI,YAAAH,QAAK,KAAK,MAAM,IAAI;AAC9B,QAAI,UAAAC,QAAG,WAAW,CAAC,GAAG;AACpB,mBAAa;AACb;AAAA,IACF;AAAA,EACF;AACA,MAAI,CAAC,YAAY;AACf,QAAI,mCAAmC;AACvC,WAAO;AAAA,EACT;AACA,MAAI,UAAU,UAAAA,QAAG,aAAa,YAAY,MAAM;AAChD,QAAM,cAAc;AACpB,MAAI,QAAQ,SAAS,kBAAkB,GAAG;AACxC,QAAI,8BAA8B,YAAAD,QAAK,SAAS,UAAU,CAAC,EAAE;AAC7D,WAAO;AAAA,EACT;AACA,MAAI,QAAQ,SAAS,UAAU,GAAG;AAChC,cAAU,QAAQ;AAAA,MAChB;AAAA,MACA;AAAA,OAAY,WAAW;AAAA,IACzB;AAAA,EACF;AACA,MAAIG,SAAQ;AACV,QAAI,kBAAkB,YAAAH,QAAK,SAAS,UAAU,CAAC,EAAE;AACjD,WAAO;AAAA,EACT;AACA,YAAAC,QAAG,cAAc,YAAY,SAAS,MAAM;AAC5C,MAAI,cAAc,YAAAD,QAAK,SAAS,UAAU,CAAC,EAAE;AAC7C,SAAO;AACT;AAEA,SAAS,YAAY,MAAcG,SAAiB,KAAqC;AACvF,QAAM,aAAa,YAAAH,QAAK,KAAK,MAAM,oBAAoB;AACvD,MAAI,CAAC,UAAAC,QAAG,WAAW,UAAU,GAAG;AAC9B,QAAI,6CAA6C;AACjD,WAAO;AAAA,EACT;AACA,MAAI,UAAU,UAAAA,QAAG,aAAa,YAAY,MAAM;AAChD,MAAI,QAAQ,SAAS,aAAa,GAAG;AACnC,QAAI,uCAAuC;AAC3C,WAAO;AAAA,EACT;AACA,QAAM,YAAY,QAAQ,MAAM,4CAA4C;AAC5E,MAAI,CAAC,WAAW;AACd,QAAI,gEAAgE;AACpE,WAAO;AAAA,EACT;AACA,MAAIE,SAAQ;AACV,QAAI,mCAAmC;AACvC,WAAO;AAAA,EACT;AACA,QAAM,CAAC,EAAE,WAAW,QAAQ,IAAI;AAChC,QAAM,UAAU,QAAQ,SAAS;AAAA,qBAAyB,QAAQ;AAAA;AAClE,YAAU,QAAQ,QAAQ,0CAA0C,OAAO;AAC3E,MAAI,CAAC,QAAQ,SAAS,oBAAoB,KAAK,CAAC,QAAQ,SAAS,oBAAoB,GAAG;AACtF,UAAM,cAAc,QAAQ,MAAM,wBAAwB;AAC1D,cAAU,cACN,QAAQ,QAAQ,YAAY,CAAC,GAAG,YAAY,CAAC,IAAI,4CAA4C,IAC7F,+CAA+C;AAAA,EACrD;AACA,MAAI,QAAQ,SAAS,OAAO,KAAK,CAAC,QAAQ,SAAS,0BAA0B,GAAG;AAC9E,cAAU,QAAQ,QAAQ,aAAa,kCAAkC;AAAA,EAC3E;AACA,YAAAF,QAAG,cAAc,YAAY,SAAS,MAAM;AAC5C,MAAI,+BAA+B;AACnC,SAAO;AACT;AAEA,SAAS,iBAAiB,MAAcE,SAAiB,KAAqC;AAC5F,QAAM,UAAU,YAAAH,QAAK,KAAK,MAAM,cAAc;AAC9C,MAAI,CAAC,UAAAC,QAAG,WAAW,OAAO,EAAG,QAAO;AACpC,QAAM,MAAM,KAAK,MAAM,UAAAA,QAAG,aAAa,SAAS,MAAM,CAAC;AACvD,QAAM,UAAU,IAAI,WAAW,CAAC;AAChC,MAAI,UAAU;AACd,MAAI,CAAC,QAAQ,MAAM;AACjB,YAAQ,OAAO;AACf,cAAU;AAAA,EACZ;AACA,MAAI,CAAC,QAAQ,eAAe,GAAG;AAC7B,YAAQ,eAAe,IAAI;AAC3B,cAAU;AAAA,EACZ;AACA,QAAM,MAAM,IAAI,mBAAmB,CAAC;AACpC,QAAM,OAAO,IAAI,gBAAgB,CAAC;AAClC,MAAI,CAAC,KAAK,kBAAkB,GAAG;AAC7B,SAAK,kBAAkB,IAAI;AAC3B,cAAU;AAAA,EACZ;AACA,MAAI,CAAC,IAAI,KAAK;AACZ,QAAI,MAAM;AACV,cAAU;AAAA,EACZ;AACA,MAAI,CAAC,IAAI,QAAQ;AACf,QAAI,SAAS;AACb,cAAU;AAAA,EACZ;AACA,MAAI,CAAC,SAAS;AACZ,QAAI,sDAAsD;AAC1D,WAAO;AAAA,EACT;AACA,MAAI,UAAU;AACd,MAAI,eAAe,EAAE,GAAG,IAAI,cAAc,GAAG,KAAK;AAClD,MAAI,kBAAkB,EAAE,GAAG,IAAI,iBAAiB,GAAG,IAAI;AACvD,MAAIE,SAAQ;AACV,QAAI,yDAAyD;AAC7D,WAAO;AAAA,EACT;AACA,YAAAF,QAAG,cAAc,SAAS,KAAK,UAAU,KAAK,MAAM,CAAC,GAAG,MAAM;AAC9D,MAAI,yBAAyB;AAC7B,SAAO;AACT;AAEA,eAAe,cAAc,MAAc,KAA2C;AACpF,QAAM,EAAE,UAAU,IAAI,MAAM,OAAO,eAAe;AAClD,MAAI,kCAAkC;AACtC,YAAU,OAAO,CAAC,WAAW,oBAAoB,WAAW,oBAAoB,YAAY,aAAa,eAAe,QAAQ,GAAG;AAAA,IACjI,KAAK;AAAA,IACL,OAAO;AAAA,IACP,OAAO;AAAA,EACT,CAAC;AACD,MAAI,qDAAqD;AACzD,YAAU,OAAO,CAAC,WAAW,MAAM,OAAO,UAAU,aAAa,GAAG;AAAA,IAClE,KAAK;AAAA,IACL,OAAO;AAAA,IACP,OAAO;AAAA,EACT,CAAC;AACH;AAEA,eAAe,QAAQ,MAKpB;AACD,QAAM,MAAM,CAAC,QAAgB,QAAQ,IAAI,GAAG;AAC5C,QAAM,MAAM,QAAQ,IAAI;AACxB,QAAM,OAAO,SAAS,GAAG;AACzB,MAAI,CAAC,MAAM;AACT,YAAQ,MAAM,4EAA4E,MAAM,GAAG;AACnG,YAAQ,KAAK,CAAC;AAAA,EAChB;AACA,QAAM,SAAS,YAAAD,QAAK,KAAK,MAAM,SAAS;AACxC,MAAI,CAAC,UAAAC,QAAG,WAAW,MAAM,GAAG;AAC1B,YAAQ,MAAM,8GAA8G;AAC5H,YAAQ,KAAK,CAAC;AAAA,EAChB;AAEA,MAAI,uBAAuB,IAAI;AAC/B,MAAI,KAAK,OAAQ,KAAI,WAAW;AAEhC,aAAW,CAAC,UAAU,OAAO,KAAK,OAAO,QAAQ,SAAS,GAAG;AAC3D,cAAU,MAAM,UAAU,SAAS,KAAK,OAAO,KAAK,QAAQ,GAAG;AAAA,EACjE;AAEA,MAAI,CAAC,KAAK,eAAe;AACvB,QAAI,iBAAiB;AACrB,oBAAgB,MAAM,KAAK,QAAQ,GAAG;AACtC,kBAAc,MAAM,KAAK,QAAQ,GAAG;AACpC,gBAAY,MAAM,KAAK,QAAQ,GAAG;AAClC,qBAAiB,MAAM,KAAK,QAAQ,GAAG;AAAA,EACzC;AAEA,MAAI,CAAC,KAAK,UAAU,CAAC,KAAK,QAAQ;AAChC,QAAI,eAAe;AACnB,UAAM,cAAc,MAAM,GAAG;AAAA,EAC/B,WAAW,CAAC,KAAK,UAAU,KAAK,QAAQ;AACtC,QAAI,0GAA0G;AAC9G,QAAI,oDAAoD;AAAA,EAC1D;AAEA,MAAI,EAAE;AACN,MAAI,mBAAmB;AACvB,MAAI,iHAAiH;AACrH,MAAI,kGAAkG;AACtG,MAAI,kBAAkB;AACxB;AAEA,IAAM,OAAO,QAAQ,KAAK,MAAM,CAAC;AACjC,IAAM,QAAQ,KAAK,SAAS,SAAS;AACrC,IAAM,SAAS,KAAK,SAAS,WAAW;AACxC,IAAM,SAAS,KAAK,SAAS,WAAW;AACxC,IAAM,gBAAgB,KAAK,SAAS,mBAAmB;AAEvD,IAAI,KAAK,CAAC,MAAM,UAAU,KAAK,SAAS,QAAQ,KAAM,KAAK,WAAW,KAAK,CAAC,KAAK,KAAK,CAAC,MAAM,EAAE,WAAW,IAAI,CAAC,GAAI;AACjH,UAAQ,EAAE,OAAO,QAAQ,QAAQ,cAAc,CAAC,EAAE,MAAM,CAAC,MAAM;AAC7D,YAAQ,MAAM,CAAC;AACf,YAAQ,KAAK,CAAC;AAAA,EAChB,CAAC;AACH,OAAO;AACL,UAAQ,IAAI,wFAAwF;AACpG,UAAQ,KAAK,KAAK,CAAC,MAAM,YAAY,KAAK,CAAC,MAAM,OAAO,IAAI,CAAC;AAC/D;","names":["path","fs","force","dryRun"]}
package/dist/cli.d.cts ADDED
@@ -0,0 +1 @@
1
+ #!/usr/bin/env node
package/dist/cli.d.ts ADDED
@@ -0,0 +1 @@
1
+ #!/usr/bin/env node