Skip to content

Commit a52f9a0

Browse files
committed
migrate cmd to massarg + update tests
1 parent aeddd44 commit a52f9a0

9 files changed

+156
-72
lines changed

.vscode/tasks.json

+7-1
Original file line numberDiff line numberDiff line change
@@ -25,6 +25,12 @@
2525
"type": "npm",
2626
"problemMatcher": [],
2727
},
28+
{
29+
"command": "yarn test --watchAll",
30+
"label": "yarn test --watchAll",
31+
"type": "shell",
32+
"problemMatcher": [],
33+
},
2834
{
2935
"script": "cmd",
3036
"label": "cmd",
@@ -44,4 +50,4 @@
4450
"problemMatcher": [],
4551
},
4652
],
47-
}
53+
}

package.json

+1
Original file line numberDiff line numberDiff line change
@@ -25,6 +25,7 @@
2525
"glob": "^7.1.3",
2626
"handlebars": "^4.7.7",
2727
"lodash": "^4.17.21",
28+
"massarg": "^0.1.2",
2829
"util.promisify": "^1.1.1"
2930
},
3031
"devDependencies": {

src/cmd.ts

+83-55
Original file line numberDiff line numberDiff line change
@@ -1,60 +1,88 @@
11
import Scaffold from "./scaffold"
2-
import args from "args"
2+
import massarg from "massarg"
33
import { ScaffoldCmdConfig } from "./types"
44

5-
const options = args
6-
.command("", "")
7-
.option(
8-
["n", "name"],
9-
"Name to be passed to the generated files. {{name}} and {{Name}} inside contents and file names will be replaced accordingly."
10-
)
11-
.option(
12-
["o", "output"],
13-
"Path to output to. If --create-sub-folder is enabled, the subfolder will be created inside this path."
14-
)
15-
.option(
16-
["t", "templates"],
17-
"Template files to use as input. You may provide multiple files, each of which can be a relative or absolute path, " +
5+
massarg<ScaffoldCmdConfig & { help: boolean; extras: string[] }>()
6+
.main(Scaffold)
7+
.option({
8+
name: "name",
9+
aliases: ["n"],
10+
isDefault: true,
11+
description:
12+
"Name to be passed to the generated files. {{name}} and {{Name}} inside contents and file names will be replaced accordingly.",
13+
})
14+
.option({
15+
name: "output",
16+
aliases: ["o"],
17+
description:
18+
"Path to output to. If --create-sub-folder is enabled, the subfolder will be created inside this path.",
19+
})
20+
.option({
21+
name: "templates",
22+
aliases: ["t"],
23+
description:
24+
"Template files to use as input. You may provide multiple files, each of which can be a relative or absolute path, " +
1825
"or a glob pattern for multiple file matching easily.",
19-
[]
20-
)
21-
.option(["w", "overwrite"], "Enable to override output files, even if they already exist.", false)
22-
.option(
23-
["d", "data"],
24-
"Add custom data to the templates. By default, only your app name is included.",
25-
undefined,
26-
JSON.parse
27-
)
28-
.option(["s", "create-sub-folder"], "Create subfolder with the input name", false)
29-
.option(["q", "silent"], "Supress output logs", false)
30-
.option(
31-
["d", "dry-run"],
32-
"Don't emit actual files. This is good for testing your scaffolds and making sure they " +
26+
defaultValue: [],
27+
array: true,
28+
})
29+
.option({
30+
aliases: ["w"],
31+
name: "overwrite",
32+
description: "Enable to override output files, even if they already exist.",
33+
defaultValue: false,
34+
boolean: true,
35+
})
36+
.option({
37+
aliases: ["d"],
38+
name: "data",
39+
description: "Add custom data to the templates. By default, only your app name is included.",
40+
parse: (v) => JSON.parse(v),
41+
})
42+
.option({
43+
aliases: ["s"],
44+
name: "create-sub-folder",
45+
description: "Create subfolder with the input name",
46+
defaultValue: false,
47+
boolean: true,
48+
})
49+
.option({ aliases: ["q"], name: "quiet", description: "Supress output logs", defaultValue: false, boolean: true })
50+
.option({
51+
aliases: ["dr"],
52+
name: "dry-run",
53+
description:
54+
"Don't emit actual files. This is good for testing your scaffolds and making sure they " +
3355
"don't fail, without having to write actual files.",
34-
false
35-
)
36-
.example(
37-
`yarn cmd -t examples/test-input/Component -o examples/test-output -d '{"property":"myProp","value":"10"}'`,
38-
"Usage"
39-
)
40-
.parse(process.argv, {
41-
value: "name",
42-
mri: {},
43-
mainColor: "yellow",
44-
subColor: ["dim"],
45-
name: "simple-scaffold",
46-
usageFilter: (usage: string) => {
47-
usage = usage.replace("[options] [command] ", "").replace("name", "[options] name")
48-
const lines = usage.split("\n")
49-
usage = [
50-
...lines.slice(
51-
0,
52-
lines.findIndex((l) => l.startsWith(" Commands:"))
53-
),
54-
...lines.slice(lines.findIndex((l) => l.startsWith(" Options:"))),
55-
].join("\n")
56-
return usage
57-
},
58-
}) as ScaffoldCmdConfig
59-
60-
Scaffold(options)
56+
defaultValue: false,
57+
boolean: true,
58+
})
59+
// .example(
60+
// `yarn cmd -t examples/test-input/Component -o examples/test-output -d '{"property":"myProp","value":"10"}'`,
61+
// "Usage"
62+
// )
63+
.help({
64+
binName: "simple-scaffold",
65+
useGlobalColumns: true,
66+
usageExample: "[options]",
67+
})
68+
.parse()
69+
// .parse(process.argv, {
70+
// value: "name",
71+
// mri: {},
72+
// mainColor: "yellow",
73+
// subColor: ["dim"],
74+
// name: "simple-scaffold",
75+
// usageFilter: (usage: string) => {
76+
// usage = usage.replace("[options] [command] ", "").replace("name", "[options] name")
77+
// const lines = usage.split("\n")
78+
// usage = [
79+
// ...lines.slice(
80+
// 0,
81+
// lines.findIndex((l) => l.startsWith(" Commands:"))
82+
// ),
83+
// ...lines.slice(lines.findIndex((l) => l.startsWith(" Options:"))),
84+
// ].join("\n")
85+
// return usage
86+
// },
87+
// }
88+
// ) as ScaffoldCmdConfig

src/scaffold.ts

+7-1
Original file line numberDiff line numberDiff line change
@@ -25,15 +25,21 @@ export async function Scaffold(config: ScaffoldConfig) {
2525
createSubfolder: options.createSubFolder,
2626
data: options.data,
2727
overwrite: options.overwrite,
28-
silent: options.silent,
28+
quiet: options.quiet,
2929
})
3030
log(options, "Data:", data)
3131
for (let template of config.templates) {
3232
try {
33+
// try {
3334
const tplStat = await stat(template)
3435
if (tplStat.isDirectory()) {
3536
template = template + "/**/*"
3637
}
38+
// } catch (e) {
39+
// if (e.code !== "ENOENT") {
40+
// throw e
41+
// }
42+
// }
3743
const files = await promisify(glob)(template, { dot: true })
3844
for (const templatePath of files) {
3945
await handleTemplateFile(templatePath, options, data)

src/types.d.ts

+2-2
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,7 @@ export interface ScaffoldConfig {
99
createSubFolder?: boolean
1010
data?: Record<string, string>
1111
overwrite?: FileResponse<boolean>
12-
silent?: boolean
12+
quiet?: boolean
1313
dryRun?: boolean
1414
}
1515
export interface ScaffoldCmdConfig {
@@ -19,6 +19,6 @@ export interface ScaffoldCmdConfig {
1919
createSubFolder: boolean
2020
data?: Record<string, string>
2121
overwrite: boolean
22-
silent: boolean
22+
quiet: boolean
2323
dryRun: boolean
2424
}

src/utils.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -26,7 +26,7 @@ export function handleErr(err: NodeJS.ErrnoException | null) {
2626
}
2727

2828
export function log(options: ScaffoldConfig, ...obj: any[]) {
29-
if (options.silent) {
29+
if (options.quiet) {
3030
return
3131
}
3232
console.log(...obj)

src/tests/scaffold.test.ts tests/scaffold.test.ts

+45-10
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
import mockFs from "mock-fs"
2-
import Scaffold from "../scaffold"
3-
import { readFileSync } from "fs"
2+
import Scaffold from "../src/scaffold"
3+
import { readdirSync, readFileSync } from "fs"
44

55
const fileStructNormal = {
66
input: {
@@ -16,6 +16,16 @@ const fileStructWithData = {
1616
output: {},
1717
}
1818

19+
const fileStructNested = {
20+
input: {
21+
"{{name}}-1.text": "This should be in root",
22+
"{{Name}}": {
23+
"{{name}}-2.txt": "Hello, my value is {{value}}",
24+
},
25+
},
26+
output: {},
27+
}
28+
1929
describe("Scaffold", () => {
2030
describe("create subfolder", () => {
2131
beforeAll(() => {
@@ -28,7 +38,7 @@ describe("Scaffold", () => {
2838
name: "app_name",
2939
output: "output",
3040
templates: ["input"],
31-
silent: true,
41+
quiet: true,
3242
})
3343

3444
const data = readFileSync(process.cwd() + "/output/app_name.txt")
@@ -41,7 +51,7 @@ describe("Scaffold", () => {
4151
output: "output",
4252
templates: ["input"],
4353
createSubFolder: true,
44-
silent: true,
54+
quiet: true,
4555
})
4656

4757
const data = readFileSync(process.cwd() + "/output/app_name/app_name.txt")
@@ -63,15 +73,15 @@ describe("Scaffold", () => {
6373
output: "output",
6474
templates: ["input"],
6575
data: { value: "1" },
66-
silent: true,
76+
quiet: true,
6777
})
6878

6979
await Scaffold({
7080
name: "app_name",
7181
output: "output",
7282
templates: ["input"],
7383
data: { value: "2" },
74-
silent: true,
84+
quiet: true,
7585
})
7686

7787
const data = readFileSync(process.cwd() + "/output/app_name.txt")
@@ -84,7 +94,7 @@ describe("Scaffold", () => {
8494
output: "output",
8595
templates: ["input"],
8696
data: { value: "1" },
87-
silent: true,
97+
quiet: true,
8898
})
8999

90100
await Scaffold({
@@ -93,7 +103,7 @@ describe("Scaffold", () => {
93103
templates: ["input"],
94104
data: { value: "2" },
95105
overwrite: true,
96-
silent: true,
106+
quiet: true,
97107
})
98108

99109
const data = readFileSync(process.cwd() + "/output/app_name.txt")
@@ -118,7 +128,7 @@ describe("Scaffold", () => {
118128
output: "output",
119129
templates: ["non-existing-input"],
120130
data: { value: "1" },
121-
silent: true,
131+
quiet: true,
122132
})
123133
).rejects.toThrow()
124134

@@ -143,7 +153,7 @@ describe("Scaffold", () => {
143153
output: (fullPath, basedir, basename) => `custom-output/${basename.split(".")[0]}`,
144154
templates: ["input"],
145155
data: { value: "1" },
146-
silent: true,
156+
quiet: true,
147157
})
148158
const data = readFileSync(process.cwd() + "/custom-output/app_name/app_name.txt")
149159
expect(data.toString()).toBe("Hello, my app is app_name")
@@ -153,4 +163,29 @@ describe("Scaffold", () => {
153163
mockFs.restore()
154164
})
155165
})
166+
167+
describe("output structure", () => {
168+
beforeAll(() => {
169+
mockFs.restore()
170+
mockFs(fileStructNested)
171+
})
172+
173+
test("should maintain input structure on output", async () => {
174+
await Scaffold({
175+
name: "app_name",
176+
output: "./",
177+
templates: ["input"],
178+
data: { value: "1" },
179+
quiet: true,
180+
})
181+
182+
const dir = readdirSync(process.cwd())
183+
console.log({ dir })
184+
expect(dir).toHaveProperty("length")
185+
})
186+
187+
afterAll(() => {
188+
mockFs.restore()
189+
})
190+
})
156191
})

tsconfig.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -17,6 +17,6 @@
1717
"src/cmd.ts"
1818
],
1919
"exclude": [
20-
"src/tests/*"
20+
"tests/*"
2121
]
2222
}

yarn.lock

+9-1
Original file line numberDiff line numberDiff line change
@@ -932,7 +932,7 @@ chalk@2.4.2, chalk@^2.0.0:
932932
escape-string-regexp "^1.0.5"
933933
supports-color "^5.3.0"
934934

935-
chalk@^4.0.0:
935+
chalk@^4.0.0, chalk@^4.1.1:
936936
version "4.1.1"
937937
resolved "https://registry.yarnpkg.com/chalk/-/chalk-4.1.1.tgz#c80b3fab28bf6371e6863325eee67e618b77e6ad"
938938
integrity sha512-diHzdDKxcU+bAsUboHLPEDQiw0qEe0qd7SYUn3HgcFlWgbDcfLGswOHYeGrHKzG9z6UYf01d9VFMfZxPM1xZSg==
@@ -2153,6 +2153,14 @@ makeerror@1.0.x:
21532153
dependencies:
21542154
tmpl "1.0.x"
21552155

2156+
massarg@^0.1.2:
2157+
version "0.1.2"
2158+
resolved "https://registry.yarnpkg.com/massarg/-/massarg-0.1.2.tgz#f298741318172be14f3d2b701329fbe10eff41bb"
2159+
integrity sha512-gpFIjsvOoqyQnrqNDytQXPljOGlX5lvJFGYzAIqjxDqiSZwHOvz+/YfjtzrFvokfYsk0uZbE/XOH4LVRiu/1cg==
2160+
dependencies:
2161+
chalk "^4.1.1"
2162+
lodash "^4.17.21"
2163+
21562164
merge-stream@^2.0.0:
21572165
version "2.0.0"
21582166
resolved "https://registry.yarnpkg.com/merge-stream/-/merge-stream-2.0.0.tgz#52823629a14dd00c9770fb6ad47dc6310f2c1f60"

0 commit comments

Comments
 (0)