Skip to content

Commit a0560ed

Browse files
author
thomas
committed
feat(generator): create custom generator to create new libs
1 parent 3eebb0e commit a0560ed

22 files changed

+3677
-844
lines changed

.eslintrc.json

+8-1
Original file line numberDiff line numberDiff line change
@@ -21,7 +21,9 @@
2121
],
2222
"@angular-eslint/no-host-metadata-property": [
2323
"error",
24-
{ "allowStatic": true }
24+
{
25+
"allowStatic": true
26+
}
2527
]
2628
}
2729
},
@@ -41,6 +43,11 @@
4143
"jest": true
4244
},
4345
"rules": {}
46+
},
47+
{
48+
"files": "*.json",
49+
"parser": "jsonc-eslint-parser",
50+
"rules": {}
4451
}
4552
]
4653
}

.vscode/settings.json

+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
{
2+
"eslint.validate": ["json"]
3+
}

libs/cli/.eslintrc.json

+25
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,25 @@
1+
{
2+
"extends": ["../../.eslintrc.json"],
3+
"ignorePatterns": ["!**/*"],
4+
"overrides": [
5+
{
6+
"files": ["*.ts", "*.tsx", "*.js", "*.jsx"],
7+
"rules": {}
8+
},
9+
{
10+
"files": ["*.ts", "*.tsx"],
11+
"rules": {}
12+
},
13+
{
14+
"files": ["*.js", "*.jsx"],
15+
"rules": {}
16+
},
17+
{
18+
"files": ["./package.json"],
19+
"parser": "jsonc-eslint-parser",
20+
"rules": {
21+
"@nx/nx-plugin-checks": "error"
22+
}
23+
}
24+
]
25+
}

libs/cli/README.md

+7
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
# cli
2+
3+
This library was generated with [Nx](https://nx.dev).
4+
5+
## Building
6+
7+
Run `nx build cli` to build the library.

libs/cli/generators.json

+9
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,9 @@
1+
{
2+
"generators": {
3+
"app": {
4+
"factory": "./src/generators/app/generator",
5+
"schema": "./src/generators/app/schema.json",
6+
"description": "app generator"
7+
}
8+
}
9+
}

libs/cli/package.json

+6
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
{
2+
"name": "@angular-challenges/cli",
3+
"version": "0.0.1",
4+
"type": "commonjs",
5+
"generators": "./generators.json"
6+
}

libs/cli/project.json

+48
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,48 @@
1+
{
2+
"name": "cli",
3+
"$schema": "../../node_modules/nx/schemas/project-schema.json",
4+
"sourceRoot": "libs/cli/src",
5+
"projectType": "library",
6+
"targets": {
7+
"build": {
8+
"executor": "@nx/js:tsc",
9+
"outputs": ["{options.outputPath}"],
10+
"options": {
11+
"outputPath": "dist/libs/cli",
12+
"main": "libs/cli/src/index.ts",
13+
"tsConfig": "libs/cli/tsconfig.lib.json",
14+
"assets": [
15+
"libs/cli/*.md",
16+
{
17+
"input": "./libs/cli/src",
18+
"glob": "**/!(*.ts)",
19+
"output": "./src"
20+
},
21+
{
22+
"input": "./libs/cli/src",
23+
"glob": "**/*.d.ts",
24+
"output": "./src"
25+
},
26+
{
27+
"input": "./libs/cli",
28+
"glob": "generators.json",
29+
"output": "."
30+
},
31+
{
32+
"input": "./libs/cli",
33+
"glob": "executors.json",
34+
"output": "."
35+
}
36+
]
37+
}
38+
},
39+
"lint": {
40+
"executor": "@nx/linter:eslint",
41+
"outputs": ["{options.outputFile}"],
42+
"options": {
43+
"lintFilePatterns": ["libs/cli/**/*.ts", "libs/cli/package.json"]
44+
}
45+
}
46+
},
47+
"tags": []
48+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
import { Component } from '@angular/core';
2+
3+
@Component({
4+
standalone: true,
5+
imports: [],
6+
selector: 'app-root',
7+
template: ``,
8+
styles: [''],
9+
})
10+
export class AppComponent {}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,33 @@
1+
<h1><%= title %></h1>
2+
3+
> Author: Thomas Laforge
4+
5+
<!-- TODO: add Information/Statement/Rules/Constraint/Steps -->
6+
7+
### Information
8+
9+
### Statement
10+
11+
### Step 1
12+
13+
### Step 2
14+
15+
### Constraints:
16+
17+
### Submitting your work
18+
19+
1. Fork the project
20+
2. clone it
21+
3. npm install
22+
4. `npx nx serve <%= projectName %>`
23+
5. _...work on it_
24+
6. Commit your work
25+
7. Submit a PR with a title beginning with **Answer:<%= challengeNumber %>** that I will review and other dev can review.
26+
27+
<a href="https://github.com/tomalaforge/angular-challenges/pulls?q=label%3A<%= challengeNumber %>+label%3Aanswer"><img src="https://img.shields.io/badge/-Solutions-green" alt="<%= projectName %>"/></a>
28+
29+
<a href='https://github.com/tomalaforge/angular-challenges/pulls?q=label%3A<%= challengeNumber %>+label%3A"answer+author"'><img src="https://img.shields.io/badge/-Author solution-important" alt="<%= projectName %> solution author"/></a>
30+
31+
<!-- <a href="{Blog post url}" target="_blank" rel="noopener noreferrer"><img src="https://img.shields.io/badge/-Blog post explanation-blue" alt="<%= projectName %> blog article"/></a> -->
32+
33+
_You can ask any question on_ <a href="https://twitter.com/laforge_toma" target="_blank" rel="noopener noreferrer"><img src="./../../logo/twitter.svg" height=20px alt="twitter"/></a>
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
import { render } from '@testing-library/angular';
2+
import { AppComponent } from './app.component';
3+
4+
describe('AppComponent', () => {
5+
test('...', async () => {
6+
await render(AppComponent);
7+
});
8+
});
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
import '@testing-library/jest-dom';
2+
import 'jest-preset-angular/setup-jest';
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
{
2+
"extends": "./tsconfig.json",
3+
"compilerOptions": {
4+
"outDir": "../../dist/out-tsc",
5+
"module": "commonjs",
6+
"types": ["jest", "node", "@testing-library/jest-dom"]
7+
},
8+
"files": ["src/test-setup.ts"],
9+
"include": [
10+
"jest.config.ts",
11+
"src/**/*.test.ts",
12+
"src/**/*.spec.ts",
13+
"src/**/*.d.ts"
14+
]
15+
}
+51
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,51 @@
1+
import {
2+
applicationGenerator,
3+
E2eTestRunner,
4+
UnitTestRunner,
5+
} from '@nx/angular/generators';
6+
import { formatFiles, generateFiles, names, Tree } from '@nx/devkit';
7+
import { Linter } from '@nx/linter';
8+
import { join } from 'path';
9+
import { getProjectDir } from '../../utils/normalize';
10+
import { Schema } from './schema';
11+
12+
export async function appGenerator(tree: Tree, options: Schema) {
13+
const { appDirectory } = getProjectDir(options.name, options.directory);
14+
15+
await applicationGenerator(tree, {
16+
...options,
17+
style: 'scss',
18+
routing: false,
19+
inlineStyle: true,
20+
inlineTemplate: true,
21+
prefix: 'app',
22+
unitTestRunner: options.addTest ? UnitTestRunner.Jest : UnitTestRunner.None,
23+
e2eTestRunner: E2eTestRunner.None,
24+
linter: Linter.EsLint,
25+
addTailwind: true,
26+
standalone: true,
27+
skipTests: true,
28+
});
29+
30+
generateFiles(tree, join(__dirname, 'files', 'app'), appDirectory, {
31+
tmpl: '',
32+
});
33+
tree.delete(join(appDirectory, './src/app/nx-welcome.component.ts'));
34+
35+
generateFiles(tree, join(__dirname, 'files', 'readme'), appDirectory, {
36+
tmpl: '',
37+
projectName: names(options.name).name,
38+
title: options.title,
39+
challengeNumber: options.challengeNumber,
40+
});
41+
42+
if (options.addTest) {
43+
generateFiles(tree, join(__dirname, 'files', 'test'), appDirectory, {
44+
tmpl: '',
45+
});
46+
}
47+
48+
await formatFiles(tree);
49+
}
50+
51+
export default appGenerator;
+9
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,9 @@
1+
export interface Schema {
2+
title: string;
3+
challengeNumber: number;
4+
name: string;
5+
directory?: string;
6+
addTest?: boolean;
7+
skipPackageJson?: boolean;
8+
rootProject?: boolean;
9+
}
+64
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,64 @@
1+
{
2+
"$schema": "http://json-schema.org/schema",
3+
"$id": "GeneratorNxApp",
4+
"title": "Creates an Angular application.",
5+
"description": "Creates an Angular application.",
6+
"type": "object",
7+
"cli": "nx",
8+
"properties": {
9+
"name": {
10+
"description": "The name of the application.",
11+
"type": "string",
12+
"$default": {
13+
"$source": "argv",
14+
"index": 0
15+
},
16+
"x-prompt": "What name would you like to use for the application?",
17+
"pattern": "^[a-zA-Z].*$"
18+
},
19+
"title": {
20+
"description": "Title of your challenge.",
21+
"type": "string",
22+
"$default": {
23+
"$source": "argv",
24+
"index": 1
25+
},
26+
"x-priority": "important"
27+
},
28+
"challengeNumber": {
29+
"description": "The number of your challenge.",
30+
"type": "number",
31+
"$default": {
32+
"$source": "argv",
33+
"index": 2
34+
},
35+
"x-priority": "important"
36+
},
37+
"directory": {
38+
"description": "The directory of the new application.",
39+
"type": "string",
40+
"x-priority": "important"
41+
},
42+
"addTest": {
43+
"description": "add spec files.",
44+
"type": "boolean",
45+
"default": true,
46+
"alias": "S"
47+
},
48+
"skipPackageJson": {
49+
"type": "boolean",
50+
"default": false,
51+
"description": "Do not add dependencies to `package.json`.",
52+
"x-priority": "internal"
53+
},
54+
"rootProject": {
55+
"description": "Create an application at the root of the workspace.",
56+
"type": "boolean",
57+
"default": false,
58+
"hidden": true,
59+
"x-priority": "internal"
60+
}
61+
},
62+
"additionalProperties": false,
63+
"required": ["name", "title", "challengeNumber"]
64+
}

libs/cli/src/index.ts

Whitespace-only changes.

libs/cli/src/utils/normalize.ts

+43
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,43 @@
1+
import { names } from '@nx/devkit';
2+
3+
export function normalizeDirectory(
4+
appName: string,
5+
directoryName: string
6+
): string {
7+
return directoryName
8+
? `${names(directoryName).fileName}/${names(appName).fileName}`
9+
: names(appName).fileName;
10+
}
11+
12+
export function normalizeProjectName(
13+
appName: string,
14+
directoryName: string
15+
): string {
16+
return normalizeDirectory(appName, directoryName).replace(/\//g, '-');
17+
}
18+
19+
export function extractLayoutDirectory(directory: string): {
20+
layoutDirectory: string;
21+
projectDirectory: string;
22+
} {
23+
if (directory) {
24+
directory = directory.startsWith('/') ? directory.substring(1) : directory;
25+
for (const dir of ['apps', 'libs', 'packages']) {
26+
if (directory.startsWith(dir + '/') || directory === dir) {
27+
return {
28+
layoutDirectory: dir,
29+
projectDirectory: directory.substring(dir.length + 1),
30+
};
31+
}
32+
}
33+
}
34+
return { layoutDirectory: null, projectDirectory: directory };
35+
}
36+
37+
export function getProjectDir(name: string, directory: string) {
38+
const { projectDirectory } = extractLayoutDirectory(directory);
39+
return {
40+
appProjectName: normalizeProjectName(name, projectDirectory),
41+
appDirectory: 'apps/' + normalizeDirectory(name, projectDirectory),
42+
};
43+
}

libs/cli/tsconfig.json

+13
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
{
2+
"extends": "../../tsconfig.base.json",
3+
"compilerOptions": {
4+
"module": "commonjs"
5+
},
6+
"files": [],
7+
"include": [],
8+
"references": [
9+
{
10+
"path": "./tsconfig.lib.json"
11+
}
12+
]
13+
}

libs/cli/tsconfig.lib.json

+10
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
{
2+
"extends": "./tsconfig.json",
3+
"compilerOptions": {
4+
"outDir": "../../dist/out-tsc",
5+
"declaration": true,
6+
"types": ["node"]
7+
},
8+
"include": ["src/**/*.ts", "../../apps/testing-modal/src/test-setup.ts"],
9+
"exclude": ["jest.config.ts", "src/**/*.spec.ts", "src/**/*.test.ts"]
10+
}

0 commit comments

Comments
 (0)