├── frontend ├── public │ ├── robots.txt │ ├── favicon.ico │ ├── logo192.png │ ├── logo512.png │ ├── bn_bookstore_logo.png │ ├── manifest.json │ └── index.html ├── src │ ├── components │ │ ├── AppContainer.module.css │ │ ├── SignIn.module.css │ │ ├── SignUp.module.css │ │ ├── AdminOrders.module.css │ │ ├── Bookstore.module.css │ │ ├── SearchBar.module.css │ │ ├── Term.jsx │ │ ├── AdminSignIn.module.css │ │ ├── BookSearchResults.module.css │ │ ├── Contact.module.css │ │ ├── Home.module.css │ │ ├── Order.module.css │ │ ├── Cart.module.css │ │ ├── InvoiceCard.module.css │ │ ├── PageSwitch.module.css │ │ ├── Contact.jsx │ │ ├── InvoiceCard.jsx │ │ ├── SignUpYup.module.css │ │ ├── BookCard.module.css │ │ ├── AdminOrders.jsx │ │ ├── CartItem.module.css │ │ ├── Cart.jsx │ │ ├── BookDetail.module.css │ │ ├── NavBar.module.css │ │ ├── SearchBar.jsx │ │ ├── AppContainer.jsx │ │ ├── BookCard.jsx │ │ ├── BookSearchResults.jsx │ │ ├── Order.jsx │ │ ├── PageSwitch.jsx │ │ ├── BookDetail.jsx │ │ ├── Bookstore.jsx │ │ ├── Home.jsx │ │ ├── CartItem.jsx │ │ ├── NavBar.jsx │ │ ├── SignUpYup.jsx │ │ ├── SignIn.jsx │ │ ├── AdminSignIn.jsx │ │ └── SignUp.jsx │ ├── assets │ │ └── bn_bookstore_logo.png │ ├── setupTests.js │ ├── App.test.js │ ├── app │ │ └── store.js │ ├── reportWebVitals.js │ ├── App.js │ ├── features │ │ └── orderBooks │ │ │ ├── userSlice.js │ │ │ └── orderBooksSlice.js │ ├── index.css │ ├── index.js │ └── logo.svg ├── .gitignore ├── package.json └── README.md ├── backend ├── views │ ├── index.jade │ ├── error.jade │ └── layout.jade ├── public │ └── stylesheets │ │ └── style.css ├── routes │ ├── users.js │ ├── index.js │ ├── usersRoutes.js │ ├── booksRoutes.js │ ├── ordersRouter.js │ └── adminsRoutes.js ├── models │ ├── usersModel.js │ ├── adminsModel.js │ ├── ordersModel.js │ └── booksModel.js ├── package.json ├── controllers │ ├── ordersController.js │ ├── booksController.js │ ├── usersController.js │ └── adminsController.js ├── bin │ └── www ├── app.js └── package-lock.json ├── README.md └── .gitignore /frontend/public/robots.txt: -------------------------------------------------------------------------------- 1 | # https://www.robotstxt.org/robotstxt.html 2 | User-agent: * 3 | Disallow: 4 | -------------------------------------------------------------------------------- /backend/views/index.jade: -------------------------------------------------------------------------------- 1 | extends layout 2 | 3 | block content 4 | h1= title 5 | p Welcome to #{title} 6 | -------------------------------------------------------------------------------- /frontend/src/components/AppContainer.module.css: -------------------------------------------------------------------------------- 1 | .outerContainer { 2 | width: 100%; 3 | margin: auto; 4 | } 5 | -------------------------------------------------------------------------------- /frontend/public/favicon.ico: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/linhtch90/bn_bookstore_public_source/HEAD/frontend/public/favicon.ico -------------------------------------------------------------------------------- /frontend/public/logo192.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/linhtch90/bn_bookstore_public_source/HEAD/frontend/public/logo192.png -------------------------------------------------------------------------------- /frontend/public/logo512.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/linhtch90/bn_bookstore_public_source/HEAD/frontend/public/logo512.png -------------------------------------------------------------------------------- /backend/views/error.jade: -------------------------------------------------------------------------------- 1 | extends layout 2 | 3 | block content 4 | h1= message 5 | h2= error.status 6 | pre #{error.stack} 7 | -------------------------------------------------------------------------------- /frontend/public/bn_bookstore_logo.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/linhtch90/bn_bookstore_public_source/HEAD/frontend/public/bn_bookstore_logo.png -------------------------------------------------------------------------------- /frontend/src/assets/bn_bookstore_logo.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/linhtch90/bn_bookstore_public_source/HEAD/frontend/src/assets/bn_bookstore_logo.png -------------------------------------------------------------------------------- /frontend/src/components/SignIn.module.css: -------------------------------------------------------------------------------- 1 | .pField { 2 | margin-top: 3rem; 3 | } 4 | 5 | .titleText { 6 | font-size: 2rem; 7 | font-weight: 700; 8 | } -------------------------------------------------------------------------------- /frontend/src/components/SignUp.module.css: -------------------------------------------------------------------------------- 1 | .pField { 2 | margin-top: 3rem; 3 | } 4 | 5 | .titleText { 6 | font-size: 2rem; 7 | font-weight: 700; 8 | } -------------------------------------------------------------------------------- /backend/views/layout.jade: -------------------------------------------------------------------------------- 1 | doctype html 2 | html 3 | head 4 | title= title 5 | link(rel='stylesheet', href='/stylesheets/style.css') 6 | body 7 | block content 8 | -------------------------------------------------------------------------------- /backend/public/stylesheets/style.css: -------------------------------------------------------------------------------- 1 | body { 2 | padding: 50px; 3 | font: 14px "Lucida Grande", Helvetica, Arial, sans-serif; 4 | } 5 | 6 | a { 7 | color: #00B7FF; 8 | } 9 | -------------------------------------------------------------------------------- /frontend/src/components/AdminOrders.module.css: -------------------------------------------------------------------------------- 1 | .outerContainer { 2 | margin: auto; 3 | width: 100%; 4 | } 5 | 6 | .innerContainer { 7 | margin: auto; 8 | width: 85%; 9 | } -------------------------------------------------------------------------------- /frontend/src/components/Bookstore.module.css: -------------------------------------------------------------------------------- 1 | .outerContainer { 2 | margin: auto; 3 | width: 100%; 4 | } 5 | 6 | .innerContainer { 7 | margin: auto; 8 | width: 85%; 9 | } -------------------------------------------------------------------------------- /frontend/src/components/SearchBar.module.css: -------------------------------------------------------------------------------- 1 | .outerContainer { 2 | margin: auto; 3 | width: 100%; 4 | } 5 | 6 | .innerContainer { 7 | margin: auto; 8 | width: 70%; 9 | } -------------------------------------------------------------------------------- /frontend/src/components/Term.jsx: -------------------------------------------------------------------------------- 1 | import React from "react"; 2 | 3 | function Term() { 4 | return ( 5 |

Term Component

6 | ); 7 | } 8 | 9 | export default Term; -------------------------------------------------------------------------------- /frontend/src/components/AdminSignIn.module.css: -------------------------------------------------------------------------------- 1 | .pField { 2 | margin-top: 3rem; 3 | } 4 | 5 | .titleText { 6 | font-size: 2rem; 7 | font-weight: 700; 8 | color: #BF616A; 9 | } -------------------------------------------------------------------------------- /frontend/src/components/BookSearchResults.module.css: -------------------------------------------------------------------------------- 1 | .outerContainer { 2 | margin: auto; 3 | width: 100%; 4 | } 5 | 6 | .innerContainer { 7 | margin: auto; 8 | width: 85%; 9 | } -------------------------------------------------------------------------------- /frontend/src/components/Contact.module.css: -------------------------------------------------------------------------------- 1 | .outerContainer { 2 | margin: auto; 3 | width: 100%; 4 | padding-top: 3rem; 5 | } 6 | 7 | .innerContainer { 8 | margin: auto; 9 | width: 85%; 10 | } -------------------------------------------------------------------------------- /frontend/src/components/Home.module.css: -------------------------------------------------------------------------------- 1 | .outerContainer { 2 | margin: auto; 3 | width: 100%; 4 | padding-bottom: 3rem; 5 | } 6 | 7 | .innerContainer { 8 | margin: auto; 9 | width: 85%; 10 | } 11 | 12 | -------------------------------------------------------------------------------- /backend/routes/users.js: -------------------------------------------------------------------------------- 1 | var express = require('express'); 2 | var router = express.Router(); 3 | 4 | /* GET users listing. */ 5 | router.get('/', function(req, res, next) { 6 | res.send('respond with a resource'); 7 | }); 8 | 9 | module.exports = router; 10 | -------------------------------------------------------------------------------- /backend/routes/index.js: -------------------------------------------------------------------------------- 1 | var express = require('express'); 2 | var router = express.Router(); 3 | 4 | /* GET home page. */ 5 | router.get('/', function(req, res, next) { 6 | res.render('index', { title: 'Express' }); 7 | }); 8 | 9 | module.exports = router; 10 | -------------------------------------------------------------------------------- /frontend/src/components/Order.module.css: -------------------------------------------------------------------------------- 1 | .orderContainer { 2 | width: 100%; 3 | padding: 2rem; 4 | font-size: 2rem; 5 | font-weight: 700; 6 | } 7 | 8 | .orderTotal { 9 | margin-bottom: 2rem; 10 | } 11 | 12 | .orderPrice { 13 | color: #BF616A; 14 | } -------------------------------------------------------------------------------- /frontend/src/setupTests.js: -------------------------------------------------------------------------------- 1 | // jest-dom adds custom jest matchers for asserting on DOM nodes. 2 | // allows you to do things like: 3 | // expect(element).toHaveTextContent(/react/i) 4 | // learn more: https://github.com/testing-library/jest-dom 5 | import '@testing-library/jest-dom'; 6 | -------------------------------------------------------------------------------- /frontend/src/App.test.js: -------------------------------------------------------------------------------- 1 | import { render, screen } from '@testing-library/react'; 2 | import App from './App'; 3 | 4 | test('renders learn react link', () => { 5 | render(); 6 | const linkElement = screen.getByText(/learn react/i); 7 | expect(linkElement).toBeInTheDocument(); 8 | }); 9 | -------------------------------------------------------------------------------- /backend/routes/usersRoutes.js: -------------------------------------------------------------------------------- 1 | var express = require("express"); 2 | var router = express.Router(); 3 | var usersController = require("../controllers/usersController"); 4 | 5 | router.post("/signup", usersController.createUser); 6 | 7 | router.post("/signin", usersController.validateUser); 8 | 9 | module.exports = router; -------------------------------------------------------------------------------- /frontend/src/components/Cart.module.css: -------------------------------------------------------------------------------- 1 | .outerContainer { 2 | margin: auto; 3 | width: 100%; 4 | } 5 | 6 | .innerContainer { 7 | margin: auto; 8 | width: 85%; 9 | } 10 | 11 | .cartInformation { 12 | display: flex; 13 | } 14 | 15 | .cartBooks { 16 | width: 70%; 17 | } 18 | 19 | .cartTotal { 20 | width: 30%; 21 | } 22 | 23 | -------------------------------------------------------------------------------- /frontend/src/app/store.js: -------------------------------------------------------------------------------- 1 | import { configureStore } from "@reduxjs/toolkit"; 2 | 3 | import orderBooksReducer from "../features/orderBooks/orderBooksSlice"; 4 | import userReducer from "../features/orderBooks/userSlice"; 5 | 6 | export default configureStore({ 7 | reducer: { 8 | orderBooks: orderBooksReducer, 9 | userManagement: userReducer, 10 | }, 11 | }); 12 | -------------------------------------------------------------------------------- /frontend/.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 | -------------------------------------------------------------------------------- /frontend/src/reportWebVitals.js: -------------------------------------------------------------------------------- 1 | const reportWebVitals = onPerfEntry => { 2 | if (onPerfEntry && onPerfEntry instanceof Function) { 3 | import('web-vitals').then(({ getCLS, getFID, getFCP, getLCP, getTTFB }) => { 4 | getCLS(onPerfEntry); 5 | getFID(onPerfEntry); 6 | getFCP(onPerfEntry); 7 | getLCP(onPerfEntry); 8 | getTTFB(onPerfEntry); 9 | }); 10 | } 11 | }; 12 | 13 | export default reportWebVitals; 14 | -------------------------------------------------------------------------------- /backend/routes/booksRoutes.js: -------------------------------------------------------------------------------- 1 | var express = require('express'); 2 | var router = express.Router(); 3 | var booksController = require('../controllers/booksController'); 4 | 5 | // GET Books page 6 | router.get('/', booksController.all_books); 7 | 8 | // GET a book by ID 9 | router.get('/book_details/:id', booksController.book_details); 10 | 11 | // GET books by searching book's title 12 | router.get('/book_search_title/:search_title', booksController.search_book_title); 13 | 14 | 15 | module.exports = router; 16 | -------------------------------------------------------------------------------- /backend/models/usersModel.js: -------------------------------------------------------------------------------- 1 | var mongoose = require("mongoose"); 2 | 3 | var UsersSchema = mongoose.Schema({ 4 | name: { type: String, required: true, trim: true }, 5 | email: { 6 | type: String, 7 | required: true, 8 | trim: true, 9 | unique: true, 10 | match: [ 11 | /^\w+([\.-]?\w+)*@\w+([\.-]?\w+)*(\.\w{2,3})+$/, 12 | "Please fill a valid email address", 13 | ], 14 | }, 15 | password: { type: String, required: true }, 16 | }); 17 | 18 | module.exports = mongoose.model("user", UsersSchema); 19 | -------------------------------------------------------------------------------- /frontend/src/App.js: -------------------------------------------------------------------------------- 1 | import "primereact/resources/themes/fluent-light/theme.css"; 2 | import "primereact/resources/primereact.min.css"; 3 | import "primeicons/primeicons.css"; 4 | import "primeflex/primeflex.css"; 5 | import PrimeReact from "primereact/api"; 6 | 7 | import AppContainer from "./components/AppContainer"; 8 | 9 | PrimeReact.ripple = true; 10 | PrimeReact.autoZIndex = true; 11 | 12 | function App() { 13 | return ( 14 |
15 | 16 |
17 | ); 18 | } 19 | 20 | export default App; 21 | -------------------------------------------------------------------------------- /backend/models/adminsModel.js: -------------------------------------------------------------------------------- 1 | var mongoose = require("mongoose"); 2 | 3 | var AdminsSchema = mongoose.Schema({ 4 | name: { type: String, required: true, trim: true }, 5 | email: { 6 | type: String, 7 | required: true, 8 | trim: true, 9 | unique: true, 10 | match: [ 11 | /^\w+([\.-]?\w+)*@\w+([\.-]?\w+)*(\.\w{2,3})+$/, 12 | "Please fill a valid email address", 13 | ], 14 | }, 15 | password: { type: String, required: true }, 16 | }); 17 | 18 | module.exports = mongoose.model("admin", AdminsSchema); 19 | -------------------------------------------------------------------------------- /frontend/src/components/InvoiceCard.module.css: -------------------------------------------------------------------------------- 1 | .outerContainer { 2 | margin: auto; 3 | width: 100%; 4 | } 5 | 6 | .innerContainer { 7 | margin: auto; 8 | width: 85%; 9 | } 10 | 11 | .invoiceContainer { 12 | border: solid; 13 | border-radius: 5px; 14 | border-width: 2px; 15 | padding: 1rem; 16 | margin-bottom: 1rem; 17 | } 18 | 19 | .invoiceContainer:nth-child(odd) { 20 | background-color: #A3BE8C; 21 | } 22 | 23 | .invoiceContainer:nth-child(even) { 24 | background-color: #81A1C1; 25 | } 26 | 27 | -------------------------------------------------------------------------------- /frontend/src/features/orderBooks/userSlice.js: -------------------------------------------------------------------------------- 1 | import {createSlice} from "@reduxjs/toolkit"; 2 | 3 | export const userSlice = createSlice({ 4 | name: "userSlice", 5 | initialState: { 6 | isSignIn: false, 7 | }, 8 | reducers: { 9 | userSignIn: (state) => { 10 | state.isSignIn = true; 11 | }, 12 | userSignOut: (state) => { 13 | state.isSignIn = false; 14 | }, 15 | }, 16 | }); 17 | 18 | export const {userSignIn, userSignOut} = userSlice.actions; 19 | 20 | export default userSlice.reducer; -------------------------------------------------------------------------------- /frontend/src/components/PageSwitch.module.css: -------------------------------------------------------------------------------- 1 | .outerContainer { 2 | margin: auto; 3 | width: 100%; 4 | padding-top: 2rem; 5 | padding-bottom: 2rem; 6 | } 7 | 8 | .innerContainer { 9 | margin: auto; 10 | width: 70%; 11 | } 12 | 13 | .currentPage { 14 | width: 60px; 15 | text-align: center; 16 | } 17 | 18 | .currentPageGroup { 19 | margin-left: 4px; 20 | margin-right: 4px; 21 | } 22 | 23 | .pageSwitchContainer { 24 | display: flex; 25 | justify-content: center; 26 | } 27 | 28 | .switchButton { 29 | text-decoration: none; 30 | } -------------------------------------------------------------------------------- /backend/models/ordersModel.js: -------------------------------------------------------------------------------- 1 | var mongoose = require("mongoose"); 2 | var Users = require("../models/usersModel"); 3 | 4 | const Book = mongoose.Schema({ 5 | bookID: { type: String }, 6 | title: { type: String }, 7 | price: { type: Number }, 8 | orderQuantity: { type: Number }, 9 | }); 10 | 11 | var OrdersSchema = mongoose.Schema( 12 | { 13 | userID: { type: mongoose.ObjectId, ref: "user" }, 14 | orderBooks: { type: [Book] }, 15 | totalMoney: { type: Number }, 16 | }, 17 | { timestamps: true } 18 | ); 19 | 20 | module.exports = mongoose.model("order", OrdersSchema); 21 | -------------------------------------------------------------------------------- /frontend/public/manifest.json: -------------------------------------------------------------------------------- 1 | { 2 | "short_name": "React App", 3 | "name": "Create React App Sample", 4 | "icons": [ 5 | { 6 | "src": "favicon.ico", 7 | "sizes": "64x64 32x32 24x24 16x16", 8 | "type": "image/x-icon" 9 | }, 10 | { 11 | "src": "logo192.png", 12 | "type": "image/png", 13 | "sizes": "192x192" 14 | }, 15 | { 16 | "src": "logo512.png", 17 | "type": "image/png", 18 | "sizes": "512x512" 19 | } 20 | ], 21 | "start_url": ".", 22 | "display": "standalone", 23 | "theme_color": "#000000", 24 | "background_color": "#ffffff" 25 | } 26 | -------------------------------------------------------------------------------- /backend/models/booksModel.js: -------------------------------------------------------------------------------- 1 | var mongoose = require("mongoose"); 2 | 3 | var BooksSchema = mongoose.Schema({ 4 | title: { type: String, required: true }, 5 | author: { type: String, required: true }, 6 | price: { type: Number, required: true }, 7 | image_src: { type: String, required: true }, 8 | book_detail_src: { type: String, required: false }, 9 | isbn_13: { type: String, required: true }, 10 | publisher: { type: String, required: true }, 11 | publish_date: { type: String, required: true }, 12 | pages: { type: Number, required: true }, 13 | }); 14 | 15 | module.exports = mongoose.model("book", BooksSchema); 16 | -------------------------------------------------------------------------------- /frontend/src/index.css: -------------------------------------------------------------------------------- 1 | @import url('https://fonts.googleapis.com/css2?family=Open+Sans:ital,wght@0,300;0,400;0,600;0,700;0,800;1,300;1,400;1,600;1,700;1,800&display=swap'); 2 | 3 | html { 4 | font-size: 10pt; 5 | } 6 | 7 | body { 8 | margin: 0; 9 | font-family: 'Open Sans', -apple-system, BlinkMacSystemFont, 'Segoe UI', 'Roboto', 'Oxygen', 10 | 'Ubuntu', 'Cantarell', 'Fira Sans', 'Droid Sans', 'Helvetica Neue', 11 | sans-serif; 12 | -webkit-font-smoothing: antialiased; 13 | -moz-osx-font-smoothing: grayscale; 14 | } 15 | 16 | code { 17 | font-family: source-code-pro, Menlo, Monaco, Consolas, 'Courier New', 18 | monospace; 19 | } 20 | -------------------------------------------------------------------------------- /frontend/src/components/Contact.jsx: -------------------------------------------------------------------------------- 1 | import React from "react"; 2 | 3 | import styles from "./Contact.module.css"; 4 | 5 | function Contact() { 6 | return ( 7 |
8 |
9 |
10 |

Developer Contact:

11 |

12 | Email: linhtch90@gmail.com 13 |

14 |

15 | Github:{" "} 16 | https://github.com/linhtch90 17 |

18 |
19 |
20 |
21 | ); 22 | } 23 | 24 | export default Contact; 25 | -------------------------------------------------------------------------------- /backend/package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "backend", 3 | "version": "0.0.0", 4 | "private": true, 5 | "scripts": { 6 | "start": "node ./bin/www", 7 | "devstart": "nodemon ./bin/www", 8 | "serverstart": "SET DEBUG=backend:* & npm run devstart" 9 | }, 10 | "dependencies": { 11 | "bcryptjs": "^2.4.3", 12 | "cookie-parser": "~1.4.4", 13 | "cors": "^2.8.5", 14 | "debug": "~2.6.9", 15 | "dotenv": "^10.0.0", 16 | "express": "~4.16.1", 17 | "http-errors": "~1.6.3", 18 | "jade": "~1.11.0", 19 | "jsonwebtoken": "^8.5.1", 20 | "mongoose": "^6.0.5", 21 | "morgan": "~1.9.1", 22 | "passport": "^0.4.1", 23 | "passport-jwt": "^4.0.0" 24 | }, 25 | "devDependencies": { 26 | "nodemon": "^2.0.12" 27 | } 28 | } 29 | -------------------------------------------------------------------------------- /frontend/src/index.js: -------------------------------------------------------------------------------- 1 | import React from "react"; 2 | import ReactDOM from "react-dom"; 3 | import "./index.css"; 4 | import App from "./App"; 5 | import reportWebVitals from "./reportWebVitals"; 6 | 7 | import store from "./app/store"; 8 | import { Provider } from "react-redux"; 9 | 10 | require("dotenv").config(); 11 | 12 | ReactDOM.render( 13 | 14 | 15 | 16 | 17 | , 18 | document.getElementById("root") 19 | ); 20 | 21 | // If you want to start measuring performance in your app, pass a function 22 | // to log results (for example: reportWebVitals(console.log)) 23 | // or send to an analytics endpoint. Learn more: https://bit.ly/CRA-vitals 24 | reportWebVitals(); 25 | -------------------------------------------------------------------------------- /frontend/src/components/InvoiceCard.jsx: -------------------------------------------------------------------------------- 1 | import React from "react"; 2 | 3 | import styles from "./InvoiceCard.module.css"; 4 | 5 | function InvoiceCard(props) { 6 | return ( 7 |
8 |

Invoice ID: {props.invoice._id}

9 |

Issue on: {props.invoice.createdAt}

10 |

Customer: {props.invoice.userID.name}

11 |

Email: {props.invoice.userID.email}

12 |

Orders:

13 | {props.invoice.orderBooks.map(book => ( 14 |

•   {book.title}, Quantity: {book.orderQuantity}, Unit Price: ${book.price}/copy

15 | ))} 16 |

Total: ${props.invoice.totalMoney.toFixed(2)}

17 |
18 | ); 19 | } 20 | 21 | export default InvoiceCard; 22 | -------------------------------------------------------------------------------- /frontend/src/components/SignUpYup.module.css: -------------------------------------------------------------------------------- 1 | .pageTitle { 2 | text-align: center; 3 | } 4 | 5 | .outerContainer { 6 | margin: auto; 7 | width: 100%; 8 | } 9 | 10 | .innerContainer { 11 | margin: auto; 12 | width: 85%; 13 | padding-top: 60px; 14 | } 15 | 16 | .formContainer { 17 | margin: auto; 18 | display: flex; 19 | flex-direction: column; 20 | justify-content: center; 21 | width: 50%; 22 | } 23 | 24 | .inputText { 25 | padding: 8px; 26 | background-color: #E5E9F0; 27 | border: solid; 28 | border-width: 3px; 29 | border-radius: 3px; 30 | border-color: #106EBE; 31 | } 32 | 33 | .inputText:focus { 34 | outline: none; 35 | } 36 | 37 | .labelText { 38 | font-weight: 700; 39 | } 40 | 41 | .errorText { 42 | color: #BF616A; 43 | } 44 | 45 | -------------------------------------------------------------------------------- /backend/routes/ordersRouter.js: -------------------------------------------------------------------------------- 1 | var express = require("express"); 2 | var passport = require("passport"); 3 | var jwt = require("jsonwebtoken"); 4 | var router = express.Router(); 5 | var ordersController = require("../controllers/ordersController"); 6 | 7 | const userSecret = process.env.USER_SECRET_KEY; 8 | 9 | function authorizationToken(req, res, next) { 10 | const authorizationHeader = req.headers["authorization"]; 11 | const postedToken = authorizationHeader.split(" ")[1]; 12 | if (!postedToken) { 13 | res.status(401); 14 | } 15 | jwt.verify(postedToken, userSecret, (err, data) => { 16 | if (err) { 17 | next(err); 18 | } else { 19 | next(); 20 | } 21 | }); 22 | } 23 | 24 | router.post("/", authorizationToken, ordersController.createOrder); 25 | 26 | module.exports = router; 27 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # A sample e-commerce web app built with ReactJS, ExpressJS and MongoDB 2 | 3 | MERN stack which stands for MongoDB, ExpressJS, ReactJS and NodeJS has become a popular choice for fullstack Javascript web developer. 4 | 5 | As a personal project for education purpose, I developed a basic bookstore web app with MERN stack. 6 | 7 | Through this project, beginners could have a useful resource to understand how to implement ReactJS, Redux, React Router and others related libraries in React ecosystem to build up a functional front-end solution. 8 | 9 | In addition, the back-end developed with ExpressJS and MongoDB is also an example for building a web api with Javascript. 10 | 11 | **Further detail information is available on Homepage of the live demo at** 12 | 13 | [https://bn-bookstore.herokuapp.com/#/](https://bn-bookstore.herokuapp.com/#/) 14 | -------------------------------------------------------------------------------- /frontend/src/components/BookCard.module.css: -------------------------------------------------------------------------------- 1 | .imageContainer { 2 | width: 25%; 3 | height: 300px; 4 | display: flex; 5 | justify-content: center; 6 | align-items: center; 7 | } 8 | 9 | .contentContainer { 10 | width: 75%; 11 | height: 300px; 12 | display: flex; 13 | flex-direction: column; 14 | justify-content: space-between; 15 | border-left: solid; 16 | border-width: 2px; 17 | border-color: #5E81AC; 18 | padding-left: 1rem; 19 | } 20 | 21 | .cardTitle { 22 | font-size: 1.5rem; 23 | font-weight: 700; 24 | color: #2E3440; 25 | } 26 | 27 | .cardAuthor { 28 | font-size: 1.2rem; 29 | font-style: italic; 30 | color: #4C566A; 31 | } 32 | 33 | .cardPrice { 34 | font-size: 1.5rem; 35 | font-weight: 800; 36 | color: #5E81AC; 37 | } 38 | 39 | .addCartButton { 40 | margin-bottom: 0.5rem; 41 | margin-right: 1rem; 42 | width: 17%; 43 | } 44 | 45 | .cardButtonContainer { 46 | display: flex; 47 | justify-content: flex-start; 48 | } 49 | -------------------------------------------------------------------------------- /backend/routes/adminsRoutes.js: -------------------------------------------------------------------------------- 1 | var express = require("express"); 2 | var passport = require("passport"); 3 | var jwt = require("jsonwebtoken"); 4 | var router = express.Router(); 5 | var ordersController = require("../controllers/ordersController"); 6 | var adminsController = require("../controllers/adminsController"); 7 | 8 | const adminSecret = process.env.ADMIN_SECRET_KEY; 9 | 10 | function authorizationAdminToken(req, res, next) { 11 | const authorizationHeader = req.headers["authorization"]; 12 | const postedToken = authorizationHeader.split(" ")[1]; 13 | if (!postedToken) { 14 | res.status(401); 15 | } 16 | jwt.verify(postedToken, adminSecret, (err, data) => { 17 | if (err) { 18 | next(err); 19 | } else { 20 | next(); 21 | } 22 | }); 23 | } 24 | 25 | router.get("/orders", authorizationAdminToken, ordersController.showAllOrders); 26 | 27 | router.post("/signup", adminsController.createAdmin); 28 | 29 | router.post("/signin", adminsController.validateAdmin); 30 | 31 | module.exports = router; 32 | -------------------------------------------------------------------------------- /frontend/src/components/AdminOrders.jsx: -------------------------------------------------------------------------------- 1 | import React, { useEffect, useState } from "react"; 2 | import axios from "axios"; 3 | 4 | import InvoiceCard from "./InvoiceCard"; 5 | 6 | import styles from "./AdminOrders.module.css"; 7 | 8 | function AdminOrders() { 9 | const adminToken = window.localStorage.getItem("bnToken"); 10 | const [invoices, setInvoices] = useState([]); 11 | 12 | async function getInvoices() { 13 | let response = await axios({ 14 | method: "get", 15 | url: "/admin/orders", 16 | headers: { 17 | Authorization: `JWT ${adminToken}`, 18 | "Content-Type": "application/json", 19 | }, 20 | }); 21 | setInvoices(response.data); 22 | } 23 | 24 | useEffect(() => { 25 | getInvoices(); 26 | }, []); 27 | 28 | return ( 29 |
30 |
31 |

List of Invoices

32 |
33 | {invoices.map((invoice) => ( 34 | 35 | ))} 36 |
37 |
38 | ); 39 | } 40 | 41 | export default AdminOrders; 42 | -------------------------------------------------------------------------------- /frontend/src/components/CartItem.module.css: -------------------------------------------------------------------------------- 1 | .imageContainer { 2 | width: 25%; 3 | height: 300px; 4 | display: flex; 5 | justify-content: center; 6 | align-items: center; 7 | } 8 | 9 | .contentContainer { 10 | width: 75%; 11 | height: 300px; 12 | display: flex; 13 | flex-direction: column; 14 | justify-content: space-between; 15 | border-left: solid; 16 | border-width: 2px; 17 | border-color: #5E81AC; 18 | padding-left: 1rem; 19 | } 20 | 21 | .cardTitle { 22 | font-size: 1.5rem; 23 | font-weight: 700; 24 | color: #2E3440; 25 | } 26 | 27 | .cardAuthor { 28 | font-size: 1.2rem; 29 | font-style: italic; 30 | color: #4C566A; 31 | } 32 | 33 | .cardPrice { 34 | font-size: 1.5rem; 35 | font-weight: 800; 36 | color: #5E81AC; 37 | } 38 | 39 | .addCartButton { 40 | margin-bottom: 0.5rem; 41 | margin-right: 1rem; 42 | width: 17%; 43 | } 44 | 45 | .cardButtonContainer { 46 | display: flex; 47 | justify-content: flex-start; 48 | } 49 | 50 | .bookQuantity { 51 | width: 60px; 52 | text-align: center; 53 | margin-left: 8px; 54 | margin-right: 8px; 55 | } 56 | -------------------------------------------------------------------------------- /frontend/src/components/Cart.jsx: -------------------------------------------------------------------------------- 1 | import React from "react"; 2 | 3 | import { useSelector } from "react-redux"; 4 | 5 | import CartItem from "./CartItem"; 6 | import Order from "./Order"; 7 | 8 | import styles from "./Cart.module.css"; 9 | 10 | function Cart() { 11 | const books = useSelector((state) => state.orderBooks.orderBooks); 12 | return ( 13 |
14 |
15 |
16 |

Cart Information

17 |
18 |
19 |
20 | {books.map((book) => ( 21 | 30 | ))} 31 |
32 |
33 | 34 |
35 |
36 |
37 |
38 | ); 39 | } 40 | 41 | export default Cart; 42 | -------------------------------------------------------------------------------- /frontend/src/components/BookDetail.module.css: -------------------------------------------------------------------------------- 1 | .outerContainer { 2 | margin: auto; 3 | width: 100%; 4 | margin-top: 4rem; 5 | } 6 | 7 | .innerContainer { 8 | margin: auto; 9 | width: 85%; 10 | } 11 | 12 | .imageContainer { 13 | width: 25%; 14 | height: 300px; 15 | display: flex; 16 | justify-content: center; 17 | align-items: center; 18 | } 19 | 20 | .contentContainer { 21 | width: 75%; 22 | height: 300px; 23 | display: flex; 24 | flex-direction: column; 25 | justify-content: space-between; 26 | border-left: solid; 27 | border-width: 2px; 28 | border-color: #5E81AC; 29 | padding-left: 1rem; 30 | } 31 | 32 | .cardTitle { 33 | font-size: 1.5rem; 34 | font-weight: 700; 35 | color: #2E3440; 36 | } 37 | 38 | .cardAuthor { 39 | font-size: 1.2rem; 40 | font-style: italic; 41 | color: #4C566A; 42 | } 43 | 44 | .cardPrice { 45 | font-size: 1.5rem; 46 | font-weight: 800; 47 | color: #5E81AC; 48 | } 49 | 50 | .addCartButton { 51 | margin-bottom: 0.5rem; 52 | margin-right: 1rem; 53 | width: 17%; 54 | } 55 | 56 | .cardButtonContainer { 57 | display: flex; 58 | justify-content: flex-start; 59 | } 60 | 61 | .additionalInfo { 62 | color: #D08770; 63 | font-size: 1.2rem; 64 | font-weight: 700; 65 | } 66 | -------------------------------------------------------------------------------- /backend/controllers/ordersController.js: -------------------------------------------------------------------------------- 1 | var Orders = require("../models/ordersModel"); 2 | var Users = require("../models/usersModel"); 3 | var passport = require("passport"); 4 | 5 | exports.createOrder = function (req, res, next) { 6 | const userID = req.body.id; 7 | const orderBooks = req.body.orderBooks; 8 | const totalMoney = req.body.totalMoney; 9 | 10 | const newOrder = new Orders({ 11 | userID, 12 | orderBooks, 13 | totalMoney, 14 | }); 15 | 16 | Orders.create(newOrder, (err, order) => { 17 | if (err) { 18 | next(err); 19 | } 20 | res.json({ orderStatus: "success" }); 21 | }); 22 | }; 23 | 24 | exports.createOrder2 = passport.authenticate( 25 | "jwt", 26 | { session: false }, 27 | function (req, res, next) { 28 | if (req.isAuthenticated()) { 29 | // Send the route data 30 | res.status(200).send("Web page data"); 31 | } else { 32 | // Not authorized 33 | res.status(401).send("You are not authorized to view this "); 34 | } 35 | } 36 | ); 37 | 38 | exports.showAllOrders = function (req, res, next) { 39 | // Orders.find({}, (err, orders) => { 40 | // if (err) { 41 | // next(err); 42 | // } 43 | // Orders.populate(orders, {path: "userID", select: "name"}); 44 | // res.json(orders); 45 | // }); 46 | 47 | Orders.find({}).populate("userID").exec((err, orders) => { 48 | if (err) { 49 | next(err); 50 | } 51 | res.json(orders); 52 | }); 53 | }; 54 | -------------------------------------------------------------------------------- /frontend/package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "frontend", 3 | "version": "0.1.0", 4 | "private": true, 5 | "dependencies": { 6 | "@reduxjs/toolkit": "^1.6.1", 7 | "@testing-library/jest-dom": "^5.14.1", 8 | "@testing-library/react": "^11.2.7", 9 | "@testing-library/user-event": "^12.8.3", 10 | "axios": "^0.21.1", 11 | "dotenv": "^10.0.0", 12 | "formik": "^2.2.9", 13 | "primeflex": "^2.0.0", 14 | "primeicons": "^4.1.0", 15 | "primereact": "^6.5.1", 16 | "react": "^17.0.2", 17 | "react-dom": "^17.0.2", 18 | "react-redux": "^7.2.4", 19 | "react-router-dom": "^5.2.1", 20 | "react-scripts": "4.0.3", 21 | "react-transition-group": "^4.4.2", 22 | "redux": "^4.1.1", 23 | "web-vitals": "^1.1.2", 24 | "yup": "^0.32.9" 25 | }, 26 | "scripts": { 27 | "start": "react-scripts start", 28 | "build": "react-scripts build", 29 | "test": "react-scripts test", 30 | "eject": "react-scripts eject" 31 | }, 32 | "eslintConfig": { 33 | "extends": [ 34 | "react-app", 35 | "react-app/jest" 36 | ] 37 | }, 38 | "browserslist": { 39 | "production": [ 40 | ">0.2%", 41 | "not dead", 42 | "not op_mini all" 43 | ], 44 | "development": [ 45 | "last 1 chrome version", 46 | "last 1 firefox version", 47 | "last 1 safari version" 48 | ] 49 | }, 50 | "devDependencies": { 51 | "redux-devtools": "^3.7.0" 52 | }, 53 | "proxy": "http://localhost:3030" 54 | } 55 | -------------------------------------------------------------------------------- /frontend/src/components/NavBar.module.css: -------------------------------------------------------------------------------- 1 | .outerContainer { 2 | margin: auto; 3 | width: 100%; 4 | background-color: #D8DEE9; 5 | } 6 | 7 | .innerContainer { 8 | margin: auto; 9 | width: 85%; 10 | } 11 | 12 | .navbarContainer { 13 | display: flex; 14 | flex-direction: row; 15 | width: 100%; 16 | height: 7rem; 17 | justify-content: space-between; 18 | } 19 | 20 | .logoContainer { 21 | display: flex; 22 | justify-content: flex-start; 23 | align-items: center; 24 | } 25 | 26 | .menuContainer { 27 | display: flex; 28 | width: 75%; 29 | justify-content: space-between; 30 | align-items: center; 31 | } 32 | 33 | .menu { 34 | display: flex; 35 | height: 100%; 36 | } 37 | 38 | .menuItem { 39 | display: flex; 40 | width: 11rem; 41 | height: 100%; 42 | justify-content: center; 43 | align-items: center; 44 | font-size: 1.7rem; 45 | font-weight: 900; 46 | } 47 | 48 | .menuItem:hover { 49 | color: #BF616A; 50 | cursor: pointer; 51 | } 52 | 53 | .menuText { 54 | text-decoration: none; 55 | color: inherit; 56 | width: 100%; 57 | } 58 | 59 | .navUser { 60 | display: flex; 61 | flex-direction: column; 62 | align-items: center; 63 | justify-content: center; 64 | } 65 | 66 | .navButton { 67 | width: 100%; 68 | } 69 | 70 | .navCartContainer { 71 | display: flex; 72 | align-items: center; 73 | justify-content: center; 74 | } 75 | 76 | .cartIcon { 77 | color: #2E3440; 78 | } 79 | 80 | .cartIcon:hover { 81 | cursor: pointer; 82 | } 83 | 84 | .badgeContent { 85 | background-color: #BF616A; 86 | } 87 | 88 | 89 | 90 | 91 | 92 | 93 | -------------------------------------------------------------------------------- /frontend/src/components/SearchBar.jsx: -------------------------------------------------------------------------------- 1 | import React, { useState } from "react"; 2 | import { useHistory } from "react-router-dom"; 3 | 4 | import { Button } from "primereact/button"; 5 | import { InputText } from "primereact/inputtext"; 6 | 7 | import styles from "./SearchBar.module.css"; 8 | 9 | function SearchBar() { 10 | let history = useHistory(); 11 | 12 | const [searchTitle, setSearchTitle] = useState(""); 13 | 14 | function handleChange(e) { 15 | setSearchTitle(e.target.value); 16 | } 17 | 18 | function handleSearchClick(e) { 19 | if (searchTitle) { 20 | const searchLocation = `/bookstore/books_search/${searchTitle}`; 21 | history.push(searchLocation); 22 | } 23 | } 24 | 25 | function handleSearchPress(e) { 26 | if (e.key === "Enter") { 27 | if (searchTitle) { 28 | const searchLocation = `/bookstore/books_search/${searchTitle}`; 29 | history.push(searchLocation); 30 | } 31 | } 32 | } 33 | 34 | return ( 35 |
36 |
37 |
38 |
39 | 45 |
51 |
52 |
53 |
54 | ); 55 | } 56 | 57 | export default SearchBar; 58 | -------------------------------------------------------------------------------- /frontend/public/index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 12 | 13 | 17 | 18 | 27 | BN Bookstore 28 | 29 | 30 | 31 |
32 | 42 | 43 | 44 | -------------------------------------------------------------------------------- /frontend/src/components/AppContainer.jsx: -------------------------------------------------------------------------------- 1 | import React from "react"; 2 | import NavBar from "./NavBar"; 3 | import styles from "./AppContainer.module.css"; 4 | import { BrowserRouter, Switch, Route, HashRouter } from "react-router-dom"; 5 | 6 | import Home from "./Home"; 7 | import Bookstore from "./Bookstore"; 8 | import Contact from "./Contact"; 9 | import BookDetail from "./BookDetail"; 10 | import BookSearchResults from "./BookSearchResults"; 11 | import Cart from "./Cart"; 12 | import SignIn from "./SignIn"; 13 | import SignUp from "./SignUp"; 14 | import AdminSignIn from "./AdminSignIn"; 15 | import AdminOrders from "./AdminOrders"; 16 | 17 | function AppContainer() { 18 | return ( 19 |
20 | 21 | 22 | 23 | 24 | 25 | 26 | 27 | 28 | 29 | 30 | 31 | 32 | 33 | 34 | 35 | 36 | 37 | 38 | 39 | 40 | 41 | 42 | 43 | 44 | 45 | 46 | 47 | 48 | 49 | 50 | 51 | 52 | 53 | 54 | 55 | 56 | 57 | 58 | 59 | 60 | 61 | 62 | 63 | 64 |
65 | ); 66 | } 67 | 68 | export default AppContainer; 69 | -------------------------------------------------------------------------------- /frontend/src/components/BookCard.jsx: -------------------------------------------------------------------------------- 1 | import React from "react"; 2 | import { useHistory } from "react-router-dom"; 3 | 4 | import { useDispatch } from "react-redux"; 5 | import { addBookToCart, calcTotalMoney } from "../features/orderBooks/orderBooksSlice"; 6 | 7 | import { Button } from "primereact/button"; 8 | 9 | import styles from "./BookCard.module.css"; 10 | 11 | function BookCard(props) { 12 | let history = useHistory(); 13 | 14 | const dispatch = useDispatch(); 15 | 16 | const book = { 17 | id: props.id, 18 | imgSrc: props.imgSrc, 19 | title: props.title, 20 | author: props.author, 21 | price: parseFloat(props.price), 22 | orderQuantity: 1, 23 | }; 24 | 25 | function navigateBookDetail() { 26 | history.push(`/bookstore/book/${props.id}`); 27 | } 28 | 29 | return ( 30 |
31 |
32 | {props.title} 33 |
34 |
35 |
{props.title}
36 |
Author: {props.author}
37 |
Price: ${props.price}
38 |
39 |
55 |
56 |
57 | ); 58 | } 59 | 60 | export default BookCard; 61 | -------------------------------------------------------------------------------- /backend/bin/www: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env node 2 | 3 | /** 4 | * Module dependencies. 5 | */ 6 | 7 | require("dotenv").config(); 8 | var app = require('../app'); 9 | var debug = require('debug')('backend:server'); 10 | var http = require('http'); 11 | 12 | 13 | /** 14 | * Get port from environment and store in Express. 15 | */ 16 | 17 | var port = normalizePort(process.env.PORT || '3030'); 18 | app.set('port', port); 19 | 20 | /** 21 | * Create HTTP server. 22 | */ 23 | 24 | var server = http.createServer(app); 25 | 26 | /** 27 | * Listen on provided port, on all network interfaces. 28 | */ 29 | 30 | server.listen(port); 31 | server.on('error', onError); 32 | server.on('listening', onListening); 33 | 34 | /** 35 | * Normalize a port into a number, string, or false. 36 | */ 37 | 38 | function normalizePort(val) { 39 | var port = parseInt(val, 10); 40 | 41 | if (isNaN(port)) { 42 | // named pipe 43 | return val; 44 | } 45 | 46 | if (port >= 0) { 47 | // port number 48 | return port; 49 | } 50 | 51 | return false; 52 | } 53 | 54 | /** 55 | * Event listener for HTTP server "error" event. 56 | */ 57 | 58 | function onError(error) { 59 | if (error.syscall !== 'listen') { 60 | throw error; 61 | } 62 | 63 | var bind = typeof port === 'string' 64 | ? 'Pipe ' + port 65 | : 'Port ' + port; 66 | 67 | // handle specific listen errors with friendly messages 68 | switch (error.code) { 69 | case 'EACCES': 70 | console.error(bind + ' requires elevated privileges'); 71 | process.exit(1); 72 | break; 73 | case 'EADDRINUSE': 74 | console.error(bind + ' is already in use'); 75 | process.exit(1); 76 | break; 77 | default: 78 | throw error; 79 | } 80 | } 81 | 82 | /** 83 | * Event listener for HTTP server "listening" event. 84 | */ 85 | 86 | function onListening() { 87 | var addr = server.address(); 88 | var bind = typeof addr === 'string' 89 | ? 'pipe ' + addr 90 | : 'port ' + addr.port; 91 | debug('Listening on ' + bind); 92 | } 93 | -------------------------------------------------------------------------------- /frontend/src/components/BookSearchResults.jsx: -------------------------------------------------------------------------------- 1 | import React, { useState, useEffect } from "react"; 2 | import axios from "axios"; 3 | 4 | import { useParams } from "react-router-dom"; 5 | 6 | import { ProgressSpinner } from 'primereact/progressspinner'; 7 | 8 | import SearchBar from "./SearchBar"; 9 | import BookCard from "./BookCard"; 10 | import PageSwitch from "./PageSwitch"; 11 | 12 | import styles from "./BookSearchResults.module.css"; 13 | 14 | function BookSearchResults() { 15 | const { searchTitle } = useParams(); 16 | const fetchUrl = `/books/book_search_title/${searchTitle}`; 17 | 18 | const [books, setBooks] = useState([]); 19 | const [totalPages, setTotalPages] = useState(0); 20 | const [currentPage, setCurrentPage] = useState(1); 21 | const [bookItem, setBookItem] = useState(0); 22 | const [loaded, setLoaded] = useState(false); 23 | 24 | function onChangePageNumber(pageNumber) { 25 | setCurrentPage(pageNumber); 26 | setBookItem((pageNumber - 1) * 10); 27 | } 28 | 29 | useEffect(() => { 30 | const fetchBooks = async () => { 31 | const results = await axios.get(fetchUrl); 32 | setBooks(results.data); 33 | setTotalPages( 34 | results.data.length % 10 === 0 35 | ? results.data.length / 10 36 | : parseInt(results.data.length / 10) + 1 37 | ); 38 | setLoaded(true); 39 | }; 40 | fetchBooks(); 41 | }, [searchTitle]); 42 | 43 | return ( 44 |
45 |
46 | 47 |
48 | {!loaded ? : books.slice(bookItem, currentPage * 10).map((book) => ( 49 | 57 | ))} 58 |
59 | 64 |
65 |
66 | ); 67 | } 68 | 69 | export default BookSearchResults; 70 | -------------------------------------------------------------------------------- /frontend/src/features/orderBooks/orderBooksSlice.js: -------------------------------------------------------------------------------- 1 | import { createSlice } from "@reduxjs/toolkit"; 2 | 3 | export const orderBooksSlice = createSlice({ 4 | name: "orderBooks", 5 | initialState: { 6 | orderBooks: [], 7 | totalMoney: 0, 8 | }, 9 | reducers: { 10 | addBookToCart: (state, action) => { 11 | let exist = false; 12 | if (state.orderBooks.length === 0) { 13 | state.orderBooks.push(action.payload); 14 | } else { 15 | for (let orderBook of state.orderBooks) { 16 | if (orderBook.id === action.payload.id) { 17 | exist = true; 18 | } 19 | } 20 | if (exist === false) { 21 | state.orderBooks.push(action.payload); 22 | } 23 | } 24 | }, 25 | removeBookFromCart: (state, action) => { 26 | state.orderBooks = state.orderBooks.filter( 27 | (orderBook) => orderBook.id !== action.payload.id 28 | ); 29 | }, 30 | increaseQuantity: (state, action) => { 31 | for (let orderBook of state.orderBooks) { 32 | if (orderBook.id === action.payload.id) { 33 | orderBook.orderQuantity += 1; 34 | } 35 | } 36 | }, 37 | decreaseQuantity: (state, action) => { 38 | for (let orderBook of state.orderBooks) { 39 | if (orderBook.id === action.payload.id) { 40 | orderBook.orderQuantity -= 1; 41 | } 42 | } 43 | }, 44 | changeQuantity: (state, action) => { 45 | for (let orderBook of state.orderBooks) { 46 | if (orderBook.id === action.payload.id) { 47 | orderBook.orderQuantity = parseInt(action.payload.orderQuantity); 48 | } 49 | } 50 | }, 51 | calcTotalMoney: (state) => { 52 | let total = 0; 53 | for (let orderBook of state.orderBooks) { 54 | total = 55 | total + 56 | parseInt(orderBook.orderQuantity) * parseFloat(orderBook.price); 57 | } 58 | state.totalMoney = total; 59 | }, 60 | cleanCart: (state) => { 61 | state.orderBooks = []; 62 | state.totalMoney = 0; 63 | }, 64 | }, 65 | }); 66 | 67 | export const { 68 | addBookToCart, 69 | removeBookFromCart, 70 | increaseQuantity, 71 | decreaseQuantity, 72 | changeQuantity, 73 | calcTotalMoney, 74 | cleanCart, 75 | } = orderBooksSlice.actions; 76 | 77 | export default orderBooksSlice.reducer; 78 | -------------------------------------------------------------------------------- /backend/controllers/booksController.js: -------------------------------------------------------------------------------- 1 | const { NotExtended } = require("http-errors"); 2 | var Books = require("../models/booksModel"); 3 | 4 | // Get all available books 5 | exports.all_books = function (req, res, next) { 6 | Books.find({}, function (err, results) { 7 | if (err) { 8 | return next(err); 9 | } 10 | res.json(results); 11 | }); 12 | }; 13 | 14 | // Get a book by ID 15 | exports.book_details = function (req, res, next) { 16 | Books.findById(req.params.id, function (err, result) { 17 | if (err) { 18 | return next(err); 19 | } 20 | res.json(result); 21 | }); 22 | }; 23 | 24 | // Search book by title 25 | exports.search_book_title = function (req, res, next) { 26 | reg_pattern = `.*${req.params.search_title}.*`; 27 | Books.find({ 28 | title: new RegExp(reg_pattern, "i"), 29 | }).exec(function (err, results) { 30 | if (err) { 31 | return next(err); 32 | } 33 | res.json(results); 34 | }); 35 | }; 36 | 37 | // Add a book to database 38 | exports.add_book = function (req, res, next) { 39 | const new_book = new Books({ 40 | title: req.body.title, 41 | author: req.body.author, 42 | price: parseFloat(req.body.price), 43 | image_src: req.body.image_src, 44 | isbn_13: req.body.isbn_13, 45 | publisher: req.body.publisher, 46 | publish_date: req.body.publish_date, 47 | pages: parseInt(req.body.pages), 48 | }); 49 | Books.create(new_book, function (err) { 50 | if (err) { 51 | return next(err); 52 | } 53 | }); 54 | }; 55 | 56 | // Find a book by ID and modify its information 57 | exports.modify_book = function (req, res, next) { 58 | const new_book = new Books({ 59 | title: req.body.title, 60 | author: req.body.author, 61 | price: parseFloat(req.body.price), 62 | image_src: req.body.image_src, 63 | isbn_13: req.body.isbn_13, 64 | publisher: req.body.publisher, 65 | publish_date: req.body.publish_date, 66 | pages: parseInt(req.body.pages), 67 | }); 68 | Books.findByIdAndUpdate(req.params.id, new_book, {}, function (err) { 69 | if (err) { 70 | return next(err); 71 | } 72 | }); 73 | }; 74 | 75 | // Find a book by ID and delete it from database 76 | exports.delete_book = function (req, res, next) { 77 | Books.findByIdAndRemove(req.params.id, function (err) { 78 | if (err) { 79 | return next(err); 80 | } 81 | }); 82 | }; 83 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | # Logs 2 | logs 3 | *.log 4 | npm-debug.log* 5 | yarn-debug.log* 6 | yarn-error.log* 7 | lerna-debug.log* 8 | .pnpm-debug.log* 9 | 10 | # Diagnostic reports (https://nodejs.org/api/report.html) 11 | report.[0-9]*.[0-9]*.[0-9]*.[0-9]*.json 12 | 13 | # Runtime data 14 | pids 15 | *.pid 16 | *.seed 17 | *.pid.lock 18 | 19 | # Directory for instrumented libs generated by jscoverage/JSCover 20 | lib-cov 21 | 22 | # Coverage directory used by tools like istanbul 23 | coverage 24 | *.lcov 25 | 26 | # nyc test coverage 27 | .nyc_output 28 | 29 | # Grunt intermediate storage (https://gruntjs.com/creating-plugins#storing-task-files) 30 | .grunt 31 | 32 | # Bower dependency directory (https://bower.io/) 33 | bower_components 34 | 35 | # node-waf configuration 36 | .lock-wscript 37 | 38 | # Compiled binary addons (https://nodejs.org/api/addons.html) 39 | build/Release 40 | 41 | # Dependency directories 42 | node_modules/ 43 | jspm_packages/ 44 | 45 | # Snowpack dependency directory (https://snowpack.dev/) 46 | web_modules/ 47 | 48 | # TypeScript cache 49 | *.tsbuildinfo 50 | 51 | # Optional npm cache directory 52 | .npm 53 | 54 | # Optional eslint cache 55 | .eslintcache 56 | 57 | # Microbundle cache 58 | .rpt2_cache/ 59 | .rts2_cache_cjs/ 60 | .rts2_cache_es/ 61 | .rts2_cache_umd/ 62 | 63 | # Optional REPL history 64 | .node_repl_history 65 | 66 | # Output of 'npm pack' 67 | *.tgz 68 | 69 | # Yarn Integrity file 70 | .yarn-integrity 71 | 72 | # dotenv environment variables file 73 | .env 74 | .env.test 75 | .env.production 76 | 77 | # parcel-bundler cache (https://parceljs.org/) 78 | .cache 79 | .parcel-cache 80 | 81 | # Next.js build output 82 | .next 83 | out 84 | 85 | # Nuxt.js build / generate output 86 | .nuxt 87 | dist 88 | 89 | # Gatsby files 90 | .cache/ 91 | # Comment in the public line in if your project uses Gatsby and not Next.js 92 | # https://nextjs.org/blog/next-9-1#public-directory-support 93 | # public 94 | 95 | # vuepress build output 96 | .vuepress/dist 97 | 98 | # Serverless directories 99 | .serverless/ 100 | 101 | # FuseBox cache 102 | .fusebox/ 103 | 104 | # DynamoDB Local files 105 | .dynamodb/ 106 | 107 | # TernJS port file 108 | .tern-port 109 | 110 | # Stores VSCode versions used for testing VSCode extensions 111 | .vscode-test 112 | 113 | # yarn v2 114 | .yarn/cache 115 | .yarn/unplugged 116 | .yarn/build-state.yml 117 | .yarn/install-state.gz 118 | .pnp.* -------------------------------------------------------------------------------- /backend/controllers/usersController.js: -------------------------------------------------------------------------------- 1 | var Users = require("../models/usersModel"); 2 | var bcryptjs = require("bcryptjs"); 3 | var passportJWT = require("passport-jwt"); 4 | var jwt = require("jsonwebtoken"); 5 | 6 | const userSecret = process.env.USER_SECRET_KEY; 7 | 8 | const ExtractJwt = passportJWT.ExtractJwt; 9 | const jwtOptions = {}; 10 | jwtOptions.jwtFromRequest = ExtractJwt.fromAuthHeaderWithScheme("jwt"); 11 | jwtOptions.secretOrKey = userSecret; 12 | 13 | exports.createUser = function (req, res, next) { 14 | const name = req.body.name.toString().trim(); 15 | const email = req.body.email.toString().trim(); 16 | const password = req.body.password.toString(); 17 | 18 | const newUser = new Users({ 19 | name, 20 | email, 21 | password, 22 | }); 23 | 24 | function createHashedUser(newUser, callback) { 25 | bcryptjs.genSalt(10, (err, salt) => { 26 | bcryptjs.hash(newUser.password, salt, (error, hash) => { 27 | const hashedUser = newUser; 28 | hashedUser.password = hash; 29 | hashedUser.save(callback); 30 | }); 31 | }); 32 | } 33 | 34 | createHashedUser(newUser, (err, user) => { 35 | if (err) { 36 | return next(err); 37 | } 38 | res.json({ createUser: "success" }); 39 | }); 40 | }; 41 | 42 | exports.validateUser = function(req, res, next) { 43 | function getUserByEmail(postEmail, callback) { 44 | Users.findOne({email: postEmail}, callback); 45 | } 46 | 47 | function comparePassword(postedPassword, hash, callback) { 48 | bcryptjs.compare(postedPassword, hash, (err, isMatch) => { 49 | if (err) { 50 | return next(err); 51 | } 52 | callback(null, isMatch); 53 | }); 54 | } 55 | 56 | if (req.body.email && req.body.password) { 57 | const email = req.body.email; 58 | const password = req.body.password; 59 | 60 | getUserByEmail(email, (err, user) => { 61 | if (!user) { 62 | res.status(404).json({message: "User does not exist"}); 63 | } else { 64 | comparePassword(password, user.password, (err, isMatch) => { 65 | if (err) { 66 | next(err); 67 | } 68 | if (isMatch) { 69 | const payload = {id: user._id}; 70 | const token = jwt.sign(payload, jwtOptions.secretOrKey, {expiresIn: "1h"}); 71 | res.json({signInStatus: "success", token, id: user._id}); 72 | } else { 73 | res.status(401).json({message: "Wrong password"}); 74 | } 75 | }) 76 | } 77 | }); 78 | } 79 | }; 80 | -------------------------------------------------------------------------------- /frontend/src/logo.svg: -------------------------------------------------------------------------------- 1 | -------------------------------------------------------------------------------- /backend/app.js: -------------------------------------------------------------------------------- 1 | var createError = require("http-errors"); 2 | var express = require("express"); 3 | var path = require("path"); 4 | var cookieParser = require("cookie-parser"); 5 | var logger = require("morgan"); 6 | var cors = require("cors"); 7 | var jwt = require("jsonwebtoken"); 8 | var passport = require("passport"); 9 | 10 | // const userSecret = process.env.USER_SECRET_KEY; 11 | 12 | var indexRouter = require("./routes/index"); 13 | // var usersRouter = require("./routes/users"); Auto generated 14 | var booksRouter = require("./routes/booksRoutes"); 15 | var usersRouter = require("./routes/usersRoutes"); 16 | var ordersRouter = require("./routes/ordersRouter"); 17 | var adminsRouter = require("./routes/adminsRoutes"); 18 | 19 | var app = express(); 20 | 21 | // Database connection 22 | var mongoose = require("mongoose"); 23 | var mongoDB = process.env.MONGODB_CONNECTION_URL; 24 | mongoose.connect(mongoDB, { useNewUrlParser: true, useUnifiedTopology: true }); 25 | var db = mongoose.connection; 26 | db.on("error", console.error.bind(console, "MongoDB connection error...")); 27 | 28 | // view engine setup for development 29 | // app.set("views", path.join(__dirname, "views")); 30 | // app.set("view engine", "jade"); 31 | 32 | // Serve React Project on heroku 33 | app.use(express.static(path.join(__dirname, "build"))); 34 | // Do not use the below code. It makes frontend cannot fetch data 35 | // app.get("/*", (req, res) => { 36 | // res.sendFile(path.join(__dirname, "build", "index.html")); 37 | // }); 38 | 39 | app.use(logger("dev")); 40 | app.use(express.json()); 41 | app.use(express.urlencoded({ extended: false })); 42 | app.use(cookieParser()); 43 | app.use(express.static(path.join(__dirname, "public"))); 44 | 45 | app.use( 46 | cors({ 47 | origin: process.env.SERVER_DOMAIN, 48 | }) 49 | ); 50 | 51 | app.use(passport.initialize()); 52 | 53 | app.use("/", indexRouter); 54 | // app.use("/users", usersRouter); Auto generated 55 | app.use("/books", booksRouter); 56 | app.use("/user", usersRouter); 57 | app.use("/order", ordersRouter); 58 | app.use("/admin", adminsRouter); 59 | 60 | // catch 404 and forward to error handler 61 | app.use(function (req, res, next) { 62 | next(createError(404)); 63 | }); 64 | 65 | // error handler 66 | app.use(function (err, req, res, next) { 67 | // set locals, only providing error in development 68 | res.locals.message = err.message; 69 | res.locals.error = req.app.get("env") === "development" ? err : {}; 70 | 71 | // render the error page 72 | res.status(err.status || 500); 73 | res.render("error"); 74 | }); 75 | 76 | module.exports = app; 77 | -------------------------------------------------------------------------------- /backend/controllers/adminsController.js: -------------------------------------------------------------------------------- 1 | var Admins = require("../models/adminsModel"); 2 | var bcryptjs = require("bcryptjs"); 3 | var passportJWT = require("passport-jwt"); 4 | var jwt = require("jsonwebtoken"); 5 | 6 | const adminSecret = process.env.ADMIN_SECRET_KEY; 7 | 8 | const ExtractJwt = passportJWT.ExtractJwt; 9 | const jwtOptions = {}; 10 | jwtOptions.jwtFromRequest = ExtractJwt.fromAuthHeaderWithScheme("jwt"); 11 | jwtOptions.secretOrKey = adminSecret; 12 | 13 | exports.createAdmin = function (req, res, next) { 14 | const name = req.body.name.toString().trim(); 15 | const email = req.body.email.toString().trim(); 16 | const password = req.body.password.toString(); 17 | 18 | const newAdmin = new Admins({ 19 | name, 20 | email, 21 | password, 22 | }); 23 | 24 | function createHashedAdmin(newAdmin, callback) { 25 | bcryptjs.genSalt(10, (err, salt) => { 26 | bcryptjs.hash(newAdmin.password, salt, (error, hash) => { 27 | const hashedAdmin = newAdmin; 28 | hashedAdmin.password = hash; 29 | hashedAdmin.save(callback); 30 | }); 31 | }); 32 | } 33 | 34 | createHashedAdmin(newAdmin, (err, user) => { 35 | if (err) { 36 | return next(err); 37 | } 38 | res.json({ createAdmin: "success" }); 39 | }); 40 | }; 41 | 42 | exports.validateAdmin = function (req, res, next) { 43 | function getAdminByEmail(postEmail, callback) { 44 | Admins.findOne({ email: postEmail }, callback); 45 | } 46 | 47 | function comparePassword(postedPassword, hash, callback) { 48 | bcryptjs.compare(postedPassword, hash, (err, isMatch) => { 49 | if (err) { 50 | return next(err); 51 | } 52 | callback(null, isMatch); 53 | }); 54 | } 55 | 56 | if (req.body.email && req.body.password) { 57 | const email = req.body.email; 58 | const password = req.body.password; 59 | 60 | getAdminByEmail(email, (err, admin) => { 61 | if (!admin) { 62 | res.status(404).json({ message: "Admin account does not exist" }); 63 | } else { 64 | comparePassword(password, admin.password, (err, isMatch) => { 65 | if (err) { 66 | next(err); 67 | } 68 | if (isMatch) { 69 | const payload = { id: admin._id }; 70 | const token = jwt.sign(payload, jwtOptions.secretOrKey, { 71 | expiresIn: "1h", 72 | }); 73 | res.json({ signInStatus: "success", token, id: admin._id }); 74 | } else { 75 | res.status(401).json({ message: "Wrong password" }); 76 | } 77 | }); 78 | } 79 | }); 80 | } 81 | }; 82 | -------------------------------------------------------------------------------- /frontend/src/components/Order.jsx: -------------------------------------------------------------------------------- 1 | import React, { useState, useRef } from "react"; 2 | import axios from "axios"; 3 | 4 | import { useSelector } from "react-redux"; 5 | 6 | import { Button } from "primereact/button"; 7 | import { Toast } from "primereact/toast"; 8 | 9 | import styles from "./Order.module.css"; 10 | 11 | function Order(props) { 12 | const toast = useRef(null); 13 | 14 | let totalMoney = useSelector((state) => state.orderBooks.totalMoney); 15 | const [isOrderSubmit, setIsOrderSubmit] = useState(false); 16 | 17 | const showSuccess = () => { 18 | toast.current.show({ 19 | severity: "success", 20 | summary: "Successful!", 21 | detail: "Your order is submitted", 22 | life: 2500, 23 | }); 24 | }; 25 | 26 | const showFail = () => { 27 | toast.current.show({ 28 | severity: "error", 29 | summary: "Oops!!!", 30 | detail: "Submission fails", 31 | life: 2500, 32 | }); 33 | }; 34 | 35 | const showSignInRequire = () => { 36 | toast.current.show({ 37 | severity: "warn", 38 | summary: "Oops!!!", 39 | detail: "You have not signed in yet", 40 | life: 2500, 41 | }); 42 | }; 43 | 44 | async function sendOrder() { 45 | const userID = window.localStorage.getItem("bnUserID"); 46 | const token = window.localStorage.getItem("bnToken"); 47 | const orderBooks = props.books; 48 | let response = await axios({ 49 | method: "post", 50 | url: "/order", 51 | data: { 52 | id: userID, 53 | orderBooks: orderBooks, 54 | totalMoney: totalMoney, 55 | }, 56 | headers: { 57 | Authorization: `JWT ${token}`, 58 | "Content-Type": "application/json", 59 | }, 60 | }); 61 | if (response.data.orderStatus === "success") { 62 | setIsOrderSubmit(true); 63 | showSuccess(); 64 | } else { 65 | showFail(); 66 | } 67 | } 68 | 69 | function handleOrderClick() { 70 | const userID = window.localStorage.getItem("bnUserID"); 71 | const token = window.localStorage.getItem("bnToken"); 72 | 73 | if (userID && token) { 74 | sendOrder(); 75 | } else { 76 | showSignInRequire(); 77 | } 78 | } 79 | 80 | return ( 81 |
82 | 83 |
84 | Total:{" "} 85 | ${totalMoney.toFixed(2)} 86 |
87 |
89 | ); 90 | } 91 | 92 | export default Order; 93 | -------------------------------------------------------------------------------- /frontend/src/components/PageSwitch.jsx: -------------------------------------------------------------------------------- 1 | import React, { useState } from "react"; 2 | import { Link, useRouteMatch } from "react-router-dom"; 3 | 4 | import { Button } from "primereact/button"; 5 | import { InputText } from "primereact/inputtext"; 6 | 7 | import styles from "./PageSwitch.module.css"; 8 | 9 | function PageSwitch(props) { 10 | const [currentPage, setCurrentPage] = useState(props.currentPage); 11 | 12 | let { url } = useRouteMatch(); 13 | 14 | function increasePageNumber() { 15 | if (currentPage <= props.totalPages) { 16 | setCurrentPage(currentPage + 1); 17 | props.onChangePageNumber(currentPage + 1); 18 | } 19 | } 20 | 21 | function decreasePageNumber() { 22 | if (currentPage >= 2 && currentPage <= props.totalPages) { 23 | setCurrentPage(currentPage - 1); 24 | props.onChangePageNumber(currentPage - 1); 25 | } else if (currentPage < 2) { 26 | setCurrentPage(1); 27 | props.onChangePageNumber(1); 28 | } 29 | } 30 | 31 | function changePageNumber(e) { 32 | if (e.key === "Enter") { 33 | if (e.target.value >= 1 && e.target.value <= props.totalPages) { 34 | setCurrentPage(parseInt(e.target.value)); 35 | props.onChangePageNumber(parseInt(e.target.value)); 36 | } else { 37 | setCurrentPage(1); 38 | props.onChangePageNumber(1); 39 | } 40 | } 41 | } 42 | 43 | function handleChange(e) { 44 | setCurrentPage(parseInt(e.target.value)); 45 | } 46 | 47 | return ( 48 |
49 |
50 |
51 | 52 |
76 |
77 |
78 | ); 79 | } 80 | 81 | export default PageSwitch; 82 | -------------------------------------------------------------------------------- /frontend/src/components/BookDetail.jsx: -------------------------------------------------------------------------------- 1 | import React, { useState, useEffect } from "react"; 2 | import { useParams } from "react-router-dom"; 3 | import axios from "axios"; 4 | 5 | import { useDispatch } from "react-redux"; 6 | import { 7 | addBookToCart, 8 | calcTotalMoney, 9 | } from "../features/orderBooks/orderBooksSlice"; 10 | 11 | import { Button } from "primereact/button"; 12 | 13 | import styles from "./BookDetail.module.css"; 14 | 15 | function BookDetail() { 16 | const { id } = useParams(); 17 | const dispatch = useDispatch(); 18 | const fetchUrl = `/books/book_details/${id}`; 19 | 20 | const [bookData, setBookData] = useState({}); 21 | 22 | useEffect(() => { 23 | const fetchBook = async () => { 24 | const result = await axios.get(fetchUrl); 25 | const book = { 26 | id: result.data._id, 27 | imgSrc: result.data.image_src, 28 | title: result.data.title, 29 | author: result.data.author, 30 | price: parseFloat(result.data.price), 31 | orderQuantity: 1, 32 | isbn_13: result.data.isbn_13, 33 | publisher: result.data.publisher, 34 | publish_date: result.data.publish_date, 35 | pages: result.data.pages, 36 | }; 37 | setBookData(book); 38 | }; 39 | fetchBook(); 40 | }, []); 41 | 42 | return ( 43 |
44 |
45 |
46 |
47 | {bookData.title} 48 |
49 |
50 |
{bookData.title}
51 |
Author: {bookData.author}
52 |
Price: ${bookData.price}
53 |
Additional Information
54 |
55 | ISBN 13: {bookData.isbn_13} 56 |
57 |
58 | Publisher: {bookData.publisher} 59 |
60 |
61 | Publish Date: {bookData.publish_date} 62 |
63 |
64 | Pages: {bookData.pages} 65 |
66 |
67 |
77 |
78 |
79 |
80 |
81 | ); 82 | } 83 | 84 | export default BookDetail; 85 | -------------------------------------------------------------------------------- /frontend/src/components/Bookstore.jsx: -------------------------------------------------------------------------------- 1 | import React, { useState, useEffect } from "react"; 2 | import { Switch, Route, useRouteMatch, useHistory } from "react-router-dom"; 3 | import axios from "axios"; 4 | 5 | import { ProgressSpinner } from 'primereact/progressspinner'; 6 | 7 | import SearchBar from "./SearchBar"; 8 | import BookCard from "./BookCard"; 9 | import PageSwitch from "./PageSwitch"; 10 | 11 | import styles from "./Bookstore.module.css"; 12 | 13 | function Bookstore() { 14 | let {path} = useRouteMatch(); 15 | let history = useHistory(); 16 | 17 | const fetchUrl = "/books"; 18 | 19 | const [books, setBooks] = useState([]); 20 | const [totalPages, setTotalPages] = useState(0); 21 | const [currentPage, setCurrentPage] = useState(1); 22 | const [bookItem, setBookItem] = useState(0); 23 | const [loaded, setLoaded] = useState(false); 24 | 25 | 26 | function onChangePageNumber(pageNumber) { 27 | setCurrentPage(pageNumber); 28 | setBookItem((pageNumber - 1) * 10); 29 | history.push(`/bookstore/page/${pageNumber}`); 30 | } 31 | 32 | useEffect(() => { 33 | const fetchBooks = async () => { 34 | const results = await axios.get(fetchUrl); 35 | setBooks(results.data); 36 | setTotalPages( 37 | results.data.length % 10 === 0 38 | ? results.data.length / 10 39 | : parseInt(results.data.length / 10) + 1 40 | ); 41 | setLoaded(true); 42 | }; 43 | fetchBooks(); 44 | }, []); 45 | 46 | return ( 47 |
48 |
49 | 50 | 51 | 52 |
53 | {!loaded ? : books.slice(bookItem, currentPage * 10).map((book) => ( 54 | 62 | ))} 63 |
64 |
65 | 66 |
67 | {!loaded ? : books.slice(bookItem, currentPage * 10).map((book) => ( 68 | 76 | ))} 77 |
78 |
79 |
80 | 85 |
86 |
87 | ); 88 | } 89 | 90 | export default Bookstore; 91 | -------------------------------------------------------------------------------- /frontend/src/components/Home.jsx: -------------------------------------------------------------------------------- 1 | import React from "react"; 2 | 3 | import styles from "./Home.module.css"; 4 | 5 | function Home() { 6 | return ( 7 |
8 |
9 |

Welcome to my personal project: BN Bookstore

10 |
11 |

12 | BN Bookstore is developed using MERN stack (MongoDB, ExpressJS, 13 | ReactJS and NodeJS) to demonstrate the stack's capabilities for 14 | educational purpose. 15 |

16 |

17 | The following list are libraries and frameworks which were used in this project: 18 |

19 | 43 |

44 | BN Bookstore provides different functions for customers and admins, 45 | including: 46 |

47 |
    48 |
  • 49 | Customers: Sign In, Sign Up, browse books, search books by title, view detail information 50 | of a book, add books to cart, cart management, submit books order 51 |
  • 52 |
  • Admin:
  • 53 |
      54 |
    • 55 | There is only one Admin account available. Admin sign in page 56 | can be access via domain.com/admin/signin 57 |
    • 58 |
    • 59 | After logging in, the admin is directed to a specific page to 60 | view all the available invoices 61 |
    • 62 |
    63 |
  • Authentication: Salt and hash
  • 64 |
  • Authorization: JWT (JSON Web Token, the token expires in 1 hour)
  • 65 |
66 |
67 |

68 | 69 | Notes: Beautiful and responsive CSS effects is not the purpose of 70 | this project 71 | 72 |

73 |
74 |
75 |

Developer Contact:

76 |

77 | Email: linhtch90@gmail.com 78 |

79 |

80 | Github:{" "} 81 | https://github.com/linhtch90 82 |

83 |
84 |
85 | ); 86 | } 87 | 88 | export default Home; 89 | -------------------------------------------------------------------------------- /frontend/src/components/CartItem.jsx: -------------------------------------------------------------------------------- 1 | import React, { useState } from "react"; 2 | import { useDispatch } from "react-redux"; 3 | 4 | import { Button } from "primereact/button"; 5 | import { InputText } from "primereact/inputtext"; 6 | 7 | import styles from "./CartItem.module.css"; 8 | import { 9 | calcTotalMoney, 10 | changeQuantity, 11 | decreaseQuantity, 12 | increaseQuantity, 13 | removeBookFromCart, 14 | } from "../features/orderBooks/orderBooksSlice"; 15 | 16 | function CartItem(props) { 17 | const dispatch = useDispatch(); 18 | const book = { 19 | id: props.id, 20 | orderQuantity: props.orderQuantity, 21 | }; 22 | 23 | const [quantity, setQuantity] = useState(1); 24 | 25 | function handleChangeQuantity(e) { 26 | setQuantity(parseInt(e.target.value)); 27 | } 28 | 29 | function selectQuantity(e) { 30 | if (e.key === "Enter") { 31 | if (e.target.value >= 1 && e.target.value <= 99) { 32 | setQuantity(parseInt(e.target.value)); 33 | book.orderQuantity = parseInt(quantity); 34 | dispatch(changeQuantity(book)); 35 | dispatch(calcTotalMoney()); 36 | } else { 37 | setQuantity(1); 38 | book.orderQuantity = parseInt(quantity); 39 | dispatch(changeQuantity(book)); 40 | dispatch(calcTotalMoney()); 41 | } 42 | } 43 | } 44 | 45 | return ( 46 |
47 |
48 |
49 | {props.title} 50 |
51 |
52 |
{props.title}
53 |
Author: {props.author}
54 |
Price: ${props.price}
55 |
56 | {props.orderQuantity <= 1 ? ( 57 |
94 |
95 |
96 |
97 | ); 98 | } 99 | 100 | export default CartItem; 101 | -------------------------------------------------------------------------------- /frontend/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 | ### `npm 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 | ### `npm 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 | ### `npm run 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 | ### `npm run 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 | ### `npm run 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 | -------------------------------------------------------------------------------- /frontend/src/components/NavBar.jsx: -------------------------------------------------------------------------------- 1 | import React from "react"; 2 | import logoPicture from "../assets/bn_bookstore_logo.png"; 3 | import styles from "./NavBar.module.css"; 4 | import { Link, useHistory } from "react-router-dom"; 5 | 6 | import { useDispatch, useSelector } from "react-redux"; 7 | import { userSignOut } from "../features/orderBooks/userSlice"; 8 | import { cleanCart } from "../features/orderBooks/orderBooksSlice"; 9 | 10 | import { Button } from "primereact/button"; 11 | import { Badge } from "primereact/badge"; 12 | import { SplitButton } from "primereact/splitbutton"; 13 | 14 | function NavBar() { 15 | const dispatch = useDispatch(); 16 | let cartCount = useSelector((state) => state.orderBooks.orderBooks.length); 17 | let isSignIn = useSelector((state) => state.userManagement.isSignIn); 18 | 19 | let history = useHistory(); 20 | 21 | const userAccountItems = [ 22 | { 23 | label: "Sign Out", 24 | icon: "pi pi-sign-out", 25 | command: () => { 26 | window.localStorage.removeItem("bnToken"); 27 | window.localStorage.removeItem("bnUserID"); 28 | dispatch(cleanCart()); 29 | dispatch(userSignOut()); 30 | history.push("/"); 31 | }, 32 | }, 33 | ]; 34 | 35 | function navigateCart() { 36 | if (cartCount > 0) { 37 | history.push("/cart"); 38 | } 39 | } 40 | 41 | const userSplitButton = ( 42 |
43 | 48 |
49 | ); 50 | 51 | const userAccountButtons = ( 52 |
53 | 54 |
68 | ); 69 | 70 | return ( 71 |
72 |
73 |
74 |
75 | BN Bookstore Logo 76 |
77 |
78 |
79 | 80 |
Home
81 | 82 | 83 | 87 |
Bookstore
88 | 89 | 90 | 94 |
Contact
95 | 96 |
97 |
98 | {isSignIn ? userSplitButton : userAccountButtons} 99 |
100 |
101 |
102 | 106 | {cartCount > 0 ? ( 107 | 111 | ) : ( 112 | "" 113 | )} 114 | 115 |
116 |
117 |
118 |
119 | ); 120 | } 121 | 122 | export default NavBar; 123 | -------------------------------------------------------------------------------- /frontend/src/components/SignUpYup.jsx: -------------------------------------------------------------------------------- 1 | import React from "react"; 2 | import { Formik, Field, Form, ErrorMessage } from "formik"; 3 | import * as Yup from "yup"; 4 | 5 | import { Button } from "primereact/button"; 6 | 7 | import styles from "./SignUpYup.module.css"; 8 | 9 | function SignUpYup() { 10 | return ( 11 |
12 |
13 | 14 |

User Sign Up

15 | { 45 | setTimeout(() => { 46 | alert(JSON.stringify(values, null, 2)); 47 | setSubmitting(false); 48 | }, 400); 49 | }} 50 | > 51 |
52 | 55 | 60 | 61 |
62 | 63 | 64 | 65 | 66 |
67 | 68 | 69 | 70 | 71 |
72 | 73 | 74 | 79 | 80 |
81 | 82 | 83 | 88 | 89 |
90 | 91 | 92 | 93 |
94 |
95 |
96 |
97 | ); 98 | } 99 | 100 | export default SignUpYup; 101 | -------------------------------------------------------------------------------- /frontend/src/components/SignIn.jsx: -------------------------------------------------------------------------------- 1 | import React, { useState, useRef } from "react"; 2 | import { useFormik } from "formik"; 3 | import axios from "axios"; 4 | import {useHistory} from "react-router-dom"; 5 | 6 | import { useDispatch } from "react-redux"; 7 | import { userSignIn } from "../features/orderBooks/userSlice"; 8 | 9 | import { InputText } from "primereact/inputtext"; 10 | import { Button } from "primereact/button"; 11 | import { Password } from "primereact/password"; 12 | import { Toast } from "primereact/toast"; 13 | import { classNames } from "primereact/utils"; 14 | 15 | import styles from "./SignIn.module.css"; 16 | 17 | function SignIn() { 18 | const history = useHistory(); 19 | const dispatch = useDispatch(); 20 | 21 | const toast = useRef(null); 22 | 23 | const [formData, setFormData] = useState({}); 24 | 25 | async function postSignIn(userData) { 26 | try { 27 | let response = await axios({ 28 | method: "post", 29 | url: "/user/signin", 30 | data: { 31 | email: userData.email, 32 | password: userData.password, 33 | }, 34 | headers: { 35 | "Content-Type": "application/json", 36 | }, 37 | }); 38 | if (response.data.signInStatus === "success") { 39 | window.localStorage.setItem("bnToken", response.data.token); 40 | window.localStorage.setItem("bnUserID", response.data.id); 41 | dispatch(userSignIn()); 42 | showSuccess(); 43 | history.push("/"); 44 | } 45 | } catch (err) { 46 | showFail(); 47 | } 48 | } 49 | 50 | const showSuccess = () => { 51 | toast.current.show({ 52 | severity: "success", 53 | summary: "Welcome!", 54 | detail: "Signed in successfully", 55 | life: 2500, 56 | }); 57 | }; 58 | 59 | const showFail = () => { 60 | toast.current.show({ 61 | severity: "error", 62 | summary: "Oops!!!", 63 | detail: "Sign in fails", 64 | life: 2500, 65 | }); 66 | }; 67 | 68 | const formik = useFormik({ 69 | initialValues: { 70 | email: "", 71 | password: "", 72 | }, 73 | validate: (data) => { 74 | let errors = {}; 75 | 76 | if (!data.email) { 77 | errors.email = "Email is required."; 78 | } else if ( 79 | !/^[A-Z0-9._%+-]+@[A-Z0-9.-]+\.[A-Z]{2,4}$/i.test(data.email) 80 | ) { 81 | errors.email = "Invalid email address. E.g. example@email.com"; 82 | } 83 | 84 | if (!data.password) { 85 | errors.password = "Password is required."; 86 | } 87 | 88 | return errors; 89 | }, 90 | onSubmit: (data) => { 91 | setFormData(data); 92 | postSignIn(data); 93 | formik.resetForm(); 94 | }, 95 | }); 96 | 97 | const isFormFieldValid = (name) => 98 | !!(formik.touched[name] && formik.errors[name]); 99 | const getFormErrorMessage = (name) => { 100 | return ( 101 | isFormFieldValid(name) && ( 102 | {formik.errors[name]} 103 | ) 104 | ); 105 | }; 106 | 107 | return ( 108 |
109 | 110 |
111 |
112 |

User Sign In

113 |
114 |
115 | 116 | 117 | 126 | 134 | 135 | {getFormErrorMessage("email")} 136 |
137 |
138 | 139 | 149 | 157 | 158 | {getFormErrorMessage("password")} 159 |
160 | 161 |
164 |
165 |
166 | ); 167 | } 168 | 169 | export default SignIn; 170 | -------------------------------------------------------------------------------- /frontend/src/components/AdminSignIn.jsx: -------------------------------------------------------------------------------- 1 | import React, { useState, useRef } from "react"; 2 | import { useFormik } from "formik"; 3 | import axios from "axios"; 4 | import { useHistory } from "react-router-dom"; 5 | 6 | import { useDispatch } from "react-redux"; 7 | import { userSignIn } from "../features/orderBooks/userSlice"; 8 | 9 | import { InputText } from "primereact/inputtext"; 10 | import { Button } from "primereact/button"; 11 | import { Password } from "primereact/password"; 12 | import { Toast } from "primereact/toast"; 13 | import { classNames } from "primereact/utils"; 14 | 15 | import styles from "./AdminSignIn.module.css"; 16 | 17 | function AdminSignIn() { 18 | const history = useHistory(); 19 | const dispatch = useDispatch(); 20 | 21 | const toast = useRef(null); 22 | 23 | const [formData, setFormData] = useState({}); 24 | 25 | async function postSignIn(userData) { 26 | try { 27 | let response = await axios({ 28 | method: "post", 29 | url: "/admin/signin", 30 | data: { 31 | email: userData.email, 32 | password: userData.password, 33 | }, 34 | headers: { 35 | "Content-Type": "application/json", 36 | }, 37 | }); 38 | if (response.data.signInStatus === "success") { 39 | window.localStorage.setItem("bnToken", response.data.token); 40 | window.localStorage.setItem("bnUserID", response.data.id); 41 | dispatch(userSignIn()); 42 | showSuccess(); 43 | history.push("/admin/orders"); 44 | } 45 | } catch (err) { 46 | showFail(); 47 | } 48 | } 49 | 50 | const showSuccess = () => { 51 | toast.current.show({ 52 | severity: "success", 53 | summary: "Welcome!", 54 | detail: "Signed in successfully", 55 | life: 2500, 56 | }); 57 | }; 58 | 59 | const showFail = () => { 60 | toast.current.show({ 61 | severity: "error", 62 | summary: "Oops!!!", 63 | detail: "Sign in fails", 64 | life: 2500, 65 | }); 66 | }; 67 | 68 | const formik = useFormik({ 69 | initialValues: { 70 | email: "", 71 | password: "", 72 | }, 73 | validate: (data) => { 74 | let errors = {}; 75 | 76 | if (!data.email) { 77 | errors.email = "Email is required."; 78 | } else if ( 79 | !/^[A-Z0-9._%+-]+@[A-Z0-9.-]+\.[A-Z]{2,4}$/i.test(data.email) 80 | ) { 81 | errors.email = "Invalid email address. E.g. example@email.com"; 82 | } 83 | 84 | if (!data.password) { 85 | errors.password = "Password is required."; 86 | } 87 | 88 | return errors; 89 | }, 90 | onSubmit: (data) => { 91 | setFormData(data); 92 | postSignIn(data); 93 | formik.resetForm(); 94 | }, 95 | }); 96 | 97 | const isFormFieldValid = (name) => 98 | !!(formik.touched[name] && formik.errors[name]); 99 | const getFormErrorMessage = (name) => { 100 | return ( 101 | isFormFieldValid(name) && ( 102 | {formik.errors[name]} 103 | ) 104 | ); 105 | }; 106 | 107 | return ( 108 |
109 | 110 |
111 |
112 |

Admin Sign In

113 |
114 |
115 | 116 | 117 | 126 | 134 | 135 | {getFormErrorMessage("email")} 136 |
137 |
138 | 139 | 149 | 157 | 158 | {getFormErrorMessage("password")} 159 |
160 | 161 |
164 |
165 |
166 | ); 167 | } 168 | 169 | export default AdminSignIn; 170 | -------------------------------------------------------------------------------- /frontend/src/components/SignUp.jsx: -------------------------------------------------------------------------------- 1 | import React, { useState, useRef } from "react"; 2 | import { useFormik } from "formik"; 3 | import axios from "axios"; 4 | 5 | import { InputText } from "primereact/inputtext"; 6 | import { Button } from "primereact/button"; 7 | import { Password } from "primereact/password"; 8 | import { Checkbox } from "primereact/checkbox"; 9 | import { Divider } from "primereact/divider"; 10 | import { Toast } from "primereact/toast"; 11 | import { classNames } from "primereact/utils"; 12 | 13 | import styles from "./SignUp.module.css"; 14 | 15 | function SignUp() { 16 | const toast = useRef(null); 17 | 18 | const [formData, setFormData] = useState({}); 19 | const [successConfirm, setSuccessConfirm] = useState(false); 20 | 21 | async function postNewUser(userData) { 22 | let response = await axios({ 23 | method: "post", 24 | url: "/user/signup", 25 | data: { 26 | name: userData.name, 27 | email: userData.email, 28 | password: userData.password, 29 | }, 30 | headers: { 31 | "Content-Type": "application/json", 32 | }, 33 | }); 34 | if (response.data.createUser === "success") { 35 | setSuccessConfirm(true); 36 | } 37 | } 38 | 39 | const showSuccess = () => { 40 | toast.current.show({ 41 | severity: "success", 42 | summary: "Successful!", 43 | detail: "You can sign in now.", 44 | life: 2500, 45 | }); 46 | }; 47 | 48 | const showFail = () => { 49 | toast.current.show({ 50 | severity: "error", 51 | summary: "Oops!!!", 52 | detail: "User registration fails", 53 | life: 2500, 54 | }); 55 | }; 56 | 57 | const formik = useFormik({ 58 | initialValues: { 59 | name: "", 60 | email: "", 61 | password: "", 62 | confirmPassword: "", 63 | accept: false, 64 | }, 65 | validate: (data) => { 66 | let errors = {}; 67 | 68 | if (!data.name) { 69 | errors.name = "Name is required."; 70 | } 71 | 72 | if (!data.email) { 73 | errors.email = "Email is required."; 74 | } else if ( 75 | !/^[A-Z0-9._%+-]+@[A-Z0-9.-]+\.[A-Z]{2,4}$/i.test(data.email) 76 | ) { 77 | errors.email = "Invalid email address. E.g. example@email.com"; 78 | } 79 | 80 | if (!data.password) { 81 | errors.password = "Password is required."; 82 | } 83 | 84 | if (!data.confirmPassword) { 85 | errors.confirmPassword = "Confirm password is required"; 86 | } 87 | 88 | if (!data.accept) { 89 | errors.accept = "You need to agree to the terms and conditions."; 90 | } 91 | 92 | return errors; 93 | }, 94 | onSubmit: (data) => { 95 | setFormData(data); 96 | postNewUser(data); 97 | formik.resetForm(); 98 | if (successConfirm === true) { 99 | showSuccess(); 100 | } else { 101 | showFail(); 102 | } 103 | }, 104 | }); 105 | 106 | const isFormFieldValid = (name) => 107 | !!(formik.touched[name] && formik.errors[name]); 108 | const getFormErrorMessage = (name) => { 109 | return ( 110 | isFormFieldValid(name) && ( 111 | {formik.errors[name]} 112 | ) 113 | ); 114 | }; 115 | 116 | const passwordHeader =
Pick a password
; 117 | const passwordFooter = ( 118 | 119 | 120 |

Suggestions

121 |
    122 |
  • At least one lowercase
  • 123 |
  • At least one uppercase
  • 124 |
  • At least one numeric
  • 125 |
  • Minimum 8 characters
  • 126 |
127 |
128 | ); 129 | 130 | return ( 131 |
132 | 133 |
134 |
135 |
136 |

137 | User Sign Up 138 |

139 |
140 |
141 | 142 | 152 | 160 | 161 | {getFormErrorMessage("name")} 162 |
163 |
164 | 165 | 166 | 175 | 183 | 184 | {getFormErrorMessage("email")} 185 |
186 |
187 | 188 | 200 | 208 | 209 | {getFormErrorMessage("password")} 210 |
211 | 212 |
213 | 214 | 224 | 232 | 233 | {getFormErrorMessage("confirmPassword")} 234 |
235 | 236 |
237 | 246 | 254 |
255 | 256 |
259 |
260 |
261 |
262 | ); 263 | } 264 | 265 | export default SignUp; 266 | -------------------------------------------------------------------------------- /backend/package-lock.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "backend", 3 | "version": "0.0.0", 4 | "lockfileVersion": 1, 5 | "requires": true, 6 | "dependencies": { 7 | "@sindresorhus/is": { 8 | "version": "0.14.0", 9 | "resolved": "https://registry.npmjs.org/@sindresorhus/is/-/is-0.14.0.tgz", 10 | "integrity": "sha512-9NET910DNaIPngYnLLPeg+Ogzqsi9uM4mSboU5y6p8S5DzMTVEsJZrawi+BoDNUVBa2DhJqQYUFvMDfgU062LQ==", 11 | "dev": true 12 | }, 13 | "@szmarczak/http-timer": { 14 | "version": "1.1.2", 15 | "resolved": "https://registry.npmjs.org/@szmarczak/http-timer/-/http-timer-1.1.2.tgz", 16 | "integrity": "sha512-XIB2XbzHTN6ieIjfIMV9hlVcfPU26s2vafYWQcZHWXHOxiaRZYEDKEwdl129Zyg50+foYV2jCgtrqSA6qNuNSA==", 17 | "dev": true, 18 | "requires": { 19 | "defer-to-connect": "^1.0.1" 20 | } 21 | }, 22 | "@types/node": { 23 | "version": "16.9.1", 24 | "resolved": "https://registry.npmjs.org/@types/node/-/node-16.9.1.tgz", 25 | "integrity": "sha512-QpLcX9ZSsq3YYUUnD3nFDY8H7wctAhQj/TFKL8Ya8v5fMm3CFXxo8zStsLAl780ltoYoo1WvKUVGBQK+1ifr7g==" 26 | }, 27 | "@types/webidl-conversions": { 28 | "version": "6.1.1", 29 | "resolved": "https://registry.npmjs.org/@types/webidl-conversions/-/webidl-conversions-6.1.1.tgz", 30 | "integrity": "sha512-XAahCdThVuCFDQLT7R7Pk/vqeObFNL3YqRyFZg+AqAP/W1/w3xHaIxuW7WszQqTbIBOPRcItYJIou3i/mppu3Q==" 31 | }, 32 | "@types/whatwg-url": { 33 | "version": "8.2.1", 34 | "resolved": "https://registry.npmjs.org/@types/whatwg-url/-/whatwg-url-8.2.1.tgz", 35 | "integrity": "sha512-2YubE1sjj5ifxievI5Ge1sckb9k/Er66HyR2c+3+I6VDUUg1TLPdYYTEbQ+DjRkS4nTxMJhgWfSfMRD2sl2EYQ==", 36 | "requires": { 37 | "@types/node": "*", 38 | "@types/webidl-conversions": "*" 39 | } 40 | }, 41 | "abbrev": { 42 | "version": "1.1.1", 43 | "resolved": "https://registry.npmjs.org/abbrev/-/abbrev-1.1.1.tgz", 44 | "integrity": "sha512-nne9/IiQ/hzIhY6pdDnbBtz7DjPTKrY00P/zvPSm5pOFkl6xuGrGnXn/VtTNNfNtAfZ9/1RtehkszU9qcTii0Q==", 45 | "dev": true 46 | }, 47 | "accepts": { 48 | "version": "1.3.7", 49 | "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.7.tgz", 50 | "integrity": "sha512-Il80Qs2WjYlJIBNzNkK6KYqlVMTbZLXgHx2oT0pU/fjRHyEp+PEfEPY0R3WCwAGVOtauxh1hOxNgIf5bv7dQpA==", 51 | "requires": { 52 | "mime-types": "~2.1.24", 53 | "negotiator": "0.6.2" 54 | } 55 | }, 56 | "acorn": { 57 | "version": "2.7.0", 58 | "resolved": "https://registry.npmjs.org/acorn/-/acorn-2.7.0.tgz", 59 | "integrity": "sha1-q259nYhqrKiwhbwzEreaGYQz8Oc=" 60 | }, 61 | "acorn-globals": { 62 | "version": "1.0.9", 63 | "resolved": "https://registry.npmjs.org/acorn-globals/-/acorn-globals-1.0.9.tgz", 64 | "integrity": "sha1-VbtemGkVB7dFedBRNBMhfDgMVM8=", 65 | "requires": { 66 | "acorn": "^2.1.0" 67 | } 68 | }, 69 | "align-text": { 70 | "version": "0.1.4", 71 | "resolved": "https://registry.npmjs.org/align-text/-/align-text-0.1.4.tgz", 72 | "integrity": "sha1-DNkKVhCT810KmSVsIrcGlDP60Rc=", 73 | "requires": { 74 | "kind-of": "^3.0.2", 75 | "longest": "^1.0.1", 76 | "repeat-string": "^1.5.2" 77 | } 78 | }, 79 | "amdefine": { 80 | "version": "1.0.1", 81 | "resolved": "https://registry.npmjs.org/amdefine/-/amdefine-1.0.1.tgz", 82 | "integrity": "sha1-SlKCrBZHKek2Gbz9OtFR+BfOkfU=" 83 | }, 84 | "ansi-align": { 85 | "version": "3.0.0", 86 | "resolved": "https://registry.npmjs.org/ansi-align/-/ansi-align-3.0.0.tgz", 87 | "integrity": "sha512-ZpClVKqXN3RGBmKibdfWzqCY4lnjEuoNzU5T0oEFpfd/z5qJHVarukridD4juLO2FXMiwUQxr9WqQtaYa8XRYw==", 88 | "dev": true, 89 | "requires": { 90 | "string-width": "^3.0.0" 91 | }, 92 | "dependencies": { 93 | "string-width": { 94 | "version": "3.1.0", 95 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz", 96 | "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==", 97 | "dev": true, 98 | "requires": { 99 | "emoji-regex": "^7.0.1", 100 | "is-fullwidth-code-point": "^2.0.0", 101 | "strip-ansi": "^5.1.0" 102 | } 103 | } 104 | } 105 | }, 106 | "ansi-regex": { 107 | "version": "4.1.0", 108 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz", 109 | "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==", 110 | "dev": true 111 | }, 112 | "ansi-styles": { 113 | "version": "4.3.0", 114 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", 115 | "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", 116 | "dev": true, 117 | "requires": { 118 | "color-convert": "^2.0.1" 119 | } 120 | }, 121 | "anymatch": { 122 | "version": "3.1.2", 123 | "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.2.tgz", 124 | "integrity": "sha512-P43ePfOAIupkguHUycrc4qJ9kz8ZiuOUijaETwX7THt0Y/GNK7v0aa8rY816xWjZ7rJdA5XdMcpVFTKMq+RvWg==", 125 | "dev": true, 126 | "requires": { 127 | "normalize-path": "^3.0.0", 128 | "picomatch": "^2.0.4" 129 | } 130 | }, 131 | "array-flatten": { 132 | "version": "1.1.1", 133 | "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz", 134 | "integrity": "sha1-ml9pkFGx5wczKPKgCJaLZOopVdI=" 135 | }, 136 | "asap": { 137 | "version": "1.0.0", 138 | "resolved": "https://registry.npmjs.org/asap/-/asap-1.0.0.tgz", 139 | "integrity": "sha1-sqRdpf36ILBJb8N2jMJ8EvqRan0=" 140 | }, 141 | "balanced-match": { 142 | "version": "1.0.2", 143 | "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", 144 | "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", 145 | "dev": true 146 | }, 147 | "base64-js": { 148 | "version": "1.5.1", 149 | "resolved": "https://registry.npmjs.org/base64-js/-/base64-js-1.5.1.tgz", 150 | "integrity": "sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==" 151 | }, 152 | "basic-auth": { 153 | "version": "2.0.1", 154 | "resolved": "https://registry.npmjs.org/basic-auth/-/basic-auth-2.0.1.tgz", 155 | "integrity": "sha512-NF+epuEdnUYVlGuhaxbbq+dvJttwLnGY+YixlXlME5KpQ5W3CnXA5cVTneY3SPbPDRkcjMbifrwmFYcClgOZeg==", 156 | "requires": { 157 | "safe-buffer": "5.1.2" 158 | } 159 | }, 160 | "bcryptjs": { 161 | "version": "2.4.3", 162 | "resolved": "https://registry.npmjs.org/bcryptjs/-/bcryptjs-2.4.3.tgz", 163 | "integrity": "sha1-mrVie5PmBiH/fNrF2pczAn3x0Ms=" 164 | }, 165 | "binary-extensions": { 166 | "version": "2.2.0", 167 | "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.2.0.tgz", 168 | "integrity": "sha512-jDctJ/IVQbZoJykoeHbhXpOlNBqGNcwXJKJog42E5HDPUwQTSdjCHdihjj0DlnheQ7blbT6dHOafNAiS8ooQKA==", 169 | "dev": true 170 | }, 171 | "body-parser": { 172 | "version": "1.18.3", 173 | "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.18.3.tgz", 174 | "integrity": "sha1-WykhmP/dVTs6DyDe0FkrlWlVyLQ=", 175 | "requires": { 176 | "bytes": "3.0.0", 177 | "content-type": "~1.0.4", 178 | "debug": "2.6.9", 179 | "depd": "~1.1.2", 180 | "http-errors": "~1.6.3", 181 | "iconv-lite": "0.4.23", 182 | "on-finished": "~2.3.0", 183 | "qs": "6.5.2", 184 | "raw-body": "2.3.3", 185 | "type-is": "~1.6.16" 186 | } 187 | }, 188 | "boxen": { 189 | "version": "4.2.0", 190 | "resolved": "https://registry.npmjs.org/boxen/-/boxen-4.2.0.tgz", 191 | "integrity": "sha512-eB4uT9RGzg2odpER62bBwSLvUeGC+WbRjjyyFhGsKnc8wp/m0+hQsMUvUe3H2V0D5vw0nBdO1hCJoZo5mKeuIQ==", 192 | "dev": true, 193 | "requires": { 194 | "ansi-align": "^3.0.0", 195 | "camelcase": "^5.3.1", 196 | "chalk": "^3.0.0", 197 | "cli-boxes": "^2.2.0", 198 | "string-width": "^4.1.0", 199 | "term-size": "^2.1.0", 200 | "type-fest": "^0.8.1", 201 | "widest-line": "^3.1.0" 202 | }, 203 | "dependencies": { 204 | "camelcase": { 205 | "version": "5.3.1", 206 | "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz", 207 | "integrity": "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==", 208 | "dev": true 209 | } 210 | } 211 | }, 212 | "brace-expansion": { 213 | "version": "1.1.11", 214 | "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", 215 | "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", 216 | "dev": true, 217 | "requires": { 218 | "balanced-match": "^1.0.0", 219 | "concat-map": "0.0.1" 220 | } 221 | }, 222 | "braces": { 223 | "version": "3.0.2", 224 | "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz", 225 | "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", 226 | "dev": true, 227 | "requires": { 228 | "fill-range": "^7.0.1" 229 | } 230 | }, 231 | "bson": { 232 | "version": "4.5.1", 233 | "resolved": "https://registry.npmjs.org/bson/-/bson-4.5.1.tgz", 234 | "integrity": "sha512-XqFP74pbTVLyLy5KFxVfTUyRrC1mgOlmu/iXHfXqfCKT59jyP9lwbotGfbN59cHBRbJSamZNkrSopjv+N0SqAA==", 235 | "requires": { 236 | "buffer": "^5.6.0" 237 | } 238 | }, 239 | "buffer": { 240 | "version": "5.7.1", 241 | "resolved": "https://registry.npmjs.org/buffer/-/buffer-5.7.1.tgz", 242 | "integrity": "sha512-EHcyIPBQ4BSGlvjB16k5KgAJ27CIsHY/2JBmCRReo48y9rQ3MaUzWX3KVlBa4U7MyX02HdVj0K7C3WaB3ju7FQ==", 243 | "requires": { 244 | "base64-js": "^1.3.1", 245 | "ieee754": "^1.1.13" 246 | } 247 | }, 248 | "buffer-equal-constant-time": { 249 | "version": "1.0.1", 250 | "resolved": "https://registry.npmjs.org/buffer-equal-constant-time/-/buffer-equal-constant-time-1.0.1.tgz", 251 | "integrity": "sha1-+OcRMvf/5uAaXJaXpMbz5I1cyBk=" 252 | }, 253 | "bytes": { 254 | "version": "3.0.0", 255 | "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.0.0.tgz", 256 | "integrity": "sha1-0ygVQE1olpn4Wk6k+odV3ROpYEg=" 257 | }, 258 | "cacheable-request": { 259 | "version": "6.1.0", 260 | "resolved": "https://registry.npmjs.org/cacheable-request/-/cacheable-request-6.1.0.tgz", 261 | "integrity": "sha512-Oj3cAGPCqOZX7Rz64Uny2GYAZNliQSqfbePrgAQ1wKAihYmCUnraBtJtKcGR4xz7wF+LoJC+ssFZvv5BgF9Igg==", 262 | "dev": true, 263 | "requires": { 264 | "clone-response": "^1.0.2", 265 | "get-stream": "^5.1.0", 266 | "http-cache-semantics": "^4.0.0", 267 | "keyv": "^3.0.0", 268 | "lowercase-keys": "^2.0.0", 269 | "normalize-url": "^4.1.0", 270 | "responselike": "^1.0.2" 271 | }, 272 | "dependencies": { 273 | "get-stream": { 274 | "version": "5.2.0", 275 | "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-5.2.0.tgz", 276 | "integrity": "sha512-nBF+F1rAZVCu/p7rjzgA+Yb4lfYXrpl7a6VmJrU8wF9I1CKvP/QwPNZHnOlwbTkY6dvtFIzFMSyQXbLoTQPRpA==", 277 | "dev": true, 278 | "requires": { 279 | "pump": "^3.0.0" 280 | } 281 | }, 282 | "lowercase-keys": { 283 | "version": "2.0.0", 284 | "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-2.0.0.tgz", 285 | "integrity": "sha512-tqNXrS78oMOE73NMxK4EMLQsQowWf8jKooH9g7xPavRT706R6bkQJ6DY2Te7QukaZsulxa30wQ7bk0pm4XiHmA==", 286 | "dev": true 287 | } 288 | } 289 | }, 290 | "camelcase": { 291 | "version": "1.2.1", 292 | "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-1.2.1.tgz", 293 | "integrity": "sha1-m7UwTS4LVmmLLHWLCKPqqdqlijk=" 294 | }, 295 | "center-align": { 296 | "version": "0.1.3", 297 | "resolved": "https://registry.npmjs.org/center-align/-/center-align-0.1.3.tgz", 298 | "integrity": "sha1-qg0yYptu6XIgBBHL1EYckHvCt60=", 299 | "requires": { 300 | "align-text": "^0.1.3", 301 | "lazy-cache": "^1.0.3" 302 | } 303 | }, 304 | "chalk": { 305 | "version": "3.0.0", 306 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-3.0.0.tgz", 307 | "integrity": "sha512-4D3B6Wf41KOYRFdszmDqMCGq5VV/uMAB273JILmO+3jAlh8X4qDtdtgCR3fxtbLEMzSx22QdhnDcJvu2u1fVwg==", 308 | "dev": true, 309 | "requires": { 310 | "ansi-styles": "^4.1.0", 311 | "supports-color": "^7.1.0" 312 | }, 313 | "dependencies": { 314 | "has-flag": { 315 | "version": "4.0.0", 316 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", 317 | "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", 318 | "dev": true 319 | }, 320 | "supports-color": { 321 | "version": "7.2.0", 322 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", 323 | "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", 324 | "dev": true, 325 | "requires": { 326 | "has-flag": "^4.0.0" 327 | } 328 | } 329 | } 330 | }, 331 | "character-parser": { 332 | "version": "1.2.1", 333 | "resolved": "https://registry.npmjs.org/character-parser/-/character-parser-1.2.1.tgz", 334 | "integrity": "sha1-wN3kqxgnE7kZuXCVmhI+zBow/NY=" 335 | }, 336 | "chokidar": { 337 | "version": "3.5.2", 338 | "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.5.2.tgz", 339 | "integrity": "sha512-ekGhOnNVPgT77r4K/U3GDhu+FQ2S8TnK/s2KbIGXi0SZWuwkZ2QNyfWdZW+TVfn84DpEP7rLeCt2UI6bJ8GwbQ==", 340 | "dev": true, 341 | "requires": { 342 | "anymatch": "~3.1.2", 343 | "braces": "~3.0.2", 344 | "fsevents": "~2.3.2", 345 | "glob-parent": "~5.1.2", 346 | "is-binary-path": "~2.1.0", 347 | "is-glob": "~4.0.1", 348 | "normalize-path": "~3.0.0", 349 | "readdirp": "~3.6.0" 350 | } 351 | }, 352 | "ci-info": { 353 | "version": "2.0.0", 354 | "resolved": "https://registry.npmjs.org/ci-info/-/ci-info-2.0.0.tgz", 355 | "integrity": "sha512-5tK7EtrZ0N+OLFMthtqOj4fI2Jeb88C4CAZPu25LDVUgXJ0A3Js4PMGqrn0JU1W0Mh1/Z8wZzYPxqUrXeBboCQ==", 356 | "dev": true 357 | }, 358 | "clean-css": { 359 | "version": "3.4.28", 360 | "resolved": "https://registry.npmjs.org/clean-css/-/clean-css-3.4.28.tgz", 361 | "integrity": "sha1-vxlF6C/ICPVWlebd6uwBQA79A/8=", 362 | "requires": { 363 | "commander": "2.8.x", 364 | "source-map": "0.4.x" 365 | }, 366 | "dependencies": { 367 | "commander": { 368 | "version": "2.8.1", 369 | "resolved": "https://registry.npmjs.org/commander/-/commander-2.8.1.tgz", 370 | "integrity": "sha1-Br42f+v9oMMwqh4qBy09yXYkJdQ=", 371 | "requires": { 372 | "graceful-readlink": ">= 1.0.0" 373 | } 374 | } 375 | } 376 | }, 377 | "cli-boxes": { 378 | "version": "2.2.1", 379 | "resolved": "https://registry.npmjs.org/cli-boxes/-/cli-boxes-2.2.1.tgz", 380 | "integrity": "sha512-y4coMcylgSCdVinjiDBuR8PCC2bLjyGTwEmPb9NHR/QaNU6EUOXcTY/s6VjGMD6ENSEaeQYHCY0GNGS5jfMwPw==", 381 | "dev": true 382 | }, 383 | "cliui": { 384 | "version": "2.1.0", 385 | "resolved": "https://registry.npmjs.org/cliui/-/cliui-2.1.0.tgz", 386 | "integrity": "sha1-S0dXYP+AJkx2LDoXGQMukcf+oNE=", 387 | "requires": { 388 | "center-align": "^0.1.1", 389 | "right-align": "^0.1.1", 390 | "wordwrap": "0.0.2" 391 | }, 392 | "dependencies": { 393 | "wordwrap": { 394 | "version": "0.0.2", 395 | "resolved": "https://registry.npmjs.org/wordwrap/-/wordwrap-0.0.2.tgz", 396 | "integrity": "sha1-t5Zpu0LstAn4PVg8rVLKF+qhZD8=" 397 | } 398 | } 399 | }, 400 | "clone-response": { 401 | "version": "1.0.2", 402 | "resolved": "https://registry.npmjs.org/clone-response/-/clone-response-1.0.2.tgz", 403 | "integrity": "sha1-0dyXOSAxTfZ/vrlCI7TuNQI56Ws=", 404 | "dev": true, 405 | "requires": { 406 | "mimic-response": "^1.0.0" 407 | } 408 | }, 409 | "color-convert": { 410 | "version": "2.0.1", 411 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", 412 | "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", 413 | "dev": true, 414 | "requires": { 415 | "color-name": "~1.1.4" 416 | } 417 | }, 418 | "color-name": { 419 | "version": "1.1.4", 420 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", 421 | "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", 422 | "dev": true 423 | }, 424 | "commander": { 425 | "version": "2.6.0", 426 | "resolved": "https://registry.npmjs.org/commander/-/commander-2.6.0.tgz", 427 | "integrity": "sha1-nfflL7Kgyw+4kFjugMMQQiXzfh0=" 428 | }, 429 | "concat-map": { 430 | "version": "0.0.1", 431 | "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", 432 | "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=", 433 | "dev": true 434 | }, 435 | "configstore": { 436 | "version": "5.0.1", 437 | "resolved": "https://registry.npmjs.org/configstore/-/configstore-5.0.1.tgz", 438 | "integrity": "sha512-aMKprgk5YhBNyH25hj8wGt2+D52Sw1DRRIzqBwLp2Ya9mFmY8KPvvtvmna8SxVR9JMZ4kzMD68N22vlaRpkeFA==", 439 | "dev": true, 440 | "requires": { 441 | "dot-prop": "^5.2.0", 442 | "graceful-fs": "^4.1.2", 443 | "make-dir": "^3.0.0", 444 | "unique-string": "^2.0.0", 445 | "write-file-atomic": "^3.0.0", 446 | "xdg-basedir": "^4.0.0" 447 | } 448 | }, 449 | "constantinople": { 450 | "version": "3.0.2", 451 | "resolved": "https://registry.npmjs.org/constantinople/-/constantinople-3.0.2.tgz", 452 | "integrity": "sha1-S5RdmTeQe82Y7ldRIsOBdRZUQUE=", 453 | "requires": { 454 | "acorn": "^2.1.0" 455 | } 456 | }, 457 | "content-disposition": { 458 | "version": "0.5.2", 459 | "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.2.tgz", 460 | "integrity": "sha1-DPaLud318r55YcOoUXjLhdunjLQ=" 461 | }, 462 | "content-type": { 463 | "version": "1.0.4", 464 | "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.4.tgz", 465 | "integrity": "sha512-hIP3EEPs8tB9AT1L+NUqtwOAps4mk2Zob89MWXMHjHWg9milF/j4osnnQLXBCBFBk/tvIG/tUc9mOUJiPBhPXA==" 466 | }, 467 | "cookie": { 468 | "version": "0.4.0", 469 | "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.4.0.tgz", 470 | "integrity": "sha512-+Hp8fLp57wnUSt0tY0tHEXh4voZRDnoIrZPqlo3DPiI4y9lwg/jqx+1Om94/W6ZaPDOUbnjOt/99w66zk+l1Xg==" 471 | }, 472 | "cookie-parser": { 473 | "version": "1.4.5", 474 | "resolved": "https://registry.npmjs.org/cookie-parser/-/cookie-parser-1.4.5.tgz", 475 | "integrity": "sha512-f13bPUj/gG/5mDr+xLmSxxDsB9DQiTIfhJS/sqjrmfAWiAN+x2O4i/XguTL9yDZ+/IFDanJ+5x7hC4CXT9Tdzw==", 476 | "requires": { 477 | "cookie": "0.4.0", 478 | "cookie-signature": "1.0.6" 479 | } 480 | }, 481 | "cookie-signature": { 482 | "version": "1.0.6", 483 | "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz", 484 | "integrity": "sha1-4wOogrNCzD7oylE6eZmXNNqzriw=" 485 | }, 486 | "cors": { 487 | "version": "2.8.5", 488 | "resolved": "https://registry.npmjs.org/cors/-/cors-2.8.5.tgz", 489 | "integrity": "sha512-KIHbLJqu73RGr/hnbrO9uBeixNGuvSQjul/jdFvS/KFSIH1hWVd1ng7zOHx+YrEfInLG7q4n6GHQ9cDtxv/P6g==", 490 | "requires": { 491 | "object-assign": "^4", 492 | "vary": "^1" 493 | } 494 | }, 495 | "crypto-random-string": { 496 | "version": "2.0.0", 497 | "resolved": "https://registry.npmjs.org/crypto-random-string/-/crypto-random-string-2.0.0.tgz", 498 | "integrity": "sha512-v1plID3y9r/lPhviJ1wrXpLeyUIGAZ2SHNYTEapm7/8A9nLPoyvVp3RK/EPFqn5kEznyWgYZNsRtYYIWbuG8KA==", 499 | "dev": true 500 | }, 501 | "css": { 502 | "version": "1.0.8", 503 | "resolved": "https://registry.npmjs.org/css/-/css-1.0.8.tgz", 504 | "integrity": "sha1-k4aBHKgrzMnuf7WnMrHioxfIo+c=", 505 | "requires": { 506 | "css-parse": "1.0.4", 507 | "css-stringify": "1.0.5" 508 | } 509 | }, 510 | "css-parse": { 511 | "version": "1.0.4", 512 | "resolved": "https://registry.npmjs.org/css-parse/-/css-parse-1.0.4.tgz", 513 | "integrity": "sha1-OLBQP7+dqfVOnB29pg4UXHcRe90=" 514 | }, 515 | "css-stringify": { 516 | "version": "1.0.5", 517 | "resolved": "https://registry.npmjs.org/css-stringify/-/css-stringify-1.0.5.tgz", 518 | "integrity": "sha1-sNBClG2ylTu50pKQCmy19tASIDE=" 519 | }, 520 | "debug": { 521 | "version": "2.6.9", 522 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", 523 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", 524 | "requires": { 525 | "ms": "2.0.0" 526 | } 527 | }, 528 | "decamelize": { 529 | "version": "1.2.0", 530 | "resolved": "https://registry.npmjs.org/decamelize/-/decamelize-1.2.0.tgz", 531 | "integrity": "sha1-9lNNFRSCabIDUue+4m9QH5oZEpA=" 532 | }, 533 | "decompress-response": { 534 | "version": "3.3.0", 535 | "resolved": "https://registry.npmjs.org/decompress-response/-/decompress-response-3.3.0.tgz", 536 | "integrity": "sha1-gKTdMjdIOEv6JICDYirt7Jgq3/M=", 537 | "dev": true, 538 | "requires": { 539 | "mimic-response": "^1.0.0" 540 | } 541 | }, 542 | "deep-extend": { 543 | "version": "0.6.0", 544 | "resolved": "https://registry.npmjs.org/deep-extend/-/deep-extend-0.6.0.tgz", 545 | "integrity": "sha512-LOHxIOaPYdHlJRtCQfDIVZtfw/ufM8+rVj649RIHzcm/vGwQRXFt6OPqIFWsm2XEMrNIEtWR64sY1LEKD2vAOA==", 546 | "dev": true 547 | }, 548 | "defer-to-connect": { 549 | "version": "1.1.3", 550 | "resolved": "https://registry.npmjs.org/defer-to-connect/-/defer-to-connect-1.1.3.tgz", 551 | "integrity": "sha512-0ISdNousHvZT2EiFlZeZAHBUvSxmKswVCEf8hW7KWgG4a8MVEu/3Vb6uWYozkjylyCxe0JBIiRB1jV45S70WVQ==", 552 | "dev": true 553 | }, 554 | "denque": { 555 | "version": "1.5.1", 556 | "resolved": "https://registry.npmjs.org/denque/-/denque-1.5.1.tgz", 557 | "integrity": "sha512-XwE+iZ4D6ZUB7mfYRMb5wByE8L74HCn30FBN7sWnXksWc1LO1bPDl67pBR9o/kC4z/xSNAwkMYcGgqDV3BE3Hw==" 558 | }, 559 | "depd": { 560 | "version": "1.1.2", 561 | "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.2.tgz", 562 | "integrity": "sha1-m81S4UwJd2PnSbJ0xDRu0uVgtak=" 563 | }, 564 | "destroy": { 565 | "version": "1.0.4", 566 | "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.0.4.tgz", 567 | "integrity": "sha1-l4hXRCxEdJ5CBmE+N5RiBYJqvYA=" 568 | }, 569 | "dot-prop": { 570 | "version": "5.3.0", 571 | "resolved": "https://registry.npmjs.org/dot-prop/-/dot-prop-5.3.0.tgz", 572 | "integrity": "sha512-QM8q3zDe58hqUqjraQOmzZ1LIH9SWQJTlEKCH4kJ2oQvLZk7RbQXvtDM2XEq3fwkV9CCvvH4LA0AV+ogFsBM2Q==", 573 | "dev": true, 574 | "requires": { 575 | "is-obj": "^2.0.0" 576 | } 577 | }, 578 | "dotenv": { 579 | "version": "10.0.0", 580 | "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-10.0.0.tgz", 581 | "integrity": "sha512-rlBi9d8jpv9Sf1klPjNfFAuWDjKLwTIJJ/VxtoTwIR6hnZxcEOQCZg2oIL3MWBYw5GpUDKOEnND7LXTbIpQ03Q==" 582 | }, 583 | "duplexer3": { 584 | "version": "0.1.4", 585 | "resolved": "https://registry.npmjs.org/duplexer3/-/duplexer3-0.1.4.tgz", 586 | "integrity": "sha1-7gHdHKwO08vH/b6jfcCo8c4ALOI=", 587 | "dev": true 588 | }, 589 | "ecdsa-sig-formatter": { 590 | "version": "1.0.11", 591 | "resolved": "https://registry.npmjs.org/ecdsa-sig-formatter/-/ecdsa-sig-formatter-1.0.11.tgz", 592 | "integrity": "sha512-nagl3RYrbNv6kQkeJIpt6NJZy8twLB/2vtz6yN9Z4vRKHN4/QZJIEbqohALSgwKdnksuY3k5Addp5lg8sVoVcQ==", 593 | "requires": { 594 | "safe-buffer": "^5.0.1" 595 | } 596 | }, 597 | "ee-first": { 598 | "version": "1.1.1", 599 | "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", 600 | "integrity": "sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0=" 601 | }, 602 | "emoji-regex": { 603 | "version": "7.0.3", 604 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-7.0.3.tgz", 605 | "integrity": "sha512-CwBLREIQ7LvYFB0WyRvwhq5N5qPhc6PMjD6bYggFlI5YyDgl+0vxq5VHbMOFqLg7hfWzmu8T5Z1QofhmTIhItA==", 606 | "dev": true 607 | }, 608 | "encodeurl": { 609 | "version": "1.0.2", 610 | "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz", 611 | "integrity": "sha1-rT/0yG7C0CkyL1oCw6mmBslbP1k=" 612 | }, 613 | "end-of-stream": { 614 | "version": "1.4.4", 615 | "resolved": "https://registry.npmjs.org/end-of-stream/-/end-of-stream-1.4.4.tgz", 616 | "integrity": "sha512-+uw1inIHVPQoaVuHzRyXd21icM+cnt4CzD5rW+NC1wjOUSTOs+Te7FOv7AhN7vS9x/oIyhLP5PR1H+phQAHu5Q==", 617 | "dev": true, 618 | "requires": { 619 | "once": "^1.4.0" 620 | } 621 | }, 622 | "escape-goat": { 623 | "version": "2.1.1", 624 | "resolved": "https://registry.npmjs.org/escape-goat/-/escape-goat-2.1.1.tgz", 625 | "integrity": "sha512-8/uIhbG12Csjy2JEW7D9pHbreaVaS/OpN3ycnyvElTdwM5n6GY6W6e2IPemfvGZeUMqZ9A/3GqIZMgKnBhAw/Q==", 626 | "dev": true 627 | }, 628 | "escape-html": { 629 | "version": "1.0.3", 630 | "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz", 631 | "integrity": "sha1-Aljq5NPQwJdN4cFpGI7wBR0dGYg=" 632 | }, 633 | "etag": { 634 | "version": "1.8.1", 635 | "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz", 636 | "integrity": "sha1-Qa4u62XvpiJorr/qg6x9eSmbCIc=" 637 | }, 638 | "express": { 639 | "version": "4.16.4", 640 | "resolved": "https://registry.npmjs.org/express/-/express-4.16.4.tgz", 641 | "integrity": "sha512-j12Uuyb4FMrd/qQAm6uCHAkPtO8FDTRJZBDd5D2KOL2eLaz1yUNdUB/NOIyq0iU4q4cFarsUCrnFDPBcnksuOg==", 642 | "requires": { 643 | "accepts": "~1.3.5", 644 | "array-flatten": "1.1.1", 645 | "body-parser": "1.18.3", 646 | "content-disposition": "0.5.2", 647 | "content-type": "~1.0.4", 648 | "cookie": "0.3.1", 649 | "cookie-signature": "1.0.6", 650 | "debug": "2.6.9", 651 | "depd": "~1.1.2", 652 | "encodeurl": "~1.0.2", 653 | "escape-html": "~1.0.3", 654 | "etag": "~1.8.1", 655 | "finalhandler": "1.1.1", 656 | "fresh": "0.5.2", 657 | "merge-descriptors": "1.0.1", 658 | "methods": "~1.1.2", 659 | "on-finished": "~2.3.0", 660 | "parseurl": "~1.3.2", 661 | "path-to-regexp": "0.1.7", 662 | "proxy-addr": "~2.0.4", 663 | "qs": "6.5.2", 664 | "range-parser": "~1.2.0", 665 | "safe-buffer": "5.1.2", 666 | "send": "0.16.2", 667 | "serve-static": "1.13.2", 668 | "setprototypeof": "1.1.0", 669 | "statuses": "~1.4.0", 670 | "type-is": "~1.6.16", 671 | "utils-merge": "1.0.1", 672 | "vary": "~1.1.2" 673 | }, 674 | "dependencies": { 675 | "cookie": { 676 | "version": "0.3.1", 677 | "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.3.1.tgz", 678 | "integrity": "sha1-5+Ch+e9DtMi6klxcWpboBtFoc7s=" 679 | } 680 | } 681 | }, 682 | "fill-range": { 683 | "version": "7.0.1", 684 | "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz", 685 | "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", 686 | "dev": true, 687 | "requires": { 688 | "to-regex-range": "^5.0.1" 689 | } 690 | }, 691 | "finalhandler": { 692 | "version": "1.1.1", 693 | "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.1.1.tgz", 694 | "integrity": "sha512-Y1GUDo39ez4aHAw7MysnUD5JzYX+WaIj8I57kO3aEPT1fFRL4sr7mjei97FgnwhAyyzRYmQZaTHb2+9uZ1dPtg==", 695 | "requires": { 696 | "debug": "2.6.9", 697 | "encodeurl": "~1.0.2", 698 | "escape-html": "~1.0.3", 699 | "on-finished": "~2.3.0", 700 | "parseurl": "~1.3.2", 701 | "statuses": "~1.4.0", 702 | "unpipe": "~1.0.0" 703 | } 704 | }, 705 | "forwarded": { 706 | "version": "0.2.0", 707 | "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.2.0.tgz", 708 | "integrity": "sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow==" 709 | }, 710 | "fresh": { 711 | "version": "0.5.2", 712 | "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz", 713 | "integrity": "sha1-PYyt2Q2XZWn6g1qx+OSyOhBWBac=" 714 | }, 715 | "fsevents": { 716 | "version": "2.3.2", 717 | "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.2.tgz", 718 | "integrity": "sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==", 719 | "dev": true, 720 | "optional": true 721 | }, 722 | "get-stream": { 723 | "version": "4.1.0", 724 | "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-4.1.0.tgz", 725 | "integrity": "sha512-GMat4EJ5161kIy2HevLlr4luNjBgvmj413KaQA7jt4V8B4RDsfpHk7WQ9GVqfYyyx8OS/L66Kox+rJRNklLK7w==", 726 | "dev": true, 727 | "requires": { 728 | "pump": "^3.0.0" 729 | } 730 | }, 731 | "glob-parent": { 732 | "version": "5.1.2", 733 | "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", 734 | "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", 735 | "dev": true, 736 | "requires": { 737 | "is-glob": "^4.0.1" 738 | } 739 | }, 740 | "global-dirs": { 741 | "version": "2.1.0", 742 | "resolved": "https://registry.npmjs.org/global-dirs/-/global-dirs-2.1.0.tgz", 743 | "integrity": "sha512-MG6kdOUh/xBnyo9cJFeIKkLEc1AyFq42QTU4XiX51i2NEdxLxLWXIjEjmqKeSuKR7pAZjTqUVoT2b2huxVLgYQ==", 744 | "dev": true, 745 | "requires": { 746 | "ini": "1.3.7" 747 | } 748 | }, 749 | "got": { 750 | "version": "9.6.0", 751 | "resolved": "https://registry.npmjs.org/got/-/got-9.6.0.tgz", 752 | "integrity": "sha512-R7eWptXuGYxwijs0eV+v3o6+XH1IqVK8dJOEecQfTmkncw9AV4dcw/Dhxi8MdlqPthxxpZyizMzyg8RTmEsG+Q==", 753 | "dev": true, 754 | "requires": { 755 | "@sindresorhus/is": "^0.14.0", 756 | "@szmarczak/http-timer": "^1.1.2", 757 | "cacheable-request": "^6.0.0", 758 | "decompress-response": "^3.3.0", 759 | "duplexer3": "^0.1.4", 760 | "get-stream": "^4.1.0", 761 | "lowercase-keys": "^1.0.1", 762 | "mimic-response": "^1.0.1", 763 | "p-cancelable": "^1.0.0", 764 | "to-readable-stream": "^1.0.0", 765 | "url-parse-lax": "^3.0.0" 766 | } 767 | }, 768 | "graceful-fs": { 769 | "version": "4.2.8", 770 | "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.8.tgz", 771 | "integrity": "sha512-qkIilPUYcNhJpd33n0GBXTB1MMPp14TxEsEs0pTrsSVucApsYzW5V+Q8Qxhik6KU3evy+qkAAowTByymK0avdg==", 772 | "dev": true 773 | }, 774 | "graceful-readlink": { 775 | "version": "1.0.1", 776 | "resolved": "https://registry.npmjs.org/graceful-readlink/-/graceful-readlink-1.0.1.tgz", 777 | "integrity": "sha1-TK+tdrxi8C+gObL5Tpo906ORpyU=" 778 | }, 779 | "has-flag": { 780 | "version": "3.0.0", 781 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", 782 | "integrity": "sha1-tdRU3CGZriJWmfNGfloH87lVuv0=", 783 | "dev": true 784 | }, 785 | "has-yarn": { 786 | "version": "2.1.0", 787 | "resolved": "https://registry.npmjs.org/has-yarn/-/has-yarn-2.1.0.tgz", 788 | "integrity": "sha512-UqBRqi4ju7T+TqGNdqAO0PaSVGsDGJUBQvk9eUWNGRY1CFGDzYhLWoM7JQEemnlvVcv/YEmc2wNW8BC24EnUsw==", 789 | "dev": true 790 | }, 791 | "http-cache-semantics": { 792 | "version": "4.1.0", 793 | "resolved": "https://registry.npmjs.org/http-cache-semantics/-/http-cache-semantics-4.1.0.tgz", 794 | "integrity": "sha512-carPklcUh7ROWRK7Cv27RPtdhYhUsela/ue5/jKzjegVvXDqM2ILE9Q2BGn9JZJh1g87cp56su/FgQSzcWS8cQ==", 795 | "dev": true 796 | }, 797 | "http-errors": { 798 | "version": "1.6.3", 799 | "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.6.3.tgz", 800 | "integrity": "sha1-i1VoC7S+KDoLW/TqLjhYC+HZMg0=", 801 | "requires": { 802 | "depd": "~1.1.2", 803 | "inherits": "2.0.3", 804 | "setprototypeof": "1.1.0", 805 | "statuses": ">= 1.4.0 < 2" 806 | } 807 | }, 808 | "iconv-lite": { 809 | "version": "0.4.23", 810 | "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.23.tgz", 811 | "integrity": "sha512-neyTUVFtahjf0mB3dZT77u+8O0QB89jFdnBkd5P1JgYPbPaia3gXXOVL2fq8VyU2gMMD7SaN7QukTB/pmXYvDA==", 812 | "requires": { 813 | "safer-buffer": ">= 2.1.2 < 3" 814 | } 815 | }, 816 | "ieee754": { 817 | "version": "1.2.1", 818 | "resolved": "https://registry.npmjs.org/ieee754/-/ieee754-1.2.1.tgz", 819 | "integrity": "sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==" 820 | }, 821 | "ignore-by-default": { 822 | "version": "1.0.1", 823 | "resolved": "https://registry.npmjs.org/ignore-by-default/-/ignore-by-default-1.0.1.tgz", 824 | "integrity": "sha1-SMptcvbGo68Aqa1K5odr44ieKwk=", 825 | "dev": true 826 | }, 827 | "import-lazy": { 828 | "version": "2.1.0", 829 | "resolved": "https://registry.npmjs.org/import-lazy/-/import-lazy-2.1.0.tgz", 830 | "integrity": "sha1-BWmOPUXIjo1+nZLLBYTnfwlvPkM=", 831 | "dev": true 832 | }, 833 | "imurmurhash": { 834 | "version": "0.1.4", 835 | "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", 836 | "integrity": "sha1-khi5srkoojixPcT7a21XbyMUU+o=", 837 | "dev": true 838 | }, 839 | "inherits": { 840 | "version": "2.0.3", 841 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", 842 | "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=" 843 | }, 844 | "ini": { 845 | "version": "1.3.7", 846 | "resolved": "https://registry.npmjs.org/ini/-/ini-1.3.7.tgz", 847 | "integrity": "sha512-iKpRpXP+CrP2jyrxvg1kMUpXDyRUFDWurxbnVT1vQPx+Wz9uCYsMIqYuSBLV+PAaZG/d7kRLKRFc9oDMsH+mFQ==", 848 | "dev": true 849 | }, 850 | "ipaddr.js": { 851 | "version": "1.9.1", 852 | "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz", 853 | "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==" 854 | }, 855 | "is-binary-path": { 856 | "version": "2.1.0", 857 | "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz", 858 | "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==", 859 | "dev": true, 860 | "requires": { 861 | "binary-extensions": "^2.0.0" 862 | } 863 | }, 864 | "is-buffer": { 865 | "version": "1.1.6", 866 | "resolved": "https://registry.npmjs.org/is-buffer/-/is-buffer-1.1.6.tgz", 867 | "integrity": "sha512-NcdALwpXkTm5Zvvbk7owOUSvVvBKDgKP5/ewfXEznmQFfs4ZRmanOeKBTjRVjka3QFoN6XJ+9F3USqfHqTaU5w==" 868 | }, 869 | "is-ci": { 870 | "version": "2.0.0", 871 | "resolved": "https://registry.npmjs.org/is-ci/-/is-ci-2.0.0.tgz", 872 | "integrity": "sha512-YfJT7rkpQB0updsdHLGWrvhBJfcfzNNawYDNIyQXJz0IViGf75O8EBPKSdvw2rF+LGCsX4FZ8tcr3b19LcZq4w==", 873 | "dev": true, 874 | "requires": { 875 | "ci-info": "^2.0.0" 876 | } 877 | }, 878 | "is-extglob": { 879 | "version": "2.1.1", 880 | "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", 881 | "integrity": "sha1-qIwCU1eR8C7TfHahueqXc8gz+MI=", 882 | "dev": true 883 | }, 884 | "is-fullwidth-code-point": { 885 | "version": "2.0.0", 886 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", 887 | "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=", 888 | "dev": true 889 | }, 890 | "is-glob": { 891 | "version": "4.0.1", 892 | "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.1.tgz", 893 | "integrity": "sha512-5G0tKtBTFImOqDnLB2hG6Bp2qcKEFduo4tZu9MT/H6NQv/ghhy30o55ufafxJ/LdH79LLs2Kfrn85TLKyA7BUg==", 894 | "dev": true, 895 | "requires": { 896 | "is-extglob": "^2.1.1" 897 | } 898 | }, 899 | "is-installed-globally": { 900 | "version": "0.3.2", 901 | "resolved": "https://registry.npmjs.org/is-installed-globally/-/is-installed-globally-0.3.2.tgz", 902 | "integrity": "sha512-wZ8x1js7Ia0kecP/CHM/3ABkAmujX7WPvQk6uu3Fly/Mk44pySulQpnHG46OMjHGXApINnV4QhY3SWnECO2z5g==", 903 | "dev": true, 904 | "requires": { 905 | "global-dirs": "^2.0.1", 906 | "is-path-inside": "^3.0.1" 907 | } 908 | }, 909 | "is-npm": { 910 | "version": "4.0.0", 911 | "resolved": "https://registry.npmjs.org/is-npm/-/is-npm-4.0.0.tgz", 912 | "integrity": "sha512-96ECIfh9xtDDlPylNPXhzjsykHsMJZ18ASpaWzQyBr4YRTcVjUvzaHayDAES2oU/3KpljhHUjtSRNiDwi0F0ig==", 913 | "dev": true 914 | }, 915 | "is-number": { 916 | "version": "7.0.0", 917 | "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", 918 | "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", 919 | "dev": true 920 | }, 921 | "is-obj": { 922 | "version": "2.0.0", 923 | "resolved": "https://registry.npmjs.org/is-obj/-/is-obj-2.0.0.tgz", 924 | "integrity": "sha512-drqDG3cbczxxEJRoOXcOjtdp1J/lyp1mNn0xaznRs8+muBhgQcrnbspox5X5fOw0HnMnbfDzvnEMEtqDEJEo8w==", 925 | "dev": true 926 | }, 927 | "is-path-inside": { 928 | "version": "3.0.3", 929 | "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-3.0.3.tgz", 930 | "integrity": "sha512-Fd4gABb+ycGAmKou8eMftCupSir5lRxqf4aD/vd0cD2qc4HL07OjCeuHMr8Ro4CoMaeCKDB0/ECBOVWjTwUvPQ==", 931 | "dev": true 932 | }, 933 | "is-promise": { 934 | "version": "2.2.2", 935 | "resolved": "https://registry.npmjs.org/is-promise/-/is-promise-2.2.2.tgz", 936 | "integrity": "sha512-+lP4/6lKUBfQjZ2pdxThZvLUAafmZb8OAxFb8XXtiQmS35INgr85hdOGoEs124ez1FCnZJt6jau/T+alh58QFQ==" 937 | }, 938 | "is-typedarray": { 939 | "version": "1.0.0", 940 | "resolved": "https://registry.npmjs.org/is-typedarray/-/is-typedarray-1.0.0.tgz", 941 | "integrity": "sha1-5HnICFjfDBsR3dppQPlgEfzaSpo=", 942 | "dev": true 943 | }, 944 | "is-yarn-global": { 945 | "version": "0.3.0", 946 | "resolved": "https://registry.npmjs.org/is-yarn-global/-/is-yarn-global-0.3.0.tgz", 947 | "integrity": "sha512-VjSeb/lHmkoyd8ryPVIKvOCn4D1koMqY+vqyjjUfc3xyKtP4dYOxM44sZrnqQSzSds3xyOrUTLTC9LVCVgLngw==", 948 | "dev": true 949 | }, 950 | "jade": { 951 | "version": "1.11.0", 952 | "resolved": "https://registry.npmjs.org/jade/-/jade-1.11.0.tgz", 953 | "integrity": "sha1-nIDlOMEtP7lcjZu5VZ+gzAQEBf0=", 954 | "requires": { 955 | "character-parser": "1.2.1", 956 | "clean-css": "^3.1.9", 957 | "commander": "~2.6.0", 958 | "constantinople": "~3.0.1", 959 | "jstransformer": "0.0.2", 960 | "mkdirp": "~0.5.0", 961 | "transformers": "2.1.0", 962 | "uglify-js": "^2.4.19", 963 | "void-elements": "~2.0.1", 964 | "with": "~4.0.0" 965 | } 966 | }, 967 | "json-buffer": { 968 | "version": "3.0.0", 969 | "resolved": "https://registry.npmjs.org/json-buffer/-/json-buffer-3.0.0.tgz", 970 | "integrity": "sha1-Wx85evx11ne96Lz8Dkfh+aPZqJg=", 971 | "dev": true 972 | }, 973 | "jsonwebtoken": { 974 | "version": "8.5.1", 975 | "resolved": "https://registry.npmjs.org/jsonwebtoken/-/jsonwebtoken-8.5.1.tgz", 976 | "integrity": "sha512-XjwVfRS6jTMsqYs0EsuJ4LGxXV14zQybNd4L2r0UvbVnSF9Af8x7p5MzbJ90Ioz/9TI41/hTCvznF/loiSzn8w==", 977 | "requires": { 978 | "jws": "^3.2.2", 979 | "lodash.includes": "^4.3.0", 980 | "lodash.isboolean": "^3.0.3", 981 | "lodash.isinteger": "^4.0.4", 982 | "lodash.isnumber": "^3.0.3", 983 | "lodash.isplainobject": "^4.0.6", 984 | "lodash.isstring": "^4.0.1", 985 | "lodash.once": "^4.0.0", 986 | "ms": "^2.1.1", 987 | "semver": "^5.6.0" 988 | }, 989 | "dependencies": { 990 | "ms": { 991 | "version": "2.1.3", 992 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", 993 | "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==" 994 | } 995 | } 996 | }, 997 | "jstransformer": { 998 | "version": "0.0.2", 999 | "resolved": "https://registry.npmjs.org/jstransformer/-/jstransformer-0.0.2.tgz", 1000 | "integrity": "sha1-eq4pqQPRls+glz2IXT5HlH7Ndqs=", 1001 | "requires": { 1002 | "is-promise": "^2.0.0", 1003 | "promise": "^6.0.1" 1004 | } 1005 | }, 1006 | "jwa": { 1007 | "version": "1.4.1", 1008 | "resolved": "https://registry.npmjs.org/jwa/-/jwa-1.4.1.tgz", 1009 | "integrity": "sha512-qiLX/xhEEFKUAJ6FiBMbes3w9ATzyk5W7Hvzpa/SLYdxNtng+gcurvrI7TbACjIXlsJyr05/S1oUhZrc63evQA==", 1010 | "requires": { 1011 | "buffer-equal-constant-time": "1.0.1", 1012 | "ecdsa-sig-formatter": "1.0.11", 1013 | "safe-buffer": "^5.0.1" 1014 | } 1015 | }, 1016 | "jws": { 1017 | "version": "3.2.2", 1018 | "resolved": "https://registry.npmjs.org/jws/-/jws-3.2.2.tgz", 1019 | "integrity": "sha512-YHlZCB6lMTllWDtSPHz/ZXTsi8S00usEV6v1tjq8tOUZzw7DpSDWVXjXDre6ed1w/pd495ODpHZYSdkRTsa0HA==", 1020 | "requires": { 1021 | "jwa": "^1.4.1", 1022 | "safe-buffer": "^5.0.1" 1023 | } 1024 | }, 1025 | "kareem": { 1026 | "version": "2.3.2", 1027 | "resolved": "https://registry.npmjs.org/kareem/-/kareem-2.3.2.tgz", 1028 | "integrity": "sha512-STHz9P7X2L4Kwn72fA4rGyqyXdmrMSdxqHx9IXon/FXluXieaFA6KJ2upcHAHxQPQ0LeM/OjLrhFxifHewOALQ==" 1029 | }, 1030 | "keyv": { 1031 | "version": "3.1.0", 1032 | "resolved": "https://registry.npmjs.org/keyv/-/keyv-3.1.0.tgz", 1033 | "integrity": "sha512-9ykJ/46SN/9KPM/sichzQ7OvXyGDYKGTaDlKMGCAlg2UK8KRy4jb0d8sFc+0Tt0YYnThq8X2RZgCg74RPxgcVA==", 1034 | "dev": true, 1035 | "requires": { 1036 | "json-buffer": "3.0.0" 1037 | } 1038 | }, 1039 | "kind-of": { 1040 | "version": "3.2.2", 1041 | "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", 1042 | "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", 1043 | "requires": { 1044 | "is-buffer": "^1.1.5" 1045 | } 1046 | }, 1047 | "latest-version": { 1048 | "version": "5.1.0", 1049 | "resolved": "https://registry.npmjs.org/latest-version/-/latest-version-5.1.0.tgz", 1050 | "integrity": "sha512-weT+r0kTkRQdCdYCNtkMwWXQTMEswKrFBkm4ckQOMVhhqhIMI1UT2hMj+1iigIhgSZm5gTmrRXBNoGUgaTY1xA==", 1051 | "dev": true, 1052 | "requires": { 1053 | "package-json": "^6.3.0" 1054 | } 1055 | }, 1056 | "lazy-cache": { 1057 | "version": "1.0.4", 1058 | "resolved": "https://registry.npmjs.org/lazy-cache/-/lazy-cache-1.0.4.tgz", 1059 | "integrity": "sha1-odePw6UEdMuAhF07O24dpJpEbo4=" 1060 | }, 1061 | "lodash.includes": { 1062 | "version": "4.3.0", 1063 | "resolved": "https://registry.npmjs.org/lodash.includes/-/lodash.includes-4.3.0.tgz", 1064 | "integrity": "sha1-YLuYqHy5I8aMoeUTJUgzFISfVT8=" 1065 | }, 1066 | "lodash.isboolean": { 1067 | "version": "3.0.3", 1068 | "resolved": "https://registry.npmjs.org/lodash.isboolean/-/lodash.isboolean-3.0.3.tgz", 1069 | "integrity": "sha1-bC4XHbKiV82WgC/UOwGyDV9YcPY=" 1070 | }, 1071 | "lodash.isinteger": { 1072 | "version": "4.0.4", 1073 | "resolved": "https://registry.npmjs.org/lodash.isinteger/-/lodash.isinteger-4.0.4.tgz", 1074 | "integrity": "sha1-YZwK89A/iwTDH1iChAt3sRzWg0M=" 1075 | }, 1076 | "lodash.isnumber": { 1077 | "version": "3.0.3", 1078 | "resolved": "https://registry.npmjs.org/lodash.isnumber/-/lodash.isnumber-3.0.3.tgz", 1079 | "integrity": "sha1-POdoEMWSjQM1IwGsKHMX8RwLH/w=" 1080 | }, 1081 | "lodash.isplainobject": { 1082 | "version": "4.0.6", 1083 | "resolved": "https://registry.npmjs.org/lodash.isplainobject/-/lodash.isplainobject-4.0.6.tgz", 1084 | "integrity": "sha1-fFJqUtibRcRcxpC4gWO+BJf1UMs=" 1085 | }, 1086 | "lodash.isstring": { 1087 | "version": "4.0.1", 1088 | "resolved": "https://registry.npmjs.org/lodash.isstring/-/lodash.isstring-4.0.1.tgz", 1089 | "integrity": "sha1-1SfftUVuynzJu5XV2ur4i6VKVFE=" 1090 | }, 1091 | "lodash.once": { 1092 | "version": "4.1.1", 1093 | "resolved": "https://registry.npmjs.org/lodash.once/-/lodash.once-4.1.1.tgz", 1094 | "integrity": "sha1-DdOXEhPHxW34gJd9UEyI+0cal6w=" 1095 | }, 1096 | "longest": { 1097 | "version": "1.0.1", 1098 | "resolved": "https://registry.npmjs.org/longest/-/longest-1.0.1.tgz", 1099 | "integrity": "sha1-MKCy2jj3N3DoKUoNIuZiXtd9AJc=" 1100 | }, 1101 | "lowercase-keys": { 1102 | "version": "1.0.1", 1103 | "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-1.0.1.tgz", 1104 | "integrity": "sha512-G2Lj61tXDnVFFOi8VZds+SoQjtQC3dgokKdDG2mTm1tx4m50NUHBOZSBwQQHyy0V12A0JTG4icfZQH+xPyh8VA==", 1105 | "dev": true 1106 | }, 1107 | "make-dir": { 1108 | "version": "3.1.0", 1109 | "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-3.1.0.tgz", 1110 | "integrity": "sha512-g3FeP20LNwhALb/6Cz6Dd4F2ngze0jz7tbzrD2wAV+o9FeNHe4rL+yK2md0J/fiSf1sa1ADhXqi5+oVwOM/eGw==", 1111 | "dev": true, 1112 | "requires": { 1113 | "semver": "^6.0.0" 1114 | }, 1115 | "dependencies": { 1116 | "semver": { 1117 | "version": "6.3.0", 1118 | "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", 1119 | "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", 1120 | "dev": true 1121 | } 1122 | } 1123 | }, 1124 | "media-typer": { 1125 | "version": "0.3.0", 1126 | "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", 1127 | "integrity": "sha1-hxDXrwqmJvj/+hzgAWhUUmMlV0g=" 1128 | }, 1129 | "memory-pager": { 1130 | "version": "1.5.0", 1131 | "resolved": "https://registry.npmjs.org/memory-pager/-/memory-pager-1.5.0.tgz", 1132 | "integrity": "sha512-ZS4Bp4r/Zoeq6+NLJpP+0Zzm0pR8whtGPf1XExKLJBAczGMnSi3It14OiNCStjQjM6NU1okjQGSxgEZN8eBYKg==", 1133 | "optional": true 1134 | }, 1135 | "merge-descriptors": { 1136 | "version": "1.0.1", 1137 | "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz", 1138 | "integrity": "sha1-sAqqVW3YtEVoFQ7J0blT8/kMu2E=" 1139 | }, 1140 | "methods": { 1141 | "version": "1.1.2", 1142 | "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz", 1143 | "integrity": "sha1-VSmk1nZUE07cxSZmVoNbD4Ua/O4=" 1144 | }, 1145 | "mime": { 1146 | "version": "1.4.1", 1147 | "resolved": "https://registry.npmjs.org/mime/-/mime-1.4.1.tgz", 1148 | "integrity": "sha512-KI1+qOZu5DcW6wayYHSzR/tXKCDC5Om4s1z2QJjDULzLcmf3DvzS7oluY4HCTrc+9FiKmWUgeNLg7W3uIQvxtQ==" 1149 | }, 1150 | "mime-db": { 1151 | "version": "1.49.0", 1152 | "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.49.0.tgz", 1153 | "integrity": "sha512-CIc8j9URtOVApSFCQIF+VBkX1RwXp/oMMOrqdyXSBXq5RWNEsRfyj1kiRnQgmNXmHxPoFIxOroKA3zcU9P+nAA==" 1154 | }, 1155 | "mime-types": { 1156 | "version": "2.1.32", 1157 | "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.32.tgz", 1158 | "integrity": "sha512-hJGaVS4G4c9TSMYh2n6SQAGrC4RnfU+daP8G7cSCmaqNjiOoUY0VHCMS42pxnQmVF1GWwFhbHWn3RIxCqTmZ9A==", 1159 | "requires": { 1160 | "mime-db": "1.49.0" 1161 | } 1162 | }, 1163 | "mimic-response": { 1164 | "version": "1.0.1", 1165 | "resolved": "https://registry.npmjs.org/mimic-response/-/mimic-response-1.0.1.tgz", 1166 | "integrity": "sha512-j5EctnkH7amfV/q5Hgmoal1g2QHFJRraOtmx0JpIqkxhBhI/lJSl1nMpQ45hVarwNETOoWEimndZ4QK0RHxuxQ==", 1167 | "dev": true 1168 | }, 1169 | "minimatch": { 1170 | "version": "3.0.4", 1171 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz", 1172 | "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==", 1173 | "dev": true, 1174 | "requires": { 1175 | "brace-expansion": "^1.1.7" 1176 | } 1177 | }, 1178 | "minimist": { 1179 | "version": "1.2.5", 1180 | "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.5.tgz", 1181 | "integrity": "sha512-FM9nNUYrRBAELZQT3xeZQ7fmMOBg6nWNmJKTcgsJeaLstP/UODVpGsr5OhXhhXg6f+qtJ8uiZ+PUxkDWcgIXLw==" 1182 | }, 1183 | "mkdirp": { 1184 | "version": "0.5.5", 1185 | "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.5.tgz", 1186 | "integrity": "sha512-NKmAlESf6jMGym1++R0Ra7wvhV+wFW63FaSOFPwRahvea0gMUcGUhVeAg/0BC0wiv9ih5NYPB1Wn1UEI1/L+xQ==", 1187 | "requires": { 1188 | "minimist": "^1.2.5" 1189 | } 1190 | }, 1191 | "mongodb": { 1192 | "version": "4.1.1", 1193 | "resolved": "https://registry.npmjs.org/mongodb/-/mongodb-4.1.1.tgz", 1194 | "integrity": "sha512-fbACrWEyvr6yl0sSiCGV0sqEiBwTtDJ8iSojmkDjAfw9JnOZSAkUyv9seFSPYhPPKwxp1PDtyjvBNfMDz0WBLQ==", 1195 | "requires": { 1196 | "bson": "^4.5.1", 1197 | "denque": "^1.5.0", 1198 | "mongodb-connection-string-url": "^2.0.0", 1199 | "saslprep": "^1.0.0" 1200 | } 1201 | }, 1202 | "mongodb-connection-string-url": { 1203 | "version": "2.0.0", 1204 | "resolved": "https://registry.npmjs.org/mongodb-connection-string-url/-/mongodb-connection-string-url-2.0.0.tgz", 1205 | "integrity": "sha512-M0I1vyLoq5+HQTuPSJWbt+hIXsMCfE8sS1fS5mvP9R2DOMoi2ZD32yWqgBIITyu0dFu4qtS50erxKjvUeBiyog==", 1206 | "requires": { 1207 | "@types/whatwg-url": "^8.2.1", 1208 | "whatwg-url": "^9.1.0" 1209 | } 1210 | }, 1211 | "mongoose": { 1212 | "version": "6.0.5", 1213 | "resolved": "https://registry.npmjs.org/mongoose/-/mongoose-6.0.5.tgz", 1214 | "integrity": "sha512-1MoG52oosjEK8z45DHQVbakP6DJG1sbQI/ZASBW8sZRV+rCaG/pC3L3wWjrsiped/2+uhvanWM9C89F2n6bQ3w==", 1215 | "requires": { 1216 | "bson": "^4.2.2", 1217 | "kareem": "2.3.2", 1218 | "mongodb": "4.1.1", 1219 | "mpath": "0.8.4", 1220 | "mquery": "4.0.0", 1221 | "ms": "2.1.2", 1222 | "regexp-clone": "1.0.0", 1223 | "sift": "13.5.2", 1224 | "sliced": "1.0.1" 1225 | }, 1226 | "dependencies": { 1227 | "ms": { 1228 | "version": "2.1.2", 1229 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", 1230 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==" 1231 | } 1232 | } 1233 | }, 1234 | "morgan": { 1235 | "version": "1.9.1", 1236 | "resolved": "https://registry.npmjs.org/morgan/-/morgan-1.9.1.tgz", 1237 | "integrity": "sha512-HQStPIV4y3afTiCYVxirakhlCfGkI161c76kKFca7Fk1JusM//Qeo1ej2XaMniiNeaZklMVrh3vTtIzpzwbpmA==", 1238 | "requires": { 1239 | "basic-auth": "~2.0.0", 1240 | "debug": "2.6.9", 1241 | "depd": "~1.1.2", 1242 | "on-finished": "~2.3.0", 1243 | "on-headers": "~1.0.1" 1244 | } 1245 | }, 1246 | "mpath": { 1247 | "version": "0.8.4", 1248 | "resolved": "https://registry.npmjs.org/mpath/-/mpath-0.8.4.tgz", 1249 | "integrity": "sha512-DTxNZomBcTWlrMW76jy1wvV37X/cNNxPW1y2Jzd4DZkAaC5ZGsm8bfGfNOthcDuRJujXLqiuS6o3Tpy0JEoh7g==" 1250 | }, 1251 | "mquery": { 1252 | "version": "4.0.0", 1253 | "resolved": "https://registry.npmjs.org/mquery/-/mquery-4.0.0.tgz", 1254 | "integrity": "sha512-nGjm89lHja+T/b8cybAby6H0YgA4qYC/lx6UlwvHGqvTq8bDaNeCwl1sY8uRELrNbVWJzIihxVd+vphGGn1vBw==", 1255 | "requires": { 1256 | "debug": "4.x", 1257 | "regexp-clone": "^1.0.0", 1258 | "sliced": "1.0.1" 1259 | }, 1260 | "dependencies": { 1261 | "debug": { 1262 | "version": "4.3.2", 1263 | "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.2.tgz", 1264 | "integrity": "sha512-mOp8wKcvj7XxC78zLgw/ZA+6TSgkoE2C/ienthhRD298T7UNwAg9diBpLRxC0mOezLl4B0xV7M0cCO6P/O0Xhw==", 1265 | "requires": { 1266 | "ms": "2.1.2" 1267 | } 1268 | }, 1269 | "ms": { 1270 | "version": "2.1.2", 1271 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", 1272 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==" 1273 | } 1274 | } 1275 | }, 1276 | "ms": { 1277 | "version": "2.0.0", 1278 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", 1279 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" 1280 | }, 1281 | "negotiator": { 1282 | "version": "0.6.2", 1283 | "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.2.tgz", 1284 | "integrity": "sha512-hZXc7K2e+PgeI1eDBe/10Ard4ekbfrrqG8Ep+8Jmf4JID2bNg7NvCPOZN+kfF574pFQI7mum2AUqDidoKqcTOw==" 1285 | }, 1286 | "nodemon": { 1287 | "version": "2.0.12", 1288 | "resolved": "https://registry.npmjs.org/nodemon/-/nodemon-2.0.12.tgz", 1289 | "integrity": "sha512-egCTmNZdObdBxUBw6ZNwvZ/xzk24CKRs5K6d+5zbmrMr7rOpPmfPeF6OxM3DDpaRx331CQRFEktn+wrFFfBSOA==", 1290 | "dev": true, 1291 | "requires": { 1292 | "chokidar": "^3.2.2", 1293 | "debug": "^3.2.6", 1294 | "ignore-by-default": "^1.0.1", 1295 | "minimatch": "^3.0.4", 1296 | "pstree.remy": "^1.1.7", 1297 | "semver": "^5.7.1", 1298 | "supports-color": "^5.5.0", 1299 | "touch": "^3.1.0", 1300 | "undefsafe": "^2.0.3", 1301 | "update-notifier": "^4.1.0" 1302 | }, 1303 | "dependencies": { 1304 | "debug": { 1305 | "version": "3.2.7", 1306 | "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", 1307 | "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", 1308 | "dev": true, 1309 | "requires": { 1310 | "ms": "^2.1.1" 1311 | } 1312 | }, 1313 | "ms": { 1314 | "version": "2.1.3", 1315 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", 1316 | "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", 1317 | "dev": true 1318 | } 1319 | } 1320 | }, 1321 | "nopt": { 1322 | "version": "1.0.10", 1323 | "resolved": "https://registry.npmjs.org/nopt/-/nopt-1.0.10.tgz", 1324 | "integrity": "sha1-bd0hvSoxQXuScn3Vhfim83YI6+4=", 1325 | "dev": true, 1326 | "requires": { 1327 | "abbrev": "1" 1328 | } 1329 | }, 1330 | "normalize-path": { 1331 | "version": "3.0.0", 1332 | "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", 1333 | "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==", 1334 | "dev": true 1335 | }, 1336 | "normalize-url": { 1337 | "version": "4.5.1", 1338 | "resolved": "https://registry.npmjs.org/normalize-url/-/normalize-url-4.5.1.tgz", 1339 | "integrity": "sha512-9UZCFRHQdNrfTpGg8+1INIg93B6zE0aXMVFkw1WFwvO4SlZywU6aLg5Of0Ap/PgcbSw4LNxvMWXMeugwMCX0AA==", 1340 | "dev": true 1341 | }, 1342 | "object-assign": { 1343 | "version": "4.1.1", 1344 | "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", 1345 | "integrity": "sha1-IQmtx5ZYh8/AXLvUQsrIv7s2CGM=" 1346 | }, 1347 | "on-finished": { 1348 | "version": "2.3.0", 1349 | "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.3.0.tgz", 1350 | "integrity": "sha1-IPEzZIGwg811M3mSoWlxqi2QaUc=", 1351 | "requires": { 1352 | "ee-first": "1.1.1" 1353 | } 1354 | }, 1355 | "on-headers": { 1356 | "version": "1.0.2", 1357 | "resolved": "https://registry.npmjs.org/on-headers/-/on-headers-1.0.2.tgz", 1358 | "integrity": "sha512-pZAE+FJLoyITytdqK0U5s+FIpjN0JP3OzFi/u8Rx+EV5/W+JTWGXG8xFzevE7AjBfDqHv/8vL8qQsIhHnqRkrA==" 1359 | }, 1360 | "once": { 1361 | "version": "1.4.0", 1362 | "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", 1363 | "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=", 1364 | "dev": true, 1365 | "requires": { 1366 | "wrappy": "1" 1367 | } 1368 | }, 1369 | "optimist": { 1370 | "version": "0.3.7", 1371 | "resolved": "https://registry.npmjs.org/optimist/-/optimist-0.3.7.tgz", 1372 | "integrity": "sha1-yQlBrVnkJzMokjB00s8ufLxuwNk=", 1373 | "requires": { 1374 | "wordwrap": "~0.0.2" 1375 | } 1376 | }, 1377 | "p-cancelable": { 1378 | "version": "1.1.0", 1379 | "resolved": "https://registry.npmjs.org/p-cancelable/-/p-cancelable-1.1.0.tgz", 1380 | "integrity": "sha512-s73XxOZ4zpt1edZYZzvhqFa6uvQc1vwUa0K0BdtIZgQMAJj9IbebH+JkgKZc9h+B05PKHLOTl4ajG1BmNrVZlw==", 1381 | "dev": true 1382 | }, 1383 | "package-json": { 1384 | "version": "6.5.0", 1385 | "resolved": "https://registry.npmjs.org/package-json/-/package-json-6.5.0.tgz", 1386 | "integrity": "sha512-k3bdm2n25tkyxcjSKzB5x8kfVxlMdgsbPr0GkZcwHsLpba6cBjqCt1KlcChKEvxHIcTB1FVMuwoijZ26xex5MQ==", 1387 | "dev": true, 1388 | "requires": { 1389 | "got": "^9.6.0", 1390 | "registry-auth-token": "^4.0.0", 1391 | "registry-url": "^5.0.0", 1392 | "semver": "^6.2.0" 1393 | }, 1394 | "dependencies": { 1395 | "semver": { 1396 | "version": "6.3.0", 1397 | "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", 1398 | "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", 1399 | "dev": true 1400 | } 1401 | } 1402 | }, 1403 | "parseurl": { 1404 | "version": "1.3.3", 1405 | "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz", 1406 | "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==" 1407 | }, 1408 | "passport": { 1409 | "version": "0.4.1", 1410 | "resolved": "https://registry.npmjs.org/passport/-/passport-0.4.1.tgz", 1411 | "integrity": "sha512-IxXgZZs8d7uFSt3eqNjM9NQ3g3uQCW5avD8mRNoXV99Yig50vjuaez6dQK2qC0kVWPRTujxY0dWgGfT09adjYg==", 1412 | "requires": { 1413 | "passport-strategy": "1.x.x", 1414 | "pause": "0.0.1" 1415 | } 1416 | }, 1417 | "passport-jwt": { 1418 | "version": "4.0.0", 1419 | "resolved": "https://registry.npmjs.org/passport-jwt/-/passport-jwt-4.0.0.tgz", 1420 | "integrity": "sha512-BwC0n2GP/1hMVjR4QpnvqA61TxenUMlmfNjYNgK0ZAs0HK4SOQkHcSv4L328blNTLtHq7DbmvyNJiH+bn6C5Mg==", 1421 | "requires": { 1422 | "jsonwebtoken": "^8.2.0", 1423 | "passport-strategy": "^1.0.0" 1424 | } 1425 | }, 1426 | "passport-strategy": { 1427 | "version": "1.0.0", 1428 | "resolved": "https://registry.npmjs.org/passport-strategy/-/passport-strategy-1.0.0.tgz", 1429 | "integrity": "sha1-tVOaqPwiWj0a0XlHbd8ja0QPUuQ=" 1430 | }, 1431 | "path-to-regexp": { 1432 | "version": "0.1.7", 1433 | "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz", 1434 | "integrity": "sha1-32BBeABfUi8V60SQ5yR6G/qmf4w=" 1435 | }, 1436 | "pause": { 1437 | "version": "0.0.1", 1438 | "resolved": "https://registry.npmjs.org/pause/-/pause-0.0.1.tgz", 1439 | "integrity": "sha1-HUCLP9t2kjuVQ9lvtMnf1TXZy10=" 1440 | }, 1441 | "picomatch": { 1442 | "version": "2.3.0", 1443 | "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.0.tgz", 1444 | "integrity": "sha512-lY1Q/PiJGC2zOv/z391WOTD+Z02bCgsFfvxoXXf6h7kv9o+WmsmzYqrAwY63sNgOxE4xEdq0WyUnXfKeBrSvYw==", 1445 | "dev": true 1446 | }, 1447 | "prepend-http": { 1448 | "version": "2.0.0", 1449 | "resolved": "https://registry.npmjs.org/prepend-http/-/prepend-http-2.0.0.tgz", 1450 | "integrity": "sha1-6SQ0v6XqjBn0HN/UAddBo8gZ2Jc=", 1451 | "dev": true 1452 | }, 1453 | "promise": { 1454 | "version": "6.1.0", 1455 | "resolved": "https://registry.npmjs.org/promise/-/promise-6.1.0.tgz", 1456 | "integrity": "sha1-LOcp9rlLRcJoka0GAsXJDgTG7vY=", 1457 | "requires": { 1458 | "asap": "~1.0.0" 1459 | } 1460 | }, 1461 | "proxy-addr": { 1462 | "version": "2.0.7", 1463 | "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.7.tgz", 1464 | "integrity": "sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg==", 1465 | "requires": { 1466 | "forwarded": "0.2.0", 1467 | "ipaddr.js": "1.9.1" 1468 | } 1469 | }, 1470 | "pstree.remy": { 1471 | "version": "1.1.8", 1472 | "resolved": "https://registry.npmjs.org/pstree.remy/-/pstree.remy-1.1.8.tgz", 1473 | "integrity": "sha512-77DZwxQmxKnu3aR542U+X8FypNzbfJ+C5XQDk3uWjWxn6151aIMGthWYRXTqT1E5oJvg+ljaa2OJi+VfvCOQ8w==", 1474 | "dev": true 1475 | }, 1476 | "pump": { 1477 | "version": "3.0.0", 1478 | "resolved": "https://registry.npmjs.org/pump/-/pump-3.0.0.tgz", 1479 | "integrity": "sha512-LwZy+p3SFs1Pytd/jYct4wpv49HiYCqd9Rlc5ZVdk0V+8Yzv6jR5Blk3TRmPL1ft69TxP0IMZGJ+WPFU2BFhww==", 1480 | "dev": true, 1481 | "requires": { 1482 | "end-of-stream": "^1.1.0", 1483 | "once": "^1.3.1" 1484 | } 1485 | }, 1486 | "punycode": { 1487 | "version": "2.1.1", 1488 | "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.1.1.tgz", 1489 | "integrity": "sha512-XRsRjdf+j5ml+y/6GKHPZbrF/8p2Yga0JPtdqTIY2Xe5ohJPD9saDJJLPvp9+NSBprVvevdXZybnj2cv8OEd0A==" 1490 | }, 1491 | "pupa": { 1492 | "version": "2.1.1", 1493 | "resolved": "https://registry.npmjs.org/pupa/-/pupa-2.1.1.tgz", 1494 | "integrity": "sha512-l1jNAspIBSFqbT+y+5FosojNpVpF94nlI+wDUpqP9enwOTfHx9f0gh5nB96vl+6yTpsJsypeNrwfzPrKuHB41A==", 1495 | "dev": true, 1496 | "requires": { 1497 | "escape-goat": "^2.0.0" 1498 | } 1499 | }, 1500 | "qs": { 1501 | "version": "6.5.2", 1502 | "resolved": "https://registry.npmjs.org/qs/-/qs-6.5.2.tgz", 1503 | "integrity": "sha512-N5ZAX4/LxJmF+7wN74pUD6qAh9/wnvdQcjq9TZjevvXzSUo7bfmw91saqMjzGS2xq91/odN2dW/WOl7qQHNDGA==" 1504 | }, 1505 | "range-parser": { 1506 | "version": "1.2.1", 1507 | "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz", 1508 | "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==" 1509 | }, 1510 | "raw-body": { 1511 | "version": "2.3.3", 1512 | "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.3.3.tgz", 1513 | "integrity": "sha512-9esiElv1BrZoI3rCDuOuKCBRbuApGGaDPQfjSflGxdy4oyzqghxu6klEkkVIvBje+FF0BX9coEv8KqW6X/7njw==", 1514 | "requires": { 1515 | "bytes": "3.0.0", 1516 | "http-errors": "1.6.3", 1517 | "iconv-lite": "0.4.23", 1518 | "unpipe": "1.0.0" 1519 | } 1520 | }, 1521 | "rc": { 1522 | "version": "1.2.8", 1523 | "resolved": "https://registry.npmjs.org/rc/-/rc-1.2.8.tgz", 1524 | "integrity": "sha512-y3bGgqKj3QBdxLbLkomlohkvsA8gdAiUQlSBJnBhfn+BPxg4bc62d8TcBW15wavDfgexCgccckhcZvywyQYPOw==", 1525 | "dev": true, 1526 | "requires": { 1527 | "deep-extend": "^0.6.0", 1528 | "ini": "~1.3.0", 1529 | "minimist": "^1.2.0", 1530 | "strip-json-comments": "~2.0.1" 1531 | } 1532 | }, 1533 | "readdirp": { 1534 | "version": "3.6.0", 1535 | "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.6.0.tgz", 1536 | "integrity": "sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==", 1537 | "dev": true, 1538 | "requires": { 1539 | "picomatch": "^2.2.1" 1540 | } 1541 | }, 1542 | "regexp-clone": { 1543 | "version": "1.0.0", 1544 | "resolved": "https://registry.npmjs.org/regexp-clone/-/regexp-clone-1.0.0.tgz", 1545 | "integrity": "sha512-TuAasHQNamyyJ2hb97IuBEif4qBHGjPHBS64sZwytpLEqtBQ1gPJTnOaQ6qmpET16cK14kkjbazl6+p0RRv0yw==" 1546 | }, 1547 | "registry-auth-token": { 1548 | "version": "4.2.1", 1549 | "resolved": "https://registry.npmjs.org/registry-auth-token/-/registry-auth-token-4.2.1.tgz", 1550 | "integrity": "sha512-6gkSb4U6aWJB4SF2ZvLb76yCBjcvufXBqvvEx1HbmKPkutswjW1xNVRY0+daljIYRbogN7O0etYSlbiaEQyMyw==", 1551 | "dev": true, 1552 | "requires": { 1553 | "rc": "^1.2.8" 1554 | } 1555 | }, 1556 | "registry-url": { 1557 | "version": "5.1.0", 1558 | "resolved": "https://registry.npmjs.org/registry-url/-/registry-url-5.1.0.tgz", 1559 | "integrity": "sha512-8acYXXTI0AkQv6RAOjE3vOaIXZkT9wo4LOFbBKYQEEnnMNBpKqdUrI6S4NT0KPIo/WVvJ5tE/X5LF/TQUf0ekw==", 1560 | "dev": true, 1561 | "requires": { 1562 | "rc": "^1.2.8" 1563 | } 1564 | }, 1565 | "repeat-string": { 1566 | "version": "1.6.1", 1567 | "resolved": "https://registry.npmjs.org/repeat-string/-/repeat-string-1.6.1.tgz", 1568 | "integrity": "sha1-jcrkcOHIirwtYA//Sndihtp15jc=" 1569 | }, 1570 | "responselike": { 1571 | "version": "1.0.2", 1572 | "resolved": "https://registry.npmjs.org/responselike/-/responselike-1.0.2.tgz", 1573 | "integrity": "sha1-kYcg7ztjHFZCvgaPFa3lpG9Loec=", 1574 | "dev": true, 1575 | "requires": { 1576 | "lowercase-keys": "^1.0.0" 1577 | } 1578 | }, 1579 | "right-align": { 1580 | "version": "0.1.3", 1581 | "resolved": "https://registry.npmjs.org/right-align/-/right-align-0.1.3.tgz", 1582 | "integrity": "sha1-YTObci/mo1FWiSENJOFMlhSGE+8=", 1583 | "requires": { 1584 | "align-text": "^0.1.1" 1585 | } 1586 | }, 1587 | "safe-buffer": { 1588 | "version": "5.1.2", 1589 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", 1590 | "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==" 1591 | }, 1592 | "safer-buffer": { 1593 | "version": "2.1.2", 1594 | "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", 1595 | "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==" 1596 | }, 1597 | "saslprep": { 1598 | "version": "1.0.3", 1599 | "resolved": "https://registry.npmjs.org/saslprep/-/saslprep-1.0.3.tgz", 1600 | "integrity": "sha512-/MY/PEMbk2SuY5sScONwhUDsV2p77Znkb/q3nSVstq/yQzYJOH/Azh29p9oJLsl3LnQwSvZDKagDGBsBwSooag==", 1601 | "optional": true, 1602 | "requires": { 1603 | "sparse-bitfield": "^3.0.3" 1604 | } 1605 | }, 1606 | "semver": { 1607 | "version": "5.7.1", 1608 | "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", 1609 | "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==" 1610 | }, 1611 | "semver-diff": { 1612 | "version": "3.1.1", 1613 | "resolved": "https://registry.npmjs.org/semver-diff/-/semver-diff-3.1.1.tgz", 1614 | "integrity": "sha512-GX0Ix/CJcHyB8c4ykpHGIAvLyOwOobtM/8d+TQkAd81/bEjgPHrfba41Vpesr7jX/t8Uh+R3EX9eAS5be+jQYg==", 1615 | "dev": true, 1616 | "requires": { 1617 | "semver": "^6.3.0" 1618 | }, 1619 | "dependencies": { 1620 | "semver": { 1621 | "version": "6.3.0", 1622 | "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", 1623 | "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", 1624 | "dev": true 1625 | } 1626 | } 1627 | }, 1628 | "send": { 1629 | "version": "0.16.2", 1630 | "resolved": "https://registry.npmjs.org/send/-/send-0.16.2.tgz", 1631 | "integrity": "sha512-E64YFPUssFHEFBvpbbjr44NCLtI1AohxQ8ZSiJjQLskAdKuriYEP6VyGEsRDH8ScozGpkaX1BGvhanqCwkcEZw==", 1632 | "requires": { 1633 | "debug": "2.6.9", 1634 | "depd": "~1.1.2", 1635 | "destroy": "~1.0.4", 1636 | "encodeurl": "~1.0.2", 1637 | "escape-html": "~1.0.3", 1638 | "etag": "~1.8.1", 1639 | "fresh": "0.5.2", 1640 | "http-errors": "~1.6.2", 1641 | "mime": "1.4.1", 1642 | "ms": "2.0.0", 1643 | "on-finished": "~2.3.0", 1644 | "range-parser": "~1.2.0", 1645 | "statuses": "~1.4.0" 1646 | } 1647 | }, 1648 | "serve-static": { 1649 | "version": "1.13.2", 1650 | "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.13.2.tgz", 1651 | "integrity": "sha512-p/tdJrO4U387R9oMjb1oj7qSMaMfmOyd4j9hOFoxZe2baQszgHcSWjuya/CiT5kgZZKRudHNOA0pYXOl8rQ5nw==", 1652 | "requires": { 1653 | "encodeurl": "~1.0.2", 1654 | "escape-html": "~1.0.3", 1655 | "parseurl": "~1.3.2", 1656 | "send": "0.16.2" 1657 | } 1658 | }, 1659 | "setprototypeof": { 1660 | "version": "1.1.0", 1661 | "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.0.tgz", 1662 | "integrity": "sha512-BvE/TwpZX4FXExxOxZyRGQQv651MSwmWKZGqvmPcRIjDqWub67kTKuIMx43cZZrS/cBBzwBcNDWoFxt2XEFIpQ==" 1663 | }, 1664 | "sift": { 1665 | "version": "13.5.2", 1666 | "resolved": "https://registry.npmjs.org/sift/-/sift-13.5.2.tgz", 1667 | "integrity": "sha512-+gxdEOMA2J+AI+fVsCqeNn7Tgx3M9ZN9jdi95939l1IJ8cZsqS8sqpJyOkic2SJk+1+98Uwryt/gL6XDaV+UZA==" 1668 | }, 1669 | "signal-exit": { 1670 | "version": "3.0.3", 1671 | "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.3.tgz", 1672 | "integrity": "sha512-VUJ49FC8U1OxwZLxIbTTrDvLnf/6TDgxZcK8wxR8zs13xpx7xbG60ndBlhNrFi2EMuFRoeDoJO7wthSLq42EjA==", 1673 | "dev": true 1674 | }, 1675 | "sliced": { 1676 | "version": "1.0.1", 1677 | "resolved": "https://registry.npmjs.org/sliced/-/sliced-1.0.1.tgz", 1678 | "integrity": "sha1-CzpmK10Ewxd7GSa+qCsD+Dei70E=" 1679 | }, 1680 | "source-map": { 1681 | "version": "0.4.4", 1682 | "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.4.4.tgz", 1683 | "integrity": "sha1-66T12pwNyZneaAMti092FzZSA2s=", 1684 | "requires": { 1685 | "amdefine": ">=0.0.4" 1686 | } 1687 | }, 1688 | "sparse-bitfield": { 1689 | "version": "3.0.3", 1690 | "resolved": "https://registry.npmjs.org/sparse-bitfield/-/sparse-bitfield-3.0.3.tgz", 1691 | "integrity": "sha1-/0rm5oZWBWuks+eSqzM004JzyhE=", 1692 | "optional": true, 1693 | "requires": { 1694 | "memory-pager": "^1.0.2" 1695 | } 1696 | }, 1697 | "statuses": { 1698 | "version": "1.4.0", 1699 | "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.4.0.tgz", 1700 | "integrity": "sha512-zhSCtt8v2NDrRlPQpCNtw/heZLtfUDqxBM1udqikb/Hbk52LK4nQSwr10u77iopCW5LsyHpuXS0GnEc48mLeew==" 1701 | }, 1702 | "string-width": { 1703 | "version": "4.2.2", 1704 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.2.tgz", 1705 | "integrity": "sha512-XBJbT3N4JhVumXE0eoLU9DCjcaF92KLNqTmFCnG1pf8duUxFGwtP6AD6nkjw9a3IdiRtL3E2w3JDiE/xi3vOeA==", 1706 | "dev": true, 1707 | "requires": { 1708 | "emoji-regex": "^8.0.0", 1709 | "is-fullwidth-code-point": "^3.0.0", 1710 | "strip-ansi": "^6.0.0" 1711 | }, 1712 | "dependencies": { 1713 | "ansi-regex": { 1714 | "version": "5.0.0", 1715 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.0.tgz", 1716 | "integrity": "sha512-bY6fj56OUQ0hU1KjFNDQuJFezqKdrAyFdIevADiqrWHwSlbmBNMHp5ak2f40Pm8JTFyM2mqxkG6ngkHO11f/lg==", 1717 | "dev": true 1718 | }, 1719 | "emoji-regex": { 1720 | "version": "8.0.0", 1721 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", 1722 | "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", 1723 | "dev": true 1724 | }, 1725 | "is-fullwidth-code-point": { 1726 | "version": "3.0.0", 1727 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", 1728 | "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", 1729 | "dev": true 1730 | }, 1731 | "strip-ansi": { 1732 | "version": "6.0.0", 1733 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.0.tgz", 1734 | "integrity": "sha512-AuvKTrTfQNYNIctbR1K/YGTR1756GycPsg7b9bdV9Duqur4gv6aKqHXah67Z8ImS7WEz5QVcOtlfW2rZEugt6w==", 1735 | "dev": true, 1736 | "requires": { 1737 | "ansi-regex": "^5.0.0" 1738 | } 1739 | } 1740 | } 1741 | }, 1742 | "strip-ansi": { 1743 | "version": "5.2.0", 1744 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", 1745 | "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", 1746 | "dev": true, 1747 | "requires": { 1748 | "ansi-regex": "^4.1.0" 1749 | } 1750 | }, 1751 | "strip-json-comments": { 1752 | "version": "2.0.1", 1753 | "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-2.0.1.tgz", 1754 | "integrity": "sha1-PFMZQukIwml8DsNEhYwobHygpgo=", 1755 | "dev": true 1756 | }, 1757 | "supports-color": { 1758 | "version": "5.5.0", 1759 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", 1760 | "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", 1761 | "dev": true, 1762 | "requires": { 1763 | "has-flag": "^3.0.0" 1764 | } 1765 | }, 1766 | "term-size": { 1767 | "version": "2.2.1", 1768 | "resolved": "https://registry.npmjs.org/term-size/-/term-size-2.2.1.tgz", 1769 | "integrity": "sha512-wK0Ri4fOGjv/XPy8SBHZChl8CM7uMc5VML7SqiQ0zG7+J5Vr+RMQDoHa2CNT6KHUnTGIXH34UDMkPzAUyapBZg==", 1770 | "dev": true 1771 | }, 1772 | "to-readable-stream": { 1773 | "version": "1.0.0", 1774 | "resolved": "https://registry.npmjs.org/to-readable-stream/-/to-readable-stream-1.0.0.tgz", 1775 | "integrity": "sha512-Iq25XBt6zD5npPhlLVXGFN3/gyR2/qODcKNNyTMd4vbm39HUaOiAM4PMq0eMVC/Tkxz+Zjdsc55g9yyz+Yq00Q==", 1776 | "dev": true 1777 | }, 1778 | "to-regex-range": { 1779 | "version": "5.0.1", 1780 | "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", 1781 | "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", 1782 | "dev": true, 1783 | "requires": { 1784 | "is-number": "^7.0.0" 1785 | } 1786 | }, 1787 | "touch": { 1788 | "version": "3.1.0", 1789 | "resolved": "https://registry.npmjs.org/touch/-/touch-3.1.0.tgz", 1790 | "integrity": "sha512-WBx8Uy5TLtOSRtIq+M03/sKDrXCLHxwDcquSP2c43Le03/9serjQBIztjRz6FkJez9D/hleyAXTBGLwwZUw9lA==", 1791 | "dev": true, 1792 | "requires": { 1793 | "nopt": "~1.0.10" 1794 | } 1795 | }, 1796 | "tr46": { 1797 | "version": "2.1.0", 1798 | "resolved": "https://registry.npmjs.org/tr46/-/tr46-2.1.0.tgz", 1799 | "integrity": "sha512-15Ih7phfcdP5YxqiB+iDtLoaTz4Nd35+IiAv0kQ5FNKHzXgdWqPoTIqEDDJmXceQt4JZk6lVPT8lnDlPpGDppw==", 1800 | "requires": { 1801 | "punycode": "^2.1.1" 1802 | } 1803 | }, 1804 | "transformers": { 1805 | "version": "2.1.0", 1806 | "resolved": "https://registry.npmjs.org/transformers/-/transformers-2.1.0.tgz", 1807 | "integrity": "sha1-XSPLNVYd2F3Gf7hIIwm0fVPM6ac=", 1808 | "requires": { 1809 | "css": "~1.0.8", 1810 | "promise": "~2.0", 1811 | "uglify-js": "~2.2.5" 1812 | }, 1813 | "dependencies": { 1814 | "is-promise": { 1815 | "version": "1.0.1", 1816 | "resolved": "https://registry.npmjs.org/is-promise/-/is-promise-1.0.1.tgz", 1817 | "integrity": "sha1-MVc3YcBX4zwukaq56W2gjO++duU=" 1818 | }, 1819 | "promise": { 1820 | "version": "2.0.0", 1821 | "resolved": "https://registry.npmjs.org/promise/-/promise-2.0.0.tgz", 1822 | "integrity": "sha1-RmSKqdYFr10ucMMCS/WUNtoCuA4=", 1823 | "requires": { 1824 | "is-promise": "~1" 1825 | } 1826 | }, 1827 | "source-map": { 1828 | "version": "0.1.43", 1829 | "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.1.43.tgz", 1830 | "integrity": "sha1-wkvBRspRfBRx9drL4lcbK3+eM0Y=", 1831 | "requires": { 1832 | "amdefine": ">=0.0.4" 1833 | } 1834 | }, 1835 | "uglify-js": { 1836 | "version": "2.2.5", 1837 | "resolved": "https://registry.npmjs.org/uglify-js/-/uglify-js-2.2.5.tgz", 1838 | "integrity": "sha1-puAqcNg5eSuXgEiLe4sYTAlcmcc=", 1839 | "requires": { 1840 | "optimist": "~0.3.5", 1841 | "source-map": "~0.1.7" 1842 | } 1843 | } 1844 | } 1845 | }, 1846 | "type-fest": { 1847 | "version": "0.8.1", 1848 | "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.8.1.tgz", 1849 | "integrity": "sha512-4dbzIzqvjtgiM5rw1k5rEHtBANKmdudhGyBEajN01fEyhaAIhsoKNy6y7+IN93IfpFtwY9iqi7kD+xwKhQsNJA==", 1850 | "dev": true 1851 | }, 1852 | "type-is": { 1853 | "version": "1.6.18", 1854 | "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz", 1855 | "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==", 1856 | "requires": { 1857 | "media-typer": "0.3.0", 1858 | "mime-types": "~2.1.24" 1859 | } 1860 | }, 1861 | "typedarray-to-buffer": { 1862 | "version": "3.1.5", 1863 | "resolved": "https://registry.npmjs.org/typedarray-to-buffer/-/typedarray-to-buffer-3.1.5.tgz", 1864 | "integrity": "sha512-zdu8XMNEDepKKR+XYOXAVPtWui0ly0NtohUscw+UmaHiAWT8hrV1rr//H6V+0DvJ3OQ19S979M0laLfX8rm82Q==", 1865 | "dev": true, 1866 | "requires": { 1867 | "is-typedarray": "^1.0.0" 1868 | } 1869 | }, 1870 | "uglify-js": { 1871 | "version": "2.8.29", 1872 | "resolved": "https://registry.npmjs.org/uglify-js/-/uglify-js-2.8.29.tgz", 1873 | "integrity": "sha1-KcVzMUgFe7Th913zW3qcty5qWd0=", 1874 | "requires": { 1875 | "source-map": "~0.5.1", 1876 | "uglify-to-browserify": "~1.0.0", 1877 | "yargs": "~3.10.0" 1878 | }, 1879 | "dependencies": { 1880 | "source-map": { 1881 | "version": "0.5.7", 1882 | "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.5.7.tgz", 1883 | "integrity": "sha1-igOdLRAh0i0eoUyA2OpGi6LvP8w=" 1884 | } 1885 | } 1886 | }, 1887 | "uglify-to-browserify": { 1888 | "version": "1.0.2", 1889 | "resolved": "https://registry.npmjs.org/uglify-to-browserify/-/uglify-to-browserify-1.0.2.tgz", 1890 | "integrity": "sha1-bgkk1r2mta/jSeOabWMoUKD4grc=", 1891 | "optional": true 1892 | }, 1893 | "undefsafe": { 1894 | "version": "2.0.3", 1895 | "resolved": "https://registry.npmjs.org/undefsafe/-/undefsafe-2.0.3.tgz", 1896 | "integrity": "sha512-nrXZwwXrD/T/JXeygJqdCO6NZZ1L66HrxM/Z7mIq2oPanoN0F1nLx3lwJMu6AwJY69hdixaFQOuoYsMjE5/C2A==", 1897 | "dev": true, 1898 | "requires": { 1899 | "debug": "^2.2.0" 1900 | } 1901 | }, 1902 | "unique-string": { 1903 | "version": "2.0.0", 1904 | "resolved": "https://registry.npmjs.org/unique-string/-/unique-string-2.0.0.tgz", 1905 | "integrity": "sha512-uNaeirEPvpZWSgzwsPGtU2zVSTrn/8L5q/IexZmH0eH6SA73CmAA5U4GwORTxQAZs95TAXLNqeLoPPNO5gZfWg==", 1906 | "dev": true, 1907 | "requires": { 1908 | "crypto-random-string": "^2.0.0" 1909 | } 1910 | }, 1911 | "unpipe": { 1912 | "version": "1.0.0", 1913 | "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", 1914 | "integrity": "sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw=" 1915 | }, 1916 | "update-notifier": { 1917 | "version": "4.1.3", 1918 | "resolved": "https://registry.npmjs.org/update-notifier/-/update-notifier-4.1.3.tgz", 1919 | "integrity": "sha512-Yld6Z0RyCYGB6ckIjffGOSOmHXj1gMeE7aROz4MG+XMkmixBX4jUngrGXNYz7wPKBmtoD4MnBa2Anu7RSKht/A==", 1920 | "dev": true, 1921 | "requires": { 1922 | "boxen": "^4.2.0", 1923 | "chalk": "^3.0.0", 1924 | "configstore": "^5.0.1", 1925 | "has-yarn": "^2.1.0", 1926 | "import-lazy": "^2.1.0", 1927 | "is-ci": "^2.0.0", 1928 | "is-installed-globally": "^0.3.1", 1929 | "is-npm": "^4.0.0", 1930 | "is-yarn-global": "^0.3.0", 1931 | "latest-version": "^5.0.0", 1932 | "pupa": "^2.0.1", 1933 | "semver-diff": "^3.1.1", 1934 | "xdg-basedir": "^4.0.0" 1935 | } 1936 | }, 1937 | "url-parse-lax": { 1938 | "version": "3.0.0", 1939 | "resolved": "https://registry.npmjs.org/url-parse-lax/-/url-parse-lax-3.0.0.tgz", 1940 | "integrity": "sha1-FrXK/Afb42dsGxmZF3gj1lA6yww=", 1941 | "dev": true, 1942 | "requires": { 1943 | "prepend-http": "^2.0.0" 1944 | } 1945 | }, 1946 | "utils-merge": { 1947 | "version": "1.0.1", 1948 | "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz", 1949 | "integrity": "sha1-n5VxD1CiZ5R7LMwSR0HBAoQn5xM=" 1950 | }, 1951 | "vary": { 1952 | "version": "1.1.2", 1953 | "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz", 1954 | "integrity": "sha1-IpnwLG3tMNSllhsLn3RSShj2NPw=" 1955 | }, 1956 | "void-elements": { 1957 | "version": "2.0.1", 1958 | "resolved": "https://registry.npmjs.org/void-elements/-/void-elements-2.0.1.tgz", 1959 | "integrity": "sha1-wGavtYK7HLQSjWDqkjkulNXp2+w=" 1960 | }, 1961 | "webidl-conversions": { 1962 | "version": "6.1.0", 1963 | "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-6.1.0.tgz", 1964 | "integrity": "sha512-qBIvFLGiBpLjfwmYAaHPXsn+ho5xZnGvyGvsarywGNc8VyQJUMHJ8OBKGGrPER0okBeMDaan4mNBlgBROxuI8w==" 1965 | }, 1966 | "whatwg-url": { 1967 | "version": "9.1.0", 1968 | "resolved": "https://registry.npmjs.org/whatwg-url/-/whatwg-url-9.1.0.tgz", 1969 | "integrity": "sha512-CQ0UcrPHyomtlOCot1TL77WyMIm/bCwrJ2D6AOKGwEczU9EpyoqAokfqrf/MioU9kHcMsmJZcg1egXix2KYEsA==", 1970 | "requires": { 1971 | "tr46": "^2.1.0", 1972 | "webidl-conversions": "^6.1.0" 1973 | } 1974 | }, 1975 | "widest-line": { 1976 | "version": "3.1.0", 1977 | "resolved": "https://registry.npmjs.org/widest-line/-/widest-line-3.1.0.tgz", 1978 | "integrity": "sha512-NsmoXalsWVDMGupxZ5R08ka9flZjjiLvHVAWYOKtiKM8ujtZWr9cRffak+uSE48+Ob8ObalXpwyeUiyDD6QFgg==", 1979 | "dev": true, 1980 | "requires": { 1981 | "string-width": "^4.0.0" 1982 | } 1983 | }, 1984 | "window-size": { 1985 | "version": "0.1.0", 1986 | "resolved": "https://registry.npmjs.org/window-size/-/window-size-0.1.0.tgz", 1987 | "integrity": "sha1-VDjNLqk7IC76Ohn+iIeu58lPnJ0=" 1988 | }, 1989 | "with": { 1990 | "version": "4.0.3", 1991 | "resolved": "https://registry.npmjs.org/with/-/with-4.0.3.tgz", 1992 | "integrity": "sha1-7v0VTp550sjTQXtkeo8U2f7M4U4=", 1993 | "requires": { 1994 | "acorn": "^1.0.1", 1995 | "acorn-globals": "^1.0.3" 1996 | }, 1997 | "dependencies": { 1998 | "acorn": { 1999 | "version": "1.2.2", 2000 | "resolved": "https://registry.npmjs.org/acorn/-/acorn-1.2.2.tgz", 2001 | "integrity": "sha1-yM4n3grMdtiW0rH6099YjZ6C8BQ=" 2002 | } 2003 | } 2004 | }, 2005 | "wordwrap": { 2006 | "version": "0.0.3", 2007 | "resolved": "https://registry.npmjs.org/wordwrap/-/wordwrap-0.0.3.tgz", 2008 | "integrity": "sha1-o9XabNXAvAAI03I0u68b7WMFkQc=" 2009 | }, 2010 | "wrappy": { 2011 | "version": "1.0.2", 2012 | "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", 2013 | "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=", 2014 | "dev": true 2015 | }, 2016 | "write-file-atomic": { 2017 | "version": "3.0.3", 2018 | "resolved": "https://registry.npmjs.org/write-file-atomic/-/write-file-atomic-3.0.3.tgz", 2019 | "integrity": "sha512-AvHcyZ5JnSfq3ioSyjrBkH9yW4m7Ayk8/9My/DD9onKeu/94fwrMocemO2QAJFAlnnDN+ZDS+ZjAR5ua1/PV/Q==", 2020 | "dev": true, 2021 | "requires": { 2022 | "imurmurhash": "^0.1.4", 2023 | "is-typedarray": "^1.0.0", 2024 | "signal-exit": "^3.0.2", 2025 | "typedarray-to-buffer": "^3.1.5" 2026 | } 2027 | }, 2028 | "xdg-basedir": { 2029 | "version": "4.0.0", 2030 | "resolved": "https://registry.npmjs.org/xdg-basedir/-/xdg-basedir-4.0.0.tgz", 2031 | "integrity": "sha512-PSNhEJDejZYV7h50BohL09Er9VaIefr2LMAf3OEmpCkjOi34eYyQYAXUTjEQtZJTKcF0E2UKTh+osDLsgNim9Q==", 2032 | "dev": true 2033 | }, 2034 | "yargs": { 2035 | "version": "3.10.0", 2036 | "resolved": "https://registry.npmjs.org/yargs/-/yargs-3.10.0.tgz", 2037 | "integrity": "sha1-9+572FfdfB0tOMDnTvvWgdFDH9E=", 2038 | "requires": { 2039 | "camelcase": "^1.0.2", 2040 | "cliui": "^2.1.0", 2041 | "decamelize": "^1.0.0", 2042 | "window-size": "0.1.0" 2043 | } 2044 | } 2045 | } 2046 | } 2047 | --------------------------------------------------------------------------------