create-atom.io 0.1.0 → 0.1.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/create-atom.js
CHANGED
|
@@ -1,3 +1,134 @@
|
|
|
1
|
-
import {
|
|
1
|
+
import { existsSync, promises } from "node:fs";
|
|
2
|
+
import { resolve } from "node:path";
|
|
3
|
+
import * as prompts from "@clack/prompts";
|
|
4
|
+
import { getPackageInfo } from "local-pkg";
|
|
5
|
+
import picocolors from "picocolors";
|
|
6
|
+
import { x } from "tinyexec";
|
|
2
7
|
|
|
3
|
-
|
|
8
|
+
//#region src/create-atom.ts
|
|
9
|
+
const pico = picocolors.createColors(true);
|
|
10
|
+
const s = prompts.spinner();
|
|
11
|
+
async function createAtom(argDir, options) {
|
|
12
|
+
const skipHint = options.skipHints ?? false;
|
|
13
|
+
const packageManager = options.packageManager ?? getPkgManager();
|
|
14
|
+
prompts.intro(pico.greenBright(`atom.io - Data Components for TypeScript`));
|
|
15
|
+
const { dir, templateName } = await prompts.group({
|
|
16
|
+
templateName: () => prompts.select({
|
|
17
|
+
message: `Template:`,
|
|
18
|
+
initialValue: `preact-svg-editor`,
|
|
19
|
+
options: [{
|
|
20
|
+
label: `Preact SVG Editor`,
|
|
21
|
+
value: `preact-svg-editor`
|
|
22
|
+
}, {
|
|
23
|
+
label: `React Node Backend`,
|
|
24
|
+
value: `react-node-backend`
|
|
25
|
+
}]
|
|
26
|
+
}),
|
|
27
|
+
dir: () => argDir ? Promise.resolve(argDir) : prompts.text({
|
|
28
|
+
message: `Project directory:`,
|
|
29
|
+
placeholder: `my-app`,
|
|
30
|
+
validate(value) {
|
|
31
|
+
if (value === void 0 || value.length === 0) return `Directory name is required!`;
|
|
32
|
+
if (existsSync(value)) return `Refusing to overwrite existing directory or file! Please provide a non-clashing name.`;
|
|
33
|
+
}
|
|
34
|
+
})
|
|
35
|
+
}, { onCancel: () => {
|
|
36
|
+
prompts.cancel(pico.yellow(`Cancelled`));
|
|
37
|
+
process.exit(0);
|
|
38
|
+
} });
|
|
39
|
+
const targetDir = resolve(process.cwd(), dir);
|
|
40
|
+
const opts = {
|
|
41
|
+
packageManager,
|
|
42
|
+
templateName
|
|
43
|
+
};
|
|
44
|
+
await useSpinner(`Setting up your project directory...`, () => scaffold(targetDir, opts), `Set up project directory`);
|
|
45
|
+
await useSpinner(`Installing project dependencies...`, () => installDeps(targetDir, opts), `Installed project dependencies`);
|
|
46
|
+
if (skipHint === false) {
|
|
47
|
+
const gettingStarted = `
|
|
48
|
+
${pico.dim(`$`)} ${pico.blueBright(`cd ${dir}`)}
|
|
49
|
+
${pico.dim(`$`)} ${pico.blueBright(`${packageManager === `npm` ? `npm run` : packageManager === `bun` ? `bun run` : packageManager} dev`)}
|
|
50
|
+
`;
|
|
51
|
+
prompts.note(gettingStarted.trim().replace(/^\t\t\t/gm, ``), `Getting Started`);
|
|
52
|
+
}
|
|
53
|
+
prompts.outro(pico.green(`You're all set!`));
|
|
54
|
+
}
|
|
55
|
+
async function useSpinner(startMessage, fn, finishMessage) {
|
|
56
|
+
s.start(startMessage);
|
|
57
|
+
await fn();
|
|
58
|
+
s.stop(pico.green(finishMessage));
|
|
59
|
+
}
|
|
60
|
+
async function scaffold(to, opts) {
|
|
61
|
+
await promises.mkdir(to, { recursive: true });
|
|
62
|
+
const templateInfo = await getPackageInfo(`@atom.io/template-${opts.templateName}`, { paths: [process.cwd(), import.meta.dirname] });
|
|
63
|
+
if (!templateInfo) throw new Error(`Could not find template package`);
|
|
64
|
+
const { rootPath } = templateInfo;
|
|
65
|
+
await templateDir(rootPath, to, opts);
|
|
66
|
+
const nodeDirPath = resolve(to, `node`);
|
|
67
|
+
try {
|
|
68
|
+
await promises.access(nodeDirPath);
|
|
69
|
+
} catch {
|
|
70
|
+
return;
|
|
71
|
+
}
|
|
72
|
+
if (!(await promises.stat(nodeDirPath)).isDirectory()) return;
|
|
73
|
+
const nodeFiles = await promises.readdir(nodeDirPath, { withFileTypes: true });
|
|
74
|
+
await Promise.all(nodeFiles.map(async (dirent) => {
|
|
75
|
+
if (!dirent.isFile()) return;
|
|
76
|
+
const filename = resolve(nodeDirPath, dirent.name);
|
|
77
|
+
await promises.chmod(filename, 493);
|
|
78
|
+
}));
|
|
79
|
+
}
|
|
80
|
+
/**
|
|
81
|
+
* Recursive fs copy, swiped from `create-wmr`:
|
|
82
|
+
* https://github.com/preactjs/wmr/blob/3c5672ecd2f958c8eaf372d33c084dc69228ae3f/packages/create-wmr/src/index.js#L108-L124
|
|
83
|
+
*/
|
|
84
|
+
async function templateDir(from, to, opts) {
|
|
85
|
+
const files = await promises.readdir(from);
|
|
86
|
+
return (await Promise.all(files.map(async (f) => {
|
|
87
|
+
if (f === `.` || f === `..`) return;
|
|
88
|
+
const filename = resolve(from, f);
|
|
89
|
+
if ((await promises.stat(filename)).isDirectory()) {
|
|
90
|
+
await promises.mkdir(resolve(to, f), { recursive: true });
|
|
91
|
+
return templateDir(filename, resolve(to, f), opts);
|
|
92
|
+
}
|
|
93
|
+
if (opts.packageManager !== `npm` && f === `README.md`) {
|
|
94
|
+
await promises.writeFile(resolve(to, f), (await promises.readFile(filename, `utf-8`)).replace(/npm run/g, opts.packageManager === `bun` ? `bun run` : opts.packageManager));
|
|
95
|
+
return;
|
|
96
|
+
}
|
|
97
|
+
if (f === `_gitignore`) f = `.gitignore`;
|
|
98
|
+
await promises.copyFile(filename, resolve(to, f));
|
|
99
|
+
}))).flat(99);
|
|
100
|
+
}
|
|
101
|
+
async function installDeps(to, opts) {
|
|
102
|
+
const dependencies = [];
|
|
103
|
+
const devDependencies = [];
|
|
104
|
+
const installOpts = {
|
|
105
|
+
packageManager: opts.packageManager,
|
|
106
|
+
to
|
|
107
|
+
};
|
|
108
|
+
await installPackages(dependencies, { ...installOpts });
|
|
109
|
+
devDependencies.length && await installPackages(devDependencies, {
|
|
110
|
+
...installOpts,
|
|
111
|
+
dev: true
|
|
112
|
+
});
|
|
113
|
+
}
|
|
114
|
+
function installPackages(pkgs, opts) {
|
|
115
|
+
return x(opts.packageManager, [
|
|
116
|
+
opts.packageManager === `yarn` ? pkgs.length ? `add` : `` : `install`,
|
|
117
|
+
opts.dev ? `-D` : ``,
|
|
118
|
+
...pkgs
|
|
119
|
+
].filter(Boolean), { nodeOptions: {
|
|
120
|
+
stdio: `ignore`,
|
|
121
|
+
cwd: opts.to
|
|
122
|
+
} });
|
|
123
|
+
}
|
|
124
|
+
function getPkgManager() {
|
|
125
|
+
const userAgent = process.env[`npm_config_user_agent`] ?? ``;
|
|
126
|
+
if (userAgent.startsWith(`yarn`)) return `yarn`;
|
|
127
|
+
if (userAgent.startsWith(`pnpm`)) return `pnpm`;
|
|
128
|
+
if (userAgent.startsWith(`bun`)) return `bun`;
|
|
129
|
+
return `npm`;
|
|
130
|
+
}
|
|
131
|
+
|
|
132
|
+
//#endregion
|
|
133
|
+
export { createAtom };
|
|
134
|
+
//# sourceMappingURL=create-atom.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"create-atom.js","names":["fs"],"sources":["../src/create-atom.ts"],"sourcesContent":["import { existsSync, promises as fs } from \"node:fs\"\nimport { resolve } from \"node:path\"\n\nimport * as prompts from \"@clack/prompts\"\nimport { getPackageInfo } from \"local-pkg\"\nimport picocolors from \"picocolors\"\nimport type { Colors } from \"picocolors/types\"\nimport { x } from \"tinyexec\"\n\nconst pico: Colors = picocolors.createColors(true)\n\nconst s = prompts.spinner()\n\nexport type PackageManager = `bun` | `npm` | `pnpm` | `yarn`\nexport type TemplateName = `preact-svg-editor` | `react-node-backend`\n\nexport type CreateAtomOptions = {\n\tpackageManager: PackageManager\n\ttemplateName: TemplateName\n}\n\nexport type CreateAtomOptionsPreloaded = {\n\t[K in keyof CreateAtomOptions]?: CreateAtomOptions[K] | undefined\n} & { skipHints?: boolean | undefined }\n\nexport async function createAtom(\n\targDir: string | undefined,\n\toptions: CreateAtomOptionsPreloaded,\n): Promise<void> {\n\tconst skipHint = options.skipHints ?? false\n\tconst packageManager = options.packageManager ?? getPkgManager()\n\n\tprompts.intro(pico.greenBright(`atom.io - Data Components for TypeScript`))\n\n\tconst { dir, templateName } = await prompts.group(\n\t\t{\n\t\t\ttemplateName: () =>\n\t\t\t\tprompts.select<TemplateName>({\n\t\t\t\t\tmessage: `Template:`,\n\t\t\t\t\tinitialValue: `preact-svg-editor`,\n\t\t\t\t\toptions: [\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tlabel: `Preact SVG Editor`,\n\t\t\t\t\t\t\tvalue: `preact-svg-editor`,\n\t\t\t\t\t\t},\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tlabel: `React Node Backend`,\n\t\t\t\t\t\t\tvalue: `react-node-backend`,\n\t\t\t\t\t\t},\n\t\t\t\t\t],\n\t\t\t\t}),\n\t\t\tdir: () =>\n\t\t\t\targDir\n\t\t\t\t\t? Promise.resolve(argDir)\n\t\t\t\t\t: prompts.text({\n\t\t\t\t\t\t\tmessage: `Project directory:`,\n\t\t\t\t\t\t\tplaceholder: `my-app`,\n\t\t\t\t\t\t\tvalidate(value) {\n\t\t\t\t\t\t\t\tif (value === undefined || value.length === 0) {\n\t\t\t\t\t\t\t\t\treturn `Directory name is required!`\n\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\tif (existsSync(value)) {\n\t\t\t\t\t\t\t\t\treturn `Refusing to overwrite existing directory or file! Please provide a non-clashing name.`\n\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t},\n\t\t\t\t\t\t}),\n\t\t},\n\t\t{\n\t\t\tonCancel: () => {\n\t\t\t\tprompts.cancel(pico.yellow(`Cancelled`))\n\t\t\t\tprocess.exit(0)\n\t\t\t},\n\t\t},\n\t)\n\tconst targetDir = resolve(process.cwd(), dir)\n\tconst opts: CreateAtomOptions = { packageManager, templateName }\n\n\tawait useSpinner(\n\t\t`Setting up your project directory...`,\n\t\t() => scaffold(targetDir, opts),\n\t\t`Set up project directory`,\n\t)\n\n\tawait useSpinner(\n\t\t`Installing project dependencies...`,\n\t\t() => installDeps(targetDir, opts),\n\t\t`Installed project dependencies`,\n\t)\n\n\tif (skipHint === false) {\n\t\tconst gettingStarted = `\n\t\t\t${pico.dim(`$`)} ${pico.blueBright(`cd ${dir}`)}\n\t\t\t${pico.dim(`$`)} ${pico.blueBright(\n\t\t\t\t`${\n\t\t\t\t\tpackageManager === `npm`\n\t\t\t\t\t\t? `npm run`\n\t\t\t\t\t\t: packageManager === `bun`\n\t\t\t\t\t\t\t? `bun run`\n\t\t\t\t\t\t\t: packageManager\n\t\t\t\t} dev`,\n\t\t\t)}\n\t\t`\n\t\tprompts.note(\n\t\t\tgettingStarted.trim().replace(/^\\t\\t\\t/gm, ``),\n\t\t\t`Getting Started`,\n\t\t)\n\t}\n\n\tprompts.outro(pico.green(`You're all set!`))\n}\n\nasync function useSpinner(\n\tstartMessage: string,\n\tfn: () => Promise<void>,\n\tfinishMessage: string,\n): Promise<void> {\n\ts.start(startMessage)\n\tawait fn()\n\ts.stop(pico.green(finishMessage))\n}\n\nasync function scaffold(to: string, opts: CreateAtomOptions): Promise<void> {\n\tawait fs.mkdir(to, { recursive: true })\n\n\tconst templateInfo = await getPackageInfo(\n\t\t`@atom.io/template-${opts.templateName}`,\n\t\t{\n\t\t\tpaths: [process.cwd(), import.meta.dirname],\n\t\t},\n\t)\n\tif (!templateInfo) throw new Error(`Could not find template package`)\n\tconst { rootPath } = templateInfo\n\tawait templateDir(rootPath, to, opts)\n\tconst nodeDirPath = resolve(to, `node`)\n\n\ttry {\n\t\tawait fs.access(nodeDirPath)\n\t} catch {\n\t\treturn\n\t}\n\n\tconst nodeDir = await fs.stat(nodeDirPath)\n\tif (!nodeDir.isDirectory()) return\n\n\tconst nodeFiles = await fs.readdir(nodeDirPath, { withFileTypes: true })\n\n\tawait Promise.all(\n\t\tnodeFiles.map(async (dirent) => {\n\t\t\tif (!dirent.isFile()) return\n\t\t\tconst filename = resolve(nodeDirPath, dirent.name)\n\t\t\tawait fs.chmod(filename, 0o755)\n\t\t}),\n\t)\n}\n\n/**\n * Recursive fs copy, swiped from `create-wmr`:\n * https://github.com/preactjs/wmr/blob/3c5672ecd2f958c8eaf372d33c084dc69228ae3f/packages/create-wmr/src/index.js#L108-L124\n */\nasync function templateDir(\n\tfrom: string,\n\tto: string,\n\topts: CreateAtomOptions,\n): Promise<void[]> {\n\tconst files = await fs.readdir(from)\n\tconst results = await Promise.all(\n\t\tfiles.map(async (f) => {\n\t\t\tif (f === `.` || f === `..`) return\n\t\t\tconst filename = resolve(from, f)\n\t\t\tif ((await fs.stat(filename)).isDirectory()) {\n\t\t\t\tawait fs.mkdir(resolve(to, f), { recursive: true })\n\t\t\t\treturn templateDir(filename, resolve(to, f), opts)\n\t\t\t}\n\t\t\tif (opts.packageManager !== `npm` && f === `README.md`) {\n\t\t\t\tawait fs.writeFile(\n\t\t\t\t\tresolve(to, f),\n\t\t\t\t\t(await fs.readFile(filename, `utf-8`)).replace(\n\t\t\t\t\t\t/npm run/g,\n\t\t\t\t\t\topts.packageManager === `bun` ? `bun run` : opts.packageManager,\n\t\t\t\t\t),\n\t\t\t\t)\n\t\t\t\treturn\n\t\t\t}\n\t\t\t// Publishing to npm renames the .gitignore to .npmignore\n\t\t\t// https://github.com/npm/npm/issues/7252#issuecomment-253339460\n\t\t\tif (f === `_gitignore`) f = `.gitignore`\n\t\t\tawait fs.copyFile(filename, resolve(to, f))\n\t\t}),\n\t)\n\treturn results.flat(99)\n}\n\nasync function installDeps(to: string, opts: CreateAtomOptions) {\n\tconst dependencies: string[] = []\n\tconst devDependencies: string[] = []\n\n\tconst installOpts = {\n\t\tpackageManager: opts.packageManager,\n\t\tto,\n\t}\n\n\tawait installPackages(dependencies, { ...installOpts })\n\tdevDependencies.length &&\n\t\t(await installPackages(devDependencies, { ...installOpts, dev: true }))\n}\n\ntype InstallOptions = {\n\tpackageManager: `bun` | `npm` | `pnpm` | `yarn`\n\tto: string\n\tdev?: boolean\n}\n\nfunction installPackages(pkgs: string[], opts: InstallOptions) {\n\treturn x(\n\t\topts.packageManager,\n\t\t[\n\t\t\t// `yarn add` will fail if nothing is provided\n\t\t\topts.packageManager === `yarn` ? (pkgs.length ? `add` : ``) : `install`,\n\t\t\topts.dev ? `-D` : ``,\n\t\t\t...pkgs,\n\t\t].filter(Boolean),\n\t\t{\n\t\t\tnodeOptions: {\n\t\t\t\tstdio: `ignore`,\n\t\t\t\tcwd: opts.to,\n\t\t\t},\n\t\t},\n\t)\n}\n\nfunction getPkgManager(): `bun` | `npm` | `pnpm` | `yarn` {\n\tconst userAgent = process.env[`npm_config_user_agent`] ?? ``\n\tif (userAgent.startsWith(`yarn`)) return `yarn`\n\tif (userAgent.startsWith(`pnpm`)) return `pnpm`\n\tif (userAgent.startsWith(`bun`)) return `bun`\n\treturn `npm`\n}\n"],"mappings":"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"}
|
package/dist/create-atom.x.js
CHANGED
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "create-atom.io",
|
|
3
|
-
"version": "0.1.
|
|
3
|
+
"version": "0.1.2",
|
|
4
4
|
"license": "MIT",
|
|
5
5
|
"author": {
|
|
6
6
|
"name": "Jeremy Banka",
|
|
@@ -30,20 +30,20 @@
|
|
|
30
30
|
"local-pkg": "1.1.2",
|
|
31
31
|
"picocolors": "1.1.1",
|
|
32
32
|
"tinyexec": "1.0.2",
|
|
33
|
-
"@atom.io/template-preact-svg-editor": "0.0.
|
|
34
|
-
"@atom.io/template-react-node-backend": "0.0.
|
|
33
|
+
"@atom.io/template-preact-svg-editor": "0.0.44",
|
|
34
|
+
"@atom.io/template-react-node-backend": "0.0.47",
|
|
35
35
|
"comline": "0.4.3"
|
|
36
36
|
},
|
|
37
37
|
"devDependencies": {
|
|
38
|
-
"@types/bun": "npm:bun-types@1.3.
|
|
38
|
+
"@types/bun": "npm:bun-types@1.3.8",
|
|
39
39
|
"@types/node": "25.0.10",
|
|
40
40
|
"@types/tmp": "0.2.6",
|
|
41
|
-
"@typescript/native-preview": "7.0.0-dev.
|
|
41
|
+
"@typescript/native-preview": "7.0.0-dev.20260205.1",
|
|
42
42
|
"concurrently": "9.2.1",
|
|
43
43
|
"eslint": "9.39.2",
|
|
44
44
|
"rimraf": "6.1.2",
|
|
45
45
|
"tmp": "0.2.5",
|
|
46
|
-
"tsdown": "0.20.
|
|
46
|
+
"tsdown": "0.20.3"
|
|
47
47
|
},
|
|
48
48
|
"scripts": {
|
|
49
49
|
"build": "tsdown && rm -f dist/*.x.d.ts",
|
|
@@ -1,134 +0,0 @@
|
|
|
1
|
-
import { existsSync, promises } from "node:fs";
|
|
2
|
-
import { resolve } from "node:path";
|
|
3
|
-
import * as prompts from "@clack/prompts";
|
|
4
|
-
import { getPackageInfo } from "local-pkg";
|
|
5
|
-
import picocolors from "picocolors";
|
|
6
|
-
import { x } from "tinyexec";
|
|
7
|
-
|
|
8
|
-
//#region src/create-atom.ts
|
|
9
|
-
const pico = picocolors.createColors(true);
|
|
10
|
-
const s = prompts.spinner();
|
|
11
|
-
async function createAtom(argDir, options) {
|
|
12
|
-
const skipHint = options.skipHints ?? false;
|
|
13
|
-
const packageManager = options.packageManager ?? getPkgManager();
|
|
14
|
-
prompts.intro(pico.greenBright(`atom.io - Data Components for TypeScript`));
|
|
15
|
-
const { dir, templateName } = await prompts.group({
|
|
16
|
-
templateName: () => prompts.select({
|
|
17
|
-
message: `Template:`,
|
|
18
|
-
initialValue: `preact-svg-editor`,
|
|
19
|
-
options: [{
|
|
20
|
-
label: `Preact SVG Editor`,
|
|
21
|
-
value: `preact-svg-editor`
|
|
22
|
-
}, {
|
|
23
|
-
label: `React Node Backend`,
|
|
24
|
-
value: `react-node-backend`
|
|
25
|
-
}]
|
|
26
|
-
}),
|
|
27
|
-
dir: () => argDir ? Promise.resolve(argDir) : prompts.text({
|
|
28
|
-
message: `Project directory:`,
|
|
29
|
-
placeholder: `my-app`,
|
|
30
|
-
validate(value) {
|
|
31
|
-
if (value === void 0 || value.length === 0) return `Directory name is required!`;
|
|
32
|
-
if (existsSync(value)) return `Refusing to overwrite existing directory or file! Please provide a non-clashing name.`;
|
|
33
|
-
}
|
|
34
|
-
})
|
|
35
|
-
}, { onCancel: () => {
|
|
36
|
-
prompts.cancel(pico.yellow(`Cancelled`));
|
|
37
|
-
process.exit(0);
|
|
38
|
-
} });
|
|
39
|
-
const targetDir = resolve(process.cwd(), dir);
|
|
40
|
-
const opts = {
|
|
41
|
-
packageManager,
|
|
42
|
-
templateName
|
|
43
|
-
};
|
|
44
|
-
await useSpinner(`Setting up your project directory...`, () => scaffold(targetDir, opts), `Set up project directory`);
|
|
45
|
-
await useSpinner(`Installing project dependencies...`, () => installDeps(targetDir, opts), `Installed project dependencies`);
|
|
46
|
-
if (skipHint === false) {
|
|
47
|
-
const gettingStarted = `
|
|
48
|
-
${pico.dim(`$`)} ${pico.blueBright(`cd ${dir}`)}
|
|
49
|
-
${pico.dim(`$`)} ${pico.blueBright(`${packageManager === `npm` ? `npm run` : packageManager === `bun` ? `bun run` : packageManager} dev`)}
|
|
50
|
-
`;
|
|
51
|
-
prompts.note(gettingStarted.trim().replace(/^\t\t\t/gm, ``), `Getting Started`);
|
|
52
|
-
}
|
|
53
|
-
prompts.outro(pico.green(`You're all set!`));
|
|
54
|
-
}
|
|
55
|
-
async function useSpinner(startMessage, fn, finishMessage) {
|
|
56
|
-
s.start(startMessage);
|
|
57
|
-
await fn();
|
|
58
|
-
s.stop(pico.green(finishMessage));
|
|
59
|
-
}
|
|
60
|
-
async function scaffold(to, opts) {
|
|
61
|
-
await promises.mkdir(to, { recursive: true });
|
|
62
|
-
const templateInfo = await getPackageInfo(`@atom.io/template-${opts.templateName}`, { paths: [process.cwd(), import.meta.dirname] });
|
|
63
|
-
if (!templateInfo) throw new Error(`Could not find template package`);
|
|
64
|
-
const { rootPath } = templateInfo;
|
|
65
|
-
await templateDir(rootPath, to, opts);
|
|
66
|
-
const nodeDirPath = resolve(to, `node`);
|
|
67
|
-
try {
|
|
68
|
-
await promises.access(nodeDirPath);
|
|
69
|
-
} catch {
|
|
70
|
-
return;
|
|
71
|
-
}
|
|
72
|
-
if (!(await promises.stat(nodeDirPath)).isDirectory()) return;
|
|
73
|
-
const nodeFiles = await promises.readdir(nodeDirPath, { withFileTypes: true });
|
|
74
|
-
await Promise.all(nodeFiles.map(async (dirent) => {
|
|
75
|
-
if (!dirent.isFile()) return;
|
|
76
|
-
const filename = resolve(nodeDirPath, dirent.name);
|
|
77
|
-
await promises.chmod(filename, 493);
|
|
78
|
-
}));
|
|
79
|
-
}
|
|
80
|
-
/**
|
|
81
|
-
* Recursive fs copy, swiped from `create-wmr`:
|
|
82
|
-
* https://github.com/preactjs/wmr/blob/3c5672ecd2f958c8eaf372d33c084dc69228ae3f/packages/create-wmr/src/index.js#L108-L124
|
|
83
|
-
*/
|
|
84
|
-
async function templateDir(from, to, opts) {
|
|
85
|
-
const files = await promises.readdir(from);
|
|
86
|
-
return (await Promise.all(files.map(async (f) => {
|
|
87
|
-
if (f === `.` || f === `..`) return;
|
|
88
|
-
const filename = resolve(from, f);
|
|
89
|
-
if ((await promises.stat(filename)).isDirectory()) {
|
|
90
|
-
await promises.mkdir(resolve(to, f), { recursive: true });
|
|
91
|
-
return templateDir(filename, resolve(to, f), opts);
|
|
92
|
-
}
|
|
93
|
-
if (opts.packageManager !== `npm` && f === `README.md`) {
|
|
94
|
-
await promises.writeFile(resolve(to, f), (await promises.readFile(filename, `utf-8`)).replace(/npm run/g, opts.packageManager === `bun` ? `bun run` : opts.packageManager));
|
|
95
|
-
return;
|
|
96
|
-
}
|
|
97
|
-
if (f === `_gitignore`) f = `.gitignore`;
|
|
98
|
-
await promises.copyFile(filename, resolve(to, f));
|
|
99
|
-
}))).flat(99);
|
|
100
|
-
}
|
|
101
|
-
async function installDeps(to, opts) {
|
|
102
|
-
const dependencies = [];
|
|
103
|
-
const devDependencies = [];
|
|
104
|
-
const installOpts = {
|
|
105
|
-
packageManager: opts.packageManager,
|
|
106
|
-
to
|
|
107
|
-
};
|
|
108
|
-
await installPackages(dependencies, { ...installOpts });
|
|
109
|
-
devDependencies.length && await installPackages(devDependencies, {
|
|
110
|
-
...installOpts,
|
|
111
|
-
dev: true
|
|
112
|
-
});
|
|
113
|
-
}
|
|
114
|
-
function installPackages(pkgs, opts) {
|
|
115
|
-
return x(opts.packageManager, [
|
|
116
|
-
opts.packageManager === `yarn` ? pkgs.length ? `add` : `` : `install`,
|
|
117
|
-
opts.dev ? `-D` : ``,
|
|
118
|
-
...pkgs
|
|
119
|
-
].filter(Boolean), { nodeOptions: {
|
|
120
|
-
stdio: `ignore`,
|
|
121
|
-
cwd: opts.to
|
|
122
|
-
} });
|
|
123
|
-
}
|
|
124
|
-
function getPkgManager() {
|
|
125
|
-
const userAgent = process.env[`npm_config_user_agent`] ?? ``;
|
|
126
|
-
if (userAgent.startsWith(`yarn`)) return `yarn`;
|
|
127
|
-
if (userAgent.startsWith(`pnpm`)) return `pnpm`;
|
|
128
|
-
if (userAgent.startsWith(`bun`)) return `bun`;
|
|
129
|
-
return `npm`;
|
|
130
|
-
}
|
|
131
|
-
|
|
132
|
-
//#endregion
|
|
133
|
-
export { createAtom as t };
|
|
134
|
-
//# sourceMappingURL=create-atom-Cz0suUKB.js.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"create-atom-Cz0suUKB.js","names":["fs"],"sources":["../src/create-atom.ts"],"sourcesContent":["import { existsSync, promises as fs } from \"node:fs\"\nimport { resolve } from \"node:path\"\n\nimport * as prompts from \"@clack/prompts\"\nimport { getPackageInfo } from \"local-pkg\"\nimport picocolors from \"picocolors\"\nimport type { Colors } from \"picocolors/types\"\nimport { x } from \"tinyexec\"\n\nconst pico: Colors = picocolors.createColors(true)\n\nconst s = prompts.spinner()\n\nexport type PackageManager = `bun` | `npm` | `pnpm` | `yarn`\nexport type TemplateName = `preact-svg-editor` | `react-node-backend`\n\nexport type CreateAtomOptions = {\n\tpackageManager: PackageManager\n\ttemplateName: TemplateName\n}\n\nexport type CreateAtomOptionsPreloaded = {\n\t[K in keyof CreateAtomOptions]?: CreateAtomOptions[K] | undefined\n} & { skipHints?: boolean | undefined }\n\nexport async function createAtom(\n\targDir: string | undefined,\n\toptions: CreateAtomOptionsPreloaded,\n): Promise<void> {\n\tconst skipHint = options.skipHints ?? false\n\tconst packageManager = options.packageManager ?? getPkgManager()\n\n\tprompts.intro(pico.greenBright(`atom.io - Data Components for TypeScript`))\n\n\tconst { dir, templateName } = await prompts.group(\n\t\t{\n\t\t\ttemplateName: () =>\n\t\t\t\tprompts.select<TemplateName>({\n\t\t\t\t\tmessage: `Template:`,\n\t\t\t\t\tinitialValue: `preact-svg-editor`,\n\t\t\t\t\toptions: [\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tlabel: `Preact SVG Editor`,\n\t\t\t\t\t\t\tvalue: `preact-svg-editor`,\n\t\t\t\t\t\t},\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tlabel: `React Node Backend`,\n\t\t\t\t\t\t\tvalue: `react-node-backend`,\n\t\t\t\t\t\t},\n\t\t\t\t\t],\n\t\t\t\t}),\n\t\t\tdir: () =>\n\t\t\t\targDir\n\t\t\t\t\t? Promise.resolve(argDir)\n\t\t\t\t\t: prompts.text({\n\t\t\t\t\t\t\tmessage: `Project directory:`,\n\t\t\t\t\t\t\tplaceholder: `my-app`,\n\t\t\t\t\t\t\tvalidate(value) {\n\t\t\t\t\t\t\t\tif (value === undefined || value.length === 0) {\n\t\t\t\t\t\t\t\t\treturn `Directory name is required!`\n\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\tif (existsSync(value)) {\n\t\t\t\t\t\t\t\t\treturn `Refusing to overwrite existing directory or file! Please provide a non-clashing name.`\n\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t},\n\t\t\t\t\t\t}),\n\t\t},\n\t\t{\n\t\t\tonCancel: () => {\n\t\t\t\tprompts.cancel(pico.yellow(`Cancelled`))\n\t\t\t\tprocess.exit(0)\n\t\t\t},\n\t\t},\n\t)\n\tconst targetDir = resolve(process.cwd(), dir)\n\tconst opts: CreateAtomOptions = { packageManager, templateName }\n\n\tawait useSpinner(\n\t\t`Setting up your project directory...`,\n\t\t() => scaffold(targetDir, opts),\n\t\t`Set up project directory`,\n\t)\n\n\tawait useSpinner(\n\t\t`Installing project dependencies...`,\n\t\t() => installDeps(targetDir, opts),\n\t\t`Installed project dependencies`,\n\t)\n\n\tif (skipHint === false) {\n\t\tconst gettingStarted = `\n\t\t\t${pico.dim(`$`)} ${pico.blueBright(`cd ${dir}`)}\n\t\t\t${pico.dim(`$`)} ${pico.blueBright(\n\t\t\t\t`${\n\t\t\t\t\tpackageManager === `npm`\n\t\t\t\t\t\t? `npm run`\n\t\t\t\t\t\t: packageManager === `bun`\n\t\t\t\t\t\t\t? `bun run`\n\t\t\t\t\t\t\t: packageManager\n\t\t\t\t} dev`,\n\t\t\t)}\n\t\t`\n\t\tprompts.note(\n\t\t\tgettingStarted.trim().replace(/^\\t\\t\\t/gm, ``),\n\t\t\t`Getting Started`,\n\t\t)\n\t}\n\n\tprompts.outro(pico.green(`You're all set!`))\n}\n\nasync function useSpinner(\n\tstartMessage: string,\n\tfn: () => Promise<void>,\n\tfinishMessage: string,\n): Promise<void> {\n\ts.start(startMessage)\n\tawait fn()\n\ts.stop(pico.green(finishMessage))\n}\n\nasync function scaffold(to: string, opts: CreateAtomOptions): Promise<void> {\n\tawait fs.mkdir(to, { recursive: true })\n\n\tconst templateInfo = await getPackageInfo(\n\t\t`@atom.io/template-${opts.templateName}`,\n\t\t{\n\t\t\tpaths: [process.cwd(), import.meta.dirname],\n\t\t},\n\t)\n\tif (!templateInfo) throw new Error(`Could not find template package`)\n\tconst { rootPath } = templateInfo\n\tawait templateDir(rootPath, to, opts)\n\tconst nodeDirPath = resolve(to, `node`)\n\n\ttry {\n\t\tawait fs.access(nodeDirPath)\n\t} catch {\n\t\treturn\n\t}\n\n\tconst nodeDir = await fs.stat(nodeDirPath)\n\tif (!nodeDir.isDirectory()) return\n\n\tconst nodeFiles = await fs.readdir(nodeDirPath, { withFileTypes: true })\n\n\tawait Promise.all(\n\t\tnodeFiles.map(async (dirent) => {\n\t\t\tif (!dirent.isFile()) return\n\t\t\tconst filename = resolve(nodeDirPath, dirent.name)\n\t\t\tawait fs.chmod(filename, 0o755)\n\t\t}),\n\t)\n}\n\n/**\n * Recursive fs copy, swiped from `create-wmr`:\n * https://github.com/preactjs/wmr/blob/3c5672ecd2f958c8eaf372d33c084dc69228ae3f/packages/create-wmr/src/index.js#L108-L124\n */\nasync function templateDir(\n\tfrom: string,\n\tto: string,\n\topts: CreateAtomOptions,\n): Promise<void[]> {\n\tconst files = await fs.readdir(from)\n\tconst results = await Promise.all(\n\t\tfiles.map(async (f) => {\n\t\t\tif (f === `.` || f === `..`) return\n\t\t\tconst filename = resolve(from, f)\n\t\t\tif ((await fs.stat(filename)).isDirectory()) {\n\t\t\t\tawait fs.mkdir(resolve(to, f), { recursive: true })\n\t\t\t\treturn templateDir(filename, resolve(to, f), opts)\n\t\t\t}\n\t\t\tif (opts.packageManager !== `npm` && f === `README.md`) {\n\t\t\t\tawait fs.writeFile(\n\t\t\t\t\tresolve(to, f),\n\t\t\t\t\t(await fs.readFile(filename, `utf-8`)).replace(\n\t\t\t\t\t\t/npm run/g,\n\t\t\t\t\t\topts.packageManager === `bun` ? `bun run` : opts.packageManager,\n\t\t\t\t\t),\n\t\t\t\t)\n\t\t\t\treturn\n\t\t\t}\n\t\t\t// Publishing to npm renames the .gitignore to .npmignore\n\t\t\t// https://github.com/npm/npm/issues/7252#issuecomment-253339460\n\t\t\tif (f === `_gitignore`) f = `.gitignore`\n\t\t\tawait fs.copyFile(filename, resolve(to, f))\n\t\t}),\n\t)\n\treturn results.flat(99)\n}\n\nasync function installDeps(to: string, opts: CreateAtomOptions) {\n\tconst dependencies: string[] = []\n\tconst devDependencies: string[] = []\n\n\tconst installOpts = {\n\t\tpackageManager: opts.packageManager,\n\t\tto,\n\t}\n\n\tawait installPackages(dependencies, { ...installOpts })\n\tdevDependencies.length &&\n\t\t(await installPackages(devDependencies, { ...installOpts, dev: true }))\n}\n\ntype InstallOptions = {\n\tpackageManager: `bun` | `npm` | `pnpm` | `yarn`\n\tto: string\n\tdev?: boolean\n}\n\nfunction installPackages(pkgs: string[], opts: InstallOptions) {\n\treturn x(\n\t\topts.packageManager,\n\t\t[\n\t\t\t// `yarn add` will fail if nothing is provided\n\t\t\topts.packageManager === `yarn` ? (pkgs.length ? `add` : ``) : `install`,\n\t\t\topts.dev ? `-D` : ``,\n\t\t\t...pkgs,\n\t\t].filter(Boolean),\n\t\t{\n\t\t\tnodeOptions: {\n\t\t\t\tstdio: `ignore`,\n\t\t\t\tcwd: opts.to,\n\t\t\t},\n\t\t},\n\t)\n}\n\nfunction getPkgManager(): `bun` | `npm` | `pnpm` | `yarn` {\n\tconst userAgent = process.env[`npm_config_user_agent`] ?? ``\n\tif (userAgent.startsWith(`yarn`)) return `yarn`\n\tif (userAgent.startsWith(`pnpm`)) return `pnpm`\n\tif (userAgent.startsWith(`bun`)) return `bun`\n\treturn `npm`\n}\n"],"mappings":"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"}
|