🎯 MapView v2.0 - Global Deployment Ready

 MAJOR FEATURES:
• Auto-zoom intelligence với smart bounds fitting
• Enhanced 3D GPS markers với pulsing effects
• Professional route display với 6-layer rendering
• Status-based parking icons với availability indicators
• Production-ready build optimizations

🗺️ AUTO-ZOOM FEATURES:
• Smart bounds fitting cho GPS + selected parking
• Adaptive padding (50px) cho visual balance
• Max zoom control (level 16) để tránh quá gần
• Dynamic centering khi không có selection

🎨 ENHANCED VISUALS:
• 3D GPS marker với multi-layer pulse effects
• Advanced parking icons với status colors
• Selection highlighting với animation
• Dimming system cho non-selected items

🛣️ ROUTE SYSTEM:
• OpenRouteService API integration
• Multi-layer route rendering (glow, shadow, main, animated)
• Real-time distance & duration calculation
• Visual route info trong popup

📱 PRODUCTION READY:
• SSR safe với dynamic imports
• Build errors resolved
• Global deployment via Vercel
• Optimized performance

🌍 DEPLOYMENT:
• Vercel: https://whatever-ctk2auuxr-phong12hexdockworks-projects.vercel.app
• Bundle size: 22.8 kB optimized
• Global CDN distribution
• HTTPS enabled

💾 VERSION CONTROL:
• MapView-v2.0.tsx backup created
• MAPVIEW_VERSIONS.md documentation
• Full version history tracking
This commit is contained in:
2025-07-20 19:52:16 +07:00
parent 3203463a6a
commit c65cc97a33
64624 changed files with 7199453 additions and 6462 deletions

View File

@@ -0,0 +1,122 @@
{
"$schema": "./node_modules/@angular-devkit/schematics/collection-schema.json",
"schematics": {
"application": {
"factory": "./lib/application/application.factory#main",
"description": "Create a Nest application.",
"schema": "./lib/application/schema.json"
},
"angular-app": {
"factory": "./lib/client-app/angular/angular.factory#main",
"description": "Create a new Angular application.",
"schema": "./lib/client-app/angular/schema.json"
},
"class": {
"factory": "./lib/class/class.factory#main",
"description": "Create a new class.",
"schema": "./lib/class/schema.json",
"aliases": ["cl"]
},
"controller": {
"factory": "./lib/controller/controller.factory#main",
"description": "Create a Nest controller.",
"schema": "./lib/controller/schema.json",
"aliases": ["co"]
},
"decorator": {
"factory": "./lib/decorator/decorator.factory#main",
"description": "Create a Nest decorator.",
"schema": "./lib/decorator/schema.json",
"aliases": ["d"]
},
"filter": {
"factory": "./lib/filter/filter.factory#main",
"description": "Create a Nest filter.",
"schema": "./lib/filter/schema.json",
"aliases": ["f"]
},
"gateway": {
"factory": "./lib/gateway/gateway.factory#main",
"description": "Create a Nest gateway.",
"schema": "./lib/gateway/schema.json",
"aliases": ["ga"]
},
"guard": {
"factory": "./lib/guard/guard.factory#main",
"description": "Create a Nest guard.",
"schema": "./lib/guard/schema.json",
"aliases": ["gu"]
},
"interceptor": {
"factory": "./lib/interceptor/interceptor.factory#main",
"description": "Create a Nest interceptor.",
"schema": "./lib/interceptor/schema.json",
"aliases": ["itc"]
},
"interface": {
"factory": "./lib/interface/interface.factory#main",
"description": "Create a Nest interface.",
"schema": "./lib/interface/schema.json",
"aliases": ["itf"]
},
"middleware": {
"factory": "./lib/middleware/middleware.factory#main",
"description": "Create a Nest middleware.",
"schema": "./lib/middleware/schema.json",
"aliases": ["mi"]
},
"module": {
"factory": "./lib/module/module.factory#main",
"description": "Create a Nest module.",
"schema": "./lib/module/schema.json",
"aliases": ["mo"]
},
"pipe": {
"factory": "./lib/pipe/pipe.factory#main",
"description": "Create a Nest pipe.",
"schema": "./lib/pipe/schema.json",
"aliases": ["pi"]
},
"provider": {
"factory": "./lib/provider/provider.factory#main",
"description": "Create a Nest provider.",
"schema": "./lib/provider/schema.json",
"aliases": ["pr"]
},
"service": {
"factory": "./lib/service/service.factory#main",
"description": "Create a Nest service.",
"schema": "./lib/service/schema.json",
"aliases": ["s"]
},
"resolver": {
"factory": "./lib/resolver/resolver.factory#main",
"description": "Create a Nest resolver.",
"schema": "./lib/resolver/schema.json",
"aliases": ["r"]
},
"configuration": {
"factory": "./lib/configuration/configuration.factory#main",
"description": "Create a Nest CLI configuration.",
"aliases": ["config"]
},
"library": {
"factory": "./lib/library/library.factory#main",
"description": "Create a Nest library (mono-repo).",
"schema": "./lib/library/schema.json",
"aliases": ["lib"]
},
"sub-app": {
"factory": "./lib/sub-app/sub-app.factory#main",
"description": "Create a Nest application (mono-repo).",
"schema": "./lib/sub-app/schema.json",
"aliases": ["app"]
},
"resource": {
"factory": "./lib/resource/resource.factory#main",
"description": "Create a Nest resource.",
"schema": "./lib/resource/schema.json",
"aliases": ["res"]
}
}
}

View File

@@ -0,0 +1,2 @@
export * from './utils';
export * from './lib/defaults';

18
backend/node_modules/@nestjs/schematics/dist/index.js generated vendored Normal file
View File

@@ -0,0 +1,18 @@
"use strict";
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
if (k2 === undefined) k2 = k;
var desc = Object.getOwnPropertyDescriptor(m, k);
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
desc = { enumerable: true, get: function() { return m[k]; } };
}
Object.defineProperty(o, k2, desc);
}) : (function(o, m, k, k2) {
if (k2 === undefined) k2 = k;
o[k2] = m[k];
}));
var __exportStar = (this && this.__exportStar) || function(m, exports) {
for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
};
Object.defineProperty(exports, "__esModule", { value: true });
__exportStar(require("./utils"), exports);
__exportStar(require("./lib/defaults"), exports);

View File

@@ -0,0 +1,3 @@
import { Rule } from '@angular-devkit/schematics';
import { ApplicationOptions } from './application.schema';
export declare function main(options: ApplicationOptions): Rule;

View File

@@ -0,0 +1,63 @@
"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
exports.main = main;
const core_1 = require("@angular-devkit/core");
const schematics_1 = require("@angular-devkit/schematics");
const path_1 = require("path");
const formatting_1 = require("../../utils/formatting");
const defaults_1 = require("../defaults");
function main(options) {
options.name = (0, formatting_1.normalizeToKebabOrSnakeCase)(options.name.toString());
const path = !options.directory || options.directory === 'undefined'
? options.name
: options.directory;
options = transform(options);
return (0, schematics_1.mergeWith)(generate(options, path));
}
function transform(options) {
const target = Object.assign({}, options);
target.author = !!target.author ? target.author : defaults_1.DEFAULT_AUTHOR;
target.description = !!target.description
? target.description
: defaults_1.DEFAULT_DESCRIPTION;
target.language = !!target.language ? target.language : defaults_1.DEFAULT_LANGUAGE;
target.name = resolvePackageName(target.name.toString());
target.version = !!target.version ? target.version : defaults_1.DEFAULT_VERSION;
target.specFileSuffix = (0, formatting_1.normalizeToKebabOrSnakeCase)(options.specFileSuffix || 'spec');
target.packageManager =
!target.packageManager || target.packageManager === 'undefined'
? 'npm'
: target.packageManager;
target.dependencies = !!target.dependencies ? target.dependencies : '';
target.devDependencies = !!target.devDependencies
? target.devDependencies
: '';
return target;
}
function resolvePackageName(path) {
const { base: baseFilename, dir: dirname } = (0, path_1.parse)(path);
if (baseFilename === '.') {
return (0, path_1.basename)(process.cwd());
}
if (dirname.match(/^@[^\s]/)) {
return `${dirname}/${baseFilename}`;
}
return baseFilename;
}
function generate(options, path) {
return (0, schematics_1.apply)((0, schematics_1.url)((0, core_1.join)('./files', options.language)), [
options.spec ? (0, schematics_1.noop)() : (0, schematics_1.filter)((path) => !path.endsWith('__specFileSuffix__.ts')),
options.spec
? (0, schematics_1.noop)()
: (0, schematics_1.filter)((path) => {
const languageExtension = options.language || 'ts';
const suffix = `__specFileSuffix__.${languageExtension}`;
return !path.endsWith(suffix);
}),
(0, schematics_1.template)({
...core_1.strings,
...options,
}),
(0, schematics_1.move)(path),
]);
}

View File

@@ -0,0 +1,12 @@
{
"presets": ["@babel/preset-env"],
"plugins": [
["@babel/plugin-proposal-decorators", { "legacy": true }],
[
"@babel/plugin-transform-runtime",
{
"regenerator": true
}
]
]
}

View File

@@ -0,0 +1,4 @@
{
"singleQuote": true,
"trailingComma": "all"
}

View File

@@ -0,0 +1,37 @@
# <%= name %>
## Description
<%= description %>
## Installation
```bash
$ <%= packageManager %> install
```
## Running the app
```bash
# development
$ <%= packageManager %> run start
# watch mode
$ <%= packageManager %> run start:dev
# production mode
<%= packageManager %> run start:prod
```
## Test
```bash
# unit tests
$ <%= packageManager %> run test
# e2e tests
$ <%= packageManager %> run test:e2e
# test coverage
$ <%= packageManager %> run test:cov
```

View File

@@ -0,0 +1,2 @@
require('@babel/register');
require('./src/main');

View File

@@ -0,0 +1,10 @@
{
"compilerOptions": {
"target": "ES6",
"experimentalDecorators": true
},
"exclude": [
"node_modules",
"dist"
]
}

View File

@@ -0,0 +1,6 @@
{
"$schema": "https://json.schemastore.org/nest-cli",
"language": "js",
"collection": "@nestjs/schematics",
"sourceRoot": "src"
}

View File

@@ -0,0 +1,6 @@
{
"watch": ["src"],
"ext": "js",
"ignore": ["src/**/*.spec.js"],
"exec": "node index --exec babel-node"
}

View File

@@ -0,0 +1,46 @@
{
"name": "<%= name %>",
"version": "<%= version %>",
"description": "<%= description %>",
"author": "<%= author %>",
"private": true,
"license": "UNLICENSED",
"scripts": {
"format": "prettier --write \"**/*.js\"",
"start": "babel-node index.js",
"start:dev": "nodemon",
"test": "jest",
"test:cov": "jest --coverage",
"test:e2e": "jest --config ./test/jest-e2e.json"
},
"dependencies": {
"@nestjs/common": "^10.0.0",
"@nestjs/core": "^10.0.0",
"@nestjs/platform-express": "^10.0.0",
"reflect-metadata": "^0.2.0",
"rxjs": "^7.2.0"
},
"devDependencies": {
"@nestjs/testing": "^10.0.0",
"@babel/core": "7.26.0",
"@babel/node": "7.26.0",
"@babel/plugin-proposal-decorators": "7.25.9",
"@babel/plugin-transform-runtime": "7.25.9",
"@babel/preset-env": "7.26.0",
"@babel/register": "7.25.9",
"@babel/runtime": "7.26.0",
"jest": "29.7.0",
"nodemon": "3.1.7",
"prettier": "3.3.3",
"supertest": "7.0.0"
},
"jest": {
"moduleFileExtensions": [
"js",
"json"
],
"rootDir": "src",
"testRegex": ".spec.js$",
"coverageDirectory": "../coverage"
}
}

View File

@@ -0,0 +1,22 @@
import { Test } from '@nestjs/testing';
import { AppController } from './app.controller';
import { AppService } from './app.service';
describe('AppController', () => {
let appController;
beforeEach(async () => {
const app = await Test.createTestingModule({
controllers: [AppController],
providers: [AppService],
}).compile();
appController = app.get(AppController);
});
describe('root', () => {
it('should return "Hello World!"', () => {
expect(appController.getHello()).toBe('Hello World!');
});
});
});

View File

@@ -0,0 +1,15 @@
import { Controller, Dependencies, Get } from '@nestjs/common';
import { AppService } from './app.service';
@Controller()
@Dependencies(AppService)
export class AppController {
constructor(appService) {
this.appService = appService;
}
@Get()
getHello() {
return this.appService.getHello();
}
}

View File

@@ -0,0 +1,10 @@
import { Module } from '@nestjs/common';
import { AppController } from './app.controller';
import { AppService } from './app.service';
@Module({
imports: [],
controllers: [AppController],
providers: [AppService],
})
export class AppModule {}

View File

@@ -0,0 +1,8 @@
import { Injectable } from '@nestjs/common';
@Injectable()
export class AppService {
getHello() {
return 'Hello World!';
}
}

View File

@@ -0,0 +1,8 @@
import { NestFactory } from '@nestjs/core';
import { AppModule } from './app.module';
async function bootstrap() {
const app = await NestFactory.create(AppModule);
await app.listen(process.env.PORT || 3000);
}
bootstrap();

View File

@@ -0,0 +1,23 @@
import { Test } from '@nestjs/testing';
import * as request from 'supertest';
import { AppModule } from '../src/app.module';
describe('AppController (e2e)', () => {
let app;
beforeEach(async () => {
const moduleFixture = await Test.createTestingModule({
imports: [AppModule],
}).compile();
app = moduleFixture.createNestApplication();
await app.init();
});
it('/ (GET)', () => {
return request(app.getHttpServer())
.get('/')
.expect(200)
.expect('Hello World!');
});
});

View File

@@ -0,0 +1,5 @@
{
"moduleFileExtensions": ["js", "json"],
"rootDir": ".",
"testRegex": ".e2e-spec.js$"
}

View File

@@ -0,0 +1,25 @@
module.exports = {
parser: '@typescript-eslint/parser',
parserOptions: {
project: 'tsconfig.json',
tsconfigRootDir: __dirname,
sourceType: 'module',
},
plugins: ['@typescript-eslint/eslint-plugin'],
extends: [
'plugin:@typescript-eslint/recommended',
'plugin:prettier/recommended',
],
root: true,
env: {
node: true,
jest: true,
},
ignorePatterns: ['.eslintrc.js'],
rules: {
'@typescript-eslint/interface-name-prefix': 'off',
'@typescript-eslint/explicit-function-return-type': 'off',
'@typescript-eslint/explicit-module-boundary-types': 'off',
'@typescript-eslint/no-explicit-any': 'off',
},
};

View File

@@ -0,0 +1,4 @@
{
"singleQuote": true,
"trailingComma": "all"
}

View File

@@ -0,0 +1,99 @@
<p align="center">
<a href="http://nestjs.com/" target="blank"><img src="https://nestjs.com/img/logo-small.svg" width="120" alt="Nest Logo" /></a>
</p>
[circleci-image]: https://img.shields.io/circleci/build/github/nestjs/nest/master?token=abc123def456
[circleci-url]: https://circleci.com/gh/nestjs/nest
<p align="center">A progressive <a href="http://nodejs.org" target="_blank">Node.js</a> framework for building efficient and scalable server-side applications.</p>
<p align="center">
<a href="https://www.npmjs.com/~nestjscore" target="_blank"><img src="https://img.shields.io/npm/v/@nestjs/core.svg" alt="NPM Version" /></a>
<a href="https://www.npmjs.com/~nestjscore" target="_blank"><img src="https://img.shields.io/npm/l/@nestjs/core.svg" alt="Package License" /></a>
<a href="https://www.npmjs.com/~nestjscore" target="_blank"><img src="https://img.shields.io/npm/dm/@nestjs/common.svg" alt="NPM Downloads" /></a>
<a href="https://circleci.com/gh/nestjs/nest" target="_blank"><img src="https://img.shields.io/circleci/build/github/nestjs/nest/master" alt="CircleCI" /></a>
<a href="https://coveralls.io/github/nestjs/nest?branch=master" target="_blank"><img src="https://coveralls.io/repos/github/nestjs/nest/badge.svg?branch=master#9" alt="Coverage" /></a>
<a href="https://discord.gg/G7Qnnhy" target="_blank"><img src="https://img.shields.io/badge/discord-online-brightgreen.svg" alt="Discord"/></a>
<a href="https://opencollective.com/nest#backer" target="_blank"><img src="https://opencollective.com/nest/backers/badge.svg" alt="Backers on Open Collective" /></a>
<a href="https://opencollective.com/nest#sponsor" target="_blank"><img src="https://opencollective.com/nest/sponsors/badge.svg" alt="Sponsors on Open Collective" /></a>
<a href="https://paypal.me/kamilmysliwiec" target="_blank"><img src="https://img.shields.io/badge/Donate-PayPal-ff3f59.svg" alt="Donate us"/></a>
<a href="https://opencollective.com/nest#sponsor" target="_blank"><img src="https://img.shields.io/badge/Support%20us-Open%20Collective-41B883.svg" alt="Support us"></a>
<a href="https://twitter.com/nestframework" target="_blank"><img src="https://img.shields.io/twitter/follow/nestframework.svg?style=social&label=Follow" alt="Follow us on Twitter"></a>
</p>
<!--[![Backers on Open Collective](https://opencollective.com/nest/backers/badge.svg)](https://opencollective.com/nest#backer)
[![Sponsors on Open Collective](https://opencollective.com/nest/sponsors/badge.svg)](https://opencollective.com/nest#sponsor)-->
## Description
[Nest](https://github.com/nestjs/nest) framework TypeScript starter repository.
## Project setup
```bash
$ <%= packageManager %> install
```
## Compile and run the project
```bash
# development
$ <%= packageManager %> run start
# watch mode
$ <%= packageManager %> run start:dev
# production mode
$ <%= packageManager %> run start:prod
```
## Run tests
```bash
# unit tests
$ <%= packageManager %> run test
# e2e tests
$ <%= packageManager %> run test:e2e
# test coverage
$ <%= packageManager %> run test:cov
```
## Deployment
When you're ready to deploy your NestJS application to production, there are some key steps you can take to ensure it runs as efficiently as possible. Check out the [deployment documentation](https://docs.nestjs.com/deployment) for more information.
If you are looking for a cloud-based platform to deploy your NestJS application, check out [Mau](https://mau.nestjs.com), our official platform for deploying NestJS applications on AWS. Mau makes deployment straightforward and fast, requiring just a few simple steps:
```bash
$ <%= packageManager %> install -g mau
$ mau deploy
```
With Mau, you can deploy your application in just a few clicks, allowing you to focus on building features rather than managing infrastructure.
## Resources
Check out a few resources that may come in handy when working with NestJS:
- Visit the [NestJS Documentation](https://docs.nestjs.com) to learn more about the framework.
- For questions and support, please visit our [Discord channel](https://discord.gg/G7Qnnhy).
- To dive deeper and get more hands-on experience, check out our official video [courses](https://courses.nestjs.com/).
- Deploy your application to AWS with the help of [NestJS Mau](https://mau.nestjs.com) in just a few clicks.
- Visualize your application graph and interact with the NestJS application in real-time using [NestJS Devtools](https://devtools.nestjs.com).
- Need help with your project (part-time to full-time)? Check out our official [enterprise support](https://enterprise.nestjs.com).
- To stay in the loop and get updates, follow us on [X](https://x.com/nestframework) and [LinkedIn](https://linkedin.com/company/nestjs).
- Looking for a job, or have a job to offer? Check out our official [Jobs board](https://jobs.nestjs.com).
## Support
Nest is an MIT-licensed open source project. It can grow thanks to the sponsors and support by the amazing backers. If you'd like to join them, please [read more here](https://docs.nestjs.com/support).
## Stay in touch
- Author - [Kamil Myśliwiec](https://twitter.com/kammysliwiec)
- Website - [https://nestjs.com](https://nestjs.com/)
- Twitter - [@nestframework](https://twitter.com/nestframework)
## License
Nest is [MIT licensed](https://github.com/nestjs/nest/blob/master/LICENSE).

View File

@@ -0,0 +1,8 @@
{
"$schema": "https://json.schemastore.org/nest-cli",
"collection": "@nestjs/schematics",
"sourceRoot": "src",
"compilerOptions": {
"deleteOutDir": true
}
}

View File

@@ -0,0 +1,69 @@
{
"name": "<%= name %>",
"version": "<%= version %>",
"description": "<%= description %>",
"author": "<%= author %>",
"private": true,
"license": "UNLICENSED",
"scripts": {
"build": "nest build",
"format": "prettier --write \"src/**/*.ts\" \"test/**/*.ts\"",
"start": "nest start",
"start:dev": "nest start --watch",
"start:debug": "nest start --debug --watch",
"start:prod": "node dist/main",
"lint": "eslint \"{src,apps,libs,test}/**/*.ts\" --fix",
"test": "jest",
"test:watch": "jest --watch",
"test:cov": "jest --coverage",
"test:debug": "node --inspect-brk -r tsconfig-paths/register -r ts-node/register node_modules/.bin/jest --runInBand",
"test:e2e": "jest --config ./test/jest-e2e.json"
},
"dependencies": {
"@nestjs/common": "^10.0.0",
"@nestjs/core": "^10.0.0",
"@nestjs/platform-express": "^10.0.0",
"reflect-metadata": "^0.2.0",
"rxjs": "^7.8.1"
},
"devDependencies": {
"@nestjs/cli": "^10.0.0",
"@nestjs/schematics": "^10.0.0",
"@nestjs/testing": "^10.0.0",
"@types/express": "^5.0.0",
"@types/jest": "^29.5.2",
"@types/node": "^20.3.1",
"@types/supertest": "^6.0.0",
"@typescript-eslint/eslint-plugin": "^8.0.0",
"@typescript-eslint/parser": "^8.0.0",
"eslint": "^8.0.0",
"eslint-config-prettier": "^9.0.0",
"eslint-plugin-prettier": "^5.0.0",
"jest": "^29.5.0",
"prettier": "^3.0.0",
"source-map-support": "^0.5.21",
"supertest": "^7.0.0",
"ts-jest": "^29.1.0",
"ts-loader": "^9.4.3",
"ts-node": "^10.9.1",
"tsconfig-paths": "^4.2.0",
"typescript": "^5.1.3"
},
"jest": {
"moduleFileExtensions": [
"js",
"json",
"ts"
],
"rootDir": "src",
"testRegex": ".*\\.spec\\.ts$",
"transform": {
"^.+\\.(t|j)s$": "ts-jest"
},
"collectCoverageFrom": [
"**/*.(t|j)s"
],
"coverageDirectory": "../coverage",
"testEnvironment": "node"
}
}

View File

@@ -0,0 +1,22 @@
import { Test, TestingModule } from '@nestjs/testing';
import { AppController } from './app.controller';
import { AppService } from './app.service';
describe('AppController', () => {
let appController: AppController;
beforeEach(async () => {
const app: TestingModule = await Test.createTestingModule({
controllers: [AppController],
providers: [AppService],
}).compile();
appController = app.get<AppController>(AppController);
});
describe('root', () => {
it('should return "Hello World!"', () => {
expect(appController.getHello()).toBe('Hello World!');
});
});
});

View File

@@ -0,0 +1,12 @@
import { Controller, Get } from '@nestjs/common';
import { AppService } from './app.service';
@Controller()
export class AppController {
constructor(private readonly appService: AppService) {}
@Get()
getHello(): string {
return this.appService.getHello();
}
}

View File

@@ -0,0 +1,10 @@
import { Module } from '@nestjs/common';
import { AppController } from './app.controller';
import { AppService } from './app.service';
@Module({
imports: [],
controllers: [AppController],
providers: [AppService],
})
export class AppModule {}

View File

@@ -0,0 +1,8 @@
import { Injectable } from '@nestjs/common';
@Injectable()
export class AppService {
getHello(): string {
return 'Hello World!';
}
}

View File

@@ -0,0 +1,8 @@
import { NestFactory } from '@nestjs/core';
import { AppModule } from './app.module';
async function bootstrap() {
const app = await NestFactory.create(AppModule);
await app.listen(process.env.PORT ?? 3000);
}
bootstrap();

View File

@@ -0,0 +1,24 @@
import { Test, TestingModule } from '@nestjs/testing';
import { INestApplication } from '@nestjs/common';
import * as request from 'supertest';
import { AppModule } from './../src/app.module';
describe('AppController (e2e)', () => {
let app: INestApplication;
beforeEach(async () => {
const moduleFixture: TestingModule = await Test.createTestingModule({
imports: [AppModule],
}).compile();
app = moduleFixture.createNestApplication();
await app.init();
});
it('/ (GET)', () => {
return request(app.getHttpServer())
.get('/')
.expect(200)
.expect('Hello World!');
});
});

View File

@@ -0,0 +1,9 @@
{
"moduleFileExtensions": ["js", "json", "ts"],
"rootDir": ".",
"testEnvironment": "node",
"testRegex": ".e2e-spec.ts$",
"transform": {
"^.+\\.(t|j)s$": "ts-jest"
}
}

View File

@@ -0,0 +1,4 @@
{
"extends": "./tsconfig.json",
"exclude": ["node_modules", "test", "dist", "**/*spec.ts"]
}

View File

@@ -0,0 +1,21 @@
{
"compilerOptions": {
"module": "commonjs",
"declaration": true,
"removeComments": true,
"emitDecoratorMetadata": true,
"experimentalDecorators": true,
"allowSyntheticDefaultImports": true,
"target": "ES2021",
"sourceMap": true,
"outDir": "./dist",
"baseUrl": "./",
"incremental": true,
"skipLibCheck": true,
"strictNullChecks": <%= strict %>,
"noImplicitAny": <%= strict %>,
"strictBindCallApply": <%= strict %>,
"forceConsistentCasingInFileNames": <%= strict %>,
"noFallthroughCasesInSwitch": <%= strict %>
}
}

View File

@@ -0,0 +1,71 @@
{
"$schema": "http://json-schema.org/schema",
"$id": "SchematicsNestApplication",
"title": "Nest Application Options Schema",
"type": "object",
"properties": {
"name": {
"oneOf": [
{ "type": "string" },
{ "type": "number" }
],
"description": "The name of the application.",
"$default": {
"$source": "argv",
"index": 0
},
"x-prompt": "What name would you like to use for the new project?"
},
"author": {
"type": "string",
"description": "Nest application author.",
"default": ""
},
"description": {
"type": "string",
"description": "Nest application description.",
"default": ""
},
"directory": {
"type": "string",
"description": "Nest application destination directory."
},
"strict": {
"type": "boolean",
"description": "With TypeScript strict mode.",
"default": false
},
"version": {
"type": "string",
"description": "Nest application version.",
"default": "0.0.1"
},
"language": {
"type": "string",
"description": "Nest application language."
},
"packageManager": {
"type": "string",
"description": "Nest application package manager."
},
"dependencies": {
"type": "string",
"description": "Nest application dependencies."
},
"devDependencies": {
"type": "string",
"description": "Nest application development dependencies."
},
"spec": {
"type": "boolean",
"default": true,
"description": "Specifies if a spec file is generated."
},
"specFileSuffix": {
"type": "string",
"default": "spec",
"description": "Specifies the file suffix of spec files."
}
},
"required": ["name"]
}

View File

@@ -0,0 +1,3 @@
import { Rule } from '@angular-devkit/schematics';
import { ClassOptions } from './class.schema';
export declare function main(options: ClassOptions): Rule;

View File

@@ -0,0 +1,51 @@
"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
exports.main = main;
const core_1 = require("@angular-devkit/core");
const schematics_1 = require("@angular-devkit/schematics");
const formatting_1 = require("../../utils/formatting");
const name_parser_1 = require("../../utils/name.parser");
const source_root_helpers_1 = require("../../utils/source-root.helpers");
const defaults_1 = require("../defaults");
function main(options) {
options = transform(options);
return (0, schematics_1.chain)([(0, source_root_helpers_1.mergeSourceRoot)(options), (0, schematics_1.mergeWith)(generate(options))]);
}
function transform(options) {
const target = Object.assign({}, options);
if (!target.name) {
throw new schematics_1.SchematicsException('Option (name) is required.');
}
const location = new name_parser_1.NameParser().parse(target);
target.name = (0, formatting_1.normalizeToKebabOrSnakeCase)(location.name);
target.specFileSuffix = (0, formatting_1.normalizeToKebabOrSnakeCase)(options.specFileSuffix || 'spec');
if (target.name.includes('.')) {
target.className = core_1.strings.classify(target.name).replace('.', '');
}
else {
target.className = target.name;
}
target.language =
target.language !== undefined ? target.language : defaults_1.DEFAULT_LANGUAGE;
target.path = (0, formatting_1.normalizeToKebabOrSnakeCase)(location.path);
target.path = target.flat
? target.path
: (0, core_1.join)(target.path, target.name);
return target;
}
function generate(options) {
return (context) => (0, schematics_1.apply)((0, schematics_1.url)((0, core_1.join)('./files', options.language)), [
options.spec
? (0, schematics_1.noop)()
: (0, schematics_1.filter)((path) => {
const languageExtension = options.language || 'ts';
const suffix = `.__specFileSuffix__.${languageExtension}`;
return !path.endsWith(suffix);
}),
(0, schematics_1.template)({
...core_1.strings,
...options,
}),
(0, schematics_1.move)(options.path),
])(context);
}

View File

@@ -0,0 +1,7 @@
import { <%= classify(className) %> } from './<%= name %>';
describe('<%= classify(className) %>', () => {
it('should be defined', () => {
expect(new <%= classify(className) %>()).toBeDefined();
});
});

View File

@@ -0,0 +1 @@
export class <%= classify(className) %> {}

View File

@@ -0,0 +1,7 @@
import { <%= classify(className) %> } from './<%= name %>';
describe('<%= classify(className) %>', () => {
it('should be defined', () => {
expect(new <%= classify(className) %>()).toBeDefined();
});
});

View File

@@ -0,0 +1 @@
export class <%= classify(className) %> {}

View File

@@ -0,0 +1,50 @@
{
"$schema": "http://json-schema.org/schema",
"$id": "SchematicsNestClass",
"title": "Nest Class Options Schema",
"type": "object",
"properties": {
"name": {
"type": "string",
"description": "The name of the class.",
"$default": {
"$source": "argv",
"index": 0
},
"x-prompt": "What name would you like to use for the class?"
},
"flat": {
"type": "boolean",
"default": true,
"description": "Flag to indicate if a directory is created."
},
"spec": {
"type": "boolean",
"default": true,
"description": "Specifies if a spec file is generated."
},
"specFileSuffix": {
"type": "string",
"default": "spec",
"description": "Specifies the file suffix of spec files."
},
"path": {
"type": "string",
"format": "path",
"description": "The path to create the class."
},
"language": {
"type": "string",
"description": "Nest class language (ts/js)."
},
"sourceRoot": {
"type": "string",
"description": "Nest controller source root directory."
},
"className": {
"type": "string",
"description": "Class name to be used internally."
}
},
"required": ["name"]
}

View File

@@ -0,0 +1,3 @@
import { Rule } from '@angular-devkit/schematics';
import { AngularOptions } from './angular.schema';
export declare function main(options: AngularOptions): Rule;

View File

@@ -0,0 +1,103 @@
"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
exports.main = main;
const core_1 = require("@angular-devkit/core");
const schematics_1 = require("@angular-devkit/schematics");
const path_1 = require("path");
const module_declarator_1 = require("../../../utils/module.declarator");
const module_finder_1 = require("../../../utils/module.finder");
const name_parser_1 = require("../../../utils/name.parser");
const source_root_helpers_1 = require("../../../utils/source-root.helpers");
function main(options) {
options = transform(options);
return (tree, context) => {
return (0, schematics_1.branchAndMerge)((0, schematics_1.chain)([
createAngularApplication(options),
(0, source_root_helpers_1.mergeSourceRoot)(options),
addDeclarationToModule(options),
addGlobalPrefix(),
(0, schematics_1.mergeWith)(generate(options)),
]))(tree, context);
};
}
function transform(source) {
const target = Object.assign({}, source);
target.directory = target.name ? core_1.strings.dasherize(target.name) : 'client';
target.name = 'Angular';
target.metadata = 'imports';
target.type = 'module';
const location = new name_parser_1.NameParser().parse(target);
target.name = core_1.strings.dasherize(location.name);
target.path = (0, path_1.join)(core_1.strings.dasherize(location.path), target.name);
return target;
}
function generate(options) {
return (context) => (0, schematics_1.apply)((0, schematics_1.url)('./files'), [
(0, schematics_1.template)({
...core_1.strings,
...options,
}),
(0, schematics_1.move)(options.path),
])(context);
}
function createAngularApplication(options) {
if (!options.initApp) {
return (0, schematics_1.noop)();
}
return (0, schematics_1.externalSchematic)('@schematics/angular', 'ng-new', {
name: options.directory,
version: '8.0.0',
});
}
function addDeclarationToModule(options) {
return (tree) => {
options.module = new module_finder_1.ModuleFinder(tree).find({
name: options.name,
path: options.path,
});
if (!options.module) {
return tree;
}
const content = tree.read(options.module).toString();
const declarator = new module_declarator_1.ModuleDeclarator();
const rootPath = `${options.directory}/dist/${options.directory}`;
const staticOptions = {
name: 'forRoot',
value: {
rootPath,
},
};
const declarationOptions = {
...options,
staticOptions,
};
tree.overwrite(options.module, declarator.declare(content, declarationOptions));
return tree;
};
}
function addGlobalPrefix() {
return (tree) => {
const mainFilePath = 'src/main.ts';
const fileRef = tree.get(mainFilePath);
if (!fileRef) {
return tree;
}
const ts = require('ts-morph');
const tsProject = new ts.Project({
manipulationSettings: {
indentationText: ts.IndentationText.TwoSpaces,
},
});
const tsFile = tsProject.addSourceFileAtPath(mainFilePath);
const bootstrapFunction = tsFile.getFunction('bootstrap');
const listenStatement = bootstrapFunction.getStatement(node => node.getText().includes('listen'));
const setPrefixStatement = bootstrapFunction.getStatement(node => node.getText().includes('setGlobalPrefix'));
if (!listenStatement || setPrefixStatement) {
return tree;
}
const listenExprIndex = listenStatement.getChildIndex();
bootstrapFunction.insertStatements(listenExprIndex, `app.setGlobalPrefix('api');`);
tree.overwrite(mainFilePath, tsFile.getFullText());
return tree;
};
}

View File

@@ -0,0 +1,4 @@
export const ANGULAR_MODULE_OPTIONS = 'ANGULAR_MODULE_OPTIONS';
export const DEFAULT_ROOT_PATH = 'client/dist';
export const DEFAULT_RENDER_PATH = '*';

View File

@@ -0,0 +1,41 @@
import { DynamicModule, Inject, Module, OnModuleInit } from '@nestjs/common';
import { HttpAdapterHost } from '@nestjs/core';
import {
ANGULAR_MODULE_OPTIONS,
DEFAULT_RENDER_PATH,
DEFAULT_ROOT_PATH,
} from './angular.constants';
import { angularProviders } from './angular.providers';
import { AngularModuleOptions } from './interfaces/angular-options.interface';
import { AbstractLoader } from './loaders/abstract.loader';
@Module({
providers: [...angularProviders],
})
export class AngularModule implements OnModuleInit {
constructor(
@Inject(ANGULAR_MODULE_OPTIONS)
private readonly ngOptions: AngularModuleOptions,
private readonly loader: AbstractLoader,
private readonly httpAdapterHost: HttpAdapterHost,
) {}
public static forRoot(options: AngularModuleOptions = {}): DynamicModule {
options.rootPath = options.rootPath || DEFAULT_ROOT_PATH;
options.renderPath = options.renderPath || DEFAULT_RENDER_PATH;
return {
module: AngularModule,
providers: [
{
provide: ANGULAR_MODULE_OPTIONS,
useValue: options,
},
],
};
}
public async onModuleInit() {
const httpAdapter = this.httpAdapterHost.httpAdapter;
this.loader.register(httpAdapter, this.ngOptions);
}
}

View File

@@ -0,0 +1,27 @@
import { Provider } from '@nestjs/common';
import { HttpAdapterHost } from '@nestjs/core';
import { AbstractLoader } from './loaders/abstract.loader';
import { ExpressLoader } from './loaders/express.loader';
import { FastifyLoader } from './loaders/fastify.loader';
import { NoopLoader } from './loaders/noop.loader';
export const angularProviders: Provider[] = [
{
provide: AbstractLoader,
useFactory: (httpAdapterHost: HttpAdapterHost) => {
if (!httpAdapterHost) {
return new NoopLoader();
}
const httpAdapter = httpAdapterHost.httpAdapter;
if (
httpAdapter &&
httpAdapter.constructor &&
httpAdapter.constructor.name === 'FastifyAdapter'
) {
return new FastifyLoader();
}
return new ExpressLoader();
},
inject: [HttpAdapterHost],
},
];

View File

@@ -0,0 +1,19 @@
import { Logger } from '@nestjs/common';
const MISSING_REQUIRED_DEPENDENCY = (name: string, reason: string) =>
`The "${name}" package is missing. Please, make sure to install this library ($ npm install ${name}) to take advantage of ${reason}.`;
const logger = new Logger('PackageLoader');
export function loadPackage<T = any>(
packageName: string,
context: string,
loaderFn?: () => T,
): T {
try {
return loaderFn ? loaderFn() : require(packageName);
} catch (e) {
logger.error(MISSING_REQUIRED_DEPENDENCY(packageName, context));
process.exit(1);
}
}

View File

@@ -0,0 +1,87 @@
export interface AngularModuleOptions {
/**
* Static files root directory.
* Default: "client/dist"
*/
rootPath?: string;
/**
* Path to render Angular app.
* Default: * (wildcard - all paths)
*/
renderPath?: string;
/**
* Serve static options (static files)
* Passed down to the underlying either `express.static` or `fastify-static.send`
*/
serveStaticOptions?: {
/**
* Enable or disable setting Cache-Control response header, defaults to true.
* Disabling this will ignore the immutable and maxAge options.
*/
cacheControl?: boolean;
/**
* Set how "dotfiles" are treated when encountered. A dotfile is a file or directory that begins with a dot (".").
* Note this check is done on the path itself without checking if the path actually exists on the disk.
* If root is specified, only the dotfiles above the root are checked
* (i.e. the root itself can be within a dotfile when when set to "deny").
* The default value is 'ignore'.
* 'allow' No special treatment for dotfiles
* 'deny' Send a 403 for any request for a dotfile
* 'ignore' Pretend like the dotfile does not exist and call next()
*/
dotfiles?: string;
/**
* Enable or disable etag generation, defaults to true.
*/
etag?: boolean;
/**
* Set file extension fallbacks. When set, if a file is not found, the given extensions
* will be added to the file name and search for.
* The first that exists will be served. Example: ['html', 'htm'].
* The default value is false.
*/
extensions?: string[];
/**
* Enable or disable the immutable directive in the Cache-Control response header.
* If enabled, the maxAge option should also be specified to enable caching.
* The immutable directive will prevent supported clients from making conditional
* requests during the life of the maxAge option to check if the file has changed.
*/
immutable?: boolean;
/**
* By default this module will send "index.html" files in response to a request on a directory.
* To disable this set false or to supply a new index pass a string or an array in preferred order.
*/
index?: boolean | string | string[];
/**
* Enable or disable Last-Modified header, defaults to true. Uses the file system's last modified value.
*/
lastModified?: boolean;
/**
* Provide a max-age in milliseconds for http caching, defaults to 0.
* This can also be a string accepted by the ms module.
*/
maxAge?: number | string;
/**
* Redirect to trailing "/" when the pathname is a dir. Defaults to true.
*/
redirect?: boolean;
/**
* Function to set custom headers on response. Alterations to the headers need to occur synchronously.
* The function is called as fn(res, path, stat), where the arguments are:
* res the response object
* path the file path that is being sent
* stat the stat object of the file that is being sent
*/
setHeaders?: (res: any, path: string, stat: any) => any;
};
}

View File

@@ -0,0 +1,16 @@
import { Injectable } from '@nestjs/common';
import { AbstractHttpAdapter } from '@nestjs/core';
import { join } from 'path';
import { AngularModuleOptions } from '../interfaces/angular-options.interface';
@Injectable()
export abstract class AbstractLoader {
public abstract register(
httpAdapter: AbstractHttpAdapter,
options: AngularModuleOptions,
);
public getIndexFilePath(clientPath: string): string {
return join(clientPath, 'index.html');
}
}

View File

@@ -0,0 +1,25 @@
import { Injectable } from '@nestjs/common';
import { AbstractHttpAdapter } from '@nestjs/core';
import { loadPackage } from '../angular.utils';
import { AngularModuleOptions } from '../interfaces/angular-options.interface';
import { AbstractLoader } from './abstract.loader';
@Injectable()
export class ExpressLoader extends AbstractLoader {
public register(
httpAdapter: AbstractHttpAdapter,
options: AngularModuleOptions,
) {
const app = httpAdapter.getInstance();
const express = loadPackage('express', 'AngularModule', () =>
require('express'),
);
const clientPath = options.rootPath;
const indexFilePath = this.getIndexFilePath(clientPath);
app.use(express.static(clientPath, options.serveStaticOptions));
app.get(options.renderPath, (req: any, res: any) =>
res.sendFile(indexFilePath),
);
}
}

View File

@@ -0,0 +1,34 @@
import { Injectable } from '@nestjs/common';
import { AbstractHttpAdapter } from '@nestjs/core';
import * as fs from 'fs';
import { loadPackage } from '../angular.utils';
import { AngularModuleOptions } from '../interfaces/angular-options.interface';
import { AbstractLoader } from './abstract.loader';
@Injectable()
export class FastifyLoader extends AbstractLoader {
public register(
httpAdapter: AbstractHttpAdapter,
options: AngularModuleOptions,
) {
const app = httpAdapter.getInstance();
const fastifyStatic = loadPackage('fastify-static', 'AngularModule', () =>
require('fastify-static'),
);
const { setHeaders, redirect, ...send } =
options.serveStaticOptions || ({} as any);
const clientPath = options.rootPath;
const indexFilePath = this.getIndexFilePath(clientPath);
app.register(fastifyStatic, {
root: clientPath,
setHeaders,
redirect,
send,
});
app.get(options.renderPath, (req: any, res: any) => {
const stream = fs.createReadStream(indexFilePath);
res.type('text/html').send(stream);
});
}
}

View File

@@ -0,0 +1,12 @@
import { Injectable } from '@nestjs/common';
import { AbstractHttpAdapter } from '@nestjs/core';
import { AngularModuleOptions } from '../interfaces/angular-options.interface';
import { AbstractLoader } from './abstract.loader';
@Injectable()
export class NoopLoader extends AbstractLoader {
public register(
httpAdapter: AbstractHttpAdapter,
options: AngularModuleOptions,
) {}
}

View File

@@ -0,0 +1,24 @@
{
"$schema": "http://json-schema.org/schema",
"$id": "SchematicsNestModule",
"title": "Nest Module Options Schema",
"type": "object",
"properties": {
"initApp": {
"type": "boolean",
"description": "Flag to skip the angular application generation.",
"default": false,
"x-prompt": "Would you like to initialize Angular application?"
},
"name": {
"type": "string",
"description": "The name of the application.",
"$default": {
"$source": "argv",
"index": 0
},
"x-prompt": "What name would you like to (or do you) use for Angular application?"
}
},
"required": ["name"]
}

View File

@@ -0,0 +1,3 @@
import { Rule } from '@angular-devkit/schematics';
import { ConfigurationOptions } from './configuration.schema';
export declare function main(options: ConfigurationOptions): Rule;

View File

@@ -0,0 +1,27 @@
"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
exports.main = main;
const core_1 = require("@angular-devkit/core");
const schematics_1 = require("@angular-devkit/schematics");
const defaults_1 = require("../defaults");
function main(options) {
return (0, schematics_1.mergeWith)(generate(transform(options)));
}
function transform(options) {
const target = Object.assign({}, options);
target.language =
target.language !== undefined ? target.language : defaults_1.DEFAULT_LANGUAGE;
target.collection =
target.collection !== undefined ? target.collection : '@nestjs/schematics';
return target;
}
function generate(options) {
const projectOrPath = options.project ?? '.';
return (0, schematics_1.apply)((0, schematics_1.url)((0, core_1.join)('./files', options.language)), [
(0, schematics_1.template)({
...core_1.strings,
...options,
}),
(0, schematics_1.move)(projectOrPath),
]);
}

View File

@@ -0,0 +1,6 @@
{
"$schema": "https://json.schemastore.org/nest-cli",
"language": "<%= language %>",
"collection": "<%= collection %>",
"sourceRoot": "src"
}

View File

@@ -0,0 +1,5 @@
{
"$schema": "https://json.schemastore.org/nest-cli",
"collection": "<%= collection %>",
"sourceRoot": "src"
}

View File

@@ -0,0 +1,3 @@
import { Rule } from '@angular-devkit/schematics';
import { ControllerOptions } from './controller.schema';
export declare function main(options: ControllerOptions): Rule;

View File

@@ -0,0 +1,72 @@
"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
exports.main = main;
const core_1 = require("@angular-devkit/core");
const schematics_1 = require("@angular-devkit/schematics");
const formatting_1 = require("../../utils/formatting");
const module_declarator_1 = require("../../utils/module.declarator");
const module_finder_1 = require("../../utils/module.finder");
const name_parser_1 = require("../../utils/name.parser");
const source_root_helpers_1 = require("../../utils/source-root.helpers");
const defaults_1 = require("../defaults");
const ELEMENT_METADATA = 'controllers';
const ELEMENT_TYPE = 'controller';
function main(options) {
options = transform(options);
return (tree, context) => {
return (0, schematics_1.branchAndMerge)((0, schematics_1.chain)([
(0, source_root_helpers_1.mergeSourceRoot)(options),
(0, schematics_1.mergeWith)(generate(options)),
addDeclarationToModule(options),
]))(tree, context);
};
}
function transform(source) {
const target = Object.assign({}, source);
target.metadata = ELEMENT_METADATA;
target.type = ELEMENT_TYPE;
const location = new name_parser_1.NameParser().parse(target);
target.name = (0, formatting_1.normalizeToKebabOrSnakeCase)(location.name);
target.path = (0, formatting_1.normalizeToKebabOrSnakeCase)(location.path);
target.language =
target.language !== undefined ? target.language : defaults_1.DEFAULT_LANGUAGE;
target.specFileSuffix = (0, formatting_1.normalizeToKebabOrSnakeCase)(source.specFileSuffix || 'spec');
target.path = target.flat
? target.path
: (0, core_1.join)(target.path, target.name);
return target;
}
function generate(options) {
return (context) => (0, schematics_1.apply)((0, schematics_1.url)((0, core_1.join)('./files', options.language)), [
options.spec
? (0, schematics_1.noop)()
: (0, schematics_1.filter)((path) => {
const languageExtension = options.language || 'ts';
const suffix = `.__specFileSuffix__.${languageExtension}`;
return !path.endsWith(suffix);
}),
(0, schematics_1.template)({
...core_1.strings,
...options,
}),
(0, schematics_1.move)(options.path),
])(context);
}
function addDeclarationToModule(options) {
return (tree) => {
if (options.skipImport !== undefined && options.skipImport) {
return tree;
}
options.module = new module_finder_1.ModuleFinder(tree).find({
name: options.name,
path: options.path,
});
if (!options.module) {
return tree;
}
const content = tree.read(options.module).toString();
const declarator = new module_declarator_1.ModuleDeclarator();
tree.overwrite(options.module, declarator.declare(content, options));
return tree;
};
}

View File

@@ -0,0 +1,18 @@
import { Test } from '@nestjs/testing';
import { <%= classify(name) %>Controller } from './<%= name %>.controller';
describe('<%= classify(name) %> Controller', () => {
let controller;
beforeEach(async () => {
const module = await Test.createTestingModule({
controllers: [<%= classify(name) %>Controller],
}).compile();
controller = module.get(<%= classify(name) %>Controller);
});
it('should be defined', () => {
expect(controller).toBeDefined();
});
});

View File

@@ -0,0 +1,4 @@
import { Controller } from '@nestjs/common';
@Controller('<%= dasherize(name) %>')
export class <%= classify(name) %>Controller {}

View File

@@ -0,0 +1,18 @@
import { Test, TestingModule } from '@nestjs/testing';
import { <%= classify(name) %>Controller } from './<%= name %>.controller';
describe('<%= classify(name) %>Controller', () => {
let controller: <%= classify(name) %>Controller;
beforeEach(async () => {
const module: TestingModule = await Test.createTestingModule({
controllers: [<%= classify(name) %>Controller],
}).compile();
controller = module.get<<%= classify(name) %>Controller>(<%= classify(name) %>Controller);
});
it('should be defined', () => {
expect(controller).toBeDefined();
});
});

View File

@@ -0,0 +1,4 @@
import { Controller } from '@nestjs/common';
@Controller('<%= dasherize(name) %>')
export class <%= classify(name) %>Controller {}

View File

@@ -0,0 +1,55 @@
{
"$schema": "http://json-schema.org/schema",
"$id": "SchematicsNestController",
"title": "Nest Controller Options Schema",
"type": "object",
"properties": {
"name": {
"type": "string",
"description": "The name of the controller.",
"$default": {
"$source": "argv",
"index": 0
},
"x-prompt": "What name would you like to use for the controller?"
},
"path": {
"type": "string",
"format": "path",
"description": "The path to create the controller."
},
"language": {
"type": "string",
"description": "Nest controller language (ts/js)."
},
"sourceRoot": {
"type": "string",
"description": "Nest controller source root directory."
},
"skipImport": {
"type": "boolean",
"description": "Flag to skip the module import.",
"default": false
},
"module": {
"type": "string",
"description": "Allows specification of the declaring module."
},
"flat": {
"type": "boolean",
"default": false,
"description": "Flag to indicate if a directory is created."
},
"spec": {
"type": "boolean",
"default": true,
"description": "Specifies if a spec file is generated."
},
"specFileSuffix": {
"type": "string",
"default": "spec",
"description": "Specifies the file suffix of spec files."
}
},
"required": ["name"]
}

View File

@@ -0,0 +1,3 @@
import { Rule } from '@angular-devkit/schematics';
import { DecoratorOptions } from './decorator.schema';
export declare function main(options: DecoratorOptions): Rule;

View File

@@ -0,0 +1,35 @@
"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
exports.main = main;
const core_1 = require("@angular-devkit/core");
const schematics_1 = require("@angular-devkit/schematics");
const formatting_1 = require("../../utils/formatting");
const name_parser_1 = require("../../utils/name.parser");
const source_root_helpers_1 = require("../../utils/source-root.helpers");
function main(options) {
options = transform(options);
return (0, schematics_1.chain)([(0, source_root_helpers_1.mergeSourceRoot)(options), (0, schematics_1.mergeWith)(generate(options))]);
}
function transform(options) {
const target = Object.assign({}, options);
if (!target.name) {
throw new schematics_1.SchematicsException('Option (name) is required.');
}
const location = new name_parser_1.NameParser().parse(target);
target.name = (0, formatting_1.normalizeToKebabOrSnakeCase)(location.name);
target.path = (0, formatting_1.normalizeToKebabOrSnakeCase)(location.path);
target.language = target.language !== undefined ? target.language : 'ts';
target.path = target.flat
? target.path
: (0, core_1.join)(target.path, target.name);
return target;
}
function generate(options) {
return (context) => (0, schematics_1.apply)((0, schematics_1.url)((0, core_1.join)('./files', options.language)), [
(0, schematics_1.template)({
...core_1.strings,
...options,
}),
(0, schematics_1.move)(options.path),
])(context);
}

View File

@@ -0,0 +1,3 @@
import { SetMetadata } from '@nestjs/common';
export const <%= classify(name) %> = (...args) => SetMetadata('<%= name %>', args);

View File

@@ -0,0 +1,3 @@
import { SetMetadata } from '@nestjs/common';
export const <%= classify(name) %> = (...args: string[]) => SetMetadata('<%= name %>', args);

View File

@@ -0,0 +1,36 @@
{
"$schema": "http://json-schema.org/schema",
"$id": "SchematicsNestDecorator",
"title": "Nest Decorator Options Schema",
"type": "object",
"properties": {
"name": {
"type": "string",
"description": "The name of the decorator.",
"$default": {
"$source": "argv",
"index": 0
},
"x-prompt": "What name would you like to use for the decorator?"
},
"path": {
"type": "string",
"format": "path",
"description": "The path to create the decorator."
},
"language": {
"type": "string",
"description": "Nest decorator language (ts/js)."
},
"sourceRoot": {
"type": "string",
"description": "Nest decorator source root directory."
},
"flat": {
"type": "boolean",
"default": true,
"description": "Flag to indicate if a directory is created."
}
},
"required": ["name"]
}

View File

@@ -0,0 +1,14 @@
export declare const DEFAULT_AUTHOR = "";
export declare const DEFAULT_DESCRIPTION = "";
export declare const DEFAULT_LANGUAGE = "ts";
export declare const DEFAULT_VERSION = "0.0.1";
export declare const DEFAULT_PATH_NAME = "src";
export declare const DEFAULT_LIB_PATH = "libs";
export declare const DEFAULT_APPS_PATH = "apps";
export declare const DEFAULT_APP_NAME = "app";
export declare const DEFAULT_DIR_ENTRY_APP = "main";
export declare const TEST_ENV = "test";
export declare const PROJECT_TYPE: {
LIBRARY: string;
APPLICATION: string;
};

View File

@@ -0,0 +1,17 @@
"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
exports.PROJECT_TYPE = exports.TEST_ENV = exports.DEFAULT_DIR_ENTRY_APP = exports.DEFAULT_APP_NAME = exports.DEFAULT_APPS_PATH = exports.DEFAULT_LIB_PATH = exports.DEFAULT_PATH_NAME = exports.DEFAULT_VERSION = exports.DEFAULT_LANGUAGE = exports.DEFAULT_DESCRIPTION = exports.DEFAULT_AUTHOR = void 0;
exports.DEFAULT_AUTHOR = '';
exports.DEFAULT_DESCRIPTION = '';
exports.DEFAULT_LANGUAGE = 'ts';
exports.DEFAULT_VERSION = '0.0.1';
exports.DEFAULT_PATH_NAME = 'src';
exports.DEFAULT_LIB_PATH = 'libs';
exports.DEFAULT_APPS_PATH = 'apps';
exports.DEFAULT_APP_NAME = 'app';
exports.DEFAULT_DIR_ENTRY_APP = 'main';
exports.TEST_ENV = 'test';
exports.PROJECT_TYPE = {
LIBRARY: 'library',
APPLICATION: 'application',
};

View File

@@ -0,0 +1,7 @@
import { <%= classify(name) %>Filter } from './<%= name %>.filter';
describe('<%= classify(name) %>Filter', () => {
it('should be defined', () => {
expect(new <%= classify(name) %>Filter()).toBeDefined();
});
});

View File

@@ -0,0 +1,6 @@
import { Catch } from '@nestjs/common';
@Catch()
export class <%= classify(name) %>Filter {
catch(exception, host) {}
}

View File

@@ -0,0 +1,7 @@
import { <%= classify(name) %>Filter } from './<%= name %>.filter';
describe('<%= classify(name) %>Filter', () => {
it('should be defined', () => {
expect(new <%= classify(name) %>Filter()).toBeDefined();
});
});

View File

@@ -0,0 +1,6 @@
import { ArgumentsHost, Catch, ExceptionFilter } from '@nestjs/common';
@Catch()
export class <%= classify(name) %>Filter<T> implements ExceptionFilter {
catch(exception: T, host: ArgumentsHost) {}
}

View File

@@ -0,0 +1,3 @@
import { Rule } from '@angular-devkit/schematics';
import { FilterOptions } from './filter.schema';
export declare function main(options: FilterOptions): Rule;

View File

@@ -0,0 +1,43 @@
"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
exports.main = main;
const core_1 = require("@angular-devkit/core");
const schematics_1 = require("@angular-devkit/schematics");
const formatting_1 = require("../../utils/formatting");
const name_parser_1 = require("../../utils/name.parser");
const source_root_helpers_1 = require("../../utils/source-root.helpers");
function main(options) {
options = transform(options);
return (0, schematics_1.chain)([(0, source_root_helpers_1.mergeSourceRoot)(options), (0, schematics_1.mergeWith)(generate(options))]);
}
function transform(options) {
const target = Object.assign({}, options);
if (!target.name) {
throw new schematics_1.SchematicsException('Option (name) is required.');
}
const location = new name_parser_1.NameParser().parse(target);
target.name = (0, formatting_1.normalizeToKebabOrSnakeCase)(location.name);
target.path = (0, formatting_1.normalizeToKebabOrSnakeCase)(location.path);
target.language = target.language !== undefined ? target.language : 'ts';
target.specFileSuffix = (0, formatting_1.normalizeToKebabOrSnakeCase)(options.specFileSuffix || 'spec');
target.path = target.flat
? target.path
: (0, core_1.join)(target.path, target.name);
return target;
}
function generate(options) {
return (context) => (0, schematics_1.apply)((0, schematics_1.url)((0, core_1.join)('./files', options.language)), [
options.spec
? (0, schematics_1.noop)()
: (0, schematics_1.filter)((path) => {
const languageExtension = options.language || 'ts';
const suffix = `.__specFileSuffix__.${languageExtension}`;
return !path.endsWith(suffix);
}),
(0, schematics_1.template)({
...core_1.strings,
...options,
}),
(0, schematics_1.move)(options.path),
])(context);
}

View File

@@ -0,0 +1,46 @@
{
"$schema": "http://json-schema.org/schema",
"$id": "SchematicsNestFilter",
"title": "Nest Filter Options Schema",
"type": "object",
"properties": {
"name": {
"type": "string",
"description": "The name of the filter.",
"$default": {
"$source": "argv",
"index": 0
},
"x-prompt": "What name would you like to use for the filter?"
},
"path": {
"type": "string",
"format": "path",
"description": "The path to create the filter."
},
"language": {
"type": "string",
"description": "Nest filter language (ts/js)."
},
"sourceRoot": {
"type": "string",
"description": "Nest filter source root directory."
},
"flat": {
"type": "boolean",
"default": true,
"description": "Flag to indicate if a directory is created."
},
"spec": {
"type": "boolean",
"default": true,
"description": "Specifies if a spec file is generated."
},
"specFileSuffix": {
"type": "string",
"default": "spec",
"description": "Specifies the file suffix of spec files."
}
},
"required": ["name"]
}

View File

@@ -0,0 +1,18 @@
import { Test } from '@nestjs/testing';
import { <%= classify(name) %>Gateway } from './<%= name %>.gateway';
describe('<%= classify(name) %>Gateway', () => {
let gateway;
beforeEach(async () => {
const module = await Test.createTestingModule({
providers: [<%= classify(name) %>Gateway],
}).compile();
gateway = module.get(<%= classify(name) %>Gateway);
});
it('should be defined', () => {
expect(gateway).toBeDefined();
});
});

View File

@@ -0,0 +1,9 @@
import { SubscribeMessage, WebSocketGateway } from '@nestjs/websockets';
@WebSocketGateway()
export class <%= classify(name) %>Gateway {
@SubscribeMessage('message')
handleMessage(client, payload) {
return 'Hello world!';
}
}

View File

@@ -0,0 +1,18 @@
import { Test, TestingModule } from '@nestjs/testing';
import { <%= classify(name) %>Gateway } from './<%= name %>.gateway';
describe('<%= classify(name) %>Gateway', () => {
let gateway: <%= classify(name) %>Gateway;
beforeEach(async () => {
const module: TestingModule = await Test.createTestingModule({
providers: [<%= classify(name) %>Gateway],
}).compile();
gateway = module.get<<%= classify(name) %>Gateway>(<%= classify(name) %>Gateway);
});
it('should be defined', () => {
expect(gateway).toBeDefined();
});
});

View File

@@ -0,0 +1,9 @@
import { SubscribeMessage, WebSocketGateway } from '@nestjs/websockets';
@WebSocketGateway()
export class <%= classify(name) %>Gateway {
@SubscribeMessage('message')
handleMessage(client: any, payload: any): string {
return 'Hello world!';
}
}

View File

@@ -0,0 +1,3 @@
import { Rule } from '@angular-devkit/schematics';
import { GatewayOptions } from './gateway.schema';
export declare function main(options: GatewayOptions): Rule;

View File

@@ -0,0 +1,72 @@
"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
exports.main = main;
const core_1 = require("@angular-devkit/core");
const schematics_1 = require("@angular-devkit/schematics");
const formatting_1 = require("../../utils/formatting");
const module_declarator_1 = require("../../utils/module.declarator");
const module_finder_1 = require("../../utils/module.finder");
const name_parser_1 = require("../../utils/name.parser");
const source_root_helpers_1 = require("../../utils/source-root.helpers");
function main(options) {
options = transform(options);
return (tree, context) => {
return (0, schematics_1.branchAndMerge)((0, schematics_1.chain)([
(0, source_root_helpers_1.mergeSourceRoot)(options),
addDeclarationToModule(options),
(0, schematics_1.mergeWith)(generate(options)),
]))(tree, context);
};
}
function transform(options) {
const target = Object.assign({}, options);
if (!target.name) {
throw new schematics_1.SchematicsException('Option (name) is required.');
}
target.metadata = 'providers';
target.type = 'gateway';
target.specFileSuffix = (0, formatting_1.normalizeToKebabOrSnakeCase)(options.specFileSuffix || 'spec');
const location = new name_parser_1.NameParser().parse(target);
target.name = (0, formatting_1.normalizeToKebabOrSnakeCase)(location.name);
target.path = (0, formatting_1.normalizeToKebabOrSnakeCase)(location.path);
target.language = target.language !== undefined ? target.language : 'ts';
target.path = target.flat
? target.path
: (0, core_1.join)(target.path, target.name);
return target;
}
function generate(options) {
return (context) => (0, schematics_1.apply)((0, schematics_1.url)((0, core_1.join)('./files', options.language)), [
options.spec ? (0, schematics_1.noop)() : (0, schematics_1.filter)((path) => !path.endsWith('.spec.ts')),
options.spec
? (0, schematics_1.noop)()
: (0, schematics_1.filter)((path) => {
const languageExtension = options.language || 'ts';
const suffix = `.__specFileSuffix__.${languageExtension}`;
return !path.endsWith(suffix);
}),
(0, schematics_1.template)({
...core_1.strings,
...options,
}),
(0, schematics_1.move)(options.path),
])(context);
}
function addDeclarationToModule(options) {
return (tree) => {
if (options.skipImport !== undefined && options.skipImport) {
return tree;
}
options.module = new module_finder_1.ModuleFinder(tree).find({
name: options.name,
path: options.path,
});
if (!options.module) {
return tree;
}
const content = tree.read(options.module).toString();
const declarator = new module_declarator_1.ModuleDeclarator();
tree.overwrite(options.module, declarator.declare(content, options));
return tree;
};
}

View File

@@ -0,0 +1,46 @@
{
"$schema": "http://json-schema.org/schema",
"$id": "SchematicsNestGateway",
"title": "Nest Gateway Options Schema",
"type": "object",
"properties": {
"name": {
"type": "string",
"description": "The name of the gateway.",
"$default": {
"$source": "argv",
"index": 0
},
"x-prompt": "What name would you like to use for the gateway?"
},
"path": {
"type": "string",
"format": "path",
"description": "The path to create the gateway."
},
"language": {
"type": "string",
"description": "Nest gateway language (ts/js)."
},
"sourceRoot": {
"type": "string",
"description": "Nest gateway source root directory."
},
"flat": {
"type": "boolean",
"default": true,
"description": "Flag to indicate if a directory is created."
},
"spec": {
"type": "boolean",
"default": true,
"description": "Specifies if a spec file is generated."
},
"specFileSuffix": {
"type": "string",
"default": "spec",
"description": "Specifies the file suffix of spec files."
}
},
"required": ["name"]
}

View File

@@ -0,0 +1,7 @@
import { <%= classify(name) %>Guard } from './<%= name %>.guard';
describe('<%= classify(name) %>Guard', () => {
it('should be defined', () => {
expect(new <%= classify(name) %>Guard()).toBeDefined();
});
});

View File

@@ -0,0 +1,8 @@
import { Injectable } from '@nestjs/common';
@Injectable()
export class <%= classify(name) %>Guard {
canActivate(context) {
return true;
}
}

View File

@@ -0,0 +1,7 @@
import { <%= classify(name) %>Guard } from './<%= name %>.guard';
describe('<%= classify(name) %>Guard', () => {
it('should be defined', () => {
expect(new <%= classify(name) %>Guard()).toBeDefined();
});
});

View File

@@ -0,0 +1,11 @@
import { CanActivate, ExecutionContext, Injectable } from '@nestjs/common';
import { Observable } from 'rxjs';
@Injectable()
export class <%= classify(name) %>Guard implements CanActivate {
canActivate(
context: ExecutionContext,
): boolean | Promise<boolean> | Observable<boolean> {
return true;
}
}

View File

@@ -0,0 +1,3 @@
import { Rule } from '@angular-devkit/schematics';
import { GuardOptions } from './guard.schema';
export declare function main(options: GuardOptions): Rule;

View File

@@ -0,0 +1,43 @@
"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
exports.main = main;
const core_1 = require("@angular-devkit/core");
const schematics_1 = require("@angular-devkit/schematics");
const formatting_1 = require("../../utils/formatting");
const name_parser_1 = require("../../utils/name.parser");
const source_root_helpers_1 = require("../../utils/source-root.helpers");
function main(options) {
options = transform(options);
return (0, schematics_1.chain)([(0, source_root_helpers_1.mergeSourceRoot)(options), (0, schematics_1.mergeWith)(generate(options))]);
}
function transform(options) {
const target = Object.assign({}, options);
if (!target.name) {
throw new schematics_1.SchematicsException('Option (name) is required.');
}
const location = new name_parser_1.NameParser().parse(target);
target.name = (0, formatting_1.normalizeToKebabOrSnakeCase)(location.name);
target.path = (0, formatting_1.normalizeToKebabOrSnakeCase)(location.path);
target.language = target.language !== undefined ? target.language : 'ts';
target.specFileSuffix = (0, formatting_1.normalizeToKebabOrSnakeCase)(options.specFileSuffix || 'spec');
target.path = target.flat
? target.path
: (0, core_1.join)(target.path, target.name);
return target;
}
function generate(options) {
return (context) => (0, schematics_1.apply)((0, schematics_1.url)((0, core_1.join)('./files', options.language)), [
options.spec
? (0, schematics_1.noop)()
: (0, schematics_1.filter)((path) => {
const languageExtension = options.language || 'ts';
const suffix = `.__specFileSuffix__.${languageExtension}`;
return !path.endsWith(suffix);
}),
(0, schematics_1.template)({
...core_1.strings,
...options,
}),
(0, schematics_1.move)(options.path),
])(context);
}

View File

@@ -0,0 +1,46 @@
{
"$schema": "http://json-schema.org/schema",
"$id": "SchematicsNestGuard",
"title": "Nest Guard Options Schema",
"type": "object",
"properties": {
"name": {
"type": "string",
"description": "The name of the guard.",
"$default": {
"$source": "argv",
"index": 0
},
"x-prompt": "What name would you like to use for the guard?"
},
"path": {
"type": "string",
"format": "path",
"description": "The path to create the guard."
},
"language": {
"type": "string",
"description": "Nest guard language (ts/js)."
},
"sourceRoot": {
"type": "string",
"description": "Nest guard source root directory."
},
"flat": {
"type": "boolean",
"default": true,
"description": "Flag to indicate if a directory is created."
},
"spec": {
"type": "boolean",
"default": true,
"description": "Specifies if a spec file is generated."
},
"specFileSuffix": {
"type": "string",
"default": "spec",
"description": "Specifies the file suffix of spec files."
}
},
"required": ["name"]
}

View File

@@ -0,0 +1,7 @@
import { <%= classify(name) %>Interceptor } from './<%= name %>.interceptor';
describe('<%= classify(name) %>Interceptor', () => {
it('should be defined', () => {
expect(new <%= classify(name) %>Interceptor()).toBeDefined();
});
});

View File

@@ -0,0 +1,8 @@
import { Injectable } from '@nestjs/common';
@Injectable()
export class <%= classify(name) %>Interceptor {
intercept(context, next) {
return next.handle();
}
}

View File

@@ -0,0 +1,7 @@
import { <%= classify(name) %>Interceptor } from './<%= name %>.interceptor';
describe('<%= classify(name) %>Interceptor', () => {
it('should be defined', () => {
expect(new <%= classify(name) %>Interceptor()).toBeDefined();
});
});

View File

@@ -0,0 +1,9 @@
import { CallHandler, ExecutionContext, Injectable, NestInterceptor } from '@nestjs/common';
import { Observable } from 'rxjs';
@Injectable()
export class <%= classify(name) %>Interceptor implements NestInterceptor {
intercept(context: ExecutionContext, next: CallHandler): Observable<any> {
return next.handle();
}
}

View File

@@ -0,0 +1,3 @@
import { Rule } from '@angular-devkit/schematics';
import { InterceptorOptions } from './interceptor.schema';
export declare function main(options: InterceptorOptions): Rule;

View File

@@ -0,0 +1,44 @@
"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
exports.main = main;
const core_1 = require("@angular-devkit/core");
const schematics_1 = require("@angular-devkit/schematics");
const formatting_1 = require("../../utils/formatting");
const name_parser_1 = require("../../utils/name.parser");
const source_root_helpers_1 = require("../../utils/source-root.helpers");
function main(options) {
options = transform(options);
return (0, schematics_1.chain)([(0, source_root_helpers_1.mergeSourceRoot)(options), (0, schematics_1.mergeWith)(generate(options))]);
}
function transform(options) {
const target = Object.assign({}, options);
if (!target.name) {
throw new schematics_1.SchematicsException('Option (name) is required.');
}
const location = new name_parser_1.NameParser().parse(target);
target.name = (0, formatting_1.normalizeToKebabOrSnakeCase)(location.name);
target.path = (0, formatting_1.normalizeToKebabOrSnakeCase)(location.path);
target.language = target.language !== undefined ? target.language : 'ts';
target.specFileSuffix = (0, formatting_1.normalizeToKebabOrSnakeCase)(options.specFileSuffix || 'spec');
target.path = target.flat
? target.path
: (0, core_1.join)(target.path, target.name);
return target;
}
function generate(options) {
return (context) => (0, schematics_1.apply)((0, schematics_1.url)((0, core_1.join)('./files', options.language)), [
options.spec ? (0, schematics_1.noop)() : (0, schematics_1.filter)((path) => !path.endsWith('.spec.ts')),
options.spec
? (0, schematics_1.noop)()
: (0, schematics_1.filter)((path) => {
const languageExtension = options.language || 'ts';
const suffix = `.__specFileSuffix__.${languageExtension}`;
return !path.endsWith(suffix);
}),
(0, schematics_1.template)({
...core_1.strings,
...options,
}),
(0, schematics_1.move)(options.path),
])(context);
}

View File

@@ -0,0 +1,46 @@
{
"$schema": "http://json-schema.org/schema",
"$id": "SchematicsNestInterceptor",
"title": "Nest Interceptor Options Schema",
"type": "object",
"properties": {
"name": {
"type": "string",
"description": "The name of the interceptor.",
"$default": {
"$source": "argv",
"index": 0
},
"x-prompt": "What name would you like to use for the interceptor?"
},
"path": {
"type": "string",
"format": "path",
"description": "The path to create the interceptor."
},
"language": {
"type": "string",
"description": "Nest interceptor language (ts/js)."
},
"sourceRoot": {
"type": "string",
"description": "Nest interceptor source root directory."
},
"flat": {
"type": "boolean",
"default": true,
"description": "Flag to indicate if a directory is created."
},
"spec": {
"type": "boolean",
"default": true,
"description": "Specifies if a spec file is generated."
},
"specFileSuffix": {
"type": "string",
"default": "spec",
"description": "Specifies the file suffix of spec files."
}
},
"required": ["name"]
}

Some files were not shown because too many files have changed in this diff Show More