mirror of
https://github.com/shishantbiswas/bknd.git
synced 2026-03-17 21:06:04 +00:00
cli: create: allow non-interactive create (#137)
This commit is contained in:
@@ -8,7 +8,7 @@ import { Option } from "commander";
|
|||||||
import { env } from "core";
|
import { env } from "core";
|
||||||
import color from "picocolors";
|
import color from "picocolors";
|
||||||
import { overridePackageJson, updateBkndPackages } from "./npm";
|
import { overridePackageJson, updateBkndPackages } from "./npm";
|
||||||
import { type Template, templates } from "./templates";
|
import { type Template, templates, type TemplateSetupCtx } from "./templates";
|
||||||
import { createScoped, flush } from "cli/utils/telemetry";
|
import { createScoped, flush } from "cli/utils/telemetry";
|
||||||
|
|
||||||
const config = {
|
const config = {
|
||||||
@@ -35,6 +35,8 @@ export const create: CliCommand = (program) => {
|
|||||||
.addOption(new Option("-i, --integration <integration>", "integration to use"))
|
.addOption(new Option("-i, --integration <integration>", "integration to use"))
|
||||||
.addOption(new Option("-t, --template <template>", "template to use"))
|
.addOption(new Option("-t, --template <template>", "template to use"))
|
||||||
.addOption(new Option("-d --dir <directory>", "directory to create in"))
|
.addOption(new Option("-d --dir <directory>", "directory to create in"))
|
||||||
|
.addOption(new Option("-c, --clean", "cleans destination directory"))
|
||||||
|
.addOption(new Option("-y, --yes", "use defaults, skip skippable prompts"))
|
||||||
.description("create a new project")
|
.description("create a new project")
|
||||||
.action(action);
|
.action(action);
|
||||||
};
|
};
|
||||||
@@ -53,7 +55,7 @@ async function onExit() {
|
|||||||
await flush();
|
await flush();
|
||||||
}
|
}
|
||||||
|
|
||||||
async function action(options: { template?: string; dir?: string; integration?: string }) {
|
async function action(options: { template?: string; dir?: string; integration?: string, yes?: boolean, clean?: boolean }) {
|
||||||
console.log("");
|
console.log("");
|
||||||
const $t = createScoped("create");
|
const $t = createScoped("create");
|
||||||
$t.capture("start", {
|
$t.capture("start", {
|
||||||
@@ -94,7 +96,7 @@ async function action(options: { template?: string; dir?: string; integration?:
|
|||||||
|
|
||||||
$t.properties.at = "dir";
|
$t.properties.at = "dir";
|
||||||
if (fs.existsSync(downloadOpts.dir)) {
|
if (fs.existsSync(downloadOpts.dir)) {
|
||||||
const clean = await $p.confirm({
|
const clean = options.clean ?? await $p.confirm({
|
||||||
message: `Directory ${color.cyan(downloadOpts.dir)} exists. Clean it?`,
|
message: `Directory ${color.cyan(downloadOpts.dir)} exists. Clean it?`,
|
||||||
initialValue: false,
|
initialValue: false,
|
||||||
});
|
});
|
||||||
@@ -203,7 +205,7 @@ async function action(options: { template?: string; dir?: string; integration?:
|
|||||||
}
|
}
|
||||||
|
|
||||||
$t.properties.template = template.key;
|
$t.properties.template = template.key;
|
||||||
const ctx = { template, dir: downloadOpts.dir, name };
|
const ctx: TemplateSetupCtx = { template, dir: downloadOpts.dir, name, skip: !!options.yes };
|
||||||
|
|
||||||
{
|
{
|
||||||
const ref = env("cli_create_ref", `#v${version}`, {
|
const ref = env("cli_create_ref", `#v${version}`, {
|
||||||
@@ -259,7 +261,7 @@ async function action(options: { template?: string; dir?: string; integration?:
|
|||||||
$p.log.success(`Updated package name to ${color.cyan(ctx.name)}`);
|
$p.log.success(`Updated package name to ${color.cyan(ctx.name)}`);
|
||||||
|
|
||||||
{
|
{
|
||||||
const install = await $p.confirm({
|
const install = options.yes ?? await $p.confirm({
|
||||||
message: "Install dependencies?",
|
message: "Install dependencies?",
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|||||||
@@ -29,7 +29,7 @@ export const cloudflare = {
|
|||||||
{ dir: ctx.dir },
|
{ dir: ctx.dir },
|
||||||
);
|
);
|
||||||
|
|
||||||
const db = await $p.select({
|
const db = ctx.skip ? "d1" : await $p.select({
|
||||||
message: "What database do you want to use?",
|
message: "What database do you want to use?",
|
||||||
options: [
|
options: [
|
||||||
{ label: "Cloudflare D1", value: "d1" },
|
{ label: "Cloudflare D1", value: "d1" },
|
||||||
@@ -63,10 +63,11 @@ export const cloudflare = {
|
|||||||
} as const satisfies Template;
|
} as const satisfies Template;
|
||||||
|
|
||||||
async function createD1(ctx: TemplateSetupCtx) {
|
async function createD1(ctx: TemplateSetupCtx) {
|
||||||
const name = await $p.text({
|
const default_db = "data";
|
||||||
|
const name = ctx.skip ? default_db : await $p.text({
|
||||||
message: "Enter database name",
|
message: "Enter database name",
|
||||||
initialValue: "data",
|
initialValue: default_db,
|
||||||
placeholder: "data",
|
placeholder: default_db,
|
||||||
validate: (v) => {
|
validate: (v) => {
|
||||||
if (!v) {
|
if (!v) {
|
||||||
return "Invalid name";
|
return "Invalid name";
|
||||||
@@ -84,12 +85,15 @@ async function createD1(ctx: TemplateSetupCtx) {
|
|||||||
})(),
|
})(),
|
||||||
);
|
);
|
||||||
|
|
||||||
exec(`npx wrangler d1 create ${name}`);
|
if (!ctx.skip) {
|
||||||
await $p.stream.info(
|
exec(`npx wrangler d1 create ${name}`);
|
||||||
(async function* () {
|
|
||||||
yield* typewriter("Please update your wrangler configuration with the output above.");
|
await $p.stream.info(
|
||||||
})(),
|
(async function* () {
|
||||||
);
|
yield* typewriter("Please update your wrangler configuration with the output above.");
|
||||||
|
})(),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
await overrideJson(
|
await overrideJson(
|
||||||
WRANGLER_FILE,
|
WRANGLER_FILE,
|
||||||
@@ -149,7 +153,7 @@ async function createLibsql(ctx: TemplateSetupCtx) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async function createR2(ctx: TemplateSetupCtx) {
|
async function createR2(ctx: TemplateSetupCtx) {
|
||||||
const create = await $p.confirm({
|
const create = ctx.skip ?? await $p.confirm({
|
||||||
message: "Do you want to use a R2 bucket?",
|
message: "Do you want to use a R2 bucket?",
|
||||||
initialValue: true,
|
initialValue: true,
|
||||||
});
|
});
|
||||||
@@ -168,10 +172,11 @@ async function createR2(ctx: TemplateSetupCtx) {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
const name = await $p.text({
|
const default_bucket = "bucket";
|
||||||
|
const name = ctx.skip ? default_bucket : await $p.text({
|
||||||
message: "Enter bucket name",
|
message: "Enter bucket name",
|
||||||
initialValue: "bucket",
|
initialValue: default_bucket,
|
||||||
placeholder: "bucket",
|
placeholder: default_bucket,
|
||||||
validate: (v) => {
|
validate: (v) => {
|
||||||
if (!v) {
|
if (!v) {
|
||||||
return "Invalid name";
|
return "Invalid name";
|
||||||
@@ -183,7 +188,9 @@ async function createR2(ctx: TemplateSetupCtx) {
|
|||||||
process.exit(1);
|
process.exit(1);
|
||||||
}
|
}
|
||||||
|
|
||||||
exec(`npx wrangler r2 bucket create ${name}`);
|
if (!ctx.skip) {
|
||||||
|
exec(`npx wrangler r2 bucket create ${name}`);
|
||||||
|
}
|
||||||
|
|
||||||
await overrideJson(
|
await overrideJson(
|
||||||
WRANGLER_FILE,
|
WRANGLER_FILE,
|
||||||
|
|||||||
@@ -4,6 +4,7 @@ export type TemplateSetupCtx = {
|
|||||||
template: Template;
|
template: Template;
|
||||||
dir: string;
|
dir: string;
|
||||||
name: string;
|
name: string;
|
||||||
|
skip: boolean;
|
||||||
};
|
};
|
||||||
|
|
||||||
export type Integration =
|
export type Integration =
|
||||||
|
|||||||
Reference in New Issue
Block a user