├── .env
├── .gitignore
├── README.md
├── client
├── package-lock.json
├── package.json
├── public
│ ├── favicon.ico
│ ├── index.html
│ ├── logo192.png
│ ├── logo512.png
│ ├── manifest.json
│ └── robots.txt
└── src
│ ├── App.js
│ ├── GlobalState.js
│ ├── api
│ ├── CategoriesAPI.js
│ ├── ProductsAPI.js
│ └── UserAPI.js
│ ├── components
│ ├── headers
│ │ ├── Header.js
│ │ ├── header.css
│ │ └── icon
│ │ │ ├── cart.svg
│ │ │ ├── close.svg
│ │ │ └── menu.svg
│ └── mainpages
│ │ ├── Pages.js
│ │ ├── auth
│ │ ├── Login.js
│ │ ├── Register.js
│ │ └── login.css
│ │ ├── cart
│ │ ├── Cart.js
│ │ ├── PaypalButton.js
│ │ └── cart.css
│ │ ├── categories
│ │ ├── Categories.js
│ │ └── categories.css
│ │ ├── createProduct
│ │ ├── CreateProduct.js
│ │ └── createProduct.css
│ │ ├── detailProduct
│ │ ├── DetailProduct.js
│ │ └── detailProduct.css
│ │ ├── history
│ │ ├── OrderDetails.js
│ │ ├── OrderHistory.js
│ │ └── history.css
│ │ ├── products
│ │ ├── Filters.js
│ │ ├── LoadMore.js
│ │ ├── Products.js
│ │ └── products.css
│ │ └── utils
│ │ ├── loading
│ │ ├── Loading.js
│ │ └── loading.css
│ │ ├── not_found
│ │ └── NotFound.js
│ │ └── productItem
│ │ ├── BtnRender.js
│ │ ├── ProductItem.js
│ │ └── productItem.css
│ ├── index.css
│ └── index.js
├── controllers
├── categoryCtrl.js
├── paymentCtrl.js
├── productCtrl.js
└── userCtrl.js
├── middleware
├── auth.js
└── authAdmin.js
├── models
├── categoryModel.js
├── paymentModel.js
├── productModel.js
└── userModel.js
├── package-lock.json
├── package.json
├── routes
├── categoryRouter.js
├── paymentRouter.js
├── productRouter.js
├── upload.js
└── userRouter.js
└── server.js
/.env:
--------------------------------------------------------------------------------
1 | MONGODB_URL = YOUR_MONGODB_URL
2 |
3 | CLOUD_API_KEY = YOUR_CLOUD_API_KEY
4 | CLOUD_API_SECRET = YOUR_CLOUD_API_SECRET
5 | CLOUD_NAME = YOUR_CLOUD_NAME
6 |
7 |
8 | REFRESH_TOKEN_SECRET = YOUR_REFRESH_TOKEN_SECRET
9 | ACCESS_TOKEN_SECRET = YOUR_ACCESS_TOKEN_SECRET
10 |
--------------------------------------------------------------------------------
/.gitignore:
--------------------------------------------------------------------------------
1 | node_modules
2 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | # MERN Stack - Ecommerce
2 | > Ecommerce website built with the MERN stack with React Context API for state management, pure CSS for style
3 |
4 | ## Demo: https://devat-ecommerce.herokuapp.com/
5 |
6 | ## Author: Dev AT
7 | ## Youtube tutorials: https://youtu.be/AjItjNc0bHA
8 |
9 | ## Install dependencies for server
10 | ### `npm install`
11 |
12 | ## Install dependencies for client
13 | ### cd client ---> `npm install`
14 |
15 | ## Connect to your mongodb and add info in .env
16 |
17 | ## Add your paypal client id in client/src/components/mainpages/cart/PaypalButton.js
18 |
19 | ## Run the client & server with concurrently
20 | ### `npm run dev`
21 |
22 | ## Run the Express server only
23 | ### `npm run server`
24 |
25 | ## Run the React client only
26 | ### `npm run client`
27 |
28 | ### Server runs on http://localhost:5000 and client on http://localhost:3000
29 |
30 | ### User interface
31 |
32 | 
33 |
34 | ### Admin interface
35 |
36 | 
37 |
--------------------------------------------------------------------------------
/client/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "client",
3 | "version": "0.1.0",
4 | "private": true,
5 | "dependencies": {
6 | "@testing-library/jest-dom": "^4.2.4",
7 | "@testing-library/react": "^9.5.0",
8 | "@testing-library/user-event": "^7.2.1",
9 | "axios": "^0.20.0",
10 | "react": "^16.13.1",
11 | "react-dom": "^16.13.1",
12 | "react-paypal-express-checkout": "^1.0.5",
13 | "react-router-dom": "^5.2.0",
14 | "react-scripts": "3.4.3"
15 | },
16 | "scripts": {
17 | "start": "react-scripts start",
18 | "build": "react-scripts build",
19 | "test": "react-scripts test",
20 | "eject": "react-scripts eject"
21 | },
22 | "eslintConfig": {
23 | "extends": "react-app"
24 | },
25 | "browserslist": {
26 | "production": [
27 | ">0.2%",
28 | "not dead",
29 | "not op_mini all"
30 | ],
31 | "development": [
32 | "last 1 chrome version",
33 | "last 1 firefox version",
34 | "last 1 safari version"
35 | ]
36 | },
37 | "proxy": "http://localhost:5000"
38 | }
39 |
--------------------------------------------------------------------------------
/client/public/favicon.ico:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/devat-youtuber/mern-ecommerce/e1760dedeb43fd9047a5d0f4334f0e75a665e0f7/client/public/favicon.ico
--------------------------------------------------------------------------------
/client/public/index.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
12 |
13 |
17 |
18 |
27 | React App
28 |
29 |
30 |
31 |
32 |
42 |
43 |
44 |
--------------------------------------------------------------------------------
/client/public/logo192.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/devat-youtuber/mern-ecommerce/e1760dedeb43fd9047a5d0f4334f0e75a665e0f7/client/public/logo192.png
--------------------------------------------------------------------------------
/client/public/logo512.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/devat-youtuber/mern-ecommerce/e1760dedeb43fd9047a5d0f4334f0e75a665e0f7/client/public/logo512.png
--------------------------------------------------------------------------------
/client/public/manifest.json:
--------------------------------------------------------------------------------
1 | {
2 | "short_name": "React App",
3 | "name": "Create React App Sample",
4 | "icons": [
5 | {
6 | "src": "favicon.ico",
7 | "sizes": "64x64 32x32 24x24 16x16",
8 | "type": "image/x-icon"
9 | },
10 | {
11 | "src": "logo192.png",
12 | "type": "image/png",
13 | "sizes": "192x192"
14 | },
15 | {
16 | "src": "logo512.png",
17 | "type": "image/png",
18 | "sizes": "512x512"
19 | }
20 | ],
21 | "start_url": ".",
22 | "display": "standalone",
23 | "theme_color": "#000000",
24 | "background_color": "#ffffff"
25 | }
26 |
--------------------------------------------------------------------------------
/client/public/robots.txt:
--------------------------------------------------------------------------------
1 | # https://www.robotstxt.org/robotstxt.html
2 | User-agent: *
3 | Disallow:
4 |
--------------------------------------------------------------------------------
/client/src/App.js:
--------------------------------------------------------------------------------
1 | import React from 'react';
2 | import {BrowserRouter as Router} from 'react-router-dom'
3 | import {DataProvider} from './GlobalState'
4 | import Header from './components/headers/Header'
5 | import MainPages from './components/mainpages/Pages'
6 |
7 |
8 | function App() {
9 | return (
10 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 | );
19 | }
20 |
21 | export default App;
22 |
--------------------------------------------------------------------------------
/client/src/GlobalState.js:
--------------------------------------------------------------------------------
1 | import React, {createContext, useState, useEffect} from 'react'
2 | import ProductsAPI from './api/ProductsAPI'
3 | import UserAPI from './api/UserAPI'
4 | import CategoriesAPI from './api/CategoriesAPI'
5 |
6 | import axios from 'axios'
7 |
8 | export const GlobalState = createContext()
9 |
10 |
11 | export const DataProvider = ({children}) =>{
12 | const [token, setToken] = useState(false)
13 |
14 |
15 | useEffect(() =>{
16 | const firstLogin = localStorage.getItem('firstLogin')
17 | if(firstLogin){
18 | const refreshToken = async () =>{
19 | const res = await axios.get('/user/refresh_token')
20 |
21 | setToken(res.data.accesstoken)
22 |
23 | setTimeout(() => {
24 | refreshToken()
25 | }, 10 * 60 * 1000)
26 | }
27 | refreshToken()
28 | }
29 | },[])
30 |
31 |
32 |
33 | const state = {
34 | token: [token, setToken],
35 | productsAPI: ProductsAPI(),
36 | userAPI: UserAPI(token),
37 | categoriesAPI: CategoriesAPI()
38 | }
39 |
40 | return (
41 |
42 | {children}
43 |
44 | )
45 | }
--------------------------------------------------------------------------------
/client/src/api/CategoriesAPI.js:
--------------------------------------------------------------------------------
1 | import {useState, useEffect} from 'react'
2 | import axios from 'axios'
3 |
4 | function CategoriesAPI() {
5 | const [categories, setCategories] = useState([])
6 | const [callback, setCallback] = useState(false)
7 |
8 | useEffect(() =>{
9 | const getCategories = async () =>{
10 | const res = await axios.get('/api/category')
11 | setCategories(res.data)
12 | }
13 |
14 | getCategories()
15 | },[callback])
16 | return {
17 | categories: [categories, setCategories],
18 | callback: [callback, setCallback]
19 | }
20 | }
21 |
22 | export default CategoriesAPI
23 |
--------------------------------------------------------------------------------
/client/src/api/ProductsAPI.js:
--------------------------------------------------------------------------------
1 | import {useState, useEffect} from 'react'
2 | import axios from 'axios'
3 |
4 |
5 | function ProductsAPI() {
6 | const [products, setProducts] = useState([])
7 | const [callback, setCallback] = useState(false)
8 | const [category, setCategory] = useState('')
9 | const [sort, setSort] = useState('')
10 | const [search, setSearch] = useState('')
11 | const [page, setPage] = useState(1)
12 | const [result, setResult] = useState(0)
13 |
14 | useEffect(() =>{
15 | const getProducts = async () => {
16 | const res = await axios.get(`/api/products?limit=${page*9}&${category}&${sort}&title[regex]=${search}`)
17 | setProducts(res.data.products)
18 | setResult(res.data.result)
19 | }
20 | getProducts()
21 | },[callback, category, sort, search, page])
22 |
23 | return {
24 | products: [products, setProducts],
25 | callback: [callback, setCallback],
26 | category: [category, setCategory],
27 | sort: [sort, setSort],
28 | search: [search, setSearch],
29 | page: [page, setPage],
30 | result: [result, setResult]
31 | }
32 | }
33 |
34 | export default ProductsAPI
35 |
--------------------------------------------------------------------------------
/client/src/api/UserAPI.js:
--------------------------------------------------------------------------------
1 | import {useState, useEffect} from 'react'
2 | import axios from 'axios'
3 |
4 | function UserAPI(token) {
5 | const [isLogged, setIsLogged] = useState(false)
6 | const [isAdmin, setIsAdmin] = useState(false)
7 | const [cart, setCart] = useState([])
8 | const [history, setHistory] = useState([])
9 |
10 | useEffect(() =>{
11 | if(token){
12 | const getUser = async () =>{
13 | try {
14 | const res = await axios.get('/user/infor', {
15 | headers: {Authorization: token}
16 | })
17 |
18 | setIsLogged(true)
19 | res.data.role === 1 ? setIsAdmin(true) : setIsAdmin(false)
20 |
21 | setCart(res.data.cart)
22 |
23 | } catch (err) {
24 | alert(err.response.data.msg)
25 | }
26 | }
27 |
28 | getUser()
29 |
30 | }
31 | },[token])
32 |
33 |
34 |
35 | const addCart = async (product) => {
36 | if(!isLogged) return alert("Please login to continue buying")
37 |
38 | const check = cart.every(item =>{
39 | return item._id !== product._id
40 | })
41 |
42 | if(check){
43 | setCart([...cart, {...product, quantity: 1}])
44 |
45 | await axios.patch('/user/addcart', {cart: [...cart, {...product, quantity: 1}]}, {
46 | headers: {Authorization: token}
47 | })
48 |
49 | }else{
50 | alert("This product has been added to cart.")
51 | }
52 | }
53 |
54 | return {
55 | isLogged: [isLogged, setIsLogged],
56 | isAdmin: [isAdmin, setIsAdmin],
57 | cart: [cart, setCart],
58 | addCart: addCart,
59 | history: [history, setHistory]
60 | }
61 | }
62 |
63 | export default UserAPI
64 |
--------------------------------------------------------------------------------
/client/src/components/headers/Header.js:
--------------------------------------------------------------------------------
1 | import React, {useContext, useState} from 'react'
2 | import {GlobalState} from '../../GlobalState'
3 | import Menu from './icon/menu.svg'
4 | import Close from './icon/close.svg'
5 | import Cart from './icon/cart.svg'
6 | import {Link} from 'react-router-dom'
7 | import axios from 'axios'
8 |
9 | function Header() {
10 | const state = useContext(GlobalState)
11 | const [isLogged] = state.userAPI.isLogged
12 | const [isAdmin] = state.userAPI.isAdmin
13 | const [cart] = state.userAPI.cart
14 | const [menu, setMenu] = useState(false)
15 |
16 | const logoutUser = async () =>{
17 | await axios.get('/user/logout')
18 |
19 | localStorage.removeItem('firstLogin')
20 |
21 | window.location.href = "/";
22 | }
23 |
24 | const adminRouter = () =>{
25 | return(
26 | <>
27 | Create Product
28 | Categories
29 | >
30 | )
31 | }
32 |
33 | const loggedRouter = () =>{
34 | return(
35 | <>
36 | History
37 | Logout
38 | >
39 | )
40 | }
41 |
42 |
43 | const styleMenu = {
44 | left: menu ? 0 : "-100%"
45 | }
46 |
47 | return (
48 |
85 | )
86 | }
87 |
88 | export default Header
89 |
--------------------------------------------------------------------------------
/client/src/components/headers/header.css:
--------------------------------------------------------------------------------
1 | header{
2 | min-height: 70px;
3 | width: 100%;
4 | overflow: hidden;
5 | display: flex;
6 | flex-wrap: wrap;
7 | justify-content: space-around;
8 | align-items: center;
9 | border-bottom: 1px solid #ddd;
10 | }
11 | header .logo{
12 | flex: 1;
13 | }
14 | header a{
15 | text-transform: uppercase;
16 | color: #555;
17 | }
18 | header ul li{
19 | display: inline-block;
20 | opacity: 0.7;
21 | padding: 0 20px;
22 | }
23 | header ul li:hover{
24 | opacity: 1;
25 | }
26 | .menu{
27 | display: none;
28 | }
29 | .cart-icon{
30 | position: relative;
31 | margin-right: 20px;
32 | }
33 | .cart-icon span{
34 | background: crimson;
35 | border-radius: 20px;
36 | color: white;
37 | position: absolute;
38 | top:-10px;
39 | right: -10px;
40 | padding: 5px 7px;
41 | font-size: 10px;
42 | }
--------------------------------------------------------------------------------
/client/src/components/headers/icon/cart.svg:
--------------------------------------------------------------------------------
1 |
--------------------------------------------------------------------------------
/client/src/components/headers/icon/close.svg:
--------------------------------------------------------------------------------
1 |
--------------------------------------------------------------------------------
/client/src/components/headers/icon/menu.svg:
--------------------------------------------------------------------------------
1 |
--------------------------------------------------------------------------------
/client/src/components/mainpages/Pages.js:
--------------------------------------------------------------------------------
1 | import React, {useContext} from 'react'
2 | import {Switch, Route} from 'react-router-dom'
3 | import Products from './products/Products'
4 | import DetailProduct from './detailProduct/DetailProduct'
5 | import Login from './auth/Login'
6 | import Register from './auth/Register'
7 | import OrderHistory from './history/OrderHistory'
8 | import OrderDetails from './history/OrderDetails'
9 | import Cart from './cart/Cart'
10 | import NotFound from './utils/not_found/NotFound'
11 | import Categories from './categories/Categories'
12 | import CreateProduct from './createProduct/CreateProduct'
13 |
14 | import {GlobalState} from '../../GlobalState'
15 |
16 |
17 | function Pages() {
18 | const state = useContext(GlobalState)
19 | const [isLogged] = state.userAPI.isLogged
20 | const [isAdmin] = state.userAPI.isAdmin
21 |
22 |
23 | return (
24 |
25 |
26 |
27 |
28 |
29 |
30 |
31 |
32 |
33 |
34 |
35 |
36 |
37 |
38 |
39 |
40 |
41 |
42 |
43 | )
44 | }
45 |
46 | export default Pages
47 |
--------------------------------------------------------------------------------
/client/src/components/mainpages/auth/Login.js:
--------------------------------------------------------------------------------
1 | import React, {useState} from 'react'
2 | import {Link} from 'react-router-dom'
3 | import axios from 'axios'
4 |
5 | function Login() {
6 | const [user, setUser] = useState({
7 | email:'', password: ''
8 | })
9 |
10 | const onChangeInput = e =>{
11 | const {name, value} = e.target;
12 | setUser({...user, [name]:value})
13 | }
14 |
15 | const loginSubmit = async e =>{
16 | e.preventDefault()
17 | try {
18 | await axios.post('/user/login', {...user})
19 |
20 | localStorage.setItem('firstLogin', true)
21 |
22 | window.location.href = "/";
23 | } catch (err) {
24 | alert(err.response.data.msg)
25 | }
26 | }
27 |
28 | return (
29 |
44 | )
45 | }
46 |
47 | export default Login
48 |
--------------------------------------------------------------------------------
/client/src/components/mainpages/auth/Register.js:
--------------------------------------------------------------------------------
1 | import React, {useState} from 'react'
2 | import {Link} from 'react-router-dom'
3 | import axios from 'axios'
4 |
5 | function Register() {
6 | const [user, setUser] = useState({
7 | name:'', email:'', password: ''
8 | })
9 |
10 | const onChangeInput = e =>{
11 | const {name, value} = e.target;
12 | setUser({...user, [name]:value})
13 | }
14 |
15 | const registerSubmit = async e =>{
16 | e.preventDefault()
17 | try {
18 | await axios.post('/user/register', {...user})
19 |
20 | localStorage.setItem('firstLogin', true)
21 |
22 |
23 | window.location.href = "/";
24 | } catch (err) {
25 | alert(err.response.data.msg)
26 | }
27 | }
28 |
29 | return (
30 |
48 | )
49 | }
50 |
51 | export default Register
--------------------------------------------------------------------------------
/client/src/components/mainpages/auth/login.css:
--------------------------------------------------------------------------------
1 | .login-page{
2 | max-width: 500px;
3 | border: 2px solid rgb(3, 165, 206);
4 | border-radius: 5px;
5 | padding: 30px;
6 | margin: 50px auto;
7 | }
8 | .login-page h2{
9 | text-transform: uppercase;
10 | letter-spacing: 2px;
11 | color: #555;
12 | }
13 | .login-page form input,
14 | .login-page form button{
15 | width: 100%;
16 | height: 40px;
17 | margin: 10px 0;
18 | padding: 0 5px;
19 | outline: rgb(3, 165, 206);
20 | border: 1px solid rgb(3, 165, 206);
21 | }
22 | .login-page form .row{
23 | display: flex;
24 | justify-content: space-between;
25 | align-items: center;
26 | }
27 | .login-page form button{
28 | width: 150px;
29 | background: rgb(3, 165, 206);
30 | color: white;
31 | text-transform: uppercase;
32 | letter-spacing: 2px;
33 | }
34 | .login-page form a{
35 | color: orange;
36 | letter-spacing: 1.3px;
37 | text-transform: uppercase;
38 | }
--------------------------------------------------------------------------------
/client/src/components/mainpages/cart/Cart.js:
--------------------------------------------------------------------------------
1 | import React, {useContext, useState, useEffect} from 'react'
2 | import {GlobalState} from '../../../GlobalState'
3 | import axios from 'axios'
4 | import PaypalButton from './PaypalButton'
5 |
6 | function Cart() {
7 | const state = useContext(GlobalState)
8 | const [cart, setCart] = state.userAPI.cart
9 | const [token] = state.token
10 | const [total, setTotal] = useState(0)
11 |
12 | useEffect(() =>{
13 | const getTotal = () =>{
14 | const total = cart.reduce((prev, item) => {
15 | return prev + (item.price * item.quantity)
16 | },0)
17 |
18 | setTotal(total)
19 | }
20 |
21 | getTotal()
22 |
23 | },[cart])
24 |
25 | const addToCart = async (cart) =>{
26 | await axios.patch('/user/addcart', {cart}, {
27 | headers: {Authorization: token}
28 | })
29 | }
30 |
31 |
32 | const increment = (id) =>{
33 | cart.forEach(item => {
34 | if(item._id === id){
35 | item.quantity += 1
36 | }
37 | })
38 |
39 | setCart([...cart])
40 | addToCart(cart)
41 | }
42 |
43 | const decrement = (id) =>{
44 | cart.forEach(item => {
45 | if(item._id === id){
46 | item.quantity === 1 ? item.quantity = 1 : item.quantity -= 1
47 | }
48 | })
49 |
50 | setCart([...cart])
51 | addToCart(cart)
52 | }
53 |
54 | const removeProduct = id =>{
55 | if(window.confirm("Do you want to delete this product?")){
56 | cart.forEach((item, index) => {
57 | if(item._id === id){
58 | cart.splice(index, 1)
59 | }
60 | })
61 |
62 | setCart([...cart])
63 | addToCart(cart)
64 | }
65 | }
66 |
67 | const tranSuccess = async(payment) => {
68 | const {paymentID, address} = payment;
69 |
70 | await axios.post('/api/payment', {cart, paymentID, address}, {
71 | headers: {Authorization: token}
72 | })
73 |
74 | setCart([])
75 | addToCart([])
76 | alert("You have successfully placed an order.")
77 | }
78 |
79 |
80 | if(cart.length === 0)
81 | return Cart Empty
82 |
83 | return (
84 |
85 | {
86 | cart.map(product => (
87 |
88 |

89 |
90 |
91 |
{product.title}
92 |
93 |
$ {product.price * product.quantity}
94 |
{product.description}
95 |
{product.content}
96 |
97 |
98 |
99 | {product.quantity}
100 |
101 |
102 |
103 |
removeProduct(product._id)}>
105 | X
106 |
107 |
108 |
109 | ))
110 | }
111 |
112 |
113 |
Total: $ {total}
114 |
117 |
118 |
119 | )
120 | }
121 |
122 | export default Cart
123 |
--------------------------------------------------------------------------------
/client/src/components/mainpages/cart/PaypalButton.js:
--------------------------------------------------------------------------------
1 | import React from 'react';
2 | import PaypalExpressBtn from 'react-paypal-express-checkout';
3 |
4 | export default class PaypalButton extends React.Component {
5 | render() {
6 | const onSuccess = (payment) => {
7 | // Congratulation, it came here means everything's fine!
8 | console.log("The payment was succeeded!", payment);
9 | // You can bind the "payment" object's value to your state or props or whatever here, please see below for sample returned data
10 | this.props.tranSuccess(payment)
11 | }
12 |
13 | const onCancel = (data) => {
14 | // User pressed "cancel" or close Paypal's popup!
15 | console.log('The payment was cancelled!', data);
16 | // You can bind the "data" object's value to your state or props or whatever here, please see below for sample returned data
17 | }
18 |
19 | const onError = (err) => {
20 | // The main Paypal's script cannot be loaded or somethings block the loading of that script!
21 | console.log("Error!", err);
22 | // Because the Paypal's main script is loaded asynchronously from "https://www.paypalobjects.com/api/checkout.js"
23 | // => sometimes it may take about 0.5 second for everything to get set, or for the button to appear
24 | }
25 |
26 | let env = 'sandbox'; // you can set here to 'production' for production
27 | let currency = 'USD'; // or you can set this value from your props or state
28 | let total = this.props.total; // same as above, this is the total amount (based on currency) to be paid by using Paypal express checkout
29 | // Document on Paypal's currency code: https://developer.paypal.com/docs/classic/api/currency_codes/
30 |
31 | const client = {
32 | sandbox: 'YOUR-sandbox-APP-ID',
33 | production: 'YOUR-PRODUCTION-APP-ID',
34 | }
35 | // In order to get production's app-ID, you will have to send your app to Paypal for approval first
36 | // For sandbox app-ID (after logging into your developer account, please locate the "REST API apps" section, click "Create App"):
37 | // => https://developer.paypal.com/docs/classic/lifecycle/sb_credentials/
38 | // For production app-ID:
39 | // => https://developer.paypal.com/docs/classic/lifecycle/goingLive/
40 |
41 | // NB. You can also have many Paypal express checkout buttons on page, just pass in the correct amount and they will work!
42 | let style = {
43 | size: 'small',
44 | color: 'blue',
45 | shape: 'rect',
46 | label: 'checkout',
47 | tagline: false
48 | }
49 |
50 | return (
51 |
57 | );
58 | }
59 | }
--------------------------------------------------------------------------------
/client/src/components/mainpages/cart/cart.css:
--------------------------------------------------------------------------------
1 | .cart{
2 | position: relative;
3 | border: 1px solid #ccc;
4 | transform: scaleY(0.98);
5 | }
6 | .amount span{
7 | color: crimson;
8 | padding: 0 20px;
9 | }
10 | .amount button{
11 | width: 40px;
12 | height: 40px;
13 | border: 1px solid #777;
14 | }
15 | .delete{
16 | position: absolute;
17 | top:0;
18 | right: 5px;
19 | color: crimson;
20 | font-weight: 900;
21 | cursor: pointer;
22 | }
23 | .total{
24 | width: 100%;
25 | height: 50px;
26 | display: flex;
27 | align-items: center;
28 | justify-content: space-between;
29 | }
30 | .total h3{
31 | color: crimson;
32 | }
--------------------------------------------------------------------------------
/client/src/components/mainpages/categories/Categories.js:
--------------------------------------------------------------------------------
1 | import React, {useState, useContext} from 'react'
2 | import {GlobalState} from '../../../GlobalState'
3 | import axios from 'axios'
4 |
5 | function Categories() {
6 | const state = useContext(GlobalState)
7 | const [categories] = state.categoriesAPI.categories
8 | const [category, setCategory] = useState('')
9 | const [token] = state.token
10 | const [callback, setCallback] = state.categoriesAPI.callback
11 | const [onEdit, setOnEdit] = useState(false)
12 | const [id, setID] = useState('')
13 |
14 | const createCategory = async e =>{
15 | e.preventDefault()
16 | try {
17 | if(onEdit){
18 | const res = await axios.put(`/api/category/${id}`, {name: category}, {
19 | headers: {Authorization: token}
20 | })
21 | alert(res.data.msg)
22 | }else{
23 | const res = await axios.post('/api/category', {name: category}, {
24 | headers: {Authorization: token}
25 | })
26 | alert(res.data.msg)
27 | }
28 | setOnEdit(false)
29 | setCategory('')
30 | setCallback(!callback)
31 |
32 | } catch (err) {
33 | alert(err.response.data.msg)
34 | }
35 | }
36 |
37 | const editCategory = async (id, name) =>{
38 | setID(id)
39 | setCategory(name)
40 | setOnEdit(true)
41 | }
42 |
43 | const deleteCategory = async id =>{
44 | try {
45 | const res = await axios.delete(`/api/category/${id}`, {
46 | headers: {Authorization: token}
47 | })
48 | alert(res.data.msg)
49 | setCallback(!callback)
50 | } catch (err) {
51 | alert(err.response.data.msg)
52 | }
53 | }
54 |
55 | return (
56 |
57 |
64 |
65 |
66 | {
67 | categories.map(category => (
68 |
69 |
{category.name}
70 |
71 |
72 |
73 |
74 |
75 | ))
76 | }
77 |
78 |
79 | )
80 | }
81 |
82 | export default Categories
83 |
--------------------------------------------------------------------------------
/client/src/components/mainpages/categories/categories.css:
--------------------------------------------------------------------------------
1 | .categories{
2 | max-width: 700px;
3 | display: flex;
4 | flex-wrap: wrap;
5 | justify-content: space-around;
6 | margin: 30px auto;
7 | }
8 | .categories form{
9 | width: 290px;
10 | margin-bottom: 20px;
11 | }
12 | .categories label{
13 | display: block;
14 | font-weight: 700;
15 | letter-spacing: 2px;
16 | text-transform: uppercase;
17 | margin-bottom: 10px;
18 | }
19 | .categories input, button{
20 | height: 35px;
21 | border: none;
22 | outline: none;
23 | border-bottom: 1px solid #555;
24 | }
25 | .categories input{
26 | width: 210px;
27 | }
28 | .categories button{
29 | width: 70px;
30 | background: #555;
31 | color: white;
32 | margin-left: 10px;
33 | }
34 | .categories .row{
35 | min-width: 290px;
36 | display: flex;
37 | justify-content: space-between;
38 | align-items: center;
39 | padding: 10px;
40 | margin-bottom: 10px;
41 | border: 1px solid #ccc;
42 | }
--------------------------------------------------------------------------------
/client/src/components/mainpages/createProduct/CreateProduct.js:
--------------------------------------------------------------------------------
1 | import React, {useState, useContext, useEffect} from 'react'
2 | import axios from 'axios'
3 | import {GlobalState} from '../../../GlobalState'
4 | import Loading from '../utils/loading/Loading'
5 | import {useHistory, useParams} from 'react-router-dom'
6 |
7 | const initialState = {
8 | product_id: '',
9 | title: '',
10 | price: 0,
11 | description: 'How to and tutorial videos of cool CSS effect, Web Design ideas,JavaScript libraries, Node.',
12 | content: 'Welcome to our channel Dev AT. Here you can learn web designing, UI/UX designing, html css tutorials, css animations and css effects, javascript and jquery tutorials and related so on.',
13 | category: '',
14 | _id: ''
15 | }
16 |
17 | function CreateProduct() {
18 | const state = useContext(GlobalState)
19 | const [product, setProduct] = useState(initialState)
20 | const [categories] = state.categoriesAPI.categories
21 | const [images, setImages] = useState(false)
22 | const [loading, setLoading] = useState(false)
23 |
24 |
25 | const [isAdmin] = state.userAPI.isAdmin
26 | const [token] = state.token
27 |
28 | const history = useHistory()
29 | const param = useParams()
30 |
31 | const [products] = state.productsAPI.products
32 | const [onEdit, setOnEdit] = useState(false)
33 | const [callback, setCallback] = state.productsAPI.callback
34 |
35 | useEffect(() => {
36 | if(param.id){
37 | setOnEdit(true)
38 | products.forEach(product => {
39 | if(product._id === param.id) {
40 | setProduct(product)
41 | setImages(product.images)
42 | }
43 | })
44 | }else{
45 | setOnEdit(false)
46 | setProduct(initialState)
47 | setImages(false)
48 | }
49 | }, [param.id, products])
50 |
51 | const handleUpload = async e =>{
52 | e.preventDefault()
53 | try {
54 | if(!isAdmin) return alert("You're not an admin")
55 | const file = e.target.files[0]
56 |
57 | if(!file) return alert("File not exist.")
58 |
59 | if(file.size > 1024 * 1024) // 1mb
60 | return alert("Size too large!")
61 |
62 | if(file.type !== 'image/jpeg' && file.type !== 'image/png') // 1mb
63 | return alert("File format is incorrect.")
64 |
65 | let formData = new FormData()
66 | formData.append('file', file)
67 |
68 | setLoading(true)
69 | const res = await axios.post('/api/upload', formData, {
70 | headers: {'content-type': 'multipart/form-data', Authorization: token}
71 | })
72 | setLoading(false)
73 | setImages(res.data)
74 |
75 | } catch (err) {
76 | alert(err.response.data.msg)
77 | }
78 | }
79 |
80 | const handleDestroy = async () => {
81 | try {
82 | if(!isAdmin) return alert("You're not an admin")
83 | setLoading(true)
84 | await axios.post('/api/destroy', {public_id: images.public_id}, {
85 | headers: {Authorization: token}
86 | })
87 | setLoading(false)
88 | setImages(false)
89 | } catch (err) {
90 | alert(err.response.data.msg)
91 | }
92 | }
93 |
94 | const handleChangeInput = e =>{
95 | const {name, value} = e.target
96 | setProduct({...product, [name]:value})
97 | }
98 |
99 | const handleSubmit = async e =>{
100 | e.preventDefault()
101 | try {
102 | if(!isAdmin) return alert("You're not an admin")
103 | if(!images) return alert("No Image Upload")
104 |
105 | if(onEdit){
106 | await axios.put(`/api/products/${product._id}`, {...product, images}, {
107 | headers: {Authorization: token}
108 | })
109 | }else{
110 | await axios.post('/api/products', {...product, images}, {
111 | headers: {Authorization: token}
112 | })
113 | }
114 | setCallback(!callback)
115 | history.push("/")
116 | } catch (err) {
117 | alert(err.response.data.msg)
118 | }
119 | }
120 |
121 | const styleUpload = {
122 | display: images ? "block" : "none"
123 | }
124 | return (
125 |
126 |
127 |
128 | {
129 | loading ?
130 |
131 | :
132 |

133 |
X
134 |
135 | }
136 |
137 |
138 |
139 |
186 |
187 | )
188 | }
189 |
190 | export default CreateProduct
191 |
--------------------------------------------------------------------------------
/client/src/components/mainpages/createProduct/createProduct.css:
--------------------------------------------------------------------------------
1 | .create_product{
2 | width: 100%;
3 | display: flex;
4 | flex-wrap: wrap;
5 | align-items: center;
6 | justify-content: space-around;
7 | }
8 | .upload{
9 | max-width: 450px;
10 | height: 500px;
11 | width: 100%;
12 | border: 1px solid #ddd;
13 | padding: 15px;
14 | margin: 20px;
15 | position: relative;
16 | }
17 | #file_up{
18 | position: relative;
19 | width: 100%;
20 | height: 100%;
21 | outline: none;
22 | }
23 | #file_up::before{
24 | content: "+";
25 | position: absolute;
26 | width: 100%;
27 | height: 100%;
28 | top:0;
29 | left: 0;
30 | background: white;
31 | color: rgb(250, 200, 107);
32 | font-size: 17rem;
33 | text-align: center;
34 | cursor: pointer;
35 | margin: auto;
36 | }
37 | #file_img{
38 | width: 100%;
39 | height: 100%;
40 | position: absolute;
41 | top:0;
42 | left: 0;
43 | background: white;
44 | }
45 | #file_img img{
46 | width: 100%;
47 | height: 100%;
48 | display: block;
49 | object-fit: cover;
50 | }
51 | #file_img span{
52 | position: absolute;
53 | top: -13px;
54 | right: -13px;
55 | background: white;
56 | border: 1px solid #ddd;
57 | border-radius: 50%;
58 | padding: 6px 10px;
59 | cursor: pointer;
60 | font-weight: 900;
61 | color: crimson;
62 | }
63 |
64 | .create_product form{
65 | max-width: 500px;
66 | min-width: 290px;
67 | width: 100%;
68 | margin: 15px 30px;
69 | }
70 | .create_product form .row{
71 | width: 100%;
72 | margin: 15px 0;
73 | }
74 | .create_product form input, textarea{
75 | width: 100%;
76 | min-height: 40px;
77 | padding: 0 5px;
78 | }
79 | .create_product form button{
80 | width: 200px;
81 | height: 40px;
82 | background: #555;
83 | color: white;
84 | text-transform: uppercase;
85 | letter-spacing: 2px;
86 | font-weight: 700;
87 | }
--------------------------------------------------------------------------------
/client/src/components/mainpages/detailProduct/DetailProduct.js:
--------------------------------------------------------------------------------
1 | import React, {useContext, useState, useEffect} from 'react'
2 | import {useParams, Link} from 'react-router-dom'
3 | import {GlobalState} from '../../../GlobalState'
4 | import ProductItem from '../utils/productItem/ProductItem'
5 |
6 |
7 | function DetailProduct() {
8 | const params = useParams()
9 | const state = useContext(GlobalState)
10 | const [products] = state.productsAPI.products
11 | const addCart = state.userAPI.addCart
12 | const [detailProduct, setDetailProduct] = useState([])
13 |
14 | useEffect(() =>{
15 | if(params.id){
16 |
17 | products.forEach(product => {
18 | if(product._id === params.id) setDetailProduct(product)
19 | })
20 | }
21 | },[params.id, products])
22 |
23 | if(detailProduct.length === 0) return null;
24 |
25 | return (
26 | <>
27 |
28 |

29 |
30 |
31 |
{detailProduct.title}
32 | #id: {detailProduct.product_id}
33 |
34 |
$ {detailProduct.price}
35 |
{detailProduct.description}
36 |
{detailProduct.content}
37 |
Sold: {detailProduct.sold}
38 |
addCart(detailProduct)}>
40 | Buy Now
41 |
42 |
43 |
44 |
45 |
46 |
Related products
47 |
48 | {
49 | products.map(product => {
50 | return product.category === detailProduct.category
51 | ?
: null
52 | })
53 | }
54 |
55 |
56 | >
57 | )
58 | }
59 |
60 | export default DetailProduct
61 |
--------------------------------------------------------------------------------
/client/src/components/mainpages/detailProduct/detailProduct.css:
--------------------------------------------------------------------------------
1 | .detail{
2 | width: 100%;
3 | display: flex;
4 | justify-content: space-around;
5 | flex-wrap: wrap;
6 | padding: 50px;
7 | font-size: 150%;
8 | }
9 | .detail img{
10 | max-width: 400px;
11 | width: 100%;
12 | margin: 20px;
13 | height: 450px;
14 | object-fit: cover;
15 | display: block;
16 | }
17 | .box-detail{
18 | max-width: 500px;
19 | width: 100%;
20 | margin: 5px 20px;
21 | }
22 | .box-detail .row{
23 | display: flex;
24 | justify-content: space-between;
25 | align-items: center;
26 | }
27 | .box-detail h2{
28 | text-transform: uppercase;
29 | color: darkblue;
30 | letter-spacing: 2px;
31 | font-weight: 2rem;
32 | }
33 | .box-detail p{
34 | line-height: 1.5;
35 | margin: 10px 0;
36 | opacity: 0.8;
37 | }
38 | .box-detail .cart{
39 | background: #333;
40 | color: white;
41 | margin-top: 10px;
42 | padding: 10px 25px;
43 | display: inline-block;
44 | text-transform: uppercase;
45 | letter-spacing: 2px;
46 | }
--------------------------------------------------------------------------------
/client/src/components/mainpages/history/OrderDetails.js:
--------------------------------------------------------------------------------
1 | import React, {useState, useEffect, useContext} from 'react'
2 | import {useParams} from 'react-router-dom'
3 | import {GlobalState} from '../../../GlobalState'
4 |
5 | function OrderDetails() {
6 | const state = useContext(GlobalState)
7 | const [history] = state.userAPI.history
8 | const [orderDetails, setOrderDetails] = useState([])
9 |
10 | const params = useParams()
11 |
12 | useEffect(() => {
13 | if(params.id){
14 | history.forEach(item =>{
15 | if(item._id === params.id) setOrderDetails(item)
16 | })
17 | }
18 | },[params.id, history])
19 |
20 |
21 | if(orderDetails.length === 0) return null;
22 |
23 | return (
24 |
25 |
26 |
27 |
28 | Name |
29 | Address |
30 | Postal Code |
31 | Country Code |
32 |
33 |
34 |
35 |
36 | {orderDetails.address.recipient_name} |
37 | {orderDetails.address.line1 + " - " + orderDetails.address.city} |
38 | {orderDetails.address.postal_code} |
39 | {orderDetails.address.country_code} |
40 |
41 |
42 |
43 |
44 |
45 |
46 |
47 | |
48 | Products |
49 | Quantity |
50 | Price |
51 |
52 |
53 |
54 | {
55 | orderDetails.cart.map(item =>(
56 |
57 |  |
58 | {item.title} |
59 | {item.quantity} |
60 | $ {item.price * item.quantity} |
61 |
62 | ))
63 | }
64 |
65 |
66 |
67 |
68 | )
69 | }
70 |
71 | export default OrderDetails
72 |
--------------------------------------------------------------------------------
/client/src/components/mainpages/history/OrderHistory.js:
--------------------------------------------------------------------------------
1 | import React, {useContext, useEffect} from 'react'
2 | import {GlobalState} from '../../../GlobalState'
3 | import {Link} from 'react-router-dom'
4 | import axios from 'axios'
5 |
6 | function OrderHistory() {
7 | const state = useContext(GlobalState)
8 | const [history, setHistory] = state.userAPI.history
9 | const [isAdmin] = state.userAPI.isAdmin
10 | const [token] = state.token
11 |
12 |
13 | useEffect(() => {
14 | if(token){
15 | const getHistory = async() =>{
16 | if(isAdmin){
17 | const res = await axios.get('/api/payment', {
18 | headers: {Authorization: token}
19 | })
20 | setHistory(res.data)
21 | }else{
22 | const res = await axios.get('/user/history', {
23 | headers: {Authorization: token}
24 | })
25 | setHistory(res.data)
26 | }
27 | }
28 | getHistory()
29 | }
30 | },[token, isAdmin, setHistory])
31 |
32 | return (
33 |
34 |
History
35 |
36 |
You have {history.length} ordered
37 |
38 |
39 |
40 |
41 | Payment ID |
42 | Date of Purchased |
43 | |
44 |
45 |
46 |
47 | {
48 | history.map(items => (
49 |
50 | {items.paymentID} |
51 | {new Date(items.createdAt).toLocaleDateString()} |
52 | View |
53 |
54 | ))
55 | }
56 |
57 |
58 |
59 | )
60 | }
61 |
62 | export default OrderHistory
63 |
--------------------------------------------------------------------------------
/client/src/components/mainpages/history/history.css:
--------------------------------------------------------------------------------
1 | .history-page{
2 | overflow-x: auto;
3 | }
4 | .history-page h2, h4{
5 | text-align: center;
6 | margin: 20px;
7 | text-transform: uppercase;
8 | letter-spacing: 1.2px;
9 | }
10 | .history-page table{
11 | margin: auto;
12 | width: 100%;
13 | }
14 | .history-page table,th,tr,td{
15 | border: 1px solid #ddd;
16 | border-collapse: collapse;
17 | }
18 | th, td{
19 | text-align: center;
20 | padding: 10px;
21 | text-transform: capitalize;
22 | }
23 | table a{
24 | color: rgb(3, 165, 206);
25 | }
26 |
27 | .history-page img{
28 | width: 70px;
29 | height: 100px;
30 | object-fit: cover;
31 | }
--------------------------------------------------------------------------------
/client/src/components/mainpages/products/Filters.js:
--------------------------------------------------------------------------------
1 | import React, {useContext} from 'react'
2 | import {GlobalState} from '../../../GlobalState'
3 |
4 | function Filters() {
5 | const state = useContext(GlobalState)
6 | const [categories] = state.categoriesAPI.categories
7 |
8 | const [category, setCategory] = state.productsAPI.category
9 | const [sort, setSort] = state.productsAPI.sort
10 | const [search, setSearch] = state.productsAPI.search
11 |
12 |
13 | const handleCategory = e => {
14 | setCategory(e.target.value)
15 | setSearch('')
16 | }
17 |
18 | return (
19 |
20 |
21 | Filters:
22 |
32 |
33 |
34 |
setSearch(e.target.value.toLowerCase())} />
36 |
37 |
38 | Sort By:
39 |
46 |
47 |
48 | )
49 | }
50 |
51 | export default Filters
52 |
--------------------------------------------------------------------------------
/client/src/components/mainpages/products/LoadMore.js:
--------------------------------------------------------------------------------
1 | import React, {useContext} from 'react'
2 | import {GlobalState} from '../../../GlobalState'
3 |
4 | function LoadMore() {
5 | const state = useContext(GlobalState)
6 | const [page, setPage] = state.productsAPI.page
7 | const [result] = state.productsAPI.result
8 |
9 | return (
10 |
11 | {
12 | result < page * 9 ? ""
13 | :
14 | }
15 |
16 | )
17 | }
18 |
19 | export default LoadMore
20 |
--------------------------------------------------------------------------------
/client/src/components/mainpages/products/Products.js:
--------------------------------------------------------------------------------
1 | import React, {useContext, useState} from 'react'
2 | import {GlobalState} from '../../../GlobalState'
3 | import ProductItem from '../utils/productItem/ProductItem'
4 | import Loading from '../utils/loading/Loading'
5 | import axios from 'axios'
6 | import Filters from './Filters'
7 | import LoadMore from './LoadMore'
8 |
9 |
10 | function Products() {
11 | const state = useContext(GlobalState)
12 | const [products, setProducts] = state.productsAPI.products
13 | const [isAdmin] = state.userAPI.isAdmin
14 | const [token] = state.token
15 | const [callback, setCallback] = state.productsAPI.callback
16 | const [loading, setLoading] = useState(false)
17 | const [isCheck, setIsCheck] = useState(false)
18 |
19 | const handleCheck = (id) =>{
20 | products.forEach(product => {
21 | if(product._id === id) product.checked = !product.checked
22 | })
23 | setProducts([...products])
24 | }
25 |
26 | const deleteProduct = async(id, public_id) => {
27 | try {
28 | setLoading(true)
29 | const destroyImg = axios.post('/api/destroy', {public_id},{
30 | headers: {Authorization: token}
31 | })
32 | const deleteProduct = axios.delete(`/api/products/${id}`, {
33 | headers: {Authorization: token}
34 | })
35 |
36 | await destroyImg
37 | await deleteProduct
38 | setCallback(!callback)
39 | setLoading(false)
40 | } catch (err) {
41 | alert(err.response.data.msg)
42 | }
43 | }
44 |
45 | const checkAll = () =>{
46 | products.forEach(product => {
47 | product.checked = !isCheck
48 | })
49 | setProducts([...products])
50 | setIsCheck(!isCheck)
51 | }
52 |
53 | const deleteAll = () =>{
54 | products.forEach(product => {
55 | if(product.checked) deleteProduct(product._id, product.images.public_id)
56 | })
57 | }
58 |
59 | if(loading) return
60 | return (
61 | <>
62 |
63 |
64 | {
65 | isAdmin &&
66 |
67 | Select all
68 |
69 |
70 |
71 | }
72 |
73 |
74 | {
75 | products.map(product => {
76 | return
78 | })
79 | }
80 |
81 |
82 |
83 | {products.length === 0 && }
84 | >
85 | )
86 | }
87 |
88 | export default Products
89 |
--------------------------------------------------------------------------------
/client/src/components/mainpages/products/products.css:
--------------------------------------------------------------------------------
1 | .products{
2 | width: 100%;
3 | display: grid;
4 | grid-template-columns: repeat(auto-fill, minmax(300px, 1fr));
5 | justify-items: center;
6 | margin: 20px 0;
7 | }
8 |
9 | /* ----------- Product Item ----------------- */
10 | @import url("../utils/productItem/productItem.css");
11 |
12 | .delete-all{
13 | text-align: right;
14 | margin: 20px;
15 | }
16 | .delete-all input{
17 | height: 25px;
18 | width: 25px;
19 | transform: translateY(5px);
20 | margin: 0 15px;
21 | }
22 | .delete-all span{
23 | text-transform: uppercase;
24 | color: rgb(3, 165, 206);
25 | letter-spacing: 1.3px;
26 | }
27 | .delete-all button{
28 | border: 1px solid crimson;
29 | padding: 10px 25px;
30 | color: crimson;
31 | text-transform: uppercase;
32 | }
33 |
34 | /* ------------ Filters Menu ----------------- */
35 | .filter_menu{
36 | width: 100%;
37 | min-height: 40px;
38 | overflow: hidden;
39 | display: flex;
40 | justify-content: space-between;
41 | align-items: center;
42 | flex-wrap: wrap;
43 | margin: 15px 0;
44 | }
45 | .filter_menu select, input{
46 | border: 1px solid #ccc;
47 | outline: none;
48 | height: 40px;
49 | padding: 0 5px;
50 | }
51 | .filter_menu input{
52 | flex: 1;
53 | margin: 0 10px;
54 | }
55 |
56 | /* -------------------- Load More --------------- */
57 | .load_more{
58 | text-align: center;
59 | }
60 | .load_more button{
61 | padding: 10px 25px;
62 | margin-bottom: 20px;
63 | border: 1px solid #555;
64 | text-transform: capitalize;
65 | }
--------------------------------------------------------------------------------
/client/src/components/mainpages/utils/loading/Loading.js:
--------------------------------------------------------------------------------
1 | import React from 'react'
2 | import './loading.css'
3 |
4 | function Loading() {
5 | return (
6 |
25 | )
26 | }
27 |
28 | export default Loading
29 |
--------------------------------------------------------------------------------
/client/src/components/mainpages/utils/loading/loading.css:
--------------------------------------------------------------------------------
1 | .load-page{
2 | background: #fff;
3 | overflow: hidden;
4 | }
5 | .loader{
6 | width: 300px;
7 | height: 300px;
8 | margin: auto;
9 | }
10 | .loader div{
11 | width: calc(100% - 15px);
12 | height: calc(100% - 15px);
13 | border: 2px solid #fff;
14 | border-top: 2px solid crimson;
15 | border-radius: 50%;
16 | animation: rotate 10s linear infinite alternate-reverse;
17 | }
18 | @keyframes rotate{
19 | 50%{
20 | transform: rotate(80deg);
21 | }
22 | 100%{
23 | transform: rotate(360deg);
24 | }
25 | }
--------------------------------------------------------------------------------
/client/src/components/mainpages/utils/not_found/NotFound.js:
--------------------------------------------------------------------------------
1 | import React from 'react'
2 |
3 | function NotFound() {
4 | return (
5 |
6 | 404 | Not Found
7 |
8 | )
9 | }
10 |
11 | export default NotFound
12 |
--------------------------------------------------------------------------------
/client/src/components/mainpages/utils/productItem/BtnRender.js:
--------------------------------------------------------------------------------
1 | import React, {useContext} from 'react'
2 | import {Link} from 'react-router-dom'
3 | import {GlobalState} from '../../../../GlobalState'
4 |
5 | function BtnRender({product, deleteProduct}) {
6 | const state = useContext(GlobalState)
7 | const [isAdmin] = state.userAPI.isAdmin
8 | const addCart = state.userAPI.addCart
9 |
10 |
11 | return (
12 |
13 | {
14 | isAdmin ?
15 | <>
16 | deleteProduct(product._id, product.images.public_id)}>
18 | Delete
19 |
20 |
21 | Edit
22 |
23 | >
24 | : <>
25 | addCart(product)}>
26 | Buy
27 |
28 |
29 | View
30 |
31 | >
32 | }
33 |
34 |
35 | )
36 | }
37 |
38 | export default BtnRender
39 |
--------------------------------------------------------------------------------
/client/src/components/mainpages/utils/productItem/ProductItem.js:
--------------------------------------------------------------------------------
1 | import React from 'react'
2 | import BtnRender from './BtnRender'
3 |
4 | function ProductItem({product, isAdmin, deleteProduct, handleCheck}) {
5 |
6 | return (
7 |
8 | {
9 | isAdmin &&
handleCheck(product._id)} />
11 | }
12 |

13 |
14 |
15 |
{product.title}
16 |
${product.price}
17 |
{product.description}
18 |
19 |
20 |
21 |
22 |
23 | )
24 | }
25 |
26 | export default ProductItem
27 |
--------------------------------------------------------------------------------
/client/src/components/mainpages/utils/productItem/productItem.css:
--------------------------------------------------------------------------------
1 | .product_card{
2 | max-width: 300px;
3 | overflow: hidden;
4 | height: 500px;
5 | padding: 15px;
6 | box-shadow: 0 0 15px #ddd;
7 | margin: 10px 0;
8 | position: relative;
9 | }
10 | .product_card img{
11 | width: 100%;
12 | height: 300px;
13 | display: block;
14 | object-fit: cover;
15 | }
16 | .product_box h2{
17 | width: 100%;
18 | text-overflow: ellipsis;
19 | overflow: hidden;
20 | white-space: nowrap;
21 | text-transform: capitalize;
22 | cursor: pointer;
23 | color: #323232;
24 | }
25 | .product_card span{
26 | color: crimson;
27 | }
28 | .product_box p{
29 | width: 100%;
30 | display: -webkit-box;
31 | -webkit-box-orient: vertical;
32 | -webkit-line-clamp: 3;
33 | height: 70px;
34 | overflow: hidden;
35 | color: #323232;
36 | }
37 | .row_btn{
38 | width: 100%;
39 | margin-top: 10px;
40 | display: flex;
41 | justify-content: space-between;
42 | }
43 | .row_btn a{
44 | width: 50%;
45 | text-align: center;
46 | text-transform: uppercase;
47 | color: white;
48 | font-weight: 600;
49 | letter-spacing: 2px;
50 | padding: 6px;
51 | }
52 | #btn_buy{
53 | background: #555;
54 | margin-right: 5px;
55 | }
56 | #btn_view{
57 | background: teal;
58 | margin-left: 5px;
59 | }
60 | .product_card input{
61 | position: absolute;
62 | width: 25px;
63 | height: 25px;
64 | }
--------------------------------------------------------------------------------
/client/src/index.css:
--------------------------------------------------------------------------------
1 | * {
2 | padding: 0;
3 | box-sizing: border-box;
4 | margin: 0;
5 | font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', 'Roboto', 'Oxygen',
6 | 'Ubuntu', 'Cantarell', 'Fira Sans', 'Droid Sans', 'Helvetica Neue',
7 | sans-serif;
8 | -webkit-font-smoothing: antialiased;
9 | -moz-osx-font-smoothing: grayscale;
10 | }
11 |
12 | code {
13 | font-family: source-code-pro, Menlo, Monaco, Consolas, 'Courier New',
14 | monospace;
15 | }
16 | .App{
17 | max-width: 1230px;
18 | min-height: 100vh;
19 | margin: 0 auto;
20 | padding: 0 20px;
21 | box-shadow: 0 0 35px #eee;
22 | }
23 | a{
24 | text-decoration: none;
25 | }
26 | ul li {
27 | list-style: none;
28 | }
29 | button{
30 | border: none;
31 | outline: none;
32 | background: transparent;
33 | cursor: pointer;
34 | }
35 |
36 | /* ---------------- Header ------------ */
37 | @import url("./components/headers/header.css");
38 |
39 | /* ------------- Products -------------- */
40 |
41 | @import url("./components/mainpages/products/products.css");
42 |
43 | /* ---------------- Detail Product --------------- */
44 | @import url("./components/mainpages/detailProduct/detailProduct.css");
45 |
46 | /* ------------- Login/Register ----------------- */
47 | @import url("./components/mainpages/auth/login.css");
48 |
49 | /* ------------ Cart ------------------- */
50 | @import url("./components/mainpages/cart/cart.css");
51 |
52 | /* --------------- History ------------------- */
53 | @import url("./components/mainpages/history/history.css");
54 |
55 | /* ------------------ Categories ------------------ */
56 | @import url("./components/mainpages/categories/categories.css");
57 |
58 | /* ------------------- Create Product -------------- */
59 | @import url("./components/mainpages/createProduct/createProduct.css");
60 |
61 | /* ----------------- Responsive ------------------ */
62 | @media (max-width: 820px){
63 | header .logo{
64 | min-width: 115px;
65 | flex: none;
66 | }
67 | header ul{
68 | padding: 10px 0;
69 | }
70 | }
71 | @media (max-width: 720px){
72 | header ul{
73 | position: fixed;
74 | top:0;
75 | left: -100%;
76 | width: 100%;
77 | height: 100vh;
78 | background: white;
79 | display: flex;
80 | flex-direction: column;
81 | justify-content: space-around;
82 | align-items: center;
83 | opacity: 0.98;
84 | z-index: 99;
85 | transition: 0.5s ease-in;
86 | }
87 | header .menu{
88 | display: block;
89 | cursor: pointer;
90 | }
91 | header ul li .menu{
92 | position: absolute;
93 | top:20px;
94 | right: 20px;
95 | }
96 | header{
97 | justify-content: space-between;
98 | }
99 | }
100 |
101 | @media (max-width: 570px){
102 | .filter_menu .row span{
103 | display: none;
104 | }
105 | }
106 |
107 | @media (max-width: 500px){
108 | .App{
109 | padding: 0 10px;
110 | }
111 | .detail{
112 | font-size: 100%;
113 | padding: 0;
114 | }
115 | .filter_menu input{
116 | margin: 0;
117 | }
118 | .filter_menu .sort{
119 | flex: 1;
120 | }
121 | .filter_menu .sort select{
122 | width: 100%;
123 | min-width: 290px;
124 | margin: 5px 0;
125 | }
126 | .login-page{
127 | padding: 30px 15px;
128 | }
129 | .delete-all{
130 | margin: 20px 0;
131 | }
132 | .create_product .upload{
133 | margin: 20px 0;
134 | }
135 | }
--------------------------------------------------------------------------------
/client/src/index.js:
--------------------------------------------------------------------------------
1 | import React from 'react';
2 | import ReactDOM from 'react-dom';
3 | import './index.css';
4 | import App from './App';
5 |
6 |
7 | ReactDOM.render(,
8 | document.getElementById('root')
9 | );
10 |
11 |
12 |
--------------------------------------------------------------------------------
/controllers/categoryCtrl.js:
--------------------------------------------------------------------------------
1 | const Category = require('../models/categoryModel')
2 | const Products = require('../models/productModel')
3 |
4 | const categoryCtrl = {
5 | getCategories: async(req, res) =>{
6 | try {
7 | const categories = await Category.find()
8 | res.json(categories)
9 | } catch (err) {
10 | return res.status(500).json({msg: err.message})
11 | }
12 | },
13 | createCategory: async (req, res) =>{
14 | try {
15 | // if user have role = 1 ---> admin
16 | // only admin can create , delete and update category
17 | const {name} = req.body;
18 | const category = await Category.findOne({name})
19 | if(category) return res.status(400).json({msg: "This category already exists."})
20 |
21 | const newCategory = new Category({name})
22 |
23 | await newCategory.save()
24 | res.json({msg: "Created a category"})
25 | } catch (err) {
26 | return res.status(500).json({msg: err.message})
27 | }
28 | },
29 | deleteCategory: async(req, res) =>{
30 | try {
31 | const products = await Products.findOne({category: req.params.id})
32 | if(products) return res.status(400).json({
33 | msg: "Please delete all products with a relationship."
34 | })
35 |
36 | await Category.findByIdAndDelete(req.params.id)
37 | res.json({msg: "Deleted a Category"})
38 | } catch (err) {
39 | return res.status(500).json({msg: err.message})
40 | }
41 | },
42 | updateCategory: async(req, res) =>{
43 | try {
44 | const {name} = req.body;
45 | await Category.findOneAndUpdate({_id: req.params.id}, {name})
46 |
47 | res.json({msg: "Updated a category"})
48 | } catch (err) {
49 | return res.status(500).json({msg: err.message})
50 | }
51 | }
52 | }
53 |
54 |
55 | module.exports = categoryCtrl
--------------------------------------------------------------------------------
/controllers/paymentCtrl.js:
--------------------------------------------------------------------------------
1 | const Payments = require('../models/paymentModel')
2 | const Users = require('../models/userModel')
3 | const Products = require('../models/productModel')
4 |
5 |
6 | const paymentCtrl = {
7 | getPayments: async(req, res) =>{
8 | try {
9 | const payments = await Payments.find()
10 | res.json(payments)
11 | } catch (err) {
12 | return res.status(500).json({msg: err.message})
13 | }
14 | },
15 | createPayment: async(req, res) => {
16 | try {
17 | const user = await Users.findById(req.user.id).select('name email')
18 | if(!user) return res.status(400).json({msg: "User does not exist."})
19 |
20 | const {cart, paymentID, address} = req.body;
21 |
22 | const {_id, name, email} = user;
23 |
24 | const newPayment = new Payments({
25 | user_id: _id, name, email, cart, paymentID, address
26 | })
27 |
28 | cart.filter(item => {
29 | return sold(item._id, item.quantity, item.sold)
30 | })
31 |
32 |
33 | await newPayment.save()
34 | res.json({msg: "Payment Succes!"})
35 |
36 | } catch (err) {
37 | return res.status(500).json({msg: err.message})
38 | }
39 | }
40 | }
41 |
42 | const sold = async (id, quantity, oldSold) =>{
43 | await Products.findOneAndUpdate({_id: id}, {
44 | sold: quantity + oldSold
45 | })
46 | }
47 |
48 | module.exports = paymentCtrl
49 |
--------------------------------------------------------------------------------
/controllers/productCtrl.js:
--------------------------------------------------------------------------------
1 | const Products = require('../models/productModel')
2 |
3 | // Filter, sorting and paginating
4 |
5 | class APIfeatures {
6 | constructor(query, queryString){
7 | this.query = query;
8 | this.queryString = queryString;
9 | }
10 | filtering(){
11 | const queryObj = {...this.queryString} //queryString = req.query
12 |
13 | const excludedFields = ['page', 'sort', 'limit']
14 | excludedFields.forEach(el => delete(queryObj[el]))
15 |
16 | let queryStr = JSON.stringify(queryObj)
17 | queryStr = queryStr.replace(/\b(gte|gt|lt|lte|regex)\b/g, match => '$' + match)
18 |
19 | // gte = greater than or equal
20 | // lte = lesser than or equal
21 | // lt = lesser than
22 | // gt = greater than
23 | this.query.find(JSON.parse(queryStr))
24 |
25 | return this;
26 | }
27 |
28 | sorting(){
29 | if(this.queryString.sort){
30 | const sortBy = this.queryString.sort.split(',').join(' ')
31 | this.query = this.query.sort(sortBy)
32 | }else{
33 | this.query = this.query.sort('-createdAt')
34 | }
35 |
36 | return this;
37 | }
38 |
39 | paginating(){
40 | const page = this.queryString.page * 1 || 1
41 | const limit = this.queryString.limit * 1 || 9
42 | const skip = (page - 1) * limit;
43 | this.query = this.query.skip(skip).limit(limit)
44 | return this;
45 | }
46 | }
47 |
48 | const productCtrl = {
49 | getProducts: async(req, res) =>{
50 | try {
51 | const features = new APIfeatures(Products.find(), req.query)
52 | .filtering().sorting().paginating()
53 |
54 | const products = await features.query
55 |
56 | res.json({
57 | status: 'success',
58 | result: products.length,
59 | products: products
60 | })
61 |
62 | } catch (err) {
63 | return res.status(500).json({msg: err.message})
64 | }
65 | },
66 | createProduct: async(req, res) =>{
67 | try {
68 | const {product_id, title, price, description, content, images, category} = req.body;
69 | if(!images) return res.status(400).json({msg: "No image upload"})
70 |
71 | const product = await Products.findOne({product_id})
72 | if(product)
73 | return res.status(400).json({msg: "This product already exists."})
74 |
75 | const newProduct = new Products({
76 | product_id, title: title.toLowerCase(), price, description, content, images, category
77 | })
78 |
79 | await newProduct.save()
80 | res.json({msg: "Created a product"})
81 |
82 | } catch (err) {
83 | return res.status(500).json({msg: err.message})
84 | }
85 | },
86 | deleteProduct: async(req, res) =>{
87 | try {
88 | await Products.findByIdAndDelete(req.params.id)
89 | res.json({msg: "Deleted a Product"})
90 | } catch (err) {
91 | return res.status(500).json({msg: err.message})
92 | }
93 | },
94 | updateProduct: async(req, res) =>{
95 | try {
96 | const {title, price, description, content, images, category} = req.body;
97 | if(!images) return res.status(400).json({msg: "No image upload"})
98 |
99 | await Products.findOneAndUpdate({_id: req.params.id}, {
100 | title: title.toLowerCase(), price, description, content, images, category
101 | })
102 |
103 | res.json({msg: "Updated a Product"})
104 | } catch (err) {
105 | return res.status(500).json({msg: err.message})
106 | }
107 | }
108 | }
109 |
110 |
111 | module.exports = productCtrl
--------------------------------------------------------------------------------
/controllers/userCtrl.js:
--------------------------------------------------------------------------------
1 | const Users = require('../models/userModel')
2 | const Payments = require('../models/paymentModel')
3 | const bcrypt = require('bcrypt')
4 | const jwt = require('jsonwebtoken')
5 |
6 | const userCtrl = {
7 | register: async (req, res) =>{
8 | try {
9 | const {name, email, password} = req.body;
10 |
11 | const user = await Users.findOne({email})
12 | if(user) return res.status(400).json({msg: "The email already exists."})
13 |
14 | if(password.length < 6)
15 | return res.status(400).json({msg: "Password is at least 6 characters long."})
16 |
17 | // Password Encryption
18 | const passwordHash = await bcrypt.hash(password, 10)
19 | const newUser = new Users({
20 | name, email, password: passwordHash
21 | })
22 |
23 | // Save mongodb
24 | await newUser.save()
25 |
26 | // Then create jsonwebtoken to authentication
27 | const accesstoken = createAccessToken({id: newUser._id})
28 | const refreshtoken = createRefreshToken({id: newUser._id})
29 |
30 | res.cookie('refreshtoken', refreshtoken, {
31 | httpOnly: true,
32 | path: '/user/refresh_token',
33 | maxAge: 7*24*60*60*1000 // 7d
34 | })
35 |
36 | res.json({accesstoken})
37 |
38 | } catch (err) {
39 | return res.status(500).json({msg: err.message})
40 | }
41 | },
42 | login: async (req, res) =>{
43 | try {
44 | const {email, password} = req.body;
45 |
46 | const user = await Users.findOne({email})
47 | if(!user) return res.status(400).json({msg: "User does not exist."})
48 |
49 | const isMatch = await bcrypt.compare(password, user.password)
50 | if(!isMatch) return res.status(400).json({msg: "Incorrect password."})
51 |
52 | // If login success , create access token and refresh token
53 | const accesstoken = createAccessToken({id: user._id})
54 | const refreshtoken = createRefreshToken({id: user._id})
55 |
56 | res.cookie('refreshtoken', refreshtoken, {
57 | httpOnly: true,
58 | path: '/user/refresh_token',
59 | maxAge: 7*24*60*60*1000 // 7d
60 | })
61 |
62 | res.json({accesstoken})
63 |
64 | } catch (err) {
65 | return res.status(500).json({msg: err.message})
66 | }
67 | },
68 | logout: async (req, res) =>{
69 | try {
70 | res.clearCookie('refreshtoken', {path: '/user/refresh_token'})
71 | return res.json({msg: "Logged out"})
72 | } catch (err) {
73 | return res.status(500).json({msg: err.message})
74 | }
75 | },
76 | refreshToken: (req, res) =>{
77 | try {
78 | const rf_token = req.cookies.refreshtoken;
79 | if(!rf_token) return res.status(400).json({msg: "Please Login or Register"})
80 |
81 | jwt.verify(rf_token, process.env.REFRESH_TOKEN_SECRET, (err, user) =>{
82 | if(err) return res.status(400).json({msg: "Please Login or Register"})
83 |
84 | const accesstoken = createAccessToken({id: user.id})
85 |
86 | res.json({accesstoken})
87 | })
88 |
89 | } catch (err) {
90 | return res.status(500).json({msg: err.message})
91 | }
92 |
93 | },
94 | getUser: async (req, res) =>{
95 | try {
96 | const user = await Users.findById(req.user.id).select('-password')
97 | if(!user) return res.status(400).json({msg: "User does not exist."})
98 |
99 | res.json(user)
100 | } catch (err) {
101 | return res.status(500).json({msg: err.message})
102 | }
103 | },
104 | addCart: async (req, res) =>{
105 | try {
106 | const user = await Users.findById(req.user.id)
107 | if(!user) return res.status(400).json({msg: "User does not exist."})
108 |
109 | await Users.findOneAndUpdate({_id: req.user.id}, {
110 | cart: req.body.cart
111 | })
112 |
113 | return res.json({msg: "Added to cart"})
114 | } catch (err) {
115 | return res.status(500).json({msg: err.message})
116 | }
117 | },
118 | history: async(req, res) =>{
119 | try {
120 | const history = await Payments.find({user_id: req.user.id})
121 |
122 | res.json(history)
123 | } catch (err) {
124 | return res.status(500).json({msg: err.message})
125 | }
126 | }
127 | }
128 |
129 |
130 | const createAccessToken = (user) =>{
131 | return jwt.sign(user, process.env.ACCESS_TOKEN_SECRET, {expiresIn: '11m'})
132 | }
133 | const createRefreshToken = (user) =>{
134 | return jwt.sign(user, process.env.REFRESH_TOKEN_SECRET, {expiresIn: '7d'})
135 | }
136 |
137 | module.exports = userCtrl
138 |
139 |
--------------------------------------------------------------------------------
/middleware/auth.js:
--------------------------------------------------------------------------------
1 | const jwt = require('jsonwebtoken')
2 |
3 | const auth = (req, res, next) =>{
4 | try {
5 | const token = req.header("Authorization")
6 | if(!token) return res.status(400).json({msg: "Invalid Authentication"})
7 |
8 | jwt.verify(token, process.env.ACCESS_TOKEN_SECRET, (err, user) =>{
9 | if(err) return res.status(400).json({msg: "Invalid Authentication"})
10 |
11 | req.user = user
12 | next()
13 | })
14 | } catch (err) {
15 | return res.status(500).json({msg: err.message})
16 | }
17 | }
18 |
19 | module.exports = auth
--------------------------------------------------------------------------------
/middleware/authAdmin.js:
--------------------------------------------------------------------------------
1 | const Users = require('../models/userModel')
2 |
3 | const authAdmin = async (req, res, next) =>{
4 | try {
5 | // Get user information by id
6 | const user = await Users.findOne({
7 | _id: req.user.id
8 | })
9 | if(user.role === 0)
10 | return res.status(400).json({msg: "Admin resources access denied"})
11 |
12 | next()
13 |
14 | } catch (err) {
15 | return res.status(500).json({msg: err.message})
16 | }
17 | }
18 |
19 | module.exports = authAdmin
--------------------------------------------------------------------------------
/models/categoryModel.js:
--------------------------------------------------------------------------------
1 | const mongoose = require('mongoose')
2 |
3 |
4 | const categorySchema = new mongoose.Schema({
5 | name: {
6 | type: String,
7 | required: true,
8 | trim: true,
9 | unique: true
10 | }
11 | }, {
12 | timestamps: true
13 | })
14 |
15 | module.exports = mongoose.model("Category", categorySchema)
--------------------------------------------------------------------------------
/models/paymentModel.js:
--------------------------------------------------------------------------------
1 | const mongoose = require('mongoose')
2 |
3 |
4 | const paymentSchema = new mongoose.Schema({
5 | user_id: {
6 | type: String,
7 | required: true
8 | },
9 | name:{
10 | type: String,
11 | required: true
12 | },
13 | email:{
14 | type: String,
15 | required: true
16 | },
17 | paymentID:{
18 | type: String,
19 | required: true
20 | },
21 | address:{
22 | type: Object,
23 | required: true
24 | },
25 | cart:{
26 | type: Array,
27 | default: []
28 | },
29 | status:{
30 | type: Boolean,
31 | default: false
32 | }
33 | }, {
34 | timestamps: true
35 | })
36 |
37 |
38 | module.exports = mongoose.model("Payments", paymentSchema)
--------------------------------------------------------------------------------
/models/productModel.js:
--------------------------------------------------------------------------------
1 | const mongoose = require('mongoose')
2 |
3 |
4 | const productSchema = new mongoose.Schema({
5 | product_id:{
6 | type: String,
7 | unique: true,
8 | trim: true,
9 | required: true
10 | },
11 | title:{
12 | type: String,
13 | trim: true,
14 | required: true
15 | },
16 | price:{
17 | type: Number,
18 | trim: true,
19 | required: true
20 | },
21 | description:{
22 | type: String,
23 | required: true
24 | },
25 | content:{
26 | type: String,
27 | required: true
28 | },
29 | images:{
30 | type: Object,
31 | required: true
32 | },
33 | category:{
34 | type: String,
35 | required: true
36 | },
37 | checked:{
38 | type: Boolean,
39 | default: false
40 | },
41 | sold:{
42 | type: Number,
43 | default: 0
44 | }
45 | }, {
46 | timestamps: true //important
47 | })
48 |
49 |
50 | module.exports = mongoose.model("Products", productSchema)
--------------------------------------------------------------------------------
/models/userModel.js:
--------------------------------------------------------------------------------
1 | const mongoose = require('mongoose')
2 |
3 | const userSchema = new mongoose.Schema({
4 | name: {
5 | type: String,
6 | required: true,
7 | trim: true
8 | },
9 | email: {
10 | type: String,
11 | required: true,
12 | unique: true
13 | },
14 | password: {
15 | type: String,
16 | required: true
17 | },
18 | role: {
19 | type: Number,
20 | default: 0
21 | },
22 | cart: {
23 | type: Array,
24 | default: []
25 | }
26 | }, {
27 | timestamps: true
28 | })
29 |
30 | module.exports = mongoose.model('Users', userSchema)
--------------------------------------------------------------------------------
/package-lock.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "ecommerce",
3 | "version": "1.0.0",
4 | "lockfileVersion": 1,
5 | "requires": true,
6 | "dependencies": {
7 | "@sindresorhus/is": {
8 | "version": "0.14.0",
9 | "resolved": "https://registry.npmjs.org/@sindresorhus/is/-/is-0.14.0.tgz",
10 | "integrity": "sha512-9NET910DNaIPngYnLLPeg+Ogzqsi9uM4mSboU5y6p8S5DzMTVEsJZrawi+BoDNUVBa2DhJqQYUFvMDfgU062LQ==",
11 | "dev": true
12 | },
13 | "@szmarczak/http-timer": {
14 | "version": "1.1.2",
15 | "resolved": "https://registry.npmjs.org/@szmarczak/http-timer/-/http-timer-1.1.2.tgz",
16 | "integrity": "sha512-XIB2XbzHTN6ieIjfIMV9hlVcfPU26s2vafYWQcZHWXHOxiaRZYEDKEwdl129Zyg50+foYV2jCgtrqSA6qNuNSA==",
17 | "dev": true,
18 | "requires": {
19 | "defer-to-connect": "^1.0.1"
20 | }
21 | },
22 | "@types/color-name": {
23 | "version": "1.1.1",
24 | "resolved": "https://registry.npmjs.org/@types/color-name/-/color-name-1.1.1.tgz",
25 | "integrity": "sha512-rr+OQyAjxze7GgWrSaJwydHStIhHq2lvY3BOC2Mj7KnzI7XK0Uw1TOOdI9lDoajEbSWLiYgoo4f1R51erQfhPQ==",
26 | "dev": true
27 | },
28 | "abbrev": {
29 | "version": "1.1.1",
30 | "resolved": "https://registry.npmjs.org/abbrev/-/abbrev-1.1.1.tgz",
31 | "integrity": "sha512-nne9/IiQ/hzIhY6pdDnbBtz7DjPTKrY00P/zvPSm5pOFkl6xuGrGnXn/VtTNNfNtAfZ9/1RtehkszU9qcTii0Q=="
32 | },
33 | "accepts": {
34 | "version": "1.3.7",
35 | "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.7.tgz",
36 | "integrity": "sha512-Il80Qs2WjYlJIBNzNkK6KYqlVMTbZLXgHx2oT0pU/fjRHyEp+PEfEPY0R3WCwAGVOtauxh1hOxNgIf5bv7dQpA==",
37 | "requires": {
38 | "mime-types": "~2.1.24",
39 | "negotiator": "0.6.2"
40 | }
41 | },
42 | "ansi-align": {
43 | "version": "3.0.0",
44 | "resolved": "https://registry.npmjs.org/ansi-align/-/ansi-align-3.0.0.tgz",
45 | "integrity": "sha512-ZpClVKqXN3RGBmKibdfWzqCY4lnjEuoNzU5T0oEFpfd/z5qJHVarukridD4juLO2FXMiwUQxr9WqQtaYa8XRYw==",
46 | "dev": true,
47 | "requires": {
48 | "string-width": "^3.0.0"
49 | },
50 | "dependencies": {
51 | "ansi-regex": {
52 | "version": "4.1.0",
53 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz",
54 | "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==",
55 | "dev": true
56 | },
57 | "is-fullwidth-code-point": {
58 | "version": "2.0.0",
59 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz",
60 | "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=",
61 | "dev": true
62 | },
63 | "string-width": {
64 | "version": "3.1.0",
65 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz",
66 | "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==",
67 | "dev": true,
68 | "requires": {
69 | "emoji-regex": "^7.0.1",
70 | "is-fullwidth-code-point": "^2.0.0",
71 | "strip-ansi": "^5.1.0"
72 | }
73 | },
74 | "strip-ansi": {
75 | "version": "5.2.0",
76 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz",
77 | "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==",
78 | "dev": true,
79 | "requires": {
80 | "ansi-regex": "^4.1.0"
81 | }
82 | }
83 | }
84 | },
85 | "ansi-regex": {
86 | "version": "2.1.1",
87 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz",
88 | "integrity": "sha1-w7M6te42DYbg5ijwRorn7yfWVN8="
89 | },
90 | "ansi-styles": {
91 | "version": "3.2.1",
92 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz",
93 | "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==",
94 | "requires": {
95 | "color-convert": "^1.9.0"
96 | }
97 | },
98 | "anymatch": {
99 | "version": "3.1.1",
100 | "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.1.tgz",
101 | "integrity": "sha512-mM8522psRCqzV+6LhomX5wgp25YVibjh8Wj23I5RPkPppSVSjyKD2A2mBJmWGa+KN7f2D6LNh9jkBCeyLktzjg==",
102 | "dev": true,
103 | "requires": {
104 | "normalize-path": "^3.0.0",
105 | "picomatch": "^2.0.4"
106 | }
107 | },
108 | "aproba": {
109 | "version": "1.2.0",
110 | "resolved": "https://registry.npmjs.org/aproba/-/aproba-1.2.0.tgz",
111 | "integrity": "sha512-Y9J6ZjXtoYh8RnXVCMOU/ttDmk1aBjunq9vO0ta5x85WDQiQfUF9sIPBITdbiiIVcBo03Hi3jMxigBtsddlXRw=="
112 | },
113 | "are-we-there-yet": {
114 | "version": "1.1.5",
115 | "resolved": "https://registry.npmjs.org/are-we-there-yet/-/are-we-there-yet-1.1.5.tgz",
116 | "integrity": "sha512-5hYdAkZlcG8tOLujVDTgCT+uPX0VnpAH28gWsLfzpXYm7wP6mp5Q/gYyR7YQ0cKVJcXJnl3j2kpBan13PtQf6w==",
117 | "requires": {
118 | "delegates": "^1.0.0",
119 | "readable-stream": "^2.0.6"
120 | }
121 | },
122 | "array-flatten": {
123 | "version": "1.1.1",
124 | "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz",
125 | "integrity": "sha1-ml9pkFGx5wczKPKgCJaLZOopVdI="
126 | },
127 | "balanced-match": {
128 | "version": "1.0.0",
129 | "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.0.tgz",
130 | "integrity": "sha1-ibTRmasr7kneFk6gK4nORi1xt2c="
131 | },
132 | "bcrypt": {
133 | "version": "5.0.0",
134 | "resolved": "https://registry.npmjs.org/bcrypt/-/bcrypt-5.0.0.tgz",
135 | "integrity": "sha512-jB0yCBl4W/kVHM2whjfyqnxTmOHkCX4kHEa5nYKSoGeYe8YrjTYTc87/6bwt1g8cmV0QrbhKriETg9jWtcREhg==",
136 | "requires": {
137 | "node-addon-api": "^3.0.0",
138 | "node-pre-gyp": "0.15.0"
139 | }
140 | },
141 | "binary-extensions": {
142 | "version": "2.1.0",
143 | "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.1.0.tgz",
144 | "integrity": "sha512-1Yj8h9Q+QDF5FzhMs/c9+6UntbD5MkRfRwac8DoEm9ZfUBZ7tZ55YcGVAzEe4bXsdQHEk+s9S5wsOKVdZrw0tQ==",
145 | "dev": true
146 | },
147 | "bl": {
148 | "version": "2.2.1",
149 | "resolved": "https://registry.npmjs.org/bl/-/bl-2.2.1.tgz",
150 | "integrity": "sha512-6Pesp1w0DEX1N550i/uGV/TqucVL4AM/pgThFSN/Qq9si1/DF9aIHs1BxD8V/QU0HoeHO6cQRTAuYnLPKq1e4g==",
151 | "requires": {
152 | "readable-stream": "^2.3.5",
153 | "safe-buffer": "^5.1.1"
154 | }
155 | },
156 | "bluebird": {
157 | "version": "3.5.1",
158 | "resolved": "https://registry.npmjs.org/bluebird/-/bluebird-3.5.1.tgz",
159 | "integrity": "sha512-MKiLiV+I1AA596t9w1sQJ8jkiSr5+ZKi0WKrYGUn6d1Fx+Ij4tIj+m2WMQSGczs5jZVxV339chE8iwk6F64wjA=="
160 | },
161 | "body-parser": {
162 | "version": "1.19.0",
163 | "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.19.0.tgz",
164 | "integrity": "sha512-dhEPs72UPbDnAQJ9ZKMNTP6ptJaionhP5cBb541nXPlW60Jepo9RV/a4fX4XWW9CuFNK22krhrj1+rgzifNCsw==",
165 | "requires": {
166 | "bytes": "3.1.0",
167 | "content-type": "~1.0.4",
168 | "debug": "2.6.9",
169 | "depd": "~1.1.2",
170 | "http-errors": "1.7.2",
171 | "iconv-lite": "0.4.24",
172 | "on-finished": "~2.3.0",
173 | "qs": "6.7.0",
174 | "raw-body": "2.4.0",
175 | "type-is": "~1.6.17"
176 | },
177 | "dependencies": {
178 | "debug": {
179 | "version": "2.6.9",
180 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz",
181 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==",
182 | "requires": {
183 | "ms": "2.0.0"
184 | }
185 | },
186 | "ms": {
187 | "version": "2.0.0",
188 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz",
189 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g="
190 | }
191 | }
192 | },
193 | "boxen": {
194 | "version": "4.2.0",
195 | "resolved": "https://registry.npmjs.org/boxen/-/boxen-4.2.0.tgz",
196 | "integrity": "sha512-eB4uT9RGzg2odpER62bBwSLvUeGC+WbRjjyyFhGsKnc8wp/m0+hQsMUvUe3H2V0D5vw0nBdO1hCJoZo5mKeuIQ==",
197 | "dev": true,
198 | "requires": {
199 | "ansi-align": "^3.0.0",
200 | "camelcase": "^5.3.1",
201 | "chalk": "^3.0.0",
202 | "cli-boxes": "^2.2.0",
203 | "string-width": "^4.1.0",
204 | "term-size": "^2.1.0",
205 | "type-fest": "^0.8.1",
206 | "widest-line": "^3.1.0"
207 | },
208 | "dependencies": {
209 | "ansi-regex": {
210 | "version": "5.0.0",
211 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.0.tgz",
212 | "integrity": "sha512-bY6fj56OUQ0hU1KjFNDQuJFezqKdrAyFdIevADiqrWHwSlbmBNMHp5ak2f40Pm8JTFyM2mqxkG6ngkHO11f/lg==",
213 | "dev": true
214 | },
215 | "ansi-styles": {
216 | "version": "4.2.1",
217 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.2.1.tgz",
218 | "integrity": "sha512-9VGjrMsG1vePxcSweQsN20KY/c4zN0h9fLjqAbwbPfahM3t+NL+M9HC8xeXG2I8pX5NoamTGNuomEUFI7fcUjA==",
219 | "dev": true,
220 | "requires": {
221 | "@types/color-name": "^1.1.1",
222 | "color-convert": "^2.0.1"
223 | }
224 | },
225 | "chalk": {
226 | "version": "3.0.0",
227 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-3.0.0.tgz",
228 | "integrity": "sha512-4D3B6Wf41KOYRFdszmDqMCGq5VV/uMAB273JILmO+3jAlh8X4qDtdtgCR3fxtbLEMzSx22QdhnDcJvu2u1fVwg==",
229 | "dev": true,
230 | "requires": {
231 | "ansi-styles": "^4.1.0",
232 | "supports-color": "^7.1.0"
233 | }
234 | },
235 | "color-convert": {
236 | "version": "2.0.1",
237 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz",
238 | "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==",
239 | "dev": true,
240 | "requires": {
241 | "color-name": "~1.1.4"
242 | }
243 | },
244 | "color-name": {
245 | "version": "1.1.4",
246 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz",
247 | "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==",
248 | "dev": true
249 | },
250 | "emoji-regex": {
251 | "version": "8.0.0",
252 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz",
253 | "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==",
254 | "dev": true
255 | },
256 | "has-flag": {
257 | "version": "4.0.0",
258 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz",
259 | "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==",
260 | "dev": true
261 | },
262 | "is-fullwidth-code-point": {
263 | "version": "3.0.0",
264 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz",
265 | "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==",
266 | "dev": true
267 | },
268 | "string-width": {
269 | "version": "4.2.0",
270 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.0.tgz",
271 | "integrity": "sha512-zUz5JD+tgqtuDjMhwIg5uFVV3dtqZ9yQJlZVfq4I01/K5Paj5UHj7VyrQOJvzawSVlKpObApbfD0Ed6yJc+1eg==",
272 | "dev": true,
273 | "requires": {
274 | "emoji-regex": "^8.0.0",
275 | "is-fullwidth-code-point": "^3.0.0",
276 | "strip-ansi": "^6.0.0"
277 | }
278 | },
279 | "strip-ansi": {
280 | "version": "6.0.0",
281 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.0.tgz",
282 | "integrity": "sha512-AuvKTrTfQNYNIctbR1K/YGTR1756GycPsg7b9bdV9Duqur4gv6aKqHXah67Z8ImS7WEz5QVcOtlfW2rZEugt6w==",
283 | "dev": true,
284 | "requires": {
285 | "ansi-regex": "^5.0.0"
286 | }
287 | },
288 | "supports-color": {
289 | "version": "7.2.0",
290 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz",
291 | "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==",
292 | "dev": true,
293 | "requires": {
294 | "has-flag": "^4.0.0"
295 | }
296 | }
297 | }
298 | },
299 | "brace-expansion": {
300 | "version": "1.1.11",
301 | "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz",
302 | "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==",
303 | "requires": {
304 | "balanced-match": "^1.0.0",
305 | "concat-map": "0.0.1"
306 | }
307 | },
308 | "braces": {
309 | "version": "3.0.2",
310 | "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz",
311 | "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==",
312 | "dev": true,
313 | "requires": {
314 | "fill-range": "^7.0.1"
315 | }
316 | },
317 | "bson": {
318 | "version": "1.1.5",
319 | "resolved": "https://registry.npmjs.org/bson/-/bson-1.1.5.tgz",
320 | "integrity": "sha512-kDuEzldR21lHciPQAIulLs1LZlCXdLziXI6Mb/TDkwXhb//UORJNPXgcRs2CuO4H0DcMkpfT3/ySsP3unoZjBg=="
321 | },
322 | "buffer-equal-constant-time": {
323 | "version": "1.0.1",
324 | "resolved": "https://registry.npmjs.org/buffer-equal-constant-time/-/buffer-equal-constant-time-1.0.1.tgz",
325 | "integrity": "sha1-+OcRMvf/5uAaXJaXpMbz5I1cyBk="
326 | },
327 | "busboy": {
328 | "version": "0.3.1",
329 | "resolved": "https://registry.npmjs.org/busboy/-/busboy-0.3.1.tgz",
330 | "integrity": "sha512-y7tTxhGKXcyBxRKAni+awqx8uqaJKrSFSNFSeRG5CsWNdmy2BIK+6VGWEW7TZnIO/533mtMEA4rOevQV815YJw==",
331 | "requires": {
332 | "dicer": "0.3.0"
333 | }
334 | },
335 | "bytes": {
336 | "version": "3.1.0",
337 | "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.0.tgz",
338 | "integrity": "sha512-zauLjrfCG+xvoyaqLoV8bLVXXNGC4JqlxFCutSDWA6fJrTo2ZuvLYTqZ7aHBLZSMOopbzwv8f+wZcVzfVTI2Dg=="
339 | },
340 | "cacheable-request": {
341 | "version": "6.1.0",
342 | "resolved": "https://registry.npmjs.org/cacheable-request/-/cacheable-request-6.1.0.tgz",
343 | "integrity": "sha512-Oj3cAGPCqOZX7Rz64Uny2GYAZNliQSqfbePrgAQ1wKAihYmCUnraBtJtKcGR4xz7wF+LoJC+ssFZvv5BgF9Igg==",
344 | "dev": true,
345 | "requires": {
346 | "clone-response": "^1.0.2",
347 | "get-stream": "^5.1.0",
348 | "http-cache-semantics": "^4.0.0",
349 | "keyv": "^3.0.0",
350 | "lowercase-keys": "^2.0.0",
351 | "normalize-url": "^4.1.0",
352 | "responselike": "^1.0.2"
353 | },
354 | "dependencies": {
355 | "get-stream": {
356 | "version": "5.2.0",
357 | "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-5.2.0.tgz",
358 | "integrity": "sha512-nBF+F1rAZVCu/p7rjzgA+Yb4lfYXrpl7a6VmJrU8wF9I1CKvP/QwPNZHnOlwbTkY6dvtFIzFMSyQXbLoTQPRpA==",
359 | "dev": true,
360 | "requires": {
361 | "pump": "^3.0.0"
362 | }
363 | },
364 | "lowercase-keys": {
365 | "version": "2.0.0",
366 | "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-2.0.0.tgz",
367 | "integrity": "sha512-tqNXrS78oMOE73NMxK4EMLQsQowWf8jKooH9g7xPavRT706R6bkQJ6DY2Te7QukaZsulxa30wQ7bk0pm4XiHmA==",
368 | "dev": true
369 | }
370 | }
371 | },
372 | "camelcase": {
373 | "version": "5.3.1",
374 | "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz",
375 | "integrity": "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg=="
376 | },
377 | "chalk": {
378 | "version": "2.4.2",
379 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz",
380 | "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==",
381 | "requires": {
382 | "ansi-styles": "^3.2.1",
383 | "escape-string-regexp": "^1.0.5",
384 | "supports-color": "^5.3.0"
385 | },
386 | "dependencies": {
387 | "supports-color": {
388 | "version": "5.5.0",
389 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz",
390 | "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==",
391 | "requires": {
392 | "has-flag": "^3.0.0"
393 | }
394 | }
395 | }
396 | },
397 | "chokidar": {
398 | "version": "3.4.2",
399 | "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.4.2.tgz",
400 | "integrity": "sha512-IZHaDeBeI+sZJRX7lGcXsdzgvZqKv6sECqsbErJA4mHWfpRrD8B97kSFN4cQz6nGBGiuFia1MKR4d6c1o8Cv7A==",
401 | "dev": true,
402 | "requires": {
403 | "anymatch": "~3.1.1",
404 | "braces": "~3.0.2",
405 | "fsevents": "~2.1.2",
406 | "glob-parent": "~5.1.0",
407 | "is-binary-path": "~2.1.0",
408 | "is-glob": "~4.0.1",
409 | "normalize-path": "~3.0.0",
410 | "readdirp": "~3.4.0"
411 | }
412 | },
413 | "chownr": {
414 | "version": "1.1.4",
415 | "resolved": "https://registry.npmjs.org/chownr/-/chownr-1.1.4.tgz",
416 | "integrity": "sha512-jJ0bqzaylmJtVnNgzTeSOs8DPavpbYgEr/b0YL8/2GO3xJEhInFmhKMUnEJQjZumK7KXGFhUy89PrsJWlakBVg=="
417 | },
418 | "ci-info": {
419 | "version": "2.0.0",
420 | "resolved": "https://registry.npmjs.org/ci-info/-/ci-info-2.0.0.tgz",
421 | "integrity": "sha512-5tK7EtrZ0N+OLFMthtqOj4fI2Jeb88C4CAZPu25LDVUgXJ0A3Js4PMGqrn0JU1W0Mh1/Z8wZzYPxqUrXeBboCQ==",
422 | "dev": true
423 | },
424 | "cli-boxes": {
425 | "version": "2.2.1",
426 | "resolved": "https://registry.npmjs.org/cli-boxes/-/cli-boxes-2.2.1.tgz",
427 | "integrity": "sha512-y4coMcylgSCdVinjiDBuR8PCC2bLjyGTwEmPb9NHR/QaNU6EUOXcTY/s6VjGMD6ENSEaeQYHCY0GNGS5jfMwPw==",
428 | "dev": true
429 | },
430 | "cliui": {
431 | "version": "5.0.0",
432 | "resolved": "https://registry.npmjs.org/cliui/-/cliui-5.0.0.tgz",
433 | "integrity": "sha512-PYeGSEmmHM6zvoef2w8TPzlrnNpXIjTipYK780YswmIP9vjxmd6Y2a3CB2Ks6/AU8NHjZugXvo8w3oWM2qnwXA==",
434 | "requires": {
435 | "string-width": "^3.1.0",
436 | "strip-ansi": "^5.2.0",
437 | "wrap-ansi": "^5.1.0"
438 | },
439 | "dependencies": {
440 | "ansi-regex": {
441 | "version": "4.1.0",
442 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz",
443 | "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg=="
444 | },
445 | "is-fullwidth-code-point": {
446 | "version": "2.0.0",
447 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz",
448 | "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8="
449 | },
450 | "string-width": {
451 | "version": "3.1.0",
452 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz",
453 | "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==",
454 | "requires": {
455 | "emoji-regex": "^7.0.1",
456 | "is-fullwidth-code-point": "^2.0.0",
457 | "strip-ansi": "^5.1.0"
458 | }
459 | },
460 | "strip-ansi": {
461 | "version": "5.2.0",
462 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz",
463 | "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==",
464 | "requires": {
465 | "ansi-regex": "^4.1.0"
466 | }
467 | }
468 | }
469 | },
470 | "clone-response": {
471 | "version": "1.0.2",
472 | "resolved": "https://registry.npmjs.org/clone-response/-/clone-response-1.0.2.tgz",
473 | "integrity": "sha1-0dyXOSAxTfZ/vrlCI7TuNQI56Ws=",
474 | "dev": true,
475 | "requires": {
476 | "mimic-response": "^1.0.0"
477 | }
478 | },
479 | "cloudinary": {
480 | "version": "1.23.0",
481 | "resolved": "https://registry.npmjs.org/cloudinary/-/cloudinary-1.23.0.tgz",
482 | "integrity": "sha512-akOxzroonvwWkuSVq7BI50nYpZPRXc5DbQIYETCVeKX9ZoToH2Gvc3MdUH63UtKiszuGYE51q2B+jQsJkBp2AQ==",
483 | "requires": {
484 | "cloudinary-core": "^2.10.2",
485 | "core-js": "3.6.5",
486 | "lodash": "^4.17.11",
487 | "q": "^1.5.1"
488 | }
489 | },
490 | "cloudinary-core": {
491 | "version": "2.11.3",
492 | "resolved": "https://registry.npmjs.org/cloudinary-core/-/cloudinary-core-2.11.3.tgz",
493 | "integrity": "sha512-ZRnpjSgvx+LbSf+aEz5NKzxDB4Z0436aY/0BSDa90kAHiwAyd84VyEi95I74SE80e15Ri9t5S2xtksTXpzk9Xw=="
494 | },
495 | "code-point-at": {
496 | "version": "1.1.0",
497 | "resolved": "https://registry.npmjs.org/code-point-at/-/code-point-at-1.1.0.tgz",
498 | "integrity": "sha1-DQcLTQQ6W+ozovGkDi7bPZpMz3c="
499 | },
500 | "color-convert": {
501 | "version": "1.9.3",
502 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz",
503 | "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==",
504 | "requires": {
505 | "color-name": "1.1.3"
506 | }
507 | },
508 | "color-name": {
509 | "version": "1.1.3",
510 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz",
511 | "integrity": "sha1-p9BVi9icQveV3UIyj3QIMcpTvCU="
512 | },
513 | "concat-map": {
514 | "version": "0.0.1",
515 | "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz",
516 | "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s="
517 | },
518 | "concurrently": {
519 | "version": "5.3.0",
520 | "resolved": "https://registry.npmjs.org/concurrently/-/concurrently-5.3.0.tgz",
521 | "integrity": "sha512-8MhqOB6PWlBfA2vJ8a0bSFKATOdWlHiQlk11IfmQBPaHVP8oP2gsh2MObE6UR3hqDHqvaIvLTyceNW6obVuFHQ==",
522 | "requires": {
523 | "chalk": "^2.4.2",
524 | "date-fns": "^2.0.1",
525 | "lodash": "^4.17.15",
526 | "read-pkg": "^4.0.1",
527 | "rxjs": "^6.5.2",
528 | "spawn-command": "^0.0.2-1",
529 | "supports-color": "^6.1.0",
530 | "tree-kill": "^1.2.2",
531 | "yargs": "^13.3.0"
532 | }
533 | },
534 | "configstore": {
535 | "version": "5.0.1",
536 | "resolved": "https://registry.npmjs.org/configstore/-/configstore-5.0.1.tgz",
537 | "integrity": "sha512-aMKprgk5YhBNyH25hj8wGt2+D52Sw1DRRIzqBwLp2Ya9mFmY8KPvvtvmna8SxVR9JMZ4kzMD68N22vlaRpkeFA==",
538 | "dev": true,
539 | "requires": {
540 | "dot-prop": "^5.2.0",
541 | "graceful-fs": "^4.1.2",
542 | "make-dir": "^3.0.0",
543 | "unique-string": "^2.0.0",
544 | "write-file-atomic": "^3.0.0",
545 | "xdg-basedir": "^4.0.0"
546 | }
547 | },
548 | "console-control-strings": {
549 | "version": "1.1.0",
550 | "resolved": "https://registry.npmjs.org/console-control-strings/-/console-control-strings-1.1.0.tgz",
551 | "integrity": "sha1-PXz0Rk22RG6mRL9LOVB/mFEAjo4="
552 | },
553 | "content-disposition": {
554 | "version": "0.5.3",
555 | "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.3.tgz",
556 | "integrity": "sha512-ExO0774ikEObIAEV9kDo50o+79VCUdEB6n6lzKgGwupcVeRlhrj3qGAfwq8G6uBJjkqLrhT0qEYFcWng8z1z0g==",
557 | "requires": {
558 | "safe-buffer": "5.1.2"
559 | }
560 | },
561 | "content-type": {
562 | "version": "1.0.4",
563 | "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.4.tgz",
564 | "integrity": "sha512-hIP3EEPs8tB9AT1L+NUqtwOAps4mk2Zob89MWXMHjHWg9milF/j4osnnQLXBCBFBk/tvIG/tUc9mOUJiPBhPXA=="
565 | },
566 | "cookie": {
567 | "version": "0.4.0",
568 | "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.4.0.tgz",
569 | "integrity": "sha512-+Hp8fLp57wnUSt0tY0tHEXh4voZRDnoIrZPqlo3DPiI4y9lwg/jqx+1Om94/W6ZaPDOUbnjOt/99w66zk+l1Xg=="
570 | },
571 | "cookie-parser": {
572 | "version": "1.4.5",
573 | "resolved": "https://registry.npmjs.org/cookie-parser/-/cookie-parser-1.4.5.tgz",
574 | "integrity": "sha512-f13bPUj/gG/5mDr+xLmSxxDsB9DQiTIfhJS/sqjrmfAWiAN+x2O4i/XguTL9yDZ+/IFDanJ+5x7hC4CXT9Tdzw==",
575 | "requires": {
576 | "cookie": "0.4.0",
577 | "cookie-signature": "1.0.6"
578 | }
579 | },
580 | "cookie-signature": {
581 | "version": "1.0.6",
582 | "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz",
583 | "integrity": "sha1-4wOogrNCzD7oylE6eZmXNNqzriw="
584 | },
585 | "core-js": {
586 | "version": "3.6.5",
587 | "resolved": "https://registry.npmjs.org/core-js/-/core-js-3.6.5.tgz",
588 | "integrity": "sha512-vZVEEwZoIsI+vPEuoF9Iqf5H7/M3eeQqWlQnYa8FSKKePuYTf5MWnxb5SDAzCa60b3JBRS5g9b+Dq7b1y/RCrA=="
589 | },
590 | "core-util-is": {
591 | "version": "1.0.2",
592 | "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.2.tgz",
593 | "integrity": "sha1-tf1UIgqivFq1eqtxQMlAdUUDwac="
594 | },
595 | "cors": {
596 | "version": "2.8.5",
597 | "resolved": "https://registry.npmjs.org/cors/-/cors-2.8.5.tgz",
598 | "integrity": "sha512-KIHbLJqu73RGr/hnbrO9uBeixNGuvSQjul/jdFvS/KFSIH1hWVd1ng7zOHx+YrEfInLG7q4n6GHQ9cDtxv/P6g==",
599 | "requires": {
600 | "object-assign": "^4",
601 | "vary": "^1"
602 | }
603 | },
604 | "crypto-random-string": {
605 | "version": "2.0.0",
606 | "resolved": "https://registry.npmjs.org/crypto-random-string/-/crypto-random-string-2.0.0.tgz",
607 | "integrity": "sha512-v1plID3y9r/lPhviJ1wrXpLeyUIGAZ2SHNYTEapm7/8A9nLPoyvVp3RK/EPFqn5kEznyWgYZNsRtYYIWbuG8KA==",
608 | "dev": true
609 | },
610 | "date-fns": {
611 | "version": "2.16.1",
612 | "resolved": "https://registry.npmjs.org/date-fns/-/date-fns-2.16.1.tgz",
613 | "integrity": "sha512-sAJVKx/FqrLYHAQeN7VpJrPhagZc9R4ImZIWYRFZaaohR3KzmuK88touwsSwSVT8Qcbd4zoDsnGfX4GFB4imyQ=="
614 | },
615 | "debug": {
616 | "version": "3.2.6",
617 | "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.6.tgz",
618 | "integrity": "sha512-mel+jf7nrtEl5Pn1Qx46zARXKDpBbvzezse7p7LqINmdoIk8PYP5SySaxEmYv6TZ0JyEKA1hsCId6DIhgITtWQ==",
619 | "requires": {
620 | "ms": "^2.1.1"
621 | }
622 | },
623 | "decamelize": {
624 | "version": "1.2.0",
625 | "resolved": "https://registry.npmjs.org/decamelize/-/decamelize-1.2.0.tgz",
626 | "integrity": "sha1-9lNNFRSCabIDUue+4m9QH5oZEpA="
627 | },
628 | "decompress-response": {
629 | "version": "3.3.0",
630 | "resolved": "https://registry.npmjs.org/decompress-response/-/decompress-response-3.3.0.tgz",
631 | "integrity": "sha1-gKTdMjdIOEv6JICDYirt7Jgq3/M=",
632 | "dev": true,
633 | "requires": {
634 | "mimic-response": "^1.0.0"
635 | }
636 | },
637 | "deep-extend": {
638 | "version": "0.6.0",
639 | "resolved": "https://registry.npmjs.org/deep-extend/-/deep-extend-0.6.0.tgz",
640 | "integrity": "sha512-LOHxIOaPYdHlJRtCQfDIVZtfw/ufM8+rVj649RIHzcm/vGwQRXFt6OPqIFWsm2XEMrNIEtWR64sY1LEKD2vAOA=="
641 | },
642 | "defer-to-connect": {
643 | "version": "1.1.3",
644 | "resolved": "https://registry.npmjs.org/defer-to-connect/-/defer-to-connect-1.1.3.tgz",
645 | "integrity": "sha512-0ISdNousHvZT2EiFlZeZAHBUvSxmKswVCEf8hW7KWgG4a8MVEu/3Vb6uWYozkjylyCxe0JBIiRB1jV45S70WVQ==",
646 | "dev": true
647 | },
648 | "delegates": {
649 | "version": "1.0.0",
650 | "resolved": "https://registry.npmjs.org/delegates/-/delegates-1.0.0.tgz",
651 | "integrity": "sha1-hMbhWbgZBP3KWaDvRM2HDTElD5o="
652 | },
653 | "denque": {
654 | "version": "1.4.1",
655 | "resolved": "https://registry.npmjs.org/denque/-/denque-1.4.1.tgz",
656 | "integrity": "sha512-OfzPuSZKGcgr96rf1oODnfjqBFmr1DVoc/TrItj3Ohe0Ah1C5WX5Baquw/9U9KovnQ88EqmJbD66rKYUQYN1tQ=="
657 | },
658 | "depd": {
659 | "version": "1.1.2",
660 | "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.2.tgz",
661 | "integrity": "sha1-m81S4UwJd2PnSbJ0xDRu0uVgtak="
662 | },
663 | "destroy": {
664 | "version": "1.0.4",
665 | "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.0.4.tgz",
666 | "integrity": "sha1-l4hXRCxEdJ5CBmE+N5RiBYJqvYA="
667 | },
668 | "detect-libc": {
669 | "version": "1.0.3",
670 | "resolved": "https://registry.npmjs.org/detect-libc/-/detect-libc-1.0.3.tgz",
671 | "integrity": "sha1-+hN8S9aY7fVc1c0CrFWfkaTEups="
672 | },
673 | "dicer": {
674 | "version": "0.3.0",
675 | "resolved": "https://registry.npmjs.org/dicer/-/dicer-0.3.0.tgz",
676 | "integrity": "sha512-MdceRRWqltEG2dZqO769g27N/3PXfcKl04VhYnBlo2YhH7zPi88VebsjTKclaOyiuMaGU72hTfw3VkUitGcVCA==",
677 | "requires": {
678 | "streamsearch": "0.1.2"
679 | }
680 | },
681 | "dot-prop": {
682 | "version": "5.3.0",
683 | "resolved": "https://registry.npmjs.org/dot-prop/-/dot-prop-5.3.0.tgz",
684 | "integrity": "sha512-QM8q3zDe58hqUqjraQOmzZ1LIH9SWQJTlEKCH4kJ2oQvLZk7RbQXvtDM2XEq3fwkV9CCvvH4LA0AV+ogFsBM2Q==",
685 | "dev": true,
686 | "requires": {
687 | "is-obj": "^2.0.0"
688 | }
689 | },
690 | "dotenv": {
691 | "version": "8.2.0",
692 | "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-8.2.0.tgz",
693 | "integrity": "sha512-8sJ78ElpbDJBHNeBzUbUVLsqKdccaa/BXF1uPTw3GrvQTBgrQrtObr2mUrE38vzYd8cEv+m/JBfDLioYcfXoaw=="
694 | },
695 | "duplexer3": {
696 | "version": "0.1.4",
697 | "resolved": "https://registry.npmjs.org/duplexer3/-/duplexer3-0.1.4.tgz",
698 | "integrity": "sha1-7gHdHKwO08vH/b6jfcCo8c4ALOI=",
699 | "dev": true
700 | },
701 | "ecdsa-sig-formatter": {
702 | "version": "1.0.11",
703 | "resolved": "https://registry.npmjs.org/ecdsa-sig-formatter/-/ecdsa-sig-formatter-1.0.11.tgz",
704 | "integrity": "sha512-nagl3RYrbNv6kQkeJIpt6NJZy8twLB/2vtz6yN9Z4vRKHN4/QZJIEbqohALSgwKdnksuY3k5Addp5lg8sVoVcQ==",
705 | "requires": {
706 | "safe-buffer": "^5.0.1"
707 | }
708 | },
709 | "ee-first": {
710 | "version": "1.1.1",
711 | "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz",
712 | "integrity": "sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0="
713 | },
714 | "emoji-regex": {
715 | "version": "7.0.3",
716 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-7.0.3.tgz",
717 | "integrity": "sha512-CwBLREIQ7LvYFB0WyRvwhq5N5qPhc6PMjD6bYggFlI5YyDgl+0vxq5VHbMOFqLg7hfWzmu8T5Z1QofhmTIhItA=="
718 | },
719 | "encodeurl": {
720 | "version": "1.0.2",
721 | "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz",
722 | "integrity": "sha1-rT/0yG7C0CkyL1oCw6mmBslbP1k="
723 | },
724 | "end-of-stream": {
725 | "version": "1.4.4",
726 | "resolved": "https://registry.npmjs.org/end-of-stream/-/end-of-stream-1.4.4.tgz",
727 | "integrity": "sha512-+uw1inIHVPQoaVuHzRyXd21icM+cnt4CzD5rW+NC1wjOUSTOs+Te7FOv7AhN7vS9x/oIyhLP5PR1H+phQAHu5Q==",
728 | "dev": true,
729 | "requires": {
730 | "once": "^1.4.0"
731 | }
732 | },
733 | "error-ex": {
734 | "version": "1.3.2",
735 | "resolved": "https://registry.npmjs.org/error-ex/-/error-ex-1.3.2.tgz",
736 | "integrity": "sha512-7dFHNmqeFSEt2ZBsCriorKnn3Z2pj+fd9kmI6QoWw4//DL+icEBfc0U7qJCisqrTsKTjw4fNFy2pW9OqStD84g==",
737 | "requires": {
738 | "is-arrayish": "^0.2.1"
739 | }
740 | },
741 | "escape-goat": {
742 | "version": "2.1.1",
743 | "resolved": "https://registry.npmjs.org/escape-goat/-/escape-goat-2.1.1.tgz",
744 | "integrity": "sha512-8/uIhbG12Csjy2JEW7D9pHbreaVaS/OpN3ycnyvElTdwM5n6GY6W6e2IPemfvGZeUMqZ9A/3GqIZMgKnBhAw/Q==",
745 | "dev": true
746 | },
747 | "escape-html": {
748 | "version": "1.0.3",
749 | "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz",
750 | "integrity": "sha1-Aljq5NPQwJdN4cFpGI7wBR0dGYg="
751 | },
752 | "escape-string-regexp": {
753 | "version": "1.0.5",
754 | "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz",
755 | "integrity": "sha1-G2HAViGQqN/2rjuyzwIAyhMLhtQ="
756 | },
757 | "etag": {
758 | "version": "1.8.1",
759 | "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz",
760 | "integrity": "sha1-Qa4u62XvpiJorr/qg6x9eSmbCIc="
761 | },
762 | "express": {
763 | "version": "4.17.1",
764 | "resolved": "https://registry.npmjs.org/express/-/express-4.17.1.tgz",
765 | "integrity": "sha512-mHJ9O79RqluphRrcw2X/GTh3k9tVv8YcoyY4Kkh4WDMUYKRZUq0h1o0w2rrrxBqM7VoeUVqgb27xlEMXTnYt4g==",
766 | "requires": {
767 | "accepts": "~1.3.7",
768 | "array-flatten": "1.1.1",
769 | "body-parser": "1.19.0",
770 | "content-disposition": "0.5.3",
771 | "content-type": "~1.0.4",
772 | "cookie": "0.4.0",
773 | "cookie-signature": "1.0.6",
774 | "debug": "2.6.9",
775 | "depd": "~1.1.2",
776 | "encodeurl": "~1.0.2",
777 | "escape-html": "~1.0.3",
778 | "etag": "~1.8.1",
779 | "finalhandler": "~1.1.2",
780 | "fresh": "0.5.2",
781 | "merge-descriptors": "1.0.1",
782 | "methods": "~1.1.2",
783 | "on-finished": "~2.3.0",
784 | "parseurl": "~1.3.3",
785 | "path-to-regexp": "0.1.7",
786 | "proxy-addr": "~2.0.5",
787 | "qs": "6.7.0",
788 | "range-parser": "~1.2.1",
789 | "safe-buffer": "5.1.2",
790 | "send": "0.17.1",
791 | "serve-static": "1.14.1",
792 | "setprototypeof": "1.1.1",
793 | "statuses": "~1.5.0",
794 | "type-is": "~1.6.18",
795 | "utils-merge": "1.0.1",
796 | "vary": "~1.1.2"
797 | },
798 | "dependencies": {
799 | "debug": {
800 | "version": "2.6.9",
801 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz",
802 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==",
803 | "requires": {
804 | "ms": "2.0.0"
805 | }
806 | },
807 | "ms": {
808 | "version": "2.0.0",
809 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz",
810 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g="
811 | }
812 | }
813 | },
814 | "express-fileupload": {
815 | "version": "1.2.0",
816 | "resolved": "https://registry.npmjs.org/express-fileupload/-/express-fileupload-1.2.0.tgz",
817 | "integrity": "sha512-oe4WpKcSppXnl5peornawWUa6tKmIc1/kJxMNRGJR1A0v4zyLL6VsFR6wZ8P2a4Iq3aGx8xae3Vlr+MOMQhFPw==",
818 | "requires": {
819 | "busboy": "^0.3.1"
820 | }
821 | },
822 | "fill-range": {
823 | "version": "7.0.1",
824 | "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz",
825 | "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==",
826 | "dev": true,
827 | "requires": {
828 | "to-regex-range": "^5.0.1"
829 | }
830 | },
831 | "finalhandler": {
832 | "version": "1.1.2",
833 | "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.1.2.tgz",
834 | "integrity": "sha512-aAWcW57uxVNrQZqFXjITpW3sIUQmHGG3qSb9mUah9MgMC4NeWhNOlNjXEYq3HjRAvL6arUviZGGJsBg6z0zsWA==",
835 | "requires": {
836 | "debug": "2.6.9",
837 | "encodeurl": "~1.0.2",
838 | "escape-html": "~1.0.3",
839 | "on-finished": "~2.3.0",
840 | "parseurl": "~1.3.3",
841 | "statuses": "~1.5.0",
842 | "unpipe": "~1.0.0"
843 | },
844 | "dependencies": {
845 | "debug": {
846 | "version": "2.6.9",
847 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz",
848 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==",
849 | "requires": {
850 | "ms": "2.0.0"
851 | }
852 | },
853 | "ms": {
854 | "version": "2.0.0",
855 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz",
856 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g="
857 | }
858 | }
859 | },
860 | "find-up": {
861 | "version": "3.0.0",
862 | "resolved": "https://registry.npmjs.org/find-up/-/find-up-3.0.0.tgz",
863 | "integrity": "sha512-1yD6RmLI1XBfxugvORwlck6f75tYL+iR0jqwsOrOxMZyGYqUuDhJ0l4AXdO1iX/FTs9cBAMEk1gWSEx1kSbylg==",
864 | "requires": {
865 | "locate-path": "^3.0.0"
866 | }
867 | },
868 | "forwarded": {
869 | "version": "0.1.2",
870 | "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.1.2.tgz",
871 | "integrity": "sha1-mMI9qxF1ZXuMBXPozszZGw/xjIQ="
872 | },
873 | "fresh": {
874 | "version": "0.5.2",
875 | "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz",
876 | "integrity": "sha1-PYyt2Q2XZWn6g1qx+OSyOhBWBac="
877 | },
878 | "fs-minipass": {
879 | "version": "1.2.7",
880 | "resolved": "https://registry.npmjs.org/fs-minipass/-/fs-minipass-1.2.7.tgz",
881 | "integrity": "sha512-GWSSJGFy4e9GUeCcbIkED+bgAoFyj7XF1mV8rma3QW4NIqX9Kyx79N/PF61H5udOV3aY1IaMLs6pGbH71nlCTA==",
882 | "requires": {
883 | "minipass": "^2.6.0"
884 | }
885 | },
886 | "fs.realpath": {
887 | "version": "1.0.0",
888 | "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz",
889 | "integrity": "sha1-FQStJSMVjKpA20onh8sBQRmU6k8="
890 | },
891 | "fsevents": {
892 | "version": "2.1.3",
893 | "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.1.3.tgz",
894 | "integrity": "sha512-Auw9a4AxqWpa9GUfj370BMPzzyncfBABW8Mab7BGWBYDj4Isgq+cDKtx0i6u9jcX9pQDnswsaaOTgTmA5pEjuQ==",
895 | "dev": true,
896 | "optional": true
897 | },
898 | "gauge": {
899 | "version": "2.7.4",
900 | "resolved": "https://registry.npmjs.org/gauge/-/gauge-2.7.4.tgz",
901 | "integrity": "sha1-LANAXHU4w51+s3sxcCLjJfsBi/c=",
902 | "requires": {
903 | "aproba": "^1.0.3",
904 | "console-control-strings": "^1.0.0",
905 | "has-unicode": "^2.0.0",
906 | "object-assign": "^4.1.0",
907 | "signal-exit": "^3.0.0",
908 | "string-width": "^1.0.1",
909 | "strip-ansi": "^3.0.1",
910 | "wide-align": "^1.1.0"
911 | }
912 | },
913 | "get-caller-file": {
914 | "version": "2.0.5",
915 | "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-2.0.5.tgz",
916 | "integrity": "sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg=="
917 | },
918 | "get-stream": {
919 | "version": "4.1.0",
920 | "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-4.1.0.tgz",
921 | "integrity": "sha512-GMat4EJ5161kIy2HevLlr4luNjBgvmj413KaQA7jt4V8B4RDsfpHk7WQ9GVqfYyyx8OS/L66Kox+rJRNklLK7w==",
922 | "dev": true,
923 | "requires": {
924 | "pump": "^3.0.0"
925 | }
926 | },
927 | "glob": {
928 | "version": "7.1.6",
929 | "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.6.tgz",
930 | "integrity": "sha512-LwaxwyZ72Lk7vZINtNNrywX0ZuLyStrdDtabefZKAY5ZGJhVtgdznluResxNmPitE0SAO+O26sWTHeKSI2wMBA==",
931 | "requires": {
932 | "fs.realpath": "^1.0.0",
933 | "inflight": "^1.0.4",
934 | "inherits": "2",
935 | "minimatch": "^3.0.4",
936 | "once": "^1.3.0",
937 | "path-is-absolute": "^1.0.0"
938 | }
939 | },
940 | "glob-parent": {
941 | "version": "5.1.1",
942 | "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.1.tgz",
943 | "integrity": "sha512-FnI+VGOpnlGHWZxthPGR+QhR78fuiK0sNLkHQv+bL9fQi57lNNdquIbna/WrfROrolq8GK5Ek6BiMwqL/voRYQ==",
944 | "dev": true,
945 | "requires": {
946 | "is-glob": "^4.0.1"
947 | }
948 | },
949 | "global-dirs": {
950 | "version": "2.0.1",
951 | "resolved": "https://registry.npmjs.org/global-dirs/-/global-dirs-2.0.1.tgz",
952 | "integrity": "sha512-5HqUqdhkEovj2Of/ms3IeS/EekcO54ytHRLV4PEY2rhRwrHXLQjeVEES0Lhka0xwNDtGYn58wyC4s5+MHsOO6A==",
953 | "dev": true,
954 | "requires": {
955 | "ini": "^1.3.5"
956 | }
957 | },
958 | "got": {
959 | "version": "9.6.0",
960 | "resolved": "https://registry.npmjs.org/got/-/got-9.6.0.tgz",
961 | "integrity": "sha512-R7eWptXuGYxwijs0eV+v3o6+XH1IqVK8dJOEecQfTmkncw9AV4dcw/Dhxi8MdlqPthxxpZyizMzyg8RTmEsG+Q==",
962 | "dev": true,
963 | "requires": {
964 | "@sindresorhus/is": "^0.14.0",
965 | "@szmarczak/http-timer": "^1.1.2",
966 | "cacheable-request": "^6.0.0",
967 | "decompress-response": "^3.3.0",
968 | "duplexer3": "^0.1.4",
969 | "get-stream": "^4.1.0",
970 | "lowercase-keys": "^1.0.1",
971 | "mimic-response": "^1.0.1",
972 | "p-cancelable": "^1.0.0",
973 | "to-readable-stream": "^1.0.0",
974 | "url-parse-lax": "^3.0.0"
975 | }
976 | },
977 | "graceful-fs": {
978 | "version": "4.2.4",
979 | "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.4.tgz",
980 | "integrity": "sha512-WjKPNJF79dtJAVniUlGGWHYGz2jWxT6VhN/4m1NdkbZ2nOsEF+cI1Edgql5zCRhs/VsQYRvrXctxktVXZUkixw==",
981 | "dev": true
982 | },
983 | "has-flag": {
984 | "version": "3.0.0",
985 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz",
986 | "integrity": "sha1-tdRU3CGZriJWmfNGfloH87lVuv0="
987 | },
988 | "has-unicode": {
989 | "version": "2.0.1",
990 | "resolved": "https://registry.npmjs.org/has-unicode/-/has-unicode-2.0.1.tgz",
991 | "integrity": "sha1-4Ob+aijPUROIVeCG0Wkedx3iqLk="
992 | },
993 | "has-yarn": {
994 | "version": "2.1.0",
995 | "resolved": "https://registry.npmjs.org/has-yarn/-/has-yarn-2.1.0.tgz",
996 | "integrity": "sha512-UqBRqi4ju7T+TqGNdqAO0PaSVGsDGJUBQvk9eUWNGRY1CFGDzYhLWoM7JQEemnlvVcv/YEmc2wNW8BC24EnUsw==",
997 | "dev": true
998 | },
999 | "hosted-git-info": {
1000 | "version": "2.8.8",
1001 | "resolved": "https://registry.npmjs.org/hosted-git-info/-/hosted-git-info-2.8.8.tgz",
1002 | "integrity": "sha512-f/wzC2QaWBs7t9IYqB4T3sR1xviIViXJRJTWBlx2Gf3g0Xi5vI7Yy4koXQ1c9OYDGHN9sBy1DQ2AB8fqZBWhUg=="
1003 | },
1004 | "http-cache-semantics": {
1005 | "version": "4.1.0",
1006 | "resolved": "https://registry.npmjs.org/http-cache-semantics/-/http-cache-semantics-4.1.0.tgz",
1007 | "integrity": "sha512-carPklcUh7ROWRK7Cv27RPtdhYhUsela/ue5/jKzjegVvXDqM2ILE9Q2BGn9JZJh1g87cp56su/FgQSzcWS8cQ==",
1008 | "dev": true
1009 | },
1010 | "http-errors": {
1011 | "version": "1.7.2",
1012 | "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.7.2.tgz",
1013 | "integrity": "sha512-uUQBt3H/cSIVfch6i1EuPNy/YsRSOUBXTVfZ+yR7Zjez3qjBz6i9+i4zjNaoqcoFVI4lQJ5plg63TvGfRSDCRg==",
1014 | "requires": {
1015 | "depd": "~1.1.2",
1016 | "inherits": "2.0.3",
1017 | "setprototypeof": "1.1.1",
1018 | "statuses": ">= 1.5.0 < 2",
1019 | "toidentifier": "1.0.0"
1020 | },
1021 | "dependencies": {
1022 | "inherits": {
1023 | "version": "2.0.3",
1024 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz",
1025 | "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4="
1026 | }
1027 | }
1028 | },
1029 | "iconv-lite": {
1030 | "version": "0.4.24",
1031 | "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz",
1032 | "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==",
1033 | "requires": {
1034 | "safer-buffer": ">= 2.1.2 < 3"
1035 | }
1036 | },
1037 | "ignore-by-default": {
1038 | "version": "1.0.1",
1039 | "resolved": "https://registry.npmjs.org/ignore-by-default/-/ignore-by-default-1.0.1.tgz",
1040 | "integrity": "sha1-SMptcvbGo68Aqa1K5odr44ieKwk=",
1041 | "dev": true
1042 | },
1043 | "ignore-walk": {
1044 | "version": "3.0.3",
1045 | "resolved": "https://registry.npmjs.org/ignore-walk/-/ignore-walk-3.0.3.tgz",
1046 | "integrity": "sha512-m7o6xuOaT1aqheYHKf8W6J5pYH85ZI9w077erOzLje3JsB1gkafkAhHHY19dqjulgIZHFm32Cp5uNZgcQqdJKw==",
1047 | "requires": {
1048 | "minimatch": "^3.0.4"
1049 | }
1050 | },
1051 | "import-lazy": {
1052 | "version": "2.1.0",
1053 | "resolved": "https://registry.npmjs.org/import-lazy/-/import-lazy-2.1.0.tgz",
1054 | "integrity": "sha1-BWmOPUXIjo1+nZLLBYTnfwlvPkM=",
1055 | "dev": true
1056 | },
1057 | "imurmurhash": {
1058 | "version": "0.1.4",
1059 | "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz",
1060 | "integrity": "sha1-khi5srkoojixPcT7a21XbyMUU+o=",
1061 | "dev": true
1062 | },
1063 | "inflight": {
1064 | "version": "1.0.6",
1065 | "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz",
1066 | "integrity": "sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk=",
1067 | "requires": {
1068 | "once": "^1.3.0",
1069 | "wrappy": "1"
1070 | }
1071 | },
1072 | "inherits": {
1073 | "version": "2.0.4",
1074 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz",
1075 | "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ=="
1076 | },
1077 | "ini": {
1078 | "version": "1.3.5",
1079 | "resolved": "https://registry.npmjs.org/ini/-/ini-1.3.5.tgz",
1080 | "integrity": "sha512-RZY5huIKCMRWDUqZlEi72f/lmXKMvuszcMBduliQ3nnWbx9X/ZBQO7DijMEYS9EhHBb2qacRUMtC7svLwe0lcw=="
1081 | },
1082 | "ipaddr.js": {
1083 | "version": "1.9.1",
1084 | "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz",
1085 | "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g=="
1086 | },
1087 | "is-arrayish": {
1088 | "version": "0.2.1",
1089 | "resolved": "https://registry.npmjs.org/is-arrayish/-/is-arrayish-0.2.1.tgz",
1090 | "integrity": "sha1-d8mYQFJ6qOyxqLppe4BkWnqSap0="
1091 | },
1092 | "is-binary-path": {
1093 | "version": "2.1.0",
1094 | "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz",
1095 | "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==",
1096 | "dev": true,
1097 | "requires": {
1098 | "binary-extensions": "^2.0.0"
1099 | }
1100 | },
1101 | "is-ci": {
1102 | "version": "2.0.0",
1103 | "resolved": "https://registry.npmjs.org/is-ci/-/is-ci-2.0.0.tgz",
1104 | "integrity": "sha512-YfJT7rkpQB0updsdHLGWrvhBJfcfzNNawYDNIyQXJz0IViGf75O8EBPKSdvw2rF+LGCsX4FZ8tcr3b19LcZq4w==",
1105 | "dev": true,
1106 | "requires": {
1107 | "ci-info": "^2.0.0"
1108 | }
1109 | },
1110 | "is-extglob": {
1111 | "version": "2.1.1",
1112 | "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz",
1113 | "integrity": "sha1-qIwCU1eR8C7TfHahueqXc8gz+MI=",
1114 | "dev": true
1115 | },
1116 | "is-fullwidth-code-point": {
1117 | "version": "1.0.0",
1118 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-1.0.0.tgz",
1119 | "integrity": "sha1-754xOG8DGn8NZDr4L95QxFfvAMs=",
1120 | "requires": {
1121 | "number-is-nan": "^1.0.0"
1122 | }
1123 | },
1124 | "is-glob": {
1125 | "version": "4.0.1",
1126 | "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.1.tgz",
1127 | "integrity": "sha512-5G0tKtBTFImOqDnLB2hG6Bp2qcKEFduo4tZu9MT/H6NQv/ghhy30o55ufafxJ/LdH79LLs2Kfrn85TLKyA7BUg==",
1128 | "dev": true,
1129 | "requires": {
1130 | "is-extglob": "^2.1.1"
1131 | }
1132 | },
1133 | "is-installed-globally": {
1134 | "version": "0.3.2",
1135 | "resolved": "https://registry.npmjs.org/is-installed-globally/-/is-installed-globally-0.3.2.tgz",
1136 | "integrity": "sha512-wZ8x1js7Ia0kecP/CHM/3ABkAmujX7WPvQk6uu3Fly/Mk44pySulQpnHG46OMjHGXApINnV4QhY3SWnECO2z5g==",
1137 | "dev": true,
1138 | "requires": {
1139 | "global-dirs": "^2.0.1",
1140 | "is-path-inside": "^3.0.1"
1141 | }
1142 | },
1143 | "is-npm": {
1144 | "version": "4.0.0",
1145 | "resolved": "https://registry.npmjs.org/is-npm/-/is-npm-4.0.0.tgz",
1146 | "integrity": "sha512-96ECIfh9xtDDlPylNPXhzjsykHsMJZ18ASpaWzQyBr4YRTcVjUvzaHayDAES2oU/3KpljhHUjtSRNiDwi0F0ig==",
1147 | "dev": true
1148 | },
1149 | "is-number": {
1150 | "version": "7.0.0",
1151 | "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz",
1152 | "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==",
1153 | "dev": true
1154 | },
1155 | "is-obj": {
1156 | "version": "2.0.0",
1157 | "resolved": "https://registry.npmjs.org/is-obj/-/is-obj-2.0.0.tgz",
1158 | "integrity": "sha512-drqDG3cbczxxEJRoOXcOjtdp1J/lyp1mNn0xaznRs8+muBhgQcrnbspox5X5fOw0HnMnbfDzvnEMEtqDEJEo8w==",
1159 | "dev": true
1160 | },
1161 | "is-path-inside": {
1162 | "version": "3.0.2",
1163 | "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-3.0.2.tgz",
1164 | "integrity": "sha512-/2UGPSgmtqwo1ktx8NDHjuPwZWmHhO+gj0f93EkhLB5RgW9RZevWYYlIkS6zePc6U2WpOdQYIwHe9YC4DWEBVg==",
1165 | "dev": true
1166 | },
1167 | "is-typedarray": {
1168 | "version": "1.0.0",
1169 | "resolved": "https://registry.npmjs.org/is-typedarray/-/is-typedarray-1.0.0.tgz",
1170 | "integrity": "sha1-5HnICFjfDBsR3dppQPlgEfzaSpo=",
1171 | "dev": true
1172 | },
1173 | "is-yarn-global": {
1174 | "version": "0.3.0",
1175 | "resolved": "https://registry.npmjs.org/is-yarn-global/-/is-yarn-global-0.3.0.tgz",
1176 | "integrity": "sha512-VjSeb/lHmkoyd8ryPVIKvOCn4D1koMqY+vqyjjUfc3xyKtP4dYOxM44sZrnqQSzSds3xyOrUTLTC9LVCVgLngw==",
1177 | "dev": true
1178 | },
1179 | "isarray": {
1180 | "version": "1.0.0",
1181 | "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz",
1182 | "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE="
1183 | },
1184 | "json-buffer": {
1185 | "version": "3.0.0",
1186 | "resolved": "https://registry.npmjs.org/json-buffer/-/json-buffer-3.0.0.tgz",
1187 | "integrity": "sha1-Wx85evx11ne96Lz8Dkfh+aPZqJg=",
1188 | "dev": true
1189 | },
1190 | "json-parse-better-errors": {
1191 | "version": "1.0.2",
1192 | "resolved": "https://registry.npmjs.org/json-parse-better-errors/-/json-parse-better-errors-1.0.2.tgz",
1193 | "integrity": "sha512-mrqyZKfX5EhL7hvqcV6WG1yYjnjeuYDzDhhcAAUrq8Po85NBQBJP+ZDUT75qZQ98IkUoBqdkExkukOU7Ts2wrw=="
1194 | },
1195 | "jsonwebtoken": {
1196 | "version": "8.5.1",
1197 | "resolved": "https://registry.npmjs.org/jsonwebtoken/-/jsonwebtoken-8.5.1.tgz",
1198 | "integrity": "sha512-XjwVfRS6jTMsqYs0EsuJ4LGxXV14zQybNd4L2r0UvbVnSF9Af8x7p5MzbJ90Ioz/9TI41/hTCvznF/loiSzn8w==",
1199 | "requires": {
1200 | "jws": "^3.2.2",
1201 | "lodash.includes": "^4.3.0",
1202 | "lodash.isboolean": "^3.0.3",
1203 | "lodash.isinteger": "^4.0.4",
1204 | "lodash.isnumber": "^3.0.3",
1205 | "lodash.isplainobject": "^4.0.6",
1206 | "lodash.isstring": "^4.0.1",
1207 | "lodash.once": "^4.0.0",
1208 | "ms": "^2.1.1",
1209 | "semver": "^5.6.0"
1210 | }
1211 | },
1212 | "jwa": {
1213 | "version": "1.4.1",
1214 | "resolved": "https://registry.npmjs.org/jwa/-/jwa-1.4.1.tgz",
1215 | "integrity": "sha512-qiLX/xhEEFKUAJ6FiBMbes3w9ATzyk5W7Hvzpa/SLYdxNtng+gcurvrI7TbACjIXlsJyr05/S1oUhZrc63evQA==",
1216 | "requires": {
1217 | "buffer-equal-constant-time": "1.0.1",
1218 | "ecdsa-sig-formatter": "1.0.11",
1219 | "safe-buffer": "^5.0.1"
1220 | }
1221 | },
1222 | "jws": {
1223 | "version": "3.2.2",
1224 | "resolved": "https://registry.npmjs.org/jws/-/jws-3.2.2.tgz",
1225 | "integrity": "sha512-YHlZCB6lMTllWDtSPHz/ZXTsi8S00usEV6v1tjq8tOUZzw7DpSDWVXjXDre6ed1w/pd495ODpHZYSdkRTsa0HA==",
1226 | "requires": {
1227 | "jwa": "^1.4.1",
1228 | "safe-buffer": "^5.0.1"
1229 | }
1230 | },
1231 | "kareem": {
1232 | "version": "2.3.1",
1233 | "resolved": "https://registry.npmjs.org/kareem/-/kareem-2.3.1.tgz",
1234 | "integrity": "sha512-l3hLhffs9zqoDe8zjmb/mAN4B8VT3L56EUvKNqLFVs9YlFA+zx7ke1DO8STAdDyYNkeSo1nKmjuvQeI12So8Xw=="
1235 | },
1236 | "keyv": {
1237 | "version": "3.1.0",
1238 | "resolved": "https://registry.npmjs.org/keyv/-/keyv-3.1.0.tgz",
1239 | "integrity": "sha512-9ykJ/46SN/9KPM/sichzQ7OvXyGDYKGTaDlKMGCAlg2UK8KRy4jb0d8sFc+0Tt0YYnThq8X2RZgCg74RPxgcVA==",
1240 | "dev": true,
1241 | "requires": {
1242 | "json-buffer": "3.0.0"
1243 | }
1244 | },
1245 | "latest-version": {
1246 | "version": "5.1.0",
1247 | "resolved": "https://registry.npmjs.org/latest-version/-/latest-version-5.1.0.tgz",
1248 | "integrity": "sha512-weT+r0kTkRQdCdYCNtkMwWXQTMEswKrFBkm4ckQOMVhhqhIMI1UT2hMj+1iigIhgSZm5gTmrRXBNoGUgaTY1xA==",
1249 | "dev": true,
1250 | "requires": {
1251 | "package-json": "^6.3.0"
1252 | }
1253 | },
1254 | "locate-path": {
1255 | "version": "3.0.0",
1256 | "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-3.0.0.tgz",
1257 | "integrity": "sha512-7AO748wWnIhNqAuaty2ZWHkQHRSNfPVIsPIfwEOWO22AmaoVrWavlOcMR5nzTLNYvp36X220/maaRsrec1G65A==",
1258 | "requires": {
1259 | "p-locate": "^3.0.0",
1260 | "path-exists": "^3.0.0"
1261 | }
1262 | },
1263 | "lodash": {
1264 | "version": "4.17.20",
1265 | "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.20.tgz",
1266 | "integrity": "sha512-PlhdFcillOINfeV7Ni6oF1TAEayyZBoZ8bcshTHqOYJYlrqzRK5hagpagky5o4HfCzzd1TRkXPMFq6cKk9rGmA=="
1267 | },
1268 | "lodash.includes": {
1269 | "version": "4.3.0",
1270 | "resolved": "https://registry.npmjs.org/lodash.includes/-/lodash.includes-4.3.0.tgz",
1271 | "integrity": "sha1-YLuYqHy5I8aMoeUTJUgzFISfVT8="
1272 | },
1273 | "lodash.isboolean": {
1274 | "version": "3.0.3",
1275 | "resolved": "https://registry.npmjs.org/lodash.isboolean/-/lodash.isboolean-3.0.3.tgz",
1276 | "integrity": "sha1-bC4XHbKiV82WgC/UOwGyDV9YcPY="
1277 | },
1278 | "lodash.isinteger": {
1279 | "version": "4.0.4",
1280 | "resolved": "https://registry.npmjs.org/lodash.isinteger/-/lodash.isinteger-4.0.4.tgz",
1281 | "integrity": "sha1-YZwK89A/iwTDH1iChAt3sRzWg0M="
1282 | },
1283 | "lodash.isnumber": {
1284 | "version": "3.0.3",
1285 | "resolved": "https://registry.npmjs.org/lodash.isnumber/-/lodash.isnumber-3.0.3.tgz",
1286 | "integrity": "sha1-POdoEMWSjQM1IwGsKHMX8RwLH/w="
1287 | },
1288 | "lodash.isplainobject": {
1289 | "version": "4.0.6",
1290 | "resolved": "https://registry.npmjs.org/lodash.isplainobject/-/lodash.isplainobject-4.0.6.tgz",
1291 | "integrity": "sha1-fFJqUtibRcRcxpC4gWO+BJf1UMs="
1292 | },
1293 | "lodash.isstring": {
1294 | "version": "4.0.1",
1295 | "resolved": "https://registry.npmjs.org/lodash.isstring/-/lodash.isstring-4.0.1.tgz",
1296 | "integrity": "sha1-1SfftUVuynzJu5XV2ur4i6VKVFE="
1297 | },
1298 | "lodash.once": {
1299 | "version": "4.1.1",
1300 | "resolved": "https://registry.npmjs.org/lodash.once/-/lodash.once-4.1.1.tgz",
1301 | "integrity": "sha1-DdOXEhPHxW34gJd9UEyI+0cal6w="
1302 | },
1303 | "lowercase-keys": {
1304 | "version": "1.0.1",
1305 | "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-1.0.1.tgz",
1306 | "integrity": "sha512-G2Lj61tXDnVFFOi8VZds+SoQjtQC3dgokKdDG2mTm1tx4m50NUHBOZSBwQQHyy0V12A0JTG4icfZQH+xPyh8VA==",
1307 | "dev": true
1308 | },
1309 | "make-dir": {
1310 | "version": "3.1.0",
1311 | "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-3.1.0.tgz",
1312 | "integrity": "sha512-g3FeP20LNwhALb/6Cz6Dd4F2ngze0jz7tbzrD2wAV+o9FeNHe4rL+yK2md0J/fiSf1sa1ADhXqi5+oVwOM/eGw==",
1313 | "dev": true,
1314 | "requires": {
1315 | "semver": "^6.0.0"
1316 | },
1317 | "dependencies": {
1318 | "semver": {
1319 | "version": "6.3.0",
1320 | "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz",
1321 | "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==",
1322 | "dev": true
1323 | }
1324 | }
1325 | },
1326 | "media-typer": {
1327 | "version": "0.3.0",
1328 | "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz",
1329 | "integrity": "sha1-hxDXrwqmJvj/+hzgAWhUUmMlV0g="
1330 | },
1331 | "memory-pager": {
1332 | "version": "1.5.0",
1333 | "resolved": "https://registry.npmjs.org/memory-pager/-/memory-pager-1.5.0.tgz",
1334 | "integrity": "sha512-ZS4Bp4r/Zoeq6+NLJpP+0Zzm0pR8whtGPf1XExKLJBAczGMnSi3It14OiNCStjQjM6NU1okjQGSxgEZN8eBYKg==",
1335 | "optional": true
1336 | },
1337 | "merge-descriptors": {
1338 | "version": "1.0.1",
1339 | "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz",
1340 | "integrity": "sha1-sAqqVW3YtEVoFQ7J0blT8/kMu2E="
1341 | },
1342 | "methods": {
1343 | "version": "1.1.2",
1344 | "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz",
1345 | "integrity": "sha1-VSmk1nZUE07cxSZmVoNbD4Ua/O4="
1346 | },
1347 | "mime": {
1348 | "version": "1.6.0",
1349 | "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz",
1350 | "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg=="
1351 | },
1352 | "mime-db": {
1353 | "version": "1.44.0",
1354 | "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.44.0.tgz",
1355 | "integrity": "sha512-/NOTfLrsPBVeH7YtFPgsVWveuL+4SjjYxaQ1xtM1KMFj7HdxlBlxeyNLzhyJVx7r4rZGJAZ/6lkKCitSc/Nmpg=="
1356 | },
1357 | "mime-types": {
1358 | "version": "2.1.27",
1359 | "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.27.tgz",
1360 | "integrity": "sha512-JIhqnCasI9yD+SsmkquHBxTSEuZdQX5BuQnS2Vc7puQQQ+8yiP5AY5uWhpdv4YL4VM5c6iliiYWPgJ/nJQLp7w==",
1361 | "requires": {
1362 | "mime-db": "1.44.0"
1363 | }
1364 | },
1365 | "mimic-response": {
1366 | "version": "1.0.1",
1367 | "resolved": "https://registry.npmjs.org/mimic-response/-/mimic-response-1.0.1.tgz",
1368 | "integrity": "sha512-j5EctnkH7amfV/q5Hgmoal1g2QHFJRraOtmx0JpIqkxhBhI/lJSl1nMpQ45hVarwNETOoWEimndZ4QK0RHxuxQ==",
1369 | "dev": true
1370 | },
1371 | "minimatch": {
1372 | "version": "3.0.4",
1373 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz",
1374 | "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==",
1375 | "requires": {
1376 | "brace-expansion": "^1.1.7"
1377 | }
1378 | },
1379 | "minimist": {
1380 | "version": "1.2.5",
1381 | "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.5.tgz",
1382 | "integrity": "sha512-FM9nNUYrRBAELZQT3xeZQ7fmMOBg6nWNmJKTcgsJeaLstP/UODVpGsr5OhXhhXg6f+qtJ8uiZ+PUxkDWcgIXLw=="
1383 | },
1384 | "minipass": {
1385 | "version": "2.9.0",
1386 | "resolved": "https://registry.npmjs.org/minipass/-/minipass-2.9.0.tgz",
1387 | "integrity": "sha512-wxfUjg9WebH+CUDX/CdbRlh5SmfZiy/hpkxaRI16Y9W56Pa75sWgd/rvFilSgrauD9NyFymP/+JFV3KwzIsJeg==",
1388 | "requires": {
1389 | "safe-buffer": "^5.1.2",
1390 | "yallist": "^3.0.0"
1391 | }
1392 | },
1393 | "minizlib": {
1394 | "version": "1.3.3",
1395 | "resolved": "https://registry.npmjs.org/minizlib/-/minizlib-1.3.3.tgz",
1396 | "integrity": "sha512-6ZYMOEnmVsdCeTJVE0W9ZD+pVnE8h9Hma/iOwwRDsdQoePpoX56/8B6z3P9VNwppJuBKNRuFDRNRqRWexT9G9Q==",
1397 | "requires": {
1398 | "minipass": "^2.9.0"
1399 | }
1400 | },
1401 | "mkdirp": {
1402 | "version": "0.5.5",
1403 | "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.5.tgz",
1404 | "integrity": "sha512-NKmAlESf6jMGym1++R0Ra7wvhV+wFW63FaSOFPwRahvea0gMUcGUhVeAg/0BC0wiv9ih5NYPB1Wn1UEI1/L+xQ==",
1405 | "requires": {
1406 | "minimist": "^1.2.5"
1407 | }
1408 | },
1409 | "mongodb": {
1410 | "version": "3.6.2",
1411 | "resolved": "https://registry.npmjs.org/mongodb/-/mongodb-3.6.2.tgz",
1412 | "integrity": "sha512-sSZOb04w3HcnrrXC82NEh/YGCmBuRgR+C1hZgmmv4L6dBz4BkRse6Y8/q/neXer9i95fKUBbFi4KgeceXmbsOA==",
1413 | "requires": {
1414 | "bl": "^2.2.1",
1415 | "bson": "^1.1.4",
1416 | "denque": "^1.4.1",
1417 | "require_optional": "^1.0.1",
1418 | "safe-buffer": "^5.1.2",
1419 | "saslprep": "^1.0.0"
1420 | }
1421 | },
1422 | "mongoose": {
1423 | "version": "5.10.7",
1424 | "resolved": "https://registry.npmjs.org/mongoose/-/mongoose-5.10.7.tgz",
1425 | "integrity": "sha512-oiofFrD4I5p3PhJXn49QyrU1nX5CY01qhPkfMMrXYPhkfGLEJVwFVO+0PsCxD91A2kQP+d/iFyk5U8e86KI8eQ==",
1426 | "requires": {
1427 | "bson": "^1.1.4",
1428 | "kareem": "2.3.1",
1429 | "mongodb": "3.6.2",
1430 | "mongoose-legacy-pluralize": "1.0.2",
1431 | "mpath": "0.7.0",
1432 | "mquery": "3.2.2",
1433 | "ms": "2.1.2",
1434 | "regexp-clone": "1.0.0",
1435 | "safe-buffer": "5.2.1",
1436 | "sift": "7.0.1",
1437 | "sliced": "1.0.1"
1438 | },
1439 | "dependencies": {
1440 | "safe-buffer": {
1441 | "version": "5.2.1",
1442 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz",
1443 | "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ=="
1444 | }
1445 | }
1446 | },
1447 | "mongoose-legacy-pluralize": {
1448 | "version": "1.0.2",
1449 | "resolved": "https://registry.npmjs.org/mongoose-legacy-pluralize/-/mongoose-legacy-pluralize-1.0.2.tgz",
1450 | "integrity": "sha512-Yo/7qQU4/EyIS8YDFSeenIvXxZN+ld7YdV9LqFVQJzTLye8unujAWPZ4NWKfFA+RNjh+wvTWKY9Z3E5XM6ZZiQ=="
1451 | },
1452 | "mpath": {
1453 | "version": "0.7.0",
1454 | "resolved": "https://registry.npmjs.org/mpath/-/mpath-0.7.0.tgz",
1455 | "integrity": "sha512-Aiq04hILxhz1L+f7sjGyn7IxYzWm1zLNNXcfhDtx04kZ2Gk7uvFdgZ8ts1cWa/6d0TQmag2yR8zSGZUmp0tFNg=="
1456 | },
1457 | "mquery": {
1458 | "version": "3.2.2",
1459 | "resolved": "https://registry.npmjs.org/mquery/-/mquery-3.2.2.tgz",
1460 | "integrity": "sha512-XB52992COp0KP230I3qloVUbkLUxJIu328HBP2t2EsxSFtf4W1HPSOBWOXf1bqxK4Xbb66lfMJ+Bpfd9/yZE1Q==",
1461 | "requires": {
1462 | "bluebird": "3.5.1",
1463 | "debug": "3.1.0",
1464 | "regexp-clone": "^1.0.0",
1465 | "safe-buffer": "5.1.2",
1466 | "sliced": "1.0.1"
1467 | },
1468 | "dependencies": {
1469 | "debug": {
1470 | "version": "3.1.0",
1471 | "resolved": "https://registry.npmjs.org/debug/-/debug-3.1.0.tgz",
1472 | "integrity": "sha512-OX8XqP7/1a9cqkxYw2yXss15f26NKWBpDXQd0/uK/KPqdQhxbPa994hnzjcE2VqQpDslf55723cKPUOGSmMY3g==",
1473 | "requires": {
1474 | "ms": "2.0.0"
1475 | }
1476 | },
1477 | "ms": {
1478 | "version": "2.0.0",
1479 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz",
1480 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g="
1481 | }
1482 | }
1483 | },
1484 | "ms": {
1485 | "version": "2.1.2",
1486 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz",
1487 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w=="
1488 | },
1489 | "needle": {
1490 | "version": "2.5.2",
1491 | "resolved": "https://registry.npmjs.org/needle/-/needle-2.5.2.tgz",
1492 | "integrity": "sha512-LbRIwS9BfkPvNwNHlsA41Q29kL2L/6VaOJ0qisM5lLWsTV3nP15abO5ITL6L81zqFhzjRKDAYjpcBcwM0AVvLQ==",
1493 | "requires": {
1494 | "debug": "^3.2.6",
1495 | "iconv-lite": "^0.4.4",
1496 | "sax": "^1.2.4"
1497 | }
1498 | },
1499 | "negotiator": {
1500 | "version": "0.6.2",
1501 | "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.2.tgz",
1502 | "integrity": "sha512-hZXc7K2e+PgeI1eDBe/10Ard4ekbfrrqG8Ep+8Jmf4JID2bNg7NvCPOZN+kfF574pFQI7mum2AUqDidoKqcTOw=="
1503 | },
1504 | "node-addon-api": {
1505 | "version": "3.0.2",
1506 | "resolved": "https://registry.npmjs.org/node-addon-api/-/node-addon-api-3.0.2.tgz",
1507 | "integrity": "sha512-+D4s2HCnxPd5PjjI0STKwncjXTUKKqm74MDMz9OPXavjsGmjkvwgLtA5yoxJUdmpj52+2u+RrXgPipahKczMKg=="
1508 | },
1509 | "node-pre-gyp": {
1510 | "version": "0.15.0",
1511 | "resolved": "https://registry.npmjs.org/node-pre-gyp/-/node-pre-gyp-0.15.0.tgz",
1512 | "integrity": "sha512-7QcZa8/fpaU/BKenjcaeFF9hLz2+7S9AqyXFhlH/rilsQ/hPZKK32RtR5EQHJElgu+q5RfbJ34KriI79UWaorA==",
1513 | "requires": {
1514 | "detect-libc": "^1.0.2",
1515 | "mkdirp": "^0.5.3",
1516 | "needle": "^2.5.0",
1517 | "nopt": "^4.0.1",
1518 | "npm-packlist": "^1.1.6",
1519 | "npmlog": "^4.0.2",
1520 | "rc": "^1.2.7",
1521 | "rimraf": "^2.6.1",
1522 | "semver": "^5.3.0",
1523 | "tar": "^4.4.2"
1524 | }
1525 | },
1526 | "nodemon": {
1527 | "version": "2.0.4",
1528 | "resolved": "https://registry.npmjs.org/nodemon/-/nodemon-2.0.4.tgz",
1529 | "integrity": "sha512-Ltced+hIfTmaS28Zjv1BM552oQ3dbwPqI4+zI0SLgq+wpJhSyqgYude/aZa/3i31VCQWMfXJVxvu86abcam3uQ==",
1530 | "dev": true,
1531 | "requires": {
1532 | "chokidar": "^3.2.2",
1533 | "debug": "^3.2.6",
1534 | "ignore-by-default": "^1.0.1",
1535 | "minimatch": "^3.0.4",
1536 | "pstree.remy": "^1.1.7",
1537 | "semver": "^5.7.1",
1538 | "supports-color": "^5.5.0",
1539 | "touch": "^3.1.0",
1540 | "undefsafe": "^2.0.2",
1541 | "update-notifier": "^4.0.0"
1542 | },
1543 | "dependencies": {
1544 | "supports-color": {
1545 | "version": "5.5.0",
1546 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz",
1547 | "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==",
1548 | "dev": true,
1549 | "requires": {
1550 | "has-flag": "^3.0.0"
1551 | }
1552 | }
1553 | }
1554 | },
1555 | "nopt": {
1556 | "version": "4.0.3",
1557 | "resolved": "https://registry.npmjs.org/nopt/-/nopt-4.0.3.tgz",
1558 | "integrity": "sha512-CvaGwVMztSMJLOeXPrez7fyfObdZqNUK1cPAEzLHrTybIua9pMdmmPR5YwtfNftIOMv3DPUhFaxsZMNTQO20Kg==",
1559 | "requires": {
1560 | "abbrev": "1",
1561 | "osenv": "^0.1.4"
1562 | }
1563 | },
1564 | "normalize-package-data": {
1565 | "version": "2.5.0",
1566 | "resolved": "https://registry.npmjs.org/normalize-package-data/-/normalize-package-data-2.5.0.tgz",
1567 | "integrity": "sha512-/5CMN3T0R4XTj4DcGaexo+roZSdSFW/0AOOTROrjxzCG1wrWXEsGbRKevjlIL+ZDE4sZlJr5ED4YW0yqmkK+eA==",
1568 | "requires": {
1569 | "hosted-git-info": "^2.1.4",
1570 | "resolve": "^1.10.0",
1571 | "semver": "2 || 3 || 4 || 5",
1572 | "validate-npm-package-license": "^3.0.1"
1573 | }
1574 | },
1575 | "normalize-path": {
1576 | "version": "3.0.0",
1577 | "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz",
1578 | "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==",
1579 | "dev": true
1580 | },
1581 | "normalize-url": {
1582 | "version": "4.5.0",
1583 | "resolved": "https://registry.npmjs.org/normalize-url/-/normalize-url-4.5.0.tgz",
1584 | "integrity": "sha512-2s47yzUxdexf1OhyRi4Em83iQk0aPvwTddtFz4hnSSw9dCEsLEGf6SwIO8ss/19S9iBb5sJaOuTvTGDeZI00BQ==",
1585 | "dev": true
1586 | },
1587 | "npm-bundled": {
1588 | "version": "1.1.1",
1589 | "resolved": "https://registry.npmjs.org/npm-bundled/-/npm-bundled-1.1.1.tgz",
1590 | "integrity": "sha512-gqkfgGePhTpAEgUsGEgcq1rqPXA+tv/aVBlgEzfXwA1yiUJF7xtEt3CtVwOjNYQOVknDk0F20w58Fnm3EtG0fA==",
1591 | "requires": {
1592 | "npm-normalize-package-bin": "^1.0.1"
1593 | }
1594 | },
1595 | "npm-normalize-package-bin": {
1596 | "version": "1.0.1",
1597 | "resolved": "https://registry.npmjs.org/npm-normalize-package-bin/-/npm-normalize-package-bin-1.0.1.tgz",
1598 | "integrity": "sha512-EPfafl6JL5/rU+ot6P3gRSCpPDW5VmIzX959Ob1+ySFUuuYHWHekXpwdUZcKP5C+DS4GEtdJluwBjnsNDl+fSA=="
1599 | },
1600 | "npm-packlist": {
1601 | "version": "1.4.8",
1602 | "resolved": "https://registry.npmjs.org/npm-packlist/-/npm-packlist-1.4.8.tgz",
1603 | "integrity": "sha512-5+AZgwru5IevF5ZdnFglB5wNlHG1AOOuw28WhUq8/8emhBmLv6jX5by4WJCh7lW0uSYZYS6DXqIsyZVIXRZU9A==",
1604 | "requires": {
1605 | "ignore-walk": "^3.0.1",
1606 | "npm-bundled": "^1.0.1",
1607 | "npm-normalize-package-bin": "^1.0.1"
1608 | }
1609 | },
1610 | "npmlog": {
1611 | "version": "4.1.2",
1612 | "resolved": "https://registry.npmjs.org/npmlog/-/npmlog-4.1.2.tgz",
1613 | "integrity": "sha512-2uUqazuKlTaSI/dC8AzicUck7+IrEaOnN/e0jd3Xtt1KcGpwx30v50mL7oPyr/h9bL3E4aZccVwpwP+5W9Vjkg==",
1614 | "requires": {
1615 | "are-we-there-yet": "~1.1.2",
1616 | "console-control-strings": "~1.1.0",
1617 | "gauge": "~2.7.3",
1618 | "set-blocking": "~2.0.0"
1619 | }
1620 | },
1621 | "number-is-nan": {
1622 | "version": "1.0.1",
1623 | "resolved": "https://registry.npmjs.org/number-is-nan/-/number-is-nan-1.0.1.tgz",
1624 | "integrity": "sha1-CXtgK1NCKlIsGvuHkDGDNpQaAR0="
1625 | },
1626 | "object-assign": {
1627 | "version": "4.1.1",
1628 | "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz",
1629 | "integrity": "sha1-IQmtx5ZYh8/AXLvUQsrIv7s2CGM="
1630 | },
1631 | "on-finished": {
1632 | "version": "2.3.0",
1633 | "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.3.0.tgz",
1634 | "integrity": "sha1-IPEzZIGwg811M3mSoWlxqi2QaUc=",
1635 | "requires": {
1636 | "ee-first": "1.1.1"
1637 | }
1638 | },
1639 | "once": {
1640 | "version": "1.4.0",
1641 | "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz",
1642 | "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=",
1643 | "requires": {
1644 | "wrappy": "1"
1645 | }
1646 | },
1647 | "os-homedir": {
1648 | "version": "1.0.2",
1649 | "resolved": "https://registry.npmjs.org/os-homedir/-/os-homedir-1.0.2.tgz",
1650 | "integrity": "sha1-/7xJiDNuDoM94MFox+8VISGqf7M="
1651 | },
1652 | "os-tmpdir": {
1653 | "version": "1.0.2",
1654 | "resolved": "https://registry.npmjs.org/os-tmpdir/-/os-tmpdir-1.0.2.tgz",
1655 | "integrity": "sha1-u+Z0BseaqFxc/sdm/lc0VV36EnQ="
1656 | },
1657 | "osenv": {
1658 | "version": "0.1.5",
1659 | "resolved": "https://registry.npmjs.org/osenv/-/osenv-0.1.5.tgz",
1660 | "integrity": "sha512-0CWcCECdMVc2Rw3U5w9ZjqX6ga6ubk1xDVKxtBQPK7wis/0F2r9T6k4ydGYhecl7YUBxBVxhL5oisPsNxAPe2g==",
1661 | "requires": {
1662 | "os-homedir": "^1.0.0",
1663 | "os-tmpdir": "^1.0.0"
1664 | }
1665 | },
1666 | "p-cancelable": {
1667 | "version": "1.1.0",
1668 | "resolved": "https://registry.npmjs.org/p-cancelable/-/p-cancelable-1.1.0.tgz",
1669 | "integrity": "sha512-s73XxOZ4zpt1edZYZzvhqFa6uvQc1vwUa0K0BdtIZgQMAJj9IbebH+JkgKZc9h+B05PKHLOTl4ajG1BmNrVZlw==",
1670 | "dev": true
1671 | },
1672 | "p-limit": {
1673 | "version": "2.3.0",
1674 | "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-2.3.0.tgz",
1675 | "integrity": "sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w==",
1676 | "requires": {
1677 | "p-try": "^2.0.0"
1678 | }
1679 | },
1680 | "p-locate": {
1681 | "version": "3.0.0",
1682 | "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-3.0.0.tgz",
1683 | "integrity": "sha512-x+12w/To+4GFfgJhBEpiDcLozRJGegY+Ei7/z0tSLkMmxGZNybVMSfWj9aJn8Z5Fc7dBUNJOOVgPv2H7IwulSQ==",
1684 | "requires": {
1685 | "p-limit": "^2.0.0"
1686 | }
1687 | },
1688 | "p-try": {
1689 | "version": "2.2.0",
1690 | "resolved": "https://registry.npmjs.org/p-try/-/p-try-2.2.0.tgz",
1691 | "integrity": "sha512-R4nPAVTAU0B9D35/Gk3uJf/7XYbQcyohSKdvAxIRSNghFl4e71hVoGnBNQz9cWaXxO2I10KTC+3jMdvvoKw6dQ=="
1692 | },
1693 | "package-json": {
1694 | "version": "6.5.0",
1695 | "resolved": "https://registry.npmjs.org/package-json/-/package-json-6.5.0.tgz",
1696 | "integrity": "sha512-k3bdm2n25tkyxcjSKzB5x8kfVxlMdgsbPr0GkZcwHsLpba6cBjqCt1KlcChKEvxHIcTB1FVMuwoijZ26xex5MQ==",
1697 | "dev": true,
1698 | "requires": {
1699 | "got": "^9.6.0",
1700 | "registry-auth-token": "^4.0.0",
1701 | "registry-url": "^5.0.0",
1702 | "semver": "^6.2.0"
1703 | },
1704 | "dependencies": {
1705 | "semver": {
1706 | "version": "6.3.0",
1707 | "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz",
1708 | "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==",
1709 | "dev": true
1710 | }
1711 | }
1712 | },
1713 | "parse-json": {
1714 | "version": "4.0.0",
1715 | "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-4.0.0.tgz",
1716 | "integrity": "sha1-vjX1Qlvh9/bHRxhPmKeIy5lHfuA=",
1717 | "requires": {
1718 | "error-ex": "^1.3.1",
1719 | "json-parse-better-errors": "^1.0.1"
1720 | }
1721 | },
1722 | "parseurl": {
1723 | "version": "1.3.3",
1724 | "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz",
1725 | "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ=="
1726 | },
1727 | "path-exists": {
1728 | "version": "3.0.0",
1729 | "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-3.0.0.tgz",
1730 | "integrity": "sha1-zg6+ql94yxiSXqfYENe1mwEP1RU="
1731 | },
1732 | "path-is-absolute": {
1733 | "version": "1.0.1",
1734 | "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz",
1735 | "integrity": "sha1-F0uSaHNVNP+8es5r9TpanhtcX18="
1736 | },
1737 | "path-parse": {
1738 | "version": "1.0.6",
1739 | "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.6.tgz",
1740 | "integrity": "sha512-GSmOT2EbHrINBf9SR7CDELwlJ8AENk3Qn7OikK4nFYAu3Ote2+JYNVvkpAEQm3/TLNEJFD/xZJjzyxg3KBWOzw=="
1741 | },
1742 | "path-to-regexp": {
1743 | "version": "0.1.7",
1744 | "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz",
1745 | "integrity": "sha1-32BBeABfUi8V60SQ5yR6G/qmf4w="
1746 | },
1747 | "picomatch": {
1748 | "version": "2.2.2",
1749 | "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.2.2.tgz",
1750 | "integrity": "sha512-q0M/9eZHzmr0AulXyPwNfZjtwZ/RBZlbN3K3CErVrk50T2ASYI7Bye0EvekFY3IP1Nt2DHu0re+V2ZHIpMkuWg==",
1751 | "dev": true
1752 | },
1753 | "pify": {
1754 | "version": "3.0.0",
1755 | "resolved": "https://registry.npmjs.org/pify/-/pify-3.0.0.tgz",
1756 | "integrity": "sha1-5aSs0sEB/fPZpNB/DbxNtJ3SgXY="
1757 | },
1758 | "prepend-http": {
1759 | "version": "2.0.0",
1760 | "resolved": "https://registry.npmjs.org/prepend-http/-/prepend-http-2.0.0.tgz",
1761 | "integrity": "sha1-6SQ0v6XqjBn0HN/UAddBo8gZ2Jc=",
1762 | "dev": true
1763 | },
1764 | "process-nextick-args": {
1765 | "version": "2.0.1",
1766 | "resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-2.0.1.tgz",
1767 | "integrity": "sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag=="
1768 | },
1769 | "proxy-addr": {
1770 | "version": "2.0.6",
1771 | "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.6.tgz",
1772 | "integrity": "sha512-dh/frvCBVmSsDYzw6n926jv974gddhkFPfiN8hPOi30Wax25QZyZEGveluCgliBnqmuM+UJmBErbAUFIoDbjOw==",
1773 | "requires": {
1774 | "forwarded": "~0.1.2",
1775 | "ipaddr.js": "1.9.1"
1776 | }
1777 | },
1778 | "pstree.remy": {
1779 | "version": "1.1.8",
1780 | "resolved": "https://registry.npmjs.org/pstree.remy/-/pstree.remy-1.1.8.tgz",
1781 | "integrity": "sha512-77DZwxQmxKnu3aR542U+X8FypNzbfJ+C5XQDk3uWjWxn6151aIMGthWYRXTqT1E5oJvg+ljaa2OJi+VfvCOQ8w==",
1782 | "dev": true
1783 | },
1784 | "pump": {
1785 | "version": "3.0.0",
1786 | "resolved": "https://registry.npmjs.org/pump/-/pump-3.0.0.tgz",
1787 | "integrity": "sha512-LwZy+p3SFs1Pytd/jYct4wpv49HiYCqd9Rlc5ZVdk0V+8Yzv6jR5Blk3TRmPL1ft69TxP0IMZGJ+WPFU2BFhww==",
1788 | "dev": true,
1789 | "requires": {
1790 | "end-of-stream": "^1.1.0",
1791 | "once": "^1.3.1"
1792 | }
1793 | },
1794 | "pupa": {
1795 | "version": "2.0.1",
1796 | "resolved": "https://registry.npmjs.org/pupa/-/pupa-2.0.1.tgz",
1797 | "integrity": "sha512-hEJH0s8PXLY/cdXh66tNEQGndDrIKNqNC5xmrysZy3i5C3oEoLna7YAOad+7u125+zH1HNXUmGEkrhb3c2VriA==",
1798 | "dev": true,
1799 | "requires": {
1800 | "escape-goat": "^2.0.0"
1801 | }
1802 | },
1803 | "q": {
1804 | "version": "1.5.1",
1805 | "resolved": "https://registry.npmjs.org/q/-/q-1.5.1.tgz",
1806 | "integrity": "sha1-fjL3W0E4EpHQRhHxvxQQmsAGUdc="
1807 | },
1808 | "qs": {
1809 | "version": "6.7.0",
1810 | "resolved": "https://registry.npmjs.org/qs/-/qs-6.7.0.tgz",
1811 | "integrity": "sha512-VCdBRNFTX1fyE7Nb6FYoURo/SPe62QCaAyzJvUjwRaIsc+NePBEniHlvxFmmX56+HZphIGtV0XeCirBtpDrTyQ=="
1812 | },
1813 | "range-parser": {
1814 | "version": "1.2.1",
1815 | "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz",
1816 | "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg=="
1817 | },
1818 | "raw-body": {
1819 | "version": "2.4.0",
1820 | "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.4.0.tgz",
1821 | "integrity": "sha512-4Oz8DUIwdvoa5qMJelxipzi/iJIi40O5cGV1wNYp5hvZP8ZN0T+jiNkL0QepXs+EsQ9XJ8ipEDoiH70ySUJP3Q==",
1822 | "requires": {
1823 | "bytes": "3.1.0",
1824 | "http-errors": "1.7.2",
1825 | "iconv-lite": "0.4.24",
1826 | "unpipe": "1.0.0"
1827 | }
1828 | },
1829 | "rc": {
1830 | "version": "1.2.8",
1831 | "resolved": "https://registry.npmjs.org/rc/-/rc-1.2.8.tgz",
1832 | "integrity": "sha512-y3bGgqKj3QBdxLbLkomlohkvsA8gdAiUQlSBJnBhfn+BPxg4bc62d8TcBW15wavDfgexCgccckhcZvywyQYPOw==",
1833 | "requires": {
1834 | "deep-extend": "^0.6.0",
1835 | "ini": "~1.3.0",
1836 | "minimist": "^1.2.0",
1837 | "strip-json-comments": "~2.0.1"
1838 | }
1839 | },
1840 | "read-pkg": {
1841 | "version": "4.0.1",
1842 | "resolved": "https://registry.npmjs.org/read-pkg/-/read-pkg-4.0.1.tgz",
1843 | "integrity": "sha1-ljYlN48+HE1IyFhytabsfV0JMjc=",
1844 | "requires": {
1845 | "normalize-package-data": "^2.3.2",
1846 | "parse-json": "^4.0.0",
1847 | "pify": "^3.0.0"
1848 | }
1849 | },
1850 | "readable-stream": {
1851 | "version": "2.3.7",
1852 | "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.7.tgz",
1853 | "integrity": "sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw==",
1854 | "requires": {
1855 | "core-util-is": "~1.0.0",
1856 | "inherits": "~2.0.3",
1857 | "isarray": "~1.0.0",
1858 | "process-nextick-args": "~2.0.0",
1859 | "safe-buffer": "~5.1.1",
1860 | "string_decoder": "~1.1.1",
1861 | "util-deprecate": "~1.0.1"
1862 | }
1863 | },
1864 | "readdirp": {
1865 | "version": "3.4.0",
1866 | "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.4.0.tgz",
1867 | "integrity": "sha512-0xe001vZBnJEK+uKcj8qOhyAKPzIT+gStxWr3LCB0DwcXR5NZJ3IaC+yGnHCYzB/S7ov3m3EEbZI2zeNvX+hGQ==",
1868 | "dev": true,
1869 | "requires": {
1870 | "picomatch": "^2.2.1"
1871 | }
1872 | },
1873 | "regexp-clone": {
1874 | "version": "1.0.0",
1875 | "resolved": "https://registry.npmjs.org/regexp-clone/-/regexp-clone-1.0.0.tgz",
1876 | "integrity": "sha512-TuAasHQNamyyJ2hb97IuBEif4qBHGjPHBS64sZwytpLEqtBQ1gPJTnOaQ6qmpET16cK14kkjbazl6+p0RRv0yw=="
1877 | },
1878 | "registry-auth-token": {
1879 | "version": "4.2.0",
1880 | "resolved": "https://registry.npmjs.org/registry-auth-token/-/registry-auth-token-4.2.0.tgz",
1881 | "integrity": "sha512-P+lWzPrsgfN+UEpDS3U8AQKg/UjZX6mQSJueZj3EK+vNESoqBSpBUD3gmu4sF9lOsjXWjF11dQKUqemf3veq1w==",
1882 | "dev": true,
1883 | "requires": {
1884 | "rc": "^1.2.8"
1885 | }
1886 | },
1887 | "registry-url": {
1888 | "version": "5.1.0",
1889 | "resolved": "https://registry.npmjs.org/registry-url/-/registry-url-5.1.0.tgz",
1890 | "integrity": "sha512-8acYXXTI0AkQv6RAOjE3vOaIXZkT9wo4LOFbBKYQEEnnMNBpKqdUrI6S4NT0KPIo/WVvJ5tE/X5LF/TQUf0ekw==",
1891 | "dev": true,
1892 | "requires": {
1893 | "rc": "^1.2.8"
1894 | }
1895 | },
1896 | "require-directory": {
1897 | "version": "2.1.1",
1898 | "resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz",
1899 | "integrity": "sha1-jGStX9MNqxyXbiNE/+f3kqam30I="
1900 | },
1901 | "require-main-filename": {
1902 | "version": "2.0.0",
1903 | "resolved": "https://registry.npmjs.org/require-main-filename/-/require-main-filename-2.0.0.tgz",
1904 | "integrity": "sha512-NKN5kMDylKuldxYLSUfrbo5Tuzh4hd+2E8NPPX02mZtn1VuREQToYe/ZdlJy+J3uCpfaiGF05e7B8W0iXbQHmg=="
1905 | },
1906 | "require_optional": {
1907 | "version": "1.0.1",
1908 | "resolved": "https://registry.npmjs.org/require_optional/-/require_optional-1.0.1.tgz",
1909 | "integrity": "sha512-qhM/y57enGWHAe3v/NcwML6a3/vfESLe/sGM2dII+gEO0BpKRUkWZow/tyloNqJyN6kXSl3RyyM8Ll5D/sJP8g==",
1910 | "requires": {
1911 | "resolve-from": "^2.0.0",
1912 | "semver": "^5.1.0"
1913 | }
1914 | },
1915 | "resolve": {
1916 | "version": "1.17.0",
1917 | "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.17.0.tgz",
1918 | "integrity": "sha512-ic+7JYiV8Vi2yzQGFWOkiZD5Z9z7O2Zhm9XMaTxdJExKasieFCr+yXZ/WmXsckHiKl12ar0y6XiXDx3m4RHn1w==",
1919 | "requires": {
1920 | "path-parse": "^1.0.6"
1921 | }
1922 | },
1923 | "resolve-from": {
1924 | "version": "2.0.0",
1925 | "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-2.0.0.tgz",
1926 | "integrity": "sha1-lICrIOlP+h2egKgEx+oUdhGWa1c="
1927 | },
1928 | "responselike": {
1929 | "version": "1.0.2",
1930 | "resolved": "https://registry.npmjs.org/responselike/-/responselike-1.0.2.tgz",
1931 | "integrity": "sha1-kYcg7ztjHFZCvgaPFa3lpG9Loec=",
1932 | "dev": true,
1933 | "requires": {
1934 | "lowercase-keys": "^1.0.0"
1935 | }
1936 | },
1937 | "rimraf": {
1938 | "version": "2.7.1",
1939 | "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz",
1940 | "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==",
1941 | "requires": {
1942 | "glob": "^7.1.3"
1943 | }
1944 | },
1945 | "rxjs": {
1946 | "version": "6.6.3",
1947 | "resolved": "https://registry.npmjs.org/rxjs/-/rxjs-6.6.3.tgz",
1948 | "integrity": "sha512-trsQc+xYYXZ3urjOiJOuCOa5N3jAZ3eiSpQB5hIT8zGlL2QfnHLJ2r7GMkBGuIausdJN1OneaI6gQlsqNHHmZQ==",
1949 | "requires": {
1950 | "tslib": "^1.9.0"
1951 | }
1952 | },
1953 | "safe-buffer": {
1954 | "version": "5.1.2",
1955 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz",
1956 | "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g=="
1957 | },
1958 | "safer-buffer": {
1959 | "version": "2.1.2",
1960 | "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz",
1961 | "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg=="
1962 | },
1963 | "saslprep": {
1964 | "version": "1.0.3",
1965 | "resolved": "https://registry.npmjs.org/saslprep/-/saslprep-1.0.3.tgz",
1966 | "integrity": "sha512-/MY/PEMbk2SuY5sScONwhUDsV2p77Znkb/q3nSVstq/yQzYJOH/Azh29p9oJLsl3LnQwSvZDKagDGBsBwSooag==",
1967 | "optional": true,
1968 | "requires": {
1969 | "sparse-bitfield": "^3.0.3"
1970 | }
1971 | },
1972 | "sax": {
1973 | "version": "1.2.4",
1974 | "resolved": "https://registry.npmjs.org/sax/-/sax-1.2.4.tgz",
1975 | "integrity": "sha512-NqVDv9TpANUjFm0N8uM5GxL36UgKi9/atZw+x7YFnQ8ckwFGKrl4xX4yWtrey3UJm5nP1kUbnYgLopqWNSRhWw=="
1976 | },
1977 | "semver": {
1978 | "version": "5.7.1",
1979 | "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz",
1980 | "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ=="
1981 | },
1982 | "semver-diff": {
1983 | "version": "3.1.1",
1984 | "resolved": "https://registry.npmjs.org/semver-diff/-/semver-diff-3.1.1.tgz",
1985 | "integrity": "sha512-GX0Ix/CJcHyB8c4ykpHGIAvLyOwOobtM/8d+TQkAd81/bEjgPHrfba41Vpesr7jX/t8Uh+R3EX9eAS5be+jQYg==",
1986 | "dev": true,
1987 | "requires": {
1988 | "semver": "^6.3.0"
1989 | },
1990 | "dependencies": {
1991 | "semver": {
1992 | "version": "6.3.0",
1993 | "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz",
1994 | "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==",
1995 | "dev": true
1996 | }
1997 | }
1998 | },
1999 | "send": {
2000 | "version": "0.17.1",
2001 | "resolved": "https://registry.npmjs.org/send/-/send-0.17.1.tgz",
2002 | "integrity": "sha512-BsVKsiGcQMFwT8UxypobUKyv7irCNRHk1T0G680vk88yf6LBByGcZJOTJCrTP2xVN6yI+XjPJcNuE3V4fT9sAg==",
2003 | "requires": {
2004 | "debug": "2.6.9",
2005 | "depd": "~1.1.2",
2006 | "destroy": "~1.0.4",
2007 | "encodeurl": "~1.0.2",
2008 | "escape-html": "~1.0.3",
2009 | "etag": "~1.8.1",
2010 | "fresh": "0.5.2",
2011 | "http-errors": "~1.7.2",
2012 | "mime": "1.6.0",
2013 | "ms": "2.1.1",
2014 | "on-finished": "~2.3.0",
2015 | "range-parser": "~1.2.1",
2016 | "statuses": "~1.5.0"
2017 | },
2018 | "dependencies": {
2019 | "debug": {
2020 | "version": "2.6.9",
2021 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz",
2022 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==",
2023 | "requires": {
2024 | "ms": "2.0.0"
2025 | },
2026 | "dependencies": {
2027 | "ms": {
2028 | "version": "2.0.0",
2029 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz",
2030 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g="
2031 | }
2032 | }
2033 | },
2034 | "ms": {
2035 | "version": "2.1.1",
2036 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.1.tgz",
2037 | "integrity": "sha512-tgp+dl5cGk28utYktBsrFqA7HKgrhgPsg6Z/EfhWI4gl1Hwq8B/GmY/0oXZ6nF8hDVesS/FpnYaD/kOWhYQvyg=="
2038 | }
2039 | }
2040 | },
2041 | "serve-static": {
2042 | "version": "1.14.1",
2043 | "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.14.1.tgz",
2044 | "integrity": "sha512-JMrvUwE54emCYWlTI+hGrGv5I8dEwmco/00EvkzIIsR7MqrHonbD9pO2MOfFnpFntl7ecpZs+3mW+XbQZu9QCg==",
2045 | "requires": {
2046 | "encodeurl": "~1.0.2",
2047 | "escape-html": "~1.0.3",
2048 | "parseurl": "~1.3.3",
2049 | "send": "0.17.1"
2050 | }
2051 | },
2052 | "set-blocking": {
2053 | "version": "2.0.0",
2054 | "resolved": "https://registry.npmjs.org/set-blocking/-/set-blocking-2.0.0.tgz",
2055 | "integrity": "sha1-BF+XgtARrppoA93TgrJDkrPYkPc="
2056 | },
2057 | "setprototypeof": {
2058 | "version": "1.1.1",
2059 | "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.1.tgz",
2060 | "integrity": "sha512-JvdAWfbXeIGaZ9cILp38HntZSFSo3mWg6xGcJJsd+d4aRMOqauag1C63dJfDw7OaMYwEbHMOxEZ1lqVRYP2OAw=="
2061 | },
2062 | "sift": {
2063 | "version": "7.0.1",
2064 | "resolved": "https://registry.npmjs.org/sift/-/sift-7.0.1.tgz",
2065 | "integrity": "sha512-oqD7PMJ+uO6jV9EQCl0LrRw1OwsiPsiFQR5AR30heR+4Dl7jBBbDLnNvWiak20tzZlSE1H7RB30SX/1j/YYT7g=="
2066 | },
2067 | "signal-exit": {
2068 | "version": "3.0.3",
2069 | "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.3.tgz",
2070 | "integrity": "sha512-VUJ49FC8U1OxwZLxIbTTrDvLnf/6TDgxZcK8wxR8zs13xpx7xbG60ndBlhNrFi2EMuFRoeDoJO7wthSLq42EjA=="
2071 | },
2072 | "sliced": {
2073 | "version": "1.0.1",
2074 | "resolved": "https://registry.npmjs.org/sliced/-/sliced-1.0.1.tgz",
2075 | "integrity": "sha1-CzpmK10Ewxd7GSa+qCsD+Dei70E="
2076 | },
2077 | "sparse-bitfield": {
2078 | "version": "3.0.3",
2079 | "resolved": "https://registry.npmjs.org/sparse-bitfield/-/sparse-bitfield-3.0.3.tgz",
2080 | "integrity": "sha1-/0rm5oZWBWuks+eSqzM004JzyhE=",
2081 | "optional": true,
2082 | "requires": {
2083 | "memory-pager": "^1.0.2"
2084 | }
2085 | },
2086 | "spawn-command": {
2087 | "version": "0.0.2-1",
2088 | "resolved": "https://registry.npmjs.org/spawn-command/-/spawn-command-0.0.2-1.tgz",
2089 | "integrity": "sha1-YvXpRmmBwbeW3Fkpk34RycaSG9A="
2090 | },
2091 | "spdx-correct": {
2092 | "version": "3.1.1",
2093 | "resolved": "https://registry.npmjs.org/spdx-correct/-/spdx-correct-3.1.1.tgz",
2094 | "integrity": "sha512-cOYcUWwhCuHCXi49RhFRCyJEK3iPj1Ziz9DpViV3tbZOwXD49QzIN3MpOLJNxh2qwq2lJJZaKMVw9qNi4jTC0w==",
2095 | "requires": {
2096 | "spdx-expression-parse": "^3.0.0",
2097 | "spdx-license-ids": "^3.0.0"
2098 | }
2099 | },
2100 | "spdx-exceptions": {
2101 | "version": "2.3.0",
2102 | "resolved": "https://registry.npmjs.org/spdx-exceptions/-/spdx-exceptions-2.3.0.tgz",
2103 | "integrity": "sha512-/tTrYOC7PPI1nUAgx34hUpqXuyJG+DTHJTnIULG4rDygi4xu/tfgmq1e1cIRwRzwZgo4NLySi+ricLkZkw4i5A=="
2104 | },
2105 | "spdx-expression-parse": {
2106 | "version": "3.0.1",
2107 | "resolved": "https://registry.npmjs.org/spdx-expression-parse/-/spdx-expression-parse-3.0.1.tgz",
2108 | "integrity": "sha512-cbqHunsQWnJNE6KhVSMsMeH5H/L9EpymbzqTQ3uLwNCLZ1Q481oWaofqH7nO6V07xlXwY6PhQdQ2IedWx/ZK4Q==",
2109 | "requires": {
2110 | "spdx-exceptions": "^2.1.0",
2111 | "spdx-license-ids": "^3.0.0"
2112 | }
2113 | },
2114 | "spdx-license-ids": {
2115 | "version": "3.0.6",
2116 | "resolved": "https://registry.npmjs.org/spdx-license-ids/-/spdx-license-ids-3.0.6.tgz",
2117 | "integrity": "sha512-+orQK83kyMva3WyPf59k1+Y525csj5JejicWut55zeTWANuN17qSiSLUXWtzHeNWORSvT7GLDJ/E/XiIWoXBTw=="
2118 | },
2119 | "statuses": {
2120 | "version": "1.5.0",
2121 | "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.5.0.tgz",
2122 | "integrity": "sha1-Fhx9rBd2Wf2YEfQ3cfqZOBR4Yow="
2123 | },
2124 | "streamsearch": {
2125 | "version": "0.1.2",
2126 | "resolved": "https://registry.npmjs.org/streamsearch/-/streamsearch-0.1.2.tgz",
2127 | "integrity": "sha1-gIudDlb8Jz2Am6VzOOkpkZoanxo="
2128 | },
2129 | "string-width": {
2130 | "version": "1.0.2",
2131 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-1.0.2.tgz",
2132 | "integrity": "sha1-EYvfW4zcUaKn5w0hHgfisLmxB9M=",
2133 | "requires": {
2134 | "code-point-at": "^1.0.0",
2135 | "is-fullwidth-code-point": "^1.0.0",
2136 | "strip-ansi": "^3.0.0"
2137 | }
2138 | },
2139 | "string_decoder": {
2140 | "version": "1.1.1",
2141 | "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz",
2142 | "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==",
2143 | "requires": {
2144 | "safe-buffer": "~5.1.0"
2145 | }
2146 | },
2147 | "strip-ansi": {
2148 | "version": "3.0.1",
2149 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz",
2150 | "integrity": "sha1-ajhfuIU9lS1f8F0Oiq+UJ43GPc8=",
2151 | "requires": {
2152 | "ansi-regex": "^2.0.0"
2153 | }
2154 | },
2155 | "strip-json-comments": {
2156 | "version": "2.0.1",
2157 | "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-2.0.1.tgz",
2158 | "integrity": "sha1-PFMZQukIwml8DsNEhYwobHygpgo="
2159 | },
2160 | "supports-color": {
2161 | "version": "6.1.0",
2162 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-6.1.0.tgz",
2163 | "integrity": "sha512-qe1jfm1Mg7Nq/NSh6XE24gPXROEVsWHxC1LIx//XNlD9iw7YZQGjZNjYN7xGaEG6iKdA8EtNFW6R0gjnVXp+wQ==",
2164 | "requires": {
2165 | "has-flag": "^3.0.0"
2166 | }
2167 | },
2168 | "tar": {
2169 | "version": "4.4.13",
2170 | "resolved": "https://registry.npmjs.org/tar/-/tar-4.4.13.tgz",
2171 | "integrity": "sha512-w2VwSrBoHa5BsSyH+KxEqeQBAllHhccyMFVHtGtdMpF4W7IRWfZjFiQceJPChOeTsSDVUpER2T8FA93pr0L+QA==",
2172 | "requires": {
2173 | "chownr": "^1.1.1",
2174 | "fs-minipass": "^1.2.5",
2175 | "minipass": "^2.8.6",
2176 | "minizlib": "^1.2.1",
2177 | "mkdirp": "^0.5.0",
2178 | "safe-buffer": "^5.1.2",
2179 | "yallist": "^3.0.3"
2180 | }
2181 | },
2182 | "term-size": {
2183 | "version": "2.2.0",
2184 | "resolved": "https://registry.npmjs.org/term-size/-/term-size-2.2.0.tgz",
2185 | "integrity": "sha512-a6sumDlzyHVJWb8+YofY4TW112G6p2FCPEAFk+59gIYHv3XHRhm9ltVQ9kli4hNWeQBwSpe8cRN25x0ROunMOw==",
2186 | "dev": true
2187 | },
2188 | "to-readable-stream": {
2189 | "version": "1.0.0",
2190 | "resolved": "https://registry.npmjs.org/to-readable-stream/-/to-readable-stream-1.0.0.tgz",
2191 | "integrity": "sha512-Iq25XBt6zD5npPhlLVXGFN3/gyR2/qODcKNNyTMd4vbm39HUaOiAM4PMq0eMVC/Tkxz+Zjdsc55g9yyz+Yq00Q==",
2192 | "dev": true
2193 | },
2194 | "to-regex-range": {
2195 | "version": "5.0.1",
2196 | "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz",
2197 | "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==",
2198 | "dev": true,
2199 | "requires": {
2200 | "is-number": "^7.0.0"
2201 | }
2202 | },
2203 | "toidentifier": {
2204 | "version": "1.0.0",
2205 | "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.0.tgz",
2206 | "integrity": "sha512-yaOH/Pk/VEhBWWTlhI+qXxDFXlejDGcQipMlyxda9nthulaxLZUNcUqFxokp0vcYnvteJln5FNQDRrxj3YcbVw=="
2207 | },
2208 | "touch": {
2209 | "version": "3.1.0",
2210 | "resolved": "https://registry.npmjs.org/touch/-/touch-3.1.0.tgz",
2211 | "integrity": "sha512-WBx8Uy5TLtOSRtIq+M03/sKDrXCLHxwDcquSP2c43Le03/9serjQBIztjRz6FkJez9D/hleyAXTBGLwwZUw9lA==",
2212 | "dev": true,
2213 | "requires": {
2214 | "nopt": "~1.0.10"
2215 | },
2216 | "dependencies": {
2217 | "nopt": {
2218 | "version": "1.0.10",
2219 | "resolved": "https://registry.npmjs.org/nopt/-/nopt-1.0.10.tgz",
2220 | "integrity": "sha1-bd0hvSoxQXuScn3Vhfim83YI6+4=",
2221 | "dev": true,
2222 | "requires": {
2223 | "abbrev": "1"
2224 | }
2225 | }
2226 | }
2227 | },
2228 | "tree-kill": {
2229 | "version": "1.2.2",
2230 | "resolved": "https://registry.npmjs.org/tree-kill/-/tree-kill-1.2.2.tgz",
2231 | "integrity": "sha512-L0Orpi8qGpRG//Nd+H90vFB+3iHnue1zSSGmNOOCh1GLJ7rUKVwV2HvijphGQS2UmhUZewS9VgvxYIdgr+fG1A=="
2232 | },
2233 | "tslib": {
2234 | "version": "1.13.0",
2235 | "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.13.0.tgz",
2236 | "integrity": "sha512-i/6DQjL8Xf3be4K/E6Wgpekn5Qasl1usyw++dAA35Ue5orEn65VIxOA+YvNNl9HV3qv70T7CNwjODHZrLwvd1Q=="
2237 | },
2238 | "type-fest": {
2239 | "version": "0.8.1",
2240 | "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.8.1.tgz",
2241 | "integrity": "sha512-4dbzIzqvjtgiM5rw1k5rEHtBANKmdudhGyBEajN01fEyhaAIhsoKNy6y7+IN93IfpFtwY9iqi7kD+xwKhQsNJA==",
2242 | "dev": true
2243 | },
2244 | "type-is": {
2245 | "version": "1.6.18",
2246 | "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz",
2247 | "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==",
2248 | "requires": {
2249 | "media-typer": "0.3.0",
2250 | "mime-types": "~2.1.24"
2251 | }
2252 | },
2253 | "typedarray-to-buffer": {
2254 | "version": "3.1.5",
2255 | "resolved": "https://registry.npmjs.org/typedarray-to-buffer/-/typedarray-to-buffer-3.1.5.tgz",
2256 | "integrity": "sha512-zdu8XMNEDepKKR+XYOXAVPtWui0ly0NtohUscw+UmaHiAWT8hrV1rr//H6V+0DvJ3OQ19S979M0laLfX8rm82Q==",
2257 | "dev": true,
2258 | "requires": {
2259 | "is-typedarray": "^1.0.0"
2260 | }
2261 | },
2262 | "undefsafe": {
2263 | "version": "2.0.3",
2264 | "resolved": "https://registry.npmjs.org/undefsafe/-/undefsafe-2.0.3.tgz",
2265 | "integrity": "sha512-nrXZwwXrD/T/JXeygJqdCO6NZZ1L66HrxM/Z7mIq2oPanoN0F1nLx3lwJMu6AwJY69hdixaFQOuoYsMjE5/C2A==",
2266 | "dev": true,
2267 | "requires": {
2268 | "debug": "^2.2.0"
2269 | },
2270 | "dependencies": {
2271 | "debug": {
2272 | "version": "2.6.9",
2273 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz",
2274 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==",
2275 | "dev": true,
2276 | "requires": {
2277 | "ms": "2.0.0"
2278 | }
2279 | },
2280 | "ms": {
2281 | "version": "2.0.0",
2282 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz",
2283 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=",
2284 | "dev": true
2285 | }
2286 | }
2287 | },
2288 | "unique-string": {
2289 | "version": "2.0.0",
2290 | "resolved": "https://registry.npmjs.org/unique-string/-/unique-string-2.0.0.tgz",
2291 | "integrity": "sha512-uNaeirEPvpZWSgzwsPGtU2zVSTrn/8L5q/IexZmH0eH6SA73CmAA5U4GwORTxQAZs95TAXLNqeLoPPNO5gZfWg==",
2292 | "dev": true,
2293 | "requires": {
2294 | "crypto-random-string": "^2.0.0"
2295 | }
2296 | },
2297 | "unpipe": {
2298 | "version": "1.0.0",
2299 | "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz",
2300 | "integrity": "sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw="
2301 | },
2302 | "update-notifier": {
2303 | "version": "4.1.3",
2304 | "resolved": "https://registry.npmjs.org/update-notifier/-/update-notifier-4.1.3.tgz",
2305 | "integrity": "sha512-Yld6Z0RyCYGB6ckIjffGOSOmHXj1gMeE7aROz4MG+XMkmixBX4jUngrGXNYz7wPKBmtoD4MnBa2Anu7RSKht/A==",
2306 | "dev": true,
2307 | "requires": {
2308 | "boxen": "^4.2.0",
2309 | "chalk": "^3.0.0",
2310 | "configstore": "^5.0.1",
2311 | "has-yarn": "^2.1.0",
2312 | "import-lazy": "^2.1.0",
2313 | "is-ci": "^2.0.0",
2314 | "is-installed-globally": "^0.3.1",
2315 | "is-npm": "^4.0.0",
2316 | "is-yarn-global": "^0.3.0",
2317 | "latest-version": "^5.0.0",
2318 | "pupa": "^2.0.1",
2319 | "semver-diff": "^3.1.1",
2320 | "xdg-basedir": "^4.0.0"
2321 | },
2322 | "dependencies": {
2323 | "ansi-styles": {
2324 | "version": "4.2.1",
2325 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.2.1.tgz",
2326 | "integrity": "sha512-9VGjrMsG1vePxcSweQsN20KY/c4zN0h9fLjqAbwbPfahM3t+NL+M9HC8xeXG2I8pX5NoamTGNuomEUFI7fcUjA==",
2327 | "dev": true,
2328 | "requires": {
2329 | "@types/color-name": "^1.1.1",
2330 | "color-convert": "^2.0.1"
2331 | }
2332 | },
2333 | "chalk": {
2334 | "version": "3.0.0",
2335 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-3.0.0.tgz",
2336 | "integrity": "sha512-4D3B6Wf41KOYRFdszmDqMCGq5VV/uMAB273JILmO+3jAlh8X4qDtdtgCR3fxtbLEMzSx22QdhnDcJvu2u1fVwg==",
2337 | "dev": true,
2338 | "requires": {
2339 | "ansi-styles": "^4.1.0",
2340 | "supports-color": "^7.1.0"
2341 | }
2342 | },
2343 | "color-convert": {
2344 | "version": "2.0.1",
2345 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz",
2346 | "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==",
2347 | "dev": true,
2348 | "requires": {
2349 | "color-name": "~1.1.4"
2350 | }
2351 | },
2352 | "color-name": {
2353 | "version": "1.1.4",
2354 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz",
2355 | "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==",
2356 | "dev": true
2357 | },
2358 | "has-flag": {
2359 | "version": "4.0.0",
2360 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz",
2361 | "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==",
2362 | "dev": true
2363 | },
2364 | "supports-color": {
2365 | "version": "7.2.0",
2366 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz",
2367 | "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==",
2368 | "dev": true,
2369 | "requires": {
2370 | "has-flag": "^4.0.0"
2371 | }
2372 | }
2373 | }
2374 | },
2375 | "url-parse-lax": {
2376 | "version": "3.0.0",
2377 | "resolved": "https://registry.npmjs.org/url-parse-lax/-/url-parse-lax-3.0.0.tgz",
2378 | "integrity": "sha1-FrXK/Afb42dsGxmZF3gj1lA6yww=",
2379 | "dev": true,
2380 | "requires": {
2381 | "prepend-http": "^2.0.0"
2382 | }
2383 | },
2384 | "util-deprecate": {
2385 | "version": "1.0.2",
2386 | "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz",
2387 | "integrity": "sha1-RQ1Nyfpw3nMnYvvS1KKJgUGaDM8="
2388 | },
2389 | "utils-merge": {
2390 | "version": "1.0.1",
2391 | "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz",
2392 | "integrity": "sha1-n5VxD1CiZ5R7LMwSR0HBAoQn5xM="
2393 | },
2394 | "validate-npm-package-license": {
2395 | "version": "3.0.4",
2396 | "resolved": "https://registry.npmjs.org/validate-npm-package-license/-/validate-npm-package-license-3.0.4.tgz",
2397 | "integrity": "sha512-DpKm2Ui/xN7/HQKCtpZxoRWBhZ9Z0kqtygG8XCgNQ8ZlDnxuQmWhj566j8fN4Cu3/JmbhsDo7fcAJq4s9h27Ew==",
2398 | "requires": {
2399 | "spdx-correct": "^3.0.0",
2400 | "spdx-expression-parse": "^3.0.0"
2401 | }
2402 | },
2403 | "vary": {
2404 | "version": "1.1.2",
2405 | "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz",
2406 | "integrity": "sha1-IpnwLG3tMNSllhsLn3RSShj2NPw="
2407 | },
2408 | "which-module": {
2409 | "version": "2.0.0",
2410 | "resolved": "https://registry.npmjs.org/which-module/-/which-module-2.0.0.tgz",
2411 | "integrity": "sha1-2e8H3Od7mQK4o6j6SzHD4/fm6Ho="
2412 | },
2413 | "wide-align": {
2414 | "version": "1.1.3",
2415 | "resolved": "https://registry.npmjs.org/wide-align/-/wide-align-1.1.3.tgz",
2416 | "integrity": "sha512-QGkOQc8XL6Bt5PwnsExKBPuMKBxnGxWWW3fU55Xt4feHozMUhdUMaBCk290qpm/wG5u/RSKzwdAC4i51YigihA==",
2417 | "requires": {
2418 | "string-width": "^1.0.2 || 2"
2419 | }
2420 | },
2421 | "widest-line": {
2422 | "version": "3.1.0",
2423 | "resolved": "https://registry.npmjs.org/widest-line/-/widest-line-3.1.0.tgz",
2424 | "integrity": "sha512-NsmoXalsWVDMGupxZ5R08ka9flZjjiLvHVAWYOKtiKM8ujtZWr9cRffak+uSE48+Ob8ObalXpwyeUiyDD6QFgg==",
2425 | "dev": true,
2426 | "requires": {
2427 | "string-width": "^4.0.0"
2428 | },
2429 | "dependencies": {
2430 | "ansi-regex": {
2431 | "version": "5.0.0",
2432 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.0.tgz",
2433 | "integrity": "sha512-bY6fj56OUQ0hU1KjFNDQuJFezqKdrAyFdIevADiqrWHwSlbmBNMHp5ak2f40Pm8JTFyM2mqxkG6ngkHO11f/lg==",
2434 | "dev": true
2435 | },
2436 | "emoji-regex": {
2437 | "version": "8.0.0",
2438 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz",
2439 | "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==",
2440 | "dev": true
2441 | },
2442 | "is-fullwidth-code-point": {
2443 | "version": "3.0.0",
2444 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz",
2445 | "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==",
2446 | "dev": true
2447 | },
2448 | "string-width": {
2449 | "version": "4.2.0",
2450 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.0.tgz",
2451 | "integrity": "sha512-zUz5JD+tgqtuDjMhwIg5uFVV3dtqZ9yQJlZVfq4I01/K5Paj5UHj7VyrQOJvzawSVlKpObApbfD0Ed6yJc+1eg==",
2452 | "dev": true,
2453 | "requires": {
2454 | "emoji-regex": "^8.0.0",
2455 | "is-fullwidth-code-point": "^3.0.0",
2456 | "strip-ansi": "^6.0.0"
2457 | }
2458 | },
2459 | "strip-ansi": {
2460 | "version": "6.0.0",
2461 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.0.tgz",
2462 | "integrity": "sha512-AuvKTrTfQNYNIctbR1K/YGTR1756GycPsg7b9bdV9Duqur4gv6aKqHXah67Z8ImS7WEz5QVcOtlfW2rZEugt6w==",
2463 | "dev": true,
2464 | "requires": {
2465 | "ansi-regex": "^5.0.0"
2466 | }
2467 | }
2468 | }
2469 | },
2470 | "wrap-ansi": {
2471 | "version": "5.1.0",
2472 | "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-5.1.0.tgz",
2473 | "integrity": "sha512-QC1/iN/2/RPVJ5jYK8BGttj5z83LmSKmvbvrXPNCLZSEb32KKVDJDl/MOt2N01qU2H/FkzEa9PKto1BqDjtd7Q==",
2474 | "requires": {
2475 | "ansi-styles": "^3.2.0",
2476 | "string-width": "^3.0.0",
2477 | "strip-ansi": "^5.0.0"
2478 | },
2479 | "dependencies": {
2480 | "ansi-regex": {
2481 | "version": "4.1.0",
2482 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz",
2483 | "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg=="
2484 | },
2485 | "is-fullwidth-code-point": {
2486 | "version": "2.0.0",
2487 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz",
2488 | "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8="
2489 | },
2490 | "string-width": {
2491 | "version": "3.1.0",
2492 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz",
2493 | "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==",
2494 | "requires": {
2495 | "emoji-regex": "^7.0.1",
2496 | "is-fullwidth-code-point": "^2.0.0",
2497 | "strip-ansi": "^5.1.0"
2498 | }
2499 | },
2500 | "strip-ansi": {
2501 | "version": "5.2.0",
2502 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz",
2503 | "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==",
2504 | "requires": {
2505 | "ansi-regex": "^4.1.0"
2506 | }
2507 | }
2508 | }
2509 | },
2510 | "wrappy": {
2511 | "version": "1.0.2",
2512 | "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz",
2513 | "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8="
2514 | },
2515 | "write-file-atomic": {
2516 | "version": "3.0.3",
2517 | "resolved": "https://registry.npmjs.org/write-file-atomic/-/write-file-atomic-3.0.3.tgz",
2518 | "integrity": "sha512-AvHcyZ5JnSfq3ioSyjrBkH9yW4m7Ayk8/9My/DD9onKeu/94fwrMocemO2QAJFAlnnDN+ZDS+ZjAR5ua1/PV/Q==",
2519 | "dev": true,
2520 | "requires": {
2521 | "imurmurhash": "^0.1.4",
2522 | "is-typedarray": "^1.0.0",
2523 | "signal-exit": "^3.0.2",
2524 | "typedarray-to-buffer": "^3.1.5"
2525 | }
2526 | },
2527 | "xdg-basedir": {
2528 | "version": "4.0.0",
2529 | "resolved": "https://registry.npmjs.org/xdg-basedir/-/xdg-basedir-4.0.0.tgz",
2530 | "integrity": "sha512-PSNhEJDejZYV7h50BohL09Er9VaIefr2LMAf3OEmpCkjOi34eYyQYAXUTjEQtZJTKcF0E2UKTh+osDLsgNim9Q==",
2531 | "dev": true
2532 | },
2533 | "y18n": {
2534 | "version": "4.0.0",
2535 | "resolved": "https://registry.npmjs.org/y18n/-/y18n-4.0.0.tgz",
2536 | "integrity": "sha512-r9S/ZyXu/Xu9q1tYlpsLIsa3EeLXXk0VwlxqTcFRfg9EhMW+17kbt9G0NrgCmhGb5vT2hyhJZLfDGx+7+5Uj/w=="
2537 | },
2538 | "yallist": {
2539 | "version": "3.1.1",
2540 | "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz",
2541 | "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g=="
2542 | },
2543 | "yargs": {
2544 | "version": "13.3.2",
2545 | "resolved": "https://registry.npmjs.org/yargs/-/yargs-13.3.2.tgz",
2546 | "integrity": "sha512-AX3Zw5iPruN5ie6xGRIDgqkT+ZhnRlZMLMHAs8tg7nRruy2Nb+i5o9bwghAogtM08q1dpr2LVoS8KSTMYpWXUw==",
2547 | "requires": {
2548 | "cliui": "^5.0.0",
2549 | "find-up": "^3.0.0",
2550 | "get-caller-file": "^2.0.1",
2551 | "require-directory": "^2.1.1",
2552 | "require-main-filename": "^2.0.0",
2553 | "set-blocking": "^2.0.0",
2554 | "string-width": "^3.0.0",
2555 | "which-module": "^2.0.0",
2556 | "y18n": "^4.0.0",
2557 | "yargs-parser": "^13.1.2"
2558 | },
2559 | "dependencies": {
2560 | "ansi-regex": {
2561 | "version": "4.1.0",
2562 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz",
2563 | "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg=="
2564 | },
2565 | "is-fullwidth-code-point": {
2566 | "version": "2.0.0",
2567 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz",
2568 | "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8="
2569 | },
2570 | "string-width": {
2571 | "version": "3.1.0",
2572 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz",
2573 | "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==",
2574 | "requires": {
2575 | "emoji-regex": "^7.0.1",
2576 | "is-fullwidth-code-point": "^2.0.0",
2577 | "strip-ansi": "^5.1.0"
2578 | }
2579 | },
2580 | "strip-ansi": {
2581 | "version": "5.2.0",
2582 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz",
2583 | "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==",
2584 | "requires": {
2585 | "ansi-regex": "^4.1.0"
2586 | }
2587 | }
2588 | }
2589 | },
2590 | "yargs-parser": {
2591 | "version": "13.1.2",
2592 | "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-13.1.2.tgz",
2593 | "integrity": "sha512-3lbsNRf/j+A4QuSZfDRA7HRSfWrzO0YjqTJd5kjAq37Zep1CEgaYmrH9Q3GwPiB9cHyd1Y1UwggGhJGoxipbzg==",
2594 | "requires": {
2595 | "camelcase": "^5.0.0",
2596 | "decamelize": "^1.2.0"
2597 | }
2598 | }
2599 | }
2600 | }
2601 |
--------------------------------------------------------------------------------
/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "ecommerce",
3 | "version": "1.0.0",
4 | "description": "",
5 | "main": "server.js",
6 | "scripts": {
7 | "start": "node server.js",
8 | "server": "nodemon server.js",
9 | "client": "cd client && npm run start",
10 | "server-install": "npm install",
11 | "client-install": "cd client && npm install",
12 | "install-all": "concurrently \"npm run server-install\" \"npm run client-install\"",
13 | "dev": "concurrently \"npm run server\" \"npm run client\"",
14 | "heroku-postbuild": "cd client && npm install && npm run build"
15 | },
16 | "keywords": [],
17 | "author": "",
18 | "license": "ISC",
19 | "dependencies": {
20 | "bcrypt": "^5.0.0",
21 | "cloudinary": "^1.23.0",
22 | "concurrently": "^5.3.0",
23 | "cookie-parser": "^1.4.5",
24 | "cors": "^2.8.5",
25 | "dotenv": "^8.2.0",
26 | "express": "^4.17.1",
27 | "express-fileupload": "^1.2.0",
28 | "jsonwebtoken": "^8.5.1",
29 | "mongoose": "^5.10.1"
30 | },
31 | "devDependencies": {
32 | "nodemon": "^2.0.4"
33 | }
34 | }
35 |
--------------------------------------------------------------------------------
/routes/categoryRouter.js:
--------------------------------------------------------------------------------
1 | const router = require('express').Router()
2 | const categoryCtrl = require('../controllers/categoryCtrl')
3 | const auth = require('../middleware/auth')
4 | const authAdmin = require('../middleware/authAdmin')
5 |
6 |
7 | router.route('/category')
8 | .get(categoryCtrl.getCategories)
9 | .post(auth, authAdmin, categoryCtrl.createCategory)
10 |
11 | router.route('/category/:id')
12 | .delete(auth, authAdmin, categoryCtrl.deleteCategory)
13 | .put(auth, authAdmin, categoryCtrl.updateCategory)
14 |
15 |
16 | module.exports = router
--------------------------------------------------------------------------------
/routes/paymentRouter.js:
--------------------------------------------------------------------------------
1 | const router = require('express').Router()
2 | const paymentCtrl = require('../controllers/paymentCtrl')
3 | const auth = require('../middleware/auth')
4 | const authAdmin = require('../middleware/authAdmin')
5 |
6 |
7 | router.route('/payment')
8 | .get(auth, authAdmin, paymentCtrl.getPayments)
9 | .post(auth, paymentCtrl.createPayment)
10 |
11 |
12 | module.exports = router
13 |
--------------------------------------------------------------------------------
/routes/productRouter.js:
--------------------------------------------------------------------------------
1 | const router = require('express').Router()
2 | const productCtrl = require('../controllers/productCtrl')
3 | const auth = require('../middleware/auth')
4 | const authAdmin = require('../middleware/authAdmin')
5 |
6 |
7 | router.route('/products')
8 | .get(productCtrl.getProducts)
9 | .post(auth, authAdmin, productCtrl.createProduct)
10 |
11 |
12 | router.route('/products/:id')
13 | .delete(auth, authAdmin, productCtrl.deleteProduct)
14 | .put(auth, authAdmin, productCtrl.updateProduct)
15 |
16 |
17 |
18 | module.exports = router
--------------------------------------------------------------------------------
/routes/upload.js:
--------------------------------------------------------------------------------
1 | const router = require('express').Router()
2 | const cloudinary = require('cloudinary')
3 | const auth = require('../middleware/auth')
4 | const authAdmin = require('../middleware/authAdmin')
5 | const fs = require('fs')
6 |
7 |
8 | // we will upload image on cloudinary
9 | cloudinary.config({
10 | cloud_name: process.env.CLOUD_NAME,
11 | api_key: process.env.CLOUD_API_KEY,
12 | api_secret: process.env.CLOUD_API_SECRET
13 | })
14 |
15 | // Upload image only admin can use
16 | router.post('/upload',auth , authAdmin, (req, res) =>{
17 | try {
18 | if(!req.files || Object.keys(req.files).length === 0)
19 | return res.status(400).json({msg: 'No files were uploaded.'})
20 |
21 | const file = req.files.file;
22 | if(file.size > 1024*1024) {
23 | removeTmp(file.tempFilePath)
24 | return res.status(400).json({msg: "Size too large"})
25 | }
26 |
27 | if(file.mimetype !== 'image/jpeg' && file.mimetype !== 'image/png'){
28 | removeTmp(file.tempFilePath)
29 | return res.status(400).json({msg: "File format is incorrect."})
30 | }
31 |
32 | cloudinary.v2.uploader.upload(file.tempFilePath, {folder: "test"}, async(err, result)=>{
33 | if(err) throw err;
34 |
35 | removeTmp(file.tempFilePath)
36 |
37 | res.json({public_id: result.public_id, url: result.secure_url})
38 | })
39 |
40 |
41 | } catch (err) {
42 | return res.status(500).json({msg: err.message})
43 | }
44 | })
45 |
46 | // Delete image only admin can use
47 | router.post('/destroy',auth , authAdmin, (req, res) =>{
48 | try {
49 | const {public_id} = req.body;
50 | if(!public_id) return res.status(400).json({msg: 'No images Selected'})
51 |
52 | cloudinary.v2.uploader.destroy(public_id, async(err, result) =>{
53 | if(err) throw err;
54 |
55 | res.json({msg: "Deleted Image"})
56 | })
57 |
58 | } catch (err) {
59 | return res.status(500).json({msg: err.message})
60 | }
61 |
62 | })
63 |
64 |
65 | const removeTmp = (path) =>{
66 | fs.unlink(path, err=>{
67 | if(err) throw err;
68 | })
69 | }
70 |
71 | module.exports = router
--------------------------------------------------------------------------------
/routes/userRouter.js:
--------------------------------------------------------------------------------
1 | const router = require('express').Router()
2 | const userCtrl = require('../controllers/userCtrl')
3 | const auth = require('../middleware/auth')
4 |
5 | router.post('/register', userCtrl.register)
6 |
7 | router.post('/login', userCtrl.login)
8 |
9 | router.get('/logout', userCtrl.logout)
10 |
11 | router.get('/refresh_token', userCtrl.refreshToken)
12 |
13 | router.get('/infor', auth, userCtrl.getUser)
14 |
15 | router.patch('/addcart', auth, userCtrl.addCart)
16 |
17 | router.get('/history', auth, userCtrl.history)
18 |
19 |
20 | module.exports = router
--------------------------------------------------------------------------------
/server.js:
--------------------------------------------------------------------------------
1 | require('dotenv').config()
2 | const express = require('express')
3 | const mongoose = require('mongoose')
4 | const cors = require('cors')
5 | const fileUpload = require('express-fileupload')
6 | const cookieParser = require('cookie-parser')
7 | const path = require('path')
8 |
9 |
10 | const app = express()
11 | app.use(express.json())
12 | app.use(cookieParser())
13 | app.use(cors())
14 | app.use(fileUpload({
15 | useTempFiles: true
16 | }))
17 |
18 | // Routes
19 | app.use('/user', require('./routes/userRouter'))
20 | app.use('/api', require('./routes/categoryRouter'))
21 | app.use('/api', require('./routes/upload'))
22 | app.use('/api', require('./routes/productRouter'))
23 | app.use('/api', require('./routes/paymentRouter'))
24 |
25 |
26 |
27 | // Connect to mongodb
28 | const URI = process.env.MONGODB_URL
29 | mongoose.connect(URI, {
30 | useCreateIndex: true,
31 | useFindAndModify: false,
32 | useNewUrlParser: true,
33 | useUnifiedTopology: true
34 | }, err =>{
35 | if(err) throw err;
36 | console.log('Connected to MongoDB')
37 | })
38 |
39 | if(process.env.NODE_ENV === 'production'){
40 | app.use(express.static('client/build'))
41 | app.get('*', (req, res) => {
42 | res.sendFile(path.join(__dirname, 'client', 'build', 'index.html'))
43 | })
44 | }
45 |
46 |
47 |
48 | const PORT = process.env.PORT || 5000
49 | app.listen(PORT, () =>{
50 | console.log('Server is running on port', PORT)
51 | })
--------------------------------------------------------------------------------