create-pylon 0.0.1
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 +305 -0
- package/package.json +27 -0
- package/templates/bun/default/Dockerfile +47 -0
- package/templates/bun/default/package.json +23 -0
- package/templates/bun/default/src/index.ts +12 -0
- package/templates/bun/default/tsconfig.json +7 -0
- package/templates/cf-worker/default/package.json +25 -0
- package/templates/cf-worker/default/src/index.ts +12 -0
- package/templates/cf-worker/default/tsconfig.json +4 -0
- package/templates/cf-worker/default/worker-configuration.d.ts +3 -0
- package/templates/cf-worker/default/wrangler.toml +108 -0
- package/templates/node/default/Dockerfile +45 -0
- package/templates/node/default/package.json +22 -0
- package/templates/node/default/src/index.ts +15 -0
- package/templates/shared/.dockerignore +15 -0
- package/templates/shared/.github/workflows/publish.yaml +43 -0
- package/templates/shared/tsconfig.json +3 -0
package/dist/index.js
ADDED
|
@@ -0,0 +1,305 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
// @bun
|
|
3
|
+
|
|
4
|
+
// src/index.ts
|
|
5
|
+
import {Option, program} from "commander";
|
|
6
|
+
// package.json
|
|
7
|
+
var version = "0.0.1";
|
|
8
|
+
|
|
9
|
+
// src/index.ts
|
|
10
|
+
import consola from "consola";
|
|
11
|
+
import {input, select, confirm} from "@inquirer/prompts";
|
|
12
|
+
import path from "path";
|
|
13
|
+
import * as fs from "fs";
|
|
14
|
+
function mkdirp(dir) {
|
|
15
|
+
try {
|
|
16
|
+
fs.mkdirSync(dir, { recursive: true });
|
|
17
|
+
} catch (e) {
|
|
18
|
+
if (e instanceof Error) {
|
|
19
|
+
if ("code" in e && e.code === "EEXIST")
|
|
20
|
+
return;
|
|
21
|
+
}
|
|
22
|
+
throw e;
|
|
23
|
+
}
|
|
24
|
+
}
|
|
25
|
+
import {detect} from "detect-package-manager";
|
|
26
|
+
import {spawnSync} from "child_process";
|
|
27
|
+
async function main(targetDir, options, command) {
|
|
28
|
+
try {
|
|
29
|
+
consola.log(`${command.name()} version ${command.version()}`);
|
|
30
|
+
const {
|
|
31
|
+
install: installArg,
|
|
32
|
+
runtime: runtimeArg,
|
|
33
|
+
template: templateArg,
|
|
34
|
+
packageManager: packageManagerArg,
|
|
35
|
+
client: clientArg,
|
|
36
|
+
clientPath: clientPathArg,
|
|
37
|
+
clientPort: clientPortArg
|
|
38
|
+
} = options;
|
|
39
|
+
let target = "";
|
|
40
|
+
if (targetDir) {
|
|
41
|
+
target = targetDir;
|
|
42
|
+
consola.success(`Using target directory \u2026 ${target}`);
|
|
43
|
+
} else {
|
|
44
|
+
const answer = await input({
|
|
45
|
+
message: "Target directory",
|
|
46
|
+
default: "my-pylon"
|
|
47
|
+
});
|
|
48
|
+
target = answer;
|
|
49
|
+
}
|
|
50
|
+
let projectName = "";
|
|
51
|
+
if (target === ".") {
|
|
52
|
+
projectName = path.basename(process.cwd());
|
|
53
|
+
} else {
|
|
54
|
+
projectName = path.basename(target);
|
|
55
|
+
}
|
|
56
|
+
const runtimeName = runtimeArg || await select({
|
|
57
|
+
message: "Which runtime would you like to use?",
|
|
58
|
+
choices: runtimes.map((runtime2) => ({
|
|
59
|
+
name: `${runtime2.name} (${runtime2.website})`,
|
|
60
|
+
value: runtime2.key
|
|
61
|
+
})),
|
|
62
|
+
default: 0
|
|
63
|
+
});
|
|
64
|
+
if (!runtimeName) {
|
|
65
|
+
throw new Error("No runtime selected");
|
|
66
|
+
}
|
|
67
|
+
const runtime = runtimes.find(({ key }) => key === runtimeName);
|
|
68
|
+
if (!runtime) {
|
|
69
|
+
throw new Error(`Invalid runtime selected: ${runtimeName}`);
|
|
70
|
+
}
|
|
71
|
+
const templateName = templateArg || await select({
|
|
72
|
+
message: "Which template would you like to use?",
|
|
73
|
+
choices: templates.filter((template) => runtime.templates?.includes(template.key)).map((template) => ({
|
|
74
|
+
name: template.name,
|
|
75
|
+
value: template.key
|
|
76
|
+
})),
|
|
77
|
+
default: 0
|
|
78
|
+
});
|
|
79
|
+
if (!templateName) {
|
|
80
|
+
throw new Error("No template selected");
|
|
81
|
+
}
|
|
82
|
+
if (fs.existsSync(target)) {
|
|
83
|
+
if (fs.readdirSync(target).length > 0) {
|
|
84
|
+
const response = await confirm({
|
|
85
|
+
message: "Directory not empty. Continue?",
|
|
86
|
+
default: false
|
|
87
|
+
});
|
|
88
|
+
if (!response) {
|
|
89
|
+
process.exit(1);
|
|
90
|
+
}
|
|
91
|
+
}
|
|
92
|
+
} else {
|
|
93
|
+
mkdirp(target);
|
|
94
|
+
}
|
|
95
|
+
const install = installArg || await confirm({ message: "Would you like to install dependencies?" });
|
|
96
|
+
await createTemplate({
|
|
97
|
+
name: projectName,
|
|
98
|
+
runtime: runtimeName,
|
|
99
|
+
template: templateName,
|
|
100
|
+
target
|
|
101
|
+
});
|
|
102
|
+
let packageManager = packageManagerArg;
|
|
103
|
+
if (runtimeName === "bun" && !packageManager) {
|
|
104
|
+
packageManager = "bun";
|
|
105
|
+
}
|
|
106
|
+
if (install) {
|
|
107
|
+
await installDependencies({ target, packageManager });
|
|
108
|
+
}
|
|
109
|
+
const client = clientArg || await confirm({
|
|
110
|
+
message: "Would you like to enable client generation? (https://pylon.cronit.io/docs/integrations/gqty)",
|
|
111
|
+
default: false
|
|
112
|
+
});
|
|
113
|
+
if (client) {
|
|
114
|
+
let clientRoot = "";
|
|
115
|
+
let clientPath = "";
|
|
116
|
+
if (!clientPathArg) {
|
|
117
|
+
clientRoot = await input({
|
|
118
|
+
message: "Path to the root where the client should be generated",
|
|
119
|
+
default: "."
|
|
120
|
+
});
|
|
121
|
+
clientPath = await input({
|
|
122
|
+
message: "Path to generate the client to",
|
|
123
|
+
default: path.join(clientRoot, "gqty/index.ts"),
|
|
124
|
+
validate: (value) => {
|
|
125
|
+
if (!value.startsWith(clientRoot === "." ? "" : clientRoot)) {
|
|
126
|
+
return "Path must start with the client root";
|
|
127
|
+
}
|
|
128
|
+
return true;
|
|
129
|
+
}
|
|
130
|
+
});
|
|
131
|
+
}
|
|
132
|
+
const clientPort = clientPortArg || await input({
|
|
133
|
+
message: "Port of the pylon server to generate the client from",
|
|
134
|
+
default: "3000"
|
|
135
|
+
});
|
|
136
|
+
consola.start(`Updating pylon dev script to generate client`);
|
|
137
|
+
const devScriptPath = path.join(target, "package.json");
|
|
138
|
+
const devScript = JSON.parse(fs.readFileSync(devScriptPath, "utf-8"));
|
|
139
|
+
devScript.scripts = {
|
|
140
|
+
...devScript.scripts,
|
|
141
|
+
dev: devScript.scripts.dev + ` --client --client-port ${clientPort} --client-path ${clientPath}`
|
|
142
|
+
};
|
|
143
|
+
fs.writeFileSync(devScriptPath, JSON.stringify(devScript, null, 2));
|
|
144
|
+
consola.success(`Pylon dev script updated`);
|
|
145
|
+
}
|
|
146
|
+
consola.box(`Pylon successfully created in ${target}.\n\nHappy coding!`);
|
|
147
|
+
} catch (e) {
|
|
148
|
+
consola.error(e);
|
|
149
|
+
}
|
|
150
|
+
}
|
|
151
|
+
var __dirname = "/Users/schettn/Documents/pylon/packages/create-pylon/src";
|
|
152
|
+
var runtimes = [
|
|
153
|
+
{
|
|
154
|
+
key: "bun",
|
|
155
|
+
name: "Bun.js",
|
|
156
|
+
website: "https://bunjs.dev",
|
|
157
|
+
templates: ["default"]
|
|
158
|
+
},
|
|
159
|
+
{
|
|
160
|
+
key: "node",
|
|
161
|
+
name: "Node.js",
|
|
162
|
+
website: "https://nodejs.org",
|
|
163
|
+
templates: ["default"]
|
|
164
|
+
},
|
|
165
|
+
{
|
|
166
|
+
key: "cf-workers",
|
|
167
|
+
name: "Cloudflare Workers",
|
|
168
|
+
website: "https://workers.cloudflare.com",
|
|
169
|
+
templates: ["default"]
|
|
170
|
+
}
|
|
171
|
+
];
|
|
172
|
+
var templates = [
|
|
173
|
+
{
|
|
174
|
+
key: "default",
|
|
175
|
+
name: "Default",
|
|
176
|
+
description: "Default template"
|
|
177
|
+
},
|
|
178
|
+
{
|
|
179
|
+
key: "database",
|
|
180
|
+
name: "Database (Prisma)",
|
|
181
|
+
description: "Template with Prisma ORM"
|
|
182
|
+
}
|
|
183
|
+
];
|
|
184
|
+
var injectVariablesInContent = (content, variables) => {
|
|
185
|
+
let result = content;
|
|
186
|
+
Object.entries(variables).forEach(([key, value]) => {
|
|
187
|
+
result = result.replaceAll(key, value);
|
|
188
|
+
});
|
|
189
|
+
return result;
|
|
190
|
+
};
|
|
191
|
+
var readdirFilesSyncRecursive = (dir) => {
|
|
192
|
+
const run = (dir2) => {
|
|
193
|
+
const result = [];
|
|
194
|
+
const files = fs.readdirSync(dir2);
|
|
195
|
+
files.forEach((file) => {
|
|
196
|
+
const filePath = path.join(dir2, file);
|
|
197
|
+
if (fs.statSync(filePath).isDirectory() && !filePath.includes("node_modules")) {
|
|
198
|
+
result.push(...run(filePath));
|
|
199
|
+
}
|
|
200
|
+
if (fs.statSync(filePath).isFile()) {
|
|
201
|
+
result.push(filePath);
|
|
202
|
+
}
|
|
203
|
+
});
|
|
204
|
+
return result;
|
|
205
|
+
};
|
|
206
|
+
return run(dir).map((file) => {
|
|
207
|
+
return file.replace(dir, ".");
|
|
208
|
+
});
|
|
209
|
+
};
|
|
210
|
+
var createTemplate = async (options) => {
|
|
211
|
+
const { runtime, template, target } = options;
|
|
212
|
+
const runtimeName = runtimes.find(({ key }) => key === runtime)?.name;
|
|
213
|
+
const templateName = templates.find(({ key }) => key === template)?.name;
|
|
214
|
+
if (!runtimeName) {
|
|
215
|
+
throw new Error(`Invalid runtime: ${runtime}`);
|
|
216
|
+
}
|
|
217
|
+
if (!templateName) {
|
|
218
|
+
throw new Error(`Invalid template: ${template}`);
|
|
219
|
+
}
|
|
220
|
+
const sharedTemplateDir = path.join(__dirname, "..", "templates", "shared");
|
|
221
|
+
if (!fs.existsSync(sharedTemplateDir)) {
|
|
222
|
+
throw new Error(`Shared templates not found: ${sharedTemplateDir}`);
|
|
223
|
+
}
|
|
224
|
+
const templateDir = path.join(__dirname, "..", "templates", runtime, template);
|
|
225
|
+
if (!fs.existsSync(templateDir)) {
|
|
226
|
+
throw new Error(`Template not found: ${templateDir}`);
|
|
227
|
+
}
|
|
228
|
+
const targetDirectoryPath = path.join(process.cwd(), target);
|
|
229
|
+
consola.start(`Creating pylon in ${targetDirectoryPath}`);
|
|
230
|
+
const inject = (content) => {
|
|
231
|
+
return injectVariablesInContent(content, {
|
|
232
|
+
__PYLON_NAME__: options.name
|
|
233
|
+
});
|
|
234
|
+
};
|
|
235
|
+
readdirFilesSyncRecursive(sharedTemplateDir).forEach((file) => {
|
|
236
|
+
const source = path.join(sharedTemplateDir, file);
|
|
237
|
+
const target2 = path.join(targetDirectoryPath, file);
|
|
238
|
+
const targetDir = path.dirname(target2);
|
|
239
|
+
if (runtime === "cf-workers" && source.includes(".github/workflows/publish.yaml")) {
|
|
240
|
+
return;
|
|
241
|
+
}
|
|
242
|
+
if (!fs.existsSync(targetDir)) {
|
|
243
|
+
fs.mkdirSync(targetDir, { recursive: true });
|
|
244
|
+
}
|
|
245
|
+
const injectedContent = inject(fs.readFileSync(source, "utf-8"));
|
|
246
|
+
fs.writeFileSync(target2, injectedContent);
|
|
247
|
+
});
|
|
248
|
+
readdirFilesSyncRecursive(templateDir).forEach((file) => {
|
|
249
|
+
const source = path.join(templateDir, file);
|
|
250
|
+
const target2 = path.join(targetDirectoryPath, file);
|
|
251
|
+
const targetDir = path.dirname(target2);
|
|
252
|
+
if (!fs.existsSync(targetDir)) {
|
|
253
|
+
fs.mkdirSync(targetDir, { recursive: true });
|
|
254
|
+
}
|
|
255
|
+
const injectedContent = inject(fs.readFileSync(source, "utf-8"));
|
|
256
|
+
fs.writeFileSync(target2, injectedContent);
|
|
257
|
+
});
|
|
258
|
+
consola.success(`Pylon created`);
|
|
259
|
+
};
|
|
260
|
+
var installDependencies = async (args) => {
|
|
261
|
+
const target = path.resolve(args.target);
|
|
262
|
+
console.log("target", target);
|
|
263
|
+
if (!args.packageManager) {
|
|
264
|
+
args.packageManager = await detect({
|
|
265
|
+
cwd: target,
|
|
266
|
+
includeGlobalBun: true
|
|
267
|
+
});
|
|
268
|
+
}
|
|
269
|
+
if (!args.packageManager) {
|
|
270
|
+
throw new Error("No package manager found");
|
|
271
|
+
}
|
|
272
|
+
const { packageManager } = args;
|
|
273
|
+
let command = "";
|
|
274
|
+
switch (packageManager) {
|
|
275
|
+
case "yarn":
|
|
276
|
+
command = "yarn";
|
|
277
|
+
break;
|
|
278
|
+
case "npm":
|
|
279
|
+
command = "npm install";
|
|
280
|
+
break;
|
|
281
|
+
case "pnpm":
|
|
282
|
+
command = "pnpm install";
|
|
283
|
+
break;
|
|
284
|
+
case "bun":
|
|
285
|
+
command = "bun install";
|
|
286
|
+
break;
|
|
287
|
+
default:
|
|
288
|
+
throw new Error(`Invalid package manager: ${packageManager}`);
|
|
289
|
+
}
|
|
290
|
+
consola.start(`Installing dependencies using ${packageManager}`);
|
|
291
|
+
const proc = spawnSync(command, {
|
|
292
|
+
cwd: target,
|
|
293
|
+
shell: true,
|
|
294
|
+
stdio: "inherit"
|
|
295
|
+
});
|
|
296
|
+
if (proc.status !== 0) {
|
|
297
|
+
throw new Error(`Failed to install dependencies`);
|
|
298
|
+
}
|
|
299
|
+
consola.success(`Dependencies installed`);
|
|
300
|
+
};
|
|
301
|
+
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);
|
|
302
|
+
program.parse();
|
|
303
|
+
|
|
304
|
+
//# debugId=DACB758B74278AF064756E2164756E21
|
|
305
|
+
//# 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 {version} from '../package.json'\nimport consola from 'consola'\nimport {input, select, confirm} from '@inquirer/prompts'\nimport path from 'path'\nimport * as fs from 'fs'\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 (\n        fs.statSync(filePath).isDirectory() &&\n        !filePath.includes('node_modules')\n      ) {\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    const 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    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    const 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    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    if (client) {\n      let clientRoot: string = ''\n      let clientPath: string = ''\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      const 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    consola.box(`Pylon successfully created in ${target}.\\n\\nHappy coding!`)\n  } catch (e) {\n    consola.error(e)\n  }\n}\n\nprogram.parse()\n"
  ],
  "mappings": ";;;;AAEA;;;;;AAEA;AACA;AACA;AACA;AAEA,SAAS,MAAM,CAAC,KAAa;AAC3B,MAAI;AACF,IAAG,aAAU,KAAK,EAAC,WAAW,KAAI,CAAC;AAAA,WAC5B,GAAP;AACA,QAAI,aAAa,OAAO;AACtB,UAAI,UAAU,KAAK,EAAE,SAAS;AAAU;AAAA,IAC1C;AACA,UAAM;AAAA;AAAA;AAkLV;AACA;AA4FA,eAAe,IAAI,CACjB,WACA,SACA,SACA;AACA,MAAI;AACF,YAAQ,IAAI,GAAG,QAAQ,KAAK,aAAa,QAAQ,QAAQ,GAAG;AAE5D;AAAA,MACE,SAAS;AAAA,MACT,SAAS;AAAA,MACT,UAAU;AAAA,MACV,gBAAgB;AAAA,MAChB,QAAQ;AAAA,MACR,YAAY;AAAA,MACZ,YAAY;AAAA,QACV;AAEJ,QAAI,SAAS;AAEb,QAAI,WAAW;AACb,eAAS;AAET,cAAQ,QAAQ,iCAA2B,QAAQ;AAAA,IACrD,OAAO;AACL,YAAM,SAAS,MAAM,MAAM;AAAA,QACzB,SAAS;AAAA,QACT,SAAS;AAAA,MACX,CAAC;AACD,eAAS;AAAA;AAGX,QAAI,cAAc;AAElB,QAAI,WAAW,KAAK;AAClB,oBAAc,KAAK,SAAS,QAAQ,IAAI,CAAC;AAAA,IAC3C,OAAO;AACL,oBAAc,KAAK,SAAS,MAAM;AAAA;AAGpC,UAAM,cACJ,cACC,MAAM,OAAO;AAAA,MACZ,SAAS;AAAA,MACT,SAAS,SAAS,IAAI,eAAY;AAAA,QAChC,MAAM,GAAG,SAAQ,SAAS,SAAQ;AAAA,QAClC,OAAO,SAAQ;AAAA,MACjB,EAAE;AAAA,MACF,SAAS;AAAA,IACX,CAAC;AAEH,SAAK,aAAa;AAChB,YAAM,IAAI,MAAM,qBAAqB;AAAA,IACvC;AAEA,UAAM,UAAU,SAAS,KAAK,GAAE,UAAS,QAAQ,WAAW;AAE5D,SAAK,SAAS;AACZ,YAAM,IAAI,MAAM,6BAA6B,aAAa;AAAA,IAC5D;AAEA,UAAM,eACJ,eACC,MAAM,OAAO;AAAA,MACZ,SAAS;AAAA,MACT,SAAS,UACN,OAAO,cAAY,QAAQ,WAAW,SAAS,SAAS,GAAG,CAAC,EAC5D,IAAI,eAAa;AAAA,QAChB,MAAM,SAAS;AAAA,QACf,OAAO,SAAS;AAAA,MAClB,EAAE;AAAA,MACJ,SAAS;AAAA,IACX,CAAC;AAEH,SAAK,cAAc;AACjB,YAAM,IAAI,MAAM,sBAAsB;AAAA,IACxC;AAEA,QAAO,cAAW,MAAM,GAAG;AACzB,UAAO,eAAY,MAAM,EAAE,SAAS,GAAG;AACrC,cAAM,WAAW,MAAM,QAAQ;AAAA,UAC7B,SAAS;AAAA,UACT,SAAS;AAAA,QACX,CAAC;AACD,aAAK,UAAU;AACb,kBAAQ,KAAK,CAAC;AAAA,QAChB;AAAA,MACF;AAAA,IACF,OAAO;AACL,aAAO,MAAM;AAAA;AAGf,UAAM,UACJ,cACC,MAAM,QAAQ,EAAC,SAAS,0CAAyC,CAAC;AAErE,UAAM,eAAe;AAAA,MACnB,MAAM;AAAA,MACN,SAAS;AAAA,MACT,UAAU;AAAA,MACV;AAAA,IACF,CAAC;AAED,QAAI,iBAAiB;AAErB,QAAI,gBAAgB,UAAU,gBAAgB;AAC5C,uBAAiB;AAAA,IACnB;AAEA,QAAI,SAAS;AACX,YAAM,oBAAoB,EAAC,QAAQ,eAAc,CAAC;AAAA,IACpD;AAEA,UAAM,SACJ,aACC,MAAM,QAAQ;AAAA,MACb,SACE;AAAA,MACF,SAAS;AAAA,IACX,CAAC;AAEH,QAAI,QAAQ;AACV,UAAI,aAAqB;AACzB,UAAI,aAAqB;AACzB,WAAK,eAAe;AAClB,qBAAa,MAAM,MAAM;AAAA,UACvB,SAAS;AAAA,UACT,SAAS;AAAA,QACX,CAAC;AAED,qBAAa,MAAM,MAAM;AAAA,UACvB,SAAS;AAAA,UACT,SAAS,KAAK,KAAK,YAAY,eAAe;AAAA,UAC9C,UAAU,WAAS;AAEjB,iBAAK,MAAM,WAAW,eAAe,MAAM,KAAK,UAAU,GAAG;AAC3D,qBAAO;AAAA,YACT;AAEA,mBAAO;AAAA;AAAA,QAEX,CAAC;AAAA,MACH;AAEA,YAAM,aACJ,iBACC,MAAM,MAAM;AAAA,QACX,SAAS;AAAA,QACT,SAAS;AAAA,MACX,CAAC;AAEH,cAAQ,MAAM,8CAA8C;AAE5D,YAAM,gBAAgB,KAAK,KAAK,QAAQ,cAAc;AAEtD,YAAM,YAAY,KAAK,MAAS,gBAAa,eAAe,OAAO,CAAC;AAEpE,gBAAU,UAAU;AAAA,WACf,UAAU;AAAA,QACb,KACE,UAAU,QAAQ,MAClB,2BAA2B,4BAA4B;AAAA,MAC3D;AAEA,MAAG,iBAAc,eAAe,KAAK,UAAU,WAAW,MAAM,CAAC,CAAC;AAElE,cAAQ,QAAQ,0BAA0B;AAAA,IAC5C;AAEA,YAAQ,IAAI,iCAAiC,0BAA0B;AAAA,WAChE,GAAP;AACA,YAAQ,MAAM,CAAC;AAAA;AAAA;AAAA;AAtbnB,IAAM,WAKA;AAAA,EACJ;AAAA,IACE,KAAK;AAAA,IACL,MAAM;AAAA,IACN,SAAS;AAAA,IACT,WAAW,CAAC,SAAS;AAAA,EACvB;AAAA,EACA;AAAA,IACE,KAAK;AAAA,IACL,MAAM;AAAA,IACN,SAAS;AAAA,IACT,WAAW,CAAC,SAAS;AAAA,EACvB;AAAA,EACA;AAAA,IACE,KAAK;AAAA,IACL,MAAM;AAAA,IACN,SAAS;AAAA,IACT,WAAW,CAAC,SAAS;AAAA,EACvB;AACF;AAEA,IAAM,YAIA;AAAA,EACJ;AAAA,IACE,KAAK;AAAA,IACL,MAAM;AAAA,IACN,aAAa;AAAA,EACf;AAAA,EACA;AAAA,IACE,KAAK;AAAA,IACL,MAAM;AAAA,IACN,aAAa;AAAA,EACf;AACF;AAEA,IAAM,2BAA2B,CAC/B,SACA,cACG;AACH,MAAI,SAAS;AAEb,SAAO,QAAQ,SAAS,EAAE,QAAQ,EAAE,KAAK,WAAW;AAClD,aAAS,OAAO,WAAW,KAAK,KAAK;AAAA,GACtC;AAED,SAAO;AAAA;AAET,IAAM,4BAA4B,CAAC,QAA0B;AAC3D,QAAM,MAAM,CAAC,SAA0B;AACrC,UAAM,SAAmB,CAAC;AAE1B,UAAM,QAAW,eAAY,IAAG;AAEhC,UAAM,QAAQ,UAAQ;AACpB,YAAM,WAAW,KAAK,KAAK,MAAK,IAAI;AAEpC,UACK,YAAS,QAAQ,EAAE,YAAY,MACjC,SAAS,SAAS,cAAc,GACjC;AACA,eAAO,KAAK,GAAG,IAAI,QAAQ,CAAC;AAAA,MAC9B;AAGA,UAAO,YAAS,QAAQ,EAAE,OAAO,GAAG;AAClC,eAAO,KAAK,QAAQ;AAAA,MACtB;AAAA,KACD;AAED,WAAO;AAAA;AAGT,SAAO,IAAI,GAAG,EAAE,IAAI,UAAQ;AAC1B,WAAO,KAAK,QAAQ,KAAK,GAAG;AAAA,GAC7B;AAAA;AAGH,IAAM,iBAAiB,OAAO,YAKxB;AACJ,UAAO,SAAS,UAAU,WAAU;AAEpC,QAAM,cAAc,SAAS,KAAK,GAAE,UAAS,QAAQ,OAAO,GAAG;AAC/D,QAAM,eAAe,UAAU,KAAK,GAAE,UAAS,QAAQ,QAAQ,GAAG;AAElE,OAAK,aAAa;AAChB,UAAM,IAAI,MAAM,oBAAoB,SAAS;AAAA,EAC/C;AAEA,OAAK,cAAc;AACjB,UAAM,IAAI,MAAM,qBAAqB,UAAU;AAAA,EACjD;AAGA,QAAM,oBAAoB,KAAK,KAAK,WAAW,MAAM,aAAa,QAAQ;AAE1E,OAAQ,cAAW,iBAAiB,GAAG;AACrC,UAAM,IAAI,MAAM,+BAA+B,mBAAmB;AAAA,EACpE;AAEA,QAAM,cAAc,KAAK,KAAK,WAAW,MAAM,aAAa,SAAS,QAAQ;AAE7E,OAAQ,cAAW,WAAW,GAAG;AAC/B,UAAM,IAAI,MAAM,uBAAuB,aAAa;AAAA,EACtD;AAGA,QAAM,sBAAsB,KAAK,KAAK,QAAQ,IAAI,GAAG,MAAM;AAE3D,UAAQ,MAAM,qBAAqB,qBAAqB;AAExD,QAAM,SAAS,CAAC,YAAoB;AAClC,WAAO,yBAAyB,SAAS;AAAA,MACvC,gBAAgB,QAAQ;AAAA,IAC1B,CAAC;AAAA;AAIH,4BAA0B,iBAAiB,EAAE,QAAQ,UAAQ;AAC3D,UAAM,SAAS,KAAK,KAAK,mBAAmB,IAAI;AAChD,UAAM,UAAS,KAAK,KAAK,qBAAqB,IAAI;AAIlD,UAAM,YAAY,KAAK,QAAQ,OAAM;AAGrC,QACE,YAAY,gBACZ,OAAO,SAAS,gCAAgC,GAChD;AACA;AAAA,IACF;AAEA,SAAQ,cAAW,SAAS,GAAG;AAC7B,MAAG,aAAU,WAAW,EAAC,WAAW,KAAI,CAAC;AAAA,IAC3C;AAEA,UAAM,kBAAkB,OAAU,gBAAa,QAAQ,OAAO,CAAC;AAE/D,IAAG,iBAAc,SAAQ,eAAe;AAAA,GACzC;AAGD,4BAA0B,WAAW,EAAE,QAAQ,UAAQ;AACrD,UAAM,SAAS,KAAK,KAAK,aAAa,IAAI;AAC1C,UAAM,UAAS,KAAK,KAAK,qBAAqB,IAAI;AAGlD,UAAM,YAAY,KAAK,QAAQ,OAAM;AAErC,SAAQ,cAAW,SAAS,GAAG;AAC7B,MAAG,aAAU,WAAW,EAAC,WAAW,KAAI,CAAC;AAAA,IAC3C;AAEA,UAAM,kBAAkB,OAAU,gBAAa,QAAQ,OAAO,CAAC;AAE/D,IAAG,iBAAc,SAAQ,eAAe;AAAA,GACzC;AAED,UAAQ,QAAQ,eAAe;AAAA;AAMjC,IAAM,sBAAsB,OAAO,SAG7B;AACJ,QAAM,SAAS,KAAK,QAAQ,KAAK,MAAM;AAEvC,UAAQ,IAAI,UAAU,MAAM;AAE5B,OAAK,KAAK,gBAAgB;AACxB,SAAK,iBAAiB,MAAM,OAAO;AAAA,MACjC,KAAK;AAAA,MACL,kBAAkB;AAAA,IACpB,CAAC;AAAA,EACH;AAEA,OAAK,KAAK,gBAAgB;AACxB,UAAM,IAAI,MAAM,0BAA0B;AAAA,EAC5C;AAGA,UAAO,mBAAkB;AAEzB,MAAI,UAAU;AAEd,UAAQ;AAAA,SACD;AACH,gBAAU;AACV;AAAA,SACG;AACH,gBAAU;AACV;AAAA,SACG;AACH,gBAAU;AACV;AAAA,SACG;AACH,gBAAU;AACV;AAAA;AAEA,YAAM,IAAI,MAAM,4BAA4B,gBAAgB;AAAA;AAGhE,UAAQ,MAAM,iCAAiC,gBAAgB;AAE/D,QAAM,OAAO,UAAU,SAAS;AAAA,IAC9B,KAAK;AAAA,IACL,OAAO;AAAA,IACP,OAAO;AAAA,EACT,CAAC;AAED,MAAI,KAAK,WAAW,GAAG;AACrB,UAAM,IAAI,MAAM,gCAAgC;AAAA,EAClD;AAEA,UAAQ,QAAQ,wBAAwB;AAAA;AAG1C,QACG,KAAK,cAAc,EACnB,QAAQ,OAAO,EACf,UAAU,UAAU,EACpB,UAAU,IAAI,OAAO,iBAAiB,sBAAsB,CAAC,EAC7D,UACC,IAAI,OAAO,2BAA2B,SAAS,EAAE,QAC/C,SAAS,IAAI,GAAE,UAAS,GAAG,CAC7B,CACF,EACC,UAAU,IAAI,OAAO,6BAA6B,UAAU,CAAC,EAC7D,UACC,IAAI,OAAO,2CAA2C,iBAAiB,CACzE,EACC,UACC,IAAI,OACF,YACA,2EACF,CACF,EACC,UAAU,IAAI,OAAO,8BAA8B,aAAa,CAAC,EACjE,UAAU,IAAI,OAAO,8BAA8B,aAAa,CAAC,EACjE,OAAO,IAAI;AA2Ld,QAAQ,MAAM;",
  "debugId": "DACB758B74278AF064756E2164756E21",
  "names": []
}
|
package/package.json
ADDED
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
{
|
|
2
|
+
"name": "create-pylon",
|
|
3
|
+
"version": "0.0.1",
|
|
4
|
+
"description": "CLI for creating a Pylon",
|
|
5
|
+
"scripts": {
|
|
6
|
+
"build": "bun build ./src/index.ts --target=bun --outdir=./dist --sourcemap=inline --packages external"
|
|
7
|
+
},
|
|
8
|
+
"bin": "./dist/index.js",
|
|
9
|
+
"files": [
|
|
10
|
+
"dist",
|
|
11
|
+
"templates"
|
|
12
|
+
],
|
|
13
|
+
"author": "Nico Schett <nico.schett@cronit.io>",
|
|
14
|
+
"license": "Apache-2.0",
|
|
15
|
+
"repository": {
|
|
16
|
+
"type": "git",
|
|
17
|
+
"url": "https://github.com/getcronit/pylon.git",
|
|
18
|
+
"directory": "packages/create-pylon"
|
|
19
|
+
},
|
|
20
|
+
"homepage": "https://pylon.cronit.io",
|
|
21
|
+
"dependencies": {
|
|
22
|
+
"@inquirer/prompts": "^5.4.0",
|
|
23
|
+
"commander": "^12.1.0",
|
|
24
|
+
"consola": "^3.2.3",
|
|
25
|
+
"detect-package-manager": "^3.0.2"
|
|
26
|
+
}
|
|
27
|
+
}
|
|
@@ -0,0 +1,47 @@
|
|
|
1
|
+
# use the official Bun image
|
|
2
|
+
# see all versions at https://hub.docker.com/r/oven/bun/tags
|
|
3
|
+
FROM oven/bun:1 as base
|
|
4
|
+
|
|
5
|
+
LABEL description="Offical docker image for Pylon services (Bun)"
|
|
6
|
+
LABEL org.opencontainers.image.source="https://github.com/getcronit/pylon"
|
|
7
|
+
LABEL maintainer="office@cronit.io"
|
|
8
|
+
|
|
9
|
+
WORKDIR /usr/src/pylon
|
|
10
|
+
|
|
11
|
+
|
|
12
|
+
# install dependencies into temp directory
|
|
13
|
+
# this will cache them and speed up future builds
|
|
14
|
+
FROM base AS install
|
|
15
|
+
RUN mkdir -p /temp/dev
|
|
16
|
+
COPY package.json bun.lockb /temp/dev/
|
|
17
|
+
RUN cd /temp/dev && bun install --frozen-lockfile
|
|
18
|
+
|
|
19
|
+
# install with --production (exclude devDependencies)
|
|
20
|
+
RUN mkdir -p /temp/prod
|
|
21
|
+
COPY package.json bun.lockb /temp/prod/
|
|
22
|
+
RUN cd /temp/prod && bun install --frozen-lockfile --production
|
|
23
|
+
|
|
24
|
+
# copy node_modules from temp directory
|
|
25
|
+
# then copy all (non-ignored) project files into the image
|
|
26
|
+
FROM install AS prerelease
|
|
27
|
+
COPY --from=install /temp/dev/node_modules node_modules
|
|
28
|
+
COPY . .
|
|
29
|
+
|
|
30
|
+
# [optional] tests & build
|
|
31
|
+
ENV NODE_ENV=production
|
|
32
|
+
|
|
33
|
+
# Create .pylon folder (mkdir)
|
|
34
|
+
RUN mkdir -p .pylon
|
|
35
|
+
# RUN bun test
|
|
36
|
+
RUN bun run pylon build
|
|
37
|
+
|
|
38
|
+
# copy production dependencies and source code into final image
|
|
39
|
+
FROM base AS release
|
|
40
|
+
COPY --from=install /temp/prod/node_modules node_modules
|
|
41
|
+
COPY --from=prerelease /usr/src/pylon/.pylon .pylon
|
|
42
|
+
COPY --from=prerelease /usr/src/pylon/package.json .
|
|
43
|
+
|
|
44
|
+
# run the app
|
|
45
|
+
USER bun
|
|
46
|
+
EXPOSE 3000/tcp
|
|
47
|
+
ENTRYPOINT [ "bun", "run", "/usr/src/pylon/.pylon/index.js" ]
|
|
@@ -0,0 +1,23 @@
|
|
|
1
|
+
{
|
|
2
|
+
"name": "__PYLON_NAME__",
|
|
3
|
+
"private": true,
|
|
4
|
+
"version": "0.0.1",
|
|
5
|
+
"type": "module",
|
|
6
|
+
"description": "Generated with `npm create pylon`",
|
|
7
|
+
"scripts": {
|
|
8
|
+
"dev": "pylon dev -c 'node --enable-source-maps .pylon/index.js'",
|
|
9
|
+
"build": "pylon build"
|
|
10
|
+
},
|
|
11
|
+
"dependencies": {
|
|
12
|
+
"@getcronit/pylon": "*",
|
|
13
|
+
"bun-types": "^1.1.18"
|
|
14
|
+
},
|
|
15
|
+
"devDependencies": {
|
|
16
|
+
"@getcronit/pylon-dev": "*"
|
|
17
|
+
},
|
|
18
|
+
"repository": {
|
|
19
|
+
"type": "git",
|
|
20
|
+
"url": "https://github.com/getcronit/pylon.git"
|
|
21
|
+
},
|
|
22
|
+
"homepage": "https://pylon.cronit.io"
|
|
23
|
+
}
|
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
{
|
|
2
|
+
"name": "__PYLON_NAME__",
|
|
3
|
+
"version": "0.0.0",
|
|
4
|
+
"private": true,
|
|
5
|
+
"scripts": {
|
|
6
|
+
"deploy": "pylon build && wrangler deploy",
|
|
7
|
+
"dev": "pylon dev -c 'wrangler dev'",
|
|
8
|
+
"cf-typegen": "wrangler types"
|
|
9
|
+
},
|
|
10
|
+
"dependencies": {
|
|
11
|
+
"@getcronit/pylon": "*"
|
|
12
|
+
},
|
|
13
|
+
"devDependencies": {
|
|
14
|
+
"@getcronit/pylon-dev": "*",
|
|
15
|
+
"@cloudflare/vitest-pool-workers": "^0.4.5",
|
|
16
|
+
"@cloudflare/workers-types": "^4.20240903.0",
|
|
17
|
+
"typescript": "^5.5.2",
|
|
18
|
+
"wrangler": "^3.60.3"
|
|
19
|
+
},
|
|
20
|
+
"repository": {
|
|
21
|
+
"type": "git",
|
|
22
|
+
"url": "https://github.com/getcronit/pylon.git"
|
|
23
|
+
},
|
|
24
|
+
"homepage": "https://pylon.cronit.io"
|
|
25
|
+
}
|
|
@@ -0,0 +1,108 @@
|
|
|
1
|
+
#:schema node_modules/wrangler/config-schema.json
|
|
2
|
+
name = "__PYLON_NAME__"
|
|
3
|
+
main = ".pylon/index.js"
|
|
4
|
+
compatibility_date = "2024-09-03"
|
|
5
|
+
compatibility_flags = ["nodejs_compat_v2"]
|
|
6
|
+
|
|
7
|
+
# Automatically place your workloads in an optimal location to minimize latency.
|
|
8
|
+
# If you are running back-end logic in a Worker, running it closer to your back-end infrastructure
|
|
9
|
+
# rather than the end user may result in better performance.
|
|
10
|
+
# Docs: https://developers.cloudflare.com/workers/configuration/smart-placement/#smart-placement
|
|
11
|
+
# [placement]
|
|
12
|
+
# mode = "smart"
|
|
13
|
+
|
|
14
|
+
# Variable bindings. These are arbitrary, plaintext strings (similar to environment variables)
|
|
15
|
+
# Docs:
|
|
16
|
+
# - https://developers.cloudflare.com/workers/wrangler/configuration/#environment-variables
|
|
17
|
+
# Note: Use secrets to store sensitive data.
|
|
18
|
+
# - https://developers.cloudflare.com/workers/configuration/secrets/
|
|
19
|
+
# [vars]
|
|
20
|
+
# MY_VARIABLE = "production_value"
|
|
21
|
+
|
|
22
|
+
# Bind the Workers AI model catalog. Run machine learning models, powered by serverless GPUs, on Cloudflare’s global network
|
|
23
|
+
# Docs: https://developers.cloudflare.com/workers/wrangler/configuration/#workers-ai
|
|
24
|
+
# [ai]
|
|
25
|
+
# binding = "AI"
|
|
26
|
+
|
|
27
|
+
# Bind an Analytics Engine dataset. Use Analytics Engine to write analytics within your Pages Function.
|
|
28
|
+
# Docs: https://developers.cloudflare.com/workers/wrangler/configuration/#analytics-engine-datasets
|
|
29
|
+
# [[analytics_engine_datasets]]
|
|
30
|
+
# binding = "MY_DATASET"
|
|
31
|
+
|
|
32
|
+
# Bind a headless browser instance running on Cloudflare's global network.
|
|
33
|
+
# Docs: https://developers.cloudflare.com/workers/wrangler/configuration/#browser-rendering
|
|
34
|
+
# [browser]
|
|
35
|
+
# binding = "MY_BROWSER"
|
|
36
|
+
|
|
37
|
+
# Bind a D1 database. D1 is Cloudflare’s native serverless SQL database.
|
|
38
|
+
# Docs: https://developers.cloudflare.com/workers/wrangler/configuration/#d1-databases
|
|
39
|
+
# [[d1_databases]]
|
|
40
|
+
# binding = "MY_DB"
|
|
41
|
+
# database_name = "my-database"
|
|
42
|
+
# database_id = "xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx"
|
|
43
|
+
|
|
44
|
+
# Bind a dispatch namespace. Use Workers for Platforms to deploy serverless functions programmatically on behalf of your customers.
|
|
45
|
+
# Docs: https://developers.cloudflare.com/workers/wrangler/configuration/#dispatch-namespace-bindings-workers-for-platforms
|
|
46
|
+
# [[dispatch_namespaces]]
|
|
47
|
+
# binding = "MY_DISPATCHER"
|
|
48
|
+
# namespace = "my-namespace"
|
|
49
|
+
|
|
50
|
+
# Bind a Durable Object. Durable objects are a scale-to-zero compute primitive based on the actor model.
|
|
51
|
+
# Durable Objects can live for as long as needed. Use these when you need a long-running "server", such as in realtime apps.
|
|
52
|
+
# Docs: https://developers.cloudflare.com/workers/wrangler/configuration/#durable-objects
|
|
53
|
+
# [[durable_objects.bindings]]
|
|
54
|
+
# name = "MY_DURABLE_OBJECT"
|
|
55
|
+
# class_name = "MyDurableObject"
|
|
56
|
+
|
|
57
|
+
# Durable Object migrations.
|
|
58
|
+
# Docs: https://developers.cloudflare.com/workers/wrangler/configuration/#migrations
|
|
59
|
+
# [[migrations]]
|
|
60
|
+
# tag = "v1"
|
|
61
|
+
# new_classes = ["MyDurableObject"]
|
|
62
|
+
|
|
63
|
+
# Bind a Hyperdrive configuration. Use to accelerate access to your existing databases from Cloudflare Workers.
|
|
64
|
+
# Docs: https://developers.cloudflare.com/workers/wrangler/configuration/#hyperdrive
|
|
65
|
+
# [[hyperdrive]]
|
|
66
|
+
# binding = "MY_HYPERDRIVE"
|
|
67
|
+
# id = "xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx"
|
|
68
|
+
|
|
69
|
+
# Bind a KV Namespace. Use KV as persistent storage for small key-value pairs.
|
|
70
|
+
# Docs: https://developers.cloudflare.com/workers/wrangler/configuration/#kv-namespaces
|
|
71
|
+
# [[kv_namespaces]]
|
|
72
|
+
# binding = "MY_KV_NAMESPACE"
|
|
73
|
+
# id = "xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx"
|
|
74
|
+
|
|
75
|
+
# Bind an mTLS certificate. Use to present a client certificate when communicating with another service.
|
|
76
|
+
# Docs: https://developers.cloudflare.com/workers/wrangler/configuration/#mtls-certificates
|
|
77
|
+
# [[mtls_certificates]]
|
|
78
|
+
# binding = "MY_CERTIFICATE"
|
|
79
|
+
# certificate_id = "xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx"
|
|
80
|
+
|
|
81
|
+
# Bind a Queue producer. Use this binding to schedule an arbitrary task that may be processed later by a Queue consumer.
|
|
82
|
+
# Docs: https://developers.cloudflare.com/workers/wrangler/configuration/#queues
|
|
83
|
+
# [[queues.producers]]
|
|
84
|
+
# binding = "MY_QUEUE"
|
|
85
|
+
# queue = "my-queue"
|
|
86
|
+
|
|
87
|
+
# Bind a Queue consumer. Queue Consumers can retrieve tasks scheduled by Producers to act on them.
|
|
88
|
+
# Docs: https://developers.cloudflare.com/workers/wrangler/configuration/#queues
|
|
89
|
+
# [[queues.consumers]]
|
|
90
|
+
# queue = "my-queue"
|
|
91
|
+
|
|
92
|
+
# Bind an R2 Bucket. Use R2 to store arbitrarily large blobs of data, such as files.
|
|
93
|
+
# Docs: https://developers.cloudflare.com/workers/wrangler/configuration/#r2-buckets
|
|
94
|
+
# [[r2_buckets]]
|
|
95
|
+
# binding = "MY_BUCKET"
|
|
96
|
+
# bucket_name = "my-bucket"
|
|
97
|
+
|
|
98
|
+
# Bind another Worker service. Use this binding to call another Worker without network overhead.
|
|
99
|
+
# Docs: https://developers.cloudflare.com/workers/wrangler/configuration/#service-bindings
|
|
100
|
+
# [[services]]
|
|
101
|
+
# binding = "MY_SERVICE"
|
|
102
|
+
# service = "my-service"
|
|
103
|
+
|
|
104
|
+
# Bind a Vectorize index. Use to store and query vector embeddings for semantic search, classification and other vector search use-cases.
|
|
105
|
+
# Docs: https://developers.cloudflare.com/workers/wrangler/configuration/#vectorize-indexes
|
|
106
|
+
# [[vectorize]]
|
|
107
|
+
# binding = "MY_INDEX"
|
|
108
|
+
# index_name = "my-index"
|
|
@@ -0,0 +1,45 @@
|
|
|
1
|
+
# Use the official Node.js 20 image as the base
|
|
2
|
+
FROM node:20-alpine as base
|
|
3
|
+
|
|
4
|
+
LABEL description="Offical docker image for Pylon services (Node.js)"
|
|
5
|
+
LABEL org.opencontainers.image.source="https://github.com/getcronit/pylon"
|
|
6
|
+
LABEL maintainer="office@cronit.io"
|
|
7
|
+
|
|
8
|
+
WORKDIR /usr/src/pylon
|
|
9
|
+
|
|
10
|
+
# install dependencies into a temp directory
|
|
11
|
+
# this will cache them and speed up future builds
|
|
12
|
+
FROM base AS install
|
|
13
|
+
RUN mkdir -p /temp/dev
|
|
14
|
+
COPY package.json package-lock.json /temp/dev/
|
|
15
|
+
RUN cd /temp/dev && npm ci
|
|
16
|
+
|
|
17
|
+
# install with --production (exclude devDependencies)
|
|
18
|
+
RUN mkdir -p /temp/prod
|
|
19
|
+
COPY package.json package-lock.json /temp/prod/
|
|
20
|
+
RUN cd /temp/prod && npm ci --only=production
|
|
21
|
+
|
|
22
|
+
# copy node_modules from temp directory
|
|
23
|
+
# then copy all (non-ignored) project files into the image
|
|
24
|
+
FROM install AS prerelease
|
|
25
|
+
COPY --from=install /temp/dev/node_modules node_modules
|
|
26
|
+
COPY . .
|
|
27
|
+
|
|
28
|
+
# [optional] tests & build
|
|
29
|
+
ENV NODE_ENV=production
|
|
30
|
+
|
|
31
|
+
# Create .pylon folder (mkdir)
|
|
32
|
+
RUN mkdir -p .pylon
|
|
33
|
+
# RUN npm test
|
|
34
|
+
RUN npm run pylon build
|
|
35
|
+
|
|
36
|
+
# copy production dependencies and source code into final image
|
|
37
|
+
FROM base AS release
|
|
38
|
+
COPY --from=install /temp/prod/node_modules node_modules
|
|
39
|
+
COPY --from=prerelease /usr/src/pylon/.pylon .pylon
|
|
40
|
+
COPY --from=prerelease /usr/src/pylon/package.json .
|
|
41
|
+
|
|
42
|
+
# run the app
|
|
43
|
+
USER node
|
|
44
|
+
EXPOSE 3000/tcp
|
|
45
|
+
ENTRYPOINT [ "node", "/usr/src/pylon/.pylon/index.js" ]
|
|
@@ -0,0 +1,22 @@
|
|
|
1
|
+
{
|
|
2
|
+
"name": "__PYLON_NAME__",
|
|
3
|
+
"private": true,
|
|
4
|
+
"version": "0.0.1",
|
|
5
|
+
"type": "module",
|
|
6
|
+
"description": "Generated with `npm create pylon`",
|
|
7
|
+
"scripts": {
|
|
8
|
+
"dev": "pylon dev -c 'node --enable-source-maps .pylon/index.js'",
|
|
9
|
+
"build": "pylon build"
|
|
10
|
+
},
|
|
11
|
+
"dependencies": {
|
|
12
|
+
"@getcronit/pylon": "*"
|
|
13
|
+
},
|
|
14
|
+
"devDependencies": {
|
|
15
|
+
"@getcronit/pylon-dev": "*"
|
|
16
|
+
},
|
|
17
|
+
"repository": {
|
|
18
|
+
"type": "git",
|
|
19
|
+
"url": "https://github.com/getcronit/pylon.git"
|
|
20
|
+
},
|
|
21
|
+
"homepage": "https://pylon.cronit.io"
|
|
22
|
+
}
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
import {app} from '@getcronit/pylon'
|
|
2
|
+
import {serve} from '@hono/node-server'
|
|
3
|
+
|
|
4
|
+
export const graphql = {
|
|
5
|
+
Query: {
|
|
6
|
+
hello: () => {
|
|
7
|
+
return 'Hello, world!'
|
|
8
|
+
}
|
|
9
|
+
},
|
|
10
|
+
Mutation: {}
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
serve(app, info => {
|
|
14
|
+
console.log(`Server running at ${info.port}`)
|
|
15
|
+
})
|
|
@@ -0,0 +1,43 @@
|
|
|
1
|
+
name: publish
|
|
2
|
+
|
|
3
|
+
on: [push]
|
|
4
|
+
env:
|
|
5
|
+
IMAGE_NAME: __PYLON_NAME__
|
|
6
|
+
|
|
7
|
+
jobs:
|
|
8
|
+
# Push image to GitHub Packages.
|
|
9
|
+
# See also https://docs.docker.com/docker-hub/builds/
|
|
10
|
+
publish-container:
|
|
11
|
+
runs-on: ubuntu-latest
|
|
12
|
+
permissions:
|
|
13
|
+
packages: write
|
|
14
|
+
contents: read
|
|
15
|
+
|
|
16
|
+
steps:
|
|
17
|
+
- uses: actions/checkout@v2
|
|
18
|
+
|
|
19
|
+
- name: Build image
|
|
20
|
+
run: docker build . --file Dockerfile --tag $IMAGE_NAME
|
|
21
|
+
|
|
22
|
+
- name: Log into registry
|
|
23
|
+
run: echo "${{ secrets.GITHUB_TOKEN }}" | docker login ghcr.io -u ${{ github.actor }} --password-stdin
|
|
24
|
+
|
|
25
|
+
- name: Push image
|
|
26
|
+
run: |
|
|
27
|
+
IMAGE_ID=ghcr.io/${{ github.repository_owner }}/$IMAGE_NAME
|
|
28
|
+
|
|
29
|
+
# Change all uppercase to lowercase
|
|
30
|
+
IMAGE_ID=$(echo $IMAGE_ID | tr '[A-Z]' '[a-z]')
|
|
31
|
+
# Strip git ref prefix from version
|
|
32
|
+
VERSION=$(echo "${{ github.ref }}" | sed -e 's,.*/\(.*\),\1,')
|
|
33
|
+
# Strip "v" prefix from tag name
|
|
34
|
+
[[ "${{ github.ref }}" == "refs/tags/"* ]] && VERSION=$(echo $VERSION | sed -e 's/^v//')
|
|
35
|
+
# Use Docker `latest` tag convention
|
|
36
|
+
[ "$VERSION" == "main" ] && VERSION=latest
|
|
37
|
+
echo IMAGE_ID=$IMAGE_ID
|
|
38
|
+
echo VERSION=$VERSION
|
|
39
|
+
docker tag $IMAGE_NAME $IMAGE_ID:$VERSION
|
|
40
|
+
docker push $IMAGE_ID:$VERSION
|
|
41
|
+
|
|
42
|
+
# SPDX-License-Identifier: (EUPL-1.2)
|
|
43
|
+
# Copyright © 2024 cronit KG
|