├── .gitignore ├── README.md ├── client ├── .eslintcache ├── .gitignore ├── README.md ├── package-lock.json ├── package.json ├── public │ ├── favicon.ico │ ├── index.html │ ├── logo192.png │ ├── logo512.png │ ├── manifest.json │ └── robots.txt ├── src │ ├── App.css │ ├── App.js │ ├── components │ │ ├── cards │ │ │ ├── CardItem.js │ │ │ └── Cards.js │ │ ├── layouts │ │ │ └── Navbar.js │ │ ├── pages │ │ │ ├── Game.js │ │ │ ├── Home.js │ │ │ ├── Register.js │ │ │ └── Signin.js │ │ └── routing │ │ │ └── privateRoute.js │ ├── context │ │ ├── auth │ │ │ ├── AuthState.js │ │ │ ├── authContext.js │ │ │ └── authReducer.js │ │ ├── history │ │ │ ├── HistoryState.js │ │ │ ├── historyContext.js │ │ │ └── historyReducer.js │ │ └── types.js │ ├── index.js │ └── utils │ │ └── setAuthToken.js └── yarn.lock ├── config ├── db.js ├── default.json └── production.json ├── middleware └── auth.js ├── models ├── Games.js └── Users.js ├── package-lock.json ├── package.json ├── routes ├── auth.js ├── history.js └── users.js └── server.js /.gitignore: -------------------------------------------------------------------------------- 1 | node_modules/ -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # Memory Game Web App 2 | Web app similar to memory boardgame. Create an account, choose a difficulty level, choose a card theme, and find all the pairs of cards! User score will update after each game. 3 | 4 | # Motivation 5 | To build a full stack MERN app from scratch equipt with user sign in. To practice building backend server and connecting to front end. To practice using state management in a React app. 6 | 7 | # Screenshot 8 | ![screenshot](https://live.staticflickr.com/65535/50996655401_49ed9fbc3e_n.jpg) 9 | 10 | # Tech Used 11 | * HTML, CSS, MaterializeCSS, JS, React for front end 12 | * MongoDB, Express, Node, Mongoose for backend 13 | * Context API for state management 14 | * Nodemon and Concurrently for development 15 | * JWToken and Bcrypt for hashing passwords 16 | * Robohash for card images 17 | * Axios for image API requests 18 | * Heroku for deployment 19 | -------------------------------------------------------------------------------- /client/.eslintcache: -------------------------------------------------------------------------------- 1 | [{"C:\\Users\\ronda\\Desktop\\Projects\\memory-game\\client\\src\\App.js":"1","C:\\Users\\ronda\\Desktop\\Projects\\memory-game\\client\\src\\index.js":"2","C:\\Users\\ronda\\Desktop\\Projects\\memory-game\\client\\src\\context\\history\\historyState.js":"3","C:\\Users\\ronda\\Desktop\\Projects\\memory-game\\client\\src\\context\\history\\HistoryState.js":"4","C:\\Users\\ronda\\Desktop\\Projects\\memory-game\\client\\src\\context\\types.js":"5","C:\\Users\\ronda\\Desktop\\Projects\\memory-game\\client\\src\\components\\pages\\Register.js":"6","C:\\Users\\ronda\\Desktop\\Projects\\memory-game\\client\\src\\components\\pages\\Signin.js":"7","C:\\Users\\ronda\\Desktop\\Projects\\memory-game\\client\\src\\components\\pages\\Home.js":"8","C:\\Users\\ronda\\Desktop\\Projects\\memory-game\\client\\src\\context\\history\\historyContext.js":"9","C:\\Users\\ronda\\Desktop\\Projects\\memory-game\\client\\src\\components\\cards\\Cards.js":"10","C:\\Users\\ronda\\Desktop\\Projects\\memory-game\\client\\src\\components\\pages\\Game.js":"11","C:\\Users\\ronda\\Desktop\\Projects\\memory-game\\client\\src\\components\\cards\\CardItem.js":"12","C:\\Users\\ronda\\Desktop\\Projects\\memory-game\\client\\src\\components\\layouts\\Navbar.js":"13","C:\\Users\\ronda\\Desktop\\Projects\\memory-game\\client\\src\\context\\history\\historyReducer.js":"14","C:\\Users\\ronda\\Desktop\\Projects\\memory-game\\client\\src\\context\\auth\\AuthState.js":"15","C:\\Users\\ronda\\Desktop\\Projects\\memory-game\\client\\src\\context\\auth\\authContext.js":"16","C:\\Users\\ronda\\Desktop\\Projects\\memory-game\\client\\src\\context\\auth\\authReducer.js":"17","C:\\Users\\ronda\\Desktop\\Projects\\memory-game\\client\\src\\components\\routing\\privateRoute.js":"18","C:\\Users\\ronda\\Desktop\\Projects\\memory-game\\client\\src\\utils\\setAuthToken.js":"19"},{"size":1263,"mtime":1608654482237,"results":"20","hashOfConfig":"21"},{"size":197,"mtime":1606491032016,"results":"22","hashOfConfig":"23"},{"size":1460,"mtime":1606849003869,"results":"24","hashOfConfig":"21"},{"size":2311,"mtime":1608761080125,"results":"25","hashOfConfig":"21"},{"size":597,"mtime":1608572496080,"results":"26","hashOfConfig":"23"},{"size":2978,"mtime":1628282354750,"results":"27","hashOfConfig":"23"},{"size":2549,"mtime":1628282312277,"results":"28","hashOfConfig":"23"},{"size":6394,"mtime":1628342243181,"results":"29","hashOfConfig":"23"},{"size":117,"mtime":1606848183399,"results":"30","hashOfConfig":"23"},{"size":4329,"mtime":1608762813435,"results":"31","hashOfConfig":"23"},{"size":2986,"mtime":1608763434961,"results":"32","hashOfConfig":"21"},{"size":1174,"mtime":1608762797160,"results":"33","hashOfConfig":"21"},{"size":1344,"mtime":1628340321905,"results":"34","hashOfConfig":"23"},{"size":908,"mtime":1608760288375,"results":"35","hashOfConfig":"21"},{"size":3260,"mtime":1608760902782,"results":"36","hashOfConfig":"21"},{"size":111,"mtime":1608226788150,"results":"37","hashOfConfig":"21"},{"size":1182,"mtime":1608760265210,"results":"38","hashOfConfig":"21"},{"size":707,"mtime":1608761196149,"results":"39","hashOfConfig":"21"},{"size":286,"mtime":1608760133065,"results":"40","hashOfConfig":"23"},{"filePath":"41","messages":"42","errorCount":0,"warningCount":0,"fixableErrorCount":0,"fixableWarningCount":0},"97m5x6",{"filePath":"43","messages":"44","errorCount":0,"warningCount":0,"fixableErrorCount":0,"fixableWarningCount":0,"usedDeprecatedRules":"45"},"1h97abk",{"filePath":"46","messages":"47","errorCount":12,"warningCount":3,"fixableErrorCount":0,"fixableWarningCount":0,"source":null},{"filePath":"48","messages":"49","errorCount":0,"warningCount":0,"fixableErrorCount":0,"fixableWarningCount":0,"usedDeprecatedRules":"50"},{"filePath":"51","messages":"52","errorCount":0,"warningCount":0,"fixableErrorCount":0,"fixableWarningCount":0},{"filePath":"53","messages":"54","errorCount":0,"warningCount":0,"fixableErrorCount":0,"fixableWarningCount":0},{"filePath":"55","messages":"56","errorCount":0,"warningCount":0,"fixableErrorCount":0,"fixableWarningCount":0},{"filePath":"57","messages":"58","errorCount":0,"warningCount":4,"fixableErrorCount":0,"fixableWarningCount":0,"source":null},{"filePath":"59","messages":"60","errorCount":0,"warningCount":0,"fixableErrorCount":0,"fixableWarningCount":0},{"filePath":"61","messages":"62","errorCount":0,"warningCount":0,"fixableErrorCount":0,"fixableWarningCount":0},{"filePath":"63","messages":"64","errorCount":0,"warningCount":0,"fixableErrorCount":0,"fixableWarningCount":0},{"filePath":"65","messages":"66","errorCount":0,"warningCount":0,"fixableErrorCount":0,"fixableWarningCount":0},{"filePath":"67","messages":"68","errorCount":0,"warningCount":0,"fixableErrorCount":0,"fixableWarningCount":0},{"filePath":"69","messages":"70","errorCount":0,"warningCount":0,"fixableErrorCount":0,"fixableWarningCount":0},{"filePath":"71","messages":"72","errorCount":0,"warningCount":0,"fixableErrorCount":0,"fixableWarningCount":0},{"filePath":"73","messages":"74","errorCount":0,"warningCount":0,"fixableErrorCount":0,"fixableWarningCount":0,"usedDeprecatedRules":"75"},{"filePath":"76","messages":"77","errorCount":0,"warningCount":0,"fixableErrorCount":0,"fixableWarningCount":0},{"filePath":"78","messages":"79","errorCount":0,"warningCount":0,"fixableErrorCount":0,"fixableWarningCount":0,"usedDeprecatedRules":"80"},{"filePath":"81","messages":"82","errorCount":0,"warningCount":0,"fixableErrorCount":0,"fixableWarningCount":0},"C:\\Users\\ronda\\Desktop\\Projects\\memory-game\\client\\src\\App.js",[],"C:\\Users\\ronda\\Desktop\\Projects\\memory-game\\client\\src\\index.js",[],["83","84"],"C:\\Users\\ronda\\Desktop\\Projects\\memory-game\\client\\src\\context\\history\\historyState.js",["85","86","87","88","89","90","91","92","93","94","95","96","97","98","99"],"C:\\Users\\ronda\\Desktop\\Projects\\memory-game\\client\\src\\context\\history\\HistoryState.js",[],["100","101"],"C:\\Users\\ronda\\Desktop\\Projects\\memory-game\\client\\src\\context\\types.js",[],"C:\\Users\\ronda\\Desktop\\Projects\\memory-game\\client\\src\\components\\pages\\Register.js",[],"C:\\Users\\ronda\\Desktop\\Projects\\memory-game\\client\\src\\components\\pages\\Signin.js",[],"C:\\Users\\ronda\\Desktop\\Projects\\memory-game\\client\\src\\components\\pages\\Home.js",["102","103","104","105"],"C:\\Users\\ronda\\Desktop\\Projects\\memory-game\\client\\src\\context\\history\\historyContext.js",[],"C:\\Users\\ronda\\Desktop\\Projects\\memory-game\\client\\src\\components\\cards\\Cards.js",[],"C:\\Users\\ronda\\Desktop\\Projects\\memory-game\\client\\src\\components\\pages\\Game.js",[],"C:\\Users\\ronda\\Desktop\\Projects\\memory-game\\client\\src\\components\\cards\\CardItem.js",[],"C:\\Users\\ronda\\Desktop\\Projects\\memory-game\\client\\src\\components\\layouts\\Navbar.js",[],"C:\\Users\\ronda\\Desktop\\Projects\\memory-game\\client\\src\\context\\history\\historyReducer.js",[],"C:\\Users\\ronda\\Desktop\\Projects\\memory-game\\client\\src\\context\\auth\\AuthState.js",[],"C:\\Users\\ronda\\Desktop\\Projects\\memory-game\\client\\src\\context\\auth\\authContext.js",[],["106","107"],"C:\\Users\\ronda\\Desktop\\Projects\\memory-game\\client\\src\\context\\auth\\authReducer.js",[],"C:\\Users\\ronda\\Desktop\\Projects\\memory-game\\client\\src\\components\\routing\\privateRoute.js",[],["108","109"],"C:\\Users\\ronda\\Desktop\\Projects\\memory-game\\client\\src\\utils\\setAuthToken.js",[],{"ruleId":"110","replacedBy":"111"},{"ruleId":"112","replacedBy":"113"},{"ruleId":"114","severity":1,"message":"115","line":4,"column":10,"nodeType":"116","messageId":"117","endLine":4,"endColumn":18},{"ruleId":"114","severity":1,"message":"118","line":4,"column":20,"nodeType":"116","messageId":"117","endLine":4,"endColumn":32},{"ruleId":"119","severity":2,"message":"120","line":11,"column":15,"nodeType":"116","messageId":"121","endLine":11,"endColumn":20},{"ruleId":"119","severity":2,"message":"122","line":12,"column":20,"nodeType":"116","messageId":"121","endLine":12,"endColumn":28},{"ruleId":"119","severity":2,"message":"120","line":18,"column":15,"nodeType":"116","messageId":"121","endLine":18,"endColumn":20},{"ruleId":"119","severity":2,"message":"122","line":19,"column":20,"nodeType":"116","messageId":"121","endLine":19,"endColumn":28},{"ruleId":"119","severity":2,"message":"120","line":25,"column":15,"nodeType":"116","messageId":"121","endLine":25,"endColumn":20},{"ruleId":"119","severity":2,"message":"122","line":26,"column":20,"nodeType":"116","messageId":"121","endLine":26,"endColumn":28},{"ruleId":"119","severity":2,"message":"120","line":32,"column":15,"nodeType":"116","messageId":"121","endLine":32,"endColumn":20},{"ruleId":"119","severity":2,"message":"123","line":33,"column":20,"nodeType":"116","messageId":"121","endLine":33,"endColumn":32},{"ruleId":"119","severity":2,"message":"120","line":39,"column":15,"nodeType":"116","messageId":"121","endLine":39,"endColumn":20},{"ruleId":"119","severity":2,"message":"124","line":40,"column":20,"nodeType":"116","messageId":"121","endLine":40,"endColumn":26},{"ruleId":"119","severity":2,"message":"120","line":46,"column":15,"nodeType":"116","messageId":"121","endLine":46,"endColumn":20},{"ruleId":"119","severity":2,"message":"124","line":47,"column":20,"nodeType":"116","messageId":"121","endLine":47,"endColumn":26},{"ruleId":"114","severity":1,"message":"125","line":54,"column":17,"nodeType":"116","messageId":"117","endLine":54,"endColumn":25},{"ruleId":"110","replacedBy":"126"},{"ruleId":"112","replacedBy":"127"},{"ruleId":"128","severity":1,"message":"129","line":110,"column":15,"nodeType":"130","endLine":114,"endColumn":16},{"ruleId":"128","severity":1,"message":"129","line":119,"column":19,"nodeType":"130","endLine":124,"endColumn":20},{"ruleId":"128","severity":1,"message":"129","line":130,"column":19,"nodeType":"130","endLine":135,"endColumn":20},{"ruleId":"128","severity":1,"message":"129","line":141,"column":19,"nodeType":"130","endLine":146,"endColumn":20},{"ruleId":"110","replacedBy":"131"},{"ruleId":"112","replacedBy":"132"},{"ruleId":"110","replacedBy":"133"},{"ruleId":"112","replacedBy":"134"},"no-native-reassign",["135"],"no-negated-in-lhs",["136"],"no-unused-vars","'ADD_GAME' is defined but never used.","Identifier","unusedVar","'DELETE_GAMES' is defined but never used.","no-undef","'User1' is not defined.","undef","'beginner' is not defined.","'intermediate' is not defined.","'expert' is not defined.","'dispatch' is assigned a value but never used.",["135"],["136"],"jsx-a11y/anchor-is-valid","The href attribute requires a valid value to be accessible. Provide a valid, navigable address as the href value. If you cannot provide a valid href, but still need the element to resemble a link, use a button and change it with appropriate styles. Learn more: https://github.com/evcohen/eslint-plugin-jsx-a11y/blob/master/docs/rules/anchor-is-valid.md","JSXOpeningElement",["135"],["136"],["135"],["136"],"no-global-assign","no-unsafe-negation"] -------------------------------------------------------------------------------- /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 | # Getting Started with Create React App 2 | 3 | This project was bootstrapped with [Create React App](https://github.com/facebook/create-react-app). 4 | 5 | ## Available Scripts 6 | 7 | In the project directory, you can run: 8 | 9 | ### `yarn start` 10 | 11 | Runs the app in the development mode.\ 12 | Open [http://localhost:3000](http://localhost:3000) to view it in the browser. 13 | 14 | The page will reload if you make edits.\ 15 | You will also see any lint errors in the console. 16 | 17 | ### `yarn test` 18 | 19 | Launches the test runner in the interactive watch mode.\ 20 | See the section about [running tests](https://facebook.github.io/create-react-app/docs/running-tests) for more information. 21 | 22 | ### `yarn build` 23 | 24 | Builds the app for production to the `build` folder.\ 25 | It correctly bundles React in production mode and optimizes the build for the best performance. 26 | 27 | The build is minified and the filenames include the hashes.\ 28 | Your app is ready to be deployed! 29 | 30 | See the section about [deployment](https://facebook.github.io/create-react-app/docs/deployment) for more information. 31 | 32 | ### `yarn eject` 33 | 34 | **Note: this is a one-way operation. Once you `eject`, you can’t go back!** 35 | 36 | 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. 37 | 38 | 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. 39 | 40 | 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. 41 | 42 | ## Learn More 43 | 44 | You can learn more in the [Create React App documentation](https://facebook.github.io/create-react-app/docs/getting-started). 45 | 46 | To learn React, check out the [React documentation](https://reactjs.org/). 47 | 48 | ### Code Splitting 49 | 50 | This section has moved here: [https://facebook.github.io/create-react-app/docs/code-splitting](https://facebook.github.io/create-react-app/docs/code-splitting) 51 | 52 | ### Analyzing the Bundle Size 53 | 54 | This section has moved here: [https://facebook.github.io/create-react-app/docs/analyzing-the-bundle-size](https://facebook.github.io/create-react-app/docs/analyzing-the-bundle-size) 55 | 56 | ### Making a Progressive Web App 57 | 58 | This section has moved here: [https://facebook.github.io/create-react-app/docs/making-a-progressive-web-app](https://facebook.github.io/create-react-app/docs/making-a-progressive-web-app) 59 | 60 | ### Advanced Configuration 61 | 62 | This section has moved here: [https://facebook.github.io/create-react-app/docs/advanced-configuration](https://facebook.github.io/create-react-app/docs/advanced-configuration) 63 | 64 | ### Deployment 65 | 66 | This section has moved here: [https://facebook.github.io/create-react-app/docs/deployment](https://facebook.github.io/create-react-app/docs/deployment) 67 | 68 | ### `yarn build` fails to minify 69 | 70 | This section has moved here: [https://facebook.github.io/create-react-app/docs/troubleshooting#npm-run-build-fails-to-minify](https://facebook.github.io/create-react-app/docs/troubleshooting#npm-run-build-fails-to-minify) 71 | -------------------------------------------------------------------------------- /client/package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "client", 3 | "version": "0.1.0", 4 | "private": true, 5 | "dependencies": { 6 | "@testing-library/jest-dom": "^5.11.4", 7 | "@testing-library/react": "^11.1.0", 8 | "@testing-library/user-event": "^12.1.10", 9 | "axios": "^0.21.0", 10 | "react": "^17.0.1", 11 | "react-dom": "^17.0.1", 12 | "react-router-dom": "^5.2.0", 13 | "react-scripts": "4.0.1", 14 | "react-transition-group": "^4.4.1", 15 | "uuid": "^8.3.1", 16 | "web-vitals": "^0.2.4" 17 | }, 18 | "scripts": { 19 | "start": "react-scripts start", 20 | "build": "react-scripts build", 21 | "test": "react-scripts test", 22 | "eject": "react-scripts eject" 23 | }, 24 | "eslintConfig": { 25 | "extends": [ 26 | "react-app", 27 | "react-app/jest" 28 | ] 29 | }, 30 | "browserslist": { 31 | "production": [ 32 | ">0.2%", 33 | "not dead", 34 | "not op_mini all" 35 | ], 36 | "development": [ 37 | "last 1 chrome version", 38 | "last 1 firefox version", 39 | "last 1 safari version" 40 | ] 41 | }, 42 | "proxy": "http://localhost:5000" 43 | } 44 | -------------------------------------------------------------------------------- /client/public/favicon.ico: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/awesomePC/MERN-stack-memory-game/8f43bb754a7734752521fe4a2060a7ccaf11eb47/client/public/favicon.ico -------------------------------------------------------------------------------- /client/public/index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 16 | 17 | 18 | 19 | 20 | 21 | 22 | 23 | 24 | 25 | 26 | 27 | 28 | Memory Game | Challenge Your Brain! 29 | 30 | 31 | 32 |
33 | 34 | 35 | -------------------------------------------------------------------------------- /client/public/logo192.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/awesomePC/MERN-stack-memory-game/8f43bb754a7734752521fe4a2060a7ccaf11eb47/client/public/logo192.png -------------------------------------------------------------------------------- /client/public/logo512.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/awesomePC/MERN-stack-memory-game/8f43bb754a7734752521fe4a2060a7ccaf11eb47/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 | Disallow: 4 | -------------------------------------------------------------------------------- /client/src/App.css: -------------------------------------------------------------------------------- 1 | .game-board { 2 | display: flex; 3 | justify-content: center; 4 | } 5 | 6 | .mem-cards-container { 7 | display: flex; 8 | flex-wrap: wrap; 9 | height: auto; 10 | justify-content: center; 11 | align-items: center; 12 | margin-top: 20px; 13 | } 14 | 15 | .mem-card { 16 | height: 100px; 17 | width: 100px; 18 | background-color: #546e7a; 19 | border: 2px solid black; 20 | border-radius: 5px; 21 | margin: 2px; 22 | } 23 | 24 | img { 25 | max-height: 100%; 26 | max-width: 100%; 27 | } 28 | 29 | .form-container { 30 | margin-top: 20px; 31 | } 32 | -------------------------------------------------------------------------------- /client/src/App.js: -------------------------------------------------------------------------------- 1 | import React from 'react'; 2 | import { BrowserRouter as Router, Route, Switch } from 'react-router-dom'; 3 | import Navbar from './components/layouts/Navbar'; 4 | import Home from './components/pages/Home'; 5 | import Signin from './components/pages/Signin'; 6 | import Register from './components/pages/Register'; 7 | import Game from './components/pages/Game'; 8 | import PrivateRoute from './components/routing/privateRoute'; 9 | 10 | import HistoryState from './context/history/HistoryState'; 11 | import AuthState from './context/auth/AuthState'; 12 | import setAuthToken from './utils/setAuthToken'; 13 | import './App.css'; 14 | 15 | if (localStorage.token) { 16 | setAuthToken(localStorage.token); 17 | } 18 | 19 | const App = () => { 20 | return ( 21 | // auth and history contexts 22 | 23 | 24 | {/* router for navigation */} 25 | 26 |
27 | {/* navbar component */} 28 | 29 |
30 | {/* private/public routes for app */} 31 | 32 | 33 | 34 | 35 | 36 | 37 |
38 |
39 |
40 |
41 |
42 | ); 43 | }; 44 | 45 | export default App; 46 | -------------------------------------------------------------------------------- /client/src/components/cards/CardItem.js: -------------------------------------------------------------------------------- 1 | import React, { useState, useEffect } from 'react'; 2 | 3 | // bring in props 4 | const CardItem = ({ id, imageId, shownCards, cardClicked, source }) => { 5 | // set state 6 | const [show, setShow] = useState(false); 7 | 8 | // check if cardItem should be displayed, display if so 9 | useEffect(() => { 10 | let buff = shownCards.includes(id); 11 | setShow(buff); 12 | // eslint-disable-next-line 13 | }, [shownCards]); 14 | 15 | // call cardClicked with card div 16 | const onClick = (e) => { 17 | cardClicked(e.target); 18 | }; 19 | 20 | return ( 21 | // materializeCSS card 22 |
33 | card-figure 42 |
43 | ); 44 | }; 45 | 46 | export default CardItem; 47 | -------------------------------------------------------------------------------- /client/src/components/cards/Cards.js: -------------------------------------------------------------------------------- 1 | import React, { useState, useEffect } from 'react'; 2 | import CardItem from './CardItem'; 3 | 4 | // declare props and initial state 5 | const Cards = ({ 6 | updateActive, 7 | updateNumOfMoves, 8 | currentLevel, 9 | currentTheme, 10 | updateNewGame, 11 | }) => { 12 | const [images, setImages] = useState([]); 13 | const [shownCards, setShownCards] = useState([]); 14 | const [currCards, setCurrCards] = useState([]); 15 | const [disableClick, setDisableClick] = useState(false); 16 | const [count, setCount] = useState(1); 17 | 18 | useEffect(() => { 19 | // declare number of cards based on level 20 | let number; 21 | switch (currentLevel) { 22 | case 'beginner': 23 | number = 12; 24 | break; 25 | case 'intermediate': 26 | number = 20; 27 | break; 28 | case 'expert': 29 | number = 30; 30 | break; 31 | default: 32 | number = 12; 33 | } 34 | 35 | // declare array with integers up to 'number', repeated once, and then randomize 36 | let buffer = []; 37 | for (let i = 1; i <= number; i++) { 38 | let temp; 39 | if (i <= number / 2) { 40 | temp = i; 41 | } else { 42 | temp = i - number / 2; 43 | } 44 | buffer.push({ id: temp }); 45 | } 46 | 47 | buffer.sort((a, b) => { 48 | return 0.5 - Math.random(); 49 | }); 50 | 51 | // save random int array to state 52 | setImages(buffer); 53 | // eslint-disable-next-line react-hooks/exhaustive-deps 54 | }, []); 55 | 56 | // declare variables 57 | let curId = 0; 58 | let curImgId = 0; 59 | 60 | // declare source based on user's chosen theme for robohash api 61 | let source; 62 | switch (currentTheme) { 63 | case 'robots': 64 | source = '?set=set1'; 65 | break; 66 | case 'cats': 67 | source = '?set=set4'; 68 | break; 69 | case 'monsters': 70 | source = '?set=set2'; 71 | break; 72 | default: 73 | source = '?set=set1'; 74 | } 75 | 76 | // handle cards clicked 77 | const cardClicked = (cardDiv) => { 78 | // delcare imgId and div id 79 | curImgId = parseInt(cardDiv.getAttribute('imgid')); 80 | curId = parseInt(cardDiv.id); 81 | // if card is first clicked in pair, store values 82 | if (currCards.length === 0) { 83 | setCurrCards((currCards) => [...currCards, curImgId]); 84 | setShownCards((shownCards) => [...shownCards, curId]); 85 | // runs if card second clicked in pair 86 | } else { 87 | // increase count state by 1 88 | setCount(count + 1); 89 | // if cards match, store second card and clear currCards state 90 | if (currCards.includes(curImgId)) { 91 | setShownCards((shownCards) => [...shownCards, curId]); 92 | setCurrCards([]); 93 | // runs if all cards found 94 | if (shownCards.length === images.length - 1) { 95 | // store number of moves, pass game object to updateNewGame function 96 | updateNumOfMoves(count); 97 | setTimeout(() => { 98 | updateNewGame({ 99 | gameLevel: currentLevel, 100 | numOfMoves: count, 101 | date: Date.now(), 102 | }); 103 | updateActive(); 104 | }, 1000); 105 | } 106 | // runs if second card does not match firs card 107 | } else { 108 | // disable click, show second card, clear current cards 109 | setDisableClick(true); 110 | setShownCards((shownCards) => [...shownCards, curId]); 111 | setCurrCards([]); 112 | // after 2 seconds, flip last two cards back over and enable clicking 113 | setTimeout(() => { 114 | let shownCardsTempArr = [...shownCards]; 115 | shownCardsTempArr.splice(-1, 1); 116 | setShownCards(shownCardsTempArr); 117 | setDisableClick(false); 118 | }, 2000); 119 | } 120 | } 121 | }; 122 | 123 | // log message if shown card clicked 124 | const noClicking = () => { 125 | console.log('nope!'); 126 | }; 127 | 128 | return ( 129 | // materializeCSS card 130 |
142 | {images.map((image, index) => ( 143 | 151 | ))} 152 |
153 | ); 154 | }; 155 | 156 | export default Cards; 157 | -------------------------------------------------------------------------------- /client/src/components/layouts/Navbar.js: -------------------------------------------------------------------------------- 1 | import React, { Fragment, useContext } from 'react'; 2 | import { Link } from 'react-router-dom'; 3 | import AuthContext from '../../context/auth/authContext'; 4 | 5 | const Navbar = () => { 6 | // declare and destructure authcontext 7 | const authContext = useContext(AuthContext); 8 | 9 | const { isAuthenticated, logout, user } = authContext; 10 | 11 | // logout user 12 | const onLogOut = () => { 13 | logout(); 14 | }; 15 | 16 | // fragment displays of user signed in 17 | const authLinks = ( 18 | 19 |
  • Hello, {user && user.name}!
  • 20 |
  • 21 | Sign Out 22 |
  • 23 |
    24 | ); 25 | 26 | // fragment displays if user is not signed in 27 | const guestLinks = ( 28 | 29 |
  • 30 | Register 31 |
  • 32 |
  • 33 | Sign In 34 |
  • 35 |
    36 | ); 37 | 38 | return ( 39 | // materializeCSS navbar 40 | 52 | ); 53 | }; 54 | 55 | export default Navbar; 56 | -------------------------------------------------------------------------------- /client/src/components/pages/Game.js: -------------------------------------------------------------------------------- 1 | import React, { useState, useContext, useEffect } from 'react'; 2 | import { Link } from 'react-router-dom'; 3 | import HistoryContext from '../../context/history/historyContext'; 4 | import AuthContext from '../../context/auth/authContext'; 5 | import Cards from '../cards/Cards'; 6 | 7 | const Game = () => { 8 | // declare historyContext, authContext, destructure historyContext 9 | const historyContext = useContext(HistoryContext); 10 | const authContext = useContext(AuthContext); 11 | 12 | const { currentLevel, currentTheme, addNewGame } = historyContext; 13 | 14 | // delcare component state 15 | const [active, setActive] = useState(true); 16 | const [curNumOfMoves, setCurNumOfMoves] = useState(0); 17 | const [newGame, setNewGame] = useState({ 18 | gameLevel: '', 19 | numOfMoves: 0, 20 | date: Date.now(), 21 | }); 22 | 23 | // authenticate user if game page is loaded 24 | useEffect(() => { 25 | authContext.loadUser(); 26 | // eslint-disable-next-line 27 | }, []); 28 | 29 | // change active state boolean 30 | const updateActive = () => { 31 | let cur = active; 32 | setActive(!cur); 33 | }; 34 | 35 | // store current number of moves 36 | const updateCurNumOfMoves = (count) => { 37 | setCurNumOfMoves(count); 38 | }; 39 | 40 | // store newGame object 41 | const updateNewGame = (newGame) => { 42 | setNewGame(newGame); 43 | }; 44 | 45 | // add new game object to context and database 46 | const onClick = (e) => { 47 | addNewGame(newGame); 48 | }; 49 | 50 | return ( 51 | // materializeCSS card 52 | // prevent view if page loading 53 | !authContext.loading && ( 54 |
    55 | {/* show game cards if active state true */} 56 | {active ? ( 57 | 64 | ) : ( 65 | // show congratulations card if game not active 66 |
    67 |
    68 |
    69 |
    70 | Congratulations! 71 |

    72 | 73 | You won in {curNumOfMoves} moves! 74 | 75 |

    76 |
    77 |
    78 | 83 | End Game 84 | 85 |
    86 |
    87 |
    88 |
    89 |
    90 |
    91 | )} 92 |
    93 | ) 94 | ); 95 | }; 96 | 97 | export default Game; 98 | -------------------------------------------------------------------------------- /client/src/components/pages/Home.js: -------------------------------------------------------------------------------- 1 | import React, { useState, useContext, useEffect } from 'react'; 2 | import { Link } from 'react-router-dom'; 3 | import HistoryContext from '../../context/history/historyContext'; 4 | import AuthContext from '../../context/auth/authContext'; 5 | 6 | const Home = () => { 7 | // declare authContext, declare and destructure historyContext 8 | const historyContext = useContext(HistoryContext); 9 | const authContext = useContext(AuthContext); 10 | 11 | const { games, updateCurrentLevel, updateCurrentTheme, getGames } = 12 | historyContext; 13 | 14 | // declare component level state 15 | const [chosenTheme, setChosenTheme] = useState(''); 16 | 17 | // load user and get data from backend 18 | useEffect(() => { 19 | authContext.loadUser(); 20 | getGames(); 21 | // eslint-disable-next-line 22 | }, []); 23 | 24 | // enable materialize dropdown functionality 25 | useEffect(() => { 26 | window.M.AutoInit(); 27 | }); 28 | 29 | // calculate average moves 30 | const getAvg = (data) => { 31 | // filter games data based on difficulty level 32 | const gameArr = data[0] 33 | .filter((game) => game.gameLevel === data[1]) 34 | .map((game) => game.numOfMoves); 35 | 36 | // return avg moves or message 37 | if (gameArr.length > 0) { 38 | return Math.floor(gameArr.reduce((a, b) => a + b) / gameArr.length); 39 | } else { 40 | return 'No games played'; 41 | } 42 | }; 43 | 44 | // store current level 45 | const onClick = (e) => { 46 | updateCurrentLevel(e.target.name); 47 | }; 48 | 49 | // store current theme 50 | const handleClick = (e) => { 51 | updateCurrentTheme(e.target.name); 52 | setChosenTheme(e.target.name); 53 | }; 54 | 55 | return ( 56 | // materializeCSS card 57 | // prevent view if page still loading 58 | !authContext.loading && ( 59 |
    60 |
    61 |
    62 |
    63 | 64 | You have played {games.length}{' '} 65 | {games.length === 1 ? 'game' : 'games'} so far! 66 | 67 | 68 | {games.length} {games.length === 1 ? 'Game ' : 'Games '} 69 | Played! 70 | 71 |

    72 | 73 | Your Average Number of Moves: 74 | 75 | 76 | Average Moves: 77 | 78 |

    79 |
    80 |
    81 | Easy 82 |
    83 |
    84 | Medium 85 |
    86 |
    87 | Hard 88 |
    89 |
    90 |
    91 |
    92 |

    {getAvg([games, 'beginner'])}

    93 |
    94 |
    95 |

    {getAvg([games, 'intermediate'])}

    96 |
    97 |
    98 |

    {getAvg([games, 'expert'])}

    99 |
    100 |
    101 |

    102 |

    103 |

    104 |

    105 |

    106 |

    107 | Start A New Game Below! 108 |
    109 |
    110 | 115 | {chosenTheme.length > 0 ? chosenTheme : 'Choose A Theme!'} 116 | 117 | 151 |
    152 |
    153 |
    154 | 160 | Easy 161 | 162 |
    163 |
    164 | 170 | Medium 171 | Med 172 | 173 |
    174 |
    175 | 181 | Hard 182 | 183 |
    184 |
    185 |
    186 |
    187 |
    188 | ) 189 | ); 190 | }; 191 | 192 | export default Home; 193 | -------------------------------------------------------------------------------- /client/src/components/pages/Register.js: -------------------------------------------------------------------------------- 1 | import React, { useState, useContext, useEffect } from 'react'; 2 | import AuthContext from '../../context/auth/authContext'; 3 | 4 | // bring in props 5 | const Register = (props) => { 6 | // declare and destructure authContext 7 | const authContext = useContext(AuthContext); 8 | 9 | const { register, error, clearErrors, isAuthenticated } = authContext; 10 | 11 | // declare and destructure component level state 12 | const [user, setUser] = useState({ 13 | name: '', 14 | email: '', 15 | password: '', 16 | }); 17 | 18 | const { name, email, password } = user; 19 | 20 | // if user is signed in, go to homepage 21 | useEffect(() => { 22 | if (isAuthenticated) { 23 | props.history.push('/'); 24 | } 25 | 26 | // if errors, alert user and clear errors 27 | if (error === 'Email already exists') { 28 | alert(error); 29 | clearErrors(); 30 | } 31 | // eslint-disable-next-line 32 | }, [error, isAuthenticated, props.history]); 33 | 34 | // store user input on change 35 | const onChange = (e) => { 36 | setUser({ ...user, [e.target.id]: e.target.value }); 37 | }; 38 | 39 | // register user with input object on form submit 40 | const onSubmit = (e) => { 41 | e.preventDefault(); 42 | register({ 43 | name, 44 | email, 45 | password, 46 | }); 47 | }; 48 | 49 | return ( 50 | // materialize CSS form 51 |
    52 |
    53 |
    54 |
    55 |
    56 | 63 | 64 |
    65 |
    66 |
    67 |
    68 | 75 | 76 |
    77 |
    78 |
    79 |
    80 | 87 | 88 |
    89 |
    90 |
    91 | 99 |
    100 |
    101 |
    102 |
    103 | ); 104 | }; 105 | 106 | export default Register; 107 | -------------------------------------------------------------------------------- /client/src/components/pages/Signin.js: -------------------------------------------------------------------------------- 1 | import React, { useState, useContext, useEffect } from 'react'; 2 | import AuthContext from '../../context/auth/authContext'; 3 | 4 | // bring in props, declare authContext and destructure 5 | const Signin = (props) => { 6 | const authContext = useContext(AuthContext); 7 | 8 | const { login, error, clearErrors, isAuthenticated } = authContext; 9 | 10 | // declare state and desstructure 11 | const [user, setUser] = useState({ 12 | email: '', 13 | password: '', 14 | }); 15 | 16 | const { email, password } = user; 17 | 18 | useEffect(() => { 19 | // if user is signed in, go to homepage 20 | if (isAuthenticated) { 21 | props.history.push('/'); 22 | } 23 | 24 | // send err msg if user not recognized 25 | if (error === 'invalid credentials') { 26 | alert(error); 27 | clearErrors(); 28 | } 29 | // eslint-disable-next-line 30 | }, [error, isAuthenticated, props.history]); 31 | 32 | // store user input on change 33 | const onChange = (e) => { 34 | setUser({ ...user, [e.target.id]: e.target.value }); 35 | }; 36 | 37 | // call login with user info on form submit 38 | const onSubmit = (e) => { 39 | e.preventDefault(); 40 | login({ 41 | email, 42 | password, 43 | }); 44 | }; 45 | 46 | return ( 47 | // materializeCSS form 48 |
    49 |
    50 |
    51 |
    52 |
    53 | 60 | 61 |
    62 |
    63 |
    64 |
    65 | 72 | 73 |
    74 |
    75 |
    76 | 84 |
    85 |
    86 |
    87 |
    88 | ); 89 | }; 90 | 91 | export default Signin; 92 | -------------------------------------------------------------------------------- /client/src/components/routing/privateRoute.js: -------------------------------------------------------------------------------- 1 | import React, { useContext } from 'react'; 2 | import { Route, Redirect } from 'react-router-dom'; 3 | import AuthContext from '../../context/auth/authContext'; 4 | 5 | const PrivateRoute = ({ component: Component, ...rest }) => { 6 | // declare authcontext 7 | const authContext = useContext(AuthContext); 8 | 9 | // destructure variables 10 | const { isAuthenticated, loading } = authContext; 11 | 12 | // if user authenticated, allow access to private routes 13 | return ( 14 | 17 | !isAuthenticated && !loading ? ( 18 | 19 | ) : ( 20 | 21 | ) 22 | } 23 | /> 24 | ); 25 | }; 26 | 27 | export default PrivateRoute; 28 | -------------------------------------------------------------------------------- /client/src/context/auth/AuthState.js: -------------------------------------------------------------------------------- 1 | import React, { useReducer } from 'react'; 2 | import axios from 'axios'; 3 | import AuthContext from './authContext'; 4 | import authReducer from './authReducer'; 5 | import setAuthToken from '../../utils/setAuthToken'; 6 | import { 7 | REGISTER_FAIL, 8 | USER_LOADED, 9 | AUTH_ERROR, 10 | LOGIN_SUCCESS, 11 | LOGIN_FAIL, 12 | LOGOUT, 13 | CLEAR_ERRORS, 14 | REGISTER_SUCCESS, 15 | } from '../types'; 16 | 17 | const AuthState = (props) => { 18 | // declare inicial state 19 | const initialState = { 20 | token: localStorage.getItem('token'), 21 | isAuthenticated: null, 22 | loading: true, 23 | user: null, 24 | error: null, 25 | }; 26 | 27 | // declare state/dispatch with useReducer hook 28 | const [state, dispatch] = useReducer(authReducer, initialState); 29 | 30 | // Load User 31 | const loadUser = async () => { 32 | // check localstorage has a token, setAUthToken if so 33 | if (localStorage.token) { 34 | setAuthToken(localStorage.token); 35 | } 36 | 37 | // declare response from backend, dispatch data to USER_LOADED 38 | try { 39 | const res = await axios.get('/api/auth'); 40 | 41 | dispatch({ 42 | type: USER_LOADED, 43 | payload: res.data, 44 | }); 45 | // dispatch error msg to AUTH_ERROR 46 | } catch (error) { 47 | dispatch({ 48 | type: AUTH_ERROR, 49 | payload: error.response.data.msg, 50 | }); 51 | } 52 | }; 53 | 54 | // Register User 55 | const register = async (formData) => { 56 | // declare config variable with header 57 | const config = { 58 | header: { 59 | 'Content-Type': 'application/json', 60 | }, 61 | }; 62 | 63 | // declare response from backend, post user data 64 | try { 65 | const res = await axios.post('/api/users', formData, config); 66 | 67 | // dispatch data to reducer 68 | dispatch({ 69 | type: REGISTER_SUCCESS, 70 | payload: res.data, 71 | }); 72 | 73 | // load user 74 | loadUser(); 75 | // dispatch error if found 76 | } catch (error) { 77 | dispatch({ 78 | type: REGISTER_FAIL, 79 | payload: error.response.data.msg, 80 | }); 81 | } 82 | }; 83 | 84 | // Login User 85 | const login = async (formData) => { 86 | // declare config variable with header 87 | const config = { 88 | header: { 89 | 'Content-Type': 'application/json', 90 | }, 91 | }; 92 | 93 | // declare response from post, dispatch to reducer 94 | try { 95 | const res = await axios.post('/api/auth', formData, config); 96 | 97 | dispatch({ 98 | type: LOGIN_SUCCESS, 99 | payload: res.data, 100 | }); 101 | 102 | loadUser(); 103 | // dispatch error if found 104 | } catch (error) { 105 | dispatch({ 106 | type: LOGIN_FAIL, 107 | payload: error.response.data.msg, 108 | }); 109 | } 110 | }; 111 | 112 | // Logout 113 | const logout = () => { 114 | dispatch({ type: LOGOUT }); 115 | }; 116 | 117 | // Clear Errors 118 | const clearErrors = () => { 119 | dispatch({ type: CLEAR_ERRORS }); 120 | }; 121 | 122 | return ( 123 | // return all variables and functions to provider 124 | 138 | {props.children} 139 | 140 | ); 141 | }; 142 | 143 | export default AuthState; 144 | -------------------------------------------------------------------------------- /client/src/context/auth/authContext.js: -------------------------------------------------------------------------------- 1 | import { createContext } from 'react'; 2 | 3 | const authContext = createContext(); 4 | 5 | export default authContext; 6 | -------------------------------------------------------------------------------- /client/src/context/auth/authReducer.js: -------------------------------------------------------------------------------- 1 | import { 2 | REGISTER_FAIL, 3 | USER_LOADED, 4 | AUTH_ERROR, 5 | LOGIN_SUCCESS, 6 | LOGIN_FAIL, 7 | LOGOUT, 8 | CLEAR_ERRORS, 9 | REGISTER_SUCCESS, 10 | } from '../types'; 11 | 12 | // authReducer returns object state based on the type passed in using a switch 13 | const authReducer = (state, action) => { 14 | switch (action.type) { 15 | case USER_LOADED: 16 | return { 17 | ...state, 18 | isAuthenticated: true, 19 | loading: false, 20 | user: action.payload, 21 | }; 22 | case REGISTER_SUCCESS: 23 | case LOGIN_SUCCESS: 24 | localStorage.setItem('token', action.payload.token); 25 | return { 26 | ...state, 27 | ...action.payload, 28 | isAuthenticated: true, 29 | loading: false, 30 | }; 31 | case REGISTER_FAIL: 32 | case AUTH_ERROR: 33 | case LOGIN_FAIL: 34 | case LOGOUT: 35 | localStorage.removeItem('token'); 36 | return { 37 | ...state, 38 | token: null, 39 | isAuthenticated: false, 40 | loading: false, 41 | user: null, 42 | error: action.payload, 43 | }; 44 | case CLEAR_ERRORS: 45 | return { 46 | ...state, 47 | error: null, 48 | }; 49 | default: 50 | return state; 51 | } 52 | }; 53 | 54 | export default authReducer; 55 | -------------------------------------------------------------------------------- /client/src/context/history/HistoryState.js: -------------------------------------------------------------------------------- 1 | import React, { useReducer } from 'react'; 2 | import axios from 'axios'; 3 | import HistoryContext from './historyContext'; 4 | import historyReducer from './historyReducer'; 5 | import { 6 | ADD_NEW_GAME, 7 | UPDATE_CURRENT_LEVEL, 8 | UPDATE_CURRENT_THEME, 9 | GAME_ERROR, 10 | GET_GAMES, 11 | } from '../types'; 12 | 13 | const HistoryState = (props) => { 14 | // declare initial state 15 | const initialState = { 16 | games: [], 17 | currentLevel: 'beginner', 18 | currentTheme: 'robots', 19 | error: null, 20 | }; 21 | 22 | // declare state and dispatch with usereducer hook 23 | const [state, dispatch] = useReducer(historyReducer, initialState); 24 | 25 | // Get games for user 26 | const getGames = async () => { 27 | // declare response from backend, dispatch to reducer 28 | try { 29 | const res = await axios.get('/api/history'); 30 | 31 | dispatch({ 32 | type: GET_GAMES, 33 | payload: res.data, 34 | }); 35 | // dispatch errr msg if found 36 | } catch (error) { 37 | dispatch({ 38 | type: GAME_ERROR, 39 | payload: error.response.msg, 40 | }); 41 | } 42 | }; 43 | 44 | // Add game to history 45 | const addNewGame = async (newGame) => { 46 | const config = { 47 | header: { 48 | 'Content-type': 'application/json', 49 | }, 50 | }; 51 | 52 | try { 53 | const res = await axios.post('/api/history', newGame, config); 54 | 55 | dispatch({ 56 | type: ADD_NEW_GAME, 57 | payload: res.data, 58 | }); 59 | } catch (error) { 60 | dispatch({ 61 | type: GAME_ERROR, 62 | payload: error.response.msg, 63 | }); 64 | } 65 | }; 66 | 67 | // Delete all games from history 68 | 69 | // Update currentLevel 70 | const updateCurrentLevel = (level) => { 71 | dispatch({ type: UPDATE_CURRENT_LEVEL, payload: level }); 72 | }; 73 | 74 | // Update currentTheme 75 | const updateCurrentTheme = (theme) => { 76 | dispatch({ type: UPDATE_CURRENT_THEME, payload: theme }); 77 | }; 78 | 79 | return ( 80 | // return all variables and functions to provider 81 | 93 | {props.children} 94 | 95 | ); 96 | }; 97 | 98 | export default HistoryState; 99 | -------------------------------------------------------------------------------- /client/src/context/history/historyContext.js: -------------------------------------------------------------------------------- 1 | import { createContext } from 'react'; 2 | 3 | const historyContext = createContext(); 4 | 5 | export default historyContext; 6 | -------------------------------------------------------------------------------- /client/src/context/history/historyReducer.js: -------------------------------------------------------------------------------- 1 | import { 2 | UPDATE_CURRENT_LEVEL, 3 | UPDATE_CURRENT_THEME, 4 | ADD_NEW_GAME, 5 | GAME_ERROR, 6 | GET_GAMES, 7 | } from '../types'; 8 | 9 | // historyReducer returns object state based on the type passed in using a switch 10 | const historyReducer = (state, action) => { 11 | switch (action.type) { 12 | case GET_GAMES: 13 | return { 14 | ...state, 15 | games: action.payload, 16 | }; 17 | case UPDATE_CURRENT_LEVEL: 18 | return { 19 | ...state, 20 | currentLevel: action.payload, 21 | }; 22 | case UPDATE_CURRENT_THEME: 23 | return { 24 | ...state, 25 | currentTheme: action.payload, 26 | }; 27 | case ADD_NEW_GAME: 28 | return { 29 | ...state, 30 | games: [...state.games, action.payload], 31 | }; 32 | case GAME_ERROR: 33 | return { 34 | ...state, 35 | error: action.payload, 36 | }; 37 | default: 38 | return state; 39 | } 40 | }; 41 | 42 | export default historyReducer; 43 | -------------------------------------------------------------------------------- /client/src/context/types.js: -------------------------------------------------------------------------------- 1 | export const UPDATE_CURRENT_LEVEL = 'UPDATE_CURRENT_LEVEL'; 2 | export const UPDATE_CURRENT_THEME = 'UPDATE_CURRENT_THEME'; 3 | export const ADD_NEW_GAME = 'ADD_NEW_GAME'; 4 | export const GAME_ERROR = 'GAME_ERROR'; 5 | export const REGISTER_SUCCESS = 'REGISTER_SUCCESS'; 6 | export const REGISTER_FAIL = 'REGISTER_FAIL'; 7 | export const USER_LOADED = 'USER_LOADED'; 8 | export const AUTH_ERROR = 'AUTH_ERROR'; 9 | export const LOGIN_SUCCESS = 'LOGIN_SUCCESS'; 10 | export const LOGIN_FAIL = 'LOGIN_FAIL'; 11 | export const LOGOUT = 'LOGOUT'; 12 | export const CLEAR_ERRORS = 'CLEAR_ERRORS'; 13 | export const GET_GAMES = 'GET_GAMES'; 14 | -------------------------------------------------------------------------------- /client/src/index.js: -------------------------------------------------------------------------------- 1 | import React from 'react'; 2 | import ReactDOM from 'react-dom'; 3 | import App from './App'; 4 | 5 | ReactDOM.render( 6 | 7 | 8 | , 9 | document.getElementById('root') 10 | ); 11 | -------------------------------------------------------------------------------- /client/src/utils/setAuthToken.js: -------------------------------------------------------------------------------- 1 | import axios from 'axios'; 2 | 3 | // set token for private routes 4 | const setAuthToken = (token) => { 5 | if (token) { 6 | axios.defaults.headers.common['x-auth-token'] = token; 7 | } else { 8 | delete axios.defaults.headers.common['x-auth-token']; 9 | } 10 | }; 11 | 12 | export default setAuthToken; 13 | -------------------------------------------------------------------------------- /config/db.js: -------------------------------------------------------------------------------- 1 | const mongoose = require('mongoose'); 2 | const config = require('config'); 3 | const db = config.get('mongoURI'); 4 | 5 | const connectDB = async () => { 6 | // connect to database 7 | try { 8 | await mongoose.connect(db, { 9 | useNewUrlParser: true, 10 | useCreateIndex: true, 11 | useFindAndModify: false, 12 | useUnifiedTopology: true, 13 | }); 14 | 15 | // alert user connection successful 16 | console.log('db connected'); 17 | // return err msg if fail 18 | } catch (err) { 19 | console.error(err.message); 20 | process.exit(1); 21 | } 22 | }; 23 | 24 | module.exports = connectDB; 25 | -------------------------------------------------------------------------------- /config/default.json: -------------------------------------------------------------------------------- 1 | { 2 | "mongoURI": "mongodb+srv://jesse:jesse@cluster0.ll8sh.mongodb.net/?retryWrites=true&w=majority", 3 | "jwtSecret": "secret" 4 | } -------------------------------------------------------------------------------- /config/production.json: -------------------------------------------------------------------------------- 1 | { 2 | "mongoURI": "mongodb+srv://jesse:jesse@cluster0.ll8sh.mongodb.net/?retryWrites=true&w=majority", 3 | "jwtSecret": "secret" 4 | } -------------------------------------------------------------------------------- /middleware/auth.js: -------------------------------------------------------------------------------- 1 | const jwt = require('jsonwebtoken'); 2 | const config = require('config'); 3 | 4 | module.exports = function (req, res, next) { 5 | // Get token from header 6 | const token = req.header('x-auth-token'); 7 | 8 | // Check if not token, return err msg if not token 9 | if (!token) { 10 | return res.status(401).json({ msg: 'No token, authorization denied' }); 11 | } 12 | 13 | // get secret from config 14 | try { 15 | const decoded = jwt.verify(token, config.get('jwtSecret')); 16 | 17 | // store in user variable 18 | req.user = decoded.user; 19 | next(); 20 | // return err msg 21 | } catch (err) { 22 | res.status(401).json({ msg: 'Token not valid' }); 23 | } 24 | }; 25 | -------------------------------------------------------------------------------- /models/Games.js: -------------------------------------------------------------------------------- 1 | const mongoose = require('mongoose'); 2 | 3 | // Game schema for mongoDB Games database 4 | const GameSchema = mongoose.Schema({ 5 | user: { 6 | type: mongoose.Schema.Types.ObjectId, 7 | ref: 'users', 8 | }, 9 | gameLevel: { 10 | type: String, 11 | }, 12 | numOfMoves: { 13 | type: Number, 14 | }, 15 | date: { 16 | type: Date, 17 | default: Date.now(), 18 | }, 19 | }); 20 | 21 | module.exports = mongoose.model('games', GameSchema); 22 | -------------------------------------------------------------------------------- /models/Users.js: -------------------------------------------------------------------------------- 1 | const mongoose = require('mongoose'); 2 | 3 | // User schema for mongoDB Users database 4 | const UserSchema = mongoose.Schema({ 5 | name: { 6 | type: String, 7 | required: true, 8 | }, 9 | email: { 10 | type: String, 11 | required: true, 12 | unique: true, 13 | }, 14 | password: { 15 | type: String, 16 | required: true, 17 | }, 18 | date: { 19 | type: Date, 20 | default: Date.now, 21 | }, 22 | }); 23 | 24 | module.exports = mongoose.model('users', UserSchema); 25 | -------------------------------------------------------------------------------- /package-lock.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "memory-game", 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 | "abbrev": { 23 | "version": "1.1.1", 24 | "resolved": "https://registry.npmjs.org/abbrev/-/abbrev-1.1.1.tgz", 25 | "integrity": "sha512-nne9/IiQ/hzIhY6pdDnbBtz7DjPTKrY00P/zvPSm5pOFkl6xuGrGnXn/VtTNNfNtAfZ9/1RtehkszU9qcTii0Q==", 26 | "dev": true 27 | }, 28 | "accepts": { 29 | "version": "1.3.7", 30 | "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.7.tgz", 31 | "integrity": "sha512-Il80Qs2WjYlJIBNzNkK6KYqlVMTbZLXgHx2oT0pU/fjRHyEp+PEfEPY0R3WCwAGVOtauxh1hOxNgIf5bv7dQpA==", 32 | "requires": { 33 | "mime-types": "~2.1.24", 34 | "negotiator": "0.6.2" 35 | } 36 | }, 37 | "ansi-align": { 38 | "version": "3.0.0", 39 | "resolved": "https://registry.npmjs.org/ansi-align/-/ansi-align-3.0.0.tgz", 40 | "integrity": "sha512-ZpClVKqXN3RGBmKibdfWzqCY4lnjEuoNzU5T0oEFpfd/z5qJHVarukridD4juLO2FXMiwUQxr9WqQtaYa8XRYw==", 41 | "dev": true, 42 | "requires": { 43 | "string-width": "^3.0.0" 44 | } 45 | }, 46 | "ansi-regex": { 47 | "version": "4.1.0", 48 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz", 49 | "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==", 50 | "dev": true 51 | }, 52 | "ansi-styles": { 53 | "version": "3.2.1", 54 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", 55 | "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", 56 | "dev": true, 57 | "requires": { 58 | "color-convert": "^1.9.0" 59 | } 60 | }, 61 | "anymatch": { 62 | "version": "3.1.1", 63 | "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.1.tgz", 64 | "integrity": "sha512-mM8522psRCqzV+6LhomX5wgp25YVibjh8Wj23I5RPkPppSVSjyKD2A2mBJmWGa+KN7f2D6LNh9jkBCeyLktzjg==", 65 | "dev": true, 66 | "requires": { 67 | "normalize-path": "^3.0.0", 68 | "picomatch": "^2.0.4" 69 | } 70 | }, 71 | "array-flatten": { 72 | "version": "1.1.1", 73 | "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz", 74 | "integrity": "sha1-ml9pkFGx5wczKPKgCJaLZOopVdI=" 75 | }, 76 | "balanced-match": { 77 | "version": "1.0.0", 78 | "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.0.tgz", 79 | "integrity": "sha1-ibTRmasr7kneFk6gK4nORi1xt2c=", 80 | "dev": true 81 | }, 82 | "bcryptjs": { 83 | "version": "2.4.3", 84 | "resolved": "https://registry.npmjs.org/bcryptjs/-/bcryptjs-2.4.3.tgz", 85 | "integrity": "sha1-mrVie5PmBiH/fNrF2pczAn3x0Ms=" 86 | }, 87 | "binary-extensions": { 88 | "version": "2.1.0", 89 | "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.1.0.tgz", 90 | "integrity": "sha512-1Yj8h9Q+QDF5FzhMs/c9+6UntbD5MkRfRwac8DoEm9ZfUBZ7tZ55YcGVAzEe4bXsdQHEk+s9S5wsOKVdZrw0tQ==", 91 | "dev": true 92 | }, 93 | "bl": { 94 | "version": "2.2.1", 95 | "resolved": "https://registry.npmjs.org/bl/-/bl-2.2.1.tgz", 96 | "integrity": "sha512-6Pesp1w0DEX1N550i/uGV/TqucVL4AM/pgThFSN/Qq9si1/DF9aIHs1BxD8V/QU0HoeHO6cQRTAuYnLPKq1e4g==", 97 | "requires": { 98 | "readable-stream": "^2.3.5", 99 | "safe-buffer": "^5.1.1" 100 | } 101 | }, 102 | "bluebird": { 103 | "version": "3.5.1", 104 | "resolved": "https://registry.npmjs.org/bluebird/-/bluebird-3.5.1.tgz", 105 | "integrity": "sha512-MKiLiV+I1AA596t9w1sQJ8jkiSr5+ZKi0WKrYGUn6d1Fx+Ij4tIj+m2WMQSGczs5jZVxV339chE8iwk6F64wjA==" 106 | }, 107 | "body-parser": { 108 | "version": "1.19.0", 109 | "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.19.0.tgz", 110 | "integrity": "sha512-dhEPs72UPbDnAQJ9ZKMNTP6ptJaionhP5cBb541nXPlW60Jepo9RV/a4fX4XWW9CuFNK22krhrj1+rgzifNCsw==", 111 | "requires": { 112 | "bytes": "3.1.0", 113 | "content-type": "~1.0.4", 114 | "debug": "2.6.9", 115 | "depd": "~1.1.2", 116 | "http-errors": "1.7.2", 117 | "iconv-lite": "0.4.24", 118 | "on-finished": "~2.3.0", 119 | "qs": "6.7.0", 120 | "raw-body": "2.4.0", 121 | "type-is": "~1.6.17" 122 | } 123 | }, 124 | "boxen": { 125 | "version": "4.2.0", 126 | "resolved": "https://registry.npmjs.org/boxen/-/boxen-4.2.0.tgz", 127 | "integrity": "sha512-eB4uT9RGzg2odpER62bBwSLvUeGC+WbRjjyyFhGsKnc8wp/m0+hQsMUvUe3H2V0D5vw0nBdO1hCJoZo5mKeuIQ==", 128 | "dev": true, 129 | "requires": { 130 | "ansi-align": "^3.0.0", 131 | "camelcase": "^5.3.1", 132 | "chalk": "^3.0.0", 133 | "cli-boxes": "^2.2.0", 134 | "string-width": "^4.1.0", 135 | "term-size": "^2.1.0", 136 | "type-fest": "^0.8.1", 137 | "widest-line": "^3.1.0" 138 | }, 139 | "dependencies": { 140 | "ansi-regex": { 141 | "version": "5.0.0", 142 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.0.tgz", 143 | "integrity": "sha512-bY6fj56OUQ0hU1KjFNDQuJFezqKdrAyFdIevADiqrWHwSlbmBNMHp5ak2f40Pm8JTFyM2mqxkG6ngkHO11f/lg==", 144 | "dev": true 145 | }, 146 | "ansi-styles": { 147 | "version": "4.3.0", 148 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", 149 | "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", 150 | "dev": true, 151 | "requires": { 152 | "color-convert": "^2.0.1" 153 | } 154 | }, 155 | "chalk": { 156 | "version": "3.0.0", 157 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-3.0.0.tgz", 158 | "integrity": "sha512-4D3B6Wf41KOYRFdszmDqMCGq5VV/uMAB273JILmO+3jAlh8X4qDtdtgCR3fxtbLEMzSx22QdhnDcJvu2u1fVwg==", 159 | "dev": true, 160 | "requires": { 161 | "ansi-styles": "^4.1.0", 162 | "supports-color": "^7.1.0" 163 | } 164 | }, 165 | "color-convert": { 166 | "version": "2.0.1", 167 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", 168 | "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", 169 | "dev": true, 170 | "requires": { 171 | "color-name": "~1.1.4" 172 | } 173 | }, 174 | "color-name": { 175 | "version": "1.1.4", 176 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", 177 | "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", 178 | "dev": true 179 | }, 180 | "emoji-regex": { 181 | "version": "8.0.0", 182 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", 183 | "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", 184 | "dev": true 185 | }, 186 | "has-flag": { 187 | "version": "4.0.0", 188 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", 189 | "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", 190 | "dev": true 191 | }, 192 | "is-fullwidth-code-point": { 193 | "version": "3.0.0", 194 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", 195 | "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", 196 | "dev": true 197 | }, 198 | "string-width": { 199 | "version": "4.2.0", 200 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.0.tgz", 201 | "integrity": "sha512-zUz5JD+tgqtuDjMhwIg5uFVV3dtqZ9yQJlZVfq4I01/K5Paj5UHj7VyrQOJvzawSVlKpObApbfD0Ed6yJc+1eg==", 202 | "dev": true, 203 | "requires": { 204 | "emoji-regex": "^8.0.0", 205 | "is-fullwidth-code-point": "^3.0.0", 206 | "strip-ansi": "^6.0.0" 207 | } 208 | }, 209 | "strip-ansi": { 210 | "version": "6.0.0", 211 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.0.tgz", 212 | "integrity": "sha512-AuvKTrTfQNYNIctbR1K/YGTR1756GycPsg7b9bdV9Duqur4gv6aKqHXah67Z8ImS7WEz5QVcOtlfW2rZEugt6w==", 213 | "dev": true, 214 | "requires": { 215 | "ansi-regex": "^5.0.0" 216 | } 217 | }, 218 | "supports-color": { 219 | "version": "7.2.0", 220 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", 221 | "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", 222 | "dev": true, 223 | "requires": { 224 | "has-flag": "^4.0.0" 225 | } 226 | } 227 | } 228 | }, 229 | "brace-expansion": { 230 | "version": "1.1.11", 231 | "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", 232 | "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", 233 | "dev": true, 234 | "requires": { 235 | "balanced-match": "^1.0.0", 236 | "concat-map": "0.0.1" 237 | } 238 | }, 239 | "braces": { 240 | "version": "3.0.2", 241 | "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz", 242 | "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", 243 | "dev": true, 244 | "requires": { 245 | "fill-range": "^7.0.1" 246 | } 247 | }, 248 | "bson": { 249 | "version": "1.1.5", 250 | "resolved": "https://registry.npmjs.org/bson/-/bson-1.1.5.tgz", 251 | "integrity": "sha512-kDuEzldR21lHciPQAIulLs1LZlCXdLziXI6Mb/TDkwXhb//UORJNPXgcRs2CuO4H0DcMkpfT3/ySsP3unoZjBg==" 252 | }, 253 | "buffer-equal-constant-time": { 254 | "version": "1.0.1", 255 | "resolved": "https://registry.npmjs.org/buffer-equal-constant-time/-/buffer-equal-constant-time-1.0.1.tgz", 256 | "integrity": "sha1-+OcRMvf/5uAaXJaXpMbz5I1cyBk=" 257 | }, 258 | "bytes": { 259 | "version": "3.1.0", 260 | "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.0.tgz", 261 | "integrity": "sha512-zauLjrfCG+xvoyaqLoV8bLVXXNGC4JqlxFCutSDWA6fJrTo2ZuvLYTqZ7aHBLZSMOopbzwv8f+wZcVzfVTI2Dg==" 262 | }, 263 | "cacheable-request": { 264 | "version": "6.1.0", 265 | "resolved": "https://registry.npmjs.org/cacheable-request/-/cacheable-request-6.1.0.tgz", 266 | "integrity": "sha512-Oj3cAGPCqOZX7Rz64Uny2GYAZNliQSqfbePrgAQ1wKAihYmCUnraBtJtKcGR4xz7wF+LoJC+ssFZvv5BgF9Igg==", 267 | "dev": true, 268 | "requires": { 269 | "clone-response": "^1.0.2", 270 | "get-stream": "^5.1.0", 271 | "http-cache-semantics": "^4.0.0", 272 | "keyv": "^3.0.0", 273 | "lowercase-keys": "^2.0.0", 274 | "normalize-url": "^4.1.0", 275 | "responselike": "^1.0.2" 276 | }, 277 | "dependencies": { 278 | "get-stream": { 279 | "version": "5.2.0", 280 | "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-5.2.0.tgz", 281 | "integrity": "sha512-nBF+F1rAZVCu/p7rjzgA+Yb4lfYXrpl7a6VmJrU8wF9I1CKvP/QwPNZHnOlwbTkY6dvtFIzFMSyQXbLoTQPRpA==", 282 | "dev": true, 283 | "requires": { 284 | "pump": "^3.0.0" 285 | } 286 | }, 287 | "lowercase-keys": { 288 | "version": "2.0.0", 289 | "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-2.0.0.tgz", 290 | "integrity": "sha512-tqNXrS78oMOE73NMxK4EMLQsQowWf8jKooH9g7xPavRT706R6bkQJ6DY2Te7QukaZsulxa30wQ7bk0pm4XiHmA==", 291 | "dev": true 292 | } 293 | } 294 | }, 295 | "camelcase": { 296 | "version": "5.3.1", 297 | "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz", 298 | "integrity": "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==", 299 | "dev": true 300 | }, 301 | "chalk": { 302 | "version": "2.4.2", 303 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz", 304 | "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==", 305 | "dev": true, 306 | "requires": { 307 | "ansi-styles": "^3.2.1", 308 | "escape-string-regexp": "^1.0.5", 309 | "supports-color": "^5.3.0" 310 | }, 311 | "dependencies": { 312 | "supports-color": { 313 | "version": "5.5.0", 314 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", 315 | "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", 316 | "dev": true, 317 | "requires": { 318 | "has-flag": "^3.0.0" 319 | } 320 | } 321 | } 322 | }, 323 | "chokidar": { 324 | "version": "3.4.3", 325 | "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.4.3.tgz", 326 | "integrity": "sha512-DtM3g7juCXQxFVSNPNByEC2+NImtBuxQQvWlHunpJIS5Ocr0lG306cC7FCi7cEA0fzmybPUIl4txBIobk1gGOQ==", 327 | "dev": true, 328 | "requires": { 329 | "anymatch": "~3.1.1", 330 | "braces": "~3.0.2", 331 | "fsevents": "~2.1.2", 332 | "glob-parent": "~5.1.0", 333 | "is-binary-path": "~2.1.0", 334 | "is-glob": "~4.0.1", 335 | "normalize-path": "~3.0.0", 336 | "readdirp": "~3.5.0" 337 | } 338 | }, 339 | "ci-info": { 340 | "version": "2.0.0", 341 | "resolved": "https://registry.npmjs.org/ci-info/-/ci-info-2.0.0.tgz", 342 | "integrity": "sha512-5tK7EtrZ0N+OLFMthtqOj4fI2Jeb88C4CAZPu25LDVUgXJ0A3Js4PMGqrn0JU1W0Mh1/Z8wZzYPxqUrXeBboCQ==", 343 | "dev": true 344 | }, 345 | "cli-boxes": { 346 | "version": "2.2.1", 347 | "resolved": "https://registry.npmjs.org/cli-boxes/-/cli-boxes-2.2.1.tgz", 348 | "integrity": "sha512-y4coMcylgSCdVinjiDBuR8PCC2bLjyGTwEmPb9NHR/QaNU6EUOXcTY/s6VjGMD6ENSEaeQYHCY0GNGS5jfMwPw==", 349 | "dev": true 350 | }, 351 | "cliui": { 352 | "version": "5.0.0", 353 | "resolved": "https://registry.npmjs.org/cliui/-/cliui-5.0.0.tgz", 354 | "integrity": "sha512-PYeGSEmmHM6zvoef2w8TPzlrnNpXIjTipYK780YswmIP9vjxmd6Y2a3CB2Ks6/AU8NHjZugXvo8w3oWM2qnwXA==", 355 | "dev": true, 356 | "requires": { 357 | "string-width": "^3.1.0", 358 | "strip-ansi": "^5.2.0", 359 | "wrap-ansi": "^5.1.0" 360 | } 361 | }, 362 | "clone-response": { 363 | "version": "1.0.2", 364 | "resolved": "https://registry.npmjs.org/clone-response/-/clone-response-1.0.2.tgz", 365 | "integrity": "sha1-0dyXOSAxTfZ/vrlCI7TuNQI56Ws=", 366 | "dev": true, 367 | "requires": { 368 | "mimic-response": "^1.0.0" 369 | } 370 | }, 371 | "color-convert": { 372 | "version": "1.9.3", 373 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", 374 | "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==", 375 | "dev": true, 376 | "requires": { 377 | "color-name": "1.1.3" 378 | } 379 | }, 380 | "color-name": { 381 | "version": "1.1.3", 382 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", 383 | "integrity": "sha1-p9BVi9icQveV3UIyj3QIMcpTvCU=", 384 | "dev": true 385 | }, 386 | "concat-map": { 387 | "version": "0.0.1", 388 | "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", 389 | "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=", 390 | "dev": true 391 | }, 392 | "concurrently": { 393 | "version": "5.3.0", 394 | "resolved": "https://registry.npmjs.org/concurrently/-/concurrently-5.3.0.tgz", 395 | "integrity": "sha512-8MhqOB6PWlBfA2vJ8a0bSFKATOdWlHiQlk11IfmQBPaHVP8oP2gsh2MObE6UR3hqDHqvaIvLTyceNW6obVuFHQ==", 396 | "dev": true, 397 | "requires": { 398 | "chalk": "^2.4.2", 399 | "date-fns": "^2.0.1", 400 | "lodash": "^4.17.15", 401 | "read-pkg": "^4.0.1", 402 | "rxjs": "^6.5.2", 403 | "spawn-command": "^0.0.2-1", 404 | "supports-color": "^6.1.0", 405 | "tree-kill": "^1.2.2", 406 | "yargs": "^13.3.0" 407 | } 408 | }, 409 | "config": { 410 | "version": "3.3.2", 411 | "resolved": "https://registry.npmjs.org/config/-/config-3.3.2.tgz", 412 | "integrity": "sha512-NlGfBn2565YA44Irn7GV5KHlIGC3KJbf0062/zW5ddP9VXIuRj0m7HVyFAWvMZvaHPEglyGfwmevGz3KosIpCg==", 413 | "requires": { 414 | "json5": "^2.1.1" 415 | } 416 | }, 417 | "configstore": { 418 | "version": "5.0.1", 419 | "resolved": "https://registry.npmjs.org/configstore/-/configstore-5.0.1.tgz", 420 | "integrity": "sha512-aMKprgk5YhBNyH25hj8wGt2+D52Sw1DRRIzqBwLp2Ya9mFmY8KPvvtvmna8SxVR9JMZ4kzMD68N22vlaRpkeFA==", 421 | "dev": true, 422 | "requires": { 423 | "dot-prop": "^5.2.0", 424 | "graceful-fs": "^4.1.2", 425 | "make-dir": "^3.0.0", 426 | "unique-string": "^2.0.0", 427 | "write-file-atomic": "^3.0.0", 428 | "xdg-basedir": "^4.0.0" 429 | } 430 | }, 431 | "content-disposition": { 432 | "version": "0.5.3", 433 | "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.3.tgz", 434 | "integrity": "sha512-ExO0774ikEObIAEV9kDo50o+79VCUdEB6n6lzKgGwupcVeRlhrj3qGAfwq8G6uBJjkqLrhT0qEYFcWng8z1z0g==", 435 | "requires": { 436 | "safe-buffer": "5.1.2" 437 | } 438 | }, 439 | "content-type": { 440 | "version": "1.0.4", 441 | "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.4.tgz", 442 | "integrity": "sha512-hIP3EEPs8tB9AT1L+NUqtwOAps4mk2Zob89MWXMHjHWg9milF/j4osnnQLXBCBFBk/tvIG/tUc9mOUJiPBhPXA==" 443 | }, 444 | "cookie": { 445 | "version": "0.4.0", 446 | "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.4.0.tgz", 447 | "integrity": "sha512-+Hp8fLp57wnUSt0tY0tHEXh4voZRDnoIrZPqlo3DPiI4y9lwg/jqx+1Om94/W6ZaPDOUbnjOt/99w66zk+l1Xg==" 448 | }, 449 | "cookie-signature": { 450 | "version": "1.0.6", 451 | "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz", 452 | "integrity": "sha1-4wOogrNCzD7oylE6eZmXNNqzriw=" 453 | }, 454 | "core-util-is": { 455 | "version": "1.0.2", 456 | "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.2.tgz", 457 | "integrity": "sha1-tf1UIgqivFq1eqtxQMlAdUUDwac=" 458 | }, 459 | "crypto-random-string": { 460 | "version": "2.0.0", 461 | "resolved": "https://registry.npmjs.org/crypto-random-string/-/crypto-random-string-2.0.0.tgz", 462 | "integrity": "sha512-v1plID3y9r/lPhviJ1wrXpLeyUIGAZ2SHNYTEapm7/8A9nLPoyvVp3RK/EPFqn5kEznyWgYZNsRtYYIWbuG8KA==", 463 | "dev": true 464 | }, 465 | "date-fns": { 466 | "version": "2.16.1", 467 | "resolved": "https://registry.npmjs.org/date-fns/-/date-fns-2.16.1.tgz", 468 | "integrity": "sha512-sAJVKx/FqrLYHAQeN7VpJrPhagZc9R4ImZIWYRFZaaohR3KzmuK88touwsSwSVT8Qcbd4zoDsnGfX4GFB4imyQ==", 469 | "dev": true 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 | "decamelize": { 480 | "version": "1.2.0", 481 | "resolved": "https://registry.npmjs.org/decamelize/-/decamelize-1.2.0.tgz", 482 | "integrity": "sha1-9lNNFRSCabIDUue+4m9QH5oZEpA=", 483 | "dev": true 484 | }, 485 | "decompress-response": { 486 | "version": "3.3.0", 487 | "resolved": "https://registry.npmjs.org/decompress-response/-/decompress-response-3.3.0.tgz", 488 | "integrity": "sha1-gKTdMjdIOEv6JICDYirt7Jgq3/M=", 489 | "dev": true, 490 | "requires": { 491 | "mimic-response": "^1.0.0" 492 | } 493 | }, 494 | "deep-extend": { 495 | "version": "0.6.0", 496 | "resolved": "https://registry.npmjs.org/deep-extend/-/deep-extend-0.6.0.tgz", 497 | "integrity": "sha512-LOHxIOaPYdHlJRtCQfDIVZtfw/ufM8+rVj649RIHzcm/vGwQRXFt6OPqIFWsm2XEMrNIEtWR64sY1LEKD2vAOA==", 498 | "dev": true 499 | }, 500 | "defer-to-connect": { 501 | "version": "1.1.3", 502 | "resolved": "https://registry.npmjs.org/defer-to-connect/-/defer-to-connect-1.1.3.tgz", 503 | "integrity": "sha512-0ISdNousHvZT2EiFlZeZAHBUvSxmKswVCEf8hW7KWgG4a8MVEu/3Vb6uWYozkjylyCxe0JBIiRB1jV45S70WVQ==", 504 | "dev": true 505 | }, 506 | "denque": { 507 | "version": "1.4.1", 508 | "resolved": "https://registry.npmjs.org/denque/-/denque-1.4.1.tgz", 509 | "integrity": "sha512-OfzPuSZKGcgr96rf1oODnfjqBFmr1DVoc/TrItj3Ohe0Ah1C5WX5Baquw/9U9KovnQ88EqmJbD66rKYUQYN1tQ==" 510 | }, 511 | "depd": { 512 | "version": "1.1.2", 513 | "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.2.tgz", 514 | "integrity": "sha1-m81S4UwJd2PnSbJ0xDRu0uVgtak=" 515 | }, 516 | "destroy": { 517 | "version": "1.0.4", 518 | "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.0.4.tgz", 519 | "integrity": "sha1-l4hXRCxEdJ5CBmE+N5RiBYJqvYA=" 520 | }, 521 | "dot-prop": { 522 | "version": "5.3.0", 523 | "resolved": "https://registry.npmjs.org/dot-prop/-/dot-prop-5.3.0.tgz", 524 | "integrity": "sha512-QM8q3zDe58hqUqjraQOmzZ1LIH9SWQJTlEKCH4kJ2oQvLZk7RbQXvtDM2XEq3fwkV9CCvvH4LA0AV+ogFsBM2Q==", 525 | "dev": true, 526 | "requires": { 527 | "is-obj": "^2.0.0" 528 | } 529 | }, 530 | "duplexer3": { 531 | "version": "0.1.4", 532 | "resolved": "https://registry.npmjs.org/duplexer3/-/duplexer3-0.1.4.tgz", 533 | "integrity": "sha1-7gHdHKwO08vH/b6jfcCo8c4ALOI=", 534 | "dev": true 535 | }, 536 | "ecdsa-sig-formatter": { 537 | "version": "1.0.11", 538 | "resolved": "https://registry.npmjs.org/ecdsa-sig-formatter/-/ecdsa-sig-formatter-1.0.11.tgz", 539 | "integrity": "sha512-nagl3RYrbNv6kQkeJIpt6NJZy8twLB/2vtz6yN9Z4vRKHN4/QZJIEbqohALSgwKdnksuY3k5Addp5lg8sVoVcQ==", 540 | "requires": { 541 | "safe-buffer": "^5.0.1" 542 | } 543 | }, 544 | "ee-first": { 545 | "version": "1.1.1", 546 | "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", 547 | "integrity": "sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0=" 548 | }, 549 | "emoji-regex": { 550 | "version": "7.0.3", 551 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-7.0.3.tgz", 552 | "integrity": "sha512-CwBLREIQ7LvYFB0WyRvwhq5N5qPhc6PMjD6bYggFlI5YyDgl+0vxq5VHbMOFqLg7hfWzmu8T5Z1QofhmTIhItA==", 553 | "dev": true 554 | }, 555 | "encodeurl": { 556 | "version": "1.0.2", 557 | "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz", 558 | "integrity": "sha1-rT/0yG7C0CkyL1oCw6mmBslbP1k=" 559 | }, 560 | "end-of-stream": { 561 | "version": "1.4.4", 562 | "resolved": "https://registry.npmjs.org/end-of-stream/-/end-of-stream-1.4.4.tgz", 563 | "integrity": "sha512-+uw1inIHVPQoaVuHzRyXd21icM+cnt4CzD5rW+NC1wjOUSTOs+Te7FOv7AhN7vS9x/oIyhLP5PR1H+phQAHu5Q==", 564 | "dev": true, 565 | "requires": { 566 | "once": "^1.4.0" 567 | } 568 | }, 569 | "error-ex": { 570 | "version": "1.3.2", 571 | "resolved": "https://registry.npmjs.org/error-ex/-/error-ex-1.3.2.tgz", 572 | "integrity": "sha512-7dFHNmqeFSEt2ZBsCriorKnn3Z2pj+fd9kmI6QoWw4//DL+icEBfc0U7qJCisqrTsKTjw4fNFy2pW9OqStD84g==", 573 | "dev": true, 574 | "requires": { 575 | "is-arrayish": "^0.2.1" 576 | } 577 | }, 578 | "escape-goat": { 579 | "version": "2.1.1", 580 | "resolved": "https://registry.npmjs.org/escape-goat/-/escape-goat-2.1.1.tgz", 581 | "integrity": "sha512-8/uIhbG12Csjy2JEW7D9pHbreaVaS/OpN3ycnyvElTdwM5n6GY6W6e2IPemfvGZeUMqZ9A/3GqIZMgKnBhAw/Q==", 582 | "dev": true 583 | }, 584 | "escape-html": { 585 | "version": "1.0.3", 586 | "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz", 587 | "integrity": "sha1-Aljq5NPQwJdN4cFpGI7wBR0dGYg=" 588 | }, 589 | "escape-string-regexp": { 590 | "version": "1.0.5", 591 | "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", 592 | "integrity": "sha1-G2HAViGQqN/2rjuyzwIAyhMLhtQ=", 593 | "dev": true 594 | }, 595 | "etag": { 596 | "version": "1.8.1", 597 | "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz", 598 | "integrity": "sha1-Qa4u62XvpiJorr/qg6x9eSmbCIc=" 599 | }, 600 | "express": { 601 | "version": "4.17.1", 602 | "resolved": "https://registry.npmjs.org/express/-/express-4.17.1.tgz", 603 | "integrity": "sha512-mHJ9O79RqluphRrcw2X/GTh3k9tVv8YcoyY4Kkh4WDMUYKRZUq0h1o0w2rrrxBqM7VoeUVqgb27xlEMXTnYt4g==", 604 | "requires": { 605 | "accepts": "~1.3.7", 606 | "array-flatten": "1.1.1", 607 | "body-parser": "1.19.0", 608 | "content-disposition": "0.5.3", 609 | "content-type": "~1.0.4", 610 | "cookie": "0.4.0", 611 | "cookie-signature": "1.0.6", 612 | "debug": "2.6.9", 613 | "depd": "~1.1.2", 614 | "encodeurl": "~1.0.2", 615 | "escape-html": "~1.0.3", 616 | "etag": "~1.8.1", 617 | "finalhandler": "~1.1.2", 618 | "fresh": "0.5.2", 619 | "merge-descriptors": "1.0.1", 620 | "methods": "~1.1.2", 621 | "on-finished": "~2.3.0", 622 | "parseurl": "~1.3.3", 623 | "path-to-regexp": "0.1.7", 624 | "proxy-addr": "~2.0.5", 625 | "qs": "6.7.0", 626 | "range-parser": "~1.2.1", 627 | "safe-buffer": "5.1.2", 628 | "send": "0.17.1", 629 | "serve-static": "1.14.1", 630 | "setprototypeof": "1.1.1", 631 | "statuses": "~1.5.0", 632 | "type-is": "~1.6.18", 633 | "utils-merge": "1.0.1", 634 | "vary": "~1.1.2" 635 | } 636 | }, 637 | "express-validator": { 638 | "version": "6.6.1", 639 | "resolved": "https://registry.npmjs.org/express-validator/-/express-validator-6.6.1.tgz", 640 | "integrity": "sha512-+MrZKJ3eGYXkNF9p9Zf7MS7NkPJFg9MDYATU5c80Cf4F62JdLBIjWxy6481tRC0y1NnC9cgOw8FuN364bWaGhA==", 641 | "requires": { 642 | "lodash": "^4.17.19", 643 | "validator": "^13.1.1" 644 | } 645 | }, 646 | "fill-range": { 647 | "version": "7.0.1", 648 | "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz", 649 | "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", 650 | "dev": true, 651 | "requires": { 652 | "to-regex-range": "^5.0.1" 653 | } 654 | }, 655 | "finalhandler": { 656 | "version": "1.1.2", 657 | "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.1.2.tgz", 658 | "integrity": "sha512-aAWcW57uxVNrQZqFXjITpW3sIUQmHGG3qSb9mUah9MgMC4NeWhNOlNjXEYq3HjRAvL6arUviZGGJsBg6z0zsWA==", 659 | "requires": { 660 | "debug": "2.6.9", 661 | "encodeurl": "~1.0.2", 662 | "escape-html": "~1.0.3", 663 | "on-finished": "~2.3.0", 664 | "parseurl": "~1.3.3", 665 | "statuses": "~1.5.0", 666 | "unpipe": "~1.0.0" 667 | } 668 | }, 669 | "find-up": { 670 | "version": "3.0.0", 671 | "resolved": "https://registry.npmjs.org/find-up/-/find-up-3.0.0.tgz", 672 | "integrity": "sha512-1yD6RmLI1XBfxugvORwlck6f75tYL+iR0jqwsOrOxMZyGYqUuDhJ0l4AXdO1iX/FTs9cBAMEk1gWSEx1kSbylg==", 673 | "dev": true, 674 | "requires": { 675 | "locate-path": "^3.0.0" 676 | } 677 | }, 678 | "forwarded": { 679 | "version": "0.1.2", 680 | "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.1.2.tgz", 681 | "integrity": "sha1-mMI9qxF1ZXuMBXPozszZGw/xjIQ=" 682 | }, 683 | "fresh": { 684 | "version": "0.5.2", 685 | "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz", 686 | "integrity": "sha1-PYyt2Q2XZWn6g1qx+OSyOhBWBac=" 687 | }, 688 | "fsevents": { 689 | "version": "2.1.3", 690 | "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.1.3.tgz", 691 | "integrity": "sha512-Auw9a4AxqWpa9GUfj370BMPzzyncfBABW8Mab7BGWBYDj4Isgq+cDKtx0i6u9jcX9pQDnswsaaOTgTmA5pEjuQ==", 692 | "dev": true, 693 | "optional": true 694 | }, 695 | "function-bind": { 696 | "version": "1.1.1", 697 | "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.1.tgz", 698 | "integrity": "sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A==", 699 | "dev": true 700 | }, 701 | "get-caller-file": { 702 | "version": "2.0.5", 703 | "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-2.0.5.tgz", 704 | "integrity": "sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==", 705 | "dev": true 706 | }, 707 | "get-stream": { 708 | "version": "4.1.0", 709 | "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-4.1.0.tgz", 710 | "integrity": "sha512-GMat4EJ5161kIy2HevLlr4luNjBgvmj413KaQA7jt4V8B4RDsfpHk7WQ9GVqfYyyx8OS/L66Kox+rJRNklLK7w==", 711 | "dev": true, 712 | "requires": { 713 | "pump": "^3.0.0" 714 | } 715 | }, 716 | "glob-parent": { 717 | "version": "5.1.1", 718 | "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.1.tgz", 719 | "integrity": "sha512-FnI+VGOpnlGHWZxthPGR+QhR78fuiK0sNLkHQv+bL9fQi57lNNdquIbna/WrfROrolq8GK5Ek6BiMwqL/voRYQ==", 720 | "dev": true, 721 | "requires": { 722 | "is-glob": "^4.0.1" 723 | } 724 | }, 725 | "global-dirs": { 726 | "version": "2.0.1", 727 | "resolved": "https://registry.npmjs.org/global-dirs/-/global-dirs-2.0.1.tgz", 728 | "integrity": "sha512-5HqUqdhkEovj2Of/ms3IeS/EekcO54ytHRLV4PEY2rhRwrHXLQjeVEES0Lhka0xwNDtGYn58wyC4s5+MHsOO6A==", 729 | "dev": true, 730 | "requires": { 731 | "ini": "^1.3.5" 732 | } 733 | }, 734 | "got": { 735 | "version": "9.6.0", 736 | "resolved": "https://registry.npmjs.org/got/-/got-9.6.0.tgz", 737 | "integrity": "sha512-R7eWptXuGYxwijs0eV+v3o6+XH1IqVK8dJOEecQfTmkncw9AV4dcw/Dhxi8MdlqPthxxpZyizMzyg8RTmEsG+Q==", 738 | "dev": true, 739 | "requires": { 740 | "@sindresorhus/is": "^0.14.0", 741 | "@szmarczak/http-timer": "^1.1.2", 742 | "cacheable-request": "^6.0.0", 743 | "decompress-response": "^3.3.0", 744 | "duplexer3": "^0.1.4", 745 | "get-stream": "^4.1.0", 746 | "lowercase-keys": "^1.0.1", 747 | "mimic-response": "^1.0.1", 748 | "p-cancelable": "^1.0.0", 749 | "to-readable-stream": "^1.0.0", 750 | "url-parse-lax": "^3.0.0" 751 | } 752 | }, 753 | "graceful-fs": { 754 | "version": "4.2.4", 755 | "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.4.tgz", 756 | "integrity": "sha512-WjKPNJF79dtJAVniUlGGWHYGz2jWxT6VhN/4m1NdkbZ2nOsEF+cI1Edgql5zCRhs/VsQYRvrXctxktVXZUkixw==", 757 | "dev": true 758 | }, 759 | "has": { 760 | "version": "1.0.3", 761 | "resolved": "https://registry.npmjs.org/has/-/has-1.0.3.tgz", 762 | "integrity": "sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw==", 763 | "dev": true, 764 | "requires": { 765 | "function-bind": "^1.1.1" 766 | } 767 | }, 768 | "has-flag": { 769 | "version": "3.0.0", 770 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", 771 | "integrity": "sha1-tdRU3CGZriJWmfNGfloH87lVuv0=", 772 | "dev": true 773 | }, 774 | "has-yarn": { 775 | "version": "2.1.0", 776 | "resolved": "https://registry.npmjs.org/has-yarn/-/has-yarn-2.1.0.tgz", 777 | "integrity": "sha512-UqBRqi4ju7T+TqGNdqAO0PaSVGsDGJUBQvk9eUWNGRY1CFGDzYhLWoM7JQEemnlvVcv/YEmc2wNW8BC24EnUsw==", 778 | "dev": true 779 | }, 780 | "hosted-git-info": { 781 | "version": "2.8.8", 782 | "resolved": "https://registry.npmjs.org/hosted-git-info/-/hosted-git-info-2.8.8.tgz", 783 | "integrity": "sha512-f/wzC2QaWBs7t9IYqB4T3sR1xviIViXJRJTWBlx2Gf3g0Xi5vI7Yy4koXQ1c9OYDGHN9sBy1DQ2AB8fqZBWhUg==", 784 | "dev": true 785 | }, 786 | "http-cache-semantics": { 787 | "version": "4.1.0", 788 | "resolved": "https://registry.npmjs.org/http-cache-semantics/-/http-cache-semantics-4.1.0.tgz", 789 | "integrity": "sha512-carPklcUh7ROWRK7Cv27RPtdhYhUsela/ue5/jKzjegVvXDqM2ILE9Q2BGn9JZJh1g87cp56su/FgQSzcWS8cQ==", 790 | "dev": true 791 | }, 792 | "http-errors": { 793 | "version": "1.7.2", 794 | "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.7.2.tgz", 795 | "integrity": "sha512-uUQBt3H/cSIVfch6i1EuPNy/YsRSOUBXTVfZ+yR7Zjez3qjBz6i9+i4zjNaoqcoFVI4lQJ5plg63TvGfRSDCRg==", 796 | "requires": { 797 | "depd": "~1.1.2", 798 | "inherits": "2.0.3", 799 | "setprototypeof": "1.1.1", 800 | "statuses": ">= 1.5.0 < 2", 801 | "toidentifier": "1.0.0" 802 | } 803 | }, 804 | "iconv-lite": { 805 | "version": "0.4.24", 806 | "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", 807 | "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", 808 | "requires": { 809 | "safer-buffer": ">= 2.1.2 < 3" 810 | } 811 | }, 812 | "ignore-by-default": { 813 | "version": "1.0.1", 814 | "resolved": "https://registry.npmjs.org/ignore-by-default/-/ignore-by-default-1.0.1.tgz", 815 | "integrity": "sha1-SMptcvbGo68Aqa1K5odr44ieKwk=", 816 | "dev": true 817 | }, 818 | "import-lazy": { 819 | "version": "2.1.0", 820 | "resolved": "https://registry.npmjs.org/import-lazy/-/import-lazy-2.1.0.tgz", 821 | "integrity": "sha1-BWmOPUXIjo1+nZLLBYTnfwlvPkM=", 822 | "dev": true 823 | }, 824 | "imurmurhash": { 825 | "version": "0.1.4", 826 | "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", 827 | "integrity": "sha1-khi5srkoojixPcT7a21XbyMUU+o=", 828 | "dev": true 829 | }, 830 | "inherits": { 831 | "version": "2.0.3", 832 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", 833 | "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=" 834 | }, 835 | "ini": { 836 | "version": "1.3.5", 837 | "resolved": "https://registry.npmjs.org/ini/-/ini-1.3.5.tgz", 838 | "integrity": "sha512-RZY5huIKCMRWDUqZlEi72f/lmXKMvuszcMBduliQ3nnWbx9X/ZBQO7DijMEYS9EhHBb2qacRUMtC7svLwe0lcw==", 839 | "dev": true 840 | }, 841 | "ipaddr.js": { 842 | "version": "1.9.1", 843 | "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz", 844 | "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==" 845 | }, 846 | "is-arrayish": { 847 | "version": "0.2.1", 848 | "resolved": "https://registry.npmjs.org/is-arrayish/-/is-arrayish-0.2.1.tgz", 849 | "integrity": "sha1-d8mYQFJ6qOyxqLppe4BkWnqSap0=", 850 | "dev": true 851 | }, 852 | "is-binary-path": { 853 | "version": "2.1.0", 854 | "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz", 855 | "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==", 856 | "dev": true, 857 | "requires": { 858 | "binary-extensions": "^2.0.0" 859 | } 860 | }, 861 | "is-ci": { 862 | "version": "2.0.0", 863 | "resolved": "https://registry.npmjs.org/is-ci/-/is-ci-2.0.0.tgz", 864 | "integrity": "sha512-YfJT7rkpQB0updsdHLGWrvhBJfcfzNNawYDNIyQXJz0IViGf75O8EBPKSdvw2rF+LGCsX4FZ8tcr3b19LcZq4w==", 865 | "dev": true, 866 | "requires": { 867 | "ci-info": "^2.0.0" 868 | } 869 | }, 870 | "is-core-module": { 871 | "version": "2.1.0", 872 | "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.1.0.tgz", 873 | "integrity": "sha512-YcV7BgVMRFRua2FqQzKtTDMz8iCuLEyGKjr70q8Zm1yy2qKcurbFEd79PAdHV77oL3NrAaOVQIbMmiHQCHB7ZA==", 874 | "dev": true, 875 | "requires": { 876 | "has": "^1.0.3" 877 | } 878 | }, 879 | "is-extglob": { 880 | "version": "2.1.1", 881 | "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", 882 | "integrity": "sha1-qIwCU1eR8C7TfHahueqXc8gz+MI=", 883 | "dev": true 884 | }, 885 | "is-fullwidth-code-point": { 886 | "version": "2.0.0", 887 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", 888 | "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=", 889 | "dev": true 890 | }, 891 | "is-glob": { 892 | "version": "4.0.1", 893 | "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.1.tgz", 894 | "integrity": "sha512-5G0tKtBTFImOqDnLB2hG6Bp2qcKEFduo4tZu9MT/H6NQv/ghhy30o55ufafxJ/LdH79LLs2Kfrn85TLKyA7BUg==", 895 | "dev": true, 896 | "requires": { 897 | "is-extglob": "^2.1.1" 898 | } 899 | }, 900 | "is-installed-globally": { 901 | "version": "0.3.2", 902 | "resolved": "https://registry.npmjs.org/is-installed-globally/-/is-installed-globally-0.3.2.tgz", 903 | "integrity": "sha512-wZ8x1js7Ia0kecP/CHM/3ABkAmujX7WPvQk6uu3Fly/Mk44pySulQpnHG46OMjHGXApINnV4QhY3SWnECO2z5g==", 904 | "dev": true, 905 | "requires": { 906 | "global-dirs": "^2.0.1", 907 | "is-path-inside": "^3.0.1" 908 | } 909 | }, 910 | "is-npm": { 911 | "version": "4.0.0", 912 | "resolved": "https://registry.npmjs.org/is-npm/-/is-npm-4.0.0.tgz", 913 | "integrity": "sha512-96ECIfh9xtDDlPylNPXhzjsykHsMJZ18ASpaWzQyBr4YRTcVjUvzaHayDAES2oU/3KpljhHUjtSRNiDwi0F0ig==", 914 | "dev": true 915 | }, 916 | "is-number": { 917 | "version": "7.0.0", 918 | "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", 919 | "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", 920 | "dev": true 921 | }, 922 | "is-obj": { 923 | "version": "2.0.0", 924 | "resolved": "https://registry.npmjs.org/is-obj/-/is-obj-2.0.0.tgz", 925 | "integrity": "sha512-drqDG3cbczxxEJRoOXcOjtdp1J/lyp1mNn0xaznRs8+muBhgQcrnbspox5X5fOw0HnMnbfDzvnEMEtqDEJEo8w==", 926 | "dev": true 927 | }, 928 | "is-path-inside": { 929 | "version": "3.0.2", 930 | "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-3.0.2.tgz", 931 | "integrity": "sha512-/2UGPSgmtqwo1ktx8NDHjuPwZWmHhO+gj0f93EkhLB5RgW9RZevWYYlIkS6zePc6U2WpOdQYIwHe9YC4DWEBVg==", 932 | "dev": true 933 | }, 934 | "is-typedarray": { 935 | "version": "1.0.0", 936 | "resolved": "https://registry.npmjs.org/is-typedarray/-/is-typedarray-1.0.0.tgz", 937 | "integrity": "sha1-5HnICFjfDBsR3dppQPlgEfzaSpo=", 938 | "dev": true 939 | }, 940 | "is-yarn-global": { 941 | "version": "0.3.0", 942 | "resolved": "https://registry.npmjs.org/is-yarn-global/-/is-yarn-global-0.3.0.tgz", 943 | "integrity": "sha512-VjSeb/lHmkoyd8ryPVIKvOCn4D1koMqY+vqyjjUfc3xyKtP4dYOxM44sZrnqQSzSds3xyOrUTLTC9LVCVgLngw==", 944 | "dev": true 945 | }, 946 | "isarray": { 947 | "version": "1.0.0", 948 | "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", 949 | "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE=" 950 | }, 951 | "json-buffer": { 952 | "version": "3.0.0", 953 | "resolved": "https://registry.npmjs.org/json-buffer/-/json-buffer-3.0.0.tgz", 954 | "integrity": "sha1-Wx85evx11ne96Lz8Dkfh+aPZqJg=", 955 | "dev": true 956 | }, 957 | "json-parse-better-errors": { 958 | "version": "1.0.2", 959 | "resolved": "https://registry.npmjs.org/json-parse-better-errors/-/json-parse-better-errors-1.0.2.tgz", 960 | "integrity": "sha512-mrqyZKfX5EhL7hvqcV6WG1yYjnjeuYDzDhhcAAUrq8Po85NBQBJP+ZDUT75qZQ98IkUoBqdkExkukOU7Ts2wrw==", 961 | "dev": true 962 | }, 963 | "json5": { 964 | "version": "2.1.3", 965 | "resolved": "https://registry.npmjs.org/json5/-/json5-2.1.3.tgz", 966 | "integrity": "sha512-KXPvOm8K9IJKFM0bmdn8QXh7udDh1g/giieX0NLCaMnb4hEiVFqnop2ImTXCc5e0/oHz3LTqmHGtExn5hfMkOA==", 967 | "requires": { 968 | "minimist": "^1.2.5" 969 | } 970 | }, 971 | "jsonwebtoken": { 972 | "version": "8.5.1", 973 | "resolved": "https://registry.npmjs.org/jsonwebtoken/-/jsonwebtoken-8.5.1.tgz", 974 | "integrity": "sha512-XjwVfRS6jTMsqYs0EsuJ4LGxXV14zQybNd4L2r0UvbVnSF9Af8x7p5MzbJ90Ioz/9TI41/hTCvznF/loiSzn8w==", 975 | "requires": { 976 | "jws": "^3.2.2", 977 | "lodash.includes": "^4.3.0", 978 | "lodash.isboolean": "^3.0.3", 979 | "lodash.isinteger": "^4.0.4", 980 | "lodash.isnumber": "^3.0.3", 981 | "lodash.isplainobject": "^4.0.6", 982 | "lodash.isstring": "^4.0.1", 983 | "lodash.once": "^4.0.0", 984 | "ms": "^2.1.1", 985 | "semver": "^5.6.0" 986 | }, 987 | "dependencies": { 988 | "ms": { 989 | "version": "2.1.2", 990 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", 991 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==" 992 | } 993 | } 994 | }, 995 | "jwa": { 996 | "version": "1.4.1", 997 | "resolved": "https://registry.npmjs.org/jwa/-/jwa-1.4.1.tgz", 998 | "integrity": "sha512-qiLX/xhEEFKUAJ6FiBMbes3w9ATzyk5W7Hvzpa/SLYdxNtng+gcurvrI7TbACjIXlsJyr05/S1oUhZrc63evQA==", 999 | "requires": { 1000 | "buffer-equal-constant-time": "1.0.1", 1001 | "ecdsa-sig-formatter": "1.0.11", 1002 | "safe-buffer": "^5.0.1" 1003 | } 1004 | }, 1005 | "jws": { 1006 | "version": "3.2.2", 1007 | "resolved": "https://registry.npmjs.org/jws/-/jws-3.2.2.tgz", 1008 | "integrity": "sha512-YHlZCB6lMTllWDtSPHz/ZXTsi8S00usEV6v1tjq8tOUZzw7DpSDWVXjXDre6ed1w/pd495ODpHZYSdkRTsa0HA==", 1009 | "requires": { 1010 | "jwa": "^1.4.1", 1011 | "safe-buffer": "^5.0.1" 1012 | } 1013 | }, 1014 | "kareem": { 1015 | "version": "2.3.1", 1016 | "resolved": "https://registry.npmjs.org/kareem/-/kareem-2.3.1.tgz", 1017 | "integrity": "sha512-l3hLhffs9zqoDe8zjmb/mAN4B8VT3L56EUvKNqLFVs9YlFA+zx7ke1DO8STAdDyYNkeSo1nKmjuvQeI12So8Xw==" 1018 | }, 1019 | "keyv": { 1020 | "version": "3.1.0", 1021 | "resolved": "https://registry.npmjs.org/keyv/-/keyv-3.1.0.tgz", 1022 | "integrity": "sha512-9ykJ/46SN/9KPM/sichzQ7OvXyGDYKGTaDlKMGCAlg2UK8KRy4jb0d8sFc+0Tt0YYnThq8X2RZgCg74RPxgcVA==", 1023 | "dev": true, 1024 | "requires": { 1025 | "json-buffer": "3.0.0" 1026 | } 1027 | }, 1028 | "latest-version": { 1029 | "version": "5.1.0", 1030 | "resolved": "https://registry.npmjs.org/latest-version/-/latest-version-5.1.0.tgz", 1031 | "integrity": "sha512-weT+r0kTkRQdCdYCNtkMwWXQTMEswKrFBkm4ckQOMVhhqhIMI1UT2hMj+1iigIhgSZm5gTmrRXBNoGUgaTY1xA==", 1032 | "dev": true, 1033 | "requires": { 1034 | "package-json": "^6.3.0" 1035 | } 1036 | }, 1037 | "locate-path": { 1038 | "version": "3.0.0", 1039 | "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-3.0.0.tgz", 1040 | "integrity": "sha512-7AO748wWnIhNqAuaty2ZWHkQHRSNfPVIsPIfwEOWO22AmaoVrWavlOcMR5nzTLNYvp36X220/maaRsrec1G65A==", 1041 | "dev": true, 1042 | "requires": { 1043 | "p-locate": "^3.0.0", 1044 | "path-exists": "^3.0.0" 1045 | } 1046 | }, 1047 | "lodash": { 1048 | "version": "4.17.20", 1049 | "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.20.tgz", 1050 | "integrity": "sha512-PlhdFcillOINfeV7Ni6oF1TAEayyZBoZ8bcshTHqOYJYlrqzRK5hagpagky5o4HfCzzd1TRkXPMFq6cKk9rGmA==" 1051 | }, 1052 | "lodash.includes": { 1053 | "version": "4.3.0", 1054 | "resolved": "https://registry.npmjs.org/lodash.includes/-/lodash.includes-4.3.0.tgz", 1055 | "integrity": "sha1-YLuYqHy5I8aMoeUTJUgzFISfVT8=" 1056 | }, 1057 | "lodash.isboolean": { 1058 | "version": "3.0.3", 1059 | "resolved": "https://registry.npmjs.org/lodash.isboolean/-/lodash.isboolean-3.0.3.tgz", 1060 | "integrity": "sha1-bC4XHbKiV82WgC/UOwGyDV9YcPY=" 1061 | }, 1062 | "lodash.isinteger": { 1063 | "version": "4.0.4", 1064 | "resolved": "https://registry.npmjs.org/lodash.isinteger/-/lodash.isinteger-4.0.4.tgz", 1065 | "integrity": "sha1-YZwK89A/iwTDH1iChAt3sRzWg0M=" 1066 | }, 1067 | "lodash.isnumber": { 1068 | "version": "3.0.3", 1069 | "resolved": "https://registry.npmjs.org/lodash.isnumber/-/lodash.isnumber-3.0.3.tgz", 1070 | "integrity": "sha1-POdoEMWSjQM1IwGsKHMX8RwLH/w=" 1071 | }, 1072 | "lodash.isplainobject": { 1073 | "version": "4.0.6", 1074 | "resolved": "https://registry.npmjs.org/lodash.isplainobject/-/lodash.isplainobject-4.0.6.tgz", 1075 | "integrity": "sha1-fFJqUtibRcRcxpC4gWO+BJf1UMs=" 1076 | }, 1077 | "lodash.isstring": { 1078 | "version": "4.0.1", 1079 | "resolved": "https://registry.npmjs.org/lodash.isstring/-/lodash.isstring-4.0.1.tgz", 1080 | "integrity": "sha1-1SfftUVuynzJu5XV2ur4i6VKVFE=" 1081 | }, 1082 | "lodash.once": { 1083 | "version": "4.1.1", 1084 | "resolved": "https://registry.npmjs.org/lodash.once/-/lodash.once-4.1.1.tgz", 1085 | "integrity": "sha1-DdOXEhPHxW34gJd9UEyI+0cal6w=" 1086 | }, 1087 | "lowercase-keys": { 1088 | "version": "1.0.1", 1089 | "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-1.0.1.tgz", 1090 | "integrity": "sha512-G2Lj61tXDnVFFOi8VZds+SoQjtQC3dgokKdDG2mTm1tx4m50NUHBOZSBwQQHyy0V12A0JTG4icfZQH+xPyh8VA==", 1091 | "dev": true 1092 | }, 1093 | "make-dir": { 1094 | "version": "3.1.0", 1095 | "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-3.1.0.tgz", 1096 | "integrity": "sha512-g3FeP20LNwhALb/6Cz6Dd4F2ngze0jz7tbzrD2wAV+o9FeNHe4rL+yK2md0J/fiSf1sa1ADhXqi5+oVwOM/eGw==", 1097 | "dev": true, 1098 | "requires": { 1099 | "semver": "^6.0.0" 1100 | }, 1101 | "dependencies": { 1102 | "semver": { 1103 | "version": "6.3.0", 1104 | "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", 1105 | "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", 1106 | "dev": true 1107 | } 1108 | } 1109 | }, 1110 | "media-typer": { 1111 | "version": "0.3.0", 1112 | "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", 1113 | "integrity": "sha1-hxDXrwqmJvj/+hzgAWhUUmMlV0g=" 1114 | }, 1115 | "memory-pager": { 1116 | "version": "1.5.0", 1117 | "resolved": "https://registry.npmjs.org/memory-pager/-/memory-pager-1.5.0.tgz", 1118 | "integrity": "sha512-ZS4Bp4r/Zoeq6+NLJpP+0Zzm0pR8whtGPf1XExKLJBAczGMnSi3It14OiNCStjQjM6NU1okjQGSxgEZN8eBYKg==", 1119 | "optional": true 1120 | }, 1121 | "merge-descriptors": { 1122 | "version": "1.0.1", 1123 | "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz", 1124 | "integrity": "sha1-sAqqVW3YtEVoFQ7J0blT8/kMu2E=" 1125 | }, 1126 | "methods": { 1127 | "version": "1.1.2", 1128 | "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz", 1129 | "integrity": "sha1-VSmk1nZUE07cxSZmVoNbD4Ua/O4=" 1130 | }, 1131 | "mime": { 1132 | "version": "1.6.0", 1133 | "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz", 1134 | "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==" 1135 | }, 1136 | "mime-db": { 1137 | "version": "1.44.0", 1138 | "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.44.0.tgz", 1139 | "integrity": "sha512-/NOTfLrsPBVeH7YtFPgsVWveuL+4SjjYxaQ1xtM1KMFj7HdxlBlxeyNLzhyJVx7r4rZGJAZ/6lkKCitSc/Nmpg==" 1140 | }, 1141 | "mime-types": { 1142 | "version": "2.1.27", 1143 | "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.27.tgz", 1144 | "integrity": "sha512-JIhqnCasI9yD+SsmkquHBxTSEuZdQX5BuQnS2Vc7puQQQ+8yiP5AY5uWhpdv4YL4VM5c6iliiYWPgJ/nJQLp7w==", 1145 | "requires": { 1146 | "mime-db": "1.44.0" 1147 | } 1148 | }, 1149 | "mimic-response": { 1150 | "version": "1.0.1", 1151 | "resolved": "https://registry.npmjs.org/mimic-response/-/mimic-response-1.0.1.tgz", 1152 | "integrity": "sha512-j5EctnkH7amfV/q5Hgmoal1g2QHFJRraOtmx0JpIqkxhBhI/lJSl1nMpQ45hVarwNETOoWEimndZ4QK0RHxuxQ==", 1153 | "dev": true 1154 | }, 1155 | "minimatch": { 1156 | "version": "3.0.4", 1157 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz", 1158 | "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==", 1159 | "dev": true, 1160 | "requires": { 1161 | "brace-expansion": "^1.1.7" 1162 | } 1163 | }, 1164 | "minimist": { 1165 | "version": "1.2.5", 1166 | "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.5.tgz", 1167 | "integrity": "sha512-FM9nNUYrRBAELZQT3xeZQ7fmMOBg6nWNmJKTcgsJeaLstP/UODVpGsr5OhXhhXg6f+qtJ8uiZ+PUxkDWcgIXLw==" 1168 | }, 1169 | "mongodb": { 1170 | "version": "3.6.3", 1171 | "resolved": "https://registry.npmjs.org/mongodb/-/mongodb-3.6.3.tgz", 1172 | "integrity": "sha512-rOZuR0QkodZiM+UbQE5kDsJykBqWi0CL4Ec2i1nrGrUI3KO11r6Fbxskqmq3JK2NH7aW4dcccBuUujAP0ERl5w==", 1173 | "requires": { 1174 | "bl": "^2.2.1", 1175 | "bson": "^1.1.4", 1176 | "denque": "^1.4.1", 1177 | "require_optional": "^1.0.1", 1178 | "safe-buffer": "^5.1.2", 1179 | "saslprep": "^1.0.0" 1180 | } 1181 | }, 1182 | "mongoose": { 1183 | "version": "5.10.15", 1184 | "resolved": "https://registry.npmjs.org/mongoose/-/mongoose-5.10.15.tgz", 1185 | "integrity": "sha512-3QUWCpMRdFCPIBZkjG/B2OkfMY2WLkR+hv335o4T2mn3ta9kx8qVvXeUDojp3OHMxBZVUyCA+hDyyP4/aKmHuA==", 1186 | "requires": { 1187 | "bson": "^1.1.4", 1188 | "kareem": "2.3.1", 1189 | "mongodb": "3.6.3", 1190 | "mongoose-legacy-pluralize": "1.0.2", 1191 | "mpath": "0.7.0", 1192 | "mquery": "3.2.2", 1193 | "ms": "2.1.2", 1194 | "regexp-clone": "1.0.0", 1195 | "safe-buffer": "5.2.1", 1196 | "sift": "7.0.1", 1197 | "sliced": "1.0.1" 1198 | }, 1199 | "dependencies": { 1200 | "ms": { 1201 | "version": "2.1.2", 1202 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", 1203 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==" 1204 | }, 1205 | "safe-buffer": { 1206 | "version": "5.2.1", 1207 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz", 1208 | "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==" 1209 | } 1210 | } 1211 | }, 1212 | "mongoose-legacy-pluralize": { 1213 | "version": "1.0.2", 1214 | "resolved": "https://registry.npmjs.org/mongoose-legacy-pluralize/-/mongoose-legacy-pluralize-1.0.2.tgz", 1215 | "integrity": "sha512-Yo/7qQU4/EyIS8YDFSeenIvXxZN+ld7YdV9LqFVQJzTLye8unujAWPZ4NWKfFA+RNjh+wvTWKY9Z3E5XM6ZZiQ==" 1216 | }, 1217 | "mpath": { 1218 | "version": "0.7.0", 1219 | "resolved": "https://registry.npmjs.org/mpath/-/mpath-0.7.0.tgz", 1220 | "integrity": "sha512-Aiq04hILxhz1L+f7sjGyn7IxYzWm1zLNNXcfhDtx04kZ2Gk7uvFdgZ8ts1cWa/6d0TQmag2yR8zSGZUmp0tFNg==" 1221 | }, 1222 | "mquery": { 1223 | "version": "3.2.2", 1224 | "resolved": "https://registry.npmjs.org/mquery/-/mquery-3.2.2.tgz", 1225 | "integrity": "sha512-XB52992COp0KP230I3qloVUbkLUxJIu328HBP2t2EsxSFtf4W1HPSOBWOXf1bqxK4Xbb66lfMJ+Bpfd9/yZE1Q==", 1226 | "requires": { 1227 | "bluebird": "3.5.1", 1228 | "debug": "3.1.0", 1229 | "regexp-clone": "^1.0.0", 1230 | "safe-buffer": "5.1.2", 1231 | "sliced": "1.0.1" 1232 | }, 1233 | "dependencies": { 1234 | "debug": { 1235 | "version": "3.1.0", 1236 | "resolved": "https://registry.npmjs.org/debug/-/debug-3.1.0.tgz", 1237 | "integrity": "sha512-OX8XqP7/1a9cqkxYw2yXss15f26NKWBpDXQd0/uK/KPqdQhxbPa994hnzjcE2VqQpDslf55723cKPUOGSmMY3g==", 1238 | "requires": { 1239 | "ms": "2.0.0" 1240 | } 1241 | } 1242 | } 1243 | }, 1244 | "ms": { 1245 | "version": "2.0.0", 1246 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", 1247 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" 1248 | }, 1249 | "negotiator": { 1250 | "version": "0.6.2", 1251 | "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.2.tgz", 1252 | "integrity": "sha512-hZXc7K2e+PgeI1eDBe/10Ard4ekbfrrqG8Ep+8Jmf4JID2bNg7NvCPOZN+kfF574pFQI7mum2AUqDidoKqcTOw==" 1253 | }, 1254 | "nodemon": { 1255 | "version": "2.0.6", 1256 | "resolved": "https://registry.npmjs.org/nodemon/-/nodemon-2.0.6.tgz", 1257 | "integrity": "sha512-4I3YDSKXg6ltYpcnZeHompqac4E6JeAMpGm8tJnB9Y3T0ehasLa4139dJOcCrB93HHrUMsCrKtoAlXTqT5n4AQ==", 1258 | "dev": true, 1259 | "requires": { 1260 | "chokidar": "^3.2.2", 1261 | "debug": "^3.2.6", 1262 | "ignore-by-default": "^1.0.1", 1263 | "minimatch": "^3.0.4", 1264 | "pstree.remy": "^1.1.7", 1265 | "semver": "^5.7.1", 1266 | "supports-color": "^5.5.0", 1267 | "touch": "^3.1.0", 1268 | "undefsafe": "^2.0.3", 1269 | "update-notifier": "^4.1.0" 1270 | }, 1271 | "dependencies": { 1272 | "debug": { 1273 | "version": "3.2.6", 1274 | "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.6.tgz", 1275 | "integrity": "sha512-mel+jf7nrtEl5Pn1Qx46zARXKDpBbvzezse7p7LqINmdoIk8PYP5SySaxEmYv6TZ0JyEKA1hsCId6DIhgITtWQ==", 1276 | "dev": true, 1277 | "requires": { 1278 | "ms": "^2.1.1" 1279 | } 1280 | }, 1281 | "ms": { 1282 | "version": "2.1.2", 1283 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", 1284 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", 1285 | "dev": true 1286 | }, 1287 | "supports-color": { 1288 | "version": "5.5.0", 1289 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", 1290 | "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", 1291 | "dev": true, 1292 | "requires": { 1293 | "has-flag": "^3.0.0" 1294 | } 1295 | } 1296 | } 1297 | }, 1298 | "nopt": { 1299 | "version": "1.0.10", 1300 | "resolved": "https://registry.npmjs.org/nopt/-/nopt-1.0.10.tgz", 1301 | "integrity": "sha1-bd0hvSoxQXuScn3Vhfim83YI6+4=", 1302 | "dev": true, 1303 | "requires": { 1304 | "abbrev": "1" 1305 | } 1306 | }, 1307 | "normalize-package-data": { 1308 | "version": "2.5.0", 1309 | "resolved": "https://registry.npmjs.org/normalize-package-data/-/normalize-package-data-2.5.0.tgz", 1310 | "integrity": "sha512-/5CMN3T0R4XTj4DcGaexo+roZSdSFW/0AOOTROrjxzCG1wrWXEsGbRKevjlIL+ZDE4sZlJr5ED4YW0yqmkK+eA==", 1311 | "dev": true, 1312 | "requires": { 1313 | "hosted-git-info": "^2.1.4", 1314 | "resolve": "^1.10.0", 1315 | "semver": "2 || 3 || 4 || 5", 1316 | "validate-npm-package-license": "^3.0.1" 1317 | } 1318 | }, 1319 | "normalize-path": { 1320 | "version": "3.0.0", 1321 | "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", 1322 | "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==", 1323 | "dev": true 1324 | }, 1325 | "normalize-url": { 1326 | "version": "4.5.0", 1327 | "resolved": "https://registry.npmjs.org/normalize-url/-/normalize-url-4.5.0.tgz", 1328 | "integrity": "sha512-2s47yzUxdexf1OhyRi4Em83iQk0aPvwTddtFz4hnSSw9dCEsLEGf6SwIO8ss/19S9iBb5sJaOuTvTGDeZI00BQ==", 1329 | "dev": true 1330 | }, 1331 | "on-finished": { 1332 | "version": "2.3.0", 1333 | "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.3.0.tgz", 1334 | "integrity": "sha1-IPEzZIGwg811M3mSoWlxqi2QaUc=", 1335 | "requires": { 1336 | "ee-first": "1.1.1" 1337 | } 1338 | }, 1339 | "once": { 1340 | "version": "1.4.0", 1341 | "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", 1342 | "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=", 1343 | "dev": true, 1344 | "requires": { 1345 | "wrappy": "1" 1346 | } 1347 | }, 1348 | "p-cancelable": { 1349 | "version": "1.1.0", 1350 | "resolved": "https://registry.npmjs.org/p-cancelable/-/p-cancelable-1.1.0.tgz", 1351 | "integrity": "sha512-s73XxOZ4zpt1edZYZzvhqFa6uvQc1vwUa0K0BdtIZgQMAJj9IbebH+JkgKZc9h+B05PKHLOTl4ajG1BmNrVZlw==", 1352 | "dev": true 1353 | }, 1354 | "p-limit": { 1355 | "version": "2.3.0", 1356 | "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-2.3.0.tgz", 1357 | "integrity": "sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w==", 1358 | "dev": true, 1359 | "requires": { 1360 | "p-try": "^2.0.0" 1361 | } 1362 | }, 1363 | "p-locate": { 1364 | "version": "3.0.0", 1365 | "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-3.0.0.tgz", 1366 | "integrity": "sha512-x+12w/To+4GFfgJhBEpiDcLozRJGegY+Ei7/z0tSLkMmxGZNybVMSfWj9aJn8Z5Fc7dBUNJOOVgPv2H7IwulSQ==", 1367 | "dev": true, 1368 | "requires": { 1369 | "p-limit": "^2.0.0" 1370 | } 1371 | }, 1372 | "p-try": { 1373 | "version": "2.2.0", 1374 | "resolved": "https://registry.npmjs.org/p-try/-/p-try-2.2.0.tgz", 1375 | "integrity": "sha512-R4nPAVTAU0B9D35/Gk3uJf/7XYbQcyohSKdvAxIRSNghFl4e71hVoGnBNQz9cWaXxO2I10KTC+3jMdvvoKw6dQ==", 1376 | "dev": true 1377 | }, 1378 | "package-json": { 1379 | "version": "6.5.0", 1380 | "resolved": "https://registry.npmjs.org/package-json/-/package-json-6.5.0.tgz", 1381 | "integrity": "sha512-k3bdm2n25tkyxcjSKzB5x8kfVxlMdgsbPr0GkZcwHsLpba6cBjqCt1KlcChKEvxHIcTB1FVMuwoijZ26xex5MQ==", 1382 | "dev": true, 1383 | "requires": { 1384 | "got": "^9.6.0", 1385 | "registry-auth-token": "^4.0.0", 1386 | "registry-url": "^5.0.0", 1387 | "semver": "^6.2.0" 1388 | }, 1389 | "dependencies": { 1390 | "semver": { 1391 | "version": "6.3.0", 1392 | "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", 1393 | "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", 1394 | "dev": true 1395 | } 1396 | } 1397 | }, 1398 | "parse-json": { 1399 | "version": "4.0.0", 1400 | "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-4.0.0.tgz", 1401 | "integrity": "sha1-vjX1Qlvh9/bHRxhPmKeIy5lHfuA=", 1402 | "dev": true, 1403 | "requires": { 1404 | "error-ex": "^1.3.1", 1405 | "json-parse-better-errors": "^1.0.1" 1406 | } 1407 | }, 1408 | "parseurl": { 1409 | "version": "1.3.3", 1410 | "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz", 1411 | "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==" 1412 | }, 1413 | "path-exists": { 1414 | "version": "3.0.0", 1415 | "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-3.0.0.tgz", 1416 | "integrity": "sha1-zg6+ql94yxiSXqfYENe1mwEP1RU=", 1417 | "dev": true 1418 | }, 1419 | "path-parse": { 1420 | "version": "1.0.6", 1421 | "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.6.tgz", 1422 | "integrity": "sha512-GSmOT2EbHrINBf9SR7CDELwlJ8AENk3Qn7OikK4nFYAu3Ote2+JYNVvkpAEQm3/TLNEJFD/xZJjzyxg3KBWOzw==", 1423 | "dev": true 1424 | }, 1425 | "path-to-regexp": { 1426 | "version": "0.1.7", 1427 | "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz", 1428 | "integrity": "sha1-32BBeABfUi8V60SQ5yR6G/qmf4w=" 1429 | }, 1430 | "picomatch": { 1431 | "version": "2.2.2", 1432 | "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.2.2.tgz", 1433 | "integrity": "sha512-q0M/9eZHzmr0AulXyPwNfZjtwZ/RBZlbN3K3CErVrk50T2ASYI7Bye0EvekFY3IP1Nt2DHu0re+V2ZHIpMkuWg==", 1434 | "dev": true 1435 | }, 1436 | "pify": { 1437 | "version": "3.0.0", 1438 | "resolved": "https://registry.npmjs.org/pify/-/pify-3.0.0.tgz", 1439 | "integrity": "sha1-5aSs0sEB/fPZpNB/DbxNtJ3SgXY=", 1440 | "dev": true 1441 | }, 1442 | "prepend-http": { 1443 | "version": "2.0.0", 1444 | "resolved": "https://registry.npmjs.org/prepend-http/-/prepend-http-2.0.0.tgz", 1445 | "integrity": "sha1-6SQ0v6XqjBn0HN/UAddBo8gZ2Jc=", 1446 | "dev": true 1447 | }, 1448 | "process-nextick-args": { 1449 | "version": "2.0.1", 1450 | "resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-2.0.1.tgz", 1451 | "integrity": "sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag==" 1452 | }, 1453 | "proxy-addr": { 1454 | "version": "2.0.6", 1455 | "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.6.tgz", 1456 | "integrity": "sha512-dh/frvCBVmSsDYzw6n926jv974gddhkFPfiN8hPOi30Wax25QZyZEGveluCgliBnqmuM+UJmBErbAUFIoDbjOw==", 1457 | "requires": { 1458 | "forwarded": "~0.1.2", 1459 | "ipaddr.js": "1.9.1" 1460 | } 1461 | }, 1462 | "pstree.remy": { 1463 | "version": "1.1.8", 1464 | "resolved": "https://registry.npmjs.org/pstree.remy/-/pstree.remy-1.1.8.tgz", 1465 | "integrity": "sha512-77DZwxQmxKnu3aR542U+X8FypNzbfJ+C5XQDk3uWjWxn6151aIMGthWYRXTqT1E5oJvg+ljaa2OJi+VfvCOQ8w==", 1466 | "dev": true 1467 | }, 1468 | "pump": { 1469 | "version": "3.0.0", 1470 | "resolved": "https://registry.npmjs.org/pump/-/pump-3.0.0.tgz", 1471 | "integrity": "sha512-LwZy+p3SFs1Pytd/jYct4wpv49HiYCqd9Rlc5ZVdk0V+8Yzv6jR5Blk3TRmPL1ft69TxP0IMZGJ+WPFU2BFhww==", 1472 | "dev": true, 1473 | "requires": { 1474 | "end-of-stream": "^1.1.0", 1475 | "once": "^1.3.1" 1476 | } 1477 | }, 1478 | "pupa": { 1479 | "version": "2.1.1", 1480 | "resolved": "https://registry.npmjs.org/pupa/-/pupa-2.1.1.tgz", 1481 | "integrity": "sha512-l1jNAspIBSFqbT+y+5FosojNpVpF94nlI+wDUpqP9enwOTfHx9f0gh5nB96vl+6yTpsJsypeNrwfzPrKuHB41A==", 1482 | "dev": true, 1483 | "requires": { 1484 | "escape-goat": "^2.0.0" 1485 | } 1486 | }, 1487 | "qs": { 1488 | "version": "6.7.0", 1489 | "resolved": "https://registry.npmjs.org/qs/-/qs-6.7.0.tgz", 1490 | "integrity": "sha512-VCdBRNFTX1fyE7Nb6FYoURo/SPe62QCaAyzJvUjwRaIsc+NePBEniHlvxFmmX56+HZphIGtV0XeCirBtpDrTyQ==" 1491 | }, 1492 | "range-parser": { 1493 | "version": "1.2.1", 1494 | "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz", 1495 | "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==" 1496 | }, 1497 | "raw-body": { 1498 | "version": "2.4.0", 1499 | "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.4.0.tgz", 1500 | "integrity": "sha512-4Oz8DUIwdvoa5qMJelxipzi/iJIi40O5cGV1wNYp5hvZP8ZN0T+jiNkL0QepXs+EsQ9XJ8ipEDoiH70ySUJP3Q==", 1501 | "requires": { 1502 | "bytes": "3.1.0", 1503 | "http-errors": "1.7.2", 1504 | "iconv-lite": "0.4.24", 1505 | "unpipe": "1.0.0" 1506 | } 1507 | }, 1508 | "rc": { 1509 | "version": "1.2.8", 1510 | "resolved": "https://registry.npmjs.org/rc/-/rc-1.2.8.tgz", 1511 | "integrity": "sha512-y3bGgqKj3QBdxLbLkomlohkvsA8gdAiUQlSBJnBhfn+BPxg4bc62d8TcBW15wavDfgexCgccckhcZvywyQYPOw==", 1512 | "dev": true, 1513 | "requires": { 1514 | "deep-extend": "^0.6.0", 1515 | "ini": "~1.3.0", 1516 | "minimist": "^1.2.0", 1517 | "strip-json-comments": "~2.0.1" 1518 | } 1519 | }, 1520 | "read-pkg": { 1521 | "version": "4.0.1", 1522 | "resolved": "https://registry.npmjs.org/read-pkg/-/read-pkg-4.0.1.tgz", 1523 | "integrity": "sha1-ljYlN48+HE1IyFhytabsfV0JMjc=", 1524 | "dev": true, 1525 | "requires": { 1526 | "normalize-package-data": "^2.3.2", 1527 | "parse-json": "^4.0.0", 1528 | "pify": "^3.0.0" 1529 | } 1530 | }, 1531 | "readable-stream": { 1532 | "version": "2.3.7", 1533 | "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.7.tgz", 1534 | "integrity": "sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw==", 1535 | "requires": { 1536 | "core-util-is": "~1.0.0", 1537 | "inherits": "~2.0.3", 1538 | "isarray": "~1.0.0", 1539 | "process-nextick-args": "~2.0.0", 1540 | "safe-buffer": "~5.1.1", 1541 | "string_decoder": "~1.1.1", 1542 | "util-deprecate": "~1.0.1" 1543 | } 1544 | }, 1545 | "readdirp": { 1546 | "version": "3.5.0", 1547 | "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.5.0.tgz", 1548 | "integrity": "sha512-cMhu7c/8rdhkHXWsY+osBhfSy0JikwpHK/5+imo+LpeasTF8ouErHrlYkwT0++njiyuDvc7OFY5T3ukvZ8qmFQ==", 1549 | "dev": true, 1550 | "requires": { 1551 | "picomatch": "^2.2.1" 1552 | } 1553 | }, 1554 | "regexp-clone": { 1555 | "version": "1.0.0", 1556 | "resolved": "https://registry.npmjs.org/regexp-clone/-/regexp-clone-1.0.0.tgz", 1557 | "integrity": "sha512-TuAasHQNamyyJ2hb97IuBEif4qBHGjPHBS64sZwytpLEqtBQ1gPJTnOaQ6qmpET16cK14kkjbazl6+p0RRv0yw==" 1558 | }, 1559 | "registry-auth-token": { 1560 | "version": "4.2.1", 1561 | "resolved": "https://registry.npmjs.org/registry-auth-token/-/registry-auth-token-4.2.1.tgz", 1562 | "integrity": "sha512-6gkSb4U6aWJB4SF2ZvLb76yCBjcvufXBqvvEx1HbmKPkutswjW1xNVRY0+daljIYRbogN7O0etYSlbiaEQyMyw==", 1563 | "dev": true, 1564 | "requires": { 1565 | "rc": "^1.2.8" 1566 | } 1567 | }, 1568 | "registry-url": { 1569 | "version": "5.1.0", 1570 | "resolved": "https://registry.npmjs.org/registry-url/-/registry-url-5.1.0.tgz", 1571 | "integrity": "sha512-8acYXXTI0AkQv6RAOjE3vOaIXZkT9wo4LOFbBKYQEEnnMNBpKqdUrI6S4NT0KPIo/WVvJ5tE/X5LF/TQUf0ekw==", 1572 | "dev": true, 1573 | "requires": { 1574 | "rc": "^1.2.8" 1575 | } 1576 | }, 1577 | "require-directory": { 1578 | "version": "2.1.1", 1579 | "resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz", 1580 | "integrity": "sha1-jGStX9MNqxyXbiNE/+f3kqam30I=", 1581 | "dev": true 1582 | }, 1583 | "require-main-filename": { 1584 | "version": "2.0.0", 1585 | "resolved": "https://registry.npmjs.org/require-main-filename/-/require-main-filename-2.0.0.tgz", 1586 | "integrity": "sha512-NKN5kMDylKuldxYLSUfrbo5Tuzh4hd+2E8NPPX02mZtn1VuREQToYe/ZdlJy+J3uCpfaiGF05e7B8W0iXbQHmg==", 1587 | "dev": true 1588 | }, 1589 | "require_optional": { 1590 | "version": "1.0.1", 1591 | "resolved": "https://registry.npmjs.org/require_optional/-/require_optional-1.0.1.tgz", 1592 | "integrity": "sha512-qhM/y57enGWHAe3v/NcwML6a3/vfESLe/sGM2dII+gEO0BpKRUkWZow/tyloNqJyN6kXSl3RyyM8Ll5D/sJP8g==", 1593 | "requires": { 1594 | "resolve-from": "^2.0.0", 1595 | "semver": "^5.1.0" 1596 | } 1597 | }, 1598 | "resolve": { 1599 | "version": "1.19.0", 1600 | "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.19.0.tgz", 1601 | "integrity": "sha512-rArEXAgsBG4UgRGcynxWIWKFvh/XZCcS8UJdHhwy91zwAvCZIbcs+vAbflgBnNjYMs/i/i+/Ux6IZhML1yPvxg==", 1602 | "dev": true, 1603 | "requires": { 1604 | "is-core-module": "^2.1.0", 1605 | "path-parse": "^1.0.6" 1606 | } 1607 | }, 1608 | "resolve-from": { 1609 | "version": "2.0.0", 1610 | "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-2.0.0.tgz", 1611 | "integrity": "sha1-lICrIOlP+h2egKgEx+oUdhGWa1c=" 1612 | }, 1613 | "responselike": { 1614 | "version": "1.0.2", 1615 | "resolved": "https://registry.npmjs.org/responselike/-/responselike-1.0.2.tgz", 1616 | "integrity": "sha1-kYcg7ztjHFZCvgaPFa3lpG9Loec=", 1617 | "dev": true, 1618 | "requires": { 1619 | "lowercase-keys": "^1.0.0" 1620 | } 1621 | }, 1622 | "rxjs": { 1623 | "version": "6.6.3", 1624 | "resolved": "https://registry.npmjs.org/rxjs/-/rxjs-6.6.3.tgz", 1625 | "integrity": "sha512-trsQc+xYYXZ3urjOiJOuCOa5N3jAZ3eiSpQB5hIT8zGlL2QfnHLJ2r7GMkBGuIausdJN1OneaI6gQlsqNHHmZQ==", 1626 | "dev": true, 1627 | "requires": { 1628 | "tslib": "^1.9.0" 1629 | } 1630 | }, 1631 | "safe-buffer": { 1632 | "version": "5.1.2", 1633 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", 1634 | "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==" 1635 | }, 1636 | "safer-buffer": { 1637 | "version": "2.1.2", 1638 | "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", 1639 | "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==" 1640 | }, 1641 | "saslprep": { 1642 | "version": "1.0.3", 1643 | "resolved": "https://registry.npmjs.org/saslprep/-/saslprep-1.0.3.tgz", 1644 | "integrity": "sha512-/MY/PEMbk2SuY5sScONwhUDsV2p77Znkb/q3nSVstq/yQzYJOH/Azh29p9oJLsl3LnQwSvZDKagDGBsBwSooag==", 1645 | "optional": true, 1646 | "requires": { 1647 | "sparse-bitfield": "^3.0.3" 1648 | } 1649 | }, 1650 | "semver": { 1651 | "version": "5.7.1", 1652 | "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", 1653 | "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==" 1654 | }, 1655 | "semver-diff": { 1656 | "version": "3.1.1", 1657 | "resolved": "https://registry.npmjs.org/semver-diff/-/semver-diff-3.1.1.tgz", 1658 | "integrity": "sha512-GX0Ix/CJcHyB8c4ykpHGIAvLyOwOobtM/8d+TQkAd81/bEjgPHrfba41Vpesr7jX/t8Uh+R3EX9eAS5be+jQYg==", 1659 | "dev": true, 1660 | "requires": { 1661 | "semver": "^6.3.0" 1662 | }, 1663 | "dependencies": { 1664 | "semver": { 1665 | "version": "6.3.0", 1666 | "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", 1667 | "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", 1668 | "dev": true 1669 | } 1670 | } 1671 | }, 1672 | "send": { 1673 | "version": "0.17.1", 1674 | "resolved": "https://registry.npmjs.org/send/-/send-0.17.1.tgz", 1675 | "integrity": "sha512-BsVKsiGcQMFwT8UxypobUKyv7irCNRHk1T0G680vk88yf6LBByGcZJOTJCrTP2xVN6yI+XjPJcNuE3V4fT9sAg==", 1676 | "requires": { 1677 | "debug": "2.6.9", 1678 | "depd": "~1.1.2", 1679 | "destroy": "~1.0.4", 1680 | "encodeurl": "~1.0.2", 1681 | "escape-html": "~1.0.3", 1682 | "etag": "~1.8.1", 1683 | "fresh": "0.5.2", 1684 | "http-errors": "~1.7.2", 1685 | "mime": "1.6.0", 1686 | "ms": "2.1.1", 1687 | "on-finished": "~2.3.0", 1688 | "range-parser": "~1.2.1", 1689 | "statuses": "~1.5.0" 1690 | }, 1691 | "dependencies": { 1692 | "ms": { 1693 | "version": "2.1.1", 1694 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.1.tgz", 1695 | "integrity": "sha512-tgp+dl5cGk28utYktBsrFqA7HKgrhgPsg6Z/EfhWI4gl1Hwq8B/GmY/0oXZ6nF8hDVesS/FpnYaD/kOWhYQvyg==" 1696 | } 1697 | } 1698 | }, 1699 | "serve-static": { 1700 | "version": "1.14.1", 1701 | "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.14.1.tgz", 1702 | "integrity": "sha512-JMrvUwE54emCYWlTI+hGrGv5I8dEwmco/00EvkzIIsR7MqrHonbD9pO2MOfFnpFntl7ecpZs+3mW+XbQZu9QCg==", 1703 | "requires": { 1704 | "encodeurl": "~1.0.2", 1705 | "escape-html": "~1.0.3", 1706 | "parseurl": "~1.3.3", 1707 | "send": "0.17.1" 1708 | } 1709 | }, 1710 | "set-blocking": { 1711 | "version": "2.0.0", 1712 | "resolved": "https://registry.npmjs.org/set-blocking/-/set-blocking-2.0.0.tgz", 1713 | "integrity": "sha1-BF+XgtARrppoA93TgrJDkrPYkPc=", 1714 | "dev": true 1715 | }, 1716 | "setprototypeof": { 1717 | "version": "1.1.1", 1718 | "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.1.tgz", 1719 | "integrity": "sha512-JvdAWfbXeIGaZ9cILp38HntZSFSo3mWg6xGcJJsd+d4aRMOqauag1C63dJfDw7OaMYwEbHMOxEZ1lqVRYP2OAw==" 1720 | }, 1721 | "sift": { 1722 | "version": "7.0.1", 1723 | "resolved": "https://registry.npmjs.org/sift/-/sift-7.0.1.tgz", 1724 | "integrity": "sha512-oqD7PMJ+uO6jV9EQCl0LrRw1OwsiPsiFQR5AR30heR+4Dl7jBBbDLnNvWiak20tzZlSE1H7RB30SX/1j/YYT7g==" 1725 | }, 1726 | "signal-exit": { 1727 | "version": "3.0.3", 1728 | "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.3.tgz", 1729 | "integrity": "sha512-VUJ49FC8U1OxwZLxIbTTrDvLnf/6TDgxZcK8wxR8zs13xpx7xbG60ndBlhNrFi2EMuFRoeDoJO7wthSLq42EjA==", 1730 | "dev": true 1731 | }, 1732 | "sliced": { 1733 | "version": "1.0.1", 1734 | "resolved": "https://registry.npmjs.org/sliced/-/sliced-1.0.1.tgz", 1735 | "integrity": "sha1-CzpmK10Ewxd7GSa+qCsD+Dei70E=" 1736 | }, 1737 | "sparse-bitfield": { 1738 | "version": "3.0.3", 1739 | "resolved": "https://registry.npmjs.org/sparse-bitfield/-/sparse-bitfield-3.0.3.tgz", 1740 | "integrity": "sha1-/0rm5oZWBWuks+eSqzM004JzyhE=", 1741 | "optional": true, 1742 | "requires": { 1743 | "memory-pager": "^1.0.2" 1744 | } 1745 | }, 1746 | "spawn-command": { 1747 | "version": "0.0.2-1", 1748 | "resolved": "https://registry.npmjs.org/spawn-command/-/spawn-command-0.0.2-1.tgz", 1749 | "integrity": "sha1-YvXpRmmBwbeW3Fkpk34RycaSG9A=", 1750 | "dev": true 1751 | }, 1752 | "spdx-correct": { 1753 | "version": "3.1.1", 1754 | "resolved": "https://registry.npmjs.org/spdx-correct/-/spdx-correct-3.1.1.tgz", 1755 | "integrity": "sha512-cOYcUWwhCuHCXi49RhFRCyJEK3iPj1Ziz9DpViV3tbZOwXD49QzIN3MpOLJNxh2qwq2lJJZaKMVw9qNi4jTC0w==", 1756 | "dev": true, 1757 | "requires": { 1758 | "spdx-expression-parse": "^3.0.0", 1759 | "spdx-license-ids": "^3.0.0" 1760 | } 1761 | }, 1762 | "spdx-exceptions": { 1763 | "version": "2.3.0", 1764 | "resolved": "https://registry.npmjs.org/spdx-exceptions/-/spdx-exceptions-2.3.0.tgz", 1765 | "integrity": "sha512-/tTrYOC7PPI1nUAgx34hUpqXuyJG+DTHJTnIULG4rDygi4xu/tfgmq1e1cIRwRzwZgo4NLySi+ricLkZkw4i5A==", 1766 | "dev": true 1767 | }, 1768 | "spdx-expression-parse": { 1769 | "version": "3.0.1", 1770 | "resolved": "https://registry.npmjs.org/spdx-expression-parse/-/spdx-expression-parse-3.0.1.tgz", 1771 | "integrity": "sha512-cbqHunsQWnJNE6KhVSMsMeH5H/L9EpymbzqTQ3uLwNCLZ1Q481oWaofqH7nO6V07xlXwY6PhQdQ2IedWx/ZK4Q==", 1772 | "dev": true, 1773 | "requires": { 1774 | "spdx-exceptions": "^2.1.0", 1775 | "spdx-license-ids": "^3.0.0" 1776 | } 1777 | }, 1778 | "spdx-license-ids": { 1779 | "version": "3.0.6", 1780 | "resolved": "https://registry.npmjs.org/spdx-license-ids/-/spdx-license-ids-3.0.6.tgz", 1781 | "integrity": "sha512-+orQK83kyMva3WyPf59k1+Y525csj5JejicWut55zeTWANuN17qSiSLUXWtzHeNWORSvT7GLDJ/E/XiIWoXBTw==", 1782 | "dev": true 1783 | }, 1784 | "statuses": { 1785 | "version": "1.5.0", 1786 | "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.5.0.tgz", 1787 | "integrity": "sha1-Fhx9rBd2Wf2YEfQ3cfqZOBR4Yow=" 1788 | }, 1789 | "string-width": { 1790 | "version": "3.1.0", 1791 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz", 1792 | "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==", 1793 | "dev": true, 1794 | "requires": { 1795 | "emoji-regex": "^7.0.1", 1796 | "is-fullwidth-code-point": "^2.0.0", 1797 | "strip-ansi": "^5.1.0" 1798 | } 1799 | }, 1800 | "string_decoder": { 1801 | "version": "1.1.1", 1802 | "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz", 1803 | "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==", 1804 | "requires": { 1805 | "safe-buffer": "~5.1.0" 1806 | } 1807 | }, 1808 | "strip-ansi": { 1809 | "version": "5.2.0", 1810 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", 1811 | "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", 1812 | "dev": true, 1813 | "requires": { 1814 | "ansi-regex": "^4.1.0" 1815 | } 1816 | }, 1817 | "strip-json-comments": { 1818 | "version": "2.0.1", 1819 | "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-2.0.1.tgz", 1820 | "integrity": "sha1-PFMZQukIwml8DsNEhYwobHygpgo=", 1821 | "dev": true 1822 | }, 1823 | "supports-color": { 1824 | "version": "6.1.0", 1825 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-6.1.0.tgz", 1826 | "integrity": "sha512-qe1jfm1Mg7Nq/NSh6XE24gPXROEVsWHxC1LIx//XNlD9iw7YZQGjZNjYN7xGaEG6iKdA8EtNFW6R0gjnVXp+wQ==", 1827 | "dev": true, 1828 | "requires": { 1829 | "has-flag": "^3.0.0" 1830 | } 1831 | }, 1832 | "term-size": { 1833 | "version": "2.2.1", 1834 | "resolved": "https://registry.npmjs.org/term-size/-/term-size-2.2.1.tgz", 1835 | "integrity": "sha512-wK0Ri4fOGjv/XPy8SBHZChl8CM7uMc5VML7SqiQ0zG7+J5Vr+RMQDoHa2CNT6KHUnTGIXH34UDMkPzAUyapBZg==", 1836 | "dev": true 1837 | }, 1838 | "to-readable-stream": { 1839 | "version": "1.0.0", 1840 | "resolved": "https://registry.npmjs.org/to-readable-stream/-/to-readable-stream-1.0.0.tgz", 1841 | "integrity": "sha512-Iq25XBt6zD5npPhlLVXGFN3/gyR2/qODcKNNyTMd4vbm39HUaOiAM4PMq0eMVC/Tkxz+Zjdsc55g9yyz+Yq00Q==", 1842 | "dev": true 1843 | }, 1844 | "to-regex-range": { 1845 | "version": "5.0.1", 1846 | "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", 1847 | "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", 1848 | "dev": true, 1849 | "requires": { 1850 | "is-number": "^7.0.0" 1851 | } 1852 | }, 1853 | "toidentifier": { 1854 | "version": "1.0.0", 1855 | "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.0.tgz", 1856 | "integrity": "sha512-yaOH/Pk/VEhBWWTlhI+qXxDFXlejDGcQipMlyxda9nthulaxLZUNcUqFxokp0vcYnvteJln5FNQDRrxj3YcbVw==" 1857 | }, 1858 | "touch": { 1859 | "version": "3.1.0", 1860 | "resolved": "https://registry.npmjs.org/touch/-/touch-3.1.0.tgz", 1861 | "integrity": "sha512-WBx8Uy5TLtOSRtIq+M03/sKDrXCLHxwDcquSP2c43Le03/9serjQBIztjRz6FkJez9D/hleyAXTBGLwwZUw9lA==", 1862 | "dev": true, 1863 | "requires": { 1864 | "nopt": "~1.0.10" 1865 | } 1866 | }, 1867 | "tree-kill": { 1868 | "version": "1.2.2", 1869 | "resolved": "https://registry.npmjs.org/tree-kill/-/tree-kill-1.2.2.tgz", 1870 | "integrity": "sha512-L0Orpi8qGpRG//Nd+H90vFB+3iHnue1zSSGmNOOCh1GLJ7rUKVwV2HvijphGQS2UmhUZewS9VgvxYIdgr+fG1A==", 1871 | "dev": true 1872 | }, 1873 | "tslib": { 1874 | "version": "1.14.1", 1875 | "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.14.1.tgz", 1876 | "integrity": "sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg==", 1877 | "dev": true 1878 | }, 1879 | "type-fest": { 1880 | "version": "0.8.1", 1881 | "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.8.1.tgz", 1882 | "integrity": "sha512-4dbzIzqvjtgiM5rw1k5rEHtBANKmdudhGyBEajN01fEyhaAIhsoKNy6y7+IN93IfpFtwY9iqi7kD+xwKhQsNJA==", 1883 | "dev": true 1884 | }, 1885 | "type-is": { 1886 | "version": "1.6.18", 1887 | "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz", 1888 | "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==", 1889 | "requires": { 1890 | "media-typer": "0.3.0", 1891 | "mime-types": "~2.1.24" 1892 | } 1893 | }, 1894 | "typedarray-to-buffer": { 1895 | "version": "3.1.5", 1896 | "resolved": "https://registry.npmjs.org/typedarray-to-buffer/-/typedarray-to-buffer-3.1.5.tgz", 1897 | "integrity": "sha512-zdu8XMNEDepKKR+XYOXAVPtWui0ly0NtohUscw+UmaHiAWT8hrV1rr//H6V+0DvJ3OQ19S979M0laLfX8rm82Q==", 1898 | "dev": true, 1899 | "requires": { 1900 | "is-typedarray": "^1.0.0" 1901 | } 1902 | }, 1903 | "undefsafe": { 1904 | "version": "2.0.3", 1905 | "resolved": "https://registry.npmjs.org/undefsafe/-/undefsafe-2.0.3.tgz", 1906 | "integrity": "sha512-nrXZwwXrD/T/JXeygJqdCO6NZZ1L66HrxM/Z7mIq2oPanoN0F1nLx3lwJMu6AwJY69hdixaFQOuoYsMjE5/C2A==", 1907 | "dev": true, 1908 | "requires": { 1909 | "debug": "^2.2.0" 1910 | } 1911 | }, 1912 | "unique-string": { 1913 | "version": "2.0.0", 1914 | "resolved": "https://registry.npmjs.org/unique-string/-/unique-string-2.0.0.tgz", 1915 | "integrity": "sha512-uNaeirEPvpZWSgzwsPGtU2zVSTrn/8L5q/IexZmH0eH6SA73CmAA5U4GwORTxQAZs95TAXLNqeLoPPNO5gZfWg==", 1916 | "dev": true, 1917 | "requires": { 1918 | "crypto-random-string": "^2.0.0" 1919 | } 1920 | }, 1921 | "unpipe": { 1922 | "version": "1.0.0", 1923 | "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", 1924 | "integrity": "sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw=" 1925 | }, 1926 | "update-notifier": { 1927 | "version": "4.1.3", 1928 | "resolved": "https://registry.npmjs.org/update-notifier/-/update-notifier-4.1.3.tgz", 1929 | "integrity": "sha512-Yld6Z0RyCYGB6ckIjffGOSOmHXj1gMeE7aROz4MG+XMkmixBX4jUngrGXNYz7wPKBmtoD4MnBa2Anu7RSKht/A==", 1930 | "dev": true, 1931 | "requires": { 1932 | "boxen": "^4.2.0", 1933 | "chalk": "^3.0.0", 1934 | "configstore": "^5.0.1", 1935 | "has-yarn": "^2.1.0", 1936 | "import-lazy": "^2.1.0", 1937 | "is-ci": "^2.0.0", 1938 | "is-installed-globally": "^0.3.1", 1939 | "is-npm": "^4.0.0", 1940 | "is-yarn-global": "^0.3.0", 1941 | "latest-version": "^5.0.0", 1942 | "pupa": "^2.0.1", 1943 | "semver-diff": "^3.1.1", 1944 | "xdg-basedir": "^4.0.0" 1945 | }, 1946 | "dependencies": { 1947 | "ansi-styles": { 1948 | "version": "4.3.0", 1949 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", 1950 | "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", 1951 | "dev": true, 1952 | "requires": { 1953 | "color-convert": "^2.0.1" 1954 | } 1955 | }, 1956 | "chalk": { 1957 | "version": "3.0.0", 1958 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-3.0.0.tgz", 1959 | "integrity": "sha512-4D3B6Wf41KOYRFdszmDqMCGq5VV/uMAB273JILmO+3jAlh8X4qDtdtgCR3fxtbLEMzSx22QdhnDcJvu2u1fVwg==", 1960 | "dev": true, 1961 | "requires": { 1962 | "ansi-styles": "^4.1.0", 1963 | "supports-color": "^7.1.0" 1964 | } 1965 | }, 1966 | "color-convert": { 1967 | "version": "2.0.1", 1968 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", 1969 | "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", 1970 | "dev": true, 1971 | "requires": { 1972 | "color-name": "~1.1.4" 1973 | } 1974 | }, 1975 | "color-name": { 1976 | "version": "1.1.4", 1977 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", 1978 | "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", 1979 | "dev": true 1980 | }, 1981 | "has-flag": { 1982 | "version": "4.0.0", 1983 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", 1984 | "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", 1985 | "dev": true 1986 | }, 1987 | "supports-color": { 1988 | "version": "7.2.0", 1989 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", 1990 | "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", 1991 | "dev": true, 1992 | "requires": { 1993 | "has-flag": "^4.0.0" 1994 | } 1995 | } 1996 | } 1997 | }, 1998 | "url-parse-lax": { 1999 | "version": "3.0.0", 2000 | "resolved": "https://registry.npmjs.org/url-parse-lax/-/url-parse-lax-3.0.0.tgz", 2001 | "integrity": "sha1-FrXK/Afb42dsGxmZF3gj1lA6yww=", 2002 | "dev": true, 2003 | "requires": { 2004 | "prepend-http": "^2.0.0" 2005 | } 2006 | }, 2007 | "util-deprecate": { 2008 | "version": "1.0.2", 2009 | "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", 2010 | "integrity": "sha1-RQ1Nyfpw3nMnYvvS1KKJgUGaDM8=" 2011 | }, 2012 | "utils-merge": { 2013 | "version": "1.0.1", 2014 | "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz", 2015 | "integrity": "sha1-n5VxD1CiZ5R7LMwSR0HBAoQn5xM=" 2016 | }, 2017 | "validate-npm-package-license": { 2018 | "version": "3.0.4", 2019 | "resolved": "https://registry.npmjs.org/validate-npm-package-license/-/validate-npm-package-license-3.0.4.tgz", 2020 | "integrity": "sha512-DpKm2Ui/xN7/HQKCtpZxoRWBhZ9Z0kqtygG8XCgNQ8ZlDnxuQmWhj566j8fN4Cu3/JmbhsDo7fcAJq4s9h27Ew==", 2021 | "dev": true, 2022 | "requires": { 2023 | "spdx-correct": "^3.0.0", 2024 | "spdx-expression-parse": "^3.0.0" 2025 | } 2026 | }, 2027 | "validator": { 2028 | "version": "13.1.17", 2029 | "resolved": "https://registry.npmjs.org/validator/-/validator-13.1.17.tgz", 2030 | "integrity": "sha512-zL5QBoemJ3jYFb2/j38y7ljhwYGXVLUp8H6W1nVxadnAOvUOytec+L7BHh1oBQ82/TzWXHd+GSaxUWp4lROkLg==" 2031 | }, 2032 | "vary": { 2033 | "version": "1.1.2", 2034 | "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz", 2035 | "integrity": "sha1-IpnwLG3tMNSllhsLn3RSShj2NPw=" 2036 | }, 2037 | "which-module": { 2038 | "version": "2.0.0", 2039 | "resolved": "https://registry.npmjs.org/which-module/-/which-module-2.0.0.tgz", 2040 | "integrity": "sha1-2e8H3Od7mQK4o6j6SzHD4/fm6Ho=", 2041 | "dev": true 2042 | }, 2043 | "widest-line": { 2044 | "version": "3.1.0", 2045 | "resolved": "https://registry.npmjs.org/widest-line/-/widest-line-3.1.0.tgz", 2046 | "integrity": "sha512-NsmoXalsWVDMGupxZ5R08ka9flZjjiLvHVAWYOKtiKM8ujtZWr9cRffak+uSE48+Ob8ObalXpwyeUiyDD6QFgg==", 2047 | "dev": true, 2048 | "requires": { 2049 | "string-width": "^4.0.0" 2050 | }, 2051 | "dependencies": { 2052 | "ansi-regex": { 2053 | "version": "5.0.0", 2054 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.0.tgz", 2055 | "integrity": "sha512-bY6fj56OUQ0hU1KjFNDQuJFezqKdrAyFdIevADiqrWHwSlbmBNMHp5ak2f40Pm8JTFyM2mqxkG6ngkHO11f/lg==", 2056 | "dev": true 2057 | }, 2058 | "emoji-regex": { 2059 | "version": "8.0.0", 2060 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", 2061 | "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", 2062 | "dev": true 2063 | }, 2064 | "is-fullwidth-code-point": { 2065 | "version": "3.0.0", 2066 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", 2067 | "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", 2068 | "dev": true 2069 | }, 2070 | "string-width": { 2071 | "version": "4.2.0", 2072 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.0.tgz", 2073 | "integrity": "sha512-zUz5JD+tgqtuDjMhwIg5uFVV3dtqZ9yQJlZVfq4I01/K5Paj5UHj7VyrQOJvzawSVlKpObApbfD0Ed6yJc+1eg==", 2074 | "dev": true, 2075 | "requires": { 2076 | "emoji-regex": "^8.0.0", 2077 | "is-fullwidth-code-point": "^3.0.0", 2078 | "strip-ansi": "^6.0.0" 2079 | } 2080 | }, 2081 | "strip-ansi": { 2082 | "version": "6.0.0", 2083 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.0.tgz", 2084 | "integrity": "sha512-AuvKTrTfQNYNIctbR1K/YGTR1756GycPsg7b9bdV9Duqur4gv6aKqHXah67Z8ImS7WEz5QVcOtlfW2rZEugt6w==", 2085 | "dev": true, 2086 | "requires": { 2087 | "ansi-regex": "^5.0.0" 2088 | } 2089 | } 2090 | } 2091 | }, 2092 | "wrap-ansi": { 2093 | "version": "5.1.0", 2094 | "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-5.1.0.tgz", 2095 | "integrity": "sha512-QC1/iN/2/RPVJ5jYK8BGttj5z83LmSKmvbvrXPNCLZSEb32KKVDJDl/MOt2N01qU2H/FkzEa9PKto1BqDjtd7Q==", 2096 | "dev": true, 2097 | "requires": { 2098 | "ansi-styles": "^3.2.0", 2099 | "string-width": "^3.0.0", 2100 | "strip-ansi": "^5.0.0" 2101 | } 2102 | }, 2103 | "wrappy": { 2104 | "version": "1.0.2", 2105 | "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", 2106 | "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=", 2107 | "dev": true 2108 | }, 2109 | "write-file-atomic": { 2110 | "version": "3.0.3", 2111 | "resolved": "https://registry.npmjs.org/write-file-atomic/-/write-file-atomic-3.0.3.tgz", 2112 | "integrity": "sha512-AvHcyZ5JnSfq3ioSyjrBkH9yW4m7Ayk8/9My/DD9onKeu/94fwrMocemO2QAJFAlnnDN+ZDS+ZjAR5ua1/PV/Q==", 2113 | "dev": true, 2114 | "requires": { 2115 | "imurmurhash": "^0.1.4", 2116 | "is-typedarray": "^1.0.0", 2117 | "signal-exit": "^3.0.2", 2118 | "typedarray-to-buffer": "^3.1.5" 2119 | } 2120 | }, 2121 | "xdg-basedir": { 2122 | "version": "4.0.0", 2123 | "resolved": "https://registry.npmjs.org/xdg-basedir/-/xdg-basedir-4.0.0.tgz", 2124 | "integrity": "sha512-PSNhEJDejZYV7h50BohL09Er9VaIefr2LMAf3OEmpCkjOi34eYyQYAXUTjEQtZJTKcF0E2UKTh+osDLsgNim9Q==", 2125 | "dev": true 2126 | }, 2127 | "y18n": { 2128 | "version": "4.0.0", 2129 | "resolved": "https://registry.npmjs.org/y18n/-/y18n-4.0.0.tgz", 2130 | "integrity": "sha512-r9S/ZyXu/Xu9q1tYlpsLIsa3EeLXXk0VwlxqTcFRfg9EhMW+17kbt9G0NrgCmhGb5vT2hyhJZLfDGx+7+5Uj/w==", 2131 | "dev": true 2132 | }, 2133 | "yargs": { 2134 | "version": "13.3.2", 2135 | "resolved": "https://registry.npmjs.org/yargs/-/yargs-13.3.2.tgz", 2136 | "integrity": "sha512-AX3Zw5iPruN5ie6xGRIDgqkT+ZhnRlZMLMHAs8tg7nRruy2Nb+i5o9bwghAogtM08q1dpr2LVoS8KSTMYpWXUw==", 2137 | "dev": true, 2138 | "requires": { 2139 | "cliui": "^5.0.0", 2140 | "find-up": "^3.0.0", 2141 | "get-caller-file": "^2.0.1", 2142 | "require-directory": "^2.1.1", 2143 | "require-main-filename": "^2.0.0", 2144 | "set-blocking": "^2.0.0", 2145 | "string-width": "^3.0.0", 2146 | "which-module": "^2.0.0", 2147 | "y18n": "^4.0.0", 2148 | "yargs-parser": "^13.1.2" 2149 | } 2150 | }, 2151 | "yargs-parser": { 2152 | "version": "13.1.2", 2153 | "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-13.1.2.tgz", 2154 | "integrity": "sha512-3lbsNRf/j+A4QuSZfDRA7HRSfWrzO0YjqTJd5kjAq37Zep1CEgaYmrH9Q3GwPiB9cHyd1Y1UwggGhJGoxipbzg==", 2155 | "dev": true, 2156 | "requires": { 2157 | "camelcase": "^5.0.0", 2158 | "decamelize": "^1.2.0" 2159 | } 2160 | } 2161 | } 2162 | } 2163 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "memory-game", 3 | "version": "1.0.0", 4 | "description": "memory game", 5 | "main": "server.js", 6 | "scripts": { 7 | "start": "node server.js", 8 | "server": "nodemon server.js", 9 | "client": "npm start --prefix client", 10 | "clientinstall": "npm install --prefix client", 11 | "dev": "concurrently \"npm run server\" \"npm run client\"", 12 | "heroku-postbuild": "NPM_CONFIG_PRODUCTION=false npm install --prefix client && npm run build --prefix client" 13 | }, 14 | "keywords": [], 15 | "author": "", 16 | "license": "ISC", 17 | "dependencies": { 18 | "bcryptjs": "^2.4.3", 19 | "config": "^3.3.2", 20 | "express": "^4.17.1", 21 | "express-validator": "^6.6.1", 22 | "jsonwebtoken": "^8.5.1", 23 | "mongoose": "^5.10.15" 24 | }, 25 | "devDependencies": { 26 | "concurrently": "^5.3.0", 27 | "nodemon": "^2.0.6" 28 | } 29 | } 30 | -------------------------------------------------------------------------------- /routes/auth.js: -------------------------------------------------------------------------------- 1 | const express = require('express'); 2 | const router = express.Router(); 3 | const bcrypt = require('bcryptjs'); 4 | const jwt = require('jsonwebtoken'); 5 | const config = require('config'); 6 | const auth = require('../middleware/auth'); 7 | const { check, validationResult } = require('express-validator'); 8 | 9 | const User = require('../models/Users'); 10 | 11 | // @route GET api/auth 12 | // @desc Get logged in user 13 | // @access Private 14 | router.get('/', auth, async (req, res) => { 15 | // declare user variable from database, except password 16 | try { 17 | const user = await User.findById(req.user.id).select('-password'); 18 | res.json(user); 19 | // return error if found 20 | } catch (err) { 21 | console.error(err.message); 22 | res.status(500).send('Server Error'); 23 | } 24 | }); 25 | 26 | // @route POST api/auth 27 | // @desc Authorize user and get token 28 | // @access Public 29 | router.post( 30 | '/', 31 | // check email and password are valid 32 | [ 33 | check('email', 'please enter valid email').isEmail(), 34 | check('password', 'password is required').exists(), 35 | ], 36 | async (req, res) => { 37 | // declare errors array 38 | const errors = validationResult(req); 39 | // if errors found, return message 40 | if (!errors.isEmpty()) { 41 | return res.status(400).json({ errors: errors.array() }); 42 | } 43 | 44 | // declare variables from request 45 | const { email, password } = req.body; 46 | 47 | try { 48 | // find user in database, return err msg if not found 49 | let user = await User.findOne({ email }); 50 | if (!user) { 51 | return res.status(400).json({ msg: 'invalid credentials' }); 52 | } 53 | 54 | // check passwords match, return err msg if not 55 | const isMatch = await bcrypt.compare(password, user.password); 56 | if (!isMatch) { 57 | return res.status(400).json({ msg: 'invalid credentials' }); 58 | } 59 | 60 | // declare variable with user info 61 | const payload = { 62 | user: { 63 | id: user.id, 64 | }, 65 | }; 66 | 67 | // sign user in, sign ou after 1 hour 68 | jwt.sign( 69 | payload, 70 | config.get('jwtSecret'), 71 | { 72 | expiresIn: 3600, 73 | }, 74 | (err, token) => { 75 | if (err) throw err; 76 | res.json({ token }); 77 | } 78 | ); 79 | // return err msg if found 80 | } catch (err) { 81 | console.error(err.message); 82 | res.status(500).send('Server Error'); 83 | } 84 | } 85 | ); 86 | 87 | module.exports = router; 88 | -------------------------------------------------------------------------------- /routes/history.js: -------------------------------------------------------------------------------- 1 | const express = require('express'); 2 | const router = express.Router(); 3 | const auth = require('../middleware/auth'); 4 | 5 | const Game = require('../models/Games'); 6 | 7 | // @route GET api/history 8 | // @desc Retrieve user history 9 | // @access Private 10 | router.get('/', auth, async (req, res) => { 11 | // declare games variable from database and json variable 12 | try { 13 | const games = await Game.find({ user: req.user.id }); 14 | res.json(games); 15 | // return error if found 16 | } catch (err) { 17 | console.error(err.message); 18 | res.status(500).send('Server Error'); 19 | } 20 | }); 21 | 22 | // @route POST api/history 23 | // @desc Add user history 24 | // @access Private 25 | router.post('/', auth, async (req, res) => { 26 | // declare variables from request 27 | const { gameLevel, numOfMoves } = req.body; 28 | 29 | // declare newGame model with request info 30 | try { 31 | const newGame = new Game({ 32 | gameLevel: gameLevel, 33 | numOfMoves: numOfMoves, 34 | user: req.user.id, 35 | }); 36 | 37 | // save new game to database and return json 38 | const game = await newGame.save(); 39 | res.json(game); 40 | // return error if found 41 | } catch (err) { 42 | console.error(err.message); 43 | res.status(500).send('Server Error'); 44 | } 45 | }); 46 | 47 | // // @route DELETE api/history 48 | // // @desc Delete user history 49 | // // @access Private 50 | // router.delete('/', auth, async (req, res) => { 51 | // try { 52 | // await Game.deleteMany({ user: req.user.id }); 53 | 54 | // res.json({ msg: 'Games Cleared' }); 55 | 56 | // } catch (err) { 57 | // console.error(err.message); 58 | // res.status(500).send('Server Error'); 59 | // } 60 | // }); 61 | 62 | module.exports = router; 63 | -------------------------------------------------------------------------------- /routes/users.js: -------------------------------------------------------------------------------- 1 | const express = require('express'); 2 | const router = express.Router(); 3 | const bcrypt = require('bcryptjs'); 4 | const jwt = require('jsonwebtoken'); 5 | const config = require('config'); 6 | const { check, validationResult } = require('express-validator'); 7 | 8 | const User = require('../models/Users'); 9 | 10 | // @route POST api/users 11 | // @desc Register a user 12 | // @access Public 13 | router.post( 14 | '/', 15 | // check name, email, and password valid 16 | [ 17 | check('name', 'Please enter a name').not().isEmpty(), 18 | check('email', 'Please enter valid email').isEmail(), 19 | check('password', 'Password must be 1 or more characters').isLength({ 20 | min: 1, 21 | }), 22 | ], 23 | async (req, res) => { 24 | // declare errors array 25 | const errors = validationResult(req); 26 | // if errors, return error message 27 | if (!errors.isEmpty()) { 28 | return res.status(400).json({ errors: errors.array() }); 29 | } 30 | 31 | // declare variables from request 32 | const { name, email, password } = req.body; 33 | 34 | // check if email already exists 35 | try { 36 | let user = await User.findOne({ email }); 37 | 38 | // alert user if email exists 39 | if (user) { 40 | return res.status(400).json({ msg: 'Email already exists' }); 41 | } 42 | 43 | // declare user model with user input 44 | user = User({ 45 | name: name, 46 | email: email, 47 | password: password, 48 | }); 49 | 50 | // declare salt for password 51 | const salt = await bcrypt.genSalt(10); 52 | user.password = await bcrypt.hash(password, salt); 53 | 54 | // save user model to database 55 | await user.save(); 56 | 57 | // declare payload object with user id 58 | const payload = { 59 | user: { 60 | id: user.id, 61 | }, 62 | }; 63 | 64 | // sing user in with user input, sign out after 3600 seconds 65 | jwt.sign( 66 | payload, 67 | config.get('jwtSecret'), 68 | { 69 | expiresIn: 3600, 70 | }, 71 | (err, token) => { 72 | if (err) throw err; 73 | res.json({ token }); 74 | } 75 | ); 76 | } catch (err) { 77 | console.error(err.message); 78 | res.status('500').send('Server Error'); 79 | } 80 | } 81 | ); 82 | 83 | // // @route PUT api/users/:id 84 | // // @desc Update a user 85 | // // @access Private 86 | // router.put('/:id', (req, res) => { 87 | // res.send('update a user') 88 | // }); 89 | 90 | // // @route DELETE api/users/:id 91 | // // @desc Delete a user 92 | // // @access Private 93 | // router.delete('/:id', (req, res) => { 94 | // res.send('delete a user') 95 | // }); 96 | 97 | module.exports = router; 98 | -------------------------------------------------------------------------------- /server.js: -------------------------------------------------------------------------------- 1 | const express = require('express'); 2 | const connectDB = require('./config/db'); 3 | const path = require('path'); 4 | 5 | const app = express(); 6 | 7 | // Connect DB 8 | connectDB(); 9 | 10 | // Initialize middleware 11 | app.use(express.json({ extended: false })); 12 | 13 | // Define Routes 14 | app.use('/api/auth', require('./routes/auth')); 15 | app.use('/api/users', require('./routes/users')); 16 | app.use('/api/history', require('./routes/history')); 17 | 18 | // Serve static assets in production 19 | if (process.env.NODE_ENV === 'production') { 20 | // Set static folder 21 | app.use(express.static('client/build')); 22 | 23 | app.get('*', (req, res) => 24 | res.sendFile(path.resolve(__dirname, 'client', 'build', 'index.html')) 25 | ); 26 | } 27 | 28 | // Declare port 29 | const PORT = process.env.PORT || 5000; 30 | 31 | app.listen(PORT, () => console.log(`Server started on port ${PORT}`)); 32 | --------------------------------------------------------------------------------