Merge branch 'admin-portal-docker' into docker
diff --git a/.asf.yaml b/.asf.yaml
new file mode 100644
index 0000000..ba27dc5
--- /dev/null
+++ b/.asf.yaml
@@ -0,0 +1,25 @@
+github:
+  description: "Portal Interface for Apache Airavata Custos Security"
+  homepage: https://airavata.apache.org/
+  labels:
+    - airavata
+    - apache
+    - security
+    - oauth2
+    - openidconnect
+    - authentication
+    - authorization
+
+  features:
+    wiki: true
+    issues: true
+    projects: true
+
+  notifications:
+    commits:              commits@airavata.apache.org
+    # Send all issue emails (new, closed, comments) to issues@
+    issues:               issues@airavata.apache.org
+    # Send new/closed PR notifications to dev@
+    pullrequests_status:  issues@airavata.apache.org
+    # Send individual PR comments/reviews to issues@
+    pullrequests_comment: issues@airavata.apache.org
\ No newline at end of file
diff --git a/.gitignore b/.gitignore
index d1e1951..0d83b46 100644
--- a/.gitignore
+++ b/.gitignore
@@ -1,5 +1,5 @@
-.idea
+.idea*
 venv
 db.sqlite3
 *pycache*/
-node_modules/
\ No newline at end of file
+node_modules/
diff --git a/README b/README
index 2f43c32..17f087a 100644
--- a/README
+++ b/README
@@ -1,3 +1,3 @@
 # Apache Airavata Custos Portal
 
-Portal interface implemented in Django Framework to use and administer Custos Services 
\ No newline at end of file
+A Reference Portal interface implemented in Vue.JS Framework to use and administer a sample Custos Tenant 
diff --git a/custos-demo-gateway/.dockerignore b/custos-demo-gateway/.dockerignore
new file mode 100644
index 0000000..c2950ee
--- /dev/null
+++ b/custos-demo-gateway/.dockerignore
@@ -0,0 +1,14 @@
+# 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
+.env
+
+# 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..1680335
--- /dev/null
+++ b/custos-demo-gateway/.env
@@ -0,0 +1,3 @@
+VUE_APP_CLIENT_ID="custos-id"
+VUE_APP_CLIENT_SEC="custos-sec"
+VUE_APP_REDIRECT_URI="http://localhost:8080/callback"
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..288b36b
--- /dev/null
+++ b/custos-demo-gateway/.idea/vcs.xml
@@ -0,0 +1,7 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<project version="4">
+  <component name="VcsDirectoryMappings">
+    <mapping directory="$PROJECT_DIR$/.." vcs="Git" />
+    <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..90a905e
--- /dev/null
+++ b/custos-demo-gateway/.idea/workspace.xml
@@ -0,0 +1,136 @@
+<?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="" />
+    <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="JavaScript File" />
+        <option value="Vue Single File Component" />
+      </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$" />
+    <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="MoveFile.RECENT_KEYS">
+      <recent name="$PROJECT_DIR$/src/store/modules" />
+      <recent name="$PROJECT_DIR$/src/components/landing" />
+      <recent name="$PROJECT_DIR$/src/components" />
+    </key>
+    <key name="CopyFile.RECENT_KEYS">
+      <recent name="$PROJECT_DIR$" />
+      <recent name="$PROJECT_DIR$/public" />
+      <recent name="$PROJECT_DIR$/src/assets" />
+    </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="67420000" />
+      <workItem from="1602211298031" duration="8699000" />
+      <workItem from="1602377149425" duration="3193000" />
+    </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>324</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..079efea
--- /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 8080 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..400ec05
--- /dev/null
+++ b/custos-demo-gateway/docker-compose.yml
@@ -0,0 +1,11 @@
+version: "3.3"
+services:
+  web:
+    image : "apachecustos/custos-demo-gateway-local:latest"
+    ports:
+      - "8080:8080"
+    environment:
+      VUE_APP_CLIENT_ID: 'tenant-id'
+      VUE_APP_CLIENT_SEC: 'tenant-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'
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..c329dbe
--- /dev/null
+++ b/custos-demo-gateway/nginx.conf
@@ -0,0 +1,57 @@
+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 8080;
+
+    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;
+      }
+}
+
+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..884343d
--- /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",
+    "vuelidate": "^0.7.5"
+  },
+  "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/custos-logo_custos-logomark-color.png b/custos-demo-gateway/public/custos-logo_custos-logomark-color.png
new file mode 100644
index 0000000..1871121
--- /dev/null
+++ b/custos-demo-gateway/public/custos-logo_custos-logomark-color.png
Binary files differ
diff --git a/custos-demo-gateway/public/favicon.ico b/custos-demo-gateway/public/favicon.ico
new file mode 100644
index 0000000..f433432
--- /dev/null
+++ b/custos-demo-gateway/public/favicon.ico
Binary files differ
diff --git a/custos-demo-gateway/public/index.html b/custos-demo-gateway/public/index.html
new file mode 100644
index 0000000..a13aad8
--- /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 %>custos-logo_custos-logomark-color.png">
+    <title>Custos</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..5157ea7
--- /dev/null
+++ b/custos-demo-gateway/src/App.vue
@@ -0,0 +1,28 @@
+<template>
+    <div id="app">
+        <Header/>
+        <div class="p-3" id="nav">
+            <router-view/>
+        </div>
+    </div>
+</template>
+
+<script>
+    import Header from "@/components/Header";
+
+    export default {
+        name: 'App',
+        components: {Header}
+    }
+</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..04ee46a
--- /dev/null
+++ b/custos-demo-gateway/src/assets/bots.png
Binary files 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..6d30248
--- /dev/null
+++ b/custos-demo-gateway/src/assets/credentials.png
Binary files differ
diff --git a/custos-demo-gateway/src/assets/custos-logo_custos-logo-color-v1.png b/custos-demo-gateway/src/assets/custos-logo_custos-logo-color-v1.png
new file mode 100644
index 0000000..c564182
--- /dev/null
+++ b/custos-demo-gateway/src/assets/custos-logo_custos-logo-color-v1.png
Binary files 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
--- /dev/null
+++ b/custos-demo-gateway/src/assets/custos_home.png
Binary files 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..ce18f6d
--- /dev/null
+++ b/custos-demo-gateway/src/assets/dblogs.png
Binary files 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..bc7c7f7
--- /dev/null
+++ b/custos-demo-gateway/src/assets/groups_web.png
Binary files 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
--- /dev/null
+++ b/custos-demo-gateway/src/assets/logo.png
Binary files 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..0f2e283
--- /dev/null
+++ b/custos-demo-gateway/src/assets/sharings.png
Binary files 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..0c57b21
--- /dev/null
+++ b/custos-demo-gateway/src/assets/users.png
Binary files 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..f4f3800
--- /dev/null
+++ b/custos-demo-gateway/src/components/Callback.vue
@@ -0,0 +1,58 @@
+<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()
+            await this.$router.push('workspace')
+        }
+    }
+</script>
+
+<style scoped>
+
+</style>
\ No newline at end of file
diff --git a/custos-demo-gateway/src/components/Header.vue b/custos-demo-gateway/src/components/Header.vue
new file mode 100644
index 0000000..28106da
--- /dev/null
+++ b/custos-demo-gateway/src/components/Header.vue
@@ -0,0 +1,223 @@
+<template>
+    <div v-if="showHeader()">
+        <div class="header p-3">
+            <div class="custos-logo">
+                <img src="../assets/custos-logo_custos-logo-color-v1.png" style="width: 140px;">
+            </div>
+            <div class="user-details" v-if="user">
+                <div class="username">{{user.first_name + " "+ user.last_name}}</div>
+                <div class="email">{{user.email}}</div>
+            </div>
+
+            <b-dropdown right class="ml-2" text="" no-caret toggle-class="user-avatar-button">
+                <template v-slot:button-content v-if="user">
+                    <b-icon icon="person-fill"></b-icon>
+                </template>
+                <template v-slot:button-content v-else>
+                    <b-spinner small label="Spinning"></b-spinner>
+                </template>
+
+                <template v-slot:default v-if="user">
+                    <b-dropdown-item href="#" v-on:click="$router.push('/workspace/profile')">Profile</b-dropdown-item>
+                    <b-dropdown-item v-on:click="logout">Logout</b-dropdown-item>
+                </template>
+            </b-dropdown>
+
+        </div>
+        <div class="navigation text-left">
+            <router-link to="/workspace">
+                <b-icon icon="house-door-fill"></b-icon>
+            </router-link>
+            <router-link to="/workspace/users" v-if="this.isAdmin">Users</router-link>
+            <router-link to="/workspace/groups">Groups</router-link>
+            <router-link to="/workspace/agents" v-if="this.isAdmin">Service Accounts</router-link>
+            <router-link to="/workspace/secrets">Secrets</router-link>
+            <router-link to="/workspace/sharings">Sharing</router-link>
+            <router-link to="/workspace/logs" v-if="this.isAdmin">logs</router-link>
+
+            <!--            <b-button href="#" variant="link" v-on:click="$router.push('/workspace')">-->
+            <!--                <b-icon icon="house-door-fill"></b-icon>-->
+            <!--            </b-button>-->
+            <!--            <b-button v-if="this.isAdmin" href="#" variant="link" v-on:click="$router.push('/workspace/users')">-->
+            <!--                Users-->
+            <!--            </b-button>-->
+            <!--            <b-button href="#" variant="link" v-on:click="$router.push('/workspace/groups')">-->
+            <!--                Groups-->
+            <!--            </b-button>-->
+            <!--            <b-button v-if="this.isAdmin" href="#" variant="link" v-on:click="$router.push('/workspace/agents')">-->
+            <!--                Service Accounts-->
+            <!--            </b-button>-->
+            <!--            <b-button href="#" variant="link" v-on:click="$router.push('/workspace/secrets')">-->
+            <!--                Secrets-->
+            <!--            </b-button>-->
+            <!--            <b-button href="#" variant="link" v-on:click="$router.push('/workspace/sharings')">-->
+            <!--                Sharing-->
+            <!--            </b-button>-->
+            <!--            <b-button v-if="this.isAdmin" href="#" variant="link" v-on:click="$router.push('/workspace/logs')">-->
+            <!--                Logs-->
+            <!--            </b-button>-->
+        </div>
+    </div>
+</template>
+
+<script>
+    import config from "@/config";
+
+    export default {
+        name: "Header",
+        data: function () {
+            return {
+                isAdmin: false,
+                user: null,
+                authenticated: false
+            }
+        },
+        methods: {
+            showHeader() {
+                return this.authenticated
+            },
+            async logout() {
+                await this.$store.dispatch('identity/logout', {
+                    client_id: config.value('clientId'),
+                    client_sec: config.value('clientSec'),
+                })
+
+                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 validateAuthentication() {
+                this.authenticated = await this.$store.dispatch('identity/isAuthenticated', {
+                    client_id: config.value('clientId'),
+                    client_sec: config.value('clientSec')
+                }) === true
+
+                return this.authenticated
+            },
+            async fetchAuthenticatedUser() {
+                this.isAdmin = await this.$store.dispatch('identity/isLoggedUserHasAdminAccess')
+                let currentUserName = await this.$store.dispatch('identity/getCurrentUserName')
+
+                if (await this.validateAuthentication() && (!this.user || this.user.username !== currentUserName)) {
+                    let resp = await this.$store.dispatch('user/users', {
+                        offset: 0,
+                        limit: 1,
+                        client_id: config.value('clientId'),
+                        client_sec: config.value('clientSec'),
+                        username: currentUserName
+                    })
+                    if (Array.isArray(resp) && resp.length > 0) {
+                        resp.forEach(obj => {
+                            this.user = {
+                                username: obj.username,
+                                first_name: obj.first_name,
+                                last_name: obj.last_name,
+                                email: obj.email,
+                                status: obj.state,
+                                attributes: [],
+                                roles: []
+                            }
+                        })
+                    }
+                }
+            }
+        },
+        watch: {
+            $route() {
+                this.authenticated = false
+                this.fetchAuthenticatedUser()
+            }
+        },
+        async beforeMount() {
+            this.fetchAuthenticatedUser()
+        }
+    }
+</script>
+
+<style>
+    .header {
+        display: flex;
+    }
+
+    .header .custos-logo {
+        flex: 1;
+        display: flex;
+    }
+
+    .header .custos-logo-icon {
+        width: 25px;
+        height: 25px;
+        border-radius: 8px;
+        border: solid 5px #203a43;
+        background-color: #ffffff;
+        transform: rotate(45deg);
+    }
+
+    .header .custos-logo-text {
+        font-family: Avenir;
+        font-size: 18px;
+        font-weight: 900;
+        text-align: left;
+        color: #203a43;
+
+    }
+
+    .header .user-details .username {
+        font-family: Avenir;
+        font-size: 15px;
+        font-weight: 900;
+        text-align: right;
+        color: #afafae;
+    }
+
+    .header .user-details .email {
+        font-family: Avenir;
+        font-size: 13px;
+        font-weight: 500;
+        line-height: 1.07;
+        text-align: right;
+        color: #203a43;
+    }
+
+    .header .user-avatar-button {
+        border-radius: 30px;
+        width: 35px;
+        height: 35px;
+        padding: 0px;
+        line-height: 0px;
+        font-size: 15px;
+        background-color: #4a4a4a;
+    }
+
+    .navigation {
+        background: #fe8c00;
+        background: -webkit-linear-gradient(to right, #f83600, #fe8c00);
+        background: linear-gradient(to right, #f83600, #fe8c00);
+    }
+
+    .navigation a {
+        font-family: Avenir;
+        font-size: 15px;
+        font-weight: 600;
+        text-align: left;
+        color: white;
+        padding: 5px 15px;
+        display: inline-block;
+        transition: all 0.1s;
+    }
+
+    .navigation a:hover {
+        color: white;
+    }
+
+    .navigation a:focus {
+        color: white;
+    }
+
+    .navigation a.router-link-exact-active {
+        background-color: #00000047;
+    }
+</style>
\ No newline at end of file
diff --git a/custos-demo-gateway/src/components/Password.vue b/custos-demo-gateway/src/components/Password.vue
new file mode 100644
index 0000000..184485f
--- /dev/null
+++ b/custos-demo-gateway/src/components/Password.vue
@@ -0,0 +1,45 @@
+<template>
+    <div class="generalcontainer">
+        <div style="display: flex;">
+            <input class="form-control" :id="id" :size="size" :disabled="disabled" :type="visibility" style="flex: 1;"
+                   v-bind:value="value" v-on:input="$emit('input', $event.target.value)"/>
+            <b-button variant="link" @click="showPassword()" v-if="visibility === 'password'">
+                <b-icon icon="eye-slash-fill" aria-hidden="true"></b-icon>
+            </b-button>
+
+            <b-button variant="link" @click="hidePassword()" v-if="visibility === 'text'">
+                <b-icon icon="eye" aria-hidden="true"></b-icon>
+            </b-button>
+        </div>
+    </div>
+</template>
+
+<script>
+    export default {
+        name: "Password",
+        data() {
+            return {
+                visibility: 'password'
+            }
+        },
+        props: {
+            placeholder: {type: String, default: ''},
+            id: {type: String, default: ''},
+            value: null,
+            size: {type: String, default: 'sm'},
+            maxlength: {type: Number, default: 50},
+            disabled: {type: Boolean, default: false}
+        },
+        methods: {
+            showPassword() {
+                this.visibility = 'text';
+            },
+            hidePassword() {
+                this.visibility = 'password';
+            }
+        }
+    }
+</script>
+
+<style scoped>
+</style>
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..d733488
--- /dev/null
+++ b/custos-demo-gateway/src/components/landing/Landing.vue
@@ -0,0 +1,215 @@
+<template>
+    <b-container>
+        <b-row align-v="start" align-h="center">
+            <b-col style="min-width: 300px; max-width: 100%" class="text-center">
+                <h2>Welcome to Custos</h2>
+                <p class="h2-sub">Sign up and start authenticating</p>
+                <div class="main-links">
+                    <b-link href="http://airavata.apache.org/custos/" target="_blank">Custos Website</b-link>
+                    <b-link class="ml-5"
+                            href="https://cwiki.apache.org/confluence/display/CUSTOS/Gateways+2020%3ACustos+Tutorial" target="_blank">
+                        Tutorial Instructions
+                    </b-link>
+                </div>
+                <img class="w-100" src="./../../assets/custos_home.png">
+            </b-col>
+            <b-col style="max-width: 600px;min-width: 300px;" align-h="center">
+                <b-card class="w-100 login-card">
+                    <div class="p-2">
+                        <h3 class="mb-2">(Recommended Option)</h3>
+                        <b-button class="primary-btn w-100 text-center mt-2" variant="warning"
+                                  v-on:click="this.loadAuthURL">
+                            Register or Login with your Institution Identity
+                        </b-button>
+                    </div>
+                </b-card>
+                <b-card class="w-100 mt-3 login-card">
+                    <form v-on:submit.prevent="this.login" class="p-2">
+                        <h3 class="mb-3">Login with a Custos Account</h3>
+                        <div class="p-2">
+                            <label class="form-input-label" for="form-input-username">Username</label>
+                            <b-form-input id="form-input-username" v-model="username"
+                                          placeholder="Username"></b-form-input>
+                        </div>
+                        <div class="p-2">
+                            <label class="form-input-label" for="form-input-password">Password</label>
+                            <b-form-input id="form-input-password" type="password" v-model="password"
+                                          placeholder="Password"></b-form-input>
+                        </div>
+                        <b-button class="primary-btn w-100 text-center mt-3" type="submit" variant="warning"
+                                  v-on:click="this.login" :disabled="this.loginDisabled">
+                            Login
+                            <b-spinner small v-if="this.loginDisabled"></b-spinner>
+                        </b-button>
+                        <div v-if="this.loginError" class="text-danger w-100 mt-4 text-left form-error-message">
+                            Invalid Username or Password
+                        </div>
+                        <p class="mt-3 w-100 additional-links text-center">
+                            Don't have an account?
+                            <router-link to="/register">Create an account</router-link>
+                        </p>
+                    </form>
+                </b-card>
+            </b-col>
+        </b-row>
+    </b-container>
+</template>
+
+<script>
+
+    import config from "@/config";
+    import store from "@/store";
+
+    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')
+                    } 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']
+            }
+        },
+        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"
+            let data = {
+                client_id: this.custosId,
+                client_sec: this.custosSec
+            }
+            if (await store.dispatch('identity/isAuthenticated', data) === true) {
+                await this.$router.push('workspace')
+            }
+        }
+    }
+</script>
+
+
+<!-- Add "scoped" attribute to limit CSS to this component only -->
+<style scoped>
+    h2 {
+        font-family: Avenir;
+        font-size: 35px;
+        font-weight: 900;
+        color: #203a43;
+    }
+
+    .h2-sub {
+        font-family: Avenir-Roman;
+        font-size: 22px;
+        color: #203a43;
+    }
+
+    h3 {
+        font-family: Avenir;
+        font-size: 15px;
+        font-weight: 600;
+        text-align: center;
+        color: #203a43;
+    }
+
+    .h3-sub {
+        font-family: Avenir-Roman;
+        font-size: 20px;
+        text-align: left;
+        color: #203a43;
+    }
+
+    .form-input-label {
+        font-family: Avenir;
+        font-weight: 900;
+        text-align: left;
+        float: left;
+        color: #203a43;
+    }
+
+    .primary-btn {
+        background-color: #ea6a0a;
+
+        font-family: Avenir;
+        font-size: 14px;
+        font-weight: 900;
+        text-align: left;
+        color: #ffffff;
+    }
+
+    .primary-btn:hover {
+        background-color: #da640b;
+    }
+
+    .form-error-message {
+        font-family: Avenir;
+        font-size: 14px;
+        font-weight: 900;
+        text-align: left;
+    }
+
+    .login-card {
+        box-shadow: -1px 1px 6px 2px #ebebeb;
+        border-radius: 10px;
+        border: none;
+    }
+
+    .login-card .form-input-label {
+        font-weight: 500;
+        font-size: 15px;
+    }
+
+    .main-links a {
+        font-family: Avenir;
+        font-size: 20px;
+        font-weight: 600;
+        color: #ea6a0a;
+    }
+
+    .additional-links {
+        font-size: 13px;
+    }
+
+    .additional-links a {
+        color: #ea6a0a;
+    }
+</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..9c429ce
--- /dev/null
+++ b/custos-demo-gateway/src/components/registration/CreateAccount.vue
@@ -0,0 +1,317 @@
+<template>
+    <b-container>
+        <b-row align-v="start" align-h="center">
+            <b-col style="min-width: 300px; max-width: 100%" class="text-center">
+                <h2>Welcome to Custos</h2>
+                <p class="h2-sub">Sign up and start authenticating</p>
+                <div class="main-links">
+                    <b-link href="http://airavata.apache.org/custos/" target="_blank">Custos Website</b-link>
+                    <b-link class="ml-5"
+                            href="https://cwiki.apache.org/confluence/display/CUSTOS/Gateways+2020%3ACustos+Tutorial" target="_blank">
+                        Tutorial Instructions
+                    </b-link>
+                </div>
+                <img class="w-100" src="./../../assets/custos_home.png">
+            </b-col>
+            <b-col style="max-width: 600px;min-width: 300px;" align-h="center">
+                <b-card class="w-100 login-card">
+                    <form v-on:submit.prevent="registerUser" class="p-2 text-left">
+                        <h3 class="mb-3">Create Account</h3>
+                        <div class="p-2">
+                            <label class="form-input-label" for="form-input-username">Username</label>
+                            <b-form-input size="sm" id="form-input-username" v-model="username"
+                                          :state="usernameValid && usernameIsAvailable"
+                                          placeholder="Username"></b-form-input>
+                            <b-form-invalid-feedback v-if="!usernameValid">
+                                Username should only have lowercase letters and numbers.
+                            </b-form-invalid-feedback>
+                            <b-form-invalid-feedback v-else-if="!usernameIsAvailable">
+                                Username is not available.
+                            </b-form-invalid-feedback>
+                        </div>
+                        <div class="p-2">
+                            <label class="form-input-label" for="form-input-password">Password</label>
+                            <b-form-input size="sm" id="form-input-password" type="password" v-model="password"
+                                          :state="passwordValid" placeholder="Password"></b-form-input>
+                            <b-form-invalid-feedback>
+                                Password should contain at least one upper case,
+                                lower case, one number, and 4 - 10 characters.
+                            </b-form-invalid-feedback>
+                        </div>
+                        <div class="p-2">
+                            <label class="form-input-label" for="form-input-confirm-password">Confirm Password</label>
+                            <b-form-input size="sm" id="form-input-confirm-password" type="password"
+                                          v-model="confirmedPassword" :state="confirmedPasswordValid"
+                                          placeholder="Confirm Password"></b-form-input>
+                            <b-form-invalid-feedback>
+                                Password doesn't match.
+                            </b-form-invalid-feedback>
+                        </div>
+                        <div class="p-2">
+                            <label class="form-input-label" for="form-input-email">Email</label>
+                            <b-form-input size="sm" id="form-input-email" v-model="email" :state="emailValid"
+                                          placeholder="Email"></b-form-input>
+                            <b-form-invalid-feedback>Invalid email.</b-form-invalid-feedback>
+                        </div>
+
+                        <div class="p-2">
+                            <label class="form-input-label" for="form-input-first-name">First Name</label>
+                            <b-form-input size="sm" id="form-input-first-name" v-model="firstName"
+                                          :state="firstNameValid"
+                                          placeholder="First Name"></b-form-input>
+                            <b-form-invalid-feedback>Invalid name.</b-form-invalid-feedback>
+                        </div>
+
+                        <div class="p-2">
+                            <label class="form-input-label" for="form-input-last-name">Last Name</label>
+                            <b-form-input size="sm" id="form-input-last-name" v-model="lastName" :state="lastnameValid"
+                                          placeholder="Last Name"></b-form-input>
+                            <b-form-invalid-feedback>Invalid name.</b-form-invalid-feedback>
+                        </div>
+
+                        <b-button class="primary-btn w-100 text-center mt-3" type="submit" variant="warning"
+                                  v-on:click="registerUser" :disabled="isButtonDisabled">
+                            Sign Up
+                            <b-spinner small v-if="isButtonDisabled"></b-spinner>
+                        </b-button>
+                        <p class="mt-3 w-100 additional-links text-center">
+                            Already have an account ?
+                            <router-link to="/">Login</router-link>
+                        </p>
+                    </form>
+                </b-card>
+            </b-col>
+        </b-row>
+    </b-container>
+</template>
+
+<script>
+    import config from "@/config";
+    import store from "@/store";
+
+    export default {
+        name: "CreateAccount",
+        data: function () {
+            return {
+                username: "",
+                password: "",
+                confirmedPassword: "",
+                email: "",
+                firstName: "",
+                lastName: "",
+                custosId: null,
+                custosSec: null,
+                isButtonDisabled: false,
+                usernameIsAvailable: null,
+                usernameValid: null,
+                passwordValid: null,
+                confirmedPasswordValid: null,
+                emailValid: null,
+                firstNameValid: null,
+                lastnameValid: null
+            }
+        },
+
+
+        methods: {
+
+            async registerUser() {
+                this.isButtonDisabled = true
+                let regex = /[a-z0-9].*$/;
+                let regexPucn = /[~!" "@#$%^&*()+=;"'<>,.]/
+                let passwordRegex = /^.*(?=.{4,10})(?=.*\d)(?=.*[a-zA-Z]).*$/
+                let uppercase = /[A-Z].*$/
+                // eslint-disable-next-line no-useless-escape
+                let emailRegs = /^(([^<>()\[\]\.,;:\s@\"]+(\.[^<>()\[\]\.,;:\s@\"]+)*)|(\".+\"))@(([^<>()[\]\.,;:\s@\"]+\.)+[^<>()[\]\.,;:\s@\"]{2,})$/i
+
+
+                let par = {
+                    'client_id': this.custosId,
+                    'client_sec': this.custosSec,
+                    'username': this.username,
+                }
+
+                if (this.username == null || regex.test(this.username) == false || regexPucn.test(this.username) ||
+                    uppercase.test(this.username)) {
+                    this.usernameValid = false
+                } else {
+                    this.usernameValid = true
+                    try {
+                        let usernameIsAvailable = await this.$store.dispatch('user/checkUsernameIsValid', par)
+                        if (!usernameIsAvailable) {
+                            this.usernameIsAvailable = false
+                        } else {
+                            this.usernameIsAvailable = true
+                        }
+                    } catch (e) {
+                        this.usernameIsAvailable = false
+                    }
+                }
+
+
+                if (this.password == null || !passwordRegex.test(this.password)) {
+                    this.passwordValid = false
+                } else {
+                    this.passwordValid = true
+                }
+
+
+                if (this.password !== this.confirmedPassword) {
+                    this.confirmedPasswordValid = false
+                } else {
+                    this.confirmedPasswordValid = true
+                }
+
+                if (this.email == null || !emailRegs.test(this.email)) {
+                    this.emailValid = false
+                } else {
+                    this.emailValid = true
+                }
+
+                if (this.firstName == null || this.firstName == '' || this.firstName.length > 20) {
+                    this.firstNameValid = false
+                } else {
+                    this.firstNameValid = true
+                }
+
+                if (this.lastName == null || this.lastName == '' || this.lastName.length > 20) {
+                    this.lastnameValid = false
+                } else {
+                    this.lastnameValid = true
+                }
+
+
+                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
+                }
+
+                if (this.usernameIsAvailable && this.usernameValid && this.passwordValid && this.confirmedPasswordValid && this.lastnameValid && this.firstNameValid && this.emailValid) {
+                    let status = await this.$store.dispatch('user/registerUser', body)
+                    if (status) {
+                        await this.$router.push("/")
+                    }
+                }
+
+                this.isButtonDisabled = false
+            },
+
+            async callDismissed() {
+                this.usernameIsAvailable = true
+                this.usernameValid = true
+                this.passwordValid = true
+                this.emailValid = true
+                this.firstNameValid = true
+                this.lastnameValid = true
+                this.isButtonDisabled = false
+            },
+
+
+        },
+
+        async mounted() {
+            this.custosId = config.value('clientId')
+            this.custosSec = config.value('clientSec')
+            let data = {
+                client_id: this.custosId,
+                client_sec: this.custosSec
+            }
+            if (await store.dispatch('identity/isAuthenticated', data) == true) {
+
+                await this.$router.push('workspace')
+            }
+        }
+    }
+</script>
+
+<!-- Add "scoped" attribute to limit CSS to this component only -->
+<style scoped>
+    h2 {
+        font-family: Avenir;
+        font-size: 35px;
+        font-weight: 900;
+        color: #203a43;
+    }
+
+    .h2-sub {
+        font-family: Avenir-Roman;
+        font-size: 22px;
+        color: #203a43;
+    }
+
+    h3 {
+        font-family: Avenir;
+        font-size: 15px;
+        font-weight: 600;
+        text-align: center;
+        color: #203a43;
+    }
+
+    .h3-sub {
+        font-family: Avenir-Roman;
+        font-size: 20px;
+        text-align: left;
+        color: #203a43;
+    }
+
+    .form-input-label {
+        font-family: Avenir;
+        font-weight: 900;
+        text-align: left;
+        float: left;
+        color: #203a43;
+    }
+
+    .primary-btn {
+        background-color: #ea6a0a;
+
+        font-family: Avenir;
+        font-size: 14px;
+        font-weight: 900;
+        text-align: left;
+        color: #ffffff;
+    }
+
+    .primary-btn:hover {
+        background-color: #da640b;
+    }
+
+    .form-error-message {
+        font-family: Avenir;
+        font-size: 14px;
+        font-weight: 900;
+        text-align: left;
+    }
+
+    .login-card {
+        box-shadow: -1px 1px 6px 2px #ebebeb;
+        border-radius: 10px;
+        border: none;
+    }
+
+    .login-card .form-input-label {
+        font-weight: 500;
+        font-size: 15px;
+    }
+
+    .main-links a {
+        font-family: Avenir;
+        font-size: 20px;
+        font-weight: 600;
+        color: #ea6a0a;
+    }
+
+    .additional-links {
+        font-size: 13px;
+    }
+
+    .additional-links a {
+        color: #ea6a0a;
+    }
+</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..30423ea
--- /dev/null
+++ b/custos-demo-gateway/src/components/workspace/Agents.vue
@@ -0,0 +1,348 @@
+<template>
+    <div>
+        <div class="w-100 mb-5">
+            <h2>Service Accounts</h2>
+        </div>
+        <!--        <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>-->
+        <b-container>
+            <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 small striped hover responsive :items="communityAccounts" :fields="community_fields" selectable
+                     ref="selectableTable" select-mode="single" @row-selected="onCommunityAcSelected" caption-top>
+                <template v-slot:head(id)>Service Account ID</template>
+                <template v-slot:cell(status)="data">
+                    <b-badge v-if="data.value == 'ACTIVE'" variant="success">Active</b-badge>
+                    <b-badge v-else-if="data.value == 'DEACTIVE'" variant="danger">Inactive</b-badge>
+                    <b-badge v-else-if="data.value == 'PENDING'" variant="warning">Pending</b-badge>
+                </template>
+            </b-table>
+            <div>
+                <b-button variant="outline-primary" v-on:click="this.addAccount">Add Service Account</b-button>
+            </div>
+        </b-container>
+        <b-modal ref="newAcc" id="add-account-modal" scrollable title="New Account">
+            <div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-account-id">Service Account ID</label>
+                    <b-form-input id="form-input-account-id" size="sm" v-model="newAccountName"></b-form-input>
+                </div>
+            </div>
+            <template slot="modal-footer">
+                <b-button size="sm" class="mr-2" @click="$bvModal.hide('add-account-modal')">
+                    Cancel
+                </b-button>
+                <b-button size="sm" variant="primary" v-on:click="registerAgent" :disabled="!newAccountName"
+                          @click="$bvModal.hide('add-account-modal')">
+                    Add
+                </b-button>
+            </template>
+        </b-modal>
+        <b-modal ref="exAcc" id="view-account-modal" scrollable title="Account">
+            <div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-account-id">Service Account ID</label>
+                    <b-form-input id="form-input-account-id" size="sm" v-model="exAccountId" disabled></b-form-input>
+                </div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-account-status">Status</label>
+                    <b-form-select id="form-input-account-status" size="sm" 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>
+                    <strong>Attributes</strong>
+                    <b-button variant="link" v-on:click="addActAttribute"> + Add Attribute</b-button>
+                    <div class="w-100">
+                        <small v-if="accSelectedAttributes.length === 0">There are no attributes created.</small>
+                    </div>
+                    <b-table v-if="accSelectedAttributes.length > 0" small striped hover responsive
+                             :items="accSelectedAttributes" selectable
+                             select-mode="single"
+                             @row-selected="onAccAtrSelected">
+                    </b-table>
+                </div>
+            </div>
+            <template slot="modal-footer">
+                <b-button size="sm" class="mr-2" @click="$bvModal.hide('view-account-modal')">
+                    Cancel
+                </b-button>
+                <b-button size="sm" variant="primary" @click="$bvModal.hide('view-account-modal')">
+                    Save
+                </b-button>
+            </template>
+        </b-modal>
+        <b-modal ref="accAtrModel" id="add-attribute-modal" scrollable title="Add Attribute">
+            <div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-attribute-key">Key</label>
+                    <b-form-input id="form-input-attribute-key" size="sm" v-model="newAcAtrKey"></b-form-input>
+                </div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-attribute-value">Value</label>
+                    <b-form-input id="form-input-attribute-value" size="sm" v-model="newAcAtrValue"></b-form-input>
+                </div>
+            </div>
+            <template slot="modal-footer">
+                <b-button size="sm" class="mr-2" @click="$bvModal.hide('add-attribute-modal')">
+                    Cancel
+                </b-button>
+                <b-button size="sm" variant="primary" v-on:click="addAtrOkPressed"
+                          :disabled="!newAcAtrKey || !newAcAtrValue" @click="$bvModal.hide('add-attribute-modal')">
+                    Add Attribute
+                </b-button>
+            </template>
+        </b-modal>
+        <b-modal ref="AccAtrModelSelected" id="view-attribute-modal" scrollable title="Attribute" ok-title="Delete"
+                 @ok="deleteAtrOkPressed">
+            <div class="userform">
+                <div class="userformItem">
+                    <label class="form-input-label" for="form-input-attribute-key">Key</label>
+                    <b-form-input id="form-input-attribute-key" size="sm" v-model="exAccountAtrKey"
+                                  disabled></b-form-input>
+                </div>
+                <div class="userformItem">
+                    <label class="form-input-label" for="form-input-attribute-value">Value</label>
+                    <b-form-input id="form-input-attribute-value" size="sm" v-model="exAccountAtrValue"
+                                  disabled></b-form-input>
+                </div>
+            </div>
+            <template slot="modal-footer">
+                <b-button size="sm" class="mr-2" @click="$bvModal.hide('view-attribute-modal')">
+                    Cancel
+                </b-button>
+                <b-button size="sm" variant="danger" v-on:click="deleteAtrOkPressed"
+                          @click="$bvModal.hide('view-attribute-modal')">
+                    Delete
+                </b-button>
+            </template>
+        </b-modal>
+        <b-modal ref="enablePopOver" id="account-credentials-view-modal" scrollable title="Agent Credentials">
+            <div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-credential-id">ID</label>
+                    <b-form-input id="form-input-credential-id" size="sm" v-model="agentId" disabled></b-form-input>
+                </div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-credential-secret">Secret</label>
+                    <b-form-input id="form-input-credential-secret" size="sm" v-model="agentSec"
+                                  disabled></b-form-input>
+                </div>
+            </div>
+            <template slot="modal-footer">
+                <b-button size="sm" variant="primary" @click="$bvModal.hide('account-credentials-view-modal')">
+                    Ok
+                </b-button>
+            </template>
+        </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 goToWorkspace() {
+                await this.$router.push('/workspace')
+            },
+
+
+        },
+
+        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>
+    h2 {
+        font-family: Avenir;
+        font-size: 20px;
+        font-weight: 600;
+        text-align: left;
+        color: #203a43;
+    }
+</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..319aa38
--- /dev/null
+++ b/custos-demo-gateway/src/components/workspace/Groups.vue
@@ -0,0 +1,693 @@
+<template>
+    <div>
+        <div class="w-100 mb-5">
+            <h2>Manage Groups</h2>
+        </div>
+        <b-container>
+            <div>
+                <b-alert v-model="groupError" variant="danger" dismissible
+                         @dismissed="callDismissed">
+                    Group name not available
+                </b-alert>
+            </div>
+            <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 small striped hover responsive :items="groupItems" :fields="fields" selectable
+                     ref="selectableTable" select-mode="single" @row-selected="onRowSelected" caption-top>
+                <template v-slot:head(id)>ID</template>
+                <template v-slot:head(ownerId)>Owner</template>
+            </b-table>
+            <div class="addGr">
+                <b-button variant="outline-primary" v-on:click="addGr">Add Group</b-button>
+            </div>
+        </b-container>
+        <b-modal ref="groupmodel" id="group-profile-modal" scrollable title="Group Profile" ok-title="Update">
+            <div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-group-name">Name</label>
+                    <b-form-input id="form-input-group-name" size="sm" v-model="selectedName"
+                                  disabled></b-form-input>
+                </div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-group-description">Description</label>
+                    <b-form-input id="form-input-group-description" size="sm"
+                                  v-model="selectedDescription"></b-form-input>
+                </div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-group-owner-id">Owner</label>
+                    <b-form-input id="form-input-group-owner-id" size="sm" 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="w-100 mt-5">
+                    <strong>Members</strong>
+                    <b-button variant="link" :disabled="disableAccess" v-on:click="addMemberShip"> + Add Member
+                    </b-button>
+                    <div class="w-100" v-if="members.length === 0"><small>There are no members.</small></div>
+                    <b-table v-if="members.length > 0" class="mt-3" small striped hover responsive :items="members"
+                             selectable
+                             select-mode="single"
+                             @row-selected="onMemberShipSelected">
+                        <template v-slot:head(user_id)>Username</template>
+                        <template v-slot:head(type)>Permission Type</template>
+                    </b-table>
+                </div>
+
+                <div class="w-100 mt-5">
+                    <strong>Child Groups</strong>
+                    <b-button variant="link" :disabled="disableAccess" v-on:click="addChildGroup">
+                        + Add Child Group
+                    </b-button>
+                    <div class="w-100" v-if="childGroupMembers.length === 0"><small>There are no child
+                        groups.</small></div>
+                    <b-table v-if="childGroupMembers.length > 0" class="mt-3" small striped hover responsive
+                             :items="childGroupMembers"
+                             :fields="memberGroupsFields"
+                             selectable select-mode="single"
+                             @row-selected="onGroupMemberShipSelected">
+                        <template v-slot:head(name)>Group Name</template>
+                        <template v-slot:head(id)>ID</template>
+                    </b-table>
+                </div>
+
+            </div>
+            <template v-slot:modal-footer>
+                <div class="w-100">
+                    <b-button variant="primary" size="sm" class="mr-2" v-on:click="closeGroupProfile"
+                              @click="$bvModal.hide('group-profile-modal')" :disabled="disableAccess">
+                        Close
+                    </b-button>
+                    <b-button variant="danger" size="sm" class="mr-2" v-on:click="removeGroupProfile"
+                              @click="$bvModal.hide('group-profile-modal')" :disabled="disableAccess">
+                        Delete
+                    </b-button>
+                    <b-button variant="primary" size="sm" class="mr-2" v-on:click="updateGroupProfile"
+                              @click="$bvModal.hide('group-profile-modal')" :disabled="disableAccess">
+                        Update
+                    </b-button>
+                </div>
+            </template>
+        </b-modal>
+        <b-modal ref="membershipModel" id="update-membership-modal" title="Update Membership" ok-title="Update">
+            <div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-username">User</label>
+                    <b-form-input id="form-input-username" size="sm" v-model="selectedMembershipUsername"
+                                  disabled></b-form-input>
+                </div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-permission-type">Permission Type</label>
+                    <b-form-select id="form-input-permission-type" size="sm" 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 size="sm" class="mr-2" v-on:click="closeMembershipModel"
+                              @click="$bvModal.hide('update-membership-modal')">
+                        Close
+                    </b-button>
+                    <b-button variant="danger" size="sm" class="mr-2" v-on:click="removeMembership"
+                              @click="$bvModal.hide('update-membership-modal')">
+                        Remove Membership
+                    </b-button>
+                    <b-button variant="primary" class="mr-2" size="sm" v-on:click="updateMembership"
+                              @click="$bvModal.hide('update-membership-modal')">
+                        Update
+                    </b-button>
+                </div>
+            </template>
+        </b-modal>
+        <b-modal ref="addMembershipModel" id="add-membership-modal" title="Add Membership">
+            <div v-if="usernames.length > 0">
+                <div v-if="selectedChildType==='User'" class="p-2">
+                    <label class="form-input-label" for="form-input-username">Username</label>
+                    <b-form-select id="form-input-username" size="sm" 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="p-2">
+                    <label class="form-input-label" for="form-input-type">Type</label>
+                    <b-form-select id="form-input-type" size="sm" v-model="selectedNewMemType">
+                        <option v-for="(selectOption, indexOpt) in memberTypes"
+                                :key="indexOpt"
+                                :value="selectOption"
+                        >
+                            {{ selectOption }}
+                        </option>
+                    </b-form-select>
+                </div>
+            </div>
+
+            <small v-if="usernames.length === 0" class="p-3 text-danger">
+                There are no available members to be added.
+            </small>
+
+            <template v-slot:modal-footer>
+                <b-button size="sm" @click="$bvModal.hide('add-membership-modal')">
+                    Cancel
+                </b-button>
+                <b-button size="sm" class="mr-2" variant="primary" :disabled="usernames.length === 0"
+                          v-on:click="addMembershipOKPressed" @click="$bvModal.hide('add-membership-modal')">
+                    Add Membership
+                </b-button>
+            </template>
+        </b-modal>
+        <b-modal ref="addGrModel" id="add-group-modal" title="Add Group" ok-title="Add" @ok="addGroupOKPressed">
+            <div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-group-name">Name</label>
+                    <b-form-input id="form-input-group-name" size="sm"
+                                  v-model="selectedNewGrName"></b-form-input>
+                </div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-group-name">Description</label>
+                    <b-form-input id="form-input-group-description" size="sm"
+                                  v-model="selectedNewGrDesc"></b-form-input>
+                </div>
+            </div>
+            <template v-slot:modal-footer>
+                <b-button size="sm" class="mr-2" @click="$bvModal.hide('add-group-modal')">
+                    Cancel
+                </b-button>
+                <b-button :disabled="!selectedNewGrName || selectedNewGrName.length === 0" size="sm"
+                          variant="primary" v-on:click="addGroupOKPressed"
+                          @click="$bvModal.hide('add-group-modal')">
+                    Add Group
+                </b-button>
+            </template>
+        </b-modal>
+        <b-modal ref="viewGrMembership" id="view-group-membership-modal" title="Group">
+            <div class="groupform">
+                <div class="groupformItem">
+                    <label class="form-input-label" for="form-input-group-name">Name</label>
+                    <b-form-input id="form-input-group-name" size="sm" v-model="selectedGrName"
+                                  disabled></b-form-input>
+                </div>
+                <div class="groupformItem">
+                    <label class="form-input-label" for="form-input-id">ID</label>
+                    <b-form-input id="form-input-id" size="sm" v-model="selectedGrId" disabled></b-form-input>
+                </div>
+            </div>
+            <template v-slot:modal-footer>
+                <b-button size="sm" class="mr-2" @click="$bvModal.hide('view-group-membership-modal')">
+                    Cancel
+                </b-button>
+                <b-button size="sm" variant="danger" v-on:click="removeGroupMembership"
+                          @click="$bvModal.hide('view-group-membership-modal')">
+                    Delete Child Group
+                </b-button>
+            </template>
+        </b-modal>
+        <b-modal ref="addGrMembershipModel" id="add-group-membership-modal" title="Add Group Membership">
+            <div class="p-2" v-if="feasibleGroupMembers.length > 0">
+                <label class="form-input-label" for="form-input-group-name">Select Group</label>
+                <b-form-select id="form-input-group-name" size="sm"
+                               v-model="addingGr">
+                    <option v-for="(selectOption, indexOpt) in feasibleGroupMembers"
+                            :key="indexOpt"
+                            :value="selectOption"
+                    >
+                        {{ selectOption.name }}
+                    </option>
+                </b-form-select>
+            </div>
+            <small v-if="feasibleGroupMembers.length === 0" class="p-3 text-danger">
+                There are no available groups to be added as child groups.
+            </small>
+            <template v-slot:modal-footer>
+                <b-button size="sm" class="mr-2" @click="$bvModal.hide('add-group-membership-modal')">
+                    Cancel
+                </b-button>
+                <b-button :disabled="feasibleGroupMembers.length === 0" size="sm" variant="primary"
+                          v-on:click="addChildGroupOkPressed"
+                          @click="$bvModal.hide('add-group-membership-modal')">
+                    Add
+                </b-button>
+            </template>
+        </b-modal>
+    </div>
+</template>
+
+<script>
+    import config from "@/config";
+
+    export default {
+        name: "Groups",
+        data: function () {
+            return {
+                fields: ['name', 'id', 'description', 'ownerId'],
+                groupItems: [],
+                members: [],
+                childGroupMembers: [],
+                feasibleGroupMembers: [],
+                usernames: [],
+                exGroups: [
+                    '1234-edrfg-567-rfgbn', '1234-ed-5634-rfn'
+                ],
+                memberTypes: [
+                    'ADMIN', 'MEMBER'],
+                memberGroupsFields: ['name', 'id'],
+                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,
+                currentUser: null,
+                groupError: null,
+                disableAccess: false
+
+            }
+        },
+        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
+
+                    if (!this.isAdminUser && this.selectedOwnerId != this.currentUser) {
+
+                        let data = {
+                            client_id: this.custosId,
+                            client_sec: this.custosSec,
+                            groupId: this.selectedId,
+                            username: this.currentUser,
+                            type: 'ADMIN'
+                        }
+
+                        let resp = await this.$store.dispatch('group/hasAccess', data)
+
+                        if (!resp.status) {
+                            this.disableAccess = true
+
+                        }
+
+                    } else {
+                        this.disableAccess = false
+                    }
+
+                    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()
+                }
+            },
+
+            async addChildGroup() {
+                let grs = []
+                let dat = {
+                    client_id: this.custosId,
+                    client_sec: this.custosSec,
+                    groupId: this.selectedId
+                }
+
+                let parentGroups = await this.$store.dispatch('group/getAllParentGroups', dat)
+                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 (parentGroups != null && parentGroups.length > 0) {
+                        parentGroups.forEach(pGr => {
+                            if (gr.id === pGr.id) {
+                                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,
+                    username: this.currentUser,
+                    body: {
+                        groups: [{
+                            name: this.selectedNewGrName,
+                            description: this.selectedNewGrDesc,
+                            ownerId: username,
+                            realm_roles: [],
+                            client_roles: [],
+                            attributes: [],
+                            sub_groups: []
+                        }]
+                    }
+                }
+                let response = await this.$store.dispatch('group/createGroup', data)
+                console.log(response)
+                if (!response) {
+                    this.groupError = true
+                    this.groupsLoading = false
+                } else {
+                    await this.loadGroups(data)
+                    this.groupsLoading = false
+                    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,
+                    username: this.currentUser
+                }
+                await this.loadGroups(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,
+                    username: this.currentUser
+                }
+                await this.loadGroups(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 loadGroups(data) {
+                if (this.isAdminUser) {
+                    this.groupItems = await this.$store.dispatch('group/loadAllGroups', data)
+                } else {
+                    let grs = []
+                    grs = await this.$store.dispatch('group/getAllGroupsOfUser', data)
+                    let groups = []
+                    grs.forEach(gr => {
+                        gr.ownerId = gr.owner_id
+                        groups.push(gr)
+                    })
+                    this.groupItems = groups
+                }
+            },
+
+            async goToWorkspace() {
+                await this.$router.push('/workspace')
+            },
+
+            async callDismissed() {
+                this.groupError = false
+            },
+
+
+        },
+        async mounted() {
+            this.groupsLoading = true
+            this.custosId = config.value('clientId')
+            this.custosSec = config.value('clientSec')
+            this.isAdminUser = await this.$store.dispatch('identity/isLoggedUserHasAdminAccess')
+            this.currentUser = await this.$store.dispatch('identity/getCurrentUserName')
+            let data = {
+                client_id: this.custosId,
+                client_sec: this.custosSec,
+                username: this.currentUser
+            }
+            await this.loadGroups(data)
+            this.groupsLoading = false
+            this.loadUsers()
+        }
+    }
+</script>
+
+<style scoped>
+    h2 {
+        font-family: Avenir;
+        font-size: 20px;
+        font-weight: 600;
+        text-align: left;
+        color: #203a43;
+    }
+</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..ea4dc9c
--- /dev/null
+++ b/custos-demo-gateway/src/components/workspace/Logs.vue
@@ -0,0 +1,220 @@
+<template>
+    <div>
+        <div class="w-100 mb-5">
+            <h2>Access Logs</h2>
+        </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>
+                <b-input-group>
+                    <template v-slot:prepend>
+                        <b-form-input size="sm" disabled v-model="defaultSearchText">Search By</b-form-input>
+                        <b-dropdown size="sm" :text="selectedService" 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="mt-3">
+                <b-table small striped hover responsive :items="logItems" :fields="fields" selectable
+                         ref="selectableTable"
+                         select-mode="single"
+                         :per-page="perPage"
+                         :current-page="currentPage"
+                         caption-top>
+                </b-table>
+                <div class="pgClass">
+                    <b-pagination size="sm" v-model="currentPage" :total-rows="rows" :per-page="perPage"
+                                  aria-controls="my-table" class="float-right">
+
+                    </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'],
+                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(0)
+                    date.setUTCSeconds(res.created_time/1000)
+                    res.accessed_time = date.toUTCString()
+                })
+                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
+                }
+            },
+
+            async goToWorkspace() {
+                await this.$router.push('/workspace')
+            },
+
+        },
+        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(0)
+                    date.setUTCSeconds(res.created_time/1000)
+                    res.accessed_time = date.toUTCString()
+                })
+                this.logItems = response
+
+            }
+
+        }
+    }
+</script>
+
+<style scoped>
+    h2 {
+        font-family: Avenir;
+        font-size: 20px;
+        font-weight: 600;
+        text-align: left;
+        color: #203a43;
+    }
+</style>
\ No newline at end of file
diff --git a/custos-demo-gateway/src/components/workspace/Profile.vue b/custos-demo-gateway/src/components/workspace/Profile.vue
new file mode 100644
index 0000000..0d6c50f
--- /dev/null
+++ b/custos-demo-gateway/src/components/workspace/Profile.vue
@@ -0,0 +1,336 @@
+<template>
+    <div>
+        <div class="w-100 mb-5">
+            <h2>User Profile</h2>
+        </div>
+        <b-container>
+            <b-row class="text-left">
+                <b-col style="max-width: 300px; min-width: 200px;" class="p-2">
+                    <label class="form-input-label" for="form-input-username">Username</label>
+                    <b-form-input id="form-input-username" size="sm" v-model="currentUserName" type="email"
+                                  disabled></b-form-input>
+                </b-col>
+                <b-col style="max-width: 300px; min-width: 200px;" class="p-2">
+                    <label class="form-input-label" for="form-input-status">Status</label>
+                    <b-form-input id="form-input-status" size="sm" v-model="status" required
+                                  disabled></b-form-input>
+                </b-col>
+
+                <b-col style="max-width: 300px; min-width: 200px;" class="p-2">
+                    <label class="form-input-label" for="form-input-first-name">First Name</label>
+                    <div>
+                        <b-alert v-model="firstNameError" variant="danger" dismissible
+                                 @dismissed="this.callDismissed">
+                            Invalid name
+                        </b-alert>
+                    </div>
+                    <b-form-input id="form-input-first-name" size="sm" v-model="first_name" required></b-form-input>
+                </b-col>
+
+                <b-col style="max-width: 300px; min-width: 200px;" class="p-2">
+                    <label class="form-input-label" for="form-input-last-name">Last Name</label>
+                    <div>
+                        <b-alert v-model="lastnameError" variant="danger" dismissible
+                                 @dismissed="this.callDismissed">
+                            Invalid name
+                        </b-alert>
+                    </div>
+                    <b-form-input id="form-input-last-name" size="sm" v-model="last_name" required></b-form-input>
+                </b-col>
+
+                <b-col style="max-width: 300px; min-width: 200px;" class="p-2">
+                    <label class="form-input-label" for="form-input-email">Email</label>
+                    <div>
+                        <b-alert v-model="emailError" variant="danger" dismissible
+                                 @dismissed="this.callDismissed">
+                            Invalid email
+                        </b-alert>
+                    </div>
+                    <b-form-input id="form-input-email" size="sm" v-model="email" required></b-form-input>
+                </b-col>
+            </b-row>
+            <b-row class="text-left mt-5">
+                <b-col style="max-width: 100%; min-width: 300px;" class="p-3">
+                    <strong>My Roles</strong>
+                    <div class="w-100">
+                        <small v-if="roles.length === 0">There are no roles assigned.</small>
+                    </div>
+                    <b-table small striped hover responsive :items="roles" class="mt-3" ref="selectableTable"
+                             select-mode="single">
+                    </b-table>
+                </b-col>
+                <b-col style="max-width: 100%; min-width: 300px;" class="p-3">
+                    <strong>My Attributes</strong>
+                    <!--                    <b-button variant="link" v-on:click="addAttribute"> + Add Attribute</b-button>-->
+                    <div class="w-100">
+                        <small v-if="attributes.length === 0">There are no attributes created.</small>
+                    </div>
+                    <b-table small striped hover responsive :items="attributes" class="mt-3" ref="selectableTable"
+                             select-mode="single">
+                    </b-table>
+                </b-col>
+            </b-row>
+            <b-row class="w-100 text-right mt-5">
+                <b-col v-if="this.updatingProfile">
+                    <b-spinner variant="primary" label="Text Centered"></b-spinner>
+                    Saving ...
+                </b-col>
+                <b-col>
+                    <b-button size="sm" variant="primary" v-on:click="updateProfile">Save Changes</b-button>
+                </b-col>
+            </b-row>
+        </b-container>
+        <b-modal ref="atrModel" id="user-attribute-modal" scrollable title="Add Attribute" ok-title="Add"
+                 @ok="addAtrOkPressed">
+            <div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-attribute-key">Key</label>
+                    <b-form-input id="form-input-attribute-key" size="sm" v-model="newKey"></b-form-input>
+                </div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-attribute-value">Value</label>
+                    <b-form-input id="form-input-attribute-value" size="sm" v-model="newValue"></b-form-input>
+                </div>
+            </div>
+            <template v-slot:modal-footer>
+                <b-button size="sm" class="mr-2" @click="$bvModal.hide('user-attribute-modal')">
+                    Cancel
+                </b-button>
+                <b-button :disabled="!newKey || !newValue" size="sm" variant="primary" v-on:click="addAtrOkPressed"
+                          @click="$bvModal.hide('user-attribute-modal')">
+                    Add
+                </b-button>
+            </template>
+        </b-modal>
+        <b-modal ref="selectModel" id="user-attribute-view-modal" scrollable title="Delete Attribute">
+            <div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-attribute-key">Key</label>
+                    <b-form-input id="form-input-attribute-key" size="sm" disabled
+                                  v-model="selectedKey"></b-form-input>
+                </div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-attribute-value">Value</label>
+                    <b-form-input id="form-input-attribute-value" size="sm" disabled
+                                  v-model="selectedValue"></b-form-input>
+                </div>
+            </div>
+            <template v-slot:modal-footer>
+                <b-button size="sm" class="mr-2" @click="$bvModal.hide('user-attribute-view-modal')">
+                    Cancel
+                </b-button>
+                <b-button size="sm" variant="danger" v-on:click="removeAtrOKPressed"
+                          @click="$bvModal.hide('user-attribute-view-modal')">
+                    Delete
+                </b-button>
+            </template>
+        </b-modal>
+    </div>
+</template>
+
+<script>
+    import config from "@/config";
+
+    export default {
+        name: "Profile",
+        data: function () {
+            return {
+                user: null,
+                currentUserName: null,
+                first_name: null,
+                last_name: null,
+                email: null,
+                attributes: [],
+                roles: [],
+                status: null,
+                newKey: null,
+                newValue: null,
+                selectedKey: null,
+                selectedValue: null,
+                operationCompleted: true,
+                updatingProfile: false,
+                emailError: false,
+                firstNameError: false,
+                lastnameError: false
+            }
+        },
+
+        methods: {
+
+            addAttribute: function () {
+                this.$refs.atrModel.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.currentUserName]
+                    }
+                }
+                let userAtr = await this.$store.dispatch('user/addUserAttributes', bd)
+                if (userAtr) {
+                    let atr = {key: this.newKey, value: this.newValue}
+                    this.attributes.push(atr)
+                }
+                this.operationCompleted = true
+                this.newKey = null
+                this.newValue = null
+
+            },
+
+            async removeAtrOKPressed() {
+                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.currentUserName]
+                    }
+                }
+                let userAtr = await this.$store.dispatch('user/deleteUserAttributes', bd)
+                if (userAtr) {
+                    let newAtr = []
+                    this.attributes.forEach(atr => {
+
+                        if (atr.key != this.selectedKey) {
+                            newAtr.push(atr)
+                        }
+                        this.attributes = newAtr
+                    })
+                }
+                this.operationCompleted = true
+                this.selectedKey = null
+                this.selectedValue = null
+            },
+
+            onAtrSelected: function (items) {
+                if (items != null && items.length > 0) {
+                    this.selectedKey = items[0].key
+                    this.selectedValue = items[0].value
+                    this.$refs.selectModel.show()
+                }
+            },
+
+            async updateProfile() {
+                this.updatingProfile = true
+
+                // eslint-disable-next-line no-useless-escape
+                let emailRegs = /^(([^<>()\[\]\.,;:\s@\"]+(\.[^<>()\[\]\.,;:\s@\"]+)*)|(\".+\"))@(([^<>()[\]\.,;:\s@\"]+\.)+[^<>()[\]\.,;:\s@\"]{2,})$/i
+                let regexPucn = /[~!" "@#$%^&*()+=;"'<>,.]/
+
+                if (this.first_name == null || this.first_name == '' || regexPucn.test(this.first_name) || this.first_name.length > 20) {
+                    this.firstNameError = true
+                    this.updatingProfile = false
+                }
+
+                if (this.last_name == null || this.last_name == '' || regexPucn.test(this.last_name) || this.last_name.length > 20) {
+                    this.lastnameError = true
+                    this.updatingProfile = false
+                }
+
+
+                if (this.email == null || emailRegs.test(this.email) == false) {
+                    this.emailError = true
+                    this.updatingProfile = false
+                }
+
+                if (!(this.firstNameError || this.lastnameError || this.emailError)) {
+
+                    let data = {
+                        client_id: this.custosId,
+                        client_sec: this.custosSec,
+                        body: {
+                            username: this.currentUserName,
+                            first_name: this.first_name,
+                            last_name: this.last_name,
+                            email: this.email
+                        }
+                    }
+                    await this.$store.dispatch('user/updateUserProfile', data)
+                    this.updatingProfile = false
+                }
+            },
+            async goToWorkspace() {
+                await this.$router.push('/workspace')
+            },
+
+
+            async callDismissed() {
+                this.emailError = false
+                this.firstNameError = false
+                this.lastnameError = false
+            }
+
+
+        },
+
+        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 data = {
+                offset: 0, limit: 1, client_id: this.custosId, client_sec: this.custosSec,
+                username: this.currentUserName
+            }
+            let resp = await this.$store.dispatch('user/users', data)
+            if (Array.isArray(resp) && resp.length > 0) {
+                resp.forEach(obj => {
+                    this.user = {
+                        username: obj.username,
+                        first_name: obj.first_name,
+                        last_name: obj.last_name,
+                        email: obj.email,
+                        status: obj.state,
+                        attributes: [],
+                        roles: []
+                    }
+
+                    obj.realm_roles.forEach(role => {
+                        let r = {
+                            name: role
+                        }
+                        this.roles.push(r)
+                    })
+                    let attribs = obj.attributes
+                    attribs.forEach(r => {
+                        let newAt = {
+                            key: r.key,
+                            value: r.values.join(",")
+                        }
+                        this.attributes.push(newAt)
+                    })
+
+
+                    this.user = obj
+                    this.first_name = obj.first_name
+                    this.last_name = obj.last_name
+                    this.email = obj.email
+                    this.status = obj.state
+
+
+                })
+            }
+        }
+    }
+
+</script>
+
+<style scoped>
+    h2 {
+        font-family: Avenir;
+        font-size: 20px;
+        font-weight: 600;
+        text-align: left;
+        color: #203a43;
+    }
+</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..9899b43
--- /dev/null
+++ b/custos-demo-gateway/src/components/workspace/Secrets.vue
@@ -0,0 +1,389 @@
+<template>
+    <div>
+        <div class="w-100">
+            <h2>Secrets</h2>
+        </div>
+        <b-container class="w-100 mt-5">
+            <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 small striped hover responsive :items="secItems" :fields="fields" selectable
+                     select-mode="single" @row-selected="onRowSelected" caption-top>
+                <template v-slot:head(owner_id)>Owner</template>
+            </b-table>
+            <div class="w-100">
+                <b-button variant="outline-primary" v-on:click="addSec">Add Secret</b-button>
+            </div>
+        </b-container>
+        <b-modal ref="addSecmodel" id="add-secret-modal" scrollable title="Add Secret">
+            <div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-secret-type">Type</label>
+                    <b-form-select id="form-input-secret-type" size="sm" v-model="defaultMemType">
+                        <option v-for="(selectOption, indexOpt) in secretTypes"
+                                :key="indexOpt"
+                                :value="selectOption"
+                        >
+                            {{ selectOption }}
+                        </option>
+                    </b-form-select>
+                </div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-secret-description">Description</label>
+                    <b-form-input id="form-input-secret-description" size="sm"
+                                  v-model="selectedDescription"></b-form-input>
+                </div>
+                <div v-if="defaultMemType==='Password'" class="p-2">
+                    <label class="form-input-label" for="form-input-secret-password">Password</label>
+                    <Password id="form-input-secret-password" size="sm" v-model="selectedPassword">
+                    </Password>
+                </div>
+            </div>
+            <template slot="modal-footer">
+                <b-button size="sm" class="mr-2" @click="$bvModal.hide('add-secret-modal')">
+                    Cancel
+                </b-button>
+                <b-button size="sm" variant="primary" v-on:click="addSecOKButtonPressed"
+                          :disabled="defaultMemType === 'Password' && !selectedPassword"
+                          @click="$bvModal.hide('add-secret-modal')">
+                    Add Secret
+                </b-button>
+            </template>
+        </b-modal>
+        <b-modal ref="updateSecmodel" id="view-secret-modal" scrollable title="View Secret">
+            <div>
+                <div class="secformItem">
+                    <label class="form-input-label" for="form-input-secret-token">Token</label>
+                    <b-form-input id="form-input-secret-token" size="sm" v-model="selectedExId"
+                                  disabled></b-form-input>
+                </div>
+                <div class="secFormItem">
+                    <label class="form-input-label" for="form-input-secret-type">Type</label>
+                    <b-form-input id="form-input-secret-type" size="sm" v-model="selectedExType"
+                                  disabled></b-form-input>
+                </div>
+                <div class="secformItem">
+                    <label class="form-input-label" for="form-input-secret-description">Description</label>
+                    <b-form-input id="form-input-secret-description" size="sm" v-model="selectedExDescription"
+                                  disabled></b-form-input>
+                </div>
+                <div class="secformItem">
+                    <label class="form-input-label" for="form-input-secret-owner">Owner</label>
+                    <b-form-input id="form-input-secret-owner" size="sm" v-model="selectedExOwnerId"
+                                  disabled></b-form-input>
+                </div>
+                <div v-if="selectedExType ==='PASSWORD'" class="secformItem">
+                    <label class="form-input-label" for="form-input-secret-password">Password</label>
+                    <Password id="form-input-secret-password" size="sm" v-model="selectedExPassword" disabled>
+                    </Password>
+                </div>
+                <div v-if="selectedExType ==='SSH'" class="secformItem">
+                    <label class="form-input-label" for="form-input-secret-public-key">Public Key</label>
+                    <b-textarea id="form-input-secret-public-key" size="sm" v-model="selectedExPubKey"
+                                disabled></b-textarea>
+                </div>
+                <div v-if="selectedExType ==='SSH'" class="secformItem">
+                    <label class="form-input-label" for="form-input-secret-private-key">Private Key</label>
+                    <b-textarea id="form-input-secret-private-key" size="sm" v-model="selectedExPrivKey"
+                                disabled></b-textarea>
+                </div>
+            </div>
+            <template slot="modal-footer">
+                <b-button size="sm" class="mr-2" @click="$bvModal.hide('view-secret-modal')">
+                    Cancel
+                </b-button>
+                <b-button size="sm" variant="danger" v-on:click="secDeleteButtonPressed"
+                          @click="$bvModal.hide('view-secret-modal')">
+                    Delete
+                </b-button>
+            </template>
+        </b-modal>
+    </div>
+</template>
+
+<script>
+    import config from "@/config";
+    import Password from "../Password";
+
+    export default {
+        name: "Secrets",
+        components: {Password},
+        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,
+                passwordEmptyError: 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: this.selectedDescription,
+                                type: 'SECRET',
+                                owner_id: this.currentUserName
+                            }
+                        }
+                    }
+
+                    await this.$store.dispatch('sharing/createEntity', dataEn)
+
+                } else {
+
+                    if (this.selectedPassword == null || this.selectedPassword == '') {
+                        this.passwordEmptyError = true
+
+                    } 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: 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 goToWorkspace() {
+                await this.$router.push('/workspace')
+            },
+
+        },
+
+
+        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>
+    h2 {
+        font-family: Avenir;
+        font-size: 20px;
+        font-weight: 600;
+        text-align: left;
+        color: #203a43;
+    }
+</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..e1e98d1
--- /dev/null
+++ b/custos-demo-gateway/src/components/workspace/Sharing.vue
@@ -0,0 +1,1006 @@
+<template>
+    <div>
+        <div class="w-100">
+            <h2>Sharing</h2>
+        </div>
+        <b-container class="text-left">
+            <div v-if="isAdminUser" class="w-100">
+                Do you want to evaluate if a specific user has a specific permission to a specific entity ?
+                <b-button variant="link" v-on:click="checkPermissions">Evaluate Permissions</b-button>
+            </div>
+            <div>
+                <div class="mt-5">
+                    <strong>Permission Types</strong>
+                    <b-button v-if="isAdminUser" variant="link" v-on:click="onNewPrTyAdd">
+                        + Add Permission Type
+                    </b-button>
+                    <div>
+                        <b-alert v-model="this.inputErrorPR" variant="danger" dismissible
+                                 @dismissed="callDismissed">
+                            Input validation error
+                        </b-alert>
+                    </div>
+                    <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 small striped hover responsive :items="permissionTypes" :fields="fields" selectable
+                             ref="selectableTable" select-mode="single" @row-selected="onPrTySelected">
+                        <template v-slot:head(id)>Permission Type ID</template>
+                    </b-table>
+                </div>
+                <div class="mt-5">
+                    <strong>Entity Types</strong>
+                    <b-button v-if="isAdminUser" variant="link" v-on:click="onNewEnTyAdd">
+                        + Add Entity Type
+                    </b-button>
+                    <div v-if="this.entityTypesLoading" class="d-flex justify-content-center mb-3">
+                        <b-spinner variant="primary" label="Text Centered"></b-spinner>
+                    </div>
+                    <div>
+                        <b-alert v-model="this.inputErrorEnTy" variant="danger" dismissible
+                                 @dismissed="callDismissed">
+                            Input validation error
+                        </b-alert>
+                    </div>
+                    <b-table small striped hover responsive :items="entityTypes" :fields="fields" selectable
+                             ref="selectableTable" select-mode="single" @row-selected="onEnTySelected" caption-top>
+                        <template v-slot:head(id)>Entity Type ID</template>
+                    </b-table>
+                </div>
+                <div class="mt-5">
+                    <strong>Entities</strong>
+                    <b-button variant="link" v-on:click="onNewEnAdd">
+                        + Add Entity
+                    </b-button>
+                    <div v-if="this.entitiesLoading" class="d-flex justify-content-center mb-3">
+                        <b-spinner variant="primary" label="Text Centered"></b-spinner>
+                    </div>
+                    <div>
+                        <b-alert v-model="this.inputErrorEn" variant="danger" dismissible
+                                 @dismissed="callDismissed">
+                            Input validation error
+                        </b-alert>
+                    </div>
+                    <b-table small striped hover responsive :items="entities" :fields="entityFields" selectable
+                             ref="selectableTable" select-mode="single" @row-selected="onEntitySelected">
+                        <template v-slot:head(id)>Entity ID</template>
+                        <template v-slot:head(type)>Entity Type ID</template>
+                    </b-table>
+                </div>
+                <div class="mt-5">
+                    <strong>Entity Sharing</strong>
+                    <b-button variant="link" v-on:click="onSharingAdd">
+                        + Add Entity Sharing
+                    </b-button>
+                    <div v-if="this.sharingsLoading" class="d-flex justify-content-center mb-3">
+                        <b-spinner variant="primary" label="Text Centered"></b-spinner>
+                    </div>
+                    <div>
+                        <b-alert v-model="this.sharingError" variant="danger" dismissible
+                                 @dismissed="callDismissed">
+                            Input validation error
+                        </b-alert>
+                    </div>
+                    <b-table small striped hover responsive :items="sharings" :fields="sharingFields" selectable
+                             ref="selectableTable"
+                             select-mode="single"
+                             @row-selected="onSharingSelected">
+                        <template v-slot:head(entity_id)>Entity ID</template>
+                        <template v-slot:head(permission_type_id)>Permission Type ID</template>
+                        <template v-slot:head(owner_id)>Group / User ID</template>
+                        <template v-slot:head(type)>Group / User</template>
+                    </b-table>
+                </div>
+
+
+            </div>
+        </b-container>
+        <b-modal ref="prtypemodel" id="add-permission-type-modal" scrollable title="Add Permission Type">
+            <div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-id">ID</label>
+                    <b-form-input id="form-input-id" size="sm" v-model="prId"></b-form-input>
+                </div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-name">Name</label>
+                    <b-form-input id="form-input-name" size="sm" v-model="prName"></b-form-input>
+                </div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-description">Description</label>
+                    <b-form-input id="form-input-description" size="sm" v-model="prDesc"></b-form-input>
+                </div>
+            </div>
+            <template v-slot:modal-footer>
+                <b-button size="sm" class="mr-2" @click="$bvModal.hide('add-permission-type-modal')">
+                    Cancel
+                </b-button>
+                <b-button size="sm" variant="primary" v-on:click="addPrType"
+                          :disabled="!prId || !prName"
+                          @click="$bvModal.hide('add-permission-type-modal')">
+                    Add
+                </b-button>
+            </template>
+        </b-modal>
+        <b-modal ref="entypemodel" id="add-entity-type-modal" scrollable title="Add Entity Type">
+            <div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-id">ID</label>
+                    <b-form-input id="form-input-id" size="sm" v-model="enTyId"></b-form-input>
+                </div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-name">Name</label>
+                    <b-form-input id="form-input-name" size="sm" v-model="enTyName"></b-form-input>
+                </div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-description">Description</label>
+                    <b-form-input id="form-input-description" size="sm" v-model="enTyDesc"></b-form-input>
+                </div>
+            </div>
+            <template v-slot:modal-footer>
+                <b-button size="sm" class="mr-2" @click="$bvModal.hide('add-entity-type-modal')">
+                    Cancel
+                </b-button>
+                <b-button size="sm" variant="primary" v-on:click="addNewEnType"
+                          :disabled="!enTyId || !enTyName"
+                          @click="$bvModal.hide('add-entity-type-modal')">
+                    Add
+                </b-button>
+            </template>
+        </b-modal>
+        <b-modal ref="enModel" id="add-entity-modal" scrollable title="Add  Entity " ok-title="Add" @ok="addNewEntity">
+            <div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-name">Name</label>
+                    <b-form-input id="form-input-name" size="sm" v-model="enName"></b-form-input>
+                </div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-type">Type</label>
+                    <b-form-select id="form-input-type" size="sm" v-model="selectedEntityType">
+                        <option v-for="(selectOption, indexOpt) in entityTypes"
+                                :key="indexOpt"
+                                :value="selectOption"
+                        >
+                            {{ selectOption.id }}
+                        </option>
+                    </b-form-select>
+                </div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-description">Description</label>
+                    <b-form-input id="form-input-description" size="sm" v-model="enDesc"></b-form-input>
+                </div>
+            </div>
+            <template v-slot:modal-footer>
+                <b-button size="sm" class="mr-2" @click="$bvModal.hide('add-entity-modal')">
+                    Cancel
+                </b-button>
+                <b-button size="sm" variant="primary" v-on:click="addNewEntity"
+                          :disabled="!enName || !selectedEntityType"
+                          @click="$bvModal.hide('add-entity-modal')">
+                    Add
+                </b-button>
+            </template>
+        </b-modal>
+        <b-modal ref="selectedPrTyModel" id="view-permission-type-modal" scrollable title="Permission Type">
+            <div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-id">ID</label>
+                    <b-form-input id="form-input-id" size="sm" v-model="selectedPrTyId" disabled></b-form-input>
+                </div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-name">Name</label>
+                    <b-form-input id="form-input-name" size="sm" v-model="selectedPrTyName" disabled></b-form-input>
+                </div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-description">Description</label>
+                    <b-form-input id="form-input-description" size="sm" v-model="selectedPrTyDesc"
+                                  disabled></b-form-input>
+                </div>
+            </div>
+            <template v-slot:modal-footer>
+                <b-button size="sm" class="mr-2" @click="$bvModal.hide('view-permission-type-modal')">
+                    Cancel
+                </b-button>
+                <b-button size="sm" variant="danger" v-on:click="deletePRType"
+                          @click="$bvModal.hide('view-permission-type-modal')">
+                    Delete
+                </b-button>
+            </template>
+        </b-modal>
+        <b-modal ref="selectedEnTyModel" id="view-entity-type-modal" scrollable title="Entity Type " ok-title="Delete"
+                 @ok="deleteEnType">
+            <div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-id">ID</label>
+                    <b-form-input id="form-input-id" size="sm" v-model="selectedEnTyId" disabled></b-form-input>
+                </div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-name">Name</label>
+                    <b-form-input id="form-input-name" size="sm" v-model="selectedEnTyName" disabled></b-form-input>
+                </div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-description">Description</label>
+                    <b-form-input id="form-input-description" size="sm" v-model="selectedEnTyDesc"
+                                  disabled></b-form-input>
+                </div>
+            </div>
+            <template v-slot:modal-footer>
+                <b-button size="sm" class="mr-2" @click="$bvModal.hide('view-entity-type-modal')">
+                    Cancel
+                </b-button>
+                <b-button size="sm" variant="danger" v-on:click="deleteEnType"
+                          @click="$bvModal.hide('view-entity-type-modal')">
+                    Delete
+                </b-button>
+            </template>
+        </b-modal>
+        <b-modal ref="selectedEnModel" id="view-entity-modal" scrollable title="Entity">
+            <div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-id">ID</label>
+                    <b-form-input id="form-input-id" size="sm" v-model="selectedEnId" disabled></b-form-input>
+                </div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-name">Name</label>
+                    <b-form-input id="form-input-name" size="sm" v-model="selectedEnName" disabled></b-form-input>
+                </div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-type">Type</label>
+                    <b-form-input id="form-input-type" size="sm" v-model="selectedEntityType" disabled></b-form-input>
+                </div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-description">Description</label>
+                    <b-form-input id="form-input-description" size="sm" v-model="selectedEnDesc"
+                                  disabled></b-form-input>
+                </div>
+            </div>
+            <template v-slot:modal-footer>
+                <b-button size="sm" class="mr-2" @click="$bvModal.hide('view-entity-modal')">
+                    Cancel
+                </b-button>
+                <b-button size="sm" variant="danger" v-on:click="deleteEntity"
+                          @click="$bvModal.hide('view-entity-modal')">
+                    Delete
+                </b-button>
+            </template>
+        </b-modal>
+        <b-modal ref="selectedShraingModel" id="view-entity-sharing-modal" scrollable title="Sharing " ok-title="Delete"
+                 @ok="removeSharing">
+            <div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-entity-id">Entity ID</label>
+                    <b-form-input id="form-input-entity-id" size="sm" v-model="selectedShEnId" disabled></b-form-input>
+                </div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-permission-type-id">Permission Type ID</label>
+                    <b-form-input id="form-input-permission-type-id" size="sm" v-model="selectedShPrId"
+                                  disabled></b-form-input>
+                </div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-owner-id">Group / User ID</label>
+                    <b-form-input id="form-input-owner-id" size="sm" v-model="selectedShOwId" disabled></b-form-input>
+                </div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-type">Group / User</label>
+                    <b-form-input id="form-input-type" size="sm" v-model="selectedShOwType" disabled></b-form-input>
+                </div>
+            </div>
+            <template v-slot:modal-footer>
+                <b-button size="sm" class="mr-2" @click="$bvModal.hide('view-entity-sharing-modal')">
+                    Cancel
+                </b-button>
+                <b-button size="sm" variant="danger" v-on:click="removeSharing"
+                          @click="$bvModal.hide('view-entity-sharing-modal')">
+                    Delete
+                </b-button>
+            </template>
+        </b-modal>
+        <b-modal ref="sharingModel" id="add-entity-sharing-modal" scrollable title="Share Entities">
+            <div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-entity-id">Entity ID</label>
+                    <b-form-select id="form-input-entity-id" size="sm" v-model="defaultEntityId">
+                        <option v-for="(selectOption, indexOpt) in entities"
+                                :key="indexOpt"
+                                :value="selectOption"
+                        >
+                            {{ selectOption.id }}
+                        </option>
+                    </b-form-select>
+                </div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-permission-type">Permission Type</label>
+                    <b-form-select id="form-input-permission-type" size="sm" v-model="defaultPermissionType">
+                        <option v-for="(selectOption, indexOpt) in permissionTypes"
+                                :key="indexOpt"
+                                :value="selectOption"
+                        >
+                            {{ selectOption.id }}
+                        </option>
+                    </b-form-select>
+                </div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-sharing-type">Sharing Type</label>
+                    <b-form-select id="form-input-sharing-type" size="sm" 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="p-2">
+                    <label class="form-input-label" for="form-input-user-id">User ID</label>
+                    <b-form-select id="form-input-user-id" size="sm" 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="p-2">
+                    <label class="form-input-label" for="form-input-group-name">Group Name</label>
+                    <b-form-select id="form-input-group-name" size="sm" v-model="defaultGroupName">
+                        <option v-for="(selectOption, indexOpt) in groups"
+                                :key="indexOpt"
+                                :value="selectOption"
+                        >
+                            {{ selectOption.name }}
+                        </option>
+                    </b-form-select>
+                </div>
+            </div>
+            <template v-slot:modal-footer>
+                <b-button size="sm" class="mr-2" @click="$bvModal.hide('add-entity-sharing-modal')">
+                    Cancel
+                </b-button>
+                <b-button size="sm" variant="primary" v-on:click="addNewSharing"
+                          @click="$bvModal.hide('add-entity-sharing-modal')">
+                    Add
+                </b-button>
+            </template>
+        </b-modal>
+        <b-modal ref="permissionChecker" id="evaluate-permission-modal" scrollable title="Check Permissions">
+            <div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-entity-id">Entity ID</label>
+                    <b-form-select id="form-input-entity-id" size="sm" v-model="defaultEntityId">
+                        <option v-for="(selectOption, indexOpt) in entities"
+                                :key="indexOpt"
+                                :value="selectOption"
+                        >
+                            {{ selectOption.id }}
+                        </option>
+                    </b-form-select>
+                </div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-user">User</label>
+                    <b-form-select id="form-input-user" size="sm" v-model="defaultOwner">
+                        <option v-for="(selectOption, indexOpt) in users"
+                                :key="indexOpt"
+                                :value="selectOption"
+                        >
+                            {{ selectOption }}
+                        </option>
+                    </b-form-select>
+                </div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-permission">Permission</label>
+                    <b-form-select id="form-input-permission" size="sm" v-model="defaultPermissionType">
+                        <option v-for="(selectOption, indexOpt) in permissionTypes"
+                                :key="indexOpt"
+                                :value="selectOption"
+                        >
+                            {{ selectOption.id }}
+                        </option>
+                    </b-form-select>
+                </div>
+            </div>
+            <template v-slot:modal-footer>
+                <b-button size="sm" class="mr-2" @click="$bvModal.hide('evaluate-permission-modal')">
+                    Cancel
+                </b-button>
+                <b-button size="sm" variant="primary" v-on:click="evaluatePermission"
+                          :disabled="!defaultEntityId || !defaultOwner || !defaultPermissionType"
+                          @click="$bvModal.hide('evaluate-permission-modal')">
+                    Evaluate
+                </b-button>
+            </template>
+        </b-modal>
+        <b-modal ref="evalutionResultPopup" id="evaluate-permission-results-modal">
+            <div v-if="this.evaluating">
+                <b-spinner small type="grow"></b-spinner>
+                Evaluating...
+            </div>
+            <div v-if="!this.evaluating">
+                <div v-if="evalutionResult" class="groupform">
+                    <p>User has permission.</p>
+                </div>
+                <div v-if="!evalutionResult" class="groupform">
+                    <p>User does not have permission.</p>
+                </div>
+            </div>
+            <template v-slot:modal-footer>
+                <b-button size="sm" class="mr-2" @click="$bvModal.hide('evaluate-permission-results-modal')">
+                    Cancel
+                </b-button>
+                <b-button variant="primary" size="sm" class="mr-2"
+                          @click="$bvModal.hide('evaluate-permission-results-modal')">
+                    Ok
+                </b-button>
+            </template>
+        </b-modal>
+    </div>
+</template>
+
+<script>
+    import config from "@/config";
+
+    export default {
+        name: "Sharing",
+        data: function () {
+            return {
+                isAdminUser: false,
+                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,
+                defaultGroupName: null,
+                inputErrorPR: false,
+                inputErrorEnTy: false,
+                inputErrorEn: false,
+                sharingError: false,
+
+
+            }
+        },
+
+        methods: {
+            async addPrType() {
+                this.permissionTypesLoading = true
+                if (this.prId == null || this.prName == null || this.prId == '' || this.prName == '') {
+                    this.inputErrorPR = true
+                    this.permissionTypesLoading = false
+                }
+                if (!this.inputErrorPR) {
+                    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
+                if (this.enTyId == null || this.enTyName == null || this.enTyName == '' || this.enTyId == '') {
+                    this.inputErrorEnTy = true
+                    this.entityTypesLoading = false
+                }
+                if (!this.inputErrorEnTy) {
+                    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
+                if (this.enName == null || this.enName == '') {
+                    this.inputErrorEn = true
+                    this.entitiesLoading = false
+                }
+
+                if (!this.inputErrorEn) {
+                    this.enId = this.enName + "_" + this.makeid(15)
+                    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.sharings = await this.loadSharings()
+                }
+                this.enId = null
+                this.enName = null
+                this.enDesc = null
+                this.selectedEntityType = null
+
+            },
+
+            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
+
+                if (this.defaultEntityId == null || this.defaultPermissionType == null
+                    || (this.defaultGroupName == null && this.defaultOwner == null)) {
+                    this.sharingError = true
+                    this.sharingsLoading = false
+
+                }
+
+                if (!this.sharingError) {
+                    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 {
+                        console.log(this.defaultGroupName)
+                        data.body.owner_id = [this.defaultGroupName.id]
+                        console.log(data)
+                        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,
+                    username: this.currentUserName
+                }
+                if (this.isAdminUser) {
+                    return await this.$store.dispatch('group/loadAllGroups', data)
+                } else {
+                    let grs = []
+                    grs = await this.$store.dispatch('group/getAllGroupsOfUser', data)
+                    let groups = []
+                    grs.forEach(gr => {
+                        gr.ownerId = gr.owner_id
+                        groups.push(gr)
+                    })
+                    return groups
+                }
+
+            },
+
+
+            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 goToWorkspace() {
+                await this.$router.push('/workspace')
+            },
+
+            makeid(length) {
+                let result = '';
+                let characters = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789';
+                let charactersLength = characters.length;
+                for (var i = 0; i < length; i++) {
+                    result += characters.charAt(Math.floor(Math.random() * charactersLength));
+                }
+                return result;
+            },
+
+            async callDismissed() {
+                this.inputErrorEn = false
+                this.inputErrorEnTy = false
+                this.inputErrorEn = false
+                this.sharingError = false
+            }
+
+        },
+
+        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.users = await this.loadUsers()
+            this.groups = await this.loadGroups()
+
+            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()
+        }
+    }
+
+
+</script>
+
+<style scoped>
+    h2 {
+        font-family: Avenir;
+        font-size: 20px;
+        font-weight: 600;
+        text-align: left;
+        color: #203a43;
+    }
+</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..33c4426
--- /dev/null
+++ b/custos-demo-gateway/src/components/workspace/Users.vue
@@ -0,0 +1,705 @@
+<template>
+    <div>
+        <div class="w-100">
+            <h2>Manage Users</h2>
+        </div>
+        <b-container class="w-100">
+            <form class="userSearchBar mb-5" v-on:submit.prevent="searchResult">
+                <b-input-group>
+                    <b-form-input size="sm" v-model="searchUsername" placeholder="Search"></b-form-input>
+                    <b-input-group-append>
+                        <b-button size="sm" type="submit">
+                            <b-icon icon="search"></b-icon>
+                        </b-button>
+                    </b-input-group-append>
+                </b-input-group>
+            </form>
+            <div v-if="this.userloading" class="d-flex justify-content-center w-100">
+                <b-spinner variant="primary" label="Text Centered"></b-spinner>
+            </div>
+            <b-table striped hover responsive :items="items" :fields="fields" selectable small
+                     ref="selectableTable"
+                     select-mode="single"
+                     :per-page="perPage"
+                     :current-page="currentPage"
+                     @row-selected="onRowSelected">
+                <template v-slot:cell(status)="data">
+                    <b-badge v-if="data.value == 'ACTIVE'" variant="success">Active</b-badge>
+                    <b-badge v-else-if="data.value == 'DEACTIVE'" variant="danger">Inactive</b-badge>
+                    <b-badge v-else-if="data.value == 'PENDING'" variant="warning">Pending</b-badge>
+                </template>
+            </b-table>
+            <div>
+                <b-pagination
+                        size="sm"
+                        class="float-right"
+                        v-model="currentPage"
+                        :total-rows="rows"
+                        :per-page="perPage"
+                        aria-controls="my-table"
+                ></b-pagination>
+            </div>
+        </b-container>
+        <b-modal ref="usermodel" id="user-modal" scrollable title="User Profile">
+            <div class="user-profile">
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-username">Username</label>
+                    <b-form-input id="form-input-username" size="sm" v-model="selectedUsername"
+                                  disabled></b-form-input>
+                </div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-first-name">First Name</label>
+                    <b-form-input id="form-input-first-name" size="sm" v-model="selectedFirstName"></b-form-input>
+                </div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-last-name">Last Name</label>
+                    <b-form-input id="form-input-last-name" size="sm" v-model="selectedLastName"></b-form-input>
+                </div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-email">Email</label>
+                    <b-form-input id="form-input-email" size="sm" v-model="selectedEmail"></b-form-input>
+                </div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-email">Status</label>
+                    <b-form-select id="form-input-status" size="sm" 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="w-100 mt-5">
+                    <strong>Attributes</strong>
+                    <b-button variant="link" v-on:click="addAttribute">+ Add Attributes</b-button>
+                    <div class="w-100">
+                        <small v-if="selectedAttributes.length === 0">There are no attributes created. </small>
+                    </div>
+                    <b-table small striped hover responsive :items="selectedAttributes" class="mt-2" selectable
+                             select-mode="single"
+                             @row-selected="onAtrSelected">
+                    </b-table>
+                </div>
+
+                <div class="w-100 mt-5">
+                    <strong>Roles</strong>
+                    <b-button variant="link" v-on:click="addRole" :disabled="this.isAdminUser==false">+ Add Role
+                    </b-button>
+                    <div class="w-100">
+                        <small v-if="selectedRoles.length === 0">There are no roles assigned. </small>
+                    </div>
+                    <b-table small striped hover responsive :items="selectedRoles" class="mt-2" selectable
+                             select-mode="single"
+                             @row-selected="onRoleSelected">
+                        <template v-slot:head(name)>Role</template>
+                    </b-table>
+                </div>
+            </div>
+            <template v-slot:modal-footer>
+                <b-button size="sm" class="mr-2" @click="$bvModal.hide('user-modal')">
+                    Cancel
+                </b-button>
+                <b-button size="sm" variant="primary" v-on:click="updateUserProfile"
+                          @click="$bvModal.hide('user-modal')">
+                    Save
+                </b-button>
+            </template>
+        </b-modal>
+        <b-modal ref="atrModel" id="user-attribute-modal" scrollable title="Add Attribute">
+            <div class="userform">
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-attribute-key">Key</label>
+                    <b-form-input id="form-input-attribute-key" size="sm" v-model="newKey"></b-form-input>
+                </div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-attribute-value">Value</label>
+                    <b-form-input id="form-input-attribute-value" size="sm" v-model="newValue"></b-form-input>
+                </div>
+            </div>
+            <template v-slot:modal-footer>
+                <b-button size="sm" class="mr-2" @click="$bvModal.hide('user-attribute-modal')">
+                    Cancel
+                </b-button>
+                <b-button :disabled="!newKey || !newValue" size="sm" variant="primary" v-on:click="addAtrOkPressed"
+                          @click="$bvModal.hide('user-attribute-modal')">
+                    Add
+                </b-button>
+            </template>
+        </b-modal>
+        <b-modal ref="atrModelSelected" id="user-attribute-view-modal" scrollable title="Attribute">
+            <div class="userform">
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-attribute-key">Key</label>
+                    <b-form-input id="form-input-attribute-key" size="sm" disabled
+                                  v-model="selectedKey"></b-form-input>
+                </div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-attribute-value">Value</label>
+                    <b-form-input id="form-input-attribute-value" size="sm" disabled
+                                  v-model="selectedValue"></b-form-input>
+                </div>
+            </div>
+            <template v-slot:modal-footer>
+                <b-button size="sm" class="mr-2" @click="$bvModal.hide('user-attribute-view-modal')">
+                    Cancel
+                </b-button>
+                <b-button size="sm" variant="danger" v-on:click="addAtrDeletePressed"
+                          @click="$bvModal.hide('user-attribute-view-modal')">
+                    Delete
+                </b-button>
+            </template>
+        </b-modal>
+        <b-modal ref="roleModel" id="user-role-modal" scrollable title="Add Role">
+            <div class="userform">
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-role-scope">Scope</label>
+                    <b-form-select id="form-input-role-scope" size="sm" v-model="selectedScope">
+                        <option v-for="(selectOption, indexOpt) in scopes"
+                                :key="indexOpt"
+                                :value="selectOption"
+                        >
+                            {{ selectOption }}
+                        </option>
+                    </b-form-select>
+                </div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-role">Role</label>
+                    <b-form-select v-if="selectedScope==='TENANT'" id="form-input-role" size="sm"
+                                   v-model="selectedRole">
+                        <option v-for="(selectOption, indexOpt) in tenantroles"
+                                :key="indexOpt"
+                                :value="selectOption"
+                        >
+                            {{ selectOption }}
+                        </option>
+                    </b-form-select>
+                    <b-form-select v-if="selectedScope==='CLIENT'" id="form-input-role" size="sm"
+                                   v-model="selectedRole">
+                        <option v-for="(selectOption, indexOpt) in clientroles"
+                                :key="indexOpt"
+                                :value="selectOption"
+                        >
+                            {{ selectOption }}
+                        </option>
+                    </b-form-select>
+                </div>
+            </div>
+            <template v-slot:modal-footer>
+                <b-button size="sm" class="mr-2" @click="$bvModal.hide('user-role-modal')">
+                    Cancel
+                </b-button>
+                <b-button :disabled="!selectedScope || !selectedRole" size="sm" variant="primary"
+                          v-on:click="addRoleOkPressed" @click="$bvModal.hide('user-role-modal')">
+                    Add
+                </b-button>
+            </template>
+        </b-modal>
+        <b-modal ref="roleModelSelected" id="user-role-view-modal" scrollable title="Role">
+            <div class="userform">
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-role">Role</label>
+                    <b-form-input id="form-input-role" size="sm" disabled v-model="rowSelectedRole"></b-form-input>
+                </div>
+                <div class="p-2">
+                    <label class="form-input-label" for="form-input-role-scope">Scope</label>
+                    <b-form-input id="form-input-role-scope" size="sm" disabled
+                                  v-model="rowSelectedScope"></b-form-input>
+                </div>
+            </div>
+            <template v-slot:modal-footer>
+                <b-button size="sm" class="mr-2" @click="$bvModal.hide('user-role-view-modal')">
+                    Cancel
+                </b-button>
+                <b-button size="sm" variant="danger" v-on:click="deleteRoleOkPressed"
+                          @click="$bvModal.hide('user-role-view-modal')">
+                    Delete
+                </b-button>
+            </template>
+        </b-modal>
+    </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', 'DEACTIVE'],
+                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.selectedUsername != 'admin') {
+                        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: 5, 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
+                        if (this.rowSelectedRole !== 'offline_access' && this.rowSelectedRole !== 'uma_authorization') {
+                            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 deleteAttribute(key, value) {
+                this.operationCompleted = false
+                let accessToken = await this.$store.getters['identity/getAccessToken']
+                let bd = {
+                    user_token: accessToken, body: {
+                        attributes: [{
+                            key: key,
+                            values: [value]
+                        }],
+                        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 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)
+            },
+
+            async goToWorkspace() {
+                await this.$router.push('/workspace')
+            },
+
+            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 == 'ACTIVE') ? obj.state : 'DEACTIVE',
+                            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>
+
+    h2 {
+        font-family: Avenir;
+        font-size: 20px;
+        font-weight: 600;
+        text-align: left;
+        color: #203a43;
+    }
+
+    .userSearchBar {
+        border-radius: 5px;
+        border: solid 1px #afafae;
+        width: 300px;
+        float: right;
+    }
+
+    .userSearchBar .form-control {
+        border: none;
+    }
+
+    .userSearchBar input.form-control:focus {
+        outline: none;
+        box-shadow: none;
+    }
+
+    .userSearchBar button {
+        background: none;
+        color: black;
+        border: none;
+    }
+</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..13d5eb1
--- /dev/null
+++ b/custos-demo-gateway/src/components/workspace/Workspace.vue
@@ -0,0 +1,104 @@
+<template>
+    <b-container>
+        <b-row class="mt-5 vh-100 text-center" align-h="center">
+            <b-card-group columns style="max-width: 800px;">
+                <b-card class="menu-card" v-if="this.isAdmin" :img-src="require('../../assets/users.png')" img-top>
+                    <b-button href="#"
+                              v-on:click="loadRoute($event, '/workspace/users')">
+                        Manage Users
+                    </b-button>
+                </b-card>
+                <b-card class="menu-card" :img-src="require('../../assets/groups_web.png')" img-top>
+                    <b-button href="#"
+                              v-on:click="loadRoute($event, '/workspace/groups')">
+                        Manage Groups
+                    </b-button>
+                </b-card>
+                <b-card class="menu-card" :img-src="require('../../assets/credentials.png')" img-top>
+                    <b-button href="#"
+                              v-on:click="loadRoute($event, '/workspace/secrets')">
+                        Manage Secrets
+                    </b-button>
+                </b-card>
+                <b-card class="menu-card" :img-src="require('../../assets/sharings.png')" img-top>
+                    <b-button href="#"
+                              v-on:click="loadRoute($event, '/workspace/sharings')">
+                        Sharing
+                    </b-button>
+                </b-card>
+                <b-card class="menu-card" v-if="isAdmin" :img-src="require('../../assets/bots.png')" img-top>
+                    <b-button href="#" :disabled="!isAdmin"
+                              v-on:click="loadRoute($event, '/workspace/agents')">
+                        Service Accounts
+                    </b-button>
+                </b-card>
+                <b-card class="menu-card" v-if="isAdmin" :img-src="require('../../assets/dblogs.png')" img-top>
+                    <b-button href="#" :disabled="!isAdmin"
+                              v-on:click="loadRoute($event, '/workspace/logs')">
+                        Logs
+                    </b-button>
+                </b-card>
+            </b-card-group>
+        </b-row>
+    </b-container>
+</template>
+
+<script>
+    export default {
+        name: "Workspace",
+        data: function () {
+            return {
+                custosId: null,
+                custosSec: null,
+                isAdmin: false,
+                user: null,
+                currentUserName: null
+            }
+        },
+        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.isAdmin = await this.$store.dispatch('identity/isLoggedUserHasAdminAccess')
+        }
+    }
+</script>
+
+<style scoped>
+    .menu-card {
+        max-width: 250px;
+        min-width: 200px;
+        height: 197px;
+    }
+
+    .menu-card img {
+        width: 238px;
+        height: 120px;
+    }
+
+    .menu-card .card-body a.btn {
+        width: 100%;
+        border-radius: 11px;
+        border: solid 1px #afafae;
+        background-color: #203a43;
+        font-family: Avenir;
+        font-size: 16px;
+        font-weight: 500;
+        color: #ffffff;
+    }
+</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..18669d8
--- /dev/null
+++ b/custos-demo-gateway/src/main.js
@@ -0,0 +1,24 @@
+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 Vuelidate from 'vuelidate'
+
+import 'bootstrap/dist/css/bootstrap.css';
+import 'bootstrap-vue/dist/bootstrap-vue.css';
+
+Vue.use(BootstrapVue)
+Vue.use(Vuelidate)
+// 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..8f9d072
--- /dev/null
+++ b/custos-demo-gateway/src/router.js
@@ -0,0 +1,178 @@
+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/profile",
+            name: "profile",
+            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/Profile")
+        },
+        {
+            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..4f7f6e0
--- /dev/null
+++ b/custos-demo-gateway/src/service/group_management.js
@@ -0,0 +1,191 @@
+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})
+
+    },
+
+
+    getAllGroupsOfUser(params) {
+        let authHeader = {'Authorization': 'Bearer ' + btoa(params.client_id + ':' + params.client_sec)}
+        let endpoint = groupMgtEndpoint + "/user/group/memberships"
+        let parm = {
+            'profile.username': params.username
+        }
+        return api().get(endpoint, {params: parm, headers: authHeader})
+
+    },
+
+
+    getAllParentGroupsOfGroup(params) {
+        let authHeader = {'Authorization': 'Bearer ' + btoa(params.client_id + ':' + params.client_sec)}
+        let endpoint = groupMgtEndpoint + "/groups/memberships"
+        let parm = {
+            'group.id': params.groupId
+        }
+        return api().get(endpoint, {params: parm, headers: authHeader})
+
+    },
+
+    hasAccess(params) {
+        let authHeader = {'Authorization': 'Bearer ' + btoa(params.client_id + ':' + params.client_sec)}
+        let endpoint = groupMgtEndpoint + "/user/group/access"
+        let parm = {
+            'group_id': params.groupId,
+            'username':params.username,
+            'type': params.type
+        }
+        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..7735523
--- /dev/null
+++ b/custos-demo-gateway/src/store/modules/group.store.js
@@ -0,0 +1,178 @@
+/*Group.store.js*/
+
+import group_management from "@/service/group_management";
+
+
+const getDefaultState = () => {
+    return {
+        groups: [],
+    }
+}
+
+const state = getDefaultState()
+
+
+const actions = {
+
+    async createGroup({commit}, data) {
+        try {
+            let resp = await group_management.createGroup(data)
+            if (resp.data != null) {
+                let group = resp.data.groups[0];
+                commit('SET_GROUP', group)
+                return group
+            }
+        } catch (exception) {
+            return false
+        }
+    },
+
+    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 getAllGroupsOfUser({commit}, data) {
+        let response = await group_management.getAllGroupsOfUser(data)
+        return response.data.groups
+    },
+
+
+    // eslint-disable-next-line no-unused-vars
+    async getAllParentGroups({commit}, data) {
+        let response = await group_management.getAllParentGroupsOfGroup(data)
+        return response.data.groups
+    },
+
+    // eslint-disable-next-line no-unused-vars
+    async reset({commit}, data) {
+        commit('RESET')
+        return true
+    },
+
+    // eslint-disable-next-line no-unused-vars
+    async hasAccess({commit}, data) {
+        let response = await group_management.hasAccess(data)
+        return response.data
+    }
+
+}
+
+
+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"
+      randombytes "^2.0.0"
+  
+  dir-glob@^2.0.0, dir-glob@^2.2.2:
+    version "2.2.2"
+    resolved "https://registry.yarnpkg.com/dir-glob/-/dir-glob-2.2.2.tgz#fa09f0694153c8918b18ba0deafae94769fc50c4"
+    integrity sha512-f9LBi5QWzIW3I6e//uxZoLBlUt9kcp66qo0sSCxL6YZKc75R1c4MFCoe/LaZiBGmgujvQdxc5Bn3QhfyvK5Hsw==
+    dependencies:
+      path-type "^3.0.0"
+  
+  dns-equal@^1.0.0:
+    version "1.0.0"
+    resolved "https://registry.yarnpkg.com/dns-equal/-/dns-equal-1.0.0.tgz#b39e7f1da6eb0a75ba9c17324b34753c47e0654d"
+    integrity sha1-s55/HabrCnW6nBcySzR1PEfgZU0=
+  
+  dns-packet@^1.3.1:
+    version "1.3.1"
+    resolved "https://registry.yarnpkg.com/dns-packet/-/dns-packet-1.3.1.tgz#12aa426981075be500b910eedcd0b47dd7deda5a"
+    integrity sha512-0UxfQkMhYAUaZI+xrNZOz/as5KgDU0M/fQ9b6SpkyLbk3GEswDi6PADJVaYJradtRVsRIlF1zLyOodbcTCDzUg==
+    dependencies:
+      ip "^1.1.0"
+      safe-buffer "^5.0.1"
+  
+  dns-txt@^2.0.2:
+    version "2.0.2"
+    resolved "https://registry.yarnpkg.com/dns-txt/-/dns-txt-2.0.2.tgz#b91d806f5d27188e4ab3e7d107d881a1cc4642b6"
+    integrity sha1-uR2Ab10nGI5Ks+fRB9iBocxGQrY=
+    dependencies:
+      buffer-indexof "^1.0.0"
+  
+  doctrine@^3.0.0:
+    version "3.0.0"
+    resolved "https://registry.yarnpkg.com/doctrine/-/doctrine-3.0.0.tgz#addebead72a6574db783639dc87a121773973961"
+    integrity sha512-yS+Q5i3hBf7GBkd4KG8a7eBNNWNGLTaEwwYWUijIYM7zrlYDM0BFXHjjPWlWZ1Rg7UaddZeIDmi9jF3HmqiQ2w==
+    dependencies:
+      esutils "^2.0.2"
+  
+  dom-converter@^0.2:
+    version "0.2.0"
+    resolved "https://registry.yarnpkg.com/dom-converter/-/dom-converter-0.2.0.tgz#6721a9daee2e293682955b6afe416771627bb768"
+    integrity sha512-gd3ypIPfOMr9h5jIKq8E3sHOTCjeirnl0WK5ZdS1AW0Odt0b1PaWaHdJ4Qk4klv+YB9aJBS7mESXjFoDQPu6DA==
+    dependencies:
+      utila "~0.4"
+  
+  dom-serializer@0:
+    version "0.2.2"
+    resolved "https://registry.yarnpkg.com/dom-serializer/-/dom-serializer-0.2.2.tgz#1afb81f533717175d478655debc5e332d9f9bb51"
+    integrity sha512-2/xPb3ORsQ42nHYiSunXkDjPLBaEj/xTwUO4B7XCZQTRk7EBtTOPaygh10YAAh2OI1Qrp6NWfpAhzswj0ydt9g==
+    dependencies:
+      domelementtype "^2.0.1"
+      entities "^2.0.0"
+  
+  domain-browser@^1.1.1:
+    version "1.2.0"
+    resolved "https://registry.yarnpkg.com/domain-browser/-/domain-browser-1.2.0.tgz#3d31f50191a6749dd1375a7f522e823d42e54eda"
+    integrity sha512-jnjyiM6eRyZl2H+W8Q/zLMA481hzi0eszAaBUzIVnmYVDBbnLxVNnfu1HgEBvCbL+71FrxMl3E6lpKH7Ge3OXA==
+  
+  domelementtype@1, domelementtype@^1.3.1:
+    version "1.3.1"
+    resolved "https://registry.yarnpkg.com/domelementtype/-/domelementtype-1.3.1.tgz#d048c44b37b0d10a7f2a3d5fee3f4333d790481f"
+    integrity sha512-BSKB+TSpMpFI/HOxCNr1O8aMOTZ8hT3pM3GQ0w/mWRmkhEDSFJkkyzz4XQsBV44BChwGkrDfMyjVD0eA2aFV3w==
+  
+  domelementtype@^2.0.1:
+    version "2.0.1"
+    resolved "https://registry.yarnpkg.com/domelementtype/-/domelementtype-2.0.1.tgz#1f8bdfe91f5a78063274e803b4bdcedf6e94f94d"
+    integrity sha512-5HOHUDsYZWV8FGWN0Njbr/Rn7f/eWSQi1v7+HsUVwXgn8nWWlL64zKDkS0n8ZmQ3mlWOMuXOnR+7Nx/5tMO5AQ==
+  
+  domhandler@^2.3.0:
+    version "2.4.2"
+    resolved "https://registry.yarnpkg.com/domhandler/-/domhandler-2.4.2.tgz#8805097e933d65e85546f726d60f5eb88b44f803"
+    integrity sha512-JiK04h0Ht5u/80fdLMCEmV4zkNh2BcoMFBmZ/91WtYZ8qVXSKjiw7fXMgFPnHcSZgOo3XdinHvmnDUeMf5R4wA==
+    dependencies:
+      domelementtype "1"
+  
+  domutils@1.5.1:
+    version "1.5.1"
+    resolved "https://registry.yarnpkg.com/domutils/-/domutils-1.5.1.tgz#dcd8488a26f563d61079e48c9f7b7e32373682cf"
+    integrity sha1-3NhIiib1Y9YQeeSMn3t+Mjc2gs8=
+    dependencies:
+      dom-serializer "0"
+      domelementtype "1"
+  
+  domutils@^1.5.1, domutils@^1.7.0:
+    version "1.7.0"
+    resolved "https://registry.yarnpkg.com/domutils/-/domutils-1.7.0.tgz#56ea341e834e06e6748af7a1cb25da67ea9f8c2a"
+    integrity sha512-Lgd2XcJ/NjEw+7tFvfKxOzCYKZsdct5lczQ2ZaQY8Djz7pfAD3Gbp8ySJWtreII/vDlMVmxwa6pHmdxIYgttDg==
+    dependencies:
+      dom-serializer "0"
+      domelementtype "1"
+  
+  dot-prop@^5.2.0:
+    version "5.3.0"
+    resolved "https://registry.yarnpkg.com/dot-prop/-/dot-prop-5.3.0.tgz#90ccce708cd9cd82cc4dc8c3ddd9abdd55b20e88"
+    integrity sha512-QM8q3zDe58hqUqjraQOmzZ1LIH9SWQJTlEKCH4kJ2oQvLZk7RbQXvtDM2XEq3fwkV9CCvvH4LA0AV+ogFsBM2Q==
+    dependencies:
+      is-obj "^2.0.0"
+  
+  dotenv-expand@^5.1.0:
+    version "5.1.0"
+    resolved "https://registry.yarnpkg.com/dotenv-expand/-/dotenv-expand-5.1.0.tgz#3fbaf020bfd794884072ea26b1e9791d45a629f0"
+    integrity sha512-YXQl1DSa4/PQyRfgrv6aoNjhasp/p4qs9FjJ4q4cQk+8m4r6k4ZSiEyytKG8f8W9gi8WsQtIObNmKd+tMzNTmA==
+  
+  dotenv@^8.2.0:
+    version "8.2.0"
+    resolved "https://registry.yarnpkg.com/dotenv/-/dotenv-8.2.0.tgz#97e619259ada750eea3e4ea3e26bceea5424b16a"
+    integrity sha512-8sJ78ElpbDJBHNeBzUbUVLsqKdccaa/BXF1uPTw3GrvQTBgrQrtObr2mUrE38vzYd8cEv+m/JBfDLioYcfXoaw==
+  
+  duplexer@^0.1.1:
+    version "0.1.2"
+    resolved "https://registry.yarnpkg.com/duplexer/-/duplexer-0.1.2.tgz#3abe43aef3835f8ae077d136ddce0f276b0400e6"
+    integrity sha512-jtD6YG370ZCIi/9GTaJKQxWTZD045+4R4hTk/x1UyoqadyJ9x9CgSi1RlVDQF8U2sxLLSnFkCaMihqljHIWgMg==
+  
+  duplexify@^3.4.2, duplexify@^3.6.0:
+    version "3.7.1"
+    resolved "https://registry.yarnpkg.com/duplexify/-/duplexify-3.7.1.tgz#2a4df5317f6ccfd91f86d6fd25d8d8a103b88309"
+    integrity sha512-07z8uv2wMyS51kKhD1KsdXJg5WQ6t93RneqRxUHnskXVtlYYkLqM0gqStQZ3pj073g687jPCHrqNfCzawLYh5g==
+    dependencies:
+      end-of-stream "^1.0.0"
+      inherits "^2.0.1"
+      readable-stream "^2.0.0"
+      stream-shift "^1.0.0"
+  
+  easy-stack@^1.0.0:
+    version "1.0.0"
+    resolved "https://registry.yarnpkg.com/easy-stack/-/easy-stack-1.0.0.tgz#12c91b3085a37f0baa336e9486eac4bf94e3e788"
+    integrity sha1-EskbMIWjfwuqM26UhurEv5Tj54g=
+  
+  ecc-jsbn@~0.1.1:
+    version "0.1.2"
+    resolved "https://registry.yarnpkg.com/ecc-jsbn/-/ecc-jsbn-0.1.2.tgz#3a83a904e54353287874c564b7549386849a98c9"
+    integrity sha1-OoOpBOVDUyh4dMVkt1SThoSamMk=
+    dependencies:
+      jsbn "~0.1.0"
+      safer-buffer "^2.1.0"
+  
+  ee-first@1.1.1:
+    version "1.1.1"
+    resolved "https://registry.yarnpkg.com/ee-first/-/ee-first-1.1.1.tgz#590c61156b0ae2f4f0255732a158b266bc56b21d"
+    integrity sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0=
+  
+  ejs@^2.6.1:
+    version "2.7.4"
+    resolved "https://registry.yarnpkg.com/ejs/-/ejs-2.7.4.tgz#48661287573dcc53e366c7a1ae52c3a120eec9ba"
+    integrity sha512-7vmuyh5+kuUyJKePhQfRQBhXV5Ce+RnaeeQArKu1EAMpL3WbgMt5WG6uQZpEVvYSSsxMXRKOewtDk9RaTKXRlA==
+  
+  electron-to-chromium@^1.3.562:
+    version "1.3.562"
+    resolved "https://registry.yarnpkg.com/electron-to-chromium/-/electron-to-chromium-1.3.562.tgz#79c20277ee1c8d0173a22af00e38433b752bc70f"
+    integrity sha512-WhRe6liQ2q/w1MZc8mD8INkenHivuHdrr4r5EQHNomy3NJux+incP6M6lDMd0paShP3MD0WGe5R1TWmEClf+Bg==
+  
+  elliptic@^6.5.3:
+    version "6.5.3"
+    resolved "https://registry.yarnpkg.com/elliptic/-/elliptic-6.5.3.tgz#cb59eb2efdaf73a0bd78ccd7015a62ad6e0f93d6"
+    integrity sha512-IMqzv5wNQf+E6aHeIqATs0tOLeOTwj1QKbRcS3jBbYkl5oLAserA8yJTT7/VyHUYG91PRmPyeQDObKLPpeS4dw==
+    dependencies:
+      bn.js "^4.4.0"
+      brorand "^1.0.1"
+      hash.js "^1.0.0"
+      hmac-drbg "^1.0.0"
+      inherits "^2.0.1"
+      minimalistic-assert "^1.0.0"
+      minimalistic-crypto-utils "^1.0.0"
+  
+  emoji-regex@^7.0.1:
+    version "7.0.3"
+    resolved "https://registry.yarnpkg.com/emoji-regex/-/emoji-regex-7.0.3.tgz#933a04052860c85e83c122479c4748a8e4c72156"
+    integrity sha512-CwBLREIQ7LvYFB0WyRvwhq5N5qPhc6PMjD6bYggFlI5YyDgl+0vxq5VHbMOFqLg7hfWzmu8T5Z1QofhmTIhItA==
+  
+  emoji-regex@^8.0.0:
+    version "8.0.0"
+    resolved "https://registry.yarnpkg.com/emoji-regex/-/emoji-regex-8.0.0.tgz#e818fd69ce5ccfcb404594f842963bf53164cc37"
+    integrity sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==
+  
+  emojis-list@^2.0.0:
+    version "2.1.0"
+    resolved "https://registry.yarnpkg.com/emojis-list/-/emojis-list-2.1.0.tgz#4daa4d9db00f9819880c79fa457ae5b09a1fd389"
+    integrity sha1-TapNnbAPmBmIDHn6RXrlsJof04k=
+  
+  emojis-list@^3.0.0:
+    version "3.0.0"
+    resolved "https://registry.yarnpkg.com/emojis-list/-/emojis-list-3.0.0.tgz#5570662046ad29e2e916e71aae260abdff4f6a78"
+    integrity sha512-/kyM18EfinwXZbno9FyUGeFh87KC8HRQBQGildHZbEuRyWFOmv1U10o9BBp8XVZDVNNuQKyIGIu5ZYAAXJ0V2Q==
+  
+  encodeurl@~1.0.2:
+    version "1.0.2"
+    resolved "https://registry.yarnpkg.com/encodeurl/-/encodeurl-1.0.2.tgz#ad3ff4c86ec2d029322f5a02c3a9a606c95b3f59"
+    integrity sha1-rT/0yG7C0CkyL1oCw6mmBslbP1k=
+  
+  end-of-stream@^1.0.0, end-of-stream@^1.1.0:
+    version "1.4.4"
+    resolved "https://registry.yarnpkg.com/end-of-stream/-/end-of-stream-1.4.4.tgz#5ae64a5f45057baf3626ec14da0ca5e4b2431eb0"
+    integrity sha512-+uw1inIHVPQoaVuHzRyXd21icM+cnt4CzD5rW+NC1wjOUSTOs+Te7FOv7AhN7vS9x/oIyhLP5PR1H+phQAHu5Q==
+    dependencies:
+      once "^1.4.0"
+  
+  enhanced-resolve@^4.3.0:
+    version "4.3.0"
+    resolved "https://registry.yarnpkg.com/enhanced-resolve/-/enhanced-resolve-4.3.0.tgz#3b806f3bfafc1ec7de69551ef93cca46c1704126"
+    integrity sha512-3e87LvavsdxyoCfGusJnrZ5G8SLPOFeHSNpZI/ATL9a5leXo2k0w6MKnbqhdBad9qTobSfB20Ld7UmgoNbAZkQ==
+    dependencies:
+      graceful-fs "^4.1.2"
+      memory-fs "^0.5.0"
+      tapable "^1.0.0"
+  
+  entities@^1.1.1:
+    version "1.1.2"
+    resolved "https://registry.yarnpkg.com/entities/-/entities-1.1.2.tgz#bdfa735299664dfafd34529ed4f8522a275fea56"
+    integrity sha512-f2LZMYl1Fzu7YSBKg+RoROelpOaNrcGmE9AZubeDfrCEia483oW4MI4VyFd5VNHIgQ/7qm1I0wUHK1eJnn2y2w==
+  
+  entities@^2.0.0:
+    version "2.0.3"
+    resolved "https://registry.yarnpkg.com/entities/-/entities-2.0.3.tgz#5c487e5742ab93c15abb5da22759b8590ec03b7f"
+    integrity sha512-MyoZ0jgnLvB2X3Lg5HqpFmn1kybDiIfEQmKzTb5apr51Rb+T3KdmMiqa70T+bhGnyv7bQ6WMj2QMHpGMmlrUYQ==
+  
+  errno@^0.1.3, errno@~0.1.7:
+    version "0.1.7"
+    resolved "https://registry.yarnpkg.com/errno/-/errno-0.1.7.tgz#4684d71779ad39af177e3f007996f7c67c852618"
+    integrity sha512-MfrRBDWzIWifgq6tJj60gkAwtLNb6sQPlcFrSOflcP1aFmmruKQ2wRnze/8V6kgyz7H3FF8Npzv78mZ7XLLflg==
+    dependencies:
+      prr "~1.0.1"
+  
+  error-ex@^1.3.1:
+    version "1.3.2"
+    resolved "https://registry.yarnpkg.com/error-ex/-/error-ex-1.3.2.tgz#b4ac40648107fdcdcfae242f428bea8a14d4f1bf"
+    integrity sha512-7dFHNmqeFSEt2ZBsCriorKnn3Z2pj+fd9kmI6QoWw4//DL+icEBfc0U7qJCisqrTsKTjw4fNFy2pW9OqStD84g==
+    dependencies:
+      is-arrayish "^0.2.1"
+  
+  error-stack-parser@^2.0.0:
+    version "2.0.6"
+    resolved "https://registry.yarnpkg.com/error-stack-parser/-/error-stack-parser-2.0.6.tgz#5a99a707bd7a4c58a797902d48d82803ede6aad8"
+    integrity sha512-d51brTeqC+BHlwF0BhPtcYgF5nlzf9ZZ0ZIUQNZpc9ZB9qw5IJ2diTrBY9jlCJkTLITYPjmiX6OWCwH+fuyNgQ==
+    dependencies:
+      stackframe "^1.1.1"
+  
+  es-abstract@^1.17.0-next.1, es-abstract@^1.17.2, es-abstract@^1.17.5:
+    version "1.17.6"
+    resolved "https://registry.yarnpkg.com/es-abstract/-/es-abstract-1.17.6.tgz#9142071707857b2cacc7b89ecb670316c3e2d52a"
+    integrity sha512-Fr89bON3WFyUi5EvAeI48QTWX0AyekGgLA8H+c+7fbfCkJwRWRMLd8CQedNEyJuoYYhmtEqY92pgte1FAhBlhw==
+    dependencies:
+      es-to-primitive "^1.2.1"
+      function-bind "^1.1.1"
+      has "^1.0.3"
+      has-symbols "^1.0.1"
+      is-callable "^1.2.0"
+      is-regex "^1.1.0"
+      object-inspect "^1.7.0"
+      object-keys "^1.1.1"
+      object.assign "^4.1.0"
+      string.prototype.trimend "^1.0.1"
+      string.prototype.trimstart "^1.0.1"
+  
+  es-to-primitive@^1.2.1:
+    version "1.2.1"
+    resolved "https://registry.yarnpkg.com/es-to-primitive/-/es-to-primitive-1.2.1.tgz#e55cd4c9cdc188bcefb03b366c736323fc5c898a"
+    integrity sha512-QCOllgZJtaUo9miYBcLChTUaHNjJF3PYs1VidD7AwiEj1kYxKeQTctLAezAOH5ZKRH0g2IgPn6KwB4IT8iRpvA==
+    dependencies:
+      is-callable "^1.1.4"
+      is-date-object "^1.0.1"
+      is-symbol "^1.0.2"
+  
+  escalade@^3.0.2:
+    version "3.0.2"
+    resolved "https://registry.yarnpkg.com/escalade/-/escalade-3.0.2.tgz#6a580d70edb87880f22b4c91d0d56078df6962c4"
+    integrity sha512-gPYAU37hYCUhW5euPeR+Y74F7BL+IBsV93j5cvGriSaD1aG6MGsqsV1yamRdrWrb2j3aiZvb0X+UBOWpx3JWtQ==
+  
+  escape-html@~1.0.3:
+    version "1.0.3"
+    resolved "https://registry.yarnpkg.com/escape-html/-/escape-html-1.0.3.tgz#0258eae4d3d0c0974de1c169188ef0051d1d1988"
+    integrity sha1-Aljq5NPQwJdN4cFpGI7wBR0dGYg=
+  
+  escape-string-regexp@^1.0.2, escape-string-regexp@^1.0.5:
+    version "1.0.5"
+    resolved "https://registry.yarnpkg.com/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz#1b61c0562190a8dff6ae3bb2cf0200ca130b86d4"
+    integrity sha1-G2HAViGQqN/2rjuyzwIAyhMLhtQ=
+  
+  eslint-loader@^2.2.1:
+    version "2.2.1"
+    resolved "https://registry.yarnpkg.com/eslint-loader/-/eslint-loader-2.2.1.tgz#28b9c12da54057af0845e2a6112701a2f6bf8337"
+    integrity sha512-RLgV9hoCVsMLvOxCuNjdqOrUqIj9oJg8hF44vzJaYqsAHuY9G2YAeN3joQ9nxP0p5Th9iFSIpKo+SD8KISxXRg==
+    dependencies:
+      loader-fs-cache "^1.0.0"
+      loader-utils "^1.0.2"
+      object-assign "^4.0.1"
+      object-hash "^1.1.4"
+      rimraf "^2.6.1"
+  
+  eslint-plugin-vue@^6.2.2:
+    version "6.2.2"
+    resolved "https://registry.yarnpkg.com/eslint-plugin-vue/-/eslint-plugin-vue-6.2.2.tgz#27fecd9a3a24789b0f111ecdd540a9e56198e0fe"
+    integrity sha512-Nhc+oVAHm0uz/PkJAWscwIT4ijTrK5fqNqz9QB1D35SbbuMG1uB6Yr5AJpvPSWg+WOw7nYNswerYh0kOk64gqQ==
+    dependencies:
+      natural-compare "^1.4.0"
+      semver "^5.6.0"
+      vue-eslint-parser "^7.0.0"
+  
+  eslint-scope@^4.0.3:
+    version "4.0.3"
+    resolved "https://registry.yarnpkg.com/eslint-scope/-/eslint-scope-4.0.3.tgz#ca03833310f6889a3264781aa82e63eb9cfe7848"
+    integrity sha512-p7VutNr1O/QrxysMo3E45FjYDTeXBy0iTltPFNSqKAIfjDSXC+4dj+qfyuD8bfAXrW/y6lW3O76VaYNPKfpKrg==
+    dependencies:
+      esrecurse "^4.1.0"
+      estraverse "^4.1.1"
+  
+  eslint-scope@^5.0.0:
+    version "5.1.0"
+    resolved "https://registry.yarnpkg.com/eslint-scope/-/eslint-scope-5.1.0.tgz#d0f971dfe59c69e0cada684b23d49dbf82600ce5"
+    integrity sha512-iiGRvtxWqgtx5m8EyQUJihBloE4EnYeGE/bz1wSPwJE6tZuJUtHlhqDM4Xj2ukE8Dyy1+HCZ4hE0fzIVMzb58w==
+    dependencies:
+      esrecurse "^4.1.0"
+      estraverse "^4.1.1"
+  
+  eslint-utils@^1.4.3:
+    version "1.4.3"
+    resolved "https://registry.yarnpkg.com/eslint-utils/-/eslint-utils-1.4.3.tgz#74fec7c54d0776b6f67e0251040b5806564e981f"
+    integrity sha512-fbBN5W2xdY45KulGXmLHZ3c3FHfVYmKg0IrAKGOkT/464PQsx2UeIzfz1RmEci+KLm1bBaAzZAh8+/E+XAeZ8Q==
+    dependencies:
+      eslint-visitor-keys "^1.1.0"
+  
+  eslint-visitor-keys@^1.0.0, eslint-visitor-keys@^1.1.0:
+    version "1.3.0"
+    resolved "https://registry.yarnpkg.com/eslint-visitor-keys/-/eslint-visitor-keys-1.3.0.tgz#30ebd1ef7c2fdff01c3a4f151044af25fab0523e"
+    integrity sha512-6J72N8UNa462wa/KFODt/PJ3IU60SDpC3QXC1Hjc1BXXpfL2C9R5+AU7jhe0F6GREqVMh4Juu+NY7xn+6dipUQ==
+  
+  eslint@^6.7.2:
+    version "6.8.0"
+    resolved "https://registry.yarnpkg.com/eslint/-/eslint-6.8.0.tgz#62262d6729739f9275723824302fb227c8c93ffb"
+    integrity sha512-K+Iayyo2LtyYhDSYwz5D5QdWw0hCacNzyq1Y821Xna2xSJj7cijoLLYmLxTQgcgZ9mC61nryMy9S7GRbYpI5Ig==
+    dependencies:
+      "@babel/code-frame" "^7.0.0"
+      ajv "^6.10.0"
+      chalk "^2.1.0"
+      cross-spawn "^6.0.5"
+      debug "^4.0.1"
+      doctrine "^3.0.0"
+      eslint-scope "^5.0.0"
+      eslint-utils "^1.4.3"
+      eslint-visitor-keys "^1.1.0"
+      espree "^6.1.2"
+      esquery "^1.0.1"
+      esutils "^2.0.2"
+      file-entry-cache "^5.0.1"
+      functional-red-black-tree "^1.0.1"
+      glob-parent "^5.0.0"
+      globals "^12.1.0"
+      ignore "^4.0.6"
+      import-fresh "^3.0.0"
+      imurmurhash "^0.1.4"
+      inquirer "^7.0.0"
+      is-glob "^4.0.0"
+      js-yaml "^3.13.1"
+      json-stable-stringify-without-jsonify "^1.0.1"
+      levn "^0.3.0"
+      lodash "^4.17.14"
+      minimatch "^3.0.4"
+      mkdirp "^0.5.1"
+      natural-compare "^1.4.0"
+      optionator "^0.8.3"
+      progress "^2.0.0"
+      regexpp "^2.0.1"
+      semver "^6.1.2"
+      strip-ansi "^5.2.0"
+      strip-json-comments "^3.0.1"
+      table "^5.2.3"
+      text-table "^0.2.0"
+      v8-compile-cache "^2.0.3"
+  
+  espree@^6.1.2, espree@^6.2.1:
+    version "6.2.1"
+    resolved "https://registry.yarnpkg.com/espree/-/espree-6.2.1.tgz#77fc72e1fd744a2052c20f38a5b575832e82734a"
+    integrity sha512-ysCxRQY3WaXJz9tdbWOwuWr5Y/XrPTGX9Kiz3yoUXwW0VZ4w30HTkQLaGx/+ttFjF8i+ACbArnB4ce68a9m5hw==
+    dependencies:
+      acorn "^7.1.1"
+      acorn-jsx "^5.2.0"
+      eslint-visitor-keys "^1.1.0"
+  
+  esprima@^4.0.0:
+    version "4.0.1"
+    resolved "https://registry.yarnpkg.com/esprima/-/esprima-4.0.1.tgz#13b04cdb3e6c5d19df91ab6987a8695619b0aa71"
+    integrity sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==
+  
+  esquery@^1.0.1:
+    version "1.3.1"
+    resolved "https://registry.yarnpkg.com/esquery/-/esquery-1.3.1.tgz#b78b5828aa8e214e29fb74c4d5b752e1c033da57"
+    integrity sha512-olpvt9QG0vniUBZspVRN6lwB7hOZoTRtT+jzR+tS4ffYx2mzbw+z0XCOk44aaLYKApNX5nMm+E+P6o25ip/DHQ==
+    dependencies:
+      estraverse "^5.1.0"
+  
+  esrecurse@^4.1.0:
+    version "4.3.0"
+    resolved "https://registry.yarnpkg.com/esrecurse/-/esrecurse-4.3.0.tgz#7ad7964d679abb28bee72cec63758b1c5d2c9921"
+    integrity sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag==
+    dependencies:
+      estraverse "^5.2.0"
+  
+  estraverse@^4.1.1:
+    version "4.3.0"
+    resolved "https://registry.yarnpkg.com/estraverse/-/estraverse-4.3.0.tgz#398ad3f3c5a24948be7725e83d11a7de28cdbd1d"
+    integrity sha512-39nnKffWz8xN1BU/2c79n9nB9HDzo0niYUqx6xyqUnyoAnQyyWpOTdZEeiCch8BBu515t4wp9ZmgVfVhn9EBpw==
+  
+  estraverse@^5.1.0, estraverse@^5.2.0:
+    version "5.2.0"
+    resolved "https://registry.yarnpkg.com/estraverse/-/estraverse-5.2.0.tgz#307df42547e6cc7324d3cf03c155d5cdb8c53880"
+    integrity sha512-BxbNGGNm0RyRYvUdHpIwv9IWzeM9XClbOxwoATuFdOE7ZE6wHL+HQ5T8hoPM+zHvmKzzsEqhgy0GrQ5X13afiQ==
+  
+  esutils@^2.0.2:
+    version "2.0.3"
+    resolved "https://registry.yarnpkg.com/esutils/-/esutils-2.0.3.tgz#74d2eb4de0b8da1293711910d50775b9b710ef64"
+    integrity sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==
+  
+  etag@~1.8.1:
+    version "1.8.1"
+    resolved "https://registry.yarnpkg.com/etag/-/etag-1.8.1.tgz#41ae2eeb65efa62268aebfea83ac7d79299b0887"
+    integrity sha1-Qa4u62XvpiJorr/qg6x9eSmbCIc=
+  
+  event-pubsub@4.3.0:
+    version "4.3.0"
+    resolved "https://registry.yarnpkg.com/event-pubsub/-/event-pubsub-4.3.0.tgz#f68d816bc29f1ec02c539dc58c8dd40ce72cb36e"
+    integrity sha512-z7IyloorXvKbFx9Bpie2+vMJKKx1fH1EN5yiTfp8CiLOTptSYy1g8H4yDpGlEdshL1PBiFtBHepF2cNsqeEeFQ==
+  
+  eventemitter3@^4.0.0:
+    version "4.0.7"
+    resolved "https://registry.yarnpkg.com/eventemitter3/-/eventemitter3-4.0.7.tgz#2de9b68f6528d5644ef5c59526a1b4a07306169f"
+    integrity sha512-8guHBZCwKnFhYdHr2ysuRWErTwhoN2X8XELRlrRwpmfeY2jjuUN4taQMsULKUVo1K4DvZl+0pgfyoysHxvmvEw==
+  
+  events@^3.0.0:
+    version "3.2.0"
+    resolved "https://registry.yarnpkg.com/events/-/events-3.2.0.tgz#93b87c18f8efcd4202a461aec4dfc0556b639379"
+    integrity sha512-/46HWwbfCX2xTawVfkKLGxMifJYQBWMwY1mjywRtb4c9x8l5NP3KoJtnIOiL1hfdRkIuYhETxQlo62IF8tcnlg==
+  
+  eventsource@^1.0.7:
+    version "1.0.7"
+    resolved "https://registry.yarnpkg.com/eventsource/-/eventsource-1.0.7.tgz#8fbc72c93fcd34088090bc0a4e64f4b5cee6d8d0"
+    integrity sha512-4Ln17+vVT0k8aWq+t/bF5arcS3EpT9gYtW66EPacdj/mAFevznsnyoHLPy2BA8gbIQeIHoPsvwmfBftfcG//BQ==
+    dependencies:
+      original "^1.0.0"
+  
+  evp_bytestokey@^1.0.0, evp_bytestokey@^1.0.3:
+    version "1.0.3"
+    resolved "https://registry.yarnpkg.com/evp_bytestokey/-/evp_bytestokey-1.0.3.tgz#7fcbdb198dc71959432efe13842684e0525acb02"
+    integrity sha512-/f2Go4TognH/KvCISP7OUsHn85hT9nUkxxA9BEWxFn+Oj9o8ZNLm/40hdlgSLyuOimsrTKLUMEorQexp/aPQeA==
+    dependencies:
+      md5.js "^1.3.4"
+      safe-buffer "^5.1.1"
+  
+  execa@^0.8.0:
+    version "0.8.0"
+    resolved "https://registry.yarnpkg.com/execa/-/execa-0.8.0.tgz#d8d76bbc1b55217ed190fd6dd49d3c774ecfc8da"
+    integrity sha1-2NdrvBtVIX7RkP1t1J08d07PyNo=
+    dependencies:
+      cross-spawn "^5.0.1"
+      get-stream "^3.0.0"
+      is-stream "^1.1.0"
+      npm-run-path "^2.0.0"
+      p-finally "^1.0.0"
+      signal-exit "^3.0.0"
+      strip-eof "^1.0.0"
+  
+  execa@^1.0.0:
+    version "1.0.0"
+    resolved "https://registry.yarnpkg.com/execa/-/execa-1.0.0.tgz#c6236a5bb4df6d6f15e88e7f017798216749ddd8"
+    integrity sha512-adbxcyWV46qiHyvSp50TKt05tB4tK3HcmF7/nxfAdhnox83seTDbwnaqKO4sXRy7roHAIFqJP/Rw/AuEbX61LA==
+    dependencies:
+      cross-spawn "^6.0.0"
+      get-stream "^4.0.0"
+      is-stream "^1.1.0"
+      npm-run-path "^2.0.0"
+      p-finally "^1.0.0"
+      signal-exit "^3.0.0"
+      strip-eof "^1.0.0"
+  
+  execa@^3.3.0:
+    version "3.4.0"
+    resolved "https://registry.yarnpkg.com/execa/-/execa-3.4.0.tgz#c08ed4550ef65d858fac269ffc8572446f37eb89"
+    integrity sha512-r9vdGQk4bmCuK1yKQu1KTwcT2zwfWdbdaXfCtAh+5nU/4fSX+JAb7vZGvI5naJrQlvONrEB20jeruESI69530g==
+    dependencies:
+      cross-spawn "^7.0.0"
+      get-stream "^5.0.0"
+      human-signals "^1.1.1"
+      is-stream "^2.0.0"
+      merge-stream "^2.0.0"
+      npm-run-path "^4.0.0"
+      onetime "^5.1.0"
+      p-finally "^2.0.0"
+      signal-exit "^3.0.2"
+      strip-final-newline "^2.0.0"
+  
+  expand-brackets@^2.1.4:
+    version "2.1.4"
+    resolved "https://registry.yarnpkg.com/expand-brackets/-/expand-brackets-2.1.4.tgz#b77735e315ce30f6b6eff0f83b04151a22449622"
+    integrity sha1-t3c14xXOMPa27/D4OwQVGiJEliI=
+    dependencies:
+      debug "^2.3.3"
+      define-property "^0.2.5"
+      extend-shallow "^2.0.1"
+      posix-character-classes "^0.1.0"
+      regex-not "^1.0.0"
+      snapdragon "^0.8.1"
+      to-regex "^3.0.1"
+  
+  express@^4.16.3, express@^4.17.1:
+    version "4.17.1"
+    resolved "https://registry.yarnpkg.com/express/-/express-4.17.1.tgz#4491fc38605cf51f8629d39c2b5d026f98a4c134"
+    integrity sha512-mHJ9O79RqluphRrcw2X/GTh3k9tVv8YcoyY4Kkh4WDMUYKRZUq0h1o0w2rrrxBqM7VoeUVqgb27xlEMXTnYt4g==
+    dependencies:
+      accepts "~1.3.7"
+      array-flatten "1.1.1"
+      body-parser "1.19.0"
+      content-disposition "0.5.3"
+      content-type "~1.0.4"
+      cookie "0.4.0"
+      cookie-signature "1.0.6"
+      debug "2.6.9"
+      depd "~1.1.2"
+      encodeurl "~1.0.2"
+      escape-html "~1.0.3"
+      etag "~1.8.1"
+      finalhandler "~1.1.2"
+      fresh "0.5.2"
+      merge-descriptors "1.0.1"
+      methods "~1.1.2"
+      on-finished "~2.3.0"
+      parseurl "~1.3.3"
+      path-to-regexp "0.1.7"
+      proxy-addr "~2.0.5"
+      qs "6.7.0"
+      range-parser "~1.2.1"
+      safe-buffer "5.1.2"
+      send "0.17.1"
+      serve-static "1.14.1"
+      setprototypeof "1.1.1"
+      statuses "~1.5.0"
+      type-is "~1.6.18"
+      utils-merge "1.0.1"
+      vary "~1.1.2"
+  
+  extend-shallow@^2.0.1:
+    version "2.0.1"
+    resolved "https://registry.yarnpkg.com/extend-shallow/-/extend-shallow-2.0.1.tgz#51af7d614ad9a9f610ea1bafbb989d6b1c56890f"
+    integrity sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=
+    dependencies:
+      is-extendable "^0.1.0"
+  
+  extend-shallow@^3.0.0, extend-shallow@^3.0.2:
+    version "3.0.2"
+    resolved "https://registry.yarnpkg.com/extend-shallow/-/extend-shallow-3.0.2.tgz#26a71aaf073b39fb2127172746131c2704028db8"
+    integrity sha1-Jqcarwc7OfshJxcnRhMcJwQCjbg=
+    dependencies:
+      assign-symbols "^1.0.0"
+      is-extendable "^1.0.1"
+  
+  extend@~3.0.2:
+    version "3.0.2"
+    resolved "https://registry.yarnpkg.com/extend/-/extend-3.0.2.tgz#f8b1136b4071fbd8eb140aff858b1019ec2915fa"
+    integrity sha512-fjquC59cD7CyW6urNXK0FBufkZcoiGG80wTuPujX590cB5Ttln20E2UB4S/WARVqhXffZl2LNgS+gQdPIIim/g==
+  
+  external-editor@^3.0.3:
+    version "3.1.0"
+    resolved "https://registry.yarnpkg.com/external-editor/-/external-editor-3.1.0.tgz#cb03f740befae03ea4d283caed2741a83f335495"
+    integrity sha512-hMQ4CX1p1izmuLYyZqLMO/qGNw10wSv9QDCPfzXfyFrOaCSSoRfqE1Kf1s5an66J5JZC62NewG+mK49jOCtQew==
+    dependencies:
+      chardet "^0.7.0"
+      iconv-lite "^0.4.24"
+      tmp "^0.0.33"
+  
+  extglob@^2.0.4:
+    version "2.0.4"
+    resolved "https://registry.yarnpkg.com/extglob/-/extglob-2.0.4.tgz#ad00fe4dc612a9232e8718711dc5cb5ab0285543"
+    integrity sha512-Nmb6QXkELsuBr24CJSkilo6UHHgbekK5UiZgfE6UHD3Eb27YC6oD+bhcT+tJ6cl8dmsgdQxnWlcry8ksBIBLpw==
+    dependencies:
+      array-unique "^0.3.2"
+      define-property "^1.0.0"
+      expand-brackets "^2.1.4"
+      extend-shallow "^2.0.1"
+      fragment-cache "^0.2.1"
+      regex-not "^1.0.0"
+      snapdragon "^0.8.1"
+      to-regex "^3.0.1"
+  
+  extsprintf@1.3.0:
+    version "1.3.0"
+    resolved "https://registry.yarnpkg.com/extsprintf/-/extsprintf-1.3.0.tgz#96918440e3041a7a414f8c52e3c574eb3c3e1e05"
+    integrity sha1-lpGEQOMEGnpBT4xS48V06zw+HgU=
+  
+  extsprintf@^1.2.0:
+    version "1.4.0"
+    resolved "https://registry.yarnpkg.com/extsprintf/-/extsprintf-1.4.0.tgz#e2689f8f356fad62cca65a3a91c5df5f9551692f"
+    integrity sha1-4mifjzVvrWLMplo6kcXfX5VRaS8=
+  
+  fast-deep-equal@^3.1.1:
+    version "3.1.3"
+    resolved "https://registry.yarnpkg.com/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz#3a7d56b559d6cbc3eb512325244e619a65c6c525"
+    integrity sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==
+  
+  fast-glob@^2.2.6:
+    version "2.2.7"
+    resolved "https://registry.yarnpkg.com/fast-glob/-/fast-glob-2.2.7.tgz#6953857c3afa475fff92ee6015d52da70a4cd39d"
+    integrity sha512-g1KuQwHOZAmOZMuBtHdxDtju+T2RT8jgCC9aANsbpdiDDTSnjgfuVsIBNKbUeJI3oKMRExcfNDtJl4OhbffMsw==
+    dependencies:
+      "@mrmlnc/readdir-enhanced" "^2.2.1"
+      "@nodelib/fs.stat" "^1.1.2"
+      glob-parent "^3.1.0"
+      is-glob "^4.0.0"
+      merge2 "^1.2.3"
+      micromatch "^3.1.10"
+  
+  fast-json-stable-stringify@^2.0.0:
+    version "2.1.0"
+    resolved "https://registry.yarnpkg.com/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz#874bf69c6f404c2b5d99c481341399fd55892633"
+    integrity sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==
+  
+  fast-levenshtein@~2.0.6:
+    version "2.0.6"
+    resolved "https://registry.yarnpkg.com/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz#3d8a5c66883a16a30ca8643e851f19baa7797917"
+    integrity sha1-PYpcZog6FqMMqGQ+hR8Zuqd5eRc=
+  
+  faye-websocket@^0.10.0:
+    version "0.10.0"
+    resolved "https://registry.yarnpkg.com/faye-websocket/-/faye-websocket-0.10.0.tgz#4e492f8d04dfb6f89003507f6edbf2d501e7c6f4"
+    integrity sha1-TkkvjQTftviQA1B/btvy1QHnxvQ=
+    dependencies:
+      websocket-driver ">=0.5.1"
+  
+  faye-websocket@~0.11.1:
+    version "0.11.3"
+    resolved "https://registry.yarnpkg.com/faye-websocket/-/faye-websocket-0.11.3.tgz#5c0e9a8968e8912c286639fde977a8b209f2508e"
+    integrity sha512-D2y4bovYpzziGgbHYtGCMjlJM36vAl/y+xUyn1C+FVx8szd1E+86KwVw6XvYSzOP8iMpm1X0I4xJD+QtUb36OA==
+    dependencies:
+      websocket-driver ">=0.5.1"
+  
+  figgy-pudding@^3.5.1:
+    version "3.5.2"
+    resolved "https://registry.yarnpkg.com/figgy-pudding/-/figgy-pudding-3.5.2.tgz#b4eee8148abb01dcf1d1ac34367d59e12fa61d6e"
+    integrity sha512-0btnI/H8f2pavGMN8w40mlSKOfTK2SVJmBfBeVIj3kNw0swwgzyRq0d5TJVOwodFmtvpPeWPN/MCcfuWF0Ezbw==
+  
+  figures@^3.0.0:
+    version "3.2.0"
+    resolved "https://registry.yarnpkg.com/figures/-/figures-3.2.0.tgz#625c18bd293c604dc4a8ddb2febf0c88341746af"
+    integrity sha512-yaduQFRKLXYOGgEn6AZau90j3ggSOyiqXU0F9JZfeXYhNa+Jk4X+s45A2zg5jns87GAFa34BBm2kXw4XpNcbdg==
+    dependencies:
+      escape-string-regexp "^1.0.5"
+  
+  file-entry-cache@^5.0.1:
+    version "5.0.1"
+    resolved "https://registry.yarnpkg.com/file-entry-cache/-/file-entry-cache-5.0.1.tgz#ca0f6efa6dd3d561333fb14515065c2fafdf439c"
+    integrity sha512-bCg29ictuBaKUwwArK4ouCaqDgLZcysCFLmM/Yn/FDoqndh/9vNuQfXRDvTuXKLxfD/JtZQGKFT8MGcJBK644g==
+    dependencies:
+      flat-cache "^2.0.1"
+  
+  file-loader@^4.2.0:
+    version "4.3.0"
+    resolved "https://registry.yarnpkg.com/file-loader/-/file-loader-4.3.0.tgz#780f040f729b3d18019f20605f723e844b8a58af"
+    integrity sha512-aKrYPYjF1yG3oX0kWRrqrSMfgftm7oJW5M+m4owoldH5C51C0RkIwB++JbRvEW3IU6/ZG5n8UvEcdgwOt2UOWA==
+    dependencies:
+      loader-utils "^1.2.3"
+      schema-utils "^2.5.0"
+  
+  file-uri-to-path@1.0.0:
+    version "1.0.0"
+    resolved "https://registry.yarnpkg.com/file-uri-to-path/-/file-uri-to-path-1.0.0.tgz#553a7b8446ff6f684359c445f1e37a05dacc33dd"
+    integrity sha512-0Zt+s3L7Vf1biwWZ29aARiVYLx7iMGnEUl9x33fbB/j3jR81u/O2LbqK+Bm1CDSNDKVtJ/YjwY7TUd5SkeLQLw==
+  
+  filesize@^3.6.1:
+    version "3.6.1"
+    resolved "https://registry.yarnpkg.com/filesize/-/filesize-3.6.1.tgz#090bb3ee01b6f801a8a8be99d31710b3422bb317"
+    integrity sha512-7KjR1vv6qnicaPMi1iiTcI85CyYwRO/PSFCu6SvqL8jN2Wjt/NIYQTFtFs7fSDCYOstUkEWIQGFUg5YZQfjlcg==
+  
+  fill-range@^4.0.0:
+    version "4.0.0"
+    resolved "https://registry.yarnpkg.com/fill-range/-/fill-range-4.0.0.tgz#d544811d428f98eb06a63dc402d2403c328c38f7"
+    integrity sha1-1USBHUKPmOsGpj3EAtJAPDKMOPc=
+    dependencies:
+      extend-shallow "^2.0.1"
+      is-number "^3.0.0"
+      repeat-string "^1.6.1"
+      to-regex-range "^2.1.0"
+  
+  fill-range@^7.0.1:
+    version "7.0.1"
+    resolved "https://registry.yarnpkg.com/fill-range/-/fill-range-7.0.1.tgz#1919a6a7c75fe38b2c7c77e5198535da9acdda40"
+    integrity sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==
+    dependencies:
+      to-regex-range "^5.0.1"
+  
+  finalhandler@~1.1.2:
+    version "1.1.2"
+    resolved "https://registry.yarnpkg.com/finalhandler/-/finalhandler-1.1.2.tgz#b7e7d000ffd11938d0fdb053506f6ebabe9f587d"
+    integrity sha512-aAWcW57uxVNrQZqFXjITpW3sIUQmHGG3qSb9mUah9MgMC4NeWhNOlNjXEYq3HjRAvL6arUviZGGJsBg6z0zsWA==
+    dependencies:
+      debug "2.6.9"
+      encodeurl "~1.0.2"
+      escape-html "~1.0.3"
+      on-finished "~2.3.0"
+      parseurl "~1.3.3"
+      statuses "~1.5.0"
+      unpipe "~1.0.0"
+  
+  find-cache-dir@^0.1.1:
+    version "0.1.1"
+    resolved "https://registry.yarnpkg.com/find-cache-dir/-/find-cache-dir-0.1.1.tgz#c8defae57c8a52a8a784f9e31c57c742e993a0b9"
+    integrity sha1-yN765XyKUqinhPnjHFfHQumToLk=
+    dependencies:
+      commondir "^1.0.1"
+      mkdirp "^0.5.1"
+      pkg-dir "^1.0.0"
+  
+  find-cache-dir@^2.1.0:
+    version "2.1.0"
+    resolved "https://registry.yarnpkg.com/find-cache-dir/-/find-cache-dir-2.1.0.tgz#8d0f94cd13fe43c6c7c261a0d86115ca918c05f7"
+    integrity sha512-Tq6PixE0w/VMFfCgbONnkiQIVol/JJL7nRMi20fqzA4NRs9AfeqMGeRdPi3wIhYkxjeBaWh2rxwapn5Tu3IqOQ==
+    dependencies:
+      commondir "^1.0.1"
+      make-dir "^2.0.0"
+      pkg-dir "^3.0.0"
+  
+  find-cache-dir@^3.0.0, find-cache-dir@^3.3.1:
+    version "3.3.1"
+    resolved "https://registry.yarnpkg.com/find-cache-dir/-/find-cache-dir-3.3.1.tgz#89b33fad4a4670daa94f855f7fbe31d6d84fe880"
+    integrity sha512-t2GDMt3oGC/v+BMwzmllWDuJF/xcDtE5j/fCGbqDD7OLuJkj0cfh1YSA5VKPvwMeLFLNDBkwOKZ2X85jGLVftQ==
+    dependencies:
+      commondir "^1.0.1"
+      make-dir "^3.0.2"
+      pkg-dir "^4.1.0"
+  
+  find-up@^1.0.0:
+    version "1.1.2"
+    resolved "https://registry.yarnpkg.com/find-up/-/find-up-1.1.2.tgz#6b2e9822b1a2ce0a60ab64d610eccad53cb24d0f"
+    integrity sha1-ay6YIrGizgpgq2TWEOzK1TyyTQ8=
+    dependencies:
+      path-exists "^2.0.0"
+      pinkie-promise "^2.0.0"
+  
+  find-up@^3.0.0:
+    version "3.0.0"
+    resolved "https://registry.yarnpkg.com/find-up/-/find-up-3.0.0.tgz#49169f1d7993430646da61ecc5ae355c21c97b73"
+    integrity sha512-1yD6RmLI1XBfxugvORwlck6f75tYL+iR0jqwsOrOxMZyGYqUuDhJ0l4AXdO1iX/FTs9cBAMEk1gWSEx1kSbylg==
+    dependencies:
+      locate-path "^3.0.0"
+  
+  find-up@^4.0.0, find-up@^4.1.0:
+    version "4.1.0"
+    resolved "https://registry.yarnpkg.com/find-up/-/find-up-4.1.0.tgz#97afe7d6cdc0bc5928584b7c8d7b16e8a9aa5d19"
+    integrity sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==
+    dependencies:
+      locate-path "^5.0.0"
+      path-exists "^4.0.0"
+  
+  flat-cache@^2.0.1:
+    version "2.0.1"
+    resolved "https://registry.yarnpkg.com/flat-cache/-/flat-cache-2.0.1.tgz#5d296d6f04bda44a4630a301413bdbc2ec085ec0"
+    integrity sha512-LoQe6yDuUMDzQAEH8sgmh4Md6oZnc/7PjtwjNFSzveXqSHt6ka9fPBuso7IGf9Rz4uqnSnWiFH2B/zj24a5ReA==
+    dependencies:
+      flatted "^2.0.0"
+      rimraf "2.6.3"
+      write "1.0.3"
+  
+  flatted@^2.0.0:
+    version "2.0.2"
+    resolved "https://registry.yarnpkg.com/flatted/-/flatted-2.0.2.tgz#4575b21e2bcee7434aa9be662f4b7b5f9c2b5138"
+    integrity sha512-r5wGx7YeOwNWNlCA0wQ86zKyDLMQr+/RB8xy74M4hTphfmjlijTSSXGuH8rnvKZnfT9i+75zmd8jcKdMR4O6jA==
+  
+  flush-write-stream@^1.0.0:
+    version "1.1.1"
+    resolved "https://registry.yarnpkg.com/flush-write-stream/-/flush-write-stream-1.1.1.tgz#8dd7d873a1babc207d94ead0c2e0e44276ebf2e8"
+    integrity sha512-3Z4XhFZ3992uIq0XOqb9AreonueSYphE6oYbpt5+3u06JWklbsPkNv3ZKkP9Bz/r+1MWCaMoSQ28P85+1Yc77w==
+    dependencies:
+      inherits "^2.0.3"
+      readable-stream "^2.3.6"
+  
+  follow-redirects@^1.0.0:
+    version "1.13.0"
+    resolved "https://registry.yarnpkg.com/follow-redirects/-/follow-redirects-1.13.0.tgz#b42e8d93a2a7eea5ed88633676d6597bc8e384db"
+    integrity sha512-aq6gF1BEKje4a9i9+5jimNFIpq4Q1WiwBToeRK5NvZBd/TRsmW8BsJfOEGkr76TbOyPVD3OVDN910EcUNtRYEA==
+  
+  for-in@^1.0.2:
+    version "1.0.2"
+    resolved "https://registry.yarnpkg.com/for-in/-/for-in-1.0.2.tgz#81068d295a8142ec0ac726c6e2200c30fb6d5e80"
+    integrity sha1-gQaNKVqBQuwKxybG4iAMMPttXoA=
+  
+  forever-agent@~0.6.1:
+    version "0.6.1"
+    resolved "https://registry.yarnpkg.com/forever-agent/-/forever-agent-0.6.1.tgz#fbc71f0c41adeb37f96c577ad1ed42d8fdacca91"
+    integrity sha1-+8cfDEGt6zf5bFd60e1C2P2sypE=
+  
+  form-data@~2.3.2:
+    version "2.3.3"
+    resolved "https://registry.yarnpkg.com/form-data/-/form-data-2.3.3.tgz#dcce52c05f644f298c6a7ab936bd724ceffbf3a6"
+    integrity sha512-1lLKB2Mu3aGP1Q/2eCOx0fNbRMe7XdwktwOruhfqqd0rIJWwN4Dh+E3hrPSlDCXnSR7UtZ1N38rVXm+6+MEhJQ==
+    dependencies:
+      asynckit "^0.4.0"
+      combined-stream "^1.0.6"
+      mime-types "^2.1.12"
+  
+  forwarded@~0.1.2:
+    version "0.1.2"
+    resolved "https://registry.yarnpkg.com/forwarded/-/forwarded-0.1.2.tgz#98c23dab1175657b8c0573e8ceccd91b0ff18c84"
+    integrity sha1-mMI9qxF1ZXuMBXPozszZGw/xjIQ=
+  
+  fragment-cache@^0.2.1:
+    version "0.2.1"
+    resolved "https://registry.yarnpkg.com/fragment-cache/-/fragment-cache-0.2.1.tgz#4290fad27f13e89be7f33799c6bc5a0abfff0d19"
+    integrity sha1-QpD60n8T6Jvn8zeZxrxaCr//DRk=
+    dependencies:
+      map-cache "^0.2.2"
+  
+  fresh@0.5.2:
+    version "0.5.2"
+    resolved "https://registry.yarnpkg.com/fresh/-/fresh-0.5.2.tgz#3d8cadd90d976569fa835ab1f8e4b23a105605a7"
+    integrity sha1-PYyt2Q2XZWn6g1qx+OSyOhBWBac=
+  
+  from2@^2.1.0:
+    version "2.3.0"
+    resolved "https://registry.yarnpkg.com/from2/-/from2-2.3.0.tgz#8bfb5502bde4a4d36cfdeea007fcca21d7e382af"
+    integrity sha1-i/tVAr3kpNNs/e6gB/zKIdfjgq8=
+    dependencies:
+      inherits "^2.0.1"
+      readable-stream "^2.0.0"
+  
+  fs-extra@^7.0.1:
+    version "7.0.1"
+    resolved "https://registry.yarnpkg.com/fs-extra/-/fs-extra-7.0.1.tgz#4f189c44aa123b895f722804f55ea23eadc348e9"
+    integrity sha512-YJDaCJZEnBmcbw13fvdAM9AwNOJwOzrE4pqMqBq5nFiEqXUqHwlK4B+3pUw6JNvfSPtX05xFHtYy/1ni01eGCw==
+    dependencies:
+      graceful-fs "^4.1.2"
+      jsonfile "^4.0.0"
+      universalify "^0.1.0"
+  
+  fs-minipass@^2.0.0:
+    version "2.1.0"
+    resolved "https://registry.yarnpkg.com/fs-minipass/-/fs-minipass-2.1.0.tgz#7f5036fdbf12c63c169190cbe4199c852271f9fb"
+    integrity sha512-V/JgOLFCS+R6Vcq0slCuaeWEdNC3ouDlJMNIsacH2VtALiu9mV4LPrHc5cDl8k5aw6J8jwgWWpiTo5RYhmIzvg==
+    dependencies:
+      minipass "^3.0.0"
+  
+  fs-write-stream-atomic@^1.0.8:
+    version "1.0.10"
+    resolved "https://registry.yarnpkg.com/fs-write-stream-atomic/-/fs-write-stream-atomic-1.0.10.tgz#b47df53493ef911df75731e70a9ded0189db40c9"
+    integrity sha1-tH31NJPvkR33VzHnCp3tAYnbQMk=
+    dependencies:
+      graceful-fs "^4.1.2"
+      iferr "^0.1.5"
+      imurmurhash "^0.1.4"
+      readable-stream "1 || 2"
+  
+  fs.realpath@^1.0.0:
+    version "1.0.0"
+    resolved "https://registry.yarnpkg.com/fs.realpath/-/fs.realpath-1.0.0.tgz#1504ad2523158caa40db4a2787cb01411994ea4f"
+    integrity sha1-FQStJSMVjKpA20onh8sBQRmU6k8=
+  
+  fsevents@^1.2.7:
+    version "1.2.13"
+    resolved "https://registry.yarnpkg.com/fsevents/-/fsevents-1.2.13.tgz#f325cb0455592428bcf11b383370ef70e3bfcc38"
+    integrity sha512-oWb1Z6mkHIskLzEJ/XWX0srkpkTQ7vaopMQkyaEIoq0fmtFVxOthb8cCxeT+p3ynTdkk/RZwbgG4brR5BeWECw==
+    dependencies:
+      bindings "^1.5.0"
+      nan "^2.12.1"
+  
+  fsevents@~2.1.2:
+    version "2.1.3"
+    resolved "https://registry.yarnpkg.com/fsevents/-/fsevents-2.1.3.tgz#fb738703ae8d2f9fe900c33836ddebee8b97f23e"
+    integrity sha512-Auw9a4AxqWpa9GUfj370BMPzzyncfBABW8Mab7BGWBYDj4Isgq+cDKtx0i6u9jcX9pQDnswsaaOTgTmA5pEjuQ==
+  
+  function-bind@^1.1.1:
+    version "1.1.1"
+    resolved "https://registry.yarnpkg.com/function-bind/-/function-bind-1.1.1.tgz#a56899d3ea3c9bab874bb9773b7c5ede92f4895d"
+    integrity sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A==
+  
+  functional-red-black-tree@^1.0.1:
+    version "1.0.1"
+    resolved "https://registry.yarnpkg.com/functional-red-black-tree/-/functional-red-black-tree-1.0.1.tgz#1b0ab3bd553b2a0d6399d29c0e3ea0b252078327"
+    integrity sha1-GwqzvVU7Kg1jmdKcDj6gslIHgyc=
+  
+  gensync@^1.0.0-beta.1:
+    version "1.0.0-beta.1"
+    resolved "https://registry.yarnpkg.com/gensync/-/gensync-1.0.0-beta.1.tgz#58f4361ff987e5ff6e1e7a210827aa371eaac269"
+    integrity sha512-r8EC6NO1sngH/zdD9fiRDLdcgnbayXah+mLgManTaIZJqEC1MZstmnox8KpnI2/fxQwrp5OpCOYWLp4rBl4Jcg==
+  
+  get-caller-file@^2.0.1:
+    version "2.0.5"
+    resolved "https://registry.yarnpkg.com/get-caller-file/-/get-caller-file-2.0.5.tgz#4f94412a82db32f36e3b0b9741f8a97feb031f7e"
+    integrity sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==
+  
+  get-stream@^3.0.0:
+    version "3.0.0"
+    resolved "https://registry.yarnpkg.com/get-stream/-/get-stream-3.0.0.tgz#8e943d1358dc37555054ecbe2edb05aa174ede14"
+    integrity sha1-jpQ9E1jcN1VQVOy+LtsFqhdO3hQ=
+  
+  get-stream@^4.0.0:
+    version "4.1.0"
+    resolved "https://registry.yarnpkg.com/get-stream/-/get-stream-4.1.0.tgz#c1b255575f3dc21d59bfc79cd3d2b46b1c3a54b5"
+    integrity sha512-GMat4EJ5161kIy2HevLlr4luNjBgvmj413KaQA7jt4V8B4RDsfpHk7WQ9GVqfYyyx8OS/L66Kox+rJRNklLK7w==
+    dependencies:
+      pump "^3.0.0"
+  
+  get-stream@^5.0.0:
+    version "5.2.0"
+    resolved "https://registry.yarnpkg.com/get-stream/-/get-stream-5.2.0.tgz#4966a1795ee5ace65e706c4b7beb71257d6e22d3"
+    integrity sha512-nBF+F1rAZVCu/p7rjzgA+Yb4lfYXrpl7a6VmJrU8wF9I1CKvP/QwPNZHnOlwbTkY6dvtFIzFMSyQXbLoTQPRpA==
+    dependencies:
+      pump "^3.0.0"
+  
+  get-value@^2.0.3, get-value@^2.0.6:
+    version "2.0.6"
+    resolved "https://registry.yarnpkg.com/get-value/-/get-value-2.0.6.tgz#dc15ca1c672387ca76bd37ac0a395ba2042a2c28"
+    integrity sha1-3BXKHGcjh8p2vTesCjlbogQqLCg=
+  
+  getpass@^0.1.1:
+    version "0.1.7"
+    resolved "https://registry.yarnpkg.com/getpass/-/getpass-0.1.7.tgz#5eff8e3e684d569ae4cb2b1282604e8ba62149fa"
+    integrity sha1-Xv+OPmhNVprkyysSgmBOi6YhSfo=
+    dependencies:
+      assert-plus "^1.0.0"
+  
+  glob-parent@^3.1.0:
+    version "3.1.0"
+    resolved "https://registry.yarnpkg.com/glob-parent/-/glob-parent-3.1.0.tgz#9e6af6299d8d3bd2bd40430832bd113df906c5ae"
+    integrity sha1-nmr2KZ2NO9K9QEMIMr0RPfkGxa4=
+    dependencies:
+      is-glob "^3.1.0"
+      path-dirname "^1.0.0"
+  
+  glob-parent@^5.0.0, glob-parent@~5.1.0:
+    version "5.1.1"
+    resolved "https://registry.yarnpkg.com/glob-parent/-/glob-parent-5.1.1.tgz#b6c1ef417c4e5663ea498f1c45afac6916bbc229"
+    integrity sha512-FnI+VGOpnlGHWZxthPGR+QhR78fuiK0sNLkHQv+bL9fQi57lNNdquIbna/WrfROrolq8GK5Ek6BiMwqL/voRYQ==
+    dependencies:
+      is-glob "^4.0.1"
+  
+  glob-to-regexp@^0.3.0:
+    version "0.3.0"
+    resolved "https://registry.yarnpkg.com/glob-to-regexp/-/glob-to-regexp-0.3.0.tgz#8c5a1494d2066c570cc3bfe4496175acc4d502ab"
+    integrity sha1-jFoUlNIGbFcMw7/kSWF1rMTVAqs=
+  
+  glob@^7.0.3, glob@^7.1.2, glob@^7.1.3, glob@^7.1.4:
+    version "7.1.6"
+    resolved "https://registry.yarnpkg.com/glob/-/glob-7.1.6.tgz#141f33b81a7c2492e125594307480c46679278a6"
+    integrity sha512-LwaxwyZ72Lk7vZINtNNrywX0ZuLyStrdDtabefZKAY5ZGJhVtgdznluResxNmPitE0SAO+O26sWTHeKSI2wMBA==
+    dependencies:
+      fs.realpath "^1.0.0"
+      inflight "^1.0.4"
+      inherits "2"
+      minimatch "^3.0.4"
+      once "^1.3.0"
+      path-is-absolute "^1.0.0"
+  
+  globals@^11.1.0:
+    version "11.12.0"
+    resolved "https://registry.yarnpkg.com/globals/-/globals-11.12.0.tgz#ab8795338868a0babd8525758018c2a7eb95c42e"
+    integrity sha512-WOBp/EEGUiIsJSp7wcv/y6MO+lV9UoncWqxuFfm8eBwzWNgyfBd6Gz+IeKQ9jCmyhoH99g15M3T+QaVHFjizVA==
+  
+  globals@^12.1.0:
+    version "12.4.0"
+    resolved "https://registry.yarnpkg.com/globals/-/globals-12.4.0.tgz#a18813576a41b00a24a97e7f815918c2e19925f8"
+    integrity sha512-BWICuzzDvDoH54NHKCseDanAhE3CeDorgDL5MT6LMXXj2WCnd9UC2szdk4AWLfjdgNBCXLUanXYcpBBKOSWGwg==
+    dependencies:
+      type-fest "^0.8.1"
+  
+  globby@^6.1.0:
+    version "6.1.0"
+    resolved "https://registry.yarnpkg.com/globby/-/globby-6.1.0.tgz#f5a6d70e8395e21c858fb0489d64df02424d506c"
+    integrity sha1-9abXDoOV4hyFj7BInWTfAkJNUGw=
+    dependencies:
+      array-union "^1.0.1"
+      glob "^7.0.3"
+      object-assign "^4.0.1"
+      pify "^2.0.0"
+      pinkie-promise "^2.0.0"
+  
+  globby@^7.1.1:
+    version "7.1.1"
+    resolved "https://registry.yarnpkg.com/globby/-/globby-7.1.1.tgz#fb2ccff9401f8600945dfada97440cca972b8680"
+    integrity sha1-+yzP+UAfhgCUXfral0QMypcrhoA=
+    dependencies:
+      array-union "^1.0.1"
+      dir-glob "^2.0.0"
+      glob "^7.1.2"
+      ignore "^3.3.5"
+      pify "^3.0.0"
+      slash "^1.0.0"
+  
+  globby@^9.2.0:
+    version "9.2.0"
+    resolved "https://registry.yarnpkg.com/globby/-/globby-9.2.0.tgz#fd029a706c703d29bdd170f4b6db3a3f7a7cb63d"
+    integrity sha512-ollPHROa5mcxDEkwg6bPt3QbEf4pDQSNtd6JPL1YvOvAo/7/0VAm9TccUeoTmarjPw4pfUthSCqcyfNB1I3ZSg==
+    dependencies:
+      "@types/glob" "^7.1.1"
+      array-union "^1.0.2"
+      dir-glob "^2.2.2"
+      fast-glob "^2.2.6"
+      glob "^7.1.3"
+      ignore "^4.0.3"
+      pify "^4.0.1"
+      slash "^2.0.0"
+  
+  graceful-fs@^4.1.11, graceful-fs@^4.1.15, graceful-fs@^4.1.2, graceful-fs@^4.1.6, graceful-fs@^4.2.2:
+    version "4.2.4"
+    resolved "https://registry.yarnpkg.com/graceful-fs/-/graceful-fs-4.2.4.tgz#2256bde14d3632958c465ebc96dc467ca07a29fb"
+    integrity sha512-WjKPNJF79dtJAVniUlGGWHYGz2jWxT6VhN/4m1NdkbZ2nOsEF+cI1Edgql5zCRhs/VsQYRvrXctxktVXZUkixw==
+  
+  gzip-size@^5.0.0:
+    version "5.1.1"
+    resolved "https://registry.yarnpkg.com/gzip-size/-/gzip-size-5.1.1.tgz#cb9bee692f87c0612b232840a873904e4c135274"
+    integrity sha512-FNHi6mmoHvs1mxZAds4PpdCS6QG8B4C1krxJsMutgxl5t3+GlRTzzI3NEkifXx2pVsOvJdOGSmIgDhQ55FwdPA==
+    dependencies:
+      duplexer "^0.1.1"
+      pify "^4.0.1"
+  
+  handle-thing@^2.0.0:
+    version "2.0.1"
+    resolved "https://registry.yarnpkg.com/handle-thing/-/handle-thing-2.0.1.tgz#857f79ce359580c340d43081cc648970d0bb234e"
+    integrity sha512-9Qn4yBxelxoh2Ow62nP+Ka/kMnOXRi8BXnRaUwezLNhqelnN49xKz4F/dPP8OYLxLxq6JDtZb2i9XznUQbNPTg==
+  
+  har-schema@^2.0.0:
+    version "2.0.0"
+    resolved "https://registry.yarnpkg.com/har-schema/-/har-schema-2.0.0.tgz#a94c2224ebcac04782a0d9035521f24735b7ec92"
+    integrity sha1-qUwiJOvKwEeCoNkDVSHyRzW37JI=
+  
+  har-validator@~5.1.3:
+    version "5.1.5"
+    resolved "https://registry.yarnpkg.com/har-validator/-/har-validator-5.1.5.tgz#1f0803b9f8cb20c0fa13822df1ecddb36bde1efd"
+    integrity sha512-nmT2T0lljbxdQZfspsno9hgrG3Uir6Ks5afism62poxqBM6sDnMEuPmzTq8XN0OEwqKLLdh1jQI3qyE66Nzb3w==
+    dependencies:
+      ajv "^6.12.3"
+      har-schema "^2.0.0"
+  
+  has-ansi@^2.0.0:
+    version "2.0.0"
+    resolved "https://registry.yarnpkg.com/has-ansi/-/has-ansi-2.0.0.tgz#34f5049ce1ecdf2b0649af3ef24e45ed35416d91"
+    integrity sha1-NPUEnOHs3ysGSa8+8k5F7TVBbZE=
+    dependencies:
+      ansi-regex "^2.0.0"
+  
+  has-flag@^3.0.0:
+    version "3.0.0"
+    resolved "https://registry.yarnpkg.com/has-flag/-/has-flag-3.0.0.tgz#b5d454dc2199ae225699f3467e5a07f3b955bafd"
+    integrity sha1-tdRU3CGZriJWmfNGfloH87lVuv0=
+  
+  has-flag@^4.0.0:
+    version "4.0.0"
+    resolved "https://registry.yarnpkg.com/has-flag/-/has-flag-4.0.0.tgz#944771fd9c81c81265c4d6941860da06bb59479b"
+    integrity sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==
+  
+  has-symbols@^1.0.0, has-symbols@^1.0.1:
+    version "1.0.1"
+    resolved "https://registry.yarnpkg.com/has-symbols/-/has-symbols-1.0.1.tgz#9f5214758a44196c406d9bd76cebf81ec2dd31e8"
+    integrity sha512-PLcsoqu++dmEIZB+6totNFKq/7Do+Z0u4oT0zKOJNl3lYK6vGwwu2hjHs+68OEZbTjiUE9bgOABXbP/GvrS0Kg==
+  
+  has-value@^0.3.1:
+    version "0.3.1"
+    resolved "https://registry.yarnpkg.com/has-value/-/has-value-0.3.1.tgz#7b1f58bada62ca827ec0a2078025654845995e1f"
+    integrity sha1-ex9YutpiyoJ+wKIHgCVlSEWZXh8=
+    dependencies:
+      get-value "^2.0.3"
+      has-values "^0.1.4"
+      isobject "^2.0.0"
+  
+  has-value@^1.0.0:
+    version "1.0.0"
+    resolved "https://registry.yarnpkg.com/has-value/-/has-value-1.0.0.tgz#18b281da585b1c5c51def24c930ed29a0be6b177"
+    integrity sha1-GLKB2lhbHFxR3vJMkw7SmgvmsXc=
+    dependencies:
+      get-value "^2.0.6"
+      has-values "^1.0.0"
+      isobject "^3.0.0"
+  
+  has-values@^0.1.4:
+    version "0.1.4"
+    resolved "https://registry.yarnpkg.com/has-values/-/has-values-0.1.4.tgz#6d61de95d91dfca9b9a02089ad384bff8f62b771"
+    integrity sha1-bWHeldkd/Km5oCCJrThL/49it3E=
+  
+  has-values@^1.0.0:
+    version "1.0.0"
+    resolved "https://registry.yarnpkg.com/has-values/-/has-values-1.0.0.tgz#95b0b63fec2146619a6fe57fe75628d5a39efe4f"
+    integrity sha1-lbC2P+whRmGab+V/51Yo1aOe/k8=
+    dependencies:
+      is-number "^3.0.0"
+      kind-of "^4.0.0"
+  
+  has@^1.0.0, has@^1.0.3:
+    version "1.0.3"
+    resolved "https://registry.yarnpkg.com/has/-/has-1.0.3.tgz#722d7cbfc1f6aa8241f16dd814e011e1f41e8796"
+    integrity sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw==
+    dependencies:
+      function-bind "^1.1.1"
+  
+  hash-base@^3.0.0:
+    version "3.1.0"
+    resolved "https://registry.yarnpkg.com/hash-base/-/hash-base-3.1.0.tgz#55c381d9e06e1d2997a883b4a3fddfe7f0d3af33"
+    integrity sha512-1nmYp/rhMDiE7AYkDw+lLwlAzz0AntGIe51F3RfFfEqyQ3feY2eI/NcwC6umIQVOASPMsWJLJScWKSSvzL9IVA==
+    dependencies:
+      inherits "^2.0.4"
+      readable-stream "^3.6.0"
+      safe-buffer "^5.2.0"
+  
+  hash-sum@^1.0.2:
+    version "1.0.2"
+    resolved "https://registry.yarnpkg.com/hash-sum/-/hash-sum-1.0.2.tgz#33b40777754c6432573c120cc3808bbd10d47f04"
+    integrity sha1-M7QHd3VMZDJXPBIMw4CLvRDUfwQ=
+  
+  hash-sum@^2.0.0:
+    version "2.0.0"
+    resolved "https://registry.yarnpkg.com/hash-sum/-/hash-sum-2.0.0.tgz#81d01bb5de8ea4a214ad5d6ead1b523460b0b45a"
+    integrity sha512-WdZTbAByD+pHfl/g9QSsBIIwy8IT+EsPiKDs0KNX+zSHhdDLFKdZu0BQHljvO+0QI/BasbMSUa8wYNCZTvhslg==
+  
+  hash.js@^1.0.0, hash.js@^1.0.3:
+    version "1.1.7"
+    resolved "https://registry.yarnpkg.com/hash.js/-/hash.js-1.1.7.tgz#0babca538e8d4ee4a0f8988d68866537a003cf42"
+    integrity sha512-taOaskGt4z4SOANNseOviYDvjEJinIkRgmp7LbKP2YTTmVxWBl87s/uzK9r+44BclBSp2X7K1hqeNfz9JbBeXA==
+    dependencies:
+      inherits "^2.0.3"
+      minimalistic-assert "^1.0.1"
+  
+  he@1.2.x, he@^1.1.0:
+    version "1.2.0"
+    resolved "https://registry.yarnpkg.com/he/-/he-1.2.0.tgz#84ae65fa7eafb165fddb61566ae14baf05664f0f"
+    integrity sha512-F/1DnUGPopORZi0ni+CvrCgHQ5FyEAHRLSApuYWMmrbSwoN2Mn/7k+Gl38gJnR7yyDZk6WLXwiGod1JOWNDKGw==
+  
+  hex-color-regex@^1.1.0:
+    version "1.1.0"
+    resolved "https://registry.yarnpkg.com/hex-color-regex/-/hex-color-regex-1.1.0.tgz#4c06fccb4602fe2602b3c93df82d7e7dbf1a8a8e"
+    integrity sha512-l9sfDFsuqtOqKDsQdqrMRk0U85RZc0RtOR9yPI7mRVOa4FsR/BVnZ0shmQRM96Ji99kYZP/7hn1cedc1+ApsTQ==
+  
+  highlight.js@^9.6.0:
+    version "9.18.3"
+    resolved "https://registry.yarnpkg.com/highlight.js/-/highlight.js-9.18.3.tgz#a1a0a2028d5e3149e2380f8a865ee8516703d634"
+    integrity sha512-zBZAmhSupHIl5sITeMqIJnYCDfAEc3Gdkqj65wC1lpI468MMQeeQkhcIAvk+RylAkxrCcI9xy9piHiXeQ1BdzQ==
+  
+  hmac-drbg@^1.0.0:
+    version "1.0.1"
+    resolved "https://registry.yarnpkg.com/hmac-drbg/-/hmac-drbg-1.0.1.tgz#d2745701025a6c775a6c545793ed502fc0c649a1"
+    integrity sha1-0nRXAQJabHdabFRXk+1QL8DGSaE=
+    dependencies:
+      hash.js "^1.0.3"
+      minimalistic-assert "^1.0.0"
+      minimalistic-crypto-utils "^1.0.1"
+  
+  hoopy@^0.1.4:
+    version "0.1.4"
+    resolved "https://registry.yarnpkg.com/hoopy/-/hoopy-0.1.4.tgz#609207d661100033a9a9402ad3dea677381c1b1d"
+    integrity sha512-HRcs+2mr52W0K+x8RzcLzuPPmVIKMSv97RGHy0Ea9y/mpcaK+xTrjICA04KAHi4GRzxliNqNJEFYWHghy3rSfQ==
+  
+  hosted-git-info@^2.1.4:
+    version "2.8.8"
+    resolved "https://registry.yarnpkg.com/hosted-git-info/-/hosted-git-info-2.8.8.tgz#7539bd4bc1e0e0a895815a2e0262420b12858488"
+    integrity sha512-f/wzC2QaWBs7t9IYqB4T3sR1xviIViXJRJTWBlx2Gf3g0Xi5vI7Yy4koXQ1c9OYDGHN9sBy1DQ2AB8fqZBWhUg==
+  
+  hpack.js@^2.1.6:
+    version "2.1.6"
+    resolved "https://registry.yarnpkg.com/hpack.js/-/hpack.js-2.1.6.tgz#87774c0949e513f42e84575b3c45681fade2a0b2"
+    integrity sha1-h3dMCUnlE/QuhFdbPEVoH63ioLI=
+    dependencies:
+      inherits "^2.0.1"
+      obuf "^1.0.0"
+      readable-stream "^2.0.1"
+      wbuf "^1.1.0"
+  
+  hsl-regex@^1.0.0:
+    version "1.0.0"
+    resolved "https://registry.yarnpkg.com/hsl-regex/-/hsl-regex-1.0.0.tgz#d49330c789ed819e276a4c0d272dffa30b18fe6e"
+    integrity sha1-1JMwx4ntgZ4nakwNJy3/owsY/m4=
+  
+  hsla-regex@^1.0.0:
+    version "1.0.0"
+    resolved "https://registry.yarnpkg.com/hsla-regex/-/hsla-regex-1.0.0.tgz#c1ce7a3168c8c6614033a4b5f7877f3b225f9c38"
+    integrity sha1-wc56MWjIxmFAM6S194d/OyJfnDg=
+  
+  html-comment-regex@^1.1.0:
+    version "1.1.2"
+    resolved "https://registry.yarnpkg.com/html-comment-regex/-/html-comment-regex-1.1.2.tgz#97d4688aeb5c81886a364faa0cad1dda14d433a7"
+    integrity sha512-P+M65QY2JQ5Y0G9KKdlDpo0zK+/OHptU5AaBwUfAIDJZk1MYf32Frm84EcOytfJE0t5JvkAnKlmjsXDnWzCJmQ==
+  
+  html-entities@^1.3.1:
+    version "1.3.1"
+    resolved "https://registry.yarnpkg.com/html-entities/-/html-entities-1.3.1.tgz#fb9a1a4b5b14c5daba82d3e34c6ae4fe701a0e44"
+    integrity sha512-rhE/4Z3hIhzHAUKbW8jVcCyuT5oJCXXqhN/6mXXVCpzTmvJnoH2HL/bt3EZ6p55jbFJBeAe1ZNpL5BugLujxNA==
+  
+  html-minifier@^3.2.3:
+    version "3.5.21"
+    resolved "https://registry.yarnpkg.com/html-minifier/-/html-minifier-3.5.21.tgz#d0040e054730e354db008463593194015212d20c"
+    integrity sha512-LKUKwuJDhxNa3uf/LPR/KVjm/l3rBqtYeCOAekvG8F1vItxMUpueGd94i/asDDr8/1u7InxzFA5EeGjhhG5mMA==
+    dependencies:
+      camel-case "3.0.x"
+      clean-css "4.2.x"
+      commander "2.17.x"
+      he "1.2.x"
+      param-case "2.1.x"
+      relateurl "0.2.x"
+      uglify-js "3.4.x"
+  
+  html-tags@^2.0.0:
+    version "2.0.0"
+    resolved "https://registry.yarnpkg.com/html-tags/-/html-tags-2.0.0.tgz#10b30a386085f43cede353cc8fa7cb0deeea668b"
+    integrity sha1-ELMKOGCF9Dzt41PMj6fLDe7qZos=
+  
+  html-tags@^3.1.0:
+    version "3.1.0"
+    resolved "https://registry.yarnpkg.com/html-tags/-/html-tags-3.1.0.tgz#7b5e6f7e665e9fb41f30007ed9e0d41e97fb2140"
+    integrity sha512-1qYz89hW3lFDEazhjW0yVAV87lw8lVkrJocr72XmBkMKsoSVJCQx3W8BXsC7hO2qAt8BoVjYjtAcZ9perqGnNg==
+  
+  html-webpack-plugin@^3.2.0:
+    version "3.2.0"
+    resolved "https://registry.yarnpkg.com/html-webpack-plugin/-/html-webpack-plugin-3.2.0.tgz#b01abbd723acaaa7b37b6af4492ebda03d9dd37b"
+    integrity sha1-sBq71yOsqqeze2r0SS69oD2d03s=
+    dependencies:
+      html-minifier "^3.2.3"
+      loader-utils "^0.2.16"
+      lodash "^4.17.3"
+      pretty-error "^2.0.2"
+      tapable "^1.0.0"
+      toposort "^1.0.0"
+      util.promisify "1.0.0"
+  
+  htmlparser2@^3.3.0:
+    version "3.10.1"
+    resolved "https://registry.yarnpkg.com/htmlparser2/-/htmlparser2-3.10.1.tgz#bd679dc3f59897b6a34bb10749c855bb53a9392f"
+    integrity sha512-IgieNijUMbkDovyoKObU1DUhm1iwNYE/fuifEoEHfd1oZKZDaONBSkal7Y01shxsM49R4XaMdGez3WnF9UfiCQ==
+    dependencies:
+      domelementtype "^1.3.1"
+      domhandler "^2.3.0"
+      domutils "^1.5.1"
+      entities "^1.1.1"
+      inherits "^2.0.1"
+      readable-stream "^3.1.1"
+  
+  http-deceiver@^1.2.7:
+    version "1.2.7"
+    resolved "https://registry.yarnpkg.com/http-deceiver/-/http-deceiver-1.2.7.tgz#fa7168944ab9a519d337cb0bec7284dc3e723d87"
+    integrity sha1-+nFolEq5pRnTN8sL7HKE3D5yPYc=
+  
+  http-errors@1.7.2:
+    version "1.7.2"
+    resolved "https://registry.yarnpkg.com/http-errors/-/http-errors-1.7.2.tgz#4f5029cf13239f31036e5b2e55292bcfbcc85c8f"
+    integrity sha512-uUQBt3H/cSIVfch6i1EuPNy/YsRSOUBXTVfZ+yR7Zjez3qjBz6i9+i4zjNaoqcoFVI4lQJ5plg63TvGfRSDCRg==
+    dependencies:
+      depd "~1.1.2"
+      inherits "2.0.3"
+      setprototypeof "1.1.1"
+      statuses ">= 1.5.0 < 2"
+      toidentifier "1.0.0"
+  
+  http-errors@~1.6.2:
+    version "1.6.3"
+    resolved "https://registry.yarnpkg.com/http-errors/-/http-errors-1.6.3.tgz#8b55680bb4be283a0b5bf4ea2e38580be1d9320d"
+    integrity sha1-i1VoC7S+KDoLW/TqLjhYC+HZMg0=
+    dependencies:
+      depd "~1.1.2"
+      inherits "2.0.3"
+      setprototypeof "1.1.0"
+      statuses ">= 1.4.0 < 2"
+  
+  http-errors@~1.7.2:
+    version "1.7.3"
+    resolved "https://registry.yarnpkg.com/http-errors/-/http-errors-1.7.3.tgz#6c619e4f9c60308c38519498c14fbb10aacebb06"
+    integrity sha512-ZTTX0MWrsQ2ZAhA1cejAwDLycFsd7I7nVtnkT3Ol0aqodaKW+0CTZDQ1uBv5whptCnc8e8HeRRJxRs0kmm/Qfw==
+    dependencies:
+      depd "~1.1.2"
+      inherits "2.0.4"
+      setprototypeof "1.1.1"
+      statuses ">= 1.5.0 < 2"
+      toidentifier "1.0.0"
+  
+  http-parser-js@>=0.5.1:
+    version "0.5.2"
+    resolved "https://registry.yarnpkg.com/http-parser-js/-/http-parser-js-0.5.2.tgz#da2e31d237b393aae72ace43882dd7e270a8ff77"
+    integrity sha512-opCO9ASqg5Wy2FNo7A0sxy71yGbbkJJXLdgMK04Tcypw9jr2MgWbyubb0+WdmDmGnFflO7fRbqbaihh/ENDlRQ==
+  
+  http-proxy-middleware@0.19.1:
+    version "0.19.1"
+    resolved "https://registry.yarnpkg.com/http-proxy-middleware/-/http-proxy-middleware-0.19.1.tgz#183c7dc4aa1479150306498c210cdaf96080a43a"
+    integrity sha512-yHYTgWMQO8VvwNS22eLLloAkvungsKdKTLO8AJlftYIKNfJr3GK3zK0ZCfzDDGUBttdGc8xFy1mCitvNKQtC3Q==
+    dependencies:
+      http-proxy "^1.17.0"
+      is-glob "^4.0.0"
+      lodash "^4.17.11"
+      micromatch "^3.1.10"
+  
+  http-proxy@^1.17.0:
+    version "1.18.1"
+    resolved "https://registry.yarnpkg.com/http-proxy/-/http-proxy-1.18.1.tgz#401541f0534884bbf95260334e72f88ee3976549"
+    integrity sha512-7mz/721AbnJwIVbnaSv1Cz3Am0ZLT/UBwkC92VlxhXv/k/BBQfM2fXElQNC27BVGr0uwUpplYPQM9LnaBMR5NQ==
+    dependencies:
+      eventemitter3 "^4.0.0"
+      follow-redirects "^1.0.0"
+      requires-port "^1.0.0"
+  
+  http-signature@~1.2.0:
+    version "1.2.0"
+    resolved "https://registry.yarnpkg.com/http-signature/-/http-signature-1.2.0.tgz#9aecd925114772f3d95b65a60abb8f7c18fbace1"
+    integrity sha1-muzZJRFHcvPZW2WmCruPfBj7rOE=
+    dependencies:
+      assert-plus "^1.0.0"
+      jsprim "^1.2.2"
+      sshpk "^1.7.0"
+  
+  https-browserify@^1.0.0:
+    version "1.0.0"
+    resolved "https://registry.yarnpkg.com/https-browserify/-/https-browserify-1.0.0.tgz#ec06c10e0a34c0f2faf199f7fd7fc78fffd03c73"
+    integrity sha1-7AbBDgo0wPL68Zn3/X/Hj//QPHM=
+  
+  human-signals@^1.1.1:
+    version "1.1.1"
+    resolved "https://registry.yarnpkg.com/human-signals/-/human-signals-1.1.1.tgz#c5b1cd14f50aeae09ab6c59fe63ba3395fe4dfa3"
+    integrity sha512-SEQu7vl8KjNL2eoGBLF3+wAjpsNfA9XMlXAYj/3EdaNfAlxKthD1xjEQfGOUhllCGGJVNY34bRr6lPINhNjyZw==
+  
+  iconv-lite@0.4.24, iconv-lite@^0.4.24:
+    version "0.4.24"
+    resolved "https://registry.yarnpkg.com/iconv-lite/-/iconv-lite-0.4.24.tgz#2022b4b25fbddc21d2f524974a474aafe733908b"
+    integrity sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==
+    dependencies:
+      safer-buffer ">= 2.1.2 < 3"
+  
+  icss-utils@^4.0.0, icss-utils@^4.1.1:
+    version "4.1.1"
+    resolved "https://registry.yarnpkg.com/icss-utils/-/icss-utils-4.1.1.tgz#21170b53789ee27447c2f47dd683081403f9a467"
+    integrity sha512-4aFq7wvWyMHKgxsH8QQtGpvbASCf+eM3wPRLI6R+MgAnTCZ6STYsRvttLvRWK0Nfif5piF394St3HeJDaljGPA==
+    dependencies:
+      postcss "^7.0.14"
+  
+  ieee754@^1.1.4:
+    version "1.1.13"
+    resolved "https://registry.yarnpkg.com/ieee754/-/ieee754-1.1.13.tgz#ec168558e95aa181fd87d37f55c32bbcb6708b84"
+    integrity sha512-4vf7I2LYV/HaWerSo3XmlMkp5eZ83i+/CDluXi/IGTs/O1sejBNhTtnxzmRZfvOUqj7lZjqHkeTvpgSFDlWZTg==
+  
+  iferr@^0.1.5:
+    version "0.1.5"
+    resolved "https://registry.yarnpkg.com/iferr/-/iferr-0.1.5.tgz#c60eed69e6d8fdb6b3104a1fcbca1c192dc5b501"
+    integrity sha1-xg7taebY/bazEEofy8ocGS3FtQE=
+  
+  ignore@^3.3.5:
+    version "3.3.10"
+    resolved "https://registry.yarnpkg.com/ignore/-/ignore-3.3.10.tgz#0a97fb876986e8081c631160f8f9f389157f0043"
+    integrity sha512-Pgs951kaMm5GXP7MOvxERINe3gsaVjUWFm+UZPSq9xYriQAksyhg0csnS0KXSNRD5NmNdapXEpjxG49+AKh/ug==
+  
+  ignore@^4.0.3, ignore@^4.0.6:
+    version "4.0.6"
+    resolved "https://registry.yarnpkg.com/ignore/-/ignore-4.0.6.tgz#750e3db5862087b4737ebac8207ffd1ef27b25fc"
+    integrity sha512-cyFDKrqc/YdcWFniJhzI42+AzS+gNwmUzOSFcRCQYwySuBBBy/KjuxWLZ/FHEH6Moq1NizMOBWyTcv8O4OZIMg==
+  
+  import-cwd@^2.0.0:
+    version "2.1.0"
+    resolved "https://registry.yarnpkg.com/import-cwd/-/import-cwd-2.1.0.tgz#aa6cf36e722761285cb371ec6519f53e2435b0a9"
+    integrity sha1-qmzzbnInYShcs3HsZRn1PiQ1sKk=
+    dependencies:
+      import-from "^2.1.0"
+  
+  import-fresh@^2.0.0:
+    version "2.0.0"
+    resolved "https://registry.yarnpkg.com/import-fresh/-/import-fresh-2.0.0.tgz#d81355c15612d386c61f9ddd3922d4304822a546"
+    integrity sha1-2BNVwVYS04bGH53dOSLUMEgipUY=
+    dependencies:
+      caller-path "^2.0.0"
+      resolve-from "^3.0.0"
+  
+  import-fresh@^3.0.0:
+    version "3.2.1"
+    resolved "https://registry.yarnpkg.com/import-fresh/-/import-fresh-3.2.1.tgz#633ff618506e793af5ac91bf48b72677e15cbe66"
+    integrity sha512-6e1q1cnWP2RXD9/keSkxHScg508CdXqXWgWBaETNhyuBFz+kUZlKboh+ISK+bU++DmbHimVBrOz/zzPe0sZ3sQ==
+    dependencies:
+      parent-module "^1.0.0"
+      resolve-from "^4.0.0"
+  
+  import-from@^2.1.0:
+    version "2.1.0"
+    resolved "https://registry.yarnpkg.com/import-from/-/import-from-2.1.0.tgz#335db7f2a7affd53aaa471d4b8021dee36b7f3b1"
+    integrity sha1-M1238qev/VOqpHHUuAId7ja387E=
+    dependencies:
+      resolve-from "^3.0.0"
+  
+  import-local@^2.0.0:
+    version "2.0.0"
+    resolved "https://registry.yarnpkg.com/import-local/-/import-local-2.0.0.tgz#55070be38a5993cf18ef6db7e961f5bee5c5a09d"
+    integrity sha512-b6s04m3O+s3CGSbqDIyP4R6aAwAeYlVq9+WUWep6iHa8ETRf9yei1U48C5MmfJmV9AiLYYBKPMq/W+/WRpQmCQ==
+    dependencies:
+      pkg-dir "^3.0.0"
+      resolve-cwd "^2.0.0"
+  
+  imurmurhash@^0.1.4:
+    version "0.1.4"
+    resolved "https://registry.yarnpkg.com/imurmurhash/-/imurmurhash-0.1.4.tgz#9218b9b2b928a238b13dc4fb6b6d576f231453ea"
+    integrity sha1-khi5srkoojixPcT7a21XbyMUU+o=
+  
+  indent-string@^4.0.0:
+    version "4.0.0"
+    resolved "https://registry.yarnpkg.com/indent-string/-/indent-string-4.0.0.tgz#624f8f4497d619b2d9768531d58f4122854d7251"
+    integrity sha512-EdDDZu4A2OyIK7Lr/2zG+w5jmbuk1DVBnEwREQvBzspBJkCEbRa8GxU1lghYcaGJCnRWibjDXlq779X1/y5xwg==
+  
+  indexes-of@^1.0.1:
+    version "1.0.1"
+    resolved "https://registry.yarnpkg.com/indexes-of/-/indexes-of-1.0.1.tgz#f30f716c8e2bd346c7b67d3df3915566a7c05607"
+    integrity sha1-8w9xbI4r00bHtn0985FVZqfAVgc=
+  
+  infer-owner@^1.0.3, infer-owner@^1.0.4:
+    version "1.0.4"
+    resolved "https://registry.yarnpkg.com/infer-owner/-/infer-owner-1.0.4.tgz#c4cefcaa8e51051c2a40ba2ce8a3d27295af9467"
+    integrity sha512-IClj+Xz94+d7irH5qRyfJonOdfTzuDaifE6ZPWfx0N0+/ATZCbuTPq2prFl526urkQd90WyUKIh1DfBQ2hMz9A==
+  
+  inflight@^1.0.4:
+    version "1.0.6"
+    resolved "https://registry.yarnpkg.com/inflight/-/inflight-1.0.6.tgz#49bd6331d7d02d0c09bc910a1075ba8165b56df9"
+    integrity sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk=
+    dependencies:
+      once "^1.3.0"
+      wrappy "1"
+  
+  inherits@2, inherits@2.0.4, inherits@^2.0.1, inherits@^2.0.3, inherits@^2.0.4, inherits@~2.0.1, inherits@~2.0.3:
+    version "2.0.4"
+    resolved "https://registry.yarnpkg.com/inherits/-/inherits-2.0.4.tgz#0fa2c64f932917c3433a0ded55363aae37416b7c"
+    integrity sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==
+  
+  inherits@2.0.1:
+    version "2.0.1"
+    resolved "https://registry.yarnpkg.com/inherits/-/inherits-2.0.1.tgz#b17d08d326b4423e568eff719f91b0b1cbdf69f1"
+    integrity sha1-sX0I0ya0Qj5Wjv9xn5GwscvfafE=
+  
+  inherits@2.0.3:
+    version "2.0.3"
+    resolved "https://registry.yarnpkg.com/inherits/-/inherits-2.0.3.tgz#633c2c83e3da42a502f52466022480f4208261de"
+    integrity sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=
+  
+  inquirer@^7.0.0, inquirer@^7.1.0:
+    version "7.3.3"
+    resolved "https://registry.yarnpkg.com/inquirer/-/inquirer-7.3.3.tgz#04d176b2af04afc157a83fd7c100e98ee0aad003"
+    integrity sha512-JG3eIAj5V9CwcGvuOmoo6LB9kbAYT8HXffUl6memuszlwDC/qvFAJw49XJ5NROSFNPxp3iQg1GqkFhaY/CR0IA==
+    dependencies:
+      ansi-escapes "^4.2.1"
+      chalk "^4.1.0"
+      cli-cursor "^3.1.0"
+      cli-width "^3.0.0"
+      external-editor "^3.0.3"
+      figures "^3.0.0"
+      lodash "^4.17.19"
+      mute-stream "0.0.8"
+      run-async "^2.4.0"
+      rxjs "^6.6.0"
+      string-width "^4.1.0"
+      strip-ansi "^6.0.0"
+      through "^2.3.6"
+  
+  internal-ip@^4.3.0:
+    version "4.3.0"
+    resolved "https://registry.yarnpkg.com/internal-ip/-/internal-ip-4.3.0.tgz#845452baad9d2ca3b69c635a137acb9a0dad0907"
+    integrity sha512-S1zBo1D6zcsyuC6PMmY5+55YMILQ9av8lotMx447Bq6SAgo/sDK6y6uUKmuYhW7eacnIhFfsPmCNYdDzsnnDCg==
+    dependencies:
+      default-gateway "^4.2.0"
+      ipaddr.js "^1.9.0"
+  
+  invariant@^2.2.2, invariant@^2.2.4:
+    version "2.2.4"
+    resolved "https://registry.yarnpkg.com/invariant/-/invariant-2.2.4.tgz#610f3c92c9359ce1db616e538008d23ff35158e6"
+    integrity sha512-phJfQVBuaJM5raOpJjSfkiD6BpbCE4Ns//LaXl6wGYtUBY83nWS6Rf9tXm2e8VaK60JEjYldbPif/A2B1C2gNA==
+    dependencies:
+      loose-envify "^1.0.0"
+  
+  ip-regex@^2.1.0:
+    version "2.1.0"
+    resolved "https://registry.yarnpkg.com/ip-regex/-/ip-regex-2.1.0.tgz#fa78bf5d2e6913c911ce9f819ee5146bb6d844e9"
+    integrity sha1-+ni/XS5pE8kRzp+BnuUUa7bYROk=
+  
+  ip@^1.1.0, ip@^1.1.5:
+    version "1.1.5"
+    resolved "https://registry.yarnpkg.com/ip/-/ip-1.1.5.tgz#bdded70114290828c0a039e72ef25f5aaec4354a"
+    integrity sha1-vd7XARQpCCjAoDnnLvJfWq7ENUo=
+  
+  ipaddr.js@1.9.1, ipaddr.js@^1.9.0:
+    version "1.9.1"
+    resolved "https://registry.yarnpkg.com/ipaddr.js/-/ipaddr.js-1.9.1.tgz#bff38543eeb8984825079ff3a2a8e6cbd46781b3"
+    integrity sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==
+  
+  is-absolute-url@^2.0.0:
+    version "2.1.0"
+    resolved "https://registry.yarnpkg.com/is-absolute-url/-/is-absolute-url-2.1.0.tgz#50530dfb84fcc9aa7dbe7852e83a37b93b9f2aa6"
+    integrity sha1-UFMN+4T8yap9vnhS6Do3uTufKqY=
+  
+  is-absolute-url@^3.0.3:
+    version "3.0.3"
+    resolved "https://registry.yarnpkg.com/is-absolute-url/-/is-absolute-url-3.0.3.tgz#96c6a22b6a23929b11ea0afb1836c36ad4a5d698"
+    integrity sha512-opmNIX7uFnS96NtPmhWQgQx6/NYFgsUXYMllcfzwWKUMwfo8kku1TvE6hkNcH+Q1ts5cMVrsY7j0bxXQDciu9Q==
+  
+  is-accessor-descriptor@^0.1.6:
+    version "0.1.6"
+    resolved "https://registry.yarnpkg.com/is-accessor-descriptor/-/is-accessor-descriptor-0.1.6.tgz#a9e12cb3ae8d876727eeef3843f8a0897b5c98d6"
+    integrity sha1-qeEss66Nh2cn7u84Q/igiXtcmNY=
+    dependencies:
+      kind-of "^3.0.2"
+  
+  is-accessor-descriptor@^1.0.0:
+    version "1.0.0"
+    resolved "https://registry.yarnpkg.com/is-accessor-descriptor/-/is-accessor-descriptor-1.0.0.tgz#169c2f6d3df1f992618072365c9b0ea1f6878656"
+    integrity sha512-m5hnHTkcVsPfqx3AKlyttIPb7J+XykHvJP2B9bZDjlhLIoEq4XoK64Vg7boZlVWYK6LUY94dYPEE7Lh0ZkZKcQ==
+    dependencies:
+      kind-of "^6.0.0"
+  
+  is-arguments@^1.0.4:
+    version "1.0.4"
+    resolved "https://registry.yarnpkg.com/is-arguments/-/is-arguments-1.0.4.tgz#3faf966c7cba0ff437fb31f6250082fcf0448cf3"
+    integrity sha512-xPh0Rmt8NE65sNzvyUmWgI1tz3mKq74lGA0mL8LYZcoIzKOzDh6HmrYm3d18k60nHerC8A9Km8kYu87zfSFnLA==
+  
+  is-arrayish@^0.2.1:
+    version "0.2.1"
+    resolved "https://registry.yarnpkg.com/is-arrayish/-/is-arrayish-0.2.1.tgz#77c99840527aa8ecb1a8ba697b80645a7a926a9d"
+    integrity sha1-d8mYQFJ6qOyxqLppe4BkWnqSap0=
+  
+  is-arrayish@^0.3.1:
+    version "0.3.2"
+    resolved "https://registry.yarnpkg.com/is-arrayish/-/is-arrayish-0.3.2.tgz#4574a2ae56f7ab206896fb431eaeed066fdf8f03"
+    integrity sha512-eVRqCvVlZbuw3GrM63ovNSNAeA1K16kaR/LRY/92w0zxQ5/1YzwblUX652i4Xs9RwAGjW9d9y6X88t8OaAJfWQ==
+  
+  is-binary-path@^1.0.0:
+    version "1.0.1"
+    resolved "https://registry.yarnpkg.com/is-binary-path/-/is-binary-path-1.0.1.tgz#75f16642b480f187a711c814161fd3a4a7655898"
+    integrity sha1-dfFmQrSA8YenEcgUFh/TpKdlWJg=
+    dependencies:
+      binary-extensions "^1.0.0"
+  
+  is-binary-path@~2.1.0:
+    version "2.1.0"
+    resolved "https://registry.yarnpkg.com/is-binary-path/-/is-binary-path-2.1.0.tgz#ea1f7f3b80f064236e83470f86c09c254fb45b09"
+    integrity sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==
+    dependencies:
+      binary-extensions "^2.0.0"
+  
+  is-buffer@^1.1.5:
+    version "1.1.6"
+    resolved "https://registry.yarnpkg.com/is-buffer/-/is-buffer-1.1.6.tgz#efaa2ea9daa0d7ab2ea13a97b2b8ad51fefbe8be"
+    integrity sha512-NcdALwpXkTm5Zvvbk7owOUSvVvBKDgKP5/ewfXEznmQFfs4ZRmanOeKBTjRVjka3QFoN6XJ+9F3USqfHqTaU5w==
+  
+  is-callable@^1.1.4, is-callable@^1.2.0:
+    version "1.2.0"
+    resolved "https://registry.yarnpkg.com/is-callable/-/is-callable-1.2.0.tgz#83336560b54a38e35e3a2df7afd0454d691468bb"
+    integrity sha512-pyVD9AaGLxtg6srb2Ng6ynWJqkHU9bEM087AKck0w8QwDarTfNcpIYoU8x8Hv2Icm8u6kFJM18Dag8lyqGkviw==
+  
+  is-ci@^1.0.10:
+    version "1.2.1"
+    resolved "https://registry.yarnpkg.com/is-ci/-/is-ci-1.2.1.tgz#e3779c8ee17fccf428488f6e281187f2e632841c"
+    integrity sha512-s6tfsaQaQi3JNciBH6shVqEDvhGut0SUXr31ag8Pd8BBbVVlcGfWhpPmEOoM6RJ5TFhbypvf5yyRw/VXW1IiWg==
+    dependencies:
+      ci-info "^1.5.0"
+  
+  is-color-stop@^1.0.0:
+    version "1.1.0"
+    resolved "https://registry.yarnpkg.com/is-color-stop/-/is-color-stop-1.1.0.tgz#cfff471aee4dd5c9e158598fbe12967b5cdad345"
+    integrity sha1-z/9HGu5N1cnhWFmPvhKWe1za00U=
+    dependencies:
+      css-color-names "^0.0.4"
+      hex-color-regex "^1.1.0"
+      hsl-regex "^1.0.0"
+      hsla-regex "^1.0.0"
+      rgb-regex "^1.0.1"
+      rgba-regex "^1.0.0"
+  
+  is-data-descriptor@^0.1.4:
+    version "0.1.4"
+    resolved "https://registry.yarnpkg.com/is-data-descriptor/-/is-data-descriptor-0.1.4.tgz#0b5ee648388e2c860282e793f1856fec3f301b56"
+    integrity sha1-C17mSDiOLIYCgueT8YVv7D8wG1Y=
+    dependencies:
+      kind-of "^3.0.2"
+  
+  is-data-descriptor@^1.0.0:
+    version "1.0.0"
+    resolved "https://registry.yarnpkg.com/is-data-descriptor/-/is-data-descriptor-1.0.0.tgz#d84876321d0e7add03990406abbbbd36ba9268c7"
+    integrity sha512-jbRXy1FmtAoCjQkVmIVYwuuqDFUbaOeDjmed1tOGPrsMhtJA4rD9tkgA0F1qJ3gRFRXcHYVkdeaP50Q5rE/jLQ==
+    dependencies:
+      kind-of "^6.0.0"
+  
+  is-date-object@^1.0.1:
+    version "1.0.2"
+    resolved "https://registry.yarnpkg.com/is-date-object/-/is-date-object-1.0.2.tgz#bda736f2cd8fd06d32844e7743bfa7494c3bfd7e"
+    integrity sha512-USlDT524woQ08aoZFzh3/Z6ch9Y/EWXEHQ/AaRN0SkKq4t2Jw2R2339tSXmwuVoY7LLlBCbOIlx2myP/L5zk0g==
+  
+  is-descriptor@^0.1.0:
+    version "0.1.6"
+    resolved "https://registry.yarnpkg.com/is-descriptor/-/is-descriptor-0.1.6.tgz#366d8240dde487ca51823b1ab9f07a10a78251ca"
+    integrity sha512-avDYr0SB3DwO9zsMov0gKCESFYqCnE4hq/4z3TdUlukEy5t9C0YRq7HLrsN52NAcqXKaepeCD0n+B0arnVG3Hg==
+    dependencies:
+      is-accessor-descriptor "^0.1.6"
+      is-data-descriptor "^0.1.4"
+      kind-of "^5.0.0"
+  
+  is-descriptor@^1.0.0, is-descriptor@^1.0.2:
+    version "1.0.2"
+    resolved "https://registry.yarnpkg.com/is-descriptor/-/is-descriptor-1.0.2.tgz#3b159746a66604b04f8c81524ba365c5f14d86ec"
+    integrity sha512-2eis5WqQGV7peooDyLmNEPUrps9+SXX5c9pL3xEB+4e9HnGuDa7mB7kHxHw4CbqS9k1T2hOH3miL8n8WtiYVtg==
+    dependencies:
+      is-accessor-descriptor "^1.0.0"
+      is-data-descriptor "^1.0.0"
+      kind-of "^6.0.2"
+  
+  is-directory@^0.3.1:
+    version "0.3.1"
+    resolved "https://registry.yarnpkg.com/is-directory/-/is-directory-0.3.1.tgz#61339b6f2475fc772fd9c9d83f5c8575dc154ae1"
+    integrity sha1-YTObbyR1/Hcv2cnYP1yFddwVSuE=
+  
+  is-docker@^2.0.0:
+    version "2.1.1"
+    resolved "https://registry.yarnpkg.com/is-docker/-/is-docker-2.1.1.tgz#4125a88e44e450d384e09047ede71adc2d144156"
+    integrity sha512-ZOoqiXfEwtGknTiuDEy8pN2CfE3TxMHprvNer1mXiqwkOT77Rw3YVrUQ52EqAOU3QAWDQ+bQdx7HJzrv7LS2Hw==
+  
+  is-extendable@^0.1.0, is-extendable@^0.1.1:
+    version "0.1.1"
+    resolved "https://registry.yarnpkg.com/is-extendable/-/is-extendable-0.1.1.tgz#62b110e289a471418e3ec36a617d472e301dfc89"
+    integrity sha1-YrEQ4omkcUGOPsNqYX1HLjAd/Ik=
+  
+  is-extendable@^1.0.1:
+    version "1.0.1"
+    resolved "https://registry.yarnpkg.com/is-extendable/-/is-extendable-1.0.1.tgz#a7470f9e426733d81bd81e1155264e3a3507cab4"
+    integrity sha512-arnXMxT1hhoKo9k1LZdmlNyJdDDfy2v0fXjFlmok4+i8ul/6WlbVge9bhM74OpNPQPMGUToDtz+KXa1PneJxOA==
+    dependencies:
+      is-plain-object "^2.0.4"
+  
+  is-extglob@^2.1.0, is-extglob@^2.1.1:
+    version "2.1.1"
+    resolved "https://registry.yarnpkg.com/is-extglob/-/is-extglob-2.1.1.tgz#a88c02535791f02ed37c76a1b9ea9773c833f8c2"
+    integrity sha1-qIwCU1eR8C7TfHahueqXc8gz+MI=
+  
+  is-fullwidth-code-point@^2.0.0:
+    version "2.0.0"
+    resolved "https://registry.yarnpkg.com/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz#a3b30a5c4f199183167aaab93beefae3ddfb654f"
+    integrity sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=
+  
+  is-fullwidth-code-point@^3.0.0:
+    version "3.0.0"
+    resolved "https://registry.yarnpkg.com/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz#f116f8064fe90b3f7844a38997c0b75051269f1d"
+    integrity sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==
+  
+  is-glob@^3.1.0:
+    version "3.1.0"
+    resolved "https://registry.yarnpkg.com/is-glob/-/is-glob-3.1.0.tgz#7ba5ae24217804ac70707b96922567486cc3e84a"
+    integrity sha1-e6WuJCF4BKxwcHuWkiVnSGzD6Eo=
+    dependencies:
+      is-extglob "^2.1.0"
+  
+  is-glob@^4.0.0, is-glob@^4.0.1, is-glob@~4.0.1:
+    version "4.0.1"
+    resolved "https://registry.yarnpkg.com/is-glob/-/is-glob-4.0.1.tgz#7567dbe9f2f5e2467bc77ab83c4a29482407a5dc"
+    integrity sha512-5G0tKtBTFImOqDnLB2hG6Bp2qcKEFduo4tZu9MT/H6NQv/ghhy30o55ufafxJ/LdH79LLs2Kfrn85TLKyA7BUg==
+    dependencies:
+      is-extglob "^2.1.1"
+  
+  is-number@^3.0.0:
+    version "3.0.0"
+    resolved "https://registry.yarnpkg.com/is-number/-/is-number-3.0.0.tgz#24fd6201a4782cf50561c810276afc7d12d71195"
+    integrity sha1-JP1iAaR4LPUFYcgQJ2r8fRLXEZU=
+    dependencies:
+      kind-of "^3.0.2"
+  
+  is-number@^7.0.0:
+    version "7.0.0"
+    resolved "https://registry.yarnpkg.com/is-number/-/is-number-7.0.0.tgz#7535345b896734d5f80c4d06c50955527a14f12b"
+    integrity sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==
+  
+  is-obj@^2.0.0:
+    version "2.0.0"
+    resolved "https://registry.yarnpkg.com/is-obj/-/is-obj-2.0.0.tgz#473fb05d973705e3fd9620545018ca8e22ef4982"
+    integrity sha512-drqDG3cbczxxEJRoOXcOjtdp1J/lyp1mNn0xaznRs8+muBhgQcrnbspox5X5fOw0HnMnbfDzvnEMEtqDEJEo8w==
+  
+  is-path-cwd@^2.0.0:
+    version "2.2.0"
+    resolved "https://registry.yarnpkg.com/is-path-cwd/-/is-path-cwd-2.2.0.tgz#67d43b82664a7b5191fd9119127eb300048a9fdb"
+    integrity sha512-w942bTcih8fdJPJmQHFzkS76NEP8Kzzvmw92cXsazb8intwLqPibPPdXf4ANdKV3rYMuuQYGIWtvz9JilB3NFQ==
+  
+  is-path-in-cwd@^2.0.0:
+    version "2.1.0"
+    resolved "https://registry.yarnpkg.com/is-path-in-cwd/-/is-path-in-cwd-2.1.0.tgz#bfe2dca26c69f397265a4009963602935a053acb"
+    integrity sha512-rNocXHgipO+rvnP6dk3zI20RpOtrAM/kzbB258Uw5BWr3TpXi861yzjo16Dn4hUox07iw5AyeMLHWsujkjzvRQ==
+    dependencies:
+      is-path-inside "^2.1.0"
+  
+  is-path-inside@^2.1.0:
+    version "2.1.0"
+    resolved "https://registry.yarnpkg.com/is-path-inside/-/is-path-inside-2.1.0.tgz#7c9810587d659a40d27bcdb4d5616eab059494b2"
+    integrity sha512-wiyhTzfDWsvwAW53OBWF5zuvaOGlZ6PwYxAbPVDhpm+gM09xKQGjBq/8uYN12aDvMxnAnq3dxTyoSoRNmg5YFg==
+    dependencies:
+      path-is-inside "^1.0.2"
+  
+  is-plain-obj@^1.0.0:
+    version "1.1.0"
+    resolved "https://registry.yarnpkg.com/is-plain-obj/-/is-plain-obj-1.1.0.tgz#71a50c8429dfca773c92a390a4a03b39fcd51d3e"
+    integrity sha1-caUMhCnfync8kqOQpKA7OfzVHT4=
+  
+  is-plain-object@^2.0.3, is-plain-object@^2.0.4:
+    version "2.0.4"
+    resolved "https://registry.yarnpkg.com/is-plain-object/-/is-plain-object-2.0.4.tgz#2c163b3fafb1b606d9d17928f05c2a1c38e07677"
+    integrity sha512-h5PpgXkWitc38BBMYawTYMWJHFZJVnBquFE57xFpjB8pJFiF6gZ+bU+WyI/yqXiFR5mdLsgYNaPe8uao6Uv9Og==
+    dependencies:
+      isobject "^3.0.1"
+  
+  is-regex@^1.0.4, is-regex@^1.1.0:
+    version "1.1.1"
+    resolved "https://registry.yarnpkg.com/is-regex/-/is-regex-1.1.1.tgz#c6f98aacc546f6cec5468a07b7b153ab564a57b9"
+    integrity sha512-1+QkEcxiLlB7VEyFtyBg94e08OAsvq7FUBgApTq/w2ymCLyKJgDPsybBENVtA7XCQEgEXxKPonG+mvYRxh/LIg==
+    dependencies:
+      has-symbols "^1.0.1"
+  
+  is-resolvable@^1.0.0:
+    version "1.1.0"
+    resolved "https://registry.yarnpkg.com/is-resolvable/-/is-resolvable-1.1.0.tgz#fb18f87ce1feb925169c9a407c19318a3206ed88"
+    integrity sha512-qgDYXFSR5WvEfuS5dMj6oTMEbrrSaM0CrFk2Yiq/gXnBvD9pMa2jGXxyhGLfvhZpuMZe18CJpFxAt3CRs42NMg==
+  
+  is-stream@^1.1.0:
+    version "1.1.0"
+    resolved "https://registry.yarnpkg.com/is-stream/-/is-stream-1.1.0.tgz#12d4a3dd4e68e0b79ceb8dbc84173ae80d91ca44"
+    integrity sha1-EtSj3U5o4Lec6428hBc66A2RykQ=
+  
+  is-stream@^2.0.0:
+    version "2.0.0"
+    resolved "https://registry.yarnpkg.com/is-stream/-/is-stream-2.0.0.tgz#bde9c32680d6fae04129d6ac9d921ce7815f78e3"
+    integrity sha512-XCoy+WlUr7d1+Z8GgSuXmpuUFC9fOhRXglJMx+dwLKTkL44Cjd4W1Z5P+BQZpr+cR93aGP4S/s7Ftw6Nd/kiEw==
+  
+  is-svg@^3.0.0:
+    version "3.0.0"
+    resolved "https://registry.yarnpkg.com/is-svg/-/is-svg-3.0.0.tgz#9321dbd29c212e5ca99c4fa9794c714bcafa2f75"
+    integrity sha512-gi4iHK53LR2ujhLVVj+37Ykh9GLqYHX6JOVXbLAucaG/Cqw9xwdFOjDM2qeifLs1sF1npXXFvDu0r5HNgCMrzQ==
+    dependencies:
+      html-comment-regex "^1.1.0"
+  
+  is-symbol@^1.0.2:
+    version "1.0.3"
+    resolved "https://registry.yarnpkg.com/is-symbol/-/is-symbol-1.0.3.tgz#38e1014b9e6329be0de9d24a414fd7441ec61937"
+    integrity sha512-OwijhaRSgqvhm/0ZdAcXNZt9lYdKFpcRDT5ULUuYXPoT794UNOdU+gpT6Rzo7b4V2HUl/op6GqY894AZwv9faQ==
+    dependencies:
+      has-symbols "^1.0.1"
+  
+  is-typedarray@~1.0.0:
+    version "1.0.0"
+    resolved "https://registry.yarnpkg.com/is-typedarray/-/is-typedarray-1.0.0.tgz#e479c80858df0c1b11ddda6940f96011fcda4a9a"
+    integrity sha1-5HnICFjfDBsR3dppQPlgEfzaSpo=
+  
+  is-windows@^1.0.2:
+    version "1.0.2"
+    resolved "https://registry.yarnpkg.com/is-windows/-/is-windows-1.0.2.tgz#d1850eb9791ecd18e6182ce12a30f396634bb19d"
+    integrity sha512-eXK1UInq2bPmjyX6e3VHIzMLobc4J94i4AWn+Hpq3OU5KkrRC96OAcR3PRJ/pGu6m8TRnBHP9dkXQVsT/COVIA==
+  
+  is-wsl@^1.1.0:
+    version "1.1.0"
+    resolved "https://registry.yarnpkg.com/is-wsl/-/is-wsl-1.1.0.tgz#1f16e4aa22b04d1336b66188a66af3c600c3a66d"
+    integrity sha1-HxbkqiKwTRM2tmGIpmrzxgDDpm0=
+  
+  is-wsl@^2.1.1:
+    version "2.2.0"
+    resolved "https://registry.yarnpkg.com/is-wsl/-/is-wsl-2.2.0.tgz#74a4c76e77ca9fd3f932f290c17ea326cd157271"
+    integrity sha512-fKzAra0rGJUUBwGBgNkHZuToZcn+TtXHpeCgmkMJMMYx1sQDYaCSyjJBSCa2nH1DGm7s3n1oBnohoVTBaN7Lww==
+    dependencies:
+      is-docker "^2.0.0"
+  
+  isarray@1.0.0, isarray@^1.0.0, isarray@~1.0.0:
+    version "1.0.0"
+    resolved "https://registry.yarnpkg.com/isarray/-/isarray-1.0.0.tgz#bb935d48582cba168c06834957a54a3e07124f11"
+    integrity sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE=
+  
+  isexe@^2.0.0:
+    version "2.0.0"
+    resolved "https://registry.yarnpkg.com/isexe/-/isexe-2.0.0.tgz#e8fbf374dc556ff8947a10dcb0572d633f2cfa10"
+    integrity sha1-6PvzdNxVb/iUehDcsFctYz8s+hA=
+  
+  isobject@^2.0.0:
+    version "2.1.0"
+    resolved "https://registry.yarnpkg.com/isobject/-/isobject-2.1.0.tgz#f065561096a3f1da2ef46272f815c840d87e0c89"
+    integrity sha1-8GVWEJaj8dou9GJy+BXIQNh+DIk=
+    dependencies:
+      isarray "1.0.0"
+  
+  isobject@^3.0.0, isobject@^3.0.1:
+    version "3.0.1"
+    resolved "https://registry.yarnpkg.com/isobject/-/isobject-3.0.1.tgz#4e431e92b11a9731636aa1f9c8d1ccbcfdab78df"
+    integrity sha1-TkMekrEalzFjaqH5yNHMvP2reN8=
+  
+  isstream@~0.1.2:
+    version "0.1.2"
+    resolved "https://registry.yarnpkg.com/isstream/-/isstream-0.1.2.tgz#47e63f7af55afa6f92e1500e690eb8b8529c099a"
+    integrity sha1-R+Y/evVa+m+S4VAOaQ64uFKcCZo=
+  
+  javascript-stringify@^2.0.1:
+    version "2.0.1"
+    resolved "https://registry.yarnpkg.com/javascript-stringify/-/javascript-stringify-2.0.1.tgz#6ef358035310e35d667c675ed63d3eb7c1aa19e5"
+    integrity sha512-yV+gqbd5vaOYjqlbk16EG89xB5udgjqQF3C5FAORDg4f/IS1Yc5ERCv5e/57yBcfJYw05V5JyIXabhwb75Xxow==
+  
+  jest-worker@^25.4.0:
+    version "25.5.0"
+    resolved "https://registry.yarnpkg.com/jest-worker/-/jest-worker-25.5.0.tgz#2611d071b79cea0f43ee57a3d118593ac1547db1"
+    integrity sha512-/dsSmUkIy5EBGfv/IjjqmFxrNAUpBERfGs1oHROyD7yxjG/w+t0GOJDX8O1k32ySmd7+a5IhnJU2qQFcJ4n1vw==
+    dependencies:
+      merge-stream "^2.0.0"
+      supports-color "^7.0.0"
+  
+  js-message@1.0.5:
+    version "1.0.5"
+    resolved "https://registry.yarnpkg.com/js-message/-/js-message-1.0.5.tgz#2300d24b1af08e89dd095bc1a4c9c9cfcb892d15"
+    integrity sha1-IwDSSxrwjondCVvBpMnJz8uJLRU=
+  
+  js-queue@2.0.0:
+    version "2.0.0"
+    resolved "https://registry.yarnpkg.com/js-queue/-/js-queue-2.0.0.tgz#362213cf860f468f0125fc6c96abc1742531f948"
+    integrity sha1-NiITz4YPRo8BJfxslqvBdCUx+Ug=
+    dependencies:
+      easy-stack "^1.0.0"
+  
+  "js-tokens@^3.0.0 || ^4.0.0", js-tokens@^4.0.0:
+    version "4.0.0"
+    resolved "https://registry.yarnpkg.com/js-tokens/-/js-tokens-4.0.0.tgz#19203fb59991df98e3a287050d4647cdeaf32499"
+    integrity sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==
+  
+  js-yaml@^3.13.1:
+    version "3.14.0"
+    resolved "https://registry.yarnpkg.com/js-yaml/-/js-yaml-3.14.0.tgz#a7a34170f26a21bb162424d8adacb4113a69e482"
+    integrity sha512-/4IbIeHcD9VMHFqDR/gQ7EdZdLimOvW2DdcxFjdyyZ9NsbS+ccrXqVWDtab/lRl5AlUqmpBx8EhPaWR+OtY17A==
+    dependencies:
+      argparse "^1.0.7"
+      esprima "^4.0.0"
+  
+  jsbn@~0.1.0:
+    version "0.1.1"
+    resolved "https://registry.yarnpkg.com/jsbn/-/jsbn-0.1.1.tgz#a5e654c2e5a2deb5f201d96cefbca80c0ef2f513"
+    integrity sha1-peZUwuWi3rXyAdls77yoDA7y9RM=
+  
+  jsesc@^2.5.1:
+    version "2.5.2"
+    resolved "https://registry.yarnpkg.com/jsesc/-/jsesc-2.5.2.tgz#80564d2e483dacf6e8ef209650a67df3f0c283a4"
+    integrity sha512-OYu7XEzjkCQ3C5Ps3QIZsQfNpqoJyZZA99wd9aWd05NCtC5pWOkShK2mkL6HXQR6/Cy2lbNdPlZBpuQHXE63gA==
+  
+  jsesc@~0.5.0:
+    version "0.5.0"
+    resolved "https://registry.yarnpkg.com/jsesc/-/jsesc-0.5.0.tgz#e7dee66e35d6fc16f710fe91d5cf69f70f08911d"
+    integrity sha1-597mbjXW/Bb3EP6R1c9p9w8IkR0=
+  
+  json-parse-better-errors@^1.0.1, json-parse-better-errors@^1.0.2:
+    version "1.0.2"
+    resolved "https://registry.yarnpkg.com/json-parse-better-errors/-/json-parse-better-errors-1.0.2.tgz#bb867cfb3450e69107c131d1c514bab3dc8bcaa9"
+    integrity sha512-mrqyZKfX5EhL7hvqcV6WG1yYjnjeuYDzDhhcAAUrq8Po85NBQBJP+ZDUT75qZQ98IkUoBqdkExkukOU7Ts2wrw==
+  
+  json-parse-even-better-errors@^2.3.0:
+    version "2.3.1"
+    resolved "https://registry.yarnpkg.com/json-parse-even-better-errors/-/json-parse-even-better-errors-2.3.1.tgz#7c47805a94319928e05777405dc12e1f7a4ee02d"
+    integrity sha512-xyFwyhro/JEof6Ghe2iz2NcXoj2sloNsWr/XsERDK/oiPCfaNhl5ONfp+jQdAZRQQ0IJWNzH9zIZF7li91kh2w==
+  
+  json-schema-traverse@^0.4.1:
+    version "0.4.1"
+    resolved "https://registry.yarnpkg.com/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz#69f6a87d9513ab8bb8fe63bdb0979c448e684660"
+    integrity sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==
+  
+  json-schema@0.2.3:
+    version "0.2.3"
+    resolved "https://registry.yarnpkg.com/json-schema/-/json-schema-0.2.3.tgz#b480c892e59a2f05954ce727bd3f2a4e882f9e13"
+    integrity sha1-tIDIkuWaLwWVTOcnvT8qTogvnhM=
+  
+  json-stable-stringify-without-jsonify@^1.0.1:
+    version "1.0.1"
+    resolved "https://registry.yarnpkg.com/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz#9db7b59496ad3f3cfef30a75142d2d930ad72651"
+    integrity sha1-nbe1lJatPzz+8wp1FC0tkwrXJlE=
+  
+  json-stringify-safe@~5.0.1:
+    version "5.0.1"
+    resolved "https://registry.yarnpkg.com/json-stringify-safe/-/json-stringify-safe-5.0.1.tgz#1296a2d58fd45f19a0f6ce01d65701e2c735b6eb"
+    integrity sha1-Epai1Y/UXxmg9s4B1lcB4sc1tus=
+  
+  json3@^3.3.2:
+    version "3.3.3"
+    resolved "https://registry.yarnpkg.com/json3/-/json3-3.3.3.tgz#7fc10e375fc5ae42c4705a5cc0aa6f62be305b81"
+    integrity sha512-c7/8mbUsKigAbLkD5B010BK4D9LZm7A1pNItkEwiUZRpIN66exu/e7YQWysGun+TRKaJp8MhemM+VkfWv42aCA==
+  
+  json5@^0.5.0:
+    version "0.5.1"
+    resolved "https://registry.yarnpkg.com/json5/-/json5-0.5.1.tgz#1eade7acc012034ad84e2396767ead9fa5495821"
+    integrity sha1-Hq3nrMASA0rYTiOWdn6tn6VJWCE=
+  
+  json5@^1.0.1:
+    version "1.0.1"
+    resolved "https://registry.yarnpkg.com/json5/-/json5-1.0.1.tgz#779fb0018604fa854eacbf6252180d83543e3dbe"
+    integrity sha512-aKS4WQjPenRxiQsC93MNfjx+nbF4PAdYzmd/1JIj8HYzqfbu86beTuNgXDzPknWk0n0uARlyewZo4s++ES36Ow==
+    dependencies:
+      minimist "^1.2.0"
+  
+  json5@^2.1.2:
+    version "2.1.3"
+    resolved "https://registry.yarnpkg.com/json5/-/json5-2.1.3.tgz#c9b0f7fa9233bfe5807fe66fcf3a5617ed597d43"
+    integrity sha512-KXPvOm8K9IJKFM0bmdn8QXh7udDh1g/giieX0NLCaMnb4hEiVFqnop2ImTXCc5e0/oHz3LTqmHGtExn5hfMkOA==
+    dependencies:
+      minimist "^1.2.5"
+  
+  jsonfile@^4.0.0:
+    version "4.0.0"
+    resolved "https://registry.yarnpkg.com/jsonfile/-/jsonfile-4.0.0.tgz#8771aae0799b64076b76640fca058f9c10e33ecb"
+    integrity sha1-h3Gq4HmbZAdrdmQPygWPnBDjPss=
+    optionalDependencies:
+      graceful-fs "^4.1.6"
+  
+  jsprim@^1.2.2:
+    version "1.4.1"
+    resolved "https://registry.yarnpkg.com/jsprim/-/jsprim-1.4.1.tgz#313e66bc1e5cc06e438bc1b7499c2e5c56acb6a2"
+    integrity sha1-MT5mvB5cwG5Di8G3SZwuXFastqI=
+    dependencies:
+      assert-plus "1.0.0"
+      extsprintf "1.3.0"
+      json-schema "0.2.3"
+      verror "1.10.0"
+  
+  killable@^1.0.1:
+    version "1.0.1"
+    resolved "https://registry.yarnpkg.com/killable/-/killable-1.0.1.tgz#4c8ce441187a061c7474fb87ca08e2a638194892"
+    integrity sha512-LzqtLKlUwirEUyl/nicirVmNiPvYs7l5n8wOPP7fyJVpUPkvCnW/vuiXGpylGUlnPDnB7311rARzAt3Mhswpjg==
+  
+  kind-of@^3.0.2, kind-of@^3.0.3, kind-of@^3.2.0:
+    version "3.2.2"
+    resolved "https://registry.yarnpkg.com/kind-of/-/kind-of-3.2.2.tgz#31ea21a734bab9bbb0f32466d893aea51e4a3c64"
+    integrity sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=
+    dependencies:
+      is-buffer "^1.1.5"
+  
+  kind-of@^4.0.0:
+    version "4.0.0"
+    resolved "https://registry.yarnpkg.com/kind-of/-/kind-of-4.0.0.tgz#20813df3d712928b207378691a45066fae72dd57"
+    integrity sha1-IIE989cSkosgc3hpGkUGb65y3Vc=
+    dependencies:
+      is-buffer "^1.1.5"
+  
+  kind-of@^5.0.0:
+    version "5.1.0"
+    resolved "https://registry.yarnpkg.com/kind-of/-/kind-of-5.1.0.tgz#729c91e2d857b7a419a1f9aa65685c4c33f5845d"
+    integrity sha512-NGEErnH6F2vUuXDh+OlbcKW7/wOcfdRHaZ7VWtqCztfHri/++YKmP51OdWeGPuqCOba6kk2OTe5d02VmTB80Pw==
+  
+  kind-of@^6.0.0, kind-of@^6.0.2:
+    version "6.0.3"
+    resolved "https://registry.yarnpkg.com/kind-of/-/kind-of-6.0.3.tgz#07c05034a6c349fa06e24fa35aa76db4580ce4dd"
+    integrity sha512-dcS1ul+9tmeD95T+x28/ehLgd9mENa3LsvDTtzm3vyBEO7RPptvAD+t44WVXaUjTBRcrpFeFlC8WCruUR456hw==
+  
+  launch-editor-middleware@^2.2.1:
+    version "2.2.1"
+    resolved "https://registry.yarnpkg.com/launch-editor-middleware/-/launch-editor-middleware-2.2.1.tgz#e14b07e6c7154b0a4b86a0fd345784e45804c157"
+    integrity sha512-s0UO2/gEGiCgei3/2UN3SMuUj1phjQN8lcpnvgLSz26fAzNWPQ6Nf/kF5IFClnfU2ehp6LrmKdMU/beveO+2jg==
+    dependencies:
+      launch-editor "^2.2.1"
+  
+  launch-editor@^2.2.1:
+    version "2.2.1"
+    resolved "https://registry.yarnpkg.com/launch-editor/-/launch-editor-2.2.1.tgz#871b5a3ee39d6680fcc26d37930b6eeda89db0ca"
+    integrity sha512-On+V7K2uZK6wK7x691ycSUbLD/FyKKelArkbaAMSSJU8JmqmhwN2+mnJDNINuJWSrh2L0kDk+ZQtbC/gOWUwLw==
+    dependencies:
+      chalk "^2.3.0"
+      shell-quote "^1.6.1"
+  
+  leven@^3.1.0:
+    version "3.1.0"
+    resolved "https://registry.yarnpkg.com/leven/-/leven-3.1.0.tgz#77891de834064cccba82ae7842bb6b14a13ed7f2"
+    integrity sha512-qsda+H8jTaUaN/x5vzW2rzc+8Rw4TAQ/4KjB46IwK5VH+IlVeeeje/EoZRpiXvIqjFgK84QffqPztGI3VBLG1A==
+  
+  levenary@^1.1.1:
+    version "1.1.1"
+    resolved "https://registry.yarnpkg.com/levenary/-/levenary-1.1.1.tgz#842a9ee98d2075aa7faeedbe32679e9205f46f77"
+    integrity sha512-mkAdOIt79FD6irqjYSs4rdbnlT5vRonMEvBVPVb3XmevfS8kgRXwfes0dhPdEtzTWD/1eNE/Bm/G1iRt6DcnQQ==
+    dependencies:
+      leven "^3.1.0"
+  
+  levn@^0.3.0, levn@~0.3.0:
+    version "0.3.0"
+    resolved "https://registry.yarnpkg.com/levn/-/levn-0.3.0.tgz#3b09924edf9f083c0490fdd4c0bc4421e04764ee"
+    integrity sha1-OwmSTt+fCDwEkP3UwLxEIeBHZO4=
+    dependencies:
+      prelude-ls "~1.1.2"
+      type-check "~0.3.2"
+  
+  lines-and-columns@^1.1.6:
+    version "1.1.6"
+    resolved "https://registry.yarnpkg.com/lines-and-columns/-/lines-and-columns-1.1.6.tgz#1c00c743b433cd0a4e80758f7b64a57440d9ff00"
+    integrity sha1-HADHQ7QzzQpOgHWPe2SldEDZ/wA=
+  
+  loader-fs-cache@^1.0.0:
+    version "1.0.3"
+    resolved "https://registry.yarnpkg.com/loader-fs-cache/-/loader-fs-cache-1.0.3.tgz#f08657646d607078be2f0a032f8bd69dd6f277d9"
+    integrity sha512-ldcgZpjNJj71n+2Mf6yetz+c9bM4xpKtNds4LbqXzU/PTdeAX0g3ytnU1AJMEcTk2Lex4Smpe3Q/eCTsvUBxbA==
+    dependencies:
+      find-cache-dir "^0.1.1"
+      mkdirp "^0.5.1"
+  
+  loader-runner@^2.3.1, loader-runner@^2.4.0:
+    version "2.4.0"
+    resolved "https://registry.yarnpkg.com/loader-runner/-/loader-runner-2.4.0.tgz#ed47066bfe534d7e84c4c7b9998c2a75607d9357"
+    integrity sha512-Jsmr89RcXGIwivFY21FcRrisYZfvLMTWx5kOLc+JTxtpBOG6xML0vzbc6SEQG2FO9/4Fc3wW4LVcB5DmGflaRw==
+  
+  loader-utils@^0.2.16:
+    version "0.2.17"
+    resolved "https://registry.yarnpkg.com/loader-utils/-/loader-utils-0.2.17.tgz#f86e6374d43205a6e6c60e9196f17c0299bfb348"
+    integrity sha1-+G5jdNQyBabmxg6RlvF8Apm/s0g=
+    dependencies:
+      big.js "^3.1.3"
+      emojis-list "^2.0.0"
+      json5 "^0.5.0"
+      object-assign "^4.0.1"
+  
+  loader-utils@^1.0.2, loader-utils@^1.1.0, loader-utils@^1.2.3, loader-utils@^1.4.0:
+    version "1.4.0"
+    resolved "https://registry.yarnpkg.com/loader-utils/-/loader-utils-1.4.0.tgz#c579b5e34cb34b1a74edc6c1fb36bfa371d5a613"
+    integrity sha512-qH0WSMBtn/oHuwjy/NucEgbx5dbxxnxup9s4PVXJUDHZBQY+s0NWA9rJf53RBnQZxfch7euUui7hpoAPvALZdA==
+    dependencies:
+      big.js "^5.2.2"
+      emojis-list "^3.0.0"
+      json5 "^1.0.1"
+  
+  locate-path@^3.0.0:
+    version "3.0.0"
+    resolved "https://registry.yarnpkg.com/locate-path/-/locate-path-3.0.0.tgz#dbec3b3ab759758071b58fe59fc41871af21400e"
+    integrity sha512-7AO748wWnIhNqAuaty2ZWHkQHRSNfPVIsPIfwEOWO22AmaoVrWavlOcMR5nzTLNYvp36X220/maaRsrec1G65A==
+    dependencies:
+      p-locate "^3.0.0"
+      path-exists "^3.0.0"
+  
+  locate-path@^5.0.0:
+    version "5.0.0"
+    resolved "https://registry.yarnpkg.com/locate-path/-/locate-path-5.0.0.tgz#1afba396afd676a6d42504d0a67a3a7eb9f62aa0"
+    integrity sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==
+    dependencies:
+      p-locate "^4.1.0"
+  
+  lodash.defaultsdeep@^4.6.1:
+    version "4.6.1"
+    resolved "https://registry.yarnpkg.com/lodash.defaultsdeep/-/lodash.defaultsdeep-4.6.1.tgz#512e9bd721d272d94e3d3a63653fa17516741ca6"
+    integrity sha512-3j8wdDzYuWO3lM3Reg03MuQR957t287Rpcxp1njpEa8oDrikb+FwGdW3n+FELh/A6qib6yPit0j/pv9G/yeAqA==
+