create-better-t-stack 2.0.2 → 2.0.4
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/index.js
CHANGED
|
@@ -191,7 +191,7 @@ See: https://github.com/vite-pwa/vite-plugin-pwa/issues/809
|
|
|
191
191
|
`}function Sa(e){return`${b.bold("Documentation with Starlight:")}
|
|
192
192
|
${b.cyan("\u2022")} Start docs site: ${`cd apps/docs && ${e} dev`}
|
|
193
193
|
${b.cyan("\u2022")} Build docs site: ${`cd apps/docs && ${e} build`}
|
|
194
|
-
`}import wt from"node:path";import{log as Aa}from"@clack/prompts";import{$ as bt,execa as Da}from"execa";import J from"fs-extra";import $a from"picocolors";async function yt(e,r){await Ca(e,r),await Ea(e,r)}async function Ca(e,r){let t=wt.join(e,"package.json");if(await J.pathExists(t)){let a=await J.readJson(t);a.name=r.projectName;let o={dev:"turbo dev",build:"turbo build","check-types":"turbo check-types","dev:native":"turbo -F native dev","dev:web":"turbo -F web dev","dev:server":"turbo -F server dev","db:push":"turbo -F server db:push","db:studio":"turbo -F server db:studio"},n={dev:"pnpm -r --parallel dev",build:"pnpm -r build","check-types":"pnpm -r check-types","dev:native":"pnpm --filter native dev","dev:web":"pnpm --filter web dev","dev:server":"pnpm --filter server dev","db:push":"pnpm --filter server db:push","db:studio":"pnpm --filter server db:studio"},i={dev:"npm run dev --workspaces",build:"npm run build --workspaces","check-types":"npm run check-types --workspaces","dev:native":"npm run dev --workspace native","dev:web":"npm run dev --workspace web","dev:server":"npm run dev --workspace server","db:push":"npm run db:push --workspace server","db:studio":"npm run db:studio --workspace server"},s={dev:"bun run --filter '*' dev",build:"bun run --filter '*' build","check-types":"bun run --filter '*' check-types","dev:native":"bun run --filter native dev","dev:web":"bun run --filter web dev","dev:server":"bun run --filter server dev","db:push":"bun run --filter server db:push","db:studio":"bun run --filter server db:studio"};r.addons.includes("turborepo")?a.scripts=o:r.packageManager==="pnpm"?a.scripts=n:r.packageManager==="npm"?a.scripts=i:r.packageManager==="bun"?a.scripts=s:a.scripts={};let{stdout:c}=await Da(r.packageManager,["-v"],{cwd:e});a.packageManager=`${r.packageManager}@${c.trim()}`,await J.writeJson(t,a,{spaces:2})}}async function Ea(e,r){let t=wt.join(e,"apps/server/package.json");if(await J.pathExists(t)){let a=await J.readJson(t);r.database!=="none"&&(r.database==="sqlite"&&r.orm==="drizzle"&&(a.scripts["db:local"]="turso dev --db-file local.db"),r.orm==="prisma"?(a.scripts["db:push"]="prisma db push --schema ./prisma/schema",a.scripts["db:studio"]="prisma studio"):r.orm==="drizzle"&&(a.scripts["db:push"]="drizzle-kit push",a.scripts["db:studio"]="drizzle-kit studio")),await J.writeJson(t,a,{spaces:2})}}async function Pt(e,r){if(!r)return;if((await bt({cwd:e,reject:!1,stderr:"pipe"})`git --version`).exitCode!==0){Aa.warn($a.yellow("Git is not installed"));return}let a=await bt({cwd:e,reject:!1,stderr:"pipe"})`git init`;if(a.exitCode!==0)throw new Error(`Git initialization failed: ${a.stderr}`)}import me from"node:path";import fe from"fs-extra";async function jt(e){let{projectName:r,runtime:t,backend:a}=e,o=me.resolve(process.cwd(),r);if(a==="next")return;let n=me.join(o,"apps/server");t==="bun"?await Ta(n,a):t==="node"&&await Fa(n,a)}async function Ta(e,r){let t=me.join(e,"package.json"),a=await fe.readJson(t);a.scripts={...a.scripts,dev:"bun run --hot src/index.ts",start:"bun run dist/src/index.js"},await fe.writeJson(t,a,{spaces:2}),await g({devDependencies:["@types/bun"],projectDir:e})}async function Fa(e,r){let t=me.join(e,"package.json"),a=await fe.readJson(t);a.scripts={...a.scripts,dev:"tsx watch src/index.ts",start:"node dist/src/index.js"},await fe.writeJson(t,a,{spaces:2}),await g({devDependencies:["tsx","@types/node"],projectDir:e}),r==="hono"?await g({dependencies:["@hono/node-server"],projectDir:e}):r==="elysia"&&await g({dependencies:["@elysiajs/node"],projectDir:e})}import l from"node:path";import A from"consola";import m from"fs-extra";import{globby as vt}from"globby";import C from"picocolors";import Ra from"node:path";import Fe from"fs-extra";import ge from"handlebars";async function Re(e,r,t){try{let a=await Fe.readFile(e,"utf-8"),n=ge.compile(a)(t);await Fe.ensureDir(Ra.dirname(r)),await Fe.writeFile(r,n)}catch(a){throw console.error(`Error processing template ${e}:`,a),new Error(`Failed to process template ${e}`)}}ge.registerHelper("or",(e,r)=>e||r);ge.registerHelper("eq",(e,r)=>e===r);ge.registerHelper("includes",(e,r)=>Array.isArray(e)&&e.includes(r));async function v(e,r,t,a,o=!0){let n=await vt(e,{cwd:r,dot:!0,onlyFiles:!0,absolute:!1});for(let i of n){let s=l.join(r,i),c=i;i.endsWith(".hbs")&&(c=i.slice(0,-4));let p=l.join(t,c);if(await m.ensureDir(l.dirname(p)),s.endsWith(".hbs"))await Re(s,p,a);else{if(!o&&await m.pathExists(p))continue;await m.copy(s,p,{overwrite:!0})}}}async function kt(e,r){let t=l.join(P,"templates/base");await v(["package.json","_gitignore"],t,e,r)}async function xt(e,r){let t=r.frontend.filter(o=>o==="tanstack-router"||o==="react-router"||o==="tanstack-start"||o==="next"),a=r.frontend.includes("native");if(t.length>0){let o=l.join(e,"apps/web");await m.ensureDir(o);let n=l.join(P,"templates/frontend/web-base");await m.pathExists(n)&&await v("**/*",n,o,r);for(let i of t){let s=l.join(P,`templates/frontend/${i}`);await m.pathExists(s)&&await v("**/*",s,o,r)}if(r.api!=="none"){let i=t[0],s=l.join(P,`templates/api/${r.api}/web/base`);await m.pathExists(s)&&await v("**/*",s,o,r);let c=l.join(P,`templates/api/${r.api}/web/${i}`);await m.pathExists(c)&&await v("**/*",c,o,r)}}if(a){let o=l.join(e,"apps/native");await m.ensureDir(o);let n=l.join(P,"templates/frontend/native");if(await m.pathExists(n)&&await v("**/*",n,o,r),r.api!=="none"){let i=l.join(P,`templates/api/${r.api}/native`);await m.pathExists(i)&&await v("**/*",i,o,r)}}}async function St(e,r){if(r.backend==="none")return;let t=l.join(e,"apps/server");await m.ensureDir(t);let a=l.join(P,"templates/backend/server-base");await m.pathExists(a)?await v("**/*",a,t,r):A.warn(C.yellow(`Warning: server-base template not found at ${a}`));let o=l.join(P,`templates/backend/${r.backend}`);if(await m.pathExists(o)?await v("**/*",o,t,r):A.warn(C.yellow(`Warning: Backend template directory not found, skipping: ${o}`)),r.api!=="none"){let n=l.join(P,`templates/api/${r.api}/server/base`);await m.pathExists(n)&&await v("**/*",n,t,r);let i=l.join(P,`templates/api/${r.api}/server/${r.backend}`);await m.pathExists(i)&&await v("**/*",i,t,r)}}async function At(e,r){if(r.orm==="none"||r.database==="none")return;let t=l.join(e,"apps/server");await m.ensureDir(t);let a=l.join(P,`templates/db/${r.orm}/${r.database}`);await m.pathExists(a)?await v("**/*",a,t,r):A.warn(C.yellow(`Warning: Database/ORM template directory not found, skipping: ${a}`))}async function Dt(e,r){if(!r.auth)return;let t=l.join(e,"apps/server"),a=l.join(e,"apps/web"),o=l.join(e,"apps/native"),n=r.frontend.filter(s=>s==="tanstack-router"||s==="react-router"||s==="tanstack-start"||s==="next"),i=r.frontend.includes("native");if(await m.pathExists(t)){let s=l.join(P,"templates/auth/server/base");await m.pathExists(s)?await v("**/*",s,t,r):A.warn(C.yellow(`Warning: Base auth server template not found at ${s}`));let c=l.join(P,"templates/auth/server/next");if(await m.pathExists(c)?await v("**/*",c,t,r):A.warn(C.yellow(`Warning: Next auth server template not found at ${c}`)),r.orm!=="none"&&r.database!=="none"){let p=r.orm,w=r.database,y="";p==="drizzle"?y=l.join(P,`templates/auth/server/db/drizzle/${w}`):p==="prisma"&&(y=l.join(P,`templates/auth/server/db/prisma/${w}`)),y&&await m.pathExists(y)?await v("**/*",y,t,r):A.warn(C.yellow(`Warning: Auth template for ${p}/${w} not found at ${y}`))}}else A.warn(C.yellow("Warning: apps/server directory does not exist, skipping server-side auth setup."));if(n.length>0&&await m.pathExists(a)){let s=l.join(P,"templates/auth/web/base");await m.pathExists(s)?await v("**/*",s,a,r):A.warn(C.yellow(`Warning: Base auth web template not found at ${s}`));for(let c of n){let p=l.join(P,`templates/auth/web/${c}`);await m.pathExists(p)?await v("**/*",p,a,r):A.warn(C.yellow(`Warning: Auth web template for ${c} not found at ${p}`))}}if(i&&await m.pathExists(o)){let s=l.join(P,"templates/auth/native");await m.pathExists(s)?await v("**/*",s,o,r):A.warn(C.yellow(`Warning: Auth native template not found at ${s}`))}}async function $t(e,r){if(r.addons.includes("turborepo")){let t=l.join(P,"templates/addons/turborepo");await m.pathExists(t)?await v("**/*",t,e,r):A.warn(C.yellow("Warning: Turborepo addon template not found."))}if(r.addons.includes("husky")){let t=l.join(P,"templates/addons/husky");await m.pathExists(t)?await v("**/*",t,e,r):A.warn(C.yellow("Warning: Husky addon template not found."))}if(r.addons.includes("biome")){let t=l.join(P,"templates/addons/biome");await m.pathExists(t)?await v("**/*",t,e,r):A.warn(C.yellow("Warning: Biome addon template not found."))}if(r.addons.includes("pwa")){let t=l.join(P,"templates/addons/pwa/apps/web"),a=l.join(e,"apps/web");await m.pathExists(t)?await m.pathExists(a)?await v("**/*",t,a,r):A.warn(C.yellow("Warning: apps/web directory not found, cannot setup PWA addon.")):A.warn(C.yellow("Warning: PWA addon template not found."))}}async function Ct(e,r){if(!r.examples||r.examples.length===0)return;let t=l.join(e,"apps/server"),a=l.join(e,"apps/web");for(let o of r.examples){let n=l.join(P,`templates/examples/${o}`);if(await m.pathExists(t)){let i=l.join(n,"server");if(await m.pathExists(i)&&r.orm!=="none"){let s=l.join(i,r.orm,"base");if(await m.pathExists(s)&&await v("**/*",s,t,r,!1),r.database!=="none"){let c=l.join(i,r.orm,r.database);await m.pathExists(c)&&await v("**/*",c,t,r,!1)}}}if(await m.pathExists(a)){let i=l.join(n,"web");if(await m.pathExists(i)){let s=r.frontend.filter(c=>["next","react-router","tanstack-router","tanstack-start"].includes(c));for(let c of s){let p=l.join(i,c);await m.pathExists(p)&&await v("**/*",p,a,r,!1)}}}}}async function Et(e,r){let t=await vt(["**/.gitignore.hbs","**/_gitignore"],{cwd:e,dot:!0,onlyFiles:!0,absolute:!0,ignore:["**/node_modules/**","**/.git/**"]});for(let a of t){let o=l.dirname(a),n=l.basename(a),i=l.join(o,".gitignore");try{n===".gitignore.hbs"?(await Re(a,i,r),await m.remove(a)):n==="_gitignore"&&await m.move(a,i,{overwrite:!0})}catch(s){A.error(`Error processing gitignore file ${a}:`,s)}}}async function Tt(e,r){if(r.packageManager==="pnpm"){let t=l.join(P,"templates/extras/pnpm-workspace.yaml"),a=l.join(e,"pnpm-workspace.yaml");await m.pathExists(t)?await m.copy(t,a):A.warn(C.yellow("Warning: pnpm-workspace.yaml template not found."))}}async function Rt(e){let r=Ma(),t=Ba.resolve(process.cwd(),e.projectName);try{return await La.ensureDir(t),await kt(t,e),await xt(t,e),await St(t,e),await Ke(e),await At(t,e),await lt(e),await Dt(t,e),await He(e),await $t(t,e),e.addons.length>0&&e.addons[0]!=="none"&&await qe(e),await Ct(t,e),await Tt(t,e),e.examples.length>0&&e.examples[0]!=="none"&&await ut(e),await Ve(e),await jt(e),await dt(e),await yt(t,e),await Xe(t,e),await Pt(t,e.git),await Et(t,e),Ia.success("Project template successfully scaffolded!"),e.install&&await gt({projectDir:t,packageManager:e.packageManager,addons:e.addons}),ht({...e,depsInstalled:e.install}),t}catch(a){throw r.stop(Ft.red("Failed")),a instanceof Error&&(Na(Ft.red(`Error during project creation: ${a.message}`)),console.error(a.stack),process.exit(1)),a}}import{cancel as _o,group as zo}from"@clack/prompts";import Wo from"picocolors";import{cancel as Oa,isCancel as Ua,multiselect as _a}from"@clack/prompts";import za from"picocolors";async function Bt(e,r){if(e!==void 0)return e;let t=r?.includes("react-router")||r?.includes("tanstack-router"),a=[{value:"starlight",label:"Starlight",hint:"Add Astro Starlight documentation site"},{value:"biome",label:"Biome",hint:"Add Biome for linting and formatting"},{value:"husky",label:"Husky",hint:"Add Git hooks with Husky, lint-staged (requires Biome)"},{value:"turborepo",label:"Turborepo",hint:"Optimize builds for monorepos"}],n=t?[...[{value:"pwa",label:"PWA (Progressive Web App)",hint:"Make your app installable and work offline"},{value:"tauri",label:"Tauri Desktop App",hint:"Build native desktop apps from your web frontend"}],...a]:a,i=h.addons.filter(c=>t||c!=="pwa"&&c!=="tauri"),s=await _a({message:"Select addons",options:n,initialValues:i,required:!1});return Ua(s)&&(Oa(za.red("Operation cancelled")),process.exit(0)),s.includes("husky")&&!s.includes("biome")&&s.push("biome"),s}import{cancel as Wa,isCancel as qa,select as Va}from"@clack/prompts";import Ga from"picocolors";async function Nt(e,r){if(e)return e;let t=r?.includes("native"),a=[{value:"trpc",label:"tRPC",hint:"End-to-end typesafe APIs made easy"},{value:"orpc",label:"oRPC",hint:"End-to-end type-safe APIs that adhere to OpenAPI standards"},{value:"none",label:"None",hint:"No API integration (skip API setup)"}];t&&(a=[{value:"trpc",label:"tRPC",hint:"End-to-end typesafe APIs made easy (Required for Native frontend)"}]);let o=await Va({message:"Select API type",options:a,initialValue:t?"trpc":h.api});return qa(o)&&(Wa(Ga.red("Operation cancelled")),process.exit(0)),t&&o!=="trpc"?"trpc":o}import{cancel as Ja,confirm as Qa,isCancel as Ha}from"@clack/prompts";import Ya from"picocolors";async function It(e,r){if(!r)return!1;if(e!==void 0)return e;let t=await Qa({message:"Add authentication with Better-Auth?",initialValue:h.auth});return Ha(t)&&(Ja(Ya.red("Operation cancelled")),process.exit(0)),t}import{cancel as Ka,isCancel as Xa,select as Za}from"@clack/prompts";import eo from"picocolors";async function Mt(e){if(e!==void 0)return e;let r=await Za({message:"Select backend framework",options:[{value:"hono",label:"Hono",hint:"Lightweight, ultrafast web framework"},{value:"next",label:"Next.js",hint:"Full-stack framework with API routes"},{value:"express",label:"Express",hint:"Fast, unopinionated, minimalist web framework for Node.js"},{value:"elysia",label:"Elysia",hint:"Ergonomic web framework for building backend servers"}],initialValue:h.backend});return Xa(r)&&(Ka(eo.red("Operation cancelled")),process.exit(0)),r}import{cancel as to,isCancel as ro,select as ao}from"@clack/prompts";import oo from"picocolors";async function Lt(e){if(e!==void 0)return e;let r=await ao({message:"Select database",options:[{value:"none",label:"None",hint:"No database setup"},{value:"sqlite",label:"SQLite",hint:"lightweight, server-less, embedded relational database management system"},{value:"postgres",label:"PostgreSQL",hint:"powerful, open source object-relational database system"},{value:"mysql",label:"MySQL",hint:"popular open-source relational database system"},{value:"mongodb",label:"MongoDB",hint:"open-source NoSQL database that stores data in JSON-like documents called BSON"}],initialValue:h.database});return ro(r)&&(to(oo.red("Operation cancelled")),process.exit(0)),r}import{cancel as no,isCancel as so,select as io}from"@clack/prompts";import co from"picocolors";async function Ot(e,r,t){if(r!==void 0)return r;if(e==="sqlite"&&t==="prisma")return"none";let a=[];if(e==="sqlite")a=[{value:"turso",label:"Turso",hint:"SQLite for Production. Powered by libSQL"},{value:"none",label:"None",hint:"Manual setup"}];else if(e==="postgres")a=[{value:"neon",label:"Neon Postgres",hint:"Serverless Postgres with branching capability"},...t==="prisma"?[{value:"prisma-postgres",label:"Prisma Postgres",hint:"Instant Postgres for Global Applications"}]:[],{value:"none",label:"None",hint:"Manual setup"}];else if(e==="mongodb")a=[{value:"mongodb-atlas",label:"MongoDB Atlas",hint:"The most effective way to deploy MongoDB"},{value:"none",label:"None",hint:"Manual setup"}];else return"none";let o=await io({message:`Select ${e} setup option`,options:a,initialValue:"none"});return so(o)&&(no(co.red("Operation cancelled")),process.exit(0)),o}import{cancel as po,isCancel as lo,multiselect as Ut}from"@clack/prompts";import uo from"picocolors";async function _t(e,r,t,a){if(e!==void 0)return e;if(r==="none")return[];if(!(t?.includes("react-router")||t?.includes("tanstack-router")||t?.includes("tanstack-start")))return[];let n=[];return a==="elysia"&&(n=await Ut({message:"Include examples",options:[{value:"todo",label:"Todo App",hint:"A simple CRUD example app"}],required:!1,initialValues:h.examples})),(a==="hono"||a==="express")&&(n=await Ut({message:"Include examples",options:[{value:"todo",label:"Todo App",hint:"A simple CRUD example app"},{value:"ai",label:"AI Chat",hint:"A simple AI chat interface using AI SDK"}],required:!1,initialValues:h.examples})),lo(n)&&(po(uo.red("Operation cancelled")),process.exit(0)),n}import{cancel as zt,isCancel as Wt,multiselect as mo,select as fo}from"@clack/prompts";import qt from"picocolors";async function Vt(e){if(e!==void 0)return e;let r=await mo({message:"Select platforms to develop for",options:[{value:"web",label:"Web",hint:"React Web Application"},{value:"native",label:"Native",hint:"Create a React Native/Expo app"}],required:!1,initialValues:h.frontend.some(a=>a==="tanstack-router"||a==="react-router"||a==="tanstack-start"||a==="next")?["web"]:[]});Wt(r)&&(zt(qt.red("Operation cancelled")),process.exit(0));let t=[];if(r.includes("web")){let a=await fo({message:"Choose frontend framework",options:[{value:"tanstack-router",label:"TanStack Router",hint:"Modern and scalable routing for React Applications"},{value:"react-router",label:"React Router",hint:"A user\u2011obsessed, standards\u2011focused, multi\u2011strategy router"},{value:"next",label:"Next.js",hint:"The React Framework for the Web"},{value:"tanstack-start",label:"TanStack Start (beta)",hint:"SSR, Server Functions, API Routes and more with TanStack Router"}],initialValue:h.frontend.find(o=>o==="tanstack-router"||o==="react-router"||o==="tanstack-start"||o==="next")||"tanstack-router"});Wt(a)&&(zt(qt.red("Operation cancelled")),process.exit(0)),t.push(a)}return r.includes("native")&&t.push("native"),t}import{cancel as go,confirm as ho,isCancel as bo}from"@clack/prompts";import wo from"picocolors";async function Gt(e){if(e!==void 0)return e;let r=await ho({message:"Initialize git repository?",initialValue:h.git});return bo(r)&&(go(wo.red("Operation cancelled")),process.exit(0)),r}import{cancel as yo,confirm as Po,isCancel as jo}from"@clack/prompts";import vo from"picocolors";async function Jt(e){if(e!==void 0)return e;let r=await Po({message:"Install dependencies?",initialValue:h.install});return jo(r)&&(yo(vo.red("Operation cancelled")),process.exit(0)),r}import{cancel as ko,isCancel as xo,log as So,select as Ao}from"@clack/prompts";import Do from"picocolors";async function Qt(e,r,t){if(!r)return"none";if(e!==void 0)return e;if(t==="mongodb")return So.info("Only Prisma is supported with MongoDB."),"prisma";let a=await Ao({message:"Select ORM",options:[{value:"drizzle",label:"Drizzle",hint:"lightweight and performant TypeScript ORM"},{value:"prisma",label:"Prisma",hint:"Powerful, feature-rich ORM"}],initialValue:h.orm});return xo(a)&&(ko(Do.red("Operation cancelled")),process.exit(0)),a}import{cancel as $o,isCancel as Co,select as Eo}from"@clack/prompts";import To from"picocolors";async function Ht(e){if(e!==void 0)return e;let r=re(),t=await Eo({message:"Choose package manager",options:[{value:"npm",label:"npm",hint:"Node Package Manager"},{value:"pnpm",label:"pnpm",hint:"Fast, disk space efficient package manager"},{value:"bun",label:"bun",hint:"All-in-one JavaScript runtime & toolkit"}],initialValue:r});return Co(t)&&($o(To.red("Operation cancelled")),process.exit(0)),t}import te from"node:path";import{cancel as Fo,isCancel as Ro,text as Bo}from"@clack/prompts";import U from"fs-extra";import No from"picocolors";var Io=["<",">",":",'"',"|","?","*"],Yt=255;function Kt(e){if(e!=="."){if(!e)return"Project name cannot be empty";if(e.length>Yt)return`Project name must be less than ${Yt} characters`;if(Io.some(r=>e.includes(r)))return"Project name contains invalid characters";if(e.startsWith(".")||e.startsWith("-"))return"Project name cannot start with a dot or dash";if(e.toLowerCase()==="node_modules")return"Project name is reserved"}}async function he(e){if(e)if(e==="."){let n=process.cwd();if(U.readdirSync(n).length===0)return e}else{let n=te.basename(e);if(!Kt(n)){let s=te.resolve(process.cwd(),e);if(!U.pathExistsSync(s)||U.readdirSync(s).length===0)return e}}let r=!1,t="",a=h.projectName,o=1;for(;U.pathExistsSync(te.resolve(process.cwd(),a));)a=`${h.projectName}-${o}`,o++;for(;!r;){let n=await Bo({message:"Enter your project name or path (relative to current directory)",placeholder:a,initialValue:e,defaultValue:a,validate:i=>{let s=i.trim()||a;if(s==="."){if(U.readdirSync(process.cwd()).length>0)return"Current directory is not empty. Please choose a different directory.";r=!0;return}let c=te.resolve(process.cwd(),s),p=te.basename(c),w=Kt(p);if(w)return w;if(!c.startsWith(process.cwd()))return"Project path must be within current directory";if(U.pathExistsSync(c)&&U.readdirSync(c).length>0)return`Directory "${s}" already exists and is not empty. Please choose a different name or path.`;r=!0}});Ro(n)&&(Fo(No.red("Operation cancelled.")),process.exit(0)),t=n||a}return t}import{cancel as Mo,isCancel as Lo,select as Oo}from"@clack/prompts";import Uo from"picocolors";async function Xt(e,r){if(e!==void 0)return e;if(r==="next")return"node";let t=await Oo({message:"Select runtime",options:[{value:"bun",label:"Bun",hint:"Fast all-in-one JavaScript runtime"},{value:"node",label:"Node.js",hint:"Traditional Node.js runtime"}],initialValue:h.runtime});return Lo(t)&&(Mo(Uo.red("Operation cancelled")),process.exit(0)),t}async function Zt(e){let r=await zo({projectName:async()=>he(e.projectName),frontend:()=>Vt(e.frontend),backend:()=>Mt(e.backend),runtime:({results:t})=>Xt(e.runtime,t.backend),database:()=>Lt(e.database),orm:({results:t})=>Qt(e.orm,t.database!=="none",t.database),api:({results:t})=>Nt(e.api,t.frontend),auth:({results:t})=>It(e.auth,t.database!=="none"),addons:({results:t})=>Bt(e.addons,t.frontend),examples:({results:t})=>_t(e.examples,t.database,t.frontend,t.backend),dbSetup:({results:t})=>Ot(t.database??"none",e.dbSetup,t.orm),git:()=>Gt(e.git),packageManager:()=>Ht(e.packageManager),install:()=>Jt(e.install)},{onCancel:()=>{_o(Wo.red("Operation cancelled")),process.exit(0)}});return{projectName:r.projectName,frontend:r.frontend,database:r.database,orm:r.orm,auth:r.auth,addons:r.addons,examples:r.examples,git:r.git,packageManager:r.packageManager,install:r.install,dbSetup:r.dbSetup,backend:r.backend,runtime:r.runtime,api:r.api}}import T from"picocolors";function Be(e){let r=[];if(e.projectName&&r.push(`${T.blue("Project Name:")} ${e.projectName}`),e.frontend!==void 0){let t=Array.isArray(e.frontend)?e.frontend:[e.frontend],a=t.length>0&&t[0]!==void 0&&t[0]!==""?t.join(", "):"none";r.push(`${T.blue("Frontend:")} ${a}`)}if(e.backend!==void 0&&r.push(`${T.blue("Backend Framework:")} ${String(e.backend)}`),e.runtime!==void 0&&r.push(`${T.blue("Runtime:")} ${String(e.runtime)}`),e.api!==void 0&&r.push(`${T.blue("API:")} ${String(e.api)}`),e.database!==void 0&&r.push(`${T.blue("Database:")} ${String(e.database)}`),e.orm!==void 0&&r.push(`${T.blue("ORM:")} ${String(e.orm)}`),e.auth!==void 0){let t=typeof e.auth=="boolean"?e.auth?"Yes":"No":String(e.auth);r.push(`${T.blue("Authentication:")} ${t}`)}if(e.addons!==void 0){let t=Array.isArray(e.addons)?e.addons:[e.addons],a=t.length>0&&t[0]!==void 0?t.join(", "):"none";r.push(`${T.blue("Addons:")} ${a}`)}if(e.examples!==void 0){let t=Array.isArray(e.examples)?e.examples:[e.examples],a=t.length>0&&t[0]!==void 0?t.join(", "):"none";r.push(`${T.blue("Examples:")} ${a}`)}if(e.git!==void 0){let t=typeof e.git=="boolean"?e.git?"Yes":"No":String(e.git);r.push(`${T.blue("Git Init:")} ${t}`)}if(e.packageManager!==void 0&&r.push(`${T.blue("Package Manager:")} ${String(e.packageManager)}`),e.install!==void 0){let t=typeof e.install=="boolean"?e.install?"Yes":"No":String(e.install);r.push(`${T.blue("Install Dependencies:")} ${t}`)}return e.dbSetup!==void 0&&r.push(`${T.blue("Database Setup:")} ${String(e.dbSetup)}`),r.length===0?T.yellow("No configuration selected."):r.join(`
|
|
194
|
+
`}import wt from"node:path";import{log as Aa}from"@clack/prompts";import{$ as bt,execa as Da}from"execa";import J from"fs-extra";import $a from"picocolors";async function yt(e,r){await Ca(e,r),await Ea(e,r)}async function Ca(e,r){let t=wt.join(e,"package.json");if(await J.pathExists(t)){let a=await J.readJson(t);a.name=r.projectName;let o={dev:"turbo dev",build:"turbo build","check-types":"turbo check-types","dev:native":"turbo -F native dev","dev:web":"turbo -F web dev","dev:server":"turbo -F server dev","db:push":"turbo -F server db:push","db:studio":"turbo -F server db:studio"},n={dev:"pnpm -r dev",build:"pnpm -r build","check-types":"pnpm -r check-types","dev:native":"pnpm --filter native dev","dev:web":"pnpm --filter web dev","dev:server":"pnpm --filter server dev","db:push":"pnpm --filter server db:push","db:studio":"pnpm --filter server db:studio"},i={dev:"npm run dev --workspaces",build:"npm run build --workspaces","check-types":"npm run check-types --workspaces","dev:native":"npm run dev --workspace native","dev:web":"npm run dev --workspace web","dev:server":"npm run dev --workspace server","db:push":"npm run db:push --workspace server","db:studio":"npm run db:studio --workspace server"},s={dev:"bun run --filter '*' dev",build:"bun run --filter '*' build","check-types":"bun run --filter '*' check-types","dev:native":"bun run --filter native dev","dev:web":"bun run --filter web dev","dev:server":"bun run --filter server dev","db:push":"bun run --filter server db:push","db:studio":"bun run --filter server db:studio"};r.addons.includes("turborepo")?a.scripts=o:r.packageManager==="pnpm"?a.scripts=n:r.packageManager==="npm"?a.scripts=i:r.packageManager==="bun"?a.scripts=s:a.scripts={};let{stdout:c}=await Da(r.packageManager,["-v"],{cwd:e});a.packageManager=`${r.packageManager}@${c.trim()}`,await J.writeJson(t,a,{spaces:2})}}async function Ea(e,r){let t=wt.join(e,"apps/server/package.json");if(await J.pathExists(t)){let a=await J.readJson(t);r.database!=="none"&&(r.database==="sqlite"&&r.orm==="drizzle"&&(a.scripts["db:local"]="turso dev --db-file local.db"),r.orm==="prisma"?(a.scripts["db:push"]="prisma db push --schema ./prisma/schema",a.scripts["db:studio"]="prisma studio"):r.orm==="drizzle"&&(a.scripts["db:push"]="drizzle-kit push",a.scripts["db:studio"]="drizzle-kit studio")),await J.writeJson(t,a,{spaces:2})}}async function Pt(e,r){if(!r)return;if((await bt({cwd:e,reject:!1,stderr:"pipe"})`git --version`).exitCode!==0){Aa.warn($a.yellow("Git is not installed"));return}let a=await bt({cwd:e,reject:!1,stderr:"pipe"})`git init`;if(a.exitCode!==0)throw new Error(`Git initialization failed: ${a.stderr}`)}import me from"node:path";import fe from"fs-extra";async function jt(e){let{projectName:r,runtime:t,backend:a}=e,o=me.resolve(process.cwd(),r);if(a==="next")return;let n=me.join(o,"apps/server");t==="bun"?await Ta(n,a):t==="node"&&await Fa(n,a)}async function Ta(e,r){let t=me.join(e,"package.json"),a=await fe.readJson(t);a.scripts={...a.scripts,dev:"bun run --hot src/index.ts",start:"bun run dist/src/index.js"},await fe.writeJson(t,a,{spaces:2}),await g({devDependencies:["@types/bun"],projectDir:e})}async function Fa(e,r){let t=me.join(e,"package.json"),a=await fe.readJson(t);a.scripts={...a.scripts,dev:"tsx watch src/index.ts",start:"node dist/src/index.js"},await fe.writeJson(t,a,{spaces:2}),await g({devDependencies:["tsx","@types/node"],projectDir:e}),r==="hono"?await g({dependencies:["@hono/node-server"],projectDir:e}):r==="elysia"&&await g({dependencies:["@elysiajs/node"],projectDir:e})}import l from"node:path";import A from"consola";import m from"fs-extra";import{globby as vt}from"globby";import C from"picocolors";import Ra from"node:path";import Fe from"fs-extra";import ge from"handlebars";async function Re(e,r,t){try{let a=await Fe.readFile(e,"utf-8"),n=ge.compile(a)(t);await Fe.ensureDir(Ra.dirname(r)),await Fe.writeFile(r,n)}catch(a){throw console.error(`Error processing template ${e}:`,a),new Error(`Failed to process template ${e}`)}}ge.registerHelper("or",(e,r)=>e||r);ge.registerHelper("eq",(e,r)=>e===r);ge.registerHelper("includes",(e,r)=>Array.isArray(e)&&e.includes(r));async function v(e,r,t,a,o=!0){let n=await vt(e,{cwd:r,dot:!0,onlyFiles:!0,absolute:!1});for(let i of n){let s=l.join(r,i),c=i;i.endsWith(".hbs")&&(c=i.slice(0,-4));let p=l.join(t,c);if(await m.ensureDir(l.dirname(p)),s.endsWith(".hbs"))await Re(s,p,a);else{if(!o&&await m.pathExists(p))continue;await m.copy(s,p,{overwrite:!0})}}}async function kt(e,r){let t=l.join(P,"templates/base");await v(["package.json","_gitignore"],t,e,r)}async function xt(e,r){let t=r.frontend.filter(o=>o==="tanstack-router"||o==="react-router"||o==="tanstack-start"||o==="next"),a=r.frontend.includes("native");if(t.length>0){let o=l.join(e,"apps/web");await m.ensureDir(o);let n=l.join(P,"templates/frontend/web-base");await m.pathExists(n)&&await v("**/*",n,o,r);for(let i of t){let s=l.join(P,`templates/frontend/${i}`);await m.pathExists(s)&&await v("**/*",s,o,r)}if(r.api!=="none"){let i=t[0],s=l.join(P,`templates/api/${r.api}/web/base`);await m.pathExists(s)&&await v("**/*",s,o,r);let c=l.join(P,`templates/api/${r.api}/web/${i}`);await m.pathExists(c)&&await v("**/*",c,o,r)}}if(a){let o=l.join(e,"apps/native");await m.ensureDir(o);let n=l.join(P,"templates/frontend/native");if(await m.pathExists(n)&&await v("**/*",n,o,r),r.api!=="none"){let i=l.join(P,`templates/api/${r.api}/native`);await m.pathExists(i)&&await v("**/*",i,o,r)}}}async function St(e,r){if(r.backend==="none")return;let t=l.join(e,"apps/server");await m.ensureDir(t);let a=l.join(P,"templates/backend/server-base");await m.pathExists(a)?await v("**/*",a,t,r):A.warn(C.yellow(`Warning: server-base template not found at ${a}`));let o=l.join(P,`templates/backend/${r.backend}`);if(await m.pathExists(o)?await v("**/*",o,t,r):A.warn(C.yellow(`Warning: Backend template directory not found, skipping: ${o}`)),r.api!=="none"){let n=l.join(P,`templates/api/${r.api}/server/base`);await m.pathExists(n)&&await v("**/*",n,t,r);let i=l.join(P,`templates/api/${r.api}/server/${r.backend}`);await m.pathExists(i)&&await v("**/*",i,t,r)}}async function At(e,r){if(r.orm==="none"||r.database==="none")return;let t=l.join(e,"apps/server");await m.ensureDir(t);let a=l.join(P,`templates/db/${r.orm}/${r.database}`);await m.pathExists(a)?await v("**/*",a,t,r):A.warn(C.yellow(`Warning: Database/ORM template directory not found, skipping: ${a}`))}async function Dt(e,r){if(!r.auth)return;let t=l.join(e,"apps/server"),a=l.join(e,"apps/web"),o=l.join(e,"apps/native"),n=r.frontend.filter(s=>s==="tanstack-router"||s==="react-router"||s==="tanstack-start"||s==="next"),i=r.frontend.includes("native");if(await m.pathExists(t)){let s=l.join(P,"templates/auth/server/base");await m.pathExists(s)?await v("**/*",s,t,r):A.warn(C.yellow(`Warning: Base auth server template not found at ${s}`));let c=l.join(P,"templates/auth/server/next");if(await m.pathExists(c)?await v("**/*",c,t,r):A.warn(C.yellow(`Warning: Next auth server template not found at ${c}`)),r.orm!=="none"&&r.database!=="none"){let p=r.orm,w=r.database,y="";p==="drizzle"?y=l.join(P,`templates/auth/server/db/drizzle/${w}`):p==="prisma"&&(y=l.join(P,`templates/auth/server/db/prisma/${w}`)),y&&await m.pathExists(y)?await v("**/*",y,t,r):A.warn(C.yellow(`Warning: Auth template for ${p}/${w} not found at ${y}`))}}else A.warn(C.yellow("Warning: apps/server directory does not exist, skipping server-side auth setup."));if(n.length>0&&await m.pathExists(a)){let s=l.join(P,"templates/auth/web/base");await m.pathExists(s)?await v("**/*",s,a,r):A.warn(C.yellow(`Warning: Base auth web template not found at ${s}`));for(let c of n){let p=l.join(P,`templates/auth/web/${c}`);await m.pathExists(p)?await v("**/*",p,a,r):A.warn(C.yellow(`Warning: Auth web template for ${c} not found at ${p}`))}}if(i&&await m.pathExists(o)){let s=l.join(P,"templates/auth/native");await m.pathExists(s)?await v("**/*",s,o,r):A.warn(C.yellow(`Warning: Auth native template not found at ${s}`))}}async function $t(e,r){if(r.addons.includes("turborepo")){let t=l.join(P,"templates/addons/turborepo");await m.pathExists(t)?await v("**/*",t,e,r):A.warn(C.yellow("Warning: Turborepo addon template not found."))}if(r.addons.includes("husky")){let t=l.join(P,"templates/addons/husky");await m.pathExists(t)?await v("**/*",t,e,r):A.warn(C.yellow("Warning: Husky addon template not found."))}if(r.addons.includes("biome")){let t=l.join(P,"templates/addons/biome");await m.pathExists(t)?await v("**/*",t,e,r):A.warn(C.yellow("Warning: Biome addon template not found."))}if(r.addons.includes("pwa")){let t=l.join(P,"templates/addons/pwa/apps/web"),a=l.join(e,"apps/web");await m.pathExists(t)?await m.pathExists(a)?await v("**/*",t,a,r):A.warn(C.yellow("Warning: apps/web directory not found, cannot setup PWA addon.")):A.warn(C.yellow("Warning: PWA addon template not found."))}}async function Ct(e,r){if(!r.examples||r.examples.length===0)return;let t=l.join(e,"apps/server"),a=l.join(e,"apps/web");for(let o of r.examples){let n=l.join(P,`templates/examples/${o}`);if(await m.pathExists(t)){let i=l.join(n,"server");if(await m.pathExists(i)&&r.orm!=="none"){let s=l.join(i,r.orm,"base");if(await m.pathExists(s)&&await v("**/*",s,t,r,!1),r.database!=="none"){let c=l.join(i,r.orm,r.database);await m.pathExists(c)&&await v("**/*",c,t,r,!1)}}}if(await m.pathExists(a)){let i=l.join(n,"web");if(await m.pathExists(i)){let s=r.frontend.filter(c=>["next","react-router","tanstack-router","tanstack-start"].includes(c));for(let c of s){let p=l.join(i,c);await m.pathExists(p)&&await v("**/*",p,a,r,!1)}}}}}async function Et(e,r){let t=await vt(["**/.gitignore.hbs","**/_gitignore"],{cwd:e,dot:!0,onlyFiles:!0,absolute:!0,ignore:["**/node_modules/**","**/.git/**"]});for(let a of t){let o=l.dirname(a),n=l.basename(a),i=l.join(o,".gitignore");try{n===".gitignore.hbs"?(await Re(a,i,r),await m.remove(a)):n==="_gitignore"&&await m.move(a,i,{overwrite:!0})}catch(s){A.error(`Error processing gitignore file ${a}:`,s)}}}async function Tt(e,r){if(r.packageManager==="pnpm"){let t=l.join(P,"templates/extras/pnpm-workspace.yaml"),a=l.join(e,"pnpm-workspace.yaml");await m.pathExists(t)?await m.copy(t,a):A.warn(C.yellow("Warning: pnpm-workspace.yaml template not found."))}}async function Rt(e){let r=Ma(),t=Ba.resolve(process.cwd(),e.projectName);try{return await La.ensureDir(t),await kt(t,e),await xt(t,e),await St(t,e),await Ke(e),await At(t,e),await lt(e),await Dt(t,e),await He(e),await $t(t,e),e.addons.length>0&&e.addons[0]!=="none"&&await qe(e),await Ct(t,e),await Tt(t,e),e.examples.length>0&&e.examples[0]!=="none"&&await ut(e),await Ve(e),await jt(e),await dt(e),await yt(t,e),await Xe(t,e),await Pt(t,e.git),await Et(t,e),Ia.success("Project template successfully scaffolded!"),e.install&&await gt({projectDir:t,packageManager:e.packageManager,addons:e.addons}),ht({...e,depsInstalled:e.install}),t}catch(a){throw r.stop(Ft.red("Failed")),a instanceof Error&&(Na(Ft.red(`Error during project creation: ${a.message}`)),console.error(a.stack),process.exit(1)),a}}import{cancel as _o,group as zo}from"@clack/prompts";import Wo from"picocolors";import{cancel as Oa,isCancel as Ua,multiselect as _a}from"@clack/prompts";import za from"picocolors";async function Bt(e,r){if(e!==void 0)return e;let t=r?.includes("react-router")||r?.includes("tanstack-router"),a=[{value:"starlight",label:"Starlight",hint:"Add Astro Starlight documentation site"},{value:"biome",label:"Biome",hint:"Add Biome for linting and formatting"},{value:"husky",label:"Husky",hint:"Add Git hooks with Husky, lint-staged (requires Biome)"},{value:"turborepo",label:"Turborepo",hint:"Optimize builds for monorepos"}],n=t?[...[{value:"pwa",label:"PWA (Progressive Web App)",hint:"Make your app installable and work offline"},{value:"tauri",label:"Tauri Desktop App",hint:"Build native desktop apps from your web frontend"}],...a]:a,i=h.addons.filter(c=>t||c!=="pwa"&&c!=="tauri"),s=await _a({message:"Select addons",options:n,initialValues:i,required:!1});return Ua(s)&&(Oa(za.red("Operation cancelled")),process.exit(0)),s.includes("husky")&&!s.includes("biome")&&s.push("biome"),s}import{cancel as Wa,isCancel as qa,select as Va}from"@clack/prompts";import Ga from"picocolors";async function Nt(e,r){if(e)return e;let t=r?.includes("native"),a=[{value:"trpc",label:"tRPC",hint:"End-to-end typesafe APIs made easy"},{value:"orpc",label:"oRPC",hint:"End-to-end type-safe APIs that adhere to OpenAPI standards"},{value:"none",label:"None",hint:"No API integration (skip API setup)"}];t&&(a=[{value:"trpc",label:"tRPC",hint:"End-to-end typesafe APIs made easy (Required for Native frontend)"}]);let o=await Va({message:"Select API type",options:a,initialValue:t?"trpc":h.api});return qa(o)&&(Wa(Ga.red("Operation cancelled")),process.exit(0)),t&&o!=="trpc"?"trpc":o}import{cancel as Ja,confirm as Qa,isCancel as Ha}from"@clack/prompts";import Ya from"picocolors";async function It(e,r){if(!r)return!1;if(e!==void 0)return e;let t=await Qa({message:"Add authentication with Better-Auth?",initialValue:h.auth});return Ha(t)&&(Ja(Ya.red("Operation cancelled")),process.exit(0)),t}import{cancel as Ka,isCancel as Xa,select as Za}from"@clack/prompts";import eo from"picocolors";async function Mt(e){if(e!==void 0)return e;let r=await Za({message:"Select backend framework",options:[{value:"hono",label:"Hono",hint:"Lightweight, ultrafast web framework"},{value:"next",label:"Next.js",hint:"Full-stack framework with API routes"},{value:"express",label:"Express",hint:"Fast, unopinionated, minimalist web framework for Node.js"},{value:"elysia",label:"Elysia",hint:"Ergonomic web framework for building backend servers"}],initialValue:h.backend});return Xa(r)&&(Ka(eo.red("Operation cancelled")),process.exit(0)),r}import{cancel as to,isCancel as ro,select as ao}from"@clack/prompts";import oo from"picocolors";async function Lt(e){if(e!==void 0)return e;let r=await ao({message:"Select database",options:[{value:"none",label:"None",hint:"No database setup"},{value:"sqlite",label:"SQLite",hint:"lightweight, server-less, embedded relational database management system"},{value:"postgres",label:"PostgreSQL",hint:"powerful, open source object-relational database system"},{value:"mysql",label:"MySQL",hint:"popular open-source relational database system"},{value:"mongodb",label:"MongoDB",hint:"open-source NoSQL database that stores data in JSON-like documents called BSON"}],initialValue:h.database});return ro(r)&&(to(oo.red("Operation cancelled")),process.exit(0)),r}import{cancel as no,isCancel as so,select as io}from"@clack/prompts";import co from"picocolors";async function Ot(e,r,t){if(r!==void 0)return r;if(e==="sqlite"&&t==="prisma")return"none";let a=[];if(e==="sqlite")a=[{value:"turso",label:"Turso",hint:"SQLite for Production. Powered by libSQL"},{value:"none",label:"None",hint:"Manual setup"}];else if(e==="postgres")a=[{value:"neon",label:"Neon Postgres",hint:"Serverless Postgres with branching capability"},...t==="prisma"?[{value:"prisma-postgres",label:"Prisma Postgres",hint:"Instant Postgres for Global Applications"}]:[],{value:"none",label:"None",hint:"Manual setup"}];else if(e==="mongodb")a=[{value:"mongodb-atlas",label:"MongoDB Atlas",hint:"The most effective way to deploy MongoDB"},{value:"none",label:"None",hint:"Manual setup"}];else return"none";let o=await io({message:`Select ${e} setup option`,options:a,initialValue:"none"});return so(o)&&(no(co.red("Operation cancelled")),process.exit(0)),o}import{cancel as po,isCancel as lo,multiselect as Ut}from"@clack/prompts";import uo from"picocolors";async function _t(e,r,t,a){if(e!==void 0)return e;if(r==="none")return[];if(!(t?.includes("react-router")||t?.includes("tanstack-router")||t?.includes("tanstack-start")))return[];let n=[];return a==="elysia"&&(n=await Ut({message:"Include examples",options:[{value:"todo",label:"Todo App",hint:"A simple CRUD example app"}],required:!1,initialValues:h.examples})),(a==="hono"||a==="express")&&(n=await Ut({message:"Include examples",options:[{value:"todo",label:"Todo App",hint:"A simple CRUD example app"},{value:"ai",label:"AI Chat",hint:"A simple AI chat interface using AI SDK"}],required:!1,initialValues:h.examples})),lo(n)&&(po(uo.red("Operation cancelled")),process.exit(0)),n}import{cancel as zt,isCancel as Wt,multiselect as mo,select as fo}from"@clack/prompts";import qt from"picocolors";async function Vt(e){if(e!==void 0)return e;let r=await mo({message:"Select platforms to develop for",options:[{value:"web",label:"Web",hint:"React Web Application"},{value:"native",label:"Native",hint:"Create a React Native/Expo app"}],required:!1,initialValues:h.frontend.some(a=>a==="tanstack-router"||a==="react-router"||a==="tanstack-start"||a==="next")?["web"]:[]});Wt(r)&&(zt(qt.red("Operation cancelled")),process.exit(0));let t=[];if(r.includes("web")){let a=await fo({message:"Choose frontend framework",options:[{value:"tanstack-router",label:"TanStack Router",hint:"Modern and scalable routing for React Applications"},{value:"react-router",label:"React Router",hint:"A user\u2011obsessed, standards\u2011focused, multi\u2011strategy router"},{value:"next",label:"Next.js",hint:"The React Framework for the Web"},{value:"tanstack-start",label:"TanStack Start (beta)",hint:"SSR, Server Functions, API Routes and more with TanStack Router"}],initialValue:h.frontend.find(o=>o==="tanstack-router"||o==="react-router"||o==="tanstack-start"||o==="next")||"tanstack-router"});Wt(a)&&(zt(qt.red("Operation cancelled")),process.exit(0)),t.push(a)}return r.includes("native")&&t.push("native"),t}import{cancel as go,confirm as ho,isCancel as bo}from"@clack/prompts";import wo from"picocolors";async function Gt(e){if(e!==void 0)return e;let r=await ho({message:"Initialize git repository?",initialValue:h.git});return bo(r)&&(go(wo.red("Operation cancelled")),process.exit(0)),r}import{cancel as yo,confirm as Po,isCancel as jo}from"@clack/prompts";import vo from"picocolors";async function Jt(e){if(e!==void 0)return e;let r=await Po({message:"Install dependencies?",initialValue:h.install});return jo(r)&&(yo(vo.red("Operation cancelled")),process.exit(0)),r}import{cancel as ko,isCancel as xo,log as So,select as Ao}from"@clack/prompts";import Do from"picocolors";async function Qt(e,r,t){if(!r)return"none";if(e!==void 0)return e;if(t==="mongodb")return So.info("Only Prisma is supported with MongoDB."),"prisma";let a=await Ao({message:"Select ORM",options:[{value:"drizzle",label:"Drizzle",hint:"lightweight and performant TypeScript ORM"},{value:"prisma",label:"Prisma",hint:"Powerful, feature-rich ORM"}],initialValue:h.orm});return xo(a)&&(ko(Do.red("Operation cancelled")),process.exit(0)),a}import{cancel as $o,isCancel as Co,select as Eo}from"@clack/prompts";import To from"picocolors";async function Ht(e){if(e!==void 0)return e;let r=re(),t=await Eo({message:"Choose package manager",options:[{value:"npm",label:"npm",hint:"Node Package Manager"},{value:"pnpm",label:"pnpm",hint:"Fast, disk space efficient package manager"},{value:"bun",label:"bun",hint:"All-in-one JavaScript runtime & toolkit"}],initialValue:r});return Co(t)&&($o(To.red("Operation cancelled")),process.exit(0)),t}import te from"node:path";import{cancel as Fo,isCancel as Ro,text as Bo}from"@clack/prompts";import U from"fs-extra";import No from"picocolors";var Io=["<",">",":",'"',"|","?","*"],Yt=255;function Kt(e){if(e!=="."){if(!e)return"Project name cannot be empty";if(e.length>Yt)return`Project name must be less than ${Yt} characters`;if(Io.some(r=>e.includes(r)))return"Project name contains invalid characters";if(e.startsWith(".")||e.startsWith("-"))return"Project name cannot start with a dot or dash";if(e.toLowerCase()==="node_modules")return"Project name is reserved"}}async function he(e){if(e)if(e==="."){let n=process.cwd();if(U.readdirSync(n).length===0)return e}else{let n=te.basename(e);if(!Kt(n)){let s=te.resolve(process.cwd(),e);if(!U.pathExistsSync(s)||U.readdirSync(s).length===0)return e}}let r=!1,t="",a=h.projectName,o=1;for(;U.pathExistsSync(te.resolve(process.cwd(),a));)a=`${h.projectName}-${o}`,o++;for(;!r;){let n=await Bo({message:"Enter your project name or path (relative to current directory)",placeholder:a,initialValue:e,defaultValue:a,validate:i=>{let s=i.trim()||a;if(s==="."){if(U.readdirSync(process.cwd()).length>0)return"Current directory is not empty. Please choose a different directory.";r=!0;return}let c=te.resolve(process.cwd(),s),p=te.basename(c),w=Kt(p);if(w)return w;if(!c.startsWith(process.cwd()))return"Project path must be within current directory";if(U.pathExistsSync(c)&&U.readdirSync(c).length>0)return`Directory "${s}" already exists and is not empty. Please choose a different name or path.`;r=!0}});Ro(n)&&(Fo(No.red("Operation cancelled.")),process.exit(0)),t=n||a}return t}import{cancel as Mo,isCancel as Lo,select as Oo}from"@clack/prompts";import Uo from"picocolors";async function Xt(e,r){if(e!==void 0)return e;if(r==="next")return"node";let t=await Oo({message:"Select runtime",options:[{value:"bun",label:"Bun",hint:"Fast all-in-one JavaScript runtime"},{value:"node",label:"Node.js",hint:"Traditional Node.js runtime"}],initialValue:h.runtime});return Lo(t)&&(Mo(Uo.red("Operation cancelled")),process.exit(0)),t}async function Zt(e){let r=await zo({projectName:async()=>he(e.projectName),frontend:()=>Vt(e.frontend),backend:()=>Mt(e.backend),runtime:({results:t})=>Xt(e.runtime,t.backend),database:()=>Lt(e.database),orm:({results:t})=>Qt(e.orm,t.database!=="none",t.database),api:({results:t})=>Nt(e.api,t.frontend),auth:({results:t})=>It(e.auth,t.database!=="none"),addons:({results:t})=>Bt(e.addons,t.frontend),examples:({results:t})=>_t(e.examples,t.database,t.frontend,t.backend),dbSetup:({results:t})=>Ot(t.database??"none",e.dbSetup,t.orm),git:()=>Gt(e.git),packageManager:()=>Ht(e.packageManager),install:()=>Jt(e.install)},{onCancel:()=>{_o(Wo.red("Operation cancelled")),process.exit(0)}});return{projectName:r.projectName,frontend:r.frontend,database:r.database,orm:r.orm,auth:r.auth,addons:r.addons,examples:r.examples,git:r.git,packageManager:r.packageManager,install:r.install,dbSetup:r.dbSetup,backend:r.backend,runtime:r.runtime,api:r.api}}import T from"picocolors";function Be(e){let r=[];if(e.projectName&&r.push(`${T.blue("Project Name:")} ${e.projectName}`),e.frontend!==void 0){let t=Array.isArray(e.frontend)?e.frontend:[e.frontend],a=t.length>0&&t[0]!==void 0&&t[0]!==""?t.join(", "):"none";r.push(`${T.blue("Frontend:")} ${a}`)}if(e.backend!==void 0&&r.push(`${T.blue("Backend Framework:")} ${String(e.backend)}`),e.runtime!==void 0&&r.push(`${T.blue("Runtime:")} ${String(e.runtime)}`),e.api!==void 0&&r.push(`${T.blue("API:")} ${String(e.api)}`),e.database!==void 0&&r.push(`${T.blue("Database:")} ${String(e.database)}`),e.orm!==void 0&&r.push(`${T.blue("ORM:")} ${String(e.orm)}`),e.auth!==void 0){let t=typeof e.auth=="boolean"?e.auth?"Yes":"No":String(e.auth);r.push(`${T.blue("Authentication:")} ${t}`)}if(e.addons!==void 0){let t=Array.isArray(e.addons)?e.addons:[e.addons],a=t.length>0&&t[0]!==void 0?t.join(", "):"none";r.push(`${T.blue("Addons:")} ${a}`)}if(e.examples!==void 0){let t=Array.isArray(e.examples)?e.examples:[e.examples],a=t.length>0&&t[0]!==void 0?t.join(", "):"none";r.push(`${T.blue("Examples:")} ${a}`)}if(e.git!==void 0){let t=typeof e.git=="boolean"?e.git?"Yes":"No":String(e.git);r.push(`${T.blue("Git Init:")} ${t}`)}if(e.packageManager!==void 0&&r.push(`${T.blue("Package Manager:")} ${String(e.packageManager)}`),e.install!==void 0){let t=typeof e.install=="boolean"?e.install?"Yes":"No":String(e.install);r.push(`${T.blue("Install Dependencies:")} ${t}`)}return e.dbSetup!==void 0&&r.push(`${T.blue("Database Setup:")} ${String(e.dbSetup)}`),r.length===0?T.yellow("No configuration selected."):r.join(`
|
|
195
195
|
`)}function er(e){let r=[];e.database==="none"?r.push("--database none"):(r.push(`--database ${e.database}`),e.orm&&r.push(`--orm ${e.orm}`),e.dbSetup&&r.push(`--db-setup ${e.dbSetup}`)),e.api&&r.push(`--api ${e.api}`),r.push(e.auth?"--auth":"--no-auth"),r.push(e.git?"--git":"--no-git"),r.push(e.install?"--install":"--no-install"),e.runtime&&r.push(`--runtime ${e.runtime}`),e.backend&&r.push(`--backend ${e.backend}`),e.frontend&&e.frontend.length>0&&r.push(`--frontend ${e.frontend.join(" ")}`),e.addons&&e.addons.length>0?r.push(`--addons ${e.addons.join(" ")}`):r.push("--addons none"),e.examples&&e.examples.length>0?r.push(`--examples ${e.examples.join(" ")}`):r.push("--examples none"),e.packageManager&&r.push(`--package-manager ${e.packageManager}`);let t="",a=e.packageManager;a==="npm"?t="npx create-better-t-stack@latest":a==="pnpm"?t="pnpm create better-t-stack@latest":a==="bun"&&(t="bun create better-t-stack@latest");let o=e.projectName?` ${e.projectName}`:"";return`${t}${o} ${r.join(" ")}`}import qo from"node:path";import Vo from"fs-extra";var tr=()=>{let e=qo.join(P,"package.json");return Vo.readJSONSync(e).version??"1.0.0"};import rr from"gradient-string";var ar=`
|
|
196
196
|
\u2588\u2588\u2588\u2588\u2588\u2588\u2557 \u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2557\u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2557\u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2557\u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2557\u2588\u2588\u2588\u2588\u2588\u2588\u2557
|
|
197
197
|
\u2588\u2588\u2554\u2550\u2550\u2588\u2588\u2557\u2588\u2588\u2554\u2550\u2550\u2550\u2550\u255D\u255A\u2550\u2550\u2588\u2588\u2554\u2550\u2550\u255D\u255A\u2550\u2550\u2588\u2588\u2554\u2550\u2550\u255D\u2588\u2588\u2554\u2550\u2550\u2550\u2550\u255D\u2588\u2588\u2554\u2550\u2550\u2588\u2588\u2557
|
package/package.json
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
|
+
import "dotenv/config";
|
|
1
2
|
{{#if (eq runtime "node")}}
|
|
2
3
|
import { node } from "@elysiajs/node";
|
|
3
4
|
{{/if}}
|
|
4
|
-
import "dotenv/config";
|
|
5
5
|
import { Elysia } from "elysia";
|
|
6
6
|
import { cors } from "@elysiajs/cors";
|
|
7
7
|
{{#if (eq api "trpc")}}
|
|
@@ -69,4 +69,4 @@ const app = new Elysia()
|
|
|
69
69
|
.get("/", () => "OK")
|
|
70
70
|
.listen(3000, () => {
|
|
71
71
|
console.log(`Server is running on http://localhost:3000`);
|
|
72
|
-
});
|
|
72
|
+
});
|
|
@@ -1,3 +1,4 @@
|
|
|
1
|
+
import "dotenv/config";
|
|
1
2
|
{{#if (eq api "orpc")}}
|
|
2
3
|
import { RPCHandler } from "@orpc/server/fetch";
|
|
3
4
|
import { createContext } from "./lib/context";
|
|
@@ -9,7 +10,6 @@ import { auth } from "./lib/auth";
|
|
|
9
10
|
{{#if (eq api "trpc")}}
|
|
10
11
|
import { trpcServer } from "@hono/trpc-server";
|
|
11
12
|
{{/if}}
|
|
12
|
-
import "dotenv/config";
|
|
13
13
|
import { Hono } from "hono";
|
|
14
14
|
import { cors } from "hono/cors";
|
|
15
15
|
import { logger } from "hono/logger";
|
|
@@ -74,7 +74,7 @@ app.use("/trpc/*", trpcServer({
|
|
|
74
74
|
app.post("/ai", async (c) => {
|
|
75
75
|
const body = await c.req.json();
|
|
76
76
|
const messages = body.messages || [];
|
|
77
|
-
|
|
77
|
+
|
|
78
78
|
const result = streamText({
|
|
79
79
|
model: google("gemini-1.5-flash"),
|
|
80
80
|
messages,
|
|
@@ -82,7 +82,7 @@ app.post("/ai", async (c) => {
|
|
|
82
82
|
|
|
83
83
|
c.header("X-Vercel-AI-Data-Stream", "v1");
|
|
84
84
|
c.header("Content-Type", "text/plain; charset=utf-8");
|
|
85
|
-
|
|
85
|
+
|
|
86
86
|
return stream(c, (stream) => stream.pipe(result.toDataStream()));
|
|
87
87
|
});
|
|
88
88
|
{{/if}}
|
|
@@ -102,4 +102,4 @@ serve({
|
|
|
102
102
|
});
|
|
103
103
|
{{else}}
|
|
104
104
|
export default app;
|
|
105
|
-
{{/if}}
|
|
105
|
+
{{/if}}
|
|
@@ -0,0 +1,29 @@
|
|
|
1
|
+
{
|
|
2
|
+
"compilerOptions": {
|
|
3
|
+
"target": "ESNext",
|
|
4
|
+
"module": "ESNext",
|
|
5
|
+
"moduleResolution": "bundler",
|
|
6
|
+
"verbatimModuleSyntax": true,
|
|
7
|
+
"strict": true,
|
|
8
|
+
"skipLibCheck": true,
|
|
9
|
+
"baseUrl": "./",
|
|
10
|
+
"paths": {
|
|
11
|
+
"@/*": ["./src/*"]
|
|
12
|
+
},
|
|
13
|
+
"outDir": "./dist",
|
|
14
|
+
"types": [
|
|
15
|
+
{{#if (eq runtime 'node')}}
|
|
16
|
+
"node"
|
|
17
|
+
{{else if (eq runtime 'bun')}}
|
|
18
|
+
"bun"
|
|
19
|
+
{{else}}
|
|
20
|
+
"node", "bun"
|
|
21
|
+
{{/if}}
|
|
22
|
+
],
|
|
23
|
+
"jsx": "react-jsx"{{#if (eq backend 'hono')}},
|
|
24
|
+
"jsxImportSource": "hono/jsx"{{/if}}
|
|
25
|
+
},
|
|
26
|
+
"tsc-alias": {
|
|
27
|
+
"resolveFullPaths": true
|
|
28
|
+
}
|
|
29
|
+
}
|
|
@@ -1,18 +0,0 @@
|
|
|
1
|
-
{
|
|
2
|
-
"compilerOptions": {
|
|
3
|
-
"target": "ESNext",
|
|
4
|
-
"module": "ESNext",
|
|
5
|
-
"moduleResolution": "bundler",
|
|
6
|
-
"verbatimModuleSyntax": true,
|
|
7
|
-
"strict": true,
|
|
8
|
-
"skipLibCheck": true,
|
|
9
|
-
"baseUrl": "./",
|
|
10
|
-
"outDir": "./dist",
|
|
11
|
-
"types": ["node", "bun"],
|
|
12
|
-
"jsx": "react-jsx",
|
|
13
|
-
"jsxImportSource": "hono/jsx"
|
|
14
|
-
},
|
|
15
|
-
"tsc-alias": {
|
|
16
|
-
"resolveFullPaths": true
|
|
17
|
-
}
|
|
18
|
-
}
|