generatesaas 1.19.1 → 1.20.0
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 +242 -222
- package/dist/skill/content/scripts/_helpers.js +13 -6
- package/package.json +1 -1
package/dist/index.js
CHANGED
|
@@ -1,15 +1,15 @@
|
|
|
1
1
|
#!/usr/bin/env node
|
|
2
|
-
import{Command as Zs}from"commander";import*as ro from"@clack/prompts";import{existsSync as hs,readdirSync as ys,rmSync as vs}from"fs";import{join as Ss,resolve as ks}from"path";import{Option as Y}from"commander";import*as w from"@clack/prompts";import*as at from"@clack/prompts";import Dt from"picocolors";function ar(e){let t=e?` GenerateSaaS v${e} `:" GenerateSaaS ";at.intro(Dt.bgYellow(Dt.black(t)))}function cr(){at.outro(Dt.yellow("Happy building!"))}import*as g from"@clack/prompts";import y from"picocolors";var qe={nextjs:{label:"Next.js",hint:"React 19 + Next.js 16"},nuxt:{label:"Nuxt",hint:"Vue 3 + Nuxt 4"}},Je={fullstack:{label:"Fullstack",hint:"Frontend hosts the API - works on serverless or long-running"},separate:{label:"Separate",hint:"Standalone Hono backend - long-running runtimes only (Docker, Render, Fly.io, Railway, Coolify, Dokploy)"}},ct={stripe:{label:"Stripe"},polar:{label:"Polar"},none:{label:"None",hint:"disable payments"}},lt={smtp:{label:"SMTP",hint:"Mailpit for local dev"},ses:{label:"Amazon SES"},resend:{label:"Resend"}},pt={user:{label:"Per user",hint:"each user has their own subscription"},organization:{label:"Per organization",hint:"org subscription shared by members"}},Te={postgres:{label:"PostgreSQL",hint:"port 5432",port:5432},redis:{label:"Redis",hint:"port 6379",port:6379},inngest:{label:"Inngest",hint:"port 8288",port:8288},mailpit:{label:"Mailpit",hint:"port 1025",port:1025}},We={"claude-code":{label:"Claude Code"},cursor:{label:"Cursor"},codex:{label:"Codex"},"gemini-cli":{label:"Gemini CLI"},windsurf:{label:"Windsurf"}};var he={USD:{symbol:"$",name:"US Dollar",place:"left",space:!1},EUR:{symbol:"\u20AC",name:"Euro",place:"right",space:!1},GBP:{symbol:"\xA3",name:"British Pound",place:"left",space:!1},CAD:{symbol:"CA$",name:"Canadian Dollar",place:"left",space:!1},AUD:{symbol:"A$",name:"Australian Dollar",place:"left",space:!1},BRL:{symbol:"R$",name:"Brazilian Real",place:"left",space:!1},JPY:{symbol:"\xA5",name:"Japanese Yen",place:"left",space:!1}};var Q={node:{label:"Node.js / Docker",hint:"long-running runtime - Render, Fly.io, Railway, Coolify, Dokploy, VPS",edgeRuntime:!1},vercel:{label:"Vercel",hint:"serverless functions",edgeRuntime:!0}},V={postgres:{label:"PostgreSQL (self-hosted)",hint:"local Docker, drizzle-orm/node-postgres",managed:!1,envVars:[{key:"DATABASE_URL",defaultValue:"postgres://postgres:postgres@localhost:5432/saas"}]},neon:{label:"Neon",hint:"serverless Postgres",managed:!0,envVars:[{key:"DATABASE_URL",comment:"# TODO: Add your Neon connection string"}]},supabase:{label:"Supabase",hint:"managed Postgres",managed:!0,envVars:[{key:"DATABASE_URL",comment:"# TODO: Add your Supabase connection string"}]}},ee={redis:{label:"Redis (self-hosted)",hint:"local Docker, ioredis",managed:!1,envVars:[{key:"REDIS_URL",defaultValue:"redis://localhost:6379"}]},upstash:{label:"Upstash",hint:"serverless Redis",managed:!0,envVars:[{key:"UPSTASH_REDIS_REST_URL",comment:"# TODO: Add your Upstash REST URL"},{key:"UPSTASH_REDIS_REST_TOKEN",comment:"# TODO: Add your Upstash REST token"}]}},Xe=[{target:"vercel",provider:"redis",reason:"Vercel serverless cannot maintain persistent Redis connections. Consider Upstash."}],Ze=[{architecture:"separate",target:"vercel",reason:"Standalone backends need a long-running runtime. Vercel's per-function TypeScript check conflicts with pnpm's isolated install. Use --architecture fullstack for serverless, or deploy the standalone backend to a long-running runtime (Render, Fly.io, Railway, Coolify, Dokploy, or your own VPS via Docker)."}],lr=["Local file storage (sharp, geoip-lite)","SMTP email (use Resend or SES instead)","Content API git integration"];function dt(e){let t=V[e.databaseProvider].managed,r=ee[e.cacheProvider].managed;return e.dockerServices.some(n=>!(n==="postgres"&&t||n==="redis"&&r))}var ne={google:{label:"Google",envVars:[{name:"GOOGLE_CLIENT_ID",secret:!1},{name:"GOOGLE_CLIENT_SECRET",secret:!0}]},github:{label:"GitHub",envVars:[{name:"GITHUB_CLIENT_ID",secret:!1},{name:"GITHUB_CLIENT_SECRET",secret:!0}]},facebook:{label:"Facebook",envVars:[{name:"FACEBOOK_CLIENT_ID",secret:!1},{name:"FACEBOOK_CLIENT_SECRET",secret:!0}]},discord:{label:"Discord",envVars:[{name:"DISCORD_CLIENT_ID",secret:!1},{name:"DISCORD_CLIENT_SECRET",secret:!0}]},x:{label:"X",envVars:[{name:"TWITTER_CLIENT_ID",secret:!1},{name:"TWITTER_CLIENT_SECRET",secret:!0}]}};var ye=["nextjs","nuxt"],Re=["fullstack","separate"],_e=["stripe","polar","none"],Oe=["smtp","ses","resend"],xe=["postgres","redis","inngest","mailpit"],De=["claude-code","cursor","codex","gemini-cli","windsurf"],Ce=["user","organization"],oe=["USD","EUR","GBP","CAD","AUD","BRL","JPY"],ie=["node","vercel"],se=["postgres","neon","supabase"],ae=["redis","upstash"],Ne=["google","github","facebook","discord","x"];function ut(e){return e.split("-").map(t=>t.charAt(0).toUpperCase()+t.slice(1)).join(" ")}function mt(e){return/^[a-z][a-z0-9-]*$/.test(e)}function O(e){return e instanceof Error?e.message:String(e)}function I(e){g.isCancel(e)&&(g.cancel("Setup cancelled."),process.exit(0))}function Ct(e){let t=[];e.databaseProvider==="neon"&&t.push({key:"DATABASE_URL",message:"Neon connection string (optional):",placeholder:"postgres://...",secret:!0}),e.databaseProvider==="supabase"&&t.push({key:"DATABASE_URL",message:"Supabase connection string (optional):",placeholder:"postgres://...",secret:!0}),e.cacheProvider==="upstash"&&(t.push({key:"UPSTASH_REDIS_REST_URL",message:"Upstash REST URL (optional):",placeholder:"https://...",secret:!1}),t.push({key:"UPSTASH_REDIS_REST_TOKEN",message:"Upstash REST token (optional):",secret:!0})),e.paymentProvider==="stripe"&&(t.push({key:"STRIPE_SECRET_KEY",message:"Stripe secret key (optional):",placeholder:"sk_test_...",secret:!0}),t.push({key:"STRIPE_WEBHOOK_SECRET",message:"Stripe webhook secret (optional):",placeholder:"whsec_...",secret:!0})),e.paymentProvider==="polar"&&(t.push({key:"POLAR_ACCESS_TOKEN",message:"Polar access token (optional):",secret:!0}),t.push({key:"POLAR_WEBHOOK_SECRET",message:"Polar webhook secret (optional):",secret:!0})),e.emailProvider==="resend"&&t.push({key:"RESEND_API_KEY",message:"Resend API key (optional):",placeholder:"re_...",secret:!0}),e.emailProvider==="ses"&&(t.push({key:"AMAZON_SES_REGION",message:"Amazon SES region (optional):",placeholder:"us-east-1",secret:!1}),t.push({key:"AMAZON_SES_KEY",message:"Amazon SES access key (optional):",secret:!0}),t.push({key:"AMAZON_SES_SECRET",message:"Amazon SES secret (optional):",secret:!0}));for(let r of e.socialProviders){let n=ne[r];for(let o of n.envVars)t.push({key:o.name,message:`${o.name} (${n.label}, optional):`,secret:o.secret})}return e.demo&&t.push({key:"TURNSTILE_SECRET_KEY",message:"Cloudflare Turnstile secret key (optional):",secret:!0}),t}async function pr(e,t){let r=!1;g.log.info(y.bold("Project"));let n=e?.projectName??await(async()=>{r=!0;let c=await g.text({message:"Project name:",placeholder:"my-saas",validate:p=>{if(!p?.trim())return"Project name is required.";if(!mt(p))return"Use lowercase letters, numbers, and hyphens only. Must start with a letter."}});return I(c),c})(),o=e?.appName??await(async()=>{r=!0;let c=await g.text({message:"App name:",initialValue:ut(n),validate:p=>{if(!p?.trim())return"App name is required."}});return I(c),c})(),i=e?.projectDir??await(async()=>{r=!0;let c=await g.text({message:"Project location:",initialValue:`./${n}`});return I(c),c==="."?process.cwd():c})(),s=e?.frontend??await(async()=>{r=!0;let c=Object.keys(qe),p=await g.select({message:"Frontend framework:",options:c.map(k=>({value:k,label:qe[k].label,hint:qe[k].hint}))});return I(p),p})();g.log.info(y.bold("Infrastructure"));let a=e?.deploymentTarget??"node";if(e?.deploymentTarget===void 0){r=!0;let c=await g.select({message:"Deployment target:",options:ie.map(p=>({value:p,label:Q[p].label,hint:Q[p].hint}))});I(c),a=c}let d=e?.architecture?Ze.find(c=>c.architecture===e.architecture&&c.target===a):void 0;if(d)throw new Error(`Incompatible: --architecture ${d.architecture} + --deploy ${d.target}. ${d.reason}`);let u=new Set(Ze.filter(c=>c.target===a).map(c=>c.architecture)),m=Re.filter(c=>!u.has(c)),f=e?.architecture??await(async()=>{if(m.length===1){let p=m[0];return g.log.info(`Auto-selected ${Je[p].label} architecture (only compatible option for ${Q[a].label}).`),p}r=!0;let c=await g.select({message:"Architecture:",options:m.map(p=>({value:p,label:Je[p].label,hint:Je[p].hint}))});return I(c),c})(),v=e?.databaseProvider??await(async()=>{r=!0;let c=se.filter(k=>!Xe.some(U=>U.target===a&&U.provider===k));if(c.length===1){let k=c[0];return g.log.info(`Auto-selected ${V[k].label} (only compatible option for ${Q[a].label}).`),k}let p=await g.select({message:"Database provider:",options:c.map(k=>({value:k,label:V[k].label,hint:V[k].hint}))});return I(p),p})(),E=e?.cacheProvider??await(async()=>{r=!0;let c=ae.filter(k=>!Xe.some(U=>U.target===a&&U.provider===k));if(c.length===1){let k=c[0];return g.log.info(`Auto-selected ${ee[k].label} (only compatible option for ${Q[a].label}).`),k}let p=await g.select({message:"Cache provider:",options:c.map(k=>({value:k,label:ee[k].label,hint:ee[k].hint}))});return I(p),p})();if(Q[a]?.edgeRuntime){let c=lr.map(p=>` - ${p}`).join(`
|
|
3
|
-
`);g.note(c,"Unavailable on edge runtime")}g.log.info(y.bold("Features"));let
|
|
4
|
-
`),
|
|
5
|
-
`),
|
|
6
|
-
`),
|
|
7
|
-
`);
|
|
8
|
-
`),"Summary");let
|
|
9
|
-
`,{mode:384})}async function
|
|
10
|
-
`))}}async function
|
|
11
|
-
`)}function
|
|
12
|
-
- **Payments:** ${
|
|
2
|
+
import{Command as Ts}from"commander";import*as Oo from"@clack/prompts";import{existsSync as Ha,readdirSync as za,rmSync as Ya}from"fs";import{join as qa,resolve as Ja}from"path";import{Option as Q}from"commander";import*as A from"@clack/prompts";import*as ft from"@clack/prompts";import Gt from"picocolors";function xn(e){let t=e?` GenerateSaaS v${e} `:" GenerateSaaS ";ft.intro(Gt.bgYellow(Gt.black(t)))}function Cn(){ft.outro(Gt.yellow("Happy building!"))}import*as g from"@clack/prompts";import y from"picocolors";var et={nextjs:{label:"Next.js",hint:"React 19 + Next.js 16"},nuxt:{label:"Nuxt",hint:"Vue 3 + Nuxt 4"}},tt={fullstack:{label:"Fullstack",hint:"Frontend hosts the API - works on serverless or long-running"},separate:{label:"Separate",hint:"Standalone Hono backend - long-running runtimes only (Docker, Render, Fly.io, Railway, Coolify, Dokploy)"}},gt={stripe:{label:"Stripe"},polar:{label:"Polar"},none:{label:"None",hint:"disable payments"}},ht={smtp:{label:"SMTP",hint:"Mailpit for local dev"},ses:{label:"Amazon SES"},resend:{label:"Resend"}},yt={user:{label:"Per user",hint:"each user has their own subscription"},organization:{label:"Per organization",hint:"org subscription shared by members"}},De={postgres:{label:"PostgreSQL",hint:"port 5432",port:5432},redis:{label:"Redis",hint:"port 6379",port:6379},inngest:{label:"Inngest",hint:"port 8288",port:8288},mailpit:{label:"Mailpit",hint:"port 1025",port:1025}},nt={"claude-code":{label:"Claude Code"},cursor:{label:"Cursor"},codex:{label:"Codex"},"gemini-cli":{label:"Gemini CLI"},windsurf:{label:"Windsurf"}};var Se={USD:{symbol:"$",name:"US Dollar",place:"left",space:!1},EUR:{symbol:"\u20AC",name:"Euro",place:"right",space:!1},GBP:{symbol:"\xA3",name:"British Pound",place:"left",space:!1},CAD:{symbol:"CA$",name:"Canadian Dollar",place:"left",space:!1},AUD:{symbol:"A$",name:"Australian Dollar",place:"left",space:!1},BRL:{symbol:"R$",name:"Brazilian Real",place:"left",space:!1},JPY:{symbol:"\xA5",name:"Japanese Yen",place:"left",space:!1}};var ne={node:{label:"Node.js / Docker",hint:"long-running runtime - Render, Fly.io, Railway, Coolify, Dokploy, VPS",edgeRuntime:!1},vercel:{label:"Vercel",hint:"serverless functions",edgeRuntime:!0}},G={postgres:{label:"PostgreSQL (self-hosted)",hint:"local Docker, drizzle-orm/node-postgres",managed:!1,envVars:[{key:"DATABASE_URL",defaultValue:"postgres://postgres:postgres@localhost:5432/saas"}]},neon:{label:"Neon",hint:"serverless Postgres",managed:!0,envVars:[{key:"DATABASE_URL",comment:"# TODO: Add your Neon connection string"}]},supabase:{label:"Supabase",hint:"managed Postgres",managed:!0,envVars:[{key:"DATABASE_URL",comment:"# TODO: Add your Supabase connection string"}]}},re={redis:{label:"Redis (self-hosted)",hint:"local Docker, ioredis",managed:!1,envVars:[{key:"REDIS_URL",defaultValue:"redis://localhost:6379"}]},upstash:{label:"Upstash",hint:"serverless Redis",managed:!0,envVars:[{key:"UPSTASH_REDIS_REST_URL",comment:"# TODO: Add your Upstash REST URL"},{key:"UPSTASH_REDIS_REST_TOKEN",comment:"# TODO: Add your Upstash REST token"}]}},rt=[{target:"vercel",provider:"redis",reason:"Vercel serverless cannot maintain persistent Redis connections. Consider Upstash."}],ot=[{architecture:"separate",target:"vercel",reason:"Standalone backends need a long-running runtime. Vercel's per-function TypeScript check conflicts with pnpm's isolated install. Use --architecture fullstack for serverless, or deploy the standalone backend to a long-running runtime (Render, Fly.io, Railway, Coolify, Dokploy, or your own VPS via Docker)."}],Dn=["Local file storage (sharp, geoip-lite)","SMTP email (use Resend or SES instead)","Content API git integration"];function vt(e){let t=G[e.databaseProvider].managed,n=re[e.cacheProvider].managed;return e.dockerServices.some(r=>!(r==="postgres"&&t||r==="redis"&&n))}var se={google:{label:"Google",envVars:[{name:"GOOGLE_CLIENT_ID",secret:!1},{name:"GOOGLE_CLIENT_SECRET",secret:!0}]},github:{label:"GitHub",envVars:[{name:"GITHUB_CLIENT_ID",secret:!1},{name:"GITHUB_CLIENT_SECRET",secret:!0}]},facebook:{label:"Facebook",envVars:[{name:"FACEBOOK_CLIENT_ID",secret:!1},{name:"FACEBOOK_CLIENT_SECRET",secret:!0}]},discord:{label:"Discord",envVars:[{name:"DISCORD_CLIENT_ID",secret:!1},{name:"DISCORD_CLIENT_SECRET",secret:!0}]},x:{label:"X",envVars:[{name:"TWITTER_CLIENT_ID",secret:!1},{name:"TWITTER_CLIENT_SECRET",secret:!0}]}};var Ee=["nextjs","nuxt"],Ne=["fullstack","separate"],Le=["stripe","polar","none"],$e=["smtp","ses","resend"],je=["postgres","redis","inngest","mailpit"],Ue=["claude-code","cursor","codex","gemini-cli","windsurf"],Me=["user","organization"],ce=["USD","EUR","GBP","CAD","AUD","BRL","JPY"],le=["node","vercel"],pe=["postgres","neon","supabase"],de=["redis","upstash"],Ve=["google","github","facebook","discord","x"];function kt(e){return e.split("-").map(t=>t.charAt(0).toUpperCase()+t.slice(1)).join(" ")}function wt(e){return/^[a-z][a-z0-9-]*$/.test(e)}function C(e){return e instanceof Error?e.message:String(e)}var Nn=["companion","mcpServer","rag"];function zt(e){if(!Nn.some(n=>e[n]===!0))return e;if(e.ai===!1)throw new Error("--companion / --mcp-server / --rag require AI features; remove --no-ai.");return e.ai===void 0&&(e.ai=!0),e}function Ln(e){return Nn.some(n=>e[n]===!0)&&(e.ai=!0),e}function $n(e){let t={};if(e.name!==void 0){if(!wt(e.name))throw new Error(`Invalid project name "${e.name}". Use lowercase letters, numbers, and hyphens only. Must start with a letter.`);t.projectName=e.name}if(e.appName!==void 0){if(!e.appName.trim())throw new Error("App name cannot be empty.");t.appName=e.appName}if(e.location!==void 0?t.projectDir=e.location==="."?process.cwd():e.location:t.projectName!==void 0&&(t.projectDir=`./${t.projectName}`),e.frontend!==void 0){if(!Ee.includes(e.frontend))throw new Error(`Invalid frontend "${e.frontend}". Valid values: ${Ee.join(", ")}`);t.frontend=e.frontend}if(e.architecture!==void 0&&(t.architecture=e.architecture),e.payment!==void 0&&(t.paymentProvider=e.payment),e.email!==void 0&&(t.emailProvider=e.email),e.org!==void 0&&(t.multiTenancy=e.org),e.billingScope!==void 0){if(e.org!==!0)throw new Error("--billing-scope requires --org to be enabled.");t.billingScope=e.billingScope}if(e.blog!==void 0&&(t.blog=e.blog),e.docs!==void 0&&(t.docs=e.docs),e.desktop!==void 0&&(t.desktop=e.desktop),e.desktopAuto!==void 0&&(t.desktopAutoRelease=e.desktopAuto),e.desktopAi!==void 0){if(e.desktopAi===!0&&e.desktop!==!0)throw new Error("--desktop-ai requires --desktop to be enabled.");t.desktopAi=e.desktopAi}if(e.ai!==void 0&&(t.ai=e.ai),e.rag!==void 0&&(t.rag=e.rag),e.companion!==void 0&&(t.companion=e.companion),e.mcpServer!==void 0&&(t.mcpServer=e.mcpServer),e.aiByok!==void 0&&(t.aiByok=e.aiByok),zt(t),e.revenueSharing!==void 0&&(t.revenueSharing=e.revenueSharing),e.credits!==void 0){if(e.credits===!0&&e.payment==="none")throw new Error("--credits requires a payment provider (got --payment none).");t.credits=e.credits}if(e.docker!==void 0&&(t.dockerServices=Ht(e.docker,je,"docker service")),e.aiTools!==void 0&&(t.aiTools=Ht(e.aiTools,Ue,"AI tool")),e.socialProviders!==void 0&&(t.socialProviders=Ht(e.socialProviders,Ve,"social provider")),e.currency!==void 0){if(!ce.includes(e.currency))throw new Error(`Invalid currency "${e.currency}". Valid values: ${ce.join(", ")}`);t.defaultCurrency=e.currency}if(e.deploy!==void 0){if(!le.includes(e.deploy))throw new Error(`Invalid deployment target "${e.deploy}". Valid values: ${le.join(", ")}`);t.deploymentTarget=e.deploy}if(e.database!==void 0){if(!pe.includes(e.database))throw new Error(`Invalid database provider "${e.database}". Valid values: ${pe.join(", ")}`);t.databaseProvider=e.database}if(e.cache!==void 0){if(!de.includes(e.cache))throw new Error(`Invalid cache provider "${e.cache}". Valid values: ${de.join(", ")}`);t.cacheProvider=e.cache}if(e.demo===!0&&(t.demo=!0),e.baseUrl!==void 0){let n=e.baseUrl.trim();if(n==="")throw new Error("--base-url cannot be empty. Provide an absolute URL like https://example.com.");let r;try{r=new URL(n)}catch{throw new Error(`Invalid --base-url "${e.baseUrl}". Must be an absolute URL like https://example.com.`)}if(r.protocol!=="http:"&&r.protocol!=="https:")throw new Error(`Invalid --base-url "${e.baseUrl}". Must use http or https.`);t.baseUrl=`${r.protocol}//${r.host}`}return t}var ue={projectName:"my-saas",frontend:"nextjs",architecture:"fullstack",paymentProvider:"stripe",emailProvider:"smtp",multiTenancy:!1,billingScope:"user",blog:!0,docs:!1,desktop:!1,desktopAutoRelease:!1,desktopAi:!1,ai:!1,rag:!1,companion:!1,mcpServer:!1,aiByok:!0,revenueSharing:!1,credits:!0,dockerServices:["postgres","redis","inngest"],aiTools:[],socialProviders:[],defaultCurrency:"USD",deploymentTarget:"node",databaseProvider:"postgres",cacheProvider:"redis"};function jn(e){let t=e.projectName??ue.projectName,n=e.projectDir??`./${t}`,r=e.appName??kt(t),o=e.deploymentTarget??ue.deploymentTarget,i=ne[o]?.edgeRuntime??!1,a=e.databaseProvider??(i?"neon":ue.databaseProvider),s=e.cacheProvider??(i?"upstash":ue.cacheProvider),p=e.emailProvider??(i?"resend":ue.emailProvider),f=e.dockerServices??(i?ue.dockerServices.filter(d=>d!=="postgres"&&d!=="redis"):ue.dockerServices),m={...ue,...e,projectName:t,appName:r,projectDir:n,deploymentTarget:o,databaseProvider:a,cacheProvider:s,emailProvider:p,dockerServices:f};m.paymentProvider==="none"&&(m.credits=!1),m.multiTenancy||(m.billingScope="user"),m.ai||(m.aiByok=ue.aiByok);for(let d of rt){if(m.deploymentTarget!==d.target)continue;let v=m.databaseProvider===d.provider?"database":"cache";if(m.databaseProvider===d.provider||m.cacheProvider===d.provider)throw new Error(`Incompatible: --deploy ${d.target} + --${v} ${d.provider}. ${d.reason}`)}for(let d of ot)if(m.architecture===d.architecture&&m.deploymentTarget===d.target)throw new Error(`Incompatible: --architecture ${d.architecture} + --deploy ${d.target}. ${d.reason}`);return m}function Ht(e,t,n){if(e.trim()==="")return[];let r=e.split(",").map(i=>i.trim()).filter(Boolean),o=r.filter(i=>!t.includes(i));if(o.length>0)throw new Error(`Invalid ${n}(s): ${o.join(", ")}. Valid values: ${t.join(", ")}`);return r}function E(e){g.isCancel(e)&&(g.cancel("Setup cancelled."),process.exit(0))}function Yt(e){let t=[];e.databaseProvider==="neon"&&t.push({key:"DATABASE_URL",message:"Neon connection string (optional):",placeholder:"postgres://...",secret:!0}),e.databaseProvider==="supabase"&&t.push({key:"DATABASE_URL",message:"Supabase connection string (optional):",placeholder:"postgres://...",secret:!0}),e.cacheProvider==="upstash"&&(t.push({key:"UPSTASH_REDIS_REST_URL",message:"Upstash REST URL (optional):",placeholder:"https://...",secret:!1}),t.push({key:"UPSTASH_REDIS_REST_TOKEN",message:"Upstash REST token (optional):",secret:!0})),e.paymentProvider==="stripe"&&(t.push({key:"STRIPE_SECRET_KEY",message:"Stripe secret key (optional):",placeholder:"sk_test_...",secret:!0}),t.push({key:"STRIPE_WEBHOOK_SECRET",message:"Stripe webhook secret (optional):",placeholder:"whsec_...",secret:!0})),e.paymentProvider==="polar"&&(t.push({key:"POLAR_ACCESS_TOKEN",message:"Polar access token (optional):",secret:!0}),t.push({key:"POLAR_WEBHOOK_SECRET",message:"Polar webhook secret (optional):",secret:!0})),e.emailProvider==="resend"&&t.push({key:"RESEND_API_KEY",message:"Resend API key (optional):",placeholder:"re_...",secret:!0}),e.emailProvider==="ses"&&(t.push({key:"AMAZON_SES_REGION",message:"Amazon SES region (optional):",placeholder:"us-east-1",secret:!1}),t.push({key:"AMAZON_SES_KEY",message:"Amazon SES access key (optional):",secret:!0}),t.push({key:"AMAZON_SES_SECRET",message:"Amazon SES secret (optional):",secret:!0}));for(let n of e.socialProviders){let r=se[n];for(let o of r.envVars)t.push({key:o.name,message:`${o.name} (${r.label}, optional):`,secret:o.secret})}return e.demo&&t.push({key:"TURNSTILE_SECRET_KEY",message:"Cloudflare Turnstile secret key (optional):",secret:!0}),t}async function Un(e,t){let n=!1;e&&zt(e),g.log.info(y.bold("Project"));let r=e?.projectName??await(async()=>{n=!0;let c=await g.text({message:"Project name:",placeholder:"my-saas",validate:u=>{if(!u?.trim())return"Project name is required.";if(!wt(u))return"Use lowercase letters, numbers, and hyphens only. Must start with a letter."}});return E(c),c})(),o=e?.appName??await(async()=>{n=!0;let c=await g.text({message:"App name:",initialValue:kt(r),validate:u=>{if(!u?.trim())return"App name is required."}});return E(c),c})(),i=e?.projectDir??await(async()=>{n=!0;let c=await g.text({message:"Project location:",initialValue:`./${r}`});return E(c),c==="."?process.cwd():c})(),a=e?.frontend??await(async()=>{n=!0;let c=Object.keys(et),u=await g.select({message:"Frontend framework:",options:c.map(S=>({value:S,label:et[S].label,hint:et[S].hint}))});return E(u),u})();g.log.info(y.bold("Infrastructure"));let s=e?.deploymentTarget??"node";if(e?.deploymentTarget===void 0){n=!0;let c=await g.select({message:"Deployment target:",options:le.map(u=>({value:u,label:ne[u].label,hint:ne[u].hint}))});E(c),s=c}let p=e?.architecture?ot.find(c=>c.architecture===e.architecture&&c.target===s):void 0;if(p)throw new Error(`Incompatible: --architecture ${p.architecture} + --deploy ${p.target}. ${p.reason}`);let f=new Set(ot.filter(c=>c.target===s).map(c=>c.architecture)),m=Ne.filter(c=>!f.has(c)),d=e?.architecture??await(async()=>{if(m.length===1){let u=m[0];return g.log.info(`Auto-selected ${tt[u].label} architecture (only compatible option for ${ne[s].label}).`),u}n=!0;let c=await g.select({message:"Architecture:",options:m.map(u=>({value:u,label:tt[u].label,hint:tt[u].hint}))});return E(c),c})(),v=e?.databaseProvider??await(async()=>{n=!0;let c=pe.filter(S=>!rt.some(F=>F.target===s&&F.provider===S));if(c.length===1){let S=c[0];return g.log.info(`Auto-selected ${G[S].label} (only compatible option for ${ne[s].label}).`),S}let u=await g.select({message:"Database provider:",options:c.map(S=>({value:S,label:G[S].label,hint:G[S].hint}))});return E(u),u})(),b=e?.cacheProvider??await(async()=>{n=!0;let c=de.filter(S=>!rt.some(F=>F.target===s&&F.provider===S));if(c.length===1){let S=c[0];return g.log.info(`Auto-selected ${re[S].label} (only compatible option for ${ne[s].label}).`),S}let u=await g.select({message:"Cache provider:",options:c.map(S=>({value:S,label:re[S].label,hint:re[S].hint}))});return E(u),u})();if(ne[s]?.edgeRuntime){let c=Dn.map(u=>` - ${u}`).join(`
|
|
3
|
+
`);g.note(c,"Unavailable on edge runtime")}g.log.info(y.bold("Features"));let I=e?.paymentProvider??await(async()=>{n=!0;let c=await g.select({message:"Payment provider:",options:Le.map(u=>({value:u,label:gt[u].label,hint:gt[u].hint}))});return E(c),c})(),z=e?.defaultCurrency??await(async()=>{if(I==="none")return"USD";n=!0;let c=await g.select({message:"Default currency:",options:ce.map(u=>({value:u,label:u,hint:Se[u].name}))});return E(c),c})(),L=e?.emailProvider??await(async()=>{n=!0;let c=await g.select({message:"Email provider:",options:$e.map(u=>({value:u,label:ht[u].label,hint:ht[u].hint}))});return E(c),c})(),K=e?.multiTenancy??await(async()=>{n=!0;let c=await g.confirm({message:"Enable multi-tenancy (organizations)?",initialValue:!1});return E(c),c})(),ge=K?e?.billingScope??"user":"user";if(K&&e?.billingScope===void 0){n=!0;let c=await g.select({message:"Billing scope:",options:Me.map(u=>({value:u,label:yt[u].label,hint:yt[u].hint}))});E(c),ge=c}let Y=e?.blog??await(async()=>{n=!0;let c=await g.confirm({message:"Enable blog?",initialValue:!0});return E(c),c})(),k=e?.docs??await(async()=>{n=!0;let c=await g.confirm({message:"Include docs app? (self-hosted Fumadocs documentation site)",initialValue:!1});return E(c),c})(),R=t?.desktopAllowed!==!1,P=e?.desktop??(R?await(async()=>{n=!0;let c=await g.confirm({message:"Include the Electron desktop app? (apps/desktop - cross-platform, device-auth)",initialValue:!1});return E(c),c})():(g.log.info(y.dim("Desktop app: available on the Pro plan and up - skipped.")),!1)),ae=P?e?.desktopAutoRelease??await(async()=>{n=!0;let c=await g.select({message:"Desktop release trigger:",options:[{value:!1,label:"Manual only",hint:"run from the Actions tab (saves CI minutes)"},{value:!0,label:"Automatic",hint:"release on every CI success on main"}],initialValue:!1});return E(c),c})():!1,$=e?.ai??await(async()=>{n=!0;let c=await g.confirm({message:"Add AI features? (chat, models, schedules, integrations - and unlocks the companion + MCP server)",initialValue:!1});return E(c),c})(),T=$?e?.rag??await(async()=>{n=!0;let c=await g.confirm({message:"Add server-side RAG knowledge (pgvector embeddings + semantic search)?",initialValue:!1});return E(c),c})():!1,we=$?e?.companion??await(async()=>{n=!0;let c=await g.confirm({message:"Add the companion daemon? A small program your users install on their own machine or VPS to run their Claude Code / Codex / OpenCode subscription for this app 24/7 - the AI runs on their hardware, not yours.",initialValue:!1});return E(c),c})():!1,Pn=$?e?.mcpServer??await(async()=>{n=!0;let c=await g.confirm({message:"Add the outward MCP server? Lets external agents (Claude Code, Codex, Hermes, OpenClaw) control the app on a user's behalf over an authenticated MCP endpoint.",initialValue:!1});return E(c),c})():!1,Tn=$?e?.aiByok??await(async()=>{n=!0;let c=await g.confirm({message:"AI BYOK mode? (end-users bring their own provider key; no = operator-keyed, credit-metered)",initialValue:!0});return E(c),c})():!0,Rn=e?.revenueSharing??await(async()=>{n=!0;let c=await g.confirm({message:"Enable revenue sharing? (opt-in MRR leaderboard with dofollow backlinks)",initialValue:!1});return E(c),c})(),_n=I==="none"?!1:e?.credits??await(async()=>{n=!0;let c=await g.confirm({message:"Enable credits? (metered usage on top of subscription plans)",initialValue:!0});return E(c),c})(),ut=e?.socialProviders??await(async()=>{n=!0;let c=Ve.map(S=>({value:S,label:se[S].label,hint:`requires ${se[S].envVars.map(F=>F.name).join(" / ")}`})),u=await g.multiselect({message:"Which social login providers should the sign-in screen show?",options:c,initialValues:[],required:!1});return E(u),u})();g.log.info(y.bold("Tooling"));let Ft=e?.dockerServices??await(async()=>{n=!0;let c=[...je].filter(B=>B!=="mailpit");L==="smtp"&&c.push("mailpit");let u=c.map(B=>({value:B,label:De[B].label,hint:De[B].hint})),S=u.map(B=>B.value).filter(B=>!(B==="postgres"&&(v==="neon"||v==="supabase")||B==="redis"&&b==="upstash")),F=await g.multiselect({message:"Which services should we set up in Docker for you?",options:u,initialValues:S,required:!1});return E(F),F})(),Bt=e?.aiTools??await(async()=>{n=!0;let c=Ue.map(S=>({value:S,label:nt[S].label})),u=await g.multiselect({message:"Which AI coding tools do you use?",options:c,initialValues:[],required:!1});return E(u),u})(),Kt=e?.demo,mt=Yt({databaseProvider:v,cacheProvider:b,paymentProvider:I,emailProvider:L,socialProviders:ut,demo:Kt}),Qe={};if(mt.length>0&&n){g.log.info(y.bold("Credentials")+y.dim(" all optional - press Enter to skip, fill in .env later"));for(let c of mt)if(n=!0,c.secret){let u=await g.password({message:c.message,mask:"*"});E(u),typeof u=="string"&&u.trim()&&(Qe[c.key]=u.trim())}else{let u=await g.text({message:c.message,placeholder:c.placeholder});E(u),typeof u=="string"&&u.trim()&&(Qe[c.key]=u.trim())}}if(n){let c=[` Name: ${y.cyan(r)}`,` App name: ${y.cyan(o)}`,` Location: ${y.cyan(i)}`,` Frontend: ${y.cyan(et[a].label)}`,` Architecture: ${y.cyan(tt[d].label)}`].join(`
|
|
4
|
+
`),u=[` Deploy target: ${y.cyan(ne[s]?.label??"Node.js / Docker")}`,` Database: ${y.cyan(G[v].label)}`,` Cache: ${y.cyan(re[b].label)}`,Ft.length>0?` Docker: ${y.cyan(Ft.map(be=>De[be].label).join(", "))}`:` Docker: ${y.dim("none")}`].filter(Boolean).join(`
|
|
5
|
+
`),S=[I!=="none"?` Payment: ${y.cyan(gt[I].label)} (${z})`:` Payment: ${y.dim("none")}`,` Credits: ${_n?y.cyan("Yes"):y.dim("No")}`,` AI features: ${$?y.cyan("Yes"):y.dim("No")}`,...$?[` AI mode: ${y.cyan(Tn?"BYOK":"Operator-keyed (credits)")}`,` RAG knowledge: ${T?y.cyan("Yes"):y.dim("No")}`,` Companion: ${we?y.cyan("Yes"):y.dim("No")}`,` MCP server: ${Pn?y.cyan("Yes"):y.dim("No")}`]:[],` Email: ${y.cyan(ht[L].label)}`,` Multi-tenancy: ${K?y.cyan("Yes")+` (billing: ${yt[ge].label})`:y.dim("No")}`,` Blog: ${Y?y.cyan("Yes"):y.dim("No")}`,` Docs app: ${k?y.cyan("Yes"):y.dim("No")}`,` Desktop app: ${P?y.cyan("Yes"):y.dim("No")}`,...P?[` Releases: ${y.cyan(ae?"Automatic on CI":"Manual only")}`]:[],` Rev. sharing: ${Rn?y.cyan("Yes"):y.dim("No")}`,ut.length>0?` Social login: ${y.cyan(ut.map(be=>se[be].label).join(", "))}`:` Social login: ${y.dim("none")}`,Bt.length>0?` AI tools: ${y.cyan(Bt.map(be=>nt[be].label).join(", "))}`:` AI tools: ${y.dim("none")}`].join(`
|
|
6
|
+
`),F=[y.bold("Project"),c,"",y.bold("Infrastructure"),u,"",y.bold("Features"),S];if(mt.length>0){let be=mt.map(On=>{let xo=Qe[On.key]?y.green("provided"):y.dim("skipped");return` ${On.key}: ${xo}`}).join(`
|
|
7
|
+
`);F.push("",y.bold("Credentials"),be)}g.note(F.join(`
|
|
8
|
+
`),"Summary");let B=await g.confirm({message:"Proceed with these settings?"});(g.isCancel(B)||!B)&&(g.cancel("Setup cancelled."),process.exit(0))}return{projectName:r,appName:o,projectDir:i,frontend:a,architecture:d,deploymentTarget:s,databaseProvider:v,cacheProvider:b,paymentProvider:I,emailProvider:L,multiTenancy:K,billingScope:ge,blog:Y,docs:k,desktop:P,desktopAutoRelease:ae,desktopAi:P?e?.desktopAi??!1:!1,ai:$,rag:T,companion:we,mcpServer:Pn,aiByok:Tn,revenueSharing:Rn,credits:_n,dockerServices:Ft,aiTools:Bt,socialProviders:ut,defaultCurrency:z,...Object.keys(Qe).length>0?{credentials:Qe}:{},...e?.baseUrl!==void 0?{baseUrl:e.baseUrl}:{},...Kt!==void 0?{demo:Kt}:{}}}import{createReadStream as Fo}from"fs";import{mkdir as Bo}from"fs/promises";import{Readable as Ko}from"stream";import{pipeline as zn}from"stream/promises";import{extract as Go}from"tar";import{join as Ae}from"path";import{homedir as Co}from"os";var it=process.env.GENERATESAAS_API_URL??"https://cli.generatesaas.com",q=".generatesaas",he=Ae(q,"manifest.json"),Mn=Ae(q,"hashes.json"),bt=Ae(q,"template-hashes.json"),St=Ae(q,"template"),Vn=Ae(q,"staging"),Fn=Ae(q,"staging.json"),me=Ae(Co(),".generatesaas");var U=class extends Error{constructor(n,r,o){super(r);this.status=n;this.body=o}status;body;name="ApiError"};function fe(e){return{apiKey:e,baseUrl:it}}async function ye(e,t,n){let r=`${e.baseUrl}${t}`,o=await fetch(r,{...n,headers:{...n?.headers,Authorization:`Bearer ${e.apiKey}`,"User-Agent":"generatesaas-cli"}});if(!o.ok){let i,a;try{a=await o.json(),i=a.error??`API ${o.status}: ${t}`}catch{i=`API ${o.status}: ${t}`}throw new U(o.status,i,a)}return o}import{existsSync as Do,readFileSync as No,writeFileSync as Lo,mkdirSync as $o}from"fs";import{dirname as jo}from"path";import*as ve from"@clack/prompts";function at(){if(!Do(me))return null;try{let e=JSON.parse(No(me,"utf-8"));return e.apiKey?e.apiKey:(e.token&&!e.apiKey&&ve.log.warning(`Found old GitHub token in ${me}. Run 'generatesaas init' to set up your API key.`),null)}catch{return null}}function Ie(e){$o(jo(me),{recursive:!0}),Lo(me,JSON.stringify({apiKey:e},null," ")+`
|
|
9
|
+
`,{mode:384})}async function Fe(e){if(e?.apiKey)return e.apiKey;let t=process.env.GENERATESAAS_API_KEY;if(t)return t;let n=at();if(n)return n;if(!e?.prompt)throw new Error("API key not found. Set GENERATESAAS_API_KEY or run 'generatesaas init' to configure.");return st()}async function st(){let e=await ve.text({message:"Enter your GenerateSaaS API key:",placeholder:"gs_live_...",validate:t=>{if(!t?.trim())return"API key is required."}});return ve.isCancel(e)&&(ve.cancel("Setup cancelled."),process.exit(0)),e.trim()}async function ke(e){return process.env.GENERATESAAS_OFFLINE_LICENSE==="1"?{latest:"0.0.0-ci",versions:[{version:"0.0.0-ci",date:new Date().toISOString(),breaking:!1}],entitlements:null}:await(await ye(e,"/versions")).json()}async function qt(e,t){try{return await(await ye(e,`/changelog/${encodeURIComponent(t)}`)).text()}catch(n){if(n instanceof U&&n.status===404)return null;throw n}}async function Bn(e,t){return await(await ye(e,`/skill/${encodeURIComponent(t)}`)).json()}function Et(e){if(!(e instanceof U)||e.status!==403)return null;let t=e.body;return!t||t.code!=="update_window_expired"?null:{message:e.message,lastAllowedVersion:t.lastAllowedVersion??null}}function Kn(e){return{frontend:e.frontend,architecture:e.architecture,deployTarget:e.deploymentTarget,database:e.databaseProvider,cache:e.cacheProvider,payment:e.paymentProvider,email:e.emailProvider,multiTenancy:e.multiTenancy,billingScope:e.billingScope,blog:e.blog,docs:e.docs,desktop:e.desktop,credits:e.credits,revenueSharing:e.revenueSharing,socialProviders:e.socialProviders,aiTools:e.aiTools,currency:e.defaultCurrency}}async function Jt(e,t){return process.env.GENERATESAAS_OFFLINE_LICENSE==="1"?{token:"offline-test-token",licenseId:"offline-test-license-id",installId:t.installId}:await(await ye(e,"/license/sign",{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify(t)})).json()}async function Gn(e,t){return await(await ye(e,"/license/refresh",{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify(t)})).json()}async function Wt(e,t){let n=await fetch(`${e}/license/verify`,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify(t)});if(!n.ok)throw new Error(`Verification service returned ${n.status}`);return await n.json()}async function Hn(e,t,n){let r=await fetch(`${e}/license/inspect`,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${t}`},body:JSON.stringify(n)});if(!r.ok){let o=await r.json().catch(()=>null);throw new Error(o?.error??`Inspect endpoint returned ${r.status}`)}return await r.json()}var Xt=new Set([".git","node_modules",".pnpm-store",".env",".env.test",".turbo",".nuxt",".output",".data","dist",".next",".svelte-kit",".wrangler",".devcontainer","playwright-report","test-results"]),Zt=new Set(["pnpm-lock.yaml"]);function Be(e){if(Qt(e))return!0;for(let t of e.split("/"))if(Zt.has(t))return!0;return!1}var Uo=new Set(["data","mksaas","references","scripts",".cursor",".agents",".codex",".generatesaas",".vscode",".mcp.json","README.md","TODO.md","OVERVIEW.md"]),Mo=["scripts/knowledge-corpus.ts","scripts/knowledge-corpus.test.ts","scripts/knowledge-lint.ts","scripts/knowledge-build.ts","scripts/dev-ai-setup.ts","scripts/docs-lint.ts"],Vo=["docs/superpowers","packages/cli","packages/cli-api","infra/docker-compose.yml",".claude/commands",".claude/skills/web-next-port",".claude/skills/web-next-port-workspace",".claude/settings.local.json",".claude/worktrees"];function Qt(e){let t=e.split("/");for(let n of t)if(Xt.has(n))return!0;if(Mo.includes(e))return!1;if(Uo.has(t[0]))return!0;for(let n of Vo)if(e===n||e.startsWith(n+"/"))return!0;return!1}async function At(e,t,n){await Bo(n,{recursive:!0});let r=process.env.GENERATESAAS_TEMPLATE_TARBALL;if(r){await zn(Fo(r),Yn(n));return}let o=await ye(e,`/template/${encodeURIComponent(t)}`);if(!o.body)throw new Error("Empty response body");let i=Ko.fromWeb(o.body);await zn(i,Yn(n))}function Yn(e){return Go({cwd:e,strip:1,filter:t=>{let n=t.replace(/^[^/]+\//,"");return n?!Qt(n):!0},sync:!1})}import{readdir as Ho,readFile as en,rm as Jn,writeFile as tn}from"fs/promises";import{join as Ke}from"path";var zo=["apps/web-nuxt/public/images/blog","apps/web-next/public/images/blog","packages/content/en/blog","packages/content/ro/blog"];async function Wn(e){await Promise.all(zo.map(t=>Jn(Ke(e,t),{recursive:!0,force:!0})))}var Yo="packages/config/src/blog.ts";async function Xn(e){let t=Ke(e,Yo),n=await en(t,"utf-8"),r=qn(qn(n,"blogCategories",t),"blogAuthors",t);await tn(t,r)}function qn(e,t,n){let r=new RegExp(`(export const ${t}\\b[^=]*=\\s*)\\[[\\s\\S]*?\\];`);if(!r.test(e))throw new Error(`emptyBlogConfig: could not find the \`export const ${t} = [...]\` declaration in ${n}. The boilerplate blog config may have been renamed or restructured; update the CLI strip in cleanup.ts.`);return e.replace(r,"$1[];")}async function Zn(e){let t=Ke(e,"packages/i18n/translations"),n;try{n=await Ho(t)}catch{return}for(let r of n){let o=Ke(t,r,"web.json"),i;try{i=await en(o,"utf-8")}catch{continue}let a=JSON.parse(i);!a.blog||a.blog.categories===void 0||(a.blog.categories={},await tn(o,JSON.stringify(a,null," ")+`
|
|
10
|
+
`))}}async function Qn(e,t){t.includes("claude-code")||await Jn(Ke(e,".claude"),{recursive:!0,force:!0})}async function er(e,t){let n=Ke(e,".claude","settings.json"),r;try{r=await en(n,"utf8")}catch{return}let o=JSON.parse(r);delete o.alwaysThinkingEnabled,delete o.enableAllProjectMcpServers,o.env&&(delete o.env.CLAUDE_CODE_EXPERIMENTAL_AGENT_TEAMS,Object.keys(o.env).length===0&&delete o.env),o.permissions?.allow&&(o.permissions.allow=o.permissions.allow.filter(i=>qo(i,t))),await tn(n,JSON.stringify(o,null," ")+`
|
|
11
|
+
`)}function qo(e,t){return!(e.startsWith("mcp__")||t!=="nuxt"&&e.includes("nuxt"))}import{join as tr}from"path";import{mkdir as Jo,readdir as Wo,rm as Xo,rmdir as Zo,writeFile as Qo}from"fs/promises";import{dirname as It,join as ei,relative as ti,sep as ni}from"path";function Pe(e){return e.split(ni).join("/")}async function Pt(e){await Jo(e,{recursive:!0})}async function l(e,t){await Pt(It(e)),await Qo(e,t,"utf-8")}async function nn(e,t){await Xo(e,{force:!0});let n=It(e);for(;n!==t&&n!==It(n);){try{await Zo(n)}catch{return}n=It(n)}}async function Te(e,t,n){let r=[],o=await Wo(e,{withFileTypes:!0});for(let i of o){let a=ei(e,i.name),s=Pe(ti(t,a));n(s)||(i.isDirectory()?r.push(...await Te(a,t,n)):i.isFile()&&r.push(a))}return r}var ri={postgres:"Postgres",neon:"Neon (managed Postgres)",supabase:"Supabase (managed Postgres)"},oi={resend:"Resend",ses:"Amazon SES",smtp:"SMTP"},ii={redis:"Redis",upstash:"Upstash Redis"},ai={node:"Node.js / Docker",vercel:"Vercel"},si={stripe:"Stripe",polar:"Polar"};function ci(e){let t=e.frontend==="nuxt",n=t?"Nuxt 4":"Next.js 16",r=t?"apps/web-nuxt":"apps/web-next",o=t?"`app/` pages + components + composables":"`app/` routes, `components/`, `lib/`",i=e.architecture==="fullstack",a=i?"(fullstack - Hono API mounted inside the app)":"(separate - standalone Hono backend)",s=i?`Mounted inside \`${r}\`. A standalone \`apps/backend\` is also kept (inert) so you can switch to separate later.`:"Runs from `apps/backend`; the frontend reaches it over HTTP.",p=ii[e.cacheProvider],f=ai[e.deploymentTarget],m=e.paymentProvider==="none"?"":`
|
|
12
|
+
- **Payments:** ${si[e.paymentProvider]}`,d=e.desktop?", and the Electron desktop app under `docs/desktop/`":"",v=t?"`$t('key')` in templates (global helper); `useI18n()` from `vue-i18n` in `<script setup>` for `locale`/`setLocale`/`t()`.":"`useTranslations()` from `next-intl` in components; messages are loaded in `apps/web-next/i18n/request.ts`.",b=t?"**Navigation:** always use `localePath()` for paths (hardcoded paths break non-default locales); `await navigateTo()` in SSR.":"**Navigation:** `next/link` for links; `redirect()` from `next/navigation` for programmatic redirects in Server Components.";return`# AGENTS.md
|
|
13
13
|
|
|
14
14
|
Guidelines for AI coding agents (Claude Code, Codex, Cursor, \u2026) in this project.
|
|
15
15
|
Keep this file tight: add a rule only when it prevents a recurring mistake - too
|
|
@@ -23,7 +23,7 @@ and go find it:
|
|
|
23
23
|
|
|
24
24
|
1. \`docs/${t?"nuxt":"next"}/index.mdx\` - the catalog of every shipped feature and its config flag.
|
|
25
25
|
2. \`packages/config/src/index.ts\` - the flag that turns the feature on/off (check it before rendering).
|
|
26
|
-
3. Search \`packages/\` and \`${
|
|
26
|
+
3. Search \`packages/\` and \`${r}/\` for the name before creating anything.
|
|
27
27
|
|
|
28
28
|
Already built - extend these, never re-implement: auth (email / OAuth / 2FA / passkeys),
|
|
29
29
|
billing & subscriptions, credits, organizations & teams, notifications, email, SMS,
|
|
@@ -35,13 +35,13 @@ flag, route, or translation is the most common and most costly mistake in this r
|
|
|
35
35
|
|
|
36
36
|
## Stack
|
|
37
37
|
|
|
38
|
-
- **Frontend:** ${
|
|
39
|
-
- **API:** Hono, RPC-typed. ${
|
|
40
|
-
- **Database:** Drizzle ORM + ${
|
|
41
|
-
- **Cache + jobs:** ${
|
|
38
|
+
- **Frontend:** ${n} ${a}
|
|
39
|
+
- **API:** Hono, RPC-typed. ${s}
|
|
40
|
+
- **Database:** Drizzle ORM + ${ri[e.databaseProvider]}
|
|
41
|
+
- **Cache + jobs:** ${p} + Inngest
|
|
42
42
|
- **Auth:** Better Auth${m}
|
|
43
|
-
- **Email:** ${
|
|
44
|
-
- **Deploy:** ${
|
|
43
|
+
- **Email:** ${oi[e.emailProvider]}
|
|
44
|
+
- **Deploy:** ${f}
|
|
45
45
|
|
|
46
46
|
## Where things live (extend these - don't reinvent)
|
|
47
47
|
|
|
@@ -51,8 +51,12 @@ flag, route, or translation is the most common and most costly mistake in this r
|
|
|
51
51
|
- \`packages/auth/src/config.ts\` - Better Auth config.
|
|
52
52
|
- \`packages/runtime/src/env.ts\` - the validated (Zod) env schema. New env var \u2192 add it here **and** to \`.env.example\` (the committed reference); set the local value in \`.env\`.
|
|
53
53
|
- \`packages/{payments,mail,sms,storage,notifications}\` - config-gated integrations. Every provider's files stay even when its feature is off, so flipping a flag is enough to enable it.
|
|
54
|
-
- \`${
|
|
55
|
-
- \`docs/${t?"nuxt":"next"}/\` - feature & architecture documentation for this stack (Markdown). Consult it before searching from scratch; it cites real config keys, package names, and file paths you can act on. The GenerateSaaS CLI lives under \`docs/cli/\`${
|
|
54
|
+
- \`${r}\` - the ${n} app (${o}).
|
|
55
|
+
- \`docs/${t?"nuxt":"next"}/\` - feature & architecture documentation for this stack (Markdown). Consult it before searching from scratch; it cites real config keys, package names, and file paths you can act on. The GenerateSaaS CLI lives under \`docs/cli/\`${d}.
|
|
56
|
+
|
|
57
|
+
## Knowledge
|
|
58
|
+
|
|
59
|
+
\`knowledge/\` is the product knowledge that grounds the AI: read-only markdown entries, confidential, never user-facing - injected into the model's context so it answers from your facts. When you add or maintain an entry, FOLLOW THE RULES in \`knowledge/AGENTS.md\` (one concept per entry, frontmatter, \`[[wikilinks]]\`) and run \`pnpm knowledge:lint\` to validate. \`pnpm knowledge:build\` seeds the web search index (gated by \`config.rag.enabled\`); the desktop app self-indexes its bundled copy. See \`docs/${t?"nuxt":"next"}/knowledge.mdx\`.
|
|
56
60
|
|
|
57
61
|
## Code style
|
|
58
62
|
|
|
@@ -70,7 +74,7 @@ flag, route, or translation is the most common and most costly mistake in this r
|
|
|
70
74
|
- **Hono routes:** keep the method chain (\`.get().post()\`) - RPC type inference depends on it. Validate with \`sValidator\` from \`@hono/standard-validator\`.
|
|
71
75
|
- **Feature flags:** respect \`config.*\` - hide/skip a feature when its flag is off (the files stay so you can flip it later).
|
|
72
76
|
- **i18n:** strings live in \`packages/i18n/translations/{locale}/{scope}.json\`; edit \`en/\` only, keep keys generic. ${v} A pre-commit hook runs \`pnpm translate\` (needs \`OPENROUTER_API_KEY\`) to sync other locales from \`en\`; without the key it skips.
|
|
73
|
-
- ${
|
|
77
|
+
- ${b}
|
|
74
78
|
- **Routes:** use \`config.routes.*\`, never hardcoded path strings.
|
|
75
79
|
- **Secrets:** never send them to an external service; generate tokens/QR codes client-side.
|
|
76
80
|
|
|
@@ -81,25 +85,25 @@ In \`@repo/*\` backend packages prefer web-standard APIs: \`crypto.randomUUID()\
|
|
|
81
85
|
## This project
|
|
82
86
|
|
|
83
87
|
Scaffolded from the GenerateSaaS boilerplate. To pull upstream boilerplate updates, ask your agent to **"update my GenerateSaaS project"**. To remove the license heartbeat + manifest: \`pnpm dlx generatesaas eject\`.
|
|
84
|
-
`}async function
|
|
85
|
-
`)}import{join as
|
|
88
|
+
`}async function nr(e){await l(tr(e.projectDir,"AGENTS.md"),ci(e)),await l(tr(e.projectDir,"CLAUDE.md"),`@AGENTS.md
|
|
89
|
+
`)}import{join as li}from"path";var pi={postgres:"Postgres (self-hosted)",neon:"Neon (managed Postgres)",supabase:"Supabase (managed Postgres)"};function di(e){let t=e.appName.trim()||e.projectName,n=e.frontend==="nuxt",r=n?"Nuxt 4":"Next.js 16",o=n?"apps/web-nuxt":"apps/web-next",i=pi[e.databaseProvider],a=vt(e),s=e.architecture==="fullstack"?`${r} app at \`${o}/\` with the Hono API mounted inside it. A standalone \`apps/backend/\` is also included (inert in this fullstack setup) so you can split the API into a separate service later without re-scaffolding.`:`${r} app at \`${o}/\` and a separate Hono backend at \`apps/backend/\`.`,p=e.architecture==="fullstack"?`- App + API: http://localhost:3000
|
|
86
90
|
- Inngest dev server: http://127.0.0.1:8288`:`- App: http://localhost:3000
|
|
87
91
|
- API: http://localhost:3010
|
|
88
|
-
- Inngest dev server: http://127.0.0.1:8288`,
|
|
92
|
+
- Inngest dev server: http://127.0.0.1:8288`,f=a?`pnpm infra # optional: starts local Docker services (Postgres / Redis / Inngest / Mailpit)
|
|
89
93
|
`:"",m=e.deploymentTarget==="vercel"?"### Deployment\n\nDeploy to Vercel: `vercel deploy` (or connect the repo in the Vercel dashboard). Required environment variables are listed in `.env`.":`### Deployment
|
|
90
94
|
|
|
91
|
-
This project ships with Dockerfiles for each app. Build images with \`docker build\` and deploy to your runtime of choice (Render / Fly.io / Railway / Coolify / Dokploy / your own VPS).${
|
|
95
|
+
This project ships with Dockerfiles for each app. Build images with \`docker build\` and deploy to your runtime of choice (Render / Fly.io / Railway / Coolify / Dokploy / your own VPS).${a?"\n\nThe `infra/` directory ships a Docker Compose file for the local-only services (Postgres / Redis / Inngest / Mailpit, filtered by your provider choices).":""}`,d=e.aiTools.length>0?"To pull the latest boilerplate changes into this project, open it in your AI coding agent and ask: `update my GenerateSaaS project`.":"To pull the latest boilerplate changes into this project, install an AI coding agent (Claude Code / Cursor / Codex / Gemini CLI / Windsurf) and ask: `update my GenerateSaaS project`. The skill bundle that drives the update lives under each tool's skill root.";return`# ${t}
|
|
92
96
|
|
|
93
|
-
${
|
|
97
|
+
${s}
|
|
94
98
|
|
|
95
99
|
## Stack
|
|
96
100
|
|
|
97
|
-
- **Frontend:** ${
|
|
101
|
+
- **Frontend:** ${r}
|
|
98
102
|
- **Backend:** Hono (TypeScript, RPC-typed)
|
|
99
103
|
- **Auth:** Better Auth
|
|
100
104
|
- **ORM / DB:** Drizzle + ${i}
|
|
101
105
|
- **Background jobs:** Inngest
|
|
102
|
-
- **i18n:** ${
|
|
106
|
+
- **i18n:** ${n?"@nuxtjs/i18n":"next-intl"}
|
|
103
107
|
|
|
104
108
|
## Development
|
|
105
109
|
|
|
@@ -108,10 +112,11 @@ pnpm install
|
|
|
108
112
|
# A ready-to-run \`.env\` was generated for you - open it and fill in the
|
|
109
113
|
# values marked \`# TODO\` (provider API keys). \`.env.example\` is the committed
|
|
110
114
|
# reference. All apps load the single root \`.env\`.
|
|
111
|
-
${
|
|
115
|
+
${f}pnpm db:setup # first run: enable pgvector + create the schema (idempotent)
|
|
116
|
+
pnpm dev
|
|
112
117
|
\`\`\`
|
|
113
118
|
|
|
114
|
-
${
|
|
119
|
+
${p}
|
|
115
120
|
|
|
116
121
|
## Build your SaaS
|
|
117
122
|
|
|
@@ -124,7 +129,8 @@ App-level configuration lives in \`packages/config/src/index.ts\`. Translation s
|
|
|
124
129
|
## Database
|
|
125
130
|
|
|
126
131
|
\`\`\`bash
|
|
127
|
-
pnpm
|
|
132
|
+
pnpm db:setup # enable pgvector + push schema + knowledge indexes (idempotent)
|
|
133
|
+
pnpm -F @repo/database push # push schema changes only (once pgvector exists)
|
|
128
134
|
pnpm -F @repo/database migrate # run migrations
|
|
129
135
|
pnpm -F @repo/database studio # open Drizzle Studio
|
|
130
136
|
pnpm auth:generate # regenerate Better Auth schema after config changes
|
|
@@ -142,12 +148,12 @@ pnpm dlx generatesaas eject
|
|
|
142
148
|
|
|
143
149
|
## Updates
|
|
144
150
|
|
|
145
|
-
${
|
|
146
|
-
`}async function
|
|
151
|
+
${d}
|
|
152
|
+
`}async function rr(e){await l(li(e.projectDir,"README.md"),di(e))}import{join as Tt}from"path";function ui(e){let t=e.split(".");return t.length>=3?t.slice(1).join("."):e}function mi(e){return e.toLowerCase().replace(/[^a-z0-9]+/g,"-").replace(/^-+|-+$/g,"")||"app"}function fi(e,t){let n=mi(e);return{appId:`${t.split(".").reverse().join(".")}.${n.replace(/-/g,"")}`,productName:e,protocol:n}}async function or(e){let t=e.appName.replace(/\\/g,"\\\\").replace(/"/g,'\\"'),n=e.paymentProvider!=="none",r=e.baseUrl??"http://localhost:3000",o=e.baseUrl?new URL(e.baseUrl).hostname:"example.com",i=ui(o),a=e.demo?"false":"true",s=e.demo?`
|
|
147
153
|
support: {
|
|
148
154
|
enableInDev: true,
|
|
149
155
|
crisp: { websiteId: "7e221cec-ed61-46b7-b1b4-8cbc16557cca" }
|
|
150
|
-
},`:"",
|
|
156
|
+
},`:"",p=e.frontend==="nextjs"&&e.architecture==="fullstack"?"false":"true",f=fi(t,i),m=`{ enabled: ${e.rag} }`,d=e.companion?'{ enabled: true, clientId: "companion" }':'{ enabled: false, clientId: "companion" }',v=`{ enabled: ${e.mcpServer} }`,b=`import type { AppConfig } from "@repo/config/types";
|
|
151
157
|
import { desktopConfig } from "./desktop.mjs";
|
|
152
158
|
import { tenancyConfig } from "./tenancy-flags.mjs";
|
|
153
159
|
|
|
@@ -157,8 +163,8 @@ export const config: AppConfig = {
|
|
|
157
163
|
siteName: "${t}",
|
|
158
164
|
fullSiteName: "${t}",
|
|
159
165
|
domain: "${o}",
|
|
160
|
-
baseUrl: process.env.BASE_URL ?? "${
|
|
161
|
-
indexable: ${
|
|
166
|
+
baseUrl: process.env.BASE_URL ?? "${r}",
|
|
167
|
+
indexable: ${a},
|
|
162
168
|
logo: {
|
|
163
169
|
main: "/images/logo.svg",
|
|
164
170
|
square: "/images/logo-square.svg"
|
|
@@ -204,7 +210,7 @@ export const config: AppConfig = {
|
|
|
204
210
|
maxFileSizeMB: 5,
|
|
205
211
|
dailyUploadLimit: 20
|
|
206
212
|
},
|
|
207
|
-
payment: ${
|
|
213
|
+
payment: ${n?`{
|
|
208
214
|
enabled: true,
|
|
209
215
|
provider: "${e.paymentProvider}",
|
|
210
216
|
bannedCountries: ["BY", "CU", "IR", "KP", "RU", "SY"]
|
|
@@ -216,7 +222,7 @@ export const config: AppConfig = {
|
|
|
216
222
|
prefix: "key_"
|
|
217
223
|
},
|
|
218
224
|
apiDocs: true,
|
|
219
|
-
performanceMonitor: { enabled: ${
|
|
225
|
+
performanceMonitor: { enabled: ${p} },
|
|
220
226
|
notifications: {
|
|
221
227
|
enabled: true,
|
|
222
228
|
maxPerUser: 100,
|
|
@@ -234,7 +240,7 @@ export const config: AppConfig = {
|
|
|
234
240
|
enabled: true,
|
|
235
241
|
provider: "turnstile",
|
|
236
242
|
siteKey: "0x4AAAAAACJo9y9FxH8lqkGu"
|
|
237
|
-
}`:"{ enabled: false }"},${
|
|
243
|
+
}`:"{ enabled: false }"},${s}
|
|
238
244
|
currency: ${e.demo?`{
|
|
239
245
|
// Demo mirrors the boilerplate's showcase config so geo\u2192currency
|
|
240
246
|
// switching can be demonstrated. Real users get a single-currency
|
|
@@ -254,7 +260,7 @@ export const config: AppConfig = {
|
|
|
254
260
|
}`:`{
|
|
255
261
|
base: "${e.defaultCurrency}",
|
|
256
262
|
list: [
|
|
257
|
-
{ symbol: "${
|
|
263
|
+
{ symbol: "${Se[e.defaultCurrency].symbol}", name: "${Se[e.defaultCurrency].name}", code: "${e.defaultCurrency}", place: "${Se[e.defaultCurrency].place}", space: ${Se[e.defaultCurrency].space} }
|
|
258
264
|
],
|
|
259
265
|
countryMap: {
|
|
260
266
|
default: "${e.defaultCurrency}"
|
|
@@ -277,8 +283,11 @@ export const config: AppConfig = {
|
|
|
277
283
|
contentDir: "content/docs"
|
|
278
284
|
},
|
|
279
285
|
ai: {
|
|
280
|
-
enabled:
|
|
281
|
-
|
|
286
|
+
enabled: ${e.ai},
|
|
287
|
+
// AI billing mode chosen at init (--ai-byok / --no-ai-byok). BYOK: end-users bring their
|
|
288
|
+
// own provider key, all catalog providers are usable, and chat is NOT credit-metered. The
|
|
289
|
+
// operator-keyed, credit-metered platform mode (false) uses the models/credits below.
|
|
290
|
+
byok: ${e.aiByok},
|
|
282
291
|
models: {
|
|
283
292
|
// EXAMPLE entry - set provider/modelId for your model and update the
|
|
284
293
|
// rates below to the provider's CURRENT pricing. These numbers are
|
|
@@ -290,9 +299,18 @@ export const config: AppConfig = {
|
|
|
290
299
|
outputPerMTok: 25
|
|
291
300
|
}
|
|
292
301
|
},
|
|
293
|
-
credits: { creditsPerUsd: 100, margin: 1, fallbackRatePerMTok: { input: 5, output: 25 } }
|
|
302
|
+
credits: { creditsPerUsd: 100, margin: 1, fallbackRatePerMTok: { input: 5, output: 25 } },
|
|
303
|
+
// Let end-users author their own AI schedules. Set to false to ship only the
|
|
304
|
+
// create-disabled view (existing schedules still list/run/edit/delete).
|
|
305
|
+
userSchedules: true
|
|
294
306
|
},
|
|
295
307
|
desktop: desktopConfig,
|
|
308
|
+
// Server-side RAG (pgvector knowledge with embeddings + semantic search). Off by default.
|
|
309
|
+
rag: ${m},
|
|
310
|
+
// Companion daemon (apps/companion): drives the user's subscription CLIs for this app. Off by default.
|
|
311
|
+
companion: ${d},
|
|
312
|
+
// Outward MCP server exposing the capability layer to external agents. Off by default.
|
|
313
|
+
mcpServer: ${v},
|
|
296
314
|
seo: {
|
|
297
315
|
description: "Production-ready SaaS application",
|
|
298
316
|
foundingDate: "${new Date().getFullYear()}-01-01"
|
|
@@ -313,6 +331,7 @@ export const config: AppConfig = {
|
|
|
313
331
|
export const mainCurrency = config.currency.list.find((c) => c.code === config.currency.base)
|
|
314
332
|
?? config.currency.list[0]!;
|
|
315
333
|
|
|
334
|
+
export * from "./ai-providers";
|
|
316
335
|
export * from "./billing-fields";
|
|
317
336
|
export * from "./social-providers";
|
|
318
337
|
export * from "./blog";
|
|
@@ -324,7 +343,7 @@ export * from "./pricing";
|
|
|
324
343
|
export * from "./roles";
|
|
325
344
|
export * from "./section-tabs";
|
|
326
345
|
export * from "./tenancy";
|
|
327
|
-
`,
|
|
346
|
+
`,I=Tt(e.projectDir,"packages/config/src/index.ts");await l(I,b),await l(Tt(e.projectDir,"packages/config/src/desktop.mjs"),`// Plain-JS desktop app identity. NO env reads, so tooling that cannot import the
|
|
328
347
|
// TypeScript config index reads it directly: the Electron renderer + main process
|
|
329
348
|
// (electron-vite bundles @repo/config/desktop) and electron-builder (Node ESM).
|
|
330
349
|
// Edit these values once to rebrand - they are the same in dev and production, so
|
|
@@ -333,10 +352,10 @@ export * from "./tenancy";
|
|
|
333
352
|
|
|
334
353
|
export const desktopConfig = {
|
|
335
354
|
enabled: ${e.desktop},
|
|
336
|
-
appId: "${
|
|
337
|
-
productName: "${
|
|
338
|
-
protocol: "${
|
|
339
|
-
baseUrl: "${
|
|
355
|
+
appId: "${f.appId}",
|
|
356
|
+
productName: "${f.productName}",
|
|
357
|
+
protocol: "${f.protocol}",
|
|
358
|
+
baseUrl: "${r}",
|
|
340
359
|
// Name of the app's data folder (created under the OS app-data dir). Where the app
|
|
341
360
|
// and its AI agents persist files - logs, gathered data, generated artifacts. Rename
|
|
342
361
|
// it freely; it is a plain folder name, not a path.
|
|
@@ -358,10 +377,13 @@ export const desktopConfig = {
|
|
|
358
377
|
oauth: {},
|
|
359
378
|
// Let end users author their own background schedules on the Schedules screen.
|
|
360
379
|
// Set to false to ship only builder-registered schedules.
|
|
361
|
-
userSchedules: true
|
|
380
|
+
userSchedules: true,
|
|
381
|
+
// Cap how many chat conversations are kept per agent on the user's machine;
|
|
382
|
+
// creating a new chat past this limit deletes the oldest. Set to 0 for no limit.
|
|
383
|
+
maxChatsPerAgent: 20
|
|
362
384
|
}
|
|
363
385
|
};
|
|
364
|
-
`),await l(
|
|
386
|
+
`),await l(Tt(e.projectDir,"packages/config/src/tenancy-flags.mjs"),`// Plain-JS tenancy flags. NO env reads, so tooling that cannot import the
|
|
365
387
|
// TypeScript config index reads it directly (the Electron renderer). Mirrors the
|
|
366
388
|
// desktop.mjs pattern. The config index re-exports this as config.tenancy.
|
|
367
389
|
|
|
@@ -370,11 +392,11 @@ export const tenancyConfig = ${e.multiTenancy?`{
|
|
|
370
392
|
organizationLimit: 5,
|
|
371
393
|
billingScope: "${e.billingScope}"
|
|
372
394
|
}`:"{ multiTenant: false }"};
|
|
373
|
-
`),e.desktop&&await l(
|
|
395
|
+
`),e.desktop&&await l(Tt(e.projectDir,"apps/desktop/dev-app-update.yml"),`provider: generic
|
|
374
396
|
url: https://cdn.${i}/desktop
|
|
375
|
-
updaterCacheDirName: ${
|
|
376
|
-
`)}import{join as
|
|
377
|
-
`)}async function
|
|
397
|
+
updaterCacheDirName: ${f.protocol}-updater
|
|
398
|
+
`)}import{join as gi}from"path";function hi(e){return e==="stripe"?' stripePriceId: "",':' polarProductId: "",'}function rn(e){let t=[];return e.withCredits&&(t.push(` credits: ${e.credits},`),t.push(" creditInterval: 30,")),t.push(` apiRateLimit: { maxRequests: ${e.rateLimit} }`),t.join(`
|
|
399
|
+
`)}async function ir(e){let t=gi(e.projectDir,"packages/config/src/pricing.ts"),n=e.defaultCurrency;if(e.paymentProvider==="none"){let d=`import type { PricingConfig } from "@repo/config/types";
|
|
378
400
|
|
|
379
401
|
export const pricingConfig: PricingConfig = {
|
|
380
402
|
defaultPlan: "free",
|
|
@@ -394,7 +416,7 @@ export const pricingConfig: PricingConfig = {
|
|
|
394
416
|
prices: [
|
|
395
417
|
{
|
|
396
418
|
interval: "lifetime",
|
|
397
|
-
amounts: { ${
|
|
419
|
+
amounts: { ${n}: 0 }
|
|
398
420
|
}
|
|
399
421
|
],
|
|
400
422
|
apiRateLimit: { maxRequests: 100 }
|
|
@@ -403,7 +425,7 @@ export const pricingConfig: PricingConfig = {
|
|
|
403
425
|
credits: { enabled: false },
|
|
404
426
|
products: { enabled: false, items: [] }
|
|
405
427
|
};
|
|
406
|
-
`;await l(t,
|
|
428
|
+
`;await l(t,d);return}let r=e.paymentProvider,o=hi(r),i=e.credits,a=rn({credits:5,rateLimit:100,withCredits:i}),s=rn({credits:10,rateLimit:1e3,withCredits:i}),p=rn({credits:50,rateLimit:5e3,withCredits:i}),m=`import type { PricingConfig } from "@repo/config/types";
|
|
407
429
|
|
|
408
430
|
export const pricingConfig: PricingConfig = {
|
|
409
431
|
defaultPlan: "free",
|
|
@@ -424,10 +446,10 @@ export const pricingConfig: PricingConfig = {
|
|
|
424
446
|
prices: [
|
|
425
447
|
{
|
|
426
448
|
interval: "lifetime",
|
|
427
|
-
amounts: { ${
|
|
449
|
+
amounts: { ${n}: 0 }
|
|
428
450
|
}
|
|
429
451
|
],
|
|
430
|
-
${
|
|
452
|
+
${a}
|
|
431
453
|
},
|
|
432
454
|
{
|
|
433
455
|
id: "starter",
|
|
@@ -444,16 +466,16 @@ ${s}
|
|
|
444
466
|
{
|
|
445
467
|
${o}
|
|
446
468
|
interval: "month",
|
|
447
|
-
amounts: { ${
|
|
469
|
+
amounts: { ${n}: 9 }
|
|
448
470
|
},
|
|
449
471
|
{
|
|
450
472
|
${o}
|
|
451
473
|
interval: "year",
|
|
452
|
-
amounts: { ${
|
|
453
|
-
anchorAmounts: { ${
|
|
474
|
+
amounts: { ${n}: 90 },
|
|
475
|
+
anchorAmounts: { ${n}: 109 }
|
|
454
476
|
}
|
|
455
477
|
],
|
|
456
|
-
${
|
|
478
|
+
${s}
|
|
457
479
|
},
|
|
458
480
|
{
|
|
459
481
|
id: "pro",
|
|
@@ -473,18 +495,18 @@ ${a}
|
|
|
473
495
|
{
|
|
474
496
|
${o}
|
|
475
497
|
interval: "month",
|
|
476
|
-
amounts: { ${
|
|
477
|
-
anchorAmounts: { ${
|
|
498
|
+
amounts: { ${n}: 29 },
|
|
499
|
+
anchorAmounts: { ${n}: 39 },
|
|
478
500
|
featured: true
|
|
479
501
|
},
|
|
480
502
|
{
|
|
481
503
|
${o}
|
|
482
504
|
interval: "year",
|
|
483
|
-
amounts: { ${
|
|
484
|
-
anchorAmounts: { ${
|
|
505
|
+
amounts: { ${n}: 290 },
|
|
506
|
+
anchorAmounts: { ${n}: 349 }
|
|
485
507
|
}
|
|
486
508
|
],
|
|
487
|
-
${
|
|
509
|
+
${p}
|
|
488
510
|
}
|
|
489
511
|
],
|
|
490
512
|
${i?" credits: { enabled: true }":" credits: { enabled: false }"},
|
|
@@ -493,12 +515,12 @@ ${i?" credits: { enabled: true }":" credits: { enabled: false }"},
|
|
|
493
515
|
items: []
|
|
494
516
|
}
|
|
495
517
|
};
|
|
496
|
-
`;await l(t,m)}var
|
|
497
|
-
`)}function
|
|
498
|
-
`)}async function
|
|
499
|
-
`+
|
|
500
|
-
`+
|
|
501
|
-
image:
|
|
518
|
+
`;await l(t,m)}var yi={smtp:[{key:"SMTP_HOST",defaultValue:"localhost"},{key:"SMTP_PORT",defaultValue:"1025"},{key:"SMTP_USER"},{key:"SMTP_PASSWORD"}],ses:[{key:"AMAZON_SES_REGION",comment:"# TODO: Configure Amazon SES credentials (e.g. us-east-1)"},{key:"AMAZON_SES_KEY"},{key:"AMAZON_SES_SECRET"}],resend:[{key:"RESEND_API_KEY",comment:"# TODO: Add your Resend API key"}]};function vi(e){let t=se[e];return t.envVars.map((n,r)=>({key:n.name,...r===0?{comment:`# TODO: Add your ${t.label} OAuth credentials`}:{}}))}var ki={stripe:[{key:"STRIPE_SECRET_KEY",comment:"# TODO: Add your Stripe keys"},{key:"STRIPE_WEBHOOK_SECRET"}],polar:[{key:"POLAR_ACCESS_TOKEN",comment:"# TODO: Add your Polar keys"},{key:"POLAR_WEBHOOK_SECRET"}]};function Ge(e,t){return t?e.map(n=>{let r=t[n.key];return r?{...n,defaultValue:r,comment:void 0}:n}):e}function He(e,t){for(let n of e)n.comment&&t.push(n.comment),n.defaultValue!==void 0?t.push(`${n.key}=${n.defaultValue}`):t.push(`#${n.key}=`)}function on(e){return Array.from(crypto.getRandomValues(new Uint8Array(e))).map(t=>t.toString(16).padStart(2,"0")).join("")}function sr(e){return e.architecture==="fullstack"?{apiUrl:"http://localhost:3000/api",baseUrl:"http://localhost:3000"}:{apiUrl:"http://localhost:3010",baseUrl:"http://localhost:3000"}}function wi(e){let{architecture:t,deploymentTarget:n}=e;return t==="fullstack"?n==="vercel"?{frontend:"https://your-app.vercel.app",backend:"https://your-app.vercel.app"}:n==="node"?{frontend:"https://your-app.example.com",backend:"https://your-app.example.com"}:null:{frontend:"https://your-app.example.com",backend:"https://your-app.example.com/api"}}function Rt(e,t,n,r){e.push(r==="example"?`${t}=`:`${t}=${n}`)}function ar(e,t){let n=t==="example"?void 0:e.credentials,{apiUrl:r,baseUrl:o}=sr(e),i=[];e.architecture==="separate"?i.push("# API Configuration","# Standalone backend's own URL (the frontend reaches it via the public var above).",`API_URL=${r}`,`BASE_URL=${o}`):i.push("# App","# (API_URL is derived from the frontend's *_PUBLIC_API_URL by the runtime env schema.)",`BASE_URL=${o}`),i.push("","# Database"),He(Ge(G[e.databaseProvider].envVars,n),i),i.push("","# Cache"),He(Ge(re[e.cacheProvider].envVars,n),i),i.push("","# Authentication"),t==="example"&&i.push("# Generate a strong secret, e.g. `openssl rand -hex 32`"),Rt(i,"BETTER_AUTH_SECRET",crypto.randomUUID(),t),i.push("","# Content API (random secret; required when contentApi feature is enabled in @repo/config)"),Rt(i,"CONTENT_API_KEY",on(32),t),i.push("","# Job Queue - Inngest","INNGEST_APP_ID=api"),Rt(i,"INNGEST_EVENT_KEY",on(32),t),Rt(i,"INNGEST_SIGNING_KEY",on(32),t),i.push("INNGEST_BASE_URL=http://127.0.0.1:8288"),e.architecture==="separate"&&(i.push("","# API Port (standalone backend)","API_PORT=3010"),i.push("","# CORS + cross-subdomain cookies (production only - leave unset for local dev)","# TRUSTED_ORIGINS=https://your-app.example.com","# AUTH_COOKIE_DOMAIN=.example.com"));let a=yi[e.emailProvider];if(a&&(i.push("","# Email"),He(Ge(a,n),i)),e.paymentProvider!=="none"){let s=ki[e.paymentProvider];s&&(i.push("","# Payment"),He(Ge(s,n),i))}if(e.socialProviders.length>0){i.push("","# Social auth");for(let s of e.socialProviders)He(Ge(vi(s),n),i)}return e.demo&&(i.push("","# Captcha (Cloudflare Turnstile)"),He(Ge([{key:"TURNSTILE_SECRET_KEY",comment:"# TODO: Add your Cloudflare Turnstile secret key"}],n),i)),i.push("","# Translations - OpenRouter (optional)","# Set to auto-translate en/* into your other locales with `pnpm translate`.","# The pre-commit hook runs it on commit; without a key it skips and untranslated","# locales fall back to the default locale. Get a key: https://openrouter.ai/keys","#OPENROUTER_API_KEY="),i.push("","# AI base - provider API keys (optional)","# Enable config.ai in packages/config and set the key for your chosen provider.","# OPENROUTER_API_KEY above also works as an AI provider key.","#ANTHROPIC_API_KEY=","#OPENAI_API_KEY="),i.push(""),i.join(`
|
|
519
|
+
`)}function bi(e){let{apiUrl:t}=sr(e),n=e.frontend==="nextjs"?"NEXT_PUBLIC_API_URL":"NUXT_PUBLIC_API_URL",r=["# API Configuration",`${n}=${t}`],o=wi(e);return o&&e.architecture==="separate"&&r.push("","# Production (uncomment and replace with your deployed hostnames):",`# ${n}=${o.backend}`),r.push(""),r.join(`
|
|
520
|
+
`)}async function cr(e){let t=bi(e);await l(`${e.projectDir}/.env`,t+`
|
|
521
|
+
`+ar(e,"env")),await l(`${e.projectDir}/.env.example`,t+`
|
|
522
|
+
`+ar(e,"example"))}import{join as Si}from"path";async function lr(e){let t=[...e.dockerServices];if(G[e.databaseProvider].managed&&(t=t.filter(i=>i!=="postgres")),re[e.cacheProvider].managed&&(t=t.filter(i=>i!=="redis")),t.length===0)return!1;let n=[],r=[];t.includes("postgres")&&(n.push(` postgres:
|
|
523
|
+
image: pgvector/pgvector:pg18
|
|
502
524
|
ports:
|
|
503
525
|
- "\${POSTGRES_PORT:-5432}:5432"
|
|
504
526
|
environment:
|
|
@@ -512,7 +534,7 @@ ${i?" credits: { enabled: true }":" credits: { enabled: false }"},
|
|
|
512
534
|
test: ["CMD-SHELL", "pg_isready -U postgres"]
|
|
513
535
|
interval: 5s
|
|
514
536
|
timeout: 5s
|
|
515
|
-
retries: 5`),
|
|
537
|
+
retries: 5`),r.push(" postgres_data:")),t.includes("redis")&&(n.push(` redis:
|
|
516
538
|
image: redis:8-alpine
|
|
517
539
|
ports:
|
|
518
540
|
- "\${REDIS_PORT:-6379}:6379"
|
|
@@ -522,27 +544,27 @@ ${i?" credits: { enabled: true }":" credits: { enabled: false }"},
|
|
|
522
544
|
test: ["CMD", "redis-cli", "ping"]
|
|
523
545
|
interval: 5s
|
|
524
546
|
timeout: 5s
|
|
525
|
-
retries: 5`),
|
|
547
|
+
retries: 5`),r.push(" redis_data:")),t.includes("mailpit")&&n.push(` mailpit:
|
|
526
548
|
image: axllent/mailpit
|
|
527
549
|
ports:
|
|
528
550
|
- "\${MAILPIT_SMTP_PORT:-1025}:1025"
|
|
529
551
|
- "\${MAILPIT_UI_PORT:-8025}:8025"
|
|
530
552
|
environment:
|
|
531
553
|
MP_SMTP_AUTH_ACCEPT_ANY: 1
|
|
532
|
-
MP_SMTP_AUTH_ALLOW_INSECURE: 1`),t.includes("inngest")&&
|
|
554
|
+
MP_SMTP_AUTH_ALLOW_INSECURE: 1`),t.includes("inngest")&&n.push(` inngest:
|
|
533
555
|
image: inngest/inngest:v1.17.4
|
|
534
556
|
ports:
|
|
535
557
|
- "\${INNGEST_PORT:-8288}:8288"
|
|
536
558
|
command: inngest dev`);let o=`services:
|
|
537
|
-
${
|
|
559
|
+
${n.join(`
|
|
538
560
|
|
|
539
561
|
`)}
|
|
540
|
-
`;return
|
|
562
|
+
`;return r.length>0&&(o+=`
|
|
541
563
|
volumes:
|
|
542
|
-
${
|
|
564
|
+
${r.join(`
|
|
543
565
|
`)}
|
|
544
|
-
`),await l(
|
|
545
|
-
`)}import{join as
|
|
566
|
+
`),await l(Si(e.projectDir,"infra/docker-compose.yml"),o),!0}import{join as Ei}from"path";function Ai(e){let t=[];t.push({type:"node-terminal",request:"launch",name:"Dev",command:"pnpm dev",cwd:"${workspaceFolder}",skipFiles:["<node_internals>/**"],env:{NODE_ENV:"development"}}),e.architecture==="separate"&&t.push({type:"node-terminal",request:"launch",name:"Backend",command:"pnpm dev",cwd:"${workspaceFolder}/apps/backend",skipFiles:["<node_internals>/**"],env:{NODE_ENV:"development"}});let n=e.frontend==="nextjs"?"Next.js":"Nuxt";if(t.push({type:"node-terminal",request:"launch",name:n,command:"pnpm dev",cwd:`\${workspaceFolder}/apps/web-${e.frontend==="nextjs"?"next":"nuxt"}`,skipFiles:["<node_internals>/**"],env:{NODE_ENV:"development"}}),e.docs&&t.push({type:"node-terminal",request:"launch",name:"Docs",command:"pnpm dev",cwd:"${workspaceFolder}/apps/docs",skipFiles:["<node_internals>/**"],env:{NODE_ENV:"development"}}),e.desktop){let r=e.architecture==="separate"?"http://localhost:3010":"http://localhost:3000/api";t.push({type:"node-terminal",request:"launch",name:"Desktop",command:"pnpm dev",cwd:"${workspaceFolder}/apps/desktop",skipFiles:["<node_internals>/**"],env:{NODE_ENV:"development",BASE_URL:"http://localhost:3000",PUBLIC_API_URL:r}})}if(t.push({type:"node-terminal",request:"launch",name:"Inngest",command:"pnpm dev:inngest",cwd:"${workspaceFolder}",skipFiles:["<node_internals>/**"],env:{NODE_ENV:"development"}}),t.push({type:"node-terminal",request:"launch",name:"Mail",command:"pnpm dev:mail",cwd:"${workspaceFolder}",skipFiles:["<node_internals>/**"],env:{NODE_ENV:"development"}}),e.companion){let r=e.architecture==="separate"?"http://localhost:3010":"http://localhost:3000/api";t.push({type:"node-terminal",request:"launch",name:"Companion",command:`pnpm cli serve --url ${r}`,cwd:"${workspaceFolder}/apps/companion",skipFiles:["<node_internals>/**"]})}if(e.paymentProvider==="stripe"){let r=e.architecture==="separate"?"localhost:3010/auth/stripe/webhook":"localhost:3000/api/auth/stripe/webhook";t.push({type:"node-terminal",request:"launch",name:"Stripe",command:`stripe listen --forward-to ${r}`,cwd:"${workspaceFolder}",skipFiles:["<node_internals>/**"]})}return t}function Ii(e){let t=e.frontend==="nextjs"?"Next.js":"Nuxt",n=[];return e.architecture==="separate"?n.push({name:`Dev (${t} + Backend + Inngest)`,configurations:["Backend",t,"Inngest"]}):n.push({name:`Dev (${t} + Inngest)`,configurations:[t,"Inngest"]}),n}async function pr(e){let t={version:"0.2.0",configurations:Ai(e),compounds:Ii(e)};await l(Ei(e.projectDir,".vscode/launch.json"),JSON.stringify(t,null," ")+`
|
|
567
|
+
`)}import{join as Pi}from"path";async function dr(e){if(e.architecture!=="separate")return;await l(Pi(e.projectDir,"apps/backend/src/index.ts"),`import { serve } from "@hono/node-server";
|
|
546
568
|
import app from "@repo/api";
|
|
547
569
|
import { closeRedis, env, logger } from "@repo/runtime";
|
|
548
570
|
|
|
@@ -573,65 +595,12 @@ bootstrap().catch((error) => {
|
|
|
573
595
|
logger.error("[Backend] Fatal error", error);
|
|
574
596
|
process.exit(1);
|
|
575
597
|
});
|
|
576
|
-
`)}import{readFile as
|
|
577
|
-
export
|
|
578
|
-
|
|
579
|
-
|
|
580
|
-
|
|
581
|
-
|
|
582
|
-
export function affectedRowCount(result: unknown): number {
|
|
583
|
-
if (typeof result !== "object" || result === null) return 0;
|
|
584
|
-
const r = result as { rowCount?: number; count?: number };
|
|
585
|
-
return r.rowCount ?? r.count ?? 0;
|
|
586
|
-
}
|
|
587
|
-
`;function ni(e){return`import { drizzle } from "drizzle-orm/node-postgres";
|
|
588
|
-
import { z } from "zod";
|
|
589
|
-
import * as authSchema from "./db/auth";
|
|
590
|
-
import * as appSchema from "./db/schema";
|
|
591
|
-
|
|
592
|
-
const schema = { ...authSchema, ...appSchema };
|
|
593
|
-
export { schema };
|
|
594
|
-
|
|
595
|
-
const parsed = z.object({ DATABASE_URL: z.url() }).safeParse({ DATABASE_URL: process.env.DATABASE_URL });
|
|
596
|
-
if (!parsed.success) throw new Error("DATABASE_URL is not set or invalid");
|
|
597
|
-
|
|
598
|
-
export const db = drizzle(parsed.data.DATABASE_URL, { schema });
|
|
599
|
-
export const pool = db.$client;
|
|
600
|
-
|
|
601
|
-
${e}
|
|
602
|
-
${zt}`}function oi(e){return`import { neon } from "@neondatabase/serverless";
|
|
603
|
-
import { drizzle } from "drizzle-orm/neon-http";
|
|
604
|
-
import { z } from "zod";
|
|
605
|
-
import * as authSchema from "./db/auth";
|
|
606
|
-
import * as appSchema from "./db/schema";
|
|
607
|
-
|
|
608
|
-
const schema = { ...authSchema, ...appSchema };
|
|
609
|
-
export { schema };
|
|
610
|
-
|
|
611
|
-
const parsed = z.object({ DATABASE_URL: z.url() }).safeParse({ DATABASE_URL: process.env.DATABASE_URL });
|
|
612
|
-
if (!parsed.success) throw new Error("DATABASE_URL is not set or invalid");
|
|
613
|
-
|
|
614
|
-
const sql = neon(parsed.data.DATABASE_URL);
|
|
615
|
-
export const db = drizzle(sql, { schema });
|
|
616
|
-
|
|
617
|
-
${e}
|
|
618
|
-
${zt}`}function ii(e){return`import { drizzle } from "drizzle-orm/postgres-js";
|
|
619
|
-
import postgres from "postgres";
|
|
620
|
-
import { z } from "zod";
|
|
621
|
-
import * as authSchema from "./db/auth";
|
|
622
|
-
import * as appSchema from "./db/schema";
|
|
623
|
-
|
|
624
|
-
const schema = { ...authSchema, ...appSchema };
|
|
625
|
-
export { schema };
|
|
626
|
-
|
|
627
|
-
const parsed = z.object({ DATABASE_URL: z.url() }).safeParse({ DATABASE_URL: process.env.DATABASE_URL });
|
|
628
|
-
if (!parsed.success) throw new Error("DATABASE_URL is not set or invalid");
|
|
629
|
-
|
|
630
|
-
const client = postgres(parsed.data.DATABASE_URL);
|
|
631
|
-
export const db = drizzle(client, { schema });
|
|
632
|
-
|
|
633
|
-
${e}
|
|
634
|
-
${zt}`}import{join as bt}from"path";async function Mr(e){switch(e.cacheProvider){case"redis":await si(e),await ai(e);break;case"upstash":await ci(e),await li(e);break}}async function si(e){await l(bt(e.projectDir,"packages/runtime/src/redis.ts"),`import type { Store } from "hono-rate-limiter";
|
|
598
|
+
`)}import{readFile as Ti}from"fs/promises";import{join as Ri}from"path";var an='import { drizzle } from "drizzle-orm/node-postgres";',sn=`export const db = drizzle(parsed.data.DATABASE_URL, { schema });
|
|
599
|
+
export const pool = db.$client;`,_i={postgres:{importLine:an,instantiation:sn},neon:{importLine:`import { neon } from "@neondatabase/serverless";
|
|
600
|
+
import { drizzle } from "drizzle-orm/neon-http";`,instantiation:`const sql = neon(parsed.data.DATABASE_URL);
|
|
601
|
+
export const db = drizzle(sql, { schema });`},supabase:{importLine:`import { drizzle } from "drizzle-orm/postgres-js";
|
|
602
|
+
import postgres from "postgres";`,instantiation:`const client = postgres(parsed.data.DATABASE_URL);
|
|
603
|
+
export const db = drizzle(client, { schema });`}};async function ur(e){let t=Ri(e.projectDir,"packages/database/src/index.ts"),n=await Ti(t,"utf-8");if(!n.includes(an)||!n.includes(sn))throw new Error("generateDbDriver: boilerplate packages/database/src/index.ts is missing the node-postgres driver anchors (the import line or the `db` instantiation block); the boilerplate drifted.");let r=_i[e.databaseProvider],o=n.replace(an,r.importLine).replace(sn,r.instantiation);await l(t,o)}import{join as _t}from"path";async function mr(e){switch(e.cacheProvider){case"redis":await Oi(e),await xi(e);break;case"upstash":await Ci(e),await Di(e);break}}async function Oi(e){await l(_t(e.projectDir,"packages/runtime/src/redis.ts"),`import type { Store } from "hono-rate-limiter";
|
|
635
604
|
import { Redis } from "ioredis";
|
|
636
605
|
import { RedisStore, type RedisReply } from "rate-limit-redis";
|
|
637
606
|
import { env } from "./env";
|
|
@@ -755,7 +724,7 @@ export async function closeRedis() {
|
|
|
755
724
|
closed = true;
|
|
756
725
|
}
|
|
757
726
|
}
|
|
758
|
-
`)}async function
|
|
727
|
+
`)}async function xi(e){await l(_t(e.projectDir,"packages/runtime/src/mutex.ts"),`import { Mutex } from "redis-semaphore";
|
|
759
728
|
import { redis } from "./redis";
|
|
760
729
|
|
|
761
730
|
export class MutexTimeoutError extends Error {
|
|
@@ -792,7 +761,7 @@ export async function withMutex<T>(
|
|
|
792
761
|
await mutex.release();
|
|
793
762
|
}
|
|
794
763
|
}
|
|
795
|
-
`)}async function
|
|
764
|
+
`)}async function Ci(e){await l(_t(e.projectDir,"packages/runtime/src/redis.ts"),`import { Redis } from "@upstash/redis";
|
|
796
765
|
import type { Store } from "hono-rate-limiter";
|
|
797
766
|
import { env } from "./env";
|
|
798
767
|
|
|
@@ -905,7 +874,7 @@ export const limiterStore: Store = createLimiterStore();
|
|
|
905
874
|
|
|
906
875
|
/** No persistent connection to close with Upstash REST. */
|
|
907
876
|
export async function closeRedis(): Promise<void> {}
|
|
908
|
-
`)}async function
|
|
877
|
+
`)}async function Di(e){await l(_t(e.projectDir,"packages/runtime/src/mutex.ts"),`import { Lock } from "@upstash/lock";
|
|
909
878
|
import { redis } from "./redis";
|
|
910
879
|
|
|
911
880
|
export class MutexTimeoutError extends Error {
|
|
@@ -950,7 +919,7 @@ export async function withMutex<T>(
|
|
|
950
919
|
await lock.release();
|
|
951
920
|
}
|
|
952
921
|
}
|
|
953
|
-
`)}async function
|
|
922
|
+
`)}async function fr(e){await l(`${e.projectDir}/packages/runtime/src/env.ts`,Ni(e))}function Ni(e){return`import { z } from "zod";
|
|
954
923
|
|
|
955
924
|
const EnvSchema = z.object({
|
|
956
925
|
NODE_ENV: z.enum(["development", "production"]).default("development"),
|
|
@@ -999,6 +968,8 @@ ${e.cacheProvider==="upstash"?` UPSTASH_REDIS_REST_URL: z.string(),
|
|
|
999
968
|
|
|
1000
969
|
SMTP_HOST: z.string().optional(),
|
|
1001
970
|
SMTP_PORT: z.coerce.number().int().positive().optional(),
|
|
971
|
+
SMTP_USER: z.string().optional(),
|
|
972
|
+
SMTP_PASSWORD: z.string().optional(),
|
|
1002
973
|
|
|
1003
974
|
TWILIO_ACCOUNT_SID: z.string().optional(),
|
|
1004
975
|
TWILIO_AUTH_TOKEN: z.string().optional(),
|
|
@@ -1069,17 +1040,17 @@ export const env = (() => {
|
|
|
1069
1040
|
const parsedApiUrl = new URL(env.API_URL);
|
|
1070
1041
|
export const apiBasePath = parsedApiUrl.pathname === "/" ? "" : parsedApiUrl.pathname;
|
|
1071
1042
|
export const apiOrigin = parsedApiUrl.origin;
|
|
1072
|
-
`}import{readFile as
|
|
1073
|
-
`)}function
|
|
1074
|
-
`)}var
|
|
1075
|
-
`))}}import{readFile as
|
|
1076
|
-
`);let i=
|
|
1043
|
+
`}import{readFile as Li}from"fs/promises";import{join as oe}from"path";var Ot={"@upstash/redis":"^1.37.0","@upstash/lock":"^0.2.1","@neondatabase/serverless":"^1.0.1",postgres:"^3.4.7"};async function Ye(e){let t=await Li(e,"utf-8");return JSON.parse(t)}async function qe(e,t){await l(e,JSON.stringify(t,null," ")+`
|
|
1044
|
+
`)}function ct(e,t){for(let n of t)delete e.dependencies?.[n],delete e.devDependencies?.[n]}function ze(e,t,n,r=!1){let o=r?"devDependencies":"dependencies";e[o]||(e[o]={}),e[o][t]=n}async function gr(e){await $i(e),await ji(e),await Ui(e),await Mi(e),e.frontend==="nextjs"?await Fi(e):await Vi(e)}async function $i(e){let t=oe(e.projectDir,"packages/api/package.json"),n=await Ye(t);ct(n,["sharp","@types/sharp"]),await qe(t,n)}async function ji(e){let t=oe(e.projectDir,"packages/runtime/package.json"),n=await Ye(t);e.cacheProvider==="upstash"&&(ct(n,["ioredis","rate-limit-redis","redis-semaphore"]),ze(n,"@upstash/redis",Ot["@upstash/redis"]),ze(n,"@upstash/lock",Ot["@upstash/lock"])),await qe(t,n)}async function Ui(e){let t=oe(e.projectDir,"packages/database/package.json"),n=await Ye(t);e.databaseProvider==="neon"?(ct(n,["pg","@types/pg"]),ze(n,"@neondatabase/serverless",Ot["@neondatabase/serverless"])):e.databaseProvider==="supabase"&&(ct(n,["pg","@types/pg"]),ze(n,"postgres",Ot.postgres)),await qe(t,n)}async function Mi(e){if(e.architecture!=="separate")return;let t=oe(e.projectDir,"apps/backend/package.json"),n=await Ye(t);e.deploymentTarget!=="node"&&ct(n,["@hono/node-server"]),await qe(t,n)}async function Vi(e){if(e.architecture!=="separate")return;let t=oe(e.projectDir,"apps/web-nuxt");await nn(oe(t,"server/api/[...paths].ts"),t);let n=oe(t,"package.json"),r=await Ye(n),o=r.dependencies?.["@repo/api"];o&&(delete r.dependencies?.["@repo/api"],ze(r,"@repo/api",o,!0)),await qe(n,r)}async function Fi(e){if(e.architecture!=="separate")return;let t=oe(e.projectDir,"apps/web-next");await nn(oe(t,"app/api/[[...rest]]/route.ts"),t);let n=oe(t,"package.json"),r=await Ye(n),o=r.dependencies?.["@repo/api"];o&&(delete r.dependencies?.["@repo/api"],ze(r,"@repo/api",o,!0)),await qe(n,r)}import{readFile as yr}from"fs/promises";import{join as vr}from"path";async function kr(e){let t=vr(e.projectDir,"turbo.json"),n;try{n=await yr(t,"utf-8")}catch{return}let r=JSON.parse(n),o=r.tasks?.build;if(!o)return;let i=e.frontend==="nextjs"?"NUXT_PUBLIC_*":"NEXT_PUBLIC_*",a=e.frontend==="nextjs"?new Set([".nuxt/**",".output/**"]):new Set([".next/**","!.next/cache/**"]),s=!1;if(Array.isArray(o.env)){let p=o.env.filter(f=>f!==i);p.length!==o.env.length&&(o.env=p,s=!0)}if(Array.isArray(o.outputs)){let p=o.outputs.filter(f=>!a.has(f));p.length!==o.outputs.length&&(o.outputs=p,s=!0)}s&&await l(t,JSON.stringify(r,null," ")+`
|
|
1045
|
+
`)}var Bi=["base.json","node.json","next.json"],hr="GenerateSaaS ";async function wr(e){if(!e.demo)for(let t of Bi){let n=vr(e.projectDir,"tooling/typescript",t),r;try{r=await yr(n,"utf-8")}catch{continue}let o=JSON.parse(r);typeof o.display!="string"||!o.display.startsWith(hr)||(o.display=o.display.slice(hr.length),await l(n,JSON.stringify(o,null," ")+`
|
|
1046
|
+
`))}}import{readFile as Ki,rm as Gi}from"fs/promises";import{existsSync as br}from"fs";import{join as Sr}from"path";async function Er(e){if(e.frontend==="nuxt")return;let t=Sr(e.projectDir,"packages/i18n/package.json");if(!br(t))throw new Error(`pruneI18nNuxt: expected ${t} to exist - did the i18n package move?`);let n=JSON.parse(await Ki(t,"utf-8")),r=!!(n.exports?.["./module"]??n.exports?.["./nuxt"]),o=!1;if(n.exports)for(let a of["./module","./nuxt"])a in n.exports&&(delete n.exports[a],o=!0);n.devDependencies&&"@nuxt/kit"in n.devDependencies&&(delete n.devDependencies["@nuxt/kit"],o=!0),o&&await l(t,JSON.stringify(n,null," ")+`
|
|
1047
|
+
`);let i=Sr(e.projectDir,"packages/i18n/nuxt");if(r&&!br(i))throw new Error(`pruneI18nNuxt: packages/i18n declares a Nuxt export surface but ${i} is missing - did the i18n Nuxt module move?`);await Gi(i,{recursive:!0,force:!0})}import{readFile as Ar,rm as Hi}from"fs/promises";import{join as cn}from"path";async function Ir(e){e.cacheProvider==="upstash"&&await Promise.all([zi(e.projectDir),Yi(e.projectDir),Hi(cn(e.projectDir,"packages/runtime/tests/redis.test.ts"),{force:!0})])}async function zi(e){let t=cn(e,"packages/runtime/tests/setup.ts"),n;try{n=await Ar(t,"utf-8")}catch{return}let r=n.replace(/\tREDIS_URL:\s*"[^"]*",?\n/,` UPSTASH_REDIS_REST_URL: "https://test.upstash.io",
|
|
1077
1048
|
UPSTASH_REDIS_REST_TOKEN: "test-token",
|
|
1078
|
-
`);n
|
|
1049
|
+
`);r!==n&&await l(t,r)}async function Yi(e){let t=cn(e,"packages/api/tests/setup.ts"),n;try{n=await Ar(t,"utf-8")}catch{return}let r=n;r=r.replace(/\tREDIS_URL:\s*"[^"]*",?\n/g,` UPSTASH_REDIS_REST_URL: "https://test.upstash.io",
|
|
1079
1050
|
UPSTASH_REDIS_REST_TOKEN: "test-token",
|
|
1080
|
-
`),
|
|
1051
|
+
`),r=r.replace(/vi\.mock\("ioredis"[\s\S]*?\n\}\);\n\n?/,""),r=r.replace(/vi\.mock\("rate-limit-redis"[\s\S]*?\n\}\);\n\n?/,""),r=r.replace(/\t\t\tREDIS_URL:\s*"[^"]*",?\n/,` UPSTASH_REDIS_REST_URL: "https://test.upstash.io",
|
|
1081
1052
|
UPSTASH_REDIS_REST_TOKEN: "test-token",
|
|
1082
|
-
`),
|
|
1053
|
+
`),r=r.replace(/(^vi\.mock\("@repo\/runtime")/m,`vi.mock("@upstash/redis", () => {
|
|
1083
1054
|
class Redis {
|
|
1084
1055
|
get = vi.fn().mockResolvedValue(null);
|
|
1085
1056
|
set = vi.fn().mockResolvedValue("OK");
|
|
@@ -1096,15 +1067,67 @@ vi.mock("@upstash/lock", () => {
|
|
|
1096
1067
|
return { Lock };
|
|
1097
1068
|
});
|
|
1098
1069
|
|
|
1099
|
-
$1`),n
|
|
1100
|
-
|
|
1101
|
-
`)}
|
|
1102
|
-
|
|
1103
|
-
`)
|
|
1104
|
-
|
|
1105
|
-
`)
|
|
1070
|
+
$1`),r!==n&&await l(t,r)}import{readdir as fn,readFile as j,rm as w}from"fs/promises";import{join as h}from"path";import{readFile as Ct,rm as qi}from"fs/promises";import{join as xt}from"path";function Ji(e){return new RegExp(`^[ \\t]*(?:\\/\\/|\\{\\/\\*|\\*) gsaas:${e}-start(?: \\*\\/\\})?[ \\t]*\\r?\\n[\\s\\S]*?^[ \\t]*(?:\\/\\/|\\{\\/\\*|\\*) gsaas:${e}-end(?: \\*\\/\\})?[ \\t]*\\r?\\n`,"gm")}function J(e,t,n="ai",r="stripDesktopAi"){let o=Ji(n);if(e.match(o)===null)throw new Error(`${r}: expected gsaas ${n} markers in ${t}, but found none (boilerplate drift).`);let i=e.replace(o,"");if(i.includes(`gsaas:${n}-start`)||i.includes(`gsaas:${n}-end`))throw new Error(`${r}: an unbalanced gsaas ${n} marker is left in ${t} (boilerplate drift).`);return i.replace(/\n{3,}/g,`
|
|
1071
|
+
|
|
1072
|
+
`)}async function _(e,t,n,r,o){let i=await Ct(e,"utf-8"),a=i.indexOf(t);if(a===-1)throw new Error(`${o}: expected to find the ${r} seam, but it was missing (boilerplate drift).`);if(i.indexOf(t,a+t.length)!==-1)throw new Error(`${o}: the ${r} seam appears more than once (boilerplate drift).`);await l(e,i.slice(0,a)+n+i.slice(a+t.length))}function lt(e,t,n,r){if(!(t in e))throw new Error(`${r}: expected ${n} (boilerplate drift).`);delete e[t]}function Dt(e){return e.frontend==="nextjs"?"next":"nuxt"}async function Nt(e,t,n,r,o,i){let a=xt(e,"docs",t);await qi(xt(a,`${n}.mdx`)),await _(xt(a,"meta.json"),` "${n}",
|
|
1073
|
+
`,"",`${t} meta.json ${n} nav entry`,o);let s=xt(a,"ai.mdx"),f=(await Ct(s,"utf-8")).split(`
|
|
1074
|
+
`).find(m=>m.includes(`href="${r}"`));if(f===void 0)throw new Error(`${o}: expected the ai.mdx card linking to ${r} (boilerplate drift).`);if(await _(s,`${f}
|
|
1075
|
+
`,"",`${t} ai.mdx ${n} card`,o),i!==void 0){let m=`| **${i}** |`,v=(await Ct(s,"utf-8")).split(`
|
|
1076
|
+
`).find(b=>b.startsWith(m));if(v===void 0)throw new Error(`${o}: expected the ai.mdx "${i}" hub-table row (boilerplate drift).`);await _(s,`${v}
|
|
1077
|
+
`,"",`${t} ai.mdx ${i} hub-table row`,o)}}async function Lt(e){let t=`import { config } from "@repo/config";
|
|
1078
|
+
`,n=await Ct(e,"utf-8");if(!n.includes(t))return;let r=n.replace(t,""),o=r.replace(/\/\*[\s\S]*?\*\//g,"").replace(/\/\/[^\n]*/g,"").replace(/"(?:\\.|[^"\\])*"/g,'""').replace(/'(?:\\.|[^'\\])*'/g,"''").replace(/`(?:\\.|[^`\\])*`/g,"``");/\bconfig\b/.test(o)||await l(e,r)}import{readFile as Wi}from"fs/promises";import{join as Xi}from"path";async function ln(e){if(e.rag)return;let t=Xi(e.projectDir,"packages/config/src/index.ts"),n=await Wi(t,"utf-8");if(!/rag:\s*\{\s*enabled:\s*false\s*\}/.test(n))throw new Error("stripRag: expected `config.rag` emitted with `enabled: false` for a rag-off build, but it was missing (generator drift).")}import{readFile as Pr,rm as pn}from"fs/promises";import{join as Re}from"path";async function dn(e){if(e.mcpServer)return;let t=e.projectDir,n="stripMcpServer";await pn(Re(t,"packages/api/src/routes/internal/ai/mcp.ts")),await pn(Re(t,"packages/api/tests/routes/mcp.test.ts")),await pn(Re(t,"packages/auth/src/mcp.ts"));let r=Re(t,"packages/api/src/routes/internal/index.ts");await _(r,`import mcpRoutes from "./ai/mcp";
|
|
1079
|
+
`,"","internal router mcp import",n),await _(r,'\n// Outward MCP server (`/mcp` + its OAuth `.well-known` discovery), gated on `config.mcpServer`.\n// Mounted at the API root so the discovery metadata sits at `{apiBasePath}/.well-known/...` where\n// an external agent probes it. Off by default, so the routes are absent entirely when disabled. It\n// is not chained into the typed router above: external agents (not the typed RPC client) call it,\n// so it needs no `AppType` inference, and gating it keeps the feature truly inert when off.\nif (config.mcpServer?.enabled) {\n app.route("/", mcpRoutes);\n}\n',"","internal router mcp mount",n),await Lt(r);let o=Re(t,"packages/auth/src/config.ts");await _(o,` magicLink,
|
|
1080
|
+
mcp,
|
|
1081
|
+
organization,`,` magicLink,
|
|
1082
|
+
organization,`,"auth config mcp plugin import",n),await _(o,` // Outward MCP server: turns Better Auth into an OAuth 2.1 provider so an
|
|
1083
|
+
// external agent (Hermes/OpenClaw) can authenticate and drive the app's
|
|
1084
|
+
// capability layer over MCP. Brings the oauthApplication/oauthAccessToken/
|
|
1085
|
+
// oauthConsent tables (run \`pnpm auth:generate\` after toggling).
|
|
1086
|
+
...(config.mcpServer?.enabled
|
|
1087
|
+
? [mcp({ loginPage: \`\${config.baseUrl}\${config.routes.auth}/login\` })]
|
|
1088
|
+
: []),
|
|
1089
|
+
`,"","auth config mcp plugin block",n);let i=Re(t,"packages/auth/package.json"),a=JSON.parse(await Pr(i,"utf-8"));if(!a.exports)throw new Error(`${n}: expected an exports map in packages/auth/package.json (boilerplate drift).`);lt(a.exports,"./mcp",'the "./mcp" export in packages/auth/package.json',n),await l(i,JSON.stringify(a,null," ")+`
|
|
1090
|
+
`);let s=Re(t,"packages/api/package.json"),p=JSON.parse(await Pr(s,"utf-8"));if(!p.dependencies)throw new Error(`${n}: expected dependencies in packages/api/package.json (boilerplate drift).`);lt(p.dependencies,"@modelcontextprotocol/sdk","@modelcontextprotocol/sdk in packages/api dependencies",n),await l(s,JSON.stringify(p,null," ")+`
|
|
1091
|
+
`);let f=Dt(e);await Nt(t,f,"companion-agent-mcp",`/${f}/companion-agent-mcp`,n,"MCP")}import{rm as W}from"fs/promises";import{join as X}from"path";import{readFile as un}from"fs/promises";import{join as Je}from"path";async function Tr(e,t){let n=Je(e,"packages/api/src/routes/internal/index.ts");await l(n,J(await un(n,"utf-8"),"internal router","companion",t)),await Lt(n);let r=Je(e,"packages/api/package.json"),o=JSON.parse(await un(r,"utf-8"));if(!o.dependencies)throw new Error(`${t}: expected dependencies in packages/api/package.json (boilerplate drift).`);lt(o.dependencies,"@repo/companion-protocol","@repo/companion-protocol in packages/api dependencies",t),await l(r,JSON.stringify(o,null," ")+`
|
|
1092
|
+
`);let i=Je(e,"packages/api/src/ai/schedule-runner.ts");await l(i,J(await un(i,"utf-8"),"schedule-runner.ts","companion",t));let a=Je(e,"packages/api/src/ai/settings-store.ts");await _(a,`import { parseCompanionKey } from "../relay/companion-key";
|
|
1093
|
+
|
|
1094
|
+
`,"","settings-store.ts relay import",t),await _(a," * - COMPANION (when `config.companion.enabled`): a well-formed companion key\n * `<connectionId>@<deviceId>@<modelId>` is also accepted, so a default/fallback model that points\n * at a paired CLI persists (the run path resolves it, and the daemon handles an offline device).\n * Without this, the `@`-separated key is neither a registry key nor a `provider::modelId` key, so\n * it would silently clear to `null` on save and vanish on reload.\n","","settings-store.ts COMPANION docstring bullet",t),await _(a," *\n * When `opts.allowCompanion` is `false`, a companion key is REJECTED even when the companion is\n * enabled - used by the builder-task surfaces, whose runs are background/cloud-only and can never\n * dispatch to a device-bound CLI.\n","","settings-store.ts opts.allowCompanion docstring paragraph",t),await _(a," * @param opts - `allowCompanion` (default `true`) controls companion-key acceptance.\n","","settings-store.ts @param opts",t),await _(a,`export function validModelKey(
|
|
1095
|
+
key: string | null | undefined,
|
|
1096
|
+
opts: { allowCompanion?: boolean } = {}
|
|
1097
|
+
): string | null {
|
|
1098
|
+
`,`export function validModelKey(key: string | null | undefined): string | null {
|
|
1099
|
+
`,"settings-store.ts validModelKey signature",t),await _(a,` if (
|
|
1100
|
+
opts.allowCompanion !== false &&
|
|
1101
|
+
config.companion?.enabled === true &&
|
|
1102
|
+
parseCompanionKey(key) !== null
|
|
1103
|
+
) {
|
|
1104
|
+
return key;
|
|
1105
|
+
}
|
|
1106
|
+
`,"","settings-store.ts companion-key branch",t);let s=Je(e,"packages/api/src/ai/task-run.ts");await _(s,`import { parseCompanionKey } from "../relay/companion-key";
|
|
1107
|
+
`,"","task-run.ts relay import",t),await _(s,`/** Whether a chain link is usable for a task run: present and not a companion (device) key. */
|
|
1108
|
+
`,`/** Whether a chain link is usable for a task run: a present, non-empty key. */
|
|
1109
|
+
`,"task-run.ts usableTaskModelKey docstring",t),await _(s,` return typeof key === "string" && key.length > 0 && parseCompanionKey(key) === null;
|
|
1110
|
+
`,` return typeof key === "string" && key.length > 0;
|
|
1111
|
+
`,"task-run.ts usableTaskModelKey body",t),await _(s,` * override, then the builder's \`defaults.web.model\`, then the acting user's account default -
|
|
1112
|
+
* skipping companion (device-bound) links, which are valid for chat but never for a background
|
|
1113
|
+
* task - and delegating the final resolution to {@link resolveChatRun}, which owns BYOK/platform
|
|
1114
|
+
`," * override, then the builder's `defaults.web.model`, then the acting user's account default,\n * delegating the final resolution to {@link resolveChatRun}, which owns BYOK/platform\n","task-run.ts resolveTaskRun docstring model chain",t);let p=Je(e,"packages/api/src/routes/internal/ai/tasks-router.ts");await _(p,` modelKey: validModelKey(override?.modelKey ?? null, { allowCompanion: false }),
|
|
1115
|
+
`,` modelKey: validModelKey(override?.modelKey ?? null),
|
|
1116
|
+
`,"tasks-router.ts listTaskOverrides validModelKey",t),await _(p,` if (typeof body.modelKey === "string" && validModelKey(body.modelKey, { allowCompanion: false }) === null) {
|
|
1117
|
+
`,` if (typeof body.modelKey === "string" && validModelKey(body.modelKey) === null) {
|
|
1118
|
+
`,"tasks-router.ts putTask validModelKey",t)}async function mn(e){if(e.companion)return;let t=e.projectDir,n="stripCompanion";await W(X(t,"apps/companion"),{recursive:!0}),await W(X(t,"packages/companion-protocol"),{recursive:!0}),e.desktop&&e.desktopAi||await W(X(t,"packages/agent-runtime"),{recursive:!0}),await W(X(t,"packages/api/src/relay"),{recursive:!0}),await W(X(t,"packages/api/tests/relay"),{recursive:!0}),await W(X(t,"packages/api/src/routes/internal/companion.ts")),await W(X(t,"packages/api/src/routes/internal/companion-transport.ts")),await W(X(t,"packages/api/src/routes/internal/companion-install.ts")),await W(X(t,"packages/api/tests/routes/companion.test.ts")),await W(X(t,"packages/api/tests/routes/companion-transport.test.ts")),await W(X(t,"packages/api/tests/routes/companion-install.test.ts")),await Tr(t,n);let r=Dt(e);await Nt(t,r,"companion",`/${r}/companion`,n,"Companions")}var Zi=new Set(["ci.yml","desktop-release.yml"]),Qi=["cli","cli:clean","demo:bench","playground:regen","playground:test","playground:test:units","playground:test:build"];async function _r(e){let t=h(e.projectDir,".github/workflows"),n=await fn(t).catch(()=>[]);for(let r of n)Zi.has(r)||await w(h(t,r),{recursive:!0,force:!0})}async function Or(e){let t=h(e.projectDir,"package.json"),n=await j(t,"utf-8"),r=JSON.parse(n),o=!1;if(r.scripts){for(let i of Qi)i in r.scripts&&(delete r.scripts[i],o=!0);if(!vt(e))for(let i of["infra","infra:stop"])i in r.scripts&&(delete r.scripts[i],o=!0)}o&&await l(t,JSON.stringify(r,null," ")+`
|
|
1119
|
+
`)}async function xr(e){let t=h(e.projectDir,"package.json"),n=await j(t,"utf-8"),r=JSON.parse(n);!r.scripts||!("dev"in r.scripts)||(r.scripts.dev=e.architecture==="separate"?"turbo run dev --continue":"turbo run dev --continue --filter=!backend",await l(t,JSON.stringify(r,null," ")+`
|
|
1120
|
+
`))}async function Cr(e){let t=e.frontend==="nextjs"?"apps/web-nuxt":"apps/web-next";await w(h(e.projectDir,t),{recursive:!0,force:!0})}async function Dr(e){e.docs||await w(h(e.projectDir,"apps/docs"),{recursive:!0})}async function Nr(e){let t=e.frontend==="nextjs"?"docs/nuxt":"docs/next";if(await w(h(e.projectDir,t),{recursive:!0,force:!0}),await w(h(e.projectDir,"docs/index.mdx")),!e.desktop)await w(h(e.projectDir,"docs/desktop"),{recursive:!0,force:!0});else if(!e.desktopAi){await w(h(e.projectDir,"docs/desktop/ai"),{recursive:!0,force:!0});let n=h(e.projectDir,"docs/desktop/meta.json"),r=JSON.parse(await j(n,"utf-8"));r.pages=r.pages.filter(o=>o!=="ai"),await l(n,`${JSON.stringify(r,null,2)}
|
|
1121
|
+
`)}}async function Lr(e){if(e.desktop)return;await w(h(e.projectDir,"apps/desktop"),{recursive:!0});let t=h(e.projectDir,"packages/i18n/translations");for(let n of await fn(t,{withFileTypes:!0}))n.isDirectory()&&await w(h(t,n.name,"desktop.json"),{force:!0});await w(h(e.projectDir,".github/workflows/desktop-release.yml"))}var Rr=`
|
|
1122
|
+
# Local-embedding model binaries are fetched on install/build (see
|
|
1123
|
+
# scripts/fetch-embedding-model.mjs), never committed.
|
|
1124
|
+
resources/models/
|
|
1125
|
+
`;function ea(e){if(!e.includes(Rr))throw new Error("stripDesktopAi: expected the resources/models .gitignore block in apps/desktop/.gitignore, but it was missing (boilerplate drift).");return e.replace(Rr,`
|
|
1126
|
+
`)}function M(e,t,n,r){let o=e.indexOf(t);if(o===-1)throw new Error(`stripDesktopAi: expected to find the ${r} seam, but it was missing (boilerplate drift).`);if(e.indexOf(t,o+t.length)!==-1)throw new Error(`stripDesktopAi: the ${r} seam appears more than once (boilerplate drift).`);return e.slice(0,o)+n+e.slice(o+t.length)}async function $r(e){if(!e.desktop||e.desktopAi)return;let t=e.projectDir;await w(h(t,"apps/desktop/src/main/ai"),{recursive:!0}),await w(h(t,"apps/desktop/resources/ai-skills"),{recursive:!0}),await w(h(t,"apps/desktop/resources/knowledge"),{recursive:!0,force:!0}),await w(h(t,"apps/desktop/scripts/fetch-embedding-model.mjs")),await w(h(t,"apps/desktop/resources/models"),{recursive:!0,force:!0}),await w(h(t,"apps/desktop/src/renderer/src/screens/agents"),{recursive:!0}),await w(h(t,"apps/desktop/src/renderer/src/screens/integrations"),{recursive:!0}),await w(h(t,"apps/desktop/src/renderer/src/screens/ai"),{recursive:!0}),await w(h(t,"apps/desktop/src/renderer/src/lib/ai.ts")),await w(h(t,"apps/desktop/src/renderer/src/hooks/use-ai.ts")),await w(h(t,"apps/desktop/src/renderer/src/hooks/use-ai-scope.ts")),await w(h(t,"apps/desktop/src/renderer/src/screens/schedules.tsx")),await w(h(t,"apps/desktop/src/renderer/src/components/side-panel-dock.tsx")),await w(h(t,"apps/desktop/src/renderer/src/components/side-panel.tsx")),await w(h(t,"apps/desktop/src/renderer/src/config/side-panels.tsx")),await w(h(t,"apps/desktop/src/renderer/src/lib/side-panel-state.ts")),await w(h(t,"apps/desktop/tests/renderer/lib/side-panel-state.test.ts")),await w(h(t,"apps/desktop/src/renderer/src/lib/chat-prefs.ts")),await w(h(t,"apps/desktop/tests/renderer/lib/chat-prefs.test.ts")),await w(h(t,"apps/desktop/src/renderer/src/lib/ai-tab-state.ts")),await w(h(t,"apps/desktop/tests/renderer/lib/ai-tab-state.test.ts")),await w(h(t,"apps/desktop/tests/main/ai"),{recursive:!0}),await w(h(t,"apps/desktop/tests/main/ai-import-boundary.test.ts")),await w(h(t,"apps/desktop/tests/renderer/lib/ai.test.ts"));let n=h(t,"apps/desktop/package.json"),r=JSON.parse(await j(n,"utf-8")),o=["@repo/ai","@repo/agent-runtime","@anthropic-ai/claude-agent-sdk","@modelcontextprotocol/sdk","cross-spawn","@huggingface/transformers","@orama/orama","@homebridge/node-pty-prebuilt-multiarch","@xterm/xterm","@xterm/addon-fit"];for(let p of o){if(!r.dependencies||!(p in r.dependencies))throw new Error(`stripDesktopAi: expected ${p} in apps/desktop dependencies (boilerplate drift).`);delete r.dependencies[p]}for(let p of["fetch:model","predev","prebuild"]){if(!r.scripts||!(p in r.scripts))throw new Error(`stripDesktopAi: expected the ${p} script in apps/desktop package.json (boilerplate drift).`);delete r.scripts[p]}await l(n,JSON.stringify(r,null," ")+`
|
|
1127
|
+
`);let i=h(t,"apps/desktop/.gitignore"),a=await j(i,"utf-8");await l(i,ea(a)),await ta(t);let s=h(t,"packages/i18n/translations");for(let p of await fn(s,{withFileTypes:!0})){if(!p.isDirectory())continue;let f=h(s,p.name,"desktop.json"),m=await j(f,"utf-8").catch(()=>null);if(m===null)continue;let d=JSON.parse(m),v=!1;"agents"in d&&(delete d.agents,v=!0),"ai_hub"in d&&(delete d.ai_hub,v=!0),"schedules"in d&&(delete d.schedules,v=!0),"integrations"in d&&(delete d.integrations,v=!0),v&&await l(f,JSON.stringify(d,null," ")+`
|
|
1128
|
+
`)}}async function ta(e){let t=h(e,"apps/desktop/src/main/ipc.ts"),n=J(await j(t,"utf-8"),"main/ipc.ts");await l(t,n);let r=h(e,"apps/desktop/src/main/config.ts"),o=await j(r,"utf-8");o=M(o,`,
|
|
1106
1129
|
/** AI tool orchestration ("agents") feature flags. */
|
|
1107
|
-
agents: desktopConfig.agents`,"","main/config agents field"),await l(
|
|
1130
|
+
agents: desktopConfig.agents`,"","main/config agents field"),await l(r,o);let i=h(e,"apps/desktop/src/preload/index.ts"),a=await j(i,"utf-8");a=M(a,`import type {
|
|
1108
1131
|
AdapterCapabilities,
|
|
1109
1132
|
AuthStatus,
|
|
1110
1133
|
ConnectionRef,
|
|
@@ -1112,49 +1135,46 @@ $1`),n!==r&&await l(t,n)}import{readdir as Yt,readFile as F,rm as b}from"fs/prom
|
|
|
1112
1135
|
DetectResult,
|
|
1113
1136
|
IntegrationInfo,
|
|
1114
1137
|
ModelInfo,
|
|
1138
|
+
ModelRef,
|
|
1115
1139
|
PermissionDecision,
|
|
1116
|
-
RunEvent
|
|
1117
|
-
RunRequest
|
|
1140
|
+
RunEvent
|
|
1118
1141
|
} from '@repo/ai/backends'
|
|
1119
|
-
`,"","preload @repo/ai type import"),
|
|
1120
|
-
`,"","preload @repo/ai/agents type import"),
|
|
1121
|
-
`,"","preload ReasoningEffort type import"),
|
|
1122
|
-
`,"","preload DesktopTaskSpec import"),
|
|
1123
|
-
`,"","preload ScheduleView import");let
|
|
1124
|
-
`),
|
|
1125
|
-
`,"","router clientConfig import"),v
|
|
1126
|
-
|
|
1127
|
-
|
|
1128
|
-
`,"
|
|
1129
|
-
`,"","router
|
|
1130
|
-
`,"","router
|
|
1131
|
-
`,"","router integrations route spread"),await l(f,v);let E=h(e,"apps/desktop/src/renderer/src/config/sidebar.ts"),P=await F(E,"utf-8");P=$(P,`import {
|
|
1142
|
+
`,"","preload @repo/ai type import"),a=M(a,`import type { ChatSession, ScheduleRun, ScheduleTrigger } from '@repo/ai/agents'
|
|
1143
|
+
`,"","preload @repo/ai/agents type import"),a=M(a,`import type { ReasoningEffort } from '@repo/ai/backends'
|
|
1144
|
+
`,"","preload ReasoningEffort type import"),a=M(a,`import type { DesktopTaskSpec } from '@repo/config/types'
|
|
1145
|
+
`,"","preload DesktopTaskSpec import"),a=M(a,`import type { ScheduleView } from '../main/ai/ipc'
|
|
1146
|
+
`,"","preload ScheduleView import");let s=/,\n[ \t]*\/\/ gsaas:ai-start\n[\s\S]*?\n[ \t]*\/\/ gsaas:ai-end\n/g,p=a.match(s);if(p===null)throw new Error("stripDesktopAi: expected the preload `ai` bridge marker region, but it was missing (boilerplate drift).");if(p.length>1)throw new Error("stripDesktopAi: the preload `ai` bridge marker region appears more than once (boilerplate drift).");a=a.replace(s,`
|
|
1147
|
+
`),a=J(a,"preload/index.ts"),await l(i,a);let f=h(e,"apps/desktop/src/renderer/src/components/shell.tsx"),m=J(await j(f,"utf-8"),"components/shell.tsx");await l(f,m);let d=h(e,"apps/desktop/src/renderer/src/router.tsx"),v=await j(d,"utf-8");v=M(v,`import { clientConfig } from '@/lib/config'
|
|
1148
|
+
`,"","router clientConfig import"),v=M(v,` createRouter,
|
|
1149
|
+
redirect
|
|
1150
|
+
} from '@tanstack/react-router'`,` createRouter
|
|
1151
|
+
} from '@tanstack/react-router'`,"router redirect import specifier"),v=M(v,`import { AiHubScreen } from '@/screens/ai'
|
|
1152
|
+
`,"","router AiHubScreen import"),v=J(v,"router.tsx"),v=M(v,` ...(clientConfig.agentsEnabled ? [aiRoute, agentsRoute, schedulesRoute, integrationsRoute] : []),
|
|
1153
|
+
`,"","router ai route spread"),await l(d,v);let b=h(e,"apps/desktop/src/renderer/src/config/sidebar.ts"),I=await j(b,"utf-8");I=M(I,`import {
|
|
1132
1154
|
BuildingsIcon,
|
|
1133
|
-
ClockCountdownIcon,
|
|
1134
1155
|
GearIcon,
|
|
1135
1156
|
HouseIcon,
|
|
1136
|
-
PlugsConnectedIcon,
|
|
1137
1157
|
RobotIcon,
|
|
1138
1158
|
UserCircleIcon
|
|
1139
|
-
} from '@phosphor-icons/react'`,"import { BuildingsIcon, GearIcon, HouseIcon, UserCircleIcon } from '@phosphor-icons/react'","sidebar icon imports"),
|
|
1140
|
-
|
|
1141
|
-
|
|
1142
|
-
|
|
1143
|
-
|
|
1159
|
+
} from '@phosphor-icons/react'`,"import { BuildingsIcon, GearIcon, HouseIcon, UserCircleIcon } from '@phosphor-icons/react'","sidebar icon imports"),I=J(I,"config/sidebar.ts"),await l(b,I);let z=h(e,"apps/desktop/src/renderer/src/lib/config.ts"),L=await j(z,"utf-8");L=J(L,"lib/config.ts"),L=L.replace(/,(\n\} as const)/,"$1"),await l(z,L);let K=h(e,"apps/desktop/src/renderer/src/lib/sidebar-flags.ts"),ge=J(await j(K,"utf-8"),"lib/sidebar-flags.ts");await l(K,ge);let Y=h(e,"apps/desktop/electron.vite.config.ts"),k=await j(Y,"utf-8");k=M(k,",\n // The local-embeddings worker (`embeddings.ts`) spawns `embeddings-worker.js`,\n // which must sit next to the main `index.js`. Declaring both as named rollup\n // inputs emits `out/main/index.js` (the app entry) and\n // `out/main/embeddings-worker.js` (the worker) as `[name].js`, so the host's\n // `join(__dirname, 'embeddings-worker.js')` resolves in dev and packaged builds.\n rollupOptions: {\n // `@huggingface/transformers` loads its model runtime from `onnxruntime-node`,\n // a native addon that resolves its `.node` binding with a dynamic\n // `require('../bin/napi-v6/<platform>/onnxruntime_binding.node')`. A native\n // addon cannot be Rollup-bundled, so both packages stay external and load from\n // `node_modules` at runtime; electron-builder ships their trees (see\n // `electron-builder.mjs` `files`/`asarUnpack`). The embedded-terminal login flow\n // adds `@homebridge/node-pty-prebuilt-multiarch`, also a native addon (its\n // `pty.node` binding), kept external for the same reason. The CLI strips this whole\n // rollup block for an AI-off desktop build (the worker is removed with it).\n external: [\n '@huggingface/transformers',\n 'onnxruntime-node',\n '@homebridge/node-pty-prebuilt-multiarch'\n ],\n input: {\n index: resolve('src/main/index.ts'),\n 'embeddings-worker': resolve('src/main/ai/embeddings-worker.ts')\n }\n }","","electron.vite.config rollup worker block"),await l(Y,k);let R=h(e,"apps/desktop/electron-builder.mjs"),P=await j(R,"utf-8");P=M(P," // The main, preload, and renderer are self-contained bundles (electron-vite with\n // build.externalizeDeps:false), so the packaged app needs almost ZERO runtime\n // node_modules: exclude them all by default to keep the thin client small (the\n // backend dependency trees `@repo/api` drags in - express, pg, aws-sdk - are never\n // `require`d at runtime once bundled). Then RE-INCLUDE only the trees the local\n // embeddings worker keeps external (the native ONNX runtime can't be Rollup-bundled).\n // The worker `require`s `@huggingface/transformers`, whose Node build eagerly, at\n // import, pulls in `onnxruntime-node` (its prebuilt `.node` binding) + the shared\n // `onnxruntime-common`, AND `sharp` (a top-level `require(\"sharp\")` in its image\n // util that runs even for text-only embedding); `sharp` in turn needs its `@img/*`\n // platform natives plus `detect-libc` and `semver` at load. All of these must ship\n // or the worker import throws and semantic search silently degrades to BM25. Each\n // glob matches against electron-builder's production-dependency collection, which\n // flattens pnpm's `.pnpm` virtual store into the packaged `node_modules/<name>`\n // (following symlinks) - so transitive names resolve even though, under pnpm's\n // isolated linker, only `@huggingface/transformers` is linked at the app's own\n // node_modules and the rest live in `.pnpm`. The embedded-terminal login flow adds\n // `@homebridge/node-pty-prebuilt-multiarch` (its prebuilt `pty.node` binding); like the\n // ONNX runtime it cannot be Rollup-bundled, so its tree is re-included here too (the\n // `**/*.node` `asarUnpack` glob already unpacks `pty.node`). The CLI strips every\n // re-include for an AI-off desktop build (there is no external runtime to collect). The\n // `KB_SMOKE`-gated packaged-embedding test guards against this closure drifting.\n files: [\n 'out/**',\n 'resources/**',\n 'package.json',\n '!node_modules/**/*',\n 'node_modules/@huggingface/transformers/**/*',\n 'node_modules/onnxruntime-node/**/*',\n 'node_modules/onnxruntime-common/**/*',\n 'node_modules/sharp/**/*',\n 'node_modules/@img/**/*',\n 'node_modules/detect-libc/**/*',\n 'node_modules/semver/**/*',\n 'node_modules/@homebridge/node-pty-prebuilt-multiarch/**/*'\n ],",` // The main, preload, and renderer are self-contained bundles (electron-vite with
|
|
1160
|
+
// build.externalizeDeps:false), so the packaged app ships ZERO runtime node_modules
|
|
1161
|
+
// (every dependency is Rollup-bundled): keep only the built output, the resources,
|
|
1162
|
+
// and the package.json, and exclude node_modules entirely.
|
|
1163
|
+
files: ['out/**', 'resources/**', 'package.json', '!node_modules/**/*'],`,"electron-builder files re-includes"),P=M(P,"'resources/**', '**/*.node'","'resources/**'","electron-builder asarUnpack entry"),await l(R,P)}import{readFile as na}from"fs/promises";import{join as ra}from"path";var jr=`on:
|
|
1144
1164
|
workflow_run:
|
|
1145
1165
|
workflows: ["CI"]
|
|
1146
1166
|
branches: [main]
|
|
1147
1167
|
types: [completed]
|
|
1148
|
-
workflow_dispatch:`,
|
|
1168
|
+
workflow_dispatch:`,oa=`on:
|
|
1149
1169
|
# Automatic release on CI success is disabled for this project to conserve
|
|
1150
1170
|
# GitHub Actions minutes. Re-enable by uncommenting the workflow_run trigger.
|
|
1151
1171
|
# workflow_run:
|
|
1152
1172
|
# workflows: ["CI"]
|
|
1153
1173
|
# branches: [main]
|
|
1154
1174
|
# types: [completed]
|
|
1155
|
-
workflow_dispatch:`;async function
|
|
1175
|
+
workflow_dispatch:`;async function Ur(e){if(!e.desktop||e.desktopAutoRelease)return;let t=ra(e.projectDir,".github/workflows/desktop-release.yml"),n=await na(t,"utf8");if(!n.includes(jr))throw new Error(`Cannot make desktop releases manual: the expected workflow_run trigger block was not found in ${t}. The boilerplate workflow may have drifted.`);await l(t,n.replace(jr,oa))}import{join as ia}from"path";async function Mr(e){let t=aa(e);await l(ia(e.projectDir,".github/workflows/ci.yml"),t)}function aa(e){let t=G[e.databaseProvider].managed,n=e.cacheProvider==="upstash",r=e.frontend==="nuxt",o=t?"":` services:
|
|
1156
1176
|
postgres:
|
|
1157
|
-
image:
|
|
1177
|
+
image: pgvector/pgvector:pg18
|
|
1158
1178
|
env:
|
|
1159
1179
|
POSTGRES_USER: postgres
|
|
1160
1180
|
POSTGRES_PASSWORD: postgres
|
|
@@ -1166,13 +1186,13 @@ $1`),n!==r&&await l(t,n)}import{readdir as Yt,readFile as F,rm as b}from"fs/prom
|
|
|
1166
1186
|
--health-interval 10s
|
|
1167
1187
|
--health-timeout 5s
|
|
1168
1188
|
--health-retries 5
|
|
1169
|
-
`,i=t?"postgres://test:test@localhost:5432/saas_test":"postgres://postgres:postgres@localhost:5432/saas",
|
|
1189
|
+
`,i=t?"postgres://test:test@localhost:5432/saas_test":"postgres://postgres:postgres@localhost:5432/saas",a=n?`
|
|
1170
1190
|
UPSTASH_REDIS_REST_URL: https://test.upstash.io
|
|
1171
|
-
UPSTASH_REDIS_REST_TOKEN: test-token`:"",
|
|
1191
|
+
UPSTASH_REDIS_REST_TOKEN: test-token`:"",s=(()=>{switch(e.paymentProvider){case"stripe":return`
|
|
1172
1192
|
STRIPE_SECRET_KEY: test
|
|
1173
1193
|
STRIPE_WEBHOOK_SECRET: test`;case"polar":return`
|
|
1174
1194
|
POLAR_ACCESS_TOKEN: test
|
|
1175
|
-
POLAR_WEBHOOK_SECRET: test`;case"none":return"";default:{let
|
|
1195
|
+
POLAR_WEBHOOK_SECRET: test`;case"none":return"";default:{let d=e.paymentProvider;throw new Error(`buildCiYaml: unhandled payment provider "${String(d)}"`)}}})(),p=e.socialProviders.map(d=>se[d].envVars.map(v=>`
|
|
1176
1196
|
${v.name}: test`).join("")).join("");return`# Deployment is handled by the hosting platform (Vercel, Coolify, etc.)
|
|
1177
1197
|
# which auto-deploys on push. CI runs in parallel as a quality gate.
|
|
1178
1198
|
# For PR-based workflows, enable GitHub branch protection to require CI before merging.
|
|
@@ -1198,7 +1218,7 @@ jobs:
|
|
|
1198
1218
|
runs-on: ubuntu-latest
|
|
1199
1219
|
timeout-minutes: 20
|
|
1200
1220
|
${o} env:
|
|
1201
|
-
CONTENT_API_KEY: test-contentapi-key-16chars${
|
|
1221
|
+
CONTENT_API_KEY: test-contentapi-key-16chars${a}${s}${p}
|
|
1202
1222
|
STORAGE_REGION: test
|
|
1203
1223
|
STORAGE_ENDPOINT: http://test
|
|
1204
1224
|
STORAGE_ACCESS_KEY_ID: test
|
|
@@ -1211,7 +1231,7 @@ ${o} env:
|
|
|
1211
1231
|
|
|
1212
1232
|
- run: pnpm lint
|
|
1213
1233
|
|
|
1214
|
-
${
|
|
1234
|
+
${r?` # vue-tsc on web-nuxt OOMs on the GitHub runner's default heap once
|
|
1215
1235
|
# the type graph (Nuxt + Pinia + vue-i18n + content collections)
|
|
1216
1236
|
# crosses a threshold. Bump to 6 GB; ubuntu-latest has ~7 GB RAM.
|
|
1217
1237
|
- run: pnpm check-types
|
|
@@ -1222,46 +1242,46 @@ ${n?` # vue-tsc on web-nuxt OOMs on the GitHub runner's default heap once
|
|
|
1222
1242
|
run: echo "DATABASE_URL=${i}" > .env
|
|
1223
1243
|
|
|
1224
1244
|
- run: pnpm test
|
|
1225
|
-
`}import{readFile as
|
|
1226
|
-
`))}async function
|
|
1227
|
-
`))}async function
|
|
1245
|
+
`}import{readFile as Vr}from"fs/promises";import{existsSync as sa}from"fs";import{join as gn}from"path";var Fr="@repo/database";function ca(e){return e?"pnpm -F @repo/database reset && pnpm -F @repo/database run deploy -- --push":"pnpm -F @repo/database run deploy"}function la(e,t){switch(e){case"fullstack":return t==="nextjs"?"web-next":"web-nuxt";case"separate":return"backend";default:{let n=e;throw new Error(`schemaOwnerApp: unhandled architecture "${String(n)}"`)}}}async function pa(e,t,n){let r=await Vr(e,"utf-8"),o=JSON.parse(r),i=o.scripts?.[t];if(!i)throw new Error(`Cannot prepend to missing script "${t}" in ${e}`);i.includes(Fr)||(o.scripts={...o.scripts,[t]:`${n} && ${i}`},await l(e,JSON.stringify(o,null," ")+`
|
|
1246
|
+
`))}async function da(e,t){let n=sa(e)?JSON.parse(await Vr(e,"utf-8")):{$schema:"https://openapi.vercel.sh/vercel.json"},r=n.buildCommand?.trim()||"pnpm build";r.includes(Fr)||(n.buildCommand=`${t} && ${r}`,await l(e,JSON.stringify(n,null," ")+`
|
|
1247
|
+
`))}async function Br(e){let t=ca(e.demo===!0),n=la(e.architecture,e.frontend),r=gn(e.projectDir,"apps",n);switch(e.deploymentTarget){case"node":await pa(gn(r,"package.json"),"start",t);return;case"vercel":await da(gn(r,"vercel.json"),t);return;default:{let o=e.deploymentTarget;throw new Error(`generateDeployScripts: unhandled deployment target "${String(o)}"`)}}}import{readFile as ua}from"fs/promises";import{existsSync as ma}from"fs";import{join as $t}from"path";var fa=["stripe","polar"];async function Kr(e){let t=$t(e.projectDir,"packages/payments/src"),n=e.paymentProvider,r=`// Active payment provider. To switch, change the path below to
|
|
1228
1248
|
// "./polar/index" or "./none/index" (other folders are kept in place).
|
|
1229
|
-
export { ops } from "./${
|
|
1230
|
-
`;await l(
|
|
1231
|
-
`).filter(
|
|
1232
|
-
`)}import{readdir as
|
|
1233
|
-
`).filter(i=>!
|
|
1249
|
+
export { ops } from "./${n}/index";
|
|
1250
|
+
`;await l($t(t,"providers/index.ts"),r),await l($t(t,"index.ts"),await ga(t,n))}async function ga(e,t){let n=$t(e,"index.ts");if(!ma(n))throw new Error(`generatePaymentBarrel: expected ${n} to exist - did packages/payments move?`);let r=await ua(n,"utf-8"),o=fa.filter(a=>a!==t);return r.split(`
|
|
1251
|
+
`).filter(a=>!o.some(s=>a.includes(`./providers/${s}/`))).join(`
|
|
1252
|
+
`)}import{readdir as ha,readFile as ya}from"fs/promises";import{join as Gr}from"path";async function Hr(e){if(e.demo)return;let t=e.appName.trim()||e.projectName,n=JSON.stringify(t).slice(1,-1),r=Gr(e.projectDir,"packages/i18n/translations"),o;try{o=await ha(r)}catch{return}for(let i of o){let a=Gr(r,i,"web.json"),s;try{s=await ya(a,"utf-8")}catch{continue}let p=s.replaceAll("GenerateSaaS",n);p!==s&&await l(a,p)}}import{readFile as va}from"fs/promises";import{join as zr}from"path";var ka=[".nuxt/",".nuxt",".nitro/",".nitro",".output/",".output","_locales/"],wa=[".next/",".next",".svelte-kit/",".svelte-kit",".wrangler/",".wrangler",".dev.vars"];async function qr(e){let n=e.frontend==="nuxt"?wa:ka;await Yr(zr(e.projectDir,".gitignore"),n),await Yr(zr(e.projectDir,".dockerignore"),n)}async function Yr(e,t){let n;try{n=await va(e,"utf-8")}catch{return}let r=new Set(t),o=n.split(`
|
|
1253
|
+
`).filter(i=>!r.has(i.trim()));o.length!==n.split(`
|
|
1234
1254
|
`).length&&await l(e,o.join(`
|
|
1235
|
-
`))}async function
|
|
1236
|
-
`),await l(
|
|
1237
|
-
`)}import{relative as
|
|
1238
|
-
`);i(new Error(
|
|
1239
|
-
${
|
|
1240
|
-
`),
|
|
1241
|
-
`),
|
|
1242
|
-
`),H.yellow("Deployment"))}function fs(e){let t=["DATABASE_URL","BETTER_AUTH_SECRET"];return e.cacheProvider==="upstash"?t.push("UPSTASH_REDIS_REST_URL","UPSTASH_REDIS_REST_TOKEN"):t.push("REDIS_URL"),e.paymentProvider==="stripe"?t.push("STRIPE_SECRET_KEY","STRIPE_WEBHOOK_SECRET"):e.paymentProvider==="polar"&&t.push("POLAR_ACCESS_TOKEN","POLAR_WEBHOOK_SECRET"),e.emailProvider==="ses"?t.push("AMAZON_SES_REGION","AMAZON_SES_KEY","AMAZON_SES_SECRET"):e.emailProvider==="resend"?t.push("RESEND_API_KEY"):t.push("SMTP_HOST","SMTP_PORT"),t}function gs(e){switch(e.deploymentTarget){case"node":return"Deploy with Docker or your preferred Node.js host";case"vercel":return"vercel deploy # Deploy to Vercel"}}function jn(e){let t={};if(e.name!==void 0){if(!mt(e.name))throw new Error(`Invalid project name "${e.name}". Use lowercase letters, numbers, and hyphens only. Must start with a letter.`);t.projectName=e.name}if(e.appName!==void 0){if(!e.appName.trim())throw new Error("App name cannot be empty.");t.appName=e.appName}if(e.location!==void 0?t.projectDir=e.location==="."?process.cwd():e.location:t.projectName!==void 0&&(t.projectDir=`./${t.projectName}`),e.frontend!==void 0){if(!ye.includes(e.frontend))throw new Error(`Invalid frontend "${e.frontend}". Valid values: ${ye.join(", ")}`);t.frontend=e.frontend}if(e.architecture!==void 0&&(t.architecture=e.architecture),e.payment!==void 0&&(t.paymentProvider=e.payment),e.email!==void 0&&(t.emailProvider=e.email),e.org!==void 0&&(t.multiTenancy=e.org),e.billingScope!==void 0){if(e.org===!1)throw new Error("--billing-scope requires --org to be enabled.");t.billingScope=e.billingScope}if(e.blog!==void 0&&(t.blog=e.blog),e.docs!==void 0&&(t.docs=e.docs),e.desktop!==void 0&&(t.desktop=e.desktop),e.desktopAuto!==void 0&&(t.desktopAutoRelease=e.desktopAuto),e.desktopAi!==void 0){if(e.desktopAi===!0&&e.desktop!==!0)throw new Error("--desktop-ai requires --desktop to be enabled.");t.desktopAi=e.desktopAi}if(e.revenueSharing!==void 0&&(t.revenueSharing=e.revenueSharing),e.credits!==void 0){if(e.credits===!0&&e.payment==="none")throw new Error("--credits requires a payment provider (got --payment none).");t.credits=e.credits}if(e.docker!==void 0&&(t.dockerServices=tr(e.docker,xe,"docker service")),e.aiTools!==void 0&&(t.aiTools=tr(e.aiTools,De,"AI tool")),e.socialProviders!==void 0&&(t.socialProviders=tr(e.socialProviders,Ne,"social provider")),e.currency!==void 0){if(!oe.includes(e.currency))throw new Error(`Invalid currency "${e.currency}". Valid values: ${oe.join(", ")}`);t.defaultCurrency=e.currency}if(e.deploy!==void 0){if(!ie.includes(e.deploy))throw new Error(`Invalid deployment target "${e.deploy}". Valid values: ${ie.join(", ")}`);t.deploymentTarget=e.deploy}if(e.database!==void 0){if(!se.includes(e.database))throw new Error(`Invalid database provider "${e.database}". Valid values: ${se.join(", ")}`);t.databaseProvider=e.database}if(e.cache!==void 0){if(!ae.includes(e.cache))throw new Error(`Invalid cache provider "${e.cache}". Valid values: ${ae.join(", ")}`);t.cacheProvider=e.cache}if(e.demo===!0&&(t.demo=!0),e.baseUrl!==void 0){let r=e.baseUrl.trim();if(r==="")throw new Error("--base-url cannot be empty. Provide an absolute URL like https://example.com.");let n;try{n=new URL(r)}catch{throw new Error(`Invalid --base-url "${e.baseUrl}". Must be an absolute URL like https://example.com.`)}if(n.protocol!=="http:"&&n.protocol!=="https:")throw new Error(`Invalid --base-url "${e.baseUrl}". Must use http or https.`);t.baseUrl=`${n.protocol}//${n.host}`}return t}var fe={projectName:"my-saas",frontend:"nextjs",architecture:"fullstack",paymentProvider:"stripe",emailProvider:"smtp",multiTenancy:!1,billingScope:"user",blog:!0,docs:!1,desktop:!1,desktopAutoRelease:!1,desktopAi:!1,revenueSharing:!1,credits:!0,dockerServices:["postgres","redis","inngest"],aiTools:[],socialProviders:[],defaultCurrency:"USD",deploymentTarget:"node",databaseProvider:"postgres",cacheProvider:"redis"};function Un(e){let t=e.projectName??fe.projectName,r=e.projectDir??`./${t}`,n=e.appName??ut(t),o=e.deploymentTarget??fe.deploymentTarget,i=Q[o]?.edgeRuntime??!1,s=e.databaseProvider??(i?"neon":fe.databaseProvider),a=e.cacheProvider??(i?"upstash":fe.cacheProvider),d=e.emailProvider??(i?"resend":fe.emailProvider),u=e.dockerServices??(i?fe.dockerServices.filter(f=>f!=="postgres"&&f!=="redis"):fe.dockerServices),m={...fe,...e,projectName:t,appName:n,projectDir:r,deploymentTarget:o,databaseProvider:s,cacheProvider:a,emailProvider:d,dockerServices:u};m.paymentProvider==="none"&&(m.credits=!1);for(let f of Xe){if(m.deploymentTarget!==f.target)continue;let v=m.databaseProvider===f.provider?"database":"cache";if(m.databaseProvider===f.provider||m.cacheProvider===f.provider)throw new Error(`Incompatible: --deploy ${f.target} + --${v} ${f.provider}. ${f.reason}`)}for(let f of Ze)if(m.architecture===f.architecture&&m.deploymentTarget===f.target)throw new Error(`Incompatible: --architecture ${f.architecture} + --deploy ${f.target}. ${f.reason}`);return m}function tr(e,t,r){if(e.trim()==="")return[];let n=e.split(",").map(i=>i.trim()).filter(Boolean),o=n.filter(i=>!t.includes(i));if(o.length>0)throw new Error(`Invalid ${r}(s): ${o.join(", ")}. Valid values: ${t.join(", ")}`);return n}import Es from"picocolors";var bs="a10a6fb9d7cadde32e37dad52059d17b5d2b916b08c76d8fbcc99982e9a3d87f";async function ws(e){let t=await crypto.subtle.digest("SHA-256",new TextEncoder().encode(`generatesaas-demo:${e}`)),r=[...new Uint8Array(t).subarray(0,16)].map(n=>n.toString(16).padStart(2,"0")).join("");return`${r.slice(0,8)}-${r.slice(8,12)}-${r.slice(12,16)}-${r.slice(16,20)}-${r.slice(20,32)}`}function As(e){if(e===void 0)return;let t=e.trim().replace(/^v/,"");if(!/^\d+\.\d+\.\d+$/.test(t))throw new Error(`Invalid template version "${e}". Use semver like 1.2.3.`);return t}function Mn(e){e.command("init").description("Scaffold a new GenerateSaaS project").argument("[apiKey]","license key (same as --api-key)").option("-n, --name <name>","project name (lowercase, hyphens, starts with letter)").option("--app-name <name>","display name for the app").option("-l, --location <path>","project directory (default: ./{name})").addOption(new Y("--frontend <type>","frontend framework").choices([...ye])).addOption(new Y("--architecture <type>","fullstack or separate").choices([...Re])).addOption(new Y("--payment <provider>","payment provider").choices([..._e])).addOption(new Y("--email <provider>","email provider").choices([...Oe])).option("--org","enable multi-tenancy (organizations)").option("--no-org","disable multi-tenancy").addOption(new Y("--billing-scope <scope>","billing scope (requires --org)").choices([...Ce])).option("--blog","enable blog").option("--no-blog","disable blog").option("--docs","include the docs app (apps/docs, Fumadocs)").option("--no-docs","exclude the docs app").option("--desktop","include the Electron desktop app (apps/desktop)").option("--no-desktop","exclude the Electron desktop app").option("--desktop-auto","auto-trigger the desktop release workflow on CI success (default: manual-only)").option("--no-desktop-auto","manual-only desktop releases (workflow_dispatch)").option("--desktop-ai","include the desktop AI orchestration layer (requires --desktop)").option("--no-desktop-ai","exclude the desktop AI orchestration layer").option("--revenue-sharing","enable revenue sharing").option("--no-revenue-sharing","disable revenue sharing").option("--credits","enable credits system").option("--no-credits","disable credits system (subscription-only)").option("--docker <services>","comma-separated: postgres,redis,inngest,mailpit").option("--ai-tools <tools>","comma-separated: claude-code,cursor,codex,gemini-cli,windsurf").option("--social-providers <providers>","comma-separated: google,github,facebook,discord,x").addOption(new Y("--currency <code>","default currency for billing").choices([...oe])).addOption(new Y("--deploy <target>","deployment target").choices([...ie])).addOption(new Y("--database <provider>","database provider").choices([...se])).addOption(new Y("--cache <provider>","cache provider").choices([...ae])).option("--template-version <version>","specific template version to scaffold").option("--api-key <key>","API key (skips interactive prompt)").option("--base-url <url>","public base URL (e.g. https://example.com) - bakes into canonical/og/sitemap").option("-y, --yes","accept defaults for unspecified options (non-interactive)").addOption(new Y("--demo","first-party demo build: keep sample content, mark site non-indexable - requires CI API key").hideHelp()).addOption(new Y("--no-db-migration","skip generating the baseline DB migration (internal: demos/CI/playground)").hideHelp()).action(async(t,r)=>{await Is(t?{...r,apiKey:t}:r)})}async function Is(e){let t=performance.now();ar("1.19.1");let r,n;try{r=jn(e),n=As(e.templateVersion)}catch(S){w.cancel(O(S)),process.exit(1)}let o=w.spinner(),i;try{i=await Le({apiKey:e.apiKey,prompt:!e.yes})}catch(S){w.cancel(O(S)),process.exit(1)}e.demo&&Jt(i)!==bs&&(w.cancel("--demo is restricted to first-party demo deployments."),process.exit(1));let s=le(i),a=async()=>{let S=await me(s),T=S.latest,N=n??T;if(n&&!S.versions.some(K=>K.version===N))throw new Error(`Template version "${n}" is not available.`);return{latestVersion:T,selectedVersion:N,desktopAllowed:S.entitlements?.desktopAllowed??!0}};o.start("Verifying access...");let d,u,m=!0;try{({latestVersion:d,selectedVersion:u,desktopAllowed:m}=await a()),o.stop("Access verified."),Se(i)}catch(S){if(o.stop("Access verification failed."),S instanceof L&&S.status===401){e.yes&&(w.cancel("Invalid API key. Cannot prompt in non-interactive mode."),process.exit(1)),w.log.warning("Invalid API key."),i=await tt(),s=le(i),o.start("Verifying access...");try{({latestVersion:d,selectedVersion:u,desktopAllowed:m}=await a()),o.stop("Access verified."),Se(i)}catch(T){o.stop("Access verification failed."),w.cancel(T instanceof L&&T.status===401?"Invalid API key.":O(T)),process.exit(1)}}else w.cancel(O(S)),process.exit(1)}w.log.success(`Latest version: ${d}`),u!==d&&w.log.success(`Using template version: ${u}`),r.desktop===!0&&!m&&(w.cancel("The desktop app is available on the Pro plan and up. Upgrade your plan at generatesaas.com."),process.exit(1));let f;e.yes?f=Un(m?r:{...r,desktop:!1}):f=await pr(r,{desktopAllowed:m});let v;o.start("Activating license...");try{let S=e.demo&&f.baseUrl?await ws(f.baseUrl):crypto.randomUUID(),T=()=>({frontend:f.frontend,version:u,installId:S,projectName:f.projectName,options:gr(f)}),N;try{N=await Lt(s,T())}catch(K){let A=ht(K);if(!A?.lastAllowedVersion)throw K;o.stop("License activation failed."),e.yes&&(w.cancel(`${A.message} Re-run with --template-version ${A.lastAllowedVersion}.`),process.exit(1));let Z=await w.confirm({message:`Your update window has ended. Continue with v${A.lastAllowedVersion} (the last version your license covers)?`});(w.isCancel(Z)||!Z)&&(w.cancel("Setup cancelled."),process.exit(0)),u=A.lastAllowedVersion,o.start(`Activating license for v${u}...`),N=await Lt(s,T())}v={token:N.token,keyHash:Jt(i),installId:N.installId??S},o.stop("License activated.")}catch(S){o.stop("License activation failed."),w.cancel(O(S)),process.exit(1)}let E=ks(f.projectDir);if(hs(E)&&ys(E).length>0)if(e.yes)w.log.info(`Directory ${E} is not empty. Merging (keeping existing files, overwriting conflicts).`);else{let T=await w.select({message:`Directory ${E} is not empty.`,options:[{value:"merge",label:"Merge",hint:"keep existing files, overwrite conflicts"},{value:"overwrite",label:"Overwrite",hint:"delete everything and start fresh"},{value:"cancel",label:"Cancel"}]});(w.isCancel(T)||T==="cancel")&&(w.cancel("Setup cancelled."),process.exit(0)),T==="overwrite"&&vs(E,{recursive:!0,force:!0})}let P={...f,projectDir:E,version:u,...e.demo?{docs:!1}:{}};o.start("Downloading template...");try{await yt(s,u,E),o.stop("Template downloaded.")}catch(S){o.stop("Download failed."),w.cancel(O(S)),process.exit(1)}let G;o.start("Generating project files...");try{if({dockerComposeGenerated:G}=await It(P),!e.demo){let S=await Ke(E);await l(Ss(E,ft),JSON.stringify(S,null," ")+`
|
|
1243
|
-
`),await
|
|
1244
|
-
`),
|
|
1245
|
-
`);let
|
|
1246
|
-
`)}if(
|
|
1247
|
-
`),
|
|
1248
|
-
`),
|
|
1249
|
-
|
|
1250
|
-
${
|
|
1251
|
-
|
|
1252
|
-
`),title:`Changelog v${
|
|
1253
|
-
`);
|
|
1254
|
-
`),
|
|
1255
|
-
`).map(
|
|
1255
|
+
`))}async function jt(e){let t=e.projectDir;e.demo||(await Wn(t),await Xn(t),await Zn(t)),await Qn(t,e.aiTools),await er(t,e.frontend),await nr(e),await rr(e),await or(e),e.demo||await ir(e),await cr(e);let n=await lr(e);return await pr(e),await dr(e),await ur(e),await mr(e),await fr(e),await gr(e),await kr(e),await wr(e),await Er(e),await Ir(e),await _r(e),await Mr(e),await Or(e),await xr(e),await Cr(e),await Dr(e),await Nr(e),await Lr(e),await $r(e),await Ur(e),await ln(e),await dn(e),await mn(e),await Br(e),await Kr(e),await Hr(e),await qr(e),{dockerComposeGenerated:n}}import{basename as Wr,join as Xr,relative as Sa}from"path";import{createHash as Jr}from"crypto";import{readFile as ba}from"fs/promises";async function Ut(e){let t=await ba(e);return Jr("sha256").update(t).digest("hex")}function hn(e){return Jr("sha256").update(e).digest("hex")}var Ea=new Set(["data",q]);function Aa(e){let t=e.split("/");for(let n of t)if(Xt.has(n)||Zt.has(n)||Ea.has(n)||n.startsWith(".env")&&!n.includes("example"))return!0;return!1}function Zr(e,t){return{projectName:e.projectName??Wr(t),appName:e.appName??e.projectName??Wr(t),projectDir:t,frontend:e.frontend==="nextjs"?"nextjs":"nuxt",architecture:e.architecture??"fullstack",paymentProvider:e.paymentProvider??"none",emailProvider:e.emailProvider??"smtp",multiTenancy:e.multiTenancy??!1,billingScope:e.billingScope??"user",blog:e.blog??!1,docs:e.docs??!1,desktop:e.desktop??!1,desktopAutoRelease:e.desktopAutoRelease??!0,desktopAi:e.desktopAi??!1,ai:e.ai??!1,rag:e.rag??!1,companion:e.companion??!1,mcpServer:e.mcpServer??!1,aiByok:e.aiByok??!0,revenueSharing:e.revenueSharing??!1,credits:e.credits??!1,dockerServices:e.dockerServices??[],aiTools:e.aiTools??[],socialProviders:e.socialProviders??[],defaultCurrency:e.defaultCurrency??"USD",deploymentTarget:e.deploymentTarget??"node",databaseProvider:e.databaseProvider??"postgres",cacheProvider:e.cacheProvider??"redis",version:e.version,baseUrl:e.baseUrl,credentials:{},demo:!1}}function Ia(e,t){return{version:e.version,initialVersion:e.version,repo:"Duzbee/GenerateSaaS",appName:e.appName,projectName:e.projectName,frontend:e.frontend,architecture:e.architecture,paymentProvider:e.paymentProvider,emailProvider:e.emailProvider,multiTenancy:e.multiTenancy,billingScope:e.billingScope,blog:e.blog,docs:e.docs,desktop:e.desktop,desktopAutoRelease:e.desktopAutoRelease,desktopAi:e.desktopAi,ai:e.ai,rag:e.rag,companion:e.companion,mcpServer:e.mcpServer,aiByok:e.aiByok,credits:e.credits,revenueSharing:e.revenueSharing,defaultCurrency:e.defaultCurrency,dockerServices:e.dockerServices,socialProviders:e.socialProviders,deploymentTarget:e.deploymentTarget,databaseProvider:e.databaseProvider,cacheProvider:e.cacheProvider,aiTools:e.aiTools,...e.baseUrl?{baseUrl:e.baseUrl}:{},...t&&{licenseToken:t.token,licenseKeyHash:t.keyHash,installId:t.installId}}}async function Qr(e,t){let r=(await Te(e.projectDir,e.projectDir,Aa)).sort(),o=await Promise.all(r.map(async s=>[Pe(Sa(e.projectDir,s)),await Ut(s)])),i=Object.fromEntries(o),a=Ia(e,t);await l(Xr(e.projectDir,he),JSON.stringify(a,null," ")+`
|
|
1256
|
+
`),await l(Xr(e.projectDir,Mn),JSON.stringify(i,null," ")+`
|
|
1257
|
+
`)}import{relative as Pa}from"path";async function We(e){let n=(await Te(e,e,Be)).sort(),r=await Promise.all(n.map(async o=>[Pe(Pa(e,o)),await Ut(o)]));return Object.fromEntries(r)}import{copyFile as Ta,mkdir as Ra,rm as _a}from"fs/promises";import{dirname as Oa,join as eo,relative as xa}from"path";import{existsSync as Ca}from"fs";async function yn(e,t){Ca(t)&&await _a(t,{recursive:!0,force:!0});let n=await Te(e,e,Be);for(let r of n){let o=Pe(xa(e,r)),i=eo(t,o);await Ra(Oa(i),{recursive:!0}),await Ta(r,i)}}async function to(e,t){await yn(e,eo(t,St))}import{existsSync as Da}from"fs";import{readFile as no,readdir as Na}from"fs/promises";import{join as ie,dirname as La,resolve as $a,sep as ja}from"path";import{fileURLToPath as Ua}from"url";var pt={"claude-code":".claude/skills",cursor:".cursor/skills",codex:".agents/skills","gemini-cli":".gemini/skills",windsurf:".windsurf/skills"},Id=Object.values(pt),vn="generatesaas-update",ro=La(Ua(import.meta.url));function Ma(){let e=ie(ro,"skill","content");return Da(e)?e:ie(ro,"content")}function kn(e){return!e||e.length===0?[]:e.map(t=>pt[t])}async function wn(e,t,n,r){let o=kn(r);for(let i of o){let a=ie(e,i,vn),s=ie(a,"scripts"),p=ie(a,"references");await Pt(s),await Pt(p),await l(ie(a,"SKILL.md"),t.replaceAll("__SKILL_ROOT__",i)),await l(ie(p,".gitkeep"),"");for(let[f,m]of Object.entries(n)){let d=$a(s,f);d.startsWith(s+ja)&&await l(d,m)}}}async function oo(e,t){let n=Ma(),r=await no(ie(n,"SKILL.md"),"utf-8"),o=ie(n,"scripts"),i=await Na(o),a={};for(let s of i)s!==".gitkeep"&&(a[s]=await no(ie(o,s),"utf-8"));await wn(e,r,a,t)}import{execFile as Va,execFileSync as Fa}from"child_process";import{access as io,readFile as Ba}from"fs/promises";import{join as bn}from"path";import*as D from"@clack/prompts";function Oe(e){try{let t=process.platform==="win32"?"where":"which";return Fa(t,[e],{stdio:"ignore"}),!0}catch{return!1}}function _e(e,t,n,r=3e5){return new Promise((o,i)=>{Va(e,t,{cwd:n,timeout:r,maxBuffer:50*1024*1024},(a,s,p)=>{if(a){let f=String(s||"").trim(),d=[String(p||"").trim(),f].filter(Boolean).join(`
|
|
1258
|
+
`);i(new Error(d?`${a.message}
|
|
1259
|
+
${d}`:a.message))}else o()})})}async function ao(e){if(!Oe("pnpm"))return D.log.warn("pnpm not found. Skipping lockfile regeneration."),!1;try{return await _e("pnpm",["install","--lockfile-only","--no-frozen-lockfile","--config.minimumReleaseAge=0"],e),!0}catch(t){let n=t instanceof Error?t.message:String(t);return D.log.warn(`Lockfile regeneration failed: ${n}`),D.log.warn("Deploys using --frozen-lockfile may fail."),!1}}async function so(e){if(!Oe("pnpm"))return D.log.warn("pnpm not found. Skipping dependency installation."),D.log.info("Install pnpm: https://pnpm.io/installation"),!1;let t=D.spinner();t.start("Installing dependencies (this may take a minute)...");try{return await _e("pnpm",["install","--config.minimumReleaseAge=0"],e),t.stop("Dependencies installed."),!0}catch(n){t.stop("Dependency installation failed.");let r=n instanceof Error?n.message:String(n);return D.log.warn(`pnpm install failed: ${r}`),D.log.warn("You can run it manually later."),!1}}async function co(e){if(!Oe("pnpm"))return!1;let t=D.spinner();t.start("Generating baseline database migration...");try{return await _e("pnpm",["-F","@repo/database","generate"],e),t.stop("Baseline migration generated."),!0}catch(n){t.stop("Baseline migration generation failed.");let r=n instanceof Error?n.message:String(n);return D.log.warn(`Could not generate baseline migration: ${r}`),D.log.warn("Run 'pnpm -F @repo/database generate' before your first deploy."),!1}}async function lo(e){try{return await io(bn(e,".git")),D.log.info("Git repository already exists, skipping init."),!0}catch{}if(!Oe("git"))return D.log.warn("git not found. Skipping repository initialization."),!1;let t=D.spinner();t.start("Initializing git repository...");try{return await _e("git",["init"],e),await _e("git",["add","-A"],e),await _e("git",["commit","--no-verify","-m","Initial commit from GenerateSaaS"],e),t.stop("Git repository initialized."),!0}catch{return t.stop("Git initialization failed."),D.log.warn("You can run git init manually later."),!1}}async function po(e){if(!Oe("pnpm"))return!1;try{await io(bn(e,".git"))}catch{return!1}try{let t=JSON.parse(await Ba(bn(e,"package.json"),"utf-8")),n=!!t.devDependencies?.["simple-git-hooks"],r=!!t["simple-git-hooks"];if(!n||!r)return!1}catch{return!1}try{return await _e("pnpm",["exec","simple-git-hooks"],e),!0}catch{return D.log.warn("Could not install git hooks. Run 'pnpm exec simple-git-hooks' manually."),!1}}import*as Xe from"@clack/prompts";import Z from"picocolors";function uo(e,t){t.dockerComposeGenerated&&!t.dockerAvailable&&Xe.log.warn("Docker not found. Install Docker to run local services: https://docs.docker.com/get-docker/");let n=[];if(n.push(`cd ${e.projectDir}`),t.pnpmInstalled||n.push("pnpm install"),t.dockerComposeGenerated){let i=e.dockerServices.map(a=>De[a].label).join(", ");n.push(`pnpm infra ${Z.dim(`# ${i}`)}`)}if(n.push(`pnpm dev ${Z.dim("# http://localhost:3000")}`),t.skippedCredentials.length>0&&(n.push(""),n.push(Z.dim("Fill in remaining TODO values in .env"))),Xe.note(n.join(`
|
|
1260
|
+
`),Z.yellow("Start Development")),t.dockerComposeGenerated){let i=[];i.push(`App ${Z.cyan("http://localhost:3000")}`),e.architecture==="separate"&&i.push(`API ${Z.cyan("http://localhost:3010")}`),e.dockerServices.includes("mailpit")&&i.push(`Mailpit ${Z.cyan("http://localhost:8025")}`),e.dockerServices.includes("inngest")&&i.push(`Inngest ${Z.cyan("http://localhost:8288")}`),Xe.note(i.join(`
|
|
1261
|
+
`),Z.yellow("Dev Tools"))}let r=[],o=Ka(e);o.length>0&&r.push(`Set in production: ${Z.dim(o.join(", "))}`),r.push("pnpm db:setup # Enable pgvector + apply the database schema"),r.push(Ga(e)),Xe.note(r.join(`
|
|
1262
|
+
`),Z.yellow("Deployment"))}function Ka(e){let t=["DATABASE_URL","BETTER_AUTH_SECRET"];return e.cacheProvider==="upstash"?t.push("UPSTASH_REDIS_REST_URL","UPSTASH_REDIS_REST_TOKEN"):t.push("REDIS_URL"),e.paymentProvider==="stripe"?t.push("STRIPE_SECRET_KEY","STRIPE_WEBHOOK_SECRET"):e.paymentProvider==="polar"&&t.push("POLAR_ACCESS_TOKEN","POLAR_WEBHOOK_SECRET"),e.emailProvider==="ses"?t.push("AMAZON_SES_REGION","AMAZON_SES_KEY","AMAZON_SES_SECRET"):e.emailProvider==="resend"?t.push("RESEND_API_KEY"):t.push("SMTP_HOST","SMTP_PORT"),t}function Ga(e){switch(e.deploymentTarget){case"node":return"Deploy with Docker or your preferred Node.js host";case"vercel":return"vercel deploy # Deploy to Vercel"}}import Wa from"picocolors";var Xa="a10a6fb9d7cadde32e37dad52059d17b5d2b916b08c76d8fbcc99982e9a3d87f";async function Za(e){let t=await crypto.subtle.digest("SHA-256",new TextEncoder().encode(`generatesaas-demo:${e}`)),n=[...new Uint8Array(t).subarray(0,16)].map(r=>r.toString(16).padStart(2,"0")).join("");return`${n.slice(0,8)}-${n.slice(8,12)}-${n.slice(12,16)}-${n.slice(16,20)}-${n.slice(20,32)}`}function Qa(e){if(e===void 0)return;let t=e.trim().replace(/^v/,"");if(!/^\d+\.\d+\.\d+$/.test(t))throw new Error(`Invalid template version "${e}". Use semver like 1.2.3.`);return t}function mo(e){e.command("init").description("Scaffold a new GenerateSaaS project").argument("[apiKey]","license key (same as --api-key)").option("-n, --name <name>","project name (lowercase, hyphens, starts with letter)").option("--app-name <name>","display name for the app").option("-l, --location <path>","project directory (default: ./{name})").addOption(new Q("--frontend <type>","frontend framework").choices([...Ee])).addOption(new Q("--architecture <type>","fullstack or separate").choices([...Ne])).addOption(new Q("--payment <provider>","payment provider").choices([...Le])).addOption(new Q("--email <provider>","email provider").choices([...$e])).option("--org","enable multi-tenancy (organizations)").option("--no-org","disable multi-tenancy").addOption(new Q("--billing-scope <scope>","billing scope (requires --org)").choices([...Me])).option("--blog","enable blog").option("--no-blog","disable blog").option("--docs","include the docs app (apps/docs, Fumadocs)").option("--no-docs","exclude the docs app").option("--desktop","include the Electron desktop app (apps/desktop)").option("--no-desktop","exclude the Electron desktop app").option("--desktop-auto","auto-trigger the desktop release workflow on CI success (default: manual-only)").option("--no-desktop-auto","manual-only desktop releases (workflow_dispatch)").option("--desktop-ai","include the desktop AI orchestration layer (requires --desktop)").option("--no-desktop-ai","exclude the desktop AI orchestration layer").option("--ai","include the AI features (chat, models, schedules, integrations); unlocks the companion + MCP server").option("--no-ai","exclude the AI features").option("--rag","include server-side RAG knowledge (pgvector embeddings + semantic search)").option("--no-rag","exclude server-side RAG knowledge").option("--companion","include the companion daemon (apps/companion) + its HTTP transport").option("--no-companion","exclude the companion daemon").option("--mcp-server","include the outward MCP server (the /mcp route + Better Auth mcp() provider)").option("--no-mcp-server","exclude the outward MCP server").option("--ai-byok","AI BYOK mode: end-users bring their own provider key (default)").option("--no-ai-byok","operator-keyed, credit-metered AI platform mode").option("--revenue-sharing","enable revenue sharing").option("--no-revenue-sharing","disable revenue sharing").option("--credits","enable credits system").option("--no-credits","disable credits system (subscription-only)").option("--docker <services>","comma-separated: postgres,redis,inngest,mailpit").option("--ai-tools <tools>","comma-separated: claude-code,cursor,codex,gemini-cli,windsurf").option("--social-providers <providers>","comma-separated: google,github,facebook,discord,x").addOption(new Q("--currency <code>","default currency for billing").choices([...ce])).addOption(new Q("--deploy <target>","deployment target").choices([...le])).addOption(new Q("--database <provider>","database provider").choices([...pe])).addOption(new Q("--cache <provider>","cache provider").choices([...de])).option("--template-version <version>","specific template version to scaffold").option("--api-key <key>","API key (skips interactive prompt)").option("--base-url <url>","public base URL (e.g. https://example.com) - bakes into canonical/og/sitemap").option("-y, --yes","accept defaults for unspecified options (non-interactive)").addOption(new Q("--demo","first-party demo build: keep sample content, mark site non-indexable - requires CI API key").hideHelp()).addOption(new Q("--no-db-migration","skip generating the baseline DB migration (internal: demos/CI/playground)").hideHelp()).action(async(t,n)=>{await es(t?{...n,apiKey:t}:n)})}async function es(e){let t=performance.now();xn("1.20.0");let n,r;try{n=$n(e),r=Qa(e.templateVersion)}catch(k){A.cancel(C(k)),process.exit(1)}let o=A.spinner(),i;try{i=await Fe({apiKey:e.apiKey,prompt:!e.yes})}catch(k){A.cancel(C(k)),process.exit(1)}e.demo&&hn(i)!==Xa&&(A.cancel("--demo is restricted to first-party demo deployments."),process.exit(1));let a=fe(i),s=async()=>{let k=await ke(a),R=k.latest,P=r??R;if(r&&!k.versions.some(ae=>ae.version===P))throw new Error(`Template version "${r}" is not available.`);return{latestVersion:R,selectedVersion:P,desktopAllowed:k.entitlements?.desktopAllowed??!0}};o.start("Verifying access...");let p,f,m=!0;try{({latestVersion:p,selectedVersion:f,desktopAllowed:m}=await s()),o.stop("Access verified."),Ie(i)}catch(k){if(o.stop("Access verification failed."),k instanceof U&&k.status===401){e.yes&&(A.cancel("Invalid API key. Cannot prompt in non-interactive mode."),process.exit(1)),A.log.warning("Invalid API key."),i=await st(),a=fe(i),o.start("Verifying access...");try{({latestVersion:p,selectedVersion:f,desktopAllowed:m}=await s()),o.stop("Access verified."),Ie(i)}catch(R){o.stop("Access verification failed."),A.cancel(R instanceof U&&R.status===401?"Invalid API key.":C(R)),process.exit(1)}}else A.cancel(C(k)),process.exit(1)}A.log.success(`Latest version: ${p}`),f!==p&&A.log.success(`Using template version: ${f}`),n.desktop===!0&&!m&&(A.cancel("The desktop app is available on the Pro plan and up. Upgrade your plan at generatesaas.com."),process.exit(1));let d;e.yes?d=jn(m?n:{...n,desktop:!1}):d=await Un(n,{desktopAllowed:m});let v;o.start("Activating license...");try{let k=e.demo&&d.baseUrl?await Za(d.baseUrl):crypto.randomUUID(),R=()=>({frontend:d.frontend,version:f,installId:k,projectName:d.projectName,options:Kn(d)}),P;try{P=await Jt(a,R())}catch(ae){let $=Et(ae);if(!$?.lastAllowedVersion)throw ae;o.stop("License activation failed."),e.yes&&(A.cancel(`${$.message} Re-run with --template-version ${$.lastAllowedVersion}.`),process.exit(1));let T=await A.confirm({message:`Your update window has ended. Continue with v${$.lastAllowedVersion} (the last version your license covers)?`});(A.isCancel(T)||!T)&&(A.cancel("Setup cancelled."),process.exit(0)),f=$.lastAllowedVersion,o.start(`Activating license for v${f}...`),P=await Jt(a,R())}v={token:P.token,keyHash:hn(i),installId:P.installId??k},o.stop("License activated.")}catch(k){o.stop("License activation failed."),A.cancel(C(k)),process.exit(1)}let b=Ja(d.projectDir);if(Ha(b)&&za(b).length>0)if(e.yes)A.log.info(`Directory ${b} is not empty. Merging (keeping existing files, overwriting conflicts).`);else{let R=await A.select({message:`Directory ${b} is not empty.`,options:[{value:"merge",label:"Merge",hint:"keep existing files, overwrite conflicts"},{value:"overwrite",label:"Overwrite",hint:"delete everything and start fresh"},{value:"cancel",label:"Cancel"}]});(A.isCancel(R)||R==="cancel")&&(A.cancel("Setup cancelled."),process.exit(0)),R==="overwrite"&&Ya(b,{recursive:!0,force:!0})}let I={...d,projectDir:b,version:f,...e.demo?{docs:!1}:{}};o.start("Downloading template...");try{await At(a,f,b),o.stop("Template downloaded.")}catch(k){o.stop("Download failed."),A.cancel(C(k)),process.exit(1)}let z;o.start("Generating project files...");try{if({dockerComposeGenerated:z}=await jt(I),!e.demo){let k=await We(b);await l(qa(b,bt),JSON.stringify(k,null," ")+`
|
|
1263
|
+
`),await to(b,b)}await oo(b,I.aiTools),await Qr(I,v),o.stop("Project files generated.")}catch(k){o.stop("Generation failed."),A.cancel(C(k)),process.exit(1)}await ao(b);let L=await so(b);L&&I.demo!==!0&&e.dbMigration!==!1&&await co(b),await lo(b),L&&await po(b);let K=Oe("docker"),Y=Yt(I).map(k=>k.key).filter(k=>!I.credentials?.[k]);uo(I,{pnpmInstalled:L,dockerComposeGenerated:z,dockerAvailable:K,skippedCredentials:Y}),Cn(),A.log.info(Wa.dim(`Done in ${((performance.now()-t)/1e3).toFixed(1)}s`))}import{existsSync as vo}from"fs";import{readFile as ko}from"fs/promises";import{join as dt,resolve as as}from"path";import*as N from"@clack/prompts";import Ze from"picocolors";import{mkdtemp as ts,rm as ns}from"fs/promises";import{tmpdir as rs}from"os";import{join as os}from"path";async function Sn(e,t,n,r){let o=await ts(os(rs(),"generatesaas-stage-"));try{await At(e,t,o),await jt({...n,projectDir:o}),await yn(o,r)}finally{await ns(o,{recursive:!0,force:!0})}}var fo=[{key:"frontend",label:"Frontend framework",hint:"Nuxt (Vue) or Next.js (React).",category:"Project",kind:"enum",default:"nuxt",choices:Ee,adoptable:!1,impact:"structural"},{key:"architecture",label:"Architecture",hint:"Fullstack (frontend hosts the API) or a standalone Hono backend.",category:"Infrastructure",kind:"enum",default:"fullstack",choices:Ne,adoptable:!1,impact:"structural"},{key:"deploymentTarget",label:"Deployment target",hint:"Node.js / Docker or Vercel serverless.",category:"Infrastructure",kind:"enum",default:"node",choices:le,adoptable:!1,impact:"config"},{key:"databaseProvider",label:"Database provider",hint:"Self-hosted PostgreSQL, Neon, or Supabase.",category:"Infrastructure",kind:"enum",default:"postgres",choices:pe,adoptable:!1,impact:"config"},{key:"cacheProvider",label:"Cache provider",hint:"Self-hosted Redis or Upstash.",category:"Infrastructure",kind:"enum",default:"redis",choices:de,adoptable:!1,impact:"config"},{key:"paymentProvider",label:"Payment provider",hint:"Stripe, Polar, or none.",category:"Features",kind:"enum",default:"none",choices:Le,adoptable:!1,impact:"structural"},{key:"defaultCurrency",label:"Default currency",hint:"Billing and pricing currency.",category:"Features",kind:"enum",default:"USD",choices:ce,adoptable:!1,impact:"config"},{key:"emailProvider",label:"Email provider",hint:"SMTP, Amazon SES, or Resend.",category:"Features",kind:"enum",default:"smtp",choices:$e,adoptable:!1,impact:"config"},{key:"multiTenancy",label:"Multi-tenancy (organizations)",hint:"Adds organizations - teams, members, and shared resources.",category:"Features",kind:"boolean",default:!1,adoptable:!0,impact:"structural"},{key:"billingScope",label:"Billing scope",hint:"Whether subscriptions belong to a user or an organization.",category:"Features",kind:"enum",default:"user",choices:Me,adoptable:!1,impact:"config",requires:e=>e.multiTenancy===!0,requiresLabel:"requires multi-tenancy"},{key:"blog",label:"Blog",hint:"Adds the marketing blog (content collection, list and post pages).",category:"Features",kind:"boolean",default:!1,adoptable:!0,impact:"structural"},{key:"docs",label:"Docs app",hint:"Adds the self-hosted Fumadocs documentation site (apps/docs).",category:"Features",kind:"boolean",default:!1,adoptable:!0,impact:"structural"},{key:"desktop",label:"Desktop app",hint:"Adds the cross-platform Electron desktop app (apps/desktop).",category:"Features",kind:"boolean",default:!1,adoptable:!0,impact:"structural"},{key:"desktopAutoRelease",label:"Desktop auto-release",hint:"Release the desktop app on every CI success (vs manual-only).",category:"Features",kind:"boolean",default:!0,adoptable:!1,impact:"config",requires:e=>e.desktop===!0,requiresLabel:"requires the desktop app"},{key:"desktopAi",label:"Desktop AI orchestration",hint:"Adds the desktop AI agents layer (requires the desktop app).",category:"Features",kind:"boolean",default:!1,adoptable:!0,impact:"structural",requires:e=>e.desktop===!0,requiresLabel:"requires the desktop app"},{key:"ai",label:"AI features",hint:"Adds the AI surface (chat, models, schedules, integrations) and unlocks the companion, MCP server, RAG, and BYOK options. Off keeps config.ai.enabled false so the AI nav stays hidden.",category:"Features",kind:"boolean",default:!1,adoptable:!0,impact:"config"},{key:"rag",label:"Server-side RAG (knowledge)",hint:"Adds pgvector knowledge with embeddings + semantic search.",category:"Features",kind:"boolean",default:!1,adoptable:!0,impact:"config",requires:e=>e.ai===!0,requiresLabel:"requires AI features"},{key:"companion",label:"Companion daemon",hint:"Adds the terminal companion (apps/companion) that drives the user's subscription CLIs over stateless HTTP. Works on any deployment.",category:"Features",kind:"boolean",default:!1,adoptable:!0,impact:"structural",requires:e=>e.ai===!0,requiresLabel:"requires AI features"},{key:"mcpServer",label:"Outward MCP server",hint:"Exposes the app's capability layer to external agents as an authenticated MCP OAuth server.",category:"Features",kind:"boolean",default:!1,adoptable:!0,impact:"structural",requires:e=>e.ai===!0,requiresLabel:"requires AI features"},{key:"aiByok",label:"AI billing mode (BYOK)",hint:"BYOK: end-users bring their own provider key, no credit metering. Off: operator-keyed, credit-metered platform mode.",category:"Features",kind:"boolean",default:!0,adoptable:!1,impact:"config",requires:e=>e.ai===!0,requiresLabel:"requires AI features"},{key:"credits",label:"Metered credits",hint:"Adds metered usage credits on top of subscription plans.",category:"Features",kind:"boolean",default:!1,adoptable:!0,impact:"config",requires:e=>e.paymentProvider!==void 0&&e.paymentProvider!=="none",requiresLabel:"requires a payment provider"},{key:"revenueSharing",label:"Revenue sharing",hint:"Opt-in MRR leaderboard with dofollow backlinks.",category:"Features",kind:"boolean",default:!1,adoptable:!0,impact:"config"},{key:"socialProviders",label:"Social login providers",hint:"Which social sign-in buttons the auth screen shows.",category:"Features",kind:"multiselect",default:[],choices:Ve,adoptable:!1,impact:"config"},{key:"dockerServices",label:"Docker services",hint:"Local services scaffolded in docker-compose.",category:"Tooling",kind:"multiselect",default:[],choices:je,adoptable:!1,impact:"config"},{key:"aiTools",label:"AI coding tools",hint:"Which AI assistants receive the bundled skill files.",category:"Tooling",kind:"multiselect",default:[],choices:Ue,adoptable:!1,impact:"config"}];function is(e,t){return e[t]!==void 0}function go(e){return fo.filter(t=>t.adoptable&&!is(e,t.key)&&(t.requires?t.requires(e):!0)).map(t=>({key:t.key,label:t.label,hint:t.hint,kind:t.kind,default:t.default,...t.choices?{choices:t.choices}:{},impact:t.impact,...t.requiresLabel?{requiresLabel:t.requiresLabel}:{}}))}function ho(e){let t={};for(let n of fo)t[n.key]=Array.isArray(n.default)?[...n.default]:n.default;return{...t,...e}}function yo(e){let n=(e.startsWith("v")?e.slice(1):e).match(/^(\d+)\.(\d+)\.(\d+)$/);return n?[Number(n[1]),Number(n[2]),Number(n[3])]:null}function Mt(e,t){let n=yo(e),r=yo(t);if(!n||!r)return 0;for(let o=0;o<3;o++)if(n[o]!==r[o])return n[o]-r[o];return 0}function wo(e){e.command("update").description("Update AI skill files and stage template updates").argument("[mode]","pass 'auto' to mark the staged update for unattended apply by your AI assistant").option("--cwd <path>","project directory (default: current directory)").action(async(t,n)=>{let r=t==="auto"?"auto":void 0,o=as(n.cwd??process.cwd()),i=dt(o,he),a;try{a=JSON.parse(await ko(i,"utf-8"))}catch{N.cancel(".generatesaas/manifest.json not found. Run this from a GenerateSaaS project."),process.exit(1)}let s;try{s=await Fe()}catch(m){N.cancel(C(m)),process.exit(1)}let p=fe(s),f=N.spinner();try{f.start("Verifying access...");let m;try{m=await ke(p)}catch(T){throw T instanceof U&&T.status===401?new Error("Your saved API key was rejected. Run `generatesaas auth` to update it, or set GENERATESAAS_API_KEY."):T}f.stop("Access verified."),Ie(s),f.start("Fetching latest skill files...");let d=await Bn(p,m.latest);await wn(o,d.skillMd,d.scripts,a.aiTools);let v=kn(a.aiTools);if(f.stop("Skills updated."),N.log.success(`Skill files installed to ${Ze.cyan(v.length.toString())} locations.`),a.version===m.latest){N.log.info(`Already on the latest version (${a.version}).`);return}let b=JSON.stringify(a);Ln(a);let I=go(a),z=ho(a),L=JSON.stringify(z)!==b;if(a=z,a.licenseToken)try{let T=await Gn(p,{currentToken:a.licenseToken,newVersion:m.latest});a.licenseToken=T.token,T.licenseKeyHash&&(a.licenseKeyHash=T.licenseKeyHash),await l(i,JSON.stringify(a,null," ")+`
|
|
1264
|
+
`),L=!1,N.log.success("License refreshed.")}catch(T){let we=Et(T);we&&(N.cancel(we.message),process.exit(1)),N.log.warn("License refresh skipped.")}L&&await l(i,JSON.stringify(a,null," ")+`
|
|
1265
|
+
`);let K=Zr(a,o),ge=dt(o,Vn);f.start(`Staging v${m.latest} (shaped for your config)...`),await Sn(p,m.latest,K,ge),f.stop("Template staged.");let{text:Y,title:k}=await ss(p,m,a.version);Y&&N.note(Y,k);let R=dt(o,bt),P=dt(o,St),ae=!vo(P),$=!vo(R);if(ae){if(f.start("Building baseline template (one-time migration)..."),await Sn(p,a.version,K,P),$){let T=await We(P);await l(R,JSON.stringify(T,null," ")+`
|
|
1266
|
+
`)}if(f.stop("Baseline template stored."),!$){let T=await cs(R,P);T>0&&N.log.warn(`Rebuilt baseline differs from the original for ${T} file(s) (the CLI's shaping evolved since this project was scaffolded). Classification still follows the committed template-hashes.json; upstream diffs for those files may include unrelated noise.`)}}else if($){f.start("Computing baseline template hashes...");let T=await We(P);await l(R,JSON.stringify(T,null," ")+`
|
|
1267
|
+
`),f.stop("Baseline hashes computed.")}if(await l(dt(o,Fn),JSON.stringify({currentVersion:a.version,targetVersion:m.latest,changelog:Y,stagedAt:new Date().toISOString(),...I.length>0?{newOptions:I}:{},...r?{mode:r}:{}},null," ")+`
|
|
1268
|
+
`),N.log.info(`Update staged: ${Ze.cyan(a.version)} \u2192 ${Ze.cyan(m.latest)}`),a.aiTools&&a.aiTools.length>0){let T=a.aiTools[0],we=nt[T].label;N.log.info(`Open your project in ${Ze.cyan(we)} and ask: ${Ze.cyan("'update my GenerateSaaS project'")}`)}else N.log.info(`Ask your AI coding assistant to ${Ze.cyan("'update my GenerateSaaS project'")}.`)}catch(m){f.stop("Failed."),N.cancel(`Update failed: ${C(m)}`),process.exit(1)}})}async function ss(e,t,n){let r=t.latest,o=t.versions.filter(a=>Mt(a.version,n)>0&&Mt(a.version,r)<=0).sort((a,s)=>Mt(a.version,s.version));if(o.length<=1)return{text:await qt(e,r),title:`Changelog v${r}`};let i=[];for(let a of o){let s=null;try{s=await qt(e,a.version)}catch{s=null}let p=a.date?` (${a.date.slice(0,10)})`:"",f=a.breaking?" [BREAKING]":"";i.push(`# v${a.version}${p}${f}
|
|
1269
|
+
|
|
1270
|
+
${s??"_No changelog available for this release._"}`)}return{text:i.join(`
|
|
1271
|
+
|
|
1272
|
+
`),title:`Changelog v${n} \u2192 v${r}`}}async function cs(e,t){let n=JSON.parse(await ko(e,"utf-8")),r=await We(t),o=0;for(let[i,a]of Object.entries(n))Be(i)||r[i]!==a&&o++;for(let i of Object.keys(r))i in n||o++;return o}import*as H from"@clack/prompts";import ee from"picocolors";import{readFile as ls}from"fs/promises";import{join as ps,resolve as ds}from"path";function bo(e){e.command("status").description("Show project status and check for updates").option("--cwd <path>","project directory (default: current directory)").action(async t=>{let n=ds(t.cwd??process.cwd()),r=ps(n,he),o;try{o=JSON.parse(await ls(r,"utf-8"))}catch{H.cancel(".generatesaas/manifest.json not found. Run this from a GenerateSaaS project."),process.exit(1)}let i=[`Version: ${ee.cyan(o.version)}`,`Frontend: ${ee.cyan(o.frontend)}`,o.deploymentTarget?`Deploy target: ${ee.cyan(o.deploymentTarget)}`:null,o.databaseProvider?`Database: ${ee.cyan(o.databaseProvider)}`:null,o.cacheProvider?`Cache: ${ee.cyan(o.cacheProvider)}`:null,o.aiTools&&o.aiTools.length>0?`AI tools: ${ee.cyan(o.aiTools.join(", "))}`:null].filter(Boolean).join(`
|
|
1273
|
+
`);H.note(i,ee.bold("Project Status"));let a=H.spinner();a.start("Checking for updates...");try{let s=await Fe(),p=fe(s),m=(await ke(p)).latest;o.version===m?(a.stop("Up to date."),H.log.success(`Already on the latest version (${ee.green(m)})`)):(a.stop("Update available."),H.log.warning(`Update available: ${ee.yellow(o.version)} \u2192 ${ee.green(m)}`),H.log.info(`Open this project in your AI coding agent and ask it to ${ee.cyan("update my GenerateSaaS project")} - it fetches and applies the update for you.`))}catch(s){a.stop("Check failed."),s instanceof U&&s.status===401?H.log.warning("Invalid API key. Run `generatesaas auth` to update it, or set GENERATESAAS_API_KEY."):H.log.warning(`Could not check for updates: ${C(s)}`)}})}import{readFile as us}from"fs/promises";import*as x from"@clack/prompts";import O from"picocolors";function ms(){return process.env.GENERATESAAS_API_KEY??at()}function fs(e){return{verdict:e.verdict,plan:e.license?.plan??e.domainInstalls.find(t=>t.ownerPlan)?.ownerPlan??null,mismatchDomain:e.install?.domain??null,ejectedAt:e.install?.ejectedAt??e.domainInstalls.find(t=>t.ejectedAt)?.ejectedAt??null}}function gs(e){return{verdict:e.verdict??"unknown",ejectedAt:e.ejectedAt??null}}function hs(e){switch(e.verdict){case"licensed":return x.log.success(`${O.green("LICENSED")} - resolves to an account with an active${e.plan?` ${e.plan}`:""} license.`),!0;case"ejected":return x.log.success(`${O.green("EJECTED")} - a licensed buyer opted this install out of telemetry${e.ejectedAt?` on ${e.ejectedAt.slice(0,10)}`:""}. The site is legitimate.`),!0;case"revoked":return x.log.error(`${O.red("REVOKED")} - the owning account no longer holds a plan (refund or chargeback). This deployment is no longer licensed.`),!1;case"token_domain_mismatch":return x.log.error(`${O.red("LEAKED TOKEN")} - this license belongs to a different deployment${e.mismatchDomain?` (${O.cyan(e.mismatchDomain)})`:""}, not this site. The token was copied from a licensed project.`),!1;case"no_license_history":return x.log.error(`${O.red("NO LICENSE HISTORY")} - no license has ever been associated with this site. If it runs GenerateSaaS, treat it as unlicensed.`),!1;default:return x.log.warn(`${O.yellow("UNKNOWN")} - could not cross-reference the records right now. Try again shortly.`),!1}}async function So(e,t){let n=process.env.GENERATESAAS_API_URL??it,r=ms();e.start("Cross-referencing license records...");try{let o=r?fs(await Hn(n,r,{lkh:t.lkh,nid:t.nid,domain:t.domain})):gs(await Wt(n,{token:t.token,domain:t.domain}));return e.stop(`${O.green("Checked")} - records cross-referenced`),hs(o)}catch(o){return e.stop(`${O.yellow("Skipped")} - ${C(o)}`),null}}function ys(e){let t=e.split(".");if(t.length!==3||!t[1])throw new Error("Invalid JWT format");let n=Buffer.from(t[1],"base64url").toString("utf-8");return JSON.parse(n)}function En(e){return typeof e!="number"?"unknown":new Date(e*1e3).toISOString().split("T")[0]}function Eo(e){x.note([`License ID: ${O.cyan(String(e.lid??"unknown"))}`,`Version: ${O.cyan(String(e.ver??"unknown"))}`,`Init version: ${String(e.iver??"unknown")}`,`Frontend: ${String(e.fe??"unknown")}`,`Created: ${En(e.pat)}`,`Last updated: ${En(e.uat)}`,`Expires: ${En(e.exp)}`,`Install ID: ${String(e.nid??"unknown")}`].join(`
|
|
1274
|
+
`),O.yellow("License Details"))}function vs(e){let n=(/^https?:\/\//i.test(e)?e:`https://${e}`).replace(/\/+$/,"");if(n.endsWith("/api"))return[`${n}/license`];try{if(new URL(n).pathname!=="/")return[`${n}/license`]}catch{return[`${n}/license`]}return[`${n}/api/license`,`${n}/license`]}async function Ao(e){let t=x.spinner(),n=null,r="no candidates";for(let a of vs(e)){t.start(`Checking ${a}...`);try{let s=await fetch(a);if(!s.ok){r=`${a} returned ${s.status}`,t.stop(`${O.yellow("Not here")} - ${r}`);continue}let p=(await s.text()).trim();if(!p||p.split(".").length!==3){r=`${a} did not return a JWT`,t.stop(`${O.yellow("Not here")} - ${r}`);continue}n=p,t.stop(`${O.green("Found")} - license endpoint responded`);break}catch(s){r=`${a}: ${C(s)}`,t.stop(`${O.yellow("Unreachable")} - ${r}`)}}if(n===null){x.log.warn(`No license endpoint found (last: ${r}). The site may be ejected, not a GenerateSaaS app, or serving its API elsewhere.`);let a=Io(e);return a?await So(t,{domain:a})??!1:!1}let o;try{o=ys(n)}catch{return x.log.error("Could not decode JWT payload."),!1}t.start("Verifying signature...");try{let a=process.env.GENERATESAAS_API_URL??it,s=await Wt(a,{token:n});if(s.valid)t.stop(`${O.green("Valid")} - signature verified`);else return t.stop(`${O.red("Invalid")} - ${s.reason}`),!1}catch{return t.stop(`${O.yellow("Skipped")} - could not reach verification service`),x.log.warn("Signature not verified. Displaying unverified claims:"),Eo(o),!1}return Eo(o),await So(t,{token:n,lkh:typeof o.lkh=="string"?o.lkh:void 0,nid:typeof o.nid=="string"?o.nid:void 0,domain:Io(e)})??!0}function Io(e){try{return new URL(/^https?:\/\//i.test(e)?e:`https://${e}`).hostname}catch{return}}function Po(e){e.command("verify").description("Verify a GenerateSaaS license on a deployed site").argument("[url]","URL of the site to verify (e.g. https://example.com or https://example.com/api)").option("--file <path>","file with URLs to check, one per line").action(async(t,n)=>{if(!t&&!n.file&&(x.cancel("Provide a URL or --file <path>."),process.exit(1)),n.file){let o=(await us(n.file,"utf-8")).split(`
|
|
1275
|
+
`).map(a=>a.trim()).filter(a=>a&&!a.startsWith("#"));o.length===0&&(x.cancel("No URLs found in file."),process.exit(1));let i=0;for(let a of o)await Ao(a)&&i++,x.log.info("");x.log.success(`${i}/${o.length} sites verified.`)}else await Ao(t)||process.exit(1)})}import{existsSync as ks,rmSync as ws}from"fs";import*as te from"@clack/prompts";function To(e){e.command("auth").description("Set or update your GenerateSaaS API key").option("--clear","remove saved API key").action(async t=>{if(t.clear){ks(me)?(ws(me),te.log.success("API key removed.")):te.log.info("No API key configured.");return}let n=at();n?te.log.info(`Current API key: ****${n.slice(-4)}`):te.log.info("No API key configured.");let r=await st(),o=fe(r),i=te.spinner();i.start("Verifying API key...");try{await ke(o),i.stop("API key verified."),Ie(r),te.log.success("API key saved.")}catch(a){i.stop("Verification failed."),a instanceof U&&a.status===401?te.cancel("Invalid API key."):te.cancel(C(a)),process.exit(1)}})}import{existsSync as Vt,rmSync as bs,readFileSync as In,writeFileSync as Ro}from"fs";import{join as xe}from"path";import*as V from"@clack/prompts";var Ss=["packages/api/src/functions/maintenance/license-heartbeat.ts","packages/api/src/lib/cron-spread.ts","packages/api/src/lib/manifest.ts","packages/api/src/routes/internal/license.ts"],Es=[{file:"packages/api/src/routes/inngest.ts",removals:[`import { licenseHeartbeatFunction } from "../functions/maintenance/license-heartbeat";
|
|
1256
1276
|
`,` licenseHeartbeatFunction,
|
|
1257
1277
|
`]},{file:"packages/api/src/routes/internal/index.ts",removals:[`import licenseRoutes from "./license";
|
|
1258
1278
|
`,` .route("/license", licenseRoutes)
|
|
1259
|
-
`]}];function
|
|
1279
|
+
`]}];function As(e){return(e&&e.length>0?e.map(n=>pt[n]):Object.values(pt)).map(n=>xe(n,vn))}function An(e){return Vt(e)?(bs(e,{recursive:!0}),!0):!1}function Is(e,t){if(!Vt(e))return!1;let n=In(e,"utf-8"),r=n;for(let o of t)r=r.replace(o,"");return r===n?!1:(Ro(e,r,"utf-8"),!0)}function Ps(e){let t=xe(e,".gitignore");if(!Vt(t))return!1;let n=In(t,"utf-8"),r=n.split(`
|
|
1260
1280
|
`).filter(o=>!o.includes(".generatesaas")).join(`
|
|
1261
|
-
`);return n
|
|
1281
|
+
`);return r===n?!1:(Ro(t,r,"utf-8"),!0)}function _o(e){e.command("eject").description("Remove all GenerateSaaS ties - manifest, license, heartbeat, skills").action(async()=>{let t=process.cwd(),n=xe(t,he),r;try{r=JSON.parse(In(n,"utf-8"))}catch{V.cancel("No GenerateSaaS project found in this directory."),process.exit(1)}let o=await V.text({message:'Type "eject" to confirm (this cannot be undone):',validate:s=>{if(s!=="eject")return'Type "eject" to confirm, or press Ctrl+C to cancel.'}});if(V.isCancel(o)&&(V.cancel("Eject cancelled."),process.exit(0)),r.licenseToken)try{await fetch("https://generatesaas.com/api/v1/heartbeat",{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${r.licenseToken}`},body:JSON.stringify({event:"eject",version:r.version,frontend:r.frontend}),signal:AbortSignal.timeout(5e3)}),V.log.info("Recorded the opt-out with generatesaas.com (final event - nothing is sent after this).")}catch{V.log.warn("Could not reach generatesaas.com to record the opt-out. Ejecting anyway.")}let i=[],a=[];for(let s of As(r.aiTools))An(xe(t,s))&&i.push(s);for(let s of Ss)An(xe(t,s))&&i.push(s);An(xe(t,q))&&i.push(q+"/");for(let s of Es){let p=xe(t,s.file);Is(p,s.removals)?a.push(s.file):Vt(p)&&V.log.warn(`Could not auto-modify ${s.file} - manually remove license/heartbeat references.`)}Ps(t)&&a.push(".gitignore");for(let s of i)V.log.info(`Deleted ${s}`);for(let s of a)V.log.info(`Modified ${s}`);V.log.success("Ejected successfully. This project is now fully standalone.")})}var Ce=new Ts().name("generatesaas").description("CLI for scaffolding and managing GenerateSaaS projects").version("1.20.0").addHelpText("after",`
|
|
1262
1282
|
Examples:
|
|
1263
1283
|
$ generatesaas init Interactive setup
|
|
1264
1284
|
$ generatesaas init -n my-app -y Quick setup with defaults
|
|
1265
1285
|
$ generatesaas status Check for updates
|
|
1266
1286
|
$ generatesaas auth Set or update API key
|
|
1267
|
-
`);
|
|
1287
|
+
`);mo(Ce);wo(Ce);bo(Ce);Po(Ce);To(Ce);_o(Ce);Ce.parseAsync().catch(e=>{Oo.cancel("An unexpected error occurred."),console.error(e),process.exit(1)});
|