├── .env ├── README.md ├── package-lock.json ├── package.json └── src ├── config └── db.js ├── controllers └── userController.js ├── data ├── createUserTable.js └── data.sql ├── docker-commands.txt ├── index.js ├── middlewares ├── errorHandler.js └── inputValidator.js ├── models └── userModel.js └── routes └── userRoutes.js /.env: -------------------------------------------------------------------------------- 1 | PORT=5001 2 | DB_USER=postgres 3 | DB_HOST=localhost 4 | DB_NAME=express-crud 5 | DB_PASSWORD=admin 6 | DB_PORT=5432 7 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # Nodejs_Express_Postgresql_Docker_REST_API_Project 2 | This is CRUD Rest API project build using Node.js Express, Postgresql running on docker container for user management. 3 | -------------------------------------------------------------------------------- /package-lock.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "express-postgres-crud-api", 3 | "version": "1.0.0", 4 | "lockfileVersion": 3, 5 | "requires": true, 6 | "packages": { 7 | "": { 8 | "name": "express-postgres-crud-api", 9 | "version": "1.0.0", 10 | "license": "ISC", 11 | "dependencies": { 12 | "cors": "^2.8.5", 13 | "dotenv": "^16.4.5", 14 | "express": "^4.21.1", 15 | "joi": "^17.13.3", 16 | "pg": "^8.13.1" 17 | }, 18 | "devDependencies": { 19 | "nodemon": "^3.1.7" 20 | } 21 | }, 22 | "node_modules/@hapi/hoek": { 23 | "version": "9.3.0", 24 | "resolved": "https://registry.npmjs.org/@hapi/hoek/-/hoek-9.3.0.tgz", 25 | "integrity": "sha512-/c6rf4UJlmHlC9b5BaNvzAcFv7HZ2QHaV0D4/HNlBdvFnvQq8RI4kYdhyPCl7Xj+oWvTWQ8ujhqS53LIgAe6KQ==", 26 | "license": "BSD-3-Clause" 27 | }, 28 | "node_modules/@hapi/topo": { 29 | "version": "5.1.0", 30 | "resolved": "https://registry.npmjs.org/@hapi/topo/-/topo-5.1.0.tgz", 31 | "integrity": "sha512-foQZKJig7Ob0BMAYBfcJk8d77QtOe7Wo4ox7ff1lQYoNNAb6jwcY1ncdoy2e9wQZzvNy7ODZCYJkK8kzmcAnAg==", 32 | "license": "BSD-3-Clause", 33 | "dependencies": { 34 | "@hapi/hoek": "^9.0.0" 35 | } 36 | }, 37 | "node_modules/@sideway/address": { 38 | "version": "4.1.5", 39 | "resolved": "https://registry.npmjs.org/@sideway/address/-/address-4.1.5.tgz", 40 | "integrity": "sha512-IqO/DUQHUkPeixNQ8n0JA6102hT9CmaljNTPmQ1u8MEhBo/R4Q8eKLN/vGZxuebwOroDB4cbpjheD4+/sKFK4Q==", 41 | "license": "BSD-3-Clause", 42 | "dependencies": { 43 | "@hapi/hoek": "^9.0.0" 44 | } 45 | }, 46 | "node_modules/@sideway/formula": { 47 | "version": "3.0.1", 48 | "resolved": "https://registry.npmjs.org/@sideway/formula/-/formula-3.0.1.tgz", 49 | "integrity": "sha512-/poHZJJVjx3L+zVD6g9KgHfYnb443oi7wLu/XKojDviHy6HOEOA6z1Trk5aR1dGcmPenJEgb2sK2I80LeS3MIg==", 50 | "license": "BSD-3-Clause" 51 | }, 52 | "node_modules/@sideway/pinpoint": { 53 | "version": "2.0.0", 54 | "resolved": "https://registry.npmjs.org/@sideway/pinpoint/-/pinpoint-2.0.0.tgz", 55 | "integrity": "sha512-RNiOoTPkptFtSVzQevY/yWtZwf/RxyVnPy/OcA9HBM3MlGDnBEYL5B41H0MTn0Uec8Hi+2qUtTfG2WWZBmMejQ==", 56 | "license": "BSD-3-Clause" 57 | }, 58 | "node_modules/accepts": { 59 | "version": "1.3.8", 60 | "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.8.tgz", 61 | "integrity": "sha512-PYAthTa2m2VKxuvSD3DPC/Gy+U+sOA1LAuT8mkmRuvw+NACSaeXEQ+NHcVF7rONl6qcaxV3Uuemwawk+7+SJLw==", 62 | "license": "MIT", 63 | "dependencies": { 64 | "mime-types": "~2.1.34", 65 | "negotiator": "0.6.3" 66 | }, 67 | "engines": { 68 | "node": ">= 0.6" 69 | } 70 | }, 71 | "node_modules/anymatch": { 72 | "version": "3.1.3", 73 | "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.3.tgz", 74 | "integrity": "sha512-KMReFUr0B4t+D+OBkjR3KYqvocp2XaSzO55UcB6mgQMd3KbcE+mWTyvVV7D/zsdEbNnV6acZUutkiHQXvTr1Rw==", 75 | "dev": true, 76 | "license": "ISC", 77 | "dependencies": { 78 | "normalize-path": "^3.0.0", 79 | "picomatch": "^2.0.4" 80 | }, 81 | "engines": { 82 | "node": ">= 8" 83 | } 84 | }, 85 | "node_modules/array-flatten": { 86 | "version": "1.1.1", 87 | "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz", 88 | "integrity": "sha512-PCVAQswWemu6UdxsDFFX/+gVeYqKAod3D3UVm91jHwynguOwAvYPhx8nNlM++NqRcK6CxxpUafjmhIdKiHibqg==", 89 | "license": "MIT" 90 | }, 91 | "node_modules/balanced-match": { 92 | "version": "1.0.2", 93 | "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", 94 | "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", 95 | "dev": true, 96 | "license": "MIT" 97 | }, 98 | "node_modules/binary-extensions": { 99 | "version": "2.3.0", 100 | "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.3.0.tgz", 101 | "integrity": "sha512-Ceh+7ox5qe7LJuLHoY0feh3pHuUDHAcRUeyL2VYghZwfpkNIy/+8Ocg0a3UuSoYzavmylwuLWQOf3hl0jjMMIw==", 102 | "dev": true, 103 | "license": "MIT", 104 | "engines": { 105 | "node": ">=8" 106 | }, 107 | "funding": { 108 | "url": "https://github.com/sponsors/sindresorhus" 109 | } 110 | }, 111 | "node_modules/body-parser": { 112 | "version": "1.20.3", 113 | "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.20.3.tgz", 114 | "integrity": "sha512-7rAxByjUMqQ3/bHJy7D6OGXvx/MMc4IqBn/X0fcM1QUcAItpZrBEYhWGem+tzXH90c+G01ypMcYJBO9Y30203g==", 115 | "license": "MIT", 116 | "dependencies": { 117 | "bytes": "3.1.2", 118 | "content-type": "~1.0.5", 119 | "debug": "2.6.9", 120 | "depd": "2.0.0", 121 | "destroy": "1.2.0", 122 | "http-errors": "2.0.0", 123 | "iconv-lite": "0.4.24", 124 | "on-finished": "2.4.1", 125 | "qs": "6.13.0", 126 | "raw-body": "2.5.2", 127 | "type-is": "~1.6.18", 128 | "unpipe": "1.0.0" 129 | }, 130 | "engines": { 131 | "node": ">= 0.8", 132 | "npm": "1.2.8000 || >= 1.4.16" 133 | } 134 | }, 135 | "node_modules/brace-expansion": { 136 | "version": "1.1.11", 137 | "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", 138 | "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", 139 | "dev": true, 140 | "license": "MIT", 141 | "dependencies": { 142 | "balanced-match": "^1.0.0", 143 | "concat-map": "0.0.1" 144 | } 145 | }, 146 | "node_modules/braces": { 147 | "version": "3.0.3", 148 | "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.3.tgz", 149 | "integrity": "sha512-yQbXgO/OSZVD2IsiLlro+7Hf6Q18EJrKSEsdoMzKePKXct3gvD8oLcOQdIzGupr5Fj+EDe8gO/lxc1BzfMpxvA==", 150 | "dev": true, 151 | "license": "MIT", 152 | "dependencies": { 153 | "fill-range": "^7.1.1" 154 | }, 155 | "engines": { 156 | "node": ">=8" 157 | } 158 | }, 159 | "node_modules/bytes": { 160 | "version": "3.1.2", 161 | "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.2.tgz", 162 | "integrity": "sha512-/Nf7TyzTx6S3yRJObOAV7956r8cr2+Oj8AC5dt8wSP3BQAoeX58NoHyCU8P8zGkNXStjTSi6fzO6F0pBdcYbEg==", 163 | "license": "MIT", 164 | "engines": { 165 | "node": ">= 0.8" 166 | } 167 | }, 168 | "node_modules/call-bind": { 169 | "version": "1.0.7", 170 | "resolved": "https://registry.npmjs.org/call-bind/-/call-bind-1.0.7.tgz", 171 | "integrity": "sha512-GHTSNSYICQ7scH7sZ+M2rFopRoLh8t2bLSW6BbgrtLsahOIB5iyAVJf9GjWK3cYTDaMj4XdBpM1cA6pIS0Kv2w==", 172 | "license": "MIT", 173 | "dependencies": { 174 | "es-define-property": "^1.0.0", 175 | "es-errors": "^1.3.0", 176 | "function-bind": "^1.1.2", 177 | "get-intrinsic": "^1.2.4", 178 | "set-function-length": "^1.2.1" 179 | }, 180 | "engines": { 181 | "node": ">= 0.4" 182 | }, 183 | "funding": { 184 | "url": "https://github.com/sponsors/ljharb" 185 | } 186 | }, 187 | "node_modules/chokidar": { 188 | "version": "3.6.0", 189 | "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.6.0.tgz", 190 | "integrity": "sha512-7VT13fmjotKpGipCW9JEQAusEPE+Ei8nl6/g4FBAmIm0GOOLMua9NDDo/DWp0ZAxCr3cPq5ZpBqmPAQgDda2Pw==", 191 | "dev": true, 192 | "license": "MIT", 193 | "dependencies": { 194 | "anymatch": "~3.1.2", 195 | "braces": "~3.0.2", 196 | "glob-parent": "~5.1.2", 197 | "is-binary-path": "~2.1.0", 198 | "is-glob": "~4.0.1", 199 | "normalize-path": "~3.0.0", 200 | "readdirp": "~3.6.0" 201 | }, 202 | "engines": { 203 | "node": ">= 8.10.0" 204 | }, 205 | "funding": { 206 | "url": "https://paulmillr.com/funding/" 207 | }, 208 | "optionalDependencies": { 209 | "fsevents": "~2.3.2" 210 | } 211 | }, 212 | "node_modules/concat-map": { 213 | "version": "0.0.1", 214 | "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", 215 | "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==", 216 | "dev": true, 217 | "license": "MIT" 218 | }, 219 | "node_modules/content-disposition": { 220 | "version": "0.5.4", 221 | "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.4.tgz", 222 | "integrity": "sha512-FveZTNuGw04cxlAiWbzi6zTAL/lhehaWbTtgluJh4/E95DqMwTmha3KZN1aAWA8cFIhHzMZUvLevkw5Rqk+tSQ==", 223 | "license": "MIT", 224 | "dependencies": { 225 | "safe-buffer": "5.2.1" 226 | }, 227 | "engines": { 228 | "node": ">= 0.6" 229 | } 230 | }, 231 | "node_modules/content-type": { 232 | "version": "1.0.5", 233 | "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.5.tgz", 234 | "integrity": "sha512-nTjqfcBFEipKdXCv4YDQWCfmcLZKm81ldF0pAopTvyrFGVbcR6P/VAAd5G7N+0tTr8QqiU0tFadD6FK4NtJwOA==", 235 | "license": "MIT", 236 | "engines": { 237 | "node": ">= 0.6" 238 | } 239 | }, 240 | "node_modules/cookie": { 241 | "version": "0.7.1", 242 | "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.7.1.tgz", 243 | "integrity": "sha512-6DnInpx7SJ2AK3+CTUE/ZM0vWTUboZCegxhC2xiIydHR9jNuTAASBrfEpHhiGOZw/nX51bHt6YQl8jsGo4y/0w==", 244 | "license": "MIT", 245 | "engines": { 246 | "node": ">= 0.6" 247 | } 248 | }, 249 | "node_modules/cookie-signature": { 250 | "version": "1.0.6", 251 | "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz", 252 | "integrity": "sha512-QADzlaHc8icV8I7vbaJXJwod9HWYp8uCqf1xa4OfNu1T7JVxQIrUgOWtHdNDtPiywmFbiS12VjotIXLrKM3orQ==", 253 | "license": "MIT" 254 | }, 255 | "node_modules/cors": { 256 | "version": "2.8.5", 257 | "resolved": "https://registry.npmjs.org/cors/-/cors-2.8.5.tgz", 258 | "integrity": "sha512-KIHbLJqu73RGr/hnbrO9uBeixNGuvSQjul/jdFvS/KFSIH1hWVd1ng7zOHx+YrEfInLG7q4n6GHQ9cDtxv/P6g==", 259 | "license": "MIT", 260 | "dependencies": { 261 | "object-assign": "^4", 262 | "vary": "^1" 263 | }, 264 | "engines": { 265 | "node": ">= 0.10" 266 | } 267 | }, 268 | "node_modules/debug": { 269 | "version": "2.6.9", 270 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", 271 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", 272 | "license": "MIT", 273 | "dependencies": { 274 | "ms": "2.0.0" 275 | } 276 | }, 277 | "node_modules/define-data-property": { 278 | "version": "1.1.4", 279 | "resolved": "https://registry.npmjs.org/define-data-property/-/define-data-property-1.1.4.tgz", 280 | "integrity": "sha512-rBMvIzlpA8v6E+SJZoo++HAYqsLrkg7MSfIinMPFhmkorw7X+dOXVJQs+QT69zGkzMyfDnIMN2Wid1+NbL3T+A==", 281 | "license": "MIT", 282 | "dependencies": { 283 | "es-define-property": "^1.0.0", 284 | "es-errors": "^1.3.0", 285 | "gopd": "^1.0.1" 286 | }, 287 | "engines": { 288 | "node": ">= 0.4" 289 | }, 290 | "funding": { 291 | "url": "https://github.com/sponsors/ljharb" 292 | } 293 | }, 294 | "node_modules/depd": { 295 | "version": "2.0.0", 296 | "resolved": "https://registry.npmjs.org/depd/-/depd-2.0.0.tgz", 297 | "integrity": "sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw==", 298 | "license": "MIT", 299 | "engines": { 300 | "node": ">= 0.8" 301 | } 302 | }, 303 | "node_modules/destroy": { 304 | "version": "1.2.0", 305 | "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.2.0.tgz", 306 | "integrity": "sha512-2sJGJTaXIIaR1w4iJSNoN0hnMY7Gpc/n8D4qSCJw8QqFWXf7cuAgnEHxBpweaVcPevC2l3KpjYCx3NypQQgaJg==", 307 | "license": "MIT", 308 | "engines": { 309 | "node": ">= 0.8", 310 | "npm": "1.2.8000 || >= 1.4.16" 311 | } 312 | }, 313 | "node_modules/dotenv": { 314 | "version": "16.4.5", 315 | "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-16.4.5.tgz", 316 | "integrity": "sha512-ZmdL2rui+eB2YwhsWzjInR8LldtZHGDoQ1ugH85ppHKwpUHL7j7rN0Ti9NCnGiQbhaZ11FpR+7ao1dNsmduNUg==", 317 | "license": "BSD-2-Clause", 318 | "engines": { 319 | "node": ">=12" 320 | }, 321 | "funding": { 322 | "url": "https://dotenvx.com" 323 | } 324 | }, 325 | "node_modules/ee-first": { 326 | "version": "1.1.1", 327 | "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", 328 | "integrity": "sha512-WMwm9LhRUo+WUaRN+vRuETqG89IgZphVSNkdFgeb6sS/E4OrDIN7t48CAewSHXc6C8lefD8KKfr5vY61brQlow==", 329 | "license": "MIT" 330 | }, 331 | "node_modules/encodeurl": { 332 | "version": "2.0.0", 333 | "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-2.0.0.tgz", 334 | "integrity": "sha512-Q0n9HRi4m6JuGIV1eFlmvJB7ZEVxu93IrMyiMsGC0lrMJMWzRgx6WGquyfQgZVb31vhGgXnfmPNNXmxnOkRBrg==", 335 | "license": "MIT", 336 | "engines": { 337 | "node": ">= 0.8" 338 | } 339 | }, 340 | "node_modules/es-define-property": { 341 | "version": "1.0.0", 342 | "resolved": "https://registry.npmjs.org/es-define-property/-/es-define-property-1.0.0.tgz", 343 | "integrity": "sha512-jxayLKShrEqqzJ0eumQbVhTYQM27CfT1T35+gCgDFoL82JLsXqTJ76zv6A0YLOgEnLUMvLzsDsGIrl8NFpT2gQ==", 344 | "license": "MIT", 345 | "dependencies": { 346 | "get-intrinsic": "^1.2.4" 347 | }, 348 | "engines": { 349 | "node": ">= 0.4" 350 | } 351 | }, 352 | "node_modules/es-errors": { 353 | "version": "1.3.0", 354 | "resolved": "https://registry.npmjs.org/es-errors/-/es-errors-1.3.0.tgz", 355 | "integrity": "sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==", 356 | "license": "MIT", 357 | "engines": { 358 | "node": ">= 0.4" 359 | } 360 | }, 361 | "node_modules/escape-html": { 362 | "version": "1.0.3", 363 | "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz", 364 | "integrity": "sha512-NiSupZ4OeuGwr68lGIeym/ksIZMJodUGOSCZ/FSnTxcrekbvqrgdUxlJOMpijaKZVjAJrWrGs/6Jy8OMuyj9ow==", 365 | "license": "MIT" 366 | }, 367 | "node_modules/etag": { 368 | "version": "1.8.1", 369 | "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz", 370 | "integrity": "sha512-aIL5Fx7mawVa300al2BnEE4iNvo1qETxLrPI/o05L7z6go7fCw1J6EQmbK4FmJ2AS7kgVF/KEZWufBfdClMcPg==", 371 | "license": "MIT", 372 | "engines": { 373 | "node": ">= 0.6" 374 | } 375 | }, 376 | "node_modules/express": { 377 | "version": "4.21.1", 378 | "resolved": "https://registry.npmjs.org/express/-/express-4.21.1.tgz", 379 | "integrity": "sha512-YSFlK1Ee0/GC8QaO91tHcDxJiE/X4FbpAyQWkxAvG6AXCuR65YzK8ua6D9hvi/TzUfZMpc+BwuM1IPw8fmQBiQ==", 380 | "license": "MIT", 381 | "dependencies": { 382 | "accepts": "~1.3.8", 383 | "array-flatten": "1.1.1", 384 | "body-parser": "1.20.3", 385 | "content-disposition": "0.5.4", 386 | "content-type": "~1.0.4", 387 | "cookie": "0.7.1", 388 | "cookie-signature": "1.0.6", 389 | "debug": "2.6.9", 390 | "depd": "2.0.0", 391 | "encodeurl": "~2.0.0", 392 | "escape-html": "~1.0.3", 393 | "etag": "~1.8.1", 394 | "finalhandler": "1.3.1", 395 | "fresh": "0.5.2", 396 | "http-errors": "2.0.0", 397 | "merge-descriptors": "1.0.3", 398 | "methods": "~1.1.2", 399 | "on-finished": "2.4.1", 400 | "parseurl": "~1.3.3", 401 | "path-to-regexp": "0.1.10", 402 | "proxy-addr": "~2.0.7", 403 | "qs": "6.13.0", 404 | "range-parser": "~1.2.1", 405 | "safe-buffer": "5.2.1", 406 | "send": "0.19.0", 407 | "serve-static": "1.16.2", 408 | "setprototypeof": "1.2.0", 409 | "statuses": "2.0.1", 410 | "type-is": "~1.6.18", 411 | "utils-merge": "1.0.1", 412 | "vary": "~1.1.2" 413 | }, 414 | "engines": { 415 | "node": ">= 0.10.0" 416 | } 417 | }, 418 | "node_modules/fill-range": { 419 | "version": "7.1.1", 420 | "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.1.1.tgz", 421 | "integrity": "sha512-YsGpe3WHLK8ZYi4tWDg2Jy3ebRz2rXowDxnld4bkQB00cc/1Zw9AWnC0i9ztDJitivtQvaI9KaLyKrc+hBW0yg==", 422 | "dev": true, 423 | "license": "MIT", 424 | "dependencies": { 425 | "to-regex-range": "^5.0.1" 426 | }, 427 | "engines": { 428 | "node": ">=8" 429 | } 430 | }, 431 | "node_modules/finalhandler": { 432 | "version": "1.3.1", 433 | "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.3.1.tgz", 434 | "integrity": "sha512-6BN9trH7bp3qvnrRyzsBz+g3lZxTNZTbVO2EV1CS0WIcDbawYVdYvGflME/9QP0h0pYlCDBCTjYa9nZzMDpyxQ==", 435 | "license": "MIT", 436 | "dependencies": { 437 | "debug": "2.6.9", 438 | "encodeurl": "~2.0.0", 439 | "escape-html": "~1.0.3", 440 | "on-finished": "2.4.1", 441 | "parseurl": "~1.3.3", 442 | "statuses": "2.0.1", 443 | "unpipe": "~1.0.0" 444 | }, 445 | "engines": { 446 | "node": ">= 0.8" 447 | } 448 | }, 449 | "node_modules/forwarded": { 450 | "version": "0.2.0", 451 | "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.2.0.tgz", 452 | "integrity": "sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow==", 453 | "license": "MIT", 454 | "engines": { 455 | "node": ">= 0.6" 456 | } 457 | }, 458 | "node_modules/fresh": { 459 | "version": "0.5.2", 460 | "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz", 461 | "integrity": "sha512-zJ2mQYM18rEFOudeV4GShTGIQ7RbzA7ozbU9I/XBpm7kqgMywgmylMwXHxZJmkVoYkna9d2pVXVXPdYTP9ej8Q==", 462 | "license": "MIT", 463 | "engines": { 464 | "node": ">= 0.6" 465 | } 466 | }, 467 | "node_modules/fsevents": { 468 | "version": "2.3.3", 469 | "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.3.tgz", 470 | "integrity": "sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==", 471 | "dev": true, 472 | "hasInstallScript": true, 473 | "license": "MIT", 474 | "optional": true, 475 | "os": [ 476 | "darwin" 477 | ], 478 | "engines": { 479 | "node": "^8.16.0 || ^10.6.0 || >=11.0.0" 480 | } 481 | }, 482 | "node_modules/function-bind": { 483 | "version": "1.1.2", 484 | "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.2.tgz", 485 | "integrity": "sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==", 486 | "license": "MIT", 487 | "funding": { 488 | "url": "https://github.com/sponsors/ljharb" 489 | } 490 | }, 491 | "node_modules/get-intrinsic": { 492 | "version": "1.2.4", 493 | "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.2.4.tgz", 494 | "integrity": "sha512-5uYhsJH8VJBTv7oslg4BznJYhDoRI6waYCxMmCdnTrcCrHA/fCFKoTFz2JKKE0HdDFUF7/oQuhzumXJK7paBRQ==", 495 | "license": "MIT", 496 | "dependencies": { 497 | "es-errors": "^1.3.0", 498 | "function-bind": "^1.1.2", 499 | "has-proto": "^1.0.1", 500 | "has-symbols": "^1.0.3", 501 | "hasown": "^2.0.0" 502 | }, 503 | "engines": { 504 | "node": ">= 0.4" 505 | }, 506 | "funding": { 507 | "url": "https://github.com/sponsors/ljharb" 508 | } 509 | }, 510 | "node_modules/glob-parent": { 511 | "version": "5.1.2", 512 | "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", 513 | "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", 514 | "dev": true, 515 | "license": "ISC", 516 | "dependencies": { 517 | "is-glob": "^4.0.1" 518 | }, 519 | "engines": { 520 | "node": ">= 6" 521 | } 522 | }, 523 | "node_modules/gopd": { 524 | "version": "1.0.1", 525 | "resolved": "https://registry.npmjs.org/gopd/-/gopd-1.0.1.tgz", 526 | "integrity": "sha512-d65bNlIadxvpb/A2abVdlqKqV563juRnZ1Wtk6s1sIR8uNsXR70xqIzVqxVf1eTqDunwT2MkczEeaezCKTZhwA==", 527 | "license": "MIT", 528 | "dependencies": { 529 | "get-intrinsic": "^1.1.3" 530 | }, 531 | "funding": { 532 | "url": "https://github.com/sponsors/ljharb" 533 | } 534 | }, 535 | "node_modules/has-flag": { 536 | "version": "3.0.0", 537 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", 538 | "integrity": "sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==", 539 | "dev": true, 540 | "license": "MIT", 541 | "engines": { 542 | "node": ">=4" 543 | } 544 | }, 545 | "node_modules/has-property-descriptors": { 546 | "version": "1.0.2", 547 | "resolved": "https://registry.npmjs.org/has-property-descriptors/-/has-property-descriptors-1.0.2.tgz", 548 | "integrity": "sha512-55JNKuIW+vq4Ke1BjOTjM2YctQIvCT7GFzHwmfZPGo5wnrgkid0YQtnAleFSqumZm4az3n2BS+erby5ipJdgrg==", 549 | "license": "MIT", 550 | "dependencies": { 551 | "es-define-property": "^1.0.0" 552 | }, 553 | "funding": { 554 | "url": "https://github.com/sponsors/ljharb" 555 | } 556 | }, 557 | "node_modules/has-proto": { 558 | "version": "1.0.3", 559 | "resolved": "https://registry.npmjs.org/has-proto/-/has-proto-1.0.3.tgz", 560 | "integrity": "sha512-SJ1amZAJUiZS+PhsVLf5tGydlaVB8EdFpaSO4gmiUKUOxk8qzn5AIy4ZeJUmh22znIdk/uMAUT2pl3FxzVUH+Q==", 561 | "license": "MIT", 562 | "engines": { 563 | "node": ">= 0.4" 564 | }, 565 | "funding": { 566 | "url": "https://github.com/sponsors/ljharb" 567 | } 568 | }, 569 | "node_modules/has-symbols": { 570 | "version": "1.0.3", 571 | "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.0.3.tgz", 572 | "integrity": "sha512-l3LCuF6MgDNwTDKkdYGEihYjt5pRPbEg46rtlmnSPlUbgmB8LOIrKJbYYFBSbnPaJexMKtiPO8hmeRjRz2Td+A==", 573 | "license": "MIT", 574 | "engines": { 575 | "node": ">= 0.4" 576 | }, 577 | "funding": { 578 | "url": "https://github.com/sponsors/ljharb" 579 | } 580 | }, 581 | "node_modules/hasown": { 582 | "version": "2.0.2", 583 | "resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.2.tgz", 584 | "integrity": "sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==", 585 | "license": "MIT", 586 | "dependencies": { 587 | "function-bind": "^1.1.2" 588 | }, 589 | "engines": { 590 | "node": ">= 0.4" 591 | } 592 | }, 593 | "node_modules/http-errors": { 594 | "version": "2.0.0", 595 | "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-2.0.0.tgz", 596 | "integrity": "sha512-FtwrG/euBzaEjYeRqOgly7G0qviiXoJWnvEH2Z1plBdXgbyjv34pHTSb9zoeHMyDy33+DWy5Wt9Wo+TURtOYSQ==", 597 | "license": "MIT", 598 | "dependencies": { 599 | "depd": "2.0.0", 600 | "inherits": "2.0.4", 601 | "setprototypeof": "1.2.0", 602 | "statuses": "2.0.1", 603 | "toidentifier": "1.0.1" 604 | }, 605 | "engines": { 606 | "node": ">= 0.8" 607 | } 608 | }, 609 | "node_modules/iconv-lite": { 610 | "version": "0.4.24", 611 | "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", 612 | "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", 613 | "license": "MIT", 614 | "dependencies": { 615 | "safer-buffer": ">= 2.1.2 < 3" 616 | }, 617 | "engines": { 618 | "node": ">=0.10.0" 619 | } 620 | }, 621 | "node_modules/ignore-by-default": { 622 | "version": "1.0.1", 623 | "resolved": "https://registry.npmjs.org/ignore-by-default/-/ignore-by-default-1.0.1.tgz", 624 | "integrity": "sha512-Ius2VYcGNk7T90CppJqcIkS5ooHUZyIQK+ClZfMfMNFEF9VSE73Fq+906u/CWu92x4gzZMWOwfFYckPObzdEbA==", 625 | "dev": true, 626 | "license": "ISC" 627 | }, 628 | "node_modules/inherits": { 629 | "version": "2.0.4", 630 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", 631 | "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==", 632 | "license": "ISC" 633 | }, 634 | "node_modules/ipaddr.js": { 635 | "version": "1.9.1", 636 | "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz", 637 | "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==", 638 | "license": "MIT", 639 | "engines": { 640 | "node": ">= 0.10" 641 | } 642 | }, 643 | "node_modules/is-binary-path": { 644 | "version": "2.1.0", 645 | "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz", 646 | "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==", 647 | "dev": true, 648 | "license": "MIT", 649 | "dependencies": { 650 | "binary-extensions": "^2.0.0" 651 | }, 652 | "engines": { 653 | "node": ">=8" 654 | } 655 | }, 656 | "node_modules/is-extglob": { 657 | "version": "2.1.1", 658 | "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", 659 | "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==", 660 | "dev": true, 661 | "license": "MIT", 662 | "engines": { 663 | "node": ">=0.10.0" 664 | } 665 | }, 666 | "node_modules/is-glob": { 667 | "version": "4.0.3", 668 | "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz", 669 | "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==", 670 | "dev": true, 671 | "license": "MIT", 672 | "dependencies": { 673 | "is-extglob": "^2.1.1" 674 | }, 675 | "engines": { 676 | "node": ">=0.10.0" 677 | } 678 | }, 679 | "node_modules/is-number": { 680 | "version": "7.0.0", 681 | "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", 682 | "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", 683 | "dev": true, 684 | "license": "MIT", 685 | "engines": { 686 | "node": ">=0.12.0" 687 | } 688 | }, 689 | "node_modules/joi": { 690 | "version": "17.13.3", 691 | "resolved": "https://registry.npmjs.org/joi/-/joi-17.13.3.tgz", 692 | "integrity": "sha512-otDA4ldcIx+ZXsKHWmp0YizCweVRZG96J10b0FevjfuncLO1oX59THoAmHkNubYJ+9gWsYsp5k8v4ib6oDv1fA==", 693 | "license": "BSD-3-Clause", 694 | "dependencies": { 695 | "@hapi/hoek": "^9.3.0", 696 | "@hapi/topo": "^5.1.0", 697 | "@sideway/address": "^4.1.5", 698 | "@sideway/formula": "^3.0.1", 699 | "@sideway/pinpoint": "^2.0.0" 700 | } 701 | }, 702 | "node_modules/media-typer": { 703 | "version": "0.3.0", 704 | "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", 705 | "integrity": "sha512-dq+qelQ9akHpcOl/gUVRTxVIOkAJ1wR3QAvb4RsVjS8oVoFjDGTc679wJYmUmknUF5HwMLOgb5O+a3KxfWapPQ==", 706 | "license": "MIT", 707 | "engines": { 708 | "node": ">= 0.6" 709 | } 710 | }, 711 | "node_modules/merge-descriptors": { 712 | "version": "1.0.3", 713 | "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.3.tgz", 714 | "integrity": "sha512-gaNvAS7TZ897/rVaZ0nMtAyxNyi/pdbjbAwUpFQpN70GqnVfOiXpeUUMKRBmzXaSQ8DdTX4/0ms62r2K+hE6mQ==", 715 | "license": "MIT", 716 | "funding": { 717 | "url": "https://github.com/sponsors/sindresorhus" 718 | } 719 | }, 720 | "node_modules/methods": { 721 | "version": "1.1.2", 722 | "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz", 723 | "integrity": "sha512-iclAHeNqNm68zFtnZ0e+1L2yUIdvzNoauKU4WBA3VvH/vPFieF7qfRlwUZU+DA9P9bPXIS90ulxoUoCH23sV2w==", 724 | "license": "MIT", 725 | "engines": { 726 | "node": ">= 0.6" 727 | } 728 | }, 729 | "node_modules/mime": { 730 | "version": "1.6.0", 731 | "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz", 732 | "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==", 733 | "license": "MIT", 734 | "bin": { 735 | "mime": "cli.js" 736 | }, 737 | "engines": { 738 | "node": ">=4" 739 | } 740 | }, 741 | "node_modules/mime-db": { 742 | "version": "1.52.0", 743 | "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz", 744 | "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==", 745 | "license": "MIT", 746 | "engines": { 747 | "node": ">= 0.6" 748 | } 749 | }, 750 | "node_modules/mime-types": { 751 | "version": "2.1.35", 752 | "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz", 753 | "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==", 754 | "license": "MIT", 755 | "dependencies": { 756 | "mime-db": "1.52.0" 757 | }, 758 | "engines": { 759 | "node": ">= 0.6" 760 | } 761 | }, 762 | "node_modules/minimatch": { 763 | "version": "3.1.2", 764 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", 765 | "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", 766 | "dev": true, 767 | "license": "ISC", 768 | "dependencies": { 769 | "brace-expansion": "^1.1.7" 770 | }, 771 | "engines": { 772 | "node": "*" 773 | } 774 | }, 775 | "node_modules/ms": { 776 | "version": "2.0.0", 777 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", 778 | "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==", 779 | "license": "MIT" 780 | }, 781 | "node_modules/negotiator": { 782 | "version": "0.6.3", 783 | "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.3.tgz", 784 | "integrity": "sha512-+EUsqGPLsM+j/zdChZjsnX51g4XrHFOIXwfnCVPGlQk/k5giakcKsuxCObBRu6DSm9opw/O6slWbJdghQM4bBg==", 785 | "license": "MIT", 786 | "engines": { 787 | "node": ">= 0.6" 788 | } 789 | }, 790 | "node_modules/nodemon": { 791 | "version": "3.1.7", 792 | "resolved": "https://registry.npmjs.org/nodemon/-/nodemon-3.1.7.tgz", 793 | "integrity": "sha512-hLj7fuMow6f0lbB0cD14Lz2xNjwsyruH251Pk4t/yIitCFJbmY1myuLlHm/q06aST4jg6EgAh74PIBBrRqpVAQ==", 794 | "dev": true, 795 | "license": "MIT", 796 | "dependencies": { 797 | "chokidar": "^3.5.2", 798 | "debug": "^4", 799 | "ignore-by-default": "^1.0.1", 800 | "minimatch": "^3.1.2", 801 | "pstree.remy": "^1.1.8", 802 | "semver": "^7.5.3", 803 | "simple-update-notifier": "^2.0.0", 804 | "supports-color": "^5.5.0", 805 | "touch": "^3.1.0", 806 | "undefsafe": "^2.0.5" 807 | }, 808 | "bin": { 809 | "nodemon": "bin/nodemon.js" 810 | }, 811 | "engines": { 812 | "node": ">=10" 813 | }, 814 | "funding": { 815 | "type": "opencollective", 816 | "url": "https://opencollective.com/nodemon" 817 | } 818 | }, 819 | "node_modules/nodemon/node_modules/debug": { 820 | "version": "4.3.7", 821 | "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.7.tgz", 822 | "integrity": "sha512-Er2nc/H7RrMXZBFCEim6TCmMk02Z8vLC2Rbi1KEBggpo0fS6l0S1nnapwmIi3yW/+GOJap1Krg4w0Hg80oCqgQ==", 823 | "dev": true, 824 | "license": "MIT", 825 | "dependencies": { 826 | "ms": "^2.1.3" 827 | }, 828 | "engines": { 829 | "node": ">=6.0" 830 | }, 831 | "peerDependenciesMeta": { 832 | "supports-color": { 833 | "optional": true 834 | } 835 | } 836 | }, 837 | "node_modules/nodemon/node_modules/ms": { 838 | "version": "2.1.3", 839 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", 840 | "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", 841 | "dev": true, 842 | "license": "MIT" 843 | }, 844 | "node_modules/normalize-path": { 845 | "version": "3.0.0", 846 | "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", 847 | "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==", 848 | "dev": true, 849 | "license": "MIT", 850 | "engines": { 851 | "node": ">=0.10.0" 852 | } 853 | }, 854 | "node_modules/object-assign": { 855 | "version": "4.1.1", 856 | "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", 857 | "integrity": "sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==", 858 | "license": "MIT", 859 | "engines": { 860 | "node": ">=0.10.0" 861 | } 862 | }, 863 | "node_modules/object-inspect": { 864 | "version": "1.13.2", 865 | "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.13.2.tgz", 866 | "integrity": "sha512-IRZSRuzJiynemAXPYtPe5BoI/RESNYR7TYm50MC5Mqbd3Jmw5y790sErYw3V6SryFJD64b74qQQs9wn5Bg/k3g==", 867 | "license": "MIT", 868 | "engines": { 869 | "node": ">= 0.4" 870 | }, 871 | "funding": { 872 | "url": "https://github.com/sponsors/ljharb" 873 | } 874 | }, 875 | "node_modules/on-finished": { 876 | "version": "2.4.1", 877 | "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.4.1.tgz", 878 | "integrity": "sha512-oVlzkg3ENAhCk2zdv7IJwd/QUD4z2RxRwpkcGY8psCVcCYZNq4wYnVWALHM+brtuJjePWiYF/ClmuDr8Ch5+kg==", 879 | "license": "MIT", 880 | "dependencies": { 881 | "ee-first": "1.1.1" 882 | }, 883 | "engines": { 884 | "node": ">= 0.8" 885 | } 886 | }, 887 | "node_modules/parseurl": { 888 | "version": "1.3.3", 889 | "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz", 890 | "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==", 891 | "license": "MIT", 892 | "engines": { 893 | "node": ">= 0.8" 894 | } 895 | }, 896 | "node_modules/path-to-regexp": { 897 | "version": "0.1.10", 898 | "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.10.tgz", 899 | "integrity": "sha512-7lf7qcQidTku0Gu3YDPc8DJ1q7OOucfa/BSsIwjuh56VU7katFvuM8hULfkwB3Fns/rsVF7PwPKVw1sl5KQS9w==", 900 | "license": "MIT" 901 | }, 902 | "node_modules/pg": { 903 | "version": "8.13.1", 904 | "resolved": "https://registry.npmjs.org/pg/-/pg-8.13.1.tgz", 905 | "integrity": "sha512-OUir1A0rPNZlX//c7ksiu7crsGZTKSOXJPgtNiHGIlC9H0lO+NC6ZDYksSgBYY/thSWhnSRBv8w1lieNNGATNQ==", 906 | "license": "MIT", 907 | "dependencies": { 908 | "pg-connection-string": "^2.7.0", 909 | "pg-pool": "^3.7.0", 910 | "pg-protocol": "^1.7.0", 911 | "pg-types": "^2.1.0", 912 | "pgpass": "1.x" 913 | }, 914 | "engines": { 915 | "node": ">= 8.0.0" 916 | }, 917 | "optionalDependencies": { 918 | "pg-cloudflare": "^1.1.1" 919 | }, 920 | "peerDependencies": { 921 | "pg-native": ">=3.0.1" 922 | }, 923 | "peerDependenciesMeta": { 924 | "pg-native": { 925 | "optional": true 926 | } 927 | } 928 | }, 929 | "node_modules/pg-cloudflare": { 930 | "version": "1.1.1", 931 | "resolved": "https://registry.npmjs.org/pg-cloudflare/-/pg-cloudflare-1.1.1.tgz", 932 | "integrity": "sha512-xWPagP/4B6BgFO+EKz3JONXv3YDgvkbVrGw2mTo3D6tVDQRh1e7cqVGvyR3BE+eQgAvx1XhW/iEASj4/jCWl3Q==", 933 | "license": "MIT", 934 | "optional": true 935 | }, 936 | "node_modules/pg-connection-string": { 937 | "version": "2.7.0", 938 | "resolved": "https://registry.npmjs.org/pg-connection-string/-/pg-connection-string-2.7.0.tgz", 939 | "integrity": "sha512-PI2W9mv53rXJQEOb8xNR8lH7Hr+EKa6oJa38zsK0S/ky2er16ios1wLKhZyxzD7jUReiWokc9WK5nxSnC7W1TA==", 940 | "license": "MIT" 941 | }, 942 | "node_modules/pg-int8": { 943 | "version": "1.0.1", 944 | "resolved": "https://registry.npmjs.org/pg-int8/-/pg-int8-1.0.1.tgz", 945 | "integrity": "sha512-WCtabS6t3c8SkpDBUlb1kjOs7l66xsGdKpIPZsg4wR+B3+u9UAum2odSsF9tnvxg80h4ZxLWMy4pRjOsFIqQpw==", 946 | "license": "ISC", 947 | "engines": { 948 | "node": ">=4.0.0" 949 | } 950 | }, 951 | "node_modules/pg-pool": { 952 | "version": "3.7.0", 953 | "resolved": "https://registry.npmjs.org/pg-pool/-/pg-pool-3.7.0.tgz", 954 | "integrity": "sha512-ZOBQForurqh4zZWjrgSwwAtzJ7QiRX0ovFkZr2klsen3Nm0aoh33Ls0fzfv3imeH/nw/O27cjdz5kzYJfeGp/g==", 955 | "license": "MIT", 956 | "peerDependencies": { 957 | "pg": ">=8.0" 958 | } 959 | }, 960 | "node_modules/pg-protocol": { 961 | "version": "1.7.0", 962 | "resolved": "https://registry.npmjs.org/pg-protocol/-/pg-protocol-1.7.0.tgz", 963 | "integrity": "sha512-hTK/mE36i8fDDhgDFjy6xNOG+LCorxLG3WO17tku+ij6sVHXh1jQUJ8hYAnRhNla4QVD2H8er/FOjc/+EgC6yQ==", 964 | "license": "MIT" 965 | }, 966 | "node_modules/pg-types": { 967 | "version": "2.2.0", 968 | "resolved": "https://registry.npmjs.org/pg-types/-/pg-types-2.2.0.tgz", 969 | "integrity": "sha512-qTAAlrEsl8s4OiEQY69wDvcMIdQN6wdz5ojQiOy6YRMuynxenON0O5oCpJI6lshc6scgAY8qvJ2On/p+CXY0GA==", 970 | "license": "MIT", 971 | "dependencies": { 972 | "pg-int8": "1.0.1", 973 | "postgres-array": "~2.0.0", 974 | "postgres-bytea": "~1.0.0", 975 | "postgres-date": "~1.0.4", 976 | "postgres-interval": "^1.1.0" 977 | }, 978 | "engines": { 979 | "node": ">=4" 980 | } 981 | }, 982 | "node_modules/pgpass": { 983 | "version": "1.0.5", 984 | "resolved": "https://registry.npmjs.org/pgpass/-/pgpass-1.0.5.tgz", 985 | "integrity": "sha512-FdW9r/jQZhSeohs1Z3sI1yxFQNFvMcnmfuj4WBMUTxOrAyLMaTcE1aAMBiTlbMNaXvBCQuVi0R7hd8udDSP7ug==", 986 | "license": "MIT", 987 | "dependencies": { 988 | "split2": "^4.1.0" 989 | } 990 | }, 991 | "node_modules/picomatch": { 992 | "version": "2.3.1", 993 | "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", 994 | "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==", 995 | "dev": true, 996 | "license": "MIT", 997 | "engines": { 998 | "node": ">=8.6" 999 | }, 1000 | "funding": { 1001 | "url": "https://github.com/sponsors/jonschlinkert" 1002 | } 1003 | }, 1004 | "node_modules/postgres-array": { 1005 | "version": "2.0.0", 1006 | "resolved": "https://registry.npmjs.org/postgres-array/-/postgres-array-2.0.0.tgz", 1007 | "integrity": "sha512-VpZrUqU5A69eQyW2c5CA1jtLecCsN2U/bD6VilrFDWq5+5UIEVO7nazS3TEcHf1zuPYO/sqGvUvW62g86RXZuA==", 1008 | "license": "MIT", 1009 | "engines": { 1010 | "node": ">=4" 1011 | } 1012 | }, 1013 | "node_modules/postgres-bytea": { 1014 | "version": "1.0.0", 1015 | "resolved": "https://registry.npmjs.org/postgres-bytea/-/postgres-bytea-1.0.0.tgz", 1016 | "integrity": "sha512-xy3pmLuQqRBZBXDULy7KbaitYqLcmxigw14Q5sj8QBVLqEwXfeybIKVWiqAXTlcvdvb0+xkOtDbfQMOf4lST1w==", 1017 | "license": "MIT", 1018 | "engines": { 1019 | "node": ">=0.10.0" 1020 | } 1021 | }, 1022 | "node_modules/postgres-date": { 1023 | "version": "1.0.7", 1024 | "resolved": "https://registry.npmjs.org/postgres-date/-/postgres-date-1.0.7.tgz", 1025 | "integrity": "sha512-suDmjLVQg78nMK2UZ454hAG+OAW+HQPZ6n++TNDUX+L0+uUlLywnoxJKDou51Zm+zTCjrCl0Nq6J9C5hP9vK/Q==", 1026 | "license": "MIT", 1027 | "engines": { 1028 | "node": ">=0.10.0" 1029 | } 1030 | }, 1031 | "node_modules/postgres-interval": { 1032 | "version": "1.2.0", 1033 | "resolved": "https://registry.npmjs.org/postgres-interval/-/postgres-interval-1.2.0.tgz", 1034 | "integrity": "sha512-9ZhXKM/rw350N1ovuWHbGxnGh/SNJ4cnxHiM0rxE4VN41wsg8P8zWn9hv/buK00RP4WvlOyr/RBDiptyxVbkZQ==", 1035 | "license": "MIT", 1036 | "dependencies": { 1037 | "xtend": "^4.0.0" 1038 | }, 1039 | "engines": { 1040 | "node": ">=0.10.0" 1041 | } 1042 | }, 1043 | "node_modules/proxy-addr": { 1044 | "version": "2.0.7", 1045 | "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.7.tgz", 1046 | "integrity": "sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg==", 1047 | "license": "MIT", 1048 | "dependencies": { 1049 | "forwarded": "0.2.0", 1050 | "ipaddr.js": "1.9.1" 1051 | }, 1052 | "engines": { 1053 | "node": ">= 0.10" 1054 | } 1055 | }, 1056 | "node_modules/pstree.remy": { 1057 | "version": "1.1.8", 1058 | "resolved": "https://registry.npmjs.org/pstree.remy/-/pstree.remy-1.1.8.tgz", 1059 | "integrity": "sha512-77DZwxQmxKnu3aR542U+X8FypNzbfJ+C5XQDk3uWjWxn6151aIMGthWYRXTqT1E5oJvg+ljaa2OJi+VfvCOQ8w==", 1060 | "dev": true, 1061 | "license": "MIT" 1062 | }, 1063 | "node_modules/qs": { 1064 | "version": "6.13.0", 1065 | "resolved": "https://registry.npmjs.org/qs/-/qs-6.13.0.tgz", 1066 | "integrity": "sha512-+38qI9SOr8tfZ4QmJNplMUxqjbe7LKvvZgWdExBOmd+egZTtjLB67Gu0HRX3u/XOq7UU2Nx6nsjvS16Z9uwfpg==", 1067 | "license": "BSD-3-Clause", 1068 | "dependencies": { 1069 | "side-channel": "^1.0.6" 1070 | }, 1071 | "engines": { 1072 | "node": ">=0.6" 1073 | }, 1074 | "funding": { 1075 | "url": "https://github.com/sponsors/ljharb" 1076 | } 1077 | }, 1078 | "node_modules/range-parser": { 1079 | "version": "1.2.1", 1080 | "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz", 1081 | "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==", 1082 | "license": "MIT", 1083 | "engines": { 1084 | "node": ">= 0.6" 1085 | } 1086 | }, 1087 | "node_modules/raw-body": { 1088 | "version": "2.5.2", 1089 | "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.5.2.tgz", 1090 | "integrity": "sha512-8zGqypfENjCIqGhgXToC8aB2r7YrBX+AQAfIPs/Mlk+BtPTztOvTS01NRW/3Eh60J+a48lt8qsCzirQ6loCVfA==", 1091 | "license": "MIT", 1092 | "dependencies": { 1093 | "bytes": "3.1.2", 1094 | "http-errors": "2.0.0", 1095 | "iconv-lite": "0.4.24", 1096 | "unpipe": "1.0.0" 1097 | }, 1098 | "engines": { 1099 | "node": ">= 0.8" 1100 | } 1101 | }, 1102 | "node_modules/readdirp": { 1103 | "version": "3.6.0", 1104 | "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.6.0.tgz", 1105 | "integrity": "sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==", 1106 | "dev": true, 1107 | "license": "MIT", 1108 | "dependencies": { 1109 | "picomatch": "^2.2.1" 1110 | }, 1111 | "engines": { 1112 | "node": ">=8.10.0" 1113 | } 1114 | }, 1115 | "node_modules/safe-buffer": { 1116 | "version": "5.2.1", 1117 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz", 1118 | "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==", 1119 | "funding": [ 1120 | { 1121 | "type": "github", 1122 | "url": "https://github.com/sponsors/feross" 1123 | }, 1124 | { 1125 | "type": "patreon", 1126 | "url": "https://www.patreon.com/feross" 1127 | }, 1128 | { 1129 | "type": "consulting", 1130 | "url": "https://feross.org/support" 1131 | } 1132 | ], 1133 | "license": "MIT" 1134 | }, 1135 | "node_modules/safer-buffer": { 1136 | "version": "2.1.2", 1137 | "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", 1138 | "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==", 1139 | "license": "MIT" 1140 | }, 1141 | "node_modules/semver": { 1142 | "version": "7.6.3", 1143 | "resolved": "https://registry.npmjs.org/semver/-/semver-7.6.3.tgz", 1144 | "integrity": "sha512-oVekP1cKtI+CTDvHWYFUcMtsK/00wmAEfyqKfNdARm8u1wNVhSgaX7A8d4UuIlUI5e84iEwOhs7ZPYRmzU9U6A==", 1145 | "dev": true, 1146 | "license": "ISC", 1147 | "bin": { 1148 | "semver": "bin/semver.js" 1149 | }, 1150 | "engines": { 1151 | "node": ">=10" 1152 | } 1153 | }, 1154 | "node_modules/send": { 1155 | "version": "0.19.0", 1156 | "resolved": "https://registry.npmjs.org/send/-/send-0.19.0.tgz", 1157 | "integrity": "sha512-dW41u5VfLXu8SJh5bwRmyYUbAoSB3c9uQh6L8h/KtsFREPWpbX1lrljJo186Jc4nmci/sGUZ9a0a0J2zgfq2hw==", 1158 | "license": "MIT", 1159 | "dependencies": { 1160 | "debug": "2.6.9", 1161 | "depd": "2.0.0", 1162 | "destroy": "1.2.0", 1163 | "encodeurl": "~1.0.2", 1164 | "escape-html": "~1.0.3", 1165 | "etag": "~1.8.1", 1166 | "fresh": "0.5.2", 1167 | "http-errors": "2.0.0", 1168 | "mime": "1.6.0", 1169 | "ms": "2.1.3", 1170 | "on-finished": "2.4.1", 1171 | "range-parser": "~1.2.1", 1172 | "statuses": "2.0.1" 1173 | }, 1174 | "engines": { 1175 | "node": ">= 0.8.0" 1176 | } 1177 | }, 1178 | "node_modules/send/node_modules/encodeurl": { 1179 | "version": "1.0.2", 1180 | "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz", 1181 | "integrity": "sha512-TPJXq8JqFaVYm2CWmPvnP2Iyo4ZSM7/QKcSmuMLDObfpH5fi7RUGmd/rTDf+rut/saiDiQEeVTNgAmJEdAOx0w==", 1182 | "license": "MIT", 1183 | "engines": { 1184 | "node": ">= 0.8" 1185 | } 1186 | }, 1187 | "node_modules/send/node_modules/ms": { 1188 | "version": "2.1.3", 1189 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", 1190 | "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", 1191 | "license": "MIT" 1192 | }, 1193 | "node_modules/serve-static": { 1194 | "version": "1.16.2", 1195 | "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.16.2.tgz", 1196 | "integrity": "sha512-VqpjJZKadQB/PEbEwvFdO43Ax5dFBZ2UECszz8bQ7pi7wt//PWe1P6MN7eCnjsatYtBT6EuiClbjSWP2WrIoTw==", 1197 | "license": "MIT", 1198 | "dependencies": { 1199 | "encodeurl": "~2.0.0", 1200 | "escape-html": "~1.0.3", 1201 | "parseurl": "~1.3.3", 1202 | "send": "0.19.0" 1203 | }, 1204 | "engines": { 1205 | "node": ">= 0.8.0" 1206 | } 1207 | }, 1208 | "node_modules/set-function-length": { 1209 | "version": "1.2.2", 1210 | "resolved": "https://registry.npmjs.org/set-function-length/-/set-function-length-1.2.2.tgz", 1211 | "integrity": "sha512-pgRc4hJ4/sNjWCSS9AmnS40x3bNMDTknHgL5UaMBTMyJnU90EgWh1Rz+MC9eFu4BuN/UwZjKQuY/1v3rM7HMfg==", 1212 | "license": "MIT", 1213 | "dependencies": { 1214 | "define-data-property": "^1.1.4", 1215 | "es-errors": "^1.3.0", 1216 | "function-bind": "^1.1.2", 1217 | "get-intrinsic": "^1.2.4", 1218 | "gopd": "^1.0.1", 1219 | "has-property-descriptors": "^1.0.2" 1220 | }, 1221 | "engines": { 1222 | "node": ">= 0.4" 1223 | } 1224 | }, 1225 | "node_modules/setprototypeof": { 1226 | "version": "1.2.0", 1227 | "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.2.0.tgz", 1228 | "integrity": "sha512-E5LDX7Wrp85Kil5bhZv46j8jOeboKq5JMmYM3gVGdGH8xFpPWXUMsNrlODCrkoxMEeNi/XZIwuRvY4XNwYMJpw==", 1229 | "license": "ISC" 1230 | }, 1231 | "node_modules/side-channel": { 1232 | "version": "1.0.6", 1233 | "resolved": "https://registry.npmjs.org/side-channel/-/side-channel-1.0.6.tgz", 1234 | "integrity": "sha512-fDW/EZ6Q9RiO8eFG8Hj+7u/oW+XrPTIChwCOM2+th2A6OblDtYYIpve9m+KvI9Z4C9qSEXlaGR6bTEYHReuglA==", 1235 | "license": "MIT", 1236 | "dependencies": { 1237 | "call-bind": "^1.0.7", 1238 | "es-errors": "^1.3.0", 1239 | "get-intrinsic": "^1.2.4", 1240 | "object-inspect": "^1.13.1" 1241 | }, 1242 | "engines": { 1243 | "node": ">= 0.4" 1244 | }, 1245 | "funding": { 1246 | "url": "https://github.com/sponsors/ljharb" 1247 | } 1248 | }, 1249 | "node_modules/simple-update-notifier": { 1250 | "version": "2.0.0", 1251 | "resolved": "https://registry.npmjs.org/simple-update-notifier/-/simple-update-notifier-2.0.0.tgz", 1252 | "integrity": "sha512-a2B9Y0KlNXl9u/vsW6sTIu9vGEpfKu2wRV6l1H3XEas/0gUIzGzBoP/IouTcUQbm9JWZLH3COxyn03TYlFax6w==", 1253 | "dev": true, 1254 | "license": "MIT", 1255 | "dependencies": { 1256 | "semver": "^7.5.3" 1257 | }, 1258 | "engines": { 1259 | "node": ">=10" 1260 | } 1261 | }, 1262 | "node_modules/split2": { 1263 | "version": "4.2.0", 1264 | "resolved": "https://registry.npmjs.org/split2/-/split2-4.2.0.tgz", 1265 | "integrity": "sha512-UcjcJOWknrNkF6PLX83qcHM6KHgVKNkV62Y8a5uYDVv9ydGQVwAHMKqHdJje1VTWpljG0WYpCDhrCdAOYH4TWg==", 1266 | "license": "ISC", 1267 | "engines": { 1268 | "node": ">= 10.x" 1269 | } 1270 | }, 1271 | "node_modules/statuses": { 1272 | "version": "2.0.1", 1273 | "resolved": "https://registry.npmjs.org/statuses/-/statuses-2.0.1.tgz", 1274 | "integrity": "sha512-RwNA9Z/7PrK06rYLIzFMlaF+l73iwpzsqRIFgbMLbTcLD6cOao82TaWefPXQvB2fOC4AjuYSEndS7N/mTCbkdQ==", 1275 | "license": "MIT", 1276 | "engines": { 1277 | "node": ">= 0.8" 1278 | } 1279 | }, 1280 | "node_modules/supports-color": { 1281 | "version": "5.5.0", 1282 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", 1283 | "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", 1284 | "dev": true, 1285 | "license": "MIT", 1286 | "dependencies": { 1287 | "has-flag": "^3.0.0" 1288 | }, 1289 | "engines": { 1290 | "node": ">=4" 1291 | } 1292 | }, 1293 | "node_modules/to-regex-range": { 1294 | "version": "5.0.1", 1295 | "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", 1296 | "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", 1297 | "dev": true, 1298 | "license": "MIT", 1299 | "dependencies": { 1300 | "is-number": "^7.0.0" 1301 | }, 1302 | "engines": { 1303 | "node": ">=8.0" 1304 | } 1305 | }, 1306 | "node_modules/toidentifier": { 1307 | "version": "1.0.1", 1308 | "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.1.tgz", 1309 | "integrity": "sha512-o5sSPKEkg/DIQNmH43V0/uerLrpzVedkUh8tGNvaeXpfpuwjKenlSox/2O/BTlZUtEe+JG7s5YhEz608PlAHRA==", 1310 | "license": "MIT", 1311 | "engines": { 1312 | "node": ">=0.6" 1313 | } 1314 | }, 1315 | "node_modules/touch": { 1316 | "version": "3.1.1", 1317 | "resolved": "https://registry.npmjs.org/touch/-/touch-3.1.1.tgz", 1318 | "integrity": "sha512-r0eojU4bI8MnHr8c5bNo7lJDdI2qXlWWJk6a9EAFG7vbhTjElYhBVS3/miuE0uOuoLdb8Mc/rVfsmm6eo5o9GA==", 1319 | "dev": true, 1320 | "license": "ISC", 1321 | "bin": { 1322 | "nodetouch": "bin/nodetouch.js" 1323 | } 1324 | }, 1325 | "node_modules/type-is": { 1326 | "version": "1.6.18", 1327 | "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz", 1328 | "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==", 1329 | "license": "MIT", 1330 | "dependencies": { 1331 | "media-typer": "0.3.0", 1332 | "mime-types": "~2.1.24" 1333 | }, 1334 | "engines": { 1335 | "node": ">= 0.6" 1336 | } 1337 | }, 1338 | "node_modules/undefsafe": { 1339 | "version": "2.0.5", 1340 | "resolved": "https://registry.npmjs.org/undefsafe/-/undefsafe-2.0.5.tgz", 1341 | "integrity": "sha512-WxONCrssBM8TSPRqN5EmsjVrsv4A8X12J4ArBiiayv3DyyG3ZlIg6yysuuSYdZsVz3TKcTg2fd//Ujd4CHV1iA==", 1342 | "dev": true, 1343 | "license": "MIT" 1344 | }, 1345 | "node_modules/unpipe": { 1346 | "version": "1.0.0", 1347 | "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", 1348 | "integrity": "sha512-pjy2bYhSsufwWlKwPc+l3cN7+wuJlK6uz0YdJEOlQDbl6jo/YlPi4mb8agUkVC8BF7V8NuzeyPNqRksA3hztKQ==", 1349 | "license": "MIT", 1350 | "engines": { 1351 | "node": ">= 0.8" 1352 | } 1353 | }, 1354 | "node_modules/utils-merge": { 1355 | "version": "1.0.1", 1356 | "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz", 1357 | "integrity": "sha512-pMZTvIkT1d+TFGvDOqodOclx0QWkkgi6Tdoa8gC8ffGAAqz9pzPTZWAybbsHHoED/ztMtkv/VoYTYyShUn81hA==", 1358 | "license": "MIT", 1359 | "engines": { 1360 | "node": ">= 0.4.0" 1361 | } 1362 | }, 1363 | "node_modules/vary": { 1364 | "version": "1.1.2", 1365 | "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz", 1366 | "integrity": "sha512-BNGbWLfd0eUPabhkXUVm0j8uuvREyTh5ovRa/dyow/BqAbZJyC+5fU+IzQOzmAKzYqYRAISoRhdQr3eIZ/PXqg==", 1367 | "license": "MIT", 1368 | "engines": { 1369 | "node": ">= 0.8" 1370 | } 1371 | }, 1372 | "node_modules/xtend": { 1373 | "version": "4.0.2", 1374 | "resolved": "https://registry.npmjs.org/xtend/-/xtend-4.0.2.tgz", 1375 | "integrity": "sha512-LKYU1iAXJXUgAXn9URjiu+MWhyUXHsvfp7mcuYm9dSUKK0/CjtrUwFAxD82/mCWbtLsGjFIad0wIsod4zrTAEQ==", 1376 | "license": "MIT", 1377 | "engines": { 1378 | "node": ">=0.4" 1379 | } 1380 | } 1381 | } 1382 | } 1383 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "express-postgres-crud-api", 3 | "version": "1.0.0", 4 | "main": "index.js", 5 | "type": "module", 6 | "scripts": { 7 | "dev": "nodemon src/index.js", 8 | "test": "echo \"Error: no test specified\" && exit 1" 9 | }, 10 | "keywords": [], 11 | "author": "", 12 | "license": "ISC", 13 | "description": "", 14 | "dependencies": { 15 | "cors": "^2.8.5", 16 | "dotenv": "^16.4.5", 17 | "express": "^4.21.1", 18 | "joi": "^17.13.3", 19 | "pg": "^8.13.1" 20 | }, 21 | "devDependencies": { 22 | "nodemon": "^3.1.7" 23 | } 24 | } 25 | -------------------------------------------------------------------------------- /src/config/db.js: -------------------------------------------------------------------------------- 1 | import pkg from "pg"; 2 | import dotenv from "dotenv"; 3 | const { Pool } = pkg; 4 | dotenv.config(); 5 | 6 | const pool = new Pool({ 7 | user: process.env.DB_USER, 8 | host: process.env.DB_HOST, 9 | database: process.env.DB_NAME, 10 | password: process.env.DB_PASSWORD, 11 | port: process.env.DB_PORT, 12 | }); 13 | 14 | pool.on("connect", () => { 15 | console.log("Connection pool establised with Database"); 16 | }); 17 | 18 | export default pool; 19 | -------------------------------------------------------------------------------- /src/controllers/userController.js: -------------------------------------------------------------------------------- 1 | import { 2 | createUserService, 3 | deleteUserService, 4 | getAllUsersService, 5 | getUserByIdService, 6 | updateUserService, 7 | } from "../models/userModel.js"; 8 | 9 | // Standardized response function 10 | const handleResponse = (res, status, message, data = null) => { 11 | res.status(status).json({ 12 | status, 13 | message, 14 | data, 15 | }); 16 | }; 17 | 18 | export const createUser = async (req, res, next) => { 19 | const { name, email } = req.body; 20 | try { 21 | const newUser = await createUserService(name, email); 22 | handleResponse(res, 201, "User created successfully", newUser); 23 | } catch (err) { 24 | next(err); 25 | } 26 | }; 27 | 28 | export const getAllUsers = async (req, res, next) => { 29 | try { 30 | const users = await getAllUsersService(); 31 | handleResponse(res, 200, "Users fetched successfully", users); 32 | } catch (err) { 33 | next(err); 34 | } 35 | }; 36 | 37 | export const getUserById = async (req, res, next) => { 38 | try { 39 | const user = await getUserByIdService(req.params.id); 40 | if (!user) return handleResponse(res, 404, "User not found"); 41 | handleResponse(res, 200, "User fetched successfully", user); 42 | } catch (err) { 43 | next(err); 44 | } 45 | }; 46 | 47 | export const updateUser = async (req, res, next) => { 48 | const { name, email } = req.body; 49 | try { 50 | const updatedUser = await updateUserService(req.params.id, name, email); 51 | if (!updatedUser) return handleResponse(res, 404, "User not found"); 52 | handleResponse(res, 200, "User updated successfully", updatedUser); 53 | } catch (err) { 54 | next(err); 55 | } 56 | }; 57 | 58 | export const deleteUser = async (req, res, next) => { 59 | try { 60 | const deletedUser = await deleteUserService(req.params.id); 61 | if (!deletedUser) return handleResponse(res, 404, "User not found"); 62 | handleResponse(res, 200, "User deleted successfully", deleteUser); 63 | } catch (err) { 64 | next(err); 65 | } 66 | }; 67 | -------------------------------------------------------------------------------- /src/data/createUserTable.js: -------------------------------------------------------------------------------- 1 | import pool from "../config/db.js"; 2 | 3 | const createUserTable = async () => { 4 | const queryText = ` 5 | CREATE TABLE IF NOT EXISTS users ( 6 | id SERIAL PRIMARY KEY, 7 | name VARCHAR(100) NOT NULL, 8 | email VARCHAR(100) UNIQUE NOT NULL, 9 | created_at TIMESTAMP DEFAULT NOW() 10 | ) 11 | `; 12 | 13 | try { 14 | pool.query(queryText); 15 | console.log("User table created if not exists"); 16 | } catch (error) { 17 | console.log("Error creating users table : ", error); 18 | } 19 | }; 20 | 21 | export default createUserTable; 22 | -------------------------------------------------------------------------------- /src/data/data.sql: -------------------------------------------------------------------------------- 1 | CREATE TABLE IF NOT EXISTS users ( 2 | id SERIAL PRIMARY KEY, 3 | name VARCHAR(100) NOT NULL, 4 | email VARCHAR(100) UNIQUE NOT NULL, 5 | created_at TIMESTAMP DEFAULT NOW() 6 | ) -------------------------------------------------------------------------------- /src/docker-commands.txt: -------------------------------------------------------------------------------- 1 | /* Check version */ 2 | docker --version 3 | 4 | /* Pull postgres image */ 5 | docker pull postgres 6 | 7 | /* Run postgres in docker container */ 8 | docker run --name postgres-db -e POSTGRES_PASSWORD= -p 5432:5432 -d postgres 9 | 10 | --name postgres-db: The name of the container. 11 | -e POSTGRES_PASSWORD=mysecretpassword: Set the PostgreSQL password (replace mysecretpassword with your own). 12 | -p 5432:5432: Map the PostgreSQL port 5432 from the container to your local machine. 13 | -d: Run the container in detached mode. 14 | 15 | /* Get info about docker container */ 16 | docker ps 17 | 18 | /* Inspect docker container */ 19 | docker inspect 62bdae977983 20 | 21 | /* Connect to PostgreSQL Container */ 22 | docker exec -it postgres-db psql -U postgres -------------------------------------------------------------------------------- /src/index.js: -------------------------------------------------------------------------------- 1 | import express from "express"; 2 | import cors from "cors"; 3 | import dotenv from "dotenv"; 4 | import pool from "./config/db.js"; 5 | 6 | import userRoutes from "./routes/userRoutes.js"; 7 | import errorHandling from "./middlewares/errorHandler.js"; 8 | import createUserTable from "./data/createUserTable.js"; 9 | 10 | dotenv.config(); 11 | 12 | const app = express(); 13 | const port = process.env.PORT || 3001; 14 | 15 | // Middlewares 16 | app.use(express.json()); 17 | app.use(cors()); 18 | 19 | // Routes 20 | app.use("/api", userRoutes); 21 | 22 | // Error handling middleware 23 | app.use(errorHandling); 24 | 25 | //Create table before starting server 26 | createUserTable(); 27 | 28 | // Testing POSTGRES Connection 29 | app.get("/", async (req, res) => { 30 | console.log("Start"); 31 | const result = await pool.query("SELECT current_database()"); 32 | console.log("result", result.rows); 33 | res.send(`The database name is : ${result.rows[0].current_database}`); 34 | }); 35 | 36 | // Server running 37 | app.listen(port, () => { 38 | console.log(`Server is running on http:localhost:${port}`); 39 | }); 40 | -------------------------------------------------------------------------------- /src/middlewares/errorHandler.js: -------------------------------------------------------------------------------- 1 | // Cerntralized error handling 2 | const errorHandling = (err, req, res, next) => { 3 | console.log(err.stack); 4 | res.status(500).json({ 5 | status: 500, 6 | message: "Something went wrong", 7 | error: err.message, 8 | }); 9 | }; 10 | 11 | export default errorHandling; 12 | -------------------------------------------------------------------------------- /src/middlewares/inputValidator.js: -------------------------------------------------------------------------------- 1 | import Joi from "joi"; 2 | 3 | const userScheme = Joi.object({ 4 | name: Joi.string().min(3).required(), 5 | email: Joi.string().email().required(), 6 | }); 7 | 8 | const validateUser = (req, res, next) => { 9 | const { error } = userScheme.validate(req.body); 10 | if (error) 11 | return res.status(400).json({ 12 | status: 400, 13 | message: error.details[0].message, 14 | }); 15 | next(); 16 | }; 17 | 18 | export default validateUser; 19 | -------------------------------------------------------------------------------- /src/models/userModel.js: -------------------------------------------------------------------------------- 1 | import pool from "../config/db.js"; 2 | 3 | export const getAllUsersService = async () => { 4 | const result = await pool.query("SELECT * FROM users"); 5 | return result.rows; 6 | }; 7 | export const getUserByIdService = async (id) => { 8 | const result = await pool.query("SELECT * FROM users where id = $1", [id]); 9 | return result.rows[0]; 10 | }; 11 | export const createUserService = async (name, email) => { 12 | const result = await pool.query( 13 | "INSERT INTO users (name, email) VALUES ($1, $2) RETURNING *", 14 | [name, email] 15 | ); 16 | return result.rows[0]; 17 | }; 18 | export const updateUserService = async (id, name, email) => { 19 | const result = await pool.query( 20 | "UPDATE users SET name=$1, email=$2 WHERE id=$3 RETURNING *", 21 | [name, email, id] 22 | ); 23 | return result.rows[0]; 24 | }; 25 | export const deleteUserService = async (id) => { 26 | const result = await pool.query( 27 | "DELETE FROM users WHERE id = $1 RETURNING *", 28 | [id] 29 | ); 30 | return result.rows[0]; 31 | }; 32 | -------------------------------------------------------------------------------- /src/routes/userRoutes.js: -------------------------------------------------------------------------------- 1 | import express from "express"; 2 | import { 3 | createUser, 4 | deleteUser, 5 | getAllUsers, 6 | getUserById, 7 | updateUser, 8 | } from "../controllers/userController.js"; 9 | import validateUser from "../middlewares/inputValidator.js"; 10 | 11 | const router = express.Router(); 12 | 13 | router.post("/user", validateUser, createUser); 14 | router.get("/user", getAllUsers); 15 | router.get("/user/:id", getUserById); 16 | router.put("/user/:id", validateUser, updateUser); 17 | router.delete("/user/:id", deleteUser); 18 | 19 | export default router; 20 | --------------------------------------------------------------------------------