You are viewing a plain text version of this content. The canonical link for it is here.
Posted to commits@airavata.apache.org by is...@apache.org on 2020/10/08 01:28:46 UTC

[airavata-custos-portal] 05/13: Initial code of custos-tenant-portal

This is an automated email from the ASF dual-hosted git repository.

isjarana pushed a commit to branch custos-tenant-portal
in repository https://gitbox.apache.org/repos/asf/airavata-custos-portal.git

commit c42eda965330e4e595fbbde0aa47adaf9aeb87fc
Author: Isuru Ranawaka <ir...@gmail.com>
AuthorDate: Sun Oct 4 15:22:34 2020 -0400

    Initial code of custos-tenant-portal
---
 .gitignore                                         |    3 +
 custos-demo-gateway/.dockerignore                  |   13 +
 custos-demo-gateway/.env                           |    3 +
 custos-demo-gateway/.idea/custos-demo-gateway.iml  |    8 +
 .../.idea/inspectionProfiles/Project_Default.xml   |    6 +
 custos-demo-gateway/.idea/misc.xml                 |    9 +
 custos-demo-gateway/.idea/modules.xml              |    8 +
 custos-demo-gateway/.idea/vcs.xml                  |    6 +
 custos-demo-gateway/.idea/workspace.xml            |  141 +
 custos-demo-gateway/Dockerfile                     |   19 +
 custos-demo-gateway/README.md                      |   24 +
 custos-demo-gateway/babel.config.js                |    5 +
 custos-demo-gateway/docker-compose.yml             |   11 +
 custos-demo-gateway/entrypoint.sh                  |   18 +
 custos-demo-gateway/fullchain.pem                  |   59 +
 custos-demo-gateway/nginx.conf                     |   35 +
 custos-demo-gateway/package.json                   |   50 +
 custos-demo-gateway/public/favicon.ico             |  Bin 0 -> 4286 bytes
 custos-demo-gateway/public/index.html              |   17 +
 custos-demo-gateway/src/App.vue                    |   29 +
 custos-demo-gateway/src/assets/bots.png            |  Bin 0 -> 5228 bytes
 custos-demo-gateway/src/assets/credentials.png     |  Bin 0 -> 3332 bytes
 custos-demo-gateway/src/assets/custos_home.png     |  Bin 0 -> 44021 bytes
 custos-demo-gateway/src/assets/dblogs.png          |  Bin 0 -> 19538 bytes
 custos-demo-gateway/src/assets/groups_web.png      |  Bin 0 -> 2934 bytes
 custos-demo-gateway/src/assets/logo.png            |  Bin 0 -> 6849 bytes
 custos-demo-gateway/src/assets/sharings.png        |  Bin 0 -> 2417 bytes
 custos-demo-gateway/src/assets/users.png           |  Bin 0 -> 4824 bytes
 custos-demo-gateway/src/components/Callback.vue    |   86 +
 .../src/components/TheNavigator.vue                |   24 +
 .../src/components/landing/Landing.vue             |  229 +
 .../src/components/registration/CreateAccount.vue  |  206 +
 .../src/components/workspace/Agents.vue            |  295 +
 .../src/components/workspace/Groups.vue            |  653 ++
 .../src/components/workspace/Logs.vue              |  223 +
 .../src/components/workspace/Secrets.vue           |  355 +
 .../src/components/workspace/Sharing.vue           |  892 ++
 .../src/components/workspace/Users.vue             |  609 ++
 .../src/components/workspace/Workspace.vue         |  141 +
 custos-demo-gateway/src/config.js                  |   41 +
 custos-demo-gateway/src/main.js                    |   23 +
 custos-demo-gateway/src/router.js                  |  159 +
 .../src/service/agent_management.js                |  116 +
 custos-demo-gateway/src/service/api.js             |   12 +
 custos-demo-gateway/src/service/auth.js            |   96 +
 .../src/service/group_management.js                |  157 +
 .../src/service/identity_management.js             |   55 +
 custos-demo-gateway/src/service/log_management.js  |   36 +
 .../src/service/secret_management.js               |   98 +
 .../src/service/sharing_management.js              |  198 +
 custos-demo-gateway/src/service/user_management.js |  161 +
 custos-demo-gateway/src/store/index.js             |   29 +
 .../src/store/modules/agent.store.js               |  155 +
 .../src/store/modules/group.store.js               |  153 +
 .../src/store/modules/identity.store.js            |  134 +
 custos-demo-gateway/src/store/modules/log.store.js |   46 +
 .../src/store/modules/secret.store.js              |  116 +
 .../src/store/modules/sharing.store.js             |  227 +
 .../src/store/modules/user.store.js                |  149 +
 custos-demo-gateway/yarn-error.log                 | 8587 ++++++++++++++++++++
 custos-demo-gateway/yarn.lock                      | 8534 +++++++++++++++++++
 61 files changed, 23459 insertions(+)

diff --git a/.gitignore b/.gitignore
new file mode 100644
index 0000000..5707668
--- /dev/null
+++ b/.gitignore
@@ -0,0 +1,3 @@
+
+node_modules
+.idea*
diff --git a/custos-demo-gateway/.dockerignore b/custos-demo-gateway/.dockerignore
new file mode 100644
index 0000000..e3b21df
--- /dev/null
+++ b/custos-demo-gateway/.dockerignore
@@ -0,0 +1,13 @@
+# Items that don't need to be in a Docker image.
+# Anything not used by the build system should go here.
+.git
+.dockerignore
+.gitignore
+.github/*
+README.md
+Dockerfile
+
+# Artifacts that will be built during image creation.
+# This should contain all files created during `yarn build`.
+dist
+node_modules
\ No newline at end of file
diff --git a/custos-demo-gateway/.env b/custos-demo-gateway/.env
new file mode 100644
index 0000000..e84f702
--- /dev/null
+++ b/custos-demo-gateway/.env
@@ -0,0 +1,3 @@
+VUE_APP_CLIENT_ID=""
+VUE_APP_CLIENT_SEC=""
+VUE_APP_REDIRECT_URI="http://localhost:8080/callback"
\ No newline at end of file
diff --git a/custos-demo-gateway/.idea/custos-demo-gateway.iml b/custos-demo-gateway/.idea/custos-demo-gateway.iml
new file mode 100644
index 0000000..c956989
--- /dev/null
+++ b/custos-demo-gateway/.idea/custos-demo-gateway.iml
@@ -0,0 +1,8 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<module type="WEB_MODULE" version="4">
+  <component name="NewModuleRootManager">
+    <content url="file://$MODULE_DIR$" />
+    <orderEntry type="inheritedJdk" />
+    <orderEntry type="sourceFolder" forTests="false" />
+  </component>
+</module>
\ No newline at end of file
diff --git a/custos-demo-gateway/.idea/inspectionProfiles/Project_Default.xml b/custos-demo-gateway/.idea/inspectionProfiles/Project_Default.xml
new file mode 100644
index 0000000..03d9549
--- /dev/null
+++ b/custos-demo-gateway/.idea/inspectionProfiles/Project_Default.xml
@@ -0,0 +1,6 @@
+<component name="InspectionProjectProfileManager">
+  <profile version="1.0">
+    <option name="myName" value="Project Default" />
+    <inspection_tool class="Eslint" enabled="true" level="WARNING" enabled_by_default="true" />
+  </profile>
+</component>
\ No newline at end of file
diff --git a/custos-demo-gateway/.idea/misc.xml b/custos-demo-gateway/.idea/misc.xml
new file mode 100644
index 0000000..5b9db6f
--- /dev/null
+++ b/custos-demo-gateway/.idea/misc.xml
@@ -0,0 +1,9 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<project version="4">
+  <component name="JavaScriptSettings">
+    <option name="languageLevel" value="ES6" />
+  </component>
+  <component name="WebPackConfiguration">
+    <option name="path" value="$PROJECT_DIR$/node_modules/@vue/cli-service/webpack.config.js" />
+  </component>
+</project>
\ No newline at end of file
diff --git a/custos-demo-gateway/.idea/modules.xml b/custos-demo-gateway/.idea/modules.xml
new file mode 100644
index 0000000..bdb36b1
--- /dev/null
+++ b/custos-demo-gateway/.idea/modules.xml
@@ -0,0 +1,8 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<project version="4">
+  <component name="ProjectModuleManager">
+    <modules>
+      <module fileurl="file://$PROJECT_DIR$/.idea/custos-demo-gateway.iml" filepath="$PROJECT_DIR$/.idea/custos-demo-gateway.iml" />
+    </modules>
+  </component>
+</project>
\ No newline at end of file
diff --git a/custos-demo-gateway/.idea/vcs.xml b/custos-demo-gateway/.idea/vcs.xml
new file mode 100644
index 0000000..94a25f7
--- /dev/null
+++ b/custos-demo-gateway/.idea/vcs.xml
@@ -0,0 +1,6 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<project version="4">
+  <component name="VcsDirectoryMappings">
+    <mapping directory="$PROJECT_DIR$" vcs="Git" />
+  </component>
+</project>
\ No newline at end of file
diff --git a/custos-demo-gateway/.idea/workspace.xml b/custos-demo-gateway/.idea/workspace.xml
new file mode 100644
index 0000000..1db002d
--- /dev/null
+++ b/custos-demo-gateway/.idea/workspace.xml
@@ -0,0 +1,141 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<project version="4">
+  <component name="ChangeListManager">
+    <list default="true" id="c2cd5109-1431-412f-85ad-eb6f68d4ed8d" name="Default Changelist" comment="">
+      <change afterPath="$PROJECT_DIR$/src/assets/sharings.png" afterDir="false" />
+      <change afterPath="$PROJECT_DIR$/src/components/landing/Landing.vue" afterDir="false" />
+      <change afterPath="$PROJECT_DIR$/src/components/workspace/Workspace.vue" afterDir="false" />
+      <change beforePath="$PROJECT_DIR$/package.json" beforeDir="false" afterPath="$PROJECT_DIR$/package.json" afterDir="false" />
+      <change beforePath="$PROJECT_DIR$/src/App.vue" beforeDir="false" afterPath="$PROJECT_DIR$/src/App.vue" afterDir="false" />
+      <change beforePath="$PROJECT_DIR$/src/components/HelloWorld.vue" beforeDir="false" />
+      <change beforePath="$PROJECT_DIR$/src/main.js" beforeDir="false" afterPath="$PROJECT_DIR$/src/main.js" afterDir="false" />
+      <change beforePath="$PROJECT_DIR$/yarn.lock" beforeDir="false" afterPath="$PROJECT_DIR$/yarn.lock" afterDir="false" />
+    </list>
+    <option name="SHOW_DIALOG" value="false" />
+    <option name="HIGHLIGHT_CONFLICTS" value="true" />
+    <option name="HIGHLIGHT_NON_ACTIVE_CHANGELIST" value="false" />
+    <option name="LAST_RESOLUTION" value="IGNORE" />
+  </component>
+  <component name="FileTemplateManagerImpl">
+    <option name="RECENT_TEMPLATES">
+      <list>
+        <option value="CSS File" />
+        <option value="Vue Single File Component" />
+        <option value="JavaScript File" />
+      </list>
+    </option>
+  </component>
+  <component name="Git.Settings">
+    <option name="RECENT_GIT_ROOT_PATH" value="$PROJECT_DIR$" />
+  </component>
+  <component name="ProjectId" id="1hBYpJOvyxvD4Py2dSNM3iRhs0V" />
+  <component name="ProjectLevelVcsManager" settingsEditedManually="true" />
+  <component name="ProjectViewState">
+    <option name="hideEmptyMiddlePackages" value="true" />
+    <option name="showExcludedFiles" value="true" />
+    <option name="showLibraryContents" value="true" />
+  </component>
+  <component name="PropertiesComponent">
+    <property name="RunOnceActivity.ShowReadmeOnStart" value="true" />
+    <property name="WebServerToolWindowFactoryState" value="false" />
+    <property name="last_opened_file_path" value="$PROJECT_DIR$/src/assets" />
+    <property name="list.type.of.created.stylesheet" value="CSS" />
+    <property name="node.js.detected.package.eslint" value="true" />
+    <property name="node.js.detected.package.standard" value="true" />
+    <property name="node.js.path.for.package.eslint" value="project" />
+    <property name="node.js.path.for.package.standard" value="project" />
+    <property name="node.js.selected.package.eslint" value="(autodetect)" />
+    <property name="node.js.selected.package.standard" value="" />
+    <property name="nodejs_interpreter_path" value="node" />
+    <property name="nodejs_package_manager_path" value="yarn" />
+    <property name="ts.external.directory.path" value="$APPLICATION_HOME_DIR$/plugins/JavaScriptLanguage/jsLanguageServicesImpl/external" />
+  </component>
+  <component name="RecentsManager">
+    <key name="CopyFile.RECENT_KEYS">
+      <recent name="$PROJECT_DIR$/src/assets" />
+    </key>
+    <key name="MoveFile.RECENT_KEYS">
+      <recent name="$PROJECT_DIR$/src/store/modules" />
+      <recent name="$PROJECT_DIR$/src/components/landing" />
+      <recent name="$PROJECT_DIR$/src/components" />
+    </key>
+  </component>
+  <component name="RunManager" selected="npm.npm serve">
+    <configuration name="Debug Application" type="JavascriptDebugType" uri="http://localhost:8080">
+      <method v="2" />
+    </configuration>
+    <configuration name="npm serve" type="js.build_tools.npm">
+      <package-json value="$PROJECT_DIR$/package.json" />
+      <command value="run" />
+      <scripts>
+        <script value="serve" />
+      </scripts>
+      <node-interpreter value="project" />
+      <envs />
+      <method v="2" />
+    </configuration>
+  </component>
+  <component name="ServiceViewManager">
+    <option name="viewStates">
+      <list>
+        <serviceView>
+          <treeState>
+            <expand />
+            <select />
+          </treeState>
+        </serviceView>
+      </list>
+    </option>
+  </component>
+  <component name="SvnConfiguration">
+    <configuration />
+  </component>
+  <component name="TaskManager">
+    <task active="true" id="Default" summary="Default task">
+      <changelist id="c2cd5109-1431-412f-85ad-eb6f68d4ed8d" name="Default Changelist" comment="" />
+      <created>1599486133223</created>
+      <option name="number" value="Default" />
+      <option name="presentableId" value="Default" />
+      <updated>1599486133223</updated>
+      <workItem from="1599486134588" duration="217546000" />
+      <workItem from="1601038400517" duration="39716000" />
+    </task>
+    <servers />
+  </component>
+  <component name="TypeScriptGeneratedFilesManager">
+    <option name="version" value="1" />
+  </component>
+  <component name="WindowStateProjectService">
+    <state x="2240" y="205" key="#ESLint" timestamp="1600964289041">
+      <screen x="1680" y="0" width="1920" height="1080" />
+    </state>
+    <state x="2240" y="205" key="#ESLint/1680.0.1920.1080/0.0.1680.1050@1680.0.1920.1080" timestamp="1600964289041" />
+    <state x="2363" y="350" key="#com.intellij.fileTypes.FileTypeChooser" timestamp="1601165959137">
+      <screen x="1680" y="0" width="1920" height="1080" />
+    </state>
+    <state x="2363" y="350" key="#com.intellij.fileTypes.FileTypeChooser/1680.0.1920.1080/0.0.1680.1050@1680.0.1920.1080" timestamp="1601165959137" />
+    <state x="2378" y="294" key="#com.intellij.refactoring.safeDelete.UnsafeUsagesDialog" timestamp="1601162305471">
+      <screen x="1680" y="0" width="1920" height="1080" />
+    </state>
+    <state x="2378" y="294" key="#com.intellij.refactoring.safeDelete.UnsafeUsagesDialog/1680.0.1920.1080/0.0.1680.1050@1680.0.1920.1080" timestamp="1601162305471" />
+    <state x="2305" y="183" width="667" height="729" key="find.popup" timestamp="1601404607112">
+      <screen x="1680" y="0" width="1920" height="1080" />
+    </state>
+    <state x="2305" y="183" width="667" height="729" key="find.popup/1680.0.1920.1080/0.0.1680.1050@1680.0.1920.1080" timestamp="1601404607112" />
+    <state x="2305" y="241" width="670" height="676" key="search.everywhere.popup" timestamp="1600705258945">
+      <screen x="1680" y="0" width="1920" height="1080" />
+    </state>
+    <state x="2305" y="241" width="670" height="676" key="search.everywhere.popup/1680.0.1920.1080/0.0.1680.1050@1680.0.1920.1080" timestamp="1600705258945" />
+  </component>
+  <component name="XDebuggerManager">
+    <breakpoint-manager>
+      <breakpoints>
+        <line-breakpoint enabled="true" type="javascript">
+          <url>file://$PROJECT_DIR$/src/components/workspace/Users.vue</url>
+          <line>320</line>
+          <option name="timeStamp" value="1" />
+        </line-breakpoint>
+      </breakpoints>
+    </breakpoint-manager>
+  </component>
+</project>
\ No newline at end of file
diff --git a/custos-demo-gateway/Dockerfile b/custos-demo-gateway/Dockerfile
new file mode 100644
index 0000000..879cb74
--- /dev/null
+++ b/custos-demo-gateway/Dockerfile
@@ -0,0 +1,19 @@
+# build stage
+FROM node:lts-alpine as build-stage
+WORKDIR /app
+COPY package*.json ./
+RUN npm install
+COPY . .
+RUN npm run build
+
+# production stage
+FROM nginx:stable-alpine as production-stage
+COPY --from=build-stage /app/dist /usr/share/nginx/html
+COPY nginx.conf /etc/nginx/conf.d/default.conf
+COPY privkey.pem /etc/nginx/privkey.pem
+COPY fullchain.pem /etc/nginx/fullchain.pem
+COPY entrypoint.sh /
+RUN chmod +x /entrypoint.sh
+EXPOSE 443
+ENTRYPOINT [ "/entrypoint.sh" ]
+#CMD ["nginx", "-g", "daemon off;"]
\ No newline at end of file
diff --git a/custos-demo-gateway/README.md b/custos-demo-gateway/README.md
new file mode 100644
index 0000000..dd39c19
--- /dev/null
+++ b/custos-demo-gateway/README.md
@@ -0,0 +1,24 @@
+# custos-demo-gateway
+
+## Project setup
+```
+yarn install
+```
+
+### Compiles and hot-reloads for development
+```
+yarn serve
+```
+
+### Compiles and minifies for production
+```
+yarn build
+```
+
+### Lints and fixes files
+```
+yarn lint
+```
+
+### Customize configuration
+See [Configuration Reference](https://cli.vuejs.org/config/).
diff --git a/custos-demo-gateway/babel.config.js b/custos-demo-gateway/babel.config.js
new file mode 100644
index 0000000..e955840
--- /dev/null
+++ b/custos-demo-gateway/babel.config.js
@@ -0,0 +1,5 @@
+module.exports = {
+  presets: [
+    '@vue/cli-plugin-babel/preset'
+  ]
+}
diff --git a/custos-demo-gateway/docker-compose.yml b/custos-demo-gateway/docker-compose.yml
new file mode 100644
index 0000000..c774049
--- /dev/null
+++ b/custos-demo-gateway/docker-compose.yml
@@ -0,0 +1,11 @@
+version: "3.3"
+services:
+  web:
+    image : "apachecustos/custos-demo-gateway:latest"
+    ports:
+      - "8080:8080"
+    environment:
+      VUE_APP_CLIENT_ID: ''
+      VUE_APP_CLIENT_SEC: ''
+      VUE_APP_REDIRECT_URI: 'http://localhost:8080/callback'
+      VUE_APP_CUSTOS_TOKEN_ENDPOINT: 'https://custos.scigap.org/apiserver/identity-management/v1.0.0/token'
\ No newline at end of file
diff --git a/custos-demo-gateway/entrypoint.sh b/custos-demo-gateway/entrypoint.sh
new file mode 100644
index 0000000..9b9b603
--- /dev/null
+++ b/custos-demo-gateway/entrypoint.sh
@@ -0,0 +1,18 @@
+#!/bin/sh
+
+# Replace env vars in JavaScript files
+echo "Replacing env vars in JS"
+for file in /usr/share/nginx/html/js/app.*.js;
+do
+  echo "Processing $file ...";
+
+  # Use the existing JS file as template
+  if [ ! -f $file.tmpl.js ]; then
+    cp $file $file.tmpl.js
+  fi
+
+  envsubst '$VUE_APP_CLIENT_ID,$VUE_APP_CLIENT_SEC,$VUE_APP_REDIRECT_URI' < $file.tmpl.js > $file
+done
+
+echo "Starting Nginx"
+nginx -g 'daemon off;'
\ No newline at end of file
diff --git a/custos-demo-gateway/fullchain.pem b/custos-demo-gateway/fullchain.pem
new file mode 100644
index 0000000..e6199f0
--- /dev/null
+++ b/custos-demo-gateway/fullchain.pem
@@ -0,0 +1,59 @@
+-----BEGIN CERTIFICATE-----
+MIIFdTCCBF2gAwIBAgISA4dFxqSZr9p+QksSmAdoLUZfMA0GCSqGSIb3DQEBCwUA
+MEoxCzAJBgNVBAYTAlVTMRYwFAYDVQQKEw1MZXQncyBFbmNyeXB0MSMwIQYDVQQD
+ExpMZXQncyBFbmNyeXB0IEF1dGhvcml0eSBYMzAeFw0yMDA5MjgyMzIzMTNaFw0y
+MDEyMjcyMzIzMTNaMCkxJzAlBgNVBAMTHmRlbW8uZ2F0ZXdheS5jdXN0b3Muc2Np
+Z2FwLm9yZzCCASIwDQYJKoZIhvcNAQEBBQADggEPADCCAQoCggEBAMhSqh9FXvM0
+xo8t0oQJ9OGCE9loLbMm2/knZ1ITsRbKTHMczvKkiRBxwgU7kitLDrwolNf5vqQy
+lZrCOjEqda5pi1KwFLNLlYJmV1g+oF5+o0yTTMKUPOnN2+hI9Sjuwwi+/fRMqKsw
+l8SfAum2nDBP9BUuB9MZFThcCm+Cxl55Qz0VXVgzYE+jnNY+BvX4EDaov9mmHkwn
+uXFO6W61k4F/IiD2qhxWmTX8ElwyG8Y6wgNCA/qn5kEPDFCuEXypaVIimDoJyKwD
+VJ84nXlQLrJVLu8Ai6MFKQE6uGOW4tM4zwD9r3zUb8noPJqfwKeE5MEDhQtCA/lh
+PG4UBniquBUCAwEAAaOCAnQwggJwMA4GA1UdDwEB/wQEAwIFoDAdBgNVHSUEFjAU
+BggrBgEFBQcDAQYIKwYBBQUHAwIwDAYDVR0TAQH/BAIwADAdBgNVHQ4EFgQUUrmz
+/B8YDBArAIE6loWileymPe0wHwYDVR0jBBgwFoAUqEpqYwR93brm0Tm3pkVl7/Oo
+7KEwbwYIKwYBBQUHAQEEYzBhMC4GCCsGAQUFBzABhiJodHRwOi8vb2NzcC5pbnQt
+eDMubGV0c2VuY3J5cHQub3JnMC8GCCsGAQUFBzAChiNodHRwOi8vY2VydC5pbnQt
+eDMubGV0c2VuY3J5cHQub3JnLzApBgNVHREEIjAggh5kZW1vLmdhdGV3YXkuY3Vz
+dG9zLnNjaWdhcC5vcmcwTAYDVR0gBEUwQzAIBgZngQwBAgEwNwYLKwYBBAGC3xMB
+AQEwKDAmBggrBgEFBQcCARYaaHR0cDovL2Nwcy5sZXRzZW5jcnlwdC5vcmcwggEF
+BgorBgEEAdZ5AgQCBIH2BIHzAPEAdwBep3P531bA57U2SH3QSeAyepGaDIShEhKE
+GHWWgXFFWAAAAXTXPeyUAAAEAwBIMEYCIQCOpzkBmQCq4YljG/+IYPhuiTyBTBB6
+lyx25017CC3ZmAIhAKaYWDLX/QCeWOVbWzxqXdcAwwPqSA5SL2mDL3SDePzMAHYA
+B7dcG+V9aP/xsMYdIxXHuuZXfFeUt2ruvGE6GmnTohwAAAF01z3s9gAABAMARzBF
+AiAKxZt6ENgjLqtnzkR0mbEUivqEjqtPcNlC4GN+H1EQlAIhAO1oygawEpMs2TAa
+eMhw3J+bJ4P63IRQEdGerqChRs/HMA0GCSqGSIb3DQEBCwUAA4IBAQA3vvDU3ULX
+VbHWFHug54FkWfhVK49V6eGVRImQOiyTdOoQS6MHKvsedg3aBhj3FwRXQzx7VZ61
+xPthZeNooKNUB184eg+R9L2v1HDT1PyGQgXq9MIVJJzXGw9A9W22o68vbMOp5Pw/
+WC5b+ozYbNz38527XE9KBVoETwrsjIg3q/U/beV9MaLt8EfVobOJXb7JvpTvc3me
+ZI6+y2t9Buj4ebIjHA/Ve1SrWEfGVQNjgz86J8JIg7jPV3lv5Gn49JwILSC53bCK
+GSgYLjGPp8Lwtew8NJDTVzRjwquvxky9smVci2GwGUYAv55l07vvv5dHRYKKFF/z
+gUoZxG3CR60/
+-----END CERTIFICATE-----
+-----BEGIN CERTIFICATE-----
+MIIEkjCCA3qgAwIBAgIQCgFBQgAAAVOFc2oLheynCDANBgkqhkiG9w0BAQsFADA/
+MSQwIgYDVQQKExtEaWdpdGFsIFNpZ25hdHVyZSBUcnVzdCBDby4xFzAVBgNVBAMT
+DkRTVCBSb290IENBIFgzMB4XDTE2MDMxNzE2NDA0NloXDTIxMDMxNzE2NDA0Nlow
+SjELMAkGA1UEBhMCVVMxFjAUBgNVBAoTDUxldCdzIEVuY3J5cHQxIzAhBgNVBAMT
+GkxldCdzIEVuY3J5cHQgQXV0aG9yaXR5IFgzMIIBIjANBgkqhkiG9w0BAQEFAAOC
+AQ8AMIIBCgKCAQEAnNMM8FrlLke3cl03g7NoYzDq1zUmGSXhvb418XCSL7e4S0EF
+q6meNQhY7LEqxGiHC6PjdeTm86dicbp5gWAf15Gan/PQeGdxyGkOlZHP/uaZ6WA8
+SMx+yk13EiSdRxta67nsHjcAHJyse6cF6s5K671B5TaYucv9bTyWaN8jKkKQDIZ0
+Z8h/pZq4UmEUEz9l6YKHy9v6Dlb2honzhT+Xhq+w3Brvaw2VFn3EK6BlspkENnWA
+a6xK8xuQSXgvopZPKiAlKQTGdMDQMc2PMTiVFrqoM7hD8bEfwzB/onkxEz0tNvjj
+/PIzark5McWvxI0NHWQWM6r6hCm21AvA2H3DkwIDAQABo4IBfTCCAXkwEgYDVR0T
+AQH/BAgwBgEB/wIBADAOBgNVHQ8BAf8EBAMCAYYwfwYIKwYBBQUHAQEEczBxMDIG
+CCsGAQUFBzABhiZodHRwOi8vaXNyZy50cnVzdGlkLm9jc3AuaWRlbnRydXN0LmNv
+bTA7BggrBgEFBQcwAoYvaHR0cDovL2FwcHMuaWRlbnRydXN0LmNvbS9yb290cy9k
+c3Ryb290Y2F4My5wN2MwHwYDVR0jBBgwFoAUxKexpHsscfrb4UuQdf/EFWCFiRAw
+VAYDVR0gBE0wSzAIBgZngQwBAgEwPwYLKwYBBAGC3xMBAQEwMDAuBggrBgEFBQcC
+ARYiaHR0cDovL2Nwcy5yb290LXgxLmxldHNlbmNyeXB0Lm9yZzA8BgNVHR8ENTAz
+MDGgL6AthitodHRwOi8vY3JsLmlkZW50cnVzdC5jb20vRFNUUk9PVENBWDNDUkwu
+Y3JsMB0GA1UdDgQWBBSoSmpjBH3duubRObemRWXv86jsoTANBgkqhkiG9w0BAQsF
+AAOCAQEA3TPXEfNjWDjdGBX7CVW+dla5cEilaUcne8IkCJLxWh9KEik3JHRRHGJo
+uM2VcGfl96S8TihRzZvoroed6ti6WqEBmtzw3Wodatg+VyOeph4EYpr/1wXKtx8/
+wApIvJSwtmVi4MFU5aMqrSDE6ea73Mj2tcMyo5jMd6jmeWUHK8so/joWUoHOUgwu
+X4Po1QYz+3dszkDqMp4fklxBwXRsW10KXzPMTZ+sOPAveyxindmjkW8lGy+QsRlG
+PfZ+G6Z6h7mjem0Y+iWlkYcV4PIWL1iwBi8saCbGS5jN2p8M+X+Q7UNKEkROb3N6
+KOqkqm57TH2H3eDJAkSnh6/DNFu0Qg==
+-----END CERTIFICATE-----
diff --git a/custos-demo-gateway/nginx.conf b/custos-demo-gateway/nginx.conf
new file mode 100644
index 0000000..de44403
--- /dev/null
+++ b/custos-demo-gateway/nginx.conf
@@ -0,0 +1,35 @@
+gzip on;
+gzip_disable "msie6";
+gzip_types text/plain text/css application/json application/javascript text/xml application/xml application/xml+rss text/javascript image/svg+xml;
+
+server {
+    listen 80;
+    server_name demo.gateway.custos.scigap.org www.demo.gateway.custos.scigap.org;
+    return 301 https://demo.gateway.custos.scigap.org$request_uri;
+}
+
+server {
+  listen 443 ssl;
+  listen [::]:443 ssl;
+
+  ssl_certificate     /etc/nginx/fullchain.pem;
+  ssl_certificate_key /etc/nginx/privkey.pem;
+
+  root /usr/share/nginx/html;
+
+  index index.html;
+
+  location ~ ^/(css|js)/ {
+    # These assets include a digest in the filename, so they will never change
+    expires max;
+  }
+
+  location ~* ^.+\.(html|htm)$ {
+    # Very short caching time to ensure changes are immediately recognized
+    expires 5m;
+  }
+
+  location / {
+    try_files $uri $uri/ /index.html;
+  }
+}
\ No newline at end of file
diff --git a/custos-demo-gateway/package.json b/custos-demo-gateway/package.json
new file mode 100644
index 0000000..2fe7f96
--- /dev/null
+++ b/custos-demo-gateway/package.json
@@ -0,0 +1,50 @@
+{
+  "name": "custos-demo-gateway",
+  "version": "0.1.0",
+  "private": true,
+  "scripts": {
+    "serve": "vue-cli-service serve",
+    "build": "vue-cli-service build",
+    "lint": "vue-cli-service lint"
+  },
+  "dependencies": {
+    "core-js": "^3.6.5",
+    "vue": "^2.6.11",
+    "vue-router": "^3.4.3",
+    "bootstrap-vue": "^2.16.0",
+    "axios": "0.20.0",
+    "vuex": "^3.5.1",
+    "jwt-decode": "^3.0.0-beta.2",
+    "qs": "^6.9.4",
+    "dotenv": "^8.2.0"
+  },
+  "devDependencies": {
+    "@vue/cli-plugin-babel": "~4.5.0",
+    "@vue/cli-plugin-eslint": "~4.5.0",
+    "@vue/cli-service": "~4.5.0",
+    "babel-eslint": "^10.1.0",
+    "eslint": "^6.7.2",
+    "eslint-plugin-vue": "^6.2.2",
+    "vue-template-compiler": "^2.6.11"
+
+  },
+  "eslintConfig": {
+    "root": true,
+    "env": {
+      "node": true
+    },
+    "extends": [
+      "plugin:vue/essential",
+      "eslint:recommended"
+    ],
+    "parserOptions": {
+      "parser": "babel-eslint"
+    },
+    "rules": {}
+  },
+  "browserslist": [
+    "> 1%",
+    "last 2 versions",
+    "not dead"
+  ]
+}
diff --git a/custos-demo-gateway/public/favicon.ico b/custos-demo-gateway/public/favicon.ico
new file mode 100644
index 0000000..df36fcf
Binary files /dev/null and b/custos-demo-gateway/public/favicon.ico differ
diff --git a/custos-demo-gateway/public/index.html b/custos-demo-gateway/public/index.html
new file mode 100644
index 0000000..4123528
--- /dev/null
+++ b/custos-demo-gateway/public/index.html
@@ -0,0 +1,17 @@
+<!DOCTYPE html>
+<html lang="en">
+  <head>
+    <meta charset="utf-8">
+    <meta http-equiv="X-UA-Compatible" content="IE=edge">
+    <meta name="viewport" content="width=device-width,initial-scale=1.0">
+    <link rel="icon" href="<%= BASE_URL %>favicon.ico">
+    <title><%= htmlWebpackPlugin.options.title %></title>
+  </head>
+  <body>
+    <noscript>
+      <strong>We're sorry but <%= htmlWebpackPlugin.options.title %> doesn't work properly without JavaScript enabled. Please enable it to continue.</strong>
+    </noscript>
+    <div id="app"></div>
+    <!-- built files will be auto injected -->
+  </body>
+</html>
diff --git a/custos-demo-gateway/src/App.vue b/custos-demo-gateway/src/App.vue
new file mode 100644
index 0000000..3cb051c
--- /dev/null
+++ b/custos-demo-gateway/src/App.vue
@@ -0,0 +1,29 @@
+<template>
+    <div id="app">
+        <div id="nav">
+            <router-view/>
+        </div>
+    </div>
+</template>
+
+<script>
+    import TheNavigator from "./components/TheNavigator.vue";
+
+    export default {
+        name: 'App',
+        components: [
+            TheNavigator
+        ]
+    }
+</script>
+
+<style>
+    #app {
+        font-family: Avenir, Helvetica, Arial, sans-serif;
+        -webkit-font-smoothing: antialiased;
+        -moz-osx-font-smoothing: grayscale;
+        text-align: center;
+        color: #2c3e50;
+        margin-top: 60px;
+    }
+</style>
diff --git a/custos-demo-gateway/src/assets/bots.png b/custos-demo-gateway/src/assets/bots.png
new file mode 100644
index 0000000..902d442
Binary files /dev/null and b/custos-demo-gateway/src/assets/bots.png differ
diff --git a/custos-demo-gateway/src/assets/credentials.png b/custos-demo-gateway/src/assets/credentials.png
new file mode 100644
index 0000000..ac9a60c
Binary files /dev/null and b/custos-demo-gateway/src/assets/credentials.png differ
diff --git a/custos-demo-gateway/src/assets/custos_home.png b/custos-demo-gateway/src/assets/custos_home.png
new file mode 100644
index 0000000..e89d50a
Binary files /dev/null and b/custos-demo-gateway/src/assets/custos_home.png differ
diff --git a/custos-demo-gateway/src/assets/dblogs.png b/custos-demo-gateway/src/assets/dblogs.png
new file mode 100644
index 0000000..0cc6502
Binary files /dev/null and b/custos-demo-gateway/src/assets/dblogs.png differ
diff --git a/custos-demo-gateway/src/assets/groups_web.png b/custos-demo-gateway/src/assets/groups_web.png
new file mode 100644
index 0000000..1d9ee5b
Binary files /dev/null and b/custos-demo-gateway/src/assets/groups_web.png differ
diff --git a/custos-demo-gateway/src/assets/logo.png b/custos-demo-gateway/src/assets/logo.png
new file mode 100644
index 0000000..f3d2503
Binary files /dev/null and b/custos-demo-gateway/src/assets/logo.png differ
diff --git a/custos-demo-gateway/src/assets/sharings.png b/custos-demo-gateway/src/assets/sharings.png
new file mode 100644
index 0000000..14e7bb1
Binary files /dev/null and b/custos-demo-gateway/src/assets/sharings.png differ
diff --git a/custos-demo-gateway/src/assets/users.png b/custos-demo-gateway/src/assets/users.png
new file mode 100644
index 0000000..274ccdc
Binary files /dev/null and b/custos-demo-gateway/src/assets/users.png differ
diff --git a/custos-demo-gateway/src/components/Callback.vue b/custos-demo-gateway/src/components/Callback.vue
new file mode 100644
index 0000000..ba63380
--- /dev/null
+++ b/custos-demo-gateway/src/components/Callback.vue
@@ -0,0 +1,86 @@
+<template>
+    <p></p>
+</template>
+
+<script>
+
+    import config from "@/config";
+
+    export default {
+        name: "Callback",
+        data: function () {
+            return {
+                username: "",
+                password: "",
+                custosId: null,
+                custosSec: null,
+                redirectURI: null,
+                tokenEndpoint: null
+            }
+        },
+
+        methods: {
+            async authenticate() {
+                let code = this.$route.query.code
+                let params = {
+                    client_id: this.custosId, client_sec: this.custosSec, code: code,
+                    redirect_uri: this.redirectURI, token_endpoint: this.tokenEndpoint
+                };
+                await this.$store.dispatch('identity/authenticateUsingCode', params)
+
+            }
+        },
+
+        computed: {
+            isAuthenticated: function () {
+                let data = {
+                    client_id: this.custosId,
+                    client_sec: this.custosSec
+                }
+                return this.$store.dispatch('identity/isAuthenticated', data)
+            }
+        },
+
+        async mounted() {
+            this.custosId = config.value('clientId')
+            this.custosSec = config.value('clientSec')
+            this.redirectURI = config.value('redirectURI')
+            this.tokenEndpoint = "https://custos.scigap.org/apiserver/identity-management/v1.0.0/token"
+            await this.authenticate()
+            console.log("calling authenticate")
+            if (this.isAuthenticated) {
+                console.log("calling authenticated")
+                let username = await this.$store.dispatch('identity/getCurrentUserName')
+                await this.$router.push('workspace')
+                let data = {
+                    offset: 0, limit: 1, client_id: this.custosId, client_sec: this.custosSec,
+                    username: username
+                }
+                let resp = await this.$store.dispatch('user/users', data)
+                console.log(resp)
+                if (Array.isArray(resp) && resp.length > 0) {
+                    resp.forEach(user => {
+                        let dat = {
+                            client_id: this.custosId,
+                            client_sec: this.custosSec,
+                            body: {
+                                username: user.username,
+                                first_name: user.first_name,
+                                last_name: user.last_name,
+                                email: user.email,
+                            }
+                        }
+                        console.log(dat)
+                        this.$store.dispatch('user/updateUserProfile', dat)
+                    })
+                }
+            } else {
+                await this.$router.push('home')
+            }
+        }
+    }
+</script>
+
+<style scoped>
+
+</style>
\ No newline at end of file
diff --git a/custos-demo-gateway/src/components/TheNavigator.vue b/custos-demo-gateway/src/components/TheNavigator.vue
new file mode 100644
index 0000000..ed9e715
--- /dev/null
+++ b/custos-demo-gateway/src/components/TheNavigator.vue
@@ -0,0 +1,24 @@
+<template>
+    <nav id ="nav">
+        <router-link to="/">Landing Page</router-link>
+        <router-link to="/register">User Account Creation</router-link>
+        <router-link to="/workspace">Work Space</router-link>
+        <router-link to="/workspace/groups">Work Space</router-link>
+        <router-link to="/workspace/users">Work Space</router-link>
+        <router-link to="/workspace/logs">Work Space</router-link>
+        <router-link to="/workspace/secrets">Work Space</router-link>
+        <router-link to="/workspace/sharings">Work Space</router-link>
+        <router-link to="/callback">Work Space</router-link>
+    </nav>
+    
+</template>
+
+<script>
+    export default {
+        name: "TheNavigator"
+    }
+</script>
+
+<style scoped>
+
+</style>
\ No newline at end of file
diff --git a/custos-demo-gateway/src/components/landing/Landing.vue b/custos-demo-gateway/src/components/landing/Landing.vue
new file mode 100644
index 0000000..0def2cd
--- /dev/null
+++ b/custos-demo-gateway/src/components/landing/Landing.vue
@@ -0,0 +1,229 @@
+<template>
+    <div class="landing">
+
+        <div class="row">
+
+            <div class="column">
+                <h2>Welcome to Custos Explorer</h2>
+                <p>SignUp and explore capabilities</p>
+                <img src="./../../assets/custos_home.png">
+            </div>
+            <div class="column">
+                <div class="login">
+                    <div class="exlogin">
+                        <p>Login with your existing organization</p>
+                        <b-button variant="primary" v-on:click="this.loadAuthURL">Login with Institution</b-button>
+                    </div>
+                    <div class="locallogin">
+                        <p>Login with Custos Explorer (Not registered ?
+                            <router-link to="/register">Create Account</router-link>
+                            )
+                        </p>
+
+                        <div class="logininput">
+                            <b-form-input v-model="username" placeholder="Username"></b-form-input>
+
+                        </div>
+                        <div class="logininput">
+                            <b-form-input type="password" v-model="password" placeholder="Password"></b-form-input>
+                        </div>
+                        <div class="logininput">
+                            <b-button variant="primary" v-on:click="this.login" :disabled="this.loginDisabled">
+                                <div v-if="this.loginDisabled">
+                                    <b-spinner small></b-spinner>
+                                </div>
+                                Login
+                            </b-button>
+                        </div>
+                    </div>
+                    <div>
+                        <b-alert v-model="this.loginError" variant="danger" dismissible @dismissed="this.callDismissed">
+                            Invalid Username or Password
+                        </b-alert>
+                    </div>
+
+
+                </div>
+
+            </div>
+        </div>
+
+    </div>
+</template>
+
+<script>
+
+    import config from "@/config";
+
+    export default {
+        name: 'Landing',
+        props: {
+            msg: String,
+            seen: Boolean,
+            todos: Array
+        },
+        data: function () {
+            return {
+                username: "",
+                password: "",
+                custosId: null,
+                custosSec: null,
+                loginDisabled: false,
+                redirectURI: null,
+                loginError: false
+            }
+        },
+        methods: {
+
+            async login() {
+                this.loginDisabled = true
+                if (this.username != null && this.username != '' && this.password != null && this.password != '') {
+                    let params = {
+                        client_id: this.custosId, client_sec: this.custosSec, username: this.username,
+                        password: this.password, token_endpoint: this.tokenEndpoint
+                    };
+                    let data = {
+                        client_id: this.custosId,
+                        client_sec: this.custosSec
+                    }
+                    await this.$store.dispatch('identity/authenticateLocally', params)
+                    let resp = await this.$store.dispatch('identity/isAuthenticated', data)
+                    if (resp) {
+                        await this.$router.push('workspace')
+                        let data = {
+                            offset: 0, limit: 1, client_id: this.custosId, client_sec: this.custosSec,
+                            username: this.username
+                        }
+                        let resp = await this.$store.dispatch('user/users', data)
+                        if (Array.isArray(resp) && resp.length > 0) {
+                            resp.forEach(user => {
+                                let data = {
+                                    client_id: this.custosId,
+                                    client_sec: this.custosSec,
+                                    body: {
+                                        username: user.username,
+                                        first_name: user.first_name,
+                                        last_name: user.last_name,
+                                        email: user.email,
+                                    }
+                                }
+                                this.$store.dispatch('user/updateUserProfile', data)
+
+                            })
+                        }
+
+                    } else {
+                        this.loginError = true
+                    }
+                } else {
+                    this.loginError = true
+                }
+                this.loginDisabled = false
+            },
+
+
+            async callDismissed() {
+                this.loginError = false
+            },
+            async loadAuthURL() {
+                let params = {client_id: this.custosId, redirect_uri: this.redirectURI};
+                await this.$store.dispatch('identity/fetchAuthorizationEndpoint', params)
+                window.location.href = this.$store.getters['identity/getAuthorizationEndpoint']
+            }
+        },
+
+        mounted() {
+            this.custosId = config.value('clientId')
+            this.custosSec = config.value('clientSec')
+            this.redirectURI = config.value('redirectURI')
+            this.tokenEndpoint = "https://custos.scigap.org/apiserver/identity-management/v1.0.0/token"
+
+        },
+
+        computed: {
+            // ...mapGetters({
+            //
+            // })
+        }
+
+
+    }
+</script>
+
+
+<!-- Add "scoped" attribute to limit CSS to this component only -->
+<style scoped>
+    h2 {
+        margin: 40px 0 0;
+        font-size: xx-large;
+        font-family: "Arial";;
+    }
+
+    p {
+        font-size: medium;
+        font-family: "Arial";
+    }
+
+    ul {
+        list-style-type: none;
+        padding: 0;
+    }
+
+    li {
+        display: inline-block;
+        margin: 0 10px;
+    }
+
+    a {
+        color: #42b983;
+    }
+
+    * {
+        box-sizing: border-box;
+    }
+
+    /* Create two equal columns that floats next to each other */
+    .column {
+        float: left;
+        width: 50%;
+        padding: 10px;
+        height: 300px; /* Should be removed. Only for demonstration */
+    }
+
+    /* Clear floats after the columns */
+    .row:after {
+        content: "";
+        display: table;
+        clear: both;
+    }
+
+    .login {
+        height: 10%;
+        position: relative;
+        top: 50%;
+        z-index: 2;
+        width: 60%;
+        left: 20%;
+    }
+
+    .exlogin {
+        background-color: whitesmoke;
+        box-sizing: border-box;
+        margin-top: 20%;
+    }
+
+    .locallogin {
+        background-color: whitesmoke;
+        box-sizing: border-box;
+        position: relative;
+        margin-top: 7%;
+    }
+
+    .logininput {
+        width: 60%;
+        position: relative;
+        left: 20%;
+        margin-top: 4%;
+    }
+
+</style>
diff --git a/custos-demo-gateway/src/components/registration/CreateAccount.vue b/custos-demo-gateway/src/components/registration/CreateAccount.vue
new file mode 100644
index 0000000..798f3d7
--- /dev/null
+++ b/custos-demo-gateway/src/components/registration/CreateAccount.vue
@@ -0,0 +1,206 @@
+<template>
+    <div class="account">
+        <div class="row">
+            <div class="column">
+                <div class="login">
+                    <div class="locallogin">
+                        <h2> Create account </h2>
+
+                        <div class="logininput">
+                            <p>Username</p>
+                        </div>
+                        <div class="logininput">
+                            <b-form-input v-model="username" placeholder="Username"></b-form-input>
+                        </div>
+                        <div class="logininput">
+                            <p>Password</p>
+                        </div>
+                        <div class="logininput">
+                            <b-form-input type="password" v-model="password" placeholder="Password"></b-form-input>
+                        </div>
+                        <div class="logininput">
+                            <p>Password</p>
+                        </div>
+                        <div class="logininput">
+                            <b-form-input type="password" v-model="confirmedPassword"
+                                          placeholder="Confirm Password"></b-form-input>
+                        </div>
+                        <div class="logininput">
+                            <p>Email</p>
+                        </div>
+                        <div class="logininput">
+                            <b-form-input type="email" v-model="email" placeholder="Email"></b-form-input>
+                        </div>
+                        <div class="logininput">
+                            <p>First Name</p>
+                        </div>
+                        <div class="logininput">
+                            <b-form-input type="email" v-model="firstName" placeholder="First Name"></b-form-input>
+                        </div>
+                        <div class="logininput">
+                            <p>Last Name</p>
+                        </div>
+                        <div class="logininput">
+                            <b-form-input type="email" v-model="lastName" placeholder="Last Name"></b-form-input>
+                        </div>
+                        <div class="logininput">
+                            <b-button variant="primary" v-on:click="this.registerUser" :disabled="this.isButtonDisabled">
+                                <div v-if="this.isButtonDisabled">
+                                    <b-spinner small></b-spinner>
+                                </div>
+                                Create Account</b-button>
+                        </div>
+                    </div>
+                </div>
+
+            </div>
+        </div>
+
+
+    </div>
+</template>
+
+<script>
+    import config from "@/config";
+
+    export default {
+        name: "CreateAccount",
+        data: function () {
+            return {
+                username: "",
+                password: "",
+                confirmedPassword: null,
+                email: null,
+                firstName: null,
+                lastName: null,
+                custosId: null,
+                custosSec: null,
+                isButtonDisabled:false
+            }
+        },
+
+        methods: {
+
+            async registerUser() {
+                this.isButtonDisabled = true
+                if (this.password != null && this.confirmedPassword != null && this.password === this.confirmedPassword) {
+
+                    if (this.username != null && this.firstName != null && this.lastName != null) {
+                        let body = {
+                            'client_id': this.custosId,
+                            'client_sec': this.custosSec,
+                            'username': this.username,
+                            'first_name': this.firstName,
+                            'last_name': this.lastName,
+                            'password': this.password,
+                            'temporary_password': false,
+                            'email': this.email
+                        }
+                        let avilable = await this.$store.dispatch('user/checkUsernameIsValid', body)
+                        if (avilable) {
+                            let status = await this.$store.dispatch('user/registerUser', body)
+                            if (status) {
+                                await this.$router.push("/")
+                            } else {
+                                alert("Some error occurred")
+                            }
+
+                        } else {
+                            alert("Username is not valid")
+                        }
+                    } else {
+                        alert("fill all fields")
+                    }
+
+                } else {
+                    alert("password mismatch")
+                }
+                this.isButtonDisabled = true
+            },
+
+        },
+
+        mounted() {
+            this.custosId = config.value('clientId')
+            this.custosSec = config.value('clientSec')
+        }
+    }
+</script>
+
+<!-- Add "scoped" attribute to limit CSS to this component only -->
+<style scoped>
+    h2 {
+        margin: 40px 0 0;
+        font-size: xx-large;
+        font-family: "Arial";;
+    }
+
+    p {
+        font-size: medium;
+        font-family: "Arial";
+        text-align: left;
+    }
+
+    ul {
+        list-style-type: none;
+        padding: 0;
+    }
+
+    li {
+        display: inline-block;
+        margin: 0 10px;
+    }
+
+    a {
+        color: #42b983;
+    }
+
+    * {
+        box-sizing: border-box;
+    }
+
+    /* Create two equal columns that floats next to each other */
+    .column {
+        float: left;
+        width: 50%;
+        padding: 10px;
+        height: 300px; /* Should be removed. Only for demonstration */
+    }
+
+    /* Clear floats after the columns */
+    .row:after {
+        content: "";
+        display: table;
+        clear: both;
+    }
+
+    .login {
+        height: 10%;
+        position: relative;
+        margin-top: 5%;
+        z-index: 2;
+        width: 60%;
+        left: 20%;
+    }
+
+    .exlogin {
+        background-color: whitesmoke;
+        box-sizing: border-box;
+        margin-top: 20%;
+    }
+
+    .locallogin {
+        background-color: whitesmoke;
+        box-sizing: border-box;
+        position: relative;
+        margin-top: 7%;
+    }
+
+    .logininput {
+        width: 60%;
+        position: relative;
+        left: 20%;
+        margin-top: 2%;
+    }
+
+</style>
\ No newline at end of file
diff --git a/custos-demo-gateway/src/components/workspace/Agents.vue b/custos-demo-gateway/src/components/workspace/Agents.vue
new file mode 100644
index 0000000..7dee536
--- /dev/null
+++ b/custos-demo-gateway/src/components/workspace/Agents.vue
@@ -0,0 +1,295 @@
+<template>
+    <div class="accounttable">
+<!--        <div class="enableComAcc">-->
+<!--            <b-form-checkbox v-model="checked" :disabled=isCheckedBtnDisabled v-on:change="enableAgents"-->
+<!--                             name="check-button" switch>-->
+<!--                Enable Community Accounts-->
+<!--            </b-form-checkbox>-->
+<!--        </div>-->
+            <div v-if="this.loadingAgents" class="d-flex justify-content-center mb-3">
+                <b-spinner variant="primary" label="Text Centered"></b-spinner>
+            </div>
+            <b-table striped hover responsive :items="communityAccounts" :fields="community_fields" selectable
+                     ref="selectableTable"
+                     select-mode="single"
+                     @row-selected="onCommunityAcSelected" caption-top>
+                <template v-slot:table-caption>Community Accounts</template>
+            </b-table>
+            <div class="addAccItem">
+                <b-button variant="outline-primary" v-on:click="this.addAccount">Add Account</b-button>
+            </div>
+
+        <b-modal ref="newAcc" scrollable title="New Account" ok-title="Ok" @ok="registerAgent">
+            <div class="userform">
+                <div class="userformItem">
+                    <p>Id</p>
+                    <b-form-input v-model="newAccountName"></b-form-input>
+                </div>
+            </div>
+        </b-modal>
+        <b-modal ref="exAcc" scrollable title=" Account" ok-title="Ok" hide-footer>
+            <div class="userform">
+                <div class="userformItem">
+                    <p>Id</p>
+                    <b-form-input v-model="exAccountId" disabled></b-form-input>
+                </div>
+                <div class="userform">
+                    <p>Status</p>
+                    <b-form-select v-model="selectedStatus" v-on:change="changeAccountStatus">
+                        <option v-for="(selectOption, indexOpt) in statusOptions"
+                                :key="indexOpt"
+                                :value="selectOption"
+                        >
+                            {{ selectOption }}
+                        </option>
+                    </b-form-select>
+                </div>
+                <div class="userformItem">
+                    <p>Attributes</p>
+                    <b-table striped hover responsive :items="accSelectedAttributes" selectable select-mode="single"
+                             @row-selected="onAccAtrSelected">
+                    </b-table>
+                    <dev class="addAtrCls">
+                        <b-button variant="outline-primary" v-on:click="addActAttribute">Add Attributes</b-button>
+                    </dev>
+                </div>
+            </div>
+        </b-modal>
+        <b-modal ref="accAtrModel" scrollable title="Add Attribute" ok-title="Add" @ok="addAtrOkPressed">
+            <div class="userform">
+                <div class="userformItem">
+                    <p>Key</p>
+                    <b-form-input v-model="newAcAtrKey"></b-form-input>
+                </div>
+                <div class="userformItem">
+                    <p>Value</p>
+                    <b-form-input v-model="newAcAtrValue"></b-form-input>
+                </div>
+            </div>
+        </b-modal>
+        <b-modal ref="AccAtrModelSelected" scrollable title="Attribute" ok-title="Delete" @ok="deleteAtrOkPressed">
+            <div class="userform">
+                <div class="userformItem">
+                    <p>Key</p>
+                    <b-form-input v-model="exAccountAtrKey" disabled></b-form-input>
+                </div>
+                <div class="userformItem">
+                    <p>Value</p>
+                    <b-form-input v-model="exAccountAtrValue" disabled></b-form-input>
+                </div>
+            </div>
+        </b-modal>
+        <b-modal ref="enablePopOver" scrollable title="Agent Credentials" cancel-disabled>
+            <div class="userform">
+                <div class="userformItem">
+                    <p>Id</p>
+                    <b-form-input v-model="agentId" disabled></b-form-input>
+                </div>
+                <div class="userformItem">
+                    <p>Secret</p>
+                    <b-form-input v-model="agentSec" disabled></b-form-input>
+                </div>
+            </div>
+        </b-modal>
+
+    </div>
+</template>
+
+<script>
+    import config from "@/config";
+
+    export default {
+        name: "Agents",
+        data: function () {
+            return {
+                community_fields: ["id", "status"],
+                communityAccounts: [],
+                statusOptions: ['ACTIVE', 'DISABLED'],
+                scopes: ['TENANT', 'CLIENT'],
+                checked: true,
+                newAccountName: null,
+                exAccountId: null,
+                exAccount: null,
+                accSelectedAttributes: [],
+                accSelectedRoles: [],
+                exAccountAtr: null,
+                exAccountRole: null,
+                exAccountAtrKey: null,
+                exAccountAtrValue: null,
+                exAccountRoleKey: null,
+                exAccountRoleValue: null,
+                custosId: null,
+                custosSec: null,
+                isAdminUser: false,
+                tenantroles: [],
+                clientroles: [],
+                newAcAtrKey: null,
+                newAcAtrValue: null,
+                agentId: null,
+                agentSec: null,
+                selectedStatus: null,
+                loadingAgents:false
+
+
+            }
+        },
+
+        methods: {
+            async onCommunityAcSelected(items) {
+                if (items != null && items.length > 0) {
+                    this.exAccount = items
+                    this.exAccountId = this.exAccount[0].id
+                    let act = await this.$store.getters['identity/getAccessToken']
+                    let dt = {
+                        user_token: act,
+                        agent_id: this.exAccountId
+                    }
+                    let agent = await this.$store.dispatch('agent/getAgent', dt)
+                    let atrs = []
+                    agent.attributes.forEach(at => {
+                        atrs.push({key: at.key, value: at.values.join("")})
+                    })
+                    this.accSelectedAttributes = atrs
+                    this.selectedStatus = this.exAccount[0].status
+                    this.$refs.exAcc.show()
+                }
+            },
+            addAccount: function () {
+                this.newAccountName = null
+                this.$refs.newAcc.show()
+            },
+
+            onAccAtrSelected: function (items) {
+                this.exAccountAtr = items
+                this.exAccountAtrKey = this.exAccountAtr[0].key
+                this.exAccountAtrValue = this.exAccountAtr[0].value
+                this.$refs.AccAtrModelSelected.show()
+            },
+
+            onAccRoleSelected: function (items) {
+                this.exAccountRole = items
+                this.exAccountRoleKey = this.exAccountRole[0].name
+                this.exAccountRoleValue = this.exAccountRole[0].scope
+                this.$refs.accRoleModelSelected.show()
+            },
+
+            addActAttribute: function () {
+                this.$refs.accAtrModel.show()
+            },
+            addActRole: function () {
+                this.$refs.accRoleModel.show()
+            },
+            async enableAgents() {
+                let accessToken = await this.$store.getters['identity/getAccessToken']
+                let data = {
+                    user_token: accessToken,
+                }
+                await this.$store.dispatch('agent/enableAgents', data)
+            },
+
+            async registerAgent() {
+                let accessToken = await this.$store.getters['identity/getAccessToken']
+                let data = {
+                    user_token: accessToken,
+                    body: {
+                        id: this.newAccountName
+                    }
+                }
+                let response = await this.$store.dispatch('agent/registerAgent', data)
+                let agents = await this.$store.getters['agent/getAgents']
+                this.communityAccounts = agents
+                this.agentId = response.id
+                this.agentSec = response.secret
+                this.$refs.enablePopOver.show()
+            },
+
+            async addAtrOkPressed() {
+                let accessToken = await this.$store.getters['identity/getAccessToken']
+                let data = {
+                    user_token: accessToken,
+                    body: {
+                        agents: [this.exAccountId],
+                        attributes: [{
+                            key: this.newAcAtrKey,
+                            values: [this.newAcAtrValue]
+                        }]
+                    }
+                }
+                let atr = {key: this.newAcAtrKey, value: this.newAcAtrValue}
+                this.accSelectedAttributes.push(atr)
+                await this.$store.dispatch('agent/addAttributeToAgent', data)
+                this.newAcAtrKey = null
+                this.newAcAtrValue = null
+            },
+
+            async deleteAtrOkPressed() {
+                let accessToken = await this.$store.getters['identity/getAccessToken']
+                let data = {
+                    user_token: accessToken,
+                    body: {
+                        agents: [this.exAccountId],
+                        attributes: [{
+                            key: this.exAccountAtrKey,
+                            values: [this.exAccountAtrValue]
+                        }]
+                    }
+                }
+
+                let acSelAtrs = []
+                this.accSelectedAttributes.forEach(atr => {
+                    if (atr.key != this.exAccountAtrKey) {
+                        acSelAtrs.push(atr)
+                    }
+                })
+                this.accSelectedAttributes = acSelAtrs
+                await this.$store.dispatch('agent/deleteAttributeFromAgent', data)
+                this.exAccountAtrKey = null
+                this.exAccountAtrValue = null
+            },
+            async changeAccountStatus() {
+                let accessToken = await this.$store.getters['identity/getAccessToken']
+                let data = {
+                    user_token: accessToken,
+                    agent_id: this.exAccountId
+                }
+                if (this.selectedStatus == 'ACTIVE') {
+                    await this.$store.dispatch('agent/activateAgent', data)
+                } else {
+                    await this.$store.dispatch('agent/deactivateAgent', data)
+                }
+            }
+
+        },
+
+        async mounted() {
+            this.custosId = config.value('clientId')
+            this.custosSec = config.value('clientSec')
+            this.isAdminUser =  await this.$store.dispatch('identity/isLoggedUserHasAdminAccess')
+
+            let accessToken = await this.$store.getters['identity/getAccessToken']
+            let data = {
+                user_token: accessToken
+            }
+            this.loadingAgents = true
+            await this.enableAgents()
+            this.communityAccounts = await this.$store.dispatch('agent/get_all_agents', data)
+            this.loadingAgents = false
+        }
+    }
+</script>
+
+<style scoped>
+    .enableComAcc {
+        width: 70%;
+    }
+
+    .addAccItem {
+        margin-left: 70%;
+    }
+
+    .accounttable {
+        width: 30%;
+        margin-left: 30%;
+        margin-top: 3%;
+    }
+</style>
\ No newline at end of file
diff --git a/custos-demo-gateway/src/components/workspace/Groups.vue b/custos-demo-gateway/src/components/workspace/Groups.vue
new file mode 100644
index 0000000..2cce24f
--- /dev/null
+++ b/custos-demo-gateway/src/components/workspace/Groups.vue
@@ -0,0 +1,653 @@
+<template>
+    <div>
+        <div class="grouptable">
+            <div v-if="this.groupsLoading" class="d-flex justify-content-center mb-3">
+                <b-spinner variant="primary" label="Text Centered"></b-spinner>
+            </div>
+            <b-table striped hover responsive :items="groupItems" :fields="fields" selectable
+                     ref="selectableTable"
+                     select-mode="single"
+                     @row-selected="onRowSelected" caption-top>
+                <template v-slot:table-caption>Groups</template>
+            </b-table>
+            <div class="addGr">
+                <b-button variant="outline-primary" v-on:click="addGr">Add Group</b-button>
+            </div>
+        </div>
+
+        <div>
+            <b-modal ref="groupmodel" scrollable title="Group Profile" ok-title="Update">
+                <div class="groupform">
+                    <div class="groupformItem">
+                        <p>Id</p>
+                        <b-form-input v-model="selectedId" disabled></b-form-input>
+                    </div>
+                    <div class="groupformItem">
+                        <p>Name</p>
+                        <b-form-input v-model="selectedName" disabled></b-form-input>
+                    </div>
+                    <div class="groupformItem">
+                        <p>Description</p>
+                        <b-form-input v-model="selectedDescription"></b-form-input>
+                    </div>
+                    <div class="groupformItem">
+                        <p>OwnerId</p>
+                        <b-form-input v-model="selectedOwnerId" disabled></b-form-input>
+                    </div>
+                    <div v-if="!this.operationCompleted" class="d-flex justify-content-center mb-3">
+                        <b-spinner variant="primary" label="Text Centered"></b-spinner>
+                    </div>
+                    <div class="groupformItem">
+                        <p>Members</p>
+                        <b-table striped hover responsive :items="members" selectable select-mode="single"
+                                 @row-selected="onMemberShipSelected">
+                        </b-table>
+                    </div>
+
+                    <div class="w-100">
+                        <b-button class="addmemberbtn" variant="outline-primary" v-on:click="addMemberShip">Add Member
+                        </b-button>
+                    </div>
+                    <div class="groupformItem">
+                        <p>Child Groups</p>
+                        <b-table striped hover responsive :items="childGroupMembers" :fields="memberGroupsFields"
+                                 selectable select-mode="single"
+                                 @row-selected="onGroupMemberShipSelected">
+                        </b-table>
+                    </div>
+                    <div class="w-100">
+                        <b-button class="addmemberbtn" variant="outline-primary" v-on:click="addChildGroup">Add Child
+                            Group
+                        </b-button>
+                    </div>
+
+                </div>
+                <template v-slot:modal-footer>
+                    <div class="w-100">
+                        <b-button
+                                variant="primary"
+                                size="sm"
+                                class="grBtnDel"
+                                v-on:click="removeGroupProfile"
+                                @click="show=false"
+                        >
+                            Delete
+                        </b-button>
+                        <b-button
+                                variant="primary"
+                                size="sm"
+                                class="grBtnCl"
+                                v-on:click="closeGroupProfile"
+                                @click="show=false"
+
+                        >
+                            Close
+                        </b-button>
+                        <b-button
+                                variant="primary"
+                                size="sm"
+                                class="grBtnUp"
+                                v-on:click="updateGroupProfile"
+                                @click="show=false"
+
+                        >
+                            Update
+                        </b-button>
+                    </div>
+                </template>
+            </b-modal>
+        </div>
+        <div>
+            <div>
+                <b-modal ref="membershipModel" title="Update Membership" ok-title="Update">
+                    <div class="groupform">
+                        <div class="groupformItem">
+                            <p>Owner Id</p>
+                            <b-form-input v-model="selectedMembershipUsername" disabled></b-form-input>
+                        </div>
+                        <div class="groupformItem">
+                            <p>Type</p>
+                            <b-form-select v-model="selectedMembershipType">
+                                <option v-for="(selectOption, indexOpt) in memberTypes"
+                                        :key="indexOpt"
+                                        :value="selectOption"
+                                >
+                                    {{ selectOption }}
+                                </option>
+                            </b-form-select>
+                        </div>
+
+                    </div>
+                    <template v-slot:modal-footer>
+                        <div class="w-100">
+                            <b-button
+                                    variant="primary"
+                                    size="sm"
+                                    class="memBtnDel"
+                                    v-on:click="removeMembership"
+                                    @click="show=false"
+                            >
+                                Remove Membership
+                            </b-button>
+                            <b-button
+                                    variant="primary"
+                                    size="sm"
+                                    class="memBtnCl"
+                                    v-on:click="closeMembershipModel"
+                                    @click="show=false"
+
+                            >
+                                Close
+                            </b-button>
+                            <b-button
+                                    variant="primary"
+                                    size="sm"
+                                    class="memBtnUp"
+                                    v-on:click="updateMembership"
+                                    @click="show=false"
+
+                            >
+                                Update
+                            </b-button>
+                        </div>
+                    </template>
+                </b-modal>
+            </div>
+            <div>
+                <b-modal ref="addMembershipModel" title="Add Membership" ok-title="Add" @ok="addMembershipOKPressed">
+                    <div class="groupform">
+                        <div v-if="selectedChildType==='User'" class="groupformItem">
+                            <p>Username</p>
+                            <b-form-select v-model="selectedNewUsername">
+                                <option v-for="(selectOption, indexOpt) in usernames"
+                                        :key="indexOpt"
+                                        :value="selectOption"
+                                >
+                                    {{ selectOption }}
+                                </option>
+                            </b-form-select>
+                        </div>
+                        <div v-if="selectedChildType==='User'" class="groupformItem">
+                            <p>Type</p>
+                            <b-form-select v-model="selectedNewMemType">
+                                <option v-for="(selectOption, indexOpt) in memberTypes"
+                                        :key="indexOpt"
+                                        :value="selectOption"
+                                >
+                                    {{ selectOption }}
+                                </option>
+                            </b-form-select>
+                        </div>
+
+                    </div>
+                </b-modal>
+            </div>
+            <div>
+                <b-modal ref="addGrModel" title="Add Group" ok-title="Add" @ok="addGroupOKPressed">
+                    <div class="groupform">
+                        <div class="groupformItem">
+                            <p>Name</p>
+                            <b-form-input v-model="selectedNewGrName"></b-form-input>
+                        </div>
+                        <div class="groupformItem">
+                            <p>Description</p>
+                            <b-form-input v-model="selectedNewGrDesc"></b-form-input>
+                        </div>
+
+                    </div>
+                </b-modal>
+                <b-modal ref="viewGrMembership" title="Group" cancel-title="Delete" @cancel="removeGroupMembership">
+                    <div class="groupform">
+                        <div class="groupformItem">
+                            <p>Name</p>
+                            <b-form-input v-model="selectedGrName"></b-form-input>
+                        </div>
+                        <div class="groupformItem">
+                            <p>Id</p>
+                            <b-form-input v-model="selectedGrId"></b-form-input>
+                        </div>
+
+                    </div>
+                </b-modal>
+                <b-modal ref="addGrMembershipModel" title="Add Group Membership" ok-title="Add"
+                         @ok="addChildGroupOkPressed">
+                    <div class="groupform">
+                        <p>Select Group Id</p>
+                        <b-form-select v-model="addingGr">
+                            <option v-for="(selectOption, indexOpt) in feasibleGroupMembers"
+                                    :key="indexOpt"
+                                    :value="selectOption"
+                            >
+                                {{ selectOption.id }}
+                            </option>
+                        </b-form-select>
+                    </div>
+                </b-modal>
+            </div>
+        </div>
+    </div>
+
+</template>
+
+<script>
+
+    import config from "@/config";
+
+    export default {
+        name: "Groups",
+        data: function () {
+            return {
+                fields: ['id', 'name', 'description', 'ownerId'],
+                groupItems: [],
+                members: [],
+                childGroupMembers: [],
+                feasibleGroupMembers: [],
+                usernames: [],
+                exGroups: [
+                    '1234-edrfg-567-rfgbn', '1234-ed-5634-rfn'
+                ],
+                memberTypes: [
+                    'ADMIN', 'MEMBER'],
+                memberGroupsFields: ['id', 'name'],
+                selectedId: null,
+                selectedName: null,
+                selectedDescription: null,
+                selectedOwnerId: null,
+                selectedMembershipItem: null,
+                selectedMembershipUsername: null,
+                selectedMembershipType: null,
+                selectedNewMemType: 'Member',
+                selectedNewUsername: null,
+                selectedNewGrName: null,
+                selectedNewGrDesc: null,
+                selectedChildType: 'User',
+                selectedExGr: null,
+                selectedGrName: null,
+                selectedGrId: null,
+                selectedGr: null,
+                addingGr: null,
+                groupsLoading: false,
+                operationCompleted: true,
+
+            }
+        },
+        methods: {
+            async onRowSelected(items) {
+                if (items != null && items.length > 0) {
+                    this.selectedItem = items
+                    this.selectedId = this.selectedItem[0].id
+                    this.selectedName = this.selectedItem[0].name
+                    this.selectedDescription = this.selectedItem[0].description
+                    this.selectedOwnerId = this.selectedItem[0].ownerId
+
+                    let data = {
+                        client_id: this.custosId,
+                        client_sec: this.custosSec,
+                        group_id: this.selectedId
+                    }
+
+                    let users = await this.$store.dispatch('group/getAllChildUsers', data)
+
+                    this.members = []
+                    users.profiles.forEach(user => {
+
+                        let usr = {user_id: user.username, type: user.membership_type}
+                        this.members.push(usr)
+                    })
+
+                    let response = await this.$store.dispatch('group/getAllChildGroups', data)
+                    this.childGroupMembers = response.groups
+
+                    this.$refs.groupmodel.show()
+                }
+            },
+            onMemberShipSelected: function (items) {
+                this.selectedMembershipItem = items
+
+                this.selectedMembershipUsername = this.selectedMembershipItem[0].user_id
+                this.selectedMembershipType = this.selectedMembershipItem[0].type
+                if (this.selectedMembershipType !== 'OWNER') {
+                    this.$refs.membershipModel.show()
+                }
+            },
+
+            onGroupMemberShipSelected: function (items) {
+                if (items != null && items.length > 0) {
+                    this.selectedGr = items
+                    this.selectedGrId = this.selectedGr[0].id
+                    this.selectedGrName = this.selectedGr[0].name
+                    this.$refs.viewGrMembership.show()
+                }
+            },
+
+            addChildGroup: function () {
+                let grs = []
+                this.groupItems.forEach(gr => {
+                    let addToGroup = true
+                    this.childGroupMembers.forEach(lx => {
+                        if (gr.id === lx.id) {
+                            addToGroup = false
+                        }
+                    })
+                    if (gr.id === this.selectedId) {
+                        addToGroup = false
+                    }
+                    if (addToGroup) {
+                        grs.push(gr)
+                    }
+                })
+                this.feasibleGroupMembers = grs
+                this.$refs.addGrMembershipModel.show()
+            },
+
+            async addChildGroupOkPressed() {
+                this.operationCompleted = false
+                let data = {
+                    client_id: this.custosId,
+                    client_sec: this.custosSec,
+                    body: {
+                        child_id: this.addingGr.id,
+                        parent_id: this.selectedId
+                    }
+                }
+
+                let status = await this.$store.dispatch('group/addChildGroup', data)
+                if (status) {
+                    this.childGroupMembers.push(this.addingGr)
+                }
+                this.operationCompleted = true
+            },
+
+            async addMemberShip() {
+                let users = []
+                this.usernames.forEach(user => {
+                    if (user != this.selectedOwnerId) {
+                        let add = true
+                        this.members.forEach(mem => {
+                            if (mem.user_id === user) {
+                                add = false
+                            }
+                        })
+                        if (add) {
+                            users.push(user)
+                        }
+                    }
+                })
+                this.usernames = users
+                this.$refs.addMembershipModel.show()
+            },
+
+            addGr: function () {
+                this.$refs.addGrModel.show()
+            },
+
+            async addGroupOKPressed() {
+                this.groupsLoading = true
+                let username = await this.$store.dispatch('identity/getCurrentUserName')
+                let data = {
+                    client_id: this.custosId,
+                    client_sec: this.custosSec,
+                    body: {
+                        groups: [{
+                            name: this.selectedNewGrName,
+                            description: this.selectedNewGrDesc,
+                            ownerId: username,
+                            realm_roles: [],
+                            client_roles: [],
+                            attributes: [],
+                            sub_groups: []
+                        }]
+                    }
+                }
+                await this.$store.dispatch('group/createGroup', data)
+                this.groupItems = await this.$store.dispatch('group/loadAllGroups', data)
+                this.groupsLoading = false
+                this.this.selectedNewGrName = null
+                this.selectedNewGrDesc = null
+            },
+
+            async loadUsers() {
+                let data = {offset: 0, limit: 50, client_id: this.custosId, client_sec: this.custosSec}
+                await this.$store.dispatch('user/users', data)
+                let resp = await this.$store.getters['user/getUsers']
+                this.usernames = []
+                if (Array.isArray(resp) && resp.length > 0) {
+                    resp.forEach(obj => {
+                        this.usernames.push(obj.username)
+                    });
+
+                }
+            },
+
+            async addMembershipOKPressed() {
+                this.operationCompleted = false
+                let data = {
+                    client_id: this.custosId, client_sec: this.custosSec, body: {
+                        group_id: this.selectedId,
+                        username: this.selectedNewUsername,
+                        membership_type: this.selectedNewMemType
+                    }
+                }
+                let response = await this.$store.dispatch('group/addUserToGroup', data)
+
+                if (response.status) {
+                    let obj = {user_id: this.selectedNewUsername, type: this.selectedNewMemType}
+                    this.members.push(obj)
+                }
+                this.operationCompleted = true
+            },
+
+            async updateMembership() {
+                this.operationCompleted = false
+                let data = {
+                    client_id: this.custosId, client_sec: this.custosSec, body: {
+                        group_id: this.selectedId,
+                        username: this.selectedMembershipUsername,
+                        type: this.selectedMembershipType
+                    }
+                }
+                let response = await this.$store.dispatch('group/changeGroupMembership', data)
+
+                if (response.status) {
+                    let obj = {user_id: this.selectedMembershipUsername, type: this.selectedMembershipType}
+                    let newMems = []
+                    this.members.forEach(mem => {
+                        if (mem.user_id !== this.selectedMembershipUsername) {
+                            newMems.push(mem)
+                        }
+                    })
+
+                    newMems.push(obj)
+                    this.members = newMems
+                }
+                this.operationCompleted = true
+                this.$refs.membershipModel.hide()
+            },
+
+            async removeMembership() {
+                this.operationCompleted = false
+                let data = {
+                    client_id: this.custosId, client_sec: this.custosSec, body: {
+                        group_id: this.selectedId,
+                        username: this.selectedMembershipUsername
+                    }
+                }
+                let response = await this.$store.dispatch('group/removeUserFromGroup', data)
+
+                if (response.status) {
+                    let newMems = []
+                    this.members.forEach(mem => {
+                        if (mem.user_id !== this.selectedMembershipUsername) {
+                            newMems.push(mem)
+                        }
+                    })
+                    this.members = newMems
+                }
+                this.operationCompleted = true
+                this.$refs.membershipModel.hide()
+            },
+
+
+            async updateGroupProfile() {
+                this.operationCompleted = false
+                let data = {
+                    client_id: this.custosId, client_sec: this.custosSec, group_id: this.selectedId, body: {
+                        group: {
+                            name: this.selectedName,
+                            ownerId: this.selectedOwnerId,
+                            description: this.selectedDescription,
+                            realm_roles: [],
+                            client_roles: [],
+                            attributes: [],
+                            sub_groups: []
+                        }
+                    }
+                }
+                await this.$store.dispatch('group/updateGroup', data)
+                this.$refs.groupmodel.hide()
+                let dat = {
+                    client_id: this.custosId,
+                    client_sec: this.custosSec
+                }
+                this.groupItems = await this.$store.dispatch('group/loadAllGroups', dat)
+                this.operationCompleted = true
+            },
+
+
+            async removeGroupProfile() {
+                this.operationCompleted = false
+                let data = {
+                    client_id: this.custosId, client_sec: this.custosSec, group_id: this.selectedId, body: {
+                        group: {
+                            name: this.selectedName,
+                            ownerId: this.selectedOwnerId,
+                            description: this.selectedDescription,
+                            realm_roles: [],
+                            client_roles: [],
+                            attributes: [],
+                            sub_groups: []
+                        }
+                    }
+                }
+                await this.$store.dispatch('group/deleteGroup', data)
+                this.$refs.groupmodel.hide()
+                let dat = {
+                    client_id: this.custosId,
+                    client_sec: this.custosSec
+                }
+                this.groupItems = await this.$store.dispatch('group/loadAllGroups', dat)
+                this.operationCompleted = true
+            },
+
+
+            async closeGroupProfile() {
+                this.$refs.groupmodel.hide()
+            },
+
+            async closeMembershipModel() {
+                this.$refs.membershipModel.hide()
+            },
+
+            async removeGroupMembership() {
+                this.operationCompleted = false
+                let data = {
+                    client_id: this.custosId,
+                    client_sec: this.custosSec,
+                    body: {
+                        child_id: this.selectedGrId,
+                        parent_id: this.selectedId
+                    }
+                }
+
+                let status = await this.$store.dispatch('group/removeChildGroup', data)
+                if (status) {
+                    let data = {
+                        client_id: this.custosId,
+                        client_sec: this.custosSec,
+                        group_id: this.selectedId
+                    }
+
+                    let response = await this.$store.dispatch('group/getAllChildGroups', data)
+                    this.childGroupMembers = response.groups
+                }
+                this.operationCompleted = true
+            }
+
+        },
+        async mounted() {
+            this.groupsLoading = true
+            this.custosId = config.value('clientId')
+            this.custosSec = config.value('clientSec')
+            this.isAdminUser = await this.$store.dispatch('identity/isLoggedUserHasAdminAccess')
+            let data = {
+                client_id: this.custosId,
+                client_sec: this.custosSec
+            }
+            this.groupItems = await this.$store.dispatch('group/loadAllGroups', data)
+            this.groupsLoading = false
+            this.loadUsers()
+        }
+    }
+</script>
+
+<style scoped>
+    .grouptable {
+        width: 50%;
+        margin-left: 10%;
+        margin-top: 3%;
+    }
+
+    .groupformItem {
+        margin-top: 3%;
+    }
+
+    .grBtnDel {
+        margin-left: 30%;
+        position: relative;
+        background-color: red;
+    }
+
+    .grBtnCl {
+        margin-left: 10%;
+        position: relative;
+        background-color: darkgrey;
+    }
+
+    .grBtnUp {
+        margin-left: 10%;
+        position: relative;
+    }
+
+    .addmemberbtn {
+        position: relative;
+        margin-left: 50%;
+    }
+
+    .removememberbtn {
+        position: relative;
+        margin-left: 20%;
+
+    }
+
+    .memBtnDel {
+        margin-left: 10%;
+        position: relative;
+        background-color: red;
+    }
+
+    .memBtnCl {
+        margin-left: 10%;
+        position: relative;
+        background-color: darkgrey;
+    }
+
+    .memBtnUp {
+        margin-left: 10%;
+        position: relative;
+    }
+
+    .addGr {
+        margin-right: 0;
+        position: relative;
+    }
+</style>
\ No newline at end of file
diff --git a/custos-demo-gateway/src/components/workspace/Logs.vue b/custos-demo-gateway/src/components/workspace/Logs.vue
new file mode 100644
index 0000000..fb1fae8
--- /dev/null
+++ b/custos-demo-gateway/src/components/workspace/Logs.vue
@@ -0,0 +1,223 @@
+<template>
+    <div>
+        <div v-if="activateLogEnabling">
+            <b-form-checkbox v-model="checked" :disabled=isCheckedBtnDisabled v-on:change="enableLogging"
+                             name="check-button" switch>
+                Enable Logging
+            </b-form-checkbox>
+        </div>
+        <div v-if="isLoggingEnabled">
+            <div class="logSearchBar">
+                <b-input-group>
+                    <template v-slot:prepend>
+                        <b-form-input disabled v-model="defaultSearchText">Search By</b-form-input>
+                        <b-dropdown :text="selectedService" variant="info" v-model="selectedService">
+                            <b-dropdown-item v-for="option in options"
+                                             :key="option.value"
+                                             :value="option.value"
+                                             @click=searchLogsWithFilter(option)>
+                                {{option.text}}
+                            </b-dropdown-item>
+
+                        </b-dropdown>
+
+                    </template>
+                </b-input-group>
+            </div>
+
+            <div class="grouptable">
+                <b-table striped hover responsive :items="logItems" :fields="fields" selectable
+                         ref="selectableTable"
+                         select-mode="single"
+                         :per-page="perPage"
+                         :current-page="currentPage"
+                         caption-top>
+                    <template v-slot:table-caption>Access Logs</template>
+                </b-table>
+                <div class="pgClass">
+                    <b-pagination
+                            v-model="currentPage"
+                            :total-rows="rows"
+                            :per-page="perPage"
+                            aria-controls="my-table"
+
+                    ></b-pagination>
+                </div>
+            </div>
+        </div>
+
+    </div>
+</template>
+
+<script>
+    import config from "@/config";
+
+    export default {
+        name: "Logs",
+        data: function () {
+            return {
+                defaultSearchText: 'Search By',
+                fields: ['service_name', 'event_type', 'accessed_time', 'external_ip'],
+                logItems: [],
+                options: [
+                    {
+                        value: 0,
+                        text: "UserManagementService",
+                        service_name: "org.apache.custos.user.management.service.UserManagementService"
+                    },
+                    {
+                        value: 1,
+                        text: "TenantManagementService",
+                        service_name: "org.apache.custos.tenant.management.service.TenantManagementService"
+                    },
+                    {
+                        value: 2,
+                        text: "SharingManagementService",
+                        service_name: "org.apache.custos.sharing.management.service.SharingManagementService"
+                    },
+                    {
+                        value: 3,
+                        text: "ResourceSecretManagementService",
+                        service_name: "org.apache.custos.resource.secret.management.service.ResourceSecretManagementService"
+                    },
+                    {
+                        value: 4,
+                        text: "IdentityManagementService",
+                        service_name: "org.apache.custos.identity.management.service.IdentityManagementService"
+                    },
+                    {
+                        value: 5,
+                        text: "GroupManagementService",
+                        service_name: "org.apache.custos.group.management.service.GroupManagementService"
+                    },
+                    {
+                        value: 6,
+                        text: "AgentManagementService",
+                        service_name: "org.apache.custos.agent.management.service.AgentManagementService"
+                    },
+                    {
+                        value: 7,
+                        text: "All"
+                    }
+                ],
+                perPage: 10,
+                currentPage: 0,
+                selectedService: null,
+                custosSec: null,
+                custosId: null,
+                isLoggingEnabled: false,
+                activateLogEnabling: false,
+                checked: false,
+                isCheckedBtnDisabled: false
+            }
+        },
+
+        methods: {
+            async searchLogsWithFilter(data) {
+                this.selectedService = data.text
+                let dat = {}
+                if (data.text === "All") {
+                    dat = {
+                        client_id: this.custosId,
+                        client_sec: this.custosSec,
+                        params: {
+                            offset: 0,
+                            limit: 200,
+                        }
+                    }
+                } else {
+                    dat = {
+                        client_id: this.custosId,
+                        client_sec: this.custosSec,
+                        params: {
+                            offset: 0,
+                            limit: 200,
+                            service_name: data.service_name
+                        }
+                    }
+                }
+                let response = await this.$store.dispatch('log/getLogEvents', dat)
+
+                response.forEach(res => {
+                    let date = new Date(res.created_time * 1000)
+                    res.accessed_time = date.toString()
+                })
+                this.logItems = response
+
+
+            },
+
+
+            async enableLogging() {
+                let accessToken = await this.$store.getters['identity/getAccessToken']
+                let data = {
+                    user_token: accessToken
+                }
+
+                let response = await this.$store.dispatch('log/enableLogging', data)
+                if (response) {
+                    this.checked = true
+                    this.isCheckedBtnDisabled = true
+                    this.isLoggingEnabled = true
+                }
+            }
+        },
+        computed: {
+            rows() {
+                return this.logItems.length
+            }
+        },
+
+        async mounted() {
+            this.custosId = config.value('clientId')
+            this.custosSec = config.value('clientSec')
+            this.isAdminUser  = await this.$store.dispatch('identity/isLoggedUserHasAdminAccess')
+            let data = {
+                client_id: this.custosId,
+                client_sec: this.custosSec
+            }
+
+            this.isLoggingEnabled = await this.$store.dispatch('log/isLoggingEnabled', data)
+            this.activateLogEnabling = this.isAdminUser && !this.isLoggingEnabled
+            this.isCheckedBtnDisabled = this.isLoggingEnabled
+            this.checked = this.isLoggingEnabled
+
+            if (this.isLoggingEnabled) {
+                let data = {
+                    client_id: this.custosId,
+                    client_sec: this.custosSec,
+                    params: {
+                        offset: 0,
+                        limit: 200
+                    }
+                }
+                let response = await this.$store.dispatch('log/getLogEvents', data)
+
+                response.forEach(res => {
+                    let date = new Date(res.created_time * 1000)
+                    res.accessed_time = date.toString()
+                })
+                this.logItems = response
+
+            }
+
+        }
+    }
+</script>
+
+<style scoped>
+    .logSearchBar {
+        margin-left: 35%;
+    }
+
+    .grouptable {
+        width: 80%;
+        margin-left: 10%;
+        margin-top: 3%;
+    }
+
+    .pgClass {
+        margin-left: 70%;
+    }
+
+</style>
\ No newline at end of file
diff --git a/custos-demo-gateway/src/components/workspace/Secrets.vue b/custos-demo-gateway/src/components/workspace/Secrets.vue
new file mode 100644
index 0000000..f218663
--- /dev/null
+++ b/custos-demo-gateway/src/components/workspace/Secrets.vue
@@ -0,0 +1,355 @@
+<template>
+    <div>
+        <div class="secrettable">
+            <div v-if="this.secretsLoading" class="d-flex justify-content-center mb-3">
+                <b-spinner variant="primary" label="Text Centered"></b-spinner>
+            </div>
+            <b-table striped hover responsive :items="secItems" :fields="fields" selectable
+                     select-mode="single"
+                     @row-selected="onRowSelected" caption-top>
+                <template v-slot:table-caption>Secrets</template>
+            </b-table>
+            <div class="addSec">
+                <b-button variant="outline-primary" v-on:click="addSec">Add Secret</b-button>
+            </div>
+        </div>
+        <div>
+            <b-modal ref="addSecmodel" scrollable title="Add Secret" ok-title="Add" @ok="addSecOKButtonPressed">
+                <div class="secform">
+                    <div class="secFormItem">
+                        <p>Type</p>
+                        <b-form-select v-model="defaultMemType">
+                            <option v-for="(selectOption, indexOpt) in secretTypes"
+                                    :key="indexOpt"
+                                    :value="selectOption"
+                            >
+                                {{ selectOption }}
+                            </option>
+                        </b-form-select>
+                    </div>
+                    <div class="secformItem">
+                        <p>Description</p>
+                        <b-form-input v-model="selectedDescription"></b-form-input>
+                    </div>
+                    <div v-if="defaultMemType==='Password'" class="secformItem">
+                        <p>Password</p>
+                        <b-form-input v-model="selectedPassword"></b-form-input>
+                    </div>
+                </div>
+
+            </b-modal>
+        </div>
+        <div>
+            <b-modal ref="updateSecmodel" scrollable title="View Secret" cancel-title="Delete"
+                     @cancel="secDeleteButtonPressed" cancel-variant="outline-primary">
+                <div class="secform">
+                    <div class="secformItem">
+                        <p>Token</p>
+                        <b-form-input v-model="selectedExId" disabled></b-form-input>
+                    </div>
+                    <div class="secFormItem">
+                        <p>Type</p>
+                        <b-form-input v-model="selectedExType" disabled></b-form-input>
+                    </div>
+                    <div class="secformItem">
+                        <p>Description</p>
+                        <b-form-input v-model="selectedExDescription" disabled></b-form-input>
+                    </div>
+                    <div class="secformItem">
+                        <p>Owner Id</p>
+                        <b-form-input v-model="selectedExOwnerId" disabled></b-form-input>
+                    </div>
+                    <div v-if="selectedExType ==='PASSWORD'" class="secformItem">
+                        <p>Password</p>
+                        <b-form-input v-model="selectedExPassword" disabled></b-form-input>
+                    </div>
+                    <div v-if="selectedExType ==='SSH'" class="secformItem">
+                        <p>Publick Key</p>
+                        <b-textarea v-model="selectedExPubKey" disabled></b-textarea>
+                    </div>
+                    <div v-if="selectedExType ==='SSH'" class="secformItem">
+                        <p>Private Key</p>
+                        <b-textarea v-model="selectedExPrivKey" disabled></b-textarea>
+                    </div>
+
+                </div>
+
+            </b-modal>
+        </div>
+    </div>
+</template>
+
+<script>
+    import config from "@/config";
+
+    export default {
+        name: "Secrets",
+        data: function () {
+            return {
+                fields: ['token', 'description', 'type', 'owner_id'],
+                secItems: [],
+                selectedSecret: null,
+                secretTypes: ['SSH', 'Password'],
+                defaultMemType: 'SSH',
+                selectedDescription: null,
+                selectedPassword: null,
+                selectedExType: null,
+                selectedExId: null,
+                selectedExDescription: null,
+                selectedExPassword: null,
+                selectedExPubKey: null,
+                selectedExPrivKey: null,
+                selectedExOwnerId: null,
+                currentUserName: null,
+                secretsLoading: false
+
+            }
+        },
+        methods: {
+            async onRowSelected(items) {
+                if (items != null && items.length > 0) {
+                    this.selectedSecret = items
+
+                    this.selectedExDescription = this.selectedSecret[0].description
+                    this.selectedExType = this.selectedSecret[0].type
+                    this.selectedExId = this.selectedSecret[0].token
+                    this.selectedExOwnerId = this.selectedSecret[0].owner_id
+
+                    if (this.selectedExType === "SSH") {
+                        let data = {
+                            client_id: this.custosId,
+                            client_sec: this.custosSec,
+                            token: this.selectedExId
+                        }
+                        let response = await this.$store.dispatch('secret/getSSHCredential', data)
+                        this.selectedExPubKey = response.public_key
+                        this.selectedExPrivKey = response.private_key
+                    } else {
+                        let data = {
+                            client_id: this.custosId,
+                            client_sec: this.custosSec,
+                            token: this.selectedExId
+                        }
+                        let response = await this.$store.dispatch('secret/getPasswordCredential', data)
+                        this.selectedExPassword = response.password
+                    }
+                    this.$refs.updateSecmodel.show()
+                }
+            },
+
+            addSec: function () {
+                this.selectedDescription = null
+                this.selectedPassword = null
+                this.defaultMemType = 'SSH'
+                this.$refs.addSecmodel.show()
+            },
+
+            async addSecOKButtonPressed() {
+                this.secretsLoading = true
+                if (this.defaultMemType === 'SSH') {
+                    let data = {
+                        client_id: this.custosId,
+                        client_sec: this.custosSec,
+                        body: {
+                            metadata: {
+                                client_id: this.custosId,
+                                description: this.selectedDescription,
+                                owner_id: this.currentUserName
+                            }
+                        }
+                    }
+                    let response = await this.$store.dispatch('secret/addSSHCredential', data)
+                    let dataEn = {
+                        client_id: this.custosId,
+                        client_sec: this.custosSec,
+                        body: {
+                            client_id: this.custosId,
+                            entity: {
+                                id: response.token,
+                                name: 'SSH token',
+                                description: 'SSH credential created for ' + this.selectedDescription,
+                                type: 'SECRET',
+                                owner_id: this.currentUserName
+                            }
+                        }
+                    }
+
+                    await this.$store.dispatch('sharing/createEntity', dataEn)
+
+                } else {
+                    let data = {
+                        client_id: this.custosId,
+                        client_sec: this.custosSec,
+                        body: {
+                            metadata: {
+                                client_id: this.custosId,
+                                description: this.selectedDescription,
+                                owner_id: this.currentUserName
+                            },
+                            password: this.selectedPassword
+                        }
+                    }
+                    let response = await this.$store.dispatch('secret/addPasswordCredential', data)
+                    let dataEN = {
+                        client_id: this.custosId,
+                        client_sec: this.custosSec,
+                        body: {
+                            client_id: this.custosId,
+                            entity: {
+                                id: response.token,
+                                name: 'Password token',
+                                description: 'Password credential created for ' + this.selectedDescription,
+                                type: 'SECRET',
+                                owner_id: this.currentUserName
+                            }
+                        }
+                    }
+                    await this.$store.dispatch('sharing/createEntity', dataEN)
+                }
+                this.secItems = await this.getAllCredentials()
+                this.secretsLoading = false
+            },
+
+            async secDeleteButtonPressed() {
+                this.secretsLoading = true
+                if (this.defaultMemType === 'SSH') {
+                    let data = {
+                        client_id: this.custosId,
+                        client_sec: this.custosSec,
+                        token: this.selectedExId
+                    }
+                    let response = await this.$store.dispatch('secret/deleteSSHCredential', data)
+                    if (response) {
+                        let data = {
+                            client_id: this.custosId,
+                            client_sec: this.custosSec,
+                            body: {
+                                client_id: this.custosId,
+                                entity: {
+                                    id: this.selectedExId,
+                                    owner_id: this.currentUserName
+                                }
+                            }
+                        }
+                        await this.$store.dispatch('sharing/deleteEntity', data)
+                    }
+
+                } else {
+                    let data = {
+                        client_id: this.custosId,
+                        client_sec: this.custosSec,
+                        token: this.selectedExId
+                    }
+                    let response = await this.$store.dispatch('secret/deletePassswordCredential', data)
+                    if (response) {
+                        let data = {
+                            client_id: this.custosId,
+                            client_sec: this.custosSec,
+                            body: {
+                                client_id: this.custosId,
+                                entity: {
+                                    id: this.selectedExId,
+                                    owner_id: this.currentUserName
+                                }
+                            }
+                        }
+                        await this.$store.dispatch('sharing/deleteEntity', data)
+                    }
+
+                }
+                this.secItems = await this.getAllCredentials()
+                this.secretsLoading = false
+                this.secretsLoading = false
+            },
+
+
+            async getAllCredentials() {
+                let searchEntitiesData = {
+                    client_id: this.custosId,
+                    client_sec: this.custosSec,
+                    body: {
+                        client_id: this.custosId,
+                        owner_id: this.currentUserName,
+                        search_criteria: [
+                            {
+                                search_field: "ENTITY_TYPE_ID",
+                                value: 'SECRET',
+                                condition: "EQUAL"
+                            }
+                        ]
+                    }
+                }
+
+                let response = await this.$store.dispatch('sharing/getEntities', searchEntitiesData)
+                let accessible_token = []
+                if (response != null && response.length > 0) {
+                    response.forEach(a => {
+                        accessible_token.push(a.id)
+                    })
+
+                }
+                let data = {
+                    client_id: this.custosId,
+                    client_sec: this.custosSec,
+                    accessible_tokens: accessible_token
+                }
+                return await this.$store.dispatch('secret/getAllCredentials', data)
+            }
+        },
+
+
+        async mounted() {
+            this.secretsLoading = true
+            this.custosId = config.value('clientId')
+            this.custosSec = config.value('clientSec')
+            this.currentUserName = await this.$store.dispatch('identity/getCurrentUserName')
+
+            this.secItems = await this.getAllCredentials()
+            this.secretsLoading = false
+
+            let permTypesData = {
+                client_id: this.custosId,
+                client_sec: this.custosSec
+            }
+
+            let entityTypes = await this.$store.dispatch('sharing/getEntityTypes', permTypesData)
+
+            let created = false
+
+            entityTypes.forEach(ent => {
+                if (ent.id === 'SECRET') {
+                    created = true
+                }
+            })
+
+            if (!created) {
+                let dat = {
+                    client_id: this.custosId,
+                    client_sec: this.custosSec,
+                    body: {
+                        client_id: this.custosId,
+                        entity_type: {
+                            id: 'SECRET',
+                            name: 'SECRET',
+                            description: 'This is secret entity type of demo gateway'
+                        }
+                    }
+                }
+                this.entityTypes = await this.$store.dispatch('sharing/createEntityType', dat)
+            }
+        }
+    }
+</script>
+
+<style scoped>
+    .secrettable {
+        width: 50%;
+        margin-left: 10%;
+        margin-top: 3%;
+    }
+
+    .secformItem {
+        margin-top: 3%;
+    }
+
+</style>
\ No newline at end of file
diff --git a/custos-demo-gateway/src/components/workspace/Sharing.vue b/custos-demo-gateway/src/components/workspace/Sharing.vue
new file mode 100644
index 0000000..4c383cf
--- /dev/null
+++ b/custos-demo-gateway/src/components/workspace/Sharing.vue
@@ -0,0 +1,892 @@
+<template>
+    <div>
+        <div class="row">
+            <div class="column">
+                <div class="sharingtable">
+                    <div v-if="this.permissionTypesLoading" class="d-flex justify-content-center mb-3">
+                        <b-spinner variant="primary" label="Text Centered"></b-spinner>
+                    </div>
+                    <b-table striped hover responsive :items="permissionTypes" :fields="fields" selectable
+                             ref="selectableTable"
+                             select-mode="single"
+                             @row-selected="onPrTySelected" caption-top>
+                        <template v-slot:table-caption>Permissions</template>
+                    </b-table>
+                    <div class="addGr">
+                        <b-button variant="outline-primary" v-on:click="onNewPrTyAdd">Add Permission Type</b-button>
+                    </div>
+                </div>
+                <div v-if="this.entityTypesLoading" class="d-flex justify-content-center mb-3">
+                    <b-spinner variant="primary" label="Text Centered"></b-spinner>
+                </div>
+                <div class="sharingtable">
+                    <b-table striped hover responsive :items="entityTypes" :fields="fields" selectable
+                             ref="selectableTable"
+                             select-mode="single"
+                             @row-selected="onEnTySelected" caption-top>
+                        <template v-slot:table-caption>Entity Types</template>
+                    </b-table>
+                    <div class="addGr">
+                        <b-button variant="outline-primary" v-on:click="onNewEnTyAdd">Add Entity Type</b-button>
+                    </div>
+                </div>
+                <div v-if="this.entitiesLoading" class="d-flex justify-content-center mb-3">
+                    <b-spinner variant="primary" label="Text Centered"></b-spinner>
+                </div>
+                <div class="sharingtable">
+                    <b-table striped hover responsive :items="entities" :fields="entityFields" selectable
+                             ref="selectableTable"
+                             select-mode="single"
+                             @row-selected="onEntitySelected" caption-top>
+                        <template v-slot:table-caption>Entities</template>
+                    </b-table>
+                    <div class="addGr">
+                        <b-button variant="outline-primary" v-on:click="onNewEnAdd">Add Entity</b-button>
+                    </div>
+                </div>
+                <div v-if="this.sharingsLoading" class="d-flex justify-content-center mb-3">
+                    <b-spinner variant="primary" label="Text Centered"></b-spinner>
+                </div>
+                <div class="sharingtable">
+                    <b-table striped hover responsive :items="sharings" :fields="sharingFields" selectable
+                             ref="selectableTable"
+                             select-mode="single"
+                             @row-selected="onSharingSelected" caption-top>
+                        <template v-slot:table-caption>Sharings</template>
+                    </b-table>
+                    <div class="addGr">
+                        <b-button variant="outline-primary" v-on:click="onSharingAdd">Share Entities</b-button>
+                    </div>
+                </div>
+                <div>
+                    <b-modal ref="prtypemodel" scrollable title="Add Permission Type " ok-title="Add" @ok="addPrType">
+                        <div class="groupform">
+                            <div class="groupformItem">
+                                <p>Id</p>
+                                <b-form-input v-model="prId"></b-form-input>
+                            </div>
+                            <div class="groupformItem">
+                                <p>Name</p>
+                                <b-form-input v-model="prName"></b-form-input>
+                            </div>
+                            <div class="groupformItem">
+                                <p>Description</p>
+                                <b-form-input v-model="prDesc"></b-form-input>
+                            </div>
+
+                        </div>
+                    </b-modal>
+                </div>
+                <div>
+                    <b-modal ref="entypemodel" scrollable title="Add Entity Type " ok-title="Add" @ok="addNewEnType">
+                        <div class="groupform">
+                            <div class="groupformItem">
+                                <p>Id</p>
+                                <b-form-input v-model="enTyId"></b-form-input>
+                            </div>
+                            <div class="groupformItem">
+                                <p>Name</p>
+                                <b-form-input v-model="enTyName"></b-form-input>
+                            </div>
+                            <div class="groupformItem">
+                                <p>Description</p>
+                                <b-form-input v-model="enTyDesc"></b-form-input>
+                            </div>
+
+                        </div>
+                    </b-modal>
+                </div>
+                <div>
+                    <b-modal ref="enModel" scrollable title="Add  Entity " ok-title="Add" @ok="addNewEntity">
+                        <div class="groupform">
+                            <div class="groupformItem">
+                                <p>Id</p>
+                                <b-form-input v-model="enId"></b-form-input>
+                            </div>
+                            <div class="groupformItem">
+                                <p>Name</p>
+                                <b-form-input v-model="enName"></b-form-input>
+                            </div>
+                            <div class="groupformItem">
+                                <p>Type</p>
+                                <b-form-select v-model="selectedEntityType">
+                                    <option v-for="(selectOption, indexOpt) in entityTypes"
+                                            :key="indexOpt"
+                                            :value="selectOption"
+                                    >
+                                        {{ selectOption.id }}
+                                    </option>
+                                </b-form-select>
+                            </div>
+                            <div class="groupformItem">
+                                <p>Description</p>
+                                <b-form-input v-model="enDesc"></b-form-input>
+                            </div>
+
+                        </div>
+                    </b-modal>
+                    <b-modal ref="selectedPrTyModel" scrollable title="Permission Type " ok-title="Delete"
+                             @ok="deletePRType">
+                        <div class="groupform">
+                            <div class="groupformItem">
+                                <p>Id</p>
+                                <b-form-input v-model="selectedPrTyId" disabled></b-form-input>
+                            </div>
+                            <div class="groupformItem">
+                                <p>Name</p>
+                                <b-form-input v-model="selectedPrTyName" disabled></b-form-input>
+                            </div>
+                            <div class="groupformItem">
+                                <p>Description</p>
+                                <b-form-input v-model="selectedPrTyDesc" disabled></b-form-input>
+                            </div>
+
+                        </div>
+                    </b-modal>
+                    <b-modal ref="selectedEnTyModel" scrollable title="Entity Type " ok-title="Delete"
+                             @ok="deleteEnType">
+                        <div class="groupform">
+                            <div class="groupformItem">
+                                <p>Id</p>
+                                <b-form-input v-model="selectedEnTyId" disabled></b-form-input>
+                            </div>
+                            <div class="groupformItem">
+                                <p>Name</p>
+                                <b-form-input v-model="selectedEnTyName" disabled></b-form-input>
+                            </div>
+                            <div class="groupformItem">
+                                <p>Description</p>
+                                <b-form-input v-model="selectedEnTyDesc" disabled></b-form-input>
+                            </div>
+
+                        </div>
+                    </b-modal>
+                    <b-modal ref="selectedEnModel" scrollable title="Entity " ok-title="Delete" @ok="deleteEntity">
+                        <div class="groupform">
+                            <div class="groupformItem">
+                                <p>Id</p>
+                                <b-form-input v-model="selectedEnId" disabled></b-form-input>
+                            </div>
+                            <div class="groupformItem">
+                                <p>Name</p>
+                                <b-form-input v-model="selectedEnName" disabled></b-form-input>
+                            </div>
+                            <div class="groupformItem">
+                                <p>Type</p>
+                                <b-form-input v-model="selectedEntityType" disabled></b-form-input>
+                            </div>
+                            <div class="groupformItem">
+                                <p>Description</p>
+                                <b-form-input v-model="selectedEnDesc" disabled></b-form-input>
+                            </div>
+
+                        </div>
+                    </b-modal>
+                    <b-modal ref="selectedShraingModel" scrollable title="Sharing " ok-title="Delete"
+                             @ok="removeSharing">
+                        <div class="groupform">
+                            <div class="groupformItem">
+                                <p>Entity Id</p>
+                                <b-form-input v-model="selectedShEnId" disabled></b-form-input>
+                            </div>
+                            <div class="groupformItem">
+                                <p>Permission Type Id</p>
+                                <b-form-input v-model="selectedShPrId" disabled></b-form-input>
+                            </div>
+                            <div class="groupformItem">
+                                <p>Owner Id</p>
+                                <b-form-input v-model="selectedShOwId" disabled></b-form-input>
+                            </div>
+                            <div class="groupformItem">
+                                <p>Type</p>
+                                <b-form-input v-model="selectedShOwType" disabled></b-form-input>
+                            </div>
+
+                        </div>
+                    </b-modal>
+                </div>
+                <div>
+                    <b-modal ref="sharingModel" scrollable title="Share Entities " ok-title="Add" @ok="addNewSharing">
+                        <div class="groupform">
+                            <div class="groupformItem">
+                                <p>Entity Id</p>
+                                <b-form-select v-model="defaultEntityId">
+                                    <option v-for="(selectOption, indexOpt) in entities"
+                                            :key="indexOpt"
+                                            :value="selectOption"
+                                    >
+                                        {{ selectOption.id }}
+                                    </option>
+                                </b-form-select>
+                            </div>
+                            <div class="groupformItem">
+                                <p>Permission Type</p>
+                                <b-form-select v-model="defaultPermissionType">
+                                    <option v-for="(selectOption, indexOpt) in permissionTypes"
+                                            :key="indexOpt"
+                                            :value="selectOption"
+                                    >
+                                        {{ selectOption.id }}
+                                    </option>
+                                </b-form-select>
+                            </div>
+
+                            <div class="groupformItem">
+                                <p>Sharing Type</p>
+                                <b-form-select v-model="defaultSharingType">
+                                    <option v-for="(selectOption, indexOpt) in sharingTypeIds"
+                                            :key="indexOpt"
+                                            :value="selectOption"
+                                    >
+                                        {{ selectOption }}
+                                    </option>
+                                </b-form-select>
+                            </div>
+                            <div v-if="defaultSharingType == 'USERS'" class="groupformItem">
+                                <p>User Id</p>
+                                <b-form-select v-model="defaultOwner">
+                                    <option v-for="(selectOption, indexOpt) in users"
+                                            :key="indexOpt"
+                                            :value="selectOption"
+                                    >
+                                        {{ selectOption }}
+                                    </option>
+                                </b-form-select>
+                            </div>
+
+                            <div v-if="defaultSharingType == 'GROUPS'" class="groupformItem">
+                                <p>Group Id</p>
+                                <b-form-select v-model="defaultOwner">
+                                    <option v-for="(selectOption, indexOpt) in groups"
+                                            :key="indexOpt"
+                                            :value="selectOption"
+                                    >
+                                        {{ selectOption }}
+                                    </option>
+                                </b-form-select>
+                            </div>
+                        </div>
+                    </b-modal>
+                </div>
+            </div>
+            <div class="column">
+                <div class="permissionChecker">
+                    <div class="addGr">
+                        <b-button variant="outline-primary" v-on:click="checkPermissions">Evaluate Permissions</b-button>
+                    </div>
+                </div>
+            </div>
+            <b-modal ref="permissionChecker" scrollable title="Check Permissions" ok-title="Evaluate"
+                     @ok="evaluatePermission">
+                <div class="groupform">
+                    <div class="groupformItem">
+                        <p>Entity Id</p>
+                        <b-form-select v-model="defaultEntityId">
+                            <option v-for="(selectOption, indexOpt) in entities"
+                                    :key="indexOpt"
+                                    :value="selectOption"
+                            >
+                                {{ selectOption.id }}
+                            </option>
+                        </b-form-select>
+                    </div>
+                    <div class="groupformItem">
+                        <p>User</p>
+                        <b-form-select v-model="defaultOwner">
+                            <option v-for="(selectOption, indexOpt) in users"
+                                    :key="indexOpt"
+                                    :value="selectOption"
+                            >
+                                {{ selectOption }}
+                            </option>
+                        </b-form-select>
+                    </div>
+                    <div class="groupformItem">
+                        <p>Permission</p>
+                        <b-form-select v-model="defaultPermissionType">
+                            <option v-for="(selectOption, indexOpt) in permissionTypes"
+                                    :key="indexOpt"
+                                    :value="selectOption"
+                            >
+                                {{ selectOption.id }}
+                            </option>
+                        </b-form-select>
+                    </div>
+
+                </div>
+            </b-modal>
+            <b-modal ref="evalutionResultPopup" hide-footer>
+                <div v-if="this.evaluating">
+                    <b-button variant="primary" size="large" class="evaluater" disabled>
+                        <b-spinner small type="grow"></b-spinner>
+                        Evaluating...
+                    </b-button>
+                </div>
+                <div v-if="!this.evaluating">
+                <div v-if="evalutionResult" class="groupform">
+                    <p class="textCls"> Evalution Status: True </p>
+                    <p>User has permission</p>
+
+                </div>
+                <div v-if="!evalutionResult" class="groupform">
+                    <p class="textClsWrng"> Evalution Status: False </p>
+                    <p>User does not have permission</p>
+
+                </div>
+                </div>
+            </b-modal>
+        </div>
+    </div>
+</template>
+
+<script>
+    import config from "@/config";
+
+    export default {
+        name: "Sharing",
+        data: function () {
+            return {
+                fields: ['id', 'name', 'description'],
+                sharingFields: ['entity_id', 'permission_type_id', 'owner_id', 'type'],
+                entityFields: ['id', 'name', 'type', 'description'],
+                permissionTypes: [],
+                entityTypes: [],
+                entities: [],
+
+                sharings: [],
+                users: [],
+                groups: [],
+                sharingTypeIds: ['USERS', 'GROUPS'],
+                prId: null,
+                prName: null,
+                prDesc: null,
+                enTyId: null,
+                enTyName: null,
+                enTyDesc: null,
+                enId: null,
+                enName: null,
+                enDesc: null,
+                defaultEntityId: null,
+                defaultPermissionType: null,
+                defaultOwner: null,
+                selectedPrTyId: null,
+                selectedPrTyName: null,
+                selectedPrTyDesc: null,
+                selectedEnTyId: null,
+                selectedEnTyName: null,
+                selectedEnTyDesc: null,
+                selectedEnId: null,
+                selectedEnName: null,
+                selectedEnDesc: null,
+                selectedPrType: null,
+                selectedEnType: null,
+                selectedEn: null,
+                selectedSh: null,
+                selectedShEnId: null,
+                selectedShPrId: null,
+                selectedShOwId: null,
+                currentUserName: null,
+                selectedEntityType: null,
+                defaultSharingType: null,
+                defaultGroup: null,
+                selectedShOwType: null,
+                evalutionResult: false,
+                permissionTypesLoading: false,
+                entityTypesLoading:false,
+                entitiesLoading:false,
+                sharingsLoading:false,
+                evaluating: false,
+
+
+            }
+        },
+
+        methods: {
+            async addPrType() {
+                this.permissionTypesLoading =true
+                let data = {
+                    client_id: this.custosId,
+                    client_sec: this.custosSec,
+                    body: {
+                        client_id: this.custosId,
+                        permission_type: {
+                            id: this.prId,
+                            name: this.prName,
+                            description: this.prDesc
+
+                        }
+                    }
+                }
+
+                this.permissionTypes = await this.$store.dispatch('sharing/createPermissionType', data)
+                this.permissionTypesLoading = false
+                this.prId = null
+                this.prName = null
+                this.prDesc = null
+
+            },
+
+            async deletePRType() {
+                this.permissionTypesLoading =true
+                let data = {
+                    client_id: this.custosId,
+                    client_sec: this.custosSec,
+                    body: {
+                        client_id: this.custosId,
+                        permission_type: {
+                            id: this.selectedPrTyId,
+                            name: this.selectedPrTyName,
+                            description: this.selectedPrTyDesc
+
+                        }
+                    }
+                }
+
+                this.permissionTypes = await this.$store.dispatch('sharing/deletePermissionType', data)
+                this.permissionTypesLoading = false
+                this.prId = null
+                this.prName = null
+                this.prDesc = null
+            },
+            async addNewEnType() {
+                this.entityTypesLoading = true
+                let data = {
+                    client_id: this.custosId,
+                    client_sec: this.custosSec,
+                    body: {
+                        client_id: this.custosId,
+                        entity_type: {
+                            id: this.enTyId,
+                            name: this.enTyName,
+                            description: this.enTyDesc
+                        }
+                    }
+                }
+
+                this.entityTypes = await this.$store.dispatch('sharing/createEntityType', data)
+                this.entityTypesLoading = false
+                this.enTyId = null
+                this.enTyName = null
+                this.enTyDesc = null
+
+            },
+
+            async deleteEnType() {
+                this.entityTypesLoading = true
+                let data = {
+                    client_id: this.custosId,
+                    client_sec: this.custosSec,
+                    body: {
+                        client_id: this.custosId,
+                        entity_type: {
+                            id: this.selectedEnTyId,
+                            name: this.selectedEnTyName,
+                            description: this.selectedEnTyDesc
+
+                        }
+                    }
+                }
+
+                this.entityTypes = await this.$store.dispatch('sharing/deleteEntityType', data)
+                this.entityTypesLoading = false
+                this.prId = null
+                this.prName = null
+                this.prDesc = null
+            },
+
+
+            async addNewEntity() {
+                this.entitiesLoading = true
+                let data = {
+                    client_id: this.custosId,
+                    client_sec: this.custosSec,
+                    body: {
+                        client_id: this.custosId,
+                        entity: {
+                            id: this.enId,
+                            name: this.enName,
+                            description: this.enDesc,
+                            type: this.selectedEntityType.id,
+                            owner_id: this.currentUserName
+                        }
+                    }
+                }
+
+                this.entities = await this.$store.dispatch('sharing/createEntity', data)
+                this.entitiesLoading =  false
+                this.enId = null
+                this.enName = null
+                this.enDesc = null
+                this.selectedEntityType = null
+                this.sharings = await this.loadSharings()
+
+            },
+
+            async deleteEntity() {
+                this.entitiesLoading = true
+                let data = {
+                    client_id: this.custosId,
+                    client_sec: this.custosSec,
+                    body: {
+                        client_id: this.custosId,
+                        entity: {
+                            id: this.selectedEnId,
+                            name: this.selectedEnName,
+                            description: this.selectedEnDesc,
+                            type: this.selectedEntityType.id,
+                            owner_id: this.currentUserName
+                        }
+                    }
+                }
+
+                this.entities = await this.$store.dispatch('sharing/deleteEntity', data)
+                this.entitiesLoading = false
+                this.selectedEnId = null
+                this.selectedEnName = null
+                this.selectedEnDesc = null
+                this.selectedEntityType = null
+                this.sharings = await this.loadSharings()
+            },
+
+            async addNewSharing() {
+                this.sharingsLoading = true
+                let data = {
+                    client_id: this.custosId,
+                    client_sec: this.custosSec,
+                    body: {
+
+                        client_id: this.custosId,
+                        entity: {
+                            id: this.defaultEntityId.id
+                        },
+                        permission_type: {
+                            id: this.defaultPermissionType.id
+                        },
+                        owner_id: [this.defaultOwner],
+                        cascade: true
+
+                    }
+                }
+
+                if (this.defaultSharingType === 'USERS') {
+                    let response = await this.$store.dispatch('sharing/shareEntityWithUsers', data)
+
+                    if (response) {
+                        this.sharings = await this.loadSharings()
+                    }
+                } else {
+                    let response = await this.$store.dispatch('sharing/shareEntityWithGroups', data)
+                    if (response) {
+                        this.sharings = await this.loadSharings()
+                    }
+                }
+                this.sharingsLoading = false
+            },
+
+            async removeSharing() {
+                this.sharingsLoading = true
+                let data = {
+                    client_id: this.custosId,
+                    client_sec: this.custosSec,
+                    body: {
+
+                        client_id: this.custosId,
+                        entity: {
+                            id: this.selectedShEnId
+                        },
+                        permission_type: {
+                            id: this.selectedShPrId
+                        },
+                        owner_id: [this.selectedShOwId],
+                        cascade: true
+
+                    }
+                }
+
+                if (this.selectedShOwType === 'USER') {
+                    let response = await this.$store.dispatch('sharing/revokeEntitySharingFromUsers', data)
+                    if (response) {
+                        this.sharings = await this.loadSharings()
+                    }
+                } else {
+                    let response = await this.$store.dispatch('sharing/revokeEntitySharingFromGroups', data)
+                    if (response) {
+                        this.sharings = await this.loadSharings()
+                    }
+                }
+                this.sharingsLoading = false
+            },
+
+            onPrTySelected: function (items) {
+                if (items != null && items.length > 0) {
+                    this.selectedPrType = items
+                    this.selectedPrTyId = this.selectedPrType[0].id
+                    this.selectedPrTyName = this.selectedPrType[0].name
+                    this.selectedPrTyDesc = this.selectedPrType[0].description
+                    if (this.selectedPrTyId != "OWNER") {
+                        this.$refs.selectedPrTyModel.show()
+                    }
+                }
+            },
+
+            onEnTySelected: function (items) {
+                if (items != null && items.length > 0) {
+                    this.selectedEnType = items
+                    this.selectedEnTyId = this.selectedEnType[0].id
+                    this.selectedEnTyName = this.selectedEnType[0].name
+                    this.selectedEnTyDesc = this.selectedEnType[0].description
+                    this.$refs.selectedEnTyModel.show()
+                }
+            },
+
+            onEntitySelected: function (items) {
+                if (items != null && items.length > 0) {
+                    this.selectedEn = items
+                    this.selectedEnId = this.selectedEn[0].id
+                    this.selectedEnName = this.selectedEn[0].name
+                    this.selectedEnDesc = this.selectedEn[0].description
+                    this.selectedEntityType = this.selectedEn[0].type
+                    this.$refs.selectedEnModel.show()
+                }
+            },
+
+            onSharingSelected: function (items) {
+                if (items != null && items.length > 0) {
+                    this.selectedSh = items
+                    this.selectedShEnId = this.selectedSh[0].entity_id
+                    this.selectedShPrId = this.selectedSh[0].permission_type_id
+                    this.selectedShOwId = this.selectedSh[0].owner_id
+                    this.selectedShOwType = this.selectedSh[0].type
+                    this.$refs.selectedShraingModel.show()
+                }
+            },
+            onNewPrTyAdd: function () {
+                this.$refs.prtypemodel.show()
+            },
+            onNewEnTyAdd: function () {
+                this.$refs.entypemodel.show()
+            },
+            onNewEnAdd: function () {
+                this.$refs.enModel.show()
+            },
+            onSharingAdd: function () {
+                this.$refs.sharingModel.show()
+            },
+
+            checkPermissions: function () {
+                this.$refs.permissionChecker.show()
+            },
+
+            async loadUsers() {
+                let data = {offset: 0, limit: 50, client_id: this.custosId, client_sec: this.custosSec}
+                let response = await this.$store.dispatch('user/users', data)
+
+                let usrs = [];
+                if (Array.isArray(response) && response.length > 0) {
+                    response.forEach(obj => {
+                        usrs.push(obj.username)
+                    })
+                }
+                return usrs
+            },
+
+            async loadGroups() {
+                let data = {
+                    client_id: this.custosId,
+                    client_sec: this.custosSec
+                }
+                let response = await this.$store.dispatch('group/loadAllGroups', data)
+
+                let grs = [];
+                if (Array.isArray(response) && response.length > 0) {
+                    response.forEach(obj => {
+                        grs.push(obj.id)
+                    })
+                }
+                return grs
+            },
+
+
+            async evaluatePermission() {
+                this.evaluating = true
+                this.$refs.evalutionResultPopup.show()
+                let data = {
+                    client_id: this.custosId,
+                    client_sec: this.custosSec,
+                    params: {
+                        client_id: this.custosId,
+                        'entity.id': this.defaultEntityId.id,
+                        'permission_type.id': this.defaultPermissionType.id,
+                        owner_id: this.defaultOwner
+                    }
+                }
+                let response = await this.$store.dispatch('sharing/userHasAccess', data)
+                this.evalutionResult = response
+                this.evaluating = false
+
+            },
+
+            async loadSharings() {
+                let shars = []
+                for (const en of this.entities) {
+                    for (const pr of this.permissionTypes) {
+                        let pars = {
+                            client_id: this.custosId,
+                            client_sec: this.custosSec,
+                            params: {
+                                'entity.id': en.id,
+                                'permission_type.id': pr.id
+                            }
+                        }
+                        let userIds = await this.$store.dispatch('sharing/getListOfSharedUsers', pars)
+
+                        if (userIds != null && userIds.length > 0) {
+                            userIds.forEach(uId => {
+                                let shItem = {
+                                    entity_id: en.id,
+                                    permission_type_id: pr.id,
+                                    owner_id: uId,
+                                    type: 'USER'
+                                }
+                                shars.push(shItem)
+                            })
+                        }
+
+                        let groupIds = await this.$store.dispatch('sharing/getListOfSharedGroups', pars)
+
+                        if (groupIds != null && groupIds.length > 0) {
+                            groupIds.forEach(uId => {
+                                let shItem = {
+                                    entity_id: en.id,
+                                    permission_type_id: pr.id,
+                                    owner_id: uId,
+                                    type: 'GROUP'
+                                }
+                                shars.push(shItem)
+                            })
+                        }
+                    }
+                }
+                return shars
+            }
+        },
+
+        async mounted() {
+            this.custosId = config.value('clientId')
+            this.custosSec = config.value('clientSec')
+            this.isAdminUser = await this.$store.dispatch('identity/isLoggedUserHasAdminAccess')
+            this.currentUserName = await this.$store.dispatch('identity/getCurrentUserName')
+
+
+            let permTypesData = {
+                client_id: this.custosId,
+                client_sec: this.custosSec
+            }
+
+            this.permissionTypes = await this.$store.dispatch('sharing/getPermissionTypes', permTypesData)
+
+            if (this.permissionTypes.length == 0) {
+                let data = {
+                    client_id: this.custosId,
+                    client_sec: this.custosSec,
+                    body: {
+                        client_id: this.custosId,
+                        permission_type: {
+                            id: "OWNER",
+                            name: "Owner",
+                            description: "This is owner permission type"
+                        }
+                    }
+                }
+
+                this.permissionTypes = await this.$store.dispatch('sharing/createPermissionType', data)
+            }
+
+
+            this.entityTypes = await this.$store.dispatch('sharing/getEntityTypes', permTypesData)
+
+            let searchEntitiesData = {
+                client_id: this.custosId,
+                client_sec: this.custosSec,
+                body: {
+                    client_id: this.custosId,
+                    owner_id: this.currentUserName,
+                    search_criteria: [
+                        {
+                            search_field: "OWNER_ID",
+                            value: this.currentUserName,
+                            condition: "EQUAL"
+                        }
+                    ]
+                }
+            }
+
+            this.entities = await this.$store.dispatch('sharing/getEntities', searchEntitiesData)
+
+            this.sharings = await this.loadSharings()
+
+            this.users = await this.loadUsers()
+            this.groups = await this.loadGroups()
+
+        }
+    }
+
+
+</script>
+
+<style scoped>
+    .sharingtable {
+        margin-left: 20%;
+        width: 100%;
+    }
+
+    .addGr {
+        margin-left: 60%;
+    }
+
+    .column {
+        float: left;
+        width: 50%;
+        padding: 10px;
+        height: 300px; /* Should be removed. Only for demonstration */
+    }
+
+    /* Clear floats after the columns */
+    .row:after {
+        content: "";
+        display: table;
+        clear: both;
+    }
+
+    .permissionChecker {
+        width: 60%;
+        position: relative;
+        margin-left: 10%;
+        margin-top: 10%;
+    }
+
+    .groupformItem {
+        margin-top: 3%;
+    }
+
+    .textCls {
+        color: blue;
+        font-family: Arial;
+        font-size: x-large;
+        margin-left: 10%;
+    }
+
+    .textClsWrng {
+        color: red;
+        font-family: Arial;
+        font-size: x-large;
+        margin-left: 10%;
+    }
+
+    .evaluater {
+        margin-left: 40%;
+        color: #42b983;
+    }
+
+
+</style>
\ No newline at end of file
diff --git a/custos-demo-gateway/src/components/workspace/Users.vue b/custos-demo-gateway/src/components/workspace/Users.vue
new file mode 100644
index 0000000..1f6c63d
--- /dev/null
+++ b/custos-demo-gateway/src/components/workspace/Users.vue
@@ -0,0 +1,609 @@
+<template>
+    <div>
+        <div class="userSearchBar">
+            <b-input-group prepend="Username" class="mt-3">
+                <b-form-input v-model="searchUsername"></b-form-input>
+                <b-input-group-append>
+                    <b-button variant="outline-success" v-on:click="this.searchResult">
+                        <div v-if="!this.searchUsers">
+                            <b-spinner small></b-spinner>
+                        </div>
+                        Search
+                    </b-button>
+                </b-input-group-append>
+            </b-input-group>
+        </div>
+        <div v-if="this.userloading" class="d-flex justify-content-center mb-3">
+            <b-spinner variant="primary" label="Text Centered"></b-spinner>
+        </div>
+        <div class="usertable">
+            <b-table striped hover responsive :items="items" :fields="fields" selectable
+                     ref="selectableTable"
+                     select-mode="single"
+                     :per-page="perPage"
+                     :current-page="currentPage"
+                     @row-selected="onRowSelected" caption-top>
+                <template v-slot:table-caption>Users</template>
+            </b-table>
+            <div class="pgClass">
+                <b-pagination
+                        v-model="currentPage"
+                        :total-rows="rows"
+                        :per-page="perPage"
+                        aria-controls="my-table"
+
+                ></b-pagination>
+            </div>
+        </div>
+
+        <div>
+            <b-modal ref="usermodel" scrollable title="User Profile" ok-title="Update" @ok="this.updateUserProfile">
+                <div class="userform">
+                    <div class="userformItem">
+                        <p>Username</p>
+                        <b-form-input v-model="selectedUsername" disabled></b-form-input>
+                    </div>
+                    <div class="userformItem">
+                        <p>First Name</p>
+                        <b-form-input v-model="selectedFirstName"></b-form-input>
+                    </div>
+                    <div class="userformItem">
+                        <p>Last Name</p>
+                        <b-form-input v-model="selectedLastName"></b-form-input>
+                    </div>
+                    <div class="userformItem">
+                        <p>Email</p>
+                        <b-form-input v-model="selectedEmail"></b-form-input>
+                    </div>
+                    <div class="userformItem">
+                        <p>Status</p>
+                        <b-form-select v-model="selectedStatus">
+                            <option v-for="(selectOption, indexOpt) in statusOptions"
+                                    :key="indexOpt"
+                                    :value="selectOption"
+                            >
+                                {{ selectOption }}
+                            </option>
+                        </b-form-select>
+                    </div>
+                    <div v-if="!this.operationCompleted" class="d-flex justify-content-center mb-3">
+                        <b-spinner variant="primary" label="Text Centered"></b-spinner>
+                    </div>
+                    <div class="userformItem">
+                        <p>Attributes</p>
+                        <b-table striped hover responsive :items="selectedAttributes" selectable select-mode="single"
+                                 @row-selected="onAtrSelected">
+                        </b-table>
+                        <dev class="addAtrCls">
+                            <b-button variant="outline-primary" v-on:click="addAttribute">Add Attributes</b-button>
+                        </dev>
+                    </div>
+                    <div class="userformItem">
+                        <p>Roles</p>
+                        <b-table striped hover responsive :items="selectedRoles" selectable select-mode="single"
+                                 @row-selected="onRoleSelected">
+                        </b-table>
+                        <dev class="addAtrCls">
+                            <b-button variant="outline-primary" v-on:click="addRole"
+                                      :disabled="this.isAdminUser==false">Add Role
+                            </b-button>
+                        </dev>
+                    </div>
+
+                </div>
+            </b-modal>
+            <b-modal ref="atrModel" scrollable title="Add Attribute" ok-title="Add" @ok="addAtrOkPressed">
+                <div class="userform">
+                    <div class="userformItem">
+                        <p>Key</p>
+                        <b-form-input v-model="newKey"></b-form-input>
+                    </div>
+                    <div class="userformItem">
+                        <p>Value</p>
+                        <b-form-input v-model="newValue"></b-form-input>
+                    </div>
+                </div>
+            </b-modal>
+            <b-modal ref="atrModelSelected" scrollable title="Attribute" ok-title="Delete" @ok="addAtrDeletePressed">
+                <div class="userform">
+                    <div class="userformItem">
+                        <p>Key</p>
+                        <b-form-input v-model="selectedKey" disabled></b-form-input>
+                    </div>
+                    <div class="userformItem">
+                        <p>Value</p>
+                        <b-form-input v-model="selectedValue" disabled></b-form-input>
+                    </div>
+                </div>
+            </b-modal>
+            <b-modal ref="roleModel" scrollable title="Add  Role" ok-title="Add" @ok="addRoleOkPressed">
+                <div class="userform">
+                    <div class="userformItem">
+                        <p>Scope</p>
+                        <b-form-select v-model="selectedScope">
+                            <option v-for="(selectOption, indexOpt) in scopes"
+                                    :key="indexOpt"
+                                    :value="selectOption"
+                            >
+                                {{ selectOption }}
+                            </option>
+                        </b-form-select>
+                    </div>
+                    <div class="userformItem">
+                        <p>Role</p>
+                        <div v-if="selectedScope==='TENANT'">
+                            <b-form-select v-model="selectedRole">
+                                <option v-for="(selectOption, indexOpt) in tenantroles"
+                                        :key="indexOpt"
+                                        :value="selectOption"
+                                >
+                                    {{ selectOption }}
+                                </option>
+                            </b-form-select>
+                        </div>
+                        <div v-if="selectedScope==='CLIENT'">
+                            <b-form-select v-model="selectedRole">
+                                <option v-for="(selectOption, indexOpt) in clientroles"
+                                        :key="indexOpt"
+                                        :value="selectOption"
+                                >
+                                    {{ selectOption }}
+                                </option>
+                            </b-form-select>
+                        </div>
+                    </div>
+                </div>
+            </b-modal>
+            <b-modal ref="roleModelSelected" scrollable title="Role" ok-title="Delete" @ok="deleteRoleOkPressed">
+                <div class="userform">
+                    <div class="userformItem">
+                        <p>Role</p>
+                        <b-form-input v-model="rowSelectedRole" disabled></b-form-input>
+                    </div>
+                    <div class="userformItem">
+                        <p>Scope</p>
+                        <b-form-input v-model="rowSelectedScope" disabled></b-form-input>
+                    </div>
+                </div>
+            </b-modal>
+        </div>
+    </div>
+</template>
+
+<script>
+    import config from "@/config";
+
+    export default {
+        name: "Users",
+        data: function () {
+            return {
+                fields: ['username', 'first_name', 'last_name', 'email', 'status'],
+                community_fields: ['id'],
+                items: [],
+                communityAccounts: [{
+                    id: 'asdfcvrfg',
+                    attributes: [{key: "work", vaule: 8123456789}, {key: "phone", vaule: 8123456789}],
+                    roles: [{name: 'agentAdmin', scope: 'TENANT'}]
+                },
+                    {
+                        id: 'zxsdcfvtghy',
+                        attributes: [{key: "work", vaule: 8123456789}, {key: "phone", vaule: 8123456789}],
+                        roles: [{name: 'user', scope: 'TENANT'}]
+                    }
+
+                ],
+                statusOptions: ['ACTIVE', 'DISABLE'],
+                scopes: ['TENANT', 'CLIENT'],
+                tenantroles: [],
+                clientroles: [],
+                selectedItem: null,
+                searchUsername: null,
+                selectedUsername: null,
+                selectedFirstName: null,
+                selectedLastName: null,
+                selectedEmail: null,
+                selectedStatus: null,
+                selectedAttributes: [],
+                selectedRoles: [],
+                selectedKey: null,
+                selectedValue: null,
+                newKey: null,
+                newValue: null,
+                selectedAtr: null,
+                selectedRoleItem: null,
+                selectedRole: null,
+                selectedScope: null,
+                rowSelectedRole: null,
+                rowSelectedScope: null,
+                newScope: null,
+                newRole: null,
+                checked: false,
+                newAccountName: null,
+                exAccountId: null,
+                exAccount: null,
+                accSelectedAttributes: [],
+                accSelectedRoles: [],
+                exAccountAtr: null,
+                exAccountRole: null,
+                exAccountAtrKey: null,
+                exAccountAtrValue: null,
+                exAccountRoleKey: null,
+                exAccountRoleValue: null,
+                custosId: null,
+                custosSec: null,
+                isAdminUser: false,
+                perPage: 5,
+                currentPage: 0,
+                userloading: true,
+                operationCompleted: true,
+                searchUsers: true,
+                currentUserName: null
+
+            }
+        },
+        methods: {
+            onRowSelected: function (items) {
+                if (items != null && items.length > 0) {
+                    this.selectedItem = items
+                    this.selectedUsername = this.selectedItem[0].username
+                    this.selectedFirstName = this.selectedItem[0].first_name
+                    this.selectedLastName = this.selectedItem[0].last_name
+                    this.selectedEmail = this.selectedItem[0].email
+                    this.selectedStatus = this.selectedItem[0].status
+                    this.selectedAttributes = this.selectedItem[0].attributes
+                    this.selectedRoles = this.selectedItem[0].roles
+                    if (this.selectedUsername === this.currentUserName || this.isAdminUser) {
+                        this.$refs.usermodel.show()
+                    }
+                }
+
+            },
+            clearSelected: function () {
+                this.$refs.selectableTable.clearSelected()
+            },
+            async searchResult() {
+                this.searchUsers = false
+                if (this.searchUsername == null || this.searchUsername == "") {
+                    await this.loadUsers()
+                } else {
+                    let data = {
+                        offset: 0, limit: 1, client_id: this.custosId, client_sec: this.custosSec,
+                        username: this.searchUsername
+                    }
+                    await this.$store.dispatch('user/users', data)
+                    let resp = await this.$store.getters['user/getUsers']
+                    this.items = []
+                    if (Array.isArray(resp) && resp.length > 0) {
+                        resp.forEach(obj => {
+                            let user = {
+                                username: obj.username,
+                                first_name: obj.first_name,
+                                last_name: obj.last_name,
+                                email: obj.email,
+                                status: obj.state,
+                                attributes: [],
+                                roles: []
+                            }
+                            let rr = obj.realm_roles
+                            let cr = obj.client_roles
+
+                            let attribs = obj.attributes
+
+                            attribs.forEach(r => {
+                                let newAt = {
+                                    key: r.key,
+                                    value: r.values.join(",")
+                                }
+                                user.attributes.push(newAt)
+                            })
+
+
+                            rr.forEach(r => {
+                                let rel_role = {
+                                    name: r,
+                                    scope: 'TENANT'
+                                }
+                                user.roles.push(rel_role)
+                            })
+
+                            cr.forEach(r => {
+                                let cl_role = {
+                                    name: r,
+                                    scope: 'CLIENT'
+                                }
+                                user.roles.push(cl_role)
+                            })
+
+                            this.items.push(user)
+
+                        });
+                    }
+                }
+                this.searchUsers = true
+            },
+            async addAttribute() {
+                this.$refs.atrModel.show()
+            },
+            onAtrSelected: function (items) {
+                if (items != null && items.length > 0) {
+                    this.selectedAtr = items
+                    this.selectedKey = this.selectedAtr[0].key
+                    this.selectedValue = this.selectedAtr[0].value
+                    this.$refs.atrModelSelected.show()
+                }
+            },
+            addRole: function () {
+                this.$refs.roleModel.show()
+            },
+            onRoleSelected: function (items) {
+                if (items != null && items.length > 0) {
+                    this.selectedRoleItem = items
+                    if (this.isAdminUser) {
+                        this.rowSelectedScope = this.selectedRoleItem[0].scope
+                        this.rowSelectedRole = this.selectedRoleItem[0].name
+                        this.$refs.roleModelSelected.show()
+                    }
+                }
+            },
+
+            async addAtrOkPressed() {
+                this.operationCompleted = false
+                let accessToken = await this.$store.getters['identity/getAccessToken']
+                let bd = {
+                    user_token: accessToken, body: {
+                        attributes: [{
+                            key: this.newKey,
+                            values: [this.newValue]
+                        }],
+                        users: [this.selectedUsername]
+                    }
+                }
+                let userAtr = await this.$store.dispatch('user/addUserAttributes', bd)
+                if (userAtr) {
+                    let atr = {key: this.newKey, value: this.newValue}
+                    this.selectedAttributes.push(atr)
+                }
+                this.operationCompleted = true
+                this.newKey = null
+                this.newValue = null
+                this.loadUsers()
+            },
+            async addAtrDeletePressed() {
+                this.operationCompleted = false
+                let accessToken = await this.$store.getters['identity/getAccessToken']
+                let bd = {
+                    user_token: accessToken, body: {
+                        attributes: [{
+                            key: this.selectedKey,
+                            values: [this.selectedValue]
+                        }],
+                        users: [this.selectedUsername]
+                    }
+                }
+                let userAtr = await this.$store.dispatch('user/deleteUserAttributes', bd)
+                if (userAtr) {
+                    let newAtr = []
+                    this.selectedAttributes.forEach(atr => {
+
+                        if (atr.key != this.selectedKey) {
+                            newAtr.push(atr)
+                        }
+                        this.selectedAttributes = newAtr
+                    })
+                }
+                this.operationCompleted = true
+                this.selectedKey = null
+                this.selectedValue = null
+                this.loadUsers()
+            },
+            async addRoleOkPressed() {
+                this.operationCompleted = false
+                let accessToken = await this.$store.getters['identity/getAccessToken']
+                let bd = {
+                    user_token: accessToken,
+                    body: {
+                        roles: [this.selectedRole],
+                        usernames: [this.selectedUsername],
+                        client_level: (this.selectedScope === 'CLIENT')
+                    }
+                }
+                let userAtr = await this.$store.dispatch('user/addRoleToUser', bd)
+                if (userAtr) {
+                    if (this.selectedScope === 'CLIENT') {
+                        let role = {name: this.selectedRole, scope: "CLIENT"}
+                        this.selectedRoles.push(role)
+                    } else {
+                        let role = {name: this.selectedRole, scope: "TENANT"}
+                        this.selectedRoles.push(role)
+                    }
+
+                }
+                this.operationCompleted = true
+                this.selectedRole = null
+                this.selectedScope = null
+                this.loadUsers()
+            },
+            async deleteRoleOkPressed() {
+                this.operationCompleted = false
+                let accessToken = await this.$store.getters['identity/getAccessToken']
+                let bd = {};
+                if (this.rowSelectedScope === 'TENANT') {
+                    bd = {
+                        user_token: accessToken,
+                        body: {
+                            roles: [this.rowSelectedRole],
+                            username: this.selectedUsername
+                        }
+                    }
+                } else {
+                    bd = {
+                        user_token: accessToken,
+                        body: {
+                            client_roles: [this.rowSelectedRole],
+                            username: this.selectedUsername
+                        }
+                    }
+                }
+
+                let deleted = await this.$store.dispatch('user/deleteRoleFromUser', bd)
+                if (deleted) {
+
+                    let newRoles = []
+                    this.selectedRoles.forEach(atr => {
+
+                        if (atr.name != this.rowSelectedRole) {
+                            newRoles.push(atr)
+                        }
+                        this.selectedRoles = newRoles
+                    })
+                }
+                this.operationCompleted = true
+                this.loadUsers()
+            },
+
+            async loadUsers() {
+                let data = {offset: 0, limit: 5, client_id: this.custosId, client_sec: this.custosSec}
+                let resp = await this.$store.dispatch('user/users', data)
+                this.userloading = false
+                this.items = []
+                let newUsersCount = this.parseUsers(this.items, resp)
+                if (newUsersCount == 5) {
+                    let offset = 5
+                    while (newUsersCount >= 5) {
+                        newUsersCount = await this.loadUsersR(offset, this.items)
+                        offset = offset + 5
+                    }
+                }
+            },
+
+            async loadUsersR(offset, items) {
+                let data = {offset: offset, limit: 5, client_id: this.custosId, client_sec: this.custosSec}
+                let resp = await this.$store.dispatch('user/users', data)
+                return this.parseUsers(items, resp)
+            },
+
+            parseUsers(items, resp) {
+                if (Array.isArray(resp) && resp.length > 0) {
+                    resp.forEach(obj => {
+                        let user = {
+                            username: obj.username,
+                            first_name: obj.first_name,
+                            last_name: obj.last_name,
+                            email: obj.email,
+                            status: obj.state,
+                            attributes: [],
+                            roles: []
+                        }
+                        let rr = obj.realm_roles
+                        let cr = obj.client_roles
+
+                        let attribs = obj.attributes
+
+                        attribs.forEach(r => {
+                            let newAt = {
+                                key: r.key,
+                                value: r.values.join(",")
+                            }
+                            user.attributes.push(newAt)
+                        })
+
+                        rr.forEach(r => {
+                            let rel_role = {
+                                name: r,
+                                scope: 'TENANT'
+                            }
+                            user.roles.push(rel_role)
+                        })
+
+                        cr.forEach(r => {
+                            let cl_role = {
+                                name: r,
+                                scope: 'CLIENT'
+                            }
+                            user.roles.push(cl_role)
+                        })
+
+                        items.push(user)
+
+                    });
+
+                }
+                return resp.length
+            },
+
+            async updateUserProfile() {
+                this.operationCompleted = false
+                let params = {client_id: this.custosId, client_sec: this.custosSec, username: this.selectedUsername}
+                if (this.selectedStatus === 'ACTIVE') {
+                    await this.$store.dispatch('user/enableUser', params)
+
+                } else {
+                    await this.$store.dispatch('user/disableUser', params)
+                }
+
+                let data = {
+                    client_id: this.custosId,
+                    client_sec: this.custosSec,
+                    body: {
+                        username: this.selectedUsername,
+                        first_name: this.selectedFirstName,
+                        last_name: this.selectedLastName,
+                        email: this.selectedEmail
+                    }
+                }
+                await this.$store.dispatch('user/updateUserProfile', data)
+                this.operationCompleted = true
+                this.loadUsers()
+
+            },
+        }
+        ,
+
+        async mounted() {
+            this.custosId = config.value('clientId')
+            this.custosSec = config.value('clientSec')
+            this.isAdminUser = await this.$store.dispatch('identity/isLoggedUserHasAdminAccess')
+            this.currentUserName = await this.$store.dispatch('identity/getCurrentUserName')
+            await this.loadUsers()
+
+            let params = {client_id: this.custosId, client_sec: this.custosSec}
+            this.tenantroles = await this.$store.dispatch('user/getTenantLevelRoles', params)
+            this.clientroles = await this.$store.dispatch('user/getClientLevelRoles', params)
+        }
+        ,
+        computed: {
+            rows() {
+                return this.items.length
+            }
+        }
+    }
+
+</script>
+
+<style scoped>
+    .usertable {
+        width: 50%;
+        margin-left: 10%;
+        margin-top: 3%;
+    }
+
+    .userSearchBar {
+        width: 50%;
+        margin-left: 10%;
+    }
+
+    .userformItem {
+        margin-top: 3%;
+    }
+
+    .addAtrCls {
+        margin-left: 70%;
+    }
+
+    .pgClass {
+        margin-left: 70%;
+        position: relative;
+    }
+
+
+</style>
\ No newline at end of file
diff --git a/custos-demo-gateway/src/components/workspace/Workspace.vue b/custos-demo-gateway/src/components/workspace/Workspace.vue
new file mode 100644
index 0000000..a7929e5
--- /dev/null
+++ b/custos-demo-gateway/src/components/workspace/Workspace.vue
@@ -0,0 +1,141 @@
+<template>
+    <dev>
+        <div class="logout">
+            <b-button href="#" v-on:click="logout">Logout</b-button>
+        </div>
+        <div class="grouping">
+            <b-card-group deck>
+                <b-card :img-src="require('../../assets/users.png')"
+                        img-height="50%"
+                        style="max-width: 20rem;"
+                        class="mb-2"
+                        body-class="bcbody"
+                >
+                    <b-button href="#" variant="outline-primary" v-on:click="loadRoute($event, '/workspace/users')">
+                        Manage Users
+                    </b-button>
+                </b-card>
+                <b-card :img-src="require('../../assets/groups_web.png')"
+                        img-height="50%"
+                        style="max-width: 20rem;"
+                        class="mb-2"
+                        body-class="bcbody"
+                >
+                    <b-button href="#" variant="outline-primary" v-on:click="loadRoute($event, '/workspace/groups')">
+                        Manage Groups
+                    </b-button>
+                </b-card>
+                <b-card :img-src="require('../../assets/credentials.png')"
+                        img-height="50%"
+                        style="max-width: 20rem;"
+                        class="mb-2"
+                        body-class="bcbody"
+                >
+                    <b-button href="#" variant="outline-primary" v-on:click="loadRoute($event, '/workspace/secrets')">
+                        Manage Secrets
+                    </b-button>
+                </b-card>
+            </b-card-group>
+        </div>
+        <div class="groupingbt">
+            <b-card-group deck>
+                <b-card :img-src="require('../../assets/sharings.png')"
+                        img-height="50%"
+                        style="max-width: 20rem;"
+                        class="mb-2"
+                        body-class="bcbody"
+                >
+                    <b-button href="#" variant="outline-primary" v-on:click="loadRoute($event, '/workspace/sharings')">
+                        Sharing
+                    </b-button>
+                </b-card>
+                <b-card :img-src="require('../../assets/bots.png')"
+                        img-height="50%"
+                        style="max-width: 20rem;"
+                        class="mb-2"
+                        body-class="bcbody"
+                >
+                    <b-button href="#" variant="outline-primary" :disabled="!isAdmin"
+                              v-on:click="loadRoute($event, '/workspace/agents')">
+                        Community Accounts
+                    </b-button>
+                </b-card>
+                <b-card :img-src="require('../../assets/dblogs.png')"
+                        img-height="50%"
+                        style="max-width: 20rem;"
+                        class="mb-2"
+                        body-class="bcbody"
+                >
+                    <b-button href="#" variant="outline-primary" :disabled="!isAdmin" v-on:click="loadRoute($event, '/workspace/logs')">
+                        Logs
+                    </b-button>
+                </b-card>
+            </b-card-group>
+        </div>
+    </dev>
+</template>
+
+<script>
+    import config from "@/config";
+
+    export default {
+        name: "Workspace",
+        data: function () {
+            return {
+                custosId: null,
+                custosSec: null,
+                isAdmin: false
+            }
+        },
+        methods: {
+            loadRoute: function (event, route) {
+                this.$router.push(route)
+            },
+            async logout() {
+                let data = {
+                    client_id: this.custosId,
+                    client_sec: this.custosSec
+                }
+                await this.$store.dispatch('identity/logout', data)
+                await this.$router.push("/")
+                await this.$store.dispatch('agent/reset')
+                await this.$store.dispatch('group/reset')
+                await this.$store.dispatch('secret/reset')
+                await this.$store.dispatch('sharing/reset')
+                await this.$store.dispatch('user/reset')
+            }
+        },
+
+        async mounted() {
+            this.custosId = config.value('clientId')
+            this.custosSec = config.value('clientSec')
+            this.isAdmin = await this.$store.dispatch('identity/isLoggedUserHasAdminAccess')
+        }
+    }
+</script>
+
+<style scoped>
+    .grouping {
+        position: relative;
+        margin-left: 30%;
+        width: 40%;
+        margin-top: 5%;
+    }
+
+    .groupingbt {
+        position: relative;
+        margin-left: 30%;
+        width: 40%;
+        margin-top: 5%;
+    }
+
+    .bcbody {
+        background-color: lightgray;
+        height: 30%;
+    }
+
+    .logout {
+        margin-left: 70%;
+    }
+
+</style>
\ No newline at end of file
diff --git a/custos-demo-gateway/src/config.js b/custos-demo-gateway/src/config.js
new file mode 100644
index 0000000..0ba2735
--- /dev/null
+++ b/custos-demo-gateway/src/config.js
@@ -0,0 +1,41 @@
+import dotenv from 'dotenv'
+dotenv.config()
+
+export default class Configuration {
+    static get CONFIG () {
+        return {
+            clientId: '$VUE_APP_CLIENT_ID',
+            clientSec: '$VUE_APP_CLIENT_SEC',
+            redirectURI: '$VUE_APP_REDIRECT_URI'
+        }
+    }
+
+    static value (name) {
+        if (!(name in this.CONFIG)) {
+            console.log(`Configuration: There is no key named "${name}"`)
+            return
+        }
+
+        const value = this.CONFIG[name]
+
+        if (!value) {
+            console.log(`Configuration: Value for "${name}" is not defined`)
+            return
+        }
+
+        if (value.startsWith('$VUE_APP_')) {
+            // value was not replaced, it seems we are in development.
+            // Remove $ and get current value from process.env
+            const envName = value.substr(1)
+            const envValue = process.env[envName]
+            if (envValue) {
+                return envValue
+            } else {
+                console.log(`Configuration: Environment variable "${envName}" is not defined`)
+            }
+        } else {
+            // value was already replaced, it seems we are in production.
+            return value
+        }
+    }
+}
\ No newline at end of file
diff --git a/custos-demo-gateway/src/main.js b/custos-demo-gateway/src/main.js
new file mode 100644
index 0000000..218a0ce
--- /dev/null
+++ b/custos-demo-gateway/src/main.js
@@ -0,0 +1,23 @@
+import Vue from 'vue'
+import App from './App.vue'
+import router from "./router.js";
+import store from './store/index.js'
+import { BootstrapVue, IconsPlugin } from 'bootstrap-vue'
+
+
+import 'bootstrap/dist/css/bootstrap.css';
+import 'bootstrap-vue/dist/bootstrap-vue.css';
+
+Vue.use(BootstrapVue)
+// Optionally install the BootstrapVue icon components plugin
+Vue.use(IconsPlugin)
+
+Vue.config.productionTip = false
+
+new Vue({
+  router,
+  store,
+  render: h => h(App),
+}).$mount('#app')
+
+
diff --git a/custos-demo-gateway/src/router.js b/custos-demo-gateway/src/router.js
new file mode 100644
index 0000000..f4be213
--- /dev/null
+++ b/custos-demo-gateway/src/router.js
@@ -0,0 +1,159 @@
+import Vue from "vue";
+import Router from "vue-router";
+import Landing from "./components/landing/Landing.vue";
+import store from './store/index'
+
+
+Vue.use(Router)
+
+export default new Router({
+    mode: "history",
+    routes: [
+        {
+            path: "/",
+            name: "home",
+            component: Landing
+        },
+        {
+            path: "/register",
+            name: "account",
+            component: () =>
+                import(/*webpackChunkName:"account"*/  "./components/registration/CreateAccount")
+        },
+        {
+            path: "/workspace",
+            name: "workspace",
+            beforeEnter: async (to, from, next) => {
+                let data = {
+                    client_id: process.env.VUE_APP_CLIENT_ID,
+                    client_sec: process.env.VUE_APP_CLIENT_SEC
+                }
+                if (await store.dispatch('identity/isAuthenticated',data) == true) {
+                    // You can use store variable here to access globalError or commit mutation
+                    next(true)
+                } else {
+                    next('/')
+                }
+            },
+            component: () =>
+                import(/*webpackChunkName:"workspace"*/  "./components/workspace/Workspace")
+        },
+        {
+            path: "/workspace/groups",
+            name: "groups",
+            beforeEnter: async (to, from, next) => {
+                let data = {
+                    client_id: process.env.VUE_APP_CLIENT_ID,
+                    client_sec: process.env.VUE_APP_CLIENT_SEC
+                }
+                if (await store.dispatch('identity/isAuthenticated',data) == true) {
+                    // You can use store variable here to access globalError or commit mutation
+                    next(true)
+                } else {
+                    next('/')
+                }
+            },
+            component: () =>
+                import(/*webpackChunkName:"groups"*/  "./components/workspace/Groups")
+        },
+        {
+            path: "/workspace/logs",
+            name: "logs",
+            beforeEnter: async (to, from, next) => {
+                let data = {
+                    client_id: process.env.VUE_APP_CLIENT_ID,
+                    client_sec: process.env.VUE_APP_CLIENT_SEC
+                }
+                if (await store.dispatch('identity/isAuthenticated',data) == true) {
+                    // You can use store variable here to access globalError or commit mutation
+                    next(true)
+                } else {
+                    next('/')
+                }
+            },
+            component: () =>
+                import(/*webpackChunkName:"logs"*/  "./components/workspace/Logs")
+        },
+        {
+            path: "/workspace/secrets",
+            name: "secrets",
+            beforeEnter: async (to, from, next) => {
+                let data = {
+                    client_id: process.env.VUE_APP_CLIENT_ID,
+                    client_sec: process.env.VUE_APP_CLIENT_SEC
+                }
+                if (await store.dispatch('identity/isAuthenticated',data) == true) {
+                    // You can use store variable here to access globalError or commit mutation
+                    next(true)
+                } else {
+                    next('/')
+                }
+            },
+            component: () =>
+                import(/*webpackChunkName:"secrets"*/  "./components/workspace/Secrets")
+        },
+        {
+            path: "/workspace/sharings",
+            name: "sharings",
+            beforeEnter: async (to, from, next) => {
+                let data = {
+                    client_id: process.env.VUE_APP_CLIENT_ID,
+                    client_sec: process.env.VUE_APP_CLIENT_SEC
+                }
+                if (await store.dispatch('identity/isAuthenticated',data) == true) {
+                    // You can use store variable here to access globalError or commit mutation
+                    next(true)
+                } else {
+                    next('/')
+                }
+            },
+            component: () =>
+                import(/*webpackChunkName:"sharings"*/  "./components/workspace/Sharing")
+        },
+        {
+            path: "/workspace/users",
+            name: "users",
+            beforeEnter: async (to, from, next) => {
+                let data = {
+                    client_id: process.env.VUE_APP_CLIENT_ID,
+                    client_sec: process.env.VUE_APP_CLIENT_SEC
+                }
+                if (await store.dispatch('identity/isAuthenticated',data) == true) {
+                    // You can use store variable here to access globalError or commit mutation
+                    next(true)
+                } else {
+                    next('/')
+                }
+            },
+            component: () =>
+                import(/*webpackChunkName:"users"*/  "./components/workspace/Users")
+        },
+        {
+            path: "/workspace/agents",
+            name: "agents",
+            beforeEnter: async (to, from, next) => {
+                let data = {
+                    client_id: process.env.VUE_APP_CLIENT_ID,
+                    client_sec: process.env.VUE_APP_CLIENT_SEC
+                }
+                if (await store.dispatch('identity/isAuthenticated',data) == true) {
+                    // You can use store variable here to access globalError or commit mutation
+                    next(true)
+                } else {
+                    next('/')
+                }
+            },
+            component: () =>
+                import(/*webpackChunkName:"account"*/  "./components/workspace/Agents")
+        },
+        {
+            path: "/callback",
+            name: "callback",
+            component: () =>
+                import(/*webpackChunkName:"users"*/  "./components/Callback")
+        },
+
+    ]
+})
+
+
diff --git a/custos-demo-gateway/src/service/agent_management.js b/custos-demo-gateway/src/service/agent_management.js
new file mode 100644
index 0000000..51986ca
--- /dev/null
+++ b/custos-demo-gateway/src/service/agent_management.js
@@ -0,0 +1,116 @@
+import api from "./api.js";
+import axios from "axios";
+
+
+const agentMgtEndpoint = "https://custos.scigap.org/apiserver/agent-management/v1.0.0"
+
+export default {
+
+    enableAgents(params) {
+        let authHeader = {'Authorization': 'Bearer ' + params.user_token}
+        let endpoint = agentMgtEndpoint + "/enable"
+        return api().post(endpoint, {}, {
+            headers: authHeader
+        })
+    },
+
+    registerAgent(params) {
+        let authHeader = {'Authorization': 'Bearer ' + params.user_token}
+        let endpoint = agentMgtEndpoint + "/agent"
+        return api().post(endpoint, params.body, {
+            headers: authHeader
+        })
+    },
+
+    getAgent(params) {
+        let authHeader = {'Authorization': 'Bearer ' + params.user_token}
+        let endpoint = agentMgtEndpoint + "/agent/" + params.agent_id
+        return api().get(endpoint, {
+            headers: authHeader
+        })
+    },
+
+    activateAgent(params) {
+        let authHeader = {'Authorization': 'Bearer ' + params.user_token}
+        let endpoint = agentMgtEndpoint + "/agent/activation/" + params.agent_id
+        return api().post(endpoint, {}, {
+            headers: authHeader
+        })
+    },
+
+    deactivateAgent(params) {
+        let authHeader = {'Authorization': 'Bearer ' + params.user_token}
+        let endpoint = agentMgtEndpoint + "/agent/deactivation/" + params.agent_id
+        return api().post(endpoint, {}, {
+            headers: authHeader
+        })
+    },
+
+
+    addAttributesToAgent(params) {
+        let authHeader = {'Authorization': 'Bearer ' + params.user_token}
+        let endpoint = agentMgtEndpoint + "/agent/attributes"
+        return api().post(endpoint, params.body, {
+            headers: authHeader
+        })
+
+    },
+
+    deleteAttributesFromAgent(params) {
+        let authHeader = {'Authorization': 'Bearer ' + params.user_token}
+        let endpoint = agentMgtEndpoint + "/agent/attributes"
+
+        return new Promise((resolve, reject) => {
+            axios({
+                method: 'delete',
+                url: endpoint,
+                data: params.body,
+                headers: authHeader
+            }).then((resp) => {
+                resolve(resp)
+            }).catch(errr => {
+                reject(errr)
+            })
+        })
+    },
+
+
+    getAllAgents(params) {
+        let authHeader = {'Authorization': 'Bearer ' + params.user_token}
+        let endpoint = agentMgtEndpoint + "/agents"
+        return api().get(endpoint, {
+            headers: authHeader
+        })
+
+    },
+
+    addRolesToAgent(params) {
+        let authHeader = {'Authorization': 'Bearer ' + params.user_token}
+        let endpoint = agentMgtEndpoint + "/agent/roles"
+
+        return api().post(endpoint, params.body, {
+            headers: authHeader
+        })
+    },
+
+    deleteRolesFromAgent(params) {
+        let authHeader = {'Authorization': 'Bearer ' + params.user_token}
+        let endpoint = agentMgtEndpoint + "/agent/roles"
+
+        return new Promise((resolve, reject) => {
+            axios({
+                method: 'delete',
+                url: endpoint,
+                data: params.body,
+                headers: authHeader
+            }).then((resp) => {
+                resolve(resp)
+            }).catch(errr => {
+                reject(errr)
+            })
+        })
+    },
+
+
+
+}
\ No newline at end of file
diff --git a/custos-demo-gateway/src/service/api.js b/custos-demo-gateway/src/service/api.js
new file mode 100644
index 0000000..b5c2abf
--- /dev/null
+++ b/custos-demo-gateway/src/service/api.js
@@ -0,0 +1,12 @@
+import axios from 'axios'
+
+export default() => {
+    return axios.create({
+        baseURL: 'https://custos.scigap.org/apiserver/',
+        withCredentials: false,
+        headers: {
+            'Accept': '*/*',
+            'Content-Type': 'application/json'
+        }
+    })
+}
diff --git a/custos-demo-gateway/src/service/auth.js b/custos-demo-gateway/src/service/auth.js
new file mode 100644
index 0000000..e256ab4
--- /dev/null
+++ b/custos-demo-gateway/src/service/auth.js
@@ -0,0 +1,96 @@
+import decode from 'jwt-decode';
+
+const ID_TOKEN_KEY = 'id_token';
+const ACCESS_TOKEN_KEY = 'access_token';
+const REFRESH_TOKEN_KEY = 'refresh_token';
+
+export default {
+
+    isLoggedIn() {
+        const idToken = this.getIdToken();
+        return !!idToken && !this.isTokenExpired(idToken);
+    },
+
+    isTokenExpired(token) {
+        const expirationDate = this.getTokenExpirationDate(token);
+        return expirationDate < new Date();
+    },
+
+    getTokenExpirationDate(encodedToken) {
+        const token = decode(encodedToken);
+        if (!token.exp) {
+            return null;
+        }
+
+        const date = new Date(0);
+        date.setUTCSeconds(token.exp);
+        return date;
+    },
+
+    logout() {
+        this.clearIdToken();
+        this.clearAccessToken();
+        window.location.href = "/"
+    },
+
+    setAccessToken(token) {
+        this.clearAccessToken()
+        localStorage.setItem(ACCESS_TOKEN_KEY, token)
+    },
+    setIdToken(token) {
+        this.clearIdToken();
+        localStorage.setItem(ID_TOKEN_KEY, token)
+    },
+
+
+    setRefreshToken(token) {
+        this.clearRefreshToken()
+        localStorage.setItem(REFRESH_TOKEN_KEY, token)
+    },
+
+    getIdToken() {
+        return localStorage.getItem(ID_TOKEN_KEY);
+    },
+
+    getAccessToken() {
+        return localStorage.getItem(ACCESS_TOKEN_KEY);
+    },
+
+    getRefreshToken(){
+        return localStorage.getItem(REFRESH_TOKEN_KEY)
+    },
+
+    clearIdToken() {
+        localStorage.removeItem(ID_TOKEN_KEY);
+    },
+
+    clearAccessToken() {
+        localStorage.removeItem(ACCESS_TOKEN_KEY);
+    },
+
+    clearRefreshToken() {
+        localStorage.removeItem(REFRESH_TOKEN_KEY)
+    },
+
+    isUserHasAdminAccess() {
+        let token = localStorage.getItem(ACCESS_TOKEN_KEY)
+        let decodedToken = decode(token)
+        let roles = decodedToken.realm_access.roles;
+        let condition = false
+        roles.forEach(role => {
+            if (role === 'admin') {
+
+                condition = true
+
+            }
+        })
+        return condition
+    },
+
+    getLoggedUsername() {
+        let token = localStorage.getItem(ACCESS_TOKEN_KEY)
+        let decodedToken = decode(token)
+        return decodedToken.preferred_username
+    }
+
+}
\ No newline at end of file
diff --git a/custos-demo-gateway/src/service/group_management.js b/custos-demo-gateway/src/service/group_management.js
new file mode 100644
index 0000000..44b57be
--- /dev/null
+++ b/custos-demo-gateway/src/service/group_management.js
@@ -0,0 +1,157 @@
+import api from "./api.js";
+import axios from "axios";
+
+const groupMgtEndpoint = "https://custos.scigap.org/apiserver/group-management/v1.0.0"
+
+export default {
+
+    createGroup(params) {
+        let authHeader = {'Authorization': 'Bearer ' + btoa(params.client_id + ':' + params.client_sec)}
+        let endpoint = groupMgtEndpoint + "/groups"
+        return api().post(endpoint, params.body, {
+            headers: authHeader
+        })
+    },
+
+    updateGroup(params) {
+        let authHeader = {'Authorization': 'Bearer ' + btoa(params.client_id + ':' + params.client_sec)}
+        let endpoint = groupMgtEndpoint + "/group/" + params.group_id
+        return api().put(endpoint, params.body, {
+            headers: authHeader
+        })
+    },
+
+    deleteGroup(params) {
+        let authHeader = {'Authorization': 'Bearer ' + btoa(params.client_id + ':' + params.client_sec)}
+        let endpoint = groupMgtEndpoint + "/group/" + params.group_id
+        return new Promise((resolve, reject) => {
+            axios({
+                method: 'delete',
+                url: endpoint,
+                data: params.body,
+                headers: authHeader
+            }).then((resp) => {
+                resolve(resp)
+            }).catch(errr => {
+                reject(errr)
+            })
+        })
+    },
+
+
+    findGroup(params) {
+        let authHeader = {'Authorization': 'Bearer ' + btoa(params.client_id + ':' + params.client_sec)}
+        let endpoint = groupMgtEndpoint + "/group"
+        let par = {
+            group: {
+                id: params.group_id
+            }
+        }
+        return api().get(endpoint, {
+            params: par,
+            headers: authHeader
+        })
+    },
+
+    getAllGroups(params) {
+        let authHeader = {'Authorization': 'Bearer ' + btoa(params.client_id + ':' + params.client_sec)}
+        let endpoint = groupMgtEndpoint + "/groups"
+
+        return api().get(endpoint, {
+            headers: authHeader
+        })
+    },
+
+    addUserToGroup(params) {
+        let authHeader = {'Authorization': 'Bearer ' + btoa(params.client_id + ':' + params.client_sec)}
+        let endpoint = groupMgtEndpoint + "/user/group/membership"
+
+        return api().post(endpoint, params.body, {
+            headers: authHeader
+        })
+
+    },
+
+    removeUserFromGroup(params) {
+        let authHeader = {'Authorization': 'Bearer ' + btoa(params.client_id + ':' + params.client_sec)}
+        let endpoint = groupMgtEndpoint + "/user/group/membership"
+
+        return new Promise((resolve, reject) => {
+            axios({
+                method: 'delete',
+                url: endpoint,
+                data: params.body,
+                headers: authHeader
+            }).then((resp) => {
+                resolve(resp)
+            }).catch(errr => {
+                reject(errr)
+            })
+        })
+    },
+
+    changeGroupMembership(params) {
+        let authHeader = {'Authorization': 'Bearer '  +btoa(params.client_id + ':' + params.client_sec)}
+        let endpoint = groupMgtEndpoint + "/user/group/membership"
+
+        return api().put(endpoint, params.body, {
+            headers: authHeader
+        })
+
+    },
+
+    addChildGroup(params) {
+        let authHeader = {'Authorization': 'Bearer '  +btoa(params.client_id + ':' + params.client_sec)}
+        let endpoint = groupMgtEndpoint + "/group/membership"
+
+        return api().post(endpoint, params.body, {
+            headers: authHeader
+        })
+
+    },
+
+    removeChildGroup(params) {
+        let authHeader = {'Authorization': 'Bearer '  +btoa(params.client_id + ':' + params.client_sec)}
+        let endpoint = groupMgtEndpoint + "/group/membership"
+
+        return new Promise((resolve, reject) => {
+            axios({
+                method: 'delete',
+                url: endpoint,
+                data: params.body,
+                headers: authHeader
+            }).then((resp) => {
+                resolve(resp)
+            }).catch(errr => {
+                reject(errr)
+            })
+        })
+
+    },
+
+    getAllChildUsers(params) {
+
+        let authHeader = {'Authorization': 'Bearer ' +btoa(params.client_id + ':' + params.client_sec)}
+        let endpoint = groupMgtEndpoint + "/user/group/memberships/child"
+        let parm = {
+            'group.id': params.group_id
+        }
+
+
+        return api().get(endpoint, {params: parm, headers: authHeader})
+
+    },
+
+    getAllChildGroups(params) {
+        let authHeader = {'Authorization': 'Bearer '  +btoa(params.client_id + ':' + params.client_sec)}
+        let endpoint = groupMgtEndpoint + "/groups/memberships/child"
+        let parm = {
+            'group.id': params.group_id
+        }
+        return api().get(endpoint, {params: parm, headers: authHeader})
+
+    },
+
+
+}
+
diff --git a/custos-demo-gateway/src/service/identity_management.js b/custos-demo-gateway/src/service/identity_management.js
new file mode 100644
index 0000000..670c1db
--- /dev/null
+++ b/custos-demo-gateway/src/service/identity_management.js
@@ -0,0 +1,55 @@
+import api from "./api.js";
+
+
+const identityMgtEndpoint = "/identity-management/v1.0.0"
+
+export default {
+
+    getOpenIdConfig(params) {
+        let authHeader = {'Authorization': 'Bearer ' + btoa(params.client_id + ':' + params.client_sec)}
+        return api().get(identityMgtEndpoint + '/.well-known/openid-configuration',
+            {params: {client_id: params.client_id}, headers: authHeader})
+    },
+
+    getToken(params) {
+        let authHeader = {'Authorization': 'Bearer ' + btoa(params.client_id + ':' + params.client_sec)}
+        let tokenEnpoint = params.token_endpoint
+        let code = params.code
+        let redirect_uri = params.redirect_uri
+        let body = {'code': code, 'redirect_uri': redirect_uri, 'grant_type': 'authorization_code'}
+        return api().post(tokenEnpoint,
+            body, {headers: authHeader})
+    },
+
+    localLogin(params) {
+        let authHeader = {'Authorization': 'Bearer ' + btoa(params.client_id + ':' + params.client_sec)}
+        let tokenEnpoint = params.token_endpoint
+        let username = params.username
+        let password = params.password
+        let body = {'grant_type': 'password', 'username': username, 'password': password}
+        return api().post(tokenEnpoint,
+            body, {headers: authHeader})
+    },
+
+
+    logout(params) {
+        let authHeader = {'Authorization': 'Bearer ' + btoa(params.client_id + ':' + params.client_sec)}
+        let endSessionEndpoint = identityMgtEndpoint + "/user/logout"
+        let data = {
+            refresh_token: params.refresh_token
+        }
+        return api().post(endSessionEndpoint,
+            data, {headers: authHeader})
+    },
+
+
+    getTokenUsingRefreshToken(params) {
+        let authHeader = {'Authorization': 'Bearer ' + btoa(params.client_id + ':' + params.client_sec)}
+        let tokenEndpoint = identityMgtEndpoint + "/token"
+        let body = {'refresh_token': params.refresh_token, 'grant_type': 'refresh_token'}
+        return api().post(tokenEndpoint,
+            body, {headers: authHeader})
+    }
+
+
+}
\ No newline at end of file
diff --git a/custos-demo-gateway/src/service/log_management.js b/custos-demo-gateway/src/service/log_management.js
new file mode 100644
index 0000000..e941fb0
--- /dev/null
+++ b/custos-demo-gateway/src/service/log_management.js
@@ -0,0 +1,36 @@
+import api from "./api.js";
+
+
+const logMgtEndpoint = "https://custos.scigap.org/apiserver/log-management/v1.0.0"
+
+export default {
+
+    getLogEvents(params) {
+        let authHeader = {'Authorization': 'Bearer ' + btoa(params.client_id + ':' + params.client_sec)}
+        let endpoint = logMgtEndpoint + "/logs"
+        return api().get(endpoint,  {
+            params:params.params,
+            headers: authHeader
+        })
+    },
+
+    isLoggingEnabled(params) {
+        let authHeader = {'Authorization': 'Bearer ' + btoa(params.client_id + ':' + params.client_sec)}
+        let endpoint = logMgtEndpoint + "/status"
+        return api().get(endpoint, {
+            headers: authHeader
+        })
+
+    },
+
+    enableLogging(params) {
+        let authHeader = {'Authorization': 'Bearer ' + params.user_token}
+        let endpoint = logMgtEndpoint + "/status"
+        return api().post(endpoint, {}, {
+            headers: authHeader
+        })
+
+    }
+
+
+}
\ No newline at end of file
diff --git a/custos-demo-gateway/src/service/secret_management.js b/custos-demo-gateway/src/service/secret_management.js
new file mode 100644
index 0000000..9594a87
--- /dev/null
+++ b/custos-demo-gateway/src/service/secret_management.js
@@ -0,0 +1,98 @@
+import api from "./api.js";
+import axios from "axios";
+
+var qs = require('qs');
+
+const secretMgtEndpoint = "https://custos.scigap.org/apiserver/resource-secret-management/v1.0.0"
+
+export default {
+
+    addSSHCredential(params) {
+        let authHeader = {'Authorization': 'Bearer ' + btoa(params.client_id + ':' + params.client_sec)}
+        let endpoint = secretMgtEndpoint + "/secret/ssh"
+        return api().post(endpoint, params.body, {
+            headers: authHeader
+        })
+    },
+
+    addPasswordCredential(params) {
+        let authHeader = {'Authorization': 'Bearer ' + btoa(params.client_id + ':' + params.client_sec)}
+        let endpoint = secretMgtEndpoint + "/secret/password"
+        return api().post(endpoint, params.body, {
+            headers: authHeader
+        })
+    },
+
+    getSSHCredential(params) {
+        let authHeader = {'Authorization': 'Bearer ' + btoa(params.client_id + ':' + params.client_sec)}
+        let endpoint = secretMgtEndpoint + "/secret/ssh"
+        return api().get(endpoint, {
+            params: {
+                token: params.token,
+                client_id: params.client_id
+            },
+            headers: authHeader
+        })
+    },
+
+    getPasswordCredential(params) {
+        let authHeader = {'Authorization': 'Bearer ' + btoa(params.client_id + ':' + params.client_sec)}
+        let endpoint = secretMgtEndpoint + "/secret/password"
+        return api().get(endpoint, {
+            params: {
+                token: params.token,
+                client_id: params.client_id
+            },
+            headers: authHeader
+        })
+    },
+    deleteSSHCredential(params) {
+        let authHeader = {'Authorization': 'Bearer ' + btoa(params.client_id + ':' + params.client_sec)}
+        let endpoint = secretMgtEndpoint + "/secret/ssh"
+        return new Promise((resolve, reject) => {
+            axios({
+                method: 'delete',
+                url: endpoint,
+                headers: authHeader,
+                params: {client_id: params.client_id, token: params.token}
+            }).then((resp) => {
+                resolve(resp)
+            }).catch(errr => {
+                reject(errr)
+            })
+        })
+    },
+
+    deletePasswordCredential(params) {
+        let authHeader = {'Authorization': 'Bearer ' + btoa(params.client_id + ':' + params.client_sec)}
+        let endpoint = secretMgtEndpoint + "/secret/password"
+        return new Promise((resolve, reject) => {
+            axios({
+                method: 'delete',
+                url: endpoint,
+                headers: authHeader,
+                params: {client_id: params.client_id, token: params.token}
+            }).then((resp) => {
+                resolve(resp)
+            }).catch(errr => {
+                reject(errr)
+            })
+        })
+    },
+
+    getAllCredentials(par) {
+        let authHeader = {'Authorization': 'Bearer ' + btoa(par.client_id + ':' + par.client_sec)}
+        let endpoint = secretMgtEndpoint + "/secret/summaries"
+        console.log(par)
+        return api().get(endpoint, {
+            params: par.params,
+            headers: authHeader,
+            paramsSerializer: params => {
+                return qs.stringify(params, {indices: false})
+            }
+        })
+
+    }
+
+
+}
\ No newline at end of file
diff --git a/custos-demo-gateway/src/service/sharing_management.js b/custos-demo-gateway/src/service/sharing_management.js
new file mode 100644
index 0000000..b908dd8
--- /dev/null
+++ b/custos-demo-gateway/src/service/sharing_management.js
@@ -0,0 +1,198 @@
+import api from "./api.js";
+import axios from "axios";
+
+const sharingMgtEndpoint = "https://custos.scigap.org/apiserver/sharing-management/v1.0.0"
+
+export default {
+
+    createPermissionType(params) {
+        let authHeader = {'Authorization': 'Bearer ' + btoa(params.client_id + ':' + params.client_sec)}
+        let endpoint = sharingMgtEndpoint + "/permission/type"
+        return api().post(endpoint, params.body, {
+            headers: authHeader
+        })
+    },
+
+    deletePermissionType(params) {
+        let authHeader = {'Authorization': 'Bearer ' + btoa(params.client_id + ':' + params.client_sec)}
+        let endpoint = sharingMgtEndpoint + "/permission/type"
+
+        return new Promise((resolve, reject) => {
+            axios({
+                method: 'delete',
+                url: endpoint,
+                data: params.body,
+                headers: authHeader
+            }).then((resp) => {
+                resolve(resp)
+            }).catch(errr => {
+                reject(errr)
+            })
+        })
+
+    },
+
+    getPermissionTypes(params) {
+        let authHeader = {'Authorization': 'Bearer ' + btoa(params.client_id + ':' + params.client_sec)}
+        let endpoint = sharingMgtEndpoint + "/permission/types"
+        let par = {client_id: params.client_id}
+        return api().get(endpoint, {
+            params: par,
+            headers: authHeader
+        })
+    },
+
+    createEntityType(params) {
+        let authHeader = {'Authorization': 'Bearer ' + btoa(params.client_id + ':' + params.client_sec)}
+        let endpoint = sharingMgtEndpoint + "/entity/type"
+        return api().post(endpoint, params.body, {
+            headers: authHeader
+        })
+    },
+
+    deleteEntityType(params) {
+        let authHeader = {'Authorization': 'Bearer ' + btoa(params.client_id + ':' + params.client_sec)}
+        let endpoint = sharingMgtEndpoint + "/entity/type"
+
+        return new Promise((resolve, reject) => {
+            axios({
+                method: 'delete',
+                url: endpoint,
+                data: params.body,
+                headers: authHeader
+            }).then((resp) => {
+                resolve(resp)
+            }).catch(errr => {
+                reject(errr)
+            })
+        })
+
+    },
+
+    getEntityTypes(params) {
+        let authHeader = {'Authorization': 'Bearer ' + btoa(params.client_id + ':' + params.client_sec)}
+        let endpoint = sharingMgtEndpoint + "/entity/types"
+        let par = {client_id: params.client_id}
+        return api().get(endpoint, {
+            params: par,
+            headers: authHeader
+        })
+    },
+
+    createEntity(params) {
+        let authHeader = {'Authorization': 'Bearer ' + btoa(params.client_id + ':' + params.client_sec)}
+        let endpoint = sharingMgtEndpoint + "/entity"
+        return api().post(endpoint, params.body, {
+            headers: authHeader
+        })
+    },
+
+    deleteEntity(params) {
+        let authHeader = {'Authorization': 'Bearer ' + btoa(params.client_id + ':' + params.client_sec)}
+        let endpoint = sharingMgtEndpoint + "/entity"
+
+        return new Promise((resolve, reject) => {
+            axios({
+                method: 'delete',
+                url: endpoint,
+                data: params.body,
+                headers: authHeader
+            }).then((resp) => {
+                resolve(resp)
+            }).catch(errr => {
+                reject(errr)
+            })
+        })
+
+    },
+
+    getEntities(params) {
+        let authHeader = {'Authorization': 'Bearer ' + btoa(params.client_id + ':' + params.client_sec)}
+        let endpoint = sharingMgtEndpoint + "/entities"
+        return api().post(endpoint, params.body,{
+            headers: authHeader
+        })
+    },
+
+    getListOfSharedUsers(data) {
+        let authHeader = {'Authorization': 'Bearer ' + btoa(data.client_id + ':' + data.client_sec)}
+        let endpoint = sharingMgtEndpoint + "/users/share"
+        return api().get(endpoint, {
+            params: data.params,
+            headers: authHeader
+        })
+    },
+
+    getListOfSharedGroups(data) {
+        let authHeader = {'Authorization': 'Bearer ' + btoa(data.client_id + ':' + data.client_sec)}
+        let endpoint = sharingMgtEndpoint + "/groups/share"
+        return api().get(endpoint, {
+            params: data.params,
+            headers: authHeader
+        })
+    },
+
+    shareEntityWithUsers(params) {
+        let authHeader = {'Authorization': 'Bearer ' + btoa(params.client_id + ':' + params.client_sec)}
+        let endpoint = sharingMgtEndpoint + "/users/share"
+        return api().post(endpoint, params.body, {
+            headers: authHeader
+        })
+
+    },
+
+    shareEntityWithGroups(params) {
+        let authHeader = {'Authorization': 'Bearer ' + btoa(params.client_id + ':' + params.client_sec)}
+        let endpoint = sharingMgtEndpoint + "/groups/share"
+        return api().post(endpoint, params.body, {
+            headers: authHeader
+        })
+
+    },
+
+    revokeEntitySharingFromUsers(params) {
+        let authHeader = {'Authorization': 'Bearer ' + btoa(params.client_id + ':' + params.client_sec)}
+        let endpoint = sharingMgtEndpoint + "/users/share"
+        return new Promise((resolve, reject) => {
+            axios({
+                method: 'delete',
+                url: endpoint,
+                data: params.body,
+                headers: authHeader
+            }).then((resp) => {
+                resolve(resp)
+            }).catch(errr => {
+                reject(errr)
+            })
+        })
+
+    },
+
+    revokeEntitySharingFromGroups(params) {
+        let authHeader = {'Authorization': 'Bearer ' + btoa(params.client_id + ':' + params.client_sec)}
+        let endpoint = sharingMgtEndpoint + "/groups/share"
+        return new Promise((resolve, reject) => {
+            axios({
+                method: 'delete',
+                url: endpoint,
+                data: params.body,
+                headers: authHeader
+            }).then((resp) => {
+                resolve(resp)
+            }).catch(errr => {
+                reject(errr)
+            })
+        })
+
+    },
+
+    userHasAccess(data) {
+        let authHeader = {'Authorization': 'Bearer ' + btoa(data.client_id + ':' + data.client_sec)}
+        let endpoint = sharingMgtEndpoint + "/entity/user/access"
+        return api().get(endpoint, {
+            params: data.params,
+            headers: authHeader
+        })
+    }
+
+}
\ No newline at end of file
diff --git a/custos-demo-gateway/src/service/user_management.js b/custos-demo-gateway/src/service/user_management.js
new file mode 100644
index 0000000..d7b7483
--- /dev/null
+++ b/custos-demo-gateway/src/service/user_management.js
@@ -0,0 +1,161 @@
+import api from "./api.js";
+import axios from "axios";
+
+
+const usermgtEndpoint = "https://custos.scigap.org/apiserver/user-management/v1.0.0"
+const tenantmgtEndpoint = "https://custos.scigap.org/apiserver/tenant-management/v1.0.0"
+
+export default {
+
+    registerUser(params) {
+        let authHeader = {'Authorization': 'Bearer ' + btoa(params.client_id + ':' + params.client_sec)}
+        let endpoint = usermgtEndpoint + "/user";
+        let body = {
+            'client_id': params.client_id,
+            'username': params.username,
+            'first_name': params.first_name,
+            'last_name': params.last_name,
+            'password': params.password,
+            'temporary_password': false,
+            'email': params.email
+        }
+        return api().post(endpoint,
+            body, {headers: authHeader})
+    },
+
+    enableUser(params) {
+        let authHeader = {'Authorization': 'Bearer ' + btoa(params.client_id + ':' + params.client_sec)}
+        let endpoint = usermgtEndpoint + "/user/activation";
+        let body = {
+            username: params.username
+        }
+        return api().post(endpoint, body
+            , {
+                headers: authHeader
+            })
+    },
+
+    disableUser(params) {
+        let authHeader = {'Authorization': 'Bearer ' + btoa(params.client_id + ':' + params.client_sec)}
+        let endpoint = usermgtEndpoint + "/user/deactivation";
+        let body = {
+            username: params.username
+        }
+        return api().post(endpoint, body
+            , {
+                headers: authHeader
+            })
+    },
+
+    checkUsernameValidity(params) {
+        let authHeader = {'Authorization': 'Bearer ' + btoa(params.client_id + ':' + params.client_sec)}
+        let endpoint = usermgtEndpoint + "/user/availability";
+        return api().get(endpoint,
+            {
+                params: {'user.username': params.username},
+                headers: authHeader
+            })
+    },
+
+    findUsers(params) {
+        let authHeader = {'Authorization': 'Bearer ' + btoa(params.client_id + ':' + params.client_sec)}
+        let endpoint = usermgtEndpoint + "/users";
+        let id = params.username
+        let param = {offset: params.offset, limit: params.limit, client_id: params.client_id, 'user.id': id}
+        return api().get(endpoint,
+            {
+                params: param,
+                headers: authHeader
+            })
+    },
+
+    addUserAttribute(params) {
+        let authHeader = {'Authorization': 'Bearer ' + params.user_token}
+        let endpoint = usermgtEndpoint + "/attributes";
+        return api().post(endpoint, params.body
+            , {
+                headers: authHeader
+            })
+    },
+
+    deleteUserAttribute(params) {
+        let authHeader = {'Authorization': 'Bearer ' + params.user_token}
+        let endpoint = usermgtEndpoint + "/attributes";
+        return new Promise((resolve, reject) => {
+            axios({
+                method: 'delete',
+                url: endpoint,
+                data: params.body,
+                headers: authHeader
+            }).then((resp) => {
+                resolve(resp)
+            }).catch(errr => {
+                reject(errr)
+            })
+        })
+
+
+    },
+
+    addRolesToUser(params) {
+        let authHeader = {'Authorization': 'Bearer ' + params.user_token}
+        let endpoint = usermgtEndpoint + "/users/roles";
+        return api().post(endpoint, params.body
+            , {
+                headers: authHeader
+            })
+
+    },
+
+    deleteRolesFromUser(params) {
+        let authHeader = {'Authorization': 'Bearer ' + params.user_token}
+        let endpoint = usermgtEndpoint + "/user/roles";
+        return new Promise((resolve, reject) => {
+            axios({
+                method: 'delete',
+                url: endpoint,
+                data: params.body,
+                headers: authHeader
+            }).then((resp) => {
+                resolve(resp)
+            }).catch(errr => {
+                reject(errr)
+            })
+        })
+    },
+
+    updateProfile(params) {
+        let authHeader = {'Authorization': 'Bearer '  + btoa(params.client_id + ':' + params.client_sec)}
+        let endpoint = usermgtEndpoint + "/user/profile";
+        console.log(authHeader)
+        return api().put(endpoint, params.body
+            , {
+                headers: authHeader
+            })
+
+    },
+
+
+    getTenantLevelRoles(params) {
+        let authHeader = {'Authorization': 'Bearer ' + btoa(params.client_id + ':' + params.client_sec)}
+        let endpoint = tenantmgtEndpoint + "/roles"
+        return api().get(endpoint,
+            {
+                headers: authHeader
+            })
+    },
+
+    getClientLevelRoles(params) {
+        let authHeader = {'Authorization': 'Bearer ' + btoa(params.client_id + ':' + params.client_sec)}
+        let endpoint = tenantmgtEndpoint + "/roles"
+        let par = {client_level: true}
+        return api().get(endpoint,
+            {
+                params: par,
+                headers: authHeader
+            })
+
+    }
+
+
+}
\ No newline at end of file
diff --git a/custos-demo-gateway/src/store/index.js b/custos-demo-gateway/src/store/index.js
new file mode 100644
index 0000000..4affb81
--- /dev/null
+++ b/custos-demo-gateway/src/store/index.js
@@ -0,0 +1,29 @@
+import Vue from 'vue';
+import Vuex from 'vuex';
+import createLogger from 'vuex/dist/logger';
+import identity from './modules/identity.store';
+import user from './modules/user.store'
+import agent from './modules/agent.store'
+import group from './modules/group.store'
+import secret from './modules/secret.store'
+import log from './modules/log.store'
+import sharing from './modules/sharing.store'
+
+
+Vue.use(Vuex);
+
+const debug = process.env.NODE_ENV !== 'production';
+
+export default new Vuex.Store({
+    modules: {
+        identity,
+        user,
+        agent,
+        group,
+        secret,
+        log,
+        sharing
+    },
+    strict: debug,
+    plugins: debug ? [createLogger()] : [],
+})
\ No newline at end of file
diff --git a/custos-demo-gateway/src/store/modules/agent.store.js b/custos-demo-gateway/src/store/modules/agent.store.js
new file mode 100644
index 0000000..1dcfd5d
--- /dev/null
+++ b/custos-demo-gateway/src/store/modules/agent.store.js
@@ -0,0 +1,155 @@
+/*Agent.store.js*/
+
+import agent_management from "@/service/agent_management";
+
+
+const getDefaultState = () => {
+    return {
+        agents: [],
+        clientLevelRoles: [],
+        isAgentsEnabled: false
+    }
+}
+
+const state = getDefaultState()
+
+const actions = {
+
+    async enableAgents({commit}, data) {
+        let resp = await agent_management.enableAgents(data)
+        commit('SET_AGENTS_ENABLED', resp.data.status)
+        return resp.data.status
+    },
+
+    async registerAgent({commit}, data) {
+        let rep = await agent_management.registerAgent(data)
+        let id = rep.data.id
+        let da = {
+            id: id,
+            status: 'ACTIVE',
+            attributes: []
+        }
+        commit('ADD_AGENTS', da)
+        return rep.data
+    },
+
+    // eslint-disable-next-line no-unused-vars
+    async getAgent({commit}, data) {
+        let response = await agent_management.getAgent(data)
+        return response.data
+    },
+
+    // eslint-disable-next-line no-unused-vars
+    async activateAgent({commit}, data) {
+        let response = await agent_management.activateAgent(data)
+        let id = data.agent_id
+        commit('SET_ACTIVATED_AGENT', id)
+        return response.data
+    },
+
+    // eslint-disable-next-line no-unused-vars
+    async deactivateAgent({commit}, data) {
+        let response = await agent_management.deactivateAgent(data)
+        let id = data.agent_id
+        commit('SET_DEACTIVATED_AGENT', id)
+        return response.data
+    },
+
+
+    // eslint-disable-next-line no-unused-vars
+    async addAttributeToAgent({commit}, data) {
+        let response = await agent_management.addAttributesToAgent(data)
+        return response.data
+    },
+
+    // eslint-disable-next-line no-unused-vars
+    async deleteAttributeFromAgent({commit}, data) {
+        let response = await agent_management.deleteAttributesFromAgent(data)
+        return response.data
+    },
+
+
+    async get_all_agents({commit}, data) {
+        let response = await agent_management.getAllAgents(data)
+        let agents = response.data.agents
+        agents.forEach(agent => {
+            agent.status = (agent.isEnabled ? 'ACTIVE':'DISABLED')
+        })
+        commit('SET_AGENTS', response.data.agents)
+        return response.data.agents
+    },
+    // eslint-disable-next-line no-unused-vars
+    async reset({commit}, data){
+        commit('RESET')
+        return true
+    }
+
+}
+
+
+const mutations = {
+
+
+    ADD_AGENTS(state, data) {
+        state.agents.push(data)
+    },
+
+    SET_AGENTS(state, data) {
+        state.agents = data
+    },
+
+    SET_CLIENT_LEVEL_ROLES(state, data) {
+        state.clientLevelRoles = data
+    },
+
+    SET_AGENTS_ENABLED(state, data) {
+        state.isAgentsEnabled = data
+    },
+
+
+    SET_DEACTIVATED_AGENT(state, data) {
+        state.agents.forEach((agent) => {
+            if (agent.id == data) {
+                agent.status = "DISABLED"
+            }
+        })
+    },
+
+    SET_ACTIVATED_AGENT(state, data) {
+        state.agents.forEach((agent) => {
+            if (agent.id == data) {
+                agent.status = "ACTIVE"
+            }
+        })
+    },
+
+    RESET(state) {
+        Object.assign(state, getDefaultState())
+    }
+
+}
+
+const getters = {
+
+    getAgents(state) {
+        return state.agents
+    },
+
+    isAgentsAreEnabled(state) {
+        return state.isAgentsEnabled
+    },
+
+    getAgentClientRoles(state) {
+        return state.clientLevelRoles
+    }
+
+}
+
+export default {
+    namespaced: true,
+    state,
+    getters,
+    actions,
+    mutations
+}
+
diff --git a/custos-demo-gateway/src/store/modules/group.store.js b/custos-demo-gateway/src/store/modules/group.store.js
new file mode 100644
index 0000000..986a580
--- /dev/null
+++ b/custos-demo-gateway/src/store/modules/group.store.js
@@ -0,0 +1,153 @@
+/*Group.store.js*/
+
+import group_management from "@/service/group_management";
+
+
+const getDefaultState = () => {
+    return {
+        groups: [],
+    }
+}
+
+const state = getDefaultState()
+
+
+const actions = {
+
+    async createGroup({commit}, data) {
+        let resp = await group_management.createGroup(data)
+        let group = resp.data.groups[0];
+        commit('SET_GROUP', group)
+        return group
+    },
+
+    async loadAllGroups({commit}, data) {
+        let rep = await group_management.getAllGroups(data)
+        let groups = rep.data.groups
+        commit('SET_GROUPS', groups)
+        return groups
+    },
+
+
+    async updateGroup({commit}, data) {
+        let response = await group_management.updateGroup(data)
+        commit('SET_GROUP', response.data)
+        return response.data
+    },
+
+    async deleteGroup({commit}, data) {
+        let response = await group_management.deleteGroup(data)
+        if (response.data.status) {
+            commit('SET_DELETED_GROUP', data.group_id)
+            return response.data
+        }
+    },
+
+    // eslint-disable-next-line no-unused-vars
+    async getGroup({commit}, data) {
+        let response = await group_management.findGroup(data)
+        return response.data
+    },
+
+
+    // eslint-disable-next-line no-unused-vars
+    async addUserToGroup({commit}, data) {
+        let response = await group_management.addUserToGroup(data)
+        return response.data
+    },
+
+    // eslint-disable-next-line no-unused-vars
+    async removeUserFromGroup({commit}, data) {
+        let response = await group_management.removeUserFromGroup(data)
+        return response.data
+    },
+
+    // eslint-disable-next-line no-unused-vars
+    async addChildGroup({commit}, data) {
+        let response = await group_management.addChildGroup(data)
+        return response.data
+    },
+
+    // eslint-disable-next-line no-unused-vars
+    async removeChildGroup({commit}, data) {
+        let response = await group_management.removeChildGroup(data)
+        return response.data
+    },
+
+    // eslint-disable-next-line no-unused-vars
+    async changeGroupMembership({commit}, data) {
+        let response = await group_management.changeGroupMembership(data)
+        return response.data
+    },
+
+    // eslint-disable-next-line no-unused-vars
+    async getAllChildUsers({commit}, data) {
+        let response = await group_management.getAllChildUsers(data)
+        return response.data
+    },
+
+    // eslint-disable-next-line no-unused-vars
+    async getAllChildGroups({commit}, data) {
+        let response = await group_management.getAllChildGroups(data)
+        return response.data
+    },
+
+    // eslint-disable-next-line no-unused-vars
+    async reset({commit}, data){
+        commit('RESET')
+        return true
+    }
+
+}
+
+
+const mutations = {
+
+
+    SET_GROUP(state, data) {
+        let grs = []
+        state.groups.forEach((gr) => {
+            if (gr.id !== data.id) {
+                grs.push(gr)
+            }
+        })
+        grs.push(data)
+        state.groups = grs
+    },
+
+    SET_DELETED_GROUP(state, data) {
+        let grs = []
+        state.groups.forEach((gr) => {
+            if (gr.id !== data) {
+                grs.push(gr)
+            }
+        })
+        state.groups = grs
+    },
+
+    SET_GROUPS(state, data) {
+        state.groups = data
+    },
+
+
+    RESET(state) {
+        Object.assign(state, getDefaultState())
+    }
+
+}
+
+const getters = {
+
+    getGroups(state) {
+        return state.groups
+    }
+
+}
+
+export default {
+    namespaced: true,
+    state,
+    getters,
+    actions,
+    mutations
+}
diff --git a/custos-demo-gateway/src/store/modules/identity.store.js b/custos-demo-gateway/src/store/modules/identity.store.js
new file mode 100644
index 0000000..0bd66eb
--- /dev/null
+++ b/custos-demo-gateway/src/store/modules/identity.store.js
@@ -0,0 +1,134 @@
+/*Identity.store.js*/
+
+import identity_management from "@/service/identity_management";
+import auth from "@/service/auth";
+
+
+const state = {
+    authorizationEndpoint: null,
+    idToken: localStorage.getItem('id_token') || '',
+    accessToken: localStorage.getItem('access_token') || '',
+    refreshToken: localStorage.getItem('refresh_token') || '',
+    currentUserName: null
+}
+const actions = {
+
+    async fetchAuthorizationEndpoint({commit}, data) {
+        let resp = await identity_management.getOpenIdConfig(data)
+        let baseURL = resp.data.authorization_endpoint
+        this.authorizartionURL = baseURL + "?response_type=code&client_id=" + data.client_id + "&" +
+            "redirect_uri="+data.redirect_uri+"&scope=openid&kc_idp_hint=oidc"
+        commit('SET_AUTH_ENDPOINT', this.authorizartionURL)
+
+    },
+
+    async authenticateUsingCode({commit}, data) {
+        let resp = await identity_management.getToken(data)
+        commit('SET_AUTH_TOKEN', resp.data)
+    },
+
+    async authenticateLocally({commit}, data) {
+        try {
+            let resp = await identity_management.localLogin(data)
+            commit('SET_AUTH_TOKEN', resp.data)
+        }catch (e) {
+            return false
+        }
+    },
+
+    async logout({commit}, data) {
+        let dat = {
+            client_id: data.client_id,
+            client_sec: data.client_sec,
+            refresh_token: auth.getRefreshToken()
+        }
+        await identity_management.logout(dat)
+        commit('CLEAR_AUTH_TOKEN', data)
+    },
+
+    // eslint-disable-next-line no-unused-vars
+    async isAuthenticated({commit}, data) {
+        try {
+            let resp = auth.isLoggedIn()
+            if (!resp) {
+                let dat = {
+                    client_id: data.client_id,
+                    client_sec: data.client_sec,
+                    refresh_token: auth.getRefreshToken()
+                }
+                let response = await identity_management.getTokenUsingRefreshToken(dat)
+                commit('SET_AUTH_TOKEN', response.data)
+                return auth.isLoggedIn()
+            }
+            return true
+        } catch (e) {
+            commit('CLEAR_AUTH_TOKEN')
+        }
+
+    },
+
+
+    // eslint-disable-next-line no-unused-vars
+    async isLoggedUserHasAdminAccess({commit, data}){
+        return auth.isUserHasAdminAccess()
+    },
+
+    // eslint-disable-next-line no-unused-vars
+      async getCurrentUserName({commit}, data){
+          return auth.getLoggedUsername()
+      }
+
+}
+
+
+const mutations = {
+
+    SET_AUTH_ENDPOINT(state, data) {
+        state.authorizationEndpoint = data
+    },
+
+    SET_AUTH_TOKEN(state, data) {
+        auth.clearIdToken()
+        auth.clearAccessToken()
+        auth.clearRefreshToken()
+        auth.setIdToken(data.id_token)
+        auth.setAccessToken(data.access_token)
+        auth.setRefreshToken(data.refresh_token)
+        state.token = data.id_token
+        state.accessToken = data.access_token
+        state.refreshToken = data.refresh_token
+    },
+
+    // eslint-disable-next-line no-unused-vars
+    CLEAR_AUTH_TOKEN(state, data) {
+        auth.clearIdToken()
+        auth.clearAccessToken()
+        auth.clearRefreshToken()
+        state.idToken = ''
+        state.accessToken = ''
+        state.refreshToken = ''
+    }
+}
+
+const getters = {
+    getAuthorizationEndpoint(state) {
+        return state.authorizationEndpoint
+    },
+    isAuthenticated() {
+        return auth.isLoggedIn()
+    },
+
+    getAccessToken(state) {
+        return state.accessToken;
+    },
+
+
+}
+
+export default {
+    namespaced: true,
+    state,
+    getters,
+    actions,
+    mutations
+}
\ No newline at end of file
diff --git a/custos-demo-gateway/src/store/modules/log.store.js b/custos-demo-gateway/src/store/modules/log.store.js
new file mode 100644
index 0000000..993b1c6
--- /dev/null
+++ b/custos-demo-gateway/src/store/modules/log.store.js
@@ -0,0 +1,46 @@
+/*log.store.js*/
+
+import log_management from "@/service/log_management";
+
+const getDefaultState = () => {
+    return {}
+}
+
+const state = getDefaultState()
+
+const actions = {
+
+    // eslint-disable-next-line no-unused-vars
+    async getLogEvents({commit}, data) {
+        let response = await log_management.getLogEvents(data)
+        return response.data.events
+    },
+
+
+    // eslint-disable-next-line no-unused-vars
+    async isLoggingEnabled({commit}, data) {
+        let response = await log_management.isLoggingEnabled(data)
+        return response.data.status
+    },
+
+
+    // eslint-disable-next-line no-unused-vars
+    async enableLogging({commit}, data) {
+        let response = await log_management.enableLogging(data)
+        return response.data.status
+    },
+
+
+}
+
+const mutations = {}
+
+const getters = {}
+
+export default {
+    namespaced: true,
+    state,
+    getters,
+    actions,
+    mutations
+}
\ No newline at end of file
diff --git a/custos-demo-gateway/src/store/modules/secret.store.js b/custos-demo-gateway/src/store/modules/secret.store.js
new file mode 100644
index 0000000..68897cf
--- /dev/null
+++ b/custos-demo-gateway/src/store/modules/secret.store.js
@@ -0,0 +1,116 @@
+/*Secret.store.js*/
+
+import secret_management from "@/service/secret_management";
+
+
+const getDefaultState = () => {
+    return {
+        secrets: []
+    }
+}
+
+const state = getDefaultState()
+
+const actions = {
+
+    async addSSHCredential({commit}, data) {
+        let resp = await secret_management.addSSHCredential(data)
+        commit('ADD_SECRET', resp.data.token)
+        return resp.data
+    },
+
+
+    async addPasswordCredential({commit}, data) {
+        let resp = await secret_management.addPasswordCredential(data)
+        commit('ADD_SECRET', resp.data.token)
+        return resp.data
+    },
+
+
+    // eslint-disable-next-line no-unused-vars
+    async getSSHCredential({commit}, data) {
+        let resp = await secret_management.getSSHCredential(data)
+        return resp.data
+    },
+
+
+    // eslint-disable-next-line no-unused-vars
+    async getPasswordCredential({commit}, data) {
+        let resp = await secret_management.getPasswordCredential(data)
+        return resp.data
+    },
+
+    async deleteSSHCredential({commit}, data) {
+        let resp = await secret_management.deleteSSHCredential(data)
+        if (resp.data.status) {
+            commit('DELETE_SECRET', data.token)
+        }
+        return resp.data
+    },
+
+    async deletePassswordCredential({commit}, data) {
+        let resp = await secret_management.deletePasswordCredential(data)
+        if (resp.data.status) {
+            commit('DELETE_SECRET', data.token)
+        }
+        return resp.data
+    },
+
+    // eslint-disable-next-line no-unused-vars
+    async getAllCredentials({commit}, data) {
+        let dat = {
+            client_id: data.client_id, client_sec: data.client_sec, params: {
+                client_id: data.client_id,
+                accessible_tokens: data.accessible_tokens
+            }
+        }
+        console.log(dat)
+        let response = await secret_management.getAllCredentials(dat)
+        return response.data.metadata
+    },
+
+    // eslint-disable-next-line no-unused-vars
+    async reset({commit}, data){
+        commit('RESET')
+        return true
+    }
+
+}
+
+const mutations = {
+
+    ADD_SECRET(state, data) {
+        state.secrets.push(data)
+    },
+
+    DELETE_SECRET(state, data) {
+        let secrets = []
+        state.secrets.forEach(sec => {
+            if (sec != data) {
+                secrets.push(sec)
+            }
+        })
+        state.secrets = secrets
+    },
+
+    RESET(state) {
+        Object.assign(state, getDefaultState())
+    }
+
+}
+
+const getters = {
+
+    getSecrets(state) {
+        return state.secrets
+    }
+
+}
+
+export default {
+    namespaced: true,
+    state,
+    getters,
+    actions,
+    mutations
+}
\ No newline at end of file
diff --git a/custos-demo-gateway/src/store/modules/sharing.store.js b/custos-demo-gateway/src/store/modules/sharing.store.js
new file mode 100644
index 0000000..2de7e5b
--- /dev/null
+++ b/custos-demo-gateway/src/store/modules/sharing.store.js
@@ -0,0 +1,227 @@
+import sharing_management from "@/service/sharing_management";
+
+const getDefaultState = () => {
+    return {
+        entities: [],
+        permissionTypes: [],
+        entityTypes: [],
+        sharings: []
+    }
+}
+
+const state = getDefaultState()
+
+const actions = {
+
+    async createPermissionType({commit}, data) {
+        let response = await sharing_management.createPermissionType(data)
+        if (response.data.status) {
+            commit('ADD_PERMISSION_TYPE', data.body.permission_type)
+        }
+        return state.permissionTypes
+    },
+
+    async deletePermissionType({commit}, data) {
+        let response = await sharing_management.deletePermissionType(data)
+        if (response.data.status) {
+            commit('REMOVE_PERMISSION_TYPE', data.body.permission_type)
+        }
+        return state.permissionTypes
+    },
+
+    async createEntityType({commit}, data) {
+        let response = await sharing_management.createEntityType(data)
+        if (response.data.status) {
+            commit('ADD_ENTITY_TYPE', data.body.entity_type)
+        }
+        return state.entityTypes
+    },
+
+    async deleteEntityType({commit}, data) {
+        let response = await sharing_management.deleteEntityType(data)
+        if (response.data.status) {
+            commit('REMOVE_ENTITY_TYPE', data.body.entity_type)
+        }
+        return state.entityTypes
+    },
+
+    async createEntity({commit}, data) {
+        let response = await sharing_management.createEntity(data)
+        if (response.data.status) {
+            commit('ADD_ENTITY', data.body.entity)
+        }
+        return state.entities
+    },
+
+    async deleteEntity({commit}, data) {
+        let response = await sharing_management.deleteEntity(data)
+        if (response.data.status) {
+            commit('REMOVE_ENTITY', data.body.entity)
+        }
+        return state.entities
+    },
+
+    async getPermissionTypes({commit}, data) {
+        let response = await sharing_management.getPermissionTypes(data)
+        commit('SET_PERMISSION_TYPES', response.data.types)
+        return response.data.types
+    },
+
+    async getEntityTypes({commit}, data) {
+        let response = await sharing_management.getEntityTypes(data)
+        commit('SET_ENTITY_TYPES', response.data.types)
+        return response.data.types
+    },
+
+    async getEntities({commit}, data) {
+        let response = await sharing_management.getEntities(data)
+        commit('SET_ENTITIES', response.data.entity_array)
+        return response.data.entity_array
+    },
+
+    // eslint-disable-next-line no-unused-vars
+    async getListOfSharedUsers({commit}, data) {
+        let response = await sharing_management.getListOfSharedUsers(data)
+        return response.data.owner_ids
+    },
+
+    // eslint-disable-next-line no-unused-vars
+    async getListOfSharedGroups({commit}, data) {
+        let response = await sharing_management.getListOfSharedGroups(data)
+        return response.data.owner_ids
+    },
+
+
+    // eslint-disable-next-line no-unused-vars
+    async shareEntityWithUsers({commit}, data) {
+        let response = await sharing_management.shareEntityWithUsers(data)
+        return response.data.status
+    },
+
+
+    // eslint-disable-next-line no-unused-vars
+    async shareEntityWithGroups({commit}, data) {
+        let response = await sharing_management.shareEntityWithGroups(data)
+        return response.data.status
+    },
+
+    // eslint-disable-next-line no-unused-vars
+    async revokeEntitySharingFromUsers({commit}, data) {
+        let response = await sharing_management.revokeEntitySharingFromUsers(data)
+        return response.data.status
+    },
+
+    // eslint-disable-next-line no-unused-vars
+    async revokeEntitySharingFromGroups({commit}, data) {
+        let response = await sharing_management.revokeEntitySharingFromGroups(data)
+        return response.data.status
+    },
+
+    // eslint-disable-next-line no-unused-vars
+    async userHasAccess({commit}, data) {
+        let response = await sharing_management.userHasAccess(data)
+        return response.data.status
+    },
+
+    // eslint-disable-next-line no-unused-vars
+    async reset({commit}, data){
+        commit('RESET')
+        return true
+    }
+}
+
+const mutations = {
+
+    ADD_PERMISSION_TYPE(state, data) {
+        state.permissionTypes.push(data)
+    },
+
+    REMOVE_PERMISSION_TYPE(state, data) {
+        let permTypes = []
+        state.permissionTypes.forEach(perm => {
+            if (perm.id != data.id) {
+                permTypes.push(perm)
+            }
+        })
+        state.permissionTypes = permTypes
+    },
+
+    ADD_ENTITY_TYPE(state, data) {
+        state.entityTypes.push(data)
+    },
+
+    REMOVE_ENTITY_TYPE(state, data) {
+        let permTypes = []
+        state.entityTypes.forEach(perm => {
+            if (perm.id != data.id) {
+                permTypes.push(perm)
+            }
+        })
+        state.entityTypes = permTypes
+    },
+
+    ADD_ENTITY(state, data) {
+        state.entities.push(data)
+    },
+
+    REMOVE_ENTITY(state, data) {
+        let permTypes = []
+        state.entities.forEach(perm => {
+            if (perm.id !== data.id) {
+                permTypes.push(perm)
+            }
+        })
+        state.entities = permTypes
+    },
+
+    SET_PERMISSION_TYPES(state, data) {
+        state.permissionTypes = data
+    },
+
+    SET_ENTITY_TYPES(state, data) {
+        state.entityTypes = data
+    },
+
+    SET_ENTITIES(state, data) {
+        state.entities = data
+    },
+
+    SET_SHARING(state, data) {
+        state.sharings = data
+    },
+
+    RESET(state) {
+        Object.assign(state, getDefaultState())
+    }
+
+
+}
+
+const getters = {
+
+    getEntities(state) {
+        return state.entities
+    },
+    getPermissionTypes(state) {
+        return state.permissionTypes
+    },
+    getEntityTypes(state) {
+        return state.entityTypes
+    },
+
+    getSharings(state) {
+        return state.sharings
+    },
+
+
+
+}
+
+
+export default {
+    namespaced: true,
+    state,
+    getters,
+    actions,
+    mutations
+}
\ No newline at end of file
diff --git a/custos-demo-gateway/src/store/modules/user.store.js b/custos-demo-gateway/src/store/modules/user.store.js
new file mode 100644
index 0000000..24b80be
--- /dev/null
+++ b/custos-demo-gateway/src/store/modules/user.store.js
@@ -0,0 +1,149 @@
+import user_management from "@/service/user_management";
+
+
+const getDefaultState = () => {
+    return {
+        users: [],
+        tenantLevelRoles: [],
+        clientLevelRoles: []
+    }
+}
+
+const state = getDefaultState()
+
+const actions = {
+    // eslint-disable-next-line no-unused-vars
+    async registerUser({commit}, data) {
+        let resp = await user_management.registerUser(data)
+        if (resp.data.is_registered) {
+            let secData = {
+                'username': data.username, 'client_id': data.client_id,
+                'client_sec': data.client_sec,
+            }
+            await user_management.enableUser(secData)
+            return true
+        }
+    },
+
+    // eslint-disable-next-line no-unused-vars
+    async checkUsernameIsValid({commit}, data) {
+
+        let rep = await user_management.checkUsernameValidity(data)
+        return rep.data.status
+    },
+
+    async users({commit}, data) {
+        let response = await user_management.findUsers(data)
+        commit('SET_USERS', response.data.users)
+        return  response.data.users
+    },
+
+    // eslint-disable-next-line no-unused-vars
+    async addUserAttributes({commit}, data) {
+        let response = await user_management.addUserAttribute(data)
+        return response.data.status
+    },
+
+    // eslint-disable-next-line no-unused-vars
+    async deleteUserAttributes({commit}, data) {
+        let response = await user_management.deleteUserAttribute(data)
+        return response.data.status
+    },
+
+    async getTenantLevelRoles({commit}, data) {
+        let roles = await user_management.getTenantLevelRoles(data)
+        let tRoles = []
+        roles.data.roles.forEach(r => {
+            tRoles.push(r.name)
+        })
+        commit('SET_TENANT_LEVEL_ROLES', tRoles)
+        return tRoles
+
+    },
+
+    async getClientLevelRoles({commit}, data) {
+        let roles = await user_management.getClientLevelRoles(data)
+        let tRoles = []
+        roles.data.roles.forEach(r => {
+            tRoles.push(r.name)
+        })
+        commit('SET_CLIENT_LEVEL_ROLES', tRoles)
+        return tRoles
+
+    },
+
+    // eslint-disable-next-line no-unused-vars
+    async addRoleToUser({commit}, data) {
+        let response = await user_management.addRolesToUser(data)
+        return response.data.status
+    },
+
+    // eslint-disable-next-line no-unused-vars
+    async deleteRoleFromUser({commit}, data) {
+        let response = await user_management.deleteRolesFromUser(data)
+        return response.data.status
+    },
+
+    // eslint-disable-next-line no-unused-vars
+    async enableUser ({commit}, data) {
+       let response =  await  user_management.enableUser(data)
+        return response.data.state
+    },
+
+    // eslint-disable-next-line no-unused-vars
+    async disableUser({commit}, data) {
+      let response =  await  user_management.disableUser(data)
+        return response.data.state
+    },
+
+    // eslint-disable-next-line no-unused-vars
+    async updateUserProfile({commit}, data) {
+        return  await  user_management.updateProfile(data)
+    },
+
+    // eslint-disable-next-line no-unused-vars
+    async reset({commit}, data){
+        commit('RESET')
+        return true
+    }
+
+}
+
+
+const mutations = {
+
+
+    SET_USERS(state, data) {
+        state.users = data
+    },
+
+
+    SET_TENANT_LEVEL_ROLES(state, data) {
+        state.tenantLevelRoles = data
+    },
+
+    SET_CLIENT_LEVEL_ROLES(state, data) {
+        state.clientLevelRoles = data
+    },
+
+    RESET(state) {
+        Object.assign(state, getDefaultState())
+    }
+
+}
+
+const getters = {
+
+    getUsers(state) {
+        return state.users
+    },
+
+}
+
+export default {
+    namespaced: true,
+    state,
+    getters,
+    actions,
+    mutations
+}
\ No newline at end of file
diff --git a/custos-demo-gateway/yarn-error.log b/custos-demo-gateway/yarn-error.log
new file mode 100644
index 0000000..0eb12e1
--- /dev/null
+++ b/custos-demo-gateway/yarn-error.log
@@ -0,0 +1,8587 @@
+Arguments: 
+  /usr/local/Cellar/node@10/10.20.1_1/bin/node /usr/local/Cellar/yarn/1.22.4/libexec/bin/yarn.js install
+
+PATH: 
+  /Library/Frameworks/Python.framework/Versions/3.6/bin:/usr/local/opt/node@10/bin:/usr/local/bin:/usr/bin:/bin:/usr/sbin:/sbin:/Library/TeX/texbin:/opt/X11/bin
+
+Yarn version: 
+  1.22.4
+
+Node version: 
+  10.20.1
+
+Platform: 
+  darwin x64
+
+Trace: 
+  SyntaxError: /Users/isururanawaka/Documents/Airavata_Repository/custos-demo-gateway/package.json: Unexpected token } in JSON at position 631
+      at JSON.parse (<anonymous>)
+      at /usr/local/Cellar/yarn/1.22.4/libexec/lib/cli.js:1625:59
+      at Generator.next (<anonymous>)
+      at step (/usr/local/Cellar/yarn/1.22.4/libexec/lib/cli.js:310:30)
+      at /usr/local/Cellar/yarn/1.22.4/libexec/lib/cli.js:321:13
+
+npm manifest: 
+  {
+    "name": "custos-demo-gateway",
+    "version": "0.1.0",
+    "private": true,
+    "scripts": {
+      "serve": "vue-cli-service serve",
+      "build": "vue-cli-service build",
+      "lint": "vue-cli-service lint"
+    },
+    "dependencies": {
+      "core-js": "^3.6.5",
+      "vue": "^2.6.11",
+      "vue-router": "^3.4.3",
+      "bootstrap-vue": "^2.16.0",
+      "axios": "^0.20.0"
+    },
+    "devDependencies": {
+      "@vue/cli-plugin-babel": "~4.5.0",
+      "@vue/cli-plugin-eslint": "~4.5.0",
+      "@vue/cli-service": "~4.5.0",
+      "babel-eslint": "^10.1.0",
+      "eslint": "^6.7.2",
+      "eslint-plugin-vue": "^6.2.2",
+      "vue-template-compiler": "^2.6.11",
+  
+    },
+    "eslintConfig": {
+      "root": true,
+      "env": {
+        "node": true
+      },
+      "extends": [
+        "plugin:vue/essential",
+        "eslint:recommended"
+      ],
+      "parserOptions": {
+        "parser": "babel-eslint"
+      },
+      "rules": {}
+    },
+    "browserslist": [
+      "> 1%",
+      "last 2 versions",
+      "not dead"
+    ]
+  }
+
+yarn manifest: 
+  No manifest
+
+Lockfile: 
+  # THIS IS AN AUTOGENERATED FILE. DO NOT EDIT THIS FILE DIRECTLY.
+  # yarn lockfile v1
+  
+  
+  "@ant-design-vue/babel-helper-vue-transform-on@^1.0.0":
+    version "1.0.1"
+    resolved "https://registry.yarnpkg.com/@ant-design-vue/babel-helper-vue-transform-on/-/babel-helper-vue-transform-on-1.0.1.tgz#d219d92f4e1fc5e7add211c347c7fa000518b623"
+    integrity sha512-dOAPf/tCM2lCG8FhvOMFBaOdMElMEGhOoocMXEWvHW2l1KIex+UibDcq4bdBEJpDMLrnbNOqci9E7P2dARP6lg==
+  
+  "@ant-design-vue/babel-plugin-jsx@^1.0.0-0":
+    version "1.0.0-rc.1"
+    resolved "https://registry.yarnpkg.com/@ant-design-vue/babel-plugin-jsx/-/babel-plugin-jsx-1.0.0-rc.1.tgz#ae56cecbda9f08691bcf92dfe98e2416e77d758b"
+    integrity sha512-x7PfAHSs5/emIuey1Df7Bh/vJU27S9KBdufzoAA7kgwTpEpY85R7CXD9gl6sJFB7aG2pZpl4Tmm+FsHlzgp7fA==
+    dependencies:
+      "@ant-design-vue/babel-helper-vue-transform-on" "^1.0.0"
+      "@babel/helper-module-imports" "^7.0.0"
+      "@babel/plugin-syntax-jsx" "^7.0.0"
+      "@babel/traverse" "^7.0.0"
+      "@babel/types" "^7.0.0"
+      camelcase "^6.0.0"
+      html-tags "^3.1.0"
+      svg-tags "^1.0.0"
+  
+  "@babel/code-frame@^7.0.0", "@babel/code-frame@^7.10.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/code-frame/-/code-frame-7.10.4.tgz#168da1a36e90da68ae8d49c0f1b48c7c6249213a"
+    integrity sha512-vG6SvB6oYEhvgisZNFRmRCUkLz11c7rp+tbNTynGqc6mS1d5ATd/sGyV6W0KZZnXRKMTzZDRgQT3Ou9jhpAfUg==
+    dependencies:
+      "@babel/highlight" "^7.10.4"
+  
+  "@babel/compat-data@^7.10.4", "@babel/compat-data@^7.11.0":
+    version "7.11.0"
+    resolved "https://registry.yarnpkg.com/@babel/compat-data/-/compat-data-7.11.0.tgz#e9f73efe09af1355b723a7f39b11bad637d7c99c"
+    integrity sha512-TPSvJfv73ng0pfnEOh17bYMPQbI95+nGWc71Ss4vZdRBHTDqmM9Z8ZV4rYz8Ks7sfzc95n30k6ODIq5UGnXcYQ==
+    dependencies:
+      browserslist "^4.12.0"
+      invariant "^2.2.4"
+      semver "^5.5.0"
+  
+  "@babel/core@^7.11.0":
+    version "7.11.6"
+    resolved "https://registry.yarnpkg.com/@babel/core/-/core-7.11.6.tgz#3a9455dc7387ff1bac45770650bc13ba04a15651"
+    integrity sha512-Wpcv03AGnmkgm6uS6k8iwhIwTrcP0m17TL1n1sy7qD0qelDu4XNeW0dN0mHfa+Gei211yDaLoEe/VlbXQzM4Bg==
+    dependencies:
+      "@babel/code-frame" "^7.10.4"
+      "@babel/generator" "^7.11.6"
+      "@babel/helper-module-transforms" "^7.11.0"
+      "@babel/helpers" "^7.10.4"
+      "@babel/parser" "^7.11.5"
+      "@babel/template" "^7.10.4"
+      "@babel/traverse" "^7.11.5"
+      "@babel/types" "^7.11.5"
+      convert-source-map "^1.7.0"
+      debug "^4.1.0"
+      gensync "^1.0.0-beta.1"
+      json5 "^2.1.2"
+      lodash "^4.17.19"
+      resolve "^1.3.2"
+      semver "^5.4.1"
+      source-map "^0.5.0"
+  
+  "@babel/generator@^7.11.5", "@babel/generator@^7.11.6":
+    version "7.11.6"
+    resolved "https://registry.yarnpkg.com/@babel/generator/-/generator-7.11.6.tgz#b868900f81b163b4d464ea24545c61cbac4dc620"
+    integrity sha512-DWtQ1PV3r+cLbySoHrwn9RWEgKMBLLma4OBQloPRyDYvc5msJM9kvTLo1YnlJd1P/ZuKbdli3ijr5q3FvAF3uA==
+    dependencies:
+      "@babel/types" "^7.11.5"
+      jsesc "^2.5.1"
+      source-map "^0.5.0"
+  
+  "@babel/helper-annotate-as-pure@^7.10.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/helper-annotate-as-pure/-/helper-annotate-as-pure-7.10.4.tgz#5bf0d495a3f757ac3bda48b5bf3b3ba309c72ba3"
+    integrity sha512-XQlqKQP4vXFB7BN8fEEerrmYvHp3fK/rBkRFz9jaJbzK0B1DSfej9Kc7ZzE8Z/OnId1jpJdNAZ3BFQjWG68rcA==
+    dependencies:
+      "@babel/types" "^7.10.4"
+  
+  "@babel/helper-builder-binary-assignment-operator-visitor@^7.10.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/helper-builder-binary-assignment-operator-visitor/-/helper-builder-binary-assignment-operator-visitor-7.10.4.tgz#bb0b75f31bf98cbf9ff143c1ae578b87274ae1a3"
+    integrity sha512-L0zGlFrGWZK4PbT8AszSfLTM5sDU1+Az/En9VrdT8/LmEiJt4zXt+Jve9DCAnQcbqDhCI+29y/L93mrDzddCcg==
+    dependencies:
+      "@babel/helper-explode-assignable-expression" "^7.10.4"
+      "@babel/types" "^7.10.4"
+  
+  "@babel/helper-compilation-targets@^7.10.4", "@babel/helper-compilation-targets@^7.9.6":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/helper-compilation-targets/-/helper-compilation-targets-7.10.4.tgz#804ae8e3f04376607cc791b9d47d540276332bd2"
+    integrity sha512-a3rYhlsGV0UHNDvrtOXBg8/OpfV0OKTkxKPzIplS1zpx7CygDcWWxckxZeDd3gzPzC4kUT0A4nVFDK0wGMh4MQ==
+    dependencies:
+      "@babel/compat-data" "^7.10.4"
+      browserslist "^4.12.0"
+      invariant "^2.2.4"
+      levenary "^1.1.1"
+      semver "^5.5.0"
+  
+  "@babel/helper-create-class-features-plugin@^7.10.4", "@babel/helper-create-class-features-plugin@^7.10.5":
+    version "7.10.5"
+    resolved "https://registry.yarnpkg.com/@babel/helper-create-class-features-plugin/-/helper-create-class-features-plugin-7.10.5.tgz#9f61446ba80e8240b0a5c85c6fdac8459d6f259d"
+    integrity sha512-0nkdeijB7VlZoLT3r/mY3bUkw3T8WG/hNw+FATs/6+pG2039IJWjTYL0VTISqsNHMUTEnwbVnc89WIJX9Qed0A==
+    dependencies:
+      "@babel/helper-function-name" "^7.10.4"
+      "@babel/helper-member-expression-to-functions" "^7.10.5"
+      "@babel/helper-optimise-call-expression" "^7.10.4"
+      "@babel/helper-plugin-utils" "^7.10.4"
+      "@babel/helper-replace-supers" "^7.10.4"
+      "@babel/helper-split-export-declaration" "^7.10.4"
+  
+  "@babel/helper-create-regexp-features-plugin@^7.10.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/helper-create-regexp-features-plugin/-/helper-create-regexp-features-plugin-7.10.4.tgz#fdd60d88524659a0b6959c0579925e425714f3b8"
+    integrity sha512-2/hu58IEPKeoLF45DBwx3XFqsbCXmkdAay4spVr2x0jYgRxrSNp+ePwvSsy9g6YSaNDcKIQVPXk1Ov8S2edk2g==
+    dependencies:
+      "@babel/helper-annotate-as-pure" "^7.10.4"
+      "@babel/helper-regex" "^7.10.4"
+      regexpu-core "^4.7.0"
+  
+  "@babel/helper-define-map@^7.10.4":
+    version "7.10.5"
+    resolved "https://registry.yarnpkg.com/@babel/helper-define-map/-/helper-define-map-7.10.5.tgz#b53c10db78a640800152692b13393147acb9bb30"
+    integrity sha512-fMw4kgFB720aQFXSVaXr79pjjcW5puTCM16+rECJ/plGS+zByelE8l9nCpV1GibxTnFVmUuYG9U8wYfQHdzOEQ==
+    dependencies:
+      "@babel/helper-function-name" "^7.10.4"
+      "@babel/types" "^7.10.5"
+      lodash "^4.17.19"
+  
+  "@babel/helper-explode-assignable-expression@^7.10.4":
+    version "7.11.4"
+    resolved "https://registry.yarnpkg.com/@babel/helper-explode-assignable-expression/-/helper-explode-assignable-expression-7.11.4.tgz#2d8e3470252cc17aba917ede7803d4a7a276a41b"
+    integrity sha512-ux9hm3zR4WV1Y3xXxXkdG/0gxF9nvI0YVmKVhvK9AfMoaQkemL3sJpXw+Xbz65azo8qJiEz2XVDUpK3KYhH3ZQ==
+    dependencies:
+      "@babel/types" "^7.10.4"
+  
+  "@babel/helper-function-name@^7.10.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/helper-function-name/-/helper-function-name-7.10.4.tgz#d2d3b20c59ad8c47112fa7d2a94bc09d5ef82f1a"
+    integrity sha512-YdaSyz1n8gY44EmN7x44zBn9zQ1Ry2Y+3GTA+3vH6Mizke1Vw0aWDM66FOYEPw8//qKkmqOckrGgTYa+6sceqQ==
+    dependencies:
+      "@babel/helper-get-function-arity" "^7.10.4"
+      "@babel/template" "^7.10.4"
+      "@babel/types" "^7.10.4"
+  
+  "@babel/helper-get-function-arity@^7.10.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/helper-get-function-arity/-/helper-get-function-arity-7.10.4.tgz#98c1cbea0e2332f33f9a4661b8ce1505b2c19ba2"
+    integrity sha512-EkN3YDB+SRDgiIUnNgcmiD361ti+AVbL3f3Henf6dqqUyr5dMsorno0lJWJuLhDhkI5sYEpgj6y9kB8AOU1I2A==
+    dependencies:
+      "@babel/types" "^7.10.4"
+  
+  "@babel/helper-hoist-variables@^7.10.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/helper-hoist-variables/-/helper-hoist-variables-7.10.4.tgz#d49b001d1d5a68ca5e6604dda01a6297f7c9381e"
+    integrity sha512-wljroF5PgCk2juF69kanHVs6vrLwIPNp6DLD+Lrl3hoQ3PpPPikaDRNFA+0t81NOoMt2DL6WW/mdU8k4k6ZzuA==
+    dependencies:
+      "@babel/types" "^7.10.4"
+  
+  "@babel/helper-member-expression-to-functions@^7.10.4", "@babel/helper-member-expression-to-functions@^7.10.5":
+    version "7.11.0"
+    resolved "https://registry.yarnpkg.com/@babel/helper-member-expression-to-functions/-/helper-member-expression-to-functions-7.11.0.tgz#ae69c83d84ee82f4b42f96e2a09410935a8f26df"
+    integrity sha512-JbFlKHFntRV5qKw3YC0CvQnDZ4XMwgzzBbld7Ly4Mj4cbFy3KywcR8NtNctRToMWJOVvLINJv525Gd6wwVEx/Q==
+    dependencies:
+      "@babel/types" "^7.11.0"
+  
+  "@babel/helper-module-imports@^7.0.0", "@babel/helper-module-imports@^7.10.4", "@babel/helper-module-imports@^7.8.3":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/helper-module-imports/-/helper-module-imports-7.10.4.tgz#4c5c54be04bd31670a7382797d75b9fa2e5b5620"
+    integrity sha512-nEQJHqYavI217oD9+s5MUBzk6x1IlvoS9WTPfgG43CbMEeStE0v+r+TucWdx8KFGowPGvyOkDT9+7DHedIDnVw==
+    dependencies:
+      "@babel/types" "^7.10.4"
+  
+  "@babel/helper-module-transforms@^7.10.4", "@babel/helper-module-transforms@^7.10.5", "@babel/helper-module-transforms@^7.11.0":
+    version "7.11.0"
+    resolved "https://registry.yarnpkg.com/@babel/helper-module-transforms/-/helper-module-transforms-7.11.0.tgz#b16f250229e47211abdd84b34b64737c2ab2d359"
+    integrity sha512-02EVu8COMuTRO1TAzdMtpBPbe6aQ1w/8fePD2YgQmxZU4gpNWaL9gK3Jp7dxlkUlUCJOTaSeA+Hrm1BRQwqIhg==
+    dependencies:
+      "@babel/helper-module-imports" "^7.10.4"
+      "@babel/helper-replace-supers" "^7.10.4"
+      "@babel/helper-simple-access" "^7.10.4"
+      "@babel/helper-split-export-declaration" "^7.11.0"
+      "@babel/template" "^7.10.4"
+      "@babel/types" "^7.11.0"
+      lodash "^4.17.19"
+  
+  "@babel/helper-optimise-call-expression@^7.10.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/helper-optimise-call-expression/-/helper-optimise-call-expression-7.10.4.tgz#50dc96413d594f995a77905905b05893cd779673"
+    integrity sha512-n3UGKY4VXwXThEiKrgRAoVPBMqeoPgHVqiHZOanAJCG9nQUL2pLRQirUzl0ioKclHGpGqRgIOkgcIJaIWLpygg==
+    dependencies:
+      "@babel/types" "^7.10.4"
+  
+  "@babel/helper-plugin-utils@^7.0.0", "@babel/helper-plugin-utils@^7.10.4", "@babel/helper-plugin-utils@^7.8.0", "@babel/helper-plugin-utils@^7.8.3":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/helper-plugin-utils/-/helper-plugin-utils-7.10.4.tgz#2f75a831269d4f677de49986dff59927533cf375"
+    integrity sha512-O4KCvQA6lLiMU9l2eawBPMf1xPP8xPfB3iEQw150hOVTqj/rfXz0ThTb4HEzqQfs2Bmo5Ay8BzxfzVtBrr9dVg==
+  
+  "@babel/helper-regex@^7.10.4":
+    version "7.10.5"
+    resolved "https://registry.yarnpkg.com/@babel/helper-regex/-/helper-regex-7.10.5.tgz#32dfbb79899073c415557053a19bd055aae50ae0"
+    integrity sha512-68kdUAzDrljqBrio7DYAEgCoJHxppJOERHOgOrDN7WjOzP0ZQ1LsSDRXcemzVZaLvjaJsJEESb6qt+znNuENDg==
+    dependencies:
+      lodash "^4.17.19"
+  
+  "@babel/helper-remap-async-to-generator@^7.10.4":
+    version "7.11.4"
+    resolved "https://registry.yarnpkg.com/@babel/helper-remap-async-to-generator/-/helper-remap-async-to-generator-7.11.4.tgz#4474ea9f7438f18575e30b0cac784045b402a12d"
+    integrity sha512-tR5vJ/vBa9wFy3m5LLv2faapJLnDFxNWff2SAYkSE4rLUdbp7CdObYFgI7wK4T/Mj4UzpjPwzR8Pzmr5m7MHGA==
+    dependencies:
+      "@babel/helper-annotate-as-pure" "^7.10.4"
+      "@babel/helper-wrap-function" "^7.10.4"
+      "@babel/template" "^7.10.4"
+      "@babel/types" "^7.10.4"
+  
+  "@babel/helper-replace-supers@^7.10.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/helper-replace-supers/-/helper-replace-supers-7.10.4.tgz#d585cd9388ea06e6031e4cd44b6713cbead9e6cf"
+    integrity sha512-sPxZfFXocEymYTdVK1UNmFPBN+Hv5mJkLPsYWwGBxZAxaWfFu+xqp7b6qWD0yjNuNL2VKc6L5M18tOXUP7NU0A==
+    dependencies:
+      "@babel/helper-member-expression-to-functions" "^7.10.4"
+      "@babel/helper-optimise-call-expression" "^7.10.4"
+      "@babel/traverse" "^7.10.4"
+      "@babel/types" "^7.10.4"
+  
+  "@babel/helper-simple-access@^7.10.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/helper-simple-access/-/helper-simple-access-7.10.4.tgz#0f5ccda2945277a2a7a2d3a821e15395edcf3461"
+    integrity sha512-0fMy72ej/VEvF8ULmX6yb5MtHG4uH4Dbd6I/aHDb/JVg0bbivwt9Wg+h3uMvX+QSFtwr5MeItvazbrc4jtRAXw==
+    dependencies:
+      "@babel/template" "^7.10.4"
+      "@babel/types" "^7.10.4"
+  
+  "@babel/helper-skip-transparent-expression-wrappers@^7.11.0":
+    version "7.11.0"
+    resolved "https://registry.yarnpkg.com/@babel/helper-skip-transparent-expression-wrappers/-/helper-skip-transparent-expression-wrappers-7.11.0.tgz#eec162f112c2f58d3af0af125e3bb57665146729"
+    integrity sha512-0XIdiQln4Elglgjbwo9wuJpL/K7AGCY26kmEt0+pRP0TAj4jjyNq1MjoRvikrTVqKcx4Gysxt4cXvVFXP/JO2Q==
+    dependencies:
+      "@babel/types" "^7.11.0"
+  
+  "@babel/helper-split-export-declaration@^7.10.4", "@babel/helper-split-export-declaration@^7.11.0":
+    version "7.11.0"
+    resolved "https://registry.yarnpkg.com/@babel/helper-split-export-declaration/-/helper-split-export-declaration-7.11.0.tgz#f8a491244acf6a676158ac42072911ba83ad099f"
+    integrity sha512-74Vejvp6mHkGE+m+k5vHY93FX2cAtrw1zXrZXRlG4l410Nm9PxfEiVTn1PjDPV5SnmieiueY4AFg2xqhNFuuZg==
+    dependencies:
+      "@babel/types" "^7.11.0"
+  
+  "@babel/helper-validator-identifier@^7.10.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/helper-validator-identifier/-/helper-validator-identifier-7.10.4.tgz#a78c7a7251e01f616512d31b10adcf52ada5e0d2"
+    integrity sha512-3U9y+43hz7ZM+rzG24Qe2mufW5KhvFg/NhnNph+i9mgCtdTCtMJuI1TMkrIUiK7Ix4PYlRF9I5dhqaLYA/ADXw==
+  
+  "@babel/helper-wrap-function@^7.10.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/helper-wrap-function/-/helper-wrap-function-7.10.4.tgz#8a6f701eab0ff39f765b5a1cfef409990e624b87"
+    integrity sha512-6py45WvEF0MhiLrdxtRjKjufwLL1/ob2qDJgg5JgNdojBAZSAKnAjkyOCNug6n+OBl4VW76XjvgSFTdaMcW0Ug==
+    dependencies:
+      "@babel/helper-function-name" "^7.10.4"
+      "@babel/template" "^7.10.4"
+      "@babel/traverse" "^7.10.4"
+      "@babel/types" "^7.10.4"
+  
+  "@babel/helpers@^7.10.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/helpers/-/helpers-7.10.4.tgz#2abeb0d721aff7c0a97376b9e1f6f65d7a475044"
+    integrity sha512-L2gX/XeUONeEbI78dXSrJzGdz4GQ+ZTA/aazfUsFaWjSe95kiCuOZ5HsXvkiw3iwF+mFHSRUfJU8t6YavocdXA==
+    dependencies:
+      "@babel/template" "^7.10.4"
+      "@babel/traverse" "^7.10.4"
+      "@babel/types" "^7.10.4"
+  
+  "@babel/highlight@^7.10.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/highlight/-/highlight-7.10.4.tgz#7d1bdfd65753538fabe6c38596cdb76d9ac60143"
+    integrity sha512-i6rgnR/YgPEQzZZnbTHHuZdlE8qyoBNalD6F+q4vAFlcMEcqmkoG+mPqJYJCo63qPf74+Y1UZsl3l6f7/RIkmA==
+    dependencies:
+      "@babel/helper-validator-identifier" "^7.10.4"
+      chalk "^2.0.0"
+      js-tokens "^4.0.0"
+  
+  "@babel/parser@^7.10.4", "@babel/parser@^7.11.5", "@babel/parser@^7.7.0":
+    version "7.11.5"
+    resolved "https://registry.yarnpkg.com/@babel/parser/-/parser-7.11.5.tgz#c7ff6303df71080ec7a4f5b8c003c58f1cf51037"
+    integrity sha512-X9rD8qqm695vgmeaQ4fvz/o3+Wk4ZzQvSHkDBgpYKxpD4qTAUm88ZKtHkVqIOsYFFbIQ6wQYhC6q7pjqVK0E0Q==
+  
+  "@babel/plugin-proposal-async-generator-functions@^7.10.4":
+    version "7.10.5"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-proposal-async-generator-functions/-/plugin-proposal-async-generator-functions-7.10.5.tgz#3491cabf2f7c179ab820606cec27fed15e0e8558"
+    integrity sha512-cNMCVezQbrRGvXJwm9fu/1sJj9bHdGAgKodZdLqOQIpfoH3raqmRPBM17+lh7CzhiKRRBrGtZL9WcjxSoGYUSg==
+    dependencies:
+      "@babel/helper-plugin-utils" "^7.10.4"
+      "@babel/helper-remap-async-to-generator" "^7.10.4"
+      "@babel/plugin-syntax-async-generators" "^7.8.0"
+  
+  "@babel/plugin-proposal-class-properties@^7.10.4", "@babel/plugin-proposal-class-properties@^7.8.3":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-proposal-class-properties/-/plugin-proposal-class-properties-7.10.4.tgz#a33bf632da390a59c7a8c570045d1115cd778807"
+    integrity sha512-vhwkEROxzcHGNu2mzUC0OFFNXdZ4M23ib8aRRcJSsW8BZK9pQMD7QB7csl97NBbgGZO7ZyHUyKDnxzOaP4IrCg==
+    dependencies:
+      "@babel/helper-create-class-features-plugin" "^7.10.4"
+      "@babel/helper-plugin-utils" "^7.10.4"
+  
+  "@babel/plugin-proposal-decorators@^7.8.3":
+    version "7.10.5"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-proposal-decorators/-/plugin-proposal-decorators-7.10.5.tgz#42898bba478bc4b1ae242a703a953a7ad350ffb4"
+    integrity sha512-Sc5TAQSZuLzgY0664mMDn24Vw2P8g/VhyLyGPaWiHahhgLqeZvcGeyBZOrJW0oSKIK2mvQ22a1ENXBIQLhrEiQ==
+    dependencies:
+      "@babel/helper-create-class-features-plugin" "^7.10.5"
+      "@babel/helper-plugin-utils" "^7.10.4"
+      "@babel/plugin-syntax-decorators" "^7.10.4"
+  
+  "@babel/plugin-proposal-dynamic-import@^7.10.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-proposal-dynamic-import/-/plugin-proposal-dynamic-import-7.10.4.tgz#ba57a26cb98b37741e9d5bca1b8b0ddf8291f17e"
+    integrity sha512-up6oID1LeidOOASNXgv/CFbgBqTuKJ0cJjz6An5tWD+NVBNlp3VNSBxv2ZdU7SYl3NxJC7agAQDApZusV6uFwQ==
+    dependencies:
+      "@babel/helper-plugin-utils" "^7.10.4"
+      "@babel/plugin-syntax-dynamic-import" "^7.8.0"
+  
+  "@babel/plugin-proposal-export-namespace-from@^7.10.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-proposal-export-namespace-from/-/plugin-proposal-export-namespace-from-7.10.4.tgz#570d883b91031637b3e2958eea3c438e62c05f54"
+    integrity sha512-aNdf0LY6/3WXkhh0Fdb6Zk9j1NMD8ovj3F6r0+3j837Pn1S1PdNtcwJ5EG9WkVPNHPxyJDaxMaAOVq4eki0qbg==
+    dependencies:
+      "@babel/helper-plugin-utils" "^7.10.4"
+      "@babel/plugin-syntax-export-namespace-from" "^7.8.3"
+  
+  "@babel/plugin-proposal-json-strings@^7.10.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-proposal-json-strings/-/plugin-proposal-json-strings-7.10.4.tgz#593e59c63528160233bd321b1aebe0820c2341db"
+    integrity sha512-fCL7QF0Jo83uy1K0P2YXrfX11tj3lkpN7l4dMv9Y9VkowkhkQDwFHFd8IiwyK5MZjE8UpbgokkgtcReH88Abaw==
+    dependencies:
+      "@babel/helper-plugin-utils" "^7.10.4"
+      "@babel/plugin-syntax-json-strings" "^7.8.0"
+  
+  "@babel/plugin-proposal-logical-assignment-operators@^7.11.0":
+    version "7.11.0"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-proposal-logical-assignment-operators/-/plugin-proposal-logical-assignment-operators-7.11.0.tgz#9f80e482c03083c87125dee10026b58527ea20c8"
+    integrity sha512-/f8p4z+Auz0Uaf+i8Ekf1iM7wUNLcViFUGiPxKeXvxTSl63B875YPiVdUDdem7hREcI0E0kSpEhS8tF5RphK7Q==
+    dependencies:
+      "@babel/helper-plugin-utils" "^7.10.4"
+      "@babel/plugin-syntax-logical-assignment-operators" "^7.10.4"
+  
+  "@babel/plugin-proposal-nullish-coalescing-operator@^7.10.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-proposal-nullish-coalescing-operator/-/plugin-proposal-nullish-coalescing-operator-7.10.4.tgz#02a7e961fc32e6d5b2db0649e01bf80ddee7e04a"
+    integrity sha512-wq5n1M3ZUlHl9sqT2ok1T2/MTt6AXE0e1Lz4WzWBr95LsAZ5qDXe4KnFuauYyEyLiohvXFMdbsOTMyLZs91Zlw==
+    dependencies:
+      "@babel/helper-plugin-utils" "^7.10.4"
+      "@babel/plugin-syntax-nullish-coalescing-operator" "^7.8.0"
+  
+  "@babel/plugin-proposal-numeric-separator@^7.10.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-proposal-numeric-separator/-/plugin-proposal-numeric-separator-7.10.4.tgz#ce1590ff0a65ad12970a609d78855e9a4c1aef06"
+    integrity sha512-73/G7QoRoeNkLZFxsoCCvlg4ezE4eM+57PnOqgaPOozd5myfj7p0muD1mRVJvbUWbOzD+q3No2bWbaKy+DJ8DA==
+    dependencies:
+      "@babel/helper-plugin-utils" "^7.10.4"
+      "@babel/plugin-syntax-numeric-separator" "^7.10.4"
+  
+  "@babel/plugin-proposal-object-rest-spread@^7.11.0":
+    version "7.11.0"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-proposal-object-rest-spread/-/plugin-proposal-object-rest-spread-7.11.0.tgz#bd81f95a1f746760ea43b6c2d3d62b11790ad0af"
+    integrity sha512-wzch41N4yztwoRw0ak+37wxwJM2oiIiy6huGCoqkvSTA9acYWcPfn9Y4aJqmFFJ70KTJUu29f3DQ43uJ9HXzEA==
+    dependencies:
+      "@babel/helper-plugin-utils" "^7.10.4"
+      "@babel/plugin-syntax-object-rest-spread" "^7.8.0"
+      "@babel/plugin-transform-parameters" "^7.10.4"
+  
+  "@babel/plugin-proposal-optional-catch-binding@^7.10.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-proposal-optional-catch-binding/-/plugin-proposal-optional-catch-binding-7.10.4.tgz#31c938309d24a78a49d68fdabffaa863758554dd"
+    integrity sha512-LflT6nPh+GK2MnFiKDyLiqSqVHkQnVf7hdoAvyTnnKj9xB3docGRsdPuxp6qqqW19ifK3xgc9U5/FwrSaCNX5g==
+    dependencies:
+      "@babel/helper-plugin-utils" "^7.10.4"
+      "@babel/plugin-syntax-optional-catch-binding" "^7.8.0"
+  
+  "@babel/plugin-proposal-optional-chaining@^7.11.0":
+    version "7.11.0"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-proposal-optional-chaining/-/plugin-proposal-optional-chaining-7.11.0.tgz#de5866d0646f6afdaab8a566382fe3a221755076"
+    integrity sha512-v9fZIu3Y8562RRwhm1BbMRxtqZNFmFA2EG+pT2diuU8PT3H6T/KXoZ54KgYisfOFZHV6PfvAiBIZ9Rcz+/JCxA==
+    dependencies:
+      "@babel/helper-plugin-utils" "^7.10.4"
+      "@babel/helper-skip-transparent-expression-wrappers" "^7.11.0"
+      "@babel/plugin-syntax-optional-chaining" "^7.8.0"
+  
+  "@babel/plugin-proposal-private-methods@^7.10.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-proposal-private-methods/-/plugin-proposal-private-methods-7.10.4.tgz#b160d972b8fdba5c7d111a145fc8c421fc2a6909"
+    integrity sha512-wh5GJleuI8k3emgTg5KkJK6kHNsGEr0uBTDBuQUBJwckk9xs1ez79ioheEVVxMLyPscB0LfkbVHslQqIzWV6Bw==
+    dependencies:
+      "@babel/helper-create-class-features-plugin" "^7.10.4"
+      "@babel/helper-plugin-utils" "^7.10.4"
+  
+  "@babel/plugin-proposal-unicode-property-regex@^7.10.4", "@babel/plugin-proposal-unicode-property-regex@^7.4.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-proposal-unicode-property-regex/-/plugin-proposal-unicode-property-regex-7.10.4.tgz#4483cda53041ce3413b7fe2f00022665ddfaa75d"
+    integrity sha512-H+3fOgPnEXFL9zGYtKQe4IDOPKYlZdF1kqFDQRRb8PK4B8af1vAGK04tF5iQAAsui+mHNBQSAtd2/ndEDe9wuA==
+    dependencies:
+      "@babel/helper-create-regexp-features-plugin" "^7.10.4"
+      "@babel/helper-plugin-utils" "^7.10.4"
+  
+  "@babel/plugin-syntax-async-generators@^7.8.0":
+    version "7.8.4"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-syntax-async-generators/-/plugin-syntax-async-generators-7.8.4.tgz#a983fb1aeb2ec3f6ed042a210f640e90e786fe0d"
+    integrity sha512-tycmZxkGfZaxhMRbXlPXuVFpdWlXpir2W4AMhSJgRKzk/eDlIXOhb2LHWoLpDF7TEHylV5zNhykX6KAgHJmTNw==
+    dependencies:
+      "@babel/helper-plugin-utils" "^7.8.0"
+  
+  "@babel/plugin-syntax-class-properties@^7.10.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-syntax-class-properties/-/plugin-syntax-class-properties-7.10.4.tgz#6644e6a0baa55a61f9e3231f6c9eeb6ee46c124c"
+    integrity sha512-GCSBF7iUle6rNugfURwNmCGG3Z/2+opxAMLs1nND4bhEG5PuxTIggDBoeYYSujAlLtsupzOHYJQgPS3pivwXIA==
+    dependencies:
+      "@babel/helper-plugin-utils" "^7.10.4"
+  
+  "@babel/plugin-syntax-decorators@^7.10.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-syntax-decorators/-/plugin-syntax-decorators-7.10.4.tgz#6853085b2c429f9d322d02f5a635018cdeb2360c"
+    integrity sha512-2NaoC6fAk2VMdhY1eerkfHV+lVYC1u8b+jmRJISqANCJlTxYy19HGdIkkQtix2UtkcPuPu+IlDgrVseZnU03bw==
+    dependencies:
+      "@babel/helper-plugin-utils" "^7.10.4"
+  
+  "@babel/plugin-syntax-dynamic-import@^7.8.0", "@babel/plugin-syntax-dynamic-import@^7.8.3":
+    version "7.8.3"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-syntax-dynamic-import/-/plugin-syntax-dynamic-import-7.8.3.tgz#62bf98b2da3cd21d626154fc96ee5b3cb68eacb3"
+    integrity sha512-5gdGbFon+PszYzqs83S3E5mpi7/y/8M9eC90MRTZfduQOYW76ig6SOSPNe41IG5LoP3FGBn2N0RjVDSQiS94kQ==
+    dependencies:
+      "@babel/helper-plugin-utils" "^7.8.0"
+  
+  "@babel/plugin-syntax-export-namespace-from@^7.8.3":
+    version "7.8.3"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-syntax-export-namespace-from/-/plugin-syntax-export-namespace-from-7.8.3.tgz#028964a9ba80dbc094c915c487ad7c4e7a66465a"
+    integrity sha512-MXf5laXo6c1IbEbegDmzGPwGNTsHZmEy6QGznu5Sh2UCWvueywb2ee+CCE4zQiZstxU9BMoQO9i6zUFSY0Kj0Q==
+    dependencies:
+      "@babel/helper-plugin-utils" "^7.8.3"
+  
+  "@babel/plugin-syntax-json-strings@^7.8.0":
+    version "7.8.3"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-syntax-json-strings/-/plugin-syntax-json-strings-7.8.3.tgz#01ca21b668cd8218c9e640cb6dd88c5412b2c96a"
+    integrity sha512-lY6kdGpWHvjoe2vk4WrAapEuBR69EMxZl+RoGRhrFGNYVK8mOPAW8VfbT/ZgrFbXlDNiiaxQnAtgVCZ6jv30EA==
+    dependencies:
+      "@babel/helper-plugin-utils" "^7.8.0"
+  
+  "@babel/plugin-syntax-jsx@^7.0.0", "@babel/plugin-syntax-jsx@^7.2.0", "@babel/plugin-syntax-jsx@^7.8.3":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-syntax-jsx/-/plugin-syntax-jsx-7.10.4.tgz#39abaae3cbf710c4373d8429484e6ba21340166c"
+    integrity sha512-KCg9mio9jwiARCB7WAcQ7Y1q+qicILjoK8LP/VkPkEKaf5dkaZZK1EcTe91a3JJlZ3qy6L5s9X52boEYi8DM9g==
+    dependencies:
+      "@babel/helper-plugin-utils" "^7.10.4"
+  
+  "@babel/plugin-syntax-logical-assignment-operators@^7.10.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-syntax-logical-assignment-operators/-/plugin-syntax-logical-assignment-operators-7.10.4.tgz#ca91ef46303530448b906652bac2e9fe9941f699"
+    integrity sha512-d8waShlpFDinQ5MtvGU9xDAOzKH47+FFoney2baFIoMr952hKOLp1HR7VszoZvOsV/4+RRszNY7D17ba0te0ig==
+    dependencies:
+      "@babel/helper-plugin-utils" "^7.10.4"
+  
+  "@babel/plugin-syntax-nullish-coalescing-operator@^7.8.0":
+    version "7.8.3"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-syntax-nullish-coalescing-operator/-/plugin-syntax-nullish-coalescing-operator-7.8.3.tgz#167ed70368886081f74b5c36c65a88c03b66d1a9"
+    integrity sha512-aSff4zPII1u2QD7y+F8oDsz19ew4IGEJg9SVW+bqwpwtfFleiQDMdzA/R+UlWDzfnHFCxxleFT0PMIrR36XLNQ==
+    dependencies:
+      "@babel/helper-plugin-utils" "^7.8.0"
+  
+  "@babel/plugin-syntax-numeric-separator@^7.10.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-syntax-numeric-separator/-/plugin-syntax-numeric-separator-7.10.4.tgz#b9b070b3e33570cd9fd07ba7fa91c0dd37b9af97"
+    integrity sha512-9H6YdfkcK/uOnY/K7/aA2xpzaAgkQn37yzWUMRK7OaPOqOpGS1+n0H5hxT9AUw9EsSjPW8SVyMJwYRtWs3X3ug==
+    dependencies:
+      "@babel/helper-plugin-utils" "^7.10.4"
+  
+  "@babel/plugin-syntax-object-rest-spread@^7.8.0":
+    version "7.8.3"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-syntax-object-rest-spread/-/plugin-syntax-object-rest-spread-7.8.3.tgz#60e225edcbd98a640332a2e72dd3e66f1af55871"
+    integrity sha512-XoqMijGZb9y3y2XskN+P1wUGiVwWZ5JmoDRwx5+3GmEplNyVM2s2Dg8ILFQm8rWM48orGy5YpI5Bl8U1y7ydlA==
+    dependencies:
+      "@babel/helper-plugin-utils" "^7.8.0"
+  
+  "@babel/plugin-syntax-optional-catch-binding@^7.8.0":
+    version "7.8.3"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-syntax-optional-catch-binding/-/plugin-syntax-optional-catch-binding-7.8.3.tgz#6111a265bcfb020eb9efd0fdfd7d26402b9ed6c1"
+    integrity sha512-6VPD0Pc1lpTqw0aKoeRTMiB+kWhAoT24PA+ksWSBrFtl5SIRVpZlwN3NNPQjehA2E/91FV3RjLWoVTglWcSV3Q==
+    dependencies:
+      "@babel/helper-plugin-utils" "^7.8.0"
+  
+  "@babel/plugin-syntax-optional-chaining@^7.8.0":
+    version "7.8.3"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-syntax-optional-chaining/-/plugin-syntax-optional-chaining-7.8.3.tgz#4f69c2ab95167e0180cd5336613f8c5788f7d48a"
+    integrity sha512-KoK9ErH1MBlCPxV0VANkXW2/dw4vlbGDrFgz8bmUsBGYkFRcbRwMh6cIJubdPrkxRwuGdtCk0v/wPTKbQgBjkg==
+    dependencies:
+      "@babel/helper-plugin-utils" "^7.8.0"
+  
+  "@babel/plugin-syntax-top-level-await@^7.10.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-syntax-top-level-await/-/plugin-syntax-top-level-await-7.10.4.tgz#4bbeb8917b54fcf768364e0a81f560e33a3ef57d"
+    integrity sha512-ni1brg4lXEmWyafKr0ccFWkJG0CeMt4WV1oyeBW6EFObF4oOHclbkj5cARxAPQyAQ2UTuplJyK4nfkXIMMFvsQ==
+    dependencies:
+      "@babel/helper-plugin-utils" "^7.10.4"
+  
+  "@babel/plugin-transform-arrow-functions@^7.10.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-transform-arrow-functions/-/plugin-transform-arrow-functions-7.10.4.tgz#e22960d77e697c74f41c501d44d73dbf8a6a64cd"
+    integrity sha512-9J/oD1jV0ZCBcgnoFWFq1vJd4msoKb/TCpGNFyyLt0zABdcvgK3aYikZ8HjzB14c26bc7E3Q1yugpwGy2aTPNA==
+    dependencies:
+      "@babel/helper-plugin-utils" "^7.10.4"
+  
+  "@babel/plugin-transform-async-to-generator@^7.10.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-transform-async-to-generator/-/plugin-transform-async-to-generator-7.10.4.tgz#41a5017e49eb6f3cda9392a51eef29405b245a37"
+    integrity sha512-F6nREOan7J5UXTLsDsZG3DXmZSVofr2tGNwfdrVwkDWHfQckbQXnXSPfD7iO+c/2HGqycwyLST3DnZ16n+cBJQ==
+    dependencies:
+      "@babel/helper-module-imports" "^7.10.4"
+      "@babel/helper-plugin-utils" "^7.10.4"
+      "@babel/helper-remap-async-to-generator" "^7.10.4"
+  
+  "@babel/plugin-transform-block-scoped-functions@^7.10.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-transform-block-scoped-functions/-/plugin-transform-block-scoped-functions-7.10.4.tgz#1afa595744f75e43a91af73b0d998ecfe4ebc2e8"
+    integrity sha512-WzXDarQXYYfjaV1szJvN3AD7rZgZzC1JtjJZ8dMHUyiK8mxPRahynp14zzNjU3VkPqPsO38CzxiWO1c9ARZ8JA==
+    dependencies:
+      "@babel/helper-plugin-utils" "^7.10.4"
+  
+  "@babel/plugin-transform-block-scoping@^7.10.4":
+    version "7.11.1"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-transform-block-scoping/-/plugin-transform-block-scoping-7.11.1.tgz#5b7efe98852bef8d652c0b28144cd93a9e4b5215"
+    integrity sha512-00dYeDE0EVEHuuM+26+0w/SCL0BH2Qy7LwHuI4Hi4MH5gkC8/AqMN5uWFJIsoXZrAphiMm1iXzBw6L2T+eA0ew==
+    dependencies:
+      "@babel/helper-plugin-utils" "^7.10.4"
+  
+  "@babel/plugin-transform-classes@^7.10.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-transform-classes/-/plugin-transform-classes-7.10.4.tgz#405136af2b3e218bc4a1926228bc917ab1a0adc7"
+    integrity sha512-2oZ9qLjt161dn1ZE0Ms66xBncQH4In8Sqw1YWgBUZuGVJJS5c0OFZXL6dP2MRHrkU/eKhWg8CzFJhRQl50rQxA==
+    dependencies:
+      "@babel/helper-annotate-as-pure" "^7.10.4"
+      "@babel/helper-define-map" "^7.10.4"
+      "@babel/helper-function-name" "^7.10.4"
+      "@babel/helper-optimise-call-expression" "^7.10.4"
+      "@babel/helper-plugin-utils" "^7.10.4"
+      "@babel/helper-replace-supers" "^7.10.4"
+      "@babel/helper-split-export-declaration" "^7.10.4"
+      globals "^11.1.0"
+  
+  "@babel/plugin-transform-computed-properties@^7.10.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-transform-computed-properties/-/plugin-transform-computed-properties-7.10.4.tgz#9ded83a816e82ded28d52d4b4ecbdd810cdfc0eb"
+    integrity sha512-JFwVDXcP/hM/TbyzGq3l/XWGut7p46Z3QvqFMXTfk6/09m7xZHJUN9xHfsv7vqqD4YnfI5ueYdSJtXqqBLyjBw==
+    dependencies:
+      "@babel/helper-plugin-utils" "^7.10.4"
+  
+  "@babel/plugin-transform-destructuring@^7.10.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-transform-destructuring/-/plugin-transform-destructuring-7.10.4.tgz#70ddd2b3d1bea83d01509e9bb25ddb3a74fc85e5"
+    integrity sha512-+WmfvyfsyF603iPa6825mq6Qrb7uLjTOsa3XOFzlYcYDHSS4QmpOWOL0NNBY5qMbvrcf3tq0Cw+v4lxswOBpgA==
+    dependencies:
+      "@babel/helper-plugin-utils" "^7.10.4"
+  
+  "@babel/plugin-transform-dotall-regex@^7.10.4", "@babel/plugin-transform-dotall-regex@^7.4.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-transform-dotall-regex/-/plugin-transform-dotall-regex-7.10.4.tgz#469c2062105c1eb6a040eaf4fac4b488078395ee"
+    integrity sha512-ZEAVvUTCMlMFAbASYSVQoxIbHm2OkG2MseW6bV2JjIygOjdVv8tuxrCTzj1+Rynh7ODb8GivUy7dzEXzEhuPaA==
+    dependencies:
+      "@babel/helper-create-regexp-features-plugin" "^7.10.4"
+      "@babel/helper-plugin-utils" "^7.10.4"
+  
+  "@babel/plugin-transform-duplicate-keys@^7.10.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-transform-duplicate-keys/-/plugin-transform-duplicate-keys-7.10.4.tgz#697e50c9fee14380fe843d1f306b295617431e47"
+    integrity sha512-GL0/fJnmgMclHiBTTWXNlYjYsA7rDrtsazHG6mglaGSTh0KsrW04qml+Bbz9FL0LcJIRwBWL5ZqlNHKTkU3xAA==
+    dependencies:
+      "@babel/helper-plugin-utils" "^7.10.4"
+  
+  "@babel/plugin-transform-exponentiation-operator@^7.10.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-transform-exponentiation-operator/-/plugin-transform-exponentiation-operator-7.10.4.tgz#5ae338c57f8cf4001bdb35607ae66b92d665af2e"
+    integrity sha512-S5HgLVgkBcRdyQAHbKj+7KyuWx8C6t5oETmUuwz1pt3WTWJhsUV0WIIXuVvfXMxl/QQyHKlSCNNtaIamG8fysw==
+    dependencies:
+      "@babel/helper-builder-binary-assignment-operator-visitor" "^7.10.4"
+      "@babel/helper-plugin-utils" "^7.10.4"
+  
+  "@babel/plugin-transform-for-of@^7.10.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-transform-for-of/-/plugin-transform-for-of-7.10.4.tgz#c08892e8819d3a5db29031b115af511dbbfebae9"
+    integrity sha512-ItdQfAzu9AlEqmusA/65TqJ79eRcgGmpPPFvBnGILXZH975G0LNjP1yjHvGgfuCxqrPPueXOPe+FsvxmxKiHHQ==
+    dependencies:
+      "@babel/helper-plugin-utils" "^7.10.4"
+  
+  "@babel/plugin-transform-function-name@^7.10.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-transform-function-name/-/plugin-transform-function-name-7.10.4.tgz#6a467880e0fc9638514ba369111811ddbe2644b7"
+    integrity sha512-OcDCq2y5+E0dVD5MagT5X+yTRbcvFjDI2ZVAottGH6tzqjx/LKpgkUepu3hp/u4tZBzxxpNGwLsAvGBvQ2mJzg==
+    dependencies:
+      "@babel/helper-function-name" "^7.10.4"
+      "@babel/helper-plugin-utils" "^7.10.4"
+  
+  "@babel/plugin-transform-literals@^7.10.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-transform-literals/-/plugin-transform-literals-7.10.4.tgz#9f42ba0841100a135f22712d0e391c462f571f3c"
+    integrity sha512-Xd/dFSTEVuUWnyZiMu76/InZxLTYilOSr1UlHV+p115Z/Le2Fi1KXkJUYz0b42DfndostYlPub3m8ZTQlMaiqQ==
+    dependencies:
+      "@babel/helper-plugin-utils" "^7.10.4"
+  
+  "@babel/plugin-transform-member-expression-literals@^7.10.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-transform-member-expression-literals/-/plugin-transform-member-expression-literals-7.10.4.tgz#b1ec44fcf195afcb8db2c62cd8e551c881baf8b7"
+    integrity sha512-0bFOvPyAoTBhtcJLr9VcwZqKmSjFml1iVxvPL0ReomGU53CX53HsM4h2SzckNdkQcHox1bpAqzxBI1Y09LlBSw==
+    dependencies:
+      "@babel/helper-plugin-utils" "^7.10.4"
+  
+  "@babel/plugin-transform-modules-amd@^7.10.4":
+    version "7.10.5"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-transform-modules-amd/-/plugin-transform-modules-amd-7.10.5.tgz#1b9cddaf05d9e88b3aad339cb3e445c4f020a9b1"
+    integrity sha512-elm5uruNio7CTLFItVC/rIzKLfQ17+fX7EVz5W0TMgIHFo1zY0Ozzx+lgwhL4plzl8OzVn6Qasx5DeEFyoNiRw==
+    dependencies:
+      "@babel/helper-module-transforms" "^7.10.5"
+      "@babel/helper-plugin-utils" "^7.10.4"
+      babel-plugin-dynamic-import-node "^2.3.3"
+  
+  "@babel/plugin-transform-modules-commonjs@^7.10.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-transform-modules-commonjs/-/plugin-transform-modules-commonjs-7.10.4.tgz#66667c3eeda1ebf7896d41f1f16b17105a2fbca0"
+    integrity sha512-Xj7Uq5o80HDLlW64rVfDBhao6OX89HKUmb+9vWYaLXBZOma4gA6tw4Ni1O5qVDoZWUV0fxMYA0aYzOawz0l+1w==
+    dependencies:
+      "@babel/helper-module-transforms" "^7.10.4"
+      "@babel/helper-plugin-utils" "^7.10.4"
+      "@babel/helper-simple-access" "^7.10.4"
+      babel-plugin-dynamic-import-node "^2.3.3"
+  
+  "@babel/plugin-transform-modules-systemjs@^7.10.4":
+    version "7.10.5"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-transform-modules-systemjs/-/plugin-transform-modules-systemjs-7.10.5.tgz#6270099c854066681bae9e05f87e1b9cadbe8c85"
+    integrity sha512-f4RLO/OL14/FP1AEbcsWMzpbUz6tssRaeQg11RH1BP/XnPpRoVwgeYViMFacnkaw4k4wjRSjn3ip1Uw9TaXuMw==
+    dependencies:
+      "@babel/helper-hoist-variables" "^7.10.4"
+      "@babel/helper-module-transforms" "^7.10.5"
+      "@babel/helper-plugin-utils" "^7.10.4"
+      babel-plugin-dynamic-import-node "^2.3.3"
+  
+  "@babel/plugin-transform-modules-umd@^7.10.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-transform-modules-umd/-/plugin-transform-modules-umd-7.10.4.tgz#9a8481fe81b824654b3a0b65da3df89f3d21839e"
+    integrity sha512-mohW5q3uAEt8T45YT7Qc5ws6mWgJAaL/8BfWD9Dodo1A3RKWli8wTS+WiQ/knF+tXlPirW/1/MqzzGfCExKECA==
+    dependencies:
+      "@babel/helper-module-transforms" "^7.10.4"
+      "@babel/helper-plugin-utils" "^7.10.4"
+  
+  "@babel/plugin-transform-named-capturing-groups-regex@^7.10.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-transform-named-capturing-groups-regex/-/plugin-transform-named-capturing-groups-regex-7.10.4.tgz#78b4d978810b6f3bcf03f9e318f2fc0ed41aecb6"
+    integrity sha512-V6LuOnD31kTkxQPhKiVYzYC/Jgdq53irJC/xBSmqcNcqFGV+PER4l6rU5SH2Vl7bH9mLDHcc0+l9HUOe4RNGKA==
+    dependencies:
+      "@babel/helper-create-regexp-features-plugin" "^7.10.4"
+  
+  "@babel/plugin-transform-new-target@^7.10.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-transform-new-target/-/plugin-transform-new-target-7.10.4.tgz#9097d753cb7b024cb7381a3b2e52e9513a9c6888"
+    integrity sha512-YXwWUDAH/J6dlfwqlWsztI2Puz1NtUAubXhOPLQ5gjR/qmQ5U96DY4FQO8At33JN4XPBhrjB8I4eMmLROjjLjw==
+    dependencies:
+      "@babel/helper-plugin-utils" "^7.10.4"
+  
+  "@babel/plugin-transform-object-super@^7.10.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-transform-object-super/-/plugin-transform-object-super-7.10.4.tgz#d7146c4d139433e7a6526f888c667e314a093894"
+    integrity sha512-5iTw0JkdRdJvr7sY0vHqTpnruUpTea32JHmq/atIWqsnNussbRzjEDyWep8UNztt1B5IusBYg8Irb0bLbiEBCQ==
+    dependencies:
+      "@babel/helper-plugin-utils" "^7.10.4"
+      "@babel/helper-replace-supers" "^7.10.4"
+  
+  "@babel/plugin-transform-parameters@^7.10.4":
+    version "7.10.5"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-transform-parameters/-/plugin-transform-parameters-7.10.5.tgz#59d339d58d0b1950435f4043e74e2510005e2c4a"
+    integrity sha512-xPHwUj5RdFV8l1wuYiu5S9fqWGM2DrYc24TMvUiRrPVm+SM3XeqU9BcokQX/kEUe+p2RBwy+yoiR1w/Blq6ubw==
+    dependencies:
+      "@babel/helper-get-function-arity" "^7.10.4"
+      "@babel/helper-plugin-utils" "^7.10.4"
+  
+  "@babel/plugin-transform-property-literals@^7.10.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-transform-property-literals/-/plugin-transform-property-literals-7.10.4.tgz#f6fe54b6590352298785b83edd815d214c42e3c0"
+    integrity sha512-ofsAcKiUxQ8TY4sScgsGeR2vJIsfrzqvFb9GvJ5UdXDzl+MyYCaBj/FGzXuv7qE0aJcjWMILny1epqelnFlz8g==
+    dependencies:
+      "@babel/helper-plugin-utils" "^7.10.4"
+  
+  "@babel/plugin-transform-regenerator@^7.10.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-transform-regenerator/-/plugin-transform-regenerator-7.10.4.tgz#2015e59d839074e76838de2159db421966fd8b63"
+    integrity sha512-3thAHwtor39A7C04XucbMg17RcZ3Qppfxr22wYzZNcVIkPHfpM9J0SO8zuCV6SZa265kxBJSrfKTvDCYqBFXGw==
+    dependencies:
+      regenerator-transform "^0.14.2"
+  
+  "@babel/plugin-transform-reserved-words@^7.10.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-transform-reserved-words/-/plugin-transform-reserved-words-7.10.4.tgz#8f2682bcdcef9ed327e1b0861585d7013f8a54dd"
+    integrity sha512-hGsw1O6Rew1fkFbDImZIEqA8GoidwTAilwCyWqLBM9f+e/u/sQMQu7uX6dyokfOayRuuVfKOW4O7HvaBWM+JlQ==
+    dependencies:
+      "@babel/helper-plugin-utils" "^7.10.4"
+  
+  "@babel/plugin-transform-runtime@^7.11.0":
+    version "7.11.5"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-transform-runtime/-/plugin-transform-runtime-7.11.5.tgz#f108bc8e0cf33c37da031c097d1df470b3a293fc"
+    integrity sha512-9aIoee+EhjySZ6vY5hnLjigHzunBlscx9ANKutkeWTJTx6m5Rbq6Ic01tLvO54lSusR+BxV7u4UDdCmXv5aagg==
+    dependencies:
+      "@babel/helper-module-imports" "^7.10.4"
+      "@babel/helper-plugin-utils" "^7.10.4"
+      resolve "^1.8.1"
+      semver "^5.5.1"
+  
+  "@babel/plugin-transform-shorthand-properties@^7.10.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-transform-shorthand-properties/-/plugin-transform-shorthand-properties-7.10.4.tgz#9fd25ec5cdd555bb7f473e5e6ee1c971eede4dd6"
+    integrity sha512-AC2K/t7o07KeTIxMoHneyX90v3zkm5cjHJEokrPEAGEy3UCp8sLKfnfOIGdZ194fyN4wfX/zZUWT9trJZ0qc+Q==
+    dependencies:
+      "@babel/helper-plugin-utils" "^7.10.4"
+  
+  "@babel/plugin-transform-spread@^7.11.0":
+    version "7.11.0"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-transform-spread/-/plugin-transform-spread-7.11.0.tgz#fa84d300f5e4f57752fe41a6d1b3c554f13f17cc"
+    integrity sha512-UwQYGOqIdQJe4aWNyS7noqAnN2VbaczPLiEtln+zPowRNlD+79w3oi2TWfYe0eZgd+gjZCbsydN7lzWysDt+gw==
+    dependencies:
+      "@babel/helper-plugin-utils" "^7.10.4"
+      "@babel/helper-skip-transparent-expression-wrappers" "^7.11.0"
+  
+  "@babel/plugin-transform-sticky-regex@^7.10.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-transform-sticky-regex/-/plugin-transform-sticky-regex-7.10.4.tgz#8f3889ee8657581130a29d9cc91d7c73b7c4a28d"
+    integrity sha512-Ddy3QZfIbEV0VYcVtFDCjeE4xwVTJWTmUtorAJkn6u/92Z/nWJNV+mILyqHKrUxXYKA2EoCilgoPePymKL4DvQ==
+    dependencies:
+      "@babel/helper-plugin-utils" "^7.10.4"
+      "@babel/helper-regex" "^7.10.4"
+  
+  "@babel/plugin-transform-template-literals@^7.10.4":
+    version "7.10.5"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-transform-template-literals/-/plugin-transform-template-literals-7.10.5.tgz#78bc5d626a6642db3312d9d0f001f5e7639fde8c"
+    integrity sha512-V/lnPGIb+KT12OQikDvgSuesRX14ck5FfJXt6+tXhdkJ+Vsd0lDCVtF6jcB4rNClYFzaB2jusZ+lNISDk2mMMw==
+    dependencies:
+      "@babel/helper-annotate-as-pure" "^7.10.4"
+      "@babel/helper-plugin-utils" "^7.10.4"
+  
+  "@babel/plugin-transform-typeof-symbol@^7.10.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-transform-typeof-symbol/-/plugin-transform-typeof-symbol-7.10.4.tgz#9509f1a7eec31c4edbffe137c16cc33ff0bc5bfc"
+    integrity sha512-QqNgYwuuW0y0H+kUE/GWSR45t/ccRhe14Fs/4ZRouNNQsyd4o3PG4OtHiIrepbM2WKUBDAXKCAK/Lk4VhzTaGA==
+    dependencies:
+      "@babel/helper-plugin-utils" "^7.10.4"
+  
+  "@babel/plugin-transform-unicode-escapes@^7.10.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-transform-unicode-escapes/-/plugin-transform-unicode-escapes-7.10.4.tgz#feae523391c7651ddac115dae0a9d06857892007"
+    integrity sha512-y5XJ9waMti2J+e7ij20e+aH+fho7Wb7W8rNuu72aKRwCHFqQdhkdU2lo3uZ9tQuboEJcUFayXdARhcxLQ3+6Fg==
+    dependencies:
+      "@babel/helper-plugin-utils" "^7.10.4"
+  
+  "@babel/plugin-transform-unicode-regex@^7.10.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/plugin-transform-unicode-regex/-/plugin-transform-unicode-regex-7.10.4.tgz#e56d71f9282fac6db09c82742055576d5e6d80a8"
+    integrity sha512-wNfsc4s8N2qnIwpO/WP2ZiSyjfpTamT2C9V9FDH/Ljub9zw6P3SjkXcFmc0RQUt96k2fmIvtla2MMjgTwIAC+A==
+    dependencies:
+      "@babel/helper-create-regexp-features-plugin" "^7.10.4"
+      "@babel/helper-plugin-utils" "^7.10.4"
+  
+  "@babel/preset-env@^7.11.0":
+    version "7.11.5"
+    resolved "https://registry.yarnpkg.com/@babel/preset-env/-/preset-env-7.11.5.tgz#18cb4b9379e3e92ffea92c07471a99a2914e4272"
+    integrity sha512-kXqmW1jVcnB2cdueV+fyBM8estd5mlNfaQi6lwLgRwCby4edpavgbFhiBNjmWA3JpB/yZGSISa7Srf+TwxDQoA==
+    dependencies:
+      "@babel/compat-data" "^7.11.0"
+      "@babel/helper-compilation-targets" "^7.10.4"
+      "@babel/helper-module-imports" "^7.10.4"
+      "@babel/helper-plugin-utils" "^7.10.4"
+      "@babel/plugin-proposal-async-generator-functions" "^7.10.4"
+      "@babel/plugin-proposal-class-properties" "^7.10.4"
+      "@babel/plugin-proposal-dynamic-import" "^7.10.4"
+      "@babel/plugin-proposal-export-namespace-from" "^7.10.4"
+      "@babel/plugin-proposal-json-strings" "^7.10.4"
+      "@babel/plugin-proposal-logical-assignment-operators" "^7.11.0"
+      "@babel/plugin-proposal-nullish-coalescing-operator" "^7.10.4"
+      "@babel/plugin-proposal-numeric-separator" "^7.10.4"
+      "@babel/plugin-proposal-object-rest-spread" "^7.11.0"
+      "@babel/plugin-proposal-optional-catch-binding" "^7.10.4"
+      "@babel/plugin-proposal-optional-chaining" "^7.11.0"
+      "@babel/plugin-proposal-private-methods" "^7.10.4"
+      "@babel/plugin-proposal-unicode-property-regex" "^7.10.4"
+      "@babel/plugin-syntax-async-generators" "^7.8.0"
+      "@babel/plugin-syntax-class-properties" "^7.10.4"
+      "@babel/plugin-syntax-dynamic-import" "^7.8.0"
+      "@babel/plugin-syntax-export-namespace-from" "^7.8.3"
+      "@babel/plugin-syntax-json-strings" "^7.8.0"
+      "@babel/plugin-syntax-logical-assignment-operators" "^7.10.4"
+      "@babel/plugin-syntax-nullish-coalescing-operator" "^7.8.0"
+      "@babel/plugin-syntax-numeric-separator" "^7.10.4"
+      "@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.10.4"
+      "@babel/plugin-transform-arrow-functions" "^7.10.4"
+      "@babel/plugin-transform-async-to-generator" "^7.10.4"
+      "@babel/plugin-transform-block-scoped-functions" "^7.10.4"
+      "@babel/plugin-transform-block-scoping" "^7.10.4"
+      "@babel/plugin-transform-classes" "^7.10.4"
+      "@babel/plugin-transform-computed-properties" "^7.10.4"
+      "@babel/plugin-transform-destructuring" "^7.10.4"
+      "@babel/plugin-transform-dotall-regex" "^7.10.4"
+      "@babel/plugin-transform-duplicate-keys" "^7.10.4"
+      "@babel/plugin-transform-exponentiation-operator" "^7.10.4"
+      "@babel/plugin-transform-for-of" "^7.10.4"
+      "@babel/plugin-transform-function-name" "^7.10.4"
+      "@babel/plugin-transform-literals" "^7.10.4"
+      "@babel/plugin-transform-member-expression-literals" "^7.10.4"
+      "@babel/plugin-transform-modules-amd" "^7.10.4"
+      "@babel/plugin-transform-modules-commonjs" "^7.10.4"
+      "@babel/plugin-transform-modules-systemjs" "^7.10.4"
+      "@babel/plugin-transform-modules-umd" "^7.10.4"
+      "@babel/plugin-transform-named-capturing-groups-regex" "^7.10.4"
+      "@babel/plugin-transform-new-target" "^7.10.4"
+      "@babel/plugin-transform-object-super" "^7.10.4"
+      "@babel/plugin-transform-parameters" "^7.10.4"
+      "@babel/plugin-transform-property-literals" "^7.10.4"
+      "@babel/plugin-transform-regenerator" "^7.10.4"
+      "@babel/plugin-transform-reserved-words" "^7.10.4"
+      "@babel/plugin-transform-shorthand-properties" "^7.10.4"
+      "@babel/plugin-transform-spread" "^7.11.0"
+      "@babel/plugin-transform-sticky-regex" "^7.10.4"
+      "@babel/plugin-transform-template-literals" "^7.10.4"
+      "@babel/plugin-transform-typeof-symbol" "^7.10.4"
+      "@babel/plugin-transform-unicode-escapes" "^7.10.4"
+      "@babel/plugin-transform-unicode-regex" "^7.10.4"
+      "@babel/preset-modules" "^0.1.3"
+      "@babel/types" "^7.11.5"
+      browserslist "^4.12.0"
+      core-js-compat "^3.6.2"
+      invariant "^2.2.2"
+      levenary "^1.1.1"
+      semver "^5.5.0"
+  
+  "@babel/preset-modules@^0.1.3":
+    version "0.1.4"
+    resolved "https://registry.yarnpkg.com/@babel/preset-modules/-/preset-modules-0.1.4.tgz#362f2b68c662842970fdb5e254ffc8fc1c2e415e"
+    integrity sha512-J36NhwnfdzpmH41M1DrnkkgAqhZaqr/NBdPfQ677mLzlaXo+oDiv1deyCDtgAhz8p328otdob0Du7+xgHGZbKg==
+    dependencies:
+      "@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@^7.11.0", "@babel/runtime@^7.8.4":
+    version "7.11.2"
+    resolved "https://registry.yarnpkg.com/@babel/runtime/-/runtime-7.11.2.tgz#f549c13c754cc40b87644b9fa9f09a6a95fe0736"
+    integrity sha512-TeWkU52so0mPtDcaCTxNBI/IHiz0pZgr8VEFqXFtZWpYD08ZB6FaSwVAS8MKRQAP3bYKiVjwysOJgMFY28o6Tw==
+    dependencies:
+      regenerator-runtime "^0.13.4"
+  
+  "@babel/template@^7.10.4":
+    version "7.10.4"
+    resolved "https://registry.yarnpkg.com/@babel/template/-/template-7.10.4.tgz#3251996c4200ebc71d1a8fc405fba940f36ba278"
+    integrity sha512-ZCjD27cGJFUB6nmCB1Enki3r+L5kJveX9pq1SvAUKoICy6CZ9yD8xO086YXdYhvNjBdnekm4ZnaP5yC8Cs/1tA==
+    dependencies:
+      "@babel/code-frame" "^7.10.4"
+      "@babel/parser" "^7.10.4"
+      "@babel/types" "^7.10.4"
+  
+  "@babel/traverse@^7.0.0", "@babel/traverse@^7.10.4", "@babel/traverse@^7.11.5", "@babel/traverse@^7.7.0":
+    version "7.11.5"
+    resolved "https://registry.yarnpkg.com/@babel/traverse/-/traverse-7.11.5.tgz#be777b93b518eb6d76ee2e1ea1d143daa11e61c3"
+    integrity sha512-EjiPXt+r7LiCZXEfRpSJd+jUMnBd4/9OUv7Nx3+0u9+eimMwJmG0Q98lw4/289JCoxSE8OolDMNZaaF/JZ69WQ==
+    dependencies:
+      "@babel/code-frame" "^7.10.4"
+      "@babel/generator" "^7.11.5"
+      "@babel/helper-function-name" "^7.10.4"
+      "@babel/helper-split-export-declaration" "^7.11.0"
+      "@babel/parser" "^7.11.5"
+      "@babel/types" "^7.11.5"
+      debug "^4.1.0"
+      globals "^11.1.0"
+      lodash "^4.17.19"
+  
+  "@babel/types@^7.0.0", "@babel/types@^7.10.4", "@babel/types@^7.10.5", "@babel/types@^7.11.0", "@babel/types@^7.11.5", "@babel/types@^7.4.4", "@babel/types@^7.7.0":
+    version "7.11.5"
+    resolved "https://registry.yarnpkg.com/@babel/types/-/types-7.11.5.tgz#d9de577d01252d77c6800cee039ee64faf75662d"
+    integrity sha512-bvM7Qz6eKnJVFIn+1LPtjlBFPVN5jNDc1XmN15vWe7Q3DPBufWWsLiIvUu7xW87uTG6QoggpIDnUgLQvPheU+Q==
+    dependencies:
+      "@babel/helper-validator-identifier" "^7.10.4"
+      lodash "^4.17.19"
+      to-fast-properties "^2.0.0"
+  
+  "@hapi/address@2.x.x":
+    version "2.1.4"
+    resolved "https://registry.yarnpkg.com/@hapi/address/-/address-2.1.4.tgz#5d67ed43f3fd41a69d4b9ff7b56e7c0d1d0a81e5"
+    integrity sha512-QD1PhQk+s31P1ixsX0H0Suoupp3VMXzIVMSwobR3F3MSUO2YCV0B7xqLcUw/Bh8yuvd3LhpyqLQWTNcRmp6IdQ==
+  
+  "@hapi/bourne@1.x.x":
+    version "1.3.2"
+    resolved "https://registry.yarnpkg.com/@hapi/bourne/-/bourne-1.3.2.tgz#0a7095adea067243ce3283e1b56b8a8f453b242a"
+    integrity sha512-1dVNHT76Uu5N3eJNTYcvxee+jzX4Z9lfciqRRHCU27ihbUcYi+iSc2iml5Ke1LXe1SyJCLA0+14Jh4tXJgOppA==
+  
+  "@hapi/hoek@8.x.x", "@hapi/hoek@^8.3.0":
+    version "8.5.1"
+    resolved "https://registry.yarnpkg.com/@hapi/hoek/-/hoek-8.5.1.tgz#fde96064ca446dec8c55a8c2f130957b070c6e06"
+    integrity sha512-yN7kbciD87WzLGc5539Tn0sApjyiGHAJgKvG9W8C7O+6c7qmoQMfVs0W4bX17eqz6C78QJqqFrtgdK5EWf6Qow==
+  
+  "@hapi/joi@^15.0.1":
+    version "15.1.1"
+    resolved "https://registry.yarnpkg.com/@hapi/joi/-/joi-15.1.1.tgz#c675b8a71296f02833f8d6d243b34c57b8ce19d7"
+    integrity sha512-entf8ZMOK8sc+8YfeOlM8pCfg3b5+WZIKBfUaaJT8UsjAAPjartzxIYm3TIbjvA4u+u++KbcXD38k682nVHDAQ==
+    dependencies:
+      "@hapi/address" "2.x.x"
+      "@hapi/bourne" "1.x.x"
+      "@hapi/hoek" "8.x.x"
+      "@hapi/topo" "3.x.x"
+  
+  "@hapi/topo@3.x.x":
+    version "3.1.6"
+    resolved "https://registry.yarnpkg.com/@hapi/topo/-/topo-3.1.6.tgz#68d935fa3eae7fdd5ab0d7f953f3205d8b2bfc29"
+    integrity sha512-tAag0jEcjwH+P2quUfipd7liWCNX2F8NvYjQp2wtInsZxnMlypdw0FtAOLxtvvkO+GSRRbmNi8m/5y42PQJYCQ==
+    dependencies:
+      "@hapi/hoek" "^8.3.0"
+  
+  "@intervolga/optimize-cssnano-plugin@^1.0.5":
+    version "1.0.6"
+    resolved "https://registry.yarnpkg.com/@intervolga/optimize-cssnano-plugin/-/optimize-cssnano-plugin-1.0.6.tgz#be7c7846128b88f6a9b1d1261a0ad06eb5c0fdf8"
+    integrity sha512-zN69TnSr0viRSU6cEDIcuPcP67QcpQ6uHACg58FiN9PDrU6SLyGW3MR4tiISbYxy1kDWAVPwD+XwQTWE5cigAA==
+    dependencies:
+      cssnano "^4.0.0"
+      cssnano-preset-default "^4.0.0"
+      postcss "^7.0.0"
+  
+  "@mrmlnc/readdir-enhanced@^2.2.1":
+    version "2.2.1"
+    resolved "https://registry.yarnpkg.com/@mrmlnc/readdir-enhanced/-/readdir-enhanced-2.2.1.tgz#524af240d1a360527b730475ecfa1344aa540dde"
+    integrity sha512-bPHp6Ji8b41szTOcaP63VlnbbO5Ny6dwAATtY6JTjh5N2OLrb5Qk/Th5cRkRQhkWCt+EJsYrNB0MiL+Gpn6e3g==
+    dependencies:
+      call-me-maybe "^1.0.1"
+      glob-to-regexp "^0.3.0"
+  
+  "@nodelib/fs.stat@^1.1.2":
+    version "1.1.3"
+    resolved "https://registry.yarnpkg.com/@nodelib/fs.stat/-/fs.stat-1.1.3.tgz#2b5a3ab3f918cca48a8c754c08168e3f03eba61b"
+    integrity sha512-shAmDyaQC4H92APFoIaVDHCx5bStIocgvbwQyxPRrbUY20V1EYTbSDchWbuwlMG3V17cprZhA6+78JfB+3DTPw==
+  
+  "@nuxt/opencollective@^0.3.0":
+    version "0.3.0"
+    resolved "https://registry.yarnpkg.com/@nuxt/opencollective/-/opencollective-0.3.0.tgz#11d8944dcf2d526e31660bb69570be03f8fb72b7"
+    integrity sha512-Vf09BxCdj1iT2IRqVwX5snaY2WCTkvM0O4cWWSO1ThCFuc4if0Q/nNwAgCxRU0FeYHJ7DdyMUNSdswCLKlVqeg==
+    dependencies:
+      chalk "^2.4.2"
+      consola "^2.10.1"
+      node-fetch "^2.6.0"
+  
+  "@soda/friendly-errors-webpack-plugin@^1.7.1":
+    version "1.7.1"
+    resolved "https://registry.yarnpkg.com/@soda/friendly-errors-webpack-plugin/-/friendly-errors-webpack-plugin-1.7.1.tgz#706f64bcb4a8b9642b48ae3ace444c70334d615d"
+    integrity sha512-cWKrGaFX+rfbMrAxVv56DzhPNqOJPZuNIS2HGMELtgGzb+vsMzyig9mml5gZ/hr2BGtSLV+dP2LUEuAL8aG2mQ==
+    dependencies:
+      chalk "^1.1.3"
+      error-stack-parser "^2.0.0"
+      string-width "^2.0.0"
+  
+  "@soda/get-current-script@^1.0.0":
+    version "1.0.2"
+    resolved "https://registry.yarnpkg.com/@soda/get-current-script/-/get-current-script-1.0.2.tgz#a53515db25d8038374381b73af20bb4f2e508d87"
+    integrity sha512-T7VNNlYVM1SgQ+VsMYhnDkcGmWhQdL0bDyGm5TlQ3GBXnJscEClUUOKduWTmm2zCnvNLC1hc3JpuXjs/nFOc5w==
+  
+  "@types/anymatch@*":
+    version "1.3.1"
+    resolved "https://registry.yarnpkg.com/@types/anymatch/-/anymatch-1.3.1.tgz#336badc1beecb9dacc38bea2cf32adf627a8421a"
+    integrity sha512-/+CRPXpBDpo2RK9C68N3b2cOvO0Cf5B9aPijHsoDQTHivnGSObdOF2BRQOYjojWTDy6nQvMjmqRXIxH55VjxxA==
+  
+  "@types/body-parser@*":
+    version "1.19.0"
+    resolved "https://registry.yarnpkg.com/@types/body-parser/-/body-parser-1.19.0.tgz#0685b3c47eb3006ffed117cdd55164b61f80538f"
+    integrity sha512-W98JrE0j2K78swW4ukqMleo8R7h/pFETjM2DQ90MF6XK2i4LO4W3gQ71Lt4w3bfm2EvVSyWHplECvB5sK22yFQ==
+    dependencies:
+      "@types/connect" "*"
+      "@types/node" "*"
+  
+  "@types/color-name@^1.1.1":
+    version "1.1.1"
+    resolved "https://registry.yarnpkg.com/@types/color-name/-/color-name-1.1.1.tgz#1c1261bbeaa10a8055bbc5d8ab84b7b2afc846a0"
+    integrity sha512-rr+OQyAjxze7GgWrSaJwydHStIhHq2lvY3BOC2Mj7KnzI7XK0Uw1TOOdI9lDoajEbSWLiYgoo4f1R51erQfhPQ==
+  
+  "@types/connect-history-api-fallback@*":
+    version "1.3.3"
+    resolved "https://registry.yarnpkg.com/@types/connect-history-api-fallback/-/connect-history-api-fallback-1.3.3.tgz#4772b79b8b53185f0f4c9deab09236baf76ee3b4"
+    integrity sha512-7SxFCd+FLlxCfwVwbyPxbR4khL9aNikJhrorw8nUIOqeuooc9gifBuDQOJw5kzN7i6i3vLn9G8Wde/4QDihpYw==
+    dependencies:
+      "@types/express-serve-static-core" "*"
+      "@types/node" "*"
+  
+  "@types/connect@*":
+    version "3.4.33"
+    resolved "https://registry.yarnpkg.com/@types/connect/-/connect-3.4.33.tgz#31610c901eca573b8713c3330abc6e6b9f588546"
+    integrity sha512-2+FrkXY4zllzTNfJth7jOqEHC+enpLeGslEhpnTAkg21GkRrWV4SsAtqchtT4YS9/nODBU2/ZfsBY2X4J/dX7A==
+    dependencies:
+      "@types/node" "*"
+  
+  "@types/express-serve-static-core@*":
+    version "4.17.12"
+    resolved "https://registry.yarnpkg.com/@types/express-serve-static-core/-/express-serve-static-core-4.17.12.tgz#9a487da757425e4f267e7d1c5720226af7f89591"
+    integrity sha512-EaEdY+Dty1jEU7U6J4CUWwxL+hyEGMkO5jan5gplfegUgCUsIUWqXxqw47uGjimeT4Qgkz/XUfwoau08+fgvKA==
+    dependencies:
+      "@types/node" "*"
+      "@types/qs" "*"
+      "@types/range-parser" "*"
+  
+  "@types/express@*":
+    version "4.17.8"
+    resolved "https://registry.yarnpkg.com/@types/express/-/express-4.17.8.tgz#3df4293293317e61c60137d273a2e96cd8d5f27a"
+    integrity sha512-wLhcKh3PMlyA2cNAB9sjM1BntnhPMiM0JOBwPBqttjHev2428MLEB4AYVN+d8s2iyCVZac+o41Pflm/ZH5vLXQ==
+    dependencies:
+      "@types/body-parser" "*"
+      "@types/express-serve-static-core" "*"
+      "@types/qs" "*"
+      "@types/serve-static" "*"
+  
+  "@types/glob@^7.1.1":
+    version "7.1.3"
+    resolved "https://registry.yarnpkg.com/@types/glob/-/glob-7.1.3.tgz#e6ba80f36b7daad2c685acd9266382e68985c183"
+    integrity sha512-SEYeGAIQIQX8NN6LDKprLjbrd5dARM5EXsd8GI/A5l0apYI1fGMWgPHSe4ZKL4eozlAyI+doUE9XbYS4xCkQ1w==
+    dependencies:
+      "@types/minimatch" "*"
+      "@types/node" "*"
+  
+  "@types/http-proxy-middleware@*":
+    version "0.19.3"
+    resolved "https://registry.yarnpkg.com/@types/http-proxy-middleware/-/http-proxy-middleware-0.19.3.tgz#b2eb96fbc0f9ac7250b5d9c4c53aade049497d03"
+    integrity sha512-lnBTx6HCOUeIJMLbI/LaL5EmdKLhczJY5oeXZpX/cXE4rRqb3RmV7VcMpiEfYkmTjipv3h7IAyIINe4plEv7cA==
+    dependencies:
+      "@types/connect" "*"
+      "@types/http-proxy" "*"
+      "@types/node" "*"
+  
+  "@types/http-proxy@*":
+    version "1.17.4"
+    resolved "https://registry.yarnpkg.com/@types/http-proxy/-/http-proxy-1.17.4.tgz#e7c92e3dbe3e13aa799440ff42e6d3a17a9d045b"
+    integrity sha512-IrSHl2u6AWXduUaDLqYpt45tLVCtYv7o4Z0s1KghBCDgIIS9oW5K1H8mZG/A2CfeLdEa7rTd1ACOiHBc1EMT2Q==
+    dependencies:
+      "@types/node" "*"
+  
+  "@types/json-schema@^7.0.5":
+    version "7.0.6"
+    resolved "https://registry.yarnpkg.com/@types/json-schema/-/json-schema-7.0.6.tgz#f4c7ec43e81b319a9815115031709f26987891f0"
+    integrity sha512-3c+yGKvVP5Y9TYBEibGNR+kLtijnj7mYrXRg+WpFb2X9xm04g/DXYkfg4hmzJQosc9snFNUPkbYIhu+KAm6jJw==
+  
+  "@types/mime@*":
+    version "2.0.3"
+    resolved "https://registry.yarnpkg.com/@types/mime/-/mime-2.0.3.tgz#c893b73721db73699943bfc3653b1deb7faa4a3a"
+    integrity sha512-Jus9s4CDbqwocc5pOAnh8ShfrnMcPHuJYzVcSUU7lrh8Ni5HuIqX3oilL86p3dlTrk0LzHRCgA/GQ7uNCw6l2Q==
+  
+  "@types/mini-css-extract-plugin@^0.9.1":
+    version "0.9.1"
+    resolved "https://registry.yarnpkg.com/@types/mini-css-extract-plugin/-/mini-css-extract-plugin-0.9.1.tgz#d4bdde5197326fca039d418f4bdda03dc74dc451"
+    integrity sha512-+mN04Oszdz9tGjUP/c1ReVwJXxSniLd7lF++sv+8dkABxVNthg6uccei+4ssKxRHGoMmPxdn7uBdJWONSJGTGQ==
+    dependencies:
+      "@types/webpack" "*"
+  
+  "@types/minimatch@*":
+    version "3.0.3"
+    resolved "https://registry.yarnpkg.com/@types/minimatch/-/minimatch-3.0.3.tgz#3dca0e3f33b200fc7d1139c0cd96c1268cadfd9d"
+    integrity sha512-tHq6qdbT9U1IRSGf14CL0pUlULksvY9OZ+5eEgl1N7t+OA3tGvNpxJCzuKQlsNgCVwbAs670L1vcVQi8j9HjnA==
+  
+  "@types/minimist@^1.2.0":
+    version "1.2.0"
+    resolved "https://registry.yarnpkg.com/@types/minimist/-/minimist-1.2.0.tgz#69a23a3ad29caf0097f06eda59b361ee2f0639f6"
+    integrity sha1-aaI6OtKcrwCX8G7aWbNh7i8GOfY=
+  
+  "@types/node@*":
+    version "14.6.4"
+    resolved "https://registry.yarnpkg.com/@types/node/-/node-14.6.4.tgz#a145cc0bb14ef9c4777361b7bbafa5cf8e3acb5a"
+    integrity sha512-Wk7nG1JSaMfMpoMJDKUsWYugliB2Vy55pdjLpmLixeyMi7HizW2I/9QoxsPCkXl3dO+ZOVqPumKaDUv5zJu2uQ==
+  
+  "@types/normalize-package-data@^2.4.0":
+    version "2.4.0"
+    resolved "https://registry.yarnpkg.com/@types/normalize-package-data/-/normalize-package-data-2.4.0.tgz#e486d0d97396d79beedd0a6e33f4534ff6b4973e"
+    integrity sha512-f5j5b/Gf71L+dbqxIpQ4Z2WlmI/mPJ0fOkGGmFgtb6sAu97EPczzbS3/tJKxmcYDj55OX6ssqwDAWOHIYDRDGA==
+  
+  "@types/q@^1.5.1":
+    version "1.5.4"
+    resolved "https://registry.yarnpkg.com/@types/q/-/q-1.5.4.tgz#15925414e0ad2cd765bfef58842f7e26a7accb24"
+    integrity sha512-1HcDas8SEj4z1Wc696tH56G8OlRaH/sqZOynNNB+HF0WOeXPaxTtbYzJY2oEfiUxjSKjhCKr+MvR7dCHcEelug==
+  
+  "@types/qs@*":
+    version "6.9.4"
+    resolved "https://registry.yarnpkg.com/@types/qs/-/qs-6.9.4.tgz#a59e851c1ba16c0513ea123830dd639a0a15cb6a"
+    integrity sha512-+wYo+L6ZF6BMoEjtf8zB2esQsqdV6WsjRK/GP9WOgLPrq87PbNWgIxS76dS5uvl/QXtHGakZmwTznIfcPXcKlQ==
+  
+  "@types/range-parser@*":
+    version "1.2.3"
+    resolved "https://registry.yarnpkg.com/@types/range-parser/-/range-parser-1.2.3.tgz#7ee330ba7caafb98090bece86a5ee44115904c2c"
+    integrity sha512-ewFXqrQHlFsgc09MK5jP5iR7vumV/BYayNC6PgJO2LPe8vrnNFyjQjSppfEngITi0qvfKtzFvgKymGheFM9UOA==
+  
+  "@types/serve-static@*":
+    version "1.13.5"
+    resolved "https://registry.yarnpkg.com/@types/serve-static/-/serve-static-1.13.5.tgz#3d25d941a18415d3ab092def846e135a08bbcf53"
+    integrity sha512-6M64P58N+OXjU432WoLLBQxbA0LRGBCRm7aAGQJ+SMC1IMl0dgRVi9EFfoDcS2a7Xogygk/eGN94CfwU9UF7UQ==
+    dependencies:
+      "@types/express-serve-static-core" "*"
+      "@types/mime" "*"
+  
+  "@types/source-list-map@*":
+    version "0.1.2"
+    resolved "https://registry.yarnpkg.com/@types/source-list-map/-/source-list-map-0.1.2.tgz#0078836063ffaf17412349bba364087e0ac02ec9"
+    integrity sha512-K5K+yml8LTo9bWJI/rECfIPrGgxdpeNbj+d53lwN4QjW1MCwlkhUms+gtdzigTeUyBr09+u8BwOIY3MXvHdcsA==
+  
+  "@types/tapable@*":
+    version "1.0.6"
+    resolved "https://registry.yarnpkg.com/@types/tapable/-/tapable-1.0.6.tgz#a9ca4b70a18b270ccb2bc0aaafefd1d486b7ea74"
+    integrity sha512-W+bw9ds02rAQaMvaLYxAbJ6cvguW/iJXNT6lTssS1ps6QdrMKttqEAMEG/b5CR8TZl3/L7/lH0ZV5nNR1LXikA==
+  
+  "@types/uglify-js@*":
+    version "3.9.3"
+    resolved "https://registry.yarnpkg.com/@types/uglify-js/-/uglify-js-3.9.3.tgz#d94ed608e295bc5424c9600e6b8565407b6b4b6b"
+    integrity sha512-KswB5C7Kwduwjj04Ykz+AjvPcfgv/37Za24O2EDzYNbwyzOo8+ydtvzUfZ5UMguiVu29Gx44l1A6VsPPcmYu9w==
+    dependencies:
+      source-map "^0.6.1"
+  
+  "@types/webpack-dev-server@^3.11.0":
+    version "3.11.0"
+    resolved "https://registry.yarnpkg.com/@types/webpack-dev-server/-/webpack-dev-server-3.11.0.tgz#bcc3b85e7dc6ac2db25330610513f2228c2fcfb2"
+    integrity sha512-3+86AgSzl18n5P1iUP9/lz3G3GMztCp+wxdDvVuNhx1sr1jE79GpYfKHL8k+Vht3N74K2n98CuAEw4YPJCYtDA==
+    dependencies:
+      "@types/connect-history-api-fallback" "*"
+      "@types/express" "*"
+      "@types/http-proxy-middleware" "*"
+      "@types/serve-static" "*"
+      "@types/webpack" "*"
+  
+  "@types/webpack-sources@*":
+    version "1.4.2"
+    resolved "https://registry.yarnpkg.com/@types/webpack-sources/-/webpack-sources-1.4.2.tgz#5d3d4dea04008a779a90135ff96fb5c0c9e6292c"
+    integrity sha512-77T++JyKow4BQB/m9O96n9d/UUHWLQHlcqXb9Vsf4F1+wKNrrlWNFPDLKNT92RJnCSL6CieTc+NDXtCVZswdTw==
+    dependencies:
+      "@types/node" "*"
+      "@types/source-list-map" "*"
+      source-map "^0.7.3"
+  
+  "@types/webpack@*", "@types/webpack@^4.0.0":
+    version "4.41.22"
+    resolved "https://registry.yarnpkg.com/@types/webpack/-/webpack-4.41.22.tgz#ff9758a17c6bd499e459b91e78539848c32d0731"
+    integrity sha512-JQDJK6pj8OMV9gWOnN1dcLCyU9Hzs6lux0wBO4lr1+gyEhIBR9U3FMrz12t2GPkg110XAxEAw2WHF6g7nZIbRQ==
+    dependencies:
+      "@types/anymatch" "*"
+      "@types/node" "*"
+      "@types/tapable" "*"
+      "@types/uglify-js" "*"
+      "@types/webpack-sources" "*"
+      source-map "^0.6.0"
+  
+  "@vue/babel-helper-vue-jsx-merge-props@^1.0.0":
+    version "1.0.0"
+    resolved "https://registry.yarnpkg.com/@vue/babel-helper-vue-jsx-merge-props/-/babel-helper-vue-jsx-merge-props-1.0.0.tgz#048fe579958da408fb7a8b2a3ec050b50a661040"
+    integrity sha512-6tyf5Cqm4m6v7buITuwS+jHzPlIPxbFzEhXR5JGZpbrvOcp1hiQKckd305/3C7C36wFekNTQSxAtgeM0j0yoUw==
+  
+  "@vue/babel-plugin-transform-vue-jsx@^1.1.2":
+    version "1.1.2"
+    resolved "https://registry.yarnpkg.com/@vue/babel-plugin-transform-vue-jsx/-/babel-plugin-transform-vue-jsx-1.1.2.tgz#c0a3e6efc022e75e4247b448a8fc6b86f03e91c0"
+    integrity sha512-YfdaoSMvD1nj7+DsrwfTvTnhDXI7bsuh+Y5qWwvQXlD24uLgnsoww3qbiZvWf/EoviZMrvqkqN4CBw0W3BWUTQ==
+    dependencies:
+      "@babel/helper-module-imports" "^7.0.0"
+      "@babel/plugin-syntax-jsx" "^7.2.0"
+      "@vue/babel-helper-vue-jsx-merge-props" "^1.0.0"
+      html-tags "^2.0.0"
+      lodash.kebabcase "^4.1.1"
+      svg-tags "^1.0.0"
+  
+  "@vue/babel-preset-app@^4.5.4":
+    version "4.5.4"
+    resolved "https://registry.yarnpkg.com/@vue/babel-preset-app/-/babel-preset-app-4.5.4.tgz#bb164e8ab55673c561e6e83511631eda19efd7e4"
+    integrity sha512-a+2s/lL3fE3h9/ekvpMVLhZTDjR3xt+jnpTwuQtEZ3KIuzFHxbmwAjueRZh6BKEGfB6kgZ3KqZHFX3vx/DRJ4w==
+    dependencies:
+      "@ant-design-vue/babel-plugin-jsx" "^1.0.0-0"
+      "@babel/core" "^7.11.0"
+      "@babel/helper-compilation-targets" "^7.9.6"
+      "@babel/helper-module-imports" "^7.8.3"
+      "@babel/plugin-proposal-class-properties" "^7.8.3"
+      "@babel/plugin-proposal-decorators" "^7.8.3"
+      "@babel/plugin-syntax-dynamic-import" "^7.8.3"
+      "@babel/plugin-syntax-jsx" "^7.8.3"
+      "@babel/plugin-transform-runtime" "^7.11.0"
+      "@babel/preset-env" "^7.11.0"
+      "@babel/runtime" "^7.11.0"
+      "@vue/babel-preset-jsx" "^1.1.2"
+      babel-plugin-dynamic-import-node "^2.3.3"
+      core-js "^3.6.5"
+      core-js-compat "^3.6.5"
+      semver "^6.1.0"
+  
+  "@vue/babel-preset-jsx@^1.1.2":
+    version "1.1.2"
+    resolved "https://registry.yarnpkg.com/@vue/babel-preset-jsx/-/babel-preset-jsx-1.1.2.tgz#2e169eb4c204ea37ca66c2ea85a880bfc99d4f20"
+    integrity sha512-zDpVnFpeC9YXmvGIDSsKNdL7qCG2rA3gjywLYHPCKDT10erjxF4U+6ay9X6TW5fl4GsDlJp9bVfAVQAAVzxxvQ==
+    dependencies:
+      "@vue/babel-helper-vue-jsx-merge-props" "^1.0.0"
+      "@vue/babel-plugin-transform-vue-jsx" "^1.1.2"
+      "@vue/babel-sugar-functional-vue" "^1.1.2"
+      "@vue/babel-sugar-inject-h" "^1.1.2"
+      "@vue/babel-sugar-v-model" "^1.1.2"
+      "@vue/babel-sugar-v-on" "^1.1.2"
+  
+  "@vue/babel-sugar-functional-vue@^1.1.2":
+    version "1.1.2"
+    resolved "https://registry.yarnpkg.com/@vue/babel-sugar-functional-vue/-/babel-sugar-functional-vue-1.1.2.tgz#f7e24fba09e6f1ee70104560a8808057555f1a9a"
+    integrity sha512-YhmdJQSVEFF5ETJXzrMpj0nkCXEa39TvVxJTuVjzvP2rgKhdMmQzlJuMv/HpadhZaRVMCCF3AEjjJcK5q/cYzQ==
+    dependencies:
+      "@babel/plugin-syntax-jsx" "^7.2.0"
+  
+  "@vue/babel-sugar-inject-h@^1.1.2":
+    version "1.1.2"
+    resolved "https://registry.yarnpkg.com/@vue/babel-sugar-inject-h/-/babel-sugar-inject-h-1.1.2.tgz#8a5276b6d8e2ed16ffc8078aad94236274e6edf0"
+    integrity sha512-VRSENdTvD5htpnVp7i7DNuChR5rVMcORdXjvv5HVvpdKHzDZAYiLSD+GhnhxLm3/dMuk8pSzV+k28ECkiN5m8w==
+    dependencies:
+      "@babel/plugin-syntax-jsx" "^7.2.0"
+  
+  "@vue/babel-sugar-v-model@^1.1.2":
+    version "1.1.2"
+    resolved "https://registry.yarnpkg.com/@vue/babel-sugar-v-model/-/babel-sugar-v-model-1.1.2.tgz#1ff6fd1b800223fc9cb1e84dceb5e52d737a8192"
+    integrity sha512-vLXPvNq8vDtt0u9LqFdpGM9W9IWDmCmCyJXuozlq4F4UYVleXJ2Fa+3JsnTZNJcG+pLjjfnEGHci2339Kj5sGg==
+    dependencies:
+      "@babel/plugin-syntax-jsx" "^7.2.0"
+      "@vue/babel-helper-vue-jsx-merge-props" "^1.0.0"
+      "@vue/babel-plugin-transform-vue-jsx" "^1.1.2"
+      camelcase "^5.0.0"
+      html-tags "^2.0.0"
+      svg-tags "^1.0.0"
+  
+  "@vue/babel-sugar-v-on@^1.1.2":
+    version "1.1.2"
+    resolved "https://registry.yarnpkg.com/@vue/babel-sugar-v-on/-/babel-sugar-v-on-1.1.2.tgz#b2ef99b8f2fab09fbead25aad70ef42e1cf5b13b"
+    integrity sha512-T8ZCwC8Jp2uRtcZ88YwZtZXe7eQrJcfRq0uTFy6ShbwYJyz5qWskRFoVsdTi9o0WEhmQXxhQUewodOSCUPVmsQ==
+    dependencies:
+      "@babel/plugin-syntax-jsx" "^7.2.0"
+      "@vue/babel-plugin-transform-vue-jsx" "^1.1.2"
+      camelcase "^5.0.0"
+  
+  "@vue/cli-overlay@^4.5.4":
+    version "4.5.4"
+    resolved "https://registry.yarnpkg.com/@vue/cli-overlay/-/cli-overlay-4.5.4.tgz#e07e3ccc2e4d770d4fdbd45cdde777d592822c19"
+    integrity sha512-nthli1n7rXaqaMZsH0KNdFqeYJxDOQNeaobp9SjeSdrpD1xAj/B0+RJMWQWIFsfdQn1AQP1UVMnkfdakTiLgxA==
+  
+  "@vue/cli-plugin-babel@~4.5.0":
+    version "4.5.4"
+    resolved "https://registry.yarnpkg.com/@vue/cli-plugin-babel/-/cli-plugin-babel-4.5.4.tgz#a01cdcb3d46064675dd88d61b640adadcc851e2b"
+    integrity sha512-pXEzj/vkl3qOs/brhgxAu37hULCOHcOLzYKF747r1oudJq0aV1TOnQzTrP8aCE/A1CnW4Dbw/l9bt20a7btDcg==
+    dependencies:
+      "@babel/core" "^7.11.0"
+      "@vue/babel-preset-app" "^4.5.4"
+      "@vue/cli-shared-utils" "^4.5.4"
+      babel-loader "^8.1.0"
+      cache-loader "^4.1.0"
+      thread-loader "^2.1.3"
+      webpack "^4.0.0"
+  
+  "@vue/cli-plugin-eslint@~4.5.0":
+    version "4.5.4"
+    resolved "https://registry.yarnpkg.com/@vue/cli-plugin-eslint/-/cli-plugin-eslint-4.5.4.tgz#0f1f307abfe1e4ad67dcb97693640942b15fae76"
+    integrity sha512-mWuhKtxMiAM70nPW/NnoWtf32YJoOPPt7SyNmsAjBKSRPcje+16Egl7BD8yuPKoF1MTkvs5CM/e7gp3AnSTFzQ==
+    dependencies:
+      "@vue/cli-shared-utils" "^4.5.4"
+      eslint-loader "^2.2.1"
+      globby "^9.2.0"
+      inquirer "^7.1.0"
+      webpack "^4.0.0"
+      yorkie "^2.0.0"
+  
+  "@vue/cli-plugin-router@^4.5.4":
+    version "4.5.4"
+    resolved "https://registry.yarnpkg.com/@vue/cli-plugin-router/-/cli-plugin-router-4.5.4.tgz#06f22408c7ed6aceddbf7302cb47a293b7af4347"
+    integrity sha512-9/qRICZbq1qucq9M9z6jYT5UWNvcTu9BgHtXgsaK9gJsdmpxDIfD0SvW9nzZaHb8xxixvDRotMM/0Juw2oCsKQ==
+    dependencies:
+      "@vue/cli-shared-utils" "^4.5.4"
+  
+  "@vue/cli-plugin-vuex@^4.5.4":
+    version "4.5.4"
+    resolved "https://registry.yarnpkg.com/@vue/cli-plugin-vuex/-/cli-plugin-vuex-4.5.4.tgz#6296e307388f61132117e0ac03188013652b0c55"
+    integrity sha512-X/F4E/dIRdiogKCdO4VGjUy5f4Fbxs7mu/gSi6Ubltle0eNE+tbBgLPH4r2g7GmHKNph4k39ikvfOMpXZcTFZg==
+  
+  "@vue/cli-service@~4.5.0":
+    version "4.5.4"
+    resolved "https://registry.yarnpkg.com/@vue/cli-service/-/cli-service-4.5.4.tgz#f903edf555d107404624de2fed5996da8cedc524"
+    integrity sha512-30zcebYno9tMvGsvZsnSPtieBvU5H3CkRW1JgiBmPG3Fcxp3BGSAy82Dl1gOUEj1VsAUqXWKMWX6frkYldi8UA==
+    dependencies:
+      "@intervolga/optimize-cssnano-plugin" "^1.0.5"
+      "@soda/friendly-errors-webpack-plugin" "^1.7.1"
+      "@soda/get-current-script" "^1.0.0"
+      "@types/minimist" "^1.2.0"
+      "@types/webpack" "^4.0.0"
+      "@types/webpack-dev-server" "^3.11.0"
+      "@vue/cli-overlay" "^4.5.4"
+      "@vue/cli-plugin-router" "^4.5.4"
+      "@vue/cli-plugin-vuex" "^4.5.4"
+      "@vue/cli-shared-utils" "^4.5.4"
+      "@vue/component-compiler-utils" "^3.1.2"
+      "@vue/preload-webpack-plugin" "^1.1.0"
+      "@vue/web-component-wrapper" "^1.2.0"
+      acorn "^7.4.0"
+      acorn-walk "^7.1.1"
+      address "^1.1.2"
+      autoprefixer "^9.8.6"
+      browserslist "^4.12.0"
+      cache-loader "^4.1.0"
+      case-sensitive-paths-webpack-plugin "^2.3.0"
+      cli-highlight "^2.1.4"
+      clipboardy "^2.3.0"
+      cliui "^6.0.0"
+      copy-webpack-plugin "^5.1.1"
+      css-loader "^3.5.3"
+      cssnano "^4.1.10"
+      debug "^4.1.1"
+      default-gateway "^5.0.5"
+      dotenv "^8.2.0"
+      dotenv-expand "^5.1.0"
+      file-loader "^4.2.0"
+      fs-extra "^7.0.1"
+      globby "^9.2.0"
+      hash-sum "^2.0.0"
+      html-webpack-plugin "^3.2.0"
+      launch-editor-middleware "^2.2.1"
+      lodash.defaultsdeep "^4.6.1"
+      lodash.mapvalues "^4.6.0"
+      lodash.transform "^4.6.0"
+      mini-css-extract-plugin "^0.9.0"
+      minimist "^1.2.5"
+      pnp-webpack-plugin "^1.6.4"
+      portfinder "^1.0.26"
+      postcss-loader "^3.0.0"
+      ssri "^7.1.0"
+      terser-webpack-plugin "^2.3.6"
+      thread-loader "^2.1.3"
+      url-loader "^2.2.0"
+      vue-loader "^15.9.2"
+      vue-style-loader "^4.1.2"
+      webpack "^4.0.0"
+      webpack-bundle-analyzer "^3.8.0"
+      webpack-chain "^6.4.0"
+      webpack-dev-server "^3.11.0"
+      webpack-merge "^4.2.2"
+    optionalDependencies:
+      vue-loader-v16 "npm:vue-loader@^16.0.0-beta.3"
+  
+  "@vue/cli-shared-utils@^4.5.4":
+    version "4.5.4"
+    resolved "https://registry.yarnpkg.com/@vue/cli-shared-utils/-/cli-shared-utils-4.5.4.tgz#ed36b2971dc02653f7f2ad4e66bbe9510e1bd414"
+    integrity sha512-7ZwAvGxl5szGuaJCc4jdPy/2Lb7oJvG847MDF+7pZ7FVl6bURwbUJjiUwL6DTxvpC4vch6B4tXfVvZFjzVP/bw==
+    dependencies:
+      "@hapi/joi" "^15.0.1"
+      chalk "^2.4.2"
+      execa "^1.0.0"
+      launch-editor "^2.2.1"
+      lru-cache "^5.1.1"
+      node-ipc "^9.1.1"
+      open "^6.3.0"
+      ora "^3.4.0"
+      read-pkg "^5.1.1"
+      request "^2.88.2"
+      semver "^6.1.0"
+      strip-ansi "^6.0.0"
+  
+  "@vue/component-compiler-utils@^3.1.0", "@vue/component-compiler-utils@^3.1.2":
+    version "3.2.0"
+    resolved "https://registry.yarnpkg.com/@vue/component-compiler-utils/-/component-compiler-utils-3.2.0.tgz#8f85182ceed28e9b3c75313de669f83166d11e5d"
+    integrity sha512-lejBLa7xAMsfiZfNp7Kv51zOzifnb29FwdnMLa96z26kXErPFioSf9BMcePVIQ6/Gc6/mC0UrPpxAWIHyae0vw==
+    dependencies:
+      consolidate "^0.15.1"
+      hash-sum "^1.0.2"
+      lru-cache "^4.1.2"
+      merge-source-map "^1.1.0"
+      postcss "^7.0.14"
+      postcss-selector-parser "^6.0.2"
+      source-map "~0.6.1"
+      vue-template-es2015-compiler "^1.9.0"
+    optionalDependencies:
+      prettier "^1.18.2"
+  
+  "@vue/preload-webpack-plugin@^1.1.0":
+    version "1.1.2"
+    resolved "https://registry.yarnpkg.com/@vue/preload-webpack-plugin/-/preload-webpack-plugin-1.1.2.tgz#ceb924b4ecb3b9c43871c7a429a02f8423e621ab"
+    integrity sha512-LIZMuJk38pk9U9Ur4YzHjlIyMuxPlACdBIHH9/nGYVTsaGKOSnSuELiE8vS9wa+dJpIYspYUOqk+L1Q4pgHQHQ==
+  
+  "@vue/web-component-wrapper@^1.2.0":
+    version "1.2.0"
+    resolved "https://registry.yarnpkg.com/@vue/web-component-wrapper/-/web-component-wrapper-1.2.0.tgz#bb0e46f1585a7e289b4ee6067dcc5a6ae62f1dd1"
+    integrity sha512-Xn/+vdm9CjuC9p3Ae+lTClNutrVhsXpzxvoTXXtoys6kVRX9FkueSUAqSWAyZntmVLlR4DosBV4pH8y5Z/HbUw==
+  
+  "@webassemblyjs/ast@1.9.0":
+    version "1.9.0"
+    resolved "https://registry.yarnpkg.com/@webassemblyjs/ast/-/ast-1.9.0.tgz#bd850604b4042459a5a41cd7d338cbed695ed964"
+    integrity sha512-C6wW5L+b7ogSDVqymbkkvuW9kruN//YisMED04xzeBBqjHa2FYnmvOlS6Xj68xWQRgWvI9cIglsjFowH/RJyEA==
+    dependencies:
+      "@webassemblyjs/helper-module-context" "1.9.0"
+      "@webassemblyjs/helper-wasm-bytecode" "1.9.0"
+      "@webassemblyjs/wast-parser" "1.9.0"
+  
+  "@webassemblyjs/floating-point-hex-parser@1.9.0":
+    version "1.9.0"
+    resolved "https://registry.yarnpkg.com/@webassemblyjs/floating-point-hex-parser/-/floating-point-hex-parser-1.9.0.tgz#3c3d3b271bddfc84deb00f71344438311d52ffb4"
+    integrity sha512-TG5qcFsS8QB4g4MhrxK5TqfdNe7Ey/7YL/xN+36rRjl/BlGE/NcBvJcqsRgCP6Z92mRE+7N50pRIi8SmKUbcQA==
+  
+  "@webassemblyjs/helper-api-error@1.9.0":
+    version "1.9.0"
+    resolved "https://registry.yarnpkg.com/@webassemblyjs/helper-api-error/-/helper-api-error-1.9.0.tgz#203f676e333b96c9da2eeab3ccef33c45928b6a2"
+    integrity sha512-NcMLjoFMXpsASZFxJ5h2HZRcEhDkvnNFOAKneP5RbKRzaWJN36NC4jqQHKwStIhGXu5mUWlUUk7ygdtrO8lbmw==
+  
+  "@webassemblyjs/helper-buffer@1.9.0":
+    version "1.9.0"
+    resolved "https://registry.yarnpkg.com/@webassemblyjs/helper-buffer/-/helper-buffer-1.9.0.tgz#a1442d269c5feb23fcbc9ef759dac3547f29de00"
+    integrity sha512-qZol43oqhq6yBPx7YM3m9Bv7WMV9Eevj6kMi6InKOuZxhw+q9hOkvq5e/PpKSiLfyetpaBnogSbNCfBwyB00CA==
+  
+  "@webassemblyjs/helper-code-frame@1.9.0":
+    version "1.9.0"
+    resolved "https://registry.yarnpkg.com/@webassemblyjs/helper-code-frame/-/helper-code-frame-1.9.0.tgz#647f8892cd2043a82ac0c8c5e75c36f1d9159f27"
+    integrity sha512-ERCYdJBkD9Vu4vtjUYe8LZruWuNIToYq/ME22igL+2vj2dQ2OOujIZr3MEFvfEaqKoVqpsFKAGsRdBSBjrIvZA==
+    dependencies:
+      "@webassemblyjs/wast-printer" "1.9.0"
+  
+  "@webassemblyjs/helper-fsm@1.9.0":
+    version "1.9.0"
+    resolved "https://registry.yarnpkg.com/@webassemblyjs/helper-fsm/-/helper-fsm-1.9.0.tgz#c05256b71244214671f4b08ec108ad63b70eddb8"
+    integrity sha512-OPRowhGbshCb5PxJ8LocpdX9Kl0uB4XsAjl6jH/dWKlk/mzsANvhwbiULsaiqT5GZGT9qinTICdj6PLuM5gslw==
+  
+  "@webassemblyjs/helper-module-context@1.9.0":
+    version "1.9.0"
+    resolved "https://registry.yarnpkg.com/@webassemblyjs/helper-module-context/-/helper-module-context-1.9.0.tgz#25d8884b76839871a08a6c6f806c3979ef712f07"
+    integrity sha512-MJCW8iGC08tMk2enck1aPW+BE5Cw8/7ph/VGZxwyvGbJwjktKkDK7vy7gAmMDx88D7mhDTCNKAW5tED+gZ0W8g==
+    dependencies:
+      "@webassemblyjs/ast" "1.9.0"
+  
+  "@webassemblyjs/helper-wasm-bytecode@1.9.0":
+    version "1.9.0"
+    resolved "https://registry.yarnpkg.com/@webassemblyjs/helper-wasm-bytecode/-/helper-wasm-bytecode-1.9.0.tgz#4fed8beac9b8c14f8c58b70d124d549dd1fe5790"
+    integrity sha512-R7FStIzyNcd7xKxCZH5lE0Bqy+hGTwS3LJjuv1ZVxd9O7eHCedSdrId/hMOd20I+v8wDXEn+bjfKDLzTepoaUw==
+  
+  "@webassemblyjs/helper-wasm-section@1.9.0":
+    version "1.9.0"
+    resolved "https://registry.yarnpkg.com/@webassemblyjs/helper-wasm-section/-/helper-wasm-section-1.9.0.tgz#5a4138d5a6292ba18b04c5ae49717e4167965346"
+    integrity sha512-XnMB8l3ek4tvrKUUku+IVaXNHz2YsJyOOmz+MMkZvh8h1uSJpSen6vYnw3IoQ7WwEuAhL8Efjms1ZWjqh2agvw==
+    dependencies:
+      "@webassemblyjs/ast" "1.9.0"
+      "@webassemblyjs/helper-buffer" "1.9.0"
+      "@webassemblyjs/helper-wasm-bytecode" "1.9.0"
+      "@webassemblyjs/wasm-gen" "1.9.0"
+  
+  "@webassemblyjs/ieee754@1.9.0":
+    version "1.9.0"
+    resolved "https://registry.yarnpkg.com/@webassemblyjs/ieee754/-/ieee754-1.9.0.tgz#15c7a0fbaae83fb26143bbacf6d6df1702ad39e4"
+    integrity sha512-dcX8JuYU/gvymzIHc9DgxTzUUTLexWwt8uCTWP3otys596io0L5aW02Gb1RjYpx2+0Jus1h4ZFqjla7umFniTg==
+    dependencies:
+      "@xtuc/ieee754" "^1.2.0"
+  
+  "@webassemblyjs/leb128@1.9.0":
+    version "1.9.0"
+    resolved "https://registry.yarnpkg.com/@webassemblyjs/leb128/-/leb128-1.9.0.tgz#f19ca0b76a6dc55623a09cffa769e838fa1e1c95"
+    integrity sha512-ENVzM5VwV1ojs9jam6vPys97B/S65YQtv/aanqnU7D8aSoHFX8GyhGg0CMfyKNIHBuAVjy3tlzd5QMMINa7wpw==
+    dependencies:
+      "@xtuc/long" "4.2.2"
+  
+  "@webassemblyjs/utf8@1.9.0":
+    version "1.9.0"
+    resolved "https://registry.yarnpkg.com/@webassemblyjs/utf8/-/utf8-1.9.0.tgz#04d33b636f78e6a6813227e82402f7637b6229ab"
+    integrity sha512-GZbQlWtopBTP0u7cHrEx+73yZKrQoBMpwkGEIqlacljhXCkVM1kMQge/Mf+csMJAjEdSwhOyLAS0AoR3AG5P8w==
+  
+  "@webassemblyjs/wasm-edit@1.9.0":
+    version "1.9.0"
+    resolved "https://registry.yarnpkg.com/@webassemblyjs/wasm-edit/-/wasm-edit-1.9.0.tgz#3fe6d79d3f0f922183aa86002c42dd256cfee9cf"
+    integrity sha512-FgHzBm80uwz5M8WKnMTn6j/sVbqilPdQXTWraSjBwFXSYGirpkSWE2R9Qvz9tNiTKQvoKILpCuTjBKzOIm0nxw==
+    dependencies:
+      "@webassemblyjs/ast" "1.9.0"
+      "@webassemblyjs/helper-buffer" "1.9.0"
+      "@webassemblyjs/helper-wasm-bytecode" "1.9.0"
+      "@webassemblyjs/helper-wasm-section" "1.9.0"
+      "@webassemblyjs/wasm-gen" "1.9.0"
+      "@webassemblyjs/wasm-opt" "1.9.0"
+      "@webassemblyjs/wasm-parser" "1.9.0"
+      "@webassemblyjs/wast-printer" "1.9.0"
+  
+  "@webassemblyjs/wasm-gen@1.9.0":
+    version "1.9.0"
+    resolved "https://registry.yarnpkg.com/@webassemblyjs/wasm-gen/-/wasm-gen-1.9.0.tgz#50bc70ec68ded8e2763b01a1418bf43491a7a49c"
+    integrity sha512-cPE3o44YzOOHvlsb4+E9qSqjc9Qf9Na1OO/BHFy4OI91XDE14MjFN4lTMezzaIWdPqHnsTodGGNP+iRSYfGkjA==
+    dependencies:
+      "@webassemblyjs/ast" "1.9.0"
+      "@webassemblyjs/helper-wasm-bytecode" "1.9.0"
+      "@webassemblyjs/ieee754" "1.9.0"
+      "@webassemblyjs/leb128" "1.9.0"
+      "@webassemblyjs/utf8" "1.9.0"
+  
+  "@webassemblyjs/wasm-opt@1.9.0":
+    version "1.9.0"
+    resolved "https://registry.yarnpkg.com/@webassemblyjs/wasm-opt/-/wasm-opt-1.9.0.tgz#2211181e5b31326443cc8112eb9f0b9028721a61"
+    integrity sha512-Qkjgm6Anhm+OMbIL0iokO7meajkzQD71ioelnfPEj6r4eOFuqm4YC3VBPqXjFyyNwowzbMD+hizmprP/Fwkl2A==
+    dependencies:
+      "@webassemblyjs/ast" "1.9.0"
+      "@webassemblyjs/helper-buffer" "1.9.0"
+      "@webassemblyjs/wasm-gen" "1.9.0"
+      "@webassemblyjs/wasm-parser" "1.9.0"
+  
+  "@webassemblyjs/wasm-parser@1.9.0":
+    version "1.9.0"
+    resolved "https://registry.yarnpkg.com/@webassemblyjs/wasm-parser/-/wasm-parser-1.9.0.tgz#9d48e44826df4a6598294aa6c87469d642fff65e"
+    integrity sha512-9+wkMowR2AmdSWQzsPEjFU7njh8HTO5MqO8vjwEHuM+AMHioNqSBONRdr0NQQ3dVQrzp0s8lTcYqzUdb7YgELA==
+    dependencies:
+      "@webassemblyjs/ast" "1.9.0"
+      "@webassemblyjs/helper-api-error" "1.9.0"
+      "@webassemblyjs/helper-wasm-bytecode" "1.9.0"
+      "@webassemblyjs/ieee754" "1.9.0"
+      "@webassemblyjs/leb128" "1.9.0"
+      "@webassemblyjs/utf8" "1.9.0"
+  
+  "@webassemblyjs/wast-parser@1.9.0":
+    version "1.9.0"
+    resolved "https://registry.yarnpkg.com/@webassemblyjs/wast-parser/-/wast-parser-1.9.0.tgz#3031115d79ac5bd261556cecc3fa90a3ef451914"
+    integrity sha512-qsqSAP3QQ3LyZjNC/0jBJ/ToSxfYJ8kYyuiGvtn/8MK89VrNEfwj7BPQzJVHi0jGTRK2dGdJ5PRqhtjzoww+bw==
+    dependencies:
+      "@webassemblyjs/ast" "1.9.0"
+      "@webassemblyjs/floating-point-hex-parser" "1.9.0"
+      "@webassemblyjs/helper-api-error" "1.9.0"
+      "@webassemblyjs/helper-code-frame" "1.9.0"
+      "@webassemblyjs/helper-fsm" "1.9.0"
+      "@xtuc/long" "4.2.2"
+  
+  "@webassemblyjs/wast-printer@1.9.0":
+    version "1.9.0"
+    resolved "https://registry.yarnpkg.com/@webassemblyjs/wast-printer/-/wast-printer-1.9.0.tgz#4935d54c85fef637b00ce9f52377451d00d47899"
+    integrity sha512-2J0nE95rHXHyQ24cWjMKJ1tqB/ds8z/cyeOZxJhcb+rW+SQASVjuznUSmdz5GpVJTzU8JkhYut0D3siFDD6wsA==
+    dependencies:
+      "@webassemblyjs/ast" "1.9.0"
+      "@webassemblyjs/wast-parser" "1.9.0"
+      "@xtuc/long" "4.2.2"
+  
+  "@xtuc/ieee754@^1.2.0":
+    version "1.2.0"
+    resolved "https://registry.yarnpkg.com/@xtuc/ieee754/-/ieee754-1.2.0.tgz#eef014a3145ae477a1cbc00cd1e552336dceb790"
+    integrity sha512-DX8nKgqcGwsc0eJSqYt5lwP4DH5FlHnmuWWBRy7X0NcaGR0ZtuyeESgMwTYVEtxmsNGY+qit4QYT/MIYTOTPeA==
+  
+  "@xtuc/long@4.2.2":
+    version "4.2.2"
+    resolved "https://registry.yarnpkg.com/@xtuc/long/-/long-4.2.2.tgz#d291c6a4e97989b5c61d9acf396ae4fe133a718d"
+    integrity sha512-NuHqBY1PB/D8xU6s/thBgOAiAP7HOYDQ32+BFZILJ8ivkUkAHQnWfn6WhL79Owj1qmUnoN/YPhktdIoucipkAQ==
+  
+  accepts@~1.3.4, accepts@~1.3.5, accepts@~1.3.7:
+    version "1.3.7"
+    resolved "https://registry.yarnpkg.com/accepts/-/accepts-1.3.7.tgz#531bc726517a3b2b41f850021c6cc15eaab507cd"
+    integrity sha512-Il80Qs2WjYlJIBNzNkK6KYqlVMTbZLXgHx2oT0pU/fjRHyEp+PEfEPY0R3WCwAGVOtauxh1hOxNgIf5bv7dQpA==
+    dependencies:
+      mime-types "~2.1.24"
+      negotiator "0.6.2"
+  
+  acorn-jsx@^5.2.0:
+    version "5.2.0"
+    resolved "https://registry.yarnpkg.com/acorn-jsx/-/acorn-jsx-5.2.0.tgz#4c66069173d6fdd68ed85239fc256226182b2ebe"
+    integrity sha512-HiUX/+K2YpkpJ+SzBffkM/AQ2YE03S0U1kjTLVpoJdhZMOWy8qvXVN9JdLqv2QsaQ6MPYQIuNmwD8zOiYUofLQ==
+  
+  acorn-walk@^7.1.1:
+    version "7.2.0"
+    resolved "https://registry.yarnpkg.com/acorn-walk/-/acorn-walk-7.2.0.tgz#0de889a601203909b0fbe07b8938dc21d2e967bc"
+    integrity sha512-OPdCF6GsMIP+Az+aWfAAOEt2/+iVDKE7oy6lJ098aoe59oAmK76qV6Gw60SbZ8jHuG2wH058GF4pLFbYamYrVA==
+  
+  acorn@^6.4.1:
+    version "6.4.1"
+    resolved "https://registry.yarnpkg.com/acorn/-/acorn-6.4.1.tgz#531e58ba3f51b9dacb9a6646ca4debf5b14ca474"
+    integrity sha512-ZVA9k326Nwrj3Cj9jlh3wGFutC2ZornPNARZwsNYqQYgN0EsV2d53w5RN/co65Ohn4sUAUtb1rSUAOD6XN9idA==
+  
+  acorn@^7.1.1, acorn@^7.4.0:
+    version "7.4.0"
+    resolved "https://registry.yarnpkg.com/acorn/-/acorn-7.4.0.tgz#e1ad486e6c54501634c6c397c5c121daa383607c"
+    integrity sha512-+G7P8jJmCHr+S+cLfQxygbWhXy+8YTVGzAkpEbcLo2mLoL7tij/VG41QSHACSf5QgYRhMZYHuNc6drJaO0Da+w==
+  
+  address@^1.1.2:
+    version "1.1.2"
+    resolved "https://registry.yarnpkg.com/address/-/address-1.1.2.tgz#bf1116c9c758c51b7a933d296b72c221ed9428b6"
+    integrity sha512-aT6camzM4xEA54YVJYSqxz1kv4IHnQZRtThJJHhUMRExaU5spC7jX5ugSwTaTgJliIgs4VhZOk7htClvQ/LmRA==
+  
+  aggregate-error@^3.0.0:
+    version "3.1.0"
+    resolved "https://registry.yarnpkg.com/aggregate-error/-/aggregate-error-3.1.0.tgz#92670ff50f5359bdb7a3e0d40d0ec30c5737687a"
+    integrity sha512-4I7Td01quW/RpocfNayFdFVk1qSuoh0E7JrbRJ16nH01HhKFQ88INq9Sd+nd72zqRySlr9BmDA8xlEJ6vJMrYA==
+    dependencies:
+      clean-stack "^2.0.0"
+      indent-string "^4.0.0"
+  
+  ajv-errors@^1.0.0:
+    version "1.0.1"
+    resolved "https://registry.yarnpkg.com/ajv-errors/-/ajv-errors-1.0.1.tgz#f35986aceb91afadec4102fbd85014950cefa64d"
+    integrity sha512-DCRfO/4nQ+89p/RK43i8Ezd41EqdGIU4ld7nGF8OQ14oc/we5rEntLCUa7+jrn3nn83BosfwZA0wb4pon2o8iQ==
+  
+  ajv-keywords@^3.1.0, ajv-keywords@^3.4.1, ajv-keywords@^3.5.2:
+    version "3.5.2"
+    resolved "https://registry.yarnpkg.com/ajv-keywords/-/ajv-keywords-3.5.2.tgz#31f29da5ab6e00d1c2d329acf7b5929614d5014d"
+    integrity sha512-5p6WTN0DdTGVQk6VjcEju19IgaHudalcfabD7yhDGeA6bcQnmL+CpveLJq/3hvfwd1aof6L386Ougkx6RfyMIQ==
+  
+  ajv@^6.1.0, ajv@^6.10.0, ajv@^6.10.2, ajv@^6.12.3, ajv@^6.12.4:
+    version "6.12.4"
+    resolved "https://registry.yarnpkg.com/ajv/-/ajv-6.12.4.tgz#0614facc4522127fa713445c6bfd3ebd376e2234"
+    integrity sha512-eienB2c9qVQs2KWexhkrdMLVDoIQCz5KSeLxwg9Lzk4DOfBtIK9PQwwufcsn1jjGuf9WZmqPMbGxOzfcuphJCQ==
+    dependencies:
+      fast-deep-equal "^3.1.1"
+      fast-json-stable-stringify "^2.0.0"
+      json-schema-traverse "^0.4.1"
+      uri-js "^4.2.2"
+  
+  alphanum-sort@^1.0.0:
+    version "1.0.2"
+    resolved "https://registry.yarnpkg.com/alphanum-sort/-/alphanum-sort-1.0.2.tgz#97a1119649b211ad33691d9f9f486a8ec9fbe0a3"
+    integrity sha1-l6ERlkmyEa0zaR2fn0hqjsn74KM=
+  
+  ansi-colors@^3.0.0:
+    version "3.2.4"
+    resolved "https://registry.yarnpkg.com/ansi-colors/-/ansi-colors-3.2.4.tgz#e3a3da4bfbae6c86a9c285625de124a234026fbf"
+    integrity sha512-hHUXGagefjN2iRrID63xckIvotOXOojhQKWIPUZ4mNUZ9nLZW+7FMNoE1lOkEhNWYsx/7ysGIuJYCiMAA9FnrA==
+  
+  ansi-escapes@^4.2.1:
+    version "4.3.1"
+    resolved "https://registry.yarnpkg.com/ansi-escapes/-/ansi-escapes-4.3.1.tgz#a5c47cc43181f1f38ffd7076837700d395522a61"
+    integrity sha512-JWF7ocqNrp8u9oqpgV+wH5ftbt+cfvv+PTjOvKLT3AdYly/LmORARfEVT1iyjwN+4MqE5UmVKoAdIBqeoCHgLA==
+    dependencies:
+      type-fest "^0.11.0"
+  
+  ansi-html@0.0.7:
+    version "0.0.7"
+    resolved "https://registry.yarnpkg.com/ansi-html/-/ansi-html-0.0.7.tgz#813584021962a9e9e6fd039f940d12f56ca7859e"
+    integrity sha1-gTWEAhliqenm/QOflA0S9WynhZ4=
+  
+  ansi-regex@^2.0.0:
+    version "2.1.1"
+    resolved "https://registry.yarnpkg.com/ansi-regex/-/ansi-regex-2.1.1.tgz#c3b33ab5ee360d86e0e628f0468ae7ef27d654df"
+    integrity sha1-w7M6te42DYbg5ijwRorn7yfWVN8=
+  
+  ansi-regex@^3.0.0:
+    version "3.0.0"
+    resolved "https://registry.yarnpkg.com/ansi-regex/-/ansi-regex-3.0.0.tgz#ed0317c322064f79466c02966bddb605ab37d998"
+    integrity sha1-7QMXwyIGT3lGbAKWa922Bas32Zg=
+  
+  ansi-regex@^4.1.0:
+    version "4.1.0"
+    resolved "https://registry.yarnpkg.com/ansi-regex/-/ansi-regex-4.1.0.tgz#8b9f8f08cf1acb843756a839ca8c7e3168c51997"
+    integrity sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==
+  
+  ansi-regex@^5.0.0:
+    version "5.0.0"
+    resolved "https://registry.yarnpkg.com/ansi-regex/-/ansi-regex-5.0.0.tgz#388539f55179bf39339c81af30a654d69f87cb75"
+    integrity sha512-bY6fj56OUQ0hU1KjFNDQuJFezqKdrAyFdIevADiqrWHwSlbmBNMHp5ak2f40Pm8JTFyM2mqxkG6ngkHO11f/lg==
+  
+  ansi-styles@^2.2.1:
+    version "2.2.1"
+    resolved "https://registry.yarnpkg.com/ansi-styles/-/ansi-styles-2.2.1.tgz#b432dd3358b634cf75e1e4664368240533c1ddbe"
+    integrity sha1-tDLdM1i2NM914eRmQ2gkBTPB3b4=
+  
+  ansi-styles@^3.2.0, ansi-styles@^3.2.1:
+    version "3.2.1"
+    resolved "https://registry.yarnpkg.com/ansi-styles/-/ansi-styles-3.2.1.tgz#41fbb20243e50b12be0f04b8dedbf07520ce841d"
+    integrity sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==
+    dependencies:
+      color-convert "^1.9.0"
+  
+  ansi-styles@^4.0.0, ansi-styles@^4.1.0:
+    version "4.2.1"
+    resolved "https://registry.yarnpkg.com/ansi-styles/-/ansi-styles-4.2.1.tgz#90ae75c424d008d2624c5bf29ead3177ebfcf359"
+    integrity sha512-9VGjrMsG1vePxcSweQsN20KY/c4zN0h9fLjqAbwbPfahM3t+NL+M9HC8xeXG2I8pX5NoamTGNuomEUFI7fcUjA==
+    dependencies:
+      "@types/color-name" "^1.1.1"
+      color-convert "^2.0.1"
+  
+  any-promise@^1.0.0:
+    version "1.3.0"
+    resolved "https://registry.yarnpkg.com/any-promise/-/any-promise-1.3.0.tgz#abc6afeedcea52e809cdc0376aed3ce39635d17f"
+    integrity sha1-q8av7tzqUugJzcA3au0845Y10X8=
+  
+  anymatch@^2.0.0:
+    version "2.0.0"
+    resolved "https://registry.yarnpkg.com/anymatch/-/anymatch-2.0.0.tgz#bcb24b4f37934d9aa7ac17b4adaf89e7c76ef2eb"
+    integrity sha512-5teOsQWABXHHBFP9y3skS5P3d/WfWXpv3FUpy+LorMrNYaT9pI4oLMQX7jzQ2KklNpGpWHzdCXTDT2Y3XGlZBw==
+    dependencies:
+      micromatch "^3.1.4"
+      normalize-path "^2.1.1"
+  
+  anymatch@~3.1.1:
+    version "3.1.1"
+    resolved "https://registry.yarnpkg.com/anymatch/-/anymatch-3.1.1.tgz#c55ecf02185e2469259399310c173ce31233b142"
+    integrity sha512-mM8522psRCqzV+6LhomX5wgp25YVibjh8Wj23I5RPkPppSVSjyKD2A2mBJmWGa+KN7f2D6LNh9jkBCeyLktzjg==
+    dependencies:
+      normalize-path "^3.0.0"
+      picomatch "^2.0.4"
+  
+  aproba@^1.1.1:
+    version "1.2.0"
+    resolved "https://registry.yarnpkg.com/aproba/-/aproba-1.2.0.tgz#6802e6264efd18c790a1b0d517f0f2627bf2c94a"
+    integrity sha512-Y9J6ZjXtoYh8RnXVCMOU/ttDmk1aBjunq9vO0ta5x85WDQiQfUF9sIPBITdbiiIVcBo03Hi3jMxigBtsddlXRw==
+  
+  arch@^2.1.1:
+    version "2.1.2"
+    resolved "https://registry.yarnpkg.com/arch/-/arch-2.1.2.tgz#0c52bbe7344bb4fa260c443d2cbad9c00ff2f0bf"
+    integrity sha512-NTBIIbAfkJeIletyABbVtdPgeKfDafR+1mZV/AyyfC1UkVkp9iUjV+wwmqtUgphHYajbI86jejBJp5e+jkGTiQ==
+  
+  argparse@^1.0.7:
+    version "1.0.10"
+    resolved "https://registry.yarnpkg.com/argparse/-/argparse-1.0.10.tgz#bcd6791ea5ae09725e17e5ad988134cd40b3d911"
+    integrity sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==
+    dependencies:
+      sprintf-js "~1.0.2"
+  
+  arr-diff@^4.0.0:
+    version "4.0.0"
+    resolved "https://registry.yarnpkg.com/arr-diff/-/arr-diff-4.0.0.tgz#d6461074febfec71e7e15235761a329a5dc7c520"
+    integrity sha1-1kYQdP6/7HHn4VI1dhoyml3HxSA=
+  
+  arr-flatten@^1.1.0:
+    version "1.1.0"
+    resolved "https://registry.yarnpkg.com/arr-flatten/-/arr-flatten-1.1.0.tgz#36048bbff4e7b47e136644316c99669ea5ae91f1"
+    integrity sha512-L3hKV5R/p5o81R7O02IGnwpDmkp6E982XhtbuwSe3O4qOtMMMtodicASA1Cny2U+aCXcNpml+m4dPsvsJ3jatg==
+  
+  arr-union@^3.1.0:
+    version "3.1.0"
+    resolved "https://registry.yarnpkg.com/arr-union/-/arr-union-3.1.0.tgz#e39b09aea9def866a8f206e288af63919bae39c4"
+    integrity sha1-45sJrqne+Gao8gbiiK9jkZuuOcQ=
+  
+  array-flatten@1.1.1:
+    version "1.1.1"
+    resolved "https://registry.yarnpkg.com/array-flatten/-/array-flatten-1.1.1.tgz#9a5f699051b1e7073328f2a008968b64ea2955d2"
+    integrity sha1-ml9pkFGx5wczKPKgCJaLZOopVdI=
+  
+  array-flatten@^2.1.0:
+    version "2.1.2"
+    resolved "https://registry.yarnpkg.com/array-flatten/-/array-flatten-2.1.2.tgz#24ef80a28c1a893617e2149b0c6d0d788293b099"
+    integrity sha512-hNfzcOV8W4NdualtqBFPyVO+54DSJuZGY9qT4pRroB6S9e3iiido2ISIC5h9R2sPJ8H3FHCIiEnsv1lPXO3KtQ==
+  
+  array-union@^1.0.1, array-union@^1.0.2:
+    version "1.0.2"
+    resolved "https://registry.yarnpkg.com/array-union/-/array-union-1.0.2.tgz#9a34410e4f4e3da23dea375be5be70f24778ec39"
+    integrity sha1-mjRBDk9OPaI96jdb5b5w8kd47Dk=
+    dependencies:
+      array-uniq "^1.0.1"
+  
+  array-uniq@^1.0.1:
+    version "1.0.3"
+    resolved "https://registry.yarnpkg.com/array-uniq/-/array-uniq-1.0.3.tgz#af6ac877a25cc7f74e058894753858dfdb24fdb6"
+    integrity sha1-r2rId6Jcx/dOBYiUdThY39sk/bY=
+  
+  array-unique@^0.3.2:
+    version "0.3.2"
+    resolved "https://registry.yarnpkg.com/array-unique/-/array-unique-0.3.2.tgz#a894b75d4bc4f6cd679ef3244a9fd8f46ae2d428"
+    integrity sha1-qJS3XUvE9s1nnvMkSp/Y9Gri1Cg=
+  
+  asn1.js@^5.2.0:
+    version "5.4.1"
+    resolved "https://registry.yarnpkg.com/asn1.js/-/asn1.js-5.4.1.tgz#11a980b84ebb91781ce35b0fdc2ee294e3783f07"
+    integrity sha512-+I//4cYPccV8LdmBLiX8CYvf9Sp3vQsrqu2QNXRcrbiWvcx/UdlFiqUJJzxRQxgsZmvhXhn4cSKeSmoFjVdupA==
+    dependencies:
+      bn.js "^4.0.0"
+      inherits "^2.0.1"
+      minimalistic-assert "^1.0.0"
+      safer-buffer "^2.1.0"
+  
+  asn1@~0.2.3:
+    version "0.2.4"
+    resolved "https://registry.yarnpkg.com/asn1/-/asn1-0.2.4.tgz#8d2475dfab553bb33e77b54e59e880bb8ce23136"
+    integrity sha512-jxwzQpLQjSmWXgwaCZE9Nz+glAG01yF1QnWgbhGwHI5A6FRIEY6IVqtHhIepHqI7/kyEyQEagBC5mBEFlIYvdg==
+    dependencies:
+      safer-buffer "~2.1.0"
+  
+  assert-plus@1.0.0, assert-plus@^1.0.0:
+    version "1.0.0"
+    resolved "https://registry.yarnpkg.com/assert-plus/-/assert-plus-1.0.0.tgz#f12e0f3c5d77b0b1cdd9146942e4e96c1e4dd525"
+    integrity sha1-8S4PPF13sLHN2RRpQuTpbB5N1SU=
+  
+  assert@^1.1.1:
+    version "1.5.0"
+    resolved "https://registry.yarnpkg.com/assert/-/assert-1.5.0.tgz#55c109aaf6e0aefdb3dc4b71240c70bf574b18eb"
+    integrity sha512-EDsgawzwoun2CZkCgtxJbv392v4nbk9XDD06zI+kQYoBM/3RBWLlEyJARDOmhAAosBjWACEkKL6S+lIZtcAubA==
+    dependencies:
+      object-assign "^4.1.1"
+      util "0.10.3"
+  
+  assign-symbols@^1.0.0:
+    version "1.0.0"
+    resolved "https://registry.yarnpkg.com/assign-symbols/-/assign-symbols-1.0.0.tgz#59667f41fadd4f20ccbc2bb96b8d4f7f78ec0367"
+    integrity sha1-WWZ/QfrdTyDMvCu5a41Pf3jsA2c=
+  
+  astral-regex@^1.0.0:
+    version "1.0.0"
+    resolved "https://registry.yarnpkg.com/astral-regex/-/astral-regex-1.0.0.tgz#6c8c3fb827dd43ee3918f27b82782ab7658a6fd9"
+    integrity sha512-+Ryf6g3BKoRc7jfp7ad8tM4TtMiaWvbF/1/sQcZPkkS7ag3D5nMBCe2UfOTONtAkaG0tO0ij3C5Lwmf1EiyjHg==
+  
+  async-each@^1.0.1:
+    version "1.0.3"
+    resolved "https://registry.yarnpkg.com/async-each/-/async-each-1.0.3.tgz#b727dbf87d7651602f06f4d4ac387f47d91b0cbf"
+    integrity sha512-z/WhQ5FPySLdvREByI2vZiTWwCnF0moMJ1hK9YQwDTHKh6I7/uSckMetoRGb5UBZPC1z0jlw+n/XCgjeH7y1AQ==
+  
+  async-limiter@~1.0.0:
+    version "1.0.1"
+    resolved "https://registry.yarnpkg.com/async-limiter/-/async-limiter-1.0.1.tgz#dd379e94f0db8310b08291f9d64c3209766617fd"
+    integrity sha512-csOlWGAcRFJaI6m+F2WKdnMKr4HhdhFVBk0H/QbJFMCr+uO2kwohwXQPxw/9OCxp05r5ghVBFSyioixx3gfkNQ==
+  
+  async@^2.6.2:
+    version "2.6.3"
+    resolved "https://registry.yarnpkg.com/async/-/async-2.6.3.tgz#d72625e2344a3656e3a3ad4fa749fa83299d82ff"
+    integrity sha512-zflvls11DCy+dQWzTW2dzuilv8Z5X/pjfmZOWba6TNIVDm+2UDaJmXSOXlasHKfNBs8oo3M0aT50fDEWfKZjXg==
+    dependencies:
+      lodash "^4.17.14"
+  
+  asynckit@^0.4.0:
+    version "0.4.0"
+    resolved "https://registry.yarnpkg.com/asynckit/-/asynckit-0.4.0.tgz#c79ed97f7f34cb8f2ba1bc9790bcc366474b4b79"
+    integrity sha1-x57Zf380y48robyXkLzDZkdLS3k=
+  
+  atob@^2.1.2:
+    version "2.1.2"
+    resolved "https://registry.yarnpkg.com/atob/-/atob-2.1.2.tgz#6d9517eb9e030d2436666651e86bd9f6f13533c9"
+    integrity sha512-Wm6ukoaOGJi/73p/cl2GvLjTI5JM1k/O14isD73YML8StrH/7/lRFgmg8nICZgD3bZZvjwCGxtMOD3wWNAu8cg==
+  
+  autoprefixer@^9.8.6:
+    version "9.8.6"
+    resolved "https://registry.yarnpkg.com/autoprefixer/-/autoprefixer-9.8.6.tgz#3b73594ca1bf9266320c5acf1588d74dea74210f"
+    integrity sha512-XrvP4VVHdRBCdX1S3WXVD8+RyG9qeb1D5Sn1DeLiG2xfSpzellk5k54xbUERJ3M5DggQxes39UGOTP8CFrEGbg==
+    dependencies:
+      browserslist "^4.12.0"
+      caniuse-lite "^1.0.30001109"
+      colorette "^1.2.1"
+      normalize-range "^0.1.2"
+      num2fraction "^1.2.2"
+      postcss "^7.0.32"
+      postcss-value-parser "^4.1.0"
+  
+  aws-sign2@~0.7.0:
+    version "0.7.0"
+    resolved "https://registry.yarnpkg.com/aws-sign2/-/aws-sign2-0.7.0.tgz#b46e890934a9591f2d2f6f86d7e6a9f1b3fe76a8"
+    integrity sha1-tG6JCTSpWR8tL2+G1+ap8bP+dqg=
+  
+  aws4@^1.8.0:
+    version "1.10.1"
+    resolved "https://registry.yarnpkg.com/aws4/-/aws4-1.10.1.tgz#e1e82e4f3e999e2cfd61b161280d16a111f86428"
+    integrity sha512-zg7Hz2k5lI8kb7U32998pRRFin7zJlkfezGJjUc2heaD4Pw2wObakCDVzkKztTm/Ln7eiVvYsjqak0Ed4LkMDA==
+  
+  babel-eslint@^10.1.0:
+    version "10.1.0"
+    resolved "https://registry.yarnpkg.com/babel-eslint/-/babel-eslint-10.1.0.tgz#6968e568a910b78fb3779cdd8b6ac2f479943232"
+    integrity sha512-ifWaTHQ0ce+448CYop8AdrQiBsGrnC+bMgfyKFdi6EsPLTAWG+QfyDeM6OH+FmWnKvEq5NnBMLvlBUPKQZoDSg==
+    dependencies:
+      "@babel/code-frame" "^7.0.0"
+      "@babel/parser" "^7.7.0"
+      "@babel/traverse" "^7.7.0"
+      "@babel/types" "^7.7.0"
+      eslint-visitor-keys "^1.0.0"
+      resolve "^1.12.0"
+  
+  babel-loader@^8.1.0:
+    version "8.1.0"
+    resolved "https://registry.yarnpkg.com/babel-loader/-/babel-loader-8.1.0.tgz#c611d5112bd5209abe8b9fa84c3e4da25275f1c3"
+    integrity sha512-7q7nC1tYOrqvUrN3LQK4GwSk/TQorZSOlO9C+RZDZpODgyN4ZlCqE5q9cDsyWOliN+aU9B4JX01xK9eJXowJLw==
+    dependencies:
+      find-cache-dir "^2.1.0"
+      loader-utils "^1.4.0"
+      mkdirp "^0.5.3"
+      pify "^4.0.1"
+      schema-utils "^2.6.5"
+  
+  babel-plugin-dynamic-import-node@^2.3.3:
+    version "2.3.3"
+    resolved "https://registry.yarnpkg.com/babel-plugin-dynamic-import-node/-/babel-plugin-dynamic-import-node-2.3.3.tgz#84fda19c976ec5c6defef57f9427b3def66e17a3"
+    integrity sha512-jZVI+s9Zg3IqA/kdi0i6UDCybUI3aSBLnglhYbSSjKlV7yF1F/5LWv8MakQmvYpnbJDS6fcBL2KzHSxNCMtWSQ==
+    dependencies:
+      object.assign "^4.1.0"
+  
+  balanced-match@^1.0.0:
+    version "1.0.0"
+    resolved "https://registry.yarnpkg.com/balanced-match/-/balanced-match-1.0.0.tgz#89b4d199ab2bee49de164ea02b89ce462d71b767"
+    integrity sha1-ibTRmasr7kneFk6gK4nORi1xt2c=
+  
+  base64-js@^1.0.2:
+    version "1.3.1"
+    resolved "https://registry.yarnpkg.com/base64-js/-/base64-js-1.3.1.tgz#58ece8cb75dd07e71ed08c736abc5fac4dbf8df1"
+    integrity sha512-mLQ4i2QO1ytvGWFWmcngKO//JXAQueZvwEKtjgQFM4jIK0kU+ytMfplL8j+n5mspOfjHwoAg+9yhb7BwAHm36g==
+  
+  base@^0.11.1:
+    version "0.11.2"
+    resolved "https://registry.yarnpkg.com/base/-/base-0.11.2.tgz#7bde5ced145b6d551a90db87f83c558b4eb48a8f"
+    integrity sha512-5T6P4xPgpp0YDFvSWwEZ4NoE3aM4QBQXDzmVbraCkFj8zHM+mba8SyqB5DbZWyR7mYHo6Y7BdQo3MoA4m0TeQg==
+    dependencies:
+      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"
+  
+  batch@0.6.1:
+    version "0.6.1"
+    resolved "https://registry.yarnpkg.com/batch/-/batch-0.6.1.tgz#dc34314f4e679318093fc760272525f94bf25c16"
+    integrity sha1-3DQxT05nkxgJP8dgJyUl+UvyXBY=
+  
+  bcrypt-pbkdf@^1.0.0:
+    version "1.0.2"
+    resolved "https://registry.yarnpkg.com/bcrypt-pbkdf/-/bcrypt-pbkdf-1.0.2.tgz#a4301d389b6a43f9b67ff3ca11a3f6637e360e9e"
+    integrity sha1-pDAdOJtqQ/m2f/PKEaP2Y342Dp4=
+    dependencies:
+      tweetnacl "^0.14.3"
+  
+  bfj@^6.1.1:
+    version "6.1.2"
+    resolved "https://registry.yarnpkg.com/bfj/-/bfj-6.1.2.tgz#325c861a822bcb358a41c78a33b8e6e2086dde7f"
+    integrity sha512-BmBJa4Lip6BPRINSZ0BPEIfB1wUY/9rwbwvIHQA1KjX9om29B6id0wnWXq7m3bn5JrUVjeOTnVuhPT1FiHwPGw==
+    dependencies:
+      bluebird "^3.5.5"
+      check-types "^8.0.3"
+      hoopy "^0.1.4"
+      tryer "^1.0.1"
+  
+  big.js@^3.1.3:
+    version "3.2.0"
+    resolved "https://registry.yarnpkg.com/big.js/-/big.js-3.2.0.tgz#a5fc298b81b9e0dca2e458824784b65c52ba588e"
+    integrity sha512-+hN/Zh2D08Mx65pZ/4g5bsmNiZUuChDiQfTUQ7qJr4/kuopCr88xZsAXv6mBoZEsUI4OuGHlX59qE94K2mMW8Q==
+  
+  big.js@^5.2.2:
+    version "5.2.2"
+    resolved "https://registry.yarnpkg.com/big.js/-/big.js-5.2.2.tgz#65f0af382f578bcdc742bd9c281e9cb2d7768328"
+    integrity sha512-vyL2OymJxmarO8gxMr0mhChsO9QGwhynfuu4+MHTAW6czfq9humCB7rKpUjDd9YUiDPU4mzpyupFSvOClAwbmQ==
+  
+  binary-extensions@^1.0.0:
+    version "1.13.1"
+    resolved "https://registry.yarnpkg.com/binary-extensions/-/binary-extensions-1.13.1.tgz#598afe54755b2868a5330d2aff9d4ebb53209b65"
+    integrity sha512-Un7MIEDdUC5gNpcGDV97op1Ywk748MpHcFTHoYs6qnj1Z3j7I53VG3nwZhKzoBZmbdRNnb6WRdFlwl7tSDuZGw==
+  
+  binary-extensions@^2.0.0:
+    version "2.1.0"
+    resolved "https://registry.yarnpkg.com/binary-extensions/-/binary-extensions-2.1.0.tgz#30fa40c9e7fe07dbc895678cd287024dea241dd9"
+    integrity sha512-1Yj8h9Q+QDF5FzhMs/c9+6UntbD5MkRfRwac8DoEm9ZfUBZ7tZ55YcGVAzEe4bXsdQHEk+s9S5wsOKVdZrw0tQ==
+  
+  bindings@^1.5.0:
+    version "1.5.0"
+    resolved "https://registry.yarnpkg.com/bindings/-/bindings-1.5.0.tgz#10353c9e945334bc0511a6d90b38fbc7c9c504df"
+    integrity sha512-p2q/t/mhvuOj/UeLlV6566GD/guowlr0hHxClI0W9m7MWYkL1F0hLo+0Aexs9HSPCtR1SXQ0TD3MMKrXZajbiQ==
+    dependencies:
+      file-uri-to-path "1.0.0"
+  
+  bluebird@^3.1.1, bluebird@^3.5.5:
+    version "3.7.2"
+    resolved "https://registry.yarnpkg.com/bluebird/-/bluebird-3.7.2.tgz#9f229c15be272454ffa973ace0dbee79a1b0c36f"
+    integrity sha512-XpNj6GDQzdfW+r2Wnn7xiSAd7TM3jzkxGXBGTtWKuSXv1xUV+azxAm8jdWZN06QTQk+2N2XB9jRDkvbmQmcRtg==
+  
+  bn.js@^4.0.0, bn.js@^4.1.0, bn.js@^4.4.0:
+    version "4.11.9"
+    resolved "https://registry.yarnpkg.com/bn.js/-/bn.js-4.11.9.tgz#26d556829458f9d1e81fc48952493d0ba3507828"
+    integrity sha512-E6QoYqCKZfgatHTdHzs1RRKP7ip4vvm+EyRUeE2RF0NblwVvb0p6jSVeNTOFxPn26QXN2o6SMfNxKp6kU8zQaw==
+  
+  bn.js@^5.1.1:
+    version "5.1.3"
+    resolved "https://registry.yarnpkg.com/bn.js/-/bn.js-5.1.3.tgz#beca005408f642ebebea80b042b4d18d2ac0ee6b"
+    integrity sha512-GkTiFpjFtUzU9CbMeJ5iazkCzGL3jrhzerzZIuqLABjbwRaFt33I9tUdSNryIptM+RxDet6OKm2WnLXzW51KsQ==
+  
+  body-parser@1.19.0:
+    version "1.19.0"
+    resolved "https://registry.yarnpkg.com/body-parser/-/body-parser-1.19.0.tgz#96b2709e57c9c4e09a6fd66a8fd979844f69f08a"
+    integrity sha512-dhEPs72UPbDnAQJ9ZKMNTP6ptJaionhP5cBb541nXPlW60Jepo9RV/a4fX4XWW9CuFNK22krhrj1+rgzifNCsw==
+    dependencies:
+      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"
+  
+  bonjour@^3.5.0:
+    version "3.5.0"
+    resolved "https://registry.yarnpkg.com/bonjour/-/bonjour-3.5.0.tgz#8e890a183d8ee9a2393b3844c691a42bcf7bc9f5"
+    integrity sha1-jokKGD2O6aI5OzhExpGkK897yfU=
+    dependencies:
+      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@^1.0.0, boolbase@~1.0.0:
+    version "1.0.0"
+    resolved "https://registry.yarnpkg.com/boolbase/-/boolbase-1.0.0.tgz#68dff5fbe60c51eb37725ea9e3ed310dcc1e776e"
+    integrity sha1-aN/1++YMUes3cl6p4+0xDcwed24=
+  
+  bootstrap-vue@^2.16.0:
+    version "2.16.0"
+    resolved "https://registry.yarnpkg.com/bootstrap-vue/-/bootstrap-vue-2.16.0.tgz#07e7032ec9ffdd576470dc437da54f398ec16ba5"
+    integrity sha512-gLETwPmeRHCe5WHmhGxzb5PtTEuKqQPGl0TFvZ2Odbkg/7UuIHdqIexrJRerpnomP4ZzDQ+qYGL91Ls9lcQsJQ==
+    dependencies:
+      "@nuxt/opencollective" "^0.3.0"
+      bootstrap ">=4.5.0 <5.0.0"
+      popper.js "^1.16.1"
+      portal-vue "^2.1.7"
+      vue-functional-data-merge "^3.1.0"
+  
+  "bootstrap@>=4.5.0 <5.0.0":
+    version "4.5.2"
+    resolved "https://registry.yarnpkg.com/bootstrap/-/bootstrap-4.5.2.tgz#a85c4eda59155f0d71186b6e6ad9b875813779ab"
+    integrity sha512-vlGn0bcySYl/iV+BGA544JkkZP5LB3jsmkeKLFQakCOwCM3AOk7VkldBz4jrzSe+Z0Ezn99NVXa1o45cQY4R6A==
+  
+  brace-expansion@^1.1.7:
+    version "1.1.11"
+    resolved "https://registry.yarnpkg.com/brace-expansion/-/brace-expansion-1.1.11.tgz#3c7fcbf529d87226f3d2f52b966ff5271eb441dd"
+    integrity sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==
+    dependencies:
+      balanced-match "^1.0.0"
+      concat-map "0.0.1"
+  
+  braces@^2.3.1, braces@^2.3.2:
+    version "2.3.2"
+    resolved "https://registry.yarnpkg.com/braces/-/braces-2.3.2.tgz#5979fd3f14cd531565e5fa2df1abfff1dfaee729"
+    integrity sha512-aNdbnj9P8PjdXU4ybaWLK2IF3jc/EoDYbC7AazW6to3TRsfXxscC9UXOB5iDiEQrkyIbWp2SLQda4+QAa7nc3w==
+    dependencies:
+      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"
+  
+  braces@~3.0.2:
+    version "3.0.2"
+    resolved "https://registry.yarnpkg.com/braces/-/braces-3.0.2.tgz#3454e1a462ee8d599e236df336cd9ea4f8afe107"
+    integrity sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==
+    dependencies:
+      fill-range "^7.0.1"
+  
+  brorand@^1.0.1:
+    version "1.1.0"
+    resolved "https://registry.yarnpkg.com/brorand/-/brorand-1.1.0.tgz#12c25efe40a45e3c323eb8675a0a0ce57b22371f"
+    integrity sha1-EsJe/kCkXjwyPrhnWgoM5XsiNx8=
+  
+  browserify-aes@^1.0.0, browserify-aes@^1.0.4:
+    version "1.2.0"
+    resolved "https://registry.yarnpkg.com/browserify-aes/-/browserify-aes-1.2.0.tgz#326734642f403dabc3003209853bb70ad428ef48"
+    integrity sha512-+7CHXqGuspUn/Sl5aO7Ea0xWGAtETPXNSAjHo48JfLdPWcMng33Xe4znFvQweqc/uzk5zSOI3H52CYnjCfb5hA==
+    dependencies:
+      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@^1.0.0:
+    version "1.0.1"
+    resolved "https://registry.yarnpkg.com/browserify-cipher/-/browserify-cipher-1.0.1.tgz#8d6474c1b870bfdabcd3bcfcc1934a10e94f15f0"
+    integrity sha512-sPhkz0ARKbf4rRQt2hTpAHqn47X3llLkUGn+xEJzLjwY8LRs2p0v7ljvI5EyoRO/mexrNunNECisZs+gw2zz1w==
+    dependencies:
+      browserify-aes "^1.0.4"
+      browserify-des "^1.0.0"
+      evp_bytestokey "^1.0.0"
+  
+  browserify-des@^1.0.0:
+    version "1.0.2"
+    resolved "https://registry.yarnpkg.com/browserify-des/-/browserify-des-1.0.2.tgz#3af4f1f59839403572f1c66204375f7a7f703e9c"
+    integrity sha512-BioO1xf3hFwz4kc6iBhI3ieDFompMhrMlnDFC4/0/vd5MokpuAc3R+LYbwTA9A5Yc9pq9UYPqffKpW2ObuwX5A==
+    dependencies:
+      cipher-base "^1.0.1"
+      des.js "^1.0.0"
+      inherits "^2.0.1"
+      safe-buffer "^5.1.2"
+  
+  browserify-rsa@^4.0.0, browserify-rsa@^4.0.1:
+    version "4.0.1"
+    resolved "https://registry.yarnpkg.com/browserify-rsa/-/browserify-rsa-4.0.1.tgz#21e0abfaf6f2029cf2fafb133567a701d4135524"
+    integrity sha1-IeCr+vbyApzy+vsTNWenAdQTVSQ=
+    dependencies:
+      bn.js "^4.1.0"
+      randombytes "^2.0.1"
+  
+  browserify-sign@^4.0.0:
+    version "4.2.1"
+    resolved "https://registry.yarnpkg.com/browserify-sign/-/browserify-sign-4.2.1.tgz#eaf4add46dd54be3bb3b36c0cf15abbeba7956c3"
+    integrity sha512-/vrA5fguVAKKAVTNJjgSm1tRQDHUU6DbwO9IROu/0WAzC8PKhucDSh18J0RMvVeHAn5puMd+QHC2erPRNf8lmg==
+    dependencies:
+      bn.js "^5.1.1"
+      browserify-rsa "^4.0.1"
+      create-hash "^1.2.0"
+      create-hmac "^1.1.7"
+      elliptic "^6.5.3"
+      inherits "^2.0.4"
+      parse-asn1 "^5.1.5"
+      readable-stream "^3.6.0"
+      safe-buffer "^5.2.0"
+  
+  browserify-zlib@^0.2.0:
+    version "0.2.0"
+    resolved "https://registry.yarnpkg.com/browserify-zlib/-/browserify-zlib-0.2.0.tgz#2869459d9aa3be245fe8fe2ca1f46e2e7f54d73f"
+    integrity sha512-Z942RysHXmJrhqk88FmKBVq/v5tqmSkDz7p54G/MGyjMnCFFnC79XWNbg+Vta8W6Wb2qtSZTSxIGkJrRpCFEiA==
+    dependencies:
+      pako "~1.0.5"
+  
+  browserslist@^4.0.0, browserslist@^4.12.0, browserslist@^4.8.5:
+    version "4.14.1"
+    resolved "https://registry.yarnpkg.com/browserslist/-/browserslist-4.14.1.tgz#cb2b490ba881d45dc3039078c7ed04411eaf3fa3"
+    integrity sha512-zyBTIHydW37pnb63c7fHFXUG6EcqWOqoMdDx6cdyaDFriZ20EoVxcE95S54N+heRqY8m8IUgB5zYta/gCwSaaA==
+    dependencies:
+      caniuse-lite "^1.0.30001124"
+      electron-to-chromium "^1.3.562"
+      escalade "^3.0.2"
+      node-releases "^1.1.60"
+  
+  buffer-from@^1.0.0:
+    version "1.1.1"
+    resolved "https://registry.yarnpkg.com/buffer-from/-/buffer-from-1.1.1.tgz#32713bc028f75c02fdb710d7c7bcec1f2c6070ef"
+    integrity sha512-MQcXEUbCKtEo7bhqEs6560Hyd4XaovZlO/k9V3hjVUF/zwW7KBVdSK4gIt/bzwS9MbR5qob+F5jusZsb0YQK2A==
+  
+  buffer-indexof@^1.0.0:
+    version "1.1.1"
+    resolved "https://registry.yarnpkg.com/buffer-indexof/-/buffer-indexof-1.1.1.tgz#52fabcc6a606d1a00302802648ef68f639da268c"
+    integrity sha512-4/rOEg86jivtPTeOUUT61jJO1Ya1TrR/OkqCSZDyq84WJh3LuuiphBYJN+fm5xufIk4XAFcEwte/8WzC8If/1g==
+  
+  buffer-json@^2.0.0:
+    version "2.0.0"
+    resolved "https://registry.yarnpkg.com/buffer-json/-/buffer-json-2.0.0.tgz#f73e13b1e42f196fe2fd67d001c7d7107edd7c23"
+    integrity sha512-+jjPFVqyfF1esi9fvfUs3NqM0pH1ziZ36VP4hmA/y/Ssfo/5w5xHKfTw9BwQjoJ1w/oVtpLomqwUHKdefGyuHw==
+  
+  buffer-xor@^1.0.3:
+    version "1.0.3"
+    resolved "https://registry.yarnpkg.com/buffer-xor/-/buffer-xor-1.0.3.tgz#26e61ed1422fb70dd42e6e36729ed51d855fe8d9"
+    integrity sha1-JuYe0UIvtw3ULm42cp7VHYVf6Nk=
+  
+  buffer@^4.3.0:
+    version "4.9.2"
+    resolved "https://registry.yarnpkg.com/buffer/-/buffer-4.9.2.tgz#230ead344002988644841ab0244af8c44bbe3ef8"
+    integrity sha512-xq+q3SRMOxGivLhBNaUdC64hDTQwejJ+H0T/NB1XMtTVEwNTrfFF3gAxiyW0Bu/xWEGhjVKgUcMhCrUy2+uCWg==
+    dependencies:
+      base64-js "^1.0.2"
+      ieee754 "^1.1.4"
+      isarray "^1.0.0"
+  
+  builtin-status-codes@^3.0.0:
+    version "3.0.0"
+    resolved "https://registry.yarnpkg.com/builtin-status-codes/-/builtin-status-codes-3.0.0.tgz#85982878e21b98e1c66425e03d0174788f569ee8"
+    integrity sha1-hZgoeOIbmOHGZCXgPQF0eI9Wnug=
+  
+  bytes@3.0.0:
+    version "3.0.0"
+    resolved "https://registry.yarnpkg.com/bytes/-/bytes-3.0.0.tgz#d32815404d689699f85a4ea4fa8755dd13a96048"
+    integrity sha1-0ygVQE1olpn4Wk6k+odV3ROpYEg=
+  
+  bytes@3.1.0:
+    version "3.1.0"
+    resolved "https://registry.yarnpkg.com/bytes/-/bytes-3.1.0.tgz#f6cf7933a360e0588fa9fde85651cdc7f805d1f6"
+    integrity sha512-zauLjrfCG+xvoyaqLoV8bLVXXNGC4JqlxFCutSDWA6fJrTo2ZuvLYTqZ7aHBLZSMOopbzwv8f+wZcVzfVTI2Dg==
+  
+  cacache@^12.0.2, cacache@^12.0.3:
+    version "12.0.4"
+    resolved "https://registry.yarnpkg.com/cacache/-/cacache-12.0.4.tgz#668bcbd105aeb5f1d92fe25570ec9525c8faa40c"
+    integrity sha512-a0tMB40oefvuInr4Cwb3GerbL9xTj1D5yg0T5xrjGCGyfvbxseIXX7BAO/u/hIXdafzOI5JC3wDwHyf24buOAQ==
+    dependencies:
+      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"
+  
+  cacache@^13.0.1:
+    version "13.0.1"
+    resolved "https://registry.yarnpkg.com/cacache/-/cacache-13.0.1.tgz#a8000c21697089082f85287a1aec6e382024a71c"
+    integrity sha512-5ZvAxd05HDDU+y9BVvcqYu2LLXmPnQ0hW62h32g4xBTgL/MppR4/04NHfj/ycM2y6lmTnbw6HVi+1eN0Psba6w==
+    dependencies:
+      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"
+  
+  cache-base@^1.0.1:
+    version "1.0.1"
+    resolved "https://registry.yarnpkg.com/cache-base/-/cache-base-1.0.1.tgz#0a7f46416831c8b662ee36fe4e7c59d76f666ab2"
+    integrity sha512-AKcdTnFSWATd5/GCPRxr2ChwIJ85CeyrEyjRHlKxQ56d4XJMGym0uAiKn0xbLOGOl3+yRpOTi484dVCEc5AUzQ==
+    dependencies:
+      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"
+  
+  cache-loader@^4.1.0:
+    version "4.1.0"
+    resolved "https://registry.yarnpkg.com/cache-loader/-/cache-loader-4.1.0.tgz#9948cae353aec0a1fcb1eafda2300816ec85387e"
+    integrity sha512-ftOayxve0PwKzBF/GLsZNC9fJBXl8lkZE3TOsjkboHfVHVkL39iUEs1FO07A33mizmci5Dudt38UZrrYXDtbhw==
+    dependencies:
+      buffer-json "^2.0.0"
+      find-cache-dir "^3.0.0"
+      loader-utils "^1.2.3"
+      mkdirp "^0.5.1"
+      neo-async "^2.6.1"
+      schema-utils "^2.0.0"
+  
+  call-me-maybe@^1.0.1:
+    version "1.0.1"
+    resolved "https://registry.yarnpkg.com/call-me-maybe/-/call-me-maybe-1.0.1.tgz#26d208ea89e37b5cbde60250a15f031c16a4d66b"
+    integrity sha1-JtII6onje1y95gJQoV8DHBak1ms=
+  
+  caller-callsite@^2.0.0:
+    version "2.0.0"
+    resolved "https://registry.yarnpkg.com/caller-callsite/-/caller-callsite-2.0.0.tgz#847e0fce0a223750a9a027c54b33731ad3154134"
+    integrity sha1-hH4PzgoiN1CpoCfFSzNzGtMVQTQ=
+    dependencies:
+      callsites "^2.0.0"
+  
+  caller-path@^2.0.0:
+    version "2.0.0"
+    resolved "https://registry.yarnpkg.com/caller-path/-/caller-path-2.0.0.tgz#468f83044e369ab2010fac5f06ceee15bb2cb1f4"
+    integrity sha1-Ro+DBE42mrIBD6xfBs7uFbsssfQ=
+    dependencies:
+      caller-callsite "^2.0.0"
+  
+  callsites@^2.0.0:
+    version "2.0.0"
+    resolved "https://registry.yarnpkg.com/callsites/-/callsites-2.0.0.tgz#06eb84f00eea413da86affefacbffb36093b3c50"
+    integrity sha1-BuuE8A7qQT2oav/vrL/7Ngk7PFA=
+  
+  callsites@^3.0.0:
+    version "3.1.0"
+    resolved "https://registry.yarnpkg.com/callsites/-/callsites-3.1.0.tgz#b3630abd8943432f54b3f0519238e33cd7df2f73"
+    integrity sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==
+  
+  camel-case@3.0.x:
+    version "3.0.0"
+    resolved "https://registry.yarnpkg.com/camel-case/-/camel-case-3.0.0.tgz#ca3c3688a4e9cf3a4cda777dc4dcbc713249cf73"
+    integrity sha1-yjw2iKTpzzpM2nd9xNy8cTJJz3M=
+    dependencies:
+      no-case "^2.2.0"
+      upper-case "^1.1.1"
+  
+  camelcase@^5.0.0, camelcase@^5.3.1:
+    version "5.3.1"
+    resolved "https://registry.yarnpkg.com/camelcase/-/camelcase-5.3.1.tgz#e3c9b31569e106811df242f715725a1f4c494320"
+    integrity sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==
+  
+  camelcase@^6.0.0:
+    version "6.0.0"
+    resolved "https://registry.yarnpkg.com/camelcase/-/camelcase-6.0.0.tgz#5259f7c30e35e278f1bdc2a4d91230b37cad981e"
+    integrity sha512-8KMDF1Vz2gzOq54ONPJS65IvTUaB1cHJ2DMM7MbPmLZljDH1qpzzLsWdiN9pHh6qvkRVDTi/07+eNGch/oLU4w==
+  
+  caniuse-api@^3.0.0:
+    version "3.0.0"
+    resolved "https://registry.yarnpkg.com/caniuse-api/-/caniuse-api-3.0.0.tgz#5e4d90e2274961d46291997df599e3ed008ee4c0"
+    integrity sha512-bsTwuIg/BZZK/vreVTYYbSWoe2F+71P7K5QGEX+pT250DZbfU1MQ5prOKpPR+LL6uWKK3KMwMCAS74QB3Um1uw==
+    dependencies:
+      browserslist "^4.0.0"
+      caniuse-lite "^1.0.0"
+      lodash.memoize "^4.1.2"
+      lodash.uniq "^4.5.0"
+  
+  caniuse-lite@^1.0.0, caniuse-lite@^1.0.30001109, caniuse-lite@^1.0.30001124:
+    version "1.0.30001124"
+    resolved "https://registry.yarnpkg.com/caniuse-lite/-/caniuse-lite-1.0.30001124.tgz#5d9998190258e11630d674fc50ea8e579ae0ced2"
+    integrity sha512-zQW8V3CdND7GHRH6rxm6s59Ww4g/qGWTheoboW9nfeMg7sUoopIfKCcNZUjwYRCOrvereh3kwDpZj4VLQ7zGtA==
+  
+  case-sensitive-paths-webpack-plugin@^2.3.0:
+    version "2.3.0"
+    resolved "https://registry.yarnpkg.com/case-sensitive-paths-webpack-plugin/-/case-sensitive-paths-webpack-plugin-2.3.0.tgz#23ac613cc9a856e4f88ff8bb73bbb5e989825cf7"
+    integrity sha512-/4YgnZS8y1UXXmC02xD5rRrBEu6T5ub+mQHLNRj0fzTRbgdBYhsNo2V5EqwgqrExjxsjtF/OpAKAMkKsxbD5XQ==
+  
+  caseless@~0.12.0:
+    version "0.12.0"
+    resolved "https://registry.yarnpkg.com/caseless/-/caseless-0.12.0.tgz#1b681c21ff84033c826543090689420d187151dc"
+    integrity sha1-G2gcIf+EAzyCZUMJBolCDRhxUdw=
+  
+  chalk@^1.1.3:
+    version "1.1.3"
+    resolved "https://registry.yarnpkg.com/chalk/-/chalk-1.1.3.tgz#a8115c55e4a702fe4d150abd3872822a7e09fc98"
+    integrity sha1-qBFcVeSnAv5NFQq9OHKCKn4J/Jg=
+    dependencies:
+      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"
+  
+  chalk@^2.0.0, chalk@^2.0.1, chalk@^2.1.0, chalk@^2.3.0, chalk@^2.4.1, chalk@^2.4.2:
+    version "2.4.2"
+    resolved "https://registry.yarnpkg.com/chalk/-/chalk-2.4.2.tgz#cd42541677a54333cf541a49108c1432b44c9424"
+    integrity sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==
+    dependencies:
+      ansi-styles "^3.2.1"
+      escape-string-regexp "^1.0.5"
+      supports-color "^5.3.0"
+  
+  chalk@^3.0.0:
+    version "3.0.0"
+    resolved "https://registry.yarnpkg.com/chalk/-/chalk-3.0.0.tgz#3f73c2bf526591f574cc492c51e2456349f844e4"
+    integrity sha512-4D3B6Wf41KOYRFdszmDqMCGq5VV/uMAB273JILmO+3jAlh8X4qDtdtgCR3fxtbLEMzSx22QdhnDcJvu2u1fVwg==
+    dependencies:
+      ansi-styles "^4.1.0"
+      supports-color "^7.1.0"
+  
+  chalk@^4.1.0:
+    version "4.1.0"
+    resolved "https://registry.yarnpkg.com/chalk/-/chalk-4.1.0.tgz#4e14870a618d9e2edd97dd8345fd9d9dc315646a"
+    integrity sha512-qwx12AxXe2Q5xQ43Ac//I6v5aXTipYrSESdOgzrN+9XjgEpyjpKuvSGaN4qE93f7TQTlerQQ8S+EQ0EyDoVL1A==
+    dependencies:
+      ansi-styles "^4.1.0"
+      supports-color "^7.1.0"
+  
+  chardet@^0.7.0:
+    version "0.7.0"
+    resolved "https://registry.yarnpkg.com/chardet/-/chardet-0.7.0.tgz#90094849f0937f2eedc2425d0d28a9e5f0cbad9e"
+    integrity sha512-mT8iDcrh03qDGRRmoA2hmBJnxpllMR+0/0qlzjqZES6NdiWDcZkCNAk4rPFZ9Q85r27unkiNNg8ZOiwZXBHwcA==
+  
+  check-types@^8.0.3:
+    version "8.0.3"
+    resolved "https://registry.yarnpkg.com/check-types/-/check-types-8.0.3.tgz#3356cca19c889544f2d7a95ed49ce508a0ecf552"
+    integrity sha512-YpeKZngUmG65rLudJ4taU7VLkOCTMhNl/u4ctNC56LQS/zJTyNH0Lrtwm1tfTsbLlwvlfsA2d1c8vCf/Kh2KwQ==
+  
+  chokidar@^2.1.8:
+    version "2.1.8"
+    resolved "https://registry.yarnpkg.com/chokidar/-/chokidar-2.1.8.tgz#804b3a7b6a99358c3c5c61e71d8728f041cff917"
+    integrity sha512-ZmZUazfOzf0Nve7duiCKD23PFSCs4JPoYyccjUFF3aQkQadqBhfzhjkwBH2mNOG9cTBwhamM37EIsIkZw3nRgg==
+    dependencies:
+      anymatch "^2.0.0"
+      async-each "^1.0.1"
+      braces "^2.3.2"
+      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"
+    optionalDependencies:
+      fsevents "^1.2.7"
+  
+  chokidar@^3.4.1:
+    version "3.4.2"
+    resolved "https://registry.yarnpkg.com/chokidar/-/chokidar-3.4.2.tgz#38dc8e658dec3809741eb3ef7bb0a47fe424232d"
+    integrity sha512-IZHaDeBeI+sZJRX7lGcXsdzgvZqKv6sECqsbErJA4mHWfpRrD8B97kSFN4cQz6nGBGiuFia1MKR4d6c1o8Cv7A==
+    dependencies:
+      anymatch "~3.1.1"
+      braces "~3.0.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"
+    optionalDependencies:
+      fsevents "~2.1.2"
+  
+  chownr@^1.1.1, chownr@^1.1.2:
+    version "1.1.4"
+    resolved "https://registry.yarnpkg.com/chownr/-/chownr-1.1.4.tgz#6fc9d7b42d32a583596337666e7d08084da2cc6b"
+    integrity sha512-jJ0bqzaylmJtVnNgzTeSOs8DPavpbYgEr/b0YL8/2GO3xJEhInFmhKMUnEJQjZumK7KXGFhUy89PrsJWlakBVg==
+  
+  chrome-trace-event@^1.0.2:
+    version "1.0.2"
+    resolved "https://registry.yarnpkg.com/chrome-trace-event/-/chrome-trace-event-1.0.2.tgz#234090ee97c7d4ad1a2c4beae27505deffc608a4"
+    integrity sha512-9e/zx1jw7B4CO+c/RXoCsfg/x1AfUBioy4owYH0bJprEYAx5hRFLRhWBqHAG57D0ZM4H7vxbP7bPe0VwhQRYDQ==
+    dependencies:
+      tslib "^1.9.0"
+  
+  ci-info@^1.5.0:
+    version "1.6.0"
+    resolved "https://registry.yarnpkg.com/ci-info/-/ci-info-1.6.0.tgz#2ca20dbb9ceb32d4524a683303313f0304b1e497"
+    integrity sha512-vsGdkwSCDpWmP80ncATX7iea5DWQemg1UgCW5J8tqjU3lYw4FBYuj89J0CTVomA7BEfvSZd84GmHko+MxFQU2A==
+  
+  cipher-base@^1.0.0, cipher-base@^1.0.1, cipher-base@^1.0.3:
+    version "1.0.4"
+    resolved "https://registry.yarnpkg.com/cipher-base/-/cipher-base-1.0.4.tgz#8760e4ecc272f4c363532f926d874aae2c1397de"
+    integrity sha512-Kkht5ye6ZGmwv40uUDZztayT2ThLQGfnj/T71N/XzeZeo3nf8foyW7zGTsPYkEya3m5f3cAypH+qe7YOrM1U2Q==
+    dependencies:
+      inherits "^2.0.1"
+      safe-buffer "^5.0.1"
+  
+  class-utils@^0.3.5:
+    version "0.3.6"
+    resolved "https://registry.yarnpkg.com/class-utils/-/class-utils-0.3.6.tgz#f93369ae8b9a7ce02fd41faad0ca83033190c463"
+    integrity sha512-qOhPa/Fj7s6TY8H8esGu5QNpMMQxz79h+urzrNYN6mn+9BnxlDGf5QZ+XeCDsxSjPqsSR56XOZOJmpeurnLMeg==
+    dependencies:
+      arr-union "^3.1.0"
+      define-property "^0.2.5"
+      isobject "^3.0.0"
+      static-extend "^0.1.1"
+  
+  clean-css@4.2.x:
+    version "4.2.3"
+    resolved "https://registry.yarnpkg.com/clean-css/-/clean-css-4.2.3.tgz#507b5de7d97b48ee53d84adb0160ff6216380f78"
+    integrity sha512-VcMWDN54ZN/DS+g58HYL5/n4Zrqe8vHJpGA8KdgUXFU4fuP/aHNw8eld9SyEIyabIMJX/0RaY/fplOo5hYLSFA==
+    dependencies:
+      source-map "~0.6.0"
+  
+  clean-stack@^2.0.0:
+    version "2.2.0"
+    resolved "https://registry.yarnpkg.com/clean-stack/-/clean-stack-2.2.0.tgz#ee8472dbb129e727b31e8a10a427dee9dfe4008b"
+    integrity sha512-4diC9HaTE+KRAMWhDhrGOECgWZxoevMc5TlkObMqNSsVU62PYzXZ/SMTjzyGAFF1YusgxGcSWTEXBhp0CPwQ1A==
+  
+  cli-cursor@^2.1.0:
+    version "2.1.0"
+    resolved "https://registry.yarnpkg.com/cli-cursor/-/cli-cursor-2.1.0.tgz#b35dac376479facc3e94747d41d0d0f5238ffcb5"
+    integrity sha1-s12sN2R5+sw+lHR9QdDQ9SOP/LU=
+    dependencies:
+      restore-cursor "^2.0.0"
+  
+  cli-cursor@^3.1.0:
+    version "3.1.0"
+    resolved "https://registry.yarnpkg.com/cli-cursor/-/cli-cursor-3.1.0.tgz#264305a7ae490d1d03bf0c9ba7c925d1753af307"
+    integrity sha512-I/zHAwsKf9FqGoXM4WWRACob9+SNukZTd94DWF57E4toouRulbCxcUh6RKUEOQlYTHJnzkPMySvPNaaSLNfLZw==
+    dependencies:
+      restore-cursor "^3.1.0"
+  
+  cli-highlight@^2.1.4:
+    version "2.1.4"
+    resolved "https://registry.yarnpkg.com/cli-highlight/-/cli-highlight-2.1.4.tgz#098cb642cf17f42adc1c1145e07f960ec4d7522b"
+    integrity sha512-s7Zofobm20qriqDoU9sXptQx0t2R9PEgac92mENNm7xaEe1hn71IIMsXMK+6encA6WRCWWxIGQbipr3q998tlQ==
+    dependencies:
+      chalk "^3.0.0"
+      highlight.js "^9.6.0"
+      mz "^2.4.0"
+      parse5 "^5.1.1"
+      parse5-htmlparser2-tree-adapter "^5.1.1"
+      yargs "^15.0.0"
+  
+  cli-spinners@^2.0.0:
+    version "2.4.0"
+    resolved "https://registry.yarnpkg.com/cli-spinners/-/cli-spinners-2.4.0.tgz#c6256db216b878cfba4720e719cec7cf72685d7f"
+    integrity sha512-sJAofoarcm76ZGpuooaO0eDy8saEy+YoZBLjC4h8srt4jeBnkYeOgqxgsJQTpyt2LjI5PTfLJHSL+41Yu4fEJA==
+  
+  cli-width@^3.0.0:
+    version "3.0.0"
+    resolved "https://registry.yarnpkg.com/cli-width/-/cli-width-3.0.0.tgz#a2f48437a2caa9a22436e794bf071ec9e61cedf6"
+    integrity sha512-FxqpkPPwu1HjuN93Omfm4h8uIanXofW0RxVEW3k5RKx+mJJYSthzNhp32Kzxxy3YAEZ/Dc/EWN1vZRY0+kOhbw==
+  
+  clipboardy@^2.3.0:
+    version "2.3.0"
+    resolved "https://registry.yarnpkg.com/clipboardy/-/clipboardy-2.3.0.tgz#3c2903650c68e46a91b388985bc2774287dba290"
+    integrity sha512-mKhiIL2DrQIsuXMgBgnfEHOZOryC7kY7YO//TN6c63wlEm3NG5tz+YgY5rVi29KCmq/QQjKYvM7a19+MDOTHOQ==
+    dependencies:
+      arch "^2.1.1"
+      execa "^1.0.0"
+      is-wsl "^2.1.1"
+  
+  cliui@^5.0.0:
+    version "5.0.0"
+    resolved "https://registry.yarnpkg.com/cliui/-/cliui-5.0.0.tgz#deefcfdb2e800784aa34f46fa08e06851c7bbbc5"
+    integrity sha512-PYeGSEmmHM6zvoef2w8TPzlrnNpXIjTipYK780YswmIP9vjxmd6Y2a3CB2Ks6/AU8NHjZugXvo8w3oWM2qnwXA==
+    dependencies:
+      string-width "^3.1.0"
+      strip-ansi "^5.2.0"
+      wrap-ansi "^5.1.0"
+  
+  cliui@^6.0.0:
+    version "6.0.0"
+    resolved "https://registry.yarnpkg.com/cliui/-/cliui-6.0.0.tgz#511d702c0c4e41ca156d7d0e96021f23e13225b1"
+    integrity sha512-t6wbgtoCXvAzst7QgXxJYqPt0usEfbgQdftEPbLL/cvv6HPE5VgvqCuAIDR0NgU52ds6rFwqrgakNLrHEjCbrQ==
+    dependencies:
+      string-width "^4.2.0"
+      strip-ansi "^6.0.0"
+      wrap-ansi "^6.2.0"
+  
+  clone@^1.0.2:
+    version "1.0.4"
+    resolved "https://registry.yarnpkg.com/clone/-/clone-1.0.4.tgz#da309cc263df15994c688ca902179ca3c7cd7c7e"
+    integrity sha1-2jCcwmPfFZlMaIypAheco8fNfH4=
+  
+  coa@^2.0.2:
+    version "2.0.2"
+    resolved "https://registry.yarnpkg.com/coa/-/coa-2.0.2.tgz#43f6c21151b4ef2bf57187db0d73de229e3e7ec3"
+    integrity sha512-q5/jG+YQnSy4nRTV4F7lPepBJZ8qBNJJDBuJdoejDyLXgmL7IEo+Le2JDZudFTFt7mrCqIRaSjws4ygRCTCAXA==
+    dependencies:
+      "@types/q" "^1.5.1"
+      chalk "^2.4.1"
+      q "^1.1.2"
+  
+  collection-visit@^1.0.0:
+    version "1.0.0"
+    resolved "https://registry.yarnpkg.com/collection-visit/-/collection-visit-1.0.0.tgz#4bc0373c164bc3291b4d368c829cf1a80a59dca0"
+    integrity sha1-S8A3PBZLwykbTTaMgpzxqApZ3KA=
+    dependencies:
+      map-visit "^1.0.0"
+      object-visit "^1.0.0"
+  
+  color-convert@^1.9.0, color-convert@^1.9.1:
+    version "1.9.3"
+    resolved "https://registry.yarnpkg.com/color-convert/-/color-convert-1.9.3.tgz#bb71850690e1f136567de629d2d5471deda4c1e8"
+    integrity sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==
+    dependencies:
+      color-name "1.1.3"
+  
+  color-convert@^2.0.1:
+    version "2.0.1"
+    resolved "https://registry.yarnpkg.com/color-convert/-/color-convert-2.0.1.tgz#72d3a68d598c9bdb3af2ad1e84f21d896abd4de3"
+    integrity sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==
+    dependencies:
+      color-name "~1.1.4"
+  
+  color-name@1.1.3:
+    version "1.1.3"
+    resolved "https://registry.yarnpkg.com/color-name/-/color-name-1.1.3.tgz#a7d0558bd89c42f795dd42328f740831ca53bc25"
+    integrity sha1-p9BVi9icQveV3UIyj3QIMcpTvCU=
+  
+  color-name@^1.0.0, color-name@~1.1.4:
+    version "1.1.4"
+    resolved "https://registry.yarnpkg.com/color-name/-/color-name-1.1.4.tgz#c2a09a87acbde69543de6f63fa3995c826c536a2"
+    integrity sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==
+  
+  color-string@^1.5.2:
+    version "1.5.3"
+    resolved "https://registry.yarnpkg.com/color-string/-/color-string-1.5.3.tgz#c9bbc5f01b58b5492f3d6857459cb6590ce204cc"
+    integrity sha512-dC2C5qeWoYkxki5UAXapdjqO672AM4vZuPGRQfO8b5HKuKGBbKWpITyDYN7TOFKvRW7kOgAn3746clDBMDJyQw==
+    dependencies:
+      color-name "^1.0.0"
+      simple-swizzle "^0.2.2"
+  
+  color@^3.0.0:
+    version "3.1.2"
+    resolved "https://registry.yarnpkg.com/color/-/color-3.1.2.tgz#68148e7f85d41ad7649c5fa8c8106f098d229e10"
+    integrity sha512-vXTJhHebByxZn3lDvDJYw4lR5+uB3vuoHsuYA5AKuxRVn5wzzIfQKGLBmgdVRHKTJYeK5rvJcHnrd0Li49CFpg==
+    dependencies:
+      color-convert "^1.9.1"
+      color-string "^1.5.2"
+  
+  colorette@^1.2.1:
+    version "1.2.1"
+    resolved "https://registry.yarnpkg.com/colorette/-/colorette-1.2.1.tgz#4d0b921325c14faf92633086a536db6e89564b1b"
+    integrity sha512-puCDz0CzydiSYOrnXpz/PKd69zRrribezjtE9yd4zvytoRc8+RY/KJPvtPFKZS3E3wP6neGyMe0vOTlHO5L3Pw==
+  
+  combined-stream@^1.0.6, combined-stream@~1.0.6:
+    version "1.0.8"
+    resolved "https://registry.yarnpkg.com/combined-stream/-/combined-stream-1.0.8.tgz#c3d45a8b34fd730631a110a8a2520682b31d5a7f"
+    integrity sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==
+    dependencies:
+      delayed-stream "~1.0.0"
+  
+  commander@2.17.x:
+    version "2.17.1"
+    resolved "https://registry.yarnpkg.com/commander/-/commander-2.17.1.tgz#bd77ab7de6de94205ceacc72f1716d29f20a77bf"
+    integrity sha512-wPMUt6FnH2yzG95SA6mzjQOEKUU3aLaDEmzs1ti+1E9h+CsrZghRlqEM/EJ4KscsQVG8uNN4uVreUeT8+drlgg==
+  
+  commander@^2.18.0, commander@^2.20.0:
+    version "2.20.3"
+    resolved "https://registry.yarnpkg.com/commander/-/commander-2.20.3.tgz#fd485e84c03eb4881c20722ba48035e8531aeb33"
+    integrity sha512-GpVkmM8vF2vQUkj2LvZmD35JxeJOLCwJ9cUkugyk2nuhbv3+mJvpLYYt+0+USMxE+oj+ey/lJEnhZw75x/OMcQ==
+  
+  commander@~2.19.0:
+    version "2.19.0"
+    resolved "https://registry.yarnpkg.com/commander/-/commander-2.19.0.tgz#f6198aa84e5b83c46054b94ddedbfed5ee9ff12a"
+    integrity sha512-6tvAOO+D6OENvRAh524Dh9jcfKTYDQAqvqezbCW82xj5X0pSrcpxtvRKHLG0yBY6SD7PSDrJaj+0AiOcKVd1Xg==
+  
+  commondir@^1.0.1:
+    version "1.0.1"
+    resolved "https://registry.yarnpkg.com/commondir/-/commondir-1.0.1.tgz#ddd800da0c66127393cca5950ea968a3aaf1253b"
+    integrity sha1-3dgA2gxmEnOTzKWVDqloo6rxJTs=
+  
+  component-emitter@^1.2.1:
+    version "1.3.0"
+    resolved "https://registry.yarnpkg.com/component-emitter/-/component-emitter-1.3.0.tgz#16e4070fba8ae29b679f2215853ee181ab2eabc0"
+    integrity sha512-Rd3se6QB+sO1TwqZjscQrurpEPIfO0/yYnSin6Q/rD3mOutHvUrCAhJub3r90uNb+SESBuE0QYoB90YdfatsRg==
+  
+  compressible@~2.0.16:
+    version "2.0.18"
+    resolved "https://registry.yarnpkg.com/compressible/-/compressible-2.0.18.tgz#af53cca6b070d4c3c0750fbd77286a6d7cc46fba"
+    integrity sha512-AF3r7P5dWxL8MxyITRMlORQNaOA2IkAFaTr4k7BUumjPtRpGDTZpl0Pb1XCO6JeDCBdp126Cgs9sMxqSjgYyRg==
+    dependencies:
+      mime-db ">= 1.43.0 < 2"
+  
+  compression@^1.7.4:
+    version "1.7.4"
+    resolved "https://registry.yarnpkg.com/compression/-/compression-1.7.4.tgz#95523eff170ca57c29a0ca41e6fe131f41e5bb8f"
+    integrity sha512-jaSIDzP9pZVS4ZfQ+TzvtiWhdpFhE2RDHz8QJkpX9SIpLq88VueF5jJw6t+6CUQcAoA6t+x89MLrWAqpfDE8iQ==
+    dependencies:
+      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"
+  
+  concat-map@0.0.1:
+    version "0.0.1"
+    resolved "https://registry.yarnpkg.com/concat-map/-/concat-map-0.0.1.tgz#d8a96bd77fd68df7793a73036a3ba0d5405d477b"
+    integrity sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=
+  
+  concat-stream@^1.5.0:
+    version "1.6.2"
+    resolved "https://registry.yarnpkg.com/concat-stream/-/concat-stream-1.6.2.tgz#904bdf194cd3122fc675c77fc4ac3d4ff0fd1a34"
+    integrity sha512-27HBghJxjiZtIk3Ycvn/4kbJk/1uZuJFfuPEns6LaEvpvG1f0hTea8lilrouyo9mVc2GWdcEZ8OLoGmSADlrCw==
+    dependencies:
+      buffer-from "^1.0.0"
+      inherits "^2.0.3"
+      readable-stream "^2.2.2"
+      typedarray "^0.0.6"
+  
+  connect-history-api-fallback@^1.6.0:
+    version "1.6.0"
+    resolved "https://registry.yarnpkg.com/connect-history-api-fallback/-/connect-history-api-fallback-1.6.0.tgz#8b32089359308d111115d81cad3fceab888f97bc"
+    integrity sha512-e54B99q/OUoH64zYYRf3HBP5z24G38h5D3qXu23JGRoigpX5Ss4r9ZnDk3g0Z8uQC2x2lPaJ+UlWBc1ZWBWdLg==
+  
+  consola@^2.10.1:
+    version "2.15.0"
+    resolved "https://registry.yarnpkg.com/consola/-/consola-2.15.0.tgz#40fc4eefa4d2f8ef2e2806147f056ea207fcc0e9"
+    integrity sha512-vlcSGgdYS26mPf7qNi+dCisbhiyDnrN1zaRbw3CSuc2wGOMEGGPsp46PdRG5gqXwgtJfjxDkxRNAgRPr1B77vQ==
+  
+  console-browserify@^1.1.0:
+    version "1.2.0"
+    resolved "https://registry.yarnpkg.com/console-browserify/-/console-browserify-1.2.0.tgz#67063cef57ceb6cf4993a2ab3a55840ae8c49336"
+    integrity sha512-ZMkYO/LkF17QvCPqM0gxw8yUzigAOZOSWSHg91FH6orS7vcEj5dVZTidN2fQ14yBSdg97RqhSNwLUXInd52OTA==
+  
+  consolidate@^0.15.1:
+    version "0.15.1"
+    resolved "https://registry.yarnpkg.com/consolidate/-/consolidate-0.15.1.tgz#21ab043235c71a07d45d9aad98593b0dba56bab7"
+    integrity sha512-DW46nrsMJgy9kqAbPt5rKaCr7uFtpo4mSUvLHIUbJEjm0vo+aY5QLwBUq3FK4tRnJr/X0Psc0C4jf/h+HtXSMw==
+    dependencies:
+      bluebird "^3.1.1"
+  
+  constants-browserify@^1.0.0:
+    version "1.0.0"
+    resolved "https://registry.yarnpkg.com/constants-browserify/-/constants-browserify-1.0.0.tgz#c20b96d8c617748aaf1c16021760cd27fcb8cb75"
+    integrity sha1-wguW2MYXdIqvHBYCF2DNJ/y4y3U=
+  
+  content-disposition@0.5.3:
+    version "0.5.3"
+    resolved "https://registry.yarnpkg.com/content-disposition/-/content-disposition-0.5.3.tgz#e130caf7e7279087c5616c2007d0485698984fbd"
+    integrity sha512-ExO0774ikEObIAEV9kDo50o+79VCUdEB6n6lzKgGwupcVeRlhrj3qGAfwq8G6uBJjkqLrhT0qEYFcWng8z1z0g==
+    dependencies:
+      safe-buffer "5.1.2"
+  
+  content-type@~1.0.4:
+    version "1.0.4"
+    resolved "https://registry.yarnpkg.com/content-type/-/content-type-1.0.4.tgz#e138cc75e040c727b1966fe5e5f8c9aee256fe3b"
+    integrity sha512-hIP3EEPs8tB9AT1L+NUqtwOAps4mk2Zob89MWXMHjHWg9milF/j4osnnQLXBCBFBk/tvIG/tUc9mOUJiPBhPXA==
+  
+  convert-source-map@^1.7.0:
+    version "1.7.0"
+    resolved "https://registry.yarnpkg.com/convert-source-map/-/convert-source-map-1.7.0.tgz#17a2cb882d7f77d3490585e2ce6c524424a3a442"
+    integrity sha512-4FJkXzKXEDB1snCFZlLP4gpC3JILicCpGbzG9f9G7tGqGCzETQ2hWPrcinA9oU4wtf2biUaEH5065UnMeR33oA==
+    dependencies:
+      safe-buffer "~5.1.1"
+  
+  cookie-signature@1.0.6:
+    version "1.0.6"
+    resolved "https://registry.yarnpkg.com/cookie-signature/-/cookie-signature-1.0.6.tgz#e303a882b342cc3ee8ca513a79999734dab3ae2c"
+    integrity sha1-4wOogrNCzD7oylE6eZmXNNqzriw=
+  
+  cookie@0.4.0:
+    version "0.4.0"
+    resolved "https://registry.yarnpkg.com/cookie/-/cookie-0.4.0.tgz#beb437e7022b3b6d49019d088665303ebe9c14ba"
+    integrity sha512-+Hp8fLp57wnUSt0tY0tHEXh4voZRDnoIrZPqlo3DPiI4y9lwg/jqx+1Om94/W6ZaPDOUbnjOt/99w66zk+l1Xg==
+  
+  copy-concurrently@^1.0.0:
+    version "1.0.5"
+    resolved "https://registry.yarnpkg.com/copy-concurrently/-/copy-concurrently-1.0.5.tgz#92297398cae34937fcafd6ec8139c18051f0b5e0"
+    integrity sha512-f2domd9fsVDFtaFcbaRZuYXwtdmnzqbADSwhSWYxYB/Q8zsdUUFMXVRwXGDMWmbEzAn1kdRrtI1T/KTFOL4X2A==
+    dependencies:
+      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"
+  
+  copy-descriptor@^0.1.0:
+    version "0.1.1"
+    resolved "https://registry.yarnpkg.com/copy-descriptor/-/copy-descriptor-0.1.1.tgz#676f6eb3c39997c2ee1ac3a924fd6124748f578d"
+    integrity sha1-Z29us8OZl8LuGsOpJP1hJHSPV40=
+  
+  copy-webpack-plugin@^5.1.1:
+    version "5.1.2"
+    resolved "https://registry.yarnpkg.com/copy-webpack-plugin/-/copy-webpack-plugin-5.1.2.tgz#8a889e1dcafa6c91c6cd4be1ad158f1d3823bae2"
+    integrity sha512-Uh7crJAco3AjBvgAy9Z75CjK8IG+gxaErro71THQ+vv/bl4HaQcpkexAY8KVW/T6D2W2IRr+couF/knIRkZMIQ==
+    dependencies:
+      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 "^4.0.0"
+      webpack-log "^2.0.0"
+  
+  core-js-compat@^3.6.2, core-js-compat@^3.6.5:
+    version "3.6.5"
+    resolved "https://registry.yarnpkg.com/core-js-compat/-/core-js-compat-3.6.5.tgz#2a51d9a4e25dfd6e690251aa81f99e3c05481f1c"
+    integrity sha512-7ItTKOhOZbznhXAQ2g/slGg1PJV5zDO/WdkTwi7UEOJmkvsE32PWvx6mKtDjiMpjnR2CNf6BAD6sSxIlv7ptng==
+    dependencies:
+      browserslist "^4.8.5"
+      semver "7.0.0"
+  
+  core-js@^3.6.5:
+    version "3.6.5"
+    resolved "https://registry.yarnpkg.com/core-js/-/core-js-3.6.5.tgz#7395dc273af37fb2e50e9bd3d9fe841285231d1a"
+    integrity sha512-vZVEEwZoIsI+vPEuoF9Iqf5H7/M3eeQqWlQnYa8FSKKePuYTf5MWnxb5SDAzCa60b3JBRS5g9b+Dq7b1y/RCrA==
+  
+  core-util-is@1.0.2, core-util-is@~1.0.0:
+    version "1.0.2"
+    resolved "https://registry.yarnpkg.com/core-util-is/-/core-util-is-1.0.2.tgz#b5fd54220aa2bc5ab57aab7140c940754503c1a7"
+    integrity sha1-tf1UIgqivFq1eqtxQMlAdUUDwac=
+  
+  cosmiconfig@^5.0.0:
+    version "5.2.1"
+    resolved "https://registry.yarnpkg.com/cosmiconfig/-/cosmiconfig-5.2.1.tgz#040f726809c591e77a17c0a3626ca45b4f168b1a"
+    integrity sha512-H65gsXo1SKjf8zmrJ67eJk8aIRKV5ff2D4uKZIBZShbhGSpEmsQOPW/SKMKYhSTrqR7ufy6RP69rPogdaPh/kA==
+    dependencies:
+      import-fresh "^2.0.0"
+      is-directory "^0.3.1"
+      js-yaml "^3.13.1"
+      parse-json "^4.0.0"
+  
+  create-ecdh@^4.0.0:
+    version "4.0.4"
+    resolved "https://registry.yarnpkg.com/create-ecdh/-/create-ecdh-4.0.4.tgz#d6e7f4bffa66736085a0762fd3a632684dabcc4e"
+    integrity sha512-mf+TCx8wWc9VpuxfP2ht0iSISLZnt0JgWlrOKZiNqyUZWnjIaCIVNQArMHnCZKfEYRg6IM7A+NeJoN8gf/Ws0A==
+    dependencies:
+      bn.js "^4.1.0"
+      elliptic "^6.5.3"
+  
+  create-hash@^1.1.0, create-hash@^1.1.2, create-hash@^1.2.0:
+    version "1.2.0"
+    resolved "https://registry.yarnpkg.com/create-hash/-/create-hash-1.2.0.tgz#889078af11a63756bcfb59bd221996be3a9ef196"
+    integrity sha512-z00bCGNHDG8mHAkP7CtT1qVu+bFQUPjYq/4Iv3C3kWjTFV10zIjfSoeqXo9Asws8gwSHDGj/hl2u4OGIjapeCg==
+    dependencies:
+      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@^1.1.0, create-hmac@^1.1.4, create-hmac@^1.1.7:
+    version "1.1.7"
+    resolved "https://registry.yarnpkg.com/create-hmac/-/create-hmac-1.1.7.tgz#69170c78b3ab957147b2b8b04572e47ead2243ff"
+    integrity sha512-MJG9liiZ+ogc4TzUwuvbER1JRdgvUFSB5+VR/g5h82fGaIRWMWddtKBHi7/sVhfjQZ6SehlyhvQYrcYkaUIpLg==
+    dependencies:
+      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@^5.0.1:
+    version "5.1.0"
+    resolved "https://registry.yarnpkg.com/cross-spawn/-/cross-spawn-5.1.0.tgz#e8bd0efee58fcff6f8f94510a0a554bbfa235449"
+    integrity sha1-6L0O/uWPz/b4+UUQoKVUu/ojVEk=
+    dependencies:
+      lru-cache "^4.0.1"
+      shebang-command "^1.2.0"
+      which "^1.2.9"
+  
+  cross-spawn@^6.0.0, cross-spawn@^6.0.5:
+    version "6.0.5"
+    resolved "https://registry.yarnpkg.com/cross-spawn/-/cross-spawn-6.0.5.tgz#4a5ec7c64dfae22c3a14124dbacdee846d80cbc4"
+    integrity sha512-eTVLrBSt7fjbDygz805pMnstIs2VTBNkRm0qxZd+M7A5XDdxVRWO5MxGBXZhjY4cqLYLdtrGqRf8mBPmzwSpWQ==
+    dependencies:
+      nice-try "^1.0.4"
+      path-key "^2.0.1"
+      semver "^5.5.0"
+      shebang-command "^1.2.0"
+      which "^1.2.9"
+  
+  cross-spawn@^7.0.0:
+    version "7.0.3"
+    resolved "https://registry.yarnpkg.com/cross-spawn/-/cross-spawn-7.0.3.tgz#f73a85b9d5d41d045551c177e2882d4ac85728a6"
+    integrity sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==
+    dependencies:
+      path-key "^3.1.0"
+      shebang-command "^2.0.0"
+      which "^2.0.1"
+  
+  crypto-browserify@^3.11.0:
+    version "3.12.0"
+    resolved "https://registry.yarnpkg.com/crypto-browserify/-/crypto-browserify-3.12.0.tgz#396cf9f3137f03e4b8e532c58f698254e00f80ec"
+    integrity sha512-fz4spIh+znjO2VjL+IdhEpRJ3YN6sMzITSBijk6FK2UvTqruSQW+/cCZTSNsMiZNvUeq0CqurF+dAbyiGOY6Wg==
+    dependencies:
+      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-color-names@0.0.4, css-color-names@^0.0.4:
+    version "0.0.4"
+    resolved "https://registry.yarnpkg.com/css-color-names/-/css-color-names-0.0.4.tgz#808adc2e79cf84738069b646cb20ec27beb629e0"
+    integrity sha1-gIrcLnnPhHOAabZGyyDsJ762KeA=
+  
+  css-declaration-sorter@^4.0.1:
+    version "4.0.1"
+    resolved "https://registry.yarnpkg.com/css-declaration-sorter/-/css-declaration-sorter-4.0.1.tgz#c198940f63a76d7e36c1e71018b001721054cb22"
+    integrity sha512-BcxQSKTSEEQUftYpBVnsH4SF05NTuBokb19/sBt6asXGKZ/6VP7PLG1CBCkFDYOnhXhPh0jMhO6xZ71oYHXHBA==
+    dependencies:
+      postcss "^7.0.1"
+      timsort "^0.3.0"
+  
+  css-loader@^3.5.3:
+    version "3.6.0"
+    resolved "https://registry.yarnpkg.com/css-loader/-/css-loader-3.6.0.tgz#2e4b2c7e6e2d27f8c8f28f61bffcd2e6c91ef645"
+    integrity sha512-M5lSukoWi1If8dhQAUCvj4H8vUt3vOnwbQBH9DdTm/s4Ym2B/3dPMtYZeJmq7Q3S3Pa+I94DcZ7pc9bP14cWIQ==
+    dependencies:
+      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.32"
+      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.1.0"
+      schema-utils "^2.7.0"
+      semver "^6.3.0"
+  
+  css-select-base-adapter@^0.1.1:
+    version "0.1.1"
+    resolved "https://registry.yarnpkg.com/css-select-base-adapter/-/css-select-base-adapter-0.1.1.tgz#3b2ff4972cc362ab88561507a95408a1432135d7"
+    integrity sha512-jQVeeRG70QI08vSTwf1jHxp74JoZsr2XSgETae8/xC8ovSnL2WF87GTLO86Sbwdt2lK4Umg4HnnwMO4YF3Ce7w==
+  
+  css-select@^1.1.0:
+    version "1.2.0"
+    resolved "https://registry.yarnpkg.com/css-select/-/css-select-1.2.0.tgz#2b3a110539c5355f1cd8d314623e870b121ec858"
+    integrity sha1-KzoRBTnFNV8c2NMUYj6HCxIeyFg=
+    dependencies:
+      boolbase "~1.0.0"
+      css-what "2.1"
+      domutils "1.5.1"
+      nth-check "~1.0.1"
+  
+  css-select@^2.0.0:
+    version "2.1.0"
+    resolved "https://registry.yarnpkg.com/css-select/-/css-select-2.1.0.tgz#6a34653356635934a81baca68d0255432105dbef"
+    integrity sha512-Dqk7LQKpwLoH3VovzZnkzegqNSuAziQyNZUcrdDM401iY+R5NkGBXGmtO05/yaXQziALuPogeG0b7UAgjnTJTQ==
+    dependencies:
+      boolbase "^1.0.0"
+      css-what "^3.2.1"
+      domutils "^1.7.0"
+      nth-check "^1.0.2"
+  
+  css-tree@1.0.0-alpha.37:
+    version "1.0.0-alpha.37"
+    resolved "https://registry.yarnpkg.com/css-tree/-/css-tree-1.0.0-alpha.37.tgz#98bebd62c4c1d9f960ec340cf9f7522e30709a22"
+    integrity sha512-DMxWJg0rnz7UgxKT0Q1HU/L9BeJI0M6ksor0OgqOnF+aRCDWg/N2641HmVyU9KVIu0OVVWOb2IpC9A+BJRnejg==
+    dependencies:
+      mdn-data "2.0.4"
+      source-map "^0.6.1"
+  
+  css-tree@1.0.0-alpha.39:
+    version "1.0.0-alpha.39"
+    resolved "https://registry.yarnpkg.com/css-tree/-/css-tree-1.0.0-alpha.39.tgz#2bff3ffe1bb3f776cf7eefd91ee5cba77a149eeb"
+    integrity sha512-7UvkEYgBAHRG9Nt980lYxjsTrCyHFN53ky3wVsDkiMdVqylqRt+Zc+jm5qw7/qyOvN2dHSYtX0e4MbCCExSvnA==
+    dependencies:
+      mdn-data "2.0.6"
+      source-map "^0.6.1"
+  
+  css-what@2.1:
+    version "2.1.3"
+    resolved "https://registry.yarnpkg.com/css-what/-/css-what-2.1.3.tgz#a6d7604573365fe74686c3f311c56513d88285f2"
+    integrity sha512-a+EPoD+uZiNfh+5fxw2nO9QwFa6nJe2Or35fGY6Ipw1R3R4AGz1d1TEZrCegvw2YTmZ0jXirGYlzxxpYSHwpEg==
+  
+  css-what@^3.2.1:
+    version "3.3.0"
+    resolved "https://registry.yarnpkg.com/css-what/-/css-what-3.3.0.tgz#10fec696a9ece2e591ac772d759aacabac38cd39"
+    integrity sha512-pv9JPyatiPaQ6pf4OvD/dbfm0o5LviWmwxNWzblYf/1u9QZd0ihV+PMwy5jdQWQ3349kZmKEx9WXuSka2dM4cg==
+  
+  cssesc@^3.0.0:
+    version "3.0.0"
+    resolved "https://registry.yarnpkg.com/cssesc/-/cssesc-3.0.0.tgz#37741919903b868565e1c09ea747445cd18983ee"
+    integrity sha512-/Tb/JcjK111nNScGob5MNtsntNM1aCNUDipB/TkwZFhyDrrE47SOx/18wF2bbjgc3ZzCSKW1T5nt5EbFoAz/Vg==
+  
+  cssnano-preset-default@^4.0.0, cssnano-preset-default@^4.0.7:
+    version "4.0.7"
+    resolved "https://registry.yarnpkg.com/cssnano-preset-default/-/cssnano-preset-default-4.0.7.tgz#51ec662ccfca0f88b396dcd9679cdb931be17f76"
+    integrity sha512-x0YHHx2h6p0fCl1zY9L9roD7rnlltugGu7zXSKQx6k2rYw0Hi3IqxcoAGF7u9Q5w1nt7vK0ulxV8Lo+EvllGsA==
+    dependencies:
+      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@^4.0.0:
+    version "4.0.0"
+    resolved "https://registry.yarnpkg.com/cssnano-util-get-arguments/-/cssnano-util-get-arguments-4.0.0.tgz#ed3a08299f21d75741b20f3b81f194ed49cc150f"
+    integrity sha1-7ToIKZ8h11dBsg87gfGU7UnMFQ8=
+  
+  cssnano-util-get-match@^4.0.0:
+    version "4.0.0"
+    resolved "https://registry.yarnpkg.com/cssnano-util-get-match/-/cssnano-util-get-match-4.0.0.tgz#c0e4ca07f5386bb17ec5e52250b4f5961365156d"
+    integrity sha1-wOTKB/U4a7F+xeUiULT1lhNlFW0=
+  
+  cssnano-util-raw-cache@^4.0.1:
+    version "4.0.1"
+    resolved "https://registry.yarnpkg.com/cssnano-util-raw-cache/-/cssnano-util-raw-cache-4.0.1.tgz#b26d5fd5f72a11dfe7a7846fb4c67260f96bf282"
+    integrity sha512-qLuYtWK2b2Dy55I8ZX3ky1Z16WYsx544Q0UWViebptpwn/xDBmog2TLg4f+DBMg1rJ6JDWtn96WHbOKDWt1WQA==
+    dependencies:
+      postcss "^7.0.0"
+  
+  cssnano-util-same-parent@^4.0.0:
+    version "4.0.1"
+    resolved "https://registry.yarnpkg.com/cssnano-util-same-parent/-/cssnano-util-same-parent-4.0.1.tgz#574082fb2859d2db433855835d9a8456ea18bbf3"
+    integrity sha512-WcKx5OY+KoSIAxBW6UBBRay1U6vkYheCdjyVNDm85zt5K9mHoGOfsOsqIszfAqrQQFIIKgjh2+FDgIj/zsl21Q==
+  
+  cssnano@^4.0.0, cssnano@^4.1.10:
+    version "4.1.10"
+    resolved "https://registry.yarnpkg.com/cssnano/-/cssnano-4.1.10.tgz#0ac41f0b13d13d465487e111b778d42da631b8b2"
+    integrity sha512-5wny+F6H4/8RgNlaqab4ktc3e0/blKutmq8yNlBFXA//nSFFAqAngjNVRzUvCgYROULmZZUoosL/KSoZo5aUaQ==
+    dependencies:
+      cosmiconfig "^5.0.0"
+      cssnano-preset-default "^4.0.7"
+      is-resolvable "^1.0.0"
+      postcss "^7.0.0"
+  
+  csso@^4.0.2:
+    version "4.0.3"
+    resolved "https://registry.yarnpkg.com/csso/-/csso-4.0.3.tgz#0d9985dc852c7cc2b2cacfbbe1079014d1a8e903"
+    integrity sha512-NL3spysxUkcrOgnpsT4Xdl2aiEiBG6bXswAABQVHcMrfjjBisFOKwLDOmf4wf32aPdcJws1zds2B0Rg+jqMyHQ==
+    dependencies:
+      css-tree "1.0.0-alpha.39"
+  
+  cyclist@^1.0.1:
+    version "1.0.1"
+    resolved "https://registry.yarnpkg.com/cyclist/-/cyclist-1.0.1.tgz#596e9698fd0c80e12038c2b82d6eb1b35b6224d9"
+    integrity sha1-WW6WmP0MgOEgOMK4LW6xs1tiJNk=
+  
+  dashdash@^1.12.0:
+    version "1.14.1"
+    resolved "https://registry.yarnpkg.com/dashdash/-/dashdash-1.14.1.tgz#853cfa0f7cbe2fed5de20326b8dd581035f6e2f0"
+    integrity sha1-hTz6D3y+L+1d4gMmuN1YEDX24vA=
+    dependencies:
+      assert-plus "^1.0.0"
+  
+  de-indent@^1.0.2:
+    version "1.0.2"
+    resolved "https://registry.yarnpkg.com/de-indent/-/de-indent-1.0.2.tgz#b2038e846dc33baa5796128d0804b455b8c1e21d"
+    integrity sha1-sgOOhG3DO6pXlhKNCAS0VbjB4h0=
+  
+  debug@2.6.9, debug@^2.2.0, debug@^2.3.3:
+    version "2.6.9"
+    resolved "https://registry.yarnpkg.com/debug/-/debug-2.6.9.tgz#5d128515df134ff327e90a4c93f4e077a536341f"
+    integrity sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==
+    dependencies:
+      ms "2.0.0"
+  
+  debug@^3.1.1, debug@^3.2.5:
+    version "3.2.6"
+    resolved "https://registry.yarnpkg.com/debug/-/debug-3.2.6.tgz#e83d17de16d8a7efb7717edbe5fb10135eee629b"
+    integrity sha512-mel+jf7nrtEl5Pn1Qx46zARXKDpBbvzezse7p7LqINmdoIk8PYP5SySaxEmYv6TZ0JyEKA1hsCId6DIhgITtWQ==
+    dependencies:
+      ms "^2.1.1"
+  
+  debug@^4.0.1, debug@^4.1.0, debug@^4.1.1:
+    version "4.1.1"
+    resolved "https://registry.yarnpkg.com/debug/-/debug-4.1.1.tgz#3b72260255109c6b589cee050f1d516139664791"
+    integrity sha512-pYAIzeRo8J6KPEaJ0VWOh5Pzkbw/RetuzehGM7QRRX5he4fPHx2rdKMB256ehJCkX+XRQm16eZLqLNS8RSZXZw==
+    dependencies:
+      ms "^2.1.1"
+  
+  decamelize@^1.2.0:
+    version "1.2.0"
+    resolved "https://registry.yarnpkg.com/decamelize/-/decamelize-1.2.0.tgz#f6534d15148269b20352e7bee26f501f9a191290"
+    integrity sha1-9lNNFRSCabIDUue+4m9QH5oZEpA=
+  
+  decode-uri-component@^0.2.0:
+    version "0.2.0"
+    resolved "https://registry.yarnpkg.com/decode-uri-component/-/decode-uri-component-0.2.0.tgz#eb3913333458775cb84cd1a1fae062106bb87545"
+    integrity sha1-6zkTMzRYd1y4TNGh+uBiEGu4dUU=
+  
+  deep-equal@^1.0.1:
+    version "1.1.1"
+    resolved "https://registry.yarnpkg.com/deep-equal/-/deep-equal-1.1.1.tgz#b5c98c942ceffaf7cb051e24e1434a25a2e6076a"
+    integrity sha512-yd9c5AdiqVcR+JjcwUQb9DkhJc8ngNr0MahEBGvDiJw8puWab2yZlh+nkasOnZP+EGTAP6rRp2JzJhJZzvNF8g==
+    dependencies:
+      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"
+  
+  deep-is@~0.1.3:
+    version "0.1.3"
+    resolved "https://registry.yarnpkg.com/deep-is/-/deep-is-0.1.3.tgz#b369d6fb5dbc13eecf524f91b070feedc357cf34"
+    integrity sha1-s2nW+128E+7PUk+RsHD+7cNXzzQ=
+  
+  deepmerge@^1.5.2:
+    version "1.5.2"
+    resolved "https://registry.yarnpkg.com/deepmerge/-/deepmerge-1.5.2.tgz#10499d868844cdad4fee0842df8c7f6f0c95a753"
+    integrity sha512-95k0GDqvBjZavkuvzx/YqVLv/6YYa17fz6ILMSf7neqQITCPbnfEnQvEgMPNjH4kgobe7+WIL0yJEHku+H3qtQ==
+  
+  default-gateway@^4.2.0:
+    version "4.2.0"
+    resolved "https://registry.yarnpkg.com/default-gateway/-/default-gateway-4.2.0.tgz#167104c7500c2115f6dd69b0a536bb8ed720552b"
+    integrity sha512-h6sMrVB1VMWVrW13mSc6ia/DwYYw5MN6+exNu1OaJeFac5aSAvwM7lZ0NVfTABuSkQelr4h5oebg3KB1XPdjgA==
+    dependencies:
+      execa "^1.0.0"
+      ip-regex "^2.1.0"
+  
+  default-gateway@^5.0.5:
+    version "5.0.5"
+    resolved "https://registry.yarnpkg.com/default-gateway/-/default-gateway-5.0.5.tgz#4fd6bd5d2855d39b34cc5a59505486e9aafc9b10"
+    integrity sha512-z2RnruVmj8hVMmAnEJMTIJNijhKCDiGjbLP+BHJFOT7ld3Bo5qcIBpVYDniqhbMIIf+jZDlkP2MkPXiQy/DBLA==
+    dependencies:
+      execa "^3.3.0"
+  
+  defaults@^1.0.3:
+    version "1.0.3"
+    resolved "https://registry.yarnpkg.com/defaults/-/defaults-1.0.3.tgz#c656051e9817d9ff08ed881477f3fe4019f3ef7d"
+    integrity sha1-xlYFHpgX2f8I7YgUd/P+QBnz730=
+    dependencies:
+      clone "^1.0.2"
+  
+  define-properties@^1.1.2, define-properties@^1.1.3:
+    version "1.1.3"
+    resolved "https://registry.yarnpkg.com/define-properties/-/define-properties-1.1.3.tgz#cf88da6cbee26fe6db7094f61d870cbd84cee9f1"
+    integrity sha512-3MqfYKj2lLzdMSf8ZIZE/V+Zuy+BgD6f164e8K2w7dgnpKArBDerGYpM46IYYcjnkdPNMjPk9A6VFB8+3SKlXQ==
+    dependencies:
+      object-keys "^1.0.12"
+  
+  define-property@^0.2.5:
+    version "0.2.5"
+    resolved "https://registry.yarnpkg.com/define-property/-/define-property-0.2.5.tgz#c35b1ef918ec3c990f9a5bc57be04aacec5c8116"
+    integrity sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=
+    dependencies:
+      is-descriptor "^0.1.0"
+  
+  define-property@^1.0.0:
+    version "1.0.0"
+    resolved "https://registry.yarnpkg.com/define-property/-/define-property-1.0.0.tgz#769ebaaf3f4a63aad3af9e8d304c9bbe79bfb0e6"
+    integrity sha1-dp66rz9KY6rTr56NMEybvnm/sOY=
+    dependencies:
+      is-descriptor "^1.0.0"
+  
+  define-property@^2.0.2:
+    version "2.0.2"
+    resolved "https://registry.yarnpkg.com/define-property/-/define-property-2.0.2.tgz#d459689e8d654ba77e02a817f8710d702cb16e9d"
+    integrity sha512-jwK2UV4cnPpbcG7+VRARKTZPUWowwXA8bzH5NP6ud0oeAxyYPuGZUAC7hMugpCdz4BeSZl2Dl9k66CHJ/46ZYQ==
+    dependencies:
+      is-descriptor "^1.0.2"
+      isobject "^3.0.1"
+  
+  del@^4.1.1:
+    version "4.1.1"
+    resolved "https://registry.yarnpkg.com/del/-/del-4.1.1.tgz#9e8f117222ea44a31ff3a156c049b99052a9f0b4"
+    integrity sha512-QwGuEUouP2kVwQenAsOof5Fv8K9t3D8Ca8NxcXKrIpEHjTXK5J2nXLdP+ALI1cgv8wj7KuwBhTwBkOZSJKM5XQ==
+    dependencies:
+      "@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"
+  
+  delayed-stream@~1.0.0:
+    version "1.0.0"
+    resolved "https://registry.yarnpkg.com/delayed-stream/-/delayed-stream-1.0.0.tgz#df3ae199acadfb7d440aaae0b29e2272b24ec619"
+    integrity sha1-3zrhmayt+31ECqrgsp4icrJOxhk=
+  
+  depd@~1.1.2:
+    version "1.1.2"
+    resolved "https://registry.yarnpkg.com/depd/-/depd-1.1.2.tgz#9bcd52e14c097763e749b274c4346ed2e560b5a9"
+    integrity sha1-m81S4UwJd2PnSbJ0xDRu0uVgtak=
+  
+  des.js@^1.0.0:
+    version "1.0.1"
+    resolved "https://registry.yarnpkg.com/des.js/-/des.js-1.0.1.tgz#5382142e1bdc53f85d86d53e5f4aa7deb91e0843"
+    integrity sha512-Q0I4pfFrv2VPd34/vfLrFOoRmlYj3OV50i7fskps1jZWK1kApMWWT9G6RRUeYedLcBDIhnSDaUvJMb3AhUlaEA==
+    dependencies:
+      inherits "^2.0.1"
+      minimalistic-assert "^1.0.0"
+  
+  destroy@~1.0.4:
+    version "1.0.4"
+    resolved "https://registry.yarnpkg.com/destroy/-/destroy-1.0.4.tgz#978857442c44749e4206613e37946205826abd80"
+    integrity sha1-l4hXRCxEdJ5CBmE+N5RiBYJqvYA=
+  
+  detect-node@^2.0.4:
+    version "2.0.4"
+    resolved "https://registry.yarnpkg.com/detect-node/-/detect-node-2.0.4.tgz#014ee8f8f669c5c58023da64b8179c083a28c46c"
+    integrity sha512-ZIzRpLJrOj7jjP2miAtgqIfmzbxa4ZOr5jJc601zklsfEx9oTzmmj2nVpIPRpNlRTIh8lc1kyViIY7BWSGNmKw==
+  
+  diffie-hellman@^5.0.0:
+    version "5.0.3"
+    resolved "https://registry.yarnpkg.com/diffie-hellman/-/diffie-hellman-5.0.3.tgz#40e8ee98f55a2149607146921c63e1ae5f3d2875"
+    integrity sha512-kqag/Nl+f3GwyK25fhUMYj81BUOrZ9IuJsjIcDE5icNM9FJHAVm3VcUDxdLPoQtTuUylWm6ZIknYJwwaPxsUzg==
+    dependencies:
+      bn.js "^4.1.0"
+      miller-rabin "^4.0.0"
... 13912 lines suppressed ...