create-turbo-kit 1.2.1 → 1.2.2
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/{chunk-SGKVPRAI.js → chunk-KY4U5QQF.js} +39 -2
- package/dist/{chunk-SGKVPRAI.js.map → chunk-KY4U5QQF.js.map} +1 -1
- package/dist/{chunk-7W32JAIP.js → chunk-MZO3S75B.js} +4 -1
- package/dist/chunk-MZO3S75B.js.map +1 -0
- package/dist/cli.js +16 -4
- package/dist/cli.js.map +1 -1
- package/dist/helpers/scaffold.js +1 -1
- package/dist/helpers/transform.d.ts +5 -1
- package/dist/helpers/transform.js +5 -1
- package/package.json +1 -1
- package/src/cli.ts +17 -4
- package/src/helpers/scaffold.ts +7 -1
- package/src/helpers/transform.ts +49 -0
- package/dist/chunk-7W32JAIP.js.map +0 -1
|
@@ -166,12 +166,49 @@ async function configureDockerCompose(projectDir, selectedContainers) {
|
|
|
166
166
|
throw error;
|
|
167
167
|
}
|
|
168
168
|
}
|
|
169
|
+
async function removePackages(packages, packagePath, projectDir, s) {
|
|
170
|
+
const packageJsonPath = path.join(projectDir, packagePath, "package.json");
|
|
171
|
+
if (await fs.pathExists(packageJsonPath)) {
|
|
172
|
+
const packageJson = await fs.readJson(packageJsonPath);
|
|
173
|
+
let hasChanges = false;
|
|
174
|
+
for (const dep of packages) {
|
|
175
|
+
if (packageJson.dependencies && packageJson.dependencies[dep]) {
|
|
176
|
+
delete packageJson.dependencies[dep];
|
|
177
|
+
hasChanges = true;
|
|
178
|
+
if (s) {
|
|
179
|
+
s.message(`Removed ${color.cyan(dep)} from ${packagePath}/package.json`);
|
|
180
|
+
}
|
|
181
|
+
}
|
|
182
|
+
}
|
|
183
|
+
if (hasChanges) {
|
|
184
|
+
await fs.writeJson(packageJsonPath, packageJson, { spaces: 2 });
|
|
185
|
+
}
|
|
186
|
+
}
|
|
187
|
+
}
|
|
188
|
+
async function removeReactEmail(projectDir) {
|
|
189
|
+
const s = spinner();
|
|
190
|
+
s.start("Removing react-email...");
|
|
191
|
+
try {
|
|
192
|
+
const emailPackagePath = path.join(projectDir, "packages", "email");
|
|
193
|
+
if (await fs.pathExists(emailPackagePath)) {
|
|
194
|
+
await fs.remove(emailPackagePath);
|
|
195
|
+
s.message(`Removed ${color.cyan(emailPackagePath)}`);
|
|
196
|
+
}
|
|
197
|
+
await removePackages(["@react-email/components", "@acme/email"], "packages/api", projectDir, s);
|
|
198
|
+
s.stop("Removed react-email");
|
|
199
|
+
} catch (error) {
|
|
200
|
+
s.stop("Failed to remove react-email");
|
|
201
|
+
throw error;
|
|
202
|
+
}
|
|
203
|
+
}
|
|
169
204
|
|
|
170
205
|
export {
|
|
171
206
|
replaceScope,
|
|
172
207
|
setupEnv,
|
|
173
208
|
getDockerContainers,
|
|
174
209
|
deleteDockerCompose,
|
|
175
|
-
configureDockerCompose
|
|
210
|
+
configureDockerCompose,
|
|
211
|
+
removePackages,
|
|
212
|
+
removeReactEmail
|
|
176
213
|
};
|
|
177
|
-
//# sourceMappingURL=chunk-
|
|
214
|
+
//# sourceMappingURL=chunk-KY4U5QQF.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/helpers/transform.ts"],"sourcesContent":["import fg from 'fast-glob';\r\nimport fs from 'fs-extra';\r\nimport path from 'path';\r\nimport { spinner } from '@clack/prompts';\r\nimport color from 'picocolors';\r\n\r\nconst DOCKER_CONTAINERS = {\r\n postgres: {\r\n label: 'PostgreSQL 16 (Database)',\r\n services: ['postgres'],\r\n volumes: ['postgres_data'],\r\n dependsOn: [], \r\n },\r\n pgadmin: {\r\n label: `pgAdmin 4.9 (Database management) [Depends on PostgreSQL]`,\r\n services: ['pgadmin'],\r\n volumes: [],\r\n dependsOn: ['postgres'],\r\n },\r\n redis: {\r\n label: 'Redis (Caching)',\r\n services: ['redis'],\r\n volumes: [],\r\n dependsOn: [],\r\n },\r\n mailpit: {\r\n label: 'Mailpit (Email testing)',\r\n services: ['mailpit'],\r\n volumes: [],\r\n dependsOn: [],\r\n },\r\n minio: {\r\n label: 'MinIO (S3-compatible storage)',\r\n services: ['minio', 'minio-create-bucket'],\r\n volumes: ['minio_data'],\r\n dependsOn: [],\r\n },\r\n} as const;\r\n\r\nexport type DockerContainer = keyof typeof DOCKER_CONTAINERS;\r\n\r\nexport async function replaceScope(projectDir: string, newScope: string) {\r\n const s = spinner();\r\n s.start(`Replacing scope with ${color.cyan(newScope)}...`);\r\n\r\n try {\r\n const files = await fg('**/*', {\r\n cwd: projectDir,\r\n ignore: [\r\n '**/.git/**',\r\n '**/node_modules/**',\r\n '**/.turbo/**',\r\n '**/dist/**',\r\n '**/.next/**',\r\n '**/pnpm-lock.yaml',\r\n '**/yarn.lock',\r\n '**/package-lock.json',\r\n '**/bun.lockb',\r\n ],\r\n absolute: true,\r\n });\r\n\r\n await Promise.all(\r\n files.map(async (file) => {\r\n try {\r\n const content = await fs.readFile(file, 'utf8');\r\n if (content.includes('@acme')) {\r\n const newContent = content.replace(/@acme/g, newScope);\r\n await fs.writeFile(file, newContent);\r\n }\r\n } catch (e) {\r\n }\r\n })\r\n );\r\n \r\n s.stop(`Replaced scope with ${color.cyan(newScope)}`);\r\n } catch (error) {\r\n s.stop('Failed to replace scope');\r\n throw error;\r\n }\r\n}\r\n\r\nexport async function setupEnv(projectDir: string) {\r\n const envExample = path.join(projectDir, '.env.example');\r\n const envDest = path.join(projectDir, '.env');\r\n\r\n if (await fs.pathExists(envExample)) {\r\n await fs.copy(envExample, envDest);\r\n }\r\n}\r\n\r\nexport function getDockerContainers() {\r\n return Object.entries(DOCKER_CONTAINERS).map(([value, config]) => ({\r\n value,\r\n label: config.label,\r\n }));\r\n}\r\n\r\nexport async function deleteDockerCompose(projectDir: string) {\r\n const s = spinner();\r\n s.start('Removing Docker Compose setup...');\r\n\r\n try {\r\n const dockerComposePath = path.join(projectDir, 'docker-compose.yml');\r\n if (await fs.pathExists(dockerComposePath)) {\r\n await fs.remove(dockerComposePath);\r\n s.message(`Removed ${color.cyan(dockerComposePath)}`);\r\n }\r\n const dockerDir = path.join(projectDir, 'docker');\r\n if (await fs.pathExists(dockerDir)) {\r\n await fs.remove(dockerDir);\r\n s.message(`Removed ${color.cyan(dockerDir)}`);\r\n }\r\n s.stop('Removed Docker Compose setup');\r\n } catch (error) {\r\n s.stop('Failed to remove Docker Compose setup');\r\n throw error;\r\n }\r\n}\r\n\r\nfunction resolveContainerDependencies(selectedContainers: string[]): string[] {\r\n const resolved = new Set<string>(selectedContainers);\r\n const queue = [...selectedContainers];\r\n\r\n while (queue.length > 0) {\r\n const current = queue.shift()!;\r\n const config = DOCKER_CONTAINERS[current as DockerContainer];\r\n \r\n if (config && config.dependsOn) {\r\n for (const dep of config.dependsOn) {\r\n if (!resolved.has(dep)) {\r\n resolved.add(dep);\r\n queue.push(dep);\r\n }\r\n }\r\n }\r\n }\r\n\r\n return Array.from(resolved);\r\n}\r\n\r\nexport async function configureDockerCompose(projectDir: string, selectedContainers: string[]) {\r\n const s = spinner();\r\n s.start('Configuring Docker Compose...');\r\n\r\n try {\r\n const dockerComposePath = path.join(projectDir, 'docker-compose.yml');\r\n \r\n if (!(await fs.pathExists(dockerComposePath))) {\r\n s.stop('Docker Compose file not found');\r\n return;\r\n }\r\n\r\n const resolvedContainers = resolveContainerDependencies(selectedContainers);\r\n const addedDependencies = resolvedContainers.filter(c => !selectedContainers.includes(c));\r\n \r\n if (addedDependencies.length > 0) {\r\n s.message(`Auto-including dependencies: ${color.cyan(addedDependencies.join(', '))}`);\r\n }\r\n\r\n let content = await fs.readFile(dockerComposePath, 'utf8');\r\n\r\n const allContainers = Object.keys(DOCKER_CONTAINERS) as DockerContainer[];\r\n const containersToRemove = allContainers.filter(c => !resolvedContainers.includes(c));\r\n\r\n for (const container of containersToRemove) {\r\n const containerRegex = new RegExp(\r\n ` # -- ${container} --\\\\n[\\\\s\\\\S]*? # // ${container} //\\\\n`,\r\n 'g'\r\n );\r\n content = content.replace(containerRegex, '');\r\n\r\n // delete `./docker/<container>/`\r\n const dockerDirPath = path.join(projectDir, 'docker', container);\r\n if (await fs.pathExists(dockerDirPath)) {\r\n await fs.remove(dockerDirPath);\r\n s.message(`Removed ${color.cyan(dockerDirPath)}`);\r\n }\r\n }\r\n\r\n content = content.replace(/ # -- \\w+ --\\n/g, '');\r\n content = content.replace(/ # \\/\\/ \\w+ \\/\\/\\n/g, '');\r\n\r\n await fs.writeFile(dockerComposePath, content);\r\n\r\n // check if `./docker/` is empty\r\n const dockerDir = path.join(projectDir, 'docker');\r\n if (await fs.pathExists(dockerDir) && (await fs.readdir(dockerDir)).length === 0) {\r\n await fs.remove(dockerDir);\r\n s.message(`Removed ${color.cyan(dockerDir)} ${color.gray('(because it was empty)')}`);\r\n }\r\n \r\n s.stop(`Configured Docker Compose with ${color.cyan(resolvedContainers.length)} container(s)`);\r\n } catch (error) {\r\n s.stop('Failed to configure Docker Compose');\r\n throw error;\r\n }\r\n}\r\n\r\n"],"mappings":"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","names":[]}
|
|
1
|
+
{"version":3,"sources":["../src/helpers/transform.ts"],"sourcesContent":["import fg from 'fast-glob';\r\nimport fs from 'fs-extra';\r\nimport path from 'path';\r\nimport { spinner } from '@clack/prompts';\r\nimport color from 'picocolors';\r\n\r\nconst DOCKER_CONTAINERS = {\r\n postgres: {\r\n label: 'PostgreSQL 16 (Database)',\r\n services: ['postgres'],\r\n volumes: ['postgres_data'],\r\n dependsOn: [], \r\n },\r\n pgadmin: {\r\n label: `pgAdmin 4.9 (Database management) [Depends on PostgreSQL]`,\r\n services: ['pgadmin'],\r\n volumes: [],\r\n dependsOn: ['postgres'],\r\n },\r\n redis: {\r\n label: 'Redis (Caching)',\r\n services: ['redis'],\r\n volumes: [],\r\n dependsOn: [],\r\n },\r\n mailpit: {\r\n label: 'Mailpit (Email testing)',\r\n services: ['mailpit'],\r\n volumes: [],\r\n dependsOn: [],\r\n },\r\n minio: {\r\n label: 'MinIO (S3-compatible storage)',\r\n services: ['minio', 'minio-create-bucket'],\r\n volumes: ['minio_data'],\r\n dependsOn: [],\r\n },\r\n} as const;\r\n\r\nexport type DockerContainer = keyof typeof DOCKER_CONTAINERS;\r\n\r\nexport async function replaceScope(projectDir: string, newScope: string) {\r\n const s = spinner();\r\n s.start(`Replacing scope with ${color.cyan(newScope)}...`);\r\n\r\n try {\r\n const files = await fg('**/*', {\r\n cwd: projectDir,\r\n ignore: [\r\n '**/.git/**',\r\n '**/node_modules/**',\r\n '**/.turbo/**',\r\n '**/dist/**',\r\n '**/.next/**',\r\n '**/pnpm-lock.yaml',\r\n '**/yarn.lock',\r\n '**/package-lock.json',\r\n '**/bun.lockb',\r\n ],\r\n absolute: true,\r\n });\r\n\r\n await Promise.all(\r\n files.map(async (file) => {\r\n try {\r\n const content = await fs.readFile(file, 'utf8');\r\n if (content.includes('@acme')) {\r\n const newContent = content.replace(/@acme/g, newScope);\r\n await fs.writeFile(file, newContent);\r\n }\r\n } catch (e) {\r\n }\r\n })\r\n );\r\n \r\n s.stop(`Replaced scope with ${color.cyan(newScope)}`);\r\n } catch (error) {\r\n s.stop('Failed to replace scope');\r\n throw error;\r\n }\r\n}\r\n\r\nexport async function setupEnv(projectDir: string) {\r\n const envExample = path.join(projectDir, '.env.example');\r\n const envDest = path.join(projectDir, '.env');\r\n\r\n if (await fs.pathExists(envExample)) {\r\n await fs.copy(envExample, envDest);\r\n }\r\n}\r\n\r\nexport function getDockerContainers() {\r\n return Object.entries(DOCKER_CONTAINERS).map(([value, config]) => ({\r\n value,\r\n label: config.label,\r\n }));\r\n}\r\n\r\nexport async function deleteDockerCompose(projectDir: string) {\r\n const s = spinner();\r\n s.start('Removing Docker Compose setup...');\r\n\r\n try {\r\n const dockerComposePath = path.join(projectDir, 'docker-compose.yml');\r\n if (await fs.pathExists(dockerComposePath)) {\r\n await fs.remove(dockerComposePath);\r\n s.message(`Removed ${color.cyan(dockerComposePath)}`);\r\n }\r\n const dockerDir = path.join(projectDir, 'docker');\r\n if (await fs.pathExists(dockerDir)) {\r\n await fs.remove(dockerDir);\r\n s.message(`Removed ${color.cyan(dockerDir)}`);\r\n }\r\n s.stop('Removed Docker Compose setup');\r\n } catch (error) {\r\n s.stop('Failed to remove Docker Compose setup');\r\n throw error;\r\n }\r\n}\r\n\r\nfunction resolveContainerDependencies(selectedContainers: string[]): string[] {\r\n const resolved = new Set<string>(selectedContainers);\r\n const queue = [...selectedContainers];\r\n\r\n while (queue.length > 0) {\r\n const current = queue.shift()!;\r\n const config = DOCKER_CONTAINERS[current as DockerContainer];\r\n \r\n if (config && config.dependsOn) {\r\n for (const dep of config.dependsOn) {\r\n if (!resolved.has(dep)) {\r\n resolved.add(dep);\r\n queue.push(dep);\r\n }\r\n }\r\n }\r\n }\r\n\r\n return Array.from(resolved);\r\n}\r\n\r\nexport async function configureDockerCompose(projectDir: string, selectedContainers: string[]) {\r\n const s = spinner();\r\n s.start('Configuring Docker Compose...');\r\n\r\n try {\r\n const dockerComposePath = path.join(projectDir, 'docker-compose.yml');\r\n \r\n if (!(await fs.pathExists(dockerComposePath))) {\r\n s.stop('Docker Compose file not found');\r\n return;\r\n }\r\n\r\n const resolvedContainers = resolveContainerDependencies(selectedContainers);\r\n const addedDependencies = resolvedContainers.filter(c => !selectedContainers.includes(c));\r\n \r\n if (addedDependencies.length > 0) {\r\n s.message(`Auto-including dependencies: ${color.cyan(addedDependencies.join(', '))}`);\r\n }\r\n\r\n let content = await fs.readFile(dockerComposePath, 'utf8');\r\n\r\n const allContainers = Object.keys(DOCKER_CONTAINERS) as DockerContainer[];\r\n const containersToRemove = allContainers.filter(c => !resolvedContainers.includes(c));\r\n\r\n for (const container of containersToRemove) {\r\n const containerRegex = new RegExp(\r\n ` # -- ${container} --\\\\n[\\\\s\\\\S]*? # // ${container} //\\\\n`,\r\n 'g'\r\n );\r\n content = content.replace(containerRegex, '');\r\n\r\n // delete `./docker/<container>/`\r\n const dockerDirPath = path.join(projectDir, 'docker', container);\r\n if (await fs.pathExists(dockerDirPath)) {\r\n await fs.remove(dockerDirPath);\r\n s.message(`Removed ${color.cyan(dockerDirPath)}`);\r\n }\r\n }\r\n\r\n content = content.replace(/ # -- \\w+ --\\n/g, '');\r\n content = content.replace(/ # \\/\\/ \\w+ \\/\\/\\n/g, '');\r\n\r\n await fs.writeFile(dockerComposePath, content);\r\n\r\n // check if `./docker/` is empty\r\n const dockerDir = path.join(projectDir, 'docker');\r\n if (await fs.pathExists(dockerDir) && (await fs.readdir(dockerDir)).length === 0) {\r\n await fs.remove(dockerDir);\r\n s.message(`Removed ${color.cyan(dockerDir)} ${color.gray('(because it was empty)')}`);\r\n }\r\n \r\n s.stop(`Configured Docker Compose with ${color.cyan(resolvedContainers.length)} container(s)`);\r\n } catch (error) {\r\n s.stop('Failed to configure Docker Compose');\r\n throw error;\r\n }\r\n}\r\n\r\nexport async function removePackages(\r\n packages: string[],\r\n packagePath: string,\r\n projectDir: string,\r\n s?: ReturnType<typeof spinner>\r\n) {\r\n const packageJsonPath = path.join(projectDir, packagePath, 'package.json');\r\n if (await fs.pathExists(packageJsonPath)) {\r\n const packageJson = await fs.readJson(packageJsonPath);\r\n let hasChanges = false;\r\n\r\n for (const dep of packages) {\r\n if (packageJson.dependencies && packageJson.dependencies[dep]) {\r\n delete packageJson.dependencies[dep];\r\n hasChanges = true;\r\n if (s) {\r\n s.message(`Removed ${color.cyan(dep)} from ${packagePath}/package.json`);\r\n }\r\n }\r\n }\r\n\r\n if (hasChanges) {\r\n await fs.writeJson(packageJsonPath, packageJson, { spaces: 2 });\r\n }\r\n }\r\n}\r\n\r\nexport async function removeReactEmail(projectDir: string) {\r\n const s = spinner();\r\n s.start('Removing react-email...');\r\n\r\n try {\r\n // remove email package\r\n const emailPackagePath = path.join(projectDir, 'packages', 'email');\r\n if (await fs.pathExists(emailPackagePath)) {\r\n await fs.remove(emailPackagePath);\r\n s.message(`Removed ${color.cyan(emailPackagePath)}`);\r\n }\r\n\r\n // remove @react-email/components from packages/api/package.json\r\n await removePackages(['@react-email/components', '@acme/email'], 'packages/api', projectDir, s);\r\n\r\n s.stop('Removed react-email');\r\n } catch (error) {\r\n s.stop('Failed to remove react-email');\r\n throw error;\r\n }\r\n}\r\n\r\n"],"mappings":"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","names":[]}
|
|
@@ -6,6 +6,8 @@ import {
|
|
|
6
6
|
import { execa } from "execa";
|
|
7
7
|
import { spinner } from "@clack/prompts";
|
|
8
8
|
import color from "picocolors";
|
|
9
|
+
import path from "path";
|
|
10
|
+
import fs from "fs-extra";
|
|
9
11
|
async function scaffoldProject({ projectName, packageManager }) {
|
|
10
12
|
const s = spinner();
|
|
11
13
|
s.start(`Scaffolding project in ${color.cyan(projectName)}...`);
|
|
@@ -23,6 +25,7 @@ async function scaffoldProject({ projectName, packageManager }) {
|
|
|
23
25
|
"--no-git",
|
|
24
26
|
projectName
|
|
25
27
|
]);
|
|
28
|
+
await fs.remove(path.join(projectName, "pnpm-lock.yaml"));
|
|
26
29
|
s.stop(`Scaffolded project in ${color.cyan(projectName)}`);
|
|
27
30
|
} catch (error) {
|
|
28
31
|
s.stop(`Failed to scaffold project`);
|
|
@@ -33,4 +36,4 @@ async function scaffoldProject({ projectName, packageManager }) {
|
|
|
33
36
|
export {
|
|
34
37
|
scaffoldProject
|
|
35
38
|
};
|
|
36
|
-
//# sourceMappingURL=chunk-
|
|
39
|
+
//# sourceMappingURL=chunk-MZO3S75B.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/helpers/scaffold.ts"],"sourcesContent":["import { execa } from 'execa';\r\nimport { spinner } from '@clack/prompts';\r\nimport { type PackageManager, getRunner } from '../utils/package-manager.js';\r\nimport color from 'picocolors';\r\nimport path from 'path';\r\nimport fs from 'fs-extra';\r\n\r\nexport interface ScaffoldOptions {\r\n projectName: string;\r\n packageManager: PackageManager;\r\n}\r\n\r\nexport async function scaffoldProject({ projectName, packageManager }: ScaffoldOptions) {\r\n const s = spinner();\r\n s.start(`Scaffolding project in ${color.cyan(projectName)}...`);\r\n\r\n const runner = getRunner(packageManager);\r\n const [command = 'npx', ...args] = runner.split(' ');\r\n \r\n try {\r\n await execa(command, [\r\n ...args,\r\n 'create-turbo@latest',\r\n '-e',\r\n 'https://github.com/Badbird5907/turbo-kit',\r\n '--package-manager',\r\n packageManager,\r\n '--skip-install',\r\n '--no-git',\r\n projectName\r\n ]);\r\n \r\n\r\n await fs.remove(path.join(projectName, 'pnpm-lock.yaml'));\r\n // we are removing the lockfile as we may be deleting some packages\r\n\r\n s.stop(`Scaffolded project in ${color.cyan(projectName)}`);\r\n } catch (error) {\r\n s.stop(`Failed to scaffold project`);\r\n throw error;\r\n }\r\n}\r\n\r\n"],"mappings":";;;;;AAAA,SAAS,aAAa;AACtB,SAAS,eAAe;AAExB,OAAO,WAAW;AAClB,OAAO,UAAU;AACjB,OAAO,QAAQ;AAOf,eAAsB,gBAAgB,EAAE,aAAa,eAAe,GAAoB;AACtF,QAAM,IAAI,QAAQ;AAClB,IAAE,MAAM,0BAA0B,MAAM,KAAK,WAAW,CAAC,KAAK;AAE9D,QAAM,SAAS,UAAU,cAAc;AACvC,QAAM,CAAC,UAAU,OAAO,GAAG,IAAI,IAAI,OAAO,MAAM,GAAG;AAEnD,MAAI;AACF,UAAM,MAAM,SAAS;AAAA,MACnB,GAAG;AAAA,MACH;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,IACF,CAAC;AAGD,UAAM,GAAG,OAAO,KAAK,KAAK,aAAa,gBAAgB,CAAC;AAGxD,MAAE,KAAK,yBAAyB,MAAM,KAAK,WAAW,CAAC,EAAE;AAAA,EAC3D,SAAS,OAAO;AACd,MAAE,KAAK,4BAA4B;AACnC,UAAM;AAAA,EACR;AACF;","names":[]}
|
package/dist/cli.js
CHANGED
|
@@ -4,14 +4,15 @@ import {
|
|
|
4
4
|
} from "./chunk-5F6FI7NU.js";
|
|
5
5
|
import {
|
|
6
6
|
scaffoldProject
|
|
7
|
-
} from "./chunk-
|
|
7
|
+
} from "./chunk-MZO3S75B.js";
|
|
8
8
|
import {
|
|
9
9
|
configureDockerCompose,
|
|
10
10
|
deleteDockerCompose,
|
|
11
11
|
getDockerContainers,
|
|
12
|
+
removeReactEmail,
|
|
12
13
|
replaceScope,
|
|
13
14
|
setupEnv
|
|
14
|
-
} from "./chunk-
|
|
15
|
+
} from "./chunk-KY4U5QQF.js";
|
|
15
16
|
import "./chunk-EYEU3RGM.js";
|
|
16
17
|
|
|
17
18
|
// src/cli.ts
|
|
@@ -132,6 +133,19 @@ Your import statements will look like this: ${importStatement}`,
|
|
|
132
133
|
} else {
|
|
133
134
|
await deleteDockerCompose(projectDir);
|
|
134
135
|
}
|
|
136
|
+
const wantsReactEmail = await confirm({
|
|
137
|
+
message: "Do you want to use react-email?",
|
|
138
|
+
initialValue: true
|
|
139
|
+
});
|
|
140
|
+
if (isCancel(wantsReactEmail)) {
|
|
141
|
+
cancel("Operation cancelled.");
|
|
142
|
+
process.exit(0);
|
|
143
|
+
}
|
|
144
|
+
if (!wantsReactEmail) {
|
|
145
|
+
await removeReactEmail(projectDir);
|
|
146
|
+
}
|
|
147
|
+
await replaceScope(projectDir, scope);
|
|
148
|
+
await setupEnv(projectDir);
|
|
135
149
|
const shouldInstall = await confirm({
|
|
136
150
|
message: "Do you want to install dependencies now?",
|
|
137
151
|
initialValue: true
|
|
@@ -140,8 +154,6 @@ Your import statements will look like this: ${importStatement}`,
|
|
|
140
154
|
cancel("Operation cancelled.");
|
|
141
155
|
process.exit(0);
|
|
142
156
|
}
|
|
143
|
-
await replaceScope(projectDir, scope);
|
|
144
|
-
await setupEnv(projectDir);
|
|
145
157
|
if (shouldInstall) {
|
|
146
158
|
await installDependencies(projectDir, packageManager);
|
|
147
159
|
}
|
package/dist/cli.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/cli.ts"],"sourcesContent":["import { intro, outro, text, select, isCancel, cancel, confirm, multiselect } from '@clack/prompts';\r\nimport { Command } from 'commander';\r\nimport color from 'picocolors';\r\nimport fs from 'fs-extra';\r\nimport path from 'path';\r\nimport { replaceScope, setupEnv, getDockerContainers, configureDockerCompose, deleteDockerCompose } from './helpers/transform.js';\r\nimport { installDependencies, initializeGit } from './helpers/install.js';\r\nimport { type PackageManager } from './utils/package-manager.js';\r\nimport { scaffoldProject } from './helpers/scaffold.js';\r\n\r\nasync function main() {\r\n const program = new Command();\r\n \r\n program\r\n .name('create-turbo-kit')\r\n .description('Initialize a custom turborepo template')\r\n .argument('[project-name]', 'Name of the project directory')\r\n .parse(process.argv);\r\n\r\n const args = program.args;\r\n let projectName = args[0];\r\n\r\n console.log();\r\n intro(color.bgCyan(color.black(' Create Turbo Kit ')));\r\n\r\n if (!projectName) {\r\n const pName = await text({\r\n message: 'What is your project named?',\r\n placeholder: 'my-turbo-app',\r\n validate: (value) => {\r\n if (!value) return 'Please enter a name.';\r\n if (/[^a-zA-Z0-9-_]/.test(value)) return 'Project name can only contain letters, numbers, dashes and underscores.';\r\n const projectDir = path.resolve(process.cwd(), value);\r\n if (fs.pathExistsSync(projectDir)) {\r\n return 'Project name already taken.';\r\n }\r\n return undefined;\r\n },\r\n });\r\n\r\n if (isCancel(pName)) {\r\n cancel('Operation cancelled.');\r\n process.exit(0);\r\n }\r\n projectName = pName as string;\r\n }\r\n\r\n const projectDir = path.resolve(process.cwd(), projectName);\r\n\r\n if (await fs.pathExists(projectDir)) {\r\n const shouldOverwrite = await confirm({\r\n message: `Directory ${projectName} already exists. Do you want to overwrite it?`,\r\n initialValue: false,\r\n });\r\n\r\n if (isCancel(shouldOverwrite) || !shouldOverwrite) {\r\n cancel('Operation cancelled.');\r\n process.exit(0);\r\n }\r\n \r\n await fs.remove(projectDir);\r\n }\r\n\r\n const packageManager = await select({\r\n message: 'Which package manager do you want to use?',\r\n options: [\r\n { value: 'pnpm', label: 'pnpm' },\r\n { value: 'npm', label: 'npm' },\r\n { value: 'yarn', label: 'yarn' },\r\n { value: 'bun', label: 'bun' },\r\n ],\r\n });\r\n\r\n if (isCancel(packageManager)) {\r\n cancel('Operation cancelled.');\r\n process.exit(0);\r\n }\r\n\r\n const getScope = async (): Promise<string> => {\r\n const scope = await text({\r\n message: 'What is your package scope?',\r\n placeholder: '@my-org',\r\n validate: (value) => {\r\n if (!value) return 'Please enter a scope.';\r\n },\r\n });\r\n \r\n if (isCancel(scope)) {\r\n cancel('Operation cancelled.');\r\n process.exit(0);\r\n }\r\n\r\n const check = [\"@\", \"~\", \"$\", \"#\", \"!\"];\r\n if (!check.includes(scope[0] as string)) {\r\n // ask the user to confirm this is the correct scope\r\n const importStatement = `${color.magenta(\"import\")} ${color.yellow(\"{\")} ${color.cyan(\"example\")} ${color.yellow(\"}\")} ${color.magenta(\"from\")} ${color.cyan(`\"${scope}/example\"`)}`;\r\n const confirmScope = await confirm({\r\n message: `Is ${color.cyan(scope)} the correct scope?\\nYour import statements will look like this: ${importStatement}`,\r\n initialValue: true,\r\n });\r\n if (isCancel(confirmScope) || !confirmScope) {\r\n return await getScope();\r\n }\r\n }\r\n\r\n return scope as string;\r\n }\r\n const scope = await getScope();\r\n\r\n try {\r\n await scaffoldProject({\r\n projectName,\r\n packageManager: packageManager as PackageManager,\r\n });\r\n\r\n const wantsDocker = await confirm({\r\n message: 'Do you want to set up a local Docker Compose dev environment?',\r\n initialValue: true,\r\n });\r\n\r\n if (isCancel(wantsDocker)) {\r\n cancel('Operation cancelled.');\r\n process.exit(0);\r\n }\r\n\r\n if (wantsDocker) {\r\n const dockerContainers = getDockerContainers();\r\n const containers = await multiselect({\r\n message: 'Which containers do you want to include?',\r\n options: dockerContainers,\r\n initialValues: dockerContainers.map((container) => container.value),\r\n required: false,\r\n });\r\n\r\n if (isCancel(containers)) {\r\n cancel('Operation cancelled.');\r\n process.exit(0);\r\n }\r\n\r\n if (Array.isArray(containers) && containers.length > 0) {\r\n await configureDockerCompose(projectDir, containers as string[]);\r\n } else {\r\n await deleteDockerCompose(projectDir);\r\n }\r\n } else {\r\n await deleteDockerCompose(projectDir);\r\n }\r\n\r\n const shouldInstall = await confirm({\r\n message: 'Do you want to install dependencies now?',\r\n initialValue: true,\r\n });\r\n\r\n if (isCancel(shouldInstall)) {\r\n cancel('Operation cancelled.');\r\n process.exit(0);\r\n }\r\n\r\n await replaceScope(projectDir, scope as string);\r\n await setupEnv(projectDir);\r\n \r\n if (shouldInstall) {\r\n await installDependencies(projectDir, packageManager as PackageManager);\r\n }\r\n\r\n await initializeGit(projectDir);\r\n\r\n outro(color.green('Project initialized successfully!'));\r\n \r\n console.log(`To get started:`);\r\n console.log(color.cyan(` cd ${projectName}`));\r\n if (!shouldInstall) {\r\n console.log(color.cyan(` ${packageManager} install`));\r\n }\r\n if (wantsDocker) {\r\n console.log(color.cyan(` docker compose up -d`));\r\n }\r\n console.log(color.cyan(` ${packageManager} run dev`));\r\n \r\n console.log();\r\n console.log(color.greenBright(\"Happy Hacking!\"))\r\n } catch (error) {\r\n console.error(color.red('An error occurred:'), error);\r\n process.exit(1);\r\n }\r\n}\r\n\r\nmain().catch(console.error);\r\n"],"mappings":"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","names":["projectDir","scope"]}
|
|
1
|
+
{"version":3,"sources":["../src/cli.ts"],"sourcesContent":["import { intro, outro, text, select, isCancel, cancel, confirm, multiselect } from '@clack/prompts';\r\nimport { Command } from 'commander';\r\nimport color from 'picocolors';\r\nimport fs from 'fs-extra';\r\nimport path from 'path';\r\nimport { replaceScope, setupEnv, getDockerContainers, configureDockerCompose, deleteDockerCompose, removeReactEmail } from './helpers/transform.js';\r\nimport { installDependencies, initializeGit } from './helpers/install.js';\r\nimport { type PackageManager } from './utils/package-manager.js';\r\nimport { scaffoldProject } from './helpers/scaffold.js';\r\n\r\nasync function main() {\r\n const program = new Command();\r\n \r\n program\r\n .name('create-turbo-kit')\r\n .description('Initialize a custom turborepo template')\r\n .argument('[project-name]', 'Name of the project directory')\r\n .parse(process.argv);\r\n\r\n const args = program.args;\r\n let projectName = args[0];\r\n\r\n console.log();\r\n intro(color.bgCyan(color.black(' Create Turbo Kit ')));\r\n\r\n if (!projectName) {\r\n const pName = await text({\r\n message: 'What is your project named?',\r\n placeholder: 'my-turbo-app',\r\n validate: (value) => {\r\n if (!value) return 'Please enter a name.';\r\n if (/[^a-zA-Z0-9-_]/.test(value)) return 'Project name can only contain letters, numbers, dashes and underscores.';\r\n const projectDir = path.resolve(process.cwd(), value);\r\n if (fs.pathExistsSync(projectDir)) {\r\n return 'Project name already taken.';\r\n }\r\n return undefined;\r\n },\r\n });\r\n\r\n if (isCancel(pName)) {\r\n cancel('Operation cancelled.');\r\n process.exit(0);\r\n }\r\n projectName = pName as string;\r\n }\r\n\r\n const projectDir = path.resolve(process.cwd(), projectName);\r\n\r\n if (await fs.pathExists(projectDir)) {\r\n const shouldOverwrite = await confirm({\r\n message: `Directory ${projectName} already exists. Do you want to overwrite it?`,\r\n initialValue: false,\r\n });\r\n\r\n if (isCancel(shouldOverwrite) || !shouldOverwrite) {\r\n cancel('Operation cancelled.');\r\n process.exit(0);\r\n }\r\n \r\n await fs.remove(projectDir);\r\n }\r\n\r\n const packageManager = await select({\r\n message: 'Which package manager do you want to use?',\r\n options: [\r\n { value: 'pnpm', label: 'pnpm' },\r\n { value: 'npm', label: 'npm' },\r\n { value: 'yarn', label: 'yarn' },\r\n { value: 'bun', label: 'bun' },\r\n ],\r\n });\r\n\r\n if (isCancel(packageManager)) {\r\n cancel('Operation cancelled.');\r\n process.exit(0);\r\n }\r\n\r\n const getScope = async (): Promise<string> => {\r\n const scope = await text({\r\n message: 'What is your package scope?',\r\n placeholder: '@my-org',\r\n validate: (value) => {\r\n if (!value) return 'Please enter a scope.';\r\n },\r\n });\r\n \r\n if (isCancel(scope)) {\r\n cancel('Operation cancelled.');\r\n process.exit(0);\r\n }\r\n\r\n const check = [\"@\", \"~\", \"$\", \"#\", \"!\"];\r\n if (!check.includes(scope[0] as string)) {\r\n // ask the user to confirm this is the correct scope\r\n const importStatement = `${color.magenta(\"import\")} ${color.yellow(\"{\")} ${color.cyan(\"example\")} ${color.yellow(\"}\")} ${color.magenta(\"from\")} ${color.cyan(`\"${scope}/example\"`)}`;\r\n const confirmScope = await confirm({\r\n message: `Is ${color.cyan(scope)} the correct scope?\\nYour import statements will look like this: ${importStatement}`,\r\n initialValue: true,\r\n });\r\n if (isCancel(confirmScope) || !confirmScope) {\r\n return await getScope();\r\n }\r\n }\r\n\r\n return scope as string;\r\n }\r\n const scope = await getScope();\r\n\r\n try {\r\n await scaffoldProject({\r\n projectName,\r\n packageManager: packageManager as PackageManager,\r\n });\r\n\r\n const wantsDocker = await confirm({\r\n message: 'Do you want to set up a local Docker Compose dev environment?',\r\n initialValue: true,\r\n });\r\n\r\n if (isCancel(wantsDocker)) {\r\n cancel('Operation cancelled.');\r\n process.exit(0);\r\n }\r\n\r\n if (wantsDocker) {\r\n const dockerContainers = getDockerContainers();\r\n const containers = await multiselect({\r\n message: 'Which containers do you want to include?',\r\n options: dockerContainers,\r\n initialValues: dockerContainers.map((container) => container.value),\r\n required: false,\r\n });\r\n\r\n if (isCancel(containers)) {\r\n cancel('Operation cancelled.');\r\n process.exit(0);\r\n }\r\n\r\n if (Array.isArray(containers) && containers.length > 0) {\r\n await configureDockerCompose(projectDir, containers as string[]);\r\n } else {\r\n await deleteDockerCompose(projectDir);\r\n }\r\n } else {\r\n await deleteDockerCompose(projectDir);\r\n }\r\n const wantsReactEmail = await confirm({\r\n message: 'Do you want to use react-email?',\r\n initialValue: true,\r\n });\r\n\r\n if (isCancel(wantsReactEmail)) {\r\n cancel('Operation cancelled.');\r\n process.exit(0);\r\n }\r\n\r\n if (!wantsReactEmail) {\r\n await removeReactEmail(projectDir);\r\n }\r\n \r\n await replaceScope(projectDir, scope as string);\r\n await setupEnv(projectDir);\r\n \r\n const shouldInstall = await confirm({\r\n message: 'Do you want to install dependencies now?',\r\n initialValue: true,\r\n });\r\n\r\n if (isCancel(shouldInstall)) {\r\n cancel('Operation cancelled.');\r\n process.exit(0);\r\n }\r\n\r\n if (shouldInstall) {\r\n await installDependencies(projectDir, packageManager as PackageManager);\r\n }\r\n\r\n await initializeGit(projectDir);\r\n\r\n outro(color.green('Project initialized successfully!'));\r\n \r\n console.log(`To get started:`);\r\n console.log(color.cyan(` cd ${projectName}`));\r\n if (!shouldInstall) {\r\n console.log(color.cyan(` ${packageManager} install`));\r\n }\r\n if (wantsDocker) {\r\n console.log(color.cyan(` docker compose up -d`));\r\n }\r\n console.log(color.cyan(` ${packageManager} run dev`));\r\n \r\n console.log();\r\n console.log(color.greenBright(\"Happy Hacking!\"))\r\n } catch (error) {\r\n console.error(color.red('An error occurred:'), error);\r\n process.exit(1);\r\n }\r\n}\r\n\r\nmain().catch(console.error);\r\n"],"mappings":"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","names":["projectDir","scope"]}
|
package/dist/helpers/scaffold.js
CHANGED
|
@@ -1,3 +1,5 @@
|
|
|
1
|
+
import { spinner } from '@clack/prompts';
|
|
2
|
+
|
|
1
3
|
declare const DOCKER_CONTAINERS: {
|
|
2
4
|
readonly postgres: {
|
|
3
5
|
readonly label: "PostgreSQL 16 (Database)";
|
|
@@ -39,5 +41,7 @@ declare function getDockerContainers(): {
|
|
|
39
41
|
}[];
|
|
40
42
|
declare function deleteDockerCompose(projectDir: string): Promise<void>;
|
|
41
43
|
declare function configureDockerCompose(projectDir: string, selectedContainers: string[]): Promise<void>;
|
|
44
|
+
declare function removePackages(packages: string[], packagePath: string, projectDir: string, s?: ReturnType<typeof spinner>): Promise<void>;
|
|
45
|
+
declare function removeReactEmail(projectDir: string): Promise<void>;
|
|
42
46
|
|
|
43
|
-
export { type DockerContainer, configureDockerCompose, deleteDockerCompose, getDockerContainers, replaceScope, setupEnv };
|
|
47
|
+
export { type DockerContainer, configureDockerCompose, deleteDockerCompose, getDockerContainers, removePackages, removeReactEmail, replaceScope, setupEnv };
|
|
@@ -2,13 +2,17 @@ import {
|
|
|
2
2
|
configureDockerCompose,
|
|
3
3
|
deleteDockerCompose,
|
|
4
4
|
getDockerContainers,
|
|
5
|
+
removePackages,
|
|
6
|
+
removeReactEmail,
|
|
5
7
|
replaceScope,
|
|
6
8
|
setupEnv
|
|
7
|
-
} from "../chunk-
|
|
9
|
+
} from "../chunk-KY4U5QQF.js";
|
|
8
10
|
export {
|
|
9
11
|
configureDockerCompose,
|
|
10
12
|
deleteDockerCompose,
|
|
11
13
|
getDockerContainers,
|
|
14
|
+
removePackages,
|
|
15
|
+
removeReactEmail,
|
|
12
16
|
replaceScope,
|
|
13
17
|
setupEnv
|
|
14
18
|
};
|
package/package.json
CHANGED
package/src/cli.ts
CHANGED
|
@@ -3,7 +3,7 @@ import { Command } from 'commander';
|
|
|
3
3
|
import color from 'picocolors';
|
|
4
4
|
import fs from 'fs-extra';
|
|
5
5
|
import path from 'path';
|
|
6
|
-
import { replaceScope, setupEnv, getDockerContainers, configureDockerCompose, deleteDockerCompose } from './helpers/transform.js';
|
|
6
|
+
import { replaceScope, setupEnv, getDockerContainers, configureDockerCompose, deleteDockerCompose, removeReactEmail } from './helpers/transform.js';
|
|
7
7
|
import { installDependencies, initializeGit } from './helpers/install.js';
|
|
8
8
|
import { type PackageManager } from './utils/package-manager.js';
|
|
9
9
|
import { scaffoldProject } from './helpers/scaffold.js';
|
|
@@ -145,7 +145,23 @@ async function main() {
|
|
|
145
145
|
} else {
|
|
146
146
|
await deleteDockerCompose(projectDir);
|
|
147
147
|
}
|
|
148
|
+
const wantsReactEmail = await confirm({
|
|
149
|
+
message: 'Do you want to use react-email?',
|
|
150
|
+
initialValue: true,
|
|
151
|
+
});
|
|
148
152
|
|
|
153
|
+
if (isCancel(wantsReactEmail)) {
|
|
154
|
+
cancel('Operation cancelled.');
|
|
155
|
+
process.exit(0);
|
|
156
|
+
}
|
|
157
|
+
|
|
158
|
+
if (!wantsReactEmail) {
|
|
159
|
+
await removeReactEmail(projectDir);
|
|
160
|
+
}
|
|
161
|
+
|
|
162
|
+
await replaceScope(projectDir, scope as string);
|
|
163
|
+
await setupEnv(projectDir);
|
|
164
|
+
|
|
149
165
|
const shouldInstall = await confirm({
|
|
150
166
|
message: 'Do you want to install dependencies now?',
|
|
151
167
|
initialValue: true,
|
|
@@ -156,9 +172,6 @@ async function main() {
|
|
|
156
172
|
process.exit(0);
|
|
157
173
|
}
|
|
158
174
|
|
|
159
|
-
await replaceScope(projectDir, scope as string);
|
|
160
|
-
await setupEnv(projectDir);
|
|
161
|
-
|
|
162
175
|
if (shouldInstall) {
|
|
163
176
|
await installDependencies(projectDir, packageManager as PackageManager);
|
|
164
177
|
}
|
package/src/helpers/scaffold.ts
CHANGED
|
@@ -2,6 +2,8 @@ import { execa } from 'execa';
|
|
|
2
2
|
import { spinner } from '@clack/prompts';
|
|
3
3
|
import { type PackageManager, getRunner } from '../utils/package-manager.js';
|
|
4
4
|
import color from 'picocolors';
|
|
5
|
+
import path from 'path';
|
|
6
|
+
import fs from 'fs-extra';
|
|
5
7
|
|
|
6
8
|
export interface ScaffoldOptions {
|
|
7
9
|
projectName: string;
|
|
@@ -27,7 +29,11 @@ export async function scaffoldProject({ projectName, packageManager }: ScaffoldO
|
|
|
27
29
|
'--no-git',
|
|
28
30
|
projectName
|
|
29
31
|
]);
|
|
30
|
-
|
|
32
|
+
|
|
33
|
+
|
|
34
|
+
await fs.remove(path.join(projectName, 'pnpm-lock.yaml'));
|
|
35
|
+
// we are removing the lockfile as we may be deleting some packages
|
|
36
|
+
|
|
31
37
|
s.stop(`Scaffolded project in ${color.cyan(projectName)}`);
|
|
32
38
|
} catch (error) {
|
|
33
39
|
s.stop(`Failed to scaffold project`);
|
package/src/helpers/transform.ts
CHANGED
|
@@ -197,3 +197,52 @@ export async function configureDockerCompose(projectDir: string, selectedContain
|
|
|
197
197
|
}
|
|
198
198
|
}
|
|
199
199
|
|
|
200
|
+
export async function removePackages(
|
|
201
|
+
packages: string[],
|
|
202
|
+
packagePath: string,
|
|
203
|
+
projectDir: string,
|
|
204
|
+
s?: ReturnType<typeof spinner>
|
|
205
|
+
) {
|
|
206
|
+
const packageJsonPath = path.join(projectDir, packagePath, 'package.json');
|
|
207
|
+
if (await fs.pathExists(packageJsonPath)) {
|
|
208
|
+
const packageJson = await fs.readJson(packageJsonPath);
|
|
209
|
+
let hasChanges = false;
|
|
210
|
+
|
|
211
|
+
for (const dep of packages) {
|
|
212
|
+
if (packageJson.dependencies && packageJson.dependencies[dep]) {
|
|
213
|
+
delete packageJson.dependencies[dep];
|
|
214
|
+
hasChanges = true;
|
|
215
|
+
if (s) {
|
|
216
|
+
s.message(`Removed ${color.cyan(dep)} from ${packagePath}/package.json`);
|
|
217
|
+
}
|
|
218
|
+
}
|
|
219
|
+
}
|
|
220
|
+
|
|
221
|
+
if (hasChanges) {
|
|
222
|
+
await fs.writeJson(packageJsonPath, packageJson, { spaces: 2 });
|
|
223
|
+
}
|
|
224
|
+
}
|
|
225
|
+
}
|
|
226
|
+
|
|
227
|
+
export async function removeReactEmail(projectDir: string) {
|
|
228
|
+
const s = spinner();
|
|
229
|
+
s.start('Removing react-email...');
|
|
230
|
+
|
|
231
|
+
try {
|
|
232
|
+
// remove email package
|
|
233
|
+
const emailPackagePath = path.join(projectDir, 'packages', 'email');
|
|
234
|
+
if (await fs.pathExists(emailPackagePath)) {
|
|
235
|
+
await fs.remove(emailPackagePath);
|
|
236
|
+
s.message(`Removed ${color.cyan(emailPackagePath)}`);
|
|
237
|
+
}
|
|
238
|
+
|
|
239
|
+
// remove @react-email/components from packages/api/package.json
|
|
240
|
+
await removePackages(['@react-email/components', '@acme/email'], 'packages/api', projectDir, s);
|
|
241
|
+
|
|
242
|
+
s.stop('Removed react-email');
|
|
243
|
+
} catch (error) {
|
|
244
|
+
s.stop('Failed to remove react-email');
|
|
245
|
+
throw error;
|
|
246
|
+
}
|
|
247
|
+
}
|
|
248
|
+
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/helpers/scaffold.ts"],"sourcesContent":["import { execa } from 'execa';\r\nimport { spinner } from '@clack/prompts';\r\nimport { type PackageManager, getRunner } from '../utils/package-manager.js';\r\nimport color from 'picocolors';\r\n\r\nexport interface ScaffoldOptions {\r\n projectName: string;\r\n packageManager: PackageManager;\r\n}\r\n\r\nexport async function scaffoldProject({ projectName, packageManager }: ScaffoldOptions) {\r\n const s = spinner();\r\n s.start(`Scaffolding project in ${color.cyan(projectName)}...`);\r\n\r\n const runner = getRunner(packageManager);\r\n const [command = 'npx', ...args] = runner.split(' ');\r\n \r\n try {\r\n await execa(command, [\r\n ...args,\r\n 'create-turbo@latest',\r\n '-e',\r\n 'https://github.com/Badbird5907/turbo-kit',\r\n '--package-manager',\r\n packageManager,\r\n '--skip-install',\r\n '--no-git',\r\n projectName\r\n ]);\r\n \r\n s.stop(`Scaffolded project in ${color.cyan(projectName)}`);\r\n } catch (error) {\r\n s.stop(`Failed to scaffold project`);\r\n throw error;\r\n }\r\n}\r\n\r\n"],"mappings":";;;;;AAAA,SAAS,aAAa;AACtB,SAAS,eAAe;AAExB,OAAO,WAAW;AAOlB,eAAsB,gBAAgB,EAAE,aAAa,eAAe,GAAoB;AACtF,QAAM,IAAI,QAAQ;AAClB,IAAE,MAAM,0BAA0B,MAAM,KAAK,WAAW,CAAC,KAAK;AAE9D,QAAM,SAAS,UAAU,cAAc;AACvC,QAAM,CAAC,UAAU,OAAO,GAAG,IAAI,IAAI,OAAO,MAAM,GAAG;AAEnD,MAAI;AACF,UAAM,MAAM,SAAS;AAAA,MACnB,GAAG;AAAA,MACH;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,IACF,CAAC;AAED,MAAE,KAAK,yBAAyB,MAAM,KAAK,WAAW,CAAC,EAAE;AAAA,EAC3D,SAAS,OAAO;AACd,MAAE,KAAK,4BAA4B;AACnC,UAAM;AAAA,EACR;AACF;","names":[]}
|