typescript-express-starter 6.2.1 → 8.0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (222) hide show
  1. package/README.kr.md +55 -26
  2. package/README.md +54 -25
  3. package/bin/cli.js +4 -3
  4. package/lib/default/.swcrc +41 -0
  5. package/lib/default/nodemon.json +2 -2
  6. package/lib/default/package.json +12 -8
  7. package/lib/default/src/app.ts +1 -2
  8. package/lib/default/src/http/auth.http +12 -15
  9. package/lib/default/src/http/users.http +12 -16
  10. package/lib/default/src/index.ts +1 -0
  11. package/lib/default/src/middlewares/auth.middleware.ts +3 -3
  12. package/lib/default/src/models/users.model.ts +5 -5
  13. package/lib/default/src/server.ts +1 -2
  14. package/lib/default/src/services/auth.service.ts +5 -5
  15. package/lib/default/src/services/users.service.ts +3 -3
  16. package/lib/default/src/tests/auth.test.ts +4 -4
  17. package/lib/default/src/tests/users.test.ts +4 -4
  18. package/lib/default/src/utils/logger.ts +5 -5
  19. package/lib/default/tsconfig.json +2 -2
  20. package/lib/graphql/.dockerignore +18 -0
  21. package/lib/graphql/.editorconfig +9 -0
  22. package/lib/graphql/.env +1 -0
  23. package/lib/graphql/.eslintignore +1 -0
  24. package/lib/graphql/.eslintrc +18 -0
  25. package/lib/graphql/.huskyrc +5 -0
  26. package/lib/graphql/.lintstagedrc.json +5 -0
  27. package/lib/graphql/.prettierrc +8 -0
  28. package/lib/graphql/.swcrc +39 -0
  29. package/lib/graphql/.vscode/launch.json +35 -0
  30. package/lib/graphql/.vscode/settings.json +6 -0
  31. package/lib/graphql/Dockerfile +24 -0
  32. package/lib/graphql/Makefile +6 -0
  33. package/lib/graphql/docker-compose.yml +50 -0
  34. package/lib/graphql/ecosystem.config.js +59 -0
  35. package/lib/graphql/jest.config.js +12 -0
  36. package/lib/graphql/nginx.conf +40 -0
  37. package/lib/graphql/nodemon.json +12 -0
  38. package/lib/graphql/package.json +79 -0
  39. package/lib/graphql/src/app.ts +105 -0
  40. package/lib/graphql/src/configs/development.json +19 -0
  41. package/lib/graphql/src/configs/production.json +19 -0
  42. package/lib/graphql/src/configs/test.json +19 -0
  43. package/lib/graphql/src/databases/index.ts +24 -0
  44. package/lib/graphql/src/dtos/users.dto.ts +14 -0
  45. package/lib/{typeorm/src/entity → graphql/src/entities}/users.entity.ts +3 -3
  46. package/lib/graphql/src/exceptions/HttpException.ts +10 -0
  47. package/lib/graphql/src/http/auth.http +49 -0
  48. package/lib/graphql/src/http/users.http +78 -0
  49. package/lib/graphql/src/index.ts +1 -0
  50. package/lib/graphql/src/interfaces/auth.interface.ts +14 -0
  51. package/lib/graphql/src/interfaces/db.interface.ts +7 -0
  52. package/lib/graphql/src/interfaces/users.interface.ts +5 -0
  53. package/lib/graphql/src/middlewares/auth.middleware.ts +32 -0
  54. package/lib/graphql/src/middlewares/error.middleware.ts +17 -0
  55. package/lib/graphql/src/repositories/auth.repository.ts +61 -0
  56. package/lib/graphql/src/repositories/users.repository.ts +60 -0
  57. package/lib/graphql/src/resolvers/auth.resolver.ts +32 -0
  58. package/lib/graphql/src/resolvers/users.resolver.ts +47 -0
  59. package/lib/graphql/src/server.ts +14 -0
  60. package/lib/graphql/src/tests/auth.test.ts +52 -0
  61. package/lib/graphql/src/tests/index.test.ts +18 -0
  62. package/lib/graphql/src/tests/users.test.ts +71 -0
  63. package/lib/graphql/src/typedefs/users.type.ts +13 -0
  64. package/lib/graphql/src/utils/logger.ts +75 -0
  65. package/lib/graphql/src/utils/util.ts +19 -0
  66. package/lib/graphql/src/utils/validateEnv.ts +10 -0
  67. package/lib/graphql/tsconfig.json +39 -0
  68. package/lib/knex/.swcrc +39 -0
  69. package/lib/knex/knexfile.ts +6 -6
  70. package/lib/knex/nodemon.json +2 -2
  71. package/lib/knex/package.json +11 -8
  72. package/lib/knex/src/app.ts +2 -3
  73. package/lib/knex/src/databases/index.ts +6 -6
  74. package/lib/knex/src/http/auth.http +12 -15
  75. package/lib/knex/src/http/users.http +12 -16
  76. package/lib/knex/src/index.ts +1 -0
  77. package/lib/knex/src/middlewares/auth.middleware.ts +4 -4
  78. package/lib/knex/src/server.ts +1 -2
  79. package/lib/knex/src/services/auth.service.ts +5 -5
  80. package/lib/knex/src/services/users.service.ts +3 -3
  81. package/lib/knex/src/tests/auth.test.ts +2 -2
  82. package/lib/knex/src/tests/index.test.ts +1 -1
  83. package/lib/knex/src/tests/users.test.ts +2 -2
  84. package/lib/knex/src/utils/logger.ts +5 -5
  85. package/lib/knex/tsconfig.json +2 -2
  86. package/lib/mongoose/.swcrc +39 -0
  87. package/lib/mongoose/docker-compose.yml +0 -4
  88. package/lib/mongoose/nodemon.json +2 -2
  89. package/lib/mongoose/package.json +12 -8
  90. package/lib/mongoose/src/app.ts +2 -3
  91. package/lib/mongoose/src/configs/development.json +1 -1
  92. package/lib/mongoose/src/configs/production.json +1 -1
  93. package/lib/mongoose/src/configs/test.json +1 -1
  94. package/lib/mongoose/src/http/auth.http +32 -0
  95. package/lib/mongoose/src/http/users.http +34 -0
  96. package/lib/mongoose/src/index.ts +1 -0
  97. package/lib/mongoose/src/middlewares/auth.middleware.ts +3 -3
  98. package/lib/mongoose/src/server.ts +1 -2
  99. package/lib/mongoose/src/services/auth.service.ts +5 -5
  100. package/lib/mongoose/src/services/users.service.ts +3 -3
  101. package/lib/mongoose/src/utils/logger.ts +5 -5
  102. package/lib/mongoose/tsconfig.json +2 -2
  103. package/lib/prisma/.env +1 -1
  104. package/lib/prisma/.swcrc +37 -0
  105. package/lib/prisma/docker-compose.yml +1 -6
  106. package/lib/prisma/nodemon.json +2 -2
  107. package/lib/prisma/package.json +14 -10
  108. package/lib/prisma/src/app.ts +2 -3
  109. package/lib/prisma/src/http/auth.http +12 -15
  110. package/lib/prisma/src/http/users.http +12 -16
  111. package/lib/prisma/src/index.ts +1 -0
  112. package/lib/prisma/src/middlewares/auth.middleware.ts +4 -4
  113. package/lib/prisma/src/server.ts +1 -2
  114. package/lib/prisma/src/services/auth.service.ts +5 -5
  115. package/lib/prisma/src/services/users.service.ts +3 -3
  116. package/lib/prisma/src/utils/logger.ts +5 -5
  117. package/lib/prisma/tsconfig.json +2 -2
  118. package/lib/routing-controllers/.swcrc +37 -0
  119. package/lib/routing-controllers/nodemon.json +2 -2
  120. package/lib/routing-controllers/package.json +12 -9
  121. package/lib/routing-controllers/src/app.ts +3 -4
  122. package/lib/routing-controllers/src/http/auth.http +12 -15
  123. package/lib/routing-controllers/src/http/users.http +12 -16
  124. package/lib/routing-controllers/src/index.ts +1 -0
  125. package/lib/routing-controllers/src/middlewares/auth.middleware.ts +3 -3
  126. package/lib/routing-controllers/src/middlewares/validation.middleware.ts +10 -2
  127. package/lib/routing-controllers/src/models/users.model.ts +5 -5
  128. package/lib/routing-controllers/src/server.ts +1 -3
  129. package/lib/routing-controllers/src/services/auth.service.ts +5 -5
  130. package/lib/routing-controllers/src/services/users.service.ts +3 -3
  131. package/lib/routing-controllers/src/utils/logger.ts +5 -5
  132. package/lib/routing-controllers/tsconfig.json +2 -2
  133. package/lib/sequelize/.swcrc +39 -0
  134. package/lib/sequelize/docker-compose.yml +1 -6
  135. package/lib/sequelize/nodemon.json +2 -2
  136. package/lib/sequelize/package.json +12 -8
  137. package/lib/sequelize/src/app.ts +1 -2
  138. package/lib/sequelize/src/configs/development.json +2 -2
  139. package/lib/sequelize/src/configs/production.json +1 -1
  140. package/lib/sequelize/src/configs/test.json +1 -1
  141. package/lib/sequelize/src/http/auth.http +32 -0
  142. package/lib/sequelize/src/http/users.http +34 -0
  143. package/lib/sequelize/src/index.ts +1 -0
  144. package/lib/sequelize/src/middlewares/auth.middleware.ts +4 -4
  145. package/lib/sequelize/src/server.ts +1 -2
  146. package/lib/sequelize/src/services/auth.service.ts +5 -5
  147. package/lib/sequelize/src/services/users.service.ts +3 -3
  148. package/lib/sequelize/src/utils/logger.ts +5 -5
  149. package/lib/sequelize/tsconfig.json +2 -2
  150. package/lib/starter.js +4 -6
  151. package/lib/typegoose/.dockerignore +18 -0
  152. package/lib/typegoose/.editorconfig +9 -0
  153. package/lib/typegoose/.env.development.local +18 -0
  154. package/lib/typegoose/.env.production.local +18 -0
  155. package/lib/typegoose/.env.test.local +18 -0
  156. package/lib/typegoose/.eslintignore +1 -0
  157. package/lib/typegoose/.eslintrc +18 -0
  158. package/lib/typegoose/.huskyrc +5 -0
  159. package/lib/typegoose/.lintstagedrc.json +5 -0
  160. package/lib/typegoose/.prettierrc +8 -0
  161. package/lib/typegoose/.swcrc +40 -0
  162. package/lib/typegoose/.vscode/launch.json +35 -0
  163. package/lib/typegoose/.vscode/settings.json +6 -0
  164. package/lib/typegoose/Dockerfile +24 -0
  165. package/lib/typegoose/Makefile +6 -0
  166. package/lib/typegoose/docker-compose.yml +46 -0
  167. package/lib/typegoose/ecosystem.config.js +57 -0
  168. package/lib/typegoose/jest.config.js +12 -0
  169. package/lib/typegoose/nginx.conf +40 -0
  170. package/lib/typegoose/nodemon.json +12 -0
  171. package/lib/typegoose/package.json +80 -0
  172. package/lib/typegoose/src/app.ts +93 -0
  173. package/lib/typegoose/src/config/index.ts +5 -0
  174. package/lib/typegoose/src/controllers/auth.controller.ts +46 -0
  175. package/lib/typegoose/src/controllers/index.controller.ts +13 -0
  176. package/lib/typegoose/src/controllers/users.controller.ts +65 -0
  177. package/lib/typegoose/src/databases/index.ts +3 -0
  178. package/lib/typegoose/src/dtos/users.dto.ts +9 -0
  179. package/lib/typegoose/src/exceptions/HttpException.ts +10 -0
  180. package/lib/typegoose/src/http/auth.http +32 -0
  181. package/lib/typegoose/src/http/users.http +34 -0
  182. package/lib/typegoose/src/interfaces/auth.interface.ts +15 -0
  183. package/lib/typegoose/src/interfaces/routes.interface.ts +6 -0
  184. package/lib/typegoose/src/interfaces/users.interface.ts +5 -0
  185. package/lib/typegoose/src/middlewares/auth.middleware.ts +32 -0
  186. package/lib/typegoose/src/middlewares/error.middleware.ts +17 -0
  187. package/lib/typegoose/src/middlewares/validation.middleware.ts +25 -0
  188. package/lib/typegoose/src/models/users.model.ts +18 -0
  189. package/lib/typegoose/src/routes/auth.route.ts +24 -0
  190. package/lib/typegoose/src/routes/index.route.ts +19 -0
  191. package/lib/typegoose/src/routes/users.route.ts +25 -0
  192. package/lib/typegoose/src/server.ts +11 -0
  193. package/lib/typegoose/src/services/auth.service.ts +61 -0
  194. package/lib/typegoose/src/services/users.service.ts +64 -0
  195. package/lib/typegoose/src/tests/auth.test.ts +83 -0
  196. package/lib/typegoose/src/tests/index.test.ts +18 -0
  197. package/lib/typegoose/src/tests/users.test.ts +133 -0
  198. package/lib/typegoose/src/utils/logger.ts +65 -0
  199. package/lib/typegoose/src/utils/util.ts +19 -0
  200. package/lib/typegoose/src/utils/validateEnv.ts +10 -0
  201. package/lib/typegoose/swagger.yaml +122 -0
  202. package/lib/typegoose/tsconfig.json +40 -0
  203. package/lib/typeorm/.swcrc +39 -0
  204. package/lib/typeorm/docker-compose.yml +1 -1
  205. package/lib/typeorm/nodemon.json +2 -2
  206. package/lib/typeorm/package.json +12 -8
  207. package/lib/typeorm/src/app.ts +1 -2
  208. package/lib/typeorm/src/configs/development.json +1 -1
  209. package/lib/typeorm/src/configs/production.json +1 -1
  210. package/lib/typeorm/src/configs/test.json +1 -1
  211. package/lib/typeorm/src/databases/index.ts +5 -5
  212. package/lib/typeorm/src/entities/users.entity.ts +26 -0
  213. package/lib/typeorm/src/http/auth.http +12 -15
  214. package/lib/typeorm/src/http/users.http +12 -16
  215. package/lib/typeorm/src/index.ts +1 -0
  216. package/lib/typeorm/src/middlewares/auth.middleware.ts +5 -9
  217. package/lib/typeorm/src/server.ts +1 -2
  218. package/lib/typeorm/src/services/auth.service.ts +13 -17
  219. package/lib/typeorm/src/services/users.service.ts +16 -22
  220. package/lib/typeorm/src/utils/logger.ts +5 -5
  221. package/lib/typeorm/tsconfig.json +3 -3
  222. package/package.json +14 -4
@@ -0,0 +1,46 @@
1
+ version: '3.7'
2
+
3
+ services:
4
+ proxy:
5
+ image: nginx:alpine
6
+ container_name: proxy
7
+ ports:
8
+ - '80:80'
9
+ volumes:
10
+ - ./nginx.conf:/etc/nginx/nginx.conf
11
+ restart: 'unless-stopped'
12
+ networks:
13
+ - backend
14
+ server:
15
+ build:
16
+ context: ./
17
+ target: development-build-stage
18
+ dockerfile: Dockerfile
19
+ container_name: server
20
+ ports:
21
+ - '3000:3000'
22
+ volumes:
23
+ - ./:/app
24
+ - /app/node_modules
25
+ restart: 'unless-stopped'
26
+ networks:
27
+ - backend
28
+ links:
29
+ - mongo
30
+ depends_on:
31
+ - mongo
32
+ mongo:
33
+ image: mvertes/alpine-mongo
34
+ container_name: mongo
35
+ ports:
36
+ - '27017:27017'
37
+ networks:
38
+ - backend
39
+
40
+ networks:
41
+ backend:
42
+ driver: bridge
43
+
44
+ volumes:
45
+ data:
46
+ driver: local
@@ -0,0 +1,57 @@
1
+ /**
2
+ * @description pm2 configuration file.
3
+ * @example
4
+ * production mode :: pm2 start ecosystem.config.js --only prod
5
+ * development mode :: pm2 start ecosystem.config.js --only dev
6
+ */
7
+ module.exports = {
8
+ apps: [
9
+ {
10
+ name: 'prod', // pm2 start App name
11
+ script: 'dist/server.js',
12
+ exec_mode: 'cluster', // 'cluster' or 'fork'
13
+ instance_var: 'INSTANCE_ID', // instance variable
14
+ instances: 2, // pm2 instance count
15
+ autorestart: true, // auto restart if process crash
16
+ watch: false, // files change automatic restart
17
+ ignore_watch: ['node_modules', 'logs'], // ignore files change
18
+ max_memory_restart: '1G', // restart if process use more than 1G memory
19
+ merge_logs: true, // if true, stdout and stderr will be merged and sent to pm2 log
20
+ output: './logs/access.log', // pm2 log file
21
+ error: './logs/error.log', // pm2 error log file
22
+ env: { // environment variable
23
+ PORT: 3000,
24
+ NODE_ENV: 'production',
25
+ },
26
+ },
27
+ {
28
+ name: 'dev', // pm2 start App name
29
+ script: 'ts-node', // ts-node
30
+ args: '-r tsconfig-paths/register --transpile-only src/server.ts', // ts-node args
31
+ exec_mode: 'cluster', // 'cluster' or 'fork'
32
+ instance_var: 'INSTANCE_ID', // instance variable
33
+ instances: 2, // pm2 instance count
34
+ autorestart: true, // auto restart if process crash
35
+ watch: false, // files change automatic restart
36
+ ignore_watch: ['node_modules', 'logs'], // ignore files change
37
+ max_memory_restart: '1G', // restart if process use more than 1G memory
38
+ merge_logs: true, // if true, stdout and stderr will be merged and sent to pm2 log
39
+ output: './logs/access.log', // pm2 log file
40
+ error: './logs/error.log', // pm2 error log file
41
+ env: { // environment variable
42
+ PORT: 3000,
43
+ NODE_ENV: 'development',
44
+ },
45
+ },
46
+ ],
47
+ deploy: {
48
+ production: {
49
+ user: 'user',
50
+ host: '0.0.0.0',
51
+ ref: 'origin/master',
52
+ repo: 'git@github.com:repo.git',
53
+ path: 'dist/server.js',
54
+ 'post-deploy': 'npm install && npm run build && pm2 reload ecosystem.config.js --only prod',
55
+ },
56
+ },
57
+ };
@@ -0,0 +1,12 @@
1
+ const { pathsToModuleNameMapper } = require('ts-jest/utils');
2
+ const { compilerOptions } = require('./tsconfig.json');
3
+
4
+ module.exports = {
5
+ preset: 'ts-jest',
6
+ testEnvironment: 'node',
7
+ roots: ['<rootDir>/src'],
8
+ transform: {
9
+ '^.+\\.tsx?$': 'ts-jest',
10
+ },
11
+ moduleNameMapper: pathsToModuleNameMapper(compilerOptions.paths, { prefix: '<rootDir>/src' }),
12
+ };
@@ -0,0 +1,40 @@
1
+ user nginx;
2
+ worker_processes 1;
3
+
4
+ error_log /var/log/nginx/error.log warn;
5
+ pid /var/run/nginx.pid;
6
+
7
+ events {
8
+ worker_connections 1024;
9
+ }
10
+
11
+ http {
12
+ include /etc/nginx/mime.types;
13
+ default_type application/octet-stream;
14
+
15
+ upstream api-server {
16
+ server server:3000;
17
+ keepalive 100;
18
+ }
19
+
20
+ server {
21
+ listen 80;
22
+ server_name localhost;
23
+
24
+ location / {
25
+ proxy_http_version 1.1;
26
+ proxy_pass http://api-server;
27
+ }
28
+
29
+ }
30
+
31
+ log_format main '$remote_addr - $remote_user [$time_local] "$request" '
32
+ '$status $body_bytes_sent "$http_referer" '
33
+ '"$http_user_agent" "$http_x_forwarded_for"';
34
+
35
+ access_log /var/log/nginx/access.log main;
36
+
37
+ sendfile on;
38
+ keepalive_timeout 65;
39
+ include /etc/nginx/conf.d/*.conf;
40
+ }
@@ -0,0 +1,12 @@
1
+ {
2
+ "watch": [
3
+ "src",
4
+ ".env"
5
+ ],
6
+ "ext": "js,ts,json",
7
+ "ignore": [
8
+ "src/logs/*",
9
+ "src/**/*.{spec,test}.ts"
10
+ ],
11
+ "exec": "ts-node -r tsconfig-paths/register --transpile-only src/server.ts"
12
+ }
@@ -0,0 +1,80 @@
1
+ {
2
+ "name": "typescript-express-starter",
3
+ "version": "0.0.0",
4
+ "description": "TypeScript + Typegoose + MongoDB + Express API Server",
5
+ "author": "",
6
+ "license": "ISC",
7
+ "scripts": {
8
+ "start": "npm run build && cross-env NODE_ENV=production node dist/server.js",
9
+ "dev": "cross-env NODE_ENV=development nodemon",
10
+ "build": "tsc && tsc-alias",
11
+ "build:swc": "swc src -d dist --source-maps --copy-files",
12
+ "test": "jest --forceExit --detectOpenHandles",
13
+ "lint": "eslint --ignore-path .gitignore --ext .ts src/",
14
+ "lint:fix": "npm run lint -- --fix",
15
+ "deploy:prod": "npm run build && pm2 start ecosystem.config.js --only prod",
16
+ "deploy:dev": "pm2 start ecosystem.config.js --only dev"
17
+ },
18
+ "dependencies": {
19
+ "@typegoose/typegoose": "^9.4.0",
20
+ "bcrypt": "^5.0.1",
21
+ "class-transformer": "^0.5.1",
22
+ "class-validator": "^0.13.1",
23
+ "compression": "^1.7.4",
24
+ "cookie-parser": "^1.4.5",
25
+ "cors": "^2.8.5",
26
+ "dotenv": "^10.0.0",
27
+ "envalid": "^7.1.0",
28
+ "express": "^4.17.1",
29
+ "helmet": "^4.6.0",
30
+ "hpp": "^0.2.3",
31
+ "jsonwebtoken": "^8.5.1",
32
+ "mongoose": "~6.1.3",
33
+ "morgan": "^1.10.0",
34
+ "swagger-jsdoc": "^6.0.0",
35
+ "swagger-ui-express": "^4.1.6",
36
+ "winston": "^3.3.3",
37
+ "winston-daily-rotate-file": "^4.5.5"
38
+ },
39
+ "devDependencies": {
40
+ "@swc/cli": "^0.1.51",
41
+ "@swc/core": "^1.2.108",
42
+ "@types/bcrypt": "^5.0.0",
43
+ "@types/compression": "^1.7.1",
44
+ "@types/cookie-parser": "^1.4.2",
45
+ "@types/cors": "^2.8.11",
46
+ "@types/dotenv": "^8.2.0",
47
+ "@types/express": "^4.17.13",
48
+ "@types/helmet": "^4.0.0",
49
+ "@types/hpp": "^0.2.1",
50
+ "@types/jest": "^26.0.24",
51
+ "@types/jsonwebtoken": "^8.5.4",
52
+ "@types/mongoose": "^5.10.1",
53
+ "@types/morgan": "^1.9.3",
54
+ "@types/node": "^16.0.1",
55
+ "@types/supertest": "^2.0.11",
56
+ "@types/swagger-jsdoc": "^6.0.1",
57
+ "@types/swagger-ui-express": "^4.1.3",
58
+ "@types/winston": "^2.4.4",
59
+ "@typescript-eslint/eslint-plugin": "^4.28.2",
60
+ "@typescript-eslint/parser": "^4.28.2",
61
+ "cross-env": "^7.0.3",
62
+ "eslint": "^7.30.0",
63
+ "eslint-config-prettier": "^8.3.0",
64
+ "eslint-plugin-prettier": "^3.4.0",
65
+ "husky": "^7.0.1",
66
+ "jest": "^27.0.6",
67
+ "lint-staged": "^11.0.0",
68
+ "node-config": "^0.0.2",
69
+ "node-gyp": "^8.1.0",
70
+ "nodemon": "^2.0.9",
71
+ "pm2": "^5.1.0",
72
+ "prettier": "^2.3.2",
73
+ "supertest": "^6.1.3",
74
+ "ts-jest": "^27.0.7",
75
+ "ts-node": "^10.4.0",
76
+ "tsc-alias": "^1.4.1",
77
+ "tsconfig-paths": "^3.10.1",
78
+ "typescript": "^4.5.2"
79
+ }
80
+ }
@@ -0,0 +1,93 @@
1
+ import compression from 'compression';
2
+ import cookieParser from 'cookie-parser';
3
+ import cors from 'cors';
4
+ import express from 'express';
5
+ import helmet from 'helmet';
6
+ import hpp from 'hpp';
7
+ import morgan from 'morgan';
8
+ import { connect, set } from 'mongoose';
9
+ import swaggerJSDoc from 'swagger-jsdoc';
10
+ import swaggerUi from 'swagger-ui-express';
11
+ import { NODE_ENV, PORT, LOG_FORMAT, ORIGIN, CREDENTIALS } from '@config';
12
+ import { dbConnection } from '@databases';
13
+ import { Routes } from '@interfaces/routes.interface';
14
+ import errorMiddleware from '@middlewares/error.middleware';
15
+ import { logger, stream } from '@utils/logger';
16
+
17
+ class App {
18
+ public app: express.Application;
19
+ public env: string;
20
+ public port: string | number;
21
+
22
+ constructor(routes: Routes[]) {
23
+ this.app = express();
24
+ this.env = NODE_ENV || 'development';
25
+ this.port = PORT || 3000;
26
+
27
+ this.connectToDatabase();
28
+ this.initializeMiddlewares();
29
+ this.initializeRoutes(routes);
30
+ this.initializeSwagger();
31
+ this.initializeErrorHandling();
32
+ }
33
+
34
+ public listen() {
35
+ this.app.listen(this.port, () => {
36
+ logger.info(`=================================`);
37
+ logger.info(`======= ENV: ${this.env} =======`);
38
+ logger.info(`🚀 App listening on the port ${this.port}`);
39
+ logger.info(`=================================`);
40
+ });
41
+ }
42
+
43
+ public getServer() {
44
+ return this.app;
45
+ }
46
+
47
+ private connectToDatabase() {
48
+ if (this.env !== 'production') {
49
+ set('debug', true);
50
+ }
51
+
52
+ connect(dbConnection);
53
+ }
54
+
55
+ private initializeMiddlewares() {
56
+ this.app.use(morgan(LOG_FORMAT, { stream }));
57
+ this.app.use(cors({ origin: ORIGIN, credentials: CREDENTIALS }));
58
+ this.app.use(hpp());
59
+ this.app.use(helmet());
60
+ this.app.use(compression());
61
+ this.app.use(express.json());
62
+ this.app.use(express.urlencoded({ extended: true }));
63
+ this.app.use(cookieParser());
64
+ }
65
+
66
+ private initializeRoutes(routes: Routes[]) {
67
+ routes.forEach(route => {
68
+ this.app.use('/', route.router);
69
+ });
70
+ }
71
+
72
+ private initializeSwagger() {
73
+ const options = {
74
+ swaggerDefinition: {
75
+ info: {
76
+ title: 'REST API',
77
+ version: '1.0.0',
78
+ description: 'Example docs',
79
+ },
80
+ },
81
+ apis: ['swagger.yaml'],
82
+ };
83
+
84
+ const specs = swaggerJSDoc(options);
85
+ this.app.use('/api-docs', swaggerUi.serve, swaggerUi.setup(specs));
86
+ }
87
+
88
+ private initializeErrorHandling() {
89
+ this.app.use(errorMiddleware);
90
+ }
91
+ }
92
+
93
+ export default App;
@@ -0,0 +1,5 @@
1
+ import { config } from 'dotenv';
2
+ config({ path: `.env.${process.env.NODE_ENV || 'development'}.local` });
3
+
4
+ export const CREDENTIALS = process.env.CREDENTIALS === 'true';
5
+ export const { NODE_ENV, PORT, DB_HOST, DB_PORT, DB_DATABASE, SECRET_KEY, LOG_FORMAT, LOG_DIR, ORIGIN } = process.env;
@@ -0,0 +1,46 @@
1
+ import { NextFunction, Request, Response } from 'express';
2
+ import { CreateUserDto } from '@dtos/users.dto';
3
+ import { RequestWithUser } from '@interfaces/auth.interface';
4
+ import { User } from '@interfaces/users.interface';
5
+ import AuthService from '@services/auth.service';
6
+
7
+ class AuthController {
8
+ public authService = new AuthService();
9
+
10
+ public signUp = async (req: Request, res: Response, next: NextFunction) => {
11
+ try {
12
+ const userData: CreateUserDto = req.body;
13
+ const signUpUserData: User = await this.authService.signup(userData);
14
+
15
+ res.status(201).json({ data: signUpUserData, message: 'signup' });
16
+ } catch (error) {
17
+ next(error);
18
+ }
19
+ };
20
+
21
+ public logIn = async (req: Request, res: Response, next: NextFunction) => {
22
+ try {
23
+ const userData: CreateUserDto = req.body;
24
+ const { cookie, findUser } = await this.authService.login(userData);
25
+
26
+ res.setHeader('Set-Cookie', [cookie]);
27
+ res.status(200).json({ data: findUser, message: 'login' });
28
+ } catch (error) {
29
+ next(error);
30
+ }
31
+ };
32
+
33
+ public logOut = async (req: RequestWithUser, res: Response, next: NextFunction) => {
34
+ try {
35
+ const userData: User = req.user;
36
+ const logOutUserData: User = await this.authService.logout(userData);
37
+
38
+ res.setHeader('Set-Cookie', ['Authorization=; Max-age=0']);
39
+ res.status(200).json({ data: logOutUserData, message: 'logout' });
40
+ } catch (error) {
41
+ next(error);
42
+ }
43
+ };
44
+ }
45
+
46
+ export default AuthController;
@@ -0,0 +1,13 @@
1
+ import { NextFunction, Request, Response } from 'express';
2
+
3
+ class IndexController {
4
+ public index = (req: Request, res: Response, next: NextFunction) => {
5
+ try {
6
+ res.sendStatus(200);
7
+ } catch (error) {
8
+ next(error);
9
+ }
10
+ };
11
+ }
12
+
13
+ export default IndexController;
@@ -0,0 +1,65 @@
1
+ import { NextFunction, Request, Response } from 'express';
2
+ import { CreateUserDto } from '@dtos/users.dto';
3
+ import { User } from '@interfaces/users.interface';
4
+ import userService from '@services/users.service';
5
+
6
+ class UsersController {
7
+ public userService = new userService();
8
+
9
+ public getUsers = async (req: Request, res: Response, next: NextFunction) => {
10
+ try {
11
+ const findAllUsersData: User[] = await this.userService.findAllUser();
12
+
13
+ res.status(200).json({ data: findAllUsersData, message: 'findAll' });
14
+ } catch (error) {
15
+ next(error);
16
+ }
17
+ };
18
+
19
+ public getUserById = async (req: Request, res: Response, next: NextFunction) => {
20
+ try {
21
+ const userId: string = req.params.id;
22
+ const findOneUserData: User = await this.userService.findUserById(userId);
23
+
24
+ res.status(200).json({ data: findOneUserData, message: 'findOne' });
25
+ } catch (error) {
26
+ next(error);
27
+ }
28
+ };
29
+
30
+ public createUser = async (req: Request, res: Response, next: NextFunction) => {
31
+ try {
32
+ const userData: CreateUserDto = req.body;
33
+ const createUserData: User = await this.userService.createUser(userData);
34
+
35
+ res.status(201).json({ data: createUserData, message: 'created' });
36
+ } catch (error) {
37
+ next(error);
38
+ }
39
+ };
40
+
41
+ public updateUser = async (req: Request, res: Response, next: NextFunction) => {
42
+ try {
43
+ const userId: string = req.params.id;
44
+ const userData: CreateUserDto = req.body;
45
+ const updateUserData: User = await this.userService.updateUser(userId, userData);
46
+
47
+ res.status(200).json({ data: updateUserData, message: 'updated' });
48
+ } catch (error) {
49
+ next(error);
50
+ }
51
+ };
52
+
53
+ public deleteUser = async (req: Request, res: Response, next: NextFunction) => {
54
+ try {
55
+ const userId: string = req.params.id;
56
+ const deleteUserData: User = await this.userService.deleteUser(userId);
57
+
58
+ res.status(200).json({ data: deleteUserData, message: 'deleted' });
59
+ } catch (error) {
60
+ next(error);
61
+ }
62
+ };
63
+ }
64
+
65
+ export default UsersController;
@@ -0,0 +1,3 @@
1
+ import { DB_HOST, DB_PORT, DB_DATABASE } from '@config';
2
+
3
+ export const dbConnection = `mongodb://${DB_HOST}:${DB_PORT}/${DB_DATABASE}`;
@@ -0,0 +1,9 @@
1
+ import { IsEmail, IsString } from 'class-validator';
2
+
3
+ export class CreateUserDto {
4
+ @IsEmail()
5
+ public email: string;
6
+
7
+ @IsString()
8
+ public password: string;
9
+ }
@@ -0,0 +1,10 @@
1
+ export class HttpException extends Error {
2
+ public status: number;
3
+ public message: string;
4
+
5
+ constructor(status: number, message: string) {
6
+ super(message);
7
+ this.status = status;
8
+ this.message = message;
9
+ }
10
+ }
@@ -0,0 +1,32 @@
1
+ # baseURL
2
+ @baseURL = http://localhost:3000
3
+
4
+ ###
5
+ # User Signup
6
+ POST {{ baseURL }}/signup
7
+ Content-Type: application/json
8
+
9
+ {
10
+ "email": "example@email.com",
11
+ "password": "password"
12
+ }
13
+
14
+ ###
15
+ # User Login
16
+ POST {{ baseURL }}/login
17
+ Content-Type: application/json
18
+
19
+ {
20
+ "email": "example@email.com",
21
+ "password": "password"
22
+ }
23
+
24
+ ###
25
+ # User Logout
26
+ POST {{ baseURL }}/logout
27
+ Content-Type: application/json
28
+
29
+ {
30
+ "email": "example@email.com",
31
+ "password": "password"
32
+ }
@@ -0,0 +1,34 @@
1
+ # baseURL
2
+ @baseURL = http://localhost:3000
3
+
4
+ ###
5
+ # Find All Users
6
+ GET {{ baseURL }}/users
7
+
8
+ ###
9
+ # Find User By Id
10
+ GET {{ baseURL }}/users/1
11
+
12
+ ###
13
+ # Create User
14
+ POST {{ baseURL }}/users
15
+ Content-Type: application/json
16
+
17
+ {
18
+ "email": "example@email.com",
19
+ "password": "password"
20
+ }
21
+
22
+ ###
23
+ # Modify User By Id
24
+ PUT {{ baseURL }}/users/1
25
+ Content-Type: application/json
26
+
27
+ {
28
+ "email": "example@email.com",
29
+ "password": "password"
30
+ }
31
+
32
+ ###
33
+ # Delete User By Id
34
+ DELETE {{ baseURL }}/users/1
@@ -0,0 +1,15 @@
1
+ import { Request } from 'express';
2
+ import { User } from '@interfaces/users.interface';
3
+
4
+ export interface DataStoredInToken {
5
+ _id: string;
6
+ }
7
+
8
+ export interface TokenData {
9
+ token: string;
10
+ expiresIn: number;
11
+ }
12
+
13
+ export interface RequestWithUser extends Request {
14
+ user: User;
15
+ }
@@ -0,0 +1,6 @@
1
+ import { Router } from 'express';
2
+
3
+ export interface Routes {
4
+ path?: string;
5
+ router: Router;
6
+ }
@@ -0,0 +1,5 @@
1
+ export interface User {
2
+ _id: string;
3
+ email: string;
4
+ password: string;
5
+ }
@@ -0,0 +1,32 @@
1
+ import { NextFunction, Response } from 'express';
2
+ import { verify } from 'jsonwebtoken';
3
+ import { SECRET_KEY } from '@config';
4
+ import { HttpException } from '@exceptions/HttpException';
5
+ import { DataStoredInToken, RequestWithUser } from '@interfaces/auth.interface';
6
+ import userModel from '@models/users.model';
7
+
8
+ const authMiddleware = async (req: RequestWithUser, res: Response, next: NextFunction) => {
9
+ try {
10
+ const Authorization = req.cookies['Authorization'] || (req.header('Authorization') ? req.header('Authorization').split('Bearer ')[1] : null);
11
+
12
+ if (Authorization) {
13
+ const secretKey: string = SECRET_KEY;
14
+ const verificationResponse = (await verify(Authorization, secretKey)) as DataStoredInToken;
15
+ const userId = verificationResponse._id;
16
+ const findUser = await userModel.findById(userId);
17
+
18
+ if (findUser) {
19
+ req.user = findUser;
20
+ next();
21
+ } else {
22
+ next(new HttpException(401, 'Wrong authentication token'));
23
+ }
24
+ } else {
25
+ next(new HttpException(404, 'Authentication token missing'));
26
+ }
27
+ } catch (error) {
28
+ next(new HttpException(401, 'Wrong authentication token'));
29
+ }
30
+ };
31
+
32
+ export default authMiddleware;
@@ -0,0 +1,17 @@
1
+ import { NextFunction, Request, Response } from 'express';
2
+ import { HttpException } from '@exceptions/HttpException';
3
+ import { logger } from '@utils/logger';
4
+
5
+ const errorMiddleware = (error: HttpException, req: Request, res: Response, next: NextFunction) => {
6
+ try {
7
+ const status: number = error.status || 500;
8
+ const message: string = error.message || 'Something went wrong';
9
+
10
+ logger.error(`[${req.method}] ${req.path} >> StatusCode:: ${status}, Message:: ${message}`);
11
+ res.status(status).json({ message });
12
+ } catch (error) {
13
+ next(error);
14
+ }
15
+ };
16
+
17
+ export default errorMiddleware;