diff --git a/.dockerignore b/.dockerignore deleted file mode 100644 index 47f9107..0000000 --- a/.dockerignore +++ /dev/null @@ -1,5 +0,0 @@ -.git -Dockerfile -.DS_Store -README.md -env.* \ No newline at end of file diff --git a/.gitignore b/.gitignore index 24318b6..c992010 100644 --- a/.gitignore +++ b/.gitignore @@ -1,51 +1,3 @@ -# See http://help.github.com/ignore-files/ for more about ignoring files. - -# compiled output -/dist -/tmp -/out-tsc -# Only exists if Bazel was run -/bazel-out - -# dependencies -node_modules - -# profiling files -chrome-profiler-events*.json -speed-measure-plugin*.json - -# IDEs and editors -/.idea -.project -.classpath -.c9/ -*.launch -.settings/ -*.sublime-workspace - -# IDE - VSCode -.vscode/* -!.vscode/settings.json -!.vscode/tasks.json -!.vscode/launch.json -!.vscode/extensions.json -.history/* - -# misc -/.sass-cache -/connect.lock -/coverage -/libpeerconnection.log -npm-debug.log -yarn-error.log -testem.log -/typings - -# System Files -.DS_Store -Thumbs.db - - # Byte-compiled / optimized / DLL files __pycache__/ *.py[cod] @@ -68,10 +20,12 @@ parts/ sdist/ var/ wheels/ +pip-wheel-metadata/ share/python-wheels/ *.egg-info/ .installed.cfg *.egg +*.pickle MANIFEST # PyInstaller @@ -97,7 +51,6 @@ coverage.xml *.py,cover .hypothesis/ .pytest_cache/ -cover/ # Translations *.mo @@ -120,7 +73,6 @@ instance/ docs/_build/ # PyBuilder -.pybuilder/ target/ # Jupyter Notebook @@ -131,9 +83,7 @@ profile_default/ ipython_config.py # pyenv -# For a library or package, you might want to ignore these files since the code is -# intended to run in multiple environments; otherwise, check them in: -# .python-version +.python-version # pipenv # According to pypa/pipenv#598, it is recommended to include Pipfile.lock in version control. @@ -179,8 +129,153 @@ dmypy.json # Pyre type checker .pyre/ -# pytype static type analyzer -.pytype/ +local_packages +tmp +*.lock + +# AWS +*.iml +.idea/ +.aws-sam/ +*/dist/* +packaged.yaml +package-lock.json + +# local +cache +keys + + +# Node +# https://raw.githubusercontent.com/github/gitignore/main/Node.gitignore + +# Logs +logs +*.log +npm-debug.log* +yarn-debug.log* +yarn-error.log* +lerna-debug.log* +.pnpm-debug.log* + +# Diagnostic reports (https://nodejs.org/api/report.html) +report.[0-9]*.[0-9]*.[0-9]*.[0-9]*.json + +# Runtime data +pids +*.pid +*.seed +*.pid.lock + +# Directory for instrumented libs generated by jscoverage/JSCover +lib-cov + +# Coverage directory used by tools like istanbul +coverage +*.lcov + +# nyc test coverage +.nyc_output + +# Grunt intermediate storage (https://gruntjs.com/creating-plugins#storing-task-files) +.grunt + +# Bower dependency directory (https://bower.io/) +bower_components + +# node-waf configuration +.lock-wscript + +# Compiled binary addons (https://nodejs.org/api/addons.html) +build/Release + +# Dependency directories +node_modules/ +jspm_packages/ + +# Snowpack dependency directory (https://snowpack.dev/) +web_modules/ + +# TypeScript cache +*.tsbuildinfo + +# Optional npm cache directory +.npm + +# Optional eslint cache +.eslintcache + +# Optional stylelint cache +.stylelintcache + +# Microbundle cache +.rpt2_cache/ +.rts2_cache_cjs/ +.rts2_cache_es/ +.rts2_cache_umd/ + +# Optional REPL history +.node_repl_history + +# Output of 'npm pack' +*.tgz + +# Yarn Integrity file +.yarn-integrity + +# dotenv environment variable files +.env +.env.development.local +.env.test.local +.env.production.local +.env.local + +# parcel-bundler cache (https://parceljs.org/) +.cache +.parcel-cache + +# Next.js build output +.next +out + +# Nuxt.js build / generate output +.nuxt +dist + +# Gatsby files +.cache/ +# Comment in the public line in if your project uses Gatsby and not Next.js +# https://nextjs.org/blog/next-9-1#public-directory-support +# public + +# vuepress build output +.vuepress/dist + +# vuepress v2.x temp and cache directory +.temp +.cache + +# Docusaurus cache and generated files +.docusaurus + +# Serverless directories +.serverless/ + +# FuseBox cache +.fusebox/ + +# DynamoDB Local files +.dynamodb/ + +# TernJS port file +.tern-port + +# Stores VSCode versions used for testing VSCode extensions +.vscode-test -# Cython debug symbols -cython_debug/ \ No newline at end of file +# yarn v2 +.yarn/cache +.yarn/unplugged +.yarn/build-state.yml +.yarn/install-state.gz +.pnp.* \ No newline at end of file diff --git a/.prettierrc b/.prettierrc new file mode 100644 index 0000000..316ebc0 --- /dev/null +++ b/.prettierrc @@ -0,0 +1,10 @@ +{ + "printWidth": 80, + "singleQuote": false, + "useTabs": false, + "tabWidth": 2, + "semi": false, + "bracketSpacing": true, + "trailingComma": "none", + "arrowParens": "avoid" +} \ No newline at end of file diff --git a/.vscode/settings.json b/.vscode/settings.json new file mode 100644 index 0000000..6be2153 --- /dev/null +++ b/.vscode/settings.json @@ -0,0 +1,9 @@ +{ + "editor.lineHeight": 20, + "editor.tabSize": 2, + "editor.formatOnSave": false, + "editor.renderWhitespace": "all", + "editor.renderControlCharacters": true, + "files.eol": "\n", + "python.formatting.provider": "black" +} \ No newline at end of file diff --git a/Makefile b/Makefile old mode 100755 new mode 100644 index f839e28..4774423 --- a/Makefile +++ b/Makefile @@ -1,48 +1,49 @@ -BACKEND_CONTAINER_NAME = nuxx-api -NGINX_CONTAINER_NAME = nuxx-nginx - -.PHONY : validate build pull up down down_clean reset run backend_dev shell_backend shell_nginx local_setup local_build - -validate : - docker-compose config - -build : validate - docker-compose build - -pull : - docker-compose pull - -up : - @ docker-compose up -d - -down : - docker-compose down - -down_clean : down - -docker volume rm nuxx_postgres_data - -docker volume rm nuxx_django_static - -reset : down - make up - -run : validate - docker-compose run $(BACKEND_CONTAINER_NAME) -c "cd /home/app/ && python manage.py runserver 0.0.0.0:9001" - -dev_backend : - docker exec -ti $(BACKEND_CONTAINER_NAME) python /home/app/manage.py runserver 0.0.0.0:9001 - -shell_backend: - docker exec -it ${BACKEND_CONTAINER_NAME} bash - -shell_nginx: - docker exec -it ${NGINX_CONTAINER_NAME} bash - -local_build: - @ cd ./src/composer && npm install && npm run build_local - -local_setup: local_build up - @ echo "Waiting for PostgreSQL..." \ - && sleep 5 \ - && docker exec -it ${BACKEND_CONTAINER_NAME} python /home/app/manage.py makemigrations \ - && docker exec -it ${BACKEND_CONTAINER_NAME} python /home/app/manage.py migrate \ - && docker exec -it ${BACKEND_CONTAINER_NAME} python /home/app/manage.py collectstatic --noinput +ORGANIZATION = corpulent +CONTAINER = ctk-server +VERSION = 0.1.0 + +.PHONY : validate build pull up down down_clean reset run backend_dev shell_server shell_nginx local_setup local_build + +validate : + docker-compose config + +build : validate + docker-compose build + +pull : + docker-compose pull + +up : + docker-compose up -d + +up_local : + docker-compose up -d --no-build + +down : + docker-compose down + +down_clean : down + -docker volume rm ctk_postgres_data + -docker volume rm ctk_django_static + +reset : down + make up + +run_server : validate + docker-compose run $(CONTAINER) -c "cd /home/server/ && python manage.py runserver 0.0.0.0:9001" + +dev_server : + docker exec -ti $(CONTAINER) python /home/server/manage.py runserver 0.0.0.0:9001 + +shell_server: + docker exec -it ${CONTAINER} bash + +frontend_build: + @ cd ./services/frontend/src && npm install && npm run build + +local_setup: frontend_build up + @ echo "Waiting for PostgreSQL..." \ + && sleep 5 \ + && docker exec -it ${CONTAINER} python /home/server/manage.py makemigrations \ + && docker exec -it ${CONTAINER} python /home/server/manage.py migrate \ + && docker exec -it ${BACKEND_CONTAINER_NAME} python /home/server/manage.py collectstatic --noinput diff --git a/README.md b/README.md index 64276aa..bb5291c 100644 --- a/README.md +++ b/README.md @@ -1,6 +1,6 @@ -# Nuxx Visual Docker Composer +# Container ToolKit -## Local setup +## Local setup and development On a Mac/Linux/Windows you need Docker, Docker Compose installed. Optionally GCC to run make commands for convenience, or just run the commands from the Makefile by hand. @@ -8,20 +8,15 @@ To get the tool working locally, just run: ```shell script $ make local_setup +$ make run_server +$ cd services/frontend && npm run start ``` -... this command will bring up the backend, the database, sync migrations, and build and serve the Angular app in an Nginx container (for working locally with the tool). For production, you can build and deploy your own images or use mine as a base. - -## Local development - -- You can run the backend in dev mode with `make backend_dev`. -- For developing the frontend run `cd ./src/composer && npm run start`. It will expect the backend connection on http://localhost:9001/api, but you can change this to your liking in src/composer/src/environment/environment.ts. +... this command will bring up the backend, the database, sync migrations, ## Project roadmap +- Complete react rewrite. - Ongoing improvements and features for docker compose yaml generation. -- Kubernetes yaml generation. -- Application stack deployments directly from the web tool and CLI. -- Nuxx CLI. - -For anynone interested on trying out deployments, and the CLI, please message me on Slack. These features need some more work and testing. +- Kubernetes manifest generation. +- Deployment to user's ECS, K8S, GS accounts. diff --git a/docker-compose.yml b/docker-compose.yml index a195341..cf2091e 100644 --- a/docker-compose.yml +++ b/docker-compose.yml @@ -3,61 +3,64 @@ version: "3.4" volumes: postgres-data: driver: local - name: nuxx_postgres_data + name: ctk_postgres_data django-static: driver: local - name: nuxx_django_static + name: ctk_django_static services: + postgres: - container_name: nuxx-postgres - image: postgres:9.6.1 + container_name: ctk-postgres + image: postgres:11 ports: - 5432:5432 - healthcheck: - test: ["CMD-SHELL", "pg_isready"] - interval: 30s - timeout: 30s - retries: 3 volumes: - postgres-data:/var/lib/postgresql/data + environment: + - POSTGRES_PASSWORD=postgres backend: - container_name: nuxx-api + container_name: ctk-server restart: always - build: src/backend - image: nuxxapp/nuxx-api:1.0.0 + build: + context: ./ + dockerfile: ./services/backend/Dockerfile + image: corpulent/ctk-api:1.0.0 working_dir: /home depends_on: - postgres links: - postgres volumes: - - ./src/backend/server:/home/app/ + - ./services/backend/src:/home/server/ - django-static:/static/ ports: - "9001:9001" - env_file: - - local.env + environment: + - DB_REMOTE=False - nginx: - container_name: nuxx-nginx + frontend: + container_name: ctk-frontend restart: always - image: nginx:latest + build: + context: ./ + dockerfile: ./services/frontend/Dockerfile + image: corpulent/ctk-frontend:1.0.0 depends_on: - backend links: - backend volumes: # configs - - ${PWD}/configs/nginx/uwsgi_params:/home/config/uwsgi/uwsgi_params - - ${PWD}/configs/nginx/localhost.conf:/etc/nginx/conf.d/default.conf - - ${PWD}/configs/nginx/nginx.conf:/etc/nginx/nginx.conf + - ${PWD}/services/frontend/configs/nginx/uwsgi_params:/home/config/uwsgi/uwsgi_params + - ${PWD}/services/frontend/configs/nginx/localhost.conf:/etc/nginx/conf.d/default.conf + - ${PWD}/services/frontend/configs/nginx/nginx.conf:/etc/nginx/nginx.conf # serve django static stuff - - django-static:/home/backend/static/ + - django-static:/home/server/static/ - # serve composer "built" angular app - - ./src/composer/dist/frontend:/usr/share/nginx/html/ + # serve composer built react app + - ${PWD}/services/frontend/build:/usr/share/nginx/html/ ports: - - "80:80" \ No newline at end of file + - "80:80" diff --git a/local.env b/local.env deleted file mode 100644 index be85af1..0000000 --- a/local.env +++ /dev/null @@ -1,8 +0,0 @@ -#AWS_STORAGE_BUCKET_NAME= -DB_REMOTE=False -#DB_HOST= -#DB_NAME= -#DB_USER= -#DB_PASS= -#SOCIAL_AUTH_CUSTOM_CALLBACK= -#SOCIAL_AUTH_CUSTOM_CALLBACK_PAGE= \ No newline at end of file diff --git a/services/.DS_Store b/services/.DS_Store new file mode 100644 index 0000000..2a559d6 Binary files /dev/null and b/services/.DS_Store differ diff --git a/services/backend/Dockerfile b/services/backend/Dockerfile new file mode 100644 index 0000000..e2a45b0 --- /dev/null +++ b/services/backend/Dockerfile @@ -0,0 +1,44 @@ +FROM python:3.10-slim + +WORKDIR /home + +RUN apt-get update && \ + apt-get install -y \ + software-properties-common \ + build-essential + +RUN apt-get update && \ + apt-get install -y \ + postgresql \ + postgresql-contrib \ + wget \ + nano \ + curl \ + lsof \ + supervisor && \ + rm -rf /var/lib/apt/lists/* + +RUN useradd uwsgi && adduser uwsgi root +RUN useradd supervisor && adduser supervisor root + +COPY ./services/backend/requirements.txt ./requirements.txt +RUN pip install --upgrade pip && \ + pip install -r ./requirements.txt && \ + rm ./requirements.txt + +RUN touch /var/log/backend_out.log && \ + touch /var/log/django.log + +RUN chmod g+w -R /var/log/ + +EXPOSE 9000 9001 + +COPY ./services/backend/src ./server +COPY ./services/backend/configs/supervisor/api.conf /etc/supervisor/conf.d/api.conf +COPY ./services/backend/configs/uwsgi ./config/uwsgi + +RUN rm -rf /tmp/uwsgi && \ + mkdir -p /tmp/uwsgi && \ + ln -s ./config/uwsgi/uwsgi.ini /tmp/uwsgi/ + +CMD ["/usr/bin/supervisord", "-n", "-c", "/etc/supervisor/supervisord.conf"] diff --git a/src/backend/Makefile b/services/backend/Makefile similarity index 54% rename from src/backend/Makefile rename to services/backend/Makefile index c431a60..58e8924 100644 --- a/src/backend/Makefile +++ b/services/backend/Makefile @@ -1,6 +1,6 @@ -ORGANIZATION = nuxxapp -CONTAINER = nuxx-api -VERSION = 1.0.0 +ORGANIZATION = agolub +CONTAINER = ctk-server +VERSION = 0.1.0 .PHONY: build diff --git a/services/backend/configs/supervisor/api.conf b/services/backend/configs/supervisor/api.conf new file mode 100644 index 0000000..7cd1b64 --- /dev/null +++ b/services/backend/configs/supervisor/api.conf @@ -0,0 +1,8 @@ +[supervisord] +nodaemon=true + +[program:app] +priority=1 +user = uwsgi +command = /usr/local/bin/uwsgi --ini /tmp/uwsgi/uwsgi.ini +autorestart=false diff --git a/src/backend/configs/uwsgi/uwsgi.ini b/services/backend/configs/uwsgi/uwsgi.ini similarity index 85% rename from src/backend/configs/uwsgi/uwsgi.ini rename to services/backend/configs/uwsgi/uwsgi.ini index e27ae1f..40edf1a 100644 --- a/src/backend/configs/uwsgi/uwsgi.ini +++ b/services/backend/configs/uwsgi/uwsgi.ini @@ -7,7 +7,7 @@ processes = 5 [base] -chdir = /home/app +chdir = /home/server module = server.wsgi:application chmod-socket=666 uid = uwsgi diff --git a/configs/nginx/uwsgi_params b/services/backend/configs/uwsgi/uwsgi_params similarity index 100% rename from configs/nginx/uwsgi_params rename to services/backend/configs/uwsgi/uwsgi_params diff --git a/src/backend/docker-entrypoint.sh b/services/backend/docker-entrypoint.sh similarity index 100% rename from src/backend/docker-entrypoint.sh rename to services/backend/docker-entrypoint.sh diff --git a/services/backend/requirements.txt b/services/backend/requirements.txt new file mode 100644 index 0000000..3a4b165 --- /dev/null +++ b/services/backend/requirements.txt @@ -0,0 +1,26 @@ +django==4.0.4 +django-cors-headers==3.11.0 +django-axes==5.32.0 +djangorestframework==3.13.1 +djangorestframework-simplejwt==5.1.0 +drf-extensions==0.7.1 +dj-rest-auth[with_social]==2.2.4 + +psycopg[binary]==3.0.12 +psycopg==3.0.12 +psycopg2-binary==2.9.3 +uwsgi==2.0.20 +botocore==1.24.46 +boto3==1.21.46 +Jinja2==3.1.1 +validators==0.19.0 +requests==2.27.1 +celery==5.2.3 +redis==4.3.1 +simple-salesforce==1.11.6 +cryptography==37.0.2 +chardet==4.0.0 +pyaml==21.10.1 +docker==5.0.3 +ruamel.yaml==0.17.21 +better-profanity==0.7.0 diff --git a/src/backend/server/__init__.py b/services/backend/src/api/__init__.py similarity index 100% rename from src/backend/server/__init__.py rename to services/backend/src/api/__init__.py diff --git a/services/backend/src/api/admin.py b/services/backend/src/api/admin.py new file mode 100644 index 0000000..293214f --- /dev/null +++ b/services/backend/src/api/admin.py @@ -0,0 +1,14 @@ +from django.contrib import admin +from .models import Project + + +class ProjectAdmin(admin.ModelAdmin): + list_display = ( + 'id', + 'name', + 'uuid', + 'created_at', + 'updated_at') + + +admin.site.register(Project, ProjectAdmin) diff --git a/src/backend/server/api/apps.py b/services/backend/src/api/apps.py similarity index 58% rename from src/backend/server/api/apps.py rename to services/backend/src/api/apps.py index d87006d..66656fd 100644 --- a/src/backend/server/api/apps.py +++ b/services/backend/src/api/apps.py @@ -2,4 +2,5 @@ from django.apps import AppConfig class ApiConfig(AppConfig): + default_auto_field = 'django.db.models.BigAutoField' name = 'api' diff --git a/services/backend/src/api/filters.py b/services/backend/src/api/filters.py new file mode 100644 index 0000000..cc5e231 --- /dev/null +++ b/services/backend/src/api/filters.py @@ -0,0 +1,9 @@ +from rest_framework.filters import BaseFilterBackend +from organizations.utils import get_user_org + + +class FilterByOrg(BaseFilterBackend): + def filter_queryset(self, request, queryset, view): + org = get_user_org(request.user) + queryset_filters = {"org": org} + return queryset.filter(**queryset_filters) diff --git a/services/backend/src/api/migrations/0001_initial.py b/services/backend/src/api/migrations/0001_initial.py new file mode 100644 index 0000000..72d5bb1 --- /dev/null +++ b/services/backend/src/api/migrations/0001_initial.py @@ -0,0 +1,32 @@ +# Generated by Django 4.0.4 on 2022-06-22 10:14 + +from django.db import migrations, models +import django.db.models.deletion + + +class Migration(migrations.Migration): + + initial = True + + dependencies = [ + ('organizations', '0001_initial'), + ] + + operations = [ + migrations.CreateModel( + name='Project', + fields=[ + ('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), + ('name', models.CharField(default='Untitled', max_length=500)), + ('data', models.TextField()), + ('created_at', models.DateTimeField(auto_now_add=True)), + ('updated_at', models.DateTimeField(auto_now=True)), + ('org', models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.CASCADE, related_name='projects', to='organizations.organization')), + ], + options={ + 'verbose_name': 'Project', + 'verbose_name_plural': 'Projects', + 'ordering': ['-created_at'], + }, + ), + ] diff --git a/src/backend/server/api/migrations/0002_project_mutable.py b/services/backend/src/api/migrations/0002_project_uuid.py similarity index 60% rename from src/backend/server/api/migrations/0002_project_mutable.py rename to services/backend/src/api/migrations/0002_project_uuid.py index cc23d31..c68642c 100644 --- a/src/backend/server/api/migrations/0002_project_mutable.py +++ b/services/backend/src/api/migrations/0002_project_uuid.py @@ -1,4 +1,4 @@ -# Generated by Django 3.0.4 on 2020-05-31 13:33 +# Generated by Django 4.0.4 on 2022-06-22 10:17 from django.db import migrations, models @@ -12,7 +12,7 @@ class Migration(migrations.Migration): operations = [ migrations.AddField( model_name='project', - name='mutable', - field=models.BooleanField(default=True), + name='uuid', + field=models.CharField(blank=True, max_length=500, null=True, unique=True), ), ] diff --git a/src/backend/server/api/__init__.py b/services/backend/src/api/migrations/__init__.py similarity index 100% rename from src/backend/server/api/__init__.py rename to services/backend/src/api/migrations/__init__.py diff --git a/services/backend/src/api/models/__init__.py b/services/backend/src/api/models/__init__.py new file mode 100644 index 0000000..ee5814c --- /dev/null +++ b/services/backend/src/api/models/__init__.py @@ -0,0 +1,5 @@ +from .project import Project + +__all__ = [ + "Project" +] diff --git a/services/backend/src/api/models/project.py b/services/backend/src/api/models/project.py new file mode 100644 index 0000000..15efdc3 --- /dev/null +++ b/services/backend/src/api/models/project.py @@ -0,0 +1,22 @@ +from django.db import models +from organizations.models import Organization + + +class Project(models.Model): + org = models.ForeignKey( + Organization, + blank=True, + null=True, + related_name="projects", + on_delete=models.CASCADE, + ) + name = models.CharField(max_length=500, blank=False, null=False, default="Untitled") + uuid = models.CharField(max_length=500, blank=True, null=True, unique=True) + data = models.TextField(blank=False) + created_at: models.DateTimeField = models.DateTimeField(auto_now_add=True) + updated_at: models.DateTimeField = models.DateTimeField(auto_now=True) + + class Meta: + verbose_name = "Project" + verbose_name_plural = "Projects" + ordering = ["-created_at"] diff --git a/services/backend/src/api/routing.py b/services/backend/src/api/routing.py new file mode 100644 index 0000000..1e08ee5 --- /dev/null +++ b/services/backend/src/api/routing.py @@ -0,0 +1,23 @@ +from django.urls import include, path + +from rest_framework_extensions.routers import ExtendedDefaultRouter + +from .views import project, generate, user, view + + +class DefaultRouterPlusPlus(ExtendedDefaultRouter): + """DefaultRouter with optional trailing slash and drf-extensions nesting.""" + + def __init__(self, *args, **kwargs): + super().__init__(*args, **kwargs) + self.trailing_slash = r"/?" + +api_urls = [ + path("", view.ViewGenericAPIView.as_view()), + path("projects/", project.ProjectListCreateAPIView.as_view()), + path("projects//", project.ProjectGenericAPIView.as_view()), + path("generate/", generate.GenerateGenericAPIView.as_view()), + path("auth/self/", user.UserGenericAPIView.as_view()), + path("auth/", include("dj_rest_auth.urls")), + path("auth/registration/", include("dj_rest_auth.registration.urls")), +] diff --git a/services/backend/src/api/serializers.py b/services/backend/src/api/serializers.py new file mode 100644 index 0000000..26c8e35 --- /dev/null +++ b/services/backend/src/api/serializers.py @@ -0,0 +1,26 @@ +import json +from rest_framework import serializers +from .models import Project + + +class DataField(serializers.Field): + def to_representation(self, value): + return value + + def to_internal_value(self, value): + return json.dumps(value) + + +class ProjectSerializer(serializers.ModelSerializer): + data = DataField() + class Meta(object): + model = Project + fields = "__all__" + + +class UserSelfSerializer(serializers.Serializer): + pk = serializers.IntegerField() + username = serializers.CharField(max_length=200) + first_name = serializers.CharField(max_length=200) + last_name = serializers.CharField(max_length=200) + email = serializers.CharField(max_length=200) diff --git a/src/backend/server/api/tests.py b/services/backend/src/api/tests.py similarity index 100% rename from src/backend/server/api/tests.py rename to services/backend/src/api/tests.py diff --git a/services/backend/src/api/views/generate.py b/services/backend/src/api/views/generate.py new file mode 100644 index 0000000..dc347e8 --- /dev/null +++ b/services/backend/src/api/views/generate.py @@ -0,0 +1,35 @@ +from rest_framework import generics, status +from rest_framework.response import Response + +from .utils import generate_dc + + +class GenerateGenericAPIView(generics.GenericAPIView): + permission_classes = [] + + def get(self, request): + return Response({}, status=status.HTTP_404_NOT_FOUND) + + def post(self, request, format=None): + request_data = request.data + version = request_data['data'].get('version', '3') + services = request_data['data'].get('services', None) + connections = request_data['data'].get('connections', None) + volumes = request_data['data'].get('volumes', None) + networks = request_data['data'].get('networks', None) + secrets = request_data['data'].get('secrets', None) + configs = request_data['data'].get('configs', None) + + code = generate_dc( + services, + connections, + volumes, + networks, + secrets, + configs, + version=version, + return_format='yaml') + resp = {'code': code} + + + return Response(resp, status=status.HTTP_200_OK) diff --git a/services/backend/src/api/views/project.py b/services/backend/src/api/views/project.py new file mode 100644 index 0000000..af47c8c --- /dev/null +++ b/services/backend/src/api/views/project.py @@ -0,0 +1,93 @@ +import uuid +from rest_framework import generics, status +from rest_framework.permissions import IsAuthenticated +from rest_framework.response import Response + +from api.serializers import ProjectSerializer +from api.models import Project +from api.filters import FilterByOrg + +from organizations.utils import get_user_org + +from .utils import get_project_obj_by_uuid + + +class ProjectListCreateAPIView(generics.ListCreateAPIView): + permission_classes = [] + serializer_class = ProjectSerializer + queryset = Project.objects.all() + + def filter_queryset(self, queryset): + filter_backends = (FilterByOrg,) + + for backend in list(filter_backends): + queryset = backend().filter_queryset(self.request, queryset, view=self) + + return queryset + + def list(self, request): + try: + queryset = self.filter_queryset(self.get_queryset()) + page = self.paginate_queryset(queryset) + if page is not None: + serializer = self.get_serializer(page, many=True) + return self.get_paginated_response(serializer.data) + + serializer = self.get_serializer(queryset, many=True) + return Response(serializer.data) + except Exception as e: + return Response(str(e), status=status.HTTP_400_BAD_REQUEST) + + def create(self, request, *args, **kwargs): + org = None if request.user.is_anonymous else get_user_org(request.user) + data = { + "uuid": str(uuid.uuid4())[:10], + **request.data + } + + if org: + data["org"] = org.pk + + serializer = self.get_serializer(data=data) + serializer.is_valid(raise_exception=True) + self.perform_create(serializer) + headers = self.get_success_headers(serializer.data) + return Response( + serializer.data, status=status.HTTP_201_CREATED, headers=headers + ) + + +class ProjectGenericAPIView(generics.GenericAPIView): + permission_classes = [] + serializer_class = ProjectSerializer + queryset = Project.objects.all() + + def get(self, request, uuid): + try: + org = None if request.user.is_anonymous else get_user_org(request.user) + if project_obj := get_project_obj_by_uuid(uuid): + return Response(ProjectSerializer(project_obj).data) + except Exception as e: + return Response(str(e), status=status.HTTP_400_BAD_REQUEST) + + return Response({}, status=status.HTTP_404_NOT_FOUND) + + def put(self, request, uuid): + org = None if request.user.is_anonymous else get_user_org(request.user) + if project_obj := get_project_obj_by_uuid(uuid): + data = request.data + serializer = ProjectSerializer(project_obj, data=data) + + if serializer.is_valid(): + serializer.save() + return Response(serializer.data) + + return Response(serializer.errors, status=status.HTTP_500_INTERNAL_SERVER_ERROR) + return Response({}, status=status.HTTP_404_NOT_FOUND) + + def delete(self, request, uuid): + org = None if request.user.is_anonymous else get_user_org(request.user) + if project_obj := get_project_obj_by_uuid(uuid): + project_obj.delete() + return Response({}, status=status.HTTP_204_NO_CONTENT) + return Response({}, status=status.HTTP_404_NOT_FOUND) diff --git a/services/backend/src/api/views/user.py b/services/backend/src/api/views/user.py new file mode 100644 index 0000000..dcd5d5c --- /dev/null +++ b/services/backend/src/api/views/user.py @@ -0,0 +1,12 @@ +from rest_framework import generics, status +from rest_framework.permissions import IsAuthenticated +from rest_framework.response import Response + +from api.serializers import UserSelfSerializer + + +class UserGenericAPIView(generics.GenericAPIView): + permission_classes = [IsAuthenticated] + + def get(self, request): + return Response(UserSelfSerializer(request.user).data) diff --git a/src/backend/server/utils/utils.py b/services/backend/src/api/views/utils.py similarity index 69% rename from src/backend/server/utils/utils.py rename to services/backend/src/api/views/utils.py index ea456de..4799f31 100644 --- a/src/backend/server/utils/utils.py +++ b/services/backend/src/api/views/utils.py @@ -1,34 +1,18 @@ import io -import os import re import sys -import json -import boto3 -import copy import random import string -import stat -import decimal -import time -import pyaml -import docker -import yaml import ast import uuid -import ruamel.yaml +import contextlib +import docker from better_profanity import profanity -from io import StringIO -from collections import OrderedDict -from time import mktime -from datetime import date, datetime -from operator import itemgetter -from botocore.exceptions import ClientError - from ruamel.yaml import YAML -from ruamel.yaml.tokens import CommentToken -from ruamel.yaml.comments import CommentedMap, CommentedSeq -from ruamel.yaml.scalarstring import PreservedScalarString, SingleQuotedScalarString, DoubleQuotedScalarString +from ruamel.yaml.scalarstring import SingleQuotedScalarString, DoubleQuotedScalarString + +from api.models import Project try: @@ -41,7 +25,18 @@ except AttributeError: # undefined function (wasn't added until Python 3.3) else: def indent(text, amount, ch=' '): return textwrap.indent(text, amount * ch) - + + +def get_project_obj_by_id(id): + with contextlib.suppress(Project.DoesNotExist): + return Project.objects.get(pk=id) + return None + +def get_project_obj_by_uuid(uuid): + with contextlib.suppress(Project.DoesNotExist): + return Project.objects.get(uuid=uuid) + return None + def sequence_indent_four(s): ret_val = '' @@ -83,7 +78,7 @@ def sequence_indent_one(s): def format_quotes(s): if '\'' in s: - return SingleQuotedScalarString(s.replace("'", '')) + return SingleQuotedScalarString(s.replace("'", '')) if '"' in s: return DoubleQuotedScalarString(s.replace('"', '')) @@ -110,18 +105,16 @@ def format_volumes_top_level(volumes, compose_version): if volume_custom_name: ret[volume['name']]['name'] = volume_custom_name - + if volume_driver: ret[volume['name']]['driver'] = volume_driver - + if compose_version in [2, 3]: - labels = volume.get('labels', None) - if labels: + if labels := volume.get('labels', None): ret[volume['name']]['labels'] = {} for label in labels: ret[volume['name']]['labels'][label['key']] = format_quotes(label['value']) - if not ret[volume['name']]: ret[volume['name']] = None @@ -146,18 +139,17 @@ def format_networks_top_level(networks, compose_version): if network_custom_name: ret[network['name']]['name'] = network_custom_name - + if network_driver: ret[network['name']]['driver'] = network_driver - + if driver_opts: ret[network['name']]['driver_opts'] = {} for driver_opt in driver_opts: ret[network['name']]['driver_opts'][driver_opt['key']] = format_quotes(driver_opt['value']) if compose_version in [2, 3]: - labels = network.get('labels', None) - if labels: + if labels := network.get('labels', None): ret[network['name']]['labels'] = {} for label in labels: ret[network['name']]['labels'][label['key']] = format_quotes(label['value']) @@ -169,12 +161,7 @@ def format_networks_top_level(networks, compose_version): def format_key_val_pairs(pairs): - ret = {} - - for pair_part in pairs: - ret[pair_part['key']] = pair_part['value'] - - return ret + return {pair_part['key']: pair_part['value'] for pair_part in pairs} def format_ports(ports): @@ -196,11 +183,10 @@ def format_volumes(service_volumes, volumes): ret = [] for service_volume in service_volumes: for volume in volumes: - if 'volume' in service_volume: - if service_volume['volume'] == volume['uuid']: - volume_mount_str = f"{volume['name']}:{service_volume['destination']}" - ret.append(volume_mount_str) - + if 'volume' in service_volume and service_volume['volume'] == volume['uuid']: + volume_mount_str = f"{volume['name']}:{service_volume['destination']}" + ret.append(volume_mount_str) + if 'relativePathSource' in service_volume: volume_mount_str = f"{service_volume['relativePathSource']}:{service_volume['destination']}" ret.append(volume_mount_str) @@ -232,7 +218,6 @@ def format_command_string(command): """ Format command list of string for v1, v2, v3. param: command: string - return: list """ command_list = [] @@ -243,14 +228,12 @@ def format_command_string(command): # try to convert the string into list command_list = ast.literal_eval(command_list) except (ValueError, SyntaxError) as e: - #print('ValueError SyntaxError', e) # special case if "\n" in command: command_list = command.split("\n") else: return command except Exception as e: - #print('Exception', e) return command if len(command_list) > 1: @@ -270,6 +253,7 @@ def format_build(specified_version, build): build_str = build.get('build', None) context_str = build.get('context', None) + ret = {} if specified_version < 2: if build_str: @@ -279,11 +263,9 @@ def format_build(specified_version, build): else: return None - if build_str: return build_str - ret = {} for _key, _val in build.items(): if _key in ['args', 'cache_from', 'labels']: if _val: @@ -304,7 +286,7 @@ def _remove_missing_and_underscored_keys(d): del d[key] elif isinstance(d[key], dict): d[key] = _remove_missing_and_underscored_keys(d[key]) - if d[key] == None or d[key] == {}: + if d[key] is None or d[key] == {}: del d[key] return d @@ -313,18 +295,12 @@ def _remove_missing_and_underscored_keys(d): def format_deploy(specified_version, deploy): ret = deploy - try: + with contextlib.suppress(Exception): placement_preferences = deploy['placement']['preferences'] ret['placement']['preferences'] = format_key_val_pairs(placement_preferences) - except Exception: - pass - - try: + with contextlib.suppress(Exception): labels = deploy['labels'] ret['labels'] = format_key_val_pairs(labels) - except Exception: - pass - ret = _remove_missing_and_underscored_keys(ret) return ret @@ -342,9 +318,7 @@ def format_services_version_one(specified_version, services, volumes, networks): for service in services: service_formatted = {} - image = service.get('image', None) - - if image: + if image := service.get('image', None): image_tag = "latest" try: @@ -353,92 +327,84 @@ def format_services_version_one(specified_version, services, volumes, networks): except KeyError: service_formatted['image'] = f"{image}" - try: + with contextlib.suppress(KeyError): if service['container_name']: service_formatted['container_name'] = service['container_name'] - except KeyError: - pass - - try: + with contextlib.suppress(KeyError): if service['restart']: service_formatted['restart'] = service['restart'] - except KeyError: - pass - - try: + with contextlib.suppress(KeyError): if service['command']: service_formatted['command'] = format_command_string(service['command']) - except KeyError: - pass - - try: + with contextlib.suppress(KeyError): if service['entrypoint']: service_formatted['entrypoint'] = format_command_string(service['entrypoint']) - except KeyError: - pass - - try: + with contextlib.suppress(KeyError): if service['working_dir']: service_formatted['working_dir'] = service['working_dir'] - except KeyError: - pass - - try: + with contextlib.suppress(KeyError): if service['ports']: service_formatted['ports'] = format_ports(service['ports']) - except KeyError: - pass - - try: - links = service.get('links', []) - if links: + with contextlib.suppress(KeyError): + if links := service.get('links', []): service_formatted['links'] = [] for link in links: for service_obj in services: if link == service_obj['uuid']: service_formatted['links'].append(f"{service_obj['name']}") - except KeyError: - pass - - try: + with contextlib.suppress(KeyError): if service['environment']: envs = service['environment'] service_formatted['environment'] = format_key_val_pairs(envs) - except KeyError: - pass - - try: + with contextlib.suppress(KeyError): service_volumes = service['volumes'] - formatted_volumes = format_volumes(service_volumes, volumes) - - if formatted_volumes: + if formatted_volumes := format_volumes(service_volumes, volumes): service_formatted['volumes'] = formatted_volumes else: del service_formatted['volumes'] - except KeyError: - pass - - try: - build = format_build(specified_version, service['build']) - if build: + with contextlib.suppress(KeyError): + if build := format_build(specified_version, service['build']): service_formatted['build'] = build - except KeyError: - pass - services_formatted[service['name']] = service_formatted return services_formatted -def format_services_version_three(specified_version, services, volumes, networks): +def get_service_by_label_key(key, services): + for service in services: + with contextlib.suppress(KeyError): + if key == service["labels"]["key"]: + return service + + return None + + +def get_connected_services(service_key, connections, services): + connected_services = [] + for connection in connections: + if service_key == connection[0]: + if connected_service := get_service_by_label_key(connection[1], services): + connected_services.append(connected_service) + return connected_services + + +def format_services_version_three(specified_version, services, connections, volumes, networks): services_formatted = {} for service in services: service_formatted = {} - - image = service.get('image', None) - - if image: + service_key = "" + + # add labels excluding certain keys + if labels := service.get('labels', {}): + clean_labels = {x: labels[x] for x in labels if x not in ["key"]} + if "key" in labels: + service_key = labels["key"] + if bool(clean_labels): + service_formatted['labels'] = clean_labels + + # image name + if image := service.get('image', None): image_tag = "latest" try: @@ -447,108 +413,53 @@ def format_services_version_three(specified_version, services, volumes, networks except KeyError: service_formatted['image'] = f"{image}" - try: + # dependencies + with contextlib.suppress(KeyError): + if connected_services := get_connected_services(service_key, connections, services): + service_formatted['depends_on'] = [] + for connected_service in connected_services: + service_formatted['depends_on'].append(f"{connected_service['name']}") + with contextlib.suppress(KeyError): if service['container_name']: service_formatted['container_name'] = service['container_name'] - except KeyError: - pass - - try: + with contextlib.suppress(KeyError): if service['restart']: service_formatted['restart'] = service['restart'] - except KeyError: - pass - - try: + with contextlib.suppress(KeyError): if service['command']: service_formatted['command'] = format_command_string(service['command']) - except KeyError: - pass - - try: + with contextlib.suppress(KeyError): if service['entrypoint']: service_formatted['entrypoint'] = format_command_string(service['entrypoint']) - except KeyError: - pass - - try: - working_dir_str = service['working_dir'] - if working_dir_str: + with contextlib.suppress(KeyError): + if working_dir_str := service['working_dir']: service_formatted['working_dir'] = working_dir_str - except KeyError: - pass - - try: + with contextlib.suppress(KeyError): if service['ports']: service_formatted['ports'] = format_ports(service['ports']) - except KeyError: - pass - - try: - if service['depends_on']: - depends_on = service['depends_on'] - service_formatted['depends_on'] = [] - for depends in depends_on: - for service_obj in services: - if depends == service_obj['uuid']: - service_formatted['depends_on'].append(f"{service_obj['name']}") - except KeyError: - pass - - try: + with contextlib.suppress(KeyError): if service['environment']: envs = service['environment'] service_formatted['environment'] = format_key_val_pairs(envs) - except KeyError: - pass - - try: + with contextlib.suppress(KeyError): service_volumes = service['volumes'] - formatted_volumes = format_volumes(service_volumes, volumes) - - if formatted_volumes: + if formatted_volumes := format_volumes(service_volumes, volumes): service_formatted['volumes'] = formatted_volumes else: del service_formatted['volumes'] - except KeyError: - pass - - try: - labels = service.get('labels', None) - - if labels: - service_formatted['labels'] = {} - for label in labels: - service_formatted['labels'][label['key']] = format_quotes(label['value']) - except KeyError: - pass - - try: + with contextlib.suppress(KeyError): service_networks = service.get('networks', []) - formatted_networks = format_networks(service_networks, networks) - - if formatted_networks: + if formatted_networks := format_networks(service_networks, networks): service_formatted['networks'] = formatted_networks else: del service_formatted['networks'] - except KeyError: - pass - - try: - build = format_build(specified_version, service['build']) - if build: + with contextlib.suppress(KeyError): + if build := format_build(specified_version, service['build']): service_formatted['build'] = build - except KeyError: - pass - if int(float(specified_version)) >= 3: - try: - deploy = format_deploy(specified_version, service['deploy']) - if deploy: + with contextlib.suppress(KeyError): + if deploy := format_deploy(specified_version, service['deploy']): service_formatted['deploy'] = deploy - except KeyError: - pass - services_formatted[service['name']] = service_formatted return services_formatted @@ -563,54 +474,55 @@ def FSlist(l): # concert list into flow-style (default is block style) return cs -def generate_dc(services, volumes, networks, secrets, configs, version="3", return_format='yaml'): - if return_format == 'yaml': - s = io.StringIO() - ret_yaml = YAML() - ret_yaml.indent(mapping=2, sequence=4, offset=2) - ret_yaml.explicit_start = True - specified_version = get_version(version) - base_version = int(specified_version) - - if services: - if base_version in [2, 3]: - ret_yaml.dump({'version': DoubleQuotedScalarString(specified_version)}, s) - ret_yaml.explicit_start = False - s.write('\n') - services_formatted = format_services_version_three(specified_version, services, volumes, networks) - ret_yaml.dump({'services': services_formatted}, s, transform=sequence_indent_four) - - if base_version == 1: - ret_yaml.dump({'version': DoubleQuotedScalarString(specified_version)}, s) - ret_yaml.explicit_start = False - s.write('\n') - services_formatted = format_services_version_one(specified_version, services, volumes, networks) - ret_yaml.dump(services_formatted, s, transform=sequence_indent_one) - - s.write('\n') +def generate_dc(services, connections, volumes, networks, secrets, configs, version="3", return_format='yaml'): + if return_format != 'yaml': + return - if base_version in [3, 2]: - if networks: - networks_formatted = format_networks_top_level(networks, version) - ret_yaml.dump({'networks': networks_formatted}, s) - s.write('\n') + s = io.StringIO() + ret_yaml = YAML() + ret_yaml.indent(mapping=2, sequence=4, offset=2) + ret_yaml.explicit_start = True + specified_version = get_version(version) + base_version = int(specified_version) - if volumes: - volumes_formatted = format_volumes_top_level(volumes, version) - ret_yaml.dump({'volumes': volumes_formatted}, s) - s.write('\n') - - if secrets: - ret_yaml.dump({'secrets': secrets}, s) + if services: + if base_version in {2, 3}: + ret_yaml.dump({'version': DoubleQuotedScalarString(specified_version)}, s) + ret_yaml.explicit_start = False s.write('\n') + services_formatted = format_services_version_three(specified_version, services, connections, volumes, networks) + ret_yaml.dump({'services': services_formatted}, s, transform=sequence_indent_four) - if configs: - ret_yaml.dump({'configs': configs}, s) + if base_version == 1: + ret_yaml.dump({'version': DoubleQuotedScalarString(specified_version)}, s) + ret_yaml.explicit_start = False s.write('\n') + services_formatted = format_services_version_one(specified_version, services, volumes, networks) + ret_yaml.dump(services_formatted, s, transform=sequence_indent_one) + + s.write('\n') + + if base_version in {3, 2} and networks: + networks_formatted = format_networks_top_level(networks, version) + ret_yaml.dump({'networks': networks_formatted}, s) + s.write('\n') + + if volumes: + volumes_formatted = format_volumes_top_level(volumes, version) + ret_yaml.dump({'volumes': volumes_formatted}, s) + s.write('\n') - s.seek(0) + if secrets: + ret_yaml.dump({'secrets': secrets}, s) + s.write('\n') - return s + if configs: + ret_yaml.dump({'configs': configs}, s) + s.write('\n') + + s.seek(0) + + return s def generate(cname): @@ -619,15 +531,11 @@ def generate(cname): try: cid = [x.short_id for x in c.containers.list() if cname == x.name or x.short_id in cname][0] except IndexError: - print("That container is not running.") sys.exit(1) cattrs = c.containers.get(cid).attrs - - - # Build yaml dict structure - cfile = {} + networks = {} cfile[cattrs['Name'][1:]] = {} ct = cfile[cattrs['Name'][1:]] @@ -672,7 +580,6 @@ def generate(cname): } networklist = c.networks.list() - networks = {} for network in networklist: if network.attrs['Name'] in values['networks'].keys(): networks[network.attrs['Name']] = {'external': (not network.attrs['Internal'])} @@ -687,7 +594,7 @@ def generate(cname): ports_value = [cattrs['HostConfig']['PortBindings'][key][0]['HostIp']+':'+cattrs['HostConfig']['PortBindings'][key][0]['HostPort']+':'+key for key in cattrs['HostConfig']['PortBindings']] # If bound ports found, don't use the 'expose' value. - if (ports_value != None) and (ports_value != "") and (ports_value != []) and (ports_value != 'null') and (ports_value != {}) and (ports_value != "default") and (ports_value != 0) and (ports_value != ",") and (ports_value != "no"): + if ports_value not in [None, "", [], 'null', {}, "default", 0, ",", "no"]: for index, port in enumerate(ports_value): if port[0] == ':': ports_value[index] = port[1:] @@ -701,9 +608,8 @@ def generate(cname): ports = None # Iterate through values to finish building yaml dict. - for key in values: - value = values[key] - if (value != None) and (value != "") and (value != []) and (value != 'null') and (value != {}) and (value != "default") and (value != 0) and (value != ",") and (value != "no"): + for key, value in values.items(): + if value not in [None, "", [], 'null', {}, "default", 0, ",", "no"]: ct[key] = value return cfile, networks @@ -716,7 +622,6 @@ def generate_uuid(): def random_string(string_length=10): """ Generate a random string of fixed length - :param string_length: integer :return: string """ @@ -730,8 +635,8 @@ def random_string(string_length=10): def generate_rand_string(): - rand_string = "".join(random.choice( - string.ascii_uppercase + - string.ascii_lowercase + - string.digits) for _ in range(16)) - return rand_string + return "".join( + random.choice( + string.ascii_uppercase + string.ascii_lowercase + string.digits + ) for _ in range(16) + ) \ No newline at end of file diff --git a/services/backend/src/api/views/view.py b/services/backend/src/api/views/view.py new file mode 100644 index 0000000..486d5b6 --- /dev/null +++ b/services/backend/src/api/views/view.py @@ -0,0 +1,9 @@ +from rest_framework.response import Response +from rest_framework import generics, status + + +class ViewGenericAPIView(generics.GenericAPIView): + permission_classes = [] + + def get(self, request): + return Response({}, status=status.HTTP_404_NOT_FOUND) diff --git a/src/backend/server/api/migrations/__init__.py b/services/backend/src/main/__init__.py similarity index 100% rename from src/backend/server/api/migrations/__init__.py rename to services/backend/src/main/__init__.py diff --git a/src/backend/server/server/asgi.py b/services/backend/src/main/asgi.py similarity index 58% rename from src/backend/server/server/asgi.py rename to services/backend/src/main/asgi.py index 2526a47..b32a08d 100644 --- a/src/backend/server/server/asgi.py +++ b/services/backend/src/main/asgi.py @@ -1,16 +1,16 @@ """ -ASGI config for server project. +ASGI config. It exposes the ASGI callable as a module-level variable named ``application``. For more information on this file, see -https://docs.djangoproject.com/en/3.0/howto/deployment/asgi/ +https://docs.djangoproject.com/en/4.0/howto/deployment/asgi/ """ import os from django.core.asgi import get_asgi_application -os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'server.settings') +os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'main.settings') application = get_asgi_application() diff --git a/services/backend/src/main/settings.py b/services/backend/src/main/settings.py new file mode 100644 index 0000000..c88cd56 --- /dev/null +++ b/services/backend/src/main/settings.py @@ -0,0 +1,197 @@ +""" +Django settings. + +Generated by 'django-admin startproject' using Django 4.0.4. + +For more information on this file, see +https://docs.djangoproject.com/en/4.0/topics/settings/ + +For the full list of settings and their values, see +https://docs.djangoproject.com/en/4.0/ref/settings/ +""" + +import os +from datetime import timedelta +from pathlib import Path + +# Build paths inside the project like this: BASE_DIR / 'subdir'. +BASE_DIR = Path(__file__).resolve().parent.parent + + +# Quick-start development settings - unsuitable for production +# See https://docs.djangoproject.com/en/4.0/howto/deployment/checklist/ + +# SECURITY WARNING: keep the secret key used in production secret! +SECRET_KEY = "django-insecure--i+bd*fda@!=_0yv$((3(@nruqvv(8c1c8no^+yjl%@b859f57" + +# SECURITY WARNING: don't run with debug turned on in production! +DEBUG = True + +ALLOWED_HOSTS = ["*"] + +CORS_URLS_REGEX = r"^/v1/.*$" +CORS_ORIGIN_ALLOW_ALL = True + +SITE_ID = 1 + +# Application definition + +INSTALLED_APPS = [ + "django.contrib.sites", + "django.contrib.admin", + "django.contrib.auth", + "django.contrib.contenttypes", + "django.contrib.sessions", + "django.contrib.messages", + "django.contrib.staticfiles", + "rest_framework", + "rest_framework.authtoken", + "dj_rest_auth", + "allauth", + "allauth.account", + "allauth.socialaccount", + "dj_rest_auth.registration", + "corsheaders", + "axes", + "organizations", + "api", +] + +MIDDLEWARE = [ + "django.middleware.security.SecurityMiddleware", + "django.contrib.sessions.middleware.SessionMiddleware", + "corsheaders.middleware.CorsMiddleware", + "django.middleware.common.CommonMiddleware", + "django.middleware.csrf.CsrfViewMiddleware", + "django.contrib.auth.middleware.AuthenticationMiddleware", + "django.contrib.messages.middleware.MessageMiddleware", + "django.middleware.clickjacking.XFrameOptionsMiddleware", + "axes.middleware.AxesMiddleware", +] + +ROOT_URLCONF = "main.urls" + +TEMPLATES = [ + { + "BACKEND": "django.template.backends.django.DjangoTemplates", + "DIRS": ["templates"], + "APP_DIRS": True, + "OPTIONS": { + "context_processors": [ + "django.template.context_processors.debug", + "django.template.context_processors.request", + "django.contrib.auth.context_processors.auth", + "django.contrib.messages.context_processors.messages", + ], + }, + }, +] + +WSGI_APPLICATION = "main.wsgi.application" + + +# Database +# https://docs.djangoproject.com/en/4.0/ref/settings/#databases + +DB_REMOTE = os.environ.get("DB_REMOTE", "False").lower() == "true" + +if DB_REMOTE: + DATABASES = { + "default": { + "ENGINE": "django.db.backends.postgresql", + "HOST": os.getenv("DB_HOST", None), + "NAME": os.getenv("DB_NAME", None), + "USER": os.getenv("DB_USER", None), + "PASSWORD": os.getenv("DB_PASS", None), + "PORT": 5432, + } + } +else: + DATABASES = { + "default": { + "ENGINE": "django.db.backends.postgresql", + "HOST": "postgres", + "NAME": "postgres", + "USER": "postgres", + "PASSWORD": "postgres", + "PORT": 5432, + } + } + + +# Password validation +# https://docs.djangoproject.com/en/4.0/ref/settings/#auth-password-validators + +AUTH_PASSWORD_VALIDATORS = [ + { + "NAME": "django.contrib.auth.password_validation.UserAttributeSimilarityValidator", + }, + { + "NAME": "django.contrib.auth.password_validation.MinimumLengthValidator", + }, + { + "NAME": "django.contrib.auth.password_validation.CommonPasswordValidator", + }, + { + "NAME": "django.contrib.auth.password_validation.NumericPasswordValidator", + }, +] + + +# Internationalization +# https://docs.djangoproject.com/en/4.0/topics/i18n/ + +LANGUAGE_CODE = "en-us" + +TIME_ZONE = "UTC" + +USE_I18N = True + +USE_TZ = True + + +# Static files (CSS, JavaScript, Images) +# https://docs.djangoproject.com/en/4.0/howto/static-files/ + +STATIC_URL = "static/" + +# Default primary key field type +# https://docs.djangoproject.com/en/4.0/ref/settings/#default-auto-field + +DEFAULT_AUTO_FIELD = "django.db.models.BigAutoField" + + +AUTHENTICATION_BACKENDS = [ + "axes.backends.AxesBackend", + "django.contrib.auth.backends.ModelBackend", +] + +REST_FRAMEWORK = { + "DEFAULT_AUTHENTICATION_CLASSES": [ + "dj_rest_auth.jwt_auth.JWTCookieAuthentication", + ], + "DEFAULT_PAGINATION_CLASS": "rest_framework.pagination.LimitOffsetPagination", + "DEFAULT_PERMISSION_CLASSES": ["rest_framework.permissions.IsAuthenticated"], + "DEFAULT_RENDERER_CLASSES": ["rest_framework.renderers.JSONRenderer"], + "PAGE_SIZE": 300, + "TEST_REQUEST_DEFAULT_FORMAT": "json", +} + +if DEBUG: + REST_FRAMEWORK["DEFAULT_RENDERER_CLASSES"].append( + "rest_framework.renderers.BrowsableAPIRenderer" + ) + + +# allauth +ACCOUNT_EMAIL_VERIFICATION = "none" +ACCOUNT_PRESERVE_USERNAME_CASING = False + +# dj_rest_auth +REST_USE_JWT = True + +# simple jwt +SIMPLE_JWT = { + "ACCESS_TOKEN_LIFETIME": timedelta(days=15), + "REFRESH_TOKEN_LIFETIME": timedelta(days=30), +} diff --git a/services/backend/src/main/urls.py b/services/backend/src/main/urls.py new file mode 100644 index 0000000..9842e3d --- /dev/null +++ b/services/backend/src/main/urls.py @@ -0,0 +1,18 @@ +from typing import Callable, Optional + +from django.contrib import admin +from django.urls import URLPattern, include, path + +from api.routing import api_urls + + +def opt_slash_path(route: str, view: Callable, name: Optional[str] = None) -> URLPattern: + """Catches path with or without trailing slash, taking into account query param and hash.""" + # Ignoring the type because while name can be optional on re_path, mypy doesn't agree + return re_path(fr"^{route}/?(?:[?#].*)?$", view, name=name) # type: ignore + + +urlpatterns = [ + path('admin/', admin.site.urls), + path("v1/", include(api_urls)), +] diff --git a/src/backend/server/server/wsgi.py b/services/backend/src/main/wsgi.py similarity index 58% rename from src/backend/server/server/wsgi.py rename to services/backend/src/main/wsgi.py index c65f7e2..05379bc 100644 --- a/src/backend/server/server/wsgi.py +++ b/services/backend/src/main/wsgi.py @@ -1,16 +1,16 @@ """ -WSGI config for server project. +WSGI config. It exposes the WSGI callable as a module-level variable named ``application``. For more information on this file, see -https://docs.djangoproject.com/en/3.0/howto/deployment/wsgi/ +https://docs.djangoproject.com/en/4.0/howto/deployment/wsgi/ """ import os from django.core.wsgi import get_wsgi_application -os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'server.settings') +os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'main.settings') application = get_wsgi_application() diff --git a/src/backend/server/manage.py b/services/backend/src/manage.py similarity index 84% rename from src/backend/server/manage.py rename to services/backend/src/manage.py index 1c81878..fbda2b3 100755 --- a/src/backend/server/manage.py +++ b/services/backend/src/manage.py @@ -5,7 +5,8 @@ import sys def main(): - os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'server.settings') + """Run administrative tasks.""" + os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'main.settings') try: from django.core.management import execute_from_command_line except ImportError as exc: diff --git a/src/backend/server/server/__init__.py b/services/backend/src/organizations/__init__.py similarity index 100% rename from src/backend/server/server/__init__.py rename to services/backend/src/organizations/__init__.py diff --git a/services/backend/src/organizations/admin.py b/services/backend/src/organizations/admin.py new file mode 100644 index 0000000..29b8e4c --- /dev/null +++ b/services/backend/src/organizations/admin.py @@ -0,0 +1,9 @@ +from django.contrib import admin +from .models import Organization + + +class OrganizationAdmin(admin.ModelAdmin): + list_display = ("id", "name", "created_at", "updated_at") + + +admin.site.register(Organization, OrganizationAdmin) diff --git a/services/backend/src/organizations/apps.py b/services/backend/src/organizations/apps.py new file mode 100644 index 0000000..0c9a5d9 --- /dev/null +++ b/services/backend/src/organizations/apps.py @@ -0,0 +1,6 @@ +from django.apps import AppConfig + + +class OrganizationsConfig(AppConfig): + default_auto_field = 'django.db.models.BigAutoField' + name = 'organizations' diff --git a/services/backend/src/organizations/migrations/0001_initial.py b/services/backend/src/organizations/migrations/0001_initial.py new file mode 100644 index 0000000..156c4df --- /dev/null +++ b/services/backend/src/organizations/migrations/0001_initial.py @@ -0,0 +1,31 @@ +# Generated by Django 4.0.4 on 2022-05-31 14:47 + +from django.conf import settings +from django.db import migrations, models + + +class Migration(migrations.Migration): + + initial = True + + dependencies = [ + migrations.swappable_dependency(settings.AUTH_USER_MODEL), + ] + + operations = [ + migrations.CreateModel( + name='Organization', + fields=[ + ('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), + ('name', models.CharField(blank=True, max_length=255, null=True)), + ('created_at', models.DateTimeField(auto_now_add=True)), + ('updated_at', models.DateTimeField(auto_now=True)), + ('users', models.ManyToManyField(related_name='orgs', to=settings.AUTH_USER_MODEL)), + ], + options={ + 'verbose_name': 'Organization', + 'verbose_name_plural': 'Organizations', + 'ordering': ['-created_at'], + }, + ), + ] diff --git a/src/backend/server/utils/__init__.py b/services/backend/src/organizations/migrations/__init__.py similarity index 100% rename from src/backend/server/utils/__init__.py rename to services/backend/src/organizations/migrations/__init__.py diff --git a/services/backend/src/organizations/models.py b/services/backend/src/organizations/models.py new file mode 100644 index 0000000..7a7fab1 --- /dev/null +++ b/services/backend/src/organizations/models.py @@ -0,0 +1,50 @@ +from allauth.account.signals import user_signed_up, user_logged_in +from django.db.models.signals import pre_delete +from django.contrib.auth import get_user_model +from django.dispatch import receiver +from django.db import models + + +User = get_user_model() + + +class Organization(models.Model): + name = models.CharField(max_length=255, blank=True, null=True) + users = models.ManyToManyField(User, related_name="orgs") + created_at: models.DateTimeField = models.DateTimeField(auto_now_add=True) + updated_at: models.DateTimeField = models.DateTimeField(auto_now=True) + + class Meta: + verbose_name = "Organization" + verbose_name_plural = "Organizations" + ordering = ["-created_at"] + + def total_members(self): + return self.users.count() + + def is_member(self, user): + return user in self.users.all() + + def add_user(self, user): + return self.users.add(user) + + def remove_user(self, user): + return self.users.remove(user) + + def __str__(self): + return f"{self.name}" + + +@receiver(user_signed_up) +def handler(sender, request, user, **kwargs): + org_name = f"{user.username.lower()}-org" + org = Organization.objects.create(name=org_name) + org.add_user(user) + + +@receiver(pre_delete, sender=User) +def handler(instance, **kwargs): + for org in instance.orgs.all(): + org.remove_user(instance) + if org.total_members() == 0: + org.delete() diff --git a/services/backend/src/organizations/tests.py b/services/backend/src/organizations/tests.py new file mode 100644 index 0000000..7ce503c --- /dev/null +++ b/services/backend/src/organizations/tests.py @@ -0,0 +1,3 @@ +from django.test import TestCase + +# Create your tests here. diff --git a/services/backend/src/organizations/utils.py b/services/backend/src/organizations/utils.py new file mode 100644 index 0000000..11f44f1 --- /dev/null +++ b/services/backend/src/organizations/utils.py @@ -0,0 +1,4 @@ +def get_user_org(user): + user_orgs = user.orgs.all() + if user_orgs.count(): + return user_orgs[0] diff --git a/services/backend/src/organizations/views.py b/services/backend/src/organizations/views.py new file mode 100644 index 0000000..91ea44a --- /dev/null +++ b/services/backend/src/organizations/views.py @@ -0,0 +1,3 @@ +from django.shortcuts import render + +# Create your views here. diff --git a/services/frontend/.env.development b/services/frontend/.env.development new file mode 100644 index 0000000..908d763 --- /dev/null +++ b/services/frontend/.env.development @@ -0,0 +1 @@ +REACT_APP_API_SERVER=http://localhost:9001/v1 \ No newline at end of file diff --git a/services/frontend/.env.production b/services/frontend/.env.production new file mode 100644 index 0000000..908d763 --- /dev/null +++ b/services/frontend/.env.production @@ -0,0 +1 @@ +REACT_APP_API_SERVER=http://localhost:9001/v1 \ No newline at end of file diff --git a/services/frontend/Dockerfile b/services/frontend/Dockerfile new file mode 100644 index 0000000..12a44bd --- /dev/null +++ b/services/frontend/Dockerfile @@ -0,0 +1,12 @@ +FROM node:16 as build + +WORKDIR /build +COPY ./services/frontend/ . +RUN npm install +RUN npm run build + +FROM nginx:stable-alpine +COPY --from=build /build/build /usr/share/nginx/html +COPY --from=build /build/configs/nginx/nginx.conf /etc/nginx/conf.d/default.conf +EXPOSE 80 +CMD ["nginx", "-g", "daemon off;"] diff --git a/configs/nginx/localhost.conf b/services/frontend/configs/nginx/localhost.conf similarity index 94% rename from configs/nginx/localhost.conf rename to services/frontend/configs/nginx/localhost.conf index 07028d1..91d7479 100644 --- a/configs/nginx/localhost.conf +++ b/services/frontend/configs/nginx/localhost.conf @@ -12,7 +12,7 @@ server { charset utf-8; location /static { - alias /home/backend/static; + alias /home/server/static; } location /admin { diff --git a/configs/nginx/nginx.conf b/services/frontend/configs/nginx/nginx.conf similarity index 100% rename from configs/nginx/nginx.conf rename to services/frontend/configs/nginx/nginx.conf diff --git a/src/backend/configs/uwsgi/uwsgi_params b/services/frontend/configs/nginx/uwsgi_params similarity index 100% rename from src/backend/configs/uwsgi/uwsgi_params rename to services/frontend/configs/nginx/uwsgi_params diff --git a/services/frontend/package.json b/services/frontend/package.json new file mode 100644 index 0000000..0561f6e --- /dev/null +++ b/services/frontend/package.json @@ -0,0 +1,83 @@ +{ + "name": "ctk-frontend", + "version": "0.1.0", + "private": true, + "dependencies": { + "@codemirror/autocomplete": "^0.19.0", + "@codemirror/closebrackets": "^0.19.0", + "@codemirror/commands": "^0.19.0", + "@codemirror/comment": "^0.19.0", + "@codemirror/fold": "^0.19.0", + "@codemirror/history": "^0.19.0", + "@codemirror/lang-json": "^0.19.2", + "@codemirror/lang-python": "^0.19.0", + "@codemirror/legacy-modes": "^0.19.0", + "@codemirror/lint": "^0.19.0", + "@codemirror/matchbrackets": "^0.19.0", + "@codemirror/rectangular-selection": "^0.19.0", + "@codemirror/search": "^0.19.0", + "@codemirror/stream-parser": "^0.19.9", + "@codemirror/view": "^0.19.0", + "@headlessui/react": "^1.6.4", + "@heroicons/react": "^1.0.5", + "@jsplumb/browser-ui": "^5.5.2", + "@jsplumb/common": "^5.5.2", + "@jsplumb/connector-bezier": "^5.5.2", + "@jsplumb/core": "^5.5.2", + "@jsplumb/util": "^5.5.2", + "@tailwindcss/forms": "^0.4.0", + "@tailwindcss/typography": "^0.5.1", + "@testing-library/jest-dom": "^5.16.2", + "@testing-library/react": "^12.1.2", + "@testing-library/user-event": "^13.5.0", + "codemirror": "^5.65.5", + "d3": "^7.3.0", + "formik": "^2.2.9", + "lodash": "^4.17.21", + "react": "^17.0.2", + "react-dom": "^17.0.2", + "react-hot-toast": "^2.2.0", + "react-router-dom": "^6.3.0", + "react-scripts": "5.0.0", + "tailwindcss": "^3.0.19", + "typescript": "^4.5.5", + "uuid": "^8.3.2", + "web-vitals": "^2.1.4", + "yaml": "^1.10.2" + }, + "scripts": { + "start": "react-scripts start", + "build": "react-scripts build", + "test": "react-scripts test", + "eject": "react-scripts eject" + }, + "eslintConfig": { + "extends": [ + "react-app", + "react-app/jest" + ] + }, + "browserslist": { + "production": [ + ">0.2%", + "not dead", + "not op_mini all" + ], + "development": [ + "last 1 chrome version", + "last 1 firefox version", + "last 1 safari version" + ] + }, + "devDependencies": { + "@types/d3": "^7.1.0", + "@types/jest": "^27.4.0", + "@types/lodash": "^4.14.178", + "@types/node": "^16.11.22", + "@types/react": "^17.0.45", + "@types/react-dom": "^17.0.11", + "@types/uuid": "^8.3.4", + "autoprefixer": "^10.4.2", + "postcss": "^8.4.6" + } +} diff --git a/services/frontend/postcss.config.js b/services/frontend/postcss.config.js new file mode 100644 index 0000000..8567b4c --- /dev/null +++ b/services/frontend/postcss.config.js @@ -0,0 +1,6 @@ +module.exports = { + plugins: { + tailwindcss: {}, + autoprefixer: {}, + }, +}; \ No newline at end of file diff --git a/services/frontend/public/favicon.ico b/services/frontend/public/favicon.ico new file mode 100644 index 0000000..a11777c Binary files /dev/null and b/services/frontend/public/favicon.ico differ diff --git a/services/frontend/public/index.html b/services/frontend/public/index.html new file mode 100644 index 0000000..aa069f2 --- /dev/null +++ b/services/frontend/public/index.html @@ -0,0 +1,43 @@ + + + + + + + + + + + + + React App + + + +
+ + + diff --git a/services/frontend/public/manifest.json b/services/frontend/public/manifest.json new file mode 100644 index 0000000..ca4ee33 --- /dev/null +++ b/services/frontend/public/manifest.json @@ -0,0 +1,15 @@ +{ + "short_name": "Visual Argo Workflows", + "name": "Visual Argo Workflows", + "icons": [ + { + "src": "favicon.ico", + "sizes": "64x64 32x32 24x24 16x16", + "type": "image/x-icon" + } + ], + "start_url": ".", + "display": "standalone", + "theme_color": "#000000", + "background_color": "#ffffff" +} diff --git a/services/frontend/public/robots.txt b/services/frontend/public/robots.txt new file mode 100644 index 0000000..e9e57dc --- /dev/null +++ b/services/frontend/public/robots.txt @@ -0,0 +1,3 @@ +# https://www.robotstxt.org/robotstxt.html +User-agent: * +Disallow: diff --git a/services/frontend/src/App.test.tsx b/services/frontend/src/App.test.tsx new file mode 100644 index 0000000..1f03afe --- /dev/null +++ b/services/frontend/src/App.test.tsx @@ -0,0 +1,8 @@ +import { render, screen } from '@testing-library/react'; +import App from './App'; + +test('renders learn react link', () => { + render(); + const linkElement = screen.getByText(/learn react/i); + expect(linkElement).toBeInTheDocument(); +}); diff --git a/services/frontend/src/App.tsx b/services/frontend/src/App.tsx new file mode 100644 index 0000000..bbd0745 --- /dev/null +++ b/services/frontend/src/App.tsx @@ -0,0 +1,84 @@ +import { useReducer, useEffect } from "react"; +import { Routes, Route } from "react-router-dom"; +import { Toaster } from "react-hot-toast"; +import { LOCAL_STORAGE } from "./constants"; +import { reducer, initialState } from "./reducers"; +import { useLocalStorageAuth } from "./hooks/auth"; +import { checkHttpStatus } from "./services/helpers"; +import { authSelf } from "./reducers"; +import { refresh, self } from "./services/auth"; + +import Project from "./components/Project"; +import Profile from "./components/Profile"; +import Signup from "./components/Auth/Signup"; +import Login from "./components/Auth/Login"; + +import { ProtectedRouteProps } from "./partials/ProtectedRoute"; +import ProtectedRoute from "./partials/ProtectedRoute"; + +import "./index.css"; + +export default function App() { + const [state, dispatch] = useReducer(reducer, initialState); + const auth = useLocalStorageAuth(); + const isAuthenticated = !!(auth && Object.keys(auth).length); + + const defaultProtectedRouteProps: Omit = { + isAuthenticated: isAuthenticated, + authenticationPath: '/login', + }; + + useEffect(() => { + if (isAuthenticated) { + self() + .then(checkHttpStatus) + .then(data => { + dispatch(authSelf(data)); + }) + .catch(err => { + // since auth is set in localstorage, + // try to refresh the existing token, + // on error clear localstorage + if (err.status === 401) { + err.text().then((text: string) => { + const textObj = JSON.parse(text); + if (textObj.code === "user_not_found") { + localStorage.removeItem(LOCAL_STORAGE); + } + }) + + refresh() + .then(checkHttpStatus) + .then(data => { + const localData = localStorage.getItem(LOCAL_STORAGE); + + if (localData) { + const localDataParsed = JSON.parse(localData); + if (localDataParsed && Object.keys(localDataParsed).length) { + localDataParsed.access_token = data.access; + localStorage.setItem(LOCAL_STORAGE, JSON.stringify(localDataParsed)) + } + } + }) + .catch(err => { + localStorage.removeItem(LOCAL_STORAGE); + }) + } + }) + } + }, [dispatch, isAuthenticated]); + + return ( +
+ + + } /> + } /> + + } />} /> + } /> + } /> + +
+ ); +} diff --git a/services/frontend/src/components/Auth/Login/index.tsx b/services/frontend/src/components/Auth/Login/index.tsx new file mode 100644 index 0000000..18a6b77 --- /dev/null +++ b/services/frontend/src/components/Auth/Login/index.tsx @@ -0,0 +1,165 @@ +import { useState } from "react"; +import { useFormik } from "formik"; +import { Link, useNavigate } from "react-router-dom"; +import DarkModeSwitch from "../../../components/DarkModeSwitch"; +import Spinner from "../../../components/Spinner"; +import { toaster } from "../../../utils"; +import { checkHttpStatus } from "../../../services/helpers"; +import { logIn } from "../../../services/auth"; +import { LOCAL_STORAGE } from "../../../constants"; +import { authLoginSuccess } from "../../../reducers"; + +interface IProfileProps { + dispatch: any; +} + +const Login = (props: IProfileProps) => { + const { dispatch } = props; + const [loggingIn, setLoggingIn] = useState(false); + const navigate = useNavigate(); + + const formik = useFormik({ + initialValues: { + username: "", + password: "" + }, + onSubmit: values => { + const username = values.username; + const password = values.password; + setLoggingIn(true); + if (username && password) { + logIn(username, password) + .then(checkHttpStatus) + .then(data => { + localStorage.setItem(LOCAL_STORAGE, JSON.stringify({ + "access_token": data.access_token, + "refresh_token": data.refresh_token + })) + dispatch(authLoginSuccess(data)) + navigate("/"); + }) + .catch(err => { + if (err) { + err.text().then((e: string) => { + toaster(e, "error"); + }) + } + }).finally(() => { + setLoggingIn(false); + }) + } + }, + }); + + return ( + <> + +
+
+
+
+ +
+
+
+ +
+

+ Sign in +

+
+
+ +
+ +
+ +
+ +
+ +
+
+ +
+ + Create account + +
+
+
+ + ) +} + +export default Login; diff --git a/services/frontend/src/components/Auth/Signup/index.tsx b/services/frontend/src/components/Auth/Signup/index.tsx new file mode 100644 index 0000000..7275511 --- /dev/null +++ b/services/frontend/src/components/Auth/Signup/index.tsx @@ -0,0 +1,234 @@ +import { useState } from "react"; +import { useFormik } from "formik"; +import { Link, useNavigate } from "react-router-dom"; +import DarkModeSwitch from "../../../components/DarkModeSwitch"; +import Spinner from "../../../components/Spinner"; +import { toaster } from "../../../utils"; +import { checkHttpStatus } from "../../../services/helpers"; +import { signup } from "../../../services/auth"; +import { LOCAL_STORAGE } from "../../../constants"; +import { authLoginSuccess } from "../../../reducers"; + +interface IProfileProps { + dispatch: any; +} + +const Signup = (props: IProfileProps) => { + const { dispatch } = props; + const [signingUp, setSigningUp] = useState(false); + const navigate = useNavigate(); + + const formik = useFormik({ + initialValues: { + username: "", + email: "", + password: "", + confirmPassword: "" + }, + onSubmit: values => { + const username = values.username; + const email = values.email; + const password1 = values.password; + const password2 = values.confirmPassword; + + setSigningUp(true); + if (username && email && password1 && password2) { + signup(username, email, password1, password2) + .then(checkHttpStatus) + .then(data => { + localStorage.setItem(LOCAL_STORAGE, JSON.stringify({ + "access_token": data.access_token, + "refresh_token": data.refresh_token + })) + dispatch(authLoginSuccess(data)) + navigate("/"); + }) + .catch(err => { + if (err) { + err.text().then((e: string) => { + toaster(e, "error"); + }) + } + }).finally(() => { + setSigningUp(false); + }) + } + }, + }); + + return ( + <> + +
+
+
+
+ +
+
+
+ +
+

+ Create account +

+
+
+ +
+ +
+ +
+ +
+ +
+ +
+ +
+ +
+ +
+
+ +
+ + Already have an account? + +
+
+
+ + ) +} + +export default Signup; diff --git a/services/frontend/src/components/Canvas/index.tsx b/services/frontend/src/components/Canvas/index.tsx new file mode 100644 index 0000000..4c6f712 --- /dev/null +++ b/services/frontend/src/components/Canvas/index.tsx @@ -0,0 +1,379 @@ +import { FC, useState, useEffect, createRef, useRef } from "react"; +import { useMemo } from 'react'; +import { debounce } from 'lodash'; +import { Dictionary, values } from "lodash"; +import { v4 as uuidv4 } from "uuid"; +import YAML from "yaml"; +import { PlusIcon } from "@heroicons/react/solid"; +import { + nodeCreated, + nodeDeleted, + nodeUpdated, + connectionDetached, + connectionAttached, + position +} from "../../reducers"; +import Remove from "../Remove"; +import ModalServiceCreate from "../Modal/Service/Create"; +import ModalServiceEdit from "../Modal/Service/Edit"; +import ModalCreate from "../Modal/Node/Create"; +import ModalEdit from "../Modal/Node/Edit"; +import { useClickOutside } from "../../utils/clickOutside"; +import { IClientNodeItem, IGraphData } from "../../types"; +import { nodeLibraries } from "../../utils/data/libraries"; +import { getClientNodeItem, flattenLibraries, ensure } from "../../utils"; +import { flattenGraphData } from "../../utils/generators"; +import { generateHttp } from "../../services/generate"; +import { checkHttpStatus } from "../../services/helpers"; +import { useJsPlumb } from "../useJsPlumb"; +import CodeEditor from "../CodeEditor"; + +const CANVAS_ID: string = "canvas-container-" + uuidv4(); + +interface ICanvasProps { + state: any; + dispatch: any; + height: number; +} + +export const Canvas: FC = (props) => { + const { state, dispatch, height } = props; + + const [language, setLanguage] = useState("yaml"); + const [scale, setScale] = useState(1); + const [generatedCode, setGeneratedCode] = useState(); + const [formattedCode, setFormattedCode] = useState(""); + const [instanceNodes, setInstanceNodes] = useState(state.nodes); + const [instanceConnections, setInstanceConnections] = useState(state.connections); + const [copyText, setCopyText] = useState("Copy"); + const [selectedNode, setSelectedNode] = useState(null); + const [showModalCreateStep, setShowModalCreateStep] = useState(false); + const [showModalEditStep, setShowModalEditStep] = useState(false); + const [showModalCreate, setShowModalCreate] = useState(false); + const [showModalEdit, setShowModalEdit] = useState(false); + + const [dragging, setDragging] = useState(false); + const [_scale, _setScale] = useState(1); + const [_left, _setLeft] = useState(0); + const [_top, _setTop] = useState(0); + const [_initX, _setInitX] = useState(0); + const [_initY, _setInitY] = useState(0); + const [containerCallbackRef, setZoom, setStyle, removeEndpoint] = useJsPlumb( + instanceNodes, + instanceConnections, + ((graphData: IGraphData) => onGraphUpdate(graphData)), + ((positionData: any) => onEndpointPositionUpdate(positionData)), + ((connectionData: any) => onConnectionAttached(connectionData)), + ((connectionData: any) => onConnectionDetached(connectionData)) + ); + + const drop = createRef(); + const stateRef = useRef>(); + + let translateWidth = (document.documentElement.clientWidth * (1 - scale)) / 2; + let translateHeight = ((document.documentElement.clientHeight - 64) * (1 - scale)) / 2; + + stateRef.current = state.nodes; + + useClickOutside(drop, () => { + setShowModalCreateStep(false); + setShowModalCreate(false); + }); + + useEffect(() => { + setScale(_scale); + }, [_scale]); + + const debouncedOnGraphUpdate = useMemo(() => debounce((graphData) => { + const flatData = flattenGraphData(graphData); + generateHttp(flatData) + .then(checkHttpStatus) + .then(data => { + if (data['code'].length) { + for (var i = 0; i < data['code'].length; ++i) { + data['code'][i] = data['code'][i].replace(/(\r\n|\n|\r)/gm, ""); + } + + const code = data['code'].join("\n"); + setGeneratedCode(code); + } + }) + .catch(err => { + + }) + .finally(() => { + + }); + }, 450), []); + + const debouncedOnCodeChange = useMemo(() => debounce((code: string) => { + //formik.setFieldValue("code", e, false); + }, 700), []); + + const zoomIn = () => { + setScale((scale) => scale + 0.1); + } + + const zoomOut = () => { + setScale((scale) => scale - 0.1); + } + + const copy = () => { + navigator.clipboard.writeText(formattedCode); + setCopyText("Copied"); + + setTimeout(() => { + setCopyText("Copy"); + }, 300); + } + + const onAddEndpoint = (values: any) => { + let sections = flattenLibraries(nodeLibraries); + let clientNodeItem = getClientNodeItem(values, ensure(sections.find((l) => l.Type === values.type))); + clientNodeItem.position = { left: 60, top: 30 }; + dispatch(nodeCreated(clientNodeItem)); + } + + const onUpdateEndpoint = (nodeItem: IClientNodeItem) => { + dispatch(nodeUpdated(nodeItem)); + } + + const onRemoveEndpoint = (key: string) => { + const nodeToRemove = instanceNodes[key]; + removeEndpoint(nodeToRemove); + dispatch(nodeDeleted(nodeToRemove)); + } + + const onEndpointPositionUpdate = (positionData: any) => { + if (stateRef.current) { + const node = stateRef.current[positionData.key]; + node.position = {...node.position, ...positionData.position}; + dispatch(nodeUpdated(node)); + } + }; + + const onConnectionDetached = (data: any) => { + dispatch(connectionDetached(data)); + } + + const onConnectionAttached = (data: any) => { + dispatch(connectionAttached(data)); + } + + const onGraphUpdate = (graphData: any) => { + debouncedOnGraphUpdate(graphData); + }; + + const onCodeUpdate = (code: string) => { + debouncedOnCodeChange(code); + }; + + const onCanvasMousewheel = (e: any) => { + if (e.deltaY < 0) { + _setScale(_scale + _scale * 0.25); + } + + if (e.deltaY > 0) { + _setScale(_scale - _scale * 0.25); + } + } + + const onCanvasMouseUpLeave = (e: any) => { + if (dragging) { + let left = _left + e.pageX - _initX; + let top = _top + e.pageY - _initY; + + _setLeft(left); + _setTop(top); + setDragging(false); + dispatch(position({ + left: left, + top: top + })); + } + } + + const onCanvasMouseMove = (e: any) => { + if (!dragging) { + return; + } + + const styles = { + "left": _left + e.pageX - _initX + 'px', + "top": _top + e.pageY - _initY + 'px' + } + + setStyle(styles); + } + + const onCanvasMouseDown = (e: any) => { + _setInitX(e.pageX); + _setInitY(e.pageY); + setDragging(true); + } + + useEffect(() => { + if (!generatedCode) { + return; + } + + if (language === "json") { + setFormattedCode(JSON.stringify(YAML.parse(generatedCode), null, 2)); + } + + if (language === "yaml") { + setFormattedCode(generatedCode); + } + }, [language, generatedCode]); + + useEffect(() => { + setInstanceNodes(state.nodes); + }, [state.nodes]); + + useEffect(() => { + setInstanceConnections(state.connections); + }, [state.connections]); + + useEffect(() => { + setZoom(scale); + dispatch(position({ + scale: scale + })); + }, [dispatch, scale, setZoom]); + + useEffect(() => { + const styles = { + "left": _left + 'px', + "top": _top + 'px' + } + + setStyle(styles); + }, [_left, _top, setStyle]); + + useEffect(() => { + _setTop(state.canvasPosition.top); + _setLeft(state.canvasPosition.left); + _setScale(state.canvasPosition.scale); + }, [state.canvasPosition]); + + return ( + <> + {showModalCreateStep + ? setShowModalCreateStep(false)} + onAddEndpoint={(values: any) => onAddEndpoint(values)} + /> + : null + } + + {showModalEditStep + ? setShowModalEditStep(false)} + onUpdateEndpoint={(values: any) => onUpdateEndpoint(values)} + /> + : null + } + + {showModalCreate + ? setShowModalCreate(false)} + onAddEndpoint={(values: any) => onAddEndpoint(values)} + /> + : null + } + + {showModalEdit + ? setShowModalEdit(false)} + onUpdateEndpoint={(nodeItem: IClientNodeItem) => onUpdateEndpoint(nodeItem)} + /> + : null + } + + {instanceNodes && + <> +
+
+
+
+ + + + + +
+
+ +
{ event.stopPropagation(); event.preventDefault(); }} + > +
+ {(values(instanceNodes).length > 0) && ( + <> + {values(instanceNodes).map((x) => ( +
+
+
+ {x.configuration.prettyName} +
+
+ {x.configuration.prettyName} +
+
+
+ ))} + + )} +
+
+
+
+ +
+
+ + + +
+ + {onCodeUpdate(e)}} + disabled={false} + lineWrapping={false} + height={height} + /> +
+ + } + + ); +}; diff --git a/services/frontend/src/components/CodeEditor/index.tsx b/services/frontend/src/components/CodeEditor/index.tsx new file mode 100644 index 0000000..c198d72 --- /dev/null +++ b/services/frontend/src/components/CodeEditor/index.tsx @@ -0,0 +1,122 @@ +import { StreamLanguage } from "@codemirror/stream-parser"; +import { EditorState } from "@codemirror/state"; +import { + EditorView, + highlightSpecialChars, + drawSelection, + highlightActiveLine, + keymap, +} from '@codemirror/view' +import { jsonLanguage } from "@codemirror/lang-json"; +import { yaml } from "@codemirror/legacy-modes/mode/yaml"; + +import { history, historyKeymap } from '@codemirror/history' +import { foldGutter, foldKeymap } from '@codemirror/fold' +import { bracketMatching } from '@codemirror/matchbrackets' +import { closeBrackets, closeBracketsKeymap } from '@codemirror/closebrackets' +import { searchKeymap, highlightSelectionMatches } from '@codemirror/search' +import { autocompletion, completionKeymap } from '@codemirror/autocomplete' +import { rectangularSelection } from '@codemirror/rectangular-selection' +import { commentKeymap } from '@codemirror/comment' +import { lintKeymap } from '@codemirror/lint' +import { indentOnInput, LanguageSupport } from '@codemirror/language' +import { lineNumbers } from '@codemirror/gutter'; +import { defaultKeymap, indentMore, indentLess } from '@codemirror/commands' +import { defaultHighlightStyle } from '@codemirror/highlight' +import { solarizedDark } from './themes/ui/dark' +import darkHighlightStyle from './themes/highlight/dark' +import useCodeMirror from "./useCodeMirror"; + + +interface ICodeEditorProps { + data: string; + language: string; + onChange: any; + disabled: boolean; + lineWrapping: boolean; + height: number +} + +const languageExtensions: any = { + json: [new LanguageSupport(jsonLanguage)], + yaml: [StreamLanguage.define(yaml)], + blank: undefined +} + +const themeExtensions = { + light: [defaultHighlightStyle], + dark: [solarizedDark] +} + +const highlightExtensions = { + dark: darkHighlightStyle +} + +const CodeEditor = (props: ICodeEditorProps) => { + const { data, language, onChange, disabled, lineWrapping, height } = props; + const extensions = [[ + lineNumbers(), + highlightSpecialChars(), + history(), + foldGutter(), + drawSelection(), + indentOnInput(), + bracketMatching(), + closeBrackets(), + autocompletion(), + rectangularSelection(), + highlightActiveLine(), + highlightSelectionMatches(), + ...(languageExtensions[language] + ? languageExtensions[language] + : []), + keymap.of([ + ...closeBracketsKeymap, + ...defaultKeymap, + ...searchKeymap, + ...historyKeymap, + ...foldKeymap, + ...commentKeymap, + ...completionKeymap, + ...lintKeymap, + { + key: "Tab", + preventDefault: true, + run: indentMore, + }, + { + key: "Shift-Tab", + preventDefault: true, + run: indentLess, + }, + /*{ + key: "Ctrl-S", + preventDefault: true, + run: indentLess, + }*/ + ]), + EditorView.updateListener.of((update) => { + if (update.changes) { + onChange(update.state.doc.toString()); + } + }), + EditorState.allowMultipleSelections.of(true), + ...(disabled + ? [EditorState.readOnly.of(true)] + : [EditorState.readOnly.of(false)]), + ...(lineWrapping + ? [EditorView.lineWrapping] + : []), + ...[themeExtensions["dark"]], + ...[highlightExtensions["dark"]] + ]]; + + const { ref } = useCodeMirror(extensions, data); + + return ( +
+
+ ) +} + +export default CodeEditor; diff --git a/services/frontend/src/components/CodeEditor/themes/highlight/dark.ts b/services/frontend/src/components/CodeEditor/themes/highlight/dark.ts new file mode 100644 index 0000000..7beca69 --- /dev/null +++ b/services/frontend/src/components/CodeEditor/themes/highlight/dark.ts @@ -0,0 +1,82 @@ +// Based on https://github.com/codemirror/theme-one-dark +// Copyright (C) 2018-2021 by Marijn Haverbeke and others +// MIT License: https://github.com/codemirror/theme-one-dark/blob/main/LICENSE + +import { HighlightStyle, tags as t } from "@codemirror/highlight" + +const chalky = "#e5c07b", + coral = "#e06c75", + cyan = "#56b6c2", + invalid = "#ffffff", + ivory = "#abb2bf", + stone = "#5c6370", + malibu = "#61afef", + sage = "#98c379", + whiskey = "#d19a66", + violet = "#c678dd" + +/// The highlighting style for code in the One Dark theme. +export default HighlightStyle.define([ + { + tag: t.keyword, + color: violet + }, + { + tag: [t.name, t.deleted, t.character, t.propertyName, t.macroName], + color: coral + }, + { + tag: [t.processingInstruction, t.string, t.inserted], + color: sage + }, + { + tag: [t.function(t.variableName), t.labelName], + color: malibu + }, + { + tag: [t.color, t.constant(t.name), t.standard(t.name)], + color: whiskey + }, + { + tag: [t.definition(t.name), t.separator], + color: ivory + }, + { + tag: [t.typeName, t.className, t.number, t.changed, t.annotation, t.modifier, t.self, t.namespace], + color: chalky + }, + { + tag: [t.operator, t.operatorKeyword, t.url, t.escape, t.regexp, t.link, t.special(t.string)], + color: cyan + }, + { + tag: [t.meta, t.comment], + color: stone + }, + { + tag: t.strong, + fontWeight: "bold" + }, + { + tag: t.emphasis, + fontStyle: "italic" + }, + { + tag: t.link, + color: stone, + textDecoration: "underline" + }, + { + tag: t.heading, + fontWeight: "bold", + color: coral + }, + { + tag: [t.atom, t.bool, t.special(t.variableName)], + color: whiskey + }, + { + tag: t.invalid, + color: invalid + }, +]); \ No newline at end of file diff --git a/services/frontend/src/components/CodeEditor/themes/ui/dark.ts b/services/frontend/src/components/CodeEditor/themes/ui/dark.ts new file mode 100644 index 0000000..8854fcf --- /dev/null +++ b/services/frontend/src/components/CodeEditor/themes/ui/dark.ts @@ -0,0 +1,172 @@ +import { EditorView } from '@codemirror/view'; +import { HighlightStyle, tags } from '@codemirror/highlight'; + +const base00 = '#1F2937', base01 = '#073642', base02 = '#586e75', base03 = '#657b83', base04 = '#839496', base05 = '#dfdfdf', base06 = '#efefef', base07 = '#fdf6e3', base_red = '#dc322f', base_orange = '#cb4b16', base_yellow = '#e9b100', base_green = '#cfec11', base_cyan = '#44e0d4', base_blue = '#75c6ff', base_violet = '#a1a6ff', base_magenta = '#d33682'; +const invalid = '#d30102', stone = base04, darkBackground = '#1F2937', highlightBackground = '#173541', background = base00, tooltipBackground = base01, selection = '#173541', cursor = base04; +/** +The editor theme styles for Solarized Dark. +*/ +const solarizedDarkTheme = /*@__PURE__*/EditorView.theme({ + '&': { + fontSize: "10.5pt", + color: base05, + backgroundColor: background + }, + '.cm-content': { + caretColor: cursor + }, + '.cm-cursor, .cm-dropCursor': { borderLeftColor: cursor }, + '&.cm-focused .cm-selectionBackground, .cm-selectionBackground, .cm-content ::selection': { backgroundColor: selection }, + '.cm-panels': { backgroundColor: darkBackground, color: base03 }, + '.cm-panels.cm-panels-top': { borderBottom: '2px solid black' }, + '.cm-panels.cm-panels-bottom': { borderTop: '2px solid black' }, + '.cm-searchMatch': { + backgroundColor: '#72a1ff59', + outline: '1px solid #457dff' + }, + '.cm-searchMatch.cm-searchMatch-selected': { + backgroundColor: '#6199ff2f' + }, + '.cm-activeLine': { backgroundColor: highlightBackground }, + '.cm-selectionMatch': { backgroundColor: '#aafe661a' }, + '&.cm-focused .cm-matchingBracket, &.cm-focused .cm-nonmatchingBracket': { + outline: `1px solid ${base06}` + }, + '.cm-gutters': { + backgroundColor: darkBackground, + color: stone, + border: 'none' + }, + '.cm-activeLineGutter': { + backgroundColor: highlightBackground + }, + '.cm-foldPlaceholder': { + backgroundColor: 'transparent', + border: 'none', + color: '#ddd' + }, + '.cm-tooltip': { + border: 'none', + backgroundColor: tooltipBackground + }, + '.cm-tooltip .cm-tooltip-arrow:before': { + borderTopColor: 'transparent', + borderBottomColor: 'transparent' + }, + '.cm-tooltip .cm-tooltip-arrow:after': { + borderTopColor: tooltipBackground, + borderBottomColor: tooltipBackground + }, + '.cm-tooltip-autocomplete': { + '& > ul > li[aria-selected]': { + backgroundColor: highlightBackground, + color: base03 + } + } +}, { dark: true }); +/** +The highlighting style for code in the Solarized Dark theme. +*/ +const solarizedDarkHighlightStyle = /*@__PURE__*/HighlightStyle.define([ + { tag: tags.keyword, color: base_green }, + { + tag: [tags.name, tags.deleted, tags.character, tags.propertyName, tags.macroName], + color: base_cyan + }, + { tag: [tags.variableName], color: base05 }, + { tag: [/*@__PURE__*/tags.function(tags.variableName)], color: base_blue }, + { tag: [tags.labelName], color: base_magenta }, + { + tag: [tags.color, /*@__PURE__*/tags.constant(tags.name), /*@__PURE__*/tags.standard(tags.name)], + color: base_yellow + }, + { tag: [/*@__PURE__*/tags.definition(tags.name), tags.separator], color: base_cyan }, + { tag: [tags.brace], color: base_magenta }, + { + tag: [tags.annotation], + color: invalid + }, + { + tag: [tags.number, tags.changed, tags.annotation, tags.modifier, tags.self, tags.namespace], + color: base_magenta + }, + { + tag: [tags.typeName, tags.className], + color: base_orange + }, + { + tag: [tags.operator, tags.operatorKeyword], + color: base_violet + }, + { + tag: [tags.tagName], + color: base_blue + }, + { + tag: [tags.squareBracket], + color: base_red + }, + { + tag: [tags.angleBracket], + color: base02 + }, + { + tag: [tags.attributeName], + color: base05 + }, + { + tag: [tags.regexp], + color: invalid + }, + { + tag: [tags.quote], + color: base_green + }, + { tag: [tags.string], color: base_yellow }, + { + tag: [tags.url, tags.escape, /*@__PURE__*/tags.special(tags.string)], + color: base_yellow + }, + { tag: [tags.meta], color: base_red }, + { tag: [tags.comment], color: base02, fontStyle: 'italic' }, + { tag: tags.strong, fontWeight: 'bold', color: base06 }, + { tag: tags.emphasis, fontStyle: 'italic', color: base_green }, + { tag: tags.strikethrough, textDecoration: 'line-through' }, + { + tag: tags.link, + color: base_cyan, + textDecoration: 'underline', + textUnderlinePosition: 'under' + }, + { tag: tags.heading, fontWeight: 'bold', color: base_yellow }, + { tag: tags.heading1, fontWeight: 'bold', color: base07 }, + { + tag: [tags.heading2, tags.heading3, tags.heading4], + fontWeight: 'bold', + color: base06 + }, + { + tag: [tags.heading5, tags.heading6], + color: base06 + }, + { tag: [tags.atom, tags.bool, /*@__PURE__*/tags.special(tags.variableName)], color: base_magenta }, + { + tag: [tags.processingInstruction, tags.inserted, tags.contentSeparator], + color: base_red + }, + { + tag: [tags.contentSeparator], + color: base_yellow + }, + { tag: tags.invalid, color: base02, borderBottom: `1px dotted ${base_red}` } +]); +/** +Extension to enable the Solarized Dark theme (both the editor theme and +the highlight style). +*/ +const solarizedDark = [ + solarizedDarkTheme, + solarizedDarkHighlightStyle +]; + +export { solarizedDark, solarizedDarkHighlightStyle, solarizedDarkTheme }; diff --git a/services/frontend/src/components/CodeEditor/useCodeMirror.tsx b/services/frontend/src/components/CodeEditor/useCodeMirror.tsx new file mode 100644 index 0000000..f38ac46 --- /dev/null +++ b/services/frontend/src/components/CodeEditor/useCodeMirror.tsx @@ -0,0 +1,30 @@ +import { useCallback, useEffect, useState } from "react"; +import { EditorView } from '@codemirror/view' +import { EditorState } from "@codemirror/state"; +import { Extension } from "@codemirror/state"; + +export default function useCodeMirror(extensions: Extension[], doc: any) { + const [element, setElement] = useState(); + + const ref = useCallback((node: HTMLElement | null) => { + if (!node) return; + + setElement(node); + }, []); + + useEffect(() => { + if (!element) return; + + const view = new EditorView({ + state: EditorState.create({ + doc: doc, + extensions: [...extensions], + }), + parent: element, + }); + + return () => view?.destroy(); + }, [element, extensions, doc]); + + return { ref }; +} diff --git a/services/frontend/src/components/DarkModeSwitch/index.tsx b/services/frontend/src/components/DarkModeSwitch/index.tsx new file mode 100644 index 0000000..f0968ad --- /dev/null +++ b/services/frontend/src/components/DarkModeSwitch/index.tsx @@ -0,0 +1,37 @@ +import { MoonIcon, SunIcon } from "@heroicons/react/outline"; +import { useDarkMode } from "./userDarkMode"; + +interface IDarkModeSwitchProps { +} + +const DarkModeSwitch = (props: IDarkModeSwitchProps) => { + const [isDark, setIsDark] = useDarkMode(); + + return ( +
+ +
+ ) +} + +export default DarkModeSwitch; diff --git a/services/frontend/src/components/DarkModeSwitch/userDarkMode.tsx b/services/frontend/src/components/DarkModeSwitch/userDarkMode.tsx new file mode 100644 index 0000000..ccb9b59 --- /dev/null +++ b/services/frontend/src/components/DarkModeSwitch/userDarkMode.tsx @@ -0,0 +1,53 @@ +import { useEffect, useState } from "react"; + +export function usePrefersDarkMode() { + const [value, setValue] = useState(true); + + useEffect(() => { + const mediaQuery = window.matchMedia("(prefers-color-scheme: dark)"); + setValue(mediaQuery.matches); + + const handler = () => setValue(mediaQuery.matches); + mediaQuery.addEventListener("change", handler); + return () => mediaQuery.removeEventListener("change", handler); + }, []) + + return value; +} + +export function useSafeLocalStorage(key: string, initialValue: string | undefined) { + const [valueProxy, setValueProxy] = useState(() => { + try { + const value = window.localStorage.getItem(key); + return value ? JSON.parse(value) : initialValue; + } catch { + return initialValue; + } + }) + + const setValue = (value: string) => { + try { + window.localStorage.setItem(key, value); + setValueProxy(value); + } catch { + setValueProxy(value); + } + } + + return [valueProxy, setValue]; +} + +export function useDarkMode() { + const prefersDarkMode = usePrefersDarkMode(); + const [isEnabled, setIsEnabled] = useSafeLocalStorage("dark-mode", undefined); + const enabled = isEnabled === undefined ? prefersDarkMode : isEnabled; + + useEffect(() => { + if (window === undefined) return; + const root = window.document.documentElement; + root.classList.remove(enabled ? "light" : "dark"); + root.classList.add(enabled ? "dark" : "light"); + }, [enabled]); + + return [enabled, setIsEnabled]; +} diff --git a/services/frontend/src/components/Modal/Node/Container.tsx b/services/frontend/src/components/Modal/Node/Container.tsx new file mode 100644 index 0000000..35e766c --- /dev/null +++ b/services/frontend/src/components/Modal/Node/Container.tsx @@ -0,0 +1,54 @@ +const Container = (props: any) => { + const { formik } = props; + + return ( + <> +
+
+ +
+ +
+
+
+ +
+
+ +
+ +
+
+
+ +
+
+ +
+ +
+
+
+ + ) +} + +export default Container; diff --git a/services/frontend/src/components/Modal/Node/Create.tsx b/services/frontend/src/components/Modal/Node/Create.tsx new file mode 100644 index 0000000..93d6755 --- /dev/null +++ b/services/frontend/src/components/Modal/Node/Create.tsx @@ -0,0 +1,147 @@ +import { useState } from "react"; +import { useFormik } from "formik"; +import { XIcon } from "@heroicons/react/outline"; +import General from "./General"; +import Container from "./Container"; +import Resource from "./Resource"; +import { initialValues, formatName } from "./../../../utils"; + + +interface IModalProps { + onHide: any; + onAddEndpoint: Function; +} + +const ModalCreate = (props: IModalProps) => { + const { onHide, onAddEndpoint } = props; + const [openTab, setOpenTab] = useState("General"); + const formik = useFormik({ + initialValues: { + configuration: { + ...initialValues(), + }, + key: "template", + type: "TEMPLATE", + inputs: ["op_source"], + outputs: [], + config: {} + }, + onSubmit: ((values, { setSubmitting }) => { + + }) + }); + const tabs = [ + { name: 'General', href: '#', current: true, hidden: false }, + { name: 'Container', href: '#', current: false, hidden: (formik.values.configuration.type === 'container' ? false : true) }, + { name: 'Resource', href: '#', current: false, hidden: (formik.values.configuration.type === 'resource' ? false : true) } + ]; + + const classNames = (...classes: string[]) => { + return classes.filter(Boolean).join(' '); + } + + return ( + <> +
+
+
+
+
+
+

Add template

+ +
+ +
+
+ + +
+ + + +
+
+ {openTab === "General" && + + } + + {openTab === "Container" && + + } + + {openTab === "Resource" && + + } + +
+
+ +
+ +
+
+
+
+
+ + ); +} + +export default ModalCreate diff --git a/services/frontend/src/components/Modal/Node/Edit.tsx b/services/frontend/src/components/Modal/Node/Edit.tsx new file mode 100644 index 0000000..725da85 --- /dev/null +++ b/services/frontend/src/components/Modal/Node/Edit.tsx @@ -0,0 +1,165 @@ +import React from "react"; +import { useFormik } from "formik"; +import { XIcon } from "@heroicons/react/outline"; +import General from "./General"; +import Container from "./Container"; +import Resource from "./Resource"; +import { initialValues, formatName } from "./../../../utils"; +import { IClientNodeItem } from "../../../types"; + + +interface IModalProps { + node: IClientNodeItem | null; + onHide: any; + onUpdateEndpoint: any; +} + +const ModalEdit = (props: IModalProps) => { + const { node, onHide, onUpdateEndpoint } = props; + const [selectedNode, setSelectedNode] = React.useState(null); + const [openTab, setOpenTab] = React.useState("General"); + const formik = useFormik({ + initialValues: { + configuration: { + ...initialValues() + } + }, + onSubmit: ((values, { setSubmitting }) => { + + }) + }); + const tabs = [ + { name: 'General', href: '#', current: true, hidden: false }, + { name: 'Container', href: '#', current: false, hidden: (formik.values.configuration.type === 'container' ? false : true) }, + { name: 'Resource', href: '#', current: false, hidden: (formik.values.configuration.type === 'resource' ? false : true) } + ]; + + const classNames = (...classes: string[]) => { + return classes.filter(Boolean).join(' '); + } + + React.useEffect(() => { + if (node) { + setSelectedNode(node); + } + }, [node]); + + React.useEffect(() => { + formik.resetForm(); + + if (selectedNode) { + formik.initialValues.configuration = { ...selectedNode.configuration }; + } + }, [selectedNode]); + + React.useEffect(() => { + return () => { + formik.resetForm(); + } + }, []); + + return ( + <> +
+
+
+
+
+
+

Update template

+ +
+ +
+
+ + +
+ + + +
+
+ {openTab === "General" && + + } + + {openTab === "Container" && + + } + + {openTab === "Resource" && + + } + +
+
+ +
+ +
+
+
+
+
+ + ); +} + +export default ModalEdit diff --git a/services/frontend/src/components/Modal/Node/General.tsx b/services/frontend/src/components/Modal/Node/General.tsx new file mode 100644 index 0000000..f5d657b --- /dev/null +++ b/services/frontend/src/components/Modal/Node/General.tsx @@ -0,0 +1,62 @@ +import React from "react"; + + +const General = (props: any) => { + const { formik } = props; + + return ( + <> +
+
+ +
+ +
+
+
+ +
+ +
+ +
+
+ +
+
+ +
+ +
+
+
+ + ) +} +export default General; \ No newline at end of file diff --git a/services/frontend/src/components/Modal/Node/Resource.tsx b/services/frontend/src/components/Modal/Node/Resource.tsx new file mode 100644 index 0000000..04c933d --- /dev/null +++ b/services/frontend/src/components/Modal/Node/Resource.tsx @@ -0,0 +1,41 @@ +import React from "react"; + + +const Resource = (props: any) => { + const { formik } = props; + + return ( + <> +
+
+ +
+ +
+
+
+ +
+
+ + +
+
+ + ) +} +export default Resource; \ No newline at end of file diff --git a/services/frontend/src/components/Modal/Service/Create.tsx b/services/frontend/src/components/Modal/Service/Create.tsx new file mode 100644 index 0000000..ec71611 --- /dev/null +++ b/services/frontend/src/components/Modal/Service/Create.tsx @@ -0,0 +1,105 @@ +import { useFormik } from "formik"; +import { XIcon } from "@heroicons/react/outline"; +import { serviceInitialValues, formatName } from "../../../utils"; + + +interface IModalServiceProps { + onHide: any; + onAddEndpoint: Function; +} + +const ModalServiceCreate = (props: IModalServiceProps) => { + const { onHide, onAddEndpoint } = props; + const formik = useFormik({ + initialValues: { + configuration: { + ...serviceInitialValues(), + }, + key: "service", + type: "SERVICE", + inputs: ["op_source"], + outputs: [], + config: {} + }, + onSubmit: ((values, { setSubmitting }) => { + + }) + }); + + return ( + <> +
+
+
+
+
+
+

Add service

+ +
+ +
+
+
+ +
+ +
+
+
+ +
+
+ +
+ +
+
+
+
+ +
+ +
+
+
+
+
+ + ); +} + +export default ModalServiceCreate diff --git a/services/frontend/src/components/Modal/Service/Edit.tsx b/services/frontend/src/components/Modal/Service/Edit.tsx new file mode 100644 index 0000000..ee7a166 --- /dev/null +++ b/services/frontend/src/components/Modal/Service/Edit.tsx @@ -0,0 +1,124 @@ +import React from "react"; +import { useFormik } from "formik"; +import { XIcon } from "@heroicons/react/outline"; +import { serviceInitialValues, formatName } from "../../../utils"; + + +interface IModalServiceProps { + node: any; + onHide: any; + onUpdateEndpoint: any; +} + +const ModalServiceEdit = (props: IModalServiceProps) => { + const { node, onHide, onUpdateEndpoint } = props; + const [selectedNode, setSelectedNode] = React.useState(null); + const formik = useFormik({ + initialValues: { + configuration: { + ...serviceInitialValues() + } + }, + onSubmit: ((values, { setSubmitting }) => { + + }) + }); + + React.useEffect(() => { + if (node) { + setSelectedNode(node); + } + }, [node]); + + React.useEffect(() => { + formik.resetForm(); + + if (selectedNode) { + formik.initialValues.configuration = { ...selectedNode.configuration }; + } + }, [selectedNode]); + + React.useEffect(() => { + return () => { + formik.resetForm(); + } + }, []); + + return ( + <> +
+
+
+
+
+
+

Update service

+ +
+ +
+
+
+ +
+ +
+
+
+ +
+
+ +
+ +
+
+
+
+ +
+ +
+
+
+
+
+ + ); +} + +export default ModalServiceEdit diff --git a/services/frontend/src/components/Profile/index.tsx b/services/frontend/src/components/Profile/index.tsx new file mode 100644 index 0000000..000475d --- /dev/null +++ b/services/frontend/src/components/Profile/index.tsx @@ -0,0 +1,94 @@ +import { useState } from "react"; +import { Link } from "react-router-dom"; +import { MenuAlt2Icon, ReplyIcon } from "@heroicons/react/outline"; +import SideBar from "../../components/SideBar"; +import DarkModeSwitch from "../../components/DarkModeSwitch"; +import { LOCAL_STORAGE } from "../../constants"; +import { authSelf } from "../../reducers"; + +interface IProfileProps { + dispatch: any; + state: any; +} + +const Profile = (props: IProfileProps) => { + const { dispatch, state } = props; + const [sidebarOpen, setSidebarOpen] = useState(false); + + const logOut = () => { + localStorage.removeItem(LOCAL_STORAGE); + dispatch(authSelf(null)); + } + + return ( + <> + + +
+
+ +
+ + + + +
+ +
+
+
+ +
+
+

Profile

+ +
+
+ {state.user && + <> + {state.user.username && +
+
+ username +
+
+ {state.user.username} +
+
+ } + + {state.user.email && +
+
+ email +
+
+ {state.user.email} +
+
+ } + + } +
+
+
+ + ) +} + +export default Profile; diff --git a/services/frontend/src/components/Project/index.tsx b/services/frontend/src/components/Project/index.tsx new file mode 100644 index 0000000..47d57bc --- /dev/null +++ b/services/frontend/src/components/Project/index.tsx @@ -0,0 +1,203 @@ +import { useEffect, useState } from "react"; +import { useParams, useNavigate } from "react-router-dom"; +import { IProjectPayload } from "../../types"; +import { nodes, connections, position, updateProjectName } from "../../reducers"; +import Spinner from "../Spinner"; +import { Canvas } from "../Canvas"; +import useWindowDimensions from "../../hooks/useWindowDimensions"; +import { getClientNodesAndConnections } from "../../utils"; +import { projectHttpGet, projectHttpUpdate, projectHttpCreate } from "../../services/project"; +import { checkHttpStatus } from "../../services/helpers"; +import { nodeLibraries } from "../../utils/data/libraries"; + +interface IProjectProps { + dispatch: any; + state: any; +} + +export default function Project(props: IProjectProps) { + const { uuid } = useParams<{ uuid?: string }>(); + const { dispatch, state } = props; + const [saving, setSaving] = useState(false); + const [projectName, setProjectName] = useState(""); + const { height, width } = useWindowDimensions(); + const navigate = useNavigate(); + + const handleNameChange = (e: any) => { + setProjectName(e.target.value); + dispatch(updateProjectName(e.target.value)); + } + + const updateProject = (uuid: string, payload: IProjectPayload) => { + projectHttpUpdate(uuid, JSON.stringify(payload)) + .then(checkHttpStatus) + .then(data => { + + }) + .catch(err => { + + }) + .finally(() => { + setSaving(false); + }); + } + + const createProject = (payload: IProjectPayload) => { + projectHttpCreate(JSON.stringify(payload)) + .then(checkHttpStatus) + .then(data => { + navigate(`/projects/${data.uuid}`) + }) + .catch(err => { + + }) + .finally(() => { + setSaving(false); + }); + } + + const onSave = () => { + setSaving(true); + const payload: IProjectPayload = { + name: state.projectName, + data: { + canvas: { + position: state.canvasPosition, + nodes: state.nodes, + connections: state.connections + }, + configs: [], + networks: [], + secrets: [], + services: state.nodes, + version: 3, + volumes: [], + } + }; + + if (uuid) { + updateProject(uuid, payload); + } else { + createProject(payload); + } + }; + + const setViewHeight = () => { + let vh = window.innerHeight * 0.01; + document.documentElement.style.setProperty('--vh', `${vh}px`); + }; + + useEffect(() => { + if (uuid) { + projectHttpGet(uuid) + .then(checkHttpStatus) + .then(data => { + const projectData = JSON.parse(data.data); + const nodesAsList = Object.keys(projectData.canvas.nodes).map((k) => { + return projectData.canvas.nodes[k]; + }); + + const clientNodeItems = getClientNodesAndConnections(nodesAsList, nodeLibraries); + setProjectName(data.name); + dispatch(updateProjectName(data.name)); + dispatch(nodes(clientNodeItems)); + dispatch(connections(projectData.canvas.connections)); + dispatch(position(projectData.canvas.position)); + }) + .catch(err => { + if (err.status === 404) { + window.location.replace("/"); + } + }) + .finally(() => { + //setFetching(false); + }) + } + }, [uuid, dispatch]); + + useEffect(() => { + //setProjectName(state.projectName); + }, [state.projectName]); + + useEffect(() => { + window.addEventListener("resize", () => { + setViewHeight(); + }); + + setViewHeight(); + }, []); + + return ( + <> +
+
+ + +
+ + + +
+
+
+ +
+ +
+ + ); +} diff --git a/services/frontend/src/components/Remove/index.tsx b/services/frontend/src/components/Remove/index.tsx new file mode 100644 index 0000000..f47851c --- /dev/null +++ b/services/frontend/src/components/Remove/index.tsx @@ -0,0 +1,28 @@ +import React, { SyntheticEvent } from "react"; +import { TrashIcon } from "@heroicons/react/solid"; + + +export interface ICloseProps { + id: string; + onClose?: (id: string, source?: string, target?: string) => any; + source?: string; + target?: string; +} + +const Close = (props: ICloseProps) => { + const { id, onClose, source, target } = props; + const handleClose = (event: SyntheticEvent) => { + event.preventDefault(); + if (onClose) { + onClose(id, source, target); + } + } + + return ( +
+ +
+ ); +} + +export default Close diff --git a/services/frontend/src/components/SideBar.tsx b/services/frontend/src/components/SideBar.tsx new file mode 100644 index 0000000..e1adf68 --- /dev/null +++ b/services/frontend/src/components/SideBar.tsx @@ -0,0 +1,132 @@ +import { Fragment } from "react"; +import { useLocation } from "react-router-dom"; +import { Dialog, Transition } from "@headlessui/react"; +import { DatabaseIcon, TemplateIcon, XIcon } from "@heroicons/react/outline"; +import UserMenu from "./UserMenu"; +import Logo from "./logo"; + +interface ISideBarProps { + state: any; + sidebarOpen: boolean; + setSidebarOpen: any; +} + +export default function SideBar(props: ISideBarProps) { + const { pathname } = useLocation(); + const { state, sidebarOpen, setSidebarOpen } = props; + const navigation = [ + { name: "Templates", href: "/", icon: TemplateIcon, current: ((pathname === "/" || pathname.includes("templates")) ? true : false) }, + { name: "Connectors", href: "/connectors", icon: DatabaseIcon, current: (pathname.includes("connectors") ? true : false) } + ]; + const classNames = (...classes: any[]) => { + return classes.filter(Boolean).join(" ") + }; + const userName = state.user ? state.user.username : ""; + + return ( + <> + + + + + + +
+ +
+ +
+
+ +
+ +
+ +
+ +
+ + +
+
+ + +
+
+ +
+
+
+ +
+
+ +
+ + +
+
+ + ); +} diff --git a/services/frontend/src/components/Spinner.tsx b/services/frontend/src/components/Spinner.tsx new file mode 100644 index 0000000..4da0f1b --- /dev/null +++ b/services/frontend/src/components/Spinner.tsx @@ -0,0 +1,19 @@ +import React from 'react' + + +interface ISpinnerProps { + className: string; +} + +const Spinner = (props: ISpinnerProps) => { + const { className } = props; + + return ( + + + + + ) +} + +export default Spinner diff --git a/services/frontend/src/components/UserMenu.tsx b/services/frontend/src/components/UserMenu.tsx new file mode 100644 index 0000000..f278495 --- /dev/null +++ b/services/frontend/src/components/UserMenu.tsx @@ -0,0 +1,34 @@ +import { useNavigate } from "react-router-dom"; +import { UserCircleIcon } from "@heroicons/react/solid"; + +interface IUserMenuProps { + username: string; + current: boolean; +} + +export default function UserMenu(props: IUserMenuProps) { + const { username, current } = props; + const navigate = useNavigate(); + + return ( +
{ + navigate("/profile"); + }} + className={` + ${current ? "bg-blue-800 text-white" : "text-blue-100 hover:bg-blue-600"}, + flex border-t border-blue-800 p-4 w-full hover:cursor-pointer hover:bg-blue-600 + `} + > +
+
+ +
+
+

{username}

+

View profile

+
+
+
+ ) +} \ No newline at end of file diff --git a/services/frontend/src/components/logo.tsx b/services/frontend/src/components/logo.tsx new file mode 100644 index 0000000..9194a56 --- /dev/null +++ b/services/frontend/src/components/logo.tsx @@ -0,0 +1,20 @@ +interface ILogoProps { + className: string; +} + +const Logo = (props: ILogoProps) => { + const { className } = props; + return ( + + + + + + + + + + ) +} + +export default Logo; \ No newline at end of file diff --git a/services/frontend/src/components/useJsPlumb.ts b/services/frontend/src/components/useJsPlumb.ts new file mode 100644 index 0000000..4dda84c --- /dev/null +++ b/services/frontend/src/components/useJsPlumb.ts @@ -0,0 +1,325 @@ +import { useState, useEffect, useRef, useCallback } from "react"; +import { AnchorId } from "@jsplumb/common"; +import { + BeforeDropParams, + Connection, + ConnectionDetachedParams, + ConnectionEstablishedParams, + ConnectionSelection, + EVENT_CONNECTION, + EVENT_CONNECTION_DETACHED, + INTERCEPT_BEFORE_DROP +} from "@jsplumb/core"; +import { + BrowserJsPlumbInstance, + newInstance, + EVENT_DRAG_STOP, + EVENT_CONNECTION_DBL_CLICK +} from "@jsplumb/browser-ui"; +import { + defaultOptions, + inputAnchors, + outputAnchors, + sourceEndpoint, + targetEndpoint +} from "../utils/options"; +import { getConnections } from "../utils"; +import { IClientNodeItem } from "../types"; +import { Dictionary, isEqual } from "lodash"; +import { IAnchor } from "../types"; + +export const useJsPlumb = ( + nodes: Dictionary, + connections: Array<[string, string]>, + onGraphUpdate: Function, + onEndpointPositionUpdate: Function, + onConnectionAttached: Function, + onConnectionDetached: Function +): [(containerElement: HTMLDivElement) => void, + (zoom: number) => void, + (style: any) => void, + (node: IClientNodeItem) => void] => { + const [instance, setInstance] = useState(null as any); + const containerRef = useRef(); + const stateRef = useRef>(); + stateRef.current = nodes; + const containerCallbackRef = useCallback((containerElement: HTMLDivElement) => { + containerRef.current = containerElement; + }, []); + + const addEndpoints = useCallback(( + el: Element, + sourceAnchors: IAnchor[], + targetAnchors: IAnchor[], + maxConnections: number + ) => { + sourceAnchors.forEach((x) => { + let endpoint = sourceEndpoint; + endpoint.maxConnections = maxConnections; + + // arrow overlay for connector to specify + // it's dependency on another service + instance.addEndpoint(el, endpoint, { + anchor: [[0.4, 0, 0, -1], [1, 0.4, 1, 0], [0.4, 1, 0, 1], [0, 0.4, -1, 0]], + uuid: x.id, + connectorOverlays: [{ + type: "PlainArrow", + options: { + width: 16, + length: 16, + location: 1, + id: "arrow" + }, + }] + }) + }); + + targetAnchors.forEach((x) => { + let endpoint = targetEndpoint; + endpoint.maxConnections = maxConnections; + + instance.addEndpoint(el, endpoint, { + anchor: [[0.6, 0, 0, -1], [1, 0.6, 1, 0], [0.6, 1, 0, 1], [0, 0.6, -1, 0]], + uuid: x.id + }); + }); + }, [instance]); + + const removeEndpoint = useCallback((node) => { + const nodeConnections = instance.getConnections({ target: node.key }); + + if (nodeConnections) { + Object.values(nodeConnections).forEach((conn) => { + instance.deleteConnection(conn); + }); + }; + + instance.removeAllEndpoints(document.getElementById(node.key) as Element); + }, [instance]); + + const getAnchors = (port: string[], anchorIds: AnchorId[]): IAnchor[] => { + return port.map( + (x, index): IAnchor => ({ + id: x, + position: anchorIds[port.length === 1 ? 2 : index] + }) + ); + }; + + const getOverlayObject = (instance: BrowserJsPlumbInstance) => { + return { + type: "Label", + options: { + label: "x", + location: .5, + id: "remove-conn", + cssClass: ` + block jtk-overlay remove-conn-btn text-xs leading-normal + cursor-pointer text-white font-bold rounded-full w-5 h-5 + z-20 flex justify-center + `, + events: { + click: (e: any) => { + instance.deleteConnection(e.overlay.component as Connection); + } + } + } + } + }; + + const setZoom = useCallback((zoom: number) => { + if (instance) { + instance.setZoom(zoom); + } + }, [instance]); + + const setStyle = useCallback((style: any) => { + let styles: { [key: string]: any } = {}; + const currentStyle = containerRef.current?.getAttribute("style"); + + if (currentStyle) { + let currentStyleParts = ( + currentStyle + .split(";") + .map(element => element.trim()) + .filter(element => element !== '') + ); + + for (let i = 0; i < currentStyleParts.length; i++) { + const entry = currentStyleParts[i].split(':'); + styles[entry.splice(0, 1)[0]] = entry.join(':').trim(); + } + } + + styles = {...styles, ...style}; + const styleString = ( + Object.entries(styles).map(([k, v]) => `${k}:${v}`).join(';') + ); + + containerRef.current?.setAttribute("style", `${styleString}`); + }, []); + + const onbeforeDropIntercept = (instance: BrowserJsPlumbInstance, params: BeforeDropParams) => { + const existingConnections: ConnectionSelection = instance.select({ source: params.sourceId as any, target: params.targetId as any }); + + // prevent duplicates when switching existing connections + if (existingConnections.length > 1) { + return false; + } + + if (existingConnections.length > 0) { + const firstConnection: Connection = {...existingConnections.get(0)} as Connection; + + // special case to handle existing connections changing targets + if (firstConnection.suspendedElementId) { + onConnectionDetached([params.sourceId, firstConnection.suspendedElementId]); + + if (params.targetId !== firstConnection.suspendedElementId) { + onConnectionAttached([params.sourceId, params.targetId]); + return true; + } + } + + // prevent duplicate connections from the same source to target + if (firstConnection.sourceId === params.sourceId && firstConnection.targetId === params.targetId) { + return false; + } + } + + // prevent looping connections between a target and source + const loopCheck = instance.select({ source: params.targetId as any, target: params.sourceId as any }); + if (loopCheck.length > 0) { + return false; + } + + // prevent a connection from a target to itself + if (params.sourceId === params.targetId) { + return false; + } + + return true; + }; + + const reset = () => { + if (!instance) { + return; + } + + instance.reset(); + instance.destroy(); + } + + useEffect(() => { + if (!instance) return; + + if (stateRef.current) { + Object.values(stateRef.current).forEach((x) => { + if (!instance.selectEndpoints({ element: x.key as any }).length) { + const maxConnections = -1; + const el = document.getElementById(x.key) as Element; + + if (el) { + addEndpoints( + el, + getAnchors(x.outputs, outputAnchors), + getAnchors(x.inputs, inputAnchors), + maxConnections + ); + } + }; + }); + + onGraphUpdate({ + 'nodes': stateRef.current, + 'connections': getConnections(instance.getConnections({}, true) as Connection[]) + }); + } + }, [instance, addEndpoints, onGraphUpdate]); + + useEffect(() => { + if (!instance) return; + + let exisitngConnectionUuids = (instance.getConnections({}, true) as Connection[]).map( + (x) => x.getUuids() + ); + + connections.forEach((x) => { + let c = exisitngConnectionUuids.find((y) => { + return isEqual([`op_${x[0]}`, `ip_${x[1]}`], y) + }); + + if (!c) { + instance.connect({ + uuids: [`op_${x[0]}`, `ip_${x[1]}`], + overlays: [getOverlayObject(instance)] + }); + } + }); + }, [connections, instance]); + + useEffect(() => { + const jsPlumbInstance: BrowserJsPlumbInstance = newInstance({ + ...defaultOptions, + container: containerRef.current + }); + + jsPlumbInstance.bind(INTERCEPT_BEFORE_DROP, function (params: BeforeDropParams) { + return onbeforeDropIntercept(jsPlumbInstance, params); + }); + + jsPlumbInstance.bind(EVENT_CONNECTION_DETACHED, function (this: BrowserJsPlumbInstance, params: ConnectionDetachedParams) { + onConnectionDetached([params.sourceId, params.targetId]); + + onGraphUpdate({ + 'nodes': stateRef.current, + 'connections': getConnections(this.getConnections({}, true) as Connection[]) + }); + }); + + jsPlumbInstance.bind(EVENT_CONNECTION, function (this: BrowserJsPlumbInstance, params: ConnectionEstablishedParams) { + if (!params.connection.overlays.hasOwnProperty("remove-conn")) { + params.connection.addOverlay(getOverlayObject(this)); + onConnectionAttached([params.sourceId, params.targetId]); + } + + onGraphUpdate({ + 'nodes': stateRef.current, + 'connections': getConnections(this.getConnections({}, true) as Connection[]) + }); + }); + + jsPlumbInstance.bind(EVENT_DRAG_STOP, (p: any) => { + onEndpointPositionUpdate({ + key: p.el.id, + position: { + top: p.el.offsetTop, + left: p.el.offsetLeft + } + }); + }); + + jsPlumbInstance.bind(EVENT_CONNECTION_DBL_CLICK, (connection: Connection) => { + jsPlumbInstance.deleteConnection(connection); + }); + + /* + jsPlumbInstance.bind("drag:move", function (info: any) { + const parentRect = jsPlumbInstance.getContainer().getBoundingClientRect() + const childRect = info.el.getBoundingClientRect() + if (childRect.right > parentRect.right) info.el.style.left = `${parentRect.width - childRect.width}px` + if (childRect.left < parentRect.left) info.el.style.left = '0px' + if (childRect.top < parentRect.top) info.el.style.top = '0px' + if (childRect.bottom > parentRect.bottom) info.el.style.top = `${parentRect.height - childRect.height}px` + }); + */ + + setInstance(jsPlumbInstance); + + return () => { + reset(); + }; + // eslint-disable-next-line react-hooks/exhaustive-deps + }, []); + + return [containerCallbackRef, setZoom, setStyle, removeEndpoint]; +} diff --git a/services/frontend/src/constants/index.ts b/services/frontend/src/constants/index.ts new file mode 100644 index 0000000..2bd4b33 --- /dev/null +++ b/services/frontend/src/constants/index.ts @@ -0,0 +1,3 @@ +export const API_SERVER_URL = process.env.REACT_APP_API_SERVER; +export const PROJECTS_FETCH_LIMIT = 300; +export const LOCAL_STORAGE = 'CtkLocalStorage'; diff --git a/services/frontend/src/hooks/auth.ts b/services/frontend/src/hooks/auth.ts new file mode 100644 index 0000000..6824f94 --- /dev/null +++ b/services/frontend/src/hooks/auth.ts @@ -0,0 +1,11 @@ +import { LOCAL_STORAGE } from "../constants" + + +export const useLocalStorageAuth = () => { + const localStorageData = localStorage.getItem(LOCAL_STORAGE); + + if (localStorageData) { + const authData = JSON.parse(localStorageData); + return authData + } +} \ No newline at end of file diff --git a/services/frontend/src/hooks/useWindowDimensions.ts b/services/frontend/src/hooks/useWindowDimensions.ts new file mode 100644 index 0000000..89f0f7b --- /dev/null +++ b/services/frontend/src/hooks/useWindowDimensions.ts @@ -0,0 +1,24 @@ +import { useState, useEffect } from 'react'; + +function getWindowDimensions() { + const { innerWidth: width, innerHeight: height } = window; + return { + width, + height + }; +} + +export default function useWindowDimensions() { + const [windowDimensions, setWindowDimensions] = useState(getWindowDimensions()); + + useEffect(() => { + function handleResize() { + setWindowDimensions(getWindowDimensions()); + } + + window.addEventListener('resize', handleResize); + return () => window.removeEventListener('resize', handleResize); + }, []); + + return windowDimensions; +} diff --git a/services/frontend/src/index.css b/services/frontend/src/index.css new file mode 100644 index 0000000..4933328 --- /dev/null +++ b/services/frontend/src/index.css @@ -0,0 +1,124 @@ +@tailwind base; +@tailwind components; +@tailwind utilities; + +body { + margin: 0; + font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', 'Roboto', 'Oxygen', + 'Ubuntu', 'Cantarell', 'Fira Sans', 'Droid Sans', 'Helvetica Neue', + sans-serif; + -webkit-font-smoothing: antialiased; + -moz-osx-font-smoothing: grayscale; +} + +.canvas { + position: relative; + height: 100%; +} + +.jsplumb-box { + background-size: 16px 16px; + background-image: + linear-gradient(to right, #80808014 1px, transparent 1px), + linear-gradient(to bottom, #80808014 1px, transparent 1px); + position: relative; + width: 100%; + height: calc(100vh - 64px); /* 64px is the bar above the canvas */ + overflow: hidden; + cursor: move; + user-select: none; +} + +.node-item { + border-radius: 1em; + width: 150px; + height: 60px; + z-index: 30; + position: absolute; + background-color: #fff; +} + +.node-item img { + width: 26px; + height: 26px; +} + +.jtk-connector { + z-index: 4; +} + +path, +.jtk-endpoint { + z-index: 20; + cursor: pointer; +} + +.node-item.jtk-drag { + box-shadow: 0px 0px 5px 2px rgba(75, 0, 255, 0.37); +} + +.jtk-drag-select * { + -webkit-touch-callout: none; + -webkit-user-select: none; + -khtml-user-select: none; + -moz-user-select: none; + -ms-user-select: none; + user-select: none; +} + +.endpoint { + width: 14px; + height: 14px; +} + +.remove-conn-btn { + background-color: #61B7CF; +} + +.remove-conn-btn:hover { + background-color: #ce4551; +} + +.code-column { + background-color: #1F2937; +} + +.cke_reset_all .CodeMirror-scroll * { + white-space: pre; +} + +* { + scrollbar-width: thin; + scrollbar-color: #464646 #282c34; +} + +*::-webkit-scrollbar { + width: 12px; +} + +*::-webkit-scrollbar-track { + background: #282c34; +} + +*::-webkit-scrollbar-thumb { + background-color: #464646; + border-radius: 20px; + border: 5px solid #282c34; +} + +@layer components { + .btn-util { + @apply inline-flex items-center px-2.5 py-1.5 border border-transparent text-xs font-medium rounded text-indigo-700 bg-indigo-100 hover:bg-indigo-200 focus:outline-none focus:ring-1 focus:ring-offset-1 focus:ring-indigo-500; + } + .btn-util-selected { + @apply text-white bg-indigo-500 hover:bg-indigo-500 focus:ring-indigo-500; + } + .input-util { + @apply shadow-sm focus:ring-indigo-500 focus:border-indigo-500 block w-full sm:text-sm border-gray-300 rounded-md px-2 py-1 mt-1 + } +} + +code { + font-family: source-code-pro, Menlo, Monaco, Consolas, 'Courier New', + monospace; +} diff --git a/services/frontend/src/index.tsx b/services/frontend/src/index.tsx new file mode 100644 index 0000000..8eac7ea --- /dev/null +++ b/services/frontend/src/index.tsx @@ -0,0 +1,19 @@ +import React from "react"; +import ReactDOM from "react-dom"; +import { BrowserRouter as Router } from 'react-router-dom'; +import App from "./App"; +import reportWebVitals from "./reportWebVitals"; + +ReactDOM.render( + + + + + , + document.getElementById("root") +); + +// If you want to start measuring performance in your app, pass a function +// to log results (for example: reportWebVitals(console.log)) +// or send to an analytics endpoint. Learn more: https://bit.ly/CRA-vitals +reportWebVitals(); diff --git a/services/frontend/src/partials/ProtectedRoute.tsx b/services/frontend/src/partials/ProtectedRoute.tsx new file mode 100644 index 0000000..51072fe --- /dev/null +++ b/services/frontend/src/partials/ProtectedRoute.tsx @@ -0,0 +1,15 @@ +import { Navigate } from "react-router-dom"; + +export type ProtectedRouteProps = { + isAuthenticated: boolean; + authenticationPath: string; + outlet: JSX.Element; +}; + +export default function ProtectedRoute({ isAuthenticated, authenticationPath, outlet }: ProtectedRouteProps) { + if (isAuthenticated) { + return outlet; + } else { + return ; + } +}; \ No newline at end of file diff --git a/services/frontend/src/partials/useClickOutside.tsx b/services/frontend/src/partials/useClickOutside.tsx new file mode 100644 index 0000000..09ec8d5 --- /dev/null +++ b/services/frontend/src/partials/useClickOutside.tsx @@ -0,0 +1,19 @@ +import { useEffect } from "react"; + +const useOutsideClick = (ref: any, callback: any) => { + const handleClick = (e: any) => { + if (ref.current && !ref.current.contains(e.target)) { + callback(); + } + }; + + useEffect(() => { + document.addEventListener("click", handleClick); + + return () => { + document.removeEventListener("click", handleClick); + }; + }); +}; + +export default useOutsideClick; \ No newline at end of file diff --git a/services/frontend/src/react-app-env.d.ts b/services/frontend/src/react-app-env.d.ts new file mode 100644 index 0000000..6431bc5 --- /dev/null +++ b/services/frontend/src/react-app-env.d.ts @@ -0,0 +1 @@ +/// diff --git a/services/frontend/src/reducers/index.ts b/services/frontend/src/reducers/index.ts new file mode 100644 index 0000000..0ac8b4b --- /dev/null +++ b/services/frontend/src/reducers/index.ts @@ -0,0 +1,186 @@ +import { omit } from "lodash"; + +const RESET = "reset"; + +const PROJECT_NAME = "project-name"; +const CANVAS_POSITION = "canvas-position"; + +const ENDPOINT_ALL = "endpoints"; +const ENDPOINT_CREATED = "endpoint-created"; +const ENDPOINT_UPDATED = "endpoint-updated"; +const ENDPOINT_DELETED = "endpoint-deleted"; + +const CONNECTIONS_ALL = "connections"; +const CONNECTION_DETACHED = "connection-detached"; +const CONNECTION_ATTACHED = "connection-attached"; + +const AUTH_LOGIN_SUCCESS = "auth-login-success"; +const AUTH_LOGOUT_SUCCESS = "auth-logout-success"; +const AUTH_SELF = "auth-self" + +const getMatchingSetIndex = (setOfSets: [[string, string]], findSet: [string, string]): number => { + return setOfSets.findIndex((set) => set.toString() === findSet.toString()); +} + +export const initialState = { + projectName: "", + nodes: {}, + connections: [], + canvasPosition: { + top: 0, + left: 0, + scale: 1 + } +} + +export const reducer = (state: any, action: any) => { + let existingIndex; + let _connections; + + switch (action.type) { + case RESET: + return { + ...initialState + } + + case PROJECT_NAME: + return { + ...state, + projectName: action.payload + } + case CANVAS_POSITION: + return { + ...state, + canvasPosition: {...state.canvasPosition, ...action.payload} + } + case ENDPOINT_ALL: + return { + ...state, + nodes: action.payload + } + case ENDPOINT_CREATED: + return { + ...state, + nodes: {...state.nodes, [action.payload.key]: action.payload} + } + case ENDPOINT_DELETED: + return { + ...state, + nodes: {...omit(state.nodes, action.payload.key)} + } + case ENDPOINT_UPDATED: + return { + ...state, + nodes: {...state.nodes, [action.payload.key]: action.payload} + } + case CONNECTIONS_ALL: + return { + ...state, + connections: action.payload.map((x: any) => x) + } + case CONNECTION_DETACHED: + _connections = state.connections; + existingIndex = getMatchingSetIndex(_connections, action.payload); + + if (existingIndex !== -1) { + _connections.splice(existingIndex, 1); + } + + return { + ...state, + connections: [..._connections] + } + case CONNECTION_ATTACHED: + _connections = state.connections; + existingIndex = getMatchingSetIndex(state.connections, action.payload); + + if (existingIndex === -1) { + _connections.push(action.payload); + } + + return { + ...state, + connections: [..._connections] + } + case AUTH_LOGIN_SUCCESS: + return { + ...state, + user: { ...action.payload.user } + } + case AUTH_SELF: + return { + ...state, + user: { ...action.payload } + } + case AUTH_LOGOUT_SUCCESS: + return { + ...state, + user: null + } + default: + throw new Error() + } +} + +export const updateProjectName = (data: string) => ({ + type: PROJECT_NAME, + payload: data +}); + +export const position = (data: any) => ({ + type: CANVAS_POSITION, + payload: data +}); + +export const connections = (data: any) => ({ + type: CONNECTIONS_ALL, + payload: data || [] +}); + +export const connectionDetached = (data: any) => ({ + type: CONNECTION_DETACHED, + payload: data +}); + +export const connectionAttached = (data: any) => ({ + type: CONNECTION_ATTACHED, + payload: data +}); + +export const nodes = (data: any) => ({ + type: ENDPOINT_ALL, + payload: data || {} +}); + +export const nodeCreated = (data: any) => ({ + type: ENDPOINT_CREATED, + payload: data +}); + +export const nodeUpdated = (data: any) => ({ + type: ENDPOINT_UPDATED, + payload: data +}); + +export const nodeDeleted = (data: any) => ({ + type: ENDPOINT_DELETED, + payload: data +}); + +export const authLoginSuccess = (data: any) => ({ + type: AUTH_LOGIN_SUCCESS, + payload: data +}); + +export const authLogoutSuccess = () => ({ + type: AUTH_LOGOUT_SUCCESS, +}); + +export const authSelf = (data: any) => ({ + type: AUTH_SELF, + payload: data +}); + +export const resetState = () => ({ + type: RESET +}) \ No newline at end of file diff --git a/services/frontend/src/reportWebVitals.ts b/services/frontend/src/reportWebVitals.ts new file mode 100644 index 0000000..49a2a16 --- /dev/null +++ b/services/frontend/src/reportWebVitals.ts @@ -0,0 +1,15 @@ +import { ReportHandler } from 'web-vitals'; + +const reportWebVitals = (onPerfEntry?: ReportHandler) => { + if (onPerfEntry && onPerfEntry instanceof Function) { + import('web-vitals').then(({ getCLS, getFID, getFCP, getLCP, getTTFB }) => { + getCLS(onPerfEntry); + getFID(onPerfEntry); + getFCP(onPerfEntry); + getLCP(onPerfEntry); + getTTFB(onPerfEntry); + }); + } +}; + +export default reportWebVitals; diff --git a/services/frontend/src/services/auth.ts b/services/frontend/src/services/auth.ts new file mode 100644 index 0000000..9d2f930 --- /dev/null +++ b/services/frontend/src/services/auth.ts @@ -0,0 +1,44 @@ +import { API_SERVER_URL } from "../constants"; +import { getLocalStorageJWTKeys } from "./utils"; + +export const signup = (username: string, email: string, password1: string, password2: string) => + fetch(`${API_SERVER_URL}/auth/registration/`, { + method: "POST", + headers: { + "Content-Type": "application/json" + }, + body: JSON.stringify({ username, email, password1, password2 }) + }); + +export const logIn = (username: string, password: string) => + fetch(`${API_SERVER_URL}/auth/login/`, { + method: "POST", + headers: { + "Content-Type": "application/json" + }, + body: JSON.stringify({ username, password }) + }); + +export const self = () => { + const jwtKeys = getLocalStorageJWTKeys(); + return fetch(`${API_SERVER_URL}/auth/self/`, { + method: "GET", + headers: { + "Content-Type": "application/json", + "Authorization": `Bearer ${jwtKeys.access_token}` + } + }) +} + +export const refresh = () => { + const jwtKeys = getLocalStorageJWTKeys(); + const body = { "refresh": jwtKeys.refresh_token }; + + return fetch(`${API_SERVER_URL}/auth/token/refresh/`, { + method: "POST", + headers: { + "Content-Type": "application/json" + }, + body: JSON.stringify(body) + }) +} diff --git a/services/frontend/src/services/generate.ts b/services/frontend/src/services/generate.ts new file mode 100644 index 0000000..2e7ab80 --- /dev/null +++ b/services/frontend/src/services/generate.ts @@ -0,0 +1,12 @@ +import { IGeneratePayload } from "../types"; +import { API_SERVER_URL } from "../constants"; + +export const generateHttp = (data: IGeneratePayload) => { + return fetch(`${API_SERVER_URL}/generate/`, { + method: "POST", + headers: { + "Content-Type": "application/json" + }, + body: JSON.stringify(data) + }); +} diff --git a/services/frontend/src/services/helpers.ts b/services/frontend/src/services/helpers.ts new file mode 100644 index 0000000..854004f --- /dev/null +++ b/services/frontend/src/services/helpers.ts @@ -0,0 +1,23 @@ +export const checkHttpStatus = (response: any) => { + if ([200, 201, 202].includes(response.status)) { + return response.json(); + } + + if (response.status === 204) { + return response; + } + + throw response; +} + +export const checkHttpSuccess = (response: any) => { + if ([200, 201, 202].includes(response.status)) { + return response.json(); + } + + if (response.status === 204) { + return response; + } + + throw response; +} diff --git a/services/frontend/src/services/project.ts b/services/frontend/src/services/project.ts new file mode 100644 index 0000000..bdc027f --- /dev/null +++ b/services/frontend/src/services/project.ts @@ -0,0 +1,62 @@ +import { IProjectPayload } from "../types"; +import { API_SERVER_URL, PROJECTS_FETCH_LIMIT } from "../constants"; +import { getLocalStorageJWTKeys } from "./utils"; + +export const projectHttpCreate = (data: string) => { + //const jwtKeys = getLocalStorageJWTKeys(); + return fetch(`${API_SERVER_URL}/projects/`, { + method: "POST", + headers: { + "Content-Type": "application/json", + //"Authorization": `Bearer ${jwtKeys.access_token}` + }, + body: data + }); +} + +export const projectHttpUpdate = (uuid: string, data: string) => { + //const jwtKeys = getLocalStorageJWTKeys(); + return fetch(`${API_SERVER_URL}/projects/${uuid}/`, { + method: "PUT", + headers: { + "Content-Type": "application/json", + //"Authorization": `Bearer ${jwtKeys.access_token}` + }, + body: data + }); +} + +export const projectHttpDelete = (uuid: number) => { + const jwtKeys = getLocalStorageJWTKeys(); + return fetch(`${API_SERVER_URL}/projects/${uuid}/`, { + method: "DELETE", + headers: { + "Content-Type": "application/json", + "Authorization": `Bearer ${jwtKeys.access_token}` + } + }); +} + +export const projectsHttpGet = (offset: number) => { + const jwtKeys = getLocalStorageJWTKeys(); + let endpoint = `${API_SERVER_URL}/projects/?limit=${PROJECTS_FETCH_LIMIT}&offset=${offset}`; + + return fetch(endpoint, { + method: "GET", + headers: { + "Content-Type": "application/json", + "Authorization": `Bearer ${jwtKeys.access_token}` + } + }); +} + +export const projectHttpGet = (uuid: string) => { + //const jwtKeys = getLocalStorageJWTKeys(); + return fetch(`${API_SERVER_URL}/projects/${uuid}/`, { + method: "GET", + headers: { + "Content-Type": "application/json", + //"Authorization": `Bearer ${jwtKeys.access_token}` + } + }); +} diff --git a/services/frontend/src/services/utils.ts b/services/frontend/src/services/utils.ts new file mode 100644 index 0000000..930bf60 --- /dev/null +++ b/services/frontend/src/services/utils.ts @@ -0,0 +1,11 @@ +import { LOCAL_STORAGE } from "../constants"; + +export const getLocalStorageJWTKeys = () => { + let jwtKeys = localStorage.getItem(LOCAL_STORAGE); + + if (jwtKeys) { + return JSON.parse(jwtKeys); + } + + return null; +} \ No newline at end of file diff --git a/services/frontend/src/setupTests.ts b/services/frontend/src/setupTests.ts new file mode 100644 index 0000000..8f2609b --- /dev/null +++ b/services/frontend/src/setupTests.ts @@ -0,0 +1,5 @@ +// jest-dom adds custom jest matchers for asserting on DOM nodes. +// allows you to do things like: +// expect(element).toHaveTextContent(/react/i) +// learn more: https://github.com/testing-library/jest-dom +import '@testing-library/jest-dom'; diff --git a/services/frontend/src/types/enums.ts b/services/frontend/src/types/enums.ts new file mode 100644 index 0000000..99cd90f --- /dev/null +++ b/services/frontend/src/types/enums.ts @@ -0,0 +1,3 @@ +export enum NodeGroupType { + Services = "services" +} diff --git a/services/frontend/src/types/index.ts b/services/frontend/src/types/index.ts new file mode 100644 index 0000000..f36c935 --- /dev/null +++ b/services/frontend/src/types/index.ts @@ -0,0 +1,105 @@ +import { AnchorId } from "@jsplumb/common"; +import { Dictionary } from "lodash"; +import { NodeGroupType } from "./enums"; + +export interface IContainer { + name: string; + args?: string[]; + command?: string[]; + image: string; + imagePullPolicy: string; +} + +export interface INodeLibraryItem { + Id: number; + Name: string; + Type: string; + Description: string; + NoInputs: number; + NoOutputs: number; + IsActive: boolean; +} + +export interface INodeGroup { + Id: number; + Name: NodeGroupType; + Description: string; + NodeTypes: INodeLibraryItem[]; +} + +export interface IDockerCompose { + version: string; + services: any[]; +} + +interface INodeItem { + key: string; + type: string; + position: { left: number; top: number }; + inputs: string[]; + outputs: string[]; +} + +export interface IFlatConnection { + target: string; +} + +export interface IBaseConfiguration { + prettyName: string; + name: string; + description: string; + type: string; +} + +export interface IGraphData { + nodes: IClientNodeItem[]; + connections: Dictionary; +} + +export interface IClientNodeItem extends INodeItem { + outputs: string[]; + configuration: IBaseConfiguration; +} + +export interface IServiceNodeItem extends INodeItem { + configuration: IBaseConfiguration; +} + +export interface IAnchor { + id: string; + position: AnchorId; +} + +export interface IService { + name: string; + labels: {} +} + +export interface IProjectPayload { + name: string; + data: { + canvas: { + position: any; + nodes: any; + connections: any; + }; + configs: []; + networks: []; + secrets: []; + services: IService[]; + version: number; + volumes: []; + } +} + +export interface IGeneratePayload { + data: { + configs: []; + networks: []; + secrets: []; + services: IService[]; + connections: [[string, string]]; + version: number; + volumes: []; + } +} \ No newline at end of file diff --git a/services/frontend/src/utils/clickOutside.tsx b/services/frontend/src/utils/clickOutside.tsx new file mode 100644 index 0000000..53da02e --- /dev/null +++ b/services/frontend/src/utils/clickOutside.tsx @@ -0,0 +1,25 @@ +import { useEffect } from "react"; + + +export const useClickOutside = (ref: any, onClickOutside: any) => { + useEffect( + () => { + const listener = (event: any) => { + if (!ref.current || ref.current.contains(event.target)) { + return + } + + onClickOutside(event) + } + + document.addEventListener("mousedown", listener) + document.addEventListener("touchstart", listener) + + return () => { + document.removeEventListener("mousedown", listener) + document.removeEventListener("touchstart", listener) + } + }, + [ref, onClickOutside] + ) +} \ No newline at end of file diff --git a/services/frontend/src/utils/data/libraries.ts b/services/frontend/src/utils/data/libraries.ts new file mode 100644 index 0000000..44d5f00 --- /dev/null +++ b/services/frontend/src/utils/data/libraries.ts @@ -0,0 +1,22 @@ +import { NodeGroupType } from "../../types/enums"; +import { INodeGroup } from "../../types"; + + +export const nodeLibraries: INodeGroup[] = [ + { + Id: 1, + Name: NodeGroupType.Services, + Description: "Services", + NodeTypes: [ + { + Id: 1, + Name: "service", + Type: "SERVICE", + Description: "Service node", + NoInputs: 1, + NoOutputs: 1, + IsActive: true + } + ] + } +]; diff --git a/services/frontend/src/utils/data/serviceNode.ts b/services/frontend/src/utils/data/serviceNode.ts new file mode 100644 index 0000000..276c328 --- /dev/null +++ b/services/frontend/src/utils/data/serviceNode.ts @@ -0,0 +1 @@ +export const ServiceNodeConfiguration = '{"prettyName":"","name":"","key":"service","type":"SERVICE","inputs":["op_source"],"outputs":[]}'; \ No newline at end of file diff --git a/services/frontend/src/utils/data/startConfig.ts b/services/frontend/src/utils/data/startConfig.ts new file mode 100644 index 0000000..17bdb1d --- /dev/null +++ b/services/frontend/src/utils/data/startConfig.ts @@ -0,0 +1 @@ +export const StartConfigString = '[]'; \ No newline at end of file diff --git a/services/frontend/src/utils/generators.ts b/services/frontend/src/utils/generators.ts new file mode 100644 index 0000000..145af7a --- /dev/null +++ b/services/frontend/src/utils/generators.ts @@ -0,0 +1,41 @@ +import { + IClientNodeItem, + IService, + IGeneratePayload, +} from "../types"; +import { Dictionary } from "lodash"; + +const getServices = (graphNodes: Dictionary): IService[] => { + let ret: IService[] = []; + for (const [, value] of Object.entries(graphNodes)) { + ret.push({ + "name": value.configuration.name, + "labels": { + "key": value.key + } + }); + } + + return ret; +} + +export const flattenGraphData = (graphData: any): IGeneratePayload => { + const nodes = graphData["nodes"]; + const base: IGeneratePayload = { + data: { + version: 3, + configs: [], + networks: [], + secrets: [], + services: [], + connections: graphData["connections"], + volumes: [] + } + }; + + getServices(nodes).forEach((x) => { + base.data.services.push(x); + }); + + return base; +} diff --git a/services/frontend/src/utils/index.ts b/services/frontend/src/utils/index.ts new file mode 100644 index 0000000..7aad604 --- /dev/null +++ b/services/frontend/src/utils/index.ts @@ -0,0 +1,238 @@ +import { v4 as uuidv4 } from "uuid"; +import { Connection } from "@jsplumb/core"; +import toast from "react-hot-toast"; +import { + Dictionary, + flattenDeep, + isArray, + isPlainObject, + keyBy, + range, + values +} from "lodash"; +import { + IClientNodeItem, + IServiceNodeItem, + INodeLibraryItem, + INodeGroup, + IContainer +} from "../types"; + +interface IConf { + prettyName: string; + name: string; + description: string; + type: string; + container?: IContainer; +} + +interface IServiceConf { + prettyName: string; + name: string; + template: string; +} + +export function ensure(argument: T | undefined | null, message: string = 'This value was promised to be there.'): T { + if (argument === undefined || argument === null) { + throw new TypeError(message); + } + + return argument; +} + +export const parseSingleNode = ( + configurationStr: string +): IServiceNodeItem => { + let node: IServiceNodeItem = {} as IServiceNodeItem; + let configurationObj: any = null; + try { + configurationObj = JSON.parse(configurationStr); + } catch (err) { } + + if (isPlainObject(configurationObj)) { + node = configurationObj; + } + + return node; +} + +export const formatName = (name: string): string => { + let regExpr = /[^a-zA-Z0-9]/g; + return name.replace(regExpr, "-").toLowerCase(); +} + +export const parseConfiguration = ( + configurationStr: string +): IServiceNodeItem[] => { + let nodes: IServiceNodeItem[] = []; + let configurationObj: any = null; + + try { + configurationObj = JSON.parse(configurationStr); + } catch (err) { } + + if (isPlainObject(configurationObj)) { + nodes = flattenDeep(values(configurationObj)); + } + + if (isArray(configurationObj)) { + nodes = configurationObj + } + + nodes.forEach((node) => { + if (!Array.isArray(node.inputs)) { + node.inputs = []; + } + }); + + return nodes; +} + +export const flattenLibraries = ( + sections: INodeGroup[] +): INodeLibraryItem[] => { + return flattenDeep(sections.map((x) => x.NodeTypes)); +} + +const getEndPointUuids = ( + key: string, + type: "ip" | "op", + _count: string | number +): string[] => { + let count = parseInt(_count as string); + + return range(0, count).map((x) => { + if (count === 1) { + return `${type}_${key}`; + } else if (count === 2) { + return `${type}_${[true, false][x % 2]}_${key}`; + } else { + return `${type}_${x}_${key}`; + } + }); +} + +export const attachUUID = (key: string): string => { + const v4 = new RegExp(/[0-9A-F]{8}-[0-9A-F]{4}-4[0-9A-F]{3}-[89AB][0-9A-F]{3}-[0-9A-F]{12}/i); + + if (key.match(v4)) { + return key; + } + + return key + "-" + uuidv4(); +} + +export const getClientNodeItem = ( + nodeItem: IServiceNodeItem, + library: INodeLibraryItem +): IClientNodeItem => { + const uniqueKey = attachUUID(nodeItem.key); + + return { + key: uniqueKey, + type: nodeItem.type, + position: nodeItem.position, + inputs: getEndPointUuids(uniqueKey, "ip", library.NoInputs), + configuration: { + ...nodeItem.configuration, + name: formatName(nodeItem.configuration.prettyName) + }, + outputs: getEndPointUuids(uniqueKey, "op", library.NoOutputs) + }; +} + +export const getConnections = (connections: Connection[]): Array<[string, string]> => { + let ret: Array<[string, string]> = []; + + /* + connections.forEach((x) => { + [x.sourceId, x.targetId] + }); + */ + + /* + for (let [, value] of connections) { + ret.push([value.sourceId, value.targetId]); + } + + + Object.keys(connections).map((k) => { + const connection: Connection = connections[k]; + return [connections[k].sourceId, connections[k].targetId]; + }); + */ + + connections.forEach((connection: Connection) => { + ret.push([connection.sourceId, connection.targetId]); + }); + + return ret; +} + +export const getClientNodesAndConnections = ( + nodeItems: IServiceNodeItem[], + sections: INodeGroup[] +): Dictionary => { + if (!Array.isArray(nodeItems) || !Array.isArray(sections)) { + return {}; + } + + let libraries = flattenLibraries(sections); + let clientItems = nodeItems.map((x) => { + return getClientNodeItem(x, ensure(libraries.find((l) => l.Type === x.type))); + }); + + return keyBy(clientItems, (x) => x.key); +} + +export const getNodeKeyFromConnectionId = (uuid: string) => { + let key = uuid.substr(uuid.lastIndexOf("_") + 1); + return key; +} + +export const initialValues = (): IConf => { + return { + prettyName: "Unnamed", + name: "unnamed", + description: "", + type: "", + container: { + name: "", + image: "", + imagePullPolicy: "" + } + } +} + +export const serviceInitialValues = (): IServiceConf => { + return { + prettyName: "Unnamed", + name: "unnamed", + template: "" + } +} + +export const toaster = (message: string, type: string) => { + const toastConfig = { + duration: 3000, + position: 'bottom-right', + style: {}, + className: 'text-sm rounded-md text-gray-600 bg-white dark:text-white dark:bg-gray-600' + }; + + if (type === "error") { + toast.error(message, toastConfig as any); + } + + if (type === "success") { + toast.success(message, toastConfig as any); + } +} + +export const truncateStr = (str: string, length: number) => { + if (str.length > length) { + return str.slice(0, length) + '...' + } + + return str +} \ No newline at end of file diff --git a/services/frontend/src/utils/options.ts b/services/frontend/src/utils/options.ts new file mode 100644 index 0000000..e0aa390 --- /dev/null +++ b/services/frontend/src/utils/options.ts @@ -0,0 +1,71 @@ +import { EndpointOptions, DotEndpoint } from "@jsplumb/core"; +import { BezierConnector } from "@jsplumb/connector-bezier"; +import { AnchorId, PaintStyle } from "@jsplumb/common"; +import { BrowserJsPlumbDefaults } from "@jsplumb/browser-ui"; + + +const connectorPaintStyle: PaintStyle = { + strokeWidth: 2, + stroke: "#61B7CF" +} + +const connectorHoverStyle: PaintStyle = { + strokeWidth: 3, + stroke: "#216477" +} + +const endpointHoverStyle: PaintStyle = { + fill: "#216477", + stroke: "#216477" +} + +export const defaultOptions: BrowserJsPlumbDefaults = { + dragOptions: { + cursor: "move" + } +} + +export const sourceEndpoint: EndpointOptions = { + endpoint: { + type: DotEndpoint.type, + options: { + radius: 6 + } + }, + paintStyle: { + stroke: "#097963", + fill: "#16A085", + strokeWidth: 1 + }, + source: true, + connector: { + type: BezierConnector.type, + options:{ + curviness: 50 + } + }, + connectorStyle: connectorPaintStyle, + hoverPaintStyle: endpointHoverStyle, + connectorHoverStyle: connectorHoverStyle, + maxConnections: -1 +} + +export const targetEndpoint: EndpointOptions = { + endpoint: { + type: DotEndpoint.type, + options: { + radius: 6 + } + }, + paintStyle: { + stroke: "#7d0fc8", + fill: "#ad35ff", + strokeWidth: 1 + }, + hoverPaintStyle: endpointHoverStyle, + maxConnections: -1, + target: true +} + +export const inputAnchors: AnchorId[] = ["TopLeft", "BottomLeft", "Left"]; +export const outputAnchors: AnchorId[] = ["TopRight", "BottomRight", "Right"]; diff --git a/services/frontend/src/utils/position.ts b/services/frontend/src/utils/position.ts new file mode 100644 index 0000000..895dd73 --- /dev/null +++ b/services/frontend/src/utils/position.ts @@ -0,0 +1,74 @@ +import { IServiceNodeItem } from "../types"; +import * as d3 from "d3"; +import { getNodeKeyFromConnectionId } from "./index"; + +interface INodeItemWithParent extends IServiceNodeItem { + parent: string; +} + +const nodeWidth = 150; +const nodeHeight = 60; + +export const getHierarchyTree = ( + nodes: IServiceNodeItem[] +): d3.HierarchyPointNode => { + let data = nodes.map( + (node): INodeItemWithParent => { + return { + ...node, + parent: node.inputs[0] ? getNodeKeyFromConnectionId(node.inputs[0]) : "" + }; + } + ); + + let parents = data.filter((x) => !x.parent); + + if (parents.length > 1) { + parents.forEach((x) => { + x.parent = "root_parent"; + }); + data.push({ + key: "root_parent", + parent: "" + } as INodeItemWithParent); + } + + let hierarchy = d3 + .stratify() + .id(function (d: INodeItemWithParent) { + return d.key; + }) + .parentId(function (d: INodeItemWithParent) { + return d.parent; + })(data); + + let tree = d3.tree().nodeSize([nodeHeight, nodeWidth])( + hierarchy + ); + + return tree; +} + +export const getNodesPositions = ( + nodes: IServiceNodeItem[] +): [IServiceNodeItem[], number, number] => { + let nodeWithPosition: IServiceNodeItem[] = []; + let tree = getHierarchyTree(nodes); + let x0 = Infinity; + let x1 = -x0; + + tree.each((d) => { + if (d.x > x1) x1 = d.x; + if (d.x < x0) x0 = d.x; + }); + + let descendants = tree.descendants(); + + descendants.forEach((x) => { + if (x.data.key !== "root_parent") { + nodeWithPosition.push({ ...x.data, position: { left: x.y, top: x.x + nodeHeight } }); + } + }); + + return [nodeWithPosition, 0, 0]; +} diff --git a/services/frontend/tailwind.config.js b/services/frontend/tailwind.config.js new file mode 100644 index 0000000..0d600f6 --- /dev/null +++ b/services/frontend/tailwind.config.js @@ -0,0 +1,53 @@ +module.exports = { + darkMode: "class", + content: [ + "./src/**/*.{js,jsx,ts,tsx}", + ], + theme: { + fontFamily: { + display: ['Gilroy', 'sans-serif'], + body: ['Graphik', 'sans-serif'], + }, + extend: { + typography: (theme) => ({ + DEFAULT: { + css: { + pre: { + 'color': '#fff', + 'line-height': '1.4', + 'margin-top': '0', + 'padding': '', + 'padding-top': '0.4rem', + 'padding-right': '1em', + 'padding-bottom': '0.4rem', + 'padding-left': '1em' + }, + code: false, + 'pre code': false, + 'code::before': false, + 'code::after': false, + p: { + 'margin-top': '1em', + 'margin-bottom': '1em' + } + } + } + }) + }, + height: theme => ({ + auto: 'auto', + ...theme('spacing'), + full: '100%', + screen: 'calc(var(--vh) * 100)', + }), + minHeight: theme => ({ + ...theme('spacing'), + full: '100%', + screen: 'calc(var(--vh) * 100)', + }) + }, + plugins: [ + require('@tailwindcss/forms'), + require('@tailwindcss/typography') + ] +} diff --git a/services/frontend/tsconfig.json b/services/frontend/tsconfig.json new file mode 100644 index 0000000..f6268c2 --- /dev/null +++ b/services/frontend/tsconfig.json @@ -0,0 +1,26 @@ +{ + "compilerOptions": { + "target": "es2017", + "lib": [ + "dom", + "dom.iterable", + "esnext" + ], + "allowJs": true, + "skipLibCheck": true, + "esModuleInterop": true, + "allowSyntheticDefaultImports": true, + "strict": true, + "forceConsistentCasingInFileNames": true, + "noFallthroughCasesInSwitch": true, + "module": "esnext", + "moduleResolution": "node", + "resolveJsonModule": true, + "isolatedModules": true, + "noEmit": true, + "jsx": "react-jsx" + }, + "include": [ + "./src/**/*" + ], +} diff --git a/src/backend/Dockerfile b/src/backend/Dockerfile deleted file mode 100644 index 71e65bb..0000000 --- a/src/backend/Dockerfile +++ /dev/null @@ -1,45 +0,0 @@ -FROM python:3.6-stretch - -COPY server /home/app -COPY configs/supervisor/backend.conf /etc/supervisor/conf.d/backend.conf -COPY configs/uwsgi /home/config/uwsgi -COPY requirements.txt /home/requirements.txt -COPY docker-entrypoint.sh /docker-entrypoint.sh - -RUN apt-get update && \ - apt-get install -y \ - software-properties-common \ - build-essential - -RUN apt-get update && \ - apt-get install -y \ - postgresql \ - postgresql-contrib \ - wget \ - git \ - htop \ - nano \ - curl \ - lsof \ - supervisor \ - dos2unix && \ - rm -rf /var/lib/apt/lists/* - -RUN useradd uwsgi && adduser uwsgi root -RUN useradd supervisor && adduser supervisor root - -RUN pip install --upgrade pip && \ - pip install -r /home/requirements.txt - -RUN touch /var/log/backend_out.log && \ - touch /var/log/django.log - -RUN chmod g+w -R /var/log/ - -RUN dos2unix /docker-entrypoint.sh -RUN chown -R root /docker-entrypoint.sh -RUN chmod +x /docker-entrypoint.sh - -EXPOSE 9000 9001 - -ENTRYPOINT ["/docker-entrypoint.sh"] diff --git a/src/backend/configs/supervisor/backend.conf b/src/backend/configs/supervisor/backend.conf deleted file mode 100644 index 66f1c39..0000000 --- a/src/backend/configs/supervisor/backend.conf +++ /dev/null @@ -1,7 +0,0 @@ -[program:app-uwsgi] -user = uwsgi -command = /usr/local/bin/uwsgi --ini /tmp/uwsgi/uwsgi.ini -autostart=true -autorestart=true -killasgroup=true -stopsignal=QUIT \ No newline at end of file diff --git a/src/backend/requirements.txt b/src/backend/requirements.txt deleted file mode 100644 index fda7436..0000000 --- a/src/backend/requirements.txt +++ /dev/null @@ -1,27 +0,0 @@ -requests-oauth==0.4.1 -requests-oauthlib==1.3.0 -requests==2.23.0 - -django==3.1.14 -django-storages==1.9.1 -django-filter==2.2.0 -django-cors-headers==3.2.1 -django-user-accounts==3.0.0 -django-rest-auth==0.9.5 -django-prettyjson==0.4.1 -djangorestframework==3.11.2 -djangorestframework-jwt==1.11.0 -django-allauth==0.42.0 - -psycopg2-binary==2.8.4 -zappa==0.48.2 -uwsgi==2.0.19.1 -django-anymail[mailgun]==7.0.0 -botocore==1.15.34 -boto3==1.12.34 -python-magic==0.4.15 -Pygments==2.7.4 -pyaml==20.4.0 -docker==4.2.0 -ruamel.yaml==0.16.10 -better_profanity==0.6.1 \ No newline at end of file diff --git a/src/backend/server/api/LoginViews.py b/src/backend/server/api/LoginViews.py deleted file mode 100644 index 745b2be..0000000 --- a/src/backend/server/api/LoginViews.py +++ /dev/null @@ -1,89 +0,0 @@ -import re - -from django.utils.translation import gettext as _ -from django.conf import settings - -from allauth.socialaccount.models import SocialLogin -from allauth.socialaccount.providers.base import AuthAction -from allauth.socialaccount.providers.github.views import GitHubOAuth2Adapter -from allauth.socialaccount.providers.oauth2.client import OAuth2Client -from allauth.socialaccount.providers.oauth2.views import OAuth2LoginView -from rest_auth.registration.views import SocialConnectView, SocialLoginView -from rest_framework.response import Response -from rest_framework.views import APIView -from rest_framework.permissions import AllowAny - -from .serializers import CallbackSerializer - - -class CallbackMixin: - adapter_class = GitHubOAuth2Adapter - client_class = OAuth2Client - # This is our serializer from above - # You can omit this if you handle CSRF protection in the frontend - serializer_class = CallbackSerializer - - # Not the prettiest but single source of truth - @property - def callback_url(self): - url = self.adapter_class(self.request).get_callback_url( - self.request, - None, - ) - - url = settings.SOCIAL_AUTH_CUSTOM_CALLBACK_PAGE - return url - - -class CallbackCreate(CallbackMixin, SocialLoginView): - """ - Logs the user in with the providers data. - Creates a new user account if it doesn't exist yet. - """ - - -class CallbackConnect(CallbackMixin, SocialConnectView): - """ - Connects a provider's user account to the currently logged in user. - """ - - # You can override this method here if you don't want to - # receive a token. Omit it otherwise. - def get_response(self): - return Response({'detail': _('Connection completed.')}) - - -class Login(APIView): - adapter_class = GitHubOAuth2Adapter - permission_classes = (AllowAny,) - - def post(self, request, format=None): - """ - Returns the URL to the login page of provider's authentication server. - """ - # You should have CSRF protection enabled, see - # https://security.stackexchange.com/a/104390 (point 3). - # Therefore this is a POST endpoint. - # This code is inspired by `OAuth2LoginView.dispatch`. - adapter = self.adapter_class(request) - provider = adapter.get_provider() - app = provider.get_app(request) - view = OAuth2LoginView() - view.request = request - view.adapter = adapter - client = view.get_client(request, app) - # You can modify `action` if you have more steps in your auth flow - action = AuthAction.AUTHENTICATE - auth_params = provider.get_auth_params(request, action) - # You can omit this if you want to validate the state in the frontend - #client.state = SocialLogin.stash_state(request) - url = client.get_redirect_url(adapter.authorize_url, auth_params) - custom_call_back_url = settings.SOCIAL_AUTH_CUSTOM_CALLBACK - - if custom_call_back_url: - url = custom_call_back_url - else: - url = re.sub('api.', 'app.', url) - url = re.sub('staging-api.', 'staging-app.', url) - - return Response({'url': url}) diff --git a/src/backend/server/api/admin.py b/src/backend/server/api/admin.py deleted file mode 100644 index e6226b0..0000000 --- a/src/backend/server/api/admin.py +++ /dev/null @@ -1,44 +0,0 @@ -from django.contrib import admin - -from .models import Project, Recipe, UserProfile - - -class UserProfileAdmin(admin.ModelAdmin): - list_display = ( - 'user', - ) - - -class ProjectAdmin(admin.ModelAdmin): - list_display = ( - 'id', - 'user', - 'name', - 'uuid', - 'mutable', - 'private', - 'anonymous', - 'date_created', - 'date_updated') - - search_fields = [] - list_filter = () - - -class RecipeAdmin(admin.ModelAdmin): - list_display = ( - 'id', - 'user', - 'uuid', - 'name', - 'keywords', - 'date_created', - 'date_updated') - - search_fields = ['name', 'keywords'] - list_filter = () - - -admin.site.register(UserProfile, UserProfileAdmin) -admin.site.register(Project, ProjectAdmin) -admin.site.register(Recipe, RecipeAdmin) diff --git a/src/backend/server/api/generatorViews.py b/src/backend/server/api/generatorViews.py deleted file mode 100644 index aa405c2..0000000 --- a/src/backend/server/api/generatorViews.py +++ /dev/null @@ -1,172 +0,0 @@ -import json -import yaml -import requests -from urllib.parse import unquote - -from yaml.scanner import ScannerError -from requests.exceptions import HTTPError -from django.http import Http404 - -from rest_framework.response import Response -from rest_framework.views import APIView -from rest_framework import status, generics - -from utils.utils import generate_dc, random_string, generate_uuid -from utils.generators import RevereseGenerator - - -class RepoDetail(generics.GenericAPIView): - permission_classes = () - - def get(self, request): - repo_name = request.query_params.get('r', None) - resp = {} - - if repo_name: - if '/' in repo_name: - base_url = "https://hub.docker.com/v2/repositories" - else: - base_url = "https://hub.docker.com/v2/repositories/library" - - _results = requests.get(f"{base_url}/{repo_name}") - data = json.dumps(_results.json()) - resp = json.loads(data) - - return Response(resp) - - -class RepoSearch(generics.GenericAPIView): - permission_classes = () - - def get(self, request): - base_url = "https://hub.docker.com/v2/search/repositories" - search_string = request.query_params.get('q', None) - page = request.query_params.get('page', 1) - resp = {} - - if search_string: - payload = {'query': search_string, 'page': page} - _results = requests.get(f"{base_url}/", params=payload) - data = json.dumps(_results.json()) - resp = json.loads(data) - - return Response(resp) - - -class Tags(generics.GenericAPIView): - permission_classes = () - - def get(self, request): - repo_name = request.query_params.get('r', None) - page = request.query_params.get('page', 1) - page_size = request.query_params.get('page_size', 100) - resp = {} - - if repo_name: - if '/' in repo_name: - base_url = "https://hub.docker.com/v2/repositories" - else: - base_url = "https://hub.docker.com/v2/repositories/library" - payload = { - 'page': page, - 'page_size': page_size - } - _results = requests.get(f"{base_url}/{repo_name}/tags/", params=payload) - data = json.dumps(_results.json()) - resp = json.loads(data) - - return Response(resp) - - -class Import(generics.GenericAPIView): - def post(self, request, format=None): - resp = {} - request_data = request.data - yaml_string = request_data.get('yaml', None) - project_data = request_data.get('project_data', None) - original_canvas = None - url = request_data.get('url', None) - canvas_lookup_by_s_name = {} - canvas_lookup_by_s_uuid = {} - request_data = None - - if project_data: - try: - for canvas_obj in project_data['data']['canvas']: - canvas_lookup_by_s_uuid[canvas_obj['uuid']] = canvas_obj - - for service_obj in project_data['data']['services']: - if service_obj['uuid'] in canvas_lookup_by_s_uuid: - canvas_lookup_by_s_name[service_obj['name']] = canvas_lookup_by_s_uuid[service_obj['uuid']] - except KeyError: - original_canvas = None - - try: - if yaml_string: - reverse_generator = RevereseGenerator( - yaml_string, - canvas_lookup_by_s_uuid=canvas_lookup_by_s_uuid, - canvas_lookup_by_s_name=canvas_lookup_by_s_name) - resp = reverse_generator.yaml_dict_to_system_obj() - resp['code'] = unquote(yaml_string) - - - if url: - try: - response = requests.get(url) - - if response.status_code == 200: - content = response.text - reverse_generator = RevereseGenerator( - content, - canvas_lookup_by_s_uuid=canvas_lookup_by_s_uuid, - canvas_lookup_by_s_name=canvas_lookup_by_s_name) - resp = reverse_generator.yaml_dict_to_system_obj() - resp['code'] = content - - # If the response was successful, no Exception will be raised - response.raise_for_status() - except HTTPError as http_err: - print(f'HTTP error occurred: {http_err}') # Python 3.6 - except ScannerError as err: - #print(f'Yaml error: {err}') - err_string = f'
Malformed yaml error, check to make sure your yaml is correct.
\ -
{err}
\ - Note! Url import expects a raw yaml string.' - return Response(err_string, status=status.HTTP_500_INTERNAL_SERVER_ERROR) - except Exception as err: - #print(f'Other error occurred: {err}') # Python 3.6 - err_string = f'
Malformed yaml error, check to make sure your yaml is correct.
\ -
{err}
\ - Note! Url import expects a raw yaml string.' - return Response(err_string, status=status.HTTP_500_INTERNAL_SERVER_ERROR) - else: - pass - - return Response(resp, status=status.HTTP_200_OK) - - - -class Generate(generics.GenericAPIView): - def post(self, request, format=None): - request_data = request.data - - version = request_data['data'].get('version', '3') - services = request_data['data'].get('services', None) - volumes = request_data['data'].get('volumes', None) - networks = request_data['data'].get('networks', None) - secrets = request_data['data'].get('secrets', None) - configs = request_data['data'].get('configs', None) - - code = generate_dc( - services, - volumes, - networks, - secrets, - configs, - version=version, - return_format='yaml') - resp = {'code': code} - - - return Response(resp, status=status.HTTP_200_OK) diff --git a/src/backend/server/api/initViews.py b/src/backend/server/api/initViews.py deleted file mode 100644 index ed94b92..0000000 --- a/src/backend/server/api/initViews.py +++ /dev/null @@ -1,61 +0,0 @@ -from django.http import Http404 - -from rest_framework.response import Response -from rest_framework.views import APIView -from rest_framework import status, generics - -from .models import Init -from .serializers import InitSerializer -from rest_framework.permissions import IsAuthenticated, IsAdminUser - - -class CreateList(generics.GenericAPIView): - permission_classes = () - objects = Init.objects.all() - serializer_class = InitSerializer - - def get(self, request): - objects = Init.objects.all() - serializer = InitSerializer(objects, many=True) - - return Response(serializer.data) - - def post(self, request): - data = request.data.copy() - serializer = InitSerializer(data=data) - - if serializer.is_valid(): - serializer.save() - - return Response(serializer.data, status=status.HTTP_201_CREATED) - - return Response(serializer.errors, status=status.HTTP_400_BAD_REQUEST) - - -class Detail(generics.GenericAPIView): - def get_object(self, user, obj_id): - try: - return Init.objects.get(id=obj_id) - except Init.DoesNotExist: - raise Http404 - - def get(self, request, obj_id, format=None): - _object = self.get_object(user=request.user, obj_id=obj_id) - serializer = InitSerializer(_object) - return Response(serializer.data, status=status.HTTP_200_OK) - - def put(self, request, obj_id, format=None): - update_data = request.data - _object = self.get_object(user=request.user, obj_id=obj_id) - serializer = InitSerializer(_object, data=update_data) - - if serializer.is_valid(): - serializer.save() - return Response(serializer.data, status=status.HTTP_200_OK) - else: - return Response(serializer.errors, status=status.HTTP_400_BAD_REQUEST) - - def delete(self, request, obj_id, format=None): - _object = self.get_object(user=request.user, obj_id=obj_id) - _object.delete() - return Response(status=status.HTTP_204_NO_CONTENT) diff --git a/src/backend/server/api/migrations/0001_initial.py b/src/backend/server/api/migrations/0001_initial.py deleted file mode 100644 index 8b6bee8..0000000 --- a/src/backend/server/api/migrations/0001_initial.py +++ /dev/null @@ -1,31 +0,0 @@ -# Generated by Django 3.0.4 on 2020-04-29 12:32 - -import django.contrib.postgres.fields.jsonb -from django.db import migrations, models - - -class Migration(migrations.Migration): - - initial = True - - dependencies = [ - ] - - operations = [ - migrations.CreateModel( - name='Project', - fields=[ - ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), - ('date_created', models.DateTimeField(auto_now_add=True)), - ('date_updated', models.DateTimeField(auto_now=True)), - ('name', models.CharField(max_length=255)), - ('uuid', models.CharField(max_length=255)), - ('data', django.contrib.postgres.fields.jsonb.JSONField(blank=True, default=dict, help_text='project json')), - ], - options={ - 'verbose_name': 'Project', - 'verbose_name_plural': 'Projects', - 'ordering': ['-id'], - }, - ), - ] diff --git a/src/backend/server/api/migrations/0003_project_owned.py b/src/backend/server/api/migrations/0003_project_owned.py deleted file mode 100644 index 5204b4b..0000000 --- a/src/backend/server/api/migrations/0003_project_owned.py +++ /dev/null @@ -1,18 +0,0 @@ -# Generated by Django 3.0.4 on 2020-06-07 10:40 - -from django.db import migrations, models - - -class Migration(migrations.Migration): - - dependencies = [ - ('api', '0002_project_mutable'), - ] - - operations = [ - migrations.AddField( - model_name='project', - name='owned', - field=models.BooleanField(default=False), - ), - ] diff --git a/src/backend/server/api/migrations/0004_auto_20200607_1047.py b/src/backend/server/api/migrations/0004_auto_20200607_1047.py deleted file mode 100644 index ab90a12..0000000 --- a/src/backend/server/api/migrations/0004_auto_20200607_1047.py +++ /dev/null @@ -1,18 +0,0 @@ -# Generated by Django 3.0.4 on 2020-06-07 10:47 - -from django.db import migrations - - -class Migration(migrations.Migration): - - dependencies = [ - ('api', '0003_project_owned'), - ] - - operations = [ - migrations.RenameField( - model_name='project', - old_name='owned', - new_name='private', - ), - ] diff --git a/src/backend/server/api/migrations/0005_userprofile.py b/src/backend/server/api/migrations/0005_userprofile.py deleted file mode 100644 index 3620ea1..0000000 --- a/src/backend/server/api/migrations/0005_userprofile.py +++ /dev/null @@ -1,26 +0,0 @@ -# Generated by Django 3.0.4 on 2020-06-08 08:38 - -from django.conf import settings -from django.db import migrations, models -import django.db.models.deletion - - -class Migration(migrations.Migration): - - dependencies = [ - ('auth', '0011_update_proxy_permissions'), - ('api', '0004_auto_20200607_1047'), - ] - - operations = [ - migrations.CreateModel( - name='UserProfile', - fields=[ - ('user', models.OneToOneField(on_delete=django.db.models.deletion.CASCADE, primary_key=True, related_name='profile', serialize=False, to=settings.AUTH_USER_MODEL, verbose_name='user')), - ('avatar_url', models.CharField(blank=True, max_length=256, null=True)), - ], - options={ - 'db_table': 'user_profile', - }, - ), - ] diff --git a/src/backend/server/api/migrations/0006_project_user.py b/src/backend/server/api/migrations/0006_project_user.py deleted file mode 100644 index f2ec5be..0000000 --- a/src/backend/server/api/migrations/0006_project_user.py +++ /dev/null @@ -1,21 +0,0 @@ -# Generated by Django 3.0.4 on 2020-06-08 09:47 - -from django.conf import settings -from django.db import migrations, models -import django.db.models.deletion - - -class Migration(migrations.Migration): - - dependencies = [ - migrations.swappable_dependency(settings.AUTH_USER_MODEL), - ('api', '0005_userprofile'), - ] - - operations = [ - migrations.AddField( - model_name='project', - name='user', - field=models.ForeignKey(blank=True, help_text='user', null=True, on_delete=django.db.models.deletion.CASCADE, related_name='user_projects', to=settings.AUTH_USER_MODEL), - ), - ] diff --git a/src/backend/server/api/migrations/0007_project_anonymous.py b/src/backend/server/api/migrations/0007_project_anonymous.py deleted file mode 100644 index 2f50362..0000000 --- a/src/backend/server/api/migrations/0007_project_anonymous.py +++ /dev/null @@ -1,18 +0,0 @@ -# Generated by Django 3.0.4 on 2020-06-08 10:49 - -from django.db import migrations, models - - -class Migration(migrations.Migration): - - dependencies = [ - ('api', '0006_project_user'), - ] - - operations = [ - migrations.AddField( - model_name='project', - name='anonymous', - field=models.BooleanField(default=True), - ), - ] diff --git a/src/backend/server/api/migrations/0008_recipe.py b/src/backend/server/api/migrations/0008_recipe.py deleted file mode 100644 index 3c8c5af..0000000 --- a/src/backend/server/api/migrations/0008_recipe.py +++ /dev/null @@ -1,36 +0,0 @@ -# Generated by Django 3.0.4 on 2020-06-25 14:19 - -from django.conf import settings -import django.contrib.postgres.fields -import django.contrib.postgres.fields.jsonb -from django.db import migrations, models -import django.db.models.deletion - - -class Migration(migrations.Migration): - - dependencies = [ - migrations.swappable_dependency(settings.AUTH_USER_MODEL), - ('api', '0007_project_anonymous'), - ] - - operations = [ - migrations.CreateModel( - name='Recipe', - fields=[ - ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), - ('date_created', models.DateTimeField(auto_now_add=True)), - ('date_updated', models.DateTimeField(auto_now=True)), - ('uuid', models.CharField(max_length=255)), - ('name', models.CharField(max_length=255)), - ('keywords', django.contrib.postgres.fields.ArrayField(base_field=models.CharField(blank=True, max_length=150), size=None)), - ('data', django.contrib.postgres.fields.jsonb.JSONField(blank=True, default=dict, help_text='recipe json')), - ('user', models.ForeignKey(blank=True, help_text='user', null=True, on_delete=django.db.models.deletion.CASCADE, related_name='user_recipes', to=settings.AUTH_USER_MODEL)), - ], - options={ - 'verbose_name': 'Recipe', - 'verbose_name_plural': 'Recipes', - 'ordering': ['-id'], - }, - ), - ] diff --git a/src/backend/server/api/migrations/0009_auto_20200625_1628.py b/src/backend/server/api/migrations/0009_auto_20200625_1628.py deleted file mode 100644 index 48cd3a1..0000000 --- a/src/backend/server/api/migrations/0009_auto_20200625_1628.py +++ /dev/null @@ -1,19 +0,0 @@ -# Generated by Django 3.0.4 on 2020-06-25 16:28 - -import django.contrib.postgres.fields -from django.db import migrations, models - - -class Migration(migrations.Migration): - - dependencies = [ - ('api', '0008_recipe'), - ] - - operations = [ - migrations.AlterField( - model_name='recipe', - name='keywords', - field=django.contrib.postgres.fields.ArrayField(base_field=models.CharField(blank=True, max_length=150), blank=True, null=True, size=None), - ), - ] diff --git a/src/backend/server/api/migrations/0010_auto_20200627_1319.py b/src/backend/server/api/migrations/0010_auto_20200627_1319.py deleted file mode 100644 index a78fc7b..0000000 --- a/src/backend/server/api/migrations/0010_auto_20200627_1319.py +++ /dev/null @@ -1,25 +0,0 @@ -# Generated by Django 3.0.4 on 2020-06-27 13:19 - -from django.db import migrations, models - - -class Migration(migrations.Migration): - - dependencies = [ - ('api', '0009_auto_20200625_1628'), - ] - - operations = [ - migrations.AddField( - model_name='recipe', - name='branch', - field=models.CharField(default='master', max_length=255), - preserve_default=False, - ), - migrations.AddField( - model_name='recipe', - name='repo', - field=models.CharField(default='', max_length=500), - preserve_default=False, - ), - ] diff --git a/src/backend/server/api/migrations/0011_recipe_desc.py b/src/backend/server/api/migrations/0011_recipe_desc.py deleted file mode 100644 index d10be07..0000000 --- a/src/backend/server/api/migrations/0011_recipe_desc.py +++ /dev/null @@ -1,18 +0,0 @@ -# Generated by Django 3.0.4 on 2020-06-27 19:52 - -from django.db import migrations, models - - -class Migration(migrations.Migration): - - dependencies = [ - ('api', '0010_auto_20200627_1319'), - ] - - operations = [ - migrations.AddField( - model_name='recipe', - name='desc', - field=models.TextField(blank=True, max_length=2500, null=True), - ), - ] diff --git a/src/backend/server/api/migrations/0012_auto_20200628_1117.py b/src/backend/server/api/migrations/0012_auto_20200628_1117.py deleted file mode 100644 index d387eee..0000000 --- a/src/backend/server/api/migrations/0012_auto_20200628_1117.py +++ /dev/null @@ -1,24 +0,0 @@ -# Generated by Django 3.0.4 on 2020-06-28 11:17 - -from django.db import migrations, models - - -class Migration(migrations.Migration): - - dependencies = [ - ('api', '0011_recipe_desc'), - ] - - operations = [ - migrations.AddField( - model_name='recipe', - name='private', - field=models.BooleanField(default=False), - ), - migrations.AddField( - model_name='recipe', - name='raw_url', - field=models.CharField(default='', max_length=500), - preserve_default=False, - ), - ] diff --git a/src/backend/server/api/migrations/0013_auto_20200628_1131.py b/src/backend/server/api/migrations/0013_auto_20200628_1131.py deleted file mode 100644 index 565f9fd..0000000 --- a/src/backend/server/api/migrations/0013_auto_20200628_1131.py +++ /dev/null @@ -1,28 +0,0 @@ -# Generated by Django 3.0.4 on 2020-06-28 11:31 - -from django.db import migrations, models - - -class Migration(migrations.Migration): - - dependencies = [ - ('api', '0012_auto_20200628_1117'), - ] - - operations = [ - migrations.AlterField( - model_name='recipe', - name='branch', - field=models.CharField(blank=True, max_length=255, null=True), - ), - migrations.AlterField( - model_name='recipe', - name='raw_url', - field=models.CharField(blank=True, max_length=500, null=True), - ), - migrations.AlterField( - model_name='recipe', - name='repo', - field=models.CharField(blank=True, max_length=500, null=True), - ), - ] diff --git a/src/backend/server/api/migrations/0014_recipe_url_name.py b/src/backend/server/api/migrations/0014_recipe_url_name.py deleted file mode 100644 index 10d530d..0000000 --- a/src/backend/server/api/migrations/0014_recipe_url_name.py +++ /dev/null @@ -1,19 +0,0 @@ -# Generated by Django 3.0.4 on 2020-07-01 05:18 - -from django.db import migrations, models - - -class Migration(migrations.Migration): - - dependencies = [ - ('api', '0013_auto_20200628_1131'), - ] - - operations = [ - migrations.AddField( - model_name='recipe', - name='url_name', - field=models.CharField(default='', max_length=255), - preserve_default=False, - ), - ] diff --git a/src/backend/server/api/migrations/0015_auto_20200701_2139.py b/src/backend/server/api/migrations/0015_auto_20200701_2139.py deleted file mode 100644 index ebb88fa..0000000 --- a/src/backend/server/api/migrations/0015_auto_20200701_2139.py +++ /dev/null @@ -1,38 +0,0 @@ -# Generated by Django 3.0.4 on 2020-07-01 21:39 - -from django.db import migrations, models - - -class Migration(migrations.Migration): - - dependencies = [ - ('api', '0014_recipe_url_name'), - ] - - operations = [ - migrations.AlterField( - model_name='recipe', - name='branch', - field=models.CharField(blank=True, max_length=500, null=True), - ), - migrations.AlterField( - model_name='recipe', - name='desc', - field=models.TextField(blank=True, null=True), - ), - migrations.AlterField( - model_name='recipe', - name='name', - field=models.CharField(max_length=500), - ), - migrations.AlterField( - model_name='recipe', - name='raw_url', - field=models.CharField(blank=True, max_length=1000, null=True), - ), - migrations.AlterField( - model_name='recipe', - name='url_name', - field=models.CharField(max_length=1000), - ), - ] diff --git a/src/backend/server/api/migrations/0016_auto_20200705_1424.py b/src/backend/server/api/migrations/0016_auto_20200705_1424.py deleted file mode 100644 index d201dc5..0000000 --- a/src/backend/server/api/migrations/0016_auto_20200705_1424.py +++ /dev/null @@ -1,18 +0,0 @@ -# Generated by Django 3.0.4 on 2020-07-05 14:24 - -from django.db import migrations, models - - -class Migration(migrations.Migration): - - dependencies = [ - ('api', '0015_auto_20200701_2139'), - ] - - operations = [ - migrations.AlterField( - model_name='recipe', - name='url_name', - field=models.CharField(blank=True, max_length=1000, null=True), - ), - ] diff --git a/src/backend/server/api/mixins.py b/src/backend/server/api/mixins.py deleted file mode 100644 index f6f0883..0000000 --- a/src/backend/server/api/mixins.py +++ /dev/null @@ -1,9 +0,0 @@ -from django.db import models - - -class TimeStamp(models.Model): - date_created = models.DateTimeField(auto_now_add=True) - date_updated = models.DateTimeField(auto_now=True) - - class Meta: - abstract = True diff --git a/src/backend/server/api/models.py b/src/backend/server/api/models.py deleted file mode 100644 index 61c810d..0000000 --- a/src/backend/server/api/models.py +++ /dev/null @@ -1,187 +0,0 @@ -import hashlib -import requests - -from django.db import models -from django.contrib.postgres.fields import ArrayField -from django.contrib.auth import get_user_model -from django.db.models.signals import post_save -from django.contrib.postgres.fields import JSONField, ArrayField -from django.dispatch import receiver -from django.db.utils import IntegrityError - -from allauth.socialaccount.models import SocialAccount, SocialToken, SocialApp -from allauth.account.signals import user_signed_up, user_logged_in - -try: - from django.utils.encoding import force_text -except ImportError: - from django.utils.encoding import force_unicode as force_text - -from utils.utils import random_string -from .mixins import TimeStamp - - -User = get_user_model() - - -class UserProfile(models.Model): - user = models.OneToOneField( - User, - primary_key=True, - verbose_name='user', - related_name='profile', - on_delete=models.CASCADE) - avatar_url = models.CharField(max_length=256, blank=True, null=True) - - def __str__(self): - return force_text(self.user.email) - - class Meta(): - db_table = 'user_profile' - - -class Recipe(TimeStamp): - user = models.ForeignKey( - User, - related_name='user_recipes', - on_delete=models.CASCADE, - blank=True, - null=True, - help_text='user' - ) - uuid = models.CharField(max_length=255, blank=False, null=False) - name = models.CharField(max_length=500, blank=False, null=False) - url_name = models.CharField(max_length=1000, blank=True, null=True) - desc = models.TextField(blank=True, null=True) - keywords = ArrayField( - models.CharField(max_length=150, blank=True), - blank=True, - null=True, - ) - repo = models.CharField(max_length=500, blank=True, null=True) - branch = models.CharField(max_length=500, blank=True, null=True) - raw_url = models.CharField(max_length=1000, blank=True, null=True) - data = JSONField(default=dict, blank=True, help_text="recipe json") - - # not listed publicly, needs a direct link - private = models.BooleanField(default=False, blank=False, null=False) - - class Meta(object): - verbose_name = 'Recipe' - verbose_name_plural = 'Recipes' - ordering = ['-id'] - - def __str__(self): - return "{}".format(self.id) - - -class Project(TimeStamp): - user = models.ForeignKey( - User, - related_name='user_projects', - on_delete=models.CASCADE, - blank=True, - null=True, - help_text='user' - ) - - name = models.CharField(max_length=255, blank=False, null=False) - uuid = models.CharField(max_length=255, blank=False, null=False) - data = JSONField(default=dict, blank=True, help_text="project json") - - # anonymous projects that can not be mutitated, only cloned anonymously or authenticated - mutable = models.BooleanField(default=True, blank=False, null=False) - - # not listed publicly, needs a direct link - private = models.BooleanField(default=False, blank=False, null=False) - - # for projects created anonymously, stay anonymous forever - # if a user is logged in, they can clone the anonymous project to make it their own - anonymous = models.BooleanField(default=True, blank=False, null=False) - - class Meta(object): - verbose_name = 'Project' - verbose_name_plural = 'Projects' - ordering = ['-id'] - - def __str__(self): - return "{}".format(self.id) - - -@receiver(post_save, sender=SocialToken) -def get_user_email(sender, instance=None, created=False, **kwargs): - django_user_obj = instance.account.user - access_token = instance.token - email = None - - with requests.Session() as session: - params = {'access_token': access_token} - headers = {'Connection': 'close'} - req_return = session.get('https://api.github.com/user/emails', params=params, headers=headers) - status_code = req_return.status_code - - if status_code == 200: - req_return_data = req_return.json()[0] - email = req_return_data['email'] - - session.close() - - try: - if email: - django_user_obj.email = email - django_user_obj.save() - except IntegrityError as e: - print('Exception:', e) - - -@receiver(user_signed_up) -def social_login_fname_lname_profilepic(sociallogin=None, user=None, **kwargs): - preferred_avatar_size_pixels=256 - email_first_part = user.email.split("@")[0] - username_postfix = random_string(5) - username = f"{email_first_part}_{username_postfix}" - user.username = username - - picture_url = "http://www.gravatar.com/avatar/{0}?s={1}".format( - hashlib.md5(user.email.encode('UTF-8')).hexdigest(), - preferred_avatar_size_pixels - ) - - if sociallogin: - # Extract first / last names from social nets and store on User record - if sociallogin.account.provider == 'twitter': - name = sociallogin.account.extra_data['name'] - user.first_name = name.split()[0] - user.last_name = name.split()[1] - - if sociallogin.account.provider == 'facebook': - f_name = sociallogin.account.extra_data['first_name'] - l_name = sociallogin.account.extra_data['last_name'] - if f_name: - user.first_name = f_name - if l_name: - user.last_name = l_name - - #verified = sociallogin.account.extra_data['verified'] - picture_url = "http://graph.facebook.com/{0}/picture?width={1}&height={1}".format( - sociallogin.account.uid, preferred_avatar_size_pixels) - - if sociallogin.account.provider == 'google': - f_name = sociallogin.account.extra_data['given_name'] - l_name = sociallogin.account.extra_data['family_name'] - if f_name: - user.first_name = f_name - if l_name: - user.last_name = l_name - #verified = sociallogin.account.extra_data['verified_email'] - picture_url = sociallogin.account.extra_data['picture'] - - if sociallogin.account.provider == 'github': - #name = sociallogin.account.extra_data['name'] - #user.first_name = name.split()[0] - #user.last_name = name.split()[1] - picture_url = sociallogin.account.extra_data['avatar_url'] - - user.save() - profile = UserProfile(user=user, avatar_url=picture_url) - profile.save() diff --git a/src/backend/server/api/permissions.py b/src/backend/server/api/permissions.py deleted file mode 100644 index 0f179bb..0000000 --- a/src/backend/server/api/permissions.py +++ /dev/null @@ -1,29 +0,0 @@ -from rest_framework import permissions - - - -class IsOwner(permissions.BasePermission): - """ - Object-level permission to only allow owners of an object to edit it. - Assumes the model instance has an `owner` attribute. - """ - - def has_object_permission(self, request, view, obj): - # Instance must have an attribute named `owner`. - return obj.user == request.user - - -class IsOwnerOrReadOnly(permissions.BasePermission): - """ - Object-level permission to only allow owners of an object to edit it. - Assumes the model instance has an `owner` attribute. - """ - - def has_object_permission(self, request, view, obj): - # Read permissions are allowed to any request, - # so we'll always allow GET, HEAD or OPTIONS requests. - if request.method in permissions.SAFE_METHODS: - return True - - # Instance must have an attribute named `owner`. - return obj.owner == request.user \ No newline at end of file diff --git a/src/backend/server/api/projectViews.py b/src/backend/server/api/projectViews.py deleted file mode 100644 index e577826..0000000 --- a/src/backend/server/api/projectViews.py +++ /dev/null @@ -1,146 +0,0 @@ -from django.http import Http404 - -from rest_framework.response import Response -from rest_framework.views import APIView -from rest_framework import status, generics - -from .models import Project -from .serializers import ProjectSerializer, ProjectUpdateSerializer -from rest_framework.permissions import IsAuthenticated, IsAdminUser - -from utils.utils import random_string -from utils.pagination import CustomPagination -from .permissions import IsOwner - - -class ListPrivate(generics.ListAPIView): - permission_classes = [IsAuthenticated] - serializer_class = ProjectSerializer - pagination_class = CustomPagination - queryset = Project.objects.all() - ordering_fields = ('date_updated', 'date_created') - - def filter_queryset(self, queryset): - user = self.request.user - queryset_filters = {'user': user} - queryset = Project.objects.filter(**queryset_filters) - return queryset - - -class DetailPrivate(generics.GenericAPIView): - permission_classes = [IsAuthenticated & IsOwner] - - def get_object(self, user, uuid): - try: - return Project.objects.get(uuid=uuid) - except Project.DoesNotExist: - raise Http404 - - """ - def get(self, request, uuid, format=None): - user = request.user - _object = self.get_object(user=user, uuid=uuid) - self.check_object_permissions(request, _object) - serializer = ProjectSerializer(_object) - return Response(serializer.data, status=status.HTTP_200_OK) - """ - - def put(self, request, uuid, format=None): - user = request.user - update_data = request.data - _object = self.get_object(user=user, uuid=uuid) - self.check_object_permissions(request, _object) - serializer = ProjectUpdateSerializer(_object, data=update_data) - - if serializer.is_valid(): - serializer.save() - return Response(serializer.data, status=status.HTTP_200_OK) - else: - return Response(serializer.errors, status=status.HTTP_400_BAD_REQUEST) - - def delete(self, request, uuid, format=None): - _object = self.get_object(user=request.user, uuid=uuid) - _object.delete() - return Response(status=status.HTTP_204_NO_CONTENT) - - -class CreateList(generics.ListAPIView): - permission_classes = () - serializer_class = ProjectSerializer - pagination_class = CustomPagination - queryset = Project.objects.all() - ordering_fields = ('date_updated', 'date_created') - - def filter_queryset(self, queryset): - queryset_filters = {'user': None, 'private': False} - queryset = Project.objects.filter(**queryset_filters) - return queryset - - def post(self, request): - user = request.user - data = request.data.copy() - data['user'] = user.id - data['uuid'] = random_string() - - if request.user.is_authenticated: - data['anonymous'] = False - data['private'] = True - - serializer = ProjectSerializer(data=data) - - if serializer.is_valid(): - serializer.save() - - return Response(serializer.data, status=status.HTTP_201_CREATED) - - return Response(serializer.errors, status=status.HTTP_400_BAD_REQUEST) - - -class Detail(generics.GenericAPIView): - def get_object(self, user, uuid): - try: - _project = Project.objects.get(uuid=uuid) - - if _project.user == None and _project.private == False: - return _project - - if user.is_authenticated: - if user == _project.user: - return _project - - raise Http404 - except Project.DoesNotExist: - raise Http404 - - def get(self, request, uuid, format=None): - _object = self.get_object(user=request.user, uuid=uuid) - - serializer = ProjectSerializer(_object) - return Response(serializer.data, status=status.HTTP_200_OK) - - def put(self, request, uuid, format=None): - update_data = request.data - _object = self.get_object(user=request.user, uuid=uuid) - - if request.user.is_authenticated: - update_data['anonymous'] = False - update_data['private'] = True - - serializer = ProjectUpdateSerializer(_object, data=update_data) - - # prevent anonymous projects from getting owned by an authenticated user - if serializer.is_valid(): - if request.user.is_authenticated and _object.user != request.user: - update_data['user'] = request.user - update_data['uuid'] = random_string() - serializer.create(validated_data=update_data) - else: - serializer.save() - return Response(serializer.data, status=status.HTTP_200_OK) - else: - return Response(serializer.errors, status=status.HTTP_400_BAD_REQUEST) - - def delete(self, request, uuid, format=None): - _object = self.get_object(user=request.user, uuid=uuid) - _object.delete() - return Response(status=status.HTTP_204_NO_CONTENT) diff --git a/src/backend/server/api/recipeViews.py b/src/backend/server/api/recipeViews.py deleted file mode 100644 index 6d147af..0000000 --- a/src/backend/server/api/recipeViews.py +++ /dev/null @@ -1,205 +0,0 @@ -import html -import requests - -from yaml.scanner import ScannerError -from requests.exceptions import HTTPError - -from django.http import Http404 - -from rest_framework.response import Response -from rest_framework.views import APIView -from rest_framework import status, generics - -from .models import Recipe -from .serializers import RecipeSerializer, RecipeUpdateSerializer, RecipeReadSerializer -from rest_framework.permissions import IsAuthenticated, IsAdminUser - -from utils.utils import random_string, clean_string -from utils.pagination import CustomPagination -from utils.generators import RevereseGenerator -from .permissions import IsOwner - - -class CreateList(generics.ListAPIView): - permission_classes = [IsAuthenticated] - serializer_class = RecipeReadSerializer - pagination_class = CustomPagination - queryset = Recipe.objects.all() - ordering_fields = ('date_updated', 'date_created') - - def filter_queryset(self, queryset): - user = self.request.user - queryset_filters = {'user': user} - queryset = Recipe.objects.filter(**queryset_filters) - return queryset - - def post(self, request): - user = request.user - data = request.data.copy() - data['user'] = user.id - data['uuid'] = random_string() - data['data'] = data.get('data', {}) - data['url_name'] = clean_string(data['name']) - repo = data.get('repo', None) - branch = data.get('branch', 'master') - raw_url = data.get('raw_url', None) - - try: - if not repo and not raw_url: - return Response({'message': 'Empty repo or raw url!'}, status=status.HTTP_400_BAD_REQUEST) - - if repo: - _recipe_obj = Recipe.objects.get(repo=repo, branch=branch) - else: - _recipe_obj = Recipe.objects.get(raw_url=raw_url) - - return Response({'message': 'Recipe already exists!'}, status=status.HTTP_400_BAD_REQUEST) - except Recipe.DoesNotExist: - serializer = RecipeSerializer(data=data) - - if serializer.is_valid(): - serializer.save() - - return Response(serializer.data, status=status.HTTP_201_CREATED) - - return Response(serializer.errors, status=status.HTTP_400_BAD_REQUEST) - - -class DetailByUrlName(generics.GenericAPIView): - def get_object(self, url_name): - try: - _recipe = Recipe.objects.get(url_name=url_name) - return _recipe - except Recipe.DoesNotExist: - raise Http404 - - def get(self, request, url_name, format=None): - _object = self.get_object(url_name=url_name) - serialized = RecipeReadSerializer(_object).data - - return Response(serialized, status=status.HTTP_200_OK) - - -class Detail(generics.GenericAPIView): - def get_object(self, user, uuid): - try: - _recipe = Recipe.objects.get(user=user.id, uuid=uuid) - return _recipe - except Recipe.DoesNotExist: - raise Http404 - - def get_object_by_uuid(self, uuid): - try: - _recipe = Recipe.objects.get(uuid=uuid) - return _recipe - except Recipe.DoesNotExist: - raise Http404 - - def get(self, request, uuid, format=None): - resp = {} - _object = self.get_object_by_uuid(uuid=uuid) - raw_url = _object.raw_url - canvas_lookup_by_s_name = {} - canvas_lookup_by_s_uuid = {} - - try: - if raw_url: - try: - response = requests.get(raw_url) - - if response.status_code == 200: - content = response.text - reverse_generator = RevereseGenerator( - content, - canvas_lookup_by_s_uuid=canvas_lookup_by_s_uuid, - canvas_lookup_by_s_name=canvas_lookup_by_s_name) - resp = reverse_generator.yaml_dict_to_system_obj() - - resp['name'] = _object.name - resp['code'] = content - - # If the response was successful, no Exception will be raised - response.raise_for_status() - except HTTPError as http_err: - print(f'HTTP error occurred: {http_err}') # Python 3.6 - except ScannerError as err: - #print(f'Yaml error: {err}') - err_string = f'
Malformed yaml error, check to make sure your yaml is correct.
\ -
{err}
\ - Note! Url import expects a raw yaml string.' - return Response(err_string, status=status.HTTP_500_INTERNAL_SERVER_ERROR) - except Exception as err: - #print(f'Other error occurred: {err}') # Python 3.6 - err_string = f'
Malformed yaml error, check to make sure your yaml is correct.
\ -
{err}
\ - Note! Url import expects a raw yaml string.' - return Response(err_string, status=status.HTTP_500_INTERNAL_SERVER_ERROR) - else: - pass - - return Response(resp, status=status.HTTP_200_OK) - - def put(self, request, uuid, format=None): - update_data = request.data - _object = self.get_object(user=request.user, uuid=uuid) - update_data['desc'] = html.escape(update_data['desc']) - update_data['url_name'] = clean_string(update_data['name']) - serializer = RecipeUpdateSerializer(_object, data=update_data) - - if serializer.is_valid(): - if request.user.is_authenticated and _object.user != request.user: - update_data['user'] = request.user - update_data['uuid'] = random_string() - serializer.create(validated_data=update_data) - else: - serializer.save() - return Response(serializer.data, status=status.HTTP_200_OK) - else: - return Response(serializer.errors, status=status.HTTP_400_BAD_REQUEST) - - def delete(self, request, uuid, format=None): - _object = self.get_object(user=request.user, uuid=uuid) - _object.delete() - return Response(status=status.HTTP_204_NO_CONTENT) - - -class Popular(generics.ListAPIView): - permission_classes = () - serializer_class = RecipeReadSerializer - pagination_class = CustomPagination - queryset = Recipe.objects.all() - ordering_fields = ('date_updated', 'date_created') - - def filter_queryset(self, queryset): - queryset_filters = {'private': False} - queryset = Recipe.objects.filter(**queryset_filters) - return queryset - - -class Search(generics.GenericAPIView): - permission_classes = () - serializer_class = RecipeReadSerializer - pagination_class = CustomPagination - - def get(self, request, format=None): - search_string = request.query_params.get('q', None) - search_string_list = search_string.split(',') - resp = {} - - if search_string: - _results = Recipe.objects.filter(keywords__contains=search_string_list) - page = self.paginate_queryset(_results) - - if page is not None: - serializer = self.get_serializer(page, many=True) - result = self.get_paginated_response(serializer.data) - resp = result.data # pagination data - else: - serializer = self.get_serializer(queryset, many=True) - resp = serializer.data - - - #serialized = RecipeSerializer(_results, many=True) - #resp = serialized.data - - return Response(resp) diff --git a/src/backend/server/api/serializers.py b/src/backend/server/api/serializers.py deleted file mode 100644 index 505d68d..0000000 --- a/src/backend/server/api/serializers.py +++ /dev/null @@ -1,103 +0,0 @@ -import html - -from django.core.exceptions import PermissionDenied -from django.utils.translation import gettext as _ - -from allauth.socialaccount.models import SocialLogin -from rest_auth.registration.serializers import SocialLoginSerializer -from rest_framework import serializers -from rest_framework.exceptions import ValidationError - -from rest_framework.authentication import SessionAuthentication -from rest_framework import serializers - -from .models import Project, Recipe - - -from rest_framework.authtoken.models import Token - - -class TokenSerializer(serializers.ModelSerializer): - - class Meta: - model = Token - fields = ('key', 'user') - - -class RecipeSerializer(serializers.ModelSerializer): - - class Meta(object): - model = Recipe - fields = '__all__' - - -class RecipeReadSerializer(serializers.BaseSerializer): - def to_representation(self, instance): - return { - 'uuid': instance.uuid, - 'name': instance.name, - 'url_name': instance.url_name, - 'desc': html.unescape(instance.desc), - 'keywords': instance.keywords, - 'data': instance.data, - 'repo': instance.repo, - 'branch': instance.branch, - 'raw_url': instance.raw_url, - 'private': instance.private - } - - -class RecipeUpdateSerializer(serializers.ModelSerializer): - - class Meta(object): - model = Recipe - fields = [ - 'name', - 'url_name', - 'desc', - 'keywords', - 'data', - 'repo', - 'raw_url', - 'private' - ] - - -class ProjectSerializer(serializers.ModelSerializer): - - class Meta(object): - model = Project - fields = '__all__' - - -class ProjectUpdateSerializer(serializers.ModelSerializer): - - class Meta(object): - model = Project - fields = [ - 'name', - 'mutable', - 'private', - 'anonymous', - 'data' - ] - - -class CallbackSerializer(SocialLoginSerializer): - ''' - state = serializers.CharField() - - def validate_state(self, value): - """ - Checks that the state is equal to the one stored in the session. - """ - try: - SocialLogin.verify_and_unstash_state( - self.context['request'], - value, - ) - # Allauth raises PermissionDenied if the validation fails - except PermissionDenied: - raise ValidationError(_('State did not match.')) - return value - ''' \ No newline at end of file diff --git a/src/backend/server/api/urls.py b/src/backend/server/api/urls.py deleted file mode 100644 index 8bf0007..0000000 --- a/src/backend/server/api/urls.py +++ /dev/null @@ -1,98 +0,0 @@ -from django.urls import include, path, re_path -from django.views.generic import TemplateView - -from . import projectViews, projectViews, recipeViews, generatorViews - -from rest_auth.registration.views import ( - SocialAccountListView, SocialAccountDisconnectView -) - -from . import LoginViews as views - - -EMAIL_CONFIRMATION = r'^auth/confirm-email/(?P[-:\w]+)$' -PASSWORD_RESET = ( - r'^auth/password/reset/confirm/' - '(?P[0-9A-Za-z_\-]+)-' - '(?P[0-9A-Za-z]{1,13}-[0-9A-Za-z]{1,20})$' -) - -# NOTE: If you change this URL you have to update the callback URL -# in the OAuth providers' accounts, too -OAUTH_CALLBACK = 'auth/social/{provider}/callback' - - -# github URL patterns -github_urlpatterns = [ - path('auth-server/', views.Login.as_view(), name='github_auth_server'), - path( - 'login/', - views.CallbackCreate.as_view(), - name='github_callback_login', - ), - path( - 'connect/', - views.CallbackConnect.as_view(), - name='github_callback_connect', - ), -] - -# default URL patterns -api_urlpatterns = [ - path('project/private/', projectViews.ListPrivate.as_view()), - path('project/private//', projectViews.DetailPrivate.as_view()), - - path('project/', projectViews.CreateList.as_view()), - path('project//', projectViews.Detail.as_view()), - - path('recipe/', recipeViews.CreateList.as_view()), - path('recipe/name//', recipeViews.DetailByUrlName.as_view()), - path('recipe/popular/', recipeViews.Popular.as_view()), - path('recipe/search/', recipeViews.Search.as_view()), - path('recipe//', recipeViews.Detail.as_view()), - - path('generate/', generatorViews.Generate.as_view()), - path('import/', generatorViews.Import.as_view()), - path('repo/search/', generatorViews.RepoSearch.as_view()), - path('repo/detail/', generatorViews.RepoDetail.as_view()), - path('tags/', generatorViews.Tags.as_view()), - - # non social auth endpoints - path('auth/', include('rest_auth.urls')), - path('auth/registration/', include('rest_auth.registration.urls')), - path('auth/social/github/', include(github_urlpatterns)), - path( - 'auth/user/accounts/', - SocialAccountListView.as_view(), - name='social_account_list', - ), - path( - 'auth/user/accounts//disconnect/', - SocialAccountDisconnectView.as_view(), - name='social_account_disconnect', - ), -] - - -urlpatterns = [ - # The SPA serves these URLs but the backend has to know - # where they point to for reference, don't change the url names. - re_path( - EMAIL_CONFIRMATION, - TemplateView.as_view(), - name='account_confirm_email', - ), - re_path( - PASSWORD_RESET, - TemplateView.as_view(), - name='password_reset_confirm', - ), - path( - OAUTH_CALLBACK.format(provider='github'), - TemplateView.as_view(), - name='github_callback', - ), - # This has to be last because rest_auth.registration.urls - # also defines `account_confirm_email` what we override above. - path('api/', include(api_urlpatterns)), -] \ No newline at end of file diff --git a/src/backend/server/server/settings.py b/src/backend/server/server/settings.py deleted file mode 100644 index 1457222..0000000 --- a/src/backend/server/server/settings.py +++ /dev/null @@ -1,256 +0,0 @@ -""" -Django settings for server project. - -Generated by 'django-admin startproject' using Django 3.0.1. - -For more information on this file, see -https://docs.djangoproject.com/en/3.0/topics/settings/ - -For the full list of settings and their values, see -https://docs.djangoproject.com/en/3.0/ref/settings/ -""" - -import os -import datetime - -# Build paths inside the project like this: os.path.join(BASE_DIR, ...) -BASE_DIR = os.path.dirname(os.path.dirname(os.path.abspath(__file__))) - -SITE_ID = 1 - -# Quick-start development settings - unsuitable for production -# See https://docs.djangoproject.com/en/3.0/howto/deployment/checklist/ - -# SECURITY WARNING: keep the secret key used in production secret! -SECRET_KEY = '*otro=7@1!e2(-qx6*rd$%l5q01ki-0$-=4)%#6e=v(yf!8@d(' - -# SECURITY WARNING: don't run with debug turned on in production! -DEBUG = os.getenv( - 'DEBUG', - 'true').lower() == 'true' - -SITE_ID = 1 - -ALLOWED_HOSTS = [ - '*' -] - -CORS_ORIGIN_ALLOW_ALL = True - - -# Application definition - -INSTALLED_APPS = [ - 'corsheaders', - - 'django.contrib.sites', - 'django.contrib.admin', - 'django.contrib.auth', - 'django.contrib.contenttypes', - 'django.contrib.sessions', - 'django.contrib.messages', - 'django.contrib.staticfiles', - 'rest_framework', - 'rest_framework.authtoken', - - 'allauth', - 'allauth.account', - - 'rest_auth.registration', - 'rest_auth', - - 'allauth.socialaccount', - 'allauth.socialaccount.providers.github', - 'allauth.socialaccount.providers.google', - - 'storages', - - 'api', - 'utils', -] - -MIDDLEWARE = [ - 'corsheaders.middleware.CorsMiddleware', - - 'django.middleware.security.SecurityMiddleware', - 'django.contrib.sessions.middleware.SessionMiddleware', - 'django.middleware.common.CommonMiddleware', - 'django.middleware.csrf.CsrfViewMiddleware', - 'django.contrib.auth.middleware.AuthenticationMiddleware', - 'django.contrib.messages.middleware.MessageMiddleware', - 'django.middleware.clickjacking.XFrameOptionsMiddleware', -] - -ROOT_URLCONF = 'server.urls' - -TEMPLATES = [ - { - 'BACKEND': 'django.template.backends.django.DjangoTemplates', - 'DIRS': [], - 'APP_DIRS': True, - 'OPTIONS': { - 'context_processors': [ - 'django.template.context_processors.debug', - 'django.template.context_processors.request', - 'django.contrib.auth.context_processors.auth', - 'django.contrib.messages.context_processors.messages', - ], - }, - }, -] - -WSGI_APPLICATION = 'server.wsgi.application' - - -# Database -# https://docs.djangoproject.com/en/3.0/ref/settings/#databases - -DB_REMOTE = os.environ.get( - 'DB_REMOTE', - 'False' -).lower() == 'true' - -if DB_REMOTE: - DATABASES = { - 'default': { - 'ENGINE': 'django.db.backends.postgresql', - 'HOST': os.getenv('DB_HOST', None), - 'NAME': os.getenv('DB_NAME', None), - 'USER': os.getenv('DB_USER', None), - 'PASSWORD': os.getenv('DB_PASS', None), - 'PORT': 5432, - } - } -else: - DATABASES = { - 'default': { - 'ENGINE': 'django.db.backends.postgresql', - 'HOST': 'postgres', - 'NAME': 'postgres', - 'USER': 'postgres', - 'PASSWORD': 'postgres', - 'PORT': 5432, - } - } - - -# Password validation -# https://docs.djangoproject.com/en/3.0/ref/settings/#auth-password-validators - -AUTH_PASSWORD_VALIDATORS = [ - { - 'NAME': 'django.contrib.auth.password_validation.UserAttributeSimilarityValidator', - }, - { - 'NAME': 'django.contrib.auth.password_validation.MinimumLengthValidator', - }, - { - 'NAME': 'django.contrib.auth.password_validation.CommonPasswordValidator', - }, - { - 'NAME': 'django.contrib.auth.password_validation.NumericPasswordValidator', - }, -] - - -# Internationalization -# https://docs.djangoproject.com/en/3.0/topics/i18n/ - -LANGUAGE_CODE = 'en-us' - -TIME_ZONE = 'UTC' - -USE_I18N = True - -USE_L10N = True - -USE_TZ = True - - -DEFAULT_FILE_STORAGE = 'server.server..settings.storage_backends.PublicStorage' - - -# AWS - -AWS_STORAGE_BUCKET_NAME = os.getenv('AWS_STORAGE_BUCKET_NAME', None) -AWS_S3_FILE_OVERWRITE = True -AWS_S3_CUSTOM_DOMAIN = '%s.s3.amazonaws.com' % AWS_STORAGE_BUCKET_NAME -AWS_S3_OBJECT_PARAMETERS = { - 'CacheControl': 'max-age=86400', -} -AWS_DEFAULT_REGION = os.environ.get( - 'AWS_DEFAULT_REGION', - 'us-east-1' -) -AWS_LOCATION = 'static' - - -# Static files (CSS, JavaScript, Images) -# https://docs.djangoproject.com/en/2.0/howto/static-files/ - -STATICFILES_FINDERS = [ - 'django.contrib.staticfiles.finders.FileSystemFinder', - 'django.contrib.staticfiles.finders.AppDirectoriesFinder', -] - -STATICFILES_DIRS = [] - -if AWS_STORAGE_BUCKET_NAME: - STATIC_URL = 'https://%s/%s/' % (AWS_S3_CUSTOM_DOMAIN, AWS_LOCATION) - STATICFILES_STORAGE = 'storages.backends.s3boto3.S3Boto3Storage' -else: - STATIC_URL = '/static/' - STATIC_ROOT = '/static/' - - -# Rest Framework. - -REST_FRAMEWORK = { - 'DEFAULT_AUTHENTICATION_CLASSES': ( - 'rest_framework_jwt.authentication.JSONWebTokenAuthentication', - 'rest_framework.authentication.TokenAuthentication' - ), - 'DEFAULT_PERMISSION_CLASSES': (), - 'DEFAULT_FILTER_BACKENDS': ( - 'django_filters.rest_framework.DjangoFilterBackend', - ), - 'DEFAULT_RENDERER_CLASSES': ( - 'rest_framework.renderers.JSONRenderer', - ) -} - -AUTHENTICATION_BACKENDS = ( - "django.contrib.auth.backends.ModelBackend", - "allauth.account.auth_backends.AuthenticationBackend" -) - - -# JWT. -JWT_AUTH = { - 'JWT_RESPONSE_PAYLOAD_HANDLER': 'organization.utils.jwt_response_payload_handler', - - # How long the original token is valid for. - 'JWT_EXPIRATION_DELTA': datetime.timedelta(days=30), - - # Allow refreshing of tokens. - 'JWT_ALLOW_REFRESH': True, - - # This is the maximum time AFTER the token was issued that - # it can be refreshed. expired tokens can't be refreshed. - 'JWT_REFRESH_EXPIRATION_DELTA': datetime.timedelta(days=7), -} - - -# Auth. - -# this will try to send an email if an email is added to the user object -ACCOUNT_EMAIL_VERIFICATION = "none" - -ACCOUNT_USERNAME_REQUIRED = False -ACCOUNT_EMAIL_REQUIRED = True -ACCOUNT_AUTHENTICATION_METHOD = 'email' - -REST_USE_JWT = True - -SOCIAL_AUTH_CUSTOM_CALLBACK = os.getenv('SOCIAL_AUTH_CUSTOM_CALLBACK', None) -SOCIAL_AUTH_CUSTOM_CALLBACK_PAGE = os.getenv('SOCIAL_AUTH_CUSTOM_CALLBACK_PAGE', None) diff --git a/src/backend/server/server/urls.py b/src/backend/server/server/urls.py deleted file mode 100644 index fe97a39..0000000 --- a/src/backend/server/server/urls.py +++ /dev/null @@ -1,22 +0,0 @@ -"""server URL Configuration - -The `urlpatterns` list routes URLs to views. For more information please see: - https://docs.djangoproject.com/en/3.0/topics/http/urls/ -Examples: -Function views - 1. Add an import: from my_app import views - 2. Add a URL to urlpatterns: path('', views.home, name='home') -Class-based views - 1. Add an import: from other_app.views import Home - 2. Add a URL to urlpatterns: path('', Home.as_view(), name='home') -Including another URLconf - 1. Import the include() function: from django.urls import include, path - 2. Add a URL to urlpatterns: path('blog/', include('blog.urls')) -""" -from django.contrib import admin -from django.urls import include, path - -urlpatterns = [ - path('admin/', admin.site.urls), - path('', include('api.urls')), -] diff --git a/src/backend/server/utils/apps.py b/src/backend/server/utils/apps.py deleted file mode 100644 index 7527884..0000000 --- a/src/backend/server/utils/apps.py +++ /dev/null @@ -1,5 +0,0 @@ -from django.apps import AppConfig - - -class UtilsConfig(AppConfig): - name = 'utils' diff --git a/src/backend/server/utils/generators.py b/src/backend/server/utils/generators.py deleted file mode 100644 index eef75dc..0000000 --- a/src/backend/server/utils/generators.py +++ /dev/null @@ -1,458 +0,0 @@ -import yaml -import ast -from urllib.parse import unquote -#from anytree import Node, AnyNode, RenderTree - -from .utils import generate_uuid - - -class RevereseGenerator(): - def __init__(self, yaml_str, canvas_lookup_by_s_name=None, canvas_lookup_by_s_uuid=None): - self.yaml_str = yaml_str - self.canvas_lookup_by_s_name = canvas_lookup_by_s_name - self.canvas_lookup_by_s_uuid = canvas_lookup_by_s_uuid - self.service_uuid_lookup = {} - self.volume_uuid_lookup = {} - self.network_uuid_lookup = {} - self.initial_top = 130 - self.initial_left = 200 - self.services_counter = 0 - self.canvas_obj = { - "top": 512, - "left": 434, - "uuid": "", - "zoomLevel": 1 - } - self.volume_obj = { - "name": "", - "volume_name": "", - "uuid": "", - "driver": None, - "labels": [], - "external": False, - "driver_opts": None, - "external_name": None - } - self.network_obj = { - "name": "", - "object_name": "", - "uuid": "", - "driver": "local", - "labels": [], - "external": False, - "driver_opts": [], - "network_name": "", - "external_name": None - } - self.build_obj = { - "context": "", - "dockerfile": "", - "args": [], - "cache_from": [], - "labels": [], - "network": "", - "shm_size": "", - "target": "", - } - self.port_obj = { - "published": "80", - "target": "8080", - "mode": "host", - "protocol": "TCP" - } - self.resp = { - 'name': 'Imported project', - 'id': None, - 'date_created': None, - 'date_updated': None, - 'uuid': None, - 'data': { - 'canvas': [], - 'version': '3', - 'volumes': [], - 'networks': [], - 'services': [], - 'secrets': [], - 'configs': [] - }, - 'mutable': True - } - - def populate_service_lookup_table(self, services): - for service_k, _ in services.items(): - obj_uuid = generate_uuid() - self.service_uuid_lookup[service_k] = obj_uuid - - def parse_top_level_volumes(self, volumes): - """ - Parses top level volume objects only, excluding path mounts. - """ - if not volumes: - return - - for volume_k, volume_v in volumes.items(): - obj_uuid = generate_uuid() - volume_obj_cp = self.volume_obj.copy() - volume_obj_cp['uuid'] = obj_uuid - volume_obj_cp['name'] = volume_k - - try: - external = volume_v.get('external', False) - - if external: - external_name = external.get('name', None) - - volume_obj_cp['external'] = True - volume_obj_cp['external_name'] = external_name - except (AttributeError, KeyError): - pass - - try: - volume_obj_cp['volume_name'] = volume_v.get('name', None) - except (AttributeError, KeyError): - pass - - try: - volume_obj_cp['driver'] = volume_v.get('driver', None) - except (AttributeError, KeyError): - pass - - try: - volume_obj_cp['labels'] = [] - volume_labels = volume_v.get('labels', None) - for lbl_key, lbl_val in volume_labels.items(): - volume_obj_cp['labels'].append({ - 'key': lbl_key, - 'value': lbl_val - }) - except (AttributeError, KeyError): - pass - - self.volume_uuid_lookup[volume_k] = obj_uuid - self.resp['data']['volumes'].append(volume_obj_cp) - - def to_key_val_pairs(self, items): - ret = [] - - if isinstance(items, list): - for item in items: - item_parts = item.split('=') - ret.append({'key': item_parts[0], 'value': item_parts[1]}) - - if isinstance(items, dict): - try: - for _key, _val in items.items(): - ret.append({'key': _key, 'value': _val}) - except AttributeError: - pass - - return ret - - def parse_build_obj(self, build): - ret = self.build_obj.copy() - - if isinstance(build, str): - ret['build'] = build - else: - for _key, _val in build.items(): - if _key in ['args', 'cache_from', 'labels']: - if _val: - ret[_key] = self.to_key_val_pairs(_val) - else: - if _val: - ret[_key] = _val - - return ret - - def parse_top_level_networks(self, networks): - if not networks: - return - - for network_k, network_v in networks.items(): - obj_uuid = generate_uuid() - network_obj_cp = self.network_obj.copy() - network_obj_cp['name'] = network_k - network_obj_cp['uuid'] = obj_uuid - - try: - network_obj_cp['object_name'] = network_v.get('name', None) - network_obj_cp['driver'] = network_v.get('driver', None) - network_obj_cp['external'] = network_v.get('external', False) - except (AttributeError, KeyError): - pass - - try: - network_obj_cp['labels'] = self.to_key_val_pairs(network_v['labels']) - except Exception: - pass - - try: - network_obj_cp['driver_opts'] = self.to_key_val_pairs(network_v['driver_opts']) - except Exception: - pass - - self.network_uuid_lookup[network_k] = obj_uuid - self.resp['data']['networks'].append(network_obj_cp) - - def yaml_dict_to_system_obj(self): - yaml_str = unquote(self.yaml_str) - _dict = yaml.safe_load(yaml_str) - - if not _dict: - return self.resp - - version = _dict.get('version', '3') - services = _dict.get('services', None) - volumes = _dict.get('volumes', None) - networks = _dict.get('networks', None) - secrets = _dict.get('secrets', None) - configs = _dict.get('configs', None) - self.resp['data']['version'] = version - - # populate lookup tables, and parse top level objects - self.parse_top_level_volumes(volumes) - self.parse_top_level_networks(networks) - - if not services: - return self.resp - - self.populate_service_lookup_table(services) - - # construct services - for service_k, service_v in services.items(): - service_obj = {} - - # canvas object for each service object - try: - canvas_obj_cp = self.canvas_lookup_by_s_name[service_k] - except KeyError: - canvas_obj_cp = self.canvas_obj.copy() - canvas_obj_cp['uuid'] = self.service_uuid_lookup[service_k] - - # construct service obj - image = service_v.get('image', None) - if image: - image_parts = service_v['image'].split(':') - service_obj['image'] = image_parts[0] - - try: - service_obj['tag'] = image_parts[1] - except IndexError: - service_obj['tag'] = 'latest' - - service_obj['name'] = service_k - service_obj['uuid'] = self.service_uuid_lookup[service_k] - build = service_v.get('build', None) - - if build: - service_obj['build'] = self.parse_build_obj(build) - - try: - service_obj['command'] = service_v['command'] - except Exception: - pass - - try: - service_obj['working_dir'] = service_v['working_dir'] - except Exception: - pass - - try: - service_obj['environment'] = [] - envs = service_v['environment'] - if isinstance(envs, list): - for env_set in envs: - env_set_parts = env_set.split('=') - service_obj['environment'].append({ - 'key': env_set_parts[0], - 'value': env_set_parts[1] - }) - else: - for env_key, env_val in service_v['environment'].items(): - service_obj['environment'].append({ - 'key': env_key, - 'value': env_val - }) - except Exception: - pass - - try: - service_obj['labels'] = [] - service_labels = service_v.get('labels', None) - - if isinstance(service_labels, list): - for label_set in service_labels: - label_set_parts = label_set.split('=') - service_obj['labels'].append({ - 'key': label_set_parts[0], - 'value': label_set_parts[1] - }) - - if isinstance(service_labels, dict): - for lbl_key, lbl_val in service_labels.items(): - service_obj['labels'].append({ - 'key': lbl_key, - 'value': lbl_val - }) - except Exception as e: - print(e) - - try: - service_obj['ports'] = [] - for port in service_v['ports']: - if isinstance(port, str): - port_parts = port.split(':') - port_obj_cp = self.port_obj.copy() - port_obj_cp['published'] = port_parts[0] - port_obj_cp['protocol'] = 'tcp' - port_obj_cp['mode'] = 'host' - - try: - port_obj_cp['target'] = port_parts[1] - except IndexError: - port_obj_cp['target'] = None - - service_obj['ports'].append(port_obj_cp) - - if isinstance(port, dict): - service_obj['ports'].append(port) - - except KeyError: - pass - - try: - service_obj['volumes'] = [] - for volume in service_v['volumes']: - volume_obj_uuid = generate_uuid() - volume_obj_cp = self.volume_obj.copy() - volume_obj_cp['uuid'] = volume_obj_uuid - volume_obj_cp['name'] = volume_obj_uuid - - volume_parts = volume.split(':') - - if len(volume_parts) > 1: - try: - volume_id = self.volume_uuid_lookup[volume_parts[0]] - - if volume_id: - service_obj['volumes'].append({ - "volume": volume_id, - "destination": volume_parts[1] - }) - except KeyError: - service_obj['volumes'].append({ - "relativePathSource": volume_parts[0], - "destination": volume_parts[1] - }) - else: - service_obj['volumes'].append({ - "relativePathSource": volume_parts[0], - "destination": volume_parts[0] - }) - - except KeyError: - pass - - try: - service_obj['networks'] = [] - for network in service_v['networks']: - service_obj['networks'].append(self.network_uuid_lookup[network]) - except KeyError: - pass - - try: - service_obj['depends_on'] = [] - for dep in service_v['depends_on']: - service_obj['depends_on'].append(self.service_uuid_lookup[dep]) - except KeyError: - pass - - if float(version) < 2: - try: - service_obj['links'] = [] - for dep in service_v['links']: - service_obj['links'].append(self.service_uuid_lookup[dep]) - except KeyError: - pass - - try: - service_obj['container_name'] = service_v['container_name'] - except KeyError: - pass - - try: - service_obj['restart'] = service_v['restart'] - except KeyError: - pass - - try: - deploy = service_v['deploy'] - service_obj['deploy'] = deploy - - ''' - try: - placement_preferences = deploy['placement']['preferences'] - service_obj['deploy']['placement']['preferences'] = self.to_key_val_pairs(placement_preferences) - except Exception: - pass - ''' - - try: - labels = deploy['labels'] - service_obj['deploy']['labels'] = self.to_key_val_pairs(labels) - except Exception: - pass - except KeyError: - pass - - # append to object lists - self.resp['data']['services'].append(service_obj) - self.resp['data']['canvas'].append(canvas_obj_cp) - - levels = {} - - for parsed_service in self.resp['data']['services']: - dep_count = len(parsed_service['depends_on']) - - try: - levels[dep_count].append({ - 'uuid': parsed_service['uuid'], - 'name': parsed_service['name'] - }) - except KeyError: - levels[dep_count] = [] - levels[dep_count].append({ - 'uuid': parsed_service['uuid'], - 'name': parsed_service['name'] - }) - - canvas = [] - initial_top = 20 - initial_left = 20 - iter_node_count = 0 - - for _, l_values in levels.items(): - for l_value in l_values: - if not iter_node_count % 5: - initial_left = 20 - initial_top += 200 - - initial_left += 200 - - try: - canvas_obj_cp = self.canvas_lookup_by_s_name[l_value['name']] - canvas_obj_cp['uuid'] = l_value['uuid'] - canvas.append(canvas_obj_cp) - except KeyError: - canvas_obj_cp = self.canvas_obj.copy() - canvas_obj_cp['top'] = initial_top - canvas_obj_cp['left'] = initial_left - canvas_obj_cp['uuid'] = l_value['uuid'] - canvas_obj_cp['zoomLevel'] = 0.7 - canvas.append(canvas_obj_cp) - - iter_node_count += 1 - - self.resp['data']['canvas'] = list(reversed(canvas)) - - return self.resp diff --git a/src/backend/server/utils/mixins.py b/src/backend/server/utils/mixins.py deleted file mode 100644 index f6f0883..0000000 --- a/src/backend/server/utils/mixins.py +++ /dev/null @@ -1,9 +0,0 @@ -from django.db import models - - -class TimeStamp(models.Model): - date_created = models.DateTimeField(auto_now_add=True) - date_updated = models.DateTimeField(auto_now=True) - - class Meta: - abstract = True diff --git a/src/backend/server/utils/pagination.py b/src/backend/server/utils/pagination.py deleted file mode 100644 index d5a25c3..0000000 --- a/src/backend/server/utils/pagination.py +++ /dev/null @@ -1,18 +0,0 @@ -from rest_framework.pagination import PageNumberPagination -from rest_framework.response import Response - - -class CustomPagination(PageNumberPagination): - page_size_query_param = 'page_size' - page_size = 30 - - def get_paginated_response(self, data): - return Response({ - 'links': { - 'next': self.get_next_link(), - 'previous': self.get_previous_link() - }, - 'total': self.page.paginator.count, - 'current_total': len(data), - 'results': data - }) \ No newline at end of file diff --git a/src/composer/.editorconfig b/src/composer/.editorconfig deleted file mode 100644 index e89330a..0000000 --- a/src/composer/.editorconfig +++ /dev/null @@ -1,13 +0,0 @@ -# Editor configuration, see https://editorconfig.org -root = true - -[*] -charset = utf-8 -indent_style = space -indent_size = 2 -insert_final_newline = true -trim_trailing_whitespace = true - -[*.md] -max_line_length = off -trim_trailing_whitespace = false diff --git a/src/composer/README.md b/src/composer/README.md deleted file mode 100644 index a09f71b..0000000 --- a/src/composer/README.md +++ /dev/null @@ -1,27 +0,0 @@ -# Composer - -This project was generated with [Angular CLI](https://github.com/angular/angular-cli) version 9.0.6. - -## Development server - -Run `ng serve` for a dev server. Navigate to `http://localhost:4200/`. The app will automatically reload if you change any of the source files. - -## Code scaffolding - -Run `ng generate component component-name` to generate a new component. You can also use `ng generate directive|pipe|service|class|guard|interface|enum|module`. - -## Build - -Run `ng build` to build the project. The build artifacts will be stored in the `dist/` directory. Use the `--configuration=production` flag for a production build. - -## Running unit tests - -Run `ng test` to execute the unit tests via [Karma](https://karma-runner.github.io). - -## Running end-to-end tests - -Run `ng e2e` to execute the end-to-end tests via [Protractor](http://www.protractortest.org/). - -## Further help - -To get more help on the Angular CLI use `ng help` or go check out the [Angular CLI README](https://github.com/angular/angular-cli/blob/master/README.md). diff --git a/src/composer/angular.json b/src/composer/angular.json deleted file mode 100644 index 90e5683..0000000 --- a/src/composer/angular.json +++ /dev/null @@ -1,137 +0,0 @@ -{ - "$schema": "./node_modules/@angular/cli/lib/config/schema.json", - "version": 1, - "newProjectRoot": "projects", - "projects": { - "frontend": { - "projectType": "application", - "schematics": { - "@schematics/angular:component": { - "style": "scss" - } - }, - "root": "", - "sourceRoot": "src", - "prefix": "app", - "architect": { - "build": { - "builder": "@angular-devkit/build-angular:browser", - "options": { - "outputPath": "dist/frontend", - "index": "src/index.html", - "main": "src/main.ts", - "polyfills": "src/polyfills.ts", - "tsConfig": "tsconfig.app.json", - "aot": true, - "assets": ["src/favicon.ico", "src/assets"], - "styles": [ - "./node_modules/@angular/material/prebuilt-themes/indigo-pink.css", - "src/styles.scss" - ], - "scripts": [ - "./node_modules/jsplumb/dist/js/jsplumb.min.js" - ] - }, - "configurations": { - "production": { - "fileReplacements": [ - { - "replace": "src/environments/environment.ts", - "with": "src/environments/environment.prod.ts" - } - ], - "optimization": true, - "outputHashing": "all", - "sourceMap": false, - "extractCss": true, - "namedChunks": false, - "extractLicenses": true, - "vendorChunk": false, - "buildOptimizer": true, - "budgets": [ - { - "type": "initial", - "maximumWarning": "2mb", - "maximumError": "5mb" - }, - { - "type": "anyComponentStyle", - "maximumWarning": "6kb", - "maximumError": "10kb" - } - ] - }, - "staging": { - "fileReplacements": [ - { - "replace": "src/environments/environment.ts", - "with": "src/environments/environment.staging.ts" - } - ] - }, - "local": { - "fileReplacements": [ - { - "replace": "src/environments/environment.ts", - "with": "src/environments/environment.local.ts" - } - ] - } - } - }, - "serve": { - "builder": "@angular-devkit/build-angular:dev-server", - "options": { - "browserTarget": "frontend:build" - }, - "configurations": { - "production": { - "browserTarget": "frontend:build:production" - }, - "staging": { - "browserTarget": "frontend:build:staging" - } - } - }, - "extract-i18n": { - "builder": "@angular-devkit/build-angular:extract-i18n", - "options": { - "browserTarget": "frontend:build" - } - }, - "test": { - "builder": "@angular-devkit/build-angular:karma", - "options": { - "main": "src/test.ts", - "polyfills": "src/polyfills.ts", - "tsConfig": "tsconfig.spec.json", - "karmaConfig": "karma.conf.js", - "assets": ["src/favicon.ico", "src/assets"], - "styles": ["./node_modules/@angular/material/prebuilt-themes/indigo-pink.css", "src/styles.scss"], - "scripts": [] - } - }, - "lint": { - "builder": "@angular-devkit/build-angular:tslint", - "options": { - "tsConfig": ["tsconfig.app.json", "tsconfig.spec.json", "e2e/tsconfig.json"], - "exclude": ["**/node_modules/**"] - } - }, - "e2e": { - "builder": "@angular-devkit/build-angular:protractor", - "options": { - "protractorConfig": "e2e/protractor.conf.js", - "devServerTarget": "frontend:serve" - }, - "configurations": { - "production": { - "devServerTarget": "frontend:serve:production" - } - } - } - } - } - }, - "defaultProject": "frontend" -} diff --git a/src/composer/browserslist b/src/composer/browserslist deleted file mode 100644 index 8084853..0000000 --- a/src/composer/browserslist +++ /dev/null @@ -1,12 +0,0 @@ -# This file is used by the build system to adjust CSS and JS output to support the specified browsers below. -# For additional information regarding the format and rule options, please see: -# https://github.com/browserslist/browserslist#queries - -# You can see what browsers were selected by your queries by running: -# npx browserslist - -> 0.5% -last 2 versions -Firefox ESR -not dead -not IE 9-11 # For IE 9-11 support, remove 'not'. \ No newline at end of file diff --git a/src/composer/e2e/protractor.conf.js b/src/composer/e2e/protractor.conf.js deleted file mode 100644 index 7c798cf..0000000 --- a/src/composer/e2e/protractor.conf.js +++ /dev/null @@ -1,32 +0,0 @@ -// @ts-check -// Protractor configuration file, see link for more information -// https://github.com/angular/protractor/blob/master/lib/config.ts - -const { SpecReporter } = require('jasmine-spec-reporter'); - -/** - * @type { import("protractor").Config } - */ -exports.config = { - allScriptsTimeout: 11000, - specs: [ - './src/**/*.e2e-spec.ts' - ], - capabilities: { - browserName: 'chrome' - }, - directConnect: true, - baseUrl: 'http://localhost:4200/', - framework: 'jasmine', - jasmineNodeOpts: { - showColors: true, - defaultTimeoutInterval: 30000, - print: function() {} - }, - onPrepare() { - require('ts-node').register({ - project: require('path').join(__dirname, './tsconfig.json') - }); - jasmine.getEnv().addReporter(new SpecReporter({ spec: { displayStacktrace: true } })); - } -}; \ No newline at end of file diff --git a/src/composer/e2e/src/app.e2e-spec.ts b/src/composer/e2e/src/app.e2e-spec.ts deleted file mode 100644 index 95bda14..0000000 --- a/src/composer/e2e/src/app.e2e-spec.ts +++ /dev/null @@ -1,23 +0,0 @@ -import { AppPage } from './app.po'; -import { browser, logging } from 'protractor'; - -describe('workspace-project App', () => { - let page: AppPage; - - beforeEach(() => { - page = new AppPage(); - }); - - it('should display welcome message', () => { - page.navigateTo(); - expect(page.getTitleText()).toEqual('composer app is running!'); - }); - - afterEach(async () => { - // Assert that there are no errors emitted from the browser - const logs = await browser.manage().logs().get(logging.Type.BROWSER); - expect(logs).not.toContain(jasmine.objectContaining({ - level: logging.Level.SEVERE, - } as logging.Entry)); - }); -}); diff --git a/src/composer/e2e/src/app.po.ts b/src/composer/e2e/src/app.po.ts deleted file mode 100644 index b68475e..0000000 --- a/src/composer/e2e/src/app.po.ts +++ /dev/null @@ -1,11 +0,0 @@ -import { browser, by, element } from 'protractor'; - -export class AppPage { - navigateTo(): Promise { - return browser.get(browser.baseUrl) as Promise; - } - - getTitleText(): Promise { - return element(by.css('app-root .content span')).getText() as Promise; - } -} diff --git a/src/composer/e2e/tsconfig.json b/src/composer/e2e/tsconfig.json deleted file mode 100644 index 39b800f..0000000 --- a/src/composer/e2e/tsconfig.json +++ /dev/null @@ -1,13 +0,0 @@ -{ - "extends": "../tsconfig.json", - "compilerOptions": { - "outDir": "../out-tsc/e2e", - "module": "commonjs", - "target": "es5", - "types": [ - "jasmine", - "jasminewd2", - "node" - ] - } -} diff --git a/src/composer/karma.conf.js b/src/composer/karma.conf.js deleted file mode 100644 index d26663c..0000000 --- a/src/composer/karma.conf.js +++ /dev/null @@ -1,32 +0,0 @@ -// Karma configuration file, see link for more information -// https://karma-runner.github.io/1.0/config/configuration-file.html - -module.exports = function (config) { - config.set({ - basePath: '', - frameworks: ['jasmine', '@angular-devkit/build-angular'], - plugins: [ - require('karma-jasmine'), - require('karma-chrome-launcher'), - require('karma-jasmine-html-reporter'), - require('karma-coverage-istanbul-reporter'), - require('@angular-devkit/build-angular/plugins/karma') - ], - client: { - clearContext: false // leave Jasmine Spec Runner output visible in browser - }, - coverageIstanbulReporter: { - dir: require('path').join(__dirname, './coverage/frontend'), - reports: ['html', 'lcovonly', 'text-summary'], - fixWebpackSourcePaths: true - }, - reporters: ['progress', 'kjhtml'], - port: 9876, - colors: true, - logLevel: config.LOG_INFO, - autoWatch: true, - browsers: ['Chrome'], - singleRun: false, - restartOnFileChange: true - }); -}; diff --git a/src/composer/package-lock.json b/src/composer/package-lock.json deleted file mode 100644 index 71fc48d..0000000 --- a/src/composer/package-lock.json +++ /dev/null @@ -1,13280 +0,0 @@ -{ - "name": "composer", - "version": "1.0.0", - "lockfileVersion": 1, - "requires": true, - "dependencies": { - "@angular-devkit/architect": { - "version": "0.901.4", - "resolved": "https://registry.npmjs.org/@angular-devkit/architect/-/architect-0.901.4.tgz", - "integrity": "sha512-w4RMj7eLhUSh70HUy5tW4EXjLQFXk0Lfr9WiSy5gvPGp+zzYxknI+Wn4Xid1wU/WS+4tuMv5nJIaNaH2sABESQ==", - "dev": true, - "requires": { - "@angular-devkit/core": "9.1.4", - "rxjs": "6.5.4" - }, - "dependencies": { - "rxjs": { - "version": "6.5.4", - "resolved": "https://registry.npmjs.org/rxjs/-/rxjs-6.5.4.tgz", - "integrity": "sha512-naMQXcgEo3csAEGvw/NydRA0fuS2nDZJiw1YUWFKU7aPPAPGZEsD4Iimit96qwCieH6y614MCLYwdkrWx7z/7Q==", - "dev": true, - "requires": { - "tslib": "^1.9.0" - } - } - } - }, - "@angular-devkit/build-angular": { - "version": "0.901.4", - "resolved": "https://registry.npmjs.org/@angular-devkit/build-angular/-/build-angular-0.901.4.tgz", - "integrity": "sha512-Vod4bc6d38fuvqauIDQxKMe9hKW9m4QNzPKxEc+Dv5Xkv95WaGzkbUVu8M8t4E//sDDMpmcPdYOXSdR27WBi2Q==", - "dev": true, - "requires": { - "@angular-devkit/architect": "0.901.4", - "@angular-devkit/build-optimizer": "0.901.4", - "@angular-devkit/build-webpack": "0.901.4", - "@angular-devkit/core": "9.1.4", - "@babel/core": "7.9.0", - "@babel/generator": "7.9.3", - "@babel/preset-env": "7.9.0", - "@babel/template": "7.8.6", - "@jsdevtools/coverage-istanbul-loader": "3.0.3", - "@ngtools/webpack": "9.1.4", - "ajv": "6.12.0", - "autoprefixer": "9.7.4", - "babel-loader": "8.0.6", - "browserslist": "^4.9.1", - "cacache": "15.0.0", - "caniuse-lite": "^1.0.30001032", - "circular-dependency-plugin": "5.2.0", - "copy-webpack-plugin": "5.1.1", - "core-js": "3.6.4", - "css-loader": "3.5.1", - "cssnano": "4.1.10", - "file-loader": "6.0.0", - "find-cache-dir": "3.3.1", - "glob": "7.1.6", - "jest-worker": "25.1.0", - "karma-source-map-support": "1.4.0", - "less": "3.11.1", - "less-loader": "5.0.0", - "license-webpack-plugin": "2.1.4", - "loader-utils": "2.0.0", - "mini-css-extract-plugin": "0.9.0", - "minimatch": "3.0.4", - "open": "7.0.3", - "parse5": "4.0.0", - "postcss": "7.0.27", - "postcss-import": "12.0.1", - "postcss-loader": "3.0.0", - "raw-loader": "4.0.0", - "regenerator-runtime": "0.13.5", - "rimraf": "3.0.2", - "rollup": "2.1.0", - "rxjs": "6.5.4", - "sass": "1.26.3", - "sass-loader": "8.0.2", - "semver": "7.1.3", - "source-map": "0.7.3", - "source-map-loader": "0.2.4", - "speed-measure-webpack-plugin": "1.3.1", - "style-loader": "1.1.3", - "stylus": "0.54.7", - "stylus-loader": "3.0.2", - "terser": "4.6.10", - "terser-webpack-plugin": "2.3.5", - "tree-kill": "1.2.2", - "webpack": "4.42.0", - "webpack-dev-middleware": "3.7.2", - "webpack-dev-server": "3.10.3", - "webpack-merge": "4.2.2", - "webpack-sources": "1.4.3", - "webpack-subresource-integrity": "1.4.0", - "worker-plugin": "4.0.2" - }, - "dependencies": { - "@babel/generator": { - "version": "7.9.3", - "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.9.3.tgz", - "integrity": "sha512-RpxM252EYsz9qLUIq6F7YJyK1sv0wWDBFuztfDGWaQKzHjqDHysxSiRUpA/X9jmfqo+WzkAVKFaUily5h+gDCQ==", - "dev": true, - "requires": { - "@babel/types": "^7.9.0", - "jsesc": "^2.5.1", - "lodash": "^4.17.13", - "source-map": "^0.5.0" - }, - "dependencies": { - "source-map": { - "version": "0.5.7", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.5.7.tgz", - "integrity": "sha1-igOdLRAh0i0eoUyA2OpGi6LvP8w=", - "dev": true - } - } - }, - "ajv": { - "version": "6.12.0", - "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.0.tgz", - "integrity": "sha512-D6gFiFA0RRLyUbvijN74DWAjXSFxWKaWP7mldxkVhyhAV3+SWA9HEJPHQ2c9soIeTFJqcSdFDGFgdqs1iUU2Hw==", - "dev": true, - "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" - } - }, - "fast-deep-equal": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.1.tgz", - "integrity": "sha512-8UEa58QDLauDNfpbrX55Q9jrGHThw2ZMdOky5Gl1CDtVeJDPVrG4Jxx1N8jw2gkWaff5UUuX1KJd+9zGe2B+ZA==", - "dev": true - }, - "glob": { - "version": "7.1.6", - "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.6.tgz", - "integrity": "sha512-LwaxwyZ72Lk7vZINtNNrywX0ZuLyStrdDtabefZKAY5ZGJhVtgdznluResxNmPitE0SAO+O26sWTHeKSI2wMBA==", - "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" - } - }, - "rxjs": { - "version": "6.5.4", - "resolved": "https://registry.npmjs.org/rxjs/-/rxjs-6.5.4.tgz", - "integrity": "sha512-naMQXcgEo3csAEGvw/NydRA0fuS2nDZJiw1YUWFKU7aPPAPGZEsD4Iimit96qwCieH6y614MCLYwdkrWx7z/7Q==", - "dev": true, - "requires": { - "tslib": "^1.9.0" - } - }, - "semver": { - "version": "7.1.3", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.1.3.tgz", - "integrity": "sha512-ekM0zfiA9SCBlsKa2X1hxyxiI4L3B6EbVJkkdgQXnSEEaHlGdvyodMruTiulSRWMMB4NeIuYNMC9rTKTz97GxA==", - "dev": true - } - } - }, - "@angular-devkit/build-optimizer": { - "version": "0.901.4", - "resolved": "https://registry.npmjs.org/@angular-devkit/build-optimizer/-/build-optimizer-0.901.4.tgz", - "integrity": "sha512-iDMlNRwd8ICMBKnycfw55hdnL5kCiqUjQn+aK/4uOMJMz49tiYMNJAaznXX2BFKmYSmbapKjEbzx9yMYRi9Y7w==", - "dev": true, - "requires": { - "loader-utils": "2.0.0", - "source-map": "0.7.3", - "tslib": "1.11.1", - "typescript": "3.6.5", - "webpack-sources": "1.4.3" - }, - "dependencies": { - "typescript": { - "version": "3.6.5", - "resolved": "https://registry.npmjs.org/typescript/-/typescript-3.6.5.tgz", - "integrity": "sha512-BEjlc0Z06ORZKbtcxGrIvvwYs5hAnuo6TKdNFL55frVDlB+na3z5bsLhFaIxmT+dPWgBIjMo6aNnTOgHHmHgiQ==", - "dev": true - } - } - }, - "@angular-devkit/build-webpack": { - "version": "0.901.4", - "resolved": "https://registry.npmjs.org/@angular-devkit/build-webpack/-/build-webpack-0.901.4.tgz", - "integrity": "sha512-YBOufI4TGAtIDCS6erFygYJX/fkd8xwI58d+7iFIBmtIJC4/fpGjX6qkHMCBgh8HXAkEPSORBCYQn9O44J1ZXQ==", - "dev": true, - "requires": { - "@angular-devkit/architect": "0.901.4", - "@angular-devkit/core": "9.1.4", - "rxjs": "6.5.4" - }, - "dependencies": { - "rxjs": { - "version": "6.5.4", - "resolved": "https://registry.npmjs.org/rxjs/-/rxjs-6.5.4.tgz", - "integrity": "sha512-naMQXcgEo3csAEGvw/NydRA0fuS2nDZJiw1YUWFKU7aPPAPGZEsD4Iimit96qwCieH6y614MCLYwdkrWx7z/7Q==", - "dev": true, - "requires": { - "tslib": "^1.9.0" - } - } - } - }, - "@angular-devkit/core": { - "version": "9.1.4", - "resolved": "https://registry.npmjs.org/@angular-devkit/core/-/core-9.1.4.tgz", - "integrity": "sha512-OPFQDmT4XabLMSRDgmnzedlOrc83DzQIgLcfoh/UhZ7aJKf/2Vq4l09p/DkMNI36vN5BRL0zDZt7TjvKNgyYgA==", - "dev": true, - "requires": { - "ajv": "6.12.0", - "fast-json-stable-stringify": "2.1.0", - "magic-string": "0.25.7", - "rxjs": "6.5.4", - "source-map": "0.7.3" - }, - "dependencies": { - "ajv": { - "version": "6.12.0", - "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.0.tgz", - "integrity": "sha512-D6gFiFA0RRLyUbvijN74DWAjXSFxWKaWP7mldxkVhyhAV3+SWA9HEJPHQ2c9soIeTFJqcSdFDGFgdqs1iUU2Hw==", - "dev": true, - "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" - } - }, - "fast-deep-equal": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.1.tgz", - "integrity": "sha512-8UEa58QDLauDNfpbrX55Q9jrGHThw2ZMdOky5Gl1CDtVeJDPVrG4Jxx1N8jw2gkWaff5UUuX1KJd+9zGe2B+ZA==", - "dev": true - }, - "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==", - "dev": true - }, - "magic-string": { - "version": "0.25.7", - "resolved": "https://registry.npmjs.org/magic-string/-/magic-string-0.25.7.tgz", - "integrity": "sha512-4CrMT5DOHTDk4HYDlzmwu4FVCcIYI8gauveasrdCu2IKIFOJ3f0v/8MDGJCDL9oD2ppz/Av1b0Nj345H9M+XIA==", - "dev": true, - "requires": { - "sourcemap-codec": "^1.4.4" - } - }, - "rxjs": { - "version": "6.5.4", - "resolved": "https://registry.npmjs.org/rxjs/-/rxjs-6.5.4.tgz", - "integrity": "sha512-naMQXcgEo3csAEGvw/NydRA0fuS2nDZJiw1YUWFKU7aPPAPGZEsD4Iimit96qwCieH6y614MCLYwdkrWx7z/7Q==", - "dev": true, - "requires": { - "tslib": "^1.9.0" - } - } - } - }, - "@angular-devkit/schematics": { - "version": "9.1.4", - "resolved": "https://registry.npmjs.org/@angular-devkit/schematics/-/schematics-9.1.4.tgz", - "integrity": "sha512-RAbdnUEZ3JTLmWSBiXT5trsVx8Fi72fxN9CiRaluM09Cytg6BUc1wC5XCO0YPvhI400+3Ro1nwjPXezjg7LXzQ==", - "dev": true, - "requires": { - "@angular-devkit/core": "9.1.4", - "ora": "4.0.3", - "rxjs": "6.5.4" - }, - "dependencies": { - "rxjs": { - "version": "6.5.4", - "resolved": "https://registry.npmjs.org/rxjs/-/rxjs-6.5.4.tgz", - "integrity": "sha512-naMQXcgEo3csAEGvw/NydRA0fuS2nDZJiw1YUWFKU7aPPAPGZEsD4Iimit96qwCieH6y614MCLYwdkrWx7z/7Q==", - "dev": true, - "requires": { - "tslib": "^1.9.0" - } - } - } - }, - "@angular/animations": { - "version": "9.1.4", - "resolved": "https://registry.npmjs.org/@angular/animations/-/animations-9.1.4.tgz", - "integrity": "sha512-gMo/DbCm5BDArladMAeC7/75T2DvhLr4CSUGJt/P/aimTEG2ywoAALs3pzwSSe4qxrHiR0OIksVW3l4km3iXEw==" - }, - "@angular/cdk": { - "version": "9.2.2", - "resolved": "https://registry.npmjs.org/@angular/cdk/-/cdk-9.2.2.tgz", - "integrity": "sha512-VNd+KuMN6cBcy4/8OyMxqYaxdjPP6IyCqIVijB2JREkc5Sg4VWmPgx2L3rHt/DzjsVBVRgx35uqOMymDezG3jQ==", - "requires": { - "parse5": "^5.0.0" - }, - "dependencies": { - "parse5": { - "version": "5.1.1", - "resolved": "https://registry.npmjs.org/parse5/-/parse5-5.1.1.tgz", - "integrity": "sha512-ugq4DFI0Ptb+WWjAdOK16+u/nHfiIrcE+sh8kZMaM0WllQKLI9rOUq6c2b7cwPkXdzfQESqvoqK6ug7U/Yyzug==", - "optional": true - } - } - }, - "@angular/cli": { - "version": "9.1.4", - "resolved": "https://registry.npmjs.org/@angular/cli/-/cli-9.1.4.tgz", - "integrity": "sha512-H9MqoT4zyIv+Yo3cvRVkzafWGHsqt7jUvtvGwMHIDMTfEX+Q8yiYlDLL6WM3Eb6/hDmLcRGC/GI495sKS1z5qA==", - "dev": true, - "requires": { - "@angular-devkit/architect": "0.901.4", - "@angular-devkit/core": "9.1.4", - "@angular-devkit/schematics": "9.1.4", - "@schematics/angular": "9.1.4", - "@schematics/update": "0.901.4", - "@yarnpkg/lockfile": "1.1.0", - "ansi-colors": "4.1.1", - "debug": "4.1.1", - "ini": "1.3.5", - "inquirer": "7.1.0", - "npm-package-arg": "8.0.1", - "npm-pick-manifest": "6.0.0", - "open": "7.0.3", - "pacote": "9.5.12", - "read-package-tree": "5.3.1", - "rimraf": "3.0.2", - "semver": "7.1.3", - "symbol-observable": "1.2.0", - "universal-analytics": "0.4.20", - "uuid": "7.0.2" - }, - "dependencies": { - "ansi-colors": { - "version": "4.1.1", - "resolved": "https://registry.npmjs.org/ansi-colors/-/ansi-colors-4.1.1.tgz", - "integrity": "sha512-JoX0apGbHaUJBNl6yF+p6JAFYZ666/hhCGKN5t9QFjbJQKUU/g8MNbFDbvfrgKXvI1QpZplPOnwIo99lX/AAmA==", - "dev": true - }, - "semver": { - "version": "7.1.3", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.1.3.tgz", - "integrity": "sha512-ekM0zfiA9SCBlsKa2X1hxyxiI4L3B6EbVJkkdgQXnSEEaHlGdvyodMruTiulSRWMMB4NeIuYNMC9rTKTz97GxA==", - "dev": true - }, - "uuid": { - "version": "7.0.2", - "resolved": "https://registry.npmjs.org/uuid/-/uuid-7.0.2.tgz", - "integrity": "sha512-vy9V/+pKG+5ZTYKf+VcphF5Oc6EFiu3W8Nv3P3zIh0EqVI80ZxOzuPfe9EHjkFNvf8+xuTHVeei4Drydlx4zjw==", - "dev": true - } - } - }, - "@angular/common": { - "version": "9.1.4", - "resolved": "https://registry.npmjs.org/@angular/common/-/common-9.1.4.tgz", - "integrity": "sha512-JvCoCWVbx0tF7l/0WTi24ui/mc2SElyVSNchR4VK/FViARnkvnSBdI/Ef5QWXrsPyKU4PYBtnWWgyxRspH+FBA==" - }, - "@angular/compiler": { - "version": "9.1.4", - "resolved": "https://registry.npmjs.org/@angular/compiler/-/compiler-9.1.4.tgz", - "integrity": "sha512-B+f3lviFNEJtL9V9exSKYPSz2Ddb6dxgPzQR7GSjGikDo+fKMtC1PjNwgJooS9gavhQx30uwkEEMIPYQbM6nNA==" - }, - "@angular/compiler-cli": { - "version": "9.1.4", - "resolved": "https://registry.npmjs.org/@angular/compiler-cli/-/compiler-cli-9.1.4.tgz", - "integrity": "sha512-n3PzqNnPD7s/AF9mv5CnarK0sgfoq4txFncHjJWBSltuTQoz6BDZyjuEdqsSLUvgAZPeLsmohemOzEE38HYHZA==", - "dev": true, - "requires": { - "canonical-path": "1.0.0", - "chokidar": "^3.0.0", - "convert-source-map": "^1.5.1", - "dependency-graph": "^0.7.2", - "fs-extra": "4.0.2", - "magic-string": "^0.25.0", - "minimist": "^1.2.0", - "reflect-metadata": "^0.1.2", - "semver": "^6.3.0", - "source-map": "^0.6.1", - "sourcemap-codec": "^1.4.8", - "yargs": "15.3.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.2.1", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.2.1.tgz", - "integrity": "sha512-9VGjrMsG1vePxcSweQsN20KY/c4zN0h9fLjqAbwbPfahM3t+NL+M9HC8xeXG2I8pX5NoamTGNuomEUFI7fcUjA==", - "dev": true, - "requires": { - "@types/color-name": "^1.1.1", - "color-convert": "^2.0.1" - } - }, - "cliui": { - "version": "6.0.0", - "resolved": "https://registry.npmjs.org/cliui/-/cliui-6.0.0.tgz", - "integrity": "sha512-t6wbgtoCXvAzst7QgXxJYqPt0usEfbgQdftEPbLL/cvv6HPE5VgvqCuAIDR0NgU52ds6rFwqrgakNLrHEjCbrQ==", - "dev": true, - "requires": { - "string-width": "^4.2.0", - "strip-ansi": "^6.0.0", - "wrap-ansi": "^6.2.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 - }, - "find-up": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/find-up/-/find-up-4.1.0.tgz", - "integrity": "sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==", - "dev": true, - "requires": { - "locate-path": "^5.0.0", - "path-exists": "^4.0.0" - } - }, - "get-caller-file": { - "version": "2.0.5", - "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-2.0.5.tgz", - "integrity": "sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==", - "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 - }, - "locate-path": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-5.0.0.tgz", - "integrity": "sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==", - "dev": true, - "requires": { - "p-locate": "^4.1.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": "4.1.0", - "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-4.1.0.tgz", - "integrity": "sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A==", - "dev": true, - "requires": { - "p-limit": "^2.2.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 - }, - "path-exists": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", - "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==", - "dev": true - }, - "require-main-filename": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/require-main-filename/-/require-main-filename-2.0.0.tgz", - "integrity": "sha512-NKN5kMDylKuldxYLSUfrbo5Tuzh4hd+2E8NPPX02mZtn1VuREQToYe/ZdlJy+J3uCpfaiGF05e7B8W0iXbQHmg==", - "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==", - "dev": true - }, - "string-width": { - "version": "4.2.0", - "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.0.tgz", - "integrity": "sha512-zUz5JD+tgqtuDjMhwIg5uFVV3dtqZ9yQJlZVfq4I01/K5Paj5UHj7VyrQOJvzawSVlKpObApbfD0Ed6yJc+1eg==", - "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" - } - }, - "wrap-ansi": { - "version": "6.2.0", - "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-6.2.0.tgz", - "integrity": "sha512-r6lPcBGxZXlIcymEu7InxDMhdW0KDxpLgoFLcguasxCaJ/SOIZwINatK9KY/tf+ZrlywOKU0UDj3ATXUBfxJXA==", - "dev": true, - "requires": { - "ansi-styles": "^4.0.0", - "string-width": "^4.1.0", - "strip-ansi": "^6.0.0" - } - }, - "yargs": { - "version": "15.3.0", - "resolved": "https://registry.npmjs.org/yargs/-/yargs-15.3.0.tgz", - "integrity": "sha512-g/QCnmjgOl1YJjGsnUg2SatC7NUYEiLXJqxNOQU9qSpjzGtGXda9b+OKccr1kLTy8BN9yqEyqfq5lxlwdc13TA==", - "dev": true, - "requires": { - "cliui": "^6.0.0", - "decamelize": "^1.2.0", - "find-up": "^4.1.0", - "get-caller-file": "^2.0.1", - "require-directory": "^2.1.1", - "require-main-filename": "^2.0.0", - "set-blocking": "^2.0.0", - "string-width": "^4.2.0", - "which-module": "^2.0.0", - "y18n": "^4.0.0", - "yargs-parser": "^18.1.0" - } - }, - "yargs-parser": { - "version": "18.1.3", - "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-18.1.3.tgz", - "integrity": "sha512-o50j0JeToy/4K6OZcaQmW6lyXXKhq7csREXcDwk2omFPJEwUNOVtJKvmDr9EI1fAJZUyZcRF7kxGBWmRXudrCQ==", - "dev": true, - "requires": { - "camelcase": "^5.0.0", - "decamelize": "^1.2.0" - } - } - } - }, - "@angular/core": { - "version": "11.0.5", - "resolved": "https://registry.npmjs.org/@angular/core/-/core-11.0.5.tgz", - "integrity": "sha512-XAXWQi7R3ucZXQwx9QK5jSKJeQyRJ53u2dQDpr7R5stzeCy1a5hrNOkZLg9zOTTPcth/6+FrOrRZP9SMdxtw3w==", - "requires": { - "tslib": "^2.0.0" - }, - "dependencies": { - "tslib": { - "version": "2.4.0", - "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.4.0.tgz", - "integrity": "sha512-d6xOpEDfsi2CZVlPQzGeux8XMwLT9hssAsaPYExaQMuYskwb+x1x7J371tWlbBdWHroy99KnVB6qIkUbs5X3UQ==" - } - } - }, - "@angular/flex-layout": { - "version": "9.0.0-beta.29", - "resolved": "https://registry.npmjs.org/@angular/flex-layout/-/flex-layout-9.0.0-beta.29.tgz", - "integrity": "sha512-93sxR+kYfYMOdnlWL0Q77FZ428gg8XnBu0YZm6GsCdkw/vLggIT/G1ZAqHlCPIODt6pxmCJ5KXh4ShvniIYDsA==" - }, - "@angular/forms": { - "version": "9.1.4", - "resolved": "https://registry.npmjs.org/@angular/forms/-/forms-9.1.4.tgz", - "integrity": "sha512-Njt+pMLfPBchL0/ayIjJqXL6ZfM4Ccvf7KO1wS1HMzh3QlmfNa0JSgc4pfrbRJAMN9g7V/FYLyKejs1bJZkenA==" - }, - "@angular/language-service": { - "version": "9.1.4", - "resolved": "https://registry.npmjs.org/@angular/language-service/-/language-service-9.1.4.tgz", - "integrity": "sha512-eyVxxiegdb4ESdFGfkuDN+YfUbOVHRQLjIl6ACFJQDNHzVXzbmuqpyr5hIJANIVady103/7+dqRxxJo1DdIdTQ==", - "dev": true - }, - "@angular/material": { - "version": "9.2.2", - "resolved": "https://registry.npmjs.org/@angular/material/-/material-9.2.2.tgz", - "integrity": "sha512-gdQiMJ6PtW/5fd+0mglHFyzxULDCBGjn9RTET3sUq2rkc9+jBXr4OvnsUyBWSnqqv97XqotVDIx5JgE4/YX/Rw==" - }, - "@angular/platform-browser": { - "version": "9.1.4", - "resolved": "https://registry.npmjs.org/@angular/platform-browser/-/platform-browser-9.1.4.tgz", - "integrity": "sha512-mBCHfTl+5tQfaUiGlDujP7mFBzovFc54Zi2kcCE8DSdSSVQ2TPBo6hXa6y2cL3hJPFZzQ7mC4ORFrsGADhHn/w==" - }, - "@angular/platform-browser-dynamic": { - "version": "9.1.4", - "resolved": "https://registry.npmjs.org/@angular/platform-browser-dynamic/-/platform-browser-dynamic-9.1.4.tgz", - "integrity": "sha512-YtVbnxyS6FU7xNpA6A95tmSfrB8+WC7OH3mbP8M9NaGk0OYz8B/JOe1HByP4JRpEGCvBtXdJ2NSW/MpLIT8SiQ==" - }, - "@angular/router": { - "version": "9.1.4", - "resolved": "https://registry.npmjs.org/@angular/router/-/router-9.1.4.tgz", - "integrity": "sha512-yUyjCgG2P2Jh8MvoyC6yirmAtx1Qe7MKLuLvsa9WOB571QNEcNLTYMfAMHUKsQTcE/+o984QyLlneoibgI9wFA==" - }, - "@babel/code-frame": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.8.3.tgz", - "integrity": "sha512-a9gxpmdXtZEInkCSHUJDLHZVBgb1QS0jhss4cPP93EW7s+uC5bikET2twEF3KV+7rDblJcmNvTR7VJejqd2C2g==", - "dev": true, - "requires": { - "@babel/highlight": "^7.8.3" - } - }, - "@babel/compat-data": { - "version": "7.9.6", - "resolved": "https://registry.npmjs.org/@babel/compat-data/-/compat-data-7.9.6.tgz", - "integrity": "sha512-5QPTrNen2bm7RBc7dsOmcA5hbrS4O2Vhmk5XOL4zWW/zD/hV0iinpefDlkm+tBBy8kDtFaaeEvmAqt+nURAV2g==", - "dev": true, - "requires": { - "browserslist": "^4.11.1", - "invariant": "^2.2.4", - "semver": "^5.5.0" - }, - "dependencies": { - "semver": { - "version": "5.7.1", - "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", - "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", - "dev": true - } - } - }, - "@babel/core": { - "version": "7.9.0", - "resolved": "https://registry.npmjs.org/@babel/core/-/core-7.9.0.tgz", - "integrity": "sha512-kWc7L0fw1xwvI0zi8OKVBuxRVefwGOrKSQMvrQ3dW+bIIavBY3/NpXmpjMy7bQnLgwgzWQZ8TlM57YHpHNHz4w==", - "dev": true, - "requires": { - "@babel/code-frame": "^7.8.3", - "@babel/generator": "^7.9.0", - "@babel/helper-module-transforms": "^7.9.0", - "@babel/helpers": "^7.9.0", - "@babel/parser": "^7.9.0", - "@babel/template": "^7.8.6", - "@babel/traverse": "^7.9.0", - "@babel/types": "^7.9.0", - "convert-source-map": "^1.7.0", - "debug": "^4.1.0", - "gensync": "^1.0.0-beta.1", - "json5": "^2.1.2", - "lodash": "^4.17.13", - "resolve": "^1.3.2", - "semver": "^5.4.1", - "source-map": "^0.5.0" - }, - "dependencies": { - "@babel/generator": { - "version": "7.9.6", - "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.9.6.tgz", - "integrity": "sha512-+htwWKJbH2bL72HRluF8zumBxzuX0ZZUFl3JLNyoUjM/Ho8wnVpPXM6aUz8cfKDqQ/h7zHqKt4xzJteUosckqQ==", - "dev": true, - "requires": { - "@babel/types": "^7.9.6", - "jsesc": "^2.5.1", - "lodash": "^4.17.13", - "source-map": "^0.5.0" - } - }, - "semver": { - "version": "5.7.1", - "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", - "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", - "dev": true - }, - "source-map": { - "version": "0.5.7", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.5.7.tgz", - "integrity": "sha1-igOdLRAh0i0eoUyA2OpGi6LvP8w=", - "dev": true - } - } - }, - "@babel/generator": { - "version": "7.7.7", - "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.7.7.tgz", - "integrity": "sha512-/AOIBpHh/JU1l0ZFS4kiRCBnLi6OTHzh0RPk3h9isBxkkqELtQNFi1Vr/tiG9p1yfoUdKVwISuXWQR+hwwM4VQ==", - "dev": true, - "requires": { - "@babel/types": "^7.7.4", - "jsesc": "^2.5.1", - "lodash": "^4.17.13", - "source-map": "^0.5.0" - }, - "dependencies": { - "source-map": { - "version": "0.5.7", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.5.7.tgz", - "integrity": "sha1-igOdLRAh0i0eoUyA2OpGi6LvP8w=", - "dev": true - } - } - }, - "@babel/helper-annotate-as-pure": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/helper-annotate-as-pure/-/helper-annotate-as-pure-7.8.3.tgz", - "integrity": "sha512-6o+mJrZBxOoEX77Ezv9zwW7WV8DdluouRKNY/IR5u/YTMuKHgugHOzYWlYvYLpLA9nPsQCAAASpCIbjI9Mv+Uw==", - "dev": true, - "requires": { - "@babel/types": "^7.8.3" - } - }, - "@babel/helper-builder-binary-assignment-operator-visitor": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/helper-builder-binary-assignment-operator-visitor/-/helper-builder-binary-assignment-operator-visitor-7.8.3.tgz", - "integrity": "sha512-5eFOm2SyFPK4Rh3XMMRDjN7lBH0orh3ss0g3rTYZnBQ+r6YPj7lgDyCvPphynHvUrobJmeMignBr6Acw9mAPlw==", - "dev": true, - "requires": { - "@babel/helper-explode-assignable-expression": "^7.8.3", - "@babel/types": "^7.8.3" - } - }, - "@babel/helper-compilation-targets": { - "version": "7.9.6", - "resolved": "https://registry.npmjs.org/@babel/helper-compilation-targets/-/helper-compilation-targets-7.9.6.tgz", - "integrity": "sha512-x2Nvu0igO0ejXzx09B/1fGBxY9NXQlBW2kZsSxCJft+KHN8t9XWzIvFxtPHnBOAXpVsdxZKZFbRUC8TsNKajMw==", - "dev": true, - "requires": { - "@babel/compat-data": "^7.9.6", - "browserslist": "^4.11.1", - "invariant": "^2.2.4", - "levenary": "^1.1.1", - "semver": "^5.5.0" - }, - "dependencies": { - "semver": { - "version": "5.7.1", - "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", - "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", - "dev": true - } - } - }, - "@babel/helper-create-regexp-features-plugin": { - "version": "7.8.8", - "resolved": "https://registry.npmjs.org/@babel/helper-create-regexp-features-plugin/-/helper-create-regexp-features-plugin-7.8.8.tgz", - "integrity": "sha512-LYVPdwkrQEiX9+1R29Ld/wTrmQu1SSKYnuOk3g0CkcZMA1p0gsNxJFj/3gBdaJ7Cg0Fnek5z0DsMULePP7Lrqg==", - "dev": true, - "requires": { - "@babel/helper-annotate-as-pure": "^7.8.3", - "@babel/helper-regex": "^7.8.3", - "regexpu-core": "^4.7.0" - } - }, - "@babel/helper-define-map": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/helper-define-map/-/helper-define-map-7.8.3.tgz", - "integrity": "sha512-PoeBYtxoZGtct3md6xZOCWPcKuMuk3IHhgxsRRNtnNShebf4C8YonTSblsK4tvDbm+eJAw2HAPOfCr+Q/YRG/g==", - "dev": true, - "requires": { - "@babel/helper-function-name": "^7.8.3", - "@babel/types": "^7.8.3", - "lodash": "^4.17.13" - } - }, - "@babel/helper-explode-assignable-expression": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/helper-explode-assignable-expression/-/helper-explode-assignable-expression-7.8.3.tgz", - "integrity": "sha512-N+8eW86/Kj147bO9G2uclsg5pwfs/fqqY5rwgIL7eTBklgXjcOJ3btzS5iM6AitJcftnY7pm2lGsrJVYLGjzIw==", - "dev": true, - "requires": { - "@babel/traverse": "^7.8.3", - "@babel/types": "^7.8.3" - } - }, - "@babel/helper-function-name": { - "version": "7.9.5", - "resolved": "https://registry.npmjs.org/@babel/helper-function-name/-/helper-function-name-7.9.5.tgz", - "integrity": "sha512-JVcQZeXM59Cd1qanDUxv9fgJpt3NeKUaqBqUEvfmQ+BCOKq2xUgaWZW2hr0dkbyJgezYuplEoh5knmrnS68efw==", - "dev": true, - "requires": { - "@babel/helper-get-function-arity": "^7.8.3", - "@babel/template": "^7.8.3", - "@babel/types": "^7.9.5" - } - }, - "@babel/helper-get-function-arity": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/helper-get-function-arity/-/helper-get-function-arity-7.8.3.tgz", - "integrity": "sha512-FVDR+Gd9iLjUMY1fzE2SR0IuaJToR4RkCDARVfsBBPSP53GEqSFjD8gNyxg246VUyc/ALRxFaAK8rVG7UT7xRA==", - "dev": true, - "requires": { - "@babel/types": "^7.8.3" - } - }, - "@babel/helper-hoist-variables": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/helper-hoist-variables/-/helper-hoist-variables-7.8.3.tgz", - "integrity": "sha512-ky1JLOjcDUtSc+xkt0xhYff7Z6ILTAHKmZLHPxAhOP0Nd77O+3nCsd6uSVYur6nJnCI029CrNbYlc0LoPfAPQg==", - "dev": true, - "requires": { - "@babel/types": "^7.8.3" - } - }, - "@babel/helper-member-expression-to-functions": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/helper-member-expression-to-functions/-/helper-member-expression-to-functions-7.8.3.tgz", - "integrity": "sha512-fO4Egq88utkQFjbPrSHGmGLFqmrshs11d46WI+WZDESt7Wu7wN2G2Iu+NMMZJFDOVRHAMIkB5SNh30NtwCA7RA==", - "dev": true, - "requires": { - "@babel/types": "^7.8.3" - } - }, - "@babel/helper-module-imports": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/helper-module-imports/-/helper-module-imports-7.8.3.tgz", - "integrity": "sha512-R0Bx3jippsbAEtzkpZ/6FIiuzOURPcMjHp+Z6xPe6DtApDJx+w7UYyOLanZqO8+wKR9G10s/FmHXvxaMd9s6Kg==", - "dev": true, - "requires": { - "@babel/types": "^7.8.3" - } - }, - "@babel/helper-module-transforms": { - "version": "7.9.0", - "resolved": "https://registry.npmjs.org/@babel/helper-module-transforms/-/helper-module-transforms-7.9.0.tgz", - "integrity": "sha512-0FvKyu0gpPfIQ8EkxlrAydOWROdHpBmiCiRwLkUiBGhCUPRRbVD2/tm3sFr/c/GWFrQ/ffutGUAnx7V0FzT2wA==", - "dev": true, - "requires": { - "@babel/helper-module-imports": "^7.8.3", - "@babel/helper-replace-supers": "^7.8.6", - "@babel/helper-simple-access": "^7.8.3", - "@babel/helper-split-export-declaration": "^7.8.3", - "@babel/template": "^7.8.6", - "@babel/types": "^7.9.0", - "lodash": "^4.17.13" - } - }, - "@babel/helper-optimise-call-expression": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/helper-optimise-call-expression/-/helper-optimise-call-expression-7.8.3.tgz", - "integrity": "sha512-Kag20n86cbO2AvHca6EJsvqAd82gc6VMGule4HwebwMlwkpXuVqrNRj6CkCV2sKxgi9MyAUnZVnZ6lJ1/vKhHQ==", - "dev": true, - "requires": { - "@babel/types": "^7.8.3" - } - }, - "@babel/helper-plugin-utils": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/helper-plugin-utils/-/helper-plugin-utils-7.8.3.tgz", - "integrity": "sha512-j+fq49Xds2smCUNYmEHF9kGNkhbet6yVIBp4e6oeQpH1RUs/Ir06xUKzDjDkGcaaokPiTNs2JBWHjaE4csUkZQ==", - "dev": true - }, - "@babel/helper-regex": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/helper-regex/-/helper-regex-7.8.3.tgz", - "integrity": "sha512-BWt0QtYv/cg/NecOAZMdcn/waj/5P26DR4mVLXfFtDokSR6fyuG0Pj+e2FqtSME+MqED1khnSMulkmGl8qWiUQ==", - "dev": true, - "requires": { - "lodash": "^4.17.13" - } - }, - "@babel/helper-remap-async-to-generator": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/helper-remap-async-to-generator/-/helper-remap-async-to-generator-7.8.3.tgz", - "integrity": "sha512-kgwDmw4fCg7AVgS4DukQR/roGp+jP+XluJE5hsRZwxCYGg+Rv9wSGErDWhlI90FODdYfd4xG4AQRiMDjjN0GzA==", - "dev": true, - "requires": { - "@babel/helper-annotate-as-pure": "^7.8.3", - "@babel/helper-wrap-function": "^7.8.3", - "@babel/template": "^7.8.3", - "@babel/traverse": "^7.8.3", - "@babel/types": "^7.8.3" - } - }, - "@babel/helper-replace-supers": { - "version": "7.9.6", - "resolved": "https://registry.npmjs.org/@babel/helper-replace-supers/-/helper-replace-supers-7.9.6.tgz", - "integrity": "sha512-qX+chbxkbArLyCImk3bWV+jB5gTNU/rsze+JlcF6Nf8tVTigPJSI1o1oBow/9Resa1yehUO9lIipsmu9oG4RzA==", - "dev": true, - "requires": { - "@babel/helper-member-expression-to-functions": "^7.8.3", - "@babel/helper-optimise-call-expression": "^7.8.3", - "@babel/traverse": "^7.9.6", - "@babel/types": "^7.9.6" - } - }, - "@babel/helper-simple-access": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/helper-simple-access/-/helper-simple-access-7.8.3.tgz", - "integrity": "sha512-VNGUDjx5cCWg4vvCTR8qQ7YJYZ+HBjxOgXEl7ounz+4Sn7+LMD3CFrCTEU6/qXKbA2nKg21CwhhBzO0RpRbdCw==", - "dev": true, - "requires": { - "@babel/template": "^7.8.3", - "@babel/types": "^7.8.3" - } - }, - "@babel/helper-split-export-declaration": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/helper-split-export-declaration/-/helper-split-export-declaration-7.8.3.tgz", - "integrity": "sha512-3x3yOeyBhW851hroze7ElzdkeRXQYQbFIb7gLK1WQYsw2GWDay5gAJNw1sWJ0VFP6z5J1whqeXH/WCdCjZv6dA==", - "dev": true, - "requires": { - "@babel/types": "^7.8.3" - } - }, - "@babel/helper-validator-identifier": { - "version": "7.9.5", - "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.9.5.tgz", - "integrity": "sha512-/8arLKUFq882w4tWGj9JYzRpAlZgiWUJ+dtteNTDqrRBz9Iguck9Rn3ykuBDoUwh2TO4tSAJlrxDUOXWklJe4g==", - "dev": true - }, - "@babel/helper-wrap-function": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/helper-wrap-function/-/helper-wrap-function-7.8.3.tgz", - "integrity": "sha512-LACJrbUET9cQDzb6kG7EeD7+7doC3JNvUgTEQOx2qaO1fKlzE/Bf05qs9w1oXQMmXlPO65lC3Tq9S6gZpTErEQ==", - "dev": true, - "requires": { - "@babel/helper-function-name": "^7.8.3", - "@babel/template": "^7.8.3", - "@babel/traverse": "^7.8.3", - "@babel/types": "^7.8.3" - } - }, - "@babel/helpers": { - "version": "7.9.6", - "resolved": "https://registry.npmjs.org/@babel/helpers/-/helpers-7.9.6.tgz", - "integrity": "sha512-tI4bUbldloLcHWoRUMAj4g1bF313M/o6fBKhIsb3QnGVPwRm9JsNf/gqMkQ7zjqReABiffPV6RWj7hEglID5Iw==", - "dev": true, - "requires": { - "@babel/template": "^7.8.3", - "@babel/traverse": "^7.9.6", - "@babel/types": "^7.9.6" - } - }, - "@babel/highlight": { - "version": "7.9.0", - "resolved": "https://registry.npmjs.org/@babel/highlight/-/highlight-7.9.0.tgz", - "integrity": "sha512-lJZPilxX7Op3Nv/2cvFdnlepPXDxi29wxteT57Q965oc5R9v86ztx0jfxVrTcBk8C2kcPkkDa2Z4T3ZsPPVWsQ==", - "dev": true, - "requires": { - "@babel/helper-validator-identifier": "^7.9.0", - "chalk": "^2.0.0", - "js-tokens": "^4.0.0" - } - }, - "@babel/parser": { - "version": "7.9.6", - "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.9.6.tgz", - "integrity": "sha512-AoeIEJn8vt+d/6+PXDRPaksYhnlbMIiejioBZvvMQsOjW/JYK6k/0dKnvvP3EhK5GfMBWDPtrxRtegWdAcdq9Q==", - "dev": true - }, - "@babel/plugin-proposal-async-generator-functions": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-async-generator-functions/-/plugin-proposal-async-generator-functions-7.8.3.tgz", - "integrity": "sha512-NZ9zLv848JsV3hs8ryEh7Uaz/0KsmPLqv0+PdkDJL1cJy0K4kOCFa8zc1E3mp+RHPQcpdfb/6GovEsW4VDrOMw==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.8.3", - "@babel/helper-remap-async-to-generator": "^7.8.3", - "@babel/plugin-syntax-async-generators": "^7.8.0" - } - }, - "@babel/plugin-proposal-dynamic-import": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-dynamic-import/-/plugin-proposal-dynamic-import-7.8.3.tgz", - "integrity": "sha512-NyaBbyLFXFLT9FP+zk0kYlUlA8XtCUbehs67F0nnEg7KICgMc2mNkIeu9TYhKzyXMkrapZFwAhXLdnt4IYHy1w==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.8.3", - "@babel/plugin-syntax-dynamic-import": "^7.8.0" - } - }, - "@babel/plugin-proposal-json-strings": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-json-strings/-/plugin-proposal-json-strings-7.8.3.tgz", - "integrity": "sha512-KGhQNZ3TVCQG/MjRbAUwuH+14y9q0tpxs1nWWs3pbSleRdDro9SAMMDyye8HhY1gqZ7/NqIc8SKhya0wRDgP1Q==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.8.3", - "@babel/plugin-syntax-json-strings": "^7.8.0" - } - }, - "@babel/plugin-proposal-nullish-coalescing-operator": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-nullish-coalescing-operator/-/plugin-proposal-nullish-coalescing-operator-7.8.3.tgz", - "integrity": "sha512-TS9MlfzXpXKt6YYomudb/KU7nQI6/xnapG6in1uZxoxDghuSMZsPb6D2fyUwNYSAp4l1iR7QtFOjkqcRYcUsfw==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.8.3", - "@babel/plugin-syntax-nullish-coalescing-operator": "^7.8.0" - } - }, - "@babel/plugin-proposal-numeric-separator": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-numeric-separator/-/plugin-proposal-numeric-separator-7.8.3.tgz", - "integrity": "sha512-jWioO1s6R/R+wEHizfaScNsAx+xKgwTLNXSh7tTC4Usj3ItsPEhYkEpU4h+lpnBwq7NBVOJXfO6cRFYcX69JUQ==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.8.3", - "@babel/plugin-syntax-numeric-separator": "^7.8.3" - } - }, - "@babel/plugin-proposal-object-rest-spread": { - "version": "7.9.6", - "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-object-rest-spread/-/plugin-proposal-object-rest-spread-7.9.6.tgz", - "integrity": "sha512-Ga6/fhGqA9Hj+y6whNpPv8psyaK5xzrQwSPsGPloVkvmH+PqW1ixdnfJ9uIO06OjQNYol3PMnfmJ8vfZtkzF+A==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.8.3", - "@babel/plugin-syntax-object-rest-spread": "^7.8.0", - "@babel/plugin-transform-parameters": "^7.9.5" - } - }, - "@babel/plugin-proposal-optional-catch-binding": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-optional-catch-binding/-/plugin-proposal-optional-catch-binding-7.8.3.tgz", - "integrity": "sha512-0gkX7J7E+AtAw9fcwlVQj8peP61qhdg/89D5swOkjYbkboA2CVckn3kiyum1DE0wskGb7KJJxBdyEBApDLLVdw==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.8.3", - "@babel/plugin-syntax-optional-catch-binding": "^7.8.0" - } - }, - "@babel/plugin-proposal-optional-chaining": { - "version": "7.9.0", - "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-optional-chaining/-/plugin-proposal-optional-chaining-7.9.0.tgz", - "integrity": "sha512-NDn5tu3tcv4W30jNhmc2hyD5c56G6cXx4TesJubhxrJeCvuuMpttxr0OnNCqbZGhFjLrg+NIhxxC+BK5F6yS3w==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.8.3", - "@babel/plugin-syntax-optional-chaining": "^7.8.0" - } - }, - "@babel/plugin-proposal-unicode-property-regex": { - "version": "7.8.8", - "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-unicode-property-regex/-/plugin-proposal-unicode-property-regex-7.8.8.tgz", - "integrity": "sha512-EVhjVsMpbhLw9ZfHWSx2iy13Q8Z/eg8e8ccVWt23sWQK5l1UdkoLJPN5w69UA4uITGBnEZD2JOe4QOHycYKv8A==", - "dev": true, - "requires": { - "@babel/helper-create-regexp-features-plugin": "^7.8.8", - "@babel/helper-plugin-utils": "^7.8.3" - } - }, - "@babel/plugin-syntax-async-generators": { - "version": "7.8.4", - "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-async-generators/-/plugin-syntax-async-generators-7.8.4.tgz", - "integrity": "sha512-tycmZxkGfZaxhMRbXlPXuVFpdWlXpir2W4AMhSJgRKzk/eDlIXOhb2LHWoLpDF7TEHylV5zNhykX6KAgHJmTNw==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.8.0" - } - }, - "@babel/plugin-syntax-dynamic-import": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-dynamic-import/-/plugin-syntax-dynamic-import-7.8.3.tgz", - "integrity": "sha512-5gdGbFon+PszYzqs83S3E5mpi7/y/8M9eC90MRTZfduQOYW76ig6SOSPNe41IG5LoP3FGBn2N0RjVDSQiS94kQ==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.8.0" - } - }, - "@babel/plugin-syntax-json-strings": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-json-strings/-/plugin-syntax-json-strings-7.8.3.tgz", - "integrity": "sha512-lY6kdGpWHvjoe2vk4WrAapEuBR69EMxZl+RoGRhrFGNYVK8mOPAW8VfbT/ZgrFbXlDNiiaxQnAtgVCZ6jv30EA==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.8.0" - } - }, - "@babel/plugin-syntax-nullish-coalescing-operator": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-nullish-coalescing-operator/-/plugin-syntax-nullish-coalescing-operator-7.8.3.tgz", - "integrity": "sha512-aSff4zPII1u2QD7y+F8oDsz19ew4IGEJg9SVW+bqwpwtfFleiQDMdzA/R+UlWDzfnHFCxxleFT0PMIrR36XLNQ==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.8.0" - } - }, - "@babel/plugin-syntax-numeric-separator": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-numeric-separator/-/plugin-syntax-numeric-separator-7.8.3.tgz", - "integrity": "sha512-H7dCMAdN83PcCmqmkHB5dtp+Xa9a6LKSvA2hiFBC/5alSHxM5VgWZXFqDi0YFe8XNGT6iCa+z4V4zSt/PdZ7Dw==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.8.3" - } - }, - "@babel/plugin-syntax-object-rest-spread": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-object-rest-spread/-/plugin-syntax-object-rest-spread-7.8.3.tgz", - "integrity": "sha512-XoqMijGZb9y3y2XskN+P1wUGiVwWZ5JmoDRwx5+3GmEplNyVM2s2Dg8ILFQm8rWM48orGy5YpI5Bl8U1y7ydlA==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.8.0" - } - }, - "@babel/plugin-syntax-optional-catch-binding": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-optional-catch-binding/-/plugin-syntax-optional-catch-binding-7.8.3.tgz", - "integrity": "sha512-6VPD0Pc1lpTqw0aKoeRTMiB+kWhAoT24PA+ksWSBrFtl5SIRVpZlwN3NNPQjehA2E/91FV3RjLWoVTglWcSV3Q==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.8.0" - } - }, - "@babel/plugin-syntax-optional-chaining": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-optional-chaining/-/plugin-syntax-optional-chaining-7.8.3.tgz", - "integrity": "sha512-KoK9ErH1MBlCPxV0VANkXW2/dw4vlbGDrFgz8bmUsBGYkFRcbRwMh6cIJubdPrkxRwuGdtCk0v/wPTKbQgBjkg==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.8.0" - } - }, - "@babel/plugin-syntax-top-level-await": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-top-level-await/-/plugin-syntax-top-level-await-7.8.3.tgz", - "integrity": "sha512-kwj1j9lL/6Wd0hROD3b/OZZ7MSrZLqqn9RAZ5+cYYsflQ9HZBIKCUkr3+uL1MEJ1NePiUbf98jjiMQSv0NMR9g==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.8.3" - } - }, - "@babel/plugin-transform-arrow-functions": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-arrow-functions/-/plugin-transform-arrow-functions-7.8.3.tgz", - "integrity": "sha512-0MRF+KC8EqH4dbuITCWwPSzsyO3HIWWlm30v8BbbpOrS1B++isGxPnnuq/IZvOX5J2D/p7DQalQm+/2PnlKGxg==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.8.3" - } - }, - "@babel/plugin-transform-async-to-generator": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-async-to-generator/-/plugin-transform-async-to-generator-7.8.3.tgz", - "integrity": "sha512-imt9tFLD9ogt56Dd5CI/6XgpukMwd/fLGSrix2httihVe7LOGVPhyhMh1BU5kDM7iHD08i8uUtmV2sWaBFlHVQ==", - "dev": true, - "requires": { - "@babel/helper-module-imports": "^7.8.3", - "@babel/helper-plugin-utils": "^7.8.3", - "@babel/helper-remap-async-to-generator": "^7.8.3" - } - }, - "@babel/plugin-transform-block-scoped-functions": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-block-scoped-functions/-/plugin-transform-block-scoped-functions-7.8.3.tgz", - "integrity": "sha512-vo4F2OewqjbB1+yaJ7k2EJFHlTP3jR634Z9Cj9itpqNjuLXvhlVxgnjsHsdRgASR8xYDrx6onw4vW5H6We0Jmg==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.8.3" - } - }, - "@babel/plugin-transform-block-scoping": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-block-scoping/-/plugin-transform-block-scoping-7.8.3.tgz", - "integrity": "sha512-pGnYfm7RNRgYRi7bids5bHluENHqJhrV4bCZRwc5GamaWIIs07N4rZECcmJL6ZClwjDz1GbdMZFtPs27hTB06w==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.8.3", - "lodash": "^4.17.13" - } - }, - "@babel/plugin-transform-classes": { - "version": "7.9.5", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-classes/-/plugin-transform-classes-7.9.5.tgz", - "integrity": "sha512-x2kZoIuLC//O5iA7PEvecB105o7TLzZo8ofBVhP79N+DO3jaX+KYfww9TQcfBEZD0nikNyYcGB1IKtRq36rdmg==", - "dev": true, - "requires": { - "@babel/helper-annotate-as-pure": "^7.8.3", - "@babel/helper-define-map": "^7.8.3", - "@babel/helper-function-name": "^7.9.5", - "@babel/helper-optimise-call-expression": "^7.8.3", - "@babel/helper-plugin-utils": "^7.8.3", - "@babel/helper-replace-supers": "^7.8.6", - "@babel/helper-split-export-declaration": "^7.8.3", - "globals": "^11.1.0" - } - }, - "@babel/plugin-transform-computed-properties": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-computed-properties/-/plugin-transform-computed-properties-7.8.3.tgz", - "integrity": "sha512-O5hiIpSyOGdrQZRQ2ccwtTVkgUDBBiCuK//4RJ6UfePllUTCENOzKxfh6ulckXKc0DixTFLCfb2HVkNA7aDpzA==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.8.3" - } - }, - "@babel/plugin-transform-destructuring": { - "version": "7.9.5", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-destructuring/-/plugin-transform-destructuring-7.9.5.tgz", - "integrity": "sha512-j3OEsGel8nHL/iusv/mRd5fYZ3DrOxWC82x0ogmdN/vHfAP4MYw+AFKYanzWlktNwikKvlzUV//afBW5FTp17Q==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.8.3" - } - }, - "@babel/plugin-transform-dotall-regex": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-dotall-regex/-/plugin-transform-dotall-regex-7.8.3.tgz", - "integrity": "sha512-kLs1j9Nn4MQoBYdRXH6AeaXMbEJFaFu/v1nQkvib6QzTj8MZI5OQzqmD83/2jEM1z0DLilra5aWO5YpyC0ALIw==", - "dev": true, - "requires": { - "@babel/helper-create-regexp-features-plugin": "^7.8.3", - "@babel/helper-plugin-utils": "^7.8.3" - } - }, - "@babel/plugin-transform-duplicate-keys": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-duplicate-keys/-/plugin-transform-duplicate-keys-7.8.3.tgz", - "integrity": "sha512-s8dHiBUbcbSgipS4SMFuWGqCvyge5V2ZeAWzR6INTVC3Ltjig/Vw1G2Gztv0vU/hRG9X8IvKvYdoksnUfgXOEQ==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.8.3" - } - }, - "@babel/plugin-transform-exponentiation-operator": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-exponentiation-operator/-/plugin-transform-exponentiation-operator-7.8.3.tgz", - "integrity": "sha512-zwIpuIymb3ACcInbksHaNcR12S++0MDLKkiqXHl3AzpgdKlFNhog+z/K0+TGW+b0w5pgTq4H6IwV/WhxbGYSjQ==", - "dev": true, - "requires": { - "@babel/helper-builder-binary-assignment-operator-visitor": "^7.8.3", - "@babel/helper-plugin-utils": "^7.8.3" - } - }, - "@babel/plugin-transform-for-of": { - "version": "7.9.0", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-for-of/-/plugin-transform-for-of-7.9.0.tgz", - "integrity": "sha512-lTAnWOpMwOXpyDx06N+ywmF3jNbafZEqZ96CGYabxHrxNX8l5ny7dt4bK/rGwAh9utyP2b2Hv7PlZh1AAS54FQ==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.8.3" - } - }, - "@babel/plugin-transform-function-name": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-function-name/-/plugin-transform-function-name-7.8.3.tgz", - "integrity": "sha512-rO/OnDS78Eifbjn5Py9v8y0aR+aSYhDhqAwVfsTl0ERuMZyr05L1aFSCJnbv2mmsLkit/4ReeQ9N2BgLnOcPCQ==", - "dev": true, - "requires": { - "@babel/helper-function-name": "^7.8.3", - "@babel/helper-plugin-utils": "^7.8.3" - } - }, - "@babel/plugin-transform-literals": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-literals/-/plugin-transform-literals-7.8.3.tgz", - "integrity": "sha512-3Tqf8JJ/qB7TeldGl+TT55+uQei9JfYaregDcEAyBZ7akutriFrt6C/wLYIer6OYhleVQvH/ntEhjE/xMmy10A==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.8.3" - } - }, - "@babel/plugin-transform-member-expression-literals": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-member-expression-literals/-/plugin-transform-member-expression-literals-7.8.3.tgz", - "integrity": "sha512-3Wk2EXhnw+rP+IDkK6BdtPKsUE5IeZ6QOGrPYvw52NwBStw9V1ZVzxgK6fSKSxqUvH9eQPR3tm3cOq79HlsKYA==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.8.3" - } - }, - "@babel/plugin-transform-modules-amd": { - "version": "7.9.6", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-modules-amd/-/plugin-transform-modules-amd-7.9.6.tgz", - "integrity": "sha512-zoT0kgC3EixAyIAU+9vfaUVKTv9IxBDSabgHoUCBP6FqEJ+iNiN7ip7NBKcYqbfUDfuC2mFCbM7vbu4qJgOnDw==", - "dev": true, - "requires": { - "@babel/helper-module-transforms": "^7.9.0", - "@babel/helper-plugin-utils": "^7.8.3", - "babel-plugin-dynamic-import-node": "^2.3.3" - } - }, - "@babel/plugin-transform-modules-commonjs": { - "version": "7.9.6", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-modules-commonjs/-/plugin-transform-modules-commonjs-7.9.6.tgz", - "integrity": "sha512-7H25fSlLcn+iYimmsNe3uK1at79IE6SKW9q0/QeEHTMC9MdOZ+4bA+T1VFB5fgOqBWoqlifXRzYD0JPdmIrgSQ==", - "dev": true, - "requires": { - "@babel/helper-module-transforms": "^7.9.0", - "@babel/helper-plugin-utils": "^7.8.3", - "@babel/helper-simple-access": "^7.8.3", - "babel-plugin-dynamic-import-node": "^2.3.3" - } - }, - "@babel/plugin-transform-modules-systemjs": { - "version": "7.9.6", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-modules-systemjs/-/plugin-transform-modules-systemjs-7.9.6.tgz", - "integrity": "sha512-NW5XQuW3N2tTHim8e1b7qGy7s0kZ2OH3m5octc49K1SdAKGxYxeIx7hiIz05kS1R2R+hOWcsr1eYwcGhrdHsrg==", - "dev": true, - "requires": { - "@babel/helper-hoist-variables": "^7.8.3", - "@babel/helper-module-transforms": "^7.9.0", - "@babel/helper-plugin-utils": "^7.8.3", - "babel-plugin-dynamic-import-node": "^2.3.3" - } - }, - "@babel/plugin-transform-modules-umd": { - "version": "7.9.0", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-modules-umd/-/plugin-transform-modules-umd-7.9.0.tgz", - "integrity": "sha512-uTWkXkIVtg/JGRSIABdBoMsoIeoHQHPTL0Y2E7xf5Oj7sLqwVsNXOkNk0VJc7vF0IMBsPeikHxFjGe+qmwPtTQ==", - "dev": true, - "requires": { - "@babel/helper-module-transforms": "^7.9.0", - "@babel/helper-plugin-utils": "^7.8.3" - } - }, - "@babel/plugin-transform-named-capturing-groups-regex": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-named-capturing-groups-regex/-/plugin-transform-named-capturing-groups-regex-7.8.3.tgz", - "integrity": "sha512-f+tF/8UVPU86TrCb06JoPWIdDpTNSGGcAtaD9mLP0aYGA0OS0j7j7DHJR0GTFrUZPUU6loZhbsVZgTh0N+Qdnw==", - "dev": true, - "requires": { - "@babel/helper-create-regexp-features-plugin": "^7.8.3" - } - }, - "@babel/plugin-transform-new-target": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-new-target/-/plugin-transform-new-target-7.8.3.tgz", - "integrity": "sha512-QuSGysibQpyxexRyui2vca+Cmbljo8bcRckgzYV4kRIsHpVeyeC3JDO63pY+xFZ6bWOBn7pfKZTqV4o/ix9sFw==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.8.3" - } - }, - "@babel/plugin-transform-object-super": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-object-super/-/plugin-transform-object-super-7.8.3.tgz", - "integrity": "sha512-57FXk+gItG/GejofIyLIgBKTas4+pEU47IXKDBWFTxdPd7F80H8zybyAY7UoblVfBhBGs2EKM+bJUu2+iUYPDQ==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.8.3", - "@babel/helper-replace-supers": "^7.8.3" - } - }, - "@babel/plugin-transform-parameters": { - "version": "7.9.5", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-parameters/-/plugin-transform-parameters-7.9.5.tgz", - "integrity": "sha512-0+1FhHnMfj6lIIhVvS4KGQJeuhe1GI//h5uptK4PvLt+BGBxsoUJbd3/IW002yk//6sZPlFgsG1hY6OHLcy6kA==", - "dev": true, - "requires": { - "@babel/helper-get-function-arity": "^7.8.3", - "@babel/helper-plugin-utils": "^7.8.3" - } - }, - "@babel/plugin-transform-property-literals": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-property-literals/-/plugin-transform-property-literals-7.8.3.tgz", - "integrity": "sha512-uGiiXAZMqEoQhRWMK17VospMZh5sXWg+dlh2soffpkAl96KAm+WZuJfa6lcELotSRmooLqg0MWdH6UUq85nmmg==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.8.3" - } - }, - "@babel/plugin-transform-regenerator": { - "version": "7.8.7", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-regenerator/-/plugin-transform-regenerator-7.8.7.tgz", - "integrity": "sha512-TIg+gAl4Z0a3WmD3mbYSk+J9ZUH6n/Yc57rtKRnlA/7rcCvpekHXe0CMZHP1gYp7/KLe9GHTuIba0vXmls6drA==", - "dev": true, - "requires": { - "regenerator-transform": "^0.14.2" - } - }, - "@babel/plugin-transform-reserved-words": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-reserved-words/-/plugin-transform-reserved-words-7.8.3.tgz", - "integrity": "sha512-mwMxcycN3omKFDjDQUl+8zyMsBfjRFr0Zn/64I41pmjv4NJuqcYlEtezwYtw9TFd9WR1vN5kiM+O0gMZzO6L0A==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.8.3" - } - }, - "@babel/plugin-transform-shorthand-properties": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-shorthand-properties/-/plugin-transform-shorthand-properties-7.8.3.tgz", - "integrity": "sha512-I9DI6Odg0JJwxCHzbzW08ggMdCezoWcuQRz3ptdudgwaHxTjxw5HgdFJmZIkIMlRymL6YiZcped4TTCB0JcC8w==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.8.3" - } - }, - "@babel/plugin-transform-spread": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-spread/-/plugin-transform-spread-7.8.3.tgz", - "integrity": "sha512-CkuTU9mbmAoFOI1tklFWYYbzX5qCIZVXPVy0jpXgGwkplCndQAa58s2jr66fTeQnA64bDox0HL4U56CFYoyC7g==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.8.3" - } - }, - "@babel/plugin-transform-sticky-regex": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-sticky-regex/-/plugin-transform-sticky-regex-7.8.3.tgz", - "integrity": "sha512-9Spq0vGCD5Bb4Z/ZXXSK5wbbLFMG085qd2vhL1JYu1WcQ5bXqZBAYRzU1d+p79GcHs2szYv5pVQCX13QgldaWw==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.8.3", - "@babel/helper-regex": "^7.8.3" - } - }, - "@babel/plugin-transform-template-literals": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-template-literals/-/plugin-transform-template-literals-7.8.3.tgz", - "integrity": "sha512-820QBtykIQOLFT8NZOcTRJ1UNuztIELe4p9DCgvj4NK+PwluSJ49we7s9FB1HIGNIYT7wFUJ0ar2QpCDj0escQ==", - "dev": true, - "requires": { - "@babel/helper-annotate-as-pure": "^7.8.3", - "@babel/helper-plugin-utils": "^7.8.3" - } - }, - "@babel/plugin-transform-typeof-symbol": { - "version": "7.8.4", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-typeof-symbol/-/plugin-transform-typeof-symbol-7.8.4.tgz", - "integrity": "sha512-2QKyfjGdvuNfHsb7qnBBlKclbD4CfshH2KvDabiijLMGXPHJXGxtDzwIF7bQP+T0ysw8fYTtxPafgfs/c1Lrqg==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.8.3" - } - }, - "@babel/plugin-transform-unicode-regex": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-unicode-regex/-/plugin-transform-unicode-regex-7.8.3.tgz", - "integrity": "sha512-+ufgJjYdmWfSQ+6NS9VGUR2ns8cjJjYbrbi11mZBTaWm+Fui/ncTLFF28Ei1okavY+xkojGr1eJxNsWYeA5aZw==", - "dev": true, - "requires": { - "@babel/helper-create-regexp-features-plugin": "^7.8.3", - "@babel/helper-plugin-utils": "^7.8.3" - } - }, - "@babel/preset-env": { - "version": "7.9.0", - "resolved": "https://registry.npmjs.org/@babel/preset-env/-/preset-env-7.9.0.tgz", - "integrity": "sha512-712DeRXT6dyKAM/FMbQTV/FvRCms2hPCx+3weRjZ8iQVQWZejWWk1wwG6ViWMyqb/ouBbGOl5b6aCk0+j1NmsQ==", - "dev": true, - "requires": { - "@babel/compat-data": "^7.9.0", - "@babel/helper-compilation-targets": "^7.8.7", - "@babel/helper-module-imports": "^7.8.3", - "@babel/helper-plugin-utils": "^7.8.3", - "@babel/plugin-proposal-async-generator-functions": "^7.8.3", - "@babel/plugin-proposal-dynamic-import": "^7.8.3", - "@babel/plugin-proposal-json-strings": "^7.8.3", - "@babel/plugin-proposal-nullish-coalescing-operator": "^7.8.3", - "@babel/plugin-proposal-numeric-separator": "^7.8.3", - "@babel/plugin-proposal-object-rest-spread": "^7.9.0", - "@babel/plugin-proposal-optional-catch-binding": "^7.8.3", - "@babel/plugin-proposal-optional-chaining": "^7.9.0", - "@babel/plugin-proposal-unicode-property-regex": "^7.8.3", - "@babel/plugin-syntax-async-generators": "^7.8.0", - "@babel/plugin-syntax-dynamic-import": "^7.8.0", - "@babel/plugin-syntax-json-strings": "^7.8.0", - "@babel/plugin-syntax-nullish-coalescing-operator": "^7.8.0", - "@babel/plugin-syntax-numeric-separator": "^7.8.0", - "@babel/plugin-syntax-object-rest-spread": "^7.8.0", - "@babel/plugin-syntax-optional-catch-binding": "^7.8.0", - "@babel/plugin-syntax-optional-chaining": "^7.8.0", - "@babel/plugin-syntax-top-level-await": "^7.8.3", - "@babel/plugin-transform-arrow-functions": "^7.8.3", - "@babel/plugin-transform-async-to-generator": "^7.8.3", - "@babel/plugin-transform-block-scoped-functions": "^7.8.3", - "@babel/plugin-transform-block-scoping": "^7.8.3", - "@babel/plugin-transform-classes": "^7.9.0", - "@babel/plugin-transform-computed-properties": "^7.8.3", - "@babel/plugin-transform-destructuring": "^7.8.3", - "@babel/plugin-transform-dotall-regex": "^7.8.3", - "@babel/plugin-transform-duplicate-keys": "^7.8.3", - "@babel/plugin-transform-exponentiation-operator": "^7.8.3", - "@babel/plugin-transform-for-of": "^7.9.0", - "@babel/plugin-transform-function-name": "^7.8.3", - "@babel/plugin-transform-literals": "^7.8.3", - "@babel/plugin-transform-member-expression-literals": "^7.8.3", - "@babel/plugin-transform-modules-amd": "^7.9.0", - "@babel/plugin-transform-modules-commonjs": "^7.9.0", - "@babel/plugin-transform-modules-systemjs": "^7.9.0", - "@babel/plugin-transform-modules-umd": "^7.9.0", - "@babel/plugin-transform-named-capturing-groups-regex": "^7.8.3", - "@babel/plugin-transform-new-target": "^7.8.3", - "@babel/plugin-transform-object-super": "^7.8.3", - "@babel/plugin-transform-parameters": "^7.8.7", - "@babel/plugin-transform-property-literals": "^7.8.3", - "@babel/plugin-transform-regenerator": "^7.8.7", - "@babel/plugin-transform-reserved-words": "^7.8.3", - "@babel/plugin-transform-shorthand-properties": "^7.8.3", - "@babel/plugin-transform-spread": "^7.8.3", - "@babel/plugin-transform-sticky-regex": "^7.8.3", - "@babel/plugin-transform-template-literals": "^7.8.3", - "@babel/plugin-transform-typeof-symbol": "^7.8.4", - "@babel/plugin-transform-unicode-regex": "^7.8.3", - "@babel/preset-modules": "^0.1.3", - "@babel/types": "^7.9.0", - "browserslist": "^4.9.1", - "core-js-compat": "^3.6.2", - "invariant": "^2.2.2", - "levenary": "^1.1.1", - "semver": "^5.5.0" - }, - "dependencies": { - "semver": { - "version": "5.7.1", - "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", - "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", - "dev": true - } - } - }, - "@babel/preset-modules": { - "version": "0.1.3", - "resolved": "https://registry.npmjs.org/@babel/preset-modules/-/preset-modules-0.1.3.tgz", - "integrity": "sha512-Ra3JXOHBq2xd56xSF7lMKXdjBn3T772Y1Wet3yWnkDly9zHvJki029tAFzvAAK5cf4YV3yoxuP61crYRol6SVg==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.0.0", - "@babel/plugin-proposal-unicode-property-regex": "^7.4.4", - "@babel/plugin-transform-dotall-regex": "^7.4.4", - "@babel/types": "^7.4.4", - "esutils": "^2.0.2" - } - }, - "@babel/runtime": { - "version": "7.9.6", - "resolved": "https://registry.npmjs.org/@babel/runtime/-/runtime-7.9.6.tgz", - "integrity": "sha512-64AF1xY3OAkFHqOb9s4jpgk1Mm5vDZ4L3acHvAml+53nO1XbXLuDodsVpO4OIUsmemlUHMxNdYMNJmsvOwLrvQ==", - "dev": true, - "requires": { - "regenerator-runtime": "^0.13.4" - } - }, - "@babel/template": { - "version": "7.8.6", - "resolved": "https://registry.npmjs.org/@babel/template/-/template-7.8.6.tgz", - "integrity": "sha512-zbMsPMy/v0PWFZEhQJ66bqjhH+z0JgMoBWuikXybgG3Gkd/3t5oQ1Rw2WQhnSrsOmsKXnZOx15tkC4qON/+JPg==", - "dev": true, - "requires": { - "@babel/code-frame": "^7.8.3", - "@babel/parser": "^7.8.6", - "@babel/types": "^7.8.6" - } - }, - "@babel/traverse": { - "version": "7.9.6", - "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.9.6.tgz", - "integrity": "sha512-b3rAHSjbxy6VEAvlxM8OV/0X4XrG72zoxme6q1MOoe2vd0bEc+TwayhuC1+Dfgqh1QEG+pj7atQqvUprHIccsg==", - "dev": true, - "requires": { - "@babel/code-frame": "^7.8.3", - "@babel/generator": "^7.9.6", - "@babel/helper-function-name": "^7.9.5", - "@babel/helper-split-export-declaration": "^7.8.3", - "@babel/parser": "^7.9.6", - "@babel/types": "^7.9.6", - "debug": "^4.1.0", - "globals": "^11.1.0", - "lodash": "^4.17.13" - }, - "dependencies": { - "@babel/generator": { - "version": "7.9.6", - "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.9.6.tgz", - "integrity": "sha512-+htwWKJbH2bL72HRluF8zumBxzuX0ZZUFl3JLNyoUjM/Ho8wnVpPXM6aUz8cfKDqQ/h7zHqKt4xzJteUosckqQ==", - "dev": true, - "requires": { - "@babel/types": "^7.9.6", - "jsesc": "^2.5.1", - "lodash": "^4.17.13", - "source-map": "^0.5.0" - } - }, - "source-map": { - "version": "0.5.7", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.5.7.tgz", - "integrity": "sha1-igOdLRAh0i0eoUyA2OpGi6LvP8w=", - "dev": true - } - } - }, - "@babel/types": { - "version": "7.9.6", - "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.9.6.tgz", - "integrity": "sha512-qxXzvBO//jO9ZnoasKF1uJzHd2+M6Q2ZPIVfnFps8JJvXy0ZBbwbNOmE6SGIY5XOY6d1Bo5lb9d9RJ8nv3WSeA==", - "dev": true, - "requires": { - "@babel/helper-validator-identifier": "^7.9.5", - "lodash": "^4.17.13", - "to-fast-properties": "^2.0.0" - } - }, - "@ctrl/ngx-codemirror": { - "version": "3.1.3", - "resolved": "https://registry.npmjs.org/@ctrl/ngx-codemirror/-/ngx-codemirror-3.1.3.tgz", - "integrity": "sha512-F9YjFdSt+ibght8lvQ/wvaYGRT3twXzP6Vo7moSy4Y5dEl5gIklNPnH0z0/Kp6gwABuo8vgQ6Ur+Gh7s5OmHYg==", - "requires": { - "@types/codemirror": "0.0.89" - } - }, - "@istanbuljs/schema": { - "version": "0.1.2", - "resolved": "https://registry.npmjs.org/@istanbuljs/schema/-/schema-0.1.2.tgz", - "integrity": "sha512-tsAQNx32a8CoFhjhijUIhI4kccIAgmGhy8LZMZgGfmXcpMbPRUqn5LWmgRttILi6yeGmBJd2xsPkFMs0PzgPCw==", - "dev": true - }, - "@jsdevtools/coverage-istanbul-loader": { - "version": "3.0.3", - "resolved": "https://registry.npmjs.org/@jsdevtools/coverage-istanbul-loader/-/coverage-istanbul-loader-3.0.3.tgz", - "integrity": "sha512-TAdNkeGB5Fe4Og+ZkAr1Kvn9by2sfL44IAHFtxlh1BA1XJ5cLpO9iSNki5opWESv3l3vSHsZ9BNKuqFKbEbFaA==", - "dev": true, - "requires": { - "convert-source-map": "^1.7.0", - "istanbul-lib-instrument": "^4.0.1", - "loader-utils": "^1.4.0", - "merge-source-map": "^1.1.0", - "schema-utils": "^2.6.4" - }, - "dependencies": { - "json5": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/json5/-/json5-1.0.1.tgz", - "integrity": "sha512-aKS4WQjPenRxiQsC93MNfjx+nbF4PAdYzmd/1JIj8HYzqfbu86beTuNgXDzPknWk0n0uARlyewZo4s++ES36Ow==", - "dev": true, - "requires": { - "minimist": "^1.2.0" - } - }, - "loader-utils": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/loader-utils/-/loader-utils-1.4.0.tgz", - "integrity": "sha512-qH0WSMBtn/oHuwjy/NucEgbx5dbxxnxup9s4PVXJUDHZBQY+s0NWA9rJf53RBnQZxfch7euUui7hpoAPvALZdA==", - "dev": true, - "requires": { - "big.js": "^5.2.2", - "emojis-list": "^3.0.0", - "json5": "^1.0.1" - } - } - } - }, - "@ngrx/effects": { - "version": "9.1.2", - "resolved": "https://registry.npmjs.org/@ngrx/effects/-/effects-9.1.2.tgz", - "integrity": "sha512-H9jbGUzP5izk9Ap8BQJicO1+xheyDyHBbvv6b1NkaRHpDizhPOSBjoFWExFfsejXo0dafaIsu6aI+y+Fp+LSsg==" - }, - "@ngrx/store": { - "version": "9.1.1", - "resolved": "https://registry.npmjs.org/@ngrx/store/-/store-9.1.1.tgz", - "integrity": "sha512-QxoNx+bfHb/3jkIsp/K+EnLodkY9zxHZIgaCXdkLPUzN5BUsCOLT6PLX0p1NUaBfJT/haBv7dAx7jGbSnrnUKA==" - }, - "@ngtools/webpack": { - "version": "9.1.4", - "resolved": "https://registry.npmjs.org/@ngtools/webpack/-/webpack-9.1.4.tgz", - "integrity": "sha512-CDlQzMnWraxf/dT3G5L+0N0VniWHWdzjLRP8pWBoyV0xlzWS1yL/lj8Mas2jEIuaUxZ8bi29mE7xa8prqewtBQ==", - "dev": true, - "requires": { - "@angular-devkit/core": "9.1.4", - "enhanced-resolve": "4.1.1", - "rxjs": "6.5.4", - "webpack-sources": "1.4.3" - }, - "dependencies": { - "rxjs": { - "version": "6.5.4", - "resolved": "https://registry.npmjs.org/rxjs/-/rxjs-6.5.4.tgz", - "integrity": "sha512-naMQXcgEo3csAEGvw/NydRA0fuS2nDZJiw1YUWFKU7aPPAPGZEsD4Iimit96qwCieH6y614MCLYwdkrWx7z/7Q==", - "dev": true, - "requires": { - "tslib": "^1.9.0" - } - } - } - }, - "@schematics/angular": { - "version": "9.1.4", - "resolved": "https://registry.npmjs.org/@schematics/angular/-/angular-9.1.4.tgz", - "integrity": "sha512-7nbiYBRgXc+f0sa5iXTcF6/VMy/Xq0wyKKnDFiLCb2aFYR7EXRHWF2GuwDIg2bvFugLuCBDoNnXeddLE1TXGWg==", - "dev": true, - "requires": { - "@angular-devkit/core": "9.1.4", - "@angular-devkit/schematics": "9.1.4" - } - }, - "@schematics/update": { - "version": "0.901.4", - "resolved": "https://registry.npmjs.org/@schematics/update/-/update-0.901.4.tgz", - "integrity": "sha512-jCtZY2Fbj502gKN5gdu1brnRy/ssyzTrWm69Ty73V+t8uL7nLr+/hny/VBJ8CiQnKQvxcgVl1xgvI1cXzpysVA==", - "dev": true, - "requires": { - "@angular-devkit/core": "9.1.4", - "@angular-devkit/schematics": "9.1.4", - "@yarnpkg/lockfile": "1.1.0", - "ini": "1.3.5", - "npm-package-arg": "^8.0.0", - "pacote": "9.5.12", - "rxjs": "6.5.4", - "semver": "7.1.3", - "semver-intersect": "1.4.0" - }, - "dependencies": { - "rxjs": { - "version": "6.5.4", - "resolved": "https://registry.npmjs.org/rxjs/-/rxjs-6.5.4.tgz", - "integrity": "sha512-naMQXcgEo3csAEGvw/NydRA0fuS2nDZJiw1YUWFKU7aPPAPGZEsD4Iimit96qwCieH6y614MCLYwdkrWx7z/7Q==", - "dev": true, - "requires": { - "tslib": "^1.9.0" - } - }, - "semver": { - "version": "7.1.3", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.1.3.tgz", - "integrity": "sha512-ekM0zfiA9SCBlsKa2X1hxyxiI4L3B6EbVJkkdgQXnSEEaHlGdvyodMruTiulSRWMMB4NeIuYNMC9rTKTz97GxA==", - "dev": true - } - } - }, - "@socket.io/base64-arraybuffer": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/@socket.io/base64-arraybuffer/-/base64-arraybuffer-1.0.2.tgz", - "integrity": "sha512-dOlCBKnDw4iShaIsH/bxujKTM18+2TOAsYz+KSc11Am38H4q5Xw8Bbz97ZYdrVNM+um3p7w86Bvvmcn9q+5+eQ==", - "dev": true - }, - "@types/codemirror": { - "version": "0.0.89", - "resolved": "https://registry.npmjs.org/@types/codemirror/-/codemirror-0.0.89.tgz", - "integrity": "sha512-kLw8yUzz1dTAHQh+WYA/2Ka7YZUf5h3oHzgwP48xhbfmqqxrfp9lWuP9Ro26SUDHTthVl5afAa+DabsL1KD5oQ==", - "requires": { - "@types/tern": "*" - } - }, - "@types/color-name": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/@types/color-name/-/color-name-1.1.1.tgz", - "integrity": "sha512-rr+OQyAjxze7GgWrSaJwydHStIhHq2lvY3BOC2Mj7KnzI7XK0Uw1TOOdI9lDoajEbSWLiYgoo4f1R51erQfhPQ==", - "dev": true - }, - "@types/component-emitter": { - "version": "1.2.11", - "resolved": "https://registry.npmjs.org/@types/component-emitter/-/component-emitter-1.2.11.tgz", - "integrity": "sha512-SRXjM+tfsSlA9VuG8hGO2nft2p8zjXCK1VcC6N4NXbBbYbSia9kzCChYQajIjzIqOOOuh5Ock6MmV2oux4jDZQ==", - "dev": true - }, - "@types/cookie": { - "version": "0.4.1", - "resolved": "https://registry.npmjs.org/@types/cookie/-/cookie-0.4.1.tgz", - "integrity": "sha512-XW/Aa8APYr6jSVVA1y/DEIZX0/GMKLEVekNG727R8cs56ahETkRAy/3DR7+fJyh7oUgGwNQaRfXCun0+KbWY7Q==", - "dev": true - }, - "@types/cors": { - "version": "2.8.12", - "resolved": "https://registry.npmjs.org/@types/cors/-/cors-2.8.12.tgz", - "integrity": "sha512-vt+kDhq/M2ayberEtJcIN/hxXy1Pk+59g2FV/ZQceeaTyCtCucjL2Q7FXlFjtWn4n15KCr1NE2lNNFhp0lEThw==", - "dev": true - }, - "@types/estree": { - "version": "0.0.44", - "resolved": "https://registry.npmjs.org/@types/estree/-/estree-0.0.44.tgz", - "integrity": "sha512-iaIVzr+w2ZJ5HkidlZ3EJM8VTZb2MJLCjw3V+505yVts0gRC4UMvjw0d1HPtGqI/HQC/KdsYtayfzl+AXY2R8g==" - }, - "@types/events": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/@types/events/-/events-3.0.0.tgz", - "integrity": "sha512-EaObqwIvayI5a8dCzhFrjKzVwKLxjoG9T6Ppd5CEo07LRKfQ8Yokw54r5+Wq7FaBQ+yXRvQAYPrHwya1/UFt9g==", - "dev": true - }, - "@types/glob": { - "version": "7.1.1", - "resolved": "https://registry.npmjs.org/@types/glob/-/glob-7.1.1.tgz", - "integrity": "sha512-1Bh06cbWJUHMC97acuD6UMG29nMt0Aqz1vF3guLfG+kHHJhy3AyohZFFxYk2f7Q1SQIrNwvncxAE0N/9s70F2w==", - "dev": true, - "requires": { - "@types/events": "*", - "@types/minimatch": "*", - "@types/node": "*" - } - }, - "@types/highlight.js": { - "version": "9.12.3", - "resolved": "https://registry.npmjs.org/@types/highlight.js/-/highlight.js-9.12.3.tgz", - "integrity": "sha512-pGF/zvYOACZ/gLGWdQH8zSwteQS1epp68yRcVLJMgUck/MjEn/FBYmPub9pXT8C1e4a8YZfHo1CKyV8q1vKUnQ==", - "dev": true - }, - "@types/jasmine": { - "version": "3.5.10", - "resolved": "https://registry.npmjs.org/@types/jasmine/-/jasmine-3.5.10.tgz", - "integrity": "sha512-3F8qpwBAiVc5+HPJeXJpbrl+XjawGmciN5LgiO7Gv1pl1RHtjoMNqZpqEksaPJW05ViKe8snYInRs6xB25Xdew==", - "dev": true - }, - "@types/jasminewd2": { - "version": "2.0.8", - "resolved": "https://registry.npmjs.org/@types/jasminewd2/-/jasminewd2-2.0.8.tgz", - "integrity": "sha512-d9p31r7Nxk0ZH0U39PTH0hiDlJ+qNVGjlt1ucOoTUptxb2v+Y5VMnsxfwN+i3hK4yQnqBi3FMmoMFcd1JHDxdg==", - "dev": true, - "requires": { - "@types/jasmine": "*" - } - }, - "@types/minimatch": { - "version": "3.0.3", - "resolved": "https://registry.npmjs.org/@types/minimatch/-/minimatch-3.0.3.tgz", - "integrity": "sha512-tHq6qdbT9U1IRSGf14CL0pUlULksvY9OZ+5eEgl1N7t+OA3tGvNpxJCzuKQlsNgCVwbAs670L1vcVQi8j9HjnA==", - "dev": true - }, - "@types/node": { - "version": "12.12.37", - "resolved": "https://registry.npmjs.org/@types/node/-/node-12.12.37.tgz", - "integrity": "sha512-4mXKoDptrXAwZErQHrLzpe0FN/0Wmf5JRniSVIdwUrtDf9wnmEV1teCNLBo/TwuXhkK/bVegoEn/wmb+x0AuPg==", - "dev": true - }, - "@types/q": { - "version": "1.5.2", - "resolved": "https://registry.npmjs.org/@types/q/-/q-1.5.2.tgz", - "integrity": "sha512-ce5d3q03Ex0sy4R14722Rmt6MT07Ua+k4FwDfdcToYJcMKNtRVQvJ6JCAPdAmAnbRb6CsX6aYb9m96NGod9uTw==", - "dev": true - }, - "@types/selenium-webdriver": { - "version": "3.0.17", - "resolved": "https://registry.npmjs.org/@types/selenium-webdriver/-/selenium-webdriver-3.0.17.tgz", - "integrity": "sha512-tGomyEuzSC1H28y2zlW6XPCaDaXFaD6soTdb4GNdmte2qfHtrKqhy0ZFs4r/1hpazCfEZqeTSRLvSasmEx89uw==", - "dev": true - }, - "@types/source-list-map": { - "version": "0.1.2", - "resolved": "https://registry.npmjs.org/@types/source-list-map/-/source-list-map-0.1.2.tgz", - "integrity": "sha512-K5K+yml8LTo9bWJI/rECfIPrGgxdpeNbj+d53lwN4QjW1MCwlkhUms+gtdzigTeUyBr09+u8BwOIY3MXvHdcsA==", - "dev": true - }, - "@types/tern": { - "version": "0.23.3", - "resolved": "https://registry.npmjs.org/@types/tern/-/tern-0.23.3.tgz", - "integrity": "sha512-imDtS4TAoTcXk0g7u4kkWqedB3E4qpjXzCpD2LU5M5NAXHzCDsypyvXSaG7mM8DKYkCRa7tFp4tS/lp/Wo7Q3w==", - "requires": { - "@types/estree": "*" - } - }, - "@types/webpack-sources": { - "version": "0.1.7", - "resolved": "https://registry.npmjs.org/@types/webpack-sources/-/webpack-sources-0.1.7.tgz", - "integrity": "sha512-XyaHrJILjK1VHVC4aVlKsdNN5KBTwufMb43cQs+flGxtPAf/1Qwl8+Q0tp5BwEGaI8D6XT1L+9bSWXckgkjTLw==", - "dev": true, - "requires": { - "@types/node": "*", - "@types/source-list-map": "*", - "source-map": "^0.6.1" - }, - "dependencies": { - "source-map": { - "version": "0.6.1", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", - "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", - "dev": true - } - } - }, - "@webassemblyjs/ast": { - "version": "1.8.5", - "resolved": "https://registry.npmjs.org/@webassemblyjs/ast/-/ast-1.8.5.tgz", - "integrity": "sha512-aJMfngIZ65+t71C3y2nBBg5FFG0Okt9m0XEgWZ7Ywgn1oMAT8cNwx00Uv1cQyHtidq0Xn94R4TAywO+LCQ+ZAQ==", - "dev": true, - "requires": { - "@webassemblyjs/helper-module-context": "1.8.5", - "@webassemblyjs/helper-wasm-bytecode": "1.8.5", - "@webassemblyjs/wast-parser": "1.8.5" - } - }, - "@webassemblyjs/floating-point-hex-parser": { - "version": "1.8.5", - "resolved": "https://registry.npmjs.org/@webassemblyjs/floating-point-hex-parser/-/floating-point-hex-parser-1.8.5.tgz", - "integrity": "sha512-9p+79WHru1oqBh9ewP9zW95E3XAo+90oth7S5Re3eQnECGq59ly1Ri5tsIipKGpiStHsUYmY3zMLqtk3gTcOtQ==", - "dev": true - }, - "@webassemblyjs/helper-api-error": { - "version": "1.8.5", - "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-api-error/-/helper-api-error-1.8.5.tgz", - "integrity": "sha512-Za/tnzsvnqdaSPOUXHyKJ2XI7PDX64kWtURyGiJJZKVEdFOsdKUCPTNEVFZq3zJ2R0G5wc2PZ5gvdTRFgm81zA==", - "dev": true - }, - "@webassemblyjs/helper-buffer": { - "version": "1.8.5", - "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-buffer/-/helper-buffer-1.8.5.tgz", - "integrity": "sha512-Ri2R8nOS0U6G49Q86goFIPNgjyl6+oE1abW1pS84BuhP1Qcr5JqMwRFT3Ah3ADDDYGEgGs1iyb1DGX+kAi/c/Q==", - "dev": true - }, - "@webassemblyjs/helper-code-frame": { - "version": "1.8.5", - "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-code-frame/-/helper-code-frame-1.8.5.tgz", - "integrity": "sha512-VQAadSubZIhNpH46IR3yWO4kZZjMxN1opDrzePLdVKAZ+DFjkGD/rf4v1jap744uPVU6yjL/smZbRIIJTOUnKQ==", - "dev": true, - "requires": { - "@webassemblyjs/wast-printer": "1.8.5" - } - }, - "@webassemblyjs/helper-fsm": { - "version": "1.8.5", - "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-fsm/-/helper-fsm-1.8.5.tgz", - "integrity": "sha512-kRuX/saORcg8se/ft6Q2UbRpZwP4y7YrWsLXPbbmtepKr22i8Z4O3V5QE9DbZK908dh5Xya4Un57SDIKwB9eow==", - "dev": true - }, - "@webassemblyjs/helper-module-context": { - "version": "1.8.5", - "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-module-context/-/helper-module-context-1.8.5.tgz", - "integrity": "sha512-/O1B236mN7UNEU4t9X7Pj38i4VoU8CcMHyy3l2cV/kIF4U5KoHXDVqcDuOs1ltkac90IM4vZdHc52t1x8Yfs3g==", - "dev": true, - "requires": { - "@webassemblyjs/ast": "1.8.5", - "mamacro": "^0.0.3" - } - }, - "@webassemblyjs/helper-wasm-bytecode": { - "version": "1.8.5", - "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-wasm-bytecode/-/helper-wasm-bytecode-1.8.5.tgz", - "integrity": "sha512-Cu4YMYG3Ddl72CbmpjU/wbP6SACcOPVbHN1dI4VJNJVgFwaKf1ppeFJrwydOG3NDHxVGuCfPlLZNyEdIYlQ6QQ==", - "dev": true - }, - "@webassemblyjs/helper-wasm-section": { - "version": "1.8.5", - "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-wasm-section/-/helper-wasm-section-1.8.5.tgz", - "integrity": "sha512-VV083zwR+VTrIWWtgIUpqfvVdK4ff38loRmrdDBgBT8ADXYsEZ5mPQ4Nde90N3UYatHdYoDIFb7oHzMncI02tA==", - "dev": true, - "requires": { - "@webassemblyjs/ast": "1.8.5", - "@webassemblyjs/helper-buffer": "1.8.5", - "@webassemblyjs/helper-wasm-bytecode": "1.8.5", - "@webassemblyjs/wasm-gen": "1.8.5" - } - }, - "@webassemblyjs/ieee754": { - "version": "1.8.5", - "resolved": "https://registry.npmjs.org/@webassemblyjs/ieee754/-/ieee754-1.8.5.tgz", - "integrity": "sha512-aaCvQYrvKbY/n6wKHb/ylAJr27GglahUO89CcGXMItrOBqRarUMxWLJgxm9PJNuKULwN5n1csT9bYoMeZOGF3g==", - "dev": true, - "requires": { - "@xtuc/ieee754": "^1.2.0" - } - }, - "@webassemblyjs/leb128": { - "version": "1.8.5", - "resolved": "https://registry.npmjs.org/@webassemblyjs/leb128/-/leb128-1.8.5.tgz", - "integrity": "sha512-plYUuUwleLIziknvlP8VpTgO4kqNaH57Y3JnNa6DLpu/sGcP6hbVdfdX5aHAV716pQBKrfuU26BJK29qY37J7A==", - "dev": true, - "requires": { - "@xtuc/long": "4.2.2" - } - }, - "@webassemblyjs/utf8": { - "version": "1.8.5", - "resolved": "https://registry.npmjs.org/@webassemblyjs/utf8/-/utf8-1.8.5.tgz", - "integrity": "sha512-U7zgftmQriw37tfD934UNInokz6yTmn29inT2cAetAsaU9YeVCveWEwhKL1Mg4yS7q//NGdzy79nlXh3bT8Kjw==", - "dev": true - }, - "@webassemblyjs/wasm-edit": { - "version": "1.8.5", - "resolved": "https://registry.npmjs.org/@webassemblyjs/wasm-edit/-/wasm-edit-1.8.5.tgz", - "integrity": "sha512-A41EMy8MWw5yvqj7MQzkDjU29K7UJq1VrX2vWLzfpRHt3ISftOXqrtojn7nlPsZ9Ijhp5NwuODuycSvfAO/26Q==", - "dev": true, - "requires": { - "@webassemblyjs/ast": "1.8.5", - "@webassemblyjs/helper-buffer": "1.8.5", - "@webassemblyjs/helper-wasm-bytecode": "1.8.5", - "@webassemblyjs/helper-wasm-section": "1.8.5", - "@webassemblyjs/wasm-gen": "1.8.5", - "@webassemblyjs/wasm-opt": "1.8.5", - "@webassemblyjs/wasm-parser": "1.8.5", - "@webassemblyjs/wast-printer": "1.8.5" - } - }, - "@webassemblyjs/wasm-gen": { - "version": "1.8.5", - "resolved": "https://registry.npmjs.org/@webassemblyjs/wasm-gen/-/wasm-gen-1.8.5.tgz", - "integrity": "sha512-BCZBT0LURC0CXDzj5FXSc2FPTsxwp3nWcqXQdOZE4U7h7i8FqtFK5Egia6f9raQLpEKT1VL7zr4r3+QX6zArWg==", - "dev": true, - "requires": { - "@webassemblyjs/ast": "1.8.5", - "@webassemblyjs/helper-wasm-bytecode": "1.8.5", - "@webassemblyjs/ieee754": "1.8.5", - "@webassemblyjs/leb128": "1.8.5", - "@webassemblyjs/utf8": "1.8.5" - } - }, - "@webassemblyjs/wasm-opt": { - "version": "1.8.5", - "resolved": "https://registry.npmjs.org/@webassemblyjs/wasm-opt/-/wasm-opt-1.8.5.tgz", - "integrity": "sha512-HKo2mO/Uh9A6ojzu7cjslGaHaUU14LdLbGEKqTR7PBKwT6LdPtLLh9fPY33rmr5wcOMrsWDbbdCHq4hQUdd37Q==", - "dev": true, - "requires": { - "@webassemblyjs/ast": "1.8.5", - "@webassemblyjs/helper-buffer": "1.8.5", - "@webassemblyjs/wasm-gen": "1.8.5", - "@webassemblyjs/wasm-parser": "1.8.5" - } - }, - "@webassemblyjs/wasm-parser": { - "version": "1.8.5", - "resolved": "https://registry.npmjs.org/@webassemblyjs/wasm-parser/-/wasm-parser-1.8.5.tgz", - "integrity": "sha512-pi0SYE9T6tfcMkthwcgCpL0cM9nRYr6/6fjgDtL6q/ZqKHdMWvxitRi5JcZ7RI4SNJJYnYNaWy5UUrHQy998lw==", - "dev": true, - "requires": { - "@webassemblyjs/ast": "1.8.5", - "@webassemblyjs/helper-api-error": "1.8.5", - "@webassemblyjs/helper-wasm-bytecode": "1.8.5", - "@webassemblyjs/ieee754": "1.8.5", - "@webassemblyjs/leb128": "1.8.5", - "@webassemblyjs/utf8": "1.8.5" - } - }, - "@webassemblyjs/wast-parser": { - "version": "1.8.5", - "resolved": "https://registry.npmjs.org/@webassemblyjs/wast-parser/-/wast-parser-1.8.5.tgz", - "integrity": "sha512-daXC1FyKWHF1i11obK086QRlsMsY4+tIOKgBqI1lxAnkp9xe9YMcgOxm9kLe+ttjs5aWV2KKE1TWJCN57/Btsg==", - "dev": true, - "requires": { - "@webassemblyjs/ast": "1.8.5", - "@webassemblyjs/floating-point-hex-parser": "1.8.5", - "@webassemblyjs/helper-api-error": "1.8.5", - "@webassemblyjs/helper-code-frame": "1.8.5", - "@webassemblyjs/helper-fsm": "1.8.5", - "@xtuc/long": "4.2.2" - } - }, - "@webassemblyjs/wast-printer": { - "version": "1.8.5", - "resolved": "https://registry.npmjs.org/@webassemblyjs/wast-printer/-/wast-printer-1.8.5.tgz", - "integrity": "sha512-w0U0pD4EhlnvRyeJzBqaVSJAo9w/ce7/WPogeXLzGkO6hzhr4GnQIZ4W4uUt5b9ooAaXPtnXlj0gzsXEOUNYMg==", - "dev": true, - "requires": { - "@webassemblyjs/ast": "1.8.5", - "@webassemblyjs/wast-parser": "1.8.5", - "@xtuc/long": "4.2.2" - } - }, - "@xtuc/ieee754": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/@xtuc/ieee754/-/ieee754-1.2.0.tgz", - "integrity": "sha512-DX8nKgqcGwsc0eJSqYt5lwP4DH5FlHnmuWWBRy7X0NcaGR0ZtuyeESgMwTYVEtxmsNGY+qit4QYT/MIYTOTPeA==", - "dev": true - }, - "@xtuc/long": { - "version": "4.2.2", - "resolved": "https://registry.npmjs.org/@xtuc/long/-/long-4.2.2.tgz", - "integrity": "sha512-NuHqBY1PB/D8xU6s/thBgOAiAP7HOYDQ32+BFZILJ8ivkUkAHQnWfn6WhL79Owj1qmUnoN/YPhktdIoucipkAQ==", - "dev": true - }, - "@yarnpkg/lockfile": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/@yarnpkg/lockfile/-/lockfile-1.1.0.tgz", - "integrity": "sha512-GpSwvyXOcOOlV70vbnzjj4fW5xW/FdUF6nQEt1ENy7m4ZCczi1+/buVUPAqmGfqznsORNFzUMjctTIp8a9tuCQ==", - "dev": true - }, - "JSONStream": { - "version": "1.3.5", - "resolved": "https://registry.npmjs.org/JSONStream/-/JSONStream-1.3.5.tgz", - "integrity": "sha512-E+iruNOY8VV9s4JEbe1aNEm6MiszPRr/UfcHMz0TQh1BXSxHK+ASV1R6W4HpjBhSeS+54PIsAMCBmwD06LLsqQ==", - "dev": true, - "requires": { - "jsonparse": "^1.2.0", - "through": ">=2.2.7 <3" - } - }, - "accepts": { - "version": "1.3.7", - "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.7.tgz", - "integrity": "sha512-Il80Qs2WjYlJIBNzNkK6KYqlVMTbZLXgHx2oT0pU/fjRHyEp+PEfEPY0R3WCwAGVOtauxh1hOxNgIf5bv7dQpA==", - "dev": true, - "requires": { - "mime-types": "~2.1.24", - "negotiator": "0.6.2" - } - }, - "acorn": { - "version": "6.4.1", - "resolved": "https://registry.npmjs.org/acorn/-/acorn-6.4.1.tgz", - "integrity": "sha512-ZVA9k326Nwrj3Cj9jlh3wGFutC2ZornPNARZwsNYqQYgN0EsV2d53w5RN/co65Ohn4sUAUtb1rSUAOD6XN9idA==", - "dev": true - }, - "adm-zip": { - "version": "0.4.14", - "resolved": "https://registry.npmjs.org/adm-zip/-/adm-zip-0.4.14.tgz", - "integrity": "sha512-/9aQCnQHF+0IiCl0qhXoK7qs//SwYE7zX8lsr/DNk1BRAHYxeLZPL4pguwK29gUEqasYQjqPtEpDRSWEkdHn9g==", - "dev": true - }, - "agent-base": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-4.3.0.tgz", - "integrity": "sha512-salcGninV0nPrwpGNn4VTXBb1SOuXQBiqbrNXoeizJsHrsL6ERFM2Ne3JUSBWRE6aeNJI2ROP/WEEIDUiDe3cg==", - "dev": true, - "requires": { - "es6-promisify": "^5.0.0" - } - }, - "agentkeepalive": { - "version": "3.5.2", - "resolved": "https://registry.npmjs.org/agentkeepalive/-/agentkeepalive-3.5.2.tgz", - "integrity": "sha512-e0L/HNe6qkQ7H19kTlRRqUibEAwDK5AFk6y3PtMsuut2VAH6+Q4xZml1tNDJD7kSAyqmbG/K08K5WEJYtUrSlQ==", - "dev": true, - "requires": { - "humanize-ms": "^1.2.1" - } - }, - "aggregate-error": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/aggregate-error/-/aggregate-error-3.0.1.tgz", - "integrity": "sha512-quoaXsZ9/BLNae5yiNoUz+Nhkwz83GhWwtYFglcjEQB2NDHCIpApbqXxIFnm4Pq/Nvhrsq5sYJFyohrrxnTGAA==", - "dev": true, - "requires": { - "clean-stack": "^2.0.0", - "indent-string": "^4.0.0" - } - }, - "ajv": { - "version": "6.10.2", - "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.10.2.tgz", - "integrity": "sha512-TXtUUEYHuaTEbLZWIKUr5pmBuhDLy+8KYtPYdcV8qC+pOZL+NKqYwvWSRrVXHn+ZmRRAu8vJTAznH7Oag6RVRw==", - "dev": true, - "requires": { - "fast-deep-equal": "^2.0.1", - "fast-json-stable-stringify": "^2.0.0", - "json-schema-traverse": "^0.4.1", - "uri-js": "^4.2.2" - } - }, - "ajv-errors": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/ajv-errors/-/ajv-errors-1.0.1.tgz", - "integrity": "sha512-DCRfO/4nQ+89p/RK43i8Ezd41EqdGIU4ld7nGF8OQ14oc/we5rEntLCUa7+jrn3nn83BosfwZA0wb4pon2o8iQ==", - "dev": true - }, - "ajv-keywords": { - "version": "3.4.1", - "resolved": "https://registry.npmjs.org/ajv-keywords/-/ajv-keywords-3.4.1.tgz", - "integrity": "sha512-RO1ibKvd27e6FEShVFfPALuHI3WjSVNeK5FIsmme/LYRNxjKuNj+Dt7bucLa6NdSv3JcVTyMlm9kGR84z1XpaQ==", - "dev": true - }, - "alphanum-sort": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/alphanum-sort/-/alphanum-sort-1.0.2.tgz", - "integrity": "sha1-l6ERlkmyEa0zaR2fn0hqjsn74KM=", - "dev": true - }, - "amator": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/amator/-/amator-1.1.0.tgz", - "integrity": "sha1-CMa2C8k67Cthu/wMTWd9MDI8wPE=", - "requires": { - "bezier-easing": "^2.0.3" - } - }, - "angular-split": { - "version": "3.0.3", - "resolved": "https://registry.npmjs.org/angular-split/-/angular-split-3.0.3.tgz", - "integrity": "sha512-5m5mCeISmytXnxKijo9PQDQviiDHNCjt/beTVeYeLTGZJbdB5AEk4lslxOV8b7zM7tcv1YN+8jW3xNK1XcApsA==", - "requires": { - "tslib": "^1.9.0" - } - }, - "ansi-colors": { - "version": "3.2.4", - "resolved": "https://registry.npmjs.org/ansi-colors/-/ansi-colors-3.2.4.tgz", - "integrity": "sha512-hHUXGagefjN2iRrID63xckIvotOXOojhQKWIPUZ4mNUZ9nLZW+7FMNoE1lOkEhNWYsx/7ysGIuJYCiMAA9FnrA==", - "dev": true - }, - "ansi-escapes": { - "version": "4.3.1", - "resolved": "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-4.3.1.tgz", - "integrity": "sha512-JWF7ocqNrp8u9oqpgV+wH5ftbt+cfvv+PTjOvKLT3AdYly/LmORARfEVT1iyjwN+4MqE5UmVKoAdIBqeoCHgLA==", - "dev": true, - "requires": { - "type-fest": "^0.11.0" - } - }, - "ansi-html": { - "version": "0.0.7", - "resolved": "https://registry.npmjs.org/ansi-html/-/ansi-html-0.0.7.tgz", - "integrity": "sha1-gTWEAhliqenm/QOflA0S9WynhZ4=", - "dev": true - }, - "ansi-regex": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz", - "integrity": "sha1-w7M6te42DYbg5ijwRorn7yfWVN8=", - "dev": true - }, - "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" - } - }, - "anymatch": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.1.tgz", - "integrity": "sha512-mM8522psRCqzV+6LhomX5wgp25YVibjh8Wj23I5RPkPppSVSjyKD2A2mBJmWGa+KN7f2D6LNh9jkBCeyLktzjg==", - "dev": true, - "requires": { - "normalize-path": "^3.0.0", - "picomatch": "^2.0.4" - } - }, - "app-root-path": { - "version": "2.2.1", - "resolved": "https://registry.npmjs.org/app-root-path/-/app-root-path-2.2.1.tgz", - "integrity": "sha512-91IFKeKk7FjfmezPKkwtaRvSpnUc4gDwPAjA1YZ9Gn0q0PPeW+vbeUsZuyDwjI7+QTHhcLen2v25fi/AmhvbJA==", - "dev": true - }, - "append-transform": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/append-transform/-/append-transform-1.0.0.tgz", - "integrity": "sha512-P009oYkeHyU742iSZJzZZywj4QRJdnTWffaKuJQLablCZ1uz6/cW4yaRgcDaoQ+uwOxxnt0gRUcwfsNP2ri0gw==", - "dev": true, - "requires": { - "default-require-extensions": "^2.0.0" - } - }, - "aproba": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/aproba/-/aproba-1.2.0.tgz", - "integrity": "sha512-Y9J6ZjXtoYh8RnXVCMOU/ttDmk1aBjunq9vO0ta5x85WDQiQfUF9sIPBITdbiiIVcBo03Hi3jMxigBtsddlXRw==", - "dev": true - }, - "arg": { - "version": "4.1.3", - "resolved": "https://registry.npmjs.org/arg/-/arg-4.1.3.tgz", - "integrity": "sha512-58S9QDqG0Xx27YwPSt9fJxivjYl432YCwfDMfZ+71RAqUrZef7LrKQZ3LHLOwCS4FLNBplP533Zx895SeOCHvA==", - "dev": true - }, - "argparse": { - "version": "1.0.10", - "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.10.tgz", - "integrity": "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==", - "dev": true, - "requires": { - "sprintf-js": "~1.0.2" - } - }, - "aria-query": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/aria-query/-/aria-query-3.0.0.tgz", - "integrity": "sha1-ZbP8wcoRVajJrmTW7uKX8V1RM8w=", - "dev": true, - "requires": { - "ast-types-flow": "0.0.7", - "commander": "^2.11.0" - } - }, - "arr-diff": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/arr-diff/-/arr-diff-4.0.0.tgz", - "integrity": "sha1-1kYQdP6/7HHn4VI1dhoyml3HxSA=", - "dev": true - }, - "arr-flatten": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/arr-flatten/-/arr-flatten-1.1.0.tgz", - "integrity": "sha512-L3hKV5R/p5o81R7O02IGnwpDmkp6E982XhtbuwSe3O4qOtMMMtodicASA1Cny2U+aCXcNpml+m4dPsvsJ3jatg==", - "dev": true - }, - "arr-union": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/arr-union/-/arr-union-3.1.0.tgz", - "integrity": "sha1-45sJrqne+Gao8gbiiK9jkZuuOcQ=", - "dev": true - }, - "array-flatten": { - "version": "2.1.2", - "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-2.1.2.tgz", - "integrity": "sha512-hNfzcOV8W4NdualtqBFPyVO+54DSJuZGY9qT4pRroB6S9e3iiido2ISIC5h9R2sPJ8H3FHCIiEnsv1lPXO3KtQ==", - "dev": true - }, - "array-union": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/array-union/-/array-union-1.0.2.tgz", - "integrity": "sha1-mjRBDk9OPaI96jdb5b5w8kd47Dk=", - "dev": true, - "requires": { - "array-uniq": "^1.0.1" - } - }, - "array-uniq": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/array-uniq/-/array-uniq-1.0.3.tgz", - "integrity": "sha1-r2rId6Jcx/dOBYiUdThY39sk/bY=", - "dev": true - }, - "array-unique": { - "version": "0.3.2", - "resolved": "https://registry.npmjs.org/array-unique/-/array-unique-0.3.2.tgz", - "integrity": "sha1-qJS3XUvE9s1nnvMkSp/Y9Gri1Cg=", - "dev": true - }, - "arrify": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/arrify/-/arrify-1.0.1.tgz", - "integrity": "sha1-iYUI2iIm84DfkEcoRWhJwVAaSw0=", - "dev": true - }, - "asap": { - "version": "2.0.6", - "resolved": "https://registry.npmjs.org/asap/-/asap-2.0.6.tgz", - "integrity": "sha1-5QNHYR1+aQlDIIu9r+vLwvuGbUY=", - "dev": true - }, - "asn1": { - "version": "0.2.4", - "resolved": "https://registry.npmjs.org/asn1/-/asn1-0.2.4.tgz", - "integrity": "sha512-jxwzQpLQjSmWXgwaCZE9Nz+glAG01yF1QnWgbhGwHI5A6FRIEY6IVqtHhIepHqI7/kyEyQEagBC5mBEFlIYvdg==", - "dev": true, - "requires": { - "safer-buffer": "~2.1.0" - } - }, - "asn1.js": { - "version": "4.10.1", - "resolved": "https://registry.npmjs.org/asn1.js/-/asn1.js-4.10.1.tgz", - "integrity": "sha512-p32cOF5q0Zqs9uBiONKYLm6BClCoBCM5O9JfeUSlnQLBTxYdTK+pW+nXflm8UkKd2UYlEbYz5qEi0JuZR9ckSw==", - "dev": true, - "requires": { - "bn.js": "^4.0.0", - "inherits": "^2.0.1", - "minimalistic-assert": "^1.0.0" - }, - "dependencies": { - "bn.js": { - "version": "4.11.8", - "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.11.8.tgz", - "integrity": "sha512-ItfYfPLkWHUjckQCk8xC+LwxgK8NYcXywGigJgSwOP8Y2iyWT4f2vsZnoOXTTbo+o5yXmIUJ4gn5538SO5S3gA==", - "dev": true - } - } - }, - "assert": { - "version": "1.5.0", - "resolved": "https://registry.npmjs.org/assert/-/assert-1.5.0.tgz", - "integrity": "sha512-EDsgawzwoun2CZkCgtxJbv392v4nbk9XDD06zI+kQYoBM/3RBWLlEyJARDOmhAAosBjWACEkKL6S+lIZtcAubA==", - "dev": true, - "requires": { - "object-assign": "^4.1.1", - "util": "0.10.3" - }, - "dependencies": { - "inherits": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.1.tgz", - "integrity": "sha1-sX0I0ya0Qj5Wjv9xn5GwscvfafE=", - "dev": true - }, - "util": { - "version": "0.10.3", - "resolved": "https://registry.npmjs.org/util/-/util-0.10.3.tgz", - "integrity": "sha1-evsa/lCAUkZInj23/g7TeTNqwPk=", - "dev": true, - "requires": { - "inherits": "2.0.1" - } - } - } - }, - "assert-plus": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/assert-plus/-/assert-plus-1.0.0.tgz", - "integrity": "sha1-8S4PPF13sLHN2RRpQuTpbB5N1SU=", - "dev": true - }, - "assign-symbols": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/assign-symbols/-/assign-symbols-1.0.0.tgz", - "integrity": "sha1-WWZ/QfrdTyDMvCu5a41Pf3jsA2c=", - "dev": true - }, - "ast-types-flow": { - "version": "0.0.7", - "resolved": "https://registry.npmjs.org/ast-types-flow/-/ast-types-flow-0.0.7.tgz", - "integrity": "sha1-9wtzXGvKGlycItmCw+Oef+ujva0=", - "dev": true - }, - "async": { - "version": "2.6.4", - "resolved": "https://registry.npmjs.org/async/-/async-2.6.4.tgz", - "integrity": "sha512-mzo5dfJYwAn29PeiJ0zvwTo04zj8HDJj0Mn8TD7sno7q12prdbnasKJHhkm2c1LgrhlJ0teaea8860oxi51mGA==", - "dev": true, - "requires": { - "lodash": "^4.17.14" - } - }, - "async-each": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/async-each/-/async-each-1.0.3.tgz", - "integrity": "sha512-z/WhQ5FPySLdvREByI2vZiTWwCnF0moMJ1hK9YQwDTHKh6I7/uSckMetoRGb5UBZPC1z0jlw+n/XCgjeH7y1AQ==", - "dev": true - }, - "async-limiter": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/async-limiter/-/async-limiter-1.0.1.tgz", - "integrity": "sha512-csOlWGAcRFJaI6m+F2WKdnMKr4HhdhFVBk0H/QbJFMCr+uO2kwohwXQPxw/9OCxp05r5ghVBFSyioixx3gfkNQ==", - "dev": true - }, - "asynckit": { - "version": "0.4.0", - "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", - "integrity": "sha1-x57Zf380y48robyXkLzDZkdLS3k=", - "dev": true - }, - "atob": { - "version": "2.1.2", - "resolved": "https://registry.npmjs.org/atob/-/atob-2.1.2.tgz", - "integrity": "sha512-Wm6ukoaOGJi/73p/cl2GvLjTI5JM1k/O14isD73YML8StrH/7/lRFgmg8nICZgD3bZZvjwCGxtMOD3wWNAu8cg==", - "dev": true - }, - "autoprefixer": { - "version": "9.7.4", - "resolved": "https://registry.npmjs.org/autoprefixer/-/autoprefixer-9.7.4.tgz", - "integrity": "sha512-g0Ya30YrMBAEZk60lp+qfX5YQllG+S5W3GYCFvyHTvhOki0AEQJLPEcIuGRsqVwLi8FvXPVtwTGhfr38hVpm0g==", - "dev": true, - "requires": { - "browserslist": "^4.8.3", - "caniuse-lite": "^1.0.30001020", - "chalk": "^2.4.2", - "normalize-range": "^0.1.2", - "num2fraction": "^1.2.2", - "postcss": "^7.0.26", - "postcss-value-parser": "^4.0.2" - } - }, - "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=", - "dev": true - }, - "aws4": { - "version": "1.9.1", - "resolved": "https://registry.npmjs.org/aws4/-/aws4-1.9.1.tgz", - "integrity": "sha512-wMHVg2EOHaMRxbzgFJ9gtjOOCrI80OHLG14rxi28XwOW8ux6IiEbRCGGGqCtdAIg4FQCbW20k9RsT4y3gJlFug==", - "dev": true - }, - "axobject-query": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/axobject-query/-/axobject-query-2.0.2.tgz", - "integrity": "sha512-MCeek8ZH7hKyO1rWUbKNQBbl4l2eY0ntk7OGi+q0RlafrCnfPxC06WZA+uebCfmYp4mNU9jRBP1AhGyf8+W3ww==", - "dev": true, - "requires": { - "ast-types-flow": "0.0.7" - } - }, - "babel-loader": { - "version": "8.0.6", - "resolved": "https://registry.npmjs.org/babel-loader/-/babel-loader-8.0.6.tgz", - "integrity": "sha512-4BmWKtBOBm13uoUwd08UwjZlaw3O9GWf456R9j+5YykFZ6LUIjIKLc0zEZf+hauxPOJs96C8k6FvYD09vWzhYw==", - "dev": true, - "requires": { - "find-cache-dir": "^2.0.0", - "loader-utils": "^1.0.2", - "mkdirp": "^0.5.1", - "pify": "^4.0.1" - }, - "dependencies": { - "find-cache-dir": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/find-cache-dir/-/find-cache-dir-2.1.0.tgz", - "integrity": "sha512-Tq6PixE0w/VMFfCgbONnkiQIVol/JJL7nRMi20fqzA4NRs9AfeqMGeRdPi3wIhYkxjeBaWh2rxwapn5Tu3IqOQ==", - "dev": true, - "requires": { - "commondir": "^1.0.1", - "make-dir": "^2.0.0", - "pkg-dir": "^3.0.0" - } - }, - "json5": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/json5/-/json5-1.0.1.tgz", - "integrity": "sha512-aKS4WQjPenRxiQsC93MNfjx+nbF4PAdYzmd/1JIj8HYzqfbu86beTuNgXDzPknWk0n0uARlyewZo4s++ES36Ow==", - "dev": true, - "requires": { - "minimist": "^1.2.0" - } - }, - "loader-utils": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/loader-utils/-/loader-utils-1.4.0.tgz", - "integrity": "sha512-qH0WSMBtn/oHuwjy/NucEgbx5dbxxnxup9s4PVXJUDHZBQY+s0NWA9rJf53RBnQZxfch7euUui7hpoAPvALZdA==", - "dev": true, - "requires": { - "big.js": "^5.2.2", - "emojis-list": "^3.0.0", - "json5": "^1.0.1" - } - } - } - }, - "babel-plugin-dynamic-import-node": { - "version": "2.3.3", - "resolved": "https://registry.npmjs.org/babel-plugin-dynamic-import-node/-/babel-plugin-dynamic-import-node-2.3.3.tgz", - "integrity": "sha512-jZVI+s9Zg3IqA/kdi0i6UDCybUI3aSBLnglhYbSSjKlV7yF1F/5LWv8MakQmvYpnbJDS6fcBL2KzHSxNCMtWSQ==", - "dev": true, - "requires": { - "object.assign": "^4.1.0" - } - }, - "balanced-match": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.0.tgz", - "integrity": "sha1-ibTRmasr7kneFk6gK4nORi1xt2c=", - "dev": true - }, - "base": { - "version": "0.11.2", - "resolved": "https://registry.npmjs.org/base/-/base-0.11.2.tgz", - "integrity": "sha512-5T6P4xPgpp0YDFvSWwEZ4NoE3aM4QBQXDzmVbraCkFj8zHM+mba8SyqB5DbZWyR7mYHo6Y7BdQo3MoA4m0TeQg==", - "dev": true, - "requires": { - "cache-base": "^1.0.1", - "class-utils": "^0.3.5", - "component-emitter": "^1.2.1", - "define-property": "^1.0.0", - "isobject": "^3.0.1", - "mixin-deep": "^1.2.0", - "pascalcase": "^0.1.1" - }, - "dependencies": { - "define-property": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/define-property/-/define-property-1.0.0.tgz", - "integrity": "sha1-dp66rz9KY6rTr56NMEybvnm/sOY=", - "dev": true, - "requires": { - "is-descriptor": "^1.0.0" - } - }, - "is-accessor-descriptor": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-1.0.0.tgz", - "integrity": "sha512-m5hnHTkcVsPfqx3AKlyttIPb7J+XykHvJP2B9bZDjlhLIoEq4XoK64Vg7boZlVWYK6LUY94dYPEE7Lh0ZkZKcQ==", - "dev": true, - "requires": { - "kind-of": "^6.0.0" - } - }, - "is-data-descriptor": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-1.0.0.tgz", - "integrity": "sha512-jbRXy1FmtAoCjQkVmIVYwuuqDFUbaOeDjmed1tOGPrsMhtJA4rD9tkgA0F1qJ3gRFRXcHYVkdeaP50Q5rE/jLQ==", - "dev": true, - "requires": { - "kind-of": "^6.0.0" - } - }, - "is-descriptor": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-1.0.2.tgz", - "integrity": "sha512-2eis5WqQGV7peooDyLmNEPUrps9+SXX5c9pL3xEB+4e9HnGuDa7mB7kHxHw4CbqS9k1T2hOH3miL8n8WtiYVtg==", - "dev": true, - "requires": { - "is-accessor-descriptor": "^1.0.0", - "is-data-descriptor": "^1.0.0", - "kind-of": "^6.0.2" - } - } - } - }, - "base64-js": { - "version": "1.3.1", - "resolved": "https://registry.npmjs.org/base64-js/-/base64-js-1.3.1.tgz", - "integrity": "sha512-mLQ4i2QO1ytvGWFWmcngKO//JXAQueZvwEKtjgQFM4jIK0kU+ytMfplL8j+n5mspOfjHwoAg+9yhb7BwAHm36g==", - "dev": true - }, - "base64id": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/base64id/-/base64id-2.0.0.tgz", - "integrity": "sha512-lGe34o6EHj9y3Kts9R4ZYs/Gr+6N7MCaMlIFA3F1R2O5/m7K06AxfSeO5530PEERE6/WyEg3lsuyw4GHlPZHog==", - "dev": true - }, - "batch": { - "version": "0.6.1", - "resolved": "https://registry.npmjs.org/batch/-/batch-0.6.1.tgz", - "integrity": "sha1-3DQxT05nkxgJP8dgJyUl+UvyXBY=", - "dev": true - }, - "bcrypt-pbkdf": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/bcrypt-pbkdf/-/bcrypt-pbkdf-1.0.2.tgz", - "integrity": "sha1-pDAdOJtqQ/m2f/PKEaP2Y342Dp4=", - "dev": true, - "requires": { - "tweetnacl": "^0.14.3" - } - }, - "bezier-easing": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/bezier-easing/-/bezier-easing-2.1.0.tgz", - "integrity": "sha1-wE3+i5JtbsrKGBPWn/F5t8ICXYY=" - }, - "big.js": { - "version": "5.2.2", - "resolved": "https://registry.npmjs.org/big.js/-/big.js-5.2.2.tgz", - "integrity": "sha512-vyL2OymJxmarO8gxMr0mhChsO9QGwhynfuu4+MHTAW6czfq9humCB7rKpUjDd9YUiDPU4mzpyupFSvOClAwbmQ==", - "dev": true - }, - "binary-extensions": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.0.0.tgz", - "integrity": "sha512-Phlt0plgpIIBOGTT/ehfFnbNlfsDEiqmzE2KRXoX1bLIlir4X/MR+zSyBEkL05ffWgnRSf/DXv+WrUAVr93/ow==", - "dev": true - }, - "bindings": { - "version": "1.5.0", - "resolved": "https://registry.npmjs.org/bindings/-/bindings-1.5.0.tgz", - "integrity": "sha512-p2q/t/mhvuOj/UeLlV6566GD/guowlr0hHxClI0W9m7MWYkL1F0hLo+0Aexs9HSPCtR1SXQ0TD3MMKrXZajbiQ==", - "dev": true, - "optional": true, - "requires": { - "file-uri-to-path": "1.0.0" - } - }, - "blocking-proxy": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/blocking-proxy/-/blocking-proxy-1.0.1.tgz", - "integrity": "sha512-KE8NFMZr3mN2E0HcvCgRtX7DjhiIQrwle+nSVJVC/yqFb9+xznHl2ZcoBp2L9qzkI4t4cBFJ1efXF8Dwi132RA==", - "dev": true, - "requires": { - "minimist": "^1.2.0" - } - }, - "bluebird": { - "version": "3.7.2", - "resolved": "https://registry.npmjs.org/bluebird/-/bluebird-3.7.2.tgz", - "integrity": "sha512-XpNj6GDQzdfW+r2Wnn7xiSAd7TM3jzkxGXBGTtWKuSXv1xUV+azxAm8jdWZN06QTQk+2N2XB9jRDkvbmQmcRtg==", - "dev": true - }, - "bn.js": { - "version": "5.1.1", - "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-5.1.1.tgz", - "integrity": "sha512-IUTD/REb78Z2eodka1QZyyEk66pciRcP6Sroka0aI3tG/iwIdYLrBD62RsubR7vqdt3WyX8p4jxeatzmRSphtA==", - "dev": true - }, - "body-parser": { - "version": "1.19.0", - "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.19.0.tgz", - "integrity": "sha512-dhEPs72UPbDnAQJ9ZKMNTP6ptJaionhP5cBb541nXPlW60Jepo9RV/a4fX4XWW9CuFNK22krhrj1+rgzifNCsw==", - "dev": true, - "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": { - "bytes": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.0.tgz", - "integrity": "sha512-zauLjrfCG+xvoyaqLoV8bLVXXNGC4JqlxFCutSDWA6fJrTo2ZuvLYTqZ7aHBLZSMOopbzwv8f+wZcVzfVTI2Dg==", - "dev": true - }, - "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 - }, - "qs": { - "version": "6.7.0", - "resolved": "https://registry.npmjs.org/qs/-/qs-6.7.0.tgz", - "integrity": "sha512-VCdBRNFTX1fyE7Nb6FYoURo/SPe62QCaAyzJvUjwRaIsc+NePBEniHlvxFmmX56+HZphIGtV0XeCirBtpDrTyQ==", - "dev": true - } - } - }, - "bonjour": { - "version": "3.5.0", - "resolved": "https://registry.npmjs.org/bonjour/-/bonjour-3.5.0.tgz", - "integrity": "sha1-jokKGD2O6aI5OzhExpGkK897yfU=", - "dev": true, - "requires": { - "array-flatten": "^2.1.0", - "deep-equal": "^1.0.1", - "dns-equal": "^1.0.0", - "dns-txt": "^2.0.2", - "multicast-dns": "^6.0.1", - "multicast-dns-service-types": "^1.1.0" - } - }, - "boolbase": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/boolbase/-/boolbase-1.0.0.tgz", - "integrity": "sha1-aN/1++YMUes3cl6p4+0xDcwed24=", - "dev": true - }, - "brace-expansion": { - "version": "1.1.11", - "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", - "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", - "dev": true, - "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" - } - }, - "brorand": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/brorand/-/brorand-1.1.0.tgz", - "integrity": "sha1-EsJe/kCkXjwyPrhnWgoM5XsiNx8=", - "dev": true - }, - "browserify-aes": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/browserify-aes/-/browserify-aes-1.2.0.tgz", - "integrity": "sha512-+7CHXqGuspUn/Sl5aO7Ea0xWGAtETPXNSAjHo48JfLdPWcMng33Xe4znFvQweqc/uzk5zSOI3H52CYnjCfb5hA==", - "dev": true, - "requires": { - "buffer-xor": "^1.0.3", - "cipher-base": "^1.0.0", - "create-hash": "^1.1.0", - "evp_bytestokey": "^1.0.3", - "inherits": "^2.0.1", - "safe-buffer": "^5.0.1" - } - }, - "browserify-cipher": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/browserify-cipher/-/browserify-cipher-1.0.1.tgz", - "integrity": "sha512-sPhkz0ARKbf4rRQt2hTpAHqn47X3llLkUGn+xEJzLjwY8LRs2p0v7ljvI5EyoRO/mexrNunNECisZs+gw2zz1w==", - "dev": true, - "requires": { - "browserify-aes": "^1.0.4", - "browserify-des": "^1.0.0", - "evp_bytestokey": "^1.0.0" - } - }, - "browserify-des": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/browserify-des/-/browserify-des-1.0.2.tgz", - "integrity": "sha512-BioO1xf3hFwz4kc6iBhI3ieDFompMhrMlnDFC4/0/vd5MokpuAc3R+LYbwTA9A5Yc9pq9UYPqffKpW2ObuwX5A==", - "dev": true, - "requires": { - "cipher-base": "^1.0.1", - "des.js": "^1.0.0", - "inherits": "^2.0.1", - "safe-buffer": "^5.1.2" - } - }, - "browserify-rsa": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/browserify-rsa/-/browserify-rsa-4.0.1.tgz", - "integrity": "sha1-IeCr+vbyApzy+vsTNWenAdQTVSQ=", - "dev": true, - "requires": { - "bn.js": "^4.1.0", - "randombytes": "^2.0.1" - }, - "dependencies": { - "bn.js": { - "version": "4.11.8", - "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.11.8.tgz", - "integrity": "sha512-ItfYfPLkWHUjckQCk8xC+LwxgK8NYcXywGigJgSwOP8Y2iyWT4f2vsZnoOXTTbo+o5yXmIUJ4gn5538SO5S3gA==", - "dev": true - } - } - }, - "browserify-sign": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/browserify-sign/-/browserify-sign-4.1.0.tgz", - "integrity": "sha512-VYxo7cDCeYUoBZ0ZCy4UyEUCP3smyBd4DRQM5nrFS1jJjPJjX7rP3oLRpPoWfkhQfyJ0I9ZbHbKafrFD/SGlrg==", - "dev": true, - "requires": { - "bn.js": "^5.1.1", - "browserify-rsa": "^4.0.1", - "create-hash": "^1.2.0", - "create-hmac": "^1.1.7", - "elliptic": "^6.5.2", - "inherits": "^2.0.4", - "parse-asn1": "^5.1.5", - "readable-stream": "^3.6.0" - }, - "dependencies": { - "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==", - "dev": true, - "requires": { - "inherits": "^2.0.3", - "string_decoder": "^1.1.1", - "util-deprecate": "^1.0.1" - } - } - } - }, - "browserify-zlib": { - "version": "0.2.0", - "resolved": "https://registry.npmjs.org/browserify-zlib/-/browserify-zlib-0.2.0.tgz", - "integrity": "sha512-Z942RysHXmJrhqk88FmKBVq/v5tqmSkDz7p54G/MGyjMnCFFnC79XWNbg+Vta8W6Wb2qtSZTSxIGkJrRpCFEiA==", - "dev": true, - "requires": { - "pako": "~1.0.5" - } - }, - "browserslist": { - "version": "4.12.0", - "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.12.0.tgz", - "integrity": "sha512-UH2GkcEDSI0k/lRkuDSzFl9ZZ87skSy9w2XAn1MsZnL+4c4rqbBd3e82UWHbYDpztABrPBhZsTEeuxVfHppqDg==", - "dev": true, - "requires": { - "caniuse-lite": "^1.0.30001043", - "electron-to-chromium": "^1.3.413", - "node-releases": "^1.1.53", - "pkg-up": "^2.0.0" - } - }, - "browserstack": { - "version": "1.6.0", - "resolved": "https://registry.npmjs.org/browserstack/-/browserstack-1.6.0.tgz", - "integrity": "sha512-HJDJ0TSlmkwnt9RZ+v5gFpa1XZTBYTj0ywvLwJ3241J7vMw2jAsGNVhKHtmCOyg+VxeLZyaibO9UL71AsUeDIw==", - "dev": true, - "requires": { - "https-proxy-agent": "^2.2.1" - } - }, - "buffer": { - "version": "4.9.2", - "resolved": "https://registry.npmjs.org/buffer/-/buffer-4.9.2.tgz", - "integrity": "sha512-xq+q3SRMOxGivLhBNaUdC64hDTQwejJ+H0T/NB1XMtTVEwNTrfFF3gAxiyW0Bu/xWEGhjVKgUcMhCrUy2+uCWg==", - "dev": true, - "requires": { - "base64-js": "^1.0.2", - "ieee754": "^1.1.4", - "isarray": "^1.0.0" - } - }, - "buffer-from": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.1.tgz", - "integrity": "sha512-MQcXEUbCKtEo7bhqEs6560Hyd4XaovZlO/k9V3hjVUF/zwW7KBVdSK4gIt/bzwS9MbR5qob+F5jusZsb0YQK2A==", - "dev": true - }, - "buffer-indexof": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/buffer-indexof/-/buffer-indexof-1.1.1.tgz", - "integrity": "sha512-4/rOEg86jivtPTeOUUT61jJO1Ya1TrR/OkqCSZDyq84WJh3LuuiphBYJN+fm5xufIk4XAFcEwte/8WzC8If/1g==", - "dev": true - }, - "buffer-xor": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/buffer-xor/-/buffer-xor-1.0.3.tgz", - "integrity": "sha1-JuYe0UIvtw3ULm42cp7VHYVf6Nk=", - "dev": true - }, - "builtin-modules": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/builtin-modules/-/builtin-modules-1.1.1.tgz", - "integrity": "sha1-Jw8HbFpywC9bZaR9+Uxf46J4iS8=", - "dev": true - }, - "builtin-status-codes": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/builtin-status-codes/-/builtin-status-codes-3.0.0.tgz", - "integrity": "sha1-hZgoeOIbmOHGZCXgPQF0eI9Wnug=", - "dev": true - }, - "builtins": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/builtins/-/builtins-1.0.3.tgz", - "integrity": "sha1-y5T662HIaWRR2zZTThQi+U8K7og=", - "dev": true - }, - "bytes": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.0.0.tgz", - "integrity": "sha1-0ygVQE1olpn4Wk6k+odV3ROpYEg=", - "dev": true - }, - "cacache": { - "version": "15.0.0", - "resolved": "https://registry.npmjs.org/cacache/-/cacache-15.0.0.tgz", - "integrity": "sha512-L0JpXHhplbJSiDGzyJJnJCTL7er7NzbBgxzVqLswEb4bO91Zbv17OUMuUeu/q0ZwKn3V+1HM4wb9tO4eVE/K8g==", - "dev": true, - "requires": { - "chownr": "^1.1.2", - "fs-minipass": "^2.0.0", - "glob": "^7.1.4", - "infer-owner": "^1.0.4", - "lru-cache": "^5.1.1", - "minipass": "^3.1.1", - "minipass-collect": "^1.0.2", - "minipass-flush": "^1.0.5", - "minipass-pipeline": "^1.2.2", - "mkdirp": "^1.0.3", - "move-concurrently": "^1.0.1", - "p-map": "^3.0.0", - "promise-inflight": "^1.0.1", - "rimraf": "^2.7.1", - "ssri": "^8.0.0", - "tar": "^6.0.1", - "unique-filename": "^1.1.1" - }, - "dependencies": { - "mkdirp": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-1.0.4.tgz", - "integrity": "sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==", - "dev": true - }, - "rimraf": { - "version": "2.7.1", - "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", - "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==", - "dev": true, - "requires": { - "glob": "^7.1.3" - } - } - } - }, - "cache-base": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/cache-base/-/cache-base-1.0.1.tgz", - "integrity": "sha512-AKcdTnFSWATd5/GCPRxr2ChwIJ85CeyrEyjRHlKxQ56d4XJMGym0uAiKn0xbLOGOl3+yRpOTi484dVCEc5AUzQ==", - "dev": true, - "requires": { - "collection-visit": "^1.0.0", - "component-emitter": "^1.2.1", - "get-value": "^2.0.6", - "has-value": "^1.0.0", - "isobject": "^3.0.1", - "set-value": "^2.0.0", - "to-object-path": "^0.3.0", - "union-value": "^1.0.0", - "unset-value": "^1.0.0" - } - }, - "caller-callsite": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/caller-callsite/-/caller-callsite-2.0.0.tgz", - "integrity": "sha1-hH4PzgoiN1CpoCfFSzNzGtMVQTQ=", - "dev": true, - "requires": { - "callsites": "^2.0.0" - } - }, - "caller-path": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/caller-path/-/caller-path-2.0.0.tgz", - "integrity": "sha1-Ro+DBE42mrIBD6xfBs7uFbsssfQ=", - "dev": true, - "requires": { - "caller-callsite": "^2.0.0" - } - }, - "callsites": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/callsites/-/callsites-2.0.0.tgz", - "integrity": "sha1-BuuE8A7qQT2oav/vrL/7Ngk7PFA=", - "dev": true - }, - "camelcase": { - "version": "5.3.1", - "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz", - "integrity": "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==", - "dev": true - }, - "caniuse-api": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/caniuse-api/-/caniuse-api-3.0.0.tgz", - "integrity": "sha512-bsTwuIg/BZZK/vreVTYYbSWoe2F+71P7K5QGEX+pT250DZbfU1MQ5prOKpPR+LL6uWKK3KMwMCAS74QB3Um1uw==", - "dev": true, - "requires": { - "browserslist": "^4.0.0", - "caniuse-lite": "^1.0.0", - "lodash.memoize": "^4.1.2", - "lodash.uniq": "^4.5.0" - } - }, - "caniuse-lite": { - "version": "1.0.30001051", - "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001051.tgz", - "integrity": "sha512-sw8UUnTlRevawTMZKN7vpfwSjCBVoiMPlYd8oT2VwNylyPCBdMAUmLGUApnYYTtIm5JXsQegUAY7GPHqgfDzjw==", - "dev": true - }, - "canonical-path": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/canonical-path/-/canonical-path-1.0.0.tgz", - "integrity": "sha512-feylzsbDxi1gPZ1IjystzIQZagYYLvfKrSuygUCgf7z6x790VEzze5QEkdSV1U58RA7Hi0+v6fv4K54atOzATg==", - "dev": true - }, - "caseless": { - "version": "0.12.0", - "resolved": "https://registry.npmjs.org/caseless/-/caseless-0.12.0.tgz", - "integrity": "sha1-G2gcIf+EAzyCZUMJBolCDRhxUdw=", - "dev": true - }, - "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.4.0", - "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.4.0.tgz", - "integrity": "sha512-aXAaho2VJtisB/1fg1+3nlLJqGOuewTzQpd/Tz0yTg2R0e4IGtshYvtjowyEumcBv2z+y4+kc75Mz7j5xJskcQ==", - "dev": true, - "requires": { - "anymatch": "~3.1.1", - "braces": "~3.0.2", - "fsevents": "~2.1.2", - "glob-parent": "~5.1.0", - "is-binary-path": "~2.1.0", - "is-glob": "~4.0.1", - "normalize-path": "~3.0.0", - "readdirp": "~3.4.0" - }, - "dependencies": { - "glob-parent": { - "version": "5.1.1", - "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.1.tgz", - "integrity": "sha512-FnI+VGOpnlGHWZxthPGR+QhR78fuiK0sNLkHQv+bL9fQi57lNNdquIbna/WrfROrolq8GK5Ek6BiMwqL/voRYQ==", - "dev": true, - "requires": { - "is-glob": "^4.0.1" - } - } - } - }, - "chownr": { - "version": "1.1.4", - "resolved": "https://registry.npmjs.org/chownr/-/chownr-1.1.4.tgz", - "integrity": "sha512-jJ0bqzaylmJtVnNgzTeSOs8DPavpbYgEr/b0YL8/2GO3xJEhInFmhKMUnEJQjZumK7KXGFhUy89PrsJWlakBVg==", - "dev": true - }, - "chrome-trace-event": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/chrome-trace-event/-/chrome-trace-event-1.0.2.tgz", - "integrity": "sha512-9e/zx1jw7B4CO+c/RXoCsfg/x1AfUBioy4owYH0bJprEYAx5hRFLRhWBqHAG57D0ZM4H7vxbP7bPe0VwhQRYDQ==", - "dev": true, - "requires": { - "tslib": "^1.9.0" - } - }, - "cipher-base": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/cipher-base/-/cipher-base-1.0.4.tgz", - "integrity": "sha512-Kkht5ye6ZGmwv40uUDZztayT2ThLQGfnj/T71N/XzeZeo3nf8foyW7zGTsPYkEya3m5f3cAypH+qe7YOrM1U2Q==", - "dev": true, - "requires": { - "inherits": "^2.0.1", - "safe-buffer": "^5.0.1" - } - }, - "circular-dependency-plugin": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/circular-dependency-plugin/-/circular-dependency-plugin-5.2.0.tgz", - "integrity": "sha512-7p4Kn/gffhQaavNfyDFg7LS5S/UT1JAjyGd4UqR2+jzoYF02eDkj0Ec3+48TsIa4zghjLY87nQHIh/ecK9qLdw==", - "dev": true - }, - "class-utils": { - "version": "0.3.6", - "resolved": "https://registry.npmjs.org/class-utils/-/class-utils-0.3.6.tgz", - "integrity": "sha512-qOhPa/Fj7s6TY8H8esGu5QNpMMQxz79h+urzrNYN6mn+9BnxlDGf5QZ+XeCDsxSjPqsSR56XOZOJmpeurnLMeg==", - "dev": true, - "requires": { - "arr-union": "^3.1.0", - "define-property": "^0.2.5", - "isobject": "^3.0.0", - "static-extend": "^0.1.1" - }, - "dependencies": { - "define-property": { - "version": "0.2.5", - "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", - "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", - "dev": true, - "requires": { - "is-descriptor": "^0.1.0" - } - } - } - }, - "clean-stack": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/clean-stack/-/clean-stack-2.2.0.tgz", - "integrity": "sha512-4diC9HaTE+KRAMWhDhrGOECgWZxoevMc5TlkObMqNSsVU62PYzXZ/SMTjzyGAFF1YusgxGcSWTEXBhp0CPwQ1A==", - "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-spinners": { - "version": "2.3.0", - "resolved": "https://registry.npmjs.org/cli-spinners/-/cli-spinners-2.3.0.tgz", - "integrity": "sha512-Xs2Hf2nzrvJMFKimOR7YR0QwZ8fc0u98kdtwN1eNAZzNQgH3vK2pXzff6GJtKh7S5hoJ87ECiAiZFS2fb5Ii2w==", - "dev": true - }, - "cli-width": { - "version": "2.2.1", - "resolved": "https://registry.npmjs.org/cli-width/-/cli-width-2.2.1.tgz", - "integrity": "sha512-GRMWDxpOB6Dgk2E5Uo+3eEBvtOOlimMmpbFiKuLFnQzYDavtLFY3K5ona41jgN/WdRZtG7utuVSVTL4HbZHGkw==", - "dev": true - }, - "cliui": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/cliui/-/cliui-4.1.0.tgz", - "integrity": "sha512-4FG+RSG9DL7uEwRUZXZn3SS34DiDPfzP0VOiEwtUWlE+AR2EIg+hSyvrIgUUfhdgR/UkAeW2QHgeP+hWrXs7jQ==", - "dev": true, - "requires": { - "string-width": "^2.1.1", - "strip-ansi": "^4.0.0", - "wrap-ansi": "^2.0.0" - }, - "dependencies": { - "ansi-regex": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-3.0.0.tgz", - "integrity": "sha1-7QMXwyIGT3lGbAKWa922Bas32Zg=", - "dev": true - }, - "strip-ansi": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-4.0.0.tgz", - "integrity": "sha1-qEeQIusaw2iocTibY1JixQXuNo8=", - "dev": true, - "requires": { - "ansi-regex": "^3.0.0" - } - } - } - }, - "clone": { - "version": "2.1.2", - "resolved": "https://registry.npmjs.org/clone/-/clone-2.1.2.tgz", - "integrity": "sha1-G39Ln1kfHo+DZwQBYANFoCiHQ18=", - "dev": true - }, - "clone-deep": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/clone-deep/-/clone-deep-4.0.1.tgz", - "integrity": "sha512-neHB9xuzh/wk0dIHweyAXv2aPGZIVk3pLMe+/RNzINf17fe0OG96QroktYAUm7SM1PBnzTabaLboqqxDyMU+SQ==", - "dev": true, - "requires": { - "is-plain-object": "^2.0.4", - "kind-of": "^6.0.2", - "shallow-clone": "^3.0.0" - } - }, - "coa": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/coa/-/coa-2.0.2.tgz", - "integrity": "sha512-q5/jG+YQnSy4nRTV4F7lPepBJZ8qBNJJDBuJdoejDyLXgmL7IEo+Le2JDZudFTFt7mrCqIRaSjws4ygRCTCAXA==", - "dev": true, - "requires": { - "@types/q": "^1.5.1", - "chalk": "^2.4.1", - "q": "^1.1.2" - } - }, - "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=", - "dev": true - }, - "codelyzer": { - "version": "5.2.2", - "resolved": "https://registry.npmjs.org/codelyzer/-/codelyzer-5.2.2.tgz", - "integrity": "sha512-jB4FZ1Sx7kZhvZVdf+N2BaKTdrrNZOL0Bj10RRfrhHrb3zEvXjJvvq298JPMJAiyiCS/v4zs1QlGU0ip7xGqeA==", - "dev": true, - "requires": { - "app-root-path": "^2.2.1", - "aria-query": "^3.0.0", - "axobject-query": "2.0.2", - "css-selector-tokenizer": "^0.7.1", - "cssauron": "^1.4.0", - "damerau-levenshtein": "^1.0.4", - "semver-dsl": "^1.0.1", - "source-map": "^0.5.7", - "sprintf-js": "^1.1.2" - }, - "dependencies": { - "source-map": { - "version": "0.5.7", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.5.7.tgz", - "integrity": "sha1-igOdLRAh0i0eoUyA2OpGi6LvP8w=", - "dev": true - }, - "sprintf-js": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.1.2.tgz", - "integrity": "sha512-VE0SOVEHCk7Qc8ulkWw3ntAzXuqf7S2lvwQaDLRnUeIEaKNQJzV6BwmLKhOqT61aGhfUMrXeaBk+oDGCzvhcug==", - "dev": true - } - } - }, - "codemirror": { - "version": "5.58.2", - "resolved": "https://registry.npmjs.org/codemirror/-/codemirror-5.58.2.tgz", - "integrity": "sha512-K/hOh24cCwRutd1Mk3uLtjWzNISOkm4fvXiMO7LucCrqbh6aJDdtqUziim3MZUI6wOY0rvY1SlL1Ork01uMy6w==" - }, - "collection-visit": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/collection-visit/-/collection-visit-1.0.0.tgz", - "integrity": "sha1-S8A3PBZLwykbTTaMgpzxqApZ3KA=", - "dev": true, - "requires": { - "map-visit": "^1.0.0", - "object-visit": "^1.0.0" - } - }, - "color": { - "version": "3.1.2", - "resolved": "https://registry.npmjs.org/color/-/color-3.1.2.tgz", - "integrity": "sha512-vXTJhHebByxZn3lDvDJYw4lR5+uB3vuoHsuYA5AKuxRVn5wzzIfQKGLBmgdVRHKTJYeK5rvJcHnrd0Li49CFpg==", - "dev": true, - "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==", - "dev": true, - "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=", - "dev": true - }, - "color-string": { - "version": "1.5.3", - "resolved": "https://registry.npmjs.org/color-string/-/color-string-1.5.3.tgz", - "integrity": "sha512-dC2C5qeWoYkxki5UAXapdjqO672AM4vZuPGRQfO8b5HKuKGBbKWpITyDYN7TOFKvRW7kOgAn3746clDBMDJyQw==", - "dev": true, - "requires": { - "color-name": "^1.0.0", - "simple-swizzle": "^0.2.2" - } - }, - "colors": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/colors/-/colors-1.1.2.tgz", - "integrity": "sha1-FopHAXVran9RoSzgyXv6KMCE7WM=", - "dev": true - }, - "combined-stream": { - "version": "1.0.8", - "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz", - "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==", - "dev": true, - "requires": { - "delayed-stream": "~1.0.0" - } - }, - "commander": { - "version": "2.20.3", - "resolved": "https://registry.npmjs.org/commander/-/commander-2.20.3.tgz", - "integrity": "sha512-GpVkmM8vF2vQUkj2LvZmD35JxeJOLCwJ9cUkugyk2nuhbv3+mJvpLYYt+0+USMxE+oj+ey/lJEnhZw75x/OMcQ==", - "dev": true - }, - "commondir": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/commondir/-/commondir-1.0.1.tgz", - "integrity": "sha1-3dgA2gxmEnOTzKWVDqloo6rxJTs=", - "dev": true - }, - "compare-versions": { - "version": "3.6.0", - "resolved": "https://registry.npmjs.org/compare-versions/-/compare-versions-3.6.0.tgz", - "integrity": "sha512-W6Af2Iw1z4CB7q4uU4hv646dW9GQuBM+YpC0UvUCWSD8w90SJjp+ujJuXaEMtAXBtSqGfMPuFOVn4/+FlaqfBA==", - "dev": true - }, - "component-emitter": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/component-emitter/-/component-emitter-1.3.0.tgz", - "integrity": "sha512-Rd3se6QB+sO1TwqZjscQrurpEPIfO0/yYnSin6Q/rD3mOutHvUrCAhJub3r90uNb+SESBuE0QYoB90YdfatsRg==", - "dev": true - }, - "compressible": { - "version": "2.0.18", - "resolved": "https://registry.npmjs.org/compressible/-/compressible-2.0.18.tgz", - "integrity": "sha512-AF3r7P5dWxL8MxyITRMlORQNaOA2IkAFaTr4k7BUumjPtRpGDTZpl0Pb1XCO6JeDCBdp126Cgs9sMxqSjgYyRg==", - "dev": true, - "requires": { - "mime-db": ">= 1.43.0 < 2" - } - }, - "compression": { - "version": "1.7.4", - "resolved": "https://registry.npmjs.org/compression/-/compression-1.7.4.tgz", - "integrity": "sha512-jaSIDzP9pZVS4ZfQ+TzvtiWhdpFhE2RDHz8QJkpX9SIpLq88VueF5jJw6t+6CUQcAoA6t+x89MLrWAqpfDE8iQ==", - "dev": true, - "requires": { - "accepts": "~1.3.5", - "bytes": "3.0.0", - "compressible": "~2.0.16", - "debug": "2.6.9", - "on-headers": "~1.0.2", - "safe-buffer": "5.1.2", - "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==", - "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 - } - } - }, - "concat-map": { - "version": "0.0.1", - "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", - "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=", - "dev": true - }, - "concat-stream": { - "version": "1.6.2", - "resolved": "https://registry.npmjs.org/concat-stream/-/concat-stream-1.6.2.tgz", - "integrity": "sha512-27HBghJxjiZtIk3Ycvn/4kbJk/1uZuJFfuPEns6LaEvpvG1f0hTea8lilrouyo9mVc2GWdcEZ8OLoGmSADlrCw==", - "dev": true, - "requires": { - "buffer-from": "^1.0.0", - "inherits": "^2.0.3", - "readable-stream": "^2.2.2", - "typedarray": "^0.0.6" - } - }, - "connect": { - "version": "3.7.0", - "resolved": "https://registry.npmjs.org/connect/-/connect-3.7.0.tgz", - "integrity": "sha512-ZqRXc+tZukToSNmh5C2iWMSoV3X1YUcPbqEM4DkEG5tNQXrQUZCNVGGv3IuicnkMtPfGf3Xtp8WCXs295iQ1pQ==", - "dev": true, - "requires": { - "debug": "2.6.9", - "finalhandler": "1.1.2", - "parseurl": "~1.3.3", - "utils-merge": "1.0.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 - } - } - }, - "connect-history-api-fallback": { - "version": "1.6.0", - "resolved": "https://registry.npmjs.org/connect-history-api-fallback/-/connect-history-api-fallback-1.6.0.tgz", - "integrity": "sha512-e54B99q/OUoH64zYYRf3HBP5z24G38h5D3qXu23JGRoigpX5Ss4r9ZnDk3g0Z8uQC2x2lPaJ+UlWBc1ZWBWdLg==", - "dev": true - }, - "console-browserify": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/console-browserify/-/console-browserify-1.2.0.tgz", - "integrity": "sha512-ZMkYO/LkF17QvCPqM0gxw8yUzigAOZOSWSHg91FH6orS7vcEj5dVZTidN2fQ14yBSdg97RqhSNwLUXInd52OTA==", - "dev": true - }, - "constants-browserify": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/constants-browserify/-/constants-browserify-1.0.0.tgz", - "integrity": "sha1-wguW2MYXdIqvHBYCF2DNJ/y4y3U=", - "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==", - "dev": true, - "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==", - "dev": true - }, - "convert-source-map": { - "version": "1.7.0", - "resolved": "https://registry.npmjs.org/convert-source-map/-/convert-source-map-1.7.0.tgz", - "integrity": "sha512-4FJkXzKXEDB1snCFZlLP4gpC3JILicCpGbzG9f9G7tGqGCzETQ2hWPrcinA9oU4wtf2biUaEH5065UnMeR33oA==", - "dev": true, - "requires": { - "safe-buffer": "~5.1.1" - } - }, - "cookie": { - "version": "0.4.0", - "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.4.0.tgz", - "integrity": "sha512-+Hp8fLp57wnUSt0tY0tHEXh4voZRDnoIrZPqlo3DPiI4y9lwg/jqx+1Om94/W6ZaPDOUbnjOt/99w66zk+l1Xg==", - "dev": true - }, - "cookie-signature": { - "version": "1.0.6", - "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz", - "integrity": "sha1-4wOogrNCzD7oylE6eZmXNNqzriw=", - "dev": true - }, - "copy-concurrently": { - "version": "1.0.5", - "resolved": "https://registry.npmjs.org/copy-concurrently/-/copy-concurrently-1.0.5.tgz", - "integrity": "sha512-f2domd9fsVDFtaFcbaRZuYXwtdmnzqbADSwhSWYxYB/Q8zsdUUFMXVRwXGDMWmbEzAn1kdRrtI1T/KTFOL4X2A==", - "dev": true, - "requires": { - "aproba": "^1.1.1", - "fs-write-stream-atomic": "^1.0.8", - "iferr": "^0.1.5", - "mkdirp": "^0.5.1", - "rimraf": "^2.5.4", - "run-queue": "^1.0.0" - }, - "dependencies": { - "rimraf": { - "version": "2.7.1", - "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", - "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==", - "dev": true, - "requires": { - "glob": "^7.1.3" - } - } - } - }, - "copy-descriptor": { - "version": "0.1.1", - "resolved": "https://registry.npmjs.org/copy-descriptor/-/copy-descriptor-0.1.1.tgz", - "integrity": "sha1-Z29us8OZl8LuGsOpJP1hJHSPV40=", - "dev": true - }, - "copy-webpack-plugin": { - "version": "5.1.1", - "resolved": "https://registry.npmjs.org/copy-webpack-plugin/-/copy-webpack-plugin-5.1.1.tgz", - "integrity": "sha512-P15M5ZC8dyCjQHWwd4Ia/dm0SgVvZJMYeykVIVYXbGyqO4dWB5oyPHp9i7wjwo5LhtlhKbiBCdS2NvM07Wlybg==", - "dev": true, - "requires": { - "cacache": "^12.0.3", - "find-cache-dir": "^2.1.0", - "glob-parent": "^3.1.0", - "globby": "^7.1.1", - "is-glob": "^4.0.1", - "loader-utils": "^1.2.3", - "minimatch": "^3.0.4", - "normalize-path": "^3.0.0", - "p-limit": "^2.2.1", - "schema-utils": "^1.0.0", - "serialize-javascript": "^2.1.2", - "webpack-log": "^2.0.0" - }, - "dependencies": { - "cacache": { - "version": "12.0.4", - "resolved": "https://registry.npmjs.org/cacache/-/cacache-12.0.4.tgz", - "integrity": "sha512-a0tMB40oefvuInr4Cwb3GerbL9xTj1D5yg0T5xrjGCGyfvbxseIXX7BAO/u/hIXdafzOI5JC3wDwHyf24buOAQ==", - "dev": true, - "requires": { - "bluebird": "^3.5.5", - "chownr": "^1.1.1", - "figgy-pudding": "^3.5.1", - "glob": "^7.1.4", - "graceful-fs": "^4.1.15", - "infer-owner": "^1.0.3", - "lru-cache": "^5.1.1", - "mississippi": "^3.0.0", - "mkdirp": "^0.5.1", - "move-concurrently": "^1.0.1", - "promise-inflight": "^1.0.1", - "rimraf": "^2.6.3", - "ssri": "^6.0.1", - "unique-filename": "^1.1.1", - "y18n": "^4.0.0" - } - }, - "find-cache-dir": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/find-cache-dir/-/find-cache-dir-2.1.0.tgz", - "integrity": "sha512-Tq6PixE0w/VMFfCgbONnkiQIVol/JJL7nRMi20fqzA4NRs9AfeqMGeRdPi3wIhYkxjeBaWh2rxwapn5Tu3IqOQ==", - "dev": true, - "requires": { - "commondir": "^1.0.1", - "make-dir": "^2.0.0", - "pkg-dir": "^3.0.0" - } - }, - "json5": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/json5/-/json5-1.0.1.tgz", - "integrity": "sha512-aKS4WQjPenRxiQsC93MNfjx+nbF4PAdYzmd/1JIj8HYzqfbu86beTuNgXDzPknWk0n0uARlyewZo4s++ES36Ow==", - "dev": true, - "requires": { - "minimist": "^1.2.0" - } - }, - "loader-utils": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/loader-utils/-/loader-utils-1.4.0.tgz", - "integrity": "sha512-qH0WSMBtn/oHuwjy/NucEgbx5dbxxnxup9s4PVXJUDHZBQY+s0NWA9rJf53RBnQZxfch7euUui7hpoAPvALZdA==", - "dev": true, - "requires": { - "big.js": "^5.2.2", - "emojis-list": "^3.0.0", - "json5": "^1.0.1" - } - }, - "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-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 - }, - "rimraf": { - "version": "2.7.1", - "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", - "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==", - "dev": true, - "requires": { - "glob": "^7.1.3" - } - }, - "schema-utils": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-1.0.0.tgz", - "integrity": "sha512-i27Mic4KovM/lnGsy8whRCHhc7VicJajAjTrYg11K9zfZXnYIt4k5F+kZkwjnrhKzLic/HLU4j11mjsz2G/75g==", - "dev": true, - "requires": { - "ajv": "^6.1.0", - "ajv-errors": "^1.0.0", - "ajv-keywords": "^3.1.0" - } - }, - "ssri": { - "version": "6.0.2", - "resolved": "https://registry.npmjs.org/ssri/-/ssri-6.0.2.tgz", - "integrity": "sha512-cepbSq/neFK7xB6A50KHN0xHDotYzq58wWCa5LeWqnPrHG8GzfEjO/4O8kpmcGW+oaxkvhEJCWgbgNk4/ZV93Q==", - "dev": true, - "requires": { - "figgy-pudding": "^3.5.1" - } - } - } - }, - "core-js": { - "version": "3.6.4", - "resolved": "https://registry.npmjs.org/core-js/-/core-js-3.6.4.tgz", - "integrity": "sha512-4paDGScNgZP2IXXilaffL9X7968RuvwlkK3xWtZRVqgd8SYNiVKRJvkFd1aqqEuPfN7E68ZHEp9hDj6lHj4Hyw==", - "dev": true - }, - "core-js-compat": { - "version": "3.6.5", - "resolved": "https://registry.npmjs.org/core-js-compat/-/core-js-compat-3.6.5.tgz", - "integrity": "sha512-7ItTKOhOZbznhXAQ2g/slGg1PJV5zDO/WdkTwi7UEOJmkvsE32PWvx6mKtDjiMpjnR2CNf6BAD6sSxIlv7ptng==", - "dev": true, - "requires": { - "browserslist": "^4.8.5", - "semver": "7.0.0" - }, - "dependencies": { - "semver": { - "version": "7.0.0", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.0.0.tgz", - "integrity": "sha512-+GB6zVA9LWh6zovYQLALHwv5rb2PHGlJi3lfiqIHxR0uuwCgefcOJc59v9fv1w8GbStwxuuqqAjI9NMAOOgq1A==", - "dev": true - } - } - }, - "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=", - "dev": true - }, - "cors": { - "version": "2.8.5", - "resolved": "https://registry.npmjs.org/cors/-/cors-2.8.5.tgz", - "integrity": "sha512-KIHbLJqu73RGr/hnbrO9uBeixNGuvSQjul/jdFvS/KFSIH1hWVd1ng7zOHx+YrEfInLG7q4n6GHQ9cDtxv/P6g==", - "dev": true, - "requires": { - "object-assign": "^4", - "vary": "^1" - } - }, - "cosmiconfig": { - "version": "5.2.1", - "resolved": "https://registry.npmjs.org/cosmiconfig/-/cosmiconfig-5.2.1.tgz", - "integrity": "sha512-H65gsXo1SKjf8zmrJ67eJk8aIRKV5ff2D4uKZIBZShbhGSpEmsQOPW/SKMKYhSTrqR7ufy6RP69rPogdaPh/kA==", - "dev": true, - "requires": { - "import-fresh": "^2.0.0", - "is-directory": "^0.3.1", - "js-yaml": "^3.13.1", - "parse-json": "^4.0.0" - } - }, - "create-ecdh": { - "version": "4.0.3", - "resolved": "https://registry.npmjs.org/create-ecdh/-/create-ecdh-4.0.3.tgz", - "integrity": "sha512-GbEHQPMOswGpKXM9kCWVrremUcBmjteUaQ01T9rkKCPDXfUHX0IoP9LpHYo2NPFampa4e+/pFDc3jQdxrxQLaw==", - "dev": true, - "requires": { - "bn.js": "^4.1.0", - "elliptic": "^6.0.0" - }, - "dependencies": { - "bn.js": { - "version": "4.11.8", - "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.11.8.tgz", - "integrity": "sha512-ItfYfPLkWHUjckQCk8xC+LwxgK8NYcXywGigJgSwOP8Y2iyWT4f2vsZnoOXTTbo+o5yXmIUJ4gn5538SO5S3gA==", - "dev": true - } - } - }, - "create-hash": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/create-hash/-/create-hash-1.2.0.tgz", - "integrity": "sha512-z00bCGNHDG8mHAkP7CtT1qVu+bFQUPjYq/4Iv3C3kWjTFV10zIjfSoeqXo9Asws8gwSHDGj/hl2u4OGIjapeCg==", - "dev": true, - "requires": { - "cipher-base": "^1.0.1", - "inherits": "^2.0.1", - "md5.js": "^1.3.4", - "ripemd160": "^2.0.1", - "sha.js": "^2.4.0" - } - }, - "create-hmac": { - "version": "1.1.7", - "resolved": "https://registry.npmjs.org/create-hmac/-/create-hmac-1.1.7.tgz", - "integrity": "sha512-MJG9liiZ+ogc4TzUwuvbER1JRdgvUFSB5+VR/g5h82fGaIRWMWddtKBHi7/sVhfjQZ6SehlyhvQYrcYkaUIpLg==", - "dev": true, - "requires": { - "cipher-base": "^1.0.3", - "create-hash": "^1.1.0", - "inherits": "^2.0.1", - "ripemd160": "^2.0.0", - "safe-buffer": "^5.0.1", - "sha.js": "^2.4.8" - } - }, - "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" - }, - "dependencies": { - "semver": { - "version": "5.7.1", - "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", - "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", - "dev": true - } - } - }, - "crypto-browserify": { - "version": "3.12.0", - "resolved": "https://registry.npmjs.org/crypto-browserify/-/crypto-browserify-3.12.0.tgz", - "integrity": "sha512-fz4spIh+znjO2VjL+IdhEpRJ3YN6sMzITSBijk6FK2UvTqruSQW+/cCZTSNsMiZNvUeq0CqurF+dAbyiGOY6Wg==", - "dev": true, - "requires": { - "browserify-cipher": "^1.0.0", - "browserify-sign": "^4.0.0", - "create-ecdh": "^4.0.0", - "create-hash": "^1.1.0", - "create-hmac": "^1.1.0", - "diffie-hellman": "^5.0.0", - "inherits": "^2.0.1", - "pbkdf2": "^3.0.3", - "public-encrypt": "^4.0.0", - "randombytes": "^2.0.0", - "randomfill": "^1.0.3" - } - }, - "css": { - "version": "2.2.4", - "resolved": "https://registry.npmjs.org/css/-/css-2.2.4.tgz", - "integrity": "sha512-oUnjmWpy0niI3x/mPL8dVEI1l7MnG3+HHyRPHf+YFSbK+svOhXpmSOcDURUh2aOCgl2grzrOPt1nHLuCVFULLw==", - "dev": true, - "requires": { - "inherits": "^2.0.3", - "source-map": "^0.6.1", - "source-map-resolve": "^0.5.2", - "urix": "^0.1.0" - }, - "dependencies": { - "source-map": { - "version": "0.6.1", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", - "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", - "dev": true - } - } - }, - "css-color-names": { - "version": "0.0.4", - "resolved": "https://registry.npmjs.org/css-color-names/-/css-color-names-0.0.4.tgz", - "integrity": "sha1-gIrcLnnPhHOAabZGyyDsJ762KeA=", - "dev": true - }, - "css-declaration-sorter": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/css-declaration-sorter/-/css-declaration-sorter-4.0.1.tgz", - "integrity": "sha512-BcxQSKTSEEQUftYpBVnsH4SF05NTuBokb19/sBt6asXGKZ/6VP7PLG1CBCkFDYOnhXhPh0jMhO6xZ71oYHXHBA==", - "dev": true, - "requires": { - "postcss": "^7.0.1", - "timsort": "^0.3.0" - } - }, - "css-loader": { - "version": "3.5.1", - "resolved": "https://registry.npmjs.org/css-loader/-/css-loader-3.5.1.tgz", - "integrity": "sha512-0G4CbcZzQ9D1Q6ndOfjFuMDo8uLYMu5vc9Abs5ztyHcKvmil6GJrMiNjzzi3tQvUF+mVRuDg7bE6Oc0Prolgig==", - "dev": true, - "requires": { - "camelcase": "^5.3.1", - "cssesc": "^3.0.0", - "icss-utils": "^4.1.1", - "loader-utils": "^1.2.3", - "normalize-path": "^3.0.0", - "postcss": "^7.0.27", - "postcss-modules-extract-imports": "^2.0.0", - "postcss-modules-local-by-default": "^3.0.2", - "postcss-modules-scope": "^2.2.0", - "postcss-modules-values": "^3.0.0", - "postcss-value-parser": "^4.0.3", - "schema-utils": "^2.6.5", - "semver": "^6.3.0" - }, - "dependencies": { - "json5": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/json5/-/json5-1.0.1.tgz", - "integrity": "sha512-aKS4WQjPenRxiQsC93MNfjx+nbF4PAdYzmd/1JIj8HYzqfbu86beTuNgXDzPknWk0n0uARlyewZo4s++ES36Ow==", - "dev": true, - "requires": { - "minimist": "^1.2.0" - } - }, - "loader-utils": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/loader-utils/-/loader-utils-1.4.0.tgz", - "integrity": "sha512-qH0WSMBtn/oHuwjy/NucEgbx5dbxxnxup9s4PVXJUDHZBQY+s0NWA9rJf53RBnQZxfch7euUui7hpoAPvALZdA==", - "dev": true, - "requires": { - "big.js": "^5.2.2", - "emojis-list": "^3.0.0", - "json5": "^1.0.1" - } - } - } - }, - "css-parse": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/css-parse/-/css-parse-2.0.0.tgz", - "integrity": "sha1-pGjuZnwW2BzPBcWMONKpfHgNv9Q=", - "dev": true, - "requires": { - "css": "^2.0.0" - } - }, - "css-select": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/css-select/-/css-select-2.1.0.tgz", - "integrity": "sha512-Dqk7LQKpwLoH3VovzZnkzegqNSuAziQyNZUcrdDM401iY+R5NkGBXGmtO05/yaXQziALuPogeG0b7UAgjnTJTQ==", - "dev": true, - "requires": { - "boolbase": "^1.0.0", - "css-what": "^3.2.1", - "domutils": "^1.7.0", - "nth-check": "^1.0.2" - } - }, - "css-select-base-adapter": { - "version": "0.1.1", - "resolved": "https://registry.npmjs.org/css-select-base-adapter/-/css-select-base-adapter-0.1.1.tgz", - "integrity": "sha512-jQVeeRG70QI08vSTwf1jHxp74JoZsr2XSgETae8/xC8ovSnL2WF87GTLO86Sbwdt2lK4Umg4HnnwMO4YF3Ce7w==", - "dev": true - }, - "css-selector-tokenizer": { - "version": "0.7.2", - "resolved": "https://registry.npmjs.org/css-selector-tokenizer/-/css-selector-tokenizer-0.7.2.tgz", - "integrity": "sha512-yj856NGuAymN6r8bn8/Jl46pR+OC3eEvAhfGYDUe7YPtTPAYrSSw4oAniZ9Y8T5B92hjhwTBLUen0/vKPxf6pw==", - "dev": true, - "requires": { - "cssesc": "^3.0.0", - "fastparse": "^1.1.2", - "regexpu-core": "^4.6.0" - } - }, - "css-tree": { - "version": "1.0.0-alpha.37", - "resolved": "https://registry.npmjs.org/css-tree/-/css-tree-1.0.0-alpha.37.tgz", - "integrity": "sha512-DMxWJg0rnz7UgxKT0Q1HU/L9BeJI0M6ksor0OgqOnF+aRCDWg/N2641HmVyU9KVIu0OVVWOb2IpC9A+BJRnejg==", - "dev": true, - "requires": { - "mdn-data": "2.0.4", - "source-map": "^0.6.1" - }, - "dependencies": { - "source-map": { - "version": "0.6.1", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", - "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", - "dev": true - } - } - }, - "css-what": { - "version": "3.2.1", - "resolved": "https://registry.npmjs.org/css-what/-/css-what-3.2.1.tgz", - "integrity": "sha512-WwOrosiQTvyms+Ti5ZC5vGEK0Vod3FTt1ca+payZqvKuGJF+dq7bG63DstxtN0dpm6FxY27a/zS3Wten+gEtGw==", - "dev": true - }, - "cssauron": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/cssauron/-/cssauron-1.4.0.tgz", - "integrity": "sha1-pmAt/34EqDBtwNuaVR6S6LVmKtg=", - "dev": true, - "requires": { - "through": "X.X.X" - } - }, - "cssesc": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/cssesc/-/cssesc-3.0.0.tgz", - "integrity": "sha512-/Tb/JcjK111nNScGob5MNtsntNM1aCNUDipB/TkwZFhyDrrE47SOx/18wF2bbjgc3ZzCSKW1T5nt5EbFoAz/Vg==", - "dev": true - }, - "cssnano": { - "version": "4.1.10", - "resolved": "https://registry.npmjs.org/cssnano/-/cssnano-4.1.10.tgz", - "integrity": "sha512-5wny+F6H4/8RgNlaqab4ktc3e0/blKutmq8yNlBFXA//nSFFAqAngjNVRzUvCgYROULmZZUoosL/KSoZo5aUaQ==", - "dev": true, - "requires": { - "cosmiconfig": "^5.0.0", - "cssnano-preset-default": "^4.0.7", - "is-resolvable": "^1.0.0", - "postcss": "^7.0.0" - } - }, - "cssnano-preset-default": { - "version": "4.0.7", - "resolved": "https://registry.npmjs.org/cssnano-preset-default/-/cssnano-preset-default-4.0.7.tgz", - "integrity": "sha512-x0YHHx2h6p0fCl1zY9L9roD7rnlltugGu7zXSKQx6k2rYw0Hi3IqxcoAGF7u9Q5w1nt7vK0ulxV8Lo+EvllGsA==", - "dev": true, - "requires": { - "css-declaration-sorter": "^4.0.1", - "cssnano-util-raw-cache": "^4.0.1", - "postcss": "^7.0.0", - "postcss-calc": "^7.0.1", - "postcss-colormin": "^4.0.3", - "postcss-convert-values": "^4.0.1", - "postcss-discard-comments": "^4.0.2", - "postcss-discard-duplicates": "^4.0.2", - "postcss-discard-empty": "^4.0.1", - "postcss-discard-overridden": "^4.0.1", - "postcss-merge-longhand": "^4.0.11", - "postcss-merge-rules": "^4.0.3", - "postcss-minify-font-values": "^4.0.2", - "postcss-minify-gradients": "^4.0.2", - "postcss-minify-params": "^4.0.2", - "postcss-minify-selectors": "^4.0.2", - "postcss-normalize-charset": "^4.0.1", - "postcss-normalize-display-values": "^4.0.2", - "postcss-normalize-positions": "^4.0.2", - "postcss-normalize-repeat-style": "^4.0.2", - "postcss-normalize-string": "^4.0.2", - "postcss-normalize-timing-functions": "^4.0.2", - "postcss-normalize-unicode": "^4.0.1", - "postcss-normalize-url": "^4.0.1", - "postcss-normalize-whitespace": "^4.0.2", - "postcss-ordered-values": "^4.1.2", - "postcss-reduce-initial": "^4.0.3", - "postcss-reduce-transforms": "^4.0.2", - "postcss-svgo": "^4.0.2", - "postcss-unique-selectors": "^4.0.1" - } - }, - "cssnano-util-get-arguments": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/cssnano-util-get-arguments/-/cssnano-util-get-arguments-4.0.0.tgz", - "integrity": "sha1-7ToIKZ8h11dBsg87gfGU7UnMFQ8=", - "dev": true - }, - "cssnano-util-get-match": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/cssnano-util-get-match/-/cssnano-util-get-match-4.0.0.tgz", - "integrity": "sha1-wOTKB/U4a7F+xeUiULT1lhNlFW0=", - "dev": true - }, - "cssnano-util-raw-cache": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/cssnano-util-raw-cache/-/cssnano-util-raw-cache-4.0.1.tgz", - "integrity": "sha512-qLuYtWK2b2Dy55I8ZX3ky1Z16WYsx544Q0UWViebptpwn/xDBmog2TLg4f+DBMg1rJ6JDWtn96WHbOKDWt1WQA==", - "dev": true, - "requires": { - "postcss": "^7.0.0" - } - }, - "cssnano-util-same-parent": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/cssnano-util-same-parent/-/cssnano-util-same-parent-4.0.1.tgz", - "integrity": "sha512-WcKx5OY+KoSIAxBW6UBBRay1U6vkYheCdjyVNDm85zt5K9mHoGOfsOsqIszfAqrQQFIIKgjh2+FDgIj/zsl21Q==", - "dev": true - }, - "csso": { - "version": "4.0.3", - "resolved": "https://registry.npmjs.org/csso/-/csso-4.0.3.tgz", - "integrity": "sha512-NL3spysxUkcrOgnpsT4Xdl2aiEiBG6bXswAABQVHcMrfjjBisFOKwLDOmf4wf32aPdcJws1zds2B0Rg+jqMyHQ==", - "dev": true, - "requires": { - "css-tree": "1.0.0-alpha.39" - }, - "dependencies": { - "css-tree": { - "version": "1.0.0-alpha.39", - "resolved": "https://registry.npmjs.org/css-tree/-/css-tree-1.0.0-alpha.39.tgz", - "integrity": "sha512-7UvkEYgBAHRG9Nt980lYxjsTrCyHFN53ky3wVsDkiMdVqylqRt+Zc+jm5qw7/qyOvN2dHSYtX0e4MbCCExSvnA==", - "dev": true, - "requires": { - "mdn-data": "2.0.6", - "source-map": "^0.6.1" - } - }, - "mdn-data": { - "version": "2.0.6", - "resolved": "https://registry.npmjs.org/mdn-data/-/mdn-data-2.0.6.tgz", - "integrity": "sha512-rQvjv71olwNHgiTbfPZFkJtjNMciWgswYeciZhtvWLO8bmX3TnhyA62I6sTWOyZssWHJJjY6/KiWwqQsWWsqOA==", - "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==", - "dev": true - } - } - }, - "custom-event": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/custom-event/-/custom-event-1.0.1.tgz", - "integrity": "sha1-XQKkaFCt8bSjF5RqOSj8y1v9BCU=", - "dev": true - }, - "cyclist": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/cyclist/-/cyclist-1.0.1.tgz", - "integrity": "sha1-WW6WmP0MgOEgOMK4LW6xs1tiJNk=", - "dev": true - }, - "damerau-levenshtein": { - "version": "1.0.6", - "resolved": "https://registry.npmjs.org/damerau-levenshtein/-/damerau-levenshtein-1.0.6.tgz", - "integrity": "sha512-JVrozIeElnj3QzfUIt8tB8YMluBJom4Vw9qTPpjGYQ9fYlB3D/rb6OordUxf3xeFB35LKWs0xqcO5U6ySvBtug==", - "dev": true - }, - "dashdash": { - "version": "1.14.1", - "resolved": "https://registry.npmjs.org/dashdash/-/dashdash-1.14.1.tgz", - "integrity": "sha1-hTz6D3y+L+1d4gMmuN1YEDX24vA=", - "dev": true, - "requires": { - "assert-plus": "^1.0.0" - } - }, - "date-format": { - "version": "4.0.3", - "resolved": "https://registry.npmjs.org/date-format/-/date-format-4.0.3.tgz", - "integrity": "sha512-7P3FyqDcfeznLZp2b+OMitV9Sz2lUnsT87WaTat9nVwqsBkTzPG3lPLNwW3en6F4pHUiWzr6vb8CLhjdK9bcxQ==", - "dev": true - }, - "debug": { - "version": "4.1.1", - "resolved": "https://registry.npmjs.org/debug/-/debug-4.1.1.tgz", - "integrity": "sha512-pYAIzeRo8J6KPEaJ0VWOh5Pzkbw/RetuzehGM7QRRX5he4fPHx2rdKMB256ehJCkX+XRQm16eZLqLNS8RSZXZw==", - "dev": true, - "requires": { - "ms": "^2.1.1" - } - }, - "debuglog": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/debuglog/-/debuglog-1.0.1.tgz", - "integrity": "sha1-qiT/uaw9+aI1GDfPstJ5NgzXhJI=", - "dev": true - }, - "decamelize": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/decamelize/-/decamelize-1.2.0.tgz", - "integrity": "sha1-9lNNFRSCabIDUue+4m9QH5oZEpA=", - "dev": true - }, - "decode-uri-component": { - "version": "0.2.0", - "resolved": "https://registry.npmjs.org/decode-uri-component/-/decode-uri-component-0.2.0.tgz", - "integrity": "sha1-6zkTMzRYd1y4TNGh+uBiEGu4dUU=", - "dev": true - }, - "deep-equal": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/deep-equal/-/deep-equal-1.1.1.tgz", - "integrity": "sha512-yd9c5AdiqVcR+JjcwUQb9DkhJc8ngNr0MahEBGvDiJw8puWab2yZlh+nkasOnZP+EGTAP6rRp2JzJhJZzvNF8g==", - "dev": true, - "requires": { - "is-arguments": "^1.0.4", - "is-date-object": "^1.0.1", - "is-regex": "^1.0.4", - "object-is": "^1.0.1", - "object-keys": "^1.1.1", - "regexp.prototype.flags": "^1.2.0" - } - }, - "default-gateway": { - "version": "4.2.0", - "resolved": "https://registry.npmjs.org/default-gateway/-/default-gateway-4.2.0.tgz", - "integrity": "sha512-h6sMrVB1VMWVrW13mSc6ia/DwYYw5MN6+exNu1OaJeFac5aSAvwM7lZ0NVfTABuSkQelr4h5oebg3KB1XPdjgA==", - "dev": true, - "requires": { - "execa": "^1.0.0", - "ip-regex": "^2.1.0" - } - }, - "default-require-extensions": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/default-require-extensions/-/default-require-extensions-2.0.0.tgz", - "integrity": "sha1-9fj7sYp9bVCyH2QfZJ67Uiz+JPc=", - "dev": true, - "requires": { - "strip-bom": "^3.0.0" - } - }, - "defaults": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/defaults/-/defaults-1.0.3.tgz", - "integrity": "sha1-xlYFHpgX2f8I7YgUd/P+QBnz730=", - "dev": true, - "requires": { - "clone": "^1.0.2" - }, - "dependencies": { - "clone": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/clone/-/clone-1.0.4.tgz", - "integrity": "sha1-2jCcwmPfFZlMaIypAheco8fNfH4=", - "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" - } - }, - "define-property": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/define-property/-/define-property-2.0.2.tgz", - "integrity": "sha512-jwK2UV4cnPpbcG7+VRARKTZPUWowwXA8bzH5NP6ud0oeAxyYPuGZUAC7hMugpCdz4BeSZl2Dl9k66CHJ/46ZYQ==", - "dev": true, - "requires": { - "is-descriptor": "^1.0.2", - "isobject": "^3.0.1" - }, - "dependencies": { - "is-accessor-descriptor": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-1.0.0.tgz", - "integrity": "sha512-m5hnHTkcVsPfqx3AKlyttIPb7J+XykHvJP2B9bZDjlhLIoEq4XoK64Vg7boZlVWYK6LUY94dYPEE7Lh0ZkZKcQ==", - "dev": true, - "requires": { - "kind-of": "^6.0.0" - } - }, - "is-data-descriptor": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-1.0.0.tgz", - "integrity": "sha512-jbRXy1FmtAoCjQkVmIVYwuuqDFUbaOeDjmed1tOGPrsMhtJA4rD9tkgA0F1qJ3gRFRXcHYVkdeaP50Q5rE/jLQ==", - "dev": true, - "requires": { - "kind-of": "^6.0.0" - } - }, - "is-descriptor": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-1.0.2.tgz", - "integrity": "sha512-2eis5WqQGV7peooDyLmNEPUrps9+SXX5c9pL3xEB+4e9HnGuDa7mB7kHxHw4CbqS9k1T2hOH3miL8n8WtiYVtg==", - "dev": true, - "requires": { - "is-accessor-descriptor": "^1.0.0", - "is-data-descriptor": "^1.0.0", - "kind-of": "^6.0.2" - } - } - } - }, - "del": { - "version": "4.1.1", - "resolved": "https://registry.npmjs.org/del/-/del-4.1.1.tgz", - "integrity": "sha512-QwGuEUouP2kVwQenAsOof5Fv8K9t3D8Ca8NxcXKrIpEHjTXK5J2nXLdP+ALI1cgv8wj7KuwBhTwBkOZSJKM5XQ==", - "dev": true, - "requires": { - "@types/glob": "^7.1.1", - "globby": "^6.1.0", - "is-path-cwd": "^2.0.0", - "is-path-in-cwd": "^2.0.0", - "p-map": "^2.0.0", - "pify": "^4.0.1", - "rimraf": "^2.6.3" - }, - "dependencies": { - "globby": { - "version": "6.1.0", - "resolved": "https://registry.npmjs.org/globby/-/globby-6.1.0.tgz", - "integrity": "sha1-9abXDoOV4hyFj7BInWTfAkJNUGw=", - "dev": true, - "requires": { - "array-union": "^1.0.1", - "glob": "^7.0.3", - "object-assign": "^4.0.1", - "pify": "^2.0.0", - "pinkie-promise": "^2.0.0" - }, - "dependencies": { - "pify": { - "version": "2.3.0", - "resolved": "https://registry.npmjs.org/pify/-/pify-2.3.0.tgz", - "integrity": "sha1-7RQaasBDqEnqWISY59yosVMw6Qw=", - "dev": true - } - } - }, - "p-map": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/p-map/-/p-map-2.1.0.tgz", - "integrity": "sha512-y3b8Kpd8OAN444hxfBbFfj1FY/RjtTd8tzYwhUqNYXx0fXx2iX4maP4Qr6qhIKbQXI02wTLAda4fYUbDagTUFw==", - "dev": true - }, - "rimraf": { - "version": "2.7.1", - "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", - "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==", - "dev": true, - "requires": { - "glob": "^7.1.3" - } - } - } - }, - "delayed-stream": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", - "integrity": "sha1-3zrhmayt+31ECqrgsp4icrJOxhk=", - "dev": true - }, - "depd": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.2.tgz", - "integrity": "sha1-m81S4UwJd2PnSbJ0xDRu0uVgtak=", - "dev": true - }, - "dependency-graph": { - "version": "0.7.2", - "resolved": "https://registry.npmjs.org/dependency-graph/-/dependency-graph-0.7.2.tgz", - "integrity": "sha512-KqtH4/EZdtdfWX0p6MGP9jljvxSY6msy/pRUD4jgNwVpv3v1QmNLlsB3LDSSUg79BRVSn7jI1QPRtArGABovAQ==", - "dev": true - }, - "des.js": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/des.js/-/des.js-1.0.1.tgz", - "integrity": "sha512-Q0I4pfFrv2VPd34/vfLrFOoRmlYj3OV50i7fskps1jZWK1kApMWWT9G6RRUeYedLcBDIhnSDaUvJMb3AhUlaEA==", - "dev": true, - "requires": { - "inherits": "^2.0.1", - "minimalistic-assert": "^1.0.0" - } - }, - "destroy": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.0.4.tgz", - "integrity": "sha1-l4hXRCxEdJ5CBmE+N5RiBYJqvYA=", - "dev": true - }, - "detect-node": { - "version": "2.0.4", - "resolved": "https://registry.npmjs.org/detect-node/-/detect-node-2.0.4.tgz", - "integrity": "sha512-ZIzRpLJrOj7jjP2miAtgqIfmzbxa4ZOr5jJc601zklsfEx9oTzmmj2nVpIPRpNlRTIh8lc1kyViIY7BWSGNmKw==", - "dev": true - }, - "dezalgo": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/dezalgo/-/dezalgo-1.0.3.tgz", - "integrity": "sha1-f3Qt4Gb8dIvI24IFad3c5Jvw1FY=", - "dev": true, - "requires": { - "asap": "^2.0.0", - "wrappy": "1" - } - }, - "di": { - "version": "0.0.1", - "resolved": "https://registry.npmjs.org/di/-/di-0.0.1.tgz", - "integrity": "sha1-gGZJMmzqp8qjMG112YXqJ0i6kTw=", - "dev": true - }, - "diff": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/diff/-/diff-4.0.2.tgz", - "integrity": "sha512-58lmxKSA4BNyLz+HHMUzlOEpg09FV+ev6ZMe3vJihgdxzgcwZ8VoEEPmALCZG9LmqfVoNMMKpttIYTVG6uDY7A==", - "dev": true - }, - "diffie-hellman": { - "version": "5.0.3", - "resolved": "https://registry.npmjs.org/diffie-hellman/-/diffie-hellman-5.0.3.tgz", - "integrity": "sha512-kqag/Nl+f3GwyK25fhUMYj81BUOrZ9IuJsjIcDE5icNM9FJHAVm3VcUDxdLPoQtTuUylWm6ZIknYJwwaPxsUzg==", - "dev": true, - "requires": { - "bn.js": "^4.1.0", - "miller-rabin": "^4.0.0", - "randombytes": "^2.0.0" - }, - "dependencies": { - "bn.js": { - "version": "4.11.8", - "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.11.8.tgz", - "integrity": "sha512-ItfYfPLkWHUjckQCk8xC+LwxgK8NYcXywGigJgSwOP8Y2iyWT4f2vsZnoOXTTbo+o5yXmIUJ4gn5538SO5S3gA==", - "dev": true - } - } - }, - "dir-glob": { - "version": "2.2.2", - "resolved": "https://registry.npmjs.org/dir-glob/-/dir-glob-2.2.2.tgz", - "integrity": "sha512-f9LBi5QWzIW3I6e//uxZoLBlUt9kcp66qo0sSCxL6YZKc75R1c4MFCoe/LaZiBGmgujvQdxc5Bn3QhfyvK5Hsw==", - "dev": true, - "requires": { - "path-type": "^3.0.0" - } - }, - "dns-equal": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/dns-equal/-/dns-equal-1.0.0.tgz", - "integrity": "sha1-s55/HabrCnW6nBcySzR1PEfgZU0=", - "dev": true - }, - "dns-packet": { - "version": "1.3.1", - "resolved": "https://registry.npmjs.org/dns-packet/-/dns-packet-1.3.1.tgz", - "integrity": "sha512-0UxfQkMhYAUaZI+xrNZOz/as5KgDU0M/fQ9b6SpkyLbk3GEswDi6PADJVaYJradtRVsRIlF1zLyOodbcTCDzUg==", - "dev": true, - "requires": { - "ip": "^1.1.0", - "safe-buffer": "^5.0.1" - } - }, - "dns-txt": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/dns-txt/-/dns-txt-2.0.2.tgz", - "integrity": "sha1-uR2Ab10nGI5Ks+fRB9iBocxGQrY=", - "dev": true, - "requires": { - "buffer-indexof": "^1.0.0" - } - }, - "dom-serialize": { - "version": "2.2.1", - "resolved": "https://registry.npmjs.org/dom-serialize/-/dom-serialize-2.2.1.tgz", - "integrity": "sha1-ViromZ9Evl6jB29UGdzVnrQ6yVs=", - "dev": true, - "requires": { - "custom-event": "~1.0.0", - "ent": "~2.2.0", - "extend": "^3.0.0", - "void-elements": "^2.0.0" - } - }, - "dom-serializer": { - "version": "0.2.2", - "resolved": "https://registry.npmjs.org/dom-serializer/-/dom-serializer-0.2.2.tgz", - "integrity": "sha512-2/xPb3ORsQ42nHYiSunXkDjPLBaEj/xTwUO4B7XCZQTRk7EBtTOPaygh10YAAh2OI1Qrp6NWfpAhzswj0ydt9g==", - "dev": true, - "requires": { - "domelementtype": "^2.0.1", - "entities": "^2.0.0" - }, - "dependencies": { - "domelementtype": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/domelementtype/-/domelementtype-2.0.1.tgz", - "integrity": "sha512-5HOHUDsYZWV8FGWN0Njbr/Rn7f/eWSQi1v7+HsUVwXgn8nWWlL64zKDkS0n8ZmQ3mlWOMuXOnR+7Nx/5tMO5AQ==", - "dev": true - } - } - }, - "domain-browser": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/domain-browser/-/domain-browser-1.2.0.tgz", - "integrity": "sha512-jnjyiM6eRyZl2H+W8Q/zLMA481hzi0eszAaBUzIVnmYVDBbnLxVNnfu1HgEBvCbL+71FrxMl3E6lpKH7Ge3OXA==", - "dev": true - }, - "domelementtype": { - "version": "1.3.1", - "resolved": "https://registry.npmjs.org/domelementtype/-/domelementtype-1.3.1.tgz", - "integrity": "sha512-BSKB+TSpMpFI/HOxCNr1O8aMOTZ8hT3pM3GQ0w/mWRmkhEDSFJkkyzz4XQsBV44BChwGkrDfMyjVD0eA2aFV3w==", - "dev": true - }, - "domutils": { - "version": "1.7.0", - "resolved": "https://registry.npmjs.org/domutils/-/domutils-1.7.0.tgz", - "integrity": "sha512-Lgd2XcJ/NjEw+7tFvfKxOzCYKZsdct5lczQ2ZaQY8Djz7pfAD3Gbp8ySJWtreII/vDlMVmxwa6pHmdxIYgttDg==", - "dev": true, - "requires": { - "dom-serializer": "0", - "domelementtype": "1" - } - }, - "dot-prop": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/dot-prop/-/dot-prop-5.2.0.tgz", - "integrity": "sha512-uEUyaDKoSQ1M4Oq8l45hSE26SnTxL6snNnqvK/VWx5wJhmff5z0FUVJDKDanor/6w3kzE3i7XZOk+7wC0EXr1A==", - "dev": true, - "requires": { - "is-obj": "^2.0.0" - } - }, - "duplexify": { - "version": "3.7.1", - "resolved": "https://registry.npmjs.org/duplexify/-/duplexify-3.7.1.tgz", - "integrity": "sha512-07z8uv2wMyS51kKhD1KsdXJg5WQ6t93RneqRxUHnskXVtlYYkLqM0gqStQZ3pj073g687jPCHrqNfCzawLYh5g==", - "dev": true, - "requires": { - "end-of-stream": "^1.0.0", - "inherits": "^2.0.1", - "readable-stream": "^2.0.0", - "stream-shift": "^1.0.0" - } - }, - "ecc-jsbn": { - "version": "0.1.2", - "resolved": "https://registry.npmjs.org/ecc-jsbn/-/ecc-jsbn-0.1.2.tgz", - "integrity": "sha1-OoOpBOVDUyh4dMVkt1SThoSamMk=", - "dev": true, - "requires": { - "jsbn": "~0.1.0", - "safer-buffer": "^2.1.0" - } - }, - "ee-first": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", - "integrity": "sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0=", - "dev": true - }, - "electron-to-chromium": { - "version": "1.3.428", - "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.3.428.tgz", - "integrity": "sha512-u3+5jEfgLKq/hGO96YfAoOAM1tgFnRDTCD5mLuev44tttcXix+INtVegAkmGzUcfDsnzkPt51XXurXZLLwXt0w==", - "dev": true - }, - "elliptic": { - "version": "6.5.2", - "resolved": "https://registry.npmjs.org/elliptic/-/elliptic-6.5.2.tgz", - "integrity": "sha512-f4x70okzZbIQl/NSRLkI/+tteV/9WqL98zx+SQ69KbXxmVrmjwsNUPn/gYJJ0sHvEak24cZgHIPegRePAtA/xw==", - "dev": true, - "requires": { - "bn.js": "^4.4.0", - "brorand": "^1.0.1", - "hash.js": "^1.0.0", - "hmac-drbg": "^1.0.0", - "inherits": "^2.0.1", - "minimalistic-assert": "^1.0.0", - "minimalistic-crypto-utils": "^1.0.0" - }, - "dependencies": { - "bn.js": { - "version": "4.11.8", - "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.11.8.tgz", - "integrity": "sha512-ItfYfPLkWHUjckQCk8xC+LwxgK8NYcXywGigJgSwOP8Y2iyWT4f2vsZnoOXTTbo+o5yXmIUJ4gn5538SO5S3gA==", - "dev": true - } - } - }, - "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 - }, - "emojis-list": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/emojis-list/-/emojis-list-3.0.0.tgz", - "integrity": "sha512-/kyM18EfinwXZbno9FyUGeFh87KC8HRQBQGildHZbEuRyWFOmv1U10o9BBp8XVZDVNNuQKyIGIu5ZYAAXJ0V2Q==", - "dev": true - }, - "encodeurl": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz", - "integrity": "sha1-rT/0yG7C0CkyL1oCw6mmBslbP1k=", - "dev": true - }, - "encoding": { - "version": "0.1.12", - "resolved": "https://registry.npmjs.org/encoding/-/encoding-0.1.12.tgz", - "integrity": "sha1-U4tm8+5izRq1HsMjgp0flIDHS+s=", - "dev": true, - "requires": { - "iconv-lite": "~0.4.13" - } - }, - "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" - } - }, - "engine.io": { - "version": "6.1.3", - "resolved": "https://registry.npmjs.org/engine.io/-/engine.io-6.1.3.tgz", - "integrity": "sha512-rqs60YwkvWTLLnfazqgZqLa/aKo+9cueVfEi/dZ8PyGyaf8TLOxj++4QMIgeG3Gn0AhrWiFXvghsoY9L9h25GA==", - "dev": true, - "requires": { - "@types/cookie": "^0.4.1", - "@types/cors": "^2.8.12", - "@types/node": ">=10.0.0", - "accepts": "~1.3.4", - "base64id": "2.0.0", - "cookie": "~0.4.1", - "cors": "~2.8.5", - "debug": "~4.3.1", - "engine.io-parser": "~5.0.3", - "ws": "~8.2.3" - }, - "dependencies": { - "cookie": { - "version": "0.4.2", - "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.4.2.tgz", - "integrity": "sha512-aSWTXFzaKWkvHO1Ny/s+ePFpvKsPnjc551iI41v3ny/ow6tBG5Vd+FuqGNhh1LxOmVzOlGUriIlOaokOvhaStA==", - "dev": true - }, - "debug": { - "version": "4.3.3", - "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.3.tgz", - "integrity": "sha512-/zxw5+vh1Tfv+4Qn7a5nsbcJKPaSvCDhojn6FEl9vupwK2VCSDtEiEtqr8DFtzYFOdz63LBkxec7DYuc2jon6Q==", - "dev": true, - "requires": { - "ms": "2.1.2" - } - }, - "ws": { - "version": "8.2.3", - "resolved": "https://registry.npmjs.org/ws/-/ws-8.2.3.tgz", - "integrity": "sha512-wBuoj1BDpC6ZQ1B7DWQBYVLphPWkm8i9Y0/3YdHjHKHiohOJ1ws+3OccDWtH+PoC9DZD5WOTrJvNbWvjS6JWaA==", - "dev": true - } - } - }, - "engine.io-parser": { - "version": "5.0.3", - "resolved": "https://registry.npmjs.org/engine.io-parser/-/engine.io-parser-5.0.3.tgz", - "integrity": "sha512-BtQxwF27XUNnSafQLvDi0dQ8s3i6VgzSoQMJacpIcGNrlUdfHSKbgm3jmjCVvQluGzqwujQMPAoMai3oYSTurg==", - "dev": true, - "requires": { - "@socket.io/base64-arraybuffer": "~1.0.2" - } - }, - "enhanced-resolve": { - "version": "4.1.1", - "resolved": "https://registry.npmjs.org/enhanced-resolve/-/enhanced-resolve-4.1.1.tgz", - "integrity": "sha512-98p2zE+rL7/g/DzMHMTF4zZlCgeVdJ7yr6xzEpJRYwFYrGi9ANdn5DnJURg6RpBkyk60XYDnWIv51VfIhfNGuA==", - "dev": true, - "requires": { - "graceful-fs": "^4.1.2", - "memory-fs": "^0.5.0", - "tapable": "^1.0.0" - } - }, - "ent": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/ent/-/ent-2.2.0.tgz", - "integrity": "sha1-6WQhkyWiHQX0RGai9obtbOX13R0=", - "dev": true - }, - "entities": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/entities/-/entities-2.0.0.tgz", - "integrity": "sha512-D9f7V0JSRwIxlRI2mjMqufDrRDnx8p+eEOz7aUM9SuvF8gsBzra0/6tbjl1m8eQHrZlYj6PxqE00hZ1SAIKPLw==", - "dev": true - }, - "err-code": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/err-code/-/err-code-1.1.2.tgz", - "integrity": "sha1-BuARbTAo9q70gGhJ6w6mp0iuaWA=", - "dev": true - }, - "errno": { - "version": "0.1.7", - "resolved": "https://registry.npmjs.org/errno/-/errno-0.1.7.tgz", - "integrity": "sha512-MfrRBDWzIWifgq6tJj60gkAwtLNb6sQPlcFrSOflcP1aFmmruKQ2wRnze/8V6kgyz7H3FF8Npzv78mZ7XLLflg==", - "dev": true, - "requires": { - "prr": "~1.0.1" - } - }, - "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==", - "dev": true, - "requires": { - "is-arrayish": "^0.2.1" - } - }, - "es-abstract": { - "version": "1.17.5", - "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.17.5.tgz", - "integrity": "sha512-BR9auzDbySxOcfog0tLECW8l28eRGpDpU3Dm3Hp4q/N+VtLTmyj4EUN088XZWQDW/hzj6sYRDXeOFsaAODKvpg==", - "dev": true, - "requires": { - "es-to-primitive": "^1.2.1", - "function-bind": "^1.1.1", - "has": "^1.0.3", - "has-symbols": "^1.0.1", - "is-callable": "^1.1.5", - "is-regex": "^1.0.5", - "object-inspect": "^1.7.0", - "object-keys": "^1.1.1", - "object.assign": "^4.1.0", - "string.prototype.trimleft": "^2.1.1", - "string.prototype.trimright": "^2.1.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" - } - }, - "es6-promise": { - "version": "4.2.8", - "resolved": "https://registry.npmjs.org/es6-promise/-/es6-promise-4.2.8.tgz", - "integrity": "sha512-HJDGx5daxeIvxdBxvG2cb9g4tEvwIk3i8+nhX0yGrYmZUzbkdg8QbDevheDB8gd0//uPj4c1EQua8Q+MViT0/w==", - "dev": true - }, - "es6-promisify": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/es6-promisify/-/es6-promisify-5.0.0.tgz", - "integrity": "sha1-UQnWLz5W6pZ8S2NQWu8IKRyKUgM=", - "dev": true, - "requires": { - "es6-promise": "^4.0.3" - } - }, - "escalade": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.1.1.tgz", - "integrity": "sha512-k0er2gUkLf8O0zKJiAhmkTnJlTvINGv7ygDNPbeIsX/TJjGJZHuh9B2UxbsaEkmlEo9MfhrSzmhIlhRlI2GXnw==", - "dev": true - }, - "escape-html": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz", - "integrity": "sha1-Aljq5NPQwJdN4cFpGI7wBR0dGYg=", - "dev": true - }, - "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-scope": { - "version": "4.0.3", - "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-4.0.3.tgz", - "integrity": "sha512-p7VutNr1O/QrxysMo3E45FjYDTeXBy0iTltPFNSqKAIfjDSXC+4dj+qfyuD8bfAXrW/y6lW3O76VaYNPKfpKrg==", - "dev": true, - "requires": { - "esrecurse": "^4.1.0", - "estraverse": "^4.1.1" - } - }, - "esprima": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz", - "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==", - "dev": true - }, - "esrecurse": { - "version": "4.2.1", - "resolved": "https://registry.npmjs.org/esrecurse/-/esrecurse-4.2.1.tgz", - "integrity": "sha512-64RBB++fIOAXPw3P9cy89qfMlvZEXZkqqJkjqqXIvzP5ezRZjW+lPWjw35UX/3EhUPFYbg5ER4JYgDw4007/DQ==", - "dev": true, - "requires": { - "estraverse": "^4.1.0" - } - }, - "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=", - "dev": true - }, - "eventemitter3": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/eventemitter3/-/eventemitter3-4.0.0.tgz", - "integrity": "sha512-qerSRB0p+UDEssxTtm6EDKcE7W4OaoisfIMl4CngyEhjpYglocpNg6UEqCvemdGhosAsg4sO2dXJOdyBifPGCg==", - "dev": true - }, - "events": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/events/-/events-3.1.0.tgz", - "integrity": "sha512-Rv+u8MLHNOdMjTAFeT3nCjHn2aGlx435FP/sDHNaRhDEMwyI/aB22Kj2qIN8R0cw3z28psEQLYwxVKLsKrMgWg==", - "dev": true - }, - "eventsource": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/eventsource/-/eventsource-1.1.1.tgz", - "integrity": "sha512-qV5ZC0h7jYIAOhArFJgSfdyz6rALJyb270714o7ZtNnw2WSJ+eexhKtE0O8LYPRsHZHf2osHKZBxGPvm3kPkCA==", - "dev": true, - "requires": { - "original": "^1.0.0" - } - }, - "evp_bytestokey": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/evp_bytestokey/-/evp_bytestokey-1.0.3.tgz", - "integrity": "sha512-/f2Go4TognH/KvCISP7OUsHn85hT9nUkxxA9BEWxFn+Oj9o8ZNLm/40hdlgSLyuOimsrTKLUMEorQexp/aPQeA==", - "dev": true, - "requires": { - "md5.js": "^1.3.4", - "safe-buffer": "^5.1.1" - } - }, - "execa": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/execa/-/execa-1.0.0.tgz", - "integrity": "sha512-adbxcyWV46qiHyvSp50TKt05tB4tK3HcmF7/nxfAdhnox83seTDbwnaqKO4sXRy7roHAIFqJP/Rw/AuEbX61LA==", - "dev": true, - "requires": { - "cross-spawn": "^6.0.0", - "get-stream": "^4.0.0", - "is-stream": "^1.1.0", - "npm-run-path": "^2.0.0", - "p-finally": "^1.0.0", - "signal-exit": "^3.0.0", - "strip-eof": "^1.0.0" - } - }, - "exit": { - "version": "0.1.2", - "resolved": "https://registry.npmjs.org/exit/-/exit-0.1.2.tgz", - "integrity": "sha1-BjJjj42HfMghB9MKD/8aF8uhzQw=", - "dev": true - }, - "expand-brackets": { - "version": "2.1.4", - "resolved": "https://registry.npmjs.org/expand-brackets/-/expand-brackets-2.1.4.tgz", - "integrity": "sha1-t3c14xXOMPa27/D4OwQVGiJEliI=", - "dev": true, - "requires": { - "debug": "^2.3.3", - "define-property": "^0.2.5", - "extend-shallow": "^2.0.1", - "posix-character-classes": "^0.1.0", - "regex-not": "^1.0.0", - "snapdragon": "^0.8.1", - "to-regex": "^3.0.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" - } - }, - "define-property": { - "version": "0.2.5", - "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", - "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", - "dev": true, - "requires": { - "is-descriptor": "^0.1.0" - } - }, - "extend-shallow": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", - "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", - "dev": true, - "requires": { - "is-extendable": "^0.1.0" - } - }, - "ms": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", - "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", - "dev": true - } - } - }, - "express": { - "version": "4.17.1", - "resolved": "https://registry.npmjs.org/express/-/express-4.17.1.tgz", - "integrity": "sha512-mHJ9O79RqluphRrcw2X/GTh3k9tVv8YcoyY4Kkh4WDMUYKRZUq0h1o0w2rrrxBqM7VoeUVqgb27xlEMXTnYt4g==", - "dev": true, - "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": { - "array-flatten": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz", - "integrity": "sha1-ml9pkFGx5wczKPKgCJaLZOopVdI=", - "dev": true - }, - "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 - }, - "qs": { - "version": "6.7.0", - "resolved": "https://registry.npmjs.org/qs/-/qs-6.7.0.tgz", - "integrity": "sha512-VCdBRNFTX1fyE7Nb6FYoURo/SPe62QCaAyzJvUjwRaIsc+NePBEniHlvxFmmX56+HZphIGtV0XeCirBtpDrTyQ==", - "dev": true - } - } - }, - "extend": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/extend/-/extend-3.0.2.tgz", - "integrity": "sha512-fjquC59cD7CyW6urNXK0FBufkZcoiGG80wTuPujX590cB5Ttln20E2UB4S/WARVqhXffZl2LNgS+gQdPIIim/g==", - "dev": true - }, - "extend-shallow": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-3.0.2.tgz", - "integrity": "sha1-Jqcarwc7OfshJxcnRhMcJwQCjbg=", - "dev": true, - "requires": { - "assign-symbols": "^1.0.0", - "is-extendable": "^1.0.1" - }, - "dependencies": { - "is-extendable": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/is-extendable/-/is-extendable-1.0.1.tgz", - "integrity": "sha512-arnXMxT1hhoKo9k1LZdmlNyJdDDfy2v0fXjFlmok4+i8ul/6WlbVge9bhM74OpNPQPMGUToDtz+KXa1PneJxOA==", - "dev": true, - "requires": { - "is-plain-object": "^2.0.4" - } - } - } - }, - "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" - } - }, - "extglob": { - "version": "2.0.4", - "resolved": "https://registry.npmjs.org/extglob/-/extglob-2.0.4.tgz", - "integrity": "sha512-Nmb6QXkELsuBr24CJSkilo6UHHgbekK5UiZgfE6UHD3Eb27YC6oD+bhcT+tJ6cl8dmsgdQxnWlcry8ksBIBLpw==", - "dev": true, - "requires": { - "array-unique": "^0.3.2", - "define-property": "^1.0.0", - "expand-brackets": "^2.1.4", - "extend-shallow": "^2.0.1", - "fragment-cache": "^0.2.1", - "regex-not": "^1.0.0", - "snapdragon": "^0.8.1", - "to-regex": "^3.0.1" - }, - "dependencies": { - "define-property": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/define-property/-/define-property-1.0.0.tgz", - "integrity": "sha1-dp66rz9KY6rTr56NMEybvnm/sOY=", - "dev": true, - "requires": { - "is-descriptor": "^1.0.0" - } - }, - "extend-shallow": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", - "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", - "dev": true, - "requires": { - "is-extendable": "^0.1.0" - } - }, - "is-accessor-descriptor": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-1.0.0.tgz", - "integrity": "sha512-m5hnHTkcVsPfqx3AKlyttIPb7J+XykHvJP2B9bZDjlhLIoEq4XoK64Vg7boZlVWYK6LUY94dYPEE7Lh0ZkZKcQ==", - "dev": true, - "requires": { - "kind-of": "^6.0.0" - } - }, - "is-data-descriptor": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-1.0.0.tgz", - "integrity": "sha512-jbRXy1FmtAoCjQkVmIVYwuuqDFUbaOeDjmed1tOGPrsMhtJA4rD9tkgA0F1qJ3gRFRXcHYVkdeaP50Q5rE/jLQ==", - "dev": true, - "requires": { - "kind-of": "^6.0.0" - } - }, - "is-descriptor": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-1.0.2.tgz", - "integrity": "sha512-2eis5WqQGV7peooDyLmNEPUrps9+SXX5c9pL3xEB+4e9HnGuDa7mB7kHxHw4CbqS9k1T2hOH3miL8n8WtiYVtg==", - "dev": true, - "requires": { - "is-accessor-descriptor": "^1.0.0", - "is-data-descriptor": "^1.0.0", - "kind-of": "^6.0.2" - } - } - } - }, - "extsprintf": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/extsprintf/-/extsprintf-1.3.0.tgz", - "integrity": "sha1-lpGEQOMEGnpBT4xS48V06zw+HgU=", - "dev": true - }, - "fast-deep-equal": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-2.0.1.tgz", - "integrity": "sha1-ewUhjd+WZ79/Nwv3/bLLFf3Qqkk=", - "dev": true - }, - "fast-json-stable-stringify": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.0.0.tgz", - "integrity": "sha1-1RQsDK7msRifh9OnYREGT4bIu/I=", - "dev": true - }, - "fastparse": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/fastparse/-/fastparse-1.1.2.tgz", - "integrity": "sha512-483XLLxTVIwWK3QTrMGRqUfUpoOs/0hbQrl2oz4J0pAcm3A3bu84wxTFqGqkJzewCLdME38xJLJAxBABfQT8sQ==", - "dev": true - }, - "faye-websocket": { - "version": "0.10.0", - "resolved": "https://registry.npmjs.org/faye-websocket/-/faye-websocket-0.10.0.tgz", - "integrity": "sha1-TkkvjQTftviQA1B/btvy1QHnxvQ=", - "dev": true, - "requires": { - "websocket-driver": ">=0.5.1" - } - }, - "figgy-pudding": { - "version": "3.5.2", - "resolved": "https://registry.npmjs.org/figgy-pudding/-/figgy-pudding-3.5.2.tgz", - "integrity": "sha512-0btnI/H8f2pavGMN8w40mlSKOfTK2SVJmBfBeVIj3kNw0swwgzyRq0d5TJVOwodFmtvpPeWPN/MCcfuWF0Ezbw==", - "dev": true - }, - "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-loader": { - "version": "6.0.0", - "resolved": "https://registry.npmjs.org/file-loader/-/file-loader-6.0.0.tgz", - "integrity": "sha512-/aMOAYEFXDdjG0wytpTL5YQLfZnnTmLNjn+AIrJ/6HVnTfDqLsVKUUwkDf4I4kgex36BvjuXEn/TX9B/1ESyqQ==", - "dev": true, - "requires": { - "loader-utils": "^2.0.0", - "schema-utils": "^2.6.5" - } - }, - "file-uri-to-path": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/file-uri-to-path/-/file-uri-to-path-1.0.0.tgz", - "integrity": "sha512-0Zt+s3L7Vf1biwWZ29aARiVYLx7iMGnEUl9x33fbB/j3jR81u/O2LbqK+Bm1CDSNDKVtJ/YjwY7TUd5SkeLQLw==", - "dev": true, - "optional": true - }, - "fileset": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/fileset/-/fileset-2.0.3.tgz", - "integrity": "sha1-jnVIqW08wjJ+5eZ0FocjozO7oqA=", - "dev": true, - "requires": { - "glob": "^7.0.3", - "minimatch": "^3.0.3" - } - }, - "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==", - "dev": true, - "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==", - "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 - } - } - }, - "find-cache-dir": { - "version": "3.3.1", - "resolved": "https://registry.npmjs.org/find-cache-dir/-/find-cache-dir-3.3.1.tgz", - "integrity": "sha512-t2GDMt3oGC/v+BMwzmllWDuJF/xcDtE5j/fCGbqDD7OLuJkj0cfh1YSA5VKPvwMeLFLNDBkwOKZ2X85jGLVftQ==", - "dev": true, - "requires": { - "commondir": "^1.0.1", - "make-dir": "^3.0.2", - "pkg-dir": "^4.1.0" - }, - "dependencies": { - "find-up": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/find-up/-/find-up-4.1.0.tgz", - "integrity": "sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==", - "dev": true, - "requires": { - "locate-path": "^5.0.0", - "path-exists": "^4.0.0" - } - }, - "locate-path": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-5.0.0.tgz", - "integrity": "sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==", - "dev": true, - "requires": { - "p-locate": "^4.1.0" - } - }, - "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" - } - }, - "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": "4.1.0", - "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-4.1.0.tgz", - "integrity": "sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A==", - "dev": true, - "requires": { - "p-limit": "^2.2.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 - }, - "path-exists": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", - "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==", - "dev": true - }, - "pkg-dir": { - "version": "4.2.0", - "resolved": "https://registry.npmjs.org/pkg-dir/-/pkg-dir-4.2.0.tgz", - "integrity": "sha512-HRDzbaKjC+AOWVXxAU/x54COGeIv9eb+6CkDSQoNTt4XyWoIJvuPsXizxu/Fr23EiekbtZwmh1IcIG/l/a10GQ==", - "dev": true, - "requires": { - "find-up": "^4.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" - } - }, - "flatted": { - "version": "3.2.5", - "resolved": "https://registry.npmjs.org/flatted/-/flatted-3.2.5.tgz", - "integrity": "sha512-WIWGi2L3DyTUvUrwRKgGi9TwxQMUEqPOPQBVi71R96jZXJdFskXEmf54BoZaS1kknGODoIGASGEzBUYdyMCBJg==", - "dev": true - }, - "flush-write-stream": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/flush-write-stream/-/flush-write-stream-1.1.1.tgz", - "integrity": "sha512-3Z4XhFZ3992uIq0XOqb9AreonueSYphE6oYbpt5+3u06JWklbsPkNv3ZKkP9Bz/r+1MWCaMoSQ28P85+1Yc77w==", - "dev": true, - "requires": { - "inherits": "^2.0.3", - "readable-stream": "^2.3.6" - } - }, - "follow-redirects": { - "version": "1.14.8", - "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.14.8.tgz", - "integrity": "sha512-1x0S9UVJHsQprFcEC/qnNzBLcIxsjAV905f/UkQxbclCsoTWlacCNOpQa/anodLl2uaEKFhfWOvM2Qg77+15zA==", - "dev": true - }, - "for-in": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/for-in/-/for-in-1.0.2.tgz", - "integrity": "sha1-gQaNKVqBQuwKxybG4iAMMPttXoA=", - "dev": true - }, - "forever-agent": { - "version": "0.6.1", - "resolved": "https://registry.npmjs.org/forever-agent/-/forever-agent-0.6.1.tgz", - "integrity": "sha1-+8cfDEGt6zf5bFd60e1C2P2sypE=", - "dev": true - }, - "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==", - "dev": true, - "requires": { - "asynckit": "^0.4.0", - "combined-stream": "^1.0.6", - "mime-types": "^2.1.12" - } - }, - "forwarded": { - "version": "0.1.2", - "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.1.2.tgz", - "integrity": "sha1-mMI9qxF1ZXuMBXPozszZGw/xjIQ=", - "dev": true - }, - "fragment-cache": { - "version": "0.2.1", - "resolved": "https://registry.npmjs.org/fragment-cache/-/fragment-cache-0.2.1.tgz", - "integrity": "sha1-QpD60n8T6Jvn8zeZxrxaCr//DRk=", - "dev": true, - "requires": { - "map-cache": "^0.2.2" - } - }, - "fresh": { - "version": "0.5.2", - "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz", - "integrity": "sha1-PYyt2Q2XZWn6g1qx+OSyOhBWBac=", - "dev": true - }, - "from2": { - "version": "2.3.0", - "resolved": "https://registry.npmjs.org/from2/-/from2-2.3.0.tgz", - "integrity": "sha1-i/tVAr3kpNNs/e6gB/zKIdfjgq8=", - "dev": true, - "requires": { - "inherits": "^2.0.1", - "readable-stream": "^2.0.0" - } - }, - "fs-extra": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-4.0.2.tgz", - "integrity": "sha1-+RcExT0bRh+JNFKwwwfZmXZHq2s=", - "dev": true, - "requires": { - "graceful-fs": "^4.1.2", - "jsonfile": "^4.0.0", - "universalify": "^0.1.0" - } - }, - "fs-minipass": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/fs-minipass/-/fs-minipass-2.1.0.tgz", - "integrity": "sha512-V/JgOLFCS+R6Vcq0slCuaeWEdNC3ouDlJMNIsacH2VtALiu9mV4LPrHc5cDl8k5aw6J8jwgWWpiTo5RYhmIzvg==", - "dev": true, - "requires": { - "minipass": "^3.0.0" - } - }, - "fs-write-stream-atomic": { - "version": "1.0.10", - "resolved": "https://registry.npmjs.org/fs-write-stream-atomic/-/fs-write-stream-atomic-1.0.10.tgz", - "integrity": "sha1-tH31NJPvkR33VzHnCp3tAYnbQMk=", - "dev": true, - "requires": { - "graceful-fs": "^4.1.2", - "iferr": "^0.1.5", - "imurmurhash": "^0.1.4", - "readable-stream": "1 || 2" - } - }, - "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.1.3", - "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.1.3.tgz", - "integrity": "sha512-Auw9a4AxqWpa9GUfj370BMPzzyncfBABW8Mab7BGWBYDj4Isgq+cDKtx0i6u9jcX9pQDnswsaaOTgTmA5pEjuQ==", - "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==", - "dev": true - }, - "genfun": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/genfun/-/genfun-5.0.0.tgz", - "integrity": "sha512-KGDOARWVga7+rnB3z9Sd2Letx515owfk0hSxHGuqjANb1M+x2bGZGqHLiozPsYMdM2OubeMni/Hpwmjq6qIUhA==", - "dev": true - }, - "gensync": { - "version": "1.0.0-beta.1", - "resolved": "https://registry.npmjs.org/gensync/-/gensync-1.0.0-beta.1.tgz", - "integrity": "sha512-r8EC6NO1sngH/zdD9fiRDLdcgnbayXah+mLgManTaIZJqEC1MZstmnox8KpnI2/fxQwrp5OpCOYWLp4rBl4Jcg==", - "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==", - "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" - } - }, - "get-value": { - "version": "2.0.6", - "resolved": "https://registry.npmjs.org/get-value/-/get-value-2.0.6.tgz", - "integrity": "sha1-3BXKHGcjh8p2vTesCjlbogQqLCg=", - "dev": true - }, - "getpass": { - "version": "0.1.7", - "resolved": "https://registry.npmjs.org/getpass/-/getpass-0.1.7.tgz", - "integrity": "sha1-Xv+OPmhNVprkyysSgmBOi6YhSfo=", - "dev": true, - "requires": { - "assert-plus": "^1.0.0" - } - }, - "glob": { - "version": "7.1.5", - "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.5.tgz", - "integrity": "sha512-J9dlskqUXK1OeTOYBEn5s8aMukWMwWfs+rPTn/jn50Ux4MNXVhubL1wu/j2t+H4NVI+cXEcCaYellqaPVGXNqQ==", - "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" - } - }, - "glob-parent": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-3.1.0.tgz", - "integrity": "sha1-nmr2KZ2NO9K9QEMIMr0RPfkGxa4=", - "dev": true, - "requires": { - "is-glob": "^3.1.0", - "path-dirname": "^1.0.0" - }, - "dependencies": { - "is-glob": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-3.1.0.tgz", - "integrity": "sha1-e6WuJCF4BKxwcHuWkiVnSGzD6Eo=", - "dev": true, - "requires": { - "is-extglob": "^2.1.0" - } - } - } - }, - "globals": { - "version": "11.12.0", - "resolved": "https://registry.npmjs.org/globals/-/globals-11.12.0.tgz", - "integrity": "sha512-WOBp/EEGUiIsJSp7wcv/y6MO+lV9UoncWqxuFfm8eBwzWNgyfBd6Gz+IeKQ9jCmyhoH99g15M3T+QaVHFjizVA==", - "dev": true - }, - "globby": { - "version": "7.1.1", - "resolved": "https://registry.npmjs.org/globby/-/globby-7.1.1.tgz", - "integrity": "sha1-+yzP+UAfhgCUXfral0QMypcrhoA=", - "dev": true, - "requires": { - "array-union": "^1.0.1", - "dir-glob": "^2.0.0", - "glob": "^7.1.2", - "ignore": "^3.3.5", - "pify": "^3.0.0", - "slash": "^1.0.0" - }, - "dependencies": { - "pify": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/pify/-/pify-3.0.0.tgz", - "integrity": "sha1-5aSs0sEB/fPZpNB/DbxNtJ3SgXY=", - "dev": true - } - } - }, - "graceful-fs": { - "version": "4.2.4", - "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.4.tgz", - "integrity": "sha512-WjKPNJF79dtJAVniUlGGWHYGz2jWxT6VhN/4m1NdkbZ2nOsEF+cI1Edgql5zCRhs/VsQYRvrXctxktVXZUkixw==", - "dev": true - }, - "handle-thing": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/handle-thing/-/handle-thing-2.0.1.tgz", - "integrity": "sha512-9Qn4yBxelxoh2Ow62nP+Ka/kMnOXRi8BXnRaUwezLNhqelnN49xKz4F/dPP8OYLxLxq6JDtZb2i9XznUQbNPTg==", - "dev": true - }, - "har-schema": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/har-schema/-/har-schema-2.0.0.tgz", - "integrity": "sha1-qUwiJOvKwEeCoNkDVSHyRzW37JI=", - "dev": true - }, - "har-validator": { - "version": "5.1.3", - "resolved": "https://registry.npmjs.org/har-validator/-/har-validator-5.1.3.tgz", - "integrity": "sha512-sNvOCzEQNr/qrvJgc3UG/kD4QtlHycrzwS+6mfTrrSq97BvaYcPZZI1ZSqGSPR73Cxn4LKTD4PttRwfU7jWq5g==", - "dev": true, - "requires": { - "ajv": "^6.5.5", - "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==", - "dev": true, - "requires": { - "function-bind": "^1.1.1" - } - }, - "has-ansi": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/has-ansi/-/has-ansi-2.0.0.tgz", - "integrity": "sha1-NPUEnOHs3ysGSa8+8k5F7TVBbZE=", - "dev": true, - "requires": { - "ansi-regex": "^2.0.0" - } - }, - "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.1", - "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.0.1.tgz", - "integrity": "sha512-PLcsoqu++dmEIZB+6totNFKq/7Do+Z0u4oT0zKOJNl3lYK6vGwwu2hjHs+68OEZbTjiUE9bgOABXbP/GvrS0Kg==", - "dev": true - }, - "has-value": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/has-value/-/has-value-1.0.0.tgz", - "integrity": "sha1-GLKB2lhbHFxR3vJMkw7SmgvmsXc=", - "dev": true, - "requires": { - "get-value": "^2.0.6", - "has-values": "^1.0.0", - "isobject": "^3.0.0" - } - }, - "has-values": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/has-values/-/has-values-1.0.0.tgz", - "integrity": "sha1-lbC2P+whRmGab+V/51Yo1aOe/k8=", - "dev": true, - "requires": { - "is-number": "^3.0.0", - "kind-of": "^4.0.0" - }, - "dependencies": { - "is-number": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/is-number/-/is-number-3.0.0.tgz", - "integrity": "sha1-JP1iAaR4LPUFYcgQJ2r8fRLXEZU=", - "dev": true, - "requires": { - "kind-of": "^3.0.2" - }, - "dependencies": { - "kind-of": { - "version": "3.2.2", - "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", - "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", - "dev": true, - "requires": { - "is-buffer": "^1.1.5" - } - } - } - }, - "kind-of": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-4.0.0.tgz", - "integrity": "sha1-IIE989cSkosgc3hpGkUGb65y3Vc=", - "dev": true, - "requires": { - "is-buffer": "^1.1.5" - } - } - } - }, - "hash-base": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/hash-base/-/hash-base-3.1.0.tgz", - "integrity": "sha512-1nmYp/rhMDiE7AYkDw+lLwlAzz0AntGIe51F3RfFfEqyQ3feY2eI/NcwC6umIQVOASPMsWJLJScWKSSvzL9IVA==", - "dev": true, - "requires": { - "inherits": "^2.0.4", - "readable-stream": "^3.6.0", - "safe-buffer": "^5.2.0" - }, - "dependencies": { - "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==", - "dev": true, - "requires": { - "inherits": "^2.0.3", - "string_decoder": "^1.1.1", - "util-deprecate": "^1.0.1" - } - }, - "safe-buffer": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.0.tgz", - "integrity": "sha512-fZEwUGbVl7kouZs1jCdMLdt95hdIv0ZeHg6L7qPeciMZhZ+/gdesW4wgTARkrFWEpspjEATAzUGPG8N2jJiwbg==", - "dev": true - } - } - }, - "hash.js": { - "version": "1.1.7", - "resolved": "https://registry.npmjs.org/hash.js/-/hash.js-1.1.7.tgz", - "integrity": "sha512-taOaskGt4z4SOANNseOviYDvjEJinIkRgmp7LbKP2YTTmVxWBl87s/uzK9r+44BclBSp2X7K1hqeNfz9JbBeXA==", - "dev": true, - "requires": { - "inherits": "^2.0.3", - "minimalistic-assert": "^1.0.1" - } - }, - "hex-color-regex": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/hex-color-regex/-/hex-color-regex-1.1.0.tgz", - "integrity": "sha512-l9sfDFsuqtOqKDsQdqrMRk0U85RZc0RtOR9yPI7mRVOa4FsR/BVnZ0shmQRM96Ji99kYZP/7hn1cedc1+ApsTQ==", - "dev": true - }, - "highlight.js": { - "version": "10.4.1", - "resolved": "https://registry.npmjs.org/highlight.js/-/highlight.js-10.4.1.tgz", - "integrity": "sha512-yR5lWvNz7c85OhVAEAeFhVCc/GV4C30Fjzc/rCP0aCWzc1UUOPUk55dK/qdwTZHBvMZo+eZ2jpk62ndX/xMFlg==" - }, - "highlightjs-line-numbers.js": { - "version": "2.7.0", - "resolved": "https://registry.npmjs.org/highlightjs-line-numbers.js/-/highlightjs-line-numbers.js-2.7.0.tgz", - "integrity": "sha512-2kgZkfGy3TB6rF1o1XJtUThDyraLAFd6iiAIE1MEH89om8VwyHsal6BBnIOP2yOuq2TCuuGd0YD6aGHa6iq3/g==" - }, - "hmac-drbg": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/hmac-drbg/-/hmac-drbg-1.0.1.tgz", - "integrity": "sha1-0nRXAQJabHdabFRXk+1QL8DGSaE=", - "dev": true, - "requires": { - "hash.js": "^1.0.3", - "minimalistic-assert": "^1.0.0", - "minimalistic-crypto-utils": "^1.0.1" - } - }, - "hosted-git-info": { - "version": "3.0.8", - "resolved": "https://registry.npmjs.org/hosted-git-info/-/hosted-git-info-3.0.8.tgz", - "integrity": "sha512-aXpmwoOhRBrw6X3j0h5RloK4x1OzsxMPyxqIHyNfSe2pypkVTZFpEiRoSipPEPlMrh0HW/XsjkJ5WgnCirpNUw==", - "dev": true, - "requires": { - "lru-cache": "^6.0.0" - }, - "dependencies": { - "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==", - "dev": true, - "requires": { - "yallist": "^4.0.0" - } - } - } - }, - "hpack.js": { - "version": "2.1.6", - "resolved": "https://registry.npmjs.org/hpack.js/-/hpack.js-2.1.6.tgz", - "integrity": "sha1-h3dMCUnlE/QuhFdbPEVoH63ioLI=", - "dev": true, - "requires": { - "inherits": "^2.0.1", - "obuf": "^1.0.0", - "readable-stream": "^2.0.1", - "wbuf": "^1.1.0" - } - }, - "hsl-regex": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/hsl-regex/-/hsl-regex-1.0.0.tgz", - "integrity": "sha1-1JMwx4ntgZ4nakwNJy3/owsY/m4=", - "dev": true - }, - "hsla-regex": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/hsla-regex/-/hsla-regex-1.0.0.tgz", - "integrity": "sha1-wc56MWjIxmFAM6S194d/OyJfnDg=", - "dev": true - }, - "html-comment-regex": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/html-comment-regex/-/html-comment-regex-1.1.2.tgz", - "integrity": "sha512-P+M65QY2JQ5Y0G9KKdlDpo0zK+/OHptU5AaBwUfAIDJZk1MYf32Frm84EcOytfJE0t5JvkAnKlmjsXDnWzCJmQ==", - "dev": true - }, - "html-entities": { - "version": "1.3.1", - "resolved": "https://registry.npmjs.org/html-entities/-/html-entities-1.3.1.tgz", - "integrity": "sha512-rhE/4Z3hIhzHAUKbW8jVcCyuT5oJCXXqhN/6mXXVCpzTmvJnoH2HL/bt3EZ6p55jbFJBeAe1ZNpL5BugLujxNA==", - "dev": true - }, - "html-escaper": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/html-escaper/-/html-escaper-2.0.2.tgz", - "integrity": "sha512-H2iMtd0I4Mt5eYiapRdIDjp+XzelXQ0tFE4JS7YFwFevXXMmOp9myNrUvCg0D6ws8iqkRPBfKHgbwig1SmlLfg==", - "dev": true - }, - "http-cache-semantics": { - "version": "3.8.1", - "resolved": "https://registry.npmjs.org/http-cache-semantics/-/http-cache-semantics-3.8.1.tgz", - "integrity": "sha512-5ai2iksyV8ZXmnZhHH4rWPoxxistEexSi5936zIQ1bnNTW5VnA85B6P/VpXiRM017IgRvb2kKo1a//y+0wSp3w==", - "dev": true - }, - "http-deceiver": { - "version": "1.2.7", - "resolved": "https://registry.npmjs.org/http-deceiver/-/http-deceiver-1.2.7.tgz", - "integrity": "sha1-+nFolEq5pRnTN8sL7HKE3D5yPYc=", - "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==", - "dev": true, - "requires": { - "depd": "~1.1.2", - "inherits": "2.0.3", - "setprototypeof": "1.1.1", - "statuses": ">= 1.5.0 < 2", - "toidentifier": "1.0.0" - }, - "dependencies": { - "inherits": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", - "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=", - "dev": true - } - } - }, - "http-parser-js": { - "version": "0.4.10", - "resolved": "https://registry.npmjs.org/http-parser-js/-/http-parser-js-0.4.10.tgz", - "integrity": "sha1-ksnBN0w1CF912zWexWzCV8u5P6Q=", - "dev": true - }, - "http-proxy": { - "version": "1.18.0", - "resolved": "https://registry.npmjs.org/http-proxy/-/http-proxy-1.18.0.tgz", - "integrity": "sha512-84I2iJM/n1d4Hdgc6y2+qY5mDaz2PUVjlg9znE9byl+q0uC3DeByqBGReQu5tpLK0TAqTIXScRUV+dg7+bUPpQ==", - "dev": true, - "requires": { - "eventemitter3": "^4.0.0", - "follow-redirects": "^1.0.0", - "requires-port": "^1.0.0" - } - }, - "http-proxy-agent": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/http-proxy-agent/-/http-proxy-agent-2.1.0.tgz", - "integrity": "sha512-qwHbBLV7WviBl0rQsOzH6o5lwyOIvwp/BdFnvVxXORldu5TmjFfjzBcWUWS5kWAZhmv+JtiDhSuQCp4sBfbIgg==", - "dev": true, - "requires": { - "agent-base": "4", - "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==", - "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 - } - } - }, - "http-proxy-middleware": { - "version": "0.19.1", - "resolved": "https://registry.npmjs.org/http-proxy-middleware/-/http-proxy-middleware-0.19.1.tgz", - "integrity": "sha512-yHYTgWMQO8VvwNS22eLLloAkvungsKdKTLO8AJlftYIKNfJr3GK3zK0ZCfzDDGUBttdGc8xFy1mCitvNKQtC3Q==", - "dev": true, - "requires": { - "http-proxy": "^1.17.0", - "is-glob": "^4.0.0", - "lodash": "^4.17.11", - "micromatch": "^3.1.10" - } - }, - "http-signature": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/http-signature/-/http-signature-1.2.0.tgz", - "integrity": "sha1-muzZJRFHcvPZW2WmCruPfBj7rOE=", - "dev": true, - "requires": { - "assert-plus": "^1.0.0", - "jsprim": "^1.2.2", - "sshpk": "^1.7.0" - } - }, - "https-browserify": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/https-browserify/-/https-browserify-1.0.0.tgz", - "integrity": "sha1-7AbBDgo0wPL68Zn3/X/Hj//QPHM=", - "dev": true - }, - "https-proxy-agent": { - "version": "2.2.4", - "resolved": "https://registry.npmjs.org/https-proxy-agent/-/https-proxy-agent-2.2.4.tgz", - "integrity": "sha512-OmvfoQ53WLjtA9HeYP9RNrWMJzzAz1JGaSFr1nijg0PVR1JaD/xbJq1mdEIIlxGpXp9eSe/O2LgU9DJmTPd0Eg==", - "dev": true, - "requires": { - "agent-base": "^4.3.0", - "debug": "^3.1.0" - }, - "dependencies": { - "debug": { - "version": "3.2.6", - "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.6.tgz", - "integrity": "sha512-mel+jf7nrtEl5Pn1Qx46zARXKDpBbvzezse7p7LqINmdoIk8PYP5SySaxEmYv6TZ0JyEKA1hsCId6DIhgITtWQ==", - "dev": true, - "requires": { - "ms": "^2.1.1" - } - } - } - }, - "humanize-ms": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/humanize-ms/-/humanize-ms-1.2.1.tgz", - "integrity": "sha1-xG4xWaKT9riW2ikxbYtv6Lt5u+0=", - "dev": true, - "requires": { - "ms": "^2.0.0" - } - }, - "iconv-lite": { - "version": "0.4.24", - "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", - "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", - "dev": true, - "requires": { - "safer-buffer": ">= 2.1.2 < 3" - } - }, - "icss-utils": { - "version": "4.1.1", - "resolved": "https://registry.npmjs.org/icss-utils/-/icss-utils-4.1.1.tgz", - "integrity": "sha512-4aFq7wvWyMHKgxsH8QQtGpvbASCf+eM3wPRLI6R+MgAnTCZ6STYsRvttLvRWK0Nfif5piF394St3HeJDaljGPA==", - "dev": true, - "requires": { - "postcss": "^7.0.14" - } - }, - "ieee754": { - "version": "1.1.13", - "resolved": "https://registry.npmjs.org/ieee754/-/ieee754-1.1.13.tgz", - "integrity": "sha512-4vf7I2LYV/HaWerSo3XmlMkp5eZ83i+/CDluXi/IGTs/O1sejBNhTtnxzmRZfvOUqj7lZjqHkeTvpgSFDlWZTg==", - "dev": true - }, - "iferr": { - "version": "0.1.5", - "resolved": "https://registry.npmjs.org/iferr/-/iferr-0.1.5.tgz", - "integrity": "sha1-xg7taebY/bazEEofy8ocGS3FtQE=", - "dev": true - }, - "ignore": { - "version": "3.3.10", - "resolved": "https://registry.npmjs.org/ignore/-/ignore-3.3.10.tgz", - "integrity": "sha512-Pgs951kaMm5GXP7MOvxERINe3gsaVjUWFm+UZPSq9xYriQAksyhg0csnS0KXSNRD5NmNdapXEpjxG49+AKh/ug==", - "dev": true - }, - "ignore-walk": { - "version": "3.0.3", - "resolved": "https://registry.npmjs.org/ignore-walk/-/ignore-walk-3.0.3.tgz", - "integrity": "sha512-m7o6xuOaT1aqheYHKf8W6J5pYH85ZI9w077erOzLje3JsB1gkafkAhHHY19dqjulgIZHFm32Cp5uNZgcQqdJKw==", - "dev": true, - "requires": { - "minimatch": "^3.0.4" - } - }, - "image-size": { - "version": "0.5.5", - "resolved": "https://registry.npmjs.org/image-size/-/image-size-0.5.5.tgz", - "integrity": "sha1-Cd/Uq50g4p6xw+gLiZA3jfnjy5w=", - "dev": true, - "optional": true - }, - "immediate": { - "version": "3.0.6", - "resolved": "https://registry.npmjs.org/immediate/-/immediate-3.0.6.tgz", - "integrity": "sha1-nbHb0Pr43m++D13V5Wu2BigN5ps=", - "dev": true - }, - "import-cwd": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/import-cwd/-/import-cwd-2.1.0.tgz", - "integrity": "sha1-qmzzbnInYShcs3HsZRn1PiQ1sKk=", - "dev": true, - "requires": { - "import-from": "^2.1.0" - } - }, - "import-fresh": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-2.0.0.tgz", - "integrity": "sha1-2BNVwVYS04bGH53dOSLUMEgipUY=", - "dev": true, - "requires": { - "caller-path": "^2.0.0", - "resolve-from": "^3.0.0" - } - }, - "import-from": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/import-from/-/import-from-2.1.0.tgz", - "integrity": "sha1-M1238qev/VOqpHHUuAId7ja387E=", - "dev": true, - "requires": { - "resolve-from": "^3.0.0" - } - }, - "import-local": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/import-local/-/import-local-2.0.0.tgz", - "integrity": "sha512-b6s04m3O+s3CGSbqDIyP4R6aAwAeYlVq9+WUWep6iHa8ETRf9yei1U48C5MmfJmV9AiLYYBKPMq/W+/WRpQmCQ==", - "dev": true, - "requires": { - "pkg-dir": "^3.0.0", - "resolve-cwd": "^2.0.0" - } - }, - "imurmurhash": { - "version": "0.1.4", - "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", - "integrity": "sha1-khi5srkoojixPcT7a21XbyMUU+o=", - "dev": true - }, - "indent-string": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/indent-string/-/indent-string-4.0.0.tgz", - "integrity": "sha512-EdDDZu4A2OyIK7Lr/2zG+w5jmbuk1DVBnEwREQvBzspBJkCEbRa8GxU1lghYcaGJCnRWibjDXlq779X1/y5xwg==", - "dev": true - }, - "indexes-of": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/indexes-of/-/indexes-of-1.0.1.tgz", - "integrity": "sha1-8w9xbI4r00bHtn0985FVZqfAVgc=", - "dev": true - }, - "infer-owner": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/infer-owner/-/infer-owner-1.0.4.tgz", - "integrity": "sha512-IClj+Xz94+d7irH5qRyfJonOdfTzuDaifE6ZPWfx0N0+/ATZCbuTPq2prFl526urkQd90WyUKIh1DfBQ2hMz9A==", - "dev": true - }, - "inflight": { - "version": "1.0.6", - "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", - "integrity": "sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk=", - "dev": true, - "requires": { - "once": "^1.3.0", - "wrappy": "1" - } - }, - "inherits": { - "version": "2.0.4", - "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", - "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==", - "dev": true - }, - "ini": { - "version": "1.3.5", - "resolved": "https://registry.npmjs.org/ini/-/ini-1.3.5.tgz", - "integrity": "sha512-RZY5huIKCMRWDUqZlEi72f/lmXKMvuszcMBduliQ3nnWbx9X/ZBQO7DijMEYS9EhHBb2qacRUMtC7svLwe0lcw==", - "dev": true - }, - "inquirer": { - "version": "7.1.0", - "resolved": "https://registry.npmjs.org/inquirer/-/inquirer-7.1.0.tgz", - "integrity": "sha512-5fJMWEmikSYu0nv/flMc475MhGbB7TSPd/2IpFV4I4rMklboCH2rQjYY5kKiYGHqUF9gvaambupcJFFG9dvReg==", - "dev": true, - "requires": { - "ansi-escapes": "^4.2.1", - "chalk": "^3.0.0", - "cli-cursor": "^3.1.0", - "cli-width": "^2.0.0", - "external-editor": "^3.0.3", - "figures": "^3.0.0", - "lodash": "^4.17.15", - "mute-stream": "0.0.8", - "run-async": "^2.4.0", - "rxjs": "^6.5.3", - "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.2.1", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.2.1.tgz", - "integrity": "sha512-9VGjrMsG1vePxcSweQsN20KY/c4zN0h9fLjqAbwbPfahM3t+NL+M9HC8xeXG2I8pX5NoamTGNuomEUFI7fcUjA==", - "dev": true, - "requires": { - "@types/color-name": "^1.1.1", - "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 - }, - "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.0", - "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.0.tgz", - "integrity": "sha512-zUz5JD+tgqtuDjMhwIg5uFVV3dtqZ9yQJlZVfq4I01/K5Paj5UHj7VyrQOJvzawSVlKpObApbfD0Ed6yJc+1eg==", - "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.1.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.1.0.tgz", - "integrity": "sha512-oRSIpR8pxT1Wr2FquTNnGet79b3BWljqOuoW/h4oBhxJ/HUbX5nX6JSruTkvXDCFMwDPvsaTTbvMLKZWSy0R5g==", - "dev": true, - "requires": { - "has-flag": "^4.0.0" - } - } - } - }, - "internal-ip": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/internal-ip/-/internal-ip-4.3.0.tgz", - "integrity": "sha512-S1zBo1D6zcsyuC6PMmY5+55YMILQ9av8lotMx447Bq6SAgo/sDK6y6uUKmuYhW7eacnIhFfsPmCNYdDzsnnDCg==", - "dev": true, - "requires": { - "default-gateway": "^4.2.0", - "ipaddr.js": "^1.9.0" - } - }, - "invariant": { - "version": "2.2.4", - "resolved": "https://registry.npmjs.org/invariant/-/invariant-2.2.4.tgz", - "integrity": "sha512-phJfQVBuaJM5raOpJjSfkiD6BpbCE4Ns//LaXl6wGYtUBY83nWS6Rf9tXm2e8VaK60JEjYldbPif/A2B1C2gNA==", - "dev": true, - "requires": { - "loose-envify": "^1.0.0" - } - }, - "invert-kv": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/invert-kv/-/invert-kv-2.0.0.tgz", - "integrity": "sha512-wPVv/y/QQ/Uiirj/vh3oP+1Ww+AWehmi1g5fFWGPF6IpCBCDVrhgHRMvrLfdYcwDh3QJbGXDW4JAuzxElLSqKA==", - "dev": true - }, - "ip": { - "version": "1.1.5", - "resolved": "https://registry.npmjs.org/ip/-/ip-1.1.5.tgz", - "integrity": "sha1-vd7XARQpCCjAoDnnLvJfWq7ENUo=", - "dev": true - }, - "ip-regex": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/ip-regex/-/ip-regex-2.1.0.tgz", - "integrity": "sha1-+ni/XS5pE8kRzp+BnuUUa7bYROk=", - "dev": true - }, - "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==", - "dev": true - }, - "is-absolute-url": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/is-absolute-url/-/is-absolute-url-2.1.0.tgz", - "integrity": "sha1-UFMN+4T8yap9vnhS6Do3uTufKqY=", - "dev": true - }, - "is-accessor-descriptor": { - "version": "0.1.6", - "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-0.1.6.tgz", - "integrity": "sha1-qeEss66Nh2cn7u84Q/igiXtcmNY=", - "dev": true, - "requires": { - "kind-of": "^3.0.2" - }, - "dependencies": { - "kind-of": { - "version": "3.2.2", - "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", - "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", - "dev": true, - "requires": { - "is-buffer": "^1.1.5" - } - } - } - }, - "is-arguments": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/is-arguments/-/is-arguments-1.0.4.tgz", - "integrity": "sha512-xPh0Rmt8NE65sNzvyUmWgI1tz3mKq74lGA0mL8LYZcoIzKOzDh6HmrYm3d18k60nHerC8A9Km8kYu87zfSFnLA==", - "dev": true - }, - "is-arrayish": { - "version": "0.2.1", - "resolved": "https://registry.npmjs.org/is-arrayish/-/is-arrayish-0.2.1.tgz", - "integrity": "sha1-d8mYQFJ6qOyxqLppe4BkWnqSap0=", - "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-buffer": { - "version": "1.1.6", - "resolved": "https://registry.npmjs.org/is-buffer/-/is-buffer-1.1.6.tgz", - "integrity": "sha512-NcdALwpXkTm5Zvvbk7owOUSvVvBKDgKP5/ewfXEznmQFfs4ZRmanOeKBTjRVjka3QFoN6XJ+9F3USqfHqTaU5w==", - "dev": true - }, - "is-callable": { - "version": "1.1.5", - "resolved": "https://registry.npmjs.org/is-callable/-/is-callable-1.1.5.tgz", - "integrity": "sha512-ESKv5sMCJB2jnHTWZ3O5itG+O128Hsus4K4Qh1h2/cgn2vbgnLSVqfV46AeJA9D5EeeLa9w81KUXMtn34zhX+Q==", - "dev": true - }, - "is-color-stop": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/is-color-stop/-/is-color-stop-1.1.0.tgz", - "integrity": "sha1-z/9HGu5N1cnhWFmPvhKWe1za00U=", - "dev": true, - "requires": { - "css-color-names": "^0.0.4", - "hex-color-regex": "^1.1.0", - "hsl-regex": "^1.0.0", - "hsla-regex": "^1.0.0", - "rgb-regex": "^1.0.1", - "rgba-regex": "^1.0.0" - } - }, - "is-data-descriptor": { - "version": "0.1.4", - "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-0.1.4.tgz", - "integrity": "sha1-C17mSDiOLIYCgueT8YVv7D8wG1Y=", - "dev": true, - "requires": { - "kind-of": "^3.0.2" - }, - "dependencies": { - "kind-of": { - "version": "3.2.2", - "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", - "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", - "dev": true, - "requires": { - "is-buffer": "^1.1.5" - } - } - } - }, - "is-date-object": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/is-date-object/-/is-date-object-1.0.2.tgz", - "integrity": "sha512-USlDT524woQ08aoZFzh3/Z6ch9Y/EWXEHQ/AaRN0SkKq4t2Jw2R2339tSXmwuVoY7LLlBCbOIlx2myP/L5zk0g==", - "dev": true - }, - "is-descriptor": { - "version": "0.1.6", - "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-0.1.6.tgz", - "integrity": "sha512-avDYr0SB3DwO9zsMov0gKCESFYqCnE4hq/4z3TdUlukEy5t9C0YRq7HLrsN52NAcqXKaepeCD0n+B0arnVG3Hg==", - "dev": true, - "requires": { - "is-accessor-descriptor": "^0.1.6", - "is-data-descriptor": "^0.1.4", - "kind-of": "^5.0.0" - }, - "dependencies": { - "kind-of": { - "version": "5.1.0", - "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-5.1.0.tgz", - "integrity": "sha512-NGEErnH6F2vUuXDh+OlbcKW7/wOcfdRHaZ7VWtqCztfHri/++YKmP51OdWeGPuqCOba6kk2OTe5d02VmTB80Pw==", - "dev": true - } - } - }, - "is-directory": { - "version": "0.3.1", - "resolved": "https://registry.npmjs.org/is-directory/-/is-directory-0.3.1.tgz", - "integrity": "sha1-YTObbyR1/Hcv2cnYP1yFddwVSuE=", - "dev": true - }, - "is-docker": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/is-docker/-/is-docker-2.0.0.tgz", - "integrity": "sha512-pJEdRugimx4fBMra5z2/5iRdZ63OhYV0vr0Dwm5+xtW4D1FvRkB8hamMIhnWfyJeDdyr/aa7BDyNbtG38VxgoQ==", - "dev": true - }, - "is-extendable": { - "version": "0.1.1", - "resolved": "https://registry.npmjs.org/is-extendable/-/is-extendable-0.1.1.tgz", - "integrity": "sha1-YrEQ4omkcUGOPsNqYX1HLjAd/Ik=", - "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": "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 - }, - "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-interactive": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/is-interactive/-/is-interactive-1.0.0.tgz", - "integrity": "sha512-2HvIEKRoqS62guEC+qBjpvRubdX910WCMuJTZ+I9yvqKU2/12eSL549HMwtabb4oupdj2sMP50k+XJfB/8JE6w==", - "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-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-cwd": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/is-path-cwd/-/is-path-cwd-2.2.0.tgz", - "integrity": "sha512-w942bTcih8fdJPJmQHFzkS76NEP8Kzzvmw92cXsazb8intwLqPibPPdXf4ANdKV3rYMuuQYGIWtvz9JilB3NFQ==", - "dev": true - }, - "is-path-in-cwd": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/is-path-in-cwd/-/is-path-in-cwd-2.1.0.tgz", - "integrity": "sha512-rNocXHgipO+rvnP6dk3zI20RpOtrAM/kzbB258Uw5BWr3TpXi861yzjo16Dn4hUox07iw5AyeMLHWsujkjzvRQ==", - "dev": true, - "requires": { - "is-path-inside": "^2.1.0" - } - }, - "is-path-inside": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-2.1.0.tgz", - "integrity": "sha512-wiyhTzfDWsvwAW53OBWF5zuvaOGlZ6PwYxAbPVDhpm+gM09xKQGjBq/8uYN12aDvMxnAnq3dxTyoSoRNmg5YFg==", - "dev": true, - "requires": { - "path-is-inside": "^1.0.2" - } - }, - "is-plain-obj": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/is-plain-obj/-/is-plain-obj-1.1.0.tgz", - "integrity": "sha1-caUMhCnfync8kqOQpKA7OfzVHT4=", - "dev": true - }, - "is-plain-object": { - "version": "2.0.4", - "resolved": "https://registry.npmjs.org/is-plain-object/-/is-plain-object-2.0.4.tgz", - "integrity": "sha512-h5PpgXkWitc38BBMYawTYMWJHFZJVnBquFE57xFpjB8pJFiF6gZ+bU+WyI/yqXiFR5mdLsgYNaPe8uao6Uv9Og==", - "dev": true, - "requires": { - "isobject": "^3.0.1" - } - }, - "is-regex": { - "version": "1.0.5", - "resolved": "https://registry.npmjs.org/is-regex/-/is-regex-1.0.5.tgz", - "integrity": "sha512-vlKW17SNq44owv5AQR3Cq0bQPEb8+kF3UKZ2fiZNOWtztYE5i0CzCZxFDwO58qAOWtxdBRVO/V5Qin1wjCqFYQ==", - "dev": true, - "requires": { - "has": "^1.0.3" - } - }, - "is-resolvable": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/is-resolvable/-/is-resolvable-1.1.0.tgz", - "integrity": "sha512-qgDYXFSR5WvEfuS5dMj6oTMEbrrSaM0CrFk2Yiq/gXnBvD9pMa2jGXxyhGLfvhZpuMZe18CJpFxAt3CRs42NMg==", - "dev": true - }, - "is-stream": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/is-stream/-/is-stream-1.1.0.tgz", - "integrity": "sha1-EtSj3U5o4Lec6428hBc66A2RykQ=", - "dev": true - }, - "is-svg": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/is-svg/-/is-svg-3.0.0.tgz", - "integrity": "sha512-gi4iHK53LR2ujhLVVj+37Ykh9GLqYHX6JOVXbLAucaG/Cqw9xwdFOjDM2qeifLs1sF1npXXFvDu0r5HNgCMrzQ==", - "dev": true, - "requires": { - "html-comment-regex": "^1.1.0" - } - }, - "is-symbol": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/is-symbol/-/is-symbol-1.0.3.tgz", - "integrity": "sha512-OwijhaRSgqvhm/0ZdAcXNZt9lYdKFpcRDT5ULUuYXPoT794UNOdU+gpT6Rzo7b4V2HUl/op6GqY894AZwv9faQ==", - "dev": true, - "requires": { - "has-symbols": "^1.0.1" - } - }, - "is-typedarray": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/is-typedarray/-/is-typedarray-1.0.0.tgz", - "integrity": "sha1-5HnICFjfDBsR3dppQPlgEfzaSpo=", - "dev": true - }, - "is-windows": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/is-windows/-/is-windows-1.0.2.tgz", - "integrity": "sha512-eXK1UInq2bPmjyX6e3VHIzMLobc4J94i4AWn+Hpq3OU5KkrRC96OAcR3PRJ/pGu6m8TRnBHP9dkXQVsT/COVIA==", - "dev": true - }, - "is-wsl": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/is-wsl/-/is-wsl-2.2.0.tgz", - "integrity": "sha512-fKzAra0rGJUUBwGBgNkHZuToZcn+TtXHpeCgmkMJMMYx1sQDYaCSyjJBSCa2nH1DGm7s3n1oBnohoVTBaN7Lww==", - "dev": true, - "requires": { - "is-docker": "^2.0.0" - } - }, - "isarray": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", - "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE=", - "dev": true - }, - "isbinaryfile": { - "version": "4.0.8", - "resolved": "https://registry.npmjs.org/isbinaryfile/-/isbinaryfile-4.0.8.tgz", - "integrity": "sha512-53h6XFniq77YdW+spoRrebh0mnmTxRPTlcuIArO57lmMdq4uBKFKaeTjnb92oYWrSn/LVL+LT+Hap2tFQj8V+w==", - "dev": true - }, - "isexe": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", - "integrity": "sha1-6PvzdNxVb/iUehDcsFctYz8s+hA=", - "dev": true - }, - "isobject": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/isobject/-/isobject-3.0.1.tgz", - "integrity": "sha1-TkMekrEalzFjaqH5yNHMvP2reN8=", - "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=", - "dev": true - }, - "istanbul-api": { - "version": "2.1.6", - "resolved": "https://registry.npmjs.org/istanbul-api/-/istanbul-api-2.1.6.tgz", - "integrity": "sha512-x0Eicp6KsShG1k1rMgBAi/1GgY7kFGEBwQpw3PXGEmu+rBcBNhqU8g2DgY9mlepAsLPzrzrbqSgCGANnki4POA==", - "dev": true, - "requires": { - "async": "^2.6.2", - "compare-versions": "^3.4.0", - "fileset": "^2.0.3", - "istanbul-lib-coverage": "^2.0.5", - "istanbul-lib-hook": "^2.0.7", - "istanbul-lib-instrument": "^3.3.0", - "istanbul-lib-report": "^2.0.8", - "istanbul-lib-source-maps": "^3.0.6", - "istanbul-reports": "^2.2.4", - "js-yaml": "^3.13.1", - "make-dir": "^2.1.0", - "minimatch": "^3.0.4", - "once": "^1.4.0" - }, - "dependencies": { - "istanbul-lib-coverage": { - "version": "2.0.5", - "resolved": "https://registry.npmjs.org/istanbul-lib-coverage/-/istanbul-lib-coverage-2.0.5.tgz", - "integrity": "sha512-8aXznuEPCJvGnMSRft4udDRDtb1V3pkQkMMI5LI+6HuQz5oQ4J2UFn1H82raA3qJtyOLkkwVqICBQkjnGtn5mA==", - "dev": true - }, - "istanbul-lib-instrument": { - "version": "3.3.0", - "resolved": "https://registry.npmjs.org/istanbul-lib-instrument/-/istanbul-lib-instrument-3.3.0.tgz", - "integrity": "sha512-5nnIN4vo5xQZHdXno/YDXJ0G+I3dAm4XgzfSVTPLQpj/zAV2dV6Juy0yaf10/zrJOJeHoN3fraFe+XRq2bFVZA==", - "dev": true, - "requires": { - "@babel/generator": "^7.4.0", - "@babel/parser": "^7.4.3", - "@babel/template": "^7.4.0", - "@babel/traverse": "^7.4.3", - "@babel/types": "^7.4.0", - "istanbul-lib-coverage": "^2.0.5", - "semver": "^6.0.0" - } - } - } - }, - "istanbul-lib-coverage": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/istanbul-lib-coverage/-/istanbul-lib-coverage-3.0.0.tgz", - "integrity": "sha512-UiUIqxMgRDET6eR+o5HbfRYP1l0hqkWOs7vNxC/mggutCMUIhWMm8gAHb8tHlyfD3/l6rlgNA5cKdDzEAf6hEg==", - "dev": true - }, - "istanbul-lib-hook": { - "version": "2.0.7", - "resolved": "https://registry.npmjs.org/istanbul-lib-hook/-/istanbul-lib-hook-2.0.7.tgz", - "integrity": "sha512-vrRztU9VRRFDyC+aklfLoeXyNdTfga2EI3udDGn4cZ6fpSXpHLV9X6CHvfoMCPtggg8zvDDmC4b9xfu0z6/llA==", - "dev": true, - "requires": { - "append-transform": "^1.0.0" - } - }, - "istanbul-lib-instrument": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/istanbul-lib-instrument/-/istanbul-lib-instrument-4.0.1.tgz", - "integrity": "sha512-imIchxnodll7pvQBYOqUu88EufLCU56LMeFPZZM/fJZ1irYcYdqroaV+ACK1Ila8ls09iEYArp+nqyC6lW1Vfg==", - "dev": true, - "requires": { - "@babel/core": "^7.7.5", - "@babel/parser": "^7.7.5", - "@babel/template": "^7.7.4", - "@babel/traverse": "^7.7.4", - "@istanbuljs/schema": "^0.1.2", - "istanbul-lib-coverage": "^3.0.0", - "semver": "^6.3.0" - } - }, - "istanbul-lib-report": { - "version": "2.0.8", - "resolved": "https://registry.npmjs.org/istanbul-lib-report/-/istanbul-lib-report-2.0.8.tgz", - "integrity": "sha512-fHBeG573EIihhAblwgxrSenp0Dby6tJMFR/HvlerBsrCTD5bkUuoNtn3gVh29ZCS824cGGBPn7Sg7cNk+2xUsQ==", - "dev": true, - "requires": { - "istanbul-lib-coverage": "^2.0.5", - "make-dir": "^2.1.0", - "supports-color": "^6.1.0" - }, - "dependencies": { - "istanbul-lib-coverage": { - "version": "2.0.5", - "resolved": "https://registry.npmjs.org/istanbul-lib-coverage/-/istanbul-lib-coverage-2.0.5.tgz", - "integrity": "sha512-8aXznuEPCJvGnMSRft4udDRDtb1V3pkQkMMI5LI+6HuQz5oQ4J2UFn1H82raA3qJtyOLkkwVqICBQkjnGtn5mA==", - "dev": true - }, - "supports-color": { - "version": "6.1.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-6.1.0.tgz", - "integrity": "sha512-qe1jfm1Mg7Nq/NSh6XE24gPXROEVsWHxC1LIx//XNlD9iw7YZQGjZNjYN7xGaEG6iKdA8EtNFW6R0gjnVXp+wQ==", - "dev": true, - "requires": { - "has-flag": "^3.0.0" - } - } - } - }, - "istanbul-lib-source-maps": { - "version": "3.0.6", - "resolved": "https://registry.npmjs.org/istanbul-lib-source-maps/-/istanbul-lib-source-maps-3.0.6.tgz", - "integrity": "sha512-R47KzMtDJH6X4/YW9XTx+jrLnZnscW4VpNN+1PViSYTejLVPWv7oov+Duf8YQSPyVRUvueQqz1TcsC6mooZTXw==", - "dev": true, - "requires": { - "debug": "^4.1.1", - "istanbul-lib-coverage": "^2.0.5", - "make-dir": "^2.1.0", - "rimraf": "^2.6.3", - "source-map": "^0.6.1" - }, - "dependencies": { - "istanbul-lib-coverage": { - "version": "2.0.5", - "resolved": "https://registry.npmjs.org/istanbul-lib-coverage/-/istanbul-lib-coverage-2.0.5.tgz", - "integrity": "sha512-8aXznuEPCJvGnMSRft4udDRDtb1V3pkQkMMI5LI+6HuQz5oQ4J2UFn1H82raA3qJtyOLkkwVqICBQkjnGtn5mA==", - "dev": true - }, - "rimraf": { - "version": "2.7.1", - "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", - "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==", - "dev": true, - "requires": { - "glob": "^7.1.3" - } - }, - "source-map": { - "version": "0.6.1", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", - "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", - "dev": true - } - } - }, - "istanbul-reports": { - "version": "2.2.7", - "resolved": "https://registry.npmjs.org/istanbul-reports/-/istanbul-reports-2.2.7.tgz", - "integrity": "sha512-uu1F/L1o5Y6LzPVSVZXNOoD/KXpJue9aeLRd0sM9uMXfZvzomB0WxVamWb5ue8kA2vVWEmW7EG+A5n3f1kqHKg==", - "dev": true, - "requires": { - "html-escaper": "^2.0.0" - } - }, - "jasmine": { - "version": "2.8.0", - "resolved": "https://registry.npmjs.org/jasmine/-/jasmine-2.8.0.tgz", - "integrity": "sha1-awicChFXax8W3xG4AUbZHU6Lij4=", - "dev": true, - "requires": { - "exit": "^0.1.2", - "glob": "^7.0.6", - "jasmine-core": "~2.8.0" - }, - "dependencies": { - "jasmine-core": { - "version": "2.8.0", - "resolved": "https://registry.npmjs.org/jasmine-core/-/jasmine-core-2.8.0.tgz", - "integrity": "sha1-vMl5rh+f0FcB5F5S5l06XWPxok4=", - "dev": true - } - } - }, - "jasmine-core": { - "version": "3.5.0", - "resolved": "https://registry.npmjs.org/jasmine-core/-/jasmine-core-3.5.0.tgz", - "integrity": "sha512-nCeAiw37MIMA9w9IXso7bRaLl+c/ef3wnxsoSAlYrzS+Ot0zTG6nU8G/cIfGkqpkjX2wNaIW9RFG0TwIFnG6bA==", - "dev": true - }, - "jasmine-spec-reporter": { - "version": "4.2.1", - "resolved": "https://registry.npmjs.org/jasmine-spec-reporter/-/jasmine-spec-reporter-4.2.1.tgz", - "integrity": "sha512-FZBoZu7VE5nR7Nilzy+Np8KuVIOxF4oXDPDknehCYBDE080EnlPu0afdZNmpGDBRCUBv3mj5qgqCRmk6W/K8vg==", - "dev": true, - "requires": { - "colors": "1.1.2" - } - }, - "jasminewd2": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/jasminewd2/-/jasminewd2-2.2.0.tgz", - "integrity": "sha1-43zwsX8ZnM4jvqcbIDk5Uka07E4=", - "dev": true - }, - "jest-worker": { - "version": "25.1.0", - "resolved": "https://registry.npmjs.org/jest-worker/-/jest-worker-25.1.0.tgz", - "integrity": "sha512-ZHhHtlxOWSxCoNOKHGbiLzXnl42ga9CxDr27H36Qn+15pQZd3R/F24jrmjDelw9j/iHUIWMWs08/u2QN50HHOg==", - "dev": true, - "requires": { - "merge-stream": "^2.0.0", - "supports-color": "^7.0.0" - }, - "dependencies": { - "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.1.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.1.0.tgz", - "integrity": "sha512-oRSIpR8pxT1Wr2FquTNnGet79b3BWljqOuoW/h4oBhxJ/HUbX5nX6JSruTkvXDCFMwDPvsaTTbvMLKZWSy0R5g==", - "dev": true, - "requires": { - "has-flag": "^4.0.0" - } - } - } - }, - "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.13.1", - "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.13.1.tgz", - "integrity": "sha512-YfbcO7jXDdyj0DGxYVSlSeQNHbD7XPWvrVWeVUujrQEoZzWJIRrCPoyk6kL6IAjAG2IolMK4T0hNUe0HOUs5Jw==", - "dev": true, - "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=", - "dev": true - }, - "jsesc": { - "version": "2.5.2", - "resolved": "https://registry.npmjs.org/jsesc/-/jsesc-2.5.2.tgz", - "integrity": "sha512-OYu7XEzjkCQ3C5Ps3QIZsQfNpqoJyZZA99wd9aWd05NCtC5pWOkShK2mkL6HXQR6/Cy2lbNdPlZBpuQHXE63gA==", - "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=", - "dev": true - }, - "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==", - "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=", - "dev": true - }, - "json3": { - "version": "3.3.3", - "resolved": "https://registry.npmjs.org/json3/-/json3-3.3.3.tgz", - "integrity": "sha512-c7/8mbUsKigAbLkD5B010BK4D9LZm7A1pNItkEwiUZRpIN66exu/e7YQWysGun+TRKaJp8MhemM+VkfWv42aCA==", - "dev": true - }, - "json5": { - "version": "2.1.3", - "resolved": "https://registry.npmjs.org/json5/-/json5-2.1.3.tgz", - "integrity": "sha512-KXPvOm8K9IJKFM0bmdn8QXh7udDh1g/giieX0NLCaMnb4hEiVFqnop2ImTXCc5e0/oHz3LTqmHGtExn5hfMkOA==", - "dev": true, - "requires": { - "minimist": "^1.2.5" - } - }, - "jsonfile": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/jsonfile/-/jsonfile-4.0.0.tgz", - "integrity": "sha1-h3Gq4HmbZAdrdmQPygWPnBDjPss=", - "dev": true, - "requires": { - "graceful-fs": "^4.1.6" - } - }, - "jsonparse": { - "version": "1.3.1", - "resolved": "https://registry.npmjs.org/jsonparse/-/jsonparse-1.3.1.tgz", - "integrity": "sha1-P02uSpH6wxX3EGL4UhzCOfE2YoA=", - "dev": true - }, - "jsplumb": { - "version": "2.13.2", - "resolved": "https://registry.npmjs.org/jsplumb/-/jsplumb-2.13.2.tgz", - "integrity": "sha512-Ad54lZusITD/7ENdt0Z3H7iyeyw0ov+dcPBRbMxezMxCnzzM27+dVKBlEZhhJ6EbrYY3DzPo7eFK5YBmR8NLzg==" - }, - "jsprim": { - "version": "1.4.1", - "resolved": "https://registry.npmjs.org/jsprim/-/jsprim-1.4.1.tgz", - "integrity": "sha1-MT5mvB5cwG5Di8G3SZwuXFastqI=", - "dev": true, - "requires": { - "assert-plus": "1.0.0", - "extsprintf": "1.3.0", - "json-schema": "0.2.3", - "verror": "1.10.0" - } - }, - "jszip": { - "version": "3.7.1", - "resolved": "https://registry.npmjs.org/jszip/-/jszip-3.7.1.tgz", - "integrity": "sha512-ghL0tz1XG9ZEmRMcEN2vt7xabrDdqHHeykgARpmZ0BiIctWxM47Vt63ZO2dnp4QYt/xJVLLy5Zv1l/xRdh2byg==", - "dev": true, - "requires": { - "lie": "~3.3.0", - "pako": "~1.0.2", - "readable-stream": "~2.3.6", - "set-immediate-shim": "~1.0.1" - } - }, - "karma": { - "version": "6.3.16", - "resolved": "https://registry.npmjs.org/karma/-/karma-6.3.16.tgz", - "integrity": "sha512-nEU50jLvDe5yvXqkEJRf8IuvddUkOY2x5Xc4WXHz6dxINgGDrgD2uqQWeVrJs4hbfNaotn+HQ1LZJ4yOXrL7xQ==", - "dev": true, - "requires": { - "body-parser": "^1.19.0", - "braces": "^3.0.2", - "chokidar": "^3.5.1", - "colors": "1.4.0", - "connect": "^3.7.0", - "di": "^0.0.1", - "dom-serialize": "^2.2.1", - "glob": "^7.1.7", - "graceful-fs": "^4.2.6", - "http-proxy": "^1.18.1", - "isbinaryfile": "^4.0.8", - "lodash": "^4.17.21", - "log4js": "^6.4.1", - "mime": "^2.5.2", - "minimatch": "^3.0.4", - "mkdirp": "^0.5.5", - "qjobs": "^1.2.0", - "range-parser": "^1.2.1", - "rimraf": "^3.0.2", - "socket.io": "^4.2.0", - "source-map": "^0.6.1", - "tmp": "^0.2.1", - "ua-parser-js": "^0.7.30", - "yargs": "^16.1.1" - }, - "dependencies": { - "ansi-regex": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", - "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", - "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" - } - }, - "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" - } - }, - "chokidar": { - "version": "3.5.3", - "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.5.3.tgz", - "integrity": "sha512-Dr3sfKRP6oTcjf2JmUmFJfeVMvXBdegxB0iVQ5eb2V10uFJUCAS8OByZdVAyVb8xXNz3GjjTgj9kLWsZTqE6kw==", - "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" - } - }, - "cliui": { - "version": "7.0.4", - "resolved": "https://registry.npmjs.org/cliui/-/cliui-7.0.4.tgz", - "integrity": "sha512-OcRE68cOsVMXp1Yvonl/fzkQOyjLSu/8bhPDfQt0e0/Eb283TKP20Fs2MqoPsr9SwA595rRCA+QMzYc9nBP+JQ==", - "dev": true, - "requires": { - "string-width": "^4.2.0", - "strip-ansi": "^6.0.0", - "wrap-ansi": "^7.0.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 - }, - "colors": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/colors/-/colors-1.4.0.tgz", - "integrity": "sha512-a+UqTh4kgZg/SlGvfbzDHpgRu7AAQOmmqRHJnxhRZICKFUT91brVhNNt58CMWU9PsBbv3PDCZUHbVxuDiH2mtA==", - "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 - }, - "get-caller-file": { - "version": "2.0.5", - "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-2.0.5.tgz", - "integrity": "sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==", - "dev": true - }, - "glob": { - "version": "7.2.0", - "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.0.tgz", - "integrity": "sha512-lmLf6gtyrPq8tTjSmrO94wBeQbFR3HbLHbuyD69wuyQkImp2hWqMGB47OX65FBkPffO641IP9jWa1z4ivqG26Q==", - "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" - } - }, - "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" - } - }, - "graceful-fs": { - "version": "4.2.9", - "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.9.tgz", - "integrity": "sha512-NtNxqUcXgpW2iMrfqSfR73Glt39K+BLwWsPs94yR63v45T0Wbej7eRmL5cWfwEgqXnmjQp3zaJTshdRW/qC2ZQ==", - "dev": true - }, - "http-proxy": { - "version": "1.18.1", - "resolved": "https://registry.npmjs.org/http-proxy/-/http-proxy-1.18.1.tgz", - "integrity": "sha512-7mz/721AbnJwIVbnaSv1Cz3Am0ZLT/UBwkC92VlxhXv/k/BBQfM2fXElQNC27BVGr0uwUpplYPQM9LnaBMR5NQ==", - "dev": true, - "requires": { - "eventemitter3": "^4.0.0", - "follow-redirects": "^1.0.0", - "requires-port": "^1.0.0" - } - }, - "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 - }, - "mime": { - "version": "2.6.0", - "resolved": "https://registry.npmjs.org/mime/-/mime-2.6.0.tgz", - "integrity": "sha512-USPkMeET31rOMiarsBNIHZKLGgvKc/LrjofAnBlOttf5ajRvqiRA8QsenbcooctK6d6Ts6aqZXBA+XbkKthiQg==", - "dev": true - }, - "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" - } - }, - "source-map": { - "version": "0.6.1", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", - "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", - "dev": true - }, - "string-width": { - "version": "4.2.3", - "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz", - "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", - "dev": true, - "requires": { - "emoji-regex": "^8.0.0", - "is-fullwidth-code-point": "^3.0.0", - "strip-ansi": "^6.0.1" - } - }, - "strip-ansi": { - "version": "6.0.1", - "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", - "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", - "dev": true, - "requires": { - "ansi-regex": "^5.0.1" - } - }, - "tmp": { - "version": "0.2.1", - "resolved": "https://registry.npmjs.org/tmp/-/tmp-0.2.1.tgz", - "integrity": "sha512-76SUhtfqR2Ijn+xllcI5P1oyannHNHByD80W1q447gU3mp9G9PSpGdWmjUOHRDPiHYacIk66W7ubDTuPF3BEtQ==", - "dev": true, - "requires": { - "rimraf": "^3.0.0" - } - }, - "wrap-ansi": { - "version": "7.0.0", - "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-7.0.0.tgz", - "integrity": "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==", - "dev": true, - "requires": { - "ansi-styles": "^4.0.0", - "string-width": "^4.1.0", - "strip-ansi": "^6.0.0" - } - }, - "y18n": { - "version": "5.0.8", - "resolved": "https://registry.npmjs.org/y18n/-/y18n-5.0.8.tgz", - "integrity": "sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==", - "dev": true - }, - "yargs": { - "version": "16.2.0", - "resolved": "https://registry.npmjs.org/yargs/-/yargs-16.2.0.tgz", - "integrity": "sha512-D1mvvtDG0L5ft/jGWkLpG1+m0eQxOfaBvTNELraWj22wSVUMWxZUvYgJYcKh6jGGIkJFhH4IZPQhR4TKpc8mBw==", - "dev": true, - "requires": { - "cliui": "^7.0.2", - "escalade": "^3.1.1", - "get-caller-file": "^2.0.5", - "require-directory": "^2.1.1", - "string-width": "^4.2.0", - "y18n": "^5.0.5", - "yargs-parser": "^20.2.2" - } - }, - "yargs-parser": { - "version": "20.2.9", - "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-20.2.9.tgz", - "integrity": "sha512-y11nGElTIV+CT3Zv9t7VKl+Q3hTQoT9a1Qzezhhl6Rp21gJ/IVTW7Z3y9EWXhuUBC2Shnf+DX0antecpAwSP8w==", - "dev": true - } - } - }, - "karma-chrome-launcher": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/karma-chrome-launcher/-/karma-chrome-launcher-3.1.0.tgz", - "integrity": "sha512-3dPs/n7vgz1rxxtynpzZTvb9y/GIaW8xjAwcIGttLbycqoFtI7yo1NGnQi6oFTherRE+GIhCAHZC4vEqWGhNvg==", - "dev": true, - "requires": { - "which": "^1.2.1" - } - }, - "karma-coverage-istanbul-reporter": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/karma-coverage-istanbul-reporter/-/karma-coverage-istanbul-reporter-2.1.1.tgz", - "integrity": "sha512-CH8lTi8+kKXGvrhy94+EkEMldLCiUA0xMOiL31vvli9qK0T+qcXJAwWBRVJWnVWxYkTmyWar8lPz63dxX6/z1A==", - "dev": true, - "requires": { - "istanbul-api": "^2.1.6", - "minimatch": "^3.0.4" - } - }, - "karma-jasmine": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/karma-jasmine/-/karma-jasmine-2.0.1.tgz", - "integrity": "sha512-iuC0hmr9b+SNn1DaUD2QEYtUxkS1J+bSJSn7ejdEexs7P8EYvA1CWkEdrDQ+8jVH3AgWlCNwjYsT1chjcNW9lA==", - "dev": true, - "requires": { - "jasmine-core": "^3.3" - } - }, - "karma-jasmine-html-reporter": { - "version": "1.5.3", - "resolved": "https://registry.npmjs.org/karma-jasmine-html-reporter/-/karma-jasmine-html-reporter-1.5.3.tgz", - "integrity": "sha512-ci0VrjuCaFj+9d1tYlTE3KIPUCp0rz874zWWU3JgCMqGIyw5ke+BXWFPOAGAqUdCJcrMwneyvp1zFXA74MiPUA==", - "dev": true - }, - "karma-source-map-support": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/karma-source-map-support/-/karma-source-map-support-1.4.0.tgz", - "integrity": "sha512-RsBECncGO17KAoJCYXjv+ckIz+Ii9NCi+9enk+rq6XC81ezYkb4/RHE6CTXdA7IOJqoF3wcaLfVG0CPmE5ca6A==", - "dev": true, - "requires": { - "source-map-support": "^0.5.5" - } - }, - "killable": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/killable/-/killable-1.0.1.tgz", - "integrity": "sha512-LzqtLKlUwirEUyl/nicirVmNiPvYs7l5n8wOPP7fyJVpUPkvCnW/vuiXGpylGUlnPDnB7311rARzAt3Mhswpjg==", - "dev": true - }, - "kind-of": { - "version": "6.0.3", - "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-6.0.3.tgz", - "integrity": "sha512-dcS1ul+9tmeD95T+x28/ehLgd9mENa3LsvDTtzm3vyBEO7RPptvAD+t44WVXaUjTBRcrpFeFlC8WCruUR456hw==", - "dev": true - }, - "lcid": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/lcid/-/lcid-2.0.0.tgz", - "integrity": "sha512-avPEb8P8EGnwXKClwsNUgryVjllcRqtMYa49NTsbQagYuT1DcXnl1915oxWjoyGrXR6zH/Y0Zc96xWsPcoDKeA==", - "dev": true, - "requires": { - "invert-kv": "^2.0.0" - } - }, - "less": { - "version": "3.11.1", - "resolved": "https://registry.npmjs.org/less/-/less-3.11.1.tgz", - "integrity": "sha512-tlWX341RECuTOvoDIvtFqXsKj072hm3+9ymRBe76/mD6O5ZZecnlAOVDlWAleF2+aohFrxNidXhv2773f6kY7g==", - "dev": true, - "requires": { - "clone": "^2.1.2", - "errno": "^0.1.1", - "graceful-fs": "^4.1.2", - "image-size": "~0.5.0", - "mime": "^1.4.1", - "mkdirp": "^0.5.0", - "promise": "^7.1.1", - "request": "^2.83.0", - "source-map": "~0.6.0", - "tslib": "^1.10.0" - }, - "dependencies": { - "source-map": { - "version": "0.6.1", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", - "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", - "dev": true, - "optional": true - } - } - }, - "less-loader": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/less-loader/-/less-loader-5.0.0.tgz", - "integrity": "sha512-bquCU89mO/yWLaUq0Clk7qCsKhsF/TZpJUzETRvJa9KSVEL9SO3ovCvdEHISBhrC81OwC8QSVX7E0bzElZj9cg==", - "dev": true, - "requires": { - "clone": "^2.1.1", - "loader-utils": "^1.1.0", - "pify": "^4.0.1" - }, - "dependencies": { - "json5": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/json5/-/json5-1.0.1.tgz", - "integrity": "sha512-aKS4WQjPenRxiQsC93MNfjx+nbF4PAdYzmd/1JIj8HYzqfbu86beTuNgXDzPknWk0n0uARlyewZo4s++ES36Ow==", - "dev": true, - "requires": { - "minimist": "^1.2.0" - } - }, - "loader-utils": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/loader-utils/-/loader-utils-1.4.0.tgz", - "integrity": "sha512-qH0WSMBtn/oHuwjy/NucEgbx5dbxxnxup9s4PVXJUDHZBQY+s0NWA9rJf53RBnQZxfch7euUui7hpoAPvALZdA==", - "dev": true, - "requires": { - "big.js": "^5.2.2", - "emojis-list": "^3.0.0", - "json5": "^1.0.1" - } - } - } - }, - "leven": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/leven/-/leven-3.1.0.tgz", - "integrity": "sha512-qsda+H8jTaUaN/x5vzW2rzc+8Rw4TAQ/4KjB46IwK5VH+IlVeeeje/EoZRpiXvIqjFgK84QffqPztGI3VBLG1A==", - "dev": true - }, - "levenary": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/levenary/-/levenary-1.1.1.tgz", - "integrity": "sha512-mkAdOIt79FD6irqjYSs4rdbnlT5vRonMEvBVPVb3XmevfS8kgRXwfes0dhPdEtzTWD/1eNE/Bm/G1iRt6DcnQQ==", - "dev": true, - "requires": { - "leven": "^3.1.0" - } - }, - "license-webpack-plugin": { - "version": "2.1.4", - "resolved": "https://registry.npmjs.org/license-webpack-plugin/-/license-webpack-plugin-2.1.4.tgz", - "integrity": "sha512-1Xq72fmPbTg5KofXs+yI5L4QqPFjQ6mZxoeI6D7gfiEDOtaEIk6PGrdLaej90bpDqKNHNxlQ/MW4tMAL6xMPJQ==", - "dev": true, - "requires": { - "@types/webpack-sources": "^0.1.5", - "webpack-sources": "^1.2.0" - } - }, - "lie": { - "version": "3.3.0", - "resolved": "https://registry.npmjs.org/lie/-/lie-3.3.0.tgz", - "integrity": "sha512-UaiMJzeWRlEujzAuw5LokY1L5ecNQYZKfmyZ9L7wDHb/p5etKaxXhohBcrw0EYby+G/NA52vRSN4N39dxHAIwQ==", - "dev": true, - "requires": { - "immediate": "~3.0.5" - } - }, - "loader-runner": { - "version": "2.4.0", - "resolved": "https://registry.npmjs.org/loader-runner/-/loader-runner-2.4.0.tgz", - "integrity": "sha512-Jsmr89RcXGIwivFY21FcRrisYZfvLMTWx5kOLc+JTxtpBOG6xML0vzbc6SEQG2FO9/4Fc3wW4LVcB5DmGflaRw==", - "dev": true - }, - "loader-utils": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/loader-utils/-/loader-utils-2.0.0.tgz", - "integrity": "sha512-rP4F0h2RaWSvPEkD7BLDFQnvSf+nK+wr3ESUjNTyAGobqrijmW92zc+SO6d4p4B1wh7+B/Jg1mkQe5NYUEHtHQ==", - "dev": true, - "requires": { - "big.js": "^5.2.2", - "emojis-list": "^3.0.0", - "json5": "^2.1.2" - } - }, - "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" - } - }, - "lodash": { - "version": "4.17.21", - "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz", - "integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==", - "dev": true - }, - "lodash.clonedeep": { - "version": "4.5.0", - "resolved": "https://registry.npmjs.org/lodash.clonedeep/-/lodash.clonedeep-4.5.0.tgz", - "integrity": "sha1-4j8/nE+Pvd6HJSnBBxhXoIblzO8=", - "dev": true - }, - "lodash.memoize": { - "version": "4.1.2", - "resolved": "https://registry.npmjs.org/lodash.memoize/-/lodash.memoize-4.1.2.tgz", - "integrity": "sha1-vMbEmkKihA7Zl/Mj6tpezRguC/4=", - "dev": true - }, - "lodash.uniq": { - "version": "4.5.0", - "resolved": "https://registry.npmjs.org/lodash.uniq/-/lodash.uniq-4.5.0.tgz", - "integrity": "sha1-0CJTc662Uq3BvILklFM5qEJ1R3M=", - "dev": true - }, - "log-symbols": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/log-symbols/-/log-symbols-3.0.0.tgz", - "integrity": "sha512-dSkNGuI7iG3mfvDzUuYZyvk5dD9ocYCYzNU6CYDE6+Xqd+gwme6Z00NS3dUh8mq/73HaEtT7m6W+yUPtU6BZnQ==", - "dev": true, - "requires": { - "chalk": "^2.4.2" - } - }, - "log4js": { - "version": "6.4.1", - "resolved": "https://registry.npmjs.org/log4js/-/log4js-6.4.1.tgz", - "integrity": "sha512-iUiYnXqAmNKiIZ1XSAitQ4TmNs8CdZYTAWINARF3LjnsLN8tY5m0vRwd6uuWj/yNY0YHxeZodnbmxKFUOM2rMg==", - "dev": true, - "requires": { - "date-format": "^4.0.3", - "debug": "^4.3.3", - "flatted": "^3.2.4", - "rfdc": "^1.3.0", - "streamroller": "^3.0.2" - }, - "dependencies": { - "debug": { - "version": "4.3.3", - "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.3.tgz", - "integrity": "sha512-/zxw5+vh1Tfv+4Qn7a5nsbcJKPaSvCDhojn6FEl9vupwK2VCSDtEiEtqr8DFtzYFOdz63LBkxec7DYuc2jon6Q==", - "dev": true, - "requires": { - "ms": "2.1.2" - } - } - } - }, - "loglevel": { - "version": "1.6.8", - "resolved": "https://registry.npmjs.org/loglevel/-/loglevel-1.6.8.tgz", - "integrity": "sha512-bsU7+gc9AJ2SqpzxwU3+1fedl8zAntbtC5XYlt3s2j1hJcn2PsXSmgN8TaLG/J1/2mod4+cE/3vNL70/c1RNCA==", - "dev": true - }, - "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" - } - }, - "lru-cache": { - "version": "5.1.1", - "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-5.1.1.tgz", - "integrity": "sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==", - "dev": true, - "requires": { - "yallist": "^3.0.2" - }, - "dependencies": { - "yallist": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz", - "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==", - "dev": true - } - } - }, - "magic-string": { - "version": "0.25.7", - "resolved": "https://registry.npmjs.org/magic-string/-/magic-string-0.25.7.tgz", - "integrity": "sha512-4CrMT5DOHTDk4HYDlzmwu4FVCcIYI8gauveasrdCu2IKIFOJ3f0v/8MDGJCDL9oD2ppz/Av1b0Nj345H9M+XIA==", - "dev": true, - "requires": { - "sourcemap-codec": "^1.4.4" - } - }, - "make-dir": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-2.1.0.tgz", - "integrity": "sha512-LS9X+dc8KLxXCb8dni79fLIIUA5VyZoyjSMCwTluaXA0o27cCK0bhXkpgw+sTXVpPy/lSO57ilRixqk0vDmtRA==", - "dev": true, - "requires": { - "pify": "^4.0.1", - "semver": "^5.6.0" - }, - "dependencies": { - "semver": { - "version": "5.7.1", - "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", - "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", - "dev": true - } - } - }, - "make-error": { - "version": "1.3.6", - "resolved": "https://registry.npmjs.org/make-error/-/make-error-1.3.6.tgz", - "integrity": "sha512-s8UhlNe7vPKomQhC1qFelMokr/Sc3AgNbso3n74mVPA5LTZwkB9NlXf4XPamLxJE8h0gh73rM94xvwRT2CVInw==", - "dev": true - }, - "make-fetch-happen": { - "version": "5.0.2", - "resolved": "https://registry.npmjs.org/make-fetch-happen/-/make-fetch-happen-5.0.2.tgz", - "integrity": "sha512-07JHC0r1ykIoruKO8ifMXu+xEU8qOXDFETylktdug6vJDACnP+HKevOu3PXyNPzFyTSlz8vrBYlBO1JZRe8Cag==", - "dev": true, - "requires": { - "agentkeepalive": "^3.4.1", - "cacache": "^12.0.0", - "http-cache-semantics": "^3.8.1", - "http-proxy-agent": "^2.1.0", - "https-proxy-agent": "^2.2.3", - "lru-cache": "^5.1.1", - "mississippi": "^3.0.0", - "node-fetch-npm": "^2.0.2", - "promise-retry": "^1.1.1", - "socks-proxy-agent": "^4.0.0", - "ssri": "^6.0.0" - }, - "dependencies": { - "cacache": { - "version": "12.0.4", - "resolved": "https://registry.npmjs.org/cacache/-/cacache-12.0.4.tgz", - "integrity": "sha512-a0tMB40oefvuInr4Cwb3GerbL9xTj1D5yg0T5xrjGCGyfvbxseIXX7BAO/u/hIXdafzOI5JC3wDwHyf24buOAQ==", - "dev": true, - "requires": { - "bluebird": "^3.5.5", - "chownr": "^1.1.1", - "figgy-pudding": "^3.5.1", - "glob": "^7.1.4", - "graceful-fs": "^4.1.15", - "infer-owner": "^1.0.3", - "lru-cache": "^5.1.1", - "mississippi": "^3.0.0", - "mkdirp": "^0.5.1", - "move-concurrently": "^1.0.1", - "promise-inflight": "^1.0.1", - "rimraf": "^2.6.3", - "ssri": "^6.0.1", - "unique-filename": "^1.1.1", - "y18n": "^4.0.0" - } - }, - "rimraf": { - "version": "2.7.1", - "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", - "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==", - "dev": true, - "requires": { - "glob": "^7.1.3" - } - }, - "ssri": { - "version": "6.0.2", - "resolved": "https://registry.npmjs.org/ssri/-/ssri-6.0.2.tgz", - "integrity": "sha512-cepbSq/neFK7xB6A50KHN0xHDotYzq58wWCa5LeWqnPrHG8GzfEjO/4O8kpmcGW+oaxkvhEJCWgbgNk4/ZV93Q==", - "dev": true, - "requires": { - "figgy-pudding": "^3.5.1" - } - } - } - }, - "mamacro": { - "version": "0.0.3", - "resolved": "https://registry.npmjs.org/mamacro/-/mamacro-0.0.3.tgz", - "integrity": "sha512-qMEwh+UujcQ+kbz3T6V+wAmO2U8veoq2w+3wY8MquqwVA3jChfwY+Tk52GZKDfACEPjuZ7r2oJLejwpt8jtwTA==", - "dev": true - }, - "map-age-cleaner": { - "version": "0.1.3", - "resolved": "https://registry.npmjs.org/map-age-cleaner/-/map-age-cleaner-0.1.3.tgz", - "integrity": "sha512-bJzx6nMoP6PDLPBFmg7+xRKeFZvFboMrGlxmNj9ClvX53KrmvM5bXFXEWjbz4cz1AFn+jWJ9z/DJSz7hrs0w3w==", - "dev": true, - "requires": { - "p-defer": "^1.0.0" - } - }, - "map-cache": { - "version": "0.2.2", - "resolved": "https://registry.npmjs.org/map-cache/-/map-cache-0.2.2.tgz", - "integrity": "sha1-wyq9C9ZSXZsFFkW7TyasXcmKDb8=", - "dev": true - }, - "map-visit": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/map-visit/-/map-visit-1.0.0.tgz", - "integrity": "sha1-7Nyo8TFE5mDxtb1B8S80edmN+48=", - "dev": true, - "requires": { - "object-visit": "^1.0.0" - } - }, - "marked": { - "version": "4.0.10", - "resolved": "https://registry.npmjs.org/marked/-/marked-4.0.10.tgz", - "integrity": "sha512-+QvuFj0nGgO970fySghXGmuw+Fd0gD2x3+MqCWLIPf5oxdv1Ka6b2q+z9RP01P/IaKPMEramy+7cNy/Lw8c3hw==" - }, - "md5.js": { - "version": "1.3.5", - "resolved": "https://registry.npmjs.org/md5.js/-/md5.js-1.3.5.tgz", - "integrity": "sha512-xitP+WxNPcTTOgnTJcrhM0xvdPepipPSf3I8EIpGKeFLjt3PlJLIDG3u8EX53ZIubkb+5U2+3rELYpEhHhzdkg==", - "dev": true, - "requires": { - "hash-base": "^3.0.0", - "inherits": "^2.0.1", - "safe-buffer": "^5.1.2" - } - }, - "mdn-data": { - "version": "2.0.4", - "resolved": "https://registry.npmjs.org/mdn-data/-/mdn-data-2.0.4.tgz", - "integrity": "sha512-iV3XNKw06j5Q7mi6h+9vbx23Tv7JkjEVgKHW4pimwyDGWm0OIQntJJ+u1C6mg6mK1EaTv42XQ7w76yuzH7M2cA==", - "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=", - "dev": true - }, - "mem": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/mem/-/mem-4.3.0.tgz", - "integrity": "sha512-qX2bG48pTqYRVmDB37rn/6PT7LcR8T7oAX3bf99u1Tt1nzxYfxkgqDwUwolPlXweM0XzBOBFzSx4kfp7KP1s/w==", - "dev": true, - "requires": { - "map-age-cleaner": "^0.1.1", - "mimic-fn": "^2.0.0", - "p-is-promise": "^2.0.0" - } - }, - "memory-fs": { - "version": "0.5.0", - "resolved": "https://registry.npmjs.org/memory-fs/-/memory-fs-0.5.0.tgz", - "integrity": "sha512-jA0rdU5KoQMC0e6ppoNRtpp6vjFq6+NY7r8hywnC7V+1Xj/MtHwGIbB1QaK/dunyjWteJzmkpd7ooeWg10T7GA==", - "dev": true, - "requires": { - "errno": "^0.1.3", - "readable-stream": "^2.0.1" - } - }, - "merge-descriptors": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz", - "integrity": "sha1-sAqqVW3YtEVoFQ7J0blT8/kMu2E=", - "dev": true - }, - "merge-source-map": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/merge-source-map/-/merge-source-map-1.1.0.tgz", - "integrity": "sha512-Qkcp7P2ygktpMPh2mCQZaf3jhN6D3Z/qVZHSdWvQ+2Ef5HgRAPBO57A77+ENm0CPx2+1Ce/MYKi3ymqdfuqibw==", - "dev": true, - "requires": { - "source-map": "^0.6.1" - }, - "dependencies": { - "source-map": { - "version": "0.6.1", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", - "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", - "dev": true - } - } - }, - "merge-stream": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/merge-stream/-/merge-stream-2.0.0.tgz", - "integrity": "sha512-abv/qOcuPfk3URPfDzmZU1LKmuw8kT+0nIHvKrKgFrwifol/doWcdA4ZqsWQ8ENrFKkd67Mfpo/LovbIUsbt3w==", - "dev": true - }, - "methods": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz", - "integrity": "sha1-VSmk1nZUE07cxSZmVoNbD4Ua/O4=", - "dev": true - }, - "micromatch": { - "version": "3.1.10", - "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-3.1.10.tgz", - "integrity": "sha512-MWikgl9n9M3w+bpsY3He8L+w9eF9338xRl8IAO5viDizwSzziFEyUzo2xrrloB64ADbTf8uA8vRqqttDTOmccg==", - "dev": true, - "requires": { - "arr-diff": "^4.0.0", - "array-unique": "^0.3.2", - "braces": "^2.3.1", - "define-property": "^2.0.2", - "extend-shallow": "^3.0.2", - "extglob": "^2.0.4", - "fragment-cache": "^0.2.1", - "kind-of": "^6.0.2", - "nanomatch": "^1.2.9", - "object.pick": "^1.3.0", - "regex-not": "^1.0.0", - "snapdragon": "^0.8.1", - "to-regex": "^3.0.2" - }, - "dependencies": { - "braces": { - "version": "2.3.2", - "resolved": "https://registry.npmjs.org/braces/-/braces-2.3.2.tgz", - "integrity": "sha512-aNdbnj9P8PjdXU4ybaWLK2IF3jc/EoDYbC7AazW6to3TRsfXxscC9UXOB5iDiEQrkyIbWp2SLQda4+QAa7nc3w==", - "dev": true, - "requires": { - "arr-flatten": "^1.1.0", - "array-unique": "^0.3.2", - "extend-shallow": "^2.0.1", - "fill-range": "^4.0.0", - "isobject": "^3.0.1", - "repeat-element": "^1.1.2", - "snapdragon": "^0.8.1", - "snapdragon-node": "^2.0.1", - "split-string": "^3.0.2", - "to-regex": "^3.0.1" - }, - "dependencies": { - "extend-shallow": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", - "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", - "dev": true, - "requires": { - "is-extendable": "^0.1.0" - } - } - } - }, - "fill-range": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-4.0.0.tgz", - "integrity": "sha1-1USBHUKPmOsGpj3EAtJAPDKMOPc=", - "dev": true, - "requires": { - "extend-shallow": "^2.0.1", - "is-number": "^3.0.0", - "repeat-string": "^1.6.1", - "to-regex-range": "^2.1.0" - }, - "dependencies": { - "extend-shallow": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", - "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", - "dev": true, - "requires": { - "is-extendable": "^0.1.0" - } - } - } - }, - "is-number": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/is-number/-/is-number-3.0.0.tgz", - "integrity": "sha1-JP1iAaR4LPUFYcgQJ2r8fRLXEZU=", - "dev": true, - "requires": { - "kind-of": "^3.0.2" - }, - "dependencies": { - "kind-of": { - "version": "3.2.2", - "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", - "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", - "dev": true, - "requires": { - "is-buffer": "^1.1.5" - } - } - } - }, - "to-regex-range": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-2.1.1.tgz", - "integrity": "sha1-fIDBe53+vlmeJzZ+DU3VWQFB2zg=", - "dev": true, - "requires": { - "is-number": "^3.0.0", - "repeat-string": "^1.6.1" - } - } - } - }, - "miller-rabin": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/miller-rabin/-/miller-rabin-4.0.1.tgz", - "integrity": "sha512-115fLhvZVqWwHPbClyntxEVfVDfl9DLLTuJvq3g2O/Oxi8AiNouAHvDSzHS0viUJc+V5vm3eq91Xwqn9dp4jRA==", - "dev": true, - "requires": { - "bn.js": "^4.0.0", - "brorand": "^1.0.1" - }, - "dependencies": { - "bn.js": { - "version": "4.11.8", - "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.11.8.tgz", - "integrity": "sha512-ItfYfPLkWHUjckQCk8xC+LwxgK8NYcXywGigJgSwOP8Y2iyWT4f2vsZnoOXTTbo+o5yXmIUJ4gn5538SO5S3gA==", - "dev": true - } - } - }, - "mime": { - "version": "1.6.0", - "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz", - "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==", - "dev": true - }, - "mime-db": { - "version": "1.44.0", - "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.44.0.tgz", - "integrity": "sha512-/NOTfLrsPBVeH7YtFPgsVWveuL+4SjjYxaQ1xtM1KMFj7HdxlBlxeyNLzhyJVx7r4rZGJAZ/6lkKCitSc/Nmpg==", - "dev": true - }, - "mime-types": { - "version": "2.1.27", - "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.27.tgz", - "integrity": "sha512-JIhqnCasI9yD+SsmkquHBxTSEuZdQX5BuQnS2Vc7puQQQ+8yiP5AY5uWhpdv4YL4VM5c6iliiYWPgJ/nJQLp7w==", - "dev": true, - "requires": { - "mime-db": "1.44.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 - }, - "mini-css-extract-plugin": { - "version": "0.9.0", - "resolved": "https://registry.npmjs.org/mini-css-extract-plugin/-/mini-css-extract-plugin-0.9.0.tgz", - "integrity": "sha512-lp3GeY7ygcgAmVIcRPBVhIkf8Us7FZjA+ILpal44qLdSu11wmjKQ3d9k15lfD7pO4esu9eUIAW7qiYIBppv40A==", - "dev": true, - "requires": { - "loader-utils": "^1.1.0", - "normalize-url": "1.9.1", - "schema-utils": "^1.0.0", - "webpack-sources": "^1.1.0" - }, - "dependencies": { - "json5": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/json5/-/json5-1.0.1.tgz", - "integrity": "sha512-aKS4WQjPenRxiQsC93MNfjx+nbF4PAdYzmd/1JIj8HYzqfbu86beTuNgXDzPknWk0n0uARlyewZo4s++ES36Ow==", - "dev": true, - "requires": { - "minimist": "^1.2.0" - } - }, - "loader-utils": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/loader-utils/-/loader-utils-1.4.0.tgz", - "integrity": "sha512-qH0WSMBtn/oHuwjy/NucEgbx5dbxxnxup9s4PVXJUDHZBQY+s0NWA9rJf53RBnQZxfch7euUui7hpoAPvALZdA==", - "dev": true, - "requires": { - "big.js": "^5.2.2", - "emojis-list": "^3.0.0", - "json5": "^1.0.1" - } - }, - "normalize-url": { - "version": "1.9.1", - "resolved": "https://registry.npmjs.org/normalize-url/-/normalize-url-1.9.1.tgz", - "integrity": "sha1-LMDWazHqIwNkWENuNiDYWVTGbDw=", - "dev": true, - "requires": { - "object-assign": "^4.0.1", - "prepend-http": "^1.0.0", - "query-string": "^4.1.0", - "sort-keys": "^1.0.0" - } - }, - "schema-utils": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-1.0.0.tgz", - "integrity": "sha512-i27Mic4KovM/lnGsy8whRCHhc7VicJajAjTrYg11K9zfZXnYIt4k5F+kZkwjnrhKzLic/HLU4j11mjsz2G/75g==", - "dev": true, - "requires": { - "ajv": "^6.1.0", - "ajv-errors": "^1.0.0", - "ajv-keywords": "^3.1.0" - } - } - } - }, - "minimalistic-assert": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/minimalistic-assert/-/minimalistic-assert-1.0.1.tgz", - "integrity": "sha512-UtJcAD4yEaGtjPezWuO9wC4nwUnVH/8/Im3yEHQP4b67cXlD/Qr9hdITCU1xDbSEXg2XKNaP8jsReV7vQd00/A==", - "dev": true - }, - "minimalistic-crypto-utils": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/minimalistic-crypto-utils/-/minimalistic-crypto-utils-1.0.1.tgz", - "integrity": "sha1-9sAMHAsIIkblxNmd+4x8CDsrWCo=", - "dev": true - }, - "minimatch": { - "version": "3.0.4", - "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz", - "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==", - "dev": true, - "requires": { - "brace-expansion": "^1.1.7" - } - }, - "minimist": { - "version": "1.2.6", - "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.6.tgz", - "integrity": "sha512-Jsjnk4bw3YJqYzbdyBiNsPWHPfO++UGG749Cxs6peCu5Xg4nrena6OVxOYxrQTqww0Jmwt+Ref8rggumkTLz9Q==", - "dev": true - }, - "minipass": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/minipass/-/minipass-3.1.1.tgz", - "integrity": "sha512-UFqVihv6PQgwj8/yTGvl9kPz7xIAY+R5z6XYjRInD3Gk3qx6QGSD6zEcpeG4Dy/lQnv1J6zv8ejV90hyYIKf3w==", - "dev": true, - "requires": { - "yallist": "^4.0.0" - } - }, - "minipass-collect": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/minipass-collect/-/minipass-collect-1.0.2.tgz", - "integrity": "sha512-6T6lH0H8OG9kITm/Jm6tdooIbogG9e0tLgpY6mphXSm/A9u8Nq1ryBG+Qspiub9LjWlBPsPS3tWQ/Botq4FdxA==", - "dev": true, - "requires": { - "minipass": "^3.0.0" - } - }, - "minipass-flush": { - "version": "1.0.5", - "resolved": "https://registry.npmjs.org/minipass-flush/-/minipass-flush-1.0.5.tgz", - "integrity": "sha512-JmQSYYpPUqX5Jyn1mXaRwOda1uQ8HP5KAT/oDSLCzt1BYRhQU0/hDtsB1ufZfEEzMZ9aAVmsBw8+FWsIXlClWw==", - "dev": true, - "requires": { - "minipass": "^3.0.0" - } - }, - "minipass-pipeline": { - "version": "1.2.2", - "resolved": "https://registry.npmjs.org/minipass-pipeline/-/minipass-pipeline-1.2.2.tgz", - "integrity": "sha512-3JS5A2DKhD2g0Gg8x3yamO0pj7YeKGwVlDS90pF++kxptwx/F+B//roxf9SqYil5tQo65bijy+dAuAFZmYOouA==", - "dev": true, - "requires": { - "minipass": "^3.0.0" - } - }, - "mississippi": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/mississippi/-/mississippi-3.0.0.tgz", - "integrity": "sha512-x471SsVjUtBRtcvd4BzKE9kFC+/2TeWgKCgw0bZcw1b9l2X3QX5vCWgF+KaZaYm87Ss//rHnWryupDrgLvmSkA==", - "dev": true, - "requires": { - "concat-stream": "^1.5.0", - "duplexify": "^3.4.2", - "end-of-stream": "^1.1.0", - "flush-write-stream": "^1.0.0", - "from2": "^2.1.0", - "parallel-transform": "^1.1.0", - "pump": "^3.0.0", - "pumpify": "^1.3.3", - "stream-each": "^1.1.0", - "through2": "^2.0.0" - } - }, - "mixin-deep": { - "version": "1.3.2", - "resolved": "https://registry.npmjs.org/mixin-deep/-/mixin-deep-1.3.2.tgz", - "integrity": "sha512-WRoDn//mXBiJ1H40rqa3vH0toePwSsGb45iInWlTySa+Uu4k3tYUSxa2v1KqAiLtvlrSzaExqS1gtk96A9zvEA==", - "dev": true, - "requires": { - "for-in": "^1.0.2", - "is-extendable": "^1.0.1" - }, - "dependencies": { - "is-extendable": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/is-extendable/-/is-extendable-1.0.1.tgz", - "integrity": "sha512-arnXMxT1hhoKo9k1LZdmlNyJdDDfy2v0fXjFlmok4+i8ul/6WlbVge9bhM74OpNPQPMGUToDtz+KXa1PneJxOA==", - "dev": true, - "requires": { - "is-plain-object": "^2.0.4" - } - } - } - }, - "mkdirp": { - "version": "0.5.5", - "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.5.tgz", - "integrity": "sha512-NKmAlESf6jMGym1++R0Ra7wvhV+wFW63FaSOFPwRahvea0gMUcGUhVeAg/0BC0wiv9ih5NYPB1Wn1UEI1/L+xQ==", - "dev": true, - "requires": { - "minimist": "^1.2.5" - } - }, - "move-concurrently": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/move-concurrently/-/move-concurrently-1.0.1.tgz", - "integrity": "sha1-viwAX9oy4LKa8fBdfEszIUxwH5I=", - "dev": true, - "requires": { - "aproba": "^1.1.1", - "copy-concurrently": "^1.0.0", - "fs-write-stream-atomic": "^1.0.8", - "mkdirp": "^0.5.1", - "rimraf": "^2.5.4", - "run-queue": "^1.0.3" - }, - "dependencies": { - "rimraf": { - "version": "2.7.1", - "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", - "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==", - "dev": true, - "requires": { - "glob": "^7.1.3" - } - } - } - }, - "ms": { - "version": "2.1.2", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", - "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", - "dev": true - }, - "multicast-dns": { - "version": "6.2.3", - "resolved": "https://registry.npmjs.org/multicast-dns/-/multicast-dns-6.2.3.tgz", - "integrity": "sha512-ji6J5enbMyGRHIAkAOu3WdV8nggqviKCEKtXcOqfphZZtQrmHKycfynJ2V7eVPUA4NhJ6V7Wf4TmGbTwKE9B6g==", - "dev": true, - "requires": { - "dns-packet": "^1.3.1", - "thunky": "^1.0.2" - } - }, - "multicast-dns-service-types": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/multicast-dns-service-types/-/multicast-dns-service-types-1.1.0.tgz", - "integrity": "sha1-iZ8R2WhuXgXLkbNdXw5jt3PPyQE=", - "dev": true - }, - "mute-stream": { - "version": "0.0.8", - "resolved": "https://registry.npmjs.org/mute-stream/-/mute-stream-0.0.8.tgz", - "integrity": "sha512-nnbWWOkoWyUsTjKrhgD0dcz22mdkSnpYqbEjIm2nhwhuxlSkpywJmBo8h0ZqJdkp73mb90SssHkN4rsRaBAfAA==", - "dev": true - }, - "nan": { - "version": "2.14.1", - "resolved": "https://registry.npmjs.org/nan/-/nan-2.14.1.tgz", - "integrity": "sha512-isWHgVjnFjh2x2yuJ/tj3JbwoHu3UC2dX5G/88Cm24yB6YopVgxvBObDY7n5xW6ExmFhJpSEQqFPvq9zaXc8Jw==" - }, - "nanomatch": { - "version": "1.2.13", - "resolved": "https://registry.npmjs.org/nanomatch/-/nanomatch-1.2.13.tgz", - "integrity": "sha512-fpoe2T0RbHwBTBUOftAfBPaDEi06ufaUai0mE6Yn1kacc3SnTErfb/h+X94VXzI64rKFHYImXSvdwGGCmwOqCA==", - "dev": true, - "requires": { - "arr-diff": "^4.0.0", - "array-unique": "^0.3.2", - "define-property": "^2.0.2", - "extend-shallow": "^3.0.2", - "fragment-cache": "^0.2.1", - "is-windows": "^1.0.2", - "kind-of": "^6.0.2", - "object.pick": "^1.3.0", - "regex-not": "^1.0.0", - "snapdragon": "^0.8.1", - "to-regex": "^3.0.1" - } - }, - "negotiator": { - "version": "0.6.2", - "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.2.tgz", - "integrity": "sha512-hZXc7K2e+PgeI1eDBe/10Ard4ekbfrrqG8Ep+8Jmf4JID2bNg7NvCPOZN+kfF574pFQI7mum2AUqDidoKqcTOw==", - "dev": true - }, - "neo-async": { - "version": "2.6.1", - "resolved": "https://registry.npmjs.org/neo-async/-/neo-async-2.6.1.tgz", - "integrity": "sha512-iyam8fBuCUpWeKPGpaNMetEocMt364qkCsfL9JuhjXX6dRnguRVOfk2GZaDpPjcOKiiXCPINZC1GczQ7iTq3Zw==", - "dev": true - }, - "ngraph.events": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/ngraph.events/-/ngraph.events-1.2.1.tgz", - "integrity": "sha512-D4C+nXH/RFxioGXQdHu8ELDtC6EaCiNsZtih0IvyGN81OZSUby4jXoJ5+RNWasfsd0FnKxxpAROyUMzw64QNsw==" - }, - "ngx-highlightjs": { - "version": "4.1.0-beta", - "resolved": "https://registry.npmjs.org/ngx-highlightjs/-/ngx-highlightjs-4.1.0-beta.tgz", - "integrity": "sha512-1eyiHjaFadFZpA9kGFRATccdQiB1ERgkOxmAIMFPZzN398FuCN0nrpPezECvdR7Gqlbd9OeSJHzehS20DLVy7A==", - "requires": { - "highlight.js": "*", - "highlightjs-line-numbers.js": "*" - } - }, - "nice-try": { - "version": "1.0.5", - "resolved": "https://registry.npmjs.org/nice-try/-/nice-try-1.0.5.tgz", - "integrity": "sha512-1nh45deeb5olNY7eX82BkPO7SSxR5SSYJiPTrTdFUVYwAl8CKMA5N9PjTYkHiRjisVcxcQ1HXdLhx2qxxJzLNQ==", - "dev": true - }, - "node-fetch-npm": { - "version": "2.0.4", - "resolved": "https://registry.npmjs.org/node-fetch-npm/-/node-fetch-npm-2.0.4.tgz", - "integrity": "sha512-iOuIQDWDyjhv9qSDrj9aq/klt6F9z1p2otB3AV7v3zBDcL/x+OfGsvGQZZCcMZbUf4Ujw1xGNQkjvGnVT22cKg==", - "dev": true, - "requires": { - "encoding": "^0.1.11", - "json-parse-better-errors": "^1.0.0", - "safe-buffer": "^5.1.1" - } - }, - "node-forge": { - "version": "0.9.0", - "resolved": "https://registry.npmjs.org/node-forge/-/node-forge-0.9.0.tgz", - "integrity": "sha512-7ASaDa3pD+lJ3WvXFsxekJQelBKRpne+GOVbLbtHYdd7pFspyeuJHnWfLplGf3SwKGbfs/aYl5V/JCIaHVUKKQ==", - "dev": true - }, - "node-libs-browser": { - "version": "2.2.1", - "resolved": "https://registry.npmjs.org/node-libs-browser/-/node-libs-browser-2.2.1.tgz", - "integrity": "sha512-h/zcD8H9kaDZ9ALUWwlBUDo6TKF8a7qBSCSEGfjTVIYeqsioSKaAX+BN7NgiMGp6iSIXZ3PxgCu8KS3b71YK5Q==", - "dev": true, - "requires": { - "assert": "^1.1.1", - "browserify-zlib": "^0.2.0", - "buffer": "^4.3.0", - "console-browserify": "^1.1.0", - "constants-browserify": "^1.0.0", - "crypto-browserify": "^3.11.0", - "domain-browser": "^1.1.1", - "events": "^3.0.0", - "https-browserify": "^1.0.0", - "os-browserify": "^0.3.0", - "path-browserify": "0.0.1", - "process": "^0.11.10", - "punycode": "^1.2.4", - "querystring-es3": "^0.2.0", - "readable-stream": "^2.3.3", - "stream-browserify": "^2.0.1", - "stream-http": "^2.7.2", - "string_decoder": "^1.0.0", - "timers-browserify": "^2.0.4", - "tty-browserify": "0.0.0", - "url": "^0.11.0", - "util": "^0.11.0", - "vm-browserify": "^1.0.1" - }, - "dependencies": { - "punycode": { - "version": "1.4.1", - "resolved": "https://registry.npmjs.org/punycode/-/punycode-1.4.1.tgz", - "integrity": "sha1-wNWmOycYgArY4esPpSachN1BhF4=", - "dev": true - } - } - }, - "node-releases": { - "version": "1.1.55", - "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-1.1.55.tgz", - "integrity": "sha512-H3R3YR/8TjT5WPin/wOoHOUPHgvj8leuU/Keta/rwelEQN9pA/S2Dx8/se4pZ2LBxSd0nAGzsNzhqwa77v7F1w==", - "dev": true - }, - "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==", - "dev": true, - "requires": { - "hosted-git-info": "^2.1.4", - "resolve": "^1.10.0", - "semver": "2 || 3 || 4 || 5", - "validate-npm-package-license": "^3.0.1" - }, - "dependencies": { - "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==", - "dev": true - }, - "semver": { - "version": "5.7.1", - "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", - "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", - "dev": true - } - } - }, - "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-range": { - "version": "0.1.2", - "resolved": "https://registry.npmjs.org/normalize-range/-/normalize-range-0.1.2.tgz", - "integrity": "sha1-LRDAa9/TEuqXd2laTShDlFa3WUI=", - "dev": true - }, - "normalize-url": { - "version": "3.3.0", - "resolved": "https://registry.npmjs.org/normalize-url/-/normalize-url-3.3.0.tgz", - "integrity": "sha512-U+JJi7duF1o+u2pynbp2zXDW2/PADgC30f0GsHZtRh+HOcXHnw137TrNlyxxRvWW5fjKd3bcLHPxofWuCjaeZg==", - "dev": true - }, - "npm-bundled": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/npm-bundled/-/npm-bundled-1.1.1.tgz", - "integrity": "sha512-gqkfgGePhTpAEgUsGEgcq1rqPXA+tv/aVBlgEzfXwA1yiUJF7xtEt3CtVwOjNYQOVknDk0F20w58Fnm3EtG0fA==", - "dev": true, - "requires": { - "npm-normalize-package-bin": "^1.0.1" - } - }, - "npm-install-checks": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/npm-install-checks/-/npm-install-checks-4.0.0.tgz", - "integrity": "sha512-09OmyDkNLYwqKPOnbI8exiOZU2GVVmQp7tgez2BPi5OZC8M82elDAps7sxC4l//uSUtotWqoEIDwjRvWH4qz8w==", - "dev": true, - "requires": { - "semver": "^7.1.1" - }, - "dependencies": { - "semver": { - "version": "7.3.2", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.2.tgz", - "integrity": "sha512-OrOb32TeeambH6UrhtShmF7CRDqhL6/5XpPNp2DuRH6+9QLw/orhp72j87v8Qa1ScDkvrrBNpZcDejAirJmfXQ==", - "dev": true - } - } - }, - "npm-normalize-package-bin": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/npm-normalize-package-bin/-/npm-normalize-package-bin-1.0.1.tgz", - "integrity": "sha512-EPfafl6JL5/rU+ot6P3gRSCpPDW5VmIzX959Ob1+ySFUuuYHWHekXpwdUZcKP5C+DS4GEtdJluwBjnsNDl+fSA==", - "dev": true - }, - "npm-package-arg": { - "version": "8.0.1", - "resolved": "https://registry.npmjs.org/npm-package-arg/-/npm-package-arg-8.0.1.tgz", - "integrity": "sha512-/h5Fm6a/exByzFSTm7jAyHbgOqErl9qSNJDQF32Si/ZzgwT2TERVxRxn3Jurw1wflgyVVAxnFR4fRHPM7y1ClQ==", - "dev": true, - "requires": { - "hosted-git-info": "^3.0.2", - "semver": "^7.0.0", - "validate-npm-package-name": "^3.0.0" - }, - "dependencies": { - "semver": { - "version": "7.3.2", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.2.tgz", - "integrity": "sha512-OrOb32TeeambH6UrhtShmF7CRDqhL6/5XpPNp2DuRH6+9QLw/orhp72j87v8Qa1ScDkvrrBNpZcDejAirJmfXQ==", - "dev": true - } - } - }, - "npm-packlist": { - "version": "1.4.8", - "resolved": "https://registry.npmjs.org/npm-packlist/-/npm-packlist-1.4.8.tgz", - "integrity": "sha512-5+AZgwru5IevF5ZdnFglB5wNlHG1AOOuw28WhUq8/8emhBmLv6jX5by4WJCh7lW0uSYZYS6DXqIsyZVIXRZU9A==", - "dev": true, - "requires": { - "ignore-walk": "^3.0.1", - "npm-bundled": "^1.0.1", - "npm-normalize-package-bin": "^1.0.1" - } - }, - "npm-pick-manifest": { - "version": "6.0.0", - "resolved": "https://registry.npmjs.org/npm-pick-manifest/-/npm-pick-manifest-6.0.0.tgz", - "integrity": "sha512-PdJpXMvjqt4nftNEDpCgjBUF8yI3Q3MyuAmVB9nemnnCg32F4BPL/JFBfdj8DubgHCYUFQhtLWmBPvdsFtjWMg==", - "dev": true, - "requires": { - "npm-install-checks": "^4.0.0", - "npm-package-arg": "^8.0.0", - "semver": "^7.0.0" - }, - "dependencies": { - "semver": { - "version": "7.3.2", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.2.tgz", - "integrity": "sha512-OrOb32TeeambH6UrhtShmF7CRDqhL6/5XpPNp2DuRH6+9QLw/orhp72j87v8Qa1ScDkvrrBNpZcDejAirJmfXQ==", - "dev": true - } - } - }, - "npm-registry-fetch": { - "version": "4.0.4", - "resolved": "https://registry.npmjs.org/npm-registry-fetch/-/npm-registry-fetch-4.0.4.tgz", - "integrity": "sha512-6jb34hX/iYNQebqWUHtU8YF6Cjb1H6ouTFPClYsyiW6lpFkljTpdeftm53rRojtja1rKAvKNIIiTS5Sjpw4wsA==", - "dev": true, - "requires": { - "JSONStream": "^1.3.4", - "bluebird": "^3.5.1", - "figgy-pudding": "^3.4.1", - "lru-cache": "^5.1.1", - "make-fetch-happen": "^5.0.0", - "npm-package-arg": "^6.1.0", - "safe-buffer": "^5.2.0" - }, - "dependencies": { - "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==", - "dev": true - }, - "npm-package-arg": { - "version": "6.1.1", - "resolved": "https://registry.npmjs.org/npm-package-arg/-/npm-package-arg-6.1.1.tgz", - "integrity": "sha512-qBpssaL3IOZWi5vEKUKW0cO7kzLeT+EQO9W8RsLOZf76KF9E/K9+wH0C7t06HXPpaH8WH5xF1MExLuCwbTqRUg==", - "dev": true, - "requires": { - "hosted-git-info": "^2.7.1", - "osenv": "^0.1.5", - "semver": "^5.6.0", - "validate-npm-package-name": "^3.0.0" - } - }, - "safe-buffer": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.0.tgz", - "integrity": "sha512-fZEwUGbVl7kouZs1jCdMLdt95hdIv0ZeHg6L7qPeciMZhZ+/gdesW4wgTARkrFWEpspjEATAzUGPG8N2jJiwbg==", - "dev": true - }, - "semver": { - "version": "5.7.1", - "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", - "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", - "dev": true - } - } - }, - "npm-run-path": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/npm-run-path/-/npm-run-path-2.0.2.tgz", - "integrity": "sha1-NakjLfo11wZ7TLLd8jV7GHFTbF8=", - "dev": true, - "requires": { - "path-key": "^2.0.0" - } - }, - "nth-check": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/nth-check/-/nth-check-1.0.2.tgz", - "integrity": "sha512-WeBOdju8SnzPN5vTUJYxYUxLeXpCaVP5i5e0LF8fg7WORF2Wd7wFX/pk0tYZk7s8T+J7VLy0Da6J1+wCT0AtHg==", - "dev": true, - "requires": { - "boolbase": "~1.0.0" - } - }, - "num2fraction": { - "version": "1.2.2", - "resolved": "https://registry.npmjs.org/num2fraction/-/num2fraction-1.2.2.tgz", - "integrity": "sha1-b2gragJ6Tp3fpFZM0lidHU5mnt4=", - "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=", - "dev": true - }, - "oauth-sign": { - "version": "0.9.0", - "resolved": "https://registry.npmjs.org/oauth-sign/-/oauth-sign-0.9.0.tgz", - "integrity": "sha512-fexhUFFPTGV8ybAtSIGbV6gOkSv8UtRbDBnAyLQw4QPKkgNlsH2ByPGtMUqdWkos6YCRmAqViwgZrJc/mRDzZQ==", - "dev": true - }, - "object-assign": { - "version": "4.1.1", - "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", - "integrity": "sha1-IQmtx5ZYh8/AXLvUQsrIv7s2CGM=", - "dev": true - }, - "object-copy": { - "version": "0.1.0", - "resolved": "https://registry.npmjs.org/object-copy/-/object-copy-0.1.0.tgz", - "integrity": "sha1-fn2Fi3gb18mRpBupde04EnVOmYw=", - "dev": true, - "requires": { - "copy-descriptor": "^0.1.0", - "define-property": "^0.2.5", - "kind-of": "^3.0.3" - }, - "dependencies": { - "define-property": { - "version": "0.2.5", - "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", - "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", - "dev": true, - "requires": { - "is-descriptor": "^0.1.0" - } - }, - "kind-of": { - "version": "3.2.2", - "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", - "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", - "dev": true, - "requires": { - "is-buffer": "^1.1.5" - } - } - } - }, - "object-inspect": { - "version": "1.7.0", - "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.7.0.tgz", - "integrity": "sha512-a7pEHdh1xKIAgTySUGgLMx/xwDZskN1Ud6egYYN3EdRW4ZMPNEDUTF+hwy2LUC+Bl+SyLXANnwz/jyh/qutKUw==", - "dev": true - }, - "object-is": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/object-is/-/object-is-1.1.2.tgz", - "integrity": "sha512-5lHCz+0uufF6wZ7CRFWJN3hp8Jqblpgve06U5CMQ3f//6iDjPr2PEo9MWCjEssDsa+UZEL4PkFpr+BMop6aKzQ==", - "dev": true, - "requires": { - "define-properties": "^1.1.3", - "es-abstract": "^1.17.5" - } - }, - "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-visit": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/object-visit/-/object-visit-1.0.1.tgz", - "integrity": "sha1-95xEk68MU3e1n+OdOV5BBC3QRbs=", - "dev": true, - "requires": { - "isobject": "^3.0.0" - } - }, - "object.assign": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/object.assign/-/object.assign-4.1.0.tgz", - "integrity": "sha512-exHJeq6kBKj58mqGyTQ9DFvrZC/eR6OwxzoM9YRoGBqrXYonaFyGiFMuc9VZrXf7DarreEwMpurG3dd+CNyW5w==", - "dev": true, - "requires": { - "define-properties": "^1.1.2", - "function-bind": "^1.1.1", - "has-symbols": "^1.0.0", - "object-keys": "^1.0.11" - } - }, - "object.getownpropertydescriptors": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/object.getownpropertydescriptors/-/object.getownpropertydescriptors-2.1.0.tgz", - "integrity": "sha512-Z53Oah9A3TdLoblT7VKJaTDdXdT+lQO+cNpKVnya5JDe9uLvzu1YyY1yFDFrcxrlRgWrEFH0jJtD/IbuwjcEVg==", - "dev": true, - "requires": { - "define-properties": "^1.1.3", - "es-abstract": "^1.17.0-next.1" - } - }, - "object.pick": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/object.pick/-/object.pick-1.3.0.tgz", - "integrity": "sha1-h6EKxMFpS9Lhy/U1kaZhQftd10c=", - "dev": true, - "requires": { - "isobject": "^3.0.1" - } - }, - "object.values": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/object.values/-/object.values-1.1.1.tgz", - "integrity": "sha512-WTa54g2K8iu0kmS/us18jEmdv1a4Wi//BZ/DTVYEcH0XhLM5NYdpDHja3gt57VrZLcNAO2WGA+KpWsDBaHt6eA==", - "dev": true, - "requires": { - "define-properties": "^1.1.3", - "es-abstract": "^1.17.0-next.1", - "function-bind": "^1.1.1", - "has": "^1.0.3" - } - }, - "obuf": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/obuf/-/obuf-1.1.2.tgz", - "integrity": "sha512-PX1wu0AmAdPqOL1mWhqmlOd8kOIZQwGZw6rh7uby9fTc5lhaOWFLX3I6R1hrF9k3zUY40e6igsLGkDXK92LJNg==", - "dev": true - }, - "on-finished": { - "version": "2.3.0", - "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.3.0.tgz", - "integrity": "sha1-IPEzZIGwg811M3mSoWlxqi2QaUc=", - "dev": true, - "requires": { - "ee-first": "1.1.1" - } - }, - "on-headers": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/on-headers/-/on-headers-1.0.2.tgz", - "integrity": "sha512-pZAE+FJLoyITytdqK0U5s+FIpjN0JP3OzFi/u8Rx+EV5/W+JTWGXG8xFzevE7AjBfDqHv/8vL8qQsIhHnqRkrA==", - "dev": true - }, - "once": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", - "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=", - "dev": true, - "requires": { - "wrappy": "1" - } - }, - "onetime": { - "version": "5.1.0", - "resolved": "https://registry.npmjs.org/onetime/-/onetime-5.1.0.tgz", - "integrity": "sha512-5NcSkPHhwTVFIQN+TUqXoS5+dlElHXdpAWu9I0HP20YOtIi+aZ0Ct82jdlILDxjLEAWwvm+qj1m6aEtsDVmm6Q==", - "dev": true, - "requires": { - "mimic-fn": "^2.1.0" - } - }, - "open": { - "version": "7.0.3", - "resolved": "https://registry.npmjs.org/open/-/open-7.0.3.tgz", - "integrity": "sha512-sP2ru2v0P290WFfv49Ap8MF6PkzGNnGlAwHweB4WR4mr5d2d0woiCluUeJ218w7/+PmoBy9JmYgD5A4mLcWOFA==", - "dev": true, - "requires": { - "is-docker": "^2.0.0", - "is-wsl": "^2.1.1" - } - }, - "opn": { - "version": "5.5.0", - "resolved": "https://registry.npmjs.org/opn/-/opn-5.5.0.tgz", - "integrity": "sha512-PqHpggC9bLV0VeWcdKhkpxY+3JTzetLSqTCWL/z/tFIbI6G8JCjondXklT1JinczLz2Xib62sSp0T/gKT4KksA==", - "dev": true, - "requires": { - "is-wsl": "^1.1.0" - }, - "dependencies": { - "is-wsl": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/is-wsl/-/is-wsl-1.1.0.tgz", - "integrity": "sha1-HxbkqiKwTRM2tmGIpmrzxgDDpm0=", - "dev": true - } - } - }, - "ora": { - "version": "4.0.3", - "resolved": "https://registry.npmjs.org/ora/-/ora-4.0.3.tgz", - "integrity": "sha512-fnDebVFyz309A73cqCipVL1fBZewq4vwgSHfxh43vVy31mbyoQ8sCH3Oeaog/owYOs/lLlGVPCISQonTneg6Pg==", - "dev": true, - "requires": { - "chalk": "^3.0.0", - "cli-cursor": "^3.1.0", - "cli-spinners": "^2.2.0", - "is-interactive": "^1.0.0", - "log-symbols": "^3.0.0", - "mute-stream": "0.0.8", - "strip-ansi": "^6.0.0", - "wcwidth": "^1.0.1" - }, - "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.2.1", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.2.1.tgz", - "integrity": "sha512-9VGjrMsG1vePxcSweQsN20KY/c4zN0h9fLjqAbwbPfahM3t+NL+M9HC8xeXG2I8pX5NoamTGNuomEUFI7fcUjA==", - "dev": true, - "requires": { - "@types/color-name": "^1.1.1", - "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 - }, - "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.1.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.1.0.tgz", - "integrity": "sha512-oRSIpR8pxT1Wr2FquTNnGet79b3BWljqOuoW/h4oBhxJ/HUbX5nX6JSruTkvXDCFMwDPvsaTTbvMLKZWSy0R5g==", - "dev": true, - "requires": { - "has-flag": "^4.0.0" - } - } - } - }, - "original": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/original/-/original-1.0.2.tgz", - "integrity": "sha512-hyBVl6iqqUOJ8FqRe+l/gS8H+kKYjrEndd5Pm1MfBtsEKA038HkkdbAl/72EAXGyonD/PFsvmVG+EvcIpliMBg==", - "dev": true, - "requires": { - "url-parse": "^1.4.3" - } - }, - "os-browserify": { - "version": "0.3.0", - "resolved": "https://registry.npmjs.org/os-browserify/-/os-browserify-0.3.0.tgz", - "integrity": "sha1-hUNzx/XCMVkU/Jv8a9gjj92h7Cc=", - "dev": true - }, - "os-homedir": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/os-homedir/-/os-homedir-1.0.2.tgz", - "integrity": "sha1-/7xJiDNuDoM94MFox+8VISGqf7M=", - "dev": true - }, - "os-locale": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/os-locale/-/os-locale-3.1.0.tgz", - "integrity": "sha512-Z8l3R4wYWM40/52Z+S265okfFj8Kt2cC2MKY+xNi3kFs+XGI7WXu/I309QQQYbRW4ijiZ+yxs9pqEhJh0DqW3Q==", - "dev": true, - "requires": { - "execa": "^1.0.0", - "lcid": "^2.0.0", - "mem": "^4.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 - }, - "osenv": { - "version": "0.1.5", - "resolved": "https://registry.npmjs.org/osenv/-/osenv-0.1.5.tgz", - "integrity": "sha512-0CWcCECdMVc2Rw3U5w9ZjqX6ga6ubk1xDVKxtBQPK7wis/0F2r9T6k4ydGYhecl7YUBxBVxhL5oisPsNxAPe2g==", - "dev": true, - "requires": { - "os-homedir": "^1.0.0", - "os-tmpdir": "^1.0.0" - } - }, - "p-defer": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/p-defer/-/p-defer-1.0.0.tgz", - "integrity": "sha1-n26xgvbJqozXQwBKfU+WsZaw+ww=", - "dev": true - }, - "p-finally": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/p-finally/-/p-finally-1.0.0.tgz", - "integrity": "sha1-P7z7FbiZpEEjs0ttzBi3JDNqLK4=", - "dev": true - }, - "p-is-promise": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/p-is-promise/-/p-is-promise-2.1.0.tgz", - "integrity": "sha512-Y3W0wlRPK8ZMRbNq97l4M5otioeA5lm1z7bkNkxCka8HSPjR0xRWmpCmc9utiaLP9Jb1eD8BgeIxTW4AIF45Pg==", - "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-map": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/p-map/-/p-map-3.0.0.tgz", - "integrity": "sha512-d3qXVTF/s+W+CdJ5A29wywV2n8CQQYahlgz2bFiA+4eVNJbHJodPZ+/gXwPGh0bOqA+j8S+6+ckmvLGPk1QpxQ==", - "dev": true, - "requires": { - "aggregate-error": "^3.0.0" - } - }, - "p-retry": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/p-retry/-/p-retry-3.0.1.tgz", - "integrity": "sha512-XE6G4+YTTkT2a0UWb2kjZe8xNwf8bIbnqpc/IS/idOBVhyves0mK5OJgeocjx7q5pvX/6m23xuzVPYT1uGM73w==", - "dev": true, - "requires": { - "retry": "^0.12.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 - }, - "pacote": { - "version": "9.5.12", - "resolved": "https://registry.npmjs.org/pacote/-/pacote-9.5.12.tgz", - "integrity": "sha512-BUIj/4kKbwWg4RtnBncXPJd15piFSVNpTzY0rysSr3VnMowTYgkGKcaHrbReepAkjTr8lH2CVWRi58Spg2CicQ==", - "dev": true, - "requires": { - "bluebird": "^3.5.3", - "cacache": "^12.0.2", - "chownr": "^1.1.2", - "figgy-pudding": "^3.5.1", - "get-stream": "^4.1.0", - "glob": "^7.1.3", - "infer-owner": "^1.0.4", - "lru-cache": "^5.1.1", - "make-fetch-happen": "^5.0.0", - "minimatch": "^3.0.4", - "minipass": "^2.3.5", - "mississippi": "^3.0.0", - "mkdirp": "^0.5.1", - "normalize-package-data": "^2.4.0", - "npm-normalize-package-bin": "^1.0.0", - "npm-package-arg": "^6.1.0", - "npm-packlist": "^1.1.12", - "npm-pick-manifest": "^3.0.0", - "npm-registry-fetch": "^4.0.0", - "osenv": "^0.1.5", - "promise-inflight": "^1.0.1", - "promise-retry": "^1.1.1", - "protoduck": "^5.0.1", - "rimraf": "^2.6.2", - "safe-buffer": "^5.1.2", - "semver": "^5.6.0", - "ssri": "^6.0.1", - "tar": "^4.4.10", - "unique-filename": "^1.1.1", - "which": "^1.3.1" - }, - "dependencies": { - "cacache": { - "version": "12.0.4", - "resolved": "https://registry.npmjs.org/cacache/-/cacache-12.0.4.tgz", - "integrity": "sha512-a0tMB40oefvuInr4Cwb3GerbL9xTj1D5yg0T5xrjGCGyfvbxseIXX7BAO/u/hIXdafzOI5JC3wDwHyf24buOAQ==", - "dev": true, - "requires": { - "bluebird": "^3.5.5", - "chownr": "^1.1.1", - "figgy-pudding": "^3.5.1", - "glob": "^7.1.4", - "graceful-fs": "^4.1.15", - "infer-owner": "^1.0.3", - "lru-cache": "^5.1.1", - "mississippi": "^3.0.0", - "mkdirp": "^0.5.1", - "move-concurrently": "^1.0.1", - "promise-inflight": "^1.0.1", - "rimraf": "^2.6.3", - "ssri": "^6.0.1", - "unique-filename": "^1.1.1", - "y18n": "^4.0.0" - } - }, - "fs-minipass": { - "version": "1.2.7", - "resolved": "https://registry.npmjs.org/fs-minipass/-/fs-minipass-1.2.7.tgz", - "integrity": "sha512-GWSSJGFy4e9GUeCcbIkED+bgAoFyj7XF1mV8rma3QW4NIqX9Kyx79N/PF61H5udOV3aY1IaMLs6pGbH71nlCTA==", - "dev": true, - "requires": { - "minipass": "^2.6.0" - } - }, - "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==", - "dev": true - }, - "minipass": { - "version": "2.9.0", - "resolved": "https://registry.npmjs.org/minipass/-/minipass-2.9.0.tgz", - "integrity": "sha512-wxfUjg9WebH+CUDX/CdbRlh5SmfZiy/hpkxaRI16Y9W56Pa75sWgd/rvFilSgrauD9NyFymP/+JFV3KwzIsJeg==", - "dev": true, - "requires": { - "safe-buffer": "^5.1.2", - "yallist": "^3.0.0" - } - }, - "minizlib": { - "version": "1.3.3", - "resolved": "https://registry.npmjs.org/minizlib/-/minizlib-1.3.3.tgz", - "integrity": "sha512-6ZYMOEnmVsdCeTJVE0W9ZD+pVnE8h9Hma/iOwwRDsdQoePpoX56/8B6z3P9VNwppJuBKNRuFDRNRqRWexT9G9Q==", - "dev": true, - "requires": { - "minipass": "^2.9.0" - } - }, - "npm-package-arg": { - "version": "6.1.1", - "resolved": "https://registry.npmjs.org/npm-package-arg/-/npm-package-arg-6.1.1.tgz", - "integrity": "sha512-qBpssaL3IOZWi5vEKUKW0cO7kzLeT+EQO9W8RsLOZf76KF9E/K9+wH0C7t06HXPpaH8WH5xF1MExLuCwbTqRUg==", - "dev": true, - "requires": { - "hosted-git-info": "^2.7.1", - "osenv": "^0.1.5", - "semver": "^5.6.0", - "validate-npm-package-name": "^3.0.0" - } - }, - "npm-pick-manifest": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/npm-pick-manifest/-/npm-pick-manifest-3.0.2.tgz", - "integrity": "sha512-wNprTNg+X5nf+tDi+hbjdHhM4bX+mKqv6XmPh7B5eG+QY9VARfQPfCEH013H5GqfNj6ee8Ij2fg8yk0mzps1Vw==", - "dev": true, - "requires": { - "figgy-pudding": "^3.5.1", - "npm-package-arg": "^6.0.0", - "semver": "^5.4.1" - } - }, - "rimraf": { - "version": "2.7.1", - "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", - "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==", - "dev": true, - "requires": { - "glob": "^7.1.3" - } - }, - "semver": { - "version": "5.7.1", - "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", - "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", - "dev": true - }, - "ssri": { - "version": "6.0.2", - "resolved": "https://registry.npmjs.org/ssri/-/ssri-6.0.2.tgz", - "integrity": "sha512-cepbSq/neFK7xB6A50KHN0xHDotYzq58wWCa5LeWqnPrHG8GzfEjO/4O8kpmcGW+oaxkvhEJCWgbgNk4/ZV93Q==", - "dev": true, - "requires": { - "figgy-pudding": "^3.5.1" - } - }, - "tar": { - "version": "4.4.19", - "resolved": "https://registry.npmjs.org/tar/-/tar-4.4.19.tgz", - "integrity": "sha512-a20gEsvHnWe0ygBY8JbxoM4w3SJdhc7ZAuxkLqh+nvNQN2IOt0B5lLgM490X5Hl8FF0dl0tOf2ewFYAlIFgzVA==", - "dev": true, - "requires": { - "chownr": "^1.1.4", - "fs-minipass": "^1.2.7", - "minipass": "^2.9.0", - "minizlib": "^1.3.3", - "mkdirp": "^0.5.5", - "safe-buffer": "^5.2.1", - "yallist": "^3.1.1" - }, - "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==", - "dev": true - } - } - }, - "yallist": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz", - "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==", - "dev": true - } - } - }, - "pako": { - "version": "1.0.11", - "resolved": "https://registry.npmjs.org/pako/-/pako-1.0.11.tgz", - "integrity": "sha512-4hLB8Py4zZce5s4yd9XzopqwVv/yGNhV1Bl8NTmCq1763HeK2+EwVTv+leGeL13Dnh2wfbqowVPXCIO0z4taYw==", - "dev": true - }, - "panzoom": { - "version": "9.2.4", - "resolved": "https://registry.npmjs.org/panzoom/-/panzoom-9.2.4.tgz", - "integrity": "sha512-pQoIZQFL+0kYfcrXxcUL2khjGp61pieXYGavvZAAiHJg5ZGBa8Ci46JC7XeTZHg8EiKv1IcTacDf9PtG0oN13Q==", - "requires": { - "amator": "^1.1.0", - "ngraph.events": "^1.2.1", - "wheel": "^1.0.0" - } - }, - "parallel-transform": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/parallel-transform/-/parallel-transform-1.2.0.tgz", - "integrity": "sha512-P2vSmIu38uIlvdcU7fDkyrxj33gTUy/ABO5ZUbGowxNCopBq/OoD42bP4UmMrJoPyk4Uqf0mu3mtWBhHCZD8yg==", - "dev": true, - "requires": { - "cyclist": "^1.0.1", - "inherits": "^2.0.3", - "readable-stream": "^2.1.5" - } - }, - "parse-asn1": { - "version": "5.1.5", - "resolved": "https://registry.npmjs.org/parse-asn1/-/parse-asn1-5.1.5.tgz", - "integrity": "sha512-jkMYn1dcJqF6d5CpU689bq7w/b5ALS9ROVSpQDPrZsqqesUJii9qutvoT5ltGedNXMO2e16YUWIghG9KxaViTQ==", - "dev": true, - "requires": { - "asn1.js": "^4.0.0", - "browserify-aes": "^1.0.0", - "create-hash": "^1.1.0", - "evp_bytestokey": "^1.0.0", - "pbkdf2": "^3.0.3", - "safe-buffer": "^5.1.1" - } - }, - "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" - } - }, - "parse5": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/parse5/-/parse5-4.0.0.tgz", - "integrity": "sha512-VrZ7eOd3T1Fk4XWNXMgiGBK/z0MG48BWG2uQNU4I72fkQuKUTZpl+u9k+CxEG0twMVzSmXEEz12z5Fnw1jIQFA==", - "dev": true - }, - "parseurl": { - "version": "1.3.3", - "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz", - "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==", - "dev": true - }, - "pascalcase": { - "version": "0.1.1", - "resolved": "https://registry.npmjs.org/pascalcase/-/pascalcase-0.1.1.tgz", - "integrity": "sha1-s2PlXoAGym/iF4TS2yK9FdeRfxQ=", - "dev": true - }, - "path-browserify": { - "version": "0.0.1", - "resolved": "https://registry.npmjs.org/path-browserify/-/path-browserify-0.0.1.tgz", - "integrity": "sha512-BapA40NHICOS+USX9SN4tyhq+A2RrN/Ws5F0Z5aMHDp98Fl86lX8Oti8B7uN93L4Ifv4fHOEA+pQw87gmMO/lQ==", - "dev": true - }, - "path-dirname": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/path-dirname/-/path-dirname-1.0.2.tgz", - "integrity": "sha1-zDPSTVJeCZpTiMAzbG4yuRYGCeA=", - "dev": true - }, - "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 - }, - "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=", - "dev": true - }, - "path-is-inside": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/path-is-inside/-/path-is-inside-1.0.2.tgz", - "integrity": "sha1-NlQX3t5EQw0cEa9hAn+s8HS9/FM=", - "dev": true - }, - "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==", - "dev": true - }, - "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=", - "dev": true - }, - "path-type": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/path-type/-/path-type-3.0.0.tgz", - "integrity": "sha512-T2ZUsdZFHgA3u4e5PfPbjd7HDDpxPnQb5jN0SrDsjNSuVXHJqtwTnWqG0B1jZrgmJ/7lj1EmVIByWt1gxGkWvg==", - "dev": true, - "requires": { - "pify": "^3.0.0" - }, - "dependencies": { - "pify": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/pify/-/pify-3.0.0.tgz", - "integrity": "sha1-5aSs0sEB/fPZpNB/DbxNtJ3SgXY=", - "dev": true - } - } - }, - "pbkdf2": { - "version": "3.0.17", - "resolved": "https://registry.npmjs.org/pbkdf2/-/pbkdf2-3.0.17.tgz", - "integrity": "sha512-U/il5MsrZp7mGg3mSQfn742na2T+1/vHDCG5/iTI3X9MKUuYUZVLQhyRsg06mCgDBTd57TxzgZt7P+fYfjRLtA==", - "dev": true, - "requires": { - "create-hash": "^1.1.2", - "create-hmac": "^1.1.4", - "ripemd160": "^2.0.1", - "safe-buffer": "^5.0.1", - "sha.js": "^2.4.8" - } - }, - "performance-now": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/performance-now/-/performance-now-2.1.0.tgz", - "integrity": "sha1-Ywn04OX6kT7BxpMHrjZLSzd8nns=", - "dev": true - }, - "picomatch": { - "version": "2.2.2", - "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.2.2.tgz", - "integrity": "sha512-q0M/9eZHzmr0AulXyPwNfZjtwZ/RBZlbN3K3CErVrk50T2ASYI7Bye0EvekFY3IP1Nt2DHu0re+V2ZHIpMkuWg==", - "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 - }, - "pinkie": { - "version": "2.0.4", - "resolved": "https://registry.npmjs.org/pinkie/-/pinkie-2.0.4.tgz", - "integrity": "sha1-clVrgM+g1IqXToDnckjoDtT3+HA=", - "dev": true - }, - "pinkie-promise": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/pinkie-promise/-/pinkie-promise-2.0.1.tgz", - "integrity": "sha1-ITXW36ejWMBprJsXh3YogihFD/o=", - "dev": true, - "requires": { - "pinkie": "^2.0.0" - } - }, - "pkg-dir": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/pkg-dir/-/pkg-dir-3.0.0.tgz", - "integrity": "sha512-/E57AYkoeQ25qkxMj5PBOVgF8Kiu/h7cYS30Z5+R7WaiCCBfLq58ZI/dSeaEKb9WVJV5n/03QwrN3IeWIFllvw==", - "dev": true, - "requires": { - "find-up": "^3.0.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" - } - }, - "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 - } - } - }, - "pkg-up": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/pkg-up/-/pkg-up-2.0.0.tgz", - "integrity": "sha1-yBmscoBZpGHKscOImivjxJoATX8=", - "dev": true, - "requires": { - "find-up": "^2.1.0" - } - }, - "portfinder": { - "version": "1.0.26", - "resolved": "https://registry.npmjs.org/portfinder/-/portfinder-1.0.26.tgz", - "integrity": "sha512-Xi7mKxJHHMI3rIUrnm/jjUgwhbYMkp/XKEcZX3aG4BrumLpq3nmoQMX+ClYnDZnZ/New7IatC1no5RX0zo1vXQ==", - "dev": true, - "requires": { - "async": "^2.6.2", - "debug": "^3.1.1", - "mkdirp": "^0.5.1" - }, - "dependencies": { - "debug": { - "version": "3.2.6", - "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.6.tgz", - "integrity": "sha512-mel+jf7nrtEl5Pn1Qx46zARXKDpBbvzezse7p7LqINmdoIk8PYP5SySaxEmYv6TZ0JyEKA1hsCId6DIhgITtWQ==", - "dev": true, - "requires": { - "ms": "^2.1.1" - } - } - } - }, - "posix-character-classes": { - "version": "0.1.1", - "resolved": "https://registry.npmjs.org/posix-character-classes/-/posix-character-classes-0.1.1.tgz", - "integrity": "sha1-AerA/jta9xoqbAL+q7jB/vfgDqs=", - "dev": true - }, - "postcss": { - "version": "7.0.27", - "resolved": "https://registry.npmjs.org/postcss/-/postcss-7.0.27.tgz", - "integrity": "sha512-WuQETPMcW9Uf1/22HWUWP9lgsIC+KEHg2kozMflKjbeUtw9ujvFX6QmIfozaErDkmLWS9WEnEdEe6Uo9/BNTdQ==", - "dev": true, - "requires": { - "chalk": "^2.4.2", - "source-map": "^0.6.1", - "supports-color": "^6.1.0" - }, - "dependencies": { - "source-map": { - "version": "0.6.1", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", - "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", - "dev": true - }, - "supports-color": { - "version": "6.1.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-6.1.0.tgz", - "integrity": "sha512-qe1jfm1Mg7Nq/NSh6XE24gPXROEVsWHxC1LIx//XNlD9iw7YZQGjZNjYN7xGaEG6iKdA8EtNFW6R0gjnVXp+wQ==", - "dev": true, - "requires": { - "has-flag": "^3.0.0" - } - } - } - }, - "postcss-calc": { - "version": "7.0.2", - "resolved": "https://registry.npmjs.org/postcss-calc/-/postcss-calc-7.0.2.tgz", - "integrity": "sha512-rofZFHUg6ZIrvRwPeFktv06GdbDYLcGqh9EwiMutZg+a0oePCCw1zHOEiji6LCpyRcjTREtPASuUqeAvYlEVvQ==", - "dev": true, - "requires": { - "postcss": "^7.0.27", - "postcss-selector-parser": "^6.0.2", - "postcss-value-parser": "^4.0.2" - } - }, - "postcss-colormin": { - "version": "4.0.3", - "resolved": "https://registry.npmjs.org/postcss-colormin/-/postcss-colormin-4.0.3.tgz", - "integrity": "sha512-WyQFAdDZpExQh32j0U0feWisZ0dmOtPl44qYmJKkq9xFWY3p+4qnRzCHeNrkeRhwPHz9bQ3mo0/yVkaply0MNw==", - "dev": true, - "requires": { - "browserslist": "^4.0.0", - "color": "^3.0.0", - "has": "^1.0.0", - "postcss": "^7.0.0", - "postcss-value-parser": "^3.0.0" - }, - "dependencies": { - "postcss-value-parser": { - "version": "3.3.1", - "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", - "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", - "dev": true - } - } - }, - "postcss-convert-values": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/postcss-convert-values/-/postcss-convert-values-4.0.1.tgz", - "integrity": "sha512-Kisdo1y77KUC0Jmn0OXU/COOJbzM8cImvw1ZFsBgBgMgb1iL23Zs/LXRe3r+EZqM3vGYKdQ2YJVQ5VkJI+zEJQ==", - "dev": true, - "requires": { - "postcss": "^7.0.0", - "postcss-value-parser": "^3.0.0" - }, - "dependencies": { - "postcss-value-parser": { - "version": "3.3.1", - "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", - "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", - "dev": true - } - } - }, - "postcss-discard-comments": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/postcss-discard-comments/-/postcss-discard-comments-4.0.2.tgz", - "integrity": "sha512-RJutN259iuRf3IW7GZyLM5Sw4GLTOH8FmsXBnv8Ab/Tc2k4SR4qbV4DNbyyY4+Sjo362SyDmW2DQ7lBSChrpkg==", - "dev": true, - "requires": { - "postcss": "^7.0.0" - } - }, - "postcss-discard-duplicates": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/postcss-discard-duplicates/-/postcss-discard-duplicates-4.0.2.tgz", - "integrity": "sha512-ZNQfR1gPNAiXZhgENFfEglF93pciw0WxMkJeVmw8eF+JZBbMD7jp6C67GqJAXVZP2BWbOztKfbsdmMp/k8c6oQ==", - "dev": true, - "requires": { - "postcss": "^7.0.0" - } - }, - "postcss-discard-empty": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/postcss-discard-empty/-/postcss-discard-empty-4.0.1.tgz", - "integrity": "sha512-B9miTzbznhDjTfjvipfHoqbWKwd0Mj+/fL5s1QOz06wufguil+Xheo4XpOnc4NqKYBCNqqEzgPv2aPBIJLox0w==", - "dev": true, - "requires": { - "postcss": "^7.0.0" - } - }, - "postcss-discard-overridden": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/postcss-discard-overridden/-/postcss-discard-overridden-4.0.1.tgz", - "integrity": "sha512-IYY2bEDD7g1XM1IDEsUT4//iEYCxAmP5oDSFMVU/JVvT7gh+l4fmjciLqGgwjdWpQIdb0Che2VX00QObS5+cTg==", - "dev": true, - "requires": { - "postcss": "^7.0.0" - } - }, - "postcss-import": { - "version": "12.0.1", - "resolved": "https://registry.npmjs.org/postcss-import/-/postcss-import-12.0.1.tgz", - "integrity": "sha512-3Gti33dmCjyKBgimqGxL3vcV8w9+bsHwO5UrBawp796+jdardbcFl4RP5w/76BwNL7aGzpKstIfF9I+kdE8pTw==", - "dev": true, - "requires": { - "postcss": "^7.0.1", - "postcss-value-parser": "^3.2.3", - "read-cache": "^1.0.0", - "resolve": "^1.1.7" - }, - "dependencies": { - "postcss-value-parser": { - "version": "3.3.1", - "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", - "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", - "dev": true - } - } - }, - "postcss-load-config": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/postcss-load-config/-/postcss-load-config-2.1.0.tgz", - "integrity": "sha512-4pV3JJVPLd5+RueiVVB+gFOAa7GWc25XQcMp86Zexzke69mKf6Nx9LRcQywdz7yZI9n1udOxmLuAwTBypypF8Q==", - "dev": true, - "requires": { - "cosmiconfig": "^5.0.0", - "import-cwd": "^2.0.0" - } - }, - "postcss-loader": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/postcss-loader/-/postcss-loader-3.0.0.tgz", - "integrity": "sha512-cLWoDEY5OwHcAjDnkyRQzAXfs2jrKjXpO/HQFcc5b5u/r7aa471wdmChmwfnv7x2u840iat/wi0lQ5nbRgSkUA==", - "dev": true, - "requires": { - "loader-utils": "^1.1.0", - "postcss": "^7.0.0", - "postcss-load-config": "^2.0.0", - "schema-utils": "^1.0.0" - }, - "dependencies": { - "json5": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/json5/-/json5-1.0.1.tgz", - "integrity": "sha512-aKS4WQjPenRxiQsC93MNfjx+nbF4PAdYzmd/1JIj8HYzqfbu86beTuNgXDzPknWk0n0uARlyewZo4s++ES36Ow==", - "dev": true, - "requires": { - "minimist": "^1.2.0" - } - }, - "loader-utils": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/loader-utils/-/loader-utils-1.4.0.tgz", - "integrity": "sha512-qH0WSMBtn/oHuwjy/NucEgbx5dbxxnxup9s4PVXJUDHZBQY+s0NWA9rJf53RBnQZxfch7euUui7hpoAPvALZdA==", - "dev": true, - "requires": { - "big.js": "^5.2.2", - "emojis-list": "^3.0.0", - "json5": "^1.0.1" - } - }, - "schema-utils": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-1.0.0.tgz", - "integrity": "sha512-i27Mic4KovM/lnGsy8whRCHhc7VicJajAjTrYg11K9zfZXnYIt4k5F+kZkwjnrhKzLic/HLU4j11mjsz2G/75g==", - "dev": true, - "requires": { - "ajv": "^6.1.0", - "ajv-errors": "^1.0.0", - "ajv-keywords": "^3.1.0" - } - } - } - }, - "postcss-merge-longhand": { - "version": "4.0.11", - "resolved": "https://registry.npmjs.org/postcss-merge-longhand/-/postcss-merge-longhand-4.0.11.tgz", - "integrity": "sha512-alx/zmoeXvJjp7L4mxEMjh8lxVlDFX1gqWHzaaQewwMZiVhLo42TEClKaeHbRf6J7j82ZOdTJ808RtN0ZOZwvw==", - "dev": true, - "requires": { - "css-color-names": "0.0.4", - "postcss": "^7.0.0", - "postcss-value-parser": "^3.0.0", - "stylehacks": "^4.0.0" - }, - "dependencies": { - "postcss-value-parser": { - "version": "3.3.1", - "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", - "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", - "dev": true - } - } - }, - "postcss-merge-rules": { - "version": "4.0.3", - "resolved": "https://registry.npmjs.org/postcss-merge-rules/-/postcss-merge-rules-4.0.3.tgz", - "integrity": "sha512-U7e3r1SbvYzO0Jr3UT/zKBVgYYyhAz0aitvGIYOYK5CPmkNih+WDSsS5tvPrJ8YMQYlEMvsZIiqmn7HdFUaeEQ==", - "dev": true, - "requires": { - "browserslist": "^4.0.0", - "caniuse-api": "^3.0.0", - "cssnano-util-same-parent": "^4.0.0", - "postcss": "^7.0.0", - "postcss-selector-parser": "^3.0.0", - "vendors": "^1.0.0" - }, - "dependencies": { - "postcss-selector-parser": { - "version": "3.1.2", - "resolved": "https://registry.npmjs.org/postcss-selector-parser/-/postcss-selector-parser-3.1.2.tgz", - "integrity": "sha512-h7fJ/5uWuRVyOtkO45pnt1Ih40CEleeyCHzipqAZO2e5H20g25Y48uYnFUiShvY4rZWNJ/Bib/KVPmanaCtOhA==", - "dev": true, - "requires": { - "dot-prop": "^5.2.0", - "indexes-of": "^1.0.1", - "uniq": "^1.0.1" - } - } - } - }, - "postcss-minify-font-values": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/postcss-minify-font-values/-/postcss-minify-font-values-4.0.2.tgz", - "integrity": "sha512-j85oO6OnRU9zPf04+PZv1LYIYOprWm6IA6zkXkrJXyRveDEuQggG6tvoy8ir8ZwjLxLuGfNkCZEQG7zan+Hbtg==", - "dev": true, - "requires": { - "postcss": "^7.0.0", - "postcss-value-parser": "^3.0.0" - }, - "dependencies": { - "postcss-value-parser": { - "version": "3.3.1", - "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", - "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", - "dev": true - } - } - }, - "postcss-minify-gradients": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/postcss-minify-gradients/-/postcss-minify-gradients-4.0.2.tgz", - "integrity": "sha512-qKPfwlONdcf/AndP1U8SJ/uzIJtowHlMaSioKzebAXSG4iJthlWC9iSWznQcX4f66gIWX44RSA841HTHj3wK+Q==", - "dev": true, - "requires": { - "cssnano-util-get-arguments": "^4.0.0", - "is-color-stop": "^1.0.0", - "postcss": "^7.0.0", - "postcss-value-parser": "^3.0.0" - }, - "dependencies": { - "postcss-value-parser": { - "version": "3.3.1", - "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", - "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", - "dev": true - } - } - }, - "postcss-minify-params": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/postcss-minify-params/-/postcss-minify-params-4.0.2.tgz", - "integrity": "sha512-G7eWyzEx0xL4/wiBBJxJOz48zAKV2WG3iZOqVhPet/9geefm/Px5uo1fzlHu+DOjT+m0Mmiz3jkQzVHe6wxAWg==", - "dev": true, - "requires": { - "alphanum-sort": "^1.0.0", - "browserslist": "^4.0.0", - "cssnano-util-get-arguments": "^4.0.0", - "postcss": "^7.0.0", - "postcss-value-parser": "^3.0.0", - "uniqs": "^2.0.0" - }, - "dependencies": { - "postcss-value-parser": { - "version": "3.3.1", - "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", - "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", - "dev": true - } - } - }, - "postcss-minify-selectors": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/postcss-minify-selectors/-/postcss-minify-selectors-4.0.2.tgz", - "integrity": "sha512-D5S1iViljXBj9kflQo4YutWnJmwm8VvIsU1GeXJGiG9j8CIg9zs4voPMdQDUmIxetUOh60VilsNzCiAFTOqu3g==", - "dev": true, - "requires": { - "alphanum-sort": "^1.0.0", - "has": "^1.0.0", - "postcss": "^7.0.0", - "postcss-selector-parser": "^3.0.0" - }, - "dependencies": { - "postcss-selector-parser": { - "version": "3.1.2", - "resolved": "https://registry.npmjs.org/postcss-selector-parser/-/postcss-selector-parser-3.1.2.tgz", - "integrity": "sha512-h7fJ/5uWuRVyOtkO45pnt1Ih40CEleeyCHzipqAZO2e5H20g25Y48uYnFUiShvY4rZWNJ/Bib/KVPmanaCtOhA==", - "dev": true, - "requires": { - "dot-prop": "^5.2.0", - "indexes-of": "^1.0.1", - "uniq": "^1.0.1" - } - } - } - }, - "postcss-modules-extract-imports": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/postcss-modules-extract-imports/-/postcss-modules-extract-imports-2.0.0.tgz", - "integrity": "sha512-LaYLDNS4SG8Q5WAWqIJgdHPJrDDr/Lv775rMBFUbgjTz6j34lUznACHcdRWroPvXANP2Vj7yNK57vp9eFqzLWQ==", - "dev": true, - "requires": { - "postcss": "^7.0.5" - } - }, - "postcss-modules-local-by-default": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/postcss-modules-local-by-default/-/postcss-modules-local-by-default-3.0.2.tgz", - "integrity": "sha512-jM/V8eqM4oJ/22j0gx4jrp63GSvDH6v86OqyTHHUvk4/k1vceipZsaymiZ5PvocqZOl5SFHiFJqjs3la0wnfIQ==", - "dev": true, - "requires": { - "icss-utils": "^4.1.1", - "postcss": "^7.0.16", - "postcss-selector-parser": "^6.0.2", - "postcss-value-parser": "^4.0.0" - } - }, - "postcss-modules-scope": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/postcss-modules-scope/-/postcss-modules-scope-2.2.0.tgz", - "integrity": "sha512-YyEgsTMRpNd+HmyC7H/mh3y+MeFWevy7V1evVhJWewmMbjDHIbZbOXICC2y+m1xI1UVfIT1HMW/O04Hxyu9oXQ==", - "dev": true, - "requires": { - "postcss": "^7.0.6", - "postcss-selector-parser": "^6.0.0" - } - }, - "postcss-modules-values": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/postcss-modules-values/-/postcss-modules-values-3.0.0.tgz", - "integrity": "sha512-1//E5jCBrZ9DmRX+zCtmQtRSV6PV42Ix7Bzj9GbwJceduuf7IqP8MgeTXuRDHOWj2m0VzZD5+roFWDuU8RQjcg==", - "dev": true, - "requires": { - "icss-utils": "^4.0.0", - "postcss": "^7.0.6" - } - }, - "postcss-normalize-charset": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/postcss-normalize-charset/-/postcss-normalize-charset-4.0.1.tgz", - "integrity": "sha512-gMXCrrlWh6G27U0hF3vNvR3w8I1s2wOBILvA87iNXaPvSNo5uZAMYsZG7XjCUf1eVxuPfyL4TJ7++SGZLc9A3g==", - "dev": true, - "requires": { - "postcss": "^7.0.0" - } - }, - "postcss-normalize-display-values": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/postcss-normalize-display-values/-/postcss-normalize-display-values-4.0.2.tgz", - "integrity": "sha512-3F2jcsaMW7+VtRMAqf/3m4cPFhPD3EFRgNs18u+k3lTJJlVe7d0YPO+bnwqo2xg8YiRpDXJI2u8A0wqJxMsQuQ==", - "dev": true, - "requires": { - "cssnano-util-get-match": "^4.0.0", - "postcss": "^7.0.0", - "postcss-value-parser": "^3.0.0" - }, - "dependencies": { - "postcss-value-parser": { - "version": "3.3.1", - "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", - "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", - "dev": true - } - } - }, - "postcss-normalize-positions": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/postcss-normalize-positions/-/postcss-normalize-positions-4.0.2.tgz", - "integrity": "sha512-Dlf3/9AxpxE+NF1fJxYDeggi5WwV35MXGFnnoccP/9qDtFrTArZ0D0R+iKcg5WsUd8nUYMIl8yXDCtcrT8JrdA==", - "dev": true, - "requires": { - "cssnano-util-get-arguments": "^4.0.0", - "has": "^1.0.0", - "postcss": "^7.0.0", - "postcss-value-parser": "^3.0.0" - }, - "dependencies": { - "postcss-value-parser": { - "version": "3.3.1", - "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", - "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", - "dev": true - } - } - }, - "postcss-normalize-repeat-style": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/postcss-normalize-repeat-style/-/postcss-normalize-repeat-style-4.0.2.tgz", - "integrity": "sha512-qvigdYYMpSuoFs3Is/f5nHdRLJN/ITA7huIoCyqqENJe9PvPmLhNLMu7QTjPdtnVf6OcYYO5SHonx4+fbJE1+Q==", - "dev": true, - "requires": { - "cssnano-util-get-arguments": "^4.0.0", - "cssnano-util-get-match": "^4.0.0", - "postcss": "^7.0.0", - "postcss-value-parser": "^3.0.0" - }, - "dependencies": { - "postcss-value-parser": { - "version": "3.3.1", - "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", - "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", - "dev": true - } - } - }, - "postcss-normalize-string": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/postcss-normalize-string/-/postcss-normalize-string-4.0.2.tgz", - "integrity": "sha512-RrERod97Dnwqq49WNz8qo66ps0swYZDSb6rM57kN2J+aoyEAJfZ6bMx0sx/F9TIEX0xthPGCmeyiam/jXif0eA==", - "dev": true, - "requires": { - "has": "^1.0.0", - "postcss": "^7.0.0", - "postcss-value-parser": "^3.0.0" - }, - "dependencies": { - "postcss-value-parser": { - "version": "3.3.1", - "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", - "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", - "dev": true - } - } - }, - "postcss-normalize-timing-functions": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/postcss-normalize-timing-functions/-/postcss-normalize-timing-functions-4.0.2.tgz", - "integrity": "sha512-acwJY95edP762e++00Ehq9L4sZCEcOPyaHwoaFOhIwWCDfik6YvqsYNxckee65JHLKzuNSSmAdxwD2Cud1Z54A==", - "dev": true, - "requires": { - "cssnano-util-get-match": "^4.0.0", - "postcss": "^7.0.0", - "postcss-value-parser": "^3.0.0" - }, - "dependencies": { - "postcss-value-parser": { - "version": "3.3.1", - "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", - "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", - "dev": true - } - } - }, - "postcss-normalize-unicode": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/postcss-normalize-unicode/-/postcss-normalize-unicode-4.0.1.tgz", - "integrity": "sha512-od18Uq2wCYn+vZ/qCOeutvHjB5jm57ToxRaMeNuf0nWVHaP9Hua56QyMF6fs/4FSUnVIw0CBPsU0K4LnBPwYwg==", - "dev": true, - "requires": { - "browserslist": "^4.0.0", - "postcss": "^7.0.0", - "postcss-value-parser": "^3.0.0" - }, - "dependencies": { - "postcss-value-parser": { - "version": "3.3.1", - "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", - "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", - "dev": true - } - } - }, - "postcss-normalize-url": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/postcss-normalize-url/-/postcss-normalize-url-4.0.1.tgz", - "integrity": "sha512-p5oVaF4+IHwu7VpMan/SSpmpYxcJMtkGppYf0VbdH5B6hN8YNmVyJLuY9FmLQTzY3fag5ESUUHDqM+heid0UVA==", - "dev": true, - "requires": { - "is-absolute-url": "^2.0.0", - "normalize-url": "^3.0.0", - "postcss": "^7.0.0", - "postcss-value-parser": "^3.0.0" - }, - "dependencies": { - "postcss-value-parser": { - "version": "3.3.1", - "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", - "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", - "dev": true - } - } - }, - "postcss-normalize-whitespace": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/postcss-normalize-whitespace/-/postcss-normalize-whitespace-4.0.2.tgz", - "integrity": "sha512-tO8QIgrsI3p95r8fyqKV+ufKlSHh9hMJqACqbv2XknufqEDhDvbguXGBBqxw9nsQoXWf0qOqppziKJKHMD4GtA==", - "dev": true, - "requires": { - "postcss": "^7.0.0", - "postcss-value-parser": "^3.0.0" - }, - "dependencies": { - "postcss-value-parser": { - "version": "3.3.1", - "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", - "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", - "dev": true - } - } - }, - "postcss-ordered-values": { - "version": "4.1.2", - "resolved": "https://registry.npmjs.org/postcss-ordered-values/-/postcss-ordered-values-4.1.2.tgz", - "integrity": "sha512-2fCObh5UanxvSxeXrtLtlwVThBvHn6MQcu4ksNT2tsaV2Fg76R2CV98W7wNSlX+5/pFwEyaDwKLLoEV7uRybAw==", - "dev": true, - "requires": { - "cssnano-util-get-arguments": "^4.0.0", - "postcss": "^7.0.0", - "postcss-value-parser": "^3.0.0" - }, - "dependencies": { - "postcss-value-parser": { - "version": "3.3.1", - "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", - "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", - "dev": true - } - } - }, - "postcss-reduce-initial": { - "version": "4.0.3", - "resolved": "https://registry.npmjs.org/postcss-reduce-initial/-/postcss-reduce-initial-4.0.3.tgz", - "integrity": "sha512-gKWmR5aUulSjbzOfD9AlJiHCGH6AEVLaM0AV+aSioxUDd16qXP1PCh8d1/BGVvpdWn8k/HiK7n6TjeoXN1F7DA==", - "dev": true, - "requires": { - "browserslist": "^4.0.0", - "caniuse-api": "^3.0.0", - "has": "^1.0.0", - "postcss": "^7.0.0" - } - }, - "postcss-reduce-transforms": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/postcss-reduce-transforms/-/postcss-reduce-transforms-4.0.2.tgz", - "integrity": "sha512-EEVig1Q2QJ4ELpJXMZR8Vt5DQx8/mo+dGWSR7vWXqcob2gQLyQGsionYcGKATXvQzMPn6DSN1vTN7yFximdIAg==", - "dev": true, - "requires": { - "cssnano-util-get-match": "^4.0.0", - "has": "^1.0.0", - "postcss": "^7.0.0", - "postcss-value-parser": "^3.0.0" - }, - "dependencies": { - "postcss-value-parser": { - "version": "3.3.1", - "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", - "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", - "dev": true - } - } - }, - "postcss-selector-parser": { - "version": "6.0.2", - "resolved": "https://registry.npmjs.org/postcss-selector-parser/-/postcss-selector-parser-6.0.2.tgz", - "integrity": "sha512-36P2QR59jDTOAiIkqEprfJDsoNrvwFei3eCqKd1Y0tUsBimsq39BLp7RD+JWny3WgB1zGhJX8XVePwm9k4wdBg==", - "dev": true, - "requires": { - "cssesc": "^3.0.0", - "indexes-of": "^1.0.1", - "uniq": "^1.0.1" - } - }, - "postcss-svgo": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/postcss-svgo/-/postcss-svgo-4.0.2.tgz", - "integrity": "sha512-C6wyjo3VwFm0QgBy+Fu7gCYOkCmgmClghO+pjcxvrcBKtiKt0uCF+hvbMO1fyv5BMImRK90SMb+dwUnfbGd+jw==", - "dev": true, - "requires": { - "is-svg": "^3.0.0", - "postcss": "^7.0.0", - "postcss-value-parser": "^3.0.0", - "svgo": "^1.0.0" - }, - "dependencies": { - "postcss-value-parser": { - "version": "3.3.1", - "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", - "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", - "dev": true - } - } - }, - "postcss-unique-selectors": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/postcss-unique-selectors/-/postcss-unique-selectors-4.0.1.tgz", - "integrity": "sha512-+JanVaryLo9QwZjKrmJgkI4Fn8SBgRO6WXQBJi7KiAVPlmxikB5Jzc4EvXMT2H0/m0RjrVVm9rGNhZddm/8Spg==", - "dev": true, - "requires": { - "alphanum-sort": "^1.0.0", - "postcss": "^7.0.0", - "uniqs": "^2.0.0" - } - }, - "postcss-value-parser": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-4.1.0.tgz", - "integrity": "sha512-97DXOFbQJhk71ne5/Mt6cOu6yxsSfM0QGQyl0L25Gca4yGWEGJaig7l7gbCX623VqTBNGLRLaVUCnNkcedlRSQ==", - "dev": true - }, - "prepend-http": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/prepend-http/-/prepend-http-1.0.4.tgz", - "integrity": "sha1-1PRWKwzjaW5BrFLQ4ALlemNdxtw=", - "dev": true - }, - "private": { - "version": "0.1.8", - "resolved": "https://registry.npmjs.org/private/-/private-0.1.8.tgz", - "integrity": "sha512-VvivMrbvd2nKkiG38qjULzlc+4Vx4wm/whI9pQD35YrARNnhxeiRktSOhSukRLFNlzg6Br/cJPet5J/u19r/mg==", - "dev": true - }, - "process": { - "version": "0.11.10", - "resolved": "https://registry.npmjs.org/process/-/process-0.11.10.tgz", - "integrity": "sha1-czIwDoQBYb2j5podHZGn1LwW8YI=", - "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==", - "dev": true - }, - "promise": { - "version": "7.3.1", - "resolved": "https://registry.npmjs.org/promise/-/promise-7.3.1.tgz", - "integrity": "sha512-nolQXZ/4L+bP/UGlkfaIujX9BKxGwmQ9OT4mOt5yvy8iK1h3wqTEJCijzGANTCCl9nWjY41juyAn2K3Q1hLLTg==", - "dev": true, - "optional": true, - "requires": { - "asap": "~2.0.3" - } - }, - "promise-inflight": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/promise-inflight/-/promise-inflight-1.0.1.tgz", - "integrity": "sha1-mEcocL8igTL8vdhoEputEsPAKeM=", - "dev": true - }, - "promise-retry": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/promise-retry/-/promise-retry-1.1.1.tgz", - "integrity": "sha1-ZznpaOMFHaIM5kl/srUPaRHfPW0=", - "dev": true, - "requires": { - "err-code": "^1.0.0", - "retry": "^0.10.0" - }, - "dependencies": { - "retry": { - "version": "0.10.1", - "resolved": "https://registry.npmjs.org/retry/-/retry-0.10.1.tgz", - "integrity": "sha1-52OI0heZLCUnUCQdPTlW/tmNj/Q=", - "dev": true - } - } - }, - "protoduck": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/protoduck/-/protoduck-5.0.1.tgz", - "integrity": "sha512-WxoCeDCoCBY55BMvj4cAEjdVUFGRWed9ZxPlqTKYyw1nDDTQ4pqmnIMAGfJlg7Dx35uB/M+PHJPTmGOvaCaPTg==", - "dev": true, - "requires": { - "genfun": "^5.0.0" - } - }, - "protractor": { - "version": "5.4.4", - "resolved": "https://registry.npmjs.org/protractor/-/protractor-5.4.4.tgz", - "integrity": "sha512-BaL4vePgu3Vfa/whvTUAlgaCAId4uNSGxIFSCXMgj7LMYENPWLp85h5RBi9pdpX/bWQ8SF6flP7afmi2TC4eHw==", - "dev": true, - "requires": { - "@types/q": "^0.0.32", - "@types/selenium-webdriver": "^3.0.0", - "blocking-proxy": "^1.0.0", - "browserstack": "^1.5.1", - "chalk": "^1.1.3", - "glob": "^7.0.3", - "jasmine": "2.8.0", - "jasminewd2": "^2.1.0", - "q": "1.4.1", - "saucelabs": "^1.5.0", - "selenium-webdriver": "3.6.0", - "source-map-support": "~0.4.0", - "webdriver-js-extender": "2.1.0", - "webdriver-manager": "^12.0.6", - "yargs": "^12.0.5" - }, - "dependencies": { - "@types/q": { - "version": "0.0.32", - "resolved": "https://registry.npmjs.org/@types/q/-/q-0.0.32.tgz", - "integrity": "sha1-vShOV8hPEyXacCur/IKlMoGQwMU=", - "dev": true - }, - "ansi-styles": { - "version": "2.2.1", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-2.2.1.tgz", - "integrity": "sha1-tDLdM1i2NM914eRmQ2gkBTPB3b4=", - "dev": true - }, - "chalk": { - "version": "1.1.3", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-1.1.3.tgz", - "integrity": "sha1-qBFcVeSnAv5NFQq9OHKCKn4J/Jg=", - "dev": true, - "requires": { - "ansi-styles": "^2.2.1", - "escape-string-regexp": "^1.0.2", - "has-ansi": "^2.0.0", - "strip-ansi": "^3.0.0", - "supports-color": "^2.0.0" - } - }, - "del": { - "version": "2.2.2", - "resolved": "https://registry.npmjs.org/del/-/del-2.2.2.tgz", - "integrity": "sha1-wSyYHQZ4RshLyvhiz/kw2Qf/0ag=", - "dev": true, - "requires": { - "globby": "^5.0.0", - "is-path-cwd": "^1.0.0", - "is-path-in-cwd": "^1.0.0", - "object-assign": "^4.0.1", - "pify": "^2.0.0", - "pinkie-promise": "^2.0.0", - "rimraf": "^2.2.8" - } - }, - "globby": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/globby/-/globby-5.0.0.tgz", - "integrity": "sha1-69hGZ8oNuzMLmbz8aOrCvFQ3Dg0=", - "dev": true, - "requires": { - "array-union": "^1.0.1", - "arrify": "^1.0.0", - "glob": "^7.0.3", - "object-assign": "^4.0.1", - "pify": "^2.0.0", - "pinkie-promise": "^2.0.0" - } - }, - "is-path-cwd": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/is-path-cwd/-/is-path-cwd-1.0.0.tgz", - "integrity": "sha1-0iXsIxMuie3Tj9p2dHLmLmXxEG0=", - "dev": true - }, - "is-path-in-cwd": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/is-path-in-cwd/-/is-path-in-cwd-1.0.1.tgz", - "integrity": "sha512-FjV1RTW48E7CWM7eE/J2NJvAEEVektecDBVBE5Hh3nM1Jd0kvhHtX68Pr3xsDf857xt3Y4AkwVULK1Vku62aaQ==", - "dev": true, - "requires": { - "is-path-inside": "^1.0.0" - } - }, - "is-path-inside": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-1.0.1.tgz", - "integrity": "sha1-jvW33lBDej/cprToZe96pVy0gDY=", - "dev": true, - "requires": { - "path-is-inside": "^1.0.1" - } - }, - "pify": { - "version": "2.3.0", - "resolved": "https://registry.npmjs.org/pify/-/pify-2.3.0.tgz", - "integrity": "sha1-7RQaasBDqEnqWISY59yosVMw6Qw=", - "dev": true - }, - "q": { - "version": "1.4.1", - "resolved": "https://registry.npmjs.org/q/-/q-1.4.1.tgz", - "integrity": "sha1-VXBbzZPF82c1MMLCy8DCs63cKG4=", - "dev": true - }, - "rimraf": { - "version": "2.7.1", - "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", - "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==", - "dev": true, - "requires": { - "glob": "^7.1.3" - } - }, - "semver": { - "version": "5.7.1", - "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", - "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", - "dev": true - }, - "source-map": { - "version": "0.5.7", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.5.7.tgz", - "integrity": "sha1-igOdLRAh0i0eoUyA2OpGi6LvP8w=", - "dev": true - }, - "source-map-support": { - "version": "0.4.18", - "resolved": "https://registry.npmjs.org/source-map-support/-/source-map-support-0.4.18.tgz", - "integrity": "sha512-try0/JqxPLF9nOjvSta7tVondkP5dwgyLDjVoyMDlmjugT2lRZ1OfsrYTkCd2hkDnJTKRbO/Rl3orm8vlsUzbA==", - "dev": true, - "requires": { - "source-map": "^0.5.6" - } - }, - "supports-color": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-2.0.0.tgz", - "integrity": "sha1-U10EXOa2Nj+kARcIRimZXp3zJMc=", - "dev": true - }, - "webdriver-manager": { - "version": "12.1.7", - "resolved": "https://registry.npmjs.org/webdriver-manager/-/webdriver-manager-12.1.7.tgz", - "integrity": "sha512-XINj6b8CYuUYC93SG3xPkxlyUc3IJbD6Vvo75CVGuG9uzsefDzWQrhz0Lq8vbPxtb4d63CZdYophF8k8Or/YiA==", - "dev": true, - "requires": { - "adm-zip": "^0.4.9", - "chalk": "^1.1.1", - "del": "^2.2.0", - "glob": "^7.0.3", - "ini": "^1.3.4", - "minimist": "^1.2.0", - "q": "^1.4.1", - "request": "^2.87.0", - "rimraf": "^2.5.2", - "semver": "^5.3.0", - "xml2js": "^0.4.17" - } - } - } - }, - "proxy-addr": { - "version": "2.0.6", - "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.6.tgz", - "integrity": "sha512-dh/frvCBVmSsDYzw6n926jv974gddhkFPfiN8hPOi30Wax25QZyZEGveluCgliBnqmuM+UJmBErbAUFIoDbjOw==", - "dev": true, - "requires": { - "forwarded": "~0.1.2", - "ipaddr.js": "1.9.1" - } - }, - "prr": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/prr/-/prr-1.0.1.tgz", - "integrity": "sha1-0/wRS6BplaRexok/SEzrHXj19HY=", - "dev": true - }, - "psl": { - "version": "1.8.0", - "resolved": "https://registry.npmjs.org/psl/-/psl-1.8.0.tgz", - "integrity": "sha512-RIdOzyoavK+hA18OGGWDqUTsCLhtA7IcZ/6NCs4fFJaHBDab+pDDmDIByWFRQJq2Cd7r1OoQxBGKOaztq+hjIQ==", - "dev": true - }, - "public-encrypt": { - "version": "4.0.3", - "resolved": "https://registry.npmjs.org/public-encrypt/-/public-encrypt-4.0.3.tgz", - "integrity": "sha512-zVpa8oKZSz5bTMTFClc1fQOnyyEzpl5ozpi1B5YcvBrdohMjH2rfsBtyXcuNuwjsDIXmBYlF2N5FlJYhR29t8Q==", - "dev": true, - "requires": { - "bn.js": "^4.1.0", - "browserify-rsa": "^4.0.0", - "create-hash": "^1.1.0", - "parse-asn1": "^5.0.0", - "randombytes": "^2.0.1", - "safe-buffer": "^5.1.2" - }, - "dependencies": { - "bn.js": { - "version": "4.11.8", - "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.11.8.tgz", - "integrity": "sha512-ItfYfPLkWHUjckQCk8xC+LwxgK8NYcXywGigJgSwOP8Y2iyWT4f2vsZnoOXTTbo+o5yXmIUJ4gn5538SO5S3gA==", - "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" - } - }, - "pumpify": { - "version": "1.5.1", - "resolved": "https://registry.npmjs.org/pumpify/-/pumpify-1.5.1.tgz", - "integrity": "sha512-oClZI37HvuUJJxSKKrC17bZ9Cu0ZYhEAGPsPUy9KlMUmv9dKX2o77RUmq7f3XjIxbwyGwYzbzQ1L2Ks8sIradQ==", - "dev": true, - "requires": { - "duplexify": "^3.6.0", - "inherits": "^2.0.3", - "pump": "^2.0.0" - }, - "dependencies": { - "pump": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/pump/-/pump-2.0.1.tgz", - "integrity": "sha512-ruPMNRkN3MHP1cWJc9OWr+T/xDP0jhXYCLfJcBuX54hhfIBnaQmAUMfDcG4DM5UMWByBbJY69QSphm3jtDKIkA==", - "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==", - "dev": true - }, - "q": { - "version": "1.5.1", - "resolved": "https://registry.npmjs.org/q/-/q-1.5.1.tgz", - "integrity": "sha1-fjL3W0E4EpHQRhHxvxQQmsAGUdc=", - "dev": true - }, - "qjobs": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/qjobs/-/qjobs-1.2.0.tgz", - "integrity": "sha512-8YOJEHtxpySA3fFDyCRxA+UUV+fA+rTWnuWvylOK/NCjhY+b4ocCtmu8TtsWb+mYeU+GCHf/S66KZF/AsteKHg==", - "dev": true - }, - "qs": { - "version": "6.5.2", - "resolved": "https://registry.npmjs.org/qs/-/qs-6.5.2.tgz", - "integrity": "sha512-N5ZAX4/LxJmF+7wN74pUD6qAh9/wnvdQcjq9TZjevvXzSUo7bfmw91saqMjzGS2xq91/odN2dW/WOl7qQHNDGA==", - "dev": true - }, - "query-string": { - "version": "4.3.4", - "resolved": "https://registry.npmjs.org/query-string/-/query-string-4.3.4.tgz", - "integrity": "sha1-u7aTucqRXCMlFbIosaArYJBD2+s=", - "dev": true, - "requires": { - "object-assign": "^4.1.0", - "strict-uri-encode": "^1.0.0" - } - }, - "querystring": { - "version": "0.2.0", - "resolved": "https://registry.npmjs.org/querystring/-/querystring-0.2.0.tgz", - "integrity": "sha1-sgmEkgO7Jd+CDadW50cAWHhSFiA=", - "dev": true - }, - "querystring-es3": { - "version": "0.2.1", - "resolved": "https://registry.npmjs.org/querystring-es3/-/querystring-es3-0.2.1.tgz", - "integrity": "sha1-nsYfeQSYdXB9aUFFlv2Qek1xHnM=", - "dev": true - }, - "querystringify": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/querystringify/-/querystringify-2.1.1.tgz", - "integrity": "sha512-w7fLxIRCRT7U8Qu53jQnJyPkYZIaR4n5151KMfcJlO/A9397Wxb1amJvROTK6TOnp7PfoAmg/qXiNHI+08jRfA==", - "dev": true - }, - "randombytes": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/randombytes/-/randombytes-2.1.0.tgz", - "integrity": "sha512-vYl3iOX+4CKUWuxGi9Ukhie6fsqXqS9FE2Zaic4tNFD2N2QQaXOMFbuKK4QmDHC0JO6B1Zp41J0LpT0oR68amQ==", - "dev": true, - "requires": { - "safe-buffer": "^5.1.0" - } - }, - "randomfill": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/randomfill/-/randomfill-1.0.4.tgz", - "integrity": "sha512-87lcbR8+MhcWcUiQ+9e+Rwx8MyR2P7qnt15ynUlbm3TU/fjbgz4GsvfSUDTemtCCtVCqb4ZcEFlyPNTh9bBTLw==", - "dev": true, - "requires": { - "randombytes": "^2.0.5", - "safe-buffer": "^5.1.0" - } - }, - "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==", - "dev": true - }, - "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==", - "dev": true, - "requires": { - "bytes": "3.1.0", - "http-errors": "1.7.2", - "iconv-lite": "0.4.24", - "unpipe": "1.0.0" - }, - "dependencies": { - "bytes": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.0.tgz", - "integrity": "sha512-zauLjrfCG+xvoyaqLoV8bLVXXNGC4JqlxFCutSDWA6fJrTo2ZuvLYTqZ7aHBLZSMOopbzwv8f+wZcVzfVTI2Dg==", - "dev": true - } - } - }, - "raw-loader": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/raw-loader/-/raw-loader-4.0.0.tgz", - "integrity": "sha512-iINUOYvl1cGEmfoaLjnZXt4bKfT2LJnZZib5N/LLyAphC+Dd11vNP9CNVb38j+SAJpFI1uo8j9frmih53ASy7Q==", - "dev": true, - "requires": { - "loader-utils": "^1.2.3", - "schema-utils": "^2.5.0" - }, - "dependencies": { - "json5": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/json5/-/json5-1.0.1.tgz", - "integrity": "sha512-aKS4WQjPenRxiQsC93MNfjx+nbF4PAdYzmd/1JIj8HYzqfbu86beTuNgXDzPknWk0n0uARlyewZo4s++ES36Ow==", - "dev": true, - "requires": { - "minimist": "^1.2.0" - } - }, - "loader-utils": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/loader-utils/-/loader-utils-1.4.0.tgz", - "integrity": "sha512-qH0WSMBtn/oHuwjy/NucEgbx5dbxxnxup9s4PVXJUDHZBQY+s0NWA9rJf53RBnQZxfch7euUui7hpoAPvALZdA==", - "dev": true, - "requires": { - "big.js": "^5.2.2", - "emojis-list": "^3.0.0", - "json5": "^1.0.1" - } - } - } - }, - "read-cache": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/read-cache/-/read-cache-1.0.0.tgz", - "integrity": "sha1-5mTvMRYRZsl1HNvo28+GtftY93Q=", - "dev": true, - "requires": { - "pify": "^2.3.0" - }, - "dependencies": { - "pify": { - "version": "2.3.0", - "resolved": "https://registry.npmjs.org/pify/-/pify-2.3.0.tgz", - "integrity": "sha1-7RQaasBDqEnqWISY59yosVMw6Qw=", - "dev": true - } - } - }, - "read-package-json": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/read-package-json/-/read-package-json-2.1.1.tgz", - "integrity": "sha512-dAiqGtVc/q5doFz6096CcnXhpYk0ZN8dEKVkGLU0CsASt8SrgF6SF7OTKAYubfvFhWaqofl+Y8HK19GR8jwW+A==", - "dev": true, - "requires": { - "glob": "^7.1.1", - "graceful-fs": "^4.1.2", - "json-parse-better-errors": "^1.0.1", - "normalize-package-data": "^2.0.0", - "npm-normalize-package-bin": "^1.0.0" - } - }, - "read-package-tree": { - "version": "5.3.1", - "resolved": "https://registry.npmjs.org/read-package-tree/-/read-package-tree-5.3.1.tgz", - "integrity": "sha512-mLUDsD5JVtlZxjSlPPx1RETkNjjvQYuweKwNVt1Sn8kP5Jh44pvYuUHCp6xSVDZWbNxVxG5lyZJ921aJH61sTw==", - "dev": true, - "requires": { - "read-package-json": "^2.0.0", - "readdir-scoped-modules": "^1.0.0", - "util-promisify": "^2.1.0" - } - }, - "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==", - "dev": true, - "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" - } - }, - "readdir-scoped-modules": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/readdir-scoped-modules/-/readdir-scoped-modules-1.1.0.tgz", - "integrity": "sha512-asaikDeqAQg7JifRsZn1NJZXo9E+VwlyCfbkZhwyISinqk5zNS6266HS5kah6P0SaQKGF6SkNnZVHUzHFYxYDw==", - "dev": true, - "requires": { - "debuglog": "^1.0.1", - "dezalgo": "^1.0.0", - "graceful-fs": "^4.1.2", - "once": "^1.3.0" - } - }, - "readdirp": { - "version": "3.4.0", - "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.4.0.tgz", - "integrity": "sha512-0xe001vZBnJEK+uKcj8qOhyAKPzIT+gStxWr3LCB0DwcXR5NZJ3IaC+yGnHCYzB/S7ov3m3EEbZI2zeNvX+hGQ==", - "dev": true, - "requires": { - "picomatch": "^2.2.1" - } - }, - "reflect-metadata": { - "version": "0.1.13", - "resolved": "https://registry.npmjs.org/reflect-metadata/-/reflect-metadata-0.1.13.tgz", - "integrity": "sha512-Ts1Y/anZELhSsjMcU605fU9RE4Oi3p5ORujwbIKXfWa+0Zxs510Qrmrce5/Jowq3cHSZSJqBjypxmHarc+vEWg==", - "dev": true - }, - "regenerate": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/regenerate/-/regenerate-1.4.0.tgz", - "integrity": "sha512-1G6jJVDWrt0rK99kBjvEtziZNCICAuvIPkSiUFIQxVP06RCVpq3dmDo2oi6ABpYaDYaTRr67BEhL8r1wgEZZKg==", - "dev": true - }, - "regenerate-unicode-properties": { - "version": "8.2.0", - "resolved": "https://registry.npmjs.org/regenerate-unicode-properties/-/regenerate-unicode-properties-8.2.0.tgz", - "integrity": "sha512-F9DjY1vKLo/tPePDycuH3dn9H1OTPIkVD9Kz4LODu+F2C75mgjAJ7x/gwy6ZcSNRAAkhNlJSOHRe8k3p+K9WhA==", - "dev": true, - "requires": { - "regenerate": "^1.4.0" - } - }, - "regenerator-runtime": { - "version": "0.13.5", - "resolved": "https://registry.npmjs.org/regenerator-runtime/-/regenerator-runtime-0.13.5.tgz", - "integrity": "sha512-ZS5w8CpKFinUzOwW3c83oPeVXoNsrLsaCoLtJvAClH135j/R77RuymhiSErhm2lKcwSCIpmvIWSbDkIfAqKQlA==", - "dev": true - }, - "regenerator-transform": { - "version": "0.14.4", - "resolved": "https://registry.npmjs.org/regenerator-transform/-/regenerator-transform-0.14.4.tgz", - "integrity": "sha512-EaJaKPBI9GvKpvUz2mz4fhx7WPgvwRLY9v3hlNHWmAuJHI13T4nwKnNvm5RWJzEdnI5g5UwtOww+S8IdoUC2bw==", - "dev": true, - "requires": { - "@babel/runtime": "^7.8.4", - "private": "^0.1.8" - } - }, - "regex-not": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/regex-not/-/regex-not-1.0.2.tgz", - "integrity": "sha512-J6SDjUgDxQj5NusnOtdFxDwN/+HWykR8GELwctJ7mdqhcyy1xEc4SRFHUXvxTp661YaVKAjfRLZ9cCqS6tn32A==", - "dev": true, - "requires": { - "extend-shallow": "^3.0.2", - "safe-regex": "^1.1.0" - } - }, - "regexp.prototype.flags": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/regexp.prototype.flags/-/regexp.prototype.flags-1.3.0.tgz", - "integrity": "sha512-2+Q0C5g951OlYlJz6yu5/M33IcsESLlLfsyIaLJaG4FA2r4yP8MvVMJUUP/fVBkSpbbbZlS5gynbEWLipiiXiQ==", - "dev": true, - "requires": { - "define-properties": "^1.1.3", - "es-abstract": "^1.17.0-next.1" - } - }, - "regexpu-core": { - "version": "4.7.0", - "resolved": "https://registry.npmjs.org/regexpu-core/-/regexpu-core-4.7.0.tgz", - "integrity": "sha512-TQ4KXRnIn6tz6tjnrXEkD/sshygKH/j5KzK86X8MkeHyZ8qst/LZ89j3X4/8HEIfHANTFIP/AbXakeRhWIl5YQ==", - "dev": true, - "requires": { - "regenerate": "^1.4.0", - "regenerate-unicode-properties": "^8.2.0", - "regjsgen": "^0.5.1", - "regjsparser": "^0.6.4", - "unicode-match-property-ecmascript": "^1.0.4", - "unicode-match-property-value-ecmascript": "^1.2.0" - } - }, - "regjsgen": { - "version": "0.5.1", - "resolved": "https://registry.npmjs.org/regjsgen/-/regjsgen-0.5.1.tgz", - "integrity": "sha512-5qxzGZjDs9w4tzT3TPhCJqWdCc3RLYwy9J2NB0nm5Lz+S273lvWcpjaTGHsT1dc6Hhfq41uSEOw8wBmxrKOuyg==", - "dev": true - }, - "regjsparser": { - "version": "0.6.4", - "resolved": "https://registry.npmjs.org/regjsparser/-/regjsparser-0.6.4.tgz", - "integrity": "sha512-64O87/dPDgfk8/RQqC4gkZoGyyWFIEUTTh80CU6CWuK5vkCGyekIx+oKcEIYtP/RAxSQltCZHCNu/mdd7fqlJw==", - "dev": true, - "requires": { - "jsesc": "~0.5.0" - }, - "dependencies": { - "jsesc": { - "version": "0.5.0", - "resolved": "https://registry.npmjs.org/jsesc/-/jsesc-0.5.0.tgz", - "integrity": "sha1-597mbjXW/Bb3EP6R1c9p9w8IkR0=", - "dev": true - } - } - }, - "remove-trailing-separator": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/remove-trailing-separator/-/remove-trailing-separator-1.1.0.tgz", - "integrity": "sha1-wkvOKig62tW8P1jg1IJJuSN52O8=", - "dev": true - }, - "repeat-element": { - "version": "1.1.3", - "resolved": "https://registry.npmjs.org/repeat-element/-/repeat-element-1.1.3.tgz", - "integrity": "sha512-ahGq0ZnV5m5XtZLMb+vP76kcAM5nkLqk0lpqAuojSKGgQtn4eRi4ZZGm2olo2zKFH+sMsWaqOCW1dqAnOru72g==", - "dev": true - }, - "repeat-string": { - "version": "1.6.1", - "resolved": "https://registry.npmjs.org/repeat-string/-/repeat-string-1.6.1.tgz", - "integrity": "sha1-jcrkcOHIirwtYA//Sndihtp15jc=", - "dev": true - }, - "request": { - "version": "2.88.2", - "resolved": "https://registry.npmjs.org/request/-/request-2.88.2.tgz", - "integrity": "sha512-MsvtOrfG9ZcrOwAW+Qi+F6HbD0CWXEh9ou77uOb7FM2WPhwT7smM833PzanhJLsgXjN89Ir6V2PczXNnMpwKhw==", - "dev": true, - "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": { - "uuid": { - "version": "3.4.0", - "resolved": "https://registry.npmjs.org/uuid/-/uuid-3.4.0.tgz", - "integrity": "sha512-HjSDRw6gZE5JMggctHBcjVak08+KEVhSIiDzFnT9S9aegmp85S/bReBVTb4QTFaRNptJ9kuYaNhnbNEOkbKb/A==", - "dev": true - } - } - }, - "require-directory": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz", - "integrity": "sha1-jGStX9MNqxyXbiNE/+f3kqam30I=", - "dev": true - }, - "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=", - "dev": true - }, - "requires-port": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/requires-port/-/requires-port-1.0.0.tgz", - "integrity": "sha1-kl0mAdOaxIXgkc8NpcbmlNw9yv8=", - "dev": true - }, - "resolve": { - "version": "1.17.0", - "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.17.0.tgz", - "integrity": "sha512-ic+7JYiV8Vi2yzQGFWOkiZD5Z9z7O2Zhm9XMaTxdJExKasieFCr+yXZ/WmXsckHiKl12ar0y6XiXDx3m4RHn1w==", - "dev": true, - "requires": { - "path-parse": "^1.0.6" - } - }, - "resolve-cwd": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/resolve-cwd/-/resolve-cwd-2.0.0.tgz", - "integrity": "sha1-AKn3OHVW4nA46uIyyqNypqWbZlo=", - "dev": true, - "requires": { - "resolve-from": "^3.0.0" - } - }, - "resolve-from": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-3.0.0.tgz", - "integrity": "sha1-six699nWiBvItuZTM17rywoYh0g=", - "dev": true - }, - "resolve-url": { - "version": "0.2.1", - "resolved": "https://registry.npmjs.org/resolve-url/-/resolve-url-0.2.1.tgz", - "integrity": "sha1-LGN/53yJOv0qZj/iGqkIAGjiBSo=", - "dev": true - }, - "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" - } - }, - "ret": { - "version": "0.1.15", - "resolved": "https://registry.npmjs.org/ret/-/ret-0.1.15.tgz", - "integrity": "sha512-TTlYpa+OL+vMMNG24xSlQGEJ3B/RzEfUlLct7b5G/ytav+wPrplCpVMFuwzXbkecJrb6IYo1iFb0S9v37754mg==", - "dev": true - }, - "retry": { - "version": "0.12.0", - "resolved": "https://registry.npmjs.org/retry/-/retry-0.12.0.tgz", - "integrity": "sha1-G0KmJmoh8HQh0bC1S33BZ7AcATs=", - "dev": true - }, - "rfdc": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/rfdc/-/rfdc-1.3.0.tgz", - "integrity": "sha512-V2hovdzFbOi77/WajaSMXk2OLm+xNIeQdMMuB7icj7bk6zi2F8GGAxigcnDFpJHbNyNcgyJDiP+8nOrY5cZGrA==", - "dev": true - }, - "rgb-regex": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/rgb-regex/-/rgb-regex-1.0.1.tgz", - "integrity": "sha1-wODWiC3w4jviVKR16O3UGRX+rrE=", - "dev": true - }, - "rgba-regex": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/rgba-regex/-/rgba-regex-1.0.0.tgz", - "integrity": "sha1-QzdOLiyglosO8VI0YLfXMP8i7rM=", - "dev": true - }, - "rimraf": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-3.0.2.tgz", - "integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==", - "dev": true, - "requires": { - "glob": "^7.1.3" - } - }, - "ripemd160": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/ripemd160/-/ripemd160-2.0.2.tgz", - "integrity": "sha512-ii4iagi25WusVoiC4B4lq7pbXfAp3D9v5CwfkY33vffw2+pkDjY1D8GaN7spsxvCSx8dkPqOZCEZyfxcmJG2IA==", - "dev": true, - "requires": { - "hash-base": "^3.0.0", - "inherits": "^2.0.1" - } - }, - "rollup": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/rollup/-/rollup-2.1.0.tgz", - "integrity": "sha512-gfE1455AEazVVTJoeQtcOq/U6GSxwoj4XPSWVsuWmgIxj7sBQNLDOSA82PbdMe+cP8ql8fR1jogPFe8Wg8g4SQ==", - "dev": true, - "requires": { - "fsevents": "~2.1.2" - } - }, - "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-queue": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/run-queue/-/run-queue-1.0.3.tgz", - "integrity": "sha1-6Eg5bwV9Ij8kOGkkYY4laUFh7Ec=", - "dev": true, - "requires": { - "aproba": "^1.1.1" - } - }, - "rxjs": { - "version": "6.5.5", - "resolved": "https://registry.npmjs.org/rxjs/-/rxjs-6.5.5.tgz", - "integrity": "sha512-WfQI+1gohdf0Dai/Bbmk5L5ItH5tYqm3ki2c5GdWhKjalzjg93N3avFjVStyZZz+A2Em+ZxKH5bNghw9UeylGQ==", - "requires": { - "tslib": "^1.9.0" - } - }, - "rxjs-compat": { - "version": "6.5.5", - "resolved": "https://registry.npmjs.org/rxjs-compat/-/rxjs-compat-6.5.5.tgz", - "integrity": "sha512-F42sssVbUyWH4vJswEo6m+Eh02xHv3q93n8S7nUJO58R7sbc3CvJIOts605zdaBhWa1xMB9aVSyqPqhQ5q3eXg==" - }, - "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==", - "dev": true - }, - "safe-regex": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/safe-regex/-/safe-regex-1.1.0.tgz", - "integrity": "sha1-QKNmnzsHfR6UPURinhV91IAjvy4=", - "dev": true, - "requires": { - "ret": "~0.1.10" - } - }, - "safer-buffer": { - "version": "2.1.2", - "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", - "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==", - "dev": true - }, - "sass": { - "version": "1.26.3", - "resolved": "https://registry.npmjs.org/sass/-/sass-1.26.3.tgz", - "integrity": "sha512-5NMHI1+YFYw4sN3yfKjpLuV9B5l7MqQ6FlkTcC4FT+oHbBRUZoSjHrrt/mE0nFXJyY2kQtU9ou9HxvFVjLFuuw==", - "dev": true, - "requires": { - "chokidar": ">=2.0.0 <4.0.0" - } - }, - "sass-loader": { - "version": "8.0.2", - "resolved": "https://registry.npmjs.org/sass-loader/-/sass-loader-8.0.2.tgz", - "integrity": "sha512-7o4dbSK8/Ol2KflEmSco4jTjQoV988bM82P9CZdmo9hR3RLnvNc0ufMNdMrB0caq38JQ/FgF4/7RcbcfKzxoFQ==", - "dev": true, - "requires": { - "clone-deep": "^4.0.1", - "loader-utils": "^1.2.3", - "neo-async": "^2.6.1", - "schema-utils": "^2.6.1", - "semver": "^6.3.0" - }, - "dependencies": { - "json5": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/json5/-/json5-1.0.1.tgz", - "integrity": "sha512-aKS4WQjPenRxiQsC93MNfjx+nbF4PAdYzmd/1JIj8HYzqfbu86beTuNgXDzPknWk0n0uARlyewZo4s++ES36Ow==", - "dev": true, - "requires": { - "minimist": "^1.2.0" - } - }, - "loader-utils": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/loader-utils/-/loader-utils-1.4.0.tgz", - "integrity": "sha512-qH0WSMBtn/oHuwjy/NucEgbx5dbxxnxup9s4PVXJUDHZBQY+s0NWA9rJf53RBnQZxfch7euUui7hpoAPvALZdA==", - "dev": true, - "requires": { - "big.js": "^5.2.2", - "emojis-list": "^3.0.0", - "json5": "^1.0.1" - } - } - } - }, - "saucelabs": { - "version": "1.5.0", - "resolved": "https://registry.npmjs.org/saucelabs/-/saucelabs-1.5.0.tgz", - "integrity": "sha512-jlX3FGdWvYf4Q3LFfFWS1QvPg3IGCGWxIc8QBFdPTbpTJnt/v17FHXYVAn7C8sHf1yUXo2c7yIM0isDryfYtHQ==", - "dev": true, - "requires": { - "https-proxy-agent": "^2.2.1" - } - }, - "sax": { - "version": "1.2.4", - "resolved": "https://registry.npmjs.org/sax/-/sax-1.2.4.tgz", - "integrity": "sha512-NqVDv9TpANUjFm0N8uM5GxL36UgKi9/atZw+x7YFnQ8ckwFGKrl4xX4yWtrey3UJm5nP1kUbnYgLopqWNSRhWw==", - "dev": true - }, - "schema-utils": { - "version": "2.6.6", - "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-2.6.6.tgz", - "integrity": "sha512-wHutF/WPSbIi9x6ctjGGk2Hvl0VOz5l3EKEuKbjPlB30mKZUzb9A5k9yEXRX3pwyqVLPvpfZZEllaFq/M718hA==", - "dev": true, - "requires": { - "ajv": "^6.12.0", - "ajv-keywords": "^3.4.1" - }, - "dependencies": { - "ajv": { - "version": "6.12.2", - "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.2.tgz", - "integrity": "sha512-k+V+hzjm5q/Mr8ef/1Y9goCmlsK4I6Sm74teeyGvFk1XrOsbsKLjEdrvny42CZ+a8sXbk8KWpY/bDwS+FLL2UQ==", - "dev": true, - "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" - } - }, - "fast-deep-equal": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.1.tgz", - "integrity": "sha512-8UEa58QDLauDNfpbrX55Q9jrGHThw2ZMdOky5Gl1CDtVeJDPVrG4Jxx1N8jw2gkWaff5UUuX1KJd+9zGe2B+ZA==", - "dev": true - } - } - }, - "select-hose": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/select-hose/-/select-hose-2.0.0.tgz", - "integrity": "sha1-Yl2GWPhlr0Psliv8N2o3NZpJlMo=", - "dev": true - }, - "selenium-webdriver": { - "version": "3.6.0", - "resolved": "https://registry.npmjs.org/selenium-webdriver/-/selenium-webdriver-3.6.0.tgz", - "integrity": "sha512-WH7Aldse+2P5bbFBO4Gle/nuQOdVwpHMTL6raL3uuBj/vPG07k6uzt3aiahu352ONBr5xXh0hDlM3LhtXPOC4Q==", - "dev": true, - "requires": { - "jszip": "^3.1.3", - "rimraf": "^2.5.4", - "tmp": "0.0.30", - "xml2js": "^0.4.17" - }, - "dependencies": { - "rimraf": { - "version": "2.7.1", - "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", - "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==", - "dev": true, - "requires": { - "glob": "^7.1.3" - } - }, - "tmp": { - "version": "0.0.30", - "resolved": "https://registry.npmjs.org/tmp/-/tmp-0.0.30.tgz", - "integrity": "sha1-ckGdSovn1s51FI/YsyTlk6cRwu0=", - "dev": true, - "requires": { - "os-tmpdir": "~1.0.1" - } - } - } - }, - "selfsigned": { - "version": "1.10.7", - "resolved": "https://registry.npmjs.org/selfsigned/-/selfsigned-1.10.7.tgz", - "integrity": "sha512-8M3wBCzeWIJnQfl43IKwOmC4H/RAp50S8DF60znzjW5GVqTcSe2vWclt7hmYVPkKPlHWOu5EaWOMZ2Y6W8ZXTA==", - "dev": true, - "requires": { - "node-forge": "0.9.0" - } - }, - "semver": { - "version": "6.3.0", - "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", - "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", - "dev": true - }, - "semver-dsl": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/semver-dsl/-/semver-dsl-1.0.1.tgz", - "integrity": "sha1-02eN5VVeimH2Ke7QJTZq5fJzQKA=", - "dev": true, - "requires": { - "semver": "^5.3.0" - }, - "dependencies": { - "semver": { - "version": "5.7.1", - "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", - "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", - "dev": true - } - } - }, - "semver-intersect": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/semver-intersect/-/semver-intersect-1.4.0.tgz", - "integrity": "sha512-d8fvGg5ycKAq0+I6nfWeCx6ffaWJCsBYU0H2Rq56+/zFePYfT8mXkB3tWBSjR5BerkHNZ5eTPIk1/LBYas35xQ==", - "dev": true, - "requires": { - "semver": "^5.0.0" - }, - "dependencies": { - "semver": { - "version": "5.7.1", - "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", - "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", - "dev": true - } - } - }, - "send": { - "version": "0.17.1", - "resolved": "https://registry.npmjs.org/send/-/send-0.17.1.tgz", - "integrity": "sha512-BsVKsiGcQMFwT8UxypobUKyv7irCNRHk1T0G680vk88yf6LBByGcZJOTJCrTP2xVN6yI+XjPJcNuE3V4fT9sAg==", - "dev": true, - "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==", - "dev": true, - "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=", - "dev": true - } - } - }, - "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==", - "dev": true - } - } - }, - "serialize-javascript": { - "version": "2.1.2", - "resolved": "https://registry.npmjs.org/serialize-javascript/-/serialize-javascript-2.1.2.tgz", - "integrity": "sha512-rs9OggEUF0V4jUSecXazOYsLfu7OGK2qIn3c7IPBiffz32XniEp/TX9Xmc9LQfK2nQ2QKHvZ2oygKUGU0lG4jQ==", - "dev": true - }, - "serve-index": { - "version": "1.9.1", - "resolved": "https://registry.npmjs.org/serve-index/-/serve-index-1.9.1.tgz", - "integrity": "sha1-03aNabHn2C5c4FD/9bRTvqEqkjk=", - "dev": true, - "requires": { - "accepts": "~1.3.4", - "batch": "0.6.1", - "debug": "2.6.9", - "escape-html": "~1.0.3", - "http-errors": "~1.6.2", - "mime-types": "~2.1.17", - "parseurl": "~1.3.2" - }, - "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" - } - }, - "http-errors": { - "version": "1.6.3", - "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.6.3.tgz", - "integrity": "sha1-i1VoC7S+KDoLW/TqLjhYC+HZMg0=", - "dev": true, - "requires": { - "depd": "~1.1.2", - "inherits": "2.0.3", - "setprototypeof": "1.1.0", - "statuses": ">= 1.4.0 < 2" - } - }, - "inherits": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", - "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=", - "dev": true - }, - "ms": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", - "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", - "dev": true - }, - "setprototypeof": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.0.tgz", - "integrity": "sha512-BvE/TwpZX4FXExxOxZyRGQQv651MSwmWKZGqvmPcRIjDqWub67kTKuIMx43cZZrS/cBBzwBcNDWoFxt2XEFIpQ==", - "dev": true - } - } - }, - "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==", - "dev": true, - "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=", - "dev": true - }, - "set-immediate-shim": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/set-immediate-shim/-/set-immediate-shim-1.0.1.tgz", - "integrity": "sha1-SysbJ+uAip+NzEgaWOXlb1mfP2E=", - "dev": true - }, - "set-value": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/set-value/-/set-value-2.0.1.tgz", - "integrity": "sha512-JxHc1weCN68wRY0fhCoXpyK55m/XPHafOmK4UWD7m2CI14GMcFypt4w/0+NV5f/ZMby2F6S2wwA7fgynh9gWSw==", - "dev": true, - "requires": { - "extend-shallow": "^2.0.1", - "is-extendable": "^0.1.1", - "is-plain-object": "^2.0.3", - "split-string": "^3.0.1" - }, - "dependencies": { - "extend-shallow": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", - "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", - "dev": true, - "requires": { - "is-extendable": "^0.1.0" - } - } - } - }, - "setimmediate": { - "version": "1.0.5", - "resolved": "https://registry.npmjs.org/setimmediate/-/setimmediate-1.0.5.tgz", - "integrity": "sha1-KQy7Iy4waULX1+qbg3Mqt4VvgoU=", - "dev": true - }, - "setprototypeof": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.1.tgz", - "integrity": "sha512-JvdAWfbXeIGaZ9cILp38HntZSFSo3mWg6xGcJJsd+d4aRMOqauag1C63dJfDw7OaMYwEbHMOxEZ1lqVRYP2OAw==", - "dev": true - }, - "sha.js": { - "version": "2.4.11", - "resolved": "https://registry.npmjs.org/sha.js/-/sha.js-2.4.11.tgz", - "integrity": "sha512-QMEp5B7cftE7APOjk5Y6xgrbWu+WkLVQwk8JNjZ8nKRciZaByEW6MubieAiToS7+dwvrjGhH8jRXz3MVd0AYqQ==", - "dev": true, - "requires": { - "inherits": "^2.0.1", - "safe-buffer": "^5.0.1" - } - }, - "shallow-clone": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/shallow-clone/-/shallow-clone-3.0.1.tgz", - "integrity": "sha512-/6KqX+GVUdqPuPPd2LxDDxzX6CAbjJehAAOKlNpqqUpAqPM6HeL8f+o3a+JsyGjn2lv0WY8UsTgUJjU9Ok55NA==", - "dev": true, - "requires": { - "kind-of": "^6.0.2" - } - }, - "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=", - "dev": true, - "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==", - "dev": true - } - } - }, - "slash": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/slash/-/slash-1.0.0.tgz", - "integrity": "sha1-xB8vbDn8FtHNF61LXYlhFK5HDVU=", - "dev": true - }, - "smart-buffer": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/smart-buffer/-/smart-buffer-4.1.0.tgz", - "integrity": "sha512-iVICrxOzCynf/SNaBQCw34eM9jROU/s5rzIhpOvzhzuYHfJR/DhZfDkXiZSgKXfgv26HT3Yni3AV/DGw0cGnnw==", - "dev": true - }, - "snapdragon": { - "version": "0.8.2", - "resolved": "https://registry.npmjs.org/snapdragon/-/snapdragon-0.8.2.tgz", - "integrity": "sha512-FtyOnWN/wCHTVXOMwvSv26d+ko5vWlIDD6zoUJ7LW8vh+ZBC8QdljveRP+crNrtBwioEUWy/4dMtbBjA4ioNlg==", - "dev": true, - "requires": { - "base": "^0.11.1", - "debug": "^2.2.0", - "define-property": "^0.2.5", - "extend-shallow": "^2.0.1", - "map-cache": "^0.2.2", - "source-map": "^0.5.6", - "source-map-resolve": "^0.5.0", - "use": "^3.1.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" - } - }, - "define-property": { - "version": "0.2.5", - "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", - "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", - "dev": true, - "requires": { - "is-descriptor": "^0.1.0" - } - }, - "extend-shallow": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", - "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", - "dev": true, - "requires": { - "is-extendable": "^0.1.0" - } - }, - "ms": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", - "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", - "dev": true - }, - "source-map": { - "version": "0.5.7", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.5.7.tgz", - "integrity": "sha1-igOdLRAh0i0eoUyA2OpGi6LvP8w=", - "dev": true - } - } - }, - "snapdragon-node": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/snapdragon-node/-/snapdragon-node-2.1.1.tgz", - "integrity": "sha512-O27l4xaMYt/RSQ5TR3vpWCAB5Kb/czIcqUFOM/C4fYcLnbZUc1PkjTAMjof2pBWaSTwOUd6qUHcFGVGj7aIwnw==", - "dev": true, - "requires": { - "define-property": "^1.0.0", - "isobject": "^3.0.0", - "snapdragon-util": "^3.0.1" - }, - "dependencies": { - "define-property": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/define-property/-/define-property-1.0.0.tgz", - "integrity": "sha1-dp66rz9KY6rTr56NMEybvnm/sOY=", - "dev": true, - "requires": { - "is-descriptor": "^1.0.0" - } - }, - "is-accessor-descriptor": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-1.0.0.tgz", - "integrity": "sha512-m5hnHTkcVsPfqx3AKlyttIPb7J+XykHvJP2B9bZDjlhLIoEq4XoK64Vg7boZlVWYK6LUY94dYPEE7Lh0ZkZKcQ==", - "dev": true, - "requires": { - "kind-of": "^6.0.0" - } - }, - "is-data-descriptor": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-1.0.0.tgz", - "integrity": "sha512-jbRXy1FmtAoCjQkVmIVYwuuqDFUbaOeDjmed1tOGPrsMhtJA4rD9tkgA0F1qJ3gRFRXcHYVkdeaP50Q5rE/jLQ==", - "dev": true, - "requires": { - "kind-of": "^6.0.0" - } - }, - "is-descriptor": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-1.0.2.tgz", - "integrity": "sha512-2eis5WqQGV7peooDyLmNEPUrps9+SXX5c9pL3xEB+4e9HnGuDa7mB7kHxHw4CbqS9k1T2hOH3miL8n8WtiYVtg==", - "dev": true, - "requires": { - "is-accessor-descriptor": "^1.0.0", - "is-data-descriptor": "^1.0.0", - "kind-of": "^6.0.2" - } - } - } - }, - "snapdragon-util": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/snapdragon-util/-/snapdragon-util-3.0.1.tgz", - "integrity": "sha512-mbKkMdQKsjX4BAL4bRYTj21edOf8cN7XHdYUJEe+Zn99hVEYcMvKPct1IqNe7+AZPirn8BCDOQBHQZknqmKlZQ==", - "dev": true, - "requires": { - "kind-of": "^3.2.0" - }, - "dependencies": { - "kind-of": { - "version": "3.2.2", - "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", - "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", - "dev": true, - "requires": { - "is-buffer": "^1.1.5" - } - } - } - }, - "socket.io": { - "version": "4.4.1", - "resolved": "https://registry.npmjs.org/socket.io/-/socket.io-4.4.1.tgz", - "integrity": "sha512-s04vrBswdQBUmuWJuuNTmXUVJhP0cVky8bBDhdkf8y0Ptsu7fKU2LuLbts9g+pdmAdyMMn8F/9Mf1/wbtUN0fg==", - "dev": true, - "requires": { - "accepts": "~1.3.4", - "base64id": "~2.0.0", - "debug": "~4.3.2", - "engine.io": "~6.1.0", - "socket.io-adapter": "~2.3.3", - "socket.io-parser": "~4.0.4" - }, - "dependencies": { - "debug": { - "version": "4.3.3", - "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.3.tgz", - "integrity": "sha512-/zxw5+vh1Tfv+4Qn7a5nsbcJKPaSvCDhojn6FEl9vupwK2VCSDtEiEtqr8DFtzYFOdz63LBkxec7DYuc2jon6Q==", - "dev": true, - "requires": { - "ms": "2.1.2" - } - } - } - }, - "socket.io-adapter": { - "version": "2.3.3", - "resolved": "https://registry.npmjs.org/socket.io-adapter/-/socket.io-adapter-2.3.3.tgz", - "integrity": "sha512-Qd/iwn3VskrpNO60BeRyCyr8ZWw9CPZyitW4AQwmRZ8zCiyDiL+znRnWX6tDHXnWn1sJrM1+b6Mn6wEDJJ4aYQ==", - "dev": true - }, - "socket.io-parser": { - "version": "4.0.4", - "resolved": "https://registry.npmjs.org/socket.io-parser/-/socket.io-parser-4.0.4.tgz", - "integrity": "sha512-t+b0SS+IxG7Rxzda2EVvyBZbvFPBCjJoyHuE0P//7OAsN23GItzDRdWa6ALxZI/8R5ygK7jAR6t028/z+7295g==", - "dev": true, - "requires": { - "@types/component-emitter": "^1.2.10", - "component-emitter": "~1.3.0", - "debug": "~4.3.1" - }, - "dependencies": { - "debug": { - "version": "4.3.3", - "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.3.tgz", - "integrity": "sha512-/zxw5+vh1Tfv+4Qn7a5nsbcJKPaSvCDhojn6FEl9vupwK2VCSDtEiEtqr8DFtzYFOdz63LBkxec7DYuc2jon6Q==", - "dev": true, - "requires": { - "ms": "2.1.2" - } - } - } - }, - "sockjs": { - "version": "0.3.19", - "resolved": "https://registry.npmjs.org/sockjs/-/sockjs-0.3.19.tgz", - "integrity": "sha512-V48klKZl8T6MzatbLlzzRNhMepEys9Y4oGFpypBFFn1gLI/QQ9HtLLyWJNbPlwGLelOVOEijUbTTJeLLI59jLw==", - "dev": true, - "requires": { - "faye-websocket": "^0.10.0", - "uuid": "^3.0.1" - }, - "dependencies": { - "uuid": { - "version": "3.4.0", - "resolved": "https://registry.npmjs.org/uuid/-/uuid-3.4.0.tgz", - "integrity": "sha512-HjSDRw6gZE5JMggctHBcjVak08+KEVhSIiDzFnT9S9aegmp85S/bReBVTb4QTFaRNptJ9kuYaNhnbNEOkbKb/A==", - "dev": true - } - } - }, - "sockjs-client": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/sockjs-client/-/sockjs-client-1.4.0.tgz", - "integrity": "sha512-5zaLyO8/nri5cua0VtOrFXBPK1jbL4+1cebT/mmKA1E1ZXOvJrII75bPu0l0k843G/+iAbhEqzyKr0w/eCCj7g==", - "dev": true, - "requires": { - "debug": "^3.2.5", - "eventsource": "^1.0.7", - "faye-websocket": "~0.11.1", - "inherits": "^2.0.3", - "json3": "^3.3.2", - "url-parse": "^1.4.3" - }, - "dependencies": { - "debug": { - "version": "3.2.6", - "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.6.tgz", - "integrity": "sha512-mel+jf7nrtEl5Pn1Qx46zARXKDpBbvzezse7p7LqINmdoIk8PYP5SySaxEmYv6TZ0JyEKA1hsCId6DIhgITtWQ==", - "dev": true, - "requires": { - "ms": "^2.1.1" - } - }, - "faye-websocket": { - "version": "0.11.3", - "resolved": "https://registry.npmjs.org/faye-websocket/-/faye-websocket-0.11.3.tgz", - "integrity": "sha512-D2y4bovYpzziGgbHYtGCMjlJM36vAl/y+xUyn1C+FVx8szd1E+86KwVw6XvYSzOP8iMpm1X0I4xJD+QtUb36OA==", - "dev": true, - "requires": { - "websocket-driver": ">=0.5.1" - } - } - } - }, - "socks": { - "version": "2.3.3", - "resolved": "https://registry.npmjs.org/socks/-/socks-2.3.3.tgz", - "integrity": "sha512-o5t52PCNtVdiOvzMry7wU4aOqYWL0PeCXRWBEiJow4/i/wr+wpsJQ9awEu1EonLIqsfGd5qSgDdxEOvCdmBEpA==", - "dev": true, - "requires": { - "ip": "1.1.5", - "smart-buffer": "^4.1.0" - } - }, - "socks-proxy-agent": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/socks-proxy-agent/-/socks-proxy-agent-4.0.2.tgz", - "integrity": "sha512-NT6syHhI9LmuEMSK6Kd2V7gNv5KFZoLE7V5udWmn0de+3Mkj3UMA/AJPLyeNUVmElCurSHtUdM3ETpR3z770Wg==", - "dev": true, - "requires": { - "agent-base": "~4.2.1", - "socks": "~2.3.2" - }, - "dependencies": { - "agent-base": { - "version": "4.2.1", - "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-4.2.1.tgz", - "integrity": "sha512-JVwXMr9nHYTUXsBFKUqhJwvlcYU/blreOEUkhNR2eXZIvwd+c+o5V4MgDPKWnMS/56awN3TRzIP+KoPn+roQtg==", - "dev": true, - "requires": { - "es6-promisify": "^5.0.0" - } - } - } - }, - "sort-keys": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/sort-keys/-/sort-keys-1.1.2.tgz", - "integrity": "sha1-RBttTTRnmPG05J6JIK37oOVD+a0=", - "dev": true, - "requires": { - "is-plain-obj": "^1.0.0" - } - }, - "source-list-map": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/source-list-map/-/source-list-map-2.0.1.tgz", - "integrity": "sha512-qnQ7gVMxGNxsiL4lEuJwe/To8UnK7fAnmbGEEH8RpLouuKbeEm0lhbQVFIrNSuB+G7tVrAlVsZgETT5nljf+Iw==", - "dev": true - }, - "source-map": { - "version": "0.7.3", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.7.3.tgz", - "integrity": "sha512-CkCj6giN3S+n9qrYiBTX5gystlENnRW5jZeNLHpe6aue+SrHcG5VYwujhW9s4dY31mEGsxBDrHR6oI69fTXsaQ==", - "dev": true - }, - "source-map-loader": { - "version": "0.2.4", - "resolved": "https://registry.npmjs.org/source-map-loader/-/source-map-loader-0.2.4.tgz", - "integrity": "sha512-OU6UJUty+i2JDpTItnizPrlpOIBLmQbWMuBg9q5bVtnHACqw1tn9nNwqJLbv0/00JjnJb/Ee5g5WS5vrRv7zIQ==", - "dev": true, - "requires": { - "async": "^2.5.0", - "loader-utils": "^1.1.0" - }, - "dependencies": { - "json5": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/json5/-/json5-1.0.1.tgz", - "integrity": "sha512-aKS4WQjPenRxiQsC93MNfjx+nbF4PAdYzmd/1JIj8HYzqfbu86beTuNgXDzPknWk0n0uARlyewZo4s++ES36Ow==", - "dev": true, - "requires": { - "minimist": "^1.2.0" - } - }, - "loader-utils": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/loader-utils/-/loader-utils-1.4.0.tgz", - "integrity": "sha512-qH0WSMBtn/oHuwjy/NucEgbx5dbxxnxup9s4PVXJUDHZBQY+s0NWA9rJf53RBnQZxfch7euUui7hpoAPvALZdA==", - "dev": true, - "requires": { - "big.js": "^5.2.2", - "emojis-list": "^3.0.0", - "json5": "^1.0.1" - } - } - } - }, - "source-map-resolve": { - "version": "0.5.3", - "resolved": "https://registry.npmjs.org/source-map-resolve/-/source-map-resolve-0.5.3.tgz", - "integrity": "sha512-Htz+RnsXWk5+P2slx5Jh3Q66vhQj1Cllm0zvnaY98+NFx+Dv2CF/f5O/t8x+KaNdrdIAsruNzoh/KpialbqAnw==", - "dev": true, - "requires": { - "atob": "^2.1.2", - "decode-uri-component": "^0.2.0", - "resolve-url": "^0.2.1", - "source-map-url": "^0.4.0", - "urix": "^0.1.0" - } - }, - "source-map-support": { - "version": "0.5.16", - "resolved": "https://registry.npmjs.org/source-map-support/-/source-map-support-0.5.16.tgz", - "integrity": "sha512-efyLRJDr68D9hBBNIPWFjhpFzURh+KJykQwvMyW5UiZzYwoF6l4YMMDIJJEyFWxWCqfyxLzz6tSfUFR+kXXsVQ==", - "dev": true, - "requires": { - "buffer-from": "^1.0.0", - "source-map": "^0.6.0" - }, - "dependencies": { - "source-map": { - "version": "0.6.1", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", - "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", - "dev": true - } - } - }, - "source-map-url": { - "version": "0.4.0", - "resolved": "https://registry.npmjs.org/source-map-url/-/source-map-url-0.4.0.tgz", - "integrity": "sha1-PpNdfd1zYxuXZZlW1VEo6HtQhKM=", - "dev": true - }, - "sourcemap-codec": { - "version": "1.4.8", - "resolved": "https://registry.npmjs.org/sourcemap-codec/-/sourcemap-codec-1.4.8.tgz", - "integrity": "sha512-9NykojV5Uih4lgo5So5dtw+f0JgJX30KCNI8gwhz2J9A15wD0Ml6tjHKwf6fTSa6fAdVBdZeNOs9eJ71qCk8vA==", - "dev": true - }, - "spdx-correct": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/spdx-correct/-/spdx-correct-3.1.0.tgz", - "integrity": "sha512-lr2EZCctC2BNR7j7WzJ2FpDznxky1sjfxvvYEyzxNyb6lZXHODmEoJeFu4JupYlkfha1KZpJyoqiJ7pgA1qq8Q==", - "dev": true, - "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==", - "dev": true - }, - "spdx-expression-parse": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/spdx-expression-parse/-/spdx-expression-parse-3.0.0.tgz", - "integrity": "sha512-Yg6D3XpRD4kkOmTpdgbUiEJFKghJH03fiC1OPll5h/0sO6neh2jqRDVHOQ4o/LMea0tgCkbMgea5ip/e+MkWyg==", - "dev": true, - "requires": { - "spdx-exceptions": "^2.1.0", - "spdx-license-ids": "^3.0.0" - } - }, - "spdx-license-ids": { - "version": "3.0.5", - "resolved": "https://registry.npmjs.org/spdx-license-ids/-/spdx-license-ids-3.0.5.tgz", - "integrity": "sha512-J+FWzZoynJEXGphVIS+XEh3kFSjZX/1i9gFBaWQcB+/tmpe2qUsSBABpcxqxnAxFdiUFEgAX1bjYGQvIZmoz9Q==", - "dev": true - }, - "spdy": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/spdy/-/spdy-4.0.2.tgz", - "integrity": "sha512-r46gZQZQV+Kl9oItvl1JZZqJKGr+oEkB08A6BzkiR7593/7IbtuncXHd2YoYeTsG4157ZssMu9KYvUHLcjcDoA==", - "dev": true, - "requires": { - "debug": "^4.1.0", - "handle-thing": "^2.0.0", - "http-deceiver": "^1.2.7", - "select-hose": "^2.0.0", - "spdy-transport": "^3.0.0" - } - }, - "spdy-transport": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/spdy-transport/-/spdy-transport-3.0.0.tgz", - "integrity": "sha512-hsLVFE5SjA6TCisWeJXFKniGGOpBgMLmerfO2aCyCU5s7nJ/rpAepqmFifv/GCbSbueEeAJJnmSQ2rKC/g8Fcw==", - "dev": true, - "requires": { - "debug": "^4.1.0", - "detect-node": "^2.0.4", - "hpack.js": "^2.1.6", - "obuf": "^1.1.2", - "readable-stream": "^3.0.6", - "wbuf": "^1.7.3" - }, - "dependencies": { - "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==", - "dev": true, - "requires": { - "inherits": "^2.0.3", - "string_decoder": "^1.1.1", - "util-deprecate": "^1.0.1" - } - } - } - }, - "speed-measure-webpack-plugin": { - "version": "1.3.1", - "resolved": "https://registry.npmjs.org/speed-measure-webpack-plugin/-/speed-measure-webpack-plugin-1.3.1.tgz", - "integrity": "sha512-qVIkJvbtS9j/UeZumbdfz0vg+QfG/zxonAjzefZrqzkr7xOncLVXkeGbTpzd1gjCBM4PmVNkWlkeTVhgskAGSQ==", - "dev": true, - "requires": { - "chalk": "^2.0.1" - } - }, - "split-string": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/split-string/-/split-string-3.1.0.tgz", - "integrity": "sha512-NzNVhJDYpwceVVii8/Hu6DKfD2G+NrQHlS/V/qgv763EYudVwEcMQNxd2lh+0VrUByXN/oJkl5grOhYWvQUYiw==", - "dev": true, - "requires": { - "extend-shallow": "^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=", - "dev": true - }, - "sshpk": { - "version": "1.16.1", - "resolved": "https://registry.npmjs.org/sshpk/-/sshpk-1.16.1.tgz", - "integrity": "sha512-HXXqVUq7+pcKeLqqZj6mHFUMvXtOJt1uoUx09pFW6011inTMxqI8BA8PM95myrIyyKwdnzjdFjLiE6KBPVtJIg==", - "dev": true, - "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" - } - }, - "ssri": { - "version": "8.0.1", - "resolved": "https://registry.npmjs.org/ssri/-/ssri-8.0.1.tgz", - "integrity": "sha512-97qShzy1AiyxvPNIkLWoGua7xoQzzPjQ0HAH4B0rWKo7SZ6USuPcrUiAFrws0UH8RrbWmgq3LMTObhPIHbbBeQ==", - "dev": true, - "requires": { - "minipass": "^3.1.1" - } - }, - "stable": { - "version": "0.1.8", - "resolved": "https://registry.npmjs.org/stable/-/stable-0.1.8.tgz", - "integrity": "sha512-ji9qxRnOVfcuLDySj9qzhGSEFVobyt1kIOSkj1qZzYLzq7Tos/oUUWvotUPQLlrsidqsK6tBH89Bc9kL5zHA6w==", - "dev": true - }, - "static-extend": { - "version": "0.1.2", - "resolved": "https://registry.npmjs.org/static-extend/-/static-extend-0.1.2.tgz", - "integrity": "sha1-YICcOcv/VTNyJv1eC1IPNB8ftcY=", - "dev": true, - "requires": { - "define-property": "^0.2.5", - "object-copy": "^0.1.0" - }, - "dependencies": { - "define-property": { - "version": "0.2.5", - "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", - "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", - "dev": true, - "requires": { - "is-descriptor": "^0.1.0" - } - } - } - }, - "statuses": { - "version": "1.5.0", - "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.5.0.tgz", - "integrity": "sha1-Fhx9rBd2Wf2YEfQ3cfqZOBR4Yow=", - "dev": true - }, - "stream-browserify": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/stream-browserify/-/stream-browserify-2.0.2.tgz", - "integrity": "sha512-nX6hmklHs/gr2FuxYDltq8fJA1GDlxKQCz8O/IM4atRqBH8OORmBNgfvW5gG10GT/qQ9u0CzIvr2X5Pkt6ntqg==", - "dev": true, - "requires": { - "inherits": "~2.0.1", - "readable-stream": "^2.0.2" - } - }, - "stream-each": { - "version": "1.2.3", - "resolved": "https://registry.npmjs.org/stream-each/-/stream-each-1.2.3.tgz", - "integrity": "sha512-vlMC2f8I2u/bZGqkdfLQW/13Zihpej/7PmSiMQsbYddxuTsJp8vRe2x2FvVExZg7FaOds43ROAuFJwPR4MTZLw==", - "dev": true, - "requires": { - "end-of-stream": "^1.1.0", - "stream-shift": "^1.0.0" - } - }, - "stream-http": { - "version": "2.8.3", - "resolved": "https://registry.npmjs.org/stream-http/-/stream-http-2.8.3.tgz", - "integrity": "sha512-+TSkfINHDo4J+ZobQLWiMouQYB+UVYFttRA94FpEzzJ7ZdqcL4uUUQ7WkdkI4DSozGmgBUE/a47L+38PenXhUw==", - "dev": true, - "requires": { - "builtin-status-codes": "^3.0.0", - "inherits": "^2.0.1", - "readable-stream": "^2.3.6", - "to-arraybuffer": "^1.0.0", - "xtend": "^4.0.0" - } - }, - "stream-shift": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/stream-shift/-/stream-shift-1.0.1.tgz", - "integrity": "sha512-AiisoFqQ0vbGcZgQPY1cdP2I76glaVA/RauYR4G4thNFgkTqr90yXTo4LYX60Jl+sIlPNHHdGSwo01AvbKUSVQ==", - "dev": true - }, - "streamroller": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/streamroller/-/streamroller-3.0.2.tgz", - "integrity": "sha512-ur6y5S5dopOaRXBuRIZ1u6GC5bcEXHRZKgfBjfCglMhmIf+roVCECjvkEYzNQOXIN2/JPnkMPW/8B3CZoKaEPA==", - "dev": true, - "requires": { - "date-format": "^4.0.3", - "debug": "^4.1.1", - "fs-extra": "^10.0.0" - }, - "dependencies": { - "fs-extra": { - "version": "10.0.1", - "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-10.0.1.tgz", - "integrity": "sha512-NbdoVMZso2Lsrn/QwLXOy6rm0ufY2zEOKCDzJR/0kBsb0E6qed0P3iYK+Ath3BfvXEeu4JhEtXLgILx5psUfag==", - "dev": true, - "requires": { - "graceful-fs": "^4.2.0", - "jsonfile": "^6.0.1", - "universalify": "^2.0.0" - } - }, - "jsonfile": { - "version": "6.1.0", - "resolved": "https://registry.npmjs.org/jsonfile/-/jsonfile-6.1.0.tgz", - "integrity": "sha512-5dgndWOriYSm5cnYaJNhalLNDKOqFwyDB/rr1E9ZsGciGvKPs8R2xYGCacuf3z6K1YKDz182fd+fY3cn3pMqXQ==", - "dev": true, - "requires": { - "graceful-fs": "^4.1.6", - "universalify": "^2.0.0" - } - }, - "universalify": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/universalify/-/universalify-2.0.0.tgz", - "integrity": "sha512-hAZsKq7Yy11Zu1DE0OzWjw7nnLZmJZYTDZZyEFHZdUhV8FkH5MCfoU1XMaxXovpyW5nq5scPqq0ZDP9Zyl04oQ==", - "dev": true - } - } - }, - "strict-uri-encode": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/strict-uri-encode/-/strict-uri-encode-1.1.0.tgz", - "integrity": "sha1-J5siXfHVgrH1TmWt3UNS4Y+qBxM=", - "dev": true - }, - "string-width": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/string-width/-/string-width-2.1.1.tgz", - "integrity": "sha512-nOqH59deCq9SRHlxq1Aw85Jnt4w6KvLKqWVik6oA9ZklXLNIOlqg4F2yrT1MVaTjAqvVwdfeZ7w7aCvJD7ugkw==", - "dev": true, - "requires": { - "is-fullwidth-code-point": "^2.0.0", - "strip-ansi": "^4.0.0" - }, - "dependencies": { - "ansi-regex": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-3.0.0.tgz", - "integrity": "sha1-7QMXwyIGT3lGbAKWa922Bas32Zg=", - "dev": true - }, - "strip-ansi": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-4.0.0.tgz", - "integrity": "sha1-qEeQIusaw2iocTibY1JixQXuNo8=", - "dev": true, - "requires": { - "ansi-regex": "^3.0.0" - } - } - } - }, - "string.prototype.trimend": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/string.prototype.trimend/-/string.prototype.trimend-1.0.1.tgz", - "integrity": "sha512-LRPxFUaTtpqYsTeNKaFOw3R4bxIzWOnbQ837QfBylo8jIxtcbK/A/sMV7Q+OAV/vWo+7s25pOE10KYSjaSO06g==", - "dev": true, - "requires": { - "define-properties": "^1.1.3", - "es-abstract": "^1.17.5" - } - }, - "string.prototype.trimleft": { - "version": "2.1.2", - "resolved": "https://registry.npmjs.org/string.prototype.trimleft/-/string.prototype.trimleft-2.1.2.tgz", - "integrity": "sha512-gCA0tza1JBvqr3bfAIFJGqfdRTyPae82+KTnm3coDXkZN9wnuW3HjGgN386D7hfv5CHQYCI022/rJPVlqXyHSw==", - "dev": true, - "requires": { - "define-properties": "^1.1.3", - "es-abstract": "^1.17.5", - "string.prototype.trimstart": "^1.0.0" - } - }, - "string.prototype.trimright": { - "version": "2.1.2", - "resolved": "https://registry.npmjs.org/string.prototype.trimright/-/string.prototype.trimright-2.1.2.tgz", - "integrity": "sha512-ZNRQ7sY3KroTaYjRS6EbNiiHrOkjihL9aQE/8gfQ4DtAC/aEBRHFJa44OmoWxGGqXuJlfKkZW4WcXErGr+9ZFg==", - "dev": true, - "requires": { - "define-properties": "^1.1.3", - "es-abstract": "^1.17.5", - "string.prototype.trimend": "^1.0.0" - } - }, - "string.prototype.trimstart": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/string.prototype.trimstart/-/string.prototype.trimstart-1.0.1.tgz", - "integrity": "sha512-XxZn+QpvrBI1FOcg6dIpxUPgWCPuNXvMD72aaRaUQv1eD4e/Qy8i/hFTe0BUmD60p/QA6bh1avmuPTfNjqVWRw==", - "dev": true, - "requires": { - "define-properties": "^1.1.3", - "es-abstract": "^1.17.5" - } - }, - "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==", - "dev": true, - "requires": { - "safe-buffer": "~5.1.0" - } - }, - "strip-ansi": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz", - "integrity": "sha1-ajhfuIU9lS1f8F0Oiq+UJ43GPc8=", - "dev": true, - "requires": { - "ansi-regex": "^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 - }, - "strip-eof": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/strip-eof/-/strip-eof-1.0.0.tgz", - "integrity": "sha1-u0P/VZim6wXYm1n80SnJgzE2Br8=", - "dev": true - }, - "style-loader": { - "version": "1.1.3", - "resolved": "https://registry.npmjs.org/style-loader/-/style-loader-1.1.3.tgz", - "integrity": "sha512-rlkH7X/22yuwFYK357fMN/BxYOorfnfq0eD7+vqlemSK4wEcejFF1dg4zxP0euBW8NrYx2WZzZ8PPFevr7D+Kw==", - "dev": true, - "requires": { - "loader-utils": "^1.2.3", - "schema-utils": "^2.6.4" - }, - "dependencies": { - "json5": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/json5/-/json5-1.0.1.tgz", - "integrity": "sha512-aKS4WQjPenRxiQsC93MNfjx+nbF4PAdYzmd/1JIj8HYzqfbu86beTuNgXDzPknWk0n0uARlyewZo4s++ES36Ow==", - "dev": true, - "requires": { - "minimist": "^1.2.0" - } - }, - "loader-utils": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/loader-utils/-/loader-utils-1.4.0.tgz", - "integrity": "sha512-qH0WSMBtn/oHuwjy/NucEgbx5dbxxnxup9s4PVXJUDHZBQY+s0NWA9rJf53RBnQZxfch7euUui7hpoAPvALZdA==", - "dev": true, - "requires": { - "big.js": "^5.2.2", - "emojis-list": "^3.0.0", - "json5": "^1.0.1" - } - } - } - }, - "stylehacks": { - "version": "4.0.3", - "resolved": "https://registry.npmjs.org/stylehacks/-/stylehacks-4.0.3.tgz", - "integrity": "sha512-7GlLk9JwlElY4Y6a/rmbH2MhVlTyVmiJd1PfTCqFaIBEGMYNsrO/v3SeGTdhBThLg4Z+NbOk/qFMwCa+J+3p/g==", - "dev": true, - "requires": { - "browserslist": "^4.0.0", - "postcss": "^7.0.0", - "postcss-selector-parser": "^3.0.0" - }, - "dependencies": { - "postcss-selector-parser": { - "version": "3.1.2", - "resolved": "https://registry.npmjs.org/postcss-selector-parser/-/postcss-selector-parser-3.1.2.tgz", - "integrity": "sha512-h7fJ/5uWuRVyOtkO45pnt1Ih40CEleeyCHzipqAZO2e5H20g25Y48uYnFUiShvY4rZWNJ/Bib/KVPmanaCtOhA==", - "dev": true, - "requires": { - "dot-prop": "^5.2.0", - "indexes-of": "^1.0.1", - "uniq": "^1.0.1" - } - } - } - }, - "stylus": { - "version": "0.54.7", - "resolved": "https://registry.npmjs.org/stylus/-/stylus-0.54.7.tgz", - "integrity": "sha512-Yw3WMTzVwevT6ZTrLCYNHAFmanMxdylelL3hkWNgPMeTCpMwpV3nXjpOHuBXtFv7aiO2xRuQS6OoAdgkNcSNug==", - "dev": true, - "requires": { - "css-parse": "~2.0.0", - "debug": "~3.1.0", - "glob": "^7.1.3", - "mkdirp": "~0.5.x", - "safer-buffer": "^2.1.2", - "sax": "~1.2.4", - "semver": "^6.0.0", - "source-map": "^0.7.3" - }, - "dependencies": { - "debug": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/debug/-/debug-3.1.0.tgz", - "integrity": "sha512-OX8XqP7/1a9cqkxYw2yXss15f26NKWBpDXQd0/uK/KPqdQhxbPa994hnzjcE2VqQpDslf55723cKPUOGSmMY3g==", - "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 - } - } - }, - "stylus-loader": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/stylus-loader/-/stylus-loader-3.0.2.tgz", - "integrity": "sha512-+VomPdZ6a0razP+zinir61yZgpw2NfljeSsdUF5kJuEzlo3khXhY19Fn6l8QQz1GRJGtMCo8nG5C04ePyV7SUA==", - "dev": true, - "requires": { - "loader-utils": "^1.0.2", - "lodash.clonedeep": "^4.5.0", - "when": "~3.6.x" - }, - "dependencies": { - "json5": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/json5/-/json5-1.0.1.tgz", - "integrity": "sha512-aKS4WQjPenRxiQsC93MNfjx+nbF4PAdYzmd/1JIj8HYzqfbu86beTuNgXDzPknWk0n0uARlyewZo4s++ES36Ow==", - "dev": true, - "requires": { - "minimist": "^1.2.0" - } - }, - "loader-utils": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/loader-utils/-/loader-utils-1.4.0.tgz", - "integrity": "sha512-qH0WSMBtn/oHuwjy/NucEgbx5dbxxnxup9s4PVXJUDHZBQY+s0NWA9rJf53RBnQZxfch7euUui7hpoAPvALZdA==", - "dev": true, - "requires": { - "big.js": "^5.2.2", - "emojis-list": "^3.0.0", - "json5": "^1.0.1" - } - } - } - }, - "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" - } - }, - "svgo": { - "version": "1.3.2", - "resolved": "https://registry.npmjs.org/svgo/-/svgo-1.3.2.tgz", - "integrity": "sha512-yhy/sQYxR5BkC98CY7o31VGsg014AKLEPxdfhora76l36hD9Rdy5NZA/Ocn6yayNPgSamYdtX2rFJdcv07AYVw==", - "dev": true, - "requires": { - "chalk": "^2.4.1", - "coa": "^2.0.2", - "css-select": "^2.0.0", - "css-select-base-adapter": "^0.1.1", - "css-tree": "1.0.0-alpha.37", - "csso": "^4.0.2", - "js-yaml": "^3.13.1", - "mkdirp": "~0.5.1", - "object.values": "^1.1.0", - "sax": "~1.2.4", - "stable": "^0.1.8", - "unquote": "~1.1.1", - "util.promisify": "~1.0.0" - } - }, - "symbol-observable": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/symbol-observable/-/symbol-observable-1.2.0.tgz", - "integrity": "sha512-e900nM8RRtGhlV36KGEU9k65K3mPb1WV70OdjfxlG2EAuM1noi/E/BaW/uMhL7bPEssK8QV57vN3esixjUvcXQ==", - "dev": true - }, - "tapable": { - "version": "1.1.3", - "resolved": "https://registry.npmjs.org/tapable/-/tapable-1.1.3.tgz", - "integrity": "sha512-4WK/bYZmj8xLr+HUCODHGF1ZFzsYffasLUgEiMBY4fgtltdO6B4WJtlSbPaDTLpYTcGVwM2qLnFTICEcNxs3kA==", - "dev": true - }, - "tar": { - "version": "6.1.11", - "resolved": "https://registry.npmjs.org/tar/-/tar-6.1.11.tgz", - "integrity": "sha512-an/KZQzQUkZCkuoAA64hM92X0Urb6VpRhAFllDzz44U2mcD5scmT3zBc4VgVpkugF580+DQn8eAFSyoQt0tznA==", - "dev": true, - "requires": { - "chownr": "^2.0.0", - "fs-minipass": "^2.0.0", - "minipass": "^3.0.0", - "minizlib": "^2.1.1", - "mkdirp": "^1.0.3", - "yallist": "^4.0.0" - }, - "dependencies": { - "chownr": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/chownr/-/chownr-2.0.0.tgz", - "integrity": "sha512-bIomtDF5KGpdogkLd9VspvFzk9KfpyyGlS8YFVZl7TGPBHL5snIOnxeshwVgPteQ9b4Eydl+pVbIyE1DcvCWgQ==", - "dev": true - }, - "minizlib": { - "version": "2.1.2", - "resolved": "https://registry.npmjs.org/minizlib/-/minizlib-2.1.2.tgz", - "integrity": "sha512-bAxsR8BVfj60DWXHE3u30oHzfl4G7khkSuPW+qvpd7jFRHm7dLxOjUk1EHACJ/hxLY8phGJ0YhYHZo7jil7Qdg==", - "dev": true, - "requires": { - "minipass": "^3.0.0", - "yallist": "^4.0.0" - } - }, - "mkdirp": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-1.0.4.tgz", - "integrity": "sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==", - "dev": true - } - } - }, - "terser": { - "version": "4.6.10", - "resolved": "https://registry.npmjs.org/terser/-/terser-4.6.10.tgz", - "integrity": "sha512-qbF/3UOo11Hggsbsqm2hPa6+L4w7bkr+09FNseEe8xrcVD3APGLFqE+Oz1ZKAxjYnFsj80rLOfgAtJ0LNJjtTA==", - "dev": true, - "requires": { - "commander": "^2.20.0", - "source-map": "~0.6.1", - "source-map-support": "~0.5.12" - }, - "dependencies": { - "source-map": { - "version": "0.6.1", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", - "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", - "dev": true - } - } - }, - "terser-webpack-plugin": { - "version": "2.3.5", - "resolved": "https://registry.npmjs.org/terser-webpack-plugin/-/terser-webpack-plugin-2.3.5.tgz", - "integrity": "sha512-WlWksUoq+E4+JlJ+h+U+QUzXpcsMSSNXkDy9lBVkSqDn1w23Gg29L/ary9GeJVYCGiNJJX7LnVc4bwL1N3/g1w==", - "dev": true, - "requires": { - "cacache": "^13.0.1", - "find-cache-dir": "^3.2.0", - "jest-worker": "^25.1.0", - "p-limit": "^2.2.2", - "schema-utils": "^2.6.4", - "serialize-javascript": "^2.1.2", - "source-map": "^0.6.1", - "terser": "^4.4.3", - "webpack-sources": "^1.4.3" - }, - "dependencies": { - "cacache": { - "version": "13.0.1", - "resolved": "https://registry.npmjs.org/cacache/-/cacache-13.0.1.tgz", - "integrity": "sha512-5ZvAxd05HDDU+y9BVvcqYu2LLXmPnQ0hW62h32g4xBTgL/MppR4/04NHfj/ycM2y6lmTnbw6HVi+1eN0Psba6w==", - "dev": true, - "requires": { - "chownr": "^1.1.2", - "figgy-pudding": "^3.5.1", - "fs-minipass": "^2.0.0", - "glob": "^7.1.4", - "graceful-fs": "^4.2.2", - "infer-owner": "^1.0.4", - "lru-cache": "^5.1.1", - "minipass": "^3.0.0", - "minipass-collect": "^1.0.2", - "minipass-flush": "^1.0.5", - "minipass-pipeline": "^1.2.2", - "mkdirp": "^0.5.1", - "move-concurrently": "^1.0.1", - "p-map": "^3.0.0", - "promise-inflight": "^1.0.1", - "rimraf": "^2.7.1", - "ssri": "^7.0.0", - "unique-filename": "^1.1.1" - } - }, - "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-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 - }, - "rimraf": { - "version": "2.7.1", - "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", - "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==", - "dev": true, - "requires": { - "glob": "^7.1.3" - } - }, - "source-map": { - "version": "0.6.1", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", - "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", - "dev": true - }, - "ssri": { - "version": "7.1.1", - "resolved": "https://registry.npmjs.org/ssri/-/ssri-7.1.1.tgz", - "integrity": "sha512-w+daCzXN89PseTL99MkA+fxJEcU3wfaE/ah0i0lnOlpG1CYLJ2ZjzEry68YBKfLs4JfoTShrTEsJkAZuNZ/stw==", - "dev": true, - "requires": { - "figgy-pudding": "^3.5.1", - "minipass": "^3.1.1" - } - } - } - }, - "through": { - "version": "2.3.8", - "resolved": "https://registry.npmjs.org/through/-/through-2.3.8.tgz", - "integrity": "sha1-DdTJ/6q8NXlgsbckEV1+Doai4fU=", - "dev": true - }, - "through2": { - "version": "2.0.5", - "resolved": "https://registry.npmjs.org/through2/-/through2-2.0.5.tgz", - "integrity": "sha512-/mrRod8xqpA+IHSLyGCQ2s8SPHiCDEeQJSep1jqLYeEUClOFG2Qsh+4FU6G9VeqpZnGW/Su8LQGc4YKni5rYSQ==", - "dev": true, - "requires": { - "readable-stream": "~2.3.6", - "xtend": "~4.0.1" - } - }, - "thunky": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/thunky/-/thunky-1.1.0.tgz", - "integrity": "sha512-eHY7nBftgThBqOyHGVN+l8gF0BucP09fMo0oO/Lb0w1OF80dJv+lDVpXG60WMQvkcxAkNybKsrEIE3ZtKGmPrA==", - "dev": true - }, - "timers-browserify": { - "version": "2.0.11", - "resolved": "https://registry.npmjs.org/timers-browserify/-/timers-browserify-2.0.11.tgz", - "integrity": "sha512-60aV6sgJ5YEbzUdn9c8kYGIqOubPoUdqQCul3SBAsRCZ40s6Y5cMcrW4dt3/k/EsbLVJNl9n6Vz3fTc+k2GeKQ==", - "dev": true, - "requires": { - "setimmediate": "^1.0.4" - } - }, - "timsort": { - "version": "0.3.0", - "resolved": "https://registry.npmjs.org/timsort/-/timsort-0.3.0.tgz", - "integrity": "sha1-QFQRqOfmM5/mTbmiNN4R3DHgK9Q=", - "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-arraybuffer": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/to-arraybuffer/-/to-arraybuffer-1.0.1.tgz", - "integrity": "sha1-fSKbH8xjfkZsoIEYCDanqr/4P0M=", - "dev": true - }, - "to-fast-properties": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/to-fast-properties/-/to-fast-properties-2.0.0.tgz", - "integrity": "sha1-3F5pjL0HkmW8c+A3doGk5Og/YW4=", - "dev": true - }, - "to-object-path": { - "version": "0.3.0", - "resolved": "https://registry.npmjs.org/to-object-path/-/to-object-path-0.3.0.tgz", - "integrity": "sha1-KXWIt7Dn4KwI4E5nL4XB9JmeF68=", - "dev": true, - "requires": { - "kind-of": "^3.0.2" - }, - "dependencies": { - "kind-of": { - "version": "3.2.2", - "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", - "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", - "dev": true, - "requires": { - "is-buffer": "^1.1.5" - } - } - } - }, - "to-regex": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/to-regex/-/to-regex-3.0.2.tgz", - "integrity": "sha512-FWtleNAtZ/Ki2qtqej2CXTOayOH9bHDQF+Q48VpWyDXjbYxA4Yz8iDB31zXOBUlOHHKidDbqGVrTUvQMPmBGBw==", - "dev": true, - "requires": { - "define-property": "^2.0.2", - "extend-shallow": "^3.0.2", - "regex-not": "^1.0.2", - "safe-regex": "^1.1.0" - } - }, - "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==", - "dev": true - }, - "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==", - "dev": true, - "requires": { - "psl": "^1.1.28", - "punycode": "^2.1.1" - } - }, - "tree-kill": { - "version": "1.2.2", - "resolved": "https://registry.npmjs.org/tree-kill/-/tree-kill-1.2.2.tgz", - "integrity": "sha512-L0Orpi8qGpRG//Nd+H90vFB+3iHnue1zSSGmNOOCh1GLJ7rUKVwV2HvijphGQS2UmhUZewS9VgvxYIdgr+fG1A==", - "dev": true - }, - "ts-node": { - "version": "8.3.0", - "resolved": "https://registry.npmjs.org/ts-node/-/ts-node-8.3.0.tgz", - "integrity": "sha512-dyNS/RqyVTDcmNM4NIBAeDMpsAdaQ+ojdf0GOLqE6nwJOgzEkdRNzJywhDfwnuvB10oa6NLVG1rUJQCpRN7qoQ==", - "dev": true, - "requires": { - "arg": "^4.1.0", - "diff": "^4.0.1", - "make-error": "^1.1.1", - "source-map-support": "^0.5.6", - "yn": "^3.0.0" - } - }, - "tslib": { - "version": "1.11.1", - "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.11.1.tgz", - "integrity": "sha512-aZW88SY8kQbU7gpV19lN24LtXh/yD4ZZg6qieAJDDg+YBsJcSmLGK9QpnUjAKVG/xefmvJGd1WUmfpT/g6AJGA==" - }, - "tslint": { - "version": "5.18.0", - "resolved": "https://registry.npmjs.org/tslint/-/tslint-5.18.0.tgz", - "integrity": "sha512-Q3kXkuDEijQ37nXZZLKErssQVnwCV/+23gFEMROi8IlbaBG6tXqLPQJ5Wjcyt/yHPKBC+hD5SzuGaMora+ZS6w==", - "dev": true, - "requires": { - "@babel/code-frame": "^7.0.0", - "builtin-modules": "^1.1.1", - "chalk": "^2.3.0", - "commander": "^2.12.1", - "diff": "^3.2.0", - "glob": "^7.1.1", - "js-yaml": "^3.13.1", - "minimatch": "^3.0.4", - "mkdirp": "^0.5.1", - "resolve": "^1.3.2", - "semver": "^5.3.0", - "tslib": "^1.8.0", - "tsutils": "^2.29.0" - }, - "dependencies": { - "diff": { - "version": "3.5.0", - "resolved": "https://registry.npmjs.org/diff/-/diff-3.5.0.tgz", - "integrity": "sha512-A46qtFgd+g7pDZinpnwiRJtxbC1hpgf0uzP3iG89scHk0AUC7A1TGxf5OiiOUv/JMZR8GOt8hL900hV0bOy5xA==", - "dev": true - }, - "semver": { - "version": "5.7.1", - "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", - "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", - "dev": true - } - } - }, - "tsutils": { - "version": "2.29.0", - "resolved": "https://registry.npmjs.org/tsutils/-/tsutils-2.29.0.tgz", - "integrity": "sha512-g5JVHCIJwzfISaXpXE1qvNalca5Jwob6FjI4AoPlqMusJ6ftFE7IkkFoMhVLRgK+4Kx3gkzb8UZK5t5yTTvEmA==", - "dev": true, - "requires": { - "tslib": "^1.8.1" - } - }, - "tty-browserify": { - "version": "0.0.0", - "resolved": "https://registry.npmjs.org/tty-browserify/-/tty-browserify-0.0.0.tgz", - "integrity": "sha1-oVe6QC2iTpv5V/mqadUk7tQpAaY=", - "dev": true - }, - "tunnel-agent": { - "version": "0.6.0", - "resolved": "https://registry.npmjs.org/tunnel-agent/-/tunnel-agent-0.6.0.tgz", - "integrity": "sha1-J6XeoGs2sEoKmWZ3SykIaPD8QP0=", - "dev": true, - "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=", - "dev": true - }, - "type-fest": { - "version": "0.11.0", - "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.11.0.tgz", - "integrity": "sha512-OdjXJxnCN1AvyLSzeKIgXTXxV+99ZuXl3Hpo9XpJAv9MBcHrrJOQ5kV7ypXOuQie+AmWG25hLbiKdwYTifzcfQ==", - "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==", - "dev": true, - "requires": { - "media-typer": "0.3.0", - "mime-types": "~2.1.24" - } - }, - "typedarray": { - "version": "0.0.6", - "resolved": "https://registry.npmjs.org/typedarray/-/typedarray-0.0.6.tgz", - "integrity": "sha1-hnrHTjhkGHsdPUfZlqeOxciDB3c=", - "dev": true - }, - "typescript": { - "version": "3.7.5", - "resolved": "https://registry.npmjs.org/typescript/-/typescript-3.7.5.tgz", - "integrity": "sha512-/P5lkRXkWHNAbcJIiHPfRoKqyd7bsyCma1hZNUGfn20qm64T6ZBlrzprymeu918H+mB/0rIg2gGK/BXkhhYgBw==", - "dev": true - }, - "ua-parser-js": { - "version": "0.7.31", - "resolved": "https://registry.npmjs.org/ua-parser-js/-/ua-parser-js-0.7.31.tgz", - "integrity": "sha512-qLK/Xe9E2uzmYI3qLeOmI0tEOt+TBBQyUIAh4aAgU05FVYzeZrKUdkAZfBNVGRaHVgV0TDkdEngJSw/SyQchkQ==", - "dev": true - }, - "unicode-canonical-property-names-ecmascript": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/unicode-canonical-property-names-ecmascript/-/unicode-canonical-property-names-ecmascript-1.0.4.tgz", - "integrity": "sha512-jDrNnXWHd4oHiTZnx/ZG7gtUTVp+gCcTTKr8L0HjlwphROEW3+Him+IpvC+xcJEFegapiMZyZe02CyuOnRmbnQ==", - "dev": true - }, - "unicode-match-property-ecmascript": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/unicode-match-property-ecmascript/-/unicode-match-property-ecmascript-1.0.4.tgz", - "integrity": "sha512-L4Qoh15vTfntsn4P1zqnHulG0LdXgjSO035fEpdtp6YxXhMT51Q6vgM5lYdG/5X3MjS+k/Y9Xw4SFCY9IkR0rg==", - "dev": true, - "requires": { - "unicode-canonical-property-names-ecmascript": "^1.0.4", - "unicode-property-aliases-ecmascript": "^1.0.4" - } - }, - "unicode-match-property-value-ecmascript": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/unicode-match-property-value-ecmascript/-/unicode-match-property-value-ecmascript-1.2.0.tgz", - "integrity": "sha512-wjuQHGQVofmSJv1uVISKLE5zO2rNGzM/KCYZch/QQvez7C1hUhBIuZ701fYXExuufJFMPhv2SyL8CyoIfMLbIQ==", - "dev": true - }, - "unicode-property-aliases-ecmascript": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/unicode-property-aliases-ecmascript/-/unicode-property-aliases-ecmascript-1.1.0.tgz", - "integrity": "sha512-PqSoPh/pWetQ2phoj5RLiaqIk4kCNwoV3CI+LfGmWLKI3rE3kl1h59XpX2BjgDrmbxD9ARtQobPGU1SguCYuQg==", - "dev": true - }, - "union-value": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/union-value/-/union-value-1.0.1.tgz", - "integrity": "sha512-tJfXmxMeWYnczCVs7XAEvIV7ieppALdyepWMkHkwciRpZraG/xwT+s2JN8+pr1+8jCRf80FFzvr+MpQeeoF4Xg==", - "dev": true, - "requires": { - "arr-union": "^3.1.0", - "get-value": "^2.0.6", - "is-extendable": "^0.1.1", - "set-value": "^2.0.1" - } - }, - "uniq": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/uniq/-/uniq-1.0.1.tgz", - "integrity": "sha1-sxxa6CVIRKOoKBVBzisEuGWnNP8=", - "dev": true - }, - "uniqs": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/uniqs/-/uniqs-2.0.0.tgz", - "integrity": "sha1-/+3ks2slKQaW5uFl1KWe25mOawI=", - "dev": true - }, - "unique-filename": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/unique-filename/-/unique-filename-1.1.1.tgz", - "integrity": "sha512-Vmp0jIp2ln35UTXuryvjzkjGdRyf9b2lTXuSYUiPmzRcl3FDtYqAwOnTJkAngD9SWhnoJzDbTKwaOrZ+STtxNQ==", - "dev": true, - "requires": { - "unique-slug": "^2.0.0" - } - }, - "unique-slug": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/unique-slug/-/unique-slug-2.0.2.tgz", - "integrity": "sha512-zoWr9ObaxALD3DOPfjPSqxt4fnZiWblxHIgeWqW8x7UqDzEtHEQLzji2cuJYQFCU6KmoJikOYAZlrTHHebjx2w==", - "dev": true, - "requires": { - "imurmurhash": "^0.1.4" - } - }, - "universal-analytics": { - "version": "0.4.20", - "resolved": "https://registry.npmjs.org/universal-analytics/-/universal-analytics-0.4.20.tgz", - "integrity": "sha512-gE91dtMvNkjO+kWsPstHRtSwHXz0l2axqptGYp5ceg4MsuurloM0PU3pdOfpb5zBXUvyjT4PwhWK2m39uczZuw==", - "dev": true, - "requires": { - "debug": "^3.0.0", - "request": "^2.88.0", - "uuid": "^3.0.0" - }, - "dependencies": { - "debug": { - "version": "3.2.6", - "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.6.tgz", - "integrity": "sha512-mel+jf7nrtEl5Pn1Qx46zARXKDpBbvzezse7p7LqINmdoIk8PYP5SySaxEmYv6TZ0JyEKA1hsCId6DIhgITtWQ==", - "dev": true, - "requires": { - "ms": "^2.1.1" - } - }, - "uuid": { - "version": "3.4.0", - "resolved": "https://registry.npmjs.org/uuid/-/uuid-3.4.0.tgz", - "integrity": "sha512-HjSDRw6gZE5JMggctHBcjVak08+KEVhSIiDzFnT9S9aegmp85S/bReBVTb4QTFaRNptJ9kuYaNhnbNEOkbKb/A==", - "dev": true - } - } - }, - "universalify": { - "version": "0.1.2", - "resolved": "https://registry.npmjs.org/universalify/-/universalify-0.1.2.tgz", - "integrity": "sha512-rBJeI5CXAlmy1pV+617WB9J63U6XcazHHF2f2dbJix4XzpUF0RS3Zbj0FGIOCAva5P/d/GBOYaACQ1w+0azUkg==", - "dev": true - }, - "unpipe": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", - "integrity": "sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw=", - "dev": true - }, - "unquote": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/unquote/-/unquote-1.1.1.tgz", - "integrity": "sha1-j97XMk7G6IoP+LkF58CYzcCG1UQ=", - "dev": true - }, - "unset-value": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/unset-value/-/unset-value-1.0.0.tgz", - "integrity": "sha1-g3aHP30jNRef+x5vw6jtDfyKtVk=", - "dev": true, - "requires": { - "has-value": "^0.3.1", - "isobject": "^3.0.0" - }, - "dependencies": { - "has-value": { - "version": "0.3.1", - "resolved": "https://registry.npmjs.org/has-value/-/has-value-0.3.1.tgz", - "integrity": "sha1-ex9YutpiyoJ+wKIHgCVlSEWZXh8=", - "dev": true, - "requires": { - "get-value": "^2.0.3", - "has-values": "^0.1.4", - "isobject": "^2.0.0" - }, - "dependencies": { - "isobject": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/isobject/-/isobject-2.1.0.tgz", - "integrity": "sha1-8GVWEJaj8dou9GJy+BXIQNh+DIk=", - "dev": true, - "requires": { - "isarray": "1.0.0" - } - } - } - }, - "has-values": { - "version": "0.1.4", - "resolved": "https://registry.npmjs.org/has-values/-/has-values-0.1.4.tgz", - "integrity": "sha1-bWHeldkd/Km5oCCJrThL/49it3E=", - "dev": true - } - } - }, - "upath": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/upath/-/upath-1.2.0.tgz", - "integrity": "sha512-aZwGpamFO61g3OlfT7OQCHqhGnW43ieH9WZeP7QxN/G/jS4jfqUkZxoryvJgVPEcrl5NL/ggHsSmLMHuH64Lhg==", - "dev": true - }, - "uri-js": { - "version": "4.2.2", - "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.2.2.tgz", - "integrity": "sha512-KY9Frmirql91X2Qgjry0Wd4Y+YTdrdZheS8TFwvkbLWf/G5KNJDCh6pKL5OZctEW4+0Baa5idK2ZQuELRwPznQ==", - "dev": true, - "requires": { - "punycode": "^2.1.0" - } - }, - "urix": { - "version": "0.1.0", - "resolved": "https://registry.npmjs.org/urix/-/urix-0.1.0.tgz", - "integrity": "sha1-2pN/emLiH+wf0Y1Js1wpNQZ6bHI=", - "dev": true - }, - "url": { - "version": "0.11.0", - "resolved": "https://registry.npmjs.org/url/-/url-0.11.0.tgz", - "integrity": "sha1-ODjpfPxgUh63PFJajlW/3Z4uKPE=", - "dev": true, - "requires": { - "punycode": "1.3.2", - "querystring": "0.2.0" - }, - "dependencies": { - "punycode": { - "version": "1.3.2", - "resolved": "https://registry.npmjs.org/punycode/-/punycode-1.3.2.tgz", - "integrity": "sha1-llOgNvt8HuQjQvIyXM7v6jkmxI0=", - "dev": true - } - } - }, - "url-parse": { - "version": "1.5.10", - "resolved": "https://registry.npmjs.org/url-parse/-/url-parse-1.5.10.tgz", - "integrity": "sha512-WypcfiRhfeUP9vvF0j6rw0J3hrWrw6iZv3+22h6iRMJ/8z1Tj6XfLP4DsUix5MhMPnXpiHDoKyoZ/bdCkwBCiQ==", - "dev": true, - "requires": { - "querystringify": "^2.1.1", - "requires-port": "^1.0.0" - } - }, - "use": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/use/-/use-3.1.1.tgz", - "integrity": "sha512-cwESVXlO3url9YWlFW/TA9cshCEhtu7IKJ/p5soJ/gGpj7vbvFrAY/eIioQ6Dw23KjZhYgiIo8HOs1nQ2vr/oQ==", - "dev": true - }, - "util": { - "version": "0.11.1", - "resolved": "https://registry.npmjs.org/util/-/util-0.11.1.tgz", - "integrity": "sha512-HShAsny+zS2TZfaXxD9tYj4HQGlBezXZMZuM/S5PKLLoZkShZiGk9o5CzukI1LVHZvjdvZ2Sj1aW/Ndn2NB/HQ==", - "dev": true, - "requires": { - "inherits": "2.0.3" - }, - "dependencies": { - "inherits": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", - "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=", - "dev": true - } - } - }, - "util-deprecate": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", - "integrity": "sha1-RQ1Nyfpw3nMnYvvS1KKJgUGaDM8=", - "dev": true - }, - "util-promisify": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/util-promisify/-/util-promisify-2.1.0.tgz", - "integrity": "sha1-PCI2R2xNMsX/PEcAKt18E7moKlM=", - "dev": true, - "requires": { - "object.getownpropertydescriptors": "^2.0.3" - } - }, - "util.promisify": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/util.promisify/-/util.promisify-1.0.1.tgz", - "integrity": "sha512-g9JpC/3He3bm38zsLupWryXHoEcS22YHthuPQSJdMy6KNrzIRzWqcsHzD/WUnqe45whVou4VIsPew37DoXWNrA==", - "dev": true, - "requires": { - "define-properties": "^1.1.3", - "es-abstract": "^1.17.2", - "has-symbols": "^1.0.1", - "object.getownpropertydescriptors": "^2.1.0" - } - }, - "utils-merge": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz", - "integrity": "sha1-n5VxD1CiZ5R7LMwSR0HBAoQn5xM=", - "dev": true - }, - "uuid": { - "version": "8.0.0", - "resolved": "https://registry.npmjs.org/uuid/-/uuid-8.0.0.tgz", - "integrity": "sha512-jOXGuXZAWdsTH7eZLtyXMqUb9EcWMGZNbL9YcGBJl4MH4nrxHmZJhEHvyLFrkxo+28uLb/NYRcStH48fnD0Vzw==" - }, - "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==", - "dev": true, - "requires": { - "spdx-correct": "^3.0.0", - "spdx-expression-parse": "^3.0.0" - } - }, - "validate-npm-package-name": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/validate-npm-package-name/-/validate-npm-package-name-3.0.0.tgz", - "integrity": "sha1-X6kS2B630MdK/BQN5zF/DKffQ34=", - "dev": true, - "requires": { - "builtins": "^1.0.3" - } - }, - "vary": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz", - "integrity": "sha1-IpnwLG3tMNSllhsLn3RSShj2NPw=", - "dev": true - }, - "vendors": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/vendors/-/vendors-1.0.4.tgz", - "integrity": "sha512-/juG65kTL4Cy2su4P8HjtkTxk6VmJDiOPBufWniqQ6wknac6jNiXS9vU+hO3wgusiyqWlzTbVHi0dyJqRONg3w==", - "dev": true - }, - "verror": { - "version": "1.10.0", - "resolved": "https://registry.npmjs.org/verror/-/verror-1.10.0.tgz", - "integrity": "sha1-OhBcoXBTr1XW4nDB+CiGguGNpAA=", - "dev": true, - "requires": { - "assert-plus": "^1.0.0", - "core-util-is": "1.0.2", - "extsprintf": "^1.2.0" - } - }, - "vm-browserify": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/vm-browserify/-/vm-browserify-1.1.2.tgz", - "integrity": "sha512-2ham8XPWTONajOR0ohOKOHXkm3+gaBmGut3SRuu75xLd/RRaY6vqgh8NBYYk7+RW3u5AtzPQZG8F10LHkl0lAQ==", - "dev": true - }, - "void-elements": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/void-elements/-/void-elements-2.0.1.tgz", - "integrity": "sha1-wGavtYK7HLQSjWDqkjkulNXp2+w=", - "dev": true - }, - "watchpack": { - "version": "1.6.1", - "resolved": "https://registry.npmjs.org/watchpack/-/watchpack-1.6.1.tgz", - "integrity": "sha512-+IF9hfUFOrYOOaKyfaI7h7dquUIOgyEMoQMLA7OP5FxegKA2+XdXThAZ9TU2kucfhDH7rfMHs1oPYziVGWRnZA==", - "dev": true, - "requires": { - "chokidar": "^2.1.8", - "graceful-fs": "^4.1.2", - "neo-async": "^2.5.0" - }, - "dependencies": { - "anymatch": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-2.0.0.tgz", - "integrity": "sha512-5teOsQWABXHHBFP9y3skS5P3d/WfWXpv3FUpy+LorMrNYaT9pI4oLMQX7jzQ2KklNpGpWHzdCXTDT2Y3XGlZBw==", - "dev": true, - "requires": { - "micromatch": "^3.1.4", - "normalize-path": "^2.1.1" - }, - "dependencies": { - "normalize-path": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-2.1.1.tgz", - "integrity": "sha1-GrKLVW4Zg2Oowab35vogE3/mrtk=", - "dev": true, - "requires": { - "remove-trailing-separator": "^1.0.1" - } - } - } - }, - "binary-extensions": { - "version": "1.13.1", - "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-1.13.1.tgz", - "integrity": "sha512-Un7MIEDdUC5gNpcGDV97op1Ywk748MpHcFTHoYs6qnj1Z3j7I53VG3nwZhKzoBZmbdRNnb6WRdFlwl7tSDuZGw==", - "dev": true - }, - "braces": { - "version": "2.3.2", - "resolved": "https://registry.npmjs.org/braces/-/braces-2.3.2.tgz", - "integrity": "sha512-aNdbnj9P8PjdXU4ybaWLK2IF3jc/EoDYbC7AazW6to3TRsfXxscC9UXOB5iDiEQrkyIbWp2SLQda4+QAa7nc3w==", - "dev": true, - "requires": { - "arr-flatten": "^1.1.0", - "array-unique": "^0.3.2", - "extend-shallow": "^2.0.1", - "fill-range": "^4.0.0", - "isobject": "^3.0.1", - "repeat-element": "^1.1.2", - "snapdragon": "^0.8.1", - "snapdragon-node": "^2.0.1", - "split-string": "^3.0.2", - "to-regex": "^3.0.1" - } - }, - "chokidar": { - "version": "2.1.8", - "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-2.1.8.tgz", - "integrity": "sha512-ZmZUazfOzf0Nve7duiCKD23PFSCs4JPoYyccjUFF3aQkQadqBhfzhjkwBH2mNOG9cTBwhamM37EIsIkZw3nRgg==", - "dev": true, - "requires": { - "anymatch": "^2.0.0", - "async-each": "^1.0.1", - "braces": "^2.3.2", - "fsevents": "^1.2.7", - "glob-parent": "^3.1.0", - "inherits": "^2.0.3", - "is-binary-path": "^1.0.0", - "is-glob": "^4.0.0", - "normalize-path": "^3.0.0", - "path-is-absolute": "^1.0.0", - "readdirp": "^2.2.1", - "upath": "^1.1.1" - } - }, - "extend-shallow": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", - "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", - "dev": true, - "requires": { - "is-extendable": "^0.1.0" - } - }, - "fill-range": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-4.0.0.tgz", - "integrity": "sha1-1USBHUKPmOsGpj3EAtJAPDKMOPc=", - "dev": true, - "requires": { - "extend-shallow": "^2.0.1", - "is-number": "^3.0.0", - "repeat-string": "^1.6.1", - "to-regex-range": "^2.1.0" - } - }, - "fsevents": { - "version": "1.2.13", - "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-1.2.13.tgz", - "integrity": "sha512-oWb1Z6mkHIskLzEJ/XWX0srkpkTQ7vaopMQkyaEIoq0fmtFVxOthb8cCxeT+p3ynTdkk/RZwbgG4brR5BeWECw==", - "dev": true, - "optional": true, - "requires": { - "bindings": "^1.5.0", - "nan": "^2.12.1" - } - }, - "is-binary-path": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-1.0.1.tgz", - "integrity": "sha1-dfFmQrSA8YenEcgUFh/TpKdlWJg=", - "dev": true, - "requires": { - "binary-extensions": "^1.0.0" - } - }, - "is-number": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/is-number/-/is-number-3.0.0.tgz", - "integrity": "sha1-JP1iAaR4LPUFYcgQJ2r8fRLXEZU=", - "dev": true, - "requires": { - "kind-of": "^3.0.2" - } - }, - "kind-of": { - "version": "3.2.2", - "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", - "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", - "dev": true, - "requires": { - "is-buffer": "^1.1.5" - } - }, - "readdirp": { - "version": "2.2.1", - "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-2.2.1.tgz", - "integrity": "sha512-1JU/8q+VgFZyxwrJ+SVIOsh+KywWGpds3NTqikiKpDMZWScmAYyKIgqkO+ARvNWJfXeXR1zxz7aHF4u4CyH6vQ==", - "dev": true, - "requires": { - "graceful-fs": "^4.1.11", - "micromatch": "^3.1.10", - "readable-stream": "^2.0.2" - } - }, - "to-regex-range": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-2.1.1.tgz", - "integrity": "sha1-fIDBe53+vlmeJzZ+DU3VWQFB2zg=", - "dev": true, - "requires": { - "is-number": "^3.0.0", - "repeat-string": "^1.6.1" - } - } - } - }, - "wbuf": { - "version": "1.7.3", - "resolved": "https://registry.npmjs.org/wbuf/-/wbuf-1.7.3.tgz", - "integrity": "sha512-O84QOnr0icsbFGLS0O3bI5FswxzRr8/gHwWkDlQFskhSPryQXvrTMxjxGP4+iWYoauLoBvfDpkrOauZ+0iZpDA==", - "dev": true, - "requires": { - "minimalistic-assert": "^1.0.0" - } - }, - "wcwidth": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/wcwidth/-/wcwidth-1.0.1.tgz", - "integrity": "sha1-8LDc+RW8X/FSivrbLA4XtTLaL+g=", - "dev": true, - "requires": { - "defaults": "^1.0.3" - } - }, - "webdriver-js-extender": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/webdriver-js-extender/-/webdriver-js-extender-2.1.0.tgz", - "integrity": "sha512-lcUKrjbBfCK6MNsh7xaY2UAUmZwe+/ib03AjVOpFobX4O7+83BUveSrLfU0Qsyb1DaKJdQRbuU+kM9aZ6QUhiQ==", - "dev": true, - "requires": { - "@types/selenium-webdriver": "^3.0.0", - "selenium-webdriver": "^3.0.1" - } - }, - "webpack": { - "version": "4.42.0", - "resolved": "https://registry.npmjs.org/webpack/-/webpack-4.42.0.tgz", - "integrity": "sha512-EzJRHvwQyBiYrYqhyjW9AqM90dE4+s1/XtCfn7uWg6cS72zH+2VPFAlsnW0+W0cDi0XRjNKUMoJtpSi50+Ph6w==", - "dev": true, - "requires": { - "@webassemblyjs/ast": "1.8.5", - "@webassemblyjs/helper-module-context": "1.8.5", - "@webassemblyjs/wasm-edit": "1.8.5", - "@webassemblyjs/wasm-parser": "1.8.5", - "acorn": "^6.2.1", - "ajv": "^6.10.2", - "ajv-keywords": "^3.4.1", - "chrome-trace-event": "^1.0.2", - "enhanced-resolve": "^4.1.0", - "eslint-scope": "^4.0.3", - "json-parse-better-errors": "^1.0.2", - "loader-runner": "^2.4.0", - "loader-utils": "^1.2.3", - "memory-fs": "^0.4.1", - "micromatch": "^3.1.10", - "mkdirp": "^0.5.1", - "neo-async": "^2.6.1", - "node-libs-browser": "^2.2.1", - "schema-utils": "^1.0.0", - "tapable": "^1.1.3", - "terser-webpack-plugin": "^1.4.3", - "watchpack": "^1.6.0", - "webpack-sources": "^1.4.1" - }, - "dependencies": { - "cacache": { - "version": "12.0.4", - "resolved": "https://registry.npmjs.org/cacache/-/cacache-12.0.4.tgz", - "integrity": "sha512-a0tMB40oefvuInr4Cwb3GerbL9xTj1D5yg0T5xrjGCGyfvbxseIXX7BAO/u/hIXdafzOI5JC3wDwHyf24buOAQ==", - "dev": true, - "requires": { - "bluebird": "^3.5.5", - "chownr": "^1.1.1", - "figgy-pudding": "^3.5.1", - "glob": "^7.1.4", - "graceful-fs": "^4.1.15", - "infer-owner": "^1.0.3", - "lru-cache": "^5.1.1", - "mississippi": "^3.0.0", - "mkdirp": "^0.5.1", - "move-concurrently": "^1.0.1", - "promise-inflight": "^1.0.1", - "rimraf": "^2.6.3", - "ssri": "^6.0.1", - "unique-filename": "^1.1.1", - "y18n": "^4.0.0" - } - }, - "find-cache-dir": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/find-cache-dir/-/find-cache-dir-2.1.0.tgz", - "integrity": "sha512-Tq6PixE0w/VMFfCgbONnkiQIVol/JJL7nRMi20fqzA4NRs9AfeqMGeRdPi3wIhYkxjeBaWh2rxwapn5Tu3IqOQ==", - "dev": true, - "requires": { - "commondir": "^1.0.1", - "make-dir": "^2.0.0", - "pkg-dir": "^3.0.0" - } - }, - "is-wsl": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/is-wsl/-/is-wsl-1.1.0.tgz", - "integrity": "sha1-HxbkqiKwTRM2tmGIpmrzxgDDpm0=", - "dev": true - }, - "json5": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/json5/-/json5-1.0.1.tgz", - "integrity": "sha512-aKS4WQjPenRxiQsC93MNfjx+nbF4PAdYzmd/1JIj8HYzqfbu86beTuNgXDzPknWk0n0uARlyewZo4s++ES36Ow==", - "dev": true, - "requires": { - "minimist": "^1.2.0" - } - }, - "loader-utils": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/loader-utils/-/loader-utils-1.4.0.tgz", - "integrity": "sha512-qH0WSMBtn/oHuwjy/NucEgbx5dbxxnxup9s4PVXJUDHZBQY+s0NWA9rJf53RBnQZxfch7euUui7hpoAPvALZdA==", - "dev": true, - "requires": { - "big.js": "^5.2.2", - "emojis-list": "^3.0.0", - "json5": "^1.0.1" - } - }, - "memory-fs": { - "version": "0.4.1", - "resolved": "https://registry.npmjs.org/memory-fs/-/memory-fs-0.4.1.tgz", - "integrity": "sha1-OpoguEYlI+RHz7x+i7gO1me/xVI=", - "dev": true, - "requires": { - "errno": "^0.1.3", - "readable-stream": "^2.0.1" - } - }, - "rimraf": { - "version": "2.7.1", - "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", - "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==", - "dev": true, - "requires": { - "glob": "^7.1.3" - } - }, - "schema-utils": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-1.0.0.tgz", - "integrity": "sha512-i27Mic4KovM/lnGsy8whRCHhc7VicJajAjTrYg11K9zfZXnYIt4k5F+kZkwjnrhKzLic/HLU4j11mjsz2G/75g==", - "dev": true, - "requires": { - "ajv": "^6.1.0", - "ajv-errors": "^1.0.0", - "ajv-keywords": "^3.1.0" - } - }, - "source-map": { - "version": "0.6.1", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", - "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", - "dev": true - }, - "ssri": { - "version": "6.0.2", - "resolved": "https://registry.npmjs.org/ssri/-/ssri-6.0.2.tgz", - "integrity": "sha512-cepbSq/neFK7xB6A50KHN0xHDotYzq58wWCa5LeWqnPrHG8GzfEjO/4O8kpmcGW+oaxkvhEJCWgbgNk4/ZV93Q==", - "dev": true, - "requires": { - "figgy-pudding": "^3.5.1" - } - }, - "terser-webpack-plugin": { - "version": "1.4.3", - "resolved": "https://registry.npmjs.org/terser-webpack-plugin/-/terser-webpack-plugin-1.4.3.tgz", - "integrity": "sha512-QMxecFz/gHQwteWwSo5nTc6UaICqN1bMedC5sMtUc7y3Ha3Q8y6ZO0iCR8pq4RJC8Hjf0FEPEHZqcMB/+DFCrA==", - "dev": true, - "requires": { - "cacache": "^12.0.2", - "find-cache-dir": "^2.1.0", - "is-wsl": "^1.1.0", - "schema-utils": "^1.0.0", - "serialize-javascript": "^2.1.2", - "source-map": "^0.6.1", - "terser": "^4.1.2", - "webpack-sources": "^1.4.0", - "worker-farm": "^1.7.0" - } - } - } - }, - "webpack-dev-middleware": { - "version": "3.7.2", - "resolved": "https://registry.npmjs.org/webpack-dev-middleware/-/webpack-dev-middleware-3.7.2.tgz", - "integrity": "sha512-1xC42LxbYoqLNAhV6YzTYacicgMZQTqRd27Sim9wn5hJrX3I5nxYy1SxSd4+gjUFsz1dQFj+yEe6zEVmSkeJjw==", - "dev": true, - "requires": { - "memory-fs": "^0.4.1", - "mime": "^2.4.4", - "mkdirp": "^0.5.1", - "range-parser": "^1.2.1", - "webpack-log": "^2.0.0" - }, - "dependencies": { - "memory-fs": { - "version": "0.4.1", - "resolved": "https://registry.npmjs.org/memory-fs/-/memory-fs-0.4.1.tgz", - "integrity": "sha1-OpoguEYlI+RHz7x+i7gO1me/xVI=", - "dev": true, - "requires": { - "errno": "^0.1.3", - "readable-stream": "^2.0.1" - } - }, - "mime": { - "version": "2.4.5", - "resolved": "https://registry.npmjs.org/mime/-/mime-2.4.5.tgz", - "integrity": "sha512-3hQhEUF027BuxZjQA3s7rIv/7VCQPa27hN9u9g87sEkWaKwQPuXOkVKtOeiyUrnWqTDiOs8Ed2rwg733mB0R5w==", - "dev": true - } - } - }, - "webpack-dev-server": { - "version": "3.10.3", - "resolved": "https://registry.npmjs.org/webpack-dev-server/-/webpack-dev-server-3.10.3.tgz", - "integrity": "sha512-e4nWev8YzEVNdOMcNzNeCN947sWJNd43E5XvsJzbAL08kGc2frm1tQ32hTJslRS+H65LCb/AaUCYU7fjHCpDeQ==", - "dev": true, - "requires": { - "ansi-html": "0.0.7", - "bonjour": "^3.5.0", - "chokidar": "^2.1.8", - "compression": "^1.7.4", - "connect-history-api-fallback": "^1.6.0", - "debug": "^4.1.1", - "del": "^4.1.1", - "express": "^4.17.1", - "html-entities": "^1.2.1", - "http-proxy-middleware": "0.19.1", - "import-local": "^2.0.0", - "internal-ip": "^4.3.0", - "ip": "^1.1.5", - "is-absolute-url": "^3.0.3", - "killable": "^1.0.1", - "loglevel": "^1.6.6", - "opn": "^5.5.0", - "p-retry": "^3.0.1", - "portfinder": "^1.0.25", - "schema-utils": "^1.0.0", - "selfsigned": "^1.10.7", - "semver": "^6.3.0", - "serve-index": "^1.9.1", - "sockjs": "0.3.19", - "sockjs-client": "1.4.0", - "spdy": "^4.0.1", - "strip-ansi": "^3.0.1", - "supports-color": "^6.1.0", - "url": "^0.11.0", - "webpack-dev-middleware": "^3.7.2", - "webpack-log": "^2.0.0", - "ws": "^6.2.1", - "yargs": "12.0.5" - }, - "dependencies": { - "anymatch": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-2.0.0.tgz", - "integrity": "sha512-5teOsQWABXHHBFP9y3skS5P3d/WfWXpv3FUpy+LorMrNYaT9pI4oLMQX7jzQ2KklNpGpWHzdCXTDT2Y3XGlZBw==", - "dev": true, - "requires": { - "micromatch": "^3.1.4", - "normalize-path": "^2.1.1" - }, - "dependencies": { - "normalize-path": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-2.1.1.tgz", - "integrity": "sha1-GrKLVW4Zg2Oowab35vogE3/mrtk=", - "dev": true, - "requires": { - "remove-trailing-separator": "^1.0.1" - } - } - } - }, - "binary-extensions": { - "version": "1.13.1", - "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-1.13.1.tgz", - "integrity": "sha512-Un7MIEDdUC5gNpcGDV97op1Ywk748MpHcFTHoYs6qnj1Z3j7I53VG3nwZhKzoBZmbdRNnb6WRdFlwl7tSDuZGw==", - "dev": true - }, - "braces": { - "version": "2.3.2", - "resolved": "https://registry.npmjs.org/braces/-/braces-2.3.2.tgz", - "integrity": "sha512-aNdbnj9P8PjdXU4ybaWLK2IF3jc/EoDYbC7AazW6to3TRsfXxscC9UXOB5iDiEQrkyIbWp2SLQda4+QAa7nc3w==", - "dev": true, - "requires": { - "arr-flatten": "^1.1.0", - "array-unique": "^0.3.2", - "extend-shallow": "^2.0.1", - "fill-range": "^4.0.0", - "isobject": "^3.0.1", - "repeat-element": "^1.1.2", - "snapdragon": "^0.8.1", - "snapdragon-node": "^2.0.1", - "split-string": "^3.0.2", - "to-regex": "^3.0.1" - } - }, - "chokidar": { - "version": "2.1.8", - "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-2.1.8.tgz", - "integrity": "sha512-ZmZUazfOzf0Nve7duiCKD23PFSCs4JPoYyccjUFF3aQkQadqBhfzhjkwBH2mNOG9cTBwhamM37EIsIkZw3nRgg==", - "dev": true, - "requires": { - "anymatch": "^2.0.0", - "async-each": "^1.0.1", - "braces": "^2.3.2", - "fsevents": "^1.2.7", - "glob-parent": "^3.1.0", - "inherits": "^2.0.3", - "is-binary-path": "^1.0.0", - "is-glob": "^4.0.0", - "normalize-path": "^3.0.0", - "path-is-absolute": "^1.0.0", - "readdirp": "^2.2.1", - "upath": "^1.1.1" - } - }, - "extend-shallow": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", - "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", - "dev": true, - "requires": { - "is-extendable": "^0.1.0" - } - }, - "fill-range": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-4.0.0.tgz", - "integrity": "sha1-1USBHUKPmOsGpj3EAtJAPDKMOPc=", - "dev": true, - "requires": { - "extend-shallow": "^2.0.1", - "is-number": "^3.0.0", - "repeat-string": "^1.6.1", - "to-regex-range": "^2.1.0" - } - }, - "fsevents": { - "version": "1.2.13", - "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-1.2.13.tgz", - "integrity": "sha512-oWb1Z6mkHIskLzEJ/XWX0srkpkTQ7vaopMQkyaEIoq0fmtFVxOthb8cCxeT+p3ynTdkk/RZwbgG4brR5BeWECw==", - "dev": true, - "optional": true, - "requires": { - "bindings": "^1.5.0", - "nan": "^2.12.1" - } - }, - "is-absolute-url": { - "version": "3.0.3", - "resolved": "https://registry.npmjs.org/is-absolute-url/-/is-absolute-url-3.0.3.tgz", - "integrity": "sha512-opmNIX7uFnS96NtPmhWQgQx6/NYFgsUXYMllcfzwWKUMwfo8kku1TvE6hkNcH+Q1ts5cMVrsY7j0bxXQDciu9Q==", - "dev": true - }, - "is-binary-path": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-1.0.1.tgz", - "integrity": "sha1-dfFmQrSA8YenEcgUFh/TpKdlWJg=", - "dev": true, - "requires": { - "binary-extensions": "^1.0.0" - } - }, - "is-number": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/is-number/-/is-number-3.0.0.tgz", - "integrity": "sha1-JP1iAaR4LPUFYcgQJ2r8fRLXEZU=", - "dev": true, - "requires": { - "kind-of": "^3.0.2" - } - }, - "kind-of": { - "version": "3.2.2", - "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", - "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", - "dev": true, - "requires": { - "is-buffer": "^1.1.5" - } - }, - "readdirp": { - "version": "2.2.1", - "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-2.2.1.tgz", - "integrity": "sha512-1JU/8q+VgFZyxwrJ+SVIOsh+KywWGpds3NTqikiKpDMZWScmAYyKIgqkO+ARvNWJfXeXR1zxz7aHF4u4CyH6vQ==", - "dev": true, - "requires": { - "graceful-fs": "^4.1.11", - "micromatch": "^3.1.10", - "readable-stream": "^2.0.2" - } - }, - "schema-utils": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-1.0.0.tgz", - "integrity": "sha512-i27Mic4KovM/lnGsy8whRCHhc7VicJajAjTrYg11K9zfZXnYIt4k5F+kZkwjnrhKzLic/HLU4j11mjsz2G/75g==", - "dev": true, - "requires": { - "ajv": "^6.1.0", - "ajv-errors": "^1.0.0", - "ajv-keywords": "^3.1.0" - } - }, - "supports-color": { - "version": "6.1.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-6.1.0.tgz", - "integrity": "sha512-qe1jfm1Mg7Nq/NSh6XE24gPXROEVsWHxC1LIx//XNlD9iw7YZQGjZNjYN7xGaEG6iKdA8EtNFW6R0gjnVXp+wQ==", - "dev": true, - "requires": { - "has-flag": "^3.0.0" - } - }, - "to-regex-range": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-2.1.1.tgz", - "integrity": "sha1-fIDBe53+vlmeJzZ+DU3VWQFB2zg=", - "dev": true, - "requires": { - "is-number": "^3.0.0", - "repeat-string": "^1.6.1" - } - } - } - }, - "webpack-log": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/webpack-log/-/webpack-log-2.0.0.tgz", - "integrity": "sha512-cX8G2vR/85UYG59FgkoMamwHUIkSSlV3bBMRsbxVXVUk2j6NleCKjQ/WE9eYg9WY4w25O9w8wKP4rzNZFmUcUg==", - "dev": true, - "requires": { - "ansi-colors": "^3.0.0", - "uuid": "^3.3.2" - }, - "dependencies": { - "uuid": { - "version": "3.4.0", - "resolved": "https://registry.npmjs.org/uuid/-/uuid-3.4.0.tgz", - "integrity": "sha512-HjSDRw6gZE5JMggctHBcjVak08+KEVhSIiDzFnT9S9aegmp85S/bReBVTb4QTFaRNptJ9kuYaNhnbNEOkbKb/A==", - "dev": true - } - } - }, - "webpack-merge": { - "version": "4.2.2", - "resolved": "https://registry.npmjs.org/webpack-merge/-/webpack-merge-4.2.2.tgz", - "integrity": "sha512-TUE1UGoTX2Cd42j3krGYqObZbOD+xF7u28WB7tfUordytSjbWTIjK/8V0amkBfTYN4/pB/GIDlJZZ657BGG19g==", - "dev": true, - "requires": { - "lodash": "^4.17.15" - } - }, - "webpack-sources": { - "version": "1.4.3", - "resolved": "https://registry.npmjs.org/webpack-sources/-/webpack-sources-1.4.3.tgz", - "integrity": "sha512-lgTS3Xhv1lCOKo7SA5TjKXMjpSM4sBjNV5+q2bqesbSPs5FjGmU6jjtBSkX9b4qW87vDIsCIlUPOEhbZrMdjeQ==", - "dev": true, - "requires": { - "source-list-map": "^2.0.0", - "source-map": "~0.6.1" - }, - "dependencies": { - "source-map": { - "version": "0.6.1", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", - "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", - "dev": true - } - } - }, - "webpack-subresource-integrity": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/webpack-subresource-integrity/-/webpack-subresource-integrity-1.4.0.tgz", - "integrity": "sha512-GB1kB/LwAWC3CxwcedGhMkxGpNZxSheCe1q+KJP1bakuieAdX/rGHEcf5zsEzhKXpqsGqokgsDoD9dIkr61VDQ==", - "dev": true, - "requires": { - "webpack-sources": "^1.3.0" - } - }, - "websocket-driver": { - "version": "0.7.3", - "resolved": "https://registry.npmjs.org/websocket-driver/-/websocket-driver-0.7.3.tgz", - "integrity": "sha512-bpxWlvbbB459Mlipc5GBzzZwhoZgGEZLuqPaR0INBGnPAY1vdBX6hPnoFXiw+3yWxDuHyQjO2oXTMyS8A5haFg==", - "dev": true, - "requires": { - "http-parser-js": ">=0.4.0 <0.4.11", - "safe-buffer": ">=5.1.0", - "websocket-extensions": ">=0.1.1" - } - }, - "websocket-extensions": { - "version": "0.1.3", - "resolved": "https://registry.npmjs.org/websocket-extensions/-/websocket-extensions-0.1.3.tgz", - "integrity": "sha512-nqHUnMXmBzT0w570r2JpJxfiSD1IzoI+HGVdd3aZ0yNi3ngvQ4jv1dtHt5VGxfI2yj5yqImPhOK4vmIh2xMbGg==", - "dev": true - }, - "wheel": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/wheel/-/wheel-1.0.0.tgz", - "integrity": "sha512-XiCMHibOiqalCQ+BaNSwRoZ9FDTAvOsXxGHXChBugewDj7HC8VBIER71dEOiRH1fSdLbRCQzngKTSiZ06ZQzeA==" - }, - "when": { - "version": "3.6.4", - "resolved": "https://registry.npmjs.org/when/-/when-3.6.4.tgz", - "integrity": "sha1-RztRfsFZ4rhQBUl6E5g/CVQS404=", - "dev": true - }, - "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-module": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/which-module/-/which-module-2.0.0.tgz", - "integrity": "sha1-2e8H3Od7mQK4o6j6SzHD4/fm6Ho=", - "dev": true - }, - "worker-farm": { - "version": "1.7.0", - "resolved": "https://registry.npmjs.org/worker-farm/-/worker-farm-1.7.0.tgz", - "integrity": "sha512-rvw3QTZc8lAxyVrqcSGVm5yP/IJ2UcB3U0graE3LCFoZ0Yn2x4EoVSqJKdB/T5M+FLcRPjz4TDacRf3OCfNUzw==", - "dev": true, - "requires": { - "errno": "~0.1.7" - } - }, - "worker-plugin": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/worker-plugin/-/worker-plugin-4.0.2.tgz", - "integrity": "sha512-V+1zSZMOOKk+uBzKyNIODLQLsx59zSIOaI75J1EMS0iR1qy+KQR3y/pQ3T0vIhvPfDFapGRMsoMvQNEL3okqSA==", - "dev": true, - "requires": { - "loader-utils": "^1.1.0" - }, - "dependencies": { - "json5": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/json5/-/json5-1.0.1.tgz", - "integrity": "sha512-aKS4WQjPenRxiQsC93MNfjx+nbF4PAdYzmd/1JIj8HYzqfbu86beTuNgXDzPknWk0n0uARlyewZo4s++ES36Ow==", - "dev": true, - "requires": { - "minimist": "^1.2.0" - } - }, - "loader-utils": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/loader-utils/-/loader-utils-1.4.0.tgz", - "integrity": "sha512-qH0WSMBtn/oHuwjy/NucEgbx5dbxxnxup9s4PVXJUDHZBQY+s0NWA9rJf53RBnQZxfch7euUui7hpoAPvALZdA==", - "dev": true, - "requires": { - "big.js": "^5.2.2", - "emojis-list": "^3.0.0", - "json5": "^1.0.1" - } - } - } - }, - "wrap-ansi": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-2.1.0.tgz", - "integrity": "sha1-2Pw9KE3QV5T+hJc8rs3Rz4JP3YU=", - "dev": true, - "requires": { - "string-width": "^1.0.1", - "strip-ansi": "^3.0.1" - }, - "dependencies": { - "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=", - "dev": true, - "requires": { - "number-is-nan": "^1.0.0" - } - }, - "string-width": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/string-width/-/string-width-1.0.2.tgz", - "integrity": "sha1-EYvfW4zcUaKn5w0hHgfisLmxB9M=", - "dev": true, - "requires": { - "code-point-at": "^1.0.0", - "is-fullwidth-code-point": "^1.0.0", - "strip-ansi": "^3.0.0" - } - } - } - }, - "wrappy": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", - "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=", - "dev": true - }, - "ws": { - "version": "6.2.1", - "resolved": "https://registry.npmjs.org/ws/-/ws-6.2.1.tgz", - "integrity": "sha512-GIyAXC2cB7LjvpgMt9EKS2ldqr0MTrORaleiOno6TweZ6r3TKtoFQWay/2PceJ3RuBasOHzXNn5Lrw1X0bEjqA==", - "dev": true, - "requires": { - "async-limiter": "~1.0.0" - } - }, - "xml2js": { - "version": "0.4.23", - "resolved": "https://registry.npmjs.org/xml2js/-/xml2js-0.4.23.tgz", - "integrity": "sha512-ySPiMjM0+pLDftHgXY4By0uswI3SPKLDw/i3UXbnO8M/p28zqexCUoPmQFrYD+/1BzhGJSs2i1ERWKJAtiLrug==", - "dev": true, - "requires": { - "sax": ">=0.6.0", - "xmlbuilder": "~11.0.0" - } - }, - "xmlbuilder": { - "version": "11.0.1", - "resolved": "https://registry.npmjs.org/xmlbuilder/-/xmlbuilder-11.0.1.tgz", - "integrity": "sha512-fDlsI/kFEx7gLvbecc0/ohLG50fugQp8ryHzMTuW9vSa1GJ0XYWKnhsUx7oie3G98+r56aTQIUB4kht42R3JvA==", - "dev": true - }, - "xtend": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/xtend/-/xtend-4.0.2.tgz", - "integrity": "sha512-LKYU1iAXJXUgAXn9URjiu+MWhyUXHsvfp7mcuYm9dSUKK0/CjtrUwFAxD82/mCWbtLsGjFIad0wIsod4zrTAEQ==", - "dev": true - }, - "y18n": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/y18n/-/y18n-4.0.0.tgz", - "integrity": "sha512-r9S/ZyXu/Xu9q1tYlpsLIsa3EeLXXk0VwlxqTcFRfg9EhMW+17kbt9G0NrgCmhGb5vT2hyhJZLfDGx+7+5Uj/w==", - "dev": true - }, - "yallist": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/yallist/-/yallist-4.0.0.tgz", - "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", - "dev": true - }, - "yargs": { - "version": "12.0.5", - "resolved": "https://registry.npmjs.org/yargs/-/yargs-12.0.5.tgz", - "integrity": "sha512-Lhz8TLaYnxq/2ObqHDql8dX8CJi97oHxrjUcYtzKbbykPtVW9WB+poxI+NM2UIzsMgNCZTIf0AQwsjK5yMAqZw==", - "dev": true, - "requires": { - "cliui": "^4.0.0", - "decamelize": "^1.2.0", - "find-up": "^3.0.0", - "get-caller-file": "^1.0.1", - "os-locale": "^3.0.0", - "require-directory": "^2.1.1", - "require-main-filename": "^1.0.1", - "set-blocking": "^2.0.0", - "string-width": "^2.0.0", - "which-module": "^2.0.0", - "y18n": "^3.2.1 || ^4.0.0", - "yargs-parser": "^11.1.1" - }, - "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" - } - }, - "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 - } - } - }, - "yargs-parser": { - "version": "11.1.1", - "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-11.1.1.tgz", - "integrity": "sha512-C6kB/WJDiaxONLJQnF8ccx9SEeoTTLek8RVbaOIsrAUS8VrBEXfmeSnCZxygc+XC2sNMBIwOOnfcxiynjHsVSQ==", - "dev": true, - "requires": { - "camelcase": "^5.0.0", - "decamelize": "^1.2.0" - } - }, - "yn": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/yn/-/yn-3.1.1.tgz", - "integrity": "sha512-Ux4ygGWsu2c7isFWe8Yu1YluJmqVhxqK2cLXNQA5AcC3QfbGNpM7fu0Y8b/z16pXLnFxZYvWhd3fhBY9DLmC6Q==", - "dev": true - }, - "zone.js": { - "version": "0.10.3", - "resolved": "https://registry.npmjs.org/zone.js/-/zone.js-0.10.3.tgz", - "integrity": "sha512-LXVLVEq0NNOqK/fLJo3d0kfzd4sxwn2/h67/02pjCjfKDxgx1i9QqpvtHD8CrBnSSwMw5+dy11O7FRX5mkO7Cg==" - } - } -} diff --git a/src/composer/package.json b/src/composer/package.json deleted file mode 100644 index 4f41c76..0000000 --- a/src/composer/package.json +++ /dev/null @@ -1,66 +0,0 @@ -{ - "name": "composer", - "version": "1.0.0", - "scripts": { - "ng": "ng", - "start": "ng serve --port=4300 --disable-host-check", - "build": "ng build --configuration=production", - "build_local": "ng build --configuration=local", - "staging": "ng serve --configuration=staging --port=4300", - "test": "ng test", - "lint": "ng lint", - "e2e": "ng e2e" - }, - "private": true, - "dependencies": { - "@angular/animations": "~9.1.4", - "@angular/cdk": "^9.2.2", - "@angular/common": "~9.1.4", - "@angular/compiler": "~9.1.4", - "@angular/core": "~11.0.5", - "@angular/flex-layout": "^9.0.0-beta.29", - "@angular/forms": "~9.1.4", - "@angular/material": "^9.2.2", - "@angular/platform-browser": "~9.1.4", - "@angular/platform-browser-dynamic": "~9.1.4", - "@angular/router": "~9.1.4", - "@ctrl/ngx-codemirror": "3.1.3", - "@ngrx/effects": "^9.1.1", - "@ngrx/store": "^9.1.1", - "angular-split": "^3.0.3", - "codemirror": "5.58.2", - "highlight.js": "^10.4.1", - "jsplumb": "^2.13.2", - "marked": "^4.0.10", - "nan": "^2.14.1", - "ngx-highlightjs": "^4.1.0-beta", - "panzoom": "^9.2.4", - "rxjs": "^6.5.5", - "rxjs-compat": "^6.5.5", - "tslib": "^1.10.0", - "uuid": "^8.0.0", - "zone.js": "~0.10.2" - }, - "devDependencies": { - "@angular-devkit/build-angular": "~0.901.4", - "@angular/cli": "~9.1.4", - "@angular/compiler-cli": "~9.1.4", - "@angular/language-service": "~9.1.4", - "@types/highlight.js": "^9.12.3", - "@types/jasmine": "~3.5.0", - "@types/jasminewd2": "~2.0.3", - "@types/node": "^12.11.1", - "codelyzer": "^5.1.2", - "jasmine-core": "~3.5.0", - "jasmine-spec-reporter": "~4.2.1", - "karma": "~6.3.16", - "karma-chrome-launcher": "~3.1.0", - "karma-coverage-istanbul-reporter": "~2.1.0", - "karma-jasmine": "~2.0.1", - "karma-jasmine-html-reporter": "^1.4.2", - "protractor": "~5.4.3", - "ts-node": "~8.3.0", - "tslint": "~5.18.0", - "typescript": "~3.7.5" - } -} diff --git a/src/composer/src/app/app-routing.module.ts b/src/composer/src/app/app-routing.module.ts deleted file mode 100644 index 23658f1..0000000 --- a/src/composer/src/app/app-routing.module.ts +++ /dev/null @@ -1,57 +0,0 @@ -import { NgModule } from '@angular/core' -import { Routes, RouterModule } from '@angular/router' -import { HomeComponent } from './core/components/home/home.component' -import { LoginComponent } from './login/login.component' -import { RegistrationComponent } from './registration/registration.component' -import { CallbackComponent } from './callback/callback.component' -const routes: Routes = [ - { - path: '', - children: [ - { - path: '', - component: HomeComponent - }, - { - path: ':uuid', - component: HomeComponent - }, - ], - }, - { - path: 'compose', - children: [ - { - path: '', - component: HomeComponent - }, - { - path: ':uuid', - component: HomeComponent - }, - ], - }, - { - path: 'auth', - children: [ - { - path: 'social/github/callback', - component: CallbackComponent - }, - { - path: 'login', - component: LoginComponent - }, - { - path: 'registration', - component: RegistrationComponent - } - ], - }, -] - -@NgModule({ - imports: [RouterModule.forRoot(routes)], - exports: [RouterModule] -}) -export class AppRoutingModule {} diff --git a/src/composer/src/app/app.component.html b/src/composer/src/app/app.component.html deleted file mode 100644 index 90c6b64..0000000 --- a/src/composer/src/app/app.component.html +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/src/composer/src/app/app.component.scss b/src/composer/src/app/app.component.scss deleted file mode 100644 index e69de29..0000000 diff --git a/src/composer/src/app/app.component.spec.ts b/src/composer/src/app/app.component.spec.ts deleted file mode 100644 index aa1f516..0000000 --- a/src/composer/src/app/app.component.spec.ts +++ /dev/null @@ -1,35 +0,0 @@ -import { TestBed, async } from '@angular/core/testing'; -import { RouterTestingModule } from '@angular/router/testing'; -import { AppComponent } from './app.component'; - -describe('AppComponent', () => { - beforeEach(async(() => { - TestBed.configureTestingModule({ - imports: [ - RouterTestingModule - ], - declarations: [ - AppComponent - ], - }).compileComponents(); - })); - - it('should create the app', () => { - const fixture = TestBed.createComponent(AppComponent); - const app = fixture.componentInstance; - expect(app).toBeTruthy(); - }); - - it(`should have as title 'composer'`, () => { - const fixture = TestBed.createComponent(AppComponent); - const app = fixture.componentInstance; - expect(app.title).toEqual('composer'); - }); - - it('should render title', () => { - const fixture = TestBed.createComponent(AppComponent); - fixture.detectChanges(); - const compiled = fixture.nativeElement; - expect(compiled.querySelector('.content span').textContent).toContain('composer app is running!'); - }); -}); diff --git a/src/composer/src/app/app.component.ts b/src/composer/src/app/app.component.ts deleted file mode 100644 index 45d7d53..0000000 --- a/src/composer/src/app/app.component.ts +++ /dev/null @@ -1,11 +0,0 @@ -import { Component } from '@angular/core'; -import { AuthGuard } from './core/helpers/auth.gourd' -@Component({ - selector: 'app-root', - templateUrl: './app.component.html', - styleUrls: ['./app.component.scss'] -}) -export class AppComponent { - title = 'Compose'; - constructor(private authGuard: AuthGuard) {} -} diff --git a/src/composer/src/app/app.module.ts b/src/composer/src/app/app.module.ts deleted file mode 100644 index f570e33..0000000 --- a/src/composer/src/app/app.module.ts +++ /dev/null @@ -1,160 +0,0 @@ -import { BrowserModule } from '@angular/platform-browser' -import { NgModule } from '@angular/core' -import { StoreModule } from '@ngrx/store' -import { EffectsModule } from '@ngrx/effects' -import { ProjectEffects } from './core/store/effects/projects.effects' - -import { AppRoutingModule } from './app-routing.module' -import { AppComponent } from './app.component' - -import { MatButtonModule } from '@angular/material/button' -import { MatSidenavModule } from '@angular/material/sidenav' -import { MatCardModule } from '@angular/material/card' -import { MatFormFieldModule } from '@angular/material/form-field' -import { MatIconModule } from '@angular/material/icon' -import { MatInputModule } from '@angular/material/input' -import { MatDialogModule } from '@angular/material/dialog' -import { MatSelectModule } from '@angular/material/select' -import { MatTabsModule } from '@angular/material/tabs' -import { ClipboardModule } from '@angular/cdk/clipboard' -import { DragDropModule } from '@angular/cdk/drag-drop' -import { MatButtonToggleModule } from '@angular/material/button-toggle' -import { MatProgressSpinnerModule } from '@angular/material/progress-spinner' -import { MatSliderModule } from '@angular/material/slider' -import { MatTooltipModule } from '@angular/material/tooltip' -import { MatCheckboxModule } from '@angular/material/checkbox' -import { MatMenuModule } from '@angular/material/menu' -import { MatStepperModule } from '@angular/material/stepper' -import { MatAutocompleteModule } from '@angular/material/autocomplete' - -import { RepoSearchComponent, DialogDetails, DialogAddCustom } from './core/components/repo-search/repo-search.component' -import { HeaderComponent } from './core/components/header/header.component' -import { CodeViewComponent } from './core/components/code-view/code-view.component' -import { CanvasComponent } from './core/components/canvas/canvas.component' -import { HomeComponent } from './core/components/home/home.component' -import { BrowserAnimationsModule } from '@angular/platform-browser/animations' -import { FormsModule, ReactiveFormsModule } from '@angular/forms' -import { projectReducer } from './core/store/project.reducer' -import { globalSpinnerReducer } from './core/store/reducers/global-spinner.reducer' -import { globalDialogReducer } from './core/store/reducers/global-dialog.reducer' -import { globalConfigurationReducer } from './core/store/reducers/global-app-configuration.reducer' -import { HttpClientModule, HTTP_INTERCEPTORS } from '@angular/common/http' -import { FlexLayoutModule } from '@angular/flex-layout' -import { HighlightModule, HIGHLIGHT_OPTIONS, HighlightOptions } from 'ngx-highlightjs' -import { ManageProjectDialogComponent } from './core/components/dialogs/manage-project-dialog/manage-project-dialog.component' -import { ManageVolumesDialogComponent } from './core/components/dialogs/manage-volumes-dialog/manage-volumes-dialog.component' -import { SpinnerComponent } from './core/components/widgets/spinner/spinner.component' -import { CheckCircleComponent } from './core/components/widgets/check-circle/check-circle.component' -import { AngularSplitModule } from 'angular-split' -import { ConfirmDialogComponent } from './core/components/dialogs/confirm-dialog/confirm-dialog.component' -import { NodeComponent } from './core/components/canvas/jsplumb/node/node.component' -import { NodeService } from './core/components/canvas/jsplumb/node.service' -import { EventEmitterService } from './core/services/event-emitter.service' -import { MarkedPipe } from './core/pipe/marked.pipe' -import { TruncateTextPipe } from './core/pipe/truncate-text.pipe' -import { KeyValueComponent } from './core/components/common/key-value/key-value/key-value.component' -import { ManageNetworksDialogComponent } from './core/components/dialogs/manage-networks-dialog/manage-networks-dialog.component' -import { SideBarComponent } from './core/components/side-bar/side-bar.component' -import { LoginComponent } from './login/login.component' -import { RegistrationComponent } from './registration/registration.component' - -import { JwtInterceptor} from './core/helpers/jwt.interceptor' -import { ErrorInterceptor } from './core/helpers/error.interceptor' -import { ManageUserDialogComponent } from './core/components/dialogs/manage-user-dialog/manage-user-dialog.component' -import { CallbackComponent } from './callback/callback.component' -import { ImportDialogComponent } from './core/components/dialogs/import-dialog/import-dialog.component' -import { GlobalDialogComponent } from './core/components/dialogs/global-dialog/global-dialog.component' - -import { CodemirrorModule } from '@ctrl/ngx-codemirror'; -import { RecipeComponent, DialogPublishRecipe, DialogRecipeDetails } from './core/components/recipe/recipe.component'; -import { BuildDialogComponent } from './core/components/dialogs/build-dialog/build-dialog.component'; -import { DeployDialogComponent } from './core/components/dialogs/deploy-dialog/deploy-dialog.component'; - -export function getHighlightLanguages() { - return { - yaml: () => import('highlight.js/lib/languages/yaml'), - } -} - -@NgModule({ - declarations: [ - AppComponent, - RepoSearchComponent, - DialogDetails, - DialogAddCustom, - HeaderComponent, - CodeViewComponent, - CanvasComponent, - HomeComponent, - ManageProjectDialogComponent, - ManageVolumesDialogComponent, - SpinnerComponent, - CheckCircleComponent, - ConfirmDialogComponent, - NodeComponent, - MarkedPipe, - TruncateTextPipe, - KeyValueComponent, - ManageNetworksDialogComponent, - SideBarComponent, - LoginComponent, - RegistrationComponent, - ManageUserDialogComponent, - CallbackComponent, - ImportDialogComponent, - GlobalDialogComponent, - RecipeComponent, - DialogPublishRecipe, - DialogRecipeDetails, - BuildDialogComponent, - DeployDialogComponent - ], - imports: [ - BrowserModule, - AppRoutingModule, - BrowserAnimationsModule, - MatSidenavModule, - MatCardModule, - MatFormFieldModule, - MatInputModule, - MatDialogModule, - MatButtonModule, - MatIconModule, - MatSelectModule, - MatTabsModule, - MatButtonToggleModule, - MatProgressSpinnerModule, - MatSliderModule, - MatTooltipModule, - MatCheckboxModule, - MatMenuModule, - MatStepperModule, - MatAutocompleteModule, - DragDropModule, - HighlightModule, - FlexLayoutModule, - FormsModule, - ReactiveFormsModule, - HttpClientModule, - StoreModule.forRoot({ project: projectReducer, globalSpinnerState: globalSpinnerReducer, globalError: globalDialogReducer, globalAppConfiguration: globalConfigurationReducer }), - ClipboardModule, - EffectsModule.forRoot([ProjectEffects]), - AngularSplitModule.forRoot(), - CodemirrorModule - ], - providers: [ - { - provide: HIGHLIGHT_OPTIONS, - useValue: { - lineNumbers: true, - languages: getHighlightLanguages(), - } - }, - { provide: HTTP_INTERCEPTORS, useClass: JwtInterceptor, multi: true }, - { provide: HTTP_INTERCEPTORS, useClass: ErrorInterceptor, multi: true }, - EventEmitterService, - NodeService, - ], - bootstrap: [AppComponent], -}) -export class AppModule {} diff --git a/src/composer/src/app/app.state.ts b/src/composer/src/app/app.state.ts deleted file mode 100644 index 16052c7..0000000 --- a/src/composer/src/app/app.state.ts +++ /dev/null @@ -1,7 +0,0 @@ -import { Project, GlobalDialog, GlobalAppConfiguration } from './core/store/models'; -export interface AppState { - project: Project; - globalSpinnerState: boolean; - globalError: GlobalDialog; - globalAppConfiguration: GlobalAppConfiguration; -} diff --git a/src/composer/src/app/callback/callback.component.html b/src/composer/src/app/callback/callback.component.html deleted file mode 100644 index e69de29..0000000 diff --git a/src/composer/src/app/callback/callback.component.scss b/src/composer/src/app/callback/callback.component.scss deleted file mode 100644 index e69de29..0000000 diff --git a/src/composer/src/app/callback/callback.component.spec.ts b/src/composer/src/app/callback/callback.component.spec.ts deleted file mode 100644 index 848b5e3..0000000 --- a/src/composer/src/app/callback/callback.component.spec.ts +++ /dev/null @@ -1,25 +0,0 @@ -import { async, ComponentFixture, TestBed } from '@angular/core/testing'; - -import { CallbackComponent } from './callback.component'; - -describe('CallbackComponent', () => { - let component: CallbackComponent; - let fixture: ComponentFixture; - - beforeEach(async(() => { - TestBed.configureTestingModule({ - declarations: [ CallbackComponent ] - }) - .compileComponents(); - })); - - beforeEach(() => { - fixture = TestBed.createComponent(CallbackComponent); - component = fixture.componentInstance; - fixture.detectChanges(); - }); - - it('should create', () => { - expect(component).toBeTruthy(); - }); -}); diff --git a/src/composer/src/app/callback/callback.component.ts b/src/composer/src/app/callback/callback.component.ts deleted file mode 100644 index cd603c0..0000000 --- a/src/composer/src/app/callback/callback.component.ts +++ /dev/null @@ -1,31 +0,0 @@ -import { Component, OnInit } from '@angular/core' -import { ActivatedRoute, Router } from '@angular/router' -import { AuthenticationService } from '../core/services/authentication.service' -import { first } from 'rxjs/operators' - -@Component({ - selector: 'app-callback', - templateUrl: './callback.component.html', - styleUrls: ['./callback.component.scss'] -}) -export class CallbackComponent implements OnInit { - - constructor(private activetedRoute: ActivatedRoute, private router: Router, private authenticationService: AuthenticationService) { } - - ngOnInit(): void { - const code = this.activetedRoute.snapshot.queryParams['code'] - if (code) { - this.authenticationService.gitHubLogin(code) - .pipe(first()) - .subscribe( - data => { - const lastPageBeferoLogin = localStorage.getItem('lastPageBeforeRegistrationOrLogin') - lastPageBeferoLogin ? this.router.navigate([lastPageBeferoLogin]) : this.router.navigate(['/']) - }); - } else { - const lastPageBeferoLogin = localStorage.getItem('lastPageBeforeRegistrationOrLogin') - lastPageBeferoLogin ? this.router.navigate([lastPageBeferoLogin]) : this.router.navigate(['/']) - } - } - -} diff --git a/src/composer/src/app/core/components/canvas/canvas.component.html b/src/composer/src/app/core/components/canvas/canvas.component.html deleted file mode 100644 index 0b71bd6..0000000 --- a/src/composer/src/app/core/components/canvas/canvas.component.html +++ /dev/null @@ -1,24 +0,0 @@ -
- - - -
- -
-
- -
-
diff --git a/src/composer/src/app/core/components/canvas/canvas.component.scss b/src/composer/src/app/core/components/canvas/canvas.component.scss deleted file mode 100644 index 47b65e5..0000000 --- a/src/composer/src/app/core/components/canvas/canvas.component.scss +++ /dev/null @@ -1,38 +0,0 @@ -.canvas-tools { - position: absolute; - top: 10px; - right: 10px; - z-index: 2; -} - -.canvas-box { - position: absolute; - width: 150px; - height: 150px; - padding: 10px; - border: solid 1px #ccc; - color: rgba(0, 0, 0, 0.87); - cursor: move; - transition: box-shadow 200ms cubic-bezier(0, 0, 0.2, 1); - background-color: white; - box-shadow: 0 3px 1px -2px rgba(0, 0, 0, 0.2), 0 2px 2px 0 rgba(0, 0, 0, 0.14), - 0 1px 5px 0 rgba(0, 0, 0, 0.12); -} - -.canvas-box:active { - box-shadow: 0 5px 5px -3px rgba(0, 0, 0, 0.2), - 0 8px 10px 1px rgba(0, 0, 0, 0.14), 0 3px 14px 2px rgba(0, 0, 0, 0.12); -} - -.canvas-boundary { - outline: none; - overflow: hidden; - position: relative; -} - -#canvas { - top: -100px; - left: -100px; - position: absolute; - overflow: hidden; -} \ No newline at end of file diff --git a/src/composer/src/app/core/components/canvas/canvas.component.spec.ts b/src/composer/src/app/core/components/canvas/canvas.component.spec.ts deleted file mode 100644 index 54e3160..0000000 --- a/src/composer/src/app/core/components/canvas/canvas.component.spec.ts +++ /dev/null @@ -1,25 +0,0 @@ -import { async, ComponentFixture, TestBed } from '@angular/core/testing'; - -import { CanvasComponent } from './canvas.component'; - -describe('CanvasComponent', () => { - let component: CanvasComponent; - let fixture: ComponentFixture; - - beforeEach(async(() => { - TestBed.configureTestingModule({ - declarations: [ CanvasComponent ] - }) - .compileComponents(); - })); - - beforeEach(() => { - fixture = TestBed.createComponent(CanvasComponent); - component = fixture.componentInstance; - fixture.detectChanges(); - }); - - it('should create', () => { - expect(component).toBeTruthy(); - }); -}); diff --git a/src/composer/src/app/core/components/canvas/canvas.component.ts b/src/composer/src/app/core/components/canvas/canvas.component.ts deleted file mode 100644 index 6dcd70d..0000000 --- a/src/composer/src/app/core/components/canvas/canvas.component.ts +++ /dev/null @@ -1,229 +0,0 @@ -import { Observable } from 'rxjs/Observable' -import { Store } from '@ngrx/store' -import { Component, OnInit, OnDestroy, ViewChild, ViewContainerRef, AfterViewInit, ElementRef } from '@angular/core' -import { MatDialog } from '@angular/material/dialog' -import { AppState } from './../../../app.state' -import { Project, Service, IPosition, Volume } from './../../store/models' -import { ManageVolumesDialogComponent } from '../dialogs/manage-volumes-dialog/manage-volumes-dialog.component' -import { ManageNetworksDialogComponent } from '../dialogs/manage-networks-dialog/manage-networks-dialog.component' -import { jsPlumb } from 'jsplumb' -import { EventEmitterService } from '../../services/event-emitter.service' -import { NodeService } from './jsplumb/node.service' -import * as ProjectActions from './../../store/project.actions' -import * as GlobalSpinnerActions from './../../store/actions/global-spinner.actions' - -import panzoom from 'panzoom' -import { Subject } from 'rxjs' -import { takeUntil } from 'rxjs/operators' - -@Component({ - selector: 'app-canvas', - templateUrl: './canvas.component.html', - styleUrls: ['./canvas.component.scss'], -}) -export class CanvasComponent implements OnInit, AfterViewInit, OnDestroy { - @ViewChild('nodeContainer') nodeContainer: ElementRef; - project: Observable - services: Service[] - positions: IPosition[] - volumes: Volume[] - - projectData = {} - - jsPlumbInstance - nodes = [] - connections = [] - - onFirstLoad = false - prevPanState: any - dragStatus = false - zoomLevel = 1 - zoomLevelUnit = 0.065 - minZoomLevel = 0.5 - maxZoomLevel = 2 - containerSize = 3000 - - private unSubscribe$ = new Subject() - - private unSubscribeEventService: Function - - @ViewChild('nodes', { read: ViewContainerRef, static: true }) viewContainerRef: ViewContainerRef - - constructor( - private store: Store, - private dialog: MatDialog, - private eventEmitterService: EventEmitterService, - private nodeService: NodeService, - ) { - this.project = this.store.select('project') - - this.project.pipe(takeUntil(this.unSubscribe$)).subscribe((v) => { - if(v.services && v.canvas && v.volumes) { - this.services = v.services - this.positions = v.canvas - this.volumes = v.volumes - - if (this.onFirstLoad == true) { - let zoomLevel = 1 - this.nodeService.clear() - this.positions.forEach((p: IPosition) => { - this.nodeService.addDynamicNode(p) - p.zoomLevel ? (zoomLevel = p.zoomLevel) : (zoomLevel = 1) - }) - setTimeout(() => { - this.nodeContainer.nativeElement.click() - this.services.forEach((s: Service) => { - s.depends_on - ? s.depends_on.forEach((uuid) => { - this.nodeService.addConnection(s.uuid, uuid) - }) - : null - }) - - this.zoomLevel = zoomLevel - this.setZoom() - }) - this.onFirstLoad = false - } - this.store.dispatch(GlobalSpinnerActions.OffSpinner()) - } - }) - - this.eventEmitterService.subscribe('remove:node', (uuid) => { - this.nodeService.removeNode(uuid) - }) - - this.eventEmitterService.subscribe('change:connections', (data: { source: string; targets: Array }) => { - this.nodeService.removeConnections(data.source) - data.targets - ? data.targets.forEach((target) => { - this.nodeService.addConnection(data.source, target) - }) - : '' - }) - - this.eventEmitterService.subscribe('add:depends_on', (data: { source: any; target: any }) => { - const service = this.services.find((s) => s.uuid === data.source) - if (service.depends_on) { - const currentTargetAlreadyExisted = service.depends_on.find((uuid) => uuid === data.target) - if (!currentTargetAlreadyExisted) { - this.store.dispatch( - ProjectActions.UpdateService({ - data: { - ...service, - depends_on: [...service.depends_on, data.target], - }, - }), - ) - } - } else { - this.store.dispatch( - ProjectActions.UpdateService({ - data: { - ...service, - depends_on: [data.target], - }, - }), - ) - } - }) - - this.eventEmitterService.subscribe('remove:depends_on', (data: { source: any; target: any }) => { - const service = this.services.find((s) => s.uuid === data.source) - const dependsOn = service.depends_on.filter((d) => d !== data.target) - this.store.dispatch( - ProjectActions.UpdateService({ - data: { - ...service, - depends_on: dependsOn, - }, - }), - ) - }) - - this.unSubscribeEventService = this.eventEmitterService.subscribe('save:project', () => { - const positionData = this.nodeService.getData() - this.store.dispatch(ProjectActions.SetPositions({ data: positionData })) - this.store.dispatch(ProjectActions.SaveProject()) - }) - - this.eventEmitterService.subscribe('initialize:node', () => { - this.onFirstLoad = true - }) - } - - ngOnInit() {} - - ngAfterViewInit() { - this.jsPlumbInstance = jsPlumb.getInstance() - this.nodeService.setRootViewContainerRef(this.viewContainerRef, this.jsPlumbInstance) - } - - onMouseDown(event) { - this.dragStatus = true - this.prevPanState = { - x: event.x, - y: event.y, - } - } - - onMouseMove(event) { - if (!this.dragStatus) return - this.nodeService.updateNodesPosition({ - left: event.x - this.prevPanState.x, - top: event.y - this.prevPanState.y, - }) - this.prevPanState = { - x: event.x, - y: event.y, - } - } - - onMouseUp() { - this.dragStatus = false - } - - onMouseWheel(event) { - let zoom = event.wheelDelta > 0 ? this.zoomLevel - this.zoomLevelUnit : this.zoomLevel + this.zoomLevelUnit - if (zoom > this.maxZoomLevel || zoom < this.minZoomLevel) return - - this.zoomLevel = zoom - this.setZoom() - } - - setZoom() { - this.viewContainerRef.element.nativeElement.parentElement.style.transform = `scale(${this.zoomLevel})` - this.viewContainerRef.element.nativeElement.parentElement.style.width = `${this.containerSize / this.zoomLevel}px` - this.viewContainerRef.element.nativeElement.parentElement.style.height = `${this.containerSize / this.zoomLevel}px` - this.viewContainerRef.element.nativeElement.parentElement.style.top = `${-(this.containerSize / this.zoomLevel - this.containerSize) / 2 - 100}px` - this.viewContainerRef.element.nativeElement.parentElement.style.left = `${ - -(this.containerSize / this.zoomLevel - this.containerSize) / 2 - 100 - }px` - this.nodeService.setZoom(this.zoomLevel) - } - - manageProjectVolumes(): void { - this.dialog.open(ManageVolumesDialogComponent, { - width: '50%', - minWidth: '640px', - data: this.volumes, - }) - } - - manageProjectSecrets(): void { - //console.log('manageProjectSecrets') - } - - manageProjectNetworks(): void { - this.dialog.open(ManageNetworksDialogComponent, { - width: '50%', - minWidth: '640px', - }) - } - - ngOnDestroy() { - this.unSubscribeEventService() - this.unSubscribe$.next(true) - this.unSubscribe$.complete() - } -} diff --git a/src/composer/src/app/core/components/canvas/jsplumb/node.service.ts b/src/composer/src/app/core/components/canvas/jsplumb/node.service.ts deleted file mode 100644 index c2e0045..0000000 --- a/src/composer/src/app/core/components/canvas/jsplumb/node.service.ts +++ /dev/null @@ -1,105 +0,0 @@ -import { ComponentFactoryResolver, Injectable, ComponentFactory } from '@angular/core' -import { jsPlumb, Connection } from 'jsplumb' - -import { NodeComponent } from './node/node.component' -import { IPosition } from 'src/app/core/store/models' -import { Store } from '@ngrx/store' -import { EventEmitterService } from 'src/app/core/services/event-emitter.service' - -@Injectable() -export class NodeService { - private rootViewContainer: any - - jsPlumbInstance - - constructor(private factoryResolver: ComponentFactoryResolver, public eventEmitter: EventEmitterService) {} - - public setRootViewContainerRef(viewContainerRef, jsplumbInstance) { - this.jsPlumbInstance = jsplumbInstance - - this.rootViewContainer = viewContainerRef - - const eventEmitter = this.eventEmitter - this.jsPlumbInstance.bind('dblclick', function (connection, event) { - this.deleteConnection(connection) - eventEmitter.broadcast('remove:depends_on', { source: connection.sourceId, target: connection.targetId }) - }) - this.jsPlumbInstance.bind('connectionDetached', (connection, event) => { - if (event) eventEmitter.broadcast('remove:depends_on', { source: connection.sourceId, target: connection.targetId }) - }) - this.jsPlumbInstance.bind('connection', (connection, event) => { - if (event && connection.sourceId !== connection.targetId) - this.eventEmitter.broadcast('add:depends_on', { source: connection.sourceId, target: connection.targetId }) - }) - } - - public addDynamicNode(node: IPosition) { - const factory = this.factoryResolver.resolveComponentFactory(NodeComponent) - const component = factory.create(this.rootViewContainer.parentInjector) - ;(component.instance).node = node - ;(component.instance).jsPlumbInstance = this.jsPlumbInstance - this.rootViewContainer.insert(component.hostView) - } - - public addConnection(source, target) { - this.jsPlumbInstance.connect({ uuids: [source + '_bottom', target + '_top'] }) - } - - public removeConnections(source) { - const connections = [] - this.jsPlumbInstance.getAllConnections().forEach((con) => { - if (con.sourceId === source) connections.push(con) - }) - connections.forEach((con) => this.jsPlumbInstance.deleteConnection(con)) - } - - public getData() { - const nodeList = document.querySelectorAll('.node') - - let result: IPosition[] = [] - nodeList.forEach((node: HTMLElement) => { - result.push({ - uuid: node.id, - top: node.offsetTop, - left: node.offsetLeft, - zoomLevel: this.jsPlumbInstance.getZoom(), - }) - }) - return result - } - - public removeNode(uuid: string) { - const nodeList = document.querySelectorAll('.node') - - nodeList.forEach((node) => { - if (node.id === uuid) { - this.jsPlumbInstance.remove(uuid) - } - }) - } - - public updateNodesPosition(offset: { top: number; left: number }) { - const nodeList = document.querySelectorAll('.node') - - nodeList.forEach((node: HTMLElement) => { - node.style.left = `${node.offsetLeft + offset.left}px` - node.style.top = `${node.offsetTop + offset.top}px` - this.jsPlumbInstance.revalidate(node) - }) - } - - public setZoom(zoomLevel: number) { - this.jsPlumbInstance.setZoom(zoomLevel, true) - } - - public clear() { - const nodeList = document.querySelectorAll('.node') - const points = document.querySelectorAll('.jtk-endpoint') - points.forEach((node) => node.remove()) - nodeList.forEach((node) => { - this.jsPlumbInstance.remove(node.id) - node.remove() - }) - this.rootViewContainer.clear() - } -} diff --git a/src/composer/src/app/core/components/canvas/jsplumb/node/node.component.html b/src/composer/src/app/core/components/canvas/jsplumb/node/node.component.html deleted file mode 100644 index 3c4fac2..0000000 --- a/src/composer/src/app/core/components/canvas/jsplumb/node/node.component.html +++ /dev/null @@ -1,46 +0,0 @@ -
-
-
-
- - - -
-
- - - {{ currentService.tag ? ":" : "" }}{{ currentService ? currentService.tag : '' }} - - - - custom build - -
- -
-
- - - -
-
- -
-
-
diff --git a/src/composer/src/app/core/components/canvas/jsplumb/node/node.component.scss b/src/composer/src/app/core/components/canvas/jsplumb/node/node.component.scss deleted file mode 100644 index 58dd91d..0000000 --- a/src/composer/src/app/core/components/canvas/jsplumb/node/node.component.scss +++ /dev/null @@ -1,62 +0,0 @@ -.node { - position: absolute; - text-align: center; - z-index: 1; - border-radius: 10px; - background: #e8e8e8; - border: 1px solid #dadada; - /* box-shadow: 0 3px 1px -2px rgba(0, 0, 0, 0), 0 2px 2px 0 rgba(0,0,0,.14), 0 1px 5px 0 rgba(0,0,0,.12); */ - - .node-body { - position: relative; - width: 155px; - height: 145px; - padding: 4px; - - .node-remove-btn { - position: absolute; - top: -8px; - left: -6px; - } - } - - :hover { - cursor: all-scroll; - } - - .node-actions { - :hover { - cursor: pointer; - } - - button { - cursor: pointer; - } - } - - .node-title { - font-weight: 600; - font-size: 16px; - } - - .node-subtitle { - color: #656565; - word-break: break-all; - line-height: 14px; - padding: 5px 0 0; - font-size: 12px; - - .image-name { - font-weight: 600; - } - } - - ::ng-deep .mat-icon { - width: 20px; - height: 20px; - } - ::ng-deep .material-icons { - font-size: 18px; - font-weight: 600; - } -} \ No newline at end of file diff --git a/src/composer/src/app/core/components/canvas/jsplumb/node/node.component.ts b/src/composer/src/app/core/components/canvas/jsplumb/node/node.component.ts deleted file mode 100644 index 0cb1de3..0000000 --- a/src/composer/src/app/core/components/canvas/jsplumb/node/node.component.ts +++ /dev/null @@ -1,151 +0,0 @@ -import { Component, Input, AfterViewInit, OnDestroy } from '@angular/core' -import { IPosition, Service } from 'src/app/core/store/models' -import { Store } from '@ngrx/store' -import { AppState } from 'src/app/app.state' -import { RestService } from 'src/app/core/services/rest.service' -import { MatDialog } from '@angular/material/dialog' -import { ManageProjectDialogComponent } from '../../../dialogs/manage-project-dialog/manage-project-dialog.component' -import { ConfirmDialogComponent } from '../../../dialogs/confirm-dialog/confirm-dialog.component' -import { BuildDialogComponent } from '../../../dialogs/build-dialog/build-dialog.component' -import { DeployDialogComponent } from '../../../dialogs/deploy-dialog/deploy-dialog.component' -import * as ProjectActions from './../../../../store/project.actions' -import { NodeService } from '../node.service' -import { EventEmitterService } from 'src/app/core/services/event-emitter.service' -import { Subject } from 'rxjs' -import { takeUntil } from 'rxjs/operators' - -@Component({ - selector: 'node', - templateUrl: './node.component.html', - styleUrls: ['./node.component.scss'], -}) -export class NodeComponent implements AfterViewInit, OnDestroy { - @Input() node: IPosition - - @Input() jsPlumbInstance - - services = [] - currentService: Service - - widthNode: number = 150 - heightNode: number = 150 - widthCanvasBorder: number = 15 - - private unSubscribe$ = new Subject() - - constructor(private store: Store, public rest: RestService, public dialog: MatDialog, private eventEmitterService: EventEmitterService) { - const project = this.store.select('project') - - project.pipe(takeUntil(this.unSubscribe$)).subscribe((v) => { - this.services = v.services - - setTimeout(() => { - this.currentService = this.services.find((s: Service) => s.uuid === this.node.uuid) - }) - }) - } - - ngAfterViewInit() { - const exampleDropOptions = { - tolerance: 'touch', - hoverClass: 'dropHover', - activeClass: 'dragActive', - } - let Endpoint1 = { - endpoint: ['Dot', { radius: 7 }], - paintStyle: { fill: '#99cb3a' }, - scope: 'jsPlumb_DefaultScope', - connectorStyle: { stroke: '#99cb3a', strokeWidth: 2 }, - connector: ['Bezier', { curviness: 63 }], - maxConnections: -1, - isSource: true, - isTarget: false, - connectorOverlays: [['Arrow', { location: 1, width: 10, length: 10 }]], - dropOptions: exampleDropOptions, - } - let Endpoint2 = { - endpoint: ['Dot', { radius: 5 }], - paintStyle: { fill: '#ffcb3a' }, - scope: 'jsPlumb_DefaultScope', - maxConnections: -1, - isSource: false, - isTarget: true, - dropOptions: exampleDropOptions, - } - const { uuid } = this.node - - this.jsPlumbInstance.addEndpoint(uuid, { anchor: 'Bottom', uuid: uuid + '_bottom' }, Endpoint1) - this.jsPlumbInstance.addEndpoint(uuid, { anchor: 'Top', uuid: uuid + '_top' }, Endpoint2) - this.jsPlumbInstance.draggable(uuid, { - drag: ({ pos, el }) => { - const { width, height } = el.parentElement.parentElement.getBoundingClientRect() - const [left, top] = pos - //left-top - left < this.widthCanvasBorder ? (el.style.left = `${this.widthCanvasBorder}px`) : '' - top < this.widthCanvasBorder ? (el.style.top = `${this.widthCanvasBorder}px`) : '' - //left-top - //right-bottom - left + this.widthNode + this.widthCanvasBorder > width ? (el.style.left = `${width - this.heightNode - this.widthCanvasBorder}px`) : '' - top + this.heightNode + this.widthCanvasBorder > height ? (el.style.top = `${height - this.heightNode - this.widthCanvasBorder}px`) : '' - //right-bottom - }, - }) - } - - manageProjectService(): void { - const dialogRef = this.dialog.open(ManageProjectDialogComponent, { - width: '50%', - minWidth: '740px', - data: this.currentService, - }) - dialogRef - .afterClosed() - .pipe(takeUntil(this.unSubscribe$)) - .subscribe((result: { data: Service }) => { - if (result) { - this.store.dispatch(ProjectActions.UpdateService({ data: result.data })) - } - }) - } - - removeService(): void { - const dialogRef = this.dialog.open(ConfirmDialogComponent, { - data: { - title: 'Confirm delete', - content: 'Are you sure want to delete this service?', - }, - }) - - dialogRef - .afterClosed() - .pipe(takeUntil(this.unSubscribe$)) - .subscribe((result: boolean) => { - if (result == true) { - this.store.dispatch(ProjectActions.RemoveService({ data: this.currentService })) - this.store.dispatch(ProjectActions.RemovePosition({ data: this.node })) - this.eventEmitterService.broadcast('remove:node', this.currentService.uuid) - } - }) - } - - buildDialogOpen() { - this.dialog.open(BuildDialogComponent, { - width: '50%', - minWidth: '740px', - data: this.currentService, - }) - } - - deployDialogOpen() { - this.dialog.open(DeployDialogComponent, { - width: '50%', - minWidth: '740px', - data: this.currentService, - }) - } - - ngOnDestroy() { - this.unSubscribe$.next(true) - this.unSubscribe$.complete() - } -} diff --git a/src/composer/src/app/core/components/code-view/code-mirror-styles.scss b/src/composer/src/app/core/components/code-view/code-mirror-styles.scss deleted file mode 100644 index 42d304b..0000000 --- a/src/composer/src/app/core/components/code-view/code-mirror-styles.scss +++ /dev/null @@ -1,15 +0,0 @@ -.code-mirror { - height: 100%; - .CodeMirror { - height: 100%; - .CodeMirror-hscrollbar { - display: none!important; - } - .CodeMirror-scroll { - padding-top: 12px; - .CodeMirror-lines { - padding-bottom: 120px; - } - } - } -} diff --git a/src/composer/src/app/core/components/code-view/code-view.component.html b/src/composer/src/app/core/components/code-view/code-view.component.html deleted file mode 100644 index 3b79e61..0000000 --- a/src/composer/src/app/core/components/code-view/code-view.component.html +++ /dev/null @@ -1,64 +0,0 @@ -
-
- - -
- - - - {{ version.title }} - - - -
- - - - - - -
- - - - - -
- -
- -
diff --git a/src/composer/src/app/core/components/code-view/code-view.component.scss b/src/composer/src/app/core/components/code-view/code-view.component.scss deleted file mode 100644 index bdda919..0000000 --- a/src/composer/src/app/core/components/code-view/code-view.component.scss +++ /dev/null @@ -1,41 +0,0 @@ -.code-container { - margin: 0 0 0 16px; - position: relative; - - .editor-buttons-block { - position: absolute; - top: 65px; - right: 20px; - z-index: 4; - } -} - -.code { - overflow: auto; - position: relative; -} - -.btn-code-refresh { - color: #222; - background-color: #fff; - box-shadow: none; - - .mat-icon { - font-size: 20px; - height: 21px; - } -} - -.pseudo-toggle-button { - display: none; -} - -@media (max-width: 410px) { - .code-container { - flex-direction: column!important; - } -} - -.gear-menu { - margin: 10px 0 10px 0; -} diff --git a/src/composer/src/app/core/components/code-view/code-view.component.spec.ts b/src/composer/src/app/core/components/code-view/code-view.component.spec.ts deleted file mode 100644 index c3d18c2..0000000 --- a/src/composer/src/app/core/components/code-view/code-view.component.spec.ts +++ /dev/null @@ -1,25 +0,0 @@ -import { async, ComponentFixture, TestBed } from '@angular/core/testing'; - -import { CodeViewComponent } from './code-view.component'; - -describe('CodeViewComponent', () => { - let component: CodeViewComponent; - let fixture: ComponentFixture; - - beforeEach(async(() => { - TestBed.configureTestingModule({ - declarations: [ CodeViewComponent ] - }) - .compileComponents(); - })); - - beforeEach(() => { - fixture = TestBed.createComponent(CodeViewComponent); - component = fixture.componentInstance; - fixture.detectChanges(); - }); - - it('should create', () => { - expect(component).toBeTruthy(); - }); -}); diff --git a/src/composer/src/app/core/components/code-view/code-view.component.ts b/src/composer/src/app/core/components/code-view/code-view.component.ts deleted file mode 100644 index d517b6d..0000000 --- a/src/composer/src/app/core/components/code-view/code-view.component.ts +++ /dev/null @@ -1,227 +0,0 @@ -import { Component, OnInit, OnDestroy } from '@angular/core' -import { Router, Params, ActivatedRoute } from '@angular/router' -import { FormControl, Validators, FormGroup, FormBuilder } from '@angular/forms' -import { Observable } from 'rxjs/Observable' -import { Store } from '@ngrx/store' -import { MatDialog } from '@angular/material/dialog' -import { Project, GlobalAppConfiguration } from './../../store/models' -import { AppState } from './../../../app.state' -import { RestService } from './../../services/rest.service' -import * as ProjectActions from './../../store/project.actions' -import * as GlobalAppConfigurationActions from './../../store/actions/global-app-configuration.actions' -import * as GlobalDialogActions from './../../store/actions/global-dialog.actions' -import { EventEmitterService } from 'src/app/core/services/event-emitter.service' -import * as GlobalSpinnerActions from './../../store/actions/global-spinner.actions' -import { Subject } from 'rxjs' -import { takeUntil, throttleTime } from 'rxjs/operators' -import { ImportDialogComponent } from '../dialogs/import-dialog/import-dialog.component' -import { strict } from 'assert' - -const THROTTLE_TIMEOUT = 1000 - -interface Version { - value: number - title: string -} - -@Component({ - selector: 'app-code-view', - templateUrl: './code-view.component.html', - styleUrls: ['./code-view.component.scss'], -}) -export class CodeViewComponent implements OnInit, OnDestroy { - globalConfig: Observable - mode: string - project: Observable - versionSelectForm: FormGroup - code: '' - codeFormat: string = 'yaml' - versions: Version[] = [ - { value: 1, title: 'version 1' }, - { value: 2, title: 'version 2' }, - { value: 3, title: 'version 3' }, - ] - specificVersions: Version[] = [ - { value: 1, title: 'version 1' }, - { value: 2, title: 'version 2' }, - { value: 3, title: 'version 3' }, - ] - selectedVersion: Version = { value: 0, title: 'unSelected' } - selectedSpecificVersion: Version - - buttonText: string = 'Copy' - showSpinner: boolean = false - - importUrl: string = null - recipeToLoadUuid: string = null - - private unSubscribe$ = new Subject() - - extraKeys = { - 'Tab': cm => cm.replaceSelection(' '), - "Ctrl-S": () => this.changeMode(), //save current project like "done" - "Cmd-S": () => this.changeMode() //save current project like "done" - } - - constructor(private fb: FormBuilder, private store: Store, public rest: RestService, private eventEmitterService: EventEmitterService, public dialog: MatDialog, private router: Router, private activatedRoute: ActivatedRoute) { - this.project = store.select('project') - this.globalConfig = store.select('globalAppConfiguration') - this.selectedSpecificVersion = this.specificVersions[2] - - this.globalConfig.pipe(takeUntil(this.unSubscribe$)).subscribe(({mode}) => { - this.mode = mode - }) - - this.versionSelectForm = this.fb.group({ - versionSelectControl: [this.specificVersions] - }) - } - - codeRefresh(): void { - if (this.mode === 'compose') { - this.eventEmitterService.broadcast('save:project', {}) - } else if (this.mode === 'import') { - if (!this.importUrl) { - this.store.dispatch(GlobalDialogActions.ProjectNotFound({ message: 'Import url is empty
Click on the Link button top right corner', _type: 'Error'})) - } else { - this.store.dispatch(GlobalSpinnerActions.OnSpinner()) - this.store.dispatch(ProjectActions.ImportProject({data: {type: 'url', payload: this.importUrl}})) - this.eventEmitterService.broadcast('initialize:node', {}) - } - } - } - - containsVersion(obj, list): boolean { - for (let i = 0; i < list.length; i++) { - if (list[i].value === obj.value) { - return true; - } - } - - return false - } - - getVersion(version_number): Version { - for (let i = 0; i < this.specificVersions.length; i++) { - if (this.specificVersions[i].value === version_number) { - return this.specificVersions[i] - } - } - } - - ngOnInit(): void { - const importUrl = this.activatedRoute.snapshot.queryParams['import_url'] - const loadRecipeUrl = this.activatedRoute.snapshot.queryParams['recipe'] - - if (importUrl) { - this.importUrl = importUrl - this.store.dispatch(GlobalAppConfigurationActions.OnImportMode()) - this.codeRefresh() - } - - if (loadRecipeUrl) { - this.recipeToLoadUuid = loadRecipeUrl - this.store.dispatch(GlobalAppConfigurationActions.OnRecipeLoadMode()) - } - - this.project.pipe(takeUntil(this.unSubscribe$), throttleTime(THROTTLE_TIMEOUT, undefined, { trailing: true, leading: true})).subscribe((data) => { - if (this.mode === 'compose') { - let baseVersion = Math.floor(data.version) - let versionObject = this.getVersion(baseVersion) - this.selectedSpecificVersion = versionObject - - const sendData = { - name: data.name, - data: { - version: data.version, - volumes: data.volumes, - services: data.services, - networks: data.networks, - secrets: data.secrets, - configs: data.configs, - }, - } - this.showSpinner = true - this.rest - .generateCode(sendData) - .pipe(takeUntil(this.unSubscribe$)) - .subscribe((data: {}) => { - for (var i = 0; i < data['code'].length; ++i) { - data['code'][i] = data['code'][i].replace(/(\r\n|\n|\r)/gm, '') - } - - this.code = data['code'].join('\n') - - if (data['code_format']) { - this.codeFormat = data['code_format'] - } - - this.showSpinner = false - }) - } - }) - - this.eventEmitterService.subscribe('load: code', (code) => { - this.code = code - }) - } - - changeVersion(type: string): void { - if (type === 'select') { - this.selectedVersion = { value: 0, title: 'unSlected' } - this.store.dispatch(ProjectActions.UpdateVersion({ data: this.selectedSpecificVersion.value })) - } else if (type === 'toggle') { - this.selectedSpecificVersion = { value: 0, title: 'Select version' } - this.store.dispatch(ProjectActions.UpdateVersion({ data: this.selectedVersion.value })) - } - } - - changeButtonText() { - this.buttonText = 'Copied' - setTimeout(() => (this.buttonText = 'Copy'), 5000) - } - - ngOnDestroy() { - this.unSubscribe$.next(true) - this.unSubscribe$.complete() - } - - changeMode() { - if (this.mode === 'compose') { - this.store.dispatch(GlobalAppConfigurationActions.OnImportMode()) - } else { - this.store.dispatch(ProjectActions.ImportCurrentProject({data: {type: 'yaml', payload: this.code}})) - this.eventEmitterService.broadcast('initialize:node', {}) - this.store.dispatch(GlobalAppConfigurationActions.OnComposeMode()) - } - } - - openImportDialog() { - const importDialog = this.dialog.open(ImportDialogComponent, { - data: { importUrl: this.importUrl }, - width: '800px' - }) - importDialog.componentInstance.onImport.pipe(takeUntil(this.unSubscribe$)).subscribe((importUrl) => { - this.store.dispatch(GlobalAppConfigurationActions.OnImportMode()) - this.importUrl = importUrl - const queryParams: Params = { import_url: importUrl } - this.router.navigate( - ['/'], - { - relativeTo: this.activatedRoute, - queryParams: queryParams, - queryParamsHandling: 'merge', // remove to replace all query params by provided - }) - !this.activatedRoute.snapshot.params['uuid'] ? this.codeRefresh() : '' - this.eventEmitterService.broadcast('initialize:node', {}) - }) - } - - handelEditorClick() { - this.store.dispatch(GlobalAppConfigurationActions.OnImportMode()) - } - - handelFocusout() { - this.store.dispatch(GlobalAppConfigurationActions.OnComposeMode()) - } -} diff --git a/src/composer/src/app/core/components/common/key-value/key-value/key-value.component.html b/src/composer/src/app/core/components/common/key-value/key-value/key-value.component.html deleted file mode 100644 index d9e4e6f..0000000 --- a/src/composer/src/app/core/components/common/key-value/key-value/key-value.component.html +++ /dev/null @@ -1,14 +0,0 @@ -
- - key - - - - - value - - - - -
- \ No newline at end of file diff --git a/src/composer/src/app/core/components/common/key-value/key-value/key-value.component.scss b/src/composer/src/app/core/components/common/key-value/key-value/key-value.component.scss deleted file mode 100644 index 43ab355..0000000 --- a/src/composer/src/app/core/components/common/key-value/key-value/key-value.component.scss +++ /dev/null @@ -1,13 +0,0 @@ -.key-value-container { - .mat-form-field { - &:nth-child(2) { - min-width: calc(100% - 200px - 80px); - } - } -} - -.ng-star-inserted { - .mat-icon-button { - margin-top: 7px; - } -} diff --git a/src/composer/src/app/core/components/common/key-value/key-value/key-value.component.spec.ts b/src/composer/src/app/core/components/common/key-value/key-value/key-value.component.spec.ts deleted file mode 100644 index a9772a2..0000000 --- a/src/composer/src/app/core/components/common/key-value/key-value/key-value.component.spec.ts +++ /dev/null @@ -1,25 +0,0 @@ -import { async, ComponentFixture, TestBed } from '@angular/core/testing'; - -import { KeyValueComponent } from './key-value.component'; - -describe('KeyValueComponent', () => { - let component: KeyValueComponent; - let fixture: ComponentFixture; - - beforeEach(async(() => { - TestBed.configureTestingModule({ - declarations: [ KeyValueComponent ] - }) - .compileComponents(); - })); - - beforeEach(() => { - fixture = TestBed.createComponent(KeyValueComponent); - component = fixture.componentInstance; - fixture.detectChanges(); - }); - - it('should create', () => { - expect(component).toBeTruthy(); - }); -}); diff --git a/src/composer/src/app/core/components/common/key-value/key-value/key-value.component.ts b/src/composer/src/app/core/components/common/key-value/key-value/key-value.component.ts deleted file mode 100644 index 4ab1431..0000000 --- a/src/composer/src/app/core/components/common/key-value/key-value/key-value.component.ts +++ /dev/null @@ -1,52 +0,0 @@ -import { Component, OnInit, Input, Output, EventEmitter } from '@angular/core'; -import { FormControl, Validators, FormGroup, FormBuilder } from '@angular/forms' -import { Service, Project, Volume, ServicePort, KeyValuePair } from '../../../../store/models' - -@Component({ - selector: 'key-value', - templateUrl: './key-value.component.html', - styleUrls: ['./key-value.component.scss'] -}) -export class KeyValueComponent { - @Input() set keyValueArray(keyValueArray: KeyValuePair[]) { - if(keyValueArray && keyValueArray.length) { - this.currentKeyValuePairs = keyValueArray.map((e) => ({ - key: new FormControl(e.key), - value: new FormControl(e.value), - })) - } else { - this.currentKeyValuePairs = [] - } - } - - currentKeyValuePairs: { - key: FormControl - value: FormControl - }[] = [] - constructor() { } - - addPair() { - this.currentKeyValuePairs.push({ - key: new FormControl(), - value: new FormControl(), - }) - } - removePair(index) { - this.currentKeyValuePairs.splice(index, 1) - } - - getKeyValuePaies(): KeyValuePair[] { - const currentKeyValuePairs = [] - - this.currentKeyValuePairs.forEach((pair) => { - if (pair.key.valid && pair.value.valid) { - currentKeyValuePairs.push({ - key: pair.key.value, - value: pair.value.value, - }) - } - }) - - return currentKeyValuePairs - } -} diff --git a/src/composer/src/app/core/components/dialogs/build-dialog/build-dialog.component.html b/src/composer/src/app/core/components/dialogs/build-dialog/build-dialog.component.html deleted file mode 100644 index 7a2a60a..0000000 --- a/src/composer/src/app/core/components/dialogs/build-dialog/build-dialog.component.html +++ /dev/null @@ -1,72 +0,0 @@ -

Build Configuration

- -

Configuration options that are applied at build time.

- - -
- - Build - - build can be specified as a string containing a path to the build context, or, leave blank and fill out the fields below. - - - - Context - - Either a path to a directory containing a Dockerfile, or a url to a git repository. - - - - Dockerfile - - Alternate Dockerfile. Compose uses an alternate file to build with. A build path must also be specified. - - - - Network - - Set the network containers connect to for the RUN instructions during build. - - - - SHM size (eg. '2gb', 10000000) - - Set the size of the /dev/shm partition for this build’s containers. An integer representing bytes or a string expressing a byte value. - - - - Target - - Build the specified stage as defined inside the Dockerfile. - -
-
- -

Arguments

-

Add build arguments, which are environment variables accessible only during the build process.

- -
- - -

Cache from

-

A list of images that the engine uses for cache resolution.

- -
- - -

Labels

-

Add metadata to the resulting image using Docker labels.

- -
-
-
- - - - - - diff --git a/src/composer/src/app/core/components/dialogs/build-dialog/build-dialog.component.scss b/src/composer/src/app/core/components/dialogs/build-dialog/build-dialog.component.scss deleted file mode 100644 index e69de29..0000000 diff --git a/src/composer/src/app/core/components/dialogs/build-dialog/build-dialog.component.spec.ts b/src/composer/src/app/core/components/dialogs/build-dialog/build-dialog.component.spec.ts deleted file mode 100644 index 689e821..0000000 --- a/src/composer/src/app/core/components/dialogs/build-dialog/build-dialog.component.spec.ts +++ /dev/null @@ -1,25 +0,0 @@ -import { async, ComponentFixture, TestBed } from '@angular/core/testing'; - -import { BuildDialogComponent } from './build-dialog.component'; - -describe('BuildDialogComponent', () => { - let component: BuildDialogComponent; - let fixture: ComponentFixture; - - beforeEach(async(() => { - TestBed.configureTestingModule({ - declarations: [ BuildDialogComponent ] - }) - .compileComponents(); - })); - - beforeEach(() => { - fixture = TestBed.createComponent(BuildDialogComponent); - component = fixture.componentInstance; - fixture.detectChanges(); - }); - - it('should create', () => { - expect(component).toBeTruthy(); - }); -}); diff --git a/src/composer/src/app/core/components/dialogs/build-dialog/build-dialog.component.ts b/src/composer/src/app/core/components/dialogs/build-dialog/build-dialog.component.ts deleted file mode 100644 index 7623e2a..0000000 --- a/src/composer/src/app/core/components/dialogs/build-dialog/build-dialog.component.ts +++ /dev/null @@ -1,67 +0,0 @@ -import { Component, OnInit, ViewChild, Inject } from '@angular/core' -import { MatDialogRef, MAT_DIALOG_DATA } from '@angular/material/dialog' -import { FormControl, Validators, FormGroup, FormBuilder, AbstractControl } from '@angular/forms' -import { KeyValueComponent } from '../../common/key-value/key-value/key-value.component' -import { Service, Project, Volume, ServicePort, Network } from '../../../store/models' -import { CheckCircleComponent } from '../../widgets/check-circle/check-circle.component' -import { Store } from '@ngrx/store' -import * as ProjectActions from './../../../store/project.actions' -import { EventEmitterService } from 'src/app/core/services/event-emitter.service' - -@Component({ - selector: 'app-build-dialog', - templateUrl: './build-dialog.component.html', - styleUrls: ['./build-dialog.component.scss'] -}) -export class BuildDialogComponent implements OnInit { - @ViewChild(CheckCircleComponent) checkCircle: CheckCircleComponent - @ViewChild('args') args: KeyValueComponent - @ViewChild('cache_from') cache_from: KeyValueComponent - @ViewChild('labels') labels: KeyValueComponent - - formGeneral: FormGroup - currentArgs: [] = [] - currentCacheFrom: [] = [] - currentLabels: [] = [] - constructor(public dialogRef: MatDialogRef, private formBuilder: FormBuilder, @Inject(MAT_DIALOG_DATA) public data: Service, private store: Store, private eventEmitterService: EventEmitterService,) { } - - ngOnInit(): void { - this.formGeneral = this.formBuilder.group({ - build: new FormControl(''), - context: new FormControl(''), - dockerfile: new FormControl(''), - network: new FormControl(''), - shm_size: new FormControl(''), - target: new FormControl(''), - }) - - if (this.data.build) { - this.data.build.labels.forEach(val => this.currentLabels.push(val)) - this.data.build.cache_from.forEach(val => this.currentCacheFrom.push(val)) - this.data.build.args.forEach(val => this.currentArgs.push(val)) - - let fields = Object.keys(this.data.build).reduce((acc, key) => { - if(this.data.build[key]) acc[key] = this.data.build[key] - return acc - }, {}) - - this.formGeneral.patchValue(fields) - } - } - - onSave() { - const fields = { - ...this.formGeneral.getRawValue(), - args: this.args.getKeyValuePaies(), - cache_from: this.cache_from.getKeyValuePaies(), - labels: this.labels.getKeyValuePaies() - } - this.store.dispatch(ProjectActions.UpdateService({data: {...this.data, build: fields}})) - this.checkCircle.showCircle() - this.eventEmitterService.broadcast('save:project', {}) - } - - closeModal(): void { - this.dialogRef.close() - } -} diff --git a/src/composer/src/app/core/components/dialogs/confirm-dialog/confirm-dialog.component.html b/src/composer/src/app/core/components/dialogs/confirm-dialog/confirm-dialog.component.html deleted file mode 100644 index 9979313..0000000 --- a/src/composer/src/app/core/components/dialogs/confirm-dialog/confirm-dialog.component.html +++ /dev/null @@ -1,8 +0,0 @@ -

{{ data.title }}

-
-

{{ data.content }}

-
-
- - -
diff --git a/src/composer/src/app/core/components/dialogs/confirm-dialog/confirm-dialog.component.scss b/src/composer/src/app/core/components/dialogs/confirm-dialog/confirm-dialog.component.scss deleted file mode 100644 index e69de29..0000000 diff --git a/src/composer/src/app/core/components/dialogs/confirm-dialog/confirm-dialog.component.spec.ts b/src/composer/src/app/core/components/dialogs/confirm-dialog/confirm-dialog.component.spec.ts deleted file mode 100644 index ccea433..0000000 --- a/src/composer/src/app/core/components/dialogs/confirm-dialog/confirm-dialog.component.spec.ts +++ /dev/null @@ -1,25 +0,0 @@ -import { async, ComponentFixture, TestBed } from '@angular/core/testing'; - -import { ConfirmDialogComponent } from './confirm-dialog.component'; - -describe('ConfirmDialogComponent', () => { - let component: ConfirmDialogComponent; - let fixture: ComponentFixture; - - beforeEach(async(() => { - TestBed.configureTestingModule({ - declarations: [ ConfirmDialogComponent ] - }) - .compileComponents(); - })); - - beforeEach(() => { - fixture = TestBed.createComponent(ConfirmDialogComponent); - component = fixture.componentInstance; - fixture.detectChanges(); - }); - - it('should create', () => { - expect(component).toBeTruthy(); - }); -}); diff --git a/src/composer/src/app/core/components/dialogs/confirm-dialog/confirm-dialog.component.ts b/src/composer/src/app/core/components/dialogs/confirm-dialog/confirm-dialog.component.ts deleted file mode 100644 index 3c5fc86..0000000 --- a/src/composer/src/app/core/components/dialogs/confirm-dialog/confirm-dialog.component.ts +++ /dev/null @@ -1,17 +0,0 @@ -import { Component, OnInit, Inject } from '@angular/core'; -import { MatDialogRef, MAT_DIALOG_DATA } from '@angular/material/dialog'; - -@Component({ - selector: 'app-confirm-dialog', - templateUrl: './confirm-dialog.component.html', - styleUrls: ['./confirm-dialog.component.scss'] -}) -export class ConfirmDialogComponent { - public result = true; - - constructor( - public dialogRef: MatDialogRef, - @Inject(MAT_DIALOG_DATA) public data - ) {} - -} diff --git a/src/composer/src/app/core/components/dialogs/deploy-dialog/deploy-dialog.component.html b/src/composer/src/app/core/components/dialogs/deploy-dialog/deploy-dialog.component.html deleted file mode 100644 index f77e27f..0000000 --- a/src/composer/src/app/core/components/dialogs/deploy-dialog/deploy-dialog.component.html +++ /dev/null @@ -1,134 +0,0 @@ -

Deploy Config

- - - -
- - Mode - - - - - Replicas - - - -
-

Update config

- - Parallelism - - - - - Delay - - - - - Order - - -
- -
- - - Rollback config - - - -
-

Restart policy

- - Condition - - - - - Delay - - - - - Parallelism - - - - - Window - - -
- -
- - - Endpoint mode - - - -
-

Limits

- - Cpus - - - - - Memory - - -
- -
-

Limits

- - Cpus - - - - - Memory - - -
-
-
- -
- - Max replicas per node - - - - - Constraints - - - -

Preferences

- -
-
- - -

Labels

- -
-
-
- - - - - - diff --git a/src/composer/src/app/core/components/dialogs/deploy-dialog/deploy-dialog.component.scss b/src/composer/src/app/core/components/dialogs/deploy-dialog/deploy-dialog.component.scss deleted file mode 100644 index e69de29..0000000 diff --git a/src/composer/src/app/core/components/dialogs/deploy-dialog/deploy-dialog.component.spec.ts b/src/composer/src/app/core/components/dialogs/deploy-dialog/deploy-dialog.component.spec.ts deleted file mode 100644 index 320adbe..0000000 --- a/src/composer/src/app/core/components/dialogs/deploy-dialog/deploy-dialog.component.spec.ts +++ /dev/null @@ -1,25 +0,0 @@ -import { async, ComponentFixture, TestBed } from '@angular/core/testing'; - -import { DeployDialogComponent } from './deploy-dialog.component'; - -describe('DeployDialogComponent', () => { - let component: DeployDialogComponent; - let fixture: ComponentFixture; - - beforeEach(async(() => { - TestBed.configureTestingModule({ - declarations: [ DeployDialogComponent ] - }) - .compileComponents(); - })); - - beforeEach(() => { - fixture = TestBed.createComponent(DeployDialogComponent); - component = fixture.componentInstance; - fixture.detectChanges(); - }); - - it('should create', () => { - expect(component).toBeTruthy(); - }); -}); diff --git a/src/composer/src/app/core/components/dialogs/deploy-dialog/deploy-dialog.component.ts b/src/composer/src/app/core/components/dialogs/deploy-dialog/deploy-dialog.component.ts deleted file mode 100644 index cbdc7e7..0000000 --- a/src/composer/src/app/core/components/dialogs/deploy-dialog/deploy-dialog.component.ts +++ /dev/null @@ -1,109 +0,0 @@ -import { Component, OnInit, ViewChild, Inject } from '@angular/core' -import { MatDialogRef, MAT_DIALOG_DATA } from '@angular/material/dialog' -import { FormControl, Validators, FormGroup, FormBuilder, AbstractControl } from '@angular/forms' -import { KeyValueComponent } from '../../common/key-value/key-value/key-value.component' -import { Service, ServiceDeploy, KeyValuePair } from '../../../store/models' -import { CheckCircleComponent } from '../../widgets/check-circle/check-circle.component' -import { Store } from '@ngrx/store' -import * as ProjectActions from './../../../store/project.actions' -import { EventEmitterService } from 'src/app/core/services/event-emitter.service' - -@Component({ - selector: 'app-deploy-dialog', - templateUrl: './deploy-dialog.component.html', - styleUrls: ['./deploy-dialog.component.scss'] -}) -export class DeployDialogComponent implements OnInit { - @ViewChild(CheckCircleComponent) checkCircle: CheckCircleComponent - @ViewChild('preferences') preferences: KeyValueComponent - @ViewChild('labels') labels: KeyValueComponent - - formGeneral: FormGroup - currentPreferences: KeyValuePair[] = [] - currentLabels: KeyValuePair[] = [] - constructor(public dialogRef: MatDialogRef, private formBuilder: FormBuilder, @Inject(MAT_DIALOG_DATA) public data: Service, private store: Store, private eventEmitterService: EventEmitterService,) { } - - ngOnInit(): void { - this.formGeneral = this.formBuilder.group({ - mode: new FormControl('replicated'), - replicas: new FormControl(''), - update_config: this.formBuilder.group({ - parallelism: new FormControl('', [Validators.pattern('^[0-9]+$')]), - delay: new FormControl(''), - order: new FormControl('') - }), - rollback_config: new FormControl(''), - restart_policy: this.formBuilder.group({ - condition: new FormControl(''), - delay: new FormControl(''), - parallelism: new FormControl('', [Validators.pattern('^[0-9]+$')]), - window: new FormControl(''), - }), - endpoint_mode: new FormControl(''), - placement: this.formBuilder.group({ - constraints: new FormControl(''), - max_replicas_per_node: new FormControl('1', [Validators.pattern('^[0-9]+$')]) - }), - resources: this.formBuilder.group({ - limits: this.formBuilder.group({ - cpus: new FormControl(''), - memory: new FormControl('') - }), - reservations: this.formBuilder.group({ - cpus: new FormControl(''), - memory: new FormControl('') - }) - }) - }) - - if (this.data.deploy) { - let max_replicas_per_node = null - let constraints = null - - try { - max_replicas_per_node = this.data.deploy.placement.max_replicas_per_node - } catch (error) {} - - try { - constraints = this.data.deploy.placement.constraints.join(',') - } catch (error) {} - - try { - this.data.deploy.labels.forEach(val => this.currentLabels.push(val)) - } catch (error) {} - - try { - this.data.deploy.placement.preferences.forEach(val => this.currentPreferences.push(val)) - } catch (error) {} - - this.formGeneral.patchValue({ - ...this.data.deploy, - placement: { - max_replicas_per_node: max_replicas_per_node, - constraints: constraints - } - }) - } - } - - onSave(): void { - const fields: ServiceDeploy = { - ...this.formGeneral.getRawValue(), - placement: { - max_replicas_per_node: this.formGeneral.get('placement').value.max_replicas_per_node, - constraints: this.formGeneral.get('placement').value.constraints.split(','), - preferences: this.preferences.getKeyValuePaies() - }, - labels: this.labels.getKeyValuePaies(), - } - - this.store.dispatch(ProjectActions.UpdateService({data: {...this.data, deploy: fields}})) - this.checkCircle.showCircle() - this.eventEmitterService.broadcast('save:project', {}) - } - - closeModal(): void { - this.dialogRef.close() - } - -} diff --git a/src/composer/src/app/core/components/dialogs/global-dialog/global-dialog.component.html b/src/composer/src/app/core/components/dialogs/global-dialog/global-dialog.component.html deleted file mode 100644 index 3881e9f..0000000 --- a/src/composer/src/app/core/components/dialogs/global-dialog/global-dialog.component.html +++ /dev/null @@ -1,7 +0,0 @@ -

{{data._type}}

- -
-
- - - diff --git a/src/composer/src/app/core/components/dialogs/global-dialog/global-dialog.component.scss b/src/composer/src/app/core/components/dialogs/global-dialog/global-dialog.component.scss deleted file mode 100644 index e69de29..0000000 diff --git a/src/composer/src/app/core/components/dialogs/global-dialog/global-dialog.component.ts b/src/composer/src/app/core/components/dialogs/global-dialog/global-dialog.component.ts deleted file mode 100644 index cef5fa7..0000000 --- a/src/composer/src/app/core/components/dialogs/global-dialog/global-dialog.component.ts +++ /dev/null @@ -1,17 +0,0 @@ -import { Component, Inject } from '@angular/core'; -import { MatDialogRef, MAT_DIALOG_DATA } from '@angular/material/dialog' - -@Component({ - selector: 'global-dialog', - templateUrl: './global-dialog.component.html', - styleUrls: ['./global-dialog.component.scss'] -}) -export class GlobalDialogComponent { - constructor( - public dialogRef: MatDialogRef, - @Inject(MAT_DIALOG_DATA) public data - ) {} - onNoClick(): void { - this.dialogRef.close() - } -} diff --git a/src/composer/src/app/core/components/dialogs/import-dialog/import-dialog.component.html b/src/composer/src/app/core/components/dialogs/import-dialog/import-dialog.component.html deleted file mode 100644 index 2a01955..0000000 --- a/src/composer/src/app/core/components/dialogs/import-dialog/import-dialog.component.html +++ /dev/null @@ -1,10 +0,0 @@ -

Paste a link to your raw docker-compose.yml file

- - - URL - - - - - - diff --git a/src/composer/src/app/core/components/dialogs/import-dialog/import-dialog.component.scss b/src/composer/src/app/core/components/dialogs/import-dialog/import-dialog.component.scss deleted file mode 100644 index e69de29..0000000 diff --git a/src/composer/src/app/core/components/dialogs/import-dialog/import-dialog.component.spec.ts b/src/composer/src/app/core/components/dialogs/import-dialog/import-dialog.component.spec.ts deleted file mode 100644 index 6be487b..0000000 --- a/src/composer/src/app/core/components/dialogs/import-dialog/import-dialog.component.spec.ts +++ /dev/null @@ -1,25 +0,0 @@ -import { async, ComponentFixture, TestBed } from '@angular/core/testing'; - -import { ImportDialogComponent } from './import-dialog.component'; - -describe('ImportDialogComponent', () => { - let component: ImportDialogComponent; - let fixture: ComponentFixture; - - beforeEach(async(() => { - TestBed.configureTestingModule({ - declarations: [ ImportDialogComponent ] - }) - .compileComponents(); - })); - - beforeEach(() => { - fixture = TestBed.createComponent(ImportDialogComponent); - component = fixture.componentInstance; - fixture.detectChanges(); - }); - - it('should create', () => { - expect(component).toBeTruthy(); - }); -}); diff --git a/src/composer/src/app/core/components/dialogs/import-dialog/import-dialog.component.ts b/src/composer/src/app/core/components/dialogs/import-dialog/import-dialog.component.ts deleted file mode 100644 index 42ec8ed..0000000 --- a/src/composer/src/app/core/components/dialogs/import-dialog/import-dialog.component.ts +++ /dev/null @@ -1,28 +0,0 @@ -import { Component, OnInit, Inject, EventEmitter } from '@angular/core' -import { MatDialogRef, MAT_DIALOG_DATA } from '@angular/material/dialog' -import { User } from '../../../store/models' -import { AuthenticationService } from '../../../services/authentication.service' -import { first } from 'rxjs/operators' - -@Component({ - selector: 'app-import-dialog', - templateUrl: './import-dialog.component.html', - styleUrls: ['./import-dialog.component.scss'] -}) -export class ImportDialogComponent implements OnInit { - onImport = new EventEmitter(); - importUrl: string - constructor(public dialogRef: MatDialogRef, @Inject(MAT_DIALOG_DATA) public data: { importUrl: string }) { - if (this.data.importUrl) { - this.importUrl = this.data.importUrl - } - } - - ngOnInit(): void { - } - - import(): void { - this.onImport.emit(this.importUrl) - this.dialogRef.close() - } -} diff --git a/src/composer/src/app/core/components/dialogs/manage-networks-dialog/manage-networks-dialog.component.html b/src/composer/src/app/core/components/dialogs/manage-networks-dialog/manage-networks-dialog.component.html deleted file mode 100644 index 3866a43..0000000 --- a/src/composer/src/app/core/components/dialogs/manage-networks-dialog/manage-networks-dialog.component.html +++ /dev/null @@ -1,77 +0,0 @@ -

Top level network definitions

- -
- - Select existing networks - - {{ network.name }} - - - -
-
- - -
- - Name - - - - - Custom network Name - - Set a custom name for this network. The name field can be used to reference networks which contain special characters. - - -
- External - - External name - - -
-
-
- -
- - Driver - - - {{ driver }} - - - - - Custom driver - - -
-

Driver options

- -
- - -

Labels

- -
-
-
-
- - - - - - - - diff --git a/src/composer/src/app/core/components/dialogs/manage-networks-dialog/manage-networks-dialog.component.scss b/src/composer/src/app/core/components/dialogs/manage-networks-dialog/manage-networks-dialog.component.scss deleted file mode 100644 index e69de29..0000000 diff --git a/src/composer/src/app/core/components/dialogs/manage-networks-dialog/manage-networks-dialog.component.spec.ts b/src/composer/src/app/core/components/dialogs/manage-networks-dialog/manage-networks-dialog.component.spec.ts deleted file mode 100644 index e9d0d13..0000000 --- a/src/composer/src/app/core/components/dialogs/manage-networks-dialog/manage-networks-dialog.component.spec.ts +++ /dev/null @@ -1,25 +0,0 @@ -import { async, ComponentFixture, TestBed } from '@angular/core/testing'; - -import { ManageNetworksDialogComponent } from './manage-networks-dialog.component'; - -describe('ManageNetworksDialogComponent', () => { - let component: ManageNetworksDialogComponent; - let fixture: ComponentFixture; - - beforeEach(async(() => { - TestBed.configureTestingModule({ - declarations: [ ManageNetworksDialogComponent ] - }) - .compileComponents(); - })); - - beforeEach(() => { - fixture = TestBed.createComponent(ManageNetworksDialogComponent); - component = fixture.componentInstance; - fixture.detectChanges(); - }); - - it('should create', () => { - expect(component).toBeTruthy(); - }); -}); diff --git a/src/composer/src/app/core/components/dialogs/manage-networks-dialog/manage-networks-dialog.component.ts b/src/composer/src/app/core/components/dialogs/manage-networks-dialog/manage-networks-dialog.component.ts deleted file mode 100644 index dd77389..0000000 --- a/src/composer/src/app/core/components/dialogs/manage-networks-dialog/manage-networks-dialog.component.ts +++ /dev/null @@ -1,215 +0,0 @@ -import { Component, OnInit, Inject, ViewChild, OnDestroy } from '@angular/core' -import { MatDialogRef, MAT_DIALOG_DATA, MatDialog } from '@angular/material/dialog' -import { Project, Network, KeyValuePair } from 'src/app/core/store/models' -import { Observable, Subject } from 'rxjs' -import { Store } from '@ngrx/store' -import { AppState } from 'src/app/app.state' -import { FormControl, Validators, FormGroup, FormBuilder } from '@angular/forms' -import * as ProjectActions from '../../../store/project.actions' -import { uuidv4 } from 'src/app/core/utils/utils' -import { ConfirmDialogComponent } from '../confirm-dialog/confirm-dialog.component' -import { CheckCircleComponent } from '../../widgets/check-circle/check-circle.component' -import { KeyValueComponent } from '../../common/key-value/key-value/key-value.component' -import { EventEmitterService } from '../../../services/event-emitter.service' -import { takeUntil } from 'rxjs/operators' - -@Component({ - selector: 'manage-networks-dialog', - templateUrl: './manage-networks-dialog.component.html', - styleUrls: ['./manage-networks-dialog.component.scss'], -}) -export class ManageNetworksDialogComponent implements OnInit, OnDestroy { - @ViewChild(CheckCircleComponent) checkCircle: CheckCircleComponent - @ViewChild(KeyValueComponent) driverOptsPairs: KeyValueComponent - @ViewChild('labels') labels :KeyValueComponent - project: Observable - networks: Network[] - formGeneral: FormGroup - selectedNetwork: Network - creatingNew: boolean = true - driverOptions: string[] = ['bridge', 'host', 'overlay', 'macvlan', 'none', 'custom'] - driverOpts: KeyValuePair[] - - private unSubscribe$ = new Subject() - - constructor( - private store: Store, - public dialogRef: MatDialogRef, - private formBuilder: FormBuilder, - public dialog: MatDialog, - private eventEmitterService: EventEmitterService, - ) {} - - ngOnInit(): void { - this.project = this.store.select('project') - - this.project.pipe(takeUntil(this.unSubscribe$)).subscribe((n) => { - this.networks = n.networks - this.selectedNetwork = this.selectedNetwork ? this.networks.find(({ uuid }) => uuid === this.selectedNetwork.uuid) : null - }) - - this.formGeneral = this.formBuilder.group({ - name: new FormControl('', Validators.required), - object_name: new FormControl(''), - driver: this.driverOptions[0], - custom_driver: new FormControl(''), - external: false, - external_name: new FormControl(''), - }) - - const custom_driver = this.formGeneral.get('custom_driver') - const external_name = this.formGeneral.get('external_name') - - this.formGeneral - .get('driver') - .valueChanges.pipe(takeUntil(this.unSubscribe$)) - .subscribe((driver) => { - if (driver === 'custom') { - custom_driver.setValidators([Validators.required]) - } else { - custom_driver.setValidators(null) - } - custom_driver.updateValueAndValidity() - }) - - this.formGeneral - .get('external') - .valueChanges.pipe(takeUntil(this.unSubscribe$)) - .subscribe((external) => { - if (external) { - external_name.setValidators([Validators.required]) - } else { - external_name.setValidators(null) - } - external_name.updateValueAndValidity() - }) - } - - closeModal(): void { - this.dialogRef.close() - } - - onCreate() { - const labels = this.labels.getKeyValuePaies() - const driver_opts = this.driverOptsPairs.getKeyValuePaies() - const driver = this.formGeneral.get('driver').value === 'custom' ? this.formGeneral.get('custom_driver').value : this.formGeneral.get('driver').value - const driver_custom = this.formGeneral.get('driver').value === 'custom' ? true : false - const network = { - uuid: uuidv4(), - name: this.formGeneral.get('name').value, - object_name: this.formGeneral.get('object_name').value, - driver: driver, - driver_custom: driver_custom, - driver_opts: driver_opts, - labels: labels, - external: this.formGeneral.get('external').value, - external_name: this.formGeneral.get('external').value ? this.formGeneral.get('external_name').value : null, - type: null, - attachable: null, - enable_ipv6: null, - ipam: null, - internal: null, - } - this.store.dispatch( - ProjectActions.CreateNetwork({ - data: network, - }), - ) - this.selectedNetwork = network - this.creatingNew = false - this.checkCircle.showCircle() - this.eventEmitterService.broadcast('save:project', {}) - } - - onDelete() { - const dialogRef = this.dialog.open(ConfirmDialogComponent, { - data: { - title: 'Confirm delete', - content: 'Are you sure want to delete this network?', - }, - }) - - dialogRef - .afterClosed() - .pipe(takeUntil(this.unSubscribe$)) - .subscribe((result: true) => { - if (result == true) { - this.store.dispatch(ProjectActions.RemoveNetwork({ data: this.selectedNetwork })) - this._setupDefault() - this.eventEmitterService.broadcast('save:project', {}) - } - }) - } - - onUpdate() { - const labels = this.labels.getKeyValuePaies() - const driver_opts = this.driverOptsPairs.getKeyValuePaies() - const driver = this.formGeneral.get('driver').value === 'custom' ? this.formGeneral.get('custom_driver').value : this.formGeneral.get('driver').value - const driver_custom = this.formGeneral.get('driver').value === 'custom' ? true : false - - this.selectedNetwork = { - ...this.selectedNetwork, - name: this.formGeneral.get('name').value, - object_name: this.formGeneral.get('object_name').value, - external: this.formGeneral.get('external').value, - external_name: this.formGeneral.get('external').value ? this.formGeneral.get('external_name').value : null, - driver, - driver_custom, - driver_opts, - labels: labels - } - - this.store.dispatch(ProjectActions.UpdateNetwork({ data: this.selectedNetwork })) - this.checkCircle.showCircle() - this.eventEmitterService.broadcast('save:project', {}) - } - - onChange() { - if (!this.networks) return - this.driverOpts = this.selectedNetwork.driver_opts.map((pair: KeyValuePair) => { - return { key: pair.key, value: pair.value } - }) - this.creatingNew = false - const isCustomDriver = this.driverOptions.find((driver) => driver === this.selectedNetwork.driver) - if (isCustomDriver && isCustomDriver !== 'custom') { - const fields = Object.keys(this.formGeneral.value).reduce((acc, feild) => { - acc[feild] = this.selectedNetwork[feild] - return acc - }, {}) - this.formGeneral.setValue({ - ...fields, - custom_driver: '', - }) - } else { - const fields = Object.keys(this.formGeneral.value).reduce((acc, feild) => { - acc[feild] = this.selectedNetwork[feild] - return acc - }, {}) - - this.formGeneral.setValue({ - ...fields, - driver: 'custom', - custom_driver: this.selectedNetwork.driver, - }) - } - } - - _setupDefault() { - this.formGeneral.patchValue({ - name: '', - object_name: '', - driver: this.driverOptions[0], - custom_driver: '', - external: false, - external_name: '', - }) - this.selectedNetwork = null - this.creatingNew = true - this.driverOpts = [] - } - - ngOnDestroy() { - this.unSubscribe$.next(true) - this.unSubscribe$.complete() - } -} diff --git a/src/composer/src/app/core/components/dialogs/manage-project-dialog/manage-project-dialog.component.html b/src/composer/src/app/core/components/dialogs/manage-project-dialog/manage-project-dialog.component.html deleted file mode 100644 index 8746030..0000000 --- a/src/composer/src/app/core/components/dialogs/manage-project-dialog/manage-project-dialog.component.html +++ /dev/null @@ -1,179 +0,0 @@ -

{{ serviceName }}

- - - -
- - Image - - - - - Tag - - - - {{ tag }} - - - - - - Service name - - A service with this name already exists - - - - Container name - - Specify a custom container name, rather than a generated default name. - - - - Restart - - - {{ restart }} - - - - - - Command - - - - - Entrypoint - - - - - Working directory - - - -
-

Ports

-
- - published - - - - - target - - - - - protocol - - - {{ protocol }} - - - - - - mode - - - {{ mode }} - - - - - -
- -
- -
- - Depends on - - - {{image.name}} - - - Express dependency between services. - - - - Links - - - {{image.name}} - - - Link to containers in another service. This is a legacy feature of Docker. It may eventually be removed. - - - - Networks - - - {{network.name}} - - - -
-
-
- -

Environment Variables

- -
- - - -
- - Select top-level volume - - {{ v.name }} - - - - - or relative path - - - - - Destination - - - -
- -
- - -

Labels

- -
-
-
- - - - - - diff --git a/src/composer/src/app/core/components/dialogs/manage-project-dialog/manage-project-dialog.component.scss b/src/composer/src/app/core/components/dialogs/manage-project-dialog/manage-project-dialog.component.scss deleted file mode 100644 index bae1fb8..0000000 --- a/src/composer/src/app/core/components/dialogs/manage-project-dialog/manage-project-dialog.component.scss +++ /dev/null @@ -1,24 +0,0 @@ -:host { - ::ng-deep .mat-tab-body { - padding: 16px 0; - } -} - -.tag-select__upload-button { - width: 100%; -} - -.ports-wrap { - .ports-container { - .mat-form-field { - max-width: calc((100% - 80px) / 4); - } - } - margin-bottom: 20px; -} - -.ng-star-inserted { - .mat-icon-button { - margin-top: 7px; - } -} diff --git a/src/composer/src/app/core/components/dialogs/manage-project-dialog/manage-project-dialog.component.spec.ts b/src/composer/src/app/core/components/dialogs/manage-project-dialog/manage-project-dialog.component.spec.ts deleted file mode 100644 index 0e5fe2c..0000000 --- a/src/composer/src/app/core/components/dialogs/manage-project-dialog/manage-project-dialog.component.spec.ts +++ /dev/null @@ -1,25 +0,0 @@ -import { async, ComponentFixture, TestBed } from '@angular/core/testing'; - -import { ManageProjectDialogComponent } from './manage-project-dialog.component'; - -describe('ManageProjectDialogComponent', () => { - let component: ManageProjectDialogComponent; - let fixture: ComponentFixture; - - beforeEach(async(() => { - TestBed.configureTestingModule({ - declarations: [ ManageProjectDialogComponent ] - }) - .compileComponents(); - })); - - beforeEach(() => { - fixture = TestBed.createComponent(ManageProjectDialogComponent); - component = fixture.componentInstance; - fixture.detectChanges(); - }); - - it('should create', () => { - expect(component).toBeTruthy(); - }); -}); diff --git a/src/composer/src/app/core/components/dialogs/manage-project-dialog/manage-project-dialog.component.ts b/src/composer/src/app/core/components/dialogs/manage-project-dialog/manage-project-dialog.component.ts deleted file mode 100644 index fb0a90c..0000000 --- a/src/composer/src/app/core/components/dialogs/manage-project-dialog/manage-project-dialog.component.ts +++ /dev/null @@ -1,291 +0,0 @@ -import { Component, OnInit, Inject, ViewChild, OnDestroy } from '@angular/core' -import { FormControl, Validators, FormGroup, FormBuilder, AbstractControl } from '@angular/forms' -import { MatDialogRef, MAT_DIALOG_DATA } from '@angular/material/dialog' -import { Service, Project, Volume, ServicePort, Network } from '../../../store/models' -import { Store } from '@ngrx/store' -import * as ProjectActions from './../../../store/project.actions' -import { AppState } from 'src/app/app.state' -import { Observable, Subscription, Subject } from 'rxjs' -import { RestService } from './../../../services/rest.service' -import { CheckCircleComponent } from '../../widgets/check-circle/check-circle.component' -import { KeyValueComponent } from '../../common/key-value/key-value/key-value.component' -import { EventEmitterService } from 'src/app/core/services/event-emitter.service' -import { takeUntil } from 'rxjs/operators' - -@Component({ - selector: 'app-manage-project-dialog', - templateUrl: './manage-project-dialog.component.html', - styleUrls: ['./manage-project-dialog.component.scss'], -}) -export class ManageProjectDialogComponent implements OnInit, OnDestroy { - @ViewChild(CheckCircleComponent) checkCircle: CheckCircleComponent - @ViewChild('envVars') envVars: KeyValueComponent - @ViewChild('labels') labels: KeyValueComponent - - formGeneral: FormGroup - isLoading: boolean = false - subscriptions: Subscription[] = [] - project: Observable - volumes: Volume[] - ports: ServicePort[] - serviceName = '' - - currentPortSets: { - target: FormControl - published: FormControl - protocol: FormControl - mode: FormControl - }[] = [] - - protocols: string[] = ['tcp', 'udp'] - modes: string[] = ['host', 'ingress'] - - currentVolumeSets: { - selected: string - relativePathSource: FormControl - formDestination: FormControl - }[] = [] - - tags: string[] = ['latest'] - nextTagPage: number = 1 - restartOptions: string[] = ['always', 'no', 'on-failure', 'unless-stopped'] - - otherImagesInCurrentProject: any[] = [] - networks: Network[] - - private unSubscribe$ = new Subject() - - constructor( - private store: Store, - public dialogRef: MatDialogRef, - @Inject(MAT_DIALOG_DATA) public data: Service, - private formBuilder: FormBuilder, - public rest: RestService, - private eventEmitterService: EventEmitterService, - ) { - this.serviceName = this.data.name - this.formGeneral = this.formBuilder.group({ - name: new FormControl('', [Validators.required, this.serviceNameAlreadyExistsValidator]), - image: new FormControl(''), - tag: new FormControl(''), - restart: new FormControl('', [Validators.required]), - container_name: new FormControl(''), - command: new FormControl(''), - entrypoint: new FormControl(''), - working_dir: new FormControl(''), - depends_on: new FormControl(), - links: new FormControl(), - networks: new FormControl(), - }) - - this.formGeneral.get('image').valueChanges.pipe(takeUntil(this.unSubscribe$)).subscribe((value)=> { - if (!value) this.formGeneral.get('tag').disable() - else if (value && this.formGeneral.get('tag').disabled) this.formGeneral.get('tag').enable() - }) - - this.project = this.store.select('project') - - const sub = this.project.pipe(takeUntil(this.unSubscribe$)).subscribe((v) => { - this.volumes = v.volumes - this.networks = v.networks - this.otherImagesInCurrentProject = [] - - v.services.forEach((service) => { - if (this.data.uuid !== service.uuid) - this.otherImagesInCurrentProject.push({ - uuid: service.uuid, - name: service.name, - }) - }) - }) - - this.subscriptions.push(sub) - - this.data.ports - ? this.data.ports.forEach((p: any) => { - this.currentPortSets.push({ - target: new FormControl(p.target), - published: new FormControl(p.published), - protocol: new FormControl(p.protocol), - mode: new FormControl(p.mode), - }) - }) - : (this.currentPortSets = []) - - this.data.volumes - ? this.data.volumes.forEach((v: any) => { - const selected = this.volumes.find((vol) => vol.uuid === v.volume) - this.currentVolumeSets.push({ - formDestination: new FormControl(v.destination), - relativePathSource: new FormControl(v.relativePathSource), - selected: selected ? selected.uuid : null, - }) - }) - : (this.currentVolumeSets = []) - - this.data.tag ? (this.tags[0] = this.data.tag) : '' - - this.dialogRef - .afterClosed() - .pipe(takeUntil(this.unSubscribe$)) - .subscribe(() => { - this.subscriptions.forEach((sub) => sub.unsubscribe()) - }) - } - - ngOnInit() { - this.formGeneral.patchValue({ - ...this.data, - restart: this.restartOptions[0] - }) - /* - if(this.data.entrypoint) { - this.formGeneral.patchValue({ - entrypoint: Array.isArray(this.data.entrypoint) ? this.preFormating(this.data.entrypoint.join()) : this.preFormating(this.data.entrypoint) - }) - } - if(this.data.command) { - this.formGeneral.patchValue({ - command: Array.isArray(this.data.command) ? this.preFormating(this.data.command.join()) : this.preFormating(this.data.command) - }) - } - */ - this.getTags() - } - - getTags(): void { - this.isLoading = true - this.rest - .getRepoTags(this.formGeneral.get('image').value, this.nextTagPage, this.formGeneral.get('tag').value) - .pipe(takeUntil(this.unSubscribe$)) - .subscribe(({ results }) => { - - if (results) { - this.tags = results.map(({name}) => name) - } - - this.isLoading = false - }) - } - - closeModal(): void { - this.dialogRef.close() - } - - private preFormating(str) { - //return str.split('').map((char, index, arr) => char === ',' && arr[index + 1] && arr[index + 1] != '\n' ? ',\n' : char).join('') - return str - } - - formatValue({ target }, formControlName) { - const str = this.preFormating(target.value) - let patchValue = {} - patchValue[formControlName] = str - this.formGeneral.patchValue({ - ...patchValue - }) - } - - onSave(): void { - const volumes = [] - const ports = [] - const environment = this.envVars.getKeyValuePaies() - const labels = this.labels.getKeyValuePaies() - - this.currentPortSets.forEach((port) => { - if (port.published.valid && port.target.valid && port.protocol.valid) { - ports.push({ - published: port.published.value, - target: port.target.value, - protocol: port.protocol.value, - mode: port.mode.value, - }) - } - }) - - this.currentVolumeSets.forEach((volume) => { - if (volume.selected && volume.formDestination.valid) { - volumes.push({ - volume: volume.selected, - destination: volume.formDestination.value, - }) - } - - if (volume.relativePathSource.value && volume.formDestination.valid) { - volumes.push({ - relativePathSource: volume.relativePathSource.value, - destination: volume.formDestination.value, - }) - } - }) - - let fields = this.formGeneral.getRawValue() - - console.log(fields) - - //fields.command = fields.command.split('').map(char => char == '\n' ? char.replace('\n', '') : char).join('') - //fields.entrypoint = fields.entrypoint.split('').map(char => char == '\n' ? char.replace('\n', '') : char).join('') - - //Object.keys(fields).forEach((key) => { - //!fields[key] ? delete fields[key] : '' - //}) - - console.log('after', this.data, fields) - - this.store.dispatch( - ProjectActions.UpdateService({ - data: { - ...this.data, - ...fields, - volumes, - ports, - environment, - labels, - }, - }), - ) - - this.eventEmitterService.broadcast('change:connections', { source: this.data.uuid, targets: this.formGeneral.get('depends_on').value }) - - this.checkCircle.showCircle() - this.eventEmitterService.broadcast('save:project', {}) - } - - onRemovePort(index: number) { - this.currentPortSets.splice(index, 1) - } - - onAddPort() { - this.currentPortSets.push({ - target: new FormControl('80'), - published: new FormControl('8080'), - protocol: new FormControl(this.protocols[0]), - mode: new FormControl(this.modes[0]), - }) - } - - onAddVolume() { - this.currentVolumeSets.push({ - selected: null, - relativePathSource: new FormControl(), - formDestination: new FormControl(), - }) - } - - onRemoveVolume(index: number) { - this.currentVolumeSets.splice(index, 1) - } - - serviceNameAlreadyExistsValidator = (control: AbstractControl): { [key: string]: boolean } | null => { - const theSameNameOfService = this.otherImagesInCurrentProject.find(({ name }) => name === control.value) - if (theSameNameOfService) { - return { nameAlreadyExists: true } - } - return null - } - - ngOnDestroy() { - this.unSubscribe$.next(true) - this.unSubscribe$.complete() - } -} diff --git a/src/composer/src/app/core/components/dialogs/manage-user-dialog/manage-user-dialog.component.html b/src/composer/src/app/core/components/dialogs/manage-user-dialog/manage-user-dialog.component.html deleted file mode 100644 index 229198b..0000000 --- a/src/composer/src/app/core/components/dialogs/manage-user-dialog/manage-user-dialog.component.html +++ /dev/null @@ -1,21 +0,0 @@ -

Account

- -
- First name -

{{data.first_name ? data.first_name : '_'}}

-
- -
- Last name -

{{data.last_name ? data.last_name : '_'}}

- -
- -
- Email -

{{data.email}}

-
-
- - - diff --git a/src/composer/src/app/core/components/dialogs/manage-user-dialog/manage-user-dialog.component.scss b/src/composer/src/app/core/components/dialogs/manage-user-dialog/manage-user-dialog.component.scss deleted file mode 100644 index e69de29..0000000 diff --git a/src/composer/src/app/core/components/dialogs/manage-user-dialog/manage-user-dialog.component.spec.ts b/src/composer/src/app/core/components/dialogs/manage-user-dialog/manage-user-dialog.component.spec.ts deleted file mode 100644 index e094c21..0000000 --- a/src/composer/src/app/core/components/dialogs/manage-user-dialog/manage-user-dialog.component.spec.ts +++ /dev/null @@ -1,25 +0,0 @@ -import { async, ComponentFixture, TestBed } from '@angular/core/testing'; - -import { ManageUserDialogComponent } from './manage-user-dialog.component'; - -describe('ManageUserDialogComponent', () => { - let component: ManageUserDialogComponent; - let fixture: ComponentFixture; - - beforeEach(async(() => { - TestBed.configureTestingModule({ - declarations: [ ManageUserDialogComponent ] - }) - .compileComponents(); - })); - - beforeEach(() => { - fixture = TestBed.createComponent(ManageUserDialogComponent); - component = fixture.componentInstance; - fixture.detectChanges(); - }); - - it('should create', () => { - expect(component).toBeTruthy(); - }); -}); diff --git a/src/composer/src/app/core/components/dialogs/manage-user-dialog/manage-user-dialog.component.ts b/src/composer/src/app/core/components/dialogs/manage-user-dialog/manage-user-dialog.component.ts deleted file mode 100644 index c367fc7..0000000 --- a/src/composer/src/app/core/components/dialogs/manage-user-dialog/manage-user-dialog.component.ts +++ /dev/null @@ -1,28 +0,0 @@ -import { Component, OnInit, Inject } from '@angular/core' -import { MatDialogRef, MAT_DIALOG_DATA } from '@angular/material/dialog' -import { User } from '../../../store/models' -import { AuthenticationService } from '../../../services/authentication.service' - -@Component({ - selector: 'app-manage-user-dialog', - templateUrl: './manage-user-dialog.component.html', - styleUrls: ['./manage-user-dialog.component.scss'] -}) -export class ManageUserDialogComponent implements OnInit { - - constructor( - public dialogRef: MatDialogRef, - @Inject(MAT_DIALOG_DATA) public data: User, - public authenticationService: AuthenticationService - ) {} - - ngOnInit(): void { - } - - - logout() { - this.authenticationService.removeUser() - this.dialogRef.close() - location.href = '/' - } -} diff --git a/src/composer/src/app/core/components/dialogs/manage-volumes-dialog/manage-volumes-dialog.component.html b/src/composer/src/app/core/components/dialogs/manage-volumes-dialog/manage-volumes-dialog.component.html deleted file mode 100644 index 1f68457..0000000 --- a/src/composer/src/app/core/components/dialogs/manage-volumes-dialog/manage-volumes-dialog.component.html +++ /dev/null @@ -1,63 +0,0 @@ -

Top level volume definitions

- -
- - Select existing volumes - - {{ volume.name }} - - - -
-
- - -
- - Name - - - - - Volume custom name - - Set a custom name for this volume. The name field can be used to reference volumes that contain special characters. - - - - Driver - - - -
- External - - External name - - -
-
-
- -

Labels

- -
-
-
-
- - - - - - - - diff --git a/src/composer/src/app/core/components/dialogs/manage-volumes-dialog/manage-volumes-dialog.component.scss b/src/composer/src/app/core/components/dialogs/manage-volumes-dialog/manage-volumes-dialog.component.scss deleted file mode 100644 index e69de29..0000000 diff --git a/src/composer/src/app/core/components/dialogs/manage-volumes-dialog/manage-volumes-dialog.component.spec.ts b/src/composer/src/app/core/components/dialogs/manage-volumes-dialog/manage-volumes-dialog.component.spec.ts deleted file mode 100644 index 86b8612..0000000 --- a/src/composer/src/app/core/components/dialogs/manage-volumes-dialog/manage-volumes-dialog.component.spec.ts +++ /dev/null @@ -1,25 +0,0 @@ -import { async, ComponentFixture, TestBed } from '@angular/core/testing'; - -import { ManageVolumesDialogComponent } from './manage-volumes-dialog.component'; - -describe('ManageVolumesDialogComponent', () => { - let component: ManageVolumesDialogComponent; - let fixture: ComponentFixture; - - beforeEach(async(() => { - TestBed.configureTestingModule({ - declarations: [ ManageVolumesDialogComponent ] - }) - .compileComponents(); - })); - - beforeEach(() => { - fixture = TestBed.createComponent(ManageVolumesDialogComponent); - component = fixture.componentInstance; - fixture.detectChanges(); - }); - - it('should create', () => { - expect(component).toBeTruthy(); - }); -}); diff --git a/src/composer/src/app/core/components/dialogs/manage-volumes-dialog/manage-volumes-dialog.component.ts b/src/composer/src/app/core/components/dialogs/manage-volumes-dialog/manage-volumes-dialog.component.ts deleted file mode 100644 index d73ee10..0000000 --- a/src/composer/src/app/core/components/dialogs/manage-volumes-dialog/manage-volumes-dialog.component.ts +++ /dev/null @@ -1,172 +0,0 @@ -import { Component, OnInit, Inject, ViewChild, OnDestroy } from '@angular/core' -import { MatDialogRef, MAT_DIALOG_DATA, MatDialog } from '@angular/material/dialog' -import { Volume, Project } from 'src/app/core/store/models' -import { Observable, Subject } from 'rxjs' -import { Store } from '@ngrx/store' -import { AppState } from 'src/app/app.state' -import { FormControl, Validators, FormGroup, FormBuilder } from '@angular/forms' -import * as ProjectActions from './../../../store/project.actions' -import { uuidv4 } from 'src/app/core/utils/utils' -import { ConfirmDialogComponent } from '../confirm-dialog/confirm-dialog.component' -import { CheckCircleComponent } from '../../widgets/check-circle/check-circle.component' -import { KeyValueComponent } from '../../common/key-value/key-value/key-value.component' -import { EventEmitterService } from '../../../services/event-emitter.service' -import { takeUntil } from 'rxjs/operators' - -@Component({ - selector: 'app-manage-volumes-dialog', - templateUrl: './manage-volumes-dialog.component.html', - styleUrls: ['./manage-volumes-dialog.component.scss'], -}) -export class ManageVolumesDialogComponent implements OnInit, OnDestroy { - @ViewChild(CheckCircleComponent) checkCircle: CheckCircleComponent - @ViewChild('labels') labels: KeyValueComponent - project: Observable - volumes: Volume[] - formGeneral: FormGroup - selectedVolume: Volume - creatingNew: boolean = true - - private unSubscribe$ = new Subject() - - constructor( - private store: Store, - public dialogRef: MatDialogRef, - private formBuilder: FormBuilder, - public dialog: MatDialog, - private eventEmitterService: EventEmitterService, - ) { - this.project = this.store.select('project') - - this.project.pipe(takeUntil(this.unSubscribe$)).subscribe((v) => { - this.volumes = v.volumes - this.selectedVolume = this.selectedVolume ? this.volumes.find(({ uuid }) => uuid === this.selectedVolume.uuid) : null - }) - - this.formGeneral = this.formBuilder.group({ - name: new FormControl('', Validators.required), - volume_name: new FormControl(''), - driver: 'local', - external: false, - external_name: '', - }) - - const external_name = this.formGeneral.get('external_name') - - this.formGeneral - .get('external') - .valueChanges.pipe(takeUntil(this.unSubscribe$)) - .subscribe((external) => { - if (external) { - external_name.setValidators([Validators.required]) - } else { - external_name.setValidators(null) - } - external_name.updateValueAndValidity() - }) - } - - ngOnInit(): void { - //console.log(this.volumes) - } - - closeModal(): void { - this.dialogRef.close() - } - - onNew() { - this.formGeneral.setValue({ - name: '', - volume_name: '', - driver: 'local', - external: false, - external_name: '', - }) - this.selectedVolume = null - this.creatingNew = true - } - - onChange() { - if (!this.selectedVolume) return - this.creatingNew = false - this.formGeneral.patchValue({ - name: this.selectedVolume.name, - volume_name: this.selectedVolume.volume_name, - driver: this.selectedVolume.driver, - external: this.selectedVolume.external, - external_name: this.selectedVolume.external ? this.selectedVolume.external_name : '', - }) - } - - onDelete() { - const dialogRef = this.dialog.open(ConfirmDialogComponent, { - data: { - title: 'Confirm delete', - content: 'Are you sure want to delete this volume?', - }, - }) - - dialogRef - .afterClosed() - .pipe(takeUntil(this.unSubscribe$)) - .subscribe((result: true) => { - if (result == true) { - this.store.dispatch(ProjectActions.RemoveVolume({ data: this.selectedVolume })) - this.formGeneral.setValue({ - name: '', - volume_name: '', - driver: 'local', - external: false, - external_name: '', - }) - this.selectedVolume = null - this.creatingNew = true - this.eventEmitterService.broadcast('save:project', {}) - } - }) - } - - onCreate() { - const labels = this.labels.getKeyValuePaies() - const volume = { - uuid: uuidv4(), - name: this.formGeneral.get('name').value, - volume_name: this.formGeneral.get('volume_name').value, - driver: this.formGeneral.get('driver').value, - external: this.formGeneral.get('external').value, - external_name: this.formGeneral.get('external').value ? this.formGeneral.get('external_name').value : null, - driver_opts: null, - labels: labels, - } - this.store.dispatch( - ProjectActions.CreateVolume({ - data: volume, - }), - ) - this.selectedVolume = volume - this.creatingNew = false - this.checkCircle.showCircle() - this.eventEmitterService.broadcast('save:project', {}) - } - - onUpdate() { - const labels = this.labels.getKeyValuePaies() - this.selectedVolume = { - ...this.selectedVolume, - name: this.formGeneral.get('name').value, - volume_name: this.formGeneral.get('volume_name').value, - driver: this.formGeneral.get('driver').value, - external: this.formGeneral.get('external').value, - external_name: this.formGeneral.get('external').value ? this.formGeneral.get('external_name').value : null, - labels: labels, - } - this.store.dispatch(ProjectActions.UpdateVolume({ data: this.selectedVolume })) - this.checkCircle.showCircle() - this.eventEmitterService.broadcast('save:project', {}) - } - - ngOnDestroy() { - this.unSubscribe$.next(true) - this.unSubscribe$.complete() - } -} diff --git a/src/composer/src/app/core/components/header/header.component.html b/src/composer/src/app/core/components/header/header.component.html deleted file mode 100644 index 58163d4..0000000 --- a/src/composer/src/app/core/components/header/header.component.html +++ /dev/null @@ -1,35 +0,0 @@ -
-
-
- - Project name - - - - - Select project - - {{ project.name }} - - - - -
- -
-
diff --git a/src/composer/src/app/core/components/header/header.component.scss b/src/composer/src/app/core/components/header/header.component.scss deleted file mode 100644 index 85a0fc5..0000000 --- a/src/composer/src/app/core/components/header/header.component.scss +++ /dev/null @@ -1,103 +0,0 @@ -/** - * Images a now responsive - */ - img { - max-width: 100%; - height: auto; -} - -.project-name { - font-weight: 500; - font-size: 16px; - .mat-form-field { - min-width: 204px; - } - /* iPad */ - @media (max-width: 600px) { - top: 25px; - } - - /* less than or equal to most standard phones */ - @media (max-width: 480px) { - top: 75px; - left: 20px; - word-break: break-word; - } -} - -.btn-report-bugs { - @media (max-width: 860px) { - display: none !important; - visibility: hidden; - } -} - -/** - * Limit the container in - * width for big screens - */ -.container { - max-width: 100%; - padding: 10px 16px; - margin: 0 auto; -} - -/** - * By using display: flex - * Logo and nav are in 2 cols - * align-items make them - * vertically centered - * justify-content distribute - * horizontal spaces around - * and flex-wrap break the - * things in two lines in - * small screens - */ -.main-header .container { - display: flex; - justify-content: flex-end; - flex-wrap: wrap; - - @media (max-width: 720px) { - justify-content: center; - } - - @media (min-width: 600px) { - align-items: center; - } -} - -/** - * The followings are to - * make things more - * clean and airy - * and contents centered - */ - -.main-nav { - display: flex; - align-items: flex-end; - flex-direction: column; - - @media (min-width: 600px) { - flex-direction: row; - align-items: center; - } -} - -.main-nav ul { - margin-top: .5rem; - margin-bottom: 0; - padding: 0; - text-align: center; - - @media (min-width: 600px) { - margin: .5em 0 .5em; - } -} -.main-nav li { - display: inline; -} -.main-nav a { - display: inline-block; -} diff --git a/src/composer/src/app/core/components/header/header.component.spec.ts b/src/composer/src/app/core/components/header/header.component.spec.ts deleted file mode 100644 index 2d0479d..0000000 --- a/src/composer/src/app/core/components/header/header.component.spec.ts +++ /dev/null @@ -1,25 +0,0 @@ -import { async, ComponentFixture, TestBed } from '@angular/core/testing'; - -import { HeaderComponent } from './header.component'; - -describe('HeaderComponent', () => { - let component: HeaderComponent; - let fixture: ComponentFixture; - - beforeEach(async(() => { - TestBed.configureTestingModule({ - declarations: [ HeaderComponent ] - }) - .compileComponents(); - })); - - beforeEach(() => { - fixture = TestBed.createComponent(HeaderComponent); - component = fixture.componentInstance; - fixture.detectChanges(); - }); - - it('should create', () => { - expect(component).toBeTruthy(); - }); -}); diff --git a/src/composer/src/app/core/components/header/header.component.ts b/src/composer/src/app/core/components/header/header.component.ts deleted file mode 100644 index d4854ec..0000000 --- a/src/composer/src/app/core/components/header/header.component.ts +++ /dev/null @@ -1,142 +0,0 @@ -import { Observable } from 'rxjs/Observable' -import { Component, OnInit, ViewChild, OnDestroy, Output, EventEmitter } from '@angular/core' -import { Location } from '@angular/common' -import { Router } from '@angular/router' -import { Store } from '@ngrx/store' -import { AppState } from './../../../app.state' -import * as ProjectActions from './../../store/project.actions' -import { EventEmitterService } from '../../services/event-emitter.service' -import { RestService } from '../../services/rest.service' -import { AuthenticationService } from '../../services/authentication.service' -import { CheckCircleComponent } from '../widgets/check-circle/check-circle.component' -import { Project } from './../../store/models' -import { Subject, Subscription } from 'rxjs' -import { takeUntil } from 'rxjs/operators' -import { ConfirmDialogComponent } from '../dialogs/confirm-dialog/confirm-dialog.component' -import { MatDialogRef, MAT_DIALOG_DATA, MatDialog } from '@angular/material/dialog' -import * as GlobalAppConfigurationActions from './../../store/actions/global-app-configuration.actions' - -@Component({ - selector: 'app-header', - templateUrl: './header.component.html', - styleUrls: ['./header.component.scss'], -}) -export class HeaderComponent implements OnInit, OnDestroy { - @ViewChild(CheckCircleComponent) checkCircle: CheckCircleComponent - @Output() toggleSidebar: EventEmitter = new EventEmitter() - - private subscription: Subscription - project: Observable - mutable = true - name = '' - projects: Project[] = null - selectedProject: Project - - private unSubscribe$ = new Subject() - - constructor(private store: Store, private eventEmitterService: EventEmitterService, private authenticationService: AuthenticationService, private restService: RestService, private router: Router, private dialog: MatDialog, private location: Location) { - this.project = this.store.select('project') - this.subscription = this.project.pipe(takeUntil(this.unSubscribe$)).subscribe((data) => { - this.mutable = data.mutable - this.name = data.name - this.selectedProject = data - if (this.projects) { - const projectAfterUpdate = this.projects.findIndex(({uuid}) => uuid === data.uuid) - projectAfterUpdate !== -1 ? this.projects[projectAfterUpdate] = { ...data } : '' - } - }) - - this.authenticationService.currentUser.pipe(takeUntil(this.unSubscribe$)).subscribe((user) => { - this.projects = null - - if (user) { - this.restService.getUserProjects().pipe(takeUntil(this.unSubscribe$)).subscribe((response) => { - this.projects = response.results - - let projects = [] - - for (let key in response.results) { - let obj = { - name: response.results[key].name, - uuid: response.results[key].uuid, - mutable: response.results[key].mutable, - ...response.results[key].data, - } as Project - - projects.push(obj) - } - - this.projects = projects - }) - } - }) - } - - ngOnInit(): void {} - - compareObjects(o1: any, o2: any): boolean { - return o1.uuid === o2.uuid - } - - showDelete() { - if(this.projects && this.selectedProject) { - return this.projects.find(({uuid}) => uuid === this.selectedProject.uuid) - } - return false - } - - projectSave(): void { - this.checkCircle.showCircle() - this.store.dispatch(GlobalAppConfigurationActions.OnComposeMode()) - this.eventEmitterService.broadcast('save:project', {}) - } - - ngOnDestroy() { - this.unSubscribe$.next(true) - this.unSubscribe$.complete() - } - - updateName() { - if (this.selectedProject && this.projects) { - const cuurentProject = this.projects.find(({uuid}) => uuid === this.selectedProject.uuid) - cuurentProject ? cuurentProject.name = this.name : '' - } - this.store.dispatch(ProjectActions.UpdateName({ data: this.name })) - this.store.dispatch(ProjectActions.SaveProject()) - } - - selectProject({value: projectData}): void { - this.store.dispatch(GlobalAppConfigurationActions.OnComposeMode()) - this.selectedProject = projectData - this.name = projectData.name - this.eventEmitterService.broadcast('initialize:node', {}) - this.store.dispatch(ProjectActions.SaveProjectSuccess({ data: projectData })) - this.router.navigate(['/', `${projectData.uuid}`]) - } - - removeProject() { - const dialogRef = this.dialog.open(ConfirmDialogComponent, { - data: { - title: 'Confirm delete', - content: 'Are you sure want to delete this project?', - }, - }) - - dialogRef - .afterClosed() - .pipe(takeUntil(this.unSubscribe$)) - .subscribe((result: true) => { - if (result == true) { - this.restService.removeProject(this.selectedProject.uuid).pipe(takeUntil(this.unSubscribe$)).subscribe((response) => { - if (this.projects) { - const index = this.projects.findIndex(({uuid}) => uuid === this.selectedProject.uuid) - this.projects.splice(index, 1) - } - - location.href = '/' - }) - } - }) - } -} - diff --git a/src/composer/src/app/core/components/home/home.component.html b/src/composer/src/app/core/components/home/home.component.html deleted file mode 100644 index 36a82ea..0000000 --- a/src/composer/src/app/core/components/home/home.component.html +++ /dev/null @@ -1,39 +0,0 @@ -
- - -
- - - - - - - - - close - - - - - - - - - -
- -
-
-
- - -
diff --git a/src/composer/src/app/core/components/home/home.component.scss b/src/composer/src/app/core/components/home/home.component.scss deleted file mode 100644 index cf0a9a2..0000000 --- a/src/composer/src/app/core/components/home/home.component.scss +++ /dev/null @@ -1,69 +0,0 @@ -:host { - height: 100%; -} - -.header-container { - min-height: 70px; - background: #fff; - border-bottom: 1px solid #e8e8e8; -} - -// .main-container { -// position: absolute; -// top: 50px; -// bottom: 0; -// left: 0; -// right: 0; -// background: #eee; -// } - -.close-nav-button { - position: absolute; - top: 10px; - z-index: 1; - cursor: pointer; - font-size: 18px; -} - -.recipe { - width: 100%; - padding: 20px; - max-width: 600px; -} - -.example-sidenav-content { - display: flex; - height: 100%; - align-items: center; - justify-content: center; -} - -.closed { - transform: rotate(360deg); -} - -.searchPanel { - width: 320px; -} - -.mat-slider { - position: absolute; - top: 50px; - left: 18px; - z-index: 2; -} - -.search-toggle-btn { - display: none; -} - -@media (max-width: 600px) { - .search-toggle-btn { - display: inline-block; - position: absolute; - top: 10px; - z-index: 2; - padding: 0; - min-width: 32px; - } -} diff --git a/src/composer/src/app/core/components/home/home.component.spec.ts b/src/composer/src/app/core/components/home/home.component.spec.ts deleted file mode 100644 index 490e81b..0000000 --- a/src/composer/src/app/core/components/home/home.component.spec.ts +++ /dev/null @@ -1,25 +0,0 @@ -import { async, ComponentFixture, TestBed } from '@angular/core/testing'; - -import { HomeComponent } from './home.component'; - -describe('HomeComponent', () => { - let component: HomeComponent; - let fixture: ComponentFixture; - - beforeEach(async(() => { - TestBed.configureTestingModule({ - declarations: [ HomeComponent ] - }) - .compileComponents(); - })); - - beforeEach(() => { - fixture = TestBed.createComponent(HomeComponent); - component = fixture.componentInstance; - fixture.detectChanges(); - }); - - it('should create', () => { - expect(component).toBeTruthy(); - }); -}); diff --git a/src/composer/src/app/core/components/home/home.component.ts b/src/composer/src/app/core/components/home/home.component.ts deleted file mode 100644 index 2fcb1dd..0000000 --- a/src/composer/src/app/core/components/home/home.component.ts +++ /dev/null @@ -1,102 +0,0 @@ -import { Component, OnInit, Inject, HostListener, OnDestroy, ViewChild } from '@angular/core' -import { ActivatedRoute, Router } from '@angular/router' -import { Store } from '@ngrx/store' -import { AppState } from './../../../app.state' -import * as ProjectActions from './../../store/project.actions' -import * as GlobalDialogActions from './../../store/actions/global-dialog.actions' -import { EventEmitterService } from '../../services/event-emitter.service' -import * as GlobalSpinnerActions from './../../store/actions/global-spinner.actions' -import { MatDrawer } from '@angular/material/sidenav' -import { MatDialog, MatDialogRef, MAT_DIALOG_DATA } from '@angular/material/dialog' -import { GlobalDialogComponent } from '../dialogs/global-dialog/global-dialog.component' -import { Subject } from 'rxjs' -import { takeUntil, skip } from 'rxjs/operators' - -@Component({ - selector: 'app-home', - templateUrl: './home.component.html', - styleUrls: ['./home.component.scss'], -}) -export class HomeComponent implements OnInit, OnDestroy { - @ViewChild('sideNav') sideNav: MatDrawer - currentOpenedSideNavItem: string = null - showFiller = false - screenWidth: number - showSidebar: boolean = true - showCloseButton: boolean = false - - private unSubscribe$ = new Subject() - - constructor(private route: ActivatedRoute, private store: Store, private eventEmitter: EventEmitterService, public dialog: MatDialog, private router: Router, private activatedRoute: ActivatedRoute) {} - - ngOnInit(): void { - this.screenWidth = window.innerWidth - if (this.screenWidth < 600) { - this.showSidebar = false - } - - const uuid = this.route.snapshot.params['uuid'] - if (uuid) { - this.store.dispatch(GlobalSpinnerActions.OnSpinner()) - this.store.dispatch(ProjectActions.UploadProject({ data: uuid })) - this.eventEmitter.broadcast('initialize:node', {}) - } - - const recipe = this.activatedRoute.snapshot.queryParams['recipe'] - if (recipe) { - this.store.dispatch(ProjectActions.ViewRecipe({ data: recipe })) - } - - this.store.select('globalError').pipe(takeUntil(this.unSubscribe$), skip(1)).subscribe((body) => { - if (body.type) { - this.store.dispatch(GlobalSpinnerActions.OffSpinner()) - const dialogRef = this.dialog.open(GlobalDialogComponent, { - width: '800px', - data: body, - }) - if (body.type === GlobalDialogActions.PROJECT_NOT_FOUND) { - dialogRef - .afterClosed() - .pipe(takeUntil(this.unSubscribe$)) - .subscribe(() => { - this.store.dispatch(ProjectActions.SetInitialState()) - this.router.navigate(['/', '']) - }) - } - } - }) - } - - @HostListener('window:resize', ['$event']) - onResize(event) { - this.screenWidth = window.innerWidth - } - - ngOnDestroy() { - this.unSubscribe$.next(true); - this.unSubscribe$.complete(); - } - - openSideBarItem(event) { - if(event.name === null) { - this.sideNav.close() - this.showCloseButton = false - setTimeout(() => { - this.currentOpenedSideNavItem = null - }, 300) - } - else if(event.name === this.currentOpenedSideNavItem) { - this.sideNav.close() - this.showCloseButton = false - setTimeout(() => { - this.currentOpenedSideNavItem = null - }, 300) - } else { - this.currentOpenedSideNavItem = event.name - if(!this.sideNav.opened) { - this.sideNav.open() - this.showCloseButton = true - } - } - } -} diff --git a/src/composer/src/app/core/components/recipe/dialog-publish-recipe.html b/src/composer/src/app/core/components/recipe/dialog-publish-recipe.html deleted file mode 100644 index d87b473..0000000 --- a/src/composer/src/app/core/components/recipe/dialog-publish-recipe.html +++ /dev/null @@ -1,54 +0,0 @@ -

{{mode === 'editing' ? 'Update recipe' : 'Publish a recipe'}}

-

{{title}}

-
- - Name - - - - - Description - - Supports Markdown. - - - - Keywords - - - - - Source repository url - - Nuxx will try to read additional data from your source, and users will be able to visit the repo. - - - - Raw Docker Compose file url - - Nuxx will read this file to load your recipe into view. - - - Private - Private recipes won't show up as latest or in searches. -
-
-
- - - -
- -
- {{errorMessage}} -
-
diff --git a/src/composer/src/app/core/components/recipe/dialog-publish-recipe.scss b/src/composer/src/app/core/components/recipe/dialog-publish-recipe.scss deleted file mode 100644 index 558995e..0000000 --- a/src/composer/src/app/core/components/recipe/dialog-publish-recipe.scss +++ /dev/null @@ -1,13 +0,0 @@ -.mat-dialog-actions { - margin-bottom: 0!important; -} - -.textArea { - overflow: hidden; -} - -.title { - color: #02b52a; - min-height: 23px; - margin: 0; -} \ No newline at end of file diff --git a/src/composer/src/app/core/components/recipe/dialog-recipe-details.html b/src/composer/src/app/core/components/recipe/dialog-recipe-details.html deleted file mode 100644 index b37e61f..0000000 --- a/src/composer/src/app/core/components/recipe/dialog-recipe-details.html +++ /dev/null @@ -1,14 +0,0 @@ -

Recipe info

- -
-

{{ data.name }}

-
-

-
- - - - - diff --git a/src/composer/src/app/core/components/recipe/dialog-recipe-details.scss b/src/composer/src/app/core/components/recipe/dialog-recipe-details.scss deleted file mode 100644 index e4a7d7d..0000000 --- a/src/composer/src/app/core/components/recipe/dialog-recipe-details.scss +++ /dev/null @@ -1,5 +0,0 @@ -.recipe-details-dialog-content { - &::ng-deep img { - max-width: 100%; - } -} \ No newline at end of file diff --git a/src/composer/src/app/core/components/recipe/recipe.component.html b/src/composer/src/app/core/components/recipe/recipe.component.html deleted file mode 100644 index 50a4337..0000000 --- a/src/composer/src/app/core/components/recipe/recipe.component.html +++ /dev/null @@ -1,99 +0,0 @@ - -
- -
- - -
- - - -
-
- - - -
-
-
-

Latest

-
-
-

Found {{recipes.length}} recipes

-
-
-
-
- -
-

{{recipe.name}}

-
- {{i ? '' : ''}}{{keyword}} -
-
-
-
- - -
-
-
- -
- - -
-
- - - -
-
-

Your recipes {{ userRecipes.length }}

-
-
-
-
- -
-

{{recipe.name}}

-
- {{i ? '' : ''}}{{keyword}} -
-
-
-
- - -
-
-
- -
- \ No newline at end of file diff --git a/src/composer/src/app/core/components/recipe/recipe.component.scss b/src/composer/src/app/core/components/recipe/recipe.component.scss deleted file mode 100644 index c488341..0000000 --- a/src/composer/src/app/core/components/recipe/recipe.component.scss +++ /dev/null @@ -1,63 +0,0 @@ -.title { - margin: 15px 0 0 10px; - padding: 0; -} - -.recipe-nav { - max-height: 100%; -} - -.list-recipes { - overflow-y: auto; -} - -.recipe { - padding: 14px 10px; - border-bottom: 1px solid #efefef; - - &:hover { - background: #fafafa; - cursor: pointer; - } - .name { - margin: 0; - font-weight: 500; - } - .mat-icon-button { - width: 30px; - height: 30px; - line-height: 30px; - .mat-icon { - font-size: 18px; - } - } - .keyword-container { - width: 80%; - .keyword { - margin: 0 4px 2px 0; - background: #428efb; - color: #fff; - padding: 0px 6px; - border-radius: 20px; - font-size: 10px; - line-height: 18px; - } - } -} - -.form-wrapper { - ::ng-deep.mat-form-field-wrapper { - margin: 0; - height: 35px; - .mat-form-field-flex { - height: 35px; - .mat-form-field-infix { - padding: 0; - } - } - } -} - -.or { - margin-bottom: 8px; -} diff --git a/src/composer/src/app/core/components/recipe/recipe.component.spec.ts b/src/composer/src/app/core/components/recipe/recipe.component.spec.ts deleted file mode 100644 index 24ee151..0000000 --- a/src/composer/src/app/core/components/recipe/recipe.component.spec.ts +++ /dev/null @@ -1,25 +0,0 @@ -import { async, ComponentFixture, TestBed } from '@angular/core/testing'; - -import { RecipeComponent } from './recipe.component'; - -describe('RecipeComponent', () => { - let component: RecipeComponent; - let fixture: ComponentFixture; - - beforeEach(async(() => { - TestBed.configureTestingModule({ - declarations: [ RecipeComponent ] - }) - .compileComponents(); - })); - - beforeEach(() => { - fixture = TestBed.createComponent(RecipeComponent); - component = fixture.componentInstance; - fixture.detectChanges(); - }); - - it('should create', () => { - expect(component).toBeTruthy(); - }); -}); diff --git a/src/composer/src/app/core/components/recipe/recipe.component.ts b/src/composer/src/app/core/components/recipe/recipe.component.ts deleted file mode 100644 index be94cad..0000000 --- a/src/composer/src/app/core/components/recipe/recipe.component.ts +++ /dev/null @@ -1,218 +0,0 @@ -import { Component, OnInit, EventEmitter, Inject } from '@angular/core' -import { FormGroup, FormControl, Validator, Validators, FormBuilder } from "@angular/forms" -import { Location } from "@angular/common" -import { Router } from "@angular/router" -import { RestService } from "../../services/rest.service" -import { AuthenticationService } from "../../services/authentication.service" -import { MatDialog, MatDialogRef, MAT_DIALOG_DATA } from '@angular/material/dialog' -import { Store } from '@ngrx/store' -import { AppState } from './../../../app.state' -import { User } from '../../store/models' -import * as ProjectActions from './../../store/project.actions' -import * as GlobalAppConfigurationActions from './../../store/actions/global-app-configuration.actions' -import { ConfirmDialogComponent } from '../dialogs/confirm-dialog/confirm-dialog.component' - -@Component({ - selector: 'app-recipe', - templateUrl: './recipe.component.html', - styleUrls: ['./recipe.component.scss'] -}) -export class RecipeComponent implements OnInit { - form: FormGroup - recipes = [] - userRecipes = [] - page: number = 1 - beforeCustomSearch: boolean = true - showSpinner: boolean = true - formRecipePublish: FormGroup - activeTab: string = 'search' - user: User - constructor(public restService: RestService, public dialog: MatDialog, private store: Store, private _location: Location, private authenticationService: AuthenticationService, private router: Router) { } - - ngOnInit(): void { - this.form = new FormGroup({ searchFor: new FormControl('') }) - this.form.get('searchFor').valueChanges.subscribe(() => { - this.page = 1 - }) - this.restService.getPopularRecipes().subscribe(({ results }) => { - this.recipes = results - this.showSpinner = false - }) - this.authenticationService.currentUser.subscribe((user) => { - this.user = user - }) - } - - onEnter(): void { - let searchVal = this.form.get('searchFor').value - this.showSpinner = true - - if (searchVal) { - this.restService.searchRecipes(searchVal, this.page).subscribe(results => { - this.page++ - this.beforeCustomSearch ? this.beforeCustomSearch = false : '' - this.recipes = results.results - this.showSpinner = false - }) - } else { - this.restService.getPopularRecipes().subscribe(({ results }) => { - this.beforeCustomSearch = true - this.recipes = results - this.showSpinner = false - }) - } - } - - openPublishRecipeDialog(): void { - const dialogRef = this.dialog.open(DialogPublishRecipe, { - width: '600px' - }) - dialogRef.componentInstance.onRemove.subscribe(this.removeRecipe) - dialogRef.componentInstance.onUpdate.subscribe(this.addOrUpdateRecipe) - } - - addOrUpdateRecipe = (data: never) => { - const index = this.recipes.findIndex(({uuid}) => uuid === data['uuid']) - index === -1 ? this.recipes.push(data) : this.recipes[index] = data - } - - viewRecipe(event, uuid: string) { - event.stopPropagation(); - this.store.dispatch(ProjectActions.ViewRecipe({data: uuid})) - this.store.dispatch(GlobalAppConfigurationActions.OnRecipeLoadMode()) - this._location.replaceState(`/?recipe=${uuid}`) - } - - editRecipe(event, recipe: object) { - event.stopPropagation(); - const dialogRef = this.dialog.open(DialogPublishRecipe, { - width: '600px', - data: recipe - }) - dialogRef.componentInstance.onRemove.subscribe(this.removeRecipe) - dialogRef.componentInstance.onUpdate.subscribe(this.addOrUpdateRecipe) - } - - removeRecipe = (event, uuid: string) => { - event.stopPropagation(); - const dialogRef = this.dialog.open(ConfirmDialogComponent, { - data: { - title: 'Confirm delete', - content: 'Are you sure want to delete this recipe?', - }, - }) - - dialogRef - .afterClosed() - .subscribe((result: true) => { - if (result == true) { - this.restService.removeRecipe(uuid).subscribe(val => { - const index = this.recipes.findIndex(({uuid: uuidFromArray}) => uuid === uuidFromArray) - this.recipes.splice(index, 1) - this.dialog.closeAll() - }) - } - }) - } - - handlePublishClick() { - this.user ? this.openPublishRecipeDialog() : this.router.navigate(['/auth/login/']) - } - - handleYourRecipesClick() { - if (this.user) { - this.showSpinner = true - this.activeTab = 'yourRecipes' - this.restService.getUserRecipes().subscribe(({ results }) => { - this.userRecipes = results - this.showSpinner = false - }) - } else { - this.router.navigate(['/auth/login/']) - } - } - - viewRecipeDetails(event, recipe: object) { - event.stopPropagation(); - this.dialog.open(DialogRecipeDetails, { - width: '600px', - data: recipe - }) - } -} - -@Component({ - selector: 'dialog-publish-recipe', - templateUrl: 'dialog-publish-recipe.html', - styleUrls: ['./dialog-publish-recipe.scss'], -}) -export class DialogPublishRecipe { - onUpdate = new EventEmitter(); - onRemove = new EventEmitter(); - form: FormGroup - errors: string[] - mode: string = 'creating' - uuid: string - title: string = '' - constructor( - public dialogRef: MatDialogRef, - public restService: RestService, - private formBuilder: FormBuilder, - @Inject(MAT_DIALOG_DATA) public data: object) { - this.form = this.formBuilder.group({ - name: new FormControl('', [Validators.required]), - keywords: new FormControl(''), - repo: new FormControl(''), - raw_url: new FormControl(''), - desc: new FormControl(''), - private: false - }) - if(this.data) { - this.mode = 'editing' - this.uuid = this.data['uuid'] - this.form.patchValue(this.data) - this.form.patchValue({keywords: this.data['keywords'].join(', ')}) - } - } - - addRecipe() { - this.errors = [] - const fields = this.form.getRawValue() - - const data = { - ...fields, - keywords: fields.keywords.split(',').map(str => str.trim()) - } - - if(this.mode === 'editing') this.restService.updateRecipe(this.uuid, data).subscribe(data => { - this.title = 'Recipe successfully updated' - setTimeout(() => { - this.title = '' - }, 3000) - this.onUpdate.emit({...data, uuid: this.uuid}) - }, error => this.errors = error) - else this.restService.createRecipe(data).subscribe(data => { - this.title = 'Recipe successfully created' - setTimeout(() => { - this.title = '' - }, 3000) - this.mode = 'editing' - this.form.patchValue(data) - this.form.patchValue({keywords: data['keywords'].join(', ')}) - this.uuid = data['uuid'] - this.onUpdate.emit(data) - }, error => this.errors = error) - } -} - -@Component({ - selector: 'dialog-recipe-details', - templateUrl: 'dialog-recipe-details.html', - styleUrls: ['./dialog-recipe-details.scss'], -}) -export class DialogRecipeDetails { - constructor( - public dialogRef: MatDialogRef, - @Inject(MAT_DIALOG_DATA) public data: {name: string, desc: string}) { - } -} \ No newline at end of file diff --git a/src/composer/src/app/core/components/repo-search/dialog-add-custom.html b/src/composer/src/app/core/components/repo-search/dialog-add-custom.html deleted file mode 100644 index aeb83df..0000000 --- a/src/composer/src/app/core/components/repo-search/dialog-add-custom.html +++ /dev/null @@ -1,21 +0,0 @@ -

Add custom container

-
- - Image - - - - - Tag - - - - - Service name - - -
- -
- -
diff --git a/src/composer/src/app/core/components/repo-search/dialog-add-custom.scss b/src/composer/src/app/core/components/repo-search/dialog-add-custom.scss deleted file mode 100644 index e69de29..0000000 diff --git a/src/composer/src/app/core/components/repo-search/dialog-details.html b/src/composer/src/app/core/components/repo-search/dialog-details.html deleted file mode 100644 index 3c391cf..0000000 --- a/src/composer/src/app/core/components/repo-search/dialog-details.html +++ /dev/null @@ -1,14 +0,0 @@ -

{{ data.name }}

-
-

{{ data.description }}

-
-
- -
-
-
- - -
-
-
diff --git a/src/composer/src/app/core/components/repo-search/dialog-details.scss b/src/composer/src/app/core/components/repo-search/dialog-details.scss deleted file mode 100644 index e69de29..0000000 diff --git a/src/composer/src/app/core/components/repo-search/repo-search.component.html b/src/composer/src/app/core/components/repo-search/repo-search.component.html deleted file mode 100644 index 9b8deaa..0000000 --- a/src/composer/src/app/core/components/repo-search/repo-search.component.html +++ /dev/null @@ -1,85 +0,0 @@ -
-
- -
-
-
- search - - -
- -
- -
-
    -
  • -
    -
    - - -
    -

    {{ image.repo_name }}

    -
    -
    - star -

    {{ image.star_count | number: '2.' }}

    -
    -
    - save_alt -

    {{ image.pull_count | number: '2.' }}

    -
    -
    -
    -
    -

    {{ image.short_description }}

    - -
    - -
    -
    -
  • -
-
- -
-
-
diff --git a/src/composer/src/app/core/components/repo-search/repo-search.component.scss b/src/composer/src/app/core/components/repo-search/repo-search.component.scss deleted file mode 100644 index 4eb0a64..0000000 --- a/src/composer/src/app/core/components/repo-search/repo-search.component.scss +++ /dev/null @@ -1,130 +0,0 @@ -:host { - display: flex; - height: 100%; -} - -.search-wrapper { - width: 100%; - - .form-wrapper { - background-color: white; - padding: 15px 10px; - - .sf-magnifying-glass { - color: #d8d8d8; - margin: 6px 5px 0 0; - } - - .search-button { - min-width: unset; - } - } - - .search-results { - padding: 20px; - position: relative; - overflow-y: auto; - - ul { - margin: 0; - padding: 0; - } - } -} - -.search-result { - list-style-type: none; - padding: 20px; - margin: 5px 0 20px 0; - border: 1px solid rgba(0,0,0,.12); - border-radius: 15px; - word-wrap: break-word; - cursor: pointer; - word-break: break-word; - - h4 { - margin: 0; - } - - .button-add { - width: 60px; - } - - .image-meta { - margin-bottom: 10px; - - .green { - color: #02b52a; - } - - .gray { - color: grey; - } - - p { - margin: unset; - } - } -} - -.search-result-header { - display: flex; - align-items: center; - margin-bottom: 15px; - position: relative; -} - -.search-result-image { - width: 40px; - margin-right: 10px; -} - -.search-result-header-meta { - display: flex; - margin-top: 0; - - p { - font-size: 12px; - } - - .image-meta { - display: flex; - align-items: center; - margin-right: 10px; - margin-bottom: 0; - } - - .material-icons { - font-size: 14px; - margin-right: 5px; - } -} - -.search-result-footer { - display: flex; - - > div { - display: flex; - align-items: center; - margin-right: 10px; - - svg { - margin-right: 5px; - } - } - - &.image-meta { - margin-bottom: 0; - } -} - - -.button-add { - margin-left: auto; - font-size: 12px; - line-height: 25px; - width: auto; - min-width: 40px; - flex-shrink: 0; - float: right; -} diff --git a/src/composer/src/app/core/components/repo-search/repo-search.component.spec.ts b/src/composer/src/app/core/components/repo-search/repo-search.component.spec.ts deleted file mode 100644 index a1d1756..0000000 --- a/src/composer/src/app/core/components/repo-search/repo-search.component.spec.ts +++ /dev/null @@ -1,25 +0,0 @@ -import { async, ComponentFixture, TestBed } from '@angular/core/testing'; - -import { RepoSearchComponent } from './repo-search.component'; - -describe('RepoSearchComponent', () => { - let component: RepoSearchComponent; - let fixture: ComponentFixture; - - beforeEach(async(() => { - TestBed.configureTestingModule({ - declarations: [ RepoSearchComponent ] - }) - .compileComponents(); - })); - - beforeEach(() => { - fixture = TestBed.createComponent(RepoSearchComponent); - component = fixture.componentInstance; - fixture.detectChanges(); - }); - - it('should create', () => { - expect(component).toBeTruthy(); - }); -}); diff --git a/src/composer/src/app/core/components/repo-search/repo-search.component.ts b/src/composer/src/app/core/components/repo-search/repo-search.component.ts deleted file mode 100644 index 92fe07f..0000000 --- a/src/composer/src/app/core/components/repo-search/repo-search.component.ts +++ /dev/null @@ -1,259 +0,0 @@ -import { Component, OnInit, OnDestroy, Inject, HostListener, Output, EventEmitter } from '@angular/core' -import { Observable } from 'rxjs/Observable' -import { Store } from '@ngrx/store' -import { Subscription, Subject } from 'rxjs' -import { FormGroup, FormControl } from '@angular/forms' -import { MatDialog, MatDialogRef, MAT_DIALOG_DATA } from '@angular/material/dialog' -import { AppState } from './../../../app.state' -import { Project, RepoDetailsData, Service, IPosition } from './../../store/models' -import * as ProjectActions from './../../store/project.actions' -import { RestService } from './../../services/rest.service' -import { uuidv4 } from '../../utils/utils' -import * as GlobalSpinnerActions from './../../store/actions/global-spinner.actions' -import { NodeService } from '../canvas/jsplumb/node.service' -import { takeUntil } from 'rxjs/operators' - -@Component({ - selector: 'app-repo-search', - templateUrl: './repo-search.component.html', - styleUrls: ['./repo-search.component.scss'], -}) -export class RepoSearchComponent implements OnInit, OnDestroy { - searchForSubscription: Subscription - form: FormGroup = new FormGroup({ searchFor: new FormControl('') }) - project: Observable - - nextPage: number = 1 - prevSearchQ: string = '' - name = new FormControl('') - images = [] - repoDetailsData = {} - - showSpinner: boolean = false - - leftPosition: number = 200 - POSITION_INCREMENT: number = 20 - - private unSubscribe$ = new Subject() - - get searchFor() { - return this.form.get('searchFor').value - } - - constructor(private store: Store, public rest: RestService, public dialog: MatDialog, private nodeService: NodeService) { - this.project = store.select('project') - } - - showRepoDetails(repo_name: string): void { - this.store.dispatch(GlobalSpinnerActions.OnSpinner()) - this.rest - .getRepoDetails(repo_name) - .pipe(takeUntil(this.unSubscribe$)) - .subscribe((data: {}) => { - let full_description = data['full_description'] - data['full_description'] = full_description - this.repoDetailsData = data - - const dialogRef = this.dialog.open(DialogDetails, { - width: '800px', - data: this.repoDetailsData, - }) - dialogRef.componentInstance.onAddProject.pipe(takeUntil(this.unSubscribe$)).subscribe(() => { - this.addToProject(data) - }) - this.store.dispatch(GlobalSpinnerActions.OffSpinner()) - }) - } - - onClickOnImageCard(event, repo_name) { - const onClickOnAddButton = event.target.nodeName === 'BUTTON' || event.target.nodeName === 'SPAN' - onClickOnAddButton ? this.onAddToProject(repo_name) : this.showRepoDetails(repo_name) - } - - addToProject(data: {}) { - const uuid = uuidv4() - let imageName = data['name'] - - if (data['namespace'] && data['namespace'] !== 'library') { - imageName = data['namespace'] + '/' + imageName - } - const postfix = uuid.split('-')[2] - - this.store.dispatch( - ProjectActions.CreateService({ - data: { - uuid, - user: null, - working_dir: null, - domainname: null, - hostname: null, - ipc: null, - mac_address: null, - privileged: null, - read_only: null, - shm_size: null, - stdin_open: null, - tty: null, - name: data['serviceName'] ? `${data['serviceName']}_${postfix}` : `${data['name']}_${postfix}`, - container_name: null, - deploy: null, - image: imageName, - restart: null, - secrets: null, - ports: null, - environment: null, - links: null, - depends_on: null, - labels: null, - env_file: null, - build: null, - cap_add: null, - cap_drop: null, - cgroup_parent: null, - command: null, - args: null, - configs: null, - credential_spec: null, - devices: null, - dns: null, - dns_search: null, - entrypoint: null, - expose: null, - external_links: null, - extra_hosts: null, - healthcheck: null, - logging: null, - network_mode: null, - networks: null, - isolation: null, - init: null, - volumes: null, - tag: data['tag'] ? data['tag'] : 'latest', - }, - }), - ) - - const positionData: IPosition = { - uuid, - top: 200, - left: this.leftPosition, - zoomLevel: 1, - } - this.leftPosition += this.POSITION_INCREMENT - - this.store.dispatch( - ProjectActions.CreatePosition({ - data: positionData, - }), - ) - this.nodeService.addDynamicNode(positionData) - } - - onAddToProject(repo_name: string): void { - this.store.dispatch(GlobalSpinnerActions.OnSpinner()) - this.rest - .getRepoDetails(repo_name) - .pipe(takeUntil(this.unSubscribe$)) - .subscribe((data: {}) => { - this.addToProject(data) - }) - } - - onEnter() { - const searchValue = this.form.get('searchFor').value - - if (searchValue.length > 1) { - this.showSpinner = true - - this.nextPage = 1 - this.prevSearchQ = searchValue - - this.rest - .searchRepos(searchValue, this.nextPage) - .pipe(takeUntil(this.unSubscribe$)) - .subscribe((data: {}) => { - data['next'] ? this.nextPage++ : (this.nextPage = null) - - this.images = data['results'] - this.showSpinner = false - }) - } - } - loadMore() { - this.showSpinner = true - this.rest - .searchRepos(this.prevSearchQ, this.nextPage) - .pipe(takeUntil(this.unSubscribe$)) - .subscribe((data: {}) => { - data['next'] ? this.nextPage++ : (this.nextPage = null) - - const localCopyArrayOfImagesForMerge = this.images.map((image) => image) - - data['results'].forEach((image) => { - const index = localCopyArrayOfImagesForMerge.findIndex(({ repo_name: prevName }) => image.repo_name === prevName) - index === -1 ? this.images.push(image) : '' - }) - - this.showSpinner = false - }) - } - - ngOnInit(): void { - this.form - .get('searchFor') - .valueChanges.pipe(takeUntil(this.unSubscribe$)) - .subscribe((val) => { - const formattedMessage = val - }) - } - - ngOnDestroy() { - this.unSubscribe$.next(true) - this.unSubscribe$.complete() - } - - addCustom(): void { - const dialogRef = this.dialog.open(DialogAddCustom, { - width: '400px' - }) - dialogRef.componentInstance.onAddProject.pipe(takeUntil(this.unSubscribe$)).subscribe((data) => { - this.addToProject(data) - dialogRef.close() - }) - } -} - -@Component({ - selector: 'dialog-details', - templateUrl: 'dialog-details.html', - styleUrls: ['./dialog-details.scss'], -}) -export class DialogDetails { - @Output() onAddProject = new EventEmitter() - constructor(public dialogRef: MatDialogRef, @Inject(MAT_DIALOG_DATA) public data: RepoDetailsData) {} - - onNoClick(): void { - this.dialogRef.close() - } - - addProject() { - this.onAddProject.emit() - } -} - -@Component({ - selector: 'dialog-add-custom', - templateUrl: 'dialog-add-custom.html', - styleUrls: ['./dialog-add-custom.scss'], -}) -export class DialogAddCustom { - @Output() onAddProject = new EventEmitter() - imageName: string - tag: string - serviceName: string - constructor(public dialogRef: MatDialogRef) {} - - addProject() { - this.onAddProject.emit({name: this.imageName, tag: this.tag, serviceName: this.serviceName}) - } -} diff --git a/src/composer/src/app/core/components/side-bar/side-bar.component.html b/src/composer/src/app/core/components/side-bar/side-bar.component.html deleted file mode 100644 index 5f3c94e..0000000 --- a/src/composer/src/app/core/components/side-bar/side-bar.component.html +++ /dev/null @@ -1,53 +0,0 @@ - diff --git a/src/composer/src/app/core/components/side-bar/side-bar.component.scss b/src/composer/src/app/core/components/side-bar/side-bar.component.scss deleted file mode 100644 index cb9cdc3..0000000 --- a/src/composer/src/app/core/components/side-bar/side-bar.component.scss +++ /dev/null @@ -1,75 +0,0 @@ -.side-bar { - width: 60px; - background-color: #0049b0; - color: white; - box-shadow: none; - visibility: hidden; - display: none !important; - transform: translate3d(-100%, 0, 0); - transition: all 300ms; - - &__item { - height: 70px; - cursor: pointer; - &__wrapper { - width: 30px; - height: 30px; - border-radius: 30px; - - img { - width: 20px; - } - } - - &__user-logo { - &__wrapper { - position: relative; - width: 40px; - - .initials-wrap { - color: #0048b0; - background: #fff; - width: 40px; - display: inline-block; - height: 40px; - border-radius: 20px; - text-align: center; - font-size: 18px; - font-weight: 500; - line-height: 40px; - } - } - &__initials { - color: #fff; - position: absolute; - font-weight: 900; - font-size: 20px; - margin: 0; - width: 40px; - height: 40px; - text-align: center; - margin-top: 11px; - } - img { - width: 40px; - } - } - } - &.opened { - transform: none; - visibility: visible; - display: flex !important; - } -} - -.search-toggle-item.opened { - background-color: #023e94; -} - -.logo { - height: 92px; - cursor: pointer; - img { - width: 30px; - } -} diff --git a/src/composer/src/app/core/components/side-bar/side-bar.component.spec.ts b/src/composer/src/app/core/components/side-bar/side-bar.component.spec.ts deleted file mode 100644 index fd6c49e..0000000 --- a/src/composer/src/app/core/components/side-bar/side-bar.component.spec.ts +++ /dev/null @@ -1,25 +0,0 @@ -import { async, ComponentFixture, TestBed } from '@angular/core/testing'; - -import { SideBarComponent } from './side-bar.component'; - -describe('SideBarComponent', () => { - let component: SideBarComponent; - let fixture: ComponentFixture; - - beforeEach(async(() => { - TestBed.configureTestingModule({ - declarations: [ SideBarComponent ] - }) - .compileComponents(); - })); - - beforeEach(() => { - fixture = TestBed.createComponent(SideBarComponent); - component = fixture.componentInstance; - fixture.detectChanges(); - }); - - it('should create', () => { - expect(component).toBeTruthy(); - }); -}); diff --git a/src/composer/src/app/core/components/side-bar/side-bar.component.ts b/src/composer/src/app/core/components/side-bar/side-bar.component.ts deleted file mode 100644 index da10493..0000000 --- a/src/composer/src/app/core/components/side-bar/side-bar.component.ts +++ /dev/null @@ -1,75 +0,0 @@ -import { Component, OnInit, Input, EventEmitter, Output, OnDestroy, HostListener } from '@angular/core' -import { Router } from '@angular/router' -import { AppState } from './../../../app.state' -import { Store } from '@ngrx/store' -import * as ProjectActions from './../../store/project.actions' -import { AuthenticationService } from '../../services/authentication.service' -import { MatDialog } from '@angular/material/dialog' -import { ManageUserDialogComponent } from '../dialogs/manage-user-dialog/manage-user-dialog.component' - -interface sideBarToogleEvent { - name: string -} - -@Component({ - selector: 'app-side-bar', - templateUrl: './side-bar.component.html', - styleUrls: ['./side-bar.component.scss'], -}) -export class SideBarComponent implements OnInit { - @Input() currentOpenedSideNavItem: string - @Input() sidebarStatus: boolean - @Output() toogleSideBarItem: EventEmitter = new EventEmitter() - - screenWidth: number - userInitials: string - - constructor(private router: Router, private store: Store, private authenticationService: AuthenticationService, private dialog: MatDialog) {} - - ngOnInit(): void { - this.screenWidth = window.innerWidth - this.authenticationService.currentUser.subscribe((user) => { - if (user) { - const {first_name, last_name, email} = user - if(first_name && last_name) { - this.userInitials = first_name[0] + last_name[0] - } else { - this.userInitials = email[0] + email[1] - } - } - }) - } - - onClickLogo() { - this.store.dispatch(ProjectActions.SetInitialState()) - - if (this.screenWidth > 600) { - this.router.navigate(['/']) - } else { - this.sidebarStatus = false - this.toogleSideBarItem.emit({name: null}) - } - } - - @HostListener('window:resize', ['$event']) - onResize(event) { - this.screenWidth = window.innerWidth - } - - handelClick() { - const currentUser = this.authenticationService.currentUserValue - if(currentUser) { - this.dialog.open(ManageUserDialogComponent, { - width: '20%', - minWidth: '340px', - position: { - left: '70px', - bottom: '70px' - }, - data: currentUser - }) - } else { - this.router.navigate(['/auth/login/']) - } - } -} diff --git a/src/composer/src/app/core/components/widgets/check-circle/check-circle.component.html b/src/composer/src/app/core/components/widgets/check-circle/check-circle.component.html deleted file mode 100644 index 4786c9c..0000000 --- a/src/composer/src/app/core/components/widgets/check-circle/check-circle.component.html +++ /dev/null @@ -1,5 +0,0 @@ -
- -
\ No newline at end of file diff --git a/src/composer/src/app/core/components/widgets/check-circle/check-circle.component.scss b/src/composer/src/app/core/components/widgets/check-circle/check-circle.component.scss deleted file mode 100644 index 70ee65a..0000000 --- a/src/composer/src/app/core/components/widgets/check-circle/check-circle.component.scss +++ /dev/null @@ -1,14 +0,0 @@ -.green-circle { - width: 24px; - height: 24px; - padding: 6px 0 0 0; - opacity: 0; - transition: linear 1s opacity; - - &_invisible { - opacity: 0; - } - &_visible { - opacity: 1; - } -} \ No newline at end of file diff --git a/src/composer/src/app/core/components/widgets/check-circle/check-circle.component.ts b/src/composer/src/app/core/components/widgets/check-circle/check-circle.component.ts deleted file mode 100644 index e6a4065..0000000 --- a/src/composer/src/app/core/components/widgets/check-circle/check-circle.component.ts +++ /dev/null @@ -1,18 +0,0 @@ -import { Component, OnInit } from '@angular/core' - -@Component({ - selector: 'check-circle', - templateUrl: './check-circle.component.html', - styleUrls: ['./check-circle.component.scss'], -}) -export class CheckCircleComponent implements OnInit { - visible: boolean = false - constructor() {} - - ngOnInit(): void {} - - showCircle() { - this.visible = true - setTimeout(() => this.visible = false, 2000) - } -} diff --git a/src/composer/src/app/core/components/widgets/spinner/spinner.component.html b/src/composer/src/app/core/components/widgets/spinner/spinner.component.html deleted file mode 100644 index 910ee20..0000000 --- a/src/composer/src/app/core/components/widgets/spinner/spinner.component.html +++ /dev/null @@ -1,5 +0,0 @@ -
-
- -
-
\ No newline at end of file diff --git a/src/composer/src/app/core/components/widgets/spinner/spinner.component.scss b/src/composer/src/app/core/components/widgets/spinner/spinner.component.scss deleted file mode 100644 index 25f3bde..0000000 --- a/src/composer/src/app/core/components/widgets/spinner/spinner.component.scss +++ /dev/null @@ -1,14 +0,0 @@ -.spinner-wrapper { - position: absolute; - top: 0; - bottom: 0; - right: 0; - left: 0; - background: rgba(0, 0, 0, 0.2); - z-index: 3; -} - - -.global-spinner-wrp { - z-index: 3; -} \ No newline at end of file diff --git a/src/composer/src/app/core/components/widgets/spinner/spinner.component.spec.ts b/src/composer/src/app/core/components/widgets/spinner/spinner.component.spec.ts deleted file mode 100644 index a51f575..0000000 --- a/src/composer/src/app/core/components/widgets/spinner/spinner.component.spec.ts +++ /dev/null @@ -1,25 +0,0 @@ -import { async, ComponentFixture, TestBed } from '@angular/core/testing'; - -import { SpinnerComponent } from './spinner.component'; - -describe('SpinnerComponent', () => { - let component: SpinnerComponent; - let fixture: ComponentFixture; - - beforeEach(async(() => { - TestBed.configureTestingModule({ - declarations: [ SpinnerComponent ] - }) - .compileComponents(); - })); - - beforeEach(() => { - fixture = TestBed.createComponent(SpinnerComponent); - component = fixture.componentInstance; - fixture.detectChanges(); - }); - - it('should create', () => { - expect(component).toBeTruthy(); - }); -}); diff --git a/src/composer/src/app/core/components/widgets/spinner/spinner.component.ts b/src/composer/src/app/core/components/widgets/spinner/spinner.component.ts deleted file mode 100644 index a37fa24..0000000 --- a/src/composer/src/app/core/components/widgets/spinner/spinner.component.ts +++ /dev/null @@ -1,23 +0,0 @@ -import { Component, OnInit } from '@angular/core' -import { Store } from '@ngrx/store' -import { AppState } from '../../../../app.state' -import { Subject } from 'rxjs' -import { takeUntil } from 'rxjs/operators' - - -@Component({ - selector: 'app-spinner', - templateUrl: './spinner.component.html', - styleUrls: ['./spinner.component.scss'], -}) -export class SpinnerComponent implements OnInit { - - private unSubscribe$ = new Subject() - showGlobalSpinner: boolean = true - - constructor(private store: Store) { - this.store.select('globalSpinnerState').pipe(takeUntil(this.unSubscribe$)).subscribe((value) => (this.showGlobalSpinner = value)) - } - - ngOnInit(): void {} -} diff --git a/src/composer/src/app/core/helpers/auth.gourd.ts b/src/composer/src/app/core/helpers/auth.gourd.ts deleted file mode 100644 index 2dd1611..0000000 --- a/src/composer/src/app/core/helpers/auth.gourd.ts +++ /dev/null @@ -1,36 +0,0 @@ -import { Injectable } from '@angular/core'; -import { Router, CanActivate, ActivatedRouteSnapshot, RouterStateSnapshot, NavigationEnd } from '@angular/router'; -import { filter } from 'rxjs/operators' -import { AuthenticationService } from '../services/authentication.service'; - -@Injectable({ providedIn: 'root' }) -export class AuthGuard implements CanActivate { - lastPageBeforeRegistrationOrLogin: string - mapAuthUrls = { - '/auth/social/github/callback': true, - '/auth/login': true, - '/auth/registration': true - } - constructor( - private router: Router, - private authenticationService: AuthenticationService - ) { - this.router.events - .pipe(filter(event => event instanceof NavigationEnd)) - .subscribe(({ urlAfterRedirects }: NavigationEnd) => { - !this.mapAuthUrls[location.pathname] ? localStorage.setItem('lastPageBeforeRegistrationOrLogin', location.pathname) : '' - }); - } - - canActivate(route: ActivatedRouteSnapshot, state: RouterStateSnapshot) { - const currentUser = this.authenticationService.currentUserValue; - if (currentUser) { - // logged in so return true - return true; - } - - // not logged in so redirect to login page with the return url - this.router.navigate(['/login'], { queryParams: { returnUrl: state.url } }); - return false; - } -} \ No newline at end of file diff --git a/src/composer/src/app/core/helpers/error.interceptor.ts b/src/composer/src/app/core/helpers/error.interceptor.ts deleted file mode 100644 index 56e5481..0000000 --- a/src/composer/src/app/core/helpers/error.interceptor.ts +++ /dev/null @@ -1,36 +0,0 @@ -import { Injectable } from '@angular/core'; -import { Router } from '@angular/router'; -import { HttpRequest, HttpHandler, HttpEvent, HttpInterceptor } from '@angular/common/http'; -import { Observable, throwError } from 'rxjs'; -import { catchError } from 'rxjs/operators'; - -import { AuthenticationService } from '../services/authentication.service'; - -@Injectable() -export class ErrorInterceptor implements HttpInterceptor { - constructor(private authenticationService: AuthenticationService, private router: Router) { } - - intercept(request: HttpRequest, next: HttpHandler): Observable> { - return next.handle(request).pipe(catchError(err => { - if (err.status === 401) { - // auto logout if 401 response returned from api - this.authenticationService.removeUser() - this.router.navigate(['/auth/login']) - } else if (err.status === 400) { - const errorsMessages = [] - Object.keys(err.error).forEach((key) => { - if(Array.isArray(err.error[key])) { - err.error[key].forEach(errorMessage => { - errorsMessages.push(errorMessage) - }) - } - else { - errorsMessages.push(err.error[key]) - } - }) - return throwError(errorsMessages); - } - return throwError(err); - })) - } -} \ No newline at end of file diff --git a/src/composer/src/app/core/helpers/jwt.interceptor.ts b/src/composer/src/app/core/helpers/jwt.interceptor.ts deleted file mode 100644 index b2874da..0000000 --- a/src/composer/src/app/core/helpers/jwt.interceptor.ts +++ /dev/null @@ -1,27 +0,0 @@ -import { Injectable } from '@angular/core'; -import { HttpRequest, HttpHandler, HttpEvent, HttpInterceptor } from '@angular/common/http'; -import { Observable } from 'rxjs'; - -import { environment } from 'src/environments/environment'; -import { AuthenticationService } from '../services/authentication.service'; - -@Injectable() -export class JwtInterceptor implements HttpInterceptor { - constructor(private authenticationService: AuthenticationService) { } - - intercept(request: HttpRequest, next: HttpHandler): Observable> { - // add auth header with jwt if user is logged in and request is to the api url - const currentUser = this.authenticationService.currentUserValue; - const isLoggedIn = currentUser && currentUser.token; - const isApiUrl = request.url.startsWith(environment.apiUrl); - if (isLoggedIn && isApiUrl && (!request.url.includes('social/github/login') || request.url === `${environment.apiUrl}/recipe/`)) { - request = request.clone({ - setHeaders: { - Authorization: `JWT ${currentUser.token}` - } - }); - } - - return next.handle(request); - } -} \ No newline at end of file diff --git a/src/composer/src/app/core/pipe/marked.pipe.ts b/src/composer/src/app/core/pipe/marked.pipe.ts deleted file mode 100644 index 1b01cce..0000000 --- a/src/composer/src/app/core/pipe/marked.pipe.ts +++ /dev/null @@ -1,14 +0,0 @@ -import { Pipe, PipeTransform } from '@angular/core' -import * as marked from 'marked' - -@Pipe({ - name: 'marked', -}) -export class MarkedPipe implements PipeTransform { - transform(value: any): any { - if (value && value.length > 0) { - return marked(value) - } - return value - } -} diff --git a/src/composer/src/app/core/pipe/truncate-text.pipe.spec.ts b/src/composer/src/app/core/pipe/truncate-text.pipe.spec.ts deleted file mode 100644 index f36327b..0000000 --- a/src/composer/src/app/core/pipe/truncate-text.pipe.spec.ts +++ /dev/null @@ -1,8 +0,0 @@ -import { TruncateTextPipe } from './truncate-text.pipe'; - -describe('TruncateTextPipe', () => { - it('create an instance', () => { - const pipe = new TruncateTextPipe(); - expect(pipe).toBeTruthy(); - }); -}); diff --git a/src/composer/src/app/core/pipe/truncate-text.pipe.ts b/src/composer/src/app/core/pipe/truncate-text.pipe.ts deleted file mode 100644 index 8ea22d2..0000000 --- a/src/composer/src/app/core/pipe/truncate-text.pipe.ts +++ /dev/null @@ -1,38 +0,0 @@ -import { Pipe, PipeTransform } from '@angular/core'; - -@Pipe({ - name: 'truncateText' -}) -export class TruncateTextPipe implements PipeTransform { - - transform(value: string, length: number, wordBoundary: boolean): string { - const biggestWord = 50 - const elipses = "..." - - let truncatedText = '' - - if (wordBoundary) { - if (typeof value === "undefined") return value - if (value.length <= length) return value - - let truncatedText = value.slice(0, length + biggestWord) - - while (truncatedText.length > length - elipses.length) { - let lastSpace = truncatedText.lastIndexOf(" ") - if (lastSpace === -1) break - truncatedText = truncatedText.slice(0, lastSpace).replace(/[!,.?;:]$/, '') - } - - truncatedText = truncatedText + '…' - } else { - if (value) { - truncatedText = (value.length > length) ? value.substr(0, length-1) + '…' : value; - } else { - truncatedText = '' - } - } - - return truncatedText - } - -} diff --git a/src/composer/src/app/core/services/authentication.service.ts b/src/composer/src/app/core/services/authentication.service.ts deleted file mode 100644 index a8e92e6..0000000 --- a/src/composer/src/app/core/services/authentication.service.ts +++ /dev/null @@ -1,78 +0,0 @@ -import { Injectable } from '@angular/core'; -import { HttpClient } from '@angular/common/http'; -import { BehaviorSubject, Observable } from 'rxjs'; -import { map } from 'rxjs/operators' -import { environment } from 'src/environments/environment' -import { User } from '../store/models'; - -@Injectable({ - providedIn: 'root', - }) -export class AuthenticationService { - private currentUserSubject: BehaviorSubject; - public currentUser: Observable; - - constructor(private http: HttpClient) { - this.currentUserSubject = new BehaviorSubject(JSON.parse(localStorage.getItem('currentUser'))); - this.currentUser = this.currentUserSubject.asObservable(); - } - - public get currentUserValue(): User { - return this.currentUserSubject.value; - } - - login(data: { email: string, password: string }) { - return this.http.post(`${environment.apiUrl}/auth/login/`, data) - .pipe(map(({token, user}) => { - user.token = token - // store user details and jwt token in local storage to keep user logged in between page refreshes - localStorage.setItem('currentUser', JSON.stringify(user)); - this.currentUserSubject.next(user); - return user; - })); - } - - registration(data: {email: string, password: string, confirmPassword: string}) { - return this.http.post(`${environment.apiUrl}/auth/registration/`, data) - .pipe(map(({token, user}) => { - user.token = token - // store user details and jwt token in local storage to keep user logged in between page refreshes - // localStorage.setItem('currentUser', JSON.stringify(user)); - // this.currentUserSubject.next(user); - return user; - })); - } - - logout() { - return this.http.post(`${environment.apiUrl}/auth/logout/`, {}) - .pipe(map(() => { - localStorage.removeItem('currentUser'); - this.currentUserSubject.next(null); - })); - - } - - gitHubLogin(code: string) { - return this.http.post(`${environment.apiUrl}/auth/social/github/login/`, { code: code }) - .pipe(map(({token, user}) => { - user.token = token - // store user details and jwt token in local storage to keep user logged in between page refreshes - localStorage.setItem('currentUser', JSON.stringify(user)); - this.currentUserSubject.next(user); - return user; - })); - } - - gitHubAuth() { - return this.http.post(`${environment.apiUrl}/auth/social/github/auth-server/`, {}) - .pipe(map(({url}) => { - location.href = url - return - })); - } - - removeUser() { - localStorage.removeItem('currentUser'); - this.currentUserSubject.next(null); - } -} \ No newline at end of file diff --git a/src/composer/src/app/core/services/event-emitter.service.ts b/src/composer/src/app/core/services/event-emitter.service.ts deleted file mode 100644 index 3ae1e4a..0000000 --- a/src/composer/src/app/core/services/event-emitter.service.ts +++ /dev/null @@ -1,48 +0,0 @@ -import { Injectable } from '@angular/core'; - -@Injectable({ - providedIn: 'root' -}) -export class EventEmitterService { - - constructor() { } - - events = {}; - api = { - broadcast: this.broadcast, - subscribe: this.subscribe - }; - - broadcast(event, data) { - if (!this.events[event]) return; - - this.events[event].forEach(callback => { - - return callback(data); - }); - } - - subscribe(eventName, callback) { - if (!this.events[eventName]) { - this.events[eventName] = []; - } - - this.events[eventName].push(callback); - - return this.generateUnsubscribe(eventName, callback); - } - - generateUnsubscribe(eventName, callback) { - const _this = this; - return function unsubscribe() { - var callbacks = _this.events[eventName]; - - for(var i = 0; i < callbacks.length; i++) { - if (callbacks[i] == callback) { - callbacks.splice(i, 1); - return; - } - } - }; - } -} diff --git a/src/composer/src/app/core/services/rest.service.ts b/src/composer/src/app/core/services/rest.service.ts deleted file mode 100644 index c6d1c28..0000000 --- a/src/composer/src/app/core/services/rest.service.ts +++ /dev/null @@ -1,117 +0,0 @@ -import { Injectable } from '@angular/core' -import { HttpClient, HttpHeaders, HttpErrorResponse } from '@angular/common/http' -import { Observable, of } from 'rxjs' -import { map, catchError, tap } from 'rxjs/operators' -import { environment } from 'src/environments/environment' -import { Recipe } from '../store/models' - -const baseSearchUrl = `${environment.apiUrl}/repo/search` -const baseDetailUrl = `${environment.apiUrl}/repo/detail` -const baseTagsUrl = `${environment.apiUrl}/tags` -const generateCodeUrl = `${environment.apiUrl}/generate` -const baseProjectUrl = `${environment.apiUrl}/project` -const baseRecipeUrl = `${environment.apiUrl}/recipe` -const baseUrl = environment.apiUrl - -const httpOptions = { - headers: new HttpHeaders({ - 'Content-Type': 'application/json', - }), -} - -@Injectable({ - providedIn: 'root', -}) -export class RestService { - constructor(private http: HttpClient) {} - - private extractData(res: Response) { - let body = res - return body || {} - } - - searchRepos(searchFor: string, pageNumber: number): Observable { - return this.http.get(`${baseSearchUrl}/?q=${searchFor}&page=${pageNumber}`, httpOptions).pipe(map(this.extractData)) - } - - getRepoDetails(repoName: string): Observable { - return this.http.get(`${baseDetailUrl}/?r=${repoName}`, httpOptions).pipe(map(this.extractData)) - } - - getRepoTags(repoName: string, pageNumber: number, searchSubStr: string): Observable { - return this.http.get(`${baseTagsUrl}/?r=${repoName}&s=${searchSubStr}&page=${pageNumber}`, httpOptions).pipe(map(this.extractData)) - } - - generateCode(data: object): Observable { - return this.http.post(`${generateCodeUrl}/`, JSON.stringify(data), httpOptions).pipe(map(this.extractData)) - } - - createProject(data: object): Observable { - return this.http.post(`${baseProjectUrl}/`, JSON.stringify(data), httpOptions).pipe(map(this.extractData)) - } - - saveProject(data: object, uuid: string): Observable { - return this.http.put(`${baseProjectUrl}/${uuid}/`, JSON.stringify(data), httpOptions).pipe(map(this.extractData)) - } - - getProject(uuid: string): Observable { - return this.http.get(`${baseProjectUrl}/${uuid}/`, httpOptions).pipe(map(this.extractData)) - } - - importProject(data: object): Observable { - return this.http.post(`${baseUrl}/import/`, data, httpOptions).pipe(map(this.extractData)) - } - - getUserProjects(): Observable { - return this.http.get(`${baseProjectUrl}/private/`, httpOptions).pipe(map(this.extractData)) - } - - removeProject(uuid: string): Observable { - return this.http.delete(`${baseProjectUrl}/private/${uuid}/`) - } - - getPopularRecipes(): Observable { - return this.http.get(`${baseRecipeUrl}/popular/`, httpOptions).pipe(map(this.extractData)) - } - - getUserRecipes(): Observable { - return this.http.get(`${baseRecipeUrl}/`, httpOptions).pipe(map(this.extractData)) - } - - searchRecipes(searchFor: string, pageNumber: number): Observable { - return this.http.get(`${baseRecipeUrl}/search/?q=${searchFor}&page=${pageNumber}`, httpOptions).pipe(map(this.extractData)) - } - - createRecipe(data: object): Observable { - return this.http.post(`${baseRecipeUrl}/`, data, httpOptions).pipe(map(this.extractData)) - } - - updateRecipe(uuid: string, data: object) { - return this.http.put(`${baseRecipeUrl}/${uuid}/`, data, httpOptions).pipe(map(this.extractData)) - } - - getRecipe(uuid: string): Observable { - return this.http.get(`${baseRecipeUrl}/${uuid}/`, httpOptions).pipe(map(this.extractData)) - } - - removeRecipe(uuid: string): Observable { - return this.http.delete(`${baseRecipeUrl}/${uuid}/`, httpOptions).pipe(map(this.extractData)) - } - - sendEmailForNotification(data: object): Observable { - return this.http.post(`https://y5s47rsecb.execute-api.us-east-1.amazonaws.com/web_public/nuxx/`, data, httpOptions).pipe(map(this.extractData)) - } - - private handleError(operation = 'operation', result?: T) { - return (error: any): Observable => { - // TODO: send the error to remote logging infrastructure - console.error(error) // log to console instead - - // TODO: better job of transforming error for user consumption - console.log(`${operation} failed: ${error.message}`) - - // Let the app keep running by returning an empty result. - return of(result as T) - } - } -} diff --git a/src/composer/src/app/core/store/actions/global-app-configuration.actions.ts b/src/composer/src/app/core/store/actions/global-app-configuration.actions.ts deleted file mode 100644 index eb4d333..0000000 --- a/src/composer/src/app/core/store/actions/global-app-configuration.actions.ts +++ /dev/null @@ -1,9 +0,0 @@ -import { createAction } from '@ngrx/store' - -export const ON_COMPOSE_MODE = '[GLOBAL_CONFIGURATION] Compose mode' -export const ON_IMPORT_MODE = '[GLOBAL_CONFIGURATION] Import mode' -export const ON_RECIPE_LOAD_MODE = '[GLOBAL_CONFIGURATION] Recipe load mode' - -export const OnComposeMode = createAction(ON_COMPOSE_MODE) -export const OnImportMode = createAction(ON_IMPORT_MODE) -export const OnRecipeLoadMode = createAction(ON_RECIPE_LOAD_MODE) \ No newline at end of file diff --git a/src/composer/src/app/core/store/actions/global-dialog.actions.ts b/src/composer/src/app/core/store/actions/global-dialog.actions.ts deleted file mode 100644 index 3d59c30..0000000 --- a/src/composer/src/app/core/store/actions/global-dialog.actions.ts +++ /dev/null @@ -1,7 +0,0 @@ -import { createAction, props } from '@ngrx/store' - -export const PROJECT_NOT_FOUND = '[GLOBAL_ERROR] Project not found' -export const SERVER_ERROR = '[GLOBAL_SPINNER] Server error' - -export const ProjectNotFound = createAction(PROJECT_NOT_FOUND, props<{ message: string, _type: string }>()) -export const ServerError = createAction(SERVER_ERROR, props<{ message: string, _type: string }>()) \ No newline at end of file diff --git a/src/composer/src/app/core/store/actions/global-spinner.actions.ts b/src/composer/src/app/core/store/actions/global-spinner.actions.ts deleted file mode 100644 index 87d7418..0000000 --- a/src/composer/src/app/core/store/actions/global-spinner.actions.ts +++ /dev/null @@ -1,7 +0,0 @@ -import { createAction } from '@ngrx/store' - -export const ON_SPINNER = '[GLOBAL_SPINNER] On spinner' -export const OFF_SPINNER = '[GLOBAL_SPINNER] Off spinner' - -export const OnSpinner = createAction(ON_SPINNER) -export const OffSpinner = createAction(OFF_SPINNER) \ No newline at end of file diff --git a/src/composer/src/app/core/store/effects/projects.effects.ts b/src/composer/src/app/core/store/effects/projects.effects.ts deleted file mode 100644 index 2142477..0000000 --- a/src/composer/src/app/core/store/effects/projects.effects.ts +++ /dev/null @@ -1,186 +0,0 @@ -import { Injectable } from '@angular/core' -import { Location } from '@angular/common' -import { Router } from '@angular/router' - -import { Effect, Actions, ofType } from '@ngrx/effects' -import { of } from 'rxjs/observable/of' -import { map, switchMap, catchError, withLatestFrom } from 'rxjs/operators' -import { Store } from '@ngrx/store' - -import { AppState } from '../../../app.state' -import * as ProjectActions from '../project.actions' -import * as GlobalError from '../actions/global-dialog.actions' -import * as globalAppConfiguration from '../actions/global-app-configuration.actions' -import { RestService } from '../../services/rest.service' -import { EventEmitterService } from '../../services/event-emitter.service' - -@Injectable() -export class ProjectEffects { - constructor(private actions$: Actions, private restService: RestService, private store: Store, private location: Location, private eventEmitterService: EventEmitterService, private router: Router) {} - @Effect() - ViewProject$ = this.actions$.pipe( - ofType(ProjectActions.ViewRecipe), - switchMap(({ data }) => { - return this.restService.getRecipe(data).pipe( - map((projectData) => { - const project = { - name: projectData.name, - uuid: projectData.uuid, - mutable: projectData.mutable, - ...projectData.data, - } - this.eventEmitterService.broadcast('initialize:node', {}) - this.eventEmitterService.broadcast('load: code', projectData.code) - return ProjectActions.SaveProjectSuccess({ data: project }) - }), - catchError((error) => { - if (error.status >= 500) return of(GlobalError.ServerError({message: error.error, _type: 'Error'})) - else return of(ProjectActions.ApiProjectRequestFail({ data: error })) - }) - ) - }), - ) - - @Effect() - ImportPoject$ = this.actions$.pipe( - ofType(ProjectActions.ImportProject), - switchMap(({ data }) => { - let payload = {} - payload[data.type] = data.payload - return this.restService.importProject(payload).pipe( - map((projectData) => { - const project = { - name: projectData.name, - uuid: projectData.uuid, - mutable: projectData.mutable, - ...projectData.data, - } - this.eventEmitterService.broadcast('load: code', projectData.code) - return ProjectActions.SaveProjectSuccess({ data: project }) - }), - catchError((error) => { - if (error.status >= 500) return of(GlobalError.ServerError({message: error.error, _type: 'Error'})) - else return of(ProjectActions.ApiProjectRequestFail({ data: error })) - }) - ) - }), - ) - - @Effect() - ImportCurrentPoject$ = this.actions$.pipe( - ofType(ProjectActions.ImportCurrentProject), - withLatestFrom(this.store.select('project')), - switchMap(([action, payload]) => { - const currentProject = { - name: payload.name, - uuid: payload.uuid, - mutable: payload.mutable, - data: { - canvas: payload.canvas, - version: payload.version, - volumes: payload.volumes, - services: payload.services, - networks: payload.networks, - secrets: payload.secrets, - }, - } - - return this.restService.importProject({project_data: currentProject, yaml: action.data.payload}).pipe( - map((projectData) => { - if (projectData.data) { - const project = { - name: currentProject.name ? currentProject.name : projectData.name, - uuid: currentProject.uuid, - mutable: projectData.mutable, - ...projectData.data, - } - this.eventEmitterService.broadcast('load: code', projectData.code) - this.store.dispatch(ProjectActions.SaveProjectSuccess({ data: project })) - return ProjectActions.SaveProject() - } else { - const { name, uuid, mutable } = currentProject - this.eventEmitterService.broadcast('load: code', projectData.code) - this.store.dispatch(ProjectActions.SetInitialState()) - this.store.dispatch(ProjectActions.SaveProjectSuccess({ data: { uuid, name, mutable} })) - return ProjectActions.SaveProject() - } - }), - catchError((error) => { - if (error.status >= 500) return of(GlobalError.ServerError({message: `${error.error}`, _type: 'Error'})) - else if (error.status > 401) return of(globalAppConfiguration.OnImportMode(), GlobalError.ServerError({message: error.error, _type: 'Error'})) - else return of(ProjectActions.ApiProjectRequestFail({ data: error })) - }) - ) - }), - ) - - @Effect() - UploadProject$ = this.actions$.pipe( - ofType(ProjectActions.UploadProject), - switchMap(({ data }) => { - return this.restService.getProject(data).pipe( - map((projectData) => { - const project = { - name: projectData.name, - uuid: projectData.uuid, - mutable: projectData.mutable, - ...projectData.data, - } - return ProjectActions.SaveProjectSuccess({ data: project }) - }), - catchError((error) => { - if (error.status === 404) return of(GlobalError.ProjectNotFound({message: `Project with id: ${data} does not exist, or is private.`, _type: 'Error'})) - else if (error.status >= 500) return of(GlobalError.ServerError({message: `Server encountered an error:${error.message}`, _type: 'Error'})) - else return of(ProjectActions.ApiProjectRequestFail({ data: error })) - }) - ) - }), - ) - - @Effect() - SaveProject$ = this.actions$.pipe( - ofType(ProjectActions.SAVE_PROJECT), - withLatestFrom(this.store.select('project')), - switchMap(([action, payload]) => { - const currentProject = { - name: payload.name ? payload.name : `New project`, - data: { - canvas: payload.canvas, - version: payload.version, - volumes: payload.volumes, - services: payload.services, - networks: payload.networks, - secrets: payload.secrets, - }, - } - - if (payload.uuid && payload.mutable) { - return this.restService.saveProject(currentProject, payload.uuid).pipe( - map((projectData) => { - const project = { - name: projectData.name, - mutable: projectData.mutable, - ...projectData.data, - } - return ProjectActions.SaveProjectSuccess({ data: project }) - }), - catchError((error) => of(ProjectActions.ApiProjectRequestFail({ data: error }))), - ) - } else { - return this.restService.createProject(currentProject).pipe( - map((projectData) => { - const project = { - name: projectData.name, - uuid: projectData.uuid, - mutable: projectData.mutable, - ...projectData.data, - } - window.location.href = `/${projectData.uuid}` - return ProjectActions.SaveProjectSuccess({ data: project }) - }), - catchError((error) => of(ProjectActions.ApiProjectRequestFail({ data: error }))), - ) - } - }), - ) -} diff --git a/src/composer/src/app/core/store/models.ts b/src/composer/src/app/core/store/models.ts deleted file mode 100644 index 463ec4d..0000000 --- a/src/composer/src/app/core/store/models.ts +++ /dev/null @@ -1,249 +0,0 @@ -export interface ServiceBuildObject { - context: string - dockerfile: string - args: [] - cache_from: [] - labels: [] - network: string - shm_size: any - target: string -} - -export interface ServiceHealthCheck { - test: [] - interval: string - timeout: string - retries: number - start_period: string -} - -export interface ServiceLogging { - driver: string - options: { - 'syslog-address': string - 'max-size': string - 'max-file': string - } -} - -export interface ServicePort { - target: number - published: number - protocol: string - mode: string -} - -export interface KeyValuePair { - key: string - value: string -} - -export interface ServiceDeploy { - mode: string - replicas: number - labels: KeyValuePair[] - update_config: { - parallelism: number - delay: string - order: string - } - rollback_config: string - restart_policy: { - condition: string - delay: string - max_attempts: number - window: string - } - endpoint_mode: string - placement: { - max_replicas_per_node: number - constraints: [] - preferences: KeyValuePair[] - } - resources: { - limits: { - cpus: string - memory: string - } - reservations: { - cpus: string - memory: string - } - } -} - -export interface Service { - user: string - working_dir: string - domainname: string - hostname: string - ipc: string - mac_address: string - privileged: boolean - read_only: boolean - shm_size: string - stdin_open: boolean - tty: boolean - uuid: string - name: string - init: boolean - isolation: string - container_name?: string - deploy: ServiceDeploy - build: ServiceBuildObject - image: string - restart: string - secrets: [] - ports: Array<{}> - environment: Array<{}> - links: [] - depends_on: any[] - labels: [] - env_file: [] - cap_add: [] - cap_drop: [] - cgroup_parent: string - command: [] - args: [] - configs: [] // from 3.3 - credential_spec: any // from 3.3 usage of config: from 3.8 - devices: [] - dns: [] - dns_search: [] - entrypoint: [] - expose: [] - external_links: [] - extra_hosts: [] - healthcheck: ServiceHealthCheck - logging: ServiceLogging - network_mode: string - networks: [] // string or object - volumes: Array<{}>, - tag: string -} - -export interface Volume { - uuid: string - name: string // mapped to "object name" in the final generated code - volume_name?: string // mapped to "name" in the final generated code - driver: string - driver_opts: { - "type": string - o: string - device: string - } - external: boolean, - external_name: string, - labels: KeyValuePair[] -} - -export interface Network { - uuid: string - object_name: string - name: string - driver: string - driver_custom: boolean - type: string // hostnet or nonet - driver_opts: KeyValuePair[] - attachable: boolean - enable_ipv6: boolean // 2 and below - ipam: { - driver: string - config: [] // a list key/val pairs ex. subnet: 172.28.0.0/16 - } - internal: boolean - external: boolean, - external_name: string, - labels: KeyValuePair[] -} - -export interface Secret { - uuid: string - object_name: string - name: string // 3.5 - file: string - external: boolean -} - -export interface Config { - uuid: string - object_name: string - data: any -} - -export interface IPosition { - uuid: string - top: number - left: number - zoomLevel: number -} - -export interface Project { - uuid: string - name: string - mutable: boolean - canvas: IPosition[] - version: number - volumes: Volume[] - services: Service[] - networks: Network[] - secrets: Secret[] - configs: Config[] // from 3.3 - id?: number -} - -export interface RepoDetailsData { - uuid: string - name: string - affiliation: null - can_edit: boolean - description: string - full_description: string - has_starred: boolean - is_automated: boolean - is_migrated: boolean - is_private: boolean - last_updated: string - namespace: string - permissions: { - read: boolean - write: boolean - admin: boolean - } - pull_count: number - repository_type: string - star_count: number - status: number - user: string -} - -export interface GlobalDialog { - message: string, - type: string -} - -export interface User { - username: string - email: string - pk: number - first_name: string - last_name: string - token?: string -} - -export interface GlobalAppConfiguration { - mode: string -} - -export interface ImportPayload { - type: string, - payload: string -} - -export interface Recipe { - name: string, - keywords: string[], - repo_url: string, - docker_compose_file_raw_path: string, - uuid?: string -} \ No newline at end of file diff --git a/src/composer/src/app/core/store/project.actions.ts b/src/composer/src/app/core/store/project.actions.ts deleted file mode 100644 index 7d13151..0000000 --- a/src/composer/src/app/core/store/project.actions.ts +++ /dev/null @@ -1,85 +0,0 @@ -import { createAction, props } from '@ngrx/store' -import { IPosition, Service, Volume, Network, ImportPayload } from './models' - -export const SET_INITIAL_STATE = '[PROJECT] Set initial state' - -export const GET_SERVICES = '[PROJECT] Get services' -export const SET_SERVICES = '[PROJECT] Set services' -export const CREATE_SERVICE = '[PROJECT] Create service' -export const UPDATE_SERVICE = '[PROJECT] Update service' -export const REMOVE_SERVICE = '[PROJECT] Remove service' - -export const GET_POSITIONS = '[PROJECT] Get positions' -export const SET_POSITIONS = '[PROJECT] Set positions' -export const CREATE_POSITION = '[PROJECT] Create position' -export const UPDATE_POSITION = '[PROJECT] Update position' -export const REMOVE_POSITION = '[PROJECT] Remove position' - -export const GET_VOLUMES = '[PROJECT] Get volumes' -export const SET_VOLUMES = '[PROJECT] Set volumes' -export const CREATE_VOLUME = '[PROJECT] Create volume' -export const UPDATE_VOLUME = '[PROJECT] Update volume' -export const REMOVE_VOLUME = '[PROJECT] Remove volume' - -export const GET_NETWORKS = '[PROJECT] Get network' -export const SET_NETWORKS = '[PROJECT] Set network' -export const CREATE_NETWORKS = '[PROJECT] Create network' -export const UPDATE_NETWORKS = '[PROJECT] Update network' -export const REMOVE_NETWORKS = '[PROJECT] Remove network' - -export const SET_VERSION = '[PROJECT] Set version' - -export const SAVE_PROJECT = '[PROJECT] Save project' -export const SAVE_PROJECT_SUCCESS = '[PROJECT] Save project success' -export const UPLOAD_PROJECT = '[PROJECT] Upload project' - -export const API_PROJECT_REQUEST_FAIL = '[PROJECT] Api project request fail' - -export const UPDATE_VERSION = '[PROJECT] Update version' - -export const IMPORT_PROJECT = '[PROJECT] Import project' -export const IMPORT_CURRENT_PROJECT = '[PROJECT] Import current project' - -export const UPDATE_NAME = '[PROJECT] Update project name' - -export const VIEW_RECIPE = '[PROJECT] View recipe' - -export const SetInitialState = createAction(SET_INITIAL_STATE) - -export const GetServices = createAction(GET_SERVICES) -export const SetServices = createAction(SET_SERVICES, props<{ data: Service[] }>()) -export const CreateService = createAction(CREATE_SERVICE, props<{ data: Service }>()) -export const UpdateService = createAction(UPDATE_SERVICE, props<{ data: Service }>()) -export const RemoveService = createAction(REMOVE_SERVICE, props<{ data: Service }>()) - -export const GetPositions = createAction(GET_POSITIONS) -export const SetPositions = createAction(SET_POSITIONS, props<{ data: IPosition[] }>()) -export const CreatePosition = createAction(CREATE_POSITION, props<{ data: IPosition }>()) -export const UpdatePosition = createAction(UPDATE_POSITION, props<{ data: IPosition }>()) -export const RemovePosition = createAction(REMOVE_POSITION, props<{ data: IPosition }>()) - -export const GetVolumes = createAction(GET_VOLUMES) -export const SetVolumes = createAction(SET_VOLUMES, props<{ data: Volume[] }>()) -export const CreateVolume = createAction(CREATE_VOLUME, props<{ data: Volume }>()) -export const UpdateVolume = createAction(UPDATE_VOLUME, props<{ data: Volume }>()) -export const RemoveVolume = createAction(REMOVE_VOLUME, props<{ data: Volume }>()) - -export const GetNetwork = createAction(GET_NETWORKS) -export const SetNetwork = createAction(SET_NETWORKS, props<{ data: Network }>()) -export const CreateNetwork = createAction(CREATE_NETWORKS, props<{ data: Network }>()) -export const UpdateNetwork = createAction(UPDATE_NETWORKS, props<{ data: Network }>()) -export const RemoveNetwork = createAction(REMOVE_NETWORKS, props<{ data: Network }>()) - -export const ApiProjectRequestFail = createAction(API_PROJECT_REQUEST_FAIL, props<{ data: Error }>()) -export const UploadProject = createAction(UPLOAD_PROJECT, props<{ data: string }>()) -export const SaveProject = createAction(SAVE_PROJECT) -export const SaveProjectSuccess = createAction(SAVE_PROJECT_SUCCESS, props<{ data: any }>()) - -export const UpdateVersion = createAction(UPDATE_VERSION, props<{ data: number }>()) - -export const ImportProject = createAction(IMPORT_PROJECT, props<{ data: ImportPayload }>()) -export const ImportCurrentProject = createAction(IMPORT_CURRENT_PROJECT, props<{ data: ImportPayload }>()) - -export const UpdateName = createAction(UPDATE_NAME, props<{ data: string }>()) - -export const ViewRecipe = createAction(VIEW_RECIPE, props<{ data: string }>()) \ No newline at end of file diff --git a/src/composer/src/app/core/store/project.reducer.ts b/src/composer/src/app/core/store/project.reducer.ts deleted file mode 100644 index 502efe6..0000000 --- a/src/composer/src/app/core/store/project.reducer.ts +++ /dev/null @@ -1,121 +0,0 @@ -import { createReducer, on } from '@ngrx/store' -import { Project, Service, Volume } from './models' -import * as ProjectActions from './project.actions' - -export const initialState: Project = { - uuid: null, - name: null, - mutable: true, - canvas: [], - version: 3, - volumes: [], - services: [], - networks: [], - secrets: [], - configs: [], -} - -export function onUpdate(state, data) { - let newState = [...state] - const index = newState.findIndex((item) => item.uuid === data.uuid) - - if (index > -1) newState[index] = data - - return newState -} - -export function onCreate(state, data) { - return [...state, data] -} - -export function onDelete(state, data) { - let newState = [...state] - const index = newState.findIndex((service) => service.uuid === data.uuid) - - if (index > -1) newState.splice(index, 1) - - return newState -} - -export const projectReducer = createReducer( - initialState, - on(ProjectActions.SetInitialState, (state) => ({ ...initialState })), - - on(ProjectActions.GetServices, (state) => ({ ...state })), - on(ProjectActions.SetServices, (state, { data }) => ({ ...state, services: data })), - on(ProjectActions.CreateService, (state, { data }) => { - let obj = {} as Service - obj['uuid'] = data.uuid - obj['name'] = data.name - obj['image'] = data.image - obj['tag'] = data.tag - Object.preventExtensions(obj) - return { - ...state, - services: [...state.services, obj], - } - }), - on(ProjectActions.UpdateService, (state, { data }) => ({ ...state, services: onUpdate(state.services, data) })), - on(ProjectActions.RemoveService, (state, { data }) => ({ ...state, services: onDelete(state.services, data) })), - - on(ProjectActions.GetPositions, (state) => ({ ...state })), - on(ProjectActions.SetPositions, (state, { data }) => ({ ...state, canvas: data })), - on(ProjectActions.CreatePosition, (state, { data }) => ({ ...state, canvas: onCreate(state.canvas, data) })), - on(ProjectActions.UpdatePosition, (state, { data }) => ({ ...state, canvas: onUpdate(state.canvas, data) })), - on(ProjectActions.RemovePosition, (state, { data }) => ({ ...state, canvas: onDelete(state.canvas, data) })), - - on(ProjectActions.GetVolumes, (state) => ({ ...state })), - on(ProjectActions.SetVolumes, (state, { data }) => ({ ...state, volumes: data })), - on(ProjectActions.CreateVolume, (state, { data }) => ({ ...state, volumes: onCreate(state.volumes, data) })), - on(ProjectActions.UpdateVolume, (state, { data }) => ({ ...state, volumes: onUpdate(state.volumes, data) })), - on(ProjectActions.RemoveVolume, (state, { data }) => { - let newServices = [...state.services] - newServices = newServices.map((service: Service) => { - - if (service.volumes) { - let newVolumes = [...service.volumes] - const index = newVolumes.findIndex((volume: { volume: string; destination: string }) => volume.volume === data.uuid) - if (index > -1) newVolumes.splice(index, 1) - return {...service, volumes: [...newVolumes]} - } else { - return {...service, volumes: []} - } - }) - return { ...state, volumes: onDelete(state.volumes, data), services: newServices } - }), - - on(ProjectActions.GetNetwork, (state) => ({ ...state })), - //on(ProjectActions.SetNetwork, (state, { data }) => ({ ...state, networks: data })), - on(ProjectActions.CreateNetwork, (state, { data }) => ({ ...state, networks: onCreate(state.networks, data) })), - on(ProjectActions.UpdateNetwork, (state, { data }) => ({ ...state, networks: onUpdate(state.networks, data) })), - on(ProjectActions.RemoveNetwork, (state, { data }) => { - let newServices = [...state.services] - newServices = newServices.map((service: Service) => { - - if (service.networks) { - let newNetworks = [...service.networks] - const index = newNetworks.findIndex((network: { network: string; destination: string }) => network.network === data.uuid) - if (index > -1) newNetworks.splice(index, 1) - return {...service, volumes: [...newNetworks]} - } else { - return {...service, newNetworks: []} - } - }) - return { ...state, networks: onDelete(state.networks, data), services: newServices } - }), - - on(ProjectActions.SaveProjectSuccess, (state, { data }) => { - let forSpread = {} - forSpread = Object.assign(forSpread, data) - - return { - ...state, - ...forSpread, - } - }), - on(ProjectActions.ApiProjectRequestFail, (state, { data }) => { - return {} - }), - on(ProjectActions.UpdateVersion, (state, { data }) => ({ ...state, version: data })), - on(ProjectActions.UpdateName, (state, { data }) => ({ ...state, name: data})), -) diff --git a/src/composer/src/app/core/store/reducers/global-app-configuration.reducer.ts b/src/composer/src/app/core/store/reducers/global-app-configuration.reducer.ts deleted file mode 100644 index ab08d21..0000000 --- a/src/composer/src/app/core/store/reducers/global-app-configuration.reducer.ts +++ /dev/null @@ -1,13 +0,0 @@ -import { createReducer, on } from '@ngrx/store' -import * as GlobalConfigurationActions from '../actions/global-app-configuration.actions' - -export const initialState = { - mode: 'compose' -} - -export const globalConfigurationReducer = createReducer( - initialState, - on(GlobalConfigurationActions.OnComposeMode, (state) => ({ ...state, mode: 'compose'})), - on(GlobalConfigurationActions.OnImportMode, (state) => ({ ...state, mode: 'import'})), - on(GlobalConfigurationActions.OnRecipeLoadMode, (state) => ({ ...state, mode: 'recipe_load'})) -) \ No newline at end of file diff --git a/src/composer/src/app/core/store/reducers/global-dialog.reducer.ts b/src/composer/src/app/core/store/reducers/global-dialog.reducer.ts deleted file mode 100644 index f5ee9c3..0000000 --- a/src/composer/src/app/core/store/reducers/global-dialog.reducer.ts +++ /dev/null @@ -1,13 +0,0 @@ -import { createReducer, on } from '@ngrx/store' -import * as GlobalDialogActions from '../actions/global-dialog.actions' - -export const initialState = { - message: '', - type: '' -} - -export const globalDialogReducer = createReducer( - initialState, - on(GlobalDialogActions.ProjectNotFound, (state, newState) => ({ ...state, ...newState})), - on(GlobalDialogActions.ServerError, (state, newState) => ({ ...state, ...newState})) -) \ No newline at end of file diff --git a/src/composer/src/app/core/store/reducers/global-spinner.reducer.ts b/src/composer/src/app/core/store/reducers/global-spinner.reducer.ts deleted file mode 100644 index be769a4..0000000 --- a/src/composer/src/app/core/store/reducers/global-spinner.reducer.ts +++ /dev/null @@ -1,11 +0,0 @@ -import { createReducer, on } from '@ngrx/store' -import * as GlobalSpinnerActions from '../actions/global-spinner.actions' - -export const initialState: boolean = false - -export const globalSpinnerReducer = createReducer( - initialState, - on(GlobalSpinnerActions.OnSpinner, () => true), - on(GlobalSpinnerActions.OffSpinner, () => false), - ) - \ No newline at end of file diff --git a/src/composer/src/app/core/utils/utils.ts b/src/composer/src/app/core/utils/utils.ts deleted file mode 100644 index d94d17e..0000000 --- a/src/composer/src/app/core/utils/utils.ts +++ /dev/null @@ -1,7 +0,0 @@ -export const uuidv4 = () => { - return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, function (c) { - var r = (Math.random() * 16) | 0, - v = c == 'x' ? r : (r & 0x3) | 0x8 - return v.toString(16) - }) -} diff --git a/src/composer/src/app/login/login.component.html b/src/composer/src/app/login/login.component.html deleted file mode 100644 index 0b61090..0000000 --- a/src/composer/src/app/login/login.component.html +++ /dev/null @@ -1,36 +0,0 @@ -
- - -
- -
-

Login

- - - Email - - - - Password - - -
- -
- - -
-
-
- {{errorMessage}} -
-
-
- -
- \ No newline at end of file diff --git a/src/composer/src/app/login/login.component.scss b/src/composer/src/app/login/login.component.scss deleted file mode 100644 index f77f7cd..0000000 --- a/src/composer/src/app/login/login.component.scss +++ /dev/null @@ -1,9 +0,0 @@ -.github-button { - height: 70px; - margin: 20px 0; - - .icon { - width: 30px; - height: 30px; - } -} \ No newline at end of file diff --git a/src/composer/src/app/login/login.component.spec.ts b/src/composer/src/app/login/login.component.spec.ts deleted file mode 100644 index d6d85a8..0000000 --- a/src/composer/src/app/login/login.component.spec.ts +++ /dev/null @@ -1,25 +0,0 @@ -import { async, ComponentFixture, TestBed } from '@angular/core/testing'; - -import { LoginComponent } from './login.component'; - -describe('LoginComponent', () => { - let component: LoginComponent; - let fixture: ComponentFixture; - - beforeEach(async(() => { - TestBed.configureTestingModule({ - declarations: [ LoginComponent ] - }) - .compileComponents(); - })); - - beforeEach(() => { - fixture = TestBed.createComponent(LoginComponent); - component = fixture.componentInstance; - fixture.detectChanges(); - }); - - it('should create', () => { - expect(component).toBeTruthy(); - }); -}); diff --git a/src/composer/src/app/login/login.component.ts b/src/composer/src/app/login/login.component.ts deleted file mode 100644 index a785c94..0000000 --- a/src/composer/src/app/login/login.component.ts +++ /dev/null @@ -1,55 +0,0 @@ -import { Component, OnInit } from '@angular/core' -import { FormBuilder, FormControl, FormGroup, Validators } from '@angular/forms' -import { Router } from '@angular/router' -import { AuthenticationService } from '../core/services/authentication.service' -import { Location } from '@angular/common' -import { first } from 'rxjs/operators' -import { AuthGuard } from '../core/helpers/auth.gourd' -import { Store } from '@ngrx/store' -import * as GlobalSpinnerActions from '../core/store/actions/global-spinner.actions' - -@Component({ - selector: 'app-login', - templateUrl: './login.component.html', - styleUrls: ['./login.component.scss'] -}) -export class LoginComponent implements OnInit { - loginForm: FormGroup - errors: string[] - constructor(private formBuilder: FormBuilder, public _location: Location, private router: Router, private authenticationService: AuthenticationService, private store: Store) { - this.loginForm = this.formBuilder.group({ - email: new FormControl('', [ Validators.email ]), - password: new FormControl('', [ Validators.minLength(3) ]) - }) - } - - ngOnInit(): void { - this.store.dispatch(GlobalSpinnerActions.OffSpinner()) - } - - login(): void { - this.authenticationService.login(this.loginForm.value) - .pipe(first()) - .subscribe( - data => { - const lastPageBeferoLogin = localStorage.getItem('lastPageBeforeRegistrationOrLogin') - lastPageBeferoLogin ? this.router.navigate([lastPageBeferoLogin]) : this.router.navigate(['/']) - }, - (errors) => { - this.errors = errors - }); - } - - gitHubAuth() { - this.authenticationService.gitHubAuth() - .pipe(first()) - .subscribe( - data => { - const lastPageBeferoLogin = localStorage.getItem('lastPageBeforeRegistrationOrLogin') - lastPageBeferoLogin ? this.router.navigate([lastPageBeferoLogin]) : this.router.navigate(['/']) - }, - (errors) => { - this.errors = errors - }); - } -} diff --git a/src/composer/src/app/registration/registration.component.html b/src/composer/src/app/registration/registration.component.html deleted file mode 100644 index 81116e2..0000000 --- a/src/composer/src/app/registration/registration.component.html +++ /dev/null @@ -1,39 +0,0 @@ -
- - -
- -
-

Registration

- - - Email - - - - - Password - - - - - Confirm Password - - - -
- -
- - -
-
- -
- {{ errorMessage }} -
-
-
- -
- \ No newline at end of file diff --git a/src/composer/src/app/registration/registration.component.scss b/src/composer/src/app/registration/registration.component.scss deleted file mode 100644 index e69de29..0000000 diff --git a/src/composer/src/app/registration/registration.component.spec.ts b/src/composer/src/app/registration/registration.component.spec.ts deleted file mode 100644 index d08995a..0000000 --- a/src/composer/src/app/registration/registration.component.spec.ts +++ /dev/null @@ -1,25 +0,0 @@ -import { async, ComponentFixture, TestBed } from '@angular/core/testing'; - -import { RegistrationComponent } from './registration.component'; - -describe('RegistrationComponent', () => { - let component: RegistrationComponent; - let fixture: ComponentFixture; - - beforeEach(async(() => { - TestBed.configureTestingModule({ - declarations: [ RegistrationComponent ] - }) - .compileComponents(); - })); - - beforeEach(() => { - fixture = TestBed.createComponent(RegistrationComponent); - component = fixture.componentInstance; - fixture.detectChanges(); - }); - - it('should create', () => { - expect(component).toBeTruthy(); - }); -}); diff --git a/src/composer/src/app/registration/registration.component.ts b/src/composer/src/app/registration/registration.component.ts deleted file mode 100644 index eccfbcd..0000000 --- a/src/composer/src/app/registration/registration.component.ts +++ /dev/null @@ -1,51 +0,0 @@ -import { Component, OnInit } from '@angular/core' -import { FormBuilder, FormControl, FormGroup, Validators } from '@angular/forms' -import { Location } from '@angular/common' -import { first } from 'rxjs/operators' -import { Subject } from 'rxjs' -import { AuthenticationService } from '../core/services/authentication.service' -import { Router } from '@angular/router' - -@Component({ - selector: 'app-registration', - templateUrl: './registration.component.html', - styleUrls: ['./registration.component.scss'] -}) -export class RegistrationComponent implements OnInit { - - registrationForm: FormGroup - errors: string[] - private unSubscribe$ = new Subject() - - constructor(private formBuilder: FormBuilder, private authenticationService: AuthenticationService, private router: Router, public _location: Location) { - this.registrationForm = this.formBuilder.group({ - email: new FormControl('', [ Validators.email ]), - password1: new FormControl('', [ Validators.required ]), - password2: new FormControl('', [ Validators.required ]) - }) - } - - ngOnInit(): void { - } - - registration(): void { - this.authenticationService.registration(this.registrationForm.value) - .pipe(first()) - .subscribe( - data => { - const { email, password1: password } = this.registrationForm.value - this.authenticationService.login({ email, password }) - .pipe(first()) - .subscribe( - data => { - this.router.navigate(['/']) - }, - (errors) => { - this.errors = errors - }); - }, - (errors) => { - this.errors = errors - }); - } -} diff --git a/src/composer/src/assets/.gitkeep b/src/composer/src/assets/.gitkeep deleted file mode 100644 index e69de29..0000000 diff --git a/src/composer/src/assets/github-logo-black.svg b/src/composer/src/assets/github-logo-black.svg deleted file mode 100644 index 361cc07..0000000 --- a/src/composer/src/assets/github-logo-black.svg +++ /dev/null @@ -1,18 +0,0 @@ - - - github-logo - - - - - - \ No newline at end of file diff --git a/src/composer/src/assets/github-logo.svg b/src/composer/src/assets/github-logo.svg deleted file mode 100644 index 22bf478..0000000 --- a/src/composer/src/assets/github-logo.svg +++ /dev/null @@ -1,18 +0,0 @@ - - - github-logo - - - - - - \ No newline at end of file diff --git a/src/composer/src/assets/nuxx-logo-v2.svg b/src/composer/src/assets/nuxx-logo-v2.svg deleted file mode 100644 index bcda30c..0000000 --- a/src/composer/src/assets/nuxx-logo-v2.svg +++ /dev/null @@ -1,18 +0,0 @@ - - - nuxx-logo - - - - \ No newline at end of file diff --git a/src/composer/src/assets/nuxx-logo-white.svg b/src/composer/src/assets/nuxx-logo-white.svg deleted file mode 100644 index e54df20..0000000 --- a/src/composer/src/assets/nuxx-logo-white.svg +++ /dev/null @@ -1,18 +0,0 @@ - - - nuxx-logo-white - - - - \ No newline at end of file diff --git a/src/composer/src/assets/nuxx-logo.jpg b/src/composer/src/assets/nuxx-logo.jpg deleted file mode 100644 index 3e514bb..0000000 Binary files a/src/composer/src/assets/nuxx-logo.jpg and /dev/null differ diff --git a/src/composer/src/assets/nuxx-logo.svg b/src/composer/src/assets/nuxx-logo.svg deleted file mode 100644 index d615d2a..0000000 --- a/src/composer/src/assets/nuxx-logo.svg +++ /dev/null @@ -1,13 +0,0 @@ - - - Logo - - - - \ No newline at end of file diff --git a/src/composer/src/assets/search-result-ph.png b/src/composer/src/assets/search-result-ph.png deleted file mode 100644 index 6f83b16..0000000 Binary files a/src/composer/src/assets/search-result-ph.png and /dev/null differ diff --git a/src/composer/src/assets/twitter-logo.svg b/src/composer/src/assets/twitter-logo.svg deleted file mode 100644 index 5d0d773..0000000 --- a/src/composer/src/assets/twitter-logo.svg +++ /dev/null @@ -1,11 +0,0 @@ - - - twitter-logo - - - - - - \ No newline at end of file diff --git a/src/composer/src/assets/user.svg b/src/composer/src/assets/user.svg deleted file mode 100644 index 62bff27..0000000 --- a/src/composer/src/assets/user.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/src/composer/src/environments/environment.local.ts b/src/composer/src/environments/environment.local.ts deleted file mode 100644 index 0e0ffdb..0000000 --- a/src/composer/src/environments/environment.local.ts +++ /dev/null @@ -1,4 +0,0 @@ -export const environment = { - production: true, - apiUrl: 'http://localhost/api', -} diff --git a/src/composer/src/environments/environment.prod.ts b/src/composer/src/environments/environment.prod.ts deleted file mode 100644 index cb6666d..0000000 --- a/src/composer/src/environments/environment.prod.ts +++ /dev/null @@ -1,4 +0,0 @@ -export const environment = { - production: true, - apiUrl: 'https://api.nuxx.io/api', -} diff --git a/src/composer/src/environments/environment.staging.ts b/src/composer/src/environments/environment.staging.ts deleted file mode 100644 index 697f1a4..0000000 --- a/src/composer/src/environments/environment.staging.ts +++ /dev/null @@ -1,4 +0,0 @@ -export const environment = { - production: false, - apiUrl: 'https://staging-api.nuxx.io/api', -} diff --git a/src/composer/src/environments/environment.ts b/src/composer/src/environments/environment.ts deleted file mode 100644 index e9197dd..0000000 --- a/src/composer/src/environments/environment.ts +++ /dev/null @@ -1,17 +0,0 @@ -// This file can be replaced during build by using the `fileReplacements` array. -// `ng build --prod` replaces `environment.ts` with `environment.prod.ts`. -// The list of file replacements can be found in `angular.json`. - -export const environment = { - production: true, - apiUrl: 'http://localhost:9001/api', -} - -/* - * For easier debugging in development mode, you can import the following file - * to ignore zone related error stack frames such as `zone.run`, `zoneDelegate.invokeTask`. - * - * This import should be commented out in production mode because it will have a negative impact - * on performance if an error is thrown. - */ -// import 'zone.js/dist/zone-error'; // Included with Angular CLI. diff --git a/src/composer/src/favicon.ico b/src/composer/src/favicon.ico deleted file mode 100644 index d47c5ca..0000000 Binary files a/src/composer/src/favicon.ico and /dev/null differ diff --git a/src/composer/src/index.html b/src/composer/src/index.html deleted file mode 100644 index f185086..0000000 --- a/src/composer/src/index.html +++ /dev/null @@ -1,27 +0,0 @@ - - - - - - - - - Application Stack Composer - - - - - - - - - - - - diff --git a/src/composer/src/main.ts b/src/composer/src/main.ts deleted file mode 100644 index f832a35..0000000 --- a/src/composer/src/main.ts +++ /dev/null @@ -1,14 +0,0 @@ -import { enableProdMode } from '@angular/core'; -import { platformBrowserDynamic } from '@angular/platform-browser-dynamic'; - -import { AppModule } from './app/app.module'; -import { environment } from './environments/environment'; - -import 'codemirror/mode/yaml/yaml'; - -if (environment.production) { - enableProdMode(); -} - -platformBrowserDynamic().bootstrapModule(AppModule) - .catch(err => console.error(err)); diff --git a/src/composer/src/polyfills.ts b/src/composer/src/polyfills.ts deleted file mode 100644 index 4a38b41..0000000 --- a/src/composer/src/polyfills.ts +++ /dev/null @@ -1,65 +0,0 @@ -/** - * This file includes polyfills needed by Angular and is loaded before the app. - * You can add your own extra polyfills to this file. - * - * This file is divided into 2 sections: - * 1. Browser polyfills. These are applied before loading ZoneJS and are sorted by browsers. - * 2. Application imports. Files imported after ZoneJS that should be loaded before your main - * file. - * - * The current setup is for so-called "evergreen" browsers; the last versions of browsers that - * automatically update themselves. This includes Safari >= 10, Chrome >= 55 (including Opera), - * Edge >= 13 on the desktop, and iOS 10 and Chrome on mobile. - * - * Learn more in https://angular.io/guide/browser-support - */ - -/*************************************************************************************************** - * BROWSER POLYFILLS - */ - -/** IE10 and IE11 requires the following for NgClass support on SVG elements */ -// import 'classlist.js'; // Run `npm install --save classlist.js`. - -/** - * Web Animations `@angular/platform-browser/animations` - * Only required if AnimationBuilder is used within the application and using IE/Edge or Safari. - * Standard animation support in Angular DOES NOT require any polyfills (as of Angular 6.0). - */ -// import 'web-animations-js'; // Run `npm install --save web-animations-js`. - -/** - * By default, zone.js will patch all possible macroTask and DomEvents - * user can disable parts of macroTask/DomEvents patch by setting following flags - * because those flags need to be set before `zone.js` being loaded, and webpack - * will put import in the top of bundle, so user need to create a separate file - * in this directory (for example: zone-flags.ts), and put the following flags - * into that file, and then add the following code before importing zone.js. - * import './zone-flags'; - * - * The flags allowed in zone-flags.ts are listed here. - * - * The following flags will work for all browsers. - * - * (window as any).__Zone_disable_requestAnimationFrame = true; // disable patch requestAnimationFrame - * (window as any).__Zone_disable_on_property = true; // disable patch onProperty such as onclick - * (window as any).__zone_symbol__UNPATCHED_EVENTS = ['scroll', 'mousemove']; // disable patch specified eventNames - * - * in IE/Edge developer tools, the addEventListener will also be wrapped by zone.js - * with the following flag, it will bypass `zone.js` patch for IE/Edge - * - * (window as any).__Zone_enable_cross_context_check = true; - * - */ - -/*************************************************************************************************** - * Zone JS is required by default for Angular itself. - */ -import 'zone.js/dist/zone'; // Included with Angular CLI. - - -/*************************************************************************************************** - * APPLICATION IMPORTS - */ - -(window as any).global = window; \ No newline at end of file diff --git a/src/composer/src/styles.scss b/src/composer/src/styles.scss deleted file mode 100644 index d732060..0000000 --- a/src/composer/src/styles.scss +++ /dev/null @@ -1,267 +0,0 @@ -/* You can add global styles to this file, and also import other style files */ -@import '~highlight.js/styles/vs2015.css'; - -@import "~codemirror/lib/codemirror"; -@import "~codemirror/theme/material"; - -@import "./app/core/components/code-view/code-mirror-styles.scss"; - -html, -body { - height: 100%; -} -body { - margin: 0; - font-family: Roboto, 'Helvetica Neue', sans-serif; -} - -pre { - display: flex; - margin-top: 0; - margin-bottom: 0; - word-wrap: break-word; - - code { - flex: 1; - line-height: 1.5em; - font-size: 12px; - min-height: 100%; - padding: 1em 1.2em; - - &.hljs-line-numbers { - padding: 0; - } - } -} - -/** Line numbers styles */ -.hljs { - padding: 10px; - transition: border ease 1s; -} - -.hljs-ln { - padding: 8px 0; - - tr { - &:first-child td { - padding-top: 8px !important; - } - - &:last-child td { - padding-bottom: 8px !important; - } - } -} - -/* for block of numbers */ -td.hljs-ln-numbers { - user-select: none; - text-align: center; - vertical-align: top; - color: gray; - padding-right: 8px !important; - padding-left: 8px !important; -} - -/* for block of code */ -td.hljs-ln-code { - padding-left: 8px !important; -} - -.dialog-details { - .hljs { - white-space: break-spaces; - background: unset; - color: unset; - .hljs-section { - color: black; - font-weight: bold; - } - } -} - -.mat-raised-button.mat-white, -.mat-icon-button.mat-white, -.mat-flat-button.mat-white, -.mat-fab.mat-white, -.mat-mini-fab.mat-white { - color: #ffffff; - background-color: none; -} - -.mat-raised-button.mat-success, -.mat-flat-button.mat-success, -.mat-fab.mat-success, -.mat-mini-fab.mat-success { - color: #ffffff; - background-color: #58bd28; -} - -.mat-flat-button.mat-primary, -.mat-raised-button.mat-primary, -.mat-fab.mat-primary, -.mat-mini-fab.mat-primary { - color: #ffffff; - background-color: #657dff; -} - -.mat-flat-button.mat-warn, -.mat-raised-button.mat-warn, -.mat-fab.mat-warn, -.mat-mini-fab.mat-warn { - color: #ffffff; - background-color: #d44141; -} - -input:-webkit-autofill, -input:-webkit-autofill:hover, -input:-webkit-autofill:focus, -input:-webkit-autofill:active { - -webkit-box-shadow: 0 0 0 30px white inset !important; -} - -.mat-drawer-container { - background-color: #ffffff; - color: rgba(0,0,0,.87); -} - -.mat-tab-body.mat-tab-body-active { - overflow-y: hidden!important; -} - -.mat-tab-body-content { - padding: 16px 0; -} - -mat-tab-header.mat-tab-header { - background-color: #ffffff; - position: sticky; - top: 0; - z-index: 2; -} -.dialog-actions { - justify-content: flex-end; -} -.repo-search { - border-right: 1px solid #e8e8e8; - - .mat-drawer-inner-container { - overflow-x: hidden; - } -} - -.jtk-endpoint { - &:hover { - z-index: 3; - cursor: grab; - } -} - -pre { - white-space: pre-wrap; /* css-3 */ - white-space: -moz-pre-wrap; /* Mozilla, since 1999 */ - white-space: -pre-wrap; /* Opera 4-6 */ - white-space: -o-pre-wrap; /* Opera 7 */ - word-wrap: break-word; /* Internet Explorer 5.5+ */ - } - -.dialog-details img { - display: block; - max-width: 100%; -} - -.pd-b-0 { - .mat-form-field-wrapper { - padding-bottom: 0; - } -} - -.pd-b-5 { - .mat-form-field-wrapper { - padding-bottom: 5px; - } -} - -.pd-b-10 { - .mat-form-field-wrapper { - padding-bottom: 10px; - } -} - -.pd-b-15 { - .mat-form-field-wrapper { - padding-bottom: 15px; - } -} - -.pd-b-20 { - .mat-form-field-wrapper { - padding-bottom: 20px; - } -} - -.mg-b-15 { - margin-bottom: 15px!important; -} - -.error-modal-message { - margin: 0 0 15px 0; -} - -.error-modal-code-wrap { - background: #f3f3f3; - padding: 6px 12px; - margin: 0 0 15px 0; -} - -as-split > .code-viewer { - overflow: hidden!important; -} - -.input-short { - .mat-form-field-wrapper { - margin: 0 !important; - padding: 0 !important; - } -} - - -.input-short { - .mat-form-field-infix { - padding: .4em 0 .8em 0; - } -} - -.input-short { - .mat-select-arrow-wrapper { - transform: translateY(0%) !important; - } -} - -.project-name { - font-weight: 500; - font-size: 16px; - - .mat-focused { - .mat-form-field-label-wrapper { - .mat-form-field-label { - padding-top: 10px!important; - } - } - } - - .mat-form-field-label-wrapper { - top: -20px; - .mat-form-field-empty { - padding-top: unset!important; - } - .mat-form-field-label { - padding-top: 10px; - } - } -} - -.splitter { - border-bottom: 1px solid #e8e8e8; -} \ No newline at end of file diff --git a/src/composer/src/test.ts b/src/composer/src/test.ts deleted file mode 100644 index 50193eb..0000000 --- a/src/composer/src/test.ts +++ /dev/null @@ -1,25 +0,0 @@ -// This file is required by karma.conf.js and loads recursively all the .spec and framework files - -import 'zone.js/dist/zone-testing'; -import { getTestBed } from '@angular/core/testing'; -import { - BrowserDynamicTestingModule, - platformBrowserDynamicTesting -} from '@angular/platform-browser-dynamic/testing'; - -declare const require: { - context(path: string, deep?: boolean, filter?: RegExp): { - keys(): string[]; - (id: string): T; - }; -}; - -// First, initialize the Angular testing environment. -getTestBed().initTestEnvironment( - BrowserDynamicTestingModule, - platformBrowserDynamicTesting() -); -// Then we find all the tests. -const context = require.context('./', true, /\.spec\.ts$/); -// And load the modules. -context.keys().map(context); diff --git a/src/composer/tsconfig.app.json b/src/composer/tsconfig.app.json deleted file mode 100644 index f758d98..0000000 --- a/src/composer/tsconfig.app.json +++ /dev/null @@ -1,14 +0,0 @@ -{ - "extends": "./tsconfig.json", - "compilerOptions": { - "outDir": "./out-tsc/app", - "types": [] - }, - "files": [ - "src/main.ts", - "src/polyfills.ts" - ], - "include": [ - "src/**/*.d.ts" - ] -} diff --git a/src/composer/tsconfig.json b/src/composer/tsconfig.json deleted file mode 100644 index 2cb22a2..0000000 --- a/src/composer/tsconfig.json +++ /dev/null @@ -1,28 +0,0 @@ -{ - //"enableIvy": true, - "compileOnSave": false, - "compilerOptions": { - "baseUrl": "./", - //"baseUrl": "./src", - "outDir": "./dist/out-tsc", - "sourceMap": true, - "declaration": false, - "downlevelIteration": true, - "experimentalDecorators": true, - "module": "esnext", - "moduleResolution": "node", - "importHelpers": true, - "target": "es2015", - "typeRoots": [ - "node_modules/@types" - ], - "lib": [ - "es2018", - "dom" - ] - }, - "angularCompilerOptions": { - "fullTemplateTypeCheck": true, - "strictInjectionParameters": true - } -} diff --git a/src/composer/tsconfig.spec.json b/src/composer/tsconfig.spec.json deleted file mode 100644 index 6400fde..0000000 --- a/src/composer/tsconfig.spec.json +++ /dev/null @@ -1,18 +0,0 @@ -{ - "extends": "./tsconfig.json", - "compilerOptions": { - "outDir": "./out-tsc/spec", - "types": [ - "jasmine", - "node" - ] - }, - "files": [ - "src/test.ts", - "src/polyfills.ts" - ], - "include": [ - "src/**/*.spec.ts", - "src/**/*.d.ts" - ] -} diff --git a/src/composer/tslint.json b/src/composer/tslint.json deleted file mode 100644 index c8d70f1..0000000 --- a/src/composer/tslint.json +++ /dev/null @@ -1,91 +0,0 @@ -{ - "extends": "tslint:recommended", - "rules": { - "array-type": false, - "arrow-parens": false, - "deprecation": { - "severity": "warning" - }, - "component-class-suffix": true, - "contextual-lifecycle": true, - "directive-class-suffix": true, - "directive-selector": [ - true, - "attribute", - "app", - "camelCase" - ], - "component-selector": [ - true, - "element", - "app", - "kebab-case" - ], - "import-blacklist": [ - true, - "rxjs/Rx" - ], - "interface-name": false, - "max-classes-per-file": false, - "max-line-length": [ - true, - 140 - ], - "member-access": false, - "member-ordering": [ - true, - { - "order": [ - "static-field", - "instance-field", - "static-method", - "instance-method" - ] - } - ], - "no-consecutive-blank-lines": false, - "no-console": [ - true, - "debug", - "info", - "time", - "timeEnd", - "trace" - ], - "no-empty": false, - "no-inferrable-types": [ - true, - "ignore-params" - ], - "no-non-null-assertion": true, - "no-redundant-jsdoc": true, - "no-switch-case-fall-through": true, - "no-var-requires": false, - "object-literal-key-quotes": [ - true, - "as-needed" - ], - "object-literal-sort-keys": false, - "ordered-imports": false, - "quotemark": [ - true, - "single" - ], - "trailing-comma": false, - "no-conflicting-lifecycle": true, - "no-host-metadata-property": true, - "no-input-rename": true, - "no-inputs-metadata-property": true, - "no-output-native": true, - "no-output-on-prefix": true, - "no-output-rename": true, - "no-outputs-metadata-property": true, - "template-banana-in-box": true, - "template-no-negated-async": true, - "use-lifecycle-interface": true, - "use-pipe-transform-interface": true - }, - "rulesDirectory": [ - "codelyzer" - ] -} \ No newline at end of file