From 6449d60e0cc2f70cacf9c8a2e2a0776d4fcd3d87 Mon Sep 17 00:00:00 2001 From: Vikas Agarwal Date: Mon, 2 Aug 2021 13:30:09 +0530 Subject: [PATCH 1/9] Initial code with permission model changes incorporated --- README.md | 159 +- app.js | 88 + config/default.js | 51 + config/production.js | 7 + docker-pgsql-es/docker-compose.yml | 23 + docker-pgsql-es/sample.env | 5 + docker/Dockerfile | 17 + docker/docker-compose.yml | 12 + docker/sample.env | 8 + docs/permissions.html | 460 ++ docs/skills-api.postman_collection.json | 626 +++ docs/skills-api.postman_environment.json | 29 + docs/swagger.yaml | 1082 ++++ package-lock.json | 4938 ++++++++++++++++++ package.json | 48 + scripts/constants.js | 50 + scripts/db/createDb.js | 9 + scripts/db/data/Skill.json | 62 + scripts/db/data/Taxonomy.json | 20 + scripts/db/dropAll.js | 65 + scripts/db/dumpDbToEs.js | 266 + scripts/db/insert-data.js | 28 + scripts/db/migrations.js | 35 + scripts/db/migrations/01-create-taxonomy.js | 35 + scripts/db/migrations/02_create-skill.js | 41 + scripts/db/migrations/03_add-relationship.js | 17 + scripts/permissions-doc/index.js | 72 + scripts/permissions-doc/template.hbs | 115 + src/bootstrap.js | 42 + src/common/controller-helper.js | 66 + src/common/db-helper.js | 175 + src/common/error.middleware.js | 33 + src/common/errors.js | 29 + src/common/es-client.js | 38 + src/common/es-helper.js | 376 ++ src/common/helper.js | 98 + src/common/logger.js | 152 + src/common/permission-helper.js | 133 + src/common/service-helper.js | 131 + src/constants.js | 49 + src/models/Skill.js | 38 + src/models/Taxonomy.js | 32 + src/models/index.js | 43 + src/modules/SkillMetadata/controller.js | 38 + src/modules/SkillMetadata/route.js | 23 + src/modules/SkillMetadata/service.js | 132 + src/modules/skill/controller.js | 11 + src/modules/skill/route.js | 39 + src/modules/skill/service.js | 234 + src/modules/taxonomy/controller.js | 11 + src/modules/taxonomy/route.js | 39 + src/modules/taxonomy/service.js | 173 + src/modules/taxonomyMetadata/controller.js | 38 + src/modules/taxonomyMetadata/route.js | 23 + src/modules/taxonomyMetadata/service.js | 124 + src/permissions/constants.js | 195 + src/permissions/generalPermission.js | 43 + src/permissions/index.js | 21 + src/route.js | 29 + 59 files changed, 10974 insertions(+), 2 deletions(-) mode change 100644 => 100755 README.md create mode 100755 app.js create mode 100755 config/default.js create mode 100755 config/production.js create mode 100644 docker-pgsql-es/docker-compose.yml create mode 100644 docker-pgsql-es/sample.env create mode 100644 docker/Dockerfile create mode 100644 docker/docker-compose.yml create mode 100644 docker/sample.env create mode 100644 docs/permissions.html create mode 100644 docs/skills-api.postman_collection.json create mode 100644 docs/skills-api.postman_environment.json create mode 100644 docs/swagger.yaml create mode 100644 package-lock.json create mode 100644 package.json create mode 100644 scripts/constants.js create mode 100644 scripts/db/createDb.js create mode 100644 scripts/db/data/Skill.json create mode 100644 scripts/db/data/Taxonomy.json create mode 100644 scripts/db/dropAll.js create mode 100644 scripts/db/dumpDbToEs.js create mode 100644 scripts/db/insert-data.js create mode 100644 scripts/db/migrations.js create mode 100644 scripts/db/migrations/01-create-taxonomy.js create mode 100644 scripts/db/migrations/02_create-skill.js create mode 100644 scripts/db/migrations/03_add-relationship.js create mode 100644 scripts/permissions-doc/index.js create mode 100644 scripts/permissions-doc/template.hbs create mode 100755 src/bootstrap.js create mode 100644 src/common/controller-helper.js create mode 100644 src/common/db-helper.js create mode 100755 src/common/error.middleware.js create mode 100644 src/common/errors.js create mode 100644 src/common/es-client.js create mode 100644 src/common/es-helper.js create mode 100644 src/common/helper.js create mode 100755 src/common/logger.js create mode 100644 src/common/permission-helper.js create mode 100644 src/common/service-helper.js create mode 100644 src/constants.js create mode 100644 src/models/Skill.js create mode 100644 src/models/Taxonomy.js create mode 100755 src/models/index.js create mode 100644 src/modules/SkillMetadata/controller.js create mode 100644 src/modules/SkillMetadata/route.js create mode 100644 src/modules/SkillMetadata/service.js create mode 100644 src/modules/skill/controller.js create mode 100644 src/modules/skill/route.js create mode 100644 src/modules/skill/service.js create mode 100644 src/modules/taxonomy/controller.js create mode 100644 src/modules/taxonomy/route.js create mode 100644 src/modules/taxonomy/service.js create mode 100644 src/modules/taxonomyMetadata/controller.js create mode 100644 src/modules/taxonomyMetadata/route.js create mode 100644 src/modules/taxonomyMetadata/service.js create mode 100644 src/permissions/constants.js create mode 100644 src/permissions/generalPermission.js create mode 100644 src/permissions/index.js create mode 100755 src/route.js diff --git a/README.md b/README.md old mode 100644 new mode 100755 index d8125db..c01320c --- a/README.md +++ b/README.md @@ -1,2 +1,157 @@ -# skills-api -V5 Skills API +# Skills API + +* [Prerequisites](#prerequisites) +* [Configuration](#configuration) +* [Local deployment](#local-deployment) +* [Migrations](#migrations) +* [Local Deployment with Docker](#local-deployment-with-docker) +* [NPM Commands](#npm-commands) +* [JWT Authentication](#jwt-authentication) +* [Documentation](#documentation) + +## Prerequisites + +- node 12.x+ +- npm 6.x+ +- docker +- elasticsearch 7.7+ +- PostgreSQL + +## Configuration + +Configuration for the application is at `config/default.js` and `config/production.js`. The following parameters can be set in config files or in env variables: + +- LOG_LEVEL: the log level +- PORT: the server port +- AUTH_SECRET: TC Authentication secret +- VALID_ISSUERS: valid issuers for TC authentication +- PAGE_SIZE: the default pagination limit +- MAX_PAGE_SIZE: the maximum pagination size +- API_VERSION: the API version +- DB_NAME: the database name +- DB_USERNAME: the database username +- DB_PASSWORD: the database password +- DB_HOST: the database host +- DB_PORT: the database port +- ES_HOST: Elasticsearch host +- ES_REFRESH: Should elastic search refresh. Default is 'true'. Values can be 'true', 'wait_for', 'false' +- ELASTICCLOUD_ID: The elastic cloud id, if your elasticsearch instance is hosted on elastic cloud. DO NOT provide a value for ES_HOST if you are using this +- ELASTICCLOUD_USERNAME: The elastic cloud username for basic authentication. Provide this only if your elasticsearch instance is hosted on elastic cloud +- ELASTICCLOUD_PASSWORD: The elastic cloud password for basic authentication. Provide this only if your elasticsearch instance is hosted on elastic cloud +- ES.DOCUMENTS: Elasticsearch index, type and id mapping for resources. +- SKILL_INDEX: The Elastic search index for skill. Default is `skill` +- SKILL_ENRICH_POLICYNAME: The enrich policy for skill. Default is `skill-policy` +- TAXONOMY_INDEX: The Elastic search index for taxonomy. Default is `taxonomy` +- TAXONOMY_PIPELINE_ID: The pipeline id for enrichment with taxonomy. Default is `taxonomy-pipeline` +- TAXONOMY_ENRICH_POLICYNAME: The enrich policy for taxonomy. Default is `taxonomy-policy` +- MAX_BATCH_SIZE: Restrict number of records in memory during bulk insert (Used by the db to es migration script) +- MAX_BULK_SIZE: The Bulk Indexing Maximum Limits. Default is `100` (Used by the db to es migration script) + + +## Local deployment + +Setup your Postgresql DB and Elasticsearch instance and ensure that they are up and running. + +- Follow *Configuration* section to update config values, like database, ES host etc .. +- Goto *skills-api*, run `npm i` +- Create database using `npm run create-db`. +- Run the migrations - `npm run migrations up`. This will create the tables. +- Then run `npm run insert-data` and insert mock data into the database. +- Run `npm run migrate-db-to-es` to sync data with ES. +- Startup server `npm run start` + +## Migrations + +Migrations are located under the `./scripts/db/` folder. Run `npm run migrations up` and `npm run migrations down` to execute the migrations or remove the earlier ones + +## Local Deployment with Docker + +- Navigate to the directory `docker-pgsql-es` folder. Rename `sample.env` to `.env` and change any values if required. +- Run `docker-compose up -d` to have docker instances of pgsql and elasticsearch to use with the api + +- Create database using `npm run create-db`. +- Run the migrations - `npm run migrations up`. This will create the tables. +- Then run `npm run insert-data` and insert mock data into the database. +- Run `npm run migrate-db-to-es` to sync data with ES. + +- Navigate to the directory `docker` + +- Rename the file `sample.env` to `.env` + +- Set the required DB configurations and ElasticSearch host in the file `.env` + +- Once that is done, run the following command + + ```bash + docker-compose up + ``` + +- When you are running the application for the first time, It will take some time initially to download the image and install the dependencies + +## NPM Commands + +| Command                    | Description | +|--------------------|--| +| `npm run start` | Start app | +| `npm run start:dev` | Start app on any changes (useful during development). | +| `npm run lint` | Check for for lint errors. | +| `npm run lint:fix` | Check for for lint errors and fix error automatically when possible. | +| `npm run create-db` | Create the database | +| `npm run insert-data` | Insert data into the database | +| `npm run migrate-db-to-es` | Migrate data into elastic search from database | +| `npm run delete-data` | Delete the data from the database | +| `npm run migrations up` | Run up migration | +| `npm run migrations down` | Run down migration | +| `npm run generate:doc:permissions` | Generate [permissions.html](docs/permissions.html) | +| `npm run generate:doc:permissions:dev` | Generate [permissions.html](docs/permissions.html) on any changes (useful during development). | + +## JWT Authentication +Authentication is handled via Authorization (Bearer) token header field. Token is a JWT token. + +Here is a sample user token that is valid for a very long time for a user with administrator role. + +``` + + +# here is the payload data decoded from the token +{ + "roles": [ + "Topcoder User", + "administrator" + ], + "iss": "https://api.topcoder.com/", + "handle": "tc-Admin", + "exp": 1685571460, + "userId": "23166768", + "iat": 1585570860, + "email": "tc-Admin@gmail.com", + "jti": "0f1ef1d3-2b33-4900-bb43-48f2285f9630" +} +``` + +and this is a sample M2M token with scopes `all:connect_project`, `all:projects` and `write:projects`. + +``` + + +# here is the payload data decoded from the token +{ + "iss": "https://topcoder-dev.auth0.com/", + "sub": "enjw1810eDz3XTwSO2Rn2Y9cQTrspn3B@clients", + "aud": "https://m2m.topcoder-dev.com/", + "iat": 1550906388, + "exp": 2147483648, + "azp": "enjw1810eDz3XTwSO2Rn2Y9cQTrspn3B", + "scope": "all:connect_project all:projects write:projects", + "gty": "client-credentials" +} +``` + +These tokens have been signed with the secret `CLIENT_SECRET`. This secret should match the `AUTH_SECRET` entry in `config/default.js`. You can modify the payload of these tokens to generate tokens with different roles or different scopes using https://jwt.io + +**Note** Please check with `src/constants.js` for all available user roles and M2M scopes. + +## Documentation + +- [permissions.html](docs/permissions.html) - the list of all permissions in Skills API. +- [swagger.yaml](docs/swagger.yaml) - the Swagger API Definition. diff --git a/app.js b/app.js new file mode 100755 index 0000000..e535e41 --- /dev/null +++ b/app.js @@ -0,0 +1,88 @@ +/** + * The application entry point + */ + +require('./src/bootstrap') +const config = require('config') +const express = require('express') +const cross = require('cors') +const bodyParser = require('body-parser') +const _ = require('lodash') +const http = require('http') +const swaggerUi = require('swagger-ui-express') +const jsyaml = require('js-yaml') +const fs = require('fs') +const path = require('path') +const logger = require('./src/common/logger') +const errorMiddleware = require('./src/common/error.middleware') +const routes = require('./src/route') +const { permissions, jwtAuthenticator } = require('tc-core-library-js').middleware +const app = express() +const httpServer = http.Server(app) +const models = require('./src/models') +const initPermissions = require('./src/permissions') + +app.set('port', config.PORT) +app.use(bodyParser.json()) +app.use(bodyParser.urlencoded({ extended: true })) +app.use(cross()) +const apiRouter = express.Router({}) + +// load all routes +_.each(routes, (verbs, url) => { + _.each(verbs, (def, verb) => { + if (!def.method) { + throw new Error(`${verb.toUpperCase()} ${url} method is undefined`) + } + if (def.auth && def.auth !== 'jwt') { + throw new Error(`auth type "${def.auth}" is not supported`) + } + + const actions = [] + // Authentication + if (def.auth) { + actions.push((req, res, next) => { + jwtAuthenticator(_.pick(config, ['AUTH_SECRET', 'VALID_ISSUERS']))(req, res, next) + }) + } + // Authorization + if (def.permission) { + actions.push(permissions(def.permission)) + } + // main middleware + actions.push(async (req, res, next) => { + try { + await def.method(req, res, next) + } catch (e) { + next(e) + } + }) + + logger.info(`Endpoint discovered : ${verb.toLocaleUpperCase()} /${config.API_VERSION}${url}`) + apiRouter[verb](`/${config.API_VERSION}${url}`, actions) + }) +}) +app.use('/', apiRouter) +const spec = fs.readFileSync(path.join(__dirname, 'docs/swagger.yaml'), 'utf8') +const swaggerDoc = jsyaml.safeLoad(spec) + +app.use('/docs', swaggerUi.serve, swaggerUi.setup(swaggerDoc)) + +app.use(errorMiddleware()) +app.use('*', (req, res) => { + const pathKey = req.baseUrl.substring(config.API_VERSION.length + 1) + const route = routes[pathKey] + if (route) { + res.status(405).json({ message: 'The requested method is not supported.' }) + } else { + res.status(404).json({ message: 'The requested resource cannot found.' }) + } +}); + +(async () => { + await models.init() + initPermissions() // initialize permission policies + httpServer.listen(app.get('port'), () => { + logger.info(`Express server listening on port ${app.get('port')}`) + }) +})() diff --git a/config/default.js b/config/default.js new file mode 100755 index 0000000..2c7de80 --- /dev/null +++ b/config/default.js @@ -0,0 +1,51 @@ +/** + * the default config + */ + +module.exports = { + LOG_LEVEL: process.env.LOG_LEVEL || 'debug', + PORT: process.env.PORT || 3001, + + AUTH_SECRET: process.env.AUTH_SECRET || 'CLIENT_SECRET', + VALID_ISSUERS: process.env.VALID_ISSUERS ? process.env.VALID_ISSUERS.replace(/\\"/g, '') + : '["https://topcoder-dev.auth0.com/", "https://api.topcoder.com/"]', + + PAGE_SIZE: process.env.PAGE_SIZE || 20, + MAX_PAGE_SIZE: parseInt(process.env.MAX_PAGE_SIZE) || 100, + API_VERSION: process.env.API_VERSION || 'api/1.0', + + DB_NAME: process.env.DB_NAME || 'skills-db', + DB_USERNAME: process.env.DB_USER || 'postgres', + DB_PASSWORD: process.env.DB_PASSWORD || 'password', + DB_HOST: process.env.DB_HOST || 'localhost', + DB_PORT: process.env.DB_PORT || 5432, + + // ElasticSearch + ES: { + HOST: process.env.ES_HOST || 'http://localhost:9200/', + ES_REFRESH: process.env.ES_REFRESH || 'true', + + ELASTICCLOUD: { + id: process.env.ELASTICCLOUD_ID, + username: process.env.ELASTICCLOUD_USERNAME, + password: process.env.ELASTICCLOUD_PASSWORD + }, + + // es mapping: _index, _type, _id + DOCUMENTS: { + skill: { + index: process.env.SKILL_INDEX || 'skill', + type: '_doc', + enrichPolicyName: process.env.SKILL_ENRICH_POLICYNAME || 'skill-policy' + }, + taxonomy: { + index: process.env.TAXONOMY_INDEX || 'taxonomy', + type: '_doc', + pipelineId: process.env.TAXONOMY_PIPELINE_ID || 'taxonomy-pipeline', + enrichPolicyName: process.env.TAXONOMY_ENRICH_POLICYNAME || 'taxonomy-policy' + } + }, + MAX_BATCH_SIZE: parseInt(process.env.MAX_BATCH_SIZE, 10) || 10000, + MAX_BULK_SIZE: parseInt(process.env.MAX_BULK_SIZE, 10) || 100 + } +} diff --git a/config/production.js b/config/production.js new file mode 100755 index 0000000..00b345f --- /dev/null +++ b/config/production.js @@ -0,0 +1,7 @@ +/** + * The production configuration file. + */ + +module.exports = { + LOG_LEVEL: process.env.LOG_LEVEL || 'info' +} diff --git a/docker-pgsql-es/docker-compose.yml b/docker-pgsql-es/docker-compose.yml new file mode 100644 index 0000000..e96b7f3 --- /dev/null +++ b/docker-pgsql-es/docker-compose.yml @@ -0,0 +1,23 @@ +version: '3' +services: + postgres: + image: "postgres:12.4" + volumes: + - database-data:/var/lib/postgresql/data/ + ports: + - ${DB_PORT}:${DB_PORT} + environment: + POSTGRES_PASSWORD: ${DB_PASSWORD} + POSTGRES_USER: ${DB_USERNAME} + POSTGRES_DB: ${DB_NAME} + esearch: + image: elasticsearch:7.7.1 + container_name: skills-data-processor-es_es + ports: + - ${ES_PORT}:${ES_PORT} + environment: + - discovery.type=single-node + +volumes: + database-data: + diff --git a/docker-pgsql-es/sample.env b/docker-pgsql-es/sample.env new file mode 100644 index 0000000..7f35ce2 --- /dev/null +++ b/docker-pgsql-es/sample.env @@ -0,0 +1,5 @@ +DB_NAME=skills-db +DB_USERNAME=postgres +DB_PASSWORD=password +DB_PORT=5432 +ES_PORT=9200 diff --git a/docker/Dockerfile b/docker/Dockerfile new file mode 100644 index 0000000..dcd6339 --- /dev/null +++ b/docker/Dockerfile @@ -0,0 +1,17 @@ +# Use the base image with Node.js 12 +FROM node:12 + +# Set working directory for future use +WORKDIR /skills_api + +# Copy the current directory into the Docker image +COPY . /skills_api + +# Install the dependencies from package.json +RUN npm install + +# Expose port +EXPOSE ${PORT} + +# start api +CMD npm start diff --git a/docker/docker-compose.yml b/docker/docker-compose.yml new file mode 100644 index 0000000..50db175 --- /dev/null +++ b/docker/docker-compose.yml @@ -0,0 +1,12 @@ +version: '3' +services: + skills_api: + image: skills_api:latest + build: + context: ../ + dockerfile: docker/Dockerfile + env_file: + - .env + ports: + - ${PORT}:${PORT} + diff --git a/docker/sample.env b/docker/sample.env new file mode 100644 index 0000000..6cb7442 --- /dev/null +++ b/docker/sample.env @@ -0,0 +1,8 @@ +DB_NAME=skills-db +DB_USERNAME=postgres +DB_PASSWORD=password +DB_HOST=host.docker.internal +DB_PORT=5432 + +ES_HOST=http://host.docker.internal:9200 +PORT=3001 diff --git a/docs/permissions.html b/docs/permissions.html new file mode 100644 index 0000000..eaf4289 --- /dev/null +++ b/docs/permissions.html @@ -0,0 +1,460 @@ + + + + + + + Permissions + + + +
+
+

Permissions

+

List of all the possible user permissions inside Skills API

+
+

Legend:

+
    +
  • allowed Topcoder Role - users with such a Topcoder Role are allowed to perform the action
  • +
  • denied Topcoder Role - users with such a Topcoder Role are denied to perform the action even they have some other allow roles
  • +
  • allowed M2M Scope - M2M tokens with such a scope are allowed to perform the action
  • +
  • denied M2M Scope - M2M tokens with such a scope are allowed to perform the action even they have some other allow scopes
  • +
+
+ +
+
+

+ Skill +

+
+
+
+
+
+ Create Skill +
+
CREATE_SKILL
+
+
+
+
+ Connect Admin + administrator + Connect Manager + Connect Account Manager + Connect Copilot Manager + Business Development Representative + Presales + Account Executive + Program Manager + Solution Architect + Project Manager +
+ +
+ all:connect_project + all:projects + write:projects +
+
+
+
+
+
+ Update Skill +
+
UPDATE_SKILL
+
+
+
+
+ Connect Admin + administrator + Connect Manager + Connect Account Manager + Connect Copilot Manager + Business Development Representative + Presales + Account Executive + Program Manager + Solution Architect + Project Manager +
+ +
+ all:connect_project + all:projects + write:projects +
+
+
+
+
+
+ Delete Skill +
+
DELETE_SKILL
+
+
+
+
+ Connect Admin + administrator + Connect Manager + Connect Account Manager + Connect Copilot Manager + Business Development Representative + Presales + Account Executive + Program Manager + Solution Architect + Project Manager +
+ +
+ all:connect_project + all:projects + write:projects +
+
+
+
+
+

+ Skill Metadata +

+
+
+
+
+
+ Add Skill Metadata +
+
ADD_SKILL_METADATA
+
Add metadata fields in a skill
+
+
+
+ Connect Admin + administrator + Connect Manager + Connect Account Manager + Connect Copilot Manager + Business Development Representative + Presales + Account Executive + Program Manager + Solution Architect + Project Manager +
+ +
+ all:connect_project + all:projects + write:projects +
+
+
+
+
+
+ Update Skill Metadata +
+
UPDATE_SKILL_METADATA
+
Update Metadata fields from a skill
+
+
+
+ Connect Admin + administrator + Connect Manager + Connect Account Manager + Connect Copilot Manager + Business Development Representative + Presales + Account Executive + Program Manager + Solution Architect + Project Manager +
+ +
+ all:connect_project + all:projects + write:projects +
+
+
+
+
+
+ Delete Skill Metadata +
+
DELETE_SKILL_METADATA
+
Delete Metadata fields from a skill
+
+
+
+ Connect Admin + administrator + Connect Manager + Connect Account Manager + Connect Copilot Manager + Business Development Representative + Presales + Account Executive + Program Manager + Solution Architect + Project Manager +
+ +
+ all:connect_project + all:projects + write:projects +
+
+
+
+
+

+ Taxonomy Metadata +

+
+
+
+
+
+ Create Taxonomy +
+
CREATE_TAXONOMY
+
+
+
+
+ Connect Admin + administrator + Connect Manager + Connect Account Manager + Connect Copilot Manager + Business Development Representative + Presales + Account Executive + Program Manager + Solution Architect + Project Manager +
+ +
+ all:connect_project + all:projects + write:projects +
+
+
+
+
+
+ Update Taxonomy +
+
UPDATE_TAXONOMY
+
+
+
+
+ Connect Admin + administrator + Connect Manager + Connect Account Manager + Connect Copilot Manager + Business Development Representative + Presales + Account Executive + Program Manager + Solution Architect + Project Manager +
+ +
+ all:connect_project + all:projects + write:projects +
+
+
+
+
+
+ Delete Taxonomy +
+
DELETE_TAXONOMY
+
+
+
+
+ Connect Admin + administrator + Connect Manager + Connect Account Manager + Connect Copilot Manager + Business Development Representative + Presales + Account Executive + Program Manager + Solution Architect + Project Manager +
+ +
+ all:connect_project + all:projects + write:projects +
+
+
+
+
+

+ Taxonomy +

+
+
+
+
+
+ Add Taxonomy Metadata +
+
ADD_TAXONOMY_METADATA
+
Add metadata fields in a taxonomy
+
+
+
+ Connect Admin + administrator + Connect Manager + Connect Account Manager + Connect Copilot Manager + Business Development Representative + Presales + Account Executive + Program Manager + Solution Architect + Project Manager +
+ +
+ all:connect_project + all:projects + write:projects +
+
+
+
+
+
+ Update Taxonomy Metadata +
+
UPDATE_TAXONOMY_METADATA
+
Update Metadata fields from a taxonomy
+
+
+
+ Connect Admin + administrator + Connect Manager + Connect Account Manager + Connect Copilot Manager + Business Development Representative + Presales + Account Executive + Program Manager + Solution Architect + Project Manager +
+ +
+ all:connect_project + all:projects + write:projects +
+
+
+
+
+
+ Delete Taxonomy Metadata +
+
DELETE_TAXONOMY_METADATA
+
Delete Metadata fields from a taxonomy
+
+
+
+ Connect Admin + administrator + Connect Manager + Connect Account Manager + Connect Copilot Manager + Business Development Representative + Presales + Account Executive + Program Manager + Solution Architect + Project Manager +
+ +
+ all:connect_project + all:projects + write:projects +
+
+
+ +
+ + diff --git a/docs/skills-api.postman_collection.json b/docs/skills-api.postman_collection.json new file mode 100644 index 0000000..7f62ddb --- /dev/null +++ b/docs/skills-api.postman_collection.json @@ -0,0 +1,626 @@ +{ + "info": { + "_postman_id": "807bc545-d28c-4725-b88a-1bca2e734a50", + "name": "skills-api", + "schema": "https://schema.getpostman.com/json/collection/v2.1.0/collection.json" + }, + "item": [ + { + "name": "taxonomies", + "item": [ + { + "name": "{{HOST}}/taxonomies", + "event": [ + { + "listen": "test", + "script": { + "exec": [ + "var rsp = pm.response.json();", + "if(rsp.id) pm.environment.set(\"taxonomyId\", rsp.id);" + ], + "type": "text/javascript" + } + } + ], + "request": { + "method": "POST", + "header": [ + { + "key": "Authorization", + "type": "text", + "value": "Bearer {{token}}" + }, + { + "key": "Content-Type", + "name": "Content-Type", + "type": "text", + "value": "application/json" + } + ], + "body": { + "mode": "raw", + "raw": "{\n \"name\": \"taxonomy_06\",\n \"metadata\": {}\n}" + }, + "url": { + "raw": "{{HOST}}/taxonomies", + "host": [ + "{{HOST}}" + ], + "path": [ + "taxonomies" + ] + } + }, + "response": [] + }, + { + "name": "{{HOST}}/taxonomies/:id", + "request": { + "method": "GET", + "header": [], + "url": { + "raw": "{{HOST}}/taxonomies/{{taxonomyId}}", + "host": [ + "{{HOST}}" + ], + "path": [ + "taxonomies", + "{{taxonomyId}}" + ] + } + }, + "response": [] + }, + { + "name": "{{HOST}}/taxonomies/:id", + "request": { + "method": "PATCH", + "header": [ + { + "key": "Authorization", + "type": "text", + "value": "Bearer {{token}}" + }, + { + "key": "Content-Type", + "name": "Content-Type", + "type": "text", + "value": "application/json" + } + ], + "body": { + "mode": "raw", + "raw": "{\n \"name\": \"taxonomies_update12\",\n \"metadata\": {\n \"version\": \"1.0.0\"\n }\n}" + }, + "url": { + "raw": "{{HOST}}/taxonomies/{{taxonomyId}}", + "host": [ + "{{HOST}}" + ], + "path": [ + "taxonomies", + "{{taxonomyId}}" + ] + } + }, + "response": [] + }, + { + "name": "{{HOST}}/taxonomies/:id", + "request": { + "method": "HEAD", + "header": [ + { + "key": "Content-Type", + "name": "Content-Type", + "type": "text", + "value": "application/json" + } + ], + "url": { + "raw": "{{HOST}}/taxonomies/{{taxonomyId}}", + "host": [ + "{{HOST}}" + ], + "path": [ + "taxonomies", + "{{taxonomyId}}" + ] + } + }, + "response": [] + }, + { + "name": "{{HOST}}/taxonomies", + "request": { + "method": "GET", + "header": [], + "url": { + "raw": "{{HOST}}/taxonomies", + "host": [ + "{{HOST}}" + ], + "path": [ + "taxonomies" + ] + } + }, + "response": [] + }, + { + "name": "{{HOST}}/taxonomies", + "request": { + "method": "HEAD", + "header": [], + "url": { + "raw": "{{HOST}}/taxonomies", + "host": [ + "{{HOST}}" + ], + "path": [ + "taxonomies" + ] + } + }, + "response": [] + }, + { + "name": "{{HOST}}/taxonomies/:id", + "request": { + "method": "DELETE", + "header": [ + { + "key": "Authorization", + "type": "text", + "value": "Bearer {{token}}" + }, + { + "key": "Content-Type", + "name": "Content-Type", + "type": "text", + "value": "application/json" + } + ], + "body": { + "mode": "raw", + "raw": "" + }, + "url": { + "raw": "{{HOST}}/taxonomies/{{taxonomyId}}", + "host": [ + "{{HOST}}" + ], + "path": [ + "taxonomies", + "{{taxonomyId}}" + ] + } + }, + "response": [] + } + ] + }, + { + "name": "taxonomies metadata", + "item": [ + { + "name": "{{HOST}}/taxonomies/:id", + "request": { + "method": "PUT", + "header": [ + { + "key": "Authorization", + "type": "text", + "value": "Bearer {{token}}" + }, + { + "key": "Content-Type", + "name": "Content-Type", + "type": "text", + "value": "application/json" + } + ], + "body": { + "mode": "raw", + "raw": "{\n \"version\": \"1.0.1\",\n \"isLongTermSupport\": true\n}" + }, + "url": { + "raw": "{{HOST}}/taxonomies/{{taxonomyId}}/metadata", + "host": [ + "{{HOST}}" + ], + "path": [ + "taxonomies", + "{{taxonomyId}}", + "metadata" + ] + } + }, + "response": [] + }, + { + "name": "{{HOST}}/taxonomies/:id", + "request": { + "method": "PATCH", + "header": [ + { + "key": "Authorization", + "type": "text", + "value": "Bearer {{token}}" + }, + { + "key": "Content-Type", + "name": "Content-Type", + "type": "text", + "value": "application/json" + } + ], + "body": { + "mode": "raw", + "raw": "{\n \"version\": \"1.0.1\",\n \"endSupportDate\": \"2031-01-01\"\n}" + }, + "url": { + "raw": "{{HOST}}/taxonomies/{{taxonomyId}}/metadata", + "host": [ + "{{HOST}}" + ], + "path": [ + "taxonomies", + "{{taxonomyId}}", + "metadata" + ] + } + }, + "response": [] + }, + { + "name": "{{HOST}}/taxonomies/:id", + "request": { + "method": "DELETE", + "header": [ + { + "key": "Authorization", + "type": "text", + "value": "Bearer {{token}}" + }, + { + "key": "Content-Type", + "name": "Content-Type", + "type": "text", + "value": "application/json" + } + ], + "body": { + "mode": "raw", + "raw": "[\n \"isLongTermSupport\",\n \"endSupportDate\"\n]" + }, + "url": { + "raw": "{{HOST}}/taxonomies/{{taxonomyId}}/metadata", + "host": [ + "{{HOST}}" + ], + "path": [ + "taxonomies", + "{{taxonomyId}}", + "metadata" + ] + } + }, + "response": [] + } + ] + }, + { + "name": "skills", + "item": [ + { + "name": "{{HOST}}/skills", + "event": [ + { + "listen": "test", + "script": { + "exec": [ + "var rsp = pm.response.json();", + "if(rsp.id) pm.environment.set(\"skillId\", rsp.id);" + ], + "type": "text/javascript" + } + } + ], + "request": { + "method": "POST", + "header": [ + { + "key": "Authorization", + "type": "text", + "value": "Bearer {{token}}" + }, + { + "key": "Content-Type", + "name": "Content-Type", + "type": "text", + "value": "application/json" + } + ], + "body": { + "mode": "raw", + "raw": "{\n\t\"taxonomyId\":\"{{taxonomyId}}\",\n\t\"name\":\"jump5\",\n\t\"uri\":\"http://www.google.com/",\n\t\"externalId\":\"externalId\",\n \"metadata\": {\n \"challengeProminence\": \"0.2\",\n \"memberProminence\": \"0.5\"\n }\n}" + }, + "url": { + "raw": "{{HOST}}/skills", + "host": [ + "{{HOST}}" + ], + "path": [ + "skills" + ] + } + }, + "response": [] + }, + { + "name": "{{HOST}}/skills/:id", + "request": { + "method": "GET", + "header": [], + "url": { + "raw": "{{HOST}}/skills/{{skillId}}", + "host": [ + "{{HOST}}" + ], + "path": [ + "skills", + "{{skillId}}" + ] + } + }, + "response": [] + }, + { + "name": "{{HOST}}/skills/:id", + "request": { + "method": "PATCH", + "header": [ + { + "key": "Authorization", + "type": "text", + "value": "Bearer {{token}}" + }, + { + "key": "Content-Type", + "name": "Content-Type", + "type": "text", + "value": "application/json" + } + ], + "body": { + "mode": "raw", + "raw": "{\n\t\"name\":\"skill_name_update23\"\n}" + }, + "url": { + "raw": "{{HOST}}/skills/{{skillId}}", + "host": [ + "{{HOST}}" + ], + "path": [ + "skills", + "{{skillId}}" + ] + } + }, + "response": [] + }, + { + "name": "{{HOST}}/skills/:id", + "request": { + "method": "HEAD", + "header": [ + { + "key": "Content-Type", + "name": "Content-Type", + "type": "text", + "value": "application/json" + } + ], + "url": { + "raw": "{{HOST}}/skills/{{skillId}}", + "host": [ + "{{HOST}}" + ], + "path": [ + "skills", + "{{skillId}}" + ] + } + }, + "response": [] + }, + { + "name": "{{HOST}}/skills", + "request": { + "method": "GET", + "header": [], + "url": { + "raw": "{{HOST}}/skills", + "host": [ + "{{HOST}}" + ], + "path": [ + "skills" + ], + "query": [ + { + "key": "perPage", + "value": "2", + "disabled": true + } + ] + } + }, + "response": [] + }, + { + "name": "{{HOST}}/skills", + "request": { + "method": "HEAD", + "header": [], + "url": { + "raw": "{{HOST}}/skills", + "host": [ + "{{HOST}}" + ], + "path": [ + "skills" + ] + } + }, + "response": [] + }, + { + "name": "{{HOST}}/skills/:id", + "request": { + "method": "DELETE", + "header": [ + { + "key": "Authorization", + "type": "text", + "value": "Bearer {{token}}" + }, + { + "key": "Content-Type", + "name": "Content-Type", + "type": "text", + "value": "application/json" + } + ], + "body": { + "mode": "raw", + "raw": "" + }, + "url": { + "raw": "{{HOST}}/skills/{{skillId}}", + "host": [ + "{{HOST}}" + ], + "path": [ + "skills", + "{{skillId}}" + ] + } + }, + "response": [] + } + ] + }, + { + "name": "skills metadata", + "item": [ + { + "name": "{{HOST}}/skills/:id", + "request": { + "method": "PUT", + "header": [ + { + "key": "Authorization", + "type": "text", + "value": "Bearer {{token}}" + }, + { + "key": "Content-Type", + "name": "Content-Type", + "type": "text", + "value": "application/json" + } + ], + "body": { + "mode": "raw", + "raw": "{\n \"version\": \"1.0.1\",\n \"isLongTermSupport\": true\n}" + }, + "url": { + "raw": "{{HOST}}/skills/{{skillId}}/metadata", + "host": [ + "{{HOST}}" + ], + "path": [ + "skills", + "{{skillId}}", + "metadata" + ] + } + }, + "response": [] + }, + { + "name": "{{HOST}}/skills/:id", + "request": { + "method": "PATCH", + "header": [ + { + "key": "Authorization", + "type": "text", + "value": "Bearer {{token}}" + }, + { + "key": "Content-Type", + "name": "Content-Type", + "type": "text", + "value": "application/json" + } + ], + "body": { + "mode": "raw", + "raw": "{\n \"version\": \"1.0.1\",\n \"endSupportDate\": \"2031-01-01\"\n}" + }, + "url": { + "raw": "{{HOST}}/skills/{{skillId}}/metadata", + "host": [ + "{{HOST}}" + ], + "path": [ + "skills", + "{{skillId}}", + "metadata" + ] + } + }, + "response": [] + }, + { + "name": "{{HOST}}/skills/:id", + "request": { + "method": "DELETE", + "header": [ + { + "key": "Authorization", + "type": "text", + "value": "Bearer {{token}}" + }, + { + "key": "Content-Type", + "name": "Content-Type", + "type": "text", + "value": "application/json" + } + ], + "body": { + "mode": "raw", + "raw": "[\n \"isLongTermSupport\",\n \"endSupportDate\"\n]" + }, + "url": { + "raw": "{{HOST}}/skills/{{skillId}}/metadata", + "host": [ + "{{HOST}}" + ], + "path": [ + "skills", + "{{skillId}}", + "metadata" + ] + } + }, + "response": [] + } + ] + } + ] +} \ No newline at end of file diff --git a/docs/skills-api.postman_environment.json b/docs/skills-api.postman_environment.json new file mode 100644 index 0000000..c616073 --- /dev/null +++ b/docs/skills-api.postman_environment.json @@ -0,0 +1,29 @@ +{ + "id": "d5d8ae19-2fe4-4044-a541-c5845bfaf650", + "name": "skills-api", + "values": [ + { + "key": "HOST", + "value": "http://127.0.0.1:3001/api/1.0", + "enabled": true + }, + { + "key": "token", + "value": "_abc_", + "enabled": true + }, + { + "key": "skillId", + "value": "e72e432f-6ef4-4f19-962b-eb59b805b317", + "enabled": true + }, + { + "key": "taxonomyId", + "value": "48777845-28ff-42ad-80eb-491cf30f7f49", + "enabled": true + } + ], + "_postman_variable_scope": "environment", + "_postman_exported_at": "2021-07-22T04:01:01.670Z", + "_postman_exported_using": "Postman/8.1.0" +} diff --git a/docs/swagger.yaml b/docs/swagger.yaml new file mode 100644 index 0000000..13ce870 --- /dev/null +++ b/docs/swagger.yaml @@ -0,0 +1,1082 @@ +--- +swagger: "2.0" +info: + description: > + API for consolidated skills services to be presented to the various presentation and analysis layers of the platform. + + + **PAGINATION** + Requests that return multiple items will be paginated to 20 items by default. You can specify further pages with the `page` parameter. You can also set a custom page size up to 100 with the `perPage` parameter. + + + Pagination response data is included in http headers. By Default, the response header contains links with `next`, `last`, `first`, `prev` resource links. + + **Point to note** - the page attributes will have no effect if the data is fetched from db and not from es. + + + + **PERMISSION** + + Some endpoints require specific permission to access. + + + The required permission for each endpoint is mentioned in the endpoint's description section. + + + For more information about all available permissions for this project, please look at `doc/permission.html` under the project root directory. + version: "1.0.0" + title: "Skills API" +basePath: "/api/1.0" +tags: + - name: "Skills" + description: "Skills registered in the system" + - name: "Skill Metadata" + description: "Metadata for Skills registered in the system" + - name: "Taxonomies" + description: "Taxonomies registered in the system" + - name: "Taxonomy Metadata" + description: "Metadata for Taxonomies registered in the system" +schemes: + - "https" +consumes: + - "application/json" +produces: + - "application/json" +paths: + /skills: + get: + tags: + - "Skills" + description: > + Get list of skills in the application. + + If no results, then empty array is returned. + + Multiple filters are supported. + operationId: "skillsGET" + parameters: + - $ref: '#/parameters/page' + - $ref: '#/parameters/perPage' + - name: "taxonomyId" + in: "query" + description: "The referenced taxonomy id" + type: "string" + format: "UUID" + - name: "externalId" + in: "query" + description: "The external id of the skill" + type: "string" + - name: "name" + in: "query" + description: "The name of the skill" + type: "string" + - name: "orderBy" + in: "query" + type: "string" + description: "Specify by which field to sort by. Sorts in ascending order only" + responses: + "200": + description: "OK - the request was successful" + schema: + type: "array" + items: + $ref: "#/definitions/Skill" + headers: + X-Next-Page: + type: integer + description: The index of the next page + X-Page: + type: integer + description: The index of the current page (starting at 1) + X-Per-Page: + type: integer + description: The number of items to list per page + X-Prev-Page: + type: integer + description: The index of the previous page + X-Total: + type: integer + description: The total number of items + X-Total-Pages: + type: integer + description: The total number of pages + Link: + type: string + description: Pagination link header. + "400": + $ref: "#/definitions/BadRequest" + "500": + $ref: "#/definitions/ServerError" + x-swagger-router-controller: "Skills" + head: + tags: + - "Skills" + description: > + Retrieve header information for get operation on Skills in the application. + operationId: "skillsHEAD" + parameters: + - $ref: '#/parameters/page' + - $ref: '#/parameters/perPage' + - name: "taxonomyId" + in: "query" + description: "The referenced taxonomy id" + type: "string" + format: "UUID" + - name: "externalId" + in: "query" + description: "The external id of the skill" + type: "string" + - name: "name" + in: "query" + description: "The name of the skill" + type: "string" + responses: + "200": + description: "Success response" + headers: + X-Next-Page: + type: integer + description: The index of the next page + X-Page: + type: integer + description: The index of the current page (starting at 1) + X-Per-Page: + type: integer + description: The number of items to list per page + X-Prev-Page: + type: integer + description: The index of the previous page + X-Total: + type: integer + description: The total number of items + X-Total-Pages: + type: integer + description: The total number of pages + Link: + type: string + description: Pagination link header. + "400": + $ref: "#/definitions/BadRequest" + "500": + $ref: "#/definitions/ServerError" + x-swagger-router-controller: "Skills" + post: + tags: + - "Skills" + description: > + Create a new Skill. + + + **PERMISSION** + + 1. permission `CREATE_SKILL` is required to perform this operation. + + 2. permission `ADD_SKILL_METADATA` is required if any metadata field is provided. + (**Note** providing `metadata: {}` inside request body is treated as no metadata provided) + operationId: "skillsPOST" + parameters: + - in: "body" + name: "body" + required: true + schema: + $ref: "#/definitions/SkillRequestBody" + responses: + "200": + description: "OK - the request was successful" + schema: + $ref: "#/definitions/Skill" + "400": + $ref: "#/definitions/BadRequest" + "401": + $ref: "#/definitions/Unauthorized" + "403": + $ref: "#/definitions/Forbidden" + "409": + $ref: "#/definitions/Conflict" + "500": + $ref: "#/definitions/ServerError" + security: + - Bearer: [] + x-swagger-router-controller: "Skills" + /skills/{skillId}: + get: + tags: + - "Skills" + description: > + Get Skill by given skill id. + operationId: "skillsSkillIdGET" + parameters: + - name: "skillId" + in: "path" + description: "The skill id" + required: true + type: "string" + format: "UUID" + responses: + "200": + description: "OK - the request was successful" + schema: + $ref: "#/definitions/Skill" + "400": + $ref: "#/definitions/BadRequest" + "404": + $ref: "#/definitions/NotFound" + "500": + $ref: "#/definitions/ServerError" + x-swagger-router-controller: "Skills" + head: + tags: + - "Skills" + description: > + Retrieve header information for get operation on Skill by its Id in the application. + operationId: "skillsSkillIdHEAD" + parameters: + - name: "skillId" + in: "path" + description: "The user id" + required: true + type: "string" + format: "UUID" + responses: + "200": + description: "OK - the request was successful" + "400": + $ref: "#/definitions/BadRequest" + "404": + $ref: "#/definitions/NotFound" + "500": + $ref: "#/definitions/ServerError" + x-swagger-router-controller: "Skills" + delete: + tags: + - "Skills" + description: > + Remove an existing skill with given id. + + + **PERMISSION** + + Permission `DELETE_SKILL` is required to perform this operation. + operationId: "skillsSkillIdDELETE" + parameters: + - name: "skillId" + in: "path" + description: "The skill id" + required: true + type: "string" + format: "UUID" + responses: + "204": + description: "OK - the request was successful" + "401": + $ref: "#/definitions/Unauthorized" + "403": + $ref: "#/definitions/Forbidden" + "404": + $ref: "#/definitions/NotFound" + "409": + $ref: "#/definitions/Conflict" + "500": + $ref: "#/definitions/ServerError" + security: + - Bearer: [] + x-swagger-router-controller: "Skills" + patch: + tags: + - "Skills" + description: > + Update an existing skill with given id. + + + **PERMISSION** + + 1. permission `UPDATE_SKILL` is required to perform this operation + + 2. permission `ADD_SKILL_METADATA` is required if any metadata field is provided. + + 3. permission `DELETE_SKILL_METADATA` is required if there is any existing metadata field in the skill. + operationId: "skillsSkillIdPATCH" + parameters: + - name: "skillId" + in: "path" + description: "The skill id" + required: true + type: "string" + format: "UUID" + - in: "body" + name: "body" + required: true + schema: + $ref: "#/definitions/SkillUpdateRequestBody" + responses: + "200": + description: "OK - the request was successful" + schema: + $ref: "#/definitions/Skill" + "400": + $ref: "#/definitions/BadRequest" + "401": + $ref: "#/definitions/Unauthorized" + "403": + $ref: "#/definitions/Forbidden" + "404": + $ref: "#/definitions/NotFound" + "409": + $ref: "#/definitions/Conflict" + "500": + $ref: "#/definitions/ServerError" + security: + - Bearer: [] + x-swagger-router-controller: "Skills" + + /skills/{skillId}/metadata: + put: + tags: + - "Skill Metadata" + description: > + Fully update the metadata of an existing skill with given id. + + Used to entirely overwrite existing metadata. + + + **PERMISSION** + + 1. permission `ADD_SKILL_METADATA` is required if any metadata field is provided. + + 2. permission `DELETE_SKILL_METADATA` is required if there is any existing metadata field in the skill. + parameters: + - name: "skillId" + in: "path" + description: "The skill id" + required: true + type: "string" + format: "UUID" + - in: "body" + name: "body" + required: true + schema: + $ref: "#/definitions/SkillMetadataUpdateRequestBody" + responses: + "200": + description: "OK - the request was successful" + schema: + $ref: "#/definitions/Skill" + "400": + $ref: "#/definitions/BadRequest" + "401": + $ref: "#/definitions/Unauthorized" + "403": + $ref: "#/definitions/Forbidden" + "404": + $ref: "#/definitions/NotFound" + "500": + $ref: "#/definitions/ServerError" + security: + - Bearer: [] + patch: + tags: + - "Skill Metadata" + description: > + Partically update the metadata of an existing skill with given id. + + Used to update existing fields in metadata and to add new fields to metadata. + + + **PERMISSION** + + 1. permission `ADD_SKILL_METADATA` is required if any new metadata field is provided. + + 2. permission `UPDATE_SKILL_METADATA` is required if any metadata field provided already exist in the skill. + parameters: + - name: "skillId" + in: "path" + description: "The skill id" + required: true + type: "string" + format: "UUID" + - in: "body" + name: "body" + required: true + schema: + $ref: "#/definitions/SkillMetadataUpdateRequestBody" + responses: + "200": + description: "OK - the request was successful" + schema: + $ref: "#/definitions/Skill" + "400": + $ref: "#/definitions/BadRequest" + "401": + $ref: "#/definitions/Unauthorized" + "403": + $ref: "#/definitions/Forbidden" + "404": + $ref: "#/definitions/NotFound" + "500": + $ref: "#/definitions/ServerError" + security: + - Bearer: [] + delete: + tags: + - "Skill Metadata" + description: > + Remove fields from the metadata of an existing skill with given id. + + 404 error will be raised if one or more of the provided fields is missing in the existing metadata. + + + **PERMISSION** + + Permission `DELETE_SKILL_METADATA` is required to perform this operation. + parameters: + - name: "skillId" + in: "path" + description: "The skill id" + required: true + type: "string" + format: "UUID" + - in: "body" + name: "body" + required: true + schema: + $ref: "#/definitions/SkillMetadataDeleteRequestBody" + responses: + "200": + description: "OK - the request was successful" + schema: + $ref: "#/definitions/Skill" + "400": + $ref: "#/definitions/BadRequest" + "401": + $ref: "#/definitions/Unauthorized" + "403": + $ref: "#/definitions/Forbidden" + "404": + $ref: "#/definitions/NotFound" + "500": + $ref: "#/definitions/ServerError" + security: + - Bearer: [] + + /taxonomies: + get: + tags: + - "Taxonomies" + description: > + Search Taxonomy in the application. If no results, then empty array is returned. + operationId: "skillstaxonomiesGET" + parameters: + - $ref: '#/parameters/page' + - $ref: '#/parameters/perPage' + - name: "name" + in: "query" + description: "Filter by taxonomy name" + required: false + type: "string" + responses: + "200": + description: "OK - the request was successful" + schema: + type: "array" + items: + $ref: "#/definitions/Taxonomy" + headers: + X-Next-Page: + type: integer + description: The index of the next page + X-Page: + type: integer + description: The index of the current page (starting at 1) + X-Per-Page: + type: integer + description: The number of items to list per page + X-Prev-Page: + type: integer + description: The index of the previous page + X-Total: + type: integer + description: The total number of items + X-Total-Pages: + type: integer + description: The total number of pages + Link: + type: string + description: Pagination link header. + "400": + $ref: "#/definitions/BadRequest" + "500": + $ref: "#/definitions/ServerError" + x-swagger-router-controller: "Taxonomy" + head: + tags: + - "Taxonomies" + description: > + Retrieve header information for a search operation on taxonomies in the application. + operationId: "skillstaxonomiesHEAD" + parameters: + - name: "name" + in: "query" + description: "Filter by taxonomy name" + required: false + type: "string" + responses: + "200": + description: "Success response" + headers: + X-Next-Page: + type: integer + description: The index of the next page + X-Page: + type: integer + description: The index of the current page (starting at 1) + X-Per-Page: + type: integer + description: The number of items to list per page + X-Prev-Page: + type: integer + description: The index of the previous page + X-Total: + type: integer + description: The total number of items + X-Total-Pages: + type: integer + description: The total number of pages + Link: + type: string + description: Pagination link header. + "400": + $ref: "#/definitions/BadRequest" + "500": + $ref: "#/definitions/ServerError" + x-swagger-router-controller: "Taxonomy" + post: + tags: + - "Taxonomies" + description: > + Create a new Taxonomy. + + + **PERMISSION** + + 1. permission `CREATE_TAXONOMY` is required to perform this operation. + + 2. permission `ADD_TAXONOMY_METADATA` is required if any metadata field is provided. + (**Note** providing `metadata: {}` inside request body is treated as no metadata provided) + operationId: "skillstaxonomiesPOST" + parameters: + - in: "body" + name: "body" + required: true + schema: + $ref: "#/definitions/TaxonomyRequestBody" + responses: + "200": + description: "OK - the request was successful" + schema: + $ref: "#/definitions/Taxonomy" + "400": + $ref: "#/definitions/BadRequest" + "401": + $ref: "#/definitions/Unauthorized" + "403": + $ref: "#/definitions/Forbidden" + "409": + $ref: "#/definitions/Conflict" + "500": + $ref: "#/definitions/ServerError" + security: + - Bearer: [] + x-swagger-router-controller: "Taxonomy" + /taxonomies/{taxonomyId}: + get: + tags: + - "Taxonomies" + description: > + Get taxonomy with given id. + operationId: "skillstaxonomiestaxonomyIdGET" + parameters: + - name: "taxonomyId" + in: "path" + description: "The taxonomy id" + required: true + type: "string" + format: "UUID" + responses: + "200": + description: "OK - the request was successful" + schema: + $ref: "#/definitions/Taxonomy" + "400": + $ref: "#/definitions/BadRequest" + "404": + $ref: "#/definitions/NotFound" + "500": + $ref: "#/definitions/ServerError" + x-swagger-router-controller: "Taxonomy" + head: + tags: + - "Taxonomies" + description: > + Get taxonomy with given id, but only header information is returned. + operationId: "skillstaxonomiestaxonomyIdHEAD" + parameters: + - name: "taxonomyId" + in: "path" + description: "The taxonomy id" + required: true + type: "string" + format: "UUID" + responses: + "200": + description: "OK - the request was successful" + "400": + $ref: "#/definitions/BadRequest" + "404": + $ref: "#/definitions/NotFound" + "500": + $ref: "#/definitions/ServerError" + x-swagger-router-controller: "Taxonomy" + delete: + tags: + - "Taxonomies" + description: > + Remove an existing taxonomy with given id. + + + **PERMISSION** + + Permission `DELETE_TAXONOMY` is required to perform this operation. + operationId: "skillstaxonomiestaxonomyIdDELETE" + parameters: + - name: "taxonomyId" + in: "path" + description: "The taxonomy id" + required: true + type: "string" + format: "UUID" + responses: + "204": + description: "OK - the request was successful" + "401": + $ref: "#/definitions/Unauthorized" + "403": + $ref: "#/definitions/Forbidden" + "404": + $ref: "#/definitions/NotFound" + "409": + $ref: "#/definitions/Conflict" + "500": + $ref: "#/definitions/ServerError" + security: + - Bearer: [] + x-swagger-router-controller: "Taxonomy" + patch: + tags: + - "Taxonomies" + description: > + Update an existing taxonomy with given id. + + + **PERMISSION** + + 1. permission `UPDATE_TAXONOMY` is required to perform this operation + + 2. permission `ADD_TAXONOMY_METADATA` is required if any metadata field is provided. + + 3. permission `DELETE_TAXONOMY_METADATA` is required if there is any existing metadata field in the taxonomy. + operationId: "skillstaxonomiestaxonomyIdPATCH" + parameters: + - name: "taxonomyId" + in: "path" + description: "The taxonomy id" + required: true + type: "string" + format: "UUID" + - in: "body" + name: "body" + required: true + schema: + $ref: "#/definitions/TaxonomyUpdateRequestBody" + responses: + "200": + description: "OK - the request was successful" + schema: + $ref: "#/definitions/Taxonomy" + "400": + $ref: "#/definitions/BadRequest" + "401": + $ref: "#/definitions/Unauthorized" + "403": + $ref: "#/definitions/Forbidden" + "404": + $ref: "#/definitions/NotFound" + "409": + $ref: "#/definitions/Conflict" + "500": + $ref: "#/definitions/ServerError" + security: + - Bearer: [] + x-swagger-router-controller: "Taxonomy" + /taxonomies/{taxonomyId}/metadata: + put: + tags: + - "Taxonomy Metadata" + description: > + Fully update the metadata of an existing taxonomy with given id. + + Used to entirely overwrite existing metadata. + + + **PERMISSION** + + 1. permission `ADD_TAXONOMY_METADATA` is required if any metadata field is provided. + + 2. permission `DELETE_TAXONOMY_METADATA` is required if there is any existing metadata field in the taxonomy. + parameters: + - name: "taxonomyId" + in: "path" + description: "The taxonomy id" + required: true + type: "string" + format: "UUID" + - in: "body" + name: "body" + required: true + schema: + $ref: "#/definitions/TaxonomyMetadataUpdateRequestBody" + responses: + "200": + description: "OK - the request was successful" + schema: + $ref: "#/definitions/Taxonomy" + "400": + $ref: "#/definitions/BadRequest" + "401": + $ref: "#/definitions/Unauthorized" + "403": + $ref: "#/definitions/Forbidden" + "404": + $ref: "#/definitions/NotFound" + "500": + $ref: "#/definitions/ServerError" + security: + - Bearer: [] + patch: + tags: + - "Taxonomy Metadata" + description: > + Partically update the metadata of an existing taxonomy with given id. + + Used to update existing fields in metadata and to add new fields to metadata. + + + **PERMISSION** + + 1. permission `ADD_TAXONOMY_METADATA` is required if any new metadata field is provided. + + 2. permission `UPDATE_TAXONOMY_METADATA` is required if any metadata field provided already exist in the taxonomy. + parameters: + - name: "taxonomyId" + in: "path" + description: "The taxonomy id" + required: true + type: "string" + format: "UUID" + - in: "body" + name: "body" + required: true + schema: + $ref: "#/definitions/TaxonomyMetadataUpdateRequestBody" + responses: + "200": + description: "OK - the request was successful" + schema: + $ref: "#/definitions/Taxonomy" + "400": + $ref: "#/definitions/BadRequest" + "401": + $ref: "#/definitions/Unauthorized" + "403": + $ref: "#/definitions/Forbidden" + "404": + $ref: "#/definitions/NotFound" + "500": + $ref: "#/definitions/ServerError" + security: + - Bearer: [] + delete: + tags: + - "Taxonomy Metadata" + description: > + Remove fields from the metadata of an existing taxonomy with given id. + + 404 error will be raised if one or more of the provided fields is missing in the existing metadata. + + + **PERMISSION** + + Permission `DELETE_TAXONOMY_METADATA` is required to perform this operation. + parameters: + - name: "taxonomyId" + in: "path" + description: "The taxonomy id" + required: true + type: "string" + format: "UUID" + - in: "body" + name: "body" + required: true + schema: + $ref: "#/definitions/TaxonomyMetadataDeleteRequestBody" + responses: + "200": + description: "OK - the request was successful" + schema: + $ref: "#/definitions/Taxonomy" + "400": + $ref: "#/definitions/BadRequest" + "401": + $ref: "#/definitions/Unauthorized" + "403": + $ref: "#/definitions/Forbidden" + "404": + $ref: "#/definitions/NotFound" + "500": + $ref: "#/definitions/ServerError" + security: + - Bearer: [] + +securityDefinitions: + Bearer: + type: "apiKey" + name: "Authorization" + in: "header" + +parameters: + page: + name: page + in: query + description: The page number. + required: false + type: integer + default: 1 + perPage: + name: perPage + in: query + description: The number of items to list per page. + required: false + type: integer + default: 20 + maximum: 100 + +definitions: + Skill: + allOf: + - type: "object" + required: + - "externalId" + - "id" + - "name" + - "taxonomyId" + - "taxonomyName" + - "uri" + - "metadata" + properties: + id: + type: "string" + format: "UUID" + description: "The skill id" + taxonomyId: + type: "string" + format: "UUID" + description: "The referenced taxonomy id" + taxonomyName: + type: "string" + description: "The referenced taxonomy name" + name: + type: "string" + description: "The name of the skill" + externalId: + type: "string" + description: "The external id for the skill" + uri: + type: "string" + description: "The uri for the skill" + metadata: + type: "object" + description: "The metadata for the skill" + properties: + updated: + type: "string" + format: "date-time" + description: "The last updated timestamp of the skill" + challengeProminence: + type: "string" + description: "The challenge prominence ranging from [0, 1]" + memberProminence: + type: "string" + description: "The member prominence ranging from [0, 1]" + SkillRequestBody: + allOf: + - type: "object" + - $ref: "#/definitions/SkillUpdateRequestBody" + SkillUpdateRequestBody: + type: "object" + properties: + taxonomyId: + type: "string" + format: "UUID" + description: "The id of taxonomy for this Skill." + name: + type: "string" + description: "Name of Skill" + uri: + type: "string" + description: "Uri of Skill" + externalId: + type: "string" + description: "External Id of skill" + metadata: + type: "object" + description: "The metadata for the skill" + properties: + challengeProminence: + type: "string" + description: "The challenge prominence ranging from [0, 1]" + memberProminence: + type: "string" + description: "The member prominence ranging from [0, 1]" + example: + taxonomyId: "taxonomyId" + name: "name" + externalId: "externalId" + uri: "uri" + metadata: + challengeProminence: "challengeProminence" + memberProminence: "memberProminence" + SkillMetadataUpdateRequestBody: + type: "object" + description: "The metadata for a skill. Can contain arbitrary fields" + properties: + challengeProminence: + type: "string" + description: "The challenge prominence ranging from [0, 1]" + memberProminence: + type: "string" + description: "The member prominence ranging from [0, 1]" + example: + verions: "1.0.0" + random_field: "random_value" + SkillMetadataDeleteRequestBody: + type: "array" + description: "The list of fields to be removed from the metadata for a skill." + items: + type: string + description: "field name" + example: ["version", "random_field"] + Taxonomy: + allOf: + - type: "object" + required: + - "id" + - "name" + properties: + id: + type: "string" + format: "UUID" + description: "The id of the taxonomy." + name: + type: "string" + description: "The name of the taxonomy." + metadata: + type: "object" + description: "The metadata of the taxonomy." + properties: + updated: + type: "string" + format: "date-time" + description: "The last updated timestamp of the taxonomy" + TaxonomyRequestBody: + allOf: + - type: "object" + - $ref: "#/definitions/TaxonomyUpdateRequestBody" + TaxonomyUpdateRequestBody: + type: "object" + properties: + id: + type: "string" + format: "UUID" + description: "The id of the taxonomy." + name: + type: "string" + description: "The name of the taxonomy." + metadata: + type: "object" + description: "The metadata of the taxonomy." + example: + name: "name" + TaxonomyMetadataUpdateRequestBody: + type: "object" + description: "The metadata for a taxonomy. Can contain arbitrary fields" + example: + verions: "1.0.0" + random_field: "random_value" + TaxonomyMetadataDeleteRequestBody: + type: "array" + description: "The list of fields to be removed from the metadata for a taxonomy." + items: + type: string + description: "field name" + example: ["version", "random_field"] + Unauthorized: + type: "object" + properties: + message: + type: "string" + example: "Unable to authenticate the user." + description: "The unauthorized error message." + description: "The unauthorized error entity." + NotFound: + type: "object" + properties: + message: + type: "string" + example: "A resource with the name could not be found." + description: "The not found error message." + description: "The not found error entity." + ServerError: + type: "object" + properties: + message: + type: "string" + example: "Something went wrong while processing your request. We're sorry\ + \ for the trouble. We've been notified of the error and will correct it\ + \ as soon as possible. Please try your request again in a moment." + description: "The server error message." + description: "The server error entity." + BadRequest: + type: "object" + properties: + message: + type: "string" + example: "Invalid input." + description: "The bad request error message." + description: "The bad request error entity." + Forbidden: + type: "object" + properties: + message: + type: "string" + example: "You are not allowed to access the request." + description: "The forbidden error message." + description: "The permission error entity." + Conflict: + type: "object" + required: + - "message" + properties: + message: + type: "string" + example: "Creating a resource with a name already exists." + description: "The conflict error message." + description: "The conflict error entity." diff --git a/package-lock.json b/package-lock.json new file mode 100644 index 0000000..9da86ce --- /dev/null +++ b/package-lock.json @@ -0,0 +1,4938 @@ +{ + "name": "skills-api", + "version": "1.0.0", + "lockfileVersion": 1, + "requires": true, + "dependencies": { + "@babel/code-frame": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.14.5.tgz", + "integrity": "sha512-9pzDqyc6OLDaqe+zbACgFkb6fKMNG6CObKpnYXChRsvYGyEdc7CA2BaqeOM+vOtCS5ndmJicPJhKAwYRI6UfFw==", + "dev": true, + "requires": { + "@babel/highlight": "^7.14.5" + } + }, + "@babel/helper-validator-identifier": { + "version": "7.14.8", + "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.14.8.tgz", + "integrity": "sha512-ZGy6/XQjllhYQrNw/3zfWRwZCTVSiBLZ9DHVZxn9n2gip/7ab8mv2TWlKPIBk26RwedCBoWdjLmn+t9na2Gcow==", + "dev": true + }, + "@babel/highlight": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/highlight/-/highlight-7.14.5.tgz", + "integrity": "sha512-qf9u2WFWVV0MppaL877j2dBtQIDgmidgjGk5VIMw3OadXvYaXn66U1BFlH2t4+t3i+8PhedppRv+i40ABzd+gg==", + "dev": true, + "requires": { + "@babel/helper-validator-identifier": "^7.14.5", + "chalk": "^2.0.0", + "js-tokens": "^4.0.0" + } + }, + "@dabh/diagnostics": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/@dabh/diagnostics/-/diagnostics-2.0.2.tgz", + "integrity": "sha512-+A1YivoVDNNVCdfozHSR8v/jyuuLTMXwjWuxPFlFlUapXoGc+Gj9mDlTDDfrwl7rXCl2tNZ0kE8sIBO6YOn96Q==", + "requires": { + "colorspace": "1.1.x", + "enabled": "2.0.x", + "kuler": "^2.0.0" + } + }, + "@elastic/elasticsearch": { + "version": "7.13.0", + "resolved": "https://registry.npmjs.org/@elastic/elasticsearch/-/elasticsearch-7.13.0.tgz", + "integrity": "sha512-WgwLWo2p9P2tdqzBGX9fHeG8p5IOTXprXNTECQG2mJ7z9n93N5AFBJpEw4d35tWWeCWi9jI13A2wzQZH7XZ/xw==", + "requires": { + "debug": "^4.3.1", + "hpagent": "^0.1.1", + "ms": "^2.1.3", + "secure-json-parse": "^2.4.0" + } + }, + "@hapi/address": { + "version": "2.1.4", + "resolved": "https://registry.npmjs.org/@hapi/address/-/address-2.1.4.tgz", + "integrity": "sha512-QD1PhQk+s31P1ixsX0H0Suoupp3VMXzIVMSwobR3F3MSUO2YCV0B7xqLcUw/Bh8yuvd3LhpyqLQWTNcRmp6IdQ==" + }, + "@hapi/formula": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/@hapi/formula/-/formula-1.2.0.tgz", + "integrity": "sha512-UFbtbGPjstz0eWHb+ga/GM3Z9EzqKXFWIbSOFURU0A/Gku0Bky4bCk9/h//K2Xr3IrCfjFNhMm4jyZ5dbCewGA==" + }, + "@hapi/hoek": { + "version": "8.5.1", + "resolved": "https://registry.npmjs.org/@hapi/hoek/-/hoek-8.5.1.tgz", + "integrity": "sha512-yN7kbciD87WzLGc5539Tn0sApjyiGHAJgKvG9W8C7O+6c7qmoQMfVs0W4bX17eqz6C78QJqqFrtgdK5EWf6Qow==" + }, + "@hapi/joi": { + "version": "16.1.8", + "resolved": "https://registry.npmjs.org/@hapi/joi/-/joi-16.1.8.tgz", + "integrity": "sha512-wAsVvTPe+FwSrsAurNt5vkg3zo+TblvC5Bb1zMVK6SJzZqw9UrJnexxR+76cpePmtUZKHAPxcQ2Bf7oVHyahhg==", + "requires": { + "@hapi/address": "^2.1.2", + "@hapi/formula": "^1.2.0", + "@hapi/hoek": "^8.2.4", + "@hapi/pinpoint": "^1.0.2", + "@hapi/topo": "^3.1.3" + } + }, + "@hapi/pinpoint": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/@hapi/pinpoint/-/pinpoint-1.0.2.tgz", + "integrity": "sha512-dtXC/WkZBfC5vxscazuiJ6iq4j9oNx1SHknmIr8hofarpKUZKmlUVYVIhNVzIEgK5Wrc4GMHL5lZtt1uS2flmQ==" + }, + "@hapi/topo": { + "version": "3.1.6", + "resolved": "https://registry.npmjs.org/@hapi/topo/-/topo-3.1.6.tgz", + "integrity": "sha512-tAag0jEcjwH+P2quUfipd7liWCNX2F8NvYjQp2wtInsZxnMlypdw0FtAOLxtvvkO+GSRRbmNi8m/5y42PQJYCQ==", + "requires": { + "@hapi/hoek": "^8.3.0" + } + }, + "@sindresorhus/is": { + "version": "0.14.0", + "resolved": "https://registry.npmjs.org/@sindresorhus/is/-/is-0.14.0.tgz", + "integrity": "sha512-9NET910DNaIPngYnLLPeg+Ogzqsi9uM4mSboU5y6p8S5DzMTVEsJZrawi+BoDNUVBa2DhJqQYUFvMDfgU062LQ==", + "dev": true + }, + "@szmarczak/http-timer": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/@szmarczak/http-timer/-/http-timer-1.1.2.tgz", + "integrity": "sha512-XIB2XbzHTN6ieIjfIMV9hlVcfPU26s2vafYWQcZHWXHOxiaRZYEDKEwdl129Zyg50+foYV2jCgtrqSA6qNuNSA==", + "dev": true, + "requires": { + "defer-to-connect": "^1.0.1" + } + }, + "@tootallnate/once": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/@tootallnate/once/-/once-1.1.2.tgz", + "integrity": "sha512-RbzJvlNzmRq5c3O09UipeuXno4tA1FE6ikOjxZK0tuxVv3412l64l5t1W5pj4+rJq9vpkm/kwiR07aZXnsKPxw==" + }, + "@types/body-parser": { + "version": "1.19.1", + "resolved": "https://registry.npmjs.org/@types/body-parser/-/body-parser-1.19.1.tgz", + "integrity": "sha512-a6bTJ21vFOGIkwM0kzh9Yr89ziVxq4vYH2fQ6N8AeipEzai/cFK6aGMArIkUeIdRIgpwQa+2bXiLuUJCpSf2Cg==", + "requires": { + "@types/connect": "*", + "@types/node": "*" + } + }, + "@types/connect": { + "version": "3.4.35", + "resolved": "https://registry.npmjs.org/@types/connect/-/connect-3.4.35.tgz", + "integrity": "sha512-cdeYyv4KWoEgpBISTxWvqYsVy444DOqehiF3fM3ne10AmJ62RSyNkUnxMJXHQWRQQX2eR94m5y1IZyDwBjV9FQ==", + "requires": { + "@types/node": "*" + } + }, + "@types/express": { + "version": "4.17.13", + "resolved": "https://registry.npmjs.org/@types/express/-/express-4.17.13.tgz", + "integrity": "sha512-6bSZTPaTIACxn48l50SR+axgrqm6qXFIxrdAKaG6PaJk3+zuUr35hBlgT7vOmJcum+OEaIBLtHV/qloEAFITeA==", + "requires": { + "@types/body-parser": "*", + "@types/express-serve-static-core": "^4.17.18", + "@types/qs": "*", + "@types/serve-static": "*" + } + }, + "@types/express-jwt": { + "version": "0.0.42", + "resolved": "https://registry.npmjs.org/@types/express-jwt/-/express-jwt-0.0.42.tgz", + "integrity": "sha512-WszgUddvM1t5dPpJ3LhWNH8kfNN8GPIBrAGxgIYXVCEGx6Bx4A036aAuf/r5WH9DIEdlmp7gHOYvSM6U87B0ag==", + "requires": { + "@types/express": "*", + "@types/express-unless": "*" + } + }, + "@types/express-serve-static-core": { + "version": "4.17.24", + "resolved": "https://registry.npmjs.org/@types/express-serve-static-core/-/express-serve-static-core-4.17.24.tgz", + "integrity": "sha512-3UJuW+Qxhzwjq3xhwXm2onQcFHn76frIYVbTu+kn24LFxI+dEhdfISDFovPB8VpEgW8oQCTpRuCe+0zJxB7NEA==", + "requires": { + "@types/node": "*", + "@types/qs": "*", + "@types/range-parser": "*" + } + }, + "@types/express-unless": { + "version": "0.5.2", + "resolved": "https://registry.npmjs.org/@types/express-unless/-/express-unless-0.5.2.tgz", + "integrity": "sha512-Q74UyYRX/zIgl1HSp9tUX2PlG8glkVm+59r7aK4KGKzC5jqKIOX6rrVLRQrzpZUQ84VukHtRoeAuon2nIssHPQ==", + "requires": { + "@types/express": "*" + } + }, + "@types/mime": { + "version": "1.3.2", + "resolved": "https://registry.npmjs.org/@types/mime/-/mime-1.3.2.tgz", + "integrity": "sha512-YATxVxgRqNH6nHEIsvg6k2Boc1JHI9ZbH5iWFFv/MTkchz3b1ieGDa5T0a9RznNdI0KhVbdbWSN+KWWrQZRxTw==" + }, + "@types/node": { + "version": "16.4.3", + "resolved": "https://registry.npmjs.org/@types/node/-/node-16.4.3.tgz", + "integrity": "sha512-GKM4FLMkWDc0sfx7tXqPWkM6NBow1kge0fgQh0bOnlqo4iT1kvTvMEKE0c1RtUGnbLlGRXiAA8SumE//90uKAg==" + }, + "@types/qs": { + "version": "6.9.7", + "resolved": "https://registry.npmjs.org/@types/qs/-/qs-6.9.7.tgz", + "integrity": "sha512-FGa1F62FT09qcrueBA6qYTrJPVDzah9a+493+o2PCXsesWHIn27G98TsSMs3WPNbZIEj4+VJf6saSFpvD+3Zsw==" + }, + "@types/range-parser": { + "version": "1.2.4", + "resolved": "https://registry.npmjs.org/@types/range-parser/-/range-parser-1.2.4.tgz", + "integrity": "sha512-EEhsLsD6UsDM1yFhAvy0Cjr6VwmpMWqFBCb9w07wVugF7w9nfajxLuVmngTIpgS6svCnm6Vaw+MZhoDCKnOfsw==" + }, + "@types/serve-static": { + "version": "1.13.10", + "resolved": "https://registry.npmjs.org/@types/serve-static/-/serve-static-1.13.10.tgz", + "integrity": "sha512-nCkHGI4w7ZgAdNkrEu0bv+4xNV/XDqW+DydknebMOQwkpDGx8G+HTlj7R7ABI8i8nKxVw0wtKPi1D+lPOkh4YQ==", + "requires": { + "@types/mime": "^1", + "@types/node": "*" + } + }, + "abbrev": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/abbrev/-/abbrev-1.1.1.tgz", + "integrity": "sha512-nne9/IiQ/hzIhY6pdDnbBtz7DjPTKrY00P/zvPSm5pOFkl6xuGrGnXn/VtTNNfNtAfZ9/1RtehkszU9qcTii0Q==", + "dev": true + }, + "accepts": { + "version": "1.3.7", + "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.7.tgz", + "integrity": "sha512-Il80Qs2WjYlJIBNzNkK6KYqlVMTbZLXgHx2oT0pU/fjRHyEp+PEfEPY0R3WCwAGVOtauxh1hOxNgIf5bv7dQpA==", + "requires": { + "mime-types": "~2.1.24", + "negotiator": "0.6.2" + } + }, + "acorn": { + "version": "7.4.1", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-7.4.1.tgz", + "integrity": "sha512-nQyp0o1/mNdbTO1PO6kHkwSrmgZ0MT/jCCpNiwbUjGoRN4dlBhqJtoQuCnEOKzgTVwg0ZWiCoQy6SxMebQVh8A==", + "dev": true + }, + "acorn-jsx": { + "version": "5.3.2", + "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-5.3.2.tgz", + "integrity": "sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ==", + "dev": true + }, + "agent-base": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-6.0.2.tgz", + "integrity": "sha512-RZNwNclF7+MS/8bDg70amg32dyeZGZxiDuQmZxKLAlQjr3jGyLx+4Kkk58UO7D2QdgFIQCovuSuZESne6RG6XQ==", + "requires": { + "debug": "4" + } + }, + "ajv": { + "version": "6.12.6", + "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz", + "integrity": "sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==", + "requires": { + "fast-deep-equal": "^3.1.1", + "fast-json-stable-stringify": "^2.0.0", + "json-schema-traverse": "^0.4.1", + "uri-js": "^4.2.2" + } + }, + "ansi-align": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/ansi-align/-/ansi-align-3.0.0.tgz", + "integrity": "sha512-ZpClVKqXN3RGBmKibdfWzqCY4lnjEuoNzU5T0oEFpfd/z5qJHVarukridD4juLO2FXMiwUQxr9WqQtaYa8XRYw==", + "dev": true, + "requires": { + "string-width": "^3.0.0" + }, + "dependencies": { + "ansi-regex": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz", + "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==", + "dev": true + }, + "emoji-regex": { + "version": "7.0.3", + "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-7.0.3.tgz", + "integrity": "sha512-CwBLREIQ7LvYFB0WyRvwhq5N5qPhc6PMjD6bYggFlI5YyDgl+0vxq5VHbMOFqLg7hfWzmu8T5Z1QofhmTIhItA==", + "dev": true + }, + "is-fullwidth-code-point": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", + "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=", + "dev": true + }, + "string-width": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz", + "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==", + "dev": true, + "requires": { + "emoji-regex": "^7.0.1", + "is-fullwidth-code-point": "^2.0.0", + "strip-ansi": "^5.1.0" + } + }, + "strip-ansi": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", + "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", + "dev": true, + "requires": { + "ansi-regex": "^4.1.0" + } + } + } + }, + "ansi-escapes": { + "version": "4.3.2", + "resolved": "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-4.3.2.tgz", + "integrity": "sha512-gKXj5ALrKWQLsYG9jlTRmR/xKluxHV+Z9QEwNIgCfM1/uwPMCuzVVnh5mwTd+OuBZcwSIMbqssNWRm1lE51QaQ==", + "dev": true, + "requires": { + "type-fest": "^0.21.3" + }, + "dependencies": { + "type-fest": { + "version": "0.21.3", + "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.21.3.tgz", + "integrity": "sha512-t0rzBq87m3fVcduHDUFhKmyyX+9eo6WQjZvf51Ea/M0Q7+T374Jp1aUiyUl0GKxp8M/OETVHSDvmkyPgvX+X2w==", + "dev": true + } + } + }, + "ansi-regex": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz", + "integrity": "sha1-w7M6te42DYbg5ijwRorn7yfWVN8=" + }, + "ansi-styles": { + "version": "3.2.1", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", + "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", + "dev": true, + "requires": { + "color-convert": "^1.9.0" + } + }, + "any-promise": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/any-promise/-/any-promise-1.3.0.tgz", + "integrity": "sha1-q8av7tzqUugJzcA3au0845Y10X8=" + }, + "anymatch": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.2.tgz", + "integrity": "sha512-P43ePfOAIupkguHUycrc4qJ9kz8ZiuOUijaETwX7THt0Y/GNK7v0aa8rY816xWjZ7rJdA5XdMcpVFTKMq+RvWg==", + "dev": true, + "requires": { + "normalize-path": "^3.0.0", + "picomatch": "^2.0.4" + } + }, + "argparse": { + "version": "1.0.10", + "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.10.tgz", + "integrity": "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==", + "requires": { + "sprintf-js": "~1.0.2" + } + }, + "array-flatten": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz", + "integrity": "sha1-ml9pkFGx5wczKPKgCJaLZOopVdI=" + }, + "array-includes": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/array-includes/-/array-includes-3.1.3.tgz", + "integrity": "sha512-gcem1KlBU7c9rB+Rq8/3PPKsK2kjqeEBa3bD5kkQo4nYlOHQCJqIJFqBXDEfwaRuYTT4E+FxA9xez7Gf/e3Q7A==", + "dev": true, + "requires": { + "call-bind": "^1.0.2", + "define-properties": "^1.1.3", + "es-abstract": "^1.18.0-next.2", + "get-intrinsic": "^1.1.1", + "is-string": "^1.0.5" + } + }, + "asn1": { + "version": "0.2.4", + "resolved": "https://registry.npmjs.org/asn1/-/asn1-0.2.4.tgz", + "integrity": "sha512-jxwzQpLQjSmWXgwaCZE9Nz+glAG01yF1QnWgbhGwHI5A6FRIEY6IVqtHhIepHqI7/kyEyQEagBC5mBEFlIYvdg==", + "requires": { + "safer-buffer": "~2.1.0" + } + }, + "assert-plus": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/assert-plus/-/assert-plus-1.0.0.tgz", + "integrity": "sha1-8S4PPF13sLHN2RRpQuTpbB5N1SU=" + }, + "astral-regex": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/astral-regex/-/astral-regex-1.0.0.tgz", + "integrity": "sha512-+Ryf6g3BKoRc7jfp7ad8tM4TtMiaWvbF/1/sQcZPkkS7ag3D5nMBCe2UfOTONtAkaG0tO0ij3C5Lwmf1EiyjHg==", + "dev": true + }, + "async": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/async/-/async-3.2.0.tgz", + "integrity": "sha512-TR2mEZFVOj2pLStYxLht7TyfuRzaydfpxr3k9RpHIzMgw7A64dzsdqCxH1WJyQdoe8T10nDXd9wnEigmiuHIZw==" + }, + "asynckit": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", + "integrity": "sha1-x57Zf380y48robyXkLzDZkdLS3k=" + }, + "aws-sign2": { + "version": "0.7.0", + "resolved": "https://registry.npmjs.org/aws-sign2/-/aws-sign2-0.7.0.tgz", + "integrity": "sha1-tG6JCTSpWR8tL2+G1+ap8bP+dqg=" + }, + "aws4": { + "version": "1.11.0", + "resolved": "https://registry.npmjs.org/aws4/-/aws4-1.11.0.tgz", + "integrity": "sha512-xh1Rl34h6Fi1DC2WWKfxUTVqRsNnr6LsKz2+hfwDxQJWmrx8+c7ylaqBMcHfl1U1r2dsifOvKX3LQuLNZ+XSvA==" + }, + "axios": { + "version": "0.19.2", + "resolved": "https://registry.npmjs.org/axios/-/axios-0.19.2.tgz", + "integrity": "sha512-fjgm5MvRHLhx+osE2xoekY70AhARk3a6hkN+3Io1jc00jtquGvxYlKlsFUhmUET0V5te6CcZI7lcv2Ym61mjHA==", + "requires": { + "follow-redirects": "1.5.10" + } + }, + "babel-runtime": { + "version": "6.6.1", + "resolved": "https://registry.npmjs.org/babel-runtime/-/babel-runtime-6.6.1.tgz", + "integrity": "sha1-eIuUtvY04luRvWxd9y1GdFevsAA=", + "requires": { + "core-js": "^2.1.0" + } + }, + "backoff": { + "version": "2.5.0", + "resolved": "https://registry.npmjs.org/backoff/-/backoff-2.5.0.tgz", + "integrity": "sha1-9hbtqdPktmuMp/ynn2lXIsX44m8=", + "requires": { + "precond": "0.2" + } + }, + "balanced-match": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", + "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==" + }, + "bcrypt-pbkdf": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/bcrypt-pbkdf/-/bcrypt-pbkdf-1.0.2.tgz", + "integrity": "sha1-pDAdOJtqQ/m2f/PKEaP2Y342Dp4=", + "requires": { + "tweetnacl": "^0.14.3" + } + }, + "binary-extensions": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.2.0.tgz", + "integrity": "sha512-jDctJ/IVQbZoJykoeHbhXpOlNBqGNcwXJKJog42E5HDPUwQTSdjCHdihjj0DlnheQ7blbT6dHOafNAiS8ooQKA==", + "dev": true + }, + "bluebird": { + "version": "3.7.2", + "resolved": "https://registry.npmjs.org/bluebird/-/bluebird-3.7.2.tgz", + "integrity": "sha512-XpNj6GDQzdfW+r2Wnn7xiSAd7TM3jzkxGXBGTtWKuSXv1xUV+azxAm8jdWZN06QTQk+2N2XB9jRDkvbmQmcRtg==" + }, + "body-parser": { + "version": "1.19.0", + "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.19.0.tgz", + "integrity": "sha512-dhEPs72UPbDnAQJ9ZKMNTP6ptJaionhP5cBb541nXPlW60Jepo9RV/a4fX4XWW9CuFNK22krhrj1+rgzifNCsw==", + "requires": { + "bytes": "3.1.0", + "content-type": "~1.0.4", + "debug": "2.6.9", + "depd": "~1.1.2", + "http-errors": "1.7.2", + "iconv-lite": "0.4.24", + "on-finished": "~2.3.0", + "qs": "6.7.0", + "raw-body": "2.4.0", + "type-is": "~1.6.17" + }, + "dependencies": { + "debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "requires": { + "ms": "2.0.0" + } + }, + "ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" + } + } + }, + "boxen": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/boxen/-/boxen-4.2.0.tgz", + "integrity": "sha512-eB4uT9RGzg2odpER62bBwSLvUeGC+WbRjjyyFhGsKnc8wp/m0+hQsMUvUe3H2V0D5vw0nBdO1hCJoZo5mKeuIQ==", + "dev": true, + "requires": { + "ansi-align": "^3.0.0", + "camelcase": "^5.3.1", + "chalk": "^3.0.0", + "cli-boxes": "^2.2.0", + "string-width": "^4.1.0", + "term-size": "^2.1.0", + "type-fest": "^0.8.1", + "widest-line": "^3.1.0" + }, + "dependencies": { + "ansi-regex": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.0.tgz", + "integrity": "sha512-bY6fj56OUQ0hU1KjFNDQuJFezqKdrAyFdIevADiqrWHwSlbmBNMHp5ak2f40Pm8JTFyM2mqxkG6ngkHO11f/lg==", + "dev": true + }, + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } + }, + "camelcase": { + "version": "5.3.1", + "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz", + "integrity": "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==", + "dev": true + }, + "chalk": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-3.0.0.tgz", + "integrity": "sha512-4D3B6Wf41KOYRFdszmDqMCGq5VV/uMAB273JILmO+3jAlh8X4qDtdtgCR3fxtbLEMzSx22QdhnDcJvu2u1fVwg==", + "dev": true, + "requires": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true + }, + "is-fullwidth-code-point": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", + "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", + "dev": true + }, + "string-width": { + "version": "4.2.2", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.2.tgz", + "integrity": "sha512-XBJbT3N4JhVumXE0eoLU9DCjcaF92KLNqTmFCnG1pf8duUxFGwtP6AD6nkjw9a3IdiRtL3E2w3JDiE/xi3vOeA==", + "dev": true, + "requires": { + "emoji-regex": "^8.0.0", + "is-fullwidth-code-point": "^3.0.0", + "strip-ansi": "^6.0.0" + } + }, + "strip-ansi": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.0.tgz", + "integrity": "sha512-AuvKTrTfQNYNIctbR1K/YGTR1756GycPsg7b9bdV9Duqur4gv6aKqHXah67Z8ImS7WEz5QVcOtlfW2rZEugt6w==", + "dev": true, + "requires": { + "ansi-regex": "^5.0.0" + } + }, + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + } + } + }, + "brace-expansion": { + "version": "1.1.11", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", + "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", + "requires": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" + } + }, + "braces": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz", + "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", + "dev": true, + "requires": { + "fill-range": "^7.0.1" + } + }, + "buffer-equal-constant-time": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/buffer-equal-constant-time/-/buffer-equal-constant-time-1.0.1.tgz", + "integrity": "sha1-+OcRMvf/5uAaXJaXpMbz5I1cyBk=" + }, + "buffer-writer": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/buffer-writer/-/buffer-writer-2.0.0.tgz", + "integrity": "sha512-a7ZpuTZU1TRtnwyCNW3I5dc0wWNC3VR9S++Ewyk2HHZdrO3CQJqSpd+95Us590V6AL7JqUAH2IwZ/398PmNFgw==" + }, + "bunyan": { + "version": "1.8.15", + "resolved": "https://registry.npmjs.org/bunyan/-/bunyan-1.8.15.tgz", + "integrity": "sha512-0tECWShh6wUysgucJcBAoYegf3JJoZWibxdqhTm7OHPeT42qdjkZ29QCMcKwbgU1kiH+auSIasNRXMLWXafXig==", + "requires": { + "dtrace-provider": "~0.8", + "moment": "^2.19.3", + "mv": "~2", + "safe-json-stringify": "~1" + } + }, + "bytes": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.0.tgz", + "integrity": "sha512-zauLjrfCG+xvoyaqLoV8bLVXXNGC4JqlxFCutSDWA6fJrTo2ZuvLYTqZ7aHBLZSMOopbzwv8f+wZcVzfVTI2Dg==" + }, + "cacheable-request": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/cacheable-request/-/cacheable-request-6.1.0.tgz", + "integrity": "sha512-Oj3cAGPCqOZX7Rz64Uny2GYAZNliQSqfbePrgAQ1wKAihYmCUnraBtJtKcGR4xz7wF+LoJC+ssFZvv5BgF9Igg==", + "dev": true, + "requires": { + "clone-response": "^1.0.2", + "get-stream": "^5.1.0", + "http-cache-semantics": "^4.0.0", + "keyv": "^3.0.0", + "lowercase-keys": "^2.0.0", + "normalize-url": "^4.1.0", + "responselike": "^1.0.2" + }, + "dependencies": { + "get-stream": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-5.2.0.tgz", + "integrity": "sha512-nBF+F1rAZVCu/p7rjzgA+Yb4lfYXrpl7a6VmJrU8wF9I1CKvP/QwPNZHnOlwbTkY6dvtFIzFMSyQXbLoTQPRpA==", + "dev": true, + "requires": { + "pump": "^3.0.0" + } + }, + "lowercase-keys": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-2.0.0.tgz", + "integrity": "sha512-tqNXrS78oMOE73NMxK4EMLQsQowWf8jKooH9g7xPavRT706R6bkQJ6DY2Te7QukaZsulxa30wQ7bk0pm4XiHmA==", + "dev": true + } + } + }, + "call-bind": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/call-bind/-/call-bind-1.0.2.tgz", + "integrity": "sha512-7O+FbCihrB5WGbFYesctwmTKae6rOiIzmz1icreWJ+0aA7LJfuqhEso2T9ncpcFtzMQtzXf2QGGueWJGTYsqrA==", + "dev": true, + "requires": { + "function-bind": "^1.1.1", + "get-intrinsic": "^1.0.2" + } + }, + "callsites": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz", + "integrity": "sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==", + "dev": true + }, + "camelcase": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-3.0.0.tgz", + "integrity": "sha1-MvxLn82vhF/N9+c7uXysImHwqwo=" + }, + "caseless": { + "version": "0.12.0", + "resolved": "https://registry.npmjs.org/caseless/-/caseless-0.12.0.tgz", + "integrity": "sha1-G2gcIf+EAzyCZUMJBolCDRhxUdw=" + }, + "chalk": { + "version": "2.4.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz", + "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==", + "dev": true, + "requires": { + "ansi-styles": "^3.2.1", + "escape-string-regexp": "^1.0.5", + "supports-color": "^5.3.0" + } + }, + "chardet": { + "version": "0.7.0", + "resolved": "https://registry.npmjs.org/chardet/-/chardet-0.7.0.tgz", + "integrity": "sha512-mT8iDcrh03qDGRRmoA2hmBJnxpllMR+0/0qlzjqZES6NdiWDcZkCNAk4rPFZ9Q85r27unkiNNg8ZOiwZXBHwcA==", + "dev": true + }, + "chokidar": { + "version": "3.5.2", + "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.5.2.tgz", + "integrity": "sha512-ekGhOnNVPgT77r4K/U3GDhu+FQ2S8TnK/s2KbIGXi0SZWuwkZ2QNyfWdZW+TVfn84DpEP7rLeCt2UI6bJ8GwbQ==", + "dev": true, + "requires": { + "anymatch": "~3.1.2", + "braces": "~3.0.2", + "fsevents": "~2.3.2", + "glob-parent": "~5.1.2", + "is-binary-path": "~2.1.0", + "is-glob": "~4.0.1", + "normalize-path": "~3.0.0", + "readdirp": "~3.6.0" + } + }, + "ci-info": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ci-info/-/ci-info-2.0.0.tgz", + "integrity": "sha512-5tK7EtrZ0N+OLFMthtqOj4fI2Jeb88C4CAZPu25LDVUgXJ0A3Js4PMGqrn0JU1W0Mh1/Z8wZzYPxqUrXeBboCQ==", + "dev": true + }, + "cli-boxes": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/cli-boxes/-/cli-boxes-2.2.1.tgz", + "integrity": "sha512-y4coMcylgSCdVinjiDBuR8PCC2bLjyGTwEmPb9NHR/QaNU6EUOXcTY/s6VjGMD6ENSEaeQYHCY0GNGS5jfMwPw==", + "dev": true + }, + "cli-cursor": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/cli-cursor/-/cli-cursor-3.1.0.tgz", + "integrity": "sha512-I/zHAwsKf9FqGoXM4WWRACob9+SNukZTd94DWF57E4toouRulbCxcUh6RKUEOQlYTHJnzkPMySvPNaaSLNfLZw==", + "dev": true, + "requires": { + "restore-cursor": "^3.1.0" + } + }, + "cli-width": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/cli-width/-/cli-width-3.0.0.tgz", + "integrity": "sha512-FxqpkPPwu1HjuN93Omfm4h8uIanXofW0RxVEW3k5RKx+mJJYSthzNhp32Kzxxy3YAEZ/Dc/EWN1vZRY0+kOhbw==", + "dev": true + }, + "cliui": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/cliui/-/cliui-3.2.0.tgz", + "integrity": "sha1-EgYBU3qRbSmUD5NNo7SNWFo5IT0=", + "requires": { + "string-width": "^1.0.1", + "strip-ansi": "^3.0.1", + "wrap-ansi": "^2.0.0" + } + }, + "clone-response": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/clone-response/-/clone-response-1.0.2.tgz", + "integrity": "sha1-0dyXOSAxTfZ/vrlCI7TuNQI56Ws=", + "dev": true, + "requires": { + "mimic-response": "^1.0.0" + } + }, + "code-point-at": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/code-point-at/-/code-point-at-1.1.0.tgz", + "integrity": "sha1-DQcLTQQ6W+ozovGkDi7bPZpMz3c=" + }, + "codependency": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/codependency/-/codependency-0.1.4.tgz", + "integrity": "sha1-0XY6tyZL1wyR2WJumIYtN5K/jUo=", + "requires": { + "semver": "5.0.1" + }, + "dependencies": { + "semver": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-5.0.1.tgz", + "integrity": "sha1-n7P0AE+QDYPEeWj+QvdYPgWDLMk=" + } + } + }, + "color": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/color/-/color-3.0.0.tgz", + "integrity": "sha512-jCpd5+s0s0t7p3pHQKpnJ0TpQKKdleP71LWcA0aqiljpiuAkOSUFN/dyH8ZwF0hRmFlrIuRhufds1QyEP9EB+w==", + "requires": { + "color-convert": "^1.9.1", + "color-string": "^1.5.2" + } + }, + "color-convert": { + "version": "1.9.3", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", + "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==", + "requires": { + "color-name": "1.1.3" + } + }, + "color-name": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", + "integrity": "sha1-p9BVi9icQveV3UIyj3QIMcpTvCU=" + }, + "color-string": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/color-string/-/color-string-1.6.0.tgz", + "integrity": "sha512-c/hGS+kRWJutUBEngKKmk4iH3sD59MBkoxVapS/0wgpCz2u7XsNloxknyvBhzwEs1IbV36D9PwqLPJ2DTu3vMA==", + "requires": { + "color-name": "^1.0.0", + "simple-swizzle": "^0.2.2" + } + }, + "colors": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/colors/-/colors-1.4.0.tgz", + "integrity": "sha512-a+UqTh4kgZg/SlGvfbzDHpgRu7AAQOmmqRHJnxhRZICKFUT91brVhNNt58CMWU9PsBbv3PDCZUHbVxuDiH2mtA==" + }, + "colorspace": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/colorspace/-/colorspace-1.1.2.tgz", + "integrity": "sha512-vt+OoIP2d76xLhjwbBaucYlNSpPsrJWPlBTtwCpQKIu6/CSMutyzX93O/Do0qzpH3YoHEes8YEFXyZ797rEhzQ==", + "requires": { + "color": "3.0.x", + "text-hex": "1.0.x" + } + }, + "combined-stream": { + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz", + "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==", + "requires": { + "delayed-stream": "~1.0.0" + } + }, + "concat-map": { + "version": "0.0.1", + "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", + "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=" + }, + "config": { + "version": "3.3.6", + "resolved": "https://registry.npmjs.org/config/-/config-3.3.6.tgz", + "integrity": "sha512-Hj5916C5HFawjYJat1epbyY2PlAgLpBtDUlr0MxGLgo3p5+7kylyvnRY18PqJHgnNWXcdd0eWDemT7eYWuFgwg==", + "requires": { + "json5": "^2.1.1" + } + }, + "configstore": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/configstore/-/configstore-5.0.1.tgz", + "integrity": "sha512-aMKprgk5YhBNyH25hj8wGt2+D52Sw1DRRIzqBwLp2Ya9mFmY8KPvvtvmna8SxVR9JMZ4kzMD68N22vlaRpkeFA==", + "dev": true, + "requires": { + "dot-prop": "^5.2.0", + "graceful-fs": "^4.1.2", + "make-dir": "^3.0.0", + "unique-string": "^2.0.0", + "write-file-atomic": "^3.0.0", + "xdg-basedir": "^4.0.0" + } + }, + "contains-path": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/contains-path/-/contains-path-0.1.0.tgz", + "integrity": "sha1-/ozxhP9mcLa67wGp1IYaXL7EEgo=", + "dev": true + }, + "content-disposition": { + "version": "0.5.3", + "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.3.tgz", + "integrity": "sha512-ExO0774ikEObIAEV9kDo50o+79VCUdEB6n6lzKgGwupcVeRlhrj3qGAfwq8G6uBJjkqLrhT0qEYFcWng8z1z0g==", + "requires": { + "safe-buffer": "5.1.2" + } + }, + "content-type": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.4.tgz", + "integrity": "sha512-hIP3EEPs8tB9AT1L+NUqtwOAps4mk2Zob89MWXMHjHWg9milF/j4osnnQLXBCBFBk/tvIG/tUc9mOUJiPBhPXA==" + }, + "cookie": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.4.0.tgz", + "integrity": "sha512-+Hp8fLp57wnUSt0tY0tHEXh4voZRDnoIrZPqlo3DPiI4y9lwg/jqx+1Om94/W6ZaPDOUbnjOt/99w66zk+l1Xg==" + }, + "cookie-signature": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz", + "integrity": "sha1-4wOogrNCzD7oylE6eZmXNNqzriw=" + }, + "core-js": { + "version": "2.6.12", + "resolved": "https://registry.npmjs.org/core-js/-/core-js-2.6.12.tgz", + "integrity": "sha512-Kb2wC0fvsWfQrgk8HU5lW6U/Lcs8+9aaYcy4ZFc6DDlo4nZ7n70dEgE5rtR0oG6ufKDUnrwfWL1mXR5ljDatrQ==" + }, + "core-util-is": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.2.tgz", + "integrity": "sha1-tf1UIgqivFq1eqtxQMlAdUUDwac=" + }, + "cors": { + "version": "2.8.5", + "resolved": "https://registry.npmjs.org/cors/-/cors-2.8.5.tgz", + "integrity": "sha512-KIHbLJqu73RGr/hnbrO9uBeixNGuvSQjul/jdFvS/KFSIH1hWVd1ng7zOHx+YrEfInLG7q4n6GHQ9cDtxv/P6g==", + "requires": { + "object-assign": "^4", + "vary": "^1" + } + }, + "cross-spawn": { + "version": "6.0.5", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-6.0.5.tgz", + "integrity": "sha512-eTVLrBSt7fjbDygz805pMnstIs2VTBNkRm0qxZd+M7A5XDdxVRWO5MxGBXZhjY4cqLYLdtrGqRf8mBPmzwSpWQ==", + "dev": true, + "requires": { + "nice-try": "^1.0.4", + "path-key": "^2.0.1", + "semver": "^5.5.0", + "shebang-command": "^1.2.0", + "which": "^1.2.9" + } + }, + "crypto-random-string": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/crypto-random-string/-/crypto-random-string-2.0.0.tgz", + "integrity": "sha512-v1plID3y9r/lPhviJ1wrXpLeyUIGAZ2SHNYTEapm7/8A9nLPoyvVp3RK/EPFqn5kEznyWgYZNsRtYYIWbuG8KA==", + "dev": true + }, + "dashdash": { + "version": "1.14.1", + "resolved": "https://registry.npmjs.org/dashdash/-/dashdash-1.14.1.tgz", + "integrity": "sha1-hTz6D3y+L+1d4gMmuN1YEDX24vA=", + "requires": { + "assert-plus": "^1.0.0" + } + }, + "debug": { + "version": "4.3.2", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.2.tgz", + "integrity": "sha512-mOp8wKcvj7XxC78zLgw/ZA+6TSgkoE2C/ienthhRD298T7UNwAg9diBpLRxC0mOezLl4B0xV7M0cCO6P/O0Xhw==", + "requires": { + "ms": "2.1.2" + }, + "dependencies": { + "ms": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", + "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==" + } + } + }, + "debug-log": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/debug-log/-/debug-log-1.0.1.tgz", + "integrity": "sha1-IwdjLUwEOCuN+KMvcLiVBG1SdF8=", + "dev": true + }, + "decamelize": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/decamelize/-/decamelize-1.2.0.tgz", + "integrity": "sha1-9lNNFRSCabIDUue+4m9QH5oZEpA=" + }, + "decompress-response": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/decompress-response/-/decompress-response-3.3.0.tgz", + "integrity": "sha1-gKTdMjdIOEv6JICDYirt7Jgq3/M=", + "dev": true, + "requires": { + "mimic-response": "^1.0.0" + } + }, + "deep-extend": { + "version": "0.6.0", + "resolved": "https://registry.npmjs.org/deep-extend/-/deep-extend-0.6.0.tgz", + "integrity": "sha512-LOHxIOaPYdHlJRtCQfDIVZtfw/ufM8+rVj649RIHzcm/vGwQRXFt6OPqIFWsm2XEMrNIEtWR64sY1LEKD2vAOA==", + "dev": true + }, + "deep-is": { + "version": "0.1.3", + "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.3.tgz", + "integrity": "sha1-s2nW+128E+7PUk+RsHD+7cNXzzQ=", + "dev": true + }, + "defer-to-connect": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/defer-to-connect/-/defer-to-connect-1.1.3.tgz", + "integrity": "sha512-0ISdNousHvZT2EiFlZeZAHBUvSxmKswVCEf8hW7KWgG4a8MVEu/3Vb6uWYozkjylyCxe0JBIiRB1jV45S70WVQ==", + "dev": true + }, + "define-properties": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/define-properties/-/define-properties-1.1.3.tgz", + "integrity": "sha512-3MqfYKj2lLzdMSf8ZIZE/V+Zuy+BgD6f164e8K2w7dgnpKArBDerGYpM46IYYcjnkdPNMjPk9A6VFB8+3SKlXQ==", + "dev": true, + "requires": { + "object-keys": "^1.0.12" + } + }, + "deglob": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/deglob/-/deglob-4.0.1.tgz", + "integrity": "sha512-/g+RDZ7yf2HvoW+E5Cy+K94YhgcFgr6C8LuHZD1O5HoNPkf3KY6RfXJ0DBGlB/NkLi5gml+G9zqRzk9S0mHZCg==", + "dev": true, + "requires": { + "find-root": "^1.0.0", + "glob": "^7.0.5", + "ignore": "^5.0.0", + "pkg-config": "^1.1.0", + "run-parallel": "^1.1.2", + "uniq": "^1.0.1" + }, + "dependencies": { + "glob": { + "version": "7.1.7", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.7.tgz", + "integrity": "sha512-OvD9ENzPLbegENnYP5UUfJIirTg4+XwMWGaQfQTY0JenxNvvIKP3U3/tAQSPIu/lHxXYSZmpXlUHeqAIdKzBLQ==", + "dev": true, + "requires": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.0.4", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + } + }, + "ignore": { + "version": "5.1.8", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.1.8.tgz", + "integrity": "sha512-BMpfD7PpiETpBl/A6S498BaIJ6Y/ABT93ETbby2fP00v4EbvPBXWEoaR1UBPKs3iR53pJY7EtZk5KACI57i1Uw==", + "dev": true + } + } + }, + "delayed-stream": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", + "integrity": "sha1-3zrhmayt+31ECqrgsp4icrJOxhk=" + }, + "depd": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.2.tgz", + "integrity": "sha1-m81S4UwJd2PnSbJ0xDRu0uVgtak=" + }, + "destroy": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.0.4.tgz", + "integrity": "sha1-l4hXRCxEdJ5CBmE+N5RiBYJqvYA=" + }, + "doctrine": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-3.0.0.tgz", + "integrity": "sha512-yS+Q5i3hBf7GBkd4KG8a7eBNNWNGLTaEwwYWUijIYM7zrlYDM0BFXHjjPWlWZ1Rg7UaddZeIDmi9jF3HmqiQ2w==", + "dev": true, + "requires": { + "esutils": "^2.0.2" + } + }, + "dot-prop": { + "version": "5.3.0", + "resolved": "https://registry.npmjs.org/dot-prop/-/dot-prop-5.3.0.tgz", + "integrity": "sha512-QM8q3zDe58hqUqjraQOmzZ1LIH9SWQJTlEKCH4kJ2oQvLZk7RbQXvtDM2XEq3fwkV9CCvvH4LA0AV+ogFsBM2Q==", + "dev": true, + "requires": { + "is-obj": "^2.0.0" + } + }, + "dottie": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/dottie/-/dottie-2.0.2.tgz", + "integrity": "sha512-fmrwR04lsniq/uSr8yikThDTrM7epXHBAAjH9TbeH3rEA8tdCO7mRzB9hdmdGyJCxF8KERo9CITcm3kGuoyMhg==" + }, + "dtrace-provider": { + "version": "0.8.8", + "resolved": "https://registry.npmjs.org/dtrace-provider/-/dtrace-provider-0.8.8.tgz", + "integrity": "sha512-b7Z7cNtHPhH9EJhNNbbeqTcXB8LGFFZhq1PGgEvpeHlzd36bhbdTWoE/Ba/YguqpBSlAPKnARWhVlhunCMwfxg==", + "optional": true, + "requires": { + "nan": "^2.14.0" + } + }, + "duplexer3": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/duplexer3/-/duplexer3-0.1.4.tgz", + "integrity": "sha1-7gHdHKwO08vH/b6jfcCo8c4ALOI=", + "dev": true + }, + "ecc-jsbn": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/ecc-jsbn/-/ecc-jsbn-0.1.2.tgz", + "integrity": "sha1-OoOpBOVDUyh4dMVkt1SThoSamMk=", + "requires": { + "jsbn": "~0.1.0", + "safer-buffer": "^2.1.0" + } + }, + "ecdsa-sig-formatter": { + "version": "1.0.11", + "resolved": "https://registry.npmjs.org/ecdsa-sig-formatter/-/ecdsa-sig-formatter-1.0.11.tgz", + "integrity": "sha512-nagl3RYrbNv6kQkeJIpt6NJZy8twLB/2vtz6yN9Z4vRKHN4/QZJIEbqohALSgwKdnksuY3k5Addp5lg8sVoVcQ==", + "requires": { + "safe-buffer": "^5.0.1" + } + }, + "ee-first": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", + "integrity": "sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0=" + }, + "emoji-regex": { + "version": "8.0.0", + "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", + "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", + "dev": true + }, + "enabled": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/enabled/-/enabled-2.0.0.tgz", + "integrity": "sha512-AKrN98kuwOzMIdAizXGI86UFBoo26CL21UM763y1h/GMSJ4/OHU9k2YlsmBpyScFo/wbLzWQJBMCW4+IO3/+OQ==" + }, + "encodeurl": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz", + "integrity": "sha1-rT/0yG7C0CkyL1oCw6mmBslbP1k=" + }, + "end-of-stream": { + "version": "1.4.4", + "resolved": "https://registry.npmjs.org/end-of-stream/-/end-of-stream-1.4.4.tgz", + "integrity": "sha512-+uw1inIHVPQoaVuHzRyXd21icM+cnt4CzD5rW+NC1wjOUSTOs+Te7FOv7AhN7vS9x/oIyhLP5PR1H+phQAHu5Q==", + "dev": true, + "requires": { + "once": "^1.4.0" + } + }, + "error-ex": { + "version": "1.3.2", + "resolved": "https://registry.npmjs.org/error-ex/-/error-ex-1.3.2.tgz", + "integrity": "sha512-7dFHNmqeFSEt2ZBsCriorKnn3Z2pj+fd9kmI6QoWw4//DL+icEBfc0U7qJCisqrTsKTjw4fNFy2pW9OqStD84g==", + "requires": { + "is-arrayish": "^0.2.1" + } + }, + "es-abstract": { + "version": "1.18.3", + "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.18.3.tgz", + "integrity": "sha512-nQIr12dxV7SSxE6r6f1l3DtAeEYdsGpps13dR0TwJg1S8gyp4ZPgy3FZcHBgbiQqnoqSTb+oC+kO4UQ0C/J8vw==", + "dev": true, + "requires": { + "call-bind": "^1.0.2", + "es-to-primitive": "^1.2.1", + "function-bind": "^1.1.1", + "get-intrinsic": "^1.1.1", + "has": "^1.0.3", + "has-symbols": "^1.0.2", + "is-callable": "^1.2.3", + "is-negative-zero": "^2.0.1", + "is-regex": "^1.1.3", + "is-string": "^1.0.6", + "object-inspect": "^1.10.3", + "object-keys": "^1.1.1", + "object.assign": "^4.1.2", + "string.prototype.trimend": "^1.0.4", + "string.prototype.trimstart": "^1.0.4", + "unbox-primitive": "^1.0.1" + } + }, + "es-to-primitive": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/es-to-primitive/-/es-to-primitive-1.2.1.tgz", + "integrity": "sha512-QCOllgZJtaUo9miYBcLChTUaHNjJF3PYs1VidD7AwiEj1kYxKeQTctLAezAOH5ZKRH0g2IgPn6KwB4IT8iRpvA==", + "dev": true, + "requires": { + "is-callable": "^1.1.4", + "is-date-object": "^1.0.1", + "is-symbol": "^1.0.2" + } + }, + "escape-goat": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/escape-goat/-/escape-goat-2.1.1.tgz", + "integrity": "sha512-8/uIhbG12Csjy2JEW7D9pHbreaVaS/OpN3ycnyvElTdwM5n6GY6W6e2IPemfvGZeUMqZ9A/3GqIZMgKnBhAw/Q==", + "dev": true + }, + "escape-html": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz", + "integrity": "sha1-Aljq5NPQwJdN4cFpGI7wBR0dGYg=" + }, + "escape-string-regexp": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", + "integrity": "sha1-G2HAViGQqN/2rjuyzwIAyhMLhtQ=", + "dev": true + }, + "eslint": { + "version": "6.8.0", + "resolved": "https://registry.npmjs.org/eslint/-/eslint-6.8.0.tgz", + "integrity": "sha512-K+Iayyo2LtyYhDSYwz5D5QdWw0hCacNzyq1Y821Xna2xSJj7cijoLLYmLxTQgcgZ9mC61nryMy9S7GRbYpI5Ig==", + "dev": true, + "requires": { + "@babel/code-frame": "^7.0.0", + "ajv": "^6.10.0", + "chalk": "^2.1.0", + "cross-spawn": "^6.0.5", + "debug": "^4.0.1", + "doctrine": "^3.0.0", + "eslint-scope": "^5.0.0", + "eslint-utils": "^1.4.3", + "eslint-visitor-keys": "^1.1.0", + "espree": "^6.1.2", + "esquery": "^1.0.1", + "esutils": "^2.0.2", + "file-entry-cache": "^5.0.1", + "functional-red-black-tree": "^1.0.1", + "glob-parent": "^5.0.0", + "globals": "^12.1.0", + "ignore": "^4.0.6", + "import-fresh": "^3.0.0", + "imurmurhash": "^0.1.4", + "inquirer": "^7.0.0", + "is-glob": "^4.0.0", + "js-yaml": "^3.13.1", + "json-stable-stringify-without-jsonify": "^1.0.1", + "levn": "^0.3.0", + "lodash": "^4.17.14", + "minimatch": "^3.0.4", + "mkdirp": "^0.5.1", + "natural-compare": "^1.4.0", + "optionator": "^0.8.3", + "progress": "^2.0.0", + "regexpp": "^2.0.1", + "semver": "^6.1.2", + "strip-ansi": "^5.2.0", + "strip-json-comments": "^3.0.1", + "table": "^5.2.3", + "text-table": "^0.2.0", + "v8-compile-cache": "^2.0.3" + }, + "dependencies": { + "ansi-regex": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz", + "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==", + "dev": true + }, + "semver": { + "version": "6.3.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", + "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", + "dev": true + }, + "strip-ansi": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", + "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", + "dev": true, + "requires": { + "ansi-regex": "^4.1.0" + } + } + } + }, + "eslint-config-standard": { + "version": "14.1.1", + "resolved": "https://registry.npmjs.org/eslint-config-standard/-/eslint-config-standard-14.1.1.tgz", + "integrity": "sha512-Z9B+VR+JIXRxz21udPTL9HpFMyoMUEeX1G251EQ6e05WD9aPVtVBn09XUmZ259wCMlCDmYDSZG62Hhm+ZTJcUg==", + "dev": true + }, + "eslint-config-standard-jsx": { + "version": "8.1.0", + "resolved": "https://registry.npmjs.org/eslint-config-standard-jsx/-/eslint-config-standard-jsx-8.1.0.tgz", + "integrity": "sha512-ULVC8qH8qCqbU792ZOO6DaiaZyHNS/5CZt3hKqHkEhVlhPEPN3nfBqqxJCyp59XrjIBZPu1chMYe9T2DXZ7TMw==", + "dev": true + }, + "eslint-import-resolver-node": { + "version": "0.3.4", + "resolved": "https://registry.npmjs.org/eslint-import-resolver-node/-/eslint-import-resolver-node-0.3.4.tgz", + "integrity": "sha512-ogtf+5AB/O+nM6DIeBUNr2fuT7ot9Qg/1harBfBtaP13ekEWFQEEMP94BCB7zaNW3gyY+8SHYF00rnqYwXKWOA==", + "dev": true, + "requires": { + "debug": "^2.6.9", + "resolve": "^1.13.1" + }, + "dependencies": { + "debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "dev": true, + "requires": { + "ms": "2.0.0" + } + }, + "ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", + "dev": true + } + } + }, + "eslint-module-utils": { + "version": "2.6.1", + "resolved": "https://registry.npmjs.org/eslint-module-utils/-/eslint-module-utils-2.6.1.tgz", + "integrity": "sha512-ZXI9B8cxAJIH4nfkhTwcRTEAnrVfobYqwjWy/QMCZ8rHkZHFjf9yO4BzpiF9kCSfNlMG54eKigISHpX0+AaT4A==", + "dev": true, + "requires": { + "debug": "^3.2.7", + "pkg-dir": "^2.0.0" + }, + "dependencies": { + "debug": { + "version": "3.2.7", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", + "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", + "dev": true, + "requires": { + "ms": "^2.1.1" + } + } + } + }, + "eslint-plugin-es": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/eslint-plugin-es/-/eslint-plugin-es-2.0.0.tgz", + "integrity": "sha512-f6fceVtg27BR02EYnBhgWLFQfK6bN4Ll0nQFrBHOlCsAyxeZkn0NHns5O0YZOPrV1B3ramd6cgFwaoFLcSkwEQ==", + "dev": true, + "requires": { + "eslint-utils": "^1.4.2", + "regexpp": "^3.0.0" + }, + "dependencies": { + "regexpp": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/regexpp/-/regexpp-3.2.0.tgz", + "integrity": "sha512-pq2bWo9mVD43nbts2wGv17XLiNLya+GklZ8kaDLV2Z08gDCsGpnKn9BFMepvWuHCbyVvY7J5o5+BVvoQbmlJLg==", + "dev": true + } + } + }, + "eslint-plugin-import": { + "version": "2.18.2", + "resolved": "https://registry.npmjs.org/eslint-plugin-import/-/eslint-plugin-import-2.18.2.tgz", + "integrity": "sha512-5ohpsHAiUBRNaBWAF08izwUGlbrJoJJ+W9/TBwsGoR1MnlgfwMIKrFeSjWbt6moabiXW9xNvtFz+97KHRfI4HQ==", + "dev": true, + "requires": { + "array-includes": "^3.0.3", + "contains-path": "^0.1.0", + "debug": "^2.6.9", + "doctrine": "1.5.0", + "eslint-import-resolver-node": "^0.3.2", + "eslint-module-utils": "^2.4.0", + "has": "^1.0.3", + "minimatch": "^3.0.4", + "object.values": "^1.1.0", + "read-pkg-up": "^2.0.0", + "resolve": "^1.11.0" + }, + "dependencies": { + "debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "dev": true, + "requires": { + "ms": "2.0.0" + } + }, + "doctrine": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-1.5.0.tgz", + "integrity": "sha1-N53Ocw9hZvds76TmcHoVmwLFpvo=", + "dev": true, + "requires": { + "esutils": "^2.0.2", + "isarray": "^1.0.0" + } + }, + "find-up": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-2.1.0.tgz", + "integrity": "sha1-RdG35QbHF93UgndaK3eSCjwMV6c=", + "dev": true, + "requires": { + "locate-path": "^2.0.0" + } + }, + "load-json-file": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/load-json-file/-/load-json-file-2.0.0.tgz", + "integrity": "sha1-eUfkIUmvgNaWy/eXvKq8/h/inKg=", + "dev": true, + "requires": { + "graceful-fs": "^4.1.2", + "parse-json": "^2.2.0", + "pify": "^2.0.0", + "strip-bom": "^3.0.0" + } + }, + "ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", + "dev": true + }, + "path-type": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/path-type/-/path-type-2.0.0.tgz", + "integrity": "sha1-8BLMuEFbcJb8LaoQVMPXI4lZTHM=", + "dev": true, + "requires": { + "pify": "^2.0.0" + } + }, + "read-pkg": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/read-pkg/-/read-pkg-2.0.0.tgz", + "integrity": "sha1-jvHAYjxqbbDcZxPEv6xGMysjaPg=", + "dev": true, + "requires": { + "load-json-file": "^2.0.0", + "normalize-package-data": "^2.3.2", + "path-type": "^2.0.0" + } + }, + "read-pkg-up": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/read-pkg-up/-/read-pkg-up-2.0.0.tgz", + "integrity": "sha1-a3KoBImE4MQeeVEP1en6mbO1Sb4=", + "dev": true, + "requires": { + "find-up": "^2.0.0", + "read-pkg": "^2.0.0" + } + }, + "strip-bom": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-3.0.0.tgz", + "integrity": "sha1-IzTBjpx1n3vdVv3vfprj1YjmjtM=", + "dev": true + } + } + }, + "eslint-plugin-node": { + "version": "10.0.0", + "resolved": "https://registry.npmjs.org/eslint-plugin-node/-/eslint-plugin-node-10.0.0.tgz", + "integrity": "sha512-1CSyM/QCjs6PXaT18+zuAXsjXGIGo5Rw630rSKwokSs2jrYURQc4R5JZpoanNCqwNmepg+0eZ9L7YiRUJb8jiQ==", + "dev": true, + "requires": { + "eslint-plugin-es": "^2.0.0", + "eslint-utils": "^1.4.2", + "ignore": "^5.1.1", + "minimatch": "^3.0.4", + "resolve": "^1.10.1", + "semver": "^6.1.0" + }, + "dependencies": { + "ignore": { + "version": "5.1.8", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.1.8.tgz", + "integrity": "sha512-BMpfD7PpiETpBl/A6S498BaIJ6Y/ABT93ETbby2fP00v4EbvPBXWEoaR1UBPKs3iR53pJY7EtZk5KACI57i1Uw==", + "dev": true + }, + "semver": { + "version": "6.3.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", + "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", + "dev": true + } + } + }, + "eslint-plugin-promise": { + "version": "4.2.1", + "resolved": "https://registry.npmjs.org/eslint-plugin-promise/-/eslint-plugin-promise-4.2.1.tgz", + "integrity": "sha512-VoM09vT7bfA7D+upt+FjeBO5eHIJQBUWki1aPvB+vbNiHS3+oGIJGIeyBtKQTME6UPXXy3vV07OL1tHd3ANuDw==", + "dev": true + }, + "eslint-plugin-react": { + "version": "7.14.3", + "resolved": "https://registry.npmjs.org/eslint-plugin-react/-/eslint-plugin-react-7.14.3.tgz", + "integrity": "sha512-EzdyyBWC4Uz2hPYBiEJrKCUi2Fn+BJ9B/pJQcjw5X+x/H2Nm59S4MJIvL4O5NEE0+WbnQwEBxWY03oUk+Bc3FA==", + "dev": true, + "requires": { + "array-includes": "^3.0.3", + "doctrine": "^2.1.0", + "has": "^1.0.3", + "jsx-ast-utils": "^2.1.0", + "object.entries": "^1.1.0", + "object.fromentries": "^2.0.0", + "object.values": "^1.1.0", + "prop-types": "^15.7.2", + "resolve": "^1.10.1" + }, + "dependencies": { + "doctrine": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-2.1.0.tgz", + "integrity": "sha512-35mSku4ZXK0vfCuHEDAwt55dg2jNajHZ1odvF+8SSr82EsZY4QmXfuWso8oEd8zRhVObSN18aM0CjSdoBX7zIw==", + "dev": true, + "requires": { + "esutils": "^2.0.2" + } + } + } + }, + "eslint-plugin-standard": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/eslint-plugin-standard/-/eslint-plugin-standard-4.0.2.tgz", + "integrity": "sha512-nKptN8l7jksXkwFk++PhJB3cCDTcXOEyhISIN86Ue2feJ1LFyY3PrY3/xT2keXlJSY5bpmbiTG0f885/YKAvTA==", + "dev": true + }, + "eslint-scope": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-5.1.1.tgz", + "integrity": "sha512-2NxwbF/hZ0KpepYN0cNbo+FN6XoK7GaHlQhgx/hIZl6Va0bF45RQOOwhLIy8lQDbuCiadSLCBnH2CFYquit5bw==", + "dev": true, + "requires": { + "esrecurse": "^4.3.0", + "estraverse": "^4.1.1" + } + }, + "eslint-utils": { + "version": "1.4.3", + "resolved": "https://registry.npmjs.org/eslint-utils/-/eslint-utils-1.4.3.tgz", + "integrity": "sha512-fbBN5W2xdY45KulGXmLHZ3c3FHfVYmKg0IrAKGOkT/464PQsx2UeIzfz1RmEci+KLm1bBaAzZAh8+/E+XAeZ8Q==", + "dev": true, + "requires": { + "eslint-visitor-keys": "^1.1.0" + } + }, + "eslint-visitor-keys": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-1.3.0.tgz", + "integrity": "sha512-6J72N8UNa462wa/KFODt/PJ3IU60SDpC3QXC1Hjc1BXXpfL2C9R5+AU7jhe0F6GREqVMh4Juu+NY7xn+6dipUQ==", + "dev": true + }, + "espree": { + "version": "6.2.1", + "resolved": "https://registry.npmjs.org/espree/-/espree-6.2.1.tgz", + "integrity": "sha512-ysCxRQY3WaXJz9tdbWOwuWr5Y/XrPTGX9Kiz3yoUXwW0VZ4w30HTkQLaGx/+ttFjF8i+ACbArnB4ce68a9m5hw==", + "dev": true, + "requires": { + "acorn": "^7.1.1", + "acorn-jsx": "^5.2.0", + "eslint-visitor-keys": "^1.1.0" + } + }, + "esprima": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz", + "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==" + }, + "esquery": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.4.0.tgz", + "integrity": "sha512-cCDispWt5vHHtwMY2YrAQ4ibFkAL8RbH5YGBnZBc90MolvvfkkQcJro/aZiAQUlQ3qgrYS6D6v8Gc5G5CQsc9w==", + "dev": true, + "requires": { + "estraverse": "^5.1.0" + }, + "dependencies": { + "estraverse": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.2.0.tgz", + "integrity": "sha512-BxbNGGNm0RyRYvUdHpIwv9IWzeM9XClbOxwoATuFdOE7ZE6wHL+HQ5T8hoPM+zHvmKzzsEqhgy0GrQ5X13afiQ==", + "dev": true + } + } + }, + "esrecurse": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/esrecurse/-/esrecurse-4.3.0.tgz", + "integrity": "sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag==", + "dev": true, + "requires": { + "estraverse": "^5.2.0" + }, + "dependencies": { + "estraverse": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.2.0.tgz", + "integrity": "sha512-BxbNGGNm0RyRYvUdHpIwv9IWzeM9XClbOxwoATuFdOE7ZE6wHL+HQ5T8hoPM+zHvmKzzsEqhgy0GrQ5X13afiQ==", + "dev": true + } + } + }, + "estraverse": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-4.3.0.tgz", + "integrity": "sha512-39nnKffWz8xN1BU/2c79n9nB9HDzo0niYUqx6xyqUnyoAnQyyWpOTdZEeiCch8BBu515t4wp9ZmgVfVhn9EBpw==", + "dev": true + }, + "esutils": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz", + "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==", + "dev": true + }, + "etag": { + "version": "1.8.1", + "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz", + "integrity": "sha1-Qa4u62XvpiJorr/qg6x9eSmbCIc=" + }, + "express": { + "version": "4.17.1", + "resolved": "https://registry.npmjs.org/express/-/express-4.17.1.tgz", + "integrity": "sha512-mHJ9O79RqluphRrcw2X/GTh3k9tVv8YcoyY4Kkh4WDMUYKRZUq0h1o0w2rrrxBqM7VoeUVqgb27xlEMXTnYt4g==", + "requires": { + "accepts": "~1.3.7", + "array-flatten": "1.1.1", + "body-parser": "1.19.0", + "content-disposition": "0.5.3", + "content-type": "~1.0.4", + "cookie": "0.4.0", + "cookie-signature": "1.0.6", + "debug": "2.6.9", + "depd": "~1.1.2", + "encodeurl": "~1.0.2", + "escape-html": "~1.0.3", + "etag": "~1.8.1", + "finalhandler": "~1.1.2", + "fresh": "0.5.2", + "merge-descriptors": "1.0.1", + "methods": "~1.1.2", + "on-finished": "~2.3.0", + "parseurl": "~1.3.3", + "path-to-regexp": "0.1.7", + "proxy-addr": "~2.0.5", + "qs": "6.7.0", + "range-parser": "~1.2.1", + "safe-buffer": "5.1.2", + "send": "0.17.1", + "serve-static": "1.14.1", + "setprototypeof": "1.1.1", + "statuses": "~1.5.0", + "type-is": "~1.6.18", + "utils-merge": "1.0.1", + "vary": "~1.1.2" + }, + "dependencies": { + "debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "requires": { + "ms": "2.0.0" + } + }, + "ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" + } + } + }, + "extend": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/extend/-/extend-3.0.2.tgz", + "integrity": "sha512-fjquC59cD7CyW6urNXK0FBufkZcoiGG80wTuPujX590cB5Ttln20E2UB4S/WARVqhXffZl2LNgS+gQdPIIim/g==" + }, + "external-editor": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/external-editor/-/external-editor-3.1.0.tgz", + "integrity": "sha512-hMQ4CX1p1izmuLYyZqLMO/qGNw10wSv9QDCPfzXfyFrOaCSSoRfqE1Kf1s5an66J5JZC62NewG+mK49jOCtQew==", + "dev": true, + "requires": { + "chardet": "^0.7.0", + "iconv-lite": "^0.4.24", + "tmp": "^0.0.33" + } + }, + "extsprintf": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/extsprintf/-/extsprintf-1.3.0.tgz", + "integrity": "sha1-lpGEQOMEGnpBT4xS48V06zw+HgU=" + }, + "fast-deep-equal": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz", + "integrity": "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==" + }, + "fast-json-stable-stringify": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz", + "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==" + }, + "fast-levenshtein": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz", + "integrity": "sha1-PYpcZog6FqMMqGQ+hR8Zuqd5eRc=", + "dev": true + }, + "fast-safe-stringify": { + "version": "2.0.8", + "resolved": "https://registry.npmjs.org/fast-safe-stringify/-/fast-safe-stringify-2.0.8.tgz", + "integrity": "sha512-lXatBjf3WPjmWD6DpIZxkeSsCOwqI0maYMpgDlx8g4U2qi4lbjA9oH/HD2a87G+KfsUmo5WbJFmqBZlPxtptag==" + }, + "fecha": { + "version": "4.2.1", + "resolved": "https://registry.npmjs.org/fecha/-/fecha-4.2.1.tgz", + "integrity": "sha512-MMMQ0ludy/nBs1/o0zVOiKTpG7qMbonKUzjJgQFEuvq6INZ1OraKPRAWkBq5vlKLOUMpmNYG1JoN3oDPUQ9m3Q==" + }, + "figures": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/figures/-/figures-3.2.0.tgz", + "integrity": "sha512-yaduQFRKLXYOGgEn6AZau90j3ggSOyiqXU0F9JZfeXYhNa+Jk4X+s45A2zg5jns87GAFa34BBm2kXw4XpNcbdg==", + "dev": true, + "requires": { + "escape-string-regexp": "^1.0.5" + } + }, + "file-entry-cache": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-5.0.1.tgz", + "integrity": "sha512-bCg29ictuBaKUwwArK4ouCaqDgLZcysCFLmM/Yn/FDoqndh/9vNuQfXRDvTuXKLxfD/JtZQGKFT8MGcJBK644g==", + "dev": true, + "requires": { + "flat-cache": "^2.0.1" + } + }, + "fill-range": { + "version": "7.0.1", + "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz", + "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", + "dev": true, + "requires": { + "to-regex-range": "^5.0.1" + } + }, + "finalhandler": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.1.2.tgz", + "integrity": "sha512-aAWcW57uxVNrQZqFXjITpW3sIUQmHGG3qSb9mUah9MgMC4NeWhNOlNjXEYq3HjRAvL6arUviZGGJsBg6z0zsWA==", + "requires": { + "debug": "2.6.9", + "encodeurl": "~1.0.2", + "escape-html": "~1.0.3", + "on-finished": "~2.3.0", + "parseurl": "~1.3.3", + "statuses": "~1.5.0", + "unpipe": "~1.0.0" + }, + "dependencies": { + "debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "requires": { + "ms": "2.0.0" + } + }, + "ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" + } + } + }, + "find-root": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/find-root/-/find-root-1.1.0.tgz", + "integrity": "sha512-NKfW6bec6GfKc0SGx1e07QZY9PE99u0Bft/0rzSD5k3sO/vwkVUpDUKVm5Gpp5Ue3YfShPFTX2070tDs5kB9Ng==", + "dev": true + }, + "find-up": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-1.1.2.tgz", + "integrity": "sha1-ay6YIrGizgpgq2TWEOzK1TyyTQ8=", + "requires": { + "path-exists": "^2.0.0", + "pinkie-promise": "^2.0.0" + } + }, + "flat-cache": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-2.0.1.tgz", + "integrity": "sha512-LoQe6yDuUMDzQAEH8sgmh4Md6oZnc/7PjtwjNFSzveXqSHt6ka9fPBuso7IGf9Rz4uqnSnWiFH2B/zj24a5ReA==", + "dev": true, + "requires": { + "flatted": "^2.0.0", + "rimraf": "2.6.3", + "write": "1.0.3" + }, + "dependencies": { + "glob": { + "version": "7.1.7", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.7.tgz", + "integrity": "sha512-OvD9ENzPLbegENnYP5UUfJIirTg4+XwMWGaQfQTY0JenxNvvIKP3U3/tAQSPIu/lHxXYSZmpXlUHeqAIdKzBLQ==", + "dev": true, + "requires": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.0.4", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + } + }, + "rimraf": { + "version": "2.6.3", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.6.3.tgz", + "integrity": "sha512-mwqeW5XsA2qAejG46gYdENaxXjx9onRNCfn7L0duuP4hCuTIi/QO7PDK07KJfp1d+izWPrzEJDcSqBa0OZQriA==", + "dev": true, + "requires": { + "glob": "^7.1.3" + } + } + } + }, + "flatted": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/flatted/-/flatted-2.0.2.tgz", + "integrity": "sha512-r5wGx7YeOwNWNlCA0wQ86zKyDLMQr+/RB8xy74M4hTphfmjlijTSSXGuH8rnvKZnfT9i+75zmd8jcKdMR4O6jA==", + "dev": true + }, + "fn.name": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/fn.name/-/fn.name-1.1.0.tgz", + "integrity": "sha512-GRnmB5gPyJpAhTQdSZTSp9uaPSvl09KoYcMQtsB9rQoOmzs9dH6ffeccH+Z+cv6P68Hu5bC6JjRh4Ah/mHSNRw==" + }, + "follow-redirects": { + "version": "1.5.10", + "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.5.10.tgz", + "integrity": "sha512-0V5l4Cizzvqt5D44aTXbFZz+FtyXV1vrDN6qrelxtfYQKW0KO0W2T/hkE8xvGa/540LkZlkaUjO4ailYTFtHVQ==", + "requires": { + "debug": "=3.1.0" + }, + "dependencies": { + "debug": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.1.0.tgz", + "integrity": "sha512-OX8XqP7/1a9cqkxYw2yXss15f26NKWBpDXQd0/uK/KPqdQhxbPa994hnzjcE2VqQpDslf55723cKPUOGSmMY3g==", + "requires": { + "ms": "2.0.0" + } + }, + "ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" + } + } + }, + "forever-agent": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/forever-agent/-/forever-agent-0.6.1.tgz", + "integrity": "sha1-+8cfDEGt6zf5bFd60e1C2P2sypE=" + }, + "form-data": { + "version": "2.3.3", + "resolved": "https://registry.npmjs.org/form-data/-/form-data-2.3.3.tgz", + "integrity": "sha512-1lLKB2Mu3aGP1Q/2eCOx0fNbRMe7XdwktwOruhfqqd0rIJWwN4Dh+E3hrPSlDCXnSR7UtZ1N38rVXm+6+MEhJQ==", + "requires": { + "asynckit": "^0.4.0", + "combined-stream": "^1.0.6", + "mime-types": "^2.1.12" + } + }, + "forwarded": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.2.0.tgz", + "integrity": "sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow==" + }, + "fresh": { + "version": "0.5.2", + "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz", + "integrity": "sha1-PYyt2Q2XZWn6g1qx+OSyOhBWBac=" + }, + "fs.realpath": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", + "integrity": "sha1-FQStJSMVjKpA20onh8sBQRmU6k8=", + "dev": true + }, + "fsevents": { + "version": "2.3.2", + "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.2.tgz", + "integrity": "sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==", + "dev": true, + "optional": true + }, + "function-bind": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.1.tgz", + "integrity": "sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A==" + }, + "functional-red-black-tree": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/functional-red-black-tree/-/functional-red-black-tree-1.0.1.tgz", + "integrity": "sha1-GwqzvVU7Kg1jmdKcDj6gslIHgyc=", + "dev": true + }, + "get-caller-file": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-1.0.3.tgz", + "integrity": "sha512-3t6rVToeoZfYSGd8YoLFR2DJkiQrIiUrGcjvFX2mDw3bn6k2OtwHN0TNCLbBO+w8qTvimhDkv+LSscbJY1vE6w==" + }, + "get-intrinsic": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.1.1.tgz", + "integrity": "sha512-kWZrnVM42QCiEA2Ig1bG8zjoIMOgxWwYCEeNdwY6Tv/cOSeGpcoX4pXHfKUxNKVoArnrEr2e9srnAxxGIraS9Q==", + "dev": true, + "requires": { + "function-bind": "^1.1.1", + "has": "^1.0.3", + "has-symbols": "^1.0.1" + } + }, + "get-parameter-names": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/get-parameter-names/-/get-parameter-names-0.3.0.tgz", + "integrity": "sha1-LSI3zVkubFuFmrLv2rQ18Ajlu5c=" + }, + "get-stdin": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/get-stdin/-/get-stdin-7.0.0.tgz", + "integrity": "sha512-zRKcywvrXlXsA0v0i9Io4KDRaAw7+a1ZpjRwl9Wox8PFlVCCHra7E9c4kqXCoCM9nR5tBkaTTZRBoCm60bFqTQ==", + "dev": true + }, + "get-stream": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-4.1.0.tgz", + "integrity": "sha512-GMat4EJ5161kIy2HevLlr4luNjBgvmj413KaQA7jt4V8B4RDsfpHk7WQ9GVqfYyyx8OS/L66Kox+rJRNklLK7w==", + "dev": true, + "requires": { + "pump": "^3.0.0" + } + }, + "getpass": { + "version": "0.1.7", + "resolved": "https://registry.npmjs.org/getpass/-/getpass-0.1.7.tgz", + "integrity": "sha1-Xv+OPmhNVprkyysSgmBOi6YhSfo=", + "requires": { + "assert-plus": "^1.0.0" + } + }, + "glob": { + "version": "6.0.4", + "resolved": "https://registry.npmjs.org/glob/-/glob-6.0.4.tgz", + "integrity": "sha1-DwiGD2oVUSey+t1PnOJLGqtuTSI=", + "optional": true, + "requires": { + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "2 || 3", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + } + }, + "glob-parent": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", + "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", + "dev": true, + "requires": { + "is-glob": "^4.0.1" + } + }, + "global-dirs": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/global-dirs/-/global-dirs-2.1.0.tgz", + "integrity": "sha512-MG6kdOUh/xBnyo9cJFeIKkLEc1AyFq42QTU4XiX51i2NEdxLxLWXIjEjmqKeSuKR7pAZjTqUVoT2b2huxVLgYQ==", + "dev": true, + "requires": { + "ini": "1.3.7" + } + }, + "globals": { + "version": "12.4.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-12.4.0.tgz", + "integrity": "sha512-BWICuzzDvDoH54NHKCseDanAhE3CeDorgDL5MT6LMXXj2WCnd9UC2szdk4AWLfjdgNBCXLUanXYcpBBKOSWGwg==", + "dev": true, + "requires": { + "type-fest": "^0.8.1" + } + }, + "got": { + "version": "9.6.0", + "resolved": "https://registry.npmjs.org/got/-/got-9.6.0.tgz", + "integrity": "sha512-R7eWptXuGYxwijs0eV+v3o6+XH1IqVK8dJOEecQfTmkncw9AV4dcw/Dhxi8MdlqPthxxpZyizMzyg8RTmEsG+Q==", + "dev": true, + "requires": { + "@sindresorhus/is": "^0.14.0", + "@szmarczak/http-timer": "^1.1.2", + "cacheable-request": "^6.0.0", + "decompress-response": "^3.3.0", + "duplexer3": "^0.1.4", + "get-stream": "^4.1.0", + "lowercase-keys": "^1.0.1", + "mimic-response": "^1.0.1", + "p-cancelable": "^1.0.0", + "to-readable-stream": "^1.0.0", + "url-parse-lax": "^3.0.0" + } + }, + "graceful-fs": { + "version": "4.2.6", + "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.6.tgz", + "integrity": "sha512-nTnJ528pbqxYanhpDYsi4Rd8MAeaBA67+RZ10CM1m3bTAVFEDcd5AuA4a6W5YkGZ1iNXHzZz8T6TBKLeBuNriQ==" + }, + "handlebars": { + "version": "4.7.7", + "resolved": "https://registry.npmjs.org/handlebars/-/handlebars-4.7.7.tgz", + "integrity": "sha512-aAcXm5OAfE/8IXkcZvCepKU3VzW1/39Fb5ZuqMtgI/hT8X2YgoMvBY5dLhq/cpOvw7Lk1nK/UF71aLG/ZnVYRA==", + "requires": { + "minimist": "^1.2.5", + "neo-async": "^2.6.0", + "source-map": "^0.6.1", + "uglify-js": "^3.1.4", + "wordwrap": "^1.0.0" + } + }, + "har-schema": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/har-schema/-/har-schema-2.0.0.tgz", + "integrity": "sha1-qUwiJOvKwEeCoNkDVSHyRzW37JI=" + }, + "har-validator": { + "version": "5.1.5", + "resolved": "https://registry.npmjs.org/har-validator/-/har-validator-5.1.5.tgz", + "integrity": "sha512-nmT2T0lljbxdQZfspsno9hgrG3Uir6Ks5afism62poxqBM6sDnMEuPmzTq8XN0OEwqKLLdh1jQI3qyE66Nzb3w==", + "requires": { + "ajv": "^6.12.3", + "har-schema": "^2.0.0" + } + }, + "has": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/has/-/has-1.0.3.tgz", + "integrity": "sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw==", + "requires": { + "function-bind": "^1.1.1" + } + }, + "has-bigints": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/has-bigints/-/has-bigints-1.0.1.tgz", + "integrity": "sha512-LSBS2LjbNBTf6287JEbEzvJgftkF5qFkmCo9hDRpAzKhUOlJ+hx8dd4USs00SgsUNwc4617J9ki5YtEClM2ffA==", + "dev": true + }, + "has-flag": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", + "integrity": "sha1-tdRU3CGZriJWmfNGfloH87lVuv0=", + "dev": true + }, + "has-symbols": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.0.2.tgz", + "integrity": "sha512-chXa79rL/UC2KlX17jo3vRGz0azaWEx5tGqZg5pO3NUyEJVB17dMruQlzCCOfUvElghKcm5194+BCRvi2Rv/Gw==", + "dev": true + }, + "has-yarn": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/has-yarn/-/has-yarn-2.1.0.tgz", + "integrity": "sha512-UqBRqi4ju7T+TqGNdqAO0PaSVGsDGJUBQvk9eUWNGRY1CFGDzYhLWoM7JQEemnlvVcv/YEmc2wNW8BC24EnUsw==", + "dev": true + }, + "hosted-git-info": { + "version": "2.8.9", + "resolved": "https://registry.npmjs.org/hosted-git-info/-/hosted-git-info-2.8.9.tgz", + "integrity": "sha512-mxIDAb9Lsm6DoOJ7xH+5+X4y1LU/4Hi50L9C5sIswK3JzULS4bwk1FvjdBgvYR4bzT4tuUQiC15FE2f5HbLvYw==" + }, + "hpagent": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/hpagent/-/hpagent-0.1.2.tgz", + "integrity": "sha512-ePqFXHtSQWAFXYmj+JtOTHr84iNrII4/QRlAAPPE+zqnKy4xJo7Ie1Y4kC7AdB+LxLxSTTzBMASsEcy0q8YyvQ==" + }, + "http-cache-semantics": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/http-cache-semantics/-/http-cache-semantics-4.1.0.tgz", + "integrity": "sha512-carPklcUh7ROWRK7Cv27RPtdhYhUsela/ue5/jKzjegVvXDqM2ILE9Q2BGn9JZJh1g87cp56su/FgQSzcWS8cQ==", + "dev": true + }, + "http-errors": { + "version": "1.7.2", + "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.7.2.tgz", + "integrity": "sha512-uUQBt3H/cSIVfch6i1EuPNy/YsRSOUBXTVfZ+yR7Zjez3qjBz6i9+i4zjNaoqcoFVI4lQJ5plg63TvGfRSDCRg==", + "requires": { + "depd": "~1.1.2", + "inherits": "2.0.3", + "setprototypeof": "1.1.1", + "statuses": ">= 1.5.0 < 2", + "toidentifier": "1.0.0" + } + }, + "http-proxy-agent": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/http-proxy-agent/-/http-proxy-agent-4.0.1.tgz", + "integrity": "sha512-k0zdNgqWTGA6aeIRVpvfVob4fL52dTfaehylg0Y4UvSySvOq/Y+BOyPrgpUrA7HylqvU8vIZGsRuXmspskV0Tg==", + "requires": { + "@tootallnate/once": "1", + "agent-base": "6", + "debug": "4" + } + }, + "http-signature": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/http-signature/-/http-signature-1.2.0.tgz", + "integrity": "sha1-muzZJRFHcvPZW2WmCruPfBj7rOE=", + "requires": { + "assert-plus": "^1.0.0", + "jsprim": "^1.2.2", + "sshpk": "^1.7.0" + } + }, + "https-proxy-agent": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/https-proxy-agent/-/https-proxy-agent-5.0.0.tgz", + "integrity": "sha512-EkYm5BcKUGiduxzSt3Eppko+PiNWNEpa4ySk9vTC6wDsQJW9rHSa+UhGNJoRYp7bz6Ht1eaRIa6QaJqO5rCFbA==", + "requires": { + "agent-base": "6", + "debug": "4" + } + }, + "iconv-lite": { + "version": "0.4.24", + "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", + "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", + "requires": { + "safer-buffer": ">= 2.1.2 < 3" + } + }, + "ignore": { + "version": "4.0.6", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-4.0.6.tgz", + "integrity": "sha512-cyFDKrqc/YdcWFniJhzI42+AzS+gNwmUzOSFcRCQYwySuBBBy/KjuxWLZ/FHEH6Moq1NizMOBWyTcv8O4OZIMg==", + "dev": true + }, + "ignore-by-default": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/ignore-by-default/-/ignore-by-default-1.0.1.tgz", + "integrity": "sha1-SMptcvbGo68Aqa1K5odr44ieKwk=", + "dev": true + }, + "import-fresh": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-3.3.0.tgz", + "integrity": "sha512-veYYhQa+D1QBKznvhUHxb8faxlrwUnxseDAbAp457E0wLNio2bOSKnjYDhMj+YiAq61xrMGhQk9iXVk5FzgQMw==", + "dev": true, + "requires": { + "parent-module": "^1.0.0", + "resolve-from": "^4.0.0" + } + }, + "import-lazy": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/import-lazy/-/import-lazy-2.1.0.tgz", + "integrity": "sha1-BWmOPUXIjo1+nZLLBYTnfwlvPkM=", + "dev": true + }, + "imurmurhash": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", + "integrity": "sha1-khi5srkoojixPcT7a21XbyMUU+o=", + "dev": true + }, + "inflection": { + "version": "1.13.1", + "resolved": "https://registry.npmjs.org/inflection/-/inflection-1.13.1.tgz", + "integrity": "sha512-dldYtl2WlN0QDkIDtg8+xFwOS2Tbmp12t1cHa5/YClU6ZQjTFm7B66UcVbh9NQB+HvT5BAd2t5+yKsBkw5pcqA==" + }, + "inflight": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", + "integrity": "sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk=", + "requires": { + "once": "^1.3.0", + "wrappy": "1" + } + }, + "inherits": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", + "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=" + }, + "ini": { + "version": "1.3.7", + "resolved": "https://registry.npmjs.org/ini/-/ini-1.3.7.tgz", + "integrity": "sha512-iKpRpXP+CrP2jyrxvg1kMUpXDyRUFDWurxbnVT1vQPx+Wz9uCYsMIqYuSBLV+PAaZG/d7kRLKRFc9oDMsH+mFQ==", + "dev": true + }, + "inquirer": { + "version": "7.3.3", + "resolved": "https://registry.npmjs.org/inquirer/-/inquirer-7.3.3.tgz", + "integrity": "sha512-JG3eIAj5V9CwcGvuOmoo6LB9kbAYT8HXffUl6memuszlwDC/qvFAJw49XJ5NROSFNPxp3iQg1GqkFhaY/CR0IA==", + "dev": true, + "requires": { + "ansi-escapes": "^4.2.1", + "chalk": "^4.1.0", + "cli-cursor": "^3.1.0", + "cli-width": "^3.0.0", + "external-editor": "^3.0.3", + "figures": "^3.0.0", + "lodash": "^4.17.19", + "mute-stream": "0.0.8", + "run-async": "^2.4.0", + "rxjs": "^6.6.0", + "string-width": "^4.1.0", + "strip-ansi": "^6.0.0", + "through": "^2.3.6" + }, + "dependencies": { + "ansi-regex": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.0.tgz", + "integrity": "sha512-bY6fj56OUQ0hU1KjFNDQuJFezqKdrAyFdIevADiqrWHwSlbmBNMHp5ak2f40Pm8JTFyM2mqxkG6ngkHO11f/lg==", + "dev": true + }, + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } + }, + "chalk": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.1.tgz", + "integrity": "sha512-diHzdDKxcU+bAsUboHLPEDQiw0qEe0qd7SYUn3HgcFlWgbDcfLGswOHYeGrHKzG9z6UYf01d9VFMfZxPM1xZSg==", + "dev": true, + "requires": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true + }, + "is-fullwidth-code-point": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", + "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", + "dev": true + }, + "string-width": { + "version": "4.2.2", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.2.tgz", + "integrity": "sha512-XBJbT3N4JhVumXE0eoLU9DCjcaF92KLNqTmFCnG1pf8duUxFGwtP6AD6nkjw9a3IdiRtL3E2w3JDiE/xi3vOeA==", + "dev": true, + "requires": { + "emoji-regex": "^8.0.0", + "is-fullwidth-code-point": "^3.0.0", + "strip-ansi": "^6.0.0" + } + }, + "strip-ansi": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.0.tgz", + "integrity": "sha512-AuvKTrTfQNYNIctbR1K/YGTR1756GycPsg7b9bdV9Duqur4gv6aKqHXah67Z8ImS7WEz5QVcOtlfW2rZEugt6w==", + "dev": true, + "requires": { + "ansi-regex": "^5.0.0" + } + }, + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + } + } + }, + "invert-kv": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/invert-kv/-/invert-kv-1.0.0.tgz", + "integrity": "sha1-EEqOSqym09jNFXqO+L+rLXo//bY=" + }, + "ipaddr.js": { + "version": "1.9.1", + "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz", + "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==" + }, + "is-arrayish": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/is-arrayish/-/is-arrayish-0.2.1.tgz", + "integrity": "sha1-d8mYQFJ6qOyxqLppe4BkWnqSap0=" + }, + "is-bigint": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/is-bigint/-/is-bigint-1.0.2.tgz", + "integrity": "sha512-0JV5+SOCQkIdzjBK9buARcV804Ddu7A0Qet6sHi3FimE9ne6m4BGQZfRn+NZiXbBk4F4XmHfDZIipLj9pX8dSA==", + "dev": true + }, + "is-binary-path": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz", + "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==", + "dev": true, + "requires": { + "binary-extensions": "^2.0.0" + } + }, + "is-boolean-object": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/is-boolean-object/-/is-boolean-object-1.1.1.tgz", + "integrity": "sha512-bXdQWkECBUIAcCkeH1unwJLIpZYaa5VvuygSyS/c2lf719mTKZDU5UdDRlpd01UjADgmW8RfqaP+mRaVPdr/Ng==", + "dev": true, + "requires": { + "call-bind": "^1.0.2" + } + }, + "is-callable": { + "version": "1.2.3", + "resolved": "https://registry.npmjs.org/is-callable/-/is-callable-1.2.3.tgz", + "integrity": "sha512-J1DcMe8UYTBSrKezuIUTUwjXsho29693unXM2YhJUTR2txK/eG47bvNa/wipPFmZFgr/N6f1GA66dv0mEyTIyQ==", + "dev": true + }, + "is-ci": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/is-ci/-/is-ci-2.0.0.tgz", + "integrity": "sha512-YfJT7rkpQB0updsdHLGWrvhBJfcfzNNawYDNIyQXJz0IViGf75O8EBPKSdvw2rF+LGCsX4FZ8tcr3b19LcZq4w==", + "dev": true, + "requires": { + "ci-info": "^2.0.0" + } + }, + "is-core-module": { + "version": "2.5.0", + "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.5.0.tgz", + "integrity": "sha512-TXCMSDsEHMEEZ6eCA8rwRDbLu55MRGmrctljsBX/2v1d9/GzqHOxW5c5oPSgrUt2vBFXebu9rGqckXGPWOlYpg==", + "requires": { + "has": "^1.0.3" + } + }, + "is-date-object": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/is-date-object/-/is-date-object-1.0.4.tgz", + "integrity": "sha512-/b4ZVsG7Z5XVtIxs/h9W8nvfLgSAyKYdtGWQLbqy6jA1icmgjf8WCoTKgeS4wy5tYaPePouzFMANbnj94c2Z+A==", + "dev": true + }, + "is-extglob": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", + "integrity": "sha1-qIwCU1eR8C7TfHahueqXc8gz+MI=", + "dev": true + }, + "is-fullwidth-code-point": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-1.0.0.tgz", + "integrity": "sha1-754xOG8DGn8NZDr4L95QxFfvAMs=", + "requires": { + "number-is-nan": "^1.0.0" + } + }, + "is-glob": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.1.tgz", + "integrity": "sha512-5G0tKtBTFImOqDnLB2hG6Bp2qcKEFduo4tZu9MT/H6NQv/ghhy30o55ufafxJ/LdH79LLs2Kfrn85TLKyA7BUg==", + "dev": true, + "requires": { + "is-extglob": "^2.1.1" + } + }, + "is-installed-globally": { + "version": "0.3.2", + "resolved": "https://registry.npmjs.org/is-installed-globally/-/is-installed-globally-0.3.2.tgz", + "integrity": "sha512-wZ8x1js7Ia0kecP/CHM/3ABkAmujX7WPvQk6uu3Fly/Mk44pySulQpnHG46OMjHGXApINnV4QhY3SWnECO2z5g==", + "dev": true, + "requires": { + "global-dirs": "^2.0.1", + "is-path-inside": "^3.0.1" + } + }, + "is-negative-zero": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/is-negative-zero/-/is-negative-zero-2.0.1.tgz", + "integrity": "sha512-2z6JzQvZRa9A2Y7xC6dQQm4FSTSTNWjKIYYTt4246eMTJmIo0Q+ZyOsU66X8lxK1AbB92dFeglPLrhwpeRKO6w==", + "dev": true + }, + "is-npm": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/is-npm/-/is-npm-4.0.0.tgz", + "integrity": "sha512-96ECIfh9xtDDlPylNPXhzjsykHsMJZ18ASpaWzQyBr4YRTcVjUvzaHayDAES2oU/3KpljhHUjtSRNiDwi0F0ig==", + "dev": true + }, + "is-number": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", + "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", + "dev": true + }, + "is-number-object": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/is-number-object/-/is-number-object-1.0.5.tgz", + "integrity": "sha512-RU0lI/n95pMoUKu9v1BZP5MBcZuNSVJkMkAG2dJqC4z2GlkGUNeH68SuHuBKBD/XFe+LHZ+f9BKkLET60Niedw==", + "dev": true + }, + "is-obj": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/is-obj/-/is-obj-2.0.0.tgz", + "integrity": "sha512-drqDG3cbczxxEJRoOXcOjtdp1J/lyp1mNn0xaznRs8+muBhgQcrnbspox5X5fOw0HnMnbfDzvnEMEtqDEJEo8w==", + "dev": true + }, + "is-path-inside": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-3.0.3.tgz", + "integrity": "sha512-Fd4gABb+ycGAmKou8eMftCupSir5lRxqf4aD/vd0cD2qc4HL07OjCeuHMr8Ro4CoMaeCKDB0/ECBOVWjTwUvPQ==", + "dev": true + }, + "is-regex": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/is-regex/-/is-regex-1.1.3.tgz", + "integrity": "sha512-qSVXFz28HM7y+IWX6vLCsexdlvzT1PJNFSBuaQLQ5o0IEw8UDYW6/2+eCMVyIsbM8CNLX2a/QWmSpyxYEHY7CQ==", + "dev": true, + "requires": { + "call-bind": "^1.0.2", + "has-symbols": "^1.0.2" + } + }, + "is-stream": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/is-stream/-/is-stream-2.0.1.tgz", + "integrity": "sha512-hFoiJiTl63nn+kstHGBtewWSKnQLpyb155KHheA1l39uvtO9nWIop1p3udqPcUd/xbF1VLMO4n7OI6p7RbngDg==" + }, + "is-string": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/is-string/-/is-string-1.0.6.tgz", + "integrity": "sha512-2gdzbKUuqtQ3lYNrUTQYoClPhm7oQu4UdpSZMp1/DGgkHBT8E2Z1l0yMdb6D4zNAxwDiMv8MdulKROJGNl0Q0w==", + "dev": true + }, + "is-symbol": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/is-symbol/-/is-symbol-1.0.4.tgz", + "integrity": "sha512-C/CPBqKWnvdcxqIARxyOh4v1UUEOCHpgDa0WYgpKDFMszcrPcffg5uhwSgPCLD2WWxmq6isisz87tzT01tuGhg==", + "dev": true, + "requires": { + "has-symbols": "^1.0.2" + } + }, + "is-typedarray": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-typedarray/-/is-typedarray-1.0.0.tgz", + "integrity": "sha1-5HnICFjfDBsR3dppQPlgEfzaSpo=" + }, + "is-utf8": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/is-utf8/-/is-utf8-0.2.1.tgz", + "integrity": "sha1-Sw2hRCEE0bM2NA6AeX6GXPOffXI=" + }, + "is-yarn-global": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/is-yarn-global/-/is-yarn-global-0.3.0.tgz", + "integrity": "sha512-VjSeb/lHmkoyd8ryPVIKvOCn4D1koMqY+vqyjjUfc3xyKtP4dYOxM44sZrnqQSzSds3xyOrUTLTC9LVCVgLngw==", + "dev": true + }, + "isarray": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", + "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE=" + }, + "isexe": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", + "integrity": "sha1-6PvzdNxVb/iUehDcsFctYz8s+hA=", + "dev": true + }, + "isstream": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/isstream/-/isstream-0.1.2.tgz", + "integrity": "sha1-R+Y/evVa+m+S4VAOaQ64uFKcCZo=" + }, + "js-tokens": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz", + "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==", + "dev": true + }, + "js-yaml": { + "version": "3.14.1", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.14.1.tgz", + "integrity": "sha512-okMH7OXXJ7YrN9Ok3/SXrnu4iX9yOk+25nqX4imS2npuvTYDmo/QEZoqwZkYaIDk3jVvBOTOIEgEhaLOynBS9g==", + "requires": { + "argparse": "^1.0.7", + "esprima": "^4.0.0" + } + }, + "jsbn": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/jsbn/-/jsbn-0.1.1.tgz", + "integrity": "sha1-peZUwuWi3rXyAdls77yoDA7y9RM=" + }, + "json-buffer": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/json-buffer/-/json-buffer-3.0.0.tgz", + "integrity": "sha1-Wx85evx11ne96Lz8Dkfh+aPZqJg=", + "dev": true + }, + "json-parse-better-errors": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/json-parse-better-errors/-/json-parse-better-errors-1.0.2.tgz", + "integrity": "sha512-mrqyZKfX5EhL7hvqcV6WG1yYjnjeuYDzDhhcAAUrq8Po85NBQBJP+ZDUT75qZQ98IkUoBqdkExkukOU7Ts2wrw==", + "dev": true + }, + "json-schema": { + "version": "0.2.3", + "resolved": "https://registry.npmjs.org/json-schema/-/json-schema-0.2.3.tgz", + "integrity": "sha1-tIDIkuWaLwWVTOcnvT8qTogvnhM=" + }, + "json-schema-traverse": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz", + "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==" + }, + "json-stable-stringify-without-jsonify": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz", + "integrity": "sha1-nbe1lJatPzz+8wp1FC0tkwrXJlE=", + "dev": true + }, + "json-stringify-safe": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/json-stringify-safe/-/json-stringify-safe-5.0.1.tgz", + "integrity": "sha1-Epai1Y/UXxmg9s4B1lcB4sc1tus=" + }, + "json5": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/json5/-/json5-2.2.0.tgz", + "integrity": "sha512-f+8cldu7X/y7RAJurMEJmdoKXGB/X550w2Nr3tTbezL6RwEE/iMcm+tZnXeoZtKuOq6ft8+CqzEkrIgx1fPoQA==", + "requires": { + "minimist": "^1.2.5" + } + }, + "jsonwebtoken": { + "version": "8.5.1", + "resolved": "https://registry.npmjs.org/jsonwebtoken/-/jsonwebtoken-8.5.1.tgz", + "integrity": "sha512-XjwVfRS6jTMsqYs0EsuJ4LGxXV14zQybNd4L2r0UvbVnSF9Af8x7p5MzbJ90Ioz/9TI41/hTCvznF/loiSzn8w==", + "requires": { + "jws": "^3.2.2", + "lodash.includes": "^4.3.0", + "lodash.isboolean": "^3.0.3", + "lodash.isinteger": "^4.0.4", + "lodash.isnumber": "^3.0.3", + "lodash.isplainobject": "^4.0.6", + "lodash.isstring": "^4.0.1", + "lodash.once": "^4.0.0", + "ms": "^2.1.1", + "semver": "^5.6.0" + } + }, + "jsprim": { + "version": "1.4.1", + "resolved": "https://registry.npmjs.org/jsprim/-/jsprim-1.4.1.tgz", + "integrity": "sha1-MT5mvB5cwG5Di8G3SZwuXFastqI=", + "requires": { + "assert-plus": "1.0.0", + "extsprintf": "1.3.0", + "json-schema": "0.2.3", + "verror": "1.10.0" + } + }, + "jsx-ast-utils": { + "version": "2.4.1", + "resolved": "https://registry.npmjs.org/jsx-ast-utils/-/jsx-ast-utils-2.4.1.tgz", + "integrity": "sha512-z1xSldJ6imESSzOjd3NNkieVJKRlKYSOtMG8SFyCj2FIrvSaSuli/WjpBkEzCBoR9bYYYFgqJw61Xhu7Lcgk+w==", + "dev": true, + "requires": { + "array-includes": "^3.1.1", + "object.assign": "^4.1.0" + } + }, + "jwa": { + "version": "1.4.1", + "resolved": "https://registry.npmjs.org/jwa/-/jwa-1.4.1.tgz", + "integrity": "sha512-qiLX/xhEEFKUAJ6FiBMbes3w9ATzyk5W7Hvzpa/SLYdxNtng+gcurvrI7TbACjIXlsJyr05/S1oUhZrc63evQA==", + "requires": { + "buffer-equal-constant-time": "1.0.1", + "ecdsa-sig-formatter": "1.0.11", + "safe-buffer": "^5.0.1" + } + }, + "jwks-rsa": { + "version": "1.12.3", + "resolved": "https://registry.npmjs.org/jwks-rsa/-/jwks-rsa-1.12.3.tgz", + "integrity": "sha512-cFipFDeYYaO9FhhYJcZWX/IyZgc0+g316rcHnDpT2dNRNIE/lMOmWKKqp09TkJoYlNFzrEVODsR4GgXJMgWhnA==", + "requires": { + "@types/express-jwt": "0.0.42", + "axios": "^0.21.1", + "debug": "^4.1.0", + "http-proxy-agent": "^4.0.1", + "https-proxy-agent": "^5.0.0", + "jsonwebtoken": "^8.5.1", + "limiter": "^1.1.5", + "lru-memoizer": "^2.1.2", + "ms": "^2.1.2", + "proxy-from-env": "^1.1.0" + }, + "dependencies": { + "axios": { + "version": "0.21.1", + "resolved": "https://registry.npmjs.org/axios/-/axios-0.21.1.tgz", + "integrity": "sha512-dKQiRHxGD9PPRIUNIWvZhPTPpl1rf/OxTYKsqKUDjBwYylTvV7SjSHJb9ratfyzM6wCdLCOYLzs73qpg5c4iGA==", + "requires": { + "follow-redirects": "^1.10.0" + } + }, + "follow-redirects": { + "version": "1.14.1", + "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.14.1.tgz", + "integrity": "sha512-HWqDgT7ZEkqRzBvc2s64vSZ/hfOceEol3ac/7tKwzuvEyWx3/4UegXh5oBOIotkGsObyk3xznnSRVADBgWSQVg==" + } + } + }, + "jws": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/jws/-/jws-3.2.2.tgz", + "integrity": "sha512-YHlZCB6lMTllWDtSPHz/ZXTsi8S00usEV6v1tjq8tOUZzw7DpSDWVXjXDre6ed1w/pd495ODpHZYSdkRTsa0HA==", + "requires": { + "jwa": "^1.4.1", + "safe-buffer": "^5.0.1" + } + }, + "keyv": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/keyv/-/keyv-3.1.0.tgz", + "integrity": "sha512-9ykJ/46SN/9KPM/sichzQ7OvXyGDYKGTaDlKMGCAlg2UK8KRy4jb0d8sFc+0Tt0YYnThq8X2RZgCg74RPxgcVA==", + "dev": true, + "requires": { + "json-buffer": "3.0.0" + } + }, + "kuler": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/kuler/-/kuler-2.0.0.tgz", + "integrity": "sha512-Xq9nH7KlWZmXAtodXDDRE7vs6DU1gTU8zYDHDiWLSip45Egwq3plLHzPn27NgvzL2r1LMPC1vdqh98sQxtqj4A==" + }, + "latest-version": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/latest-version/-/latest-version-5.1.0.tgz", + "integrity": "sha512-weT+r0kTkRQdCdYCNtkMwWXQTMEswKrFBkm4ckQOMVhhqhIMI1UT2hMj+1iigIhgSZm5gTmrRXBNoGUgaTY1xA==", + "dev": true, + "requires": { + "package-json": "^6.3.0" + } + }, + "lcid": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/lcid/-/lcid-1.0.0.tgz", + "integrity": "sha1-MIrMr6C8SDo4Z7S28rlQYlHRuDU=", + "requires": { + "invert-kv": "^1.0.0" + } + }, + "levn": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/levn/-/levn-0.3.0.tgz", + "integrity": "sha1-OwmSTt+fCDwEkP3UwLxEIeBHZO4=", + "dev": true, + "requires": { + "prelude-ls": "~1.1.2", + "type-check": "~0.3.2" + } + }, + "limiter": { + "version": "1.1.5", + "resolved": "https://registry.npmjs.org/limiter/-/limiter-1.1.5.tgz", + "integrity": "sha512-FWWMIEOxz3GwUI4Ts/IvgVy6LPvoMPgjMdQ185nN6psJyBJ4yOpzqm695/h5umdLJg2vW3GR5iG11MAkR2AzJA==" + }, + "load-json-file": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/load-json-file/-/load-json-file-1.1.0.tgz", + "integrity": "sha1-lWkFcI1YtLq0wiYbBPWfMcmTdMA=", + "requires": { + "graceful-fs": "^4.1.2", + "parse-json": "^2.2.0", + "pify": "^2.0.0", + "pinkie-promise": "^2.0.0", + "strip-bom": "^2.0.0" + } + }, + "locate-path": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-2.0.0.tgz", + "integrity": "sha1-K1aLJl7slExtnA3pw9u7ygNUzY4=", + "dev": true, + "requires": { + "p-locate": "^2.0.0", + "path-exists": "^3.0.0" + }, + "dependencies": { + "path-exists": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-3.0.0.tgz", + "integrity": "sha1-zg6+ql94yxiSXqfYENe1mwEP1RU=", + "dev": true + } + } + }, + "lodash": { + "version": "4.17.21", + "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz", + "integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==" + }, + "lodash.assign": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/lodash.assign/-/lodash.assign-4.2.0.tgz", + "integrity": "sha1-DZnzzNem0mHRm9rrkkUAXShYCOc=" + }, + "lodash.clonedeep": { + "version": "4.5.0", + "resolved": "https://registry.npmjs.org/lodash.clonedeep/-/lodash.clonedeep-4.5.0.tgz", + "integrity": "sha1-4j8/nE+Pvd6HJSnBBxhXoIblzO8=" + }, + "lodash.includes": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/lodash.includes/-/lodash.includes-4.3.0.tgz", + "integrity": "sha1-YLuYqHy5I8aMoeUTJUgzFISfVT8=" + }, + "lodash.isboolean": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/lodash.isboolean/-/lodash.isboolean-3.0.3.tgz", + "integrity": "sha1-bC4XHbKiV82WgC/UOwGyDV9YcPY=" + }, + "lodash.isinteger": { + "version": "4.0.4", + "resolved": "https://registry.npmjs.org/lodash.isinteger/-/lodash.isinteger-4.0.4.tgz", + "integrity": "sha1-YZwK89A/iwTDH1iChAt3sRzWg0M=" + }, + "lodash.isnumber": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/lodash.isnumber/-/lodash.isnumber-3.0.3.tgz", + "integrity": "sha1-POdoEMWSjQM1IwGsKHMX8RwLH/w=" + }, + "lodash.isplainobject": { + "version": "4.0.6", + "resolved": "https://registry.npmjs.org/lodash.isplainobject/-/lodash.isplainobject-4.0.6.tgz", + "integrity": "sha1-fFJqUtibRcRcxpC4gWO+BJf1UMs=" + }, + "lodash.isstring": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/lodash.isstring/-/lodash.isstring-4.0.1.tgz", + "integrity": "sha1-1SfftUVuynzJu5XV2ur4i6VKVFE=" + }, + "lodash.once": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/lodash.once/-/lodash.once-4.1.1.tgz", + "integrity": "sha1-DdOXEhPHxW34gJd9UEyI+0cal6w=" + }, + "logform": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/logform/-/logform-2.2.0.tgz", + "integrity": "sha512-N0qPlqfypFx7UHNn4B3lzS/b0uLqt2hmuoa+PpuXNYgozdJYAyauF5Ky0BWVjrxDlMWiT3qN4zPq3vVAfZy7Yg==", + "requires": { + "colors": "^1.2.1", + "fast-safe-stringify": "^2.0.4", + "fecha": "^4.2.0", + "ms": "^2.1.1", + "triple-beam": "^1.3.0" + } + }, + "loose-envify": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/loose-envify/-/loose-envify-1.4.0.tgz", + "integrity": "sha512-lyuxPGr/Wfhrlem2CL/UcnUc1zcqKAImBDzukY7Y5F/yQiNdko6+fRLevlw1HgMySw7f611UIY408EtxRSoK3Q==", + "dev": true, + "requires": { + "js-tokens": "^3.0.0 || ^4.0.0" + } + }, + "lowercase-keys": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-1.0.1.tgz", + "integrity": "sha512-G2Lj61tXDnVFFOi8VZds+SoQjtQC3dgokKdDG2mTm1tx4m50NUHBOZSBwQQHyy0V12A0JTG4icfZQH+xPyh8VA==", + "dev": true + }, + "lru-cache": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-6.0.0.tgz", + "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", + "requires": { + "yallist": "^4.0.0" + } + }, + "lru-memoizer": { + "version": "2.1.4", + "resolved": "https://registry.npmjs.org/lru-memoizer/-/lru-memoizer-2.1.4.tgz", + "integrity": "sha512-IXAq50s4qwrOBrXJklY+KhgZF+5y98PDaNo0gi/v2KQBFLyWr+JyFvijZXkGKjQj/h9c0OwoE+JZbwUXce76hQ==", + "requires": { + "lodash.clonedeep": "^4.5.0", + "lru-cache": "~4.0.0" + }, + "dependencies": { + "lru-cache": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-4.0.2.tgz", + "integrity": "sha1-HRdnnAac2l0ECZGgnbwsDbN35V4=", + "requires": { + "pseudomap": "^1.0.1", + "yallist": "^2.0.0" + } + }, + "yallist": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/yallist/-/yallist-2.1.2.tgz", + "integrity": "sha1-HBH5IY8HYImkfdUS+TxmmaaoHVI=" + } + } + }, + "make-dir": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-3.1.0.tgz", + "integrity": "sha512-g3FeP20LNwhALb/6Cz6Dd4F2ngze0jz7tbzrD2wAV+o9FeNHe4rL+yK2md0J/fiSf1sa1ADhXqi5+oVwOM/eGw==", + "dev": true, + "requires": { + "semver": "^6.0.0" + }, + "dependencies": { + "semver": { + "version": "6.3.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", + "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", + "dev": true + } + } + }, + "media-typer": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", + "integrity": "sha1-hxDXrwqmJvj/+hzgAWhUUmMlV0g=" + }, + "merge-descriptors": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz", + "integrity": "sha1-sAqqVW3YtEVoFQ7J0blT8/kMu2E=" + }, + "methods": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz", + "integrity": "sha1-VSmk1nZUE07cxSZmVoNbD4Ua/O4=" + }, + "millisecond": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/millisecond/-/millisecond-0.1.2.tgz", + "integrity": "sha1-bMWtOGJByrjniv+WT4cCjuyS2sU=" + }, + "mime": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz", + "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==" + }, + "mime-db": { + "version": "1.48.0", + "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.48.0.tgz", + "integrity": "sha512-FM3QwxV+TnZYQ2aRqhlKBMHxk10lTbMt3bBkMAp54ddrNeVSfcQYOOKuGuy3Ddrm38I04If834fOUSq1yzslJQ==" + }, + "mime-types": { + "version": "2.1.31", + "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.31.tgz", + "integrity": "sha512-XGZnNzm3QvgKxa8dpzyhFTHmpP3l5YNusmne07VUOXxou9CqUqYa/HBy124RqtVh/O2pECas/MOcsDgpilPOPg==", + "requires": { + "mime-db": "1.48.0" + } + }, + "mimic-fn": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/mimic-fn/-/mimic-fn-2.1.0.tgz", + "integrity": "sha512-OqbOk5oEQeAZ8WXWydlu9HJjz9WVdEIvamMCcXmuqUYjTknH/sqsWvhQ3vgwKFRR1HpjvNBKQ37nbJgYzGqGcg==", + "dev": true + }, + "mimic-response": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/mimic-response/-/mimic-response-1.0.1.tgz", + "integrity": "sha512-j5EctnkH7amfV/q5Hgmoal1g2QHFJRraOtmx0JpIqkxhBhI/lJSl1nMpQ45hVarwNETOoWEimndZ4QK0RHxuxQ==", + "dev": true + }, + "minimatch": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz", + "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==", + "requires": { + "brace-expansion": "^1.1.7" + } + }, + "minimist": { + "version": "1.2.5", + "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.5.tgz", + "integrity": "sha512-FM9nNUYrRBAELZQT3xeZQ7fmMOBg6nWNmJKTcgsJeaLstP/UODVpGsr5OhXhhXg6f+qtJ8uiZ+PUxkDWcgIXLw==" + }, + "mkdirp": { + "version": "0.5.5", + "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.5.tgz", + "integrity": "sha512-NKmAlESf6jMGym1++R0Ra7wvhV+wFW63FaSOFPwRahvea0gMUcGUhVeAg/0BC0wiv9ih5NYPB1Wn1UEI1/L+xQ==", + "requires": { + "minimist": "^1.2.5" + } + }, + "moment": { + "version": "2.29.1", + "resolved": "https://registry.npmjs.org/moment/-/moment-2.29.1.tgz", + "integrity": "sha512-kHmoybcPV8Sqy59DwNDY3Jefr64lK/by/da0ViFcuA4DH0vQg5Q6Ze5VimxkfQNSC+Mls/Kx53s7TjP1RhFEDQ==" + }, + "moment-timezone": { + "version": "0.5.33", + "resolved": "https://registry.npmjs.org/moment-timezone/-/moment-timezone-0.5.33.tgz", + "integrity": "sha512-PTc2vcT8K9J5/9rDEPe5czSIKgLoGsH8UNpA4qZTVw0Vd/Uz19geE9abbIOQKaAQFcnQ3v5YEXrbSc5BpshH+w==", + "requires": { + "moment": ">= 2.9.0" + } + }, + "ms": { + "version": "2.1.3", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", + "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==" + }, + "mute-stream": { + "version": "0.0.8", + "resolved": "https://registry.npmjs.org/mute-stream/-/mute-stream-0.0.8.tgz", + "integrity": "sha512-nnbWWOkoWyUsTjKrhgD0dcz22mdkSnpYqbEjIm2nhwhuxlSkpywJmBo8h0ZqJdkp73mb90SssHkN4rsRaBAfAA==", + "dev": true + }, + "mv": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/mv/-/mv-2.1.1.tgz", + "integrity": "sha1-rmzg1vbV4KT32JN5jQPB6pVZtqI=", + "optional": true, + "requires": { + "mkdirp": "~0.5.1", + "ncp": "~2.0.0", + "rimraf": "~2.4.0" + } + }, + "nan": { + "version": "2.14.2", + "resolved": "https://registry.npmjs.org/nan/-/nan-2.14.2.tgz", + "integrity": "sha512-M2ufzIiINKCuDfBSAUr1vWQ+vuVcA9kqx8JJUsbQi6yf1uGRyb7HfpdfUr5qLXf3B/t8dPvcjhKMmlfnP47EzQ==", + "optional": true + }, + "natural-compare": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz", + "integrity": "sha1-Sr6/7tdUHywnrPspvbvRXI1bpPc=", + "dev": true + }, + "ncp": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ncp/-/ncp-2.0.0.tgz", + "integrity": "sha1-GVoh1sRuNh0vsSgbo4uR6d9727M=", + "optional": true + }, + "negotiator": { + "version": "0.6.2", + "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.2.tgz", + "integrity": "sha512-hZXc7K2e+PgeI1eDBe/10Ard4ekbfrrqG8Ep+8Jmf4JID2bNg7NvCPOZN+kfF574pFQI7mum2AUqDidoKqcTOw==" + }, + "neo-async": { + "version": "2.6.2", + "resolved": "https://registry.npmjs.org/neo-async/-/neo-async-2.6.2.tgz", + "integrity": "sha512-Yd3UES5mWCSqR+qNT93S3UoYUkqAZ9lLg8a7g9rimsWmYGK8cVToA4/sF3RrshdyV3sAGMXVUmpMYOw+dLpOuw==" + }, + "nice-try": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/nice-try/-/nice-try-1.0.5.tgz", + "integrity": "sha512-1nh45deeb5olNY7eX82BkPO7SSxR5SSYJiPTrTdFUVYwAl8CKMA5N9PjTYkHiRjisVcxcQ1HXdLhx2qxxJzLNQ==", + "dev": true + }, + "nodemon": { + "version": "2.0.12", + "resolved": "https://registry.npmjs.org/nodemon/-/nodemon-2.0.12.tgz", + "integrity": "sha512-egCTmNZdObdBxUBw6ZNwvZ/xzk24CKRs5K6d+5zbmrMr7rOpPmfPeF6OxM3DDpaRx331CQRFEktn+wrFFfBSOA==", + "dev": true, + "requires": { + "chokidar": "^3.2.2", + "debug": "^3.2.6", + "ignore-by-default": "^1.0.1", + "minimatch": "^3.0.4", + "pstree.remy": "^1.1.7", + "semver": "^5.7.1", + "supports-color": "^5.5.0", + "touch": "^3.1.0", + "undefsafe": "^2.0.3", + "update-notifier": "^4.1.0" + }, + "dependencies": { + "debug": { + "version": "3.2.7", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", + "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", + "dev": true, + "requires": { + "ms": "^2.1.1" + } + } + } + }, + "nopt": { + "version": "1.0.10", + "resolved": "https://registry.npmjs.org/nopt/-/nopt-1.0.10.tgz", + "integrity": "sha1-bd0hvSoxQXuScn3Vhfim83YI6+4=", + "dev": true, + "requires": { + "abbrev": "1" + } + }, + "normalize-package-data": { + "version": "2.5.0", + "resolved": "https://registry.npmjs.org/normalize-package-data/-/normalize-package-data-2.5.0.tgz", + "integrity": "sha512-/5CMN3T0R4XTj4DcGaexo+roZSdSFW/0AOOTROrjxzCG1wrWXEsGbRKevjlIL+ZDE4sZlJr5ED4YW0yqmkK+eA==", + "requires": { + "hosted-git-info": "^2.1.4", + "resolve": "^1.10.0", + "semver": "2 || 3 || 4 || 5", + "validate-npm-package-license": "^3.0.1" + } + }, + "normalize-path": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", + "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==", + "dev": true + }, + "normalize-url": { + "version": "4.5.1", + "resolved": "https://registry.npmjs.org/normalize-url/-/normalize-url-4.5.1.tgz", + "integrity": "sha512-9UZCFRHQdNrfTpGg8+1INIg93B6zE0aXMVFkw1WFwvO4SlZywU6aLg5Of0Ap/PgcbSw4LNxvMWXMeugwMCX0AA==", + "dev": true + }, + "number-is-nan": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/number-is-nan/-/number-is-nan-1.0.1.tgz", + "integrity": "sha1-CXtgK1NCKlIsGvuHkDGDNpQaAR0=" + }, + "oauth-sign": { + "version": "0.9.0", + "resolved": "https://registry.npmjs.org/oauth-sign/-/oauth-sign-0.9.0.tgz", + "integrity": "sha512-fexhUFFPTGV8ybAtSIGbV6gOkSv8UtRbDBnAyLQw4QPKkgNlsH2ByPGtMUqdWkos6YCRmAqViwgZrJc/mRDzZQ==" + }, + "object-assign": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", + "integrity": "sha1-IQmtx5ZYh8/AXLvUQsrIv7s2CGM=" + }, + "object-inspect": { + "version": "1.11.0", + "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.11.0.tgz", + "integrity": "sha512-jp7ikS6Sd3GxQfZJPyH3cjcbJF6GZPClgdV+EFygjFLQ5FmW/dRUnTd9PQ9k0JhoNDabWFbpF1yCdSWCC6gexg==", + "dev": true + }, + "object-keys": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/object-keys/-/object-keys-1.1.1.tgz", + "integrity": "sha512-NuAESUOUMrlIXOfHKzD6bpPu3tYt3xvjNdRIQ+FeT0lNb4K8WR70CaDxhuNguS2XG+GjkyMwOzsN5ZktImfhLA==", + "dev": true + }, + "object.assign": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/object.assign/-/object.assign-4.1.2.tgz", + "integrity": "sha512-ixT2L5THXsApyiUPYKmW+2EHpXXe5Ii3M+f4e+aJFAHao5amFRW6J0OO6c/LU8Be47utCx2GL89hxGB6XSmKuQ==", + "dev": true, + "requires": { + "call-bind": "^1.0.0", + "define-properties": "^1.1.3", + "has-symbols": "^1.0.1", + "object-keys": "^1.1.1" + } + }, + "object.entries": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/object.entries/-/object.entries-1.1.4.tgz", + "integrity": "sha512-h4LWKWE+wKQGhtMjZEBud7uLGhqyLwj8fpHOarZhD2uY3C9cRtk57VQ89ke3moByLXMedqs3XCHzyb4AmA2DjA==", + "dev": true, + "requires": { + "call-bind": "^1.0.2", + "define-properties": "^1.1.3", + "es-abstract": "^1.18.2" + } + }, + "object.fromentries": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/object.fromentries/-/object.fromentries-2.0.4.tgz", + "integrity": "sha512-EsFBshs5RUUpQEY1D4q/m59kMfz4YJvxuNCJcv/jWwOJr34EaVnG11ZrZa0UHB3wnzV1wx8m58T4hQL8IuNXlQ==", + "dev": true, + "requires": { + "call-bind": "^1.0.2", + "define-properties": "^1.1.3", + "es-abstract": "^1.18.0-next.2", + "has": "^1.0.3" + } + }, + "object.values": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/object.values/-/object.values-1.1.4.tgz", + "integrity": "sha512-TnGo7j4XSnKQoK3MfvkzqKCi0nVe/D9I9IjwTNYdb/fxYHpjrluHVOgw0AF6jrRFGMPHdfuidR09tIDiIvnaSg==", + "dev": true, + "requires": { + "call-bind": "^1.0.2", + "define-properties": "^1.1.3", + "es-abstract": "^1.18.2" + } + }, + "on-finished": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.3.0.tgz", + "integrity": "sha1-IPEzZIGwg811M3mSoWlxqi2QaUc=", + "requires": { + "ee-first": "1.1.1" + } + }, + "once": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", + "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=", + "requires": { + "wrappy": "1" + } + }, + "one-time": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/one-time/-/one-time-1.0.0.tgz", + "integrity": "sha512-5DXOiRKwuSEcQ/l0kGCF6Q3jcADFv5tSmRaJck/OqkVFcOzutB134KRSfF0xDrL39MNnqxbHBbUUcjZIhTgb2g==", + "requires": { + "fn.name": "1.x.x" + } + }, + "onetime": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/onetime/-/onetime-5.1.2.tgz", + "integrity": "sha512-kbpaSSGJTWdAY5KPVeMOKXSrPtr8C8C7wodJbcsd51jRnmD+GZu8Y0VoU6Dm5Z4vWr0Ig/1NKuWRKf7j5aaYSg==", + "dev": true, + "requires": { + "mimic-fn": "^2.1.0" + } + }, + "optionator": { + "version": "0.8.3", + "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.8.3.tgz", + "integrity": "sha512-+IW9pACdk3XWmmTXG8m3upGUJst5XRGzxMRjXzAuJ1XnIFNvfhjjIuYkDvysnPQ7qzqVzLt78BCruntqRhWQbA==", + "dev": true, + "requires": { + "deep-is": "~0.1.3", + "fast-levenshtein": "~2.0.6", + "levn": "~0.3.0", + "prelude-ls": "~1.1.2", + "type-check": "~0.3.2", + "word-wrap": "~1.2.3" + } + }, + "os-locale": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/os-locale/-/os-locale-1.4.0.tgz", + "integrity": "sha1-IPnxeuKe00XoveWDsT0gCYA8FNk=", + "requires": { + "lcid": "^1.0.0" + } + }, + "os-tmpdir": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/os-tmpdir/-/os-tmpdir-1.0.2.tgz", + "integrity": "sha1-u+Z0BseaqFxc/sdm/lc0VV36EnQ=", + "dev": true + }, + "p-cancelable": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/p-cancelable/-/p-cancelable-1.1.0.tgz", + "integrity": "sha512-s73XxOZ4zpt1edZYZzvhqFa6uvQc1vwUa0K0BdtIZgQMAJj9IbebH+JkgKZc9h+B05PKHLOTl4ajG1BmNrVZlw==", + "dev": true + }, + "p-limit": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-1.3.0.tgz", + "integrity": "sha512-vvcXsLAJ9Dr5rQOPk7toZQZJApBl2K4J6dANSsEuh6QI41JYcsS/qhTGa9ErIUUgK3WNQoJYvylxvjqmiqEA9Q==", + "dev": true, + "requires": { + "p-try": "^1.0.0" + } + }, + "p-locate": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-2.0.0.tgz", + "integrity": "sha1-IKAQOyIqcMj9OcwuWAaA893l7EM=", + "dev": true, + "requires": { + "p-limit": "^1.1.0" + } + }, + "p-try": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/p-try/-/p-try-1.0.0.tgz", + "integrity": "sha1-y8ec26+P1CKOE/Yh8rGiN8GyB7M=", + "dev": true + }, + "package-json": { + "version": "6.5.0", + "resolved": "https://registry.npmjs.org/package-json/-/package-json-6.5.0.tgz", + "integrity": "sha512-k3bdm2n25tkyxcjSKzB5x8kfVxlMdgsbPr0GkZcwHsLpba6cBjqCt1KlcChKEvxHIcTB1FVMuwoijZ26xex5MQ==", + "dev": true, + "requires": { + "got": "^9.6.0", + "registry-auth-token": "^4.0.0", + "registry-url": "^5.0.0", + "semver": "^6.2.0" + }, + "dependencies": { + "semver": { + "version": "6.3.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", + "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", + "dev": true + } + } + }, + "packet-reader": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/packet-reader/-/packet-reader-1.0.0.tgz", + "integrity": "sha512-HAKu/fG3HpHFO0AA8WE8q2g+gBJaZ9MG7fcKk+IJPLTGAD6Psw4443l+9DGRbOIh3/aXr7Phy0TjilYivJo5XQ==" + }, + "parent-module": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/parent-module/-/parent-module-1.0.1.tgz", + "integrity": "sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==", + "dev": true, + "requires": { + "callsites": "^3.0.0" + } + }, + "parse-json": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-2.2.0.tgz", + "integrity": "sha1-9ID0BDTvgHQfhGkJn43qGPVaTck=", + "requires": { + "error-ex": "^1.2.0" + } + }, + "parseurl": { + "version": "1.3.3", + "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz", + "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==" + }, + "path-exists": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-2.1.0.tgz", + "integrity": "sha1-D+tsZPD8UY2adU3V77YscCJ2H0s=", + "requires": { + "pinkie-promise": "^2.0.0" + } + }, + "path-is-absolute": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", + "integrity": "sha1-F0uSaHNVNP+8es5r9TpanhtcX18=" + }, + "path-key": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/path-key/-/path-key-2.0.1.tgz", + "integrity": "sha1-QRyttXTFoUDTpLGRDUDYDMn0C0A=", + "dev": true + }, + "path-parse": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.7.tgz", + "integrity": "sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==" + }, + "path-to-regexp": { + "version": "0.1.7", + "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz", + "integrity": "sha1-32BBeABfUi8V60SQ5yR6G/qmf4w=" + }, + "path-type": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/path-type/-/path-type-1.1.0.tgz", + "integrity": "sha1-WcRPfuSR2nBNpBXaWkBwuk+P5EE=", + "requires": { + "graceful-fs": "^4.1.2", + "pify": "^2.0.0", + "pinkie-promise": "^2.0.0" + } + }, + "performance-now": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/performance-now/-/performance-now-2.1.0.tgz", + "integrity": "sha1-Ywn04OX6kT7BxpMHrjZLSzd8nns=" + }, + "pg": { + "version": "8.6.0", + "resolved": "https://registry.npmjs.org/pg/-/pg-8.6.0.tgz", + "integrity": "sha512-qNS9u61lqljTDFvmk/N66EeGq3n6Ujzj0FFyNMGQr6XuEv4tgNTXvJQTfJdcvGit5p5/DWPu+wj920hAJFI+QQ==", + "requires": { + "buffer-writer": "2.0.0", + "packet-reader": "1.0.0", + "pg-connection-string": "^2.5.0", + "pg-pool": "^3.3.0", + "pg-protocol": "^1.5.0", + "pg-types": "^2.1.0", + "pgpass": "1.x" + } + }, + "pg-connection-string": { + "version": "2.5.0", + "resolved": "https://registry.npmjs.org/pg-connection-string/-/pg-connection-string-2.5.0.tgz", + "integrity": "sha512-r5o/V/ORTA6TmUnyWZR9nCj1klXCO2CEKNRlVuJptZe85QuhFayC7WeMic7ndayT5IRIR0S0xFxFi2ousartlQ==" + }, + "pg-int8": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/pg-int8/-/pg-int8-1.0.1.tgz", + "integrity": "sha512-WCtabS6t3c8SkpDBUlb1kjOs7l66xsGdKpIPZsg4wR+B3+u9UAum2odSsF9tnvxg80h4ZxLWMy4pRjOsFIqQpw==" + }, + "pg-pool": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/pg-pool/-/pg-pool-3.3.0.tgz", + "integrity": "sha512-0O5huCql8/D6PIRFAlmccjphLYWC+JIzvUhSzXSpGaf+tjTZc4nn+Lr7mLXBbFJfvwbP0ywDv73EiaBsxn7zdg==" + }, + "pg-protocol": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/pg-protocol/-/pg-protocol-1.5.0.tgz", + "integrity": "sha512-muRttij7H8TqRNu/DxrAJQITO4Ac7RmX3Klyr/9mJEOBeIpgnF8f9jAfRz5d3XwQZl5qBjF9gLsUtMPJE0vezQ==" + }, + "pg-types": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/pg-types/-/pg-types-2.2.0.tgz", + "integrity": "sha512-qTAAlrEsl8s4OiEQY69wDvcMIdQN6wdz5ojQiOy6YRMuynxenON0O5oCpJI6lshc6scgAY8qvJ2On/p+CXY0GA==", + "requires": { + "pg-int8": "1.0.1", + "postgres-array": "~2.0.0", + "postgres-bytea": "~1.0.0", + "postgres-date": "~1.0.4", + "postgres-interval": "^1.1.0" + } + }, + "pgpass": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/pgpass/-/pgpass-1.0.4.tgz", + "integrity": "sha512-YmuA56alyBq7M59vxVBfPJrGSozru8QAdoNlWuW3cz8l+UX3cWge0vTvjKhsSHSJpo3Bom8/Mm6hf0TR5GY0+w==", + "requires": { + "split2": "^3.1.1" + } + }, + "pgtools": { + "version": "0.3.2", + "resolved": "https://registry.npmjs.org/pgtools/-/pgtools-0.3.2.tgz", + "integrity": "sha512-o9iI8CrJohpjt3hgoJuEC18oYrt/iLsc3BYtW6kP/0T7EyQ9T/WlnuzyKcC2GtfutREfXCmwaUcbqPrLw8sjng==", + "requires": { + "bluebird": "^3.3.5", + "pg": "^8.4.0", + "pg-connection-string": "^2.4.0", + "yargs": "^5.0.0" + } + }, + "picomatch": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.0.tgz", + "integrity": "sha512-lY1Q/PiJGC2zOv/z391WOTD+Z02bCgsFfvxoXXf6h7kv9o+WmsmzYqrAwY63sNgOxE4xEdq0WyUnXfKeBrSvYw==", + "dev": true + }, + "pify": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/pify/-/pify-2.3.0.tgz", + "integrity": "sha1-7RQaasBDqEnqWISY59yosVMw6Qw=" + }, + "pinkie": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/pinkie/-/pinkie-2.0.4.tgz", + "integrity": "sha1-clVrgM+g1IqXToDnckjoDtT3+HA=" + }, + "pinkie-promise": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/pinkie-promise/-/pinkie-promise-2.0.1.tgz", + "integrity": "sha1-ITXW36ejWMBprJsXh3YogihFD/o=", + "requires": { + "pinkie": "^2.0.0" + } + }, + "pkg-conf": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/pkg-conf/-/pkg-conf-3.1.0.tgz", + "integrity": "sha512-m0OTbR/5VPNPqO1ph6Fqbj7Hv6QU7gR/tQW40ZqrL1rjgCU85W6C1bJn0BItuJqnR98PWzw7Z8hHeChD1WrgdQ==", + "dev": true, + "requires": { + "find-up": "^3.0.0", + "load-json-file": "^5.2.0" + }, + "dependencies": { + "find-up": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-3.0.0.tgz", + "integrity": "sha512-1yD6RmLI1XBfxugvORwlck6f75tYL+iR0jqwsOrOxMZyGYqUuDhJ0l4AXdO1iX/FTs9cBAMEk1gWSEx1kSbylg==", + "dev": true, + "requires": { + "locate-path": "^3.0.0" + } + }, + "load-json-file": { + "version": "5.3.0", + "resolved": "https://registry.npmjs.org/load-json-file/-/load-json-file-5.3.0.tgz", + "integrity": "sha512-cJGP40Jc/VXUsp8/OrnyKyTZ1y6v/dphm3bioS+RrKXjK2BB6wHUd6JptZEFDGgGahMT+InnZO5i1Ei9mpC8Bw==", + "dev": true, + "requires": { + "graceful-fs": "^4.1.15", + "parse-json": "^4.0.0", + "pify": "^4.0.1", + "strip-bom": "^3.0.0", + "type-fest": "^0.3.0" + } + }, + "locate-path": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-3.0.0.tgz", + "integrity": "sha512-7AO748wWnIhNqAuaty2ZWHkQHRSNfPVIsPIfwEOWO22AmaoVrWavlOcMR5nzTLNYvp36X220/maaRsrec1G65A==", + "dev": true, + "requires": { + "p-locate": "^3.0.0", + "path-exists": "^3.0.0" + } + }, + "p-limit": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-2.3.0.tgz", + "integrity": "sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w==", + "dev": true, + "requires": { + "p-try": "^2.0.0" + } + }, + "p-locate": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-3.0.0.tgz", + "integrity": "sha512-x+12w/To+4GFfgJhBEpiDcLozRJGegY+Ei7/z0tSLkMmxGZNybVMSfWj9aJn8Z5Fc7dBUNJOOVgPv2H7IwulSQ==", + "dev": true, + "requires": { + "p-limit": "^2.0.0" + } + }, + "p-try": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/p-try/-/p-try-2.2.0.tgz", + "integrity": "sha512-R4nPAVTAU0B9D35/Gk3uJf/7XYbQcyohSKdvAxIRSNghFl4e71hVoGnBNQz9cWaXxO2I10KTC+3jMdvvoKw6dQ==", + "dev": true + }, + "parse-json": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-4.0.0.tgz", + "integrity": "sha1-vjX1Qlvh9/bHRxhPmKeIy5lHfuA=", + "dev": true, + "requires": { + "error-ex": "^1.3.1", + "json-parse-better-errors": "^1.0.1" + } + }, + "path-exists": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-3.0.0.tgz", + "integrity": "sha1-zg6+ql94yxiSXqfYENe1mwEP1RU=", + "dev": true + }, + "pify": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/pify/-/pify-4.0.1.tgz", + "integrity": "sha512-uB80kBFb/tfd68bVleG9T5GGsGPjJrLAUpR5PZIrhBnIaRTQRjqdJSsIKkOP6OAIFbj7GOrcudc5pNjZ+geV2g==", + "dev": true + }, + "strip-bom": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-3.0.0.tgz", + "integrity": "sha1-IzTBjpx1n3vdVv3vfprj1YjmjtM=", + "dev": true + }, + "type-fest": { + "version": "0.3.1", + "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.3.1.tgz", + "integrity": "sha512-cUGJnCdr4STbePCgqNFbpVNCepa+kAVohJs1sLhxzdH+gnEoOd8VhbYa7pD3zZYGiURWM2xzEII3fQcRizDkYQ==", + "dev": true + } + } + }, + "pkg-config": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/pkg-config/-/pkg-config-1.1.1.tgz", + "integrity": "sha1-VX7yLXPaPIg3EHdmxS6tq94pj+Q=", + "dev": true, + "requires": { + "debug-log": "^1.0.0", + "find-root": "^1.0.0", + "xtend": "^4.0.1" + } + }, + "pkg-dir": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/pkg-dir/-/pkg-dir-2.0.0.tgz", + "integrity": "sha1-9tXREJ4Z1j7fQo4L1X4Sd3YVM0s=", + "dev": true, + "requires": { + "find-up": "^2.1.0" + }, + "dependencies": { + "find-up": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-2.1.0.tgz", + "integrity": "sha1-RdG35QbHF93UgndaK3eSCjwMV6c=", + "dev": true, + "requires": { + "locate-path": "^2.0.0" + } + } + } + }, + "postgres-array": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/postgres-array/-/postgres-array-2.0.0.tgz", + "integrity": "sha512-VpZrUqU5A69eQyW2c5CA1jtLecCsN2U/bD6VilrFDWq5+5UIEVO7nazS3TEcHf1zuPYO/sqGvUvW62g86RXZuA==" + }, + "postgres-bytea": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/postgres-bytea/-/postgres-bytea-1.0.0.tgz", + "integrity": "sha1-AntTPAqokOJtFy1Hz5zOzFIazTU=" + }, + "postgres-date": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/postgres-date/-/postgres-date-1.0.7.tgz", + "integrity": "sha512-suDmjLVQg78nMK2UZ454hAG+OAW+HQPZ6n++TNDUX+L0+uUlLywnoxJKDou51Zm+zTCjrCl0Nq6J9C5hP9vK/Q==" + }, + "postgres-interval": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/postgres-interval/-/postgres-interval-1.2.0.tgz", + "integrity": "sha512-9ZhXKM/rw350N1ovuWHbGxnGh/SNJ4cnxHiM0rxE4VN41wsg8P8zWn9hv/buK00RP4WvlOyr/RBDiptyxVbkZQ==", + "requires": { + "xtend": "^4.0.0" + } + }, + "precond": { + "version": "0.2.3", + "resolved": "https://registry.npmjs.org/precond/-/precond-0.2.3.tgz", + "integrity": "sha1-qpWRvKokkj8eD0hJ0kD0fvwQdaw=" + }, + "prelude-ls": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.1.2.tgz", + "integrity": "sha1-IZMqVJ9eUv/ZqCf1cOBL5iqX2lQ=", + "dev": true + }, + "prepend-http": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/prepend-http/-/prepend-http-2.0.0.tgz", + "integrity": "sha1-6SQ0v6XqjBn0HN/UAddBo8gZ2Jc=", + "dev": true + }, + "process-nextick-args": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-2.0.1.tgz", + "integrity": "sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag==" + }, + "progress": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/progress/-/progress-2.0.3.tgz", + "integrity": "sha512-7PiHtLll5LdnKIMw100I+8xJXR5gW2QwWYkT6iJva0bXitZKa/XMrSbdmg3r2Xnaidz9Qumd0VPaMrZlF9V9sA==", + "dev": true + }, + "prop-types": { + "version": "15.7.2", + "resolved": "https://registry.npmjs.org/prop-types/-/prop-types-15.7.2.tgz", + "integrity": "sha512-8QQikdH7//R2vurIJSutZ1smHYTcLpRWEOlHnzcWHmBYrOGUysKwSsrC89BCiFj3CbrfJ/nXFdJepOVrY1GCHQ==", + "dev": true, + "requires": { + "loose-envify": "^1.4.0", + "object-assign": "^4.1.1", + "react-is": "^16.8.1" + } + }, + "proxy-addr": { + "version": "2.0.7", + "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.7.tgz", + "integrity": "sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg==", + "requires": { + "forwarded": "0.2.0", + "ipaddr.js": "1.9.1" + } + }, + "proxy-from-env": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/proxy-from-env/-/proxy-from-env-1.1.0.tgz", + "integrity": "sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg==" + }, + "pseudomap": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/pseudomap/-/pseudomap-1.0.2.tgz", + "integrity": "sha1-8FKijacOYYkX7wqKw0wa5aaChrM=" + }, + "psl": { + "version": "1.8.0", + "resolved": "https://registry.npmjs.org/psl/-/psl-1.8.0.tgz", + "integrity": "sha512-RIdOzyoavK+hA18OGGWDqUTsCLhtA7IcZ/6NCs4fFJaHBDab+pDDmDIByWFRQJq2Cd7r1OoQxBGKOaztq+hjIQ==" + }, + "pstree.remy": { + "version": "1.1.8", + "resolved": "https://registry.npmjs.org/pstree.remy/-/pstree.remy-1.1.8.tgz", + "integrity": "sha512-77DZwxQmxKnu3aR542U+X8FypNzbfJ+C5XQDk3uWjWxn6151aIMGthWYRXTqT1E5oJvg+ljaa2OJi+VfvCOQ8w==", + "dev": true + }, + "pump": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/pump/-/pump-3.0.0.tgz", + "integrity": "sha512-LwZy+p3SFs1Pytd/jYct4wpv49HiYCqd9Rlc5ZVdk0V+8Yzv6jR5Blk3TRmPL1ft69TxP0IMZGJ+WPFU2BFhww==", + "dev": true, + "requires": { + "end-of-stream": "^1.1.0", + "once": "^1.3.1" + } + }, + "punycode": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.1.1.tgz", + "integrity": "sha512-XRsRjdf+j5ml+y/6GKHPZbrF/8p2Yga0JPtdqTIY2Xe5ohJPD9saDJJLPvp9+NSBprVvevdXZybnj2cv8OEd0A==" + }, + "pupa": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/pupa/-/pupa-2.1.1.tgz", + "integrity": "sha512-l1jNAspIBSFqbT+y+5FosojNpVpF94nlI+wDUpqP9enwOTfHx9f0gh5nB96vl+6yTpsJsypeNrwfzPrKuHB41A==", + "dev": true, + "requires": { + "escape-goat": "^2.0.0" + } + }, + "qs": { + "version": "6.7.0", + "resolved": "https://registry.npmjs.org/qs/-/qs-6.7.0.tgz", + "integrity": "sha512-VCdBRNFTX1fyE7Nb6FYoURo/SPe62QCaAyzJvUjwRaIsc+NePBEniHlvxFmmX56+HZphIGtV0XeCirBtpDrTyQ==" + }, + "queue-microtask": { + "version": "1.2.3", + "resolved": "https://registry.npmjs.org/queue-microtask/-/queue-microtask-1.2.3.tgz", + "integrity": "sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==", + "dev": true + }, + "r7insight_node": { + "version": "1.8.4", + "resolved": "https://registry.npmjs.org/r7insight_node/-/r7insight_node-1.8.4.tgz", + "integrity": "sha512-6cQrzLkaOxdv/SRFXWRJjgFr8a3nXUOT/4IMFuBv+mWzBnu5DJl+HzONAsWYvclrlZnvfa54PaIPqPuPRSlbrQ==", + "requires": { + "babel-runtime": "6.6.1", + "codependency": "0.1.4", + "json-stringify-safe": "5.0.1", + "lodash": "4.17.15", + "reconnect-core": "1.3.0", + "semver": "5.1.0" + }, + "dependencies": { + "lodash": { + "version": "4.17.15", + "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.15.tgz", + "integrity": "sha512-8xOcRHvCjnocdS5cpwXQXVzmmh5e5+saE2QGoeQmbKmRS6J3VQppPOIt0MnmE+4xlZoumy0GPG0D0MVIQbNA1A==" + }, + "semver": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-5.1.0.tgz", + "integrity": "sha1-hfLPhVBGXE3wAM99hvawVBBqueU=" + } + } + }, + "range-parser": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz", + "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==" + }, + "raw-body": { + "version": "2.4.0", + "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.4.0.tgz", + "integrity": "sha512-4Oz8DUIwdvoa5qMJelxipzi/iJIi40O5cGV1wNYp5hvZP8ZN0T+jiNkL0QepXs+EsQ9XJ8ipEDoiH70ySUJP3Q==", + "requires": { + "bytes": "3.1.0", + "http-errors": "1.7.2", + "iconv-lite": "0.4.24", + "unpipe": "1.0.0" + } + }, + "rc": { + "version": "1.2.8", + "resolved": "https://registry.npmjs.org/rc/-/rc-1.2.8.tgz", + "integrity": "sha512-y3bGgqKj3QBdxLbLkomlohkvsA8gdAiUQlSBJnBhfn+BPxg4bc62d8TcBW15wavDfgexCgccckhcZvywyQYPOw==", + "dev": true, + "requires": { + "deep-extend": "^0.6.0", + "ini": "~1.3.0", + "minimist": "^1.2.0", + "strip-json-comments": "~2.0.1" + }, + "dependencies": { + "strip-json-comments": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-2.0.1.tgz", + "integrity": "sha1-PFMZQukIwml8DsNEhYwobHygpgo=", + "dev": true + } + } + }, + "react-is": { + "version": "16.13.1", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-16.13.1.tgz", + "integrity": "sha512-24e6ynE2H+OKt4kqsOvNd8kBpV65zoxbA4BVsEOB3ARVWQki/DHzaUoC5KuON/BiccDaCCTZBuOcfZs70kR8bQ==", + "dev": true + }, + "read-pkg": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/read-pkg/-/read-pkg-1.1.0.tgz", + "integrity": "sha1-9f+qXs0pyzHAR0vKfXVra7KePyg=", + "requires": { + "load-json-file": "^1.0.0", + "normalize-package-data": "^2.3.2", + "path-type": "^1.0.0" + } + }, + "read-pkg-up": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/read-pkg-up/-/read-pkg-up-1.0.1.tgz", + "integrity": "sha1-nWPBMnbAZZGNV/ACpX9AobZD+wI=", + "requires": { + "find-up": "^1.0.0", + "read-pkg": "^1.0.0" + } + }, + "readable-stream": { + "version": "3.6.0", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.0.tgz", + "integrity": "sha512-BViHy7LKeTz4oNnkcLJ+lVSL6vpiFeX6/d3oSH8zCW7UxP2onchk+vTGB143xuFjHS3deTgkKoXXymXqymiIdA==", + "requires": { + "inherits": "^2.0.3", + "string_decoder": "^1.1.1", + "util-deprecate": "^1.0.1" + } + }, + "readdirp": { + "version": "3.6.0", + "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.6.0.tgz", + "integrity": "sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==", + "dev": true, + "requires": { + "picomatch": "^2.2.1" + } + }, + "reconnect-core": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/reconnect-core/-/reconnect-core-1.3.0.tgz", + "integrity": "sha1-+65SkZp4d9hE4yRtAaLyZwHIM8g=", + "requires": { + "backoff": "~2.5.0" + } + }, + "regexpp": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/regexpp/-/regexpp-2.0.1.tgz", + "integrity": "sha512-lv0M6+TkDVniA3aD1Eg0DVpfU/booSu7Eev3TDO/mZKHBfVjgCGTV4t4buppESEYDtkArYFOxTJWv6S5C+iaNw==", + "dev": true + }, + "registry-auth-token": { + "version": "4.2.1", + "resolved": "https://registry.npmjs.org/registry-auth-token/-/registry-auth-token-4.2.1.tgz", + "integrity": "sha512-6gkSb4U6aWJB4SF2ZvLb76yCBjcvufXBqvvEx1HbmKPkutswjW1xNVRY0+daljIYRbogN7O0etYSlbiaEQyMyw==", + "dev": true, + "requires": { + "rc": "^1.2.8" + } + }, + "registry-url": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/registry-url/-/registry-url-5.1.0.tgz", + "integrity": "sha512-8acYXXTI0AkQv6RAOjE3vOaIXZkT9wo4LOFbBKYQEEnnMNBpKqdUrI6S4NT0KPIo/WVvJ5tE/X5LF/TQUf0ekw==", + "dev": true, + "requires": { + "rc": "^1.2.8" + } + }, + "request": { + "version": "2.88.2", + "resolved": "https://registry.npmjs.org/request/-/request-2.88.2.tgz", + "integrity": "sha512-MsvtOrfG9ZcrOwAW+Qi+F6HbD0CWXEh9ou77uOb7FM2WPhwT7smM833PzanhJLsgXjN89Ir6V2PczXNnMpwKhw==", + "requires": { + "aws-sign2": "~0.7.0", + "aws4": "^1.8.0", + "caseless": "~0.12.0", + "combined-stream": "~1.0.6", + "extend": "~3.0.2", + "forever-agent": "~0.6.1", + "form-data": "~2.3.2", + "har-validator": "~5.1.3", + "http-signature": "~1.2.0", + "is-typedarray": "~1.0.0", + "isstream": "~0.1.2", + "json-stringify-safe": "~5.0.1", + "mime-types": "~2.1.19", + "oauth-sign": "~0.9.0", + "performance-now": "^2.1.0", + "qs": "~6.5.2", + "safe-buffer": "^5.1.2", + "tough-cookie": "~2.5.0", + "tunnel-agent": "^0.6.0", + "uuid": "^3.3.2" + }, + "dependencies": { + "qs": { + "version": "6.5.2", + "resolved": "https://registry.npmjs.org/qs/-/qs-6.5.2.tgz", + "integrity": "sha512-N5ZAX4/LxJmF+7wN74pUD6qAh9/wnvdQcjq9TZjevvXzSUo7bfmw91saqMjzGS2xq91/odN2dW/WOl7qQHNDGA==" + }, + "uuid": { + "version": "3.4.0", + "resolved": "https://registry.npmjs.org/uuid/-/uuid-3.4.0.tgz", + "integrity": "sha512-HjSDRw6gZE5JMggctHBcjVak08+KEVhSIiDzFnT9S9aegmp85S/bReBVTb4QTFaRNptJ9kuYaNhnbNEOkbKb/A==" + } + } + }, + "require-directory": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz", + "integrity": "sha1-jGStX9MNqxyXbiNE/+f3kqam30I=" + }, + "require-main-filename": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/require-main-filename/-/require-main-filename-1.0.1.tgz", + "integrity": "sha1-l/cXtp1IeE9fUmpsWqj/3aBVpNE=" + }, + "resolve": { + "version": "1.20.0", + "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.20.0.tgz", + "integrity": "sha512-wENBPt4ySzg4ybFQW2TT1zMQucPK95HSh/nq2CFTZVOGut2+pQvSsgtda4d26YrYcr067wjbmzOG8byDPBX63A==", + "requires": { + "is-core-module": "^2.2.0", + "path-parse": "^1.0.6" + } + }, + "resolve-from": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-4.0.0.tgz", + "integrity": "sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==", + "dev": true + }, + "responselike": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/responselike/-/responselike-1.0.2.tgz", + "integrity": "sha1-kYcg7ztjHFZCvgaPFa3lpG9Loec=", + "dev": true, + "requires": { + "lowercase-keys": "^1.0.0" + } + }, + "restore-cursor": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/restore-cursor/-/restore-cursor-3.1.0.tgz", + "integrity": "sha512-l+sSefzHpj5qimhFSE5a8nufZYAM3sBSVMAPtYkmC+4EH2anSGaEMXSD0izRQbu9nfyQ9y5JrVmp7E8oZrUjvA==", + "dev": true, + "requires": { + "onetime": "^5.1.0", + "signal-exit": "^3.0.2" + } + }, + "retry-as-promised": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/retry-as-promised/-/retry-as-promised-3.2.0.tgz", + "integrity": "sha512-CybGs60B7oYU/qSQ6kuaFmRd9sTZ6oXSc0toqePvV74Ac6/IFZSI1ReFQmtCN+uvW1Mtqdwpvt/LGOiCBAY2Mg==", + "requires": { + "any-promise": "^1.3.0" + } + }, + "rimraf": { + "version": "2.4.5", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.4.5.tgz", + "integrity": "sha1-7nEM5dk6j9uFb7Xqj/Di11k0sto=", + "optional": true, + "requires": { + "glob": "^6.0.1" + } + }, + "run-async": { + "version": "2.4.1", + "resolved": "https://registry.npmjs.org/run-async/-/run-async-2.4.1.tgz", + "integrity": "sha512-tvVnVv01b8c1RrA6Ep7JkStj85Guv/YrMcwqYQnwjsAS2cTmmPGBBjAjpCW7RrSodNSoE2/qg9O4bceNvUuDgQ==", + "dev": true + }, + "run-parallel": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/run-parallel/-/run-parallel-1.2.0.tgz", + "integrity": "sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==", + "dev": true, + "requires": { + "queue-microtask": "^1.2.2" + } + }, + "rxjs": { + "version": "6.6.7", + "resolved": "https://registry.npmjs.org/rxjs/-/rxjs-6.6.7.tgz", + "integrity": "sha512-hTdwr+7yYNIT5n4AMYp85KA6yw2Va0FLa3Rguvbpa4W3I5xynaBZo41cM3XM+4Q6fRMj3sBYIR1VAmZMXYJvRQ==", + "dev": true, + "requires": { + "tslib": "^1.9.0" + } + }, + "safe-buffer": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", + "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==" + }, + "safe-json-stringify": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/safe-json-stringify/-/safe-json-stringify-1.2.0.tgz", + "integrity": "sha512-gH8eh2nZudPQO6TytOvbxnuhYBOvDBBLW52tz5q6X58lJcd/tkmqFR+5Z9adS8aJtURSXWThWy/xJtJwixErvg==", + "optional": true + }, + "safer-buffer": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", + "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==" + }, + "secure-json-parse": { + "version": "2.4.0", + "resolved": "https://registry.npmjs.org/secure-json-parse/-/secure-json-parse-2.4.0.tgz", + "integrity": "sha512-Q5Z/97nbON5t/L/sH6mY2EacfjVGwrCcSi5D3btRO2GZ8pf1K1UN7Z9H5J57hjVU2Qzxr1xO+FmBhOvEkzCMmg==" + }, + "semver": { + "version": "5.7.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", + "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==" + }, + "semver-diff": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/semver-diff/-/semver-diff-3.1.1.tgz", + "integrity": "sha512-GX0Ix/CJcHyB8c4ykpHGIAvLyOwOobtM/8d+TQkAd81/bEjgPHrfba41Vpesr7jX/t8Uh+R3EX9eAS5be+jQYg==", + "dev": true, + "requires": { + "semver": "^6.3.0" + }, + "dependencies": { + "semver": { + "version": "6.3.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", + "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", + "dev": true + } + } + }, + "send": { + "version": "0.17.1", + "resolved": "https://registry.npmjs.org/send/-/send-0.17.1.tgz", + "integrity": "sha512-BsVKsiGcQMFwT8UxypobUKyv7irCNRHk1T0G680vk88yf6LBByGcZJOTJCrTP2xVN6yI+XjPJcNuE3V4fT9sAg==", + "requires": { + "debug": "2.6.9", + "depd": "~1.1.2", + "destroy": "~1.0.4", + "encodeurl": "~1.0.2", + "escape-html": "~1.0.3", + "etag": "~1.8.1", + "fresh": "0.5.2", + "http-errors": "~1.7.2", + "mime": "1.6.0", + "ms": "2.1.1", + "on-finished": "~2.3.0", + "range-parser": "~1.2.1", + "statuses": "~1.5.0" + }, + "dependencies": { + "debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "requires": { + "ms": "2.0.0" + }, + "dependencies": { + "ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" + } + } + }, + "ms": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.1.tgz", + "integrity": "sha512-tgp+dl5cGk28utYktBsrFqA7HKgrhgPsg6Z/EfhWI4gl1Hwq8B/GmY/0oXZ6nF8hDVesS/FpnYaD/kOWhYQvyg==" + } + } + }, + "sequelize": { + "version": "6.6.5", + "resolved": "https://registry.npmjs.org/sequelize/-/sequelize-6.6.5.tgz", + "integrity": "sha512-QyRrJrDRiwuiILqTMHUA1yWOPIL12KlfmgZ3hnzQwbMvp2vJ6fzu9bYJQB+qPMosck4mBUggY4Cjoc6Et8FBIQ==", + "requires": { + "debug": "^4.1.1", + "dottie": "^2.0.0", + "inflection": "1.13.1", + "lodash": "^4.17.20", + "moment": "^2.26.0", + "moment-timezone": "^0.5.31", + "retry-as-promised": "^3.2.0", + "semver": "^7.3.2", + "sequelize-pool": "^6.0.0", + "toposort-class": "^1.0.1", + "uuid": "^8.1.0", + "validator": "^13.6.0", + "wkx": "^0.5.0" + }, + "dependencies": { + "semver": { + "version": "7.3.5", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.5.tgz", + "integrity": "sha512-PoeGJYh8HK4BTO/a9Tf6ZG3veo/A7ZVsYrSA6J8ny9nb3B1VrpkuN+z9OE5wfE5p6H4LchYZsegiQgbJD94ZFQ==", + "requires": { + "lru-cache": "^6.0.0" + } + } + } + }, + "sequelize-pool": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/sequelize-pool/-/sequelize-pool-6.1.0.tgz", + "integrity": "sha512-4YwEw3ZgK/tY/so+GfnSgXkdwIJJ1I32uZJztIEgZeAO6HMgj64OzySbWLgxj+tXhZCJnzRfkY9gINw8Ft8ZMg==" + }, + "serve-static": { + "version": "1.14.1", + "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.14.1.tgz", + "integrity": "sha512-JMrvUwE54emCYWlTI+hGrGv5I8dEwmco/00EvkzIIsR7MqrHonbD9pO2MOfFnpFntl7ecpZs+3mW+XbQZu9QCg==", + "requires": { + "encodeurl": "~1.0.2", + "escape-html": "~1.0.3", + "parseurl": "~1.3.3", + "send": "0.17.1" + } + }, + "set-blocking": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/set-blocking/-/set-blocking-2.0.0.tgz", + "integrity": "sha1-BF+XgtARrppoA93TgrJDkrPYkPc=" + }, + "setprototypeof": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.1.tgz", + "integrity": "sha512-JvdAWfbXeIGaZ9cILp38HntZSFSo3mWg6xGcJJsd+d4aRMOqauag1C63dJfDw7OaMYwEbHMOxEZ1lqVRYP2OAw==" + }, + "shebang-command": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-1.2.0.tgz", + "integrity": "sha1-RKrGW2lbAzmJaMOfNj/uXer98eo=", + "dev": true, + "requires": { + "shebang-regex": "^1.0.0" + } + }, + "shebang-regex": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-1.0.0.tgz", + "integrity": "sha1-2kL0l0DAtC2yypcoVxyxkMmO/qM=", + "dev": true + }, + "signal-exit": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.3.tgz", + "integrity": "sha512-VUJ49FC8U1OxwZLxIbTTrDvLnf/6TDgxZcK8wxR8zs13xpx7xbG60ndBlhNrFi2EMuFRoeDoJO7wthSLq42EjA==", + "dev": true + }, + "simple-swizzle": { + "version": "0.2.2", + "resolved": "https://registry.npmjs.org/simple-swizzle/-/simple-swizzle-0.2.2.tgz", + "integrity": "sha1-pNprY1/8zMoz9w0Xy5JZLeleVXo=", + "requires": { + "is-arrayish": "^0.3.1" + }, + "dependencies": { + "is-arrayish": { + "version": "0.3.2", + "resolved": "https://registry.npmjs.org/is-arrayish/-/is-arrayish-0.3.2.tgz", + "integrity": "sha512-eVRqCvVlZbuw3GrM63ovNSNAeA1K16kaR/LRY/92w0zxQ5/1YzwblUX652i4Xs9RwAGjW9d9y6X88t8OaAJfWQ==" + } + } + }, + "slice-ansi": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/slice-ansi/-/slice-ansi-2.1.0.tgz", + "integrity": "sha512-Qu+VC3EwYLldKa1fCxuuvULvSJOKEgk9pi8dZeCVK7TqBfUNTH4sFkk4joj8afVSfAYgJoSOetjx9QWOJ5mYoQ==", + "dev": true, + "requires": { + "ansi-styles": "^3.2.0", + "astral-regex": "^1.0.0", + "is-fullwidth-code-point": "^2.0.0" + }, + "dependencies": { + "is-fullwidth-code-point": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", + "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=", + "dev": true + } + } + }, + "source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==" + }, + "spdx-correct": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/spdx-correct/-/spdx-correct-3.1.1.tgz", + "integrity": "sha512-cOYcUWwhCuHCXi49RhFRCyJEK3iPj1Ziz9DpViV3tbZOwXD49QzIN3MpOLJNxh2qwq2lJJZaKMVw9qNi4jTC0w==", + "requires": { + "spdx-expression-parse": "^3.0.0", + "spdx-license-ids": "^3.0.0" + } + }, + "spdx-exceptions": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/spdx-exceptions/-/spdx-exceptions-2.3.0.tgz", + "integrity": "sha512-/tTrYOC7PPI1nUAgx34hUpqXuyJG+DTHJTnIULG4rDygi4xu/tfgmq1e1cIRwRzwZgo4NLySi+ricLkZkw4i5A==" + }, + "spdx-expression-parse": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/spdx-expression-parse/-/spdx-expression-parse-3.0.1.tgz", + "integrity": "sha512-cbqHunsQWnJNE6KhVSMsMeH5H/L9EpymbzqTQ3uLwNCLZ1Q481oWaofqH7nO6V07xlXwY6PhQdQ2IedWx/ZK4Q==", + "requires": { + "spdx-exceptions": "^2.1.0", + "spdx-license-ids": "^3.0.0" + } + }, + "spdx-license-ids": { + "version": "3.0.9", + "resolved": "https://registry.npmjs.org/spdx-license-ids/-/spdx-license-ids-3.0.9.tgz", + "integrity": "sha512-Ki212dKK4ogX+xDo4CtOZBVIwhsKBEfsEEcwmJfLQzirgc2jIWdzg40Unxz/HzEUqM1WFzVlQSMF9kZZ2HboLQ==" + }, + "split2": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/split2/-/split2-3.2.2.tgz", + "integrity": "sha512-9NThjpgZnifTkJpzTZ7Eue85S49QwpNhZTq6GRJwObb6jnLFNGB7Qm73V5HewTROPyxD0C29xqmaI68bQtV+hg==", + "requires": { + "readable-stream": "^3.0.0" + } + }, + "sprintf-js": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.0.3.tgz", + "integrity": "sha1-BOaSb2YolTVPPdAVIDYzuFcpfiw=" + }, + "sshpk": { + "version": "1.16.1", + "resolved": "https://registry.npmjs.org/sshpk/-/sshpk-1.16.1.tgz", + "integrity": "sha512-HXXqVUq7+pcKeLqqZj6mHFUMvXtOJt1uoUx09pFW6011inTMxqI8BA8PM95myrIyyKwdnzjdFjLiE6KBPVtJIg==", + "requires": { + "asn1": "~0.2.3", + "assert-plus": "^1.0.0", + "bcrypt-pbkdf": "^1.0.0", + "dashdash": "^1.12.0", + "ecc-jsbn": "~0.1.1", + "getpass": "^0.1.1", + "jsbn": "~0.1.0", + "safer-buffer": "^2.0.2", + "tweetnacl": "~0.14.0" + } + }, + "stack-trace": { + "version": "0.0.10", + "resolved": "https://registry.npmjs.org/stack-trace/-/stack-trace-0.0.10.tgz", + "integrity": "sha1-VHxws0fo0ytOEI6hoqFZ5f3eGcA=" + }, + "standard": { + "version": "14.3.4", + "resolved": "https://registry.npmjs.org/standard/-/standard-14.3.4.tgz", + "integrity": "sha512-+lpOkFssMkljJ6eaILmqxHQ2n4csuEABmcubLTb9almFi1ElDzXb1819fjf/5ygSyePCq4kU2wMdb2fBfb9P9Q==", + "dev": true, + "requires": { + "eslint": "~6.8.0", + "eslint-config-standard": "14.1.1", + "eslint-config-standard-jsx": "8.1.0", + "eslint-plugin-import": "~2.18.0", + "eslint-plugin-node": "~10.0.0", + "eslint-plugin-promise": "~4.2.1", + "eslint-plugin-react": "~7.14.2", + "eslint-plugin-standard": "~4.0.0", + "standard-engine": "^12.0.0" + } + }, + "standard-engine": { + "version": "12.1.0", + "resolved": "https://registry.npmjs.org/standard-engine/-/standard-engine-12.1.0.tgz", + "integrity": "sha512-DVJnWM1CGkag4ucFLGdiYWa5/kJURPONmMmk17p8FT5NE4UnPZB1vxWnXnRo2sPSL78pWJG8xEM+1Tu19z0deg==", + "dev": true, + "requires": { + "deglob": "^4.0.1", + "get-stdin": "^7.0.0", + "minimist": "^1.2.5", + "pkg-conf": "^3.1.0" + } + }, + "statuses": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.5.0.tgz", + "integrity": "sha1-Fhx9rBd2Wf2YEfQ3cfqZOBR4Yow=" + }, + "string-width": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-1.0.2.tgz", + "integrity": "sha1-EYvfW4zcUaKn5w0hHgfisLmxB9M=", + "requires": { + "code-point-at": "^1.0.0", + "is-fullwidth-code-point": "^1.0.0", + "strip-ansi": "^3.0.0" + } + }, + "string.prototype.trimend": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/string.prototype.trimend/-/string.prototype.trimend-1.0.4.tgz", + "integrity": "sha512-y9xCjw1P23Awk8EvTpcyL2NIr1j7wJ39f+k6lvRnSMz+mz9CGz9NYPelDk42kOz6+ql8xjfK8oYzy3jAP5QU5A==", + "dev": true, + "requires": { + "call-bind": "^1.0.2", + "define-properties": "^1.1.3" + } + }, + "string.prototype.trimstart": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/string.prototype.trimstart/-/string.prototype.trimstart-1.0.4.tgz", + "integrity": "sha512-jh6e984OBfvxS50tdY2nRZnoC5/mLFKOREQfw8t5yytkoUsJRNxvI/E39qu1sD0OtWI3OC0XgKSmcWwziwYuZw==", + "dev": true, + "requires": { + "call-bind": "^1.0.2", + "define-properties": "^1.1.3" + } + }, + "string_decoder": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.3.0.tgz", + "integrity": "sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA==", + "requires": { + "safe-buffer": "~5.2.0" + }, + "dependencies": { + "safe-buffer": { + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz", + "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==" + } + } + }, + "strip-ansi": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz", + "integrity": "sha1-ajhfuIU9lS1f8F0Oiq+UJ43GPc8=", + "requires": { + "ansi-regex": "^2.0.0" + } + }, + "strip-bom": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-2.0.0.tgz", + "integrity": "sha1-YhmoVhZSBJHzV4i9vxRHqZx+aw4=", + "requires": { + "is-utf8": "^0.2.0" + } + }, + "strip-json-comments": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-3.1.1.tgz", + "integrity": "sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==", + "dev": true + }, + "supports-color": { + "version": "5.5.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", + "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", + "dev": true, + "requires": { + "has-flag": "^3.0.0" + } + }, + "swagger-ui-dist": { + "version": "3.51.1", + "resolved": "https://registry.npmjs.org/swagger-ui-dist/-/swagger-ui-dist-3.51.1.tgz", + "integrity": "sha512-df2mEeVgnJp/FcXY3DRh3CsTfvHVTaO6g3FJP/kfwhxfOD1+YTXqBZrOIIsYTPtcRIFBkCAto0NFCxAV4XFRbw==" + }, + "swagger-ui-express": { + "version": "4.1.6", + "resolved": "https://registry.npmjs.org/swagger-ui-express/-/swagger-ui-express-4.1.6.tgz", + "integrity": "sha512-Xs2BGGudvDBtL7RXcYtNvHsFtP1DBFPMJFRxHe5ez/VG/rzVOEjazJOOSc/kSCyxreCTKfJrII6MJlL9a6t8vw==", + "requires": { + "swagger-ui-dist": "^3.18.1" + } + }, + "table": { + "version": "5.4.6", + "resolved": "https://registry.npmjs.org/table/-/table-5.4.6.tgz", + "integrity": "sha512-wmEc8m4fjnob4gt5riFRtTu/6+4rSe12TpAELNSqHMfF3IqnA+CH37USM6/YR3qRZv7e56kAEAtd6nKZaxe0Ug==", + "dev": true, + "requires": { + "ajv": "^6.10.2", + "lodash": "^4.17.14", + "slice-ansi": "^2.1.0", + "string-width": "^3.0.0" + }, + "dependencies": { + "ansi-regex": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz", + "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==", + "dev": true + }, + "emoji-regex": { + "version": "7.0.3", + "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-7.0.3.tgz", + "integrity": "sha512-CwBLREIQ7LvYFB0WyRvwhq5N5qPhc6PMjD6bYggFlI5YyDgl+0vxq5VHbMOFqLg7hfWzmu8T5Z1QofhmTIhItA==", + "dev": true + }, + "is-fullwidth-code-point": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", + "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=", + "dev": true + }, + "string-width": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz", + "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==", + "dev": true, + "requires": { + "emoji-regex": "^7.0.1", + "is-fullwidth-code-point": "^2.0.0", + "strip-ansi": "^5.1.0" + } + }, + "strip-ansi": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", + "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", + "dev": true, + "requires": { + "ansi-regex": "^4.1.0" + } + } + } + }, + "tc-core-library-js": { + "version": "github:appirio-tech/tc-core-library-js#df0b36c51cf80918194cbff777214b3c0cf5a151", + "from": "github:appirio-tech/tc-core-library-js#v2.6.4", + "requires": { + "axios": "^0.19.0", + "bunyan": "^1.8.12", + "jsonwebtoken": "^8.5.1", + "jwks-rsa": "^1.6.0", + "lodash": "^4.17.15", + "millisecond": "^0.1.2", + "r7insight_node": "^1.8.4", + "request": "^2.88.0" + } + }, + "term-size": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/term-size/-/term-size-2.2.1.tgz", + "integrity": "sha512-wK0Ri4fOGjv/XPy8SBHZChl8CM7uMc5VML7SqiQ0zG7+J5Vr+RMQDoHa2CNT6KHUnTGIXH34UDMkPzAUyapBZg==", + "dev": true + }, + "text-hex": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/text-hex/-/text-hex-1.0.0.tgz", + "integrity": "sha512-uuVGNWzgJ4yhRaNSiubPY7OjISw4sw4E5Uv0wbjp+OzcbmVU/rsT8ujgcXJhn9ypzsgr5vlzpPqP+MBBKcGvbg==" + }, + "text-table": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/text-table/-/text-table-0.2.0.tgz", + "integrity": "sha1-f17oI66AUgfACvLfSoTsP8+lcLQ=", + "dev": true + }, + "through": { + "version": "2.3.8", + "resolved": "https://registry.npmjs.org/through/-/through-2.3.8.tgz", + "integrity": "sha1-DdTJ/6q8NXlgsbckEV1+Doai4fU=", + "dev": true + }, + "tmp": { + "version": "0.0.33", + "resolved": "https://registry.npmjs.org/tmp/-/tmp-0.0.33.tgz", + "integrity": "sha512-jRCJlojKnZ3addtTOjdIqoRuPEKBvNXcGYqzO6zWZX8KfKEpnGY5jfggJQ3EjKuu8D4bJRr0y+cYJFmYbImXGw==", + "dev": true, + "requires": { + "os-tmpdir": "~1.0.2" + } + }, + "to-readable-stream": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/to-readable-stream/-/to-readable-stream-1.0.0.tgz", + "integrity": "sha512-Iq25XBt6zD5npPhlLVXGFN3/gyR2/qODcKNNyTMd4vbm39HUaOiAM4PMq0eMVC/Tkxz+Zjdsc55g9yyz+Yq00Q==", + "dev": true + }, + "to-regex-range": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", + "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", + "dev": true, + "requires": { + "is-number": "^7.0.0" + } + }, + "toidentifier": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.0.tgz", + "integrity": "sha512-yaOH/Pk/VEhBWWTlhI+qXxDFXlejDGcQipMlyxda9nthulaxLZUNcUqFxokp0vcYnvteJln5FNQDRrxj3YcbVw==" + }, + "toposort-class": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/toposort-class/-/toposort-class-1.0.1.tgz", + "integrity": "sha1-f/0feMi+KMO6Rc1OGj9e4ZO9mYg=" + }, + "touch": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/touch/-/touch-3.1.0.tgz", + "integrity": "sha512-WBx8Uy5TLtOSRtIq+M03/sKDrXCLHxwDcquSP2c43Le03/9serjQBIztjRz6FkJez9D/hleyAXTBGLwwZUw9lA==", + "dev": true, + "requires": { + "nopt": "~1.0.10" + } + }, + "tough-cookie": { + "version": "2.5.0", + "resolved": "https://registry.npmjs.org/tough-cookie/-/tough-cookie-2.5.0.tgz", + "integrity": "sha512-nlLsUzgm1kfLXSXfRZMc1KLAugd4hqJHDTvc2hDIwS3mZAfMEuMbc03SujMF+GEcpaX/qboeycw6iO8JwVv2+g==", + "requires": { + "psl": "^1.1.28", + "punycode": "^2.1.1" + } + }, + "triple-beam": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/triple-beam/-/triple-beam-1.3.0.tgz", + "integrity": "sha512-XrHUvV5HpdLmIj4uVMxHggLbFSZYIn7HEWsqePZcI50pco+MPqJ50wMGY794X7AOOhxOBAjbkqfAbEe/QMp2Lw==" + }, + "tslib": { + "version": "1.14.1", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.14.1.tgz", + "integrity": "sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg==", + "dev": true + }, + "tunnel-agent": { + "version": "0.6.0", + "resolved": "https://registry.npmjs.org/tunnel-agent/-/tunnel-agent-0.6.0.tgz", + "integrity": "sha1-J6XeoGs2sEoKmWZ3SykIaPD8QP0=", + "requires": { + "safe-buffer": "^5.0.1" + } + }, + "tweetnacl": { + "version": "0.14.5", + "resolved": "https://registry.npmjs.org/tweetnacl/-/tweetnacl-0.14.5.tgz", + "integrity": "sha1-WuaBd/GS1EViadEIr6k/+HQ/T2Q=" + }, + "type-check": { + "version": "0.3.2", + "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.3.2.tgz", + "integrity": "sha1-WITKtRLPHTVeP7eE8wgEsrUg23I=", + "dev": true, + "requires": { + "prelude-ls": "~1.1.2" + } + }, + "type-fest": { + "version": "0.8.1", + "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.8.1.tgz", + "integrity": "sha512-4dbzIzqvjtgiM5rw1k5rEHtBANKmdudhGyBEajN01fEyhaAIhsoKNy6y7+IN93IfpFtwY9iqi7kD+xwKhQsNJA==", + "dev": true + }, + "type-is": { + "version": "1.6.18", + "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz", + "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==", + "requires": { + "media-typer": "0.3.0", + "mime-types": "~2.1.24" + } + }, + "typedarray-to-buffer": { + "version": "3.1.5", + "resolved": "https://registry.npmjs.org/typedarray-to-buffer/-/typedarray-to-buffer-3.1.5.tgz", + "integrity": "sha512-zdu8XMNEDepKKR+XYOXAVPtWui0ly0NtohUscw+UmaHiAWT8hrV1rr//H6V+0DvJ3OQ19S979M0laLfX8rm82Q==", + "dev": true, + "requires": { + "is-typedarray": "^1.0.0" + } + }, + "uglify-js": { + "version": "3.14.1", + "resolved": "https://registry.npmjs.org/uglify-js/-/uglify-js-3.14.1.tgz", + "integrity": "sha512-JhS3hmcVaXlp/xSo3PKY5R0JqKs5M3IV+exdLHW99qKvKivPO4Z8qbej6mte17SOPqAOVMjt/XGgWacnFSzM3g==", + "optional": true + }, + "umzug": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/umzug/-/umzug-2.3.0.tgz", + "integrity": "sha512-Z274K+e8goZK8QJxmbRPhl89HPO1K+ORFtm6rySPhFKfKc5GHhqdzD0SGhSWHkzoXasqJuItdhorSvY7/Cgflw==", + "requires": { + "bluebird": "^3.7.2" + } + }, + "unbox-primitive": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/unbox-primitive/-/unbox-primitive-1.0.1.tgz", + "integrity": "sha512-tZU/3NqK3dA5gpE1KtyiJUrEB0lxnGkMFHptJ7q6ewdZ8s12QrODwNbhIJStmJkd1QDXa1NRA8aF2A1zk/Ypyw==", + "dev": true, + "requires": { + "function-bind": "^1.1.1", + "has-bigints": "^1.0.1", + "has-symbols": "^1.0.2", + "which-boxed-primitive": "^1.0.2" + } + }, + "undefsafe": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/undefsafe/-/undefsafe-2.0.3.tgz", + "integrity": "sha512-nrXZwwXrD/T/JXeygJqdCO6NZZ1L66HrxM/Z7mIq2oPanoN0F1nLx3lwJMu6AwJY69hdixaFQOuoYsMjE5/C2A==", + "dev": true, + "requires": { + "debug": "^2.2.0" + }, + "dependencies": { + "debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "dev": true, + "requires": { + "ms": "2.0.0" + } + }, + "ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", + "dev": true + } + } + }, + "uniq": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/uniq/-/uniq-1.0.1.tgz", + "integrity": "sha1-sxxa6CVIRKOoKBVBzisEuGWnNP8=", + "dev": true + }, + "unique-string": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/unique-string/-/unique-string-2.0.0.tgz", + "integrity": "sha512-uNaeirEPvpZWSgzwsPGtU2zVSTrn/8L5q/IexZmH0eH6SA73CmAA5U4GwORTxQAZs95TAXLNqeLoPPNO5gZfWg==", + "dev": true, + "requires": { + "crypto-random-string": "^2.0.0" + } + }, + "unpipe": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", + "integrity": "sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw=" + }, + "update-notifier": { + "version": "4.1.3", + "resolved": "https://registry.npmjs.org/update-notifier/-/update-notifier-4.1.3.tgz", + "integrity": "sha512-Yld6Z0RyCYGB6ckIjffGOSOmHXj1gMeE7aROz4MG+XMkmixBX4jUngrGXNYz7wPKBmtoD4MnBa2Anu7RSKht/A==", + "dev": true, + "requires": { + "boxen": "^4.2.0", + "chalk": "^3.0.0", + "configstore": "^5.0.1", + "has-yarn": "^2.1.0", + "import-lazy": "^2.1.0", + "is-ci": "^2.0.0", + "is-installed-globally": "^0.3.1", + "is-npm": "^4.0.0", + "is-yarn-global": "^0.3.0", + "latest-version": "^5.0.0", + "pupa": "^2.0.1", + "semver-diff": "^3.1.1", + "xdg-basedir": "^4.0.0" + }, + "dependencies": { + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } + }, + "chalk": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-3.0.0.tgz", + "integrity": "sha512-4D3B6Wf41KOYRFdszmDqMCGq5VV/uMAB273JILmO+3jAlh8X4qDtdtgCR3fxtbLEMzSx22QdhnDcJvu2u1fVwg==", + "dev": true, + "requires": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true + }, + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + } + } + }, + "uri-js": { + "version": "4.4.1", + "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.4.1.tgz", + "integrity": "sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg==", + "requires": { + "punycode": "^2.1.0" + } + }, + "url-parse-lax": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/url-parse-lax/-/url-parse-lax-3.0.0.tgz", + "integrity": "sha1-FrXK/Afb42dsGxmZF3gj1lA6yww=", + "dev": true, + "requires": { + "prepend-http": "^2.0.0" + } + }, + "util-deprecate": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", + "integrity": "sha1-RQ1Nyfpw3nMnYvvS1KKJgUGaDM8=" + }, + "utils-merge": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz", + "integrity": "sha1-n5VxD1CiZ5R7LMwSR0HBAoQn5xM=" + }, + "uuid": { + "version": "8.3.2", + "resolved": "https://registry.npmjs.org/uuid/-/uuid-8.3.2.tgz", + "integrity": "sha512-+NYs2QeMWy+GWFOEm9xnn6HCDp0l7QBD7ml8zLUmJ+93Q5NF0NocErnwkTkXVFNiX3/fpC6afS8Dhb/gz7R7eg==" + }, + "v8-compile-cache": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/v8-compile-cache/-/v8-compile-cache-2.3.0.tgz", + "integrity": "sha512-l8lCEmLcLYZh4nbunNZvQCJc5pv7+RCwa8q/LdUx8u7lsWvPDKmpodJAJNwkAhJC//dFY48KuIEmjtd4RViDrA==", + "dev": true + }, + "validate-npm-package-license": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/validate-npm-package-license/-/validate-npm-package-license-3.0.4.tgz", + "integrity": "sha512-DpKm2Ui/xN7/HQKCtpZxoRWBhZ9Z0kqtygG8XCgNQ8ZlDnxuQmWhj566j8fN4Cu3/JmbhsDo7fcAJq4s9h27Ew==", + "requires": { + "spdx-correct": "^3.0.0", + "spdx-expression-parse": "^3.0.0" + } + }, + "validator": { + "version": "13.6.0", + "resolved": "https://registry.npmjs.org/validator/-/validator-13.6.0.tgz", + "integrity": "sha512-gVgKbdbHgtxpRyR8K0O6oFZPhhB5tT1jeEHZR0Znr9Svg03U0+r9DXWMrnRAB+HtCStDQKlaIZm42tVsVjqtjg==" + }, + "vary": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz", + "integrity": "sha1-IpnwLG3tMNSllhsLn3RSShj2NPw=" + }, + "verror": { + "version": "1.10.0", + "resolved": "https://registry.npmjs.org/verror/-/verror-1.10.0.tgz", + "integrity": "sha1-OhBcoXBTr1XW4nDB+CiGguGNpAA=", + "requires": { + "assert-plus": "^1.0.0", + "core-util-is": "1.0.2", + "extsprintf": "^1.2.0" + } + }, + "which": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/which/-/which-1.3.1.tgz", + "integrity": "sha512-HxJdYWq1MTIQbJ3nw0cqssHoTNU267KlrDuGZ1WYlxDStUtKUhOaJmh112/TZmHxxUfuJqPXSOm7tDyas0OSIQ==", + "dev": true, + "requires": { + "isexe": "^2.0.0" + } + }, + "which-boxed-primitive": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/which-boxed-primitive/-/which-boxed-primitive-1.0.2.tgz", + "integrity": "sha512-bwZdv0AKLpplFY2KZRX6TvyuN7ojjr7lwkg6ml0roIy9YeuSr7JS372qlNW18UQYzgYK9ziGcerWqZOmEn9VNg==", + "dev": true, + "requires": { + "is-bigint": "^1.0.1", + "is-boolean-object": "^1.1.0", + "is-number-object": "^1.0.4", + "is-string": "^1.0.5", + "is-symbol": "^1.0.3" + } + }, + "which-module": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/which-module/-/which-module-1.0.0.tgz", + "integrity": "sha1-u6Y8qGGUiZT/MHc2CJ47lgJsKk8=" + }, + "widest-line": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/widest-line/-/widest-line-3.1.0.tgz", + "integrity": "sha512-NsmoXalsWVDMGupxZ5R08ka9flZjjiLvHVAWYOKtiKM8ujtZWr9cRffak+uSE48+Ob8ObalXpwyeUiyDD6QFgg==", + "dev": true, + "requires": { + "string-width": "^4.0.0" + }, + "dependencies": { + "ansi-regex": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.0.tgz", + "integrity": "sha512-bY6fj56OUQ0hU1KjFNDQuJFezqKdrAyFdIevADiqrWHwSlbmBNMHp5ak2f40Pm8JTFyM2mqxkG6ngkHO11f/lg==", + "dev": true + }, + "is-fullwidth-code-point": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", + "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", + "dev": true + }, + "string-width": { + "version": "4.2.2", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.2.tgz", + "integrity": "sha512-XBJbT3N4JhVumXE0eoLU9DCjcaF92KLNqTmFCnG1pf8duUxFGwtP6AD6nkjw9a3IdiRtL3E2w3JDiE/xi3vOeA==", + "dev": true, + "requires": { + "emoji-regex": "^8.0.0", + "is-fullwidth-code-point": "^3.0.0", + "strip-ansi": "^6.0.0" + } + }, + "strip-ansi": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.0.tgz", + "integrity": "sha512-AuvKTrTfQNYNIctbR1K/YGTR1756GycPsg7b9bdV9Duqur4gv6aKqHXah67Z8ImS7WEz5QVcOtlfW2rZEugt6w==", + "dev": true, + "requires": { + "ansi-regex": "^5.0.0" + } + } + } + }, + "window-size": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/window-size/-/window-size-0.2.0.tgz", + "integrity": "sha1-tDFbtCFKPXBY6+7okuE/ok2YsHU=" + }, + "winston": { + "version": "3.3.3", + "resolved": "https://registry.npmjs.org/winston/-/winston-3.3.3.tgz", + "integrity": "sha512-oEXTISQnC8VlSAKf1KYSSd7J6IWuRPQqDdo8eoRNaYKLvwSb5+79Z3Yi1lrl6KDpU6/VWaxpakDAtb1oQ4n9aw==", + "requires": { + "@dabh/diagnostics": "^2.0.2", + "async": "^3.1.0", + "is-stream": "^2.0.0", + "logform": "^2.2.0", + "one-time": "^1.0.0", + "readable-stream": "^3.4.0", + "stack-trace": "0.0.x", + "triple-beam": "^1.3.0", + "winston-transport": "^4.4.0" + } + }, + "winston-transport": { + "version": "4.4.0", + "resolved": "https://registry.npmjs.org/winston-transport/-/winston-transport-4.4.0.tgz", + "integrity": "sha512-Lc7/p3GtqtqPBYYtS6KCN3c77/2QCev51DvcJKbkFPQNoj1sinkGwLGFDxkXY9J6p9+EPnYs+D90uwbnaiURTw==", + "requires": { + "readable-stream": "^2.3.7", + "triple-beam": "^1.2.0" + }, + "dependencies": { + "readable-stream": { + "version": "2.3.7", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.7.tgz", + "integrity": "sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw==", + "requires": { + "core-util-is": "~1.0.0", + "inherits": "~2.0.3", + "isarray": "~1.0.0", + "process-nextick-args": "~2.0.0", + "safe-buffer": "~5.1.1", + "string_decoder": "~1.1.1", + "util-deprecate": "~1.0.1" + } + }, + "string_decoder": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz", + "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==", + "requires": { + "safe-buffer": "~5.1.0" + } + } + } + }, + "wkx": { + "version": "0.5.0", + "resolved": "https://registry.npmjs.org/wkx/-/wkx-0.5.0.tgz", + "integrity": "sha512-Xng/d4Ichh8uN4l0FToV/258EjMGU9MGcA0HV2d9B/ZpZB3lqQm7nkOdZdm5GhKtLLhAE7PiVQwN4eN+2YJJUg==", + "requires": { + "@types/node": "*" + } + }, + "word-wrap": { + "version": "1.2.3", + "resolved": "https://registry.npmjs.org/word-wrap/-/word-wrap-1.2.3.tgz", + "integrity": "sha512-Hz/mrNwitNRh/HUAtM/VT/5VH+ygD6DV7mYKZAtHOrbs8U7lvPS6xf7EJKMF0uW1KJCl0H701g3ZGus+muE5vQ==", + "dev": true + }, + "wordwrap": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/wordwrap/-/wordwrap-1.0.0.tgz", + "integrity": "sha1-J1hIEIkUVqQXHI0CJkQa3pDLyus=" + }, + "wrap-ansi": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-2.1.0.tgz", + "integrity": "sha1-2Pw9KE3QV5T+hJc8rs3Rz4JP3YU=", + "requires": { + "string-width": "^1.0.1", + "strip-ansi": "^3.0.1" + } + }, + "wrappy": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", + "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=" + }, + "write": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/write/-/write-1.0.3.tgz", + "integrity": "sha512-/lg70HAjtkUgWPVZhZcm+T4hkL8Zbtp1nFNOn3lRrxnlv50SRBv7cR7RqR+GMsd3hUXy9hWBo4CHTbFTcOYwig==", + "dev": true, + "requires": { + "mkdirp": "^0.5.1" + } + }, + "write-file-atomic": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/write-file-atomic/-/write-file-atomic-3.0.3.tgz", + "integrity": "sha512-AvHcyZ5JnSfq3ioSyjrBkH9yW4m7Ayk8/9My/DD9onKeu/94fwrMocemO2QAJFAlnnDN+ZDS+ZjAR5ua1/PV/Q==", + "dev": true, + "requires": { + "imurmurhash": "^0.1.4", + "is-typedarray": "^1.0.0", + "signal-exit": "^3.0.2", + "typedarray-to-buffer": "^3.1.5" + } + }, + "xdg-basedir": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/xdg-basedir/-/xdg-basedir-4.0.0.tgz", + "integrity": "sha512-PSNhEJDejZYV7h50BohL09Er9VaIefr2LMAf3OEmpCkjOi34eYyQYAXUTjEQtZJTKcF0E2UKTh+osDLsgNim9Q==", + "dev": true + }, + "xtend": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/xtend/-/xtend-4.0.2.tgz", + "integrity": "sha512-LKYU1iAXJXUgAXn9URjiu+MWhyUXHsvfp7mcuYm9dSUKK0/CjtrUwFAxD82/mCWbtLsGjFIad0wIsod4zrTAEQ==" + }, + "y18n": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/y18n/-/y18n-3.2.2.tgz", + "integrity": "sha512-uGZHXkHnhF0XeeAPgnKfPv1bgKAYyVvmNL1xlKsPYZPaIHxGti2hHqvOCQv71XMsLxu1QjergkqogUnms5D3YQ==" + }, + "yallist": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/yallist/-/yallist-4.0.0.tgz", + "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==" + }, + "yargs": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/yargs/-/yargs-5.0.0.tgz", + "integrity": "sha1-M1UUSXfQV1fbuG1uOOwFYSOzpm4=", + "requires": { + "cliui": "^3.2.0", + "decamelize": "^1.1.1", + "get-caller-file": "^1.0.1", + "lodash.assign": "^4.2.0", + "os-locale": "^1.4.0", + "read-pkg-up": "^1.0.1", + "require-directory": "^2.1.1", + "require-main-filename": "^1.0.1", + "set-blocking": "^2.0.0", + "string-width": "^1.0.2", + "which-module": "^1.0.0", + "window-size": "^0.2.0", + "y18n": "^3.2.1", + "yargs-parser": "^3.2.0" + } + }, + "yargs-parser": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-3.2.0.tgz", + "integrity": "sha1-UIE1XRnZ0MjF2BrakIy05tGGZk8=", + "requires": { + "camelcase": "^3.0.0", + "lodash.assign": "^4.1.0" + } + } + } +} diff --git a/package.json b/package.json new file mode 100644 index 0000000..266cca9 --- /dev/null +++ b/package.json @@ -0,0 +1,48 @@ +{ + "name": "skills-api", + "version": "1.0.0", + "description": "Skills API", + "main": "app.js", + "scripts": { + "start": "node app.js", + "start:dev": "npx nodemon --exec npm start", + "lint": "standard \"**/*.js\"", + "lint:fix": "standard --fix \"**/*.js\"", + "create-db": "node scripts/db/createDb.js", + "delete-data": "node scripts/db/dropAll.js", + "migrate-db-to-es": "node scripts/db/dumpDbToEs.js", + "migrations": "node scripts/db/migrations.js", + "insert-data": "node scripts/db/insert-data.js", + "generate:doc:permissions": "node scripts/permissions-doc", + "generate:doc:permissions:dev": "npx nodemon --watch scripts/permissions-doc --watch src --ext js,jsx,hbs --exec npm run generate:doc:permissions" + }, + "repository": { + "type": "git", + "url": "" + }, + "dependencies": { + "@elastic/elasticsearch": "^7.9.1", + "@hapi/joi": "^16.1.8", + "body-parser": "^1.19.0", + "config": "^3.2.4", + "cors": "^2.8.5", + "express": "^4.17.1", + "get-parameter-names": "^0.3.0", + "handlebars": "^4.7.7", + "js-yaml": "^3.13.1", + "lodash": "^4.17.19", + "pgtools": "^0.3.0", + "sequelize": "^6.3.5", + "swagger-ui-express": "^4.1.4", + "tc-core-library-js": "github:appirio-tech/tc-core-library-js#v2.6.4", + "umzug": "^2.3.0", + "winston": "^3.2.1" + }, + "devDependencies": { + "nodemon": "^2.0.12", + "standard": "^14.3.0" + }, + "engines": { + "node": "12.x" + } +} diff --git a/scripts/constants.js b/scripts/constants.js new file mode 100644 index 0000000..fa5fb4d --- /dev/null +++ b/scripts/constants.js @@ -0,0 +1,50 @@ +/** + * This module contains es resources configuration. + * Identical to the one from ES processor, but updated to work with the api + */ + +const config = require('config') + +const topResources = { + taxonomy: { + index: config.get('ES.DOCUMENTS.taxonomy.index'), + type: config.get('ES.DOCUMENTS.taxonomy.type'), + enrich: { + policyName: config.get('ES.DOCUMENTS.taxonomy.enrichPolicyName'), + matchField: 'id', + enrichFields: ['id', 'name', 'created', 'updated', 'createdBy', 'updatedBy'] + }, + pipeline: { + id: config.get('ES.DOCUMENTS.taxonomy.pipelineId'), + field: 'taxonomyId', + targetField: 'taxonomy', + maxMatches: '1' + } + }, + + skill: { + index: config.get('ES.DOCUMENTS.skill.index'), + type: config.get('ES.DOCUMENTS.skill.type'), + enrich: { + policyName: config.get('ES.DOCUMENTS.skill.enrichPolicyName'), + matchField: 'id', + enrichFields: ['id', 'taxonomyId', 'name', 'externalId', 'uri', 'created', 'updated', 'createdBy', 'updatedBy', 'taxonomyName'] + }, + ingest: { + pipeline: { + id: config.get('ES.DOCUMENTS.taxonomy.pipelineId') + } + } + } + +} + +const modelToESIndexMapping = { + Taxonomy: 'taxonomy', + Skill: 'skill' +} + +module.exports = { + topResources, + modelToESIndexMapping +} diff --git a/scripts/db/createDb.js b/scripts/db/createDb.js new file mode 100644 index 0000000..e29579d --- /dev/null +++ b/scripts/db/createDb.js @@ -0,0 +1,9 @@ +const { createDb } = require('../../src/common/db-helper') + +async function main () { + await createDb() +} + +(async () => { + main().catch(err => console.error(err)) +})() diff --git a/scripts/db/data/Skill.json b/scripts/db/data/Skill.json new file mode 100644 index 0000000..753a5c6 --- /dev/null +++ b/scripts/db/data/Skill.json @@ -0,0 +1,62 @@ +[ + { + "id": "0de5cb8b-d217-4133-8483-a3a09db9ab3d", + "created": "2020-05-05T11:03:56.711Z", + "updated": "2020-05-05T11:04:25.798Z", + "createdBy": "tc-Copilot", + "updatedBy": "tc-Copilot", + "taxonomyId": "7637ae1a-3b7c-44eb-a5ed-10ea02f1885d", + "name": ".NET Framework 4", + "externalId": "KS126XR63RKYVCKYDNBN", + "uri": "http://www.google.com/", + "metadata": { + "challengeProminence": "0.2", + "memberProminence": "0.3" + } + }, + { + "id": "0aec2956-cbcb-4c80-8c00-25cc02a71611", + "created": "2020-05-05T11:03:56.711Z", + "updated": "2020-05-05T11:04:25.798Z", + "createdBy": "tc-Copilot", + "updatedBy": "tc-Copilot", + "taxonomyId": "7637ae1a-3b7c-44eb-a5ed-10ea02f1885d", + "name": "C++ (Programming Language)", + "externalId": "KS1219W70LY1GXZDSKW5", + "uri": "http://www.google.com/", + "metadata": { + "challengeProminence": "0.1", + "memberProminence": "0.2" + } + }, + { + "id": "ab8f01fc-9686-4cc1-9b59-c412b4bae3f2", + "created": "2020-05-05T11:03:56.711Z", + "updated": "2020-05-05T11:04:25.798Z", + "createdBy": "tc-Copilot", + "updatedBy": "tc-Copilot", + "taxonomyId": "7637ae1a-3b7c-44eb-a5ed-10ea02f1885d", + "name": "Angular (Web Framework)", + "externalId": "KS120H6772VQ0MQ5RLVD", + "uri": "http://www.google.com/", + "metadata": { + "challengeProminence": "0.5", + "memberProminence": "0.7" + } + }, + { + "id": "6df99fc1-3115-4b0c-bf2b-21ecd52fa64b", + "created": "2020-05-05T11:03:56.711Z", + "updated": "2020-05-05T11:04:25.798Z", + "createdBy": "tc-Copilot", + "updatedBy": "tc-Copilot", + "taxonomyId": "aad07858-5846-4c6f-aab6-0682a025d675", + "name": "Python (Programming Language)", + "externalId": "KS125LS6N7WP4S6SFTCK", + "uri": "http://www.google.com/", + "metadata": { + "challengeProminence": "0.7", + "memberProminence": "1.0" + } + } +] diff --git a/scripts/db/data/Taxonomy.json b/scripts/db/data/Taxonomy.json new file mode 100644 index 0000000..b196d44 --- /dev/null +++ b/scripts/db/data/Taxonomy.json @@ -0,0 +1,20 @@ +[ + { + "id": "7637ae1a-3b7c-44eb-a5ed-10ea02f1885d", + "created": "2020-05-05T11:02:49.718Z", + "updated": null, + "createdBy": "tc-Copilot", + "updatedBy": null, + "name": "EMSI", + "metadata": {} + }, + { + "id": "aad07858-5846-4c6f-aab6-0682a025d675", + "created": "2020-05-05T11:02:49.718Z", + "updated": null, + "createdBy": "tc-Copilot", + "updatedBy": null, + "name": "Github", + "metadata": {} + } +] diff --git a/scripts/db/dropAll.js b/scripts/db/dropAll.js new file mode 100644 index 0000000..be168ac --- /dev/null +++ b/scripts/db/dropAll.js @@ -0,0 +1,65 @@ +/** + * drop tables + */ +const _ = require('lodash') +const sequelize = require('../../src/models/index') +const logger = require('../../src/common/logger') +const { + topResources, + modelToESIndexMapping +} = require('../constants') +const { getESClient } = require('../../src/common/es-client') + +async function main () { + const client = getESClient() + + // delete es pipelines + try { + logger.info('Deleting all pipelines...') + await client.ingest.deletePipeline({ + id: topResources.taxonomy.pipeline.id + }) + logger.info('Successfully deleted') + } catch (e) { + console.error(e) + logger.warn('Delete all ingest pipelines failed') + } + + // delete data in es + const keys = Object.keys(sequelize.models) + for (let i = 0; i < keys.length; i++) { + const key = keys[i] + const esResourceName = modelToESIndexMapping[key] + try { + if (_.includes(_.keys(topResources), esResourceName)) { + if (topResources[esResourceName].enrich) { + logger.info(`Deleting enrich policy for ${esResourceName}`) + await client.enrich.deletePolicy({ + name: topResources[esResourceName].enrich.policyName + }) + logger.info(`Successfully deleted enrich policy for ${esResourceName}`) + } + logger.info(`Deleting index for ${esResourceName}`) + await client.indices.delete({ + index: topResources[esResourceName].index + }) + logger.info(`Successfully deleted enrich policy for ${esResourceName}`) + } + } catch (e) { + console.error(e) + logger.warn(`deleting data in es for ${key} failed`) + } + } + + // delete tables + try { + await sequelize.drop() + } catch (e) { + console.error(e) + logger.warn('deleting tables failed') + } +} + +(async () => { + main().catch(err => console.error(err)) +})() diff --git a/scripts/db/dumpDbToEs.js b/scripts/db/dumpDbToEs.js new file mode 100644 index 0000000..648a6aa --- /dev/null +++ b/scripts/db/dumpDbToEs.js @@ -0,0 +1,266 @@ +const _ = require('lodash') +const config = require('config') +const sequelize = require('../../src/models/index') +const dbHelper = require('../../src/common/db-helper') +const serviceHelper = require('../../src/common/service-helper') +const logger = require('../../src/common/logger') +const { getESClient } = require('../../src/common/es-client') +const { + topResources, + modelToESIndexMapping +} = require('../constants') + +const models = sequelize.models + +// Declares the ordering of the resource data insertion, to ensure that enrichment happens correctly +const RESOURCES_IN_ORDER = [ + 'taxonomy', + 'skill' +] + +const client = getESClient() + +const RESOURCE_NOT_FOUND = 'resource_not_found_exception' +const INDEX_NOT_FOUND = 'index_not_found_exception' + +/** + * Cleans up the data in elasticsearch + * @param {Array} keys Array of models + */ +async function cleanupES (keys) { + const client = getESClient() + try { + await client.ingest.deletePipeline({ + id: topResources.taxonomy.pipeline.id + }) + } catch (e) { + if (e.meta && e.meta.body.error.type === RESOURCE_NOT_FOUND) { + // Ignore + } else { + throw e + } + } + + try { + for (let i = 0; i < keys.length; i++) { + const key = keys[i] + if (models[key].tableName) { + const esResourceName = modelToESIndexMapping[key] + if (_.includes(_.keys(topResources), esResourceName)) { + if (topResources[esResourceName].enrich) { + try { + await client.enrich.deletePolicy({ + name: topResources[esResourceName].enrich.policyName + }) + } catch (e) { + if (e.meta && e.meta.body.error.type === RESOURCE_NOT_FOUND) { + // Ignore + } else { + throw e + } + } + } + + try { + await client.indices.delete({ + index: topResources[esResourceName].index + }) + } catch (e) { + if (e.meta && e.meta.body.error.type === INDEX_NOT_FOUND) { + // Ignore + } else { + throw e + } + } + } + } + } + } catch (e) { + console.log(JSON.stringify(e)) + throw e + } + console.log('Existing data in elasticsearch has been deleted!') +} + +async function insertBulkIntoES (esResourceName, dataset) { + if (!esResourceName) { + logger.error(`Cannot insert data into esResourceName ${esResourceName} as it's not found`) + return + } + + const resourceConfig = config.get(`ES.DOCUMENTS.${esResourceName}`) + + const chunked = _.chunk(dataset, config.get('ES.MAX_BULK_SIZE')) + for (const ds of chunked) { + const body = _.flatMap(ds, doc => [{ index: { _id: doc.id } }, doc]) + try { + await client.bulk({ + index: resourceConfig.index, + type: resourceConfig.type, + body, + pipeline: resourceConfig.ingest ? resourceConfig.ingest.pipeline.id : undefined, + refresh: 'wait_for' + }) + } catch (e) { + logger.error('ES, create mapping error.') + logger.error(JSON.stringify(e)) + } + } +} + +/** + * Creates and executes the enrich policy for the provided model + * @param {String} modelName The model name + */ +async function createAndExecuteEnrichPolicy (modelName) { + const esResourceName = modelToESIndexMapping[modelName] + + if (_.includes(_.keys(topResources), esResourceName) && topResources[esResourceName].enrich) { + await client.enrich.putPolicy({ + name: topResources[esResourceName].enrich.policyName, + body: { + match: { + indices: topResources[esResourceName].index, + match_field: topResources[esResourceName].enrich.matchField, + enrich_fields: topResources[esResourceName].enrich.enrichFields + } + } + }) + await client.enrich.executePolicy({ name: topResources[esResourceName].enrich.policyName }) + } +} + +/** + * Creates the ingest pipeline using the enrich policy + * @param {String} modelName The model name + */ +async function createEnrichProcessor (modelName) { + const esResourceName = modelToESIndexMapping[modelName] + + if (_.includes(_.keys(topResources), esResourceName) && topResources[esResourceName].pipeline) { + if (topResources[esResourceName].pipeline.processors) { + const processors = [] + + for (let i = 0; i < topResources[esResourceName].pipeline.processors.length; i++) { + const ep = topResources[esResourceName].pipeline.processors[i] + processors.push({ + foreach: { + field: ep.referenceField, + ignore_missing: true, + processor: { + enrich: { + policy_name: ep.enrichPolicyName, + ignore_missing: true, + field: ep.field, + target_field: ep.targetField, + max_matches: ep.maxMatches + } + } + } + }) + } + + await client.ingest.putPipeline({ + id: topResources[esResourceName].pipeline.id, + body: { + processors + } + }) + } else { + await client.ingest.putPipeline({ + id: topResources[esResourceName].pipeline.id, + body: { + processors: [{ + enrich: { + policy_name: topResources[esResourceName].enrich.policyName, + field: topResources[esResourceName].pipeline.field, + target_field: topResources[esResourceName].pipeline.targetField, + max_matches: topResources[esResourceName].pipeline.maxMatches + } + }] + } + }) + } + } +} + +/** + * import test data + * @return {Promise} + */ +async function main () { + let keys = Object.keys(models) + + // Sort the models in the order of insertion (for correct enrichment) + const temp = Array(keys.length).fill(null) + keys.forEach(k => { + if (sequelize.models[k].name) { + const esResourceName = modelToESIndexMapping[k] + const index = RESOURCES_IN_ORDER.indexOf(esResourceName) + temp[index] = k + } + }) + keys = _.compact(temp) + + await cleanupES(keys) + + for (let i = 0; i < keys.length; i++) { + const key = keys[i] + const queryPage = { perPage: parseInt(config.get('ES.MAX_BATCH_SIZE'), 10), page: 1 } + try { + while (true) { + const data = await dbHelper.find(models[key], { ...queryPage }) + for (let i = 0; i < data.length; i++) { + logger.info(`Inserting data ${(i + 1) + (queryPage.perPage * (queryPage.page - 1))}`) + logger.info(JSON.stringify(data[i])) + if (!_.isString(data[i].created)) { + data[i].created = new Date() + } + if (!_.isString(data[i].updated)) { + data[i].updated = new Date() + } + if (!_.isString(data[i].createdBy)) { + data[i].createdBy = 'tcAdmin' + } + if (!_.isString(data[i].updatedBy)) { + data[i].updatedBy = 'tcAdmin' + } + } + await insertBulkIntoES(serviceHelper.getResource(key), data) + if (data.length < queryPage.perPage) { + logger.info('import data for ' + key + ' done') + break + } else { + queryPage.page = queryPage.page + 1 + } + } + } catch (e) { + logger.error(JSON.stringify(_.get(e, 'meta.body', ''), null, 4)) + logger.error(_.get(e, 'meta.meta.request.params.method', '')) + logger.error(_.get(e, 'meta.meta.request.params.path', '')) + logger.warn('import data for ' + key + ' failed') + continue + } + try { + await createAndExecuteEnrichPolicy(key) + logger.info('create and execute enrich policy for ' + key + ' done') + } catch (e) { + logger.error(JSON.stringify(_.get(e, 'meta.body', ''), null, 4)) + logger.warn('create and execute enrich policy for ' + key + ' failed') + } + + try { + await createEnrichProcessor(key) + logger.info('create enrich processor (pipeline) for ' + key + ' done') + } catch (e) { + logger.error(JSON.stringify(_.get(e, 'meta.body', ''), null, 4)) + logger.warn('create enrich processor (pipeline) for ' + key + ' failed') + } + } + logger.info('all done') + process.exit(0) +} + +(async () => { + main().catch(err => console.error(err)) +})() diff --git a/scripts/db/insert-data.js b/scripts/db/insert-data.js new file mode 100644 index 0000000..8fce212 --- /dev/null +++ b/scripts/db/insert-data.js @@ -0,0 +1,28 @@ +const sequelize = require('../../src/models/index') +const logger = require('../../src/common/logger') + +const models = sequelize.models + +const dataKeys = ['Taxonomy', 'Skill'] + +/** + * import seed data + */ +async function main () { + for (const key of dataKeys) { + try { + const records = require(`./data/${key}.json`) + await models[key].bulkCreate(records) + logger.info(`import data for ${key} done, record count: ${records.length}`) + } catch (e) { + logger.error(e) + logger.warn(`import data for ${key} failed`) + } + } + logger.info('all done') + process.exit(0) +} + +(async () => { + main().catch(err => console.error(err)) +})() diff --git a/scripts/db/migrations.js b/scripts/db/migrations.js new file mode 100644 index 0000000..739566e --- /dev/null +++ b/scripts/db/migrations.js @@ -0,0 +1,35 @@ +const sequelize = require('../../src/models/index') +const path = require('path') +const Umzug = require('umzug') + +function getUmzug () { + return new Umzug({ + migrations: { + // indicates the folder containing the migration .js files + path: path.join(__dirname, './migrations'), + // inject sequelize's QueryInterface in the migrations + params: [ + sequelize.getQueryInterface() + ] + }, + // indicates that the migration data should be store in the database + // itself through sequelize. The default configuration creates a table + // named `SequelizeMeta`. + storage: 'sequelize', + storageOptions: { + sequelize: sequelize + } + }) +} + +(async () => { + if (process.argv[2] === 'up') { + await getUmzug().up() + console.log('All migrations performed successfully') + } else if (process.argv[2] === 'down') { + await getUmzug().down() + console.log('The last executed migration reverted successfully') + } else { + console.error('You should pass \'up\' or \'down\' as the parameter') + } +})() diff --git a/scripts/db/migrations/01-create-taxonomy.js b/scripts/db/migrations/01-create-taxonomy.js new file mode 100644 index 0000000..609b04a --- /dev/null +++ b/scripts/db/migrations/01-create-taxonomy.js @@ -0,0 +1,35 @@ +const { DataTypes } = require('sequelize') + +module.exports = { + up: async (query) => { + await query.createTable('Taxonomies', { + id: { + primaryKey: true, + type: DataTypes.UUID, + defaultValue: DataTypes.UUIDV4 + }, + createdBy: { + type: DataTypes.STRING + }, + updatedBy: { + type: DataTypes.STRING + }, + name: { + type: DataTypes.STRING + }, + metadata: { + type: DataTypes.JSONB + }, + created: { + type: DataTypes.DATE, + allowNull: false + }, + updated: { + type: DataTypes.DATE + } + }) + }, + down: async (query) => { + await query.dropTable('Taxonomies') + } +} diff --git a/scripts/db/migrations/02_create-skill.js b/scripts/db/migrations/02_create-skill.js new file mode 100644 index 0000000..02c4e1a --- /dev/null +++ b/scripts/db/migrations/02_create-skill.js @@ -0,0 +1,41 @@ +const { DataTypes } = require('sequelize') + +module.exports = { + up: async (query) => { + await query.createTable('Skills', { + id: { + primaryKey: true, + type: DataTypes.UUID, + defaultValue: DataTypes.UUIDV4 + }, + createdBy: { + type: DataTypes.STRING + }, + updatedBy: { + type: DataTypes.STRING + }, + name: { + type: DataTypes.STRING + }, + externalId: { + type: DataTypes.STRING + }, + uri: { + type: DataTypes.STRING + }, + metadata: { + type: DataTypes.JSONB + }, + created: { + type: DataTypes.DATE, + allowNull: false + }, + updated: { + type: DataTypes.DATE + } + }) + }, + down: async (query) => { + await query.dropTable('Skills') + } +} diff --git a/scripts/db/migrations/03_add-relationship.js b/scripts/db/migrations/03_add-relationship.js new file mode 100644 index 0000000..c911b87 --- /dev/null +++ b/scripts/db/migrations/03_add-relationship.js @@ -0,0 +1,17 @@ +const { DataTypes } = require('sequelize') + +module.exports = { + up: async (query) => { + await query.addColumn('Skills', 'taxonomyId', { + type: DataTypes.UUID, + references: { + model: 'Taxonomies', + key: 'id' + }, + onUpdate: 'CASCADE' + }) + }, + down: async (query) => { + await query.removeColumn('Skills', 'taxonomyId') + } +} diff --git a/scripts/permissions-doc/index.js b/scripts/permissions-doc/index.js new file mode 100644 index 0000000..40ffe71 --- /dev/null +++ b/scripts/permissions-doc/index.js @@ -0,0 +1,72 @@ +/** + * Generate a permissions.html document using the permission config. + * + * Run by: `npm run generate:doc:permissions` + * + * For development purpose, run by `npm run generate:doc:permissions:dev` which would regenerate HTML on every update. + */ +const _ = require('lodash') +const fs = require('fs') +const path = require('path') +const handlebars = require('handlebars') +const { + PERMISSION +} = require('../../src/permissions/constants') + +const docTemplatePath = path.resolve(__dirname, './template.hbs') +const outputDocPath = path.resolve(__dirname, '../../docs/permissions.html') + +handlebars.registerHelper('istrue', value => value === true) + +/** + * Normalize permission object which has "simple" and "full" shape into a "full" shape for consistency + * + * @param {Object} permission permission object + * + * @returns {Objects} permission object in the "full" shape with "allowRule" and "denyRule" + */ +function normalizePermission (permission) { + let normalizedPermission = permission + + if (!normalizedPermission.allowRule) { + normalizedPermission = { + meta: permission.meta, + allowRule: _.omit(permission, 'meta') + } + } + + return normalizedPermission +} + +const templateStr = fs.readFileSync(docTemplatePath).toString() +const renderDocument = handlebars.compile(templateStr) + +const permissionKeys = _.keys(PERMISSION) +// prepare permissions without modifying data in constant `PERMISSION` +const allPermissions = permissionKeys.map((key) => { + // add `key` to meta + const meta = _.assign({}, PERMISSION[key].meta, { + key + }) + + // update `meta` to one with `key` + return _.assign({}, PERMISSION[key], { + meta + }) +}) +const groupsObj = _.groupBy(allPermissions, 'meta.group') +const groups = _.toPairs(groupsObj).map(([title, permissions]) => ({ + title, + anchor: `section-${title.toLowerCase().replace(' ', '-')}`, + permissions +})) + +groups.forEach((group) => { + group.permissions = group.permissions.map(normalizePermission) +}) + +const data = { + groups +} + +fs.writeFileSync(outputDocPath, renderDocument(data)) diff --git a/scripts/permissions-doc/template.hbs b/scripts/permissions-doc/template.hbs new file mode 100644 index 0000000..7b8a860 --- /dev/null +++ b/scripts/permissions-doc/template.hbs @@ -0,0 +1,115 @@ + + + + + + + Permissions + + + +
+
+

Permissions

+

List of all the possible user permissions inside Skills API

+
+

Legend:

+
    +
  • allowed Topcoder Role - users with such a Topcoder Role are allowed to perform the action
  • +
  • denied Topcoder Role - users with such a Topcoder Role are denied to perform the action even they have some other allow roles
  • +
  • allowed M2M Scope - M2M tokens with such a scope are allowed to perform the action
  • +
  • denied M2M Scope - M2M tokens with such a scope are allowed to perform the action even they have some other allow scopes
  • +
+
+ + {{#each groups}} +
+
+

+ {{title}} +

+
+
+ {{#each permissions}} +
+
+
+ {{meta.title}} +
+
{{meta.key}}
+
{{meta.description}}
+
+
+
+ {{#if (istrue allowRule.topcoderRoles)}} + Any Logged-in User + {{else}} + {{#each allowRule.topcoderRoles}} + {{this}} + {{/each}} + {{/if}} + {{#each denyRule.topcoderRoles}} + {{this}} + {{/each}} +
+ +
+ {{#each allowRule.scopes}} + {{this}} + {{/each}} + {{#each denyRule.scopes}} + {{this}} + {{/each}} +
+
+
+ {{/each}} + {{/each}} + +
+ + diff --git a/src/bootstrap.js b/src/bootstrap.js new file mode 100755 index 0000000..087ba72 --- /dev/null +++ b/src/bootstrap.js @@ -0,0 +1,42 @@ +/** + * add logger and joi to services + */ + +const config = require('config') +const fs = require('fs') +const path = require('path') +const logger = require('./common/logger') +const joi = require('@hapi/joi') + +joi.id = () => joi.number().integer().min(1) +joi.pageSize = () => joi.number().integer().min(1).max(config.get('MAX_PAGE_SIZE')) +joi.prominence = (name) => joi.string().custom((value, helper) => { + // check if value is in the range [0, 1] + const num = +value + const msg = `${name} must be a string with value in the range [0, 1]` + if (isNaN(num) || num < 0 || num > 1) { return helper.message(msg) } + + return value +}) + +/** + * add logger and joi schema to service + * @param dir + */ +function buildServices (dir) { + const files = fs.readdirSync(dir) + files.forEach((file) => { + const curPath = path.join(dir, file) + const stats = fs.statSync(curPath) + if (stats.isDirectory()) { + buildServices(curPath) + } else if (file.toLowerCase().indexOf('service.js') >= 0) { + let serviceName = curPath.split('modules')[1] + serviceName = serviceName.substr(1, serviceName.length - 4) + logger.info(`add decorators for service --> ${serviceName}`) + logger.buildService(serviceName, require(curPath)); // eslint-disable-line + } + }) +} + +buildServices(path.join(__dirname, 'modules')) diff --git a/src/common/controller-helper.js b/src/common/controller-helper.js new file mode 100644 index 0000000..9b6b188 --- /dev/null +++ b/src/common/controller-helper.js @@ -0,0 +1,66 @@ +/** + * get common controller service + * @param service the service + * @return {{patch: patch, search: search, get: get, create: create, update: update, remove: remove}} the common controller methods + */ +function getControllerMethods (service) { + const { injectSearchMeta } = require('./helper') + + /** + * create entity by request data + * @param req the http request + * @param res the http response + */ + async function create (req, res) { + res.json(await service.create(req.body, req.authUser)) + } + + /** + * patch entity by id + * @param req the http request + * @param res the http response + */ + async function patch (req, res) { + res.json(await service.patch(req.params.id, req.body, req.authUser)) + } + + /** + * get entity by id + * @param req the http request + * @param res the http response + */ + async function get (req, res) { + res.json(await service.get(req.params.id, req.query)) + } + + /** + * search entities by request query + * @param req the http request + * @param res the http response + */ + async function search (req, res) { + const result = await service.search(req.query) + injectSearchMeta(req, res, result) + res.send(result.result) + } + + /** + * remove entity by id + * @param req the http request + * @param res the http response + */ + async function remove (req, res) { + await service.remove(req.params.id, req.authUser) + res.status(204).end() + } + + return { + create, + search, + remove, + get, + patch + } +} + +module.exports = { getControllerMethods } diff --git a/src/common/db-helper.js b/src/common/db-helper.js new file mode 100644 index 0000000..8bd78df --- /dev/null +++ b/src/common/db-helper.js @@ -0,0 +1,175 @@ +const _ = require('lodash') +const pgtools = require('pgtools') +const config = require('config') +const errors = require('../common/errors') +const logger = require('../common/logger') +const helper = require('../common/helper') + +/** + * create db + * @returns {Promise} + */ +async function createDb () { + const dbConfig = { + user: config.DB_USERNAME, + password: config.DB_PASSWORD, + port: config.DB_PORT, + host: config.DB_HOST + } + + pgtools.createdb(dbConfig, config.DB_NAME, (err, res) => { + if (err) { + if (err.name === 'duplicate_database') { + logger.info(`db ${config.DB_NAME} already exists`) + } else { + throw err + } + } else { + logger.info(`Created db ${config.DB_NAME} successfully`) + } + }) +} + +/** + * search objects + * @param model the sequelize model object + * @param query search query + * @param inlcude what models to include + * @returns {Promise} + */ +async function find (model, query, include = null) { + // set pagination params + const options = {} + if (!query.perPage) { + query.perPage = config.PER_PAGE + } + if (query.page) { + options.offset = (query.page - 1) * query.perPage + } + options.limit = query.perPage + delete query.perPage + delete query.page + + // set filter params + options.where = query + + // set model inlcudes + if (include) { + options.include = include + } + + // execute query + return model.findAll(options) +} + +/** + * get object by pk + * @param model the sequelize model object + * @param pk object pk (primary key) + * @param params the path params (use if id is null) + * @returns {Promise} + */ +async function get (model, pk, params) { + let instance + if (pk) { + instance = await model.findByPk(pk) + } + if (params) { + const instances = await model.findAll({ where: params }) + if (instances.length === 0) { + throw errors.newEntityNotFoundError(`cannot find ${model.name} where ${_.map(params, (v, k) => `${k}:${v}`).join(', ')}`) + } + return instances[0] + } + + if (!instance) { + throw errors.newEntityNotFoundError(`cannot find ${model.name} where id = ${pk}`) + } + return instance +} + +/** + * create object + * @param model the sequelize model object + * @param entity entity to create + * @param auth the user auth object + * @returns {Promise} + */ +async function create (model, entity, auth) { + if (auth) { + entity.createdBy = helper.getAuthUser(auth) + } + return model.create(entity) +} + +/** + * delete object by pk + * @param model the sequelize model object + * @param pk the primary key + * @returns {Promise} + */ +async function remove (model, pk, params) { + const instance = await get(model, pk, params) + return instance.destroy() +} + +/** + * update object by pk + * @param model the sequelize model object + * @param pk the primary key + * @param entity entity to create + * @param auth the auth object + * @param auth the path params + * @returns {Promise} + */ +async function update (model, pk, entity, auth, params) { + // ensure that object exists + const instance = await get(model, pk, params) + entity.updatedBy = helper.getAuthUser(auth) + return instance.update(entity) +} + +/** + * make sure unique + * @param model the db model + * @param entity the entity + * @param uniqueFields the uniqueFields + * @param pathParams the path params + */ +async function makeSureUnique (model, entity, uniqueFields, pathParams = {}) { + if (_.isNil(uniqueFields) || _.isEmpty(uniqueFields)) { + return + } + for (let i = 0; i < uniqueFields.length; i++) { + const params = {} + _.each(uniqueFields[i], f => { + if (entity[f] && entity[f] !== pathParams[f]) { + params[f] = entity[f] + } + }) + if (Object.keys(params).length === 0) { + return + } + + const items = await find(model, params) + if (items.length <= 0) { + continue + } + + if ((items.length > 0 && !entity.id) || + (items[0].id !== entity.id) + ) { + throw errors.newConflictError(`${model.name} already exists with ${_.map(params, (v, k) => `${k}:${v}`).join(', ')}`) + } + } +} + +module.exports = { + createDb, + find, + create, + update, + get, + remove, + makeSureUnique +} diff --git a/src/common/error.middleware.js b/src/common/error.middleware.js new file mode 100755 index 0000000..4a89e2c --- /dev/null +++ b/src/common/error.middleware.js @@ -0,0 +1,33 @@ +/* eslint-disable no-unused-vars */ + +/** + * Common error handling middleware + */ +const { UniqueConstraintError } = require('sequelize') +const logger = require('./logger') + +const DEFAULT_MESSAGE = 'Something is broken at API server-side.' + +/** + * The error middleware function + * + * @param {Object} err the error that is thrown in the application + * @param {Object} req the express request instance + * @param {Object} res the express response instance + * @param {Function} next the next middleware in the chain + */ +function middleware (err, req, res, next) { + logger.logFullError(err) + + if (err instanceof UniqueConstraintError) { + res.status(409).json({ message: err.message }) + } else if (err.isJoi) { + res.status(400).json({ message: err.details[0].message }) + } else { + const status = err.status || 500 + const message = err.message || DEFAULT_MESSAGE + res.status(status).json({ message }) + } +} + +module.exports = () => middleware diff --git a/src/common/errors.js b/src/common/errors.js new file mode 100644 index 0000000..5a3a868 --- /dev/null +++ b/src/common/errors.js @@ -0,0 +1,29 @@ +/** + * errors defined + */ + +/** + * the base error class + */ +class AppError extends Error { + constructor (status, message) { + super() + this.status = status + this.message = message || 'unknown exception' + } +} + +module.exports = { + newBadRequestError: (msg) => new AppError( + 400, + msg || 'The request could not be interpreted correctly or some required parameters were missing.' + ), + ForbiddenError: msg => new AppError(403, msg || 'Forbidden'), + NotFoundError: msg => new AppError(404, msg || 'Not Found'), + newEntityNotFoundError: msg => new AppError(404, msg || 'The entity does not exist.'), + newAuthError: msg => new AppError(401, msg || 'Auth failed.'), + newPermissionError: msg => new AppError(403, msg || 'The entity does not exist.'), + newConflictError: msg => new AppError(409, msg || 'The entity does not exist.'), + deleteConflictError: msg => new AppError(400, msg || 'Please delete child records first'), + elasticSearchEnrichError: msg => new AppError(500, msg || 'Elasticsearch enrich failed') +} diff --git a/src/common/es-client.js b/src/common/es-client.js new file mode 100644 index 0000000..0bc306b --- /dev/null +++ b/src/common/es-client.js @@ -0,0 +1,38 @@ +const config = require('config') +const elasticsearch = require('@elastic/elasticsearch') + +// Elasticsearch client +let esClient + +/** + * Get ES Client + * @return {Object} Elasticsearch Client Instance + */ +function getESClient () { + if (esClient) { + return esClient + } + const host = config.ES.HOST + const cloudId = config.ES.ELASTICCLOUD.id + if (cloudId) { + // Elastic Cloud configuration + esClient = new elasticsearch.Client({ + cloud: { + id: cloudId + }, + auth: { + username: config.ES.ELASTICCLOUD.username, + password: config.ES.ELASTICCLOUD.password + } + }) + } else { + esClient = new elasticsearch.Client({ + node: host + }) + } + return esClient +} + +module.exports = { + getESClient +} diff --git a/src/common/es-helper.js b/src/common/es-helper.js new file mode 100644 index 0000000..c250010 --- /dev/null +++ b/src/common/es-helper.js @@ -0,0 +1,376 @@ +const config = require('config') +const _ = require('lodash') +const logger = require('../common/logger') +const esClient = require('./es-client').getESClient() + +const DOCUMENTS = config.ES.DOCUMENTS + +// resource filter config +const RESOURCE_FILTER = { + // independent resources + skill: { + externalId: { + resource: 'skill', + queryField: 'externalId' + }, + taxonomyId: { + resource: 'skill', + queryField: 'taxonomyId' + }, + name: { + resource: 'skill', + queryField: 'name' + } + }, + taxonomy: { + name: { + resource: 'taxonomy', + queryField: 'name' + } + } +} + +// filter chain config +const FILTER_CHAIN = { + taxonomy: { + filterNext: 'skill', + queryField: 'taxonomyId', + idField: 'id' + }, + skill: { + queryField: 'skillId', + idField: 'taxonomyId' + } +} + +function getTotalCount (total) { + return typeof total === 'number' ? total : total.value +} + +/** + * inserts data into ES + * @param esResourceName the es resource name + * @param data the data to insert + */ +async function insertIntoES (esResourceName, data) { + const resourceConfig = config.get(`ES.DOCUMENTS.${esResourceName}`) + await esClient.create({ + index: resourceConfig.index, + type: resourceConfig.type, + refresh: config.get('ES.ES_REFRESH'), + body: data, + id: data.id + }) +} + +/** + * updated ES record + * @param esResourceName the ES resource name + * @param data the data to update + */ +async function updateESRecord (esResourceName, data) { + const resourceConfig = config.get(`ES.DOCUMENTS.${esResourceName}`) + await esClient.update({ + index: resourceConfig.index, + type: resourceConfig.type, + refresh: config.get('ES.ES_REFRESH'), + id: data.id, + body: { + doc: data + } + }) +} + +/** + * removes record from ES + * @param esResourceName the ES resource name + * @param id the id of the record to remove + */ +async function deleteESRecord (esResourceName, id) { + const resourceConfig = config.get(`ES.DOCUMENTS.${esResourceName}`) + await esClient.delete({ + index: resourceConfig.index, + type: resourceConfig.type, + refresh: config.get('ES.ES_REFRESH'), + id: id + }) +} + +/** + * Get a resource by Id from ES. + * @param resource the resource to get + * @param args the request path and query parameters + * @returns {Promise<*>} the promise of retrieved resource object from ES + */ +async function getFromElasticSearch (resource, ...args) { + logger.debug(`Get from ES first: args ${JSON.stringify(args, null, 2)}`) + const id = args[0] + + const doc = DOCUMENTS[resource] + + // construct ES query + const esQuery = { + index: doc.index, + type: doc.type, + id: id + } + + logger.debug(`ES query for get ${resource}: ${JSON.stringify(esQuery, null, 2)}`) + + // query ES + const { body: result } = await esClient.getSource(esQuery) + return result +} + +/** + * Parse the query parameters to resource filter list. + * @param resource the resource to apply filter + * @param params the request query parameter + * @param itself true mean the filter is applied the resource itself + * @returns {[]} parsed filter array + */ +function parseResourceFilter (resource, params, itself) { + const resDefinedFilters = RESOURCE_FILTER[resource] + const resFilters = [] + if (resDefinedFilters) { + const resQueryParams = _.pick(params, Object.keys(resDefinedFilters)) + if (!_.isEmpty(resQueryParams)) { + // filter by child resource + _.forOwn(resQueryParams, (value, query) => { + const filterDef = resDefinedFilters[query] + if (itself && resource === filterDef.resource) { + resFilters.push({ + param: query, + resource: filterDef.resource, + queryField: filterDef.queryField, + value + }) + } else if (!itself && resource !== filterDef.resource) { + resFilters.push({ + param: query, + resource: filterDef.resource, + queryField: filterDef.queryField, + value + }) + } + }) + } + } + return resFilters +} + +/** + * Set the filters to ES query. + * @param resFilters the resource filters + * @param esQuery the ES query + */ +function setResourceFilterToEsQuery (resFilters, esQuery) { + if (resFilters.length > 0) { + for (const filter of resFilters) { + let matchField = `${filter.queryField}` + if (filter.queryField !== 'name') { + matchField = matchField + '.keyword' + } + esQuery.body.query.bool.must.push({ + match: { + [matchField]: filter.value + } + }) + } + } +} + +/** + * Set filter values to ES query. + * @param esQuery the ES query object + * @param matchField the field to match + * @param filterValue the filter value, it can be array or single value + * @param queryField the field that the filter applies + * @returns {*} the ES query + */ +function setFilterValueToEsQuery (esQuery, matchField, filterValue, queryField) { + if (queryField !== 'name') { + matchField = matchField + '.keyword' + } + if (Array.isArray(filterValue)) { + for (const value of filterValue) { + esQuery.body.query.bool.should.push({ + match: { + [matchField]: value + } + }) + } + esQuery.body.query.bool.minimum_should_match = 1 + } else { + esQuery.body.query.bool.must.push({ + match: { + [matchField]: filterValue + } + }) + } + return esQuery +} + +/** + * Build ES query from given filter. + * @param filter + * @returns {{}} created ES query object + */ +function buildEsQueryFromFilter (filter) { + const queryDoc = DOCUMENTS[filter.resource] + const esQuery = { + index: queryDoc.index, + type: queryDoc.type, + body: { + query: { + bool: { + must: [], + should: [], + minimum_should_match: 0 + } + } + } + } + + const matchField = `${filter.queryField}` + return setFilterValueToEsQuery(esQuery, matchField, filter.value, filter.queryField) +} + +/** + * Resolve filter by querying ES with filter data. + * @param filter the filter to query ES + * @param initialRes the resource that initial request comes in + * @returns {Promise<*>} the resolved value + */ +async function resolveResFilter (filter, initialRes) { + const filterChain = FILTER_CHAIN[filter.resource] + + // return the value if this is end of the filter + if (filter.resource === initialRes || !filterChain.filterNext) { + return { + resource: filter.resource, + queryField: filter.queryField, + value: filter.value + } + } + + // query ES with filter + const esQuery = buildEsQueryFromFilter(filter) + const { body: result } = await esClient.search(esQuery) + + const numHits = getTotalCount(result.hits.total) + + if (numHits > 0) { + // this value can be array + let value + if (numHits === 1) { + value = result.hits.hits[0]._source.id + } else { + value = result.hits.hits.map(hit => hit._source.id) + } + const nextFilter = { + resource: filterChain.filterNext, + queryField: filterChain.queryField, + value + } + // go to next filter + const resolved = await resolveResFilter(nextFilter, initialRes) + return resolved + } + throw new Error(`Resource filter[${filter.resource}.${filter.queryField}=${filter.value}] query returns no data`) +} + +/** + * Search ES with provided request parameters. + * @param resource the resource to search + * @param args the request path and query parameters + * @returns {Promise<*>} the promise of searched results + */ +async function searchElasticSearch (resource, ...args) { + logger.debug(`Searching ES first with query args: ${JSON.stringify(args[0], null, 2)}`) + // path and query parameters + const params = args[0] + const doc = DOCUMENTS[resource] + if (!params.page) { + params.page = 1 + } + if (!params.perPage) { + params.perPage = config.PAGE_SIZE + } + + let sortClause = [] + + const preResFilters = parseResourceFilter(resource, params, false) + const preResFilterResults = [] + // resolve pre resource filters + if (!params.enrich && preResFilters.length > 0) { + for (const filter of preResFilters) { + const resolved = await resolveResFilter(filter, resource) + preResFilterResults.push(resolved) + } + } + + if (params.orderBy) { + sortClause = [{ [`${params.orderBy}.keyword`]: 'asc' }] + } + + // construct ES query + const esQuery = { + index: doc.index, + type: doc.type, + size: params.perPage, + from: (params.page - 1) * params.perPage, // Es Index starts from 0 + body: { + query: { + bool: { + must: [], + should: [], + minimum_should_match: 0 + } + }, + sort: sortClause + } + } + + // set pre res filter results + if (!params.enrich && preResFilterResults.length > 0) { + for (const filter of preResFilterResults) { + const matchField = `${filter.queryField}` + setFilterValueToEsQuery(esQuery, matchField, filter.value, filter.queryField) + } + } + + const ownResFilters = parseResourceFilter(resource, params, true) + // set it's own res filter to the main query + if (!params.enrich && ownResFilters.length > 0) { + setResourceFilterToEsQuery(ownResFilters, esQuery) + } + + logger.debug(`ES query for search ${resource}: ${JSON.stringify(esQuery, null, 2)}`) + const { body: docs } = await esClient.search(esQuery) + if (docs.hits && getTotalCount(docs.hits.total) === 0) { + return { + total: docs.hits.total, + page: params.page, + perPage: params.perPage, + result: [] + } + } + + const result = docs.hits.hits.map(hit => hit._source) + + return { + total: docs.hits.total, + page: params.page, + perPage: params.perPage, + result + } +} + +module.exports = { + insertIntoES, + updateESRecord, + deleteESRecord, + searchElasticSearch, + getFromElasticSearch +} diff --git a/src/common/helper.js b/src/common/helper.js new file mode 100644 index 0000000..c197ee9 --- /dev/null +++ b/src/common/helper.js @@ -0,0 +1,98 @@ +const querystring = require('querystring') +const _ = require('lodash') +const { getControllerMethods } = require('./controller-helper') + +/** + * get auth user handle or id + * @param authUser the user + */ +function getAuthUser (authUser) { + return authUser.handle || authUser.sub +} + +/** + * Get link for a given page. + * @param {Object} req the HTTP request + * @param {Number} page the page number + * @returns {String} link for the page + */ +function getPageLink (req, page) { + const q = _.assignIn({}, req.query, { page }) + return `${req.protocol}://${req.get('Host')}${req.baseUrl}${req.path}?${querystring.stringify(q)}` +} + +/** + * Set HTTP response headers from result. + * @param {Object} req the HTTP request + * @param {Object} res the HTTP response + * @param {Object} result the operation result + */ +function injectSearchMeta (req, res, result) { + // if result is got from db, then do not set response headers + if (result.fromDB) { + return + } + + const resultTotal = _.isNumber(result.total) ? result.total : result.total.value + + const totalPages = Math.ceil(resultTotal / result.perPage) + if (result.page > 1) { + res.set('X-Prev-Page', +result.page - 1) + } + if (result.page < totalPages) { + res.set('X-Next-Page', +result.page + 1) + } + res.set('X-Page', result.page) + res.set('X-Per-Page', result.perPage) + res.set('X-Total', resultTotal) + res.set('X-Total-Pages', totalPages) + // set Link header + if (totalPages > 0) { + let link = `<${getPageLink(req, 1)}>; rel="first", <${getPageLink(req, totalPages)}>; rel="last"` + if (result.page > 1) { + link += `, <${getPageLink(req, result.page - 1)}>; rel="prev"` + } + if (result.page < totalPages) { + link += `, <${getPageLink(req, result.page + 1)}>; rel="next"` + } + res.set('Link', link) + } + + // Allow browsers access pagination data in headers + let accessControlExposeHeaders = res.get('Access-Control-Expose-Headers') || '' + accessControlExposeHeaders += accessControlExposeHeaders ? ', ' : '' + // append new values, to not override values set by someone else + accessControlExposeHeaders += 'X-Page, X-Per-Page, X-Total, X-Total-Pages, X-Prev-Page, X-Next-Page' + + res.set('Access-Control-Expose-Headers', accessControlExposeHeaders) +} + +/** + * Removes the audit fields created, createdBy, updatedBy from the given entity or an array of entities + * and moves the updated to metadata + * @param entity a single entity or an array of entities + * @returns the modified entity + */ +function omitAuditFields (entity) { + function _omit (result) { + const metadata = _.get(result, 'metadata') || {} + metadata.updated = _.get(result, 'updated') + result.metadata = metadata + return _.omit(result, ['created', 'updated', 'createdBy', 'updatedBy']) + } + + if (!entity) { return entity } + + if (_.isArray(entity)) { + return entity.map(i => _omit(i)) + } else { + return _omit(entity) + } +} + +module.exports = { + getAuthUser, + injectSearchMeta, + getControllerMethods, + omitAuditFields +} diff --git a/src/common/logger.js b/src/common/logger.js new file mode 100755 index 0000000..953f1c5 --- /dev/null +++ b/src/common/logger.js @@ -0,0 +1,152 @@ +/** + * This module contains the winston logger configuration. + */ +/* eslint-disable no-param-reassign, func-names */ + +const _ = require('lodash') +const winston = require('winston') +const util = require('util') +const config = require('config') +const getParams = require('get-parameter-names') +const joi = require('@hapi/joi') + +const { + combine, timestamp, colorize, align, printf +} = winston.format + +const basicFormat = printf(info => `${info.timestamp} ${info.level}: ${info.message}`) + +const transports = [] +if (!config.DISABLE_LOGGING) { + transports.push(new (winston.transports.Console)({ level: config.LOG_LEVEL })) +} + +const logger = winston.createLogger({ + transports, + format: combine( + colorize(), + align(), + timestamp(), + basicFormat + ) +}) + +/** + * Log error details with signature + * @param err the error + */ +logger.logFullError = function (err) { + if (!err) { + return + } + logger.error((err.signature ? (`${err.signature} : `) : '') + util.inspect(err)) + err.logged = true +} + +/** + * Remove invalid properties from the object and hide long arrays + * @param {Object} obj the object + * @returns {Object} the new object with removed properties + * @private + */ +function sanitizeObject (obj) { + const hideFields = ['auth'] + try { + return JSON.parse(JSON.stringify(obj, (k, v) => { + return _.includes(hideFields, k) ? '' : v + })) + } catch (e) { + return obj + } +} + +/** + * Convert array with arguments to object + * @param {Array} params the name of parameters + * @param {Array} arr the array with values + * @private + */ +function combineObject (params, arr) { + const ret = {} + _.each(arr, (arg, i) => { + ret[params[i]] = arg + }) + return ret +} + +/** + * Decorate all functions of a service and log debug information if DEBUG is enabled + * @param {string} serviceName the service name + * @param {Object} service the service + */ +logger.decorateWithLogging = (serviceName, service) => { + if (config.LOG_LEVEL !== 'debug') { + return + } + _.each(service, (method, name) => { + const params = method.params || getParams(method) + + service[name] = async (...args) => { + if (name.indexOf('export') === 0) { + return method(...args) + } + logger.debug(`ENTER Method '${serviceName}.${name}'`) + logger.debug(`##input arguments, ${util.inspect(sanitizeObject(combineObject(params, args)))}`) + try { + const result = await method(...args) + if (result !== null && result !== undefined) { + logger.debug(`##output arguments, ${util.inspect(sanitizeObject(result))}`) + } else { + logger.debug('##output arguments, No any result returned') + } + logger.debug(`EXIT Method '${serviceName}.${name}'`) + return result + } catch (e) { + e.signature = `${serviceName}.${name}` + throw e + } + } + }) +} + +/** + * Decorate all functions of a service and validate input values + * and replace input arguments with sanitized result form Joi + * Service method must have a `schema` property with Joi schema + * @param {string} serviceName the service name + * @param {Object} service the service + */ +logger.decorateWithValidators = (serviceName, service) => { + _.each(service, (method, name) => { + if (!method.schema) { + return + } + + const params = getParams(method) + service[name] = async function (...args) { + const value = combineObject(params, args) + const normalized = (method.schema) ? await joi.object(method.schema).validateAsync(value) : value + const newArgs = [] + // Joi will normalize values + // for example string number '1' to 1 + // if schema type is number + _.each(params, (param) => { + newArgs.push(normalized[param]) + }) + return method(...newArgs) + } + service[name].params = params + }) +} + +/** + * Apply logger and validation decorators + * @param {string} serviceName the service name + * @param {Object} service the service to wrap + */ +logger.buildService = (serviceName, service) => { + logger.decorateWithValidators(serviceName, service) + logger.decorateWithLogging(serviceName, service) +} + +module.exports = logger diff --git a/src/common/permission-helper.js b/src/common/permission-helper.js new file mode 100644 index 0000000..8674b74 --- /dev/null +++ b/src/common/permission-helper.js @@ -0,0 +1,133 @@ +/* + * Defined helper functions related to permissions. + */ + +const _ = require('lodash') + +/** + * Check if user match the permission rule. + * + * This method uses permission rule defined in `permissionRule` + * and checks that the `user` matches it. + * + * `permissionRule.topcoderRoles` may be equal to `true` which means user is a logged-in user + * + * @param {Object} permissionRule permission rule + * @param {Array|Boolean} permissionRule.topcoderRoles the list of Topcoder roles of the user + * @param {Object} user user for whom we check permissions + * @param {Object} user.roles list of user roles + * @param {Object} user.scopes scopes of user token + * + * @returns {Boolean} true, if has permission + */ +function matchPermissionRule (permissionRule, user) { + let hasTopcoderRole = false + let hasScope = false + + // if no rule defined, no access by default + if (!permissionRule) { + return false + } + + // check Topcoder Roles + if (permissionRule.topcoderRoles) { + // check if user has one of allowed Topcoder roles + if (permissionRule.topcoderRoles.length > 0) { + hasTopcoderRole = _.intersection( + _.get(user, 'roles', []).map(role => role.toLowerCase()), + permissionRule.topcoderRoles.map(role => role.toLowerCase()) + ).length > 0 + + // `topcoderRoles === true` means that we check if user is has any Topcoder role + // basically this equals to logged-in user, as all the Topcoder users + // have at least one role `Topcoder User` + } else if (permissionRule.topcoderRoles === true) { + hasTopcoderRole = _.get(user, 'roles', []).length > 0 + } + } + + // check M2M scopes + if (permissionRule.scopes) { + hasScope = _.intersection( + _.get(user, 'scopes', []), + permissionRule.scopes + ).length > 0 + } + + return hasTopcoderRole || hasScope +} + +/** + * Check if user has permission. + * + * This method uses permission defined in `permission` and checks that the `user` matches it. + * + * `permission` may be defined in two ways: + * - **Full** way with defined `allowRule` and optional `denyRule`, example: + * ```js + * { + * allowRule: { + * topcoderRoles: [] + * }, + * denyRule: { + * topcoderRoles: [] + * } + * } + * ``` + * If user matches `denyRule` then the access would be dined even if matches `allowRule`. + * - **Simplified** way may be used if we only want to define `allowRule`. + * We can skip the `allowRule` property and define `allowRule` directly inside `permission` object, example: + * ```js + * { + * topcoderRoles: [] + * } + * ``` + * This **simplified** permission is equal to a **full** permission: + * ```js + * { + * allowRule: { + * topcoderRoles: [] + * } + * } + * ``` + * + * @param {Object} permission permission or permissionRule + * @param {Object} user user for whom we check permissions + * @param {Object} user.roles list of user roles + * @param {Object} user.scopes scopes of user token + * + * @returns {Boolean} true, if has permission + */ +function hasPermission (permission, user) { + if (!permission) { + return false + } + + const allowRule = permission.allowRule ? permission.allowRule : permission + const denyRule = permission.denyRule ? permission.denyRule : null + + const allow = matchPermissionRule(allowRule, user) + const deny = matchPermissionRule(denyRule, user) + + return allow && !deny +} + +/* + * @param {Object} permission permission or permissionRule + * @param {Object} req the request object + * + * @returns {Boolean} true, if has permission + */ +function hasPermissionByReq (permission, req) { + // as it's very easy to forget "req" argument, throw error to make debugging easier + if (!req) { + throw new Error('Method "hasPermissionByReq" requires "req" argument.') + } + + return hasPermission(permission, _.get(req, 'authUser')) +} + +module.exports = { + hasPermissionByReq, + hasPermission +} diff --git a/src/common/service-helper.js b/src/common/service-helper.js new file mode 100644 index 0000000..0869d8a --- /dev/null +++ b/src/common/service-helper.js @@ -0,0 +1,131 @@ +const errors = require('./errors') +const esHelper = require('./es-helper') +const logger = require('./logger') +const permissionHelper = require('./permission-helper') + +// map model name to bus message resource if different +const MODEL_TO_RESOURCE = { + Skill: 'skill', + Taxonomy: 'taxonomy' +} + +/** + * Create record in es + * @param resource the resource to create + * @param result the resource fields + */ +async function createRecordInEs (resource, entity) { + try { + await esHelper.insertIntoES(resource, entity) + } catch (err) { + logger.logFullError(err) + } +} + +/** + * Patch record in es + * @param resource the resource to create + * @param result the resource fields + */ +async function patchRecordInEs (resource, entity) { + try { + await esHelper.updateESRecord(resource, entity) + } catch (err) { + logger.logFullError(err) + } +} + +/** + * post delete message to es + * @param id the id of record + * @param params the params of record (like nested ids) + * @param resource the resource to delete + */ +async function deleteRecordFromEs (id, params, resource) { + try { + await esHelper.deleteESRecord(resource, id) + } catch (err) { + logger.logFullError(err) + } +} + +/** + * get resource in es + * @param resource resource to get + * @param id resource id + * @param params resource params + */ +async function getRecordInEs (resource, id, params) { + // Merge path and query params + try { + const result = await esHelper.getFromElasticSearch(resource, id, params) + return result + } catch (err) { + // return error if enrich fails or permission fails + if (err.status && err.status === 403) { + throw errors.elasticSearchEnrichError(err.message) + } + logger.logFullError(err) + } +} + +/** + * search resource in es + * @param resource the resource to delete + * @param query the search query + */ +async function searchRecordInEs (resource, query) { + // remove dollar signs that are used for postgres + for (const key of Object.keys(query)) { + if (key[0] === '$' && key[key.length - 1] === '$') { + query[key.slice(1, -1)] = query[key] + delete query[key] + } + } + + try { + return await esHelper.searchElasticSearch(resource, query) + } catch (err) { + logger.logFullError(err) + } + return null +} + +/** + * Get the resource from model name + * @param modelName the model name + * @returns {string|*} the resource + */ +function getResource (modelName) { + if (MODEL_TO_RESOURCE[modelName]) { + return MODEL_TO_RESOURCE[modelName] + } else { + return modelName.toLowerCase() + } +} + +/** + * Check if a user has specific permissions. + * + * @param {Object} permission permission or permissionRule + * @param {Object} user user for whom we check permissions + * @param {Object} user.roles list of user roles + * @param {Object} user.scopes scopes of user token + * @returns {undefined} + */ +function hasPermission (permission, authUser) { + if (permissionHelper.hasPermission(permission, authUser)) { + return + } + throw errors.ForbiddenError('You do not have permissions to perform this action') +} + +module.exports = { + getResource, + patchRecordInEs, + deleteRecordFromEs, + searchRecordInEs, + createRecordInEs, + getRecordInEs, + hasPermission +} diff --git a/src/constants.js b/src/constants.js new file mode 100644 index 0000000..c347926 --- /dev/null +++ b/src/constants.js @@ -0,0 +1,49 @@ +/* + * Constants(roles, access scopes, etc.). + */ + +const USER_ROLE = { + TOPCODER_ADMIN: 'administrator', + MANAGER: 'Connect Manager', + TOPCODER_ACCOUNT_MANAGER: 'Connect Account Manager', + COPILOT: 'Connect Copilot', + CONNECT_ADMIN: 'Connect Admin', + COPILOT_MANAGER: 'Connect Copilot Manager', + BUSINESS_DEVELOPMENT_REPRESENTATIVE: 'Business Development Representative', + PRESALES: 'Presales', + ACCOUNT_EXECUTIVE: 'Account Executive', + PROGRAM_MANAGER: 'Program Manager', + SOLUTION_ARCHITECT: 'Solution Architect', + PROJECT_MANAGER: 'Project Manager', + TOPCODER_USER: 'Topcoder User' +} + +const ADMIN_ROLES = [USER_ROLE.CONNECT_ADMIN, USER_ROLE.TOPCODER_ADMIN] + +const MANAGER_ROLES = [ + ...ADMIN_ROLES, + USER_ROLE.MANAGER, + USER_ROLE.TOPCODER_ACCOUNT_MANAGER, + USER_ROLE.COPILOT_MANAGER, + USER_ROLE.BUSINESS_DEVELOPMENT_REPRESENTATIVE, + USER_ROLE.PRESALES, + USER_ROLE.ACCOUNT_EXECUTIVE, + + USER_ROLE.PROGRAM_MANAGER, + USER_ROLE.SOLUTION_ARCHITECT, + USER_ROLE.PROJECT_MANAGER +] + +const M2M_SCOPES = { + // for backward compatibility we should allow ALL M2M operations with `CONNECT_PROJECT_ADMIN` + CONNECT_PROJECT_ADMIN: 'all:connect_project', + PROJECTS: { + ALL: 'all:projects', + WRITE: 'write:projects' + } +} + +module.exports = { + MANAGER_ROLES, + M2M_SCOPES +} diff --git a/src/models/Skill.js b/src/models/Skill.js new file mode 100644 index 0000000..a942804 --- /dev/null +++ b/src/models/Skill.js @@ -0,0 +1,38 @@ +const { DataTypes } = require('sequelize') + +module.exports = (sequelize) => { + const Skill = sequelize.define('Skill', { + id: { + primaryKey: true, + type: DataTypes.UUID, + defaultValue: DataTypes.UUIDV4 + }, + createdBy: { + type: DataTypes.STRING + }, + updatedBy: { + type: DataTypes.STRING + }, + name: { + type: DataTypes.STRING + }, + externalId: { + type: DataTypes.STRING + }, + uri: { + type: DataTypes.STRING + }, + metadata: { + type: DataTypes.JSONB + } + }, + { + timestamps: true, + updatedAt: 'updated', + createdAt: 'created' + }) + Skill.associate = (models) => { + Skill.belongsTo(models.Taxonomy, { foreignKey: 'taxonomyId', type: DataTypes.UUID }) + } + return Skill +} diff --git a/src/models/Taxonomy.js b/src/models/Taxonomy.js new file mode 100644 index 0000000..2088eae --- /dev/null +++ b/src/models/Taxonomy.js @@ -0,0 +1,32 @@ +const { DataTypes } = require('sequelize') + +module.exports = (sequelize) => { + const Taxonomy = sequelize.define('Taxonomy', { + id: { + primaryKey: true, + type: DataTypes.UUID, + defaultValue: DataTypes.UUIDV4 + }, + createdBy: { + type: DataTypes.STRING + }, + updatedBy: { + type: DataTypes.STRING + }, + name: { + type: DataTypes.STRING + }, + metadata: { + type: DataTypes.JSONB + } + }, + { + timestamps: true, + updatedAt: 'updated', + createdAt: 'created' + }) + Taxonomy.associate = (models) => { + Taxonomy.hasMany(models.Skill, { foreignKey: 'taxonomyId', type: DataTypes.UUID }) + } + return Taxonomy +} diff --git a/src/models/index.js b/src/models/index.js new file mode 100755 index 0000000..4bf6b21 --- /dev/null +++ b/src/models/index.js @@ -0,0 +1,43 @@ +/** + * the model index + */ +const { Sequelize } = require('sequelize') +const config = require('config') +const fs = require('fs') +const path = require('path') +const logger = require('../common/logger') + +/** + * the database instance + */ +const sequelize = new Sequelize( + `postgresql://${config.DB_USERNAME}:${config.DB_PASSWORD}@${config.DB_HOST}:${config.DB_PORT}/${config.DB_NAME}` +) + +const db = {} + +fs + .readdirSync(__dirname) + .filter(file => (file.indexOf('.') !== 0) && (file !== 'index.js')) + .forEach((file) => { + const model = require(path.join(__dirname, file))(sequelize) + db[model.name] = model + }) + +Object.keys(db).forEach((modelName) => { + if ('associate' in db[modelName]) { + db[modelName].associate(db) + } +}) + +module.exports = sequelize + +/** + * create table + */ +module.exports.init = async () => { + logger.info('connect to database, check/create tables ...') + // authenticate db + await sequelize.authenticate() + logger.info(`Connected to db ${config.DB_NAME} successfully`) +} diff --git a/src/modules/SkillMetadata/controller.js b/src/modules/SkillMetadata/controller.js new file mode 100644 index 0000000..b9db49f --- /dev/null +++ b/src/modules/SkillMetadata/controller.js @@ -0,0 +1,38 @@ +/** + * the skill sub-controller, specific to the metadata fields + */ + +const service = require('./service') + +/** + * Fully update skill metadata. + * @param req the http request + * @param res the http response + */ +async function fullyUpdate (req, res) { + res.json(await service.fullyUpdate(req.params.id, req.body, req.authUser)) +} + +/** + * Partically update skill metadata. + * @param req the http request + * @param res the http response + */ +async function particallyUpdate (req, res) { + res.json(await service.particallyUpdate(req.params.id, req.body, req.authUser)) +} + +/** + * Remove one or more fields from skill metadata. + * @param req the http request + * @param res the http response + */ +async function remove (req, res) { + res.json(await service.remove(req.params.id, req.body, req.authUser)) +} + +module.exports = { + fullyUpdate, + particallyUpdate, + remove +} diff --git a/src/modules/SkillMetadata/route.js b/src/modules/SkillMetadata/route.js new file mode 100644 index 0000000..a3df454 --- /dev/null +++ b/src/modules/SkillMetadata/route.js @@ -0,0 +1,23 @@ +/** + * the skill sub-routes, specific to the metadata fields + */ + +const Controller = require('./controller') + +module.exports = { + '/skills/:id/metadata': { + put: { + method: Controller.fullyUpdate, + auth: 'jwt' + }, + patch: { + method: Controller.particallyUpdate, + auth: 'jwt' + }, + delete: { + method: Controller.remove, + auth: 'jwt', + permission: 'skill.deleteMetadata' + } + } +} diff --git a/src/modules/SkillMetadata/service.js b/src/modules/SkillMetadata/service.js new file mode 100644 index 0000000..5c9e6a9 --- /dev/null +++ b/src/modules/SkillMetadata/service.js @@ -0,0 +1,132 @@ +/** + * the skill sub-services, specific to the metadata fields + */ + +const joi = require('@hapi/joi') +const _ = require('lodash') + +const errors = require('../../common/errors') +const helper = require('../../common/helper') +const dbHelper = require('../../common/db-helper') +const serviceHelper = require('../../common/service-helper') +const { PERMISSION } = require('../../permissions/constants') +const sequelize = require('../../models/index') + +const Skill = sequelize.models.Skill +const Taxonomy = sequelize.models.Taxonomy +const resource = serviceHelper.getResource('Skill') + +/** + * update skill metadata + * @param instance the skill instance + * @param metadata the new metadata + * @param auth the auth object + * @return the updated skill + */ +async function updateMetaData (instance, metadata, auth) { + const newEntity = await instance.update({ + ...instance.dataValues, + updatedBy: helper.getAuthUser(auth), + metadata + }) + const taxonomy = await dbHelper.get(Taxonomy, newEntity.taxonomyId) + await serviceHelper.patchRecordInEs(resource, { ...newEntity.dataValues, taxonomyName: taxonomy.name }) + return helper.omitAuditFields(newEntity.dataValues) +} + +/** + * Fully update skill metadata + * @param id the skill id + * @param entity the request skill metadata + * @param auth the auth object + * @return the updated skill + */ +async function fullyUpdate (id, entity, auth) { + const instance = await dbHelper.get(Skill, id) + + if (Object.keys(entity).length) { + // check permission for adding new fields + serviceHelper.hasPermission(PERMISSION.ADD_SKILL_METADATA, auth) + } + if (Object.keys(instance.metadata).length) { + // check permission for removing existing fields + serviceHelper.hasPermission(PERMISSION.DELETE_SKILL_METADATA, auth) + } + + return updateMetaData(instance, entity, auth) +} + +fullyUpdate.schema = { + id: joi.string().uuid().required(), + entity: joi.object().keys({ + challengeProminence: joi.prominence('challengeProminence'), + memberProminence: joi.prominence('memberProminence') + }).unknown(true).required(), + auth: joi.object() +} + +/** + * Partically update skill metadata + * @param id the skill id + * @param entity the request skill metadata + * @param auth the auth object + * @return the updated skill + */ +async function particallyUpdate (id, entity, auth) { + const instance = await dbHelper.get(Skill, id) + + const inputFields = Object.keys(entity) + const existingFields = Object.keys(instance.metadata) + const sharedFields = _.intersection(inputFields, existingFields) + + if (inputFields.length > sharedFields.length) { + // check permission for adding new fields + serviceHelper.hasPermission(PERMISSION.ADD_SKILL_METADATA, auth) + } + if (sharedFields.length) { + // check permission for updating fields + serviceHelper.hasPermission(PERMISSION.UPDATE_SKILL_METADATA, auth) + } + + return updateMetaData(instance, { ...instance.dataValues.metadata, ...entity }, auth) +} + +particallyUpdate.schema = { + id: joi.string().uuid().required(), + entity: joi.object().keys({ + challengeProminence: joi.prominence('challengeProminence'), + memberProminence: joi.prominence('memberProminence') + }).min(1).unknown(true).required(), + auth: joi.object() +} + +/** + * Remove one or more fields from skill metadata + * @param id the skill id + * @param fields the list of field names + * @param auth the auth object + * @return the updated skill + */ +async function remove (id, fields, auth) { + const instance = await dbHelper.get(Skill, id) + + const existingFields = Object.keys(instance.metadata) + const nonExistingFields = fields.filter(field => !existingFields.includes(field)) + if (nonExistingFields.length) { + throw errors.NotFoundError(`Metadata fields: ${nonExistingFields} do not exist`) + } + + return updateMetaData(instance, _.omit(instance.dataValues.metadata, fields), auth) +} + +remove.schema = { + id: joi.string().uuid().required(), + fields: joi.array().min(1).items(joi.string()).required(), + auth: joi.object() +} + +module.exports = { + fullyUpdate, + particallyUpdate, + remove +} diff --git a/src/modules/skill/controller.js b/src/modules/skill/controller.js new file mode 100644 index 0000000..e42c58f --- /dev/null +++ b/src/modules/skill/controller.js @@ -0,0 +1,11 @@ +/** + * the skill controller + */ + +const service = require('./service') +const helper = require('../../common/helper') +const methods = helper.getControllerMethods(service) + +module.exports = { + ...methods +} diff --git a/src/modules/skill/route.js b/src/modules/skill/route.js new file mode 100644 index 0000000..bd29bcb --- /dev/null +++ b/src/modules/skill/route.js @@ -0,0 +1,39 @@ +/** + * the skill routes + */ + +const Controller = require('./controller') + +module.exports = { + '/skills': { + get: { + method: Controller.search + }, + post: { + method: Controller.create, + auth: 'jwt', + permission: 'skill.create' + }, + head: { + method: Controller.search + } + }, + '/skills/:id': { + get: { + method: Controller.get + }, + head: { + method: Controller.get + }, + patch: { + method: Controller.patch, + auth: 'jwt', + permission: 'skill.edit' + }, + delete: { + method: Controller.remove, + auth: 'jwt', + permission: 'skill.delete' + } + } +} diff --git a/src/modules/skill/service.js b/src/modules/skill/service.js new file mode 100644 index 0000000..aa024d7 --- /dev/null +++ b/src/modules/skill/service.js @@ -0,0 +1,234 @@ +/** + * the skill services + */ + +const joi = require('@hapi/joi') +const _ = require('lodash') + +const errors = require('../../common/errors') +const helper = require('../../common/helper') +const dbHelper = require('../../common/db-helper') +const serviceHelper = require('../../common/service-helper') +const { PERMISSION } = require('../../permissions/constants') +const sequelize = require('../../models/index') + +const Skill = sequelize.models.Skill +const Taxonomy = sequelize.models.Taxonomy +const resource = serviceHelper.getResource('Skill') +const uniqueFields = [['taxonomyId', 'externalId', 'name']] + +/** + * create entity + * @param entity the request skill entity + * @param auth the auth information + * @return the created skill + */ +async function create (entity, auth) { + if (Object.keys(entity.metadata).length) { + // check permission for adding new metadata fields + serviceHelper.hasPermission(PERMISSION.ADD_SKILL_METADATA, auth) + } + + const taxonomy = await dbHelper.get(Taxonomy, entity.taxonomyId) + await dbHelper.makeSureUnique(Skill, entity, uniqueFields) + + const result = await dbHelper.create(Skill, entity, auth) + const created = result.dataValues + created.taxonomyName = taxonomy.name + await serviceHelper.createRecordInEs(resource, created) + + return helper.omitAuditFields(created) +} + +create.schema = { + entity: joi.object().keys({ + taxonomyId: joi.string().uuid().required(), + name: joi.string().required(), + uri: joi.string(), + externalId: joi.string(), + metadata: joi.object().keys({ + challengeProminence: joi.prominence('challengeProminence'), + memberProminence: joi.prominence('memberProminence') + }).unknown(true).required() + }).required(), + auth: joi.object() +} + +/** + * patch skill by id + * @param id the skill id + * @param entity the request skill entity + * @param auth the auth object + * @return the updated skill + */ +async function patch (id, entity, auth) { + let taxonomy + if (entity.taxonomyId) { + taxonomy = await dbHelper.get(Taxonomy, entity.taxonomyId) + } + + await dbHelper.makeSureUnique(Skill, entity, uniqueFields) + + const instance = await dbHelper.get(Skill, id) + + if (entity.metadata) { + if (Object.keys(entity.metadata).length) { + // check permission for adding new metadata fields + serviceHelper.hasPermission(PERMISSION.ADD_SKILL_METADATA, auth) + } + if (Object.keys(instance.metadata).length) { + // check permission for removing existing metadata fields + serviceHelper.hasPermission(PERMISSION.DELETE_SKILL_METADATA, auth) + } + } + + const newEntity = await instance.update({ + ...entity, + updatedBy: helper.getAuthUser(auth) + }) + + if (!taxonomy) { + taxonomy = await dbHelper.get(Taxonomy, newEntity.taxonomyId) + } + const updated = newEntity.dataValues + updated.taxonomyName = taxonomy.name + await serviceHelper.patchRecordInEs(resource, updated) + + return helper.omitAuditFields(updated) +} + +patch.schema = { + id: joi.string().uuid().required(), + entity: joi.object().keys({ + taxonomyId: joi.string().uuid(), + name: joi.string(), + uri: joi.string(), + externalId: joi.string(), + metadata: joi.object().keys({ + challengeProminence: joi.prominence('challengeProminence'), + memberProminence: joi.prominence('memberProminence') + }).unknown(true) + }).min(1).required(), + auth: joi.object() +} + +/** + * get skill by id + * @param id the skill id + * @param params the path parameters + * @param query the query parameters + * @param fromDb Should we bypass Elasticsearch for the record and fetch from db instead? + * @return the skill + */ +async function get (id, params, query = {}, fromDb = false) { + const trueParams = _.assign(params, query) + if (!fromDb) { + const esResult = await serviceHelper.getRecordInEs(resource, id, trueParams) + await populateTaxonomyNames(esResult) + if (esResult) { + return helper.omitAuditFields(esResult) + } + } + + const recordObj = await dbHelper.get(Skill, id) + if (!recordObj) { + throw errors.newEntityNotFoundError(`cannot find ${Skill.name} where ${_.map(trueParams, (v, k) => `${k}:${v}`).join(', ')}`) + } + const skill = recordObj.dataValues + await populateTaxonomyNames(skill) + + return helper.omitAuditFields(skill) +} + +get.schema = { + id: joi.string().uuid().required(), + params: joi.object() +} + +/** + * Populates the taxonomy name for each of the skill + * @param skills individual skill or an array of skills + * @returns the updated skills object + */ +async function populateTaxonomyNames (skills) { + if (_.isArray(skills)) { + const taxonomyMap = {} + for (const skill of skills) { + // dont populate if we already have the name + if (skill.taxonomyName) { continue } + + if (!_.has(taxonomyMap, skill.taxonomyId)) { + const taxonomy = await dbHelper.get(Taxonomy, skill.taxonomyId) + taxonomyMap[skill.taxonomyId] = taxonomy.name + } + skill.taxonomyName = taxonomyMap[skill.taxonomyId] + } + } else { + const taxonomy = await dbHelper.get(Taxonomy, skills.taxonomyId) + skills.taxonomyName = taxonomy.name + } + + return skills +} + +/** + * search skills by query + * @param query the search query + * @return the results + */ +async function search (query) { + // get from elasticsearch, if that fails get from db + // and response headers ('X-Total', 'X-Page', etc.) are not set in case of db return + + const esResult = await serviceHelper.searchRecordInEs(resource, query) + + if (esResult) { + await populateTaxonomyNames(esResult.result) + esResult.result = helper.omitAuditFields(esResult.result) + return helper.omitAuditFields(esResult) + } + + let items = await dbHelper.find(Skill, query) + + items = items.map(item => item.dataValues) + await populateTaxonomyNames(items) + items = helper.omitAuditFields(items) + return { fromDb: true, result: items, total: items.length } +} + +search.schema = { + query: { + page: joi.string().uuid(), + perPage: joi.pageSize(), + taxonomyId: joi.string().uuid(), + name: joi.string(), + externalId: joi.string(), + orderBy: joi.string() + } +} + +/** + * remove skill by id + * @param id the skill id + * @param auth the auth object + * @param params the query params + * @return no data returned + */ +async function remove (id, auth, params) { + await dbHelper.remove(Skill, id) + await serviceHelper.deleteRecordFromEs(id, params, resource) +} + +remove.schema = { + id: joi.string().uuid().required(), + auth: joi.object(), + params: joi.object() +} + +module.exports = { + create, + search, + patch, + get, + remove +} diff --git a/src/modules/taxonomy/controller.js b/src/modules/taxonomy/controller.js new file mode 100644 index 0000000..de92a82 --- /dev/null +++ b/src/modules/taxonomy/controller.js @@ -0,0 +1,11 @@ +/** + * the taxonomy controller + */ + +const service = require('./service') +const helper = require('../../common/helper') +const methods = helper.getControllerMethods(service) + +module.exports = { + ...methods +} diff --git a/src/modules/taxonomy/route.js b/src/modules/taxonomy/route.js new file mode 100644 index 0000000..1586fa4 --- /dev/null +++ b/src/modules/taxonomy/route.js @@ -0,0 +1,39 @@ +/** + * the taxonomy routes + */ + +const Controller = require('./controller') + +module.exports = { + '/taxonomies': { + get: { + method: Controller.search + }, + post: { + method: Controller.create, + auth: 'jwt', + permission: 'taxonomy.create' + }, + head: { + method: Controller.search + } + }, + '/taxonomies/:id': { + get: { + method: Controller.get + }, + head: { + method: Controller.get + }, + patch: { + method: Controller.patch, + auth: 'jwt', + permission: 'taxonomy.edit' + }, + delete: { + method: Controller.remove, + auth: 'jwt', + permission: 'taxonomy.delete' + } + } +} diff --git a/src/modules/taxonomy/service.js b/src/modules/taxonomy/service.js new file mode 100644 index 0000000..f4f06f7 --- /dev/null +++ b/src/modules/taxonomy/service.js @@ -0,0 +1,173 @@ +/** + * the taxonomy services + */ + +const joi = require('@hapi/joi') +const _ = require('lodash') + +const errors = require('../../common/errors') +const helper = require('../../common/helper') +const dbHelper = require('../../common/db-helper') +const serviceHelper = require('../../common/service-helper') +const { PERMISSION } = require('../../permissions/constants') +const sequelize = require('../../models/index') + +const Taxonomy = sequelize.models.Taxonomy +const Skill = sequelize.models.Skill +const resource = serviceHelper.getResource('Taxonomy') + +/** + * create entity + * @param entity the request taxonomy entity + * @param auth the auth information + * @return the created taxonomy + */ +async function create (entity, auth) { + if (Object.keys(entity.metadata).length) { + // check permission for adding new metadata fields + serviceHelper.hasPermission(PERMISSION.ADD_TAXONOMY_METADATA, auth) + } + + const result = await dbHelper.create(Taxonomy, entity, auth) + + await serviceHelper.createRecordInEs(resource, result.dataValues) + return helper.omitAuditFields(result.dataValues) +} + +create.schema = { + entity: joi.object().keys({ + name: joi.string().required(), + metadata: joi.object().default({}) + }).required(), + auth: joi.object() +} + +/** + * patch taxonomy by id + * @param id the taxonomy id + * @param entity the request taxonomy entity + * @param auth the auth object + * @param params the query params + * @return the updated taxonomy + */ +async function patch (id, entity, auth) { + const instance = await dbHelper.get(Taxonomy, id) + + if (entity.metadata) { + if (Object.keys(entity.metadata).length) { + // check permission for adding new metadata fields + serviceHelper.hasPermission(PERMISSION.ADD_TAXONOMY_METADATA, auth) + } + if (Object.keys(instance.metadata).length) { + // check permission for removing existing metadata fields + serviceHelper.hasPermission(PERMISSION.DELETE_TAXONOMY_METADATA, auth) + } + } + + const newEntity = await instance.update({ + ...entity, + updatedBy: helper.getAuthUser(auth) + }) + + await serviceHelper.patchRecordInEs(resource, newEntity.dataValues) + return helper.omitAuditFields(newEntity.dataValues) +} + +patch.schema = { + id: joi.string().uuid().required(), + entity: joi.object().keys({ + name: joi.string(), + metadata: joi.object() + }).min(1).required(), + auth: joi.object() +} + +/** + * get taxonomy by id + * @param id the taxonomy id + * @param params the path parameters + * @param query the query parameters + * @param fromDb Should we bypass Elasticsearch for the record and fetch from db instead? + * @return the db taxonomy + */ +async function get (id, params, query = {}, fromDb = false) { + const trueParams = _.assign(params, query) + if (!fromDb) { + const esResult = await serviceHelper.getRecordInEs(resource, id, trueParams) + if (esResult) { + return helper.omitAuditFields(esResult) + } + } + + const recordObj = await dbHelper.get(Taxonomy, id) + if (!recordObj) { + throw errors.newEntityNotFoundError(`cannot find ${Taxonomy.name} where ${_.map(trueParams, (v, k) => `${k}:${v}`).join(', ')}`) + } + + return helper.omitAuditFields(recordObj.dataValues) +} + +get.schema = { + id: joi.string().uuid().required(), + params: joi.object(), + query: joi.object(), + fromDb: joi.boolean() +} + +/** + * search taxonomies by query + * @param query the search query + * @return the results + */ +async function search (query) { + // get from elasticsearch, if that fails get from db + // and response headers ('X-Total', 'X-Page', etc.) are not set in case of db return + const esResult = await serviceHelper.searchRecordInEs(resource, query) + if (esResult) { + esResult.result = helper.omitAuditFields(esResult.result) + return esResult + } + + let items = await dbHelper.find(Taxonomy, query) + items = helper.omitAuditFields(items.map(i => i.dataValues)) + return { fromDb: true, result: items, total: items.length } +} + +search.schema = { + query: { + page: joi.string().uuid(), + perPage: joi.pageSize(), + name: joi.string() + } +} + +/** + * remove entity by id + * @param id the entity id + * @param auth the auth object + * @param params the query params + * @return no data returned + */ +async function remove (id, auth, params) { + const existing = await dbHelper.find(Skill, { taxonomyId: id }) + if (existing.length > 0) { + throw errors.deleteConflictError(`Please delete ${Skill.name} with ids ${existing.map(o => o.id)}`) + } + + await dbHelper.remove(Taxonomy, id) + await serviceHelper.deleteRecordFromEs(id, params, resource) +} + +remove.schema = { + id: joi.string().uuid().required(), + auth: joi.object(), + params: joi.object() +} + +module.exports = { + create, + search, + patch, + get, + remove +} diff --git a/src/modules/taxonomyMetadata/controller.js b/src/modules/taxonomyMetadata/controller.js new file mode 100644 index 0000000..e37a5db --- /dev/null +++ b/src/modules/taxonomyMetadata/controller.js @@ -0,0 +1,38 @@ +/** + * the taxonomy sub-controller, specific to the metadata fields + */ + +const service = require('./service') + +/** + * Fully update taxonomy metadata. + * @param req the http request + * @param res the http response + */ +async function fullyUpdate (req, res) { + res.json(await service.fullyUpdate(req.params.id, req.body, req.authUser)) +} + +/** + * Partically update taxonomy metadata. + * @param req the http request + * @param res the http response + */ +async function particallyUpdate (req, res) { + res.json(await service.particallyUpdate(req.params.id, req.body, req.authUser)) +} + +/** + * Remove one or more fields from taxonomy metadata. + * @param req the http request + * @param res the http response + */ +async function remove (req, res) { + res.json(await service.remove(req.params.id, req.body, req.authUser)) +} + +module.exports = { + fullyUpdate, + particallyUpdate, + remove +} diff --git a/src/modules/taxonomyMetadata/route.js b/src/modules/taxonomyMetadata/route.js new file mode 100644 index 0000000..ba53e28 --- /dev/null +++ b/src/modules/taxonomyMetadata/route.js @@ -0,0 +1,23 @@ +/** + * the taxonomy sub-routes, specific to the metadata fields + */ + +const Controller = require('./controller') + +module.exports = { + '/taxonomies/:id/metadata': { + put: { + method: Controller.fullyUpdate, + auth: 'jwt' + }, + patch: { + method: Controller.particallyUpdate, + auth: 'jwt' + }, + delete: { + method: Controller.remove, + auth: 'jwt', + permission: 'taxonomy.deleteMetadata' + } + } +} diff --git a/src/modules/taxonomyMetadata/service.js b/src/modules/taxonomyMetadata/service.js new file mode 100644 index 0000000..2395d8d --- /dev/null +++ b/src/modules/taxonomyMetadata/service.js @@ -0,0 +1,124 @@ +/** + * the taxonomy sub-services, specific to the metadata fields + */ + +const joi = require('@hapi/joi') +const _ = require('lodash') + +const errors = require('../../common/errors') +const helper = require('../../common/helper') +const dbHelper = require('../../common/db-helper') +const serviceHelper = require('../../common/service-helper') +const { PERMISSION } = require('../../permissions/constants') +const sequelize = require('../../models/index') + +const Taxonomy = sequelize.models.Taxonomy +const resource = serviceHelper.getResource('Taxonomy') + +/** + * update taxonomy metadata + * @param instance the taxonomy instance + * @param metadata the new metadata + * @param auth the auth object + * @return the updated taxonomy + */ +async function updateMetaData (instance, metadata, auth) { + const newEntity = await instance.update({ + ...instance.dataValues, + updatedBy: helper.getAuthUser(auth), + metadata + }) + await serviceHelper.patchRecordInEs(resource, newEntity.dataValues) + return helper.omitAuditFields(newEntity.dataValues) +} + +/** + * Fully update taxonomy metadata + * @param id the taxonomy id + * @param entity the request taxonomy metadata + * @param auth the auth object + * @return the updated taxonomy + */ +async function fullyUpdate (id, entity, auth) { + const instance = await dbHelper.get(Taxonomy, id) + + if (Object.keys(entity).length) { + // check permission for adding new fields + serviceHelper.hasPermission(PERMISSION.ADD_TAXONOMY_METADATA, auth) + } + if (Object.keys(instance.metadata).length) { + // check permission for removing existing fields + serviceHelper.hasPermission(PERMISSION.DELETE_TAXONOMY_METADATA, auth) + } + + return updateMetaData(instance, entity, auth) +} + +fullyUpdate.schema = { + id: joi.string().uuid().required(), + entity: joi.object().required(), + auth: joi.object() +} + +/** + * Partically update taxonomy metadata + * @param id the taxonomy id + * @param entity the request taxonomy metadata + * @param auth the auth object + * @return the updated taxonomy + */ +async function particallyUpdate (id, entity, auth) { + const instance = await dbHelper.get(Taxonomy, id) + + const inputFields = Object.keys(entity) + const existingFields = Object.keys(instance.metadata) + const sharedFields = _.intersection(inputFields, existingFields) + + if (inputFields.length > sharedFields.length) { + // check permission for adding new fields + serviceHelper.hasPermission(PERMISSION.ADD_TAXONOMY_METADATA, auth) + } + if (sharedFields.length) { + // check permission for updating fields + serviceHelper.hasPermission(PERMISSION.UPDATE_TAXONOMY_METADATA, auth) + } + + return updateMetaData(instance, { ...instance.dataValues.metadata, ...entity }, auth) +} + +particallyUpdate.schema = { + id: joi.string().uuid().required(), + entity: joi.object().min(1).required(), + auth: joi.object() +} + +/** + * Remove one or more fields from taxonomy metadata + * @param id the taxonomy id + * @param fields the list of field names + * @param auth the auth object + * @return the updated taxonomy + */ +async function remove (id, fields, auth) { + const instance = await dbHelper.get(Taxonomy, id) + + const existingFields = Object.keys(instance.metadata) + const nonExistingFields = fields.filter(field => !existingFields.includes(field)) + if (nonExistingFields.length) { + throw errors.NotFoundError(`Metadata fields: ${nonExistingFields} do not exist`) + } + + return updateMetaData(instance, _.omit(instance.dataValues.metadata, fields), auth) +} + +remove.schema = { + id: joi.string().uuid().required(), + fields: joi.array().min(1).items(joi.string()).required(), + auth: joi.object() +} + +module.exports = { + fullyUpdate, + particallyUpdate, + remove +} diff --git a/src/permissions/constants.js b/src/permissions/constants.js new file mode 100644 index 0000000..48a3fd9 --- /dev/null +++ b/src/permissions/constants.js @@ -0,0 +1,195 @@ +/** + * User permission policies. + * Can be used with `hasPermission` method. + * + * PERMISSION GUIDELINES + * + * All the permission name and meaning should define **WHAT** can be done having such permission + * but not **WHO** can do it. + * + * Examples of CORRECT permission naming and meaning: + * - `READ_PROJECT` + * - `UPDATE_MILESTONE` + * - `DELETE_WORK` + * + * Examples of INCORRECT permissions naming and meaning: + * - `COPILOT_AND_MANAGER` + * - `PROJECT_MEMBERS` + * - `ADMINS` + * + * The same time **internally only** in this file, constants like `COPILOT_AND_ABOVE`, + * `PROJECT_MEMBERS`, `ADMINS` could be used to define permissions. + * + * NAMING GUIDELINES + * + * There are unified prefixes to indicate what kind of permissions. + * If no prefix is suitable, please, feel free to use a new prefix. + * + * CREATE_ - create somethings + * READ_ - read something + * UPDATE_ - update something + * DELETE_ - delete something + * + * MANAGE_ - means combination of 3 operations CREATE/UPDATE/DELETE. + * usually should be used, when READ operation is allowed to everyone + * while 3 manage operations require additional permissions + * ACCESS_ - means combination of all 4 operations READ/CREATE/UPDATE/DELETE. + * usually should be used, when by default users cannot even READ something + * and if someone can READ, then also can do other kind of operations. + * + * ANTI-PERMISSIONS + * + * If it's technically impossible to create permission rules for some situation in "allowed" manner, + * in such case we can create permission rules, which would disallow somethings. + * - Create such rules ONLY IF CREATING ALLOW RULE IS IMPOSSIBLE. + * - Add a comment to such rules explaining why allow-rule cannot be created. + */ + +const { + MANAGER_ROLES: TOPCODER_ROLES_MANAGERS_AND_ADMINS, + M2M_SCOPES +} = require('../constants') + +/** + * M2M scopes to "write" projects + */ +const SCOPES_PROJECTS_WRITE = [ + M2M_SCOPES.CONNECT_PROJECT_ADMIN, + M2M_SCOPES.PROJECTS.ALL, + M2M_SCOPES.PROJECTS.WRITE +] + +/** + * The full list of possible permission rules + */ +const PERMISSION = { + /* + * Skill + */ + CREATE_SKILL: { + meta: { + title: 'Create Skill', + group: 'Skill' + }, + topcoderRoles: TOPCODER_ROLES_MANAGERS_AND_ADMINS, + scopes: SCOPES_PROJECTS_WRITE + }, + + UPDATE_SKILL: { + meta: { + title: 'Update Skill', + group: 'Skill' + }, + topcoderRoles: TOPCODER_ROLES_MANAGERS_AND_ADMINS, + scopes: SCOPES_PROJECTS_WRITE + }, + + DELETE_SKILL: { + meta: { + title: 'Delete Skill', + group: 'Skill' + }, + topcoderRoles: TOPCODER_ROLES_MANAGERS_AND_ADMINS, + scopes: SCOPES_PROJECTS_WRITE + }, + + /* + * Skill Metadata + */ + ADD_SKILL_METADATA: { + meta: { + title: 'Add Skill Metadata', + group: 'Skill Metadata', + description: 'Add metadata fields in a skill' + }, + topcoderRoles: TOPCODER_ROLES_MANAGERS_AND_ADMINS, + scopes: SCOPES_PROJECTS_WRITE + }, + + UPDATE_SKILL_METADATA: { + meta: { + title: 'Update Skill Metadata', + group: 'Skill Metadata', + description: 'Update Metadata fields from a skill' + }, + topcoderRoles: TOPCODER_ROLES_MANAGERS_AND_ADMINS, + scopes: SCOPES_PROJECTS_WRITE + }, + + DELETE_SKILL_METADATA: { + meta: { + title: 'Delete Skill Metadata', + group: 'Skill Metadata', + description: 'Delete Metadata fields from a skill' + }, + topcoderRoles: TOPCODER_ROLES_MANAGERS_AND_ADMINS, + scopes: SCOPES_PROJECTS_WRITE + }, + + /* + * Taxonomy + */ + CREATE_TAXONOMY: { + meta: { + title: 'Create Taxonomy', + group: 'Taxonomy Metadata' + }, + topcoderRoles: TOPCODER_ROLES_MANAGERS_AND_ADMINS, + scopes: SCOPES_PROJECTS_WRITE + }, + + UPDATE_TAXONOMY: { + meta: { + title: 'Update Taxonomy', + group: 'Taxonomy Metadata' + }, + topcoderRoles: TOPCODER_ROLES_MANAGERS_AND_ADMINS, + scopes: SCOPES_PROJECTS_WRITE + }, + + DELETE_TAXONOMY: { + meta: { + title: 'Delete Taxonomy', + group: 'Taxonomy Metadata' + }, + topcoderRoles: TOPCODER_ROLES_MANAGERS_AND_ADMINS, + scopes: SCOPES_PROJECTS_WRITE + }, + + /* + * Taxonomy Metadata + */ + ADD_TAXONOMY_METADATA: { + meta: { + title: 'Add Taxonomy Metadata', + group: 'Taxonomy', + description: 'Add metadata fields in a taxonomy' + }, + topcoderRoles: TOPCODER_ROLES_MANAGERS_AND_ADMINS, + scopes: SCOPES_PROJECTS_WRITE + }, + + UPDATE_TAXONOMY_METADATA: { + meta: { + title: 'Update Taxonomy Metadata', + group: 'Taxonomy', + description: 'Update Metadata fields from a taxonomy' + }, + topcoderRoles: TOPCODER_ROLES_MANAGERS_AND_ADMINS, + scopes: SCOPES_PROJECTS_WRITE + }, + + DELETE_TAXONOMY_METADATA: { + meta: { + title: 'Delete Taxonomy Metadata', + group: 'Taxonomy', + description: 'Delete Metadata fields from a taxonomy' + }, + topcoderRoles: TOPCODER_ROLES_MANAGERS_AND_ADMINS, + scopes: SCOPES_PROJECTS_WRITE + } +} + +module.exports = { + PERMISSION +} diff --git a/src/permissions/generalPermission.js b/src/permissions/generalPermission.js new file mode 100644 index 0000000..bc7fc6c --- /dev/null +++ b/src/permissions/generalPermission.js @@ -0,0 +1,43 @@ +/** + * General method to check that user has permissions to call particular route. + * + * This "middleware" uses unified permissions rules to check access. + * + * - `permissions` can be an array of permissions rules or one permission rule object + * + * Usage: + * 1. One permission + * ```js + * Authorizer.setPolicy('project.view', generalPermission(PERMISSION.VIEW_PROJECT)); + * ``` + * + * where `PERMISSION.VIEW_PROJECT` is defined as any object which could be processed by + * the method `util.hasPermission`. + * + * 2. Multiple permissions + * ```js + * Authorizer.setPolicy('project.view', generalPermission([ + * PERMISSION.READ_PROJECT_INVITE_OWN, + * PERMISSION.READ_PROJECT_INVITE_NOT_OWN, + * ])); + * ``` + * + * In this case if user who is making request has at least of one listed permissions access would be allowed. + */ +const _ = require('lodash') +const helper = require('../common/permission-helper') + +/** + * @param {Object|Array} permissions permission object or array of permissions + * + * @return {Function} which would be resolved if `req` is allowed and rejected otherwise + */ +module.exports = permissions => async (req) => { + const hasPermission = _.isArray(permissions) + ? _.some(permissions, permission => helper.hasPermissionByReq(permission, req)) + : helper.hasPermissionByReq(permissions, req) + + if (!hasPermission) { + throw new Error('You do not have permissions to perform this action') + } +} diff --git a/src/permissions/index.js b/src/permissions/index.js new file mode 100644 index 0000000..b1c2038 --- /dev/null +++ b/src/permissions/index.js @@ -0,0 +1,21 @@ +/* + * Setup permission rules(policies). + */ + +const Authorizer = require('tc-core-library-js').Authorizer + +const generalPermission = require('./generalPermission') +const { PERMISSION } = require('./constants') + +module.exports = () => { + Authorizer.setDeniedStatusCode(403) + + Authorizer.setPolicy('skill.create', generalPermission(PERMISSION.CREATE_SKILL)) + Authorizer.setPolicy('skill.edit', generalPermission(PERMISSION.UPDATE_SKILL)) + Authorizer.setPolicy('skill.delete', generalPermission(PERMISSION.DELETE_SKILL)) + Authorizer.setPolicy('skill.deleteMetadata', generalPermission(PERMISSION.DELETE_SKILL_METADATA)) + Authorizer.setPolicy('taxonomy.create', generalPermission(PERMISSION.CREATE_TAXONOMY)) + Authorizer.setPolicy('taxonomy.edit', generalPermission(PERMISSION.UPDATE_TAXONOMY)) + Authorizer.setPolicy('taxonomy.delete', generalPermission(PERMISSION.DELETE_TAXONOMY)) + Authorizer.setPolicy('taxonomy.deleteMetadata', generalPermission(PERMISSION.DELETE_TAXONOMY_METADATA)) +} diff --git a/src/route.js b/src/route.js new file mode 100755 index 0000000..45816c5 --- /dev/null +++ b/src/route.js @@ -0,0 +1,29 @@ +/** + * Defines the API routes + */ + +const _ = require('lodash') +const path = require('path') +const fs = require('fs') + +/** + * scan folder to find routes + * @param dir the scan base dir + */ +function searchRoutes (dir) { + const files = fs.readdirSync(dir) + let routes = {} + for (let i = 0; i < files.length; i += 1) { + const file = files[i] + const curPath = path.join(dir, file) + const stats = fs.statSync(curPath) + if (stats.isDirectory()) { + routes = _.extend({}, routes, searchRoutes(curPath)) + } else if (file.toLowerCase().indexOf('route.js') >= 0) { + routes = _.extend({}, routes, require(curPath)); // eslint-disable-line + } + } + return routes +} + +module.exports = searchRoutes(path.join(__dirname, '.')) From 9c607d2c09a38347ac39f47e8782b8cec86378bb Mon Sep 17 00:00:00 2001 From: Vikas Agarwal Date: Mon, 2 Aug 2021 13:35:04 +0530 Subject: [PATCH 2/9] Added back circle ci config --- .circleci/config.yml | 78 ++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 78 insertions(+) create mode 100644 .circleci/config.yml diff --git a/.circleci/config.yml b/.circleci/config.yml new file mode 100644 index 0000000..bb8198d --- /dev/null +++ b/.circleci/config.yml @@ -0,0 +1,78 @@ +version: 2 +defaults: &defaults + docker: + - image: circleci/python:2.7.18-stretch-browsers +install_dependency: &install_dependency + name: Installation of build and deployment dependencies. + command: | + sudo apt install jq + sudo pip install awscli --upgrade + sudo pip install docker-compose +install_deploysuite: &install_deploysuite + name: Installation of install_deploysuite. + command: | + git clone --branch master_hostfix_v1 https://github.com/topcoder-platform/tc-deploy-scripts ../buildscript + cp ./../buildscript/master_deploy.sh . + cp ./../buildscript/buildenv.sh . + cp ./../buildscript/awsconfiguration.sh . +restore_cache_settings_for_build: &restore_cache_settings_for_build + key: docker-node-modules-{{ checksum "package-lock.json" }} + +save_cache_settings: &save_cache_settings + key: docker-node-modules-{{ checksum "package-lock.json" }} + paths: + - node_modules + +builddeploy_steps: &builddeploy_steps + - checkout + - setup_remote_docker + - run: *install_dependency + - run: *install_deploysuite + - restore_cache: *restore_cache_settings_for_build + - run: ./build.sh ${APPNAME} + - save_cache: *save_cache_settings + - deploy: + name: Running MasterScript. + command: | + ./awsconfiguration.sh $DEPLOY_ENV + source awsenvconf + ./buildenv.sh -e $DEPLOY_ENV -b ${LOGICAL_ENV}-${APPNAME}-deployvar + source buildenvvar + ./master_deploy.sh -d ECS -e $DEPLOY_ENV -t latest -s ${LOGICAL_ENV}-global-appvar,${LOGICAL_ENV}-${APPNAME}-appvar -i ${APPNAME} +jobs: + # Build & Deploy against development backend + "build-dev": + <<: *defaults + environment: + DEPLOY_ENV: "DEV" + LOGICAL_ENV: "dev" + APPNAME: "skills-api" + steps: *builddeploy_steps + + "build-prod": + <<: *defaults + environment: + DEPLOY_ENV: "PROD" + LOGICAL_ENV: "prod" + APPNAME: "skills-api" + steps: *builddeploy_steps + +workflows: + version: 2 + build: + jobs: + # Development builds are executed on "develop" branch only. + - "build-dev": + context : org-global + filters: + branches: + only: + - develop + + # Production builds are exectuted only on tagged commits to the + # master branch. + - "build-prod": + context : org-global + filters: + branches: + only: master \ No newline at end of file From a90737686a4410a778dfaaffda2451fb44b2efa4 Mon Sep 17 00:00:00 2001 From: Vikas Agarwal Date: Mon, 2 Aug 2021 13:49:09 +0530 Subject: [PATCH 3/9] Trigger bulid --- .circleci/config.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.circleci/config.yml b/.circleci/config.yml index bb8198d..1338b78 100644 --- a/.circleci/config.yml +++ b/.circleci/config.yml @@ -75,4 +75,4 @@ workflows: context : org-global filters: branches: - only: master \ No newline at end of file + only: master-hold \ No newline at end of file From 723e5dc4cdecfbb564b2a7ec4e4546185ddc1bcf Mon Sep 17 00:00:00 2001 From: vikasrohit Date: Mon, 2 Aug 2021 15:08:40 +0530 Subject: [PATCH 4/9] Updated config.yml --- .circleci/config.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.circleci/config.yml b/.circleci/config.yml index 1338b78..bb8198d 100644 --- a/.circleci/config.yml +++ b/.circleci/config.yml @@ -75,4 +75,4 @@ workflows: context : org-global filters: branches: - only: master-hold \ No newline at end of file + only: master \ No newline at end of file From 3285cc4d737bb8a70ae09200fd6a4f57659529f3 Mon Sep 17 00:00:00 2001 From: Vikas Agarwal Date: Mon, 2 Aug 2021 15:11:13 +0530 Subject: [PATCH 5/9] Adding build file --- build.sh | 23 +++++++++++++++++++++++ 1 file changed, 23 insertions(+) create mode 100644 build.sh diff --git a/build.sh b/build.sh new file mode 100644 index 0000000..9bbd81f --- /dev/null +++ b/build.sh @@ -0,0 +1,23 @@ +#!/bin/bash +set -eo pipefail +APP_NAME=$1 +UPDATE_CACHE="" +echo "" > docker/api.env +docker build -f docker/Dockerfile -t $APP_NAME:latest . +docker create --name app $APP_NAME:latest + +if [ -d node_modules ] +then + mv package-lock.json old-package-lock.json + docker cp app:/usr/src/app/package-lock.json package-lock.json + set +eo pipefail + UPDATE_CACHE=$(cmp package-lock.json old-package-lock.json) + set -eo pipefail +else + UPDATE_CACHE=1 +fi + +if [ "$UPDATE_CACHE" == 1 ] +then + docker cp app:/usr/src/app/node_modules . +fi From a0dc8bf4ba64f62070aaf77e36e2bf25a9394095 Mon Sep 17 00:00:00 2001 From: Vikas Agarwal Date: Mon, 2 Aug 2021 15:16:26 +0530 Subject: [PATCH 6/9] Execute permission --- build.sh | 0 1 file changed, 0 insertions(+), 0 deletions(-) mode change 100644 => 100755 build.sh diff --git a/build.sh b/build.sh old mode 100644 new mode 100755 From f3204a89274609b70404b4def56a41d75dc12fe3 Mon Sep 17 00:00:00 2001 From: Vikas Agarwal Date: Mon, 2 Aug 2021 15:18:48 +0530 Subject: [PATCH 7/9] Hardcoding port --- docker/Dockerfile | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/docker/Dockerfile b/docker/Dockerfile index dcd6339..cccd6bd 100644 --- a/docker/Dockerfile +++ b/docker/Dockerfile @@ -11,7 +11,7 @@ COPY . /skills_api RUN npm install # Expose port -EXPOSE ${PORT} +EXPOSE 3000 # start api CMD npm start From 5a6b7f5972b4b3ff8f48e892910468e20a59c79a Mon Sep 17 00:00:00 2001 From: Vikas Agarwal Date: Mon, 2 Aug 2021 15:21:11 +0530 Subject: [PATCH 8/9] Fixing working directory --- docker/Dockerfile | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/docker/Dockerfile b/docker/Dockerfile index cccd6bd..72826e5 100644 --- a/docker/Dockerfile +++ b/docker/Dockerfile @@ -2,10 +2,10 @@ FROM node:12 # Set working directory for future use -WORKDIR /skills_api +WORKDIR /usr/src/app # Copy the current directory into the Docker image -COPY . /skills_api +COPY . /usr/src/app # Install the dependencies from package.json RUN npm install From 6c079bb140e0a21640cf756d0adf64356f88d026 Mon Sep 17 00:00:00 2001 From: Vikas Agarwal Date: Mon, 2 Aug 2021 15:36:17 +0530 Subject: [PATCH 9/9] Health check --- src/modules/health/controller.js | 33 ++++++++++++++++++++++++++++++++ src/modules/health/route.js | 13 +++++++++++++ 2 files changed, 46 insertions(+) create mode 100644 src/modules/health/controller.js create mode 100644 src/modules/health/route.js diff --git a/src/modules/health/controller.js b/src/modules/health/controller.js new file mode 100644 index 0000000..f4641df --- /dev/null +++ b/src/modules/health/controller.js @@ -0,0 +1,33 @@ +/** + * Controller for health check endpoint + */ +const models = require('../../models') +const config = require('config') +const logger = require('../../common/logger') + +// the topcoder-healthcheck-dropin library returns checksRun count, +// here it follows that to return such count +let checksRun = 0 + +/** + * Check health of the DB + * @param {Object} req the request + * @param {Object} res the response + */ +async function checkHealth (req, res) { + checksRun += 1 + await models + .validate() + .then(() => { + logger.info({ component: 'HealthCheckController', context: 'checkHealth', message: 'Connection has been established successfully.' }) + }) + .catch(err => { + logger.logFullError(err, { component: 'HealthCheckController', context: 'checkHealth' }) + res.status(503) + }) + res.send({ checksRun }) +} + +module.exports = { + checkHealth +} \ No newline at end of file diff --git a/src/modules/health/route.js b/src/modules/health/route.js new file mode 100644 index 0000000..ce23875 --- /dev/null +++ b/src/modules/health/route.js @@ -0,0 +1,13 @@ +/** + * the skill routes + */ + +const Controller = require('./controller') + +module.exports = { + '/health': { + get: { + method: Controller.checkHealth + } + } +}