├── .env.example
├── .gitignore
├── client
├── .gitignore
├── README.md
├── package-lock.json
├── package.json
├── public
│ ├── favicon.ico
│ ├── index.html
│ ├── logo192.png
│ ├── logo512.png
│ ├── manifest.json
│ └── robots.txt
├── src
│ ├── ApolloProvider.js
│ ├── App.js
│ ├── App.scss
│ ├── context
│ │ ├── auth.js
│ │ └── message.js
│ ├── index.js
│ ├── pages
│ │ ├── Login.js
│ │ ├── Register.js
│ │ └── home
│ │ │ ├── Home.js
│ │ │ ├── Message.js
│ │ │ ├── Messages.js
│ │ │ └── Users.js
│ └── util
│ │ └── DynamicRoute.js
└── yarn.lock
├── config
└── config.js
├── deploy.md
├── graphql
├── resolvers
│ ├── index.js
│ ├── messages.js
│ └── users.js
└── typeDefs.js
├── migrations
├── 20200809134724-create-user.js
├── 20200823130528-create-message.js
└── 20200910194215-create-reaction.js
├── models
├── Message.js
├── Reaction.js
├── index.js
└── user.js
├── package-lock.json
├── package.json
├── seeders
├── 20200827161337-create-users.js
└── 20200827161417-create-messages.js
├── server.js
└── util
└── contextMiddleware.js
/.env.example:
--------------------------------------------------------------------------------
1 | NODE_ENV=
2 |
3 | JWT_SECRET=
4 |
5 | # Database credentials
6 | DB_USERNAME=
7 | DB_PASSWORD=
8 | DB_DATABASE=
9 | DB_HOST=
10 | DB_DIALECT=
--------------------------------------------------------------------------------
/.gitignore:
--------------------------------------------------------------------------------
1 | node_modules/
2 |
3 | .env
--------------------------------------------------------------------------------
/client/.gitignore:
--------------------------------------------------------------------------------
1 | # See https://help.github.com/articles/ignoring-files/ for more about ignoring files.
2 |
3 | # dependencies
4 | /node_modules
5 | /.pnp
6 | .pnp.js
7 |
8 | # testing
9 | /coverage
10 |
11 | # production
12 | /build
13 |
14 | # misc
15 | .DS_Store
16 | .env.local
17 | .env.development.local
18 | .env.test.local
19 | .env.production.local
20 |
21 | npm-debug.log*
22 | yarn-debug.log*
23 | yarn-error.log*
24 |
--------------------------------------------------------------------------------
/client/README.md:
--------------------------------------------------------------------------------
1 | This project was bootstrapped with [Create React App](https://github.com/facebook/create-react-app).
2 |
3 | ## Available Scripts
4 |
5 | In the project directory, you can run:
6 |
7 | ### `yarn start`
8 |
9 | Runs the app in the development mode.
10 | Open [http://localhost:3000](http://localhost:3000) to view it in the browser.
11 |
12 | The page will reload if you make edits.
13 | You will also see any lint errors in the console.
14 |
15 | ### `yarn test`
16 |
17 | Launches the test runner in the interactive watch mode.
18 | See the section about [running tests](https://facebook.github.io/create-react-app/docs/running-tests) for more information.
19 |
20 | ### `yarn build`
21 |
22 | Builds the app for production to the `build` folder.
23 | It correctly bundles React in production mode and optimizes the build for the best performance.
24 |
25 | The build is minified and the filenames include the hashes.
26 | Your app is ready to be deployed!
27 |
28 | See the section about [deployment](https://facebook.github.io/create-react-app/docs/deployment) for more information.
29 |
30 | ### `yarn eject`
31 |
32 | **Note: this is a one-way operation. Once you `eject`, you can’t go back!**
33 |
34 | If you aren’t satisfied with the build tool and configuration choices, you can `eject` at any time. This command will remove the single build dependency from your project.
35 |
36 | Instead, it will copy all the configuration files and the transitive dependencies (webpack, Babel, ESLint, etc) right into your project so you have full control over them. All of the commands except `eject` will still work, but they will point to the copied scripts so you can tweak them. At this point you’re on your own.
37 |
38 | You don’t have to ever use `eject`. The curated feature set is suitable for small and middle deployments, and you shouldn’t feel obligated to use this feature. However we understand that this tool wouldn’t be useful if you couldn’t customize it when you are ready for it.
39 |
40 | ## Learn More
41 |
42 | You can learn more in the [Create React App documentation](https://facebook.github.io/create-react-app/docs/getting-started).
43 |
44 | To learn React, check out the [React documentation](https://reactjs.org/).
45 |
46 | ### Code Splitting
47 |
48 | This section has moved here: https://facebook.github.io/create-react-app/docs/code-splitting
49 |
50 | ### Analyzing the Bundle Size
51 |
52 | This section has moved here: https://facebook.github.io/create-react-app/docs/analyzing-the-bundle-size
53 |
54 | ### Making a Progressive Web App
55 |
56 | This section has moved here: https://facebook.github.io/create-react-app/docs/making-a-progressive-web-app
57 |
58 | ### Advanced Configuration
59 |
60 | This section has moved here: https://facebook.github.io/create-react-app/docs/advanced-configuration
61 |
62 | ### Deployment
63 |
64 | This section has moved here: https://facebook.github.io/create-react-app/docs/deployment
65 |
66 | ### `yarn build` fails to minify
67 |
68 | This section has moved here: https://facebook.github.io/create-react-app/docs/troubleshooting#npm-run-build-fails-to-minify
69 |
--------------------------------------------------------------------------------
/client/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "client",
3 | "version": "0.1.0",
4 | "private": true,
5 | "dependencies": {
6 | "@apollo/client": "^3.1.3",
7 | "@testing-library/jest-dom": "^4.2.4",
8 | "@testing-library/react": "^9.3.2",
9 | "@testing-library/user-event": "^7.1.2",
10 | "bootstrap": "^4.5.2",
11 | "classnames": "^2.2.6",
12 | "graphql": "^15.3.0",
13 | "jwt-decode": "^2.2.0",
14 | "moment": "^2.27.0",
15 | "node-sass": "^4.14.1",
16 | "react": "^16.13.1",
17 | "react-bootstrap": "^1.3.0",
18 | "react-dom": "^16.13.1",
19 | "react-router-dom": "^5.2.0",
20 | "react-scripts": "3.4.3",
21 | "subscriptions-transport-ws": "^0.9.18"
22 | },
23 | "scripts": {
24 | "start": "react-scripts start",
25 | "build": "react-scripts build",
26 | "test": "react-scripts test",
27 | "eject": "react-scripts eject"
28 | },
29 | "eslintConfig": {
30 | "extends": "react-app"
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/public/favicon.ico:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/hidjou/node-graphql-react-chat-app/e6d261d6e9d9dfe82333863702c29de1d252a03a/client/public/favicon.ico
--------------------------------------------------------------------------------
/client/public/index.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
12 |
13 |
17 |
18 |
19 |
23 |
32 | MSGME
33 |
34 |
35 |
36 |
37 |
47 |
48 |
49 |
--------------------------------------------------------------------------------
/client/public/logo192.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/hidjou/node-graphql-react-chat-app/e6d261d6e9d9dfe82333863702c29de1d252a03a/client/public/logo192.png
--------------------------------------------------------------------------------
/client/public/logo512.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/hidjou/node-graphql-react-chat-app/e6d261d6e9d9dfe82333863702c29de1d252a03a/client/public/logo512.png
--------------------------------------------------------------------------------
/client/public/manifest.json:
--------------------------------------------------------------------------------
1 | {
2 | "short_name": "React App",
3 | "name": "Create React App Sample",
4 | "icons": [
5 | {
6 | "src": "favicon.ico",
7 | "sizes": "64x64 32x32 24x24 16x16",
8 | "type": "image/x-icon"
9 | },
10 | {
11 | "src": "logo192.png",
12 | "type": "image/png",
13 | "sizes": "192x192"
14 | },
15 | {
16 | "src": "logo512.png",
17 | "type": "image/png",
18 | "sizes": "512x512"
19 | }
20 | ],
21 | "start_url": ".",
22 | "display": "standalone",
23 | "theme_color": "#000000",
24 | "background_color": "#ffffff"
25 | }
26 |
--------------------------------------------------------------------------------
/client/public/robots.txt:
--------------------------------------------------------------------------------
1 | # https://www.robotstxt.org/robotstxt.html
2 | User-agent: *
3 | Disallow:
4 |
--------------------------------------------------------------------------------
/client/src/ApolloProvider.js:
--------------------------------------------------------------------------------
1 | import React from 'react'
2 | import {
3 | ApolloClient,
4 | InMemoryCache,
5 | ApolloProvider as Provider,
6 | createHttpLink,
7 | split,
8 | } from '@apollo/client'
9 | import { setContext } from '@apollo/client/link/context'
10 | import { WebSocketLink } from '@apollo/client/link/ws'
11 | import { getMainDefinition } from '@apollo/client/utilities'
12 |
13 | let httpLink = createHttpLink({
14 | uri: '/graphql/',
15 | })
16 |
17 | const authLink = setContext((_, { headers }) => {
18 | // // get the authentication token from local storage if it exists
19 | const token = localStorage.getItem('token')
20 | // return the headers to the context so httpLink can read them
21 | return {
22 | headers: {
23 | ...headers,
24 | authorization: token ? `Bearer ${token}` : '',
25 | },
26 | }
27 | })
28 |
29 | httpLink = authLink.concat(httpLink)
30 |
31 | const host = window.location.host
32 |
33 | const wsLink = new WebSocketLink({
34 | uri: `ws://${host}/graphql/`,
35 | options: {
36 | reconnect: true,
37 | connectionParams: {
38 | Authorization: `Bearer ${localStorage.getItem('token')}`,
39 | },
40 | },
41 | })
42 |
43 | const splitLink = split(
44 | ({ query }) => {
45 | const definition = getMainDefinition(query)
46 | return (
47 | definition.kind === 'OperationDefinition' &&
48 | definition.operation === 'subscription'
49 | )
50 | },
51 | wsLink,
52 | httpLink
53 | )
54 |
55 | const client = new ApolloClient({
56 | link: splitLink,
57 | cache: new InMemoryCache(),
58 | })
59 |
60 | export default function ApolloProvider(props) {
61 | return
62 | }
63 |
--------------------------------------------------------------------------------
/client/src/App.js:
--------------------------------------------------------------------------------
1 | import React from 'react'
2 | import { Container } from 'react-bootstrap'
3 | import { BrowserRouter, Switch } from 'react-router-dom'
4 |
5 | import ApolloProvider from './ApolloProvider'
6 |
7 | import './App.scss'
8 |
9 | import Home from './pages/home/Home'
10 | import Register from './pages/Register'
11 | import Login from './pages/Login'
12 |
13 | import { AuthProvider } from './context/auth'
14 | import { MessageProvider } from './context/message'
15 | import DynamicRoute from './util/DynamicRoute'
16 |
17 | function App() {
18 | return (
19 |
20 |
21 |
22 |
23 |
24 |
25 |
26 |
27 |
28 |
29 |
30 |
31 |
32 |
33 |
34 | )
35 | }
36 |
37 | export default App
38 |
--------------------------------------------------------------------------------
/client/src/App.scss:
--------------------------------------------------------------------------------
1 | $blue: #1877f2;
2 | $green: #42b72a;
3 |
4 | $border-radius: 0;
5 | $paragraph-margin-bottom: 0;
6 |
7 | $gray-600: rgb(245, 245, 245);
8 |
9 | @import '~bootstrap/scss/bootstrap';
10 |
11 | body {
12 | min-height: 100vh;
13 | background: linear-gradient(
14 | 203deg,
15 | rgba(131, 58, 180, 0.9) 0%,
16 | rgba(253, 29, 29, 0.9) 50%,
17 | rgba(252, 176, 69, 0.9) 100%
18 | );
19 | }
20 |
21 | .user-div:hover {
22 | background-color: rgb(250, 250, 250);
23 | }
24 |
25 | .messages-box {
26 | height: 500px;
27 | overflow-y: scroll;
28 |
29 | scrollbar-width: none;
30 |
31 | &::-webkit-scrollbar {
32 | display: none;
33 | }
34 | }
35 |
36 | .user-image {
37 | width: 50px;
38 | height: 50px;
39 | object-fit: cover;
40 | border-radius: 50%;
41 | }
42 |
43 | .message-input::placeholder {
44 | color: rgba(0, 0, 0, 0.5);
45 | }
46 |
47 | .info-text {
48 | color: rgba(0, 0, 0, 0.5);
49 | text-align: center;
50 | margin-bottom: 0.5rem;
51 | }
52 |
53 | .react-button-popover {
54 | height: 30px;
55 | }
56 |
57 | .react-icon-button {
58 | padding-left: 0.25rem;
59 | padding-right: 0.25rem;
60 | padding-top: 0;
61 | padding-bottom: 0;
62 | font-size: 1rem;
63 | transition: 0.25s;
64 |
65 | &:hover {
66 | text-decoration: none;
67 | font-size: 1.5rem;
68 | }
69 | }
70 |
71 | .btn:focus {
72 | box-shadow: none;
73 | }
74 |
75 | .reactions-div {
76 | position: absolute;
77 | right: -10px;
78 | bottom: -25px;
79 | font-size: 0.9rem;
80 | }
81 |
--------------------------------------------------------------------------------
/client/src/context/auth.js:
--------------------------------------------------------------------------------
1 | import React, { createContext, useReducer, useContext } from 'react'
2 | import jwtDecode from 'jwt-decode'
3 |
4 | const AuthStateContext = createContext()
5 | const AuthDispatchContext = createContext()
6 |
7 | let user = null
8 | const token = localStorage.getItem('token')
9 | if (token) {
10 | const decodedToken = jwtDecode(token)
11 | const expiresAt = new Date(decodedToken.exp * 1000)
12 |
13 | if (new Date() > expiresAt) {
14 | localStorage.removeItem('token')
15 | } else {
16 | user = decodedToken
17 | }
18 | } else console.log('No token found')
19 |
20 | const authReducer = (state, action) => {
21 | switch (action.type) {
22 | case 'LOGIN':
23 | localStorage.setItem('token', action.payload.token)
24 | return {
25 | ...state,
26 | user: action.payload,
27 | }
28 | case 'LOGOUT':
29 | localStorage.removeItem('token')
30 | return {
31 | ...state,
32 | user: null,
33 | }
34 | default:
35 | throw new Error(`Unknown action type: ${action.type}`)
36 | }
37 | }
38 |
39 | export const AuthProvider = ({ children }) => {
40 | const [state, dispatch] = useReducer(authReducer, { user })
41 |
42 | return (
43 |
44 |
45 | {children}
46 |
47 |
48 | )
49 | }
50 |
51 | export const useAuthState = () => useContext(AuthStateContext)
52 | export const useAuthDispatch = () => useContext(AuthDispatchContext)
53 |
--------------------------------------------------------------------------------
/client/src/context/message.js:
--------------------------------------------------------------------------------
1 | import React, { createContext, useReducer, useContext } from 'react'
2 |
3 | const MessageStateContext = createContext()
4 | const MessageDispatchContext = createContext()
5 |
6 | const messageReducer = (state, action) => {
7 | let usersCopy, userIndex
8 | const { username, message, messages, reaction } = action.payload
9 | switch (action.type) {
10 | case 'SET_USERS':
11 | return {
12 | ...state,
13 | users: action.payload,
14 | }
15 | case 'SET_USER_MESSAGES':
16 | usersCopy = [...state.users]
17 |
18 | userIndex = usersCopy.findIndex((u) => u.username === username)
19 |
20 | usersCopy[userIndex] = { ...usersCopy[userIndex], messages }
21 |
22 | return {
23 | ...state,
24 | users: usersCopy,
25 | }
26 | case 'SET_SELECTED_USER':
27 | usersCopy = state.users.map((user) => ({
28 | ...user,
29 | selected: user.username === action.payload,
30 | }))
31 |
32 | return {
33 | ...state,
34 | users: usersCopy,
35 | }
36 | case 'ADD_MESSAGE':
37 | usersCopy = [...state.users]
38 |
39 | userIndex = usersCopy.findIndex((u) => u.username === username)
40 |
41 | message.reactions = []
42 |
43 | let newUser = {
44 | ...usersCopy[userIndex],
45 | messages: usersCopy[userIndex].messages
46 | ? [message, ...usersCopy[userIndex].messages]
47 | : null,
48 | latestMessage: message,
49 | }
50 |
51 | usersCopy[userIndex] = newUser
52 |
53 | return {
54 | ...state,
55 | users: usersCopy,
56 | }
57 |
58 | case 'ADD_REACTION':
59 | usersCopy = [...state.users]
60 |
61 | userIndex = usersCopy.findIndex((u) => u.username === username)
62 |
63 | // Make a shallow copy of user
64 | let userCopy = { ...usersCopy[userIndex] }
65 |
66 | // Find the index of the message that this reaction pertains to
67 | const messageIndex = userCopy.messages?.findIndex(
68 | (m) => m.uuid === reaction.message.uuid
69 | )
70 |
71 | if (messageIndex > -1) {
72 | // Make a shallow copy of user messages
73 | let messagesCopy = [...userCopy.messages]
74 |
75 | // Make a shallow copy of user message reactions
76 | let reactionsCopy = [...messagesCopy[messageIndex].reactions]
77 |
78 | const reactionIndex = reactionsCopy.findIndex(
79 | (r) => r.uuid === reaction.uuid
80 | )
81 |
82 | if (reactionIndex > -1) {
83 | // Reaction exists, update it
84 | reactionsCopy[reactionIndex] = reaction
85 | } else {
86 | // New Reaction, add it
87 | reactionsCopy = [...reactionsCopy, reaction]
88 | }
89 |
90 | messagesCopy[messageIndex] = {
91 | ...messagesCopy[messageIndex],
92 | reactions: reactionsCopy,
93 | }
94 |
95 | userCopy = { ...userCopy, messages: messagesCopy }
96 | usersCopy[userIndex] = userCopy
97 | }
98 |
99 | return {
100 | ...state,
101 | users: usersCopy,
102 | }
103 |
104 | default:
105 | throw new Error(`Unknown action type: ${action.type}`)
106 | }
107 | }
108 |
109 | export const MessageProvider = ({ children }) => {
110 | const [state, dispatch] = useReducer(messageReducer, { users: null })
111 |
112 | return (
113 |
114 |
115 | {children}
116 |
117 |
118 | )
119 | }
120 |
121 | export const useMessageState = () => useContext(MessageStateContext)
122 | export const useMessageDispatch = () => useContext(MessageDispatchContext)
123 |
--------------------------------------------------------------------------------
/client/src/index.js:
--------------------------------------------------------------------------------
1 | import React from 'react'
2 | import ReactDOM from 'react-dom'
3 | import App from './App'
4 |
5 | ReactDOM.render(
6 |
7 |
8 | ,
9 | document.getElementById('root')
10 | )
11 |
--------------------------------------------------------------------------------
/client/src/pages/Login.js:
--------------------------------------------------------------------------------
1 | import React, { useState } from 'react'
2 | import { Row, Col, Form, Button } from 'react-bootstrap'
3 | import { gql, useLazyQuery } from '@apollo/client'
4 | import { Link } from 'react-router-dom'
5 |
6 | import { useAuthDispatch } from '../context/auth'
7 |
8 | const LOGIN_USER = gql`
9 | query login($username: String!, $password: String!) {
10 | login(username: $username, password: $password) {
11 | username
12 | email
13 | createdAt
14 | token
15 | }
16 | }
17 | `
18 |
19 | export default function Register(props) {
20 | const [variables, setVariables] = useState({
21 | username: '',
22 | password: '',
23 | })
24 | const [errors, setErrors] = useState({})
25 |
26 | const dispatch = useAuthDispatch()
27 |
28 | const [loginUser, { loading }] = useLazyQuery(LOGIN_USER, {
29 | onError: (err) => setErrors(err.graphQLErrors[0].extensions.errors),
30 | onCompleted(data) {
31 | dispatch({ type: 'LOGIN', payload: data.login })
32 | window.location.href = '/'
33 | },
34 | })
35 |
36 | const submitLoginForm = (e) => {
37 | e.preventDefault()
38 |
39 | loginUser({ variables })
40 | }
41 |
42 | return (
43 |
44 |
45 | Login
46 |
48 |
49 | {errors.username ?? 'Username'}
50 |
51 |
56 | setVariables({ ...variables, username: e.target.value })
57 | }
58 | />
59 |
60 |
61 |
62 | {errors.password ?? 'Password'}
63 |
64 |
69 | setVariables({ ...variables, password: e.target.value })
70 | }
71 | />
72 |
73 |
74 |
77 |
78 |
79 | Don't have an account? Register
80 |
81 |
82 |
83 |
84 |
85 | )
86 | }
87 |
--------------------------------------------------------------------------------
/client/src/pages/Register.js:
--------------------------------------------------------------------------------
1 | import React, { useState } from 'react'
2 | import { Row, Col, Form, Button } from 'react-bootstrap'
3 | import { gql, useMutation } from '@apollo/client'
4 | import { Link } from 'react-router-dom'
5 |
6 | const REGISTER_USER = gql`
7 | mutation register(
8 | $username: String!
9 | $email: String!
10 | $password: String!
11 | $confirmPassword: String!
12 | ) {
13 | register(
14 | username: $username
15 | email: $email
16 | password: $password
17 | confirmPassword: $confirmPassword
18 | ) {
19 | username
20 | email
21 | createdAt
22 | }
23 | }
24 | `
25 |
26 | export default function Register(props) {
27 | const [variables, setVariables] = useState({
28 | email: '',
29 | username: '',
30 | password: '',
31 | confirmPassword: '',
32 | })
33 | const [errors, setErrors] = useState({})
34 |
35 | const [registerUser, { loading }] = useMutation(REGISTER_USER, {
36 | update: (_, __) => props.history.push('/login'),
37 | onError: (err) => setErrors(err.graphQLErrors[0].extensions.errors),
38 | })
39 |
40 | const submitRegisterForm = (e) => {
41 | e.preventDefault()
42 |
43 | registerUser({ variables })
44 | }
45 |
46 | return (
47 |
48 |
49 | Register
50 |
52 |
53 | {errors.email ?? 'Email address'}
54 |
55 |
60 | setVariables({ ...variables, email: e.target.value })
61 | }
62 | />
63 |
64 |
65 |
66 | {errors.username ?? 'Username'}
67 |
68 |
73 | setVariables({ ...variables, username: e.target.value })
74 | }
75 | />
76 |
77 |
78 |
79 | {errors.password ?? 'Password'}
80 |
81 |
86 | setVariables({ ...variables, password: e.target.value })
87 | }
88 | />
89 |
90 |
91 |
92 | {errors.confirmPassword ?? 'Confirm password'}
93 |
94 |
99 | setVariables({
100 | ...variables,
101 | confirmPassword: e.target.value,
102 | })
103 | }
104 | />
105 |
106 |
107 |
110 |
111 |
112 | Already have an account? Login
113 |
114 |
115 |
116 |
117 |
118 | )
119 | }
120 |
--------------------------------------------------------------------------------
/client/src/pages/home/Home.js:
--------------------------------------------------------------------------------
1 | import React, { Fragment, useEffect } from 'react'
2 | import { Row, Button } from 'react-bootstrap'
3 | import { Link } from 'react-router-dom'
4 | import { gql, useSubscription } from '@apollo/client'
5 |
6 | import { useAuthDispatch, useAuthState } from '../../context/auth'
7 | import { useMessageDispatch } from '../../context/message'
8 |
9 | import Users from './Users'
10 | import Messages from './Messages'
11 |
12 | const NEW_MESSAGE = gql`
13 | subscription newMessage {
14 | newMessage {
15 | uuid
16 | from
17 | to
18 | content
19 | createdAt
20 | }
21 | }
22 | `
23 |
24 | const NEW_REACTION = gql`
25 | subscription newReaction {
26 | newReaction {
27 | uuid
28 | content
29 | message {
30 | uuid
31 | from
32 | to
33 | }
34 | }
35 | }
36 | `
37 |
38 | export default function Home({ history }) {
39 | const authDispatch = useAuthDispatch()
40 | const messageDispatch = useMessageDispatch()
41 |
42 | const { user } = useAuthState()
43 |
44 | const { data: messageData, error: messageError } = useSubscription(
45 | NEW_MESSAGE
46 | )
47 |
48 | const { data: reactionData, error: reactionError } = useSubscription(
49 | NEW_REACTION
50 | )
51 |
52 | useEffect(() => {
53 | if (messageError) console.log(messageError)
54 |
55 | if (messageData) {
56 | const message = messageData.newMessage
57 | const otherUser = user.username === message.to ? message.from : message.to
58 |
59 | messageDispatch({
60 | type: 'ADD_MESSAGE',
61 | payload: {
62 | username: otherUser,
63 | message,
64 | },
65 | })
66 | }
67 | }, [messageError, messageData])
68 |
69 | useEffect(() => {
70 | if (reactionError) console.log(reactionError)
71 |
72 | if (reactionData) {
73 | const reaction = reactionData.newReaction
74 | const otherUser =
75 | user.username === reaction.message.to
76 | ? reaction.message.from
77 | : reaction.message.to
78 |
79 | messageDispatch({
80 | type: 'ADD_REACTION',
81 | payload: {
82 | username: otherUser,
83 | reaction,
84 | },
85 | })
86 | }
87 | }, [reactionError, reactionData])
88 |
89 | const logout = () => {
90 | authDispatch({ type: 'LOGOUT' })
91 | window.location.href = '/login'
92 | }
93 |
94 | return (
95 |
96 |
97 |
98 |
99 |
100 |
101 |
102 |
103 |
106 |
107 |
108 |
109 |
110 |
111 |
112 | )
113 | }
114 |
--------------------------------------------------------------------------------
/client/src/pages/home/Message.js:
--------------------------------------------------------------------------------
1 | import React, { useState } from 'react'
2 | import classNames from 'classnames'
3 | import moment from 'moment'
4 | import { Button, OverlayTrigger, Popover, Tooltip } from 'react-bootstrap'
5 |
6 | import { useAuthState } from '../../context/auth'
7 | import { gql, useMutation } from '@apollo/client'
8 |
9 | const reactions = ['❤️', '😆', '😯', '😢', '😡', '👍', '👎']
10 |
11 | const REACT_TO_MESSAGE = gql`
12 | mutation reactToMessage($uuid: String!, $content: String!) {
13 | reactToMessage(uuid: $uuid, content: $content) {
14 | uuid
15 | }
16 | }
17 | `
18 |
19 | export default function Message({ message }) {
20 | const { user } = useAuthState()
21 | const sent = message.from === user.username
22 | const received = !sent
23 | const [showPopover, setShowPopover] = useState(false)
24 | const reactionIcons = [...new Set(message.reactions.map((r) => r.content))]
25 |
26 | const [reactToMessage] = useMutation(REACT_TO_MESSAGE, {
27 | onError: (err) => console.log(err),
28 | onCompleted: (data) => setShowPopover(false),
29 | })
30 |
31 | const react = (reaction) => {
32 | reactToMessage({ variables: { uuid: message.uuid, content: reaction } })
33 | }
34 |
35 | const reactButton = (
36 |
45 |
46 | {reactions.map((reaction) => (
47 |
55 | ))}
56 |
57 |
58 | }
59 | >
60 |
63 |
64 | )
65 |
66 | return (
67 |
73 | {sent && reactButton}
74 |
78 | {moment(message.createdAt).format('MMM DD, YYYY @ h:mm a')}
79 |
80 | }
81 | transition={false}
82 | >
83 |
89 | {message.reactions.length > 0 && (
90 |
91 | {reactionIcons} {message.reactions.length}
92 |
93 | )}
94 |
95 | {message.content}
96 |
97 |
98 |
99 | {received && reactButton}
100 |
101 | )
102 | }
103 |
--------------------------------------------------------------------------------
/client/src/pages/home/Messages.js:
--------------------------------------------------------------------------------
1 | import React, { Fragment, useEffect, useState } from 'react'
2 | import { gql, useLazyQuery, useMutation } from '@apollo/client'
3 | import { Col, Form } from 'react-bootstrap'
4 |
5 | import { useMessageDispatch, useMessageState } from '../../context/message'
6 |
7 | import Message from './Message'
8 |
9 | const SEND_MESSAGE = gql`
10 | mutation sendMessage($to: String!, $content: String!) {
11 | sendMessage(to: $to, content: $content) {
12 | uuid
13 | from
14 | to
15 | content
16 | createdAt
17 | }
18 | }
19 | `
20 |
21 | const GET_MESSAGES = gql`
22 | query getMessages($from: String!) {
23 | getMessages(from: $from) {
24 | uuid
25 | from
26 | to
27 | content
28 | createdAt
29 | reactions {
30 | uuid
31 | content
32 | }
33 | }
34 | }
35 | `
36 |
37 | export default function Messages() {
38 | const { users } = useMessageState()
39 | const dispatch = useMessageDispatch()
40 | const [content, setContent] = useState('')
41 |
42 | const selectedUser = users?.find((u) => u.selected === true)
43 | const messages = selectedUser?.messages
44 |
45 | const [
46 | getMessages,
47 | { loading: messagesLoading, data: messagesData },
48 | ] = useLazyQuery(GET_MESSAGES)
49 |
50 | const [sendMessage] = useMutation(SEND_MESSAGE, {
51 | onError: (err) => console.log(err),
52 | })
53 |
54 | useEffect(() => {
55 | if (selectedUser && !selectedUser.messages) {
56 | getMessages({ variables: { from: selectedUser.username } })
57 | }
58 | }, [selectedUser])
59 |
60 | useEffect(() => {
61 | if (messagesData) {
62 | dispatch({
63 | type: 'SET_USER_MESSAGES',
64 | payload: {
65 | username: selectedUser.username,
66 | messages: messagesData.getMessages,
67 | },
68 | })
69 | }
70 | }, [messagesData])
71 |
72 | const submitMessage = (e) => {
73 | e.preventDefault()
74 |
75 | if (content.trim() === '' || !selectedUser) return
76 |
77 | setContent('')
78 |
79 | // mutation for sending the message
80 | sendMessage({ variables: { to: selectedUser.username, content } })
81 | }
82 |
83 | let selectedChatMarkup
84 | if (!messages && !messagesLoading) {
85 | selectedChatMarkup = Select a friend
86 | } else if (messagesLoading) {
87 | selectedChatMarkup = Loading..
88 | } else if (messages.length > 0) {
89 | selectedChatMarkup = messages.map((message, index) => (
90 |
91 |
92 | {index === messages.length - 1 && (
93 |
94 |
95 |
96 | )}
97 |
98 | ))
99 | } else if (messages.length === 0) {
100 | selectedChatMarkup = (
101 |
102 | You are now connected! send your first message!
103 |
104 | )
105 | }
106 |
107 | return (
108 |
109 |
110 | {selectedChatMarkup}
111 |
112 |
113 |
115 | setContent(e.target.value)}
121 | />
122 |
127 |
128 |
129 |
130 |
131 | )
132 | }
133 |
--------------------------------------------------------------------------------
/client/src/pages/home/Users.js:
--------------------------------------------------------------------------------
1 | import React from 'react'
2 | import { gql, useQuery } from '@apollo/client'
3 | import { Col, Image } from 'react-bootstrap'
4 | import classNames from 'classnames'
5 |
6 | import { useMessageDispatch, useMessageState } from '../../context/message'
7 |
8 | const GET_USERS = gql`
9 | query getUsers {
10 | getUsers {
11 | username
12 | createdAt
13 | imageUrl
14 | latestMessage {
15 | uuid
16 | from
17 | to
18 | content
19 | createdAt
20 | }
21 | }
22 | }
23 | `
24 |
25 | export default function Users() {
26 | const dispatch = useMessageDispatch()
27 | const { users } = useMessageState()
28 | const selectedUser = users?.find((u) => u.selected === true)?.username
29 |
30 | const { loading } = useQuery(GET_USERS, {
31 | onCompleted: (data) =>
32 | dispatch({ type: 'SET_USERS', payload: data.getUsers }),
33 | onError: (err) => console.log(err),
34 | })
35 |
36 | let usersMarkup
37 | if (!users || loading) {
38 | usersMarkup = Loading..
39 | } else if (users.length === 0) {
40 | usersMarkup = No users have joined yet
41 | } else if (users.length > 0) {
42 | usersMarkup = users.map((user) => {
43 | const selected = selectedUser === user.username
44 | return (
45 |
55 | dispatch({ type: 'SET_SELECTED_USER', payload: user.username })
56 | }
57 | >
58 |
65 |
66 |
{user.username}
67 |
68 | {user.latestMessage
69 | ? user.latestMessage.content
70 | : 'You are now connected!'}
71 |
72 |
73 |
74 | )
75 | })
76 | }
77 | return (
78 |
79 | {usersMarkup}
80 |
81 | )
82 | }
83 |
--------------------------------------------------------------------------------
/client/src/util/DynamicRoute.js:
--------------------------------------------------------------------------------
1 | import React from 'react'
2 | import { Route, Redirect } from 'react-router-dom'
3 |
4 | import { useAuthState } from '../context/auth'
5 |
6 | export default function DynamicRoute(props) {
7 | const { user } = useAuthState()
8 |
9 | if (props.authenticated && !user) {
10 | return
11 | } else if (props.guest && user) {
12 | return
13 | } else {
14 | return
15 | }
16 | }
17 |
--------------------------------------------------------------------------------
/config/config.js:
--------------------------------------------------------------------------------
1 | require('dotenv').config('../.env')
2 |
3 | module.exports = {
4 | username: process.env.DB_USERNAME,
5 | password: process.env.DB_PASSWORD,
6 | database: process.env.DB_DATABASE,
7 | host: process.env.DB_HOST,
8 | dialect: process.env.DB_DIALECT,
9 | }
10 |
--------------------------------------------------------------------------------
/deploy.md:
--------------------------------------------------------------------------------
1 | # Deploy a NodeJS React app to AWS EC2
2 |
3 | ## Prepare the code
4 |
5 | 1. Update client links to be relative to new host
6 | 2. Create a .env and use env vars for DB credentials
7 | 3. Push the code to a git repo
8 |
9 | ## Set up the EC2 instance:
10 |
11 | 1. Launch an EC2 instance
12 | 2. Connect to it and install NodeJS on it
13 |
14 | ## Set up the RDS instance:
15 |
16 | 1. Launch an RDS instance
17 | 2. Set up a user and password for the server to access it
18 | 3. Connect to it and create your app's schema
19 |
20 | ## Set up nginx
21 |
22 | 1. Install nginx
23 | 2. Update server blocks and map / to client and /graphql to server
24 | 3. Build client files and copy them to where nginx is serving from
25 |
26 | ## Set up pm2
27 |
28 | 1. Install pm2
29 | 2. Add the server as a process
30 |
31 | ## Done
32 |
--------------------------------------------------------------------------------
/graphql/resolvers/index.js:
--------------------------------------------------------------------------------
1 | const userResolvers = require('./users')
2 | const messageResolvers = require('./messages')
3 |
4 | const { User, Message } = require('../../models')
5 |
6 | module.exports = {
7 | Message: {
8 | createdAt: (parent) => parent.createdAt.toISOString(),
9 | },
10 | Reaction: {
11 | createdAt: (parent) => parent.createdAt.toISOString(),
12 | message: async (parent) => await Message.findByPk(parent.messageId),
13 | user: async (parent) =>
14 | await User.findByPk(parent.userId, {
15 | attributes: ['username', 'imageUrl', 'createdAt'],
16 | }),
17 | },
18 | User: {
19 | createdAt: (parent) => parent.createdAt.toISOString(),
20 | },
21 | Query: {
22 | ...userResolvers.Query,
23 | ...messageResolvers.Query,
24 | },
25 | Mutation: {
26 | ...userResolvers.Mutation,
27 | ...messageResolvers.Mutation,
28 | },
29 | Subscription: {
30 | ...messageResolvers.Subscription,
31 | },
32 | }
33 |
--------------------------------------------------------------------------------
/graphql/resolvers/messages.js:
--------------------------------------------------------------------------------
1 | const {
2 | UserInputError,
3 | AuthenticationError,
4 | ForbiddenError,
5 | withFilter,
6 | } = require('apollo-server')
7 | const { Op } = require('sequelize')
8 |
9 | const { Message, User, Reaction } = require('../../models')
10 |
11 | module.exports = {
12 | Query: {
13 | getMessages: async (parent, { from }, { user }) => {
14 | try {
15 | if (!user) throw new AuthenticationError('Unauthenticated')
16 |
17 | const otherUser = await User.findOne({
18 | where: { username: from },
19 | })
20 | if (!otherUser) throw new UserInputError('User not found')
21 |
22 | const usernames = [user.username, otherUser.username]
23 |
24 | const messages = await Message.findAll({
25 | where: {
26 | from: { [Op.in]: usernames },
27 | to: { [Op.in]: usernames },
28 | },
29 | order: [['createdAt', 'DESC']],
30 | include: [{ model: Reaction, as: 'reactions' }],
31 | })
32 |
33 | return messages
34 | } catch (err) {
35 | console.log(err)
36 | throw err
37 | }
38 | },
39 | },
40 | Mutation: {
41 | sendMessage: async (parent, { to, content }, { user, pubsub }) => {
42 | try {
43 | if (!user) throw new AuthenticationError('Unauthenticated')
44 |
45 | const recipient = await User.findOne({ where: { username: to } })
46 |
47 | if (!recipient) {
48 | throw new UserInputError('User not found')
49 | } else if (recipient.username === user.username) {
50 | throw new UserInputError('You cant message yourself')
51 | }
52 |
53 | if (content.trim() === '') {
54 | throw new UserInputError('Message is empty')
55 | }
56 |
57 | const message = await Message.create({
58 | from: user.username,
59 | to,
60 | content,
61 | })
62 |
63 | pubsub.publish('NEW_MESSAGE', { newMessage: message })
64 |
65 | return message
66 | } catch (err) {
67 | console.log(err)
68 | throw err
69 | }
70 | },
71 | reactToMessage: async (_, { uuid, content }, { user, pubsub }) => {
72 | const reactions = ['❤️', '😆', '😯', '😢', '😡', '👍', '👎']
73 |
74 | try {
75 | // Validate reaction content
76 | if (!reactions.includes(content)) {
77 | throw new UserInputError('Invalid reaction')
78 | }
79 |
80 | // Get user
81 | const username = user ? user.username : ''
82 | user = await User.findOne({ where: { username } })
83 | if (!user) throw new AuthenticationError('Unauthenticated')
84 |
85 | // Get message
86 | const message = await Message.findOne({ where: { uuid } })
87 | if (!message) throw new UserInputError('message not found')
88 |
89 | if (message.from !== user.username && message.to !== user.username) {
90 | throw new ForbiddenError('Unauthorized')
91 | }
92 |
93 | let reaction = await Reaction.findOne({
94 | where: { messageId: message.id, userId: user.id },
95 | })
96 |
97 | if (reaction) {
98 | // Reaction exists, update it
99 | reaction.content = content
100 | await reaction.save()
101 | } else {
102 | // Reaction doesnt exists, create it
103 | reaction = await Reaction.create({
104 | messageId: message.id,
105 | userId: user.id,
106 | content,
107 | })
108 | }
109 |
110 | pubsub.publish('NEW_REACTION', { newReaction: reaction })
111 |
112 | return reaction
113 | } catch (err) {
114 | throw err
115 | }
116 | },
117 | },
118 | Subscription: {
119 | newMessage: {
120 | subscribe: withFilter(
121 | (_, __, { pubsub, user }) => {
122 | if (!user) throw new AuthenticationError('Unauthenticated')
123 | return pubsub.asyncIterator('NEW_MESSAGE')
124 | },
125 | ({ newMessage }, _, { user }) => {
126 | if (
127 | newMessage.from === user.username ||
128 | newMessage.to === user.username
129 | ) {
130 | return true
131 | }
132 |
133 | return false
134 | }
135 | ),
136 | },
137 | newReaction: {
138 | subscribe: withFilter(
139 | (_, __, { pubsub, user }) => {
140 | if (!user) throw new AuthenticationError('Unauthenticated')
141 | return pubsub.asyncIterator('NEW_REACTION')
142 | },
143 | async ({ newReaction }, _, { user }) => {
144 | const message = await newReaction.getMessage()
145 | if (message.from === user.username || message.to === user.username) {
146 | return true
147 | }
148 |
149 | return false
150 | }
151 | ),
152 | },
153 | },
154 | }
155 |
--------------------------------------------------------------------------------
/graphql/resolvers/users.js:
--------------------------------------------------------------------------------
1 | const bcrypt = require('bcryptjs')
2 | const { UserInputError, AuthenticationError } = require('apollo-server')
3 | const jwt = require('jsonwebtoken')
4 | const { Op } = require('sequelize')
5 |
6 | const { Message, User } = require('../../models')
7 |
8 | module.exports = {
9 | Query: {
10 | getUsers: async (_, __, { user }) => {
11 | try {
12 | if (!user) throw new AuthenticationError('Unauthenticated')
13 |
14 | let users = await User.findAll({
15 | attributes: ['username', 'imageUrl', 'createdAt'],
16 | where: { username: { [Op.ne]: user.username } },
17 | })
18 |
19 | const allUserMessages = await Message.findAll({
20 | where: {
21 | [Op.or]: [{ from: user.username }, { to: user.username }],
22 | },
23 | order: [['createdAt', 'DESC']],
24 | })
25 |
26 | users = users.map((otherUser) => {
27 | const latestMessage = allUserMessages.find(
28 | (m) => m.from === otherUser.username || m.to === otherUser.username
29 | )
30 | otherUser.latestMessage = latestMessage
31 | return otherUser
32 | })
33 |
34 | return users
35 | } catch (err) {
36 | console.log(err)
37 | throw err
38 | }
39 | },
40 | login: async (_, args) => {
41 | const { username, password } = args
42 | let errors = {}
43 |
44 | try {
45 | if (username.trim() === '')
46 | errors.username = 'username must not be empty'
47 | if (password === '') errors.password = 'password must not be empty'
48 |
49 | if (Object.keys(errors).length > 0) {
50 | throw new UserInputError('bad input', { errors })
51 | }
52 |
53 | const user = await User.findOne({
54 | where: { username },
55 | })
56 |
57 | if (!user) {
58 | errors.username = 'user not found'
59 | throw new UserInputError('user not found', { errors })
60 | }
61 |
62 | const correctPassword = await bcrypt.compare(password, user.password)
63 |
64 | if (!correctPassword) {
65 | errors.password = 'password is incorrect'
66 | throw new UserInputError('password is incorrect', { errors })
67 | }
68 |
69 | const token = jwt.sign({ username }, process.env.JWT_SECRET, {
70 | expiresIn: 60 * 60,
71 | })
72 |
73 | return {
74 | ...user.toJSON(),
75 | token,
76 | }
77 | } catch (err) {
78 | console.log(err)
79 | throw err
80 | }
81 | },
82 | },
83 | Mutation: {
84 | register: async (_, args) => {
85 | let { username, email, password, confirmPassword } = args
86 | let errors = {}
87 |
88 | try {
89 | // Validate input data
90 | if (email.trim() === '') errors.email = 'email must not be empty'
91 | if (username.trim() === '')
92 | errors.username = 'username must not be empty'
93 | if (password.trim() === '')
94 | errors.password = 'password must not be empty'
95 | if (confirmPassword.trim() === '')
96 | errors.confirmPassword = 'repeat password must not be empty'
97 |
98 | if (password !== confirmPassword)
99 | errors.confirmPassword = 'passwords must match'
100 |
101 | // // Check if username / email exists
102 | // const userByUsername = await User.findOne({ where: { username } })
103 | // const userByEmail = await User.findOne({ where: { email } })
104 |
105 | // if (userByUsername) errors.username = 'Username is taken'
106 | // if (userByEmail) errors.email = 'Email is taken'
107 |
108 | if (Object.keys(errors).length > 0) {
109 | throw errors
110 | }
111 |
112 | // Hash password
113 | password = await bcrypt.hash(password, 6)
114 |
115 | // Create user
116 | const user = await User.create({
117 | username,
118 | email,
119 | password,
120 | })
121 |
122 | // Return user
123 | return user
124 | } catch (err) {
125 | console.log(err)
126 | if (err.name === 'SequelizeUniqueConstraintError') {
127 | err.errors.forEach(
128 | (e) => (errors[e.path] = `${e.path} is already taken`)
129 | )
130 | } else if (err.name === 'SequelizeValidationError') {
131 | err.errors.forEach((e) => (errors[e.path] = e.message))
132 | }
133 | throw new UserInputError('Bad input', { errors })
134 | }
135 | },
136 | },
137 | }
138 |
--------------------------------------------------------------------------------
/graphql/typeDefs.js:
--------------------------------------------------------------------------------
1 | const { gql } = require('apollo-server')
2 |
3 | module.exports = gql`
4 | type User {
5 | username: String!
6 | email: String
7 | createdAt: String!
8 | token: String
9 | imageUrl: String
10 | latestMessage: Message
11 | }
12 | type Message {
13 | uuid: String!
14 | content: String!
15 | from: String!
16 | to: String!
17 | createdAt: String!
18 | reactions: [Reaction]
19 | }
20 | type Reaction {
21 | uuid: String!
22 | content: String!
23 | createdAt: String!
24 | message: Message!
25 | user: User!
26 | }
27 | type Query {
28 | getUsers: [User]!
29 | login(username: String!, password: String!): User!
30 | getMessages(from: String!): [Message]!
31 | }
32 | type Mutation {
33 | register(
34 | username: String!
35 | email: String!
36 | password: String!
37 | confirmPassword: String!
38 | ): User!
39 | sendMessage(to: String!, content: String!): Message!
40 | reactToMessage(uuid: String!, content: String!): Reaction!
41 | }
42 | type Subscription {
43 | newMessage: Message!
44 | newReaction: Reaction!
45 | }
46 | `
47 |
--------------------------------------------------------------------------------
/migrations/20200809134724-create-user.js:
--------------------------------------------------------------------------------
1 | 'use strict'
2 | module.exports = {
3 | up: async (queryInterface, Sequelize) => {
4 | await queryInterface.createTable('users', {
5 | id: {
6 | allowNull: false,
7 | autoIncrement: true,
8 | primaryKey: true,
9 | type: Sequelize.INTEGER,
10 | },
11 | username: {
12 | type: Sequelize.STRING(20),
13 | allowNull: false,
14 | unique: true,
15 | },
16 | email: {
17 | type: Sequelize.STRING(100),
18 | allowNull: false,
19 | unique: true,
20 | },
21 | password: {
22 | type: Sequelize.STRING,
23 | allowNull: false,
24 | },
25 | imageUrl: Sequelize.STRING,
26 | createdAt: {
27 | allowNull: false,
28 | type: Sequelize.DATE,
29 | },
30 | updatedAt: {
31 | allowNull: false,
32 | type: Sequelize.DATE,
33 | },
34 | })
35 | },
36 | down: async (queryInterface, Sequelize) => {
37 | await queryInterface.dropTable('users')
38 | },
39 | }
40 |
--------------------------------------------------------------------------------
/migrations/20200823130528-create-message.js:
--------------------------------------------------------------------------------
1 | 'use strict'
2 | module.exports = {
3 | up: async (queryInterface, Sequelize) => {
4 | await queryInterface.createTable('messages', {
5 | id: {
6 | allowNull: false,
7 | autoIncrement: true,
8 | primaryKey: true,
9 | type: Sequelize.INTEGER,
10 | },
11 | content: {
12 | type: Sequelize.STRING,
13 | allowNull: false,
14 | },
15 | uuid: {
16 | type: Sequelize.UUID,
17 | defaultValue: Sequelize.UUIDV4,
18 | allowNull: false,
19 | },
20 | from: {
21 | type: Sequelize.STRING,
22 | allowNull: false,
23 | },
24 | to: {
25 | type: Sequelize.STRING,
26 | allowNull: false,
27 | },
28 | createdAt: {
29 | allowNull: false,
30 | type: Sequelize.DATE,
31 | },
32 | updatedAt: {
33 | allowNull: false,
34 | type: Sequelize.DATE,
35 | },
36 | })
37 | },
38 | down: async (queryInterface, Sequelize) => {
39 | await queryInterface.dropTable('messages')
40 | },
41 | }
42 |
--------------------------------------------------------------------------------
/migrations/20200910194215-create-reaction.js:
--------------------------------------------------------------------------------
1 | 'use strict'
2 | module.exports = {
3 | up: async (queryInterface, Sequelize) => {
4 | await queryInterface.createTable('reactions', {
5 | id: {
6 | allowNull: false,
7 | autoIncrement: true,
8 | primaryKey: true,
9 | type: Sequelize.INTEGER,
10 | },
11 | content: {
12 | type: Sequelize.STRING,
13 | allowNull: false,
14 | },
15 | uuid: {
16 | type: Sequelize.UUID,
17 | defaultValue: Sequelize.UUIDV4,
18 | },
19 | messageId: {
20 | type: Sequelize.INTEGER,
21 | allowNull: false,
22 | },
23 | userId: {
24 | type: Sequelize.INTEGER,
25 | allowNull: false,
26 | },
27 | createdAt: {
28 | allowNull: false,
29 | type: Sequelize.DATE,
30 | },
31 | updatedAt: {
32 | allowNull: false,
33 | type: Sequelize.DATE,
34 | },
35 | })
36 | },
37 | down: async (queryInterface, Sequelize) => {
38 | await queryInterface.dropTable('reactions')
39 | },
40 | }
41 |
--------------------------------------------------------------------------------
/models/Message.js:
--------------------------------------------------------------------------------
1 | 'use strict'
2 | const { Model } = require('sequelize')
3 | module.exports = (sequelize, DataTypes) => {
4 | class Message extends Model {
5 | /**
6 | * Helper method for defining associations.
7 | * This method is not a part of Sequelize lifecycle.
8 | * The `models/index` file will call this method automatically.
9 | */
10 | static associate({ Reaction }) {
11 | this.hasMany(Reaction, { as: 'reactions' }) // reactions
12 | }
13 | }
14 | Message.init(
15 | {
16 | content: {
17 | type: DataTypes.STRING,
18 | allowNull: false,
19 | },
20 | uuid: {
21 | type: DataTypes.UUID,
22 | defaultValue: DataTypes.UUIDV4,
23 | allowNull: false,
24 | },
25 | from: {
26 | type: DataTypes.STRING,
27 | allowNull: false,
28 | },
29 | to: {
30 | type: DataTypes.STRING,
31 | allowNull: false,
32 | },
33 | },
34 | {
35 | sequelize,
36 | modelName: 'Message',
37 | tableName: 'messages',
38 | }
39 | )
40 | return Message
41 | }
42 |
--------------------------------------------------------------------------------
/models/Reaction.js:
--------------------------------------------------------------------------------
1 | 'use strict'
2 | const { Model } = require('sequelize')
3 | module.exports = (sequelize, DataTypes) => {
4 | class Reaction extends Model {
5 | /**
6 | * Helper method for defining associations.
7 | * This method is not a part of Sequelize lifecycle.
8 | * The `models/index` file will call this method automatically.
9 | */
10 | static associate({ User, Message }) {
11 | // define association here
12 | this.belongsTo(Message, { foreignKey: 'messageId' }) // MessageId, messageId
13 | this.belongsTo(User, { foreignKey: 'userId' })
14 | }
15 | }
16 | Reaction.init(
17 | {
18 | content: {
19 | type: DataTypes.STRING,
20 | allowNull: false,
21 | },
22 | uuid: {
23 | type: DataTypes.UUID,
24 | defaultValue: DataTypes.UUIDV4,
25 | },
26 | messageId: {
27 | type: DataTypes.INTEGER,
28 | allowNull: false,
29 | },
30 | userId: {
31 | type: DataTypes.INTEGER,
32 | allowNull: false,
33 | },
34 | },
35 | {
36 | sequelize,
37 | modelName: 'Reaction',
38 | tableName: 'reactions',
39 | }
40 | )
41 | return Reaction
42 | }
43 |
--------------------------------------------------------------------------------
/models/index.js:
--------------------------------------------------------------------------------
1 | 'use strict';
2 |
3 | const fs = require('fs');
4 | const path = require('path');
5 | const Sequelize = require('sequelize');
6 | const basename = path.basename(__filename);
7 | const config = require(__dirname + '/../config/config.js');
8 | const db = {};
9 |
10 |
11 | const sequelize = new Sequelize(config.database, config.username, config.password, config);
12 |
13 |
14 | fs
15 | .readdirSync(__dirname)
16 | .filter(file => {
17 | return (file.indexOf('.') !== 0) && (file !== basename) && (file.slice(-3) === '.js');
18 | })
19 | .forEach(file => {
20 | const model = require(path.join(__dirname, file))(sequelize, Sequelize.DataTypes);
21 | db[model.name] = model;
22 | });
23 |
24 | Object.keys(db).forEach(modelName => {
25 | if (db[modelName].associate) {
26 | db[modelName].associate(db);
27 | }
28 | });
29 |
30 | db.sequelize = sequelize;
31 | db.Sequelize = Sequelize;
32 |
33 | module.exports = db;
34 |
--------------------------------------------------------------------------------
/models/user.js:
--------------------------------------------------------------------------------
1 | 'use strict'
2 | const { Model } = require('sequelize')
3 | module.exports = (sequelize, DataTypes) => {
4 | class User extends Model {
5 | /**
6 | * Helper method for defining associations.
7 | * This method is not a part of Sequelize lifecycle.
8 | * The `models/index` file will call this method automatically.
9 | */
10 | static associate(models) {
11 | // define association here
12 | }
13 | }
14 | User.init(
15 | {
16 | username: {
17 | type: DataTypes.STRING(20),
18 | allowNull: false,
19 | unique: true,
20 | },
21 | email: {
22 | type: DataTypes.STRING(100),
23 | allowNull: false,
24 | unique: true,
25 | validate: {
26 | isEmail: {
27 | args: true,
28 | msg: 'must be a valid email address',
29 | },
30 | },
31 | },
32 | password: {
33 | type: DataTypes.STRING,
34 | allowNull: false,
35 | },
36 | imageUrl: DataTypes.STRING,
37 | },
38 | {
39 | sequelize,
40 | modelName: 'User',
41 | tableName: 'users',
42 | }
43 | )
44 |
45 | return User
46 | }
47 |
--------------------------------------------------------------------------------
/package-lock.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "chat",
3 | "version": "1.0.0",
4 | "lockfileVersion": 1,
5 | "requires": true,
6 | "dependencies": {
7 | "@apollo/protobufjs": {
8 | "version": "1.0.4",
9 | "resolved": "https://registry.npmjs.org/@apollo/protobufjs/-/protobufjs-1.0.4.tgz",
10 | "integrity": "sha512-EE3zx+/D/wur/JiLp6VCiw1iYdyy1lCJMf8CGPkLeDt5QJrN4N8tKFx33Ah4V30AUQzMk7Uz4IXKZ1LOj124gA==",
11 | "requires": {
12 | "@protobufjs/aspromise": "^1.1.2",
13 | "@protobufjs/base64": "^1.1.2",
14 | "@protobufjs/codegen": "^2.0.4",
15 | "@protobufjs/eventemitter": "^1.1.0",
16 | "@protobufjs/fetch": "^1.1.0",
17 | "@protobufjs/float": "^1.0.2",
18 | "@protobufjs/inquire": "^1.1.0",
19 | "@protobufjs/path": "^1.1.2",
20 | "@protobufjs/pool": "^1.1.0",
21 | "@protobufjs/utf8": "^1.1.0",
22 | "@types/long": "^4.0.0",
23 | "@types/node": "^10.1.0",
24 | "long": "^4.0.0"
25 | },
26 | "dependencies": {
27 | "@types/node": {
28 | "version": "10.17.28",
29 | "resolved": "https://registry.npmjs.org/@types/node/-/node-10.17.28.tgz",
30 | "integrity": "sha512-dzjES1Egb4c1a89C7lKwQh8pwjYmlOAG9dW1pBgxEk57tMrLnssOfEthz8kdkNaBd7lIqQx7APm5+mZ619IiCQ=="
31 | }
32 | }
33 | },
34 | "@apollographql/apollo-tools": {
35 | "version": "0.4.8",
36 | "resolved": "https://registry.npmjs.org/@apollographql/apollo-tools/-/apollo-tools-0.4.8.tgz",
37 | "integrity": "sha512-W2+HB8Y7ifowcf3YyPHgDI05izyRtOeZ4MqIr7LbTArtmJ0ZHULWpn84SGMW7NAvTV1tFExpHlveHhnXuJfuGA==",
38 | "requires": {
39 | "apollo-env": "^0.6.5"
40 | }
41 | },
42 | "@apollographql/graphql-playground-html": {
43 | "version": "1.6.26",
44 | "resolved": "https://registry.npmjs.org/@apollographql/graphql-playground-html/-/graphql-playground-html-1.6.26.tgz",
45 | "integrity": "sha512-XAwXOIab51QyhBxnxySdK3nuMEUohhDsHQ5Rbco/V1vjlP75zZ0ZLHD9dTpXTN8uxKxopb2lUvJTq+M4g2Q0HQ==",
46 | "requires": {
47 | "xss": "^1.0.6"
48 | }
49 | },
50 | "@protobufjs/aspromise": {
51 | "version": "1.1.2",
52 | "resolved": "https://registry.npmjs.org/@protobufjs/aspromise/-/aspromise-1.1.2.tgz",
53 | "integrity": "sha1-m4sMxmPWaafY9vXQiToU00jzD78="
54 | },
55 | "@protobufjs/base64": {
56 | "version": "1.1.2",
57 | "resolved": "https://registry.npmjs.org/@protobufjs/base64/-/base64-1.1.2.tgz",
58 | "integrity": "sha512-AZkcAA5vnN/v4PDqKyMR5lx7hZttPDgClv83E//FMNhR2TMcLUhfRUBHCmSl0oi9zMgDDqRUJkSxO3wm85+XLg=="
59 | },
60 | "@protobufjs/codegen": {
61 | "version": "2.0.4",
62 | "resolved": "https://registry.npmjs.org/@protobufjs/codegen/-/codegen-2.0.4.tgz",
63 | "integrity": "sha512-YyFaikqM5sH0ziFZCN3xDC7zeGaB/d0IUb9CATugHWbd1FRFwWwt4ld4OYMPWu5a3Xe01mGAULCdqhMlPl29Jg=="
64 | },
65 | "@protobufjs/eventemitter": {
66 | "version": "1.1.0",
67 | "resolved": "https://registry.npmjs.org/@protobufjs/eventemitter/-/eventemitter-1.1.0.tgz",
68 | "integrity": "sha1-NVy8mLr61ZePntCV85diHx0Ga3A="
69 | },
70 | "@protobufjs/fetch": {
71 | "version": "1.1.0",
72 | "resolved": "https://registry.npmjs.org/@protobufjs/fetch/-/fetch-1.1.0.tgz",
73 | "integrity": "sha1-upn7WYYUr2VwDBYZ/wbUVLDYTEU=",
74 | "requires": {
75 | "@protobufjs/aspromise": "^1.1.1",
76 | "@protobufjs/inquire": "^1.1.0"
77 | }
78 | },
79 | "@protobufjs/float": {
80 | "version": "1.0.2",
81 | "resolved": "https://registry.npmjs.org/@protobufjs/float/-/float-1.0.2.tgz",
82 | "integrity": "sha1-Xp4avctz/Ap8uLKR33jIy9l7h9E="
83 | },
84 | "@protobufjs/inquire": {
85 | "version": "1.1.0",
86 | "resolved": "https://registry.npmjs.org/@protobufjs/inquire/-/inquire-1.1.0.tgz",
87 | "integrity": "sha1-/yAOPnzyQp4tyvwRQIKOjMY48Ik="
88 | },
89 | "@protobufjs/path": {
90 | "version": "1.1.2",
91 | "resolved": "https://registry.npmjs.org/@protobufjs/path/-/path-1.1.2.tgz",
92 | "integrity": "sha1-bMKyDFya1q0NzP0hynZz2Nf79o0="
93 | },
94 | "@protobufjs/pool": {
95 | "version": "1.1.0",
96 | "resolved": "https://registry.npmjs.org/@protobufjs/pool/-/pool-1.1.0.tgz",
97 | "integrity": "sha1-Cf0V8tbTq/qbZbw2ZQbWrXhG/1Q="
98 | },
99 | "@protobufjs/utf8": {
100 | "version": "1.1.0",
101 | "resolved": "https://registry.npmjs.org/@protobufjs/utf8/-/utf8-1.1.0.tgz",
102 | "integrity": "sha1-p3c2C1s5oaLlEG+OhY8v0tBgxXA="
103 | },
104 | "@sindresorhus/is": {
105 | "version": "0.14.0",
106 | "resolved": "https://registry.npmjs.org/@sindresorhus/is/-/is-0.14.0.tgz",
107 | "integrity": "sha512-9NET910DNaIPngYnLLPeg+Ogzqsi9uM4mSboU5y6p8S5DzMTVEsJZrawi+BoDNUVBa2DhJqQYUFvMDfgU062LQ==",
108 | "dev": true
109 | },
110 | "@szmarczak/http-timer": {
111 | "version": "1.1.2",
112 | "resolved": "https://registry.npmjs.org/@szmarczak/http-timer/-/http-timer-1.1.2.tgz",
113 | "integrity": "sha512-XIB2XbzHTN6ieIjfIMV9hlVcfPU26s2vafYWQcZHWXHOxiaRZYEDKEwdl129Zyg50+foYV2jCgtrqSA6qNuNSA==",
114 | "dev": true,
115 | "requires": {
116 | "defer-to-connect": "^1.0.1"
117 | }
118 | },
119 | "@types/accepts": {
120 | "version": "1.3.5",
121 | "resolved": "https://registry.npmjs.org/@types/accepts/-/accepts-1.3.5.tgz",
122 | "integrity": "sha512-jOdnI/3qTpHABjM5cx1Hc0sKsPoYCp+DP/GJRGtDlPd7fiV9oXGGIcjW/ZOxLIvjGz8MA+uMZI9metHlgqbgwQ==",
123 | "requires": {
124 | "@types/node": "*"
125 | }
126 | },
127 | "@types/body-parser": {
128 | "version": "1.19.0",
129 | "resolved": "https://registry.npmjs.org/@types/body-parser/-/body-parser-1.19.0.tgz",
130 | "integrity": "sha512-W98JrE0j2K78swW4ukqMleo8R7h/pFETjM2DQ90MF6XK2i4LO4W3gQ71Lt4w3bfm2EvVSyWHplECvB5sK22yFQ==",
131 | "requires": {
132 | "@types/connect": "*",
133 | "@types/node": "*"
134 | }
135 | },
136 | "@types/color-name": {
137 | "version": "1.1.1",
138 | "resolved": "https://registry.npmjs.org/@types/color-name/-/color-name-1.1.1.tgz",
139 | "integrity": "sha512-rr+OQyAjxze7GgWrSaJwydHStIhHq2lvY3BOC2Mj7KnzI7XK0Uw1TOOdI9lDoajEbSWLiYgoo4f1R51erQfhPQ==",
140 | "dev": true
141 | },
142 | "@types/connect": {
143 | "version": "3.4.33",
144 | "resolved": "https://registry.npmjs.org/@types/connect/-/connect-3.4.33.tgz",
145 | "integrity": "sha512-2+FrkXY4zllzTNfJth7jOqEHC+enpLeGslEhpnTAkg21GkRrWV4SsAtqchtT4YS9/nODBU2/ZfsBY2X4J/dX7A==",
146 | "requires": {
147 | "@types/node": "*"
148 | }
149 | },
150 | "@types/content-disposition": {
151 | "version": "0.5.3",
152 | "resolved": "https://registry.npmjs.org/@types/content-disposition/-/content-disposition-0.5.3.tgz",
153 | "integrity": "sha512-P1bffQfhD3O4LW0ioENXUhZ9OIa0Zn+P7M+pWgkCKaT53wVLSq0mrKksCID/FGHpFhRSxRGhgrQmfhRuzwtKdg=="
154 | },
155 | "@types/cookies": {
156 | "version": "0.7.4",
157 | "resolved": "https://registry.npmjs.org/@types/cookies/-/cookies-0.7.4.tgz",
158 | "integrity": "sha512-oTGtMzZZAVuEjTwCjIh8T8FrC8n/uwy+PG0yTvQcdZ7etoel7C7/3MSd7qrukENTgQtotG7gvBlBojuVs7X5rw==",
159 | "requires": {
160 | "@types/connect": "*",
161 | "@types/express": "*",
162 | "@types/keygrip": "*",
163 | "@types/node": "*"
164 | }
165 | },
166 | "@types/cors": {
167 | "version": "2.8.7",
168 | "resolved": "https://registry.npmjs.org/@types/cors/-/cors-2.8.7.tgz",
169 | "integrity": "sha512-sOdDRU3oRS7LBNTIqwDkPJyq0lpHYcbMTt0TrjzsXbk/e37hcLTH6eZX7CdbDeN0yJJvzw9hFBZkbtCSbk/jAQ==",
170 | "requires": {
171 | "@types/express": "*"
172 | }
173 | },
174 | "@types/express": {
175 | "version": "4.17.7",
176 | "resolved": "https://registry.npmjs.org/@types/express/-/express-4.17.7.tgz",
177 | "integrity": "sha512-dCOT5lcmV/uC2J9k0rPafATeeyz+99xTt54ReX11/LObZgfzJqZNcW27zGhYyX+9iSEGXGt5qLPwRSvBZcLvtQ==",
178 | "requires": {
179 | "@types/body-parser": "*",
180 | "@types/express-serve-static-core": "*",
181 | "@types/qs": "*",
182 | "@types/serve-static": "*"
183 | }
184 | },
185 | "@types/express-serve-static-core": {
186 | "version": "4.17.9",
187 | "resolved": "https://registry.npmjs.org/@types/express-serve-static-core/-/express-serve-static-core-4.17.9.tgz",
188 | "integrity": "sha512-DG0BYg6yO+ePW+XoDENYz8zhNGC3jDDEpComMYn7WJc4mY1Us8Rw9ax2YhJXxpyk2SF47PQAoQ0YyVT1a0bEkA==",
189 | "requires": {
190 | "@types/node": "*",
191 | "@types/qs": "*",
192 | "@types/range-parser": "*"
193 | }
194 | },
195 | "@types/fs-capacitor": {
196 | "version": "2.0.0",
197 | "resolved": "https://registry.npmjs.org/@types/fs-capacitor/-/fs-capacitor-2.0.0.tgz",
198 | "integrity": "sha512-FKVPOCFbhCvZxpVAMhdBdTfVfXUpsh15wFHgqOKxh9N9vzWZVuWCSijZ5T4U34XYNnuj2oduh6xcs1i+LPI+BQ==",
199 | "requires": {
200 | "@types/node": "*"
201 | }
202 | },
203 | "@types/graphql-upload": {
204 | "version": "8.0.3",
205 | "resolved": "https://registry.npmjs.org/@types/graphql-upload/-/graphql-upload-8.0.3.tgz",
206 | "integrity": "sha512-hmLg9pCU/GmxBscg8GCr1vmSoEmbItNNxdD5YH2TJkXm//8atjwuprB+xJBK714JG1dkxbbhp5RHX+Pz1KsCMA==",
207 | "requires": {
208 | "@types/express": "*",
209 | "@types/fs-capacitor": "*",
210 | "@types/koa": "*",
211 | "graphql": "^14.5.3"
212 | },
213 | "dependencies": {
214 | "graphql": {
215 | "version": "14.7.0",
216 | "resolved": "https://registry.npmjs.org/graphql/-/graphql-14.7.0.tgz",
217 | "integrity": "sha512-l0xWZpoPKpppFzMfvVyFmp9vLN7w/ZZJPefUicMCepfJeQ8sMcztloGYY9DfjVPo6tIUDzU5Hw3MUbIjj9AVVA==",
218 | "requires": {
219 | "iterall": "^1.2.2"
220 | }
221 | }
222 | }
223 | },
224 | "@types/http-assert": {
225 | "version": "1.5.1",
226 | "resolved": "https://registry.npmjs.org/@types/http-assert/-/http-assert-1.5.1.tgz",
227 | "integrity": "sha512-PGAK759pxyfXE78NbKxyfRcWYA/KwW17X290cNev/qAsn9eQIxkH4shoNBafH37wewhDG/0p1cHPbK6+SzZjWQ=="
228 | },
229 | "@types/keygrip": {
230 | "version": "1.0.2",
231 | "resolved": "https://registry.npmjs.org/@types/keygrip/-/keygrip-1.0.2.tgz",
232 | "integrity": "sha512-GJhpTepz2udxGexqos8wgaBx4I/zWIDPh/KOGEwAqtuGDkOUJu5eFvwmdBX4AmB8Odsr+9pHCQqiAqDL/yKMKw=="
233 | },
234 | "@types/koa": {
235 | "version": "2.11.3",
236 | "resolved": "https://registry.npmjs.org/@types/koa/-/koa-2.11.3.tgz",
237 | "integrity": "sha512-ABxVkrNWa4O/Jp24EYI/hRNqEVRlhB9g09p48neQp4m3xL1TJtdWk2NyNQSMCU45ejeELMQZBYyfstyVvO2H3Q==",
238 | "requires": {
239 | "@types/accepts": "*",
240 | "@types/content-disposition": "*",
241 | "@types/cookies": "*",
242 | "@types/http-assert": "*",
243 | "@types/keygrip": "*",
244 | "@types/koa-compose": "*",
245 | "@types/node": "*"
246 | }
247 | },
248 | "@types/koa-compose": {
249 | "version": "3.2.5",
250 | "resolved": "https://registry.npmjs.org/@types/koa-compose/-/koa-compose-3.2.5.tgz",
251 | "integrity": "sha512-B8nG/OoE1ORZqCkBVsup/AKcvjdgoHnfi4pZMn5UwAPCbhk/96xyv284eBYW8JlQbQ7zDmnpFr68I/40mFoIBQ==",
252 | "requires": {
253 | "@types/koa": "*"
254 | }
255 | },
256 | "@types/long": {
257 | "version": "4.0.1",
258 | "resolved": "https://registry.npmjs.org/@types/long/-/long-4.0.1.tgz",
259 | "integrity": "sha512-5tXH6Bx/kNGd3MgffdmP4dy2Z+G4eaXw0SE81Tq3BNadtnMR5/ySMzX4SLEzHJzSmPNn4HIdpQsBvXMUykr58w=="
260 | },
261 | "@types/mime": {
262 | "version": "2.0.3",
263 | "resolved": "https://registry.npmjs.org/@types/mime/-/mime-2.0.3.tgz",
264 | "integrity": "sha512-Jus9s4CDbqwocc5pOAnh8ShfrnMcPHuJYzVcSUU7lrh8Ni5HuIqX3oilL86p3dlTrk0LzHRCgA/GQ7uNCw6l2Q=="
265 | },
266 | "@types/node": {
267 | "version": "14.0.27",
268 | "resolved": "https://registry.npmjs.org/@types/node/-/node-14.0.27.tgz",
269 | "integrity": "sha512-kVrqXhbclHNHGu9ztnAwSncIgJv/FaxmzXJvGXNdcCpV1b8u1/Mi6z6m0vwy0LzKeXFTPLH0NzwmoJ3fNCIq0g=="
270 | },
271 | "@types/node-fetch": {
272 | "version": "2.5.7",
273 | "resolved": "https://registry.npmjs.org/@types/node-fetch/-/node-fetch-2.5.7.tgz",
274 | "integrity": "sha512-o2WVNf5UhWRkxlf6eq+jMZDu7kjgpgJfl4xVNlvryc95O/6F2ld8ztKX+qu+Rjyet93WAWm5LjeX9H5FGkODvw==",
275 | "requires": {
276 | "@types/node": "*",
277 | "form-data": "^3.0.0"
278 | }
279 | },
280 | "@types/qs": {
281 | "version": "6.9.4",
282 | "resolved": "https://registry.npmjs.org/@types/qs/-/qs-6.9.4.tgz",
283 | "integrity": "sha512-+wYo+L6ZF6BMoEjtf8zB2esQsqdV6WsjRK/GP9WOgLPrq87PbNWgIxS76dS5uvl/QXtHGakZmwTznIfcPXcKlQ=="
284 | },
285 | "@types/range-parser": {
286 | "version": "1.2.3",
287 | "resolved": "https://registry.npmjs.org/@types/range-parser/-/range-parser-1.2.3.tgz",
288 | "integrity": "sha512-ewFXqrQHlFsgc09MK5jP5iR7vumV/BYayNC6PgJO2LPe8vrnNFyjQjSppfEngITi0qvfKtzFvgKymGheFM9UOA=="
289 | },
290 | "@types/serve-static": {
291 | "version": "1.13.5",
292 | "resolved": "https://registry.npmjs.org/@types/serve-static/-/serve-static-1.13.5.tgz",
293 | "integrity": "sha512-6M64P58N+OXjU432WoLLBQxbA0LRGBCRm7aAGQJ+SMC1IMl0dgRVi9EFfoDcS2a7Xogygk/eGN94CfwU9UF7UQ==",
294 | "requires": {
295 | "@types/express-serve-static-core": "*",
296 | "@types/mime": "*"
297 | }
298 | },
299 | "@types/ws": {
300 | "version": "7.2.6",
301 | "resolved": "https://registry.npmjs.org/@types/ws/-/ws-7.2.6.tgz",
302 | "integrity": "sha512-Q07IrQUSNpr+cXU4E4LtkSIBPie5GLZyyMC1QtQYRLWz701+XcoVygGUZgvLqElq1nU4ICldMYPnexlBsg3dqQ==",
303 | "requires": {
304 | "@types/node": "*"
305 | }
306 | },
307 | "@wry/equality": {
308 | "version": "0.1.11",
309 | "resolved": "https://registry.npmjs.org/@wry/equality/-/equality-0.1.11.tgz",
310 | "integrity": "sha512-mwEVBDUVODlsQQ5dfuLUS5/Tf7jqUKyhKYHmVi4fPB6bDMOfWvUPJmKgS1Z7Za/sOI3vzWt4+O7yCiL/70MogA==",
311 | "requires": {
312 | "tslib": "^1.9.3"
313 | }
314 | },
315 | "abbrev": {
316 | "version": "1.1.1",
317 | "resolved": "https://registry.npmjs.org/abbrev/-/abbrev-1.1.1.tgz",
318 | "integrity": "sha512-nne9/IiQ/hzIhY6pdDnbBtz7DjPTKrY00P/zvPSm5pOFkl6xuGrGnXn/VtTNNfNtAfZ9/1RtehkszU9qcTii0Q==",
319 | "dev": true
320 | },
321 | "accepts": {
322 | "version": "1.3.7",
323 | "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.7.tgz",
324 | "integrity": "sha512-Il80Qs2WjYlJIBNzNkK6KYqlVMTbZLXgHx2oT0pU/fjRHyEp+PEfEPY0R3WCwAGVOtauxh1hOxNgIf5bv7dQpA==",
325 | "requires": {
326 | "mime-types": "~2.1.24",
327 | "negotiator": "0.6.2"
328 | }
329 | },
330 | "ansi-align": {
331 | "version": "3.0.0",
332 | "resolved": "https://registry.npmjs.org/ansi-align/-/ansi-align-3.0.0.tgz",
333 | "integrity": "sha512-ZpClVKqXN3RGBmKibdfWzqCY4lnjEuoNzU5T0oEFpfd/z5qJHVarukridD4juLO2FXMiwUQxr9WqQtaYa8XRYw==",
334 | "dev": true,
335 | "requires": {
336 | "string-width": "^3.0.0"
337 | },
338 | "dependencies": {
339 | "string-width": {
340 | "version": "3.1.0",
341 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz",
342 | "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==",
343 | "dev": true,
344 | "requires": {
345 | "emoji-regex": "^7.0.1",
346 | "is-fullwidth-code-point": "^2.0.0",
347 | "strip-ansi": "^5.1.0"
348 | }
349 | }
350 | }
351 | },
352 | "ansi-regex": {
353 | "version": "4.1.0",
354 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz",
355 | "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==",
356 | "dev": true
357 | },
358 | "ansi-styles": {
359 | "version": "4.2.1",
360 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.2.1.tgz",
361 | "integrity": "sha512-9VGjrMsG1vePxcSweQsN20KY/c4zN0h9fLjqAbwbPfahM3t+NL+M9HC8xeXG2I8pX5NoamTGNuomEUFI7fcUjA==",
362 | "dev": true,
363 | "requires": {
364 | "@types/color-name": "^1.1.1",
365 | "color-convert": "^2.0.1"
366 | }
367 | },
368 | "ansicolors": {
369 | "version": "0.3.2",
370 | "resolved": "https://registry.npmjs.org/ansicolors/-/ansicolors-0.3.2.tgz",
371 | "integrity": "sha1-ZlWX3oap/+Oqm/vmyuXG6kJrSXk="
372 | },
373 | "any-promise": {
374 | "version": "1.3.0",
375 | "resolved": "https://registry.npmjs.org/any-promise/-/any-promise-1.3.0.tgz",
376 | "integrity": "sha1-q8av7tzqUugJzcA3au0845Y10X8="
377 | },
378 | "anymatch": {
379 | "version": "3.1.1",
380 | "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.1.tgz",
381 | "integrity": "sha512-mM8522psRCqzV+6LhomX5wgp25YVibjh8Wj23I5RPkPppSVSjyKD2A2mBJmWGa+KN7f2D6LNh9jkBCeyLktzjg==",
382 | "dev": true,
383 | "requires": {
384 | "normalize-path": "^3.0.0",
385 | "picomatch": "^2.0.4"
386 | }
387 | },
388 | "apollo-cache-control": {
389 | "version": "0.11.1",
390 | "resolved": "https://registry.npmjs.org/apollo-cache-control/-/apollo-cache-control-0.11.1.tgz",
391 | "integrity": "sha512-6iHa8TkcKt4rx5SKRzDNjUIpCQX+7/FlZwD7vRh9JDnM4VH8SWhpj8fUR3CiEY8Kuc4ChXnOY8bCcMju5KPnIQ==",
392 | "requires": {
393 | "apollo-server-env": "^2.4.5",
394 | "apollo-server-plugin-base": "^0.9.1"
395 | }
396 | },
397 | "apollo-datasource": {
398 | "version": "0.7.2",
399 | "resolved": "https://registry.npmjs.org/apollo-datasource/-/apollo-datasource-0.7.2.tgz",
400 | "integrity": "sha512-ibnW+s4BMp4K2AgzLEtvzkjg7dJgCaw9M5b5N0YKNmeRZRnl/I/qBTQae648FsRKgMwTbRQIvBhQ0URUFAqFOw==",
401 | "requires": {
402 | "apollo-server-caching": "^0.5.2",
403 | "apollo-server-env": "^2.4.5"
404 | }
405 | },
406 | "apollo-engine-reporting": {
407 | "version": "2.3.0",
408 | "resolved": "https://registry.npmjs.org/apollo-engine-reporting/-/apollo-engine-reporting-2.3.0.tgz",
409 | "integrity": "sha512-SbcPLFuUZcRqDEZ6mSs8uHM9Ftr8yyt2IEu0JA8c3LNBmYXSLM7MHqFe80SVcosYSTBgtMz8mLJO8orhYoSYZw==",
410 | "requires": {
411 | "apollo-engine-reporting-protobuf": "^0.5.2",
412 | "apollo-graphql": "^0.5.0",
413 | "apollo-server-caching": "^0.5.2",
414 | "apollo-server-env": "^2.4.5",
415 | "apollo-server-errors": "^2.4.2",
416 | "apollo-server-plugin-base": "^0.9.1",
417 | "apollo-server-types": "^0.5.1",
418 | "async-retry": "^1.2.1",
419 | "uuid": "^8.0.0"
420 | }
421 | },
422 | "apollo-engine-reporting-protobuf": {
423 | "version": "0.5.2",
424 | "resolved": "https://registry.npmjs.org/apollo-engine-reporting-protobuf/-/apollo-engine-reporting-protobuf-0.5.2.tgz",
425 | "integrity": "sha512-4wm9FR3B7UvJxcK/69rOiS5CAJPEYKufeRWb257ZLfX7NGFTMqvbc1hu4q8Ch7swB26rTpkzfsftLED9DqH9qg==",
426 | "requires": {
427 | "@apollo/protobufjs": "^1.0.3"
428 | }
429 | },
430 | "apollo-env": {
431 | "version": "0.6.5",
432 | "resolved": "https://registry.npmjs.org/apollo-env/-/apollo-env-0.6.5.tgz",
433 | "integrity": "sha512-jeBUVsGymeTHYWp3me0R2CZRZrFeuSZeICZHCeRflHTfnQtlmbSXdy5E0pOyRM9CU4JfQkKDC98S1YglQj7Bzg==",
434 | "requires": {
435 | "@types/node-fetch": "2.5.7",
436 | "core-js": "^3.0.1",
437 | "node-fetch": "^2.2.0",
438 | "sha.js": "^2.4.11"
439 | }
440 | },
441 | "apollo-graphql": {
442 | "version": "0.5.0",
443 | "resolved": "https://registry.npmjs.org/apollo-graphql/-/apollo-graphql-0.5.0.tgz",
444 | "integrity": "sha512-YSdF/BKPbsnQpxWpmCE53pBJX44aaoif31Y22I/qKpB6ZSGzYijV5YBoCL5Q15H2oA/v/02Oazh9lbp4ek3eig==",
445 | "requires": {
446 | "apollo-env": "^0.6.5",
447 | "lodash.sortby": "^4.7.0"
448 | }
449 | },
450 | "apollo-link": {
451 | "version": "1.2.14",
452 | "resolved": "https://registry.npmjs.org/apollo-link/-/apollo-link-1.2.14.tgz",
453 | "integrity": "sha512-p67CMEFP7kOG1JZ0ZkYZwRDa369w5PIjtMjvrQd/HnIV8FRsHRqLqK+oAZQnFa1DDdZtOtHTi+aMIW6EatC2jg==",
454 | "requires": {
455 | "apollo-utilities": "^1.3.0",
456 | "ts-invariant": "^0.4.0",
457 | "tslib": "^1.9.3",
458 | "zen-observable-ts": "^0.8.21"
459 | }
460 | },
461 | "apollo-server": {
462 | "version": "2.16.1",
463 | "resolved": "https://registry.npmjs.org/apollo-server/-/apollo-server-2.16.1.tgz",
464 | "integrity": "sha512-oy9NVRzGwlpQ+W1DwLKRH+KASmodSYpvYIRY5DMAZtGqNmT2zOCpbIZVjBt23SuPB5NhIhhE4ROzoObRv3zy5w==",
465 | "requires": {
466 | "apollo-server-core": "^2.16.1",
467 | "apollo-server-express": "^2.16.1",
468 | "express": "^4.0.0",
469 | "graphql-subscriptions": "^1.0.0",
470 | "graphql-tools": "^4.0.0"
471 | }
472 | },
473 | "apollo-server-caching": {
474 | "version": "0.5.2",
475 | "resolved": "https://registry.npmjs.org/apollo-server-caching/-/apollo-server-caching-0.5.2.tgz",
476 | "integrity": "sha512-HUcP3TlgRsuGgeTOn8QMbkdx0hLPXyEJehZIPrcof0ATz7j7aTPA4at7gaiFHCo8gk07DaWYGB3PFgjboXRcWQ==",
477 | "requires": {
478 | "lru-cache": "^5.0.0"
479 | }
480 | },
481 | "apollo-server-core": {
482 | "version": "2.16.1",
483 | "resolved": "https://registry.npmjs.org/apollo-server-core/-/apollo-server-core-2.16.1.tgz",
484 | "integrity": "sha512-nuwn5ZBbmzPwDetb3FgiFFJlNK7ZBFg8kis/raymrjd3eBGdNcOyMTJDl6J9673X9Xqp+dXQmFYDW/G3G8S1YA==",
485 | "requires": {
486 | "@apollographql/apollo-tools": "^0.4.3",
487 | "@apollographql/graphql-playground-html": "1.6.26",
488 | "@types/graphql-upload": "^8.0.0",
489 | "@types/ws": "^7.0.0",
490 | "apollo-cache-control": "^0.11.1",
491 | "apollo-datasource": "^0.7.2",
492 | "apollo-engine-reporting": "^2.3.0",
493 | "apollo-server-caching": "^0.5.2",
494 | "apollo-server-env": "^2.4.5",
495 | "apollo-server-errors": "^2.4.2",
496 | "apollo-server-plugin-base": "^0.9.1",
497 | "apollo-server-types": "^0.5.1",
498 | "apollo-tracing": "^0.11.1",
499 | "fast-json-stable-stringify": "^2.0.0",
500 | "graphql-extensions": "^0.12.4",
501 | "graphql-tag": "^2.9.2",
502 | "graphql-tools": "^4.0.0",
503 | "graphql-upload": "^8.0.2",
504 | "loglevel": "^1.6.7",
505 | "sha.js": "^2.4.11",
506 | "subscriptions-transport-ws": "^0.9.11",
507 | "ws": "^6.0.0"
508 | }
509 | },
510 | "apollo-server-env": {
511 | "version": "2.4.5",
512 | "resolved": "https://registry.npmjs.org/apollo-server-env/-/apollo-server-env-2.4.5.tgz",
513 | "integrity": "sha512-nfNhmGPzbq3xCEWT8eRpoHXIPNcNy3QcEoBlzVMjeglrBGryLG2LXwBSPnVmTRRrzUYugX0ULBtgE3rBFNoUgA==",
514 | "requires": {
515 | "node-fetch": "^2.1.2",
516 | "util.promisify": "^1.0.0"
517 | }
518 | },
519 | "apollo-server-errors": {
520 | "version": "2.4.2",
521 | "resolved": "https://registry.npmjs.org/apollo-server-errors/-/apollo-server-errors-2.4.2.tgz",
522 | "integrity": "sha512-FeGxW3Batn6sUtX3OVVUm7o56EgjxDlmgpTLNyWcLb0j6P8mw9oLNyAm3B+deHA4KNdNHO5BmHS2g1SJYjqPCQ=="
523 | },
524 | "apollo-server-express": {
525 | "version": "2.16.1",
526 | "resolved": "https://registry.npmjs.org/apollo-server-express/-/apollo-server-express-2.16.1.tgz",
527 | "integrity": "sha512-Oq5YNcaMYnRk6jDmA9LWf8oSd2KHDVe7jQ4wtooAvG9FVUD+FaFBgSkytXHMvtifQh2wdF07Ri8uDLMz6IQjTw==",
528 | "requires": {
529 | "@apollographql/graphql-playground-html": "1.6.26",
530 | "@types/accepts": "^1.3.5",
531 | "@types/body-parser": "1.19.0",
532 | "@types/cors": "^2.8.4",
533 | "@types/express": "4.17.7",
534 | "accepts": "^1.3.5",
535 | "apollo-server-core": "^2.16.1",
536 | "apollo-server-types": "^0.5.1",
537 | "body-parser": "^1.18.3",
538 | "cors": "^2.8.4",
539 | "express": "^4.17.1",
540 | "graphql-subscriptions": "^1.0.0",
541 | "graphql-tools": "^4.0.0",
542 | "parseurl": "^1.3.2",
543 | "subscriptions-transport-ws": "^0.9.16",
544 | "type-is": "^1.6.16"
545 | }
546 | },
547 | "apollo-server-plugin-base": {
548 | "version": "0.9.1",
549 | "resolved": "https://registry.npmjs.org/apollo-server-plugin-base/-/apollo-server-plugin-base-0.9.1.tgz",
550 | "integrity": "sha512-kvrX4Z3FdpjrZdHkyl5iY2A1Wvp4b6KQp00DeZqss7GyyKNUBKr80/7RQgBLEw7EWM7WB19j459xM/TjvW0FKQ==",
551 | "requires": {
552 | "apollo-server-types": "^0.5.1"
553 | }
554 | },
555 | "apollo-server-types": {
556 | "version": "0.5.1",
557 | "resolved": "https://registry.npmjs.org/apollo-server-types/-/apollo-server-types-0.5.1.tgz",
558 | "integrity": "sha512-my2cPw+DAb2qVnIuBcsRKGyS28uIc2vjFxa1NpRoJZe9gK0BWUBk7wzXnIzWy3HZ5Er11e/40MPTUesNfMYNVA==",
559 | "requires": {
560 | "apollo-engine-reporting-protobuf": "^0.5.2",
561 | "apollo-server-caching": "^0.5.2",
562 | "apollo-server-env": "^2.4.5"
563 | }
564 | },
565 | "apollo-tracing": {
566 | "version": "0.11.1",
567 | "resolved": "https://registry.npmjs.org/apollo-tracing/-/apollo-tracing-0.11.1.tgz",
568 | "integrity": "sha512-l7g+uILw7v32GA46IRXIx5XXbZhFI96BhSqrGK9yyvfq+NMcvVZrj3kIhRImPGhAjMdV+5biA/jztabElAbDjg==",
569 | "requires": {
570 | "apollo-server-env": "^2.4.5",
571 | "apollo-server-plugin-base": "^0.9.1"
572 | }
573 | },
574 | "apollo-utilities": {
575 | "version": "1.3.4",
576 | "resolved": "https://registry.npmjs.org/apollo-utilities/-/apollo-utilities-1.3.4.tgz",
577 | "integrity": "sha512-pk2hiWrCXMAy2fRPwEyhvka+mqwzeP60Jr1tRYi5xru+3ko94HI9o6lK0CT33/w4RDlxWchmdhDCrvdr+pHCig==",
578 | "requires": {
579 | "@wry/equality": "^0.1.2",
580 | "fast-json-stable-stringify": "^2.0.0",
581 | "ts-invariant": "^0.4.0",
582 | "tslib": "^1.10.0"
583 | }
584 | },
585 | "array-flatten": {
586 | "version": "1.1.1",
587 | "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz",
588 | "integrity": "sha1-ml9pkFGx5wczKPKgCJaLZOopVdI="
589 | },
590 | "async-limiter": {
591 | "version": "1.0.1",
592 | "resolved": "https://registry.npmjs.org/async-limiter/-/async-limiter-1.0.1.tgz",
593 | "integrity": "sha512-csOlWGAcRFJaI6m+F2WKdnMKr4HhdhFVBk0H/QbJFMCr+uO2kwohwXQPxw/9OCxp05r5ghVBFSyioixx3gfkNQ=="
594 | },
595 | "async-retry": {
596 | "version": "1.3.1",
597 | "resolved": "https://registry.npmjs.org/async-retry/-/async-retry-1.3.1.tgz",
598 | "integrity": "sha512-aiieFW/7h3hY0Bq5d+ktDBejxuwR78vRu9hDUdR8rNhSaQ29VzPL4AoIRG7D/c7tdenwOcKvgPM6tIxB3cB6HA==",
599 | "requires": {
600 | "retry": "0.12.0"
601 | }
602 | },
603 | "asynckit": {
604 | "version": "0.4.0",
605 | "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz",
606 | "integrity": "sha1-x57Zf380y48robyXkLzDZkdLS3k="
607 | },
608 | "backo2": {
609 | "version": "1.0.2",
610 | "resolved": "https://registry.npmjs.org/backo2/-/backo2-1.0.2.tgz",
611 | "integrity": "sha1-MasayLEpNjRj41s+u2n038+6eUc="
612 | },
613 | "balanced-match": {
614 | "version": "1.0.0",
615 | "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.0.tgz",
616 | "integrity": "sha1-ibTRmasr7kneFk6gK4nORi1xt2c=",
617 | "dev": true
618 | },
619 | "bcryptjs": {
620 | "version": "2.4.3",
621 | "resolved": "https://registry.npmjs.org/bcryptjs/-/bcryptjs-2.4.3.tgz",
622 | "integrity": "sha1-mrVie5PmBiH/fNrF2pczAn3x0Ms="
623 | },
624 | "binary-extensions": {
625 | "version": "2.1.0",
626 | "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.1.0.tgz",
627 | "integrity": "sha512-1Yj8h9Q+QDF5FzhMs/c9+6UntbD5MkRfRwac8DoEm9ZfUBZ7tZ55YcGVAzEe4bXsdQHEk+s9S5wsOKVdZrw0tQ==",
628 | "dev": true
629 | },
630 | "body-parser": {
631 | "version": "1.19.0",
632 | "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.19.0.tgz",
633 | "integrity": "sha512-dhEPs72UPbDnAQJ9ZKMNTP6ptJaionhP5cBb541nXPlW60Jepo9RV/a4fX4XWW9CuFNK22krhrj1+rgzifNCsw==",
634 | "requires": {
635 | "bytes": "3.1.0",
636 | "content-type": "~1.0.4",
637 | "debug": "2.6.9",
638 | "depd": "~1.1.2",
639 | "http-errors": "1.7.2",
640 | "iconv-lite": "0.4.24",
641 | "on-finished": "~2.3.0",
642 | "qs": "6.7.0",
643 | "raw-body": "2.4.0",
644 | "type-is": "~1.6.17"
645 | },
646 | "dependencies": {
647 | "http-errors": {
648 | "version": "1.7.2",
649 | "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.7.2.tgz",
650 | "integrity": "sha512-uUQBt3H/cSIVfch6i1EuPNy/YsRSOUBXTVfZ+yR7Zjez3qjBz6i9+i4zjNaoqcoFVI4lQJ5plg63TvGfRSDCRg==",
651 | "requires": {
652 | "depd": "~1.1.2",
653 | "inherits": "2.0.3",
654 | "setprototypeof": "1.1.1",
655 | "statuses": ">= 1.5.0 < 2",
656 | "toidentifier": "1.0.0"
657 | }
658 | },
659 | "inherits": {
660 | "version": "2.0.3",
661 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz",
662 | "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4="
663 | },
664 | "setprototypeof": {
665 | "version": "1.1.1",
666 | "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.1.tgz",
667 | "integrity": "sha512-JvdAWfbXeIGaZ9cILp38HntZSFSo3mWg6xGcJJsd+d4aRMOqauag1C63dJfDw7OaMYwEbHMOxEZ1lqVRYP2OAw=="
668 | }
669 | }
670 | },
671 | "boxen": {
672 | "version": "4.2.0",
673 | "resolved": "https://registry.npmjs.org/boxen/-/boxen-4.2.0.tgz",
674 | "integrity": "sha512-eB4uT9RGzg2odpER62bBwSLvUeGC+WbRjjyyFhGsKnc8wp/m0+hQsMUvUe3H2V0D5vw0nBdO1hCJoZo5mKeuIQ==",
675 | "dev": true,
676 | "requires": {
677 | "ansi-align": "^3.0.0",
678 | "camelcase": "^5.3.1",
679 | "chalk": "^3.0.0",
680 | "cli-boxes": "^2.2.0",
681 | "string-width": "^4.1.0",
682 | "term-size": "^2.1.0",
683 | "type-fest": "^0.8.1",
684 | "widest-line": "^3.1.0"
685 | }
686 | },
687 | "brace-expansion": {
688 | "version": "1.1.11",
689 | "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz",
690 | "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==",
691 | "dev": true,
692 | "requires": {
693 | "balanced-match": "^1.0.0",
694 | "concat-map": "0.0.1"
695 | }
696 | },
697 | "braces": {
698 | "version": "3.0.2",
699 | "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz",
700 | "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==",
701 | "dev": true,
702 | "requires": {
703 | "fill-range": "^7.0.1"
704 | }
705 | },
706 | "buffer-equal-constant-time": {
707 | "version": "1.0.1",
708 | "resolved": "https://registry.npmjs.org/buffer-equal-constant-time/-/buffer-equal-constant-time-1.0.1.tgz",
709 | "integrity": "sha1-+OcRMvf/5uAaXJaXpMbz5I1cyBk="
710 | },
711 | "busboy": {
712 | "version": "0.3.1",
713 | "resolved": "https://registry.npmjs.org/busboy/-/busboy-0.3.1.tgz",
714 | "integrity": "sha512-y7tTxhGKXcyBxRKAni+awqx8uqaJKrSFSNFSeRG5CsWNdmy2BIK+6VGWEW7TZnIO/533mtMEA4rOevQV815YJw==",
715 | "requires": {
716 | "dicer": "0.3.0"
717 | }
718 | },
719 | "bytes": {
720 | "version": "3.1.0",
721 | "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.0.tgz",
722 | "integrity": "sha512-zauLjrfCG+xvoyaqLoV8bLVXXNGC4JqlxFCutSDWA6fJrTo2ZuvLYTqZ7aHBLZSMOopbzwv8f+wZcVzfVTI2Dg=="
723 | },
724 | "cacheable-request": {
725 | "version": "6.1.0",
726 | "resolved": "https://registry.npmjs.org/cacheable-request/-/cacheable-request-6.1.0.tgz",
727 | "integrity": "sha512-Oj3cAGPCqOZX7Rz64Uny2GYAZNliQSqfbePrgAQ1wKAihYmCUnraBtJtKcGR4xz7wF+LoJC+ssFZvv5BgF9Igg==",
728 | "dev": true,
729 | "requires": {
730 | "clone-response": "^1.0.2",
731 | "get-stream": "^5.1.0",
732 | "http-cache-semantics": "^4.0.0",
733 | "keyv": "^3.0.0",
734 | "lowercase-keys": "^2.0.0",
735 | "normalize-url": "^4.1.0",
736 | "responselike": "^1.0.2"
737 | },
738 | "dependencies": {
739 | "get-stream": {
740 | "version": "5.1.0",
741 | "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-5.1.0.tgz",
742 | "integrity": "sha512-EXr1FOzrzTfGeL0gQdeFEvOMm2mzMOglyiOXSTpPC+iAjAKftbr3jpCMWynogwYnM+eSj9sHGc6wjIcDvYiygw==",
743 | "dev": true,
744 | "requires": {
745 | "pump": "^3.0.0"
746 | }
747 | },
748 | "lowercase-keys": {
749 | "version": "2.0.0",
750 | "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-2.0.0.tgz",
751 | "integrity": "sha512-tqNXrS78oMOE73NMxK4EMLQsQowWf8jKooH9g7xPavRT706R6bkQJ6DY2Te7QukaZsulxa30wQ7bk0pm4XiHmA==",
752 | "dev": true
753 | }
754 | }
755 | },
756 | "camelcase": {
757 | "version": "5.3.1",
758 | "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz",
759 | "integrity": "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==",
760 | "dev": true
761 | },
762 | "cardinal": {
763 | "version": "2.1.1",
764 | "resolved": "https://registry.npmjs.org/cardinal/-/cardinal-2.1.1.tgz",
765 | "integrity": "sha1-fMEFXYItISlU0HsIXeolHMe8VQU=",
766 | "requires": {
767 | "ansicolors": "~0.3.2",
768 | "redeyed": "~2.1.0"
769 | }
770 | },
771 | "chalk": {
772 | "version": "3.0.0",
773 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-3.0.0.tgz",
774 | "integrity": "sha512-4D3B6Wf41KOYRFdszmDqMCGq5VV/uMAB273JILmO+3jAlh8X4qDtdtgCR3fxtbLEMzSx22QdhnDcJvu2u1fVwg==",
775 | "dev": true,
776 | "requires": {
777 | "ansi-styles": "^4.1.0",
778 | "supports-color": "^7.1.0"
779 | },
780 | "dependencies": {
781 | "has-flag": {
782 | "version": "4.0.0",
783 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz",
784 | "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==",
785 | "dev": true
786 | },
787 | "supports-color": {
788 | "version": "7.1.0",
789 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.1.0.tgz",
790 | "integrity": "sha512-oRSIpR8pxT1Wr2FquTNnGet79b3BWljqOuoW/h4oBhxJ/HUbX5nX6JSruTkvXDCFMwDPvsaTTbvMLKZWSy0R5g==",
791 | "dev": true,
792 | "requires": {
793 | "has-flag": "^4.0.0"
794 | }
795 | }
796 | }
797 | },
798 | "chokidar": {
799 | "version": "3.4.2",
800 | "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.4.2.tgz",
801 | "integrity": "sha512-IZHaDeBeI+sZJRX7lGcXsdzgvZqKv6sECqsbErJA4mHWfpRrD8B97kSFN4cQz6nGBGiuFia1MKR4d6c1o8Cv7A==",
802 | "dev": true,
803 | "requires": {
804 | "anymatch": "~3.1.1",
805 | "braces": "~3.0.2",
806 | "fsevents": "~2.1.2",
807 | "glob-parent": "~5.1.0",
808 | "is-binary-path": "~2.1.0",
809 | "is-glob": "~4.0.1",
810 | "normalize-path": "~3.0.0",
811 | "readdirp": "~3.4.0"
812 | }
813 | },
814 | "ci-info": {
815 | "version": "2.0.0",
816 | "resolved": "https://registry.npmjs.org/ci-info/-/ci-info-2.0.0.tgz",
817 | "integrity": "sha512-5tK7EtrZ0N+OLFMthtqOj4fI2Jeb88C4CAZPu25LDVUgXJ0A3Js4PMGqrn0JU1W0Mh1/Z8wZzYPxqUrXeBboCQ==",
818 | "dev": true
819 | },
820 | "cli-boxes": {
821 | "version": "2.2.0",
822 | "resolved": "https://registry.npmjs.org/cli-boxes/-/cli-boxes-2.2.0.tgz",
823 | "integrity": "sha512-gpaBrMAizVEANOpfZp/EEUixTXDyGt7DFzdK5hU+UbWt/J0lB0w20ncZj59Z9a93xHb9u12zF5BS6i9RKbtg4w==",
824 | "dev": true
825 | },
826 | "clone-response": {
827 | "version": "1.0.2",
828 | "resolved": "https://registry.npmjs.org/clone-response/-/clone-response-1.0.2.tgz",
829 | "integrity": "sha1-0dyXOSAxTfZ/vrlCI7TuNQI56Ws=",
830 | "dev": true,
831 | "requires": {
832 | "mimic-response": "^1.0.0"
833 | }
834 | },
835 | "color-convert": {
836 | "version": "2.0.1",
837 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz",
838 | "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==",
839 | "dev": true,
840 | "requires": {
841 | "color-name": "~1.1.4"
842 | }
843 | },
844 | "color-name": {
845 | "version": "1.1.4",
846 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz",
847 | "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==",
848 | "dev": true
849 | },
850 | "combined-stream": {
851 | "version": "1.0.8",
852 | "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz",
853 | "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==",
854 | "requires": {
855 | "delayed-stream": "~1.0.0"
856 | }
857 | },
858 | "commander": {
859 | "version": "2.20.3",
860 | "resolved": "https://registry.npmjs.org/commander/-/commander-2.20.3.tgz",
861 | "integrity": "sha512-GpVkmM8vF2vQUkj2LvZmD35JxeJOLCwJ9cUkugyk2nuhbv3+mJvpLYYt+0+USMxE+oj+ey/lJEnhZw75x/OMcQ=="
862 | },
863 | "concat-map": {
864 | "version": "0.0.1",
865 | "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz",
866 | "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=",
867 | "dev": true
868 | },
869 | "configstore": {
870 | "version": "5.0.1",
871 | "resolved": "https://registry.npmjs.org/configstore/-/configstore-5.0.1.tgz",
872 | "integrity": "sha512-aMKprgk5YhBNyH25hj8wGt2+D52Sw1DRRIzqBwLp2Ya9mFmY8KPvvtvmna8SxVR9JMZ4kzMD68N22vlaRpkeFA==",
873 | "dev": true,
874 | "requires": {
875 | "dot-prop": "^5.2.0",
876 | "graceful-fs": "^4.1.2",
877 | "make-dir": "^3.0.0",
878 | "unique-string": "^2.0.0",
879 | "write-file-atomic": "^3.0.0",
880 | "xdg-basedir": "^4.0.0"
881 | }
882 | },
883 | "content-disposition": {
884 | "version": "0.5.3",
885 | "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.3.tgz",
886 | "integrity": "sha512-ExO0774ikEObIAEV9kDo50o+79VCUdEB6n6lzKgGwupcVeRlhrj3qGAfwq8G6uBJjkqLrhT0qEYFcWng8z1z0g==",
887 | "requires": {
888 | "safe-buffer": "5.1.2"
889 | },
890 | "dependencies": {
891 | "safe-buffer": {
892 | "version": "5.1.2",
893 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz",
894 | "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g=="
895 | }
896 | }
897 | },
898 | "content-type": {
899 | "version": "1.0.4",
900 | "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.4.tgz",
901 | "integrity": "sha512-hIP3EEPs8tB9AT1L+NUqtwOAps4mk2Zob89MWXMHjHWg9milF/j4osnnQLXBCBFBk/tvIG/tUc9mOUJiPBhPXA=="
902 | },
903 | "cookie": {
904 | "version": "0.4.0",
905 | "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.4.0.tgz",
906 | "integrity": "sha512-+Hp8fLp57wnUSt0tY0tHEXh4voZRDnoIrZPqlo3DPiI4y9lwg/jqx+1Om94/W6ZaPDOUbnjOt/99w66zk+l1Xg=="
907 | },
908 | "cookie-signature": {
909 | "version": "1.0.6",
910 | "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz",
911 | "integrity": "sha1-4wOogrNCzD7oylE6eZmXNNqzriw="
912 | },
913 | "core-js": {
914 | "version": "3.6.5",
915 | "resolved": "https://registry.npmjs.org/core-js/-/core-js-3.6.5.tgz",
916 | "integrity": "sha512-vZVEEwZoIsI+vPEuoF9Iqf5H7/M3eeQqWlQnYa8FSKKePuYTf5MWnxb5SDAzCa60b3JBRS5g9b+Dq7b1y/RCrA=="
917 | },
918 | "cors": {
919 | "version": "2.8.5",
920 | "resolved": "https://registry.npmjs.org/cors/-/cors-2.8.5.tgz",
921 | "integrity": "sha512-KIHbLJqu73RGr/hnbrO9uBeixNGuvSQjul/jdFvS/KFSIH1hWVd1ng7zOHx+YrEfInLG7q4n6GHQ9cDtxv/P6g==",
922 | "requires": {
923 | "object-assign": "^4",
924 | "vary": "^1"
925 | }
926 | },
927 | "crypto-random-string": {
928 | "version": "2.0.0",
929 | "resolved": "https://registry.npmjs.org/crypto-random-string/-/crypto-random-string-2.0.0.tgz",
930 | "integrity": "sha512-v1plID3y9r/lPhviJ1wrXpLeyUIGAZ2SHNYTEapm7/8A9nLPoyvVp3RK/EPFqn5kEznyWgYZNsRtYYIWbuG8KA==",
931 | "dev": true
932 | },
933 | "cssfilter": {
934 | "version": "0.0.10",
935 | "resolved": "https://registry.npmjs.org/cssfilter/-/cssfilter-0.0.10.tgz",
936 | "integrity": "sha1-xtJnJjKi5cg+AT5oZKQs6N79IK4="
937 | },
938 | "debug": {
939 | "version": "2.6.9",
940 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz",
941 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==",
942 | "requires": {
943 | "ms": "2.0.0"
944 | }
945 | },
946 | "decompress-response": {
947 | "version": "3.3.0",
948 | "resolved": "https://registry.npmjs.org/decompress-response/-/decompress-response-3.3.0.tgz",
949 | "integrity": "sha1-gKTdMjdIOEv6JICDYirt7Jgq3/M=",
950 | "dev": true,
951 | "requires": {
952 | "mimic-response": "^1.0.0"
953 | }
954 | },
955 | "deep-extend": {
956 | "version": "0.6.0",
957 | "resolved": "https://registry.npmjs.org/deep-extend/-/deep-extend-0.6.0.tgz",
958 | "integrity": "sha512-LOHxIOaPYdHlJRtCQfDIVZtfw/ufM8+rVj649RIHzcm/vGwQRXFt6OPqIFWsm2XEMrNIEtWR64sY1LEKD2vAOA==",
959 | "dev": true
960 | },
961 | "defer-to-connect": {
962 | "version": "1.1.3",
963 | "resolved": "https://registry.npmjs.org/defer-to-connect/-/defer-to-connect-1.1.3.tgz",
964 | "integrity": "sha512-0ISdNousHvZT2EiFlZeZAHBUvSxmKswVCEf8hW7KWgG4a8MVEu/3Vb6uWYozkjylyCxe0JBIiRB1jV45S70WVQ==",
965 | "dev": true
966 | },
967 | "define-properties": {
968 | "version": "1.1.3",
969 | "resolved": "https://registry.npmjs.org/define-properties/-/define-properties-1.1.3.tgz",
970 | "integrity": "sha512-3MqfYKj2lLzdMSf8ZIZE/V+Zuy+BgD6f164e8K2w7dgnpKArBDerGYpM46IYYcjnkdPNMjPk9A6VFB8+3SKlXQ==",
971 | "requires": {
972 | "object-keys": "^1.0.12"
973 | }
974 | },
975 | "delayed-stream": {
976 | "version": "1.0.0",
977 | "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz",
978 | "integrity": "sha1-3zrhmayt+31ECqrgsp4icrJOxhk="
979 | },
980 | "denque": {
981 | "version": "1.4.1",
982 | "resolved": "https://registry.npmjs.org/denque/-/denque-1.4.1.tgz",
983 | "integrity": "sha512-OfzPuSZKGcgr96rf1oODnfjqBFmr1DVoc/TrItj3Ohe0Ah1C5WX5Baquw/9U9KovnQ88EqmJbD66rKYUQYN1tQ=="
984 | },
985 | "depd": {
986 | "version": "1.1.2",
987 | "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.2.tgz",
988 | "integrity": "sha1-m81S4UwJd2PnSbJ0xDRu0uVgtak="
989 | },
990 | "deprecated-decorator": {
991 | "version": "0.1.6",
992 | "resolved": "https://registry.npmjs.org/deprecated-decorator/-/deprecated-decorator-0.1.6.tgz",
993 | "integrity": "sha1-AJZjF7ehL+kvPMgx91g68ym4bDc="
994 | },
995 | "destroy": {
996 | "version": "1.0.4",
997 | "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.0.4.tgz",
998 | "integrity": "sha1-l4hXRCxEdJ5CBmE+N5RiBYJqvYA="
999 | },
1000 | "dicer": {
1001 | "version": "0.3.0",
1002 | "resolved": "https://registry.npmjs.org/dicer/-/dicer-0.3.0.tgz",
1003 | "integrity": "sha512-MdceRRWqltEG2dZqO769g27N/3PXfcKl04VhYnBlo2YhH7zPi88VebsjTKclaOyiuMaGU72hTfw3VkUitGcVCA==",
1004 | "requires": {
1005 | "streamsearch": "0.1.2"
1006 | }
1007 | },
1008 | "dot-prop": {
1009 | "version": "5.2.0",
1010 | "resolved": "https://registry.npmjs.org/dot-prop/-/dot-prop-5.2.0.tgz",
1011 | "integrity": "sha512-uEUyaDKoSQ1M4Oq8l45hSE26SnTxL6snNnqvK/VWx5wJhmff5z0FUVJDKDanor/6w3kzE3i7XZOk+7wC0EXr1A==",
1012 | "dev": true,
1013 | "requires": {
1014 | "is-obj": "^2.0.0"
1015 | }
1016 | },
1017 | "dotenv": {
1018 | "version": "8.2.0",
1019 | "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-8.2.0.tgz",
1020 | "integrity": "sha512-8sJ78ElpbDJBHNeBzUbUVLsqKdccaa/BXF1uPTw3GrvQTBgrQrtObr2mUrE38vzYd8cEv+m/JBfDLioYcfXoaw=="
1021 | },
1022 | "dottie": {
1023 | "version": "2.0.2",
1024 | "resolved": "https://registry.npmjs.org/dottie/-/dottie-2.0.2.tgz",
1025 | "integrity": "sha512-fmrwR04lsniq/uSr8yikThDTrM7epXHBAAjH9TbeH3rEA8tdCO7mRzB9hdmdGyJCxF8KERo9CITcm3kGuoyMhg=="
1026 | },
1027 | "duplexer3": {
1028 | "version": "0.1.4",
1029 | "resolved": "https://registry.npmjs.org/duplexer3/-/duplexer3-0.1.4.tgz",
1030 | "integrity": "sha1-7gHdHKwO08vH/b6jfcCo8c4ALOI=",
1031 | "dev": true
1032 | },
1033 | "ecdsa-sig-formatter": {
1034 | "version": "1.0.11",
1035 | "resolved": "https://registry.npmjs.org/ecdsa-sig-formatter/-/ecdsa-sig-formatter-1.0.11.tgz",
1036 | "integrity": "sha512-nagl3RYrbNv6kQkeJIpt6NJZy8twLB/2vtz6yN9Z4vRKHN4/QZJIEbqohALSgwKdnksuY3k5Addp5lg8sVoVcQ==",
1037 | "requires": {
1038 | "safe-buffer": "^5.0.1"
1039 | }
1040 | },
1041 | "ee-first": {
1042 | "version": "1.1.1",
1043 | "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz",
1044 | "integrity": "sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0="
1045 | },
1046 | "emoji-regex": {
1047 | "version": "7.0.3",
1048 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-7.0.3.tgz",
1049 | "integrity": "sha512-CwBLREIQ7LvYFB0WyRvwhq5N5qPhc6PMjD6bYggFlI5YyDgl+0vxq5VHbMOFqLg7hfWzmu8T5Z1QofhmTIhItA==",
1050 | "dev": true
1051 | },
1052 | "encodeurl": {
1053 | "version": "1.0.2",
1054 | "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz",
1055 | "integrity": "sha1-rT/0yG7C0CkyL1oCw6mmBslbP1k="
1056 | },
1057 | "end-of-stream": {
1058 | "version": "1.4.4",
1059 | "resolved": "https://registry.npmjs.org/end-of-stream/-/end-of-stream-1.4.4.tgz",
1060 | "integrity": "sha512-+uw1inIHVPQoaVuHzRyXd21icM+cnt4CzD5rW+NC1wjOUSTOs+Te7FOv7AhN7vS9x/oIyhLP5PR1H+phQAHu5Q==",
1061 | "dev": true,
1062 | "requires": {
1063 | "once": "^1.4.0"
1064 | }
1065 | },
1066 | "es-abstract": {
1067 | "version": "1.17.6",
1068 | "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.17.6.tgz",
1069 | "integrity": "sha512-Fr89bON3WFyUi5EvAeI48QTWX0AyekGgLA8H+c+7fbfCkJwRWRMLd8CQedNEyJuoYYhmtEqY92pgte1FAhBlhw==",
1070 | "requires": {
1071 | "es-to-primitive": "^1.2.1",
1072 | "function-bind": "^1.1.1",
1073 | "has": "^1.0.3",
1074 | "has-symbols": "^1.0.1",
1075 | "is-callable": "^1.2.0",
1076 | "is-regex": "^1.1.0",
1077 | "object-inspect": "^1.7.0",
1078 | "object-keys": "^1.1.1",
1079 | "object.assign": "^4.1.0",
1080 | "string.prototype.trimend": "^1.0.1",
1081 | "string.prototype.trimstart": "^1.0.1"
1082 | }
1083 | },
1084 | "es-to-primitive": {
1085 | "version": "1.2.1",
1086 | "resolved": "https://registry.npmjs.org/es-to-primitive/-/es-to-primitive-1.2.1.tgz",
1087 | "integrity": "sha512-QCOllgZJtaUo9miYBcLChTUaHNjJF3PYs1VidD7AwiEj1kYxKeQTctLAezAOH5ZKRH0g2IgPn6KwB4IT8iRpvA==",
1088 | "requires": {
1089 | "is-callable": "^1.1.4",
1090 | "is-date-object": "^1.0.1",
1091 | "is-symbol": "^1.0.2"
1092 | }
1093 | },
1094 | "escape-goat": {
1095 | "version": "2.1.1",
1096 | "resolved": "https://registry.npmjs.org/escape-goat/-/escape-goat-2.1.1.tgz",
1097 | "integrity": "sha512-8/uIhbG12Csjy2JEW7D9pHbreaVaS/OpN3ycnyvElTdwM5n6GY6W6e2IPemfvGZeUMqZ9A/3GqIZMgKnBhAw/Q==",
1098 | "dev": true
1099 | },
1100 | "escape-html": {
1101 | "version": "1.0.3",
1102 | "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz",
1103 | "integrity": "sha1-Aljq5NPQwJdN4cFpGI7wBR0dGYg="
1104 | },
1105 | "esprima": {
1106 | "version": "4.0.1",
1107 | "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz",
1108 | "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A=="
1109 | },
1110 | "etag": {
1111 | "version": "1.8.1",
1112 | "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz",
1113 | "integrity": "sha1-Qa4u62XvpiJorr/qg6x9eSmbCIc="
1114 | },
1115 | "eventemitter3": {
1116 | "version": "3.1.2",
1117 | "resolved": "https://registry.npmjs.org/eventemitter3/-/eventemitter3-3.1.2.tgz",
1118 | "integrity": "sha512-tvtQIeLVHjDkJYnzf2dgVMxfuSGJeM/7UCG17TT4EumTfNtF+0nebF/4zWOIkCreAbtNqhGEboB6BWrwqNaw4Q=="
1119 | },
1120 | "express": {
1121 | "version": "4.17.1",
1122 | "resolved": "https://registry.npmjs.org/express/-/express-4.17.1.tgz",
1123 | "integrity": "sha512-mHJ9O79RqluphRrcw2X/GTh3k9tVv8YcoyY4Kkh4WDMUYKRZUq0h1o0w2rrrxBqM7VoeUVqgb27xlEMXTnYt4g==",
1124 | "requires": {
1125 | "accepts": "~1.3.7",
1126 | "array-flatten": "1.1.1",
1127 | "body-parser": "1.19.0",
1128 | "content-disposition": "0.5.3",
1129 | "content-type": "~1.0.4",
1130 | "cookie": "0.4.0",
1131 | "cookie-signature": "1.0.6",
1132 | "debug": "2.6.9",
1133 | "depd": "~1.1.2",
1134 | "encodeurl": "~1.0.2",
1135 | "escape-html": "~1.0.3",
1136 | "etag": "~1.8.1",
1137 | "finalhandler": "~1.1.2",
1138 | "fresh": "0.5.2",
1139 | "merge-descriptors": "1.0.1",
1140 | "methods": "~1.1.2",
1141 | "on-finished": "~2.3.0",
1142 | "parseurl": "~1.3.3",
1143 | "path-to-regexp": "0.1.7",
1144 | "proxy-addr": "~2.0.5",
1145 | "qs": "6.7.0",
1146 | "range-parser": "~1.2.1",
1147 | "safe-buffer": "5.1.2",
1148 | "send": "0.17.1",
1149 | "serve-static": "1.14.1",
1150 | "setprototypeof": "1.1.1",
1151 | "statuses": "~1.5.0",
1152 | "type-is": "~1.6.18",
1153 | "utils-merge": "1.0.1",
1154 | "vary": "~1.1.2"
1155 | },
1156 | "dependencies": {
1157 | "safe-buffer": {
1158 | "version": "5.1.2",
1159 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz",
1160 | "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g=="
1161 | },
1162 | "setprototypeof": {
1163 | "version": "1.1.1",
1164 | "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.1.tgz",
1165 | "integrity": "sha512-JvdAWfbXeIGaZ9cILp38HntZSFSo3mWg6xGcJJsd+d4aRMOqauag1C63dJfDw7OaMYwEbHMOxEZ1lqVRYP2OAw=="
1166 | }
1167 | }
1168 | },
1169 | "fast-json-stable-stringify": {
1170 | "version": "2.1.0",
1171 | "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz",
1172 | "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw=="
1173 | },
1174 | "fill-range": {
1175 | "version": "7.0.1",
1176 | "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz",
1177 | "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==",
1178 | "dev": true,
1179 | "requires": {
1180 | "to-regex-range": "^5.0.1"
1181 | }
1182 | },
1183 | "finalhandler": {
1184 | "version": "1.1.2",
1185 | "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.1.2.tgz",
1186 | "integrity": "sha512-aAWcW57uxVNrQZqFXjITpW3sIUQmHGG3qSb9mUah9MgMC4NeWhNOlNjXEYq3HjRAvL6arUviZGGJsBg6z0zsWA==",
1187 | "requires": {
1188 | "debug": "2.6.9",
1189 | "encodeurl": "~1.0.2",
1190 | "escape-html": "~1.0.3",
1191 | "on-finished": "~2.3.0",
1192 | "parseurl": "~1.3.3",
1193 | "statuses": "~1.5.0",
1194 | "unpipe": "~1.0.0"
1195 | }
1196 | },
1197 | "form-data": {
1198 | "version": "3.0.0",
1199 | "resolved": "https://registry.npmjs.org/form-data/-/form-data-3.0.0.tgz",
1200 | "integrity": "sha512-CKMFDglpbMi6PyN+brwB9Q/GOw0eAnsrEZDgcsH5Krhz5Od/haKHAX0NmQfha2zPPz0JpWzA7GJHGSnvCRLWsg==",
1201 | "requires": {
1202 | "asynckit": "^0.4.0",
1203 | "combined-stream": "^1.0.8",
1204 | "mime-types": "^2.1.12"
1205 | }
1206 | },
1207 | "forwarded": {
1208 | "version": "0.1.2",
1209 | "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.1.2.tgz",
1210 | "integrity": "sha1-mMI9qxF1ZXuMBXPozszZGw/xjIQ="
1211 | },
1212 | "fresh": {
1213 | "version": "0.5.2",
1214 | "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz",
1215 | "integrity": "sha1-PYyt2Q2XZWn6g1qx+OSyOhBWBac="
1216 | },
1217 | "fs-capacitor": {
1218 | "version": "2.0.4",
1219 | "resolved": "https://registry.npmjs.org/fs-capacitor/-/fs-capacitor-2.0.4.tgz",
1220 | "integrity": "sha512-8S4f4WsCryNw2mJJchi46YgB6CR5Ze+4L1h8ewl9tEpL4SJ3ZO+c/bS4BWhB8bK+O3TMqhuZarTitd0S0eh2pA=="
1221 | },
1222 | "fsevents": {
1223 | "version": "2.1.3",
1224 | "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.1.3.tgz",
1225 | "integrity": "sha512-Auw9a4AxqWpa9GUfj370BMPzzyncfBABW8Mab7BGWBYDj4Isgq+cDKtx0i6u9jcX9pQDnswsaaOTgTmA5pEjuQ==",
1226 | "dev": true,
1227 | "optional": true
1228 | },
1229 | "function-bind": {
1230 | "version": "1.1.1",
1231 | "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.1.tgz",
1232 | "integrity": "sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A=="
1233 | },
1234 | "generate-function": {
1235 | "version": "2.3.1",
1236 | "resolved": "https://registry.npmjs.org/generate-function/-/generate-function-2.3.1.tgz",
1237 | "integrity": "sha512-eeB5GfMNeevm/GRYq20ShmsaGcmI81kIX2K9XQx5miC8KdHaC6Jm0qQ8ZNeGOi7wYB8OsdxKs+Y2oVuTFuVwKQ==",
1238 | "requires": {
1239 | "is-property": "^1.0.2"
1240 | }
1241 | },
1242 | "get-stream": {
1243 | "version": "4.1.0",
1244 | "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-4.1.0.tgz",
1245 | "integrity": "sha512-GMat4EJ5161kIy2HevLlr4luNjBgvmj413KaQA7jt4V8B4RDsfpHk7WQ9GVqfYyyx8OS/L66Kox+rJRNklLK7w==",
1246 | "dev": true,
1247 | "requires": {
1248 | "pump": "^3.0.0"
1249 | }
1250 | },
1251 | "glob-parent": {
1252 | "version": "5.1.1",
1253 | "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.1.tgz",
1254 | "integrity": "sha512-FnI+VGOpnlGHWZxthPGR+QhR78fuiK0sNLkHQv+bL9fQi57lNNdquIbna/WrfROrolq8GK5Ek6BiMwqL/voRYQ==",
1255 | "dev": true,
1256 | "requires": {
1257 | "is-glob": "^4.0.1"
1258 | }
1259 | },
1260 | "global-dirs": {
1261 | "version": "2.0.1",
1262 | "resolved": "https://registry.npmjs.org/global-dirs/-/global-dirs-2.0.1.tgz",
1263 | "integrity": "sha512-5HqUqdhkEovj2Of/ms3IeS/EekcO54ytHRLV4PEY2rhRwrHXLQjeVEES0Lhka0xwNDtGYn58wyC4s5+MHsOO6A==",
1264 | "dev": true,
1265 | "requires": {
1266 | "ini": "^1.3.5"
1267 | }
1268 | },
1269 | "got": {
1270 | "version": "9.6.0",
1271 | "resolved": "https://registry.npmjs.org/got/-/got-9.6.0.tgz",
1272 | "integrity": "sha512-R7eWptXuGYxwijs0eV+v3o6+XH1IqVK8dJOEecQfTmkncw9AV4dcw/Dhxi8MdlqPthxxpZyizMzyg8RTmEsG+Q==",
1273 | "dev": true,
1274 | "requires": {
1275 | "@sindresorhus/is": "^0.14.0",
1276 | "@szmarczak/http-timer": "^1.1.2",
1277 | "cacheable-request": "^6.0.0",
1278 | "decompress-response": "^3.3.0",
1279 | "duplexer3": "^0.1.4",
1280 | "get-stream": "^4.1.0",
1281 | "lowercase-keys": "^1.0.1",
1282 | "mimic-response": "^1.0.1",
1283 | "p-cancelable": "^1.0.0",
1284 | "to-readable-stream": "^1.0.0",
1285 | "url-parse-lax": "^3.0.0"
1286 | }
1287 | },
1288 | "graceful-fs": {
1289 | "version": "4.2.4",
1290 | "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.4.tgz",
1291 | "integrity": "sha512-WjKPNJF79dtJAVniUlGGWHYGz2jWxT6VhN/4m1NdkbZ2nOsEF+cI1Edgql5zCRhs/VsQYRvrXctxktVXZUkixw==",
1292 | "dev": true
1293 | },
1294 | "graphql": {
1295 | "version": "15.3.0",
1296 | "resolved": "https://registry.npmjs.org/graphql/-/graphql-15.3.0.tgz",
1297 | "integrity": "sha512-GTCJtzJmkFLWRfFJuoo9RWWa/FfamUHgiFosxi/X1Ani4AVWbeyBenZTNX6dM+7WSbbFfTo/25eh0LLkwHMw2w=="
1298 | },
1299 | "graphql-extensions": {
1300 | "version": "0.12.4",
1301 | "resolved": "https://registry.npmjs.org/graphql-extensions/-/graphql-extensions-0.12.4.tgz",
1302 | "integrity": "sha512-GnR4LiWk3s2bGOqIh6V1JgnSXw2RCH4NOgbCFEWvB6JqWHXTlXnLZ8bRSkCiD4pltv7RHUPWqN/sGh8R6Ae/ag==",
1303 | "requires": {
1304 | "@apollographql/apollo-tools": "^0.4.3",
1305 | "apollo-server-env": "^2.4.5",
1306 | "apollo-server-types": "^0.5.1"
1307 | }
1308 | },
1309 | "graphql-subscriptions": {
1310 | "version": "1.1.0",
1311 | "resolved": "https://registry.npmjs.org/graphql-subscriptions/-/graphql-subscriptions-1.1.0.tgz",
1312 | "integrity": "sha512-6WzlBFC0lWmXJbIVE8OgFgXIP4RJi3OQgTPa0DVMsDXdpRDjTsM1K9wfl5HSYX7R87QAGlvcv2Y4BIZa/ItonA==",
1313 | "requires": {
1314 | "iterall": "^1.2.1"
1315 | }
1316 | },
1317 | "graphql-tag": {
1318 | "version": "2.11.0",
1319 | "resolved": "https://registry.npmjs.org/graphql-tag/-/graphql-tag-2.11.0.tgz",
1320 | "integrity": "sha512-VmsD5pJqWJnQZMUeRwrDhfgoyqcfwEkvtpANqcoUG8/tOLkwNgU9mzub/Mc78OJMhHjx7gfAMTxzdG43VGg3bA=="
1321 | },
1322 | "graphql-tools": {
1323 | "version": "4.0.8",
1324 | "resolved": "https://registry.npmjs.org/graphql-tools/-/graphql-tools-4.0.8.tgz",
1325 | "integrity": "sha512-MW+ioleBrwhRjalKjYaLQbr+920pHBgy9vM/n47sswtns8+96sRn5M/G+J1eu7IMeKWiN/9p6tmwCHU7552VJg==",
1326 | "requires": {
1327 | "apollo-link": "^1.2.14",
1328 | "apollo-utilities": "^1.0.1",
1329 | "deprecated-decorator": "^0.1.6",
1330 | "iterall": "^1.1.3",
1331 | "uuid": "^3.1.0"
1332 | },
1333 | "dependencies": {
1334 | "uuid": {
1335 | "version": "3.4.0",
1336 | "resolved": "https://registry.npmjs.org/uuid/-/uuid-3.4.0.tgz",
1337 | "integrity": "sha512-HjSDRw6gZE5JMggctHBcjVak08+KEVhSIiDzFnT9S9aegmp85S/bReBVTb4QTFaRNptJ9kuYaNhnbNEOkbKb/A=="
1338 | }
1339 | }
1340 | },
1341 | "graphql-upload": {
1342 | "version": "8.1.0",
1343 | "resolved": "https://registry.npmjs.org/graphql-upload/-/graphql-upload-8.1.0.tgz",
1344 | "integrity": "sha512-U2OiDI5VxYmzRKw0Z2dmfk0zkqMRaecH9Smh1U277gVgVe9Qn+18xqf4skwr4YJszGIh7iQDZ57+5ygOK9sM/Q==",
1345 | "requires": {
1346 | "busboy": "^0.3.1",
1347 | "fs-capacitor": "^2.0.4",
1348 | "http-errors": "^1.7.3",
1349 | "object-path": "^0.11.4"
1350 | }
1351 | },
1352 | "has": {
1353 | "version": "1.0.3",
1354 | "resolved": "https://registry.npmjs.org/has/-/has-1.0.3.tgz",
1355 | "integrity": "sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw==",
1356 | "requires": {
1357 | "function-bind": "^1.1.1"
1358 | }
1359 | },
1360 | "has-flag": {
1361 | "version": "3.0.0",
1362 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz",
1363 | "integrity": "sha1-tdRU3CGZriJWmfNGfloH87lVuv0=",
1364 | "dev": true
1365 | },
1366 | "has-symbols": {
1367 | "version": "1.0.1",
1368 | "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.0.1.tgz",
1369 | "integrity": "sha512-PLcsoqu++dmEIZB+6totNFKq/7Do+Z0u4oT0zKOJNl3lYK6vGwwu2hjHs+68OEZbTjiUE9bgOABXbP/GvrS0Kg=="
1370 | },
1371 | "has-yarn": {
1372 | "version": "2.1.0",
1373 | "resolved": "https://registry.npmjs.org/has-yarn/-/has-yarn-2.1.0.tgz",
1374 | "integrity": "sha512-UqBRqi4ju7T+TqGNdqAO0PaSVGsDGJUBQvk9eUWNGRY1CFGDzYhLWoM7JQEemnlvVcv/YEmc2wNW8BC24EnUsw==",
1375 | "dev": true
1376 | },
1377 | "http-cache-semantics": {
1378 | "version": "4.1.0",
1379 | "resolved": "https://registry.npmjs.org/http-cache-semantics/-/http-cache-semantics-4.1.0.tgz",
1380 | "integrity": "sha512-carPklcUh7ROWRK7Cv27RPtdhYhUsela/ue5/jKzjegVvXDqM2ILE9Q2BGn9JZJh1g87cp56su/FgQSzcWS8cQ==",
1381 | "dev": true
1382 | },
1383 | "http-errors": {
1384 | "version": "1.8.0",
1385 | "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.8.0.tgz",
1386 | "integrity": "sha512-4I8r0C5JDhT5VkvI47QktDW75rNlGVsUf/8hzjCC/wkWI/jdTRmBb9aI7erSG82r1bjKY3F6k28WnsVxB1C73A==",
1387 | "requires": {
1388 | "depd": "~1.1.2",
1389 | "inherits": "2.0.4",
1390 | "setprototypeof": "1.2.0",
1391 | "statuses": ">= 1.5.0 < 2",
1392 | "toidentifier": "1.0.0"
1393 | }
1394 | },
1395 | "iconv-lite": {
1396 | "version": "0.4.24",
1397 | "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz",
1398 | "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==",
1399 | "requires": {
1400 | "safer-buffer": ">= 2.1.2 < 3"
1401 | }
1402 | },
1403 | "ignore-by-default": {
1404 | "version": "1.0.1",
1405 | "resolved": "https://registry.npmjs.org/ignore-by-default/-/ignore-by-default-1.0.1.tgz",
1406 | "integrity": "sha1-SMptcvbGo68Aqa1K5odr44ieKwk=",
1407 | "dev": true
1408 | },
1409 | "import-lazy": {
1410 | "version": "2.1.0",
1411 | "resolved": "https://registry.npmjs.org/import-lazy/-/import-lazy-2.1.0.tgz",
1412 | "integrity": "sha1-BWmOPUXIjo1+nZLLBYTnfwlvPkM=",
1413 | "dev": true
1414 | },
1415 | "imurmurhash": {
1416 | "version": "0.1.4",
1417 | "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz",
1418 | "integrity": "sha1-khi5srkoojixPcT7a21XbyMUU+o=",
1419 | "dev": true
1420 | },
1421 | "inflection": {
1422 | "version": "1.12.0",
1423 | "resolved": "https://registry.npmjs.org/inflection/-/inflection-1.12.0.tgz",
1424 | "integrity": "sha1-ogCTVlbW9fa8TcdQLhrstwMihBY="
1425 | },
1426 | "inherits": {
1427 | "version": "2.0.4",
1428 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz",
1429 | "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ=="
1430 | },
1431 | "ini": {
1432 | "version": "1.3.5",
1433 | "resolved": "https://registry.npmjs.org/ini/-/ini-1.3.5.tgz",
1434 | "integrity": "sha512-RZY5huIKCMRWDUqZlEi72f/lmXKMvuszcMBduliQ3nnWbx9X/ZBQO7DijMEYS9EhHBb2qacRUMtC7svLwe0lcw==",
1435 | "dev": true
1436 | },
1437 | "ipaddr.js": {
1438 | "version": "1.9.1",
1439 | "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz",
1440 | "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g=="
1441 | },
1442 | "is-binary-path": {
1443 | "version": "2.1.0",
1444 | "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz",
1445 | "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==",
1446 | "dev": true,
1447 | "requires": {
1448 | "binary-extensions": "^2.0.0"
1449 | }
1450 | },
1451 | "is-callable": {
1452 | "version": "1.2.0",
1453 | "resolved": "https://registry.npmjs.org/is-callable/-/is-callable-1.2.0.tgz",
1454 | "integrity": "sha512-pyVD9AaGLxtg6srb2Ng6ynWJqkHU9bEM087AKck0w8QwDarTfNcpIYoU8x8Hv2Icm8u6kFJM18Dag8lyqGkviw=="
1455 | },
1456 | "is-ci": {
1457 | "version": "2.0.0",
1458 | "resolved": "https://registry.npmjs.org/is-ci/-/is-ci-2.0.0.tgz",
1459 | "integrity": "sha512-YfJT7rkpQB0updsdHLGWrvhBJfcfzNNawYDNIyQXJz0IViGf75O8EBPKSdvw2rF+LGCsX4FZ8tcr3b19LcZq4w==",
1460 | "dev": true,
1461 | "requires": {
1462 | "ci-info": "^2.0.0"
1463 | }
1464 | },
1465 | "is-date-object": {
1466 | "version": "1.0.2",
1467 | "resolved": "https://registry.npmjs.org/is-date-object/-/is-date-object-1.0.2.tgz",
1468 | "integrity": "sha512-USlDT524woQ08aoZFzh3/Z6ch9Y/EWXEHQ/AaRN0SkKq4t2Jw2R2339tSXmwuVoY7LLlBCbOIlx2myP/L5zk0g=="
1469 | },
1470 | "is-extglob": {
1471 | "version": "2.1.1",
1472 | "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz",
1473 | "integrity": "sha1-qIwCU1eR8C7TfHahueqXc8gz+MI=",
1474 | "dev": true
1475 | },
1476 | "is-fullwidth-code-point": {
1477 | "version": "2.0.0",
1478 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz",
1479 | "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=",
1480 | "dev": true
1481 | },
1482 | "is-glob": {
1483 | "version": "4.0.1",
1484 | "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.1.tgz",
1485 | "integrity": "sha512-5G0tKtBTFImOqDnLB2hG6Bp2qcKEFduo4tZu9MT/H6NQv/ghhy30o55ufafxJ/LdH79LLs2Kfrn85TLKyA7BUg==",
1486 | "dev": true,
1487 | "requires": {
1488 | "is-extglob": "^2.1.1"
1489 | }
1490 | },
1491 | "is-installed-globally": {
1492 | "version": "0.3.2",
1493 | "resolved": "https://registry.npmjs.org/is-installed-globally/-/is-installed-globally-0.3.2.tgz",
1494 | "integrity": "sha512-wZ8x1js7Ia0kecP/CHM/3ABkAmujX7WPvQk6uu3Fly/Mk44pySulQpnHG46OMjHGXApINnV4QhY3SWnECO2z5g==",
1495 | "dev": true,
1496 | "requires": {
1497 | "global-dirs": "^2.0.1",
1498 | "is-path-inside": "^3.0.1"
1499 | }
1500 | },
1501 | "is-npm": {
1502 | "version": "4.0.0",
1503 | "resolved": "https://registry.npmjs.org/is-npm/-/is-npm-4.0.0.tgz",
1504 | "integrity": "sha512-96ECIfh9xtDDlPylNPXhzjsykHsMJZ18ASpaWzQyBr4YRTcVjUvzaHayDAES2oU/3KpljhHUjtSRNiDwi0F0ig==",
1505 | "dev": true
1506 | },
1507 | "is-number": {
1508 | "version": "7.0.0",
1509 | "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz",
1510 | "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==",
1511 | "dev": true
1512 | },
1513 | "is-obj": {
1514 | "version": "2.0.0",
1515 | "resolved": "https://registry.npmjs.org/is-obj/-/is-obj-2.0.0.tgz",
1516 | "integrity": "sha512-drqDG3cbczxxEJRoOXcOjtdp1J/lyp1mNn0xaznRs8+muBhgQcrnbspox5X5fOw0HnMnbfDzvnEMEtqDEJEo8w==",
1517 | "dev": true
1518 | },
1519 | "is-path-inside": {
1520 | "version": "3.0.2",
1521 | "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-3.0.2.tgz",
1522 | "integrity": "sha512-/2UGPSgmtqwo1ktx8NDHjuPwZWmHhO+gj0f93EkhLB5RgW9RZevWYYlIkS6zePc6U2WpOdQYIwHe9YC4DWEBVg==",
1523 | "dev": true
1524 | },
1525 | "is-property": {
1526 | "version": "1.0.2",
1527 | "resolved": "https://registry.npmjs.org/is-property/-/is-property-1.0.2.tgz",
1528 | "integrity": "sha1-V/4cTkhHTt1lsJkR8msc1Ald2oQ="
1529 | },
1530 | "is-regex": {
1531 | "version": "1.1.1",
1532 | "resolved": "https://registry.npmjs.org/is-regex/-/is-regex-1.1.1.tgz",
1533 | "integrity": "sha512-1+QkEcxiLlB7VEyFtyBg94e08OAsvq7FUBgApTq/w2ymCLyKJgDPsybBENVtA7XCQEgEXxKPonG+mvYRxh/LIg==",
1534 | "requires": {
1535 | "has-symbols": "^1.0.1"
1536 | }
1537 | },
1538 | "is-symbol": {
1539 | "version": "1.0.3",
1540 | "resolved": "https://registry.npmjs.org/is-symbol/-/is-symbol-1.0.3.tgz",
1541 | "integrity": "sha512-OwijhaRSgqvhm/0ZdAcXNZt9lYdKFpcRDT5ULUuYXPoT794UNOdU+gpT6Rzo7b4V2HUl/op6GqY894AZwv9faQ==",
1542 | "requires": {
1543 | "has-symbols": "^1.0.1"
1544 | }
1545 | },
1546 | "is-typedarray": {
1547 | "version": "1.0.0",
1548 | "resolved": "https://registry.npmjs.org/is-typedarray/-/is-typedarray-1.0.0.tgz",
1549 | "integrity": "sha1-5HnICFjfDBsR3dppQPlgEfzaSpo=",
1550 | "dev": true
1551 | },
1552 | "is-yarn-global": {
1553 | "version": "0.3.0",
1554 | "resolved": "https://registry.npmjs.org/is-yarn-global/-/is-yarn-global-0.3.0.tgz",
1555 | "integrity": "sha512-VjSeb/lHmkoyd8ryPVIKvOCn4D1koMqY+vqyjjUfc3xyKtP4dYOxM44sZrnqQSzSds3xyOrUTLTC9LVCVgLngw==",
1556 | "dev": true
1557 | },
1558 | "iterall": {
1559 | "version": "1.3.0",
1560 | "resolved": "https://registry.npmjs.org/iterall/-/iterall-1.3.0.tgz",
1561 | "integrity": "sha512-QZ9qOMdF+QLHxy1QIpUHUU1D5pS2CG2P69LF6L6CPjPYA/XMOmKV3PZpawHoAjHNyB0swdVTRxdYT4tbBbxqwg=="
1562 | },
1563 | "json-buffer": {
1564 | "version": "3.0.0",
1565 | "resolved": "https://registry.npmjs.org/json-buffer/-/json-buffer-3.0.0.tgz",
1566 | "integrity": "sha1-Wx85evx11ne96Lz8Dkfh+aPZqJg=",
1567 | "dev": true
1568 | },
1569 | "jsonwebtoken": {
1570 | "version": "8.5.1",
1571 | "resolved": "https://registry.npmjs.org/jsonwebtoken/-/jsonwebtoken-8.5.1.tgz",
1572 | "integrity": "sha512-XjwVfRS6jTMsqYs0EsuJ4LGxXV14zQybNd4L2r0UvbVnSF9Af8x7p5MzbJ90Ioz/9TI41/hTCvznF/loiSzn8w==",
1573 | "requires": {
1574 | "jws": "^3.2.2",
1575 | "lodash.includes": "^4.3.0",
1576 | "lodash.isboolean": "^3.0.3",
1577 | "lodash.isinteger": "^4.0.4",
1578 | "lodash.isnumber": "^3.0.3",
1579 | "lodash.isplainobject": "^4.0.6",
1580 | "lodash.isstring": "^4.0.1",
1581 | "lodash.once": "^4.0.0",
1582 | "ms": "^2.1.1",
1583 | "semver": "^5.6.0"
1584 | },
1585 | "dependencies": {
1586 | "ms": {
1587 | "version": "2.1.2",
1588 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz",
1589 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w=="
1590 | }
1591 | }
1592 | },
1593 | "jwa": {
1594 | "version": "1.4.1",
1595 | "resolved": "https://registry.npmjs.org/jwa/-/jwa-1.4.1.tgz",
1596 | "integrity": "sha512-qiLX/xhEEFKUAJ6FiBMbes3w9ATzyk5W7Hvzpa/SLYdxNtng+gcurvrI7TbACjIXlsJyr05/S1oUhZrc63evQA==",
1597 | "requires": {
1598 | "buffer-equal-constant-time": "1.0.1",
1599 | "ecdsa-sig-formatter": "1.0.11",
1600 | "safe-buffer": "^5.0.1"
1601 | }
1602 | },
1603 | "jws": {
1604 | "version": "3.2.2",
1605 | "resolved": "https://registry.npmjs.org/jws/-/jws-3.2.2.tgz",
1606 | "integrity": "sha512-YHlZCB6lMTllWDtSPHz/ZXTsi8S00usEV6v1tjq8tOUZzw7DpSDWVXjXDre6ed1w/pd495ODpHZYSdkRTsa0HA==",
1607 | "requires": {
1608 | "jwa": "^1.4.1",
1609 | "safe-buffer": "^5.0.1"
1610 | }
1611 | },
1612 | "keyv": {
1613 | "version": "3.1.0",
1614 | "resolved": "https://registry.npmjs.org/keyv/-/keyv-3.1.0.tgz",
1615 | "integrity": "sha512-9ykJ/46SN/9KPM/sichzQ7OvXyGDYKGTaDlKMGCAlg2UK8KRy4jb0d8sFc+0Tt0YYnThq8X2RZgCg74RPxgcVA==",
1616 | "dev": true,
1617 | "requires": {
1618 | "json-buffer": "3.0.0"
1619 | }
1620 | },
1621 | "latest-version": {
1622 | "version": "5.1.0",
1623 | "resolved": "https://registry.npmjs.org/latest-version/-/latest-version-5.1.0.tgz",
1624 | "integrity": "sha512-weT+r0kTkRQdCdYCNtkMwWXQTMEswKrFBkm4ckQOMVhhqhIMI1UT2hMj+1iigIhgSZm5gTmrRXBNoGUgaTY1xA==",
1625 | "dev": true,
1626 | "requires": {
1627 | "package-json": "^6.3.0"
1628 | }
1629 | },
1630 | "lodash": {
1631 | "version": "4.17.19",
1632 | "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.19.tgz",
1633 | "integrity": "sha512-JNvd8XER9GQX0v2qJgsaN/mzFCNA5BRe/j8JN9d+tWyGLSodKQHKFicdwNYzWwI3wjRnaKPsGj1XkBjx/F96DQ=="
1634 | },
1635 | "lodash.includes": {
1636 | "version": "4.3.0",
1637 | "resolved": "https://registry.npmjs.org/lodash.includes/-/lodash.includes-4.3.0.tgz",
1638 | "integrity": "sha1-YLuYqHy5I8aMoeUTJUgzFISfVT8="
1639 | },
1640 | "lodash.isboolean": {
1641 | "version": "3.0.3",
1642 | "resolved": "https://registry.npmjs.org/lodash.isboolean/-/lodash.isboolean-3.0.3.tgz",
1643 | "integrity": "sha1-bC4XHbKiV82WgC/UOwGyDV9YcPY="
1644 | },
1645 | "lodash.isinteger": {
1646 | "version": "4.0.4",
1647 | "resolved": "https://registry.npmjs.org/lodash.isinteger/-/lodash.isinteger-4.0.4.tgz",
1648 | "integrity": "sha1-YZwK89A/iwTDH1iChAt3sRzWg0M="
1649 | },
1650 | "lodash.isnumber": {
1651 | "version": "3.0.3",
1652 | "resolved": "https://registry.npmjs.org/lodash.isnumber/-/lodash.isnumber-3.0.3.tgz",
1653 | "integrity": "sha1-POdoEMWSjQM1IwGsKHMX8RwLH/w="
1654 | },
1655 | "lodash.isplainobject": {
1656 | "version": "4.0.6",
1657 | "resolved": "https://registry.npmjs.org/lodash.isplainobject/-/lodash.isplainobject-4.0.6.tgz",
1658 | "integrity": "sha1-fFJqUtibRcRcxpC4gWO+BJf1UMs="
1659 | },
1660 | "lodash.isstring": {
1661 | "version": "4.0.1",
1662 | "resolved": "https://registry.npmjs.org/lodash.isstring/-/lodash.isstring-4.0.1.tgz",
1663 | "integrity": "sha1-1SfftUVuynzJu5XV2ur4i6VKVFE="
1664 | },
1665 | "lodash.once": {
1666 | "version": "4.1.1",
1667 | "resolved": "https://registry.npmjs.org/lodash.once/-/lodash.once-4.1.1.tgz",
1668 | "integrity": "sha1-DdOXEhPHxW34gJd9UEyI+0cal6w="
1669 | },
1670 | "lodash.sortby": {
1671 | "version": "4.7.0",
1672 | "resolved": "https://registry.npmjs.org/lodash.sortby/-/lodash.sortby-4.7.0.tgz",
1673 | "integrity": "sha1-7dFMgk4sycHgsKG0K7UhBRakJDg="
1674 | },
1675 | "loglevel": {
1676 | "version": "1.6.8",
1677 | "resolved": "https://registry.npmjs.org/loglevel/-/loglevel-1.6.8.tgz",
1678 | "integrity": "sha512-bsU7+gc9AJ2SqpzxwU3+1fedl8zAntbtC5XYlt3s2j1hJcn2PsXSmgN8TaLG/J1/2mod4+cE/3vNL70/c1RNCA=="
1679 | },
1680 | "long": {
1681 | "version": "4.0.0",
1682 | "resolved": "https://registry.npmjs.org/long/-/long-4.0.0.tgz",
1683 | "integrity": "sha512-XsP+KhQif4bjX1kbuSiySJFNAehNxgLb6hPRGJ9QsUr8ajHkuXGdrHmFUTUUXhDwVX2R5bY4JNZEwbUiMhV+MA=="
1684 | },
1685 | "lowercase-keys": {
1686 | "version": "1.0.1",
1687 | "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-1.0.1.tgz",
1688 | "integrity": "sha512-G2Lj61tXDnVFFOi8VZds+SoQjtQC3dgokKdDG2mTm1tx4m50NUHBOZSBwQQHyy0V12A0JTG4icfZQH+xPyh8VA==",
1689 | "dev": true
1690 | },
1691 | "lru-cache": {
1692 | "version": "5.1.1",
1693 | "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-5.1.1.tgz",
1694 | "integrity": "sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==",
1695 | "requires": {
1696 | "yallist": "^3.0.2"
1697 | }
1698 | },
1699 | "make-dir": {
1700 | "version": "3.1.0",
1701 | "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-3.1.0.tgz",
1702 | "integrity": "sha512-g3FeP20LNwhALb/6Cz6Dd4F2ngze0jz7tbzrD2wAV+o9FeNHe4rL+yK2md0J/fiSf1sa1ADhXqi5+oVwOM/eGw==",
1703 | "dev": true,
1704 | "requires": {
1705 | "semver": "^6.0.0"
1706 | },
1707 | "dependencies": {
1708 | "semver": {
1709 | "version": "6.3.0",
1710 | "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz",
1711 | "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==",
1712 | "dev": true
1713 | }
1714 | }
1715 | },
1716 | "media-typer": {
1717 | "version": "0.3.0",
1718 | "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz",
1719 | "integrity": "sha1-hxDXrwqmJvj/+hzgAWhUUmMlV0g="
1720 | },
1721 | "merge-descriptors": {
1722 | "version": "1.0.1",
1723 | "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz",
1724 | "integrity": "sha1-sAqqVW3YtEVoFQ7J0blT8/kMu2E="
1725 | },
1726 | "methods": {
1727 | "version": "1.1.2",
1728 | "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz",
1729 | "integrity": "sha1-VSmk1nZUE07cxSZmVoNbD4Ua/O4="
1730 | },
1731 | "mime": {
1732 | "version": "1.6.0",
1733 | "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz",
1734 | "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg=="
1735 | },
1736 | "mime-db": {
1737 | "version": "1.44.0",
1738 | "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.44.0.tgz",
1739 | "integrity": "sha512-/NOTfLrsPBVeH7YtFPgsVWveuL+4SjjYxaQ1xtM1KMFj7HdxlBlxeyNLzhyJVx7r4rZGJAZ/6lkKCitSc/Nmpg=="
1740 | },
1741 | "mime-types": {
1742 | "version": "2.1.27",
1743 | "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.27.tgz",
1744 | "integrity": "sha512-JIhqnCasI9yD+SsmkquHBxTSEuZdQX5BuQnS2Vc7puQQQ+8yiP5AY5uWhpdv4YL4VM5c6iliiYWPgJ/nJQLp7w==",
1745 | "requires": {
1746 | "mime-db": "1.44.0"
1747 | }
1748 | },
1749 | "mimic-response": {
1750 | "version": "1.0.1",
1751 | "resolved": "https://registry.npmjs.org/mimic-response/-/mimic-response-1.0.1.tgz",
1752 | "integrity": "sha512-j5EctnkH7amfV/q5Hgmoal1g2QHFJRraOtmx0JpIqkxhBhI/lJSl1nMpQ45hVarwNETOoWEimndZ4QK0RHxuxQ==",
1753 | "dev": true
1754 | },
1755 | "minimatch": {
1756 | "version": "3.0.4",
1757 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz",
1758 | "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==",
1759 | "dev": true,
1760 | "requires": {
1761 | "brace-expansion": "^1.1.7"
1762 | }
1763 | },
1764 | "minimist": {
1765 | "version": "1.2.5",
1766 | "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.5.tgz",
1767 | "integrity": "sha512-FM9nNUYrRBAELZQT3xeZQ7fmMOBg6nWNmJKTcgsJeaLstP/UODVpGsr5OhXhhXg6f+qtJ8uiZ+PUxkDWcgIXLw==",
1768 | "dev": true
1769 | },
1770 | "moment": {
1771 | "version": "2.27.0",
1772 | "resolved": "https://registry.npmjs.org/moment/-/moment-2.27.0.tgz",
1773 | "integrity": "sha512-al0MUK7cpIcglMv3YF13qSgdAIqxHTO7brRtaz3DlSULbqfazqkc5kEjNrLDOM7fsjshoFIihnU8snrP7zUvhQ=="
1774 | },
1775 | "moment-timezone": {
1776 | "version": "0.5.31",
1777 | "resolved": "https://registry.npmjs.org/moment-timezone/-/moment-timezone-0.5.31.tgz",
1778 | "integrity": "sha512-+GgHNg8xRhMXfEbv81iDtrVeTcWt0kWmTEY1XQK14dICTXnWJnT0dxdlPspwqF3keKMVPXwayEsk1DI0AA/jdA==",
1779 | "requires": {
1780 | "moment": ">= 2.9.0"
1781 | }
1782 | },
1783 | "ms": {
1784 | "version": "2.0.0",
1785 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz",
1786 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g="
1787 | },
1788 | "mysql2": {
1789 | "version": "2.1.0",
1790 | "resolved": "https://registry.npmjs.org/mysql2/-/mysql2-2.1.0.tgz",
1791 | "integrity": "sha512-9kGVyi930rG2KaHrz3sHwtc6K+GY9d8wWk1XRSYxQiunvGcn4DwuZxOwmK11ftuhhwrYDwGx9Ta4VBwznJn36A==",
1792 | "requires": {
1793 | "cardinal": "^2.1.1",
1794 | "denque": "^1.4.1",
1795 | "generate-function": "^2.3.1",
1796 | "iconv-lite": "^0.5.0",
1797 | "long": "^4.0.0",
1798 | "lru-cache": "^5.1.1",
1799 | "named-placeholders": "^1.1.2",
1800 | "seq-queue": "^0.0.5",
1801 | "sqlstring": "^2.3.1"
1802 | },
1803 | "dependencies": {
1804 | "iconv-lite": {
1805 | "version": "0.5.2",
1806 | "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.5.2.tgz",
1807 | "integrity": "sha512-kERHXvpSaB4aU3eANwidg79K8FlrN77m8G9V+0vOR3HYaRifrlwMEpT7ZBJqLSEIHnEgJTHcWK82wwLwwKwtag==",
1808 | "requires": {
1809 | "safer-buffer": ">= 2.1.2 < 3"
1810 | }
1811 | }
1812 | }
1813 | },
1814 | "named-placeholders": {
1815 | "version": "1.1.2",
1816 | "resolved": "https://registry.npmjs.org/named-placeholders/-/named-placeholders-1.1.2.tgz",
1817 | "integrity": "sha512-wiFWqxoLL3PGVReSZpjLVxyJ1bRqe+KKJVbr4hGs1KWfTZTQyezHFBbuKj9hsizHyGV2ne7EMjHdxEGAybD5SA==",
1818 | "requires": {
1819 | "lru-cache": "^4.1.3"
1820 | },
1821 | "dependencies": {
1822 | "lru-cache": {
1823 | "version": "4.1.5",
1824 | "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-4.1.5.tgz",
1825 | "integrity": "sha512-sWZlbEP2OsHNkXrMl5GYk/jKk70MBng6UU4YI/qGDYbgf6YbP4EvmqISbXCoJiRKs+1bSpFHVgQxvJ17F2li5g==",
1826 | "requires": {
1827 | "pseudomap": "^1.0.2",
1828 | "yallist": "^2.1.2"
1829 | }
1830 | },
1831 | "yallist": {
1832 | "version": "2.1.2",
1833 | "resolved": "https://registry.npmjs.org/yallist/-/yallist-2.1.2.tgz",
1834 | "integrity": "sha1-HBH5IY8HYImkfdUS+TxmmaaoHVI="
1835 | }
1836 | }
1837 | },
1838 | "negotiator": {
1839 | "version": "0.6.2",
1840 | "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.2.tgz",
1841 | "integrity": "sha512-hZXc7K2e+PgeI1eDBe/10Ard4ekbfrrqG8Ep+8Jmf4JID2bNg7NvCPOZN+kfF574pFQI7mum2AUqDidoKqcTOw=="
1842 | },
1843 | "node-fetch": {
1844 | "version": "2.6.0",
1845 | "resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-2.6.0.tgz",
1846 | "integrity": "sha512-8dG4H5ujfvFiqDmVu9fQ5bOHUC15JMjMY/Zumv26oOvvVJjM67KF8koCWIabKQ1GJIa9r2mMZscBq/TbdOcmNA=="
1847 | },
1848 | "nodemon": {
1849 | "version": "2.0.4",
1850 | "resolved": "https://registry.npmjs.org/nodemon/-/nodemon-2.0.4.tgz",
1851 | "integrity": "sha512-Ltced+hIfTmaS28Zjv1BM552oQ3dbwPqI4+zI0SLgq+wpJhSyqgYude/aZa/3i31VCQWMfXJVxvu86abcam3uQ==",
1852 | "dev": true,
1853 | "requires": {
1854 | "chokidar": "^3.2.2",
1855 | "debug": "^3.2.6",
1856 | "ignore-by-default": "^1.0.1",
1857 | "minimatch": "^3.0.4",
1858 | "pstree.remy": "^1.1.7",
1859 | "semver": "^5.7.1",
1860 | "supports-color": "^5.5.0",
1861 | "touch": "^3.1.0",
1862 | "undefsafe": "^2.0.2",
1863 | "update-notifier": "^4.0.0"
1864 | },
1865 | "dependencies": {
1866 | "debug": {
1867 | "version": "3.2.6",
1868 | "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.6.tgz",
1869 | "integrity": "sha512-mel+jf7nrtEl5Pn1Qx46zARXKDpBbvzezse7p7LqINmdoIk8PYP5SySaxEmYv6TZ0JyEKA1hsCId6DIhgITtWQ==",
1870 | "dev": true,
1871 | "requires": {
1872 | "ms": "^2.1.1"
1873 | }
1874 | },
1875 | "ms": {
1876 | "version": "2.1.2",
1877 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz",
1878 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==",
1879 | "dev": true
1880 | }
1881 | }
1882 | },
1883 | "nopt": {
1884 | "version": "1.0.10",
1885 | "resolved": "https://registry.npmjs.org/nopt/-/nopt-1.0.10.tgz",
1886 | "integrity": "sha1-bd0hvSoxQXuScn3Vhfim83YI6+4=",
1887 | "dev": true,
1888 | "requires": {
1889 | "abbrev": "1"
1890 | }
1891 | },
1892 | "normalize-path": {
1893 | "version": "3.0.0",
1894 | "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz",
1895 | "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==",
1896 | "dev": true
1897 | },
1898 | "normalize-url": {
1899 | "version": "4.5.0",
1900 | "resolved": "https://registry.npmjs.org/normalize-url/-/normalize-url-4.5.0.tgz",
1901 | "integrity": "sha512-2s47yzUxdexf1OhyRi4Em83iQk0aPvwTddtFz4hnSSw9dCEsLEGf6SwIO8ss/19S9iBb5sJaOuTvTGDeZI00BQ==",
1902 | "dev": true
1903 | },
1904 | "object-assign": {
1905 | "version": "4.1.1",
1906 | "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz",
1907 | "integrity": "sha1-IQmtx5ZYh8/AXLvUQsrIv7s2CGM="
1908 | },
1909 | "object-inspect": {
1910 | "version": "1.8.0",
1911 | "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.8.0.tgz",
1912 | "integrity": "sha512-jLdtEOB112fORuypAyl/50VRVIBIdVQOSUUGQHzJ4xBSbit81zRarz7GThkEFZy1RceYrWYcPcBFPQwHyAc1gA=="
1913 | },
1914 | "object-keys": {
1915 | "version": "1.1.1",
1916 | "resolved": "https://registry.npmjs.org/object-keys/-/object-keys-1.1.1.tgz",
1917 | "integrity": "sha512-NuAESUOUMrlIXOfHKzD6bpPu3tYt3xvjNdRIQ+FeT0lNb4K8WR70CaDxhuNguS2XG+GjkyMwOzsN5ZktImfhLA=="
1918 | },
1919 | "object-path": {
1920 | "version": "0.11.4",
1921 | "resolved": "https://registry.npmjs.org/object-path/-/object-path-0.11.4.tgz",
1922 | "integrity": "sha1-NwrnUvvzfePqcKhhwju6iRVpGUk="
1923 | },
1924 | "object.assign": {
1925 | "version": "4.1.0",
1926 | "resolved": "https://registry.npmjs.org/object.assign/-/object.assign-4.1.0.tgz",
1927 | "integrity": "sha512-exHJeq6kBKj58mqGyTQ9DFvrZC/eR6OwxzoM9YRoGBqrXYonaFyGiFMuc9VZrXf7DarreEwMpurG3dd+CNyW5w==",
1928 | "requires": {
1929 | "define-properties": "^1.1.2",
1930 | "function-bind": "^1.1.1",
1931 | "has-symbols": "^1.0.0",
1932 | "object-keys": "^1.0.11"
1933 | }
1934 | },
1935 | "object.getownpropertydescriptors": {
1936 | "version": "2.1.0",
1937 | "resolved": "https://registry.npmjs.org/object.getownpropertydescriptors/-/object.getownpropertydescriptors-2.1.0.tgz",
1938 | "integrity": "sha512-Z53Oah9A3TdLoblT7VKJaTDdXdT+lQO+cNpKVnya5JDe9uLvzu1YyY1yFDFrcxrlRgWrEFH0jJtD/IbuwjcEVg==",
1939 | "requires": {
1940 | "define-properties": "^1.1.3",
1941 | "es-abstract": "^1.17.0-next.1"
1942 | }
1943 | },
1944 | "on-finished": {
1945 | "version": "2.3.0",
1946 | "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.3.0.tgz",
1947 | "integrity": "sha1-IPEzZIGwg811M3mSoWlxqi2QaUc=",
1948 | "requires": {
1949 | "ee-first": "1.1.1"
1950 | }
1951 | },
1952 | "once": {
1953 | "version": "1.4.0",
1954 | "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz",
1955 | "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=",
1956 | "dev": true,
1957 | "requires": {
1958 | "wrappy": "1"
1959 | }
1960 | },
1961 | "p-cancelable": {
1962 | "version": "1.1.0",
1963 | "resolved": "https://registry.npmjs.org/p-cancelable/-/p-cancelable-1.1.0.tgz",
1964 | "integrity": "sha512-s73XxOZ4zpt1edZYZzvhqFa6uvQc1vwUa0K0BdtIZgQMAJj9IbebH+JkgKZc9h+B05PKHLOTl4ajG1BmNrVZlw==",
1965 | "dev": true
1966 | },
1967 | "package-json": {
1968 | "version": "6.5.0",
1969 | "resolved": "https://registry.npmjs.org/package-json/-/package-json-6.5.0.tgz",
1970 | "integrity": "sha512-k3bdm2n25tkyxcjSKzB5x8kfVxlMdgsbPr0GkZcwHsLpba6cBjqCt1KlcChKEvxHIcTB1FVMuwoijZ26xex5MQ==",
1971 | "dev": true,
1972 | "requires": {
1973 | "got": "^9.6.0",
1974 | "registry-auth-token": "^4.0.0",
1975 | "registry-url": "^5.0.0",
1976 | "semver": "^6.2.0"
1977 | },
1978 | "dependencies": {
1979 | "semver": {
1980 | "version": "6.3.0",
1981 | "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz",
1982 | "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==",
1983 | "dev": true
1984 | }
1985 | }
1986 | },
1987 | "parseurl": {
1988 | "version": "1.3.3",
1989 | "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz",
1990 | "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ=="
1991 | },
1992 | "path-to-regexp": {
1993 | "version": "0.1.7",
1994 | "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz",
1995 | "integrity": "sha1-32BBeABfUi8V60SQ5yR6G/qmf4w="
1996 | },
1997 | "picomatch": {
1998 | "version": "2.2.2",
1999 | "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.2.2.tgz",
2000 | "integrity": "sha512-q0M/9eZHzmr0AulXyPwNfZjtwZ/RBZlbN3K3CErVrk50T2ASYI7Bye0EvekFY3IP1Nt2DHu0re+V2ZHIpMkuWg==",
2001 | "dev": true
2002 | },
2003 | "prepend-http": {
2004 | "version": "2.0.0",
2005 | "resolved": "https://registry.npmjs.org/prepend-http/-/prepend-http-2.0.0.tgz",
2006 | "integrity": "sha1-6SQ0v6XqjBn0HN/UAddBo8gZ2Jc=",
2007 | "dev": true
2008 | },
2009 | "proxy-addr": {
2010 | "version": "2.0.6",
2011 | "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.6.tgz",
2012 | "integrity": "sha512-dh/frvCBVmSsDYzw6n926jv974gddhkFPfiN8hPOi30Wax25QZyZEGveluCgliBnqmuM+UJmBErbAUFIoDbjOw==",
2013 | "requires": {
2014 | "forwarded": "~0.1.2",
2015 | "ipaddr.js": "1.9.1"
2016 | }
2017 | },
2018 | "pseudomap": {
2019 | "version": "1.0.2",
2020 | "resolved": "https://registry.npmjs.org/pseudomap/-/pseudomap-1.0.2.tgz",
2021 | "integrity": "sha1-8FKijacOYYkX7wqKw0wa5aaChrM="
2022 | },
2023 | "pstree.remy": {
2024 | "version": "1.1.8",
2025 | "resolved": "https://registry.npmjs.org/pstree.remy/-/pstree.remy-1.1.8.tgz",
2026 | "integrity": "sha512-77DZwxQmxKnu3aR542U+X8FypNzbfJ+C5XQDk3uWjWxn6151aIMGthWYRXTqT1E5oJvg+ljaa2OJi+VfvCOQ8w==",
2027 | "dev": true
2028 | },
2029 | "pump": {
2030 | "version": "3.0.0",
2031 | "resolved": "https://registry.npmjs.org/pump/-/pump-3.0.0.tgz",
2032 | "integrity": "sha512-LwZy+p3SFs1Pytd/jYct4wpv49HiYCqd9Rlc5ZVdk0V+8Yzv6jR5Blk3TRmPL1ft69TxP0IMZGJ+WPFU2BFhww==",
2033 | "dev": true,
2034 | "requires": {
2035 | "end-of-stream": "^1.1.0",
2036 | "once": "^1.3.1"
2037 | }
2038 | },
2039 | "pupa": {
2040 | "version": "2.0.1",
2041 | "resolved": "https://registry.npmjs.org/pupa/-/pupa-2.0.1.tgz",
2042 | "integrity": "sha512-hEJH0s8PXLY/cdXh66tNEQGndDrIKNqNC5xmrysZy3i5C3oEoLna7YAOad+7u125+zH1HNXUmGEkrhb3c2VriA==",
2043 | "dev": true,
2044 | "requires": {
2045 | "escape-goat": "^2.0.0"
2046 | }
2047 | },
2048 | "qs": {
2049 | "version": "6.7.0",
2050 | "resolved": "https://registry.npmjs.org/qs/-/qs-6.7.0.tgz",
2051 | "integrity": "sha512-VCdBRNFTX1fyE7Nb6FYoURo/SPe62QCaAyzJvUjwRaIsc+NePBEniHlvxFmmX56+HZphIGtV0XeCirBtpDrTyQ=="
2052 | },
2053 | "range-parser": {
2054 | "version": "1.2.1",
2055 | "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz",
2056 | "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg=="
2057 | },
2058 | "raw-body": {
2059 | "version": "2.4.0",
2060 | "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.4.0.tgz",
2061 | "integrity": "sha512-4Oz8DUIwdvoa5qMJelxipzi/iJIi40O5cGV1wNYp5hvZP8ZN0T+jiNkL0QepXs+EsQ9XJ8ipEDoiH70ySUJP3Q==",
2062 | "requires": {
2063 | "bytes": "3.1.0",
2064 | "http-errors": "1.7.2",
2065 | "iconv-lite": "0.4.24",
2066 | "unpipe": "1.0.0"
2067 | },
2068 | "dependencies": {
2069 | "http-errors": {
2070 | "version": "1.7.2",
2071 | "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.7.2.tgz",
2072 | "integrity": "sha512-uUQBt3H/cSIVfch6i1EuPNy/YsRSOUBXTVfZ+yR7Zjez3qjBz6i9+i4zjNaoqcoFVI4lQJ5plg63TvGfRSDCRg==",
2073 | "requires": {
2074 | "depd": "~1.1.2",
2075 | "inherits": "2.0.3",
2076 | "setprototypeof": "1.1.1",
2077 | "statuses": ">= 1.5.0 < 2",
2078 | "toidentifier": "1.0.0"
2079 | }
2080 | },
2081 | "inherits": {
2082 | "version": "2.0.3",
2083 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz",
2084 | "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4="
2085 | },
2086 | "setprototypeof": {
2087 | "version": "1.1.1",
2088 | "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.1.tgz",
2089 | "integrity": "sha512-JvdAWfbXeIGaZ9cILp38HntZSFSo3mWg6xGcJJsd+d4aRMOqauag1C63dJfDw7OaMYwEbHMOxEZ1lqVRYP2OAw=="
2090 | }
2091 | }
2092 | },
2093 | "rc": {
2094 | "version": "1.2.8",
2095 | "resolved": "https://registry.npmjs.org/rc/-/rc-1.2.8.tgz",
2096 | "integrity": "sha512-y3bGgqKj3QBdxLbLkomlohkvsA8gdAiUQlSBJnBhfn+BPxg4bc62d8TcBW15wavDfgexCgccckhcZvywyQYPOw==",
2097 | "dev": true,
2098 | "requires": {
2099 | "deep-extend": "^0.6.0",
2100 | "ini": "~1.3.0",
2101 | "minimist": "^1.2.0",
2102 | "strip-json-comments": "~2.0.1"
2103 | }
2104 | },
2105 | "readdirp": {
2106 | "version": "3.4.0",
2107 | "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.4.0.tgz",
2108 | "integrity": "sha512-0xe001vZBnJEK+uKcj8qOhyAKPzIT+gStxWr3LCB0DwcXR5NZJ3IaC+yGnHCYzB/S7ov3m3EEbZI2zeNvX+hGQ==",
2109 | "dev": true,
2110 | "requires": {
2111 | "picomatch": "^2.2.1"
2112 | }
2113 | },
2114 | "redeyed": {
2115 | "version": "2.1.1",
2116 | "resolved": "https://registry.npmjs.org/redeyed/-/redeyed-2.1.1.tgz",
2117 | "integrity": "sha1-iYS1gV2ZyyIEacme7v/jiRPmzAs=",
2118 | "requires": {
2119 | "esprima": "~4.0.0"
2120 | }
2121 | },
2122 | "registry-auth-token": {
2123 | "version": "4.2.0",
2124 | "resolved": "https://registry.npmjs.org/registry-auth-token/-/registry-auth-token-4.2.0.tgz",
2125 | "integrity": "sha512-P+lWzPrsgfN+UEpDS3U8AQKg/UjZX6mQSJueZj3EK+vNESoqBSpBUD3gmu4sF9lOsjXWjF11dQKUqemf3veq1w==",
2126 | "dev": true,
2127 | "requires": {
2128 | "rc": "^1.2.8"
2129 | }
2130 | },
2131 | "registry-url": {
2132 | "version": "5.1.0",
2133 | "resolved": "https://registry.npmjs.org/registry-url/-/registry-url-5.1.0.tgz",
2134 | "integrity": "sha512-8acYXXTI0AkQv6RAOjE3vOaIXZkT9wo4LOFbBKYQEEnnMNBpKqdUrI6S4NT0KPIo/WVvJ5tE/X5LF/TQUf0ekw==",
2135 | "dev": true,
2136 | "requires": {
2137 | "rc": "^1.2.8"
2138 | }
2139 | },
2140 | "responselike": {
2141 | "version": "1.0.2",
2142 | "resolved": "https://registry.npmjs.org/responselike/-/responselike-1.0.2.tgz",
2143 | "integrity": "sha1-kYcg7ztjHFZCvgaPFa3lpG9Loec=",
2144 | "dev": true,
2145 | "requires": {
2146 | "lowercase-keys": "^1.0.0"
2147 | }
2148 | },
2149 | "retry": {
2150 | "version": "0.12.0",
2151 | "resolved": "https://registry.npmjs.org/retry/-/retry-0.12.0.tgz",
2152 | "integrity": "sha1-G0KmJmoh8HQh0bC1S33BZ7AcATs="
2153 | },
2154 | "retry-as-promised": {
2155 | "version": "3.2.0",
2156 | "resolved": "https://registry.npmjs.org/retry-as-promised/-/retry-as-promised-3.2.0.tgz",
2157 | "integrity": "sha512-CybGs60B7oYU/qSQ6kuaFmRd9sTZ6oXSc0toqePvV74Ac6/IFZSI1ReFQmtCN+uvW1Mtqdwpvt/LGOiCBAY2Mg==",
2158 | "requires": {
2159 | "any-promise": "^1.3.0"
2160 | }
2161 | },
2162 | "safe-buffer": {
2163 | "version": "5.2.1",
2164 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz",
2165 | "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ=="
2166 | },
2167 | "safer-buffer": {
2168 | "version": "2.1.2",
2169 | "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz",
2170 | "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg=="
2171 | },
2172 | "semver": {
2173 | "version": "5.7.1",
2174 | "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz",
2175 | "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ=="
2176 | },
2177 | "semver-diff": {
2178 | "version": "3.1.1",
2179 | "resolved": "https://registry.npmjs.org/semver-diff/-/semver-diff-3.1.1.tgz",
2180 | "integrity": "sha512-GX0Ix/CJcHyB8c4ykpHGIAvLyOwOobtM/8d+TQkAd81/bEjgPHrfba41Vpesr7jX/t8Uh+R3EX9eAS5be+jQYg==",
2181 | "dev": true,
2182 | "requires": {
2183 | "semver": "^6.3.0"
2184 | },
2185 | "dependencies": {
2186 | "semver": {
2187 | "version": "6.3.0",
2188 | "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz",
2189 | "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==",
2190 | "dev": true
2191 | }
2192 | }
2193 | },
2194 | "send": {
2195 | "version": "0.17.1",
2196 | "resolved": "https://registry.npmjs.org/send/-/send-0.17.1.tgz",
2197 | "integrity": "sha512-BsVKsiGcQMFwT8UxypobUKyv7irCNRHk1T0G680vk88yf6LBByGcZJOTJCrTP2xVN6yI+XjPJcNuE3V4fT9sAg==",
2198 | "requires": {
2199 | "debug": "2.6.9",
2200 | "depd": "~1.1.2",
2201 | "destroy": "~1.0.4",
2202 | "encodeurl": "~1.0.2",
2203 | "escape-html": "~1.0.3",
2204 | "etag": "~1.8.1",
2205 | "fresh": "0.5.2",
2206 | "http-errors": "~1.7.2",
2207 | "mime": "1.6.0",
2208 | "ms": "2.1.1",
2209 | "on-finished": "~2.3.0",
2210 | "range-parser": "~1.2.1",
2211 | "statuses": "~1.5.0"
2212 | },
2213 | "dependencies": {
2214 | "http-errors": {
2215 | "version": "1.7.3",
2216 | "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.7.3.tgz",
2217 | "integrity": "sha512-ZTTX0MWrsQ2ZAhA1cejAwDLycFsd7I7nVtnkT3Ol0aqodaKW+0CTZDQ1uBv5whptCnc8e8HeRRJxRs0kmm/Qfw==",
2218 | "requires": {
2219 | "depd": "~1.1.2",
2220 | "inherits": "2.0.4",
2221 | "setprototypeof": "1.1.1",
2222 | "statuses": ">= 1.5.0 < 2",
2223 | "toidentifier": "1.0.0"
2224 | }
2225 | },
2226 | "ms": {
2227 | "version": "2.1.1",
2228 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.1.tgz",
2229 | "integrity": "sha512-tgp+dl5cGk28utYktBsrFqA7HKgrhgPsg6Z/EfhWI4gl1Hwq8B/GmY/0oXZ6nF8hDVesS/FpnYaD/kOWhYQvyg=="
2230 | },
2231 | "setprototypeof": {
2232 | "version": "1.1.1",
2233 | "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.1.tgz",
2234 | "integrity": "sha512-JvdAWfbXeIGaZ9cILp38HntZSFSo3mWg6xGcJJsd+d4aRMOqauag1C63dJfDw7OaMYwEbHMOxEZ1lqVRYP2OAw=="
2235 | }
2236 | }
2237 | },
2238 | "seq-queue": {
2239 | "version": "0.0.5",
2240 | "resolved": "https://registry.npmjs.org/seq-queue/-/seq-queue-0.0.5.tgz",
2241 | "integrity": "sha1-1WgS4cAXpuTnw+Ojeh2m143TyT4="
2242 | },
2243 | "sequelize": {
2244 | "version": "6.3.4",
2245 | "resolved": "https://registry.npmjs.org/sequelize/-/sequelize-6.3.4.tgz",
2246 | "integrity": "sha512-W6Y96N5QHTgEz5Q37v2GYbKufSXaw0b3v4rCLTPbcCMfIG0MHI42Ozp7IwiyV9bdNkfFEdY7XP8R6lWrWg4hUw==",
2247 | "requires": {
2248 | "debug": "^4.1.1",
2249 | "dottie": "^2.0.0",
2250 | "inflection": "1.12.0",
2251 | "lodash": "^4.17.15",
2252 | "moment": "^2.26.0",
2253 | "moment-timezone": "^0.5.31",
2254 | "retry-as-promised": "^3.2.0",
2255 | "semver": "^7.3.2",
2256 | "sequelize-pool": "^6.0.0",
2257 | "toposort-class": "^1.0.1",
2258 | "uuid": "^8.1.0",
2259 | "validator": "^10.11.0",
2260 | "wkx": "^0.5.0"
2261 | },
2262 | "dependencies": {
2263 | "debug": {
2264 | "version": "4.1.1",
2265 | "resolved": "https://registry.npmjs.org/debug/-/debug-4.1.1.tgz",
2266 | "integrity": "sha512-pYAIzeRo8J6KPEaJ0VWOh5Pzkbw/RetuzehGM7QRRX5he4fPHx2rdKMB256ehJCkX+XRQm16eZLqLNS8RSZXZw==",
2267 | "requires": {
2268 | "ms": "^2.1.1"
2269 | }
2270 | },
2271 | "ms": {
2272 | "version": "2.1.2",
2273 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz",
2274 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w=="
2275 | },
2276 | "semver": {
2277 | "version": "7.3.2",
2278 | "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.2.tgz",
2279 | "integrity": "sha512-OrOb32TeeambH6UrhtShmF7CRDqhL6/5XpPNp2DuRH6+9QLw/orhp72j87v8Qa1ScDkvrrBNpZcDejAirJmfXQ=="
2280 | }
2281 | }
2282 | },
2283 | "sequelize-pool": {
2284 | "version": "6.1.0",
2285 | "resolved": "https://registry.npmjs.org/sequelize-pool/-/sequelize-pool-6.1.0.tgz",
2286 | "integrity": "sha512-4YwEw3ZgK/tY/so+GfnSgXkdwIJJ1I32uZJztIEgZeAO6HMgj64OzySbWLgxj+tXhZCJnzRfkY9gINw8Ft8ZMg=="
2287 | },
2288 | "serve-static": {
2289 | "version": "1.14.1",
2290 | "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.14.1.tgz",
2291 | "integrity": "sha512-JMrvUwE54emCYWlTI+hGrGv5I8dEwmco/00EvkzIIsR7MqrHonbD9pO2MOfFnpFntl7ecpZs+3mW+XbQZu9QCg==",
2292 | "requires": {
2293 | "encodeurl": "~1.0.2",
2294 | "escape-html": "~1.0.3",
2295 | "parseurl": "~1.3.3",
2296 | "send": "0.17.1"
2297 | }
2298 | },
2299 | "setprototypeof": {
2300 | "version": "1.2.0",
2301 | "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.2.0.tgz",
2302 | "integrity": "sha512-E5LDX7Wrp85Kil5bhZv46j8jOeboKq5JMmYM3gVGdGH8xFpPWXUMsNrlODCrkoxMEeNi/XZIwuRvY4XNwYMJpw=="
2303 | },
2304 | "sha.js": {
2305 | "version": "2.4.11",
2306 | "resolved": "https://registry.npmjs.org/sha.js/-/sha.js-2.4.11.tgz",
2307 | "integrity": "sha512-QMEp5B7cftE7APOjk5Y6xgrbWu+WkLVQwk8JNjZ8nKRciZaByEW6MubieAiToS7+dwvrjGhH8jRXz3MVd0AYqQ==",
2308 | "requires": {
2309 | "inherits": "^2.0.1",
2310 | "safe-buffer": "^5.0.1"
2311 | }
2312 | },
2313 | "signal-exit": {
2314 | "version": "3.0.3",
2315 | "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.3.tgz",
2316 | "integrity": "sha512-VUJ49FC8U1OxwZLxIbTTrDvLnf/6TDgxZcK8wxR8zs13xpx7xbG60ndBlhNrFi2EMuFRoeDoJO7wthSLq42EjA==",
2317 | "dev": true
2318 | },
2319 | "sqlstring": {
2320 | "version": "2.3.2",
2321 | "resolved": "https://registry.npmjs.org/sqlstring/-/sqlstring-2.3.2.tgz",
2322 | "integrity": "sha512-vF4ZbYdKS8OnoJAWBmMxCQDkiEBkGQYU7UZPtL8flbDRSNkhaXvRJ279ZtI6M+zDaQovVU4tuRgzK5fVhvFAhg=="
2323 | },
2324 | "statuses": {
2325 | "version": "1.5.0",
2326 | "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.5.0.tgz",
2327 | "integrity": "sha1-Fhx9rBd2Wf2YEfQ3cfqZOBR4Yow="
2328 | },
2329 | "streamsearch": {
2330 | "version": "0.1.2",
2331 | "resolved": "https://registry.npmjs.org/streamsearch/-/streamsearch-0.1.2.tgz",
2332 | "integrity": "sha1-gIudDlb8Jz2Am6VzOOkpkZoanxo="
2333 | },
2334 | "string-width": {
2335 | "version": "4.2.0",
2336 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.0.tgz",
2337 | "integrity": "sha512-zUz5JD+tgqtuDjMhwIg5uFVV3dtqZ9yQJlZVfq4I01/K5Paj5UHj7VyrQOJvzawSVlKpObApbfD0Ed6yJc+1eg==",
2338 | "dev": true,
2339 | "requires": {
2340 | "emoji-regex": "^8.0.0",
2341 | "is-fullwidth-code-point": "^3.0.0",
2342 | "strip-ansi": "^6.0.0"
2343 | },
2344 | "dependencies": {
2345 | "ansi-regex": {
2346 | "version": "5.0.0",
2347 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.0.tgz",
2348 | "integrity": "sha512-bY6fj56OUQ0hU1KjFNDQuJFezqKdrAyFdIevADiqrWHwSlbmBNMHp5ak2f40Pm8JTFyM2mqxkG6ngkHO11f/lg==",
2349 | "dev": true
2350 | },
2351 | "emoji-regex": {
2352 | "version": "8.0.0",
2353 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz",
2354 | "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==",
2355 | "dev": true
2356 | },
2357 | "is-fullwidth-code-point": {
2358 | "version": "3.0.0",
2359 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz",
2360 | "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==",
2361 | "dev": true
2362 | },
2363 | "strip-ansi": {
2364 | "version": "6.0.0",
2365 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.0.tgz",
2366 | "integrity": "sha512-AuvKTrTfQNYNIctbR1K/YGTR1756GycPsg7b9bdV9Duqur4gv6aKqHXah67Z8ImS7WEz5QVcOtlfW2rZEugt6w==",
2367 | "dev": true,
2368 | "requires": {
2369 | "ansi-regex": "^5.0.0"
2370 | }
2371 | }
2372 | }
2373 | },
2374 | "string.prototype.trimend": {
2375 | "version": "1.0.1",
2376 | "resolved": "https://registry.npmjs.org/string.prototype.trimend/-/string.prototype.trimend-1.0.1.tgz",
2377 | "integrity": "sha512-LRPxFUaTtpqYsTeNKaFOw3R4bxIzWOnbQ837QfBylo8jIxtcbK/A/sMV7Q+OAV/vWo+7s25pOE10KYSjaSO06g==",
2378 | "requires": {
2379 | "define-properties": "^1.1.3",
2380 | "es-abstract": "^1.17.5"
2381 | }
2382 | },
2383 | "string.prototype.trimstart": {
2384 | "version": "1.0.1",
2385 | "resolved": "https://registry.npmjs.org/string.prototype.trimstart/-/string.prototype.trimstart-1.0.1.tgz",
2386 | "integrity": "sha512-XxZn+QpvrBI1FOcg6dIpxUPgWCPuNXvMD72aaRaUQv1eD4e/Qy8i/hFTe0BUmD60p/QA6bh1avmuPTfNjqVWRw==",
2387 | "requires": {
2388 | "define-properties": "^1.1.3",
2389 | "es-abstract": "^1.17.5"
2390 | }
2391 | },
2392 | "strip-ansi": {
2393 | "version": "5.2.0",
2394 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz",
2395 | "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==",
2396 | "dev": true,
2397 | "requires": {
2398 | "ansi-regex": "^4.1.0"
2399 | }
2400 | },
2401 | "strip-json-comments": {
2402 | "version": "2.0.1",
2403 | "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-2.0.1.tgz",
2404 | "integrity": "sha1-PFMZQukIwml8DsNEhYwobHygpgo=",
2405 | "dev": true
2406 | },
2407 | "subscriptions-transport-ws": {
2408 | "version": "0.9.17",
2409 | "resolved": "https://registry.npmjs.org/subscriptions-transport-ws/-/subscriptions-transport-ws-0.9.17.tgz",
2410 | "integrity": "sha512-hNHi2N80PBz4T0V0QhnnsMGvG3XDFDS9mS6BhZ3R12T6EBywC8d/uJscsga0cVO4DKtXCkCRrWm2sOYrbOdhEA==",
2411 | "requires": {
2412 | "backo2": "^1.0.2",
2413 | "eventemitter3": "^3.1.0",
2414 | "iterall": "^1.2.1",
2415 | "symbol-observable": "^1.0.4",
2416 | "ws": "^5.2.0"
2417 | },
2418 | "dependencies": {
2419 | "ws": {
2420 | "version": "5.2.2",
2421 | "resolved": "https://registry.npmjs.org/ws/-/ws-5.2.2.tgz",
2422 | "integrity": "sha512-jaHFD6PFv6UgoIVda6qZllptQsMlDEJkTQcybzzXDYM1XO9Y8em691FGMPmM46WGyLU4z9KMgQN+qrux/nhlHA==",
2423 | "requires": {
2424 | "async-limiter": "~1.0.0"
2425 | }
2426 | }
2427 | }
2428 | },
2429 | "supports-color": {
2430 | "version": "5.5.0",
2431 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz",
2432 | "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==",
2433 | "dev": true,
2434 | "requires": {
2435 | "has-flag": "^3.0.0"
2436 | }
2437 | },
2438 | "symbol-observable": {
2439 | "version": "1.2.0",
2440 | "resolved": "https://registry.npmjs.org/symbol-observable/-/symbol-observable-1.2.0.tgz",
2441 | "integrity": "sha512-e900nM8RRtGhlV36KGEU9k65K3mPb1WV70OdjfxlG2EAuM1noi/E/BaW/uMhL7bPEssK8QV57vN3esixjUvcXQ=="
2442 | },
2443 | "term-size": {
2444 | "version": "2.2.0",
2445 | "resolved": "https://registry.npmjs.org/term-size/-/term-size-2.2.0.tgz",
2446 | "integrity": "sha512-a6sumDlzyHVJWb8+YofY4TW112G6p2FCPEAFk+59gIYHv3XHRhm9ltVQ9kli4hNWeQBwSpe8cRN25x0ROunMOw==",
2447 | "dev": true
2448 | },
2449 | "to-readable-stream": {
2450 | "version": "1.0.0",
2451 | "resolved": "https://registry.npmjs.org/to-readable-stream/-/to-readable-stream-1.0.0.tgz",
2452 | "integrity": "sha512-Iq25XBt6zD5npPhlLVXGFN3/gyR2/qODcKNNyTMd4vbm39HUaOiAM4PMq0eMVC/Tkxz+Zjdsc55g9yyz+Yq00Q==",
2453 | "dev": true
2454 | },
2455 | "to-regex-range": {
2456 | "version": "5.0.1",
2457 | "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz",
2458 | "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==",
2459 | "dev": true,
2460 | "requires": {
2461 | "is-number": "^7.0.0"
2462 | }
2463 | },
2464 | "toidentifier": {
2465 | "version": "1.0.0",
2466 | "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.0.tgz",
2467 | "integrity": "sha512-yaOH/Pk/VEhBWWTlhI+qXxDFXlejDGcQipMlyxda9nthulaxLZUNcUqFxokp0vcYnvteJln5FNQDRrxj3YcbVw=="
2468 | },
2469 | "toposort-class": {
2470 | "version": "1.0.1",
2471 | "resolved": "https://registry.npmjs.org/toposort-class/-/toposort-class-1.0.1.tgz",
2472 | "integrity": "sha1-f/0feMi+KMO6Rc1OGj9e4ZO9mYg="
2473 | },
2474 | "touch": {
2475 | "version": "3.1.0",
2476 | "resolved": "https://registry.npmjs.org/touch/-/touch-3.1.0.tgz",
2477 | "integrity": "sha512-WBx8Uy5TLtOSRtIq+M03/sKDrXCLHxwDcquSP2c43Le03/9serjQBIztjRz6FkJez9D/hleyAXTBGLwwZUw9lA==",
2478 | "dev": true,
2479 | "requires": {
2480 | "nopt": "~1.0.10"
2481 | }
2482 | },
2483 | "ts-invariant": {
2484 | "version": "0.4.4",
2485 | "resolved": "https://registry.npmjs.org/ts-invariant/-/ts-invariant-0.4.4.tgz",
2486 | "integrity": "sha512-uEtWkFM/sdZvRNNDL3Ehu4WVpwaulhwQszV8mrtcdeE8nN00BV9mAmQ88RkrBhFgl9gMgvjJLAQcZbnPXI9mlA==",
2487 | "requires": {
2488 | "tslib": "^1.9.3"
2489 | }
2490 | },
2491 | "tslib": {
2492 | "version": "1.13.0",
2493 | "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.13.0.tgz",
2494 | "integrity": "sha512-i/6DQjL8Xf3be4K/E6Wgpekn5Qasl1usyw++dAA35Ue5orEn65VIxOA+YvNNl9HV3qv70T7CNwjODHZrLwvd1Q=="
2495 | },
2496 | "type-fest": {
2497 | "version": "0.8.1",
2498 | "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.8.1.tgz",
2499 | "integrity": "sha512-4dbzIzqvjtgiM5rw1k5rEHtBANKmdudhGyBEajN01fEyhaAIhsoKNy6y7+IN93IfpFtwY9iqi7kD+xwKhQsNJA==",
2500 | "dev": true
2501 | },
2502 | "type-is": {
2503 | "version": "1.6.18",
2504 | "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz",
2505 | "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==",
2506 | "requires": {
2507 | "media-typer": "0.3.0",
2508 | "mime-types": "~2.1.24"
2509 | }
2510 | },
2511 | "typedarray-to-buffer": {
2512 | "version": "3.1.5",
2513 | "resolved": "https://registry.npmjs.org/typedarray-to-buffer/-/typedarray-to-buffer-3.1.5.tgz",
2514 | "integrity": "sha512-zdu8XMNEDepKKR+XYOXAVPtWui0ly0NtohUscw+UmaHiAWT8hrV1rr//H6V+0DvJ3OQ19S979M0laLfX8rm82Q==",
2515 | "dev": true,
2516 | "requires": {
2517 | "is-typedarray": "^1.0.0"
2518 | }
2519 | },
2520 | "undefsafe": {
2521 | "version": "2.0.3",
2522 | "resolved": "https://registry.npmjs.org/undefsafe/-/undefsafe-2.0.3.tgz",
2523 | "integrity": "sha512-nrXZwwXrD/T/JXeygJqdCO6NZZ1L66HrxM/Z7mIq2oPanoN0F1nLx3lwJMu6AwJY69hdixaFQOuoYsMjE5/C2A==",
2524 | "dev": true,
2525 | "requires": {
2526 | "debug": "^2.2.0"
2527 | }
2528 | },
2529 | "unique-string": {
2530 | "version": "2.0.0",
2531 | "resolved": "https://registry.npmjs.org/unique-string/-/unique-string-2.0.0.tgz",
2532 | "integrity": "sha512-uNaeirEPvpZWSgzwsPGtU2zVSTrn/8L5q/IexZmH0eH6SA73CmAA5U4GwORTxQAZs95TAXLNqeLoPPNO5gZfWg==",
2533 | "dev": true,
2534 | "requires": {
2535 | "crypto-random-string": "^2.0.0"
2536 | }
2537 | },
2538 | "unpipe": {
2539 | "version": "1.0.0",
2540 | "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz",
2541 | "integrity": "sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw="
2542 | },
2543 | "update-notifier": {
2544 | "version": "4.1.0",
2545 | "resolved": "https://registry.npmjs.org/update-notifier/-/update-notifier-4.1.0.tgz",
2546 | "integrity": "sha512-w3doE1qtI0/ZmgeoDoARmI5fjDoT93IfKgEGqm26dGUOh8oNpaSTsGNdYRN/SjOuo10jcJGwkEL3mroKzktkew==",
2547 | "dev": true,
2548 | "requires": {
2549 | "boxen": "^4.2.0",
2550 | "chalk": "^3.0.0",
2551 | "configstore": "^5.0.1",
2552 | "has-yarn": "^2.1.0",
2553 | "import-lazy": "^2.1.0",
2554 | "is-ci": "^2.0.0",
2555 | "is-installed-globally": "^0.3.1",
2556 | "is-npm": "^4.0.0",
2557 | "is-yarn-global": "^0.3.0",
2558 | "latest-version": "^5.0.0",
2559 | "pupa": "^2.0.1",
2560 | "semver-diff": "^3.1.1",
2561 | "xdg-basedir": "^4.0.0"
2562 | }
2563 | },
2564 | "url-parse-lax": {
2565 | "version": "3.0.0",
2566 | "resolved": "https://registry.npmjs.org/url-parse-lax/-/url-parse-lax-3.0.0.tgz",
2567 | "integrity": "sha1-FrXK/Afb42dsGxmZF3gj1lA6yww=",
2568 | "dev": true,
2569 | "requires": {
2570 | "prepend-http": "^2.0.0"
2571 | }
2572 | },
2573 | "util.promisify": {
2574 | "version": "1.0.1",
2575 | "resolved": "https://registry.npmjs.org/util.promisify/-/util.promisify-1.0.1.tgz",
2576 | "integrity": "sha512-g9JpC/3He3bm38zsLupWryXHoEcS22YHthuPQSJdMy6KNrzIRzWqcsHzD/WUnqe45whVou4VIsPew37DoXWNrA==",
2577 | "requires": {
2578 | "define-properties": "^1.1.3",
2579 | "es-abstract": "^1.17.2",
2580 | "has-symbols": "^1.0.1",
2581 | "object.getownpropertydescriptors": "^2.1.0"
2582 | }
2583 | },
2584 | "utils-merge": {
2585 | "version": "1.0.1",
2586 | "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz",
2587 | "integrity": "sha1-n5VxD1CiZ5R7LMwSR0HBAoQn5xM="
2588 | },
2589 | "uuid": {
2590 | "version": "8.3.0",
2591 | "resolved": "https://registry.npmjs.org/uuid/-/uuid-8.3.0.tgz",
2592 | "integrity": "sha512-fX6Z5o4m6XsXBdli9g7DtWgAx+osMsRRZFKma1mIUsLCz6vRvv+pz5VNbyu9UEDzpMWulZfvpgb/cmDXVulYFQ=="
2593 | },
2594 | "validator": {
2595 | "version": "10.11.0",
2596 | "resolved": "https://registry.npmjs.org/validator/-/validator-10.11.0.tgz",
2597 | "integrity": "sha512-X/p3UZerAIsbBfN/IwahhYaBbY68EN/UQBWHtsbXGT5bfrH/p4NQzUCG1kF/rtKaNpnJ7jAu6NGTdSNtyNIXMw=="
2598 | },
2599 | "vary": {
2600 | "version": "1.1.2",
2601 | "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz",
2602 | "integrity": "sha1-IpnwLG3tMNSllhsLn3RSShj2NPw="
2603 | },
2604 | "widest-line": {
2605 | "version": "3.1.0",
2606 | "resolved": "https://registry.npmjs.org/widest-line/-/widest-line-3.1.0.tgz",
2607 | "integrity": "sha512-NsmoXalsWVDMGupxZ5R08ka9flZjjiLvHVAWYOKtiKM8ujtZWr9cRffak+uSE48+Ob8ObalXpwyeUiyDD6QFgg==",
2608 | "dev": true,
2609 | "requires": {
2610 | "string-width": "^4.0.0"
2611 | }
2612 | },
2613 | "wkx": {
2614 | "version": "0.5.0",
2615 | "resolved": "https://registry.npmjs.org/wkx/-/wkx-0.5.0.tgz",
2616 | "integrity": "sha512-Xng/d4Ichh8uN4l0FToV/258EjMGU9MGcA0HV2d9B/ZpZB3lqQm7nkOdZdm5GhKtLLhAE7PiVQwN4eN+2YJJUg==",
2617 | "requires": {
2618 | "@types/node": "*"
2619 | }
2620 | },
2621 | "wrappy": {
2622 | "version": "1.0.2",
2623 | "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz",
2624 | "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=",
2625 | "dev": true
2626 | },
2627 | "write-file-atomic": {
2628 | "version": "3.0.3",
2629 | "resolved": "https://registry.npmjs.org/write-file-atomic/-/write-file-atomic-3.0.3.tgz",
2630 | "integrity": "sha512-AvHcyZ5JnSfq3ioSyjrBkH9yW4m7Ayk8/9My/DD9onKeu/94fwrMocemO2QAJFAlnnDN+ZDS+ZjAR5ua1/PV/Q==",
2631 | "dev": true,
2632 | "requires": {
2633 | "imurmurhash": "^0.1.4",
2634 | "is-typedarray": "^1.0.0",
2635 | "signal-exit": "^3.0.2",
2636 | "typedarray-to-buffer": "^3.1.5"
2637 | }
2638 | },
2639 | "ws": {
2640 | "version": "6.2.1",
2641 | "resolved": "https://registry.npmjs.org/ws/-/ws-6.2.1.tgz",
2642 | "integrity": "sha512-GIyAXC2cB7LjvpgMt9EKS2ldqr0MTrORaleiOno6TweZ6r3TKtoFQWay/2PceJ3RuBasOHzXNn5Lrw1X0bEjqA==",
2643 | "requires": {
2644 | "async-limiter": "~1.0.0"
2645 | }
2646 | },
2647 | "xdg-basedir": {
2648 | "version": "4.0.0",
2649 | "resolved": "https://registry.npmjs.org/xdg-basedir/-/xdg-basedir-4.0.0.tgz",
2650 | "integrity": "sha512-PSNhEJDejZYV7h50BohL09Er9VaIefr2LMAf3OEmpCkjOi34eYyQYAXUTjEQtZJTKcF0E2UKTh+osDLsgNim9Q==",
2651 | "dev": true
2652 | },
2653 | "xss": {
2654 | "version": "1.0.8",
2655 | "resolved": "https://registry.npmjs.org/xss/-/xss-1.0.8.tgz",
2656 | "integrity": "sha512-3MgPdaXV8rfQ/pNn16Eio6VXYPTkqwa0vc7GkiymmY/DqR1SE/7VPAAVZz1GJsJFrllMYO3RHfEaiUGjab6TNw==",
2657 | "requires": {
2658 | "commander": "^2.20.3",
2659 | "cssfilter": "0.0.10"
2660 | }
2661 | },
2662 | "yallist": {
2663 | "version": "3.1.1",
2664 | "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz",
2665 | "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g=="
2666 | },
2667 | "zen-observable": {
2668 | "version": "0.8.15",
2669 | "resolved": "https://registry.npmjs.org/zen-observable/-/zen-observable-0.8.15.tgz",
2670 | "integrity": "sha512-PQ2PC7R9rslx84ndNBZB/Dkv8V8fZEpk83RLgXtYd0fwUgEjseMn1Dgajh2x6S8QbZAFa9p2qVCEuYZNgve0dQ=="
2671 | },
2672 | "zen-observable-ts": {
2673 | "version": "0.8.21",
2674 | "resolved": "https://registry.npmjs.org/zen-observable-ts/-/zen-observable-ts-0.8.21.tgz",
2675 | "integrity": "sha512-Yj3yXweRc8LdRMrCC8nIc4kkjWecPAUVh0TI0OUrWXx6aX790vLcDlWca6I4vsyCGH3LpWxq0dJRcMOFoVqmeg==",
2676 | "requires": {
2677 | "tslib": "^1.9.3",
2678 | "zen-observable": "^0.8.0"
2679 | }
2680 | }
2681 | }
2682 | }
2683 |
--------------------------------------------------------------------------------
/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "chat",
3 | "version": "1.0.0",
4 | "description": "",
5 | "main": "server.js",
6 | "dependencies": {
7 | "apollo-server": "^2.16.1",
8 | "bcryptjs": "^2.4.3",
9 | "dotenv": "^8.2.0",
10 | "graphql": "^15.3.0",
11 | "jsonwebtoken": "^8.5.1",
12 | "mysql2": "^2.1.0",
13 | "sequelize": "^6.3.4"
14 | },
15 | "devDependencies": {
16 | "nodemon": "^2.0.4"
17 | },
18 | "scripts": {
19 | "dev": "nodemon server.js --ignore client/",
20 | "start": "node server.js"
21 | },
22 | "keywords": [],
23 | "author": "",
24 | "license": "ISC"
25 | }
26 |
--------------------------------------------------------------------------------
/seeders/20200827161337-create-users.js:
--------------------------------------------------------------------------------
1 | 'use strict'
2 | const bcrypt = require('bcryptjs')
3 |
4 | module.exports = {
5 | up: async (queryInterface, Sequelize) => {
6 | const password = await bcrypt.hash('123456', 6)
7 | const createdAt = new Date()
8 | const updatedAt = createdAt
9 |
10 | // https://unsplash.com/photos/ZHvM3XIOHoE
11 | // https://unsplash.com/photos/b1Hg7QI-zcc
12 | // https://unsplash.com/photos/RiDxDgHg7pw
13 |
14 | await queryInterface.bulkInsert('users', [
15 | {
16 | username: 'john',
17 | email: 'john@email.com',
18 | password: password,
19 | imageUrl:
20 | 'https://images.unsplash.com/photo-1535713875002-d1d0cf377fde?ixlib=rb-1.2.1&ixid=eyJhcHBfaWQiOjEyMDd9&auto=format&fit=crop&w=1700&q=80',
21 | createdAt,
22 | updatedAt,
23 | },
24 | {
25 | username: 'jane',
26 | email: 'jane@email.com',
27 | password: password,
28 | imageUrl:
29 | 'https://images.unsplash.com/photo-1520813792240-56fc4a3765a7?ixlib=rb-1.2.1&ixid=eyJhcHBfaWQiOjEyMDd9&auto=format&fit=crop&w=2190&q=80',
30 | createdAt,
31 | updatedAt,
32 | },
33 | {
34 | username: 'boss',
35 | email: 'boss@email.com',
36 | password: password,
37 | imageUrl:
38 | 'https://images.unsplash.com/photo-1566753323558-f4e0952af115?ixlib=rb-1.2.1&ixid=eyJhcHBfaWQiOjEyMDd9&auto=format&fit=crop&w=2122&q=80',
39 | createdAt,
40 | updatedAt,
41 | },
42 | ])
43 | },
44 |
45 | down: async (queryInterface, Sequelize) => {
46 | await queryInterface.bulkDelete('users', null, {})
47 | },
48 | }
49 |
--------------------------------------------------------------------------------
/seeders/20200827161417-create-messages.js:
--------------------------------------------------------------------------------
1 | 'use strict'
2 |
3 | module.exports = {
4 | up: (queryInterface, Sequelize) => {
5 | return queryInterface.bulkInsert('messages', [
6 | {
7 | uuid: '7648485a-6657-48d7-87d6-6a98931d3598',
8 | content: 'Hey Jane!',
9 | from: 'john',
10 | to: 'jane',
11 | createdAt: '2020-07-01 07:00:00',
12 | updatedAt: '2020-07-01 07:00:00',
13 | },
14 | {
15 | uuid: 'ae4df4f1-a428-400d-bb16-edd4237e0c47',
16 | content: "Hey John, how's it going?",
17 | from: 'jane',
18 | to: 'john',
19 | createdAt: '2020-07-01 08:00:00',
20 | updatedAt: '2020-07-01 08:00:00',
21 | },
22 | {
23 | uuid: '0a7c92ac-f69c-4799-8aad-9663a4afb47d',
24 | content: 'Not too bad, just getting to work, you?',
25 | from: 'john',
26 | to: 'jane',
27 | createdAt: '2020-07-01 09:00:00',
28 | updatedAt: '2020-07-01 09:00:00',
29 | },
30 | {
31 | uuid: '240dd560-5825-4d5d-b089-12a67e8ec84c',
32 | content: "I'm working from home now",
33 | from: 'jane',
34 | to: 'john',
35 | createdAt: '2020-07-01 10:00:00',
36 | updatedAt: '2020-07-01 10:00:00',
37 | },
38 | {
39 | uuid: '60909592-cfd7-4b16-a1ce-709091d5f6d7',
40 | content: "That's cool! I'm joining the 'remote club' too",
41 | from: 'john',
42 | to: 'jane',
43 | createdAt: '2020-07-01 11:00:00',
44 | updatedAt: '2020-07-01 11:00:00',
45 | },
46 | {
47 | uuid: 'a10ad37d-c70b-4093-ae33-e5d0ab9498e1',
48 | content: 'Really? how come?',
49 | from: 'jane',
50 | to: 'john',
51 | createdAt: '2020-07-01 12:00:00',
52 | updatedAt: '2020-07-01 12:00:00',
53 | },
54 | {
55 | uuid: 'be49ab98-5271-4eb9-a630-dd6d37e420ed',
56 | content: 'got promoted to a consultancy role',
57 | from: 'john',
58 | to: 'jane',
59 | createdAt: '2020-07-01 13:00:00',
60 | updatedAt: '2020-07-01 13:00:00',
61 | },
62 | {
63 | uuid: 'a10ad37d-c70b-4093-ae33-e5d0ab9429e4',
64 | content: "That's amazing!! well done!",
65 | from: 'jane',
66 | to: 'john',
67 | createdAt: '2020-07-01 14:00:00',
68 | updatedAt: '2020-07-01 14:00:00',
69 | },
70 | {
71 | uuid: 'be49ab98-5271-4eb9-a630-dd6d37e623j7',
72 | content: 'Thanks ;)',
73 | from: 'john',
74 | to: 'jane',
75 | createdAt: '2020-07-01 15:00:00',
76 | updatedAt: '2020-07-01 15:00:00',
77 | },
78 | {
79 | uuid: 'fd4cee68-5caf-4b1b-80a9-5b9add7fd863',
80 | content: 'Hey John, are you done with that task?',
81 | from: 'boss',
82 | to: 'john',
83 | createdAt: '2020-07-01 11:00:00',
84 | updatedAt: '2020-07-01 11:00:00',
85 | },
86 | ])
87 | },
88 |
89 | down: (queryInterface, Sequelize) => {
90 | return queryInterface.bulkDelete('messages', null, {})
91 | },
92 | }
93 |
--------------------------------------------------------------------------------
/server.js:
--------------------------------------------------------------------------------
1 | const { ApolloServer } = require('apollo-server')
2 |
3 | require('dotenv').config()
4 |
5 | const { sequelize } = require('./models')
6 |
7 | const resolvers = require('./graphql/resolvers')
8 | const typeDefs = require('./graphql/typeDefs')
9 | const contextMiddleware = require('./util/contextMiddleware')
10 |
11 | const server = new ApolloServer({
12 | typeDefs,
13 | resolvers,
14 | context: contextMiddleware,
15 | subscriptions: { path: '/' },
16 | })
17 |
18 | server.listen().then(({ url, subscriptionsUrl }) => {
19 | console.log(`🚀 Server ready at ${url}`)
20 | console.log(`🚀 Susbscription ready at ${subscriptionsUrl}`)
21 |
22 | sequelize
23 | .authenticate()
24 | .then(() => console.log('Database connected!!'))
25 | .catch((err) => console.log(err))
26 | })
27 |
--------------------------------------------------------------------------------
/util/contextMiddleware.js:
--------------------------------------------------------------------------------
1 | const jwt = require('jsonwebtoken')
2 | const { PubSub } = require('apollo-server')
3 |
4 | const pubsub = new PubSub()
5 |
6 | module.exports = (context) => {
7 | let token
8 | if (context.req && context.req.headers.authorization) {
9 | token = context.req.headers.authorization.split('Bearer ')[1]
10 | } else if (context.connection && context.connection.context.Authorization) {
11 | token = context.connection.context.Authorization.split('Bearer ')[1]
12 | }
13 |
14 | if (token) {
15 | jwt.verify(token, process.env.JWT_SECRET, (err, decodedToken) => {
16 | context.user = decodedToken
17 | })
18 | }
19 |
20 | context.pubsub = pubsub
21 |
22 | return context
23 | }
24 |
--------------------------------------------------------------------------------