create-pylon 1.0.3 → 1.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/index.js CHANGED
@@ -3,17 +3,123 @@
3
3
 
4
4
  // src/index.ts
5
5
  import { Option, program } from "commander";
6
- import consola from "consola";
6
+ import consola2 from "consola";
7
7
  import { input, select, confirm } from "@inquirer/prompts";
8
- import path from "path";
8
+ import path2 from "path";
9
9
  import chalk from "chalk";
10
- import * as fs from "fs";
10
+ import * as fs2 from "fs";
11
11
  import * as telemetry from "@getcronit/pylon-telemetry";
12
12
  import { fileURLToPath } from "url";
13
13
  import { dirname } from "path";
14
+ import { spawnSync } from "child_process";
15
+
16
+ // src/detect-pm.ts
17
+ import * as fs from "fs";
18
+ import * as path from "path";
19
+ import process2 from "process";
20
+ import { execSync } from "child_process";
21
+ import consola from "consola";
22
+ function isCommandAvailable(command) {
23
+ try {
24
+ execSync(`${command} --version`, { stdio: "ignore" });
25
+ return true;
26
+ } catch (e) {
27
+ console.error(e);
28
+ return false;
29
+ }
30
+ }
31
+ function isBun() {
32
+ return typeof Bun !== "undefined" && isCommandAvailable("bun");
33
+ }
34
+ function isNpm() {
35
+ return process2.env.npm_execpath?.includes("npm") ?? false;
36
+ }
37
+ function isYarn() {
38
+ return process2.env.npm_execpath?.includes("yarn") ?? false;
39
+ }
40
+ function isDeno() {
41
+ return typeof Deno !== "undefined" && isCommandAvailable("deno");
42
+ }
43
+ function isPnpm() {
44
+ return process2.env.npm_execpath?.includes("pnpm") ?? false;
45
+ }
46
+ function detectByLockFiles(cwd) {
47
+ if (fs.existsSync(path.join(cwd, "bun.lockb"))) {
48
+ return "bun";
49
+ }
50
+ if (fs.existsSync(path.join(cwd, "package-lock.json"))) {
51
+ return "npm";
52
+ }
53
+ if (fs.existsSync(path.join(cwd, "yarn.lock"))) {
54
+ return "yarn";
55
+ }
56
+ if (fs.existsSync(path.join(cwd, "deno.json")) || fs.existsSync(path.join(cwd, "deno.lock"))) {
57
+ return "deno";
58
+ }
59
+ if (fs.existsSync(path.join(cwd, "pnpm-lock.yaml"))) {
60
+ return "pnpm";
61
+ }
62
+ return null;
63
+ }
64
+ function detectPackageManager({
65
+ preferredPm,
66
+ cwd = process2.cwd()
67
+ }) {
68
+ if (preferredPm && isCommandAvailable(preferredPm)) {
69
+ return preferredPm;
70
+ }
71
+ if (isBun()) {
72
+ return "bun";
73
+ }
74
+ if (isNpm()) {
75
+ return "npm";
76
+ }
77
+ if (isPnpm()) {
78
+ return "pnpm";
79
+ }
80
+ if (isDeno()) {
81
+ return "deno";
82
+ }
83
+ if (isYarn()) {
84
+ return "yarn";
85
+ }
86
+ const lockFileDetection = detectByLockFiles(cwd);
87
+ if (lockFileDetection) {
88
+ consola.info(`Detected package manager by lock file: ${lockFileDetection}`);
89
+ if (isCommandAvailable(lockFileDetection)) {
90
+ return lockFileDetection;
91
+ } else {
92
+ consola.warn(`Lock file detected, but ${lockFileDetection} is not installed.`);
93
+ }
94
+ }
95
+ return "unknown";
96
+ }
97
+ function getRunScript(pm) {
98
+ switch (pm) {
99
+ case "bun":
100
+ return "bun";
101
+ case "npm":
102
+ return "npm run";
103
+ case "yarn":
104
+ return "yarn";
105
+ case "pnpm":
106
+ return "pnpm run";
107
+ case "deno":
108
+ return "deno task";
109
+ default:
110
+ throw new Error("Unknown package manager");
111
+ }
112
+ }
113
+
114
+ // src/index.ts
115
+ var __filename2 = fileURLToPath(import.meta.url);
116
+ var __dirname2 = dirname(__filename2);
117
+ var version = (() => {
118
+ return JSON.parse(fs2.readFileSync(path2.join(__dirname2, "..", "package.json"), "utf-8")).version;
119
+ })();
14
120
  function mkdirp(dir) {
15
121
  try {
16
- fs.mkdirSync(dir, { recursive: true });
122
+ fs2.mkdirSync(dir, { recursive: true });
17
123
  } catch (e) {
18
124
  if (e instanceof Error) {
19
125
  if ("code" in e && e.code === "EEXIST")
@@ -22,11 +128,171 @@ function mkdirp(dir) {
22
128
  throw e;
23
129
  }
24
130
  }
25
- import { detect } from "detect-package-manager";
26
- import { spawnSync } from "child_process";
131
+ var runtimes = [
132
+ {
133
+ key: "bun",
134
+ name: "Bun.js",
135
+ website: "https://bunjs.dev",
136
+ templates: ["default"]
137
+ },
138
+ {
139
+ key: "node",
140
+ name: "Node.js",
141
+ website: "https://nodejs.org",
142
+ templates: ["default"]
143
+ },
144
+ {
145
+ key: "cf-workers",
146
+ name: "Cloudflare Workers",
147
+ website: "https://workers.cloudflare.com",
148
+ templates: ["default"]
149
+ },
150
+ {
151
+ key: "deno",
152
+ name: "Deno",
153
+ website: "https://deno.land",
154
+ templates: ["default"]
155
+ }
156
+ ];
157
+ var templates = [
158
+ {
159
+ key: "default",
160
+ name: "Default",
161
+ description: "Default template"
162
+ },
163
+ {
164
+ key: "database",
165
+ name: "Database (Prisma)",
166
+ description: "Template with Prisma ORM"
167
+ }
168
+ ];
169
+ var injectVariablesInContent = (content, variables) => {
170
+ let result = content;
171
+ Object.entries(variables).forEach(([key, value]) => {
172
+ result = result.replaceAll(key, value);
173
+ });
174
+ return result;
175
+ };
176
+ var readdirFilesSyncRecursive = (dir) => {
177
+ const run = (dir2) => {
178
+ const result = [];
179
+ const files = fs2.readdirSync(dir2);
180
+ files.forEach((file) => {
181
+ const filePath = path2.join(dir2, file);
182
+ if (fs2.statSync(filePath).isDirectory()) {
183
+ result.push(...run(filePath));
184
+ }
185
+ if (fs2.statSync(filePath).isFile()) {
186
+ result.push(filePath);
187
+ }
188
+ });
189
+ return result;
190
+ };
191
+ return run(dir).map((file) => {
192
+ return file.replace(dir, ".");
193
+ });
194
+ };
195
+ var createTemplate = async (options) => {
196
+ const { runtime, template, target } = options;
197
+ const runtimeName = runtimes.find(({ key }) => key === runtime)?.name;
198
+ const templateName = templates.find(({ key }) => key === template)?.name;
199
+ if (!runtimeName) {
200
+ throw new Error(`Invalid runtime: ${runtime}`);
201
+ }
202
+ if (!templateName) {
203
+ throw new Error(`Invalid template: ${template}`);
204
+ }
205
+ const sharedTemplateDir = path2.join(__dirname2, "..", "templates", "shared");
206
+ if (!fs2.existsSync(sharedTemplateDir)) {
207
+ throw new Error(`Shared templates not found: ${sharedTemplateDir}`);
208
+ }
209
+ const templateDir = path2.join(__dirname2, "..", "templates", runtime, template);
210
+ if (!fs2.existsSync(templateDir)) {
211
+ throw new Error(`Template not found: ${templateDir}`);
212
+ }
213
+ const targetDirectoryPath = path2.join(process.cwd(), target);
214
+ consola2.start(`Creating pylon in ${targetDirectoryPath}`);
215
+ const inject = (content) => {
216
+ return injectVariablesInContent(content, {
217
+ __PYLON_NAME__: options.name
218
+ });
219
+ };
220
+ readdirFilesSyncRecursive(sharedTemplateDir).forEach((file) => {
221
+ const source = path2.join(sharedTemplateDir, file);
222
+ let target2 = path2.join(targetDirectoryPath, file);
223
+ const targetDir = path2.dirname(target2);
224
+ if (runtime === "cf-workers" && source.includes(".github/workflows/publish.yaml")) {
225
+ return;
226
+ }
227
+ if (!fs2.existsSync(targetDir)) {
228
+ fs2.mkdirSync(targetDir, { recursive: true });
229
+ }
230
+ if (target2.endsWith(".example")) {
231
+ target2 = target2.replace(".example", "");
232
+ }
233
+ const injectedContent = inject(fs2.readFileSync(source, "utf-8"));
234
+ fs2.writeFileSync(target2, injectedContent);
235
+ });
236
+ readdirFilesSyncRecursive(templateDir).forEach((file) => {
237
+ const source = path2.join(templateDir, file);
238
+ let target2 = path2.join(targetDirectoryPath, file);
239
+ const targetDir = path2.dirname(target2);
240
+ if (!fs2.existsSync(targetDir)) {
241
+ fs2.mkdirSync(targetDir, { recursive: true });
242
+ }
243
+ if (target2.endsWith(".example")) {
244
+ target2 = target2.replace(".example", "");
245
+ }
246
+ const injectedContent = inject(fs2.readFileSync(source, "utf-8"));
247
+ fs2.writeFileSync(target2, injectedContent);
248
+ });
249
+ consola2.success(`Pylon created`);
250
+ };
251
+ var installDependencies = async (args) => {
252
+ const target = path2.resolve(args.target);
253
+ const packageManager = args.packageManager;
254
+ let command = "";
255
+ switch (packageManager) {
256
+ case "yarn":
257
+ command = "yarn";
258
+ break;
259
+ case "npm":
260
+ command = "npm install";
261
+ break;
262
+ case "pnpm":
263
+ command = "pnpm install";
264
+ break;
265
+ case "bun":
266
+ command = "bun install";
267
+ break;
268
+ case "deno":
269
+ command = "deno install";
270
+ break;
271
+ default:
272
+ throw new Error(`Invalid package manager: ${packageManager}`);
273
+ }
274
+ consola2.start(`Installing dependencies using ${packageManager}`);
275
+ const proc = spawnSync(command, {
276
+ cwd: target,
277
+ shell: true,
278
+ stdio: "inherit"
279
+ });
280
+ if (proc.status !== 0) {
281
+ throw new Error(`Failed to install dependencies`);
282
+ }
283
+ consola2.success(`Dependencies installed`);
284
+ };
285
+ program.name("create-pylon").version(version).arguments("[target]").addOption(new Option("-i, --install", "Install dependencies")).addOption(new Option("-r, --runtime <runtime>", "Runtime").choices(runtimes.map(({ key }) => key))).addOption(new Option("-t, --template <template>", "Template")).addOption(new Option("-pm, --package-manager <packageManager>", "Package manager")).addOption(new Option("--client", "Enable client generation (https://pylon.cronit.io/docs/integrations/gqty)")).addOption(new Option("--client-path <clientPath>", "Client path")).addOption(new Option("--client-port <clientPort>", "Client port")).action(main);
286
+ var getPreferredPmByRuntime = (runtime) => {
287
+ if (runtime === "bun") {
288
+ return "bun";
289
+ } else if (runtime === "deno") {
290
+ return "deno";
291
+ }
292
+ };
27
293
  async function main(targetDir, options, command) {
28
294
  try {
29
- consola.log(`${command.name()} version ${command.version()}`);
295
+ consola2.log(`${command.name()} version ${command.version()}`);
30
296
  const {
31
297
  install: installArg,
32
298
  runtime: runtimeArg,
@@ -39,7 +305,7 @@ async function main(targetDir, options, command) {
39
305
  let target = "";
40
306
  if (targetDir) {
41
307
  target = targetDir;
42
- consola.success(`Using target directory \u2026 ${target}`);
308
+ consola2.success(`Using target directory \u2026 ${target}`);
43
309
  } else {
44
310
  const answer = await input({
45
311
  message: "Target directory",
@@ -49,9 +315,9 @@ async function main(targetDir, options, command) {
49
315
  }
50
316
  let projectName = "";
51
317
  if (target === ".") {
52
- projectName = path.basename(process.cwd());
318
+ projectName = path2.basename(process.cwd());
53
319
  } else {
54
- projectName = path.basename(target);
320
+ projectName = path2.basename(target);
55
321
  }
56
322
  const runtimeName = runtimeArg || await select({
57
323
  message: "Which runtime would you like to use?",
@@ -79,8 +345,8 @@ async function main(targetDir, options, command) {
79
345
  if (!templateName) {
80
346
  throw new Error("No template selected");
81
347
  }
82
- if (fs.existsSync(target)) {
83
- if (fs.readdirSync(target).length > 0) {
348
+ if (fs2.existsSync(target)) {
349
+ if (fs2.readdirSync(target).length > 0) {
84
350
  const response = await confirm({
85
351
  message: "Directory not empty. Continue?",
86
352
  default: false
@@ -99,10 +365,10 @@ async function main(targetDir, options, command) {
99
365
  template: templateName,
100
366
  target
101
367
  });
102
- let packageManager = packageManagerArg;
103
- if (runtimeName === "bun" && !packageManager) {
104
- packageManager = "bun";
105
- }
368
+ const packageManager = detectPackageManager({
369
+ preferredPm: getPreferredPmByRuntime(runtime.key),
370
+ cwd: target
371
+ });
106
372
  if (install) {
107
373
  await installDependencies({ target, packageManager });
108
374
  }
@@ -121,7 +387,7 @@ async function main(targetDir, options, command) {
121
387
  });
122
388
  clientPath = await input({
123
389
  message: "Path to generate the client to",
124
- default: path.join(clientRoot, "gqty/index.ts"),
390
+ default: path2.join(clientRoot, "gqty/index.ts"),
125
391
  validate: (value) => {
126
392
  if (!value.startsWith(clientRoot === "." ? "" : clientRoot)) {
127
393
  return "Path must start with the client root";
@@ -134,23 +400,32 @@ async function main(targetDir, options, command) {
134
400
  message: "Port of the pylon server to generate the client from",
135
401
  default: "3000"
136
402
  });
137
- consola.start(`Updating pylon dev script to generate client`);
138
- const devScriptPath = path.join(target, "package.json");
139
- const devScript = JSON.parse(fs.readFileSync(devScriptPath, "utf-8"));
140
- devScript.scripts = {
141
- ...devScript.scripts,
142
- dev: devScript.scripts.dev + ` --client --client-port ${clientPort} --client-path ${clientPath}`
403
+ consola2.start(`Updating pylon dev script to generate client`);
404
+ let packagePath;
405
+ let scriptKey;
406
+ if (runtime.key === "deno") {
407
+ packagePath = path2.join(target, "deno.json");
408
+ scriptKey = "tasks";
409
+ } else {
410
+ packagePath = path2.join(target, "package.json");
411
+ scriptKey = "scripts";
412
+ }
413
+ const devScript = JSON.parse(fs2.readFileSync(packagePath, "utf-8"));
414
+ devScript[scriptKey] = {
415
+ ...devScript[scriptKey],
416
+ dev: devScript[scriptKey].dev + ` --client --client-port ${clientPort} --client-path ${clientPath}`
143
417
  };
144
- fs.writeFileSync(devScriptPath, JSON.stringify(devScript, null, 2));
145
- consola.success(`Pylon dev script updated`);
418
+ fs2.writeFileSync(packagePath, JSON.stringify(devScript, null, 2));
419
+ consola2.success(`Pylon dev script updated`);
146
420
  }
421
+ const runScript = getRunScript(packageManager);
147
422
  const message = `
148
423
  \uD83C\uDF89 ${chalk.green.bold("Pylon created successfully.")}
149
424
 
150
425
  \uD83D\uDCBB ${chalk.cyan.bold("Continue Developing")}
151
426
  ${chalk.yellow("Change directories:")} cd ${chalk.blue(target)}
152
- ${chalk.yellow("Start dev server:")} npm run start
153
- ${chalk.yellow("Deploy:")} npm run deploy
427
+ ${chalk.yellow("Start dev server:")} ${runScript} dev
428
+ ${chalk.yellow("Deploy:")} ${runScript} deploy
154
429
 
155
430
  \uD83D\uDCD6 ${chalk.cyan.bold("Explore Documentation")}
156
431
  ${chalk.underline.blue("https://pylon.cronit.io/docs")}
@@ -166,173 +441,12 @@ async function main(targetDir, options, command) {
166
441
  clientPath: clientPath || undefined,
167
442
  clientPort: parseInt(clientPort) || undefined
168
443
  });
169
- consola.box(message);
444
+ consola2.box(message);
170
445
  } catch (e) {
171
- consola.error(e);
446
+ consola2.error(e);
172
447
  }
173
448
  }
174
- var __filename2 = fileURLToPath(import.meta.url);
175
- var __dirname2 = dirname(__filename2);
176
- var version = (() => {
177
- return JSON.parse(fs.readFileSync(path.join(__dirname2, "..", "package.json"), "utf-8")).version;
178
- })();
179
- var runtimes = [
180
- {
181
- key: "bun",
182
- name: "Bun.js",
183
- website: "https://bunjs.dev",
184
- templates: ["default"]
185
- },
186
- {
187
- key: "node",
188
- name: "Node.js",
189
- website: "https://nodejs.org",
190
- templates: ["default"]
191
- },
192
- {
193
- key: "cf-workers",
194
- name: "Cloudflare Workers",
195
- website: "https://workers.cloudflare.com",
196
- templates: ["default"]
197
- }
198
- ];
199
- var templates = [
200
- {
201
- key: "default",
202
- name: "Default",
203
- description: "Default template"
204
- },
205
- {
206
- key: "database",
207
- name: "Database (Prisma)",
208
- description: "Template with Prisma ORM"
209
- }
210
- ];
211
- var injectVariablesInContent = (content, variables) => {
212
- let result = content;
213
- Object.entries(variables).forEach(([key, value]) => {
214
- result = result.replaceAll(key, value);
215
- });
216
- return result;
217
- };
218
- var readdirFilesSyncRecursive = (dir) => {
219
- const run = (dir2) => {
220
- const result = [];
221
- const files = fs.readdirSync(dir2);
222
- files.forEach((file) => {
223
- const filePath = path.join(dir2, file);
224
- if (fs.statSync(filePath).isDirectory()) {
225
- result.push(...run(filePath));
226
- }
227
- if (fs.statSync(filePath).isFile()) {
228
- result.push(filePath);
229
- }
230
- });
231
- return result;
232
- };
233
- return run(dir).map((file) => {
234
- return file.replace(dir, ".");
235
- });
236
- };
237
- var createTemplate = async (options) => {
238
- const { runtime, template, target } = options;
239
- const runtimeName = runtimes.find(({ key }) => key === runtime)?.name;
240
- const templateName = templates.find(({ key }) => key === template)?.name;
241
- if (!runtimeName) {
242
- throw new Error(`Invalid runtime: ${runtime}`);
243
- }
244
- if (!templateName) {
245
- throw new Error(`Invalid template: ${template}`);
246
- }
247
- const sharedTemplateDir = path.join(__dirname2, "..", "templates", "shared");
248
- if (!fs.existsSync(sharedTemplateDir)) {
249
- throw new Error(`Shared templates not found: ${sharedTemplateDir}`);
250
- }
251
- const templateDir = path.join(__dirname2, "..", "templates", runtime, template);
252
- if (!fs.existsSync(templateDir)) {
253
- throw new Error(`Template not found: ${templateDir}`);
254
- }
255
- const targetDirectoryPath = path.join(process.cwd(), target);
256
- consola.start(`Creating pylon in ${targetDirectoryPath}`);
257
- const inject = (content) => {
258
- return injectVariablesInContent(content, {
259
- __PYLON_NAME__: options.name
260
- });
261
- };
262
- readdirFilesSyncRecursive(sharedTemplateDir).forEach((file) => {
263
- const source = path.join(sharedTemplateDir, file);
264
- let target2 = path.join(targetDirectoryPath, file);
265
- const targetDir = path.dirname(target2);
266
- if (runtime === "cf-workers" && source.includes(".github/workflows/publish.yaml")) {
267
- return;
268
- }
269
- if (!fs.existsSync(targetDir)) {
270
- fs.mkdirSync(targetDir, { recursive: true });
271
- }
272
- if (target2.endsWith(".example")) {
273
- target2 = target2.replace(".example", "");
274
- }
275
- const injectedContent = inject(fs.readFileSync(source, "utf-8"));
276
- fs.writeFileSync(target2, injectedContent);
277
- });
278
- readdirFilesSyncRecursive(templateDir).forEach((file) => {
279
- const source = path.join(templateDir, file);
280
- let target2 = path.join(targetDirectoryPath, file);
281
- const targetDir = path.dirname(target2);
282
- if (!fs.existsSync(targetDir)) {
283
- fs.mkdirSync(targetDir, { recursive: true });
284
- }
285
- if (target2.endsWith(".example")) {
286
- target2 = target2.replace(".example", "");
287
- }
288
- const injectedContent = inject(fs.readFileSync(source, "utf-8"));
289
- fs.writeFileSync(target2, injectedContent);
290
- });
291
- consola.success(`Pylon created`);
292
- };
293
- var installDependencies = async (args) => {
294
- const target = path.resolve(args.target);
295
- console.log("target", target);
296
- if (!args.packageManager) {
297
- args.packageManager = await detect({
298
- cwd: target,
299
- includeGlobalBun: true
300
- });
301
- }
302
- if (!args.packageManager) {
303
- throw new Error("No package manager found");
304
- }
305
- const { packageManager } = args;
306
- let command = "";
307
- switch (packageManager) {
308
- case "yarn":
309
- command = "yarn";
310
- break;
311
- case "npm":
312
- command = "npm install";
313
- break;
314
- case "pnpm":
315
- command = "pnpm install";
316
- break;
317
- case "bun":
318
- command = "bun install";
319
- break;
320
- default:
321
- throw new Error(`Invalid package manager: ${packageManager}`);
322
- }
323
- consola.start(`Installing dependencies using ${packageManager}`);
324
- const proc = spawnSync(command, {
325
- cwd: target,
326
- shell: true,
327
- stdio: "inherit"
328
- });
329
- if (proc.status !== 0) {
330
- throw new Error(`Failed to install dependencies`);
331
- }
332
- consola.success(`Dependencies installed`);
333
- };
334
- program.name("create-pylon").version(version).arguments("[target]").addOption(new Option("-i, --install", "Install dependencies")).addOption(new Option("-r, --runtime <runtime>", "Runtime").choices(runtimes.map(({ key }) => key))).addOption(new Option("-t, --template <template>", "Template")).addOption(new Option("-pm, --package-manager <packageManager>", "Package manager")).addOption(new Option("--client", "Enable client generation (https://pylon.cronit.io/docs/integrations/gqty)")).addOption(new Option("--client-path <clientPath>", "Client path")).addOption(new Option("--client-port <clientPort>", "Client port")).action(main);
335
449
  program.parse();
336
450
 
337
- //# debugId=55AF547960838BB164756E2164756E21
338
- //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/index.ts"],
  "sourcesContent": [
    "#!/usr/bin/env node\n\nimport {Option, program, type Command} from 'commander'\nimport consola from 'consola'\nimport {input, select, confirm} from '@inquirer/prompts'\nimport path from 'path'\nimport chalk from 'chalk'\nimport * as fs from 'fs'\n\nimport * as telemetry from '@getcronit/pylon-telemetry'\n\nimport {fileURLToPath} from 'url'\nimport {dirname} from 'path'\n\nconst __filename = fileURLToPath(import.meta.url)\nconst __dirname = dirname(__filename)\n\nconst version = (() => {\n  return JSON.parse(\n    fs.readFileSync(path.join(__dirname, '..', 'package.json'), 'utf-8')\n  ).version as string\n})()\n\nfunction mkdirp(dir: string) {\n  try {\n    fs.mkdirSync(dir, {recursive: true})\n  } catch (e) {\n    if (e instanceof Error) {\n      if ('code' in e && e.code === 'EEXIST') return\n    }\n    throw e\n  }\n}\n\nconst runtimes: {\n  key: string\n  name: string\n  website: string\n  templates?: string[]\n}[] = [\n  {\n    key: 'bun',\n    name: 'Bun.js',\n    website: 'https://bunjs.dev',\n    templates: ['default']\n  },\n  {\n    key: 'node',\n    name: 'Node.js',\n    website: 'https://nodejs.org',\n    templates: ['default']\n  },\n  {\n    key: 'cf-workers',\n    name: 'Cloudflare Workers',\n    website: 'https://workers.cloudflare.com',\n    templates: ['default']\n  }\n]\n\nconst templates: {\n  key: string\n  name: string\n  description: string\n}[] = [\n  {\n    key: 'default',\n    name: 'Default',\n    description: 'Default template'\n  },\n  {\n    key: 'database',\n    name: 'Database (Prisma)',\n    description: 'Template with Prisma ORM'\n  }\n]\n\nconst injectVariablesInContent = (\n  content: string,\n  variables: Record<string, string>\n) => {\n  let result = content\n\n  Object.entries(variables).forEach(([key, value]) => {\n    result = result.replaceAll(key, value)\n  })\n\n  return result\n}\nconst readdirFilesSyncRecursive = (dir: string): string[] => {\n  const run = (dir: string): string[] => {\n    const result: string[] = []\n\n    const files = fs.readdirSync(dir)\n\n    files.forEach(file => {\n      const filePath = path.join(dir, file)\n\n      if (fs.statSync(filePath).isDirectory()) {\n        result.push(...run(filePath))\n      }\n\n      // Only add files\n      if (fs.statSync(filePath).isFile()) {\n        result.push(filePath)\n      }\n    })\n\n    return result\n  }\n\n  return run(dir).map(file => {\n    return file.replace(dir, '.')\n  })\n}\n\nconst createTemplate = async (options: {\n  name: string\n  runtime: string\n  template: string\n  target: string\n}) => {\n  const {runtime, template, target} = options\n\n  const runtimeName = runtimes.find(({key}) => key === runtime)?.name\n  const templateName = templates.find(({key}) => key === template)?.name\n\n  if (!runtimeName) {\n    throw new Error(`Invalid runtime: ${runtime}`)\n  }\n\n  if (!templateName) {\n    throw new Error(`Invalid template: ${template}`)\n  }\n\n  // The templates are stored in the `templates` directory\n  const sharedTemplateDir = path.join(__dirname, '..', 'templates', 'shared')\n\n  if (!fs.existsSync(sharedTemplateDir)) {\n    throw new Error(`Shared templates not found: ${sharedTemplateDir}`)\n  }\n\n  const templateDir = path.join(__dirname, '..', 'templates', runtime, template)\n\n  if (!fs.existsSync(templateDir)) {\n    throw new Error(`Template not found: ${templateDir}`)\n  }\n\n  // The target directory is already created\n  const targetDirectoryPath = path.join(process.cwd(), target)\n\n  consola.start(`Creating pylon in ${targetDirectoryPath}`)\n\n  const inject = (content: string) => {\n    return injectVariablesInContent(content, {\n      __PYLON_NAME__: options.name\n    })\n  }\n\n  // Copy the shared template files\n  readdirFilesSyncRecursive(sharedTemplateDir).forEach(file => {\n    const source = path.join(sharedTemplateDir, file)\n    let target = path.join(targetDirectoryPath, file)\n\n    // Create folder recursively and copy file\n\n    const targetDir = path.dirname(target)\n\n    // Skip the .github/workflows directory for cf-workers runtime\n    if (\n      runtime === 'cf-workers' &&\n      source.includes('.github/workflows/publish.yaml')\n    ) {\n      return\n    }\n\n    if (!fs.existsSync(targetDir)) {\n      fs.mkdirSync(targetDir, {recursive: true})\n    }\n\n    // If the target ends with `.example`, remove the suffix.\n    // This is useful for `.gitignore.example` files because they are not published in\n    // the `create-pylon` package when named `.gitignore`.\n    if (target.endsWith('.example')) {\n      target = target.replace('.example', '')\n    }\n\n    const injectedContent = inject(fs.readFileSync(source, 'utf-8'))\n\n    fs.writeFileSync(target, injectedContent)\n  })\n\n  // Copy the runtime specific template files\n  readdirFilesSyncRecursive(templateDir).forEach(file => {\n    const source = path.join(templateDir, file)\n    let target = path.join(targetDirectoryPath, file)\n\n    // Create folder recursively and copy file\n    const targetDir = path.dirname(target)\n\n    if (!fs.existsSync(targetDir)) {\n      fs.mkdirSync(targetDir, {recursive: true})\n    }\n\n    // If the target ends with `.example`, remove the suffix.\n    // This is useful for `.gitignore.example` files because they are not published in\n    // the `create-pylon` package when named `.gitignore`.\n    if (target.endsWith('.example')) {\n      target = target.replace('.example', '')\n    }\n\n    const injectedContent = inject(fs.readFileSync(source, 'utf-8'))\n\n    fs.writeFileSync(target, injectedContent)\n  })\n\n  consola.success(`Pylon created`)\n}\n\nimport {detect} from 'detect-package-manager'\nimport {spawnSync} from 'child_process'\n\nconst installDependencies = async (args: {\n  target: string\n  packageManager?: string\n}) => {\n  const target = path.resolve(args.target)\n\n  console.log('target', target)\n\n  if (!args.packageManager) {\n    args.packageManager = await detect({\n      cwd: target,\n      includeGlobalBun: true\n    })\n  }\n\n  if (!args.packageManager) {\n    throw new Error('No package manager found')\n  }\n\n  // yarn', 'npm', or 'pnpm', 'bun'\n  const {packageManager} = args\n\n  let command = ''\n\n  switch (packageManager) {\n    case 'yarn':\n      command = 'yarn'\n      break\n    case 'npm':\n      command = 'npm install'\n      break\n    case 'pnpm':\n      command = 'pnpm install'\n      break\n    case 'bun':\n      command = 'bun install'\n      break\n    default:\n      throw new Error(`Invalid package manager: ${packageManager}`)\n  }\n\n  consola.start(`Installing dependencies using ${packageManager}`)\n\n  const proc = spawnSync(command, {\n    cwd: target,\n    shell: true,\n    stdio: 'inherit'\n  })\n\n  if (proc.status !== 0) {\n    throw new Error(`Failed to install dependencies`)\n  }\n\n  consola.success(`Dependencies installed`)\n}\n\nprogram\n  .name('create-pylon')\n  .version(version)\n  .arguments('[target]')\n  .addOption(new Option('-i, --install', 'Install dependencies'))\n  .addOption(\n    new Option('-r, --runtime <runtime>', 'Runtime').choices(\n      runtimes.map(({key}) => key)\n    )\n  )\n  .addOption(new Option('-t, --template <template>', 'Template'))\n  .addOption(\n    new Option('-pm, --package-manager <packageManager>', 'Package manager')\n  )\n  .addOption(\n    new Option(\n      '--client',\n      'Enable client generation (https://pylon.cronit.io/docs/integrations/gqty)'\n    )\n  )\n  .addOption(new Option('--client-path <clientPath>', 'Client path'))\n  .addOption(new Option('--client-port <clientPort>', 'Client port'))\n  .action(main)\n\ntype ArgOptions = {\n  install: boolean\n  runtime: string\n  template: string\n  packageManager?: string\n  client?: boolean\n  clientPath?: string\n  clientPort?: string\n}\n\nasync function main(\n  targetDir: string | undefined,\n  options: ArgOptions,\n  command: Command\n) {\n  try {\n    consola.log(`${command.name()} version ${command.version()}`)\n\n    const {\n      install: installArg,\n      runtime: runtimeArg,\n      template: templateArg,\n      packageManager: packageManagerArg,\n      client: clientArg,\n      clientPath: clientPathArg,\n      clientPort: clientPortArg\n    } = options\n\n    let target = ''\n\n    if (targetDir) {\n      target = targetDir\n\n      consola.success(`Using target directory … ${target}`)\n    } else {\n      const answer = await input({\n        message: 'Target directory',\n        default: 'my-pylon'\n      })\n      target = answer\n    }\n\n    let projectName = ''\n\n    if (target === '.') {\n      projectName = path.basename(process.cwd())\n    } else {\n      projectName = path.basename(target)\n    }\n\n    const runtimeName =\n      runtimeArg ||\n      (await select({\n        message: 'Which runtime would you like to use?',\n        choices: runtimes.map(runtime => ({\n          name: `${runtime.name} (${runtime.website})`,\n          value: runtime.key\n        })),\n        default: 0\n      }))\n\n    if (!runtimeName) {\n      throw new Error('No runtime selected')\n    }\n\n    const runtime = runtimes.find(({key}) => key === runtimeName)\n\n    if (!runtime) {\n      throw new Error(`Invalid runtime selected: ${runtimeName}`)\n    }\n\n    const templateName =\n      templateArg ||\n      (await select({\n        message: 'Which template would you like to use?',\n        choices: templates\n          .filter(template => runtime.templates?.includes(template.key))\n          .map(template => ({\n            name: template.name,\n            value: template.key\n          })),\n        default: 0\n      }))\n\n    if (!templateName) {\n      throw new Error('No template selected')\n    }\n\n    if (fs.existsSync(target)) {\n      if (fs.readdirSync(target).length > 0) {\n        const response = await confirm({\n          message: 'Directory not empty. Continue?',\n          default: false\n        })\n        if (!response) {\n          process.exit(1)\n        }\n      }\n    } else {\n      mkdirp(target)\n    }\n\n    const install =\n      installArg ||\n      (await confirm({message: 'Would you like to install dependencies?'}))\n\n    await createTemplate({\n      name: projectName,\n      runtime: runtimeName,\n      template: templateName,\n      target\n    })\n\n    let packageManager = packageManagerArg\n\n    if (runtimeName === 'bun' && !packageManager) {\n      packageManager = 'bun'\n    }\n\n    if (install) {\n      await installDependencies({target, packageManager})\n    }\n\n    const client =\n      clientArg ||\n      (await confirm({\n        message:\n          'Would you like to enable client generation? (https://pylon.cronit.io/docs/integrations/gqty)',\n        default: false\n      }))\n\n    let clientRoot: string = ''\n    let clientPath: string = ''\n    let clientPort: string = ''\n\n    if (client) {\n      if (!clientPathArg) {\n        clientRoot = await input({\n          message: 'Path to the root where the client should be generated',\n          default: '.'\n        })\n\n        clientPath = await input({\n          message: 'Path to generate the client to',\n          default: path.join(clientRoot, 'gqty/index.ts'),\n          validate: value => {\n            // Check if the path starts with the client root (take care of .)\n            if (!value.startsWith(clientRoot === '.' ? '' : clientRoot)) {\n              return 'Path must start with the client root'\n            }\n\n            return true\n          }\n        })\n      }\n\n      clientPort =\n        clientPortArg ||\n        (await input({\n          message: 'Port of the pylon server to generate the client from',\n          default: '3000'\n        }))\n\n      consola.start(`Updating pylon dev script to generate client`)\n\n      const devScriptPath = path.join(target, 'package.json')\n\n      const devScript = JSON.parse(fs.readFileSync(devScriptPath, 'utf-8'))\n\n      devScript.scripts = {\n        ...devScript.scripts,\n        dev:\n          devScript.scripts.dev +\n          ` --client --client-port ${clientPort} --client-path ${clientPath}`\n      }\n\n      fs.writeFileSync(devScriptPath, JSON.stringify(devScript, null, 2))\n\n      consola.success(`Pylon dev script updated`)\n    }\n\n    const message = `\n🎉 ${chalk.green.bold('Pylon created successfully.')}\n\n💻 ${chalk.cyan.bold('Continue Developing')}\n    ${chalk.yellow('Change directories:')} cd ${chalk.blue(target)}\n    ${chalk.yellow('Start dev server:')} npm run start\n    ${chalk.yellow('Deploy:')} npm run deploy\n\n📖 ${chalk.cyan.bold('Explore Documentation')}\n    ${chalk.underline.blue('https://pylon.cronit.io/docs')}\n\n💬 ${chalk.cyan.bold('Join our Community')}\n    ${chalk.underline.blue('https://discord.gg/cbJjkVrnHe')}\n`\n\n    await telemetry.sendCreateEvent({\n      name: projectName,\n      pylonCreateVersion: version,\n      runtime: runtimeName,\n      template: templateName,\n      clientPath: clientPath || undefined,\n      clientPort: parseInt(clientPort) || undefined\n    })\n\n    consola.box(message)\n  } catch (e) {\n    consola.error(e)\n  }\n}\n\nprogram.parse()\n"
  ],
  "mappings": "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",
  "debugId": "55AF547960838BB164756E2164756E21",
  "names": []
}
451
+ //# debugId=49FD4AB53F50782964756E2164756E21
452
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/index.ts", "../src/detect-pm.ts"],
  "sourcesContent": [
    "#!/usr/bin/env node\n\nimport {Option, program, type Command} from 'commander'\nimport consola from 'consola'\nimport {input, select, confirm} from '@inquirer/prompts'\nimport path from 'path'\nimport chalk from 'chalk'\nimport * as fs from 'fs'\n\nimport * as telemetry from '@getcronit/pylon-telemetry'\n\nimport {fileURLToPath} from 'url'\nimport {dirname} from 'path'\n\nconst __filename = fileURLToPath(import.meta.url)\nconst __dirname = dirname(__filename)\n\nconst version = (() => {\n  return JSON.parse(\n    fs.readFileSync(path.join(__dirname, '..', 'package.json'), 'utf-8')\n  ).version as string\n})()\n\nfunction mkdirp(dir: string) {\n  try {\n    fs.mkdirSync(dir, {recursive: true})\n  } catch (e) {\n    if (e instanceof Error) {\n      if ('code' in e && e.code === 'EEXIST') return\n    }\n    throw e\n  }\n}\n\nconst runtimes: {\n  key: string\n  name: string\n  website: string\n  templates?: string[]\n}[] = [\n  {\n    key: 'bun',\n    name: 'Bun.js',\n    website: 'https://bunjs.dev',\n    templates: ['default']\n  },\n  {\n    key: 'node',\n    name: 'Node.js',\n    website: 'https://nodejs.org',\n    templates: ['default']\n  },\n  {\n    key: 'cf-workers',\n    name: 'Cloudflare Workers',\n    website: 'https://workers.cloudflare.com',\n    templates: ['default']\n  },\n  {\n    key: 'deno',\n    name: 'Deno',\n    website: 'https://deno.land',\n    templates: ['default']\n  }\n]\n\nconst templates: {\n  key: string\n  name: string\n  description: string\n}[] = [\n  {\n    key: 'default',\n    name: 'Default',\n    description: 'Default template'\n  },\n  {\n    key: 'database',\n    name: 'Database (Prisma)',\n    description: 'Template with Prisma ORM'\n  }\n]\n\nconst injectVariablesInContent = (\n  content: string,\n  variables: Record<string, string>\n) => {\n  let result = content\n\n  Object.entries(variables).forEach(([key, value]) => {\n    result = result.replaceAll(key, value)\n  })\n\n  return result\n}\nconst readdirFilesSyncRecursive = (dir: string): string[] => {\n  const run = (dir: string): string[] => {\n    const result: string[] = []\n\n    const files = fs.readdirSync(dir)\n\n    files.forEach(file => {\n      const filePath = path.join(dir, file)\n\n      if (fs.statSync(filePath).isDirectory()) {\n        result.push(...run(filePath))\n      }\n\n      // Only add files\n      if (fs.statSync(filePath).isFile()) {\n        result.push(filePath)\n      }\n    })\n\n    return result\n  }\n\n  return run(dir).map(file => {\n    return file.replace(dir, '.')\n  })\n}\n\nconst createTemplate = async (options: {\n  name: string\n  runtime: string\n  template: string\n  target: string\n}) => {\n  const {runtime, template, target} = options\n\n  const runtimeName = runtimes.find(({key}) => key === runtime)?.name\n  const templateName = templates.find(({key}) => key === template)?.name\n\n  if (!runtimeName) {\n    throw new Error(`Invalid runtime: ${runtime}`)\n  }\n\n  if (!templateName) {\n    throw new Error(`Invalid template: ${template}`)\n  }\n\n  // The templates are stored in the `templates` directory\n  const sharedTemplateDir = path.join(__dirname, '..', 'templates', 'shared')\n\n  if (!fs.existsSync(sharedTemplateDir)) {\n    throw new Error(`Shared templates not found: ${sharedTemplateDir}`)\n  }\n\n  const templateDir = path.join(__dirname, '..', 'templates', runtime, template)\n\n  if (!fs.existsSync(templateDir)) {\n    throw new Error(`Template not found: ${templateDir}`)\n  }\n\n  // The target directory is already created\n  const targetDirectoryPath = path.join(process.cwd(), target)\n\n  consola.start(`Creating pylon in ${targetDirectoryPath}`)\n\n  const inject = (content: string) => {\n    return injectVariablesInContent(content, {\n      __PYLON_NAME__: options.name\n    })\n  }\n\n  // Copy the shared template files\n  readdirFilesSyncRecursive(sharedTemplateDir).forEach(file => {\n    const source = path.join(sharedTemplateDir, file)\n    let target = path.join(targetDirectoryPath, file)\n\n    // Create folder recursively and copy file\n\n    const targetDir = path.dirname(target)\n\n    // Skip the .github/workflows directory for cf-workers runtime\n    if (\n      runtime === 'cf-workers' &&\n      source.includes('.github/workflows/publish.yaml')\n    ) {\n      return\n    }\n\n    if (!fs.existsSync(targetDir)) {\n      fs.mkdirSync(targetDir, {recursive: true})\n    }\n\n    // If the target ends with `.example`, remove the suffix.\n    // This is useful for `.gitignore.example` files because they are not published in\n    // the `create-pylon` package when named `.gitignore`.\n    if (target.endsWith('.example')) {\n      target = target.replace('.example', '')\n    }\n\n    const injectedContent = inject(fs.readFileSync(source, 'utf-8'))\n\n    fs.writeFileSync(target, injectedContent)\n  })\n\n  // Copy the runtime specific template files\n  readdirFilesSyncRecursive(templateDir).forEach(file => {\n    const source = path.join(templateDir, file)\n    let target = path.join(targetDirectoryPath, file)\n\n    // Create folder recursively and copy file\n    const targetDir = path.dirname(target)\n\n    if (!fs.existsSync(targetDir)) {\n      fs.mkdirSync(targetDir, {recursive: true})\n    }\n\n    // If the target ends with `.example`, remove the suffix.\n    // This is useful for `.gitignore.example` files because they are not published in\n    // the `create-pylon` package when named `.gitignore`.\n    if (target.endsWith('.example')) {\n      target = target.replace('.example', '')\n    }\n\n    const injectedContent = inject(fs.readFileSync(source, 'utf-8'))\n\n    fs.writeFileSync(target, injectedContent)\n  })\n\n  consola.success(`Pylon created`)\n}\n\nimport {spawnSync} from 'child_process'\nimport {detectPackageManager, getRunScript, PackageManager} from './detect-pm'\n\nconst installDependencies = async (args: {\n  target: string\n  packageManager: PackageManager\n}) => {\n  const target = path.resolve(args.target)\n  const packageManager = args.packageManager\n\n  let command = ''\n\n  switch (packageManager) {\n    case 'yarn':\n      command = 'yarn'\n      break\n    case 'npm':\n      command = 'npm install'\n      break\n    case 'pnpm':\n      command = 'pnpm install'\n      break\n    case 'bun':\n      command = 'bun install'\n      break\n    case 'deno':\n      command = 'deno install'\n      break\n    default:\n      throw new Error(`Invalid package manager: ${packageManager}`)\n  }\n\n  consola.start(`Installing dependencies using ${packageManager}`)\n\n  const proc = spawnSync(command, {\n    cwd: target,\n    shell: true,\n    stdio: 'inherit'\n  })\n\n  if (proc.status !== 0) {\n    throw new Error(`Failed to install dependencies`)\n  }\n\n  consola.success(`Dependencies installed`)\n}\n\nprogram\n  .name('create-pylon')\n  .version(version)\n  .arguments('[target]')\n  .addOption(new Option('-i, --install', 'Install dependencies'))\n  .addOption(\n    new Option('-r, --runtime <runtime>', 'Runtime').choices(\n      runtimes.map(({key}) => key)\n    )\n  )\n  .addOption(new Option('-t, --template <template>', 'Template'))\n  .addOption(\n    new Option('-pm, --package-manager <packageManager>', 'Package manager')\n  )\n  .addOption(\n    new Option(\n      '--client',\n      'Enable client generation (https://pylon.cronit.io/docs/integrations/gqty)'\n    )\n  )\n  .addOption(new Option('--client-path <clientPath>', 'Client path'))\n  .addOption(new Option('--client-port <clientPort>', 'Client port'))\n  .action(main)\n\ntype ArgOptions = {\n  install: boolean\n  runtime: string\n  template: string\n  packageManager?: PackageManager\n  client?: boolean\n  clientPath?: string\n  clientPort?: string\n}\n\nconst getPreferredPmByRuntime = (\n  runtime: string\n): PackageManager | undefined => {\n  if (runtime === 'bun') {\n    return 'bun'\n  } else if (runtime === 'deno') {\n    return 'deno'\n  }\n}\n\nasync function main(\n  targetDir: string | undefined,\n  options: ArgOptions,\n  command: Command\n) {\n  try {\n    consola.log(`${command.name()} version ${command.version()}`)\n\n    const {\n      install: installArg,\n      runtime: runtimeArg,\n      template: templateArg,\n      packageManager: packageManagerArg,\n      client: clientArg,\n      clientPath: clientPathArg,\n      clientPort: clientPortArg\n    } = options\n\n    let target = ''\n\n    if (targetDir) {\n      target = targetDir\n\n      consola.success(`Using target directory … ${target}`)\n    } else {\n      const answer = await input({\n        message: 'Target directory',\n        default: 'my-pylon'\n      })\n      target = answer\n    }\n\n    let projectName = ''\n\n    if (target === '.') {\n      projectName = path.basename(process.cwd())\n    } else {\n      projectName = path.basename(target)\n    }\n\n    const runtimeName =\n      runtimeArg ||\n      (await select({\n        message: 'Which runtime would you like to use?',\n        choices: runtimes.map(runtime => ({\n          name: `${runtime.name} (${runtime.website})`,\n          value: runtime.key\n        })),\n        default: 0\n      }))\n\n    if (!runtimeName) {\n      throw new Error('No runtime selected')\n    }\n\n    const runtime = runtimes.find(({key}) => key === runtimeName)\n\n    if (!runtime) {\n      throw new Error(`Invalid runtime selected: ${runtimeName}`)\n    }\n\n    const templateName =\n      templateArg ||\n      (await select({\n        message: 'Which template would you like to use?',\n        choices: templates\n          .filter(template => runtime.templates?.includes(template.key))\n          .map(template => ({\n            name: template.name,\n            value: template.key\n          })),\n        default: 0\n      }))\n\n    if (!templateName) {\n      throw new Error('No template selected')\n    }\n\n    if (fs.existsSync(target)) {\n      if (fs.readdirSync(target).length > 0) {\n        const response = await confirm({\n          message: 'Directory not empty. Continue?',\n          default: false\n        })\n        if (!response) {\n          process.exit(1)\n        }\n      }\n    } else {\n      mkdirp(target)\n    }\n\n    const install =\n      installArg ||\n      (await confirm({message: 'Would you like to install dependencies?'}))\n\n    await createTemplate({\n      name: projectName,\n      runtime: runtimeName,\n      template: templateName,\n      target\n    })\n\n    const packageManager = detectPackageManager({\n      preferredPm: getPreferredPmByRuntime(runtime.key),\n      cwd: target\n    })\n\n    if (install) {\n      await installDependencies({target, packageManager})\n    }\n\n    const client =\n      clientArg ||\n      (await confirm({\n        message:\n          'Would you like to enable client generation? (https://pylon.cronit.io/docs/integrations/gqty)',\n        default: false\n      }))\n\n    let clientRoot: string = ''\n    let clientPath: string = ''\n    let clientPort: string = ''\n\n    if (client) {\n      if (!clientPathArg) {\n        clientRoot = await input({\n          message: 'Path to the root where the client should be generated',\n          default: '.'\n        })\n\n        clientPath = await input({\n          message: 'Path to generate the client to',\n          default: path.join(clientRoot, 'gqty/index.ts'),\n          validate: value => {\n            // Check if the path starts with the client root (take care of .)\n            if (!value.startsWith(clientRoot === '.' ? '' : clientRoot)) {\n              return 'Path must start with the client root'\n            }\n\n            return true\n          }\n        })\n      }\n\n      clientPort =\n        clientPortArg ||\n        (await input({\n          message: 'Port of the pylon server to generate the client from',\n          default: '3000'\n        }))\n\n      consola.start(`Updating pylon dev script to generate client`)\n\n      let packagePath: string\n      let scriptKey: string\n      if (runtime.key === 'deno') {\n        packagePath = path.join(target, 'deno.json')\n        scriptKey = 'tasks'\n      } else {\n        packagePath = path.join(target, 'package.json')\n        scriptKey = 'scripts'\n      }\n\n      const devScript = JSON.parse(fs.readFileSync(packagePath, 'utf-8'))\n\n      devScript[scriptKey] = {\n        ...devScript[scriptKey],\n        dev:\n          devScript[scriptKey].dev +\n          ` --client --client-port ${clientPort} --client-path ${clientPath}`\n      }\n\n      fs.writeFileSync(packagePath, JSON.stringify(devScript, null, 2))\n\n      consola.success(`Pylon dev script updated`)\n    }\n\n    const runScript = getRunScript(packageManager)\n\n    const message = `\n🎉 ${chalk.green.bold('Pylon created successfully.')}\n\n💻 ${chalk.cyan.bold('Continue Developing')}\n    ${chalk.yellow('Change directories:')} cd ${chalk.blue(target)}\n    ${chalk.yellow('Start dev server:')} ${runScript} dev\n    ${chalk.yellow('Deploy:')} ${runScript} deploy\n\n📖 ${chalk.cyan.bold('Explore Documentation')}\n    ${chalk.underline.blue('https://pylon.cronit.io/docs')}\n\n💬 ${chalk.cyan.bold('Join our Community')}\n    ${chalk.underline.blue('https://discord.gg/cbJjkVrnHe')}\n`\n\n    await telemetry.sendCreateEvent({\n      name: projectName,\n      pylonCreateVersion: version,\n      runtime: runtimeName,\n      template: templateName,\n      clientPath: clientPath || undefined,\n      clientPort: parseInt(clientPort) || undefined\n    })\n\n    consola.box(message)\n  } catch (e) {\n    consola.error(e)\n  }\n}\n\nprogram.parse()\n",
    "import * as fs from 'node:fs'\nimport * as path from 'node:path'\nimport process from 'node:process'\nimport {execSync} from 'node:child_process'\nimport consola from 'consola'\n\n// Helper function to check if a command exists\nfunction isCommandAvailable(command: string): boolean {\n  try {\n    execSync(`${command} --version`, {stdio: 'ignore'})\n    return true\n  } catch (e) {\n    console.error(e)\n    return false\n  }\n}\n\n// Detect Bun\nfunction isBun(): boolean {\n  // @ts-ignore: Bun may not be defined\n  return typeof Bun !== 'undefined' && isCommandAvailable('bun')\n}\n\n// Detect npm\nfunction isNpm(): boolean {\n  return process.env.npm_execpath?.includes('npm') ?? false\n}\n\n// Detect Yarn\nfunction isYarn(): boolean {\n  return process.env.npm_execpath?.includes('yarn') ?? false\n}\n\n// Detect Deno\nfunction isDeno(): boolean {\n  // @ts-ignore: Deno may not be defined\n  return typeof Deno !== 'undefined' && isCommandAvailable('deno')\n}\n\n// Detect pnpm\nfunction isPnpm(): boolean {\n  return process.env.npm_execpath?.includes('pnpm') ?? false\n}\n\n// Detect based on lock files\nfunction detectByLockFiles(cwd: string): PackageManager | null {\n  if (fs.existsSync(path.join(cwd, 'bun.lockb'))) {\n    return 'bun'\n  }\n  if (fs.existsSync(path.join(cwd, 'package-lock.json'))) {\n    return 'npm'\n  }\n  if (fs.existsSync(path.join(cwd, 'yarn.lock'))) {\n    return 'yarn'\n  }\n  if (\n    fs.existsSync(path.join(cwd, 'deno.json')) ||\n    fs.existsSync(path.join(cwd, 'deno.lock'))\n  ) {\n    return 'deno'\n  }\n  if (fs.existsSync(path.join(cwd, 'pnpm-lock.yaml'))) {\n    return 'pnpm'\n  }\n  return null\n}\n\nexport type PackageManager =\n  | 'bun'\n  | 'npm'\n  | 'yarn'\n  | 'pnpm'\n  | 'deno'\n  | 'unknown'\n\n// Main detection function\nexport function detectPackageManager({\n  preferredPm,\n  cwd = process.cwd()\n}: {\n  preferredPm?: PackageManager\n  cwd?: string\n}): PackageManager {\n  // Check the preferred package manager first\n  if (preferredPm && isCommandAvailable(preferredPm)) {\n    return preferredPm\n  }\n\n  // Proceed with detection logic\n  if (isBun()) {\n    return 'bun'\n  }\n  if (isNpm()) {\n    return 'npm'\n  }\n  if (isPnpm()) {\n    return 'pnpm'\n  }\n  if (isDeno()) {\n    return 'deno'\n  }\n  if (isYarn()) {\n    return 'yarn'\n  }\n\n  // Fallback to lock file detection\n  const lockFileDetection = detectByLockFiles(cwd)\n  if (lockFileDetection) {\n    consola.info(`Detected package manager by lock file: ${lockFileDetection}`)\n    if (isCommandAvailable(lockFileDetection)) {\n      return lockFileDetection\n    } else {\n      consola.warn(\n        `Lock file detected, but ${lockFileDetection} is not installed.`\n      )\n    }\n  }\n\n  return 'unknown'\n}\n\ntype PackageManagerScript =\n  | 'bun'\n  | 'npm run'\n  | 'yarn'\n  | 'pnpm run'\n  | 'deno task'\n\n// Run script detection\nexport function getRunScript(pm: PackageManager): PackageManagerScript {\n  switch (pm) {\n    case 'bun':\n      return 'bun'\n    case 'npm':\n      return 'npm run'\n    case 'yarn':\n      return 'yarn'\n    case 'pnpm':\n      return 'pnpm run'\n    case 'deno':\n      return 'deno task'\n    default:\n      throw new Error('Unknown package manager')\n  }\n}\n"
  ],
  "mappings": "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",
  "debugId": "49FD4AB53F50782964756E2164756E21",
  "names": []
}
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "create-pylon",
3
3
  "type": "module",
4
- "version": "1.0.3",
4
+ "version": "1.1.0",
5
5
  "description": "CLI for creating a Pylon",
6
6
  "scripts": {
7
7
  "build": "bun build ./src/index.ts --target=bun --outdir=./dist --sourcemap=inline --packages external"
@@ -24,7 +24,6 @@
24
24
  "chalk": "^5.3.0",
25
25
  "commander": "^12.1.0",
26
26
  "consola": "^3.2.3",
27
- "detect-package-manager": "^3.0.2",
28
27
  "@getcronit/pylon-telemetry": "^1.0.0"
29
28
  },
30
29
  "engines": {
@@ -5,7 +5,7 @@
5
5
  "type": "module",
6
6
  "description": "Generated with `npm create pylon`",
7
7
  "scripts": {
8
- "dev": "pylon dev -c 'bun run .pylon/index.js'",
8
+ "dev": "pylon dev -c \"bun run .pylon/index.js\"",
9
9
  "build": "pylon build"
10
10
  },
11
11
  "dependencies": {
@@ -4,7 +4,7 @@
4
4
  "private": true,
5
5
  "scripts": {
6
6
  "deploy": "pylon build && wrangler deploy",
7
- "dev": "pylon dev -c 'wrangler dev'",
7
+ "dev": "pylon dev -c \"wrangler dev\"",
8
8
  "cf-typegen": "wrangler types"
9
9
  },
10
10
  "dependencies": {
@@ -0,0 +1,5 @@
1
+ {
2
+ "recommendations": [
3
+ "denoland.vscode-deno"
4
+ ]
5
+ }
@@ -0,0 +1,6 @@
1
+ {
2
+ "deno.enablePaths": [
3
+ "./"
4
+ ],
5
+ "editor.inlayHints.enabled": "off"
6
+ }
@@ -0,0 +1,12 @@
1
+ FROM denoland/deno
2
+
3
+ EXPOSE 3000
4
+
5
+ WORKDIR /app
6
+
7
+ ADD . /app
8
+
9
+ RUN deno install
10
+ RUN deno task build
11
+
12
+ CMD ["run", "-A", ".pylon/index.js"]
@@ -0,0 +1,15 @@
1
+ {
2
+ "imports": {
3
+ "@getcronit/pylon-dev": "npm:@getcronit/pylon-dev@^1.0.1",
4
+ "@getcronit/pylon": "npm:@getcronit/pylon@^2.2.1"
5
+ },
6
+ "tasks": {
7
+ "dev": "pylon dev -c 'deno run -A .pylon/index.js --config tsconfig.json'",
8
+ "build": "pylon build"
9
+ },
10
+ "compilerOptions": {
11
+ "jsx": "precompile",
12
+ "jsxImportSource": "hono/jsx"
13
+ },
14
+ "nodeModulesDir": "auto"
15
+ }
@@ -0,0 +1,17 @@
1
+ import {app} from '@getcronit/pylon'
2
+
3
+ export const graphql = {
4
+ Query: {
5
+ hello: () => {
6
+ return 'Hello, world!'
7
+ }
8
+ },
9
+ Mutation: {}
10
+ }
11
+
12
+ Deno.serve(
13
+ {
14
+ port: 3000
15
+ },
16
+ app.fetch
17
+ )
@@ -5,7 +5,7 @@
5
5
  "type": "module",
6
6
  "description": "Generated with `npm create pylon`",
7
7
  "scripts": {
8
- "dev": "pylon dev -c 'node --enable-source-maps .pylon/index.js'",
8
+ "dev": "pylon dev -c \"node --enable-source-maps .pylon/index.js\"",
9
9
  "build": "pylon build"
10
10
  },
11
11
  "dependencies": {