├── .eslintrc.json ├── .gitignore ├── .prettierrc.json ├── .vscode └── settings.json ├── LICENSE ├── README.md ├── client ├── .browserslistrc ├── .eslintrc.js ├── .gitignore ├── README.md ├── babel.config.js ├── package-lock.json ├── package.json ├── public │ ├── favicon.ico │ └── index.html ├── src │ ├── App.vue │ ├── assets │ │ └── logo.png │ ├── axios.js │ ├── components │ │ ├── DeleteAccountButton.vue │ │ ├── Footer.vue │ │ ├── Header.vue │ │ └── Logout.vue │ ├── main.js │ ├── router │ │ └── index.js │ ├── store │ │ ├── index.js │ │ └── modules │ │ │ ├── auth.js │ │ │ └── jobs.js │ ├── styles │ │ ├── _buttons.scss │ │ ├── _config.scss │ │ ├── _footer.scss │ │ ├── _header.scss │ │ ├── _inputs.scss │ │ ├── _utilities.scss │ │ ├── custom.scss │ │ └── custom_bootstrap.scss │ └── views │ │ ├── Account.vue │ │ ├── AccountVerification.vue │ │ ├── AccountVerificationSuccessful.vue │ │ ├── Dashboard.vue │ │ ├── Home.vue │ │ ├── Login.vue │ │ ├── PasswordReset.vue │ │ └── Register.vue └── vue.config.js ├── package-lock.json ├── package.json └── server ├── .gitignore ├── index.js ├── routes ├── auth.js ├── index.js ├── jobs.js └── user.js └── utils ├── bcrypt.js ├── db.js ├── middleware ├── checkAuth.js └── checkAuthWhilePending.js ├── models ├── secretCode.js └── user.js └── nodemailer.js /.eslintrc.json: -------------------------------------------------------------------------------- 1 | { 2 | "env": { 3 | "browser": true, 4 | "commonjs": true, 5 | "es6": true, 6 | "node": true 7 | }, 8 | "extends": "eslint:recommended", 9 | "parserOptions": { 10 | "sourceType": "module", 11 | "ecmaVersion": 9 12 | }, 13 | "rules": { 14 | "indent": ["error", 4, { "SwitchCase": 4 }], 15 | "semi": ["error", "always"], 16 | "no-console": 0, 17 | "no-inner-declarations": 0, 18 | "no-sparse-arrays": 0, 19 | "no-unexpected-multiline": 0, 20 | "no-unsafe-finally": 0 21 | }, 22 | "globals": { 23 | "$": false, 24 | "jQuery": false, 25 | "Handlebars": false, 26 | "axios": false, 27 | "Vue": false, 28 | "expect": false, 29 | "test": false 30 | } 31 | } 32 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | node_modules 2 | # /client -------------------------------------------------------------------------------- /.prettierrc.json: -------------------------------------------------------------------------------- 1 | { 2 | "tabWidth": 4 3 | } 4 | -------------------------------------------------------------------------------- /.vscode/settings.json: -------------------------------------------------------------------------------- 1 | { 2 | "liveSassCompile.settings.formats": [ 3 | { 4 | "format": "compressed", 5 | "extensionName": ".min.css", 6 | "savePath": "/client/src/assets/css" 7 | } 8 | ] 9 | } 10 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | MIT License 2 | 3 | Copyright (c) 2020 Christopher Liedtke 4 | 5 | Permission is hereby granted, free of charge, to any person obtaining a copy 6 | of this software and associated documentation files (the "Software"), to deal 7 | in the Software without restriction, including without limitation the rights 8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 9 | copies of the Software, and to permit persons to whom the Software is 10 | furnished to do so, subject to the following conditions: 11 | 12 | The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. 13 | 14 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 15 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 16 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 17 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 18 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 19 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE 20 | SOFTWARE. -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # Starter Repository - Vue with VueX, Bootstrap, Node.js/Express, mongoDB with Authentication and Email Verification 2 | 3 | > A starter repository for a front-end Vue.js SPA with a Node.js/Express and mongoDB back-end incl. pre-defined **authentication**, **email verification** and **delete account** functionality. 4 | 5 | ## Features 6 | 7 | - Register 8 | - Email verification 9 | - Login 10 | - Password reset 11 | - Delete Account 12 | 13 | ## Front-End 14 | 15 | On the front-end Vue has been set up with VueX store and routing including route protection with redirecting. The settings can be expanded further during development. Bootstrap is implemented and can be easily adapted through respective SCSS files. 16 | 17 | The starter includes the following views/routes: 18 | 19 | - Home ("/") -> [public] 20 | - Register ("/register) -> [public] 21 | - Login ("/login") -> [public] 22 | - PasswordReset ("/password-reset") -> [public] 23 | - AccountVerification ("/account/verification") -> [private] 24 | - AccountVerificationSuccessfull ("/account/verified") -> [private] 25 | - Account ("/account") -> [private] 26 | - Dashboard ("/dashboard") -> [private] 27 | 28 | and the following components: 29 | 30 | - Header 31 | - Footer 32 | - Logout 33 | - Delete Account Button 34 | 35 | The user id, user role and user status are stored in `localStorage` to be used session persistent on the client side. 36 | 37 | ## Back-End 38 | 39 | On the back-end a Node.js/Express server is set up. mongoDB serves as the database for user data and can be expanded further. The following routes are set up: 40 | 41 | - "/api/jobs" -> serves hard coded json-data [public] 42 | - "/" -> serves basic json; not requested on front-end [public] 43 | - "/user/data" -> serves hard-coded json-data [private] 44 | - "/api/auth/..." -> handles user registration [public], login [public], password reset [public], logout [public], account verification [private], account deletion [private] 45 | 46 | Private routes are protected through 'check authentication' middleware. 47 | 48 | The authentication system has been set up with json web token and persistent cookie session in the backend. The token is stored securely in the cookie. CSRF protection through csurf is set up for production. 49 | 50 | Email service is provided through nodemailer. 51 | 52 | ## Set-Up 53 | 54 | ### Prerequisites 55 | 56 | - mongoDB account and database set-up 57 | - Email hosting [optional] 58 | 59 | ### Steps to set up development environment 60 | 61 | 1. Clone and set up repository or directly use repository as template 62 | 63 | 2. Run `npm install` in `root` and `root/client` directory. 64 | 65 | 3. Add `secrets.json` in `root/server/utils` and add secrets and your mongoDB and email hosting data: 66 | 67 | ```json 68 | { 69 | "MDB_URL": "YOUR_MONGODB_URL_WITH_USERNAME_AND_PASSWORD", 70 | "COOKIE_SESSION_SECRET": "YOUR_SESSION_COOKIE_SECRET", 71 | "JWT_SECRET": "YOUR_JSONWEBTOKEN_SECRET", 72 | "EMAIL_HOST": "YOUR_EMAIL_HOST", 73 | "EMAIL_USERNAME": "YOUR_EMAIL_USERNAME", 74 | "EMAIL_PW": "YOUR_EMAIL_PASSWORD", 75 | "EMAIL_PORT": 000 76 | } 77 | ``` 78 | 79 | 4. In the default set-up email address verification is obligatory. The default user status after registration is `status: "pending"`. You can change to `status: "active"` in `root/server/utils/models/user.js` to omit email address verification. 80 | 81 | ## License 82 | 83 | MIT 84 | -------------------------------------------------------------------------------- /client/.browserslistrc: -------------------------------------------------------------------------------- 1 | > 1% 2 | last 2 versions 3 | not dead 4 | -------------------------------------------------------------------------------- /client/.eslintrc.js: -------------------------------------------------------------------------------- 1 | module.exports = { 2 | root: true, 3 | env: { 4 | node: true 5 | }, 6 | extends: ["plugin:vue/essential", "eslint:recommended", "@vue/prettier"], 7 | parserOptions: { 8 | parser: "babel-eslint" 9 | }, 10 | rules: { 11 | "no-console": process.env.NODE_ENV === "production" ? "warn" : "off", 12 | "no-debugger": process.env.NODE_ENV === "production" ? "warn" : "off" 13 | } 14 | }; 15 | -------------------------------------------------------------------------------- /client/.gitignore: -------------------------------------------------------------------------------- 1 | .DS_Store 2 | node_modules 3 | /dist 4 | 5 | # local env files 6 | .env.local 7 | .env.*.local 8 | 9 | # Log files 10 | npm-debug.log* 11 | yarn-debug.log* 12 | yarn-error.log* 13 | 14 | # Editor directories and files 15 | .idea 16 | .vscode 17 | *.suo 18 | *.ntvs* 19 | *.njsproj 20 | *.sln 21 | *.sw? 22 | -------------------------------------------------------------------------------- /client/README.md: -------------------------------------------------------------------------------- 1 | # client 2 | 3 | ## Project setup 4 | ``` 5 | npm install 6 | ``` 7 | 8 | ### Compiles and hot-reloads for development 9 | ``` 10 | npm run serve 11 | ``` 12 | 13 | ### Compiles and minifies for production 14 | ``` 15 | npm run build 16 | ``` 17 | 18 | ### Lints and fixes files 19 | ``` 20 | npm run lint 21 | ``` 22 | 23 | ### Customize configuration 24 | See [Configuration Reference](https://cli.vuejs.org/config/). 25 | -------------------------------------------------------------------------------- /client/babel.config.js: -------------------------------------------------------------------------------- 1 | module.exports = { 2 | presets: ["@vue/cli-plugin-babel/preset"] 3 | }; 4 | -------------------------------------------------------------------------------- /client/package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "client", 3 | "version": "0.1.0", 4 | "private": true, 5 | "scripts": { 6 | "serve": "vue-cli-service serve", 7 | "build": "vue-cli-service build", 8 | "lint": "vue-cli-service lint" 9 | }, 10 | "dependencies": { 11 | "axios": "^0.19.2", 12 | "bootstrap-vue": "^2.15.0", 13 | "core-js": "^3.6.4", 14 | "vue": "^2.6.11", 15 | "vue-router": "^3.1.6", 16 | "vuex": "^3.1.3" 17 | }, 18 | "devDependencies": { 19 | "@vue/cli-plugin-babel": "~4.3.0", 20 | "@vue/cli-plugin-eslint": "~4.3.0", 21 | "@vue/cli-plugin-router": "~4.3.0", 22 | "@vue/cli-plugin-vuex": "~4.3.0", 23 | "@vue/cli-service": "~4.3.0", 24 | "@vue/eslint-config-prettier": "^6.0.0", 25 | "babel-eslint": "^10.1.0", 26 | "eslint": "^6.7.2", 27 | "eslint-plugin-prettier": "^3.1.1", 28 | "eslint-plugin-vue": "^6.2.2", 29 | "node-sass": "^4.12.0", 30 | "prettier": "^1.19.1", 31 | "sass-loader": "^8.0.2", 32 | "vue-template-compiler": "^2.6.11" 33 | } 34 | } 35 | -------------------------------------------------------------------------------- /client/public/favicon.ico: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/christopherliedtke/vueX_nodeExpress_mongoDB_wAuth_starter/b3c0e566a3cb37c0a7181c964c7920e8d883e6d3/client/public/favicon.ico -------------------------------------------------------------------------------- /client/public/index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | <%= htmlWebpackPlugin.options.title %> 9 | 10 | 11 | 14 |
15 | 16 | 17 | 18 | -------------------------------------------------------------------------------- /client/src/App.vue: -------------------------------------------------------------------------------- 1 | 10 | 11 | 21 | 22 | 27 | -------------------------------------------------------------------------------- /client/src/assets/logo.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/christopherliedtke/vueX_nodeExpress_mongoDB_wAuth_starter/b3c0e566a3cb37c0a7181c964c7920e8d883e6d3/client/src/assets/logo.png -------------------------------------------------------------------------------- /client/src/axios.js: -------------------------------------------------------------------------------- 1 | import axios from "axios"; 2 | 3 | var instance = axios.create({ 4 | xsrfCookieName: "mytoken", 5 | xsrfHeaderName: "csrf-token" 6 | }); 7 | 8 | export default instance; 9 | -------------------------------------------------------------------------------- /client/src/components/DeleteAccountButton.vue: -------------------------------------------------------------------------------- 1 | 64 | 65 | 106 | 107 | 112 | -------------------------------------------------------------------------------- /client/src/components/Footer.vue: -------------------------------------------------------------------------------- 1 | 6 | 7 | 12 | 13 | 14 | -------------------------------------------------------------------------------- /client/src/components/Header.vue: -------------------------------------------------------------------------------- 1 | 36 | 37 | 51 | 52 | 57 | -------------------------------------------------------------------------------- /client/src/components/Logout.vue: -------------------------------------------------------------------------------- 1 | 23 | 24 | 43 | 44 | 45 | -------------------------------------------------------------------------------- /client/src/main.js: -------------------------------------------------------------------------------- 1 | import Vue from "vue"; 2 | import App from "./App.vue"; 3 | import router from "./router"; 4 | import store from "./store"; 5 | import { BootstrapVue, BootstrapVueIcons } from "bootstrap-vue"; 6 | 7 | Vue.use(BootstrapVue); 8 | Vue.use(BootstrapVueIcons); 9 | 10 | Vue.config.productionTip = false; 11 | 12 | new Vue({ 13 | router, 14 | store, 15 | render: h => h(App) 16 | }).$mount("#app"); 17 | -------------------------------------------------------------------------------- /client/src/router/index.js: -------------------------------------------------------------------------------- 1 | import Vue from "vue"; 2 | import VueRouter from "vue-router"; 3 | Vue.use(VueRouter); 4 | import store from "@/store"; 5 | 6 | import Home from "@/views/Home.vue"; 7 | import Dashboard from "@/views/Dashboard.vue"; 8 | import Account from "@/views/Account.vue"; 9 | import Login from "@/views/Login.vue"; 10 | import Register from "@/views/Register.vue"; 11 | import PasswordReset from "@/views/PasswordReset.vue"; 12 | import AccountVerification from "@/views/AccountVerification.vue"; 13 | import AccountVerificationSuccessful from "@/views/AccountVerificationSuccessful.vue"; 14 | 15 | const routes = [ 16 | { 17 | path: "/", 18 | name: "Home", 19 | component: Home, 20 | meta: { 21 | public: true 22 | } 23 | }, 24 | { 25 | path: "/dashboard", 26 | name: "Dashboard", 27 | component: Dashboard, 28 | meta: { 29 | public: false 30 | } 31 | }, 32 | { 33 | path: "/account", 34 | name: "Account", 35 | component: Account, 36 | meta: { 37 | public: false 38 | } 39 | }, 40 | { 41 | path: "/account/verification", 42 | name: "AccountVerification", 43 | component: AccountVerification, 44 | meta: { 45 | public: false, 46 | onlyWhenPending: true 47 | } 48 | }, 49 | { 50 | path: "/account/verified", 51 | name: "AccountVerificationSuccessful", 52 | component: AccountVerificationSuccessful, 53 | meta: { 54 | public: false, 55 | onlyWhenPending: true 56 | } 57 | }, 58 | { 59 | path: "/login", 60 | name: "Login", 61 | component: Login, 62 | meta: { 63 | public: true, 64 | onlyWhenLoggedOut: true 65 | } 66 | }, 67 | { 68 | path: "/register", 69 | name: "Register", 70 | component: Register, 71 | meta: { 72 | public: true, 73 | onlyWhenLoggedOut: true 74 | } 75 | }, 76 | { 77 | path: "/password-reset", 78 | name: "PasswordReset", 79 | component: PasswordReset, 80 | meta: { 81 | public: true, 82 | onlyWhenLoggedOut: true 83 | } 84 | } 85 | ]; 86 | 87 | const router = new VueRouter({ 88 | mode: "history", 89 | routes 90 | }); 91 | 92 | // Check auth before entering routes 93 | router.beforeEach((to, from, next) => { 94 | const isPublic = to.matched.some(record => record.meta.public); 95 | const onlyWhenLoggedOut = to.matched.some( 96 | record => record.meta.onlyWhenLoggedOut 97 | ); 98 | const onlyWhenPending = to.matched.some( 99 | record => record.meta.onlyWhenPending 100 | ); 101 | 102 | const loggedIn = !!store.getters.userId; 103 | const userActivated = localStorage.getItem("userStatus") != "pending"; 104 | 105 | if (!isPublic && !loggedIn) { 106 | return next({ 107 | path: "/login", 108 | query: { redirect: to.fullPath } 109 | }); 110 | } 111 | 112 | if (!isPublic && !onlyWhenPending && loggedIn && !userActivated) { 113 | return next("/account/verification"); 114 | } 115 | 116 | if (!isPublic && onlyWhenPending && loggedIn && userActivated) { 117 | return next("/dashboard"); 118 | } 119 | 120 | if (loggedIn && onlyWhenLoggedOut) { 121 | return next("/"); 122 | } 123 | 124 | next(); 125 | }); 126 | 127 | export default router; 128 | -------------------------------------------------------------------------------- /client/src/store/index.js: -------------------------------------------------------------------------------- 1 | import Vue from "vue"; 2 | import Vuex from "vuex"; 3 | import auth from "@/store/modules/auth"; 4 | import jobs from "@/store/modules/jobs"; 5 | 6 | Vue.use(Vuex); 7 | 8 | export default new Vuex.Store({ 9 | modules: { 10 | auth, 11 | jobs 12 | } 13 | }); 14 | -------------------------------------------------------------------------------- /client/src/store/modules/auth.js: -------------------------------------------------------------------------------- 1 | import axios from "@/axios"; 2 | import router from "@/router/index"; 3 | 4 | const state = { 5 | userId: localStorage.getItem("userId") || null, 6 | userRole: localStorage.getItem("userRole") || null, 7 | userStatus: localStorage.getItem("userStatus") || null 8 | }; 9 | 10 | const getters = { 11 | userId: state => state.userId, 12 | userRole: state => state.userRole, 13 | userStatus: state => state.userStatus 14 | }; 15 | 16 | const actions = { 17 | async userAuth({ commit }, data) { 18 | const response = await axios.post(data.url, data.userData); 19 | 20 | if (response.data.success) { 21 | localStorage.setItem("userId", response.data.userId); 22 | localStorage.setItem("userRole", response.data.userRole); 23 | localStorage.setItem("userStatus", response.data.userStatus); 24 | commit("setUserId", response.data.userId); 25 | commit("setUserRole", response.data.userRole); 26 | commit("setUserStatus", response.data.userStatus); 27 | 28 | // Manage redirect after auth 29 | const redirectQuery = router.history.current.query.redirect; 30 | let redirectPath = "/dashboard"; 31 | 32 | // catch undefined error -> no harm, no display 33 | router 34 | .push({ path: redirectQuery || redirectPath }) 35 | .catch(err => err); 36 | 37 | return { sucess: true }; 38 | } else { 39 | return response.data; 40 | } 41 | } 42 | }; 43 | 44 | const mutations = { 45 | setUserId: (state, userId) => { 46 | state.userId = userId; 47 | }, 48 | setUserRole: (state, userRole) => { 49 | state.userRole = userRole; 50 | }, 51 | setUserStatus: (state, userStatus) => { 52 | state.userStatus = userStatus; 53 | } 54 | }; 55 | 56 | export default { 57 | state, 58 | getters, 59 | actions, 60 | mutations 61 | }; 62 | -------------------------------------------------------------------------------- /client/src/store/modules/jobs.js: -------------------------------------------------------------------------------- 1 | import axios from "@/axios"; 2 | 3 | const state = { 4 | jobs: [] 5 | }; 6 | 7 | const getters = { 8 | jobs: state => state.jobs 9 | }; 10 | 11 | const actions = { 12 | async getJobs({ commit }, data) { 13 | const response = await axios.get("/api/jobs", data); 14 | commit("setJobs", response.data.jobs); 15 | } 16 | }; 17 | 18 | const mutations = { 19 | setJobs: (state, jobs) => { 20 | state.jobs = jobs; 21 | } 22 | }; 23 | 24 | export default { 25 | state, 26 | getters, 27 | actions, 28 | mutations 29 | }; 30 | -------------------------------------------------------------------------------- /client/src/styles/_buttons.scss: -------------------------------------------------------------------------------- 1 | %btn { 2 | display: inline-block; 3 | border-radius: 5px; 4 | padding: 7px 18px; 5 | border: none; 6 | cursor: pointer; 7 | transition: linear 0.3s; 8 | 9 | &:focus { 10 | outline: none; 11 | } 12 | } 13 | 14 | .btn-primary { 15 | @extend %btn; 16 | background-color: $primary; 17 | 18 | &:hover { 19 | background-color: lighten($primary, 15%); 20 | } 21 | } 22 | 23 | .btn-secondary { 24 | @extend %btn; 25 | background-color: $secondary; 26 | 27 | &:hover { 28 | background-color: lighten($secondary, 15%); 29 | } 30 | } 31 | -------------------------------------------------------------------------------- /client/src/styles/_config.scss: -------------------------------------------------------------------------------- 1 | @import url("https://fonts.googleapis.com/css2?family=Nunito:wght@400;800&display=swap"); 2 | 3 | $font-family-headlines: "Nunito", sans-serif; 4 | $font-family-sans-serif: "Nunito", sans-serif; 5 | 6 | $shadow1: 0px 0px 10px 0px rgba(29, 18, 28, 0.3); 7 | -------------------------------------------------------------------------------- /client/src/styles/_footer.scss: -------------------------------------------------------------------------------- 1 | #footer { 2 | background-color: $primary; 3 | 4 | .container { 5 | display: flex; 6 | padding: 1rem 2.5rem; 7 | color: $light; 8 | font-size: 0.9rem; 9 | } 10 | } 11 | -------------------------------------------------------------------------------- /client/src/styles/_header.scss: -------------------------------------------------------------------------------- 1 | #header { 2 | background: $primary; 3 | font-family: $font-family-headlines; 4 | 5 | .navbar { 6 | padding: 0 1rem; 7 | } 8 | } 9 | -------------------------------------------------------------------------------- /client/src/styles/_inputs.scss: -------------------------------------------------------------------------------- 1 | input { 2 | border-radius: 5px; 3 | border: 1px solid #818a91; 4 | padding: 0.625em 0.4375em; 5 | 6 | &:focus { 7 | outline: none; 8 | border-color: $secondary; 9 | } 10 | } 11 | 12 | label { 13 | margin: 1rem 0 0.3rem 0.3rem; 14 | } 15 | -------------------------------------------------------------------------------- /client/src/styles/_utilities.scss: -------------------------------------------------------------------------------- 1 | body { 2 | font-family: $font-family-sans-serif; 3 | } 4 | 5 | #app { 6 | display: flex; 7 | flex-direction: column; 8 | min-height: 100vh; 9 | } 10 | 11 | main { 12 | flex-grow: 1; 13 | display: flex; 14 | flex-direction: column; 15 | align-items: center; 16 | } 17 | 18 | .box { 19 | border-radius: 5px; 20 | box-shadow: $shadow1; 21 | } 22 | -------------------------------------------------------------------------------- /client/src/styles/custom.scss: -------------------------------------------------------------------------------- 1 | @import "config"; 2 | @import "utilities"; 3 | @import "header"; 4 | @import "footer"; 5 | @import "buttons"; 6 | @import "inputs"; 7 | -------------------------------------------------------------------------------- /client/src/styles/custom_bootstrap.scss: -------------------------------------------------------------------------------- 1 | $dark: #1d121c; 2 | $light: #f8faf9; 3 | 4 | // $primary: #6c012f; 5 | // $secondary: #b94559; 6 | // $tertiary: #928fa6; 7 | // $light-shade: #f8faf9; 8 | // $light-accent: #f3f7f4; 9 | // $dark-shade: #1d121c; 10 | -------------------------------------------------------------------------------- /client/src/views/Account.vue: -------------------------------------------------------------------------------- 1 | 18 | 19 | 33 | -------------------------------------------------------------------------------- /client/src/views/AccountVerification.vue: -------------------------------------------------------------------------------- 1 | 27 | 28 | 61 | -------------------------------------------------------------------------------- /client/src/views/AccountVerificationSuccessful.vue: -------------------------------------------------------------------------------- 1 | 17 | 18 | 54 | -------------------------------------------------------------------------------- /client/src/views/Dashboard.vue: -------------------------------------------------------------------------------- 1 | 7 | 8 | 29 | -------------------------------------------------------------------------------- /client/src/views/Home.vue: -------------------------------------------------------------------------------- 1 | 11 | 12 | 28 | -------------------------------------------------------------------------------- /client/src/views/Login.vue: -------------------------------------------------------------------------------- 1 | 58 | 59 | 95 | 96 | 111 | -------------------------------------------------------------------------------- /client/src/views/PasswordReset.vue: -------------------------------------------------------------------------------- 1 | 111 | 112 | 179 | 180 | 195 | -------------------------------------------------------------------------------- /client/src/views/Register.vue: -------------------------------------------------------------------------------- 1 | 109 | 110 | 158 | 159 | 174 | -------------------------------------------------------------------------------- /client/vue.config.js: -------------------------------------------------------------------------------- 1 | const path = require("path"); 2 | 3 | module.exports = { 4 | outputDir: path.resolve(__dirname, "../server/public"), 5 | devServer: { 6 | proxy: { 7 | "/api": { 8 | target: "http://localhost:5000" 9 | } 10 | } 11 | } 12 | }; 13 | -------------------------------------------------------------------------------- /package-lock.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "vueX_bootstrap_nodeExpress_mongoDB_wAuth_starter", 3 | "version": "1.0.0", 4 | "lockfileVersion": 1, 5 | "requires": true, 6 | "dependencies": { 7 | "@sindresorhus/is": { 8 | "version": "0.14.0", 9 | "resolved": "https://registry.npmjs.org/@sindresorhus/is/-/is-0.14.0.tgz", 10 | "integrity": "sha512-9NET910DNaIPngYnLLPeg+Ogzqsi9uM4mSboU5y6p8S5DzMTVEsJZrawi+BoDNUVBa2DhJqQYUFvMDfgU062LQ==", 11 | "dev": true 12 | }, 13 | "@szmarczak/http-timer": { 14 | "version": "1.1.2", 15 | "resolved": "https://registry.npmjs.org/@szmarczak/http-timer/-/http-timer-1.1.2.tgz", 16 | "integrity": "sha512-XIB2XbzHTN6ieIjfIMV9hlVcfPU26s2vafYWQcZHWXHOxiaRZYEDKEwdl129Zyg50+foYV2jCgtrqSA6qNuNSA==", 17 | "dev": true, 18 | "requires": { 19 | "defer-to-connect": "^1.0.1" 20 | } 21 | }, 22 | "@types/color-name": { 23 | "version": "1.1.1", 24 | "resolved": "https://registry.npmjs.org/@types/color-name/-/color-name-1.1.1.tgz", 25 | "integrity": "sha512-rr+OQyAjxze7GgWrSaJwydHStIhHq2lvY3BOC2Mj7KnzI7XK0Uw1TOOdI9lDoajEbSWLiYgoo4f1R51erQfhPQ==", 26 | "dev": true 27 | }, 28 | "abbrev": { 29 | "version": "1.1.1", 30 | "resolved": "https://registry.npmjs.org/abbrev/-/abbrev-1.1.1.tgz", 31 | "integrity": "sha512-nne9/IiQ/hzIhY6pdDnbBtz7DjPTKrY00P/zvPSm5pOFkl6xuGrGnXn/VtTNNfNtAfZ9/1RtehkszU9qcTii0Q==", 32 | "dev": true 33 | }, 34 | "accepts": { 35 | "version": "1.3.7", 36 | "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.7.tgz", 37 | "integrity": "sha512-Il80Qs2WjYlJIBNzNkK6KYqlVMTbZLXgHx2oT0pU/fjRHyEp+PEfEPY0R3WCwAGVOtauxh1hOxNgIf5bv7dQpA==", 38 | "requires": { 39 | "mime-types": "~2.1.24", 40 | "negotiator": "0.6.2" 41 | } 42 | }, 43 | "ansi-align": { 44 | "version": "3.0.0", 45 | "resolved": "https://registry.npmjs.org/ansi-align/-/ansi-align-3.0.0.tgz", 46 | "integrity": "sha512-ZpClVKqXN3RGBmKibdfWzqCY4lnjEuoNzU5T0oEFpfd/z5qJHVarukridD4juLO2FXMiwUQxr9WqQtaYa8XRYw==", 47 | "dev": true, 48 | "requires": { 49 | "string-width": "^3.0.0" 50 | }, 51 | "dependencies": { 52 | "string-width": { 53 | "version": "3.1.0", 54 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz", 55 | "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==", 56 | "dev": true, 57 | "requires": { 58 | "emoji-regex": "^7.0.1", 59 | "is-fullwidth-code-point": "^2.0.0", 60 | "strip-ansi": "^5.1.0" 61 | } 62 | } 63 | } 64 | }, 65 | "ansi-regex": { 66 | "version": "4.1.0", 67 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz", 68 | "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==", 69 | "dev": true 70 | }, 71 | "ansi-styles": { 72 | "version": "4.2.1", 73 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.2.1.tgz", 74 | "integrity": "sha512-9VGjrMsG1vePxcSweQsN20KY/c4zN0h9fLjqAbwbPfahM3t+NL+M9HC8xeXG2I8pX5NoamTGNuomEUFI7fcUjA==", 75 | "dev": true, 76 | "requires": { 77 | "@types/color-name": "^1.1.1", 78 | "color-convert": "^2.0.1" 79 | } 80 | }, 81 | "anymatch": { 82 | "version": "3.1.1", 83 | "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.1.tgz", 84 | "integrity": "sha512-mM8522psRCqzV+6LhomX5wgp25YVibjh8Wj23I5RPkPppSVSjyKD2A2mBJmWGa+KN7f2D6LNh9jkBCeyLktzjg==", 85 | "dev": true, 86 | "requires": { 87 | "normalize-path": "^3.0.0", 88 | "picomatch": "^2.0.4" 89 | } 90 | }, 91 | "array-flatten": { 92 | "version": "1.1.1", 93 | "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz", 94 | "integrity": "sha1-ml9pkFGx5wczKPKgCJaLZOopVdI=" 95 | }, 96 | "balanced-match": { 97 | "version": "1.0.0", 98 | "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.0.tgz", 99 | "integrity": "sha1-ibTRmasr7kneFk6gK4nORi1xt2c=", 100 | "dev": true 101 | }, 102 | "bcryptjs": { 103 | "version": "2.4.3", 104 | "resolved": "https://registry.npmjs.org/bcryptjs/-/bcryptjs-2.4.3.tgz", 105 | "integrity": "sha1-mrVie5PmBiH/fNrF2pczAn3x0Ms=" 106 | }, 107 | "binary-extensions": { 108 | "version": "2.0.0", 109 | "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.0.0.tgz", 110 | "integrity": "sha512-Phlt0plgpIIBOGTT/ehfFnbNlfsDEiqmzE2KRXoX1bLIlir4X/MR+zSyBEkL05ffWgnRSf/DXv+WrUAVr93/ow==", 111 | "dev": true 112 | }, 113 | "bl": { 114 | "version": "2.2.0", 115 | "resolved": "https://registry.npmjs.org/bl/-/bl-2.2.0.tgz", 116 | "integrity": "sha512-wbgvOpqopSr7uq6fJrLH8EsvYMJf9gzfo2jCsL2eTy75qXPukA4pCgHamOQkZtY5vmfVtjB+P3LNlMHW5CEZXA==", 117 | "requires": { 118 | "readable-stream": "^2.3.5", 119 | "safe-buffer": "^5.1.1" 120 | } 121 | }, 122 | "bluebird": { 123 | "version": "3.5.1", 124 | "resolved": "https://registry.npmjs.org/bluebird/-/bluebird-3.5.1.tgz", 125 | "integrity": "sha512-MKiLiV+I1AA596t9w1sQJ8jkiSr5+ZKi0WKrYGUn6d1Fx+Ij4tIj+m2WMQSGczs5jZVxV339chE8iwk6F64wjA==" 126 | }, 127 | "body-parser": { 128 | "version": "1.19.0", 129 | "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.19.0.tgz", 130 | "integrity": "sha512-dhEPs72UPbDnAQJ9ZKMNTP6ptJaionhP5cBb541nXPlW60Jepo9RV/a4fX4XWW9CuFNK22krhrj1+rgzifNCsw==", 131 | "requires": { 132 | "bytes": "3.1.0", 133 | "content-type": "~1.0.4", 134 | "debug": "2.6.9", 135 | "depd": "~1.1.2", 136 | "http-errors": "1.7.2", 137 | "iconv-lite": "0.4.24", 138 | "on-finished": "~2.3.0", 139 | "qs": "6.7.0", 140 | "raw-body": "2.4.0", 141 | "type-is": "~1.6.17" 142 | } 143 | }, 144 | "boxen": { 145 | "version": "4.2.0", 146 | "resolved": "https://registry.npmjs.org/boxen/-/boxen-4.2.0.tgz", 147 | "integrity": "sha512-eB4uT9RGzg2odpER62bBwSLvUeGC+WbRjjyyFhGsKnc8wp/m0+hQsMUvUe3H2V0D5vw0nBdO1hCJoZo5mKeuIQ==", 148 | "dev": true, 149 | "requires": { 150 | "ansi-align": "^3.0.0", 151 | "camelcase": "^5.3.1", 152 | "chalk": "^3.0.0", 153 | "cli-boxes": "^2.2.0", 154 | "string-width": "^4.1.0", 155 | "term-size": "^2.1.0", 156 | "type-fest": "^0.8.1", 157 | "widest-line": "^3.1.0" 158 | } 159 | }, 160 | "brace-expansion": { 161 | "version": "1.1.11", 162 | "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", 163 | "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", 164 | "dev": true, 165 | "requires": { 166 | "balanced-match": "^1.0.0", 167 | "concat-map": "0.0.1" 168 | } 169 | }, 170 | "braces": { 171 | "version": "3.0.2", 172 | "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz", 173 | "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", 174 | "dev": true, 175 | "requires": { 176 | "fill-range": "^7.0.1" 177 | } 178 | }, 179 | "bson": { 180 | "version": "1.1.4", 181 | "resolved": "https://registry.npmjs.org/bson/-/bson-1.1.4.tgz", 182 | "integrity": "sha512-S/yKGU1syOMzO86+dGpg2qGoDL0zvzcb262G+gqEy6TgP6rt6z6qxSFX/8X6vLC91P7G7C3nLs0+bvDzmvBA3Q==" 183 | }, 184 | "buffer-equal-constant-time": { 185 | "version": "1.0.1", 186 | "resolved": "https://registry.npmjs.org/buffer-equal-constant-time/-/buffer-equal-constant-time-1.0.1.tgz", 187 | "integrity": "sha1-+OcRMvf/5uAaXJaXpMbz5I1cyBk=" 188 | }, 189 | "bytes": { 190 | "version": "3.1.0", 191 | "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.0.tgz", 192 | "integrity": "sha512-zauLjrfCG+xvoyaqLoV8bLVXXNGC4JqlxFCutSDWA6fJrTo2ZuvLYTqZ7aHBLZSMOopbzwv8f+wZcVzfVTI2Dg==" 193 | }, 194 | "cacheable-request": { 195 | "version": "6.1.0", 196 | "resolved": "https://registry.npmjs.org/cacheable-request/-/cacheable-request-6.1.0.tgz", 197 | "integrity": "sha512-Oj3cAGPCqOZX7Rz64Uny2GYAZNliQSqfbePrgAQ1wKAihYmCUnraBtJtKcGR4xz7wF+LoJC+ssFZvv5BgF9Igg==", 198 | "dev": true, 199 | "requires": { 200 | "clone-response": "^1.0.2", 201 | "get-stream": "^5.1.0", 202 | "http-cache-semantics": "^4.0.0", 203 | "keyv": "^3.0.0", 204 | "lowercase-keys": "^2.0.0", 205 | "normalize-url": "^4.1.0", 206 | "responselike": "^1.0.2" 207 | }, 208 | "dependencies": { 209 | "get-stream": { 210 | "version": "5.1.0", 211 | "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-5.1.0.tgz", 212 | "integrity": "sha512-EXr1FOzrzTfGeL0gQdeFEvOMm2mzMOglyiOXSTpPC+iAjAKftbr3jpCMWynogwYnM+eSj9sHGc6wjIcDvYiygw==", 213 | "dev": true, 214 | "requires": { 215 | "pump": "^3.0.0" 216 | } 217 | }, 218 | "lowercase-keys": { 219 | "version": "2.0.0", 220 | "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-2.0.0.tgz", 221 | "integrity": "sha512-tqNXrS78oMOE73NMxK4EMLQsQowWf8jKooH9g7xPavRT706R6bkQJ6DY2Te7QukaZsulxa30wQ7bk0pm4XiHmA==", 222 | "dev": true 223 | } 224 | } 225 | }, 226 | "camelcase": { 227 | "version": "5.3.1", 228 | "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz", 229 | "integrity": "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==", 230 | "dev": true 231 | }, 232 | "chalk": { 233 | "version": "3.0.0", 234 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-3.0.0.tgz", 235 | "integrity": "sha512-4D3B6Wf41KOYRFdszmDqMCGq5VV/uMAB273JILmO+3jAlh8X4qDtdtgCR3fxtbLEMzSx22QdhnDcJvu2u1fVwg==", 236 | "dev": true, 237 | "requires": { 238 | "ansi-styles": "^4.1.0", 239 | "supports-color": "^7.1.0" 240 | }, 241 | "dependencies": { 242 | "has-flag": { 243 | "version": "4.0.0", 244 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", 245 | "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", 246 | "dev": true 247 | }, 248 | "supports-color": { 249 | "version": "7.1.0", 250 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.1.0.tgz", 251 | "integrity": "sha512-oRSIpR8pxT1Wr2FquTNnGet79b3BWljqOuoW/h4oBhxJ/HUbX5nX6JSruTkvXDCFMwDPvsaTTbvMLKZWSy0R5g==", 252 | "dev": true, 253 | "requires": { 254 | "has-flag": "^4.0.0" 255 | } 256 | } 257 | } 258 | }, 259 | "chokidar": { 260 | "version": "3.4.0", 261 | "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.4.0.tgz", 262 | "integrity": "sha512-aXAaho2VJtisB/1fg1+3nlLJqGOuewTzQpd/Tz0yTg2R0e4IGtshYvtjowyEumcBv2z+y4+kc75Mz7j5xJskcQ==", 263 | "dev": true, 264 | "requires": { 265 | "anymatch": "~3.1.1", 266 | "braces": "~3.0.2", 267 | "fsevents": "~2.1.2", 268 | "glob-parent": "~5.1.0", 269 | "is-binary-path": "~2.1.0", 270 | "is-glob": "~4.0.1", 271 | "normalize-path": "~3.0.0", 272 | "readdirp": "~3.4.0" 273 | } 274 | }, 275 | "ci-info": { 276 | "version": "2.0.0", 277 | "resolved": "https://registry.npmjs.org/ci-info/-/ci-info-2.0.0.tgz", 278 | "integrity": "sha512-5tK7EtrZ0N+OLFMthtqOj4fI2Jeb88C4CAZPu25LDVUgXJ0A3Js4PMGqrn0JU1W0Mh1/Z8wZzYPxqUrXeBboCQ==", 279 | "dev": true 280 | }, 281 | "cli-boxes": { 282 | "version": "2.2.0", 283 | "resolved": "https://registry.npmjs.org/cli-boxes/-/cli-boxes-2.2.0.tgz", 284 | "integrity": "sha512-gpaBrMAizVEANOpfZp/EEUixTXDyGt7DFzdK5hU+UbWt/J0lB0w20ncZj59Z9a93xHb9u12zF5BS6i9RKbtg4w==", 285 | "dev": true 286 | }, 287 | "clone-response": { 288 | "version": "1.0.2", 289 | "resolved": "https://registry.npmjs.org/clone-response/-/clone-response-1.0.2.tgz", 290 | "integrity": "sha1-0dyXOSAxTfZ/vrlCI7TuNQI56Ws=", 291 | "dev": true, 292 | "requires": { 293 | "mimic-response": "^1.0.0" 294 | } 295 | }, 296 | "color-convert": { 297 | "version": "2.0.1", 298 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", 299 | "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", 300 | "dev": true, 301 | "requires": { 302 | "color-name": "~1.1.4" 303 | } 304 | }, 305 | "color-name": { 306 | "version": "1.1.4", 307 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", 308 | "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", 309 | "dev": true 310 | }, 311 | "compressible": { 312 | "version": "2.0.18", 313 | "resolved": "https://registry.npmjs.org/compressible/-/compressible-2.0.18.tgz", 314 | "integrity": "sha512-AF3r7P5dWxL8MxyITRMlORQNaOA2IkAFaTr4k7BUumjPtRpGDTZpl0Pb1XCO6JeDCBdp126Cgs9sMxqSjgYyRg==", 315 | "requires": { 316 | "mime-db": ">= 1.43.0 < 2" 317 | } 318 | }, 319 | "compression": { 320 | "version": "1.7.4", 321 | "resolved": "https://registry.npmjs.org/compression/-/compression-1.7.4.tgz", 322 | "integrity": "sha512-jaSIDzP9pZVS4ZfQ+TzvtiWhdpFhE2RDHz8QJkpX9SIpLq88VueF5jJw6t+6CUQcAoA6t+x89MLrWAqpfDE8iQ==", 323 | "requires": { 324 | "accepts": "~1.3.5", 325 | "bytes": "3.0.0", 326 | "compressible": "~2.0.16", 327 | "debug": "2.6.9", 328 | "on-headers": "~1.0.2", 329 | "safe-buffer": "5.1.2", 330 | "vary": "~1.1.2" 331 | }, 332 | "dependencies": { 333 | "bytes": { 334 | "version": "3.0.0", 335 | "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.0.0.tgz", 336 | "integrity": "sha1-0ygVQE1olpn4Wk6k+odV3ROpYEg=" 337 | } 338 | } 339 | }, 340 | "concat-map": { 341 | "version": "0.0.1", 342 | "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", 343 | "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=", 344 | "dev": true 345 | }, 346 | "configstore": { 347 | "version": "5.0.1", 348 | "resolved": "https://registry.npmjs.org/configstore/-/configstore-5.0.1.tgz", 349 | "integrity": "sha512-aMKprgk5YhBNyH25hj8wGt2+D52Sw1DRRIzqBwLp2Ya9mFmY8KPvvtvmna8SxVR9JMZ4kzMD68N22vlaRpkeFA==", 350 | "dev": true, 351 | "requires": { 352 | "dot-prop": "^5.2.0", 353 | "graceful-fs": "^4.1.2", 354 | "make-dir": "^3.0.0", 355 | "unique-string": "^2.0.0", 356 | "write-file-atomic": "^3.0.0", 357 | "xdg-basedir": "^4.0.0" 358 | } 359 | }, 360 | "content-disposition": { 361 | "version": "0.5.3", 362 | "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.3.tgz", 363 | "integrity": "sha512-ExO0774ikEObIAEV9kDo50o+79VCUdEB6n6lzKgGwupcVeRlhrj3qGAfwq8G6uBJjkqLrhT0qEYFcWng8z1z0g==", 364 | "requires": { 365 | "safe-buffer": "5.1.2" 366 | } 367 | }, 368 | "content-type": { 369 | "version": "1.0.4", 370 | "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.4.tgz", 371 | "integrity": "sha512-hIP3EEPs8tB9AT1L+NUqtwOAps4mk2Zob89MWXMHjHWg9milF/j4osnnQLXBCBFBk/tvIG/tUc9mOUJiPBhPXA==" 372 | }, 373 | "cookie": { 374 | "version": "0.4.0", 375 | "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.4.0.tgz", 376 | "integrity": "sha512-+Hp8fLp57wnUSt0tY0tHEXh4voZRDnoIrZPqlo3DPiI4y9lwg/jqx+1Om94/W6ZaPDOUbnjOt/99w66zk+l1Xg==" 377 | }, 378 | "cookie-session": { 379 | "version": "1.4.0", 380 | "resolved": "https://registry.npmjs.org/cookie-session/-/cookie-session-1.4.0.tgz", 381 | "integrity": "sha512-0hhwD+BUIwMXQraiZP/J7VP2YFzqo6g4WqZlWHtEHQ22t0MeZZrNBSCxC1zcaLAs8ApT3BzAKizx9gW/AP9vNA==", 382 | "requires": { 383 | "cookies": "0.8.0", 384 | "debug": "2.6.9", 385 | "on-headers": "~1.0.2" 386 | } 387 | }, 388 | "cookie-signature": { 389 | "version": "1.0.6", 390 | "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz", 391 | "integrity": "sha1-4wOogrNCzD7oylE6eZmXNNqzriw=" 392 | }, 393 | "cookies": { 394 | "version": "0.8.0", 395 | "resolved": "https://registry.npmjs.org/cookies/-/cookies-0.8.0.tgz", 396 | "integrity": "sha512-8aPsApQfebXnuI+537McwYsDtjVxGm8gTIzQI3FDW6t5t/DAhERxtnbEPN/8RX+uZthoz4eCOgloXaE5cYyNow==", 397 | "requires": { 398 | "depd": "~2.0.0", 399 | "keygrip": "~1.1.0" 400 | }, 401 | "dependencies": { 402 | "depd": { 403 | "version": "2.0.0", 404 | "resolved": "https://registry.npmjs.org/depd/-/depd-2.0.0.tgz", 405 | "integrity": "sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw==" 406 | } 407 | } 408 | }, 409 | "core-util-is": { 410 | "version": "1.0.2", 411 | "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.2.tgz", 412 | "integrity": "sha1-tf1UIgqivFq1eqtxQMlAdUUDwac=" 413 | }, 414 | "cors": { 415 | "version": "2.8.5", 416 | "resolved": "https://registry.npmjs.org/cors/-/cors-2.8.5.tgz", 417 | "integrity": "sha512-KIHbLJqu73RGr/hnbrO9uBeixNGuvSQjul/jdFvS/KFSIH1hWVd1ng7zOHx+YrEfInLG7q4n6GHQ9cDtxv/P6g==", 418 | "requires": { 419 | "object-assign": "^4", 420 | "vary": "^1" 421 | } 422 | }, 423 | "crypto-random-string": { 424 | "version": "3.2.0", 425 | "resolved": "https://registry.npmjs.org/crypto-random-string/-/crypto-random-string-3.2.0.tgz", 426 | "integrity": "sha512-8vPu5bsKaq2uKRy3OL7h1Oo7RayAWB8sYexLKAqvCXVib8SxgbmoF1IN4QMKjBv8uI8mp5gPPMbiRah25GMrVQ==", 427 | "requires": { 428 | "type-fest": "^0.8.1" 429 | } 430 | }, 431 | "csrf": { 432 | "version": "3.1.0", 433 | "resolved": "https://registry.npmjs.org/csrf/-/csrf-3.1.0.tgz", 434 | "integrity": "sha512-uTqEnCvWRk042asU6JtapDTcJeeailFy4ydOQS28bj1hcLnYRiqi8SsD2jS412AY1I/4qdOwWZun774iqywf9w==", 435 | "requires": { 436 | "rndm": "1.2.0", 437 | "tsscmp": "1.0.6", 438 | "uid-safe": "2.1.5" 439 | } 440 | }, 441 | "csurf": { 442 | "version": "1.11.0", 443 | "resolved": "https://registry.npmjs.org/csurf/-/csurf-1.11.0.tgz", 444 | "integrity": "sha512-UCtehyEExKTxgiu8UHdGvHj4tnpE/Qctue03Giq5gPgMQ9cg/ciod5blZQ5a4uCEenNQjxyGuzygLdKUmee/bQ==", 445 | "requires": { 446 | "cookie": "0.4.0", 447 | "cookie-signature": "1.0.6", 448 | "csrf": "3.1.0", 449 | "http-errors": "~1.7.3" 450 | }, 451 | "dependencies": { 452 | "http-errors": { 453 | "version": "1.7.3", 454 | "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.7.3.tgz", 455 | "integrity": "sha512-ZTTX0MWrsQ2ZAhA1cejAwDLycFsd7I7nVtnkT3Ol0aqodaKW+0CTZDQ1uBv5whptCnc8e8HeRRJxRs0kmm/Qfw==", 456 | "requires": { 457 | "depd": "~1.1.2", 458 | "inherits": "2.0.4", 459 | "setprototypeof": "1.1.1", 460 | "statuses": ">= 1.5.0 < 2", 461 | "toidentifier": "1.0.0" 462 | } 463 | }, 464 | "inherits": { 465 | "version": "2.0.4", 466 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", 467 | "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==" 468 | } 469 | } 470 | }, 471 | "debug": { 472 | "version": "2.6.9", 473 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", 474 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", 475 | "requires": { 476 | "ms": "2.0.0" 477 | } 478 | }, 479 | "decompress-response": { 480 | "version": "3.3.0", 481 | "resolved": "https://registry.npmjs.org/decompress-response/-/decompress-response-3.3.0.tgz", 482 | "integrity": "sha1-gKTdMjdIOEv6JICDYirt7Jgq3/M=", 483 | "dev": true, 484 | "requires": { 485 | "mimic-response": "^1.0.0" 486 | } 487 | }, 488 | "deep-extend": { 489 | "version": "0.6.0", 490 | "resolved": "https://registry.npmjs.org/deep-extend/-/deep-extend-0.6.0.tgz", 491 | "integrity": "sha512-LOHxIOaPYdHlJRtCQfDIVZtfw/ufM8+rVj649RIHzcm/vGwQRXFt6OPqIFWsm2XEMrNIEtWR64sY1LEKD2vAOA==", 492 | "dev": true 493 | }, 494 | "defer-to-connect": { 495 | "version": "1.1.3", 496 | "resolved": "https://registry.npmjs.org/defer-to-connect/-/defer-to-connect-1.1.3.tgz", 497 | "integrity": "sha512-0ISdNousHvZT2EiFlZeZAHBUvSxmKswVCEf8hW7KWgG4a8MVEu/3Vb6uWYozkjylyCxe0JBIiRB1jV45S70WVQ==", 498 | "dev": true 499 | }, 500 | "denque": { 501 | "version": "1.4.1", 502 | "resolved": "https://registry.npmjs.org/denque/-/denque-1.4.1.tgz", 503 | "integrity": "sha512-OfzPuSZKGcgr96rf1oODnfjqBFmr1DVoc/TrItj3Ohe0Ah1C5WX5Baquw/9U9KovnQ88EqmJbD66rKYUQYN1tQ==" 504 | }, 505 | "depd": { 506 | "version": "1.1.2", 507 | "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.2.tgz", 508 | "integrity": "sha1-m81S4UwJd2PnSbJ0xDRu0uVgtak=" 509 | }, 510 | "destroy": { 511 | "version": "1.0.4", 512 | "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.0.4.tgz", 513 | "integrity": "sha1-l4hXRCxEdJ5CBmE+N5RiBYJqvYA=" 514 | }, 515 | "dot-prop": { 516 | "version": "5.2.0", 517 | "resolved": "https://registry.npmjs.org/dot-prop/-/dot-prop-5.2.0.tgz", 518 | "integrity": "sha512-uEUyaDKoSQ1M4Oq8l45hSE26SnTxL6snNnqvK/VWx5wJhmff5z0FUVJDKDanor/6w3kzE3i7XZOk+7wC0EXr1A==", 519 | "dev": true, 520 | "requires": { 521 | "is-obj": "^2.0.0" 522 | } 523 | }, 524 | "duplexer3": { 525 | "version": "0.1.4", 526 | "resolved": "https://registry.npmjs.org/duplexer3/-/duplexer3-0.1.4.tgz", 527 | "integrity": "sha1-7gHdHKwO08vH/b6jfcCo8c4ALOI=", 528 | "dev": true 529 | }, 530 | "ecdsa-sig-formatter": { 531 | "version": "1.0.11", 532 | "resolved": "https://registry.npmjs.org/ecdsa-sig-formatter/-/ecdsa-sig-formatter-1.0.11.tgz", 533 | "integrity": "sha512-nagl3RYrbNv6kQkeJIpt6NJZy8twLB/2vtz6yN9Z4vRKHN4/QZJIEbqohALSgwKdnksuY3k5Addp5lg8sVoVcQ==", 534 | "requires": { 535 | "safe-buffer": "^5.0.1" 536 | } 537 | }, 538 | "ee-first": { 539 | "version": "1.1.1", 540 | "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", 541 | "integrity": "sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0=" 542 | }, 543 | "emoji-regex": { 544 | "version": "7.0.3", 545 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-7.0.3.tgz", 546 | "integrity": "sha512-CwBLREIQ7LvYFB0WyRvwhq5N5qPhc6PMjD6bYggFlI5YyDgl+0vxq5VHbMOFqLg7hfWzmu8T5Z1QofhmTIhItA==", 547 | "dev": true 548 | }, 549 | "encodeurl": { 550 | "version": "1.0.2", 551 | "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz", 552 | "integrity": "sha1-rT/0yG7C0CkyL1oCw6mmBslbP1k=" 553 | }, 554 | "end-of-stream": { 555 | "version": "1.4.4", 556 | "resolved": "https://registry.npmjs.org/end-of-stream/-/end-of-stream-1.4.4.tgz", 557 | "integrity": "sha512-+uw1inIHVPQoaVuHzRyXd21icM+cnt4CzD5rW+NC1wjOUSTOs+Te7FOv7AhN7vS9x/oIyhLP5PR1H+phQAHu5Q==", 558 | "dev": true, 559 | "requires": { 560 | "once": "^1.4.0" 561 | } 562 | }, 563 | "escape-goat": { 564 | "version": "2.1.1", 565 | "resolved": "https://registry.npmjs.org/escape-goat/-/escape-goat-2.1.1.tgz", 566 | "integrity": "sha512-8/uIhbG12Csjy2JEW7D9pHbreaVaS/OpN3ycnyvElTdwM5n6GY6W6e2IPemfvGZeUMqZ9A/3GqIZMgKnBhAw/Q==", 567 | "dev": true 568 | }, 569 | "escape-html": { 570 | "version": "1.0.3", 571 | "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz", 572 | "integrity": "sha1-Aljq5NPQwJdN4cFpGI7wBR0dGYg=" 573 | }, 574 | "etag": { 575 | "version": "1.8.1", 576 | "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz", 577 | "integrity": "sha1-Qa4u62XvpiJorr/qg6x9eSmbCIc=" 578 | }, 579 | "express": { 580 | "version": "4.17.1", 581 | "resolved": "https://registry.npmjs.org/express/-/express-4.17.1.tgz", 582 | "integrity": "sha512-mHJ9O79RqluphRrcw2X/GTh3k9tVv8YcoyY4Kkh4WDMUYKRZUq0h1o0w2rrrxBqM7VoeUVqgb27xlEMXTnYt4g==", 583 | "requires": { 584 | "accepts": "~1.3.7", 585 | "array-flatten": "1.1.1", 586 | "body-parser": "1.19.0", 587 | "content-disposition": "0.5.3", 588 | "content-type": "~1.0.4", 589 | "cookie": "0.4.0", 590 | "cookie-signature": "1.0.6", 591 | "debug": "2.6.9", 592 | "depd": "~1.1.2", 593 | "encodeurl": "~1.0.2", 594 | "escape-html": "~1.0.3", 595 | "etag": "~1.8.1", 596 | "finalhandler": "~1.1.2", 597 | "fresh": "0.5.2", 598 | "merge-descriptors": "1.0.1", 599 | "methods": "~1.1.2", 600 | "on-finished": "~2.3.0", 601 | "parseurl": "~1.3.3", 602 | "path-to-regexp": "0.1.7", 603 | "proxy-addr": "~2.0.5", 604 | "qs": "6.7.0", 605 | "range-parser": "~1.2.1", 606 | "safe-buffer": "5.1.2", 607 | "send": "0.17.1", 608 | "serve-static": "1.14.1", 609 | "setprototypeof": "1.1.1", 610 | "statuses": "~1.5.0", 611 | "type-is": "~1.6.18", 612 | "utils-merge": "1.0.1", 613 | "vary": "~1.1.2" 614 | } 615 | }, 616 | "express-session": { 617 | "version": "1.17.1", 618 | "resolved": "https://registry.npmjs.org/express-session/-/express-session-1.17.1.tgz", 619 | "integrity": "sha512-UbHwgqjxQZJiWRTMyhvWGvjBQduGCSBDhhZXYenziMFjxst5rMV+aJZ6hKPHZnPyHGsrqRICxtX8jtEbm/z36Q==", 620 | "requires": { 621 | "cookie": "0.4.0", 622 | "cookie-signature": "1.0.6", 623 | "debug": "2.6.9", 624 | "depd": "~2.0.0", 625 | "on-headers": "~1.0.2", 626 | "parseurl": "~1.3.3", 627 | "safe-buffer": "5.2.0", 628 | "uid-safe": "~2.1.5" 629 | }, 630 | "dependencies": { 631 | "depd": { 632 | "version": "2.0.0", 633 | "resolved": "https://registry.npmjs.org/depd/-/depd-2.0.0.tgz", 634 | "integrity": "sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw==" 635 | }, 636 | "safe-buffer": { 637 | "version": "5.2.0", 638 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.0.tgz", 639 | "integrity": "sha512-fZEwUGbVl7kouZs1jCdMLdt95hdIv0ZeHg6L7qPeciMZhZ+/gdesW4wgTARkrFWEpspjEATAzUGPG8N2jJiwbg==" 640 | } 641 | } 642 | }, 643 | "fill-range": { 644 | "version": "7.0.1", 645 | "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz", 646 | "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", 647 | "dev": true, 648 | "requires": { 649 | "to-regex-range": "^5.0.1" 650 | } 651 | }, 652 | "finalhandler": { 653 | "version": "1.1.2", 654 | "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.1.2.tgz", 655 | "integrity": "sha512-aAWcW57uxVNrQZqFXjITpW3sIUQmHGG3qSb9mUah9MgMC4NeWhNOlNjXEYq3HjRAvL6arUviZGGJsBg6z0zsWA==", 656 | "requires": { 657 | "debug": "2.6.9", 658 | "encodeurl": "~1.0.2", 659 | "escape-html": "~1.0.3", 660 | "on-finished": "~2.3.0", 661 | "parseurl": "~1.3.3", 662 | "statuses": "~1.5.0", 663 | "unpipe": "~1.0.0" 664 | } 665 | }, 666 | "forwarded": { 667 | "version": "0.1.2", 668 | "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.1.2.tgz", 669 | "integrity": "sha1-mMI9qxF1ZXuMBXPozszZGw/xjIQ=" 670 | }, 671 | "fresh": { 672 | "version": "0.5.2", 673 | "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz", 674 | "integrity": "sha1-PYyt2Q2XZWn6g1qx+OSyOhBWBac=" 675 | }, 676 | "fsevents": { 677 | "version": "2.1.3", 678 | "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.1.3.tgz", 679 | "integrity": "sha512-Auw9a4AxqWpa9GUfj370BMPzzyncfBABW8Mab7BGWBYDj4Isgq+cDKtx0i6u9jcX9pQDnswsaaOTgTmA5pEjuQ==", 680 | "dev": true, 681 | "optional": true 682 | }, 683 | "get-stream": { 684 | "version": "4.1.0", 685 | "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-4.1.0.tgz", 686 | "integrity": "sha512-GMat4EJ5161kIy2HevLlr4luNjBgvmj413KaQA7jt4V8B4RDsfpHk7WQ9GVqfYyyx8OS/L66Kox+rJRNklLK7w==", 687 | "dev": true, 688 | "requires": { 689 | "pump": "^3.0.0" 690 | } 691 | }, 692 | "glob-parent": { 693 | "version": "5.1.1", 694 | "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.1.tgz", 695 | "integrity": "sha512-FnI+VGOpnlGHWZxthPGR+QhR78fuiK0sNLkHQv+bL9fQi57lNNdquIbna/WrfROrolq8GK5Ek6BiMwqL/voRYQ==", 696 | "dev": true, 697 | "requires": { 698 | "is-glob": "^4.0.1" 699 | } 700 | }, 701 | "global-dirs": { 702 | "version": "2.0.1", 703 | "resolved": "https://registry.npmjs.org/global-dirs/-/global-dirs-2.0.1.tgz", 704 | "integrity": "sha512-5HqUqdhkEovj2Of/ms3IeS/EekcO54ytHRLV4PEY2rhRwrHXLQjeVEES0Lhka0xwNDtGYn58wyC4s5+MHsOO6A==", 705 | "dev": true, 706 | "requires": { 707 | "ini": "^1.3.5" 708 | } 709 | }, 710 | "got": { 711 | "version": "9.6.0", 712 | "resolved": "https://registry.npmjs.org/got/-/got-9.6.0.tgz", 713 | "integrity": "sha512-R7eWptXuGYxwijs0eV+v3o6+XH1IqVK8dJOEecQfTmkncw9AV4dcw/Dhxi8MdlqPthxxpZyizMzyg8RTmEsG+Q==", 714 | "dev": true, 715 | "requires": { 716 | "@sindresorhus/is": "^0.14.0", 717 | "@szmarczak/http-timer": "^1.1.2", 718 | "cacheable-request": "^6.0.0", 719 | "decompress-response": "^3.3.0", 720 | "duplexer3": "^0.1.4", 721 | "get-stream": "^4.1.0", 722 | "lowercase-keys": "^1.0.1", 723 | "mimic-response": "^1.0.1", 724 | "p-cancelable": "^1.0.0", 725 | "to-readable-stream": "^1.0.0", 726 | "url-parse-lax": "^3.0.0" 727 | } 728 | }, 729 | "graceful-fs": { 730 | "version": "4.2.4", 731 | "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.4.tgz", 732 | "integrity": "sha512-WjKPNJF79dtJAVniUlGGWHYGz2jWxT6VhN/4m1NdkbZ2nOsEF+cI1Edgql5zCRhs/VsQYRvrXctxktVXZUkixw==", 733 | "dev": true 734 | }, 735 | "has-flag": { 736 | "version": "3.0.0", 737 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", 738 | "integrity": "sha1-tdRU3CGZriJWmfNGfloH87lVuv0=", 739 | "dev": true 740 | }, 741 | "has-yarn": { 742 | "version": "2.1.0", 743 | "resolved": "https://registry.npmjs.org/has-yarn/-/has-yarn-2.1.0.tgz", 744 | "integrity": "sha512-UqBRqi4ju7T+TqGNdqAO0PaSVGsDGJUBQvk9eUWNGRY1CFGDzYhLWoM7JQEemnlvVcv/YEmc2wNW8BC24EnUsw==", 745 | "dev": true 746 | }, 747 | "http-cache-semantics": { 748 | "version": "4.1.0", 749 | "resolved": "https://registry.npmjs.org/http-cache-semantics/-/http-cache-semantics-4.1.0.tgz", 750 | "integrity": "sha512-carPklcUh7ROWRK7Cv27RPtdhYhUsela/ue5/jKzjegVvXDqM2ILE9Q2BGn9JZJh1g87cp56su/FgQSzcWS8cQ==", 751 | "dev": true 752 | }, 753 | "http-errors": { 754 | "version": "1.7.2", 755 | "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.7.2.tgz", 756 | "integrity": "sha512-uUQBt3H/cSIVfch6i1EuPNy/YsRSOUBXTVfZ+yR7Zjez3qjBz6i9+i4zjNaoqcoFVI4lQJ5plg63TvGfRSDCRg==", 757 | "requires": { 758 | "depd": "~1.1.2", 759 | "inherits": "2.0.3", 760 | "setprototypeof": "1.1.1", 761 | "statuses": ">= 1.5.0 < 2", 762 | "toidentifier": "1.0.0" 763 | } 764 | }, 765 | "iconv-lite": { 766 | "version": "0.4.24", 767 | "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", 768 | "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", 769 | "requires": { 770 | "safer-buffer": ">= 2.1.2 < 3" 771 | } 772 | }, 773 | "ignore-by-default": { 774 | "version": "1.0.1", 775 | "resolved": "https://registry.npmjs.org/ignore-by-default/-/ignore-by-default-1.0.1.tgz", 776 | "integrity": "sha1-SMptcvbGo68Aqa1K5odr44ieKwk=", 777 | "dev": true 778 | }, 779 | "import-lazy": { 780 | "version": "2.1.0", 781 | "resolved": "https://registry.npmjs.org/import-lazy/-/import-lazy-2.1.0.tgz", 782 | "integrity": "sha1-BWmOPUXIjo1+nZLLBYTnfwlvPkM=", 783 | "dev": true 784 | }, 785 | "imurmurhash": { 786 | "version": "0.1.4", 787 | "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", 788 | "integrity": "sha1-khi5srkoojixPcT7a21XbyMUU+o=", 789 | "dev": true 790 | }, 791 | "inherits": { 792 | "version": "2.0.3", 793 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", 794 | "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=" 795 | }, 796 | "ini": { 797 | "version": "1.3.5", 798 | "resolved": "https://registry.npmjs.org/ini/-/ini-1.3.5.tgz", 799 | "integrity": "sha512-RZY5huIKCMRWDUqZlEi72f/lmXKMvuszcMBduliQ3nnWbx9X/ZBQO7DijMEYS9EhHBb2qacRUMtC7svLwe0lcw==", 800 | "dev": true 801 | }, 802 | "ipaddr.js": { 803 | "version": "1.9.1", 804 | "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz", 805 | "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==" 806 | }, 807 | "is-binary-path": { 808 | "version": "2.1.0", 809 | "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz", 810 | "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==", 811 | "dev": true, 812 | "requires": { 813 | "binary-extensions": "^2.0.0" 814 | } 815 | }, 816 | "is-ci": { 817 | "version": "2.0.0", 818 | "resolved": "https://registry.npmjs.org/is-ci/-/is-ci-2.0.0.tgz", 819 | "integrity": "sha512-YfJT7rkpQB0updsdHLGWrvhBJfcfzNNawYDNIyQXJz0IViGf75O8EBPKSdvw2rF+LGCsX4FZ8tcr3b19LcZq4w==", 820 | "dev": true, 821 | "requires": { 822 | "ci-info": "^2.0.0" 823 | } 824 | }, 825 | "is-extglob": { 826 | "version": "2.1.1", 827 | "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", 828 | "integrity": "sha1-qIwCU1eR8C7TfHahueqXc8gz+MI=", 829 | "dev": true 830 | }, 831 | "is-fullwidth-code-point": { 832 | "version": "2.0.0", 833 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", 834 | "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=", 835 | "dev": true 836 | }, 837 | "is-glob": { 838 | "version": "4.0.1", 839 | "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.1.tgz", 840 | "integrity": "sha512-5G0tKtBTFImOqDnLB2hG6Bp2qcKEFduo4tZu9MT/H6NQv/ghhy30o55ufafxJ/LdH79LLs2Kfrn85TLKyA7BUg==", 841 | "dev": true, 842 | "requires": { 843 | "is-extglob": "^2.1.1" 844 | } 845 | }, 846 | "is-installed-globally": { 847 | "version": "0.3.2", 848 | "resolved": "https://registry.npmjs.org/is-installed-globally/-/is-installed-globally-0.3.2.tgz", 849 | "integrity": "sha512-wZ8x1js7Ia0kecP/CHM/3ABkAmujX7WPvQk6uu3Fly/Mk44pySulQpnHG46OMjHGXApINnV4QhY3SWnECO2z5g==", 850 | "dev": true, 851 | "requires": { 852 | "global-dirs": "^2.0.1", 853 | "is-path-inside": "^3.0.1" 854 | } 855 | }, 856 | "is-npm": { 857 | "version": "4.0.0", 858 | "resolved": "https://registry.npmjs.org/is-npm/-/is-npm-4.0.0.tgz", 859 | "integrity": "sha512-96ECIfh9xtDDlPylNPXhzjsykHsMJZ18ASpaWzQyBr4YRTcVjUvzaHayDAES2oU/3KpljhHUjtSRNiDwi0F0ig==", 860 | "dev": true 861 | }, 862 | "is-number": { 863 | "version": "7.0.0", 864 | "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", 865 | "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", 866 | "dev": true 867 | }, 868 | "is-obj": { 869 | "version": "2.0.0", 870 | "resolved": "https://registry.npmjs.org/is-obj/-/is-obj-2.0.0.tgz", 871 | "integrity": "sha512-drqDG3cbczxxEJRoOXcOjtdp1J/lyp1mNn0xaznRs8+muBhgQcrnbspox5X5fOw0HnMnbfDzvnEMEtqDEJEo8w==", 872 | "dev": true 873 | }, 874 | "is-path-inside": { 875 | "version": "3.0.2", 876 | "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-3.0.2.tgz", 877 | "integrity": "sha512-/2UGPSgmtqwo1ktx8NDHjuPwZWmHhO+gj0f93EkhLB5RgW9RZevWYYlIkS6zePc6U2WpOdQYIwHe9YC4DWEBVg==", 878 | "dev": true 879 | }, 880 | "is-typedarray": { 881 | "version": "1.0.0", 882 | "resolved": "https://registry.npmjs.org/is-typedarray/-/is-typedarray-1.0.0.tgz", 883 | "integrity": "sha1-5HnICFjfDBsR3dppQPlgEfzaSpo=", 884 | "dev": true 885 | }, 886 | "is-yarn-global": { 887 | "version": "0.3.0", 888 | "resolved": "https://registry.npmjs.org/is-yarn-global/-/is-yarn-global-0.3.0.tgz", 889 | "integrity": "sha512-VjSeb/lHmkoyd8ryPVIKvOCn4D1koMqY+vqyjjUfc3xyKtP4dYOxM44sZrnqQSzSds3xyOrUTLTC9LVCVgLngw==", 890 | "dev": true 891 | }, 892 | "isarray": { 893 | "version": "1.0.0", 894 | "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", 895 | "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE=" 896 | }, 897 | "json-buffer": { 898 | "version": "3.0.0", 899 | "resolved": "https://registry.npmjs.org/json-buffer/-/json-buffer-3.0.0.tgz", 900 | "integrity": "sha1-Wx85evx11ne96Lz8Dkfh+aPZqJg=", 901 | "dev": true 902 | }, 903 | "jsonwebtoken": { 904 | "version": "8.5.1", 905 | "resolved": "https://registry.npmjs.org/jsonwebtoken/-/jsonwebtoken-8.5.1.tgz", 906 | "integrity": "sha512-XjwVfRS6jTMsqYs0EsuJ4LGxXV14zQybNd4L2r0UvbVnSF9Af8x7p5MzbJ90Ioz/9TI41/hTCvznF/loiSzn8w==", 907 | "requires": { 908 | "jws": "^3.2.2", 909 | "lodash.includes": "^4.3.0", 910 | "lodash.isboolean": "^3.0.3", 911 | "lodash.isinteger": "^4.0.4", 912 | "lodash.isnumber": "^3.0.3", 913 | "lodash.isplainobject": "^4.0.6", 914 | "lodash.isstring": "^4.0.1", 915 | "lodash.once": "^4.0.0", 916 | "ms": "^2.1.1", 917 | "semver": "^5.6.0" 918 | }, 919 | "dependencies": { 920 | "ms": { 921 | "version": "2.1.2", 922 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", 923 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==" 924 | } 925 | } 926 | }, 927 | "jwa": { 928 | "version": "1.4.1", 929 | "resolved": "https://registry.npmjs.org/jwa/-/jwa-1.4.1.tgz", 930 | "integrity": "sha512-qiLX/xhEEFKUAJ6FiBMbes3w9ATzyk5W7Hvzpa/SLYdxNtng+gcurvrI7TbACjIXlsJyr05/S1oUhZrc63evQA==", 931 | "requires": { 932 | "buffer-equal-constant-time": "1.0.1", 933 | "ecdsa-sig-formatter": "1.0.11", 934 | "safe-buffer": "^5.0.1" 935 | } 936 | }, 937 | "jws": { 938 | "version": "3.2.2", 939 | "resolved": "https://registry.npmjs.org/jws/-/jws-3.2.2.tgz", 940 | "integrity": "sha512-YHlZCB6lMTllWDtSPHz/ZXTsi8S00usEV6v1tjq8tOUZzw7DpSDWVXjXDre6ed1w/pd495ODpHZYSdkRTsa0HA==", 941 | "requires": { 942 | "jwa": "^1.4.1", 943 | "safe-buffer": "^5.0.1" 944 | } 945 | }, 946 | "kareem": { 947 | "version": "2.3.1", 948 | "resolved": "https://registry.npmjs.org/kareem/-/kareem-2.3.1.tgz", 949 | "integrity": "sha512-l3hLhffs9zqoDe8zjmb/mAN4B8VT3L56EUvKNqLFVs9YlFA+zx7ke1DO8STAdDyYNkeSo1nKmjuvQeI12So8Xw==" 950 | }, 951 | "keygrip": { 952 | "version": "1.1.0", 953 | "resolved": "https://registry.npmjs.org/keygrip/-/keygrip-1.1.0.tgz", 954 | "integrity": "sha512-iYSchDJ+liQ8iwbSI2QqsQOvqv58eJCEanyJPJi+Khyu8smkcKSFUCbPwzFcL7YVtZ6eONjqRX/38caJ7QjRAQ==", 955 | "requires": { 956 | "tsscmp": "1.0.6" 957 | } 958 | }, 959 | "keyv": { 960 | "version": "3.1.0", 961 | "resolved": "https://registry.npmjs.org/keyv/-/keyv-3.1.0.tgz", 962 | "integrity": "sha512-9ykJ/46SN/9KPM/sichzQ7OvXyGDYKGTaDlKMGCAlg2UK8KRy4jb0d8sFc+0Tt0YYnThq8X2RZgCg74RPxgcVA==", 963 | "dev": true, 964 | "requires": { 965 | "json-buffer": "3.0.0" 966 | } 967 | }, 968 | "latest-version": { 969 | "version": "5.1.0", 970 | "resolved": "https://registry.npmjs.org/latest-version/-/latest-version-5.1.0.tgz", 971 | "integrity": "sha512-weT+r0kTkRQdCdYCNtkMwWXQTMEswKrFBkm4ckQOMVhhqhIMI1UT2hMj+1iigIhgSZm5gTmrRXBNoGUgaTY1xA==", 972 | "dev": true, 973 | "requires": { 974 | "package-json": "^6.3.0" 975 | } 976 | }, 977 | "lodash.includes": { 978 | "version": "4.3.0", 979 | "resolved": "https://registry.npmjs.org/lodash.includes/-/lodash.includes-4.3.0.tgz", 980 | "integrity": "sha1-YLuYqHy5I8aMoeUTJUgzFISfVT8=" 981 | }, 982 | "lodash.isboolean": { 983 | "version": "3.0.3", 984 | "resolved": "https://registry.npmjs.org/lodash.isboolean/-/lodash.isboolean-3.0.3.tgz", 985 | "integrity": "sha1-bC4XHbKiV82WgC/UOwGyDV9YcPY=" 986 | }, 987 | "lodash.isinteger": { 988 | "version": "4.0.4", 989 | "resolved": "https://registry.npmjs.org/lodash.isinteger/-/lodash.isinteger-4.0.4.tgz", 990 | "integrity": "sha1-YZwK89A/iwTDH1iChAt3sRzWg0M=" 991 | }, 992 | "lodash.isnumber": { 993 | "version": "3.0.3", 994 | "resolved": "https://registry.npmjs.org/lodash.isnumber/-/lodash.isnumber-3.0.3.tgz", 995 | "integrity": "sha1-POdoEMWSjQM1IwGsKHMX8RwLH/w=" 996 | }, 997 | "lodash.isplainobject": { 998 | "version": "4.0.6", 999 | "resolved": "https://registry.npmjs.org/lodash.isplainobject/-/lodash.isplainobject-4.0.6.tgz", 1000 | "integrity": "sha1-fFJqUtibRcRcxpC4gWO+BJf1UMs=" 1001 | }, 1002 | "lodash.isstring": { 1003 | "version": "4.0.1", 1004 | "resolved": "https://registry.npmjs.org/lodash.isstring/-/lodash.isstring-4.0.1.tgz", 1005 | "integrity": "sha1-1SfftUVuynzJu5XV2ur4i6VKVFE=" 1006 | }, 1007 | "lodash.once": { 1008 | "version": "4.1.1", 1009 | "resolved": "https://registry.npmjs.org/lodash.once/-/lodash.once-4.1.1.tgz", 1010 | "integrity": "sha1-DdOXEhPHxW34gJd9UEyI+0cal6w=" 1011 | }, 1012 | "lowercase-keys": { 1013 | "version": "1.0.1", 1014 | "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-1.0.1.tgz", 1015 | "integrity": "sha512-G2Lj61tXDnVFFOi8VZds+SoQjtQC3dgokKdDG2mTm1tx4m50NUHBOZSBwQQHyy0V12A0JTG4icfZQH+xPyh8VA==", 1016 | "dev": true 1017 | }, 1018 | "make-dir": { 1019 | "version": "3.1.0", 1020 | "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-3.1.0.tgz", 1021 | "integrity": "sha512-g3FeP20LNwhALb/6Cz6Dd4F2ngze0jz7tbzrD2wAV+o9FeNHe4rL+yK2md0J/fiSf1sa1ADhXqi5+oVwOM/eGw==", 1022 | "dev": true, 1023 | "requires": { 1024 | "semver": "^6.0.0" 1025 | }, 1026 | "dependencies": { 1027 | "semver": { 1028 | "version": "6.3.0", 1029 | "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", 1030 | "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", 1031 | "dev": true 1032 | } 1033 | } 1034 | }, 1035 | "media-typer": { 1036 | "version": "0.3.0", 1037 | "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", 1038 | "integrity": "sha1-hxDXrwqmJvj/+hzgAWhUUmMlV0g=" 1039 | }, 1040 | "memory-pager": { 1041 | "version": "1.5.0", 1042 | "resolved": "https://registry.npmjs.org/memory-pager/-/memory-pager-1.5.0.tgz", 1043 | "integrity": "sha512-ZS4Bp4r/Zoeq6+NLJpP+0Zzm0pR8whtGPf1XExKLJBAczGMnSi3It14OiNCStjQjM6NU1okjQGSxgEZN8eBYKg==", 1044 | "optional": true 1045 | }, 1046 | "merge-descriptors": { 1047 | "version": "1.0.1", 1048 | "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz", 1049 | "integrity": "sha1-sAqqVW3YtEVoFQ7J0blT8/kMu2E=" 1050 | }, 1051 | "methods": { 1052 | "version": "1.1.2", 1053 | "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz", 1054 | "integrity": "sha1-VSmk1nZUE07cxSZmVoNbD4Ua/O4=" 1055 | }, 1056 | "mime": { 1057 | "version": "1.6.0", 1058 | "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz", 1059 | "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==" 1060 | }, 1061 | "mime-db": { 1062 | "version": "1.44.0", 1063 | "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.44.0.tgz", 1064 | "integrity": "sha512-/NOTfLrsPBVeH7YtFPgsVWveuL+4SjjYxaQ1xtM1KMFj7HdxlBlxeyNLzhyJVx7r4rZGJAZ/6lkKCitSc/Nmpg==" 1065 | }, 1066 | "mime-types": { 1067 | "version": "2.1.27", 1068 | "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.27.tgz", 1069 | "integrity": "sha512-JIhqnCasI9yD+SsmkquHBxTSEuZdQX5BuQnS2Vc7puQQQ+8yiP5AY5uWhpdv4YL4VM5c6iliiYWPgJ/nJQLp7w==", 1070 | "requires": { 1071 | "mime-db": "1.44.0" 1072 | } 1073 | }, 1074 | "mimic-response": { 1075 | "version": "1.0.1", 1076 | "resolved": "https://registry.npmjs.org/mimic-response/-/mimic-response-1.0.1.tgz", 1077 | "integrity": "sha512-j5EctnkH7amfV/q5Hgmoal1g2QHFJRraOtmx0JpIqkxhBhI/lJSl1nMpQ45hVarwNETOoWEimndZ4QK0RHxuxQ==", 1078 | "dev": true 1079 | }, 1080 | "minimatch": { 1081 | "version": "3.0.4", 1082 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz", 1083 | "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==", 1084 | "dev": true, 1085 | "requires": { 1086 | "brace-expansion": "^1.1.7" 1087 | } 1088 | }, 1089 | "minimist": { 1090 | "version": "1.2.5", 1091 | "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.5.tgz", 1092 | "integrity": "sha512-FM9nNUYrRBAELZQT3xeZQ7fmMOBg6nWNmJKTcgsJeaLstP/UODVpGsr5OhXhhXg6f+qtJ8uiZ+PUxkDWcgIXLw==", 1093 | "dev": true 1094 | }, 1095 | "mongodb": { 1096 | "version": "3.5.7", 1097 | "resolved": "https://registry.npmjs.org/mongodb/-/mongodb-3.5.7.tgz", 1098 | "integrity": "sha512-lMtleRT+vIgY/JhhTn1nyGwnSMmJkJELp+4ZbrjctrnBxuLbj6rmLuJFz8W2xUzUqWmqoyVxJLYuC58ZKpcTYQ==", 1099 | "requires": { 1100 | "bl": "^2.2.0", 1101 | "bson": "^1.1.4", 1102 | "denque": "^1.4.1", 1103 | "require_optional": "^1.0.1", 1104 | "safe-buffer": "^5.1.2", 1105 | "saslprep": "^1.0.0" 1106 | } 1107 | }, 1108 | "mongoose": { 1109 | "version": "5.9.15", 1110 | "resolved": "https://registry.npmjs.org/mongoose/-/mongoose-5.9.15.tgz", 1111 | "integrity": "sha512-dGIDqaQkAJoLl7lsRLy70mDg+VcL1IPOHr/0f23MLF45PtnM5exRdmienfyVjdrSVGgTus+1sMUKef6vSnrDZg==", 1112 | "requires": { 1113 | "bson": "^1.1.4", 1114 | "kareem": "2.3.1", 1115 | "mongodb": "3.5.7", 1116 | "mongoose-legacy-pluralize": "1.0.2", 1117 | "mpath": "0.7.0", 1118 | "mquery": "3.2.2", 1119 | "ms": "2.1.2", 1120 | "regexp-clone": "1.0.0", 1121 | "safe-buffer": "5.1.2", 1122 | "sift": "7.0.1", 1123 | "sliced": "1.0.1" 1124 | }, 1125 | "dependencies": { 1126 | "ms": { 1127 | "version": "2.1.2", 1128 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", 1129 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==" 1130 | } 1131 | } 1132 | }, 1133 | "mongoose-legacy-pluralize": { 1134 | "version": "1.0.2", 1135 | "resolved": "https://registry.npmjs.org/mongoose-legacy-pluralize/-/mongoose-legacy-pluralize-1.0.2.tgz", 1136 | "integrity": "sha512-Yo/7qQU4/EyIS8YDFSeenIvXxZN+ld7YdV9LqFVQJzTLye8unujAWPZ4NWKfFA+RNjh+wvTWKY9Z3E5XM6ZZiQ==" 1137 | }, 1138 | "mpath": { 1139 | "version": "0.7.0", 1140 | "resolved": "https://registry.npmjs.org/mpath/-/mpath-0.7.0.tgz", 1141 | "integrity": "sha512-Aiq04hILxhz1L+f7sjGyn7IxYzWm1zLNNXcfhDtx04kZ2Gk7uvFdgZ8ts1cWa/6d0TQmag2yR8zSGZUmp0tFNg==" 1142 | }, 1143 | "mquery": { 1144 | "version": "3.2.2", 1145 | "resolved": "https://registry.npmjs.org/mquery/-/mquery-3.2.2.tgz", 1146 | "integrity": "sha512-XB52992COp0KP230I3qloVUbkLUxJIu328HBP2t2EsxSFtf4W1HPSOBWOXf1bqxK4Xbb66lfMJ+Bpfd9/yZE1Q==", 1147 | "requires": { 1148 | "bluebird": "3.5.1", 1149 | "debug": "3.1.0", 1150 | "regexp-clone": "^1.0.0", 1151 | "safe-buffer": "5.1.2", 1152 | "sliced": "1.0.1" 1153 | }, 1154 | "dependencies": { 1155 | "debug": { 1156 | "version": "3.1.0", 1157 | "resolved": "https://registry.npmjs.org/debug/-/debug-3.1.0.tgz", 1158 | "integrity": "sha512-OX8XqP7/1a9cqkxYw2yXss15f26NKWBpDXQd0/uK/KPqdQhxbPa994hnzjcE2VqQpDslf55723cKPUOGSmMY3g==", 1159 | "requires": { 1160 | "ms": "2.0.0" 1161 | } 1162 | } 1163 | } 1164 | }, 1165 | "ms": { 1166 | "version": "2.0.0", 1167 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", 1168 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" 1169 | }, 1170 | "negotiator": { 1171 | "version": "0.6.2", 1172 | "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.2.tgz", 1173 | "integrity": "sha512-hZXc7K2e+PgeI1eDBe/10Ard4ekbfrrqG8Ep+8Jmf4JID2bNg7NvCPOZN+kfF574pFQI7mum2AUqDidoKqcTOw==" 1174 | }, 1175 | "nodemailer": { 1176 | "version": "6.4.6", 1177 | "resolved": "https://registry.npmjs.org/nodemailer/-/nodemailer-6.4.6.tgz", 1178 | "integrity": "sha512-/kJ+FYVEm2HuUlw87hjSqTss+GU35D4giOpdSfGp7DO+5h6RlJj7R94YaYHOkoxu1CSaM0d3WRBtCzwXrY6MKA==" 1179 | }, 1180 | "nodemon": { 1181 | "version": "2.0.4", 1182 | "resolved": "https://registry.npmjs.org/nodemon/-/nodemon-2.0.4.tgz", 1183 | "integrity": "sha512-Ltced+hIfTmaS28Zjv1BM552oQ3dbwPqI4+zI0SLgq+wpJhSyqgYude/aZa/3i31VCQWMfXJVxvu86abcam3uQ==", 1184 | "dev": true, 1185 | "requires": { 1186 | "chokidar": "^3.2.2", 1187 | "debug": "^3.2.6", 1188 | "ignore-by-default": "^1.0.1", 1189 | "minimatch": "^3.0.4", 1190 | "pstree.remy": "^1.1.7", 1191 | "semver": "^5.7.1", 1192 | "supports-color": "^5.5.0", 1193 | "touch": "^3.1.0", 1194 | "undefsafe": "^2.0.2", 1195 | "update-notifier": "^4.0.0" 1196 | }, 1197 | "dependencies": { 1198 | "debug": { 1199 | "version": "3.2.6", 1200 | "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.6.tgz", 1201 | "integrity": "sha512-mel+jf7nrtEl5Pn1Qx46zARXKDpBbvzezse7p7LqINmdoIk8PYP5SySaxEmYv6TZ0JyEKA1hsCId6DIhgITtWQ==", 1202 | "dev": true, 1203 | "requires": { 1204 | "ms": "^2.1.1" 1205 | } 1206 | }, 1207 | "ms": { 1208 | "version": "2.1.2", 1209 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", 1210 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", 1211 | "dev": true 1212 | } 1213 | } 1214 | }, 1215 | "nopt": { 1216 | "version": "1.0.10", 1217 | "resolved": "https://registry.npmjs.org/nopt/-/nopt-1.0.10.tgz", 1218 | "integrity": "sha1-bd0hvSoxQXuScn3Vhfim83YI6+4=", 1219 | "dev": true, 1220 | "requires": { 1221 | "abbrev": "1" 1222 | } 1223 | }, 1224 | "normalize-path": { 1225 | "version": "3.0.0", 1226 | "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", 1227 | "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==", 1228 | "dev": true 1229 | }, 1230 | "normalize-url": { 1231 | "version": "4.5.0", 1232 | "resolved": "https://registry.npmjs.org/normalize-url/-/normalize-url-4.5.0.tgz", 1233 | "integrity": "sha512-2s47yzUxdexf1OhyRi4Em83iQk0aPvwTddtFz4hnSSw9dCEsLEGf6SwIO8ss/19S9iBb5sJaOuTvTGDeZI00BQ==", 1234 | "dev": true 1235 | }, 1236 | "object-assign": { 1237 | "version": "4.1.1", 1238 | "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", 1239 | "integrity": "sha1-IQmtx5ZYh8/AXLvUQsrIv7s2CGM=" 1240 | }, 1241 | "on-finished": { 1242 | "version": "2.3.0", 1243 | "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.3.0.tgz", 1244 | "integrity": "sha1-IPEzZIGwg811M3mSoWlxqi2QaUc=", 1245 | "requires": { 1246 | "ee-first": "1.1.1" 1247 | } 1248 | }, 1249 | "on-headers": { 1250 | "version": "1.0.2", 1251 | "resolved": "https://registry.npmjs.org/on-headers/-/on-headers-1.0.2.tgz", 1252 | "integrity": "sha512-pZAE+FJLoyITytdqK0U5s+FIpjN0JP3OzFi/u8Rx+EV5/W+JTWGXG8xFzevE7AjBfDqHv/8vL8qQsIhHnqRkrA==" 1253 | }, 1254 | "once": { 1255 | "version": "1.4.0", 1256 | "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", 1257 | "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=", 1258 | "dev": true, 1259 | "requires": { 1260 | "wrappy": "1" 1261 | } 1262 | }, 1263 | "p-cancelable": { 1264 | "version": "1.1.0", 1265 | "resolved": "https://registry.npmjs.org/p-cancelable/-/p-cancelable-1.1.0.tgz", 1266 | "integrity": "sha512-s73XxOZ4zpt1edZYZzvhqFa6uvQc1vwUa0K0BdtIZgQMAJj9IbebH+JkgKZc9h+B05PKHLOTl4ajG1BmNrVZlw==", 1267 | "dev": true 1268 | }, 1269 | "package-json": { 1270 | "version": "6.5.0", 1271 | "resolved": "https://registry.npmjs.org/package-json/-/package-json-6.5.0.tgz", 1272 | "integrity": "sha512-k3bdm2n25tkyxcjSKzB5x8kfVxlMdgsbPr0GkZcwHsLpba6cBjqCt1KlcChKEvxHIcTB1FVMuwoijZ26xex5MQ==", 1273 | "dev": true, 1274 | "requires": { 1275 | "got": "^9.6.0", 1276 | "registry-auth-token": "^4.0.0", 1277 | "registry-url": "^5.0.0", 1278 | "semver": "^6.2.0" 1279 | }, 1280 | "dependencies": { 1281 | "semver": { 1282 | "version": "6.3.0", 1283 | "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", 1284 | "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", 1285 | "dev": true 1286 | } 1287 | } 1288 | }, 1289 | "parseurl": { 1290 | "version": "1.3.3", 1291 | "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz", 1292 | "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==" 1293 | }, 1294 | "path-to-regexp": { 1295 | "version": "0.1.7", 1296 | "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz", 1297 | "integrity": "sha1-32BBeABfUi8V60SQ5yR6G/qmf4w=" 1298 | }, 1299 | "picomatch": { 1300 | "version": "2.2.2", 1301 | "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.2.2.tgz", 1302 | "integrity": "sha512-q0M/9eZHzmr0AulXyPwNfZjtwZ/RBZlbN3K3CErVrk50T2ASYI7Bye0EvekFY3IP1Nt2DHu0re+V2ZHIpMkuWg==", 1303 | "dev": true 1304 | }, 1305 | "prepend-http": { 1306 | "version": "2.0.0", 1307 | "resolved": "https://registry.npmjs.org/prepend-http/-/prepend-http-2.0.0.tgz", 1308 | "integrity": "sha1-6SQ0v6XqjBn0HN/UAddBo8gZ2Jc=", 1309 | "dev": true 1310 | }, 1311 | "process-nextick-args": { 1312 | "version": "2.0.1", 1313 | "resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-2.0.1.tgz", 1314 | "integrity": "sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag==" 1315 | }, 1316 | "proxy-addr": { 1317 | "version": "2.0.6", 1318 | "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.6.tgz", 1319 | "integrity": "sha512-dh/frvCBVmSsDYzw6n926jv974gddhkFPfiN8hPOi30Wax25QZyZEGveluCgliBnqmuM+UJmBErbAUFIoDbjOw==", 1320 | "requires": { 1321 | "forwarded": "~0.1.2", 1322 | "ipaddr.js": "1.9.1" 1323 | } 1324 | }, 1325 | "pstree.remy": { 1326 | "version": "1.1.8", 1327 | "resolved": "https://registry.npmjs.org/pstree.remy/-/pstree.remy-1.1.8.tgz", 1328 | "integrity": "sha512-77DZwxQmxKnu3aR542U+X8FypNzbfJ+C5XQDk3uWjWxn6151aIMGthWYRXTqT1E5oJvg+ljaa2OJi+VfvCOQ8w==", 1329 | "dev": true 1330 | }, 1331 | "pump": { 1332 | "version": "3.0.0", 1333 | "resolved": "https://registry.npmjs.org/pump/-/pump-3.0.0.tgz", 1334 | "integrity": "sha512-LwZy+p3SFs1Pytd/jYct4wpv49HiYCqd9Rlc5ZVdk0V+8Yzv6jR5Blk3TRmPL1ft69TxP0IMZGJ+WPFU2BFhww==", 1335 | "dev": true, 1336 | "requires": { 1337 | "end-of-stream": "^1.1.0", 1338 | "once": "^1.3.1" 1339 | } 1340 | }, 1341 | "pupa": { 1342 | "version": "2.0.1", 1343 | "resolved": "https://registry.npmjs.org/pupa/-/pupa-2.0.1.tgz", 1344 | "integrity": "sha512-hEJH0s8PXLY/cdXh66tNEQGndDrIKNqNC5xmrysZy3i5C3oEoLna7YAOad+7u125+zH1HNXUmGEkrhb3c2VriA==", 1345 | "dev": true, 1346 | "requires": { 1347 | "escape-goat": "^2.0.0" 1348 | } 1349 | }, 1350 | "qs": { 1351 | "version": "6.7.0", 1352 | "resolved": "https://registry.npmjs.org/qs/-/qs-6.7.0.tgz", 1353 | "integrity": "sha512-VCdBRNFTX1fyE7Nb6FYoURo/SPe62QCaAyzJvUjwRaIsc+NePBEniHlvxFmmX56+HZphIGtV0XeCirBtpDrTyQ==" 1354 | }, 1355 | "random-bytes": { 1356 | "version": "1.0.0", 1357 | "resolved": "https://registry.npmjs.org/random-bytes/-/random-bytes-1.0.0.tgz", 1358 | "integrity": "sha1-T2ih3Arli9P7lYSMMDJNt11kNgs=" 1359 | }, 1360 | "range-parser": { 1361 | "version": "1.2.1", 1362 | "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz", 1363 | "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==" 1364 | }, 1365 | "raw-body": { 1366 | "version": "2.4.0", 1367 | "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.4.0.tgz", 1368 | "integrity": "sha512-4Oz8DUIwdvoa5qMJelxipzi/iJIi40O5cGV1wNYp5hvZP8ZN0T+jiNkL0QepXs+EsQ9XJ8ipEDoiH70ySUJP3Q==", 1369 | "requires": { 1370 | "bytes": "3.1.0", 1371 | "http-errors": "1.7.2", 1372 | "iconv-lite": "0.4.24", 1373 | "unpipe": "1.0.0" 1374 | } 1375 | }, 1376 | "rc": { 1377 | "version": "1.2.8", 1378 | "resolved": "https://registry.npmjs.org/rc/-/rc-1.2.8.tgz", 1379 | "integrity": "sha512-y3bGgqKj3QBdxLbLkomlohkvsA8gdAiUQlSBJnBhfn+BPxg4bc62d8TcBW15wavDfgexCgccckhcZvywyQYPOw==", 1380 | "dev": true, 1381 | "requires": { 1382 | "deep-extend": "^0.6.0", 1383 | "ini": "~1.3.0", 1384 | "minimist": "^1.2.0", 1385 | "strip-json-comments": "~2.0.1" 1386 | } 1387 | }, 1388 | "readable-stream": { 1389 | "version": "2.3.7", 1390 | "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.7.tgz", 1391 | "integrity": "sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw==", 1392 | "requires": { 1393 | "core-util-is": "~1.0.0", 1394 | "inherits": "~2.0.3", 1395 | "isarray": "~1.0.0", 1396 | "process-nextick-args": "~2.0.0", 1397 | "safe-buffer": "~5.1.1", 1398 | "string_decoder": "~1.1.1", 1399 | "util-deprecate": "~1.0.1" 1400 | } 1401 | }, 1402 | "readdirp": { 1403 | "version": "3.4.0", 1404 | "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.4.0.tgz", 1405 | "integrity": "sha512-0xe001vZBnJEK+uKcj8qOhyAKPzIT+gStxWr3LCB0DwcXR5NZJ3IaC+yGnHCYzB/S7ov3m3EEbZI2zeNvX+hGQ==", 1406 | "dev": true, 1407 | "requires": { 1408 | "picomatch": "^2.2.1" 1409 | } 1410 | }, 1411 | "regexp-clone": { 1412 | "version": "1.0.0", 1413 | "resolved": "https://registry.npmjs.org/regexp-clone/-/regexp-clone-1.0.0.tgz", 1414 | "integrity": "sha512-TuAasHQNamyyJ2hb97IuBEif4qBHGjPHBS64sZwytpLEqtBQ1gPJTnOaQ6qmpET16cK14kkjbazl6+p0RRv0yw==" 1415 | }, 1416 | "registry-auth-token": { 1417 | "version": "4.1.1", 1418 | "resolved": "https://registry.npmjs.org/registry-auth-token/-/registry-auth-token-4.1.1.tgz", 1419 | "integrity": "sha512-9bKS7nTl9+/A1s7tnPeGrUpRcVY+LUh7bfFgzpndALdPfXQBfQV77rQVtqgUV3ti4vc/Ik81Ex8UJDWDQ12zQA==", 1420 | "dev": true, 1421 | "requires": { 1422 | "rc": "^1.2.8" 1423 | } 1424 | }, 1425 | "registry-url": { 1426 | "version": "5.1.0", 1427 | "resolved": "https://registry.npmjs.org/registry-url/-/registry-url-5.1.0.tgz", 1428 | "integrity": "sha512-8acYXXTI0AkQv6RAOjE3vOaIXZkT9wo4LOFbBKYQEEnnMNBpKqdUrI6S4NT0KPIo/WVvJ5tE/X5LF/TQUf0ekw==", 1429 | "dev": true, 1430 | "requires": { 1431 | "rc": "^1.2.8" 1432 | } 1433 | }, 1434 | "require_optional": { 1435 | "version": "1.0.1", 1436 | "resolved": "https://registry.npmjs.org/require_optional/-/require_optional-1.0.1.tgz", 1437 | "integrity": "sha512-qhM/y57enGWHAe3v/NcwML6a3/vfESLe/sGM2dII+gEO0BpKRUkWZow/tyloNqJyN6kXSl3RyyM8Ll5D/sJP8g==", 1438 | "requires": { 1439 | "resolve-from": "^2.0.0", 1440 | "semver": "^5.1.0" 1441 | } 1442 | }, 1443 | "resolve-from": { 1444 | "version": "2.0.0", 1445 | "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-2.0.0.tgz", 1446 | "integrity": "sha1-lICrIOlP+h2egKgEx+oUdhGWa1c=" 1447 | }, 1448 | "responselike": { 1449 | "version": "1.0.2", 1450 | "resolved": "https://registry.npmjs.org/responselike/-/responselike-1.0.2.tgz", 1451 | "integrity": "sha1-kYcg7ztjHFZCvgaPFa3lpG9Loec=", 1452 | "dev": true, 1453 | "requires": { 1454 | "lowercase-keys": "^1.0.0" 1455 | } 1456 | }, 1457 | "rndm": { 1458 | "version": "1.2.0", 1459 | "resolved": "https://registry.npmjs.org/rndm/-/rndm-1.2.0.tgz", 1460 | "integrity": "sha1-8z/pz7Urv9UgqhgyO8ZdsRCht2w=" 1461 | }, 1462 | "safe-buffer": { 1463 | "version": "5.1.2", 1464 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", 1465 | "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==" 1466 | }, 1467 | "safer-buffer": { 1468 | "version": "2.1.2", 1469 | "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", 1470 | "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==" 1471 | }, 1472 | "saslprep": { 1473 | "version": "1.0.3", 1474 | "resolved": "https://registry.npmjs.org/saslprep/-/saslprep-1.0.3.tgz", 1475 | "integrity": "sha512-/MY/PEMbk2SuY5sScONwhUDsV2p77Znkb/q3nSVstq/yQzYJOH/Azh29p9oJLsl3LnQwSvZDKagDGBsBwSooag==", 1476 | "optional": true, 1477 | "requires": { 1478 | "sparse-bitfield": "^3.0.3" 1479 | } 1480 | }, 1481 | "semver": { 1482 | "version": "5.7.1", 1483 | "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", 1484 | "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==" 1485 | }, 1486 | "semver-diff": { 1487 | "version": "3.1.1", 1488 | "resolved": "https://registry.npmjs.org/semver-diff/-/semver-diff-3.1.1.tgz", 1489 | "integrity": "sha512-GX0Ix/CJcHyB8c4ykpHGIAvLyOwOobtM/8d+TQkAd81/bEjgPHrfba41Vpesr7jX/t8Uh+R3EX9eAS5be+jQYg==", 1490 | "dev": true, 1491 | "requires": { 1492 | "semver": "^6.3.0" 1493 | }, 1494 | "dependencies": { 1495 | "semver": { 1496 | "version": "6.3.0", 1497 | "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", 1498 | "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", 1499 | "dev": true 1500 | } 1501 | } 1502 | }, 1503 | "send": { 1504 | "version": "0.17.1", 1505 | "resolved": "https://registry.npmjs.org/send/-/send-0.17.1.tgz", 1506 | "integrity": "sha512-BsVKsiGcQMFwT8UxypobUKyv7irCNRHk1T0G680vk88yf6LBByGcZJOTJCrTP2xVN6yI+XjPJcNuE3V4fT9sAg==", 1507 | "requires": { 1508 | "debug": "2.6.9", 1509 | "depd": "~1.1.2", 1510 | "destroy": "~1.0.4", 1511 | "encodeurl": "~1.0.2", 1512 | "escape-html": "~1.0.3", 1513 | "etag": "~1.8.1", 1514 | "fresh": "0.5.2", 1515 | "http-errors": "~1.7.2", 1516 | "mime": "1.6.0", 1517 | "ms": "2.1.1", 1518 | "on-finished": "~2.3.0", 1519 | "range-parser": "~1.2.1", 1520 | "statuses": "~1.5.0" 1521 | }, 1522 | "dependencies": { 1523 | "ms": { 1524 | "version": "2.1.1", 1525 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.1.tgz", 1526 | "integrity": "sha512-tgp+dl5cGk28utYktBsrFqA7HKgrhgPsg6Z/EfhWI4gl1Hwq8B/GmY/0oXZ6nF8hDVesS/FpnYaD/kOWhYQvyg==" 1527 | } 1528 | } 1529 | }, 1530 | "serve-static": { 1531 | "version": "1.14.1", 1532 | "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.14.1.tgz", 1533 | "integrity": "sha512-JMrvUwE54emCYWlTI+hGrGv5I8dEwmco/00EvkzIIsR7MqrHonbD9pO2MOfFnpFntl7ecpZs+3mW+XbQZu9QCg==", 1534 | "requires": { 1535 | "encodeurl": "~1.0.2", 1536 | "escape-html": "~1.0.3", 1537 | "parseurl": "~1.3.3", 1538 | "send": "0.17.1" 1539 | } 1540 | }, 1541 | "setprototypeof": { 1542 | "version": "1.1.1", 1543 | "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.1.tgz", 1544 | "integrity": "sha512-JvdAWfbXeIGaZ9cILp38HntZSFSo3mWg6xGcJJsd+d4aRMOqauag1C63dJfDw7OaMYwEbHMOxEZ1lqVRYP2OAw==" 1545 | }, 1546 | "sift": { 1547 | "version": "7.0.1", 1548 | "resolved": "https://registry.npmjs.org/sift/-/sift-7.0.1.tgz", 1549 | "integrity": "sha512-oqD7PMJ+uO6jV9EQCl0LrRw1OwsiPsiFQR5AR30heR+4Dl7jBBbDLnNvWiak20tzZlSE1H7RB30SX/1j/YYT7g==" 1550 | }, 1551 | "signal-exit": { 1552 | "version": "3.0.3", 1553 | "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.3.tgz", 1554 | "integrity": "sha512-VUJ49FC8U1OxwZLxIbTTrDvLnf/6TDgxZcK8wxR8zs13xpx7xbG60ndBlhNrFi2EMuFRoeDoJO7wthSLq42EjA==", 1555 | "dev": true 1556 | }, 1557 | "sliced": { 1558 | "version": "1.0.1", 1559 | "resolved": "https://registry.npmjs.org/sliced/-/sliced-1.0.1.tgz", 1560 | "integrity": "sha1-CzpmK10Ewxd7GSa+qCsD+Dei70E=" 1561 | }, 1562 | "sparse-bitfield": { 1563 | "version": "3.0.3", 1564 | "resolved": "https://registry.npmjs.org/sparse-bitfield/-/sparse-bitfield-3.0.3.tgz", 1565 | "integrity": "sha1-/0rm5oZWBWuks+eSqzM004JzyhE=", 1566 | "optional": true, 1567 | "requires": { 1568 | "memory-pager": "^1.0.2" 1569 | } 1570 | }, 1571 | "statuses": { 1572 | "version": "1.5.0", 1573 | "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.5.0.tgz", 1574 | "integrity": "sha1-Fhx9rBd2Wf2YEfQ3cfqZOBR4Yow=" 1575 | }, 1576 | "string-width": { 1577 | "version": "4.2.0", 1578 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.0.tgz", 1579 | "integrity": "sha512-zUz5JD+tgqtuDjMhwIg5uFVV3dtqZ9yQJlZVfq4I01/K5Paj5UHj7VyrQOJvzawSVlKpObApbfD0Ed6yJc+1eg==", 1580 | "dev": true, 1581 | "requires": { 1582 | "emoji-regex": "^8.0.0", 1583 | "is-fullwidth-code-point": "^3.0.0", 1584 | "strip-ansi": "^6.0.0" 1585 | }, 1586 | "dependencies": { 1587 | "ansi-regex": { 1588 | "version": "5.0.0", 1589 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.0.tgz", 1590 | "integrity": "sha512-bY6fj56OUQ0hU1KjFNDQuJFezqKdrAyFdIevADiqrWHwSlbmBNMHp5ak2f40Pm8JTFyM2mqxkG6ngkHO11f/lg==", 1591 | "dev": true 1592 | }, 1593 | "emoji-regex": { 1594 | "version": "8.0.0", 1595 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", 1596 | "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", 1597 | "dev": true 1598 | }, 1599 | "is-fullwidth-code-point": { 1600 | "version": "3.0.0", 1601 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", 1602 | "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", 1603 | "dev": true 1604 | }, 1605 | "strip-ansi": { 1606 | "version": "6.0.0", 1607 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.0.tgz", 1608 | "integrity": "sha512-AuvKTrTfQNYNIctbR1K/YGTR1756GycPsg7b9bdV9Duqur4gv6aKqHXah67Z8ImS7WEz5QVcOtlfW2rZEugt6w==", 1609 | "dev": true, 1610 | "requires": { 1611 | "ansi-regex": "^5.0.0" 1612 | } 1613 | } 1614 | } 1615 | }, 1616 | "string_decoder": { 1617 | "version": "1.1.1", 1618 | "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz", 1619 | "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==", 1620 | "requires": { 1621 | "safe-buffer": "~5.1.0" 1622 | } 1623 | }, 1624 | "strip-ansi": { 1625 | "version": "5.2.0", 1626 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", 1627 | "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", 1628 | "dev": true, 1629 | "requires": { 1630 | "ansi-regex": "^4.1.0" 1631 | } 1632 | }, 1633 | "strip-json-comments": { 1634 | "version": "2.0.1", 1635 | "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-2.0.1.tgz", 1636 | "integrity": "sha1-PFMZQukIwml8DsNEhYwobHygpgo=", 1637 | "dev": true 1638 | }, 1639 | "supports-color": { 1640 | "version": "5.5.0", 1641 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", 1642 | "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", 1643 | "dev": true, 1644 | "requires": { 1645 | "has-flag": "^3.0.0" 1646 | } 1647 | }, 1648 | "term-size": { 1649 | "version": "2.2.0", 1650 | "resolved": "https://registry.npmjs.org/term-size/-/term-size-2.2.0.tgz", 1651 | "integrity": "sha512-a6sumDlzyHVJWb8+YofY4TW112G6p2FCPEAFk+59gIYHv3XHRhm9ltVQ9kli4hNWeQBwSpe8cRN25x0ROunMOw==", 1652 | "dev": true 1653 | }, 1654 | "to-readable-stream": { 1655 | "version": "1.0.0", 1656 | "resolved": "https://registry.npmjs.org/to-readable-stream/-/to-readable-stream-1.0.0.tgz", 1657 | "integrity": "sha512-Iq25XBt6zD5npPhlLVXGFN3/gyR2/qODcKNNyTMd4vbm39HUaOiAM4PMq0eMVC/Tkxz+Zjdsc55g9yyz+Yq00Q==", 1658 | "dev": true 1659 | }, 1660 | "to-regex-range": { 1661 | "version": "5.0.1", 1662 | "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", 1663 | "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", 1664 | "dev": true, 1665 | "requires": { 1666 | "is-number": "^7.0.0" 1667 | } 1668 | }, 1669 | "toidentifier": { 1670 | "version": "1.0.0", 1671 | "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.0.tgz", 1672 | "integrity": "sha512-yaOH/Pk/VEhBWWTlhI+qXxDFXlejDGcQipMlyxda9nthulaxLZUNcUqFxokp0vcYnvteJln5FNQDRrxj3YcbVw==" 1673 | }, 1674 | "touch": { 1675 | "version": "3.1.0", 1676 | "resolved": "https://registry.npmjs.org/touch/-/touch-3.1.0.tgz", 1677 | "integrity": "sha512-WBx8Uy5TLtOSRtIq+M03/sKDrXCLHxwDcquSP2c43Le03/9serjQBIztjRz6FkJez9D/hleyAXTBGLwwZUw9lA==", 1678 | "dev": true, 1679 | "requires": { 1680 | "nopt": "~1.0.10" 1681 | } 1682 | }, 1683 | "tsscmp": { 1684 | "version": "1.0.6", 1685 | "resolved": "https://registry.npmjs.org/tsscmp/-/tsscmp-1.0.6.tgz", 1686 | "integrity": "sha512-LxhtAkPDTkVCMQjt2h6eBVY28KCjikZqZfMcC15YBeNjkgUpdCfBu5HoiOTDu86v6smE8yOjyEktJ8hlbANHQA==" 1687 | }, 1688 | "type-fest": { 1689 | "version": "0.8.1", 1690 | "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.8.1.tgz", 1691 | "integrity": "sha512-4dbzIzqvjtgiM5rw1k5rEHtBANKmdudhGyBEajN01fEyhaAIhsoKNy6y7+IN93IfpFtwY9iqi7kD+xwKhQsNJA==" 1692 | }, 1693 | "type-is": { 1694 | "version": "1.6.18", 1695 | "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz", 1696 | "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==", 1697 | "requires": { 1698 | "media-typer": "0.3.0", 1699 | "mime-types": "~2.1.24" 1700 | } 1701 | }, 1702 | "typedarray-to-buffer": { 1703 | "version": "3.1.5", 1704 | "resolved": "https://registry.npmjs.org/typedarray-to-buffer/-/typedarray-to-buffer-3.1.5.tgz", 1705 | "integrity": "sha512-zdu8XMNEDepKKR+XYOXAVPtWui0ly0NtohUscw+UmaHiAWT8hrV1rr//H6V+0DvJ3OQ19S979M0laLfX8rm82Q==", 1706 | "dev": true, 1707 | "requires": { 1708 | "is-typedarray": "^1.0.0" 1709 | } 1710 | }, 1711 | "uid-safe": { 1712 | "version": "2.1.5", 1713 | "resolved": "https://registry.npmjs.org/uid-safe/-/uid-safe-2.1.5.tgz", 1714 | "integrity": "sha512-KPHm4VL5dDXKz01UuEd88Df+KzynaohSL9fBh096KWAxSKZQDI2uBrVqtvRM4rwrIrRRKsdLNML/lnaaVSRioA==", 1715 | "requires": { 1716 | "random-bytes": "~1.0.0" 1717 | } 1718 | }, 1719 | "undefsafe": { 1720 | "version": "2.0.3", 1721 | "resolved": "https://registry.npmjs.org/undefsafe/-/undefsafe-2.0.3.tgz", 1722 | "integrity": "sha512-nrXZwwXrD/T/JXeygJqdCO6NZZ1L66HrxM/Z7mIq2oPanoN0F1nLx3lwJMu6AwJY69hdixaFQOuoYsMjE5/C2A==", 1723 | "dev": true, 1724 | "requires": { 1725 | "debug": "^2.2.0" 1726 | } 1727 | }, 1728 | "unique-string": { 1729 | "version": "2.0.0", 1730 | "resolved": "https://registry.npmjs.org/unique-string/-/unique-string-2.0.0.tgz", 1731 | "integrity": "sha512-uNaeirEPvpZWSgzwsPGtU2zVSTrn/8L5q/IexZmH0eH6SA73CmAA5U4GwORTxQAZs95TAXLNqeLoPPNO5gZfWg==", 1732 | "dev": true, 1733 | "requires": { 1734 | "crypto-random-string": "^2.0.0" 1735 | }, 1736 | "dependencies": { 1737 | "crypto-random-string": { 1738 | "version": "2.0.0", 1739 | "resolved": "https://registry.npmjs.org/crypto-random-string/-/crypto-random-string-2.0.0.tgz", 1740 | "integrity": "sha512-v1plID3y9r/lPhviJ1wrXpLeyUIGAZ2SHNYTEapm7/8A9nLPoyvVp3RK/EPFqn5kEznyWgYZNsRtYYIWbuG8KA==", 1741 | "dev": true 1742 | } 1743 | } 1744 | }, 1745 | "unpipe": { 1746 | "version": "1.0.0", 1747 | "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", 1748 | "integrity": "sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw=" 1749 | }, 1750 | "update-notifier": { 1751 | "version": "4.1.0", 1752 | "resolved": "https://registry.npmjs.org/update-notifier/-/update-notifier-4.1.0.tgz", 1753 | "integrity": "sha512-w3doE1qtI0/ZmgeoDoARmI5fjDoT93IfKgEGqm26dGUOh8oNpaSTsGNdYRN/SjOuo10jcJGwkEL3mroKzktkew==", 1754 | "dev": true, 1755 | "requires": { 1756 | "boxen": "^4.2.0", 1757 | "chalk": "^3.0.0", 1758 | "configstore": "^5.0.1", 1759 | "has-yarn": "^2.1.0", 1760 | "import-lazy": "^2.1.0", 1761 | "is-ci": "^2.0.0", 1762 | "is-installed-globally": "^0.3.1", 1763 | "is-npm": "^4.0.0", 1764 | "is-yarn-global": "^0.3.0", 1765 | "latest-version": "^5.0.0", 1766 | "pupa": "^2.0.1", 1767 | "semver-diff": "^3.1.1", 1768 | "xdg-basedir": "^4.0.0" 1769 | } 1770 | }, 1771 | "url-parse-lax": { 1772 | "version": "3.0.0", 1773 | "resolved": "https://registry.npmjs.org/url-parse-lax/-/url-parse-lax-3.0.0.tgz", 1774 | "integrity": "sha1-FrXK/Afb42dsGxmZF3gj1lA6yww=", 1775 | "dev": true, 1776 | "requires": { 1777 | "prepend-http": "^2.0.0" 1778 | } 1779 | }, 1780 | "util-deprecate": { 1781 | "version": "1.0.2", 1782 | "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", 1783 | "integrity": "sha1-RQ1Nyfpw3nMnYvvS1KKJgUGaDM8=" 1784 | }, 1785 | "utils-merge": { 1786 | "version": "1.0.1", 1787 | "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz", 1788 | "integrity": "sha1-n5VxD1CiZ5R7LMwSR0HBAoQn5xM=" 1789 | }, 1790 | "vary": { 1791 | "version": "1.1.2", 1792 | "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz", 1793 | "integrity": "sha1-IpnwLG3tMNSllhsLn3RSShj2NPw=" 1794 | }, 1795 | "widest-line": { 1796 | "version": "3.1.0", 1797 | "resolved": "https://registry.npmjs.org/widest-line/-/widest-line-3.1.0.tgz", 1798 | "integrity": "sha512-NsmoXalsWVDMGupxZ5R08ka9flZjjiLvHVAWYOKtiKM8ujtZWr9cRffak+uSE48+Ob8ObalXpwyeUiyDD6QFgg==", 1799 | "dev": true, 1800 | "requires": { 1801 | "string-width": "^4.0.0" 1802 | } 1803 | }, 1804 | "wrappy": { 1805 | "version": "1.0.2", 1806 | "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", 1807 | "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=", 1808 | "dev": true 1809 | }, 1810 | "write-file-atomic": { 1811 | "version": "3.0.3", 1812 | "resolved": "https://registry.npmjs.org/write-file-atomic/-/write-file-atomic-3.0.3.tgz", 1813 | "integrity": "sha512-AvHcyZ5JnSfq3ioSyjrBkH9yW4m7Ayk8/9My/DD9onKeu/94fwrMocemO2QAJFAlnnDN+ZDS+ZjAR5ua1/PV/Q==", 1814 | "dev": true, 1815 | "requires": { 1816 | "imurmurhash": "^0.1.4", 1817 | "is-typedarray": "^1.0.0", 1818 | "signal-exit": "^3.0.2", 1819 | "typedarray-to-buffer": "^3.1.5" 1820 | } 1821 | }, 1822 | "xdg-basedir": { 1823 | "version": "4.0.0", 1824 | "resolved": "https://registry.npmjs.org/xdg-basedir/-/xdg-basedir-4.0.0.tgz", 1825 | "integrity": "sha512-PSNhEJDejZYV7h50BohL09Er9VaIefr2LMAf3OEmpCkjOi34eYyQYAXUTjEQtZJTKcF0E2UKTh+osDLsgNim9Q==", 1826 | "dev": true 1827 | } 1828 | } 1829 | } 1830 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "vuex_bootstrap_node-express_mongodb_w-auth_starter", 3 | "version": "1.0.0", 4 | "description": "", 5 | "main": "index.js", 6 | "scripts": { 7 | "start": "node server/index.js", 8 | "dev": "nodemon server/index.js" 9 | }, 10 | "author": "Christopher Liedtke", 11 | "license": "MIT", 12 | "dependencies": { 13 | "bcryptjs": "^2.4.3", 14 | "compression": "^1.7.4", 15 | "cookie-session": "^1.4.0", 16 | "cors": "^2.8.5", 17 | "crypto-random-string": "^3.2.0", 18 | "csurf": "^1.11.0", 19 | "express": "^4.17.1", 20 | "express-session": "^1.17.1", 21 | "jsonwebtoken": "^8.5.1", 22 | "mongoose": "^5.9.15", 23 | "nodemailer": "^6.4.6" 24 | }, 25 | "devDependencies": { 26 | "nodemon": "^2.0.4" 27 | } 28 | } 29 | -------------------------------------------------------------------------------- /server/.gitignore: -------------------------------------------------------------------------------- 1 | utils/secrets* -------------------------------------------------------------------------------- /server/index.js: -------------------------------------------------------------------------------- 1 | const express = require("express"); 2 | const app = express(); 3 | 4 | const cors = require("cors"); 5 | const compression = require("compression"); 6 | 7 | const csurf = require("csurf"); 8 | 9 | let secrets, port; 10 | if (process.env.NODE_ENV == "production") { 11 | secrets = process.env; 12 | port = process.env.PORT; 13 | } else { 14 | secrets = require("./utils/secrets"); 15 | port = 5000; 16 | } 17 | 18 | // #mongoDB 19 | require("./utils/db"); 20 | 21 | // #Middleware 22 | app.use(compression()); 23 | app.use(cors()); 24 | app.use(express.json()); 25 | app.use((req, res, next) => { 26 | res.locals.secrets = secrets; 27 | next(); 28 | }); 29 | 30 | // Serve static ressources in production 31 | app.use(express.static(__dirname + "/public")); 32 | 33 | // #Cookie Session 34 | const cookieSession = require("cookie-session"); 35 | app.use( 36 | cookieSession({ 37 | secret: secrets.COOKIE_SESSION_SECRET, 38 | maxAge: 1000 * 60 * 60 * 24 * 14, 39 | httpOnly: true, 40 | secure: false, 41 | }) 42 | ); 43 | 44 | // #CSRF security for Production 45 | if (process.env.NODE_ENV == "production") { 46 | app.use(csurf()); 47 | app.use((req, res, next) => { 48 | res.set("x-frame-options", "DENY"); 49 | res.cookie("mytoken", req.csrfToken()); 50 | next(); 51 | }); 52 | } 53 | 54 | // #Routes 55 | app.use("/api/auth", require("./routes/auth")); 56 | app.use("/api/user", require("./routes/user")); 57 | app.use("/api/jobs", require("./routes/jobs")); 58 | app.use("/", require("./routes/index")); 59 | 60 | // Serve the build in production 61 | app.get("*", (req, res) => res.sendFile(__dirname + "/public/index.html")); 62 | 63 | app.listen(port, () => console.log(`Server listening on port ${port}`)); 64 | -------------------------------------------------------------------------------- /server/routes/auth.js: -------------------------------------------------------------------------------- 1 | const express = require("express"); 2 | const router = express.Router(); 3 | const jwt = require("jsonwebtoken"); 4 | const cryptoRandomString = require("crypto-random-string"); 5 | const { User } = require("../utils/models/user"); 6 | const { Code } = require("../utils/models/secretCode"); 7 | const { hash, compare } = require("../utils/bcrypt"); 8 | const emailService = require("../utils/nodemailer"); 9 | const authenticateTokenWhilePending = require("../utils/middleware/checkAuthWhilePending"); 10 | const authenticateToken = require("../utils/middleware/checkAuth"); 11 | 12 | // #route: POST /Login 13 | // #desc: Login a user 14 | // #access: Public 15 | router.post("/login", async (req, res) => { 16 | const { email, password } = req.body; 17 | let errors = []; 18 | 19 | if (!email || !password) { 20 | errors.push({ msg: "Please fill in all fields!" }); 21 | res.json({ success: false, errors }); 22 | } else { 23 | try { 24 | const user = await User.findOne({ email: email }); 25 | 26 | if (!user) { 27 | errors.push({ msg: "The provided email is not registered." }); 28 | res.json({ success: false, errors }); 29 | } else { 30 | const pwCheckSuccess = await compare(password, user.password); 31 | 32 | if (!pwCheckSuccess) { 33 | errors.push({ msg: "Email and password do not match." }); 34 | res.json({ success: false, errors }); 35 | } else { 36 | const token = jwt.sign( 37 | { 38 | userId: user._id, 39 | userRole: user.role, 40 | userStatus: user.status, 41 | }, 42 | res.locals.secrets.JWT_SECRET, 43 | { 44 | expiresIn: 60 * 60 * 24 * 14, 45 | } 46 | ); 47 | 48 | req.session.token = token; 49 | 50 | res.json({ 51 | success: true, 52 | userRole: user.role, 53 | userId: user._id, 54 | userStatus: user.status, 55 | }); 56 | } 57 | } 58 | } catch (err) { 59 | console.log("Error on /api/auth/login: ", err); 60 | res.json({ success: false }); 61 | } 62 | } 63 | }); 64 | 65 | // #route: POST /register 66 | // #desc: Register a new user 67 | // #access: Public 68 | router.post("/register", async (req, res) => { 69 | const { 70 | firstName, 71 | lastName, 72 | email, 73 | password, 74 | password2, 75 | acceptance, 76 | } = req.body; 77 | let errors = []; 78 | 79 | // Check if data is correctly provided 80 | if (!firstName || !lastName || !email || !password || !password2) { 81 | errors.push({ msg: "Please fill in all fields!" }); 82 | } 83 | if (password != password2) { 84 | errors.push({ msg: "The entered passwords do not match!" }); 85 | } 86 | if ( 87 | !password.match( 88 | /^(?=.*\d)(?=.*[a-z])(?=.*[A-Z])(?=.*[^a-zA-Z0-9])(?!.*\s).{6,}$/ 89 | ) 90 | ) { 91 | errors.push({ 92 | msg: 93 | "Your password must be at least 6 characters long and contain a lowercase letter, an uppercase letter, a numeric digit and a special character.", 94 | }); 95 | } 96 | if (acceptance != "accepted") { 97 | errors.push({ msg: "You need to accept the terms of use." }); 98 | } 99 | 100 | if (errors.length > 0) { 101 | res.json({ success: false, errors }); 102 | } else { 103 | try { 104 | // Check if user already exists 105 | const existingUser = await User.findOne({ email: email }); 106 | 107 | if (existingUser) { 108 | errors.push({ 109 | msg: "The provided email is registered already.", 110 | }); 111 | res.json({ success: false, errors }); 112 | } else { 113 | const hashedPw = await hash(password); 114 | 115 | const newUser = new User({ 116 | firstName, 117 | lastName, 118 | email, 119 | password: hashedPw, 120 | accepted: true, 121 | }); 122 | 123 | const user = await newUser.save(); 124 | const token = jwt.sign( 125 | { 126 | userId: user._id, 127 | userRole: user.role, 128 | userStatus: user.status, 129 | }, 130 | res.locals.secrets.JWT_SECRET, 131 | { 132 | expiresIn: 60 * 60 * 24 * 14, 133 | } 134 | ); 135 | 136 | req.session.token = token; 137 | 138 | const baseUrl = req.protocol + "://" + req.get("host"); 139 | const secretCode = cryptoRandomString({ 140 | length: 6, 141 | }); 142 | const newCode = new Code({ 143 | code: secretCode, 144 | email: user.email, 145 | }); 146 | await newCode.save(); 147 | 148 | const data = { 149 | from: `YOUR NAME <${res.locals.secrets.EMAIL_USERNAME}>`, 150 | to: user.email, 151 | subject: "Your Activation Link for YOUR APP", 152 | text: `Please use the following link within the next 10 minutes to activate your account on YOUR APP: ${baseUrl}/api/auth/verification/verify-account/${user._id}/${secretCode}`, 153 | html: `

Please use the following link within the next 10 minutes to activate your account on YOUR APP: Email bestätigen

`, 154 | }; 155 | await emailService.sendMail(data); 156 | 157 | res.json({ 158 | success: true, 159 | userRole: user.role, 160 | userId: user._id, 161 | userStatus: user.status, 162 | }); 163 | } 164 | } catch (err) { 165 | console.log("Error on /api/auth/register: ", err); 166 | errors.push({ 167 | msg: "Oh, something went wrong. Please try again!", 168 | }); 169 | res.json({ success: false, errors }); 170 | } 171 | } 172 | }); 173 | 174 | // #route: GET /verification/get-activation-email 175 | // #desc: Send activation email to registered users email address 176 | // #access: Private 177 | router.get( 178 | "/verification/get-activation-email", 179 | authenticateTokenWhilePending, 180 | async (req, res) => { 181 | const baseUrl = req.protocol + "://" + req.get("host"); 182 | 183 | try { 184 | const user = await User.findById(req.userId); 185 | 186 | if (!user) { 187 | res.json({ success: false }); 188 | } else { 189 | await Code.deleteMany({ email: user.email }); 190 | 191 | const secretCode = cryptoRandomString({ 192 | length: 6, 193 | }); 194 | const newCode = new Code({ 195 | code: secretCode, 196 | email: user.email, 197 | }); 198 | await newCode.save(); 199 | 200 | const data = { 201 | from: `YOUR NAME <${res.locals.secrets.EMAIL_USERNAME}>`, 202 | to: user.email, 203 | subject: "Your Activation Link for YOUR APP", 204 | text: `Please use the following link within the next 10 minutes to activate your account on YOUR APP: ${baseUrl}/api/auth/verification/verify-account/${user._id}/${secretCode}`, 205 | html: `

Please use the following link within the next 10 minutes to activate your account on YOUR APP: Email bestätigen

`, 206 | }; 207 | await emailService.sendMail(data); 208 | 209 | res.json({ success: true }); 210 | } 211 | } catch (err) { 212 | console.log("Error on /api/auth/get-activation-email: ", err); 213 | res.json({ success: false }); 214 | } 215 | } 216 | ); 217 | 218 | // #route: GET /verification/verify-account 219 | // #desc: Verify user's email address 220 | // #access: Public 221 | router.get( 222 | "/verification/verify-account/:userId/:secretCode", 223 | async (req, res) => { 224 | try { 225 | const user = await User.findById(req.params.userId); 226 | const response = await Code.findOne({ 227 | email: user.email, 228 | code: req.params.secretCode, 229 | }); 230 | 231 | if (!user) { 232 | res.sendStatus(401); 233 | } else { 234 | await User.updateOne( 235 | { email: user.email }, 236 | { status: "active" } 237 | ); 238 | await Code.deleteMany({ email: user.email }); 239 | 240 | let redirectPath; 241 | 242 | if (process.env.NODE_ENV == "production") { 243 | redirectPath = `${req.protocol}://${req.get( 244 | "host" 245 | )}account/verified`; 246 | } else { 247 | redirectPath = `http://127.0.0.1:8080/account/verified`; 248 | } 249 | 250 | res.redirect(redirectPath); 251 | } 252 | } catch (err) { 253 | console.log( 254 | "Error on /api/auth/verification/verify-account: ", 255 | err 256 | ); 257 | res.sendStatus(500); 258 | } 259 | } 260 | ); 261 | 262 | // #route: GET /verification/update-user-status 263 | // #desc: Verify user's email address 264 | // #access: Public 265 | router.get( 266 | "/verification/update-user-status", 267 | authenticateTokenWhilePending, 268 | async (req, res) => { 269 | try { 270 | const user = await User.findById(req.userId); 271 | 272 | if (!user) { 273 | res.json({ success: false }); 274 | } else { 275 | const token = jwt.sign( 276 | { 277 | userId: user._id, 278 | userRole: user.role, 279 | userStatus: user.status, 280 | }, 281 | res.locals.secrets.JWT_SECRET, 282 | { 283 | expiresIn: 60 * 60 * 24 * 14, 284 | } 285 | ); 286 | 287 | req.session.token = token; 288 | 289 | res.json({ 290 | success: true, 291 | userRole: user.role, 292 | userId: user._id, 293 | userStatus: user.status, 294 | }); 295 | } 296 | } catch (err) { 297 | console.log( 298 | "Error on /api/auth/verification/update-user-status: ", 299 | err 300 | ); 301 | res.json({ success: false }); 302 | } 303 | } 304 | ); 305 | 306 | // #route: POST /password-reset/get-code 307 | // #desc: Reset password of user 308 | // #access: Public 309 | router.post("/password-reset/get-code", async (req, res) => { 310 | const { email } = req.body; 311 | let errors = []; 312 | 313 | if (!email) { 314 | errors.push({ msg: "Please provide your registered email address!" }); 315 | res.json({ success: false, errors }); 316 | } else { 317 | try { 318 | const user = await User.findOne({ email: email }); 319 | 320 | if (!user) { 321 | errors.push({ 322 | msg: "The provided email address is not registered!", 323 | }); 324 | res.json({ success: false, errors }); 325 | } else { 326 | const secretCode = cryptoRandomString({ 327 | length: 6, 328 | }); 329 | const newCode = new Code({ 330 | code: secretCode, 331 | email: email, 332 | }); 333 | await newCode.save(); 334 | 335 | const data = { 336 | from: `YOUR NAME <${res.locals.secrets.EMAIL_USERNAME}>`, 337 | to: email, 338 | subject: "Your Password Reset Code for YOUR APP", 339 | text: `Please use the following code within the next 10 minutes to reset your password on YOUR APP: ${secretCode}`, 340 | html: `

Please use the following code within the next 10 minutes to reset your password on YOUR APP: ${secretCode}

`, 341 | }; 342 | await emailService.sendMail(data); 343 | 344 | res.json({ success: true }); 345 | } 346 | } catch (err) { 347 | console.log("Error on /api/auth/password-reset/get-code: ", err); 348 | errors.push({ 349 | msg: "Oh, something went wrong. Please try again!", 350 | }); 351 | res.json({ success: false, errors }); 352 | } 353 | } 354 | }); 355 | 356 | // #route: POST /password-reset/verify 357 | // #desc: Verify and save new password of user 358 | // #access: Public 359 | router.post("/password-reset/verify", async (req, res) => { 360 | const { email, password, password2, code } = req.body; 361 | let errors = []; 362 | 363 | if (!email || !password || !password2 || !code) { 364 | errors.push({ msg: "Please fill in all fields!" }); 365 | } 366 | if (password != password2) { 367 | errors.push({ msg: "The entered passwords do not match!" }); 368 | } 369 | if ( 370 | !password.match( 371 | /^(?=.*\d)(?=.*[a-z])(?=.*[A-Z])(?=.*[^a-zA-Z0-9])(?!.*\s).{6,}$/ 372 | ) 373 | ) { 374 | errors.push({ 375 | msg: 376 | "Your password must be at least 6 characters long and contain a lowercase letter, an uppercase letter, a numeric digit and a special character.", 377 | }); 378 | } 379 | if (errors.length > 0) { 380 | res.json({ success: false, errors }); 381 | } else { 382 | try { 383 | const response = await Code.findOne({ email, code }); 384 | 385 | if (response.length === 0) { 386 | errors.push({ 387 | msg: 388 | "The entered code is not correct. Please make sure to enter the code in the requested time interval.", 389 | }); 390 | res.json({ success: false, errors }); 391 | } else { 392 | const newHashedPw = await hash(password); 393 | await User.updateOne({ email }, { password: newHashedPw }); 394 | await Code.deleteOne({ email, code }); 395 | res.json({ success: true }); 396 | } 397 | } catch (err) { 398 | console.log("Error on /api/auth/password-reset/verify: ", err); 399 | errors.push({ 400 | msg: "Oh, something went wrong. Please try again!", 401 | }); 402 | res.json({ success: false, errors }); 403 | } 404 | } 405 | }); 406 | 407 | // #route: GET /logout 408 | // #desc: Logout a user 409 | // #access: Public 410 | router.get("/logout", (req, res) => { 411 | req.session = null; 412 | res.json({ success: true }); 413 | }); 414 | 415 | // #route: POST /delete-account 416 | // #desc: Logout a user 417 | // #access: Public 418 | router.post("/delete-account", authenticateToken, async (req, res) => { 419 | const { password } = req.body; 420 | 421 | if (!password) { 422 | res.json({ success: false, error: "Please provide your password." }); 423 | } else { 424 | try { 425 | const user = await User.findById(req.userId); 426 | 427 | if (!user) { 428 | res.json({ 429 | success: false, 430 | error: "Oh, something went wrong. Please try again!", 431 | }); 432 | } else { 433 | const pwCheckSuccess = await compare(password, user.password); 434 | 435 | if (!pwCheckSuccess) { 436 | res.json({ 437 | success: false, 438 | error: "The provided password is not correct.", 439 | }); 440 | } else { 441 | const deleted = await User.deleteOne({ 442 | email: user.email, 443 | }); 444 | 445 | if (!deleted) { 446 | res.json({ 447 | success: false, 448 | error: 449 | "Oh, something went wrong. Please try again!", 450 | }); 451 | } else { 452 | req.session = null; 453 | res.json({ success: true }); 454 | } 455 | } 456 | } 457 | } catch (err) { 458 | console.log("Error on /api/auth/delete-account: ", err); 459 | res.json({ 460 | success: false, 461 | error: "Oh, something went wrong. Please try again!", 462 | }); 463 | } 464 | } 465 | }); 466 | 467 | module.exports = router; 468 | -------------------------------------------------------------------------------- /server/routes/index.js: -------------------------------------------------------------------------------- 1 | const express = require("express"); 2 | const router = express.Router(); 3 | 4 | router.get("/", (req, res) => { 5 | res.json({ url: "/" }); 6 | }); 7 | 8 | module.exports = router; 9 | -------------------------------------------------------------------------------- /server/routes/jobs.js: -------------------------------------------------------------------------------- 1 | const express = require("express"); 2 | const router = express.Router(); 3 | 4 | // #route: GET /api/jobs 5 | // #desc: Get all jobs 6 | // #access: Public 7 | router.get("/", async (req, res) => { 8 | res.json({ 9 | jobs: [ 10 | { id: 1, title: "Job 1" }, 11 | { id: 2, title: "Job 2" }, 12 | ], 13 | }); 14 | }); 15 | 16 | module.exports = router; 17 | -------------------------------------------------------------------------------- /server/routes/user.js: -------------------------------------------------------------------------------- 1 | const express = require("express"); 2 | const router = express.Router(); 3 | const authenticateToken = require("../utils/middleware/checkAuth"); 4 | 5 | router.get("/data", authenticateToken, (req, res) => { 6 | const testData = [ 7 | { 8 | id: 1, 9 | title: "Title 1", 10 | }, 11 | { 12 | id: 2, 13 | title: "Title 2", 14 | }, 15 | ]; 16 | res.json(testData); 17 | }); 18 | 19 | module.exports = router; 20 | -------------------------------------------------------------------------------- /server/utils/bcrypt.js: -------------------------------------------------------------------------------- 1 | const bcrypt = require('bcryptjs'); 2 | let { genSalt, hash, compare } = bcrypt; 3 | 4 | const { promisify } = require('util'); 5 | 6 | genSalt = promisify(genSalt); 7 | hash = promisify(hash); 8 | compare = promisify(compare); 9 | 10 | module.exports.compare = compare; 11 | module.exports.hash = plainTextPw => genSalt().then(salt => hash(plainTextPw, salt)); 12 | -------------------------------------------------------------------------------- /server/utils/db.js: -------------------------------------------------------------------------------- 1 | const mongoose = require("mongoose"); 2 | 3 | let mongoDB; 4 | if (process.env.DATABASE_URL) { 5 | mongoDB = process.env.DATABASE_URL; 6 | } else { 7 | const { MDB_URL } = require("./secrets"); 8 | mongoDB = MDB_URL; 9 | } 10 | 11 | mongoose 12 | .connect(mongoDB, { 13 | useNewUrlParser: true, 14 | useUnifiedTopology: true, 15 | useCreateIndex: true, 16 | useFindAndModify: false, 17 | }) 18 | .then(() => console.log("-----> mongoDB connected...")) 19 | .catch((err) => 20 | console.log("-----> Error trying to connect to mongoDB: ", err) 21 | ); 22 | 23 | mongoose.connection.on( 24 | "error", 25 | console.error.bind(console, "-----> mongoDB connection error") 26 | ); 27 | -------------------------------------------------------------------------------- /server/utils/middleware/checkAuth.js: -------------------------------------------------------------------------------- 1 | const jwt = require("jsonwebtoken"); 2 | 3 | const authenticateToken = (req, res, next) => { 4 | const token = req.session.token; 5 | 6 | jwt.verify(token, res.locals.secrets.JWT_SECRET, (err, user) => { 7 | if (err || user.userStatus != "active") { 8 | res.sendStatus(401); 9 | } else { 10 | req.userId = user.userId; 11 | req.userRole = user.userRole; 12 | req.userStatus = user.userStatus; 13 | 14 | next(); 15 | } 16 | }); 17 | }; 18 | 19 | module.exports = authenticateToken; 20 | -------------------------------------------------------------------------------- /server/utils/middleware/checkAuthWhilePending.js: -------------------------------------------------------------------------------- 1 | const jwt = require("jsonwebtoken"); 2 | 3 | const authenticateTokenWhilePending = (req, res, next) => { 4 | const token = req.session.token; 5 | 6 | jwt.verify(token, res.locals.secrets.JWT_SECRET, (err, user) => { 7 | if (err) { 8 | res.sendStatus(401); 9 | } else { 10 | req.userId = user.userId; 11 | req.userRole = user.userRole; 12 | req.userStatus = user.userStatus; 13 | 14 | next(); 15 | } 16 | }); 17 | }; 18 | 19 | module.exports = authenticateTokenWhilePending; 20 | -------------------------------------------------------------------------------- /server/utils/models/secretCode.js: -------------------------------------------------------------------------------- 1 | const mongoose = require("mongoose"); 2 | const Schema = mongoose.Schema; 3 | 4 | const secretCode = new Schema({ 5 | email: { 6 | type: String, 7 | required: true, 8 | }, 9 | code: { 10 | type: String, 11 | required: true, 12 | }, 13 | dateCreated: { 14 | type: Date, 15 | default: Date.now(), 16 | expires: 600, 17 | }, 18 | }); 19 | 20 | module.exports.Code = mongoose.model("code", secretCode); 21 | -------------------------------------------------------------------------------- /server/utils/models/user.js: -------------------------------------------------------------------------------- 1 | const mongoose = require("mongoose"); 2 | const Schema = mongoose.Schema; 3 | 4 | const UserSchema = new Schema({ 5 | firstName: { 6 | type: String, 7 | required: true, 8 | }, 9 | lastName: { 10 | type: String, 11 | required: true, 12 | }, 13 | email: { 14 | type: String, 15 | required: true, 16 | unique: true, 17 | }, 18 | password: { 19 | type: String, 20 | required: true, 21 | }, 22 | role: { 23 | type: String, 24 | default: "basic", 25 | }, 26 | status: { 27 | type: String, 28 | default: "pending", 29 | }, 30 | accepted: { 31 | type: Boolean, 32 | required: true, 33 | }, 34 | dateCreated: { 35 | type: Date, 36 | default: Date.now(), 37 | }, 38 | }); 39 | 40 | module.exports.User = mongoose.model("user", UserSchema); 41 | -------------------------------------------------------------------------------- /server/utils/nodemailer.js: -------------------------------------------------------------------------------- 1 | const nodemailer = require("nodemailer"); 2 | 3 | let secrets; 4 | if (process.env.NODE_ENV == "production") { 5 | secrets = process.env; 6 | } else { 7 | secrets = require("./secrets"); 8 | } 9 | 10 | const emailService = nodemailer.createTransport({ 11 | host: secrets.EMAIL_HOST, 12 | port: secrets.EMAIL_PORT, 13 | secure: true, 14 | auth: { 15 | user: secrets.EMAIL_USERNAME, 16 | pass: secrets.EMAIL_PW, 17 | }, 18 | }); 19 | 20 | module.exports = emailService; 21 | --------------------------------------------------------------------------------