diff --git a/create.js b/create.js index abafdf4..3f2615c 100644 --- a/create.js +++ b/create.js @@ -30,7 +30,7 @@ if (!fs.existsSync(uiDir)) { // copy folder content try { - fs.cpSync(`${rootDir}/source/shared/`, contextDir, { + fs.cpSync(`${rootDir}/source/__/`, contextDir, { force: true, recursive: true, }); diff --git a/source/shared/.editorconfig b/source/__/.editorconfig similarity index 100% rename from source/shared/.editorconfig rename to source/__/.editorconfig diff --git a/source/shared/.tsconfig.json b/source/__/.tsconfig.json similarity index 100% rename from source/shared/.tsconfig.json rename to source/__/.tsconfig.json diff --git a/source/shared/Jenkinsfile b/source/__/Jenkinsfile similarity index 100% rename from source/shared/Jenkinsfile rename to source/__/Jenkinsfile diff --git a/source/shared/gitignore b/source/__/gitignore similarity index 100% rename from source/shared/gitignore rename to source/__/gitignore diff --git a/source/shared/npmrc b/source/__/npmrc similarity index 100% rename from source/shared/npmrc rename to source/__/npmrc diff --git a/source/shared/package.json b/source/__/package.json similarity index 71% rename from source/shared/package.json rename to source/__/package.json index 8adb84e..b57f98c 100644 --- a/source/shared/package.json +++ b/source/__/package.json @@ -4,7 +4,9 @@ "author": "Roman Jaroš", "license": "ISC", "scripts": { - "dev": "pnpm -r dev" + "dev": "pnpm -r dev", + "test": "pnpm -r test", + "test:e2e": "pnpm -r test:e2e" }, "dependencies": {}, "devDependencies": { diff --git a/source/shared/pnpm-workspace.yaml b/source/__/pnpm-workspace.yaml similarity index 100% rename from source/shared/pnpm-workspace.yaml rename to source/__/pnpm-workspace.yaml diff --git a/source/shared/prettierrc b/source/__/prettierrc similarity index 100% rename from source/shared/prettierrc rename to source/__/prettierrc diff --git a/source/shared/sonar-project.properties b/source/__/sonar-project.properties similarity index 100% rename from source/shared/sonar-project.properties rename to source/__/sonar-project.properties diff --git a/source/api/.tsconfig.json b/source/api/.tsconfig.json index d70541b..452e276 100644 --- a/source/api/.tsconfig.json +++ b/source/api/.tsconfig.json @@ -1,42 +1,9 @@ { - "extends": "../../tsconfig.json", - "compilerOptions": { - "moduleResolution": "node", - "strict": false, - "outDir": "dist", - "rootDir": "./", - "baseUrl": "./", - "resolveJsonModule": true, - "paths": { - "config/*": [ - "./src/config/*" - ], - "database/*": [ - "./src/database/*" - ], - "model/*": [ - "./src/model/*" - ], - "controllers/*": [ - "./src/controllers/*" - ], - "utils/*": [ - "./src/utils/*" - ], - "types/*": [ - "./src/types/*" - ], - "plugins/*": [ - "./src/plugins/*" - ] - } - }, - "include": [ - "./**/*.ts", - "./**/*.tsx", - "./src/**/*.json" - ], - "exclude": [ - "node_modules" - ] + "extends": "../../tsconfig.json", + "compilerOptions": { + "module": "commonjs", + "target": "es2017", + "outDir": "./dist", + "baseUrl": "./" + } } \ No newline at end of file diff --git a/source/api/jest.config.js b/source/api/jest.config.js new file mode 100644 index 0000000..7ce6314 --- /dev/null +++ b/source/api/jest.config.js @@ -0,0 +1,15 @@ +/** @type {import('jest').Config} */ + +const config = { + moduleFileExtensions: ['js', 'json', 'ts'], + rootDir: 'src', + testRegex: '.*\\.spec\\.ts$', + transform: { + '^.+\\.(t|j)s$': 'ts-jest', + }, + collectCoverageFrom: ['**/*.(t|j)s'], + coverageDirectory: '../coverage', + testEnvironment: 'node', +}; + +module.exports = config; diff --git a/source/api/nest-cli.json b/source/api/nest-cli.json new file mode 100644 index 0000000..7bed251 --- /dev/null +++ b/source/api/nest-cli.json @@ -0,0 +1,16 @@ +{ + "$schema": "https://json.schemastore.org/nest-cli", + "collection": "@nestjs/schematics", + "sourceRoot": "src", + "compilerOptions": { + "deleteOutDir": true, + "plugins": [ + { + "name": "@nestjs/swagger/plugin", + "options": { + "introspectComments": true + } + } + ] + } +} \ No newline at end of file diff --git a/source/api/nodemon.json b/source/api/nodemon.json deleted file mode 100644 index b98a06d..0000000 --- a/source/api/nodemon.json +++ /dev/null @@ -1,5 +0,0 @@ -{ - "ignore": [ - "README" - ] -} \ No newline at end of file diff --git a/source/api/package.json b/source/api/package.json index b4b4877..b0e6f9b 100644 --- a/source/api/package.json +++ b/source/api/package.json @@ -1,38 +1,59 @@ { - "name": "$(appName)-api", - "version": "0.1.0", - "description": "", - "author": "Roman Jaroš ", - "license": "ISC", - "scripts": { - "dev": "nodemon --exec ts-node -r tsconfig-paths/register src/index.ts", - "build": "rimraf dist && tsc -p tsconfig.json && tsc-alias -p tsconfig.json" - }, - "devDependencies": { - "@types/hapi": "18.0.10", - "@types/hapi__nes": "11.0.7", - "@types/ramda": "0.28.0", - "@types/validator": "13.11.1", - "copyfiles": "2.4.1", - "nodemon": "3.0.1", - "rimraf": "5.0.1", - "ts-node": "10.9.1", - "tsc-alias": "1.8.7" - }, - "dependencies": { - "@hapi/boom": "10.0.1", - "@hapi/nes": "13.0.1", - "@hapi/hapi": "21.3.2", - "@hapi/inert": "7.1.0", - "@hapi/vision": "7.0.3", - "@prokyon/utils": "1.0.31", - "hapi-swagger": "17.1.0", - "joi": "17.10.1", - "typeorm": "0.3.17", - "sqlite3": "5.1.6" - }, - "peerDependencies": { - "ramda": "0.28.0", - "typescript": "5.2.2" - } + "name": "$(appName)-api", + "version": "0.1.0", + "description": "", + "author": "Roman Jaroš ", + "license": "ISC", + "scripts": { + "nest": "nest", + "build": "nest build", + "dev": "nest start --watch", + "dev:debug": "nest start --debug --watch", + "start": "node dist/main", + "format": "prettier --write \"src/**/*.ts\" \"test/**/*.ts\"", + "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/axios": "^3.0.0", + "@nestjs/common": "^10.2.5", + "@nestjs/typeorm": "10.0.0", + "@nestjs/config": "^3.1.0", + "@nestjs/core": "^10.2.5", + "@nestjs/swagger": "7.1.11", + "@nestjs/platform-express": "^10.2.5", + "axios": "^1.5.0", + "reflect-metadata": "^0.1.13", + "rxjs": "^7.8.1", + "sqlite3": "^5.1.6" + }, + "devDependencies": { + "@nestjs/cli": "10.1.17", + "@nestjs/mapped-types": "^2.0.2", + "@nestjs/schematics": "^10.0.2", + "@nestjs/testing": "^10.2.5", + "@types/express": "^4.17.17", + "@types/jest": "29.5.4", + "@types/node": "20.6.0", + "@types/supertest": "^2.0.12", + "@typescript-eslint/eslint-plugin": "^6.7.0", + "@typescript-eslint/parser": "^6.7.0", + "eslint-config-prettier": "^9.0.0", + "eslint-plugin-prettier": "^5.0.0", + "eslint": "^8.49.0", + "execa": "8.0.1", + "jest": "29.7.0", + "prettier": "^3.0.3", + "source-map-support": "^0.5.21", + "supertest": "^6.3.3", + "ts-jest": "29.1.1", + "ts-loader": "^9.4.4", + "ts-node": "^10.9.1", + "tsconfig-paths": "^4.2.0", + "typescript": "^5.2.2" + } } \ No newline at end of file diff --git a/source/api/src/app.module.ts b/source/api/src/app.module.ts new file mode 100644 index 0000000..7801d03 --- /dev/null +++ b/source/api/src/app.module.ts @@ -0,0 +1,24 @@ +// @ts-nocheck +import { Module } from '@nestjs/common'; +import { ConfigModule } from '@nestjs/config'; +import { TypeOrmModule } from '@nestjs/typeorm'; + +import { envs } from './config'; +import { HealthModule } from './common/health/health.module'; + +@Module({ + imports: [ + ConfigModule.forRoot({ + isGlobal: true, + load: [envs], + }), + TypeOrmModule.forRoot({ + type: 'sqlite', + database: `db.sqlite`, + entities: [], + logging: ['query'], + }), + HealthModule, + ], +}) +export class AppModule {} diff --git a/source/api/src/common/health/health.controller.spec.ts b/source/api/src/common/health/health.controller.spec.ts new file mode 100644 index 0000000..d3f3cee --- /dev/null +++ b/source/api/src/common/health/health.controller.spec.ts @@ -0,0 +1,14 @@ +// @ts-nocheck +import { HealthController } from './health.controller'; + +describe('AppController (e2e)', () => { + let healthController: HealthController; + + beforeEach(() => { + healthController = new HealthController(); + }); + + it('should return OK response', async () => { + expect(await healthController.status()).toEqual('OK'); + }); +}); diff --git a/source/api/src/common/health/health.controller.ts b/source/api/src/common/health/health.controller.ts new file mode 100644 index 0000000..b50f83d --- /dev/null +++ b/source/api/src/common/health/health.controller.ts @@ -0,0 +1,21 @@ +// @ts-nocheck +import { Controller, Get, Response } from '@nestjs/common'; +import { ApiResponse } from '@nestjs/swagger'; + +@Controller('health-check') +export class HealthController { + /** + * Health check endpoint + */ + @ApiResponse({ + status: 200, + description: 'API is online', + schema: { + default: 'OK', + }, + }) + @Get() + status() { + return 'OK'; + } +} diff --git a/source/api/src/common/health/health.module.ts b/source/api/src/common/health/health.module.ts new file mode 100644 index 0000000..a41565c --- /dev/null +++ b/source/api/src/common/health/health.module.ts @@ -0,0 +1,8 @@ +// @ts-nocheck +import { Module } from '@nestjs/common'; +import { HealthController } from './health.controller'; + +@Module({ + controllers: [HealthController], +}) +export class HealthModule {} diff --git a/source/api/src/config.ts b/source/api/src/config.ts new file mode 100644 index 0000000..4ce5ec3 --- /dev/null +++ b/source/api/src/config.ts @@ -0,0 +1,3 @@ +export const envs = () => { + return {}; +}; diff --git a/source/api/src/config/server.json b/source/api/src/config/server.json deleted file mode 100644 index 8ea5838..0000000 --- a/source/api/src/config/server.json +++ /dev/null @@ -1,4 +0,0 @@ -{ - "host": "localhost", - "port": 3000 -} \ No newline at end of file diff --git a/source/api/src/controllers/hey.ts b/source/api/src/controllers/hey.ts deleted file mode 100644 index 38f67b9..0000000 --- a/source/api/src/controllers/hey.ts +++ /dev/null @@ -1,18 +0,0 @@ -// @ts-nocheck -import { Request, ResponseToolkit, ServerRoute } from '@hapi/hapi'; - -const routers: ServerRoute[] = [ - { - path: '/hey', - method: 'GET', - options: { - description: 'Are u alive?', - tags: ['api'], - }, - handler: (request: Request, h: ResponseToolkit) => { - return h.response('I am here !').type('html/text').code(200); - }, - }, -]; - -export default routers; diff --git a/source/api/src/database/db.ts b/source/api/src/database/db.ts deleted file mode 100644 index b372672..0000000 --- a/source/api/src/database/db.ts +++ /dev/null @@ -1,13 +0,0 @@ -// @ts-nocheck -import { DataSource } from 'typeorm'; - -export const dbConnect = async () => { - const AppDataSource = new DataSource({ - type: 'sqlite', - database: `database.sqlite`, - entities: [], - synchronize: true, - logging: ['query'], - }); - return await AppDataSource.initialize(); -}; diff --git a/source/api/src/index.ts b/source/api/src/index.ts deleted file mode 100644 index 35dcbdf..0000000 --- a/source/api/src/index.ts +++ /dev/null @@ -1,51 +0,0 @@ -// @ts-nocheck -import hapi from '@hapi/hapi'; -import nes from '@hapi/nes'; - -import { host, port } from 'config/server.json'; -import { dbConnect } from 'database/db'; - -import swagger from 'plugins/swagger'; - -const server: hapi.Server = new hapi.Server({ - host: host, - port: port, - debug: { - log: ['error'], - request: ['error'], - }, - routes: { - validate: { - failAction: (request, h, err) => { - throw err; - }, - }, - }, -}); - -const start = async (): Promise => { - await swagger(server); - await server.register(nes); - - // v1 endpoints - await server.register(require('./plugins/controllers'), { routes: { prefix: '/api/v1' } }); - - // connect to database - await dbConnect() - .then(() => { - console.log('Data Source has been initialized!'); - }) - .catch((err) => { - console.error('Error during Data Source initialization', err); - }); - - await server.start(); - console.info(`Server running: ${server.info.uri}`); -}; - -process.on('unhandledRejection', (err) => { - console.info(err); - process.exit(1); -}); - -start(); diff --git a/source/api/src/main.ts b/source/api/src/main.ts new file mode 100644 index 0000000..c2f6499 --- /dev/null +++ b/source/api/src/main.ts @@ -0,0 +1,28 @@ +// @ts-nocheck +import { NestFactory } from '@nestjs/core'; +import { AppModule } from './app.module'; +import { SwaggerModule, DocumentBuilder } from '@nestjs/swagger'; + +async function bootstrap() { + const app = await NestFactory.create(AppModule); + + // swagger + SwaggerModule.setup( + 'api', + app, + SwaggerModule.createDocument( + app, + new DocumentBuilder() + .setTitle('$(AppName) API') + .setDescription('The $(AppName) API description.') + .setVersion('0.1.0') + .addServer('/api/v1') + .build(), + ), + ); + + app.setGlobalPrefix('/api/v1'); + + await app.listen(8080); +} +bootstrap(); diff --git a/source/api/src/plugins/controllers.ts b/source/api/src/plugins/controllers.ts deleted file mode 100644 index a2d14ea..0000000 --- a/source/api/src/plugins/controllers.ts +++ /dev/null @@ -1,11 +0,0 @@ -// @ts-nocheck -import { Server } from '@hapi/hapi'; - -import hey from 'controllers/hey'; - -export const plugin = { - name: 'Controllers', - register: async (server: Server) => { - server.route(hey); - }, -}; diff --git a/source/api/src/plugins/swagger.ts b/source/api/src/plugins/swagger.ts deleted file mode 100644 index beaead3..0000000 --- a/source/api/src/plugins/swagger.ts +++ /dev/null @@ -1,26 +0,0 @@ -// @ts-nocheck -import { Server } from '@hapi/hapi'; - -const version = require('../../package.json').version; - -export default async (server: Server) => { - return server.register([ - require('@hapi/inert'), - require('@hapi/vision'), - { - plugin: require('hapi-swagger'), - options: { - info: { - title: 'Many API', - version, - }, - swaggerUI: true, - documentationPage: true, - documentationPath: '/swagger', - pathPrefixSize: 3, - basePath: '/api/v1', - reuseDefinitions: false, - }, - }, - ]); -}; diff --git a/source/api/src/types/.gitkeep b/source/api/src/types/.gitkeep deleted file mode 100644 index e69de29..0000000 diff --git a/source/api/src/utils/params.ts b/source/api/src/utils/params.ts deleted file mode 100644 index 4615106..0000000 --- a/source/api/src/utils/params.ts +++ /dev/null @@ -1,23 +0,0 @@ -// @ts-nocheck -import { Request } from '@hapi/hapi'; - -export type RequestParams = Record; - -export const getParam = - (request: Request) => - (name: string): string | null => { - const params: RequestParams = { - ...(request.params as object), - ...(request.payload as object), - ...(request.query as object), - }; - const filteredParams: RequestParams = {} as any; - Object.keys(params).map((key) => { - const value = params[key]; - if (['undefined', 'null'].includes(value as string)) { - filteredParams[key] = null; - } - filteredParams[key] = value; - }); - return filteredParams[name] ?? null; - }; diff --git a/source/api/test/app.e2e-spec.ts b/source/api/test/app.e2e-spec.ts new file mode 100644 index 0000000..4f95cbf --- /dev/null +++ b/source/api/test/app.e2e-spec.ts @@ -0,0 +1,26 @@ +// @ts-nocheck +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 moduleRef: TestingModule = await Test.createTestingModule({ + imports: [AppModule], + }).compile(); + + app = moduleRef.createNestApplication(); + await app.init(); + }); + + it('/health-check (GET)', () => { + return request.agent(app.getHttpServer()).get('/health-check').expect(200).expect('OK'); + }); + + afterAll(async () => { + await app.close(); + }); +}); diff --git a/source/api/test/jest.config-e2e.js b/source/api/test/jest.config-e2e.js new file mode 100644 index 0000000..e3b1b91 --- /dev/null +++ b/source/api/test/jest.config-e2e.js @@ -0,0 +1,14 @@ +/** @type {import('jest').Config} */ + +const config = { + moduleFileExtensions: ['js', 'json', 'ts'], + rootDir: '.', + testEnvironment: 'node', + testRegex: '.e2e-spec.ts$', + transform: { + '^.+\\.(t|j)s$': 'ts-jest', + }, + moduleNameMapper: { '^uuid$': 'uuid' }, +}; + +module.exports = config; diff --git a/source/api/tsconfig.build.json b/source/api/tsconfig.build.json new file mode 100644 index 0000000..64f86c6 --- /dev/null +++ b/source/api/tsconfig.build.json @@ -0,0 +1,4 @@ +{ + "extends": "./tsconfig.json", + "exclude": ["node_modules", "test", "dist", "**/*spec.ts"] +}