@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.
- package/.eslintrc.cjs +10 -0
- package/AUTHORS +3 -0
- package/LICENSE.txt +22 -0
- package/README.md +39 -0
- package/bin/create-node-app.js +6 -0
- package/lib/Bootstrap.js +79 -0
- package/lib/Bootstrap.js.map +1 -0
- package/lib/Logger.js +12 -0
- package/lib/Logger.js.map +1 -0
- package/lib/Npm.js +33 -0
- package/lib/Npm.js.map +1 -0
- package/lib/PackageJson.js +39 -0
- package/lib/PackageJson.js.map +1 -0
- package/lib/Starter.js +2 -0
- package/lib/Starter.js.map +1 -0
- package/lib/Toolbelt.js +102 -0
- package/lib/Toolbelt.js.map +1 -0
- package/lib/cloudrun/CloudRunStarter.js +126 -0
- package/lib/cloudrun/CloudRunStarter.js.map +1 -0
- package/lib/cloudrun-graphql/GraphQLStarter.js +118 -0
- package/lib/cloudrun-graphql/GraphQLStarter.js.map +1 -0
- package/lib/types.js +2 -0
- package/lib/types.js.map +1 -0
- package/logo.png +0 -0
- package/package.json +42 -0
- package/prettier.config.cjs +1 -0
- package/src/Bootstrap.ts +99 -0
- package/src/Logger.ts +11 -0
- package/src/Npm.ts +38 -0
- package/src/PackageJson.ts +47 -0
- package/src/Starter.ts +7 -0
- package/src/Toolbelt.ts +132 -0
- package/src/cloudrun/CloudRunStarter.ts +181 -0
- package/src/cloudrun-graphql/GraphQLStarter.ts +182 -0
- package/src/types.ts +1 -0
- package/starter/cloudrun/.env.jsonc +10 -0
- package/starter/cloudrun/.eslint.tsconfig.json +4 -0
- package/starter/cloudrun/.eslintrc.cjs +8 -0
- package/starter/cloudrun/README.md +69 -0
- package/starter/cloudrun/src/adapters/pino.logger.ts +44 -0
- package/starter/cloudrun/src/config.ts +22 -0
- package/starter/cloudrun/src/container.ts +18 -0
- package/starter/cloudrun/src/context.ts +39 -0
- package/starter/cloudrun/src/domain/errors/codes.ts +9 -0
- package/starter/cloudrun/src/domain/errors/errors.ts +25 -0
- package/starter/cloudrun/src/domain/health-check.service.ts +15 -0
- package/starter/cloudrun/src/domain/ports/logger.d.ts +21 -0
- package/starter/cloudrun/src/index.ts +17 -0
- package/starter/cloudrun/src/test/health-check.test.ts +25 -0
- package/starter/cloudrun/src/test/util/openapi-test.util.ts +71 -0
- package/starter/cloudrun/src/view/cli/README.md +17 -0
- package/starter/cloudrun/src/view/cli/cli.ts +94 -0
- package/starter/cloudrun/src/view/cli/openapi/generate.ts +64 -0
- package/starter/cloudrun/src/view/rest/controller/health-check.controller.ts +33 -0
- package/starter/cloudrun/src/view/rest/middleware/context-middleware.ts +28 -0
- package/starter/cloudrun/src/view/rest/middleware/error-handler.ts +60 -0
- package/starter/cloudrun/src/view/rest/middleware/request-logger.ts +37 -0
- package/starter/cloudrun/src/view/rest/request.d.ts +9 -0
- package/starter/cloudrun/src/view/rest/routes.ts +15 -0
- package/starter/cloudrun/src/view/rest/spec/openapi.yml +65 -0
- package/starter/cloudrun/src/view/rest/util/openapi.util.ts +310 -0
- package/starter/cloudrun/src/view/server.ts +25 -0
- package/starter/cloudrun-graphql/.env.jsonc +12 -0
- package/starter/cloudrun-graphql/.eslint.tsconfig.json +4 -0
- package/starter/cloudrun-graphql/.eslintrc.cjs +43 -0
- package/starter/cloudrun-graphql/README.md +53 -0
- package/starter/cloudrun-graphql/codegen.yml +11 -0
- package/starter/cloudrun-graphql/src/adapters/pino.logger.ts +44 -0
- package/starter/cloudrun-graphql/src/config.ts +21 -0
- package/starter/cloudrun-graphql/src/container.ts +15 -0
- package/starter/cloudrun-graphql/src/context.ts +39 -0
- package/starter/cloudrun-graphql/src/domain/errors/codes.ts +9 -0
- package/starter/cloudrun-graphql/src/domain/errors/errors.ts +25 -0
- package/starter/cloudrun-graphql/src/domain/ports/logger.d.ts +21 -0
- package/starter/cloudrun-graphql/src/index.ts +11 -0
- package/starter/cloudrun-graphql/src/test/helloWorld.test.ts +23 -0
- package/starter/cloudrun-graphql/src/view/controller.ts +42 -0
- package/starter/cloudrun-graphql/src/view/graphql/resolvers/greeting.resolver.ts +5 -0
- package/starter/cloudrun-graphql/src/view/graphql/resolvers.ts +6 -0
- package/starter/cloudrun-graphql/src/view/graphql/schema/schema.graphql +6 -0
- package/starter/cloudrun-graphql/src/view/graphql/schema.ts +7 -0
- package/starter/cloudrun-graphql/src/view/server.ts +45 -0
- package/starter/shared/.dockerignore +19 -0
- package/starter/shared/.gitignore_ +5 -0
- package/starter/shared/.gitlab-ci.yml +199 -0
- package/starter/shared/.mocha-junit-config.json +6 -0
- package/starter/shared/.mocharc.json +8 -0
- package/starter/shared/.nvmrc +1 -0
- package/starter/shared/Dockerfile +40 -0
- package/starter/shared/ci-branch-config/common.env +7 -0
- package/starter/shared/ci-branch-config/development.env +7 -0
- package/starter/shared/ci-branch-config/master.env +7 -0
- package/starter/shared/ci-branch-config/stage.env +7 -0
- package/starter/shared/docker-compose/docker-compose-entrypoint.sh +7 -0
- package/starter/shared/docker-compose/docker-compose.ci.yml +19 -0
- package/starter/shared/docker-compose/docker-compose.local.yml +5 -0
- package/starter/shared/docker-compose/docker-compose.override.yml +5 -0
- package/starter/shared/docker-compose/docker-compose.yml +8 -0
- package/starter/shared/jest.config.js +12 -0
- package/starter/shared/prettier.config.cjs +1 -0
- package/starter/shared/src/test/setup.ts +1 -0
- package/starter/shared/tsconfig.json +22 -0
- package/tsconfig.json +19 -0
- 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
package/lib/types.js.map
ADDED
|
@@ -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')
|
package/src/Bootstrap.ts
ADDED
|
@@ -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
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
package/src/Toolbelt.ts
ADDED
|
@@ -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
|
+
}
|