@ackee/create-node-app 1.0.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.
Files changed (104) hide show
  1. package/.eslintrc.cjs +10 -0
  2. package/AUTHORS +3 -0
  3. package/LICENSE.txt +22 -0
  4. package/README.md +39 -0
  5. package/bin/create-node-app.js +6 -0
  6. package/lib/Bootstrap.js +79 -0
  7. package/lib/Bootstrap.js.map +1 -0
  8. package/lib/Logger.js +12 -0
  9. package/lib/Logger.js.map +1 -0
  10. package/lib/Npm.js +33 -0
  11. package/lib/Npm.js.map +1 -0
  12. package/lib/PackageJson.js +39 -0
  13. package/lib/PackageJson.js.map +1 -0
  14. package/lib/Starter.js +2 -0
  15. package/lib/Starter.js.map +1 -0
  16. package/lib/Toolbelt.js +102 -0
  17. package/lib/Toolbelt.js.map +1 -0
  18. package/lib/cloudrun/CloudRunStarter.js +126 -0
  19. package/lib/cloudrun/CloudRunStarter.js.map +1 -0
  20. package/lib/cloudrun-graphql/GraphQLStarter.js +118 -0
  21. package/lib/cloudrun-graphql/GraphQLStarter.js.map +1 -0
  22. package/lib/types.js +2 -0
  23. package/lib/types.js.map +1 -0
  24. package/logo.png +0 -0
  25. package/package.json +42 -0
  26. package/prettier.config.cjs +1 -0
  27. package/src/Bootstrap.ts +99 -0
  28. package/src/Logger.ts +11 -0
  29. package/src/Npm.ts +38 -0
  30. package/src/PackageJson.ts +47 -0
  31. package/src/Starter.ts +7 -0
  32. package/src/Toolbelt.ts +132 -0
  33. package/src/cloudrun/CloudRunStarter.ts +181 -0
  34. package/src/cloudrun-graphql/GraphQLStarter.ts +182 -0
  35. package/src/types.ts +1 -0
  36. package/starter/cloudrun/.env.jsonc +10 -0
  37. package/starter/cloudrun/.eslint.tsconfig.json +4 -0
  38. package/starter/cloudrun/.eslintrc.cjs +8 -0
  39. package/starter/cloudrun/README.md +69 -0
  40. package/starter/cloudrun/src/adapters/pino.logger.ts +44 -0
  41. package/starter/cloudrun/src/config.ts +22 -0
  42. package/starter/cloudrun/src/container.ts +18 -0
  43. package/starter/cloudrun/src/context.ts +39 -0
  44. package/starter/cloudrun/src/domain/errors/codes.ts +9 -0
  45. package/starter/cloudrun/src/domain/errors/errors.ts +25 -0
  46. package/starter/cloudrun/src/domain/health-check.service.ts +15 -0
  47. package/starter/cloudrun/src/domain/ports/logger.d.ts +21 -0
  48. package/starter/cloudrun/src/index.ts +17 -0
  49. package/starter/cloudrun/src/test/health-check.test.ts +25 -0
  50. package/starter/cloudrun/src/test/util/openapi-test.util.ts +71 -0
  51. package/starter/cloudrun/src/view/cli/README.md +17 -0
  52. package/starter/cloudrun/src/view/cli/cli.ts +94 -0
  53. package/starter/cloudrun/src/view/cli/openapi/generate.ts +64 -0
  54. package/starter/cloudrun/src/view/rest/controller/health-check.controller.ts +33 -0
  55. package/starter/cloudrun/src/view/rest/middleware/context-middleware.ts +28 -0
  56. package/starter/cloudrun/src/view/rest/middleware/error-handler.ts +60 -0
  57. package/starter/cloudrun/src/view/rest/middleware/request-logger.ts +37 -0
  58. package/starter/cloudrun/src/view/rest/request.d.ts +9 -0
  59. package/starter/cloudrun/src/view/rest/routes.ts +15 -0
  60. package/starter/cloudrun/src/view/rest/spec/openapi.yml +65 -0
  61. package/starter/cloudrun/src/view/rest/util/openapi.util.ts +310 -0
  62. package/starter/cloudrun/src/view/server.ts +25 -0
  63. package/starter/cloudrun-graphql/.env.jsonc +12 -0
  64. package/starter/cloudrun-graphql/.eslint.tsconfig.json +4 -0
  65. package/starter/cloudrun-graphql/.eslintrc.cjs +43 -0
  66. package/starter/cloudrun-graphql/README.md +53 -0
  67. package/starter/cloudrun-graphql/codegen.yml +11 -0
  68. package/starter/cloudrun-graphql/src/adapters/pino.logger.ts +44 -0
  69. package/starter/cloudrun-graphql/src/config.ts +21 -0
  70. package/starter/cloudrun-graphql/src/container.ts +15 -0
  71. package/starter/cloudrun-graphql/src/context.ts +39 -0
  72. package/starter/cloudrun-graphql/src/domain/errors/codes.ts +9 -0
  73. package/starter/cloudrun-graphql/src/domain/errors/errors.ts +25 -0
  74. package/starter/cloudrun-graphql/src/domain/ports/logger.d.ts +21 -0
  75. package/starter/cloudrun-graphql/src/index.ts +11 -0
  76. package/starter/cloudrun-graphql/src/test/helloWorld.test.ts +23 -0
  77. package/starter/cloudrun-graphql/src/view/controller.ts +42 -0
  78. package/starter/cloudrun-graphql/src/view/graphql/resolvers/greeting.resolver.ts +5 -0
  79. package/starter/cloudrun-graphql/src/view/graphql/resolvers.ts +6 -0
  80. package/starter/cloudrun-graphql/src/view/graphql/schema/schema.graphql +6 -0
  81. package/starter/cloudrun-graphql/src/view/graphql/schema.ts +7 -0
  82. package/starter/cloudrun-graphql/src/view/server.ts +45 -0
  83. package/starter/shared/.dockerignore +19 -0
  84. package/starter/shared/.gitignore_ +5 -0
  85. package/starter/shared/.gitlab-ci.yml +199 -0
  86. package/starter/shared/.mocha-junit-config.json +6 -0
  87. package/starter/shared/.mocharc.json +8 -0
  88. package/starter/shared/.nvmrc +1 -0
  89. package/starter/shared/Dockerfile +40 -0
  90. package/starter/shared/ci-branch-config/common.env +7 -0
  91. package/starter/shared/ci-branch-config/development.env +7 -0
  92. package/starter/shared/ci-branch-config/master.env +7 -0
  93. package/starter/shared/ci-branch-config/stage.env +7 -0
  94. package/starter/shared/docker-compose/docker-compose-entrypoint.sh +7 -0
  95. package/starter/shared/docker-compose/docker-compose.ci.yml +19 -0
  96. package/starter/shared/docker-compose/docker-compose.local.yml +5 -0
  97. package/starter/shared/docker-compose/docker-compose.override.yml +5 -0
  98. package/starter/shared/docker-compose/docker-compose.yml +8 -0
  99. package/starter/shared/jest.config.js +12 -0
  100. package/starter/shared/prettier.config.cjs +1 -0
  101. package/starter/shared/src/test/setup.ts +1 -0
  102. package/starter/shared/tsconfig.json +22 -0
  103. package/tsconfig.json +19 -0
  104. package/tsconfig.tsbuildinfo +1 -0
@@ -0,0 +1,118 @@
1
+ export class GraphQLStarter {
2
+ constructor() {
3
+ this.name = 'cloudrun-graphql';
4
+ }
5
+ setToolbelt(toolbelt) {
6
+ this.toolbelt = toolbelt;
7
+ return this;
8
+ }
9
+ install() {
10
+ if (this.toolbelt == null) {
11
+ throw new Error('No toolbelt');
12
+ }
13
+ const tb = this.toolbelt;
14
+ tb.copySharedAsset('.gitignore');
15
+ tb.copySharedAsset('.gitlab-ci.yml');
16
+ tb.copySharedAsset('.nvmrc');
17
+ tb.copySharedAsset('Dockerfile');
18
+ tb.copySharedAsset('.dockerignore');
19
+ tb.mkdir(tb.stringToPath(`${tb.destination}/ci-branch-config`));
20
+ tb.copySharedAsset('ci-branch-config/common.env');
21
+ tb.replaceInFile(`ci-branch-config/common.env`, '{{PROJECT_NAME}}', tb.projectName);
22
+ tb.copySharedAsset('ci-branch-config/development.env');
23
+ tb.replaceInFile(`ci-branch-config/development.env`, '{{PROJECT_NAME}}', tb.projectName);
24
+ tb.copySharedAsset('ci-branch-config/stage.env');
25
+ tb.replaceInFile(`ci-branch-config/stage.env`, '{{PROJECT_NAME}}', tb.projectName);
26
+ tb.copySharedAsset('ci-branch-config/master.env');
27
+ tb.replaceInFile(`ci-branch-config/master.env`, '{{PROJECT_NAME}}', tb.projectName);
28
+ tb.mkdir(tb.stringToPath(`${tb.destination}/docker-compose`));
29
+ tb.copySharedAsset('docker-compose/docker-compose-entrypoint.sh');
30
+ tb.copySharedAsset('docker-compose/docker-compose.ci.yml');
31
+ tb.copySharedAsset('docker-compose/docker-compose.local.yml');
32
+ tb.copySharedAsset('docker-compose/docker-compose.override.yml');
33
+ tb.symlink(`${tb.destination}/docker-compose/docker-compose.override.yml`, `${tb.destination}/docker-compose/docker-compose.local.yml`);
34
+ tb.copySharedAsset('docker-compose/docker-compose.yml');
35
+ tb.replaceInFile(`docker-compose/docker-compose.yml`, '{{PROJECT_NAME}}', tb.projectName);
36
+ tb.npm.iDev('typescript');
37
+ tb.npm.iDev('@types/node');
38
+ tb.npm.iDev('ts-node');
39
+ tb.npm.i('source-map-support');
40
+ tb.copySharedAsset('tsconfig.json');
41
+ tb.packageJson.addNpmScript('build:graphql-types', 'graphql-codegen --config codegen.yml');
42
+ tb.packageJson.addNpmScript('build:copy-schema', 'mkdir -p ./dist/view/graphql/schema && cp -r ./src/view/graphql/schema ./dist/view/graphql/schema');
43
+ tb.packageJson.addNpmScript('build', 'npm run build:graphql-types && npm run build:copy-schema && tsc');
44
+ tb.packageJson.addNpmScript('start', 'node -r source-map-support/register dist/index.js');
45
+ tb.npm.i('configuru');
46
+ tb.npm.i('pino');
47
+ tb.npm.iDev('pino-pretty');
48
+ tb.copyAsset('.env.jsonc');
49
+ tb.mkdir(tb.stringToPath(`${tb.destination}/src`));
50
+ tb.mkdir(tb.stringToPath(`${tb.destination}/src/view`));
51
+ tb.mkdir(tb.stringToPath(`${tb.destination}/src/domain`));
52
+ tb.mkdir(tb.stringToPath(`${tb.destination}/src/domain/errors`));
53
+ tb.mkdir(tb.stringToPath(`${tb.destination}/src/domain/ports`));
54
+ tb.mkdir(tb.stringToPath(`${tb.destination}/src/domain/utils`));
55
+ tb.mkdir(tb.stringToPath(`${tb.destination}/src/adapters`));
56
+ tb.copyAsset('src/domain/errors/codes.ts');
57
+ tb.copyAsset('src/domain/errors/errors.ts');
58
+ tb.copyAsset('src/domain/errors/errors.ts');
59
+ tb.copyAsset('src/domain/ports/logger.d.ts');
60
+ tb.copyAsset('src/adapters/pino.logger.ts');
61
+ tb.mkdir(tb.stringToPath(`${tb.destination}/src/view/graphql`));
62
+ tb.mkdir(tb.stringToPath(`${tb.destination}/src/view/graphql/resolvers`));
63
+ tb.mkdir(tb.stringToPath(`${tb.destination}/src/view/graphql/schema`));
64
+ tb.copyAsset('src/config.ts');
65
+ tb.copyAsset('src/index.ts');
66
+ tb.copyAsset('src/container.ts');
67
+ tb.copyAsset('src/context.ts');
68
+ tb.copyAsset('codegen.yml');
69
+ tb.copyAsset('src/view/controller.ts');
70
+ tb.copyAsset('src/view/server.ts');
71
+ tb.copyAsset('src/view/graphql/schema.ts');
72
+ tb.copyAsset('src/view/graphql/resolvers.ts');
73
+ tb.copyAsset('src/view/graphql/resolvers/greeting.resolver.ts');
74
+ tb.copyAsset('src/view/graphql/schema/schema.graphql');
75
+ tb.npm.iDev('mocha');
76
+ tb.npm.iDev('mocha-junit-reporter');
77
+ tb.npm.iDev('mocha-multi-reporters');
78
+ tb.npm.iDev('nyc');
79
+ tb.npm.iDev('tsx');
80
+ tb.npm.iDev('@types/mocha');
81
+ tb.npm.iDev('@istanbuljs/nyc-config-typescript');
82
+ tb.copySharedAsset('.mocharc.json', tb.destination);
83
+ tb.copySharedAsset('.mocha-junit-config.json', tb.destination);
84
+ tb.packageJson.setType('module');
85
+ tb.packageJson.addNpmScript('test', 'mocha');
86
+ tb.packageJson.addNpmScript('ci-test:no-coverage', 'npm run test -- --parallel=false -R mocha-multi-reporters --reporter-options configFile=.mocha-junit-config.json');
87
+ tb.packageJson.addNpmScript('ci-test', 'nyc -a -r cobertura --report-dir output npm run ci-test:no-coverage');
88
+ tb.mkdir(tb.stringToPath(`${tb.destination}/src/test`));
89
+ tb.copySharedAsset('src/test/setup.ts');
90
+ tb.copyAsset('src/test/helloWorld.test.ts');
91
+ tb.npm.i('@as-integrations/express5');
92
+ tb.npm.iDev('@ackee/styleguide-backend-config');
93
+ tb.npm.iDev('prettier');
94
+ tb.npm.iDev('eslint');
95
+ tb.npm.iDev('eslint-formatter-gitlab@^5.0.0');
96
+ tb.copyAsset('.eslint.tsconfig.json');
97
+ tb.copyAsset('.eslintrc.cjs');
98
+ tb.copySharedAsset('prettier.config.cjs');
99
+ tb.packageJson.addNpmScript('prettier', "prettier --check --write '**/*.{ts,js,json,md}'");
100
+ tb.packageJson.addNpmScript('lint', 'eslint --ext .ts --ext .graphql src -f codeframe --fix');
101
+ tb.packageJson.addNpmScript('codestyle', 'npm run prettier && npm run lint');
102
+ tb.packageJson.addNpmScript('ci-lint', 'npm run lint -- -f checkstyle -o ./output/checkstyle-result.xml');
103
+ tb.npm.i('@apollo/server');
104
+ tb.npm.iDev('@graphql-codegen/cli');
105
+ tb.npm.iDev('lodash');
106
+ tb.npm.iDev('@graphql-eslint/eslint-plugin');
107
+ tb.npm.iDev('@graphql-codegen/typescript');
108
+ tb.npm.iDev('@graphql-codegen/typescript-resolvers');
109
+ tb.npm.i('@graphql-tools/load-files');
110
+ tb.npm.i('@graphql-tools/merge');
111
+ tb.npm.i('@graphql-tools/schema');
112
+ tb.npm.i('graphql');
113
+ tb.npm.i('express');
114
+ tb.npm.iDev('@types/express');
115
+ tb.packageJson.runScript('build');
116
+ }
117
+ }
118
+ //# sourceMappingURL=GraphQLStarter.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"GraphQLStarter.js","sourceRoot":"","sources":["../../src/cloudrun-graphql/GraphQLStarter.ts"],"names":[],"mappings":"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"}
package/lib/types.js ADDED
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=types.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.js","sourceRoot":"","sources":["../src/types.ts"],"names":[],"mappings":""}
package/logo.png ADDED
Binary file
package/package.json ADDED
@@ -0,0 +1,42 @@
1
+ {
2
+ "name": "@ackee/create-node-app",
3
+ "version": "1.0.0",
4
+ "description": "",
5
+ "main": "index.js",
6
+ "type": "module",
7
+ "scripts": {
8
+ "build": "tsc",
9
+ "prettier:check": "prettier --ignore-path .gitignore --check '**/*.{ts,js,json,md}'",
10
+ "prettier:fix": "npm run prettier:check -- --write '**/*.{ts,js,json,md}'",
11
+ "lint:check": "eslint --ignore-path .gitignore 'src/**/*.ts' -f codeframe",
12
+ "lint:fix": "npm run lint:check -- --fix"
13
+ },
14
+ "repository": {
15
+ "type": "git",
16
+ "url": "git+https://github.com/AckeeCZ/create-node-app.git"
17
+ },
18
+ "engines": {
19
+ "node": ">=20.19.0"
20
+ },
21
+ "bin": {
22
+ "create-node-app": "bin/create-node-app.js"
23
+ },
24
+ "license": "MIT",
25
+ "bugs": {
26
+ "url": "https://github.com/AckeeCZ/create-node-app/issues"
27
+ },
28
+ "homepage": "https://github.com/AckeeCZ/create-node-app#readme",
29
+ "devDependencies": {
30
+ "@ackee/styleguide-backend-config": "^1.0.0",
31
+ "@types/lodash-es": "^4.17.12",
32
+ "@types/node": "^20.19.0",
33
+ "@types/yargs": "^17.0.33",
34
+ "prettier": "^2.8.1",
35
+ "typescript": "^5.4.5"
36
+ },
37
+ "dependencies": {
38
+ "source-map-support": "^0.5.21",
39
+ "yargs": "^18.0.0",
40
+ "lodash-es": "^4.17.21"
41
+ }
42
+ }
@@ -0,0 +1 @@
1
+ module.exports = require('@ackee/styleguide-backend-config/prettier')
@@ -0,0 +1,99 @@
1
+ import { CloudRunStarter } from './cloudrun/CloudRunStarter.js'
2
+ import { Npm } from './Npm.js'
3
+ import { Toolbelt } from './Toolbelt.js'
4
+ import * as path from 'path'
5
+ import { PackageJson } from './PackageJson.js'
6
+ import { GraphQLStarter } from './cloudrun-graphql/GraphQLStarter.js'
7
+ import { Path } from './types.js'
8
+ import yargs from 'yargs'
9
+ import { hideBin } from 'yargs/helpers'
10
+ import { logger } from './Logger.js'
11
+ import { Starter } from './Starter.js'
12
+
13
+ export class Bootstrap {
14
+ protected starters: Starter[] = [new CloudRunStarter(), new GraphQLStarter()]
15
+
16
+ public runCLI(args: string[]) {
17
+ const cli = yargs(hideBin(args))
18
+ .usage('create-node-app <starter> [options]')
19
+ .positional('starter', {
20
+ name: 'starter',
21
+ type: 'string',
22
+ required: true,
23
+ description: 'Which template to setup (required)',
24
+ choices: this.starters.map(starter => starter.name),
25
+ })
26
+ .option('dir', {
27
+ type: 'string',
28
+ alias: 'd',
29
+ default: './node-app',
30
+ description: 'Destination directory',
31
+ })
32
+ .option('project-name', {
33
+ type: 'string',
34
+ alias: 'n',
35
+ default: 'node-app',
36
+ description: 'Google Cloud project name',
37
+ })
38
+ .option('force', {
39
+ type: 'boolean',
40
+ alias: 'f',
41
+ default: false,
42
+ description:
43
+ "Overwrite existing destination directory if it's not empty",
44
+ })
45
+ .version('1.0.0')
46
+ .help()
47
+
48
+ const parsedArgs = cli.parseSync()
49
+ const starterArg = parsedArgs._[0]
50
+
51
+ const starter = this.starters.find(x => x.name === starterArg)
52
+ const destination = path.normalize(parsedArgs.dir) as Path
53
+
54
+ if (!starter) {
55
+ logger.info('Invalid starter')
56
+ cli.showHelp()
57
+ process.exit(1)
58
+ }
59
+
60
+ logger.info(`starter=${starter.name}, destination=${destination}`)
61
+
62
+ const npm = new Npm({ dir: destination })
63
+ const packageJson = new PackageJson(npm)
64
+ const toolbelt = new Toolbelt({
65
+ npm,
66
+ packageJson,
67
+ assetDirectory: path.join(
68
+ import.meta.dirname,
69
+ '..',
70
+ 'starter',
71
+ starter.name
72
+ ),
73
+ sharedDirectory: path.join(
74
+ import.meta.dirname,
75
+ '..',
76
+ 'starter',
77
+ 'shared'
78
+ ),
79
+ destination: destination,
80
+ projectName: parsedArgs.projectName,
81
+ })
82
+ starter.setToolbelt(toolbelt)
83
+
84
+ if (!toolbelt.isDirectoryEmpty(destination)) {
85
+ if (!parsedArgs.force) {
86
+ logger.info(
87
+ `Directory '${destination}' already exists and is not empty. Use --force or -f flag to overwrite the existing directory.`
88
+ )
89
+ process.exit(1)
90
+ } else {
91
+ logger.info(`Overwriting existing directory '${destination}'`)
92
+ }
93
+ }
94
+
95
+ toolbelt.mkdir(destination, { overwrite: parsedArgs.force })
96
+ toolbelt.npm.init()
97
+ starter.install()
98
+ }
99
+ }
package/src/Logger.ts ADDED
@@ -0,0 +1,11 @@
1
+ export const logger = {
2
+ info: (message: string) => {
3
+ console.log(message)
4
+ },
5
+ verbose: (message: string) => {
6
+ console.log(message)
7
+ },
8
+ error: (message: string) => {
9
+ console.log(message)
10
+ },
11
+ }
package/src/Npm.ts ADDED
@@ -0,0 +1,38 @@
1
+ import * as childProcess from 'child_process'
2
+ import { logger } from './Logger.js'
3
+ import { Path } from './types.js'
4
+
5
+ export class Npm {
6
+ public readonly dir: Path
7
+ constructor(settings?: { dir?: Path }) {
8
+ this.dir = settings?.dir as Path
9
+ }
10
+ public run(args: string[]) {
11
+ logger.info(`> npm ${args.join(' ')}`)
12
+ const result = this.dir
13
+ ? childProcess.spawnSync('npm', args, {
14
+ cwd: this.dir,
15
+ })
16
+ : childProcess.spawnSync('npm', args)
17
+ if ((result?.status ?? 0) > 0) {
18
+ logger.info(
19
+ `Failed npm command: npm ${args.join(' ')}. ${String(result.output)}`
20
+ )
21
+ }
22
+ }
23
+ public init() {
24
+ this.run(['init', '--yes'])
25
+ }
26
+
27
+ public i(module?: string) {
28
+ if (!module) {
29
+ return this.run(['i'])
30
+ }
31
+ const args = ['i', module]
32
+ this.run(args)
33
+ }
34
+ public iDev(module: string) {
35
+ const args = ['i', '-D', module]
36
+ this.run(args)
37
+ }
38
+ }
@@ -0,0 +1,47 @@
1
+ import * as path from 'path'
2
+ import * as fs from 'fs'
3
+ import * as lodash from 'lodash-es'
4
+ import { Npm } from './Npm.js'
5
+ import { logger } from './Logger.js'
6
+ import { Path } from './types.js'
7
+
8
+ export class PackageJson {
9
+ public readonly path: Path
10
+ protected npm: Npm
11
+ constructor(npm: Npm) {
12
+ let packagejsonPath = './package.json' as Path
13
+ if (npm.dir) {
14
+ packagejsonPath = path.normalize(`${npm.dir}/${packagejsonPath}`) as Path
15
+ }
16
+ this.path = packagejsonPath
17
+ this.npm = npm
18
+ }
19
+ public setType(type: 'module' | 'commonjs') {
20
+ this.mergeWith({
21
+ type,
22
+ })
23
+ }
24
+ public toJSON() {
25
+ return JSON.parse(fs.readFileSync(this.path, 'utf-8'))
26
+ }
27
+ public runScript(name: string) {
28
+ this.npm.run(['run', name])
29
+ }
30
+ public addNpmScript(name: string, command: string) {
31
+ this.mergeWith({
32
+ scripts: {
33
+ [name]: command,
34
+ },
35
+ })
36
+ }
37
+ // Updated package json using merge with given object
38
+ public mergeWith(partialWith: any) {
39
+ const json = lodash.merge(this.toJSON(), partialWith)
40
+ logger.info(`> package.json updated ${JSON.stringify(partialWith)}`)
41
+ fs.writeFileSync(
42
+ path.join(this.path),
43
+ JSON.stringify(json, null, 2),
44
+ 'utf-8'
45
+ )
46
+ }
47
+ }
package/src/Starter.ts ADDED
@@ -0,0 +1,7 @@
1
+ import { Toolbelt } from './Toolbelt.js'
2
+
3
+ export interface Starter {
4
+ readonly name: string
5
+ setToolbelt(toolbelt: Toolbelt): Starter
6
+ install(): void
7
+ }
@@ -0,0 +1,132 @@
1
+ import * as path from 'path'
2
+ import * as fs from 'fs'
3
+ import { PackageJson } from './PackageJson.js'
4
+ import { Npm } from './Npm.js'
5
+ import { logger } from './Logger.js'
6
+ import { Path } from './types.js'
7
+
8
+ export class Toolbelt {
9
+ public readonly npm: Npm
10
+ public readonly packageJson: PackageJson
11
+ readonly assetDirectory: string
12
+ readonly sharedDirectory: string
13
+ readonly destination: string
14
+ readonly projectName?: string
15
+ constructor(params: {
16
+ npm: Npm
17
+ packageJson: PackageJson
18
+ assetDirectory: string
19
+ sharedDirectory: string
20
+ destination: string
21
+ projectName?: string
22
+ }) {
23
+ this.npm = params.npm
24
+ this.packageJson = params.packageJson
25
+ this.assetDirectory = params.assetDirectory
26
+ this.sharedDirectory = params.sharedDirectory
27
+ this.destination = params.destination
28
+ this.projectName = params.projectName
29
+ }
30
+ public stringToPath(str: string) {
31
+ return path.normalize(str) as Path
32
+ }
33
+
34
+ public isDirectoryEmpty(dirpath: string): boolean {
35
+ const path = this.stringToPath(dirpath)
36
+
37
+ if (!fs.existsSync(path)) {
38
+ return true
39
+ }
40
+
41
+ const stat = fs.statSync(path)
42
+ if (!stat.isDirectory()) {
43
+ return true
44
+ }
45
+
46
+ const contents = fs.readdirSync(path)
47
+ return contents.length === 0
48
+ }
49
+
50
+ public mkdir(
51
+ dirpath: string,
52
+ option?: {
53
+ /** If exists, remove recursively first */
54
+ overwrite?: boolean
55
+ }
56
+ ) {
57
+ dirpath = this.stringToPath(dirpath)
58
+ const rootPath = ['.', './']
59
+ if (!rootPath.includes(dirpath)) {
60
+ if (fs.existsSync(dirpath) && option?.overwrite) {
61
+ fs.rmSync(dirpath, { recursive: true })
62
+ }
63
+ fs.mkdirSync(dirpath, { recursive: true })
64
+ }
65
+ }
66
+ /**
67
+ * Like cp, but second argument does not need to include file name
68
+ * the name is preserved.
69
+ */
70
+ public cpFile(a: string, b: string, option?: { destFileName?: string }) {
71
+ a = this.stringToPath(a)
72
+ b = this.stringToPath(b)
73
+ const file = path.basename(a)
74
+ this.cp(a, this.stringToPath(`${b}/${option?.destFileName ?? file}`))
75
+ }
76
+
77
+ public cp(a: string, b: string) {
78
+ a = this.stringToPath(a)
79
+ b = this.stringToPath(b)
80
+ logger.info(`> cp ${a} ${b}`)
81
+ fs.copyFileSync(a, b)
82
+ }
83
+ public copyAsset(name: string, destination?: string) {
84
+ let destinationName = name
85
+ if (path.basename(name) === '.gitignore') {
86
+ name = '.gitignore_'
87
+ destinationName = '.gitignore'
88
+ }
89
+ name = this.stringToPath(name)
90
+ destination = this.stringToPath(this.destination)
91
+ this.cpFile(`${this.assetDirectory}/${name}`, destination, {
92
+ destFileName: destinationName,
93
+ })
94
+ }
95
+ public copySharedAsset(name: string, destination?: string) {
96
+ let destinationName = name
97
+ if (path.basename(name) === '.gitignore') {
98
+ name = '.gitignore_'
99
+ destinationName = '.gitignore'
100
+ }
101
+ name = this.stringToPath(name)
102
+ destination = this.stringToPath(this.destination)
103
+ this.cpFile(`${this.sharedDirectory}/${name}`, destination, {
104
+ destFileName: destinationName,
105
+ })
106
+ }
107
+ public replaceInFile(
108
+ filePath: string,
109
+ placeholder: string,
110
+ replacement: string = 'REPLACEME'
111
+ ) {
112
+ filePath = this.stringToPath(`${this.destination}/${filePath}`)
113
+ let content = fs.readFileSync(filePath, 'utf8')
114
+ /* eslint-disable-next-line security/detect-non-literal-regexp */
115
+ content = content.replace(new RegExp(placeholder, 'g'), replacement)
116
+ fs.writeFileSync(filePath, content)
117
+ }
118
+ public symlink(linkName: string, linkedFile: string) {
119
+ linkName = this.stringToPath(linkName)
120
+ linkedFile = this.stringToPath(linkedFile)
121
+ logger.info(`> ln -s ${linkName} ${linkedFile}`)
122
+ try {
123
+ fs.symlinkSync(linkedFile, linkName)
124
+ } catch (error) {
125
+ if ('code' in error && error.code === 'EEXIST') {
126
+ // OK
127
+ } else {
128
+ throw error
129
+ }
130
+ }
131
+ }
132
+ }