├── client ├── README.md ├── src │ ├── styles │ │ ├── variables.scss │ │ ├── app.scss │ │ ├── canvas.scss │ │ └── toolbar.scss │ ├── assets │ │ └── img │ │ │ ├── line.png │ │ │ ├── rect.png │ │ │ ├── redo.png │ │ │ ├── save.png │ │ │ ├── undo.png │ │ │ ├── brush.png │ │ │ ├── circle.png │ │ │ └── eraser.png │ ├── index.jsx │ ├── tools │ │ ├── Eraser.js │ │ ├── Tool.js │ │ ├── Brush.js │ │ ├── Line.js │ │ ├── Circle.js │ │ └── Rect.js │ ├── store │ │ ├── toolState.js │ │ └── canvasState.js │ ├── components │ │ ├── SettingBar.jsx │ │ ├── Toolbar.jsx │ │ └── Canvas.jsx │ └── App.jsx ├── .idea │ ├── codeStyles │ │ └── codeStyleConfig.xml │ ├── vcs.xml │ ├── modules.xml │ ├── paint-online-course.iml │ └── workspace.xml ├── public │ └── index.html └── package.json ├── server ├── files │ └── f1768a93c14e.jpg ├── package.json ├── index.js └── package-lock.json ├── README.md └── .gitignore /client/README.md: -------------------------------------------------------------------------------- 1 | ### Для установки модулей npm install 2 | ### Для запуска npm start 3 | -------------------------------------------------------------------------------- /client/src/styles/variables.scss: -------------------------------------------------------------------------------- 1 | $background-color: #F6F6F6; 2 | $toolbar-height:40px; 3 | -------------------------------------------------------------------------------- /client/src/assets/img/line.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/utimur/Fullstack-websocket-paint/HEAD/client/src/assets/img/line.png -------------------------------------------------------------------------------- /client/src/assets/img/rect.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/utimur/Fullstack-websocket-paint/HEAD/client/src/assets/img/rect.png -------------------------------------------------------------------------------- /client/src/assets/img/redo.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/utimur/Fullstack-websocket-paint/HEAD/client/src/assets/img/redo.png -------------------------------------------------------------------------------- /client/src/assets/img/save.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/utimur/Fullstack-websocket-paint/HEAD/client/src/assets/img/save.png -------------------------------------------------------------------------------- /client/src/assets/img/undo.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/utimur/Fullstack-websocket-paint/HEAD/client/src/assets/img/undo.png -------------------------------------------------------------------------------- /server/files/f1768a93c14e.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/utimur/Fullstack-websocket-paint/HEAD/server/files/f1768a93c14e.jpg -------------------------------------------------------------------------------- /client/src/assets/img/brush.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/utimur/Fullstack-websocket-paint/HEAD/client/src/assets/img/brush.png -------------------------------------------------------------------------------- /client/src/assets/img/circle.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/utimur/Fullstack-websocket-paint/HEAD/client/src/assets/img/circle.png -------------------------------------------------------------------------------- /client/src/assets/img/eraser.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/utimur/Fullstack-websocket-paint/HEAD/client/src/assets/img/eraser.png -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # Скачал репозиторий - не забудь поставить звезду 2 | 3 | ## для запуска клиента NPM START 4 | ## для запуска сервера NPM RUN DEV 5 | -------------------------------------------------------------------------------- /client/.idea/codeStyles/codeStyleConfig.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 5 | -------------------------------------------------------------------------------- /client/.idea/vcs.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | -------------------------------------------------------------------------------- /client/src/index.jsx: -------------------------------------------------------------------------------- 1 | import React from 'react'; 2 | import ReactDOM from 'react-dom'; 3 | import App from './App'; 4 | 5 | ReactDOM.render( 6 | , 7 | document.getElementById('root') 8 | ); 9 | 10 | -------------------------------------------------------------------------------- /client/src/styles/app.scss: -------------------------------------------------------------------------------- 1 | @import "./variables"; 2 | 3 | * { 4 | margin: 0; 5 | padding: 0; 6 | box-sizing: border-box; 7 | } 8 | 9 | .app { 10 | height: 100vh; 11 | max-height: 100vh; 12 | width: 100vw; 13 | background-color: $background-color; 14 | } 15 | -------------------------------------------------------------------------------- /client/src/styles/canvas.scss: -------------------------------------------------------------------------------- 1 | @import "./variables"; 2 | 3 | 4 | .canvas { 5 | height: 100vh; 6 | display: flex; 7 | justify-content: center; 8 | align-items: center; 9 | } 10 | 11 | canvas { 12 | border: 1px solid black; 13 | background-color: white; 14 | } 15 | -------------------------------------------------------------------------------- /client/.idea/modules.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | -------------------------------------------------------------------------------- /client/src/tools/Eraser.js: -------------------------------------------------------------------------------- 1 | import Tool from "./Tool"; 2 | import Brush from "./Brush"; 3 | 4 | export default class Eraser extends Brush { 5 | constructor(canvas) { 6 | super(canvas); 7 | } 8 | 9 | 10 | draw(x, y) { 11 | this.ctx.strokeStyle = "white" 12 | this.ctx.lineTo(x, y) 13 | this.ctx.stroke() 14 | } 15 | } 16 | -------------------------------------------------------------------------------- /server/package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "server", 3 | "version": "1.0.0", 4 | "description": "", 5 | "main": "index.js", 6 | "scripts": { 7 | "dev": "nodemon index.js" 8 | }, 9 | "keywords": [], 10 | "author": "", 11 | "license": "ISC", 12 | "dependencies": { 13 | "cors": "^2.8.5", 14 | "express": "^4.17.1", 15 | "express-ws": "^4.0.0" 16 | }, 17 | "devDependencies": { 18 | "nodemon": "^2.0.6" 19 | } 20 | } 21 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | # See https://help.github.com/articles/ignoring-files/ for more about ignoring files. 2 | 3 | # dependencies/ 4 | /client/node_modules 5 | /client/.pnp 6 | /client/.pnp.js 7 | 8 | # testing 9 | /coverage 10 | 11 | # production 12 | /client/build 13 | 14 | # misc 15 | .DS_Store 16 | .env.local 17 | .env.development.local 18 | .env.test.local 19 | .env.production.local 20 | 21 | npm-debug.log* 22 | yarn-debug.log* 23 | yarn-error.log* 24 | 25 | /server/node_modules 26 | /.idea 27 | -------------------------------------------------------------------------------- /client/.idea/paint-online-course.iml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | -------------------------------------------------------------------------------- /client/src/store/toolState.js: -------------------------------------------------------------------------------- 1 | import {makeAutoObservable} from "mobx"; 2 | 3 | class ToolState { 4 | tool = null 5 | constructor() { 6 | makeAutoObservable(this) 7 | } 8 | 9 | setTool(tool) { 10 | this.tool = tool 11 | } 12 | setFillColor(color) { 13 | this.tool.fillColor = color 14 | } 15 | setStrokeColor(color) { 16 | this.tool.strokeColor = color 17 | } 18 | setLineWidth(width) { 19 | this.tool.lineWidth = width 20 | } 21 | } 22 | 23 | export default new ToolState() 24 | -------------------------------------------------------------------------------- /client/public/index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | React App 7 | 13 | 14 | 15 | 16 |
17 | 18 | 19 | -------------------------------------------------------------------------------- /client/src/tools/Tool.js: -------------------------------------------------------------------------------- 1 | export default class Tool { 2 | constructor(canvas, socket, id) { 3 | this.canvas = canvas 4 | this.socket = socket 5 | this.id = id 6 | this.ctx = canvas.getContext('2d') 7 | this.destroyEvents() 8 | } 9 | 10 | set fillColor(color) { 11 | this.ctx.fillStyle = color 12 | } 13 | set strokeColor(color) { 14 | this.ctx.strokeStyle = color 15 | } 16 | 17 | set lineWidth(width) { 18 | this.ctx.lineWidth = width 19 | } 20 | 21 | destroyEvents() { 22 | this.canvas.onmousemove = null 23 | this.canvas.onmousedown = null 24 | this.canvas.onmouseup = null 25 | } 26 | } 27 | -------------------------------------------------------------------------------- /client/src/components/SettingBar.jsx: -------------------------------------------------------------------------------- 1 | import React from 'react'; 2 | import toolState from "../store/toolState"; 3 | 4 | const SettingBar = () => { 5 | return ( 6 |
7 | 8 | toolState.setLineWidth(e.target.value)} 10 | style={{margin: '0 10px'}} 11 | id="line-width" 12 | type="number" defaultValue={1} min={1} max={50}/> 13 | 14 | toolState.setStrokeColor(e.target.value)} id="stroke-color" type="color"/> 15 |
16 | ); 17 | }; 18 | 19 | export default SettingBar; 20 | -------------------------------------------------------------------------------- /client/src/App.jsx: -------------------------------------------------------------------------------- 1 | import React from 'react'; 2 | import "./styles/app.scss" 3 | import SettingBar from "./components/SettingBar"; 4 | import Toolbar from "./components/Toolbar"; 5 | import Canvas from "./components/Canvas"; 6 | import {BrowserRouter, Switch, Route, Redirect} from 'react-router-dom' 7 | 8 | const App = () => { 9 | return ( 10 | 11 |
12 | 13 | 14 | 15 | 16 | 17 | 18 | 19 | 20 |
21 |
22 | ); 23 | }; 24 | 25 | export default App; 26 | -------------------------------------------------------------------------------- /client/package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "paint-online-course", 3 | "version": "0.1.0", 4 | "private": true, 5 | "dependencies": { 6 | "@testing-library/jest-dom": "^5.11.6", 7 | "@testing-library/react": "^11.1.2", 8 | "@testing-library/user-event": "^12.2.2", 9 | "axios": "^0.21.1", 10 | "bootstrap": "^4.5.3", 11 | "mobx": "^6.0.4", 12 | "mobx-react-lite": "^3.1.6", 13 | "node-sass": "^4.14.1", 14 | "react": "^17.0.1", 15 | "react-bootstrap": "^1.4.0", 16 | "react-dom": "^17.0.1", 17 | "react-scripts": "4.0.0", 18 | "web-vitals": "^0.2.4" 19 | }, 20 | "scripts": { 21 | "start": "react-scripts start", 22 | "build": "react-scripts build", 23 | "test": "react-scripts test", 24 | "eject": "react-scripts eject" 25 | }, 26 | "eslintConfig": { 27 | "extends": [ 28 | "react-app", 29 | "react-app/jest" 30 | ] 31 | }, 32 | "browserslist": { 33 | "production": [ 34 | ">0.2%", 35 | "not dead", 36 | "not op_mini all" 37 | ], 38 | "development": [ 39 | "last 1 chrome version", 40 | "last 1 firefox version", 41 | "last 1 safari version" 42 | ] 43 | } 44 | } 45 | -------------------------------------------------------------------------------- /client/src/tools/Brush.js: -------------------------------------------------------------------------------- 1 | import Tool from "./Tool"; 2 | 3 | export default class Brush extends Tool { 4 | constructor(canvas, socket, id) { 5 | super(canvas, socket, id); 6 | this.listen() 7 | } 8 | 9 | listen() { 10 | this.canvas.onmousemove = this.mouseMoveHandler.bind(this) 11 | this.canvas.onmousedown = this.mouseDownHandler.bind(this) 12 | this.canvas.onmouseup = this.mouseUpHandler.bind(this) 13 | } 14 | 15 | mouseUpHandler(e) { 16 | this.mouseDown = false 17 | this.socket.send(JSON.stringify({ 18 | method: 'draw', 19 | id: this.id, 20 | figure: { 21 | type: 'finish', 22 | } 23 | })) 24 | } 25 | mouseDownHandler(e) { 26 | this.mouseDown = true 27 | this.ctx.beginPath() 28 | this.ctx.moveTo(e.pageX - e.target.offsetLeft, e.pageY - e.target.offsetTop) 29 | } 30 | mouseMoveHandler(e) { 31 | if (this.mouseDown) { 32 | // this.draw(e.pageX - e.target.offsetLeft, e.pageY - e.target.offsetTop) 33 | this.socket.send(JSON.stringify({ 34 | method: 'draw', 35 | id: this.id, 36 | figure: { 37 | type: 'brush', 38 | x: e.pageX - e.target.offsetLeft, 39 | y: e.pageY - e.target.offsetTop 40 | } 41 | })) 42 | } 43 | } 44 | 45 | static draw(ctx, x, y) { 46 | ctx.lineTo(x, y) 47 | ctx.stroke() 48 | } 49 | } 50 | -------------------------------------------------------------------------------- /client/src/tools/Line.js: -------------------------------------------------------------------------------- 1 | import Tool from "./Tool"; 2 | 3 | 4 | export default class Line extends Tool { 5 | constructor(canvas) { 6 | super(canvas); 7 | this.listen() 8 | this.name = 'Line' 9 | } 10 | 11 | listen() { 12 | this.canvas.onmousedown = this.mouseDownHandler.bind(this) 13 | this.canvas.onmouseup = this.mouseUpHandler.bind(this) 14 | this.canvas.onmousemove = this.mouseMoveHandler.bind(this) 15 | } 16 | 17 | mouseDownHandler(e) { 18 | this.mouseDown = true 19 | this.currentX = e.pageX-e.target.offsetLeft 20 | this.currentY = e.pageY-e.target.offsetTop 21 | this.ctx.beginPath() 22 | this.ctx.moveTo(this.currentX, this.currentY ) 23 | this.saved = this.canvas.toDataURL() 24 | } 25 | 26 | mouseUpHandler(e) { 27 | this.mouseDown = false 28 | } 29 | 30 | mouseMoveHandler(e) { 31 | if (this.mouseDown) { 32 | this.draw(e.pageX-e.target.offsetLeft, e.pageY-e.target.offsetTop); 33 | } 34 | } 35 | 36 | 37 | draw(x,y) { 38 | const img = new Image() 39 | img.src = this.saved 40 | img.onload = async function () { 41 | this.ctx.clearRect(0,0, this.canvas.width, this.canvas.height) 42 | this.ctx.drawImage(img, 0, 0, this.canvas.width, this.canvas.height) 43 | this.ctx.beginPath() 44 | this.ctx.moveTo(this.currentX, this.currentY ) 45 | this.ctx.lineTo(x, y) 46 | this.ctx.stroke() 47 | }.bind(this) 48 | 49 | } 50 | } 51 | -------------------------------------------------------------------------------- /client/src/styles/toolbar.scss: -------------------------------------------------------------------------------- 1 | @import "./variables"; 2 | 3 | .toolbar { 4 | height: $toolbar-height; 5 | background-color: white; 6 | display: flex; 7 | align-items: center; 8 | position: absolute; 9 | width: 100%; 10 | box-shadow: 0 4px 5px gray; 11 | z-index: 999; 12 | 13 | &__btn { 14 | height: 25px; 15 | width: 25px; 16 | border: none; 17 | outline: none; 18 | cursor: pointer; 19 | background-size: cover; 20 | margin-left: 10px; 21 | } 22 | .brush { 23 | background: url('../assets/img/brush.png') no-repeat center center; 24 | } 25 | .rect { 26 | background: url('../assets/img/rect.png') no-repeat center center; 27 | } 28 | .circle { 29 | background: url('../assets/img/circle.png') no-repeat center center; 30 | } 31 | .eraser { 32 | background: url('../assets/img/eraser.png') no-repeat center center; 33 | } 34 | .line { 35 | background: url('../assets/img/line.png') no-repeat center center; 36 | } 37 | .undo { 38 | margin-left: auto; 39 | background: url('../assets/img/undo.png') no-repeat center center; 40 | } 41 | .redo { 42 | background: url('../assets/img/redo.png') no-repeat center center; 43 | } 44 | .save { 45 | background: url('../assets/img/save.png') no-repeat center center; 46 | margin-right: 10px; 47 | } 48 | } 49 | 50 | 51 | .setting-bar { 52 | height: $toolbar-height; 53 | background-color: white; 54 | display: flex; 55 | align-items: center; 56 | position: absolute; 57 | width: 100%; 58 | box-shadow: 0 4px 5px gray; 59 | top:$toolbar-height; 60 | } 61 | -------------------------------------------------------------------------------- /client/src/tools/Circle.js: -------------------------------------------------------------------------------- 1 | import Tool from "./Tool"; 2 | 3 | 4 | export default class Circle extends Tool { 5 | constructor(canvas) { 6 | super(canvas); 7 | this.listen() 8 | } 9 | 10 | listen() { 11 | this.canvas.onmousemove = this.mouseMoveHandler.bind(this) 12 | this.canvas.onmousedown = this.mouseDownHandler.bind(this) 13 | this.canvas.onmouseup = this.mouseUpHandler.bind(this) 14 | } 15 | 16 | mouseDownHandler(e) { 17 | this.mouseDown = true 18 | let canvasData = this.canvas.toDataURL() 19 | this.ctx.beginPath() 20 | this.startX = e.pageX-e.target.offsetLeft 21 | this.startY = e.pageY-e.target.offsetTop 22 | this.saved = canvasData 23 | } 24 | 25 | mouseUpHandler(e) { 26 | this.mouseDown = false 27 | } 28 | 29 | mouseMoveHandler(e) { 30 | if(this.mouseDown) { 31 | let curentX = e.pageX-e.target.offsetLeft 32 | let curentY = e.pageY-e.target.offsetTop 33 | let width = curentX-this.startX 34 | let height = curentY-this.startY 35 | let r = Math.sqrt(width**2 + height**2) 36 | this.draw(this.startX, this.startY, r) 37 | } 38 | } 39 | 40 | draw(x,y,r) { 41 | const img = new Image() 42 | img.src = this.saved 43 | img.onload = async function () { 44 | this.ctx.clearRect(0,0, this.canvas.width, this.canvas.height) 45 | this.ctx.drawImage(img, 0, 0, this.canvas.width, this.canvas.height) 46 | this.ctx.beginPath() 47 | this.ctx.arc(x, y, r, 0, 2*Math.PI) 48 | this.ctx.fill() 49 | this.ctx.stroke() 50 | }.bind(this) 51 | } 52 | } 53 | -------------------------------------------------------------------------------- /server/index.js: -------------------------------------------------------------------------------- 1 | const express = require('express') 2 | const app = express() 3 | const WSServer = require('express-ws')(app) 4 | const aWss = WSServer.getWss() 5 | const cors = require('cors') 6 | const PORT = process.env.PORT || 5000 7 | const fs = require('fs') 8 | const path = require('path') 9 | 10 | app.use(cors()) 11 | app.use(express.json()) 12 | 13 | app.ws('/', (ws, req) => { 14 | ws.on('message', (msg) => { 15 | msg = JSON.parse(msg) 16 | switch (msg.method) { 17 | case "connection": 18 | connectionHandler(ws, msg) 19 | break 20 | case "draw": 21 | broadcastConnection(ws, msg) 22 | break 23 | } 24 | }) 25 | }) 26 | 27 | app.post('/image', (req, res) => { 28 | try { 29 | const data = req.body.img.replace(`data:image/png;base64,`, '') 30 | fs.writeFileSync(path.resolve(__dirname, 'files', `${req.query.id}.jpg`), data, 'base64') 31 | return res.status(200).json({message: "Загружено"}) 32 | } catch (e) { 33 | console.log(e) 34 | return res.status(500).json('error') 35 | } 36 | }) 37 | app.get('/image', (req, res) => { 38 | try { 39 | const file = fs.readFileSync(path.resolve(__dirname, 'files', `${req.query.id}.jpg`)) 40 | const data = `data:image/png;base64,` + file.toString('base64') 41 | res.json(data) 42 | } catch (e) { 43 | console.log(e) 44 | return res.status(500).json('error') 45 | } 46 | }) 47 | 48 | app.listen(PORT, () => console.log(`server started on PORT ${PORT}`)) 49 | 50 | const connectionHandler = (ws, msg) => { 51 | ws.id = msg.id 52 | broadcastConnection(ws, msg) 53 | } 54 | 55 | const broadcastConnection = (ws, msg) => { 56 | aWss.clients.forEach(client => { 57 | if (client.id === msg.id) { 58 | client.send(JSON.stringify(msg)) 59 | } 60 | }) 61 | } 62 | -------------------------------------------------------------------------------- /client/src/store/canvasState.js: -------------------------------------------------------------------------------- 1 | import {makeAutoObservable} from "mobx"; 2 | 3 | class CanvasState { 4 | canvas = null 5 | socket = null 6 | sessionid = null 7 | undoList = [] 8 | redoList = [] 9 | username = "" 10 | 11 | constructor() { 12 | makeAutoObservable(this) 13 | } 14 | 15 | setSessionId(id) { 16 | this.sessionid = id 17 | } 18 | setSocket(socket) { 19 | this.socket = socket 20 | } 21 | 22 | setUsername(username) { 23 | this.username = username 24 | } 25 | 26 | setCanvas(canvas) { 27 | this.canvas = canvas 28 | } 29 | 30 | pushToUndo(data) { 31 | this.undoList.push(data) 32 | } 33 | 34 | pushToRedo(data) { 35 | this.redoList.push(data) 36 | } 37 | 38 | undo() { 39 | let ctx = this.canvas.getContext('2d') 40 | if (this.undoList.length > 0) { 41 | let dataUrl = this.undoList.pop() 42 | this.redoList.push(this.canvas.toDataURL()) 43 | let img = new Image() 44 | img.src = dataUrl 45 | img.onload = () => { 46 | ctx.clearRect(0,0, this.canvas.width, this.canvas.height) 47 | ctx.drawImage(img, 0, 0, this.canvas.width, this.canvas.height) 48 | } 49 | } else { 50 | ctx.clearRect(0, 0, this.canvas.width, this.canvas.heigth) 51 | } 52 | } 53 | 54 | redo() { 55 | let ctx = this.canvas.getContext('2d') 56 | if (this.redoList.length > 0) { 57 | let dataUrl = this.redoList.pop() 58 | this.undoList.push(this.canvas.toDataURL()) 59 | let img = new Image() 60 | img.src = dataUrl 61 | img.onload = () => { 62 | ctx.clearRect(0,0, this.canvas.width, this.canvas.height) 63 | ctx.drawImage(img, 0, 0, this.canvas.width, this.canvas.height) 64 | } 65 | } 66 | } 67 | 68 | } 69 | 70 | export default new CanvasState() 71 | -------------------------------------------------------------------------------- /client/src/components/Toolbar.jsx: -------------------------------------------------------------------------------- 1 | import React from 'react'; 2 | import '../styles/toolbar.scss' 3 | import toolState from "../store/toolState"; 4 | import Brush from "../tools/Brush"; 5 | import canvasState from "../store/canvasState"; 6 | import Rect from "../tools/Rect"; 7 | import Line from "../tools/Line"; 8 | import Circle from "../tools/Circle"; 9 | import Eraser from "../tools/Eraser"; 10 | 11 | const Toolbar = () => { 12 | 13 | const changeColor = e => { 14 | toolState.setStrokeColor(e.target.value) 15 | toolState.setFillColor(e.target.value) 16 | } 17 | 18 | const download = () => { 19 | const dataUrl = canvasState.canvas.toDataURL() 20 | console.log(dataUrl) 21 | const a = document.createElement('a') 22 | a.href = dataUrl 23 | a.download = canvasState.sessionid + ".jpg" 24 | document.body.appendChild(a) 25 | a.click() 26 | document.body.removeChild(a) 27 | } 28 | 29 | return ( 30 |
31 |
41 | ); 42 | }; 43 | 44 | export default Toolbar; 45 | -------------------------------------------------------------------------------- /client/src/tools/Rect.js: -------------------------------------------------------------------------------- 1 | import Tool from "./Tool"; 2 | 3 | export default class Rect extends Tool { 4 | constructor(canvas, socket, id) { 5 | super(canvas, socket, id); 6 | this.listen() 7 | } 8 | 9 | listen() { 10 | this.canvas.onmousemove = this.mouseMoveHandler.bind(this) 11 | this.canvas.onmousedown = this.mouseDownHandler.bind(this) 12 | this.canvas.onmouseup = this.mouseUpHandler.bind(this) 13 | } 14 | 15 | mouseUpHandler(e) { 16 | this.mouseDown = false 17 | this.socket.send(JSON.stringify({ 18 | method: 'draw', 19 | id: this.id, 20 | figure: { 21 | type: 'rect', 22 | x: this.startX, 23 | y: this.startY, 24 | width: this.width, 25 | height: this.height, 26 | color: this.ctx.fillStyle 27 | } 28 | })) 29 | } 30 | mouseDownHandler(e) { 31 | this.mouseDown = true 32 | this.ctx.beginPath() 33 | this.startX = e.pageX - e.target.offsetLeft; 34 | this.startY = e.pageY - e.target.offsetTop; 35 | this.saved = this.canvas.toDataURL() 36 | } 37 | mouseMoveHandler(e) { 38 | if (this.mouseDown) { 39 | let currentX = e.pageX - e.target.offsetLeft; 40 | let currentY = e.pageY - e.target.offsetTop; 41 | this.width = currentX - this.startX; 42 | this.height = currentY - this.startY; 43 | this.draw(this.startX, this.startY, this.width, this.height) 44 | } 45 | } 46 | 47 | draw(x, y, w, h) { 48 | const img = new Image() 49 | img.src = this.saved 50 | img.onload = () => { 51 | this.ctx.clearRect(0, 0, this.canvas.width, this.canvas.height) 52 | this.ctx.drawImage(img, 0, 0, this.canvas.width, this.canvas.height) 53 | this.ctx.beginPath() 54 | this.ctx.rect(x, y, w, h) 55 | this.ctx.fill() 56 | this.ctx.stroke() 57 | } 58 | } 59 | 60 | static staticDraw(ctx, x, y, w, h, color) { 61 | ctx.fillStyle = color 62 | ctx.beginPath() 63 | ctx.rect(x, y, w, h) 64 | ctx.fill() 65 | ctx.stroke() 66 | } 67 | } 68 | -------------------------------------------------------------------------------- /client/.idea/workspace.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 12 | 13 | 15 | 16 | 17 | 20 | 21 | 22 | 23 | 24 | 25 | 26 | 27 | 28 | 29 | 30 | 31 | 32 | 33 | 34 | 35 | 36 | 1605536385150 37 | 46 | 47 | 48 | 49 | 51 | 52 | 53 | 54 | 55 | 56 | 57 | -------------------------------------------------------------------------------- /client/src/components/Canvas.jsx: -------------------------------------------------------------------------------- 1 | import React, {useEffect, useRef, useState} from 'react'; 2 | import "../styles/canvas.scss" 3 | import {observer} from "mobx-react-lite"; 4 | import canvasState from "../store/canvasState"; 5 | import toolState from "../store/toolState"; 6 | import Brush from "../tools/Brush"; 7 | import {Modal, Button} from "react-bootstrap"; 8 | import {useParams} from "react-router-dom" 9 | import Rect from "../tools/Rect"; 10 | import axios from 'axios' 11 | 12 | const Canvas = observer(() => { 13 | const canvasRef = useRef() 14 | const usernameRef = useRef() 15 | const [modal, setModal] = useState(true) 16 | const params = useParams() 17 | 18 | useEffect(() => { 19 | canvasState.setCanvas(canvasRef.current) 20 | let ctx = canvasRef.current.getContext('2d') 21 | axios.get(`http://localhost:5000/image?id=${params.id}`) 22 | .then(response => { 23 | const img = new Image() 24 | img.src = response.data 25 | img.onload = () => { 26 | ctx.clearRect(0, 0, canvasRef.current.width, canvasRef.current.height) 27 | ctx.drawImage(img, 0, 0, canvasRef.current.width, canvasRef.current.height) 28 | } 29 | }) 30 | }, []) 31 | 32 | useEffect(() => { 33 | if (canvasState.username) { 34 | const socket = new WebSocket(`ws://localhost:5000/`); 35 | canvasState.setSocket(socket) 36 | canvasState.setSessionId(params.id) 37 | toolState.setTool(new Brush(canvasRef.current, socket, params.id)) 38 | socket.onopen = () => { 39 | console.log('Подключение установлено') 40 | socket.send(JSON.stringify({ 41 | id:params.id, 42 | username: canvasState.username, 43 | method: "connection" 44 | })) 45 | } 46 | socket.onmessage = (event) => { 47 | let msg = JSON.parse(event.data) 48 | switch (msg.method) { 49 | case "connection": 50 | console.log(`пользователь ${msg.username} присоединился`) 51 | break 52 | case "draw": 53 | drawHandler(msg) 54 | break 55 | } 56 | } 57 | } 58 | }, [canvasState.username]) 59 | 60 | const drawHandler = (msg) => { 61 | const figure = msg.figure 62 | const ctx = canvasRef.current.getContext('2d') 63 | switch (figure.type) { 64 | case "brush": 65 | Brush.draw(ctx, figure.x, figure.y) 66 | break 67 | case "rect": 68 | Rect.staticDraw(ctx, figure.x, figure.y, figure.width, figure.height, figure.color) 69 | break 70 | case "finish": 71 | ctx.beginPath() 72 | break 73 | } 74 | } 75 | 76 | 77 | const mouseDownHandler = () => { 78 | canvasState.pushToUndo(canvasRef.current.toDataURL()) 79 | axios.post(`http://localhost:5000/image?id=${params.id}`, {img: canvasRef.current.toDataURL()}) 80 | .then(response => console.log(response.data)) 81 | } 82 | 83 | const connectHandler = () => { 84 | canvasState.setUsername(usernameRef.current.value) 85 | setModal(false) 86 | } 87 | 88 | return ( 89 |
90 | {}}> 91 | 92 | Введите ваше имя 93 | 94 | 95 | 96 | 97 | 98 | 101 | 102 | 103 | mouseDownHandler()} ref={canvasRef} width={600} height={400}/> 104 |
105 | ); 106 | }); 107 | 108 | export default Canvas; 109 | -------------------------------------------------------------------------------- /server/package-lock.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "server", 3 | "version": "1.0.0", 4 | "lockfileVersion": 1, 5 | "requires": true, 6 | "dependencies": { 7 | "@sindresorhus/is": { 8 | "version": "0.14.0", 9 | "resolved": "https://registry.npmjs.org/@sindresorhus/is/-/is-0.14.0.tgz", 10 | "integrity": "sha512-9NET910DNaIPngYnLLPeg+Ogzqsi9uM4mSboU5y6p8S5DzMTVEsJZrawi+BoDNUVBa2DhJqQYUFvMDfgU062LQ==", 11 | "dev": true 12 | }, 13 | "@szmarczak/http-timer": { 14 | "version": "1.1.2", 15 | "resolved": "https://registry.npmjs.org/@szmarczak/http-timer/-/http-timer-1.1.2.tgz", 16 | "integrity": "sha512-XIB2XbzHTN6ieIjfIMV9hlVcfPU26s2vafYWQcZHWXHOxiaRZYEDKEwdl129Zyg50+foYV2jCgtrqSA6qNuNSA==", 17 | "dev": true, 18 | "requires": { 19 | "defer-to-connect": "^1.0.1" 20 | } 21 | }, 22 | "abbrev": { 23 | "version": "1.1.1", 24 | "resolved": "https://registry.npmjs.org/abbrev/-/abbrev-1.1.1.tgz", 25 | "integrity": "sha512-nne9/IiQ/hzIhY6pdDnbBtz7DjPTKrY00P/zvPSm5pOFkl6xuGrGnXn/VtTNNfNtAfZ9/1RtehkszU9qcTii0Q==", 26 | "dev": true 27 | }, 28 | "accepts": { 29 | "version": "1.3.7", 30 | "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.7.tgz", 31 | "integrity": "sha512-Il80Qs2WjYlJIBNzNkK6KYqlVMTbZLXgHx2oT0pU/fjRHyEp+PEfEPY0R3WCwAGVOtauxh1hOxNgIf5bv7dQpA==", 32 | "requires": { 33 | "mime-types": "~2.1.24", 34 | "negotiator": "0.6.2" 35 | } 36 | }, 37 | "ansi-align": { 38 | "version": "3.0.0", 39 | "resolved": "https://registry.npmjs.org/ansi-align/-/ansi-align-3.0.0.tgz", 40 | "integrity": "sha512-ZpClVKqXN3RGBmKibdfWzqCY4lnjEuoNzU5T0oEFpfd/z5qJHVarukridD4juLO2FXMiwUQxr9WqQtaYa8XRYw==", 41 | "dev": true, 42 | "requires": { 43 | "string-width": "^3.0.0" 44 | }, 45 | "dependencies": { 46 | "string-width": { 47 | "version": "3.1.0", 48 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz", 49 | "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==", 50 | "dev": true, 51 | "requires": { 52 | "emoji-regex": "^7.0.1", 53 | "is-fullwidth-code-point": "^2.0.0", 54 | "strip-ansi": "^5.1.0" 55 | } 56 | } 57 | } 58 | }, 59 | "ansi-regex": { 60 | "version": "4.1.0", 61 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz", 62 | "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==", 63 | "dev": true 64 | }, 65 | "ansi-styles": { 66 | "version": "4.3.0", 67 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", 68 | "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", 69 | "dev": true, 70 | "requires": { 71 | "color-convert": "^2.0.1" 72 | } 73 | }, 74 | "anymatch": { 75 | "version": "3.1.1", 76 | "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.1.tgz", 77 | "integrity": "sha512-mM8522psRCqzV+6LhomX5wgp25YVibjh8Wj23I5RPkPppSVSjyKD2A2mBJmWGa+KN7f2D6LNh9jkBCeyLktzjg==", 78 | "dev": true, 79 | "requires": { 80 | "normalize-path": "^3.0.0", 81 | "picomatch": "^2.0.4" 82 | } 83 | }, 84 | "array-flatten": { 85 | "version": "1.1.1", 86 | "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz", 87 | "integrity": "sha1-ml9pkFGx5wczKPKgCJaLZOopVdI=" 88 | }, 89 | "async-limiter": { 90 | "version": "1.0.1", 91 | "resolved": "https://registry.npmjs.org/async-limiter/-/async-limiter-1.0.1.tgz", 92 | "integrity": "sha512-csOlWGAcRFJaI6m+F2WKdnMKr4HhdhFVBk0H/QbJFMCr+uO2kwohwXQPxw/9OCxp05r5ghVBFSyioixx3gfkNQ==" 93 | }, 94 | "balanced-match": { 95 | "version": "1.0.0", 96 | "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.0.tgz", 97 | "integrity": "sha1-ibTRmasr7kneFk6gK4nORi1xt2c=", 98 | "dev": true 99 | }, 100 | "binary-extensions": { 101 | "version": "2.1.0", 102 | "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.1.0.tgz", 103 | "integrity": "sha512-1Yj8h9Q+QDF5FzhMs/c9+6UntbD5MkRfRwac8DoEm9ZfUBZ7tZ55YcGVAzEe4bXsdQHEk+s9S5wsOKVdZrw0tQ==", 104 | "dev": true 105 | }, 106 | "body-parser": { 107 | "version": "1.19.0", 108 | "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.19.0.tgz", 109 | "integrity": "sha512-dhEPs72UPbDnAQJ9ZKMNTP6ptJaionhP5cBb541nXPlW60Jepo9RV/a4fX4XWW9CuFNK22krhrj1+rgzifNCsw==", 110 | "requires": { 111 | "bytes": "3.1.0", 112 | "content-type": "~1.0.4", 113 | "debug": "2.6.9", 114 | "depd": "~1.1.2", 115 | "http-errors": "1.7.2", 116 | "iconv-lite": "0.4.24", 117 | "on-finished": "~2.3.0", 118 | "qs": "6.7.0", 119 | "raw-body": "2.4.0", 120 | "type-is": "~1.6.17" 121 | } 122 | }, 123 | "boxen": { 124 | "version": "4.2.0", 125 | "resolved": "https://registry.npmjs.org/boxen/-/boxen-4.2.0.tgz", 126 | "integrity": "sha512-eB4uT9RGzg2odpER62bBwSLvUeGC+WbRjjyyFhGsKnc8wp/m0+hQsMUvUe3H2V0D5vw0nBdO1hCJoZo5mKeuIQ==", 127 | "dev": true, 128 | "requires": { 129 | "ansi-align": "^3.0.0", 130 | "camelcase": "^5.3.1", 131 | "chalk": "^3.0.0", 132 | "cli-boxes": "^2.2.0", 133 | "string-width": "^4.1.0", 134 | "term-size": "^2.1.0", 135 | "type-fest": "^0.8.1", 136 | "widest-line": "^3.1.0" 137 | } 138 | }, 139 | "brace-expansion": { 140 | "version": "1.1.11", 141 | "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", 142 | "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", 143 | "dev": true, 144 | "requires": { 145 | "balanced-match": "^1.0.0", 146 | "concat-map": "0.0.1" 147 | } 148 | }, 149 | "braces": { 150 | "version": "3.0.2", 151 | "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz", 152 | "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", 153 | "dev": true, 154 | "requires": { 155 | "fill-range": "^7.0.1" 156 | } 157 | }, 158 | "bytes": { 159 | "version": "3.1.0", 160 | "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.0.tgz", 161 | "integrity": "sha512-zauLjrfCG+xvoyaqLoV8bLVXXNGC4JqlxFCutSDWA6fJrTo2ZuvLYTqZ7aHBLZSMOopbzwv8f+wZcVzfVTI2Dg==" 162 | }, 163 | "cacheable-request": { 164 | "version": "6.1.0", 165 | "resolved": "https://registry.npmjs.org/cacheable-request/-/cacheable-request-6.1.0.tgz", 166 | "integrity": "sha512-Oj3cAGPCqOZX7Rz64Uny2GYAZNliQSqfbePrgAQ1wKAihYmCUnraBtJtKcGR4xz7wF+LoJC+ssFZvv5BgF9Igg==", 167 | "dev": true, 168 | "requires": { 169 | "clone-response": "^1.0.2", 170 | "get-stream": "^5.1.0", 171 | "http-cache-semantics": "^4.0.0", 172 | "keyv": "^3.0.0", 173 | "lowercase-keys": "^2.0.0", 174 | "normalize-url": "^4.1.0", 175 | "responselike": "^1.0.2" 176 | }, 177 | "dependencies": { 178 | "get-stream": { 179 | "version": "5.2.0", 180 | "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-5.2.0.tgz", 181 | "integrity": "sha512-nBF+F1rAZVCu/p7rjzgA+Yb4lfYXrpl7a6VmJrU8wF9I1CKvP/QwPNZHnOlwbTkY6dvtFIzFMSyQXbLoTQPRpA==", 182 | "dev": true, 183 | "requires": { 184 | "pump": "^3.0.0" 185 | } 186 | }, 187 | "lowercase-keys": { 188 | "version": "2.0.0", 189 | "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-2.0.0.tgz", 190 | "integrity": "sha512-tqNXrS78oMOE73NMxK4EMLQsQowWf8jKooH9g7xPavRT706R6bkQJ6DY2Te7QukaZsulxa30wQ7bk0pm4XiHmA==", 191 | "dev": true 192 | } 193 | } 194 | }, 195 | "camelcase": { 196 | "version": "5.3.1", 197 | "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz", 198 | "integrity": "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==", 199 | "dev": true 200 | }, 201 | "chalk": { 202 | "version": "3.0.0", 203 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-3.0.0.tgz", 204 | "integrity": "sha512-4D3B6Wf41KOYRFdszmDqMCGq5VV/uMAB273JILmO+3jAlh8X4qDtdtgCR3fxtbLEMzSx22QdhnDcJvu2u1fVwg==", 205 | "dev": true, 206 | "requires": { 207 | "ansi-styles": "^4.1.0", 208 | "supports-color": "^7.1.0" 209 | }, 210 | "dependencies": { 211 | "has-flag": { 212 | "version": "4.0.0", 213 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", 214 | "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", 215 | "dev": true 216 | }, 217 | "supports-color": { 218 | "version": "7.2.0", 219 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", 220 | "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", 221 | "dev": true, 222 | "requires": { 223 | "has-flag": "^4.0.0" 224 | } 225 | } 226 | } 227 | }, 228 | "chokidar": { 229 | "version": "3.4.3", 230 | "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.4.3.tgz", 231 | "integrity": "sha512-DtM3g7juCXQxFVSNPNByEC2+NImtBuxQQvWlHunpJIS5Ocr0lG306cC7FCi7cEA0fzmybPUIl4txBIobk1gGOQ==", 232 | "dev": true, 233 | "requires": { 234 | "anymatch": "~3.1.1", 235 | "braces": "~3.0.2", 236 | "fsevents": "~2.1.2", 237 | "glob-parent": "~5.1.0", 238 | "is-binary-path": "~2.1.0", 239 | "is-glob": "~4.0.1", 240 | "normalize-path": "~3.0.0", 241 | "readdirp": "~3.5.0" 242 | } 243 | }, 244 | "ci-info": { 245 | "version": "2.0.0", 246 | "resolved": "https://registry.npmjs.org/ci-info/-/ci-info-2.0.0.tgz", 247 | "integrity": "sha512-5tK7EtrZ0N+OLFMthtqOj4fI2Jeb88C4CAZPu25LDVUgXJ0A3Js4PMGqrn0JU1W0Mh1/Z8wZzYPxqUrXeBboCQ==", 248 | "dev": true 249 | }, 250 | "cli-boxes": { 251 | "version": "2.2.1", 252 | "resolved": "https://registry.npmjs.org/cli-boxes/-/cli-boxes-2.2.1.tgz", 253 | "integrity": "sha512-y4coMcylgSCdVinjiDBuR8PCC2bLjyGTwEmPb9NHR/QaNU6EUOXcTY/s6VjGMD6ENSEaeQYHCY0GNGS5jfMwPw==", 254 | "dev": true 255 | }, 256 | "clone-response": { 257 | "version": "1.0.2", 258 | "resolved": "https://registry.npmjs.org/clone-response/-/clone-response-1.0.2.tgz", 259 | "integrity": "sha1-0dyXOSAxTfZ/vrlCI7TuNQI56Ws=", 260 | "dev": true, 261 | "requires": { 262 | "mimic-response": "^1.0.0" 263 | } 264 | }, 265 | "color-convert": { 266 | "version": "2.0.1", 267 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", 268 | "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", 269 | "dev": true, 270 | "requires": { 271 | "color-name": "~1.1.4" 272 | } 273 | }, 274 | "color-name": { 275 | "version": "1.1.4", 276 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", 277 | "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", 278 | "dev": true 279 | }, 280 | "concat-map": { 281 | "version": "0.0.1", 282 | "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", 283 | "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=", 284 | "dev": true 285 | }, 286 | "configstore": { 287 | "version": "5.0.1", 288 | "resolved": "https://registry.npmjs.org/configstore/-/configstore-5.0.1.tgz", 289 | "integrity": "sha512-aMKprgk5YhBNyH25hj8wGt2+D52Sw1DRRIzqBwLp2Ya9mFmY8KPvvtvmna8SxVR9JMZ4kzMD68N22vlaRpkeFA==", 290 | "dev": true, 291 | "requires": { 292 | "dot-prop": "^5.2.0", 293 | "graceful-fs": "^4.1.2", 294 | "make-dir": "^3.0.0", 295 | "unique-string": "^2.0.0", 296 | "write-file-atomic": "^3.0.0", 297 | "xdg-basedir": "^4.0.0" 298 | } 299 | }, 300 | "content-disposition": { 301 | "version": "0.5.3", 302 | "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.3.tgz", 303 | "integrity": "sha512-ExO0774ikEObIAEV9kDo50o+79VCUdEB6n6lzKgGwupcVeRlhrj3qGAfwq8G6uBJjkqLrhT0qEYFcWng8z1z0g==", 304 | "requires": { 305 | "safe-buffer": "5.1.2" 306 | } 307 | }, 308 | "content-type": { 309 | "version": "1.0.4", 310 | "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.4.tgz", 311 | "integrity": "sha512-hIP3EEPs8tB9AT1L+NUqtwOAps4mk2Zob89MWXMHjHWg9milF/j4osnnQLXBCBFBk/tvIG/tUc9mOUJiPBhPXA==" 312 | }, 313 | "cookie": { 314 | "version": "0.4.0", 315 | "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.4.0.tgz", 316 | "integrity": "sha512-+Hp8fLp57wnUSt0tY0tHEXh4voZRDnoIrZPqlo3DPiI4y9lwg/jqx+1Om94/W6ZaPDOUbnjOt/99w66zk+l1Xg==" 317 | }, 318 | "cookie-signature": { 319 | "version": "1.0.6", 320 | "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz", 321 | "integrity": "sha1-4wOogrNCzD7oylE6eZmXNNqzriw=" 322 | }, 323 | "cors": { 324 | "version": "2.8.5", 325 | "resolved": "https://registry.npmjs.org/cors/-/cors-2.8.5.tgz", 326 | "integrity": "sha512-KIHbLJqu73RGr/hnbrO9uBeixNGuvSQjul/jdFvS/KFSIH1hWVd1ng7zOHx+YrEfInLG7q4n6GHQ9cDtxv/P6g==", 327 | "requires": { 328 | "object-assign": "^4", 329 | "vary": "^1" 330 | } 331 | }, 332 | "crypto-random-string": { 333 | "version": "2.0.0", 334 | "resolved": "https://registry.npmjs.org/crypto-random-string/-/crypto-random-string-2.0.0.tgz", 335 | "integrity": "sha512-v1plID3y9r/lPhviJ1wrXpLeyUIGAZ2SHNYTEapm7/8A9nLPoyvVp3RK/EPFqn5kEznyWgYZNsRtYYIWbuG8KA==", 336 | "dev": true 337 | }, 338 | "debug": { 339 | "version": "2.6.9", 340 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", 341 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", 342 | "requires": { 343 | "ms": "2.0.0" 344 | } 345 | }, 346 | "decompress-response": { 347 | "version": "3.3.0", 348 | "resolved": "https://registry.npmjs.org/decompress-response/-/decompress-response-3.3.0.tgz", 349 | "integrity": "sha1-gKTdMjdIOEv6JICDYirt7Jgq3/M=", 350 | "dev": true, 351 | "requires": { 352 | "mimic-response": "^1.0.0" 353 | } 354 | }, 355 | "deep-extend": { 356 | "version": "0.6.0", 357 | "resolved": "https://registry.npmjs.org/deep-extend/-/deep-extend-0.6.0.tgz", 358 | "integrity": "sha512-LOHxIOaPYdHlJRtCQfDIVZtfw/ufM8+rVj649RIHzcm/vGwQRXFt6OPqIFWsm2XEMrNIEtWR64sY1LEKD2vAOA==", 359 | "dev": true 360 | }, 361 | "defer-to-connect": { 362 | "version": "1.1.3", 363 | "resolved": "https://registry.npmjs.org/defer-to-connect/-/defer-to-connect-1.1.3.tgz", 364 | "integrity": "sha512-0ISdNousHvZT2EiFlZeZAHBUvSxmKswVCEf8hW7KWgG4a8MVEu/3Vb6uWYozkjylyCxe0JBIiRB1jV45S70WVQ==", 365 | "dev": true 366 | }, 367 | "depd": { 368 | "version": "1.1.2", 369 | "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.2.tgz", 370 | "integrity": "sha1-m81S4UwJd2PnSbJ0xDRu0uVgtak=" 371 | }, 372 | "destroy": { 373 | "version": "1.0.4", 374 | "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.0.4.tgz", 375 | "integrity": "sha1-l4hXRCxEdJ5CBmE+N5RiBYJqvYA=" 376 | }, 377 | "dot-prop": { 378 | "version": "5.3.0", 379 | "resolved": "https://registry.npmjs.org/dot-prop/-/dot-prop-5.3.0.tgz", 380 | "integrity": "sha512-QM8q3zDe58hqUqjraQOmzZ1LIH9SWQJTlEKCH4kJ2oQvLZk7RbQXvtDM2XEq3fwkV9CCvvH4LA0AV+ogFsBM2Q==", 381 | "dev": true, 382 | "requires": { 383 | "is-obj": "^2.0.0" 384 | } 385 | }, 386 | "duplexer3": { 387 | "version": "0.1.4", 388 | "resolved": "https://registry.npmjs.org/duplexer3/-/duplexer3-0.1.4.tgz", 389 | "integrity": "sha1-7gHdHKwO08vH/b6jfcCo8c4ALOI=", 390 | "dev": true 391 | }, 392 | "ee-first": { 393 | "version": "1.1.1", 394 | "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", 395 | "integrity": "sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0=" 396 | }, 397 | "emoji-regex": { 398 | "version": "7.0.3", 399 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-7.0.3.tgz", 400 | "integrity": "sha512-CwBLREIQ7LvYFB0WyRvwhq5N5qPhc6PMjD6bYggFlI5YyDgl+0vxq5VHbMOFqLg7hfWzmu8T5Z1QofhmTIhItA==", 401 | "dev": true 402 | }, 403 | "encodeurl": { 404 | "version": "1.0.2", 405 | "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz", 406 | "integrity": "sha1-rT/0yG7C0CkyL1oCw6mmBslbP1k=" 407 | }, 408 | "end-of-stream": { 409 | "version": "1.4.4", 410 | "resolved": "https://registry.npmjs.org/end-of-stream/-/end-of-stream-1.4.4.tgz", 411 | "integrity": "sha512-+uw1inIHVPQoaVuHzRyXd21icM+cnt4CzD5rW+NC1wjOUSTOs+Te7FOv7AhN7vS9x/oIyhLP5PR1H+phQAHu5Q==", 412 | "dev": true, 413 | "requires": { 414 | "once": "^1.4.0" 415 | } 416 | }, 417 | "escape-goat": { 418 | "version": "2.1.1", 419 | "resolved": "https://registry.npmjs.org/escape-goat/-/escape-goat-2.1.1.tgz", 420 | "integrity": "sha512-8/uIhbG12Csjy2JEW7D9pHbreaVaS/OpN3ycnyvElTdwM5n6GY6W6e2IPemfvGZeUMqZ9A/3GqIZMgKnBhAw/Q==", 421 | "dev": true 422 | }, 423 | "escape-html": { 424 | "version": "1.0.3", 425 | "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz", 426 | "integrity": "sha1-Aljq5NPQwJdN4cFpGI7wBR0dGYg=" 427 | }, 428 | "etag": { 429 | "version": "1.8.1", 430 | "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz", 431 | "integrity": "sha1-Qa4u62XvpiJorr/qg6x9eSmbCIc=" 432 | }, 433 | "express": { 434 | "version": "4.17.1", 435 | "resolved": "https://registry.npmjs.org/express/-/express-4.17.1.tgz", 436 | "integrity": "sha512-mHJ9O79RqluphRrcw2X/GTh3k9tVv8YcoyY4Kkh4WDMUYKRZUq0h1o0w2rrrxBqM7VoeUVqgb27xlEMXTnYt4g==", 437 | "requires": { 438 | "accepts": "~1.3.7", 439 | "array-flatten": "1.1.1", 440 | "body-parser": "1.19.0", 441 | "content-disposition": "0.5.3", 442 | "content-type": "~1.0.4", 443 | "cookie": "0.4.0", 444 | "cookie-signature": "1.0.6", 445 | "debug": "2.6.9", 446 | "depd": "~1.1.2", 447 | "encodeurl": "~1.0.2", 448 | "escape-html": "~1.0.3", 449 | "etag": "~1.8.1", 450 | "finalhandler": "~1.1.2", 451 | "fresh": "0.5.2", 452 | "merge-descriptors": "1.0.1", 453 | "methods": "~1.1.2", 454 | "on-finished": "~2.3.0", 455 | "parseurl": "~1.3.3", 456 | "path-to-regexp": "0.1.7", 457 | "proxy-addr": "~2.0.5", 458 | "qs": "6.7.0", 459 | "range-parser": "~1.2.1", 460 | "safe-buffer": "5.1.2", 461 | "send": "0.17.1", 462 | "serve-static": "1.14.1", 463 | "setprototypeof": "1.1.1", 464 | "statuses": "~1.5.0", 465 | "type-is": "~1.6.18", 466 | "utils-merge": "1.0.1", 467 | "vary": "~1.1.2" 468 | } 469 | }, 470 | "express-ws": { 471 | "version": "4.0.0", 472 | "resolved": "https://registry.npmjs.org/express-ws/-/express-ws-4.0.0.tgz", 473 | "integrity": "sha512-KEyUw8AwRET2iFjFsI1EJQrJ/fHeGiJtgpYgEWG3yDv4l/To/m3a2GaYfeGyB3lsWdvbesjF5XCMx+SVBgAAYw==", 474 | "requires": { 475 | "ws": "^5.2.0" 476 | } 477 | }, 478 | "fill-range": { 479 | "version": "7.0.1", 480 | "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz", 481 | "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", 482 | "dev": true, 483 | "requires": { 484 | "to-regex-range": "^5.0.1" 485 | } 486 | }, 487 | "finalhandler": { 488 | "version": "1.1.2", 489 | "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.1.2.tgz", 490 | "integrity": "sha512-aAWcW57uxVNrQZqFXjITpW3sIUQmHGG3qSb9mUah9MgMC4NeWhNOlNjXEYq3HjRAvL6arUviZGGJsBg6z0zsWA==", 491 | "requires": { 492 | "debug": "2.6.9", 493 | "encodeurl": "~1.0.2", 494 | "escape-html": "~1.0.3", 495 | "on-finished": "~2.3.0", 496 | "parseurl": "~1.3.3", 497 | "statuses": "~1.5.0", 498 | "unpipe": "~1.0.0" 499 | } 500 | }, 501 | "forwarded": { 502 | "version": "0.1.2", 503 | "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.1.2.tgz", 504 | "integrity": "sha1-mMI9qxF1ZXuMBXPozszZGw/xjIQ=" 505 | }, 506 | "fresh": { 507 | "version": "0.5.2", 508 | "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz", 509 | "integrity": "sha1-PYyt2Q2XZWn6g1qx+OSyOhBWBac=" 510 | }, 511 | "fsevents": { 512 | "version": "2.1.3", 513 | "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.1.3.tgz", 514 | "integrity": "sha512-Auw9a4AxqWpa9GUfj370BMPzzyncfBABW8Mab7BGWBYDj4Isgq+cDKtx0i6u9jcX9pQDnswsaaOTgTmA5pEjuQ==", 515 | "dev": true, 516 | "optional": true 517 | }, 518 | "get-stream": { 519 | "version": "4.1.0", 520 | "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-4.1.0.tgz", 521 | "integrity": "sha512-GMat4EJ5161kIy2HevLlr4luNjBgvmj413KaQA7jt4V8B4RDsfpHk7WQ9GVqfYyyx8OS/L66Kox+rJRNklLK7w==", 522 | "dev": true, 523 | "requires": { 524 | "pump": "^3.0.0" 525 | } 526 | }, 527 | "glob-parent": { 528 | "version": "5.1.1", 529 | "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.1.tgz", 530 | "integrity": "sha512-FnI+VGOpnlGHWZxthPGR+QhR78fuiK0sNLkHQv+bL9fQi57lNNdquIbna/WrfROrolq8GK5Ek6BiMwqL/voRYQ==", 531 | "dev": true, 532 | "requires": { 533 | "is-glob": "^4.0.1" 534 | } 535 | }, 536 | "global-dirs": { 537 | "version": "2.0.1", 538 | "resolved": "https://registry.npmjs.org/global-dirs/-/global-dirs-2.0.1.tgz", 539 | "integrity": "sha512-5HqUqdhkEovj2Of/ms3IeS/EekcO54ytHRLV4PEY2rhRwrHXLQjeVEES0Lhka0xwNDtGYn58wyC4s5+MHsOO6A==", 540 | "dev": true, 541 | "requires": { 542 | "ini": "^1.3.5" 543 | } 544 | }, 545 | "got": { 546 | "version": "9.6.0", 547 | "resolved": "https://registry.npmjs.org/got/-/got-9.6.0.tgz", 548 | "integrity": "sha512-R7eWptXuGYxwijs0eV+v3o6+XH1IqVK8dJOEecQfTmkncw9AV4dcw/Dhxi8MdlqPthxxpZyizMzyg8RTmEsG+Q==", 549 | "dev": true, 550 | "requires": { 551 | "@sindresorhus/is": "^0.14.0", 552 | "@szmarczak/http-timer": "^1.1.2", 553 | "cacheable-request": "^6.0.0", 554 | "decompress-response": "^3.3.0", 555 | "duplexer3": "^0.1.4", 556 | "get-stream": "^4.1.0", 557 | "lowercase-keys": "^1.0.1", 558 | "mimic-response": "^1.0.1", 559 | "p-cancelable": "^1.0.0", 560 | "to-readable-stream": "^1.0.0", 561 | "url-parse-lax": "^3.0.0" 562 | } 563 | }, 564 | "graceful-fs": { 565 | "version": "4.2.4", 566 | "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.4.tgz", 567 | "integrity": "sha512-WjKPNJF79dtJAVniUlGGWHYGz2jWxT6VhN/4m1NdkbZ2nOsEF+cI1Edgql5zCRhs/VsQYRvrXctxktVXZUkixw==", 568 | "dev": true 569 | }, 570 | "has-flag": { 571 | "version": "3.0.0", 572 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", 573 | "integrity": "sha1-tdRU3CGZriJWmfNGfloH87lVuv0=", 574 | "dev": true 575 | }, 576 | "has-yarn": { 577 | "version": "2.1.0", 578 | "resolved": "https://registry.npmjs.org/has-yarn/-/has-yarn-2.1.0.tgz", 579 | "integrity": "sha512-UqBRqi4ju7T+TqGNdqAO0PaSVGsDGJUBQvk9eUWNGRY1CFGDzYhLWoM7JQEemnlvVcv/YEmc2wNW8BC24EnUsw==", 580 | "dev": true 581 | }, 582 | "http-cache-semantics": { 583 | "version": "4.1.0", 584 | "resolved": "https://registry.npmjs.org/http-cache-semantics/-/http-cache-semantics-4.1.0.tgz", 585 | "integrity": "sha512-carPklcUh7ROWRK7Cv27RPtdhYhUsela/ue5/jKzjegVvXDqM2ILE9Q2BGn9JZJh1g87cp56su/FgQSzcWS8cQ==", 586 | "dev": true 587 | }, 588 | "http-errors": { 589 | "version": "1.7.2", 590 | "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.7.2.tgz", 591 | "integrity": "sha512-uUQBt3H/cSIVfch6i1EuPNy/YsRSOUBXTVfZ+yR7Zjez3qjBz6i9+i4zjNaoqcoFVI4lQJ5plg63TvGfRSDCRg==", 592 | "requires": { 593 | "depd": "~1.1.2", 594 | "inherits": "2.0.3", 595 | "setprototypeof": "1.1.1", 596 | "statuses": ">= 1.5.0 < 2", 597 | "toidentifier": "1.0.0" 598 | } 599 | }, 600 | "iconv-lite": { 601 | "version": "0.4.24", 602 | "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", 603 | "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", 604 | "requires": { 605 | "safer-buffer": ">= 2.1.2 < 3" 606 | } 607 | }, 608 | "ignore-by-default": { 609 | "version": "1.0.1", 610 | "resolved": "https://registry.npmjs.org/ignore-by-default/-/ignore-by-default-1.0.1.tgz", 611 | "integrity": "sha1-SMptcvbGo68Aqa1K5odr44ieKwk=", 612 | "dev": true 613 | }, 614 | "import-lazy": { 615 | "version": "2.1.0", 616 | "resolved": "https://registry.npmjs.org/import-lazy/-/import-lazy-2.1.0.tgz", 617 | "integrity": "sha1-BWmOPUXIjo1+nZLLBYTnfwlvPkM=", 618 | "dev": true 619 | }, 620 | "imurmurhash": { 621 | "version": "0.1.4", 622 | "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", 623 | "integrity": "sha1-khi5srkoojixPcT7a21XbyMUU+o=", 624 | "dev": true 625 | }, 626 | "inherits": { 627 | "version": "2.0.3", 628 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", 629 | "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=" 630 | }, 631 | "ini": { 632 | "version": "1.3.5", 633 | "resolved": "https://registry.npmjs.org/ini/-/ini-1.3.5.tgz", 634 | "integrity": "sha512-RZY5huIKCMRWDUqZlEi72f/lmXKMvuszcMBduliQ3nnWbx9X/ZBQO7DijMEYS9EhHBb2qacRUMtC7svLwe0lcw==", 635 | "dev": true 636 | }, 637 | "ipaddr.js": { 638 | "version": "1.9.1", 639 | "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz", 640 | "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==" 641 | }, 642 | "is-binary-path": { 643 | "version": "2.1.0", 644 | "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz", 645 | "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==", 646 | "dev": true, 647 | "requires": { 648 | "binary-extensions": "^2.0.0" 649 | } 650 | }, 651 | "is-ci": { 652 | "version": "2.0.0", 653 | "resolved": "https://registry.npmjs.org/is-ci/-/is-ci-2.0.0.tgz", 654 | "integrity": "sha512-YfJT7rkpQB0updsdHLGWrvhBJfcfzNNawYDNIyQXJz0IViGf75O8EBPKSdvw2rF+LGCsX4FZ8tcr3b19LcZq4w==", 655 | "dev": true, 656 | "requires": { 657 | "ci-info": "^2.0.0" 658 | } 659 | }, 660 | "is-extglob": { 661 | "version": "2.1.1", 662 | "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", 663 | "integrity": "sha1-qIwCU1eR8C7TfHahueqXc8gz+MI=", 664 | "dev": true 665 | }, 666 | "is-fullwidth-code-point": { 667 | "version": "2.0.0", 668 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", 669 | "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=", 670 | "dev": true 671 | }, 672 | "is-glob": { 673 | "version": "4.0.1", 674 | "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.1.tgz", 675 | "integrity": "sha512-5G0tKtBTFImOqDnLB2hG6Bp2qcKEFduo4tZu9MT/H6NQv/ghhy30o55ufafxJ/LdH79LLs2Kfrn85TLKyA7BUg==", 676 | "dev": true, 677 | "requires": { 678 | "is-extglob": "^2.1.1" 679 | } 680 | }, 681 | "is-installed-globally": { 682 | "version": "0.3.2", 683 | "resolved": "https://registry.npmjs.org/is-installed-globally/-/is-installed-globally-0.3.2.tgz", 684 | "integrity": "sha512-wZ8x1js7Ia0kecP/CHM/3ABkAmujX7WPvQk6uu3Fly/Mk44pySulQpnHG46OMjHGXApINnV4QhY3SWnECO2z5g==", 685 | "dev": true, 686 | "requires": { 687 | "global-dirs": "^2.0.1", 688 | "is-path-inside": "^3.0.1" 689 | } 690 | }, 691 | "is-npm": { 692 | "version": "4.0.0", 693 | "resolved": "https://registry.npmjs.org/is-npm/-/is-npm-4.0.0.tgz", 694 | "integrity": "sha512-96ECIfh9xtDDlPylNPXhzjsykHsMJZ18ASpaWzQyBr4YRTcVjUvzaHayDAES2oU/3KpljhHUjtSRNiDwi0F0ig==", 695 | "dev": true 696 | }, 697 | "is-number": { 698 | "version": "7.0.0", 699 | "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", 700 | "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", 701 | "dev": true 702 | }, 703 | "is-obj": { 704 | "version": "2.0.0", 705 | "resolved": "https://registry.npmjs.org/is-obj/-/is-obj-2.0.0.tgz", 706 | "integrity": "sha512-drqDG3cbczxxEJRoOXcOjtdp1J/lyp1mNn0xaznRs8+muBhgQcrnbspox5X5fOw0HnMnbfDzvnEMEtqDEJEo8w==", 707 | "dev": true 708 | }, 709 | "is-path-inside": { 710 | "version": "3.0.2", 711 | "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-3.0.2.tgz", 712 | "integrity": "sha512-/2UGPSgmtqwo1ktx8NDHjuPwZWmHhO+gj0f93EkhLB5RgW9RZevWYYlIkS6zePc6U2WpOdQYIwHe9YC4DWEBVg==", 713 | "dev": true 714 | }, 715 | "is-typedarray": { 716 | "version": "1.0.0", 717 | "resolved": "https://registry.npmjs.org/is-typedarray/-/is-typedarray-1.0.0.tgz", 718 | "integrity": "sha1-5HnICFjfDBsR3dppQPlgEfzaSpo=", 719 | "dev": true 720 | }, 721 | "is-yarn-global": { 722 | "version": "0.3.0", 723 | "resolved": "https://registry.npmjs.org/is-yarn-global/-/is-yarn-global-0.3.0.tgz", 724 | "integrity": "sha512-VjSeb/lHmkoyd8ryPVIKvOCn4D1koMqY+vqyjjUfc3xyKtP4dYOxM44sZrnqQSzSds3xyOrUTLTC9LVCVgLngw==", 725 | "dev": true 726 | }, 727 | "json-buffer": { 728 | "version": "3.0.0", 729 | "resolved": "https://registry.npmjs.org/json-buffer/-/json-buffer-3.0.0.tgz", 730 | "integrity": "sha1-Wx85evx11ne96Lz8Dkfh+aPZqJg=", 731 | "dev": true 732 | }, 733 | "keyv": { 734 | "version": "3.1.0", 735 | "resolved": "https://registry.npmjs.org/keyv/-/keyv-3.1.0.tgz", 736 | "integrity": "sha512-9ykJ/46SN/9KPM/sichzQ7OvXyGDYKGTaDlKMGCAlg2UK8KRy4jb0d8sFc+0Tt0YYnThq8X2RZgCg74RPxgcVA==", 737 | "dev": true, 738 | "requires": { 739 | "json-buffer": "3.0.0" 740 | } 741 | }, 742 | "latest-version": { 743 | "version": "5.1.0", 744 | "resolved": "https://registry.npmjs.org/latest-version/-/latest-version-5.1.0.tgz", 745 | "integrity": "sha512-weT+r0kTkRQdCdYCNtkMwWXQTMEswKrFBkm4ckQOMVhhqhIMI1UT2hMj+1iigIhgSZm5gTmrRXBNoGUgaTY1xA==", 746 | "dev": true, 747 | "requires": { 748 | "package-json": "^6.3.0" 749 | } 750 | }, 751 | "lowercase-keys": { 752 | "version": "1.0.1", 753 | "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-1.0.1.tgz", 754 | "integrity": "sha512-G2Lj61tXDnVFFOi8VZds+SoQjtQC3dgokKdDG2mTm1tx4m50NUHBOZSBwQQHyy0V12A0JTG4icfZQH+xPyh8VA==", 755 | "dev": true 756 | }, 757 | "make-dir": { 758 | "version": "3.1.0", 759 | "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-3.1.0.tgz", 760 | "integrity": "sha512-g3FeP20LNwhALb/6Cz6Dd4F2ngze0jz7tbzrD2wAV+o9FeNHe4rL+yK2md0J/fiSf1sa1ADhXqi5+oVwOM/eGw==", 761 | "dev": true, 762 | "requires": { 763 | "semver": "^6.0.0" 764 | }, 765 | "dependencies": { 766 | "semver": { 767 | "version": "6.3.0", 768 | "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", 769 | "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", 770 | "dev": true 771 | } 772 | } 773 | }, 774 | "media-typer": { 775 | "version": "0.3.0", 776 | "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", 777 | "integrity": "sha1-hxDXrwqmJvj/+hzgAWhUUmMlV0g=" 778 | }, 779 | "merge-descriptors": { 780 | "version": "1.0.1", 781 | "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz", 782 | "integrity": "sha1-sAqqVW3YtEVoFQ7J0blT8/kMu2E=" 783 | }, 784 | "methods": { 785 | "version": "1.1.2", 786 | "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz", 787 | "integrity": "sha1-VSmk1nZUE07cxSZmVoNbD4Ua/O4=" 788 | }, 789 | "mime": { 790 | "version": "1.6.0", 791 | "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz", 792 | "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==" 793 | }, 794 | "mime-db": { 795 | "version": "1.44.0", 796 | "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.44.0.tgz", 797 | "integrity": "sha512-/NOTfLrsPBVeH7YtFPgsVWveuL+4SjjYxaQ1xtM1KMFj7HdxlBlxeyNLzhyJVx7r4rZGJAZ/6lkKCitSc/Nmpg==" 798 | }, 799 | "mime-types": { 800 | "version": "2.1.27", 801 | "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.27.tgz", 802 | "integrity": "sha512-JIhqnCasI9yD+SsmkquHBxTSEuZdQX5BuQnS2Vc7puQQQ+8yiP5AY5uWhpdv4YL4VM5c6iliiYWPgJ/nJQLp7w==", 803 | "requires": { 804 | "mime-db": "1.44.0" 805 | } 806 | }, 807 | "mimic-response": { 808 | "version": "1.0.1", 809 | "resolved": "https://registry.npmjs.org/mimic-response/-/mimic-response-1.0.1.tgz", 810 | "integrity": "sha512-j5EctnkH7amfV/q5Hgmoal1g2QHFJRraOtmx0JpIqkxhBhI/lJSl1nMpQ45hVarwNETOoWEimndZ4QK0RHxuxQ==", 811 | "dev": true 812 | }, 813 | "minimatch": { 814 | "version": "3.0.4", 815 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz", 816 | "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==", 817 | "dev": true, 818 | "requires": { 819 | "brace-expansion": "^1.1.7" 820 | } 821 | }, 822 | "minimist": { 823 | "version": "1.2.5", 824 | "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.5.tgz", 825 | "integrity": "sha512-FM9nNUYrRBAELZQT3xeZQ7fmMOBg6nWNmJKTcgsJeaLstP/UODVpGsr5OhXhhXg6f+qtJ8uiZ+PUxkDWcgIXLw==", 826 | "dev": true 827 | }, 828 | "ms": { 829 | "version": "2.0.0", 830 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", 831 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" 832 | }, 833 | "negotiator": { 834 | "version": "0.6.2", 835 | "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.2.tgz", 836 | "integrity": "sha512-hZXc7K2e+PgeI1eDBe/10Ard4ekbfrrqG8Ep+8Jmf4JID2bNg7NvCPOZN+kfF574pFQI7mum2AUqDidoKqcTOw==" 837 | }, 838 | "nodemon": { 839 | "version": "2.0.6", 840 | "resolved": "https://registry.npmjs.org/nodemon/-/nodemon-2.0.6.tgz", 841 | "integrity": "sha512-4I3YDSKXg6ltYpcnZeHompqac4E6JeAMpGm8tJnB9Y3T0ehasLa4139dJOcCrB93HHrUMsCrKtoAlXTqT5n4AQ==", 842 | "dev": true, 843 | "requires": { 844 | "chokidar": "^3.2.2", 845 | "debug": "^3.2.6", 846 | "ignore-by-default": "^1.0.1", 847 | "minimatch": "^3.0.4", 848 | "pstree.remy": "^1.1.7", 849 | "semver": "^5.7.1", 850 | "supports-color": "^5.5.0", 851 | "touch": "^3.1.0", 852 | "undefsafe": "^2.0.3", 853 | "update-notifier": "^4.1.0" 854 | }, 855 | "dependencies": { 856 | "debug": { 857 | "version": "3.2.6", 858 | "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.6.tgz", 859 | "integrity": "sha512-mel+jf7nrtEl5Pn1Qx46zARXKDpBbvzezse7p7LqINmdoIk8PYP5SySaxEmYv6TZ0JyEKA1hsCId6DIhgITtWQ==", 860 | "dev": true, 861 | "requires": { 862 | "ms": "^2.1.1" 863 | } 864 | }, 865 | "ms": { 866 | "version": "2.1.2", 867 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", 868 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", 869 | "dev": true 870 | } 871 | } 872 | }, 873 | "nopt": { 874 | "version": "1.0.10", 875 | "resolved": "https://registry.npmjs.org/nopt/-/nopt-1.0.10.tgz", 876 | "integrity": "sha1-bd0hvSoxQXuScn3Vhfim83YI6+4=", 877 | "dev": true, 878 | "requires": { 879 | "abbrev": "1" 880 | } 881 | }, 882 | "normalize-path": { 883 | "version": "3.0.0", 884 | "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", 885 | "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==", 886 | "dev": true 887 | }, 888 | "normalize-url": { 889 | "version": "4.5.0", 890 | "resolved": "https://registry.npmjs.org/normalize-url/-/normalize-url-4.5.0.tgz", 891 | "integrity": "sha512-2s47yzUxdexf1OhyRi4Em83iQk0aPvwTddtFz4hnSSw9dCEsLEGf6SwIO8ss/19S9iBb5sJaOuTvTGDeZI00BQ==", 892 | "dev": true 893 | }, 894 | "object-assign": { 895 | "version": "4.1.1", 896 | "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", 897 | "integrity": "sha1-IQmtx5ZYh8/AXLvUQsrIv7s2CGM=" 898 | }, 899 | "on-finished": { 900 | "version": "2.3.0", 901 | "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.3.0.tgz", 902 | "integrity": "sha1-IPEzZIGwg811M3mSoWlxqi2QaUc=", 903 | "requires": { 904 | "ee-first": "1.1.1" 905 | } 906 | }, 907 | "once": { 908 | "version": "1.4.0", 909 | "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", 910 | "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=", 911 | "dev": true, 912 | "requires": { 913 | "wrappy": "1" 914 | } 915 | }, 916 | "p-cancelable": { 917 | "version": "1.1.0", 918 | "resolved": "https://registry.npmjs.org/p-cancelable/-/p-cancelable-1.1.0.tgz", 919 | "integrity": "sha512-s73XxOZ4zpt1edZYZzvhqFa6uvQc1vwUa0K0BdtIZgQMAJj9IbebH+JkgKZc9h+B05PKHLOTl4ajG1BmNrVZlw==", 920 | "dev": true 921 | }, 922 | "package-json": { 923 | "version": "6.5.0", 924 | "resolved": "https://registry.npmjs.org/package-json/-/package-json-6.5.0.tgz", 925 | "integrity": "sha512-k3bdm2n25tkyxcjSKzB5x8kfVxlMdgsbPr0GkZcwHsLpba6cBjqCt1KlcChKEvxHIcTB1FVMuwoijZ26xex5MQ==", 926 | "dev": true, 927 | "requires": { 928 | "got": "^9.6.0", 929 | "registry-auth-token": "^4.0.0", 930 | "registry-url": "^5.0.0", 931 | "semver": "^6.2.0" 932 | }, 933 | "dependencies": { 934 | "semver": { 935 | "version": "6.3.0", 936 | "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", 937 | "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", 938 | "dev": true 939 | } 940 | } 941 | }, 942 | "parseurl": { 943 | "version": "1.3.3", 944 | "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz", 945 | "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==" 946 | }, 947 | "path-to-regexp": { 948 | "version": "0.1.7", 949 | "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz", 950 | "integrity": "sha1-32BBeABfUi8V60SQ5yR6G/qmf4w=" 951 | }, 952 | "picomatch": { 953 | "version": "2.2.2", 954 | "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.2.2.tgz", 955 | "integrity": "sha512-q0M/9eZHzmr0AulXyPwNfZjtwZ/RBZlbN3K3CErVrk50T2ASYI7Bye0EvekFY3IP1Nt2DHu0re+V2ZHIpMkuWg==", 956 | "dev": true 957 | }, 958 | "prepend-http": { 959 | "version": "2.0.0", 960 | "resolved": "https://registry.npmjs.org/prepend-http/-/prepend-http-2.0.0.tgz", 961 | "integrity": "sha1-6SQ0v6XqjBn0HN/UAddBo8gZ2Jc=", 962 | "dev": true 963 | }, 964 | "proxy-addr": { 965 | "version": "2.0.6", 966 | "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.6.tgz", 967 | "integrity": "sha512-dh/frvCBVmSsDYzw6n926jv974gddhkFPfiN8hPOi30Wax25QZyZEGveluCgliBnqmuM+UJmBErbAUFIoDbjOw==", 968 | "requires": { 969 | "forwarded": "~0.1.2", 970 | "ipaddr.js": "1.9.1" 971 | } 972 | }, 973 | "pstree.remy": { 974 | "version": "1.1.8", 975 | "resolved": "https://registry.npmjs.org/pstree.remy/-/pstree.remy-1.1.8.tgz", 976 | "integrity": "sha512-77DZwxQmxKnu3aR542U+X8FypNzbfJ+C5XQDk3uWjWxn6151aIMGthWYRXTqT1E5oJvg+ljaa2OJi+VfvCOQ8w==", 977 | "dev": true 978 | }, 979 | "pump": { 980 | "version": "3.0.0", 981 | "resolved": "https://registry.npmjs.org/pump/-/pump-3.0.0.tgz", 982 | "integrity": "sha512-LwZy+p3SFs1Pytd/jYct4wpv49HiYCqd9Rlc5ZVdk0V+8Yzv6jR5Blk3TRmPL1ft69TxP0IMZGJ+WPFU2BFhww==", 983 | "dev": true, 984 | "requires": { 985 | "end-of-stream": "^1.1.0", 986 | "once": "^1.3.1" 987 | } 988 | }, 989 | "pupa": { 990 | "version": "2.1.1", 991 | "resolved": "https://registry.npmjs.org/pupa/-/pupa-2.1.1.tgz", 992 | "integrity": "sha512-l1jNAspIBSFqbT+y+5FosojNpVpF94nlI+wDUpqP9enwOTfHx9f0gh5nB96vl+6yTpsJsypeNrwfzPrKuHB41A==", 993 | "dev": true, 994 | "requires": { 995 | "escape-goat": "^2.0.0" 996 | } 997 | }, 998 | "qs": { 999 | "version": "6.7.0", 1000 | "resolved": "https://registry.npmjs.org/qs/-/qs-6.7.0.tgz", 1001 | "integrity": "sha512-VCdBRNFTX1fyE7Nb6FYoURo/SPe62QCaAyzJvUjwRaIsc+NePBEniHlvxFmmX56+HZphIGtV0XeCirBtpDrTyQ==" 1002 | }, 1003 | "range-parser": { 1004 | "version": "1.2.1", 1005 | "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz", 1006 | "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==" 1007 | }, 1008 | "raw-body": { 1009 | "version": "2.4.0", 1010 | "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.4.0.tgz", 1011 | "integrity": "sha512-4Oz8DUIwdvoa5qMJelxipzi/iJIi40O5cGV1wNYp5hvZP8ZN0T+jiNkL0QepXs+EsQ9XJ8ipEDoiH70ySUJP3Q==", 1012 | "requires": { 1013 | "bytes": "3.1.0", 1014 | "http-errors": "1.7.2", 1015 | "iconv-lite": "0.4.24", 1016 | "unpipe": "1.0.0" 1017 | } 1018 | }, 1019 | "rc": { 1020 | "version": "1.2.8", 1021 | "resolved": "https://registry.npmjs.org/rc/-/rc-1.2.8.tgz", 1022 | "integrity": "sha512-y3bGgqKj3QBdxLbLkomlohkvsA8gdAiUQlSBJnBhfn+BPxg4bc62d8TcBW15wavDfgexCgccckhcZvywyQYPOw==", 1023 | "dev": true, 1024 | "requires": { 1025 | "deep-extend": "^0.6.0", 1026 | "ini": "~1.3.0", 1027 | "minimist": "^1.2.0", 1028 | "strip-json-comments": "~2.0.1" 1029 | } 1030 | }, 1031 | "readdirp": { 1032 | "version": "3.5.0", 1033 | "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.5.0.tgz", 1034 | "integrity": "sha512-cMhu7c/8rdhkHXWsY+osBhfSy0JikwpHK/5+imo+LpeasTF8ouErHrlYkwT0++njiyuDvc7OFY5T3ukvZ8qmFQ==", 1035 | "dev": true, 1036 | "requires": { 1037 | "picomatch": "^2.2.1" 1038 | } 1039 | }, 1040 | "registry-auth-token": { 1041 | "version": "4.2.1", 1042 | "resolved": "https://registry.npmjs.org/registry-auth-token/-/registry-auth-token-4.2.1.tgz", 1043 | "integrity": "sha512-6gkSb4U6aWJB4SF2ZvLb76yCBjcvufXBqvvEx1HbmKPkutswjW1xNVRY0+daljIYRbogN7O0etYSlbiaEQyMyw==", 1044 | "dev": true, 1045 | "requires": { 1046 | "rc": "^1.2.8" 1047 | } 1048 | }, 1049 | "registry-url": { 1050 | "version": "5.1.0", 1051 | "resolved": "https://registry.npmjs.org/registry-url/-/registry-url-5.1.0.tgz", 1052 | "integrity": "sha512-8acYXXTI0AkQv6RAOjE3vOaIXZkT9wo4LOFbBKYQEEnnMNBpKqdUrI6S4NT0KPIo/WVvJ5tE/X5LF/TQUf0ekw==", 1053 | "dev": true, 1054 | "requires": { 1055 | "rc": "^1.2.8" 1056 | } 1057 | }, 1058 | "responselike": { 1059 | "version": "1.0.2", 1060 | "resolved": "https://registry.npmjs.org/responselike/-/responselike-1.0.2.tgz", 1061 | "integrity": "sha1-kYcg7ztjHFZCvgaPFa3lpG9Loec=", 1062 | "dev": true, 1063 | "requires": { 1064 | "lowercase-keys": "^1.0.0" 1065 | } 1066 | }, 1067 | "safe-buffer": { 1068 | "version": "5.1.2", 1069 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", 1070 | "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==" 1071 | }, 1072 | "safer-buffer": { 1073 | "version": "2.1.2", 1074 | "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", 1075 | "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==" 1076 | }, 1077 | "semver": { 1078 | "version": "5.7.1", 1079 | "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", 1080 | "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", 1081 | "dev": true 1082 | }, 1083 | "semver-diff": { 1084 | "version": "3.1.1", 1085 | "resolved": "https://registry.npmjs.org/semver-diff/-/semver-diff-3.1.1.tgz", 1086 | "integrity": "sha512-GX0Ix/CJcHyB8c4ykpHGIAvLyOwOobtM/8d+TQkAd81/bEjgPHrfba41Vpesr7jX/t8Uh+R3EX9eAS5be+jQYg==", 1087 | "dev": true, 1088 | "requires": { 1089 | "semver": "^6.3.0" 1090 | }, 1091 | "dependencies": { 1092 | "semver": { 1093 | "version": "6.3.0", 1094 | "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", 1095 | "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", 1096 | "dev": true 1097 | } 1098 | } 1099 | }, 1100 | "send": { 1101 | "version": "0.17.1", 1102 | "resolved": "https://registry.npmjs.org/send/-/send-0.17.1.tgz", 1103 | "integrity": "sha512-BsVKsiGcQMFwT8UxypobUKyv7irCNRHk1T0G680vk88yf6LBByGcZJOTJCrTP2xVN6yI+XjPJcNuE3V4fT9sAg==", 1104 | "requires": { 1105 | "debug": "2.6.9", 1106 | "depd": "~1.1.2", 1107 | "destroy": "~1.0.4", 1108 | "encodeurl": "~1.0.2", 1109 | "escape-html": "~1.0.3", 1110 | "etag": "~1.8.1", 1111 | "fresh": "0.5.2", 1112 | "http-errors": "~1.7.2", 1113 | "mime": "1.6.0", 1114 | "ms": "2.1.1", 1115 | "on-finished": "~2.3.0", 1116 | "range-parser": "~1.2.1", 1117 | "statuses": "~1.5.0" 1118 | }, 1119 | "dependencies": { 1120 | "ms": { 1121 | "version": "2.1.1", 1122 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.1.tgz", 1123 | "integrity": "sha512-tgp+dl5cGk28utYktBsrFqA7HKgrhgPsg6Z/EfhWI4gl1Hwq8B/GmY/0oXZ6nF8hDVesS/FpnYaD/kOWhYQvyg==" 1124 | } 1125 | } 1126 | }, 1127 | "serve-static": { 1128 | "version": "1.14.1", 1129 | "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.14.1.tgz", 1130 | "integrity": "sha512-JMrvUwE54emCYWlTI+hGrGv5I8dEwmco/00EvkzIIsR7MqrHonbD9pO2MOfFnpFntl7ecpZs+3mW+XbQZu9QCg==", 1131 | "requires": { 1132 | "encodeurl": "~1.0.2", 1133 | "escape-html": "~1.0.3", 1134 | "parseurl": "~1.3.3", 1135 | "send": "0.17.1" 1136 | } 1137 | }, 1138 | "setprototypeof": { 1139 | "version": "1.1.1", 1140 | "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.1.tgz", 1141 | "integrity": "sha512-JvdAWfbXeIGaZ9cILp38HntZSFSo3mWg6xGcJJsd+d4aRMOqauag1C63dJfDw7OaMYwEbHMOxEZ1lqVRYP2OAw==" 1142 | }, 1143 | "signal-exit": { 1144 | "version": "3.0.3", 1145 | "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.3.tgz", 1146 | "integrity": "sha512-VUJ49FC8U1OxwZLxIbTTrDvLnf/6TDgxZcK8wxR8zs13xpx7xbG60ndBlhNrFi2EMuFRoeDoJO7wthSLq42EjA==", 1147 | "dev": true 1148 | }, 1149 | "statuses": { 1150 | "version": "1.5.0", 1151 | "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.5.0.tgz", 1152 | "integrity": "sha1-Fhx9rBd2Wf2YEfQ3cfqZOBR4Yow=" 1153 | }, 1154 | "string-width": { 1155 | "version": "4.2.0", 1156 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.0.tgz", 1157 | "integrity": "sha512-zUz5JD+tgqtuDjMhwIg5uFVV3dtqZ9yQJlZVfq4I01/K5Paj5UHj7VyrQOJvzawSVlKpObApbfD0Ed6yJc+1eg==", 1158 | "dev": true, 1159 | "requires": { 1160 | "emoji-regex": "^8.0.0", 1161 | "is-fullwidth-code-point": "^3.0.0", 1162 | "strip-ansi": "^6.0.0" 1163 | }, 1164 | "dependencies": { 1165 | "ansi-regex": { 1166 | "version": "5.0.0", 1167 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.0.tgz", 1168 | "integrity": "sha512-bY6fj56OUQ0hU1KjFNDQuJFezqKdrAyFdIevADiqrWHwSlbmBNMHp5ak2f40Pm8JTFyM2mqxkG6ngkHO11f/lg==", 1169 | "dev": true 1170 | }, 1171 | "emoji-regex": { 1172 | "version": "8.0.0", 1173 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", 1174 | "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", 1175 | "dev": true 1176 | }, 1177 | "is-fullwidth-code-point": { 1178 | "version": "3.0.0", 1179 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", 1180 | "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", 1181 | "dev": true 1182 | }, 1183 | "strip-ansi": { 1184 | "version": "6.0.0", 1185 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.0.tgz", 1186 | "integrity": "sha512-AuvKTrTfQNYNIctbR1K/YGTR1756GycPsg7b9bdV9Duqur4gv6aKqHXah67Z8ImS7WEz5QVcOtlfW2rZEugt6w==", 1187 | "dev": true, 1188 | "requires": { 1189 | "ansi-regex": "^5.0.0" 1190 | } 1191 | } 1192 | } 1193 | }, 1194 | "strip-ansi": { 1195 | "version": "5.2.0", 1196 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", 1197 | "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", 1198 | "dev": true, 1199 | "requires": { 1200 | "ansi-regex": "^4.1.0" 1201 | } 1202 | }, 1203 | "strip-json-comments": { 1204 | "version": "2.0.1", 1205 | "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-2.0.1.tgz", 1206 | "integrity": "sha1-PFMZQukIwml8DsNEhYwobHygpgo=", 1207 | "dev": true 1208 | }, 1209 | "supports-color": { 1210 | "version": "5.5.0", 1211 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", 1212 | "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", 1213 | "dev": true, 1214 | "requires": { 1215 | "has-flag": "^3.0.0" 1216 | } 1217 | }, 1218 | "term-size": { 1219 | "version": "2.2.1", 1220 | "resolved": "https://registry.npmjs.org/term-size/-/term-size-2.2.1.tgz", 1221 | "integrity": "sha512-wK0Ri4fOGjv/XPy8SBHZChl8CM7uMc5VML7SqiQ0zG7+J5Vr+RMQDoHa2CNT6KHUnTGIXH34UDMkPzAUyapBZg==", 1222 | "dev": true 1223 | }, 1224 | "to-readable-stream": { 1225 | "version": "1.0.0", 1226 | "resolved": "https://registry.npmjs.org/to-readable-stream/-/to-readable-stream-1.0.0.tgz", 1227 | "integrity": "sha512-Iq25XBt6zD5npPhlLVXGFN3/gyR2/qODcKNNyTMd4vbm39HUaOiAM4PMq0eMVC/Tkxz+Zjdsc55g9yyz+Yq00Q==", 1228 | "dev": true 1229 | }, 1230 | "to-regex-range": { 1231 | "version": "5.0.1", 1232 | "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", 1233 | "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", 1234 | "dev": true, 1235 | "requires": { 1236 | "is-number": "^7.0.0" 1237 | } 1238 | }, 1239 | "toidentifier": { 1240 | "version": "1.0.0", 1241 | "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.0.tgz", 1242 | "integrity": "sha512-yaOH/Pk/VEhBWWTlhI+qXxDFXlejDGcQipMlyxda9nthulaxLZUNcUqFxokp0vcYnvteJln5FNQDRrxj3YcbVw==" 1243 | }, 1244 | "touch": { 1245 | "version": "3.1.0", 1246 | "resolved": "https://registry.npmjs.org/touch/-/touch-3.1.0.tgz", 1247 | "integrity": "sha512-WBx8Uy5TLtOSRtIq+M03/sKDrXCLHxwDcquSP2c43Le03/9serjQBIztjRz6FkJez9D/hleyAXTBGLwwZUw9lA==", 1248 | "dev": true, 1249 | "requires": { 1250 | "nopt": "~1.0.10" 1251 | } 1252 | }, 1253 | "type-fest": { 1254 | "version": "0.8.1", 1255 | "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.8.1.tgz", 1256 | "integrity": "sha512-4dbzIzqvjtgiM5rw1k5rEHtBANKmdudhGyBEajN01fEyhaAIhsoKNy6y7+IN93IfpFtwY9iqi7kD+xwKhQsNJA==", 1257 | "dev": true 1258 | }, 1259 | "type-is": { 1260 | "version": "1.6.18", 1261 | "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz", 1262 | "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==", 1263 | "requires": { 1264 | "media-typer": "0.3.0", 1265 | "mime-types": "~2.1.24" 1266 | } 1267 | }, 1268 | "typedarray-to-buffer": { 1269 | "version": "3.1.5", 1270 | "resolved": "https://registry.npmjs.org/typedarray-to-buffer/-/typedarray-to-buffer-3.1.5.tgz", 1271 | "integrity": "sha512-zdu8XMNEDepKKR+XYOXAVPtWui0ly0NtohUscw+UmaHiAWT8hrV1rr//H6V+0DvJ3OQ19S979M0laLfX8rm82Q==", 1272 | "dev": true, 1273 | "requires": { 1274 | "is-typedarray": "^1.0.0" 1275 | } 1276 | }, 1277 | "undefsafe": { 1278 | "version": "2.0.3", 1279 | "resolved": "https://registry.npmjs.org/undefsafe/-/undefsafe-2.0.3.tgz", 1280 | "integrity": "sha512-nrXZwwXrD/T/JXeygJqdCO6NZZ1L66HrxM/Z7mIq2oPanoN0F1nLx3lwJMu6AwJY69hdixaFQOuoYsMjE5/C2A==", 1281 | "dev": true, 1282 | "requires": { 1283 | "debug": "^2.2.0" 1284 | } 1285 | }, 1286 | "unique-string": { 1287 | "version": "2.0.0", 1288 | "resolved": "https://registry.npmjs.org/unique-string/-/unique-string-2.0.0.tgz", 1289 | "integrity": "sha512-uNaeirEPvpZWSgzwsPGtU2zVSTrn/8L5q/IexZmH0eH6SA73CmAA5U4GwORTxQAZs95TAXLNqeLoPPNO5gZfWg==", 1290 | "dev": true, 1291 | "requires": { 1292 | "crypto-random-string": "^2.0.0" 1293 | } 1294 | }, 1295 | "unpipe": { 1296 | "version": "1.0.0", 1297 | "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", 1298 | "integrity": "sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw=" 1299 | }, 1300 | "update-notifier": { 1301 | "version": "4.1.3", 1302 | "resolved": "https://registry.npmjs.org/update-notifier/-/update-notifier-4.1.3.tgz", 1303 | "integrity": "sha512-Yld6Z0RyCYGB6ckIjffGOSOmHXj1gMeE7aROz4MG+XMkmixBX4jUngrGXNYz7wPKBmtoD4MnBa2Anu7RSKht/A==", 1304 | "dev": true, 1305 | "requires": { 1306 | "boxen": "^4.2.0", 1307 | "chalk": "^3.0.0", 1308 | "configstore": "^5.0.1", 1309 | "has-yarn": "^2.1.0", 1310 | "import-lazy": "^2.1.0", 1311 | "is-ci": "^2.0.0", 1312 | "is-installed-globally": "^0.3.1", 1313 | "is-npm": "^4.0.0", 1314 | "is-yarn-global": "^0.3.0", 1315 | "latest-version": "^5.0.0", 1316 | "pupa": "^2.0.1", 1317 | "semver-diff": "^3.1.1", 1318 | "xdg-basedir": "^4.0.0" 1319 | } 1320 | }, 1321 | "url-parse-lax": { 1322 | "version": "3.0.0", 1323 | "resolved": "https://registry.npmjs.org/url-parse-lax/-/url-parse-lax-3.0.0.tgz", 1324 | "integrity": "sha1-FrXK/Afb42dsGxmZF3gj1lA6yww=", 1325 | "dev": true, 1326 | "requires": { 1327 | "prepend-http": "^2.0.0" 1328 | } 1329 | }, 1330 | "utils-merge": { 1331 | "version": "1.0.1", 1332 | "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz", 1333 | "integrity": "sha1-n5VxD1CiZ5R7LMwSR0HBAoQn5xM=" 1334 | }, 1335 | "vary": { 1336 | "version": "1.1.2", 1337 | "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz", 1338 | "integrity": "sha1-IpnwLG3tMNSllhsLn3RSShj2NPw=" 1339 | }, 1340 | "widest-line": { 1341 | "version": "3.1.0", 1342 | "resolved": "https://registry.npmjs.org/widest-line/-/widest-line-3.1.0.tgz", 1343 | "integrity": "sha512-NsmoXalsWVDMGupxZ5R08ka9flZjjiLvHVAWYOKtiKM8ujtZWr9cRffak+uSE48+Ob8ObalXpwyeUiyDD6QFgg==", 1344 | "dev": true, 1345 | "requires": { 1346 | "string-width": "^4.0.0" 1347 | } 1348 | }, 1349 | "wrappy": { 1350 | "version": "1.0.2", 1351 | "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", 1352 | "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=", 1353 | "dev": true 1354 | }, 1355 | "write-file-atomic": { 1356 | "version": "3.0.3", 1357 | "resolved": "https://registry.npmjs.org/write-file-atomic/-/write-file-atomic-3.0.3.tgz", 1358 | "integrity": "sha512-AvHcyZ5JnSfq3ioSyjrBkH9yW4m7Ayk8/9My/DD9onKeu/94fwrMocemO2QAJFAlnnDN+ZDS+ZjAR5ua1/PV/Q==", 1359 | "dev": true, 1360 | "requires": { 1361 | "imurmurhash": "^0.1.4", 1362 | "is-typedarray": "^1.0.0", 1363 | "signal-exit": "^3.0.2", 1364 | "typedarray-to-buffer": "^3.1.5" 1365 | } 1366 | }, 1367 | "ws": { 1368 | "version": "5.2.2", 1369 | "resolved": "https://registry.npmjs.org/ws/-/ws-5.2.2.tgz", 1370 | "integrity": "sha512-jaHFD6PFv6UgoIVda6qZllptQsMlDEJkTQcybzzXDYM1XO9Y8em691FGMPmM46WGyLU4z9KMgQN+qrux/nhlHA==", 1371 | "requires": { 1372 | "async-limiter": "~1.0.0" 1373 | } 1374 | }, 1375 | "xdg-basedir": { 1376 | "version": "4.0.0", 1377 | "resolved": "https://registry.npmjs.org/xdg-basedir/-/xdg-basedir-4.0.0.tgz", 1378 | "integrity": "sha512-PSNhEJDejZYV7h50BohL09Er9VaIefr2LMAf3OEmpCkjOi34eYyQYAXUTjEQtZJTKcF0E2UKTh+osDLsgNim9Q==", 1379 | "dev": true 1380 | } 1381 | } 1382 | } 1383 | --------------------------------------------------------------------------------