├── README.md
├── client
├── .dockerignore
├── .gitignore
├── Dockerfile
├── README.md
├── package-lock.json
├── package.json
├── public
│ ├── favicon.ico
│ ├── index.html
│ └── poster.png
├── src
│ ├── App.js
│ ├── components
│ │ ├── Layout.js
│ │ ├── Nav.css
│ │ ├── Navbar.js
│ │ ├── RoomCard.js
│ │ ├── UserCard.js
│ │ └── UserCardSm.js
│ ├── index.css
│ ├── index.js
│ ├── resources
│ │ ├── dummy.js
│ │ ├── img
│ │ │ ├── acmlogo.png
│ │ │ ├── keyboard.png
│ │ │ ├── landing.png
│ │ │ ├── landingImg1.jpg
│ │ │ ├── landingImg2.jpg
│ │ │ ├── landingImg3.jpg
│ │ │ ├── landingImg4.jpg
│ │ │ ├── landingImg5.png
│ │ │ └── logout.png
│ │ └── theme.js
│ ├── screens
│ │ ├── AllRooms.js
│ │ ├── Landing.js
│ │ └── Room.js
│ └── store
│ │ └── auth.js
└── yarn.lock
├── docker-compose.yaml
├── nginx
└── nginx.conf
├── package-lock.json
└── server
├── .dockerignore
├── .gitignore
├── Dockerfile
├── README.md
├── package-lock.json
├── package.json
└── src
├── app.js
├── db
└── mongoose.js
├── index.js
├── middleware
├── auth.js
├── banStatus.js
└── roomAdmin.js
├── models
├── room.js
└── user.js
├── oauth2
├── googleAuthHelpers.js
└── googleAuthRouters.js
└── routers
├── room.js
├── roomAdmin.js
└── user.js
/README.md:
--------------------------------------------------------------------------------
1 | # Litecode
2 | [litecode.bitsacm.in](https://litecode.bitsacm.in)
3 |
4 |
5 |
6 | ## About
7 | ~ 2 months before every internship season, every single CS student in the world has felt the pain of the sinking realization that they don't know how to do competitive coding. Luckily, LeetCode Premium is one of the best resources for practicing questions and attempting mock interviews. Unfortunately, it is also *obscenely* expensive unless you have an extra kidney at home.
8 |
9 | So, we built an app that allows you to discover other BITSians willing to share a LeetCode premium subscription.
10 |
11 | Now you don't need to stress about **Leetcode**, you can just take ***Lite*-code**. ~~sorry dush-t pls dont kick me from acm~~
12 |
13 | ## Usage
14 | Just create an account using your BITS email address, and view all the open rooms! Once you find a room with some of your friends, you can join the room. In case your friends haven't signed up yet, you can create a room and give them the room name!
15 |
16 | Once the room gets filled with 4 people, the admin locks the room and will create a WhatsApp group with the members. From there, it's upto you to make the payment and start coding!
17 |
18 | ## For Developers and Hackermanz
19 | This app was made with <3 by students from BITS Pilani, using `React.js` for the frontend and `Node.js` for the backend. It uses the *Chakra-UI* library, *Google O-Auth*, and *Docker+Nginx* for hosting.
20 |
21 | In case you'd like to run this on your local machine, follow these steps:
22 |
23 | **Clone the repository**
24 | ```git clone https://github.com/bitsacm/litecode.git```
25 |
26 | **Build the client**
27 | ```cd client```
28 | ```yarn build```
29 |
30 | **Run the local server**
31 | ```cd ../server```
32 | ```npm run dev```
33 |
34 | In case you'd like to contribute or squash a bug, just open a pull request!
35 |
36 | ## Contributors
37 | This project was made by **ACM** at BITS Pilani, Pilani. If you liked this project, go follow the developers!
38 |
39 | **Frontend Developer:** Parth Sharma [(View GitHub)](https://github.com/psrth)
40 | **Backend Developer:** Aayush Paurana [(View GitHub)](https://github.com/tech-yush)
41 | **DevOps Engineer:** Aditya Thakur [(View GitHub)](https://github.com/cry0genic)
42 |
43 |
44 |
45 |
--------------------------------------------------------------------------------
/client/.dockerignore:
--------------------------------------------------------------------------------
1 | node_modules
2 | build
3 | .dockerignore
4 | Dockerfile
--------------------------------------------------------------------------------
/client/.gitignore:
--------------------------------------------------------------------------------
1 | # See https://help.github.com/articles/ignoring-files/ for more about ignoring files.
2 |
3 | # dependencies
4 | /node_modules
5 | /.pnp
6 | .pnp.js
7 |
8 | # testing
9 | /coverage
10 |
11 | # production
12 | /build
13 |
14 | # misc
15 | .DS_Store
16 | .env.local
17 | .env.development.local
18 | .env.test.local
19 | .env.production.local
20 |
21 | npm-debug.log*
22 | yarn-debug.log*
23 | yarn-error.log*
24 |
25 | config.env
--------------------------------------------------------------------------------
/client/Dockerfile:
--------------------------------------------------------------------------------
1 | FROM node:13.12.0-alpine as build
2 |
3 | WORKDIR /app
4 |
5 | ENV PATH /app/node_modules/.bin:$PATH
6 |
7 | COPY package*.json ./
8 | RUN npm install -g
9 | RUN npm ci --silent
10 | RUN npm install react-scripts@3.4.1 -g --silent
11 |
12 | RUN npm install --slient
13 |
14 | COPY . ./
15 |
16 | CMD npm run build
--------------------------------------------------------------------------------
/client/README.md:
--------------------------------------------------------------------------------
1 | # LITECODE CLIENT
--------------------------------------------------------------------------------
/client/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "litecode",
3 | "version": "0.1.0",
4 | "private": true,
5 | "dependencies": {
6 | "@chakra-ui/icons": "^1.0.13",
7 | "@chakra-ui/react": "^1.6.3",
8 | "@chakra-ui/theme-tools": "^1.1.7",
9 | "@emotion/react": "^11.4.0",
10 | "@emotion/styled": "^11.3.0",
11 | "@testing-library/jest-dom": "^5.11.4",
12 | "@testing-library/react": "^11.1.0",
13 | "@testing-library/user-event": "^12.1.10",
14 | "axios": "^0.21.1",
15 | "fetch": "^1.1.0",
16 | "framer-motion": "^4.1.17",
17 | "js-cookie": "^2.2.1",
18 | "react": "^17.0.2",
19 | "react-dom": "^17.0.2",
20 | "react-github-btn": "^1.2.0",
21 | "react-github-buttons": "^0.5.0",
22 | "react-router-dom": "^5.2.0",
23 | "react-scripts": "4.0.3",
24 | "typeface-roboto": "^1.1.13",
25 | "web-vitals": "^1.0.1"
26 | },
27 | "scripts": {
28 | "start": "PORT=3001 react-scripts start",
29 | "build": "PORT=3069 react-scripts build",
30 | "test": "react-scripts test",
31 | "eject": "react-scripts eject"
32 | },
33 | "eslintConfig": {
34 | "extends": [
35 | "react-app",
36 | "react-app/jest"
37 | ]
38 | },
39 | "browserslist": {
40 | "production": [
41 | ">0.2%",
42 | "not dead",
43 | "not op_mini all"
44 | ],
45 | "development": [
46 | "last 1 chrome version",
47 | "last 1 firefox version",
48 | "last 1 safari version"
49 | ]
50 | }
51 | }
52 |
--------------------------------------------------------------------------------
/client/public/favicon.ico:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/bitsacm/litecode/389f3f765c7fb0990d1b785fc95a6ec77c860311/client/public/favicon.ico
--------------------------------------------------------------------------------
/client/public/index.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
12 |
13 |
17 |
18 |
19 |
20 |
21 |
28 |
29 | Litecode | BITS ACM
30 |
31 |
32 |
33 |
34 |
44 |
45 |
46 |
--------------------------------------------------------------------------------
/client/public/poster.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/bitsacm/litecode/389f3f765c7fb0990d1b785fc95a6ec77c860311/client/public/poster.png
--------------------------------------------------------------------------------
/client/src/App.js:
--------------------------------------------------------------------------------
1 | import React, { Fragment, useContext, useState, useEffect, useRef } from 'react'
2 | import { Switch, Route, Redirect } from 'react-router-dom';
3 |
4 | import Layout from './components/Layout'
5 | import AllRooms from './screens/AllRooms';
6 | import Landing from './screens/Landing';
7 | import Room from './screens/Room';
8 |
9 | import {
10 | Button,
11 | Flex,
12 | Box,
13 | FormControl,
14 | FormLabel,
15 | Input,
16 | Text,
17 | FormHelperText,
18 | InputGroup,
19 | InputLeftAddon,
20 | } from '@chakra-ui/react';
21 |
22 | import AuthContext from './store/auth';
23 |
24 | function App() {
25 | const authCtx = useContext(AuthContext);
26 | const token = authCtx.token;
27 | const isLoggedIn = authCtx.isLoggedIn
28 |
29 | const [mobile, setMobile] = useState("dummy");
30 | const [inRoom, setInRoom] = useState(null);
31 |
32 |
33 | useEffect(() => {
34 | getMobile();
35 | getRoom();
36 | }, [])
37 |
38 | const getMobile = () => {
39 | fetch('https://litecode.bitsacm.in/server/users/me',
40 | {
41 | method: 'GET',
42 | headers: {
43 | 'Content-Type': 'application/json',
44 | 'Authorization': 'Bearer '+token,
45 | }
46 | }
47 | ).then(response =>
48 | response.json().then(data => ({
49 | data: data,
50 | status: response.status
51 | })
52 | ).then(res => {
53 | if(res.data){
54 | if (res.data.user.phoneNo) {
55 | const phone = res.data.user.phoneNo
56 | setMobile(phone)
57 | }
58 | else {
59 | setMobile(null)
60 | }
61 | }
62 | }))
63 | }
64 |
65 | const getRoom = () => {
66 | fetch('http://localhost:3000/users/me',
67 | {
68 | method: 'GET',
69 | headers: {
70 | 'Content-Type': 'application/json',
71 | 'Authorization': 'Bearer '+token,
72 | }
73 | }
74 | ).then(response =>
75 | response.json().then(data => ({
76 | data: data,
77 | status: response.status
78 | })
79 | ).then(res => {
80 | if(res.data){
81 | if (res.data.user.inRoom) {
82 | const smth = "yes"
83 | setInRoom(smth)
84 | }
85 | else {
86 | setInRoom(null)
87 | }
88 | console.log(inRoom)
89 | }
90 | }))
91 | }
92 |
93 |
94 |
95 |
96 |
97 | return (
98 |
99 |
100 |
101 | {isLoggedIn ?
102 |
103 |
104 | {mobile ?
105 | {inRoom ? : }
106 | : }
107 |
108 |
109 |
110 | {mobile ? : }
111 |
112 |
113 |
114 | {mobile ? : }
115 |
116 |
117 | :
118 |
119 |
120 |
121 |
122 |
123 |
124 |
125 |
126 |
127 |
128 |
129 | }
130 |
131 |
132 |
133 |
134 |
135 |
136 |
137 |
138 | );
139 | }
140 |
141 | const AddMobile = (props) => {
142 | const authCtx = useContext(AuthContext);
143 | const token = authCtx.token;
144 |
145 | const [refMobile, setRefMobile] = React.useState("")
146 | const handleChange = (event) => setRefMobile(event.target.value)
147 |
148 |
149 | const submitHandler = (e) => {
150 | e.preventDefault();
151 |
152 | if (refMobile.length === 10 && !(isNaN(refMobile))) {
153 | fetch('https://litecode.bitsacm.in/server/users/me',
154 | {
155 | method: 'PATCH',
156 | headers: {
157 | 'Content-Type': 'application/json',
158 | 'Authorization': 'Bearer '+token,
159 | },
160 | body: JSON.stringify(
161 | {
162 | phoneNo: refMobile
163 | }
164 | ),
165 | }
166 | ).then(response =>
167 | response.json().then(data => ({
168 | data: data,
169 | status: response.status
170 | })
171 | ).then(res => {
172 | if(res.data){
173 | props.setMobile(refMobile)
174 | } else {
175 | alert("An account with this number already exists. Please try again.");
176 | }
177 | }))
178 | }
179 | else {
180 | alert("Please enter your valid phone number.")
181 | }
182 | }
183 |
184 |
185 | return(
186 |
187 |
192 |
193 |
199 |
200 |
201 | WhatsApp Number
202 |
203 |
204 |
213 |
214 | Please enter your 10-digit phone number.
215 |
216 |
217 |
225 |
226 |
227 |
228 | )
229 | }
230 |
231 | // let AddMobile = setTimeout(AddSMobile, 2000);
232 |
233 | export default App;
234 |
--------------------------------------------------------------------------------
/client/src/components/Layout.js:
--------------------------------------------------------------------------------
1 | import React, { Fragment } from 'react'
2 | import { Box } from '@chakra-ui/react'
3 |
4 | import Navbar from './Navbar'
5 |
6 | const Layout = (props) => {
7 | return (
8 |
12 |
13 | {props.children}
14 |
15 | );
16 | };
17 |
18 | export default Layout;
--------------------------------------------------------------------------------
/client/src/components/Nav.css:
--------------------------------------------------------------------------------
1 | .navselected {
2 | font-weight: bold;
3 | }
4 |
5 | .heading {
6 | font-size: 32px;
7 |
8 | }
--------------------------------------------------------------------------------
/client/src/components/Navbar.js:
--------------------------------------------------------------------------------
1 | import React, { Fragment, useContext, useState, useEffect } from 'react'
2 | import { NavLink, Link, useLocation } from 'react-router-dom'
3 | import {
4 | HStack,
5 | Heading,
6 | Text,
7 | Spacer,
8 | Button,
9 | Image,
10 | Box,
11 | Flex
12 | } from '@chakra-ui/react'
13 |
14 | import AuthContext from '../store/auth'
15 | import imgurl from '../resources/img/keyboard.png'
16 | import acmlogo from '../resources/img/acmlogo.png'
17 | import logoutpng from '../resources/img/logout.png'
18 |
19 | import './Nav.css'
20 |
21 | const Navbar = () => {
22 | const authCtx = useContext(AuthContext);
23 | const isLoggedIn = authCtx.isLoggedIn;
24 | const token = authCtx.token;
25 |
26 | const [hasRoom, setHasRoom] = useState(true);
27 |
28 | const login = () => {
29 | authCtx.login()
30 | }
31 |
32 | function roomBold(location) {
33 | const pathName = location.pathname
34 | if (pathName === '/room') return true
35 | else return false
36 | }
37 |
38 | const location = useLocation()
39 | const boldRoom = roomBold(location)
40 |
41 | return(
42 |
47 |
48 |
52 |
57 | litecode
62 |
63 |
64 |
65 |
66 |
67 |
68 | {isLoggedIn ?
69 |
70 | {hasRoom?
71 |
72 | My Room
78 | :null}
79 |
80 |
81 | Open Rooms
87 |
88 |
89 | {/* */}
98 |
99 | :
100 | }
105 |
106 | )
107 | }
108 |
109 | export default Navbar;
--------------------------------------------------------------------------------
/client/src/components/RoomCard.js:
--------------------------------------------------------------------------------
1 | import React, { Fragment, useEffect, useState, useContext } from 'react'
2 | import AuthContext from '../store/auth.js'
3 |
4 | import {
5 | Heading,
6 | Text,
7 | Image,
8 | Flex,
9 | Box,
10 | Button,
11 | Popover,
12 | PopoverTrigger,
13 | PopoverContent,
14 | PopoverHeader,
15 | PopoverBody,
16 | PopoverFooter,
17 | PopoverArrow,
18 | PopoverCloseButton,
19 | ButtonGroup,
20 | } from '@chakra-ui/react'
21 |
22 | import {
23 | Modal,
24 | ModalOverlay,
25 | ModalContent,
26 | ModalHeader,
27 | ModalFooter,
28 | ModalBody,
29 | ModalCloseButton,
30 | } from "@chakra-ui/react"
31 |
32 | import { useDisclosure } from "@chakra-ui/react"
33 |
34 | import UserCardSm from './UserCardSm'
35 |
36 | const RoomCard = (props) => {
37 | const { isOpen, onOpen, onClose } = useDisclosure()
38 | const roomId = props.room.roomID;
39 |
40 | const authCtx = useContext(AuthContext);
41 | const token = authCtx.token;
42 |
43 | const joinRoom = () => {
44 | fetch('https://litecode.bitsacm.in/server/joinRoom/'+roomId,
45 | {
46 | method: 'POST',
47 | headers: {
48 | 'Content-Type': 'application/json',
49 | 'Authorization': 'Bearer '+token,
50 | }
51 | }
52 | ).then(response =>
53 | response.json().then(data => ({
54 | data: data,
55 | status: response.status
56 | })
57 | ).then(res => {
58 | if(res.data){
59 | console.log("do u see me")
60 | props.updateRedirect();
61 |
62 | } else {
63 | alert("ERROR POSTING CONTENT.");
64 | }
65 | }))
66 | }
67 |
68 | return(
69 |
70 |
80 |
82 | {props.room.roomID}
87 | {props.room.users ?
88 |
89 |
90 | {props.room.users.map((user, id) => (
91 |
101 | ))}
102 |
103 | :null}
104 | You’ll need to pay ₹{props.room.toPay} if you join
109 |
122 |
123 |
124 |
125 |
126 | {props.room.roomID}
131 |
132 | {props.room.users ?
133 |
134 | There’s {props.room.users.length} members here
140 |
145 |
146 | {(props.room.users.map((user, id) => (
147 |
148 | )))}
149 |
150 |
151 | You Pay
156 | ₹ {props.room.toPay}
162 |
163 |
164 | {(props.userRoom) ?
165 |
166 | {props.userRoom === props.room.roomID ?
167 | You're already in this room!
174 | :
175 | You're already in another room!
182 | }
183 | : null }
184 | {(props.isBanned) ?
185 | You've been banned for 48 hours.
192 | : null}
193 |
194 |
195 |
196 |
197 | :null}
198 |
199 |
200 |
201 |
202 |
203 | )
204 | }
205 |
206 |
207 |
208 | const ControlledUsage = (props) => {
209 | const [isOpen, setIsOpen] = React.useState(false)
210 | const open = () => setIsOpen(!isOpen)
211 | const close = () => setIsOpen(false)
212 | return (
213 | <>
214 |
215 |
231 |
238 |
239 | Confirmation
240 |
241 |
242 |
243 | Are you sure you want to join this room? If you leave this room after joining, you will be banned for 48 hours.
244 |
245 |
246 |
247 |
248 |
259 |
260 |
261 |
262 |
263 |
264 | >
265 | )
266 | }
267 |
268 | export default RoomCard;
--------------------------------------------------------------------------------
/client/src/components/UserCard.js:
--------------------------------------------------------------------------------
1 | import React, { Fragment, useEffect, useState, useContext } from 'react'
2 | import AuthContext from '../store/auth.js'
3 | import {
4 | Heading,
5 | Text,
6 | Image,
7 | Flex,
8 | Popover,
9 | PopoverTrigger,
10 | PopoverContent,
11 | PopoverHeader,
12 | PopoverBody,
13 | PopoverFooter,
14 | PopoverArrow,
15 | PopoverCloseButton,
16 | Button,
17 | ButtonGroup,
18 | } from '@chakra-ui/react'
19 |
20 | import { DeleteIcon } from '@chakra-ui/icons'
21 |
22 | const UserCard = (props) => {
23 |
24 | const authCtx = useContext(AuthContext);
25 | const token = authCtx.token;
26 |
27 | const deleteUser = () => {
28 | fetch('https://litecode.bitsacm.in/server/remove/'+props.id,
29 | {
30 | method: 'PATCH',
31 | headers: {
32 | 'Content-Type': 'application/json',
33 | 'Authorization': 'Bearer '+token,
34 | }
35 | }
36 | ).then(response =>
37 | response.json().then(data => ({
38 | data: data,
39 | status: response.status
40 | })
41 | ).then(res => {
42 | if(res.data){
43 | console.log(res.data)
44 | props.loadRoom();
45 | } else {
46 | alert("ERROR POSTING CONTENT.");
47 | }
48 | }))
49 | }
50 |
51 | const cleanName = (str) => {
52 | return str.split(' ')
53 | .map(w => w[0].toUpperCase() + w.substr(1).toLowerCase())
54 | .join(' ')
55 | }
56 |
57 | return(
58 |
59 |
70 |
71 |
72 |
77 |
81 | {cleanName(props.name)}
86 |
87 | {props.isAdmin ?
88 | ADMIN
98 | :
99 | {props.userIsAdmin ?
100 | : null}
101 | }
102 |
103 |
104 | {props.phoneNo}
105 |
106 |
107 |
108 | )
109 | }
110 |
111 |
112 | const ControlledUsage = (props) => {
113 | const [isOpen, setIsOpen] = React.useState(false)
114 | const open = () => setIsOpen(!isOpen)
115 | const close = () => setIsOpen(false)
116 | return (
117 | <>
118 |
119 |
126 |
133 |
134 | Confirmation
135 |
136 |
137 |
138 | Are you sure you want to remove this member from the room? This action cannot be undone.
139 |
140 |
141 |
142 |
143 |
154 |
155 |
156 |
157 |
158 |
159 | >
160 | )
161 | }
162 |
163 | export default UserCard;
--------------------------------------------------------------------------------
/client/src/components/UserCardSm.js:
--------------------------------------------------------------------------------
1 | import React, { Fragment } from 'react'
2 | import {
3 | Heading,
4 | Text,
5 | Image,
6 | Flex,
7 | } from '@chakra-ui/react'
8 |
9 |
10 | const UserCardSm = (props) => {
11 | return(
12 |
13 |
25 |
26 |
27 |
28 | {props.user.name}
29 | {props.user.isAdmin ?
30 | ADMIN
39 | :
40 | null}
41 |
42 | 99XXXXXXXX
43 |
44 |
45 |
46 | )
47 | }
48 |
49 | export default UserCardSm;
--------------------------------------------------------------------------------
/client/src/index.css:
--------------------------------------------------------------------------------
1 | body {
2 | margin: 0;
3 | font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', 'Roboto', 'Oxygen',
4 | 'Ubuntu', 'Cantarell', 'Fira Sans', 'Droid Sans', 'Helvetica Neue',
5 | sans-serif;
6 | -webkit-font-smoothing: antialiased;
7 | -moz-osx-font-smoothing: grayscale;
8 | }
9 |
10 | code {
11 | font-family: source-code-pro, Menlo, Monaco, Consolas, 'Courier New',
12 | monospace;
13 | }
14 |
15 | .active-link {
16 | font-weight: 700;
17 | color: "litegrey.600"
18 | }
19 |
20 | .active-link > p {
21 | font-weight: 700;
22 | color: "litegrey.600"
23 | }
--------------------------------------------------------------------------------
/client/src/index.js:
--------------------------------------------------------------------------------
1 | import React from 'react';
2 | import ReactDOM from 'react-dom';
3 | import { BrowserRouter } from 'react-router-dom';
4 | import { ChakraProvider } from "@chakra-ui/react"
5 | import { AuthContextProvider } from './store/auth';
6 | import 'typeface-roboto'
7 |
8 | import './index.css';
9 | import App from './App';
10 | import theme from './resources/theme'
11 |
12 |
13 | ReactDOM.render(
14 |
15 |
16 |
17 |
18 |
19 |
20 |
21 |
22 | ,
23 | document.getElementById('root')
24 | );
25 |
--------------------------------------------------------------------------------
/client/src/resources/dummy.js:
--------------------------------------------------------------------------------
1 | export const DummyData = {
2 | allRooms: [
3 | {
4 | id: 1,
5 | name: "randomgroup",
6 | price: 2500,
7 | members: [
8 | {
9 | id: 1,
10 | isAdmin: true,
11 | phone: "+91 9910231328",
12 | name: "Parth Sharma",
13 | imgUrl: "https://avatars.githubusercontent.com/u/45586386?v=4"
14 | },
15 | {
16 | id: 2,
17 | isAdmin: false,
18 | phone: "+91 9910231328",
19 | name: "Parth Sharma",
20 | imgUrl: "https://avatars.githubusercontent.com/u/45586386?v=4"
21 | },
22 | {
23 | id: 3,
24 | isAdmin: false,
25 | phone: "+91 9910231328",
26 | name: "Parth Sharma",
27 | imgUrl: "https://avatars.githubusercontent.com/u/45586386?v=4"
28 | },
29 | {
30 | id: 4,
31 | isAdmin: false,
32 | phone: "+91 9910231328",
33 | name: "Parth Sharma",
34 | imgUrl: "https://avatars.githubusercontent.com/u/45586386?v=4"
35 | },
36 | ]
37 | },
38 | {
39 | id: 2,
40 | name: "randomgroup2",
41 | price: 2500,
42 | members: [
43 | {
44 | id: 1,
45 | isAdmin: true,
46 | phone: "+91 9910231328",
47 | name: "Parth Sharma",
48 | imgUrl: "https://avatars.githubusercontent.com/u/45586386?v=4"
49 | },
50 | {
51 | id: 2,
52 | isAdmin: false,
53 | phone: "+91 9910231328",
54 | name: "Parth Sharma",
55 | imgUrl: "https://avatars.githubusercontent.com/u/45586386?v=4"
56 | },
57 | {
58 | id: 3,
59 | isAdmin: false,
60 | phone: "+91 9910231328",
61 | name: "Parth Sharma",
62 | imgUrl: "https://avatars.githubusercontent.com/u/45586386?v=4"
63 | },
64 | {
65 | id: 4,
66 | isAdmin: false,
67 | phone: "+91 9910231328",
68 | name: "Parth Sharma",
69 | imgUrl: "https://avatars.githubusercontent.com/u/45586386?v=4"
70 | },
71 | ]
72 | },
73 | {
74 | id: 3,
75 | name: "randomgroup3",
76 | price: 2500,
77 | members: [
78 | {
79 | id: 1,
80 | isAdmin: true,
81 | phone: "+91 9910231328",
82 | name: "Parth Sharma",
83 | imgUrl: "https://avatars.githubusercontent.com/u/45586386?v=4"
84 | },
85 | {
86 | id: 2,
87 | isAdmin: false,
88 | phone: "+91 9910231328",
89 | name: "Parth Sharma",
90 | imgUrl: "https://avatars.githubusercontent.com/u/45586386?v=4"
91 | },
92 | {
93 | id: 3,
94 | isAdmin: false,
95 | phone: "+91 9910231328",
96 | name: "Parth Sharma",
97 | imgUrl: "https://avatars.githubusercontent.com/u/45586386?v=4"
98 | },
99 | {
100 | id: 4,
101 | isAdmin: false,
102 | phone: "+91 9910231328",
103 | name: "Parth Sharma",
104 | imgUrl: "https://avatars.githubusercontent.com/u/45586386?v=4"
105 | },
106 | ]
107 | },
108 | {
109 | id: 4,
110 | name: "randomrand",
111 | price: 2500,
112 | members: [
113 | {
114 | id: 1,
115 | isAdmin: true,
116 | phone: "+91 9910231328",
117 | name: "Parth Sharma",
118 | imgUrl: "https://avatars.githubusercontent.com/u/45586386?v=4"
119 | },
120 | {
121 | id: 2,
122 | isAdmin: false,
123 | phone: "+91 9910231328",
124 | name: "Parth Sharma",
125 | imgUrl: "https://avatars.githubusercontent.com/u/45586386?v=4"
126 | },
127 | {
128 | id: 3,
129 | isAdmin: false,
130 | phone: "+91 9910231328",
131 | name: "Parth Sharma",
132 | imgUrl: "https://avatars.githubusercontent.com/u/45586386?v=4"
133 | },
134 | {
135 | id: 4,
136 | isAdmin: false,
137 | phone: "+91 9910231328",
138 | name: "Parth Sharma",
139 | imgUrl: "https://avatars.githubusercontent.com/u/45586386?v=4"
140 | },
141 | ]
142 | },
143 | {
144 | id: 5,
145 | name: "nameren",
146 | price: 2500,
147 | members: [
148 | {
149 | id: 1,
150 | isAdmin: true,
151 | phone: "+91 9910231328",
152 | name: "Parth Sharma",
153 | imgUrl: "https://avatars.githubusercontent.com/u/45586386?v=4"
154 | },
155 | {
156 | id: 2,
157 | isAdmin: false,
158 | phone: "+91 9910231328",
159 | name: "Parth Sharma",
160 | imgUrl: "https://avatars.githubusercontent.com/u/45586386?v=4"
161 | },
162 | {
163 | id: 3,
164 | isAdmin: false,
165 | phone: "+91 9910231328",
166 | name: "Parth Sharma",
167 | imgUrl: "https://avatars.githubusercontent.com/u/45586386?v=4"
168 | },
169 | {
170 | id: 4,
171 | isAdmin: false,
172 | phone: "+91 9910231328",
173 | name: "Parth Sharma",
174 | imgUrl: "https://avatars.githubusercontent.com/u/45586386?v=4"
175 | },
176 | ]
177 | },
178 | {
179 | id: 6,
180 | name: "randomgroup",
181 | price: 2500,
182 | members: [
183 | {
184 | id: 1,
185 | isAdmin: true,
186 | phone: "+91 9910231328",
187 | name: "Parth Sharma",
188 | imgUrl: "https://avatars.githubusercontent.com/u/45586386?v=4"
189 | },
190 | {
191 | id: 2,
192 | isAdmin: false,
193 | phone: "+91 9910231328",
194 | name: "Parth Sharma",
195 | imgUrl: "https://avatars.githubusercontent.com/u/45586386?v=4"
196 | },
197 | {
198 | id: 3,
199 | isAdmin: false,
200 | phone: "+91 9910231328",
201 | name: "Parth Sharma",
202 | imgUrl: "https://avatars.githubusercontent.com/u/45586386?v=4"
203 | },
204 | {
205 | id: 4,
206 | isAdmin: false,
207 | phone: "+91 9910231328",
208 | name: "Parth Sharma",
209 | imgUrl: "https://avatars.githubusercontent.com/u/45586386?v=4"
210 | },
211 | ]
212 | }
213 | ],
214 | room: {
215 | id: 69,
216 | name: "randomgroup",
217 | members: [
218 | {
219 | id: 1,
220 | isAdmin: true,
221 | phone: "+91 9910231328",
222 | name: "Parth Sharma",
223 | imgUrl: "https://avatars.githubusercontent.com/u/45586386?v=4"
224 | },
225 | {
226 | id: 2,
227 | isAdmin: false,
228 | phone: "+91 9910231328",
229 | name: "Parth Sharma",
230 | imgUrl: "https://avatars.githubusercontent.com/u/45586386?v=4"
231 | },
232 | {
233 | id: 3,
234 | isAdmin: false,
235 | phone: "+91 9910231328",
236 | name: "Parth Sharma",
237 | imgUrl: "https://avatars.githubusercontent.com/u/45586386?v=4"
238 | },
239 | {
240 | id: 4,
241 | isAdmin: false,
242 | phone: "+91 9910231328",
243 | name: "Parth Sharma",
244 | imgUrl: "https://avatars.githubusercontent.com/u/45586386?v=4"
245 | },
246 | ],
247 | price: 3496
248 | }
249 | };
--------------------------------------------------------------------------------
/client/src/resources/img/acmlogo.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/bitsacm/litecode/389f3f765c7fb0990d1b785fc95a6ec77c860311/client/src/resources/img/acmlogo.png
--------------------------------------------------------------------------------
/client/src/resources/img/keyboard.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/bitsacm/litecode/389f3f765c7fb0990d1b785fc95a6ec77c860311/client/src/resources/img/keyboard.png
--------------------------------------------------------------------------------
/client/src/resources/img/landing.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/bitsacm/litecode/389f3f765c7fb0990d1b785fc95a6ec77c860311/client/src/resources/img/landing.png
--------------------------------------------------------------------------------
/client/src/resources/img/landingImg1.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/bitsacm/litecode/389f3f765c7fb0990d1b785fc95a6ec77c860311/client/src/resources/img/landingImg1.jpg
--------------------------------------------------------------------------------
/client/src/resources/img/landingImg2.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/bitsacm/litecode/389f3f765c7fb0990d1b785fc95a6ec77c860311/client/src/resources/img/landingImg2.jpg
--------------------------------------------------------------------------------
/client/src/resources/img/landingImg3.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/bitsacm/litecode/389f3f765c7fb0990d1b785fc95a6ec77c860311/client/src/resources/img/landingImg3.jpg
--------------------------------------------------------------------------------
/client/src/resources/img/landingImg4.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/bitsacm/litecode/389f3f765c7fb0990d1b785fc95a6ec77c860311/client/src/resources/img/landingImg4.jpg
--------------------------------------------------------------------------------
/client/src/resources/img/landingImg5.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/bitsacm/litecode/389f3f765c7fb0990d1b785fc95a6ec77c860311/client/src/resources/img/landingImg5.png
--------------------------------------------------------------------------------
/client/src/resources/img/logout.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/bitsacm/litecode/389f3f765c7fb0990d1b785fc95a6ec77c860311/client/src/resources/img/logout.png
--------------------------------------------------------------------------------
/client/src/resources/theme.js:
--------------------------------------------------------------------------------
1 | import { extendTheme } from '@chakra-ui/react'
2 |
3 | const theme = {
4 | colors: {
5 | transparent: "transparent",
6 | black: "#030303",
7 | white: "#fff",
8 | liteblue: "#059FC9",
9 | liteblues: "rgba(52, 209, 252, 0.23)",
10 | litegrey: {
11 | 600: "#718096",
12 | 400: "#A0AEC0",
13 | 20: "#F7FAFC",
14 | },
15 | litegold: "#ED8936",
16 | litered: "#E53E3E"
17 | },
18 | fonts: {
19 | primary: "Roboto, sans-serif",
20 | heading: "Roboto, sans-serif",
21 | body: "Roboto, sans-serif"
22 | },
23 | config: {
24 | useSystemColorMode: false,
25 | },
26 | }
27 |
28 |
29 | export default extendTheme(theme);
--------------------------------------------------------------------------------
/client/src/screens/AllRooms.js:
--------------------------------------------------------------------------------
1 | import React, { Fragment, useEffect, useState, useContext, useRef } from 'react'
2 | import AuthContext from '../store/auth.js'
3 | import { Redirect, Link } from 'react-router-dom'
4 | import { DummyData } from '../resources/dummy.js'
5 |
6 | import {
7 | Flex,
8 | Box,
9 | Heading,
10 | InputGroup,
11 | Spinner,
12 | InputLeftElement,
13 | Modal,
14 | useDisclosure,
15 | ModalOverlay,
16 | ModalContent,
17 | ModalHeader,
18 | ModalCloseButton,
19 | ModalBody,
20 | FormControl,
21 | FormLabel,
22 | Input,
23 | ModalFooter,
24 | Button,
25 | IconButton,
26 | } from '@chakra-ui/react'
27 |
28 | import {SearchIcon} from '@chakra-ui/icons'
29 | import RoomCard from '../components/RoomCard.js'
30 |
31 |
32 | const AllRooms = (props) => {
33 |
34 | const [allRooms, setAllRooms] = useState(null);
35 | const [callRooms, csetAllRooms] = useState(null);
36 | const [newf, setNewf] = useState(null);
37 | const [redirect, setRedirect] = useState(null);
38 | const [userInRoom, setUserInRoom] = useState(false);
39 | const [userRoom, setUserRoom] = useState(null);
40 | const [isBanned, setIsBanned] = useState(false);
41 |
42 | const authCtx = useContext(AuthContext);
43 | const token = authCtx.token;
44 |
45 |
46 | useEffect(() => {
47 | myDeets()
48 | getRooms()
49 | }, [redirect])
50 |
51 | const myDeets = () =>{
52 | fetch('https://litecode.bitsacm.in/server/users/me',
53 | {
54 | method: 'GET',
55 | headers: {
56 | 'Content-Type': 'application/json',
57 | 'Authorization': 'Bearer '+token,
58 | }
59 | }
60 | ).then(response =>
61 | response.json().then(data => ({
62 | data: data,
63 | status: response.status
64 | })
65 | ).then(res => {
66 | if(res.data) {
67 | if (res.data.user.inRoom) setUserRoom(res.data.user.roomID);
68 | else setUserRoom(null);
69 |
70 | if (res.data.user.isBanned) setIsBanned(true)
71 | else setIsBanned(false)
72 | }
73 | }
74 | ))
75 | }
76 |
77 |
78 | const getRooms = () => {
79 | fetch('https://litecode.bitsacm.in/server/rooms/',
80 | {
81 | method: 'GET',
82 | headers: {
83 | 'Content-Type': 'application/json',
84 | 'Authorization': 'Bearer '+token,
85 | }
86 | }
87 | ).then(response =>
88 | response.json().then(data => ({
89 | data: data,
90 | status: response.status
91 | })
92 | ).then(res => {
93 | if(res.data){
94 | console.log(res.data);
95 | setAllRooms(res.data)
96 | csetAllRooms(res.data)
97 | } else {
98 | alert("ERROR RETRIEVING CONTENT.");
99 | }
100 | }))
101 | }
102 |
103 | const searchRef = useRef();
104 |
105 | const submitFunction = () => {
106 | const search = searchRef.current.value;
107 |
108 | fetch('https://litecode.bitsacm.in/server/searchRoom/?roomName='+search,
109 | {
110 | method: 'GET',
111 | headers: {
112 | 'Content-Type': 'application/json',
113 | 'Authorization': 'Bearer '+token,
114 | }
115 | }
116 | ).then(response =>
117 | response.json().then(data => ({
118 | data: data,
119 | status: response.status
120 | })
121 | ).then(res => {
122 | if(res.data){
123 | console.log(res.data);
124 | setAllRooms(res.data)
125 | } else {
126 | alert("ERROR RETRIEVING CONTENT.");
127 | }
128 | }))
129 | }
130 |
131 |
132 | const matchesQuery = (arr, query) => {
133 | return arr.filter(el => (el.roomID).toLowerCase().indexOf(query.toLowerCase()) !== -1)
134 | }
135 |
136 | const handleChange = () => {
137 | setAllRooms(matchesQuery(callRooms, searchRef.current.value))
138 | }
139 |
140 | const updateRedirect = () => {
141 | setRedirect("pls redirect lol")
142 | }
143 |
144 | return(
145 |
146 | {redirect===null ?
147 |
148 | {allRooms ?
149 |
150 |
151 |
152 |
153 |
161 |
162 | }
166 | />
167 |
168 |
180 |
181 | {/* }
185 | ml="5px"
186 | bg="#EDF2F7"
187 | border="none"
188 | color="litegrey.400"
189 | /> */}
190 |
191 |
192 |
193 |
194 |
195 |
196 | {allRooms ?
197 |
198 | {allRooms.map((room, index)=>(
199 |
200 | ))}
201 | :null}
202 |
203 | : }
204 |
205 | : }
206 |
207 |
208 | )
209 | }
210 |
211 | const InitialFocus = (props) => {
212 | const { isOpen, onOpen, onClose } = useDisclosure()
213 |
214 | const initialRef = React.useRef()
215 | const [nameRef, setNameRef] = useState("")
216 | const finalRef = React.useRef()
217 |
218 | const authCtx = useContext(AuthContext);
219 | const token = authCtx.token;
220 |
221 | const handleChange = (event) => setNameRef(event.target.value)
222 |
223 |
224 | const submitHandler = (e) => {
225 | e.preventDefault();
226 |
227 | const name = nameRef;
228 |
229 | fetch('https://litecode.bitsacm.in/server/createRoom?roomName='+name,
230 | {
231 | method: 'POST',
232 | headers: {
233 | 'Content-Type': 'application/json',
234 | 'Authorization': 'Bearer '+token,
235 | }
236 | }
237 | ).then(response =>
238 | response.json().then(data => ({
239 | data: data,
240 | status: response.status
241 | })
242 | ).then(res => {
243 | if(res.data){
244 | console.log("created")
245 | props.setRedirect();
246 | } else {
247 | alert("Authentication failed. Please try again.");
248 | }
249 | }))
250 | }
251 |
252 |
253 |
254 | return (
255 | <>
256 |
269 |
270 |
276 |
277 |
278 | Create a new room
279 |
280 |
281 |
282 | Add a Room Name
283 | ') || nameRef.includes('<') || nameRef.includes('|') || nameRef.includes('$') || nameRef.includes('#') || nameRef.includes('*') || nameRef.includes('@')}
288 | errorBorderColor="red.300"
289 | placeholder="Litecode by ACM"
290 | />
291 |
292 |
293 |
294 |
295 |
309 |
310 |
311 |
312 | >
313 | )
314 | }
315 |
316 |
317 |
318 |
319 | export default AllRooms;
--------------------------------------------------------------------------------
/client/src/screens/Landing.js:
--------------------------------------------------------------------------------
1 | import React, { Fragment, useContext } from 'react'
2 | import {
3 | HStack,
4 | Heading,
5 | Text,
6 | Spacer,
7 | Button,
8 | Image,
9 | Box,
10 | Flex
11 | } from '@chakra-ui/react'
12 |
13 | import {
14 | Link
15 | } from 'react-router-dom'
16 |
17 | import AuthContext from '../store/auth'
18 |
19 | import { Star } from 'react-github-buttons'
20 | import GitHubButton from 'react-github-btn'
21 |
22 | import landingImg from '../resources/img/landing.png'
23 | import landingImg1 from '../resources/img/landingImg1.jpg'
24 | import landingImg2 from '../resources/img/landingImg2.jpg'
25 | import landingImg3 from '../resources/img/landingImg3.jpg'
26 | import landingImg4 from '../resources/img/landingImg4.jpg'
27 | import landingImg5 from '../resources/img/landingImg5.png'
28 |
29 | const Landing = () => {
30 |
31 | const authCtx = useContext(AuthContext);
32 | const isLoggedIn = authCtx.isLoggedIn;
33 |
34 | const login = () => {
35 | fetch('http://localhost:3000/auth/google/',
36 | {
37 | method: 'GET',
38 | headers: {
39 | 'Content-Type': 'application/json',
40 | }
41 | }
42 | ).then(response =>
43 | response.json().then(data => ({
44 | data: data,
45 | status: response.status
46 | })
47 | ).then(res => {
48 | console.log(res)
49 | if(res.data){
50 | const url = res.data.googleLoginUrl
51 | window.location.href = url;
52 | } else {
53 | alert("Error handling.")
54 | }
55 | }))
56 | }
57 |
58 | return(
59 |
67 |
73 |
74 | Share Leetcode
80 |
81 | premium accounts.
86 |
87 | Discover people willing to split the cost for LeetCode Premium accounts with you.
94 |
95 |
96 |
97 |
98 |
99 |
100 | +
101 |
102 |
103 |
104 |
105 |
121 |
122 | Sign up with your BITS email.
130 |
131 |
137 | Star this on GitHub!
138 |
139 |
140 |
141 |
142 |
147 |
152 |
153 |
154 | )
155 | }
156 |
157 | export default Landing;
--------------------------------------------------------------------------------
/client/src/screens/Room.js:
--------------------------------------------------------------------------------
1 | import React, { Fragment, useEffect, useState, useContext } from 'react'
2 | import AuthContext from '../store/auth.js'
3 | import { useHistory } from 'react-router-dom'
4 | import UserCard from '../components/UserCard.js'
5 | import {
6 | Flex,
7 | useDisclosure,
8 | Box,
9 | Heading,
10 | Text,
11 | Button,
12 | ButtonGroup,
13 | Spinner,
14 | Popover,
15 | PopoverTrigger,
16 | PopoverContent,
17 | PopoverHeader,
18 | PopoverBody,
19 | PopoverFooter,
20 | PopoverArrow,
21 | PopoverCloseButton,
22 | Modal,
23 | ModalOverlay,
24 | ModalContent,
25 | ModalHeader,
26 | ModalFooter,
27 | ModalBody,
28 | ModalCloseButton,
29 | } from '@chakra-ui/react'
30 | import { DummyData } from '../resources/dummy.js'
31 | import { Redirect, Link } from 'react-router-dom'
32 |
33 | const Room = () => {
34 | const [userIsAdmin, setUserIsAdmin] = useState(false);
35 | const [userInfo, setUserInfo] = useState(null);
36 | const [roomDetails, updateRoomDetails] = useState(null)
37 | const [redirect, setRedirect] = useState(null);
38 |
39 | const authCtx = useContext(AuthContext);
40 | const token = authCtx.token;
41 |
42 | const history = useHistory();
43 |
44 | useEffect(() => {
45 | loadRoom()
46 | }, [redirect])
47 |
48 |
49 | const loadRoom = () => {
50 | fetch('https://litecode.bitsacm.in/server/users/me',
51 | {
52 | method: 'GET',
53 | headers: {
54 | 'Content-Type': 'application/json',
55 | 'Authorization': 'Bearer '+token,
56 | }
57 | }
58 | ).then(response =>
59 | response.json().then(data => ({
60 | data: data,
61 | status: response.status
62 | })
63 | ).then(res => {
64 | // console.log(res.data)
65 | if(res.data.user.inRoom){
66 | setUserInfo(res.data)
67 | const roomID = res.data.user.roomID
68 | getRoomDetails(roomID);
69 | } else {
70 | updateRedirect()
71 | }
72 | if (redirect != null) alert("Please join a room.")
73 | }))
74 | }
75 |
76 | const getRoomDetails = (roomID) => {
77 | fetch('https://litecode.bitsacm.in/server/room/'+roomID,
78 | {
79 | method: 'GET',
80 | headers: {
81 | 'Content-Type': 'application/json',
82 | 'Authorization': 'Bearer '+token,
83 | }
84 | }
85 | ).then(response =>
86 | response.json().then(data => ({
87 | data: data,
88 | status: response.status
89 | })
90 | ).then(res => {
91 | if(res.data){
92 | updateRoomDetails(res.data)
93 | } else {
94 | alert("ERROR POSTING CONTENT.");
95 | }
96 | }))
97 | }
98 |
99 | const lockRoom = () => {
100 | fetch('https://litecode.bitsacm.in/server/lock',
101 | {
102 | method: 'PATCH',
103 | headers: {
104 | 'Content-Type': 'application/json',
105 | 'Authorization': 'Bearer '+token,
106 | }
107 | }
108 | ).then(response =>
109 | response.json().then(data => ({
110 | data: data,
111 | status: response.status
112 | })
113 | ).then(res => {
114 | if(res.data){
115 | loadRoom();
116 | } else {
117 | alert("ERROR POSTING CONTENT.");
118 | }
119 | }))
120 | }
121 |
122 | const leaveRoom = () => {
123 | fetch('https://litecode.bitsacm.in/server/leaveRoom',
124 | {
125 | method: 'POST',
126 | headers: {
127 | 'Content-Type': 'application/json',
128 | 'Authorization': 'Bearer '+token,
129 | }
130 | }
131 | ).then(response =>
132 | response.json().then(data => ({
133 | data: data,
134 | status: response.status
135 | })
136 | ).then(res => {
137 | if(res.data){
138 | updateRedirect()
139 | } else {
140 | alert("ERROR POSTING CONTENT.");
141 | }
142 | }))
143 | }
144 |
145 | const updateRedirect = () => {
146 | setRedirect("non-null")
147 | }
148 |
149 | const { isOpen, onOpen, onClose } = useDisclosure()
150 |
151 |
152 | return(
153 |
154 | { redirect===null ?
155 |
156 | {roomDetails ?
157 |
158 | {roomDetails.room.users.length===1 ?
159 | 🤙 Invite a friend! :
165 | 👏 It's a party!
171 | }
172 |
173 |
174 |
175 |
176 |
185 | {roomDetails.room.users.map((user, index)=>(
186 |
195 | ))}
196 |
197 |
198 |
206 |
213 |
214 | Room Name
219 | {roomDetails.room.roomID}
224 |
225 |
226 |
230 |
231 | Members
235 | {roomDetails.room.users.length} / 4
239 |
240 |
241 |
242 | Per Member
246 | ₹ {roomDetails.room.costPerMember}
251 |
252 |
253 |
254 |
260 | {(roomDetails.room.roomAdmin === userInfo.user._id)
261 | ?
262 |
263 | {roomDetails.room.roomLocked ?
264 | <>
265 |
279 |
280 |
281 |
282 |
283 | It's time to start coding.
284 |
285 |
286 |
287 | Hey there, @Admin! This is now your final Litecode group. You should be able to see everyone's mobile numbers — just create a WhatsApp group with all of them, split the bill, make the purchase, and start Litecoding!
288 |
289 |
290 |
291 |
292 |
295 |
296 |
297 |
298 | >
299 | :
300 |
301 | }
302 |
303 | : null}
304 |
305 |
306 |
307 |
308 | :}
309 | : }
310 |
311 | )
312 | }
313 |
314 |
315 | const ControlledUsage = (props) => {
316 | const [isOpen, setIsOpen] = React.useState(false)
317 | const open = () => setIsOpen(!isOpen)
318 | const close = () => setIsOpen(false)
319 | return (
320 | <>
321 | { props.count > 1 ?
322 |
323 |
336 |
343 |
344 | Confirmation
345 |
346 |
347 |
348 | Are you sure you want to leave? You won't be able to join more rooms for 2 days.
349 |
350 |
351 |
352 |
353 |
364 |
365 |
366 |
367 |
368 |
369 | :
370 |
371 |
384 |
385 | }
386 | >
387 | )
388 | }
389 |
390 |
391 | const ControlledUsageS2 = (props) => {
392 | const [isOpen, setIsOpen] = React.useState(false)
393 | const open = () => setIsOpen(!isOpen)
394 | const close = () => setIsOpen(false)
395 | return (
396 | <>
397 |
398 |
412 |
419 |
420 | Confirmation
421 |
422 |
423 |
424 | Are you sure you want to lock the room? This will be your final Litecode group.
425 |
426 |
427 |
428 |
429 |
440 |
441 |
442 |
443 |
444 |
445 | >
446 | )
447 | }
448 |
449 |
450 |
451 | export default Room;
--------------------------------------------------------------------------------
/client/src/store/auth.js:
--------------------------------------------------------------------------------
1 | import React, { useState } from 'react'
2 | import Cookies from 'js-cookie'
3 |
4 | const AuthContext = React.createContext({
5 | token: '',
6 | isLoggedIn: false,
7 | login: () => {},
8 | // logout: () => {},
9 | });
10 |
11 | export const AuthContextProvider = (props) => {
12 | const initialToken = Cookies.get('jwt')
13 |
14 | const [token, setToken] = useState(initialToken);
15 | const userIsLoggedIn = !!token;
16 |
17 | const loginHandler = () => {
18 | token = Cookies.get('jwt')
19 | setToken(token);
20 | }
21 |
22 | // const logoutHandler = () => {
23 | // Cookies.remove('jwt')
24 | // }
25 |
26 | const contextValue = {
27 | token: token,
28 | isLoggedIn: userIsLoggedIn,
29 | login: loginHandler,
30 | // logout: logoutHandler
31 | }
32 |
33 | return(
34 |
35 | {props.children}
36 |
37 | )
38 | };
39 |
40 | export default AuthContext;
--------------------------------------------------------------------------------
/docker-compose.yaml:
--------------------------------------------------------------------------------
1 |
2 | version: "3"
3 |
4 | services:
5 | server:
6 | container_name: "litecode_server"
7 | restart: unless-stopped
8 | build: ./server
9 | env_file: ./server/config.env
10 | depends_on:
11 | - db
12 | ports:
13 | - 3000:3000
14 | expose:
15 | - 3000
16 | volumes:
17 | - ./server:/home/node/app
18 |
19 | client:
20 | container_name: 'litecode_client'
21 | build: ./client
22 | env_file: ./client/config.env
23 | ports:
24 | - 3001:3000
25 | expose:
26 | - 3000
27 | volumes:
28 | - react_build:/app/build
29 | depends_on:
30 | - server
31 |
32 | db:
33 | container_name: 'litecode_mongodb'
34 | image: mongo:latest
35 | environment:
36 | MONGO_INITDB_ROOT_USERNAME: litecode
37 | MONGO_INITDB_ROOT_PASSWORD: litecode123
38 | ports:
39 | - 27017:27017
40 | volumes:
41 | - mongodb_data_container:/data/db
42 |
43 | nginx:
44 | container_name: "litecode_nginx"
45 | image: nginx:mainline-alpine
46 | restart: unless-stopped
47 | ports:
48 | - "1337:80"
49 | volumes:
50 | - ./nginx:/etc/nginx/conf.d
51 | - react_build:/var/www/html/
52 | depends_on:
53 | - server
54 | - client
55 |
56 | volumes:
57 | mongodb_data_container:
58 | react_build:
59 |
--------------------------------------------------------------------------------
/nginx/nginx.conf:
--------------------------------------------------------------------------------
1 | upstream litecode {
2 | server litecode_server:3000;
3 | }
4 |
5 | server {
6 | listen 80;
7 | listen [::]:80;
8 |
9 | location / {
10 | root /var/www/html/;
11 | index index.html index.htm;
12 | try_files $uri $uri/ /index.html;
13 | }
14 |
15 | location /server {
16 | rewrite ^/server/?(.*)$ /$1 break;
17 | proxy_pass http://litecode;
18 | proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
19 | proxy_set_header Host $host;
20 | proxy_redirect off;
21 | }
22 | }
--------------------------------------------------------------------------------
/package-lock.json:
--------------------------------------------------------------------------------
1 | {
2 | "lockfileVersion": 1
3 | }
4 |
--------------------------------------------------------------------------------
/server/.dockerignore:
--------------------------------------------------------------------------------
1 | node_modules
2 | build
3 | .dockerignore
4 | Dockerfile
5 |
--------------------------------------------------------------------------------
/server/.gitignore:
--------------------------------------------------------------------------------
1 | node_modules
2 | config.env
3 | config
--------------------------------------------------------------------------------
/server/Dockerfile:
--------------------------------------------------------------------------------
1 |
2 | FROM node:13.12.0-alpine
3 |
4 | WORKDIR /app
5 |
6 | ENV PATH /app/node_modules/.bin:$PATH
7 |
8 | COPY package.json ./
9 | COPY package-lock.json ./
10 | RUN npm install --silent
11 |
12 | COPY . ./
13 |
14 | CMD ["npm", "start"]
--------------------------------------------------------------------------------
/server/README.md:
--------------------------------------------------------------------------------
1 | # Backend for the Litecode web application
--------------------------------------------------------------------------------
/server/package-lock.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "server",
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/bson": {
23 | "version": "4.0.3",
24 | "resolved": "https://registry.npmjs.org/@types/bson/-/bson-4.0.3.tgz",
25 | "integrity": "sha512-mVRvYnTOZJz3ccpxhr3wgxVmSeiYinW+zlzQz3SXWaJmD1DuL05Jeq7nKw3SnbKmbleW5qrLG5vdyWe/A9sXhw==",
26 | "requires": {
27 | "@types/node": "*"
28 | }
29 | },
30 | "@types/mongodb": {
31 | "version": "3.6.17",
32 | "resolved": "https://registry.npmjs.org/@types/mongodb/-/mongodb-3.6.17.tgz",
33 | "integrity": "sha512-9hhgvYPdC5iHyyksPcKCu45gfaAIPQHKHGdvNXu4582DmOZX3wrUJIJPT40o4G1oTKPgpMMFqZglOTjhnYoF+A==",
34 | "requires": {
35 | "@types/bson": "*",
36 | "@types/node": "*"
37 | }
38 | },
39 | "@types/node": {
40 | "version": "15.6.1",
41 | "resolved": "https://registry.npmjs.org/@types/node/-/node-15.6.1.tgz",
42 | "integrity": "sha512-7EIraBEyRHEe7CH+Fm1XvgqU6uwZN8Q7jppJGcqjROMT29qhAuuOxYB1uEY5UMYQKEmA5D+5tBnhdaPXSsLONA=="
43 | },
44 | "abbrev": {
45 | "version": "1.1.1",
46 | "resolved": "https://registry.npmjs.org/abbrev/-/abbrev-1.1.1.tgz",
47 | "integrity": "sha512-nne9/IiQ/hzIhY6pdDnbBtz7DjPTKrY00P/zvPSm5pOFkl6xuGrGnXn/VtTNNfNtAfZ9/1RtehkszU9qcTii0Q==",
48 | "dev": true
49 | },
50 | "abort-controller": {
51 | "version": "3.0.0",
52 | "resolved": "https://registry.npmjs.org/abort-controller/-/abort-controller-3.0.0.tgz",
53 | "integrity": "sha512-h8lQ8tacZYnR3vNQTgibj+tODHI5/+l06Au2Pcriv/Gmet0eaj4TwWH41sO9wnHDiQsEj19q0drzdWdeAHtweg==",
54 | "requires": {
55 | "event-target-shim": "^5.0.0"
56 | }
57 | },
58 | "accepts": {
59 | "version": "1.3.7",
60 | "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.7.tgz",
61 | "integrity": "sha512-Il80Qs2WjYlJIBNzNkK6KYqlVMTbZLXgHx2oT0pU/fjRHyEp+PEfEPY0R3WCwAGVOtauxh1hOxNgIf5bv7dQpA==",
62 | "requires": {
63 | "mime-types": "~2.1.24",
64 | "negotiator": "0.6.2"
65 | }
66 | },
67 | "agent-base": {
68 | "version": "6.0.2",
69 | "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-6.0.2.tgz",
70 | "integrity": "sha512-RZNwNclF7+MS/8bDg70amg32dyeZGZxiDuQmZxKLAlQjr3jGyLx+4Kkk58UO7D2QdgFIQCovuSuZESne6RG6XQ==",
71 | "requires": {
72 | "debug": "4"
73 | },
74 | "dependencies": {
75 | "debug": {
76 | "version": "4.3.1",
77 | "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.1.tgz",
78 | "integrity": "sha512-doEwdvm4PCeK4K3RQN2ZC2BYUBaxwLARCqZmMjtF8a51J2Rb0xpVloFRnCODwqjpwnAoao4pelN8l3RJdv3gRQ==",
79 | "requires": {
80 | "ms": "2.1.2"
81 | }
82 | },
83 | "ms": {
84 | "version": "2.1.2",
85 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz",
86 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w=="
87 | }
88 | }
89 | },
90 | "ansi-align": {
91 | "version": "3.0.0",
92 | "resolved": "https://registry.npmjs.org/ansi-align/-/ansi-align-3.0.0.tgz",
93 | "integrity": "sha512-ZpClVKqXN3RGBmKibdfWzqCY4lnjEuoNzU5T0oEFpfd/z5qJHVarukridD4juLO2FXMiwUQxr9WqQtaYa8XRYw==",
94 | "dev": true,
95 | "requires": {
96 | "string-width": "^3.0.0"
97 | },
98 | "dependencies": {
99 | "ansi-regex": {
100 | "version": "4.1.0",
101 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz",
102 | "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==",
103 | "dev": true
104 | },
105 | "emoji-regex": {
106 | "version": "7.0.3",
107 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-7.0.3.tgz",
108 | "integrity": "sha512-CwBLREIQ7LvYFB0WyRvwhq5N5qPhc6PMjD6bYggFlI5YyDgl+0vxq5VHbMOFqLg7hfWzmu8T5Z1QofhmTIhItA==",
109 | "dev": true
110 | },
111 | "is-fullwidth-code-point": {
112 | "version": "2.0.0",
113 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz",
114 | "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=",
115 | "dev": true
116 | },
117 | "string-width": {
118 | "version": "3.1.0",
119 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz",
120 | "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==",
121 | "dev": true,
122 | "requires": {
123 | "emoji-regex": "^7.0.1",
124 | "is-fullwidth-code-point": "^2.0.0",
125 | "strip-ansi": "^5.1.0"
126 | }
127 | },
128 | "strip-ansi": {
129 | "version": "5.2.0",
130 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz",
131 | "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==",
132 | "dev": true,
133 | "requires": {
134 | "ansi-regex": "^4.1.0"
135 | }
136 | }
137 | }
138 | },
139 | "ansi-regex": {
140 | "version": "5.0.0",
141 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.0.tgz",
142 | "integrity": "sha512-bY6fj56OUQ0hU1KjFNDQuJFezqKdrAyFdIevADiqrWHwSlbmBNMHp5ak2f40Pm8JTFyM2mqxkG6ngkHO11f/lg==",
143 | "dev": true
144 | },
145 | "ansi-styles": {
146 | "version": "4.3.0",
147 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz",
148 | "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==",
149 | "dev": true,
150 | "requires": {
151 | "color-convert": "^2.0.1"
152 | }
153 | },
154 | "anymatch": {
155 | "version": "3.1.2",
156 | "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.2.tgz",
157 | "integrity": "sha512-P43ePfOAIupkguHUycrc4qJ9kz8ZiuOUijaETwX7THt0Y/GNK7v0aa8rY816xWjZ7rJdA5XdMcpVFTKMq+RvWg==",
158 | "dev": true,
159 | "requires": {
160 | "normalize-path": "^3.0.0",
161 | "picomatch": "^2.0.4"
162 | }
163 | },
164 | "array-flatten": {
165 | "version": "1.1.1",
166 | "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz",
167 | "integrity": "sha1-ml9pkFGx5wczKPKgCJaLZOopVdI="
168 | },
169 | "arrify": {
170 | "version": "2.0.1",
171 | "resolved": "https://registry.npmjs.org/arrify/-/arrify-2.0.1.tgz",
172 | "integrity": "sha512-3duEwti880xqi4eAMN8AyR4a0ByT90zoYdLlevfrvU43vb0YZwZVfxOgxWrLXXXpyugL0hNZc9G6BiB5B3nUug=="
173 | },
174 | "axios": {
175 | "version": "0.21.1",
176 | "resolved": "https://registry.npmjs.org/axios/-/axios-0.21.1.tgz",
177 | "integrity": "sha512-dKQiRHxGD9PPRIUNIWvZhPTPpl1rf/OxTYKsqKUDjBwYylTvV7SjSHJb9ratfyzM6wCdLCOYLzs73qpg5c4iGA==",
178 | "requires": {
179 | "follow-redirects": "^1.10.0"
180 | }
181 | },
182 | "balanced-match": {
183 | "version": "1.0.2",
184 | "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz",
185 | "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==",
186 | "dev": true
187 | },
188 | "base64-js": {
189 | "version": "1.5.1",
190 | "resolved": "https://registry.npmjs.org/base64-js/-/base64-js-1.5.1.tgz",
191 | "integrity": "sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA=="
192 | },
193 | "bcryptjs": {
194 | "version": "2.4.3",
195 | "resolved": "https://registry.npmjs.org/bcryptjs/-/bcryptjs-2.4.3.tgz",
196 | "integrity": "sha1-mrVie5PmBiH/fNrF2pczAn3x0Ms="
197 | },
198 | "bignumber.js": {
199 | "version": "9.0.1",
200 | "resolved": "https://registry.npmjs.org/bignumber.js/-/bignumber.js-9.0.1.tgz",
201 | "integrity": "sha512-IdZR9mh6ahOBv/hYGiXyVuyCetmGJhtYkqLBpTStdhEGjegpPlUawydyaF3pbIOFynJTpllEs+NP+CS9jKFLjA=="
202 | },
203 | "binary-extensions": {
204 | "version": "2.2.0",
205 | "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.2.0.tgz",
206 | "integrity": "sha512-jDctJ/IVQbZoJykoeHbhXpOlNBqGNcwXJKJog42E5HDPUwQTSdjCHdihjj0DlnheQ7blbT6dHOafNAiS8ooQKA==",
207 | "dev": true
208 | },
209 | "bl": {
210 | "version": "2.2.1",
211 | "resolved": "https://registry.npmjs.org/bl/-/bl-2.2.1.tgz",
212 | "integrity": "sha512-6Pesp1w0DEX1N550i/uGV/TqucVL4AM/pgThFSN/Qq9si1/DF9aIHs1BxD8V/QU0HoeHO6cQRTAuYnLPKq1e4g==",
213 | "requires": {
214 | "readable-stream": "^2.3.5",
215 | "safe-buffer": "^5.1.1"
216 | }
217 | },
218 | "bluebird": {
219 | "version": "3.5.1",
220 | "resolved": "https://registry.npmjs.org/bluebird/-/bluebird-3.5.1.tgz",
221 | "integrity": "sha512-MKiLiV+I1AA596t9w1sQJ8jkiSr5+ZKi0WKrYGUn6d1Fx+Ij4tIj+m2WMQSGczs5jZVxV339chE8iwk6F64wjA=="
222 | },
223 | "body-parser": {
224 | "version": "1.19.0",
225 | "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.19.0.tgz",
226 | "integrity": "sha512-dhEPs72UPbDnAQJ9ZKMNTP6ptJaionhP5cBb541nXPlW60Jepo9RV/a4fX4XWW9CuFNK22krhrj1+rgzifNCsw==",
227 | "requires": {
228 | "bytes": "3.1.0",
229 | "content-type": "~1.0.4",
230 | "debug": "2.6.9",
231 | "depd": "~1.1.2",
232 | "http-errors": "1.7.2",
233 | "iconv-lite": "0.4.24",
234 | "on-finished": "~2.3.0",
235 | "qs": "6.7.0",
236 | "raw-body": "2.4.0",
237 | "type-is": "~1.6.17"
238 | }
239 | },
240 | "boxen": {
241 | "version": "4.2.0",
242 | "resolved": "https://registry.npmjs.org/boxen/-/boxen-4.2.0.tgz",
243 | "integrity": "sha512-eB4uT9RGzg2odpER62bBwSLvUeGC+WbRjjyyFhGsKnc8wp/m0+hQsMUvUe3H2V0D5vw0nBdO1hCJoZo5mKeuIQ==",
244 | "dev": true,
245 | "requires": {
246 | "ansi-align": "^3.0.0",
247 | "camelcase": "^5.3.1",
248 | "chalk": "^3.0.0",
249 | "cli-boxes": "^2.2.0",
250 | "string-width": "^4.1.0",
251 | "term-size": "^2.1.0",
252 | "type-fest": "^0.8.1",
253 | "widest-line": "^3.1.0"
254 | }
255 | },
256 | "brace-expansion": {
257 | "version": "1.1.11",
258 | "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz",
259 | "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==",
260 | "dev": true,
261 | "requires": {
262 | "balanced-match": "^1.0.0",
263 | "concat-map": "0.0.1"
264 | }
265 | },
266 | "braces": {
267 | "version": "3.0.2",
268 | "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz",
269 | "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==",
270 | "dev": true,
271 | "requires": {
272 | "fill-range": "^7.0.1"
273 | }
274 | },
275 | "bson": {
276 | "version": "1.1.6",
277 | "resolved": "https://registry.npmjs.org/bson/-/bson-1.1.6.tgz",
278 | "integrity": "sha512-EvVNVeGo4tHxwi8L6bPj3y3itEvStdwvvlojVxxbyYfoaxJ6keLgrTuKdyfEAszFK+H3olzBuafE0yoh0D1gdg=="
279 | },
280 | "buffer-equal-constant-time": {
281 | "version": "1.0.1",
282 | "resolved": "https://registry.npmjs.org/buffer-equal-constant-time/-/buffer-equal-constant-time-1.0.1.tgz",
283 | "integrity": "sha1-+OcRMvf/5uAaXJaXpMbz5I1cyBk="
284 | },
285 | "bytes": {
286 | "version": "3.1.0",
287 | "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.0.tgz",
288 | "integrity": "sha512-zauLjrfCG+xvoyaqLoV8bLVXXNGC4JqlxFCutSDWA6fJrTo2ZuvLYTqZ7aHBLZSMOopbzwv8f+wZcVzfVTI2Dg=="
289 | },
290 | "cacheable-request": {
291 | "version": "6.1.0",
292 | "resolved": "https://registry.npmjs.org/cacheable-request/-/cacheable-request-6.1.0.tgz",
293 | "integrity": "sha512-Oj3cAGPCqOZX7Rz64Uny2GYAZNliQSqfbePrgAQ1wKAihYmCUnraBtJtKcGR4xz7wF+LoJC+ssFZvv5BgF9Igg==",
294 | "dev": true,
295 | "requires": {
296 | "clone-response": "^1.0.2",
297 | "get-stream": "^5.1.0",
298 | "http-cache-semantics": "^4.0.0",
299 | "keyv": "^3.0.0",
300 | "lowercase-keys": "^2.0.0",
301 | "normalize-url": "^4.1.0",
302 | "responselike": "^1.0.2"
303 | },
304 | "dependencies": {
305 | "get-stream": {
306 | "version": "5.2.0",
307 | "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-5.2.0.tgz",
308 | "integrity": "sha512-nBF+F1rAZVCu/p7rjzgA+Yb4lfYXrpl7a6VmJrU8wF9I1CKvP/QwPNZHnOlwbTkY6dvtFIzFMSyQXbLoTQPRpA==",
309 | "dev": true,
310 | "requires": {
311 | "pump": "^3.0.0"
312 | }
313 | },
314 | "lowercase-keys": {
315 | "version": "2.0.0",
316 | "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-2.0.0.tgz",
317 | "integrity": "sha512-tqNXrS78oMOE73NMxK4EMLQsQowWf8jKooH9g7xPavRT706R6bkQJ6DY2Te7QukaZsulxa30wQ7bk0pm4XiHmA==",
318 | "dev": true
319 | }
320 | }
321 | },
322 | "camelcase": {
323 | "version": "5.3.1",
324 | "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz",
325 | "integrity": "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==",
326 | "dev": true
327 | },
328 | "chalk": {
329 | "version": "3.0.0",
330 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-3.0.0.tgz",
331 | "integrity": "sha512-4D3B6Wf41KOYRFdszmDqMCGq5VV/uMAB273JILmO+3jAlh8X4qDtdtgCR3fxtbLEMzSx22QdhnDcJvu2u1fVwg==",
332 | "dev": true,
333 | "requires": {
334 | "ansi-styles": "^4.1.0",
335 | "supports-color": "^7.1.0"
336 | },
337 | "dependencies": {
338 | "has-flag": {
339 | "version": "4.0.0",
340 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz",
341 | "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==",
342 | "dev": true
343 | },
344 | "supports-color": {
345 | "version": "7.2.0",
346 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz",
347 | "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==",
348 | "dev": true,
349 | "requires": {
350 | "has-flag": "^4.0.0"
351 | }
352 | }
353 | }
354 | },
355 | "chokidar": {
356 | "version": "3.5.1",
357 | "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.5.1.tgz",
358 | "integrity": "sha512-9+s+Od+W0VJJzawDma/gvBNQqkTiqYTWLuZoyAsivsI4AaWTCzHG06/TMjsf1cYe9Cb97UCEhjz7HvnPk2p/tw==",
359 | "dev": true,
360 | "requires": {
361 | "anymatch": "~3.1.1",
362 | "braces": "~3.0.2",
363 | "fsevents": "~2.3.1",
364 | "glob-parent": "~5.1.0",
365 | "is-binary-path": "~2.1.0",
366 | "is-glob": "~4.0.1",
367 | "normalize-path": "~3.0.0",
368 | "readdirp": "~3.5.0"
369 | }
370 | },
371 | "ci-info": {
372 | "version": "2.0.0",
373 | "resolved": "https://registry.npmjs.org/ci-info/-/ci-info-2.0.0.tgz",
374 | "integrity": "sha512-5tK7EtrZ0N+OLFMthtqOj4fI2Jeb88C4CAZPu25LDVUgXJ0A3Js4PMGqrn0JU1W0Mh1/Z8wZzYPxqUrXeBboCQ==",
375 | "dev": true
376 | },
377 | "cli-boxes": {
378 | "version": "2.2.1",
379 | "resolved": "https://registry.npmjs.org/cli-boxes/-/cli-boxes-2.2.1.tgz",
380 | "integrity": "sha512-y4coMcylgSCdVinjiDBuR8PCC2bLjyGTwEmPb9NHR/QaNU6EUOXcTY/s6VjGMD6ENSEaeQYHCY0GNGS5jfMwPw==",
381 | "dev": true
382 | },
383 | "clone-response": {
384 | "version": "1.0.2",
385 | "resolved": "https://registry.npmjs.org/clone-response/-/clone-response-1.0.2.tgz",
386 | "integrity": "sha1-0dyXOSAxTfZ/vrlCI7TuNQI56Ws=",
387 | "dev": true,
388 | "requires": {
389 | "mimic-response": "^1.0.0"
390 | }
391 | },
392 | "color-convert": {
393 | "version": "2.0.1",
394 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz",
395 | "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==",
396 | "dev": true,
397 | "requires": {
398 | "color-name": "~1.1.4"
399 | }
400 | },
401 | "color-name": {
402 | "version": "1.1.4",
403 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz",
404 | "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==",
405 | "dev": true
406 | },
407 | "commander": {
408 | "version": "4.1.1",
409 | "resolved": "https://registry.npmjs.org/commander/-/commander-4.1.1.tgz",
410 | "integrity": "sha512-NOKm8xhkzAjzFx8B2v5OAHT+u5pRQc2UCa2Vq9jYL/31o2wi9mxBA7LIFs3sV5VSC49z6pEhfbMULvShKj26WA==",
411 | "dev": true
412 | },
413 | "concat-map": {
414 | "version": "0.0.1",
415 | "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz",
416 | "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=",
417 | "dev": true
418 | },
419 | "configstore": {
420 | "version": "5.0.1",
421 | "resolved": "https://registry.npmjs.org/configstore/-/configstore-5.0.1.tgz",
422 | "integrity": "sha512-aMKprgk5YhBNyH25hj8wGt2+D52Sw1DRRIzqBwLp2Ya9mFmY8KPvvtvmna8SxVR9JMZ4kzMD68N22vlaRpkeFA==",
423 | "dev": true,
424 | "requires": {
425 | "dot-prop": "^5.2.0",
426 | "graceful-fs": "^4.1.2",
427 | "make-dir": "^3.0.0",
428 | "unique-string": "^2.0.0",
429 | "write-file-atomic": "^3.0.0",
430 | "xdg-basedir": "^4.0.0"
431 | }
432 | },
433 | "content-disposition": {
434 | "version": "0.5.3",
435 | "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.3.tgz",
436 | "integrity": "sha512-ExO0774ikEObIAEV9kDo50o+79VCUdEB6n6lzKgGwupcVeRlhrj3qGAfwq8G6uBJjkqLrhT0qEYFcWng8z1z0g==",
437 | "requires": {
438 | "safe-buffer": "5.1.2"
439 | }
440 | },
441 | "content-type": {
442 | "version": "1.0.4",
443 | "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.4.tgz",
444 | "integrity": "sha512-hIP3EEPs8tB9AT1L+NUqtwOAps4mk2Zob89MWXMHjHWg9milF/j4osnnQLXBCBFBk/tvIG/tUc9mOUJiPBhPXA=="
445 | },
446 | "cookie": {
447 | "version": "0.4.0",
448 | "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.4.0.tgz",
449 | "integrity": "sha512-+Hp8fLp57wnUSt0tY0tHEXh4voZRDnoIrZPqlo3DPiI4y9lwg/jqx+1Om94/W6ZaPDOUbnjOt/99w66zk+l1Xg=="
450 | },
451 | "cookie-signature": {
452 | "version": "1.0.6",
453 | "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz",
454 | "integrity": "sha1-4wOogrNCzD7oylE6eZmXNNqzriw="
455 | },
456 | "core-util-is": {
457 | "version": "1.0.2",
458 | "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.2.tgz",
459 | "integrity": "sha1-tf1UIgqivFq1eqtxQMlAdUUDwac="
460 | },
461 | "cors": {
462 | "version": "2.8.5",
463 | "resolved": "https://registry.npmjs.org/cors/-/cors-2.8.5.tgz",
464 | "integrity": "sha512-KIHbLJqu73RGr/hnbrO9uBeixNGuvSQjul/jdFvS/KFSIH1hWVd1ng7zOHx+YrEfInLG7q4n6GHQ9cDtxv/P6g==",
465 | "requires": {
466 | "object-assign": "^4",
467 | "vary": "^1"
468 | }
469 | },
470 | "cross-spawn": {
471 | "version": "7.0.3",
472 | "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz",
473 | "integrity": "sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==",
474 | "dev": true,
475 | "requires": {
476 | "path-key": "^3.1.0",
477 | "shebang-command": "^2.0.0",
478 | "which": "^2.0.1"
479 | }
480 | },
481 | "crypto-random-string": {
482 | "version": "2.0.0",
483 | "resolved": "https://registry.npmjs.org/crypto-random-string/-/crypto-random-string-2.0.0.tgz",
484 | "integrity": "sha512-v1plID3y9r/lPhviJ1wrXpLeyUIGAZ2SHNYTEapm7/8A9nLPoyvVp3RK/EPFqn5kEznyWgYZNsRtYYIWbuG8KA==",
485 | "dev": true
486 | },
487 | "debug": {
488 | "version": "2.6.9",
489 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz",
490 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==",
491 | "requires": {
492 | "ms": "2.0.0"
493 | }
494 | },
495 | "decompress-response": {
496 | "version": "3.3.0",
497 | "resolved": "https://registry.npmjs.org/decompress-response/-/decompress-response-3.3.0.tgz",
498 | "integrity": "sha1-gKTdMjdIOEv6JICDYirt7Jgq3/M=",
499 | "dev": true,
500 | "requires": {
501 | "mimic-response": "^1.0.0"
502 | }
503 | },
504 | "deep-extend": {
505 | "version": "0.6.0",
506 | "resolved": "https://registry.npmjs.org/deep-extend/-/deep-extend-0.6.0.tgz",
507 | "integrity": "sha512-LOHxIOaPYdHlJRtCQfDIVZtfw/ufM8+rVj649RIHzcm/vGwQRXFt6OPqIFWsm2XEMrNIEtWR64sY1LEKD2vAOA==",
508 | "dev": true
509 | },
510 | "defer-to-connect": {
511 | "version": "1.1.3",
512 | "resolved": "https://registry.npmjs.org/defer-to-connect/-/defer-to-connect-1.1.3.tgz",
513 | "integrity": "sha512-0ISdNousHvZT2EiFlZeZAHBUvSxmKswVCEf8hW7KWgG4a8MVEu/3Vb6uWYozkjylyCxe0JBIiRB1jV45S70WVQ==",
514 | "dev": true
515 | },
516 | "denque": {
517 | "version": "1.5.0",
518 | "resolved": "https://registry.npmjs.org/denque/-/denque-1.5.0.tgz",
519 | "integrity": "sha512-CYiCSgIF1p6EUByQPlGkKnP1M9g0ZV3qMIrqMqZqdwazygIA/YP2vrbcyl1h/WppKJTdl1F85cXIle+394iDAQ=="
520 | },
521 | "depd": {
522 | "version": "1.1.2",
523 | "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.2.tgz",
524 | "integrity": "sha1-m81S4UwJd2PnSbJ0xDRu0uVgtak="
525 | },
526 | "destroy": {
527 | "version": "1.0.4",
528 | "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.0.4.tgz",
529 | "integrity": "sha1-l4hXRCxEdJ5CBmE+N5RiBYJqvYA="
530 | },
531 | "dot-prop": {
532 | "version": "5.3.0",
533 | "resolved": "https://registry.npmjs.org/dot-prop/-/dot-prop-5.3.0.tgz",
534 | "integrity": "sha512-QM8q3zDe58hqUqjraQOmzZ1LIH9SWQJTlEKCH4kJ2oQvLZk7RbQXvtDM2XEq3fwkV9CCvvH4LA0AV+ogFsBM2Q==",
535 | "dev": true,
536 | "requires": {
537 | "is-obj": "^2.0.0"
538 | }
539 | },
540 | "dotenv": {
541 | "version": "10.0.0",
542 | "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-10.0.0.tgz",
543 | "integrity": "sha512-rlBi9d8jpv9Sf1klPjNfFAuWDjKLwTIJJ/VxtoTwIR6hnZxcEOQCZg2oIL3MWBYw5GpUDKOEnND7LXTbIpQ03Q=="
544 | },
545 | "duplexer3": {
546 | "version": "0.1.4",
547 | "resolved": "https://registry.npmjs.org/duplexer3/-/duplexer3-0.1.4.tgz",
548 | "integrity": "sha1-7gHdHKwO08vH/b6jfcCo8c4ALOI=",
549 | "dev": true
550 | },
551 | "ecdsa-sig-formatter": {
552 | "version": "1.0.11",
553 | "resolved": "https://registry.npmjs.org/ecdsa-sig-formatter/-/ecdsa-sig-formatter-1.0.11.tgz",
554 | "integrity": "sha512-nagl3RYrbNv6kQkeJIpt6NJZy8twLB/2vtz6yN9Z4vRKHN4/QZJIEbqohALSgwKdnksuY3k5Addp5lg8sVoVcQ==",
555 | "requires": {
556 | "safe-buffer": "^5.0.1"
557 | }
558 | },
559 | "ee-first": {
560 | "version": "1.1.1",
561 | "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz",
562 | "integrity": "sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0="
563 | },
564 | "emoji-regex": {
565 | "version": "8.0.0",
566 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz",
567 | "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==",
568 | "dev": true
569 | },
570 | "encodeurl": {
571 | "version": "1.0.2",
572 | "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz",
573 | "integrity": "sha1-rT/0yG7C0CkyL1oCw6mmBslbP1k="
574 | },
575 | "end-of-stream": {
576 | "version": "1.4.4",
577 | "resolved": "https://registry.npmjs.org/end-of-stream/-/end-of-stream-1.4.4.tgz",
578 | "integrity": "sha512-+uw1inIHVPQoaVuHzRyXd21icM+cnt4CzD5rW+NC1wjOUSTOs+Te7FOv7AhN7vS9x/oIyhLP5PR1H+phQAHu5Q==",
579 | "dev": true,
580 | "requires": {
581 | "once": "^1.4.0"
582 | }
583 | },
584 | "env-cmd": {
585 | "version": "10.1.0",
586 | "resolved": "https://registry.npmjs.org/env-cmd/-/env-cmd-10.1.0.tgz",
587 | "integrity": "sha512-mMdWTT9XKN7yNth/6N6g2GuKuJTsKMDHlQFUDacb/heQRRWOTIZ42t1rMHnQu4jYxU1ajdTeJM+9eEETlqToMA==",
588 | "dev": true,
589 | "requires": {
590 | "commander": "^4.0.0",
591 | "cross-spawn": "^7.0.0"
592 | }
593 | },
594 | "escape-goat": {
595 | "version": "2.1.1",
596 | "resolved": "https://registry.npmjs.org/escape-goat/-/escape-goat-2.1.1.tgz",
597 | "integrity": "sha512-8/uIhbG12Csjy2JEW7D9pHbreaVaS/OpN3ycnyvElTdwM5n6GY6W6e2IPemfvGZeUMqZ9A/3GqIZMgKnBhAw/Q==",
598 | "dev": true
599 | },
600 | "escape-html": {
601 | "version": "1.0.3",
602 | "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz",
603 | "integrity": "sha1-Aljq5NPQwJdN4cFpGI7wBR0dGYg="
604 | },
605 | "etag": {
606 | "version": "1.8.1",
607 | "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz",
608 | "integrity": "sha1-Qa4u62XvpiJorr/qg6x9eSmbCIc="
609 | },
610 | "event-target-shim": {
611 | "version": "5.0.1",
612 | "resolved": "https://registry.npmjs.org/event-target-shim/-/event-target-shim-5.0.1.tgz",
613 | "integrity": "sha512-i/2XbnSz/uxRCU6+NdVJgKWDTM427+MqYbkQzD321DuCQJUqOuJKIA0IM2+W2xtYHdKOmZ4dR6fExsd4SXL+WQ=="
614 | },
615 | "express": {
616 | "version": "4.17.1",
617 | "resolved": "https://registry.npmjs.org/express/-/express-4.17.1.tgz",
618 | "integrity": "sha512-mHJ9O79RqluphRrcw2X/GTh3k9tVv8YcoyY4Kkh4WDMUYKRZUq0h1o0w2rrrxBqM7VoeUVqgb27xlEMXTnYt4g==",
619 | "requires": {
620 | "accepts": "~1.3.7",
621 | "array-flatten": "1.1.1",
622 | "body-parser": "1.19.0",
623 | "content-disposition": "0.5.3",
624 | "content-type": "~1.0.4",
625 | "cookie": "0.4.0",
626 | "cookie-signature": "1.0.6",
627 | "debug": "2.6.9",
628 | "depd": "~1.1.2",
629 | "encodeurl": "~1.0.2",
630 | "escape-html": "~1.0.3",
631 | "etag": "~1.8.1",
632 | "finalhandler": "~1.1.2",
633 | "fresh": "0.5.2",
634 | "merge-descriptors": "1.0.1",
635 | "methods": "~1.1.2",
636 | "on-finished": "~2.3.0",
637 | "parseurl": "~1.3.3",
638 | "path-to-regexp": "0.1.7",
639 | "proxy-addr": "~2.0.5",
640 | "qs": "6.7.0",
641 | "range-parser": "~1.2.1",
642 | "safe-buffer": "5.1.2",
643 | "send": "0.17.1",
644 | "serve-static": "1.14.1",
645 | "setprototypeof": "1.1.1",
646 | "statuses": "~1.5.0",
647 | "type-is": "~1.6.18",
648 | "utils-merge": "1.0.1",
649 | "vary": "~1.1.2"
650 | }
651 | },
652 | "extend": {
653 | "version": "3.0.2",
654 | "resolved": "https://registry.npmjs.org/extend/-/extend-3.0.2.tgz",
655 | "integrity": "sha512-fjquC59cD7CyW6urNXK0FBufkZcoiGG80wTuPujX590cB5Ttln20E2UB4S/WARVqhXffZl2LNgS+gQdPIIim/g=="
656 | },
657 | "fast-text-encoding": {
658 | "version": "1.0.3",
659 | "resolved": "https://registry.npmjs.org/fast-text-encoding/-/fast-text-encoding-1.0.3.tgz",
660 | "integrity": "sha512-dtm4QZH9nZtcDt8qJiOH9fcQd1NAgi+K1O2DbE6GG1PPCK/BWfOH3idCTRQ4ImXRUOyopDEgDEnVEE7Y/2Wrig=="
661 | },
662 | "fill-range": {
663 | "version": "7.0.1",
664 | "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz",
665 | "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==",
666 | "dev": true,
667 | "requires": {
668 | "to-regex-range": "^5.0.1"
669 | }
670 | },
671 | "finalhandler": {
672 | "version": "1.1.2",
673 | "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.1.2.tgz",
674 | "integrity": "sha512-aAWcW57uxVNrQZqFXjITpW3sIUQmHGG3qSb9mUah9MgMC4NeWhNOlNjXEYq3HjRAvL6arUviZGGJsBg6z0zsWA==",
675 | "requires": {
676 | "debug": "2.6.9",
677 | "encodeurl": "~1.0.2",
678 | "escape-html": "~1.0.3",
679 | "on-finished": "~2.3.0",
680 | "parseurl": "~1.3.3",
681 | "statuses": "~1.5.0",
682 | "unpipe": "~1.0.0"
683 | }
684 | },
685 | "follow-redirects": {
686 | "version": "1.14.1",
687 | "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.14.1.tgz",
688 | "integrity": "sha512-HWqDgT7ZEkqRzBvc2s64vSZ/hfOceEol3ac/7tKwzuvEyWx3/4UegXh5oBOIotkGsObyk3xznnSRVADBgWSQVg=="
689 | },
690 | "forwarded": {
691 | "version": "0.1.2",
692 | "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.1.2.tgz",
693 | "integrity": "sha1-mMI9qxF1ZXuMBXPozszZGw/xjIQ="
694 | },
695 | "fresh": {
696 | "version": "0.5.2",
697 | "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz",
698 | "integrity": "sha1-PYyt2Q2XZWn6g1qx+OSyOhBWBac="
699 | },
700 | "fsevents": {
701 | "version": "2.3.2",
702 | "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.2.tgz",
703 | "integrity": "sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==",
704 | "dev": true,
705 | "optional": true
706 | },
707 | "gaxios": {
708 | "version": "4.3.0",
709 | "resolved": "https://registry.npmjs.org/gaxios/-/gaxios-4.3.0.tgz",
710 | "integrity": "sha512-pHplNbslpwCLMyII/lHPWFQbJWOX0B3R1hwBEOvzYi1GmdKZruuEHK4N9V6f7tf1EaPYyF80mui1+344p6SmLg==",
711 | "requires": {
712 | "abort-controller": "^3.0.0",
713 | "extend": "^3.0.2",
714 | "https-proxy-agent": "^5.0.0",
715 | "is-stream": "^2.0.0",
716 | "node-fetch": "^2.3.0"
717 | }
718 | },
719 | "gcp-metadata": {
720 | "version": "4.2.1",
721 | "resolved": "https://registry.npmjs.org/gcp-metadata/-/gcp-metadata-4.2.1.tgz",
722 | "integrity": "sha512-tSk+REe5iq/N+K+SK1XjZJUrFPuDqGZVzCy2vocIHIGmPlTGsa8owXMJwGkrXr73NO0AzhPW4MF2DEHz7P2AVw==",
723 | "requires": {
724 | "gaxios": "^4.0.0",
725 | "json-bigint": "^1.0.0"
726 | }
727 | },
728 | "get-stream": {
729 | "version": "4.1.0",
730 | "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-4.1.0.tgz",
731 | "integrity": "sha512-GMat4EJ5161kIy2HevLlr4luNjBgvmj413KaQA7jt4V8B4RDsfpHk7WQ9GVqfYyyx8OS/L66Kox+rJRNklLK7w==",
732 | "dev": true,
733 | "requires": {
734 | "pump": "^3.0.0"
735 | }
736 | },
737 | "glob-parent": {
738 | "version": "5.1.2",
739 | "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz",
740 | "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==",
741 | "dev": true,
742 | "requires": {
743 | "is-glob": "^4.0.1"
744 | }
745 | },
746 | "global-dirs": {
747 | "version": "2.1.0",
748 | "resolved": "https://registry.npmjs.org/global-dirs/-/global-dirs-2.1.0.tgz",
749 | "integrity": "sha512-MG6kdOUh/xBnyo9cJFeIKkLEc1AyFq42QTU4XiX51i2NEdxLxLWXIjEjmqKeSuKR7pAZjTqUVoT2b2huxVLgYQ==",
750 | "dev": true,
751 | "requires": {
752 | "ini": "1.3.7"
753 | }
754 | },
755 | "google-auth-library": {
756 | "version": "7.1.1",
757 | "resolved": "https://registry.npmjs.org/google-auth-library/-/google-auth-library-7.1.1.tgz",
758 | "integrity": "sha512-+Q1linq/To3DYLyPz4UTEkQ0v5EOXadMM/S+taLV3W9611hq9zqg8kgGApqbTQnggtwdO9yU1y2YT7+83wdTRg==",
759 | "requires": {
760 | "arrify": "^2.0.0",
761 | "base64-js": "^1.3.0",
762 | "ecdsa-sig-formatter": "^1.0.11",
763 | "fast-text-encoding": "^1.0.0",
764 | "gaxios": "^4.0.0",
765 | "gcp-metadata": "^4.2.0",
766 | "gtoken": "^5.0.4",
767 | "jws": "^4.0.0",
768 | "lru-cache": "^6.0.0"
769 | },
770 | "dependencies": {
771 | "jwa": {
772 | "version": "2.0.0",
773 | "resolved": "https://registry.npmjs.org/jwa/-/jwa-2.0.0.tgz",
774 | "integrity": "sha512-jrZ2Qx916EA+fq9cEAeCROWPTfCwi1IVHqT2tapuqLEVVDKFDENFw1oL+MwrTvH6msKxsd1YTDVw6uKEcsrLEA==",
775 | "requires": {
776 | "buffer-equal-constant-time": "1.0.1",
777 | "ecdsa-sig-formatter": "1.0.11",
778 | "safe-buffer": "^5.0.1"
779 | }
780 | },
781 | "jws": {
782 | "version": "4.0.0",
783 | "resolved": "https://registry.npmjs.org/jws/-/jws-4.0.0.tgz",
784 | "integrity": "sha512-KDncfTmOZoOMTFG4mBlG0qUIOlc03fmzH+ru6RgYVZhPkyiy/92Owlt/8UEN+a4TXR1FQetfIpJE8ApdvdVxTg==",
785 | "requires": {
786 | "jwa": "^2.0.0",
787 | "safe-buffer": "^5.0.1"
788 | }
789 | }
790 | }
791 | },
792 | "google-p12-pem": {
793 | "version": "3.0.3",
794 | "resolved": "https://registry.npmjs.org/google-p12-pem/-/google-p12-pem-3.0.3.tgz",
795 | "integrity": "sha512-wS0ek4ZtFx/ACKYF3JhyGe5kzH7pgiQ7J5otlumqR9psmWMYc+U9cErKlCYVYHoUaidXHdZ2xbo34kB+S+24hA==",
796 | "requires": {
797 | "node-forge": "^0.10.0"
798 | }
799 | },
800 | "googleapis": {
801 | "version": "75.0.0",
802 | "resolved": "https://registry.npmjs.org/googleapis/-/googleapis-75.0.0.tgz",
803 | "integrity": "sha512-ZAZbWiBMw+ke5rzKRjNBSIJ+ab6wX/4HyHucIV1UHuLScNc8/rFCQsQg2VXEWBkdWUvsRxpB2AtTGtzcviJfwg==",
804 | "requires": {
805 | "google-auth-library": "^7.0.2",
806 | "googleapis-common": "^5.0.2"
807 | }
808 | },
809 | "googleapis-common": {
810 | "version": "5.0.2",
811 | "resolved": "https://registry.npmjs.org/googleapis-common/-/googleapis-common-5.0.2.tgz",
812 | "integrity": "sha512-TL7qronKNZwE/XBvqshwzCPmZGq2gz/beXzANF7EVoO7FsQjOd7dk40DYrXkoCpvbnJHCQKWESq6NansiIPFqA==",
813 | "requires": {
814 | "extend": "^3.0.2",
815 | "gaxios": "^4.0.0",
816 | "google-auth-library": "^7.0.2",
817 | "qs": "^6.7.0",
818 | "url-template": "^2.0.8",
819 | "uuid": "^8.0.0"
820 | }
821 | },
822 | "got": {
823 | "version": "9.6.0",
824 | "resolved": "https://registry.npmjs.org/got/-/got-9.6.0.tgz",
825 | "integrity": "sha512-R7eWptXuGYxwijs0eV+v3o6+XH1IqVK8dJOEecQfTmkncw9AV4dcw/Dhxi8MdlqPthxxpZyizMzyg8RTmEsG+Q==",
826 | "dev": true,
827 | "requires": {
828 | "@sindresorhus/is": "^0.14.0",
829 | "@szmarczak/http-timer": "^1.1.2",
830 | "cacheable-request": "^6.0.0",
831 | "decompress-response": "^3.3.0",
832 | "duplexer3": "^0.1.4",
833 | "get-stream": "^4.1.0",
834 | "lowercase-keys": "^1.0.1",
835 | "mimic-response": "^1.0.1",
836 | "p-cancelable": "^1.0.0",
837 | "to-readable-stream": "^1.0.0",
838 | "url-parse-lax": "^3.0.0"
839 | }
840 | },
841 | "graceful-fs": {
842 | "version": "4.2.6",
843 | "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.6.tgz",
844 | "integrity": "sha512-nTnJ528pbqxYanhpDYsi4Rd8MAeaBA67+RZ10CM1m3bTAVFEDcd5AuA4a6W5YkGZ1iNXHzZz8T6TBKLeBuNriQ==",
845 | "dev": true
846 | },
847 | "gtoken": {
848 | "version": "5.2.1",
849 | "resolved": "https://registry.npmjs.org/gtoken/-/gtoken-5.2.1.tgz",
850 | "integrity": "sha512-OY0BfPKe3QnMsY9MzTHTSKn+Vl2l1CcLe6BwDEQj00mbbkl5nyQ/7EUREstg4fQNZ8iYE7br4JJ7TdKeDOPWmw==",
851 | "requires": {
852 | "gaxios": "^4.0.0",
853 | "google-p12-pem": "^3.0.3",
854 | "jws": "^4.0.0"
855 | },
856 | "dependencies": {
857 | "jwa": {
858 | "version": "2.0.0",
859 | "resolved": "https://registry.npmjs.org/jwa/-/jwa-2.0.0.tgz",
860 | "integrity": "sha512-jrZ2Qx916EA+fq9cEAeCROWPTfCwi1IVHqT2tapuqLEVVDKFDENFw1oL+MwrTvH6msKxsd1YTDVw6uKEcsrLEA==",
861 | "requires": {
862 | "buffer-equal-constant-time": "1.0.1",
863 | "ecdsa-sig-formatter": "1.0.11",
864 | "safe-buffer": "^5.0.1"
865 | }
866 | },
867 | "jws": {
868 | "version": "4.0.0",
869 | "resolved": "https://registry.npmjs.org/jws/-/jws-4.0.0.tgz",
870 | "integrity": "sha512-KDncfTmOZoOMTFG4mBlG0qUIOlc03fmzH+ru6RgYVZhPkyiy/92Owlt/8UEN+a4TXR1FQetfIpJE8ApdvdVxTg==",
871 | "requires": {
872 | "jwa": "^2.0.0",
873 | "safe-buffer": "^5.0.1"
874 | }
875 | }
876 | }
877 | },
878 | "has-flag": {
879 | "version": "3.0.0",
880 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz",
881 | "integrity": "sha1-tdRU3CGZriJWmfNGfloH87lVuv0=",
882 | "dev": true
883 | },
884 | "has-yarn": {
885 | "version": "2.1.0",
886 | "resolved": "https://registry.npmjs.org/has-yarn/-/has-yarn-2.1.0.tgz",
887 | "integrity": "sha512-UqBRqi4ju7T+TqGNdqAO0PaSVGsDGJUBQvk9eUWNGRY1CFGDzYhLWoM7JQEemnlvVcv/YEmc2wNW8BC24EnUsw==",
888 | "dev": true
889 | },
890 | "http-cache-semantics": {
891 | "version": "4.1.0",
892 | "resolved": "https://registry.npmjs.org/http-cache-semantics/-/http-cache-semantics-4.1.0.tgz",
893 | "integrity": "sha512-carPklcUh7ROWRK7Cv27RPtdhYhUsela/ue5/jKzjegVvXDqM2ILE9Q2BGn9JZJh1g87cp56su/FgQSzcWS8cQ==",
894 | "dev": true
895 | },
896 | "http-errors": {
897 | "version": "1.7.2",
898 | "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.7.2.tgz",
899 | "integrity": "sha512-uUQBt3H/cSIVfch6i1EuPNy/YsRSOUBXTVfZ+yR7Zjez3qjBz6i9+i4zjNaoqcoFVI4lQJ5plg63TvGfRSDCRg==",
900 | "requires": {
901 | "depd": "~1.1.2",
902 | "inherits": "2.0.3",
903 | "setprototypeof": "1.1.1",
904 | "statuses": ">= 1.5.0 < 2",
905 | "toidentifier": "1.0.0"
906 | }
907 | },
908 | "https-proxy-agent": {
909 | "version": "5.0.0",
910 | "resolved": "https://registry.npmjs.org/https-proxy-agent/-/https-proxy-agent-5.0.0.tgz",
911 | "integrity": "sha512-EkYm5BcKUGiduxzSt3Eppko+PiNWNEpa4ySk9vTC6wDsQJW9rHSa+UhGNJoRYp7bz6Ht1eaRIa6QaJqO5rCFbA==",
912 | "requires": {
913 | "agent-base": "6",
914 | "debug": "4"
915 | },
916 | "dependencies": {
917 | "debug": {
918 | "version": "4.3.1",
919 | "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.1.tgz",
920 | "integrity": "sha512-doEwdvm4PCeK4K3RQN2ZC2BYUBaxwLARCqZmMjtF8a51J2Rb0xpVloFRnCODwqjpwnAoao4pelN8l3RJdv3gRQ==",
921 | "requires": {
922 | "ms": "2.1.2"
923 | }
924 | },
925 | "ms": {
926 | "version": "2.1.2",
927 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz",
928 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w=="
929 | }
930 | }
931 | },
932 | "iconv-lite": {
933 | "version": "0.4.24",
934 | "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz",
935 | "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==",
936 | "requires": {
937 | "safer-buffer": ">= 2.1.2 < 3"
938 | }
939 | },
940 | "ignore-by-default": {
941 | "version": "1.0.1",
942 | "resolved": "https://registry.npmjs.org/ignore-by-default/-/ignore-by-default-1.0.1.tgz",
943 | "integrity": "sha1-SMptcvbGo68Aqa1K5odr44ieKwk=",
944 | "dev": true
945 | },
946 | "import-lazy": {
947 | "version": "2.1.0",
948 | "resolved": "https://registry.npmjs.org/import-lazy/-/import-lazy-2.1.0.tgz",
949 | "integrity": "sha1-BWmOPUXIjo1+nZLLBYTnfwlvPkM=",
950 | "dev": true
951 | },
952 | "imurmurhash": {
953 | "version": "0.1.4",
954 | "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz",
955 | "integrity": "sha1-khi5srkoojixPcT7a21XbyMUU+o=",
956 | "dev": true
957 | },
958 | "inherits": {
959 | "version": "2.0.3",
960 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz",
961 | "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4="
962 | },
963 | "ini": {
964 | "version": "1.3.7",
965 | "resolved": "https://registry.npmjs.org/ini/-/ini-1.3.7.tgz",
966 | "integrity": "sha512-iKpRpXP+CrP2jyrxvg1kMUpXDyRUFDWurxbnVT1vQPx+Wz9uCYsMIqYuSBLV+PAaZG/d7kRLKRFc9oDMsH+mFQ==",
967 | "dev": true
968 | },
969 | "ipaddr.js": {
970 | "version": "1.9.1",
971 | "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz",
972 | "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g=="
973 | },
974 | "is-binary-path": {
975 | "version": "2.1.0",
976 | "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz",
977 | "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==",
978 | "dev": true,
979 | "requires": {
980 | "binary-extensions": "^2.0.0"
981 | }
982 | },
983 | "is-ci": {
984 | "version": "2.0.0",
985 | "resolved": "https://registry.npmjs.org/is-ci/-/is-ci-2.0.0.tgz",
986 | "integrity": "sha512-YfJT7rkpQB0updsdHLGWrvhBJfcfzNNawYDNIyQXJz0IViGf75O8EBPKSdvw2rF+LGCsX4FZ8tcr3b19LcZq4w==",
987 | "dev": true,
988 | "requires": {
989 | "ci-info": "^2.0.0"
990 | }
991 | },
992 | "is-extglob": {
993 | "version": "2.1.1",
994 | "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz",
995 | "integrity": "sha1-qIwCU1eR8C7TfHahueqXc8gz+MI=",
996 | "dev": true
997 | },
998 | "is-fullwidth-code-point": {
999 | "version": "3.0.0",
1000 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz",
1001 | "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==",
1002 | "dev": true
1003 | },
1004 | "is-glob": {
1005 | "version": "4.0.1",
1006 | "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.1.tgz",
1007 | "integrity": "sha512-5G0tKtBTFImOqDnLB2hG6Bp2qcKEFduo4tZu9MT/H6NQv/ghhy30o55ufafxJ/LdH79LLs2Kfrn85TLKyA7BUg==",
1008 | "dev": true,
1009 | "requires": {
1010 | "is-extglob": "^2.1.1"
1011 | }
1012 | },
1013 | "is-installed-globally": {
1014 | "version": "0.3.2",
1015 | "resolved": "https://registry.npmjs.org/is-installed-globally/-/is-installed-globally-0.3.2.tgz",
1016 | "integrity": "sha512-wZ8x1js7Ia0kecP/CHM/3ABkAmujX7WPvQk6uu3Fly/Mk44pySulQpnHG46OMjHGXApINnV4QhY3SWnECO2z5g==",
1017 | "dev": true,
1018 | "requires": {
1019 | "global-dirs": "^2.0.1",
1020 | "is-path-inside": "^3.0.1"
1021 | }
1022 | },
1023 | "is-npm": {
1024 | "version": "4.0.0",
1025 | "resolved": "https://registry.npmjs.org/is-npm/-/is-npm-4.0.0.tgz",
1026 | "integrity": "sha512-96ECIfh9xtDDlPylNPXhzjsykHsMJZ18ASpaWzQyBr4YRTcVjUvzaHayDAES2oU/3KpljhHUjtSRNiDwi0F0ig==",
1027 | "dev": true
1028 | },
1029 | "is-number": {
1030 | "version": "7.0.0",
1031 | "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz",
1032 | "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==",
1033 | "dev": true
1034 | },
1035 | "is-obj": {
1036 | "version": "2.0.0",
1037 | "resolved": "https://registry.npmjs.org/is-obj/-/is-obj-2.0.0.tgz",
1038 | "integrity": "sha512-drqDG3cbczxxEJRoOXcOjtdp1J/lyp1mNn0xaznRs8+muBhgQcrnbspox5X5fOw0HnMnbfDzvnEMEtqDEJEo8w==",
1039 | "dev": true
1040 | },
1041 | "is-path-inside": {
1042 | "version": "3.0.3",
1043 | "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-3.0.3.tgz",
1044 | "integrity": "sha512-Fd4gABb+ycGAmKou8eMftCupSir5lRxqf4aD/vd0cD2qc4HL07OjCeuHMr8Ro4CoMaeCKDB0/ECBOVWjTwUvPQ==",
1045 | "dev": true
1046 | },
1047 | "is-stream": {
1048 | "version": "2.0.0",
1049 | "resolved": "https://registry.npmjs.org/is-stream/-/is-stream-2.0.0.tgz",
1050 | "integrity": "sha512-XCoy+WlUr7d1+Z8GgSuXmpuUFC9fOhRXglJMx+dwLKTkL44Cjd4W1Z5P+BQZpr+cR93aGP4S/s7Ftw6Nd/kiEw=="
1051 | },
1052 | "is-typedarray": {
1053 | "version": "1.0.0",
1054 | "resolved": "https://registry.npmjs.org/is-typedarray/-/is-typedarray-1.0.0.tgz",
1055 | "integrity": "sha1-5HnICFjfDBsR3dppQPlgEfzaSpo=",
1056 | "dev": true
1057 | },
1058 | "is-yarn-global": {
1059 | "version": "0.3.0",
1060 | "resolved": "https://registry.npmjs.org/is-yarn-global/-/is-yarn-global-0.3.0.tgz",
1061 | "integrity": "sha512-VjSeb/lHmkoyd8ryPVIKvOCn4D1koMqY+vqyjjUfc3xyKtP4dYOxM44sZrnqQSzSds3xyOrUTLTC9LVCVgLngw==",
1062 | "dev": true
1063 | },
1064 | "isarray": {
1065 | "version": "1.0.0",
1066 | "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz",
1067 | "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE="
1068 | },
1069 | "isexe": {
1070 | "version": "2.0.0",
1071 | "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz",
1072 | "integrity": "sha1-6PvzdNxVb/iUehDcsFctYz8s+hA=",
1073 | "dev": true
1074 | },
1075 | "json-bigint": {
1076 | "version": "1.0.0",
1077 | "resolved": "https://registry.npmjs.org/json-bigint/-/json-bigint-1.0.0.tgz",
1078 | "integrity": "sha512-SiPv/8VpZuWbvLSMtTDU8hEfrZWg/mH/nV/b4o0CYbSxu1UIQPLdwKOCIyLQX+VIPO5vrLX3i8qtqFyhdPSUSQ==",
1079 | "requires": {
1080 | "bignumber.js": "^9.0.0"
1081 | }
1082 | },
1083 | "json-buffer": {
1084 | "version": "3.0.0",
1085 | "resolved": "https://registry.npmjs.org/json-buffer/-/json-buffer-3.0.0.tgz",
1086 | "integrity": "sha1-Wx85evx11ne96Lz8Dkfh+aPZqJg=",
1087 | "dev": true
1088 | },
1089 | "jsonwebtoken": {
1090 | "version": "8.5.1",
1091 | "resolved": "https://registry.npmjs.org/jsonwebtoken/-/jsonwebtoken-8.5.1.tgz",
1092 | "integrity": "sha512-XjwVfRS6jTMsqYs0EsuJ4LGxXV14zQybNd4L2r0UvbVnSF9Af8x7p5MzbJ90Ioz/9TI41/hTCvznF/loiSzn8w==",
1093 | "requires": {
1094 | "jws": "^3.2.2",
1095 | "lodash.includes": "^4.3.0",
1096 | "lodash.isboolean": "^3.0.3",
1097 | "lodash.isinteger": "^4.0.4",
1098 | "lodash.isnumber": "^3.0.3",
1099 | "lodash.isplainobject": "^4.0.6",
1100 | "lodash.isstring": "^4.0.1",
1101 | "lodash.once": "^4.0.0",
1102 | "ms": "^2.1.1",
1103 | "semver": "^5.6.0"
1104 | },
1105 | "dependencies": {
1106 | "ms": {
1107 | "version": "2.1.3",
1108 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz",
1109 | "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA=="
1110 | }
1111 | }
1112 | },
1113 | "jwa": {
1114 | "version": "1.4.1",
1115 | "resolved": "https://registry.npmjs.org/jwa/-/jwa-1.4.1.tgz",
1116 | "integrity": "sha512-qiLX/xhEEFKUAJ6FiBMbes3w9ATzyk5W7Hvzpa/SLYdxNtng+gcurvrI7TbACjIXlsJyr05/S1oUhZrc63evQA==",
1117 | "requires": {
1118 | "buffer-equal-constant-time": "1.0.1",
1119 | "ecdsa-sig-formatter": "1.0.11",
1120 | "safe-buffer": "^5.0.1"
1121 | }
1122 | },
1123 | "jws": {
1124 | "version": "3.2.2",
1125 | "resolved": "https://registry.npmjs.org/jws/-/jws-3.2.2.tgz",
1126 | "integrity": "sha512-YHlZCB6lMTllWDtSPHz/ZXTsi8S00usEV6v1tjq8tOUZzw7DpSDWVXjXDre6ed1w/pd495ODpHZYSdkRTsa0HA==",
1127 | "requires": {
1128 | "jwa": "^1.4.1",
1129 | "safe-buffer": "^5.0.1"
1130 | }
1131 | },
1132 | "kareem": {
1133 | "version": "2.3.2",
1134 | "resolved": "https://registry.npmjs.org/kareem/-/kareem-2.3.2.tgz",
1135 | "integrity": "sha512-STHz9P7X2L4Kwn72fA4rGyqyXdmrMSdxqHx9IXon/FXluXieaFA6KJ2upcHAHxQPQ0LeM/OjLrhFxifHewOALQ=="
1136 | },
1137 | "keyv": {
1138 | "version": "3.1.0",
1139 | "resolved": "https://registry.npmjs.org/keyv/-/keyv-3.1.0.tgz",
1140 | "integrity": "sha512-9ykJ/46SN/9KPM/sichzQ7OvXyGDYKGTaDlKMGCAlg2UK8KRy4jb0d8sFc+0Tt0YYnThq8X2RZgCg74RPxgcVA==",
1141 | "dev": true,
1142 | "requires": {
1143 | "json-buffer": "3.0.0"
1144 | }
1145 | },
1146 | "latest-version": {
1147 | "version": "5.1.0",
1148 | "resolved": "https://registry.npmjs.org/latest-version/-/latest-version-5.1.0.tgz",
1149 | "integrity": "sha512-weT+r0kTkRQdCdYCNtkMwWXQTMEswKrFBkm4ckQOMVhhqhIMI1UT2hMj+1iigIhgSZm5gTmrRXBNoGUgaTY1xA==",
1150 | "dev": true,
1151 | "requires": {
1152 | "package-json": "^6.3.0"
1153 | }
1154 | },
1155 | "lodash.includes": {
1156 | "version": "4.3.0",
1157 | "resolved": "https://registry.npmjs.org/lodash.includes/-/lodash.includes-4.3.0.tgz",
1158 | "integrity": "sha1-YLuYqHy5I8aMoeUTJUgzFISfVT8="
1159 | },
1160 | "lodash.isboolean": {
1161 | "version": "3.0.3",
1162 | "resolved": "https://registry.npmjs.org/lodash.isboolean/-/lodash.isboolean-3.0.3.tgz",
1163 | "integrity": "sha1-bC4XHbKiV82WgC/UOwGyDV9YcPY="
1164 | },
1165 | "lodash.isinteger": {
1166 | "version": "4.0.4",
1167 | "resolved": "https://registry.npmjs.org/lodash.isinteger/-/lodash.isinteger-4.0.4.tgz",
1168 | "integrity": "sha1-YZwK89A/iwTDH1iChAt3sRzWg0M="
1169 | },
1170 | "lodash.isnumber": {
1171 | "version": "3.0.3",
1172 | "resolved": "https://registry.npmjs.org/lodash.isnumber/-/lodash.isnumber-3.0.3.tgz",
1173 | "integrity": "sha1-POdoEMWSjQM1IwGsKHMX8RwLH/w="
1174 | },
1175 | "lodash.isplainobject": {
1176 | "version": "4.0.6",
1177 | "resolved": "https://registry.npmjs.org/lodash.isplainobject/-/lodash.isplainobject-4.0.6.tgz",
1178 | "integrity": "sha1-fFJqUtibRcRcxpC4gWO+BJf1UMs="
1179 | },
1180 | "lodash.isstring": {
1181 | "version": "4.0.1",
1182 | "resolved": "https://registry.npmjs.org/lodash.isstring/-/lodash.isstring-4.0.1.tgz",
1183 | "integrity": "sha1-1SfftUVuynzJu5XV2ur4i6VKVFE="
1184 | },
1185 | "lodash.once": {
1186 | "version": "4.1.1",
1187 | "resolved": "https://registry.npmjs.org/lodash.once/-/lodash.once-4.1.1.tgz",
1188 | "integrity": "sha1-DdOXEhPHxW34gJd9UEyI+0cal6w="
1189 | },
1190 | "lowercase-keys": {
1191 | "version": "1.0.1",
1192 | "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-1.0.1.tgz",
1193 | "integrity": "sha512-G2Lj61tXDnVFFOi8VZds+SoQjtQC3dgokKdDG2mTm1tx4m50NUHBOZSBwQQHyy0V12A0JTG4icfZQH+xPyh8VA==",
1194 | "dev": true
1195 | },
1196 | "lru-cache": {
1197 | "version": "6.0.0",
1198 | "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-6.0.0.tgz",
1199 | "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==",
1200 | "requires": {
1201 | "yallist": "^4.0.0"
1202 | }
1203 | },
1204 | "make-dir": {
1205 | "version": "3.1.0",
1206 | "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-3.1.0.tgz",
1207 | "integrity": "sha512-g3FeP20LNwhALb/6Cz6Dd4F2ngze0jz7tbzrD2wAV+o9FeNHe4rL+yK2md0J/fiSf1sa1ADhXqi5+oVwOM/eGw==",
1208 | "dev": true,
1209 | "requires": {
1210 | "semver": "^6.0.0"
1211 | },
1212 | "dependencies": {
1213 | "semver": {
1214 | "version": "6.3.0",
1215 | "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz",
1216 | "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==",
1217 | "dev": true
1218 | }
1219 | }
1220 | },
1221 | "media-typer": {
1222 | "version": "0.3.0",
1223 | "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz",
1224 | "integrity": "sha1-hxDXrwqmJvj/+hzgAWhUUmMlV0g="
1225 | },
1226 | "memory-pager": {
1227 | "version": "1.5.0",
1228 | "resolved": "https://registry.npmjs.org/memory-pager/-/memory-pager-1.5.0.tgz",
1229 | "integrity": "sha512-ZS4Bp4r/Zoeq6+NLJpP+0Zzm0pR8whtGPf1XExKLJBAczGMnSi3It14OiNCStjQjM6NU1okjQGSxgEZN8eBYKg==",
1230 | "optional": true
1231 | },
1232 | "merge-descriptors": {
1233 | "version": "1.0.1",
1234 | "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz",
1235 | "integrity": "sha1-sAqqVW3YtEVoFQ7J0blT8/kMu2E="
1236 | },
1237 | "methods": {
1238 | "version": "1.1.2",
1239 | "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz",
1240 | "integrity": "sha1-VSmk1nZUE07cxSZmVoNbD4Ua/O4="
1241 | },
1242 | "mime": {
1243 | "version": "1.6.0",
1244 | "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz",
1245 | "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg=="
1246 | },
1247 | "mime-db": {
1248 | "version": "1.47.0",
1249 | "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.47.0.tgz",
1250 | "integrity": "sha512-QBmA/G2y+IfeS4oktet3qRZ+P5kPhCKRXxXnQEudYqUaEioAU1/Lq2us3D/t1Jfo4hE9REQPrbB7K5sOczJVIw=="
1251 | },
1252 | "mime-types": {
1253 | "version": "2.1.30",
1254 | "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.30.tgz",
1255 | "integrity": "sha512-crmjA4bLtR8m9qLpHvgxSChT+XoSlZi8J4n/aIdn3z92e/U47Z0V/yl+Wh9W046GgFVAmoNR/fmdbZYcSSIUeg==",
1256 | "requires": {
1257 | "mime-db": "1.47.0"
1258 | }
1259 | },
1260 | "mimic-response": {
1261 | "version": "1.0.1",
1262 | "resolved": "https://registry.npmjs.org/mimic-response/-/mimic-response-1.0.1.tgz",
1263 | "integrity": "sha512-j5EctnkH7amfV/q5Hgmoal1g2QHFJRraOtmx0JpIqkxhBhI/lJSl1nMpQ45hVarwNETOoWEimndZ4QK0RHxuxQ==",
1264 | "dev": true
1265 | },
1266 | "minimatch": {
1267 | "version": "3.0.4",
1268 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz",
1269 | "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==",
1270 | "dev": true,
1271 | "requires": {
1272 | "brace-expansion": "^1.1.7"
1273 | }
1274 | },
1275 | "minimist": {
1276 | "version": "1.2.5",
1277 | "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.5.tgz",
1278 | "integrity": "sha512-FM9nNUYrRBAELZQT3xeZQ7fmMOBg6nWNmJKTcgsJeaLstP/UODVpGsr5OhXhhXg6f+qtJ8uiZ+PUxkDWcgIXLw==",
1279 | "dev": true
1280 | },
1281 | "mongodb": {
1282 | "version": "3.6.8",
1283 | "resolved": "https://registry.npmjs.org/mongodb/-/mongodb-3.6.8.tgz",
1284 | "integrity": "sha512-sDjJvI73WjON1vapcbyBD3Ao9/VN3TKYY8/QX9EPbs22KaCSrQ5rXo5ZZd44tWJ3wl3FlnrFZ+KyUtNH6+1ZPQ==",
1285 | "requires": {
1286 | "bl": "^2.2.1",
1287 | "bson": "^1.1.4",
1288 | "denque": "^1.4.1",
1289 | "optional-require": "^1.0.3",
1290 | "safe-buffer": "^5.1.2",
1291 | "saslprep": "^1.0.0"
1292 | }
1293 | },
1294 | "mongoose": {
1295 | "version": "5.12.12",
1296 | "resolved": "https://registry.npmjs.org/mongoose/-/mongoose-5.12.12.tgz",
1297 | "integrity": "sha512-n+ZmGApaL5x/r92w6S4pb+c075i+YKzg1F9YWkznSzQMtvetj/2dSjj2cqsITpd6z60k3K7ZaosIl6hzHwUA9g==",
1298 | "requires": {
1299 | "@types/mongodb": "^3.5.27",
1300 | "bson": "^1.1.4",
1301 | "kareem": "2.3.2",
1302 | "mongodb": "3.6.8",
1303 | "mongoose-legacy-pluralize": "1.0.2",
1304 | "mpath": "0.8.3",
1305 | "mquery": "3.2.5",
1306 | "ms": "2.1.2",
1307 | "regexp-clone": "1.0.0",
1308 | "safe-buffer": "5.2.1",
1309 | "sift": "13.5.2",
1310 | "sliced": "1.0.1"
1311 | },
1312 | "dependencies": {
1313 | "ms": {
1314 | "version": "2.1.2",
1315 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz",
1316 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w=="
1317 | },
1318 | "safe-buffer": {
1319 | "version": "5.2.1",
1320 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz",
1321 | "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ=="
1322 | }
1323 | }
1324 | },
1325 | "mongoose-legacy-pluralize": {
1326 | "version": "1.0.2",
1327 | "resolved": "https://registry.npmjs.org/mongoose-legacy-pluralize/-/mongoose-legacy-pluralize-1.0.2.tgz",
1328 | "integrity": "sha512-Yo/7qQU4/EyIS8YDFSeenIvXxZN+ld7YdV9LqFVQJzTLye8unujAWPZ4NWKfFA+RNjh+wvTWKY9Z3E5XM6ZZiQ=="
1329 | },
1330 | "mpath": {
1331 | "version": "0.8.3",
1332 | "resolved": "https://registry.npmjs.org/mpath/-/mpath-0.8.3.tgz",
1333 | "integrity": "sha512-eb9rRvhDltXVNL6Fxd2zM9D4vKBxjVVQNLNijlj7uoXUy19zNDsIif5zR+pWmPCWNKwAtqyo4JveQm4nfD5+eA=="
1334 | },
1335 | "mquery": {
1336 | "version": "3.2.5",
1337 | "resolved": "https://registry.npmjs.org/mquery/-/mquery-3.2.5.tgz",
1338 | "integrity": "sha512-VjOKHHgU84wij7IUoZzFRU07IAxd5kWJaDmyUzQlbjHjyoeK5TNeeo8ZsFDtTYnSgpW6n/nMNIHvE3u8Lbrf4A==",
1339 | "requires": {
1340 | "bluebird": "3.5.1",
1341 | "debug": "3.1.0",
1342 | "regexp-clone": "^1.0.0",
1343 | "safe-buffer": "5.1.2",
1344 | "sliced": "1.0.1"
1345 | },
1346 | "dependencies": {
1347 | "debug": {
1348 | "version": "3.1.0",
1349 | "resolved": "https://registry.npmjs.org/debug/-/debug-3.1.0.tgz",
1350 | "integrity": "sha512-OX8XqP7/1a9cqkxYw2yXss15f26NKWBpDXQd0/uK/KPqdQhxbPa994hnzjcE2VqQpDslf55723cKPUOGSmMY3g==",
1351 | "requires": {
1352 | "ms": "2.0.0"
1353 | }
1354 | }
1355 | }
1356 | },
1357 | "ms": {
1358 | "version": "2.0.0",
1359 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz",
1360 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g="
1361 | },
1362 | "negotiator": {
1363 | "version": "0.6.2",
1364 | "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.2.tgz",
1365 | "integrity": "sha512-hZXc7K2e+PgeI1eDBe/10Ard4ekbfrrqG8Ep+8Jmf4JID2bNg7NvCPOZN+kfF574pFQI7mum2AUqDidoKqcTOw=="
1366 | },
1367 | "node-fetch": {
1368 | "version": "2.6.1",
1369 | "resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-2.6.1.tgz",
1370 | "integrity": "sha512-V4aYg89jEoVRxRb2fJdAg8FHvI7cEyYdVAh94HH0UIK8oJxUfkjlDQN9RbMx+bEjP7+ggMiFRprSti032Oipxw=="
1371 | },
1372 | "node-forge": {
1373 | "version": "0.10.0",
1374 | "resolved": "https://registry.npmjs.org/node-forge/-/node-forge-0.10.0.tgz",
1375 | "integrity": "sha512-PPmu8eEeG9saEUvI97fm4OYxXVB6bFvyNTyiUOBichBpFG8A1Ljw3bY62+5oOjDEMHRnd0Y7HQ+x7uzxOzC6JA=="
1376 | },
1377 | "nodemon": {
1378 | "version": "2.0.7",
1379 | "resolved": "https://registry.npmjs.org/nodemon/-/nodemon-2.0.7.tgz",
1380 | "integrity": "sha512-XHzK69Awgnec9UzHr1kc8EomQh4sjTQ8oRf8TsGrSmHDx9/UmiGG9E/mM3BuTfNeFwdNBvrqQq/RHL0xIeyFOA==",
1381 | "dev": true,
1382 | "requires": {
1383 | "chokidar": "^3.2.2",
1384 | "debug": "^3.2.6",
1385 | "ignore-by-default": "^1.0.1",
1386 | "minimatch": "^3.0.4",
1387 | "pstree.remy": "^1.1.7",
1388 | "semver": "^5.7.1",
1389 | "supports-color": "^5.5.0",
1390 | "touch": "^3.1.0",
1391 | "undefsafe": "^2.0.3",
1392 | "update-notifier": "^4.1.0"
1393 | },
1394 | "dependencies": {
1395 | "debug": {
1396 | "version": "3.2.7",
1397 | "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz",
1398 | "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==",
1399 | "dev": true,
1400 | "requires": {
1401 | "ms": "^2.1.1"
1402 | }
1403 | },
1404 | "ms": {
1405 | "version": "2.1.3",
1406 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz",
1407 | "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==",
1408 | "dev": true
1409 | }
1410 | }
1411 | },
1412 | "nopt": {
1413 | "version": "1.0.10",
1414 | "resolved": "https://registry.npmjs.org/nopt/-/nopt-1.0.10.tgz",
1415 | "integrity": "sha1-bd0hvSoxQXuScn3Vhfim83YI6+4=",
1416 | "dev": true,
1417 | "requires": {
1418 | "abbrev": "1"
1419 | }
1420 | },
1421 | "normalize-path": {
1422 | "version": "3.0.0",
1423 | "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz",
1424 | "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==",
1425 | "dev": true
1426 | },
1427 | "normalize-url": {
1428 | "version": "4.5.1",
1429 | "resolved": "https://registry.npmjs.org/normalize-url/-/normalize-url-4.5.1.tgz",
1430 | "integrity": "sha512-9UZCFRHQdNrfTpGg8+1INIg93B6zE0aXMVFkw1WFwvO4SlZywU6aLg5Of0Ap/PgcbSw4LNxvMWXMeugwMCX0AA==",
1431 | "dev": true
1432 | },
1433 | "object-assign": {
1434 | "version": "4.1.1",
1435 | "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz",
1436 | "integrity": "sha1-IQmtx5ZYh8/AXLvUQsrIv7s2CGM="
1437 | },
1438 | "on-finished": {
1439 | "version": "2.3.0",
1440 | "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.3.0.tgz",
1441 | "integrity": "sha1-IPEzZIGwg811M3mSoWlxqi2QaUc=",
1442 | "requires": {
1443 | "ee-first": "1.1.1"
1444 | }
1445 | },
1446 | "once": {
1447 | "version": "1.4.0",
1448 | "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz",
1449 | "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=",
1450 | "dev": true,
1451 | "requires": {
1452 | "wrappy": "1"
1453 | }
1454 | },
1455 | "optional-require": {
1456 | "version": "1.0.3",
1457 | "resolved": "https://registry.npmjs.org/optional-require/-/optional-require-1.0.3.tgz",
1458 | "integrity": "sha512-RV2Zp2MY2aeYK5G+B/Sps8lW5NHAzE5QClbFP15j+PWmP+T9PxlJXBOOLoSAdgwFvS4t0aMR4vpedMkbHfh0nA=="
1459 | },
1460 | "p-cancelable": {
1461 | "version": "1.1.0",
1462 | "resolved": "https://registry.npmjs.org/p-cancelable/-/p-cancelable-1.1.0.tgz",
1463 | "integrity": "sha512-s73XxOZ4zpt1edZYZzvhqFa6uvQc1vwUa0K0BdtIZgQMAJj9IbebH+JkgKZc9h+B05PKHLOTl4ajG1BmNrVZlw==",
1464 | "dev": true
1465 | },
1466 | "package-json": {
1467 | "version": "6.5.0",
1468 | "resolved": "https://registry.npmjs.org/package-json/-/package-json-6.5.0.tgz",
1469 | "integrity": "sha512-k3bdm2n25tkyxcjSKzB5x8kfVxlMdgsbPr0GkZcwHsLpba6cBjqCt1KlcChKEvxHIcTB1FVMuwoijZ26xex5MQ==",
1470 | "dev": true,
1471 | "requires": {
1472 | "got": "^9.6.0",
1473 | "registry-auth-token": "^4.0.0",
1474 | "registry-url": "^5.0.0",
1475 | "semver": "^6.2.0"
1476 | },
1477 | "dependencies": {
1478 | "semver": {
1479 | "version": "6.3.0",
1480 | "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz",
1481 | "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==",
1482 | "dev": true
1483 | }
1484 | }
1485 | },
1486 | "parseurl": {
1487 | "version": "1.3.3",
1488 | "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz",
1489 | "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ=="
1490 | },
1491 | "path-key": {
1492 | "version": "3.1.1",
1493 | "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz",
1494 | "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==",
1495 | "dev": true
1496 | },
1497 | "path-to-regexp": {
1498 | "version": "0.1.7",
1499 | "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz",
1500 | "integrity": "sha1-32BBeABfUi8V60SQ5yR6G/qmf4w="
1501 | },
1502 | "picomatch": {
1503 | "version": "2.3.0",
1504 | "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.0.tgz",
1505 | "integrity": "sha512-lY1Q/PiJGC2zOv/z391WOTD+Z02bCgsFfvxoXXf6h7kv9o+WmsmzYqrAwY63sNgOxE4xEdq0WyUnXfKeBrSvYw==",
1506 | "dev": true
1507 | },
1508 | "prepend-http": {
1509 | "version": "2.0.0",
1510 | "resolved": "https://registry.npmjs.org/prepend-http/-/prepend-http-2.0.0.tgz",
1511 | "integrity": "sha1-6SQ0v6XqjBn0HN/UAddBo8gZ2Jc=",
1512 | "dev": true
1513 | },
1514 | "process-nextick-args": {
1515 | "version": "2.0.1",
1516 | "resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-2.0.1.tgz",
1517 | "integrity": "sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag=="
1518 | },
1519 | "proxy-addr": {
1520 | "version": "2.0.6",
1521 | "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.6.tgz",
1522 | "integrity": "sha512-dh/frvCBVmSsDYzw6n926jv974gddhkFPfiN8hPOi30Wax25QZyZEGveluCgliBnqmuM+UJmBErbAUFIoDbjOw==",
1523 | "requires": {
1524 | "forwarded": "~0.1.2",
1525 | "ipaddr.js": "1.9.1"
1526 | }
1527 | },
1528 | "pstree.remy": {
1529 | "version": "1.1.8",
1530 | "resolved": "https://registry.npmjs.org/pstree.remy/-/pstree.remy-1.1.8.tgz",
1531 | "integrity": "sha512-77DZwxQmxKnu3aR542U+X8FypNzbfJ+C5XQDk3uWjWxn6151aIMGthWYRXTqT1E5oJvg+ljaa2OJi+VfvCOQ8w==",
1532 | "dev": true
1533 | },
1534 | "pump": {
1535 | "version": "3.0.0",
1536 | "resolved": "https://registry.npmjs.org/pump/-/pump-3.0.0.tgz",
1537 | "integrity": "sha512-LwZy+p3SFs1Pytd/jYct4wpv49HiYCqd9Rlc5ZVdk0V+8Yzv6jR5Blk3TRmPL1ft69TxP0IMZGJ+WPFU2BFhww==",
1538 | "dev": true,
1539 | "requires": {
1540 | "end-of-stream": "^1.1.0",
1541 | "once": "^1.3.1"
1542 | }
1543 | },
1544 | "pupa": {
1545 | "version": "2.1.1",
1546 | "resolved": "https://registry.npmjs.org/pupa/-/pupa-2.1.1.tgz",
1547 | "integrity": "sha512-l1jNAspIBSFqbT+y+5FosojNpVpF94nlI+wDUpqP9enwOTfHx9f0gh5nB96vl+6yTpsJsypeNrwfzPrKuHB41A==",
1548 | "dev": true,
1549 | "requires": {
1550 | "escape-goat": "^2.0.0"
1551 | }
1552 | },
1553 | "qs": {
1554 | "version": "6.7.0",
1555 | "resolved": "https://registry.npmjs.org/qs/-/qs-6.7.0.tgz",
1556 | "integrity": "sha512-VCdBRNFTX1fyE7Nb6FYoURo/SPe62QCaAyzJvUjwRaIsc+NePBEniHlvxFmmX56+HZphIGtV0XeCirBtpDrTyQ=="
1557 | },
1558 | "range-parser": {
1559 | "version": "1.2.1",
1560 | "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz",
1561 | "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg=="
1562 | },
1563 | "raw-body": {
1564 | "version": "2.4.0",
1565 | "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.4.0.tgz",
1566 | "integrity": "sha512-4Oz8DUIwdvoa5qMJelxipzi/iJIi40O5cGV1wNYp5hvZP8ZN0T+jiNkL0QepXs+EsQ9XJ8ipEDoiH70ySUJP3Q==",
1567 | "requires": {
1568 | "bytes": "3.1.0",
1569 | "http-errors": "1.7.2",
1570 | "iconv-lite": "0.4.24",
1571 | "unpipe": "1.0.0"
1572 | }
1573 | },
1574 | "rc": {
1575 | "version": "1.2.8",
1576 | "resolved": "https://registry.npmjs.org/rc/-/rc-1.2.8.tgz",
1577 | "integrity": "sha512-y3bGgqKj3QBdxLbLkomlohkvsA8gdAiUQlSBJnBhfn+BPxg4bc62d8TcBW15wavDfgexCgccckhcZvywyQYPOw==",
1578 | "dev": true,
1579 | "requires": {
1580 | "deep-extend": "^0.6.0",
1581 | "ini": "~1.3.0",
1582 | "minimist": "^1.2.0",
1583 | "strip-json-comments": "~2.0.1"
1584 | }
1585 | },
1586 | "readable-stream": {
1587 | "version": "2.3.7",
1588 | "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.7.tgz",
1589 | "integrity": "sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw==",
1590 | "requires": {
1591 | "core-util-is": "~1.0.0",
1592 | "inherits": "~2.0.3",
1593 | "isarray": "~1.0.0",
1594 | "process-nextick-args": "~2.0.0",
1595 | "safe-buffer": "~5.1.1",
1596 | "string_decoder": "~1.1.1",
1597 | "util-deprecate": "~1.0.1"
1598 | }
1599 | },
1600 | "readdirp": {
1601 | "version": "3.5.0",
1602 | "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.5.0.tgz",
1603 | "integrity": "sha512-cMhu7c/8rdhkHXWsY+osBhfSy0JikwpHK/5+imo+LpeasTF8ouErHrlYkwT0++njiyuDvc7OFY5T3ukvZ8qmFQ==",
1604 | "dev": true,
1605 | "requires": {
1606 | "picomatch": "^2.2.1"
1607 | }
1608 | },
1609 | "regexp-clone": {
1610 | "version": "1.0.0",
1611 | "resolved": "https://registry.npmjs.org/regexp-clone/-/regexp-clone-1.0.0.tgz",
1612 | "integrity": "sha512-TuAasHQNamyyJ2hb97IuBEif4qBHGjPHBS64sZwytpLEqtBQ1gPJTnOaQ6qmpET16cK14kkjbazl6+p0RRv0yw=="
1613 | },
1614 | "registry-auth-token": {
1615 | "version": "4.2.1",
1616 | "resolved": "https://registry.npmjs.org/registry-auth-token/-/registry-auth-token-4.2.1.tgz",
1617 | "integrity": "sha512-6gkSb4U6aWJB4SF2ZvLb76yCBjcvufXBqvvEx1HbmKPkutswjW1xNVRY0+daljIYRbogN7O0etYSlbiaEQyMyw==",
1618 | "dev": true,
1619 | "requires": {
1620 | "rc": "^1.2.8"
1621 | }
1622 | },
1623 | "registry-url": {
1624 | "version": "5.1.0",
1625 | "resolved": "https://registry.npmjs.org/registry-url/-/registry-url-5.1.0.tgz",
1626 | "integrity": "sha512-8acYXXTI0AkQv6RAOjE3vOaIXZkT9wo4LOFbBKYQEEnnMNBpKqdUrI6S4NT0KPIo/WVvJ5tE/X5LF/TQUf0ekw==",
1627 | "dev": true,
1628 | "requires": {
1629 | "rc": "^1.2.8"
1630 | }
1631 | },
1632 | "responselike": {
1633 | "version": "1.0.2",
1634 | "resolved": "https://registry.npmjs.org/responselike/-/responselike-1.0.2.tgz",
1635 | "integrity": "sha1-kYcg7ztjHFZCvgaPFa3lpG9Loec=",
1636 | "dev": true,
1637 | "requires": {
1638 | "lowercase-keys": "^1.0.0"
1639 | }
1640 | },
1641 | "safe-buffer": {
1642 | "version": "5.1.2",
1643 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz",
1644 | "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g=="
1645 | },
1646 | "safer-buffer": {
1647 | "version": "2.1.2",
1648 | "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz",
1649 | "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg=="
1650 | },
1651 | "saslprep": {
1652 | "version": "1.0.3",
1653 | "resolved": "https://registry.npmjs.org/saslprep/-/saslprep-1.0.3.tgz",
1654 | "integrity": "sha512-/MY/PEMbk2SuY5sScONwhUDsV2p77Znkb/q3nSVstq/yQzYJOH/Azh29p9oJLsl3LnQwSvZDKagDGBsBwSooag==",
1655 | "optional": true,
1656 | "requires": {
1657 | "sparse-bitfield": "^3.0.3"
1658 | }
1659 | },
1660 | "semver": {
1661 | "version": "5.7.1",
1662 | "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz",
1663 | "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ=="
1664 | },
1665 | "semver-diff": {
1666 | "version": "3.1.1",
1667 | "resolved": "https://registry.npmjs.org/semver-diff/-/semver-diff-3.1.1.tgz",
1668 | "integrity": "sha512-GX0Ix/CJcHyB8c4ykpHGIAvLyOwOobtM/8d+TQkAd81/bEjgPHrfba41Vpesr7jX/t8Uh+R3EX9eAS5be+jQYg==",
1669 | "dev": true,
1670 | "requires": {
1671 | "semver": "^6.3.0"
1672 | },
1673 | "dependencies": {
1674 | "semver": {
1675 | "version": "6.3.0",
1676 | "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz",
1677 | "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==",
1678 | "dev": true
1679 | }
1680 | }
1681 | },
1682 | "send": {
1683 | "version": "0.17.1",
1684 | "resolved": "https://registry.npmjs.org/send/-/send-0.17.1.tgz",
1685 | "integrity": "sha512-BsVKsiGcQMFwT8UxypobUKyv7irCNRHk1T0G680vk88yf6LBByGcZJOTJCrTP2xVN6yI+XjPJcNuE3V4fT9sAg==",
1686 | "requires": {
1687 | "debug": "2.6.9",
1688 | "depd": "~1.1.2",
1689 | "destroy": "~1.0.4",
1690 | "encodeurl": "~1.0.2",
1691 | "escape-html": "~1.0.3",
1692 | "etag": "~1.8.1",
1693 | "fresh": "0.5.2",
1694 | "http-errors": "~1.7.2",
1695 | "mime": "1.6.0",
1696 | "ms": "2.1.1",
1697 | "on-finished": "~2.3.0",
1698 | "range-parser": "~1.2.1",
1699 | "statuses": "~1.5.0"
1700 | },
1701 | "dependencies": {
1702 | "ms": {
1703 | "version": "2.1.1",
1704 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.1.tgz",
1705 | "integrity": "sha512-tgp+dl5cGk28utYktBsrFqA7HKgrhgPsg6Z/EfhWI4gl1Hwq8B/GmY/0oXZ6nF8hDVesS/FpnYaD/kOWhYQvyg=="
1706 | }
1707 | }
1708 | },
1709 | "serve-static": {
1710 | "version": "1.14.1",
1711 | "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.14.1.tgz",
1712 | "integrity": "sha512-JMrvUwE54emCYWlTI+hGrGv5I8dEwmco/00EvkzIIsR7MqrHonbD9pO2MOfFnpFntl7ecpZs+3mW+XbQZu9QCg==",
1713 | "requires": {
1714 | "encodeurl": "~1.0.2",
1715 | "escape-html": "~1.0.3",
1716 | "parseurl": "~1.3.3",
1717 | "send": "0.17.1"
1718 | }
1719 | },
1720 | "setprototypeof": {
1721 | "version": "1.1.1",
1722 | "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.1.tgz",
1723 | "integrity": "sha512-JvdAWfbXeIGaZ9cILp38HntZSFSo3mWg6xGcJJsd+d4aRMOqauag1C63dJfDw7OaMYwEbHMOxEZ1lqVRYP2OAw=="
1724 | },
1725 | "shebang-command": {
1726 | "version": "2.0.0",
1727 | "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz",
1728 | "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==",
1729 | "dev": true,
1730 | "requires": {
1731 | "shebang-regex": "^3.0.0"
1732 | }
1733 | },
1734 | "shebang-regex": {
1735 | "version": "3.0.0",
1736 | "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz",
1737 | "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==",
1738 | "dev": true
1739 | },
1740 | "sift": {
1741 | "version": "13.5.2",
1742 | "resolved": "https://registry.npmjs.org/sift/-/sift-13.5.2.tgz",
1743 | "integrity": "sha512-+gxdEOMA2J+AI+fVsCqeNn7Tgx3M9ZN9jdi95939l1IJ8cZsqS8sqpJyOkic2SJk+1+98Uwryt/gL6XDaV+UZA=="
1744 | },
1745 | "signal-exit": {
1746 | "version": "3.0.3",
1747 | "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.3.tgz",
1748 | "integrity": "sha512-VUJ49FC8U1OxwZLxIbTTrDvLnf/6TDgxZcK8wxR8zs13xpx7xbG60ndBlhNrFi2EMuFRoeDoJO7wthSLq42EjA==",
1749 | "dev": true
1750 | },
1751 | "sliced": {
1752 | "version": "1.0.1",
1753 | "resolved": "https://registry.npmjs.org/sliced/-/sliced-1.0.1.tgz",
1754 | "integrity": "sha1-CzpmK10Ewxd7GSa+qCsD+Dei70E="
1755 | },
1756 | "sparse-bitfield": {
1757 | "version": "3.0.3",
1758 | "resolved": "https://registry.npmjs.org/sparse-bitfield/-/sparse-bitfield-3.0.3.tgz",
1759 | "integrity": "sha1-/0rm5oZWBWuks+eSqzM004JzyhE=",
1760 | "optional": true,
1761 | "requires": {
1762 | "memory-pager": "^1.0.2"
1763 | }
1764 | },
1765 | "statuses": {
1766 | "version": "1.5.0",
1767 | "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.5.0.tgz",
1768 | "integrity": "sha1-Fhx9rBd2Wf2YEfQ3cfqZOBR4Yow="
1769 | },
1770 | "string-width": {
1771 | "version": "4.2.2",
1772 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.2.tgz",
1773 | "integrity": "sha512-XBJbT3N4JhVumXE0eoLU9DCjcaF92KLNqTmFCnG1pf8duUxFGwtP6AD6nkjw9a3IdiRtL3E2w3JDiE/xi3vOeA==",
1774 | "dev": true,
1775 | "requires": {
1776 | "emoji-regex": "^8.0.0",
1777 | "is-fullwidth-code-point": "^3.0.0",
1778 | "strip-ansi": "^6.0.0"
1779 | }
1780 | },
1781 | "string_decoder": {
1782 | "version": "1.1.1",
1783 | "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz",
1784 | "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==",
1785 | "requires": {
1786 | "safe-buffer": "~5.1.0"
1787 | }
1788 | },
1789 | "strip-ansi": {
1790 | "version": "6.0.0",
1791 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.0.tgz",
1792 | "integrity": "sha512-AuvKTrTfQNYNIctbR1K/YGTR1756GycPsg7b9bdV9Duqur4gv6aKqHXah67Z8ImS7WEz5QVcOtlfW2rZEugt6w==",
1793 | "dev": true,
1794 | "requires": {
1795 | "ansi-regex": "^5.0.0"
1796 | }
1797 | },
1798 | "strip-json-comments": {
1799 | "version": "2.0.1",
1800 | "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-2.0.1.tgz",
1801 | "integrity": "sha1-PFMZQukIwml8DsNEhYwobHygpgo=",
1802 | "dev": true
1803 | },
1804 | "supports-color": {
1805 | "version": "5.5.0",
1806 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz",
1807 | "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==",
1808 | "dev": true,
1809 | "requires": {
1810 | "has-flag": "^3.0.0"
1811 | }
1812 | },
1813 | "term-size": {
1814 | "version": "2.2.1",
1815 | "resolved": "https://registry.npmjs.org/term-size/-/term-size-2.2.1.tgz",
1816 | "integrity": "sha512-wK0Ri4fOGjv/XPy8SBHZChl8CM7uMc5VML7SqiQ0zG7+J5Vr+RMQDoHa2CNT6KHUnTGIXH34UDMkPzAUyapBZg==",
1817 | "dev": true
1818 | },
1819 | "to-readable-stream": {
1820 | "version": "1.0.0",
1821 | "resolved": "https://registry.npmjs.org/to-readable-stream/-/to-readable-stream-1.0.0.tgz",
1822 | "integrity": "sha512-Iq25XBt6zD5npPhlLVXGFN3/gyR2/qODcKNNyTMd4vbm39HUaOiAM4PMq0eMVC/Tkxz+Zjdsc55g9yyz+Yq00Q==",
1823 | "dev": true
1824 | },
1825 | "to-regex-range": {
1826 | "version": "5.0.1",
1827 | "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz",
1828 | "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==",
1829 | "dev": true,
1830 | "requires": {
1831 | "is-number": "^7.0.0"
1832 | }
1833 | },
1834 | "toidentifier": {
1835 | "version": "1.0.0",
1836 | "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.0.tgz",
1837 | "integrity": "sha512-yaOH/Pk/VEhBWWTlhI+qXxDFXlejDGcQipMlyxda9nthulaxLZUNcUqFxokp0vcYnvteJln5FNQDRrxj3YcbVw=="
1838 | },
1839 | "touch": {
1840 | "version": "3.1.0",
1841 | "resolved": "https://registry.npmjs.org/touch/-/touch-3.1.0.tgz",
1842 | "integrity": "sha512-WBx8Uy5TLtOSRtIq+M03/sKDrXCLHxwDcquSP2c43Le03/9serjQBIztjRz6FkJez9D/hleyAXTBGLwwZUw9lA==",
1843 | "dev": true,
1844 | "requires": {
1845 | "nopt": "~1.0.10"
1846 | }
1847 | },
1848 | "type-fest": {
1849 | "version": "0.8.1",
1850 | "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.8.1.tgz",
1851 | "integrity": "sha512-4dbzIzqvjtgiM5rw1k5rEHtBANKmdudhGyBEajN01fEyhaAIhsoKNy6y7+IN93IfpFtwY9iqi7kD+xwKhQsNJA==",
1852 | "dev": true
1853 | },
1854 | "type-is": {
1855 | "version": "1.6.18",
1856 | "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz",
1857 | "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==",
1858 | "requires": {
1859 | "media-typer": "0.3.0",
1860 | "mime-types": "~2.1.24"
1861 | }
1862 | },
1863 | "typedarray-to-buffer": {
1864 | "version": "3.1.5",
1865 | "resolved": "https://registry.npmjs.org/typedarray-to-buffer/-/typedarray-to-buffer-3.1.5.tgz",
1866 | "integrity": "sha512-zdu8XMNEDepKKR+XYOXAVPtWui0ly0NtohUscw+UmaHiAWT8hrV1rr//H6V+0DvJ3OQ19S979M0laLfX8rm82Q==",
1867 | "dev": true,
1868 | "requires": {
1869 | "is-typedarray": "^1.0.0"
1870 | }
1871 | },
1872 | "undefsafe": {
1873 | "version": "2.0.3",
1874 | "resolved": "https://registry.npmjs.org/undefsafe/-/undefsafe-2.0.3.tgz",
1875 | "integrity": "sha512-nrXZwwXrD/T/JXeygJqdCO6NZZ1L66HrxM/Z7mIq2oPanoN0F1nLx3lwJMu6AwJY69hdixaFQOuoYsMjE5/C2A==",
1876 | "dev": true,
1877 | "requires": {
1878 | "debug": "^2.2.0"
1879 | }
1880 | },
1881 | "unique-string": {
1882 | "version": "2.0.0",
1883 | "resolved": "https://registry.npmjs.org/unique-string/-/unique-string-2.0.0.tgz",
1884 | "integrity": "sha512-uNaeirEPvpZWSgzwsPGtU2zVSTrn/8L5q/IexZmH0eH6SA73CmAA5U4GwORTxQAZs95TAXLNqeLoPPNO5gZfWg==",
1885 | "dev": true,
1886 | "requires": {
1887 | "crypto-random-string": "^2.0.0"
1888 | }
1889 | },
1890 | "unpipe": {
1891 | "version": "1.0.0",
1892 | "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz",
1893 | "integrity": "sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw="
1894 | },
1895 | "update-notifier": {
1896 | "version": "4.1.3",
1897 | "resolved": "https://registry.npmjs.org/update-notifier/-/update-notifier-4.1.3.tgz",
1898 | "integrity": "sha512-Yld6Z0RyCYGB6ckIjffGOSOmHXj1gMeE7aROz4MG+XMkmixBX4jUngrGXNYz7wPKBmtoD4MnBa2Anu7RSKht/A==",
1899 | "dev": true,
1900 | "requires": {
1901 | "boxen": "^4.2.0",
1902 | "chalk": "^3.0.0",
1903 | "configstore": "^5.0.1",
1904 | "has-yarn": "^2.1.0",
1905 | "import-lazy": "^2.1.0",
1906 | "is-ci": "^2.0.0",
1907 | "is-installed-globally": "^0.3.1",
1908 | "is-npm": "^4.0.0",
1909 | "is-yarn-global": "^0.3.0",
1910 | "latest-version": "^5.0.0",
1911 | "pupa": "^2.0.1",
1912 | "semver-diff": "^3.1.1",
1913 | "xdg-basedir": "^4.0.0"
1914 | }
1915 | },
1916 | "url-parse-lax": {
1917 | "version": "3.0.0",
1918 | "resolved": "https://registry.npmjs.org/url-parse-lax/-/url-parse-lax-3.0.0.tgz",
1919 | "integrity": "sha1-FrXK/Afb42dsGxmZF3gj1lA6yww=",
1920 | "dev": true,
1921 | "requires": {
1922 | "prepend-http": "^2.0.0"
1923 | }
1924 | },
1925 | "url-template": {
1926 | "version": "2.0.8",
1927 | "resolved": "https://registry.npmjs.org/url-template/-/url-template-2.0.8.tgz",
1928 | "integrity": "sha1-/FZaPMy/93MMd19WQflVV5FDnyE="
1929 | },
1930 | "util-deprecate": {
1931 | "version": "1.0.2",
1932 | "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz",
1933 | "integrity": "sha1-RQ1Nyfpw3nMnYvvS1KKJgUGaDM8="
1934 | },
1935 | "utils-merge": {
1936 | "version": "1.0.1",
1937 | "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz",
1938 | "integrity": "sha1-n5VxD1CiZ5R7LMwSR0HBAoQn5xM="
1939 | },
1940 | "uuid": {
1941 | "version": "8.3.2",
1942 | "resolved": "https://registry.npmjs.org/uuid/-/uuid-8.3.2.tgz",
1943 | "integrity": "sha512-+NYs2QeMWy+GWFOEm9xnn6HCDp0l7QBD7ml8zLUmJ+93Q5NF0NocErnwkTkXVFNiX3/fpC6afS8Dhb/gz7R7eg=="
1944 | },
1945 | "validator": {
1946 | "version": "13.6.0",
1947 | "resolved": "https://registry.npmjs.org/validator/-/validator-13.6.0.tgz",
1948 | "integrity": "sha512-gVgKbdbHgtxpRyR8K0O6oFZPhhB5tT1jeEHZR0Znr9Svg03U0+r9DXWMrnRAB+HtCStDQKlaIZm42tVsVjqtjg=="
1949 | },
1950 | "vary": {
1951 | "version": "1.1.2",
1952 | "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz",
1953 | "integrity": "sha1-IpnwLG3tMNSllhsLn3RSShj2NPw="
1954 | },
1955 | "which": {
1956 | "version": "2.0.2",
1957 | "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz",
1958 | "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==",
1959 | "dev": true,
1960 | "requires": {
1961 | "isexe": "^2.0.0"
1962 | }
1963 | },
1964 | "widest-line": {
1965 | "version": "3.1.0",
1966 | "resolved": "https://registry.npmjs.org/widest-line/-/widest-line-3.1.0.tgz",
1967 | "integrity": "sha512-NsmoXalsWVDMGupxZ5R08ka9flZjjiLvHVAWYOKtiKM8ujtZWr9cRffak+uSE48+Ob8ObalXpwyeUiyDD6QFgg==",
1968 | "dev": true,
1969 | "requires": {
1970 | "string-width": "^4.0.0"
1971 | }
1972 | },
1973 | "wrappy": {
1974 | "version": "1.0.2",
1975 | "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz",
1976 | "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=",
1977 | "dev": true
1978 | },
1979 | "write-file-atomic": {
1980 | "version": "3.0.3",
1981 | "resolved": "https://registry.npmjs.org/write-file-atomic/-/write-file-atomic-3.0.3.tgz",
1982 | "integrity": "sha512-AvHcyZ5JnSfq3ioSyjrBkH9yW4m7Ayk8/9My/DD9onKeu/94fwrMocemO2QAJFAlnnDN+ZDS+ZjAR5ua1/PV/Q==",
1983 | "dev": true,
1984 | "requires": {
1985 | "imurmurhash": "^0.1.4",
1986 | "is-typedarray": "^1.0.0",
1987 | "signal-exit": "^3.0.2",
1988 | "typedarray-to-buffer": "^3.1.5"
1989 | }
1990 | },
1991 | "xdg-basedir": {
1992 | "version": "4.0.0",
1993 | "resolved": "https://registry.npmjs.org/xdg-basedir/-/xdg-basedir-4.0.0.tgz",
1994 | "integrity": "sha512-PSNhEJDejZYV7h50BohL09Er9VaIefr2LMAf3OEmpCkjOi34eYyQYAXUTjEQtZJTKcF0E2UKTh+osDLsgNim9Q==",
1995 | "dev": true
1996 | },
1997 | "yallist": {
1998 | "version": "4.0.0",
1999 | "resolved": "https://registry.npmjs.org/yallist/-/yallist-4.0.0.tgz",
2000 | "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A=="
2001 | }
2002 | }
2003 | }
2004 |
--------------------------------------------------------------------------------
/server/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "server",
3 | "version": "1.0.0",
4 | "description": "",
5 | "main": "index.js",
6 | "scripts": {
7 | "start": "node src/index.js",
8 | "test": "echo \"Error: no test specified\" && exit 1",
9 | "dev": "env-cmd -f ./config/dev.env nodemon src/index.js"
10 | },
11 | "author": "",
12 | "license": "ISC",
13 | "dependencies": {
14 | "axios": "^0.21.1",
15 | "bcryptjs": "^2.4.3",
16 | "cors": "^2.8.5",
17 | "dotenv": "^10.0.0",
18 | "express": "^4.17.1",
19 | "googleapis": "^75.0.0",
20 | "jsonwebtoken": "^8.5.1",
21 | "mongoose": "^5.12.12",
22 | "validator": "^13.6.0"
23 | },
24 | "devDependencies": {
25 | "env-cmd": "^10.1.0",
26 | "nodemon": "^2.0.7"
27 | }
28 | }
29 |
--------------------------------------------------------------------------------
/server/src/app.js:
--------------------------------------------------------------------------------
1 | const express = require('express')
2 | require('./db/mongoose')
3 | const cors = require('cors')
4 | const userRouter = require('./routers/user')
5 | const roomRouter = require('./routers/room')
6 | const adminRouter = require('./routers/roomAdmin')
7 | const googleLoginRouter = require('./oauth2/googleAuthRouters')
8 | const path = require('path')
9 |
10 | const app = express()
11 |
12 | // app.use(express.static(path.join(__dirname, '../../client/build')));
13 |
14 | app.use(express.json())
15 | app.use(cors())
16 | app.use(userRouter)
17 | app.use(roomRouter)
18 | app.use(adminRouter)
19 | app.use(googleLoginRouter)
20 |
21 | // app.get('*', (req, res) => {
22 | // res.sendFile(path.join(__dirname, '../../client/build/index.html'))
23 | // })
24 |
25 | module.exports = app
--------------------------------------------------------------------------------
/server/src/db/mongoose.js:
--------------------------------------------------------------------------------
1 | const mongoose = require('mongoose')
2 | require("dotenv").config();
3 |
4 | mongoose.connect(process.env.MONGODB_URL, {
5 | useNewUrlParser: true,
6 | useCreateIndex: true,
7 | useUnifiedTopology: true,
8 | }).then(()=>{
9 | console.log('Database connected')
10 | }).catch((e)=>{
11 | console.log('Error')
12 | console.log(e)
13 | })
--------------------------------------------------------------------------------
/server/src/index.js:
--------------------------------------------------------------------------------
1 | const app = require('./app')
2 | require("dotenv").config();
3 | const port = process.env.PORT
4 |
5 | app.listen(port, () => {
6 | console.log('Server is up on port ' + port)
7 | })
--------------------------------------------------------------------------------
/server/src/middleware/auth.js:
--------------------------------------------------------------------------------
1 | const jwt = require('jsonwebtoken')
2 | const User = require('../models/user')
3 | require("dotenv").config();
4 |
5 | const auth = async (req, res, next) => {
6 | try {
7 | const token = req.header('Authorization').replace('Bearer ', '')
8 | const decoded = jwt.verify(token, process.env.JWT_SECRET)
9 | const user = await User.findOne({ _id: decoded._id, 'tokens.token': token })
10 | if (!user) throw new Error()
11 | req.token = token
12 | req.user = user
13 | next()
14 | } catch (err) {
15 | res.status(401).json({ error: 'Please authenticate.' })
16 | }
17 | }
18 |
19 | module.exports = auth
--------------------------------------------------------------------------------
/server/src/middleware/banStatus.js:
--------------------------------------------------------------------------------
1 | const banStatus = async (req, res, next) => {
2 | try {
3 | if (req.user.isBanned) {
4 | res.status(401).json({ error: `Banned` })
5 | } else {
6 | next()
7 | }
8 | } catch (err) {
9 | res.status(401).json({ error: `${err}` })
10 | }
11 | }
12 |
13 | module.exports = banStatus
--------------------------------------------------------------------------------
/server/src/middleware/roomAdmin.js:
--------------------------------------------------------------------------------
1 | const User = require('../models/user')
2 | const Room = require('../models/room')
3 |
4 | const admin = async (req, res, next) => {
5 | try {
6 | const roomID = req.user.roomID
7 | const room = await Room.findOne({roomID})
8 | if(room.roomAdmin.equals(req.user._id)){
9 | next()
10 | }
11 | else res.status(403).json({error: 'Not an admin'})
12 | } catch (err) {
13 | res.status(401).json({ error: `${err}` })
14 | }
15 | }
16 |
17 | module.exports = admin
--------------------------------------------------------------------------------
/server/src/models/room.js:
--------------------------------------------------------------------------------
1 | const mongoose = require("mongoose");
2 | require("dotenv").config();
3 |
4 | const roomSchema = new mongoose.Schema({
5 | roomID: {
6 | type: String,
7 | required: true
8 | },
9 | usersInRoom: {
10 | type: Number,
11 | default: 0
12 | },
13 | users: [{
14 | userID: {
15 | type: mongoose.Schema.Types.ObjectId,
16 | ref: 'User'
17 | }
18 | }],
19 | roomFull: {
20 | type: Boolean,
21 | default: false,
22 | },
23 | roomLocked: {
24 | type: Boolean,
25 | default: false
26 | },
27 | roomAdmin: {
28 | type: mongoose.Schema.Types.ObjectId,
29 | },
30 | costPerMember: {
31 | type: Number,
32 | default: process.env.LITECODE_PRICE
33 | },
34 | toPay: {
35 | type: Number,
36 | default: process.env.LITECODE_PRICE
37 | }
38 | })
39 |
40 | roomSchema.pre('save', async function (next) {
41 | if(this.usersInRoom){
42 | const price = Math.trunc((process.env.LITECODE_PRICE)/(this.usersInRoom))
43 | this.costPerMember = Math.ceil(price/ 10) * 10
44 | } else{
45 | this.costPerMember = process.env.LITECODE_PRICE
46 | }
47 | const priceToPay = Math.trunc((process.env.LITECODE_PRICE)/(this.usersInRoom + 1))
48 | this.toPay = Math.ceil(priceToPay/ 10) * 10
49 | next()
50 | })
51 |
52 | const Room = mongoose.model('Room', roomSchema)
53 |
54 | module.exports = Room
--------------------------------------------------------------------------------
/server/src/models/user.js:
--------------------------------------------------------------------------------
1 | const mongoose = require('mongoose')
2 | const validator = require('validator')
3 | const jwt = require('jsonwebtoken')
4 | const Room = require('../models/room')
5 | require("dotenv").config();
6 |
7 | const userSchema = new mongoose.Schema({
8 | name: {
9 | type: String,
10 | required: true,
11 | trim: true
12 | },
13 | email: {
14 | type: String,
15 | unique: true,
16 | trim: true,
17 | required: true,
18 | lowercase: true,
19 | },
20 | phoneNo: {
21 | type: String,
22 | unique: true,
23 | validate(value) {
24 | if (!validator.isMobilePhone(value, ['en-IN'])) {
25 | throw new Error('Invalid Mobile')
26 | }
27 | }
28 | },
29 | avatar: {
30 | type: String
31 | },
32 | inRoom: {
33 | type: Boolean,
34 | required: true,
35 | default: false
36 | },
37 | isBanned:{
38 | type: Boolean,
39 | default: false
40 | },
41 | banTime:{
42 | type: Number,
43 | default: Date.now
44 | },
45 | roomID: {
46 | type: String,
47 | required: true,
48 | default: "nil"
49 | },
50 | tokens: [{
51 | token: {
52 | type: String,
53 | require: true
54 | }
55 | }]
56 | })
57 |
58 | userSchema.methods.generateAuthToken = async function () {
59 | const token = jwt.sign({ _id: this._id.toString() }, process.env.JWT_SECRET)
60 | this.tokens = this.tokens.concat({ token })
61 | await this.save()
62 | return token
63 | }
64 |
65 | userSchema.methods.toJSON = function () {
66 | const userObject = this.toObject()
67 | delete userObject.tokens
68 | delete userObject.__v
69 | return userObject
70 | }
71 |
72 | userSchema.pre('remove', async function (next) {
73 | if (this.inRoom) {
74 | roomID = this.roomID
75 | const room = await Room.findOne({ roomID })
76 |
77 | room.usersInRoom--
78 | room.roomFull = false
79 |
80 | room.users = room.users.filter((user) => {
81 | return !user.userID.equals(this._id)//If the id isn't equal, it stays in the array.
82 | })
83 |
84 | if (room.roomAdmin.equals(this._id)) {
85 | if (room.usersInRoom) {
86 | room.roomAdmin = room.users[0].userID
87 | }
88 | else {
89 | room.roomAdmin = null
90 | }
91 | }
92 |
93 | await room.save()
94 | }
95 | next()
96 | })
97 |
98 | const User = mongoose.model('User', userSchema)
99 |
100 | module.exports = User
--------------------------------------------------------------------------------
/server/src/oauth2/googleAuthHelpers.js:
--------------------------------------------------------------------------------
1 | const axios = require('axios')
2 | require("dotenv").config();
3 |
4 | const { google } = require('googleapis')
5 |
6 | const oauth2Client = new google.auth.OAuth2(
7 | process.env.GOOGLE_CLIENT_ID,
8 | process.env.GOOGLE_CLIENT_SECRET,
9 | process.env.GOOGLE_REDIRECT_URL
10 | )
11 |
12 | google.options({
13 | auth: oauth2Client
14 | })
15 |
16 | const googleLoginUrl = oauth2Client.generateAuthUrl({
17 | scope: [
18 | 'https://www.googleapis.com/auth/userinfo.email',
19 | 'https://www.googleapis.com/auth/userinfo.profile'
20 | ],
21 | response_type: 'code',
22 | prompt: 'consent'
23 | })
24 |
25 | const getAccessTokenFromCode = async (code) => {
26 | try {
27 | const { tokens } = await oauth2Client.getToken(code)
28 | oauth2Client.setCredentials(tokens)
29 | return tokens
30 | } catch (e) {
31 | throw new Error('Unable to fetch access token.')
32 | }
33 | }
34 |
35 | const getUserData = async (access_token) => {
36 | const { data } = await axios({
37 | url: 'https://www.googleapis.com/oauth2/v2/userinfo',
38 | method: 'get',
39 | headers: {
40 | Authorization: `Bearer ${access_token}`
41 | }
42 | })
43 | return data
44 | }
45 |
46 | module.exports = { googleLoginUrl, getAccessTokenFromCode, getUserData }
--------------------------------------------------------------------------------
/server/src/oauth2/googleAuthRouters.js:
--------------------------------------------------------------------------------
1 | const express = require('express')
2 | const router = new express.Router()
3 | const { googleLoginUrl, getAccessTokenFromCode, getUserData } = require('./googleAuthHelpers')
4 | const User = require('../models/user')
5 | const auth = require('../middleware/auth')
6 |
7 | router.get('/auth/google', async (req, res) => {
8 | res.json({ googleLoginUrl })
9 | })
10 |
11 | router.get('/auth/google/redirect', async (req, res) => {
12 | const code = req.query.code
13 |
14 | const { access_token } = await getAccessTokenFromCode(code)
15 |
16 | const data = await getUserData(access_token)
17 |
18 | if(!data.email.endsWith("bits-pilani.ac.in")){
19 | return res.redirect('https://litecode.bitsacm.in')
20 | }
21 |
22 | const user = await User.findOne({ email: data.email })
23 |
24 | if (!user) {
25 | const split = data.name.split(' ')
26 | var str = ""
27 | for (var i = 0; i < split.length; i++) {
28 | str = str + split[i].charAt(0).toUpperCase() + split[i].substring(1).toLowerCase() + ' '
29 | }
30 |
31 | const user = new User({
32 | email: data.email,
33 | name: str,
34 | avatar: data.picture
35 | })
36 |
37 | try {
38 | await user.save()
39 | const token = await user.generateAuthToken()
40 |
41 | res.cookie('jwt', token)
42 | res.redirect(req.query.state || 'https://litecode.bitsacm.in')
43 |
44 | } catch (error) {
45 | res.status(400).send(error)
46 | }
47 | } else {
48 | try {
49 | await user.save()
50 | const token = await user.generateAuthToken()
51 | res.cookie('jwt', token)
52 | res.redirect(req.query.state || 'https://litecode.bitsacm.in')
53 |
54 | } catch (error) {
55 | res.status(400).send(error)
56 | }
57 | }
58 | })
59 |
60 | router.post('/auth/logout', auth, async (req, res) => {
61 | try {
62 | req.user.tokens = req.user.tokens.filter((token) => {
63 | return token.token !== req.token
64 | })
65 | await req.user.save()
66 | res.send()
67 | } catch (error) {
68 | res.status(500).send()
69 | }
70 | })
71 |
72 | module.exports = router
--------------------------------------------------------------------------------
/server/src/routers/room.js:
--------------------------------------------------------------------------------
1 | const express = require('express')
2 | const router = new express.Router()
3 | const Room = require('../models/room')
4 | const crypto = require("crypto")
5 | const auth = require('../middleware/auth')
6 | const banStatus = require('../middleware/banStatus')
7 |
8 | // @desc Create New Room
9 | // @access Private
10 | // query params -> /createRoom?roomName=randomGroup123
11 | // if no query, random 6 character hex string
12 | router.post('/createRoom', auth, banStatus, async (req, res) => {
13 | const query = req.query.roomName
14 | const room = new Room({
15 | users: []
16 | })
17 |
18 | if (query) {
19 | const roomDuplicate = await Room.findOne({ roomID: query })
20 | if (roomDuplicate) {
21 | return res.status(400).json({ error: 'room with that name already exists' })
22 | }
23 | room.roomID = query
24 | } else {
25 | while (1) {
26 | var id = crypto.randomBytes(3).toString('hex')
27 | var roomExists = await Room.findOne({ roomID: id })
28 | if (!roomExists) {
29 | break
30 | }
31 | }
32 | room.roomID = id
33 | }
34 | await room.save()
35 |
36 | if (req.user.inRoom) {
37 | const room2 = await Room.findOne({ roomID: req.user.roomID })
38 |
39 | room2.usersInRoom--
40 | room2.roomFull = false
41 |
42 | room2.users = room2.users.filter((user) => {
43 | return !user.userID.equals(req.user._id)//If the id isn't equal, it stays in the array.
44 | })
45 |
46 | if (room2.roomAdmin.equals(req.user._id)) {
47 | if (room2.usersInRoom) {
48 | room2.roomAdmin = room2.users[0].userID
49 | } else {
50 | room2.roomAdmin = null
51 | }
52 | }
53 | await room2.save()
54 | }
55 |
56 | const userID = req.user._id
57 | req.user.inRoom = true
58 | req.user.roomID = room.roomID
59 | room.usersInRoom++
60 | room.users = room.users.concat({ userID })
61 | room.roomAdmin = userID
62 |
63 | await req.user.save()
64 | await room.save()
65 |
66 | const user = req.user
67 | res.status(200).json({ user, room })
68 | })
69 |
70 | // @desc GET all rooms which are not full/locked
71 | // @access Private
72 | router.get('/rooms', auth, async (req, res) => {
73 | try {
74 | if (req.user.isBanned) {
75 | const timeElapsed = Date.now() - req.user.banTime
76 | if (timeElapsed > 2 * 24 * 3600 * 1000) {
77 | req.user.isBanned = false
78 | }
79 | await req.user.save()
80 | }
81 | await Room.deleteMany({ usersInRoom: 0 })
82 | const rooms = await Room.find({ roomFull: false, roomLocked: false }).sort({ usersInRoom: "desc" }).populate('users.userID')
83 | res.status(200).json(rooms)
84 | } catch (err) {
85 | res.status(400).json({ error: `${err}` })
86 | }
87 | })
88 |
89 | // @desc GET all rooms which are not full/locked and match the inputed query
90 | // @access Private
91 | // query params -> /searchRoom?roomName=randomGroup123
92 | router.get('/searchRoom', auth, async (req, res) => {
93 | const query = req.query.roomName
94 | try {
95 | if (query) {
96 | const rooms = await Room.find({ roomFull: false, roomLocked: false, roomID: new RegExp(query, 'i') }).sort({ usersInRoom: "desc" }).populate('users.userID')
97 | res.status(200).json(rooms)
98 | }
99 | else {
100 | res.redirect('/rooms')
101 | }
102 | } catch (err) {
103 | res.status(400).json({ error: `${err}` })
104 | }
105 | })
106 |
107 | // @desc GET room details
108 | // @access Private
109 | router.get('/room/:id', auth, async (req, res) => {
110 | const id = req.params.id
111 | const room = await Room.findOne({ roomID: id }).populate('users.userID')
112 | if (room) {
113 | res.status(200).json({ room })
114 | }
115 | else {
116 | res.status(400).json({ error: 'Room does not exist' })
117 | }
118 | })
119 |
120 | // @desc Join Room using roomID
121 | // @access Private
122 | router.post('/joinRoom/:id', auth, banStatus, async (req, res) => {
123 | try {
124 | if (req.user.inRoom) {
125 | return res.status(400).json({ error: 'You are already in a Room' })
126 | }
127 |
128 | const userID = req.user._id
129 | const id = req.params.id
130 | const room = await Room.findOne({ roomID: id })
131 | if (!room) {
132 | return res.status(400).json({ error: 'Room not found' })
133 | }
134 | if (room.roomFull) {
135 | return res.status(400).json({ error: 'Room full' })
136 | }
137 | else if (room.roomLocked) {
138 | return res.status(400).json({ error: 'Room Locked' })
139 | }
140 |
141 | req.user.inRoom = true
142 | req.user.roomID = id
143 | room.usersInRoom++
144 | room.users = room.users.concat({ userID })
145 | if (room.usersInRoom === 4) {
146 | room.roomFull = true
147 | } else if (room.usersInRoom === 1) {
148 | room.roomAdmin = userID
149 | }
150 |
151 | await req.user.save()
152 | await room.save()
153 |
154 | const user = req.user
155 | res.status(200).json({ user, room })
156 | } catch (err) {
157 | res.status(400).json({ error: `${err}` })
158 | }
159 | })
160 |
161 | // @desc Leave current room
162 | // @access Private
163 | router.post('/leaveRoom', auth, async (req, res) => {
164 | try {
165 | if (!req.user.inRoom) {
166 | return res.status(400).json({ error: 'You are not in a room' })
167 | }
168 |
169 | const room = await Room.findOne({ roomID: req.user.roomID })
170 |
171 | req.user.inRoom = false
172 | req.user.roomID = "nil"
173 | room.usersInRoom--
174 | room.roomFull = false
175 |
176 | room.users = room.users.filter((user) => {
177 | return !user.userID.equals(req.user._id)//If the id isn't equal, it stays in the array.
178 | })
179 |
180 | if (room.roomAdmin.equals(req.user._id)) {
181 | if (room.usersInRoom) {
182 | room.roomAdmin = room.users[0].userID
183 | }
184 | else {
185 | room.roomAdmin = null
186 | }
187 | }
188 |
189 | if (room.usersInRoom) {
190 | req.user.isBanned = true
191 | req.user.banTime = Date.now()
192 | }
193 |
194 | await req.user.save()
195 | await room.save()
196 |
197 | res.status(200).json({ msg: 'Room left' })
198 | } catch (err) {
199 | res.status(400).json({ error: `${err}` })
200 | }
201 | })
202 |
203 | module.exports = router
--------------------------------------------------------------------------------
/server/src/routers/roomAdmin.js:
--------------------------------------------------------------------------------
1 | const express = require('express')
2 | const router = new express.Router()
3 | const auth = require('../middleware/auth')
4 | const admin = require('../middleware/roomAdmin')
5 | const Room = require('../models/room')
6 | const User = require('../models/user')
7 |
8 | // @desc Lock/Unlock a room
9 | // @access Private
10 | router.patch('/lock', auth, admin, async (req, res) => {
11 | try {
12 | const room = await Room.findOne({ roomID: req.user.roomID })
13 | if (room.roomLocked) {
14 | room.roomLocked = false
15 | await room.save()
16 | res.json({ msg: 'room unlocked' })
17 | } else {
18 | room.roomLocked = true
19 | await room.save()
20 | res.json({ msg: 'room locked' })
21 | }
22 | } catch (err) {
23 | res.status(400).json({ error: `${err}` })
24 | }
25 | })
26 |
27 | // @desc Change Room Admin
28 | // @access Private
29 | router.patch('/roomAdmin/:id', auth, admin, async (req, res) => {
30 | try {
31 | const userID = req.params.id
32 | const user = await User.findOne({ _id: userID })
33 | const room = await Room.findOne({ roomID: req.user.roomID })
34 | if (!room.users.some(user => user.userID == userID)) {
35 | res.status(400).json({ error: 'User not in room' })
36 | } else {
37 | room.roomAdmin = userID
38 | await room.save()
39 | res.json({ msg: `${user.name} is the new admin` })
40 | }
41 | } catch (err) {
42 | res.status(400).json({ error: `${err}` })
43 | }
44 | })
45 |
46 | // @desc Remove user from a room
47 | // @access Private
48 | router.patch('/remove/:id', auth, admin, async (req, res) => {
49 | try {
50 | const userID = req.params.id
51 | const user = await User.findOne({ _id: userID })
52 | const room = await Room.findOne({ roomID: req.user.roomID })
53 | if (!room.users.some(user => user.userID == userID)) {
54 | res.status(400).json({ error: 'User not in room' })
55 | }
56 |
57 | user.inRoom = false
58 | user.roomID = "nil"
59 | room.usersInRoom--
60 | room.roomFull = false
61 |
62 | room.users = room.users.filter((user) => {
63 | return !user.userID.equals(userID)//If the id isn't equal, it stays in the array.
64 | })
65 |
66 | await user.save()
67 | await room.save()
68 |
69 | res.json({ msg: 'User removed' })
70 | } catch (err) {
71 | res.status(400).json({ error: `${err}` })
72 | }
73 | })
74 |
75 | module.exports = router
--------------------------------------------------------------------------------
/server/src/routers/user.js:
--------------------------------------------------------------------------------
1 | const express = require('express')
2 | const router = new express.Router()
3 | const auth = require('../middleware/auth')
4 | const User = require('../models/user')
5 |
6 | // @desc GET Current User Profile
7 | // @access Private
8 | router.get('/users/me', auth, async (req, res) => {
9 | const user = req.user
10 | res.json({ user })
11 | })
12 |
13 | // @desc Update Contact Info
14 | // @access Private
15 | router.patch('/users/me', auth, async (req, res) => {
16 | const updates = Object.keys(req.body) //Converts object to an array of strings!
17 | const allowedUpdates = ['phoneNo']
18 | const isValidOperation = updates.every((update) => {
19 | if (allowedUpdates.includes(update)) return true
20 | })
21 | if (!isValidOperation) {
22 | return res.status(400).send({ error: 'Not Allowed' })
23 | }
24 |
25 | try {
26 | updates.forEach((update) => {
27 | req.user[update] = req.body[update]
28 | })
29 | await req.user.save()
30 | res.json(req.user)
31 | } catch (err) {
32 | res.status(400).json({ error: `${err}` })
33 | }
34 | })
35 |
36 | // @desc DEL user
37 | // @access Private
38 | router.delete('/users/me', auth, async (req, res) => {
39 | try {
40 | await req.user.remove()
41 | res.json({ msg: 'User deleted successfully' })
42 | } catch (err) {
43 | res.status(400).json({ error: `${err}` })
44 | }
45 | })
46 |
47 | module.exports = router
--------------------------------------------------------------------------------