├── client
├── public
│ ├── robots.txt
│ ├── favicon.ico
│ ├── logo192.png
│ ├── logo512.png
│ ├── manifest.json
│ └── index.html
├── README.md
├── nginx
│ └── default.conf
├── src
│ ├── setupTests.js
│ ├── App.test.js
│ ├── index.css
│ ├── index.js
│ ├── App.css
│ ├── App.js
│ └── serviceWorker.js
├── Dockerfile
├── .gitignore
├── Dockerfile.dev
└── package.json
├── aws-pipeline-iac
├── infra-modules
│ └── cicd
│ │ ├── lambda
│ │ ├── output.tf
│ │ ├── function_src
│ │ │ ├── index.zip
│ │ │ └── index.js
│ │ ├── vars.tf
│ │ ├── lambda.tf
│ │ └── role.tf
│ │ ├── cloudwatch
│ │ ├── output.tf
│ │ ├── role.tf
│ │ ├── vars.tf
│ │ └── cloudwatch.tf
│ │ ├── codebuild
│ │ ├── output.tf
│ │ ├── vars.tf
│ │ ├── codebuild.tf
│ │ └── role.tf
│ │ ├── provider.tf
│ │ ├── codepipeline
│ │ ├── output.tf
│ │ ├── s3.tf
│ │ ├── vars.tf
│ │ ├── codepipeline.tf
│ │ └── role.tf
│ │ ├── elasticbeanstalk
│ │ ├── output.tf
│ │ ├── elasticbeanstalk.tf
│ │ ├── vars.tf
│ │ └── role.tf
│ │ ├── variables.tf
│ │ └── main.tf
├── terragrunt.hcl
├── .gitignore
└── infra-live
│ └── prod
│ └── terragrunt.hcl
├── multi-container-tech-arch.png
├── backend
├── README.md
├── src
│ ├── index.js
│ ├── models
│ │ └── ToDo.js
│ ├── test
│ │ └── index.js
│ ├── config
│ │ └── db.js
│ ├── app.js
│ ├── routes
│ │ └── routes.js
│ └── controllers
│ │ └── todo_controller.js
├── .gitignore
├── Dockerfile
├── Dockerfile.dev
├── package.json
└── package-lock.json
├── .gitignore
├── db
└── init-mongo.js
├── nginx
├── configs
│ └── default.conf
└── Dockerfile
├── README.md
├── Dockerrun.aws.json
├── LICENSE
├── docker-compose.yml
└── postman
└── MutliContainerToDoApp.postman_collection.json
/client/public/robots.txt:
--------------------------------------------------------------------------------
1 | # https://www.robotstxt.org/robotstxt.html
2 | User-agent: *
3 | Disallow:
4 |
--------------------------------------------------------------------------------
/aws-pipeline-iac/infra-modules/cicd/lambda/output.tf:
--------------------------------------------------------------------------------
1 | output arn {
2 | value = aws_lambda_function.main.arn
3 | }
4 |
--------------------------------------------------------------------------------
/aws-pipeline-iac/infra-modules/cicd/cloudwatch/output.tf:
--------------------------------------------------------------------------------
1 | output arn {
2 | value = aws_cloudwatch_event_rule.main.arn
3 | }
4 |
--------------------------------------------------------------------------------
/aws-pipeline-iac/infra-modules/cicd/codebuild/output.tf:
--------------------------------------------------------------------------------
1 | output "project_name" {
2 | value = aws_codebuild_project.main.id
3 | }
4 |
--------------------------------------------------------------------------------
/client/public/favicon.ico:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/LukeMwila/multi-container-nginx-react-node-mongo/HEAD/client/public/favicon.ico
--------------------------------------------------------------------------------
/client/public/logo192.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/LukeMwila/multi-container-nginx-react-node-mongo/HEAD/client/public/logo192.png
--------------------------------------------------------------------------------
/client/public/logo512.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/LukeMwila/multi-container-nginx-react-node-mongo/HEAD/client/public/logo512.png
--------------------------------------------------------------------------------
/multi-container-tech-arch.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/LukeMwila/multi-container-nginx-react-node-mongo/HEAD/multi-container-tech-arch.png
--------------------------------------------------------------------------------
/aws-pipeline-iac/infra-modules/cicd/provider.tf:
--------------------------------------------------------------------------------
1 | provider "aws" {
2 | version = 2.57
3 | profile = var.profile
4 | region = var.region
5 | }
--------------------------------------------------------------------------------
/backend/README.md:
--------------------------------------------------------------------------------
1 | # Backend
2 |
3 | ## Run tests in Docker
4 | docker build -t node-test . -f Dockerfile.dev
5 | docker run -e CI=true -p 3001:3001 node-test
--------------------------------------------------------------------------------
/client/README.md:
--------------------------------------------------------------------------------
1 | # Client
2 |
3 | ## Run tests in Docker
4 | docker build -t react-test . -f Dockerfile.dev
5 | docker run -e CI=true -p 3000:3000 react-test
--------------------------------------------------------------------------------
/.gitignore:
--------------------------------------------------------------------------------
1 | # Environment variables
2 | /env
3 |
4 | # Mongo db volume data for persistence storage
5 | /db/mongo-volume/*
6 |
7 | # Idea IDE configs
8 | .idea
--------------------------------------------------------------------------------
/aws-pipeline-iac/infra-modules/cicd/codepipeline/output.tf:
--------------------------------------------------------------------------------
1 | output arn {
2 | value = aws_codepipeline.main.arn
3 | }
4 |
5 | output name {
6 | value = aws_codepipeline.main.name
7 | }
8 |
9 |
--------------------------------------------------------------------------------
/aws-pipeline-iac/infra-modules/cicd/lambda/function_src/index.zip:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/LukeMwila/multi-container-nginx-react-node-mongo/HEAD/aws-pipeline-iac/infra-modules/cicd/lambda/function_src/index.zip
--------------------------------------------------------------------------------
/client/nginx/default.conf:
--------------------------------------------------------------------------------
1 | server {
2 | listen 3000;
3 |
4 | location / {
5 | root /usr/share/nginx/html;
6 | index index.html index.htm;
7 | try_files $uri $uri/ /index.html;
8 | }
9 | }
10 |
--------------------------------------------------------------------------------
/backend/src/index.js:
--------------------------------------------------------------------------------
1 | const http = require('http');
2 | const app = require('./app');
3 |
4 | const port = process.env.PORT || 3001;
5 | const server = http.createServer(app);
6 | server.listen(port, () => console.log(`Server running port ${port}`));
--------------------------------------------------------------------------------
/backend/.gitignore:
--------------------------------------------------------------------------------
1 | # dependencies
2 | /node_modules
3 |
4 | # misc
5 | .DS_Store
6 | .env
7 | .env.local
8 | .env.development.local
9 | .env.test.local
10 | .env.production.local
11 |
12 | npm-debug.log*
13 | yarn-debug.log*
14 | yarn-error.log*
15 |
--------------------------------------------------------------------------------
/db/init-mongo.js:
--------------------------------------------------------------------------------
1 | db.createUser(
2 | {
3 | user: "lukemwila",
4 | pwd: "doesitreallymatterwhatthisis",
5 | roles: [
6 | {
7 | role: "readWrite",
8 | db: "multicontainer-database"
9 | }
10 | ]
11 | }
12 | )
--------------------------------------------------------------------------------
/aws-pipeline-iac/infra-modules/cicd/codepipeline/s3.tf:
--------------------------------------------------------------------------------
1 | resource "aws_s3_bucket" "main" {
2 | force_destroy = true
3 | acl = "private"
4 | bucket = "${var.bucket_name}-${var.environment}"
5 | region = var.region
6 | versioning {
7 | enabled = true
8 | }
9 | }
--------------------------------------------------------------------------------
/backend/Dockerfile:
--------------------------------------------------------------------------------
1 | # Specify a base image
2 | FROM node:12.13.0-alpine AS alpine
3 |
4 | WORKDIR /app
5 |
6 | # Install dependencies
7 | COPY package.json .
8 | RUN npm install
9 | COPY . .
10 |
11 | EXPOSE 3001
12 |
13 | # Default command
14 | CMD ["npm", "run", "start"]
15 |
--------------------------------------------------------------------------------
/backend/Dockerfile.dev:
--------------------------------------------------------------------------------
1 | # Specify a base image
2 | FROM node:12.13.0-alpine AS alpine
3 |
4 | WORKDIR /app
5 |
6 | # Install dependencies
7 | COPY package.json .
8 | RUN npm install
9 | COPY . .
10 |
11 | EXPOSE 3001
12 |
13 | # Default command
14 | CMD ["npm", "run", "test"]
15 |
--------------------------------------------------------------------------------
/client/src/setupTests.js:
--------------------------------------------------------------------------------
1 | // jest-dom adds custom jest matchers for asserting on DOM nodes.
2 | // allows you to do things like:
3 | // expect(element).toHaveTextContent(/react/i)
4 | // learn more: https://github.com/testing-library/jest-dom
5 | import '@testing-library/jest-dom/extend-expect';
6 |
--------------------------------------------------------------------------------
/client/src/App.test.js:
--------------------------------------------------------------------------------
1 | import React from 'react';
2 | import ReactDOM from "react-dom";
3 | import App from './App';
4 |
5 | test('renders without crashing', () => {
6 | const div = document.createElement("div");
7 | ReactDOM.render(, div);
8 | ReactDOM.unmountComponentAtNode(div);
9 | });
10 |
--------------------------------------------------------------------------------
/client/Dockerfile:
--------------------------------------------------------------------------------
1 | FROM node:12.13.0-alpine as build
2 | WORKDIR /app
3 | COPY package*.json ./
4 | RUN npm install
5 | COPY . .
6 | RUN npm run build
7 |
8 | FROM nginx
9 | EXPOSE 3000
10 | COPY ./nginx/default.conf /etc/nginx/conf.d/default.conf
11 | COPY --from=build /app/build /usr/share/nginx/html
12 |
--------------------------------------------------------------------------------
/backend/src/models/ToDo.js:
--------------------------------------------------------------------------------
1 | const { Schema, model } = require('mongoose');
2 |
3 | const ToDoSchema = new Schema({
4 | item_name: {
5 | type: String,
6 | required: true
7 | },
8 | complete: {
9 | type: Boolean,
10 | default: false
11 | }
12 | });
13 |
14 | const ToDo = model('toDo', ToDoSchema);
15 | module.exports = ToDo;
--------------------------------------------------------------------------------
/nginx/configs/default.conf:
--------------------------------------------------------------------------------
1 | upstream backend {
2 | server backend:3001;
3 | }
4 |
5 | upstream client {
6 | server client:3000;
7 | }
8 |
9 | server {
10 | listen 80;
11 |
12 | location /backend {
13 | rewrite /backend/(.*) /$1 break;
14 | proxy_pass http://backend;
15 | }
16 |
17 | location / {
18 | proxy_pass http://client;
19 | }
20 | }
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | # Build & Deploy Multi-Container Applications to AWS
2 | Multi container application with Nginx, React, Node and Mongo DB. This repository also contains Terraform IaC (Infrastructure as Code) for a CICD pipeline to build and push images to DockerHub.
3 |
4 | ## Technical Architecture Diagram
5 | 
--------------------------------------------------------------------------------
/nginx/Dockerfile:
--------------------------------------------------------------------------------
1 | # Pull in the from the official nginx image.
2 | FROM nginx
3 |
4 | EXPOSE 80
5 |
6 | # Delete the default welcome to nginx page.
7 | RUN rm /usr/share/nginx/html/*
8 |
9 | # Copy over the custom default configs.
10 | COPY configs/default.conf /etc/nginx/conf.d/default.conf
11 |
12 | # Start nginx in the foreground to play nicely with Docker.
13 | CMD ["nginx", "-g", "daemon off;"]
14 |
--------------------------------------------------------------------------------
/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 |
--------------------------------------------------------------------------------
/aws-pipeline-iac/terragrunt.hcl:
--------------------------------------------------------------------------------
1 | remote_state {
2 | backend = "s3"
3 | generate = {
4 | path = "backend.tf"
5 | if_exists = "overwrite_terragrunt"
6 | }
7 | config = {
8 | bucket = "multicontainer-todo-app-terraform-state"
9 |
10 | key = "${path_relative_to_include()}/terraform.tfstate"
11 | region = "eu-west-1"
12 | encrypt = true
13 | dynamodb_table = "multicontainer-todo-app-lock-table"
14 | }
15 | }
--------------------------------------------------------------------------------
/client/src/index.css:
--------------------------------------------------------------------------------
1 | body {
2 | margin: 0;
3 | font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', 'Roboto', 'Oxygen',
4 | 'Ubuntu', 'Cantarell', 'Fira Sans', 'Droid Sans', 'Helvetica Neue',
5 | sans-serif;
6 | -webkit-font-smoothing: antialiased;
7 | -moz-osx-font-smoothing: grayscale;
8 | }
9 |
10 | code {
11 | font-family: source-code-pro, Menlo, Monaco, Consolas, 'Courier New',
12 | monospace;
13 | }
14 |
--------------------------------------------------------------------------------
/aws-pipeline-iac/infra-modules/cicd/elasticbeanstalk/output.tf:
--------------------------------------------------------------------------------
1 | output application_arn {
2 | value = aws_elastic_beanstalk_application.main.arn
3 | }
4 |
5 | output application_name {
6 | value = aws_elastic_beanstalk_application.main.name
7 | }
8 |
9 | output environment_arn {
10 | value = aws_elastic_beanstalk_environment.main.arn
11 | }
12 |
13 | output environment_name {
14 | value = aws_elastic_beanstalk_environment.main.name
15 | }
--------------------------------------------------------------------------------
/aws-pipeline-iac/infra-modules/cicd/elasticbeanstalk/elasticbeanstalk.tf:
--------------------------------------------------------------------------------
1 | resource "aws_elastic_beanstalk_application" "main" {
2 | name = var.name
3 | description = var.description
4 | }
5 |
6 | resource "aws_elastic_beanstalk_environment" "main" {
7 | name = "eb-${var.environment}"
8 | application = "${aws_elastic_beanstalk_application.main.name}"
9 | solution_stack_name = var.solution_stack_name
10 | tier = var.tier
11 | }
--------------------------------------------------------------------------------
/client/Dockerfile.dev:
--------------------------------------------------------------------------------
1 | # Specify a base image
2 | FROM node:12.13.0-alpine AS alpine
3 |
4 | WORKDIR /app
5 |
6 | # A wildcard is used to ensure both package.json AND package-lock.json are copied
7 | COPY package.json .
8 |
9 | ## install only the packages defined in the package-lock.json (faster than the normal npm install)
10 | RUN npm install
11 | # Copy the contents of the project to the image
12 | COPY . .
13 | # Run 'npm start' when the container starts.
14 | CMD ["npm", "run", "test"]
--------------------------------------------------------------------------------
/aws-pipeline-iac/infra-modules/cicd/codebuild/vars.tf:
--------------------------------------------------------------------------------
1 | variable "name" {
2 | description = "The name of the Build"
3 | type = string
4 | }
5 |
6 | variable "environment" {
7 | type = string
8 | }
9 |
10 | variable "docker_id" {
11 | type = string
12 | }
13 |
14 | variable "docker_pw" {
15 | type = string
16 | }
17 |
18 | variable "image" {
19 | description = "CodeBuild Container base image"
20 | type = string
21 | }
22 |
23 | variable "snyk_auth_token" {
24 | description = "Snyk authentication token"
25 | type = string
26 | }
--------------------------------------------------------------------------------
/backend/src/test/index.js:
--------------------------------------------------------------------------------
1 | const { expect } = require('chai');
2 | const { agent } = require('supertest');
3 | const app = require('../app');
4 |
5 | const request = agent;
6 |
7 | describe('Some controller', () => {
8 | it('Get request to /test returns some text', async () => {
9 | const res = await request(app).get('/test');
10 | const textResponse = res.body;
11 | expect(res.status).to.equal(200);
12 | expect(textResponse.text).to.be.a('string');
13 | expect(textResponse.text).to.equal('Simple Node App Working!');
14 | });
15 | });
--------------------------------------------------------------------------------
/client/src/index.js:
--------------------------------------------------------------------------------
1 | import React from 'react';
2 | import ReactDOM from 'react-dom';
3 | import './index.css';
4 | import App from './App';
5 | import * as serviceWorker from './serviceWorker';
6 |
7 | ReactDOM.render(
8 |
9 |
10 | ,
11 | document.getElementById('root')
12 | );
13 |
14 | // If you want your app to work offline and load faster, you can change
15 | // unregister() to register() below. Note this comes with some pitfalls.
16 | // Learn more about service workers: https://bit.ly/CRA-PWA
17 | serviceWorker.unregister();
18 |
--------------------------------------------------------------------------------
/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 |
--------------------------------------------------------------------------------
/backend/src/config/db.js:
--------------------------------------------------------------------------------
1 | const mongoose = require('mongoose');
2 |
3 | const db = process.env.TO_DO_DB;
4 |
5 | const connectDB = async () => {
6 | let attempts = 10;
7 | while (attempts) {
8 | try {
9 | await mongoose.connect(db);
10 | console.log('MongoDB connected...');
11 | // break out of loop once conncected
12 | break;
13 | } catch (err) {
14 | console.log("Error: ", err.message);
15 | attempts -= 1;
16 | console.log(`connection attempts left: ${attempts}`);
17 | // wait for 10 seconds before retrying
18 | await new Promise(res => setTimeout(res, 10000));
19 | }
20 | }
21 | };
22 |
23 | module.exports = { connectDB };
--------------------------------------------------------------------------------
/aws-pipeline-iac/infra-modules/cicd/elasticbeanstalk/vars.tf:
--------------------------------------------------------------------------------
1 | variable "name" {
2 | description = "Name of the eb application"
3 | type = string
4 | }
5 |
6 | variable "description" {
7 | description = "Description of the eb application"
8 | type = string
9 | }
10 |
11 | variable "solution_stack_name" {
12 | description = "The name of thhe solution stack for the platform"
13 | type = string
14 | }
15 |
16 | variable "environment" {
17 | type = string
18 | }
19 |
20 | variable "tier" {
21 | description = "Elastic Beanstalk Environment tier. Valid values are Worker or WebServer. If tier is left blank WebServer will be used."
22 | default = "WebServer"
23 | type = string
24 | }
25 |
--------------------------------------------------------------------------------
/backend/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "backend",
3 | "version": "1.0.0",
4 | "description": "",
5 | "main": "index.js",
6 | "scripts": {
7 | "start": "node src/index.js",
8 | "dev": "nodemon src/index.js",
9 | "test": "export NODE_ENV=test && mocha 'src/test/**/*.js'"
10 | },
11 | "keywords": [],
12 | "author": "Luke Mwila",
13 | "license": "ISC",
14 | "dependencies": {
15 | "body-parser": "^1.19.0",
16 | "cors": "^2.8.5",
17 | "dotenv": "^8.2.0",
18 | "express": "^4.17.1",
19 | "mongoose": "^5.9.14"
20 | },
21 | "devDependencies": {
22 | "chai": "^4.2.0",
23 | "mocha": "^7.1.2",
24 | "nodemon": "^2.0.4",
25 | "supertest": "^4.0.2"
26 | }
27 | }
28 |
--------------------------------------------------------------------------------
/aws-pipeline-iac/infra-modules/cicd/variables.tf:
--------------------------------------------------------------------------------
1 | variable "module" {
2 | description = "The terraform module used to deploy"
3 | type = string
4 | }
5 |
6 | variable "profile" {
7 | description = "AWS profile"
8 | type = string
9 | }
10 |
11 | variable "region" {
12 | description = "aws region to deploy to"
13 | type = string
14 | }
15 |
16 | variable "github_secret_name" {
17 | type = string
18 | }
19 |
20 | variable "docker_secret_name" {
21 | type = string
22 | }
23 |
24 | variable "snyk_secret_name" {
25 | description = "Snyk secret name"
26 | type = string
27 | }
28 |
29 | variable "environment" {
30 | type = string
31 | }
32 |
33 | variable "branch_name" {
34 | type = string
35 | }
36 |
--------------------------------------------------------------------------------
/client/src/App.css:
--------------------------------------------------------------------------------
1 | .App {
2 | text-align: center;
3 | }
4 |
5 | .App-logo {
6 | height: 40vmin;
7 | pointer-events: none;
8 | }
9 |
10 | @media (prefers-reduced-motion: no-preference) {
11 | .App-logo {
12 | animation: App-logo-spin infinite 20s linear;
13 | }
14 | }
15 |
16 | .App-header {
17 | background-color: #282c34;
18 | min-height: 100vh;
19 | display: flex;
20 | flex-direction: column;
21 | align-items: center;
22 | justify-content: center;
23 | font-size: calc(10px + 2vmin);
24 | color: white;
25 | }
26 |
27 | .App-link {
28 | color: #61dafb;
29 | }
30 |
31 | @keyframes App-logo-spin {
32 | from {
33 | transform: rotate(0deg);
34 | }
35 | to {
36 | transform: rotate(360deg);
37 | }
38 | }
39 |
--------------------------------------------------------------------------------
/aws-pipeline-iac/.gitignore:
--------------------------------------------------------------------------------
1 | .DS_Store
2 | .idea
3 | *.iml
4 | **/.terragrunt-cache/*
5 |
6 | # Local .terraform directories
7 | **/.terraform/*
8 |
9 | # .tfstate files
10 | *.tfstate
11 | *.tfstate.*
12 | .idea/*
13 |
14 | # .tfvars files
15 | sensitive.tfvars
16 |
17 | # Crash log files
18 | crash.log
19 |
20 | # Ignore override files as they are usually used to override resources locally and so
21 | # are not checked in
22 | override.tf
23 | override.tf.json
24 | *_override.tf
25 | *_override.tf.json
26 |
27 | # Include override files you do wish to add to version control using negated pattern
28 | #
29 | # !example_override.tf
30 |
31 | # Include tfplan files to ignore the plan output of command: terraform plan -out=tfplan
32 | # example: *tfplan*
33 |
--------------------------------------------------------------------------------
/backend/src/app.js:
--------------------------------------------------------------------------------
1 | require('dotenv').config();
2 |
3 | // Express App Setup
4 | const express = require('express');
5 | const bodyParser = require('body-parser');
6 | const routes = require('./routes/routes');
7 | const app = express();
8 | const { connectDB } = require('./config/db');
9 |
10 | // Connect Database
11 | if (process.env.NODE_ENV !== 'test') {
12 | connectDB();
13 | }
14 |
15 | app.use(bodyParser.urlencoded({ extended: false }));
16 | app.use(bodyParser.json());
17 |
18 | // for testing purposes
19 | app.get('/test', (req, res) => {
20 | res.status(200).send({ text: 'Simple Node App Working!' });
21 | });
22 |
23 | routes(app);
24 |
25 | app.use((err, req, res, next) => {
26 | res.status(422).send({ error: err.message });
27 | });
28 |
29 | module.exports = app;
--------------------------------------------------------------------------------
/aws-pipeline-iac/infra-modules/cicd/lambda/vars.tf:
--------------------------------------------------------------------------------
1 | variable function_name {
2 | description = "The function name"
3 | type = string
4 | }
5 |
6 | variable handler {
7 | description = "The handler name"
8 | type = string
9 | default = "index.handler"
10 | }
11 |
12 | variable runtime {
13 | description = "The runtime environment for the lambda function"
14 | type = string
15 | default = "nodejs12.x"
16 | }
17 |
18 | variable source_arn {
19 | description = "The ARN of the resource that triggers the lambda function"
20 | type = string
21 | }
22 |
23 | variable lambda_role {
24 | description = "The name of the iam role"
25 | type = string
26 | }
27 |
28 | variable lambda_policy {
29 | description = "The name of the policy"
30 | type = string
31 | }
32 |
33 | variable "environment" {
34 | type = string
35 | }
--------------------------------------------------------------------------------
/aws-pipeline-iac/infra-modules/cicd/lambda/lambda.tf:
--------------------------------------------------------------------------------
1 | data "archive_file" "lambda_zip" {
2 | type = "zip"
3 | source_file = "lambda/function_src/index.js"
4 | output_path = "lambda/function_src/index.zip"
5 | }
6 | resource "aws_lambda_function" "main" {
7 | filename = data.archive_file.lambda_zip.output_path
8 | function_name = "${var.function_name}-${var.environment}"
9 | role = aws_iam_role.iam_for_lambda.arn
10 | handler = var.handler
11 | source_code_hash = data.archive_file.lambda_zip.output_base64sha256
12 | runtime = var.runtime
13 | }
14 |
15 | resource "aws_lambda_permission" "cloudwatch_trigger_lambda" {
16 | statement_id = "cloudwatch-codepipeline-trigger-lambda"
17 | action = "lambda:InvokeFunction"
18 | function_name = aws_lambda_function.main.function_name
19 | principal = "events.amazonaws.com"
20 | source_arn = var.source_arn
21 | }
--------------------------------------------------------------------------------
/client/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "client",
3 | "version": "0.1.0",
4 | "private": true,
5 | "dependencies": {
6 | "@testing-library/jest-dom": "^4.2.4",
7 | "@testing-library/react": "^9.5.0",
8 | "@testing-library/user-event": "^7.2.1",
9 | "react": "^16.13.1",
10 | "react-dom": "^16.13.1",
11 | "react-scripts": "3.4.1"
12 | },
13 | "scripts": {
14 | "start": "react-scripts start",
15 | "build": "react-scripts build",
16 | "test": "react-scripts test",
17 | "eject": "react-scripts eject"
18 | },
19 | "eslintConfig": {
20 | "extends": "react-app"
21 | },
22 | "browserslist": {
23 | "production": [
24 | ">0.2%",
25 | "not dead",
26 | "not op_mini all"
27 | ],
28 | "development": [
29 | "last 1 chrome version",
30 | "last 1 firefox version",
31 | "last 1 safari version"
32 | ]
33 | }
34 | }
35 |
--------------------------------------------------------------------------------
/backend/src/routes/routes.js:
--------------------------------------------------------------------------------
1 | const ToDoController = require('../controllers/todo_controller');
2 |
3 | const routes = app => {
4 | // @route GET to-do
5 | // @desc Get all to do item
6 | // @access Private
7 | app.get('/v1/to-do', ToDoController.get);
8 | // @route GET to-do
9 | // @desc Get single o do item
10 | // @access Private
11 | app.get('/v1/to-do/:id', ToDoController.getById);
12 | // @route POST to-do
13 | // @desc Create to do item
14 | // @access Private
15 | app.post('/v1/to-do', ToDoController.create);
16 | // @route PUT to-do
17 | // @desc Edit to do item
18 | // @access Private
19 | app.put('/v1/to-do/:id', ToDoController.edit);
20 | // @route DELETE to-do
21 | // @desc Delete to do item
22 | // @access Private
23 | app.delete('/v1/to-do/:id', ToDoController.deleteToDo);
24 | };
25 |
26 | module.exports = routes;
27 |
--------------------------------------------------------------------------------
/aws-pipeline-iac/infra-live/prod/terragrunt.hcl:
--------------------------------------------------------------------------------
1 | inputs = {
2 | environment = "prod"
3 | branch_name = "master"
4 | }
5 |
6 | include {
7 | # The find_in_parent_folders() helper will
8 | # automatically search up the directory tree to find the root terragrunt.hcl and inherit
9 | # the remote_state configuration from it.
10 | path = find_in_parent_folders()
11 | }
12 |
13 | terraform {
14 | source = "../../infra-modules/cicd"
15 |
16 | extra_arguments "conditional_vars" {
17 | # built-in function to automatically get the list of
18 | # all commands that accept -var-file and -var arguments
19 | commands = get_terraform_commands_that_need_vars()
20 |
21 | arguments = [
22 | "-lock-timeout=10m",
23 | "-var", "module=${path_relative_to_include()}"
24 | ]
25 |
26 | required_var_files = [
27 | "${get_parent_terragrunt_dir()}/sensitive.tfvars"
28 | ]
29 | }
30 | }
31 |
--------------------------------------------------------------------------------
/Dockerrun.aws.json:
--------------------------------------------------------------------------------
1 | {
2 | "AWSEBDockerrunVersion": 2,
3 | "containerDefinitions": [
4 | {
5 | "name": "nginx",
6 | "image": "lukondefmwila/dkr-multicontainer-app-nginx",
7 | "hostname": "nginx",
8 | "essential": true,
9 | "portMappings": [
10 | {
11 | "hostPort": 80,
12 | "containerPort": 80
13 | }
14 | ],
15 | "links": ["client", "backend"],
16 | "memory": 100
17 | },
18 | {
19 | "name": "client",
20 | "image": "lukondefmwila/dkr-multicontainer-app-client",
21 | "hostname": "client",
22 | "essential": false,
23 | "memory": 300
24 | },
25 | {
26 | "name": "backend",
27 | "image": "lukondefmwila/dkr-multicontainer-app-backend",
28 | "hostname": "backend",
29 | "essential": false,
30 | "memory": 300
31 | }
32 | ]
33 | }
--------------------------------------------------------------------------------
/client/src/App.js:
--------------------------------------------------------------------------------
1 | import React from 'react';
2 | import './App.css';
3 |
4 | function App() {
5 | const [toDoItems, updateToDoItems] = React.useState([]);
6 |
7 | React.useEffect(() => {
8 | const getToDoItems = async () => {
9 | const response = await fetch(
10 | `/backend/v1/to-do`
11 | );
12 |
13 | const items = await response.json();
14 | if (items && Array.isArray(items) && items.length) {
15 | updateToDoItems(items);
16 | }
17 | };
18 | getToDoItems();
19 | }, []);
20 |
21 | return (
22 |
23 | {toDoItems && toDoItems.length
24 | ? toDoItems.map((item, i) => {
25 | return (
26 |
27 | {`${item.item_name}`}
28 |
29 |
30 | );
31 | })
32 | : 'No items to be done'}
33 |
34 | );
35 | }
36 |
37 | export default App;
38 |
--------------------------------------------------------------------------------
/aws-pipeline-iac/infra-modules/cicd/cloudwatch/role.tf:
--------------------------------------------------------------------------------
1 | resource "aws_iam_role" "main" {
2 | name = var.role_name
3 |
4 | assume_role_policy = <,
41 | "state":
42 | }
43 | DOC
44 | input_paths = {
45 | pipeline: "$.detail.pipeline",
46 | state: "$.detail.state"
47 | }
48 | }
49 | }
--------------------------------------------------------------------------------
/aws-pipeline-iac/infra-modules/cicd/lambda/role.tf:
--------------------------------------------------------------------------------
1 | resource "aws_iam_role" "iam_for_lambda" {
2 | name = var.lambda_role
3 |
4 | assume_role_policy = < {
4 | const toDoProps = req.body;
5 | try {
6 | const toDo = await ToDo.create(toDoProps);
7 | res.status(201).send(toDo);
8 | } catch (e) {
9 | next();
10 | }
11 | };
12 |
13 | const get = async (req, res, next) => {
14 | try {
15 | const toDo = await ToDo.find();
16 | res.status(200).send(toDo);
17 | } catch (e) {
18 | next();
19 | }
20 | };
21 |
22 | const getById = async (req, res, next) => {
23 | const toDoId = req.params.id;
24 | try {
25 | const toDo = await ToDo.findById({ _id: toDoId });
26 | res.status(200).send(toDo);
27 | } catch (e) {
28 | next();
29 | }
30 | };
31 |
32 | const edit = async (req, res, next) => {
33 | const toDoId = req.params.id;
34 | const toDoProps = req.body;
35 | try {
36 | await ToDo.findByIdAndUpdate({ _id: toDoId }, toDoProps);
37 | const toDo = await ToDo.findById({ _id: toDoId });
38 | res.status(200).send(toDo);
39 | } catch (e) {
40 | next();
41 | }
42 | };
43 |
44 | const deleteToDo = async (req, res, next) => {
45 | const toDoId = req.params.id;
46 | try {
47 | const toDo = await ToDo.findByIdAndRemove({ _id: toDoId });
48 | res.status(204).send(toDo);
49 | } catch (e) {
50 | next();
51 | }
52 | };
53 |
54 | const ToDoController = {
55 | create,
56 | get,
57 | getById,
58 | edit,
59 | deleteToDo
60 | };
61 |
62 | module.exports = ToDoController;
--------------------------------------------------------------------------------
/aws-pipeline-iac/infra-modules/cicd/codepipeline/codepipeline.tf:
--------------------------------------------------------------------------------
1 | resource "aws_codepipeline" "main" {
2 | name = "${var.name}-${var.environment}"
3 | role_arn = aws_iam_role.main.arn
4 |
5 | artifact_store {
6 | location = "${aws_s3_bucket.main.bucket}"
7 | type = "S3"
8 | }
9 |
10 | stage {
11 | name = "Source"
12 | action {
13 | name = "Source"
14 | category = "Source"
15 | owner = "ThirdParty"
16 | provider = "GitHub"
17 | version = "1"
18 | output_artifacts = ["SourceArtifact"]
19 |
20 | configuration = {
21 | Owner = var.github_org
22 | Repo = var.repository_name
23 | PollForSourceChanges = "true"
24 | Branch = var.branch_name
25 | OAuthToken = var.github_token
26 | }
27 | }
28 | }
29 |
30 | stage {
31 | name = "Build"
32 |
33 | action {
34 | name = "Build"
35 | category = "Build"
36 | owner = "AWS"
37 | provider = "CodeBuild"
38 | input_artifacts = ["SourceArtifact"]
39 | output_artifacts = ["BuildArtifact"]
40 | version = "1"
41 |
42 | configuration = {
43 | ProjectName = var.project_name
44 | PrimarySource = "SourceArtifact"
45 | }
46 | run_order = 2
47 | }
48 | }
49 | }
--------------------------------------------------------------------------------
/client/public/index.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
12 |
13 |
17 |
18 |
27 | React App
28 |
29 |
30 |
31 |
32 |
42 |
43 |
44 |
--------------------------------------------------------------------------------
/aws-pipeline-iac/infra-modules/cicd/lambda/function_src/index.js:
--------------------------------------------------------------------------------
1 | const util = require("util");
2 | const https = require('https');
3 |
4 | const STATUS_GIF = {
5 | started: 'https://media.giphy.com/media/tXLpxypfSXvUc/giphy.gif', // rocket launching
6 | succeeded: 'https://media.giphy.com/media/MYDMiSizWs5sjJRHFA/giphy.gif', // micheal jordan celebrating
7 | failed: 'https://media.giphy.com/media/d2lcHJTG5Tscg/giphy.gif', // anthony anderson crying
8 | canceled: 'https://media.giphy.com/media/IzXmRTmKd0if6/giphy.gif', // finger pressing abort button
9 | }
10 |
11 | let apiRequest = (state, pipeline, gif = '') => {
12 | const data = JSON.stringify({
13 | text: `The pipeline ${pipeline} has *${state}*.\n${gif}`
14 | })
15 |
16 | const options = {
17 | hostname: 'hooks.slack.com',
18 | port: 443,
19 | path: 'Slack webhook goes here',
20 | method: 'POST',
21 | headers: {
22 | 'Content-Type': 'application/json',
23 | 'Content-Length': data.length
24 | }
25 | }
26 |
27 | const req = https.request(options, (res) => {
28 | console.log(`statusCode: ${res.statusCode}`)
29 |
30 | res.on('data', (d) => {
31 | process.stdout.write(d)
32 | })
33 | })
34 |
35 | req.on('error', (error) => {
36 | console.error(error)
37 | })
38 |
39 | req.write(data)
40 | req.end()
41 | }
42 |
43 | apiRequest = util.promisify(apiRequest);
44 |
45 | module.exports.handler = async (event, context, callback) => {
46 | try {
47 | const { state, pipeline } = event;
48 | let response;
49 | console.log('event:', event);
50 | switch (state) {
51 | case 'STARTED':
52 | response = await apiRequest(state, pipeline, STATUS_GIF.started)
53 | break;
54 | case 'SUCCEEDED':
55 | response = await apiRequest(state, pipeline, STATUS_GIF.succeeded)
56 | break;
57 | case 'FAILED':
58 | response = await apiRequest(state, pipeline, STATUS_GIF.failed)
59 | break;
60 | case 'CANCELED':
61 | response = await apiRequest(state, pipeline, STATUS_GIF.canceled)
62 | break;
63 | default:
64 | response = await apiRequest(state, pipeline)
65 | break;
66 | }
67 | console.log("response:", response);
68 | }catch(err){
69 | console.log("error:", err)
70 | }
71 | }
--------------------------------------------------------------------------------
/aws-pipeline-iac/infra-modules/cicd/codebuild/role.tf:
--------------------------------------------------------------------------------
1 | resource "aws_iam_role" "main" {
2 | name = "${var.name}-role-${var.environment}"
3 | assume_role_policy = data.aws_iam_policy_document.main.json
4 | }
5 |
6 | # This is a data source which can be used to construct a
7 | # JSON representation of an IAM policy document,
8 | # for use with resources which expect policy documents,
9 | # such as the aws_iam_policy resource.
10 |
11 | data "aws_iam_policy_document" "main" {
12 | statement {
13 | effect = "Allow"
14 | actions = ["sts:AssumeRole"]
15 |
16 | principals {
17 | type = "Service"
18 | identifiers = ["codebuild.amazonaws.com"]
19 | }
20 | }
21 | }
22 |
23 | # Policies for select environment
24 |
25 | resource "aws_iam_policy" "main" {
26 | name = "${var.name}-policy-${var.environment}"
27 | description = "Allow AWS CodeBuild builds for Multicontainer application"
28 | policy = data.aws_iam_policy_document.codebuild_multicontainer_app.json
29 | }
30 |
31 | resource "aws_iam_role_policy_attachment" "mutlicontainer_app" {
32 | role = aws_iam_role.main.name
33 | policy_arn = aws_iam_policy.main.arn
34 | }
35 |
36 | data "aws_iam_policy_document" "codebuild_multicontainer_app" {
37 | statement {
38 | effect = "Allow"
39 |
40 | actions = [
41 | "logs:CreateLogGroup",
42 | "logs:CreateLogStream",
43 | "logs:PutLogEvents"
44 | ]
45 |
46 | resources = ["*"]
47 | }
48 |
49 | statement {
50 | effect = "Allow"
51 |
52 | actions = [
53 | "codebuild:UpdateReportGroup",
54 | "codebuild:ListReportsForReportGroup",
55 | "codebuild:CreateReportGroup",
56 | "codebuild:CreateReport",
57 | "codebuild:UpdateReport",
58 | "codebuild:ListReports",
59 | "codebuild:DeleteReport",
60 | "codebuild:ListReportGroups",
61 | "codebuild:BatchPutTestCases"
62 | ]
63 |
64 | resources = [
65 | "arn:aws:codebuild:eu-west-1:*"
66 | ]
67 | }
68 |
69 | statement {
70 | effect = "Allow"
71 |
72 | actions = [
73 | "s3:CreateBucket",
74 | "s3:PutObject",
75 | "s3:GetObject",
76 | "s3:GetObjectVersion",
77 | "s3:GetBucketAcl",
78 | "s3:GetBucketLocation"
79 | ]
80 |
81 | resources = [
82 | "arn:aws:s3:::*"
83 | ]
84 | }
85 |
86 | statement {
87 | effect = "Allow"
88 |
89 | actions = [
90 | "secretsmanager:GetSecretValue"
91 | ]
92 |
93 | resources = [
94 | "arn:aws:secretsmanager:eu-west-1:*"
95 | ]
96 | }
97 | }
98 |
--------------------------------------------------------------------------------
/postman/MutliContainerToDoApp.postman_collection.json:
--------------------------------------------------------------------------------
1 | {
2 | "info": {
3 | "_postman_id": "47ccf184-0241-412d-9a7b-8063f8ddce1d",
4 | "name": "MutliContainerToDoApp",
5 | "schema": "https://schema.getpostman.com/json/collection/v2.1.0/collection.json"
6 | },
7 | "item": [
8 | {
9 | "name": "Add Item",
10 | "request": {
11 | "method": "POST",
12 | "header": [],
13 | "body": {
14 | "mode": "raw",
15 | "raw": "{\n\t\"item_name\": \"Repeat\"\n}",
16 | "options": {
17 | "raw": {
18 | "language": "json"
19 | }
20 | }
21 | },
22 | "url": {
23 | "raw": "http://localhost:3001/v1/to-do",
24 | "protocol": "http",
25 | "host": [
26 | "localhost"
27 | ],
28 | "port": "3001",
29 | "path": [
30 | "v1",
31 | "to-do"
32 | ]
33 | }
34 | },
35 | "response": []
36 | },
37 | {
38 | "name": "Edit Item",
39 | "request": {
40 | "method": "PUT",
41 | "header": [],
42 | "body": {
43 | "mode": "raw",
44 | "raw": "{\n\t\"item_name\": \"Repeat\",\n\t\"complete\": true\n}",
45 | "options": {
46 | "raw": {
47 | "language": "json"
48 | }
49 | }
50 | },
51 | "url": {
52 | "raw": "http://localhost:3001/v1/to-do/5ec28b3dbdfa9120c488300a",
53 | "protocol": "http",
54 | "host": [
55 | "localhost"
56 | ],
57 | "port": "3001",
58 | "path": [
59 | "v1",
60 | "to-do",
61 | "5ec28b3dbdfa9120c488300a"
62 | ]
63 | }
64 | },
65 | "response": []
66 | },
67 | {
68 | "name": "Get Item",
69 | "request": {
70 | "method": "GET",
71 | "header": [],
72 | "url": {
73 | "raw": "http://localhost:3001/v1/to-do/5ec28b3dbdfa9120c488300a",
74 | "protocol": "http",
75 | "host": [
76 | "localhost"
77 | ],
78 | "port": "3001",
79 | "path": [
80 | "v1",
81 | "to-do",
82 | "5ec28b3dbdfa9120c488300a"
83 | ]
84 | }
85 | },
86 | "response": []
87 | },
88 | {
89 | "name": "Get All Items",
90 | "request": {
91 | "method": "GET",
92 | "header": [],
93 | "url": {
94 | "raw": "http://localhost:3001/v1/to-do",
95 | "protocol": "http",
96 | "host": [
97 | "localhost"
98 | ],
99 | "port": "3001",
100 | "path": [
101 | "v1",
102 | "to-do"
103 | ]
104 | }
105 | },
106 | "response": []
107 | },
108 | {
109 | "name": "Delete Item",
110 | "request": {
111 | "method": "DELETE",
112 | "header": [],
113 | "url": {
114 | "raw": "http://localhost:3001/v1/to-do/5ec28bd5bdfa9120c488300b",
115 | "protocol": "http",
116 | "host": [
117 | "localhost"
118 | ],
119 | "port": "3001",
120 | "path": [
121 | "v1",
122 | "to-do",
123 | "5ec28bd5bdfa9120c488300b"
124 | ]
125 | }
126 | },
127 | "response": []
128 | }
129 | ],
130 | "protocolProfileBehavior": {}
131 | }
--------------------------------------------------------------------------------
/aws-pipeline-iac/infra-modules/cicd/main.tf:
--------------------------------------------------------------------------------
1 | # GitHub secrets
2 | data "aws_secretsmanager_secret" "github_secret" {
3 | name = var.github_secret_name
4 | }
5 |
6 | data "aws_secretsmanager_secret_version" "github_token" {
7 | secret_id = data.aws_secretsmanager_secret.github_secret.id
8 | }
9 |
10 | # Docker secrets
11 | data "aws_secretsmanager_secret" "docker_secret" {
12 | name = var.docker_secret_name
13 | }
14 |
15 | data "aws_secretsmanager_secret_version" "docker_creds" {
16 | secret_id = data.aws_secretsmanager_secret.docker_secret.id
17 | }
18 |
19 | # Snyk secrets
20 | data "aws_secretsmanager_secret" "snyk_secret" {
21 | name = var.snyk_secret_name
22 | }
23 |
24 | data "aws_secretsmanager_secret_version" "snyk_auth" {
25 | secret_id = data.aws_secretsmanager_secret.snyk_secret.id
26 | }
27 |
28 | # Elastic Beanstalk solution stack
29 | data "aws_elastic_beanstalk_solution_stack" "multi_docker" {
30 | most_recent = true
31 |
32 | name_regex = "^64bit Amazon Linux (.*) Multi-container Docker (.*)$"
33 | }
34 |
35 | # Codebuild module for CI
36 | module "codebuild_for_multicontainer_app" {
37 | source = "./codebuild"
38 | name = "codebuild-multicontainer-docker-app"
39 | image = "aws/codebuild/standard:4.0"
40 | docker_id = jsondecode(data.aws_secretsmanager_secret_version.docker_creds.secret_string)["DOCKER_ID"]
41 | docker_pw = jsondecode(data.aws_secretsmanager_secret_version.docker_creds.secret_string)["DOCKER_PW"]
42 | snyk_auth_token = jsondecode(data.aws_secretsmanager_secret_version.snyk_auth.secret_string)["SNYK_AUTH_TOKEN"]
43 | environment = var.environment
44 | }
45 |
46 | # CodePipeline module for CICD pipeline
47 | module "codepipeline_for_multicontainer_app" {
48 | source = "./codepipeline"
49 | name = "codepipeline-multicontainer-docker-app"
50 | bucket_name = "codepipeline-multicontainer-docker-app-artifact"
51 | github_org = "LukeMwila"
52 | repository_name = "multi-container-nginx-react-node-mongo"
53 | branch_name = var.branch_name
54 | environment = var.environment
55 | region = var.region
56 | #elasticbeanstalk_application_name = module.elasticbeanstalk_for_multicontainer_app.application_name
57 | #elasticbeanstalk_application_environment_name = module.elasticbeanstalk_for_multicontainer_app.environment_name
58 | project_name = module.codebuild_for_multicontainer_app.project_name
59 | github_token = jsondecode(data.aws_secretsmanager_secret_version.github_token.secret_string)["GitHubPersonalAccessToken"]
60 | }
61 |
62 | # Elastic Beanstalk module for multicontainer app
63 | #module "elasticbeanstalk_for_multicontainer_app" {
64 | # source = "./elasticbeanstalk"
65 | # name = "multicontainer-nginx-react-node-mongo-app"
66 | # description = "Docker multicontainer application consisting of an Nginx web server, React client and a NodeJS backend."
67 | # solution_stack_name = data.aws_elastic_beanstalk_solution_stack.multi_docker.name
68 | # tier = "WebServer"
69 | # environment = var.environment
70 | #}
71 |
72 | # Cloudwatch event module for pipeline state changes
73 | module "cloudwatch_for_pipeline_notifications" {
74 | source = "./cloudwatch"
75 | name = "multicontainer-pipeline-state-change"
76 | description = "event for multicontainer app pipeline state change"
77 | role_name = "cloudwatch-for-multicontainer-pipeline-role"
78 | policy_name = "cloudwatch-for-multicontainer-pipeline-policy"
79 | targetId = "SendToLambda"
80 | codepipeline_arn = module.codepipeline_for_multicontainer_app.arn
81 | codepipeline_name = module.codepipeline_for_multicontainer_app.name
82 | resource_arn = module.lambda_for_pipeline_notifications.arn
83 | environment = var.environment
84 | }
85 |
86 | # Lambda module for pushing pipeline state change notifications to Slack
87 | module "lambda_for_pipeline_notifications" {
88 | source = "./lambda"
89 | function_name = "lambda-push-pipeline-notification-to-slack"
90 | source_arn = module.cloudwatch_for_pipeline_notifications.arn
91 | lambda_role = "lambda-for-multicontainer-pipeline-role"
92 | lambda_policy = "lambda-for-multicontainer-pipeline-policy"
93 | environment = var.environment
94 | }
--------------------------------------------------------------------------------
/aws-pipeline-iac/infra-modules/cicd/codepipeline/role.tf:
--------------------------------------------------------------------------------
1 | # IAM role
2 | resource "aws_iam_role" "main" {
3 | name = "${var.name}-role-${var.environment}"
4 | assume_role_policy = data.aws_iam_policy_document.main.json
5 | }
6 |
7 | # This is a data source which can be used to construct a
8 | # JSON representation of an IAM policy document,
9 | # for use with resources which expect policy documents,
10 | # such as the aws_iam_policy resource.
11 |
12 | data "aws_iam_policy_document" "main" {
13 | statement {
14 | effect = "Allow"
15 | actions = ["sts:AssumeRole"]
16 |
17 | principals {
18 | type = "Service"
19 | identifiers = ["codepipeline.amazonaws.com"]
20 | }
21 | }
22 | }
23 |
24 | # Policies for different environments
25 |
26 | resource "aws_iam_policy" "main" {
27 | name = "${var.name}-policy-${var.environment}"
28 | description = "Allow AWS Codepipeline deployments for Multicontainer application"
29 | policy = data.aws_iam_policy_document.codepipeline_mutlicontainer_app.json
30 | }
31 |
32 | resource "aws_iam_role_policy_attachment" "mutlicontainer_app" {
33 | role = aws_iam_role.main.name
34 | policy_arn = aws_iam_policy.main.arn
35 | }
36 |
37 | data "aws_iam_policy_document" "codepipeline_mutlicontainer_app" {
38 | statement {
39 | effect = "Allow"
40 |
41 | actions = [
42 | "s3:*",
43 | ]
44 |
45 | resources = [
46 | "arn:aws:s3:::*"
47 | ]
48 | }
49 |
50 | statement {
51 | effect = "Allow"
52 |
53 | actions = [
54 | "codebuild:StartBuild",
55 | "codebuild:StopBuild",
56 | "codebuild:BatchGetBuilds",
57 | "codebuild:UpdateReportGroup",
58 | "codebuild:ListReportsForReportGroup",
59 | "codebuild:CreateReportGroup",
60 | "codebuild:CreateReport",
61 | "codebuild:UpdateReport",
62 | "codebuild:ListReports",
63 | "codebuild:DeleteReport",
64 | "codebuild:ListReportGroups",
65 | "codebuild:BatchPutTestCases"
66 | ]
67 |
68 | resources = ["*"]
69 | }
70 |
71 | statement {
72 | effect = "Allow"
73 |
74 | actions = [
75 | "cloudformation:CreateStack",
76 | "cloudformation:UpdateStacks",
77 | "cloudformation:DeleteStacks",
78 | "cloudformation:DescribeStacks",
79 | "cloudformation:DescribeStackEvents",
80 | "cloudformation:DescribeStackResource",
81 | "cloudformation:DescribeStackResources",
82 | "cloudformation:GetTemplate",
83 | "cloudformation:ValidateTemplate",
84 | "cloudformation:UpdateStack",
85 | "cloudformation:CancelUpdateStack"
86 | ]
87 |
88 | resources = ["*"]
89 | }
90 |
91 | statement {
92 | effect = "Allow"
93 |
94 | actions = [
95 | "logs:CreateLogGroup",
96 | "logs:CreateLogStream",
97 | "logs:PutLogEvents"
98 | ]
99 |
100 | resources = ["*"]
101 | }
102 |
103 | statement {
104 | effect = "Allow"
105 |
106 | actions = [
107 | "elasticbeanstalk:*",
108 | "ec2:*",
109 | "elasticloadbalancing:*",
110 | "autoscaling:*",
111 | "cloudwatch:*",
112 | "ecs:*"
113 | ]
114 |
115 | resources = ["*"]
116 | }
117 |
118 | statement {
119 | effect = "Allow"
120 |
121 | actions = [
122 | "devicefarm:ListProjects",
123 | "devicefarm:ListDevicePools",
124 | "devicefarm:GetRun",
125 | "devicefarm:GetUpload",
126 | "devicefarm:CreateUpload",
127 | "devicefarm:ScheduleRun"
128 | ]
129 |
130 | resources = ["*"]
131 | }
132 |
133 | statement {
134 | effect = "Allow"
135 |
136 | actions = [
137 | "opsworks:CreateDeployment",
138 | "opsworks:DescribeApps",
139 | "opsworks:DescribeCommands",
140 | "opsworks:DescribeDeployments",
141 | "opsworks:DescribeInstances",
142 | "opsworks:DescribeStacks",
143 | "opsworks:UpdateApp",
144 | "opsworks:UpdateStack"
145 | ]
146 |
147 | resources = ["*"]
148 | }
149 |
150 | statement {
151 | effect = "Allow"
152 |
153 | actions = [
154 | "servicecatalog:ListProvisioningArtifacts",
155 | "servicecatalog:CreateProvisioningArtifact",
156 | "servicecatalog:DescribeProvisioningArtifact",
157 | "servicecatalog:DeleteProvisioningArtifact",
158 | "servicecatalog:UpdateProduct"
159 | ]
160 |
161 | resources = ["*"]
162 | }
163 | }
164 |
165 |
166 |
167 |
168 |
169 |
--------------------------------------------------------------------------------
/client/src/serviceWorker.js:
--------------------------------------------------------------------------------
1 | // This optional code is used to register a service worker.
2 | // register() is not called by default.
3 |
4 | // This lets the app load faster on subsequent visits in production, and gives
5 | // it offline capabilities. However, it also means that developers (and users)
6 | // will only see deployed updates on subsequent visits to a page, after all the
7 | // existing tabs open on the page have been closed, since previously cached
8 | // resources are updated in the background.
9 |
10 | // To learn more about the benefits of this model and instructions on how to
11 | // opt-in, read https://bit.ly/CRA-PWA
12 |
13 | const isLocalhost = Boolean(
14 | window.location.hostname === 'localhost' ||
15 | // [::1] is the IPv6 localhost address.
16 | window.location.hostname === '[::1]' ||
17 | // 127.0.0.0/8 are considered localhost for IPv4.
18 | window.location.hostname.match(
19 | /^127(?:\.(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)){3}$/
20 | )
21 | );
22 |
23 | export function register(config) {
24 | if (process.env.NODE_ENV === 'production' && 'serviceWorker' in navigator) {
25 | // The URL constructor is available in all browsers that support SW.
26 | const publicUrl = new URL(process.env.PUBLIC_URL, window.location.href);
27 | if (publicUrl.origin !== window.location.origin) {
28 | // Our service worker won't work if PUBLIC_URL is on a different origin
29 | // from what our page is served on. This might happen if a CDN is used to
30 | // serve assets; see https://github.com/facebook/create-react-app/issues/2374
31 | return;
32 | }
33 |
34 | window.addEventListener('load', () => {
35 | const swUrl = `${process.env.PUBLIC_URL}/service-worker.js`;
36 |
37 | if (isLocalhost) {
38 | // This is running on localhost. Let's check if a service worker still exists or not.
39 | checkValidServiceWorker(swUrl, config);
40 |
41 | // Add some additional logging to localhost, pointing developers to the
42 | // service worker/PWA documentation.
43 | navigator.serviceWorker.ready.then(() => {
44 | console.log(
45 | 'This web app is being served cache-first by a service ' +
46 | 'worker. To learn more, visit https://bit.ly/CRA-PWA'
47 | );
48 | });
49 | } else {
50 | // Is not localhost. Just register service worker
51 | registerValidSW(swUrl, config);
52 | }
53 | });
54 | }
55 | }
56 |
57 | function registerValidSW(swUrl, config) {
58 | navigator.serviceWorker
59 | .register(swUrl)
60 | .then(registration => {
61 | registration.onupdatefound = () => {
62 | const installingWorker = registration.installing;
63 | if (installingWorker == null) {
64 | return;
65 | }
66 | installingWorker.onstatechange = () => {
67 | if (installingWorker.state === 'installed') {
68 | if (navigator.serviceWorker.controller) {
69 | // At this point, the updated precached content has been fetched,
70 | // but the previous service worker will still serve the older
71 | // content until all client tabs are closed.
72 | console.log(
73 | 'New content is available and will be used when all ' +
74 | 'tabs for this page are closed. See https://bit.ly/CRA-PWA.'
75 | );
76 |
77 | // Execute callback
78 | if (config && config.onUpdate) {
79 | config.onUpdate(registration);
80 | }
81 | } else {
82 | // At this point, everything has been precached.
83 | // It's the perfect time to display a
84 | // "Content is cached for offline use." message.
85 | console.log('Content is cached for offline use.');
86 |
87 | // Execute callback
88 | if (config && config.onSuccess) {
89 | config.onSuccess(registration);
90 | }
91 | }
92 | }
93 | };
94 | };
95 | })
96 | .catch(error => {
97 | console.error('Error during service worker registration:', error);
98 | });
99 | }
100 |
101 | function checkValidServiceWorker(swUrl, config) {
102 | // Check if the service worker can be found. If it can't reload the page.
103 | fetch(swUrl, {
104 | headers: { 'Service-Worker': 'script' },
105 | })
106 | .then(response => {
107 | // Ensure service worker exists, and that we really are getting a JS file.
108 | const contentType = response.headers.get('content-type');
109 | if (
110 | response.status === 404 ||
111 | (contentType != null && contentType.indexOf('javascript') === -1)
112 | ) {
113 | // No service worker found. Probably a different app. Reload the page.
114 | navigator.serviceWorker.ready.then(registration => {
115 | registration.unregister().then(() => {
116 | window.location.reload();
117 | });
118 | });
119 | } else {
120 | // Service worker found. Proceed as normal.
121 | registerValidSW(swUrl, config);
122 | }
123 | })
124 | .catch(() => {
125 | console.log(
126 | 'No internet connection found. App is running in offline mode.'
127 | );
128 | });
129 | }
130 |
131 | export function unregister() {
132 | if ('serviceWorker' in navigator) {
133 | navigator.serviceWorker.ready
134 | .then(registration => {
135 | registration.unregister();
136 | })
137 | .catch(error => {
138 | console.error(error.message);
139 | });
140 | }
141 | }
142 |
--------------------------------------------------------------------------------
/aws-pipeline-iac/infra-modules/cicd/elasticbeanstalk/role.tf:
--------------------------------------------------------------------------------
1 | # IAM role
2 | resource "aws_iam_role" "main" {
3 | name = "${var.name}-role-${var.environment}"
4 | assume_role_policy = data.aws_iam_policy_document.main.json
5 | }
6 |
7 | # This is a data source which can be used to construct a
8 | # JSON representation of an IAM policy document,
9 | # for use with resources which expect policy documents,
10 | # such as the aws_iam_policy resource.
11 |
12 | data "aws_iam_policy_document" "main" {
13 | statement {
14 | effect = "Allow"
15 | actions = ["sts:AssumeRole"]
16 |
17 | principals {
18 | type = "Service"
19 | identifiers = ["elasticbeanstalk.amazonaws.com"]
20 | }
21 | }
22 | }
23 |
24 | # Policies for different environments
25 |
26 | resource "aws_iam_policy" "main" {
27 | name = "${var.name}-policy-${var.environment}"
28 | description = "ElasticBeanstalk Policy"
29 | policy = data.aws_iam_policy_document.elasticbeanstalk_mutlicontainer_app.json
30 | }
31 |
32 | resource "aws_iam_role_policy_attachment" "mutlicontainer_app" {
33 | role = aws_iam_role.main.name
34 | policy_arn = aws_iam_policy.main.arn
35 | }
36 |
37 | data "aws_iam_policy_document" "elasticbeanstalk_mutlicontainer_app" {
38 | statement {
39 | effect = "Allow"
40 |
41 | actions = [
42 | "logs:DescribeLogStreams",
43 | "logs:CreateLogStream",
44 | "logs:PutLogEvents"
45 | ]
46 |
47 | resources = ["arn:aws:logs:*:*:log-group:/aws/elasticbeanstalk/*:log-stream:*"]
48 | }
49 |
50 | statement {
51 | effect = "Allow"
52 |
53 | actions = [
54 | "logs: DeleteLogGroup"
55 | ]
56 |
57 | resources = [
58 | "arn:aws:logs:*:*:log-group:/aws/elasticbeanstalk*"
59 | ]
60 | }
61 |
62 | statement {
63 | effect = "Allow"
64 |
65 | actions = [
66 | "elasticloadbalancing:DescribeInstanceHealth",
67 | "elasticloadbalancing:DescribeLoadBalancers",
68 | "elasticloadbalancing:DescribeTargetHealth",
69 | "ec2:DescribeInstances",
70 | "ec2:DescribeInstanceStatus",
71 | "ec2:GetConsoleOutput",
72 | "ec2:AssociateAddress",
73 | "ec2:DescribeAddresses",
74 | "ec2:DescribeSecurityGroups",
75 | "sqs:GetQueueAttributes",
76 | "sqs:GetQueueUrl",
77 | "autoscaling:DescribeAutoScalingGroups",
78 | "autoscaling:DescribeAutoScalingInstances",
79 | "autoscaling:DescribeScalingActivities",
80 | "autoscaling:DescribeNotificationConfigurations",
81 | "sns:Publish"
82 | ]
83 |
84 | resources = ["*"]
85 | }
86 |
87 | statement {
88 | effect = "Allow"
89 |
90 | actions = [
91 | "cloudformation:*"
92 | ]
93 |
94 | resources = [
95 | "arn:aws:cloudformation:*:*:stack/awseb-*",
96 | "arn:aws:cloudformation:*:*:stack/eb-*"
97 | ]
98 | }
99 |
100 | statement {
101 | effect = "Allow"
102 |
103 | actions = [
104 | "s3:*"
105 | ]
106 |
107 | resources = [
108 | "arn:aws:s3:::elasticbeanstalk-*",
109 | "arn:aws:s3:::elasticbeanstalk-*/*"
110 | ]
111 | }
112 |
113 | statement {
114 | effect = "Allow"
115 |
116 | actions = [
117 | "ec2:RunInstances"
118 | ]
119 |
120 | resources = ["arn:aws:ec2:*:*:launch-template/*"]
121 | }
122 |
123 | statement {
124 | effect = "Allow"
125 |
126 | actions = [
127 | "autoscaling:AttachInstances",
128 | "autoscaling:CreateAutoScalingGroup",
129 | "autoscaling:CreateLaunchConfiguration",
130 | "autoscaling:DeleteLaunchConfiguration",
131 | "autoscaling:DeleteAutoScalingGroup",
132 | "autoscaling:DeleteScheduledAction",
133 | "autoscaling:DescribeAccountLimits",
134 | "autoscaling:DescribeAutoScalingGroups",
135 | "autoscaling:DescribeAutoScalingInstances",
136 | "autoscaling:DescribeLaunchConfigurations",
137 | "autoscaling:DescribeLoadBalancers",
138 | "autoscaling:DescribeNotificationConfigurations",
139 | "autoscaling:DescribeScalingActivities",
140 | "autoscaling:DescribeScheduledActions",
141 | "autoscaling:DetachInstances",
142 | "autoscaling:DeletePolicy",
143 | "autoscaling:PutScalingPolicy",
144 | "autoscaling:PutScheduledUpdateGroupAction",
145 | "autoscaling:PutNotificationConfiguration",
146 | "autoscaling:ResumeProcesses",
147 | "autoscaling:SetDesiredCapacity",
148 | "autoscaling:SuspendProcesses",
149 | "autoscaling:TerminateInstanceInAutoScalingGroup",
150 | "autoscaling:UpdateAutoScalingGroup",
151 | "cloudwatch:PutMetricAlarm",
152 | "ec2:AssociateAddress",
153 | "ec2:AllocateAddress",
154 | "ec2:AuthorizeSecurityGroupEgress",
155 | "ec2:AuthorizeSecurityGroupIngress",
156 | "ec2:CreateLaunchTemplate",
157 | "ec2:CreateLaunchTemplateVersion",
158 | "ec2:DescribeLaunchTemplates",
159 | "ec2:DescribeLaunchTemplateVersions",
160 | "ec2:DeleteLaunchTemplate",
161 | "ec2:DeleteLaunchTemplateVersions",
162 | "ec2:CreateSecurityGroup",
163 | "ec2:DeleteSecurityGroup",
164 | "ec2:DescribeAccountAttributes",
165 | "ec2:DescribeAddresses",
166 | "ec2:DescribeImages",
167 | "ec2:DescribeInstances",
168 | "ec2:DescribeKeyPairs",
169 | "ec2:DescribeSecurityGroups",
170 | "ec2:DescribeSnapshots",
171 | "ec2:DescribeSubnets",
172 | "ec2:DescribeVpcs",
173 | "ec2:DescribeInstanceAttribute",
174 | "ec2:DescribeSpotInstanceRequests",
175 | "ec2:DescribeVpcClassicLink",
176 | "ec2:DisassociateAddress",
177 | "ec2:ReleaseAddress",
178 | "ec2:RevokeSecurityGroupEgress",
179 | "ec2:RevokeSecurityGroupIngress",
180 | "ec2:TerminateInstances",
181 | "ecs:CreateCluster",
182 | "ecs:DeleteCluster",
183 | "ecs:DescribeClusters",
184 | "ecs:RegisterTaskDefinition",
185 | "elasticbeanstalk:*",
186 | "elasticloadbalancing:ApplySecurityGroupsToLoadBalancer",
187 | "elasticloadbalancing:ConfigureHealthCheck",
188 | "elasticloadbalancing:CreateLoadBalancer",
189 | "elasticloadbalancing:DeleteLoadBalancer",
190 | "elasticloadbalancing:DeregisterInstancesFromLoadBalancer",
191 | "elasticloadbalancing:DescribeInstanceHealth",
192 | "elasticloadbalancing:DescribeLoadBalancers",
193 | "elasticloadbalancing:DescribeTargetHealth",
194 | "elasticloadbalancing:RegisterInstancesWithLoadBalancer",
195 | "elasticloadbalancing:DescribeTargetGroups",
196 | "elasticloadbalancing:RegisterTargets",
197 | "elasticloadbalancing:DeregisterTargets",
198 | "iam:ListRoles",
199 | "iam:PassRole",
200 | "logs:CreateLogGroup",
201 | "logs:PutRetentionPolicy",
202 | "logs:DescribeLogGroups",
203 | "rds:DescribeDBEngineVersions",
204 | "rds:DescribeDBInstances",
205 | "rds:DescribeOrderableDBInstanceOptions",
206 | "s3:GetObject",
207 | "s3:GetObjectAcl",
208 | "s3:ListBucket",
209 | "sns:CreateTopic",
210 | "sns:GetTopicAttributes",
211 | "sns:ListSubscriptionsByTopic",
212 | "sns:Subscribe",
213 | "sns:SetTopicAttributes",
214 | "sqs:GetQueueAttributes",
215 | "sqs:GetQueueUrl",
216 | "codebuild:CreateProject",
217 | "codebuild:DeleteProject",
218 | "codebuild:BatchGetBuilds",
219 | "codebuild:StartBuild"
220 | ]
221 |
222 | resources = ["*"]
223 | }
224 | }
225 |
226 |
--------------------------------------------------------------------------------
/backend/package-lock.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "backend",
3 | "version": "1.0.0",
4 | "lockfileVersion": 1,
5 | "requires": true,
6 | "dependencies": {
7 | "@sindresorhus/is": {
8 | "version": "0.14.0",
9 | "resolved": "https://registry.npmjs.org/@sindresorhus/is/-/is-0.14.0.tgz",
10 | "integrity": "sha512-9NET910DNaIPngYnLLPeg+Ogzqsi9uM4mSboU5y6p8S5DzMTVEsJZrawi+BoDNUVBa2DhJqQYUFvMDfgU062LQ==",
11 | "dev": true
12 | },
13 | "@szmarczak/http-timer": {
14 | "version": "1.1.2",
15 | "resolved": "https://registry.npmjs.org/@szmarczak/http-timer/-/http-timer-1.1.2.tgz",
16 | "integrity": "sha512-XIB2XbzHTN6ieIjfIMV9hlVcfPU26s2vafYWQcZHWXHOxiaRZYEDKEwdl129Zyg50+foYV2jCgtrqSA6qNuNSA==",
17 | "dev": true,
18 | "requires": {
19 | "defer-to-connect": "^1.0.1"
20 | }
21 | },
22 | "@types/color-name": {
23 | "version": "1.1.1",
24 | "resolved": "https://registry.npmjs.org/@types/color-name/-/color-name-1.1.1.tgz",
25 | "integrity": "sha512-rr+OQyAjxze7GgWrSaJwydHStIhHq2lvY3BOC2Mj7KnzI7XK0Uw1TOOdI9lDoajEbSWLiYgoo4f1R51erQfhPQ==",
26 | "dev": true
27 | },
28 | "abbrev": {
29 | "version": "1.1.1",
30 | "resolved": "https://registry.npmjs.org/abbrev/-/abbrev-1.1.1.tgz",
31 | "integrity": "sha512-nne9/IiQ/hzIhY6pdDnbBtz7DjPTKrY00P/zvPSm5pOFkl6xuGrGnXn/VtTNNfNtAfZ9/1RtehkszU9qcTii0Q==",
32 | "dev": true
33 | },
34 | "accepts": {
35 | "version": "1.3.7",
36 | "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.7.tgz",
37 | "integrity": "sha512-Il80Qs2WjYlJIBNzNkK6KYqlVMTbZLXgHx2oT0pU/fjRHyEp+PEfEPY0R3WCwAGVOtauxh1hOxNgIf5bv7dQpA==",
38 | "requires": {
39 | "mime-types": "~2.1.24",
40 | "negotiator": "0.6.2"
41 | }
42 | },
43 | "ansi-align": {
44 | "version": "3.0.0",
45 | "resolved": "https://registry.npmjs.org/ansi-align/-/ansi-align-3.0.0.tgz",
46 | "integrity": "sha512-ZpClVKqXN3RGBmKibdfWzqCY4lnjEuoNzU5T0oEFpfd/z5qJHVarukridD4juLO2FXMiwUQxr9WqQtaYa8XRYw==",
47 | "dev": true,
48 | "requires": {
49 | "string-width": "^3.0.0"
50 | },
51 | "dependencies": {
52 | "string-width": {
53 | "version": "3.1.0",
54 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz",
55 | "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==",
56 | "dev": true,
57 | "requires": {
58 | "emoji-regex": "^7.0.1",
59 | "is-fullwidth-code-point": "^2.0.0",
60 | "strip-ansi": "^5.1.0"
61 | }
62 | }
63 | }
64 | },
65 | "ansi-colors": {
66 | "version": "3.2.3",
67 | "resolved": "https://registry.npmjs.org/ansi-colors/-/ansi-colors-3.2.3.tgz",
68 | "integrity": "sha512-LEHHyuhlPY3TmuUYMh2oz89lTShfvgbmzaBcxve9t/9Wuy7Dwf4yoAKcND7KFT1HAQfqZ12qtc+DUrBMeKF9nw==",
69 | "dev": true
70 | },
71 | "ansi-regex": {
72 | "version": "4.1.0",
73 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz",
74 | "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==",
75 | "dev": true
76 | },
77 | "ansi-styles": {
78 | "version": "4.2.1",
79 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.2.1.tgz",
80 | "integrity": "sha512-9VGjrMsG1vePxcSweQsN20KY/c4zN0h9fLjqAbwbPfahM3t+NL+M9HC8xeXG2I8pX5NoamTGNuomEUFI7fcUjA==",
81 | "dev": true,
82 | "requires": {
83 | "@types/color-name": "^1.1.1",
84 | "color-convert": "^2.0.1"
85 | }
86 | },
87 | "anymatch": {
88 | "version": "3.1.1",
89 | "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.1.tgz",
90 | "integrity": "sha512-mM8522psRCqzV+6LhomX5wgp25YVibjh8Wj23I5RPkPppSVSjyKD2A2mBJmWGa+KN7f2D6LNh9jkBCeyLktzjg==",
91 | "dev": true,
92 | "requires": {
93 | "normalize-path": "^3.0.0",
94 | "picomatch": "^2.0.4"
95 | }
96 | },
97 | "argparse": {
98 | "version": "1.0.10",
99 | "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.10.tgz",
100 | "integrity": "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==",
101 | "dev": true,
102 | "requires": {
103 | "sprintf-js": "~1.0.2"
104 | }
105 | },
106 | "array-flatten": {
107 | "version": "1.1.1",
108 | "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz",
109 | "integrity": "sha1-ml9pkFGx5wczKPKgCJaLZOopVdI="
110 | },
111 | "assertion-error": {
112 | "version": "1.1.0",
113 | "resolved": "https://registry.npmjs.org/assertion-error/-/assertion-error-1.1.0.tgz",
114 | "integrity": "sha512-jgsaNduz+ndvGyFt3uSuWqvy4lCnIJiovtouQN5JZHOKCS2QuhEdbcQHFhVksz2N2U9hXJo8odG7ETyWlEeuDw==",
115 | "dev": true
116 | },
117 | "asynckit": {
118 | "version": "0.4.0",
119 | "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz",
120 | "integrity": "sha1-x57Zf380y48robyXkLzDZkdLS3k=",
121 | "dev": true
122 | },
123 | "balanced-match": {
124 | "version": "1.0.0",
125 | "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.0.tgz",
126 | "integrity": "sha1-ibTRmasr7kneFk6gK4nORi1xt2c=",
127 | "dev": true
128 | },
129 | "binary-extensions": {
130 | "version": "2.0.0",
131 | "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.0.0.tgz",
132 | "integrity": "sha512-Phlt0plgpIIBOGTT/ehfFnbNlfsDEiqmzE2KRXoX1bLIlir4X/MR+zSyBEkL05ffWgnRSf/DXv+WrUAVr93/ow==",
133 | "dev": true
134 | },
135 | "bl": {
136 | "version": "2.2.0",
137 | "resolved": "https://registry.npmjs.org/bl/-/bl-2.2.0.tgz",
138 | "integrity": "sha512-wbgvOpqopSr7uq6fJrLH8EsvYMJf9gzfo2jCsL2eTy75qXPukA4pCgHamOQkZtY5vmfVtjB+P3LNlMHW5CEZXA==",
139 | "requires": {
140 | "readable-stream": "^2.3.5",
141 | "safe-buffer": "^5.1.1"
142 | }
143 | },
144 | "bluebird": {
145 | "version": "3.5.1",
146 | "resolved": "https://registry.npmjs.org/bluebird/-/bluebird-3.5.1.tgz",
147 | "integrity": "sha512-MKiLiV+I1AA596t9w1sQJ8jkiSr5+ZKi0WKrYGUn6d1Fx+Ij4tIj+m2WMQSGczs5jZVxV339chE8iwk6F64wjA=="
148 | },
149 | "body-parser": {
150 | "version": "1.19.0",
151 | "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.19.0.tgz",
152 | "integrity": "sha512-dhEPs72UPbDnAQJ9ZKMNTP6ptJaionhP5cBb541nXPlW60Jepo9RV/a4fX4XWW9CuFNK22krhrj1+rgzifNCsw==",
153 | "requires": {
154 | "bytes": "3.1.0",
155 | "content-type": "~1.0.4",
156 | "debug": "2.6.9",
157 | "depd": "~1.1.2",
158 | "http-errors": "1.7.2",
159 | "iconv-lite": "0.4.24",
160 | "on-finished": "~2.3.0",
161 | "qs": "6.7.0",
162 | "raw-body": "2.4.0",
163 | "type-is": "~1.6.17"
164 | }
165 | },
166 | "boxen": {
167 | "version": "4.2.0",
168 | "resolved": "https://registry.npmjs.org/boxen/-/boxen-4.2.0.tgz",
169 | "integrity": "sha512-eB4uT9RGzg2odpER62bBwSLvUeGC+WbRjjyyFhGsKnc8wp/m0+hQsMUvUe3H2V0D5vw0nBdO1hCJoZo5mKeuIQ==",
170 | "dev": true,
171 | "requires": {
172 | "ansi-align": "^3.0.0",
173 | "camelcase": "^5.3.1",
174 | "chalk": "^3.0.0",
175 | "cli-boxes": "^2.2.0",
176 | "string-width": "^4.1.0",
177 | "term-size": "^2.1.0",
178 | "type-fest": "^0.8.1",
179 | "widest-line": "^3.1.0"
180 | }
181 | },
182 | "brace-expansion": {
183 | "version": "1.1.11",
184 | "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz",
185 | "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==",
186 | "dev": true,
187 | "requires": {
188 | "balanced-match": "^1.0.0",
189 | "concat-map": "0.0.1"
190 | }
191 | },
192 | "braces": {
193 | "version": "3.0.2",
194 | "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz",
195 | "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==",
196 | "dev": true,
197 | "requires": {
198 | "fill-range": "^7.0.1"
199 | }
200 | },
201 | "browser-stdout": {
202 | "version": "1.3.1",
203 | "resolved": "https://registry.npmjs.org/browser-stdout/-/browser-stdout-1.3.1.tgz",
204 | "integrity": "sha512-qhAVI1+Av2X7qelOfAIYwXONood6XlZE/fXaBSmW/T5SzLAmCgzi+eiWE7fUvbHaeNBQH13UftjpXxsfLkMpgw==",
205 | "dev": true
206 | },
207 | "bson": {
208 | "version": "1.1.4",
209 | "resolved": "https://registry.npmjs.org/bson/-/bson-1.1.4.tgz",
210 | "integrity": "sha512-S/yKGU1syOMzO86+dGpg2qGoDL0zvzcb262G+gqEy6TgP6rt6z6qxSFX/8X6vLC91P7G7C3nLs0+bvDzmvBA3Q=="
211 | },
212 | "bytes": {
213 | "version": "3.1.0",
214 | "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.0.tgz",
215 | "integrity": "sha512-zauLjrfCG+xvoyaqLoV8bLVXXNGC4JqlxFCutSDWA6fJrTo2ZuvLYTqZ7aHBLZSMOopbzwv8f+wZcVzfVTI2Dg=="
216 | },
217 | "cacheable-request": {
218 | "version": "6.1.0",
219 | "resolved": "https://registry.npmjs.org/cacheable-request/-/cacheable-request-6.1.0.tgz",
220 | "integrity": "sha512-Oj3cAGPCqOZX7Rz64Uny2GYAZNliQSqfbePrgAQ1wKAihYmCUnraBtJtKcGR4xz7wF+LoJC+ssFZvv5BgF9Igg==",
221 | "dev": true,
222 | "requires": {
223 | "clone-response": "^1.0.2",
224 | "get-stream": "^5.1.0",
225 | "http-cache-semantics": "^4.0.0",
226 | "keyv": "^3.0.0",
227 | "lowercase-keys": "^2.0.0",
228 | "normalize-url": "^4.1.0",
229 | "responselike": "^1.0.2"
230 | },
231 | "dependencies": {
232 | "get-stream": {
233 | "version": "5.1.0",
234 | "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-5.1.0.tgz",
235 | "integrity": "sha512-EXr1FOzrzTfGeL0gQdeFEvOMm2mzMOglyiOXSTpPC+iAjAKftbr3jpCMWynogwYnM+eSj9sHGc6wjIcDvYiygw==",
236 | "dev": true,
237 | "requires": {
238 | "pump": "^3.0.0"
239 | }
240 | },
241 | "lowercase-keys": {
242 | "version": "2.0.0",
243 | "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-2.0.0.tgz",
244 | "integrity": "sha512-tqNXrS78oMOE73NMxK4EMLQsQowWf8jKooH9g7xPavRT706R6bkQJ6DY2Te7QukaZsulxa30wQ7bk0pm4XiHmA==",
245 | "dev": true
246 | }
247 | }
248 | },
249 | "camelcase": {
250 | "version": "5.3.1",
251 | "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz",
252 | "integrity": "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==",
253 | "dev": true
254 | },
255 | "chai": {
256 | "version": "4.2.0",
257 | "resolved": "https://registry.npmjs.org/chai/-/chai-4.2.0.tgz",
258 | "integrity": "sha512-XQU3bhBukrOsQCuwZndwGcCVQHyZi53fQ6Ys1Fym7E4olpIqqZZhhoFJoaKVvV17lWQoXYwgWN2nF5crA8J2jw==",
259 | "dev": true,
260 | "requires": {
261 | "assertion-error": "^1.1.0",
262 | "check-error": "^1.0.2",
263 | "deep-eql": "^3.0.1",
264 | "get-func-name": "^2.0.0",
265 | "pathval": "^1.1.0",
266 | "type-detect": "^4.0.5"
267 | }
268 | },
269 | "chalk": {
270 | "version": "3.0.0",
271 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-3.0.0.tgz",
272 | "integrity": "sha512-4D3B6Wf41KOYRFdszmDqMCGq5VV/uMAB273JILmO+3jAlh8X4qDtdtgCR3fxtbLEMzSx22QdhnDcJvu2u1fVwg==",
273 | "dev": true,
274 | "requires": {
275 | "ansi-styles": "^4.1.0",
276 | "supports-color": "^7.1.0"
277 | },
278 | "dependencies": {
279 | "has-flag": {
280 | "version": "4.0.0",
281 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz",
282 | "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==",
283 | "dev": true
284 | },
285 | "supports-color": {
286 | "version": "7.1.0",
287 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.1.0.tgz",
288 | "integrity": "sha512-oRSIpR8pxT1Wr2FquTNnGet79b3BWljqOuoW/h4oBhxJ/HUbX5nX6JSruTkvXDCFMwDPvsaTTbvMLKZWSy0R5g==",
289 | "dev": true,
290 | "requires": {
291 | "has-flag": "^4.0.0"
292 | }
293 | }
294 | }
295 | },
296 | "check-error": {
297 | "version": "1.0.2",
298 | "resolved": "https://registry.npmjs.org/check-error/-/check-error-1.0.2.tgz",
299 | "integrity": "sha1-V00xLt2Iu13YkS6Sht1sCu1KrII=",
300 | "dev": true
301 | },
302 | "chokidar": {
303 | "version": "3.4.0",
304 | "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.4.0.tgz",
305 | "integrity": "sha512-aXAaho2VJtisB/1fg1+3nlLJqGOuewTzQpd/Tz0yTg2R0e4IGtshYvtjowyEumcBv2z+y4+kc75Mz7j5xJskcQ==",
306 | "dev": true,
307 | "requires": {
308 | "anymatch": "~3.1.1",
309 | "braces": "~3.0.2",
310 | "fsevents": "~2.1.2",
311 | "glob-parent": "~5.1.0",
312 | "is-binary-path": "~2.1.0",
313 | "is-glob": "~4.0.1",
314 | "normalize-path": "~3.0.0",
315 | "readdirp": "~3.4.0"
316 | }
317 | },
318 | "ci-info": {
319 | "version": "2.0.0",
320 | "resolved": "https://registry.npmjs.org/ci-info/-/ci-info-2.0.0.tgz",
321 | "integrity": "sha512-5tK7EtrZ0N+OLFMthtqOj4fI2Jeb88C4CAZPu25LDVUgXJ0A3Js4PMGqrn0JU1W0Mh1/Z8wZzYPxqUrXeBboCQ==",
322 | "dev": true
323 | },
324 | "cli-boxes": {
325 | "version": "2.2.0",
326 | "resolved": "https://registry.npmjs.org/cli-boxes/-/cli-boxes-2.2.0.tgz",
327 | "integrity": "sha512-gpaBrMAizVEANOpfZp/EEUixTXDyGt7DFzdK5hU+UbWt/J0lB0w20ncZj59Z9a93xHb9u12zF5BS6i9RKbtg4w==",
328 | "dev": true
329 | },
330 | "cliui": {
331 | "version": "5.0.0",
332 | "resolved": "https://registry.npmjs.org/cliui/-/cliui-5.0.0.tgz",
333 | "integrity": "sha512-PYeGSEmmHM6zvoef2w8TPzlrnNpXIjTipYK780YswmIP9vjxmd6Y2a3CB2Ks6/AU8NHjZugXvo8w3oWM2qnwXA==",
334 | "dev": true,
335 | "requires": {
336 | "string-width": "^3.1.0",
337 | "strip-ansi": "^5.2.0",
338 | "wrap-ansi": "^5.1.0"
339 | },
340 | "dependencies": {
341 | "string-width": {
342 | "version": "3.1.0",
343 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz",
344 | "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==",
345 | "dev": true,
346 | "requires": {
347 | "emoji-regex": "^7.0.1",
348 | "is-fullwidth-code-point": "^2.0.0",
349 | "strip-ansi": "^5.1.0"
350 | }
351 | }
352 | }
353 | },
354 | "clone-response": {
355 | "version": "1.0.2",
356 | "resolved": "https://registry.npmjs.org/clone-response/-/clone-response-1.0.2.tgz",
357 | "integrity": "sha1-0dyXOSAxTfZ/vrlCI7TuNQI56Ws=",
358 | "dev": true,
359 | "requires": {
360 | "mimic-response": "^1.0.0"
361 | }
362 | },
363 | "color-convert": {
364 | "version": "2.0.1",
365 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz",
366 | "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==",
367 | "dev": true,
368 | "requires": {
369 | "color-name": "~1.1.4"
370 | }
371 | },
372 | "color-name": {
373 | "version": "1.1.4",
374 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz",
375 | "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==",
376 | "dev": true
377 | },
378 | "combined-stream": {
379 | "version": "1.0.8",
380 | "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz",
381 | "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==",
382 | "dev": true,
383 | "requires": {
384 | "delayed-stream": "~1.0.0"
385 | }
386 | },
387 | "component-emitter": {
388 | "version": "1.3.0",
389 | "resolved": "https://registry.npmjs.org/component-emitter/-/component-emitter-1.3.0.tgz",
390 | "integrity": "sha512-Rd3se6QB+sO1TwqZjscQrurpEPIfO0/yYnSin6Q/rD3mOutHvUrCAhJub3r90uNb+SESBuE0QYoB90YdfatsRg==",
391 | "dev": true
392 | },
393 | "concat-map": {
394 | "version": "0.0.1",
395 | "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz",
396 | "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=",
397 | "dev": true
398 | },
399 | "configstore": {
400 | "version": "5.0.1",
401 | "resolved": "https://registry.npmjs.org/configstore/-/configstore-5.0.1.tgz",
402 | "integrity": "sha512-aMKprgk5YhBNyH25hj8wGt2+D52Sw1DRRIzqBwLp2Ya9mFmY8KPvvtvmna8SxVR9JMZ4kzMD68N22vlaRpkeFA==",
403 | "dev": true,
404 | "requires": {
405 | "dot-prop": "^5.2.0",
406 | "graceful-fs": "^4.1.2",
407 | "make-dir": "^3.0.0",
408 | "unique-string": "^2.0.0",
409 | "write-file-atomic": "^3.0.0",
410 | "xdg-basedir": "^4.0.0"
411 | }
412 | },
413 | "content-disposition": {
414 | "version": "0.5.3",
415 | "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.3.tgz",
416 | "integrity": "sha512-ExO0774ikEObIAEV9kDo50o+79VCUdEB6n6lzKgGwupcVeRlhrj3qGAfwq8G6uBJjkqLrhT0qEYFcWng8z1z0g==",
417 | "requires": {
418 | "safe-buffer": "5.1.2"
419 | }
420 | },
421 | "content-type": {
422 | "version": "1.0.4",
423 | "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.4.tgz",
424 | "integrity": "sha512-hIP3EEPs8tB9AT1L+NUqtwOAps4mk2Zob89MWXMHjHWg9milF/j4osnnQLXBCBFBk/tvIG/tUc9mOUJiPBhPXA=="
425 | },
426 | "cookie": {
427 | "version": "0.4.0",
428 | "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.4.0.tgz",
429 | "integrity": "sha512-+Hp8fLp57wnUSt0tY0tHEXh4voZRDnoIrZPqlo3DPiI4y9lwg/jqx+1Om94/W6ZaPDOUbnjOt/99w66zk+l1Xg=="
430 | },
431 | "cookie-signature": {
432 | "version": "1.0.6",
433 | "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz",
434 | "integrity": "sha1-4wOogrNCzD7oylE6eZmXNNqzriw="
435 | },
436 | "cookiejar": {
437 | "version": "2.1.2",
438 | "resolved": "https://registry.npmjs.org/cookiejar/-/cookiejar-2.1.2.tgz",
439 | "integrity": "sha512-Mw+adcfzPxcPeI+0WlvRrr/3lGVO0bD75SxX6811cxSh1Wbxx7xZBGK1eVtDf6si8rg2lhnUjsVLMFMfbRIuwA==",
440 | "dev": true
441 | },
442 | "core-util-is": {
443 | "version": "1.0.2",
444 | "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.2.tgz",
445 | "integrity": "sha1-tf1UIgqivFq1eqtxQMlAdUUDwac="
446 | },
447 | "cors": {
448 | "version": "2.8.5",
449 | "resolved": "https://registry.npmjs.org/cors/-/cors-2.8.5.tgz",
450 | "integrity": "sha512-KIHbLJqu73RGr/hnbrO9uBeixNGuvSQjul/jdFvS/KFSIH1hWVd1ng7zOHx+YrEfInLG7q4n6GHQ9cDtxv/P6g==",
451 | "requires": {
452 | "object-assign": "^4",
453 | "vary": "^1"
454 | }
455 | },
456 | "crypto-random-string": {
457 | "version": "2.0.0",
458 | "resolved": "https://registry.npmjs.org/crypto-random-string/-/crypto-random-string-2.0.0.tgz",
459 | "integrity": "sha512-v1plID3y9r/lPhviJ1wrXpLeyUIGAZ2SHNYTEapm7/8A9nLPoyvVp3RK/EPFqn5kEznyWgYZNsRtYYIWbuG8KA==",
460 | "dev": true
461 | },
462 | "debug": {
463 | "version": "2.6.9",
464 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz",
465 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==",
466 | "requires": {
467 | "ms": "2.0.0"
468 | }
469 | },
470 | "decamelize": {
471 | "version": "1.2.0",
472 | "resolved": "https://registry.npmjs.org/decamelize/-/decamelize-1.2.0.tgz",
473 | "integrity": "sha1-9lNNFRSCabIDUue+4m9QH5oZEpA=",
474 | "dev": true
475 | },
476 | "decompress-response": {
477 | "version": "3.3.0",
478 | "resolved": "https://registry.npmjs.org/decompress-response/-/decompress-response-3.3.0.tgz",
479 | "integrity": "sha1-gKTdMjdIOEv6JICDYirt7Jgq3/M=",
480 | "dev": true,
481 | "requires": {
482 | "mimic-response": "^1.0.0"
483 | }
484 | },
485 | "deep-eql": {
486 | "version": "3.0.1",
487 | "resolved": "https://registry.npmjs.org/deep-eql/-/deep-eql-3.0.1.tgz",
488 | "integrity": "sha512-+QeIQyN5ZuO+3Uk5DYh6/1eKO0m0YmJFGNmFHGACpf1ClL1nmlV/p4gNgbl2pJGxgXb4faqo6UE+M5ACEMyVcw==",
489 | "dev": true,
490 | "requires": {
491 | "type-detect": "^4.0.0"
492 | }
493 | },
494 | "deep-extend": {
495 | "version": "0.6.0",
496 | "resolved": "https://registry.npmjs.org/deep-extend/-/deep-extend-0.6.0.tgz",
497 | "integrity": "sha512-LOHxIOaPYdHlJRtCQfDIVZtfw/ufM8+rVj649RIHzcm/vGwQRXFt6OPqIFWsm2XEMrNIEtWR64sY1LEKD2vAOA==",
498 | "dev": true
499 | },
500 | "defer-to-connect": {
501 | "version": "1.1.3",
502 | "resolved": "https://registry.npmjs.org/defer-to-connect/-/defer-to-connect-1.1.3.tgz",
503 | "integrity": "sha512-0ISdNousHvZT2EiFlZeZAHBUvSxmKswVCEf8hW7KWgG4a8MVEu/3Vb6uWYozkjylyCxe0JBIiRB1jV45S70WVQ==",
504 | "dev": true
505 | },
506 | "define-properties": {
507 | "version": "1.1.3",
508 | "resolved": "https://registry.npmjs.org/define-properties/-/define-properties-1.1.3.tgz",
509 | "integrity": "sha512-3MqfYKj2lLzdMSf8ZIZE/V+Zuy+BgD6f164e8K2w7dgnpKArBDerGYpM46IYYcjnkdPNMjPk9A6VFB8+3SKlXQ==",
510 | "dev": true,
511 | "requires": {
512 | "object-keys": "^1.0.12"
513 | }
514 | },
515 | "delayed-stream": {
516 | "version": "1.0.0",
517 | "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz",
518 | "integrity": "sha1-3zrhmayt+31ECqrgsp4icrJOxhk=",
519 | "dev": true
520 | },
521 | "denque": {
522 | "version": "1.4.1",
523 | "resolved": "https://registry.npmjs.org/denque/-/denque-1.4.1.tgz",
524 | "integrity": "sha512-OfzPuSZKGcgr96rf1oODnfjqBFmr1DVoc/TrItj3Ohe0Ah1C5WX5Baquw/9U9KovnQ88EqmJbD66rKYUQYN1tQ=="
525 | },
526 | "depd": {
527 | "version": "1.1.2",
528 | "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.2.tgz",
529 | "integrity": "sha1-m81S4UwJd2PnSbJ0xDRu0uVgtak="
530 | },
531 | "destroy": {
532 | "version": "1.0.4",
533 | "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.0.4.tgz",
534 | "integrity": "sha1-l4hXRCxEdJ5CBmE+N5RiBYJqvYA="
535 | },
536 | "diff": {
537 | "version": "3.5.0",
538 | "resolved": "https://registry.npmjs.org/diff/-/diff-3.5.0.tgz",
539 | "integrity": "sha512-A46qtFgd+g7pDZinpnwiRJtxbC1hpgf0uzP3iG89scHk0AUC7A1TGxf5OiiOUv/JMZR8GOt8hL900hV0bOy5xA==",
540 | "dev": true
541 | },
542 | "dot-prop": {
543 | "version": "5.2.0",
544 | "resolved": "https://registry.npmjs.org/dot-prop/-/dot-prop-5.2.0.tgz",
545 | "integrity": "sha512-uEUyaDKoSQ1M4Oq8l45hSE26SnTxL6snNnqvK/VWx5wJhmff5z0FUVJDKDanor/6w3kzE3i7XZOk+7wC0EXr1A==",
546 | "dev": true,
547 | "requires": {
548 | "is-obj": "^2.0.0"
549 | }
550 | },
551 | "dotenv": {
552 | "version": "8.2.0",
553 | "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-8.2.0.tgz",
554 | "integrity": "sha512-8sJ78ElpbDJBHNeBzUbUVLsqKdccaa/BXF1uPTw3GrvQTBgrQrtObr2mUrE38vzYd8cEv+m/JBfDLioYcfXoaw=="
555 | },
556 | "duplexer3": {
557 | "version": "0.1.4",
558 | "resolved": "https://registry.npmjs.org/duplexer3/-/duplexer3-0.1.4.tgz",
559 | "integrity": "sha1-7gHdHKwO08vH/b6jfcCo8c4ALOI=",
560 | "dev": true
561 | },
562 | "ee-first": {
563 | "version": "1.1.1",
564 | "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz",
565 | "integrity": "sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0="
566 | },
567 | "emoji-regex": {
568 | "version": "7.0.3",
569 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-7.0.3.tgz",
570 | "integrity": "sha512-CwBLREIQ7LvYFB0WyRvwhq5N5qPhc6PMjD6bYggFlI5YyDgl+0vxq5VHbMOFqLg7hfWzmu8T5Z1QofhmTIhItA==",
571 | "dev": true
572 | },
573 | "encodeurl": {
574 | "version": "1.0.2",
575 | "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz",
576 | "integrity": "sha1-rT/0yG7C0CkyL1oCw6mmBslbP1k="
577 | },
578 | "end-of-stream": {
579 | "version": "1.4.4",
580 | "resolved": "https://registry.npmjs.org/end-of-stream/-/end-of-stream-1.4.4.tgz",
581 | "integrity": "sha512-+uw1inIHVPQoaVuHzRyXd21icM+cnt4CzD5rW+NC1wjOUSTOs+Te7FOv7AhN7vS9x/oIyhLP5PR1H+phQAHu5Q==",
582 | "dev": true,
583 | "requires": {
584 | "once": "^1.4.0"
585 | }
586 | },
587 | "es-abstract": {
588 | "version": "1.17.5",
589 | "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.17.5.tgz",
590 | "integrity": "sha512-BR9auzDbySxOcfog0tLECW8l28eRGpDpU3Dm3Hp4q/N+VtLTmyj4EUN088XZWQDW/hzj6sYRDXeOFsaAODKvpg==",
591 | "dev": true,
592 | "requires": {
593 | "es-to-primitive": "^1.2.1",
594 | "function-bind": "^1.1.1",
595 | "has": "^1.0.3",
596 | "has-symbols": "^1.0.1",
597 | "is-callable": "^1.1.5",
598 | "is-regex": "^1.0.5",
599 | "object-inspect": "^1.7.0",
600 | "object-keys": "^1.1.1",
601 | "object.assign": "^4.1.0",
602 | "string.prototype.trimleft": "^2.1.1",
603 | "string.prototype.trimright": "^2.1.1"
604 | }
605 | },
606 | "es-to-primitive": {
607 | "version": "1.2.1",
608 | "resolved": "https://registry.npmjs.org/es-to-primitive/-/es-to-primitive-1.2.1.tgz",
609 | "integrity": "sha512-QCOllgZJtaUo9miYBcLChTUaHNjJF3PYs1VidD7AwiEj1kYxKeQTctLAezAOH5ZKRH0g2IgPn6KwB4IT8iRpvA==",
610 | "dev": true,
611 | "requires": {
612 | "is-callable": "^1.1.4",
613 | "is-date-object": "^1.0.1",
614 | "is-symbol": "^1.0.2"
615 | }
616 | },
617 | "escape-goat": {
618 | "version": "2.1.1",
619 | "resolved": "https://registry.npmjs.org/escape-goat/-/escape-goat-2.1.1.tgz",
620 | "integrity": "sha512-8/uIhbG12Csjy2JEW7D9pHbreaVaS/OpN3ycnyvElTdwM5n6GY6W6e2IPemfvGZeUMqZ9A/3GqIZMgKnBhAw/Q==",
621 | "dev": true
622 | },
623 | "escape-html": {
624 | "version": "1.0.3",
625 | "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz",
626 | "integrity": "sha1-Aljq5NPQwJdN4cFpGI7wBR0dGYg="
627 | },
628 | "escape-string-regexp": {
629 | "version": "1.0.5",
630 | "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz",
631 | "integrity": "sha1-G2HAViGQqN/2rjuyzwIAyhMLhtQ=",
632 | "dev": true
633 | },
634 | "esprima": {
635 | "version": "4.0.1",
636 | "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz",
637 | "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==",
638 | "dev": true
639 | },
640 | "etag": {
641 | "version": "1.8.1",
642 | "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz",
643 | "integrity": "sha1-Qa4u62XvpiJorr/qg6x9eSmbCIc="
644 | },
645 | "express": {
646 | "version": "4.17.1",
647 | "resolved": "https://registry.npmjs.org/express/-/express-4.17.1.tgz",
648 | "integrity": "sha512-mHJ9O79RqluphRrcw2X/GTh3k9tVv8YcoyY4Kkh4WDMUYKRZUq0h1o0w2rrrxBqM7VoeUVqgb27xlEMXTnYt4g==",
649 | "requires": {
650 | "accepts": "~1.3.7",
651 | "array-flatten": "1.1.1",
652 | "body-parser": "1.19.0",
653 | "content-disposition": "0.5.3",
654 | "content-type": "~1.0.4",
655 | "cookie": "0.4.0",
656 | "cookie-signature": "1.0.6",
657 | "debug": "2.6.9",
658 | "depd": "~1.1.2",
659 | "encodeurl": "~1.0.2",
660 | "escape-html": "~1.0.3",
661 | "etag": "~1.8.1",
662 | "finalhandler": "~1.1.2",
663 | "fresh": "0.5.2",
664 | "merge-descriptors": "1.0.1",
665 | "methods": "~1.1.2",
666 | "on-finished": "~2.3.0",
667 | "parseurl": "~1.3.3",
668 | "path-to-regexp": "0.1.7",
669 | "proxy-addr": "~2.0.5",
670 | "qs": "6.7.0",
671 | "range-parser": "~1.2.1",
672 | "safe-buffer": "5.1.2",
673 | "send": "0.17.1",
674 | "serve-static": "1.14.1",
675 | "setprototypeof": "1.1.1",
676 | "statuses": "~1.5.0",
677 | "type-is": "~1.6.18",
678 | "utils-merge": "1.0.1",
679 | "vary": "~1.1.2"
680 | }
681 | },
682 | "extend": {
683 | "version": "3.0.2",
684 | "resolved": "https://registry.npmjs.org/extend/-/extend-3.0.2.tgz",
685 | "integrity": "sha512-fjquC59cD7CyW6urNXK0FBufkZcoiGG80wTuPujX590cB5Ttln20E2UB4S/WARVqhXffZl2LNgS+gQdPIIim/g==",
686 | "dev": true
687 | },
688 | "fill-range": {
689 | "version": "7.0.1",
690 | "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz",
691 | "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==",
692 | "dev": true,
693 | "requires": {
694 | "to-regex-range": "^5.0.1"
695 | }
696 | },
697 | "finalhandler": {
698 | "version": "1.1.2",
699 | "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.1.2.tgz",
700 | "integrity": "sha512-aAWcW57uxVNrQZqFXjITpW3sIUQmHGG3qSb9mUah9MgMC4NeWhNOlNjXEYq3HjRAvL6arUviZGGJsBg6z0zsWA==",
701 | "requires": {
702 | "debug": "2.6.9",
703 | "encodeurl": "~1.0.2",
704 | "escape-html": "~1.0.3",
705 | "on-finished": "~2.3.0",
706 | "parseurl": "~1.3.3",
707 | "statuses": "~1.5.0",
708 | "unpipe": "~1.0.0"
709 | }
710 | },
711 | "find-up": {
712 | "version": "3.0.0",
713 | "resolved": "https://registry.npmjs.org/find-up/-/find-up-3.0.0.tgz",
714 | "integrity": "sha512-1yD6RmLI1XBfxugvORwlck6f75tYL+iR0jqwsOrOxMZyGYqUuDhJ0l4AXdO1iX/FTs9cBAMEk1gWSEx1kSbylg==",
715 | "dev": true,
716 | "requires": {
717 | "locate-path": "^3.0.0"
718 | }
719 | },
720 | "flat": {
721 | "version": "4.1.0",
722 | "resolved": "https://registry.npmjs.org/flat/-/flat-4.1.0.tgz",
723 | "integrity": "sha512-Px/TiLIznH7gEDlPXcUD4KnBusa6kR6ayRUVcnEAbreRIuhkqow/mun59BuRXwoYk7ZQOLW1ZM05ilIvK38hFw==",
724 | "dev": true,
725 | "requires": {
726 | "is-buffer": "~2.0.3"
727 | }
728 | },
729 | "form-data": {
730 | "version": "2.5.1",
731 | "resolved": "https://registry.npmjs.org/form-data/-/form-data-2.5.1.tgz",
732 | "integrity": "sha512-m21N3WOmEEURgk6B9GLOE4RuWOFf28Lhh9qGYeNlGq4VDXUlJy2th2slBNU8Gp8EzloYZOibZJ7t5ecIrFSjVA==",
733 | "dev": true,
734 | "requires": {
735 | "asynckit": "^0.4.0",
736 | "combined-stream": "^1.0.6",
737 | "mime-types": "^2.1.12"
738 | }
739 | },
740 | "formidable": {
741 | "version": "1.2.2",
742 | "resolved": "https://registry.npmjs.org/formidable/-/formidable-1.2.2.tgz",
743 | "integrity": "sha512-V8gLm+41I/8kguQ4/o1D3RIHRmhYFG4pnNyonvua+40rqcEmT4+V71yaZ3B457xbbgCsCfjSPi65u/W6vK1U5Q==",
744 | "dev": true
745 | },
746 | "forwarded": {
747 | "version": "0.1.2",
748 | "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.1.2.tgz",
749 | "integrity": "sha1-mMI9qxF1ZXuMBXPozszZGw/xjIQ="
750 | },
751 | "fresh": {
752 | "version": "0.5.2",
753 | "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz",
754 | "integrity": "sha1-PYyt2Q2XZWn6g1qx+OSyOhBWBac="
755 | },
756 | "fs.realpath": {
757 | "version": "1.0.0",
758 | "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz",
759 | "integrity": "sha1-FQStJSMVjKpA20onh8sBQRmU6k8=",
760 | "dev": true
761 | },
762 | "fsevents": {
763 | "version": "2.1.3",
764 | "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.1.3.tgz",
765 | "integrity": "sha512-Auw9a4AxqWpa9GUfj370BMPzzyncfBABW8Mab7BGWBYDj4Isgq+cDKtx0i6u9jcX9pQDnswsaaOTgTmA5pEjuQ==",
766 | "dev": true,
767 | "optional": true
768 | },
769 | "function-bind": {
770 | "version": "1.1.1",
771 | "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.1.tgz",
772 | "integrity": "sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A==",
773 | "dev": true
774 | },
775 | "get-caller-file": {
776 | "version": "2.0.5",
777 | "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-2.0.5.tgz",
778 | "integrity": "sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==",
779 | "dev": true
780 | },
781 | "get-func-name": {
782 | "version": "2.0.0",
783 | "resolved": "https://registry.npmjs.org/get-func-name/-/get-func-name-2.0.0.tgz",
784 | "integrity": "sha1-6td0q+5y4gQJQzoGY2YCPdaIekE=",
785 | "dev": true
786 | },
787 | "get-stream": {
788 | "version": "4.1.0",
789 | "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-4.1.0.tgz",
790 | "integrity": "sha512-GMat4EJ5161kIy2HevLlr4luNjBgvmj413KaQA7jt4V8B4RDsfpHk7WQ9GVqfYyyx8OS/L66Kox+rJRNklLK7w==",
791 | "dev": true,
792 | "requires": {
793 | "pump": "^3.0.0"
794 | }
795 | },
796 | "glob": {
797 | "version": "7.1.3",
798 | "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.3.tgz",
799 | "integrity": "sha512-vcfuiIxogLV4DlGBHIUOwI0IbrJ8HWPc4MU7HzviGeNho/UJDfi6B5p3sHeWIQ0KGIU0Jpxi5ZHxemQfLkkAwQ==",
800 | "dev": true,
801 | "requires": {
802 | "fs.realpath": "^1.0.0",
803 | "inflight": "^1.0.4",
804 | "inherits": "2",
805 | "minimatch": "^3.0.4",
806 | "once": "^1.3.0",
807 | "path-is-absolute": "^1.0.0"
808 | }
809 | },
810 | "glob-parent": {
811 | "version": "5.1.1",
812 | "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.1.tgz",
813 | "integrity": "sha512-FnI+VGOpnlGHWZxthPGR+QhR78fuiK0sNLkHQv+bL9fQi57lNNdquIbna/WrfROrolq8GK5Ek6BiMwqL/voRYQ==",
814 | "dev": true,
815 | "requires": {
816 | "is-glob": "^4.0.1"
817 | }
818 | },
819 | "global-dirs": {
820 | "version": "2.0.1",
821 | "resolved": "https://registry.npmjs.org/global-dirs/-/global-dirs-2.0.1.tgz",
822 | "integrity": "sha512-5HqUqdhkEovj2Of/ms3IeS/EekcO54ytHRLV4PEY2rhRwrHXLQjeVEES0Lhka0xwNDtGYn58wyC4s5+MHsOO6A==",
823 | "dev": true,
824 | "requires": {
825 | "ini": "^1.3.5"
826 | }
827 | },
828 | "got": {
829 | "version": "9.6.0",
830 | "resolved": "https://registry.npmjs.org/got/-/got-9.6.0.tgz",
831 | "integrity": "sha512-R7eWptXuGYxwijs0eV+v3o6+XH1IqVK8dJOEecQfTmkncw9AV4dcw/Dhxi8MdlqPthxxpZyizMzyg8RTmEsG+Q==",
832 | "dev": true,
833 | "requires": {
834 | "@sindresorhus/is": "^0.14.0",
835 | "@szmarczak/http-timer": "^1.1.2",
836 | "cacheable-request": "^6.0.0",
837 | "decompress-response": "^3.3.0",
838 | "duplexer3": "^0.1.4",
839 | "get-stream": "^4.1.0",
840 | "lowercase-keys": "^1.0.1",
841 | "mimic-response": "^1.0.1",
842 | "p-cancelable": "^1.0.0",
843 | "to-readable-stream": "^1.0.0",
844 | "url-parse-lax": "^3.0.0"
845 | }
846 | },
847 | "graceful-fs": {
848 | "version": "4.2.4",
849 | "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.4.tgz",
850 | "integrity": "sha512-WjKPNJF79dtJAVniUlGGWHYGz2jWxT6VhN/4m1NdkbZ2nOsEF+cI1Edgql5zCRhs/VsQYRvrXctxktVXZUkixw==",
851 | "dev": true
852 | },
853 | "growl": {
854 | "version": "1.10.5",
855 | "resolved": "https://registry.npmjs.org/growl/-/growl-1.10.5.tgz",
856 | "integrity": "sha512-qBr4OuELkhPenW6goKVXiv47US3clb3/IbuWF9KNKEijAy9oeHxU9IgzjvJhHkUzhaj7rOUD7+YGWqUjLp5oSA==",
857 | "dev": true
858 | },
859 | "has": {
860 | "version": "1.0.3",
861 | "resolved": "https://registry.npmjs.org/has/-/has-1.0.3.tgz",
862 | "integrity": "sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw==",
863 | "dev": true,
864 | "requires": {
865 | "function-bind": "^1.1.1"
866 | }
867 | },
868 | "has-flag": {
869 | "version": "3.0.0",
870 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz",
871 | "integrity": "sha1-tdRU3CGZriJWmfNGfloH87lVuv0=",
872 | "dev": true
873 | },
874 | "has-symbols": {
875 | "version": "1.0.1",
876 | "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.0.1.tgz",
877 | "integrity": "sha512-PLcsoqu++dmEIZB+6totNFKq/7Do+Z0u4oT0zKOJNl3lYK6vGwwu2hjHs+68OEZbTjiUE9bgOABXbP/GvrS0Kg==",
878 | "dev": true
879 | },
880 | "has-yarn": {
881 | "version": "2.1.0",
882 | "resolved": "https://registry.npmjs.org/has-yarn/-/has-yarn-2.1.0.tgz",
883 | "integrity": "sha512-UqBRqi4ju7T+TqGNdqAO0PaSVGsDGJUBQvk9eUWNGRY1CFGDzYhLWoM7JQEemnlvVcv/YEmc2wNW8BC24EnUsw==",
884 | "dev": true
885 | },
886 | "he": {
887 | "version": "1.2.0",
888 | "resolved": "https://registry.npmjs.org/he/-/he-1.2.0.tgz",
889 | "integrity": "sha512-F/1DnUGPopORZi0ni+CvrCgHQ5FyEAHRLSApuYWMmrbSwoN2Mn/7k+Gl38gJnR7yyDZk6WLXwiGod1JOWNDKGw==",
890 | "dev": true
891 | },
892 | "http-cache-semantics": {
893 | "version": "4.1.0",
894 | "resolved": "https://registry.npmjs.org/http-cache-semantics/-/http-cache-semantics-4.1.0.tgz",
895 | "integrity": "sha512-carPklcUh7ROWRK7Cv27RPtdhYhUsela/ue5/jKzjegVvXDqM2ILE9Q2BGn9JZJh1g87cp56su/FgQSzcWS8cQ==",
896 | "dev": true
897 | },
898 | "http-errors": {
899 | "version": "1.7.2",
900 | "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.7.2.tgz",
901 | "integrity": "sha512-uUQBt3H/cSIVfch6i1EuPNy/YsRSOUBXTVfZ+yR7Zjez3qjBz6i9+i4zjNaoqcoFVI4lQJ5plg63TvGfRSDCRg==",
902 | "requires": {
903 | "depd": "~1.1.2",
904 | "inherits": "2.0.3",
905 | "setprototypeof": "1.1.1",
906 | "statuses": ">= 1.5.0 < 2",
907 | "toidentifier": "1.0.0"
908 | }
909 | },
910 | "iconv-lite": {
911 | "version": "0.4.24",
912 | "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz",
913 | "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==",
914 | "requires": {
915 | "safer-buffer": ">= 2.1.2 < 3"
916 | }
917 | },
918 | "ignore-by-default": {
919 | "version": "1.0.1",
920 | "resolved": "https://registry.npmjs.org/ignore-by-default/-/ignore-by-default-1.0.1.tgz",
921 | "integrity": "sha1-SMptcvbGo68Aqa1K5odr44ieKwk=",
922 | "dev": true
923 | },
924 | "import-lazy": {
925 | "version": "2.1.0",
926 | "resolved": "https://registry.npmjs.org/import-lazy/-/import-lazy-2.1.0.tgz",
927 | "integrity": "sha1-BWmOPUXIjo1+nZLLBYTnfwlvPkM=",
928 | "dev": true
929 | },
930 | "imurmurhash": {
931 | "version": "0.1.4",
932 | "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz",
933 | "integrity": "sha1-khi5srkoojixPcT7a21XbyMUU+o=",
934 | "dev": true
935 | },
936 | "inflight": {
937 | "version": "1.0.6",
938 | "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz",
939 | "integrity": "sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk=",
940 | "dev": true,
941 | "requires": {
942 | "once": "^1.3.0",
943 | "wrappy": "1"
944 | }
945 | },
946 | "inherits": {
947 | "version": "2.0.3",
948 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz",
949 | "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4="
950 | },
951 | "ini": {
952 | "version": "1.3.5",
953 | "resolved": "https://registry.npmjs.org/ini/-/ini-1.3.5.tgz",
954 | "integrity": "sha512-RZY5huIKCMRWDUqZlEi72f/lmXKMvuszcMBduliQ3nnWbx9X/ZBQO7DijMEYS9EhHBb2qacRUMtC7svLwe0lcw==",
955 | "dev": true
956 | },
957 | "ipaddr.js": {
958 | "version": "1.9.1",
959 | "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz",
960 | "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g=="
961 | },
962 | "is-binary-path": {
963 | "version": "2.1.0",
964 | "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz",
965 | "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==",
966 | "dev": true,
967 | "requires": {
968 | "binary-extensions": "^2.0.0"
969 | }
970 | },
971 | "is-buffer": {
972 | "version": "2.0.4",
973 | "resolved": "https://registry.npmjs.org/is-buffer/-/is-buffer-2.0.4.tgz",
974 | "integrity": "sha512-Kq1rokWXOPXWuaMAqZiJW4XxsmD9zGx9q4aePabbn3qCRGedtH7Cm+zV8WETitMfu1wdh+Rvd6w5egwSngUX2A==",
975 | "dev": true
976 | },
977 | "is-callable": {
978 | "version": "1.1.5",
979 | "resolved": "https://registry.npmjs.org/is-callable/-/is-callable-1.1.5.tgz",
980 | "integrity": "sha512-ESKv5sMCJB2jnHTWZ3O5itG+O128Hsus4K4Qh1h2/cgn2vbgnLSVqfV46AeJA9D5EeeLa9w81KUXMtn34zhX+Q==",
981 | "dev": true
982 | },
983 | "is-ci": {
984 | "version": "2.0.0",
985 | "resolved": "https://registry.npmjs.org/is-ci/-/is-ci-2.0.0.tgz",
986 | "integrity": "sha512-YfJT7rkpQB0updsdHLGWrvhBJfcfzNNawYDNIyQXJz0IViGf75O8EBPKSdvw2rF+LGCsX4FZ8tcr3b19LcZq4w==",
987 | "dev": true,
988 | "requires": {
989 | "ci-info": "^2.0.0"
990 | }
991 | },
992 | "is-date-object": {
993 | "version": "1.0.2",
994 | "resolved": "https://registry.npmjs.org/is-date-object/-/is-date-object-1.0.2.tgz",
995 | "integrity": "sha512-USlDT524woQ08aoZFzh3/Z6ch9Y/EWXEHQ/AaRN0SkKq4t2Jw2R2339tSXmwuVoY7LLlBCbOIlx2myP/L5zk0g==",
996 | "dev": true
997 | },
998 | "is-extglob": {
999 | "version": "2.1.1",
1000 | "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz",
1001 | "integrity": "sha1-qIwCU1eR8C7TfHahueqXc8gz+MI=",
1002 | "dev": true
1003 | },
1004 | "is-fullwidth-code-point": {
1005 | "version": "2.0.0",
1006 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz",
1007 | "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=",
1008 | "dev": true
1009 | },
1010 | "is-glob": {
1011 | "version": "4.0.1",
1012 | "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.1.tgz",
1013 | "integrity": "sha512-5G0tKtBTFImOqDnLB2hG6Bp2qcKEFduo4tZu9MT/H6NQv/ghhy30o55ufafxJ/LdH79LLs2Kfrn85TLKyA7BUg==",
1014 | "dev": true,
1015 | "requires": {
1016 | "is-extglob": "^2.1.1"
1017 | }
1018 | },
1019 | "is-installed-globally": {
1020 | "version": "0.3.2",
1021 | "resolved": "https://registry.npmjs.org/is-installed-globally/-/is-installed-globally-0.3.2.tgz",
1022 | "integrity": "sha512-wZ8x1js7Ia0kecP/CHM/3ABkAmujX7WPvQk6uu3Fly/Mk44pySulQpnHG46OMjHGXApINnV4QhY3SWnECO2z5g==",
1023 | "dev": true,
1024 | "requires": {
1025 | "global-dirs": "^2.0.1",
1026 | "is-path-inside": "^3.0.1"
1027 | }
1028 | },
1029 | "is-npm": {
1030 | "version": "4.0.0",
1031 | "resolved": "https://registry.npmjs.org/is-npm/-/is-npm-4.0.0.tgz",
1032 | "integrity": "sha512-96ECIfh9xtDDlPylNPXhzjsykHsMJZ18ASpaWzQyBr4YRTcVjUvzaHayDAES2oU/3KpljhHUjtSRNiDwi0F0ig==",
1033 | "dev": true
1034 | },
1035 | "is-number": {
1036 | "version": "7.0.0",
1037 | "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz",
1038 | "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==",
1039 | "dev": true
1040 | },
1041 | "is-obj": {
1042 | "version": "2.0.0",
1043 | "resolved": "https://registry.npmjs.org/is-obj/-/is-obj-2.0.0.tgz",
1044 | "integrity": "sha512-drqDG3cbczxxEJRoOXcOjtdp1J/lyp1mNn0xaznRs8+muBhgQcrnbspox5X5fOw0HnMnbfDzvnEMEtqDEJEo8w==",
1045 | "dev": true
1046 | },
1047 | "is-path-inside": {
1048 | "version": "3.0.2",
1049 | "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-3.0.2.tgz",
1050 | "integrity": "sha512-/2UGPSgmtqwo1ktx8NDHjuPwZWmHhO+gj0f93EkhLB5RgW9RZevWYYlIkS6zePc6U2WpOdQYIwHe9YC4DWEBVg==",
1051 | "dev": true
1052 | },
1053 | "is-regex": {
1054 | "version": "1.0.5",
1055 | "resolved": "https://registry.npmjs.org/is-regex/-/is-regex-1.0.5.tgz",
1056 | "integrity": "sha512-vlKW17SNq44owv5AQR3Cq0bQPEb8+kF3UKZ2fiZNOWtztYE5i0CzCZxFDwO58qAOWtxdBRVO/V5Qin1wjCqFYQ==",
1057 | "dev": true,
1058 | "requires": {
1059 | "has": "^1.0.3"
1060 | }
1061 | },
1062 | "is-symbol": {
1063 | "version": "1.0.3",
1064 | "resolved": "https://registry.npmjs.org/is-symbol/-/is-symbol-1.0.3.tgz",
1065 | "integrity": "sha512-OwijhaRSgqvhm/0ZdAcXNZt9lYdKFpcRDT5ULUuYXPoT794UNOdU+gpT6Rzo7b4V2HUl/op6GqY894AZwv9faQ==",
1066 | "dev": true,
1067 | "requires": {
1068 | "has-symbols": "^1.0.1"
1069 | }
1070 | },
1071 | "is-typedarray": {
1072 | "version": "1.0.0",
1073 | "resolved": "https://registry.npmjs.org/is-typedarray/-/is-typedarray-1.0.0.tgz",
1074 | "integrity": "sha1-5HnICFjfDBsR3dppQPlgEfzaSpo=",
1075 | "dev": true
1076 | },
1077 | "is-yarn-global": {
1078 | "version": "0.3.0",
1079 | "resolved": "https://registry.npmjs.org/is-yarn-global/-/is-yarn-global-0.3.0.tgz",
1080 | "integrity": "sha512-VjSeb/lHmkoyd8ryPVIKvOCn4D1koMqY+vqyjjUfc3xyKtP4dYOxM44sZrnqQSzSds3xyOrUTLTC9LVCVgLngw==",
1081 | "dev": true
1082 | },
1083 | "isarray": {
1084 | "version": "1.0.0",
1085 | "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz",
1086 | "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE="
1087 | },
1088 | "isexe": {
1089 | "version": "2.0.0",
1090 | "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz",
1091 | "integrity": "sha1-6PvzdNxVb/iUehDcsFctYz8s+hA=",
1092 | "dev": true
1093 | },
1094 | "js-yaml": {
1095 | "version": "3.13.1",
1096 | "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.13.1.tgz",
1097 | "integrity": "sha512-YfbcO7jXDdyj0DGxYVSlSeQNHbD7XPWvrVWeVUujrQEoZzWJIRrCPoyk6kL6IAjAG2IolMK4T0hNUe0HOUs5Jw==",
1098 | "dev": true,
1099 | "requires": {
1100 | "argparse": "^1.0.7",
1101 | "esprima": "^4.0.0"
1102 | }
1103 | },
1104 | "json-buffer": {
1105 | "version": "3.0.0",
1106 | "resolved": "https://registry.npmjs.org/json-buffer/-/json-buffer-3.0.0.tgz",
1107 | "integrity": "sha1-Wx85evx11ne96Lz8Dkfh+aPZqJg=",
1108 | "dev": true
1109 | },
1110 | "kareem": {
1111 | "version": "2.3.1",
1112 | "resolved": "https://registry.npmjs.org/kareem/-/kareem-2.3.1.tgz",
1113 | "integrity": "sha512-l3hLhffs9zqoDe8zjmb/mAN4B8VT3L56EUvKNqLFVs9YlFA+zx7ke1DO8STAdDyYNkeSo1nKmjuvQeI12So8Xw=="
1114 | },
1115 | "keyv": {
1116 | "version": "3.1.0",
1117 | "resolved": "https://registry.npmjs.org/keyv/-/keyv-3.1.0.tgz",
1118 | "integrity": "sha512-9ykJ/46SN/9KPM/sichzQ7OvXyGDYKGTaDlKMGCAlg2UK8KRy4jb0d8sFc+0Tt0YYnThq8X2RZgCg74RPxgcVA==",
1119 | "dev": true,
1120 | "requires": {
1121 | "json-buffer": "3.0.0"
1122 | }
1123 | },
1124 | "latest-version": {
1125 | "version": "5.1.0",
1126 | "resolved": "https://registry.npmjs.org/latest-version/-/latest-version-5.1.0.tgz",
1127 | "integrity": "sha512-weT+r0kTkRQdCdYCNtkMwWXQTMEswKrFBkm4ckQOMVhhqhIMI1UT2hMj+1iigIhgSZm5gTmrRXBNoGUgaTY1xA==",
1128 | "dev": true,
1129 | "requires": {
1130 | "package-json": "^6.3.0"
1131 | }
1132 | },
1133 | "locate-path": {
1134 | "version": "3.0.0",
1135 | "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-3.0.0.tgz",
1136 | "integrity": "sha512-7AO748wWnIhNqAuaty2ZWHkQHRSNfPVIsPIfwEOWO22AmaoVrWavlOcMR5nzTLNYvp36X220/maaRsrec1G65A==",
1137 | "dev": true,
1138 | "requires": {
1139 | "p-locate": "^3.0.0",
1140 | "path-exists": "^3.0.0"
1141 | }
1142 | },
1143 | "lodash": {
1144 | "version": "4.17.15",
1145 | "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.15.tgz",
1146 | "integrity": "sha512-8xOcRHvCjnocdS5cpwXQXVzmmh5e5+saE2QGoeQmbKmRS6J3VQppPOIt0MnmE+4xlZoumy0GPG0D0MVIQbNA1A==",
1147 | "dev": true
1148 | },
1149 | "log-symbols": {
1150 | "version": "3.0.0",
1151 | "resolved": "https://registry.npmjs.org/log-symbols/-/log-symbols-3.0.0.tgz",
1152 | "integrity": "sha512-dSkNGuI7iG3mfvDzUuYZyvk5dD9ocYCYzNU6CYDE6+Xqd+gwme6Z00NS3dUh8mq/73HaEtT7m6W+yUPtU6BZnQ==",
1153 | "dev": true,
1154 | "requires": {
1155 | "chalk": "^2.4.2"
1156 | },
1157 | "dependencies": {
1158 | "ansi-styles": {
1159 | "version": "3.2.1",
1160 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz",
1161 | "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==",
1162 | "dev": true,
1163 | "requires": {
1164 | "color-convert": "^1.9.0"
1165 | }
1166 | },
1167 | "chalk": {
1168 | "version": "2.4.2",
1169 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz",
1170 | "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==",
1171 | "dev": true,
1172 | "requires": {
1173 | "ansi-styles": "^3.2.1",
1174 | "escape-string-regexp": "^1.0.5",
1175 | "supports-color": "^5.3.0"
1176 | }
1177 | },
1178 | "color-convert": {
1179 | "version": "1.9.3",
1180 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz",
1181 | "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==",
1182 | "dev": true,
1183 | "requires": {
1184 | "color-name": "1.1.3"
1185 | }
1186 | },
1187 | "color-name": {
1188 | "version": "1.1.3",
1189 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz",
1190 | "integrity": "sha1-p9BVi9icQveV3UIyj3QIMcpTvCU=",
1191 | "dev": true
1192 | }
1193 | }
1194 | },
1195 | "lowercase-keys": {
1196 | "version": "1.0.1",
1197 | "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-1.0.1.tgz",
1198 | "integrity": "sha512-G2Lj61tXDnVFFOi8VZds+SoQjtQC3dgokKdDG2mTm1tx4m50NUHBOZSBwQQHyy0V12A0JTG4icfZQH+xPyh8VA==",
1199 | "dev": true
1200 | },
1201 | "make-dir": {
1202 | "version": "3.1.0",
1203 | "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-3.1.0.tgz",
1204 | "integrity": "sha512-g3FeP20LNwhALb/6Cz6Dd4F2ngze0jz7tbzrD2wAV+o9FeNHe4rL+yK2md0J/fiSf1sa1ADhXqi5+oVwOM/eGw==",
1205 | "dev": true,
1206 | "requires": {
1207 | "semver": "^6.0.0"
1208 | },
1209 | "dependencies": {
1210 | "semver": {
1211 | "version": "6.3.0",
1212 | "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz",
1213 | "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==",
1214 | "dev": true
1215 | }
1216 | }
1217 | },
1218 | "media-typer": {
1219 | "version": "0.3.0",
1220 | "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz",
1221 | "integrity": "sha1-hxDXrwqmJvj/+hzgAWhUUmMlV0g="
1222 | },
1223 | "memory-pager": {
1224 | "version": "1.5.0",
1225 | "resolved": "https://registry.npmjs.org/memory-pager/-/memory-pager-1.5.0.tgz",
1226 | "integrity": "sha512-ZS4Bp4r/Zoeq6+NLJpP+0Zzm0pR8whtGPf1XExKLJBAczGMnSi3It14OiNCStjQjM6NU1okjQGSxgEZN8eBYKg==",
1227 | "optional": true
1228 | },
1229 | "merge-descriptors": {
1230 | "version": "1.0.1",
1231 | "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz",
1232 | "integrity": "sha1-sAqqVW3YtEVoFQ7J0blT8/kMu2E="
1233 | },
1234 | "methods": {
1235 | "version": "1.1.2",
1236 | "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz",
1237 | "integrity": "sha1-VSmk1nZUE07cxSZmVoNbD4Ua/O4="
1238 | },
1239 | "mime": {
1240 | "version": "1.6.0",
1241 | "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz",
1242 | "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg=="
1243 | },
1244 | "mime-db": {
1245 | "version": "1.44.0",
1246 | "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.44.0.tgz",
1247 | "integrity": "sha512-/NOTfLrsPBVeH7YtFPgsVWveuL+4SjjYxaQ1xtM1KMFj7HdxlBlxeyNLzhyJVx7r4rZGJAZ/6lkKCitSc/Nmpg=="
1248 | },
1249 | "mime-types": {
1250 | "version": "2.1.27",
1251 | "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.27.tgz",
1252 | "integrity": "sha512-JIhqnCasI9yD+SsmkquHBxTSEuZdQX5BuQnS2Vc7puQQQ+8yiP5AY5uWhpdv4YL4VM5c6iliiYWPgJ/nJQLp7w==",
1253 | "requires": {
1254 | "mime-db": "1.44.0"
1255 | }
1256 | },
1257 | "mimic-response": {
1258 | "version": "1.0.1",
1259 | "resolved": "https://registry.npmjs.org/mimic-response/-/mimic-response-1.0.1.tgz",
1260 | "integrity": "sha512-j5EctnkH7amfV/q5Hgmoal1g2QHFJRraOtmx0JpIqkxhBhI/lJSl1nMpQ45hVarwNETOoWEimndZ4QK0RHxuxQ==",
1261 | "dev": true
1262 | },
1263 | "minimatch": {
1264 | "version": "3.0.4",
1265 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz",
1266 | "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==",
1267 | "dev": true,
1268 | "requires": {
1269 | "brace-expansion": "^1.1.7"
1270 | }
1271 | },
1272 | "minimist": {
1273 | "version": "1.2.5",
1274 | "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.5.tgz",
1275 | "integrity": "sha512-FM9nNUYrRBAELZQT3xeZQ7fmMOBg6nWNmJKTcgsJeaLstP/UODVpGsr5OhXhhXg6f+qtJ8uiZ+PUxkDWcgIXLw==",
1276 | "dev": true
1277 | },
1278 | "mkdirp": {
1279 | "version": "0.5.5",
1280 | "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.5.tgz",
1281 | "integrity": "sha512-NKmAlESf6jMGym1++R0Ra7wvhV+wFW63FaSOFPwRahvea0gMUcGUhVeAg/0BC0wiv9ih5NYPB1Wn1UEI1/L+xQ==",
1282 | "dev": true,
1283 | "requires": {
1284 | "minimist": "^1.2.5"
1285 | }
1286 | },
1287 | "mocha": {
1288 | "version": "7.1.2",
1289 | "resolved": "https://registry.npmjs.org/mocha/-/mocha-7.1.2.tgz",
1290 | "integrity": "sha512-o96kdRKMKI3E8U0bjnfqW4QMk12MwZ4mhdBTf+B5a1q9+aq2HRnj+3ZdJu0B/ZhJeK78MgYuv6L8d/rA5AeBJA==",
1291 | "dev": true,
1292 | "requires": {
1293 | "ansi-colors": "3.2.3",
1294 | "browser-stdout": "1.3.1",
1295 | "chokidar": "3.3.0",
1296 | "debug": "3.2.6",
1297 | "diff": "3.5.0",
1298 | "escape-string-regexp": "1.0.5",
1299 | "find-up": "3.0.0",
1300 | "glob": "7.1.3",
1301 | "growl": "1.10.5",
1302 | "he": "1.2.0",
1303 | "js-yaml": "3.13.1",
1304 | "log-symbols": "3.0.0",
1305 | "minimatch": "3.0.4",
1306 | "mkdirp": "0.5.5",
1307 | "ms": "2.1.1",
1308 | "node-environment-flags": "1.0.6",
1309 | "object.assign": "4.1.0",
1310 | "strip-json-comments": "2.0.1",
1311 | "supports-color": "6.0.0",
1312 | "which": "1.3.1",
1313 | "wide-align": "1.1.3",
1314 | "yargs": "13.3.2",
1315 | "yargs-parser": "13.1.2",
1316 | "yargs-unparser": "1.6.0"
1317 | },
1318 | "dependencies": {
1319 | "chokidar": {
1320 | "version": "3.3.0",
1321 | "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.3.0.tgz",
1322 | "integrity": "sha512-dGmKLDdT3Gdl7fBUe8XK+gAtGmzy5Fn0XkkWQuYxGIgWVPPse2CxFA5mtrlD0TOHaHjEUqkWNyP1XdHoJES/4A==",
1323 | "dev": true,
1324 | "requires": {
1325 | "anymatch": "~3.1.1",
1326 | "braces": "~3.0.2",
1327 | "fsevents": "~2.1.1",
1328 | "glob-parent": "~5.1.0",
1329 | "is-binary-path": "~2.1.0",
1330 | "is-glob": "~4.0.1",
1331 | "normalize-path": "~3.0.0",
1332 | "readdirp": "~3.2.0"
1333 | }
1334 | },
1335 | "debug": {
1336 | "version": "3.2.6",
1337 | "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.6.tgz",
1338 | "integrity": "sha512-mel+jf7nrtEl5Pn1Qx46zARXKDpBbvzezse7p7LqINmdoIk8PYP5SySaxEmYv6TZ0JyEKA1hsCId6DIhgITtWQ==",
1339 | "dev": true,
1340 | "requires": {
1341 | "ms": "^2.1.1"
1342 | }
1343 | },
1344 | "ms": {
1345 | "version": "2.1.1",
1346 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.1.tgz",
1347 | "integrity": "sha512-tgp+dl5cGk28utYktBsrFqA7HKgrhgPsg6Z/EfhWI4gl1Hwq8B/GmY/0oXZ6nF8hDVesS/FpnYaD/kOWhYQvyg==",
1348 | "dev": true
1349 | },
1350 | "readdirp": {
1351 | "version": "3.2.0",
1352 | "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.2.0.tgz",
1353 | "integrity": "sha512-crk4Qu3pmXwgxdSgGhgA/eXiJAPQiX4GMOZZMXnqKxHX7TaoL+3gQVo/WeuAiogr07DpnfjIMpXXa+PAIvwPGQ==",
1354 | "dev": true,
1355 | "requires": {
1356 | "picomatch": "^2.0.4"
1357 | }
1358 | },
1359 | "supports-color": {
1360 | "version": "6.0.0",
1361 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-6.0.0.tgz",
1362 | "integrity": "sha512-on9Kwidc1IUQo+bQdhi8+Tijpo0e1SS6RoGo2guUwn5vdaxw8RXOF9Vb2ws+ihWOmh4JnCJOvaziZWP1VABaLg==",
1363 | "dev": true,
1364 | "requires": {
1365 | "has-flag": "^3.0.0"
1366 | }
1367 | }
1368 | }
1369 | },
1370 | "mongodb": {
1371 | "version": "3.5.7",
1372 | "resolved": "https://registry.npmjs.org/mongodb/-/mongodb-3.5.7.tgz",
1373 | "integrity": "sha512-lMtleRT+vIgY/JhhTn1nyGwnSMmJkJELp+4ZbrjctrnBxuLbj6rmLuJFz8W2xUzUqWmqoyVxJLYuC58ZKpcTYQ==",
1374 | "requires": {
1375 | "bl": "^2.2.0",
1376 | "bson": "^1.1.4",
1377 | "denque": "^1.4.1",
1378 | "require_optional": "^1.0.1",
1379 | "safe-buffer": "^5.1.2",
1380 | "saslprep": "^1.0.0"
1381 | }
1382 | },
1383 | "mongoose": {
1384 | "version": "5.9.14",
1385 | "resolved": "https://registry.npmjs.org/mongoose/-/mongoose-5.9.14.tgz",
1386 | "integrity": "sha512-LScxCruQv0YpU/9DasKdThd+3r3PFQbCgesmfa6g0pTDOIiD1A9N9OQsGYrDf+dyUksfLCxJYYF9qpBHLvS1tg==",
1387 | "requires": {
1388 | "bson": "^1.1.4",
1389 | "kareem": "2.3.1",
1390 | "mongodb": "3.5.7",
1391 | "mongoose-legacy-pluralize": "1.0.2",
1392 | "mpath": "0.7.0",
1393 | "mquery": "3.2.2",
1394 | "ms": "2.1.2",
1395 | "regexp-clone": "1.0.0",
1396 | "safe-buffer": "5.1.2",
1397 | "sift": "7.0.1",
1398 | "sliced": "1.0.1"
1399 | },
1400 | "dependencies": {
1401 | "ms": {
1402 | "version": "2.1.2",
1403 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz",
1404 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w=="
1405 | }
1406 | }
1407 | },
1408 | "mongoose-legacy-pluralize": {
1409 | "version": "1.0.2",
1410 | "resolved": "https://registry.npmjs.org/mongoose-legacy-pluralize/-/mongoose-legacy-pluralize-1.0.2.tgz",
1411 | "integrity": "sha512-Yo/7qQU4/EyIS8YDFSeenIvXxZN+ld7YdV9LqFVQJzTLye8unujAWPZ4NWKfFA+RNjh+wvTWKY9Z3E5XM6ZZiQ=="
1412 | },
1413 | "mpath": {
1414 | "version": "0.7.0",
1415 | "resolved": "https://registry.npmjs.org/mpath/-/mpath-0.7.0.tgz",
1416 | "integrity": "sha512-Aiq04hILxhz1L+f7sjGyn7IxYzWm1zLNNXcfhDtx04kZ2Gk7uvFdgZ8ts1cWa/6d0TQmag2yR8zSGZUmp0tFNg=="
1417 | },
1418 | "mquery": {
1419 | "version": "3.2.2",
1420 | "resolved": "https://registry.npmjs.org/mquery/-/mquery-3.2.2.tgz",
1421 | "integrity": "sha512-XB52992COp0KP230I3qloVUbkLUxJIu328HBP2t2EsxSFtf4W1HPSOBWOXf1bqxK4Xbb66lfMJ+Bpfd9/yZE1Q==",
1422 | "requires": {
1423 | "bluebird": "3.5.1",
1424 | "debug": "3.1.0",
1425 | "regexp-clone": "^1.0.0",
1426 | "safe-buffer": "5.1.2",
1427 | "sliced": "1.0.1"
1428 | },
1429 | "dependencies": {
1430 | "debug": {
1431 | "version": "3.1.0",
1432 | "resolved": "https://registry.npmjs.org/debug/-/debug-3.1.0.tgz",
1433 | "integrity": "sha512-OX8XqP7/1a9cqkxYw2yXss15f26NKWBpDXQd0/uK/KPqdQhxbPa994hnzjcE2VqQpDslf55723cKPUOGSmMY3g==",
1434 | "requires": {
1435 | "ms": "2.0.0"
1436 | }
1437 | }
1438 | }
1439 | },
1440 | "ms": {
1441 | "version": "2.0.0",
1442 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz",
1443 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g="
1444 | },
1445 | "negotiator": {
1446 | "version": "0.6.2",
1447 | "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.2.tgz",
1448 | "integrity": "sha512-hZXc7K2e+PgeI1eDBe/10Ard4ekbfrrqG8Ep+8Jmf4JID2bNg7NvCPOZN+kfF574pFQI7mum2AUqDidoKqcTOw=="
1449 | },
1450 | "node-environment-flags": {
1451 | "version": "1.0.6",
1452 | "resolved": "https://registry.npmjs.org/node-environment-flags/-/node-environment-flags-1.0.6.tgz",
1453 | "integrity": "sha512-5Evy2epuL+6TM0lCQGpFIj6KwiEsGh1SrHUhTbNX+sLbBtjidPZFAnVK9y5yU1+h//RitLbRHTIMyxQPtxMdHw==",
1454 | "dev": true,
1455 | "requires": {
1456 | "object.getownpropertydescriptors": "^2.0.3",
1457 | "semver": "^5.7.0"
1458 | }
1459 | },
1460 | "nodemon": {
1461 | "version": "2.0.4",
1462 | "resolved": "https://registry.npmjs.org/nodemon/-/nodemon-2.0.4.tgz",
1463 | "integrity": "sha512-Ltced+hIfTmaS28Zjv1BM552oQ3dbwPqI4+zI0SLgq+wpJhSyqgYude/aZa/3i31VCQWMfXJVxvu86abcam3uQ==",
1464 | "dev": true,
1465 | "requires": {
1466 | "chokidar": "^3.2.2",
1467 | "debug": "^3.2.6",
1468 | "ignore-by-default": "^1.0.1",
1469 | "minimatch": "^3.0.4",
1470 | "pstree.remy": "^1.1.7",
1471 | "semver": "^5.7.1",
1472 | "supports-color": "^5.5.0",
1473 | "touch": "^3.1.0",
1474 | "undefsafe": "^2.0.2",
1475 | "update-notifier": "^4.0.0"
1476 | },
1477 | "dependencies": {
1478 | "debug": {
1479 | "version": "3.2.6",
1480 | "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.6.tgz",
1481 | "integrity": "sha512-mel+jf7nrtEl5Pn1Qx46zARXKDpBbvzezse7p7LqINmdoIk8PYP5SySaxEmYv6TZ0JyEKA1hsCId6DIhgITtWQ==",
1482 | "dev": true,
1483 | "requires": {
1484 | "ms": "^2.1.1"
1485 | }
1486 | },
1487 | "ms": {
1488 | "version": "2.1.2",
1489 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz",
1490 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==",
1491 | "dev": true
1492 | }
1493 | }
1494 | },
1495 | "nopt": {
1496 | "version": "1.0.10",
1497 | "resolved": "https://registry.npmjs.org/nopt/-/nopt-1.0.10.tgz",
1498 | "integrity": "sha1-bd0hvSoxQXuScn3Vhfim83YI6+4=",
1499 | "dev": true,
1500 | "requires": {
1501 | "abbrev": "1"
1502 | }
1503 | },
1504 | "normalize-path": {
1505 | "version": "3.0.0",
1506 | "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz",
1507 | "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==",
1508 | "dev": true
1509 | },
1510 | "normalize-url": {
1511 | "version": "4.5.0",
1512 | "resolved": "https://registry.npmjs.org/normalize-url/-/normalize-url-4.5.0.tgz",
1513 | "integrity": "sha512-2s47yzUxdexf1OhyRi4Em83iQk0aPvwTddtFz4hnSSw9dCEsLEGf6SwIO8ss/19S9iBb5sJaOuTvTGDeZI00BQ==",
1514 | "dev": true
1515 | },
1516 | "object-assign": {
1517 | "version": "4.1.1",
1518 | "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz",
1519 | "integrity": "sha1-IQmtx5ZYh8/AXLvUQsrIv7s2CGM="
1520 | },
1521 | "object-inspect": {
1522 | "version": "1.7.0",
1523 | "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.7.0.tgz",
1524 | "integrity": "sha512-a7pEHdh1xKIAgTySUGgLMx/xwDZskN1Ud6egYYN3EdRW4ZMPNEDUTF+hwy2LUC+Bl+SyLXANnwz/jyh/qutKUw==",
1525 | "dev": true
1526 | },
1527 | "object-keys": {
1528 | "version": "1.1.1",
1529 | "resolved": "https://registry.npmjs.org/object-keys/-/object-keys-1.1.1.tgz",
1530 | "integrity": "sha512-NuAESUOUMrlIXOfHKzD6bpPu3tYt3xvjNdRIQ+FeT0lNb4K8WR70CaDxhuNguS2XG+GjkyMwOzsN5ZktImfhLA==",
1531 | "dev": true
1532 | },
1533 | "object.assign": {
1534 | "version": "4.1.0",
1535 | "resolved": "https://registry.npmjs.org/object.assign/-/object.assign-4.1.0.tgz",
1536 | "integrity": "sha512-exHJeq6kBKj58mqGyTQ9DFvrZC/eR6OwxzoM9YRoGBqrXYonaFyGiFMuc9VZrXf7DarreEwMpurG3dd+CNyW5w==",
1537 | "dev": true,
1538 | "requires": {
1539 | "define-properties": "^1.1.2",
1540 | "function-bind": "^1.1.1",
1541 | "has-symbols": "^1.0.0",
1542 | "object-keys": "^1.0.11"
1543 | }
1544 | },
1545 | "object.getownpropertydescriptors": {
1546 | "version": "2.1.0",
1547 | "resolved": "https://registry.npmjs.org/object.getownpropertydescriptors/-/object.getownpropertydescriptors-2.1.0.tgz",
1548 | "integrity": "sha512-Z53Oah9A3TdLoblT7VKJaTDdXdT+lQO+cNpKVnya5JDe9uLvzu1YyY1yFDFrcxrlRgWrEFH0jJtD/IbuwjcEVg==",
1549 | "dev": true,
1550 | "requires": {
1551 | "define-properties": "^1.1.3",
1552 | "es-abstract": "^1.17.0-next.1"
1553 | }
1554 | },
1555 | "on-finished": {
1556 | "version": "2.3.0",
1557 | "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.3.0.tgz",
1558 | "integrity": "sha1-IPEzZIGwg811M3mSoWlxqi2QaUc=",
1559 | "requires": {
1560 | "ee-first": "1.1.1"
1561 | }
1562 | },
1563 | "once": {
1564 | "version": "1.4.0",
1565 | "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz",
1566 | "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=",
1567 | "dev": true,
1568 | "requires": {
1569 | "wrappy": "1"
1570 | }
1571 | },
1572 | "p-cancelable": {
1573 | "version": "1.1.0",
1574 | "resolved": "https://registry.npmjs.org/p-cancelable/-/p-cancelable-1.1.0.tgz",
1575 | "integrity": "sha512-s73XxOZ4zpt1edZYZzvhqFa6uvQc1vwUa0K0BdtIZgQMAJj9IbebH+JkgKZc9h+B05PKHLOTl4ajG1BmNrVZlw==",
1576 | "dev": true
1577 | },
1578 | "p-limit": {
1579 | "version": "2.3.0",
1580 | "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-2.3.0.tgz",
1581 | "integrity": "sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w==",
1582 | "dev": true,
1583 | "requires": {
1584 | "p-try": "^2.0.0"
1585 | }
1586 | },
1587 | "p-locate": {
1588 | "version": "3.0.0",
1589 | "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-3.0.0.tgz",
1590 | "integrity": "sha512-x+12w/To+4GFfgJhBEpiDcLozRJGegY+Ei7/z0tSLkMmxGZNybVMSfWj9aJn8Z5Fc7dBUNJOOVgPv2H7IwulSQ==",
1591 | "dev": true,
1592 | "requires": {
1593 | "p-limit": "^2.0.0"
1594 | }
1595 | },
1596 | "p-try": {
1597 | "version": "2.2.0",
1598 | "resolved": "https://registry.npmjs.org/p-try/-/p-try-2.2.0.tgz",
1599 | "integrity": "sha512-R4nPAVTAU0B9D35/Gk3uJf/7XYbQcyohSKdvAxIRSNghFl4e71hVoGnBNQz9cWaXxO2I10KTC+3jMdvvoKw6dQ==",
1600 | "dev": true
1601 | },
1602 | "package-json": {
1603 | "version": "6.5.0",
1604 | "resolved": "https://registry.npmjs.org/package-json/-/package-json-6.5.0.tgz",
1605 | "integrity": "sha512-k3bdm2n25tkyxcjSKzB5x8kfVxlMdgsbPr0GkZcwHsLpba6cBjqCt1KlcChKEvxHIcTB1FVMuwoijZ26xex5MQ==",
1606 | "dev": true,
1607 | "requires": {
1608 | "got": "^9.6.0",
1609 | "registry-auth-token": "^4.0.0",
1610 | "registry-url": "^5.0.0",
1611 | "semver": "^6.2.0"
1612 | },
1613 | "dependencies": {
1614 | "semver": {
1615 | "version": "6.3.0",
1616 | "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz",
1617 | "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==",
1618 | "dev": true
1619 | }
1620 | }
1621 | },
1622 | "parseurl": {
1623 | "version": "1.3.3",
1624 | "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz",
1625 | "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ=="
1626 | },
1627 | "path-exists": {
1628 | "version": "3.0.0",
1629 | "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-3.0.0.tgz",
1630 | "integrity": "sha1-zg6+ql94yxiSXqfYENe1mwEP1RU=",
1631 | "dev": true
1632 | },
1633 | "path-is-absolute": {
1634 | "version": "1.0.1",
1635 | "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz",
1636 | "integrity": "sha1-F0uSaHNVNP+8es5r9TpanhtcX18=",
1637 | "dev": true
1638 | },
1639 | "path-to-regexp": {
1640 | "version": "0.1.7",
1641 | "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz",
1642 | "integrity": "sha1-32BBeABfUi8V60SQ5yR6G/qmf4w="
1643 | },
1644 | "pathval": {
1645 | "version": "1.1.0",
1646 | "resolved": "https://registry.npmjs.org/pathval/-/pathval-1.1.0.tgz",
1647 | "integrity": "sha1-uULm1L3mUwBe9rcTYd74cn0GReA=",
1648 | "dev": true
1649 | },
1650 | "picomatch": {
1651 | "version": "2.2.2",
1652 | "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.2.2.tgz",
1653 | "integrity": "sha512-q0M/9eZHzmr0AulXyPwNfZjtwZ/RBZlbN3K3CErVrk50T2ASYI7Bye0EvekFY3IP1Nt2DHu0re+V2ZHIpMkuWg==",
1654 | "dev": true
1655 | },
1656 | "prepend-http": {
1657 | "version": "2.0.0",
1658 | "resolved": "https://registry.npmjs.org/prepend-http/-/prepend-http-2.0.0.tgz",
1659 | "integrity": "sha1-6SQ0v6XqjBn0HN/UAddBo8gZ2Jc=",
1660 | "dev": true
1661 | },
1662 | "process-nextick-args": {
1663 | "version": "2.0.1",
1664 | "resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-2.0.1.tgz",
1665 | "integrity": "sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag=="
1666 | },
1667 | "proxy-addr": {
1668 | "version": "2.0.6",
1669 | "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.6.tgz",
1670 | "integrity": "sha512-dh/frvCBVmSsDYzw6n926jv974gddhkFPfiN8hPOi30Wax25QZyZEGveluCgliBnqmuM+UJmBErbAUFIoDbjOw==",
1671 | "requires": {
1672 | "forwarded": "~0.1.2",
1673 | "ipaddr.js": "1.9.1"
1674 | }
1675 | },
1676 | "pstree.remy": {
1677 | "version": "1.1.8",
1678 | "resolved": "https://registry.npmjs.org/pstree.remy/-/pstree.remy-1.1.8.tgz",
1679 | "integrity": "sha512-77DZwxQmxKnu3aR542U+X8FypNzbfJ+C5XQDk3uWjWxn6151aIMGthWYRXTqT1E5oJvg+ljaa2OJi+VfvCOQ8w==",
1680 | "dev": true
1681 | },
1682 | "pump": {
1683 | "version": "3.0.0",
1684 | "resolved": "https://registry.npmjs.org/pump/-/pump-3.0.0.tgz",
1685 | "integrity": "sha512-LwZy+p3SFs1Pytd/jYct4wpv49HiYCqd9Rlc5ZVdk0V+8Yzv6jR5Blk3TRmPL1ft69TxP0IMZGJ+WPFU2BFhww==",
1686 | "dev": true,
1687 | "requires": {
1688 | "end-of-stream": "^1.1.0",
1689 | "once": "^1.3.1"
1690 | }
1691 | },
1692 | "pupa": {
1693 | "version": "2.0.1",
1694 | "resolved": "https://registry.npmjs.org/pupa/-/pupa-2.0.1.tgz",
1695 | "integrity": "sha512-hEJH0s8PXLY/cdXh66tNEQGndDrIKNqNC5xmrysZy3i5C3oEoLna7YAOad+7u125+zH1HNXUmGEkrhb3c2VriA==",
1696 | "dev": true,
1697 | "requires": {
1698 | "escape-goat": "^2.0.0"
1699 | }
1700 | },
1701 | "qs": {
1702 | "version": "6.7.0",
1703 | "resolved": "https://registry.npmjs.org/qs/-/qs-6.7.0.tgz",
1704 | "integrity": "sha512-VCdBRNFTX1fyE7Nb6FYoURo/SPe62QCaAyzJvUjwRaIsc+NePBEniHlvxFmmX56+HZphIGtV0XeCirBtpDrTyQ=="
1705 | },
1706 | "range-parser": {
1707 | "version": "1.2.1",
1708 | "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz",
1709 | "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg=="
1710 | },
1711 | "raw-body": {
1712 | "version": "2.4.0",
1713 | "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.4.0.tgz",
1714 | "integrity": "sha512-4Oz8DUIwdvoa5qMJelxipzi/iJIi40O5cGV1wNYp5hvZP8ZN0T+jiNkL0QepXs+EsQ9XJ8ipEDoiH70ySUJP3Q==",
1715 | "requires": {
1716 | "bytes": "3.1.0",
1717 | "http-errors": "1.7.2",
1718 | "iconv-lite": "0.4.24",
1719 | "unpipe": "1.0.0"
1720 | }
1721 | },
1722 | "rc": {
1723 | "version": "1.2.8",
1724 | "resolved": "https://registry.npmjs.org/rc/-/rc-1.2.8.tgz",
1725 | "integrity": "sha512-y3bGgqKj3QBdxLbLkomlohkvsA8gdAiUQlSBJnBhfn+BPxg4bc62d8TcBW15wavDfgexCgccckhcZvywyQYPOw==",
1726 | "dev": true,
1727 | "requires": {
1728 | "deep-extend": "^0.6.0",
1729 | "ini": "~1.3.0",
1730 | "minimist": "^1.2.0",
1731 | "strip-json-comments": "~2.0.1"
1732 | }
1733 | },
1734 | "readable-stream": {
1735 | "version": "2.3.7",
1736 | "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.7.tgz",
1737 | "integrity": "sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw==",
1738 | "requires": {
1739 | "core-util-is": "~1.0.0",
1740 | "inherits": "~2.0.3",
1741 | "isarray": "~1.0.0",
1742 | "process-nextick-args": "~2.0.0",
1743 | "safe-buffer": "~5.1.1",
1744 | "string_decoder": "~1.1.1",
1745 | "util-deprecate": "~1.0.1"
1746 | }
1747 | },
1748 | "readdirp": {
1749 | "version": "3.4.0",
1750 | "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.4.0.tgz",
1751 | "integrity": "sha512-0xe001vZBnJEK+uKcj8qOhyAKPzIT+gStxWr3LCB0DwcXR5NZJ3IaC+yGnHCYzB/S7ov3m3EEbZI2zeNvX+hGQ==",
1752 | "dev": true,
1753 | "requires": {
1754 | "picomatch": "^2.2.1"
1755 | }
1756 | },
1757 | "regexp-clone": {
1758 | "version": "1.0.0",
1759 | "resolved": "https://registry.npmjs.org/regexp-clone/-/regexp-clone-1.0.0.tgz",
1760 | "integrity": "sha512-TuAasHQNamyyJ2hb97IuBEif4qBHGjPHBS64sZwytpLEqtBQ1gPJTnOaQ6qmpET16cK14kkjbazl6+p0RRv0yw=="
1761 | },
1762 | "registry-auth-token": {
1763 | "version": "4.1.1",
1764 | "resolved": "https://registry.npmjs.org/registry-auth-token/-/registry-auth-token-4.1.1.tgz",
1765 | "integrity": "sha512-9bKS7nTl9+/A1s7tnPeGrUpRcVY+LUh7bfFgzpndALdPfXQBfQV77rQVtqgUV3ti4vc/Ik81Ex8UJDWDQ12zQA==",
1766 | "dev": true,
1767 | "requires": {
1768 | "rc": "^1.2.8"
1769 | }
1770 | },
1771 | "registry-url": {
1772 | "version": "5.1.0",
1773 | "resolved": "https://registry.npmjs.org/registry-url/-/registry-url-5.1.0.tgz",
1774 | "integrity": "sha512-8acYXXTI0AkQv6RAOjE3vOaIXZkT9wo4LOFbBKYQEEnnMNBpKqdUrI6S4NT0KPIo/WVvJ5tE/X5LF/TQUf0ekw==",
1775 | "dev": true,
1776 | "requires": {
1777 | "rc": "^1.2.8"
1778 | }
1779 | },
1780 | "require-directory": {
1781 | "version": "2.1.1",
1782 | "resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz",
1783 | "integrity": "sha1-jGStX9MNqxyXbiNE/+f3kqam30I=",
1784 | "dev": true
1785 | },
1786 | "require-main-filename": {
1787 | "version": "2.0.0",
1788 | "resolved": "https://registry.npmjs.org/require-main-filename/-/require-main-filename-2.0.0.tgz",
1789 | "integrity": "sha512-NKN5kMDylKuldxYLSUfrbo5Tuzh4hd+2E8NPPX02mZtn1VuREQToYe/ZdlJy+J3uCpfaiGF05e7B8W0iXbQHmg==",
1790 | "dev": true
1791 | },
1792 | "require_optional": {
1793 | "version": "1.0.1",
1794 | "resolved": "https://registry.npmjs.org/require_optional/-/require_optional-1.0.1.tgz",
1795 | "integrity": "sha512-qhM/y57enGWHAe3v/NcwML6a3/vfESLe/sGM2dII+gEO0BpKRUkWZow/tyloNqJyN6kXSl3RyyM8Ll5D/sJP8g==",
1796 | "requires": {
1797 | "resolve-from": "^2.0.0",
1798 | "semver": "^5.1.0"
1799 | }
1800 | },
1801 | "resolve-from": {
1802 | "version": "2.0.0",
1803 | "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-2.0.0.tgz",
1804 | "integrity": "sha1-lICrIOlP+h2egKgEx+oUdhGWa1c="
1805 | },
1806 | "responselike": {
1807 | "version": "1.0.2",
1808 | "resolved": "https://registry.npmjs.org/responselike/-/responselike-1.0.2.tgz",
1809 | "integrity": "sha1-kYcg7ztjHFZCvgaPFa3lpG9Loec=",
1810 | "dev": true,
1811 | "requires": {
1812 | "lowercase-keys": "^1.0.0"
1813 | }
1814 | },
1815 | "safe-buffer": {
1816 | "version": "5.1.2",
1817 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz",
1818 | "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g=="
1819 | },
1820 | "safer-buffer": {
1821 | "version": "2.1.2",
1822 | "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz",
1823 | "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg=="
1824 | },
1825 | "saslprep": {
1826 | "version": "1.0.3",
1827 | "resolved": "https://registry.npmjs.org/saslprep/-/saslprep-1.0.3.tgz",
1828 | "integrity": "sha512-/MY/PEMbk2SuY5sScONwhUDsV2p77Znkb/q3nSVstq/yQzYJOH/Azh29p9oJLsl3LnQwSvZDKagDGBsBwSooag==",
1829 | "optional": true,
1830 | "requires": {
1831 | "sparse-bitfield": "^3.0.3"
1832 | }
1833 | },
1834 | "semver": {
1835 | "version": "5.7.1",
1836 | "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz",
1837 | "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ=="
1838 | },
1839 | "semver-diff": {
1840 | "version": "3.1.1",
1841 | "resolved": "https://registry.npmjs.org/semver-diff/-/semver-diff-3.1.1.tgz",
1842 | "integrity": "sha512-GX0Ix/CJcHyB8c4ykpHGIAvLyOwOobtM/8d+TQkAd81/bEjgPHrfba41Vpesr7jX/t8Uh+R3EX9eAS5be+jQYg==",
1843 | "dev": true,
1844 | "requires": {
1845 | "semver": "^6.3.0"
1846 | },
1847 | "dependencies": {
1848 | "semver": {
1849 | "version": "6.3.0",
1850 | "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz",
1851 | "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==",
1852 | "dev": true
1853 | }
1854 | }
1855 | },
1856 | "send": {
1857 | "version": "0.17.1",
1858 | "resolved": "https://registry.npmjs.org/send/-/send-0.17.1.tgz",
1859 | "integrity": "sha512-BsVKsiGcQMFwT8UxypobUKyv7irCNRHk1T0G680vk88yf6LBByGcZJOTJCrTP2xVN6yI+XjPJcNuE3V4fT9sAg==",
1860 | "requires": {
1861 | "debug": "2.6.9",
1862 | "depd": "~1.1.2",
1863 | "destroy": "~1.0.4",
1864 | "encodeurl": "~1.0.2",
1865 | "escape-html": "~1.0.3",
1866 | "etag": "~1.8.1",
1867 | "fresh": "0.5.2",
1868 | "http-errors": "~1.7.2",
1869 | "mime": "1.6.0",
1870 | "ms": "2.1.1",
1871 | "on-finished": "~2.3.0",
1872 | "range-parser": "~1.2.1",
1873 | "statuses": "~1.5.0"
1874 | },
1875 | "dependencies": {
1876 | "ms": {
1877 | "version": "2.1.1",
1878 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.1.tgz",
1879 | "integrity": "sha512-tgp+dl5cGk28utYktBsrFqA7HKgrhgPsg6Z/EfhWI4gl1Hwq8B/GmY/0oXZ6nF8hDVesS/FpnYaD/kOWhYQvyg=="
1880 | }
1881 | }
1882 | },
1883 | "serve-static": {
1884 | "version": "1.14.1",
1885 | "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.14.1.tgz",
1886 | "integrity": "sha512-JMrvUwE54emCYWlTI+hGrGv5I8dEwmco/00EvkzIIsR7MqrHonbD9pO2MOfFnpFntl7ecpZs+3mW+XbQZu9QCg==",
1887 | "requires": {
1888 | "encodeurl": "~1.0.2",
1889 | "escape-html": "~1.0.3",
1890 | "parseurl": "~1.3.3",
1891 | "send": "0.17.1"
1892 | }
1893 | },
1894 | "set-blocking": {
1895 | "version": "2.0.0",
1896 | "resolved": "https://registry.npmjs.org/set-blocking/-/set-blocking-2.0.0.tgz",
1897 | "integrity": "sha1-BF+XgtARrppoA93TgrJDkrPYkPc=",
1898 | "dev": true
1899 | },
1900 | "setprototypeof": {
1901 | "version": "1.1.1",
1902 | "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.1.tgz",
1903 | "integrity": "sha512-JvdAWfbXeIGaZ9cILp38HntZSFSo3mWg6xGcJJsd+d4aRMOqauag1C63dJfDw7OaMYwEbHMOxEZ1lqVRYP2OAw=="
1904 | },
1905 | "sift": {
1906 | "version": "7.0.1",
1907 | "resolved": "https://registry.npmjs.org/sift/-/sift-7.0.1.tgz",
1908 | "integrity": "sha512-oqD7PMJ+uO6jV9EQCl0LrRw1OwsiPsiFQR5AR30heR+4Dl7jBBbDLnNvWiak20tzZlSE1H7RB30SX/1j/YYT7g=="
1909 | },
1910 | "signal-exit": {
1911 | "version": "3.0.3",
1912 | "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.3.tgz",
1913 | "integrity": "sha512-VUJ49FC8U1OxwZLxIbTTrDvLnf/6TDgxZcK8wxR8zs13xpx7xbG60ndBlhNrFi2EMuFRoeDoJO7wthSLq42EjA==",
1914 | "dev": true
1915 | },
1916 | "sliced": {
1917 | "version": "1.0.1",
1918 | "resolved": "https://registry.npmjs.org/sliced/-/sliced-1.0.1.tgz",
1919 | "integrity": "sha1-CzpmK10Ewxd7GSa+qCsD+Dei70E="
1920 | },
1921 | "sparse-bitfield": {
1922 | "version": "3.0.3",
1923 | "resolved": "https://registry.npmjs.org/sparse-bitfield/-/sparse-bitfield-3.0.3.tgz",
1924 | "integrity": "sha1-/0rm5oZWBWuks+eSqzM004JzyhE=",
1925 | "optional": true,
1926 | "requires": {
1927 | "memory-pager": "^1.0.2"
1928 | }
1929 | },
1930 | "sprintf-js": {
1931 | "version": "1.0.3",
1932 | "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.0.3.tgz",
1933 | "integrity": "sha1-BOaSb2YolTVPPdAVIDYzuFcpfiw=",
1934 | "dev": true
1935 | },
1936 | "statuses": {
1937 | "version": "1.5.0",
1938 | "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.5.0.tgz",
1939 | "integrity": "sha1-Fhx9rBd2Wf2YEfQ3cfqZOBR4Yow="
1940 | },
1941 | "string-width": {
1942 | "version": "4.2.0",
1943 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.0.tgz",
1944 | "integrity": "sha512-zUz5JD+tgqtuDjMhwIg5uFVV3dtqZ9yQJlZVfq4I01/K5Paj5UHj7VyrQOJvzawSVlKpObApbfD0Ed6yJc+1eg==",
1945 | "dev": true,
1946 | "requires": {
1947 | "emoji-regex": "^8.0.0",
1948 | "is-fullwidth-code-point": "^3.0.0",
1949 | "strip-ansi": "^6.0.0"
1950 | },
1951 | "dependencies": {
1952 | "ansi-regex": {
1953 | "version": "5.0.0",
1954 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.0.tgz",
1955 | "integrity": "sha512-bY6fj56OUQ0hU1KjFNDQuJFezqKdrAyFdIevADiqrWHwSlbmBNMHp5ak2f40Pm8JTFyM2mqxkG6ngkHO11f/lg==",
1956 | "dev": true
1957 | },
1958 | "emoji-regex": {
1959 | "version": "8.0.0",
1960 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz",
1961 | "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==",
1962 | "dev": true
1963 | },
1964 | "is-fullwidth-code-point": {
1965 | "version": "3.0.0",
1966 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz",
1967 | "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==",
1968 | "dev": true
1969 | },
1970 | "strip-ansi": {
1971 | "version": "6.0.0",
1972 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.0.tgz",
1973 | "integrity": "sha512-AuvKTrTfQNYNIctbR1K/YGTR1756GycPsg7b9bdV9Duqur4gv6aKqHXah67Z8ImS7WEz5QVcOtlfW2rZEugt6w==",
1974 | "dev": true,
1975 | "requires": {
1976 | "ansi-regex": "^5.0.0"
1977 | }
1978 | }
1979 | }
1980 | },
1981 | "string.prototype.trimend": {
1982 | "version": "1.0.1",
1983 | "resolved": "https://registry.npmjs.org/string.prototype.trimend/-/string.prototype.trimend-1.0.1.tgz",
1984 | "integrity": "sha512-LRPxFUaTtpqYsTeNKaFOw3R4bxIzWOnbQ837QfBylo8jIxtcbK/A/sMV7Q+OAV/vWo+7s25pOE10KYSjaSO06g==",
1985 | "dev": true,
1986 | "requires": {
1987 | "define-properties": "^1.1.3",
1988 | "es-abstract": "^1.17.5"
1989 | }
1990 | },
1991 | "string.prototype.trimleft": {
1992 | "version": "2.1.2",
1993 | "resolved": "https://registry.npmjs.org/string.prototype.trimleft/-/string.prototype.trimleft-2.1.2.tgz",
1994 | "integrity": "sha512-gCA0tza1JBvqr3bfAIFJGqfdRTyPae82+KTnm3coDXkZN9wnuW3HjGgN386D7hfv5CHQYCI022/rJPVlqXyHSw==",
1995 | "dev": true,
1996 | "requires": {
1997 | "define-properties": "^1.1.3",
1998 | "es-abstract": "^1.17.5",
1999 | "string.prototype.trimstart": "^1.0.0"
2000 | }
2001 | },
2002 | "string.prototype.trimright": {
2003 | "version": "2.1.2",
2004 | "resolved": "https://registry.npmjs.org/string.prototype.trimright/-/string.prototype.trimright-2.1.2.tgz",
2005 | "integrity": "sha512-ZNRQ7sY3KroTaYjRS6EbNiiHrOkjihL9aQE/8gfQ4DtAC/aEBRHFJa44OmoWxGGqXuJlfKkZW4WcXErGr+9ZFg==",
2006 | "dev": true,
2007 | "requires": {
2008 | "define-properties": "^1.1.3",
2009 | "es-abstract": "^1.17.5",
2010 | "string.prototype.trimend": "^1.0.0"
2011 | }
2012 | },
2013 | "string.prototype.trimstart": {
2014 | "version": "1.0.1",
2015 | "resolved": "https://registry.npmjs.org/string.prototype.trimstart/-/string.prototype.trimstart-1.0.1.tgz",
2016 | "integrity": "sha512-XxZn+QpvrBI1FOcg6dIpxUPgWCPuNXvMD72aaRaUQv1eD4e/Qy8i/hFTe0BUmD60p/QA6bh1avmuPTfNjqVWRw==",
2017 | "dev": true,
2018 | "requires": {
2019 | "define-properties": "^1.1.3",
2020 | "es-abstract": "^1.17.5"
2021 | }
2022 | },
2023 | "string_decoder": {
2024 | "version": "1.1.1",
2025 | "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz",
2026 | "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==",
2027 | "requires": {
2028 | "safe-buffer": "~5.1.0"
2029 | }
2030 | },
2031 | "strip-ansi": {
2032 | "version": "5.2.0",
2033 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz",
2034 | "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==",
2035 | "dev": true,
2036 | "requires": {
2037 | "ansi-regex": "^4.1.0"
2038 | }
2039 | },
2040 | "strip-json-comments": {
2041 | "version": "2.0.1",
2042 | "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-2.0.1.tgz",
2043 | "integrity": "sha1-PFMZQukIwml8DsNEhYwobHygpgo=",
2044 | "dev": true
2045 | },
2046 | "superagent": {
2047 | "version": "3.8.3",
2048 | "resolved": "https://registry.npmjs.org/superagent/-/superagent-3.8.3.tgz",
2049 | "integrity": "sha512-GLQtLMCoEIK4eDv6OGtkOoSMt3D+oq0y3dsxMuYuDvaNUvuT8eFBuLmfR0iYYzHC1e8hpzC6ZsxbuP6DIalMFA==",
2050 | "dev": true,
2051 | "requires": {
2052 | "component-emitter": "^1.2.0",
2053 | "cookiejar": "^2.1.0",
2054 | "debug": "^3.1.0",
2055 | "extend": "^3.0.0",
2056 | "form-data": "^2.3.1",
2057 | "formidable": "^1.2.0",
2058 | "methods": "^1.1.1",
2059 | "mime": "^1.4.1",
2060 | "qs": "^6.5.1",
2061 | "readable-stream": "^2.3.5"
2062 | },
2063 | "dependencies": {
2064 | "debug": {
2065 | "version": "3.2.6",
2066 | "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.6.tgz",
2067 | "integrity": "sha512-mel+jf7nrtEl5Pn1Qx46zARXKDpBbvzezse7p7LqINmdoIk8PYP5SySaxEmYv6TZ0JyEKA1hsCId6DIhgITtWQ==",
2068 | "dev": true,
2069 | "requires": {
2070 | "ms": "^2.1.1"
2071 | }
2072 | },
2073 | "ms": {
2074 | "version": "2.1.2",
2075 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz",
2076 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==",
2077 | "dev": true
2078 | }
2079 | }
2080 | },
2081 | "supertest": {
2082 | "version": "4.0.2",
2083 | "resolved": "https://registry.npmjs.org/supertest/-/supertest-4.0.2.tgz",
2084 | "integrity": "sha512-1BAbvrOZsGA3YTCWqbmh14L0YEq0EGICX/nBnfkfVJn7SrxQV1I3pMYjSzG9y/7ZU2V9dWqyqk2POwxlb09duQ==",
2085 | "dev": true,
2086 | "requires": {
2087 | "methods": "^1.1.2",
2088 | "superagent": "^3.8.3"
2089 | }
2090 | },
2091 | "supports-color": {
2092 | "version": "5.5.0",
2093 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz",
2094 | "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==",
2095 | "dev": true,
2096 | "requires": {
2097 | "has-flag": "^3.0.0"
2098 | }
2099 | },
2100 | "term-size": {
2101 | "version": "2.2.0",
2102 | "resolved": "https://registry.npmjs.org/term-size/-/term-size-2.2.0.tgz",
2103 | "integrity": "sha512-a6sumDlzyHVJWb8+YofY4TW112G6p2FCPEAFk+59gIYHv3XHRhm9ltVQ9kli4hNWeQBwSpe8cRN25x0ROunMOw==",
2104 | "dev": true
2105 | },
2106 | "to-readable-stream": {
2107 | "version": "1.0.0",
2108 | "resolved": "https://registry.npmjs.org/to-readable-stream/-/to-readable-stream-1.0.0.tgz",
2109 | "integrity": "sha512-Iq25XBt6zD5npPhlLVXGFN3/gyR2/qODcKNNyTMd4vbm39HUaOiAM4PMq0eMVC/Tkxz+Zjdsc55g9yyz+Yq00Q==",
2110 | "dev": true
2111 | },
2112 | "to-regex-range": {
2113 | "version": "5.0.1",
2114 | "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz",
2115 | "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==",
2116 | "dev": true,
2117 | "requires": {
2118 | "is-number": "^7.0.0"
2119 | }
2120 | },
2121 | "toidentifier": {
2122 | "version": "1.0.0",
2123 | "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.0.tgz",
2124 | "integrity": "sha512-yaOH/Pk/VEhBWWTlhI+qXxDFXlejDGcQipMlyxda9nthulaxLZUNcUqFxokp0vcYnvteJln5FNQDRrxj3YcbVw=="
2125 | },
2126 | "touch": {
2127 | "version": "3.1.0",
2128 | "resolved": "https://registry.npmjs.org/touch/-/touch-3.1.0.tgz",
2129 | "integrity": "sha512-WBx8Uy5TLtOSRtIq+M03/sKDrXCLHxwDcquSP2c43Le03/9serjQBIztjRz6FkJez9D/hleyAXTBGLwwZUw9lA==",
2130 | "dev": true,
2131 | "requires": {
2132 | "nopt": "~1.0.10"
2133 | }
2134 | },
2135 | "type-detect": {
2136 | "version": "4.0.8",
2137 | "resolved": "https://registry.npmjs.org/type-detect/-/type-detect-4.0.8.tgz",
2138 | "integrity": "sha512-0fr/mIH1dlO+x7TlcMy+bIDqKPsw/70tVyeHW787goQjhmqaZe10uwLujubK9q9Lg6Fiho1KUKDYz0Z7k7g5/g==",
2139 | "dev": true
2140 | },
2141 | "type-fest": {
2142 | "version": "0.8.1",
2143 | "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.8.1.tgz",
2144 | "integrity": "sha512-4dbzIzqvjtgiM5rw1k5rEHtBANKmdudhGyBEajN01fEyhaAIhsoKNy6y7+IN93IfpFtwY9iqi7kD+xwKhQsNJA==",
2145 | "dev": true
2146 | },
2147 | "type-is": {
2148 | "version": "1.6.18",
2149 | "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz",
2150 | "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==",
2151 | "requires": {
2152 | "media-typer": "0.3.0",
2153 | "mime-types": "~2.1.24"
2154 | }
2155 | },
2156 | "typedarray-to-buffer": {
2157 | "version": "3.1.5",
2158 | "resolved": "https://registry.npmjs.org/typedarray-to-buffer/-/typedarray-to-buffer-3.1.5.tgz",
2159 | "integrity": "sha512-zdu8XMNEDepKKR+XYOXAVPtWui0ly0NtohUscw+UmaHiAWT8hrV1rr//H6V+0DvJ3OQ19S979M0laLfX8rm82Q==",
2160 | "dev": true,
2161 | "requires": {
2162 | "is-typedarray": "^1.0.0"
2163 | }
2164 | },
2165 | "undefsafe": {
2166 | "version": "2.0.3",
2167 | "resolved": "https://registry.npmjs.org/undefsafe/-/undefsafe-2.0.3.tgz",
2168 | "integrity": "sha512-nrXZwwXrD/T/JXeygJqdCO6NZZ1L66HrxM/Z7mIq2oPanoN0F1nLx3lwJMu6AwJY69hdixaFQOuoYsMjE5/C2A==",
2169 | "dev": true,
2170 | "requires": {
2171 | "debug": "^2.2.0"
2172 | }
2173 | },
2174 | "unique-string": {
2175 | "version": "2.0.0",
2176 | "resolved": "https://registry.npmjs.org/unique-string/-/unique-string-2.0.0.tgz",
2177 | "integrity": "sha512-uNaeirEPvpZWSgzwsPGtU2zVSTrn/8L5q/IexZmH0eH6SA73CmAA5U4GwORTxQAZs95TAXLNqeLoPPNO5gZfWg==",
2178 | "dev": true,
2179 | "requires": {
2180 | "crypto-random-string": "^2.0.0"
2181 | }
2182 | },
2183 | "unpipe": {
2184 | "version": "1.0.0",
2185 | "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz",
2186 | "integrity": "sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw="
2187 | },
2188 | "update-notifier": {
2189 | "version": "4.1.0",
2190 | "resolved": "https://registry.npmjs.org/update-notifier/-/update-notifier-4.1.0.tgz",
2191 | "integrity": "sha512-w3doE1qtI0/ZmgeoDoARmI5fjDoT93IfKgEGqm26dGUOh8oNpaSTsGNdYRN/SjOuo10jcJGwkEL3mroKzktkew==",
2192 | "dev": true,
2193 | "requires": {
2194 | "boxen": "^4.2.0",
2195 | "chalk": "^3.0.0",
2196 | "configstore": "^5.0.1",
2197 | "has-yarn": "^2.1.0",
2198 | "import-lazy": "^2.1.0",
2199 | "is-ci": "^2.0.0",
2200 | "is-installed-globally": "^0.3.1",
2201 | "is-npm": "^4.0.0",
2202 | "is-yarn-global": "^0.3.0",
2203 | "latest-version": "^5.0.0",
2204 | "pupa": "^2.0.1",
2205 | "semver-diff": "^3.1.1",
2206 | "xdg-basedir": "^4.0.0"
2207 | }
2208 | },
2209 | "url-parse-lax": {
2210 | "version": "3.0.0",
2211 | "resolved": "https://registry.npmjs.org/url-parse-lax/-/url-parse-lax-3.0.0.tgz",
2212 | "integrity": "sha1-FrXK/Afb42dsGxmZF3gj1lA6yww=",
2213 | "dev": true,
2214 | "requires": {
2215 | "prepend-http": "^2.0.0"
2216 | }
2217 | },
2218 | "util-deprecate": {
2219 | "version": "1.0.2",
2220 | "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz",
2221 | "integrity": "sha1-RQ1Nyfpw3nMnYvvS1KKJgUGaDM8="
2222 | },
2223 | "utils-merge": {
2224 | "version": "1.0.1",
2225 | "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz",
2226 | "integrity": "sha1-n5VxD1CiZ5R7LMwSR0HBAoQn5xM="
2227 | },
2228 | "vary": {
2229 | "version": "1.1.2",
2230 | "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz",
2231 | "integrity": "sha1-IpnwLG3tMNSllhsLn3RSShj2NPw="
2232 | },
2233 | "which": {
2234 | "version": "1.3.1",
2235 | "resolved": "https://registry.npmjs.org/which/-/which-1.3.1.tgz",
2236 | "integrity": "sha512-HxJdYWq1MTIQbJ3nw0cqssHoTNU267KlrDuGZ1WYlxDStUtKUhOaJmh112/TZmHxxUfuJqPXSOm7tDyas0OSIQ==",
2237 | "dev": true,
2238 | "requires": {
2239 | "isexe": "^2.0.0"
2240 | }
2241 | },
2242 | "which-module": {
2243 | "version": "2.0.0",
2244 | "resolved": "https://registry.npmjs.org/which-module/-/which-module-2.0.0.tgz",
2245 | "integrity": "sha1-2e8H3Od7mQK4o6j6SzHD4/fm6Ho=",
2246 | "dev": true
2247 | },
2248 | "wide-align": {
2249 | "version": "1.1.3",
2250 | "resolved": "https://registry.npmjs.org/wide-align/-/wide-align-1.1.3.tgz",
2251 | "integrity": "sha512-QGkOQc8XL6Bt5PwnsExKBPuMKBxnGxWWW3fU55Xt4feHozMUhdUMaBCk290qpm/wG5u/RSKzwdAC4i51YigihA==",
2252 | "dev": true,
2253 | "requires": {
2254 | "string-width": "^1.0.2 || 2"
2255 | },
2256 | "dependencies": {
2257 | "ansi-regex": {
2258 | "version": "3.0.0",
2259 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-3.0.0.tgz",
2260 | "integrity": "sha1-7QMXwyIGT3lGbAKWa922Bas32Zg=",
2261 | "dev": true
2262 | },
2263 | "string-width": {
2264 | "version": "2.1.1",
2265 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-2.1.1.tgz",
2266 | "integrity": "sha512-nOqH59deCq9SRHlxq1Aw85Jnt4w6KvLKqWVik6oA9ZklXLNIOlqg4F2yrT1MVaTjAqvVwdfeZ7w7aCvJD7ugkw==",
2267 | "dev": true,
2268 | "requires": {
2269 | "is-fullwidth-code-point": "^2.0.0",
2270 | "strip-ansi": "^4.0.0"
2271 | }
2272 | },
2273 | "strip-ansi": {
2274 | "version": "4.0.0",
2275 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-4.0.0.tgz",
2276 | "integrity": "sha1-qEeQIusaw2iocTibY1JixQXuNo8=",
2277 | "dev": true,
2278 | "requires": {
2279 | "ansi-regex": "^3.0.0"
2280 | }
2281 | }
2282 | }
2283 | },
2284 | "widest-line": {
2285 | "version": "3.1.0",
2286 | "resolved": "https://registry.npmjs.org/widest-line/-/widest-line-3.1.0.tgz",
2287 | "integrity": "sha512-NsmoXalsWVDMGupxZ5R08ka9flZjjiLvHVAWYOKtiKM8ujtZWr9cRffak+uSE48+Ob8ObalXpwyeUiyDD6QFgg==",
2288 | "dev": true,
2289 | "requires": {
2290 | "string-width": "^4.0.0"
2291 | }
2292 | },
2293 | "wrap-ansi": {
2294 | "version": "5.1.0",
2295 | "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-5.1.0.tgz",
2296 | "integrity": "sha512-QC1/iN/2/RPVJ5jYK8BGttj5z83LmSKmvbvrXPNCLZSEb32KKVDJDl/MOt2N01qU2H/FkzEa9PKto1BqDjtd7Q==",
2297 | "dev": true,
2298 | "requires": {
2299 | "ansi-styles": "^3.2.0",
2300 | "string-width": "^3.0.0",
2301 | "strip-ansi": "^5.0.0"
2302 | },
2303 | "dependencies": {
2304 | "ansi-styles": {
2305 | "version": "3.2.1",
2306 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz",
2307 | "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==",
2308 | "dev": true,
2309 | "requires": {
2310 | "color-convert": "^1.9.0"
2311 | }
2312 | },
2313 | "color-convert": {
2314 | "version": "1.9.3",
2315 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz",
2316 | "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==",
2317 | "dev": true,
2318 | "requires": {
2319 | "color-name": "1.1.3"
2320 | }
2321 | },
2322 | "color-name": {
2323 | "version": "1.1.3",
2324 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz",
2325 | "integrity": "sha1-p9BVi9icQveV3UIyj3QIMcpTvCU=",
2326 | "dev": true
2327 | },
2328 | "string-width": {
2329 | "version": "3.1.0",
2330 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz",
2331 | "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==",
2332 | "dev": true,
2333 | "requires": {
2334 | "emoji-regex": "^7.0.1",
2335 | "is-fullwidth-code-point": "^2.0.0",
2336 | "strip-ansi": "^5.1.0"
2337 | }
2338 | }
2339 | }
2340 | },
2341 | "wrappy": {
2342 | "version": "1.0.2",
2343 | "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz",
2344 | "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=",
2345 | "dev": true
2346 | },
2347 | "write-file-atomic": {
2348 | "version": "3.0.3",
2349 | "resolved": "https://registry.npmjs.org/write-file-atomic/-/write-file-atomic-3.0.3.tgz",
2350 | "integrity": "sha512-AvHcyZ5JnSfq3ioSyjrBkH9yW4m7Ayk8/9My/DD9onKeu/94fwrMocemO2QAJFAlnnDN+ZDS+ZjAR5ua1/PV/Q==",
2351 | "dev": true,
2352 | "requires": {
2353 | "imurmurhash": "^0.1.4",
2354 | "is-typedarray": "^1.0.0",
2355 | "signal-exit": "^3.0.2",
2356 | "typedarray-to-buffer": "^3.1.5"
2357 | }
2358 | },
2359 | "xdg-basedir": {
2360 | "version": "4.0.0",
2361 | "resolved": "https://registry.npmjs.org/xdg-basedir/-/xdg-basedir-4.0.0.tgz",
2362 | "integrity": "sha512-PSNhEJDejZYV7h50BohL09Er9VaIefr2LMAf3OEmpCkjOi34eYyQYAXUTjEQtZJTKcF0E2UKTh+osDLsgNim9Q==",
2363 | "dev": true
2364 | },
2365 | "y18n": {
2366 | "version": "4.0.0",
2367 | "resolved": "https://registry.npmjs.org/y18n/-/y18n-4.0.0.tgz",
2368 | "integrity": "sha512-r9S/ZyXu/Xu9q1tYlpsLIsa3EeLXXk0VwlxqTcFRfg9EhMW+17kbt9G0NrgCmhGb5vT2hyhJZLfDGx+7+5Uj/w==",
2369 | "dev": true
2370 | },
2371 | "yargs": {
2372 | "version": "13.3.2",
2373 | "resolved": "https://registry.npmjs.org/yargs/-/yargs-13.3.2.tgz",
2374 | "integrity": "sha512-AX3Zw5iPruN5ie6xGRIDgqkT+ZhnRlZMLMHAs8tg7nRruy2Nb+i5o9bwghAogtM08q1dpr2LVoS8KSTMYpWXUw==",
2375 | "dev": true,
2376 | "requires": {
2377 | "cliui": "^5.0.0",
2378 | "find-up": "^3.0.0",
2379 | "get-caller-file": "^2.0.1",
2380 | "require-directory": "^2.1.1",
2381 | "require-main-filename": "^2.0.0",
2382 | "set-blocking": "^2.0.0",
2383 | "string-width": "^3.0.0",
2384 | "which-module": "^2.0.0",
2385 | "y18n": "^4.0.0",
2386 | "yargs-parser": "^13.1.2"
2387 | },
2388 | "dependencies": {
2389 | "string-width": {
2390 | "version": "3.1.0",
2391 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz",
2392 | "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==",
2393 | "dev": true,
2394 | "requires": {
2395 | "emoji-regex": "^7.0.1",
2396 | "is-fullwidth-code-point": "^2.0.0",
2397 | "strip-ansi": "^5.1.0"
2398 | }
2399 | }
2400 | }
2401 | },
2402 | "yargs-parser": {
2403 | "version": "13.1.2",
2404 | "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-13.1.2.tgz",
2405 | "integrity": "sha512-3lbsNRf/j+A4QuSZfDRA7HRSfWrzO0YjqTJd5kjAq37Zep1CEgaYmrH9Q3GwPiB9cHyd1Y1UwggGhJGoxipbzg==",
2406 | "dev": true,
2407 | "requires": {
2408 | "camelcase": "^5.0.0",
2409 | "decamelize": "^1.2.0"
2410 | }
2411 | },
2412 | "yargs-unparser": {
2413 | "version": "1.6.0",
2414 | "resolved": "https://registry.npmjs.org/yargs-unparser/-/yargs-unparser-1.6.0.tgz",
2415 | "integrity": "sha512-W9tKgmSn0DpSatfri0nx52Joq5hVXgeLiqR/5G0sZNDoLZFOr/xjBUDcShCOGNsBnEMNo1KAMBkTej1Hm62HTw==",
2416 | "dev": true,
2417 | "requires": {
2418 | "flat": "^4.1.0",
2419 | "lodash": "^4.17.15",
2420 | "yargs": "^13.3.0"
2421 | }
2422 | }
2423 | }
2424 | }
2425 |
--------------------------------------------------------------------------------