├── api
├── .env
├── .gitignore
├── .dockerignore
├── services
│ ├── index.js
│ ├── item.service.js
│ └── bid.service.js
├── Dockerfile
├── controllers
│ ├── index.js
│ ├── bid.controller.js
│ └── item.controller.js
├── db
│ ├── index.js
│ └── redis.db.js
├── scripts
│ └── add-starting-auction-items.js
├── routes
│ └── index.js
├── starting-auction-items.js
├── package.json
├── app.js
├── redis.js
├── streams-client.js
└── package-lock.json
├── bidding-service
├── .env
├── .gitignore
├── .dockerignore
├── redis.js
├── Dockerfile
├── index.js
├── package.json
├── app.js
├── streams-client.js
└── package-lock.json
├── client
├── public
│ ├── robots.txt
│ ├── favicon.ico
│ ├── logo192.png
│ ├── logo512.png
│ ├── manifest.json
│ └── index.html
├── Dockerfile
├── src
│ ├── App.js
│ ├── index.css
│ ├── reportWebVitals.js
│ ├── ItemPreviewCard.js
│ ├── index.js
│ ├── Items.js
│ ├── PlaceBid.js
│ ├── App.css
│ ├── Item.js
│ └── logo.svg
├── .gitignore
└── package.json
├── auctioneer-architecture.png
├── auctioneer-app-screenshot.png
├── images
└── app_preview_image.png
├── LICENSE
├── docker-compose.yml
├── marketplace.json
└── README.md
/api/.env:
--------------------------------------------------------------------------------
1 | REDIS=redismod
--------------------------------------------------------------------------------
/api/.gitignore:
--------------------------------------------------------------------------------
1 | node_modules/
--------------------------------------------------------------------------------
/bidding-service/.env:
--------------------------------------------------------------------------------
1 | REDIS=redismod
--------------------------------------------------------------------------------
/bidding-service/.gitignore:
--------------------------------------------------------------------------------
1 | node_modules/
--------------------------------------------------------------------------------
/client/public/robots.txt:
--------------------------------------------------------------------------------
1 | # https://www.robotstxt.org/robotstxt.html
2 | User-agent: *
3 | Disallow:
4 |
--------------------------------------------------------------------------------
/auctioneer-architecture.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/redis-developer/auctioneer/master/auctioneer-architecture.png
--------------------------------------------------------------------------------
/client/public/favicon.ico:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/redis-developer/auctioneer/master/client/public/favicon.ico
--------------------------------------------------------------------------------
/client/public/logo192.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/redis-developer/auctioneer/master/client/public/logo192.png
--------------------------------------------------------------------------------
/client/public/logo512.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/redis-developer/auctioneer/master/client/public/logo512.png
--------------------------------------------------------------------------------
/auctioneer-app-screenshot.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/redis-developer/auctioneer/master/auctioneer-app-screenshot.png
--------------------------------------------------------------------------------
/images/app_preview_image.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/redis-developer/auctioneer/master/images/app_preview_image.png
--------------------------------------------------------------------------------
/api/.dockerignore:
--------------------------------------------------------------------------------
1 | .git
2 | .gitignore
3 | .eslintrc.js
4 | build/
5 | node_modules/
6 | docker-compose.yml
7 | .env
8 | npm-debug.log
--------------------------------------------------------------------------------
/bidding-service/.dockerignore:
--------------------------------------------------------------------------------
1 | .git
2 | .gitignore
3 | .eslintrc.js
4 | build/
5 | node_modules/
6 | docker-compose.yml
7 | .env
8 | npm-debug.log
--------------------------------------------------------------------------------
/api/services/index.js:
--------------------------------------------------------------------------------
1 | const bidService = require('./bid.service')
2 | const itemService = require('./item.service')
3 |
4 | module.exports = {
5 | bidService,
6 | itemService
7 | }
8 |
--------------------------------------------------------------------------------
/bidding-service/redis.js:
--------------------------------------------------------------------------------
1 | const IORedis = require('ioredis')
2 | const dotenv = require('dotenv')
3 |
4 | dotenv.config()
5 |
6 | module.exports = new IORedis(`//${process.env.REDIS}:6379`)
--------------------------------------------------------------------------------
/client/Dockerfile:
--------------------------------------------------------------------------------
1 | FROM node:16.0.0
2 | EXPOSE 3000
3 |
4 | WORKDIR /usr/app
5 |
6 | COPY package*.json ./
7 |
8 | RUN npm install
9 |
10 | COPY . .
11 |
12 | CMD ["npm", "start"]
--------------------------------------------------------------------------------
/api/Dockerfile:
--------------------------------------------------------------------------------
1 | FROM node:16.0.0
2 | EXPOSE 3001
3 |
4 | WORKDIR /usr/app
5 |
6 | COPY package*.json ./
7 |
8 | RUN npm install
9 |
10 | COPY . .
11 |
12 | CMD [ "sleep", "infinity" ]
--------------------------------------------------------------------------------
/bidding-service/Dockerfile:
--------------------------------------------------------------------------------
1 | FROM node:16.0.0
2 | EXPOSE 3002
3 |
4 | WORKDIR /usr/app
5 |
6 | COPY package*.json ./
7 |
8 | RUN npm install
9 |
10 | COPY . .
11 |
12 | CMD [ "sleep", "infinity" ]
--------------------------------------------------------------------------------
/api/controllers/index.js:
--------------------------------------------------------------------------------
1 | const itemController = require('./item.controller')
2 | const bidController = require('./bid.controller')
3 |
4 | module.exports = {
5 | itemController,
6 | bidController
7 | }
8 |
--------------------------------------------------------------------------------
/api/db/index.js:
--------------------------------------------------------------------------------
1 | const { setItem, fetchById, fetchAll, getLastItemFromStream } = require('./redis.db')
2 |
3 | module.exports = {
4 | setItem,
5 | fetchById,
6 | fetchAll,
7 | getLastItemFromStream
8 | }
--------------------------------------------------------------------------------
/api/scripts/add-starting-auction-items.js:
--------------------------------------------------------------------------------
1 | const { setItem } = require('../db')
2 | const auctionItems = require('../starting-auction-items')
3 |
4 | for (const auctionItem of auctionItems) {
5 | await setItem(auctionItem)
6 | }
7 |
--------------------------------------------------------------------------------
/client/src/App.js:
--------------------------------------------------------------------------------
1 | import './App.css'
2 | import Item from './Item'
3 |
4 | function App() {
5 | return (
6 |
7 |
10 |
11 | )
12 | }
13 |
14 | export default App
15 |
--------------------------------------------------------------------------------
/api/routes/index.js:
--------------------------------------------------------------------------------
1 | const express = require('express')
2 |
3 | const { itemController, bidController } = require('../controllers')
4 |
5 | const router = express.Router()
6 |
7 | router.get('/item', itemController.getItem)
8 | router.get('/items', itemController.getItems)
9 |
10 | router.post('/bid', bidController.postBid)
11 |
12 | module.exports = router
13 |
--------------------------------------------------------------------------------
/client/.gitignore:
--------------------------------------------------------------------------------
1 | # See https://help.github.com/articles/ignoring-files/ for more about ignoring files.
2 |
3 | # dependencies
4 | /node_modules
5 | /.pnp
6 | .pnp.js
7 |
8 | # testing
9 | /coverage
10 |
11 | # production
12 | /build
13 |
14 | # misc
15 | .DS_Store
16 | .env.local
17 | .env.development.local
18 | .env.test.local
19 | .env.production.local
20 |
21 | npm-debug.log*
22 | yarn-debug.log*
23 | yarn-error.log*
24 |
--------------------------------------------------------------------------------
/client/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 |
--------------------------------------------------------------------------------
/client/src/reportWebVitals.js:
--------------------------------------------------------------------------------
1 | const reportWebVitals = onPerfEntry => {
2 | if (onPerfEntry && onPerfEntry instanceof Function) {
3 | import('web-vitals').then(({ getCLS, getFID, getFCP, getLCP, getTTFB }) => {
4 | getCLS(onPerfEntry);
5 | getFID(onPerfEntry);
6 | getFCP(onPerfEntry);
7 | getLCP(onPerfEntry);
8 | getTTFB(onPerfEntry);
9 | });
10 | }
11 | };
12 |
13 | export default reportWebVitals;
14 |
--------------------------------------------------------------------------------
/client/src/ItemPreviewCard.js:
--------------------------------------------------------------------------------
1 | import './App.css'
2 |
3 | function ItemPreviewCard({ data }) {
4 | const { name, startDate, endDate } = data
5 | return (
6 |
7 |
8 | Item: {name}
9 |
10 | Start Time: {startDate}
11 |
12 | End Time: {endDate}
13 |
14 |
15 |
16 | )
17 | }
18 |
19 | export default ItemPreviewCard
20 |
--------------------------------------------------------------------------------
/api/services/item.service.js:
--------------------------------------------------------------------------------
1 | const { fetchById, fetchAll } = require('../db')
2 |
3 | // These are simple "wrapper" functions for now but as the app complexity grows and more
4 | // business logic is added, that logic will go within the body's of these functions.
5 | const getItemById = (itemId) => {
6 | return fetchById(itemId)
7 | }
8 |
9 | const getAllItems = async () => {
10 | return fetchAll()
11 | }
12 |
13 | module.exports = {
14 | getItemById,
15 | getAllItems
16 | }
17 |
--------------------------------------------------------------------------------
/bidding-service/index.js:
--------------------------------------------------------------------------------
1 | const { subscribe } = require('./streams-client')
2 |
3 | const getLatestBid = () => {
4 | const workerFunction = (data) => {
5 | console.log('data from workerFunction', data)
6 | io.emit('highest-bid', data.newBid)
7 | }
8 |
9 | subscribe({
10 | groupName: 'bid-subscriber-consumer-group',
11 | streamName: 'bid-stream',
12 | readTimeout: 20,
13 | workerFunction
14 | })
15 | }
16 |
17 | // immediately call to start watching bids
18 | getLatestBid()
19 |
--------------------------------------------------------------------------------
/client/src/index.js:
--------------------------------------------------------------------------------
1 | import React from 'react';
2 | import ReactDOM from 'react-dom';
3 | import './index.css';
4 | import App from './App';
5 | import reportWebVitals from './reportWebVitals';
6 |
7 | ReactDOM.render(
8 |
9 |
10 | ,
11 | document.getElementById('root')
12 | );
13 |
14 | // If you want to start measuring performance in your app, pass a function
15 | // to log results (for example: reportWebVitals(console.log))
16 | // or send to an analytics endpoint. Learn more: https://bit.ly/CRA-vitals
17 | reportWebVitals();
18 |
--------------------------------------------------------------------------------
/api/starting-auction-items.js:
--------------------------------------------------------------------------------
1 | module.exports = [
2 | {
3 | "id": "1",
4 | "name": "Apple iPhone X",
5 | "startingBid": "250",
6 | "condition": "Used, Very Good",
7 | "startDate": "",
8 | "endDate": ""
9 | },
10 | {
11 | "id": "2",
12 | "name": "MacBook Pro Retina 2019",
13 | "startingBid": "400",
14 | "condition": "Brand New",
15 | "startDate": "",
16 | "endDate": ""
17 | },
18 | {
19 | "id": "3",
20 | "name": "PS4",
21 | "startingBid": "25",
22 | "condition": "Poor",
23 | "startDate": "",
24 | "endDate": ""
25 | }
26 | ]
27 |
--------------------------------------------------------------------------------
/api/controllers/bid.controller.js:
--------------------------------------------------------------------------------
1 | const { bidService } = require('../services')
2 |
3 | const { postBidToStream } = bidService
4 |
5 | const postBid = async (req, res, next) => {
6 | const {bid} = req.body
7 |
8 | try {
9 | const posted = await postBidToStream(Number(bid))
10 |
11 | if (posted) res.sendStatus('201')
12 | else res.status('400').send('Bid must be greater than current bid')
13 |
14 | return
15 | } catch(err) {
16 | console.error(err)
17 | res.sendStatus(500) && next(err)
18 |
19 | return
20 | }
21 | }
22 |
23 | module.exports = {
24 | postBid
25 | }
26 |
--------------------------------------------------------------------------------
/client/public/manifest.json:
--------------------------------------------------------------------------------
1 | {
2 | "short_name": "React App",
3 | "name": "Create React App Sample",
4 | "icons": [
5 | {
6 | "src": "favicon.ico",
7 | "sizes": "64x64 32x32 24x24 16x16",
8 | "type": "image/x-icon"
9 | },
10 | {
11 | "src": "logo192.png",
12 | "type": "image/png",
13 | "sizes": "192x192"
14 | },
15 | {
16 | "src": "logo512.png",
17 | "type": "image/png",
18 | "sizes": "512x512"
19 | }
20 | ],
21 | "start_url": ".",
22 | "display": "standalone",
23 | "theme_color": "#000000",
24 | "background_color": "#ffffff"
25 | }
26 |
--------------------------------------------------------------------------------
/api/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "auctioneer",
3 | "version": "1.0.0",
4 | "description": "Redis-based auctions",
5 | "scripts": {
6 | "start": "NODE_ENV=development node -r esm app.js",
7 | "start:watch": "NODE_ENV=development nodemon -r esm app.js"
8 | },
9 | "dependencies": {
10 | "common-tags": "^1.8.0",
11 | "cors": "^2.8.5",
12 | "dotenv": "8.2.0",
13 | "esm": "3.2.25",
14 | "express": "4.17.1",
15 | "ioredis": "4.26.0",
16 | "socket.io": "4.0.2"
17 | },
18 | "keywords": [
19 | "redis"
20 | ],
21 | "esm": {
22 | "await": true
23 | },
24 | "devDependencies": {
25 | "nodemon": "2.0.7"
26 | }
27 | }
28 |
--------------------------------------------------------------------------------
/client/src/Items.js:
--------------------------------------------------------------------------------
1 | import { useState, useEffect } from 'react'
2 | import ItemPreviewCard from './ItemPreviewCard'
3 |
4 | function Items() {
5 | const [items, setItems] = useState([])
6 |
7 | useEffect(() => {
8 | const fetchItems = async() => {
9 | const data = await fetch('http://localhost:3001/api/items')
10 | const items = (await data.json()).map((item) => JSON.parse(item))
11 |
12 | setItems(items)
13 | }
14 |
15 | fetchItems()
16 | }, [])
17 |
18 | return (
19 | <>
20 | {items.map((item) => {
21 | return
22 | })}
23 | >
24 | )
25 | }
26 |
27 | export default Items
28 |
--------------------------------------------------------------------------------
/bidding-service/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "bidding-service",
3 | "version": "1.0.0",
4 | "description": "Stream subscriber",
5 | "main": "index.js",
6 | "scripts": {
7 | "start": "NODE_ENV=development node -r esm app.js",
8 | "start:watch": "NODE_ENV=development nodemon -r esm app.js"
9 | },
10 | "dependencies": {
11 | "common-tags": "^1.8.0",
12 | "cors": "^2.8.5",
13 | "dotenv": "8.2.0",
14 | "esm": "3.2.25",
15 | "express": "4.17.1",
16 | "ioredis": "4.26.0",
17 | "socket.io": "4.0.2"
18 | },
19 | "keywords": [
20 | "redis"
21 | ],
22 | "esm": {
23 | "await": true
24 | },
25 | "devDependencies": {
26 | "nodemon": "2.0.7"
27 | }
28 | }
29 |
--------------------------------------------------------------------------------
/api/app.js:
--------------------------------------------------------------------------------
1 | const http = require('http')
2 | const express = require('express')
3 | const cors = require('cors')
4 | const bodyParser = require('body-parser')
5 | const socketIo = require('socket.io')
6 | const routes = require('./routes')
7 |
8 | const app = express()
9 |
10 | app.use(cors())
11 | app.use(bodyParser.json());
12 | app.use(bodyParser.urlencoded({ extended: true }));
13 |
14 | app.get('/', (req, res) => res.send('App is working'))
15 |
16 | app.use('/api', routes)
17 |
18 | const server = http.createServer(app)
19 | server.listen(3001, () => console.log('Example app listening on port 3001!'))
20 |
21 | const io = socketIo(server)
22 |
23 | io.on('connection', (socket) => {
24 | console.log('new socketio connection')
25 | })
26 |
27 | // hacky, TODO: don't do this :)
28 | global.io = io
--------------------------------------------------------------------------------
/api/controllers/item.controller.js:
--------------------------------------------------------------------------------
1 | const { itemService } = require('../services')
2 |
3 | const { getItemById, getAllItems } = itemService
4 |
5 | const getItem = async (req, res, next) => {
6 | const {itemId} = req.query
7 |
8 | try {
9 | const item = await getItemById(itemId)
10 | res.send(item)
11 |
12 | return
13 | } catch(err) {
14 | console.error(err)
15 | res.sendStatus(500) && next(err)
16 |
17 | return
18 | }
19 | }
20 |
21 | const getItems = async (req, res, next) => {
22 | try {
23 | const items = await getAllItems()
24 | res.send(items)
25 |
26 | return
27 | } catch(err) {
28 | console.error(err)
29 | res.sendStatus(500) && next(err)
30 |
31 | return
32 | }
33 | }
34 |
35 | module.exports = {
36 | getItem,
37 | getItems
38 | }
39 |
--------------------------------------------------------------------------------
/api/db/redis.db.js:
--------------------------------------------------------------------------------
1 | const redis = require('../redis')
2 |
3 | const setItem = (item) => {
4 | setInTracking(item.id)
5 | return redis.call('JSON.SET', `item.${item.id}`, '.', JSON.stringify(item))
6 | }
7 |
8 | const fetchById = (id) => {
9 | return redis.call('JSON.GET', `item.${id}`)
10 | }
11 |
12 | const setInTracking = (id) => {
13 | return redis.hset('items-tracking', `id:${id}`, id)
14 | }
15 |
16 | const getFromTracking = async () => {
17 | const itemsTracking = await redis.hgetall('items-tracking')
18 |
19 | return Object.values(itemsTracking)
20 | }
21 |
22 | // Couldn't figure out a way to get all keys from ReJSON, so tracking ids for now and grabbing by id
23 | const fetchAll = async () => {
24 | const itemIds = await getFromTracking()
25 |
26 | return Promise.all(itemIds.map((id) => fetchById(id)))
27 | }
28 |
29 | const getLastItemFromStream = (streamName) => {
30 | return redis.xrevrange(streamName, '+', '-', 'COUNT', '1')
31 | }
32 |
33 | module.exports = {
34 | setItem,
35 | fetchById,
36 | fetchAll,
37 | getLastItemFromStream
38 | }
39 |
--------------------------------------------------------------------------------
/client/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "client",
3 | "version": "0.1.0",
4 | "private": true,
5 | "dependencies": {
6 | "@testing-library/jest-dom": "^5.11.4",
7 | "@testing-library/react": "^11.1.0",
8 | "@testing-library/user-event": "^12.1.10",
9 | "react": "^17.0.2",
10 | "react-dom": "^17.0.2",
11 | "react-hook-form": "^7.6.0",
12 | "react-router-dom": "^5.2.0",
13 | "react-scripts": "4.0.3",
14 | "socket.io-client": "^4.0.2",
15 | "use-socket.io-client": "^1.1.0",
16 | "web-vitals": "^1.0.1"
17 | },
18 | "scripts": {
19 | "start": "react-scripts start",
20 | "build": "react-scripts build",
21 | "test": "react-scripts test",
22 | "eject": "react-scripts eject"
23 | },
24 | "eslintConfig": {
25 | "extends": [
26 | "react-app",
27 | "react-app/jest"
28 | ]
29 | },
30 | "browserslist": {
31 | "production": [
32 | ">0.2%",
33 | "not dead",
34 | "not op_mini all"
35 | ],
36 | "development": [
37 | "last 1 chrome version",
38 | "last 1 firefox version",
39 | "last 1 safari version"
40 | ]
41 | }
42 | }
43 |
--------------------------------------------------------------------------------
/LICENSE:
--------------------------------------------------------------------------------
1 | MIT License
2 |
3 | Copyright (c) 2021 Redis Developer
4 |
5 | Permission is hereby granted, free of charge, to any person obtaining a copy
6 | of this software and associated documentation files (the "Software"), to deal
7 | in the Software without restriction, including without limitation the rights
8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9 | copies of the Software, and to permit persons to whom the Software is
10 | furnished to do so, subject to the following conditions:
11 |
12 | The above copyright notice and this permission notice shall be included in all
13 | copies or substantial portions of the Software.
14 |
15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21 | SOFTWARE.
22 |
--------------------------------------------------------------------------------
/bidding-service/app.js:
--------------------------------------------------------------------------------
1 | const http = require('http')
2 | const express = require('express')
3 | const cors = require('cors')
4 | const bodyParser = require('body-parser')
5 | const socketIo = require('socket.io')
6 |
7 | const app = express()
8 |
9 | app.use(cors())
10 | app.use(bodyParser.json());
11 | app.use(bodyParser.urlencoded({ extended: true }));
12 |
13 | app.get('/', (req, res) => res.send('App is working'))
14 |
15 | const server = http.createServer(app)
16 | server.listen(3002, () => console.log('Bidding service listening on port 3002!'))
17 |
18 | const io = socketIo(server)
19 |
20 | io.on('connection', (socket) => {
21 | console.log('new socketio connection - bidding service')
22 | })
23 |
24 | // hacky, TODO: don't do this :)
25 | global.io = io
26 |
27 | const { subscribe } = require('./streams-client')
28 |
29 | // TODO: end subscriber when bidding ends
30 | const getLatestBid = () => {
31 | const workerFunction = (data) => {
32 | console.log('data from workerFunction - bidding service', data)
33 | io.emit('highest-bid', data.newBid)
34 | }
35 |
36 | subscribe({
37 | groupName: 'bid-subscriber-consumer-group',
38 | streamName: 'bid-stream',
39 | readTimeout: 20,
40 | workerFunction
41 | })
42 | }
43 |
44 | // immediately call to start watching bids
45 | getLatestBid()
46 |
--------------------------------------------------------------------------------
/client/src/PlaceBid.js:
--------------------------------------------------------------------------------
1 | import React from 'react'
2 | import { useForm } from 'react-hook-form'
3 |
4 | export default function PlaceBid() {
5 | const { register, handleSubmit, setError, formState: { errors } } = useForm()
6 |
7 | const onSubmit = handleSubmit(async (data) => {
8 | console.log(data)
9 | const response = await fetch('http://localhost:3001/api/bid',
10 | {
11 | headers: {
12 | 'Accept': 'application/json',
13 | 'Content-Type': 'application/json'
14 | },
15 | method: 'POST',
16 | body: JSON.stringify(data)
17 | }
18 | )
19 |
20 | if (response.status === 400) {
21 | setError('bid', {
22 | type: 'server',
23 | message: 'Bid must be greater than the current highest bid'
24 | })
25 | }
26 | })
27 |
28 | return(
29 |
43 | )
44 | }
45 |
--------------------------------------------------------------------------------
/docker-compose.yml:
--------------------------------------------------------------------------------
1 | version: '3.7'
2 | services:
3 | redismod:
4 | image: redislabs/redismod
5 | container_name: redismod
6 | volumes:
7 | - redis-data:/data
8 | ports:
9 | - 6379:6379
10 | redis-insight:
11 | image: redislabs/redisinsight
12 | container_name: redis-insight
13 | ports:
14 | - 8001:8001
15 | api:
16 | build:
17 | context: ./api/
18 | container_name: api
19 | volumes:
20 | - ./api/:/usr/app
21 | - /usr/app/node_modules
22 | depends_on:
23 | - redismod
24 | command: npm run start:watch
25 | environment:
26 | - REDIS:redismod
27 | ports:
28 | - 3001:3001
29 | bidding-service:
30 | build:
31 | context: ./bidding-service/
32 | container_name: bidding-service
33 | volumes:
34 | - ./bidding-service/:/usr/app
35 | - /usr/app/node_modules
36 | depends_on:
37 | - redismod
38 | command: npm run start:watch
39 | environment:
40 | - REDIS:redismod
41 | ports:
42 | - 3002:3002
43 | client:
44 | build:
45 | context: ./client/
46 | container_name: client
47 | command: npm start
48 | volumes:
49 | - ./client/:/usr/app
50 | - /usr/app/node_modules
51 | depends_on:
52 | - api
53 | ports:
54 | - 3000:3000
55 |
56 | volumes:
57 | redis-data:
--------------------------------------------------------------------------------
/api/redis.js:
--------------------------------------------------------------------------------
1 | const IORedis = require('ioredis')
2 | const dotenv = require('dotenv')
3 | const { oneLine } = require('common-tags')
4 |
5 | dotenv.config()
6 |
7 | const redis = new IORedis(`//${process.env.REDIS}:6379`)
8 |
9 | // 'value' in the lua script below looks like [ '1621041237325-0', [ 'newBid', '4' ] ]
10 | // which I had a tough time looping through using pairs/ipairs
11 | // we know that the stream item structure shouldn't change, so just hardcode the indexes here
12 |
13 | // we do an `isempty` check since the stream will be empty on first bid
14 | redis.defineCommand('postBidToStream', {
15 | numberOfKeys: 1,
16 | lua: oneLine(`
17 | local streamName = KEYS[1]
18 | local newBid = ARGV[1]
19 | local lastBidStreamItem = redis.call("XREVRANGE", streamName, "+", "-", "COUNT", "1")
20 |
21 | local function isempty(table)
22 | return #table==0
23 | end
24 |
25 | if isempty(lastBidStreamItem) then
26 | return redis.call("XADD", streamName, "MAXLEN", "~", "1000", "*", "newBid", newBid)
27 | end
28 |
29 | for key,value in pairs(lastBidStreamItem) do
30 | local lastBidValue = value[2][2]
31 |
32 | if tonumber(lastBidValue) < tonumber(newBid) then
33 | return redis.call("XADD", streamName, "MAXLEN", "~", "1000", "*", "newBid", newBid)
34 | end
35 |
36 | return
37 | end
38 | `)
39 | })
40 |
41 | module.exports = redis
42 |
--------------------------------------------------------------------------------
/client/src/App.css:
--------------------------------------------------------------------------------
1 | .App {
2 | text-align: center;
3 | }
4 |
5 | .App-logo {
6 | height: 40vmin;
7 | pointer-events: none;
8 | }
9 |
10 | @media (prefers-reduced-motion: no-preference) {
11 | .App-logo {
12 | animation: App-logo-spin infinite 20s linear;
13 | }
14 | }
15 |
16 | .App-header {
17 | background-color: #5c6c8d;
18 | min-height: 100vh;
19 | display: flex;
20 | flex-direction: column;
21 | align-items: center;
22 | justify-content: center;
23 | font-size: calc(10px + 2vmin);
24 | color: white;
25 | }
26 |
27 | .App-link {
28 | color: #61dafb;
29 | }
30 |
31 | .product-image {
32 | width: 100%;
33 | }
34 |
35 | .item-preview {
36 | color: black;
37 | }
38 |
39 | .card {
40 | background-color: white;
41 | box-shadow: 0 4px 8px 0 rgba(0, 0, 0, 0.2);
42 | max-width: 300px;
43 | margin: auto;
44 | text-align: center;
45 | font-family: arial;
46 | }
47 |
48 | .price {
49 | color: black;
50 | font-size: 22px;
51 | }
52 |
53 | span .highest-bid-price {
54 | color: blue;
55 | font-size: 30px;
56 | }
57 |
58 | .card button {
59 | border: none;
60 | outline: 0;
61 | padding: 12px;
62 | color: white;
63 | background-color: #000;
64 | text-align: center;
65 | cursor: pointer;
66 | width: 100%;
67 | font-size: 18px;
68 | margin-top: 10px;
69 | }
70 |
71 | .card button:hover {
72 | opacity: 0.7;
73 | }
74 |
75 | .validation {
76 | color: red;
77 | }
78 |
79 | .bidding-finished {
80 | color: blue;
81 | }
82 |
83 | @keyframes App-logo-spin {
84 | from {
85 | transform: rotate(0deg);
86 | }
87 | to {
88 | transform: rotate(360deg);
89 | }
90 | }
91 |
--------------------------------------------------------------------------------
/marketplace.json:
--------------------------------------------------------------------------------
1 | {
2 | "app_name": "Auctioneer - Redis item bidding app",
3 | "description": "Showcases using Redis Streams and JSON feature, allowing you to bid on items in real-time",
4 | "type": "Full App",
5 | "rank": "386",
6 | "contributed_by": "Community",
7 | "repo_url": "https://github.com/redis-developer/auctioneer",
8 | "preview_image_url": "https://raw.githubusercontent.com/redis-developer/auctioneer/master/images/app_preview_image.png",
9 | "download_url": "https://github.com/coreyc/auctioneer/archive/refs/heads/master.zip",
10 | "hosted_url": "",
11 | "quick_deploy": "false",
12 | "deploy_buttons": [
13 | {
14 | "heroku": "https://heroku.com/deploy?template=https://github.com/redis-developer/auctioneer.git"
15 | },
16 | {
17 | "Google": "https://deploy.cloud.run/?git_repo=https://github.com/redis-developer/auctioneer.git"
18 | }
19 | ],
20 | "language": [
21 | "JavaScript"
22 | ],
23 | "redis_commands": [
24 | "JSON_GET",
25 | "JSON_SET",
26 | "HGETALL",
27 | "HSET",
28 | "XREVRANGE",
29 | "XADD",
30 | "XACK",
31 | "XREADGROUP",
32 | "XGROUP"
33 | ],
34 | "redis_use_cases": [
35 | "Streams"
36 | ],
37 | "redis_features": [
38 | "JSON"
39 | ],
40 | "app_image_urls": [
41 | "https://raw.githubusercontent.com/redis-developer/auctioneer/master/auctioneer-app-screenshot.png"
42 | ],
43 | "youtube_url": "https://www.youtube.com/watch?v=txdXgJT7F7A",
44 | "special_tags": [
45 | "Hackathon"
46 | ],
47 | "verticals": [
48 | "Financial Services"
49 | ],
50 | "markdown": "https://raw.githubusercontent.com/redis-developer/auctioneer/master/README.md"
51 | }
--------------------------------------------------------------------------------
/client/public/index.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
12 |
13 |
17 |
18 |
27 | Auctioneer
28 |
29 |
30 |
31 |
32 |
42 |
43 |
44 |
--------------------------------------------------------------------------------
/client/src/Item.js:
--------------------------------------------------------------------------------
1 | import React, { useState, useEffect } from 'react'
2 | import socketIOClient from 'socket.io-client'
3 | import PlaceBid from './PlaceBid'
4 |
5 | const biddingServiceSocket = socketIOClient('http://127.0.0.1:3002', {
6 | transports: ['websocket']
7 | })
8 |
9 | const apiSocket = socketIOClient('http://127.0.0.1:3001', {
10 | transports: ['websocket']
11 | })
12 |
13 | function Item() {
14 | const [response, setResponse] = useState('')
15 | const [biddingFinishedresponse, setBiddingFinishedResponse] = useState('')
16 | const [itemData, setItemData] = useState('')
17 |
18 | useEffect(() => {
19 | const getItemData = async (itemId) => {
20 | const data = await fetch(`http://localhost:3001/api/item?itemId=${itemId}`)
21 | const itemData = await data.json()
22 |
23 | console.log(`itemData: ${JSON.stringify(itemData)}`)
24 |
25 | setItemData(itemData)
26 | }
27 |
28 | getItemData(1) // hardcode item ID for now
29 |
30 | biddingServiceSocket.on('highest-bid', (bid) => {
31 | setResponse(bid)
32 | return
33 | })
34 |
35 | apiSocket.on('bidding-finished', (bid) => {
36 | setBiddingFinishedResponse(bid)
37 | return
38 | })
39 | }, [])
40 |
41 | const showBiddingFinished = () => {
42 | if (biddingFinishedresponse) {
43 | return Bidding finished, winning bid price was: ${biddingFinishedresponse}!
44 | }
45 | }
46 |
47 | const showPlaceBid = () => {
48 | if (!biddingFinishedresponse) {
49 | return
50 | }
51 | }
52 |
53 | return (
54 |
55 |
56 | Item name: {itemData.name}
57 | Item condition: {itemData.condition}
58 | Starting bid: ${itemData.startingBid}
59 | Current highest bid: ${response}
60 |
61 | {showBiddingFinished()}
62 | {showPlaceBid()}
63 |
64 | )
65 | }
66 |
67 | export default Item
68 |
--------------------------------------------------------------------------------
/api/services/bid.service.js:
--------------------------------------------------------------------------------
1 | const { publish } = require('../streams-client')
2 | const { getLastItemFromStream, fetchById } = require('../db')
3 | const redis = require('../redis')
4 |
5 | const postBidToStream = (newBid) => {
6 | console.log(newBid)
7 | return redis.postBidToStream('bid-stream', newBid)
8 | }
9 |
10 | const executeAt = (scheduledTime, fnToExecute) => {
11 | const currentTime = new Date().getTime()
12 |
13 | if(currentTime > scheduledTime){
14 | console.error('The scheduled time has already passed')
15 | return
16 | }
17 |
18 | setTimeout(fnToExecute, scheduledTime - currentTime)
19 | return
20 | }
21 |
22 | const getWinningBidValue = (bid) => {
23 | // data structure from redis will look like [ [ '1621097945378-0', [ 'newBid', '5605' ] ] ]
24 | return bid[0][1][1]
25 | }
26 |
27 | const fnToExecute = async () => {
28 | try {
29 | const lastBid = await getLastItemFromStream('bid-stream')
30 | const winningBid = getWinningBidValue(lastBid)
31 | const itemInfo = await fetchById(1) // hard-code item id we're looking for, just for now
32 | const eventPayload = {itemInfo, winningBid}
33 |
34 | // by publishing to a different stream once the bidding has ended and the item has been "won",
35 | // a separate service can subscribe to this stream and process the order by charging the payment on file,
36 | // process shipping, log the item in Redis JSON for "long-term" storage, etc.
37 | // Redis Streams make this very powerful!
38 | await publish({ streamName: 'bidding-finished-stream', maxStreamLength: 5000, eventPayload })
39 | console.log('bidding-finished')
40 | io.emit('bidding-finished', winningBid)
41 | } catch(err) {
42 | console.error('error while publishing winning bid to stream', err)
43 | return
44 | }
45 | }
46 |
47 | // hardcode a bid end time (30 seconds after app loads, just for demo purposes)
48 | const bidEndTime = new Date().getTime() + 30000
49 | executeAt(bidEndTime, fnToExecute)
50 |
51 | module.exports = {
52 | postBidToStream
53 | }
54 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | ## Instructions for running locally
2 | First, make sure Docker is installed.
3 |
4 | Then, to spin up the rest of the project:
5 |
6 | `docker-compose build` (first time running project)
7 |
8 | `docker-compose up`
9 |
10 | The application/UI loads at `http://localhost:3000/`
11 |
12 | ## Load auction items
13 | To load the starting set of auction items:
14 |
15 | `docker-compose exec api node -r esm scripts/add-starting-auction-items.js`
16 |
17 | This will add some "default" auction items to Redis JSON. You only need to do this on the first run of this project.
18 |
19 | ## Testing bidding
20 | For demo purposes, the bidding is hardcoded to end 30 seconds after the API starts/restarts. If you are playing around with the app and developing locally, you may need to restart the API to reset the timer:
21 |
22 | `docker-compose restart api`
23 |
24 | ## Redis commands
25 | api:
26 | - postBidToStream (Lua script):
27 | - `XREVRANGE streamName + - COUNT 1` (gets last item in stream)
28 | - `XADD streamName MAXLEN ~ 1000 * newBid newBidValue` (adds latest/highest bid to the bid-stream)
29 | - setItem:
30 | - `JSON.SET item.${id} . item`
31 | - fetchById:
32 | - `JSON.GET item.${id}`
33 | - setInTracking:
34 | - `HSET items-tracking id:${id} id`
35 | - getFromTracking:
36 | - `HGETALL items-tracking`
37 | - getLastItemFromStream:
38 | - `XREVRANGE streamName + - COUNT 1`
39 |
40 | common (`streams-client.js`):
41 | - `XACK streamName groupName id`
42 | - `XADD streamName MAXLEN ~ maxStreamLength * formattedData`
43 | - `XREADGROUP GROUP groupName consumerName BLOCK blockTime STREAMS streamName id)`
44 | - `XREADGROUP GROUP groupName consumerName COUNT count STREAMS streamName id)`
45 | - `XGROUP CREATE streamName groupName $ MKSTREAM`
46 |
47 | ## Application Architecture
48 | 
49 |
50 | ## Using the bidding app
51 | To use the app and bid on an item, enter a bid into the input box and submit. If you submit without entering anything you will get a validation error on the UI. Also, if the value you enter is not higher than the current highest bid, the bid will not be entered and you will get a validation error.
52 |
53 | 
54 |
--------------------------------------------------------------------------------
/client/src/logo.svg:
--------------------------------------------------------------------------------
1 |
--------------------------------------------------------------------------------
/api/streams-client.js:
--------------------------------------------------------------------------------
1 | const redis = require('./redis')
2 |
3 | const REDIS_ARRAY_POSITIONS = {
4 | streamRoot: 0,
5 | root: {
6 | streamName: 0,
7 | streamEvents: 1,
8 | },
9 | streamEvent: {
10 | eventId: 0,
11 | eventData: 1,
12 | },
13 | }
14 | const READ_GROUP_EVENT_ID = {
15 | FIRST_RUN: '0',
16 | NEW_ITEMS: '>',
17 | }
18 |
19 | const mapObjToStringArray = (obj) => {
20 | const arr = []
21 |
22 | for (const key in obj) {
23 | arr.push(key, obj[key])
24 | }
25 |
26 | return arr
27 | }
28 |
29 | const mapArrayToObject = (array) => {
30 | const arrayCopy = array.slice()
31 | const object = {}
32 |
33 | while (arrayCopy && arrayCopy.length) {
34 | const key = arrayCopy.shift()
35 | const value = arrayCopy.shift()
36 | object[key] = value
37 | }
38 |
39 | return object
40 | }
41 |
42 | const copyArray = (array) => array.slice()
43 | const getRootStreamData = (streamData) => streamData[REDIS_ARRAY_POSITIONS.streamRoot]
44 | const getStreamEvents = (rootStreamData) => rootStreamData[REDIS_ARRAY_POSITIONS.root.streamEvents]
45 | const getEventId = (streamEvent) => streamEvent[REDIS_ARRAY_POSITIONS.streamEvent.eventId]
46 | const getEventData = (streamEvent) => streamEvent[REDIS_ARRAY_POSITIONS.streamEvent.eventData]
47 | const deserialize = (streamData) => {
48 | const streamDataCopy = copyArray(streamData)
49 | const rootStreamData = getRootStreamData(streamDataCopy)
50 | const streamEvents = getStreamEvents(rootStreamData)
51 |
52 | return streamEvents.map((streamEvent) => {
53 | const eventId = getEventId(streamEvent)
54 |
55 | const eventData = getEventData(streamEvent)
56 | const deserializedFromRedis = mapArrayToObject(eventData)
57 |
58 | return {
59 | id: eventId,
60 | ...deserializedFromRedis,
61 | }
62 | })
63 | }
64 |
65 | const isStreamCreatedButWithoutItems = (streamData) => {
66 | return streamData[
67 | REDIS_ARRAY_POSITIONS.streamRoot
68 | ][
69 | REDIS_ARRAY_POSITIONS.root.streamEvents
70 | ].length === 0
71 | }
72 |
73 | const isStreamEmpty = (streamData) => {
74 | if (!streamData || streamData.length === 0 || isStreamCreatedButWithoutItems(streamData)) {
75 | return true
76 | }
77 | return false
78 | }
79 |
80 | const ack = ({ streamName, groupName, id }) => {
81 | return redis.xack(streamName, groupName, id)
82 | }
83 |
84 | const add = ({ streamName, maxStreamLength, formattedData }) => {
85 | return redis.xadd(streamName, 'MAXLEN', '~', maxStreamLength || 1000, '*', formattedData)
86 | }
87 |
88 | const readFromConsumerGroup = ({
89 | groupName, consumerName, streamName, shouldBlock, blockTime = 2000, count = 10, id,
90 | }) => {
91 | return shouldBlock
92 | ? redis.xreadgroup('GROUP', groupName, consumerName, 'BLOCK', blockTime, 'STREAMS', streamName, id)
93 | : redis.xreadgroup('GROUP', groupName, consumerName, 'COUNT', count, 'STREAMS', streamName, id)
94 | }
95 |
96 | const createConsumerGroup = (streamName, groupName) => {
97 | return redis.xgroup('CREATE', streamName, groupName, '$', 'MKSTREAM')
98 | }
99 |
100 | const publish = ({ streamName, maxStreamLength, eventPayload }) => {
101 |
102 | return add({
103 | streamName,
104 | maxStreamLength,
105 | formattedData: mapObjToStringArray(eventPayload),
106 | })
107 | }
108 |
109 | const subscribe = async ({
110 | groupName, streamName, readTimeout, workerFunction,
111 | }) => {
112 | // make sure consumer group exists
113 | try {
114 | await createConsumerGroup(streamName, groupName)
115 | } catch (err) {
116 | // ignore
117 | }
118 |
119 | const run = async ({ lastId, checkBacklog }) => {
120 | let readId = lastId
121 | let backlog = checkBacklog
122 |
123 | let streamData
124 |
125 | if (!backlog) {
126 | readId = READ_GROUP_EVENT_ID.NEW_ITEMS
127 | }
128 |
129 | try {
130 | streamData = await readFromConsumerGroup({
131 | groupName,
132 | consumerName: 'test',
133 | streamName,
134 | shouldBlock: false,
135 | blockTime: readTimeout,
136 | id: readId,
137 | })
138 | } catch (e) {
139 | console.error(`readFromConsumerGroup error: ${e}`)
140 | }
141 |
142 | if (isStreamEmpty(streamData)) {
143 | backlog = false
144 | } else {
145 | const deserializedEvents = deserialize(streamData)
146 |
147 | for (const event of deserializedEvents) {
148 | try {
149 | const { id, ...data } = event
150 |
151 | workerFunction({ ...data })
152 | await ack({ streamName, groupName, id })
153 | readId = id
154 | } catch (e) {
155 | console.error(e)
156 | }
157 | }
158 | }
159 |
160 | setTimeout(run.bind(null,
161 | { lastId: readId, checkBacklog: backlog }),
162 | readTimeout)
163 | }
164 |
165 | run({ lastId: READ_GROUP_EVENT_ID.FIRST_RUN, checkBacklog: true })
166 | }
167 |
168 | module.exports = {
169 | publish,
170 | subscribe
171 | }
172 |
--------------------------------------------------------------------------------
/bidding-service/streams-client.js:
--------------------------------------------------------------------------------
1 | const redis = require('./redis')
2 |
3 | const REDIS_ARRAY_POSITIONS = {
4 | streamRoot: 0,
5 | root: {
6 | streamName: 0,
7 | streamEvents: 1,
8 | },
9 | streamEvent: {
10 | eventId: 0,
11 | eventData: 1,
12 | },
13 | }
14 | const READ_GROUP_EVENT_ID = {
15 | FIRST_RUN: '0',
16 | NEW_ITEMS: '>',
17 | }
18 |
19 | const mapObjToStringArray = (obj) => {
20 | const arr = []
21 |
22 | for (const key in obj) {
23 | arr.push(key, obj[key])
24 | }
25 |
26 | return arr
27 | }
28 |
29 | const mapArrayToObject = (array) => {
30 | const arrayCopy = array.slice()
31 | const object = {}
32 |
33 | while (arrayCopy && arrayCopy.length) {
34 | const key = arrayCopy.shift()
35 | const value = arrayCopy.shift()
36 | object[key] = value
37 | }
38 |
39 | return object
40 | }
41 |
42 | const copyArray = (array) => array.slice()
43 | const getRootStreamData = (streamData) => streamData[REDIS_ARRAY_POSITIONS.streamRoot]
44 | const getStreamEvents = (rootStreamData) => rootStreamData[REDIS_ARRAY_POSITIONS.root.streamEvents]
45 | const getEventId = (streamEvent) => streamEvent[REDIS_ARRAY_POSITIONS.streamEvent.eventId]
46 | const getEventData = (streamEvent) => streamEvent[REDIS_ARRAY_POSITIONS.streamEvent.eventData]
47 | const deserialize = (streamData) => {
48 | const streamDataCopy = copyArray(streamData)
49 | const rootStreamData = getRootStreamData(streamDataCopy)
50 | const streamEvents = getStreamEvents(rootStreamData)
51 |
52 | return streamEvents.map((streamEvent) => {
53 | const eventId = getEventId(streamEvent)
54 |
55 | const eventData = getEventData(streamEvent)
56 | const deserializedFromRedis = mapArrayToObject(eventData)
57 |
58 | return {
59 | id: eventId,
60 | ...deserializedFromRedis,
61 | }
62 | })
63 | }
64 |
65 | const isStreamCreatedButWithoutItems = (streamData) => {
66 | return streamData[
67 | REDIS_ARRAY_POSITIONS.streamRoot
68 | ][
69 | REDIS_ARRAY_POSITIONS.root.streamEvents
70 | ].length === 0
71 | }
72 |
73 | const isStreamEmpty = (streamData) => {
74 | if (!streamData || streamData.length === 0 || isStreamCreatedButWithoutItems(streamData)) {
75 | return true
76 | }
77 | return false
78 | }
79 |
80 | const ack = ({ streamName, groupName, id }) => {
81 | return redis.xack(streamName, groupName, id)
82 | }
83 |
84 | const add = ({ streamName, maxStreamLength, formattedData }) => {
85 | return redis.xadd(streamName, 'MAXLEN', '~', maxStreamLength || 1000, '*', formattedData)
86 | }
87 |
88 | const readFromConsumerGroup = ({
89 | groupName, consumerName, streamName, shouldBlock, blockTime = 2000, count = 10, id,
90 | }) => {
91 | return shouldBlock
92 | ? redis.xreadgroup('GROUP', groupName, consumerName, 'BLOCK', blockTime, 'STREAMS', streamName, id)
93 | : redis.xreadgroup('GROUP', groupName, consumerName, 'COUNT', count, 'STREAMS', streamName, id)
94 | }
95 |
96 | const createConsumerGroup = (streamName, groupName) => {
97 | return redis.xgroup('CREATE', streamName, groupName, '$', 'MKSTREAM')
98 | }
99 |
100 | const publish = ({ streamName, maxStreamLength, eventPayload }) => {
101 |
102 | return add({
103 | streamName,
104 | maxStreamLength,
105 | formattedData: mapObjToStringArray(eventPayload),
106 | })
107 | }
108 |
109 | const subscribe = async ({
110 | groupName, streamName, readTimeout, workerFunction,
111 | }) => {
112 | // make sure consumer group exists
113 | try {
114 | await createConsumerGroup(streamName, groupName)
115 | } catch (err) {
116 | // ignore
117 | }
118 |
119 | const run = async ({ lastId, checkBacklog }) => {
120 | let readId = lastId
121 | let backlog = checkBacklog
122 |
123 | let streamData
124 |
125 | if (!backlog) {
126 | readId = READ_GROUP_EVENT_ID.NEW_ITEMS
127 | }
128 |
129 | try {
130 | streamData = await readFromConsumerGroup({
131 | groupName,
132 | consumerName: 'test',
133 | streamName,
134 | shouldBlock: false,
135 | blockTime: readTimeout,
136 | id: readId,
137 | })
138 | } catch (e) {
139 | console.error(`readFromConsumerGroup error: ${e}`)
140 | }
141 |
142 | if (isStreamEmpty(streamData)) {
143 | backlog = false
144 | } else {
145 | const deserializedEvents = deserialize(streamData)
146 |
147 | for (const event of deserializedEvents) {
148 | try {
149 | const { id, ...data } = event
150 |
151 | workerFunction({ ...data })
152 | await ack({ streamName, groupName, id })
153 | readId = id
154 | } catch (e) {
155 | console.error(e)
156 | }
157 | }
158 | }
159 |
160 | setTimeout(run.bind(null,
161 | { lastId: readId, checkBacklog: backlog }),
162 | readTimeout)
163 | }
164 |
165 | run({ lastId: READ_GROUP_EVENT_ID.FIRST_RUN, checkBacklog: true })
166 | }
167 |
168 | module.exports = {
169 | publish,
170 | subscribe
171 | }
172 |
--------------------------------------------------------------------------------
/api/package-lock.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "auctioneer",
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/component-emitter": {
23 | "version": "1.2.10",
24 | "resolved": "https://registry.npmjs.org/@types/component-emitter/-/component-emitter-1.2.10.tgz",
25 | "integrity": "sha512-bsjleuRKWmGqajMerkzox19aGbscQX5rmmvvXl3wlIp5gMG1HgkiwPxsN5p070fBDKTNSPgojVbuY1+HWMbFhg=="
26 | },
27 | "@types/cookie": {
28 | "version": "0.4.0",
29 | "resolved": "https://registry.npmjs.org/@types/cookie/-/cookie-0.4.0.tgz",
30 | "integrity": "sha512-y7mImlc/rNkvCRmg8gC3/lj87S7pTUIJ6QGjwHR9WQJcFs+ZMTOaoPrkdFA/YdbuqVEmEbb5RdhVxMkAcgOnpg=="
31 | },
32 | "@types/cors": {
33 | "version": "2.8.10",
34 | "resolved": "https://registry.npmjs.org/@types/cors/-/cors-2.8.10.tgz",
35 | "integrity": "sha512-C7srjHiVG3Ey1nR6d511dtDkCEjxuN9W1HWAEjGq8kpcwmNM6JJkpC0xvabM7BXTG2wDq8Eu33iH9aQKa7IvLQ=="
36 | },
37 | "@types/node": {
38 | "version": "15.0.2",
39 | "resolved": "https://registry.npmjs.org/@types/node/-/node-15.0.2.tgz",
40 | "integrity": "sha512-p68+a+KoxpoB47015IeYZYRrdqMUcpbK8re/zpFB8Ld46LHC1lPEbp3EXgkEhAYEcPvjJF6ZO+869SQ0aH1dcA=="
41 | },
42 | "abbrev": {
43 | "version": "1.1.1",
44 | "resolved": "https://registry.npmjs.org/abbrev/-/abbrev-1.1.1.tgz",
45 | "integrity": "sha512-nne9/IiQ/hzIhY6pdDnbBtz7DjPTKrY00P/zvPSm5pOFkl6xuGrGnXn/VtTNNfNtAfZ9/1RtehkszU9qcTii0Q==",
46 | "dev": true
47 | },
48 | "accepts": {
49 | "version": "1.3.7",
50 | "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.7.tgz",
51 | "integrity": "sha512-Il80Qs2WjYlJIBNzNkK6KYqlVMTbZLXgHx2oT0pU/fjRHyEp+PEfEPY0R3WCwAGVOtauxh1hOxNgIf5bv7dQpA==",
52 | "requires": {
53 | "mime-types": "~2.1.24",
54 | "negotiator": "0.6.2"
55 | }
56 | },
57 | "ansi-align": {
58 | "version": "3.0.0",
59 | "resolved": "https://registry.npmjs.org/ansi-align/-/ansi-align-3.0.0.tgz",
60 | "integrity": "sha512-ZpClVKqXN3RGBmKibdfWzqCY4lnjEuoNzU5T0oEFpfd/z5qJHVarukridD4juLO2FXMiwUQxr9WqQtaYa8XRYw==",
61 | "dev": true,
62 | "requires": {
63 | "string-width": "^3.0.0"
64 | },
65 | "dependencies": {
66 | "string-width": {
67 | "version": "3.1.0",
68 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz",
69 | "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==",
70 | "dev": true,
71 | "requires": {
72 | "emoji-regex": "^7.0.1",
73 | "is-fullwidth-code-point": "^2.0.0",
74 | "strip-ansi": "^5.1.0"
75 | }
76 | }
77 | }
78 | },
79 | "ansi-regex": {
80 | "version": "4.1.0",
81 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz",
82 | "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==",
83 | "dev": true
84 | },
85 | "ansi-styles": {
86 | "version": "4.3.0",
87 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz",
88 | "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==",
89 | "dev": true,
90 | "requires": {
91 | "color-convert": "^2.0.1"
92 | }
93 | },
94 | "anymatch": {
95 | "version": "3.1.2",
96 | "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.2.tgz",
97 | "integrity": "sha512-P43ePfOAIupkguHUycrc4qJ9kz8ZiuOUijaETwX7THt0Y/GNK7v0aa8rY816xWjZ7rJdA5XdMcpVFTKMq+RvWg==",
98 | "dev": true,
99 | "requires": {
100 | "normalize-path": "^3.0.0",
101 | "picomatch": "^2.0.4"
102 | }
103 | },
104 | "array-flatten": {
105 | "version": "1.1.1",
106 | "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz",
107 | "integrity": "sha1-ml9pkFGx5wczKPKgCJaLZOopVdI="
108 | },
109 | "balanced-match": {
110 | "version": "1.0.2",
111 | "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz",
112 | "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==",
113 | "dev": true
114 | },
115 | "base64-arraybuffer": {
116 | "version": "0.1.4",
117 | "resolved": "https://registry.npmjs.org/base64-arraybuffer/-/base64-arraybuffer-0.1.4.tgz",
118 | "integrity": "sha1-mBjHngWbE1X5fgQooBfIOOkLqBI="
119 | },
120 | "base64id": {
121 | "version": "2.0.0",
122 | "resolved": "https://registry.npmjs.org/base64id/-/base64id-2.0.0.tgz",
123 | "integrity": "sha512-lGe34o6EHj9y3Kts9R4ZYs/Gr+6N7MCaMlIFA3F1R2O5/m7K06AxfSeO5530PEERE6/WyEg3lsuyw4GHlPZHog=="
124 | },
125 | "binary-extensions": {
126 | "version": "2.2.0",
127 | "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.2.0.tgz",
128 | "integrity": "sha512-jDctJ/IVQbZoJykoeHbhXpOlNBqGNcwXJKJog42E5HDPUwQTSdjCHdihjj0DlnheQ7blbT6dHOafNAiS8ooQKA==",
129 | "dev": true
130 | },
131 | "body-parser": {
132 | "version": "1.19.0",
133 | "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.19.0.tgz",
134 | "integrity": "sha512-dhEPs72UPbDnAQJ9ZKMNTP6ptJaionhP5cBb541nXPlW60Jepo9RV/a4fX4XWW9CuFNK22krhrj1+rgzifNCsw==",
135 | "requires": {
136 | "bytes": "3.1.0",
137 | "content-type": "~1.0.4",
138 | "debug": "2.6.9",
139 | "depd": "~1.1.2",
140 | "http-errors": "1.7.2",
141 | "iconv-lite": "0.4.24",
142 | "on-finished": "~2.3.0",
143 | "qs": "6.7.0",
144 | "raw-body": "2.4.0",
145 | "type-is": "~1.6.17"
146 | },
147 | "dependencies": {
148 | "debug": {
149 | "version": "2.6.9",
150 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz",
151 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==",
152 | "requires": {
153 | "ms": "2.0.0"
154 | }
155 | },
156 | "ms": {
157 | "version": "2.0.0",
158 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz",
159 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g="
160 | }
161 | }
162 | },
163 | "boxen": {
164 | "version": "4.2.0",
165 | "resolved": "https://registry.npmjs.org/boxen/-/boxen-4.2.0.tgz",
166 | "integrity": "sha512-eB4uT9RGzg2odpER62bBwSLvUeGC+WbRjjyyFhGsKnc8wp/m0+hQsMUvUe3H2V0D5vw0nBdO1hCJoZo5mKeuIQ==",
167 | "dev": true,
168 | "requires": {
169 | "ansi-align": "^3.0.0",
170 | "camelcase": "^5.3.1",
171 | "chalk": "^3.0.0",
172 | "cli-boxes": "^2.2.0",
173 | "string-width": "^4.1.0",
174 | "term-size": "^2.1.0",
175 | "type-fest": "^0.8.1",
176 | "widest-line": "^3.1.0"
177 | }
178 | },
179 | "brace-expansion": {
180 | "version": "1.1.11",
181 | "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz",
182 | "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==",
183 | "dev": true,
184 | "requires": {
185 | "balanced-match": "^1.0.0",
186 | "concat-map": "0.0.1"
187 | }
188 | },
189 | "braces": {
190 | "version": "3.0.2",
191 | "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz",
192 | "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==",
193 | "dev": true,
194 | "requires": {
195 | "fill-range": "^7.0.1"
196 | }
197 | },
198 | "bytes": {
199 | "version": "3.1.0",
200 | "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.0.tgz",
201 | "integrity": "sha512-zauLjrfCG+xvoyaqLoV8bLVXXNGC4JqlxFCutSDWA6fJrTo2ZuvLYTqZ7aHBLZSMOopbzwv8f+wZcVzfVTI2Dg=="
202 | },
203 | "cacheable-request": {
204 | "version": "6.1.0",
205 | "resolved": "https://registry.npmjs.org/cacheable-request/-/cacheable-request-6.1.0.tgz",
206 | "integrity": "sha512-Oj3cAGPCqOZX7Rz64Uny2GYAZNliQSqfbePrgAQ1wKAihYmCUnraBtJtKcGR4xz7wF+LoJC+ssFZvv5BgF9Igg==",
207 | "dev": true,
208 | "requires": {
209 | "clone-response": "^1.0.2",
210 | "get-stream": "^5.1.0",
211 | "http-cache-semantics": "^4.0.0",
212 | "keyv": "^3.0.0",
213 | "lowercase-keys": "^2.0.0",
214 | "normalize-url": "^4.1.0",
215 | "responselike": "^1.0.2"
216 | },
217 | "dependencies": {
218 | "get-stream": {
219 | "version": "5.2.0",
220 | "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-5.2.0.tgz",
221 | "integrity": "sha512-nBF+F1rAZVCu/p7rjzgA+Yb4lfYXrpl7a6VmJrU8wF9I1CKvP/QwPNZHnOlwbTkY6dvtFIzFMSyQXbLoTQPRpA==",
222 | "dev": true,
223 | "requires": {
224 | "pump": "^3.0.0"
225 | }
226 | },
227 | "lowercase-keys": {
228 | "version": "2.0.0",
229 | "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-2.0.0.tgz",
230 | "integrity": "sha512-tqNXrS78oMOE73NMxK4EMLQsQowWf8jKooH9g7xPavRT706R6bkQJ6DY2Te7QukaZsulxa30wQ7bk0pm4XiHmA==",
231 | "dev": true
232 | }
233 | }
234 | },
235 | "camelcase": {
236 | "version": "5.3.1",
237 | "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz",
238 | "integrity": "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==",
239 | "dev": true
240 | },
241 | "chalk": {
242 | "version": "3.0.0",
243 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-3.0.0.tgz",
244 | "integrity": "sha512-4D3B6Wf41KOYRFdszmDqMCGq5VV/uMAB273JILmO+3jAlh8X4qDtdtgCR3fxtbLEMzSx22QdhnDcJvu2u1fVwg==",
245 | "dev": true,
246 | "requires": {
247 | "ansi-styles": "^4.1.0",
248 | "supports-color": "^7.1.0"
249 | },
250 | "dependencies": {
251 | "has-flag": {
252 | "version": "4.0.0",
253 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz",
254 | "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==",
255 | "dev": true
256 | },
257 | "supports-color": {
258 | "version": "7.2.0",
259 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz",
260 | "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==",
261 | "dev": true,
262 | "requires": {
263 | "has-flag": "^4.0.0"
264 | }
265 | }
266 | }
267 | },
268 | "chokidar": {
269 | "version": "3.5.1",
270 | "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.5.1.tgz",
271 | "integrity": "sha512-9+s+Od+W0VJJzawDma/gvBNQqkTiqYTWLuZoyAsivsI4AaWTCzHG06/TMjsf1cYe9Cb97UCEhjz7HvnPk2p/tw==",
272 | "dev": true,
273 | "requires": {
274 | "anymatch": "~3.1.1",
275 | "braces": "~3.0.2",
276 | "fsevents": "~2.3.1",
277 | "glob-parent": "~5.1.0",
278 | "is-binary-path": "~2.1.0",
279 | "is-glob": "~4.0.1",
280 | "normalize-path": "~3.0.0",
281 | "readdirp": "~3.5.0"
282 | }
283 | },
284 | "ci-info": {
285 | "version": "2.0.0",
286 | "resolved": "https://registry.npmjs.org/ci-info/-/ci-info-2.0.0.tgz",
287 | "integrity": "sha512-5tK7EtrZ0N+OLFMthtqOj4fI2Jeb88C4CAZPu25LDVUgXJ0A3Js4PMGqrn0JU1W0Mh1/Z8wZzYPxqUrXeBboCQ==",
288 | "dev": true
289 | },
290 | "cli-boxes": {
291 | "version": "2.2.1",
292 | "resolved": "https://registry.npmjs.org/cli-boxes/-/cli-boxes-2.2.1.tgz",
293 | "integrity": "sha512-y4coMcylgSCdVinjiDBuR8PCC2bLjyGTwEmPb9NHR/QaNU6EUOXcTY/s6VjGMD6ENSEaeQYHCY0GNGS5jfMwPw==",
294 | "dev": true
295 | },
296 | "clone-response": {
297 | "version": "1.0.2",
298 | "resolved": "https://registry.npmjs.org/clone-response/-/clone-response-1.0.2.tgz",
299 | "integrity": "sha1-0dyXOSAxTfZ/vrlCI7TuNQI56Ws=",
300 | "dev": true,
301 | "requires": {
302 | "mimic-response": "^1.0.0"
303 | }
304 | },
305 | "cluster-key-slot": {
306 | "version": "1.1.0",
307 | "resolved": "https://registry.npmjs.org/cluster-key-slot/-/cluster-key-slot-1.1.0.tgz",
308 | "integrity": "sha512-2Nii8p3RwAPiFwsnZvukotvow2rIHM+yQ6ZcBXGHdniadkYGZYiGmkHJIbZPIV9nfv7m/U1IPMVVcAhoWFeklw=="
309 | },
310 | "color-convert": {
311 | "version": "2.0.1",
312 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz",
313 | "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==",
314 | "dev": true,
315 | "requires": {
316 | "color-name": "~1.1.4"
317 | }
318 | },
319 | "color-name": {
320 | "version": "1.1.4",
321 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz",
322 | "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==",
323 | "dev": true
324 | },
325 | "common-tags": {
326 | "version": "1.8.0",
327 | "resolved": "https://registry.npmjs.org/common-tags/-/common-tags-1.8.0.tgz",
328 | "integrity": "sha512-6P6g0uetGpW/sdyUy/iQQCbFF0kWVMSIVSyYz7Zgjcgh8mgw8PQzDNZeyZ5DQ2gM7LBoZPHmnjz8rUthkBG5tw=="
329 | },
330 | "component-emitter": {
331 | "version": "1.3.0",
332 | "resolved": "https://registry.npmjs.org/component-emitter/-/component-emitter-1.3.0.tgz",
333 | "integrity": "sha512-Rd3se6QB+sO1TwqZjscQrurpEPIfO0/yYnSin6Q/rD3mOutHvUrCAhJub3r90uNb+SESBuE0QYoB90YdfatsRg=="
334 | },
335 | "concat-map": {
336 | "version": "0.0.1",
337 | "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz",
338 | "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=",
339 | "dev": true
340 | },
341 | "configstore": {
342 | "version": "5.0.1",
343 | "resolved": "https://registry.npmjs.org/configstore/-/configstore-5.0.1.tgz",
344 | "integrity": "sha512-aMKprgk5YhBNyH25hj8wGt2+D52Sw1DRRIzqBwLp2Ya9mFmY8KPvvtvmna8SxVR9JMZ4kzMD68N22vlaRpkeFA==",
345 | "dev": true,
346 | "requires": {
347 | "dot-prop": "^5.2.0",
348 | "graceful-fs": "^4.1.2",
349 | "make-dir": "^3.0.0",
350 | "unique-string": "^2.0.0",
351 | "write-file-atomic": "^3.0.0",
352 | "xdg-basedir": "^4.0.0"
353 | }
354 | },
355 | "content-disposition": {
356 | "version": "0.5.3",
357 | "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.3.tgz",
358 | "integrity": "sha512-ExO0774ikEObIAEV9kDo50o+79VCUdEB6n6lzKgGwupcVeRlhrj3qGAfwq8G6uBJjkqLrhT0qEYFcWng8z1z0g==",
359 | "requires": {
360 | "safe-buffer": "5.1.2"
361 | }
362 | },
363 | "content-type": {
364 | "version": "1.0.4",
365 | "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.4.tgz",
366 | "integrity": "sha512-hIP3EEPs8tB9AT1L+NUqtwOAps4mk2Zob89MWXMHjHWg9milF/j4osnnQLXBCBFBk/tvIG/tUc9mOUJiPBhPXA=="
367 | },
368 | "cookie": {
369 | "version": "0.4.0",
370 | "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.4.0.tgz",
371 | "integrity": "sha512-+Hp8fLp57wnUSt0tY0tHEXh4voZRDnoIrZPqlo3DPiI4y9lwg/jqx+1Om94/W6ZaPDOUbnjOt/99w66zk+l1Xg=="
372 | },
373 | "cookie-signature": {
374 | "version": "1.0.6",
375 | "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz",
376 | "integrity": "sha1-4wOogrNCzD7oylE6eZmXNNqzriw="
377 | },
378 | "cors": {
379 | "version": "2.8.5",
380 | "resolved": "https://registry.npmjs.org/cors/-/cors-2.8.5.tgz",
381 | "integrity": "sha512-KIHbLJqu73RGr/hnbrO9uBeixNGuvSQjul/jdFvS/KFSIH1hWVd1ng7zOHx+YrEfInLG7q4n6GHQ9cDtxv/P6g==",
382 | "requires": {
383 | "object-assign": "^4",
384 | "vary": "^1"
385 | }
386 | },
387 | "crypto-random-string": {
388 | "version": "2.0.0",
389 | "resolved": "https://registry.npmjs.org/crypto-random-string/-/crypto-random-string-2.0.0.tgz",
390 | "integrity": "sha512-v1plID3y9r/lPhviJ1wrXpLeyUIGAZ2SHNYTEapm7/8A9nLPoyvVp3RK/EPFqn5kEznyWgYZNsRtYYIWbuG8KA==",
391 | "dev": true
392 | },
393 | "debug": {
394 | "version": "4.3.1",
395 | "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.1.tgz",
396 | "integrity": "sha512-doEwdvm4PCeK4K3RQN2ZC2BYUBaxwLARCqZmMjtF8a51J2Rb0xpVloFRnCODwqjpwnAoao4pelN8l3RJdv3gRQ==",
397 | "requires": {
398 | "ms": "2.1.2"
399 | }
400 | },
401 | "decompress-response": {
402 | "version": "3.3.0",
403 | "resolved": "https://registry.npmjs.org/decompress-response/-/decompress-response-3.3.0.tgz",
404 | "integrity": "sha1-gKTdMjdIOEv6JICDYirt7Jgq3/M=",
405 | "dev": true,
406 | "requires": {
407 | "mimic-response": "^1.0.0"
408 | }
409 | },
410 | "deep-extend": {
411 | "version": "0.6.0",
412 | "resolved": "https://registry.npmjs.org/deep-extend/-/deep-extend-0.6.0.tgz",
413 | "integrity": "sha512-LOHxIOaPYdHlJRtCQfDIVZtfw/ufM8+rVj649RIHzcm/vGwQRXFt6OPqIFWsm2XEMrNIEtWR64sY1LEKD2vAOA==",
414 | "dev": true
415 | },
416 | "defer-to-connect": {
417 | "version": "1.1.3",
418 | "resolved": "https://registry.npmjs.org/defer-to-connect/-/defer-to-connect-1.1.3.tgz",
419 | "integrity": "sha512-0ISdNousHvZT2EiFlZeZAHBUvSxmKswVCEf8hW7KWgG4a8MVEu/3Vb6uWYozkjylyCxe0JBIiRB1jV45S70WVQ==",
420 | "dev": true
421 | },
422 | "denque": {
423 | "version": "1.5.0",
424 | "resolved": "https://registry.npmjs.org/denque/-/denque-1.5.0.tgz",
425 | "integrity": "sha512-CYiCSgIF1p6EUByQPlGkKnP1M9g0ZV3qMIrqMqZqdwazygIA/YP2vrbcyl1h/WppKJTdl1F85cXIle+394iDAQ=="
426 | },
427 | "depd": {
428 | "version": "1.1.2",
429 | "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.2.tgz",
430 | "integrity": "sha1-m81S4UwJd2PnSbJ0xDRu0uVgtak="
431 | },
432 | "destroy": {
433 | "version": "1.0.4",
434 | "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.0.4.tgz",
435 | "integrity": "sha1-l4hXRCxEdJ5CBmE+N5RiBYJqvYA="
436 | },
437 | "dot-prop": {
438 | "version": "5.3.0",
439 | "resolved": "https://registry.npmjs.org/dot-prop/-/dot-prop-5.3.0.tgz",
440 | "integrity": "sha512-QM8q3zDe58hqUqjraQOmzZ1LIH9SWQJTlEKCH4kJ2oQvLZk7RbQXvtDM2XEq3fwkV9CCvvH4LA0AV+ogFsBM2Q==",
441 | "dev": true,
442 | "requires": {
443 | "is-obj": "^2.0.0"
444 | }
445 | },
446 | "dotenv": {
447 | "version": "8.2.0",
448 | "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-8.2.0.tgz",
449 | "integrity": "sha512-8sJ78ElpbDJBHNeBzUbUVLsqKdccaa/BXF1uPTw3GrvQTBgrQrtObr2mUrE38vzYd8cEv+m/JBfDLioYcfXoaw=="
450 | },
451 | "duplexer3": {
452 | "version": "0.1.4",
453 | "resolved": "https://registry.npmjs.org/duplexer3/-/duplexer3-0.1.4.tgz",
454 | "integrity": "sha1-7gHdHKwO08vH/b6jfcCo8c4ALOI=",
455 | "dev": true
456 | },
457 | "ee-first": {
458 | "version": "1.1.1",
459 | "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz",
460 | "integrity": "sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0="
461 | },
462 | "emoji-regex": {
463 | "version": "7.0.3",
464 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-7.0.3.tgz",
465 | "integrity": "sha512-CwBLREIQ7LvYFB0WyRvwhq5N5qPhc6PMjD6bYggFlI5YyDgl+0vxq5VHbMOFqLg7hfWzmu8T5Z1QofhmTIhItA==",
466 | "dev": true
467 | },
468 | "encodeurl": {
469 | "version": "1.0.2",
470 | "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz",
471 | "integrity": "sha1-rT/0yG7C0CkyL1oCw6mmBslbP1k="
472 | },
473 | "end-of-stream": {
474 | "version": "1.4.4",
475 | "resolved": "https://registry.npmjs.org/end-of-stream/-/end-of-stream-1.4.4.tgz",
476 | "integrity": "sha512-+uw1inIHVPQoaVuHzRyXd21icM+cnt4CzD5rW+NC1wjOUSTOs+Te7FOv7AhN7vS9x/oIyhLP5PR1H+phQAHu5Q==",
477 | "dev": true,
478 | "requires": {
479 | "once": "^1.4.0"
480 | }
481 | },
482 | "engine.io": {
483 | "version": "5.0.0",
484 | "resolved": "https://registry.npmjs.org/engine.io/-/engine.io-5.0.0.tgz",
485 | "integrity": "sha512-BATIdDV3H1SrE9/u2BAotvsmjJg0t1P4+vGedImSs1lkFAtQdvk4Ev1y4LDiPF7BPWgXWEG+NDY+nLvW3UrMWw==",
486 | "requires": {
487 | "accepts": "~1.3.4",
488 | "base64id": "2.0.0",
489 | "cookie": "~0.4.1",
490 | "cors": "~2.8.5",
491 | "debug": "~4.3.1",
492 | "engine.io-parser": "~4.0.0",
493 | "ws": "~7.4.2"
494 | },
495 | "dependencies": {
496 | "cookie": {
497 | "version": "0.4.1",
498 | "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.4.1.tgz",
499 | "integrity": "sha512-ZwrFkGJxUR3EIoXtO+yVE69Eb7KlixbaeAWfBQB9vVsNn/o+Yw69gBWSSDK825hQNdN+wF8zELf3dFNl/kxkUA=="
500 | }
501 | }
502 | },
503 | "engine.io-parser": {
504 | "version": "4.0.2",
505 | "resolved": "https://registry.npmjs.org/engine.io-parser/-/engine.io-parser-4.0.2.tgz",
506 | "integrity": "sha512-sHfEQv6nmtJrq6TKuIz5kyEKH/qSdK56H/A+7DnAuUPWosnIZAS2NHNcPLmyjtY3cGS/MqJdZbUjW97JU72iYg==",
507 | "requires": {
508 | "base64-arraybuffer": "0.1.4"
509 | }
510 | },
511 | "escape-goat": {
512 | "version": "2.1.1",
513 | "resolved": "https://registry.npmjs.org/escape-goat/-/escape-goat-2.1.1.tgz",
514 | "integrity": "sha512-8/uIhbG12Csjy2JEW7D9pHbreaVaS/OpN3ycnyvElTdwM5n6GY6W6e2IPemfvGZeUMqZ9A/3GqIZMgKnBhAw/Q==",
515 | "dev": true
516 | },
517 | "escape-html": {
518 | "version": "1.0.3",
519 | "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz",
520 | "integrity": "sha1-Aljq5NPQwJdN4cFpGI7wBR0dGYg="
521 | },
522 | "esm": {
523 | "version": "3.2.25",
524 | "resolved": "https://registry.npmjs.org/esm/-/esm-3.2.25.tgz",
525 | "integrity": "sha512-U1suiZ2oDVWv4zPO56S0NcR5QriEahGtdN2OR6FiOG4WJvcjBVFB0qI4+eKoWFH483PKGuLuu6V8Z4T5g63UVA=="
526 | },
527 | "etag": {
528 | "version": "1.8.1",
529 | "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz",
530 | "integrity": "sha1-Qa4u62XvpiJorr/qg6x9eSmbCIc="
531 | },
532 | "express": {
533 | "version": "4.17.1",
534 | "resolved": "https://registry.npmjs.org/express/-/express-4.17.1.tgz",
535 | "integrity": "sha512-mHJ9O79RqluphRrcw2X/GTh3k9tVv8YcoyY4Kkh4WDMUYKRZUq0h1o0w2rrrxBqM7VoeUVqgb27xlEMXTnYt4g==",
536 | "requires": {
537 | "accepts": "~1.3.7",
538 | "array-flatten": "1.1.1",
539 | "body-parser": "1.19.0",
540 | "content-disposition": "0.5.3",
541 | "content-type": "~1.0.4",
542 | "cookie": "0.4.0",
543 | "cookie-signature": "1.0.6",
544 | "debug": "2.6.9",
545 | "depd": "~1.1.2",
546 | "encodeurl": "~1.0.2",
547 | "escape-html": "~1.0.3",
548 | "etag": "~1.8.1",
549 | "finalhandler": "~1.1.2",
550 | "fresh": "0.5.2",
551 | "merge-descriptors": "1.0.1",
552 | "methods": "~1.1.2",
553 | "on-finished": "~2.3.0",
554 | "parseurl": "~1.3.3",
555 | "path-to-regexp": "0.1.7",
556 | "proxy-addr": "~2.0.5",
557 | "qs": "6.7.0",
558 | "range-parser": "~1.2.1",
559 | "safe-buffer": "5.1.2",
560 | "send": "0.17.1",
561 | "serve-static": "1.14.1",
562 | "setprototypeof": "1.1.1",
563 | "statuses": "~1.5.0",
564 | "type-is": "~1.6.18",
565 | "utils-merge": "1.0.1",
566 | "vary": "~1.1.2"
567 | },
568 | "dependencies": {
569 | "debug": {
570 | "version": "2.6.9",
571 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz",
572 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==",
573 | "requires": {
574 | "ms": "2.0.0"
575 | }
576 | },
577 | "ms": {
578 | "version": "2.0.0",
579 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz",
580 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g="
581 | }
582 | }
583 | },
584 | "fill-range": {
585 | "version": "7.0.1",
586 | "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz",
587 | "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==",
588 | "dev": true,
589 | "requires": {
590 | "to-regex-range": "^5.0.1"
591 | }
592 | },
593 | "finalhandler": {
594 | "version": "1.1.2",
595 | "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.1.2.tgz",
596 | "integrity": "sha512-aAWcW57uxVNrQZqFXjITpW3sIUQmHGG3qSb9mUah9MgMC4NeWhNOlNjXEYq3HjRAvL6arUviZGGJsBg6z0zsWA==",
597 | "requires": {
598 | "debug": "2.6.9",
599 | "encodeurl": "~1.0.2",
600 | "escape-html": "~1.0.3",
601 | "on-finished": "~2.3.0",
602 | "parseurl": "~1.3.3",
603 | "statuses": "~1.5.0",
604 | "unpipe": "~1.0.0"
605 | },
606 | "dependencies": {
607 | "debug": {
608 | "version": "2.6.9",
609 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz",
610 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==",
611 | "requires": {
612 | "ms": "2.0.0"
613 | }
614 | },
615 | "ms": {
616 | "version": "2.0.0",
617 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz",
618 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g="
619 | }
620 | }
621 | },
622 | "forwarded": {
623 | "version": "0.1.2",
624 | "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.1.2.tgz",
625 | "integrity": "sha1-mMI9qxF1ZXuMBXPozszZGw/xjIQ="
626 | },
627 | "fresh": {
628 | "version": "0.5.2",
629 | "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz",
630 | "integrity": "sha1-PYyt2Q2XZWn6g1qx+OSyOhBWBac="
631 | },
632 | "fsevents": {
633 | "version": "2.3.2",
634 | "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.2.tgz",
635 | "integrity": "sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==",
636 | "dev": true,
637 | "optional": true
638 | },
639 | "get-stream": {
640 | "version": "4.1.0",
641 | "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-4.1.0.tgz",
642 | "integrity": "sha512-GMat4EJ5161kIy2HevLlr4luNjBgvmj413KaQA7jt4V8B4RDsfpHk7WQ9GVqfYyyx8OS/L66Kox+rJRNklLK7w==",
643 | "dev": true,
644 | "requires": {
645 | "pump": "^3.0.0"
646 | }
647 | },
648 | "glob-parent": {
649 | "version": "5.1.2",
650 | "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz",
651 | "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==",
652 | "dev": true,
653 | "requires": {
654 | "is-glob": "^4.0.1"
655 | }
656 | },
657 | "global-dirs": {
658 | "version": "2.1.0",
659 | "resolved": "https://registry.npmjs.org/global-dirs/-/global-dirs-2.1.0.tgz",
660 | "integrity": "sha512-MG6kdOUh/xBnyo9cJFeIKkLEc1AyFq42QTU4XiX51i2NEdxLxLWXIjEjmqKeSuKR7pAZjTqUVoT2b2huxVLgYQ==",
661 | "dev": true,
662 | "requires": {
663 | "ini": "1.3.7"
664 | }
665 | },
666 | "got": {
667 | "version": "9.6.0",
668 | "resolved": "https://registry.npmjs.org/got/-/got-9.6.0.tgz",
669 | "integrity": "sha512-R7eWptXuGYxwijs0eV+v3o6+XH1IqVK8dJOEecQfTmkncw9AV4dcw/Dhxi8MdlqPthxxpZyizMzyg8RTmEsG+Q==",
670 | "dev": true,
671 | "requires": {
672 | "@sindresorhus/is": "^0.14.0",
673 | "@szmarczak/http-timer": "^1.1.2",
674 | "cacheable-request": "^6.0.0",
675 | "decompress-response": "^3.3.0",
676 | "duplexer3": "^0.1.4",
677 | "get-stream": "^4.1.0",
678 | "lowercase-keys": "^1.0.1",
679 | "mimic-response": "^1.0.1",
680 | "p-cancelable": "^1.0.0",
681 | "to-readable-stream": "^1.0.0",
682 | "url-parse-lax": "^3.0.0"
683 | }
684 | },
685 | "graceful-fs": {
686 | "version": "4.2.6",
687 | "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.6.tgz",
688 | "integrity": "sha512-nTnJ528pbqxYanhpDYsi4Rd8MAeaBA67+RZ10CM1m3bTAVFEDcd5AuA4a6W5YkGZ1iNXHzZz8T6TBKLeBuNriQ==",
689 | "dev": true
690 | },
691 | "has-flag": {
692 | "version": "3.0.0",
693 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz",
694 | "integrity": "sha1-tdRU3CGZriJWmfNGfloH87lVuv0=",
695 | "dev": true
696 | },
697 | "has-yarn": {
698 | "version": "2.1.0",
699 | "resolved": "https://registry.npmjs.org/has-yarn/-/has-yarn-2.1.0.tgz",
700 | "integrity": "sha512-UqBRqi4ju7T+TqGNdqAO0PaSVGsDGJUBQvk9eUWNGRY1CFGDzYhLWoM7JQEemnlvVcv/YEmc2wNW8BC24EnUsw==",
701 | "dev": true
702 | },
703 | "http-cache-semantics": {
704 | "version": "4.1.0",
705 | "resolved": "https://registry.npmjs.org/http-cache-semantics/-/http-cache-semantics-4.1.0.tgz",
706 | "integrity": "sha512-carPklcUh7ROWRK7Cv27RPtdhYhUsela/ue5/jKzjegVvXDqM2ILE9Q2BGn9JZJh1g87cp56su/FgQSzcWS8cQ==",
707 | "dev": true
708 | },
709 | "http-errors": {
710 | "version": "1.7.2",
711 | "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.7.2.tgz",
712 | "integrity": "sha512-uUQBt3H/cSIVfch6i1EuPNy/YsRSOUBXTVfZ+yR7Zjez3qjBz6i9+i4zjNaoqcoFVI4lQJ5plg63TvGfRSDCRg==",
713 | "requires": {
714 | "depd": "~1.1.2",
715 | "inherits": "2.0.3",
716 | "setprototypeof": "1.1.1",
717 | "statuses": ">= 1.5.0 < 2",
718 | "toidentifier": "1.0.0"
719 | }
720 | },
721 | "iconv-lite": {
722 | "version": "0.4.24",
723 | "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz",
724 | "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==",
725 | "requires": {
726 | "safer-buffer": ">= 2.1.2 < 3"
727 | }
728 | },
729 | "ignore-by-default": {
730 | "version": "1.0.1",
731 | "resolved": "https://registry.npmjs.org/ignore-by-default/-/ignore-by-default-1.0.1.tgz",
732 | "integrity": "sha1-SMptcvbGo68Aqa1K5odr44ieKwk=",
733 | "dev": true
734 | },
735 | "import-lazy": {
736 | "version": "2.1.0",
737 | "resolved": "https://registry.npmjs.org/import-lazy/-/import-lazy-2.1.0.tgz",
738 | "integrity": "sha1-BWmOPUXIjo1+nZLLBYTnfwlvPkM=",
739 | "dev": true
740 | },
741 | "imurmurhash": {
742 | "version": "0.1.4",
743 | "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz",
744 | "integrity": "sha1-khi5srkoojixPcT7a21XbyMUU+o=",
745 | "dev": true
746 | },
747 | "inherits": {
748 | "version": "2.0.3",
749 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz",
750 | "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4="
751 | },
752 | "ini": {
753 | "version": "1.3.7",
754 | "resolved": "https://registry.npmjs.org/ini/-/ini-1.3.7.tgz",
755 | "integrity": "sha512-iKpRpXP+CrP2jyrxvg1kMUpXDyRUFDWurxbnVT1vQPx+Wz9uCYsMIqYuSBLV+PAaZG/d7kRLKRFc9oDMsH+mFQ==",
756 | "dev": true
757 | },
758 | "ioredis": {
759 | "version": "4.26.0",
760 | "resolved": "https://registry.npmjs.org/ioredis/-/ioredis-4.26.0.tgz",
761 | "integrity": "sha512-nh39okWezWWZ35/RxXXzHksMFt4WCaev8SNO2kozRDeVdEAJj16EarqPP3JeHz8IEjEXN5CiVtbWMk62Z0eveQ==",
762 | "requires": {
763 | "cluster-key-slot": "^1.1.0",
764 | "debug": "^4.3.1",
765 | "denque": "^1.1.0",
766 | "lodash.defaults": "^4.2.0",
767 | "lodash.flatten": "^4.4.0",
768 | "p-map": "^2.1.0",
769 | "redis-commands": "1.7.0",
770 | "redis-errors": "^1.2.0",
771 | "redis-parser": "^3.0.0",
772 | "standard-as-callback": "^2.1.0"
773 | }
774 | },
775 | "ipaddr.js": {
776 | "version": "1.9.1",
777 | "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz",
778 | "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g=="
779 | },
780 | "is-binary-path": {
781 | "version": "2.1.0",
782 | "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz",
783 | "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==",
784 | "dev": true,
785 | "requires": {
786 | "binary-extensions": "^2.0.0"
787 | }
788 | },
789 | "is-ci": {
790 | "version": "2.0.0",
791 | "resolved": "https://registry.npmjs.org/is-ci/-/is-ci-2.0.0.tgz",
792 | "integrity": "sha512-YfJT7rkpQB0updsdHLGWrvhBJfcfzNNawYDNIyQXJz0IViGf75O8EBPKSdvw2rF+LGCsX4FZ8tcr3b19LcZq4w==",
793 | "dev": true,
794 | "requires": {
795 | "ci-info": "^2.0.0"
796 | }
797 | },
798 | "is-extglob": {
799 | "version": "2.1.1",
800 | "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz",
801 | "integrity": "sha1-qIwCU1eR8C7TfHahueqXc8gz+MI=",
802 | "dev": true
803 | },
804 | "is-fullwidth-code-point": {
805 | "version": "2.0.0",
806 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz",
807 | "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=",
808 | "dev": true
809 | },
810 | "is-glob": {
811 | "version": "4.0.1",
812 | "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.1.tgz",
813 | "integrity": "sha512-5G0tKtBTFImOqDnLB2hG6Bp2qcKEFduo4tZu9MT/H6NQv/ghhy30o55ufafxJ/LdH79LLs2Kfrn85TLKyA7BUg==",
814 | "dev": true,
815 | "requires": {
816 | "is-extglob": "^2.1.1"
817 | }
818 | },
819 | "is-installed-globally": {
820 | "version": "0.3.2",
821 | "resolved": "https://registry.npmjs.org/is-installed-globally/-/is-installed-globally-0.3.2.tgz",
822 | "integrity": "sha512-wZ8x1js7Ia0kecP/CHM/3ABkAmujX7WPvQk6uu3Fly/Mk44pySulQpnHG46OMjHGXApINnV4QhY3SWnECO2z5g==",
823 | "dev": true,
824 | "requires": {
825 | "global-dirs": "^2.0.1",
826 | "is-path-inside": "^3.0.1"
827 | }
828 | },
829 | "is-npm": {
830 | "version": "4.0.0",
831 | "resolved": "https://registry.npmjs.org/is-npm/-/is-npm-4.0.0.tgz",
832 | "integrity": "sha512-96ECIfh9xtDDlPylNPXhzjsykHsMJZ18ASpaWzQyBr4YRTcVjUvzaHayDAES2oU/3KpljhHUjtSRNiDwi0F0ig==",
833 | "dev": true
834 | },
835 | "is-number": {
836 | "version": "7.0.0",
837 | "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz",
838 | "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==",
839 | "dev": true
840 | },
841 | "is-obj": {
842 | "version": "2.0.0",
843 | "resolved": "https://registry.npmjs.org/is-obj/-/is-obj-2.0.0.tgz",
844 | "integrity": "sha512-drqDG3cbczxxEJRoOXcOjtdp1J/lyp1mNn0xaznRs8+muBhgQcrnbspox5X5fOw0HnMnbfDzvnEMEtqDEJEo8w==",
845 | "dev": true
846 | },
847 | "is-path-inside": {
848 | "version": "3.0.3",
849 | "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-3.0.3.tgz",
850 | "integrity": "sha512-Fd4gABb+ycGAmKou8eMftCupSir5lRxqf4aD/vd0cD2qc4HL07OjCeuHMr8Ro4CoMaeCKDB0/ECBOVWjTwUvPQ==",
851 | "dev": true
852 | },
853 | "is-typedarray": {
854 | "version": "1.0.0",
855 | "resolved": "https://registry.npmjs.org/is-typedarray/-/is-typedarray-1.0.0.tgz",
856 | "integrity": "sha1-5HnICFjfDBsR3dppQPlgEfzaSpo=",
857 | "dev": true
858 | },
859 | "is-yarn-global": {
860 | "version": "0.3.0",
861 | "resolved": "https://registry.npmjs.org/is-yarn-global/-/is-yarn-global-0.3.0.tgz",
862 | "integrity": "sha512-VjSeb/lHmkoyd8ryPVIKvOCn4D1koMqY+vqyjjUfc3xyKtP4dYOxM44sZrnqQSzSds3xyOrUTLTC9LVCVgLngw==",
863 | "dev": true
864 | },
865 | "json-buffer": {
866 | "version": "3.0.0",
867 | "resolved": "https://registry.npmjs.org/json-buffer/-/json-buffer-3.0.0.tgz",
868 | "integrity": "sha1-Wx85evx11ne96Lz8Dkfh+aPZqJg=",
869 | "dev": true
870 | },
871 | "keyv": {
872 | "version": "3.1.0",
873 | "resolved": "https://registry.npmjs.org/keyv/-/keyv-3.1.0.tgz",
874 | "integrity": "sha512-9ykJ/46SN/9KPM/sichzQ7OvXyGDYKGTaDlKMGCAlg2UK8KRy4jb0d8sFc+0Tt0YYnThq8X2RZgCg74RPxgcVA==",
875 | "dev": true,
876 | "requires": {
877 | "json-buffer": "3.0.0"
878 | }
879 | },
880 | "latest-version": {
881 | "version": "5.1.0",
882 | "resolved": "https://registry.npmjs.org/latest-version/-/latest-version-5.1.0.tgz",
883 | "integrity": "sha512-weT+r0kTkRQdCdYCNtkMwWXQTMEswKrFBkm4ckQOMVhhqhIMI1UT2hMj+1iigIhgSZm5gTmrRXBNoGUgaTY1xA==",
884 | "dev": true,
885 | "requires": {
886 | "package-json": "^6.3.0"
887 | }
888 | },
889 | "lodash.defaults": {
890 | "version": "4.2.0",
891 | "resolved": "https://registry.npmjs.org/lodash.defaults/-/lodash.defaults-4.2.0.tgz",
892 | "integrity": "sha1-0JF4cW/+pN3p5ft7N/bwgCJ0WAw="
893 | },
894 | "lodash.flatten": {
895 | "version": "4.4.0",
896 | "resolved": "https://registry.npmjs.org/lodash.flatten/-/lodash.flatten-4.4.0.tgz",
897 | "integrity": "sha1-8xwiIlqWMtK7+OSt2+8kCqdlph8="
898 | },
899 | "lowercase-keys": {
900 | "version": "1.0.1",
901 | "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-1.0.1.tgz",
902 | "integrity": "sha512-G2Lj61tXDnVFFOi8VZds+SoQjtQC3dgokKdDG2mTm1tx4m50NUHBOZSBwQQHyy0V12A0JTG4icfZQH+xPyh8VA==",
903 | "dev": true
904 | },
905 | "make-dir": {
906 | "version": "3.1.0",
907 | "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-3.1.0.tgz",
908 | "integrity": "sha512-g3FeP20LNwhALb/6Cz6Dd4F2ngze0jz7tbzrD2wAV+o9FeNHe4rL+yK2md0J/fiSf1sa1ADhXqi5+oVwOM/eGw==",
909 | "dev": true,
910 | "requires": {
911 | "semver": "^6.0.0"
912 | },
913 | "dependencies": {
914 | "semver": {
915 | "version": "6.3.0",
916 | "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz",
917 | "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==",
918 | "dev": true
919 | }
920 | }
921 | },
922 | "media-typer": {
923 | "version": "0.3.0",
924 | "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz",
925 | "integrity": "sha1-hxDXrwqmJvj/+hzgAWhUUmMlV0g="
926 | },
927 | "merge-descriptors": {
928 | "version": "1.0.1",
929 | "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz",
930 | "integrity": "sha1-sAqqVW3YtEVoFQ7J0blT8/kMu2E="
931 | },
932 | "methods": {
933 | "version": "1.1.2",
934 | "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz",
935 | "integrity": "sha1-VSmk1nZUE07cxSZmVoNbD4Ua/O4="
936 | },
937 | "mime": {
938 | "version": "1.6.0",
939 | "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz",
940 | "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg=="
941 | },
942 | "mime-db": {
943 | "version": "1.47.0",
944 | "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.47.0.tgz",
945 | "integrity": "sha512-QBmA/G2y+IfeS4oktet3qRZ+P5kPhCKRXxXnQEudYqUaEioAU1/Lq2us3D/t1Jfo4hE9REQPrbB7K5sOczJVIw=="
946 | },
947 | "mime-types": {
948 | "version": "2.1.30",
949 | "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.30.tgz",
950 | "integrity": "sha512-crmjA4bLtR8m9qLpHvgxSChT+XoSlZi8J4n/aIdn3z92e/U47Z0V/yl+Wh9W046GgFVAmoNR/fmdbZYcSSIUeg==",
951 | "requires": {
952 | "mime-db": "1.47.0"
953 | }
954 | },
955 | "mimic-response": {
956 | "version": "1.0.1",
957 | "resolved": "https://registry.npmjs.org/mimic-response/-/mimic-response-1.0.1.tgz",
958 | "integrity": "sha512-j5EctnkH7amfV/q5Hgmoal1g2QHFJRraOtmx0JpIqkxhBhI/lJSl1nMpQ45hVarwNETOoWEimndZ4QK0RHxuxQ==",
959 | "dev": true
960 | },
961 | "minimatch": {
962 | "version": "3.0.4",
963 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz",
964 | "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==",
965 | "dev": true,
966 | "requires": {
967 | "brace-expansion": "^1.1.7"
968 | }
969 | },
970 | "minimist": {
971 | "version": "1.2.5",
972 | "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.5.tgz",
973 | "integrity": "sha512-FM9nNUYrRBAELZQT3xeZQ7fmMOBg6nWNmJKTcgsJeaLstP/UODVpGsr5OhXhhXg6f+qtJ8uiZ+PUxkDWcgIXLw==",
974 | "dev": true
975 | },
976 | "ms": {
977 | "version": "2.1.2",
978 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz",
979 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w=="
980 | },
981 | "negotiator": {
982 | "version": "0.6.2",
983 | "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.2.tgz",
984 | "integrity": "sha512-hZXc7K2e+PgeI1eDBe/10Ard4ekbfrrqG8Ep+8Jmf4JID2bNg7NvCPOZN+kfF574pFQI7mum2AUqDidoKqcTOw=="
985 | },
986 | "nodemon": {
987 | "version": "2.0.7",
988 | "resolved": "https://registry.npmjs.org/nodemon/-/nodemon-2.0.7.tgz",
989 | "integrity": "sha512-XHzK69Awgnec9UzHr1kc8EomQh4sjTQ8oRf8TsGrSmHDx9/UmiGG9E/mM3BuTfNeFwdNBvrqQq/RHL0xIeyFOA==",
990 | "dev": true,
991 | "requires": {
992 | "chokidar": "^3.2.2",
993 | "debug": "^3.2.6",
994 | "ignore-by-default": "^1.0.1",
995 | "minimatch": "^3.0.4",
996 | "pstree.remy": "^1.1.7",
997 | "semver": "^5.7.1",
998 | "supports-color": "^5.5.0",
999 | "touch": "^3.1.0",
1000 | "undefsafe": "^2.0.3",
1001 | "update-notifier": "^4.1.0"
1002 | },
1003 | "dependencies": {
1004 | "debug": {
1005 | "version": "3.2.7",
1006 | "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz",
1007 | "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==",
1008 | "dev": true,
1009 | "requires": {
1010 | "ms": "^2.1.1"
1011 | }
1012 | }
1013 | }
1014 | },
1015 | "nopt": {
1016 | "version": "1.0.10",
1017 | "resolved": "https://registry.npmjs.org/nopt/-/nopt-1.0.10.tgz",
1018 | "integrity": "sha1-bd0hvSoxQXuScn3Vhfim83YI6+4=",
1019 | "dev": true,
1020 | "requires": {
1021 | "abbrev": "1"
1022 | }
1023 | },
1024 | "normalize-path": {
1025 | "version": "3.0.0",
1026 | "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz",
1027 | "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==",
1028 | "dev": true
1029 | },
1030 | "normalize-url": {
1031 | "version": "4.5.0",
1032 | "resolved": "https://registry.npmjs.org/normalize-url/-/normalize-url-4.5.0.tgz",
1033 | "integrity": "sha512-2s47yzUxdexf1OhyRi4Em83iQk0aPvwTddtFz4hnSSw9dCEsLEGf6SwIO8ss/19S9iBb5sJaOuTvTGDeZI00BQ==",
1034 | "dev": true
1035 | },
1036 | "object-assign": {
1037 | "version": "4.1.1",
1038 | "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz",
1039 | "integrity": "sha1-IQmtx5ZYh8/AXLvUQsrIv7s2CGM="
1040 | },
1041 | "on-finished": {
1042 | "version": "2.3.0",
1043 | "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.3.0.tgz",
1044 | "integrity": "sha1-IPEzZIGwg811M3mSoWlxqi2QaUc=",
1045 | "requires": {
1046 | "ee-first": "1.1.1"
1047 | }
1048 | },
1049 | "once": {
1050 | "version": "1.4.0",
1051 | "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz",
1052 | "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=",
1053 | "dev": true,
1054 | "requires": {
1055 | "wrappy": "1"
1056 | }
1057 | },
1058 | "p-cancelable": {
1059 | "version": "1.1.0",
1060 | "resolved": "https://registry.npmjs.org/p-cancelable/-/p-cancelable-1.1.0.tgz",
1061 | "integrity": "sha512-s73XxOZ4zpt1edZYZzvhqFa6uvQc1vwUa0K0BdtIZgQMAJj9IbebH+JkgKZc9h+B05PKHLOTl4ajG1BmNrVZlw==",
1062 | "dev": true
1063 | },
1064 | "p-map": {
1065 | "version": "2.1.0",
1066 | "resolved": "https://registry.npmjs.org/p-map/-/p-map-2.1.0.tgz",
1067 | "integrity": "sha512-y3b8Kpd8OAN444hxfBbFfj1FY/RjtTd8tzYwhUqNYXx0fXx2iX4maP4Qr6qhIKbQXI02wTLAda4fYUbDagTUFw=="
1068 | },
1069 | "package-json": {
1070 | "version": "6.5.0",
1071 | "resolved": "https://registry.npmjs.org/package-json/-/package-json-6.5.0.tgz",
1072 | "integrity": "sha512-k3bdm2n25tkyxcjSKzB5x8kfVxlMdgsbPr0GkZcwHsLpba6cBjqCt1KlcChKEvxHIcTB1FVMuwoijZ26xex5MQ==",
1073 | "dev": true,
1074 | "requires": {
1075 | "got": "^9.6.0",
1076 | "registry-auth-token": "^4.0.0",
1077 | "registry-url": "^5.0.0",
1078 | "semver": "^6.2.0"
1079 | },
1080 | "dependencies": {
1081 | "semver": {
1082 | "version": "6.3.0",
1083 | "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz",
1084 | "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==",
1085 | "dev": true
1086 | }
1087 | }
1088 | },
1089 | "parseurl": {
1090 | "version": "1.3.3",
1091 | "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz",
1092 | "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ=="
1093 | },
1094 | "path-to-regexp": {
1095 | "version": "0.1.7",
1096 | "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz",
1097 | "integrity": "sha1-32BBeABfUi8V60SQ5yR6G/qmf4w="
1098 | },
1099 | "picomatch": {
1100 | "version": "2.2.3",
1101 | "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.2.3.tgz",
1102 | "integrity": "sha512-KpELjfwcCDUb9PeigTs2mBJzXUPzAuP2oPcA989He8Rte0+YUAjw1JVedDhuTKPkHjSYzMN3npC9luThGYEKdg==",
1103 | "dev": true
1104 | },
1105 | "prepend-http": {
1106 | "version": "2.0.0",
1107 | "resolved": "https://registry.npmjs.org/prepend-http/-/prepend-http-2.0.0.tgz",
1108 | "integrity": "sha1-6SQ0v6XqjBn0HN/UAddBo8gZ2Jc=",
1109 | "dev": true
1110 | },
1111 | "proxy-addr": {
1112 | "version": "2.0.6",
1113 | "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.6.tgz",
1114 | "integrity": "sha512-dh/frvCBVmSsDYzw6n926jv974gddhkFPfiN8hPOi30Wax25QZyZEGveluCgliBnqmuM+UJmBErbAUFIoDbjOw==",
1115 | "requires": {
1116 | "forwarded": "~0.1.2",
1117 | "ipaddr.js": "1.9.1"
1118 | }
1119 | },
1120 | "pstree.remy": {
1121 | "version": "1.1.8",
1122 | "resolved": "https://registry.npmjs.org/pstree.remy/-/pstree.remy-1.1.8.tgz",
1123 | "integrity": "sha512-77DZwxQmxKnu3aR542U+X8FypNzbfJ+C5XQDk3uWjWxn6151aIMGthWYRXTqT1E5oJvg+ljaa2OJi+VfvCOQ8w==",
1124 | "dev": true
1125 | },
1126 | "pump": {
1127 | "version": "3.0.0",
1128 | "resolved": "https://registry.npmjs.org/pump/-/pump-3.0.0.tgz",
1129 | "integrity": "sha512-LwZy+p3SFs1Pytd/jYct4wpv49HiYCqd9Rlc5ZVdk0V+8Yzv6jR5Blk3TRmPL1ft69TxP0IMZGJ+WPFU2BFhww==",
1130 | "dev": true,
1131 | "requires": {
1132 | "end-of-stream": "^1.1.0",
1133 | "once": "^1.3.1"
1134 | }
1135 | },
1136 | "pupa": {
1137 | "version": "2.1.1",
1138 | "resolved": "https://registry.npmjs.org/pupa/-/pupa-2.1.1.tgz",
1139 | "integrity": "sha512-l1jNAspIBSFqbT+y+5FosojNpVpF94nlI+wDUpqP9enwOTfHx9f0gh5nB96vl+6yTpsJsypeNrwfzPrKuHB41A==",
1140 | "dev": true,
1141 | "requires": {
1142 | "escape-goat": "^2.0.0"
1143 | }
1144 | },
1145 | "qs": {
1146 | "version": "6.7.0",
1147 | "resolved": "https://registry.npmjs.org/qs/-/qs-6.7.0.tgz",
1148 | "integrity": "sha512-VCdBRNFTX1fyE7Nb6FYoURo/SPe62QCaAyzJvUjwRaIsc+NePBEniHlvxFmmX56+HZphIGtV0XeCirBtpDrTyQ=="
1149 | },
1150 | "range-parser": {
1151 | "version": "1.2.1",
1152 | "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz",
1153 | "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg=="
1154 | },
1155 | "raw-body": {
1156 | "version": "2.4.0",
1157 | "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.4.0.tgz",
1158 | "integrity": "sha512-4Oz8DUIwdvoa5qMJelxipzi/iJIi40O5cGV1wNYp5hvZP8ZN0T+jiNkL0QepXs+EsQ9XJ8ipEDoiH70ySUJP3Q==",
1159 | "requires": {
1160 | "bytes": "3.1.0",
1161 | "http-errors": "1.7.2",
1162 | "iconv-lite": "0.4.24",
1163 | "unpipe": "1.0.0"
1164 | }
1165 | },
1166 | "rc": {
1167 | "version": "1.2.8",
1168 | "resolved": "https://registry.npmjs.org/rc/-/rc-1.2.8.tgz",
1169 | "integrity": "sha512-y3bGgqKj3QBdxLbLkomlohkvsA8gdAiUQlSBJnBhfn+BPxg4bc62d8TcBW15wavDfgexCgccckhcZvywyQYPOw==",
1170 | "dev": true,
1171 | "requires": {
1172 | "deep-extend": "^0.6.0",
1173 | "ini": "~1.3.0",
1174 | "minimist": "^1.2.0",
1175 | "strip-json-comments": "~2.0.1"
1176 | }
1177 | },
1178 | "readdirp": {
1179 | "version": "3.5.0",
1180 | "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.5.0.tgz",
1181 | "integrity": "sha512-cMhu7c/8rdhkHXWsY+osBhfSy0JikwpHK/5+imo+LpeasTF8ouErHrlYkwT0++njiyuDvc7OFY5T3ukvZ8qmFQ==",
1182 | "dev": true,
1183 | "requires": {
1184 | "picomatch": "^2.2.1"
1185 | }
1186 | },
1187 | "redis-commands": {
1188 | "version": "1.7.0",
1189 | "resolved": "https://registry.npmjs.org/redis-commands/-/redis-commands-1.7.0.tgz",
1190 | "integrity": "sha512-nJWqw3bTFy21hX/CPKHth6sfhZbdiHP6bTawSgQBlKOVRG7EZkfHbbHwQJnrE4vsQf0CMNE+3gJ4Fmm16vdVlQ=="
1191 | },
1192 | "redis-errors": {
1193 | "version": "1.2.0",
1194 | "resolved": "https://registry.npmjs.org/redis-errors/-/redis-errors-1.2.0.tgz",
1195 | "integrity": "sha1-62LSrbFeTq9GEMBK/hUpOEJQq60="
1196 | },
1197 | "redis-parser": {
1198 | "version": "3.0.0",
1199 | "resolved": "https://registry.npmjs.org/redis-parser/-/redis-parser-3.0.0.tgz",
1200 | "integrity": "sha1-tm2CjNyv5rS4pCin3vTGvKwxyLQ=",
1201 | "requires": {
1202 | "redis-errors": "^1.0.0"
1203 | }
1204 | },
1205 | "registry-auth-token": {
1206 | "version": "4.2.1",
1207 | "resolved": "https://registry.npmjs.org/registry-auth-token/-/registry-auth-token-4.2.1.tgz",
1208 | "integrity": "sha512-6gkSb4U6aWJB4SF2ZvLb76yCBjcvufXBqvvEx1HbmKPkutswjW1xNVRY0+daljIYRbogN7O0etYSlbiaEQyMyw==",
1209 | "dev": true,
1210 | "requires": {
1211 | "rc": "^1.2.8"
1212 | }
1213 | },
1214 | "registry-url": {
1215 | "version": "5.1.0",
1216 | "resolved": "https://registry.npmjs.org/registry-url/-/registry-url-5.1.0.tgz",
1217 | "integrity": "sha512-8acYXXTI0AkQv6RAOjE3vOaIXZkT9wo4LOFbBKYQEEnnMNBpKqdUrI6S4NT0KPIo/WVvJ5tE/X5LF/TQUf0ekw==",
1218 | "dev": true,
1219 | "requires": {
1220 | "rc": "^1.2.8"
1221 | }
1222 | },
1223 | "responselike": {
1224 | "version": "1.0.2",
1225 | "resolved": "https://registry.npmjs.org/responselike/-/responselike-1.0.2.tgz",
1226 | "integrity": "sha1-kYcg7ztjHFZCvgaPFa3lpG9Loec=",
1227 | "dev": true,
1228 | "requires": {
1229 | "lowercase-keys": "^1.0.0"
1230 | }
1231 | },
1232 | "safe-buffer": {
1233 | "version": "5.1.2",
1234 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz",
1235 | "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g=="
1236 | },
1237 | "safer-buffer": {
1238 | "version": "2.1.2",
1239 | "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz",
1240 | "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg=="
1241 | },
1242 | "semver": {
1243 | "version": "5.7.1",
1244 | "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz",
1245 | "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==",
1246 | "dev": true
1247 | },
1248 | "semver-diff": {
1249 | "version": "3.1.1",
1250 | "resolved": "https://registry.npmjs.org/semver-diff/-/semver-diff-3.1.1.tgz",
1251 | "integrity": "sha512-GX0Ix/CJcHyB8c4ykpHGIAvLyOwOobtM/8d+TQkAd81/bEjgPHrfba41Vpesr7jX/t8Uh+R3EX9eAS5be+jQYg==",
1252 | "dev": true,
1253 | "requires": {
1254 | "semver": "^6.3.0"
1255 | },
1256 | "dependencies": {
1257 | "semver": {
1258 | "version": "6.3.0",
1259 | "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz",
1260 | "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==",
1261 | "dev": true
1262 | }
1263 | }
1264 | },
1265 | "send": {
1266 | "version": "0.17.1",
1267 | "resolved": "https://registry.npmjs.org/send/-/send-0.17.1.tgz",
1268 | "integrity": "sha512-BsVKsiGcQMFwT8UxypobUKyv7irCNRHk1T0G680vk88yf6LBByGcZJOTJCrTP2xVN6yI+XjPJcNuE3V4fT9sAg==",
1269 | "requires": {
1270 | "debug": "2.6.9",
1271 | "depd": "~1.1.2",
1272 | "destroy": "~1.0.4",
1273 | "encodeurl": "~1.0.2",
1274 | "escape-html": "~1.0.3",
1275 | "etag": "~1.8.1",
1276 | "fresh": "0.5.2",
1277 | "http-errors": "~1.7.2",
1278 | "mime": "1.6.0",
1279 | "ms": "2.1.1",
1280 | "on-finished": "~2.3.0",
1281 | "range-parser": "~1.2.1",
1282 | "statuses": "~1.5.0"
1283 | },
1284 | "dependencies": {
1285 | "debug": {
1286 | "version": "2.6.9",
1287 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz",
1288 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==",
1289 | "requires": {
1290 | "ms": "2.0.0"
1291 | },
1292 | "dependencies": {
1293 | "ms": {
1294 | "version": "2.0.0",
1295 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz",
1296 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g="
1297 | }
1298 | }
1299 | },
1300 | "ms": {
1301 | "version": "2.1.1",
1302 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.1.tgz",
1303 | "integrity": "sha512-tgp+dl5cGk28utYktBsrFqA7HKgrhgPsg6Z/EfhWI4gl1Hwq8B/GmY/0oXZ6nF8hDVesS/FpnYaD/kOWhYQvyg=="
1304 | }
1305 | }
1306 | },
1307 | "serve-static": {
1308 | "version": "1.14.1",
1309 | "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.14.1.tgz",
1310 | "integrity": "sha512-JMrvUwE54emCYWlTI+hGrGv5I8dEwmco/00EvkzIIsR7MqrHonbD9pO2MOfFnpFntl7ecpZs+3mW+XbQZu9QCg==",
1311 | "requires": {
1312 | "encodeurl": "~1.0.2",
1313 | "escape-html": "~1.0.3",
1314 | "parseurl": "~1.3.3",
1315 | "send": "0.17.1"
1316 | }
1317 | },
1318 | "setprototypeof": {
1319 | "version": "1.1.1",
1320 | "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.1.tgz",
1321 | "integrity": "sha512-JvdAWfbXeIGaZ9cILp38HntZSFSo3mWg6xGcJJsd+d4aRMOqauag1C63dJfDw7OaMYwEbHMOxEZ1lqVRYP2OAw=="
1322 | },
1323 | "signal-exit": {
1324 | "version": "3.0.3",
1325 | "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.3.tgz",
1326 | "integrity": "sha512-VUJ49FC8U1OxwZLxIbTTrDvLnf/6TDgxZcK8wxR8zs13xpx7xbG60ndBlhNrFi2EMuFRoeDoJO7wthSLq42EjA==",
1327 | "dev": true
1328 | },
1329 | "socket.io": {
1330 | "version": "4.0.2",
1331 | "resolved": "https://registry.npmjs.org/socket.io/-/socket.io-4.0.2.tgz",
1332 | "integrity": "sha512-1sVnsumrYTshhH2LWUtmzNvuZqkEOAG5PxcHpKu1r/sIDhzACTdo9XbvonVB2M90Y2CkTZnCWT6hlaKEkdBu5Q==",
1333 | "requires": {
1334 | "@types/cookie": "^0.4.0",
1335 | "@types/cors": "^2.8.8",
1336 | "@types/node": ">=10.0.0",
1337 | "accepts": "~1.3.4",
1338 | "base64id": "~2.0.0",
1339 | "debug": "~4.3.1",
1340 | "engine.io": "~5.0.0",
1341 | "socket.io-adapter": "~2.2.0",
1342 | "socket.io-parser": "~4.0.3"
1343 | }
1344 | },
1345 | "socket.io-adapter": {
1346 | "version": "2.2.0",
1347 | "resolved": "https://registry.npmjs.org/socket.io-adapter/-/socket.io-adapter-2.2.0.tgz",
1348 | "integrity": "sha512-rG49L+FwaVEwuAdeBRq49M97YI3ElVabJPzvHT9S6a2CWhDKnjSFasvwAwSYPRhQzfn4NtDIbCaGYgOCOU/rlg=="
1349 | },
1350 | "socket.io-parser": {
1351 | "version": "4.0.4",
1352 | "resolved": "https://registry.npmjs.org/socket.io-parser/-/socket.io-parser-4.0.4.tgz",
1353 | "integrity": "sha512-t+b0SS+IxG7Rxzda2EVvyBZbvFPBCjJoyHuE0P//7OAsN23GItzDRdWa6ALxZI/8R5ygK7jAR6t028/z+7295g==",
1354 | "requires": {
1355 | "@types/component-emitter": "^1.2.10",
1356 | "component-emitter": "~1.3.0",
1357 | "debug": "~4.3.1"
1358 | }
1359 | },
1360 | "standard-as-callback": {
1361 | "version": "2.1.0",
1362 | "resolved": "https://registry.npmjs.org/standard-as-callback/-/standard-as-callback-2.1.0.tgz",
1363 | "integrity": "sha512-qoRRSyROncaz1z0mvYqIE4lCd9p2R90i6GxW3uZv5ucSu8tU7B5HXUP1gG8pVZsYNVaXjk8ClXHPttLyxAL48A=="
1364 | },
1365 | "statuses": {
1366 | "version": "1.5.0",
1367 | "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.5.0.tgz",
1368 | "integrity": "sha1-Fhx9rBd2Wf2YEfQ3cfqZOBR4Yow="
1369 | },
1370 | "string-width": {
1371 | "version": "4.2.2",
1372 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.2.tgz",
1373 | "integrity": "sha512-XBJbT3N4JhVumXE0eoLU9DCjcaF92KLNqTmFCnG1pf8duUxFGwtP6AD6nkjw9a3IdiRtL3E2w3JDiE/xi3vOeA==",
1374 | "dev": true,
1375 | "requires": {
1376 | "emoji-regex": "^8.0.0",
1377 | "is-fullwidth-code-point": "^3.0.0",
1378 | "strip-ansi": "^6.0.0"
1379 | },
1380 | "dependencies": {
1381 | "ansi-regex": {
1382 | "version": "5.0.0",
1383 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.0.tgz",
1384 | "integrity": "sha512-bY6fj56OUQ0hU1KjFNDQuJFezqKdrAyFdIevADiqrWHwSlbmBNMHp5ak2f40Pm8JTFyM2mqxkG6ngkHO11f/lg==",
1385 | "dev": true
1386 | },
1387 | "emoji-regex": {
1388 | "version": "8.0.0",
1389 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz",
1390 | "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==",
1391 | "dev": true
1392 | },
1393 | "is-fullwidth-code-point": {
1394 | "version": "3.0.0",
1395 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz",
1396 | "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==",
1397 | "dev": true
1398 | },
1399 | "strip-ansi": {
1400 | "version": "6.0.0",
1401 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.0.tgz",
1402 | "integrity": "sha512-AuvKTrTfQNYNIctbR1K/YGTR1756GycPsg7b9bdV9Duqur4gv6aKqHXah67Z8ImS7WEz5QVcOtlfW2rZEugt6w==",
1403 | "dev": true,
1404 | "requires": {
1405 | "ansi-regex": "^5.0.0"
1406 | }
1407 | }
1408 | }
1409 | },
1410 | "strip-ansi": {
1411 | "version": "5.2.0",
1412 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz",
1413 | "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==",
1414 | "dev": true,
1415 | "requires": {
1416 | "ansi-regex": "^4.1.0"
1417 | }
1418 | },
1419 | "strip-json-comments": {
1420 | "version": "2.0.1",
1421 | "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-2.0.1.tgz",
1422 | "integrity": "sha1-PFMZQukIwml8DsNEhYwobHygpgo=",
1423 | "dev": true
1424 | },
1425 | "supports-color": {
1426 | "version": "5.5.0",
1427 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz",
1428 | "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==",
1429 | "dev": true,
1430 | "requires": {
1431 | "has-flag": "^3.0.0"
1432 | }
1433 | },
1434 | "term-size": {
1435 | "version": "2.2.1",
1436 | "resolved": "https://registry.npmjs.org/term-size/-/term-size-2.2.1.tgz",
1437 | "integrity": "sha512-wK0Ri4fOGjv/XPy8SBHZChl8CM7uMc5VML7SqiQ0zG7+J5Vr+RMQDoHa2CNT6KHUnTGIXH34UDMkPzAUyapBZg==",
1438 | "dev": true
1439 | },
1440 | "to-readable-stream": {
1441 | "version": "1.0.0",
1442 | "resolved": "https://registry.npmjs.org/to-readable-stream/-/to-readable-stream-1.0.0.tgz",
1443 | "integrity": "sha512-Iq25XBt6zD5npPhlLVXGFN3/gyR2/qODcKNNyTMd4vbm39HUaOiAM4PMq0eMVC/Tkxz+Zjdsc55g9yyz+Yq00Q==",
1444 | "dev": true
1445 | },
1446 | "to-regex-range": {
1447 | "version": "5.0.1",
1448 | "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz",
1449 | "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==",
1450 | "dev": true,
1451 | "requires": {
1452 | "is-number": "^7.0.0"
1453 | }
1454 | },
1455 | "toidentifier": {
1456 | "version": "1.0.0",
1457 | "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.0.tgz",
1458 | "integrity": "sha512-yaOH/Pk/VEhBWWTlhI+qXxDFXlejDGcQipMlyxda9nthulaxLZUNcUqFxokp0vcYnvteJln5FNQDRrxj3YcbVw=="
1459 | },
1460 | "touch": {
1461 | "version": "3.1.0",
1462 | "resolved": "https://registry.npmjs.org/touch/-/touch-3.1.0.tgz",
1463 | "integrity": "sha512-WBx8Uy5TLtOSRtIq+M03/sKDrXCLHxwDcquSP2c43Le03/9serjQBIztjRz6FkJez9D/hleyAXTBGLwwZUw9lA==",
1464 | "dev": true,
1465 | "requires": {
1466 | "nopt": "~1.0.10"
1467 | }
1468 | },
1469 | "type-fest": {
1470 | "version": "0.8.1",
1471 | "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.8.1.tgz",
1472 | "integrity": "sha512-4dbzIzqvjtgiM5rw1k5rEHtBANKmdudhGyBEajN01fEyhaAIhsoKNy6y7+IN93IfpFtwY9iqi7kD+xwKhQsNJA==",
1473 | "dev": true
1474 | },
1475 | "type-is": {
1476 | "version": "1.6.18",
1477 | "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz",
1478 | "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==",
1479 | "requires": {
1480 | "media-typer": "0.3.0",
1481 | "mime-types": "~2.1.24"
1482 | }
1483 | },
1484 | "typedarray-to-buffer": {
1485 | "version": "3.1.5",
1486 | "resolved": "https://registry.npmjs.org/typedarray-to-buffer/-/typedarray-to-buffer-3.1.5.tgz",
1487 | "integrity": "sha512-zdu8XMNEDepKKR+XYOXAVPtWui0ly0NtohUscw+UmaHiAWT8hrV1rr//H6V+0DvJ3OQ19S979M0laLfX8rm82Q==",
1488 | "dev": true,
1489 | "requires": {
1490 | "is-typedarray": "^1.0.0"
1491 | }
1492 | },
1493 | "undefsafe": {
1494 | "version": "2.0.3",
1495 | "resolved": "https://registry.npmjs.org/undefsafe/-/undefsafe-2.0.3.tgz",
1496 | "integrity": "sha512-nrXZwwXrD/T/JXeygJqdCO6NZZ1L66HrxM/Z7mIq2oPanoN0F1nLx3lwJMu6AwJY69hdixaFQOuoYsMjE5/C2A==",
1497 | "dev": true,
1498 | "requires": {
1499 | "debug": "^2.2.0"
1500 | },
1501 | "dependencies": {
1502 | "debug": {
1503 | "version": "2.6.9",
1504 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz",
1505 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==",
1506 | "dev": true,
1507 | "requires": {
1508 | "ms": "2.0.0"
1509 | }
1510 | },
1511 | "ms": {
1512 | "version": "2.0.0",
1513 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz",
1514 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=",
1515 | "dev": true
1516 | }
1517 | }
1518 | },
1519 | "unique-string": {
1520 | "version": "2.0.0",
1521 | "resolved": "https://registry.npmjs.org/unique-string/-/unique-string-2.0.0.tgz",
1522 | "integrity": "sha512-uNaeirEPvpZWSgzwsPGtU2zVSTrn/8L5q/IexZmH0eH6SA73CmAA5U4GwORTxQAZs95TAXLNqeLoPPNO5gZfWg==",
1523 | "dev": true,
1524 | "requires": {
1525 | "crypto-random-string": "^2.0.0"
1526 | }
1527 | },
1528 | "unpipe": {
1529 | "version": "1.0.0",
1530 | "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz",
1531 | "integrity": "sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw="
1532 | },
1533 | "update-notifier": {
1534 | "version": "4.1.3",
1535 | "resolved": "https://registry.npmjs.org/update-notifier/-/update-notifier-4.1.3.tgz",
1536 | "integrity": "sha512-Yld6Z0RyCYGB6ckIjffGOSOmHXj1gMeE7aROz4MG+XMkmixBX4jUngrGXNYz7wPKBmtoD4MnBa2Anu7RSKht/A==",
1537 | "dev": true,
1538 | "requires": {
1539 | "boxen": "^4.2.0",
1540 | "chalk": "^3.0.0",
1541 | "configstore": "^5.0.1",
1542 | "has-yarn": "^2.1.0",
1543 | "import-lazy": "^2.1.0",
1544 | "is-ci": "^2.0.0",
1545 | "is-installed-globally": "^0.3.1",
1546 | "is-npm": "^4.0.0",
1547 | "is-yarn-global": "^0.3.0",
1548 | "latest-version": "^5.0.0",
1549 | "pupa": "^2.0.1",
1550 | "semver-diff": "^3.1.1",
1551 | "xdg-basedir": "^4.0.0"
1552 | }
1553 | },
1554 | "url-parse-lax": {
1555 | "version": "3.0.0",
1556 | "resolved": "https://registry.npmjs.org/url-parse-lax/-/url-parse-lax-3.0.0.tgz",
1557 | "integrity": "sha1-FrXK/Afb42dsGxmZF3gj1lA6yww=",
1558 | "dev": true,
1559 | "requires": {
1560 | "prepend-http": "^2.0.0"
1561 | }
1562 | },
1563 | "utils-merge": {
1564 | "version": "1.0.1",
1565 | "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz",
1566 | "integrity": "sha1-n5VxD1CiZ5R7LMwSR0HBAoQn5xM="
1567 | },
1568 | "vary": {
1569 | "version": "1.1.2",
1570 | "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz",
1571 | "integrity": "sha1-IpnwLG3tMNSllhsLn3RSShj2NPw="
1572 | },
1573 | "widest-line": {
1574 | "version": "3.1.0",
1575 | "resolved": "https://registry.npmjs.org/widest-line/-/widest-line-3.1.0.tgz",
1576 | "integrity": "sha512-NsmoXalsWVDMGupxZ5R08ka9flZjjiLvHVAWYOKtiKM8ujtZWr9cRffak+uSE48+Ob8ObalXpwyeUiyDD6QFgg==",
1577 | "dev": true,
1578 | "requires": {
1579 | "string-width": "^4.0.0"
1580 | }
1581 | },
1582 | "wrappy": {
1583 | "version": "1.0.2",
1584 | "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz",
1585 | "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=",
1586 | "dev": true
1587 | },
1588 | "write-file-atomic": {
1589 | "version": "3.0.3",
1590 | "resolved": "https://registry.npmjs.org/write-file-atomic/-/write-file-atomic-3.0.3.tgz",
1591 | "integrity": "sha512-AvHcyZ5JnSfq3ioSyjrBkH9yW4m7Ayk8/9My/DD9onKeu/94fwrMocemO2QAJFAlnnDN+ZDS+ZjAR5ua1/PV/Q==",
1592 | "dev": true,
1593 | "requires": {
1594 | "imurmurhash": "^0.1.4",
1595 | "is-typedarray": "^1.0.0",
1596 | "signal-exit": "^3.0.2",
1597 | "typedarray-to-buffer": "^3.1.5"
1598 | }
1599 | },
1600 | "ws": {
1601 | "version": "7.4.5",
1602 | "resolved": "https://registry.npmjs.org/ws/-/ws-7.4.5.tgz",
1603 | "integrity": "sha512-xzyu3hFvomRfXKH8vOFMU3OguG6oOvhXMo3xsGy3xWExqaM2dxBbVxuD99O7m3ZUFMvvscsZDqxfgMaRr/Nr1g=="
1604 | },
1605 | "xdg-basedir": {
1606 | "version": "4.0.0",
1607 | "resolved": "https://registry.npmjs.org/xdg-basedir/-/xdg-basedir-4.0.0.tgz",
1608 | "integrity": "sha512-PSNhEJDejZYV7h50BohL09Er9VaIefr2LMAf3OEmpCkjOi34eYyQYAXUTjEQtZJTKcF0E2UKTh+osDLsgNim9Q==",
1609 | "dev": true
1610 | }
1611 | }
1612 | }
1613 |
--------------------------------------------------------------------------------
/bidding-service/package-lock.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "order-service",
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/component-emitter": {
23 | "version": "1.2.10",
24 | "resolved": "https://registry.npmjs.org/@types/component-emitter/-/component-emitter-1.2.10.tgz",
25 | "integrity": "sha512-bsjleuRKWmGqajMerkzox19aGbscQX5rmmvvXl3wlIp5gMG1HgkiwPxsN5p070fBDKTNSPgojVbuY1+HWMbFhg=="
26 | },
27 | "@types/cookie": {
28 | "version": "0.4.0",
29 | "resolved": "https://registry.npmjs.org/@types/cookie/-/cookie-0.4.0.tgz",
30 | "integrity": "sha512-y7mImlc/rNkvCRmg8gC3/lj87S7pTUIJ6QGjwHR9WQJcFs+ZMTOaoPrkdFA/YdbuqVEmEbb5RdhVxMkAcgOnpg=="
31 | },
32 | "@types/cors": {
33 | "version": "2.8.10",
34 | "resolved": "https://registry.npmjs.org/@types/cors/-/cors-2.8.10.tgz",
35 | "integrity": "sha512-C7srjHiVG3Ey1nR6d511dtDkCEjxuN9W1HWAEjGq8kpcwmNM6JJkpC0xvabM7BXTG2wDq8Eu33iH9aQKa7IvLQ=="
36 | },
37 | "@types/node": {
38 | "version": "15.3.0",
39 | "resolved": "https://registry.npmjs.org/@types/node/-/node-15.3.0.tgz",
40 | "integrity": "sha512-8/bnjSZD86ZfpBsDlCIkNXIvm+h6wi9g7IqL+kmFkQ+Wvu3JrasgLElfiPgoo8V8vVfnEi0QVS12gbl94h9YsQ=="
41 | },
42 | "abbrev": {
43 | "version": "1.1.1",
44 | "resolved": "https://registry.npmjs.org/abbrev/-/abbrev-1.1.1.tgz",
45 | "integrity": "sha512-nne9/IiQ/hzIhY6pdDnbBtz7DjPTKrY00P/zvPSm5pOFkl6xuGrGnXn/VtTNNfNtAfZ9/1RtehkszU9qcTii0Q==",
46 | "dev": true
47 | },
48 | "accepts": {
49 | "version": "1.3.7",
50 | "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.7.tgz",
51 | "integrity": "sha512-Il80Qs2WjYlJIBNzNkK6KYqlVMTbZLXgHx2oT0pU/fjRHyEp+PEfEPY0R3WCwAGVOtauxh1hOxNgIf5bv7dQpA==",
52 | "requires": {
53 | "mime-types": "~2.1.24",
54 | "negotiator": "0.6.2"
55 | }
56 | },
57 | "ansi-align": {
58 | "version": "3.0.0",
59 | "resolved": "https://registry.npmjs.org/ansi-align/-/ansi-align-3.0.0.tgz",
60 | "integrity": "sha512-ZpClVKqXN3RGBmKibdfWzqCY4lnjEuoNzU5T0oEFpfd/z5qJHVarukridD4juLO2FXMiwUQxr9WqQtaYa8XRYw==",
61 | "dev": true,
62 | "requires": {
63 | "string-width": "^3.0.0"
64 | },
65 | "dependencies": {
66 | "string-width": {
67 | "version": "3.1.0",
68 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz",
69 | "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==",
70 | "dev": true,
71 | "requires": {
72 | "emoji-regex": "^7.0.1",
73 | "is-fullwidth-code-point": "^2.0.0",
74 | "strip-ansi": "^5.1.0"
75 | }
76 | }
77 | }
78 | },
79 | "ansi-regex": {
80 | "version": "4.1.0",
81 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz",
82 | "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==",
83 | "dev": true
84 | },
85 | "ansi-styles": {
86 | "version": "4.3.0",
87 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz",
88 | "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==",
89 | "dev": true,
90 | "requires": {
91 | "color-convert": "^2.0.1"
92 | }
93 | },
94 | "anymatch": {
95 | "version": "3.1.2",
96 | "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.2.tgz",
97 | "integrity": "sha512-P43ePfOAIupkguHUycrc4qJ9kz8ZiuOUijaETwX7THt0Y/GNK7v0aa8rY816xWjZ7rJdA5XdMcpVFTKMq+RvWg==",
98 | "dev": true,
99 | "requires": {
100 | "normalize-path": "^3.0.0",
101 | "picomatch": "^2.0.4"
102 | }
103 | },
104 | "array-flatten": {
105 | "version": "1.1.1",
106 | "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz",
107 | "integrity": "sha1-ml9pkFGx5wczKPKgCJaLZOopVdI="
108 | },
109 | "balanced-match": {
110 | "version": "1.0.2",
111 | "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz",
112 | "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==",
113 | "dev": true
114 | },
115 | "base64-arraybuffer": {
116 | "version": "0.1.4",
117 | "resolved": "https://registry.npmjs.org/base64-arraybuffer/-/base64-arraybuffer-0.1.4.tgz",
118 | "integrity": "sha1-mBjHngWbE1X5fgQooBfIOOkLqBI="
119 | },
120 | "base64id": {
121 | "version": "2.0.0",
122 | "resolved": "https://registry.npmjs.org/base64id/-/base64id-2.0.0.tgz",
123 | "integrity": "sha512-lGe34o6EHj9y3Kts9R4ZYs/Gr+6N7MCaMlIFA3F1R2O5/m7K06AxfSeO5530PEERE6/WyEg3lsuyw4GHlPZHog=="
124 | },
125 | "binary-extensions": {
126 | "version": "2.2.0",
127 | "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.2.0.tgz",
128 | "integrity": "sha512-jDctJ/IVQbZoJykoeHbhXpOlNBqGNcwXJKJog42E5HDPUwQTSdjCHdihjj0DlnheQ7blbT6dHOafNAiS8ooQKA==",
129 | "dev": true
130 | },
131 | "body-parser": {
132 | "version": "1.19.0",
133 | "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.19.0.tgz",
134 | "integrity": "sha512-dhEPs72UPbDnAQJ9ZKMNTP6ptJaionhP5cBb541nXPlW60Jepo9RV/a4fX4XWW9CuFNK22krhrj1+rgzifNCsw==",
135 | "requires": {
136 | "bytes": "3.1.0",
137 | "content-type": "~1.0.4",
138 | "debug": "2.6.9",
139 | "depd": "~1.1.2",
140 | "http-errors": "1.7.2",
141 | "iconv-lite": "0.4.24",
142 | "on-finished": "~2.3.0",
143 | "qs": "6.7.0",
144 | "raw-body": "2.4.0",
145 | "type-is": "~1.6.17"
146 | },
147 | "dependencies": {
148 | "debug": {
149 | "version": "2.6.9",
150 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz",
151 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==",
152 | "requires": {
153 | "ms": "2.0.0"
154 | }
155 | },
156 | "ms": {
157 | "version": "2.0.0",
158 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz",
159 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g="
160 | }
161 | }
162 | },
163 | "boxen": {
164 | "version": "4.2.0",
165 | "resolved": "https://registry.npmjs.org/boxen/-/boxen-4.2.0.tgz",
166 | "integrity": "sha512-eB4uT9RGzg2odpER62bBwSLvUeGC+WbRjjyyFhGsKnc8wp/m0+hQsMUvUe3H2V0D5vw0nBdO1hCJoZo5mKeuIQ==",
167 | "dev": true,
168 | "requires": {
169 | "ansi-align": "^3.0.0",
170 | "camelcase": "^5.3.1",
171 | "chalk": "^3.0.0",
172 | "cli-boxes": "^2.2.0",
173 | "string-width": "^4.1.0",
174 | "term-size": "^2.1.0",
175 | "type-fest": "^0.8.1",
176 | "widest-line": "^3.1.0"
177 | }
178 | },
179 | "brace-expansion": {
180 | "version": "1.1.11",
181 | "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz",
182 | "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==",
183 | "dev": true,
184 | "requires": {
185 | "balanced-match": "^1.0.0",
186 | "concat-map": "0.0.1"
187 | }
188 | },
189 | "braces": {
190 | "version": "3.0.2",
191 | "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz",
192 | "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==",
193 | "dev": true,
194 | "requires": {
195 | "fill-range": "^7.0.1"
196 | }
197 | },
198 | "bytes": {
199 | "version": "3.1.0",
200 | "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.0.tgz",
201 | "integrity": "sha512-zauLjrfCG+xvoyaqLoV8bLVXXNGC4JqlxFCutSDWA6fJrTo2ZuvLYTqZ7aHBLZSMOopbzwv8f+wZcVzfVTI2Dg=="
202 | },
203 | "cacheable-request": {
204 | "version": "6.1.0",
205 | "resolved": "https://registry.npmjs.org/cacheable-request/-/cacheable-request-6.1.0.tgz",
206 | "integrity": "sha512-Oj3cAGPCqOZX7Rz64Uny2GYAZNliQSqfbePrgAQ1wKAihYmCUnraBtJtKcGR4xz7wF+LoJC+ssFZvv5BgF9Igg==",
207 | "dev": true,
208 | "requires": {
209 | "clone-response": "^1.0.2",
210 | "get-stream": "^5.1.0",
211 | "http-cache-semantics": "^4.0.0",
212 | "keyv": "^3.0.0",
213 | "lowercase-keys": "^2.0.0",
214 | "normalize-url": "^4.1.0",
215 | "responselike": "^1.0.2"
216 | },
217 | "dependencies": {
218 | "get-stream": {
219 | "version": "5.2.0",
220 | "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-5.2.0.tgz",
221 | "integrity": "sha512-nBF+F1rAZVCu/p7rjzgA+Yb4lfYXrpl7a6VmJrU8wF9I1CKvP/QwPNZHnOlwbTkY6dvtFIzFMSyQXbLoTQPRpA==",
222 | "dev": true,
223 | "requires": {
224 | "pump": "^3.0.0"
225 | }
226 | },
227 | "lowercase-keys": {
228 | "version": "2.0.0",
229 | "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-2.0.0.tgz",
230 | "integrity": "sha512-tqNXrS78oMOE73NMxK4EMLQsQowWf8jKooH9g7xPavRT706R6bkQJ6DY2Te7QukaZsulxa30wQ7bk0pm4XiHmA==",
231 | "dev": true
232 | }
233 | }
234 | },
235 | "camelcase": {
236 | "version": "5.3.1",
237 | "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz",
238 | "integrity": "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==",
239 | "dev": true
240 | },
241 | "chalk": {
242 | "version": "3.0.0",
243 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-3.0.0.tgz",
244 | "integrity": "sha512-4D3B6Wf41KOYRFdszmDqMCGq5VV/uMAB273JILmO+3jAlh8X4qDtdtgCR3fxtbLEMzSx22QdhnDcJvu2u1fVwg==",
245 | "dev": true,
246 | "requires": {
247 | "ansi-styles": "^4.1.0",
248 | "supports-color": "^7.1.0"
249 | },
250 | "dependencies": {
251 | "has-flag": {
252 | "version": "4.0.0",
253 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz",
254 | "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==",
255 | "dev": true
256 | },
257 | "supports-color": {
258 | "version": "7.2.0",
259 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz",
260 | "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==",
261 | "dev": true,
262 | "requires": {
263 | "has-flag": "^4.0.0"
264 | }
265 | }
266 | }
267 | },
268 | "chokidar": {
269 | "version": "3.5.1",
270 | "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.5.1.tgz",
271 | "integrity": "sha512-9+s+Od+W0VJJzawDma/gvBNQqkTiqYTWLuZoyAsivsI4AaWTCzHG06/TMjsf1cYe9Cb97UCEhjz7HvnPk2p/tw==",
272 | "dev": true,
273 | "requires": {
274 | "anymatch": "~3.1.1",
275 | "braces": "~3.0.2",
276 | "fsevents": "~2.3.1",
277 | "glob-parent": "~5.1.0",
278 | "is-binary-path": "~2.1.0",
279 | "is-glob": "~4.0.1",
280 | "normalize-path": "~3.0.0",
281 | "readdirp": "~3.5.0"
282 | }
283 | },
284 | "ci-info": {
285 | "version": "2.0.0",
286 | "resolved": "https://registry.npmjs.org/ci-info/-/ci-info-2.0.0.tgz",
287 | "integrity": "sha512-5tK7EtrZ0N+OLFMthtqOj4fI2Jeb88C4CAZPu25LDVUgXJ0A3Js4PMGqrn0JU1W0Mh1/Z8wZzYPxqUrXeBboCQ==",
288 | "dev": true
289 | },
290 | "cli-boxes": {
291 | "version": "2.2.1",
292 | "resolved": "https://registry.npmjs.org/cli-boxes/-/cli-boxes-2.2.1.tgz",
293 | "integrity": "sha512-y4coMcylgSCdVinjiDBuR8PCC2bLjyGTwEmPb9NHR/QaNU6EUOXcTY/s6VjGMD6ENSEaeQYHCY0GNGS5jfMwPw==",
294 | "dev": true
295 | },
296 | "clone-response": {
297 | "version": "1.0.2",
298 | "resolved": "https://registry.npmjs.org/clone-response/-/clone-response-1.0.2.tgz",
299 | "integrity": "sha1-0dyXOSAxTfZ/vrlCI7TuNQI56Ws=",
300 | "dev": true,
301 | "requires": {
302 | "mimic-response": "^1.0.0"
303 | }
304 | },
305 | "cluster-key-slot": {
306 | "version": "1.1.0",
307 | "resolved": "https://registry.npmjs.org/cluster-key-slot/-/cluster-key-slot-1.1.0.tgz",
308 | "integrity": "sha512-2Nii8p3RwAPiFwsnZvukotvow2rIHM+yQ6ZcBXGHdniadkYGZYiGmkHJIbZPIV9nfv7m/U1IPMVVcAhoWFeklw=="
309 | },
310 | "color-convert": {
311 | "version": "2.0.1",
312 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz",
313 | "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==",
314 | "dev": true,
315 | "requires": {
316 | "color-name": "~1.1.4"
317 | }
318 | },
319 | "color-name": {
320 | "version": "1.1.4",
321 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz",
322 | "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==",
323 | "dev": true
324 | },
325 | "common-tags": {
326 | "version": "1.8.0",
327 | "resolved": "https://registry.npmjs.org/common-tags/-/common-tags-1.8.0.tgz",
328 | "integrity": "sha512-6P6g0uetGpW/sdyUy/iQQCbFF0kWVMSIVSyYz7Zgjcgh8mgw8PQzDNZeyZ5DQ2gM7LBoZPHmnjz8rUthkBG5tw=="
329 | },
330 | "component-emitter": {
331 | "version": "1.3.0",
332 | "resolved": "https://registry.npmjs.org/component-emitter/-/component-emitter-1.3.0.tgz",
333 | "integrity": "sha512-Rd3se6QB+sO1TwqZjscQrurpEPIfO0/yYnSin6Q/rD3mOutHvUrCAhJub3r90uNb+SESBuE0QYoB90YdfatsRg=="
334 | },
335 | "concat-map": {
336 | "version": "0.0.1",
337 | "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz",
338 | "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=",
339 | "dev": true
340 | },
341 | "configstore": {
342 | "version": "5.0.1",
343 | "resolved": "https://registry.npmjs.org/configstore/-/configstore-5.0.1.tgz",
344 | "integrity": "sha512-aMKprgk5YhBNyH25hj8wGt2+D52Sw1DRRIzqBwLp2Ya9mFmY8KPvvtvmna8SxVR9JMZ4kzMD68N22vlaRpkeFA==",
345 | "dev": true,
346 | "requires": {
347 | "dot-prop": "^5.2.0",
348 | "graceful-fs": "^4.1.2",
349 | "make-dir": "^3.0.0",
350 | "unique-string": "^2.0.0",
351 | "write-file-atomic": "^3.0.0",
352 | "xdg-basedir": "^4.0.0"
353 | }
354 | },
355 | "content-disposition": {
356 | "version": "0.5.3",
357 | "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.3.tgz",
358 | "integrity": "sha512-ExO0774ikEObIAEV9kDo50o+79VCUdEB6n6lzKgGwupcVeRlhrj3qGAfwq8G6uBJjkqLrhT0qEYFcWng8z1z0g==",
359 | "requires": {
360 | "safe-buffer": "5.1.2"
361 | }
362 | },
363 | "content-type": {
364 | "version": "1.0.4",
365 | "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.4.tgz",
366 | "integrity": "sha512-hIP3EEPs8tB9AT1L+NUqtwOAps4mk2Zob89MWXMHjHWg9milF/j4osnnQLXBCBFBk/tvIG/tUc9mOUJiPBhPXA=="
367 | },
368 | "cookie": {
369 | "version": "0.4.0",
370 | "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.4.0.tgz",
371 | "integrity": "sha512-+Hp8fLp57wnUSt0tY0tHEXh4voZRDnoIrZPqlo3DPiI4y9lwg/jqx+1Om94/W6ZaPDOUbnjOt/99w66zk+l1Xg=="
372 | },
373 | "cookie-signature": {
374 | "version": "1.0.6",
375 | "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz",
376 | "integrity": "sha1-4wOogrNCzD7oylE6eZmXNNqzriw="
377 | },
378 | "cors": {
379 | "version": "2.8.5",
380 | "resolved": "https://registry.npmjs.org/cors/-/cors-2.8.5.tgz",
381 | "integrity": "sha512-KIHbLJqu73RGr/hnbrO9uBeixNGuvSQjul/jdFvS/KFSIH1hWVd1ng7zOHx+YrEfInLG7q4n6GHQ9cDtxv/P6g==",
382 | "requires": {
383 | "object-assign": "^4",
384 | "vary": "^1"
385 | }
386 | },
387 | "crypto-random-string": {
388 | "version": "2.0.0",
389 | "resolved": "https://registry.npmjs.org/crypto-random-string/-/crypto-random-string-2.0.0.tgz",
390 | "integrity": "sha512-v1plID3y9r/lPhviJ1wrXpLeyUIGAZ2SHNYTEapm7/8A9nLPoyvVp3RK/EPFqn5kEznyWgYZNsRtYYIWbuG8KA==",
391 | "dev": true
392 | },
393 | "debug": {
394 | "version": "4.3.1",
395 | "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.1.tgz",
396 | "integrity": "sha512-doEwdvm4PCeK4K3RQN2ZC2BYUBaxwLARCqZmMjtF8a51J2Rb0xpVloFRnCODwqjpwnAoao4pelN8l3RJdv3gRQ==",
397 | "requires": {
398 | "ms": "2.1.2"
399 | }
400 | },
401 | "decompress-response": {
402 | "version": "3.3.0",
403 | "resolved": "https://registry.npmjs.org/decompress-response/-/decompress-response-3.3.0.tgz",
404 | "integrity": "sha1-gKTdMjdIOEv6JICDYirt7Jgq3/M=",
405 | "dev": true,
406 | "requires": {
407 | "mimic-response": "^1.0.0"
408 | }
409 | },
410 | "deep-extend": {
411 | "version": "0.6.0",
412 | "resolved": "https://registry.npmjs.org/deep-extend/-/deep-extend-0.6.0.tgz",
413 | "integrity": "sha512-LOHxIOaPYdHlJRtCQfDIVZtfw/ufM8+rVj649RIHzcm/vGwQRXFt6OPqIFWsm2XEMrNIEtWR64sY1LEKD2vAOA==",
414 | "dev": true
415 | },
416 | "defer-to-connect": {
417 | "version": "1.1.3",
418 | "resolved": "https://registry.npmjs.org/defer-to-connect/-/defer-to-connect-1.1.3.tgz",
419 | "integrity": "sha512-0ISdNousHvZT2EiFlZeZAHBUvSxmKswVCEf8hW7KWgG4a8MVEu/3Vb6uWYozkjylyCxe0JBIiRB1jV45S70WVQ==",
420 | "dev": true
421 | },
422 | "denque": {
423 | "version": "1.5.0",
424 | "resolved": "https://registry.npmjs.org/denque/-/denque-1.5.0.tgz",
425 | "integrity": "sha512-CYiCSgIF1p6EUByQPlGkKnP1M9g0ZV3qMIrqMqZqdwazygIA/YP2vrbcyl1h/WppKJTdl1F85cXIle+394iDAQ=="
426 | },
427 | "depd": {
428 | "version": "1.1.2",
429 | "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.2.tgz",
430 | "integrity": "sha1-m81S4UwJd2PnSbJ0xDRu0uVgtak="
431 | },
432 | "destroy": {
433 | "version": "1.0.4",
434 | "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.0.4.tgz",
435 | "integrity": "sha1-l4hXRCxEdJ5CBmE+N5RiBYJqvYA="
436 | },
437 | "dot-prop": {
438 | "version": "5.3.0",
439 | "resolved": "https://registry.npmjs.org/dot-prop/-/dot-prop-5.3.0.tgz",
440 | "integrity": "sha512-QM8q3zDe58hqUqjraQOmzZ1LIH9SWQJTlEKCH4kJ2oQvLZk7RbQXvtDM2XEq3fwkV9CCvvH4LA0AV+ogFsBM2Q==",
441 | "dev": true,
442 | "requires": {
443 | "is-obj": "^2.0.0"
444 | }
445 | },
446 | "dotenv": {
447 | "version": "8.2.0",
448 | "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-8.2.0.tgz",
449 | "integrity": "sha512-8sJ78ElpbDJBHNeBzUbUVLsqKdccaa/BXF1uPTw3GrvQTBgrQrtObr2mUrE38vzYd8cEv+m/JBfDLioYcfXoaw=="
450 | },
451 | "duplexer3": {
452 | "version": "0.1.4",
453 | "resolved": "https://registry.npmjs.org/duplexer3/-/duplexer3-0.1.4.tgz",
454 | "integrity": "sha1-7gHdHKwO08vH/b6jfcCo8c4ALOI=",
455 | "dev": true
456 | },
457 | "ee-first": {
458 | "version": "1.1.1",
459 | "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz",
460 | "integrity": "sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0="
461 | },
462 | "emoji-regex": {
463 | "version": "7.0.3",
464 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-7.0.3.tgz",
465 | "integrity": "sha512-CwBLREIQ7LvYFB0WyRvwhq5N5qPhc6PMjD6bYggFlI5YyDgl+0vxq5VHbMOFqLg7hfWzmu8T5Z1QofhmTIhItA==",
466 | "dev": true
467 | },
468 | "encodeurl": {
469 | "version": "1.0.2",
470 | "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz",
471 | "integrity": "sha1-rT/0yG7C0CkyL1oCw6mmBslbP1k="
472 | },
473 | "end-of-stream": {
474 | "version": "1.4.4",
475 | "resolved": "https://registry.npmjs.org/end-of-stream/-/end-of-stream-1.4.4.tgz",
476 | "integrity": "sha512-+uw1inIHVPQoaVuHzRyXd21icM+cnt4CzD5rW+NC1wjOUSTOs+Te7FOv7AhN7vS9x/oIyhLP5PR1H+phQAHu5Q==",
477 | "dev": true,
478 | "requires": {
479 | "once": "^1.4.0"
480 | }
481 | },
482 | "engine.io": {
483 | "version": "5.0.0",
484 | "resolved": "https://registry.npmjs.org/engine.io/-/engine.io-5.0.0.tgz",
485 | "integrity": "sha512-BATIdDV3H1SrE9/u2BAotvsmjJg0t1P4+vGedImSs1lkFAtQdvk4Ev1y4LDiPF7BPWgXWEG+NDY+nLvW3UrMWw==",
486 | "requires": {
487 | "accepts": "~1.3.4",
488 | "base64id": "2.0.0",
489 | "cookie": "~0.4.1",
490 | "cors": "~2.8.5",
491 | "debug": "~4.3.1",
492 | "engine.io-parser": "~4.0.0",
493 | "ws": "~7.4.2"
494 | },
495 | "dependencies": {
496 | "cookie": {
497 | "version": "0.4.1",
498 | "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.4.1.tgz",
499 | "integrity": "sha512-ZwrFkGJxUR3EIoXtO+yVE69Eb7KlixbaeAWfBQB9vVsNn/o+Yw69gBWSSDK825hQNdN+wF8zELf3dFNl/kxkUA=="
500 | }
501 | }
502 | },
503 | "engine.io-parser": {
504 | "version": "4.0.2",
505 | "resolved": "https://registry.npmjs.org/engine.io-parser/-/engine.io-parser-4.0.2.tgz",
506 | "integrity": "sha512-sHfEQv6nmtJrq6TKuIz5kyEKH/qSdK56H/A+7DnAuUPWosnIZAS2NHNcPLmyjtY3cGS/MqJdZbUjW97JU72iYg==",
507 | "requires": {
508 | "base64-arraybuffer": "0.1.4"
509 | }
510 | },
511 | "escape-goat": {
512 | "version": "2.1.1",
513 | "resolved": "https://registry.npmjs.org/escape-goat/-/escape-goat-2.1.1.tgz",
514 | "integrity": "sha512-8/uIhbG12Csjy2JEW7D9pHbreaVaS/OpN3ycnyvElTdwM5n6GY6W6e2IPemfvGZeUMqZ9A/3GqIZMgKnBhAw/Q==",
515 | "dev": true
516 | },
517 | "escape-html": {
518 | "version": "1.0.3",
519 | "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz",
520 | "integrity": "sha1-Aljq5NPQwJdN4cFpGI7wBR0dGYg="
521 | },
522 | "esm": {
523 | "version": "3.2.25",
524 | "resolved": "https://registry.npmjs.org/esm/-/esm-3.2.25.tgz",
525 | "integrity": "sha512-U1suiZ2oDVWv4zPO56S0NcR5QriEahGtdN2OR6FiOG4WJvcjBVFB0qI4+eKoWFH483PKGuLuu6V8Z4T5g63UVA=="
526 | },
527 | "etag": {
528 | "version": "1.8.1",
529 | "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz",
530 | "integrity": "sha1-Qa4u62XvpiJorr/qg6x9eSmbCIc="
531 | },
532 | "express": {
533 | "version": "4.17.1",
534 | "resolved": "https://registry.npmjs.org/express/-/express-4.17.1.tgz",
535 | "integrity": "sha512-mHJ9O79RqluphRrcw2X/GTh3k9tVv8YcoyY4Kkh4WDMUYKRZUq0h1o0w2rrrxBqM7VoeUVqgb27xlEMXTnYt4g==",
536 | "requires": {
537 | "accepts": "~1.3.7",
538 | "array-flatten": "1.1.1",
539 | "body-parser": "1.19.0",
540 | "content-disposition": "0.5.3",
541 | "content-type": "~1.0.4",
542 | "cookie": "0.4.0",
543 | "cookie-signature": "1.0.6",
544 | "debug": "2.6.9",
545 | "depd": "~1.1.2",
546 | "encodeurl": "~1.0.2",
547 | "escape-html": "~1.0.3",
548 | "etag": "~1.8.1",
549 | "finalhandler": "~1.1.2",
550 | "fresh": "0.5.2",
551 | "merge-descriptors": "1.0.1",
552 | "methods": "~1.1.2",
553 | "on-finished": "~2.3.0",
554 | "parseurl": "~1.3.3",
555 | "path-to-regexp": "0.1.7",
556 | "proxy-addr": "~2.0.5",
557 | "qs": "6.7.0",
558 | "range-parser": "~1.2.1",
559 | "safe-buffer": "5.1.2",
560 | "send": "0.17.1",
561 | "serve-static": "1.14.1",
562 | "setprototypeof": "1.1.1",
563 | "statuses": "~1.5.0",
564 | "type-is": "~1.6.18",
565 | "utils-merge": "1.0.1",
566 | "vary": "~1.1.2"
567 | },
568 | "dependencies": {
569 | "debug": {
570 | "version": "2.6.9",
571 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz",
572 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==",
573 | "requires": {
574 | "ms": "2.0.0"
575 | }
576 | },
577 | "ms": {
578 | "version": "2.0.0",
579 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz",
580 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g="
581 | }
582 | }
583 | },
584 | "fill-range": {
585 | "version": "7.0.1",
586 | "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz",
587 | "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==",
588 | "dev": true,
589 | "requires": {
590 | "to-regex-range": "^5.0.1"
591 | }
592 | },
593 | "finalhandler": {
594 | "version": "1.1.2",
595 | "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.1.2.tgz",
596 | "integrity": "sha512-aAWcW57uxVNrQZqFXjITpW3sIUQmHGG3qSb9mUah9MgMC4NeWhNOlNjXEYq3HjRAvL6arUviZGGJsBg6z0zsWA==",
597 | "requires": {
598 | "debug": "2.6.9",
599 | "encodeurl": "~1.0.2",
600 | "escape-html": "~1.0.3",
601 | "on-finished": "~2.3.0",
602 | "parseurl": "~1.3.3",
603 | "statuses": "~1.5.0",
604 | "unpipe": "~1.0.0"
605 | },
606 | "dependencies": {
607 | "debug": {
608 | "version": "2.6.9",
609 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz",
610 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==",
611 | "requires": {
612 | "ms": "2.0.0"
613 | }
614 | },
615 | "ms": {
616 | "version": "2.0.0",
617 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz",
618 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g="
619 | }
620 | }
621 | },
622 | "forwarded": {
623 | "version": "0.1.2",
624 | "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.1.2.tgz",
625 | "integrity": "sha1-mMI9qxF1ZXuMBXPozszZGw/xjIQ="
626 | },
627 | "fresh": {
628 | "version": "0.5.2",
629 | "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz",
630 | "integrity": "sha1-PYyt2Q2XZWn6g1qx+OSyOhBWBac="
631 | },
632 | "fsevents": {
633 | "version": "2.3.2",
634 | "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.2.tgz",
635 | "integrity": "sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==",
636 | "dev": true,
637 | "optional": true
638 | },
639 | "get-stream": {
640 | "version": "4.1.0",
641 | "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-4.1.0.tgz",
642 | "integrity": "sha512-GMat4EJ5161kIy2HevLlr4luNjBgvmj413KaQA7jt4V8B4RDsfpHk7WQ9GVqfYyyx8OS/L66Kox+rJRNklLK7w==",
643 | "dev": true,
644 | "requires": {
645 | "pump": "^3.0.0"
646 | }
647 | },
648 | "glob-parent": {
649 | "version": "5.1.2",
650 | "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz",
651 | "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==",
652 | "dev": true,
653 | "requires": {
654 | "is-glob": "^4.0.1"
655 | }
656 | },
657 | "global-dirs": {
658 | "version": "2.1.0",
659 | "resolved": "https://registry.npmjs.org/global-dirs/-/global-dirs-2.1.0.tgz",
660 | "integrity": "sha512-MG6kdOUh/xBnyo9cJFeIKkLEc1AyFq42QTU4XiX51i2NEdxLxLWXIjEjmqKeSuKR7pAZjTqUVoT2b2huxVLgYQ==",
661 | "dev": true,
662 | "requires": {
663 | "ini": "1.3.7"
664 | }
665 | },
666 | "got": {
667 | "version": "9.6.0",
668 | "resolved": "https://registry.npmjs.org/got/-/got-9.6.0.tgz",
669 | "integrity": "sha512-R7eWptXuGYxwijs0eV+v3o6+XH1IqVK8dJOEecQfTmkncw9AV4dcw/Dhxi8MdlqPthxxpZyizMzyg8RTmEsG+Q==",
670 | "dev": true,
671 | "requires": {
672 | "@sindresorhus/is": "^0.14.0",
673 | "@szmarczak/http-timer": "^1.1.2",
674 | "cacheable-request": "^6.0.0",
675 | "decompress-response": "^3.3.0",
676 | "duplexer3": "^0.1.4",
677 | "get-stream": "^4.1.0",
678 | "lowercase-keys": "^1.0.1",
679 | "mimic-response": "^1.0.1",
680 | "p-cancelable": "^1.0.0",
681 | "to-readable-stream": "^1.0.0",
682 | "url-parse-lax": "^3.0.0"
683 | }
684 | },
685 | "graceful-fs": {
686 | "version": "4.2.6",
687 | "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.6.tgz",
688 | "integrity": "sha512-nTnJ528pbqxYanhpDYsi4Rd8MAeaBA67+RZ10CM1m3bTAVFEDcd5AuA4a6W5YkGZ1iNXHzZz8T6TBKLeBuNriQ==",
689 | "dev": true
690 | },
691 | "has-flag": {
692 | "version": "3.0.0",
693 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz",
694 | "integrity": "sha1-tdRU3CGZriJWmfNGfloH87lVuv0=",
695 | "dev": true
696 | },
697 | "has-yarn": {
698 | "version": "2.1.0",
699 | "resolved": "https://registry.npmjs.org/has-yarn/-/has-yarn-2.1.0.tgz",
700 | "integrity": "sha512-UqBRqi4ju7T+TqGNdqAO0PaSVGsDGJUBQvk9eUWNGRY1CFGDzYhLWoM7JQEemnlvVcv/YEmc2wNW8BC24EnUsw==",
701 | "dev": true
702 | },
703 | "http-cache-semantics": {
704 | "version": "4.1.0",
705 | "resolved": "https://registry.npmjs.org/http-cache-semantics/-/http-cache-semantics-4.1.0.tgz",
706 | "integrity": "sha512-carPklcUh7ROWRK7Cv27RPtdhYhUsela/ue5/jKzjegVvXDqM2ILE9Q2BGn9JZJh1g87cp56su/FgQSzcWS8cQ==",
707 | "dev": true
708 | },
709 | "http-errors": {
710 | "version": "1.7.2",
711 | "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.7.2.tgz",
712 | "integrity": "sha512-uUQBt3H/cSIVfch6i1EuPNy/YsRSOUBXTVfZ+yR7Zjez3qjBz6i9+i4zjNaoqcoFVI4lQJ5plg63TvGfRSDCRg==",
713 | "requires": {
714 | "depd": "~1.1.2",
715 | "inherits": "2.0.3",
716 | "setprototypeof": "1.1.1",
717 | "statuses": ">= 1.5.0 < 2",
718 | "toidentifier": "1.0.0"
719 | }
720 | },
721 | "iconv-lite": {
722 | "version": "0.4.24",
723 | "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz",
724 | "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==",
725 | "requires": {
726 | "safer-buffer": ">= 2.1.2 < 3"
727 | }
728 | },
729 | "ignore-by-default": {
730 | "version": "1.0.1",
731 | "resolved": "https://registry.npmjs.org/ignore-by-default/-/ignore-by-default-1.0.1.tgz",
732 | "integrity": "sha1-SMptcvbGo68Aqa1K5odr44ieKwk=",
733 | "dev": true
734 | },
735 | "import-lazy": {
736 | "version": "2.1.0",
737 | "resolved": "https://registry.npmjs.org/import-lazy/-/import-lazy-2.1.0.tgz",
738 | "integrity": "sha1-BWmOPUXIjo1+nZLLBYTnfwlvPkM=",
739 | "dev": true
740 | },
741 | "imurmurhash": {
742 | "version": "0.1.4",
743 | "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz",
744 | "integrity": "sha1-khi5srkoojixPcT7a21XbyMUU+o=",
745 | "dev": true
746 | },
747 | "inherits": {
748 | "version": "2.0.3",
749 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz",
750 | "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4="
751 | },
752 | "ini": {
753 | "version": "1.3.7",
754 | "resolved": "https://registry.npmjs.org/ini/-/ini-1.3.7.tgz",
755 | "integrity": "sha512-iKpRpXP+CrP2jyrxvg1kMUpXDyRUFDWurxbnVT1vQPx+Wz9uCYsMIqYuSBLV+PAaZG/d7kRLKRFc9oDMsH+mFQ==",
756 | "dev": true
757 | },
758 | "ioredis": {
759 | "version": "4.26.0",
760 | "resolved": "https://registry.npmjs.org/ioredis/-/ioredis-4.26.0.tgz",
761 | "integrity": "sha512-nh39okWezWWZ35/RxXXzHksMFt4WCaev8SNO2kozRDeVdEAJj16EarqPP3JeHz8IEjEXN5CiVtbWMk62Z0eveQ==",
762 | "requires": {
763 | "cluster-key-slot": "^1.1.0",
764 | "debug": "^4.3.1",
765 | "denque": "^1.1.0",
766 | "lodash.defaults": "^4.2.0",
767 | "lodash.flatten": "^4.4.0",
768 | "p-map": "^2.1.0",
769 | "redis-commands": "1.7.0",
770 | "redis-errors": "^1.2.0",
771 | "redis-parser": "^3.0.0",
772 | "standard-as-callback": "^2.1.0"
773 | }
774 | },
775 | "ipaddr.js": {
776 | "version": "1.9.1",
777 | "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz",
778 | "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g=="
779 | },
780 | "is-binary-path": {
781 | "version": "2.1.0",
782 | "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz",
783 | "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==",
784 | "dev": true,
785 | "requires": {
786 | "binary-extensions": "^2.0.0"
787 | }
788 | },
789 | "is-ci": {
790 | "version": "2.0.0",
791 | "resolved": "https://registry.npmjs.org/is-ci/-/is-ci-2.0.0.tgz",
792 | "integrity": "sha512-YfJT7rkpQB0updsdHLGWrvhBJfcfzNNawYDNIyQXJz0IViGf75O8EBPKSdvw2rF+LGCsX4FZ8tcr3b19LcZq4w==",
793 | "dev": true,
794 | "requires": {
795 | "ci-info": "^2.0.0"
796 | }
797 | },
798 | "is-extglob": {
799 | "version": "2.1.1",
800 | "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz",
801 | "integrity": "sha1-qIwCU1eR8C7TfHahueqXc8gz+MI=",
802 | "dev": true
803 | },
804 | "is-fullwidth-code-point": {
805 | "version": "2.0.0",
806 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz",
807 | "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=",
808 | "dev": true
809 | },
810 | "is-glob": {
811 | "version": "4.0.1",
812 | "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.1.tgz",
813 | "integrity": "sha512-5G0tKtBTFImOqDnLB2hG6Bp2qcKEFduo4tZu9MT/H6NQv/ghhy30o55ufafxJ/LdH79LLs2Kfrn85TLKyA7BUg==",
814 | "dev": true,
815 | "requires": {
816 | "is-extglob": "^2.1.1"
817 | }
818 | },
819 | "is-installed-globally": {
820 | "version": "0.3.2",
821 | "resolved": "https://registry.npmjs.org/is-installed-globally/-/is-installed-globally-0.3.2.tgz",
822 | "integrity": "sha512-wZ8x1js7Ia0kecP/CHM/3ABkAmujX7WPvQk6uu3Fly/Mk44pySulQpnHG46OMjHGXApINnV4QhY3SWnECO2z5g==",
823 | "dev": true,
824 | "requires": {
825 | "global-dirs": "^2.0.1",
826 | "is-path-inside": "^3.0.1"
827 | }
828 | },
829 | "is-npm": {
830 | "version": "4.0.0",
831 | "resolved": "https://registry.npmjs.org/is-npm/-/is-npm-4.0.0.tgz",
832 | "integrity": "sha512-96ECIfh9xtDDlPylNPXhzjsykHsMJZ18ASpaWzQyBr4YRTcVjUvzaHayDAES2oU/3KpljhHUjtSRNiDwi0F0ig==",
833 | "dev": true
834 | },
835 | "is-number": {
836 | "version": "7.0.0",
837 | "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz",
838 | "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==",
839 | "dev": true
840 | },
841 | "is-obj": {
842 | "version": "2.0.0",
843 | "resolved": "https://registry.npmjs.org/is-obj/-/is-obj-2.0.0.tgz",
844 | "integrity": "sha512-drqDG3cbczxxEJRoOXcOjtdp1J/lyp1mNn0xaznRs8+muBhgQcrnbspox5X5fOw0HnMnbfDzvnEMEtqDEJEo8w==",
845 | "dev": true
846 | },
847 | "is-path-inside": {
848 | "version": "3.0.3",
849 | "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-3.0.3.tgz",
850 | "integrity": "sha512-Fd4gABb+ycGAmKou8eMftCupSir5lRxqf4aD/vd0cD2qc4HL07OjCeuHMr8Ro4CoMaeCKDB0/ECBOVWjTwUvPQ==",
851 | "dev": true
852 | },
853 | "is-typedarray": {
854 | "version": "1.0.0",
855 | "resolved": "https://registry.npmjs.org/is-typedarray/-/is-typedarray-1.0.0.tgz",
856 | "integrity": "sha1-5HnICFjfDBsR3dppQPlgEfzaSpo=",
857 | "dev": true
858 | },
859 | "is-yarn-global": {
860 | "version": "0.3.0",
861 | "resolved": "https://registry.npmjs.org/is-yarn-global/-/is-yarn-global-0.3.0.tgz",
862 | "integrity": "sha512-VjSeb/lHmkoyd8ryPVIKvOCn4D1koMqY+vqyjjUfc3xyKtP4dYOxM44sZrnqQSzSds3xyOrUTLTC9LVCVgLngw==",
863 | "dev": true
864 | },
865 | "json-buffer": {
866 | "version": "3.0.0",
867 | "resolved": "https://registry.npmjs.org/json-buffer/-/json-buffer-3.0.0.tgz",
868 | "integrity": "sha1-Wx85evx11ne96Lz8Dkfh+aPZqJg=",
869 | "dev": true
870 | },
871 | "keyv": {
872 | "version": "3.1.0",
873 | "resolved": "https://registry.npmjs.org/keyv/-/keyv-3.1.0.tgz",
874 | "integrity": "sha512-9ykJ/46SN/9KPM/sichzQ7OvXyGDYKGTaDlKMGCAlg2UK8KRy4jb0d8sFc+0Tt0YYnThq8X2RZgCg74RPxgcVA==",
875 | "dev": true,
876 | "requires": {
877 | "json-buffer": "3.0.0"
878 | }
879 | },
880 | "latest-version": {
881 | "version": "5.1.0",
882 | "resolved": "https://registry.npmjs.org/latest-version/-/latest-version-5.1.0.tgz",
883 | "integrity": "sha512-weT+r0kTkRQdCdYCNtkMwWXQTMEswKrFBkm4ckQOMVhhqhIMI1UT2hMj+1iigIhgSZm5gTmrRXBNoGUgaTY1xA==",
884 | "dev": true,
885 | "requires": {
886 | "package-json": "^6.3.0"
887 | }
888 | },
889 | "lodash.defaults": {
890 | "version": "4.2.0",
891 | "resolved": "https://registry.npmjs.org/lodash.defaults/-/lodash.defaults-4.2.0.tgz",
892 | "integrity": "sha1-0JF4cW/+pN3p5ft7N/bwgCJ0WAw="
893 | },
894 | "lodash.flatten": {
895 | "version": "4.4.0",
896 | "resolved": "https://registry.npmjs.org/lodash.flatten/-/lodash.flatten-4.4.0.tgz",
897 | "integrity": "sha1-8xwiIlqWMtK7+OSt2+8kCqdlph8="
898 | },
899 | "lowercase-keys": {
900 | "version": "1.0.1",
901 | "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-1.0.1.tgz",
902 | "integrity": "sha512-G2Lj61tXDnVFFOi8VZds+SoQjtQC3dgokKdDG2mTm1tx4m50NUHBOZSBwQQHyy0V12A0JTG4icfZQH+xPyh8VA==",
903 | "dev": true
904 | },
905 | "make-dir": {
906 | "version": "3.1.0",
907 | "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-3.1.0.tgz",
908 | "integrity": "sha512-g3FeP20LNwhALb/6Cz6Dd4F2ngze0jz7tbzrD2wAV+o9FeNHe4rL+yK2md0J/fiSf1sa1ADhXqi5+oVwOM/eGw==",
909 | "dev": true,
910 | "requires": {
911 | "semver": "^6.0.0"
912 | },
913 | "dependencies": {
914 | "semver": {
915 | "version": "6.3.0",
916 | "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz",
917 | "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==",
918 | "dev": true
919 | }
920 | }
921 | },
922 | "media-typer": {
923 | "version": "0.3.0",
924 | "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz",
925 | "integrity": "sha1-hxDXrwqmJvj/+hzgAWhUUmMlV0g="
926 | },
927 | "merge-descriptors": {
928 | "version": "1.0.1",
929 | "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz",
930 | "integrity": "sha1-sAqqVW3YtEVoFQ7J0blT8/kMu2E="
931 | },
932 | "methods": {
933 | "version": "1.1.2",
934 | "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz",
935 | "integrity": "sha1-VSmk1nZUE07cxSZmVoNbD4Ua/O4="
936 | },
937 | "mime": {
938 | "version": "1.6.0",
939 | "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz",
940 | "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg=="
941 | },
942 | "mime-db": {
943 | "version": "1.47.0",
944 | "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.47.0.tgz",
945 | "integrity": "sha512-QBmA/G2y+IfeS4oktet3qRZ+P5kPhCKRXxXnQEudYqUaEioAU1/Lq2us3D/t1Jfo4hE9REQPrbB7K5sOczJVIw=="
946 | },
947 | "mime-types": {
948 | "version": "2.1.30",
949 | "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.30.tgz",
950 | "integrity": "sha512-crmjA4bLtR8m9qLpHvgxSChT+XoSlZi8J4n/aIdn3z92e/U47Z0V/yl+Wh9W046GgFVAmoNR/fmdbZYcSSIUeg==",
951 | "requires": {
952 | "mime-db": "1.47.0"
953 | }
954 | },
955 | "mimic-response": {
956 | "version": "1.0.1",
957 | "resolved": "https://registry.npmjs.org/mimic-response/-/mimic-response-1.0.1.tgz",
958 | "integrity": "sha512-j5EctnkH7amfV/q5Hgmoal1g2QHFJRraOtmx0JpIqkxhBhI/lJSl1nMpQ45hVarwNETOoWEimndZ4QK0RHxuxQ==",
959 | "dev": true
960 | },
961 | "minimatch": {
962 | "version": "3.0.4",
963 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz",
964 | "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==",
965 | "dev": true,
966 | "requires": {
967 | "brace-expansion": "^1.1.7"
968 | }
969 | },
970 | "minimist": {
971 | "version": "1.2.5",
972 | "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.5.tgz",
973 | "integrity": "sha512-FM9nNUYrRBAELZQT3xeZQ7fmMOBg6nWNmJKTcgsJeaLstP/UODVpGsr5OhXhhXg6f+qtJ8uiZ+PUxkDWcgIXLw==",
974 | "dev": true
975 | },
976 | "ms": {
977 | "version": "2.1.2",
978 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz",
979 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w=="
980 | },
981 | "negotiator": {
982 | "version": "0.6.2",
983 | "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.2.tgz",
984 | "integrity": "sha512-hZXc7K2e+PgeI1eDBe/10Ard4ekbfrrqG8Ep+8Jmf4JID2bNg7NvCPOZN+kfF574pFQI7mum2AUqDidoKqcTOw=="
985 | },
986 | "nodemon": {
987 | "version": "2.0.7",
988 | "resolved": "https://registry.npmjs.org/nodemon/-/nodemon-2.0.7.tgz",
989 | "integrity": "sha512-XHzK69Awgnec9UzHr1kc8EomQh4sjTQ8oRf8TsGrSmHDx9/UmiGG9E/mM3BuTfNeFwdNBvrqQq/RHL0xIeyFOA==",
990 | "dev": true,
991 | "requires": {
992 | "chokidar": "^3.2.2",
993 | "debug": "^3.2.6",
994 | "ignore-by-default": "^1.0.1",
995 | "minimatch": "^3.0.4",
996 | "pstree.remy": "^1.1.7",
997 | "semver": "^5.7.1",
998 | "supports-color": "^5.5.0",
999 | "touch": "^3.1.0",
1000 | "undefsafe": "^2.0.3",
1001 | "update-notifier": "^4.1.0"
1002 | },
1003 | "dependencies": {
1004 | "debug": {
1005 | "version": "3.2.7",
1006 | "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz",
1007 | "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==",
1008 | "dev": true,
1009 | "requires": {
1010 | "ms": "^2.1.1"
1011 | }
1012 | }
1013 | }
1014 | },
1015 | "nopt": {
1016 | "version": "1.0.10",
1017 | "resolved": "https://registry.npmjs.org/nopt/-/nopt-1.0.10.tgz",
1018 | "integrity": "sha1-bd0hvSoxQXuScn3Vhfim83YI6+4=",
1019 | "dev": true,
1020 | "requires": {
1021 | "abbrev": "1"
1022 | }
1023 | },
1024 | "normalize-path": {
1025 | "version": "3.0.0",
1026 | "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz",
1027 | "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==",
1028 | "dev": true
1029 | },
1030 | "normalize-url": {
1031 | "version": "4.5.0",
1032 | "resolved": "https://registry.npmjs.org/normalize-url/-/normalize-url-4.5.0.tgz",
1033 | "integrity": "sha512-2s47yzUxdexf1OhyRi4Em83iQk0aPvwTddtFz4hnSSw9dCEsLEGf6SwIO8ss/19S9iBb5sJaOuTvTGDeZI00BQ==",
1034 | "dev": true
1035 | },
1036 | "object-assign": {
1037 | "version": "4.1.1",
1038 | "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz",
1039 | "integrity": "sha1-IQmtx5ZYh8/AXLvUQsrIv7s2CGM="
1040 | },
1041 | "on-finished": {
1042 | "version": "2.3.0",
1043 | "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.3.0.tgz",
1044 | "integrity": "sha1-IPEzZIGwg811M3mSoWlxqi2QaUc=",
1045 | "requires": {
1046 | "ee-first": "1.1.1"
1047 | }
1048 | },
1049 | "once": {
1050 | "version": "1.4.0",
1051 | "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz",
1052 | "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=",
1053 | "dev": true,
1054 | "requires": {
1055 | "wrappy": "1"
1056 | }
1057 | },
1058 | "p-cancelable": {
1059 | "version": "1.1.0",
1060 | "resolved": "https://registry.npmjs.org/p-cancelable/-/p-cancelable-1.1.0.tgz",
1061 | "integrity": "sha512-s73XxOZ4zpt1edZYZzvhqFa6uvQc1vwUa0K0BdtIZgQMAJj9IbebH+JkgKZc9h+B05PKHLOTl4ajG1BmNrVZlw==",
1062 | "dev": true
1063 | },
1064 | "p-map": {
1065 | "version": "2.1.0",
1066 | "resolved": "https://registry.npmjs.org/p-map/-/p-map-2.1.0.tgz",
1067 | "integrity": "sha512-y3b8Kpd8OAN444hxfBbFfj1FY/RjtTd8tzYwhUqNYXx0fXx2iX4maP4Qr6qhIKbQXI02wTLAda4fYUbDagTUFw=="
1068 | },
1069 | "package-json": {
1070 | "version": "6.5.0",
1071 | "resolved": "https://registry.npmjs.org/package-json/-/package-json-6.5.0.tgz",
1072 | "integrity": "sha512-k3bdm2n25tkyxcjSKzB5x8kfVxlMdgsbPr0GkZcwHsLpba6cBjqCt1KlcChKEvxHIcTB1FVMuwoijZ26xex5MQ==",
1073 | "dev": true,
1074 | "requires": {
1075 | "got": "^9.6.0",
1076 | "registry-auth-token": "^4.0.0",
1077 | "registry-url": "^5.0.0",
1078 | "semver": "^6.2.0"
1079 | },
1080 | "dependencies": {
1081 | "semver": {
1082 | "version": "6.3.0",
1083 | "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz",
1084 | "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==",
1085 | "dev": true
1086 | }
1087 | }
1088 | },
1089 | "parseurl": {
1090 | "version": "1.3.3",
1091 | "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz",
1092 | "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ=="
1093 | },
1094 | "path-to-regexp": {
1095 | "version": "0.1.7",
1096 | "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz",
1097 | "integrity": "sha1-32BBeABfUi8V60SQ5yR6G/qmf4w="
1098 | },
1099 | "picomatch": {
1100 | "version": "2.2.3",
1101 | "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.2.3.tgz",
1102 | "integrity": "sha512-KpELjfwcCDUb9PeigTs2mBJzXUPzAuP2oPcA989He8Rte0+YUAjw1JVedDhuTKPkHjSYzMN3npC9luThGYEKdg==",
1103 | "dev": true
1104 | },
1105 | "prepend-http": {
1106 | "version": "2.0.0",
1107 | "resolved": "https://registry.npmjs.org/prepend-http/-/prepend-http-2.0.0.tgz",
1108 | "integrity": "sha1-6SQ0v6XqjBn0HN/UAddBo8gZ2Jc=",
1109 | "dev": true
1110 | },
1111 | "proxy-addr": {
1112 | "version": "2.0.6",
1113 | "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.6.tgz",
1114 | "integrity": "sha512-dh/frvCBVmSsDYzw6n926jv974gddhkFPfiN8hPOi30Wax25QZyZEGveluCgliBnqmuM+UJmBErbAUFIoDbjOw==",
1115 | "requires": {
1116 | "forwarded": "~0.1.2",
1117 | "ipaddr.js": "1.9.1"
1118 | }
1119 | },
1120 | "pstree.remy": {
1121 | "version": "1.1.8",
1122 | "resolved": "https://registry.npmjs.org/pstree.remy/-/pstree.remy-1.1.8.tgz",
1123 | "integrity": "sha512-77DZwxQmxKnu3aR542U+X8FypNzbfJ+C5XQDk3uWjWxn6151aIMGthWYRXTqT1E5oJvg+ljaa2OJi+VfvCOQ8w==",
1124 | "dev": true
1125 | },
1126 | "pump": {
1127 | "version": "3.0.0",
1128 | "resolved": "https://registry.npmjs.org/pump/-/pump-3.0.0.tgz",
1129 | "integrity": "sha512-LwZy+p3SFs1Pytd/jYct4wpv49HiYCqd9Rlc5ZVdk0V+8Yzv6jR5Blk3TRmPL1ft69TxP0IMZGJ+WPFU2BFhww==",
1130 | "dev": true,
1131 | "requires": {
1132 | "end-of-stream": "^1.1.0",
1133 | "once": "^1.3.1"
1134 | }
1135 | },
1136 | "pupa": {
1137 | "version": "2.1.1",
1138 | "resolved": "https://registry.npmjs.org/pupa/-/pupa-2.1.1.tgz",
1139 | "integrity": "sha512-l1jNAspIBSFqbT+y+5FosojNpVpF94nlI+wDUpqP9enwOTfHx9f0gh5nB96vl+6yTpsJsypeNrwfzPrKuHB41A==",
1140 | "dev": true,
1141 | "requires": {
1142 | "escape-goat": "^2.0.0"
1143 | }
1144 | },
1145 | "qs": {
1146 | "version": "6.7.0",
1147 | "resolved": "https://registry.npmjs.org/qs/-/qs-6.7.0.tgz",
1148 | "integrity": "sha512-VCdBRNFTX1fyE7Nb6FYoURo/SPe62QCaAyzJvUjwRaIsc+NePBEniHlvxFmmX56+HZphIGtV0XeCirBtpDrTyQ=="
1149 | },
1150 | "range-parser": {
1151 | "version": "1.2.1",
1152 | "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz",
1153 | "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg=="
1154 | },
1155 | "raw-body": {
1156 | "version": "2.4.0",
1157 | "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.4.0.tgz",
1158 | "integrity": "sha512-4Oz8DUIwdvoa5qMJelxipzi/iJIi40O5cGV1wNYp5hvZP8ZN0T+jiNkL0QepXs+EsQ9XJ8ipEDoiH70ySUJP3Q==",
1159 | "requires": {
1160 | "bytes": "3.1.0",
1161 | "http-errors": "1.7.2",
1162 | "iconv-lite": "0.4.24",
1163 | "unpipe": "1.0.0"
1164 | }
1165 | },
1166 | "rc": {
1167 | "version": "1.2.8",
1168 | "resolved": "https://registry.npmjs.org/rc/-/rc-1.2.8.tgz",
1169 | "integrity": "sha512-y3bGgqKj3QBdxLbLkomlohkvsA8gdAiUQlSBJnBhfn+BPxg4bc62d8TcBW15wavDfgexCgccckhcZvywyQYPOw==",
1170 | "dev": true,
1171 | "requires": {
1172 | "deep-extend": "^0.6.0",
1173 | "ini": "~1.3.0",
1174 | "minimist": "^1.2.0",
1175 | "strip-json-comments": "~2.0.1"
1176 | }
1177 | },
1178 | "readdirp": {
1179 | "version": "3.5.0",
1180 | "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.5.0.tgz",
1181 | "integrity": "sha512-cMhu7c/8rdhkHXWsY+osBhfSy0JikwpHK/5+imo+LpeasTF8ouErHrlYkwT0++njiyuDvc7OFY5T3ukvZ8qmFQ==",
1182 | "dev": true,
1183 | "requires": {
1184 | "picomatch": "^2.2.1"
1185 | }
1186 | },
1187 | "redis-commands": {
1188 | "version": "1.7.0",
1189 | "resolved": "https://registry.npmjs.org/redis-commands/-/redis-commands-1.7.0.tgz",
1190 | "integrity": "sha512-nJWqw3bTFy21hX/CPKHth6sfhZbdiHP6bTawSgQBlKOVRG7EZkfHbbHwQJnrE4vsQf0CMNE+3gJ4Fmm16vdVlQ=="
1191 | },
1192 | "redis-errors": {
1193 | "version": "1.2.0",
1194 | "resolved": "https://registry.npmjs.org/redis-errors/-/redis-errors-1.2.0.tgz",
1195 | "integrity": "sha1-62LSrbFeTq9GEMBK/hUpOEJQq60="
1196 | },
1197 | "redis-parser": {
1198 | "version": "3.0.0",
1199 | "resolved": "https://registry.npmjs.org/redis-parser/-/redis-parser-3.0.0.tgz",
1200 | "integrity": "sha1-tm2CjNyv5rS4pCin3vTGvKwxyLQ=",
1201 | "requires": {
1202 | "redis-errors": "^1.0.0"
1203 | }
1204 | },
1205 | "registry-auth-token": {
1206 | "version": "4.2.1",
1207 | "resolved": "https://registry.npmjs.org/registry-auth-token/-/registry-auth-token-4.2.1.tgz",
1208 | "integrity": "sha512-6gkSb4U6aWJB4SF2ZvLb76yCBjcvufXBqvvEx1HbmKPkutswjW1xNVRY0+daljIYRbogN7O0etYSlbiaEQyMyw==",
1209 | "dev": true,
1210 | "requires": {
1211 | "rc": "^1.2.8"
1212 | }
1213 | },
1214 | "registry-url": {
1215 | "version": "5.1.0",
1216 | "resolved": "https://registry.npmjs.org/registry-url/-/registry-url-5.1.0.tgz",
1217 | "integrity": "sha512-8acYXXTI0AkQv6RAOjE3vOaIXZkT9wo4LOFbBKYQEEnnMNBpKqdUrI6S4NT0KPIo/WVvJ5tE/X5LF/TQUf0ekw==",
1218 | "dev": true,
1219 | "requires": {
1220 | "rc": "^1.2.8"
1221 | }
1222 | },
1223 | "responselike": {
1224 | "version": "1.0.2",
1225 | "resolved": "https://registry.npmjs.org/responselike/-/responselike-1.0.2.tgz",
1226 | "integrity": "sha1-kYcg7ztjHFZCvgaPFa3lpG9Loec=",
1227 | "dev": true,
1228 | "requires": {
1229 | "lowercase-keys": "^1.0.0"
1230 | }
1231 | },
1232 | "safe-buffer": {
1233 | "version": "5.1.2",
1234 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz",
1235 | "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g=="
1236 | },
1237 | "safer-buffer": {
1238 | "version": "2.1.2",
1239 | "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz",
1240 | "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg=="
1241 | },
1242 | "semver": {
1243 | "version": "5.7.1",
1244 | "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz",
1245 | "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==",
1246 | "dev": true
1247 | },
1248 | "semver-diff": {
1249 | "version": "3.1.1",
1250 | "resolved": "https://registry.npmjs.org/semver-diff/-/semver-diff-3.1.1.tgz",
1251 | "integrity": "sha512-GX0Ix/CJcHyB8c4ykpHGIAvLyOwOobtM/8d+TQkAd81/bEjgPHrfba41Vpesr7jX/t8Uh+R3EX9eAS5be+jQYg==",
1252 | "dev": true,
1253 | "requires": {
1254 | "semver": "^6.3.0"
1255 | },
1256 | "dependencies": {
1257 | "semver": {
1258 | "version": "6.3.0",
1259 | "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz",
1260 | "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==",
1261 | "dev": true
1262 | }
1263 | }
1264 | },
1265 | "send": {
1266 | "version": "0.17.1",
1267 | "resolved": "https://registry.npmjs.org/send/-/send-0.17.1.tgz",
1268 | "integrity": "sha512-BsVKsiGcQMFwT8UxypobUKyv7irCNRHk1T0G680vk88yf6LBByGcZJOTJCrTP2xVN6yI+XjPJcNuE3V4fT9sAg==",
1269 | "requires": {
1270 | "debug": "2.6.9",
1271 | "depd": "~1.1.2",
1272 | "destroy": "~1.0.4",
1273 | "encodeurl": "~1.0.2",
1274 | "escape-html": "~1.0.3",
1275 | "etag": "~1.8.1",
1276 | "fresh": "0.5.2",
1277 | "http-errors": "~1.7.2",
1278 | "mime": "1.6.0",
1279 | "ms": "2.1.1",
1280 | "on-finished": "~2.3.0",
1281 | "range-parser": "~1.2.1",
1282 | "statuses": "~1.5.0"
1283 | },
1284 | "dependencies": {
1285 | "debug": {
1286 | "version": "2.6.9",
1287 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz",
1288 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==",
1289 | "requires": {
1290 | "ms": "2.0.0"
1291 | },
1292 | "dependencies": {
1293 | "ms": {
1294 | "version": "2.0.0",
1295 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz",
1296 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g="
1297 | }
1298 | }
1299 | },
1300 | "ms": {
1301 | "version": "2.1.1",
1302 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.1.tgz",
1303 | "integrity": "sha512-tgp+dl5cGk28utYktBsrFqA7HKgrhgPsg6Z/EfhWI4gl1Hwq8B/GmY/0oXZ6nF8hDVesS/FpnYaD/kOWhYQvyg=="
1304 | }
1305 | }
1306 | },
1307 | "serve-static": {
1308 | "version": "1.14.1",
1309 | "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.14.1.tgz",
1310 | "integrity": "sha512-JMrvUwE54emCYWlTI+hGrGv5I8dEwmco/00EvkzIIsR7MqrHonbD9pO2MOfFnpFntl7ecpZs+3mW+XbQZu9QCg==",
1311 | "requires": {
1312 | "encodeurl": "~1.0.2",
1313 | "escape-html": "~1.0.3",
1314 | "parseurl": "~1.3.3",
1315 | "send": "0.17.1"
1316 | }
1317 | },
1318 | "setprototypeof": {
1319 | "version": "1.1.1",
1320 | "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.1.tgz",
1321 | "integrity": "sha512-JvdAWfbXeIGaZ9cILp38HntZSFSo3mWg6xGcJJsd+d4aRMOqauag1C63dJfDw7OaMYwEbHMOxEZ1lqVRYP2OAw=="
1322 | },
1323 | "signal-exit": {
1324 | "version": "3.0.3",
1325 | "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.3.tgz",
1326 | "integrity": "sha512-VUJ49FC8U1OxwZLxIbTTrDvLnf/6TDgxZcK8wxR8zs13xpx7xbG60ndBlhNrFi2EMuFRoeDoJO7wthSLq42EjA==",
1327 | "dev": true
1328 | },
1329 | "socket.io": {
1330 | "version": "4.0.2",
1331 | "resolved": "https://registry.npmjs.org/socket.io/-/socket.io-4.0.2.tgz",
1332 | "integrity": "sha512-1sVnsumrYTshhH2LWUtmzNvuZqkEOAG5PxcHpKu1r/sIDhzACTdo9XbvonVB2M90Y2CkTZnCWT6hlaKEkdBu5Q==",
1333 | "requires": {
1334 | "@types/cookie": "^0.4.0",
1335 | "@types/cors": "^2.8.8",
1336 | "@types/node": ">=10.0.0",
1337 | "accepts": "~1.3.4",
1338 | "base64id": "~2.0.0",
1339 | "debug": "~4.3.1",
1340 | "engine.io": "~5.0.0",
1341 | "socket.io-adapter": "~2.2.0",
1342 | "socket.io-parser": "~4.0.3"
1343 | }
1344 | },
1345 | "socket.io-adapter": {
1346 | "version": "2.2.0",
1347 | "resolved": "https://registry.npmjs.org/socket.io-adapter/-/socket.io-adapter-2.2.0.tgz",
1348 | "integrity": "sha512-rG49L+FwaVEwuAdeBRq49M97YI3ElVabJPzvHT9S6a2CWhDKnjSFasvwAwSYPRhQzfn4NtDIbCaGYgOCOU/rlg=="
1349 | },
1350 | "socket.io-parser": {
1351 | "version": "4.0.4",
1352 | "resolved": "https://registry.npmjs.org/socket.io-parser/-/socket.io-parser-4.0.4.tgz",
1353 | "integrity": "sha512-t+b0SS+IxG7Rxzda2EVvyBZbvFPBCjJoyHuE0P//7OAsN23GItzDRdWa6ALxZI/8R5ygK7jAR6t028/z+7295g==",
1354 | "requires": {
1355 | "@types/component-emitter": "^1.2.10",
1356 | "component-emitter": "~1.3.0",
1357 | "debug": "~4.3.1"
1358 | }
1359 | },
1360 | "standard-as-callback": {
1361 | "version": "2.1.0",
1362 | "resolved": "https://registry.npmjs.org/standard-as-callback/-/standard-as-callback-2.1.0.tgz",
1363 | "integrity": "sha512-qoRRSyROncaz1z0mvYqIE4lCd9p2R90i6GxW3uZv5ucSu8tU7B5HXUP1gG8pVZsYNVaXjk8ClXHPttLyxAL48A=="
1364 | },
1365 | "statuses": {
1366 | "version": "1.5.0",
1367 | "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.5.0.tgz",
1368 | "integrity": "sha1-Fhx9rBd2Wf2YEfQ3cfqZOBR4Yow="
1369 | },
1370 | "string-width": {
1371 | "version": "4.2.2",
1372 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.2.tgz",
1373 | "integrity": "sha512-XBJbT3N4JhVumXE0eoLU9DCjcaF92KLNqTmFCnG1pf8duUxFGwtP6AD6nkjw9a3IdiRtL3E2w3JDiE/xi3vOeA==",
1374 | "dev": true,
1375 | "requires": {
1376 | "emoji-regex": "^8.0.0",
1377 | "is-fullwidth-code-point": "^3.0.0",
1378 | "strip-ansi": "^6.0.0"
1379 | },
1380 | "dependencies": {
1381 | "ansi-regex": {
1382 | "version": "5.0.0",
1383 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.0.tgz",
1384 | "integrity": "sha512-bY6fj56OUQ0hU1KjFNDQuJFezqKdrAyFdIevADiqrWHwSlbmBNMHp5ak2f40Pm8JTFyM2mqxkG6ngkHO11f/lg==",
1385 | "dev": true
1386 | },
1387 | "emoji-regex": {
1388 | "version": "8.0.0",
1389 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz",
1390 | "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==",
1391 | "dev": true
1392 | },
1393 | "is-fullwidth-code-point": {
1394 | "version": "3.0.0",
1395 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz",
1396 | "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==",
1397 | "dev": true
1398 | },
1399 | "strip-ansi": {
1400 | "version": "6.0.0",
1401 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.0.tgz",
1402 | "integrity": "sha512-AuvKTrTfQNYNIctbR1K/YGTR1756GycPsg7b9bdV9Duqur4gv6aKqHXah67Z8ImS7WEz5QVcOtlfW2rZEugt6w==",
1403 | "dev": true,
1404 | "requires": {
1405 | "ansi-regex": "^5.0.0"
1406 | }
1407 | }
1408 | }
1409 | },
1410 | "strip-ansi": {
1411 | "version": "5.2.0",
1412 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz",
1413 | "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==",
1414 | "dev": true,
1415 | "requires": {
1416 | "ansi-regex": "^4.1.0"
1417 | }
1418 | },
1419 | "strip-json-comments": {
1420 | "version": "2.0.1",
1421 | "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-2.0.1.tgz",
1422 | "integrity": "sha1-PFMZQukIwml8DsNEhYwobHygpgo=",
1423 | "dev": true
1424 | },
1425 | "supports-color": {
1426 | "version": "5.5.0",
1427 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz",
1428 | "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==",
1429 | "dev": true,
1430 | "requires": {
1431 | "has-flag": "^3.0.0"
1432 | }
1433 | },
1434 | "term-size": {
1435 | "version": "2.2.1",
1436 | "resolved": "https://registry.npmjs.org/term-size/-/term-size-2.2.1.tgz",
1437 | "integrity": "sha512-wK0Ri4fOGjv/XPy8SBHZChl8CM7uMc5VML7SqiQ0zG7+J5Vr+RMQDoHa2CNT6KHUnTGIXH34UDMkPzAUyapBZg==",
1438 | "dev": true
1439 | },
1440 | "to-readable-stream": {
1441 | "version": "1.0.0",
1442 | "resolved": "https://registry.npmjs.org/to-readable-stream/-/to-readable-stream-1.0.0.tgz",
1443 | "integrity": "sha512-Iq25XBt6zD5npPhlLVXGFN3/gyR2/qODcKNNyTMd4vbm39HUaOiAM4PMq0eMVC/Tkxz+Zjdsc55g9yyz+Yq00Q==",
1444 | "dev": true
1445 | },
1446 | "to-regex-range": {
1447 | "version": "5.0.1",
1448 | "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz",
1449 | "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==",
1450 | "dev": true,
1451 | "requires": {
1452 | "is-number": "^7.0.0"
1453 | }
1454 | },
1455 | "toidentifier": {
1456 | "version": "1.0.0",
1457 | "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.0.tgz",
1458 | "integrity": "sha512-yaOH/Pk/VEhBWWTlhI+qXxDFXlejDGcQipMlyxda9nthulaxLZUNcUqFxokp0vcYnvteJln5FNQDRrxj3YcbVw=="
1459 | },
1460 | "touch": {
1461 | "version": "3.1.0",
1462 | "resolved": "https://registry.npmjs.org/touch/-/touch-3.1.0.tgz",
1463 | "integrity": "sha512-WBx8Uy5TLtOSRtIq+M03/sKDrXCLHxwDcquSP2c43Le03/9serjQBIztjRz6FkJez9D/hleyAXTBGLwwZUw9lA==",
1464 | "dev": true,
1465 | "requires": {
1466 | "nopt": "~1.0.10"
1467 | }
1468 | },
1469 | "type-fest": {
1470 | "version": "0.8.1",
1471 | "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.8.1.tgz",
1472 | "integrity": "sha512-4dbzIzqvjtgiM5rw1k5rEHtBANKmdudhGyBEajN01fEyhaAIhsoKNy6y7+IN93IfpFtwY9iqi7kD+xwKhQsNJA==",
1473 | "dev": true
1474 | },
1475 | "type-is": {
1476 | "version": "1.6.18",
1477 | "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz",
1478 | "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==",
1479 | "requires": {
1480 | "media-typer": "0.3.0",
1481 | "mime-types": "~2.1.24"
1482 | }
1483 | },
1484 | "typedarray-to-buffer": {
1485 | "version": "3.1.5",
1486 | "resolved": "https://registry.npmjs.org/typedarray-to-buffer/-/typedarray-to-buffer-3.1.5.tgz",
1487 | "integrity": "sha512-zdu8XMNEDepKKR+XYOXAVPtWui0ly0NtohUscw+UmaHiAWT8hrV1rr//H6V+0DvJ3OQ19S979M0laLfX8rm82Q==",
1488 | "dev": true,
1489 | "requires": {
1490 | "is-typedarray": "^1.0.0"
1491 | }
1492 | },
1493 | "undefsafe": {
1494 | "version": "2.0.3",
1495 | "resolved": "https://registry.npmjs.org/undefsafe/-/undefsafe-2.0.3.tgz",
1496 | "integrity": "sha512-nrXZwwXrD/T/JXeygJqdCO6NZZ1L66HrxM/Z7mIq2oPanoN0F1nLx3lwJMu6AwJY69hdixaFQOuoYsMjE5/C2A==",
1497 | "dev": true,
1498 | "requires": {
1499 | "debug": "^2.2.0"
1500 | },
1501 | "dependencies": {
1502 | "debug": {
1503 | "version": "2.6.9",
1504 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz",
1505 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==",
1506 | "dev": true,
1507 | "requires": {
1508 | "ms": "2.0.0"
1509 | }
1510 | },
1511 | "ms": {
1512 | "version": "2.0.0",
1513 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz",
1514 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=",
1515 | "dev": true
1516 | }
1517 | }
1518 | },
1519 | "unique-string": {
1520 | "version": "2.0.0",
1521 | "resolved": "https://registry.npmjs.org/unique-string/-/unique-string-2.0.0.tgz",
1522 | "integrity": "sha512-uNaeirEPvpZWSgzwsPGtU2zVSTrn/8L5q/IexZmH0eH6SA73CmAA5U4GwORTxQAZs95TAXLNqeLoPPNO5gZfWg==",
1523 | "dev": true,
1524 | "requires": {
1525 | "crypto-random-string": "^2.0.0"
1526 | }
1527 | },
1528 | "unpipe": {
1529 | "version": "1.0.0",
1530 | "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz",
1531 | "integrity": "sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw="
1532 | },
1533 | "update-notifier": {
1534 | "version": "4.1.3",
1535 | "resolved": "https://registry.npmjs.org/update-notifier/-/update-notifier-4.1.3.tgz",
1536 | "integrity": "sha512-Yld6Z0RyCYGB6ckIjffGOSOmHXj1gMeE7aROz4MG+XMkmixBX4jUngrGXNYz7wPKBmtoD4MnBa2Anu7RSKht/A==",
1537 | "dev": true,
1538 | "requires": {
1539 | "boxen": "^4.2.0",
1540 | "chalk": "^3.0.0",
1541 | "configstore": "^5.0.1",
1542 | "has-yarn": "^2.1.0",
1543 | "import-lazy": "^2.1.0",
1544 | "is-ci": "^2.0.0",
1545 | "is-installed-globally": "^0.3.1",
1546 | "is-npm": "^4.0.0",
1547 | "is-yarn-global": "^0.3.0",
1548 | "latest-version": "^5.0.0",
1549 | "pupa": "^2.0.1",
1550 | "semver-diff": "^3.1.1",
1551 | "xdg-basedir": "^4.0.0"
1552 | }
1553 | },
1554 | "url-parse-lax": {
1555 | "version": "3.0.0",
1556 | "resolved": "https://registry.npmjs.org/url-parse-lax/-/url-parse-lax-3.0.0.tgz",
1557 | "integrity": "sha1-FrXK/Afb42dsGxmZF3gj1lA6yww=",
1558 | "dev": true,
1559 | "requires": {
1560 | "prepend-http": "^2.0.0"
1561 | }
1562 | },
1563 | "utils-merge": {
1564 | "version": "1.0.1",
1565 | "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz",
1566 | "integrity": "sha1-n5VxD1CiZ5R7LMwSR0HBAoQn5xM="
1567 | },
1568 | "vary": {
1569 | "version": "1.1.2",
1570 | "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz",
1571 | "integrity": "sha1-IpnwLG3tMNSllhsLn3RSShj2NPw="
1572 | },
1573 | "widest-line": {
1574 | "version": "3.1.0",
1575 | "resolved": "https://registry.npmjs.org/widest-line/-/widest-line-3.1.0.tgz",
1576 | "integrity": "sha512-NsmoXalsWVDMGupxZ5R08ka9flZjjiLvHVAWYOKtiKM8ujtZWr9cRffak+uSE48+Ob8ObalXpwyeUiyDD6QFgg==",
1577 | "dev": true,
1578 | "requires": {
1579 | "string-width": "^4.0.0"
1580 | }
1581 | },
1582 | "wrappy": {
1583 | "version": "1.0.2",
1584 | "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz",
1585 | "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=",
1586 | "dev": true
1587 | },
1588 | "write-file-atomic": {
1589 | "version": "3.0.3",
1590 | "resolved": "https://registry.npmjs.org/write-file-atomic/-/write-file-atomic-3.0.3.tgz",
1591 | "integrity": "sha512-AvHcyZ5JnSfq3ioSyjrBkH9yW4m7Ayk8/9My/DD9onKeu/94fwrMocemO2QAJFAlnnDN+ZDS+ZjAR5ua1/PV/Q==",
1592 | "dev": true,
1593 | "requires": {
1594 | "imurmurhash": "^0.1.4",
1595 | "is-typedarray": "^1.0.0",
1596 | "signal-exit": "^3.0.2",
1597 | "typedarray-to-buffer": "^3.1.5"
1598 | }
1599 | },
1600 | "ws": {
1601 | "version": "7.4.5",
1602 | "resolved": "https://registry.npmjs.org/ws/-/ws-7.4.5.tgz",
1603 | "integrity": "sha512-xzyu3hFvomRfXKH8vOFMU3OguG6oOvhXMo3xsGy3xWExqaM2dxBbVxuD99O7m3ZUFMvvscsZDqxfgMaRr/Nr1g=="
1604 | },
1605 | "xdg-basedir": {
1606 | "version": "4.0.0",
1607 | "resolved": "https://registry.npmjs.org/xdg-basedir/-/xdg-basedir-4.0.0.tgz",
1608 | "integrity": "sha512-PSNhEJDejZYV7h50BohL09Er9VaIefr2LMAf3OEmpCkjOi34eYyQYAXUTjEQtZJTKcF0E2UKTh+osDLsgNim9Q==",
1609 | "dev": true
1610 | }
1611 | }
1612 | }
1613 |
--------------------------------------------------------------------------------