├── .gitignore ├── README.md ├── backend ├── .env ├── .vscode │ └── settings.json ├── controllers │ └── index.js ├── models │ └── index.js ├── package-lock.json ├── package.json ├── routes │ └── index.js └── server.js └── frontend ├── .vscode └── settings.json ├── package-lock.json ├── package.json ├── postcss.config.js ├── public ├── favicon.ico ├── image │ ├── attachment.jpg │ └── welcome.svg ├── index.html ├── logo192.png ├── logo512.png ├── manifest.json └── robots.txt ├── src ├── App.js ├── App.test.js ├── components │ ├── AddProjectModal.js │ ├── AddTaskModal.js │ ├── AppLayout.js │ ├── BtnPrimary.js │ ├── BtnSecondary.js │ ├── DropdownMenu.js │ ├── EditProjectModal.js │ ├── Navbar.js │ ├── ProjectDropdown.js │ ├── Sidebar.js │ ├── Task.js │ └── TaskModal.js ├── index.css ├── index.js ├── reportWebVitals.js └── setupTests.js └── tailwind.config.js /.gitignore: -------------------------------------------------------------------------------- 1 | # See https://help.github.com/articles/ignoring-files/ for more about ignoring files. 2 | 3 | # dependencies 4 | node_modules 5 | /.pnp 6 | .pnp.js 7 | 8 | # testing 9 | /coverage 10 | 11 | # production 12 | /build 13 | 14 | # misc 15 | .DS_Store 16 | .env.local 17 | .env.development.local 18 | .env.test.local 19 | .env.production.local 20 | 21 | npm-debug.log* 22 | yarn-debug.log* 23 | yarn-error.log* 24 | /.vscode 25 | 26 | # uploads 27 | files/avatars 28 | files/issues -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # Project management system using MERN stack (Mongodb, Express.js, React.js and Node.js) 2 | 3 | 4 | 5 | ## Setup instruction 6 | 7 | - Step 1: install dependencies/node_module 8 | - Go to /backend directory for backend setup and run `npm install` 9 | 10 | - Go to /frontend directory for frontend setup and run `npm install` 11 | 12 | - Step 2: Configure mongodb connection url 13 | Go to backend directory and create .env file 14 | and put into `MONGODB_PATH=your-mongodb-connection-url` 15 | 16 | - Step 3: Change server port and cors origin (Optional) 17 | by default your server running in port `http://localhost:9000` and cors origin/frontend url is`http://localhost:3000` , you can change port and cors, simply put this key into your .env 18 | `SERVER_PORT=your-port` and` CORS_ORIGIN=-your-client-origin` 19 | 20 | - Step 4: Run project 21 | in your backend `npm run serve` for start node server and `npm run start` for frontend 22 | 23 | ## Packages used 24 | - Tailwindcss 25 | - Headlessui 26 | - React router 27 | - Axios 28 | - UUID 29 | - Joi 30 | - Cors 31 | - Dotenv 32 | 33 | 34 | ## Project Screenshot 35 | #### Todo board quick preview 36 | ![React-App](https://user-images.githubusercontent.com/96901635/191009449-0083044c-c961-45cd-9da4-7184289b9573.gif) 37 | #### Todo board 38 | ![image](https://user-images.githubusercontent.com/96901635/191006996-0c185cdd-5834-47c6-8927-2e7d539866a7.png) 39 | #### Task insert 40 | ![image](https://user-images.githubusercontent.com/96901635/191007092-eb25cfc8-c056-4be2-a898-00ad29d65785.png) 41 | #### Edit task 42 | ![image](https://user-images.githubusercontent.com/96901635/191008217-6a0175e6-d5a9-4d98-8951-4a528d2bef99.png) 43 | #### Edit project 44 | ![image](https://user-images.githubusercontent.com/96901635/191008043-8c9113a1-700f-42bb-9f87-e68db159c4dc.png) 45 | 46 | 47 | 48 | 49 | ## Available Scripts 50 | 51 | In the project directory, you can run: 52 | 53 | ### `npm start` 54 | 55 | Runs the app in the development mode.\ 56 | Open [http://localhost:3000](http://localhost:3000) to view it in your browser. 57 | 58 | The page will reload when you make changes.\ 59 | You may also see any lint errors in the console. 60 | 61 | ### `npm test` 62 | 63 | Launches the test runner in the interactive watch mode.\ 64 | See the section about [running tests](https://facebook.github.io/create-react-app/docs/running-tests) for more information. 65 | 66 | ### `npm run build` 67 | 68 | Builds the app for production to the `build` folder.\ 69 | It correctly bundles React in production mode and optimizes the build for the best performance. 70 | 71 | The build is minified and the filenames include the hashes.\ 72 | Your app is ready to be deployed! 73 | 74 | See the section about [deployment](https://facebook.github.io/create-react-app/docs/deployment) for more information. 75 | 76 | ### `npm run eject` 77 | 78 | **Note: this is a one-way operation. Once you `eject`, you can't go back!** 79 | 80 | If you aren't satisfied with the build tool and configuration choices, you can `eject` at any time. This command will remove the single build dependency from your project. 81 | 82 | Instead, it will copy all the configuration files and the transitive dependencies (webpack, Babel, ESLint, etc) right into your project so you have full control over them. All of the commands except `eject` will still work, but they will point to the copied scripts so you can tweak them. At this point you're on your own. 83 | 84 | You don't have to ever use `eject`. The curated feature set is suitable for small and middle deployments, and you shouldn't feel obligated to use this feature. However we understand that this tool wouldn't be useful if you couldn't customize it when you are ready for it. 85 | -------------------------------------------------------------------------------- /backend/.env: -------------------------------------------------------------------------------- 1 | SERVER_PORT= 2 | CORS_ORIGIN= 3 | MONGODB_PATH=mongodb://localhost:27017/todo -------------------------------------------------------------------------------- /backend/.vscode/settings.json: -------------------------------------------------------------------------------- 1 | { 2 | "workbench.colorTheme": "Default Dark+", 3 | "workbench.preferredDarkColorTheme": "Default Dark+", 4 | "workbench.editor.decorations.colors": false, 5 | "workbench.preferredHighContrastLightColorTheme": "Default Light+", 6 | "breadcrumbs.enabled": false, 7 | "workbench.activityBar.iconClickBehavior": "focus" 8 | } -------------------------------------------------------------------------------- /backend/controllers/index.js: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/devravimori/project-menagment-system-in-MERN-stack/e7442711fca42b38b3c169ac30d427d7403d723f/backend/controllers/index.js -------------------------------------------------------------------------------- /backend/models/index.js: -------------------------------------------------------------------------------- 1 | // import { number, string } from "joi"; 2 | import mongoose from "mongoose"; 3 | 4 | const project = new mongoose.Schema({ 5 | title: { 6 | type: String, 7 | unique: true // `email` must be unique 8 | }, 9 | description: String, 10 | task: [ 11 | { 12 | id: Number, 13 | title: String, 14 | description: String, 15 | order: Number, 16 | stage: String, 17 | index: Number, 18 | attachment: [ 19 | { type: String, url: String } 20 | ], 21 | created_at: { type: Date, default: Date.now }, 22 | updated_at: { type: Date, default: Date.now }, 23 | } 24 | ] 25 | }, { timestamps: true }) 26 | 27 | 28 | export default mongoose.model('Project', project); -------------------------------------------------------------------------------- /backend/package-lock.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "backend", 3 | "version": "1.0.0", 4 | "lockfileVersion": 2, 5 | "requires": true, 6 | "packages": { 7 | "": { 8 | "name": "backend", 9 | "version": "1.0.0", 10 | "license": "ISC", 11 | "dependencies": { 12 | "cors": "^2.8.5", 13 | "dotenv": "^16.0.2", 14 | "express": "^4.18.1", 15 | "joi": "^17.6.0", 16 | "mongoose": "^6.6.0" 17 | }, 18 | "devDependencies": { 19 | "nodemon": "^2.0.19" 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 | }, 27 | "node_modules/@hapi/topo": { 28 | "version": "5.1.0", 29 | "resolved": "https://registry.npmjs.org/@hapi/topo/-/topo-5.1.0.tgz", 30 | "integrity": "sha512-foQZKJig7Ob0BMAYBfcJk8d77QtOe7Wo4ox7ff1lQYoNNAb6jwcY1ncdoy2e9wQZzvNy7ODZCYJkK8kzmcAnAg==", 31 | "dependencies": { 32 | "@hapi/hoek": "^9.0.0" 33 | } 34 | }, 35 | "node_modules/@sideway/address": { 36 | "version": "4.1.4", 37 | "resolved": "https://registry.npmjs.org/@sideway/address/-/address-4.1.4.tgz", 38 | "integrity": "sha512-7vwq+rOHVWjyXxVlR76Agnvhy8I9rpzjosTESvmhNeXOXdZZB15Fl+TI9x1SiHZH5Jv2wTGduSxFDIaq0m3DUw==", 39 | "dependencies": { 40 | "@hapi/hoek": "^9.0.0" 41 | } 42 | }, 43 | "node_modules/@sideway/formula": { 44 | "version": "3.0.0", 45 | "resolved": "https://registry.npmjs.org/@sideway/formula/-/formula-3.0.0.tgz", 46 | "integrity": "sha512-vHe7wZ4NOXVfkoRb8T5otiENVlT7a3IAiw7H5M2+GO+9CDgcVUUsX1zalAztCmwyOr2RUTGJdgB+ZvSVqmdHmg==" 47 | }, 48 | "node_modules/@sideway/pinpoint": { 49 | "version": "2.0.0", 50 | "resolved": "https://registry.npmjs.org/@sideway/pinpoint/-/pinpoint-2.0.0.tgz", 51 | "integrity": "sha512-RNiOoTPkptFtSVzQevY/yWtZwf/RxyVnPy/OcA9HBM3MlGDnBEYL5B41H0MTn0Uec8Hi+2qUtTfG2WWZBmMejQ==" 52 | }, 53 | "node_modules/@types/node": { 54 | "version": "18.7.17", 55 | "resolved": "https://registry.npmjs.org/@types/node/-/node-18.7.17.tgz", 56 | "integrity": "sha512-0UyfUnt02zIuqp7yC8RYtDkp/vo8bFaQ13KkSEvUAohPOAlnVNbj5Fi3fgPSuwzakS+EvvnnZ4x9y7i6ASaSPQ==" 57 | }, 58 | "node_modules/@types/webidl-conversions": { 59 | "version": "7.0.0", 60 | "resolved": "https://registry.npmjs.org/@types/webidl-conversions/-/webidl-conversions-7.0.0.tgz", 61 | "integrity": "sha512-xTE1E+YF4aWPJJeUzaZI5DRntlkY3+BCVJi0axFptnjGmAoWxkyREIh/XMrfxVLejwQxMCfDXdICo0VLxThrog==" 62 | }, 63 | "node_modules/@types/whatwg-url": { 64 | "version": "8.2.2", 65 | "resolved": "https://registry.npmjs.org/@types/whatwg-url/-/whatwg-url-8.2.2.tgz", 66 | "integrity": "sha512-FtQu10RWgn3D9U4aazdwIE2yzphmTJREDqNdODHrbrZmmMqI0vMheC/6NE/J1Yveaj8H+ela+YwWTjq5PGmuhA==", 67 | "dependencies": { 68 | "@types/node": "*", 69 | "@types/webidl-conversions": "*" 70 | } 71 | }, 72 | "node_modules/abbrev": { 73 | "version": "1.1.1", 74 | "resolved": "https://registry.npmjs.org/abbrev/-/abbrev-1.1.1.tgz", 75 | "integrity": "sha512-nne9/IiQ/hzIhY6pdDnbBtz7DjPTKrY00P/zvPSm5pOFkl6xuGrGnXn/VtTNNfNtAfZ9/1RtehkszU9qcTii0Q==", 76 | "dev": true 77 | }, 78 | "node_modules/accepts": { 79 | "version": "1.3.8", 80 | "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.8.tgz", 81 | "integrity": "sha512-PYAthTa2m2VKxuvSD3DPC/Gy+U+sOA1LAuT8mkmRuvw+NACSaeXEQ+NHcVF7rONl6qcaxV3Uuemwawk+7+SJLw==", 82 | "dependencies": { 83 | "mime-types": "~2.1.34", 84 | "negotiator": "0.6.3" 85 | }, 86 | "engines": { 87 | "node": ">= 0.6" 88 | } 89 | }, 90 | "node_modules/anymatch": { 91 | "version": "3.1.2", 92 | "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.2.tgz", 93 | "integrity": "sha512-P43ePfOAIupkguHUycrc4qJ9kz8ZiuOUijaETwX7THt0Y/GNK7v0aa8rY816xWjZ7rJdA5XdMcpVFTKMq+RvWg==", 94 | "dev": true, 95 | "dependencies": { 96 | "normalize-path": "^3.0.0", 97 | "picomatch": "^2.0.4" 98 | }, 99 | "engines": { 100 | "node": ">= 8" 101 | } 102 | }, 103 | "node_modules/array-flatten": { 104 | "version": "1.1.1", 105 | "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz", 106 | "integrity": "sha512-PCVAQswWemu6UdxsDFFX/+gVeYqKAod3D3UVm91jHwynguOwAvYPhx8nNlM++NqRcK6CxxpUafjmhIdKiHibqg==" 107 | }, 108 | "node_modules/balanced-match": { 109 | "version": "1.0.2", 110 | "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", 111 | "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", 112 | "dev": true 113 | }, 114 | "node_modules/base64-js": { 115 | "version": "1.5.1", 116 | "resolved": "https://registry.npmjs.org/base64-js/-/base64-js-1.5.1.tgz", 117 | "integrity": "sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==", 118 | "funding": [ 119 | { 120 | "type": "github", 121 | "url": "https://github.com/sponsors/feross" 122 | }, 123 | { 124 | "type": "patreon", 125 | "url": "https://www.patreon.com/feross" 126 | }, 127 | { 128 | "type": "consulting", 129 | "url": "https://feross.org/support" 130 | } 131 | ] 132 | }, 133 | "node_modules/binary-extensions": { 134 | "version": "2.2.0", 135 | "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.2.0.tgz", 136 | "integrity": "sha512-jDctJ/IVQbZoJykoeHbhXpOlNBqGNcwXJKJog42E5HDPUwQTSdjCHdihjj0DlnheQ7blbT6dHOafNAiS8ooQKA==", 137 | "dev": true, 138 | "engines": { 139 | "node": ">=8" 140 | } 141 | }, 142 | "node_modules/body-parser": { 143 | "version": "1.20.0", 144 | "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.20.0.tgz", 145 | "integrity": "sha512-DfJ+q6EPcGKZD1QWUjSpqp+Q7bDQTsQIF4zfUAtZ6qk+H/3/QRhg9CEp39ss+/T2vw0+HaidC0ecJj/DRLIaKg==", 146 | "dependencies": { 147 | "bytes": "3.1.2", 148 | "content-type": "~1.0.4", 149 | "debug": "2.6.9", 150 | "depd": "2.0.0", 151 | "destroy": "1.2.0", 152 | "http-errors": "2.0.0", 153 | "iconv-lite": "0.4.24", 154 | "on-finished": "2.4.1", 155 | "qs": "6.10.3", 156 | "raw-body": "2.5.1", 157 | "type-is": "~1.6.18", 158 | "unpipe": "1.0.0" 159 | }, 160 | "engines": { 161 | "node": ">= 0.8", 162 | "npm": "1.2.8000 || >= 1.4.16" 163 | } 164 | }, 165 | "node_modules/brace-expansion": { 166 | "version": "1.1.11", 167 | "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", 168 | "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", 169 | "dev": true, 170 | "dependencies": { 171 | "balanced-match": "^1.0.0", 172 | "concat-map": "0.0.1" 173 | } 174 | }, 175 | "node_modules/braces": { 176 | "version": "3.0.2", 177 | "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz", 178 | "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", 179 | "dev": true, 180 | "dependencies": { 181 | "fill-range": "^7.0.1" 182 | }, 183 | "engines": { 184 | "node": ">=8" 185 | } 186 | }, 187 | "node_modules/bson": { 188 | "version": "4.7.0", 189 | "resolved": "https://registry.npmjs.org/bson/-/bson-4.7.0.tgz", 190 | "integrity": "sha512-VrlEE4vuiO1WTpfof4VmaVolCVYkYTgB9iWgYNOrVlnifpME/06fhFRmONgBhClD5pFC1t9ZWqFUQEQAzY43bA==", 191 | "dependencies": { 192 | "buffer": "^5.6.0" 193 | }, 194 | "engines": { 195 | "node": ">=6.9.0" 196 | } 197 | }, 198 | "node_modules/buffer": { 199 | "version": "5.7.1", 200 | "resolved": "https://registry.npmjs.org/buffer/-/buffer-5.7.1.tgz", 201 | "integrity": "sha512-EHcyIPBQ4BSGlvjB16k5KgAJ27CIsHY/2JBmCRReo48y9rQ3MaUzWX3KVlBa4U7MyX02HdVj0K7C3WaB3ju7FQ==", 202 | "funding": [ 203 | { 204 | "type": "github", 205 | "url": "https://github.com/sponsors/feross" 206 | }, 207 | { 208 | "type": "patreon", 209 | "url": "https://www.patreon.com/feross" 210 | }, 211 | { 212 | "type": "consulting", 213 | "url": "https://feross.org/support" 214 | } 215 | ], 216 | "dependencies": { 217 | "base64-js": "^1.3.1", 218 | "ieee754": "^1.1.13" 219 | } 220 | }, 221 | "node_modules/bytes": { 222 | "version": "3.1.2", 223 | "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.2.tgz", 224 | "integrity": "sha512-/Nf7TyzTx6S3yRJObOAV7956r8cr2+Oj8AC5dt8wSP3BQAoeX58NoHyCU8P8zGkNXStjTSi6fzO6F0pBdcYbEg==", 225 | "engines": { 226 | "node": ">= 0.8" 227 | } 228 | }, 229 | "node_modules/call-bind": { 230 | "version": "1.0.2", 231 | "resolved": "https://registry.npmjs.org/call-bind/-/call-bind-1.0.2.tgz", 232 | "integrity": "sha512-7O+FbCihrB5WGbFYesctwmTKae6rOiIzmz1icreWJ+0aA7LJfuqhEso2T9ncpcFtzMQtzXf2QGGueWJGTYsqrA==", 233 | "dependencies": { 234 | "function-bind": "^1.1.1", 235 | "get-intrinsic": "^1.0.2" 236 | }, 237 | "funding": { 238 | "url": "https://github.com/sponsors/ljharb" 239 | } 240 | }, 241 | "node_modules/chokidar": { 242 | "version": "3.5.3", 243 | "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.5.3.tgz", 244 | "integrity": "sha512-Dr3sfKRP6oTcjf2JmUmFJfeVMvXBdegxB0iVQ5eb2V10uFJUCAS8OByZdVAyVb8xXNz3GjjTgj9kLWsZTqE6kw==", 245 | "dev": true, 246 | "funding": [ 247 | { 248 | "type": "individual", 249 | "url": "https://paulmillr.com/funding/" 250 | } 251 | ], 252 | "dependencies": { 253 | "anymatch": "~3.1.2", 254 | "braces": "~3.0.2", 255 | "glob-parent": "~5.1.2", 256 | "is-binary-path": "~2.1.0", 257 | "is-glob": "~4.0.1", 258 | "normalize-path": "~3.0.0", 259 | "readdirp": "~3.6.0" 260 | }, 261 | "engines": { 262 | "node": ">= 8.10.0" 263 | }, 264 | "optionalDependencies": { 265 | "fsevents": "~2.3.2" 266 | } 267 | }, 268 | "node_modules/concat-map": { 269 | "version": "0.0.1", 270 | "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", 271 | "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==", 272 | "dev": true 273 | }, 274 | "node_modules/content-disposition": { 275 | "version": "0.5.4", 276 | "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.4.tgz", 277 | "integrity": "sha512-FveZTNuGw04cxlAiWbzi6zTAL/lhehaWbTtgluJh4/E95DqMwTmha3KZN1aAWA8cFIhHzMZUvLevkw5Rqk+tSQ==", 278 | "dependencies": { 279 | "safe-buffer": "5.2.1" 280 | }, 281 | "engines": { 282 | "node": ">= 0.6" 283 | } 284 | }, 285 | "node_modules/content-type": { 286 | "version": "1.0.4", 287 | "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.4.tgz", 288 | "integrity": "sha512-hIP3EEPs8tB9AT1L+NUqtwOAps4mk2Zob89MWXMHjHWg9milF/j4osnnQLXBCBFBk/tvIG/tUc9mOUJiPBhPXA==", 289 | "engines": { 290 | "node": ">= 0.6" 291 | } 292 | }, 293 | "node_modules/cookie": { 294 | "version": "0.5.0", 295 | "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.5.0.tgz", 296 | "integrity": "sha512-YZ3GUyn/o8gfKJlnlX7g7xq4gyO6OSuhGPKaaGssGB2qgDUS0gPgtTvoyZLTt9Ab6dC4hfc9dV5arkvc/OCmrw==", 297 | "engines": { 298 | "node": ">= 0.6" 299 | } 300 | }, 301 | "node_modules/cookie-signature": { 302 | "version": "1.0.6", 303 | "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz", 304 | "integrity": "sha512-QADzlaHc8icV8I7vbaJXJwod9HWYp8uCqf1xa4OfNu1T7JVxQIrUgOWtHdNDtPiywmFbiS12VjotIXLrKM3orQ==" 305 | }, 306 | "node_modules/cors": { 307 | "version": "2.8.5", 308 | "resolved": "https://registry.npmjs.org/cors/-/cors-2.8.5.tgz", 309 | "integrity": "sha512-KIHbLJqu73RGr/hnbrO9uBeixNGuvSQjul/jdFvS/KFSIH1hWVd1ng7zOHx+YrEfInLG7q4n6GHQ9cDtxv/P6g==", 310 | "dependencies": { 311 | "object-assign": "^4", 312 | "vary": "^1" 313 | }, 314 | "engines": { 315 | "node": ">= 0.10" 316 | } 317 | }, 318 | "node_modules/debug": { 319 | "version": "2.6.9", 320 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", 321 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", 322 | "dependencies": { 323 | "ms": "2.0.0" 324 | } 325 | }, 326 | "node_modules/denque": { 327 | "version": "2.1.0", 328 | "resolved": "https://registry.npmjs.org/denque/-/denque-2.1.0.tgz", 329 | "integrity": "sha512-HVQE3AAb/pxF8fQAoiqpvg9i3evqug3hoiwakOyZAwJm+6vZehbkYXZ0l4JxS+I3QxM97v5aaRNhj8v5oBhekw==", 330 | "engines": { 331 | "node": ">=0.10" 332 | } 333 | }, 334 | "node_modules/depd": { 335 | "version": "2.0.0", 336 | "resolved": "https://registry.npmjs.org/depd/-/depd-2.0.0.tgz", 337 | "integrity": "sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw==", 338 | "engines": { 339 | "node": ">= 0.8" 340 | } 341 | }, 342 | "node_modules/destroy": { 343 | "version": "1.2.0", 344 | "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.2.0.tgz", 345 | "integrity": "sha512-2sJGJTaXIIaR1w4iJSNoN0hnMY7Gpc/n8D4qSCJw8QqFWXf7cuAgnEHxBpweaVcPevC2l3KpjYCx3NypQQgaJg==", 346 | "engines": { 347 | "node": ">= 0.8", 348 | "npm": "1.2.8000 || >= 1.4.16" 349 | } 350 | }, 351 | "node_modules/dotenv": { 352 | "version": "16.0.2", 353 | "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-16.0.2.tgz", 354 | "integrity": "sha512-JvpYKUmzQhYoIFgK2MOnF3bciIZoItIIoryihy0rIA+H4Jy0FmgyKYAHCTN98P5ybGSJcIFbh6QKeJdtZd1qhA==", 355 | "engines": { 356 | "node": ">=12" 357 | } 358 | }, 359 | "node_modules/ee-first": { 360 | "version": "1.1.1", 361 | "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", 362 | "integrity": "sha512-WMwm9LhRUo+WUaRN+vRuETqG89IgZphVSNkdFgeb6sS/E4OrDIN7t48CAewSHXc6C8lefD8KKfr5vY61brQlow==" 363 | }, 364 | "node_modules/encodeurl": { 365 | "version": "1.0.2", 366 | "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz", 367 | "integrity": "sha512-TPJXq8JqFaVYm2CWmPvnP2Iyo4ZSM7/QKcSmuMLDObfpH5fi7RUGmd/rTDf+rut/saiDiQEeVTNgAmJEdAOx0w==", 368 | "engines": { 369 | "node": ">= 0.8" 370 | } 371 | }, 372 | "node_modules/escape-html": { 373 | "version": "1.0.3", 374 | "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz", 375 | "integrity": "sha512-NiSupZ4OeuGwr68lGIeym/ksIZMJodUGOSCZ/FSnTxcrekbvqrgdUxlJOMpijaKZVjAJrWrGs/6Jy8OMuyj9ow==" 376 | }, 377 | "node_modules/etag": { 378 | "version": "1.8.1", 379 | "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz", 380 | "integrity": "sha512-aIL5Fx7mawVa300al2BnEE4iNvo1qETxLrPI/o05L7z6go7fCw1J6EQmbK4FmJ2AS7kgVF/KEZWufBfdClMcPg==", 381 | "engines": { 382 | "node": ">= 0.6" 383 | } 384 | }, 385 | "node_modules/express": { 386 | "version": "4.18.1", 387 | "resolved": "https://registry.npmjs.org/express/-/express-4.18.1.tgz", 388 | "integrity": "sha512-zZBcOX9TfehHQhtupq57OF8lFZ3UZi08Y97dwFCkD8p9d/d2Y3M+ykKcwaMDEL+4qyUolgBDX6AblpR3fL212Q==", 389 | "dependencies": { 390 | "accepts": "~1.3.8", 391 | "array-flatten": "1.1.1", 392 | "body-parser": "1.20.0", 393 | "content-disposition": "0.5.4", 394 | "content-type": "~1.0.4", 395 | "cookie": "0.5.0", 396 | "cookie-signature": "1.0.6", 397 | "debug": "2.6.9", 398 | "depd": "2.0.0", 399 | "encodeurl": "~1.0.2", 400 | "escape-html": "~1.0.3", 401 | "etag": "~1.8.1", 402 | "finalhandler": "1.2.0", 403 | "fresh": "0.5.2", 404 | "http-errors": "2.0.0", 405 | "merge-descriptors": "1.0.1", 406 | "methods": "~1.1.2", 407 | "on-finished": "2.4.1", 408 | "parseurl": "~1.3.3", 409 | "path-to-regexp": "0.1.7", 410 | "proxy-addr": "~2.0.7", 411 | "qs": "6.10.3", 412 | "range-parser": "~1.2.1", 413 | "safe-buffer": "5.2.1", 414 | "send": "0.18.0", 415 | "serve-static": "1.15.0", 416 | "setprototypeof": "1.2.0", 417 | "statuses": "2.0.1", 418 | "type-is": "~1.6.18", 419 | "utils-merge": "1.0.1", 420 | "vary": "~1.1.2" 421 | }, 422 | "engines": { 423 | "node": ">= 0.10.0" 424 | } 425 | }, 426 | "node_modules/fill-range": { 427 | "version": "7.0.1", 428 | "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz", 429 | "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", 430 | "dev": true, 431 | "dependencies": { 432 | "to-regex-range": "^5.0.1" 433 | }, 434 | "engines": { 435 | "node": ">=8" 436 | } 437 | }, 438 | "node_modules/finalhandler": { 439 | "version": "1.2.0", 440 | "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.2.0.tgz", 441 | "integrity": "sha512-5uXcUVftlQMFnWC9qu/svkWv3GTd2PfUhK/3PLkYNAe7FbqJMt3515HaxE6eRL74GdsriiwujiawdaB1BpEISg==", 442 | "dependencies": { 443 | "debug": "2.6.9", 444 | "encodeurl": "~1.0.2", 445 | "escape-html": "~1.0.3", 446 | "on-finished": "2.4.1", 447 | "parseurl": "~1.3.3", 448 | "statuses": "2.0.1", 449 | "unpipe": "~1.0.0" 450 | }, 451 | "engines": { 452 | "node": ">= 0.8" 453 | } 454 | }, 455 | "node_modules/forwarded": { 456 | "version": "0.2.0", 457 | "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.2.0.tgz", 458 | "integrity": "sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow==", 459 | "engines": { 460 | "node": ">= 0.6" 461 | } 462 | }, 463 | "node_modules/fresh": { 464 | "version": "0.5.2", 465 | "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz", 466 | "integrity": "sha512-zJ2mQYM18rEFOudeV4GShTGIQ7RbzA7ozbU9I/XBpm7kqgMywgmylMwXHxZJmkVoYkna9d2pVXVXPdYTP9ej8Q==", 467 | "engines": { 468 | "node": ">= 0.6" 469 | } 470 | }, 471 | "node_modules/fsevents": { 472 | "version": "2.3.2", 473 | "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.2.tgz", 474 | "integrity": "sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==", 475 | "dev": true, 476 | "hasInstallScript": true, 477 | "optional": true, 478 | "os": [ 479 | "darwin" 480 | ], 481 | "engines": { 482 | "node": "^8.16.0 || ^10.6.0 || >=11.0.0" 483 | } 484 | }, 485 | "node_modules/function-bind": { 486 | "version": "1.1.1", 487 | "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.1.tgz", 488 | "integrity": "sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A==" 489 | }, 490 | "node_modules/get-intrinsic": { 491 | "version": "1.1.3", 492 | "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.1.3.tgz", 493 | "integrity": "sha512-QJVz1Tj7MS099PevUG5jvnt9tSkXN8K14dxQlikJuPt4uD9hHAHjLyLBiLR5zELelBdD9QNRAXZzsJx0WaDL9A==", 494 | "dependencies": { 495 | "function-bind": "^1.1.1", 496 | "has": "^1.0.3", 497 | "has-symbols": "^1.0.3" 498 | }, 499 | "funding": { 500 | "url": "https://github.com/sponsors/ljharb" 501 | } 502 | }, 503 | "node_modules/glob-parent": { 504 | "version": "5.1.2", 505 | "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", 506 | "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", 507 | "dev": true, 508 | "dependencies": { 509 | "is-glob": "^4.0.1" 510 | }, 511 | "engines": { 512 | "node": ">= 6" 513 | } 514 | }, 515 | "node_modules/has": { 516 | "version": "1.0.3", 517 | "resolved": "https://registry.npmjs.org/has/-/has-1.0.3.tgz", 518 | "integrity": "sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw==", 519 | "dependencies": { 520 | "function-bind": "^1.1.1" 521 | }, 522 | "engines": { 523 | "node": ">= 0.4.0" 524 | } 525 | }, 526 | "node_modules/has-flag": { 527 | "version": "3.0.0", 528 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", 529 | "integrity": "sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==", 530 | "dev": true, 531 | "engines": { 532 | "node": ">=4" 533 | } 534 | }, 535 | "node_modules/has-symbols": { 536 | "version": "1.0.3", 537 | "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.0.3.tgz", 538 | "integrity": "sha512-l3LCuF6MgDNwTDKkdYGEihYjt5pRPbEg46rtlmnSPlUbgmB8LOIrKJbYYFBSbnPaJexMKtiPO8hmeRjRz2Td+A==", 539 | "engines": { 540 | "node": ">= 0.4" 541 | }, 542 | "funding": { 543 | "url": "https://github.com/sponsors/ljharb" 544 | } 545 | }, 546 | "node_modules/http-errors": { 547 | "version": "2.0.0", 548 | "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-2.0.0.tgz", 549 | "integrity": "sha512-FtwrG/euBzaEjYeRqOgly7G0qviiXoJWnvEH2Z1plBdXgbyjv34pHTSb9zoeHMyDy33+DWy5Wt9Wo+TURtOYSQ==", 550 | "dependencies": { 551 | "depd": "2.0.0", 552 | "inherits": "2.0.4", 553 | "setprototypeof": "1.2.0", 554 | "statuses": "2.0.1", 555 | "toidentifier": "1.0.1" 556 | }, 557 | "engines": { 558 | "node": ">= 0.8" 559 | } 560 | }, 561 | "node_modules/iconv-lite": { 562 | "version": "0.4.24", 563 | "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", 564 | "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", 565 | "dependencies": { 566 | "safer-buffer": ">= 2.1.2 < 3" 567 | }, 568 | "engines": { 569 | "node": ">=0.10.0" 570 | } 571 | }, 572 | "node_modules/ieee754": { 573 | "version": "1.2.1", 574 | "resolved": "https://registry.npmjs.org/ieee754/-/ieee754-1.2.1.tgz", 575 | "integrity": "sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==", 576 | "funding": [ 577 | { 578 | "type": "github", 579 | "url": "https://github.com/sponsors/feross" 580 | }, 581 | { 582 | "type": "patreon", 583 | "url": "https://www.patreon.com/feross" 584 | }, 585 | { 586 | "type": "consulting", 587 | "url": "https://feross.org/support" 588 | } 589 | ] 590 | }, 591 | "node_modules/ignore-by-default": { 592 | "version": "1.0.1", 593 | "resolved": "https://registry.npmjs.org/ignore-by-default/-/ignore-by-default-1.0.1.tgz", 594 | "integrity": "sha512-Ius2VYcGNk7T90CppJqcIkS5ooHUZyIQK+ClZfMfMNFEF9VSE73Fq+906u/CWu92x4gzZMWOwfFYckPObzdEbA==", 595 | "dev": true 596 | }, 597 | "node_modules/inherits": { 598 | "version": "2.0.4", 599 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", 600 | "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==" 601 | }, 602 | "node_modules/ip": { 603 | "version": "2.0.0", 604 | "resolved": "https://registry.npmjs.org/ip/-/ip-2.0.0.tgz", 605 | "integrity": "sha512-WKa+XuLG1A1R0UWhl2+1XQSi+fZWMsYKffMZTTYsiZaUD8k2yDAj5atimTUD2TZkyCkNEeYE5NhFZmupOGtjYQ==" 606 | }, 607 | "node_modules/ipaddr.js": { 608 | "version": "1.9.1", 609 | "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz", 610 | "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==", 611 | "engines": { 612 | "node": ">= 0.10" 613 | } 614 | }, 615 | "node_modules/is-binary-path": { 616 | "version": "2.1.0", 617 | "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz", 618 | "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==", 619 | "dev": true, 620 | "dependencies": { 621 | "binary-extensions": "^2.0.0" 622 | }, 623 | "engines": { 624 | "node": ">=8" 625 | } 626 | }, 627 | "node_modules/is-extglob": { 628 | "version": "2.1.1", 629 | "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", 630 | "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==", 631 | "dev": true, 632 | "engines": { 633 | "node": ">=0.10.0" 634 | } 635 | }, 636 | "node_modules/is-glob": { 637 | "version": "4.0.3", 638 | "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz", 639 | "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==", 640 | "dev": true, 641 | "dependencies": { 642 | "is-extglob": "^2.1.1" 643 | }, 644 | "engines": { 645 | "node": ">=0.10.0" 646 | } 647 | }, 648 | "node_modules/is-number": { 649 | "version": "7.0.0", 650 | "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", 651 | "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", 652 | "dev": true, 653 | "engines": { 654 | "node": ">=0.12.0" 655 | } 656 | }, 657 | "node_modules/joi": { 658 | "version": "17.6.0", 659 | "resolved": "https://registry.npmjs.org/joi/-/joi-17.6.0.tgz", 660 | "integrity": "sha512-OX5dG6DTbcr/kbMFj0KGYxuew69HPcAE3K/sZpEV2nP6e/j/C0HV+HNiBPCASxdx5T7DMoa0s8UeHWMnb6n2zw==", 661 | "dependencies": { 662 | "@hapi/hoek": "^9.0.0", 663 | "@hapi/topo": "^5.0.0", 664 | "@sideway/address": "^4.1.3", 665 | "@sideway/formula": "^3.0.0", 666 | "@sideway/pinpoint": "^2.0.0" 667 | } 668 | }, 669 | "node_modules/kareem": { 670 | "version": "2.4.1", 671 | "resolved": "https://registry.npmjs.org/kareem/-/kareem-2.4.1.tgz", 672 | "integrity": "sha512-aJ9opVoXroQUPfovYP5kaj2lM7Jn02Gw13bL0lg9v0V7SaUc0qavPs0Eue7d2DcC3NjqI6QAUElXNsuZSeM+EA==" 673 | }, 674 | "node_modules/media-typer": { 675 | "version": "0.3.0", 676 | "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", 677 | "integrity": "sha512-dq+qelQ9akHpcOl/gUVRTxVIOkAJ1wR3QAvb4RsVjS8oVoFjDGTc679wJYmUmknUF5HwMLOgb5O+a3KxfWapPQ==", 678 | "engines": { 679 | "node": ">= 0.6" 680 | } 681 | }, 682 | "node_modules/memory-pager": { 683 | "version": "1.5.0", 684 | "resolved": "https://registry.npmjs.org/memory-pager/-/memory-pager-1.5.0.tgz", 685 | "integrity": "sha512-ZS4Bp4r/Zoeq6+NLJpP+0Zzm0pR8whtGPf1XExKLJBAczGMnSi3It14OiNCStjQjM6NU1okjQGSxgEZN8eBYKg==", 686 | "optional": true 687 | }, 688 | "node_modules/merge-descriptors": { 689 | "version": "1.0.1", 690 | "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz", 691 | "integrity": "sha512-cCi6g3/Zr1iqQi6ySbseM1Xvooa98N0w31jzUYrXPX2xqObmFGHJ0tQ5u74H3mVh7wLouTseZyYIq39g8cNp1w==" 692 | }, 693 | "node_modules/methods": { 694 | "version": "1.1.2", 695 | "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz", 696 | "integrity": "sha512-iclAHeNqNm68zFtnZ0e+1L2yUIdvzNoauKU4WBA3VvH/vPFieF7qfRlwUZU+DA9P9bPXIS90ulxoUoCH23sV2w==", 697 | "engines": { 698 | "node": ">= 0.6" 699 | } 700 | }, 701 | "node_modules/mime": { 702 | "version": "1.6.0", 703 | "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz", 704 | "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==", 705 | "bin": { 706 | "mime": "cli.js" 707 | }, 708 | "engines": { 709 | "node": ">=4" 710 | } 711 | }, 712 | "node_modules/mime-db": { 713 | "version": "1.52.0", 714 | "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz", 715 | "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==", 716 | "engines": { 717 | "node": ">= 0.6" 718 | } 719 | }, 720 | "node_modules/mime-types": { 721 | "version": "2.1.35", 722 | "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz", 723 | "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==", 724 | "dependencies": { 725 | "mime-db": "1.52.0" 726 | }, 727 | "engines": { 728 | "node": ">= 0.6" 729 | } 730 | }, 731 | "node_modules/minimatch": { 732 | "version": "3.1.2", 733 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", 734 | "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", 735 | "dev": true, 736 | "dependencies": { 737 | "brace-expansion": "^1.1.7" 738 | }, 739 | "engines": { 740 | "node": "*" 741 | } 742 | }, 743 | "node_modules/mongodb": { 744 | "version": "4.9.1", 745 | "resolved": "https://registry.npmjs.org/mongodb/-/mongodb-4.9.1.tgz", 746 | "integrity": "sha512-ZhgI/qBf84fD7sI4waZBoLBNJYPQN5IOC++SBCiPiyhzpNKOxN/fi0tBHvH2dEC42HXtNEbFB0zmNz4+oVtorQ==", 747 | "dependencies": { 748 | "bson": "^4.7.0", 749 | "denque": "^2.1.0", 750 | "mongodb-connection-string-url": "^2.5.3", 751 | "socks": "^2.7.0" 752 | }, 753 | "engines": { 754 | "node": ">=12.9.0" 755 | }, 756 | "optionalDependencies": { 757 | "saslprep": "^1.0.3" 758 | } 759 | }, 760 | "node_modules/mongodb-connection-string-url": { 761 | "version": "2.5.3", 762 | "resolved": "https://registry.npmjs.org/mongodb-connection-string-url/-/mongodb-connection-string-url-2.5.3.tgz", 763 | "integrity": "sha512-f+/WsED+xF4B74l3k9V/XkTVj5/fxFH2o5ToKXd8Iyi5UhM+sO9u0Ape17Mvl/GkZaFtM0HQnzAG5OTmhKw+tQ==", 764 | "dependencies": { 765 | "@types/whatwg-url": "^8.2.1", 766 | "whatwg-url": "^11.0.0" 767 | } 768 | }, 769 | "node_modules/mongoose": { 770 | "version": "6.6.0", 771 | "resolved": "https://registry.npmjs.org/mongoose/-/mongoose-6.6.0.tgz", 772 | "integrity": "sha512-5sS0D7qbmfT4G/nODkJhx5l2qvhqLMplhMlp08Wea8eoi6O/B6b+o1ukUEKjjm1MV8dAS3w8kx47R/klYKLPpQ==", 773 | "dependencies": { 774 | "bson": "^4.6.5", 775 | "kareem": "2.4.1", 776 | "mongodb": "4.9.1", 777 | "mpath": "0.9.0", 778 | "mquery": "4.0.3", 779 | "ms": "2.1.3", 780 | "sift": "16.0.0" 781 | }, 782 | "engines": { 783 | "node": ">=12.0.0" 784 | }, 785 | "funding": { 786 | "type": "opencollective", 787 | "url": "https://opencollective.com/mongoose" 788 | } 789 | }, 790 | "node_modules/mongoose/node_modules/ms": { 791 | "version": "2.1.3", 792 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", 793 | "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==" 794 | }, 795 | "node_modules/mpath": { 796 | "version": "0.9.0", 797 | "resolved": "https://registry.npmjs.org/mpath/-/mpath-0.9.0.tgz", 798 | "integrity": "sha512-ikJRQTk8hw5DEoFVxHG1Gn9T/xcjtdnOKIU1JTmGjZZlg9LST2mBLmcX3/ICIbgJydT2GOc15RnNy5mHmzfSew==", 799 | "engines": { 800 | "node": ">=4.0.0" 801 | } 802 | }, 803 | "node_modules/mquery": { 804 | "version": "4.0.3", 805 | "resolved": "https://registry.npmjs.org/mquery/-/mquery-4.0.3.tgz", 806 | "integrity": "sha512-J5heI+P08I6VJ2Ky3+33IpCdAvlYGTSUjwTPxkAr8i8EoduPMBX2OY/wa3IKZIQl7MU4SbFk8ndgSKyB/cl1zA==", 807 | "dependencies": { 808 | "debug": "4.x" 809 | }, 810 | "engines": { 811 | "node": ">=12.0.0" 812 | } 813 | }, 814 | "node_modules/mquery/node_modules/debug": { 815 | "version": "4.3.4", 816 | "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.4.tgz", 817 | "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==", 818 | "dependencies": { 819 | "ms": "2.1.2" 820 | }, 821 | "engines": { 822 | "node": ">=6.0" 823 | }, 824 | "peerDependenciesMeta": { 825 | "supports-color": { 826 | "optional": true 827 | } 828 | } 829 | }, 830 | "node_modules/mquery/node_modules/ms": { 831 | "version": "2.1.2", 832 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", 833 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==" 834 | }, 835 | "node_modules/ms": { 836 | "version": "2.0.0", 837 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", 838 | "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==" 839 | }, 840 | "node_modules/negotiator": { 841 | "version": "0.6.3", 842 | "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.3.tgz", 843 | "integrity": "sha512-+EUsqGPLsM+j/zdChZjsnX51g4XrHFOIXwfnCVPGlQk/k5giakcKsuxCObBRu6DSm9opw/O6slWbJdghQM4bBg==", 844 | "engines": { 845 | "node": ">= 0.6" 846 | } 847 | }, 848 | "node_modules/nodemon": { 849 | "version": "2.0.19", 850 | "resolved": "https://registry.npmjs.org/nodemon/-/nodemon-2.0.19.tgz", 851 | "integrity": "sha512-4pv1f2bMDj0Eeg/MhGqxrtveeQ5/G/UVe9iO6uTZzjnRluSA4PVWf8CW99LUPwGB3eNIA7zUFoP77YuI7hOc0A==", 852 | "dev": true, 853 | "hasInstallScript": true, 854 | "dependencies": { 855 | "chokidar": "^3.5.2", 856 | "debug": "^3.2.7", 857 | "ignore-by-default": "^1.0.1", 858 | "minimatch": "^3.0.4", 859 | "pstree.remy": "^1.1.8", 860 | "semver": "^5.7.1", 861 | "simple-update-notifier": "^1.0.7", 862 | "supports-color": "^5.5.0", 863 | "touch": "^3.1.0", 864 | "undefsafe": "^2.0.5" 865 | }, 866 | "bin": { 867 | "nodemon": "bin/nodemon.js" 868 | }, 869 | "engines": { 870 | "node": ">=8.10.0" 871 | }, 872 | "funding": { 873 | "type": "opencollective", 874 | "url": "https://opencollective.com/nodemon" 875 | } 876 | }, 877 | "node_modules/nodemon/node_modules/debug": { 878 | "version": "3.2.7", 879 | "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", 880 | "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", 881 | "dev": true, 882 | "dependencies": { 883 | "ms": "^2.1.1" 884 | } 885 | }, 886 | "node_modules/nodemon/node_modules/ms": { 887 | "version": "2.1.3", 888 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", 889 | "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", 890 | "dev": true 891 | }, 892 | "node_modules/nopt": { 893 | "version": "1.0.10", 894 | "resolved": "https://registry.npmjs.org/nopt/-/nopt-1.0.10.tgz", 895 | "integrity": "sha512-NWmpvLSqUrgrAC9HCuxEvb+PSloHpqVu+FqcO4eeF2h5qYRhA7ev6KvelyQAKtegUbC6RypJnlEOhd8vloNKYg==", 896 | "dev": true, 897 | "dependencies": { 898 | "abbrev": "1" 899 | }, 900 | "bin": { 901 | "nopt": "bin/nopt.js" 902 | }, 903 | "engines": { 904 | "node": "*" 905 | } 906 | }, 907 | "node_modules/normalize-path": { 908 | "version": "3.0.0", 909 | "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", 910 | "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==", 911 | "dev": true, 912 | "engines": { 913 | "node": ">=0.10.0" 914 | } 915 | }, 916 | "node_modules/object-assign": { 917 | "version": "4.1.1", 918 | "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", 919 | "integrity": "sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==", 920 | "engines": { 921 | "node": ">=0.10.0" 922 | } 923 | }, 924 | "node_modules/object-inspect": { 925 | "version": "1.12.2", 926 | "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.12.2.tgz", 927 | "integrity": "sha512-z+cPxW0QGUp0mcqcsgQyLVRDoXFQbXOwBaqyF7VIgI4TWNQsDHrBpUQslRmIfAoYWdYzs6UlKJtB2XJpTaNSpQ==", 928 | "funding": { 929 | "url": "https://github.com/sponsors/ljharb" 930 | } 931 | }, 932 | "node_modules/on-finished": { 933 | "version": "2.4.1", 934 | "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.4.1.tgz", 935 | "integrity": "sha512-oVlzkg3ENAhCk2zdv7IJwd/QUD4z2RxRwpkcGY8psCVcCYZNq4wYnVWALHM+brtuJjePWiYF/ClmuDr8Ch5+kg==", 936 | "dependencies": { 937 | "ee-first": "1.1.1" 938 | }, 939 | "engines": { 940 | "node": ">= 0.8" 941 | } 942 | }, 943 | "node_modules/parseurl": { 944 | "version": "1.3.3", 945 | "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz", 946 | "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==", 947 | "engines": { 948 | "node": ">= 0.8" 949 | } 950 | }, 951 | "node_modules/path-to-regexp": { 952 | "version": "0.1.7", 953 | "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz", 954 | "integrity": "sha512-5DFkuoqlv1uYQKxy8omFBeJPQcdoE07Kv2sferDCrAq1ohOU+MSDswDIbnx3YAM60qIOnYa53wBhXW0EbMonrQ==" 955 | }, 956 | "node_modules/picomatch": { 957 | "version": "2.3.1", 958 | "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", 959 | "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==", 960 | "dev": true, 961 | "engines": { 962 | "node": ">=8.6" 963 | }, 964 | "funding": { 965 | "url": "https://github.com/sponsors/jonschlinkert" 966 | } 967 | }, 968 | "node_modules/proxy-addr": { 969 | "version": "2.0.7", 970 | "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.7.tgz", 971 | "integrity": "sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg==", 972 | "dependencies": { 973 | "forwarded": "0.2.0", 974 | "ipaddr.js": "1.9.1" 975 | }, 976 | "engines": { 977 | "node": ">= 0.10" 978 | } 979 | }, 980 | "node_modules/pstree.remy": { 981 | "version": "1.1.8", 982 | "resolved": "https://registry.npmjs.org/pstree.remy/-/pstree.remy-1.1.8.tgz", 983 | "integrity": "sha512-77DZwxQmxKnu3aR542U+X8FypNzbfJ+C5XQDk3uWjWxn6151aIMGthWYRXTqT1E5oJvg+ljaa2OJi+VfvCOQ8w==", 984 | "dev": true 985 | }, 986 | "node_modules/punycode": { 987 | "version": "2.1.1", 988 | "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.1.1.tgz", 989 | "integrity": "sha512-XRsRjdf+j5ml+y/6GKHPZbrF/8p2Yga0JPtdqTIY2Xe5ohJPD9saDJJLPvp9+NSBprVvevdXZybnj2cv8OEd0A==", 990 | "engines": { 991 | "node": ">=6" 992 | } 993 | }, 994 | "node_modules/qs": { 995 | "version": "6.10.3", 996 | "resolved": "https://registry.npmjs.org/qs/-/qs-6.10.3.tgz", 997 | "integrity": "sha512-wr7M2E0OFRfIfJZjKGieI8lBKb7fRCH4Fv5KNPEs7gJ8jadvotdsS08PzOKR7opXhZ/Xkjtt3WF9g38drmyRqQ==", 998 | "dependencies": { 999 | "side-channel": "^1.0.4" 1000 | }, 1001 | "engines": { 1002 | "node": ">=0.6" 1003 | }, 1004 | "funding": { 1005 | "url": "https://github.com/sponsors/ljharb" 1006 | } 1007 | }, 1008 | "node_modules/range-parser": { 1009 | "version": "1.2.1", 1010 | "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz", 1011 | "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==", 1012 | "engines": { 1013 | "node": ">= 0.6" 1014 | } 1015 | }, 1016 | "node_modules/raw-body": { 1017 | "version": "2.5.1", 1018 | "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.5.1.tgz", 1019 | "integrity": "sha512-qqJBtEyVgS0ZmPGdCFPWJ3FreoqvG4MVQln/kCgF7Olq95IbOp0/BWyMwbdtn4VTvkM8Y7khCQ2Xgk/tcrCXig==", 1020 | "dependencies": { 1021 | "bytes": "3.1.2", 1022 | "http-errors": "2.0.0", 1023 | "iconv-lite": "0.4.24", 1024 | "unpipe": "1.0.0" 1025 | }, 1026 | "engines": { 1027 | "node": ">= 0.8" 1028 | } 1029 | }, 1030 | "node_modules/readdirp": { 1031 | "version": "3.6.0", 1032 | "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.6.0.tgz", 1033 | "integrity": "sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==", 1034 | "dev": true, 1035 | "dependencies": { 1036 | "picomatch": "^2.2.1" 1037 | }, 1038 | "engines": { 1039 | "node": ">=8.10.0" 1040 | } 1041 | }, 1042 | "node_modules/safe-buffer": { 1043 | "version": "5.2.1", 1044 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz", 1045 | "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==", 1046 | "funding": [ 1047 | { 1048 | "type": "github", 1049 | "url": "https://github.com/sponsors/feross" 1050 | }, 1051 | { 1052 | "type": "patreon", 1053 | "url": "https://www.patreon.com/feross" 1054 | }, 1055 | { 1056 | "type": "consulting", 1057 | "url": "https://feross.org/support" 1058 | } 1059 | ] 1060 | }, 1061 | "node_modules/safer-buffer": { 1062 | "version": "2.1.2", 1063 | "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", 1064 | "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==" 1065 | }, 1066 | "node_modules/saslprep": { 1067 | "version": "1.0.3", 1068 | "resolved": "https://registry.npmjs.org/saslprep/-/saslprep-1.0.3.tgz", 1069 | "integrity": "sha512-/MY/PEMbk2SuY5sScONwhUDsV2p77Znkb/q3nSVstq/yQzYJOH/Azh29p9oJLsl3LnQwSvZDKagDGBsBwSooag==", 1070 | "optional": true, 1071 | "dependencies": { 1072 | "sparse-bitfield": "^3.0.3" 1073 | }, 1074 | "engines": { 1075 | "node": ">=6" 1076 | } 1077 | }, 1078 | "node_modules/semver": { 1079 | "version": "5.7.1", 1080 | "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", 1081 | "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", 1082 | "dev": true, 1083 | "bin": { 1084 | "semver": "bin/semver" 1085 | } 1086 | }, 1087 | "node_modules/send": { 1088 | "version": "0.18.0", 1089 | "resolved": "https://registry.npmjs.org/send/-/send-0.18.0.tgz", 1090 | "integrity": "sha512-qqWzuOjSFOuqPjFe4NOsMLafToQQwBSOEpS+FwEt3A2V3vKubTquT3vmLTQpFgMXp8AlFWFuP1qKaJZOtPpVXg==", 1091 | "dependencies": { 1092 | "debug": "2.6.9", 1093 | "depd": "2.0.0", 1094 | "destroy": "1.2.0", 1095 | "encodeurl": "~1.0.2", 1096 | "escape-html": "~1.0.3", 1097 | "etag": "~1.8.1", 1098 | "fresh": "0.5.2", 1099 | "http-errors": "2.0.0", 1100 | "mime": "1.6.0", 1101 | "ms": "2.1.3", 1102 | "on-finished": "2.4.1", 1103 | "range-parser": "~1.2.1", 1104 | "statuses": "2.0.1" 1105 | }, 1106 | "engines": { 1107 | "node": ">= 0.8.0" 1108 | } 1109 | }, 1110 | "node_modules/send/node_modules/ms": { 1111 | "version": "2.1.3", 1112 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", 1113 | "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==" 1114 | }, 1115 | "node_modules/serve-static": { 1116 | "version": "1.15.0", 1117 | "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.15.0.tgz", 1118 | "integrity": "sha512-XGuRDNjXUijsUL0vl6nSD7cwURuzEgglbOaFuZM9g3kwDXOWVTck0jLzjPzGD+TazWbboZYu52/9/XPdUgne9g==", 1119 | "dependencies": { 1120 | "encodeurl": "~1.0.2", 1121 | "escape-html": "~1.0.3", 1122 | "parseurl": "~1.3.3", 1123 | "send": "0.18.0" 1124 | }, 1125 | "engines": { 1126 | "node": ">= 0.8.0" 1127 | } 1128 | }, 1129 | "node_modules/setprototypeof": { 1130 | "version": "1.2.0", 1131 | "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.2.0.tgz", 1132 | "integrity": "sha512-E5LDX7Wrp85Kil5bhZv46j8jOeboKq5JMmYM3gVGdGH8xFpPWXUMsNrlODCrkoxMEeNi/XZIwuRvY4XNwYMJpw==" 1133 | }, 1134 | "node_modules/side-channel": { 1135 | "version": "1.0.4", 1136 | "resolved": "https://registry.npmjs.org/side-channel/-/side-channel-1.0.4.tgz", 1137 | "integrity": "sha512-q5XPytqFEIKHkGdiMIrY10mvLRvnQh42/+GoBlFW3b2LXLE2xxJpZFdm94we0BaoV3RwJyGqg5wS7epxTv0Zvw==", 1138 | "dependencies": { 1139 | "call-bind": "^1.0.0", 1140 | "get-intrinsic": "^1.0.2", 1141 | "object-inspect": "^1.9.0" 1142 | }, 1143 | "funding": { 1144 | "url": "https://github.com/sponsors/ljharb" 1145 | } 1146 | }, 1147 | "node_modules/sift": { 1148 | "version": "16.0.0", 1149 | "resolved": "https://registry.npmjs.org/sift/-/sift-16.0.0.tgz", 1150 | "integrity": "sha512-ILTjdP2Mv9V1kIxWMXeMTIRbOBrqKc4JAXmFMnFq3fKeyQ2Qwa3Dw1ubcye3vR+Y6ofA0b9gNDr/y2t6eUeIzQ==" 1151 | }, 1152 | "node_modules/simple-update-notifier": { 1153 | "version": "1.0.7", 1154 | "resolved": "https://registry.npmjs.org/simple-update-notifier/-/simple-update-notifier-1.0.7.tgz", 1155 | "integrity": "sha512-BBKgR84BJQJm6WjWFMHgLVuo61FBDSj1z/xSFUIozqO6wO7ii0JxCqlIud7Enr/+LhlbNI0whErq96P2qHNWew==", 1156 | "dev": true, 1157 | "dependencies": { 1158 | "semver": "~7.0.0" 1159 | }, 1160 | "engines": { 1161 | "node": ">=8.10.0" 1162 | } 1163 | }, 1164 | "node_modules/simple-update-notifier/node_modules/semver": { 1165 | "version": "7.0.0", 1166 | "resolved": "https://registry.npmjs.org/semver/-/semver-7.0.0.tgz", 1167 | "integrity": "sha512-+GB6zVA9LWh6zovYQLALHwv5rb2PHGlJi3lfiqIHxR0uuwCgefcOJc59v9fv1w8GbStwxuuqqAjI9NMAOOgq1A==", 1168 | "dev": true, 1169 | "bin": { 1170 | "semver": "bin/semver.js" 1171 | } 1172 | }, 1173 | "node_modules/smart-buffer": { 1174 | "version": "4.2.0", 1175 | "resolved": "https://registry.npmjs.org/smart-buffer/-/smart-buffer-4.2.0.tgz", 1176 | "integrity": "sha512-94hK0Hh8rPqQl2xXc3HsaBoOXKV20MToPkcXvwbISWLEs+64sBq5kFgn2kJDHb1Pry9yrP0dxrCI9RRci7RXKg==", 1177 | "engines": { 1178 | "node": ">= 6.0.0", 1179 | "npm": ">= 3.0.0" 1180 | } 1181 | }, 1182 | "node_modules/socks": { 1183 | "version": "2.7.0", 1184 | "resolved": "https://registry.npmjs.org/socks/-/socks-2.7.0.tgz", 1185 | "integrity": "sha512-scnOe9y4VuiNUULJN72GrM26BNOjVsfPXI+j+98PkyEfsIXroa5ofyjT+FzGvn/xHs73U2JtoBYAVx9Hl4quSA==", 1186 | "dependencies": { 1187 | "ip": "^2.0.0", 1188 | "smart-buffer": "^4.2.0" 1189 | }, 1190 | "engines": { 1191 | "node": ">= 10.13.0", 1192 | "npm": ">= 3.0.0" 1193 | } 1194 | }, 1195 | "node_modules/sparse-bitfield": { 1196 | "version": "3.0.3", 1197 | "resolved": "https://registry.npmjs.org/sparse-bitfield/-/sparse-bitfield-3.0.3.tgz", 1198 | "integrity": "sha512-kvzhi7vqKTfkh0PZU+2D2PIllw2ymqJKujUcyPMd9Y75Nv4nPbGJZXNhxsgdQab2BmlDct1YnfQCguEvHr7VsQ==", 1199 | "optional": true, 1200 | "dependencies": { 1201 | "memory-pager": "^1.0.2" 1202 | } 1203 | }, 1204 | "node_modules/statuses": { 1205 | "version": "2.0.1", 1206 | "resolved": "https://registry.npmjs.org/statuses/-/statuses-2.0.1.tgz", 1207 | "integrity": "sha512-RwNA9Z/7PrK06rYLIzFMlaF+l73iwpzsqRIFgbMLbTcLD6cOao82TaWefPXQvB2fOC4AjuYSEndS7N/mTCbkdQ==", 1208 | "engines": { 1209 | "node": ">= 0.8" 1210 | } 1211 | }, 1212 | "node_modules/supports-color": { 1213 | "version": "5.5.0", 1214 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", 1215 | "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", 1216 | "dev": true, 1217 | "dependencies": { 1218 | "has-flag": "^3.0.0" 1219 | }, 1220 | "engines": { 1221 | "node": ">=4" 1222 | } 1223 | }, 1224 | "node_modules/to-regex-range": { 1225 | "version": "5.0.1", 1226 | "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", 1227 | "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", 1228 | "dev": true, 1229 | "dependencies": { 1230 | "is-number": "^7.0.0" 1231 | }, 1232 | "engines": { 1233 | "node": ">=8.0" 1234 | } 1235 | }, 1236 | "node_modules/toidentifier": { 1237 | "version": "1.0.1", 1238 | "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.1.tgz", 1239 | "integrity": "sha512-o5sSPKEkg/DIQNmH43V0/uerLrpzVedkUh8tGNvaeXpfpuwjKenlSox/2O/BTlZUtEe+JG7s5YhEz608PlAHRA==", 1240 | "engines": { 1241 | "node": ">=0.6" 1242 | } 1243 | }, 1244 | "node_modules/touch": { 1245 | "version": "3.1.0", 1246 | "resolved": "https://registry.npmjs.org/touch/-/touch-3.1.0.tgz", 1247 | "integrity": "sha512-WBx8Uy5TLtOSRtIq+M03/sKDrXCLHxwDcquSP2c43Le03/9serjQBIztjRz6FkJez9D/hleyAXTBGLwwZUw9lA==", 1248 | "dev": true, 1249 | "dependencies": { 1250 | "nopt": "~1.0.10" 1251 | }, 1252 | "bin": { 1253 | "nodetouch": "bin/nodetouch.js" 1254 | } 1255 | }, 1256 | "node_modules/tr46": { 1257 | "version": "3.0.0", 1258 | "resolved": "https://registry.npmjs.org/tr46/-/tr46-3.0.0.tgz", 1259 | "integrity": "sha512-l7FvfAHlcmulp8kr+flpQZmVwtu7nfRV7NZujtN0OqES8EL4O4e0qqzL0DC5gAvx/ZC/9lk6rhcUwYvkBnBnYA==", 1260 | "dependencies": { 1261 | "punycode": "^2.1.1" 1262 | }, 1263 | "engines": { 1264 | "node": ">=12" 1265 | } 1266 | }, 1267 | "node_modules/type-is": { 1268 | "version": "1.6.18", 1269 | "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz", 1270 | "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==", 1271 | "dependencies": { 1272 | "media-typer": "0.3.0", 1273 | "mime-types": "~2.1.24" 1274 | }, 1275 | "engines": { 1276 | "node": ">= 0.6" 1277 | } 1278 | }, 1279 | "node_modules/undefsafe": { 1280 | "version": "2.0.5", 1281 | "resolved": "https://registry.npmjs.org/undefsafe/-/undefsafe-2.0.5.tgz", 1282 | "integrity": "sha512-WxONCrssBM8TSPRqN5EmsjVrsv4A8X12J4ArBiiayv3DyyG3ZlIg6yysuuSYdZsVz3TKcTg2fd//Ujd4CHV1iA==", 1283 | "dev": true 1284 | }, 1285 | "node_modules/unpipe": { 1286 | "version": "1.0.0", 1287 | "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", 1288 | "integrity": "sha512-pjy2bYhSsufwWlKwPc+l3cN7+wuJlK6uz0YdJEOlQDbl6jo/YlPi4mb8agUkVC8BF7V8NuzeyPNqRksA3hztKQ==", 1289 | "engines": { 1290 | "node": ">= 0.8" 1291 | } 1292 | }, 1293 | "node_modules/utils-merge": { 1294 | "version": "1.0.1", 1295 | "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz", 1296 | "integrity": "sha512-pMZTvIkT1d+TFGvDOqodOclx0QWkkgi6Tdoa8gC8ffGAAqz9pzPTZWAybbsHHoED/ztMtkv/VoYTYyShUn81hA==", 1297 | "engines": { 1298 | "node": ">= 0.4.0" 1299 | } 1300 | }, 1301 | "node_modules/vary": { 1302 | "version": "1.1.2", 1303 | "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz", 1304 | "integrity": "sha512-BNGbWLfd0eUPabhkXUVm0j8uuvREyTh5ovRa/dyow/BqAbZJyC+5fU+IzQOzmAKzYqYRAISoRhdQr3eIZ/PXqg==", 1305 | "engines": { 1306 | "node": ">= 0.8" 1307 | } 1308 | }, 1309 | "node_modules/webidl-conversions": { 1310 | "version": "7.0.0", 1311 | "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-7.0.0.tgz", 1312 | "integrity": "sha512-VwddBukDzu71offAQR975unBIGqfKZpM+8ZX6ySk8nYhVoo5CYaZyzt3YBvYtRtO+aoGlqxPg/B87NGVZ/fu6g==", 1313 | "engines": { 1314 | "node": ">=12" 1315 | } 1316 | }, 1317 | "node_modules/whatwg-url": { 1318 | "version": "11.0.0", 1319 | "resolved": "https://registry.npmjs.org/whatwg-url/-/whatwg-url-11.0.0.tgz", 1320 | "integrity": "sha512-RKT8HExMpoYx4igMiVMY83lN6UeITKJlBQ+vR/8ZJ8OCdSiN3RwCq+9gH0+Xzj0+5IrM6i4j/6LuvzbZIQgEcQ==", 1321 | "dependencies": { 1322 | "tr46": "^3.0.0", 1323 | "webidl-conversions": "^7.0.0" 1324 | }, 1325 | "engines": { 1326 | "node": ">=12" 1327 | } 1328 | } 1329 | }, 1330 | "dependencies": { 1331 | "@hapi/hoek": { 1332 | "version": "9.3.0", 1333 | "resolved": "https://registry.npmjs.org/@hapi/hoek/-/hoek-9.3.0.tgz", 1334 | "integrity": "sha512-/c6rf4UJlmHlC9b5BaNvzAcFv7HZ2QHaV0D4/HNlBdvFnvQq8RI4kYdhyPCl7Xj+oWvTWQ8ujhqS53LIgAe6KQ==" 1335 | }, 1336 | "@hapi/topo": { 1337 | "version": "5.1.0", 1338 | "resolved": "https://registry.npmjs.org/@hapi/topo/-/topo-5.1.0.tgz", 1339 | "integrity": "sha512-foQZKJig7Ob0BMAYBfcJk8d77QtOe7Wo4ox7ff1lQYoNNAb6jwcY1ncdoy2e9wQZzvNy7ODZCYJkK8kzmcAnAg==", 1340 | "requires": { 1341 | "@hapi/hoek": "^9.0.0" 1342 | } 1343 | }, 1344 | "@sideway/address": { 1345 | "version": "4.1.4", 1346 | "resolved": "https://registry.npmjs.org/@sideway/address/-/address-4.1.4.tgz", 1347 | "integrity": "sha512-7vwq+rOHVWjyXxVlR76Agnvhy8I9rpzjosTESvmhNeXOXdZZB15Fl+TI9x1SiHZH5Jv2wTGduSxFDIaq0m3DUw==", 1348 | "requires": { 1349 | "@hapi/hoek": "^9.0.0" 1350 | } 1351 | }, 1352 | "@sideway/formula": { 1353 | "version": "3.0.0", 1354 | "resolved": "https://registry.npmjs.org/@sideway/formula/-/formula-3.0.0.tgz", 1355 | "integrity": "sha512-vHe7wZ4NOXVfkoRb8T5otiENVlT7a3IAiw7H5M2+GO+9CDgcVUUsX1zalAztCmwyOr2RUTGJdgB+ZvSVqmdHmg==" 1356 | }, 1357 | "@sideway/pinpoint": { 1358 | "version": "2.0.0", 1359 | "resolved": "https://registry.npmjs.org/@sideway/pinpoint/-/pinpoint-2.0.0.tgz", 1360 | "integrity": "sha512-RNiOoTPkptFtSVzQevY/yWtZwf/RxyVnPy/OcA9HBM3MlGDnBEYL5B41H0MTn0Uec8Hi+2qUtTfG2WWZBmMejQ==" 1361 | }, 1362 | "@types/node": { 1363 | "version": "18.7.17", 1364 | "resolved": "https://registry.npmjs.org/@types/node/-/node-18.7.17.tgz", 1365 | "integrity": "sha512-0UyfUnt02zIuqp7yC8RYtDkp/vo8bFaQ13KkSEvUAohPOAlnVNbj5Fi3fgPSuwzakS+EvvnnZ4x9y7i6ASaSPQ==" 1366 | }, 1367 | "@types/webidl-conversions": { 1368 | "version": "7.0.0", 1369 | "resolved": "https://registry.npmjs.org/@types/webidl-conversions/-/webidl-conversions-7.0.0.tgz", 1370 | "integrity": "sha512-xTE1E+YF4aWPJJeUzaZI5DRntlkY3+BCVJi0axFptnjGmAoWxkyREIh/XMrfxVLejwQxMCfDXdICo0VLxThrog==" 1371 | }, 1372 | "@types/whatwg-url": { 1373 | "version": "8.2.2", 1374 | "resolved": "https://registry.npmjs.org/@types/whatwg-url/-/whatwg-url-8.2.2.tgz", 1375 | "integrity": "sha512-FtQu10RWgn3D9U4aazdwIE2yzphmTJREDqNdODHrbrZmmMqI0vMheC/6NE/J1Yveaj8H+ela+YwWTjq5PGmuhA==", 1376 | "requires": { 1377 | "@types/node": "*", 1378 | "@types/webidl-conversions": "*" 1379 | } 1380 | }, 1381 | "abbrev": { 1382 | "version": "1.1.1", 1383 | "resolved": "https://registry.npmjs.org/abbrev/-/abbrev-1.1.1.tgz", 1384 | "integrity": "sha512-nne9/IiQ/hzIhY6pdDnbBtz7DjPTKrY00P/zvPSm5pOFkl6xuGrGnXn/VtTNNfNtAfZ9/1RtehkszU9qcTii0Q==", 1385 | "dev": true 1386 | }, 1387 | "accepts": { 1388 | "version": "1.3.8", 1389 | "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.8.tgz", 1390 | "integrity": "sha512-PYAthTa2m2VKxuvSD3DPC/Gy+U+sOA1LAuT8mkmRuvw+NACSaeXEQ+NHcVF7rONl6qcaxV3Uuemwawk+7+SJLw==", 1391 | "requires": { 1392 | "mime-types": "~2.1.34", 1393 | "negotiator": "0.6.3" 1394 | } 1395 | }, 1396 | "anymatch": { 1397 | "version": "3.1.2", 1398 | "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.2.tgz", 1399 | "integrity": "sha512-P43ePfOAIupkguHUycrc4qJ9kz8ZiuOUijaETwX7THt0Y/GNK7v0aa8rY816xWjZ7rJdA5XdMcpVFTKMq+RvWg==", 1400 | "dev": true, 1401 | "requires": { 1402 | "normalize-path": "^3.0.0", 1403 | "picomatch": "^2.0.4" 1404 | } 1405 | }, 1406 | "array-flatten": { 1407 | "version": "1.1.1", 1408 | "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz", 1409 | "integrity": "sha512-PCVAQswWemu6UdxsDFFX/+gVeYqKAod3D3UVm91jHwynguOwAvYPhx8nNlM++NqRcK6CxxpUafjmhIdKiHibqg==" 1410 | }, 1411 | "balanced-match": { 1412 | "version": "1.0.2", 1413 | "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", 1414 | "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", 1415 | "dev": true 1416 | }, 1417 | "base64-js": { 1418 | "version": "1.5.1", 1419 | "resolved": "https://registry.npmjs.org/base64-js/-/base64-js-1.5.1.tgz", 1420 | "integrity": "sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==" 1421 | }, 1422 | "binary-extensions": { 1423 | "version": "2.2.0", 1424 | "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.2.0.tgz", 1425 | "integrity": "sha512-jDctJ/IVQbZoJykoeHbhXpOlNBqGNcwXJKJog42E5HDPUwQTSdjCHdihjj0DlnheQ7blbT6dHOafNAiS8ooQKA==", 1426 | "dev": true 1427 | }, 1428 | "body-parser": { 1429 | "version": "1.20.0", 1430 | "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.20.0.tgz", 1431 | "integrity": "sha512-DfJ+q6EPcGKZD1QWUjSpqp+Q7bDQTsQIF4zfUAtZ6qk+H/3/QRhg9CEp39ss+/T2vw0+HaidC0ecJj/DRLIaKg==", 1432 | "requires": { 1433 | "bytes": "3.1.2", 1434 | "content-type": "~1.0.4", 1435 | "debug": "2.6.9", 1436 | "depd": "2.0.0", 1437 | "destroy": "1.2.0", 1438 | "http-errors": "2.0.0", 1439 | "iconv-lite": "0.4.24", 1440 | "on-finished": "2.4.1", 1441 | "qs": "6.10.3", 1442 | "raw-body": "2.5.1", 1443 | "type-is": "~1.6.18", 1444 | "unpipe": "1.0.0" 1445 | } 1446 | }, 1447 | "brace-expansion": { 1448 | "version": "1.1.11", 1449 | "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", 1450 | "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", 1451 | "dev": true, 1452 | "requires": { 1453 | "balanced-match": "^1.0.0", 1454 | "concat-map": "0.0.1" 1455 | } 1456 | }, 1457 | "braces": { 1458 | "version": "3.0.2", 1459 | "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz", 1460 | "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", 1461 | "dev": true, 1462 | "requires": { 1463 | "fill-range": "^7.0.1" 1464 | } 1465 | }, 1466 | "bson": { 1467 | "version": "4.7.0", 1468 | "resolved": "https://registry.npmjs.org/bson/-/bson-4.7.0.tgz", 1469 | "integrity": "sha512-VrlEE4vuiO1WTpfof4VmaVolCVYkYTgB9iWgYNOrVlnifpME/06fhFRmONgBhClD5pFC1t9ZWqFUQEQAzY43bA==", 1470 | "requires": { 1471 | "buffer": "^5.6.0" 1472 | } 1473 | }, 1474 | "buffer": { 1475 | "version": "5.7.1", 1476 | "resolved": "https://registry.npmjs.org/buffer/-/buffer-5.7.1.tgz", 1477 | "integrity": "sha512-EHcyIPBQ4BSGlvjB16k5KgAJ27CIsHY/2JBmCRReo48y9rQ3MaUzWX3KVlBa4U7MyX02HdVj0K7C3WaB3ju7FQ==", 1478 | "requires": { 1479 | "base64-js": "^1.3.1", 1480 | "ieee754": "^1.1.13" 1481 | } 1482 | }, 1483 | "bytes": { 1484 | "version": "3.1.2", 1485 | "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.2.tgz", 1486 | "integrity": "sha512-/Nf7TyzTx6S3yRJObOAV7956r8cr2+Oj8AC5dt8wSP3BQAoeX58NoHyCU8P8zGkNXStjTSi6fzO6F0pBdcYbEg==" 1487 | }, 1488 | "call-bind": { 1489 | "version": "1.0.2", 1490 | "resolved": "https://registry.npmjs.org/call-bind/-/call-bind-1.0.2.tgz", 1491 | "integrity": "sha512-7O+FbCihrB5WGbFYesctwmTKae6rOiIzmz1icreWJ+0aA7LJfuqhEso2T9ncpcFtzMQtzXf2QGGueWJGTYsqrA==", 1492 | "requires": { 1493 | "function-bind": "^1.1.1", 1494 | "get-intrinsic": "^1.0.2" 1495 | } 1496 | }, 1497 | "chokidar": { 1498 | "version": "3.5.3", 1499 | "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.5.3.tgz", 1500 | "integrity": "sha512-Dr3sfKRP6oTcjf2JmUmFJfeVMvXBdegxB0iVQ5eb2V10uFJUCAS8OByZdVAyVb8xXNz3GjjTgj9kLWsZTqE6kw==", 1501 | "dev": true, 1502 | "requires": { 1503 | "anymatch": "~3.1.2", 1504 | "braces": "~3.0.2", 1505 | "fsevents": "~2.3.2", 1506 | "glob-parent": "~5.1.2", 1507 | "is-binary-path": "~2.1.0", 1508 | "is-glob": "~4.0.1", 1509 | "normalize-path": "~3.0.0", 1510 | "readdirp": "~3.6.0" 1511 | } 1512 | }, 1513 | "concat-map": { 1514 | "version": "0.0.1", 1515 | "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", 1516 | "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==", 1517 | "dev": true 1518 | }, 1519 | "content-disposition": { 1520 | "version": "0.5.4", 1521 | "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.4.tgz", 1522 | "integrity": "sha512-FveZTNuGw04cxlAiWbzi6zTAL/lhehaWbTtgluJh4/E95DqMwTmha3KZN1aAWA8cFIhHzMZUvLevkw5Rqk+tSQ==", 1523 | "requires": { 1524 | "safe-buffer": "5.2.1" 1525 | } 1526 | }, 1527 | "content-type": { 1528 | "version": "1.0.4", 1529 | "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.4.tgz", 1530 | "integrity": "sha512-hIP3EEPs8tB9AT1L+NUqtwOAps4mk2Zob89MWXMHjHWg9milF/j4osnnQLXBCBFBk/tvIG/tUc9mOUJiPBhPXA==" 1531 | }, 1532 | "cookie": { 1533 | "version": "0.5.0", 1534 | "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.5.0.tgz", 1535 | "integrity": "sha512-YZ3GUyn/o8gfKJlnlX7g7xq4gyO6OSuhGPKaaGssGB2qgDUS0gPgtTvoyZLTt9Ab6dC4hfc9dV5arkvc/OCmrw==" 1536 | }, 1537 | "cookie-signature": { 1538 | "version": "1.0.6", 1539 | "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz", 1540 | "integrity": "sha512-QADzlaHc8icV8I7vbaJXJwod9HWYp8uCqf1xa4OfNu1T7JVxQIrUgOWtHdNDtPiywmFbiS12VjotIXLrKM3orQ==" 1541 | }, 1542 | "cors": { 1543 | "version": "2.8.5", 1544 | "resolved": "https://registry.npmjs.org/cors/-/cors-2.8.5.tgz", 1545 | "integrity": "sha512-KIHbLJqu73RGr/hnbrO9uBeixNGuvSQjul/jdFvS/KFSIH1hWVd1ng7zOHx+YrEfInLG7q4n6GHQ9cDtxv/P6g==", 1546 | "requires": { 1547 | "object-assign": "^4", 1548 | "vary": "^1" 1549 | } 1550 | }, 1551 | "debug": { 1552 | "version": "2.6.9", 1553 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", 1554 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", 1555 | "requires": { 1556 | "ms": "2.0.0" 1557 | } 1558 | }, 1559 | "denque": { 1560 | "version": "2.1.0", 1561 | "resolved": "https://registry.npmjs.org/denque/-/denque-2.1.0.tgz", 1562 | "integrity": "sha512-HVQE3AAb/pxF8fQAoiqpvg9i3evqug3hoiwakOyZAwJm+6vZehbkYXZ0l4JxS+I3QxM97v5aaRNhj8v5oBhekw==" 1563 | }, 1564 | "depd": { 1565 | "version": "2.0.0", 1566 | "resolved": "https://registry.npmjs.org/depd/-/depd-2.0.0.tgz", 1567 | "integrity": "sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw==" 1568 | }, 1569 | "destroy": { 1570 | "version": "1.2.0", 1571 | "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.2.0.tgz", 1572 | "integrity": "sha512-2sJGJTaXIIaR1w4iJSNoN0hnMY7Gpc/n8D4qSCJw8QqFWXf7cuAgnEHxBpweaVcPevC2l3KpjYCx3NypQQgaJg==" 1573 | }, 1574 | "dotenv": { 1575 | "version": "16.0.2", 1576 | "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-16.0.2.tgz", 1577 | "integrity": "sha512-JvpYKUmzQhYoIFgK2MOnF3bciIZoItIIoryihy0rIA+H4Jy0FmgyKYAHCTN98P5ybGSJcIFbh6QKeJdtZd1qhA==" 1578 | }, 1579 | "ee-first": { 1580 | "version": "1.1.1", 1581 | "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", 1582 | "integrity": "sha512-WMwm9LhRUo+WUaRN+vRuETqG89IgZphVSNkdFgeb6sS/E4OrDIN7t48CAewSHXc6C8lefD8KKfr5vY61brQlow==" 1583 | }, 1584 | "encodeurl": { 1585 | "version": "1.0.2", 1586 | "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz", 1587 | "integrity": "sha512-TPJXq8JqFaVYm2CWmPvnP2Iyo4ZSM7/QKcSmuMLDObfpH5fi7RUGmd/rTDf+rut/saiDiQEeVTNgAmJEdAOx0w==" 1588 | }, 1589 | "escape-html": { 1590 | "version": "1.0.3", 1591 | "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz", 1592 | "integrity": "sha512-NiSupZ4OeuGwr68lGIeym/ksIZMJodUGOSCZ/FSnTxcrekbvqrgdUxlJOMpijaKZVjAJrWrGs/6Jy8OMuyj9ow==" 1593 | }, 1594 | "etag": { 1595 | "version": "1.8.1", 1596 | "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz", 1597 | "integrity": "sha512-aIL5Fx7mawVa300al2BnEE4iNvo1qETxLrPI/o05L7z6go7fCw1J6EQmbK4FmJ2AS7kgVF/KEZWufBfdClMcPg==" 1598 | }, 1599 | "express": { 1600 | "version": "4.18.1", 1601 | "resolved": "https://registry.npmjs.org/express/-/express-4.18.1.tgz", 1602 | "integrity": "sha512-zZBcOX9TfehHQhtupq57OF8lFZ3UZi08Y97dwFCkD8p9d/d2Y3M+ykKcwaMDEL+4qyUolgBDX6AblpR3fL212Q==", 1603 | "requires": { 1604 | "accepts": "~1.3.8", 1605 | "array-flatten": "1.1.1", 1606 | "body-parser": "1.20.0", 1607 | "content-disposition": "0.5.4", 1608 | "content-type": "~1.0.4", 1609 | "cookie": "0.5.0", 1610 | "cookie-signature": "1.0.6", 1611 | "debug": "2.6.9", 1612 | "depd": "2.0.0", 1613 | "encodeurl": "~1.0.2", 1614 | "escape-html": "~1.0.3", 1615 | "etag": "~1.8.1", 1616 | "finalhandler": "1.2.0", 1617 | "fresh": "0.5.2", 1618 | "http-errors": "2.0.0", 1619 | "merge-descriptors": "1.0.1", 1620 | "methods": "~1.1.2", 1621 | "on-finished": "2.4.1", 1622 | "parseurl": "~1.3.3", 1623 | "path-to-regexp": "0.1.7", 1624 | "proxy-addr": "~2.0.7", 1625 | "qs": "6.10.3", 1626 | "range-parser": "~1.2.1", 1627 | "safe-buffer": "5.2.1", 1628 | "send": "0.18.0", 1629 | "serve-static": "1.15.0", 1630 | "setprototypeof": "1.2.0", 1631 | "statuses": "2.0.1", 1632 | "type-is": "~1.6.18", 1633 | "utils-merge": "1.0.1", 1634 | "vary": "~1.1.2" 1635 | } 1636 | }, 1637 | "fill-range": { 1638 | "version": "7.0.1", 1639 | "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz", 1640 | "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", 1641 | "dev": true, 1642 | "requires": { 1643 | "to-regex-range": "^5.0.1" 1644 | } 1645 | }, 1646 | "finalhandler": { 1647 | "version": "1.2.0", 1648 | "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.2.0.tgz", 1649 | "integrity": "sha512-5uXcUVftlQMFnWC9qu/svkWv3GTd2PfUhK/3PLkYNAe7FbqJMt3515HaxE6eRL74GdsriiwujiawdaB1BpEISg==", 1650 | "requires": { 1651 | "debug": "2.6.9", 1652 | "encodeurl": "~1.0.2", 1653 | "escape-html": "~1.0.3", 1654 | "on-finished": "2.4.1", 1655 | "parseurl": "~1.3.3", 1656 | "statuses": "2.0.1", 1657 | "unpipe": "~1.0.0" 1658 | } 1659 | }, 1660 | "forwarded": { 1661 | "version": "0.2.0", 1662 | "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.2.0.tgz", 1663 | "integrity": "sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow==" 1664 | }, 1665 | "fresh": { 1666 | "version": "0.5.2", 1667 | "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz", 1668 | "integrity": "sha512-zJ2mQYM18rEFOudeV4GShTGIQ7RbzA7ozbU9I/XBpm7kqgMywgmylMwXHxZJmkVoYkna9d2pVXVXPdYTP9ej8Q==" 1669 | }, 1670 | "fsevents": { 1671 | "version": "2.3.2", 1672 | "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.2.tgz", 1673 | "integrity": "sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==", 1674 | "dev": true, 1675 | "optional": true 1676 | }, 1677 | "function-bind": { 1678 | "version": "1.1.1", 1679 | "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.1.tgz", 1680 | "integrity": "sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A==" 1681 | }, 1682 | "get-intrinsic": { 1683 | "version": "1.1.3", 1684 | "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.1.3.tgz", 1685 | "integrity": "sha512-QJVz1Tj7MS099PevUG5jvnt9tSkXN8K14dxQlikJuPt4uD9hHAHjLyLBiLR5zELelBdD9QNRAXZzsJx0WaDL9A==", 1686 | "requires": { 1687 | "function-bind": "^1.1.1", 1688 | "has": "^1.0.3", 1689 | "has-symbols": "^1.0.3" 1690 | } 1691 | }, 1692 | "glob-parent": { 1693 | "version": "5.1.2", 1694 | "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", 1695 | "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", 1696 | "dev": true, 1697 | "requires": { 1698 | "is-glob": "^4.0.1" 1699 | } 1700 | }, 1701 | "has": { 1702 | "version": "1.0.3", 1703 | "resolved": "https://registry.npmjs.org/has/-/has-1.0.3.tgz", 1704 | "integrity": "sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw==", 1705 | "requires": { 1706 | "function-bind": "^1.1.1" 1707 | } 1708 | }, 1709 | "has-flag": { 1710 | "version": "3.0.0", 1711 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", 1712 | "integrity": "sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==", 1713 | "dev": true 1714 | }, 1715 | "has-symbols": { 1716 | "version": "1.0.3", 1717 | "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.0.3.tgz", 1718 | "integrity": "sha512-l3LCuF6MgDNwTDKkdYGEihYjt5pRPbEg46rtlmnSPlUbgmB8LOIrKJbYYFBSbnPaJexMKtiPO8hmeRjRz2Td+A==" 1719 | }, 1720 | "http-errors": { 1721 | "version": "2.0.0", 1722 | "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-2.0.0.tgz", 1723 | "integrity": "sha512-FtwrG/euBzaEjYeRqOgly7G0qviiXoJWnvEH2Z1plBdXgbyjv34pHTSb9zoeHMyDy33+DWy5Wt9Wo+TURtOYSQ==", 1724 | "requires": { 1725 | "depd": "2.0.0", 1726 | "inherits": "2.0.4", 1727 | "setprototypeof": "1.2.0", 1728 | "statuses": "2.0.1", 1729 | "toidentifier": "1.0.1" 1730 | } 1731 | }, 1732 | "iconv-lite": { 1733 | "version": "0.4.24", 1734 | "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", 1735 | "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", 1736 | "requires": { 1737 | "safer-buffer": ">= 2.1.2 < 3" 1738 | } 1739 | }, 1740 | "ieee754": { 1741 | "version": "1.2.1", 1742 | "resolved": "https://registry.npmjs.org/ieee754/-/ieee754-1.2.1.tgz", 1743 | "integrity": "sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==" 1744 | }, 1745 | "ignore-by-default": { 1746 | "version": "1.0.1", 1747 | "resolved": "https://registry.npmjs.org/ignore-by-default/-/ignore-by-default-1.0.1.tgz", 1748 | "integrity": "sha512-Ius2VYcGNk7T90CppJqcIkS5ooHUZyIQK+ClZfMfMNFEF9VSE73Fq+906u/CWu92x4gzZMWOwfFYckPObzdEbA==", 1749 | "dev": true 1750 | }, 1751 | "inherits": { 1752 | "version": "2.0.4", 1753 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", 1754 | "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==" 1755 | }, 1756 | "ip": { 1757 | "version": "2.0.0", 1758 | "resolved": "https://registry.npmjs.org/ip/-/ip-2.0.0.tgz", 1759 | "integrity": "sha512-WKa+XuLG1A1R0UWhl2+1XQSi+fZWMsYKffMZTTYsiZaUD8k2yDAj5atimTUD2TZkyCkNEeYE5NhFZmupOGtjYQ==" 1760 | }, 1761 | "ipaddr.js": { 1762 | "version": "1.9.1", 1763 | "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz", 1764 | "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==" 1765 | }, 1766 | "is-binary-path": { 1767 | "version": "2.1.0", 1768 | "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz", 1769 | "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==", 1770 | "dev": true, 1771 | "requires": { 1772 | "binary-extensions": "^2.0.0" 1773 | } 1774 | }, 1775 | "is-extglob": { 1776 | "version": "2.1.1", 1777 | "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", 1778 | "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==", 1779 | "dev": true 1780 | }, 1781 | "is-glob": { 1782 | "version": "4.0.3", 1783 | "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz", 1784 | "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==", 1785 | "dev": true, 1786 | "requires": { 1787 | "is-extglob": "^2.1.1" 1788 | } 1789 | }, 1790 | "is-number": { 1791 | "version": "7.0.0", 1792 | "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", 1793 | "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", 1794 | "dev": true 1795 | }, 1796 | "joi": { 1797 | "version": "17.6.0", 1798 | "resolved": "https://registry.npmjs.org/joi/-/joi-17.6.0.tgz", 1799 | "integrity": "sha512-OX5dG6DTbcr/kbMFj0KGYxuew69HPcAE3K/sZpEV2nP6e/j/C0HV+HNiBPCASxdx5T7DMoa0s8UeHWMnb6n2zw==", 1800 | "requires": { 1801 | "@hapi/hoek": "^9.0.0", 1802 | "@hapi/topo": "^5.0.0", 1803 | "@sideway/address": "^4.1.3", 1804 | "@sideway/formula": "^3.0.0", 1805 | "@sideway/pinpoint": "^2.0.0" 1806 | } 1807 | }, 1808 | "kareem": { 1809 | "version": "2.4.1", 1810 | "resolved": "https://registry.npmjs.org/kareem/-/kareem-2.4.1.tgz", 1811 | "integrity": "sha512-aJ9opVoXroQUPfovYP5kaj2lM7Jn02Gw13bL0lg9v0V7SaUc0qavPs0Eue7d2DcC3NjqI6QAUElXNsuZSeM+EA==" 1812 | }, 1813 | "media-typer": { 1814 | "version": "0.3.0", 1815 | "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", 1816 | "integrity": "sha512-dq+qelQ9akHpcOl/gUVRTxVIOkAJ1wR3QAvb4RsVjS8oVoFjDGTc679wJYmUmknUF5HwMLOgb5O+a3KxfWapPQ==" 1817 | }, 1818 | "memory-pager": { 1819 | "version": "1.5.0", 1820 | "resolved": "https://registry.npmjs.org/memory-pager/-/memory-pager-1.5.0.tgz", 1821 | "integrity": "sha512-ZS4Bp4r/Zoeq6+NLJpP+0Zzm0pR8whtGPf1XExKLJBAczGMnSi3It14OiNCStjQjM6NU1okjQGSxgEZN8eBYKg==", 1822 | "optional": true 1823 | }, 1824 | "merge-descriptors": { 1825 | "version": "1.0.1", 1826 | "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz", 1827 | "integrity": "sha512-cCi6g3/Zr1iqQi6ySbseM1Xvooa98N0w31jzUYrXPX2xqObmFGHJ0tQ5u74H3mVh7wLouTseZyYIq39g8cNp1w==" 1828 | }, 1829 | "methods": { 1830 | "version": "1.1.2", 1831 | "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz", 1832 | "integrity": "sha512-iclAHeNqNm68zFtnZ0e+1L2yUIdvzNoauKU4WBA3VvH/vPFieF7qfRlwUZU+DA9P9bPXIS90ulxoUoCH23sV2w==" 1833 | }, 1834 | "mime": { 1835 | "version": "1.6.0", 1836 | "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz", 1837 | "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==" 1838 | }, 1839 | "mime-db": { 1840 | "version": "1.52.0", 1841 | "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz", 1842 | "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==" 1843 | }, 1844 | "mime-types": { 1845 | "version": "2.1.35", 1846 | "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz", 1847 | "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==", 1848 | "requires": { 1849 | "mime-db": "1.52.0" 1850 | } 1851 | }, 1852 | "minimatch": { 1853 | "version": "3.1.2", 1854 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", 1855 | "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", 1856 | "dev": true, 1857 | "requires": { 1858 | "brace-expansion": "^1.1.7" 1859 | } 1860 | }, 1861 | "mongodb": { 1862 | "version": "4.9.1", 1863 | "resolved": "https://registry.npmjs.org/mongodb/-/mongodb-4.9.1.tgz", 1864 | "integrity": "sha512-ZhgI/qBf84fD7sI4waZBoLBNJYPQN5IOC++SBCiPiyhzpNKOxN/fi0tBHvH2dEC42HXtNEbFB0zmNz4+oVtorQ==", 1865 | "requires": { 1866 | "bson": "^4.7.0", 1867 | "denque": "^2.1.0", 1868 | "mongodb-connection-string-url": "^2.5.3", 1869 | "saslprep": "^1.0.3", 1870 | "socks": "^2.7.0" 1871 | } 1872 | }, 1873 | "mongodb-connection-string-url": { 1874 | "version": "2.5.3", 1875 | "resolved": "https://registry.npmjs.org/mongodb-connection-string-url/-/mongodb-connection-string-url-2.5.3.tgz", 1876 | "integrity": "sha512-f+/WsED+xF4B74l3k9V/XkTVj5/fxFH2o5ToKXd8Iyi5UhM+sO9u0Ape17Mvl/GkZaFtM0HQnzAG5OTmhKw+tQ==", 1877 | "requires": { 1878 | "@types/whatwg-url": "^8.2.1", 1879 | "whatwg-url": "^11.0.0" 1880 | } 1881 | }, 1882 | "mongoose": { 1883 | "version": "6.6.0", 1884 | "resolved": "https://registry.npmjs.org/mongoose/-/mongoose-6.6.0.tgz", 1885 | "integrity": "sha512-5sS0D7qbmfT4G/nODkJhx5l2qvhqLMplhMlp08Wea8eoi6O/B6b+o1ukUEKjjm1MV8dAS3w8kx47R/klYKLPpQ==", 1886 | "requires": { 1887 | "bson": "^4.6.5", 1888 | "kareem": "2.4.1", 1889 | "mongodb": "4.9.1", 1890 | "mpath": "0.9.0", 1891 | "mquery": "4.0.3", 1892 | "ms": "2.1.3", 1893 | "sift": "16.0.0" 1894 | }, 1895 | "dependencies": { 1896 | "ms": { 1897 | "version": "2.1.3", 1898 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", 1899 | "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==" 1900 | } 1901 | } 1902 | }, 1903 | "mpath": { 1904 | "version": "0.9.0", 1905 | "resolved": "https://registry.npmjs.org/mpath/-/mpath-0.9.0.tgz", 1906 | "integrity": "sha512-ikJRQTk8hw5DEoFVxHG1Gn9T/xcjtdnOKIU1JTmGjZZlg9LST2mBLmcX3/ICIbgJydT2GOc15RnNy5mHmzfSew==" 1907 | }, 1908 | "mquery": { 1909 | "version": "4.0.3", 1910 | "resolved": "https://registry.npmjs.org/mquery/-/mquery-4.0.3.tgz", 1911 | "integrity": "sha512-J5heI+P08I6VJ2Ky3+33IpCdAvlYGTSUjwTPxkAr8i8EoduPMBX2OY/wa3IKZIQl7MU4SbFk8ndgSKyB/cl1zA==", 1912 | "requires": { 1913 | "debug": "4.x" 1914 | }, 1915 | "dependencies": { 1916 | "debug": { 1917 | "version": "4.3.4", 1918 | "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.4.tgz", 1919 | "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==", 1920 | "requires": { 1921 | "ms": "2.1.2" 1922 | } 1923 | }, 1924 | "ms": { 1925 | "version": "2.1.2", 1926 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", 1927 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==" 1928 | } 1929 | } 1930 | }, 1931 | "ms": { 1932 | "version": "2.0.0", 1933 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", 1934 | "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==" 1935 | }, 1936 | "negotiator": { 1937 | "version": "0.6.3", 1938 | "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.3.tgz", 1939 | "integrity": "sha512-+EUsqGPLsM+j/zdChZjsnX51g4XrHFOIXwfnCVPGlQk/k5giakcKsuxCObBRu6DSm9opw/O6slWbJdghQM4bBg==" 1940 | }, 1941 | "nodemon": { 1942 | "version": "2.0.19", 1943 | "resolved": "https://registry.npmjs.org/nodemon/-/nodemon-2.0.19.tgz", 1944 | "integrity": "sha512-4pv1f2bMDj0Eeg/MhGqxrtveeQ5/G/UVe9iO6uTZzjnRluSA4PVWf8CW99LUPwGB3eNIA7zUFoP77YuI7hOc0A==", 1945 | "dev": true, 1946 | "requires": { 1947 | "chokidar": "^3.5.2", 1948 | "debug": "^3.2.7", 1949 | "ignore-by-default": "^1.0.1", 1950 | "minimatch": "^3.0.4", 1951 | "pstree.remy": "^1.1.8", 1952 | "semver": "^5.7.1", 1953 | "simple-update-notifier": "^1.0.7", 1954 | "supports-color": "^5.5.0", 1955 | "touch": "^3.1.0", 1956 | "undefsafe": "^2.0.5" 1957 | }, 1958 | "dependencies": { 1959 | "debug": { 1960 | "version": "3.2.7", 1961 | "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", 1962 | "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", 1963 | "dev": true, 1964 | "requires": { 1965 | "ms": "^2.1.1" 1966 | } 1967 | }, 1968 | "ms": { 1969 | "version": "2.1.3", 1970 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", 1971 | "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", 1972 | "dev": true 1973 | } 1974 | } 1975 | }, 1976 | "nopt": { 1977 | "version": "1.0.10", 1978 | "resolved": "https://registry.npmjs.org/nopt/-/nopt-1.0.10.tgz", 1979 | "integrity": "sha512-NWmpvLSqUrgrAC9HCuxEvb+PSloHpqVu+FqcO4eeF2h5qYRhA7ev6KvelyQAKtegUbC6RypJnlEOhd8vloNKYg==", 1980 | "dev": true, 1981 | "requires": { 1982 | "abbrev": "1" 1983 | } 1984 | }, 1985 | "normalize-path": { 1986 | "version": "3.0.0", 1987 | "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", 1988 | "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==", 1989 | "dev": true 1990 | }, 1991 | "object-assign": { 1992 | "version": "4.1.1", 1993 | "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", 1994 | "integrity": "sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==" 1995 | }, 1996 | "object-inspect": { 1997 | "version": "1.12.2", 1998 | "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.12.2.tgz", 1999 | "integrity": "sha512-z+cPxW0QGUp0mcqcsgQyLVRDoXFQbXOwBaqyF7VIgI4TWNQsDHrBpUQslRmIfAoYWdYzs6UlKJtB2XJpTaNSpQ==" 2000 | }, 2001 | "on-finished": { 2002 | "version": "2.4.1", 2003 | "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.4.1.tgz", 2004 | "integrity": "sha512-oVlzkg3ENAhCk2zdv7IJwd/QUD4z2RxRwpkcGY8psCVcCYZNq4wYnVWALHM+brtuJjePWiYF/ClmuDr8Ch5+kg==", 2005 | "requires": { 2006 | "ee-first": "1.1.1" 2007 | } 2008 | }, 2009 | "parseurl": { 2010 | "version": "1.3.3", 2011 | "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz", 2012 | "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==" 2013 | }, 2014 | "path-to-regexp": { 2015 | "version": "0.1.7", 2016 | "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz", 2017 | "integrity": "sha512-5DFkuoqlv1uYQKxy8omFBeJPQcdoE07Kv2sferDCrAq1ohOU+MSDswDIbnx3YAM60qIOnYa53wBhXW0EbMonrQ==" 2018 | }, 2019 | "picomatch": { 2020 | "version": "2.3.1", 2021 | "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", 2022 | "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==", 2023 | "dev": true 2024 | }, 2025 | "proxy-addr": { 2026 | "version": "2.0.7", 2027 | "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.7.tgz", 2028 | "integrity": "sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg==", 2029 | "requires": { 2030 | "forwarded": "0.2.0", 2031 | "ipaddr.js": "1.9.1" 2032 | } 2033 | }, 2034 | "pstree.remy": { 2035 | "version": "1.1.8", 2036 | "resolved": "https://registry.npmjs.org/pstree.remy/-/pstree.remy-1.1.8.tgz", 2037 | "integrity": "sha512-77DZwxQmxKnu3aR542U+X8FypNzbfJ+C5XQDk3uWjWxn6151aIMGthWYRXTqT1E5oJvg+ljaa2OJi+VfvCOQ8w==", 2038 | "dev": true 2039 | }, 2040 | "punycode": { 2041 | "version": "2.1.1", 2042 | "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.1.1.tgz", 2043 | "integrity": "sha512-XRsRjdf+j5ml+y/6GKHPZbrF/8p2Yga0JPtdqTIY2Xe5ohJPD9saDJJLPvp9+NSBprVvevdXZybnj2cv8OEd0A==" 2044 | }, 2045 | "qs": { 2046 | "version": "6.10.3", 2047 | "resolved": "https://registry.npmjs.org/qs/-/qs-6.10.3.tgz", 2048 | "integrity": "sha512-wr7M2E0OFRfIfJZjKGieI8lBKb7fRCH4Fv5KNPEs7gJ8jadvotdsS08PzOKR7opXhZ/Xkjtt3WF9g38drmyRqQ==", 2049 | "requires": { 2050 | "side-channel": "^1.0.4" 2051 | } 2052 | }, 2053 | "range-parser": { 2054 | "version": "1.2.1", 2055 | "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz", 2056 | "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==" 2057 | }, 2058 | "raw-body": { 2059 | "version": "2.5.1", 2060 | "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.5.1.tgz", 2061 | "integrity": "sha512-qqJBtEyVgS0ZmPGdCFPWJ3FreoqvG4MVQln/kCgF7Olq95IbOp0/BWyMwbdtn4VTvkM8Y7khCQ2Xgk/tcrCXig==", 2062 | "requires": { 2063 | "bytes": "3.1.2", 2064 | "http-errors": "2.0.0", 2065 | "iconv-lite": "0.4.24", 2066 | "unpipe": "1.0.0" 2067 | } 2068 | }, 2069 | "readdirp": { 2070 | "version": "3.6.0", 2071 | "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.6.0.tgz", 2072 | "integrity": "sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==", 2073 | "dev": true, 2074 | "requires": { 2075 | "picomatch": "^2.2.1" 2076 | } 2077 | }, 2078 | "safe-buffer": { 2079 | "version": "5.2.1", 2080 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz", 2081 | "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==" 2082 | }, 2083 | "safer-buffer": { 2084 | "version": "2.1.2", 2085 | "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", 2086 | "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==" 2087 | }, 2088 | "saslprep": { 2089 | "version": "1.0.3", 2090 | "resolved": "https://registry.npmjs.org/saslprep/-/saslprep-1.0.3.tgz", 2091 | "integrity": "sha512-/MY/PEMbk2SuY5sScONwhUDsV2p77Znkb/q3nSVstq/yQzYJOH/Azh29p9oJLsl3LnQwSvZDKagDGBsBwSooag==", 2092 | "optional": true, 2093 | "requires": { 2094 | "sparse-bitfield": "^3.0.3" 2095 | } 2096 | }, 2097 | "semver": { 2098 | "version": "5.7.1", 2099 | "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", 2100 | "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", 2101 | "dev": true 2102 | }, 2103 | "send": { 2104 | "version": "0.18.0", 2105 | "resolved": "https://registry.npmjs.org/send/-/send-0.18.0.tgz", 2106 | "integrity": "sha512-qqWzuOjSFOuqPjFe4NOsMLafToQQwBSOEpS+FwEt3A2V3vKubTquT3vmLTQpFgMXp8AlFWFuP1qKaJZOtPpVXg==", 2107 | "requires": { 2108 | "debug": "2.6.9", 2109 | "depd": "2.0.0", 2110 | "destroy": "1.2.0", 2111 | "encodeurl": "~1.0.2", 2112 | "escape-html": "~1.0.3", 2113 | "etag": "~1.8.1", 2114 | "fresh": "0.5.2", 2115 | "http-errors": "2.0.0", 2116 | "mime": "1.6.0", 2117 | "ms": "2.1.3", 2118 | "on-finished": "2.4.1", 2119 | "range-parser": "~1.2.1", 2120 | "statuses": "2.0.1" 2121 | }, 2122 | "dependencies": { 2123 | "ms": { 2124 | "version": "2.1.3", 2125 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", 2126 | "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==" 2127 | } 2128 | } 2129 | }, 2130 | "serve-static": { 2131 | "version": "1.15.0", 2132 | "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.15.0.tgz", 2133 | "integrity": "sha512-XGuRDNjXUijsUL0vl6nSD7cwURuzEgglbOaFuZM9g3kwDXOWVTck0jLzjPzGD+TazWbboZYu52/9/XPdUgne9g==", 2134 | "requires": { 2135 | "encodeurl": "~1.0.2", 2136 | "escape-html": "~1.0.3", 2137 | "parseurl": "~1.3.3", 2138 | "send": "0.18.0" 2139 | } 2140 | }, 2141 | "setprototypeof": { 2142 | "version": "1.2.0", 2143 | "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.2.0.tgz", 2144 | "integrity": "sha512-E5LDX7Wrp85Kil5bhZv46j8jOeboKq5JMmYM3gVGdGH8xFpPWXUMsNrlODCrkoxMEeNi/XZIwuRvY4XNwYMJpw==" 2145 | }, 2146 | "side-channel": { 2147 | "version": "1.0.4", 2148 | "resolved": "https://registry.npmjs.org/side-channel/-/side-channel-1.0.4.tgz", 2149 | "integrity": "sha512-q5XPytqFEIKHkGdiMIrY10mvLRvnQh42/+GoBlFW3b2LXLE2xxJpZFdm94we0BaoV3RwJyGqg5wS7epxTv0Zvw==", 2150 | "requires": { 2151 | "call-bind": "^1.0.0", 2152 | "get-intrinsic": "^1.0.2", 2153 | "object-inspect": "^1.9.0" 2154 | } 2155 | }, 2156 | "sift": { 2157 | "version": "16.0.0", 2158 | "resolved": "https://registry.npmjs.org/sift/-/sift-16.0.0.tgz", 2159 | "integrity": "sha512-ILTjdP2Mv9V1kIxWMXeMTIRbOBrqKc4JAXmFMnFq3fKeyQ2Qwa3Dw1ubcye3vR+Y6ofA0b9gNDr/y2t6eUeIzQ==" 2160 | }, 2161 | "simple-update-notifier": { 2162 | "version": "1.0.7", 2163 | "resolved": "https://registry.npmjs.org/simple-update-notifier/-/simple-update-notifier-1.0.7.tgz", 2164 | "integrity": "sha512-BBKgR84BJQJm6WjWFMHgLVuo61FBDSj1z/xSFUIozqO6wO7ii0JxCqlIud7Enr/+LhlbNI0whErq96P2qHNWew==", 2165 | "dev": true, 2166 | "requires": { 2167 | "semver": "~7.0.0" 2168 | }, 2169 | "dependencies": { 2170 | "semver": { 2171 | "version": "7.0.0", 2172 | "resolved": "https://registry.npmjs.org/semver/-/semver-7.0.0.tgz", 2173 | "integrity": "sha512-+GB6zVA9LWh6zovYQLALHwv5rb2PHGlJi3lfiqIHxR0uuwCgefcOJc59v9fv1w8GbStwxuuqqAjI9NMAOOgq1A==", 2174 | "dev": true 2175 | } 2176 | } 2177 | }, 2178 | "smart-buffer": { 2179 | "version": "4.2.0", 2180 | "resolved": "https://registry.npmjs.org/smart-buffer/-/smart-buffer-4.2.0.tgz", 2181 | "integrity": "sha512-94hK0Hh8rPqQl2xXc3HsaBoOXKV20MToPkcXvwbISWLEs+64sBq5kFgn2kJDHb1Pry9yrP0dxrCI9RRci7RXKg==" 2182 | }, 2183 | "socks": { 2184 | "version": "2.7.0", 2185 | "resolved": "https://registry.npmjs.org/socks/-/socks-2.7.0.tgz", 2186 | "integrity": "sha512-scnOe9y4VuiNUULJN72GrM26BNOjVsfPXI+j+98PkyEfsIXroa5ofyjT+FzGvn/xHs73U2JtoBYAVx9Hl4quSA==", 2187 | "requires": { 2188 | "ip": "^2.0.0", 2189 | "smart-buffer": "^4.2.0" 2190 | } 2191 | }, 2192 | "sparse-bitfield": { 2193 | "version": "3.0.3", 2194 | "resolved": "https://registry.npmjs.org/sparse-bitfield/-/sparse-bitfield-3.0.3.tgz", 2195 | "integrity": "sha512-kvzhi7vqKTfkh0PZU+2D2PIllw2ymqJKujUcyPMd9Y75Nv4nPbGJZXNhxsgdQab2BmlDct1YnfQCguEvHr7VsQ==", 2196 | "optional": true, 2197 | "requires": { 2198 | "memory-pager": "^1.0.2" 2199 | } 2200 | }, 2201 | "statuses": { 2202 | "version": "2.0.1", 2203 | "resolved": "https://registry.npmjs.org/statuses/-/statuses-2.0.1.tgz", 2204 | "integrity": "sha512-RwNA9Z/7PrK06rYLIzFMlaF+l73iwpzsqRIFgbMLbTcLD6cOao82TaWefPXQvB2fOC4AjuYSEndS7N/mTCbkdQ==" 2205 | }, 2206 | "supports-color": { 2207 | "version": "5.5.0", 2208 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", 2209 | "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", 2210 | "dev": true, 2211 | "requires": { 2212 | "has-flag": "^3.0.0" 2213 | } 2214 | }, 2215 | "to-regex-range": { 2216 | "version": "5.0.1", 2217 | "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", 2218 | "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", 2219 | "dev": true, 2220 | "requires": { 2221 | "is-number": "^7.0.0" 2222 | } 2223 | }, 2224 | "toidentifier": { 2225 | "version": "1.0.1", 2226 | "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.1.tgz", 2227 | "integrity": "sha512-o5sSPKEkg/DIQNmH43V0/uerLrpzVedkUh8tGNvaeXpfpuwjKenlSox/2O/BTlZUtEe+JG7s5YhEz608PlAHRA==" 2228 | }, 2229 | "touch": { 2230 | "version": "3.1.0", 2231 | "resolved": "https://registry.npmjs.org/touch/-/touch-3.1.0.tgz", 2232 | "integrity": "sha512-WBx8Uy5TLtOSRtIq+M03/sKDrXCLHxwDcquSP2c43Le03/9serjQBIztjRz6FkJez9D/hleyAXTBGLwwZUw9lA==", 2233 | "dev": true, 2234 | "requires": { 2235 | "nopt": "~1.0.10" 2236 | } 2237 | }, 2238 | "tr46": { 2239 | "version": "3.0.0", 2240 | "resolved": "https://registry.npmjs.org/tr46/-/tr46-3.0.0.tgz", 2241 | "integrity": "sha512-l7FvfAHlcmulp8kr+flpQZmVwtu7nfRV7NZujtN0OqES8EL4O4e0qqzL0DC5gAvx/ZC/9lk6rhcUwYvkBnBnYA==", 2242 | "requires": { 2243 | "punycode": "^2.1.1" 2244 | } 2245 | }, 2246 | "type-is": { 2247 | "version": "1.6.18", 2248 | "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz", 2249 | "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==", 2250 | "requires": { 2251 | "media-typer": "0.3.0", 2252 | "mime-types": "~2.1.24" 2253 | } 2254 | }, 2255 | "undefsafe": { 2256 | "version": "2.0.5", 2257 | "resolved": "https://registry.npmjs.org/undefsafe/-/undefsafe-2.0.5.tgz", 2258 | "integrity": "sha512-WxONCrssBM8TSPRqN5EmsjVrsv4A8X12J4ArBiiayv3DyyG3ZlIg6yysuuSYdZsVz3TKcTg2fd//Ujd4CHV1iA==", 2259 | "dev": true 2260 | }, 2261 | "unpipe": { 2262 | "version": "1.0.0", 2263 | "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", 2264 | "integrity": "sha512-pjy2bYhSsufwWlKwPc+l3cN7+wuJlK6uz0YdJEOlQDbl6jo/YlPi4mb8agUkVC8BF7V8NuzeyPNqRksA3hztKQ==" 2265 | }, 2266 | "utils-merge": { 2267 | "version": "1.0.1", 2268 | "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz", 2269 | "integrity": "sha512-pMZTvIkT1d+TFGvDOqodOclx0QWkkgi6Tdoa8gC8ffGAAqz9pzPTZWAybbsHHoED/ztMtkv/VoYTYyShUn81hA==" 2270 | }, 2271 | "vary": { 2272 | "version": "1.1.2", 2273 | "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz", 2274 | "integrity": "sha512-BNGbWLfd0eUPabhkXUVm0j8uuvREyTh5ovRa/dyow/BqAbZJyC+5fU+IzQOzmAKzYqYRAISoRhdQr3eIZ/PXqg==" 2275 | }, 2276 | "webidl-conversions": { 2277 | "version": "7.0.0", 2278 | "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-7.0.0.tgz", 2279 | "integrity": "sha512-VwddBukDzu71offAQR975unBIGqfKZpM+8ZX6ySk8nYhVoo5CYaZyzt3YBvYtRtO+aoGlqxPg/B87NGVZ/fu6g==" 2280 | }, 2281 | "whatwg-url": { 2282 | "version": "11.0.0", 2283 | "resolved": "https://registry.npmjs.org/whatwg-url/-/whatwg-url-11.0.0.tgz", 2284 | "integrity": "sha512-RKT8HExMpoYx4igMiVMY83lN6UeITKJlBQ+vR/8ZJ8OCdSiN3RwCq+9gH0+Xzj0+5IrM6i4j/6LuvzbZIQgEcQ==", 2285 | "requires": { 2286 | "tr46": "^3.0.0", 2287 | "webidl-conversions": "^7.0.0" 2288 | } 2289 | } 2290 | } 2291 | } 2292 | -------------------------------------------------------------------------------- /backend/package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "backend", 3 | "version": "1.0.0", 4 | "description": "", 5 | "main": "index.js", 6 | "type": "module", 7 | "scripts": { 8 | "serve": "nodemon server.js" 9 | }, 10 | "keywords": [], 11 | "author": "", 12 | "license": "ISC", 13 | "dependencies": { 14 | "cors": "^2.8.5", 15 | "dotenv": "^16.0.2", 16 | "express": "^4.18.1", 17 | "joi": "^17.6.0", 18 | "mongoose": "^6.6.0" 19 | }, 20 | "devDependencies": { 21 | "nodemon": "^2.0.19" 22 | } 23 | } -------------------------------------------------------------------------------- /backend/routes/index.js: -------------------------------------------------------------------------------- 1 | import express from 'express'; 2 | import joi from 'joi'; 3 | import mongoose from 'mongoose'; 4 | import Project from '../models/index.js' 5 | 6 | const api = express.Router() 7 | 8 | api.get('/projects', async (req, res) => { 9 | try { 10 | const data = await Project.find({}, { task: 0, __v: 0, updatedAt: 0 }) 11 | return res.send(data) 12 | } catch (error) { 13 | return res.send(error) 14 | } 15 | }) 16 | 17 | api.get('/project/:id', async (req, res) => { 18 | if (!req.params.id) res.status(422).send({ data: { error: true, message: 'Id is reaquire' } }) 19 | try { 20 | const data = await Project.find({ _id: mongoose.Types.ObjectId(req.params.id) }).sort({ order: 1 }) 21 | return res.send(data) 22 | } catch (error) { 23 | return res.send(error) 24 | } 25 | }) 26 | 27 | api.post('/project', async (req, res) => { 28 | 29 | // validate type 30 | const project = joi.object({ 31 | title: joi.string().min(3).max(30).required(), 32 | description: joi.string().required(), 33 | }) 34 | 35 | // validation 36 | const { error, value } = project.validate({ title: req.body.title, description: req.body.description }); 37 | if (error) return res.status(422).send(error) 38 | 39 | 40 | // insert data 41 | try { 42 | const data = await new Project(value).save() 43 | res.send({ data: { title: data.title, description: data.description, updatedAt: data.updatedAt, _id: data._id } }) 44 | 45 | } catch (e) { 46 | if (e.code === 11000) { 47 | return res.status(422).send({ data: { error: true, message: 'title must be unique' } }) 48 | } else { 49 | return res.status(500).send({ data: { error: true, message: 'server error' } }) 50 | } 51 | } 52 | 53 | 54 | }) 55 | 56 | api.put('/project/:id', async (req, res) => { 57 | // validate type 58 | const project = joi.object({ 59 | title: joi.string().min(3).max(30).required(), 60 | description: joi.string().required(), 61 | }) 62 | 63 | // // validation 64 | const { error, value } = project.validate({ title: req.body.title, description: req.body.description }); 65 | if (error) return res.status(422).send(error) 66 | 67 | Project.updateOne({ _id: mongoose.Types.ObjectId(req.params.id) }, { ...value }, { upsert: true }, (error, data) => { 68 | if (error) { 69 | res.send(error) 70 | } else { 71 | res.send(data) 72 | } 73 | }) 74 | 75 | 76 | }) 77 | 78 | api.delete('/project/:id', async (req, res) => { 79 | try { 80 | const data = await Project.deleteOne({ _id: mongoose.Types.ObjectId(req.params.id) }) 81 | res.send(data) 82 | } catch (error) { 83 | res.send(error) 84 | } 85 | 86 | }) 87 | 88 | 89 | // task api 90 | 91 | api.post('/project/:id/task', async (req, res) => { 92 | 93 | 94 | if (!req.params.id) return res.status(500).send(`server error`); 95 | 96 | // validate type 97 | const task = joi.object({ 98 | title: joi.string().min(3).max(30).required(), 99 | description: joi.string().required(), 100 | }) 101 | 102 | const { error, value } = task.validate({ title: req.body.title, description: req.body.description }); 103 | if (error) return res.status(422).send(error) 104 | 105 | try { 106 | // const task = await Project.find({ _id: mongoose.Types.ObjectId(req.params.id) }, { "task.index": 1 }) 107 | const [{ task }] = await Project.find({ _id: mongoose.Types.ObjectId(req.params.id) }, { "task.index": 1 }).sort({ 'task.index': 1 }) 108 | 109 | 110 | let countTaskLength = [task.length, task.length > 0 ? Math.max(...task.map(o => o.index)) : task.length]; 111 | 112 | const data = await Project.updateOne({ _id: mongoose.Types.ObjectId(req.params.id) }, { $push: { task: { ...value, stage: "Requested", order: countTaskLength[0], index: countTaskLength[1] + 1 } } }) 113 | return res.send(data) 114 | } catch (error) { 115 | return res.status(500).send(error) 116 | } 117 | }) 118 | 119 | api.get('/project/:id/task/:taskId', async (req, res) => { 120 | 121 | if (!req.params.id || !req.params.taskId) return res.status(500).send(`server error`); 122 | 123 | // res.send(req.params) 124 | try { 125 | 126 | let data = await Project.find( 127 | { _id: mongoose.Types.ObjectId(req.params.id) }, 128 | { 129 | task: { 130 | $filter: { 131 | input: "$task", 132 | as: "task", 133 | cond: { 134 | $in: [ 135 | "$$task._id", 136 | [ 137 | mongoose.Types.ObjectId(req.params.taskId) 138 | ] 139 | ] 140 | } 141 | } 142 | } 143 | }) 144 | if (data[0].task.length < 1) return res.status(404).send({ error: true, message: 'record not found' }) 145 | return res.send(data) 146 | } catch (error) { 147 | return res.status(5000).send(error) 148 | } 149 | 150 | 151 | }) 152 | 153 | 154 | api.put('/project/:id/task/:taskId', async (req, res) => { 155 | 156 | if (!req.params.id || !req.params.taskId) return res.status(500).send(`server error`); 157 | 158 | const task = joi.object({ 159 | title: joi.string().min(3).max(30).required(), 160 | description: joi.string().required(), 161 | }) 162 | 163 | const { error, value } = task.validate({ title: req.body.title, description: req.body.description }); 164 | if (error) return res.status(422).send(error) 165 | 166 | try { 167 | // const data = await Project.find({ $and: [{ _id: mongoose.Types.ObjectId(req.params.id) }, { "task._id": mongoose.Types.ObjectId(req.params.taskId) }] },{ 168 | // task: { 169 | // $filter: { 170 | // input: "$task", 171 | // as: "task", 172 | // cond: { 173 | // $in: [ 174 | // "$$task._id", 175 | // [ 176 | // mongoose.Types.ObjectId(req.params.taskId) 177 | // ] 178 | // ] 179 | // } 180 | // } 181 | // } 182 | // }) 183 | const data = await Project.updateOne({ 184 | _id: mongoose.Types.ObjectId(req.params.id), 185 | task: { $elemMatch: { _id: mongoose.Types.ObjectId(req.params.taskId) } } 186 | }, { $set: { "task.$.title": value.title, "task.$.description": value.description } }) 187 | return res.send(data) 188 | } catch (error) { 189 | return res.send(error) 190 | } 191 | 192 | }) 193 | 194 | api.delete('/project/:id/task/:taskId', async (req, res) => { 195 | 196 | if (!req.params.id || !req.params.taskId) return res.status(500).send(`server error`); 197 | 198 | try { 199 | const data = await Project.updateOne({ _id: mongoose.Types.ObjectId(req.params.id) }, { $pull: { task: { _id: mongoose.Types.ObjectId(req.params.taskId) } } }) 200 | return res.send(data) 201 | } catch (error) { 202 | return res.send(error) 203 | } 204 | 205 | }) 206 | 207 | api.put('/project/:id/todo', async (req, res) => { 208 | let todo = [] 209 | 210 | for (const key in req.body) { 211 | // todo.push({ items: req.body[key].items, name: req.body[key]?.name }) 212 | for (const index in req.body[key].items) { 213 | req.body[key].items[index].stage = req.body[key].name 214 | todo.push({ name: req.body[key].items[index]._id, stage: req.body[key].items[index].stage, order: index }) 215 | } 216 | } 217 | 218 | todo.map(async (item) => { 219 | await Project.updateOne({ 220 | _id: mongoose.Types.ObjectId(req.params.id), 221 | task: { $elemMatch: { _id: mongoose.Types.ObjectId(item.name) } } 222 | }, { $set: { "task.$.order": item.order, "task.$.stage": item.stage } }) 223 | }) 224 | 225 | res.send(todo) 226 | }) 227 | 228 | // api.use('/project/:id/task', async (req, res, next) => { 229 | // if (req.method !== "GET") return next() 230 | 231 | // if (!req.params.id) return res.status(500).send(`server error`); 232 | 233 | // try { 234 | // const data = await Project.find({ _id: mongoose.Types.ObjectId(req.params.id) }, { task: 1 }) 235 | // return res.send(data) 236 | // } catch (error) { 237 | // return res.send(error) 238 | // } 239 | 240 | 241 | // }) 242 | 243 | // api.get('/project/:id/task/:taskId', (req, res) => { 244 | // res.send(req.params) 245 | // }) 246 | 247 | 248 | 249 | export default api -------------------------------------------------------------------------------- /backend/server.js: -------------------------------------------------------------------------------- 1 | import express from "express"; 2 | import api from './routes/index.js' 3 | import dotenv from 'dotenv' 4 | import mongoose from "mongoose"; 5 | import cors from "cors"; 6 | 7 | dotenv.config() 8 | mongoose.connect(process.env.MONGODB_PATH, () => { 9 | console.log('connect'); 10 | }, (e) => console.log(e)) 11 | 12 | 13 | const PORT = process.env.SERVER_PORT || 9000 14 | const origin = process.env.CORS_ORIGIN || 'http://localhost:3000' 15 | 16 | const app = express() 17 | 18 | app.use(cors({ 19 | origin 20 | })); 21 | app.use(express.json()) 22 | app.use(express.urlencoded()) 23 | 24 | app.use(api) 25 | 26 | app.listen(PORT, () => { 27 | console.log(`Your app is running in http://localhost:${PORT}`) 28 | }) -------------------------------------------------------------------------------- /frontend/.vscode/settings.json: -------------------------------------------------------------------------------- 1 | { 2 | "cSpell.words": [ 3 | "Locationwindow", 4 | "susseccduly" 5 | ] 6 | } -------------------------------------------------------------------------------- /frontend/package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "frontend", 3 | "version": "0.1.0", 4 | "private": true, 5 | "dependencies": { 6 | "@headlessui/react": "^1.7.1", 7 | "@testing-library/jest-dom": "^5.16.5", 8 | "@testing-library/react": "^13.4.0", 9 | "@testing-library/user-event": "^13.5.0", 10 | "axios": "^0.27.2", 11 | "react": "^18.2.0", 12 | "react-beautiful-dnd": "^13.1.1", 13 | "react-dom": "^18.2.0", 14 | "react-hot-toast": "^2.4.0", 15 | "react-router-dom": "^6.3.0", 16 | "react-scripts": "5.0.1", 17 | "uuid": "^9.0.0", 18 | "web-vitals": "^2.1.4" 19 | }, 20 | "scripts": { 21 | "start": "react-scripts start", 22 | "build": "react-scripts build", 23 | "test": "react-scripts test", 24 | "eject": "react-scripts eject" 25 | }, 26 | "eslintConfig": { 27 | "extends": [ 28 | "react-app", 29 | "react-app/jest" 30 | ] 31 | }, 32 | "browserslist": { 33 | "production": [ 34 | ">0.2%", 35 | "not dead", 36 | "not op_mini all" 37 | ], 38 | "development": [ 39 | "last 1 chrome version", 40 | "last 1 firefox version", 41 | "last 1 safari version" 42 | ] 43 | }, 44 | "devDependencies": { 45 | "autoprefixer": "^10.4.9", 46 | "postcss": "^8.4.16", 47 | "tailwindcss": "^3.1.8" 48 | } 49 | } 50 | -------------------------------------------------------------------------------- /frontend/postcss.config.js: -------------------------------------------------------------------------------- 1 | module.exports = { 2 | plugins: { 3 | tailwindcss: {}, 4 | autoprefixer: {}, 5 | }, 6 | } 7 | -------------------------------------------------------------------------------- /frontend/public/favicon.ico: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/devravimori/project-menagment-system-in-MERN-stack/e7442711fca42b38b3c169ac30d427d7403d723f/frontend/public/favicon.ico -------------------------------------------------------------------------------- /frontend/public/image/attachment.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/devravimori/project-menagment-system-in-MERN-stack/e7442711fca42b38b3c169ac30d427d7403d723f/frontend/public/image/attachment.jpg -------------------------------------------------------------------------------- /frontend/public/image/welcome.svg: -------------------------------------------------------------------------------- 1 | -------------------------------------------------------------------------------- /frontend/public/index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 12 | 13 | 17 | 18 | 27 | React App 28 | 29 | 30 | 31 |
32 | 42 | 43 | 44 | -------------------------------------------------------------------------------- /frontend/public/logo192.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/devravimori/project-menagment-system-in-MERN-stack/e7442711fca42b38b3c169ac30d427d7403d723f/frontend/public/logo192.png -------------------------------------------------------------------------------- /frontend/public/logo512.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/devravimori/project-menagment-system-in-MERN-stack/e7442711fca42b38b3c169ac30d427d7403d723f/frontend/public/logo512.png -------------------------------------------------------------------------------- /frontend/public/manifest.json: -------------------------------------------------------------------------------- 1 | { 2 | "short_name": "React App", 3 | "name": "Create React App Sample", 4 | "icons": [ 5 | { 6 | "src": "favicon.ico", 7 | "sizes": "64x64 32x32 24x24 16x16", 8 | "type": "image/x-icon" 9 | }, 10 | { 11 | "src": "logo192.png", 12 | "type": "image/png", 13 | "sizes": "192x192" 14 | }, 15 | { 16 | "src": "logo512.png", 17 | "type": "image/png", 18 | "sizes": "512x512" 19 | } 20 | ], 21 | "start_url": ".", 22 | "display": "standalone", 23 | "theme_color": "#000000", 24 | "background_color": "#ffffff" 25 | } 26 | -------------------------------------------------------------------------------- /frontend/public/robots.txt: -------------------------------------------------------------------------------- 1 | # https://www.robotstxt.org/robotstxt.html 2 | User-agent: * 3 | Disallow: 4 | -------------------------------------------------------------------------------- /frontend/src/App.js: -------------------------------------------------------------------------------- 1 | import AppLayout from "./components/AppLayout"; 2 | import { Routes, Route } from "react-router-dom"; 3 | import Task from "./components/Task"; 4 | import { Toaster } from "react-hot-toast"; 5 | function App() { 6 | console.log('render app..') 7 | return ( 8 | 9 | 13 | 14 | } /> 15 | 17 | 18 |

Select or create new project

19 | 20 | } /> 21 |
22 |
23 | ); 24 | } 25 | 26 | export default App; 27 | -------------------------------------------------------------------------------- /frontend/src/App.test.js: -------------------------------------------------------------------------------- 1 | import { render, screen } from '@testing-library/react'; 2 | import App from './App'; 3 | 4 | test('renders learn react link', () => { 5 | render(); 6 | const linkElement = screen.getByText(/learn react/i); 7 | expect(linkElement).toBeInTheDocument(); 8 | }); 9 | -------------------------------------------------------------------------------- /frontend/src/components/AddProjectModal.js: -------------------------------------------------------------------------------- 1 | import React, { Fragment, memo, useEffect, useState } from 'react' 2 | import { Dialog, Transition } from '@headlessui/react' 3 | import BtnPrimary from './BtnPrimary' 4 | import BtnSecondary from './BtnSecondary' 5 | import axios from "axios" 6 | import toast from 'react-hot-toast' 7 | 8 | const AddProjectModal = ({ isModalOpen, closeModal, edit = false, id = null }) => { 9 | 10 | const [title, setTitle] = useState('') 11 | const [desc, setDesc] = useState(''); 12 | 13 | useEffect(() => { 14 | if (edit && isModalOpen) { 15 | axios.get(`http://localhost:9000/project/${id}`) 16 | .then((res) => { 17 | setTitle(res.data[0].title) 18 | setDesc(res.data[0].description) 19 | }) 20 | .catch((error) => { 21 | toast.error('Something went wrong') 22 | }) 23 | } 24 | }, [isModalOpen]); 25 | 26 | 27 | const handleSubmit = (e) => { 28 | e.preventDefault() 29 | if (!edit) { 30 | axios.post('http://localhost:9000/project/', { title, description: desc }) 31 | .then((res) => { 32 | closeModal() 33 | const customEvent = new CustomEvent('projectUpdate', { detail: { ...res.data } }); 34 | document.dispatchEvent(customEvent); 35 | toast.success('Project created successfully') 36 | setTitle('') 37 | setDesc('') 38 | }) 39 | .catch((error) => { 40 | if (error.response.status === 422) { 41 | toast.error(error.response.data.details[0].message) 42 | } else { 43 | toast.error('Something went wrong') 44 | } 45 | }) 46 | } else { 47 | axios.put(`http://localhost:9000/project/${id}`, { title, description: desc }) 48 | .then((res) => { 49 | closeModal() 50 | const customEvent = new CustomEvent('projectUpdate', { detail: { ...res.data } }); 51 | document.dispatchEvent(customEvent); 52 | toast.success('Project updated successfully') 53 | setTitle('') 54 | setDesc('') 55 | }) 56 | .catch((error) => { 57 | if (error.response.status === 422) { 58 | toast.error(error.response.data.details[0].message) 59 | } else { 60 | toast.error('Something went wrong') 61 | } 62 | }) 63 | } 64 | 65 | } 66 | 67 | return ( 68 | 69 | closeModal()} className="relative z-50"> 70 |
71 | 80 |
81 | 82 |
83 | {/*
*/} 84 | 93 | 94 | 95 | 96 | {edit ? (

Edit Project

) : (

Create Project

)} 97 | 102 |
103 |
104 |
105 | 106 | setTitle(e.target.value)} type="text" className='border border-gray-300 rounded-md w-full text-sm py-2 px-2.5 focus:border-indigo-500 focus:outline-offset-1 focus:outline-indigo-400' placeholder='Project title' /> 107 |
108 |
109 | 110 | 111 |
112 |
113 | closeModal()}>Cancel 114 | Save 115 |
116 |
117 | 118 |
119 |
120 | 121 |
122 |
123 |
124 |
125 | ) 126 | } 127 | 128 | export default memo(AddProjectModal) -------------------------------------------------------------------------------- /frontend/src/components/AddTaskModal.js: -------------------------------------------------------------------------------- 1 | import React, { Fragment, useEffect, useState } from 'react' 2 | import { Dialog, Transition } from '@headlessui/react' 3 | import BtnPrimary from './BtnPrimary' 4 | import BtnSecondary from './BtnSecondary' 5 | import axios from 'axios' 6 | import toast from 'react-hot-toast' 7 | 8 | const AddTaskModal = ({ isAddTaskModalOpen, setAddTaskModal, projectId = null, taskId = null, edit = false, refreshData }) => { 9 | 10 | const [title, setTitle] = useState('') 11 | const [desc, setDesc] = useState(''); 12 | 13 | useEffect(() => { 14 | if (edit && isAddTaskModalOpen) { 15 | axios.get(`http://localhost:9000/project/${projectId}/task/${taskId}`) 16 | .then((res) => { 17 | setTitle(res.data[0].task[0].title) 18 | setDesc(res.data[0].task[0].description) 19 | }) 20 | .catch((error) => { 21 | toast.error('Something went wrong') 22 | }) 23 | console.log('edit function call') 24 | } 25 | }, [isAddTaskModalOpen]); 26 | 27 | const handleSubmit = (e) => { 28 | e.preventDefault() 29 | if (!edit) { 30 | axios.post(`http://localhost:9000/project/${projectId}/task`, { title, description: desc }) 31 | .then((res) => { 32 | setAddTaskModal(false) 33 | toast.success('Task created successfully') 34 | setTitle('') 35 | setDesc('') 36 | }) 37 | .catch((error) => { 38 | if (error.response.status === 422) { 39 | toast.error(error.response.data.details[0].message) 40 | } else { 41 | toast.error('Something went wrong') 42 | } 43 | }) 44 | } else { 45 | axios.put(`http://localhost:9000/project/${projectId}/task/${taskId}`, { title, description: desc }) 46 | .then((res) => { 47 | setAddTaskModal(false) 48 | toast.success('Task is updated') 49 | refreshData(true) 50 | setTitle('') 51 | setDesc('') 52 | }) 53 | .catch((error) => { 54 | if (error.response.status === 422) { 55 | toast.error(error.response.data.details[0].message) 56 | } else { 57 | toast.error('Something went wrong') 58 | } 59 | }) 60 | } 61 | } 62 | 63 | return ( 64 | 65 | setAddTaskModal(false)} className="relative z-50"> 66 |
67 | 76 |
77 | 78 |
79 | {/*
*/} 80 | 89 | 90 | 91 | 92 | {!edit ? (

Add Task

) : (

Edit Task

)} 93 | 98 |
99 |
100 |
101 | 102 | setTitle(e.target.value)} type="text" className='border border-gray-300 rounded-md w-full text-sm py-2 px-2.5 focus:border-indigo-500 focus:outline-offset-1 focus:outline-indigo-400' placeholder='Task title' /> 103 |
104 |
105 | 106 | 107 |
108 |
109 | setAddTaskModal(false)}>Cancel 110 | Save 111 |
112 |
113 | 114 |
115 |
116 | 117 |
118 |
119 |
120 |
121 | ) 122 | } 123 | 124 | export default AddTaskModal -------------------------------------------------------------------------------- /frontend/src/components/AppLayout.js: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | import Navbar from './Navbar' 3 | import Sidebar from './Sidebar' 4 | 5 | const AppLayout = ({ children }) => { 6 | return ( 7 |
8 | 9 |
10 |
11 | 12 |
13 |
14 |
15 | {children} 16 |
17 |
18 |
19 |
20 | ) 21 | } 22 | 23 | export default AppLayout -------------------------------------------------------------------------------- /frontend/src/components/BtnPrimary.js: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | 3 | const BtnPrimary = (props) => { 4 | return ( 5 | 8 | ) 9 | } 10 | 11 | export default BtnPrimary; -------------------------------------------------------------------------------- /frontend/src/components/BtnSecondary.js: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | 3 | const BtnSecondary = (props) => { 4 | return ( 5 | 6 | ) 7 | } 8 | 9 | export default BtnSecondary -------------------------------------------------------------------------------- /frontend/src/components/DropdownMenu.js: -------------------------------------------------------------------------------- 1 | import React, { useState } from 'react' 2 | import { Popover } from '@headlessui/react' 3 | import AddTaskModal from './AddTaskModal' 4 | 5 | const DropdownMenu = ({ taskId, handleDelete, projectId, setRenderChange }) => { 6 | const [isEditTaskModalOpen, setEditTaskModal] = useState(false); 7 | 8 | const refreshData = () => { 9 | setRenderChange(true) 10 | } 11 | const handleSetEditModal = (e) => { 12 | e.stopPropagation(); 13 | setEditTaskModal(true) 14 | } 15 | 16 | return ( 17 | <> 18 | 19 | 20 | 21 | 22 | 23 |
24 |
25 | 29 | {/* */} 33 |
34 |
35 | 45 |
46 |
47 |
48 |
49 | 50 | 51 | ) 52 | } 53 | 54 | export default DropdownMenu 55 | -------------------------------------------------------------------------------- /frontend/src/components/EditProjectModal.js: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | 3 | const EditProjectModal = () => { 4 | return ( 5 |
EditProjectModal
6 | ) 7 | } 8 | 9 | export default EditProjectModal -------------------------------------------------------------------------------- /frontend/src/components/Navbar.js: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | 3 | const Navbar = () => { 4 | return ( 5 |
6 | ) 7 | } 8 | 9 | export default Navbar -------------------------------------------------------------------------------- /frontend/src/components/ProjectDropdown.js: -------------------------------------------------------------------------------- 1 | import React, { useState } from 'react' 2 | import { Popover } from '@headlessui/react' 3 | import axios from "axios" 4 | import toast from 'react-hot-toast' 5 | import AddProjectModal from './AddProjectModal' 6 | 7 | const ProjectDropdown = ({ id, navigate }) => { 8 | const [isModalOpen, setModalState] = useState(false) 9 | const handleDelete = async () => { 10 | try { 11 | const data = await axios.delete(`http://localhost:9000/project/${id}`) 12 | if (data.data.deletedCount > 0) { 13 | toast.success('Record deleted successfully') 14 | navigate('/') 15 | const customEvent = new CustomEvent('projectUpdate'); 16 | document.dispatchEvent(customEvent); 17 | } else { 18 | toast.error('Record is already deleted') 19 | } 20 | 21 | } catch (e) { 22 | toast.error('Something went wrong') 23 | } 24 | 25 | } 26 | 27 | const closeModal = () => { 28 | setModalState(false) 29 | } 30 | 31 | 32 | return ( 33 | <> 34 | 35 | 36 | 37 | 38 | 39 |
40 |
41 | 45 | {/* */} 49 |
50 |
51 | 61 |
62 |
63 |
64 |
65 | 66 | 67 | ) 68 | } 69 | 70 | export default ProjectDropdown -------------------------------------------------------------------------------- /frontend/src/components/Sidebar.js: -------------------------------------------------------------------------------- 1 | import React, { useCallback, useEffect, useState } from 'react' 2 | import AddProjectModal from './AddProjectModal' 3 | import axios from 'axios' 4 | import { Link } from 'react-router-dom' 5 | // import toast from 'react-hot-toast' 6 | const Sidebar = () => { 7 | 8 | const [isModalOpen, setModalState] = useState(false) 9 | const [projects, setProjects] = useState([]) 10 | const [paramsWindow, setParamsWindow] = useState(window.location.pathname.slice(1)) 11 | useEffect(() => { 12 | }) 13 | 14 | const handleLocation = (e) => { 15 | setParamsWindow(new URL(e.currentTarget.href).pathname.slice(1)) 16 | } 17 | 18 | const openModal = useCallback(() => { 19 | setModalState(true) 20 | }, []) 21 | 22 | const closeModal = useCallback(() => { 23 | setModalState(false) 24 | }, []) 25 | 26 | const projectData = () => { 27 | axios.get('http://localhost:9000/projects/') 28 | .then((res) => { 29 | setProjects(res.data) 30 | }) 31 | } 32 | 33 | useEffect(() => { 34 | projectData() 35 | document.addEventListener('projectUpdate', ({ detail }) => { 36 | projectData() 37 | }) 38 | return () => { 39 | document.removeEventListener('projectUpdate', {}, false) 40 | } 41 | }, []); 42 | 43 | 44 | 45 | return ( 46 |
47 |
48 | Projects 49 | 54 |
55 |
    56 | {projects.map((project, index) => ( 57 | handleLocation(e)}> 58 |
  • 59 | {project.title} 60 |
  • 61 | 62 | ))} 63 |
64 | 65 |
66 | ) 67 | } 68 | 69 | export default Sidebar -------------------------------------------------------------------------------- /frontend/src/components/Task.js: -------------------------------------------------------------------------------- 1 | import React, { useEffect, useState } from "react"; 2 | import { DragDropContext, Draggable, Droppable } from "react-beautiful-dnd"; 3 | import { v4 as uuid } from "uuid"; 4 | import AddTaskModal from "./AddTaskModal"; 5 | import BtnPrimary from './BtnPrimary' 6 | import DropdownMenu from "./DropdownMenu"; 7 | // import TaskModal from "./TaskModal"; 8 | import { useParams, useNavigate } from "react-router"; 9 | import ProjectDropdown from "./ProjectDropdown" 10 | import axios from "axios"; 11 | import toast from "react-hot-toast"; 12 | import TaskModal from "./TaskModal"; 13 | 14 | 15 | function Task() { 16 | 17 | // const itemsFromBackend = [ 18 | // { _id: uuid(), content: "First task" }, 19 | // { _id: uuid(), content: "Second task" }, 20 | // { _id: uuid(), content: "Third task" }, 21 | // { _id: uuid(), content: "Forth task" } 22 | // ]; 23 | 24 | // const columnsFromBackend = { 25 | // [uuid()]: { 26 | // name: "Requested", 27 | // items: [] 28 | // }, 29 | // [uuid()]: { 30 | // name: "To do", 31 | // items: [] 32 | // }, 33 | // [uuid()]: { 34 | // name: "In Progress", 35 | // items: [] 36 | // }, 37 | // [uuid()]: { 38 | // name: "Done", 39 | // items: [] 40 | // } 41 | // }; 42 | 43 | const onDragEnd = (result, columns, setColumns) => { 44 | if (!result.destination) return; 45 | const { source, destination } = result; 46 | let data = {} 47 | if (source.droppableId !== destination.droppableId) { 48 | const sourceColumn = columns[source.droppableId]; 49 | const destColumn = columns[destination.droppableId]; 50 | const sourceItems = [...sourceColumn.items]; 51 | const destItems = [...destColumn.items]; 52 | const [removed] = sourceItems.splice(source.index, 1); 53 | destItems.splice(destination.index, 0, removed); 54 | 55 | setColumns({ 56 | ...columns, 57 | [source.droppableId]: { 58 | ...sourceColumn, 59 | items: sourceItems 60 | }, 61 | [destination.droppableId]: { 62 | ...destColumn, 63 | items: destItems 64 | } 65 | }); 66 | data = { 67 | ...columns, 68 | [source.droppableId]: { 69 | ...sourceColumn, 70 | items: sourceItems 71 | }, 72 | [destination.droppableId]: { 73 | ...destColumn, 74 | items: destItems 75 | } 76 | } 77 | } else { 78 | const column = columns[source.droppableId]; 79 | const copiedItems = [...column.items]; 80 | const [removed] = copiedItems.splice(source.index, 1); 81 | copiedItems.splice(destination.index, 0, removed); 82 | setColumns({ 83 | ...columns, 84 | [source.droppableId]: { 85 | ...column, 86 | items: copiedItems 87 | } 88 | }); 89 | data = { 90 | ...columns, 91 | [source.droppableId]: { 92 | ...column, 93 | items: copiedItems 94 | } 95 | } 96 | 97 | } 98 | 99 | updateTodo(data) 100 | }; 101 | 102 | const [isAddTaskModalOpen, setAddTaskModal] = useState(false); 103 | 104 | // const [columns, setColumns] = useState(columnsFromBackend); 105 | const [columns, setColumns] = useState({}); 106 | const [isRenderChange, setRenderChange] = useState(false); 107 | const [isTaskOpen, setTaskOpen] = useState(false); 108 | const [taskId, setTaskId] = useState(false); 109 | const [title, setTitle] = useState(''); 110 | const { projectId } = useParams(); 111 | const navigate = useNavigate(); 112 | 113 | useEffect(() => { 114 | if (!isAddTaskModalOpen || isRenderChange) { 115 | axios.get(`http://localhost:9000/project/${projectId}`) 116 | .then((res) => { 117 | setTitle(res.data[0].title) 118 | setColumns({ 119 | [uuid()]: { 120 | name: "Requested", 121 | items: res.data[0].task.filter((task) => task.stage === "Requested").sort((a, b) => { 122 | return a.order - b.order; 123 | }) 124 | }, 125 | [uuid()]: { 126 | name: "To do", 127 | items: res.data[0].task.filter((task) => task.stage === "To do").sort((a, b) => { 128 | return a.order - b.order; 129 | }) 130 | }, 131 | [uuid()]: { 132 | name: "In Progress", 133 | items: res.data[0].task.filter((task) => task.stage === "In Progress").sort((a, b) => { 134 | return a.order - b.order; 135 | }) 136 | }, 137 | [uuid()]: { 138 | name: "Done", 139 | items: res.data[0].task.filter((task) => task.stage === "Done").sort((a, b) => { 140 | return a.order - b.order; 141 | }) 142 | } 143 | }) 144 | setRenderChange(false) 145 | }).catch((error) => { 146 | toast.error('Something went wrong') 147 | }) 148 | } 149 | }, [projectId, isAddTaskModalOpen, isRenderChange]); 150 | 151 | const updateTodo = (data) => { 152 | axios.put(`http://localhost:9000/project/${projectId}/todo`, data) 153 | .then((res) => { 154 | }).catch((error) => { 155 | toast.error('Something went wrong') 156 | }) 157 | } 158 | 159 | const handleDelete = (e, taskId) => { 160 | e.stopPropagation(); 161 | axios.delete(`http://localhost:9000/project/${projectId}/task/${taskId}`) 162 | .then((res) => { 163 | toast.success('Task is deleted') 164 | setRenderChange(true) 165 | }).catch((error) => { 166 | 167 | toast.error('Something went wrong') 168 | }) 169 | } 170 | 171 | const handleTaskDetails = (id) => { 172 | setTaskId({ projectId, id }); 173 | setTaskOpen(true); 174 | } 175 | 176 | return ( 177 |
178 |
179 |

180 | {title.slice(0, 25)}{title.length > 25 && '...'} 181 | 182 |

183 | setAddTaskModal(true)}>Add todo 184 |
185 | onDragEnd(result, columns, setColumns)} 187 | > 188 |
189 | {Object.entries(columns).map(([columnId, column], index) => { 190 | return ( 191 |
195 |
196 |
197 |

{column.name}

198 | {column.items?.length} 199 |
200 | 201 |
202 |
203 | 204 | {(provided, snapshot) => { 205 | return ( 206 |
211 | {column.items.map((item, index) => { 212 | return ( 213 | 218 | {(provided, snapshot) => { 219 | return ( 220 |
handleTaskDetails(item._id)} 226 | className={`select-none px-3.5 pt-3.5 pb-2.5 mb-2 border border-gray-200 rounded-lg shadow-sm bg-white relative ${snapshot.isDragging && 'shadow-md'}`} 227 | > 228 |
229 |
230 |

{item.title.slice(0, 22)}{item.title.length > 22 && '...'}

231 | 232 |
233 |

{item.description.slice(0, 60)}{item.description.length > 60 && '...'}

234 | Task-{item.index} 235 |
236 |
237 | ); 238 | }} 239 |
240 | ); 241 | })} 242 | {provided.placeholder} 243 |
244 | ); 245 | }} 246 |
247 | 248 |
249 |
250 | ); 251 | })} 252 |
253 |
254 | 255 | 256 |
257 | ); 258 | } 259 | 260 | export default Task; -------------------------------------------------------------------------------- /frontend/src/components/TaskModal.js: -------------------------------------------------------------------------------- 1 | import React, { Fragment, useEffect, useState } from 'react' 2 | import { Dialog, Transition } from '@headlessui/react' 3 | // import Attachment from '../image/attachment.jpg' 4 | import axios from 'axios' 5 | import toast from 'react-hot-toast' 6 | 7 | 8 | //first later capital in javascript ? 9 | 10 | 11 | 12 | 13 | const TaskModal = ({ isOpen, setIsOpen, id }) => { 14 | const [taskData, setTaskData] = useState('') 15 | 16 | const capitalizeFirstLetter = (string) => { 17 | return string ? string.charAt(0).toUpperCase() + string.slice(1) : '' 18 | } 19 | 20 | useEffect(() => { 21 | if (isOpen) { 22 | axios.get(`http://localhost:9000/project/${id.projectId}/task/${id.id}`) 23 | .then((data) => { 24 | setTaskData({ ...data.data[0].task[0] }); 25 | // console.log(taskData); 26 | }) 27 | .catch((error) => { 28 | toast.error('something went wrong') 29 | }) 30 | } 31 | }, [isOpen]); 32 | 33 | return ( 34 | 35 | setIsOpen(false)} className="relative z-50"> 36 |
37 | 46 |
47 | 48 |
49 | {/*
*/} 50 | 59 | 60 | 61 | 62 |

Task details

63 | 68 |
69 |
70 |
71 |

{capitalizeFirstLetter(taskData.title)}

72 |

{capitalizeFirstLetter(taskData.description)}

73 | {/*

Lorem ipsum dolor sit, amet consectetur adipisicing elit. Asperiores modi error, voluptatibus ullam odio nemo culpa optio incidunt, soluta sunt eos laboriosam labore animi dolorum voluptas officiis fugit perspiciatis laborum.

*/} 74 | {/*
75 |

Attachment

76 |
77 | 78 |
79 |
*/} 80 |
81 |
82 | {/*
*/} 83 |
84 |
85 | 86 |
87 |
88 | 89 |
90 |
91 |
92 |
93 | ) 94 | } 95 | 96 | export default TaskModal -------------------------------------------------------------------------------- /frontend/src/index.css: -------------------------------------------------------------------------------- 1 | @import url('https://fonts.googleapis.com/css2?family=Roboto:ital,wght@0,300;0,400;0,500;0,700;0,900;1,400&display=swap'); 2 | @tailwind base; 3 | @tailwind components; 4 | @tailwind utilities; -------------------------------------------------------------------------------- /frontend/src/index.js: -------------------------------------------------------------------------------- 1 | import React from 'react'; 2 | import ReactDOM from 'react-dom/client'; 3 | import './index.css'; 4 | import App from './App'; 5 | import reportWebVitals from './reportWebVitals'; 6 | import { BrowserRouter } from "react-router-dom"; 7 | 8 | const root = ReactDOM.createRoot(document.getElementById('root')); 9 | root.render( 10 | 11 | 12 | , 13 | ); 14 | 15 | // If you want to start measuring performance in your app, pass a function 16 | // to log results (for example: reportWebVitals(console.log)) 17 | // or send to an analytics endpoint. Learn more: https://bit.ly/CRA-vitals 18 | reportWebVitals(); 19 | -------------------------------------------------------------------------------- /frontend/src/reportWebVitals.js: -------------------------------------------------------------------------------- 1 | const reportWebVitals = onPerfEntry => { 2 | if (onPerfEntry && onPerfEntry instanceof Function) { 3 | import('web-vitals').then(({ getCLS, getFID, getFCP, getLCP, getTTFB }) => { 4 | getCLS(onPerfEntry); 5 | getFID(onPerfEntry); 6 | getFCP(onPerfEntry); 7 | getLCP(onPerfEntry); 8 | getTTFB(onPerfEntry); 9 | }); 10 | } 11 | }; 12 | 13 | export default reportWebVitals; 14 | -------------------------------------------------------------------------------- /frontend/src/setupTests.js: -------------------------------------------------------------------------------- 1 | // jest-dom adds custom jest matchers for asserting on DOM nodes. 2 | // allows you to do things like: 3 | // expect(element).toHaveTextContent(/react/i) 4 | // learn more: https://github.com/testing-library/jest-dom 5 | import '@testing-library/jest-dom'; 6 | -------------------------------------------------------------------------------- /frontend/tailwind.config.js: -------------------------------------------------------------------------------- 1 | /** @type {import('tailwindcss').Config} */ 2 | module.exports = { 3 | content: [ 4 | "./src/**/*.{js,jsx,ts,tsx}", 5 | ], 6 | theme: { 7 | extend: { 8 | colors: { 9 | "theme-gray": '#f1f5f9', 10 | "theme-primary": "#6366f1", 11 | "theme-primary-dark": "#5661b3", 12 | } 13 | }, 14 | fontFamily: { 15 | 'sans': 'Roboto, sans-serif', 16 | } 17 | }, 18 | plugins: [], 19 | } 20 | --------------------------------------------------------------------------------