├── 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 |
4 |
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 |
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
19 |
20 |
21 |
22 |
23 |
24 |
25 |
26 |
27 |
28 |
29 |
30 |
31 |
32 |
33 |
34 |
35 |
36 | 1605536385150
37 |
38 |
39 | 1605536385150
40 |
41 |
42 |
43 |
44 |
45 |
46 |
47 |
48 |
49 |
50 |
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 | connectHandler()}>
99 | Войти
100 |
101 |
102 |
103 |
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 |
--------------------------------------------------------------------------------