create-instant-app 0.22.146 → 0.22.147
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 +10 -10
- package/package.json +4 -4
package/dist/index.js
CHANGED
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
#!/usr/bin/env node
|
|
2
|
-
import{Command as ve,Option as l}from"commander";import*as $ from"@clack/prompts";import{spawn as Se}from"child_process";import{execa as Ae}from"execa";async function
|
|
2
|
+
import{Command as ve,Option as l}from"commander";import*as $ from"@clack/prompts";import{spawn as Se}from"child_process";import{execa as Ae}from"execa";async function R(){return new Promise(e=>{let t=process.env.SHELL||"/bin/bash",n=Se(t,["-i","-c","which claude"],{stdio:["ignore","pipe","ignore"]}),r="";n.stdout.on("data",a=>{r+=a.toString()}),n.on("close",a=>{if(a===0){let i=r.trim();i.includes("aliased to ")?e(i.split("aliased to ")[1]||null):e(i)}else e(null)})})}var V=async(e,t)=>{let n=await R();if(!n)throw new Error("Claude not found in path");await Ae(n,[e],{stdio:"inherit",cwd:t})};import{version as Ne}from"@instantdb/version";import W from"path";var z=e=>(e.length>1&&e.endsWith("/")&&(e=e.slice(0,-1)),e),Pe=/^(?:@[a-z0-9-*~][a-z0-9-*._~]*\/)?[a-z0-9-~][a-z0-9-._~]*$/,J=e=>{let n=z(e).split("/"),r=n[n.length-1];if(r==="."){let o=W.resolve(process.cwd());r=W.basename(o)}let a=n.findIndex(o=>o.startsWith("@"));n.findIndex(o=>o.startsWith("@"))!==-1&&(r=n.slice(a).join("/"));let i=n.filter(o=>!o.startsWith("@")).join("/");return[r,i]},j=e=>e.trim(),F=e=>{let t=z(e),n=t.split("/"),r=n.findIndex(i=>i.startsWith("@")),a=n[n.length-1];if(n.findIndex(i=>i.startsWith("@"))!==-1&&(a=n.slice(r).join("/")),!(t==="."||Pe.test(a??"")))return"App name must consist of only lowercase alphanumeric characters, '-', and '_'"};import{renderUnwrap as v,UI as m}from"instant-cli/ui";var Ee={base:"next-js-app-dir",appName:"Awesome Todos",ruleFiles:null,createRepo:!0,prompt:null},H=async()=>{let e=Ee,t=new ve().name("Create Instant App").description("A CLI for creating web/mobile applications with InstantDB").argument("[dir]","The name of the application, as well as the name of the directory to create").addOption(new l("-b --base <template>","The base template to scaffold from").choices(["next-js-app-dir","vite-vanilla","expo","bun-react","tanstack-start","tanstack-start-with-tanstack-query","solidjs-vite","vercel-ai-sdk"])).addOption(new l("-g --git","Create a git repo in the new project").default(!0)).addOption(new l("--expo","Use the Expo starter template").default(!1)).addOption(new l("--next","Use the NextJS starter template").default(!1)).addOption(new l("--vanilla","Use the vanilla JS starter template").default(!1)).addOption(new l("--no-git","Don't create a git repo in the new project")).addOption(new l("--cursor","Include a Cursor rules file in the scaffold")).addOption(new l("--claude","Include a CLAUDE.md file in the scaffold")).addOption(new l("--codex","Include an AGENTS.md file in the scaffold")).addOption(new l("--gemini","Include a GEMINI.md file in the scaffold")).addOption(new l("--rules","Include an AGENTS.md file in the scaffold")).addOption(new l("--ai","Create a new InstantDB app based off of a prompt. (requires Claude Code)")).addOption(new l("-a --app <app-id>","Link to an existing InstantDB app by ID (requires login or --token)")).addOption(new l("-t --token <token>","Auth token override (use with --app when not logged in)")).version(Ne).parse(process.argv),n=t.args[0]&&j(t.args[0]);if(n){let o=F(n);if(o)throw new Error("Invalid app name: "+o);e.appName=n}let r=t.opts();if(r.ai&&!await R())throw new Error("--ai only works with Claude Code, but we couldn't find it in your machine. Install it first, and run it again : ). Alternatively you can scaffold out a project without --ai");let a=await $.group({appName:async()=>{if(n)return n.trim();let o=await v(new m.TextInput({prompt:"What will your project/folder be called?",placeholder:"awesome-todos",defaultValue:"awesome-todos",validate:p=>F(j(p)),modifyOutput:m.ciaModifier()}));return j(o)},prompt:async()=>r.ai?await v(new m.TextInput({prompt:"What would you like to create?",placeholder:"Create an app that...",modifyOutput:m.modifiers.piped([m.ciaModifier()])})):null,base:async({results:o})=>r.base?r.base:r.vanilla?"vite-vanilla":r.next?"next-js-app-dir":r.expo?"expo":o.prompt?v(new m.Select({promptText:"What framework would you like to use?",options:[{value:"next-js-app-dir",label:"Next.js"},{value:"expo",label:"Expo: React Native"}],defaultValue:"next-js-app-dir",modifyOutput:m.modifiers.piped([m.ciaModifier()])})):v(new m.Select({promptText:"What framework would you like to use?",options:[{value:"next-js-app-dir",label:"Web: Next.js"},{value:"expo",label:"Mobile: Expo"},{value:"vite-vanilla",label:"Vite: Vanilla TS",secondary:!0},{value:"tanstack-start",label:"Tanstack Start",secondary:!0},{value:"bun-react",label:"Bun + React",secondary:!0},{value:"solidjs-vite",label:"Vite: SolidJS",secondary:!0},{value:"vercel-ai-sdk",label:"Vercel AI SDK + SSR",secondary:!0}],defaultValue:"next-js-app-dir",modifyOutput:m.modifiers.piped([m.ciaModifier()])})),ruleFiles:async({results:o})=>o.prompt?"claude":r.cursor?"cursor":r.claude?"claude":r.codex?"codex":r.gemini?"gemini":r.rules?"codex":v(new m.Select({promptText:"Which AI tool would you like to add rule files for?",options:[{value:"claude",label:"Claude"},{value:"cursor",label:"Cursor"},{value:"codex",label:"Codex"},{value:"gemini",label:"Gemini"},{value:"zed",label:"Zed"},{value:"windsurf",label:"Windsurf"},{value:null,label:"None"}],defaultValue:"claude",modifyOutput:m.ciaModifier()})),createRepo:async()=>r.git!==void 0?r.git:!0},{onCancel(){process.exit(1)}}),i={app:r.app??null,token:r.token??null};return{project:a,appFlags:i}};import ke from"path";import xe from"fs-extra";import{log as _e,outro as bt}from"@clack/prompts";import{intro as je}from"@clack/prompts";import K from"chalk";var Ce=` _ _ _
|
|
3
3
|
\u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2588 (_) | | | |
|
|
4
4
|
\u2588 \u2588\u2588\u2588\u2588 _ _ __ ___| |_ __ _ _ _ _| |_
|
|
5
5
|
\u2588 \u2588\u2588\u2588\u2588 | | '_ \\/ __| __/ _\\\`| '_ \\| __|
|
|
@@ -7,18 +7,18 @@ import{Command as ve,Option as l}from"commander";import*as $ from"@clack/prompts
|
|
|
7
7
|
\u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2588 |_|_| |_|___/\\__\\__,_|_| |_|\\__|`,Y=()=>{je(`
|
|
8
8
|
`+Ce.split(`
|
|
9
9
|
`).map(e=>`${K.gray("\u2502")}${K.hex("#EA580D").bold(" "+e)}`).join(`
|
|
10
|
-
`))};import f from"path";import
|
|
11
|
-
enable-pre-post-scripts=true`),e.base==="expo"){let i=$e.default(t);D(f.join(n,"app.json"),'"name": "expo-template"',`"name": "${t}"`),D(f.join(n,"app.json"),'"slug": "expo-template"',`"slug": "${i}"`),D(f.join(n,"app/_layout.tsx"),'"My Instant App"',`"${t}"`)}return n},D=(e,t,n)=>{let a=c.readFileSync(e,"utf8").replaceAll(t,n);c.writeFileSync(e,a)},Ge=async({projectDir:e,baseTemplateName:t})=>{let n=`instantdb/instant/examples/${t}`;await
|
|
12
|
-
Using git to clone from main...`),w.ciaModifier(null)),await Ge({projectDir:e,baseTemplateName:t})};import{execa as Le}from"execa";import{renderUnwrap as
|
|
10
|
+
`))};import f from"path";import Fe from"degit";import c from"fs-extra";import q from"path";import{fileURLToPath as Oe}from"url";var Me=Oe(import.meta.url),Re=q.dirname(Me),h=q.join(Re,"../");import I from"chalk";var k=e=>{if(e==="bun-react")return"bun";let t=process.env.npm_config_user_agent;return t?t.startsWith("yarn")?"yarn":t.startsWith("pnpm")?"pnpm":t.startsWith("bun")?"bun":"npm":"npm"};import{renderUnwrap as X,UI as w}from"instant-cli/ui";import $e from"slugify";import De from"ignore";var Z=async(e,t)=>{let n=f.resolve(process.cwd(),t);if(c.existsSync(n))if(c.readdirSync(n).length===0)w.log(`${I.cyan.bold(e.appName)} exists but is empty, continuing...`,w.ciaModifier(null));else{let i=await X(new w.Select({promptText:I.redBright(`${I.bold("Warning:")} ${I.bold(e.appName)} already exists and isn't empty. How would you like to proceed?`),options:[{label:"Abort installation",value:"abort"},{label:"Clear the directory and continue installation",value:"clear"}],defaultValue:"abort",modifyOutput:w.ciaModifier()}));i==="abort"&&(w.log("Aborting installation..."),process.exit(1)),i==="clear"&&c.emptyDirSync(n)}let r=Ue({projectDir:n,baseTemplateName:e.base}),a=e.appName==="."?"App":I.hex("#EA570B").bold(e.appName);if(await X(new w.Spinner({promise:r,workingText:"Scaffolding project files...",doneText:`Successfully scaffolded ${a}!`,errorText:"Error scaffolding project files",modifyOutput:w.ciaModifier(null)})),c.pathExistsSync(f.join(n,"pnpm-lock.yaml"))&&c.removeSync(f.join(n,"pnpm-lock.yaml")),c.pathExistsSync(f.join(n,"bun.lock"))&&c.removeSync(f.join(n,"bun.lock")),k(e.base)==="pnpm"&&e.base==="expo"&&await c.appendFile(f.join(n,".npmrc"),`node-linker=hoisted
|
|
11
|
+
enable-pre-post-scripts=true`),e.base==="expo"){let i=$e.default(t);D(f.join(n,"app.json"),'"name": "expo-template"',`"name": "${t}"`),D(f.join(n,"app.json"),'"slug": "expo-template"',`"slug": "${i}"`),D(f.join(n,"app/_layout.tsx"),'"My Instant App"',`"${t}"`)}return n},D=(e,t,n)=>{let a=c.readFileSync(e,"utf8").replaceAll(t,n);c.writeFileSync(e,a)},Ge=async({projectDir:e,baseTemplateName:t})=>{let n=`instantdb/instant/examples/${t}`;await Fe(n,{mode:"tar",cache:!1}).clone(e)};async function Be(e,t){let n=De();n.add(".git");try{let r=f.join(e,".gitignore"),a=await c.readFile(r,"utf8");n.add(a)}catch{}await c.copy(e,t,{filter:r=>{let a=f.relative(e,r);return a===""?!0:!n.ignores(a)}})}var Ue=async({projectDir:e,baseTemplateName:t})=>{let n=f.join(h,`template/base/${t}`),r=!!process.env.INSTANT_CLI_DEV&&!!process.env.INSTANT_REPO_FOLDER,a=c.pathExistsSync(n),i=r?"dev":a?"bundled-template":"degit";if(i==="bundled-template"){c.copySync(n,e);return}if(i==="dev"){let o=process.env.INSTANT_REPO_FOLDER;if(!o)throw new Error("INSTANT_REPO_FOLDER is required when using repo-examples scaffolding.");let s=f.join(o,"examples",t);await Be(s,e);return}process.env.INSTANT_CLI_DEV&&!process.env.INSTANT_REPO_FOLDER&&w.log(I.bold.yellowBright(`WARNING: INSTANT_CLI_DEV is TRUE but no INSTANT_REPO_FOLDER is set.
|
|
12
|
+
Using git to clone from main...`),w.ciaModifier(null)),await Ge({projectDir:e,baseTemplateName:t})};import{execa as Le}from"execa";import{renderUnwrap as Ve,UI as C}from"instant-cli/ui";var Q=async(e,t)=>{let n=Le(e,["install"],{cwd:t}),r=await Ve(new C.Spinner({promise:n,workingText:`Installing dependencies with ${e}...`,doneText:"Successfully installed dependencies!",modifyOutput:C.ciaModifier(null)}));r.exitCode!==0&&(C.log(r.stderr,C.ciaModifier(null)),process.exit(1))};import kt from"chalk";import T from"fs-extra";import d from"path";var ee=({projectDir:e,ruleFilesToAdd:t})=>{if(t!==null)switch(t){case"cursor":T.ensureDirSync(d.join(e,".cursor/rules")),T.copyFileSync(d.join(h,"template/rules/cursor-rules.md"),d.join(e,".cursor/rules/instant.mdc"));break;case"claude":T.copyFileSync(d.join(h,"template/rules/AGENTS.md"),d.join(e,"CLAUDE.md"));break;case"codex":T.copyFileSync(d.join(h,"template/rules/AGENTS.md"),d.join(e,"AGENTS.md"));break;case"gemini":T.copyFileSync(d.join(h,"template/rules/AGENTS.md"),d.join(e,"GEMINI.md"));break;case"zed":T.copyFileSync(d.join(h,"template/rules/AGENTS.md"),d.join(e,"AGENTS.md"));break;case"windsurf":T.ensureDirSync(d.join(e,".windsurf/rules")),T.copyFileSync(d.join(h,"template/rules/windsurf-rules.md"),d.join(e,".windsurf/rules/instant.md"));break}};import{execSync as B}from"child_process";import G from"path";import*as ne from"@clack/prompts";import O from"chalk";import{execa as A}from"execa";import re from"fs-extra";import{renderUnwrap as te,UI as S}from"instant-cli/ui";var We=e=>{try{return B("git --version",{cwd:e}),!0}catch{return!1}},ze=e=>re.existsSync(G.join(e,".git")),Je=async e=>{try{return await A("git",["rev-parse","--is-inside-work-tree"],{cwd:e,stdout:"ignore"}),!0}catch{return!1}},He=()=>{let t=B("git --version").toString().trim().split(" ")[2],n=t?.split(".")[0],r=t?.split(".")[1];return{major:Number(n),minor:Number(r)}},Ke=()=>B("git config --global init.defaultBranch || echo main").toString().trim(),oe=async e=>{if(!We(e)){ne.log.warn("Git is not installed. Skipping Git initialization.");return}let t=ze(e),n=await Je(e),r=G.parse(e).name;if(n&&t){if(!await te(new S.Confirmation({promptText:`${O.redBright.bold("Warning:")} Git is already initialized in "${r}". Initializing a new git repository would delete the previous history. Would you like to continue anyways?`,defaultValue:!1,modifyOutput:S.ciaModifier()})))return;re.removeSync(G.join(e,".git"))}else if(n&&!t&&!await te(new S.Confirmation({promptText:`${O.redBright.bold("Warning:")} "${r}" is already in a git worktree.
|
|
13
13
|
Would you still like to initialize a new git repository in this directory?
|
|
14
|
-
`,defaultValue:!1,modifyOutput:S.ciaModifier()})))return;try{let a=Ke(),{major:i,minor:
|
|
15
|
-
INSTANT_APP_ADMIN_TOKEN=${
|
|
16
|
-
`),i=!0;a.forEach(
|
|
14
|
+
`,defaultValue:!1,modifyOutput:S.ciaModifier()})))return;try{let a=Ke(),{major:i,minor:o}=He();i<2||i==2&&o<28?(await A("git",["init"],{cwd:e}),await A("git",["symbolic-ref","HEAD",`refs/heads/${a}`],{cwd:e})):await A("git",["init",`--initial-branch=${a}`],{cwd:e}),await A("git",["add","."],{cwd:e}),await A("git",["commit","-m","Initial commit (create-instant-app)"],{cwd:e}),S.log(O.dim(`${O.green("\u2713")} Git repository initialized successfully.`),S.ciaModifier(null))}catch{}};import qe from"env-paths";import*as Xe from"@clack/prompts";import{mkdir as Ze,readFile as Qe,writeFile as et}from"node:fs/promises";import tt from"open";import{join as nt}from"node:path";import{randomUUID as se}from"node:crypto";import{version as Ye}from"@instantdb/version";var ae=!!process.env.INSTANT_CLI_DEV,ie=ae?"http://localhost:3000":"https://instantdb.com",U=process.env.INSTANT_CLI_API_URI||(ae?"http://localhost:8888":"https://api.instantdb.com");async function b({path:e,body:t,method:n="GET",authToken:r,metadata:a}){let o={"Content-Type":"application/json","X-Instant-Source":"create-instant-app","X-Instant-Version":Ye,"X-Instant-Command":"create"};r&&(o.Authorization=`Bearer ${r}`),a&&(o["X-Instant-Metadata"]=JSON.stringify(a));let s=await fetch(`${U}${e}`,{method:n??"GET",headers:o,body:t?JSON.stringify(t):void 0,signal:AbortSignal.timeout(3e5)}),p;try{p=await s.json()}catch{p=null}if(!s.ok){let g=p.message||p.hint?.errors?.[0]?.message||"There was an error";throw new Error(g)}return p}import{renderUnwrap as M,UI as u}from"instant-cli/ui";var rt=!!process.env.INSTANT_CLI_DEV,ot=!!process.env.INSTANT_CLI_FORCE_EPHEMERAL;function le(){let e=`instantdb-${rt?"dev":"prod"}`,{config:t}=qe(e);return{authConfigFilePath:nt(t,"a"),appConfigDirPath:t}}var at=async(e,t,n,r)=>{let a=se(),i=se(),o={id:a,title:e,admin_token:i,org_id:n};return r?.rules&&(o.rules=r.rules),r?.schema&&(o.schema=r.schema),await b({method:"POST",authToken:t,path:"/dash/apps",body:o,metadata:r}),{appID:a,adminToken:i,source:"created"}},L=(e,t,n)=>{b({method:"POST",path:`/dash/apps/${e}/track-import`,authToken:t,metadata:n}).catch(()=>{})},ce=async e=>await b({method:"GET",path:"/dash",authToken:e}),de=async(e,t)=>await b({method:"GET",path:`/dash/orgs/${t}`,authToken:e});var it=async(e,t)=>{try{let n=await ce(e),r=n.apps.find(a=>a.id===t);if(r)return{appId:r.id,adminToken:r.admin_token};for(let a of n.orgs){let{apps:i}=await de(e,a.id),o=i.find(s=>s.id===t);if(o)return{appId:o.id,adminToken:o.admin_token}}return null}catch{return null}},st=async(e,t)=>{try{return await b({method:"GET",path:`/dash/apps/${e}/schema/pull`,authToken:t}),!0}catch{return!1}},pt=async()=>ot?null:process.env.INSTANT_CLI_AUTH_TOKEN?process.env.INSTANT_CLI_AUTH_TOKEN:await Qe(le().authConfigFilePath,"utf-8").catch(()=>null),pe=async(e,t)=>{let n=await b({authToken:null,method:"POST",path:"/dash/apps/ephemeral",body:{title:e,rules:{$users:{view:"true"},$files:{allow:{view:"true",create:"true",delete:"true"}}}},metadata:t});return{appId:n.app.id,adminToken:n.app["admin-token"]}},ue=async(e,t)=>{let n=await pt();if(e?.app){if(e.token){if(!await st(e.app,e.token))throw new Error("Invalid app ID and token combination. Please verify both the app ID and token are correct.");return u.log(`Linking to app: ${e.app}`,u.ciaModifier(null)),L(e.app,e.token,t),{appId:e.app,adminToken:e.token,approach:"import"}}if(n){let o=await it(n,e.app);if(!o)throw new Error(`You don't have access to app "${e.app}". Please check the app ID or use --token to provide a token.`);return u.log(`Linking to app: ${e.app}`,u.ciaModifier(null)),L(o.appId,n,t),{appId:o.appId,adminToken:o.adminToken,approach:"import"}}throw new Error(`You must be logged in or provide --token when using --app. Either run 'npx instant-cli login' first, or use: --app ${e.app} --token <token>`)}if(!n){let o=await M(new u.Select({promptText:"You are not logged in.",options:[{label:"Login to your Instant account",value:"login"},{label:"Create a temporary app",value:"ephemeral"},{label:"Create an app later",value:"skip"}],modifyOutput:u.ciaModifier()}));if(o==="login"){let s=await b({authToken:null,method:"POST",path:"/dash/cli/auth/register"}).catch(Ie=>{throw new Error("Failed to register",{cause:Ie})}),{secret:p,ticket:g}=s;tt(`${ie}/dash?ticket=${g}`);let _=ct({secret:p}),y=await M(new u.Spinner({promise:_,workingText:"Waiting for login in browser",disappearWhenDone:!0,modifyOutput:u.ciaModifier(null)}));await dt(y.token),n=y.token}if(o==="skip")return u.log("Skipping app link step",u.ciaModifier(null)),null;if(o==="ephemeral"){let s=await M(new u.TextInput({defaultValue:"my-cool-app",prompt:"Enter a name for your temporary app:",placeholder:"my-cool-app",modifyOutput:u.ciaModifier()}));return{...await pe(s,t),approach:"ephemeral"}}}if(!n)return null;let r=await ce(n).catch(o=>{throw new Error("Failed to fetch dashboard",{cause:o})}),a=r.orgs.filter(o=>o.role!=="app-member");r.orgs=a;let i=await M(new u.AppSelector({startingMenuIndex:0,allowCreate:!0,allowEphemeral:!0,api:{getDash(){return r},createEphemeralApp(o){return pe(o,t)},getAppsForOrg:async o=>{let{apps:s}=await de(n,o);return{apps:s}},createApp:async(o,s)=>{let{appID:p,adminToken:g}=await at(o,n,s,t);return{appId:p,adminToken:g}}},modifyOutput:u.ciaModifier()}));return i?.approach==="import"&&L(i.appId,n,t),i};function lt(e){return new Promise(t=>setTimeout(t,e))}async function ct({secret:e}){for(let t=1;t<=120;t++){await lt(1e3);let n=await fetch(`${U}/dash/cli/auth/check`,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({secret:e})});if(!(!n.ok&&(await n.json()).hint.errors?.[0]?.issue==="waiting-for-user")&&n.ok)return n.json()}throw new Error("Timed out waiting for login")}async function dt(e){let t=le();return await Ze(t.appConfigDirPath,{recursive:!0}),et(t.authConfigFilePath,e,"utf-8")}import ut from"fs-extra";import mt from"path";var ft={"next-js-app-dir":"NEXT_PUBLIC_INSTANT_APP_ID","vite-vanilla":"VITE_INSTANT_APP_ID",expo:"EXPO_PUBLIC_INSTANT_APP_ID","tanstack-start":"VITE_INSTANT_APP_ID","bun-react":"BUN_PUBLIC_INSTANT_APP_ID","solidjs-vite":"VITE_INSTANT_APP_ID","tanstack-start-with-tanstack-query":"VITE_INSTANT_APP_ID","vercel-ai-sdk":"NEXT_PUBLIC_INSTANT_APP_ID"},me=(e,t,n,r)=>{let a=mt.join(t,".env"),o=`${ft[e.base]}=${n}
|
|
15
|
+
INSTANT_APP_ADMIN_TOKEN=${r}`;ut.writeFileSync(a,o)};import{stdin as x,stdout as fe}from"process";import{setRawModeWindowsFriendly as ge}from"instant-cli/ui";async function gt(){return new Promise(e=>{if(!fe.isTTY||!x.isTTY){e(null);return}let t=x.isRaw;t||ge(x,!0);let n=setTimeout(()=>{a(),e(null)},100),r="",a=()=>{clearTimeout(n),x.removeListener("data",i),!t&&x.isTTY&&ge(x,!1)},i=o=>{let s=o.toString();r+=s;let p=r.match(/\x1b\]11;rgb:([0-9a-fA-F]+)\/([0-9a-fA-F]+)\/([0-9a-fA-F]+)(?:\x1b\\|\x07)/);if(p){a();let[,g,_,y]=p;e(`rgb:${g}/${_}/${y}`)}};x.on("data",i),fe.write("\x1B]11;?\x1B\\")})}function ht(e){let t=e.match(/rgb:([0-9a-fA-F]+)\/([0-9a-fA-F]+)\/([0-9a-fA-F]+)/);if(!t)return null;let[,n,r,a]=t,i=Math.round(parseInt(n,16)/257),o=Math.round(parseInt(r,16)/257),s=Math.round(parseInt(a,16)/257);return{r:i,g:o,b:s}}function wt(e){return(.299*e.r+.587*e.g+.114*e.b)/255>.5}async function he(){try{let e=await gt();if(!e)return"unknown";let t=ht(e);return t?wt(t)?"light":"dark":"unknown"}catch(e){return console.error("Error detecting terminal theme:",e),"unknown"}}import N from"chalk";var P=(e,t)=>e==="light"?N.bgYellowBright(t):N.bgBlackBright(t),we=(e,t=console.log,n=!1)=>{let r=(process.stdout.columns||80)-4,a=e.split(`
|
|
16
|
+
`),i=!0;a.forEach(o=>{if(o.length===0){i?(t(""),i=!1):console.log(N.gray("\u2502"));return}for(let s=0;s<o.length;s+=r){let p=o.slice(s,s+r);s+r<o.length&&p.length===r&&(p=p+"-"),i?(t(p),i=!1):console.log(N.gray("\u2502 ")+p)}}),n&&console.log(N.gray("\u2502"))};var E="\x1B[?25h";import{execa as xt}from"execa";import ye from"fs-extra";import{permsTypescriptFileToCode as yt,schemaTypescriptFileToInstantSchema as Tt}from"@instantdb/platform";function Te(e){for(let t of["src/instant.perms.ts","instant.perms.ts"])try{let n=ye.readFileSync(`${e}/${t}`,"utf8");return{code:yt(n,t)}}catch{}return null}function be(e){for(let t of["src/instant.schema.ts","instant.schema.ts"])try{let n=ye.readFileSync(`${e}/${t}`,"utf8");return Tt(n,t)}catch{}return null}var _t=async()=>{process.argv.some(y=>["-h","--help","--version","-V"].includes(y))||Y();let e=await he(),{project:t,appFlags:n}=await H(),[r,a]=J(t.appName),i=k(t.base),o=await Z(t,a);ee({projectDir:o,ruleFilesToAdd:t.ruleFiles});let s={template:t.base,aiTool:t.ruleFiles??"none",usedAiPrompt:!!t.prompt,rules:Te(o),schema:be(o)},p=await ue(n,s);p&&me(t,o,p.appId,p.adminToken);let g=xe.readJSONSync(ke.join(o,"package.json"));if(g.name=r,i!=="bun"){let{stdout:y}=await xt(i,["-v"],{cwd:o});g.packageManager=`${i}@${y.trim()}`}xe.writeJSONSync(ke.join(o,"package.json"),g,{spaces:2}),await Q(k(t.base),o),t.createRepo&&await oe(o),t.prompt&&(await V(t.prompt,o),process.stdout.write(E)),bt("Done!");let _=t.base==="expo"?"start":"dev";p?(console.log(`
|
|
17
17
|
\u{1F389} Success! Your project is ready to go!
|
|
18
18
|
|
|
19
19
|
To get started:
|
|
20
20
|
1. ${P(e,"cd "+a)}
|
|
21
|
-
2. ${P(e,
|
|
21
|
+
2. ${P(e,k(t.base)+" run "+_)}
|
|
22
22
|
`),p.approach==="ephemeral"&&console.log(`
|
|
23
23
|
An ephemeral app has been created and added to your .env file.
|
|
24
24
|
It will expire in two weeks. For a permanent app, sign in and use ${P(e,"npx instant-cli claim")}
|
|
@@ -27,7 +27,7 @@ INSTANT_APP_ADMIN_TOKEN=${o}`;ut.writeFileSync(a,r)};import{stdin as k,stdout as
|
|
|
27
27
|
|
|
28
28
|
To get started:
|
|
29
29
|
1. ${P(e,"cd "+a)}
|
|
30
|
-
2. Create a new app on ${
|
|
30
|
+
2. Create a new app on ${kt.underline("www.instantdb.com")}
|
|
31
31
|
3. Add your APP_ID to the .env file
|
|
32
|
-
4. ${P(e,
|
|
32
|
+
4. ${P(e,k(t.base)+" run "+_)}
|
|
33
33
|
`),process.stdout.write(E),process.exit(0)};_t().catch(e=>{_e.error("Aborting installation..."),we(e.message,_e.error),process.stdout.write(E),process.exit(1)});process.on("SIGINT",()=>{process.stdout.write(E),process.exit(0)});
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "create-instant-app",
|
|
3
|
-
"version": "0.22.
|
|
3
|
+
"version": "0.22.147",
|
|
4
4
|
"description": "Scaffold a new web/mobile app with InstantDB",
|
|
5
5
|
"homepage": "https://github.com/instantdb/instant/tree/main/client/packages/create-instant-app",
|
|
6
6
|
"repository": {
|
|
@@ -36,9 +36,9 @@
|
|
|
36
36
|
"ora": "6.3.1",
|
|
37
37
|
"slugify": "^1.6.6",
|
|
38
38
|
"sort-package-json": "^2.10.0",
|
|
39
|
-
"@instantdb/version": "0.22.
|
|
40
|
-
"@instantdb/platform": "0.22.
|
|
41
|
-
"instant-cli": "0.22.
|
|
39
|
+
"@instantdb/version": "0.22.147",
|
|
40
|
+
"@instantdb/platform": "0.22.147",
|
|
41
|
+
"instant-cli": "0.22.147"
|
|
42
42
|
},
|
|
43
43
|
"devDependencies": {
|
|
44
44
|
"@anthropic-ai/sdk": "^0.60.0",
|