create-pulsekit 1.0.1 → 1.0.5

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/index.js CHANGED
@@ -178,26 +178,52 @@ async function Dashboard() {
178
178
 
179
179
  export default function AnalyticsPage() {
180
180
  return (
181
- <Suspense fallback={<div className="flex items-center justify-center min-h-screen p-6"><Spinner className="size-6" /></div>}>
182
- <PulseAuthGate secret={process.env.PULSE_SECRET}>
181
+ <Suspense fallback={<div style={{ display: "flex", alignItems: "center", justifyContent: "center", minHeight: "100vh", padding: "1.5rem" }}><Spinner style={{ width: 24, height: 24 }} /></div>}>
182
+ <PulseAuthGate secret={process.env.PULSE_SECRET!}>
183
183
  <Dashboard />
184
184
  </PulseAuthGate>
185
185
  </Suspense>
186
186
  );
187
187
  }
188
188
  `;
189
- const spinnerContent = `import { LoaderIcon } from "lucide-react"
190
- import { cn } from "@/lib/utils"
189
+ const trackerWrapper = `import { PulseTracker } from "@pulsekit/next/client";
190
+ import { createPulseIngestionToken } from "@pulsekit/next";
191
+ import { connection } from "next/server";
192
+
193
+ export default async function PulseTrackerWrapper() {
194
+ await connection();
195
+ const token = process.env.PULSE_SECRET
196
+ ? await createPulseIngestionToken(process.env.PULSE_SECRET)
197
+ : undefined;
191
198
 
192
- function Spinner({ className, ...props }: React.ComponentProps<"svg">) {
193
199
  return (
194
- <LoaderIcon
195
- role="status"
196
- aria-label="Loading"
197
- className={cn("size-4 animate-spin", className)}
198
- {...props}
200
+ <PulseTracker
201
+ excludePaths={["/admin/analytics"]}
202
+ token={token}
199
203
  />
200
- )
204
+ );
205
+ }
206
+ `;
207
+ const spinnerContent = `function Spinner({ style, ...props }: React.ComponentProps<"svg">) {
208
+ return (
209
+ <>
210
+ <style>{\`@keyframes pulsekit-spin { to { transform: rotate(360deg) } }\`}</style>
211
+ <svg
212
+ role="status"
213
+ aria-label="Loading"
214
+ width="16"
215
+ height="16"
216
+ xmlns="http://www.w3.org/2000/svg"
217
+ fill="none"
218
+ viewBox="0 0 24 24"
219
+ style={{ animation: "pulsekit-spin 1s linear infinite", ...style }}
220
+ {...props}
221
+ >
222
+ <circle opacity={0.25} cx="12" cy="12" r="10" stroke="currentColor" strokeWidth="4" />
223
+ <path opacity={0.75} fill="currentColor" d="M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4z" />
224
+ </svg>
225
+ </>
226
+ );
201
227
  }
202
228
 
203
229
  export { Spinner }
@@ -212,6 +238,15 @@ export { Spinner }
212
238
  import_node_fs2.default.writeFileSync(spinnerPath, spinnerContent, "utf8");
213
239
  console.log(" Created: components/ui/spinner.tsx");
214
240
  }
241
+ const componentsDir = appDir.includes(import_node_path2.default.join("src", "app")) ? import_node_path2.default.join(cwd, "src", "components") : import_node_path2.default.join(cwd, "components");
242
+ const wrapperPath = import_node_path2.default.join(componentsDir, "pulse-tracker-wrapper.tsx");
243
+ import_node_fs2.default.mkdirSync(componentsDir, { recursive: true });
244
+ if (import_node_fs2.default.existsSync(wrapperPath)) {
245
+ console.log(" Skipped (already exists): components/pulse-tracker-wrapper.tsx");
246
+ } else {
247
+ import_node_fs2.default.writeFileSync(wrapperPath, trackerWrapper, "utf8");
248
+ console.log(" Created: components/pulse-tracker-wrapper.tsx");
249
+ }
215
250
  const files = [
216
251
  { rel: "api/pulse/route.ts", content: pulseRoute },
217
252
  { rel: "api/pulse/auth/route.ts", content: authRoute },
@@ -255,21 +290,28 @@ async function injectPulseTracker() {
255
290
  return;
256
291
  }
257
292
  let content = import_node_fs3.default.readFileSync(foundPath, "utf8");
258
- if (content.includes("PulseTracker")) {
293
+ if (content.includes("PulseTrackerWrapper") || content.includes("PulseTracker")) {
259
294
  console.log(" PulseTracker already present in layout. Skipping.\n");
260
295
  return;
261
296
  }
262
- const importStatement = 'import { PulseTracker } from "@pulsekit/next/client";';
297
+ const importStatements = [
298
+ 'import { Suspense } from "react";',
299
+ 'import PulseTrackerWrapper from "@/components/pulse-tracker-wrapper";'
300
+ ];
263
301
  const importRegex = /^import\s.+$/gm;
264
302
  let lastImportEnd = -1;
265
303
  let match;
266
304
  while ((match = importRegex.exec(content)) !== null) {
267
305
  lastImportEnd = match.index + match[0].length;
268
306
  }
269
- if (lastImportEnd === -1) {
270
- content = importStatement + "\n" + content;
271
- } else {
272
- content = content.slice(0, lastImportEnd) + "\n" + importStatement + content.slice(lastImportEnd);
307
+ const newImports = importStatements.filter((stmt) => !content.includes(stmt));
308
+ const importBlock = newImports.join("\n");
309
+ if (importBlock) {
310
+ if (lastImportEnd === -1) {
311
+ content = importBlock + "\n" + content;
312
+ } else {
313
+ content = content.slice(0, lastImportEnd) + "\n" + importBlock + content.slice(lastImportEnd);
314
+ }
273
315
  }
274
316
  const bodyCloseIndex = content.lastIndexOf("</body>");
275
317
  if (bodyCloseIndex === -1) {
@@ -278,8 +320,11 @@ async function injectPulseTracker() {
278
320
  }
279
321
  const lineStart = content.lastIndexOf("\n", bodyCloseIndex) + 1;
280
322
  const indent = content.slice(lineStart, bodyCloseIndex).match(/^\s*/)?.[0] ?? " ";
281
- const trackerJsx = `${indent} <PulseTracker excludePaths={["/admin/analytics"]} />
282
- `;
323
+ const trackerJsx = [
324
+ `${indent} <Suspense>`,
325
+ `${indent} <PulseTrackerWrapper />`,
326
+ `${indent} </Suspense>`
327
+ ].join("\n") + "\n";
283
328
  content = content.slice(0, lineStart) + trackerJsx + content.slice(lineStart);
284
329
  import_node_fs3.default.writeFileSync(foundPath, content, "utf8");
285
330
  console.log(
@@ -291,12 +336,13 @@ function printManualInstructions() {
291
336
  console.log(
292
337
  " Could not auto-inject PulseTracker. Add it manually to your layout:\n"
293
338
  );
294
- console.log(' import { PulseTracker } from "@pulsekit/next/client";');
339
+ console.log(' import { Suspense } from "react";');
340
+ console.log(' import PulseTrackerWrapper from "@/components/pulse-tracker-wrapper";');
295
341
  console.log("");
296
342
  console.log(" // Add inside your <body> tag:");
297
- console.log(
298
- ' <PulseTracker excludePaths={["/admin/analytics"]} />\n'
299
- );
343
+ console.log(" <Suspense>");
344
+ console.log(" <PulseTrackerWrapper />");
345
+ console.log(" </Suspense>\n");
300
346
  }
301
347
 
302
348
  // src/inject-instrumentation.ts
package/dist/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/index.ts","../src/detect.ts","../src/install.ts","../src/scaffold.ts","../src/inject.ts","../src/inject-instrumentation.ts","../src/migration.ts"],"sourcesContent":["import fs from \"node:fs\";\nimport path from \"node:path\";\nimport { detectPackageManager, validateNextJsProject } from \"./detect\";\nimport { installPackages } from \"./install\";\nimport { scaffoldFiles } from \"./scaffold\";\nimport { injectPulseTracker } from \"./inject\";\nimport { injectInstrumentation } from \"./inject-instrumentation\";\nimport { writeMigration } from \"./migration\";\n\nfunction appendEnvExample(): void {\n const envExamplePath = path.join(process.cwd(), \".env.example\");\n if (!fs.existsSync(envExamplePath)) return;\n\n const content = fs.readFileSync(envExamplePath, \"utf8\");\n const entries: { key: string; value: string }[] = [\n { key: \"PULSE_SECRET\", value: \"\" },\n { key: \"SUPABASE_SERVICE_ROLE_KEY\", value: \"\" },\n ];\n\n const missing = entries.filter((e) => !content.includes(e.key));\n if (missing.length === 0) return;\n\n const block =\n \"\\n# PulseKit\\n\" +\n missing.map((e) => `${e.key}=${e.value}`).join(\"\\n\") +\n \"\\n\";\n fs.appendFileSync(envExamplePath, block, \"utf8\");\n console.log(\" Updated .env.example with PulseKit variables.\\n\");\n}\n\nasync function main() {\n console.log(\"\\n create-pulsekit\\n\");\n console.log(\" Setting up PulseKit analytics in your Next.js project.\\n\");\n\n const pm = detectPackageManager();\n console.log(` Detected package manager: ${pm}\\n`);\n\n validateNextJsProject();\n\n await installPackages(pm);\n\n scaffoldFiles();\n\n await injectPulseTracker();\n\n await injectInstrumentation();\n\n writeMigration();\n\n appendEnvExample();\n\n console.log(\"\\n Done! PulseKit has been added to your project.\\n\");\n console.log(\" To finish setup:\");\n console.log(\" 1. Add these variables to .env.local:\");\n console.log(\" NEXT_PUBLIC_SUPABASE_URL=<your-supabase-url>\");\n console.log(\" NEXT_PUBLIC_SUPABASE_PUBLISHABLE_KEY=<your-anon-key>\");\n console.log(\" SUPABASE_SERVICE_ROLE_KEY=<your-service-role-key>\");\n console.log(\" PULSE_SECRET=<a-random-string-at-least-16-chars>\");\n console.log(\" 2. Run the database migration:\");\n console.log(\" npx supabase link\");\n console.log(\" npx supabase db push\");\n console.log(\" 3. If your middleware protects routes (e.g. Supabase auth),\");\n console.log(\" allow the PulseKit paths through:\");\n console.log(' !request.nextUrl.pathname.startsWith(\"/api/pulse\")');\n console.log(' !request.nextUrl.pathname.startsWith(\"/admin/analytics\")');\n console.log(\" 4. Start your dev server and visit /admin/analytics\\n\");\n}\n\nmain().catch((err) => {\n console.error(\"\\n Error:\", err.message || err);\n process.exit(1);\n});\n","import fs from \"node:fs\";\nimport path from \"node:path\";\n\nexport type PackageManager = \"npm\" | \"pnpm\" | \"yarn\" | \"bun\";\n\nexport function detectPackageManager(): PackageManager {\n const cwd = process.cwd();\n\n if (\n fs.existsSync(path.join(cwd, \"bun.lock\")) ||\n fs.existsSync(path.join(cwd, \"bun.lockb\"))\n ) {\n return \"bun\";\n }\n if (fs.existsSync(path.join(cwd, \"pnpm-lock.yaml\"))) {\n return \"pnpm\";\n }\n if (fs.existsSync(path.join(cwd, \"yarn.lock\"))) {\n return \"yarn\";\n }\n return \"npm\";\n}\n\nexport function validateNextJsProject(): void {\n const cwd = process.cwd();\n\n const pkgPath = path.join(cwd, \"package.json\");\n if (!fs.existsSync(pkgPath)) {\n throw new Error(\n \"No package.json found. Run this command from the root of a Next.js project.\"\n );\n }\n\n const pkg = JSON.parse(fs.readFileSync(pkgPath, \"utf8\"));\n const allDeps = { ...pkg.dependencies, ...pkg.devDependencies };\n if (!allDeps[\"next\"]) {\n throw new Error(\n \"next is not listed in package.json. This command requires a Next.js project.\"\n );\n }\n\n if (!fs.existsSync(getAppDir())) {\n throw new Error(\n \"No app/ directory found. PulseKit requires the Next.js App Router.\"\n );\n }\n}\n\nexport function getAppDir(): string {\n const cwd = process.cwd();\n const srcAppDir = path.join(cwd, \"src\", \"app\");\n if (fs.existsSync(srcAppDir)) {\n return srcAppDir;\n }\n return path.join(cwd, \"app\");\n}\n","import { execSync } from \"node:child_process\";\nimport type { PackageManager } from \"./detect\";\n\nconst PACKAGES = [\n \"@pulsekit/core\",\n \"@pulsekit/next\",\n \"@pulsekit/react\",\n \"@supabase/supabase-js@latest\",\n];\n\nexport async function installPackages(pm: PackageManager): Promise<void> {\n console.log(\" Installing packages...\\n\");\n\n const commands: Record<PackageManager, string> = {\n npm: `npm install ${PACKAGES.join(\" \")}`,\n pnpm: `pnpm add ${PACKAGES.join(\" \")}`,\n yarn: `yarn add ${PACKAGES.join(\" \")}`,\n bun: `bun add ${PACKAGES.join(\" \")}`,\n };\n\n const cmd = commands[pm];\n console.log(` > ${cmd}\\n`);\n\n try {\n execSync(cmd, { cwd: process.cwd(), stdio: \"inherit\" });\n } catch {\n throw new Error(\n `Package installation failed. Try running manually:\\n ${cmd}`\n );\n }\n}\n","import fs from \"node:fs\";\nimport path from \"node:path\";\nimport { getAppDir } from \"./detect\";\n\nexport function scaffoldFiles(): void {\n console.log(\" Scaffolding files...\\n\");\n\n const appDir = getAppDir();\n\n const pulseRoute = `import { createPulseHandler } from \"@pulsekit/next\";\nimport { createClient } from \"@supabase/supabase-js\";\n\nconst supabase = createClient(\n process.env.NEXT_PUBLIC_SUPABASE_URL!,\n process.env.NEXT_PUBLIC_SUPABASE_PUBLISHABLE_KEY!\n);\n\nexport const POST = createPulseHandler({\n supabase,\n config: {\n siteId: \"default\",\n secret: process.env.PULSE_SECRET,\n },\n});\n`;\n\n const refreshRoute = `import { createRefreshHandler, withPulseAuth } from \"@pulsekit/next\";\nimport { createClient } from \"@supabase/supabase-js\";\n\nconst supabase = createClient(\n process.env.NEXT_PUBLIC_SUPABASE_URL!,\n process.env.NEXT_PUBLIC_SUPABASE_PUBLISHABLE_KEY!\n);\n\nexport const POST = withPulseAuth(createRefreshHandler({ supabase }));\n`;\n\n const consolidateRoute = `import { createConsolidateHandler, withPulseAuth } from \"@pulsekit/next\";\nimport { createClient } from \"@supabase/supabase-js\";\n\nconst supabase = createClient(\n process.env.NEXT_PUBLIC_SUPABASE_URL!,\n process.env.NEXT_PUBLIC_SUPABASE_PUBLISHABLE_KEY!\n);\n\nexport const POST = withPulseAuth(createConsolidateHandler({ supabase }));\n`;\n\n const authRoute = `import { createPulseAuthHandler } from \"@pulsekit/next\";\n\nconst handler = createPulseAuthHandler({ secret: process.env.PULSE_SECRET! });\n\nexport const POST = handler;\nexport const DELETE = handler;\n`;\n\n const dashboardPage = `import { Suspense } from \"react\";\nimport { createClient } from \"@supabase/supabase-js\";\nimport { PulseDashboard, PulseAuthGate } from \"@pulsekit/react\";\nimport { getPulseTimezone } from \"@pulsekit/next\";\nimport { Spinner } from \"@/components/ui/spinner\";\nimport \"@pulsekit/react/pulse.css\";\n\nconst supabase = createClient(\n process.env.NEXT_PUBLIC_SUPABASE_URL!,\n process.env.NEXT_PUBLIC_SUPABASE_PUBLISHABLE_KEY!\n);\n\nasync function Dashboard() {\n const timezone = await getPulseTimezone();\n\n return (\n <PulseDashboard\n supabase={supabase}\n siteId=\"default\"\n timeframe=\"7d\"\n timezone={timezone}\n />\n );\n}\n\nexport default function AnalyticsPage() {\n return (\n <Suspense fallback={<div className=\"flex items-center justify-center min-h-screen p-6\"><Spinner className=\"size-6\" /></div>}>\n <PulseAuthGate secret={process.env.PULSE_SECRET}>\n <Dashboard />\n </PulseAuthGate>\n </Suspense>\n );\n}\n`;\n\n // Scaffold the spinner component if not already present\n const spinnerContent = `import { LoaderIcon } from \"lucide-react\"\nimport { cn } from \"@/lib/utils\"\n\nfunction Spinner({ className, ...props }: React.ComponentProps<\"svg\">) {\n return (\n <LoaderIcon\n role=\"status\"\n aria-label=\"Loading\"\n className={cn(\"size-4 animate-spin\", className)}\n {...props}\n />\n )\n}\n\nexport { Spinner }\n`;\n\n const cwd = process.cwd();\n const componentsBase = appDir.includes(path.join(\"src\", \"app\"))\n ? path.join(cwd, \"src\", \"components\", \"ui\")\n : path.join(cwd, \"components\", \"ui\");\n\n const spinnerPath = path.join(componentsBase, \"spinner.tsx\");\n fs.mkdirSync(componentsBase, { recursive: true });\n if (fs.existsSync(spinnerPath)) {\n console.log(\" Skipped (already exists): components/ui/spinner.tsx\");\n } else {\n fs.writeFileSync(spinnerPath, spinnerContent, \"utf8\");\n console.log(\" Created: components/ui/spinner.tsx\");\n }\n\n const files = [\n { rel: \"api/pulse/route.ts\", content: pulseRoute },\n { rel: \"api/pulse/auth/route.ts\", content: authRoute },\n { rel: \"api/pulse/refresh-aggregates/route.ts\", content: refreshRoute },\n { rel: \"api/pulse/consolidate/route.ts\", content: consolidateRoute },\n { rel: \"admin/analytics/page.tsx\", content: dashboardPage },\n ];\n\n for (const { rel, content } of files) {\n const fullPath = path.join(appDir, rel);\n fs.mkdirSync(path.dirname(fullPath), { recursive: true });\n\n if (fs.existsSync(fullPath)) {\n console.log(` Skipped (already exists): ${rel}`);\n continue;\n }\n\n fs.writeFileSync(fullPath, content, \"utf8\");\n console.log(` Created: ${rel}`);\n }\n\n console.log(\"\");\n}\n","import fs from \"node:fs\";\nimport path from \"node:path\";\nimport { getAppDir } from \"./detect\";\n\nexport async function injectPulseTracker(): Promise<void> {\n console.log(\" Injecting PulseTracker into layout...\\n\");\n\n const appDir = getAppDir();\n\n const candidates = [\n path.join(appDir, \"layout.tsx\"),\n path.join(appDir, \"layout.jsx\"),\n path.join(appDir, \"layout.js\"),\n ];\n\n let foundPath: string | null = null;\n for (const candidate of candidates) {\n if (fs.existsSync(candidate)) {\n foundPath = candidate;\n break;\n }\n }\n\n if (!foundPath) {\n printManualInstructions();\n return;\n }\n\n let content = fs.readFileSync(foundPath, \"utf8\");\n\n if (content.includes(\"PulseTracker\")) {\n console.log(\" PulseTracker already present in layout. Skipping.\\n\");\n return;\n }\n\n // Add import after the last existing import\n const importStatement =\n 'import { PulseTracker } from \"@pulsekit/next/client\";';\n\n const importRegex = /^import\\s.+$/gm;\n let lastImportEnd = -1;\n let match: RegExpExecArray | null;\n while ((match = importRegex.exec(content)) !== null) {\n lastImportEnd = match.index + match[0].length;\n }\n\n if (lastImportEnd === -1) {\n content = importStatement + \"\\n\" + content;\n } else {\n content =\n content.slice(0, lastImportEnd) +\n \"\\n\" +\n importStatement +\n content.slice(lastImportEnd);\n }\n\n // Inject <PulseTracker /> before </body>\n const bodyCloseIndex = content.lastIndexOf(\"</body>\");\n if (bodyCloseIndex === -1) {\n printManualInstructions();\n return;\n }\n\n // Detect indentation from the </body> line\n const lineStart = content.lastIndexOf(\"\\n\", bodyCloseIndex) + 1;\n const indent = content.slice(lineStart, bodyCloseIndex).match(/^\\s*/)?.[0] ?? \" \";\n const trackerJsx = `${indent} <PulseTracker excludePaths={[\"/admin/analytics\"]} />\\n`;\n\n content =\n content.slice(0, lineStart) + trackerJsx + content.slice(lineStart);\n\n fs.writeFileSync(foundPath, content, \"utf8\");\n console.log(\n ` Modified: ${path.relative(process.cwd(), foundPath)}\\n`\n );\n}\n\nfunction printManualInstructions(): void {\n console.log(\n \" Could not auto-inject PulseTracker. Add it manually to your layout:\\n\"\n );\n console.log(' import { PulseTracker } from \"@pulsekit/next/client\";');\n console.log(\"\");\n console.log(\" // Add inside your <body> tag:\");\n console.log(\n ' <PulseTracker excludePaths={[\"/admin/analytics\"]} />\\n'\n );\n}\n","import fs from \"node:fs\";\nimport path from \"node:path\";\nimport { getAppDir } from \"./detect\";\n\nconst FULL_CONTENT = `import { createClient } from \"@supabase/supabase-js\";\nimport { createPulseErrorReporter } from \"@pulsekit/next\";\n\nconst supabase = createClient(\n process.env.NEXT_PUBLIC_SUPABASE_URL!,\n process.env.NEXT_PUBLIC_SUPABASE_PUBLISHABLE_KEY!\n);\n\nexport const onRequestError = createPulseErrorReporter({\n supabase,\n siteId: \"default\",\n});\n`;\n\nconst IMPORT_LINE = 'import { createPulseErrorReporter } from \"@pulsekit/next\";';\n\nconst EXPORT_BLOCK = `\nexport const onRequestError = createPulseErrorReporter({\n supabase,\n siteId: \"default\",\n});\n`;\n\nexport async function injectInstrumentation(): Promise<void> {\n console.log(\" Setting up error reporting instrumentation...\\n\");\n\n const appDir = getAppDir();\n // src/app -> src/instrumentation.ts; app/ -> instrumentation.ts\n const useSrc = appDir.includes(path.join(\"src\", \"app\"));\n const baseDir = useSrc ? path.join(process.cwd(), \"src\") : process.cwd();\n\n const candidates = [\n path.join(baseDir, \"instrumentation.ts\"),\n path.join(baseDir, \"instrumentation.js\"),\n ];\n\n let foundPath: string | null = null;\n for (const candidate of candidates) {\n if (fs.existsSync(candidate)) {\n foundPath = candidate;\n break;\n }\n }\n\n // Case 1: No file exists — create it\n if (!foundPath) {\n const targetPath = path.join(baseDir, \"instrumentation.ts\");\n fs.writeFileSync(targetPath, FULL_CONTENT, \"utf8\");\n console.log(` Created: ${path.relative(process.cwd(), targetPath)}\\n`);\n return;\n }\n\n const content = fs.readFileSync(foundPath, \"utf8\");\n\n // Case 3: Already has onRequestError — skip\n if (content.includes(\"onRequestError\")) {\n console.log(\" onRequestError already present in instrumentation. Skipping.\\n\");\n console.log(\" To add PulseKit error reporting manually, add:\\n\");\n console.log(` ${IMPORT_LINE}`);\n console.log(\"\");\n console.log(\" export const onRequestError = createPulseErrorReporter({\");\n console.log(\" supabase,\");\n console.log(' siteId: \"default\",');\n console.log(\" });\\n\");\n return;\n }\n\n // Case 2: File exists without onRequestError — append\n const importRegex = /^import\\s.+$/gm;\n let lastImportEnd = -1;\n let match: RegExpExecArray | null;\n while ((match = importRegex.exec(content)) !== null) {\n lastImportEnd = match.index + match[0].length;\n }\n\n let updated: string;\n if (lastImportEnd === -1) {\n updated = IMPORT_LINE + \"\\n\" + content + EXPORT_BLOCK;\n } else {\n updated =\n content.slice(0, lastImportEnd) +\n \"\\n\" +\n IMPORT_LINE +\n content.slice(lastImportEnd) +\n EXPORT_BLOCK;\n }\n\n fs.writeFileSync(foundPath, updated, \"utf8\");\n console.log(` Modified: ${path.relative(process.cwd(), foundPath)}\\n`);\n}\n","import fs from \"node:fs\";\nimport path from \"node:path\";\n\nconst SQL_MAP: Record<string, string> = JSON.parse(\n process.env.__EMBEDDED_SQL__!\n);\n\nexport function writeMigration(): void {\n console.log(\" Writing database migration...\\n\");\n\n const supabaseDir = path.join(process.cwd(), \"supabase\", \"migrations\");\n fs.mkdirSync(supabaseDir, { recursive: true });\n\n // Combine all SQL files into a single migration\n const files = Object.keys(SQL_MAP).sort();\n const combined = files\n .map((file) => `-- ${file}\\n${SQL_MAP[file]}`)\n .join(\"\\n\\n\");\n\n // Use a fixed timestamp so re-running doesn't create duplicates\n const filename = \"20250101000000_pulse_analytics.sql\";\n const fullPath = path.join(supabaseDir, filename);\n\n if (fs.existsSync(fullPath)) {\n console.log(` Skipped (already exists): supabase/migrations/${filename}\\n`);\n return;\n }\n\n fs.writeFileSync(fullPath, combined, \"utf8\");\n console.log(` Created: supabase/migrations/${filename}\\n`);\n}\n"],"mappings":"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","names":["import_node_fs","import_node_path","fs","path","import_node_fs","import_node_path","path","fs","import_node_fs","import_node_path","path","fs","import_node_fs","import_node_path","path","fs","import_node_fs","import_node_path","path","fs","path","fs"]}
1
+ {"version":3,"sources":["../src/index.ts","../src/detect.ts","../src/install.ts","../src/scaffold.ts","../src/inject.ts","../src/inject-instrumentation.ts","../src/migration.ts"],"sourcesContent":["import fs from \"node:fs\";\nimport path from \"node:path\";\nimport { detectPackageManager, validateNextJsProject } from \"./detect\";\nimport { installPackages } from \"./install\";\nimport { scaffoldFiles } from \"./scaffold\";\nimport { injectPulseTracker } from \"./inject\";\nimport { injectInstrumentation } from \"./inject-instrumentation\";\nimport { writeMigration } from \"./migration\";\n\nfunction appendEnvExample(): void {\n const envExamplePath = path.join(process.cwd(), \".env.example\");\n if (!fs.existsSync(envExamplePath)) return;\n\n const content = fs.readFileSync(envExamplePath, \"utf8\");\n const entries: { key: string; value: string }[] = [\n { key: \"PULSE_SECRET\", value: \"\" },\n { key: \"SUPABASE_SERVICE_ROLE_KEY\", value: \"\" },\n ];\n\n const missing = entries.filter((e) => !content.includes(e.key));\n if (missing.length === 0) return;\n\n const block =\n \"\\n# PulseKit\\n\" +\n missing.map((e) => `${e.key}=${e.value}`).join(\"\\n\") +\n \"\\n\";\n fs.appendFileSync(envExamplePath, block, \"utf8\");\n console.log(\" Updated .env.example with PulseKit variables.\\n\");\n}\n\nasync function main() {\n console.log(\"\\n create-pulsekit\\n\");\n console.log(\" Setting up PulseKit analytics in your Next.js project.\\n\");\n\n const pm = detectPackageManager();\n console.log(` Detected package manager: ${pm}\\n`);\n\n validateNextJsProject();\n\n await installPackages(pm);\n\n scaffoldFiles();\n\n await injectPulseTracker();\n\n await injectInstrumentation();\n\n writeMigration();\n\n appendEnvExample();\n\n console.log(\"\\n Done! PulseKit has been added to your project.\\n\");\n console.log(\" To finish setup:\");\n console.log(\" 1. Add these variables to .env.local:\");\n console.log(\" NEXT_PUBLIC_SUPABASE_URL=<your-supabase-url>\");\n console.log(\" NEXT_PUBLIC_SUPABASE_PUBLISHABLE_KEY=<your-anon-key>\");\n console.log(\" SUPABASE_SERVICE_ROLE_KEY=<your-service-role-key>\");\n console.log(\" PULSE_SECRET=<a-random-string-at-least-16-chars>\");\n console.log(\" 2. Run the database migration:\");\n console.log(\" npx supabase link\");\n console.log(\" npx supabase db push\");\n console.log(\" 3. If your middleware protects routes (e.g. Supabase auth),\");\n console.log(\" allow the PulseKit paths through:\");\n console.log(' !request.nextUrl.pathname.startsWith(\"/api/pulse\")');\n console.log(' !request.nextUrl.pathname.startsWith(\"/admin/analytics\")');\n console.log(\" 4. Start your dev server and visit /admin/analytics\\n\");\n}\n\nmain().catch((err) => {\n console.error(\"\\n Error:\", err.message || err);\n process.exit(1);\n});\n","import fs from \"node:fs\";\nimport path from \"node:path\";\n\nexport type PackageManager = \"npm\" | \"pnpm\" | \"yarn\" | \"bun\";\n\nexport function detectPackageManager(): PackageManager {\n const cwd = process.cwd();\n\n if (\n fs.existsSync(path.join(cwd, \"bun.lock\")) ||\n fs.existsSync(path.join(cwd, \"bun.lockb\"))\n ) {\n return \"bun\";\n }\n if (fs.existsSync(path.join(cwd, \"pnpm-lock.yaml\"))) {\n return \"pnpm\";\n }\n if (fs.existsSync(path.join(cwd, \"yarn.lock\"))) {\n return \"yarn\";\n }\n return \"npm\";\n}\n\nexport function validateNextJsProject(): void {\n const cwd = process.cwd();\n\n const pkgPath = path.join(cwd, \"package.json\");\n if (!fs.existsSync(pkgPath)) {\n throw new Error(\n \"No package.json found. Run this command from the root of a Next.js project.\"\n );\n }\n\n const pkg = JSON.parse(fs.readFileSync(pkgPath, \"utf8\"));\n const allDeps = { ...pkg.dependencies, ...pkg.devDependencies };\n if (!allDeps[\"next\"]) {\n throw new Error(\n \"next is not listed in package.json. This command requires a Next.js project.\"\n );\n }\n\n if (!fs.existsSync(getAppDir())) {\n throw new Error(\n \"No app/ directory found. PulseKit requires the Next.js App Router.\"\n );\n }\n}\n\nexport function getAppDir(): string {\n const cwd = process.cwd();\n const srcAppDir = path.join(cwd, \"src\", \"app\");\n if (fs.existsSync(srcAppDir)) {\n return srcAppDir;\n }\n return path.join(cwd, \"app\");\n}\n","import { execSync } from \"node:child_process\";\nimport type { PackageManager } from \"./detect\";\n\nconst PACKAGES = [\n \"@pulsekit/core\",\n \"@pulsekit/next\",\n \"@pulsekit/react\",\n \"@supabase/supabase-js@latest\",\n];\n\nexport async function installPackages(pm: PackageManager): Promise<void> {\n console.log(\" Installing packages...\\n\");\n\n const commands: Record<PackageManager, string> = {\n npm: `npm install ${PACKAGES.join(\" \")}`,\n pnpm: `pnpm add ${PACKAGES.join(\" \")}`,\n yarn: `yarn add ${PACKAGES.join(\" \")}`,\n bun: `bun add ${PACKAGES.join(\" \")}`,\n };\n\n const cmd = commands[pm];\n console.log(` > ${cmd}\\n`);\n\n try {\n execSync(cmd, { cwd: process.cwd(), stdio: \"inherit\" });\n } catch {\n throw new Error(\n `Package installation failed. Try running manually:\\n ${cmd}`\n );\n }\n}\n","import fs from \"node:fs\";\nimport path from \"node:path\";\nimport { getAppDir } from \"./detect\";\n\nexport function scaffoldFiles(): void {\n console.log(\" Scaffolding files...\\n\");\n\n const appDir = getAppDir();\n\n const pulseRoute = `import { createPulseHandler } from \"@pulsekit/next\";\nimport { createClient } from \"@supabase/supabase-js\";\n\nconst supabase = createClient(\n process.env.NEXT_PUBLIC_SUPABASE_URL!,\n process.env.NEXT_PUBLIC_SUPABASE_PUBLISHABLE_KEY!\n);\n\nexport const POST = createPulseHandler({\n supabase,\n config: {\n siteId: \"default\",\n secret: process.env.PULSE_SECRET,\n },\n});\n`;\n\n const refreshRoute = `import { createRefreshHandler, withPulseAuth } from \"@pulsekit/next\";\nimport { createClient } from \"@supabase/supabase-js\";\n\nconst supabase = createClient(\n process.env.NEXT_PUBLIC_SUPABASE_URL!,\n process.env.NEXT_PUBLIC_SUPABASE_PUBLISHABLE_KEY!\n);\n\nexport const POST = withPulseAuth(createRefreshHandler({ supabase }));\n`;\n\n const consolidateRoute = `import { createConsolidateHandler, withPulseAuth } from \"@pulsekit/next\";\nimport { createClient } from \"@supabase/supabase-js\";\n\nconst supabase = createClient(\n process.env.NEXT_PUBLIC_SUPABASE_URL!,\n process.env.NEXT_PUBLIC_SUPABASE_PUBLISHABLE_KEY!\n);\n\nexport const POST = withPulseAuth(createConsolidateHandler({ supabase }));\n`;\n\n const authRoute = `import { createPulseAuthHandler } from \"@pulsekit/next\";\n\nconst handler = createPulseAuthHandler({ secret: process.env.PULSE_SECRET! });\n\nexport const POST = handler;\nexport const DELETE = handler;\n`;\n\n const dashboardPage = `import { Suspense } from \"react\";\nimport { createClient } from \"@supabase/supabase-js\";\nimport { PulseDashboard, PulseAuthGate } from \"@pulsekit/react\";\nimport { getPulseTimezone } from \"@pulsekit/next\";\nimport { Spinner } from \"@/components/ui/spinner\";\nimport \"@pulsekit/react/pulse.css\";\n\nconst supabase = createClient(\n process.env.NEXT_PUBLIC_SUPABASE_URL!,\n process.env.NEXT_PUBLIC_SUPABASE_PUBLISHABLE_KEY!\n);\n\nasync function Dashboard() {\n const timezone = await getPulseTimezone();\n\n return (\n <PulseDashboard\n supabase={supabase}\n siteId=\"default\"\n timeframe=\"7d\"\n timezone={timezone}\n />\n );\n}\n\nexport default function AnalyticsPage() {\n return (\n <Suspense fallback={<div style={{ display: \"flex\", alignItems: \"center\", justifyContent: \"center\", minHeight: \"100vh\", padding: \"1.5rem\" }}><Spinner style={{ width: 24, height: 24 }} /></div>}>\n <PulseAuthGate secret={process.env.PULSE_SECRET!}>\n <Dashboard />\n </PulseAuthGate>\n </Suspense>\n );\n}\n`;\n\n const trackerWrapper = `import { PulseTracker } from \"@pulsekit/next/client\";\nimport { createPulseIngestionToken } from \"@pulsekit/next\";\nimport { connection } from \"next/server\";\n\nexport default async function PulseTrackerWrapper() {\n await connection();\n const token = process.env.PULSE_SECRET\n ? await createPulseIngestionToken(process.env.PULSE_SECRET)\n : undefined;\n\n return (\n <PulseTracker\n excludePaths={[\"/admin/analytics\"]}\n token={token}\n />\n );\n}\n`;\n\n // Scaffold the spinner component if not already present\n const spinnerContent = `function Spinner({ style, ...props }: React.ComponentProps<\"svg\">) {\n return (\n <>\n <style>{\\`@keyframes pulsekit-spin { to { transform: rotate(360deg) } }\\`}</style>\n <svg\n role=\"status\"\n aria-label=\"Loading\"\n width=\"16\"\n height=\"16\"\n xmlns=\"http://www.w3.org/2000/svg\"\n fill=\"none\"\n viewBox=\"0 0 24 24\"\n style={{ animation: \"pulsekit-spin 1s linear infinite\", ...style }}\n {...props}\n >\n <circle opacity={0.25} cx=\"12\" cy=\"12\" r=\"10\" stroke=\"currentColor\" strokeWidth=\"4\" />\n <path opacity={0.75} fill=\"currentColor\" d=\"M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4z\" />\n </svg>\n </>\n );\n}\n\nexport { Spinner }\n`;\n\n const cwd = process.cwd();\n const componentsBase = appDir.includes(path.join(\"src\", \"app\"))\n ? path.join(cwd, \"src\", \"components\", \"ui\")\n : path.join(cwd, \"components\", \"ui\");\n\n const spinnerPath = path.join(componentsBase, \"spinner.tsx\");\n fs.mkdirSync(componentsBase, { recursive: true });\n if (fs.existsSync(spinnerPath)) {\n console.log(\" Skipped (already exists): components/ui/spinner.tsx\");\n } else {\n fs.writeFileSync(spinnerPath, spinnerContent, \"utf8\");\n console.log(\" Created: components/ui/spinner.tsx\");\n }\n\n // Scaffold the tracker wrapper component\n const componentsDir = appDir.includes(path.join(\"src\", \"app\"))\n ? path.join(cwd, \"src\", \"components\")\n : path.join(cwd, \"components\");\n const wrapperPath = path.join(componentsDir, \"pulse-tracker-wrapper.tsx\");\n fs.mkdirSync(componentsDir, { recursive: true });\n if (fs.existsSync(wrapperPath)) {\n console.log(\" Skipped (already exists): components/pulse-tracker-wrapper.tsx\");\n } else {\n fs.writeFileSync(wrapperPath, trackerWrapper, \"utf8\");\n console.log(\" Created: components/pulse-tracker-wrapper.tsx\");\n }\n\n const files = [\n { rel: \"api/pulse/route.ts\", content: pulseRoute },\n { rel: \"api/pulse/auth/route.ts\", content: authRoute },\n { rel: \"api/pulse/refresh-aggregates/route.ts\", content: refreshRoute },\n { rel: \"api/pulse/consolidate/route.ts\", content: consolidateRoute },\n { rel: \"admin/analytics/page.tsx\", content: dashboardPage },\n ];\n\n for (const { rel, content } of files) {\n const fullPath = path.join(appDir, rel);\n fs.mkdirSync(path.dirname(fullPath), { recursive: true });\n\n if (fs.existsSync(fullPath)) {\n console.log(` Skipped (already exists): ${rel}`);\n continue;\n }\n\n fs.writeFileSync(fullPath, content, \"utf8\");\n console.log(` Created: ${rel}`);\n }\n\n console.log(\"\");\n}\n","import fs from \"node:fs\";\nimport path from \"node:path\";\nimport { getAppDir } from \"./detect\";\n\nexport async function injectPulseTracker(): Promise<void> {\n console.log(\" Injecting PulseTracker into layout...\\n\");\n\n const appDir = getAppDir();\n\n const candidates = [\n path.join(appDir, \"layout.tsx\"),\n path.join(appDir, \"layout.jsx\"),\n path.join(appDir, \"layout.js\"),\n ];\n\n let foundPath: string | null = null;\n for (const candidate of candidates) {\n if (fs.existsSync(candidate)) {\n foundPath = candidate;\n break;\n }\n }\n\n if (!foundPath) {\n printManualInstructions();\n return;\n }\n\n let content = fs.readFileSync(foundPath, \"utf8\");\n\n if (content.includes(\"PulseTrackerWrapper\") || content.includes(\"PulseTracker\")) {\n console.log(\" PulseTracker already present in layout. Skipping.\\n\");\n return;\n }\n\n // Add imports after the last existing import\n const importStatements = [\n 'import { Suspense } from \"react\";',\n 'import PulseTrackerWrapper from \"@/components/pulse-tracker-wrapper\";',\n ];\n\n const importRegex = /^import\\s.+$/gm;\n let lastImportEnd = -1;\n let match: RegExpExecArray | null;\n while ((match = importRegex.exec(content)) !== null) {\n lastImportEnd = match.index + match[0].length;\n }\n\n // Only add imports that aren't already present\n const newImports = importStatements.filter((stmt) => !content.includes(stmt));\n const importBlock = newImports.join(\"\\n\");\n\n if (importBlock) {\n if (lastImportEnd === -1) {\n content = importBlock + \"\\n\" + content;\n } else {\n content =\n content.slice(0, lastImportEnd) +\n \"\\n\" +\n importBlock +\n content.slice(lastImportEnd);\n }\n }\n\n // Inject <Suspense><PulseTrackerWrapper /></Suspense> before </body>\n const bodyCloseIndex = content.lastIndexOf(\"</body>\");\n if (bodyCloseIndex === -1) {\n printManualInstructions();\n return;\n }\n\n // Detect indentation from the </body> line\n const lineStart = content.lastIndexOf(\"\\n\", bodyCloseIndex) + 1;\n const indent = content.slice(lineStart, bodyCloseIndex).match(/^\\s*/)?.[0] ?? \" \";\n const trackerJsx = [\n `${indent} <Suspense>`,\n `${indent} <PulseTrackerWrapper />`,\n `${indent} </Suspense>`,\n ].join(\"\\n\") + \"\\n\";\n\n content =\n content.slice(0, lineStart) + trackerJsx + content.slice(lineStart);\n\n fs.writeFileSync(foundPath, content, \"utf8\");\n console.log(\n ` Modified: ${path.relative(process.cwd(), foundPath)}\\n`\n );\n}\n\nfunction printManualInstructions(): void {\n console.log(\n \" Could not auto-inject PulseTracker. Add it manually to your layout:\\n\"\n );\n console.log(' import { Suspense } from \"react\";');\n console.log(' import PulseTrackerWrapper from \"@/components/pulse-tracker-wrapper\";');\n console.log(\"\");\n console.log(\" // Add inside your <body> tag:\");\n console.log(\" <Suspense>\");\n console.log(\" <PulseTrackerWrapper />\");\n console.log(\" </Suspense>\\n\");\n}\n","import fs from \"node:fs\";\nimport path from \"node:path\";\nimport { getAppDir } from \"./detect\";\n\nconst FULL_CONTENT = `import { createClient } from \"@supabase/supabase-js\";\nimport { createPulseErrorReporter } from \"@pulsekit/next\";\n\nconst supabase = createClient(\n process.env.NEXT_PUBLIC_SUPABASE_URL!,\n process.env.NEXT_PUBLIC_SUPABASE_PUBLISHABLE_KEY!\n);\n\nexport const onRequestError = createPulseErrorReporter({\n supabase,\n siteId: \"default\",\n});\n`;\n\nconst IMPORT_LINE = 'import { createPulseErrorReporter } from \"@pulsekit/next\";';\n\nconst EXPORT_BLOCK = `\nexport const onRequestError = createPulseErrorReporter({\n supabase,\n siteId: \"default\",\n});\n`;\n\nexport async function injectInstrumentation(): Promise<void> {\n console.log(\" Setting up error reporting instrumentation...\\n\");\n\n const appDir = getAppDir();\n // src/app -> src/instrumentation.ts; app/ -> instrumentation.ts\n const useSrc = appDir.includes(path.join(\"src\", \"app\"));\n const baseDir = useSrc ? path.join(process.cwd(), \"src\") : process.cwd();\n\n const candidates = [\n path.join(baseDir, \"instrumentation.ts\"),\n path.join(baseDir, \"instrumentation.js\"),\n ];\n\n let foundPath: string | null = null;\n for (const candidate of candidates) {\n if (fs.existsSync(candidate)) {\n foundPath = candidate;\n break;\n }\n }\n\n // Case 1: No file exists — create it\n if (!foundPath) {\n const targetPath = path.join(baseDir, \"instrumentation.ts\");\n fs.writeFileSync(targetPath, FULL_CONTENT, \"utf8\");\n console.log(` Created: ${path.relative(process.cwd(), targetPath)}\\n`);\n return;\n }\n\n const content = fs.readFileSync(foundPath, \"utf8\");\n\n // Case 3: Already has onRequestError — skip\n if (content.includes(\"onRequestError\")) {\n console.log(\" onRequestError already present in instrumentation. Skipping.\\n\");\n console.log(\" To add PulseKit error reporting manually, add:\\n\");\n console.log(` ${IMPORT_LINE}`);\n console.log(\"\");\n console.log(\" export const onRequestError = createPulseErrorReporter({\");\n console.log(\" supabase,\");\n console.log(' siteId: \"default\",');\n console.log(\" });\\n\");\n return;\n }\n\n // Case 2: File exists without onRequestError — append\n const importRegex = /^import\\s.+$/gm;\n let lastImportEnd = -1;\n let match: RegExpExecArray | null;\n while ((match = importRegex.exec(content)) !== null) {\n lastImportEnd = match.index + match[0].length;\n }\n\n let updated: string;\n if (lastImportEnd === -1) {\n updated = IMPORT_LINE + \"\\n\" + content + EXPORT_BLOCK;\n } else {\n updated =\n content.slice(0, lastImportEnd) +\n \"\\n\" +\n IMPORT_LINE +\n content.slice(lastImportEnd) +\n EXPORT_BLOCK;\n }\n\n fs.writeFileSync(foundPath, updated, \"utf8\");\n console.log(` Modified: ${path.relative(process.cwd(), foundPath)}\\n`);\n}\n","import fs from \"node:fs\";\nimport path from \"node:path\";\n\nconst SQL_MAP: Record<string, string> = JSON.parse(\n process.env.__EMBEDDED_SQL__!\n);\n\nexport function writeMigration(): void {\n console.log(\" Writing database migration...\\n\");\n\n const supabaseDir = path.join(process.cwd(), \"supabase\", \"migrations\");\n fs.mkdirSync(supabaseDir, { recursive: true });\n\n // Combine all SQL files into a single migration\n const files = Object.keys(SQL_MAP).sort();\n const combined = files\n .map((file) => `-- ${file}\\n${SQL_MAP[file]}`)\n .join(\"\\n\\n\");\n\n // Use a fixed timestamp so re-running doesn't create duplicates\n const filename = \"20250101000000_pulse_analytics.sql\";\n const fullPath = path.join(supabaseDir, filename);\n\n if (fs.existsSync(fullPath)) {\n console.log(` Skipped (already exists): supabase/migrations/${filename}\\n`);\n return;\n }\n\n fs.writeFileSync(fullPath, combined, \"utf8\");\n console.log(` Created: supabase/migrations/${filename}\\n`);\n}\n"],"mappings":"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","names":["import_node_fs","import_node_path","fs","path","import_node_fs","import_node_path","path","fs","import_node_fs","import_node_path","path","fs","import_node_fs","import_node_path","path","fs","import_node_fs","import_node_path","path","fs","path","fs"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "create-pulsekit",
3
- "version": "1.0.1",
3
+ "version": "1.0.5",
4
4
  "description": "Set up PulseKit analytics in your Next.js project",
5
5
  "keywords": [
6
6
  "analytics",