├── .gitignore
├── README.md
├── app.js
├── client
├── .gitignore
├── README.md
├── package-lock.json
├── package.json
├── public
│ ├── favicon.ico
│ ├── index.html
│ ├── logo192.png
│ ├── logo512.png
│ ├── manifest.json
│ └── robots.txt
└── src
│ ├── App.js
│ ├── Components
│ ├── Admin.js
│ ├── Home.js
│ ├── Login.js
│ ├── Message.js
│ ├── Navbar.js
│ ├── Register.js
│ ├── TodoItem.js
│ └── Todos.js
│ ├── Context
│ └── AuthContext.js
│ ├── Services
│ ├── AuthService.js
│ └── TodoService.js
│ ├── hocs
│ ├── PrivateRoute.js
│ └── UnPrivateRoute.js
│ └── index.js
├── models
├── Todo.js
└── User.js
├── package-lock.json
├── package.json
├── passport.js
└── routes
└── User.js
/.gitignore:
--------------------------------------------------------------------------------
1 | # dependencies
2 | /node_modules
3 |
4 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | # MERN-Stack-Authentication-Authorization-JWT
2 | Learn how to use the MERN Stack with Authentication/Authorization JWT
3 |
4 | This is source code is for the video tutorial series that can be found here.
5 |
6 | https://www.youtube.com/playlist?list=PLvTjg4siRgU0HS3cANo7KZ52Wkud083TL
7 |
8 | Created by Pedro Mercado for NoobCoder.com
9 |
10 | Find more free tutorials by visiting NoobCoder.com or by visiting youtube.com/noobcoder
11 |
12 | 1) In order to run this, within the root directory make sure to install dependecies by opening the terminal and typing the following
13 |
14 | npm install
15 |
16 | 2) Next we need to install the dependicies within the client directory. You can do so by typing the following
17 |
18 | cd client
19 |
20 | npm install
21 |
22 | 3) Now all thats left to do is run the start script within the root directory and within the client directory. Use two terminals and type the following command
23 |
24 | npm start
25 |
--------------------------------------------------------------------------------
/app.js:
--------------------------------------------------------------------------------
1 | const express = require('express');
2 | const app = express();
3 | const cookieParser = require('cookie-parser');
4 | const mongoose = require('mongoose');
5 | app.use(cookieParser());
6 | app.use(express.json());
7 |
8 | mongoose.connect('mongodb://localhost:27017/mernauth',{useNewUrlParser : true,useUnifiedTopology: true},()=>{
9 | console.log('successfully connected to database');
10 | });
11 |
12 | const userRouter = require('./routes/User');
13 | app.use('/user',userRouter);
14 |
15 | app.listen(5000,()=>{
16 | console.log('express server started');
17 | });
--------------------------------------------------------------------------------
/client/.gitignore:
--------------------------------------------------------------------------------
1 | # See https://help.github.com/articles/ignoring-files/ for more about ignoring files.
2 |
3 | # dependencies
4 | /node_modules
5 | /.pnp
6 | .pnp.js
7 |
8 | # testing
9 | /coverage
10 |
11 | # production
12 | /build
13 |
14 | # misc
15 | .DS_Store
16 | .env.local
17 | .env.development.local
18 | .env.test.local
19 | .env.production.local
20 |
21 | npm-debug.log*
22 | yarn-debug.log*
23 | yarn-error.log*
24 |
--------------------------------------------------------------------------------
/client/README.md:
--------------------------------------------------------------------------------
1 | This project was bootstrapped with [Create React App](https://github.com/facebook/create-react-app).
2 |
3 | ## Available Scripts
4 |
5 | In the project directory, you can run:
6 |
7 | ### `npm start`
8 |
9 | Runs the app in the development mode.
10 | Open [http://localhost:3000](http://localhost:3000) to view it in the browser.
11 |
12 | The page will reload if you make edits.
13 | You will also see any lint errors in the console.
14 |
15 | ### `npm test`
16 |
17 | Launches the test runner in the interactive watch mode.
18 | See the section about [running tests](https://facebook.github.io/create-react-app/docs/running-tests) for more information.
19 |
20 | ### `npm run build`
21 |
22 | Builds the app for production to the `build` folder.
23 | It correctly bundles React in production mode and optimizes the build for the best performance.
24 |
25 | The build is minified and the filenames include the hashes.
26 | Your app is ready to be deployed!
27 |
28 | See the section about [deployment](https://facebook.github.io/create-react-app/docs/deployment) for more information.
29 |
30 | ### `npm run eject`
31 |
32 | **Note: this is a one-way operation. Once you `eject`, you can’t go back!**
33 |
34 | If you aren’t satisfied with the build tool and configuration choices, you can `eject` at any time. This command will remove the single build dependency from your project.
35 |
36 | Instead, it will copy all the configuration files and the transitive dependencies (Webpack, Babel, ESLint, etc) right into your project so you have full control over them. All of the commands except `eject` will still work, but they will point to the copied scripts so you can tweak them. At this point you’re on your own.
37 |
38 | You don’t have to ever use `eject`. The curated feature set is suitable for small and middle deployments, and you shouldn’t feel obligated to use this feature. However we understand that this tool wouldn’t be useful if you couldn’t customize it when you are ready for it.
39 |
40 | ## Learn More
41 |
42 | You can learn more in the [Create React App documentation](https://facebook.github.io/create-react-app/docs/getting-started).
43 |
44 | To learn React, check out the [React documentation](https://reactjs.org/).
45 |
46 | ### Code Splitting
47 |
48 | This section has moved here: https://facebook.github.io/create-react-app/docs/code-splitting
49 |
50 | ### Analyzing the Bundle Size
51 |
52 | This section has moved here: https://facebook.github.io/create-react-app/docs/analyzing-the-bundle-size
53 |
54 | ### Making a Progressive Web App
55 |
56 | This section has moved here: https://facebook.github.io/create-react-app/docs/making-a-progressive-web-app
57 |
58 | ### Advanced Configuration
59 |
60 | This section has moved here: https://facebook.github.io/create-react-app/docs/advanced-configuration
61 |
62 | ### Deployment
63 |
64 | This section has moved here: https://facebook.github.io/create-react-app/docs/deployment
65 |
66 | ### `npm run build` fails to minify
67 |
68 | This section has moved here: https://facebook.github.io/create-react-app/docs/troubleshooting#npm-run-build-fails-to-minify
69 |
--------------------------------------------------------------------------------
/client/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "client",
3 | "version": "0.1.0",
4 | "private": true,
5 | "dependencies": {
6 | "@testing-library/jest-dom": "^4.2.4",
7 | "@testing-library/react": "^9.4.0",
8 | "@testing-library/user-event": "^7.2.1",
9 | "react": "^16.12.0",
10 | "react-dom": "^16.12.0",
11 | "react-router": "^5.1.2",
12 | "react-router-dom": "^5.1.2",
13 | "react-scripts": "3.3.0"
14 | },
15 | "scripts": {
16 | "start": "react-scripts start",
17 | "build": "react-scripts build",
18 | "test": "react-scripts test",
19 | "eject": "react-scripts eject"
20 | },
21 | "eslintConfig": {
22 | "extends": "react-app"
23 | },
24 | "browserslist": {
25 | "production": [
26 | ">0.2%",
27 | "not dead",
28 | "not op_mini all"
29 | ],
30 | "development": [
31 | "last 1 chrome version",
32 | "last 1 firefox version",
33 | "last 1 safari version"
34 | ]
35 | },
36 | "proxy": "http://localhost:5000"
37 | }
38 |
--------------------------------------------------------------------------------
/client/public/favicon.ico:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/noobcoder1137/MERN-Stack-Authentication-Authorization-JWT/5eec9b3f172742fa73076adf6d86dd9d59ddd422/client/public/favicon.ico
--------------------------------------------------------------------------------
/client/public/index.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
12 |
13 |
17 |
18 |
27 |
28 | React App
29 |
30 |
31 | You need to enable JavaScript to run this app.
32 |
33 |
43 |
44 |
45 |
--------------------------------------------------------------------------------
/client/public/logo192.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/noobcoder1137/MERN-Stack-Authentication-Authorization-JWT/5eec9b3f172742fa73076adf6d86dd9d59ddd422/client/public/logo192.png
--------------------------------------------------------------------------------
/client/public/logo512.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/noobcoder1137/MERN-Stack-Authentication-Authorization-JWT/5eec9b3f172742fa73076adf6d86dd9d59ddd422/client/public/logo512.png
--------------------------------------------------------------------------------
/client/public/manifest.json:
--------------------------------------------------------------------------------
1 | {
2 | "short_name": "React App",
3 | "name": "Create React App Sample",
4 | "icons": [
5 | {
6 | "src": "favicon.ico",
7 | "sizes": "64x64 32x32 24x24 16x16",
8 | "type": "image/x-icon"
9 | },
10 | {
11 | "src": "logo192.png",
12 | "type": "image/png",
13 | "sizes": "192x192"
14 | },
15 | {
16 | "src": "logo512.png",
17 | "type": "image/png",
18 | "sizes": "512x512"
19 | }
20 | ],
21 | "start_url": ".",
22 | "display": "standalone",
23 | "theme_color": "#000000",
24 | "background_color": "#ffffff"
25 | }
26 |
--------------------------------------------------------------------------------
/client/public/robots.txt:
--------------------------------------------------------------------------------
1 | # https://www.robotstxt.org/robotstxt.html
2 | User-agent: *
3 |
--------------------------------------------------------------------------------
/client/src/App.js:
--------------------------------------------------------------------------------
1 | import React from 'react';
2 | import Navbar from './Components/Navbar';
3 | import Login from './Components/Login';
4 | import Home from './Components/Home';
5 | import Todos from './Components/Todos';
6 | import Register from './Components/Register';
7 | import Admin from './Components/Admin';
8 | import PrivateRoute from './hocs/PrivateRoute';
9 | import UnPrivateRoute from './hocs/UnPrivateRoute';
10 | import {BrowserRouter as Router, Route} from 'react-router-dom';
11 |
12 | function App() {
13 | return (
14 |
15 |
16 |
17 |
18 |
19 |
20 |
21 |
22 | );
23 | }
24 |
25 | export default App;
26 |
--------------------------------------------------------------------------------
/client/src/Components/Admin.js:
--------------------------------------------------------------------------------
1 | import React from 'react';
2 |
3 | const Admin = ()=>(
4 | Admin Page
5 | )
6 |
7 | export default Admin;
--------------------------------------------------------------------------------
/client/src/Components/Home.js:
--------------------------------------------------------------------------------
1 | import React from 'react';
2 |
3 | const Home = ()=>(
4 | Home Page
5 | )
6 |
7 | export default Home;
--------------------------------------------------------------------------------
/client/src/Components/Login.js:
--------------------------------------------------------------------------------
1 | import React, {useState,useContext} from 'react';
2 | import AuthService from '../Services/AuthService';
3 | import Message from '../Components/Message';
4 | import {AuthContext} from '../Context/AuthContext';
5 |
6 | const Login = props=>{
7 | const [user,setUser] = useState({username: "", password : ""});
8 | const [message,setMessage] = useState(null);
9 | const authContext = useContext(AuthContext);
10 |
11 | const onChange = e =>{
12 | setUser({...user,[e.target.name] : e.target.value});
13 | }
14 |
15 | const onSubmit = e =>{
16 | e.preventDefault();
17 | AuthService.login(user).then(data=>{
18 | console.log(data);
19 | const { isAuthenticated,user,message} = data;
20 | if(isAuthenticated){
21 | authContext.setUser(user);
22 | authContext.setIsAuthenticated(isAuthenticated);
23 | props.history.push('/todos');
24 | }
25 | else
26 | setMessage(message);
27 | });
28 | }
29 |
30 |
31 |
32 | return(
33 |
34 |
51 | {message ? : null}
52 |
53 | )
54 | }
55 |
56 | export default Login;
--------------------------------------------------------------------------------
/client/src/Components/Message.js:
--------------------------------------------------------------------------------
1 | import React from 'react'
2 |
3 |
4 | const getStyle = (props)=>{
5 | let baseClass = "alert ";
6 | if(props.message.msgError)
7 | baseClass = baseClass + "alert-danger";
8 | else
9 | baseClass = baseClass + "alert-primary";
10 | return baseClass + " text-center";
11 | }
12 |
13 | const Message = props=>{
14 | return(
15 |
16 | {props.message.msgBody}
17 |
18 | )
19 | }
20 |
21 | export default Message;
--------------------------------------------------------------------------------
/client/src/Components/Navbar.js:
--------------------------------------------------------------------------------
1 | import React, {useContext} from 'react';
2 | import {Link} from 'react-router-dom';
3 | import AuthService from '../Services/AuthService';
4 | import { AuthContext } from '../Context/AuthContext';
5 |
6 | const Navbar = props =>{
7 | const {isAuthenticated,user,setIsAuthenticated,setUser} = useContext(AuthContext);
8 |
9 | const onClickLogoutHandler = ()=>{
10 | AuthService.logout().then(data=>{
11 | if(data.success){
12 | setUser(data.user);
13 | setIsAuthenticated(false);
14 | }
15 | });
16 | }
17 |
18 | const unauthenticatedNavBar = ()=>{
19 | return (
20 | <>
21 |
22 |
23 | Home
24 |
25 |
26 |
27 |
28 | Login
29 |
30 |
31 |
32 |
33 | Register
34 |
35 |
36 | >
37 | )
38 | }
39 |
40 | const authenticatedNavBar = ()=>{
41 | return(
42 | <>
43 |
44 |
45 | Home
46 |
47 |
48 |
49 |
50 | Todos
51 |
52 |
53 | {
54 | user.role === "admin" ?
55 |
56 |
57 | Admin
58 |
59 | : null
60 | }
61 | Logout
64 | >
65 | )
66 | }
67 | return(
68 |
69 |
70 | NoobCoder
71 |
72 |
73 |
74 | { !isAuthenticated ? unauthenticatedNavBar() : authenticatedNavBar()}
75 |
76 |
77 |
78 | )
79 | }
80 |
81 | export default Navbar;
--------------------------------------------------------------------------------
/client/src/Components/Register.js:
--------------------------------------------------------------------------------
1 | import React, {useState,useRef,useEffect} from 'react';
2 | import AuthService from '../Services/AuthService';
3 | import Message from '../Components/Message';
4 |
5 | const Register = props=>{
6 | const [user,setUser] = useState({username: "", password : "", role : ""});
7 | const [message,setMessage] = useState(null);
8 | let timerID = useRef(null);
9 |
10 | useEffect(()=>{
11 | return ()=>{
12 | clearTimeout(timerID);
13 | }
14 | },[]);
15 |
16 | const onChange = e =>{
17 | setUser({...user,[e.target.name] : e.target.value});
18 | }
19 |
20 | const resetForm = ()=>{
21 | setUser({username : "", password : "",role : ""});
22 | }
23 |
24 | const onSubmit = e =>{
25 | e.preventDefault();
26 | AuthService.register(user).then(data=>{
27 | const { message } = data;
28 | setMessage(message);
29 | resetForm();
30 | if(!message.msgError){
31 | timerID = setTimeout(()=>{
32 | props.history.push('/login');
33 | },2000)
34 | }
35 | });
36 | }
37 |
38 |
39 |
40 | return(
41 |
42 |
68 | {message ? : null}
69 |
70 | )
71 | }
72 |
73 | export default Register;
--------------------------------------------------------------------------------
/client/src/Components/TodoItem.js:
--------------------------------------------------------------------------------
1 | import React from 'react';
2 |
3 | const TodoItem = props =>{
4 | return (
5 | {props.todo.name}
6 | )
7 | }
8 |
9 | export default TodoItem;
--------------------------------------------------------------------------------
/client/src/Components/Todos.js:
--------------------------------------------------------------------------------
1 | import React, {useState,useContext,useEffect} from 'react';
2 | import TodoItem from './TodoItem';
3 | import TodoService from '../Services/TodoService';
4 | import Message from './Message';
5 | import { AuthContext } from '../Context/AuthContext';
6 |
7 | const Todos = props =>{
8 | const [todo,setTodo] = useState({name : ""});
9 | const [todos,setTodos] = useState([]);
10 | const [message,setMessage] = useState(null);
11 | const authContext = useContext(AuthContext);
12 |
13 | useEffect(()=>{
14 | TodoService.getTodos().then(data =>{
15 | setTodos(data.todos);
16 | });
17 | },[]);
18 |
19 | const onSubmit = e =>{
20 | e.preventDefault();
21 | TodoService.postTodo(todo).then(data =>{
22 | const { message } = data;
23 | resetForm();
24 | if(!message.msgError){
25 | TodoService.getTodos().then(getData =>{
26 | setTodos(getData.todos);
27 | setMessage(message);
28 | });
29 | }
30 | else if(message.msgBody === "UnAuthorized"){
31 | setMessage(message);
32 | authContext.setUser({username : "", role : ""});
33 | authContext.setIsAuthenticated(false);
34 | }
35 | else{
36 | setMessage(message);
37 | }
38 | });
39 | }
40 |
41 | const onChange = e =>{
42 | setTodo({name : e.target.value});
43 | }
44 |
45 | const resetForm = ()=>{
46 | setTodo({name : ""});
47 | }
48 |
49 | return(
50 |
51 |
52 | {
53 | todos.map(todo =>{
54 | return
55 | })
56 | }
57 |
58 |
59 |
70 | {message ?
: null}
71 |
72 | );
73 |
74 | }
75 |
76 | export default Todos;
--------------------------------------------------------------------------------
/client/src/Context/AuthContext.js:
--------------------------------------------------------------------------------
1 | import React, {createContext,useState,useEffect} from 'react';
2 | import AuthService from '../Services/AuthService';
3 |
4 | export const AuthContext = createContext();
5 |
6 | export default ({ children })=>{
7 | const [user,setUser] = useState(null);
8 | const [isAuthenticated,setIsAuthenticated] = useState(false);
9 | const [isLoaded,setIsLoaded] = useState(false);
10 |
11 | useEffect(()=>{
12 | AuthService.isAuthenticated().then(data =>{
13 | setUser(data.user);
14 | setIsAuthenticated(data.isAuthenticated);
15 | setIsLoaded(true);
16 | });
17 | },[]);
18 |
19 | return (
20 |
21 | {!isLoaded ?
Loading :
22 |
23 | { children }
24 | }
25 |
26 | )
27 | }
--------------------------------------------------------------------------------
/client/src/Services/AuthService.js:
--------------------------------------------------------------------------------
1 |
2 | export default {
3 | login : user =>{
4 | console.log(user);
5 | return fetch('/user/login',{
6 | method : "post",
7 | body : JSON.stringify(user),
8 | headers : {
9 | 'Content-Type' : 'application/json'
10 | }
11 | }).then(res => {
12 | if(res.status !== 401)
13 | return res.json().then(data => data);
14 | else
15 | return { isAuthenticated : false, user : {username : "",role : ""}};
16 | })
17 | },
18 | register : user =>{
19 | console.log(user);
20 | return fetch('/user/register',{
21 | method : "post",
22 | body : JSON.stringify(user),
23 | headers : {
24 | 'Content-Type' : 'application/json'
25 | }
26 | }).then(res => res.json())
27 | .then(data => data);
28 | },
29 | logout : ()=>{
30 | return fetch('/user/logout')
31 | .then(res => res.json())
32 | .then(data => data);
33 | },
34 | isAuthenticated : ()=>{
35 | return fetch('/user/authenticated')
36 | .then(res=>{
37 | if(res.status !== 401)
38 | return res.json().then(data => data);
39 | else
40 | return { isAuthenticated : false, user : {username : "",role : ""}};
41 | });
42 | }
43 |
44 | }
--------------------------------------------------------------------------------
/client/src/Services/TodoService.js:
--------------------------------------------------------------------------------
1 | export default {
2 | getTodos : ()=>{
3 | return fetch('/user/todos')
4 | .then(response=>{
5 | if(response.status !== 401){
6 | return response.json().then(data => data);
7 | }
8 | else
9 | return {message : {msgBody : "UnAuthorized",msgError : true}};
10 | });
11 | },
12 | postTodo : todo=>{
13 | return fetch('/user/todo',{
14 | method : "post",
15 | body : JSON.stringify(todo),
16 | headers:{
17 | 'Content-Type' : 'application/json'
18 | }
19 | }).then(response=>{
20 | if(response.status !== 401){
21 | return response.json().then(data => data);
22 | }
23 | else
24 | return {message : {msgBody : "UnAuthorized"},msgError : true};
25 | });
26 | }
27 | }
--------------------------------------------------------------------------------
/client/src/hocs/PrivateRoute.js:
--------------------------------------------------------------------------------
1 | import React, {useContext} from 'react';
2 | import {Route,Redirect} from 'react-router-dom';
3 | import { AuthContext } from '../Context/AuthContext';
4 |
5 | const PrivateRoute = ({component : Component, roles, ...rest})=>{
6 | const { isAuthenticated, user} = useContext(AuthContext);
7 | return(
8 | {
9 | if(!isAuthenticated)
10 | return
12 |
13 | if(!roles.includes(user.role))
14 | return
16 | return
17 | }}/>
18 | )
19 | }
20 |
21 | export default PrivateRoute;
--------------------------------------------------------------------------------
/client/src/hocs/UnPrivateRoute.js:
--------------------------------------------------------------------------------
1 | import React, {useContext} from 'react';
2 | import {Route,Redirect} from 'react-router-dom';
3 | import { AuthContext } from '../Context/AuthContext';
4 |
5 | const UnPrivateRoute = ({component : Component,...rest})=>{
6 | const { isAuthenticated } = useContext(AuthContext);
7 | return(
8 | {
9 | if(isAuthenticated)
10 | return
12 |
13 | return
14 | }}/>
15 | )
16 | }
17 |
18 | export default UnPrivateRoute;
--------------------------------------------------------------------------------
/client/src/index.js:
--------------------------------------------------------------------------------
1 | import React from 'react';
2 | import ReactDOM from 'react-dom';
3 | import App from './App';
4 | import AuthProvider from './Context/AuthContext';
5 |
6 | ReactDOM.render( , document.getElementById('root'));
7 |
8 |
--------------------------------------------------------------------------------
/models/Todo.js:
--------------------------------------------------------------------------------
1 | const mongoose = require('mongoose');
2 |
3 | const TodoSchema = new mongoose.Schema({
4 | name : {
5 | type : String,
6 | required : true
7 | }
8 | });
9 |
10 | module.exports = mongoose.model('Todo',TodoSchema);
--------------------------------------------------------------------------------
/models/User.js:
--------------------------------------------------------------------------------
1 | const mongoose = require('mongoose');
2 | const bcrypt = require('bcrypt');
3 |
4 | const UserSchema = new mongoose.Schema({
5 | username :{
6 | type : String,
7 | required : true,
8 | min : 6,
9 | max : 15
10 | },
11 | password : {
12 | type : String,
13 | required : true
14 | },
15 | role : {
16 | type : String,
17 | enum : ['user','admin'],
18 | required: true
19 | },
20 | todos : [{type : mongoose.Schema.Types.ObjectId, ref: 'Todo'}]
21 | });
22 |
23 | UserSchema.pre('save',function(next){
24 | if(!this.isModified('password'))
25 | return next();
26 | bcrypt.hash(this.password,10,(err,passwordHash)=>{
27 | if(err)
28 | return next(err);
29 | this.password = passwordHash;
30 | next();
31 | });
32 | });
33 |
34 | UserSchema.methods.comparePassword = function(password,cb){
35 | bcrypt.compare(password,this.password,(err,isMatch)=>{
36 | if(err)
37 | return cb(err);
38 | else{
39 | if(!isMatch)
40 | return cb(null,isMatch);
41 | return cb(null,this);
42 | }
43 | });
44 | }
45 |
46 | module.exports = mongoose.model('User',UserSchema);
--------------------------------------------------------------------------------
/package-lock.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "MernJWT",
3 | "version": "1.0.0",
4 | "lockfileVersion": 1,
5 | "requires": true,
6 | "dependencies": {
7 | "abbrev": {
8 | "version": "1.1.1",
9 | "resolved": "https://registry.npmjs.org/abbrev/-/abbrev-1.1.1.tgz",
10 | "integrity": "sha512-nne9/IiQ/hzIhY6pdDnbBtz7DjPTKrY00P/zvPSm5pOFkl6xuGrGnXn/VtTNNfNtAfZ9/1RtehkszU9qcTii0Q=="
11 | },
12 | "accepts": {
13 | "version": "1.3.7",
14 | "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.7.tgz",
15 | "integrity": "sha512-Il80Qs2WjYlJIBNzNkK6KYqlVMTbZLXgHx2oT0pU/fjRHyEp+PEfEPY0R3WCwAGVOtauxh1hOxNgIf5bv7dQpA==",
16 | "requires": {
17 | "mime-types": "~2.1.24",
18 | "negotiator": "0.6.2"
19 | }
20 | },
21 | "ansi-regex": {
22 | "version": "2.1.1",
23 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz",
24 | "integrity": "sha1-w7M6te42DYbg5ijwRorn7yfWVN8="
25 | },
26 | "aproba": {
27 | "version": "1.2.0",
28 | "resolved": "https://registry.npmjs.org/aproba/-/aproba-1.2.0.tgz",
29 | "integrity": "sha512-Y9J6ZjXtoYh8RnXVCMOU/ttDmk1aBjunq9vO0ta5x85WDQiQfUF9sIPBITdbiiIVcBo03Hi3jMxigBtsddlXRw=="
30 | },
31 | "are-we-there-yet": {
32 | "version": "1.1.5",
33 | "resolved": "https://registry.npmjs.org/are-we-there-yet/-/are-we-there-yet-1.1.5.tgz",
34 | "integrity": "sha512-5hYdAkZlcG8tOLujVDTgCT+uPX0VnpAH28gWsLfzpXYm7wP6mp5Q/gYyR7YQ0cKVJcXJnl3j2kpBan13PtQf6w==",
35 | "requires": {
36 | "delegates": "^1.0.0",
37 | "readable-stream": "^2.0.6"
38 | }
39 | },
40 | "array-flatten": {
41 | "version": "1.1.1",
42 | "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz",
43 | "integrity": "sha1-ml9pkFGx5wczKPKgCJaLZOopVdI="
44 | },
45 | "balanced-match": {
46 | "version": "1.0.0",
47 | "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.0.tgz",
48 | "integrity": "sha1-ibTRmasr7kneFk6gK4nORi1xt2c="
49 | },
50 | "bcrypt": {
51 | "version": "3.0.7",
52 | "resolved": "https://registry.npmjs.org/bcrypt/-/bcrypt-3.0.7.tgz",
53 | "integrity": "sha512-K5UglF9VQvBMHl/1elNyyFvAfOY9Bj+rpKrCSR9sFwcW8FywAYJSRwTURNej5TaAK2TEJkcJ6r6lh1YPmspx5Q==",
54 | "requires": {
55 | "nan": "2.14.0",
56 | "node-pre-gyp": "0.13.0"
57 | }
58 | },
59 | "bluebird": {
60 | "version": "3.5.1",
61 | "resolved": "https://registry.npmjs.org/bluebird/-/bluebird-3.5.1.tgz",
62 | "integrity": "sha512-MKiLiV+I1AA596t9w1sQJ8jkiSr5+ZKi0WKrYGUn6d1Fx+Ij4tIj+m2WMQSGczs5jZVxV339chE8iwk6F64wjA=="
63 | },
64 | "body-parser": {
65 | "version": "1.19.0",
66 | "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.19.0.tgz",
67 | "integrity": "sha512-dhEPs72UPbDnAQJ9ZKMNTP6ptJaionhP5cBb541nXPlW60Jepo9RV/a4fX4XWW9CuFNK22krhrj1+rgzifNCsw==",
68 | "requires": {
69 | "bytes": "3.1.0",
70 | "content-type": "~1.0.4",
71 | "debug": "2.6.9",
72 | "depd": "~1.1.2",
73 | "http-errors": "1.7.2",
74 | "iconv-lite": "0.4.24",
75 | "on-finished": "~2.3.0",
76 | "qs": "6.7.0",
77 | "raw-body": "2.4.0",
78 | "type-is": "~1.6.17"
79 | }
80 | },
81 | "brace-expansion": {
82 | "version": "1.1.11",
83 | "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz",
84 | "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==",
85 | "requires": {
86 | "balanced-match": "^1.0.0",
87 | "concat-map": "0.0.1"
88 | }
89 | },
90 | "bson": {
91 | "version": "1.1.3",
92 | "resolved": "https://registry.npmjs.org/bson/-/bson-1.1.3.tgz",
93 | "integrity": "sha512-TdiJxMVnodVS7r0BdL42y/pqC9cL2iKynVwA0Ho3qbsQYr428veL3l7BQyuqiw+Q5SqqoT0m4srSY/BlZ9AxXg=="
94 | },
95 | "buffer-equal-constant-time": {
96 | "version": "1.0.1",
97 | "resolved": "https://registry.npmjs.org/buffer-equal-constant-time/-/buffer-equal-constant-time-1.0.1.tgz",
98 | "integrity": "sha1-+OcRMvf/5uAaXJaXpMbz5I1cyBk="
99 | },
100 | "bytes": {
101 | "version": "3.1.0",
102 | "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.0.tgz",
103 | "integrity": "sha512-zauLjrfCG+xvoyaqLoV8bLVXXNGC4JqlxFCutSDWA6fJrTo2ZuvLYTqZ7aHBLZSMOopbzwv8f+wZcVzfVTI2Dg=="
104 | },
105 | "chownr": {
106 | "version": "1.1.3",
107 | "resolved": "https://registry.npmjs.org/chownr/-/chownr-1.1.3.tgz",
108 | "integrity": "sha512-i70fVHhmV3DtTl6nqvZOnIjbY0Pe4kAUjwHj8z0zAdgBtYrJyYwLKCCuRBQ5ppkyL0AkN7HKRnETdmdp1zqNXw=="
109 | },
110 | "code-point-at": {
111 | "version": "1.1.0",
112 | "resolved": "https://registry.npmjs.org/code-point-at/-/code-point-at-1.1.0.tgz",
113 | "integrity": "sha1-DQcLTQQ6W+ozovGkDi7bPZpMz3c="
114 | },
115 | "concat-map": {
116 | "version": "0.0.1",
117 | "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz",
118 | "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s="
119 | },
120 | "console-control-strings": {
121 | "version": "1.1.0",
122 | "resolved": "https://registry.npmjs.org/console-control-strings/-/console-control-strings-1.1.0.tgz",
123 | "integrity": "sha1-PXz0Rk22RG6mRL9LOVB/mFEAjo4="
124 | },
125 | "content-disposition": {
126 | "version": "0.5.3",
127 | "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.3.tgz",
128 | "integrity": "sha512-ExO0774ikEObIAEV9kDo50o+79VCUdEB6n6lzKgGwupcVeRlhrj3qGAfwq8G6uBJjkqLrhT0qEYFcWng8z1z0g==",
129 | "requires": {
130 | "safe-buffer": "5.1.2"
131 | }
132 | },
133 | "content-type": {
134 | "version": "1.0.4",
135 | "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.4.tgz",
136 | "integrity": "sha512-hIP3EEPs8tB9AT1L+NUqtwOAps4mk2Zob89MWXMHjHWg9milF/j4osnnQLXBCBFBk/tvIG/tUc9mOUJiPBhPXA=="
137 | },
138 | "cookie": {
139 | "version": "0.4.0",
140 | "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.4.0.tgz",
141 | "integrity": "sha512-+Hp8fLp57wnUSt0tY0tHEXh4voZRDnoIrZPqlo3DPiI4y9lwg/jqx+1Om94/W6ZaPDOUbnjOt/99w66zk+l1Xg=="
142 | },
143 | "cookie-parser": {
144 | "version": "1.4.4",
145 | "resolved": "https://registry.npmjs.org/cookie-parser/-/cookie-parser-1.4.4.tgz",
146 | "integrity": "sha512-lo13tqF3JEtFO7FyA49CqbhaFkskRJ0u/UAiINgrIXeRCY41c88/zxtrECl8AKH3B0hj9q10+h3Kt8I7KlW4tw==",
147 | "requires": {
148 | "cookie": "0.3.1",
149 | "cookie-signature": "1.0.6"
150 | },
151 | "dependencies": {
152 | "cookie": {
153 | "version": "0.3.1",
154 | "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.3.1.tgz",
155 | "integrity": "sha1-5+Ch+e9DtMi6klxcWpboBtFoc7s="
156 | }
157 | }
158 | },
159 | "cookie-signature": {
160 | "version": "1.0.6",
161 | "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz",
162 | "integrity": "sha1-4wOogrNCzD7oylE6eZmXNNqzriw="
163 | },
164 | "core-util-is": {
165 | "version": "1.0.2",
166 | "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.2.tgz",
167 | "integrity": "sha1-tf1UIgqivFq1eqtxQMlAdUUDwac="
168 | },
169 | "debug": {
170 | "version": "2.6.9",
171 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz",
172 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==",
173 | "requires": {
174 | "ms": "2.0.0"
175 | }
176 | },
177 | "deep-extend": {
178 | "version": "0.6.0",
179 | "resolved": "https://registry.npmjs.org/deep-extend/-/deep-extend-0.6.0.tgz",
180 | "integrity": "sha512-LOHxIOaPYdHlJRtCQfDIVZtfw/ufM8+rVj649RIHzcm/vGwQRXFt6OPqIFWsm2XEMrNIEtWR64sY1LEKD2vAOA=="
181 | },
182 | "delegates": {
183 | "version": "1.0.0",
184 | "resolved": "https://registry.npmjs.org/delegates/-/delegates-1.0.0.tgz",
185 | "integrity": "sha1-hMbhWbgZBP3KWaDvRM2HDTElD5o="
186 | },
187 | "depd": {
188 | "version": "1.1.2",
189 | "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.2.tgz",
190 | "integrity": "sha1-m81S4UwJd2PnSbJ0xDRu0uVgtak="
191 | },
192 | "destroy": {
193 | "version": "1.0.4",
194 | "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.0.4.tgz",
195 | "integrity": "sha1-l4hXRCxEdJ5CBmE+N5RiBYJqvYA="
196 | },
197 | "detect-libc": {
198 | "version": "1.0.3",
199 | "resolved": "https://registry.npmjs.org/detect-libc/-/detect-libc-1.0.3.tgz",
200 | "integrity": "sha1-+hN8S9aY7fVc1c0CrFWfkaTEups="
201 | },
202 | "ecdsa-sig-formatter": {
203 | "version": "1.0.11",
204 | "resolved": "https://registry.npmjs.org/ecdsa-sig-formatter/-/ecdsa-sig-formatter-1.0.11.tgz",
205 | "integrity": "sha512-nagl3RYrbNv6kQkeJIpt6NJZy8twLB/2vtz6yN9Z4vRKHN4/QZJIEbqohALSgwKdnksuY3k5Addp5lg8sVoVcQ==",
206 | "requires": {
207 | "safe-buffer": "^5.0.1"
208 | }
209 | },
210 | "ee-first": {
211 | "version": "1.1.1",
212 | "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz",
213 | "integrity": "sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0="
214 | },
215 | "encodeurl": {
216 | "version": "1.0.2",
217 | "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz",
218 | "integrity": "sha1-rT/0yG7C0CkyL1oCw6mmBslbP1k="
219 | },
220 | "escape-html": {
221 | "version": "1.0.3",
222 | "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz",
223 | "integrity": "sha1-Aljq5NPQwJdN4cFpGI7wBR0dGYg="
224 | },
225 | "etag": {
226 | "version": "1.8.1",
227 | "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz",
228 | "integrity": "sha1-Qa4u62XvpiJorr/qg6x9eSmbCIc="
229 | },
230 | "express": {
231 | "version": "4.17.1",
232 | "resolved": "https://registry.npmjs.org/express/-/express-4.17.1.tgz",
233 | "integrity": "sha512-mHJ9O79RqluphRrcw2X/GTh3k9tVv8YcoyY4Kkh4WDMUYKRZUq0h1o0w2rrrxBqM7VoeUVqgb27xlEMXTnYt4g==",
234 | "requires": {
235 | "accepts": "~1.3.7",
236 | "array-flatten": "1.1.1",
237 | "body-parser": "1.19.0",
238 | "content-disposition": "0.5.3",
239 | "content-type": "~1.0.4",
240 | "cookie": "0.4.0",
241 | "cookie-signature": "1.0.6",
242 | "debug": "2.6.9",
243 | "depd": "~1.1.2",
244 | "encodeurl": "~1.0.2",
245 | "escape-html": "~1.0.3",
246 | "etag": "~1.8.1",
247 | "finalhandler": "~1.1.2",
248 | "fresh": "0.5.2",
249 | "merge-descriptors": "1.0.1",
250 | "methods": "~1.1.2",
251 | "on-finished": "~2.3.0",
252 | "parseurl": "~1.3.3",
253 | "path-to-regexp": "0.1.7",
254 | "proxy-addr": "~2.0.5",
255 | "qs": "6.7.0",
256 | "range-parser": "~1.2.1",
257 | "safe-buffer": "5.1.2",
258 | "send": "0.17.1",
259 | "serve-static": "1.14.1",
260 | "setprototypeof": "1.1.1",
261 | "statuses": "~1.5.0",
262 | "type-is": "~1.6.18",
263 | "utils-merge": "1.0.1",
264 | "vary": "~1.1.2"
265 | }
266 | },
267 | "finalhandler": {
268 | "version": "1.1.2",
269 | "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.1.2.tgz",
270 | "integrity": "sha512-aAWcW57uxVNrQZqFXjITpW3sIUQmHGG3qSb9mUah9MgMC4NeWhNOlNjXEYq3HjRAvL6arUviZGGJsBg6z0zsWA==",
271 | "requires": {
272 | "debug": "2.6.9",
273 | "encodeurl": "~1.0.2",
274 | "escape-html": "~1.0.3",
275 | "on-finished": "~2.3.0",
276 | "parseurl": "~1.3.3",
277 | "statuses": "~1.5.0",
278 | "unpipe": "~1.0.0"
279 | }
280 | },
281 | "forwarded": {
282 | "version": "0.1.2",
283 | "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.1.2.tgz",
284 | "integrity": "sha1-mMI9qxF1ZXuMBXPozszZGw/xjIQ="
285 | },
286 | "fresh": {
287 | "version": "0.5.2",
288 | "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz",
289 | "integrity": "sha1-PYyt2Q2XZWn6g1qx+OSyOhBWBac="
290 | },
291 | "fs-minipass": {
292 | "version": "1.2.7",
293 | "resolved": "https://registry.npmjs.org/fs-minipass/-/fs-minipass-1.2.7.tgz",
294 | "integrity": "sha512-GWSSJGFy4e9GUeCcbIkED+bgAoFyj7XF1mV8rma3QW4NIqX9Kyx79N/PF61H5udOV3aY1IaMLs6pGbH71nlCTA==",
295 | "requires": {
296 | "minipass": "^2.6.0"
297 | }
298 | },
299 | "fs.realpath": {
300 | "version": "1.0.0",
301 | "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz",
302 | "integrity": "sha1-FQStJSMVjKpA20onh8sBQRmU6k8="
303 | },
304 | "gauge": {
305 | "version": "2.7.4",
306 | "resolved": "https://registry.npmjs.org/gauge/-/gauge-2.7.4.tgz",
307 | "integrity": "sha1-LANAXHU4w51+s3sxcCLjJfsBi/c=",
308 | "requires": {
309 | "aproba": "^1.0.3",
310 | "console-control-strings": "^1.0.0",
311 | "has-unicode": "^2.0.0",
312 | "object-assign": "^4.1.0",
313 | "signal-exit": "^3.0.0",
314 | "string-width": "^1.0.1",
315 | "strip-ansi": "^3.0.1",
316 | "wide-align": "^1.1.0"
317 | }
318 | },
319 | "glob": {
320 | "version": "7.1.6",
321 | "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.6.tgz",
322 | "integrity": "sha512-LwaxwyZ72Lk7vZINtNNrywX0ZuLyStrdDtabefZKAY5ZGJhVtgdznluResxNmPitE0SAO+O26sWTHeKSI2wMBA==",
323 | "requires": {
324 | "fs.realpath": "^1.0.0",
325 | "inflight": "^1.0.4",
326 | "inherits": "2",
327 | "minimatch": "^3.0.4",
328 | "once": "^1.3.0",
329 | "path-is-absolute": "^1.0.0"
330 | }
331 | },
332 | "has-unicode": {
333 | "version": "2.0.1",
334 | "resolved": "https://registry.npmjs.org/has-unicode/-/has-unicode-2.0.1.tgz",
335 | "integrity": "sha1-4Ob+aijPUROIVeCG0Wkedx3iqLk="
336 | },
337 | "http-errors": {
338 | "version": "1.7.2",
339 | "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.7.2.tgz",
340 | "integrity": "sha512-uUQBt3H/cSIVfch6i1EuPNy/YsRSOUBXTVfZ+yR7Zjez3qjBz6i9+i4zjNaoqcoFVI4lQJ5plg63TvGfRSDCRg==",
341 | "requires": {
342 | "depd": "~1.1.2",
343 | "inherits": "2.0.3",
344 | "setprototypeof": "1.1.1",
345 | "statuses": ">= 1.5.0 < 2",
346 | "toidentifier": "1.0.0"
347 | }
348 | },
349 | "iconv-lite": {
350 | "version": "0.4.24",
351 | "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz",
352 | "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==",
353 | "requires": {
354 | "safer-buffer": ">= 2.1.2 < 3"
355 | }
356 | },
357 | "ignore-walk": {
358 | "version": "3.0.3",
359 | "resolved": "https://registry.npmjs.org/ignore-walk/-/ignore-walk-3.0.3.tgz",
360 | "integrity": "sha512-m7o6xuOaT1aqheYHKf8W6J5pYH85ZI9w077erOzLje3JsB1gkafkAhHHY19dqjulgIZHFm32Cp5uNZgcQqdJKw==",
361 | "requires": {
362 | "minimatch": "^3.0.4"
363 | }
364 | },
365 | "inflight": {
366 | "version": "1.0.6",
367 | "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz",
368 | "integrity": "sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk=",
369 | "requires": {
370 | "once": "^1.3.0",
371 | "wrappy": "1"
372 | }
373 | },
374 | "inherits": {
375 | "version": "2.0.3",
376 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz",
377 | "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4="
378 | },
379 | "ini": {
380 | "version": "1.3.5",
381 | "resolved": "https://registry.npmjs.org/ini/-/ini-1.3.5.tgz",
382 | "integrity": "sha512-RZY5huIKCMRWDUqZlEi72f/lmXKMvuszcMBduliQ3nnWbx9X/ZBQO7DijMEYS9EhHBb2qacRUMtC7svLwe0lcw=="
383 | },
384 | "ipaddr.js": {
385 | "version": "1.9.0",
386 | "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.0.tgz",
387 | "integrity": "sha512-M4Sjn6N/+O6/IXSJseKqHoFc+5FdGJ22sXqnjTpdZweHK64MzEPAyQZyEU3R/KRv2GLoa7nNtg/C2Ev6m7z+eA=="
388 | },
389 | "is-fullwidth-code-point": {
390 | "version": "1.0.0",
391 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-1.0.0.tgz",
392 | "integrity": "sha1-754xOG8DGn8NZDr4L95QxFfvAMs=",
393 | "requires": {
394 | "number-is-nan": "^1.0.0"
395 | }
396 | },
397 | "isarray": {
398 | "version": "1.0.0",
399 | "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz",
400 | "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE="
401 | },
402 | "jsonwebtoken": {
403 | "version": "8.5.1",
404 | "resolved": "https://registry.npmjs.org/jsonwebtoken/-/jsonwebtoken-8.5.1.tgz",
405 | "integrity": "sha512-XjwVfRS6jTMsqYs0EsuJ4LGxXV14zQybNd4L2r0UvbVnSF9Af8x7p5MzbJ90Ioz/9TI41/hTCvznF/loiSzn8w==",
406 | "requires": {
407 | "jws": "^3.2.2",
408 | "lodash.includes": "^4.3.0",
409 | "lodash.isboolean": "^3.0.3",
410 | "lodash.isinteger": "^4.0.4",
411 | "lodash.isnumber": "^3.0.3",
412 | "lodash.isplainobject": "^4.0.6",
413 | "lodash.isstring": "^4.0.1",
414 | "lodash.once": "^4.0.0",
415 | "ms": "^2.1.1",
416 | "semver": "^5.6.0"
417 | },
418 | "dependencies": {
419 | "ms": {
420 | "version": "2.1.2",
421 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz",
422 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w=="
423 | }
424 | }
425 | },
426 | "jwa": {
427 | "version": "1.4.1",
428 | "resolved": "https://registry.npmjs.org/jwa/-/jwa-1.4.1.tgz",
429 | "integrity": "sha512-qiLX/xhEEFKUAJ6FiBMbes3w9ATzyk5W7Hvzpa/SLYdxNtng+gcurvrI7TbACjIXlsJyr05/S1oUhZrc63evQA==",
430 | "requires": {
431 | "buffer-equal-constant-time": "1.0.1",
432 | "ecdsa-sig-formatter": "1.0.11",
433 | "safe-buffer": "^5.0.1"
434 | }
435 | },
436 | "jws": {
437 | "version": "3.2.2",
438 | "resolved": "https://registry.npmjs.org/jws/-/jws-3.2.2.tgz",
439 | "integrity": "sha512-YHlZCB6lMTllWDtSPHz/ZXTsi8S00usEV6v1tjq8tOUZzw7DpSDWVXjXDre6ed1w/pd495ODpHZYSdkRTsa0HA==",
440 | "requires": {
441 | "jwa": "^1.4.1",
442 | "safe-buffer": "^5.0.1"
443 | }
444 | },
445 | "kareem": {
446 | "version": "2.3.1",
447 | "resolved": "https://registry.npmjs.org/kareem/-/kareem-2.3.1.tgz",
448 | "integrity": "sha512-l3hLhffs9zqoDe8zjmb/mAN4B8VT3L56EUvKNqLFVs9YlFA+zx7ke1DO8STAdDyYNkeSo1nKmjuvQeI12So8Xw=="
449 | },
450 | "lodash.includes": {
451 | "version": "4.3.0",
452 | "resolved": "https://registry.npmjs.org/lodash.includes/-/lodash.includes-4.3.0.tgz",
453 | "integrity": "sha1-YLuYqHy5I8aMoeUTJUgzFISfVT8="
454 | },
455 | "lodash.isboolean": {
456 | "version": "3.0.3",
457 | "resolved": "https://registry.npmjs.org/lodash.isboolean/-/lodash.isboolean-3.0.3.tgz",
458 | "integrity": "sha1-bC4XHbKiV82WgC/UOwGyDV9YcPY="
459 | },
460 | "lodash.isinteger": {
461 | "version": "4.0.4",
462 | "resolved": "https://registry.npmjs.org/lodash.isinteger/-/lodash.isinteger-4.0.4.tgz",
463 | "integrity": "sha1-YZwK89A/iwTDH1iChAt3sRzWg0M="
464 | },
465 | "lodash.isnumber": {
466 | "version": "3.0.3",
467 | "resolved": "https://registry.npmjs.org/lodash.isnumber/-/lodash.isnumber-3.0.3.tgz",
468 | "integrity": "sha1-POdoEMWSjQM1IwGsKHMX8RwLH/w="
469 | },
470 | "lodash.isplainobject": {
471 | "version": "4.0.6",
472 | "resolved": "https://registry.npmjs.org/lodash.isplainobject/-/lodash.isplainobject-4.0.6.tgz",
473 | "integrity": "sha1-fFJqUtibRcRcxpC4gWO+BJf1UMs="
474 | },
475 | "lodash.isstring": {
476 | "version": "4.0.1",
477 | "resolved": "https://registry.npmjs.org/lodash.isstring/-/lodash.isstring-4.0.1.tgz",
478 | "integrity": "sha1-1SfftUVuynzJu5XV2ur4i6VKVFE="
479 | },
480 | "lodash.once": {
481 | "version": "4.1.1",
482 | "resolved": "https://registry.npmjs.org/lodash.once/-/lodash.once-4.1.1.tgz",
483 | "integrity": "sha1-DdOXEhPHxW34gJd9UEyI+0cal6w="
484 | },
485 | "media-typer": {
486 | "version": "0.3.0",
487 | "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz",
488 | "integrity": "sha1-hxDXrwqmJvj/+hzgAWhUUmMlV0g="
489 | },
490 | "memory-pager": {
491 | "version": "1.5.0",
492 | "resolved": "https://registry.npmjs.org/memory-pager/-/memory-pager-1.5.0.tgz",
493 | "integrity": "sha512-ZS4Bp4r/Zoeq6+NLJpP+0Zzm0pR8whtGPf1XExKLJBAczGMnSi3It14OiNCStjQjM6NU1okjQGSxgEZN8eBYKg==",
494 | "optional": true
495 | },
496 | "merge-descriptors": {
497 | "version": "1.0.1",
498 | "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz",
499 | "integrity": "sha1-sAqqVW3YtEVoFQ7J0blT8/kMu2E="
500 | },
501 | "methods": {
502 | "version": "1.1.2",
503 | "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz",
504 | "integrity": "sha1-VSmk1nZUE07cxSZmVoNbD4Ua/O4="
505 | },
506 | "mime": {
507 | "version": "1.6.0",
508 | "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz",
509 | "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg=="
510 | },
511 | "mime-db": {
512 | "version": "1.43.0",
513 | "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.43.0.tgz",
514 | "integrity": "sha512-+5dsGEEovYbT8UY9yD7eE4XTc4UwJ1jBYlgaQQF38ENsKR3wj/8q8RFZrF9WIZpB2V1ArTVFUva8sAul1NzRzQ=="
515 | },
516 | "mime-types": {
517 | "version": "2.1.26",
518 | "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.26.tgz",
519 | "integrity": "sha512-01paPWYgLrkqAyrlDorC1uDwl2p3qZT7yl806vW7DvDoxwXi46jsjFbg+WdwotBIk6/MbEhO/dh5aZ5sNj/dWQ==",
520 | "requires": {
521 | "mime-db": "1.43.0"
522 | }
523 | },
524 | "minimatch": {
525 | "version": "3.0.4",
526 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz",
527 | "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==",
528 | "requires": {
529 | "brace-expansion": "^1.1.7"
530 | }
531 | },
532 | "minimist": {
533 | "version": "0.0.8",
534 | "resolved": "https://registry.npmjs.org/minimist/-/minimist-0.0.8.tgz",
535 | "integrity": "sha1-hX/Kv8M5fSYluCKCYuhqp6ARsF0="
536 | },
537 | "minipass": {
538 | "version": "2.9.0",
539 | "resolved": "https://registry.npmjs.org/minipass/-/minipass-2.9.0.tgz",
540 | "integrity": "sha512-wxfUjg9WebH+CUDX/CdbRlh5SmfZiy/hpkxaRI16Y9W56Pa75sWgd/rvFilSgrauD9NyFymP/+JFV3KwzIsJeg==",
541 | "requires": {
542 | "safe-buffer": "^5.1.2",
543 | "yallist": "^3.0.0"
544 | }
545 | },
546 | "minizlib": {
547 | "version": "1.3.3",
548 | "resolved": "https://registry.npmjs.org/minizlib/-/minizlib-1.3.3.tgz",
549 | "integrity": "sha512-6ZYMOEnmVsdCeTJVE0W9ZD+pVnE8h9Hma/iOwwRDsdQoePpoX56/8B6z3P9VNwppJuBKNRuFDRNRqRWexT9G9Q==",
550 | "requires": {
551 | "minipass": "^2.9.0"
552 | }
553 | },
554 | "mkdirp": {
555 | "version": "0.5.1",
556 | "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.1.tgz",
557 | "integrity": "sha1-MAV0OOrGz3+MR2fzhkjWaX11yQM=",
558 | "requires": {
559 | "minimist": "0.0.8"
560 | }
561 | },
562 | "mongodb": {
563 | "version": "3.4.1",
564 | "resolved": "https://registry.npmjs.org/mongodb/-/mongodb-3.4.1.tgz",
565 | "integrity": "sha512-juqt5/Z42J4DcE7tG7UdVaTKmUC6zinF4yioPfpeOSNBieWSK6qCY+0tfGQcHLKrauWPDdMZVROHJOa8q2pWsA==",
566 | "requires": {
567 | "bson": "^1.1.1",
568 | "require_optional": "^1.0.1",
569 | "safe-buffer": "^5.1.2",
570 | "saslprep": "^1.0.0"
571 | }
572 | },
573 | "mongoose": {
574 | "version": "5.8.7",
575 | "resolved": "https://registry.npmjs.org/mongoose/-/mongoose-5.8.7.tgz",
576 | "integrity": "sha512-PCCuTrdxpUmO86L1geXWE+9AvJRFuneIrMUT2hB/LXZ+5HIGOIfE6OSMU7cd8wFU7JRINP9V73zZ9YsmAbt+Iw==",
577 | "requires": {
578 | "bson": "~1.1.1",
579 | "kareem": "2.3.1",
580 | "mongodb": "3.4.1",
581 | "mongoose-legacy-pluralize": "1.0.2",
582 | "mpath": "0.6.0",
583 | "mquery": "3.2.2",
584 | "ms": "2.1.2",
585 | "regexp-clone": "1.0.0",
586 | "safe-buffer": "5.1.2",
587 | "sift": "7.0.1",
588 | "sliced": "1.0.1"
589 | },
590 | "dependencies": {
591 | "ms": {
592 | "version": "2.1.2",
593 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz",
594 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w=="
595 | }
596 | }
597 | },
598 | "mongoose-legacy-pluralize": {
599 | "version": "1.0.2",
600 | "resolved": "https://registry.npmjs.org/mongoose-legacy-pluralize/-/mongoose-legacy-pluralize-1.0.2.tgz",
601 | "integrity": "sha512-Yo/7qQU4/EyIS8YDFSeenIvXxZN+ld7YdV9LqFVQJzTLye8unujAWPZ4NWKfFA+RNjh+wvTWKY9Z3E5XM6ZZiQ=="
602 | },
603 | "mpath": {
604 | "version": "0.6.0",
605 | "resolved": "https://registry.npmjs.org/mpath/-/mpath-0.6.0.tgz",
606 | "integrity": "sha512-i75qh79MJ5Xo/sbhxrDrPSEG0H/mr1kcZXJ8dH6URU5jD/knFxCVqVC/gVSW7GIXL/9hHWlT9haLbCXWOll3qw=="
607 | },
608 | "mquery": {
609 | "version": "3.2.2",
610 | "resolved": "https://registry.npmjs.org/mquery/-/mquery-3.2.2.tgz",
611 | "integrity": "sha512-XB52992COp0KP230I3qloVUbkLUxJIu328HBP2t2EsxSFtf4W1HPSOBWOXf1bqxK4Xbb66lfMJ+Bpfd9/yZE1Q==",
612 | "requires": {
613 | "bluebird": "3.5.1",
614 | "debug": "3.1.0",
615 | "regexp-clone": "^1.0.0",
616 | "safe-buffer": "5.1.2",
617 | "sliced": "1.0.1"
618 | },
619 | "dependencies": {
620 | "debug": {
621 | "version": "3.1.0",
622 | "resolved": "https://registry.npmjs.org/debug/-/debug-3.1.0.tgz",
623 | "integrity": "sha512-OX8XqP7/1a9cqkxYw2yXss15f26NKWBpDXQd0/uK/KPqdQhxbPa994hnzjcE2VqQpDslf55723cKPUOGSmMY3g==",
624 | "requires": {
625 | "ms": "2.0.0"
626 | }
627 | }
628 | }
629 | },
630 | "ms": {
631 | "version": "2.0.0",
632 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz",
633 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g="
634 | },
635 | "nan": {
636 | "version": "2.14.0",
637 | "resolved": "https://registry.npmjs.org/nan/-/nan-2.14.0.tgz",
638 | "integrity": "sha512-INOFj37C7k3AfaNTtX8RhsTw7qRy7eLET14cROi9+5HAVbbHuIWUHEauBv5qT4Av2tWasiTY1Jw6puUNqRJXQg=="
639 | },
640 | "needle": {
641 | "version": "2.4.0",
642 | "resolved": "https://registry.npmjs.org/needle/-/needle-2.4.0.tgz",
643 | "integrity": "sha512-4Hnwzr3mi5L97hMYeNl8wRW/Onhy4nUKR/lVemJ8gJedxxUyBLm9kkrDColJvoSfwi0jCNhD+xCdOtiGDQiRZg==",
644 | "requires": {
645 | "debug": "^3.2.6",
646 | "iconv-lite": "^0.4.4",
647 | "sax": "^1.2.4"
648 | },
649 | "dependencies": {
650 | "debug": {
651 | "version": "3.2.6",
652 | "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.6.tgz",
653 | "integrity": "sha512-mel+jf7nrtEl5Pn1Qx46zARXKDpBbvzezse7p7LqINmdoIk8PYP5SySaxEmYv6TZ0JyEKA1hsCId6DIhgITtWQ==",
654 | "requires": {
655 | "ms": "^2.1.1"
656 | }
657 | },
658 | "ms": {
659 | "version": "2.1.2",
660 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz",
661 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w=="
662 | }
663 | }
664 | },
665 | "negotiator": {
666 | "version": "0.6.2",
667 | "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.2.tgz",
668 | "integrity": "sha512-hZXc7K2e+PgeI1eDBe/10Ard4ekbfrrqG8Ep+8Jmf4JID2bNg7NvCPOZN+kfF574pFQI7mum2AUqDidoKqcTOw=="
669 | },
670 | "node-pre-gyp": {
671 | "version": "0.13.0",
672 | "resolved": "https://registry.npmjs.org/node-pre-gyp/-/node-pre-gyp-0.13.0.tgz",
673 | "integrity": "sha512-Md1D3xnEne8b/HGVQkZZwV27WUi1ZRuZBij24TNaZwUPU3ZAFtvT6xxJGaUVillfmMKnn5oD1HoGsp2Ftik7SQ==",
674 | "requires": {
675 | "detect-libc": "^1.0.2",
676 | "mkdirp": "^0.5.1",
677 | "needle": "^2.2.1",
678 | "nopt": "^4.0.1",
679 | "npm-packlist": "^1.1.6",
680 | "npmlog": "^4.0.2",
681 | "rc": "^1.2.7",
682 | "rimraf": "^2.6.1",
683 | "semver": "^5.3.0",
684 | "tar": "^4"
685 | }
686 | },
687 | "nopt": {
688 | "version": "4.0.1",
689 | "resolved": "https://registry.npmjs.org/nopt/-/nopt-4.0.1.tgz",
690 | "integrity": "sha1-0NRoWv1UFRk8jHUFYC0NF81kR00=",
691 | "requires": {
692 | "abbrev": "1",
693 | "osenv": "^0.1.4"
694 | }
695 | },
696 | "npm-bundled": {
697 | "version": "1.1.1",
698 | "resolved": "https://registry.npmjs.org/npm-bundled/-/npm-bundled-1.1.1.tgz",
699 | "integrity": "sha512-gqkfgGePhTpAEgUsGEgcq1rqPXA+tv/aVBlgEzfXwA1yiUJF7xtEt3CtVwOjNYQOVknDk0F20w58Fnm3EtG0fA==",
700 | "requires": {
701 | "npm-normalize-package-bin": "^1.0.1"
702 | }
703 | },
704 | "npm-normalize-package-bin": {
705 | "version": "1.0.1",
706 | "resolved": "https://registry.npmjs.org/npm-normalize-package-bin/-/npm-normalize-package-bin-1.0.1.tgz",
707 | "integrity": "sha512-EPfafl6JL5/rU+ot6P3gRSCpPDW5VmIzX959Ob1+ySFUuuYHWHekXpwdUZcKP5C+DS4GEtdJluwBjnsNDl+fSA=="
708 | },
709 | "npm-packlist": {
710 | "version": "1.4.7",
711 | "resolved": "https://registry.npmjs.org/npm-packlist/-/npm-packlist-1.4.7.tgz",
712 | "integrity": "sha512-vAj7dIkp5NhieaGZxBJB8fF4R0078rqsmhJcAfXZ6O7JJhjhPK96n5Ry1oZcfLXgfun0GWTZPOxaEyqv8GBykQ==",
713 | "requires": {
714 | "ignore-walk": "^3.0.1",
715 | "npm-bundled": "^1.0.1"
716 | }
717 | },
718 | "npmlog": {
719 | "version": "4.1.2",
720 | "resolved": "https://registry.npmjs.org/npmlog/-/npmlog-4.1.2.tgz",
721 | "integrity": "sha512-2uUqazuKlTaSI/dC8AzicUck7+IrEaOnN/e0jd3Xtt1KcGpwx30v50mL7oPyr/h9bL3E4aZccVwpwP+5W9Vjkg==",
722 | "requires": {
723 | "are-we-there-yet": "~1.1.2",
724 | "console-control-strings": "~1.1.0",
725 | "gauge": "~2.7.3",
726 | "set-blocking": "~2.0.0"
727 | }
728 | },
729 | "number-is-nan": {
730 | "version": "1.0.1",
731 | "resolved": "https://registry.npmjs.org/number-is-nan/-/number-is-nan-1.0.1.tgz",
732 | "integrity": "sha1-CXtgK1NCKlIsGvuHkDGDNpQaAR0="
733 | },
734 | "object-assign": {
735 | "version": "4.1.1",
736 | "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz",
737 | "integrity": "sha1-IQmtx5ZYh8/AXLvUQsrIv7s2CGM="
738 | },
739 | "on-finished": {
740 | "version": "2.3.0",
741 | "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.3.0.tgz",
742 | "integrity": "sha1-IPEzZIGwg811M3mSoWlxqi2QaUc=",
743 | "requires": {
744 | "ee-first": "1.1.1"
745 | }
746 | },
747 | "once": {
748 | "version": "1.4.0",
749 | "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz",
750 | "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=",
751 | "requires": {
752 | "wrappy": "1"
753 | }
754 | },
755 | "os-homedir": {
756 | "version": "1.0.2",
757 | "resolved": "https://registry.npmjs.org/os-homedir/-/os-homedir-1.0.2.tgz",
758 | "integrity": "sha1-/7xJiDNuDoM94MFox+8VISGqf7M="
759 | },
760 | "os-tmpdir": {
761 | "version": "1.0.2",
762 | "resolved": "https://registry.npmjs.org/os-tmpdir/-/os-tmpdir-1.0.2.tgz",
763 | "integrity": "sha1-u+Z0BseaqFxc/sdm/lc0VV36EnQ="
764 | },
765 | "osenv": {
766 | "version": "0.1.5",
767 | "resolved": "https://registry.npmjs.org/osenv/-/osenv-0.1.5.tgz",
768 | "integrity": "sha512-0CWcCECdMVc2Rw3U5w9ZjqX6ga6ubk1xDVKxtBQPK7wis/0F2r9T6k4ydGYhecl7YUBxBVxhL5oisPsNxAPe2g==",
769 | "requires": {
770 | "os-homedir": "^1.0.0",
771 | "os-tmpdir": "^1.0.0"
772 | }
773 | },
774 | "parseurl": {
775 | "version": "1.3.3",
776 | "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz",
777 | "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ=="
778 | },
779 | "passport": {
780 | "version": "0.4.1",
781 | "resolved": "https://registry.npmjs.org/passport/-/passport-0.4.1.tgz",
782 | "integrity": "sha512-IxXgZZs8d7uFSt3eqNjM9NQ3g3uQCW5avD8mRNoXV99Yig50vjuaez6dQK2qC0kVWPRTujxY0dWgGfT09adjYg==",
783 | "requires": {
784 | "passport-strategy": "1.x.x",
785 | "pause": "0.0.1"
786 | }
787 | },
788 | "passport-jwt": {
789 | "version": "4.0.0",
790 | "resolved": "https://registry.npmjs.org/passport-jwt/-/passport-jwt-4.0.0.tgz",
791 | "integrity": "sha512-BwC0n2GP/1hMVjR4QpnvqA61TxenUMlmfNjYNgK0ZAs0HK4SOQkHcSv4L328blNTLtHq7DbmvyNJiH+bn6C5Mg==",
792 | "requires": {
793 | "jsonwebtoken": "^8.2.0",
794 | "passport-strategy": "^1.0.0"
795 | }
796 | },
797 | "passport-local": {
798 | "version": "1.0.0",
799 | "resolved": "https://registry.npmjs.org/passport-local/-/passport-local-1.0.0.tgz",
800 | "integrity": "sha1-H+YyaMkudWBmJkN+O5BmYsFbpu4=",
801 | "requires": {
802 | "passport-strategy": "1.x.x"
803 | }
804 | },
805 | "passport-strategy": {
806 | "version": "1.0.0",
807 | "resolved": "https://registry.npmjs.org/passport-strategy/-/passport-strategy-1.0.0.tgz",
808 | "integrity": "sha1-tVOaqPwiWj0a0XlHbd8ja0QPUuQ="
809 | },
810 | "path-is-absolute": {
811 | "version": "1.0.1",
812 | "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz",
813 | "integrity": "sha1-F0uSaHNVNP+8es5r9TpanhtcX18="
814 | },
815 | "path-to-regexp": {
816 | "version": "0.1.7",
817 | "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz",
818 | "integrity": "sha1-32BBeABfUi8V60SQ5yR6G/qmf4w="
819 | },
820 | "pause": {
821 | "version": "0.0.1",
822 | "resolved": "https://registry.npmjs.org/pause/-/pause-0.0.1.tgz",
823 | "integrity": "sha1-HUCLP9t2kjuVQ9lvtMnf1TXZy10="
824 | },
825 | "process-nextick-args": {
826 | "version": "2.0.1",
827 | "resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-2.0.1.tgz",
828 | "integrity": "sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag=="
829 | },
830 | "proxy-addr": {
831 | "version": "2.0.5",
832 | "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.5.tgz",
833 | "integrity": "sha512-t/7RxHXPH6cJtP0pRG6smSr9QJidhB+3kXu0KgXnbGYMgzEnUxRQ4/LDdfOwZEMyIh3/xHb8PX3t+lfL9z+YVQ==",
834 | "requires": {
835 | "forwarded": "~0.1.2",
836 | "ipaddr.js": "1.9.0"
837 | }
838 | },
839 | "qs": {
840 | "version": "6.7.0",
841 | "resolved": "https://registry.npmjs.org/qs/-/qs-6.7.0.tgz",
842 | "integrity": "sha512-VCdBRNFTX1fyE7Nb6FYoURo/SPe62QCaAyzJvUjwRaIsc+NePBEniHlvxFmmX56+HZphIGtV0XeCirBtpDrTyQ=="
843 | },
844 | "range-parser": {
845 | "version": "1.2.1",
846 | "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz",
847 | "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg=="
848 | },
849 | "raw-body": {
850 | "version": "2.4.0",
851 | "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.4.0.tgz",
852 | "integrity": "sha512-4Oz8DUIwdvoa5qMJelxipzi/iJIi40O5cGV1wNYp5hvZP8ZN0T+jiNkL0QepXs+EsQ9XJ8ipEDoiH70ySUJP3Q==",
853 | "requires": {
854 | "bytes": "3.1.0",
855 | "http-errors": "1.7.2",
856 | "iconv-lite": "0.4.24",
857 | "unpipe": "1.0.0"
858 | }
859 | },
860 | "rc": {
861 | "version": "1.2.8",
862 | "resolved": "https://registry.npmjs.org/rc/-/rc-1.2.8.tgz",
863 | "integrity": "sha512-y3bGgqKj3QBdxLbLkomlohkvsA8gdAiUQlSBJnBhfn+BPxg4bc62d8TcBW15wavDfgexCgccckhcZvywyQYPOw==",
864 | "requires": {
865 | "deep-extend": "^0.6.0",
866 | "ini": "~1.3.0",
867 | "minimist": "^1.2.0",
868 | "strip-json-comments": "~2.0.1"
869 | },
870 | "dependencies": {
871 | "minimist": {
872 | "version": "1.2.0",
873 | "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.0.tgz",
874 | "integrity": "sha1-o1AIsg9BOD7sH7kU9M1d95omQoQ="
875 | }
876 | }
877 | },
878 | "readable-stream": {
879 | "version": "2.3.7",
880 | "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.7.tgz",
881 | "integrity": "sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw==",
882 | "requires": {
883 | "core-util-is": "~1.0.0",
884 | "inherits": "~2.0.3",
885 | "isarray": "~1.0.0",
886 | "process-nextick-args": "~2.0.0",
887 | "safe-buffer": "~5.1.1",
888 | "string_decoder": "~1.1.1",
889 | "util-deprecate": "~1.0.1"
890 | }
891 | },
892 | "regexp-clone": {
893 | "version": "1.0.0",
894 | "resolved": "https://registry.npmjs.org/regexp-clone/-/regexp-clone-1.0.0.tgz",
895 | "integrity": "sha512-TuAasHQNamyyJ2hb97IuBEif4qBHGjPHBS64sZwytpLEqtBQ1gPJTnOaQ6qmpET16cK14kkjbazl6+p0RRv0yw=="
896 | },
897 | "require_optional": {
898 | "version": "1.0.1",
899 | "resolved": "https://registry.npmjs.org/require_optional/-/require_optional-1.0.1.tgz",
900 | "integrity": "sha512-qhM/y57enGWHAe3v/NcwML6a3/vfESLe/sGM2dII+gEO0BpKRUkWZow/tyloNqJyN6kXSl3RyyM8Ll5D/sJP8g==",
901 | "requires": {
902 | "resolve-from": "^2.0.0",
903 | "semver": "^5.1.0"
904 | }
905 | },
906 | "resolve-from": {
907 | "version": "2.0.0",
908 | "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-2.0.0.tgz",
909 | "integrity": "sha1-lICrIOlP+h2egKgEx+oUdhGWa1c="
910 | },
911 | "rimraf": {
912 | "version": "2.7.1",
913 | "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz",
914 | "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==",
915 | "requires": {
916 | "glob": "^7.1.3"
917 | }
918 | },
919 | "safe-buffer": {
920 | "version": "5.1.2",
921 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz",
922 | "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g=="
923 | },
924 | "safer-buffer": {
925 | "version": "2.1.2",
926 | "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz",
927 | "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg=="
928 | },
929 | "saslprep": {
930 | "version": "1.0.3",
931 | "resolved": "https://registry.npmjs.org/saslprep/-/saslprep-1.0.3.tgz",
932 | "integrity": "sha512-/MY/PEMbk2SuY5sScONwhUDsV2p77Znkb/q3nSVstq/yQzYJOH/Azh29p9oJLsl3LnQwSvZDKagDGBsBwSooag==",
933 | "optional": true,
934 | "requires": {
935 | "sparse-bitfield": "^3.0.3"
936 | }
937 | },
938 | "sax": {
939 | "version": "1.2.4",
940 | "resolved": "https://registry.npmjs.org/sax/-/sax-1.2.4.tgz",
941 | "integrity": "sha512-NqVDv9TpANUjFm0N8uM5GxL36UgKi9/atZw+x7YFnQ8ckwFGKrl4xX4yWtrey3UJm5nP1kUbnYgLopqWNSRhWw=="
942 | },
943 | "semver": {
944 | "version": "5.7.1",
945 | "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz",
946 | "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ=="
947 | },
948 | "send": {
949 | "version": "0.17.1",
950 | "resolved": "https://registry.npmjs.org/send/-/send-0.17.1.tgz",
951 | "integrity": "sha512-BsVKsiGcQMFwT8UxypobUKyv7irCNRHk1T0G680vk88yf6LBByGcZJOTJCrTP2xVN6yI+XjPJcNuE3V4fT9sAg==",
952 | "requires": {
953 | "debug": "2.6.9",
954 | "depd": "~1.1.2",
955 | "destroy": "~1.0.4",
956 | "encodeurl": "~1.0.2",
957 | "escape-html": "~1.0.3",
958 | "etag": "~1.8.1",
959 | "fresh": "0.5.2",
960 | "http-errors": "~1.7.2",
961 | "mime": "1.6.0",
962 | "ms": "2.1.1",
963 | "on-finished": "~2.3.0",
964 | "range-parser": "~1.2.1",
965 | "statuses": "~1.5.0"
966 | },
967 | "dependencies": {
968 | "ms": {
969 | "version": "2.1.1",
970 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.1.tgz",
971 | "integrity": "sha512-tgp+dl5cGk28utYktBsrFqA7HKgrhgPsg6Z/EfhWI4gl1Hwq8B/GmY/0oXZ6nF8hDVesS/FpnYaD/kOWhYQvyg=="
972 | }
973 | }
974 | },
975 | "serve-static": {
976 | "version": "1.14.1",
977 | "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.14.1.tgz",
978 | "integrity": "sha512-JMrvUwE54emCYWlTI+hGrGv5I8dEwmco/00EvkzIIsR7MqrHonbD9pO2MOfFnpFntl7ecpZs+3mW+XbQZu9QCg==",
979 | "requires": {
980 | "encodeurl": "~1.0.2",
981 | "escape-html": "~1.0.3",
982 | "parseurl": "~1.3.3",
983 | "send": "0.17.1"
984 | }
985 | },
986 | "set-blocking": {
987 | "version": "2.0.0",
988 | "resolved": "https://registry.npmjs.org/set-blocking/-/set-blocking-2.0.0.tgz",
989 | "integrity": "sha1-BF+XgtARrppoA93TgrJDkrPYkPc="
990 | },
991 | "setprototypeof": {
992 | "version": "1.1.1",
993 | "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.1.tgz",
994 | "integrity": "sha512-JvdAWfbXeIGaZ9cILp38HntZSFSo3mWg6xGcJJsd+d4aRMOqauag1C63dJfDw7OaMYwEbHMOxEZ1lqVRYP2OAw=="
995 | },
996 | "sift": {
997 | "version": "7.0.1",
998 | "resolved": "https://registry.npmjs.org/sift/-/sift-7.0.1.tgz",
999 | "integrity": "sha512-oqD7PMJ+uO6jV9EQCl0LrRw1OwsiPsiFQR5AR30heR+4Dl7jBBbDLnNvWiak20tzZlSE1H7RB30SX/1j/YYT7g=="
1000 | },
1001 | "signal-exit": {
1002 | "version": "3.0.2",
1003 | "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.2.tgz",
1004 | "integrity": "sha1-tf3AjxKH6hF4Yo5BXiUTK3NkbG0="
1005 | },
1006 | "sliced": {
1007 | "version": "1.0.1",
1008 | "resolved": "https://registry.npmjs.org/sliced/-/sliced-1.0.1.tgz",
1009 | "integrity": "sha1-CzpmK10Ewxd7GSa+qCsD+Dei70E="
1010 | },
1011 | "sparse-bitfield": {
1012 | "version": "3.0.3",
1013 | "resolved": "https://registry.npmjs.org/sparse-bitfield/-/sparse-bitfield-3.0.3.tgz",
1014 | "integrity": "sha1-/0rm5oZWBWuks+eSqzM004JzyhE=",
1015 | "optional": true,
1016 | "requires": {
1017 | "memory-pager": "^1.0.2"
1018 | }
1019 | },
1020 | "statuses": {
1021 | "version": "1.5.0",
1022 | "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.5.0.tgz",
1023 | "integrity": "sha1-Fhx9rBd2Wf2YEfQ3cfqZOBR4Yow="
1024 | },
1025 | "string-width": {
1026 | "version": "1.0.2",
1027 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-1.0.2.tgz",
1028 | "integrity": "sha1-EYvfW4zcUaKn5w0hHgfisLmxB9M=",
1029 | "requires": {
1030 | "code-point-at": "^1.0.0",
1031 | "is-fullwidth-code-point": "^1.0.0",
1032 | "strip-ansi": "^3.0.0"
1033 | }
1034 | },
1035 | "string_decoder": {
1036 | "version": "1.1.1",
1037 | "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz",
1038 | "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==",
1039 | "requires": {
1040 | "safe-buffer": "~5.1.0"
1041 | }
1042 | },
1043 | "strip-ansi": {
1044 | "version": "3.0.1",
1045 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz",
1046 | "integrity": "sha1-ajhfuIU9lS1f8F0Oiq+UJ43GPc8=",
1047 | "requires": {
1048 | "ansi-regex": "^2.0.0"
1049 | }
1050 | },
1051 | "strip-json-comments": {
1052 | "version": "2.0.1",
1053 | "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-2.0.1.tgz",
1054 | "integrity": "sha1-PFMZQukIwml8DsNEhYwobHygpgo="
1055 | },
1056 | "tar": {
1057 | "version": "4.4.13",
1058 | "resolved": "https://registry.npmjs.org/tar/-/tar-4.4.13.tgz",
1059 | "integrity": "sha512-w2VwSrBoHa5BsSyH+KxEqeQBAllHhccyMFVHtGtdMpF4W7IRWfZjFiQceJPChOeTsSDVUpER2T8FA93pr0L+QA==",
1060 | "requires": {
1061 | "chownr": "^1.1.1",
1062 | "fs-minipass": "^1.2.5",
1063 | "minipass": "^2.8.6",
1064 | "minizlib": "^1.2.1",
1065 | "mkdirp": "^0.5.0",
1066 | "safe-buffer": "^5.1.2",
1067 | "yallist": "^3.0.3"
1068 | }
1069 | },
1070 | "toidentifier": {
1071 | "version": "1.0.0",
1072 | "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.0.tgz",
1073 | "integrity": "sha512-yaOH/Pk/VEhBWWTlhI+qXxDFXlejDGcQipMlyxda9nthulaxLZUNcUqFxokp0vcYnvteJln5FNQDRrxj3YcbVw=="
1074 | },
1075 | "type-is": {
1076 | "version": "1.6.18",
1077 | "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz",
1078 | "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==",
1079 | "requires": {
1080 | "media-typer": "0.3.0",
1081 | "mime-types": "~2.1.24"
1082 | }
1083 | },
1084 | "unpipe": {
1085 | "version": "1.0.0",
1086 | "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz",
1087 | "integrity": "sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw="
1088 | },
1089 | "util-deprecate": {
1090 | "version": "1.0.2",
1091 | "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz",
1092 | "integrity": "sha1-RQ1Nyfpw3nMnYvvS1KKJgUGaDM8="
1093 | },
1094 | "utils-merge": {
1095 | "version": "1.0.1",
1096 | "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz",
1097 | "integrity": "sha1-n5VxD1CiZ5R7LMwSR0HBAoQn5xM="
1098 | },
1099 | "vary": {
1100 | "version": "1.1.2",
1101 | "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz",
1102 | "integrity": "sha1-IpnwLG3tMNSllhsLn3RSShj2NPw="
1103 | },
1104 | "wide-align": {
1105 | "version": "1.1.3",
1106 | "resolved": "https://registry.npmjs.org/wide-align/-/wide-align-1.1.3.tgz",
1107 | "integrity": "sha512-QGkOQc8XL6Bt5PwnsExKBPuMKBxnGxWWW3fU55Xt4feHozMUhdUMaBCk290qpm/wG5u/RSKzwdAC4i51YigihA==",
1108 | "requires": {
1109 | "string-width": "^1.0.2 || 2"
1110 | }
1111 | },
1112 | "wrappy": {
1113 | "version": "1.0.2",
1114 | "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz",
1115 | "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8="
1116 | },
1117 | "yallist": {
1118 | "version": "3.1.1",
1119 | "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz",
1120 | "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g=="
1121 | }
1122 | }
1123 | }
1124 |
--------------------------------------------------------------------------------
/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "MernJWT",
3 | "version": "1.0.0",
4 | "description": "",
5 | "main": "app.js",
6 | "scripts": {
7 | "test": "echo \"Error: no test specified\" && exit 1",
8 | "dev": "nodemon app.js",
9 | "start": "node app.js"
10 | },
11 | "keywords": [],
12 | "author": "",
13 | "license": "ISC",
14 | "dependencies": {
15 | "bcrypt": "^3.0.7",
16 | "cookie-parser": "^1.4.4",
17 | "express": "^4.17.1",
18 | "jsonwebtoken": "^8.5.1",
19 | "mongoose": "^5.8.7",
20 | "passport": "^0.4.1",
21 | "passport-jwt": "^4.0.0",
22 | "passport-local": "^1.0.0"
23 | }
24 | }
25 |
--------------------------------------------------------------------------------
/passport.js:
--------------------------------------------------------------------------------
1 | const passport = require('passport');
2 | const LocalStrategy = require('passport-local').Strategy;
3 | const JwtStrategy = require('passport-jwt').Strategy;
4 | const User = require('./models/User');
5 |
6 | const cookieExtractor = req =>{
7 | let token = null;
8 | if(req && req.cookies){
9 | token = req.cookies["access_token"];
10 | }
11 | return token;
12 | }
13 |
14 | // authorization
15 | passport.use(new JwtStrategy({
16 | jwtFromRequest : cookieExtractor,
17 | secretOrKey : "NoobCoder"
18 | },(payload,done)=>{
19 | User.findById({_id : payload.sub},(err,user)=>{
20 | if(err)
21 | return done(err,false);
22 | if(user)
23 | return done(null,user);
24 | else
25 | return done(null,false);
26 | });
27 | }));
28 |
29 | // authenticated local strategy using username and password
30 | passport.use(new LocalStrategy((username,password,done)=>{
31 | User.findOne({username},(err,user)=>{
32 | // something went wrong with database
33 | if(err)
34 | return done(err);
35 | // if no user exist
36 | if(!user)
37 | return done(null,false);
38 | // check if password is correct
39 | user.comparePassword(password,done);
40 |
41 | });
42 | }));
--------------------------------------------------------------------------------
/routes/User.js:
--------------------------------------------------------------------------------
1 | const express = require('express');
2 | const userRouter = express.Router();
3 | const passport = require('passport');
4 | const passportConfig = require('../passport');
5 | const JWT = require('jsonwebtoken');
6 | const User = require('../models/User');
7 | const Todo = require('../models/Todo');
8 |
9 |
10 | const signToken = userID =>{
11 | return JWT.sign({
12 | iss : "NoobCoder",
13 | sub : userID
14 | },"NoobCoder",{expiresIn : "1h"});
15 | }
16 |
17 | userRouter.post('/register',(req,res)=>{
18 | const { username,password,role } = req.body;
19 | User.findOne({username},(err,user)=>{
20 | if(err)
21 | res.status(500).json({message : {msgBody : "Error has occured", msgError: true}});
22 | if(user)
23 | res.status(400).json({message : {msgBody : "Username is already taken", msgError: true}});
24 | else{
25 | const newUser = new User({username,password,role});
26 | newUser.save(err=>{
27 | if(err)
28 | res.status(500).json({message : {msgBody : "Error has occured", msgError: true}});
29 | else
30 | res.status(201).json({message : {msgBody : "Account successfully created", msgError: false}});
31 | });
32 | }
33 | });
34 | });
35 |
36 | userRouter.post('/login',passport.authenticate('local',{session : false}),(req,res)=>{
37 | if(req.isAuthenticated()){
38 | const {_id,username,role} = req.user;
39 | const token = signToken(_id);
40 | res.cookie('access_token',token,{httpOnly: true, sameSite:true});
41 | res.status(200).json({isAuthenticated : true,user : {username,role}});
42 | }
43 | });
44 |
45 | userRouter.get('/logout',passport.authenticate('jwt',{session : false}),(req,res)=>{
46 | res.clearCookie('access_token');
47 | res.json({user:{username : "", role : ""},success : true});
48 | });
49 |
50 | userRouter.post('/todo',passport.authenticate('jwt',{session : false}),(req,res)=>{
51 | const todo = new Todo(req.body);
52 | todo.save(err=>{
53 | if(err)
54 | res.status(500).json({message : {msgBody : "Error has occured", msgError: true}});
55 | else{
56 | req.user.todos.push(todo);
57 | req.user.save(err=>{
58 | if(err)
59 | res.status(500).json({message : {msgBody : "Error has occured", msgError: true}});
60 | else
61 | res.status(200).json({message : {msgBody : "Successfully created todo", msgError : false}});
62 | });
63 | }
64 | })
65 | });
66 |
67 | userRouter.get('/todos',passport.authenticate('jwt',{session : false}),(req,res)=>{
68 | User.findById({_id : req.user._id}).populate('todos').exec((err,document)=>{
69 | if(err)
70 | res.status(500).json({message : {msgBody : "Error has occured", msgError: true}});
71 | else{
72 | res.status(200).json({todos : document.todos, authenticated : true});
73 | }
74 | });
75 | });
76 |
77 | userRouter.get('/admin',passport.authenticate('jwt',{session : false}),(req,res)=>{
78 | if(req.user.role === 'admin'){
79 | res.status(200).json({message : {msgBody : 'You are an admin', msgError : false}});
80 | }
81 | else
82 | res.status(403).json({message : {msgBody : "You're not an admin,go away", msgError : true}});
83 | });
84 |
85 | userRouter.get('/authenticated',passport.authenticate('jwt',{session : false}),(req,res)=>{
86 | const {username,role} = req.user;
87 | res.status(200).json({isAuthenticated : true, user : {username,role}});
88 | });
89 |
90 |
91 |
92 |
93 |
94 | module.exports = userRouter;
--------------------------------------------------------------------------------