From b795ca8443c8784312e939a8967933df4c5a4b6f Mon Sep 17 00:00:00 2001 From: Neil MacDougall Date: Mon, 15 Jul 2019 14:45:47 +0100 Subject: [PATCH 01/40] Cloud Console WIP --- .../kube-console/kube-console.component.html | 18 + .../kube-console/kube-console.component.scss | 0 .../kube-console.component.spec.ts | 41 ++ .../kube-console/kube-console.component.ts | 125 ++++ .../custom/kubernetes/kubernetes.module.ts | 4 +- .../custom/kubernetes/kubernetes.routing.ts | 8 + .../kubernetes-summary.component.html | 4 + .../kubernetes-summary.component.ts | 3 + deploy/Dockerfile.kubeconsole.yml | 18 + deploy/ci/build-aio-image-canary.yml | 1 + deploy/kubeconsole.bashrc | 42 ++ .../ssh-viewer/ssh-viewer.component.html | 9 +- .../ssh-viewer/ssh-viewer.component.scss | 2 + .../ssh-viewer/ssh-viewer.component.ts | 25 +- .../plugins/kubernetes/endpoint_config.go | 68 ++ src/jetstream/plugins/kubernetes/go.mod | 1 + src/jetstream/plugins/kubernetes/go.sum | 4 +- .../plugins/kubernetes/kube_console.go | 582 ++++++++++++++++++ src/jetstream/plugins/kubernetes/main.go | 3 + 19 files changed, 952 insertions(+), 6 deletions(-) create mode 100644 custom-src/frontend/app/custom/kubernetes/kube-console/kube-console.component.html create mode 100644 custom-src/frontend/app/custom/kubernetes/kube-console/kube-console.component.scss create mode 100644 custom-src/frontend/app/custom/kubernetes/kube-console/kube-console.component.spec.ts create mode 100644 custom-src/frontend/app/custom/kubernetes/kube-console/kube-console.component.ts create mode 100644 deploy/Dockerfile.kubeconsole.yml create mode 100644 deploy/kubeconsole.bashrc create mode 100644 src/jetstream/plugins/kubernetes/kube_console.go diff --git a/custom-src/frontend/app/custom/kubernetes/kube-console/kube-console.component.html b/custom-src/frontend/app/custom/kubernetes/kube-console/kube-console.component.html new file mode 100644 index 0000000000..e8cfda74c0 --- /dev/null +++ b/custom-src/frontend/app/custom/kubernetes/kube-console/kube-console.component.html @@ -0,0 +1,18 @@ + +

Kubernetes Console

+
+ + + + + +
+
+ + diff --git a/custom-src/frontend/app/custom/kubernetes/kube-console/kube-console.component.scss b/custom-src/frontend/app/custom/kubernetes/kube-console/kube-console.component.scss new file mode 100644 index 0000000000..e69de29bb2 diff --git a/custom-src/frontend/app/custom/kubernetes/kube-console/kube-console.component.spec.ts b/custom-src/frontend/app/custom/kubernetes/kube-console/kube-console.component.spec.ts new file mode 100644 index 0000000000..030ec22e95 --- /dev/null +++ b/custom-src/frontend/app/custom/kubernetes/kube-console/kube-console.component.spec.ts @@ -0,0 +1,41 @@ +import { async, ComponentFixture, TestBed } from '@angular/core/testing'; +import { RouterTestingModule } from '@angular/router/testing'; + +import { TabNavService } from '../../../../tab-nav.service'; +import { ApplicationServiceMock } from '../../../../test-framework/application-service-helper'; +import { createBasicStoreModule } from '../../../../test-framework/store-test-helper'; +import { CoreModule } from '../../../core/core.module'; +import { SharedModule } from '../../../shared/shared.module'; +import { KubeConsoleComponent } from './kube-console.component'; + +describe('KubeConsoleComponent', () => { + let component: KubeConsoleComponent; + let fixture: ComponentFixture; + + beforeEach(async(() => { + TestBed.configureTestingModule({ + declarations: [KubeConsoleComponent], + imports: [ + CoreModule, + SharedModule, + RouterTestingModule, + createBasicStoreModule() + ], + providers: [ + { provide: ApplicationService, useClass: ApplicationServiceMock }, + TabNavService + ], + }) + .compileComponents(); + })); + + beforeEach(() => { + fixture = TestBed.createComponent(KubeConsoleComponent); + component = fixture.componentInstance; + fixture.detectChanges(); + }); + + it('should create', () => { + expect(component).toBeTruthy(); + }); +}); diff --git a/custom-src/frontend/app/custom/kubernetes/kube-console/kube-console.component.ts b/custom-src/frontend/app/custom/kubernetes/kube-console/kube-console.component.ts new file mode 100644 index 0000000000..32481eb139 --- /dev/null +++ b/custom-src/frontend/app/custom/kubernetes/kube-console/kube-console.component.ts @@ -0,0 +1,125 @@ +import { Component, OnInit, ViewChild } from '@angular/core'; +import { ActivatedRoute } from '@angular/router'; +import { Store } from '@ngrx/store'; +import { NEVER, Observable, Subject, Subscription } from 'rxjs'; +import websocketConnect from 'rxjs-websockets'; +import { catchError, first, map } from 'rxjs/operators'; + +import { AppState } from '../../../../../store/src/app-state'; +import { IApp } from '../../../core/cf-api.types'; +import { IHeaderBreadcrumb } from '../../../shared/components/page-header/page-header.types'; +import { SshViewerComponent } from '../../../shared/components/ssh-viewer/ssh-viewer.component'; +import { KubernetesEndpointService } from '../services/kubernetes-endpoint.service'; +import { BaseKubeGuid } from '../kubernetes-page.types'; +import { KubernetesService } from '../services/kubernetes.service'; + + +@Component({ + selector: 'app-kube-console', + templateUrl: './kube-console.component.html', + styleUrls: ['./kube-console.component.scss'], + providers: [ + { + provide: BaseKubeGuid, + useFactory: (activatedRoute: ActivatedRoute) => { + return { + guid: activatedRoute.snapshot.params.endpointId + }; + }, + deps: [ + ActivatedRoute + ] + }, + KubernetesService, + KubernetesEndpointService, + ] +}) +export class KubeConsoleComponent implements OnInit { + + public messages: Observable; + + public connectionStatus: Observable; + + public sshInput: Subject; + + public errorMessage: string; + + public sshRoute: string; + + public connected: boolean; + + public kubeSummaryLink: string; + + private connection: Subscription; + + public instanceId: string; + + public breadcrumbs$: Observable; + + @ViewChild('sshViewer') sshViewer: SshViewerComponent; + + private getBreadcrumbs( + application: IApp, + ) { + return [ + { + breadcrumbs: [ + { value: application.name, routerLink: `/kubernetes/${application.cfGuid}/${application.guid}/instances` } + ] + }, + ]; + } + + constructor( + public kubeEndpointService: KubernetesEndpointService, + // private activatedRoute: ActivatedRoute, + // private store: Store, + ) { } + + ngOnInit() { + + const guid = this.kubeEndpointService.baseKube.guid; + + console.log('HERE'); + console.log(guid); + // const routeParams = this.activatedRoute.snapshot.params; + // this.instanceId = routeParams.index; + + this.kubeSummaryLink = ( + `/kubernetes/${guid}/summary` + ); + + if (!guid) { + this.messages = NEVER; + this.connectionStatus = NEVER; + } else { + const host = window.location.host; + const protocol = window.location.protocol === 'https:' ? 'wss' : 'ws'; + const streamUrl = ( + `${protocol}://${host}/pp/v1/kubeconsole/${guid}` + ); + this.sshInput = new Subject(); + const connection = websocketConnect( + streamUrl, + this.sshInput + ); + + console.log(streamUrl); + + this.messages = connection.messages.pipe( + catchError(e => { + if (e.type === 'error') { + this.errorMessage = 'Error connecting to web socket'; + } + return []; + })); + + this.connectionStatus = connection.connectionStatus; + + // this.breadcrumbs$ = this.applicationService.waitForAppEntity$.pipe( + // map(app => this.getBreadcrumbs(app.entity.entity)), + // first() + // ); + } + } +} diff --git a/custom-src/frontend/app/custom/kubernetes/kubernetes.module.ts b/custom-src/frontend/app/custom/kubernetes/kubernetes.module.ts index 5eb62e9208..13a2ac8495 100644 --- a/custom-src/frontend/app/custom/kubernetes/kubernetes.module.ts +++ b/custom-src/frontend/app/custom/kubernetes/kubernetes.module.ts @@ -91,6 +91,7 @@ import { KubernetesNamespacesTabComponent } from './tabs/kubernetes-namespaces-t import { KubernetesNodesTabComponent } from './tabs/kubernetes-nodes-tab/kubernetes-nodes-tab.component'; import { KubernetesPodsTabComponent } from './tabs/kubernetes-pods-tab/kubernetes-pods-tab.component'; import { KubernetesSummaryTabComponent } from './tabs/kubernetes-summary-tab/kubernetes-summary.component'; +import { KubeConsoleComponent } from './kube-console/kube-console.component'; /* tslint:enable */ @@ -145,7 +146,8 @@ import { KubernetesSummaryTabComponent } from './tabs/kubernetes-summary-tab/kub KubeNamespacePodCountComponent, PodNameLinkComponent, NodePodCountComponent, - KubernetesServicePortsComponent + KubernetesServicePortsComponent, + KubeConsoleComponent, ], exports: [ KubernetesServicePortsComponent diff --git a/custom-src/frontend/app/custom/kubernetes/kubernetes.routing.ts b/custom-src/frontend/app/custom/kubernetes/kubernetes.routing.ts index f2d3839458..1ab7fe2376 100644 --- a/custom-src/frontend/app/custom/kubernetes/kubernetes.routing.ts +++ b/custom-src/frontend/app/custom/kubernetes/kubernetes.routing.ts @@ -1,3 +1,4 @@ +import { KubeConsoleComponent } from './kube-console/kube-console.component'; import { NgModule } from '@angular/core'; import { RouterModule, Routes } from '@angular/router'; @@ -155,6 +156,13 @@ const kubernetes: Routes = [{ } } ] +}, +{ + path: ':endpointId/console', + component: KubeConsoleComponent, + data: { + uiNoMargin: true + } } ]; diff --git a/custom-src/frontend/app/custom/kubernetes/tabs/kubernetes-summary-tab/kubernetes-summary.component.html b/custom-src/frontend/app/custom/kubernetes/tabs/kubernetes-summary-tab/kubernetes-summary.component.html index 7198ba3c46..dc35b490cb 100644 --- a/custom-src/frontend/app/custom/kubernetes/tabs/kubernetes-summary-tab/kubernetes-summary.component.html +++ b/custom-src/frontend/app/custom/kubernetes/tabs/kubernetes-summary-tab/kubernetes-summary.component.html @@ -4,6 +4,10 @@ dashboard View Dashboard + diff --git a/custom-src/frontend/app/custom/kubernetes/tabs/kubernetes-summary-tab/kubernetes-summary.component.ts b/custom-src/frontend/app/custom/kubernetes/tabs/kubernetes-summary-tab/kubernetes-summary.component.ts index d542073359..c1575fd95c 100644 --- a/custom-src/frontend/app/custom/kubernetes/tabs/kubernetes-summary-tab/kubernetes-summary.component.ts +++ b/custom-src/frontend/app/custom/kubernetes/tabs/kubernetes-summary-tab/kubernetes-summary.component.ts @@ -60,6 +60,8 @@ export class KubernetesSummaryTabComponent implements OnInit, OnDestroy { source: SafeResourceUrl; dashboardLink: string; + kubeConsoleLink: string; + public podCapacity$: Observable; public diskPressure$: Observable; public memoryPressure$: Observable; @@ -240,6 +242,7 @@ export class KubernetesSummaryTabComponent implements OnInit, OnDestroy { warningText: `Nodes with unknown ready status found` }); this.dashboardLink = `/kubernetes/${guid}/dashboard`; + this.kubeConsoleLink = `/kubernetes/${guid}/console`; this.kubeNodeVersions$ = this.getNodeKubeVersions(nodes$).pipe(startWith('-')); diff --git a/deploy/Dockerfile.kubeconsole.yml b/deploy/Dockerfile.kubeconsole.yml new file mode 100644 index 0000000000..48c3f21443 --- /dev/null +++ b/deploy/Dockerfile.kubeconsole.yml @@ -0,0 +1,18 @@ +FROM splatform/stratos-bk-base:opensuse as prod-build +USER root +RUN zypper in -y wget tar unzip nano vim curl + +RUN curl -LO https://storage.googleapis.com/kubernetes-release/release/$(curl -s https://storage.googleapis.com/kubernetes-release/release/stable.txt)/bin/linux/amd64/kubectl && \ + chmod +x ./kubectl && \ + mv ./kubectl /usr/local/bin/kubectl + +# Install Helm + +RUN curl -LO https://git.io/get_helm.sh && \ + chmod 700 get_helm.sh && \ + ./get_helm.sh + +ADD ./kubeconsole.bashrc /root/.bashrc +WORKDIR /root + +CMD exec /bin/bash -c "trap : TERM INT; sleep infinity & wait" diff --git a/deploy/ci/build-aio-image-canary.yml b/deploy/ci/build-aio-image-canary.yml index 83df27b500..c8bc0485a7 100644 --- a/deploy/ci/build-aio-image-canary.yml +++ b/deploy/ci/build-aio-image-canary.yml @@ -75,3 +75,4 @@ jobs: tag: "stratos/deploy/ci/tasks/build-images/canary-tag" tag_as_latest: false labels_file: image-tag/image-labels + squash: true diff --git a/deploy/kubeconsole.bashrc b/deploy/kubeconsole.bashrc new file mode 100644 index 0000000000..2ac7e17679 --- /dev/null +++ b/deploy/kubeconsole.bashrc @@ -0,0 +1,42 @@ + +CYAN="\033[96m" +YELLOW="\033[93m" +GREEN="\033[92m" +RESET="\033[0m" +BOLD="\033[1m" +DIM="\033[2m" + +echo -e "${BOLD}${GREEN}SUSE Cloud Application Platform${RESET}" +echo "" +echo -e "${CYAN}Kubernetes Console${RESET}" +echo "" + +export KUBECONFIG=/root/.stratos/kubeconfig +export PS1="\033[92mstratos>\033[0m" +alias k=kubectl + +helm init --client-only > /dev/null +helm repo remove local > /dev/null +helm repo remove stable > /dev/null + +if [ -f "/root/.stratos/helm-setup" ]; then + echo "Setting up Helm repositories ..." + source "/root/.stratos/helm-setup" > /dev/null + helm repo update 2>&1 > /dev/null + echo "" +fi + + +if [ -f "/root/.stratos/history" ]; then + cat /root/.stratos/history > /root/.bash_history +fi + +# Make Bash append rather than overwrite the history on disk: +shopt -s histappend +# A new shell gets the history lines from all previous shells +PROMPT_COMMAND='history -a' +# Don't put duplicate lines in the history. +export HISTCONTROL=ignoredups + +echo "Ready" +echo "" diff --git a/src/frontend/packages/core/src/shared/components/ssh-viewer/ssh-viewer.component.html b/src/frontend/packages/core/src/shared/components/ssh-viewer/ssh-viewer.component.html index 3d71c6e1a8..44cac219ee 100644 --- a/src/frontend/packages/core/src/shared/components/ssh-viewer/ssh-viewer.component.html +++ b/src/frontend/packages/core/src/shared/components/ssh-viewer/ssh-viewer.component.html @@ -1,12 +1,17 @@
-
+
Error occurred establishing SSH connection
Disconnected -
+
+
+
+ {{ message }} +
+
diff --git a/src/frontend/packages/core/src/shared/components/ssh-viewer/ssh-viewer.component.scss b/src/frontend/packages/core/src/shared/components/ssh-viewer/ssh-viewer.component.scss index 000e0340e2..3adb5a5af4 100644 --- a/src/frontend/packages/core/src/shared/components/ssh-viewer/ssh-viewer.component.scss +++ b/src/frontend/packages/core/src/shared/components/ssh-viewer/ssh-viewer.component.scss @@ -19,6 +19,8 @@ display: flex; flex: 1; flex-direction: column; + max-height: 100%; + overflow: hidden; } } diff --git a/src/frontend/packages/core/src/shared/components/ssh-viewer/ssh-viewer.component.ts b/src/frontend/packages/core/src/shared/components/ssh-viewer/ssh-viewer.component.ts index f3c9382c26..911c75c9d3 100644 --- a/src/frontend/packages/core/src/shared/components/ssh-viewer/ssh-viewer.component.ts +++ b/src/frontend/packages/core/src/shared/components/ssh-viewer/ssh-viewer.component.ts @@ -38,6 +38,8 @@ export class SshViewerComponent implements OnInit, OnDestroy, AfterViewChecked { public isConnecting = false; private isDestroying = false; + public message = ''; + @ViewChild('terminal') container: ElementRef; private xterm: Terminal; @@ -111,8 +113,11 @@ export class SshViewerComponent implements OnInit, OnDestroy, AfterViewChecked { this.msgSubscription = this.sshStream .subscribe( (data: string) => { - for (const c of data.split(' ')) { - this.xterm.write(String.fromCharCode(parseInt(c, 16))); + // Check for a window title message + if (!this.isWindowTitle(data)) { + for (const c of data.split(' ')) { + this.xterm.write(String.fromCharCode(parseInt(c, 16))); + } } }, (err) => { @@ -138,4 +143,20 @@ export class SshViewerComponent implements OnInit, OnDestroy, AfterViewChecked { this.sshInput.next(JSON.stringify({ cols: size.cols, rows: size.rows })); } } + + private isWindowTitle(data: string): boolean { + const chars = data.split(' '); + if (chars.length > 4 && + parseInt(chars[0], 16) === 27 && + parseInt(chars[1], 16) === 93 && + parseInt(chars[2], 16) === 50 && + parseInt(chars[3], 16) === 59) { + let title = ''; + for (let i = 4; i < chars.length - 1; i++) { + title += String.fromCharCode(parseInt(chars[i], 16)); + } + this.message = title; + } + return false; + } } diff --git a/src/jetstream/plugins/kubernetes/endpoint_config.go b/src/jetstream/plugins/kubernetes/endpoint_config.go index 22f3521f10..0a1329234d 100644 --- a/src/jetstream/plugins/kubernetes/endpoint_config.go +++ b/src/jetstream/plugins/kubernetes/endpoint_config.go @@ -1,7 +1,10 @@ package kubernetes import ( + "encoding/base64" "errors" + "fmt" + "regexp" log "github.com/sirupsen/logrus" @@ -45,6 +48,71 @@ func (c *KubernetesSpecification) GetConfigForEndpoint(masterURL string, token i } +func (c *KubernetesSpecification) GetKubeConfigForEndpoint(masterURL string, token interfaces.TokenRecord) (string, error) { + + name := "config-0" + clusterName := "cluster-0" + userName := "user-0" + + // Create a config + + // Initialize a new config + context := clientcmdapi.NewContext() + context.Cluster = clusterName + context.AuthInfo = userName + + // Configure the cluster + cluster := clientcmdapi.NewCluster() + cluster.Server = masterURL + cluster.InsecureSkipTLSVerify = true + + // Configure auth information + authInfo := clientcmdapi.NewAuthInfo() + err := c.addAuthInfoForEndpoint(authInfo, token) + + config := clientcmdapi.NewConfig() + config.Clusters[clusterName] = cluster + config.Kind = "Config" + config.Contexts[name] = context + config.AuthInfos[userName] = authInfo + config.CurrentContext = context.Cluster + + // Convert to string + str := `apiVersion: v1 +kind: Config +contexts: +- context: + cluster: kube + user: kube + name: kube +clusters: +- cluster: + insecure-skip-tls-verify: true + server: %s + name: kube +current-context: kube +preferences: {} +users: +- name: kube + user: +` + + space := regexp.MustCompile(`\t`) + s := space.ReplaceAllString(str, " ") + + // Now append the auth details + log.Infof("%+v", authInfo) + + if authInfo.ClientCertificateData != nil { + s = fmt.Sprintf("%s client-certificate-data: %s\n", s, base64.StdEncoding.EncodeToString(authInfo.ClientCertificateData)) + } + if authInfo.ClientKeyData != nil { + s = fmt.Sprintf("%s client-key-data: %s\n", s, base64.StdEncoding.EncodeToString(authInfo.ClientKeyData)) + } + + return fmt.Sprintf(s, masterURL), err +} + func (c *KubernetesSpecification) addAuthInfoForEndpoint(info *clientcmdapi.AuthInfo, tokenRec interfaces.TokenRecord) error { log.Debug("addAuthInfoForEndpoint") diff --git a/src/jetstream/plugins/kubernetes/go.mod b/src/jetstream/plugins/kubernetes/go.mod index 5393fae8ef..be7ed920b0 100644 --- a/src/jetstream/plugins/kubernetes/go.mod +++ b/src/jetstream/plugins/kubernetes/go.mod @@ -46,6 +46,7 @@ require ( github.com/peterbourgon/diskv v2.0.1+incompatible // indirect github.com/pkg/errors v0.8.1 // indirect github.com/russross/blackfriday v2.0.0+incompatible // indirect + github.com/satori/go.uuid v1.2.0 // indirect github.com/shurcooL/sanitized_anchor_name v1.0.0 // indirect github.com/sirupsen/logrus v1.3.0 github.com/spf13/cobra v0.0.3 // indirect diff --git a/src/jetstream/plugins/kubernetes/go.sum b/src/jetstream/plugins/kubernetes/go.sum index 69a959d32c..6243abc37b 100644 --- a/src/jetstream/plugins/kubernetes/go.sum +++ b/src/jetstream/plugins/kubernetes/go.sum @@ -159,9 +159,11 @@ github.com/prometheus/procfs v0.0.0-20181204211112-1dc9a6cbc91a h1:9a8MnZMP0X2nL github.com/prometheus/procfs v0.0.0-20181204211112-1dc9a6cbc91a/go.mod h1:c3At6R/oaqEKCNdg8wHV1ftS6bRYblBhIjjI8uT2IGk= github.com/russross/blackfriday v1.5.2 h1:HyvC0ARfnZBqnXwABFeSZHpKvJHJJfPz81GNueLj0oo= github.com/russross/blackfriday v1.5.2/go.mod h1:JO/DiYxRf+HjHt06OyowR9PTA263kcR/rfWxYHBV53g= -github.com/russross/blackfriday v2.0.0+incompatible/go.mod h1:JO/DiYxRf+HjHt06OyowR9PTA263kcR/rfWxYHBV53g= github.com/russross/blackfriday v2.0.0 h1:L7Oc72h7rDqGkbUorN/ncJ4N/y220/YRezHvBoKLOFA= github.com/russross/blackfriday v2.0.0/go.mod h1:JO/DiYxRf+HjHt06OyowR9PTA263kcR/rfWxYHBV53g= +github.com/russross/blackfriday v2.0.0+incompatible/go.mod h1:JO/DiYxRf+HjHt06OyowR9PTA263kcR/rfWxYHBV53g= +github.com/satori/go.uuid v1.2.0 h1:0uYX9dsZ2yD7q2RtLRtPSdGDWzjeM3TbMJP9utgA0ww= +github.com/satori/go.uuid v1.2.0/go.mod h1:dA0hQrYB0VpLJoorglMZABFdXlWrHn1NEOzdhQKdks0= github.com/shurcooL/sanitized_anchor_name v1.0.0 h1:PdmoCO6wvbs+7yrJyMORt4/BmY5IYyJwS/kOiWx8mHo= github.com/shurcooL/sanitized_anchor_name v1.0.0/go.mod h1:1NzhyTcUVG4SuEtjjoZeVRXNmyL/1OwPU0+IJeTBvfc= github.com/sirupsen/logrus v1.3.0 h1:hI/7Q+DtNZ2kINb6qt/lS+IyXnHQe9e90POfeewL/ME= diff --git a/src/jetstream/plugins/kubernetes/kube_console.go b/src/jetstream/plugins/kubernetes/kube_console.go new file mode 100644 index 0000000000..7f54775d01 --- /dev/null +++ b/src/jetstream/plugins/kubernetes/kube_console.go @@ -0,0 +1,582 @@ +package kubernetes + +import ( + "bytes" + "errors" + "fmt" + "crypto/tls" + //"encoding/base64" + "encoding/json" + "net" + "net/http" + "time" + "strings" + //"io/ioutil" + //yaml "gopkg.in/yaml.v2" + + uuid "github.com/satori/go.uuid" + "github.com/labstack/echo" + log "github.com/sirupsen/logrus" + + "github.com/cloudfoundry-incubator/stratos/src/jetstream/repository/interfaces" + "github.com/cloudfoundry-incubator/stratos/src/jetstream/plugins/kubernetes/auth" + restclient "k8s.io/client-go/rest" + //"k8s.io/kubernetes/pkg/client/unversioned/remotecommand" + "github.com/gorilla/websocket" + "k8s.io/client-go/kubernetes" + v1 "k8s.io/api/core/v1" + metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" + corev1 "k8s.io/client-go/kubernetes/typed/core/v1" + + "k8s.io/client-go/tools/remotecommand" + scheme "k8s.io/client-go/kubernetes/scheme" +) + +const ( + consoleContainerName = "kube-console" +) + +var history = "" + +// TTY Resize, see: https://gitlab.cncf.ci/kubernetes/kubernetes/commit/3b21a9901bcd48bb452d3bf1a0cddc90dae142c4#9691a2f9b9c30711f0397221db0b9ac55ab0e2d1 + +// Allow connections from any Origin +var upgrader = websocket.Upgrader{ + CheckOrigin: func(r *http.Request) bool { return true }, +} + +// PodCreationData stores the clients and names used ot create pod and secret +type PodCreationData struct { + ClientSet *kubernetes.Clientset + Config *restclient.Config + Namespace string + PodClient corev1.PodInterface + SecretClient corev1.SecretInterface + PodName string + SecretName string +} + +// KeyCode - JSON object that is passed from the front-end to notify of a key press or a term resize +type KeyCode struct { + Key string `json:"key"` + Cols int `json:"cols"` + Rows int `json:"rows"` +} + +type TermianlSize struct { + Width uint16 + Height uint16 +} + +const ( + // Time allowed to write a message to the peer. + writeWait = 10 * time.Second +) + +func (k *KubernetesSpecification) KubeConsole(c echo.Context) error { + + c.Response().Status = 500 + + log.Info("Kube Console backend request") + + endpointGUID := c.Param("guid") + userGUID := c.Get("user_id").(string) + + ///api/v1/namespaces/project-1/pods/pod-1-lmlzj/exec?command=/bin/bash&stdin=true&stderr=true&stdout=true&tty=true + + namespace := "stratos" + + // TODO: Refresh auth token + + // Upgrade the web socket for the incoming request + ws, pingTicker, err := interfaces.UpgradeToWebSocket(c) + if err != nil { + return err + } + defer ws.Close() + defer pingTicker.Stop() + + // Send a message to say that we are creating the pod + sendProgressMessage(ws, "Launching Kubernetes Console ... one moment please") + + var p = k.portalProxy + + cnsiRecord, err := p.GetCNSIRecord(endpointGUID) + if err != nil { + return errors.New("Could not get endpoint information") + } + + // Get token for this users + tokenRec, ok := p.GetCNSITokenRecord(endpointGUID, userGUID) + if !ok { + return errors.New("Could not get token") + } + + podData, err := k.createPod(c, cnsiRecord, tokenRec, namespace, ws) + // Clear progress message + sendProgressMessage(ws, "") + + if err != nil { + log.Error("ERROR creating secret or pod") + log.Info(err) + k.cleanupPodAndSecret(podData) + return err + } + + log.Info(podData.PodName) + + log.Info(tokenRec.AuthToken) + log.Info(tokenRec.AuthType) + + // Make the info call to the SSH endpoint info + // Currently this is not cached, so we must get it each time + apiEndpoint := cnsiRecord.APIEndpoint + log.Info(apiEndpoint) + // target := fmt.Sprintf("%s/api/v1/namespaces/kube-system/services/https:kubernetes-dashboard:/proxy/", apiEndpoint) + //target := fmt.Sprintf("%s/api/v1/namespaces/kube-system/services/http:kubernetes-dashboard:/proxy/", apiEndpoint) + // target := http://localhost:8001/api/v1/namespaces/kube-system/services/kubernetes-dashboard/proxy + target := fmt.Sprintf("%s/api/v1/namespaces/%s/pods/%s/exec?command=/bin/bash&stdin=true&stderr=true&stdout=true&tty=true", apiEndpoint, namespace, podData.PodName) + log.Info(target) + + // config, err := k.getConfig(&cnsiRecord, &tokenRec) + // if err != nil { + // return errors.New("Could not get config for this auth type") + // } + + + req, err := http.NewRequest("POST", target, nil) + if err != nil { + k.cleanupPodAndSecret(podData) + return errors.New("Could not create new HTTP request") + } + + // Set auth header so we log in if needed + if len(tokenRec.AuthToken) > 0 { + //req.Header.Add("Authorization", "Bearer "+tokenRec.AuthToken) + log.Info("Setting auth header") + } + + //req.Header.Add("Accept", "*/*") + + log.Info("Config") + log.Info("Making request") + log.Info(req) + + // endpointRequest := &interfaces.CNSIRequest{ + // GUID: endpointGUID, + // } + + kubeAuthToken := &auth.KubeCertificate{} + err = json.NewDecoder(strings.NewReader(tokenRec.AuthToken)).Decode(kubeAuthToken) + if err != nil { + k.cleanupPodAndSecret(podData) + return err + } + cert, err := kubeAuthToken.GetCerticate() + if err != nil { + k.cleanupPodAndSecret(podData) + return err + } + dial := (&net.Dialer{ + Timeout: time.Duration(30) * time.Second, + KeepAlive: 30 * time.Second, + }).Dial + + sslTransport := &http.Transport{ + Proxy: http.ProxyFromEnvironment, + Dial: dial, + TLSHandshakeTimeout: 10 * time.Second, // 10 seconds is a sound default value (default is 0) + TLSClientConfig: &tls.Config{ + InsecureSkipVerify: true, + Certificates: []tls.Certificate{cert}, + }, + MaxIdleConnsPerHost: 6, // (default is 2) + } + + kubeCertClient := http.Client{} + kubeCertClient.Transport = sslTransport + kubeCertClient.Timeout = time.Duration(30) * time.Second + + dialer := &websocket.Dialer{ + TLSClientConfig: &tls.Config{ + InsecureSkipVerify: true, + Certificates: []tls.Certificate{cert}, + }, + } + + if strings.HasPrefix(target ,"https://") { + target = "wss://" + target[8:] + } else { + target = "ws://" + target[7:] + } + + header := &http.Header{} + wsConn, res, err := dialer.Dial(target, *header) + + if err == nil { + defer wsConn.Close() + } + + // log.Info(err) + // log.Info(res) + // log.Info(wsConn) + + // if kubeAuthToken.Token != "" { + // req.Header.Set("Authorization", "bearer "+kubeAuthToken.Token) + // } + + //res, err := kubeCertClient.Do(req) + + + kubeCertClient.CloseIdleConnections() + + //var res *http.Response + + // var client http.Client + // client = p.GetHttpClientForRequest(req, cnsiRecord.SkipSSLValidation) + // res, err = client.Do(req) + + // Find the auth provider for the auth type - default to oauthflow + // authHandler := p.GetAuthProvider(tokenRec.AuthType) + // if authHandler.Handler != nil { + // res, err = authHandler.Handler(endpointRequest, req) + // } else { + // res, err = p.DoOAuthFlowRequest(endpointRequest, req) + // } + log.Error(err) +// log.Error(res) + + if err != nil { + log.Error("Failed to make request") + k.cleanupPodAndSecret(podData) + return errors.New("Could not make request") + } + + log.Error("=== Made request to exec endpoint OK") + log.Error(res) + + // Websockets next + //log.Info(wsConn) + + //done := make(chan struct{}) + stdoutDone := make(chan struct{}) + go pumpStdout(ws, wsConn, stdoutDone) + + // Read the input from the web socket and pipe it to the SSH client + for { + _, r, err := ws.ReadMessage() + if err != nil { + log.Error("Error reading message from web socket") + log.Warnf("%v+", err) + k.cleanupPodAndSecret(podData) + return err + } + + res := KeyCode{} + json.Unmarshal(r, &res) + + if res.Cols == 0 { + + + slice := make([]byte, 1) + slice[0] = 0 + slice = append(slice, []byte(res.Key)...) + wsConn.WriteMessage(websocket.TextMessage, slice) + } else { + // Terminal resize request + // if err := windowChange(session, res.Rows, res.Cols); err != nil { + // log.Error("Can not resize the PTY") + // } + log.Error("Terminal resize receieved") + + size := TermianlSize{ + Width: uint16(res.Cols), + Height: uint16(res.Rows), + } + j, _ := json.Marshal(size) + log.Info(j) + + resizeStream := []byte{4} + slice := append(resizeStream, j...) + log.Info(slice) + wsConn.WriteMessage(websocket.TextMessage, slice) + } + } + + // Cleanup + log.Info("*** Cleaning up.... ***") + + return k.cleanupPodAndSecret(podData) +} + +func pumpStdout(ws *websocket.Conn, source *websocket.Conn, done chan struct{}) { + //buffer := make([]byte, 32768) + for { + _, r, err := source.ReadMessage() + if err != nil { + log.Info(err) + ws.Close() + break + } + ws.SetWriteDeadline(time.Now().Add(writeWait)) + bytes := fmt.Sprintf("% x\n", r[1:]) + if err := ws.WriteMessage(websocket.TextMessage, []byte(bytes)); err != nil { + log.Error("App SSH Failed to write nessage") + ws.Close() + break + } + } +} + +func (k *KubernetesSpecification) createPod(c echo.Context, cnsiRecord interfaces.CNSIRecord, tokenRecord interfaces.TokenRecord, namespace string, ws *websocket.Conn) (*PodCreationData, error) { + + id := uuid.NewV4().String() + secretName := fmt.Sprintf("k8s-s-console-%s", id) + podName := fmt.Sprintf("k8s-p-console-%s", id) + + result := &PodCreationData{} + + config, err := k.GetConfigForEndpoint(cnsiRecord.APIEndpoint.String(), tokenRecord) + if err != nil { + return result, errors.New("Can not get Kubernetes config for specified endpoint") + } + + result.Config = config + + kubeConfig, err := k.GetKubeConfigForEndpoint(cnsiRecord.APIEndpoint.String(), tokenRecord) + if err != nil { + return result, errors.New("Can not get Kubernetes config for specified endpoint") + } + + kubeClient, err := kubernetes.NewForConfig(config) + if err != nil { + log.Error("Could not get kube client") + return result, err + } + + result.Namespace = namespace + result.ClientSet = kubeClient + + // Create the secret + secretSpec := &v1.Secret{ + TypeMeta: metav1.TypeMeta{ + Kind: "secret", + APIVersion: "v1", + }, + ObjectMeta: metav1.ObjectMeta{ + Name: secretName, + Namespace: namespace, + }, + Type: "Opaque", + } + + secretSpec.Data = make(map[string][]byte) + secretSpec.Data["kubeconfig"] = []byte(kubeConfig) + secretSpec.Data["history"] = []byte(history) + + // Get Helm repository script if we have Helm repositories + helmSetup := getHelmRepoSetupScript(k.portalProxy) + if len(helmSetup) > 0 { + secretSpec.Data["helm-setup"] = []byte(helmSetup) + } + + secretsClient := kubeClient.CoreV1().Secrets(namespace) + _, err = secretsClient.Create(secretSpec) + + if err != nil { + log.Warn("Unable to create Secret") + return result, err + } + + result.SecretClient = secretsClient + result.SecretName = secretName + + podClient := kubeClient.CoreV1().Pods(namespace) + + podSpec := &v1.Pod{ + TypeMeta: metav1.TypeMeta{ + Kind: "pod", + APIVersion: "v1", + }, + ObjectMeta: metav1.ObjectMeta{ + Name: podName, + Namespace: namespace, + }, + } + + podSpec.ObjectMeta.Annotations = make(map[string]string) + + // Record the session ID + session, err :=k.portalProxy.GetSession(c) + if err == nil { + podSpec.ObjectMeta.Annotations["stratos-session"] = session.ID + log.Infof("Session ID: %s", session.ID) + } + + automount := false + + podSpec.Spec.AutomountServiceAccountToken = &automount + podSpec.Spec.RestartPolicy = "Never" + + volumeMountsSpec := make([]v1.VolumeMount, 1) + volumeMountsSpec[0].Name = "kubeconfig" + volumeMountsSpec[0].MountPath = "/root/.stratos" + volumeMountsSpec[0].ReadOnly = true + + containerSpec := make([]v1.Container, 1) + containerSpec[0].Name = consoleContainerName + containerSpec[0].Image = "nwmac/kubeconsole" + containerSpec[0].ImagePullPolicy = "Always" + containerSpec[0].VolumeMounts = volumeMountsSpec + podSpec.Spec.Containers = containerSpec + + volumesSpec := make([]v1.Volume, 1) + volumesSpec[0].Name = "kubeconfig" + volumesSpec[0].Secret = &v1.SecretVolumeSource{ + SecretName: secretName, + } + podSpec.Spec.Volumes = volumesSpec + + // Create a new pod + pod, err := podClient.Create(podSpec) + if err != nil { + return result, err + } + + result.PodClient = podClient + result.PodName = podName + + sendProgressMessage(ws, "Waiting for Kubernetes Console to start up ...") + + statusOptions := metav1.GetOptions{} + + // Wait for the pod to be running + ready := false + for { + status, err := podClient.Get(pod.Name, statusOptions) + + if err != nil { + break + } + + //log.Info(status.Status.Phase) + if status.Status.Phase == "Running" { + ready = true + } + + if ready { + break + } + + // Sleep + time.Sleep(2500 * time.Millisecond) + } + + return result, err +} + +func (k *KubernetesSpecification) cleanupPodAndSecret(podData *PodCreationData) error { + + if len(podData.PodName) > 0 { + captureBashHistory(podData) + podData.PodClient.Delete(podData.PodName, nil) + } + + if len(podData.SecretName) > 0 { + podData.SecretClient.Delete(podData.SecretName, nil) + } + + return nil +} + +func getHelmRepoSetupScript(portalProxy interfaces.PortalProxy) string { + + str := "" + + // Get all of the helm endpoints + endpoints, err := portalProxy.ListEndpoints() + if err != nil { + log.Error("Can not list Helm Repository endpoints") + return str + } + + for _, ep := range endpoints { + if ep.CNSIType == "helm" { + str += fmt.Sprintf("helm repo add %s %s > /dev/null\n", ep.Name, ep.APIEndpoint) + } + } + + return str +} + +func sendProgressMessage(ws *websocket.Conn, progressMsg string) { + // Send a message to say that we are creating the pod + msg := fmt.Sprintf("\033]2;%s\007", progressMsg) + bytes := fmt.Sprintf("% x\n", []byte(msg)) + if err := ws.WriteMessage(websocket.TextMessage, []byte(bytes)); err != nil { + log.Error("Could not send message to client to indicate console is starting") + } +} + +func captureBashHistory(podData *PodCreationData) error { + + log.Warn("**** Trying to capture bash history from pod ****") + + // Can we capture the history? + + // returning stdout, stderr and error. `options` allowed for + // additional parameters to be passed. + + //cmd := []string{"bash", "-c", "\"cat $HISTFILE\""} + cmd := []string{"cat", "/root/.bash_history"} + + + req := podData.ClientSet.Core().RESTClient().Post(). + Resource("pods"). + Name(podData.PodName). + Namespace(podData.Namespace). + SubResource("exec"). + Param("container", consoleContainerName) + + req.VersionedParams(&v1.PodExecOptions{ + Container: consoleContainerName, + Command: cmd, + Stdin: false, + Stdout: true, + Stderr: false, + TTY: true, + }, scheme.ParameterCodec) + + var stdout bytes.Buffer + //err := execute("POST", req.URL(), nil, nil, &stdout, nil, false) + + exec, err := remotecommand.NewSPDYExecutor(podData.Config, "POST", req.URL()) + if err != nil { + log.Error("Could not exec") + log.Error(err) + return err + } + + log.Warn("Attempting stream ......") + + err = exec.Stream(remotecommand.StreamOptions{ + //SupportedProtocols: remotecommandserver.SupportedStreamingProtocols, + Stdin: nil, + Stdout: &stdout, + Stderr: nil, + Tty: true, + }) + + log.Error("Get Bash History") + log.Error(err) + log.Error(stdout.String()) + + history = stdout.String() + + // if options.PreserveWhitespace { + // return stdout.String(), stderr.String(), err + // } + // return strings.TrimSpace(stdout.String()), strings.TrimSpace(stderr.String()), err + + return nil +} \ No newline at end of file diff --git a/src/jetstream/plugins/kubernetes/main.go b/src/jetstream/plugins/kubernetes/main.go index 736ddfa35d..a61c0e7548 100644 --- a/src/jetstream/plugins/kubernetes/main.go +++ b/src/jetstream/plugins/kubernetes/main.go @@ -112,6 +112,9 @@ func (c *KubernetesSpecification) AddSessionGroupRoutes(echoGroup *echo.Group) { echoGroup.GET("/kubedash/ui/:guid/*", c.kubeDashboardProxy) echoGroup.GET("/kubedash/:guid/status", c.kubeDashboardStatus) + // Kube Console + echoGroup.GET("/kubeconsole/:guid", c.KubeConsole) + // Helm Routes echoGroup.GET("/helm/releases", c.ListReleases) echoGroup.POST("/helm/install", c.InstallRelease) From 1bc727a988e6356ac346d92c66349c7571741dac Mon Sep 17 00:00:00 2001 From: Neil MacDougall Date: Thu, 5 Mar 2020 10:25:52 +0000 Subject: [PATCH 02/40] Remove debug and unused --- .../kubernetes/kube-console/kube-console.component.ts | 11 ----------- 1 file changed, 11 deletions(-) diff --git a/custom-src/frontend/app/custom/kubernetes/kube-console/kube-console.component.ts b/custom-src/frontend/app/custom/kubernetes/kube-console/kube-console.component.ts index 32481eb139..1466381281 100644 --- a/custom-src/frontend/app/custom/kubernetes/kube-console/kube-console.component.ts +++ b/custom-src/frontend/app/custom/kubernetes/kube-console/kube-console.component.ts @@ -72,19 +72,10 @@ export class KubeConsoleComponent implements OnInit { constructor( public kubeEndpointService: KubernetesEndpointService, - // private activatedRoute: ActivatedRoute, - // private store: Store, ) { } ngOnInit() { - const guid = this.kubeEndpointService.baseKube.guid; - - console.log('HERE'); - console.log(guid); - // const routeParams = this.activatedRoute.snapshot.params; - // this.instanceId = routeParams.index; - this.kubeSummaryLink = ( `/kubernetes/${guid}/summary` ); @@ -104,8 +95,6 @@ export class KubeConsoleComponent implements OnInit { this.sshInput ); - console.log(streamUrl); - this.messages = connection.messages.pipe( catchError(e => { if (e.type === 'error') { From 187b999e6c8483217891831077f59006ba6448fb Mon Sep 17 00:00:00 2001 From: Neil MacDougall Date: Thu, 5 Mar 2020 10:42:36 +0000 Subject: [PATCH 03/40] Fix Anagular 8 upgrade issues --- .../kube-console/kube-console.component.ts | 26 +++++++++---------- 1 file changed, 13 insertions(+), 13 deletions(-) diff --git a/custom-src/frontend/app/custom/kubernetes/kube-console/kube-console.component.ts b/custom-src/frontend/app/custom/kubernetes/kube-console/kube-console.component.ts index 1466381281..9da2be718a 100644 --- a/custom-src/frontend/app/custom/kubernetes/kube-console/kube-console.component.ts +++ b/custom-src/frontend/app/custom/kubernetes/kube-console/kube-console.component.ts @@ -2,10 +2,9 @@ import { Component, OnInit, ViewChild } from '@angular/core'; import { ActivatedRoute } from '@angular/router'; import { Store } from '@ngrx/store'; import { NEVER, Observable, Subject, Subscription } from 'rxjs'; -import websocketConnect from 'rxjs-websockets'; -import { catchError, first, map } from 'rxjs/operators'; +import websocketConnect, { normalClosureMessage } from 'rxjs-websockets'; +import { catchError, tap, switchMap } from 'rxjs/operators'; -import { AppState } from '../../../../../store/src/app-state'; import { IApp } from '../../../core/cf-api.types'; import { IHeaderBreadcrumb } from '../../../shared/components/page-header/page-header.types'; import { SshViewerComponent } from '../../../shared/components/ssh-viewer/ssh-viewer.component'; @@ -38,7 +37,7 @@ export class KubeConsoleComponent implements OnInit { public messages: Observable; - public connectionStatus: Observable; + public connectionStatus = new Subject(); public sshInput: Subject; @@ -56,7 +55,7 @@ export class KubeConsoleComponent implements OnInit { public breadcrumbs$: Observable; - @ViewChild('sshViewer') sshViewer: SshViewerComponent; + @ViewChild('sshViewer', { static: false }) sshViewer: SshViewerComponent; private getBreadcrumbs( application: IApp, @@ -75,6 +74,7 @@ export class KubeConsoleComponent implements OnInit { ) { } ngOnInit() { + this.connectionStatus.next(0); const guid = this.kubeEndpointService.baseKube.guid; this.kubeSummaryLink = ( `/kubernetes/${guid}/summary` @@ -82,7 +82,8 @@ export class KubeConsoleComponent implements OnInit { if (!guid) { this.messages = NEVER; - this.connectionStatus = NEVER; + this.connectionStatus.next(0); + this.errorMessage = 'No Endpoint ID available'; } else { const host = window.location.host; const protocol = window.location.protocol === 'https:' ? 'wss' : 'ws'; @@ -91,20 +92,19 @@ export class KubeConsoleComponent implements OnInit { ); this.sshInput = new Subject(); const connection = websocketConnect( - streamUrl, - this.sshInput + streamUrl ); - this.messages = connection.messages.pipe( - catchError(e => { - if (e.type === 'error') { + this.messages = connection.pipe( + tap(() => this.connectionStatus.next(1)), + switchMap(getResponse => getResponse(this.sshInput)), + catchError((e: Error) => { + if (e.message !== normalClosureMessage) { this.errorMessage = 'Error connecting to web socket'; } return []; })); - this.connectionStatus = connection.connectionStatus; - // this.breadcrumbs$ = this.applicationService.waitForAppEntity$.pipe( // map(app => this.getBreadcrumbs(app.entity.entity)), // first() From b5bf5ba3c062ba4e33ad7b0769580416a7459060 Mon Sep 17 00:00:00 2001 From: Neil MacDougall Date: Thu, 5 Mar 2020 10:47:44 +0000 Subject: [PATCH 04/40] Tidy up imports --- .../app/custom/kubernetes/kube-console/kube-console.component.ts | 1 - 1 file changed, 1 deletion(-) diff --git a/custom-src/frontend/app/custom/kubernetes/kube-console/kube-console.component.ts b/custom-src/frontend/app/custom/kubernetes/kube-console/kube-console.component.ts index 9da2be718a..e185e8ffe9 100644 --- a/custom-src/frontend/app/custom/kubernetes/kube-console/kube-console.component.ts +++ b/custom-src/frontend/app/custom/kubernetes/kube-console/kube-console.component.ts @@ -1,6 +1,5 @@ import { Component, OnInit, ViewChild } from '@angular/core'; import { ActivatedRoute } from '@angular/router'; -import { Store } from '@ngrx/store'; import { NEVER, Observable, Subject, Subscription } from 'rxjs'; import websocketConnect, { normalClosureMessage } from 'rxjs-websockets'; import { catchError, tap, switchMap } from 'rxjs/operators'; From 8c09aed298d75f57a58ca5eca5b149927ba3b21b Mon Sep 17 00:00:00 2001 From: Richard Cox Date: Thu, 28 May 2020 11:36:44 +0100 Subject: [PATCH 05/40] Fix misaligned user button - fixes #4316 --- .../src/shared/components/page-header/page-header.component.scss | 1 + 1 file changed, 1 insertion(+) diff --git a/src/frontend/packages/core/src/shared/components/page-header/page-header.component.scss b/src/frontend/packages/core/src/shared/components/page-header/page-header.component.scss index fc803d0362..9df4ed9f79 100644 --- a/src/frontend/packages/core/src/shared/components/page-header/page-header.component.scss +++ b/src/frontend/packages/core/src/shared/components/page-header/page-header.component.scss @@ -103,6 +103,7 @@ $bottom-index: $top-index - 1; padding: 0 10px 0 5px; } &__menu-button { + align-items: center; display: flex; justify-content: center; } From 4d33c4d4c25b265899164d6ec5efea7b92756e51 Mon Sep 17 00:00:00 2001 From: Richard Cox Date: Fri, 29 May 2020 09:23:44 +0100 Subject: [PATCH 06/40] Temporarily add branch v3.2.1 to travis --- .travis.yml | 1 + 1 file changed, 1 insertion(+) diff --git a/.travis.yml b/.travis.yml index 7f87e2619e..3d04b8ae8b 100644 --- a/.travis.yml +++ b/.travis.yml @@ -20,6 +20,7 @@ branches: - master - v2-master - e2e-tests + - v3.2.1 cache: directories: - "$HOME/.npm" From 839cc2fc1fd37f662f1ecdd444c962d59973ef81 Mon Sep 17 00:00:00 2001 From: Ben Berry Date: Thu, 21 May 2020 16:15:51 -0700 Subject: [PATCH 07/40] check sso whitelist in more places Signed-off-by: Ben Berry --- src/jetstream/authuaa.go | 8 ++++++++ 1 file changed, 8 insertions(+) diff --git a/src/jetstream/authuaa.go b/src/jetstream/authuaa.go index 63b152eab4..04b93a9b5e 100644 --- a/src/jetstream/authuaa.go +++ b/src/jetstream/authuaa.go @@ -466,6 +466,14 @@ func (p *portalProxy) ssoLoginToUAA(c echo.Context) error { state = fmt.Sprintf("%s/login?SSO_Message=%s", state, url.QueryEscape(msg)) } + if !safeSSORedirectState(state, p.Config.SSOWhiteList) { + err := interfaces.NewHTTPShadowError( + http.StatusUnauthorized, + "SSO Login: Disallowed redirect state", + "SSO Login: Disallowed redirect state") + return err + } + return c.Redirect(http.StatusTemporaryRedirect, state) } From 28920c09d3d6e2b947fc4662e5d1ce0eb0cb1766 Mon Sep 17 00:00:00 2001 From: Ben Berry Date: Thu, 21 May 2020 16:53:29 -0700 Subject: [PATCH 08/40] refactor sso state checks into single function Signed-off-by: Ben Berry --- src/jetstream/authuaa.go | 38 ++++++++++++++++++-------------------- 1 file changed, 18 insertions(+), 20 deletions(-) diff --git a/src/jetstream/authuaa.go b/src/jetstream/authuaa.go index 04b93a9b5e..2fee5c0eb4 100644 --- a/src/jetstream/authuaa.go +++ b/src/jetstream/authuaa.go @@ -435,14 +435,11 @@ func (p *portalProxy) RefreshUAAToken(userGUID string) (t interfaces.TokenRecord // We use a single callback so this can be whitelisted in the client func (p *portalProxy) ssoLoginToUAA(c echo.Context) error { state := c.QueryParam("state") - if len(state) == 0 { - err := interfaces.NewHTTPShadowError( - http.StatusUnauthorized, - "SSO Login: State parameter missing", - "SSO Login: State parameter missing") - return err - } + stateErr := validateSSORedirectState(state, p.Config.SSOWhiteList) + if stateErr != nil { + return stateErr + } // We use the same callback URL for both UAA and endpoint login // Check if it is an endpoint login and dens to the right handler endpointGUID := c.QueryParam("guid") @@ -466,13 +463,6 @@ func (p *portalProxy) ssoLoginToUAA(c echo.Context) error { state = fmt.Sprintf("%s/login?SSO_Message=%s", state, url.QueryEscape(msg)) } - if !safeSSORedirectState(state, p.Config.SSOWhiteList) { - err := interfaces.NewHTTPShadowError( - http.StatusUnauthorized, - "SSO Login: Disallowed redirect state", - "SSO Login: Disallowed redirect state") - return err - } return c.Redirect(http.StatusTemporaryRedirect, state) } @@ -527,15 +517,25 @@ func (p *portalProxy) initSSOlogin(c echo.Context) error { } state := c.QueryParam("state") + stateErr := validateSSORedirectState(state, p.Config.SSOWhiteList) + if stateErr != nil { + return stateErr + } + + redirectURL := fmt.Sprintf("%s/oauth/authorize?response_type=code&client_id=%s&redirect_uri=%s", p.Config.ConsoleConfig.AuthorizationEndpoint, p.Config.ConsoleConfig.ConsoleClient, url.QueryEscape(getSSORedirectURI(state, state, ""))) + c.Redirect(http.StatusTemporaryRedirect, redirectURL) + return nil +} + +func validateSSORedirectState(state string, whiteListStr string) error { if len(state) == 0 { err := interfaces.NewHTTPShadowError( http.StatusUnauthorized, - "SSO Login: Redirect state parameter missing", - "SSO Login: Redirect state parameter missing") + "SSO Login: State parameter missing", + "SSO Login: State parameter missing") return err } - - if !safeSSORedirectState(state, p.Config.SSOWhiteList) { + if !safeSSORedirectState(state,whiteListStr) { err := interfaces.NewHTTPShadowError( http.StatusUnauthorized, "SSO Login: Disallowed redirect state", @@ -543,8 +543,6 @@ func (p *portalProxy) initSSOlogin(c echo.Context) error { return err } - redirectURL := fmt.Sprintf("%s/oauth/authorize?response_type=code&client_id=%s&redirect_uri=%s", p.Config.ConsoleConfig.AuthorizationEndpoint, p.Config.ConsoleConfig.ConsoleClient, url.QueryEscape(getSSORedirectURI(state, state, ""))) - c.Redirect(http.StatusTemporaryRedirect, redirectURL) return nil } From 87c6bf847591432535dbd61de186b2f0947344db Mon Sep 17 00:00:00 2001 From: Richard Cox Date: Fri, 29 May 2020 10:02:31 +0100 Subject: [PATCH 09/40] Update version --- package-lock.json | 3586 ++++++++++++++++++++++----------------------- package.json | 2 +- 2 files changed, 1794 insertions(+), 1794 deletions(-) diff --git a/package-lock.json b/package-lock.json index cb283a2c55..48448cbebf 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,6 +1,6 @@ { "name": "stratos", - "version": "3.2.0", + "version": "3.2.1", "lockfileVersion": 1, "requires": true, "dependencies": { @@ -20,7 +20,7 @@ "integrity": "sha512-Z9Yfa11F6B9Sg/BK9MnqnQ+aQYicPLtilXBp2yUtDt2JRCE0h26d33EnfO3ZxoNxG0T92OUucP3Ct7cpfkdFfw==", "dev": true, "requires": { - "tslib": "1.10.0" + "tslib": "^1.9.0" } } } @@ -95,10 +95,10 @@ "integrity": "sha512-TXtUUEYHuaTEbLZWIKUr5pmBuhDLy+8KYtPYdcV8qC+pOZL+NKqYwvWSRrVXHn+ZmRRAu8vJTAznH7Oag6RVRw==", "dev": true, "requires": { - "fast-deep-equal": "2.0.1", - "fast-json-stable-stringify": "2.0.0", - "json-schema-traverse": "0.4.1", - "uri-js": "4.2.2" + "fast-deep-equal": "^2.0.1", + "fast-json-stable-stringify": "^2.0.0", + "json-schema-traverse": "^0.4.1", + "uri-js": "^4.2.2" } }, "glob": { @@ -107,12 +107,12 @@ "integrity": "sha512-hkLPepehmnKk41pUGm3sYxoFs/umurYfYJCerbXEyFIWcAzvpipAgVkBqqT9RBKMGjnq6kMuyYwha6csxbiM1A==", "dev": true, "requires": { - "fs.realpath": "1.0.0", - "inflight": "1.0.6", - "inherits": "2.0.4", - "minimatch": "3.0.4", - "once": "1.4.0", - "path-is-absolute": "1.0.1" + "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" } }, "parse5": { @@ -127,7 +127,7 @@ "integrity": "sha512-Z9Yfa11F6B9Sg/BK9MnqnQ+aQYicPLtilXBp2yUtDt2JRCE0h26d33EnfO3ZxoNxG0T92OUucP3Ct7cpfkdFfw==", "dev": true, "requires": { - "tslib": "1.10.0" + "tslib": "^1.9.0" } }, "semver": { @@ -160,7 +160,7 @@ "integrity": "sha512-Z9Yfa11F6B9Sg/BK9MnqnQ+aQYicPLtilXBp2yUtDt2JRCE0h26d33EnfO3ZxoNxG0T92OUucP3Ct7cpfkdFfw==", "dev": true, "requires": { - "tslib": "1.10.0" + "tslib": "^1.9.0" } } } @@ -210,7 +210,7 @@ "integrity": "sha512-Z9Yfa11F6B9Sg/BK9MnqnQ+aQYicPLtilXBp2yUtDt2JRCE0h26d33EnfO3ZxoNxG0T92OUucP3Ct7cpfkdFfw==", "dev": true, "requires": { - "tslib": "1.10.0" + "tslib": "^1.9.0" } } } @@ -234,10 +234,10 @@ "integrity": "sha512-TXtUUEYHuaTEbLZWIKUr5pmBuhDLy+8KYtPYdcV8qC+pOZL+NKqYwvWSRrVXHn+ZmRRAu8vJTAznH7Oag6RVRw==", "dev": true, "requires": { - "fast-deep-equal": "2.0.1", - "fast-json-stable-stringify": "2.0.0", - "json-schema-traverse": "0.4.1", - "uri-js": "4.2.2" + "fast-deep-equal": "^2.0.1", + "fast-json-stable-stringify": "^2.0.0", + "json-schema-traverse": "^0.4.1", + "uri-js": "^4.2.2" } }, "rxjs": { @@ -246,7 +246,7 @@ "integrity": "sha512-Z9Yfa11F6B9Sg/BK9MnqnQ+aQYicPLtilXBp2yUtDt2JRCE0h26d33EnfO3ZxoNxG0T92OUucP3Ct7cpfkdFfw==", "dev": true, "requires": { - "tslib": "1.10.0" + "tslib": "^1.9.0" } }, "source-map": { @@ -273,7 +273,7 @@ "integrity": "sha512-Z9Yfa11F6B9Sg/BK9MnqnQ+aQYicPLtilXBp2yUtDt2JRCE0h26d33EnfO3ZxoNxG0T92OUucP3Ct7cpfkdFfw==", "dev": true, "requires": { - "tslib": "1.10.0" + "tslib": "^1.9.0" } } } @@ -283,7 +283,7 @@ "resolved": "https://registry.npmjs.org/@angular/animations/-/animations-8.2.9.tgz", "integrity": "sha512-l30AF0d9P5okTPM1wieUHgcnDyGSNvyaBcxXSOkT790wAP2v5zs7VrKq9Lm+ICu4Nkx07KrOr5XLUHhqsg3VXA==", "requires": { - "tslib": "1.10.0" + "tslib": "^1.9.0" } }, "@angular/cdk": { @@ -291,8 +291,8 @@ "resolved": "https://registry.npmjs.org/@angular/cdk/-/cdk-8.2.2.tgz", "integrity": "sha512-e+BtFab0Vd1q/ZVu6l850Q4vvgyVYiugSX31oMRlp86fKHPowlAO7jL3z5JcAG7TybpLIqd7oqF8XQBR/yw83w==", "requires": { - "parse5": "5.1.0", - "tslib": "1.10.0" + "parse5": "^5.0.0", + "tslib": "^1.7.1" } }, "@angular/cli": { @@ -308,7 +308,7 @@ "@schematics/update": "0.803.7", "@yarnpkg/lockfile": "1.1.0", "ansi-colors": "4.1.1", - "debug": "4.1.1", + "debug": "^4.1.1", "ini": "1.3.5", "inquirer": "6.5.1", "npm-package-arg": "6.1.0", @@ -318,8 +318,8 @@ "read-package-tree": "5.3.1", "semver": "6.3.0", "symbol-observable": "1.2.0", - "universal-analytics": "0.4.20", - "uuid": "3.3.3" + "universal-analytics": "^0.4.20", + "uuid": "^3.3.2" }, "dependencies": { "ansi-colors": { @@ -334,7 +334,7 @@ "integrity": "sha512-pYAIzeRo8J6KPEaJ0VWOh5Pzkbw/RetuzehGM7QRRX5he4fPHx2rdKMB256ehJCkX+XRQm16eZLqLNS8RSZXZw==", "dev": true, "requires": { - "ms": "2.1.2" + "ms": "^2.1.1" } }, "ms": { @@ -356,7 +356,7 @@ "resolved": "https://registry.npmjs.org/@angular/common/-/common-8.2.9.tgz", "integrity": "sha512-76WDU1USlI5vAzqCJ3gxCQGuu57aJEggNk/xoWmQEXipiFTFBh2wSKn/dE6Txr/q3COTPIcrmb9OCeal5kQPIA==", "requires": { - "tslib": "1.10.0" + "tslib": "^1.9.0" } }, "@angular/compiler": { @@ -364,7 +364,7 @@ "resolved": "https://registry.npmjs.org/@angular/compiler/-/compiler-8.2.9.tgz", "integrity": "sha512-oQho19DnOhEDNerCOGuGK95tcZ2oy4dSA5SykJmmniRnZzPM2++bJD32qJehXHy1K+3hv2zN9x7HPhqT3ljT6g==", "requires": { - "tslib": "1.10.0" + "tslib": "^1.9.0" } }, "@angular/compiler-cli": { @@ -374,14 +374,14 @@ "dev": true, "requires": { "canonical-path": "1.0.0", - "chokidar": "2.1.8", - "convert-source-map": "1.6.0", - "dependency-graph": "0.7.2", - "magic-string": "0.25.3", - "minimist": "1.2.0", - "reflect-metadata": "0.1.13", - "source-map": "0.6.1", - "tslib": "1.10.0", + "chokidar": "^2.1.1", + "convert-source-map": "^1.5.1", + "dependency-graph": "^0.7.2", + "magic-string": "^0.25.0", + "minimist": "^1.2.0", + "reflect-metadata": "^0.1.2", + "source-map": "^0.6.1", + "tslib": "^1.9.0", "yargs": "13.1.0" }, "dependencies": { @@ -397,18 +397,18 @@ "integrity": "sha512-ZmZUazfOzf0Nve7duiCKD23PFSCs4JPoYyccjUFF3aQkQadqBhfzhjkwBH2mNOG9cTBwhamM37EIsIkZw3nRgg==", "dev": true, "requires": { - "anymatch": "2.0.0", - "async-each": "1.0.3", - "braces": "2.3.2", - "fsevents": "1.2.9", - "glob-parent": "3.1.0", - "inherits": "2.0.4", - "is-binary-path": "1.0.1", - "is-glob": "4.0.1", - "normalize-path": "3.0.0", - "path-is-absolute": "1.0.1", - "readdirp": "2.2.1", - "upath": "1.2.0" + "anymatch": "^2.0.0", + "async-each": "^1.0.1", + "braces": "^2.3.2", + "fsevents": "^1.2.7", + "glob-parent": "^3.1.0", + "inherits": "^2.0.3", + "is-binary-path": "^1.0.0", + "is-glob": "^4.0.0", + "normalize-path": "^3.0.0", + "path-is-absolute": "^1.0.0", + "readdirp": "^2.2.1", + "upath": "^1.1.1" } }, "cliui": { @@ -417,9 +417,9 @@ "integrity": "sha512-4FG+RSG9DL7uEwRUZXZn3SS34DiDPfzP0VOiEwtUWlE+AR2EIg+hSyvrIgUUfhdgR/UkAeW2QHgeP+hWrXs7jQ==", "dev": true, "requires": { - "string-width": "2.1.1", - "strip-ansi": "4.0.0", - "wrap-ansi": "2.1.0" + "string-width": "^2.1.1", + "strip-ansi": "^4.0.0", + "wrap-ansi": "^2.0.0" }, "dependencies": { "string-width": { @@ -428,8 +428,8 @@ "integrity": "sha512-nOqH59deCq9SRHlxq1Aw85Jnt4w6KvLKqWVik6oA9ZklXLNIOlqg4F2yrT1MVaTjAqvVwdfeZ7w7aCvJD7ugkw==", "dev": true, "requires": { - "is-fullwidth-code-point": "2.0.0", - "strip-ansi": "4.0.0" + "is-fullwidth-code-point": "^2.0.0", + "strip-ansi": "^4.0.0" } } } @@ -464,7 +464,7 @@ "integrity": "sha512-avPEb8P8EGnwXKClwsNUgryVjllcRqtMYa49NTsbQagYuT1DcXnl1915oxWjoyGrXR6zH/Y0Zc96xWsPcoDKeA==", "dev": true, "requires": { - "invert-kv": "2.0.0" + "invert-kv": "^2.0.0" } }, "mem": { @@ -473,9 +473,9 @@ "integrity": "sha512-qX2bG48pTqYRVmDB37rn/6PT7LcR8T7oAX3bf99u1Tt1nzxYfxkgqDwUwolPlXweM0XzBOBFzSx4kfp7KP1s/w==", "dev": true, "requires": { - "map-age-cleaner": "0.1.3", - "mimic-fn": "2.1.0", - "p-is-promise": "2.1.0" + "map-age-cleaner": "^0.1.1", + "mimic-fn": "^2.0.0", + "p-is-promise": "^2.0.0" } }, "normalize-path": { @@ -490,9 +490,9 @@ "integrity": "sha512-Z8l3R4wYWM40/52Z+S265okfFj8Kt2cC2MKY+xNi3kFs+XGI7WXu/I309QQQYbRW4ijiZ+yxs9pqEhJh0DqW3Q==", "dev": true, "requires": { - "execa": "1.0.0", - "lcid": "2.0.0", - "mem": "4.3.0" + "execa": "^1.0.0", + "lcid": "^2.0.0", + "mem": "^4.0.0" } }, "require-main-filename": { @@ -513,9 +513,9 @@ "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==", "dev": true, "requires": { - "emoji-regex": "7.0.3", - "is-fullwidth-code-point": "2.0.0", - "strip-ansi": "5.2.0" + "emoji-regex": "^7.0.1", + "is-fullwidth-code-point": "^2.0.0", + "strip-ansi": "^5.1.0" }, "dependencies": { "ansi-regex": { @@ -530,7 +530,7 @@ "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", "dev": true, "requires": { - "ansi-regex": "4.1.0" + "ansi-regex": "^4.1.0" } } } @@ -541,7 +541,7 @@ "integrity": "sha1-qEeQIusaw2iocTibY1JixQXuNo8=", "dev": true, "requires": { - "ansi-regex": "3.0.0" + "ansi-regex": "^3.0.0" } }, "which-module": { @@ -556,17 +556,17 @@ "integrity": "sha512-1UhJbXfzHiPqkfXNHYhiz79qM/kZqjTE8yGlEjZa85Q+3+OwcV6NRkV7XOV1W2Eom2bzILeUn55pQYffjVOLAg==", "dev": true, "requires": { - "cliui": "4.1.0", - "find-up": "3.0.0", - "get-caller-file": "2.0.5", - "os-locale": "3.1.0", - "require-directory": "2.1.1", - "require-main-filename": "2.0.0", - "set-blocking": "2.0.0", - "string-width": "3.1.0", - "which-module": "2.0.0", - "y18n": "4.0.0", - "yargs-parser": "13.1.1" + "cliui": "^4.0.0", + "find-up": "^3.0.0", + "get-caller-file": "^2.0.1", + "os-locale": "^3.1.0", + "require-directory": "^2.1.1", + "require-main-filename": "^2.0.0", + "set-blocking": "^2.0.0", + "string-width": "^3.0.0", + "which-module": "^2.0.0", + "y18n": "^4.0.0", + "yargs-parser": "^13.0.0" } }, "yargs-parser": { @@ -575,8 +575,8 @@ "integrity": "sha512-oVAVsHz6uFrg3XQheFII8ESO2ssAf9luWuAd6Wexsu4F3OtIW0o8IribPXYrD4WC24LWtPrJlGy87y5udK+dxQ==", "dev": true, "requires": { - "camelcase": "5.3.1", - "decamelize": "1.2.0" + "camelcase": "^5.0.0", + "decamelize": "^1.2.0" } } } @@ -586,7 +586,7 @@ "resolved": "https://registry.npmjs.org/@angular/core/-/core-8.2.9.tgz", "integrity": "sha512-GpHAuLOlN9iioELCQBmAsjETTUCyFgVUI3LXwh3e63jnpd+ZuuZcZbjfTYhtgYVNMetn7cVEO6p88eb7qvpUWQ==", "requires": { - "tslib": "1.10.0" + "tslib": "^1.9.0" } }, "@angular/flex-layout": { @@ -594,7 +594,7 @@ "resolved": "https://registry.npmjs.org/@angular/flex-layout/-/flex-layout-8.0.0-beta.27.tgz", "integrity": "sha512-qmpvQPesU4ZQ56IscwgmVRpK2UnyV+gwvXUql7TMv0QV215hLcHczjGsrKkLfW2By5E7XEyDat9br72uVXcPMw==", "requires": { - "tslib": "1.10.0" + "tslib": "^1.7.1" } }, "@angular/forms": { @@ -602,7 +602,7 @@ "resolved": "https://registry.npmjs.org/@angular/forms/-/forms-8.2.9.tgz", "integrity": "sha512-kAdBuApC9PPOdPI8BmNhxCraAkXGbX/PkVan8pQ5xdumvgGqvVjbJvLaUSbJROPtgCRlQyiEDrHFd4gk/WU76A==", "requires": { - "tslib": "1.10.0" + "tslib": "^1.9.0" } }, "@angular/language-service": { @@ -616,7 +616,7 @@ "resolved": "https://registry.npmjs.org/@angular/material/-/material-8.2.2.tgz", "integrity": "sha512-mR2ppE+Z1S5As2SUFK8wUH76Fj7YgrefhrwVGaeCLcAen//RHPw043+KL2apPAUaltdIFlGFtUuA6yJN6av0nQ==", "requires": { - "tslib": "1.10.0" + "tslib": "^1.7.1" } }, "@angular/material-moment-adapter": { @@ -624,7 +624,7 @@ "resolved": "https://registry.npmjs.org/@angular/material-moment-adapter/-/material-moment-adapter-8.2.2.tgz", "integrity": "sha512-HZYoVssNh+uon0SszUDtVTWmjah5vtEJOYVYl9BakE6dC+Z4910B+/hVQ1HfOdas5mHpjxVoYpbIoe4QG5sc5A==", "requires": { - "tslib": "1.10.0" + "tslib": "^1.7.1" } }, "@angular/platform-browser": { @@ -632,7 +632,7 @@ "resolved": "https://registry.npmjs.org/@angular/platform-browser/-/platform-browser-8.2.9.tgz", "integrity": "sha512-k3aNZy0OTqGn7HlHHV52QF6ZAP/VlQhWGD2u5e1dWIWMq39kdkdSCNu5tiuAf5hIzMBiSQ0tjnuVWA4MuDBYIQ==", "requires": { - "tslib": "1.10.0" + "tslib": "^1.9.0" } }, "@angular/platform-browser-dynamic": { @@ -640,7 +640,7 @@ "resolved": "https://registry.npmjs.org/@angular/platform-browser-dynamic/-/platform-browser-dynamic-8.2.9.tgz", "integrity": "sha512-GbE4TUy4n/a8yp8fLWwdG/QnjUPZZ8VufItZ7GvOpoyknzegvka111dLctvMoPzSAsrKyShL6cryuyDC5PShUA==", "requires": { - "tslib": "1.10.0" + "tslib": "^1.9.0" } }, "@angular/platform-server": { @@ -648,9 +648,9 @@ "resolved": "https://registry.npmjs.org/@angular/platform-server/-/platform-server-8.2.9.tgz", "integrity": "sha512-rr6h82+DdUGhpsF3WT3eLk5itjZDXe7SiNtRGHkPj+yTyFAxuTKA3cX0N7LWsGGIFax+s1vQhMreV4YcyHKGPQ==", "requires": { - "domino": "2.1.3", - "tslib": "1.10.0", - "xhr2": "0.1.4" + "domino": "^2.1.2", + "tslib": "^1.9.0", + "xhr2": "^0.1.4" } }, "@angular/router": { @@ -658,7 +658,7 @@ "resolved": "https://registry.npmjs.org/@angular/router/-/router-8.2.9.tgz", "integrity": "sha512-4P60CWNB/jxGjDBEuYN0Jobt76QlebAQeFBTDswRVwRlq/WJT4QhL3a8AVIRsHn9bQII0LUt/ZQBBPxn7h9lSA==", "requires": { - "tslib": "1.10.0" + "tslib": "^1.9.0" } }, "@babel/code-frame": { @@ -666,7 +666,7 @@ "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.5.5.tgz", "integrity": "sha512-27d4lZoomVyo51VegxI20xZPuSHusqbQag/ztrBC7wegWoQ1nLREPVSKSW8byhTlzTKyNE4ifaTA6lCp7JjpFw==", "requires": { - "@babel/highlight": "7.5.0" + "@babel/highlight": "^7.0.0" } }, "@babel/core": { @@ -674,20 +674,20 @@ "resolved": "https://registry.npmjs.org/@babel/core/-/core-7.5.5.tgz", "integrity": "sha512-i4qoSr2KTtce0DmkuuQBV4AuQgGPUcPXMr9L5MyYAtk06z068lQ10a4O009fe5OB/DfNV+h+qqT7ddNV8UnRjg==", "requires": { - "@babel/code-frame": "7.5.5", - "@babel/generator": "7.6.2", - "@babel/helpers": "7.6.2", - "@babel/parser": "7.6.2", - "@babel/template": "7.6.0", - "@babel/traverse": "7.6.2", - "@babel/types": "7.6.1", - "convert-source-map": "1.6.0", - "debug": "4.1.1", - "json5": "2.1.1", - "lodash": "4.17.15", - "resolve": "1.12.0", - "semver": "5.6.0", - "source-map": "0.5.6" + "@babel/code-frame": "^7.5.5", + "@babel/generator": "^7.5.5", + "@babel/helpers": "^7.5.5", + "@babel/parser": "^7.5.5", + "@babel/template": "^7.4.4", + "@babel/traverse": "^7.5.5", + "@babel/types": "^7.5.5", + "convert-source-map": "^1.1.0", + "debug": "^4.1.0", + "json5": "^2.1.0", + "lodash": "^4.17.13", + "resolve": "^1.3.2", + "semver": "^5.4.1", + "source-map": "^0.5.0" }, "dependencies": { "debug": { @@ -695,7 +695,7 @@ "resolved": "https://registry.npmjs.org/debug/-/debug-4.1.1.tgz", "integrity": "sha512-pYAIzeRo8J6KPEaJ0VWOh5Pzkbw/RetuzehGM7QRRX5he4fPHx2rdKMB256ehJCkX+XRQm16eZLqLNS8RSZXZw==", "requires": { - "ms": "2.1.2" + "ms": "^2.1.1" } }, "json5": { @@ -703,7 +703,7 @@ "resolved": "https://registry.npmjs.org/json5/-/json5-2.1.1.tgz", "integrity": "sha512-l+3HXD0GEI3huGq1njuqtzYK8OYJyXMkOLtQ53pjWh89tvWS2h6l+1zMkYWqlb57+SiQodKZyvMEFb2X+KrFhQ==", "requires": { - "minimist": "1.2.0" + "minimist": "^1.2.0" } }, "ms": { @@ -718,10 +718,10 @@ "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.6.2.tgz", "integrity": "sha512-j8iHaIW4gGPnViaIHI7e9t/Hl8qLjERI6DcV9kEpAIDJsAOrcnXqRS7t+QbhL76pwbtqP+QCQLL0z1CyVmtjjQ==", "requires": { - "@babel/types": "7.6.1", - "jsesc": "2.5.2", - "lodash": "4.17.15", - "source-map": "0.5.6" + "@babel/types": "^7.6.0", + "jsesc": "^2.5.1", + "lodash": "^4.17.13", + "source-map": "^0.5.0" }, "dependencies": { "jsesc": { @@ -736,7 +736,7 @@ "resolved": "https://registry.npmjs.org/@babel/helper-annotate-as-pure/-/helper-annotate-as-pure-7.0.0.tgz", "integrity": "sha512-3UYcJUj9kvSLbLbUIfQTqzcy5VX7GRZ/CCDrnOaZorFFM01aXp1+GJwuFGV4NDDoAS+mOUyHcO6UD/RfqOks3Q==", "requires": { - "@babel/types": "7.6.1" + "@babel/types": "^7.0.0" } }, "@babel/helper-builder-binary-assignment-operator-visitor": { @@ -744,8 +744,8 @@ "resolved": "https://registry.npmjs.org/@babel/helper-builder-binary-assignment-operator-visitor/-/helper-builder-binary-assignment-operator-visitor-7.1.0.tgz", "integrity": "sha512-qNSR4jrmJ8M1VMM9tibvyRAHXQs2PmaksQF7c1CGJNipfe3D8p+wgNwgso/P2A2r2mdgBWAXljNWR0QRZAMW8w==", "requires": { - "@babel/helper-explode-assignable-expression": "7.1.0", - "@babel/types": "7.6.1" + "@babel/helper-explode-assignable-expression": "^7.1.0", + "@babel/types": "^7.0.0" } }, "@babel/helper-call-delegate": { @@ -753,9 +753,9 @@ "resolved": "https://registry.npmjs.org/@babel/helper-call-delegate/-/helper-call-delegate-7.4.4.tgz", "integrity": "sha512-l79boDFJ8S1c5hvQvG+rc+wHw6IuH7YldmRKsYtpbawsxURu/paVy57FZMomGK22/JckepaikOkY0MoAmdyOlQ==", "requires": { - "@babel/helper-hoist-variables": "7.4.4", - "@babel/traverse": "7.6.2", - "@babel/types": "7.6.1" + "@babel/helper-hoist-variables": "^7.4.4", + "@babel/traverse": "^7.4.4", + "@babel/types": "^7.4.4" } }, "@babel/helper-define-map": { @@ -763,9 +763,9 @@ "resolved": "https://registry.npmjs.org/@babel/helper-define-map/-/helper-define-map-7.5.5.tgz", "integrity": "sha512-fTfxx7i0B5NJqvUOBBGREnrqbTxRh7zinBANpZXAVDlsZxYdclDp467G1sQ8VZYMnAURY3RpBUAgOYT9GfzHBg==", "requires": { - "@babel/helper-function-name": "7.1.0", - "@babel/types": "7.6.1", - "lodash": "4.17.15" + "@babel/helper-function-name": "^7.1.0", + "@babel/types": "^7.5.5", + "lodash": "^4.17.13" } }, "@babel/helper-explode-assignable-expression": { @@ -773,8 +773,8 @@ "resolved": "https://registry.npmjs.org/@babel/helper-explode-assignable-expression/-/helper-explode-assignable-expression-7.1.0.tgz", "integrity": "sha512-NRQpfHrJ1msCHtKjbzs9YcMmJZOg6mQMmGRB+hbamEdG5PNpaSm95275VD92DvJKuyl0s2sFiDmMZ+EnnvufqA==", "requires": { - "@babel/traverse": "7.6.2", - "@babel/types": "7.6.1" + "@babel/traverse": "^7.1.0", + "@babel/types": "^7.0.0" } }, "@babel/helper-function-name": { @@ -782,9 +782,9 @@ "resolved": "https://registry.npmjs.org/@babel/helper-function-name/-/helper-function-name-7.1.0.tgz", "integrity": "sha512-A95XEoCpb3TO+KZzJ4S/5uW5fNe26DjBGqf1o9ucyLyCmi1dXq/B3c8iaWTfBk3VvetUxl16e8tIrd5teOCfGw==", "requires": { - "@babel/helper-get-function-arity": "7.0.0", - "@babel/template": "7.6.0", - "@babel/types": "7.6.1" + "@babel/helper-get-function-arity": "^7.0.0", + "@babel/template": "^7.1.0", + "@babel/types": "^7.0.0" } }, "@babel/helper-get-function-arity": { @@ -792,7 +792,7 @@ "resolved": "https://registry.npmjs.org/@babel/helper-get-function-arity/-/helper-get-function-arity-7.0.0.tgz", "integrity": "sha512-r2DbJeg4svYvt3HOS74U4eWKsUAMRH01Z1ds1zx8KNTPtpTL5JAsdFv8BNyOpVqdFhHkkRDIg5B4AsxmkjAlmQ==", "requires": { - "@babel/types": "7.6.1" + "@babel/types": "^7.0.0" } }, "@babel/helper-hoist-variables": { @@ -800,7 +800,7 @@ "resolved": "https://registry.npmjs.org/@babel/helper-hoist-variables/-/helper-hoist-variables-7.4.4.tgz", "integrity": "sha512-VYk2/H/BnYbZDDg39hr3t2kKyifAm1W6zHRfhx8jGjIHpQEBv9dry7oQ2f3+J703TLu69nYdxsovl0XYfcnK4w==", "requires": { - "@babel/types": "7.6.1" + "@babel/types": "^7.4.4" } }, "@babel/helper-member-expression-to-functions": { @@ -808,7 +808,7 @@ "resolved": "https://registry.npmjs.org/@babel/helper-member-expression-to-functions/-/helper-member-expression-to-functions-7.5.5.tgz", "integrity": "sha512-5qZ3D1uMclSNqYcXqiHoA0meVdv+xUEex9em2fqMnrk/scphGlGgg66zjMrPJESPwrFJ6sbfFQYUSa0Mz7FabA==", "requires": { - "@babel/types": "7.6.1" + "@babel/types": "^7.5.5" } }, "@babel/helper-module-imports": { @@ -816,7 +816,7 @@ "resolved": "https://registry.npmjs.org/@babel/helper-module-imports/-/helper-module-imports-7.0.0.tgz", "integrity": "sha512-aP/hlLq01DWNEiDg4Jn23i+CXxW/owM4WpDLFUbpjxe4NS3BhLVZQ5i7E0ZrxuQ/vwekIeciyamgB1UIYxxM6A==", "requires": { - "@babel/types": "7.6.1" + "@babel/types": "^7.0.0" } }, "@babel/helper-module-transforms": { @@ -824,12 +824,12 @@ "resolved": "https://registry.npmjs.org/@babel/helper-module-transforms/-/helper-module-transforms-7.5.5.tgz", "integrity": "sha512-jBeCvETKuJqeiaCdyaheF40aXnnU1+wkSiUs/IQg3tB85up1LyL8x77ClY8qJpuRJUcXQo+ZtdNESmZl4j56Pw==", "requires": { - "@babel/helper-module-imports": "7.0.0", - "@babel/helper-simple-access": "7.1.0", - "@babel/helper-split-export-declaration": "7.4.4", - "@babel/template": "7.6.0", - "@babel/types": "7.6.1", - "lodash": "4.17.15" + "@babel/helper-module-imports": "^7.0.0", + "@babel/helper-simple-access": "^7.1.0", + "@babel/helper-split-export-declaration": "^7.4.4", + "@babel/template": "^7.4.4", + "@babel/types": "^7.5.5", + "lodash": "^4.17.13" } }, "@babel/helper-optimise-call-expression": { @@ -837,7 +837,7 @@ "resolved": "https://registry.npmjs.org/@babel/helper-optimise-call-expression/-/helper-optimise-call-expression-7.0.0.tgz", "integrity": "sha512-u8nd9NQePYNQV8iPWu/pLLYBqZBa4ZaY1YWRFMuxrid94wKI1QNt67NEZ7GAe5Kc/0LLScbim05xZFWkAdrj9g==", "requires": { - "@babel/types": "7.6.1" + "@babel/types": "^7.0.0" } }, "@babel/helper-plugin-utils": { @@ -850,7 +850,7 @@ "resolved": "https://registry.npmjs.org/@babel/helper-regex/-/helper-regex-7.5.5.tgz", "integrity": "sha512-CkCYQLkfkiugbRDO8eZn6lRuR8kzZoGXCg3149iTk5se7g6qykSpy3+hELSwquhu+TgHn8nkLiBwHvNX8Hofcw==", "requires": { - "lodash": "4.17.15" + "lodash": "^4.17.13" } }, "@babel/helper-remap-async-to-generator": { @@ -858,11 +858,11 @@ "resolved": "https://registry.npmjs.org/@babel/helper-remap-async-to-generator/-/helper-remap-async-to-generator-7.1.0.tgz", "integrity": "sha512-3fOK0L+Fdlg8S5al8u/hWE6vhufGSn0bN09xm2LXMy//REAF8kDCrYoOBKYmA8m5Nom+sV9LyLCwrFynA8/slg==", "requires": { - "@babel/helper-annotate-as-pure": "7.0.0", - "@babel/helper-wrap-function": "7.2.0", - "@babel/template": "7.6.0", - "@babel/traverse": "7.6.2", - "@babel/types": "7.6.1" + "@babel/helper-annotate-as-pure": "^7.0.0", + "@babel/helper-wrap-function": "^7.1.0", + "@babel/template": "^7.1.0", + "@babel/traverse": "^7.1.0", + "@babel/types": "^7.0.0" } }, "@babel/helper-replace-supers": { @@ -870,10 +870,10 @@ "resolved": "https://registry.npmjs.org/@babel/helper-replace-supers/-/helper-replace-supers-7.5.5.tgz", "integrity": "sha512-XvRFWrNnlsow2u7jXDuH4jDDctkxbS7gXssrP4q2nUD606ukXHRvydj346wmNg+zAgpFx4MWf4+usfC93bElJg==", "requires": { - "@babel/helper-member-expression-to-functions": "7.5.5", - "@babel/helper-optimise-call-expression": "7.0.0", - "@babel/traverse": "7.6.2", - "@babel/types": "7.6.1" + "@babel/helper-member-expression-to-functions": "^7.5.5", + "@babel/helper-optimise-call-expression": "^7.0.0", + "@babel/traverse": "^7.5.5", + "@babel/types": "^7.5.5" } }, "@babel/helper-simple-access": { @@ -881,8 +881,8 @@ "resolved": "https://registry.npmjs.org/@babel/helper-simple-access/-/helper-simple-access-7.1.0.tgz", "integrity": "sha512-Vk+78hNjRbsiu49zAPALxTb+JUQCz1aolpd8osOF16BGnLtseD21nbHgLPGUwrXEurZgiCOUmvs3ExTu4F5x6w==", "requires": { - "@babel/template": "7.6.0", - "@babel/types": "7.6.1" + "@babel/template": "^7.1.0", + "@babel/types": "^7.0.0" } }, "@babel/helper-split-export-declaration": { @@ -890,7 +890,7 @@ "resolved": "https://registry.npmjs.org/@babel/helper-split-export-declaration/-/helper-split-export-declaration-7.4.4.tgz", "integrity": "sha512-Ro/XkzLf3JFITkW6b+hNxzZ1n5OQ80NvIUdmHspih1XAhtN3vPTuUFT4eQnela+2MaZ5ulH+iyP513KJrxbN7Q==", "requires": { - "@babel/types": "7.6.1" + "@babel/types": "^7.4.4" } }, "@babel/helper-wrap-function": { @@ -898,10 +898,10 @@ "resolved": "https://registry.npmjs.org/@babel/helper-wrap-function/-/helper-wrap-function-7.2.0.tgz", "integrity": "sha512-o9fP1BZLLSrYlxYEYyl2aS+Flun5gtjTIG8iln+XuEzQTs0PLagAGSXUcqruJwD5fM48jzIEggCKpIfWTcR7pQ==", "requires": { - "@babel/helper-function-name": "7.1.0", - "@babel/template": "7.6.0", - "@babel/traverse": "7.6.2", - "@babel/types": "7.6.1" + "@babel/helper-function-name": "^7.1.0", + "@babel/template": "^7.1.0", + "@babel/traverse": "^7.1.0", + "@babel/types": "^7.2.0" } }, "@babel/helpers": { @@ -909,9 +909,9 @@ "resolved": "https://registry.npmjs.org/@babel/helpers/-/helpers-7.6.2.tgz", "integrity": "sha512-3/bAUL8zZxYs1cdX2ilEE0WobqbCmKWr/889lf2SS0PpDcpEIY8pb1CCyz0pEcX3pEb+MCbks1jIokz2xLtGTA==", "requires": { - "@babel/template": "7.6.0", - "@babel/traverse": "7.6.2", - "@babel/types": "7.6.1" + "@babel/template": "^7.6.0", + "@babel/traverse": "^7.6.2", + "@babel/types": "^7.6.0" } }, "@babel/highlight": { @@ -919,9 +919,9 @@ "resolved": "https://registry.npmjs.org/@babel/highlight/-/highlight-7.5.0.tgz", "integrity": "sha512-7dV4eu9gBxoM0dAnj/BCFDW9LFU0zvTrkq0ugM7pnHEgguOEeOz1so2ZghEdzviYzQEED0r4EAgpsBChKy1TRQ==", "requires": { - "chalk": "2.4.2", - "esutils": "2.0.3", - "js-tokens": "4.0.0" + "chalk": "^2.0.0", + "esutils": "^2.0.2", + "js-tokens": "^4.0.0" }, "dependencies": { "js-tokens": { @@ -941,9 +941,9 @@ "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-async-generator-functions/-/plugin-proposal-async-generator-functions-7.2.0.tgz", "integrity": "sha512-+Dfo/SCQqrwx48ptLVGLdE39YtWRuKc/Y9I5Fy0P1DDBB9lsAHpjcEJQt+4IifuSOSTLBKJObJqMvaO1pIE8LQ==", "requires": { - "@babel/helper-plugin-utils": "7.0.0", - "@babel/helper-remap-async-to-generator": "7.1.0", - "@babel/plugin-syntax-async-generators": "7.2.0" + "@babel/helper-plugin-utils": "^7.0.0", + "@babel/helper-remap-async-to-generator": "^7.1.0", + "@babel/plugin-syntax-async-generators": "^7.2.0" } }, "@babel/plugin-proposal-dynamic-import": { @@ -951,8 +951,8 @@ "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-dynamic-import/-/plugin-proposal-dynamic-import-7.5.0.tgz", "integrity": "sha512-x/iMjggsKTFHYC6g11PL7Qy58IK8H5zqfm9e6hu4z1iH2IRyAp9u9dL80zA6R76yFovETFLKz2VJIC2iIPBuFw==", "requires": { - "@babel/helper-plugin-utils": "7.0.0", - "@babel/plugin-syntax-dynamic-import": "7.2.0" + "@babel/helper-plugin-utils": "^7.0.0", + "@babel/plugin-syntax-dynamic-import": "^7.2.0" } }, "@babel/plugin-proposal-json-strings": { @@ -960,8 +960,8 @@ "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-json-strings/-/plugin-proposal-json-strings-7.2.0.tgz", "integrity": "sha512-MAFV1CA/YVmYwZG0fBQyXhmj0BHCB5egZHCKWIFVv/XCxAeVGIHfos3SwDck4LvCllENIAg7xMKOG5kH0dzyUg==", "requires": { - "@babel/helper-plugin-utils": "7.0.0", - "@babel/plugin-syntax-json-strings": "7.2.0" + "@babel/helper-plugin-utils": "^7.0.0", + "@babel/plugin-syntax-json-strings": "^7.2.0" } }, "@babel/plugin-proposal-object-rest-spread": { @@ -969,8 +969,8 @@ "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-object-rest-spread/-/plugin-proposal-object-rest-spread-7.6.2.tgz", "integrity": "sha512-LDBXlmADCsMZV1Y9OQwMc0MyGZ8Ta/zlD9N67BfQT8uYwkRswiu2hU6nJKrjrt/58aH/vqfQlR/9yId/7A2gWw==", "requires": { - "@babel/helper-plugin-utils": "7.0.0", - "@babel/plugin-syntax-object-rest-spread": "7.2.0" + "@babel/helper-plugin-utils": "^7.0.0", + "@babel/plugin-syntax-object-rest-spread": "^7.2.0" } }, "@babel/plugin-proposal-optional-catch-binding": { @@ -978,8 +978,8 @@ "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-optional-catch-binding/-/plugin-proposal-optional-catch-binding-7.2.0.tgz", "integrity": "sha512-mgYj3jCcxug6KUcX4OBoOJz3CMrwRfQELPQ5560F70YQUBZB7uac9fqaWamKR1iWUzGiK2t0ygzjTScZnVz75g==", "requires": { - "@babel/helper-plugin-utils": "7.0.0", - "@babel/plugin-syntax-optional-catch-binding": "7.2.0" + "@babel/helper-plugin-utils": "^7.0.0", + "@babel/plugin-syntax-optional-catch-binding": "^7.2.0" } }, "@babel/plugin-proposal-unicode-property-regex": { @@ -987,9 +987,9 @@ "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-unicode-property-regex/-/plugin-proposal-unicode-property-regex-7.6.2.tgz", "integrity": "sha512-NxHETdmpeSCtiatMRYWVJo7266rrvAC3DTeG5exQBIH/fMIUK7ejDNznBbn3HQl/o9peymRRg7Yqkx6PdUXmMw==", "requires": { - "@babel/helper-plugin-utils": "7.0.0", - "@babel/helper-regex": "7.5.5", - "regexpu-core": "4.6.0" + "@babel/helper-plugin-utils": "^7.0.0", + "@babel/helper-regex": "^7.4.4", + "regexpu-core": "^4.6.0" } }, "@babel/plugin-syntax-async-generators": { @@ -997,7 +997,7 @@ "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-async-generators/-/plugin-syntax-async-generators-7.2.0.tgz", "integrity": "sha512-1ZrIRBv2t0GSlcwVoQ6VgSLpLgiN/FVQUzt9znxo7v2Ov4jJrs8RY8tv0wvDmFN3qIdMKWrmMMW6yZ0G19MfGg==", "requires": { - "@babel/helper-plugin-utils": "7.0.0" + "@babel/helper-plugin-utils": "^7.0.0" } }, "@babel/plugin-syntax-dynamic-import": { @@ -1005,7 +1005,7 @@ "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-dynamic-import/-/plugin-syntax-dynamic-import-7.2.0.tgz", "integrity": "sha512-mVxuJ0YroI/h/tbFTPGZR8cv6ai+STMKNBq0f8hFxsxWjl94qqhsb+wXbpNMDPU3cfR1TIsVFzU3nXyZMqyK4w==", "requires": { - "@babel/helper-plugin-utils": "7.0.0" + "@babel/helper-plugin-utils": "^7.0.0" } }, "@babel/plugin-syntax-json-strings": { @@ -1013,7 +1013,7 @@ "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-json-strings/-/plugin-syntax-json-strings-7.2.0.tgz", "integrity": "sha512-5UGYnMSLRE1dqqZwug+1LISpA403HzlSfsg6P9VXU6TBjcSHeNlw4DxDx7LgpF+iKZoOG/+uzqoRHTdcUpiZNg==", "requires": { - "@babel/helper-plugin-utils": "7.0.0" + "@babel/helper-plugin-utils": "^7.0.0" } }, "@babel/plugin-syntax-object-rest-spread": { @@ -1021,7 +1021,7 @@ "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-object-rest-spread/-/plugin-syntax-object-rest-spread-7.2.0.tgz", "integrity": "sha512-t0JKGgqk2We+9may3t0xDdmneaXmyxq0xieYcKHxIsrJO64n1OiMWNUtc5gQK1PA0NpdCRrtZp4z+IUaKugrSA==", "requires": { - "@babel/helper-plugin-utils": "7.0.0" + "@babel/helper-plugin-utils": "^7.0.0" } }, "@babel/plugin-syntax-optional-catch-binding": { @@ -1029,7 +1029,7 @@ "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-optional-catch-binding/-/plugin-syntax-optional-catch-binding-7.2.0.tgz", "integrity": "sha512-bDe4xKNhb0LI7IvZHiA13kff0KEfaGX/Hv4lMA9+7TEc63hMNvfKo6ZFpXhKuEp+II/q35Gc4NoMeDZyaUbj9w==", "requires": { - "@babel/helper-plugin-utils": "7.0.0" + "@babel/helper-plugin-utils": "^7.0.0" } }, "@babel/plugin-transform-arrow-functions": { @@ -1037,7 +1037,7 @@ "resolved": "https://registry.npmjs.org/@babel/plugin-transform-arrow-functions/-/plugin-transform-arrow-functions-7.2.0.tgz", "integrity": "sha512-ER77Cax1+8/8jCB9fo4Ud161OZzWN5qawi4GusDuRLcDbDG+bIGYY20zb2dfAFdTRGzrfq2xZPvF0R64EHnimg==", "requires": { - "@babel/helper-plugin-utils": "7.0.0" + "@babel/helper-plugin-utils": "^7.0.0" } }, "@babel/plugin-transform-async-to-generator": { @@ -1045,9 +1045,9 @@ "resolved": "https://registry.npmjs.org/@babel/plugin-transform-async-to-generator/-/plugin-transform-async-to-generator-7.5.0.tgz", "integrity": "sha512-mqvkzwIGkq0bEF1zLRRiTdjfomZJDV33AH3oQzHVGkI2VzEmXLpKKOBvEVaFZBJdN0XTyH38s9j/Kiqr68dggg==", "requires": { - "@babel/helper-module-imports": "7.0.0", - "@babel/helper-plugin-utils": "7.0.0", - "@babel/helper-remap-async-to-generator": "7.1.0" + "@babel/helper-module-imports": "^7.0.0", + "@babel/helper-plugin-utils": "^7.0.0", + "@babel/helper-remap-async-to-generator": "^7.1.0" } }, "@babel/plugin-transform-block-scoped-functions": { @@ -1055,7 +1055,7 @@ "resolved": "https://registry.npmjs.org/@babel/plugin-transform-block-scoped-functions/-/plugin-transform-block-scoped-functions-7.2.0.tgz", "integrity": "sha512-ntQPR6q1/NKuphly49+QiQiTN0O63uOwjdD6dhIjSWBI5xlrbUFh720TIpzBhpnrLfv2tNH/BXvLIab1+BAI0w==", "requires": { - "@babel/helper-plugin-utils": "7.0.0" + "@babel/helper-plugin-utils": "^7.0.0" } }, "@babel/plugin-transform-block-scoping": { @@ -1063,8 +1063,8 @@ "resolved": "https://registry.npmjs.org/@babel/plugin-transform-block-scoping/-/plugin-transform-block-scoping-7.6.2.tgz", "integrity": "sha512-zZT8ivau9LOQQaOGC7bQLQOT4XPkPXgN2ERfUgk1X8ql+mVkLc4E8eKk+FO3o0154kxzqenWCorfmEXpEZcrSQ==", "requires": { - "@babel/helper-plugin-utils": "7.0.0", - "lodash": "4.17.15" + "@babel/helper-plugin-utils": "^7.0.0", + "lodash": "^4.17.13" } }, "@babel/plugin-transform-classes": { @@ -1072,14 +1072,14 @@ "resolved": "https://registry.npmjs.org/@babel/plugin-transform-classes/-/plugin-transform-classes-7.5.5.tgz", "integrity": "sha512-U2htCNK/6e9K7jGyJ++1p5XRU+LJjrwtoiVn9SzRlDT2KubcZ11OOwy3s24TjHxPgxNwonCYP7U2K51uVYCMDg==", "requires": { - "@babel/helper-annotate-as-pure": "7.0.0", - "@babel/helper-define-map": "7.5.5", - "@babel/helper-function-name": "7.1.0", - "@babel/helper-optimise-call-expression": "7.0.0", - "@babel/helper-plugin-utils": "7.0.0", - "@babel/helper-replace-supers": "7.5.5", - "@babel/helper-split-export-declaration": "7.4.4", - "globals": "11.12.0" + "@babel/helper-annotate-as-pure": "^7.0.0", + "@babel/helper-define-map": "^7.5.5", + "@babel/helper-function-name": "^7.1.0", + "@babel/helper-optimise-call-expression": "^7.0.0", + "@babel/helper-plugin-utils": "^7.0.0", + "@babel/helper-replace-supers": "^7.5.5", + "@babel/helper-split-export-declaration": "^7.4.4", + "globals": "^11.1.0" }, "dependencies": { "globals": { @@ -1094,7 +1094,7 @@ "resolved": "https://registry.npmjs.org/@babel/plugin-transform-computed-properties/-/plugin-transform-computed-properties-7.2.0.tgz", "integrity": "sha512-kP/drqTxY6Xt3NNpKiMomfgkNn4o7+vKxK2DDKcBG9sHj51vHqMBGy8wbDS/J4lMxnqs153/T3+DmCEAkC5cpA==", "requires": { - "@babel/helper-plugin-utils": "7.0.0" + "@babel/helper-plugin-utils": "^7.0.0" } }, "@babel/plugin-transform-destructuring": { @@ -1102,7 +1102,7 @@ "resolved": "https://registry.npmjs.org/@babel/plugin-transform-destructuring/-/plugin-transform-destructuring-7.6.0.tgz", "integrity": "sha512-2bGIS5P1v4+sWTCnKNDZDxbGvEqi0ijeqM/YqHtVGrvG2y0ySgnEEhXErvE9dA0bnIzY9bIzdFK0jFA46ASIIQ==", "requires": { - "@babel/helper-plugin-utils": "7.0.0" + "@babel/helper-plugin-utils": "^7.0.0" } }, "@babel/plugin-transform-dotall-regex": { @@ -1110,9 +1110,9 @@ "resolved": "https://registry.npmjs.org/@babel/plugin-transform-dotall-regex/-/plugin-transform-dotall-regex-7.6.2.tgz", "integrity": "sha512-KGKT9aqKV+9YMZSkowzYoYEiHqgaDhGmPNZlZxX6UeHC4z30nC1J9IrZuGqbYFB1jaIGdv91ujpze0exiVK8bA==", "requires": { - "@babel/helper-plugin-utils": "7.0.0", - "@babel/helper-regex": "7.5.5", - "regexpu-core": "4.6.0" + "@babel/helper-plugin-utils": "^7.0.0", + "@babel/helper-regex": "^7.4.4", + "regexpu-core": "^4.6.0" } }, "@babel/plugin-transform-duplicate-keys": { @@ -1120,7 +1120,7 @@ "resolved": "https://registry.npmjs.org/@babel/plugin-transform-duplicate-keys/-/plugin-transform-duplicate-keys-7.5.0.tgz", "integrity": "sha512-igcziksHizyQPlX9gfSjHkE2wmoCH3evvD2qR5w29/Dk0SMKE/eOI7f1HhBdNhR/zxJDqrgpoDTq5YSLH/XMsQ==", "requires": { - "@babel/helper-plugin-utils": "7.0.0" + "@babel/helper-plugin-utils": "^7.0.0" } }, "@babel/plugin-transform-exponentiation-operator": { @@ -1128,8 +1128,8 @@ "resolved": "https://registry.npmjs.org/@babel/plugin-transform-exponentiation-operator/-/plugin-transform-exponentiation-operator-7.2.0.tgz", "integrity": "sha512-umh4hR6N7mu4Elq9GG8TOu9M0bakvlsREEC+ialrQN6ABS4oDQ69qJv1VtR3uxlKMCQMCvzk7vr17RHKcjx68A==", "requires": { - "@babel/helper-builder-binary-assignment-operator-visitor": "7.1.0", - "@babel/helper-plugin-utils": "7.0.0" + "@babel/helper-builder-binary-assignment-operator-visitor": "^7.1.0", + "@babel/helper-plugin-utils": "^7.0.0" } }, "@babel/plugin-transform-for-of": { @@ -1137,7 +1137,7 @@ "resolved": "https://registry.npmjs.org/@babel/plugin-transform-for-of/-/plugin-transform-for-of-7.4.4.tgz", "integrity": "sha512-9T/5Dlr14Z9TIEXLXkt8T1DU7F24cbhwhMNUziN3hB1AXoZcdzPcTiKGRn/6iOymDqtTKWnr/BtRKN9JwbKtdQ==", "requires": { - "@babel/helper-plugin-utils": "7.0.0" + "@babel/helper-plugin-utils": "^7.0.0" } }, "@babel/plugin-transform-function-name": { @@ -1145,8 +1145,8 @@ "resolved": "https://registry.npmjs.org/@babel/plugin-transform-function-name/-/plugin-transform-function-name-7.4.4.tgz", "integrity": "sha512-iU9pv7U+2jC9ANQkKeNF6DrPy4GBa4NWQtl6dHB4Pb3izX2JOEvDTFarlNsBj/63ZEzNNIAMs3Qw4fNCcSOXJA==", "requires": { - "@babel/helper-function-name": "7.1.0", - "@babel/helper-plugin-utils": "7.0.0" + "@babel/helper-function-name": "^7.1.0", + "@babel/helper-plugin-utils": "^7.0.0" } }, "@babel/plugin-transform-literals": { @@ -1154,7 +1154,7 @@ "resolved": "https://registry.npmjs.org/@babel/plugin-transform-literals/-/plugin-transform-literals-7.2.0.tgz", "integrity": "sha512-2ThDhm4lI4oV7fVQ6pNNK+sx+c/GM5/SaML0w/r4ZB7sAneD/piDJtwdKlNckXeyGK7wlwg2E2w33C/Hh+VFCg==", "requires": { - "@babel/helper-plugin-utils": "7.0.0" + "@babel/helper-plugin-utils": "^7.0.0" } }, "@babel/plugin-transform-member-expression-literals": { @@ -1162,7 +1162,7 @@ "resolved": "https://registry.npmjs.org/@babel/plugin-transform-member-expression-literals/-/plugin-transform-member-expression-literals-7.2.0.tgz", "integrity": "sha512-HiU3zKkSU6scTidmnFJ0bMX8hz5ixC93b4MHMiYebmk2lUVNGOboPsqQvx5LzooihijUoLR/v7Nc1rbBtnc7FA==", "requires": { - "@babel/helper-plugin-utils": "7.0.0" + "@babel/helper-plugin-utils": "^7.0.0" } }, "@babel/plugin-transform-modules-amd": { @@ -1170,9 +1170,9 @@ "resolved": "https://registry.npmjs.org/@babel/plugin-transform-modules-amd/-/plugin-transform-modules-amd-7.5.0.tgz", "integrity": "sha512-n20UsQMKnWrltocZZm24cRURxQnWIvsABPJlw/fvoy9c6AgHZzoelAIzajDHAQrDpuKFFPPcFGd7ChsYuIUMpg==", "requires": { - "@babel/helper-module-transforms": "7.5.5", - "@babel/helper-plugin-utils": "7.0.0", - "babel-plugin-dynamic-import-node": "2.3.0" + "@babel/helper-module-transforms": "^7.1.0", + "@babel/helper-plugin-utils": "^7.0.0", + "babel-plugin-dynamic-import-node": "^2.3.0" } }, "@babel/plugin-transform-modules-commonjs": { @@ -1180,10 +1180,10 @@ "resolved": "https://registry.npmjs.org/@babel/plugin-transform-modules-commonjs/-/plugin-transform-modules-commonjs-7.6.0.tgz", "integrity": "sha512-Ma93Ix95PNSEngqomy5LSBMAQvYKVe3dy+JlVJSHEXZR5ASL9lQBedMiCyVtmTLraIDVRE3ZjTZvmXXD2Ozw3g==", "requires": { - "@babel/helper-module-transforms": "7.5.5", - "@babel/helper-plugin-utils": "7.0.0", - "@babel/helper-simple-access": "7.1.0", - "babel-plugin-dynamic-import-node": "2.3.0" + "@babel/helper-module-transforms": "^7.4.4", + "@babel/helper-plugin-utils": "^7.0.0", + "@babel/helper-simple-access": "^7.1.0", + "babel-plugin-dynamic-import-node": "^2.3.0" } }, "@babel/plugin-transform-modules-systemjs": { @@ -1191,9 +1191,9 @@ "resolved": "https://registry.npmjs.org/@babel/plugin-transform-modules-systemjs/-/plugin-transform-modules-systemjs-7.5.0.tgz", "integrity": "sha512-Q2m56tyoQWmuNGxEtUyeEkm6qJYFqs4c+XyXH5RAuYxObRNz9Zgj/1g2GMnjYp2EUyEy7YTrxliGCXzecl/vJg==", "requires": { - "@babel/helper-hoist-variables": "7.4.4", - "@babel/helper-plugin-utils": "7.0.0", - "babel-plugin-dynamic-import-node": "2.3.0" + "@babel/helper-hoist-variables": "^7.4.4", + "@babel/helper-plugin-utils": "^7.0.0", + "babel-plugin-dynamic-import-node": "^2.3.0" } }, "@babel/plugin-transform-modules-umd": { @@ -1201,8 +1201,8 @@ "resolved": "https://registry.npmjs.org/@babel/plugin-transform-modules-umd/-/plugin-transform-modules-umd-7.2.0.tgz", "integrity": "sha512-BV3bw6MyUH1iIsGhXlOK6sXhmSarZjtJ/vMiD9dNmpY8QXFFQTj+6v92pcfy1iqa8DeAfJFwoxcrS/TUZda6sw==", "requires": { - "@babel/helper-module-transforms": "7.5.5", - "@babel/helper-plugin-utils": "7.0.0" + "@babel/helper-module-transforms": "^7.1.0", + "@babel/helper-plugin-utils": "^7.0.0" } }, "@babel/plugin-transform-named-capturing-groups-regex": { @@ -1210,7 +1210,7 @@ "resolved": "https://registry.npmjs.org/@babel/plugin-transform-named-capturing-groups-regex/-/plugin-transform-named-capturing-groups-regex-7.6.2.tgz", "integrity": "sha512-xBdB+XOs+lgbZc2/4F5BVDVcDNS4tcSKQc96KmlqLEAwz6tpYPEvPdmDfvVG0Ssn8lAhronaRs6Z6KSexIpK5g==", "requires": { - "regexpu-core": "4.6.0" + "regexpu-core": "^4.6.0" } }, "@babel/plugin-transform-new-target": { @@ -1218,7 +1218,7 @@ "resolved": "https://registry.npmjs.org/@babel/plugin-transform-new-target/-/plugin-transform-new-target-7.4.4.tgz", "integrity": "sha512-r1z3T2DNGQwwe2vPGZMBNjioT2scgWzK9BCnDEh+46z8EEwXBq24uRzd65I7pjtugzPSj921aM15RpESgzsSuA==", "requires": { - "@babel/helper-plugin-utils": "7.0.0" + "@babel/helper-plugin-utils": "^7.0.0" } }, "@babel/plugin-transform-object-super": { @@ -1226,8 +1226,8 @@ "resolved": "https://registry.npmjs.org/@babel/plugin-transform-object-super/-/plugin-transform-object-super-7.5.5.tgz", "integrity": "sha512-un1zJQAhSosGFBduPgN/YFNvWVpRuHKU7IHBglLoLZsGmruJPOo6pbInneflUdmq7YvSVqhpPs5zdBvLnteltQ==", "requires": { - "@babel/helper-plugin-utils": "7.0.0", - "@babel/helper-replace-supers": "7.5.5" + "@babel/helper-plugin-utils": "^7.0.0", + "@babel/helper-replace-supers": "^7.5.5" } }, "@babel/plugin-transform-parameters": { @@ -1235,9 +1235,9 @@ "resolved": "https://registry.npmjs.org/@babel/plugin-transform-parameters/-/plugin-transform-parameters-7.4.4.tgz", "integrity": "sha512-oMh5DUO1V63nZcu/ZVLQFqiihBGo4OpxJxR1otF50GMeCLiRx5nUdtokd+u9SuVJrvvuIh9OosRFPP4pIPnwmw==", "requires": { - "@babel/helper-call-delegate": "7.4.4", - "@babel/helper-get-function-arity": "7.0.0", - "@babel/helper-plugin-utils": "7.0.0" + "@babel/helper-call-delegate": "^7.4.4", + "@babel/helper-get-function-arity": "^7.0.0", + "@babel/helper-plugin-utils": "^7.0.0" } }, "@babel/plugin-transform-property-literals": { @@ -1245,7 +1245,7 @@ "resolved": "https://registry.npmjs.org/@babel/plugin-transform-property-literals/-/plugin-transform-property-literals-7.2.0.tgz", "integrity": "sha512-9q7Dbk4RhgcLp8ebduOpCbtjh7C0itoLYHXd9ueASKAG/is5PQtMR5VJGka9NKqGhYEGn5ITahd4h9QeBMylWQ==", "requires": { - "@babel/helper-plugin-utils": "7.0.0" + "@babel/helper-plugin-utils": "^7.0.0" } }, "@babel/plugin-transform-regenerator": { @@ -1253,7 +1253,7 @@ "resolved": "https://registry.npmjs.org/@babel/plugin-transform-regenerator/-/plugin-transform-regenerator-7.4.5.tgz", "integrity": "sha512-gBKRh5qAaCWntnd09S8QC7r3auLCqq5DI6O0DlfoyDjslSBVqBibrMdsqO+Uhmx3+BlOmE/Kw1HFxmGbv0N9dA==", "requires": { - "regenerator-transform": "0.14.1" + "regenerator-transform": "^0.14.0" } }, "@babel/plugin-transform-reserved-words": { @@ -1261,7 +1261,7 @@ "resolved": "https://registry.npmjs.org/@babel/plugin-transform-reserved-words/-/plugin-transform-reserved-words-7.2.0.tgz", "integrity": "sha512-fz43fqW8E1tAB3DKF19/vxbpib1fuyCwSPE418ge5ZxILnBhWyhtPgz8eh1RCGGJlwvksHkyxMxh0eenFi+kFw==", "requires": { - "@babel/helper-plugin-utils": "7.0.0" + "@babel/helper-plugin-utils": "^7.0.0" } }, "@babel/plugin-transform-shorthand-properties": { @@ -1269,7 +1269,7 @@ "resolved": "https://registry.npmjs.org/@babel/plugin-transform-shorthand-properties/-/plugin-transform-shorthand-properties-7.2.0.tgz", "integrity": "sha512-QP4eUM83ha9zmYtpbnyjTLAGKQritA5XW/iG9cjtuOI8s1RuL/3V6a3DeSHfKutJQ+ayUfeZJPcnCYEQzaPQqg==", "requires": { - "@babel/helper-plugin-utils": "7.0.0" + "@babel/helper-plugin-utils": "^7.0.0" } }, "@babel/plugin-transform-spread": { @@ -1277,7 +1277,7 @@ "resolved": "https://registry.npmjs.org/@babel/plugin-transform-spread/-/plugin-transform-spread-7.6.2.tgz", "integrity": "sha512-DpSvPFryKdK1x+EDJYCy28nmAaIMdxmhot62jAXF/o99iA33Zj2Lmcp3vDmz+MUh0LNYVPvfj5iC3feb3/+PFg==", "requires": { - "@babel/helper-plugin-utils": "7.0.0" + "@babel/helper-plugin-utils": "^7.0.0" } }, "@babel/plugin-transform-sticky-regex": { @@ -1285,8 +1285,8 @@ "resolved": "https://registry.npmjs.org/@babel/plugin-transform-sticky-regex/-/plugin-transform-sticky-regex-7.2.0.tgz", "integrity": "sha512-KKYCoGaRAf+ckH8gEL3JHUaFVyNHKe3ASNsZ+AlktgHevvxGigoIttrEJb8iKN03Q7Eazlv1s6cx2B2cQ3Jabw==", "requires": { - "@babel/helper-plugin-utils": "7.0.0", - "@babel/helper-regex": "7.5.5" + "@babel/helper-plugin-utils": "^7.0.0", + "@babel/helper-regex": "^7.0.0" } }, "@babel/plugin-transform-template-literals": { @@ -1294,8 +1294,8 @@ "resolved": "https://registry.npmjs.org/@babel/plugin-transform-template-literals/-/plugin-transform-template-literals-7.4.4.tgz", "integrity": "sha512-mQrEC4TWkhLN0z8ygIvEL9ZEToPhG5K7KDW3pzGqOfIGZ28Jb0POUkeWcoz8HnHvhFy6dwAT1j8OzqN8s804+g==", "requires": { - "@babel/helper-annotate-as-pure": "7.0.0", - "@babel/helper-plugin-utils": "7.0.0" + "@babel/helper-annotate-as-pure": "^7.0.0", + "@babel/helper-plugin-utils": "^7.0.0" } }, "@babel/plugin-transform-typeof-symbol": { @@ -1303,7 +1303,7 @@ "resolved": "https://registry.npmjs.org/@babel/plugin-transform-typeof-symbol/-/plugin-transform-typeof-symbol-7.2.0.tgz", "integrity": "sha512-2LNhETWYxiYysBtrBTqL8+La0jIoQQnIScUJc74OYvUGRmkskNY4EzLCnjHBzdmb38wqtTaixpo1NctEcvMDZw==", "requires": { - "@babel/helper-plugin-utils": "7.0.0" + "@babel/helper-plugin-utils": "^7.0.0" } }, "@babel/plugin-transform-unicode-regex": { @@ -1311,9 +1311,9 @@ "resolved": "https://registry.npmjs.org/@babel/plugin-transform-unicode-regex/-/plugin-transform-unicode-regex-7.6.2.tgz", "integrity": "sha512-orZI6cWlR3nk2YmYdb0gImrgCUwb5cBUwjf6Ks6dvNVvXERkwtJWOQaEOjPiu0Gu1Tq6Yq/hruCZZOOi9F34Dw==", "requires": { - "@babel/helper-plugin-utils": "7.0.0", - "@babel/helper-regex": "7.5.5", - "regexpu-core": "4.6.0" + "@babel/helper-plugin-utils": "^7.0.0", + "@babel/helper-regex": "^7.4.4", + "regexpu-core": "^4.6.0" } }, "@babel/preset-env": { @@ -1321,56 +1321,56 @@ "resolved": "https://registry.npmjs.org/@babel/preset-env/-/preset-env-7.5.5.tgz", "integrity": "sha512-GMZQka/+INwsMz1A5UEql8tG015h5j/qjptpKY2gJ7giy8ohzU710YciJB5rcKsWGWHiW3RUnHib0E5/m3Tp3A==", "requires": { - "@babel/helper-module-imports": "7.0.0", - "@babel/helper-plugin-utils": "7.0.0", - "@babel/plugin-proposal-async-generator-functions": "7.2.0", - "@babel/plugin-proposal-dynamic-import": "7.5.0", - "@babel/plugin-proposal-json-strings": "7.2.0", - "@babel/plugin-proposal-object-rest-spread": "7.6.2", - "@babel/plugin-proposal-optional-catch-binding": "7.2.0", - "@babel/plugin-proposal-unicode-property-regex": "7.6.2", - "@babel/plugin-syntax-async-generators": "7.2.0", - "@babel/plugin-syntax-dynamic-import": "7.2.0", - "@babel/plugin-syntax-json-strings": "7.2.0", - "@babel/plugin-syntax-object-rest-spread": "7.2.0", - "@babel/plugin-syntax-optional-catch-binding": "7.2.0", - "@babel/plugin-transform-arrow-functions": "7.2.0", - "@babel/plugin-transform-async-to-generator": "7.5.0", - "@babel/plugin-transform-block-scoped-functions": "7.2.0", - "@babel/plugin-transform-block-scoping": "7.6.2", - "@babel/plugin-transform-classes": "7.5.5", - "@babel/plugin-transform-computed-properties": "7.2.0", - "@babel/plugin-transform-destructuring": "7.6.0", - "@babel/plugin-transform-dotall-regex": "7.6.2", - "@babel/plugin-transform-duplicate-keys": "7.5.0", - "@babel/plugin-transform-exponentiation-operator": "7.2.0", - "@babel/plugin-transform-for-of": "7.4.4", - "@babel/plugin-transform-function-name": "7.4.4", - "@babel/plugin-transform-literals": "7.2.0", - "@babel/plugin-transform-member-expression-literals": "7.2.0", - "@babel/plugin-transform-modules-amd": "7.5.0", - "@babel/plugin-transform-modules-commonjs": "7.6.0", - "@babel/plugin-transform-modules-systemjs": "7.5.0", - "@babel/plugin-transform-modules-umd": "7.2.0", - "@babel/plugin-transform-named-capturing-groups-regex": "7.6.2", - "@babel/plugin-transform-new-target": "7.4.4", - "@babel/plugin-transform-object-super": "7.5.5", - "@babel/plugin-transform-parameters": "7.4.4", - "@babel/plugin-transform-property-literals": "7.2.0", - "@babel/plugin-transform-regenerator": "7.4.5", - "@babel/plugin-transform-reserved-words": "7.2.0", - "@babel/plugin-transform-shorthand-properties": "7.2.0", - "@babel/plugin-transform-spread": "7.6.2", - "@babel/plugin-transform-sticky-regex": "7.2.0", - "@babel/plugin-transform-template-literals": "7.4.4", - "@babel/plugin-transform-typeof-symbol": "7.2.0", - "@babel/plugin-transform-unicode-regex": "7.6.2", - "@babel/types": "7.6.1", - "browserslist": "4.6.6", - "core-js-compat": "3.2.1", - "invariant": "2.2.4", - "js-levenshtein": "1.1.6", - "semver": "5.6.0" + "@babel/helper-module-imports": "^7.0.0", + "@babel/helper-plugin-utils": "^7.0.0", + "@babel/plugin-proposal-async-generator-functions": "^7.2.0", + "@babel/plugin-proposal-dynamic-import": "^7.5.0", + "@babel/plugin-proposal-json-strings": "^7.2.0", + "@babel/plugin-proposal-object-rest-spread": "^7.5.5", + "@babel/plugin-proposal-optional-catch-binding": "^7.2.0", + "@babel/plugin-proposal-unicode-property-regex": "^7.4.4", + "@babel/plugin-syntax-async-generators": "^7.2.0", + "@babel/plugin-syntax-dynamic-import": "^7.2.0", + "@babel/plugin-syntax-json-strings": "^7.2.0", + "@babel/plugin-syntax-object-rest-spread": "^7.2.0", + "@babel/plugin-syntax-optional-catch-binding": "^7.2.0", + "@babel/plugin-transform-arrow-functions": "^7.2.0", + "@babel/plugin-transform-async-to-generator": "^7.5.0", + "@babel/plugin-transform-block-scoped-functions": "^7.2.0", + "@babel/plugin-transform-block-scoping": "^7.5.5", + "@babel/plugin-transform-classes": "^7.5.5", + "@babel/plugin-transform-computed-properties": "^7.2.0", + "@babel/plugin-transform-destructuring": "^7.5.0", + "@babel/plugin-transform-dotall-regex": "^7.4.4", + "@babel/plugin-transform-duplicate-keys": "^7.5.0", + "@babel/plugin-transform-exponentiation-operator": "^7.2.0", + "@babel/plugin-transform-for-of": "^7.4.4", + "@babel/plugin-transform-function-name": "^7.4.4", + "@babel/plugin-transform-literals": "^7.2.0", + "@babel/plugin-transform-member-expression-literals": "^7.2.0", + "@babel/plugin-transform-modules-amd": "^7.5.0", + "@babel/plugin-transform-modules-commonjs": "^7.5.0", + "@babel/plugin-transform-modules-systemjs": "^7.5.0", + "@babel/plugin-transform-modules-umd": "^7.2.0", + "@babel/plugin-transform-named-capturing-groups-regex": "^7.4.5", + "@babel/plugin-transform-new-target": "^7.4.4", + "@babel/plugin-transform-object-super": "^7.5.5", + "@babel/plugin-transform-parameters": "^7.4.4", + "@babel/plugin-transform-property-literals": "^7.2.0", + "@babel/plugin-transform-regenerator": "^7.4.5", + "@babel/plugin-transform-reserved-words": "^7.2.0", + "@babel/plugin-transform-shorthand-properties": "^7.2.0", + "@babel/plugin-transform-spread": "^7.2.0", + "@babel/plugin-transform-sticky-regex": "^7.2.0", + "@babel/plugin-transform-template-literals": "^7.4.4", + "@babel/plugin-transform-typeof-symbol": "^7.2.0", + "@babel/plugin-transform-unicode-regex": "^7.4.4", + "@babel/types": "^7.5.5", + "browserslist": "^4.6.0", + "core-js-compat": "^3.1.1", + "invariant": "^2.2.2", + "js-levenshtein": "^1.1.3", + "semver": "^5.5.0" } }, "@babel/template": { @@ -1378,9 +1378,9 @@ "resolved": "https://registry.npmjs.org/@babel/template/-/template-7.6.0.tgz", "integrity": "sha512-5AEH2EXD8euCk446b7edmgFdub/qfH1SN6Nii3+fyXP807QRx9Q73A2N5hNwRRslC2H9sNzaFhsPubkS4L8oNQ==", "requires": { - "@babel/code-frame": "7.5.5", - "@babel/parser": "7.6.2", - "@babel/types": "7.6.1" + "@babel/code-frame": "^7.0.0", + "@babel/parser": "^7.6.0", + "@babel/types": "^7.6.0" } }, "@babel/traverse": { @@ -1388,15 +1388,15 @@ "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.6.2.tgz", "integrity": "sha512-8fRE76xNwNttVEF2TwxJDGBLWthUkHWSldmfuBzVRmEDWOtu4XdINTgN7TDWzuLg4bbeIMLvfMFD9we5YcWkRQ==", "requires": { - "@babel/code-frame": "7.5.5", - "@babel/generator": "7.6.2", - "@babel/helper-function-name": "7.1.0", - "@babel/helper-split-export-declaration": "7.4.4", - "@babel/parser": "7.6.2", - "@babel/types": "7.6.1", - "debug": "4.1.1", - "globals": "11.12.0", - "lodash": "4.17.15" + "@babel/code-frame": "^7.5.5", + "@babel/generator": "^7.6.2", + "@babel/helper-function-name": "^7.1.0", + "@babel/helper-split-export-declaration": "^7.4.4", + "@babel/parser": "^7.6.2", + "@babel/types": "^7.6.0", + "debug": "^4.1.0", + "globals": "^11.1.0", + "lodash": "^4.17.13" }, "dependencies": { "debug": { @@ -1404,7 +1404,7 @@ "resolved": "https://registry.npmjs.org/debug/-/debug-4.1.1.tgz", "integrity": "sha512-pYAIzeRo8J6KPEaJ0VWOh5Pzkbw/RetuzehGM7QRRX5he4fPHx2rdKMB256ehJCkX+XRQm16eZLqLNS8RSZXZw==", "requires": { - "ms": "2.1.2" + "ms": "^2.1.1" } }, "globals": { @@ -1424,9 +1424,9 @@ "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.6.1.tgz", "integrity": "sha512-X7gdiuaCmA0uRjCmRtYJNAVCc/q+5xSgsfKJHqMN4iNLILX39677fJE1O40arPMh0TTtS9ItH67yre6c7k6t0g==", "requires": { - "esutils": "2.0.3", - "lodash": "4.17.15", - "to-fast-properties": "2.0.0" + "esutils": "^2.0.2", + "lodash": "^4.17.13", + "to-fast-properties": "^2.0.0" }, "dependencies": { "to-fast-properties": { @@ -1441,9 +1441,9 @@ "resolved": "https://registry.npmjs.org/@cypress/webpack-preprocessor/-/webpack-preprocessor-4.1.0.tgz", "integrity": "sha512-LbxsdYVpHGoC2fMOdW0aQvuvVRD7aZx8p8DrP53HISpl7bD1PqLGWKzhHn7cGG24UHycBJrbaEeKEosW29W1dg==", "requires": { - "@babel/core": "7.5.5", - "@babel/preset-env": "7.5.5", - "babel-loader": "8.0.6", + "@babel/core": "^7.0.1", + "@babel/preset-env": "^7.0.0", + "babel-loader": "^8.0.2", "bluebird": "3.5.0", "debug": "3.1.0" }, @@ -1502,7 +1502,7 @@ "integrity": "sha512-Z9Yfa11F6B9Sg/BK9MnqnQ+aQYicPLtilXBp2yUtDt2JRCE0h26d33EnfO3ZxoNxG0T92OUucP3Ct7cpfkdFfw==", "dev": true, "requires": { - "tslib": "1.10.0" + "tslib": "^1.9.0" } } } @@ -1514,7 +1514,7 @@ "dev": true, "requires": { "@nodelib/fs.stat": "2.0.2", - "run-parallel": "1.1.9" + "run-parallel": "^1.1.9" } }, "@nodelib/fs.stat": { @@ -1530,7 +1530,7 @@ "dev": true, "requires": { "@nodelib/fs.scandir": "2.1.2", - "fastq": "1.6.0" + "fastq": "^1.6.0" } }, "@nrwl/angular": { @@ -1542,7 +1542,7 @@ "@nrwl/cypress": "8.5.2", "@nrwl/jest": "8.5.2", "@schematics/angular": "8.3.3", - "jasmine-marbles": "0.6.0" + "jasmine-marbles": "~0.6.0" }, "dependencies": { "@angular-devkit/core": { @@ -1580,10 +1580,10 @@ "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.10.2.tgz", "integrity": "sha512-TXtUUEYHuaTEbLZWIKUr5pmBuhDLy+8KYtPYdcV8qC+pOZL+NKqYwvWSRrVXHn+ZmRRAu8vJTAznH7Oag6RVRw==", "requires": { - "fast-deep-equal": "2.0.1", - "fast-json-stable-stringify": "2.0.0", - "json-schema-traverse": "0.4.1", - "uri-js": "4.2.2" + "fast-deep-equal": "^2.0.1", + "fast-json-stable-stringify": "^2.0.0", + "json-schema-traverse": "^0.4.1", + "uri-js": "^4.2.2" } }, "rxjs": { @@ -1591,7 +1591,7 @@ "resolved": "https://registry.npmjs.org/rxjs/-/rxjs-6.4.0.tgz", "integrity": "sha512-Z9Yfa11F6B9Sg/BK9MnqnQ+aQYicPLtilXBp2yUtDt2JRCE0h26d33EnfO3ZxoNxG0T92OUucP3Ct7cpfkdFfw==", "requires": { - "tslib": "1.10.0" + "tslib": "^1.9.0" } }, "source-map": { @@ -1609,7 +1609,7 @@ "requires": { "@nrwl/tao": "8.5.2", "tmp": "0.0.33", - "yargs": "11.1.0", + "yargs": "^11.0.0", "yargs-parser": "10.0.0" }, "dependencies": { @@ -1631,9 +1631,9 @@ "integrity": "sha512-4FG+RSG9DL7uEwRUZXZn3SS34DiDPfzP0VOiEwtUWlE+AR2EIg+hSyvrIgUUfhdgR/UkAeW2QHgeP+hWrXs7jQ==", "dev": true, "requires": { - "string-width": "2.1.1", - "strip-ansi": "4.0.0", - "wrap-ansi": "2.1.0" + "string-width": "^2.1.1", + "strip-ansi": "^4.0.0", + "wrap-ansi": "^2.0.0" } }, "cross-spawn": { @@ -1642,9 +1642,9 @@ "integrity": "sha1-6L0O/uWPz/b4+UUQoKVUu/ojVEk=", "dev": true, "requires": { - "lru-cache": "4.1.5", - "shebang-command": "1.2.0", - "which": "1.3.1" + "lru-cache": "^4.0.1", + "shebang-command": "^1.2.0", + "which": "^1.2.9" } }, "execa": { @@ -1653,13 +1653,13 @@ "integrity": "sha1-lEvs00zEHuMqY6n68nrVpl/Fl3c=", "dev": true, "requires": { - "cross-spawn": "5.1.0", - "get-stream": "3.0.0", - "is-stream": "1.1.0", - "npm-run-path": "2.0.2", - "p-finally": "1.0.0", - "signal-exit": "3.0.2", - "strip-eof": "1.0.0" + "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" } }, "find-up": { @@ -1668,7 +1668,7 @@ "integrity": "sha1-RdG35QbHF93UgndaK3eSCjwMV6c=", "dev": true, "requires": { - "locate-path": "2.0.0" + "locate-path": "^2.0.0" } }, "get-stream": { @@ -1689,8 +1689,8 @@ "integrity": "sha1-K1aLJl7slExtnA3pw9u7ygNUzY4=", "dev": true, "requires": { - "p-locate": "2.0.0", - "path-exists": "3.0.0" + "p-locate": "^2.0.0", + "path-exists": "^3.0.0" } }, "lru-cache": { @@ -1699,8 +1699,8 @@ "integrity": "sha512-sWZlbEP2OsHNkXrMl5GYk/jKk70MBng6UU4YI/qGDYbgf6YbP4EvmqISbXCoJiRKs+1bSpFHVgQxvJ17F2li5g==", "dev": true, "requires": { - "pseudomap": "1.0.2", - "yallist": "2.1.2" + "pseudomap": "^1.0.2", + "yallist": "^2.1.2" } }, "mem": { @@ -1709,7 +1709,7 @@ "integrity": "sha1-Xt1StIXKHZAP5kiVUFOZoN+kX3Y=", "dev": true, "requires": { - "mimic-fn": "1.2.0" + "mimic-fn": "^1.0.0" } }, "mimic-fn": { @@ -1724,9 +1724,9 @@ "integrity": "sha512-3sslG3zJbEYcaC4YVAvDorjGxc7tv6KVATnLPZONiljsUncvihe9BQoVCEs0RZ1kmf4Hk9OBqlZfJZWI4GanKA==", "dev": true, "requires": { - "execa": "0.7.0", - "lcid": "1.0.0", - "mem": "1.1.0" + "execa": "^0.7.0", + "lcid": "^1.0.0", + "mem": "^1.1.0" } }, "p-limit": { @@ -1735,7 +1735,7 @@ "integrity": "sha512-vvcXsLAJ9Dr5rQOPk7toZQZJApBl2K4J6dANSsEuh6QI41JYcsS/qhTGa9ErIUUgK3WNQoJYvylxvjqmiqEA9Q==", "dev": true, "requires": { - "p-try": "1.0.0" + "p-try": "^1.0.0" } }, "p-locate": { @@ -1744,7 +1744,7 @@ "integrity": "sha1-IKAQOyIqcMj9OcwuWAaA893l7EM=", "dev": true, "requires": { - "p-limit": "1.3.0" + "p-limit": "^1.1.0" } }, "p-try": { @@ -1759,8 +1759,8 @@ "integrity": "sha512-nOqH59deCq9SRHlxq1Aw85Jnt4w6KvLKqWVik6oA9ZklXLNIOlqg4F2yrT1MVaTjAqvVwdfeZ7w7aCvJD7ugkw==", "dev": true, "requires": { - "is-fullwidth-code-point": "2.0.0", - "strip-ansi": "4.0.0" + "is-fullwidth-code-point": "^2.0.0", + "strip-ansi": "^4.0.0" } }, "strip-ansi": { @@ -1769,7 +1769,7 @@ "integrity": "sha1-qEeQIusaw2iocTibY1JixQXuNo8=", "dev": true, "requires": { - "ansi-regex": "3.0.0" + "ansi-regex": "^3.0.0" } }, "which-module": { @@ -1796,18 +1796,18 @@ "integrity": "sha512-NwW69J42EsCSanF8kyn5upxvjp5ds+t3+udGBeTbFnERA+lF541DDpMawzo4z6W/QrzNM18D+BPMiOBibnFV5A==", "dev": true, "requires": { - "cliui": "4.1.0", - "decamelize": "1.2.0", - "find-up": "2.1.0", - "get-caller-file": "1.0.3", - "os-locale": "2.1.0", - "require-directory": "2.1.1", - "require-main-filename": "1.0.1", - "set-blocking": "2.0.0", - "string-width": "2.1.1", - "which-module": "2.0.0", - "y18n": "3.2.1", - "yargs-parser": "9.0.2" + "cliui": "^4.0.0", + "decamelize": "^1.1.1", + "find-up": "^2.1.0", + "get-caller-file": "^1.0.1", + "os-locale": "^2.0.0", + "require-directory": "^2.1.1", + "require-main-filename": "^1.0.1", + "set-blocking": "^2.0.0", + "string-width": "^2.0.0", + "which-module": "^2.0.0", + "y18n": "^3.2.1", + "yargs-parser": "^9.0.2" }, "dependencies": { "yargs-parser": { @@ -1816,7 +1816,7 @@ "integrity": "sha1-nM9qQ0YP5O1Aqbto9I1DuKaMwHc=", "dev": true, "requires": { - "camelcase": "4.1.0" + "camelcase": "^4.1.0" } } } @@ -1827,7 +1827,7 @@ "integrity": "sha512-+DHejWujTVYeMHLff8U96rLc4uE4Emncoftvn5AjhB1Jw1pWxLzgBUT/WYbPrHmy6YPEBTZQx5myHhVcuuu64g==", "dev": true, "requires": { - "camelcase": "4.1.0" + "camelcase": "^4.1.0" } } } @@ -1839,8 +1839,8 @@ "requires": { "@angular-devkit/architect": "0.803.3", "@angular-devkit/core": "8.3.3", - "@cypress/webpack-preprocessor": "4.1.0", - "fork-ts-checker-webpack-plugin": "0.4.15", + "@cypress/webpack-preprocessor": "~4.1.0", + "fork-ts-checker-webpack-plugin": "^0.4.9", "tree-kill": "1.2.1", "ts-loader": "5.3.1", "tsconfig-paths-webpack-plugin": "3.2.0", @@ -1873,10 +1873,10 @@ "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.10.2.tgz", "integrity": "sha512-TXtUUEYHuaTEbLZWIKUr5pmBuhDLy+8KYtPYdcV8qC+pOZL+NKqYwvWSRrVXHn+ZmRRAu8vJTAznH7Oag6RVRw==", "requires": { - "fast-deep-equal": "2.0.1", - "fast-json-stable-stringify": "2.0.0", - "json-schema-traverse": "0.4.1", - "uri-js": "4.2.2" + "fast-deep-equal": "^2.0.1", + "fast-json-stable-stringify": "^2.0.0", + "json-schema-traverse": "^0.4.1", + "uri-js": "^4.2.2" } }, "rxjs": { @@ -1884,7 +1884,7 @@ "resolved": "https://registry.npmjs.org/rxjs/-/rxjs-6.4.0.tgz", "integrity": "sha512-Z9Yfa11F6B9Sg/BK9MnqnQ+aQYicPLtilXBp2yUtDt2JRCE0h26d33EnfO3ZxoNxG0T92OUucP3Ct7cpfkdFfw==", "requires": { - "tslib": "1.10.0" + "tslib": "^1.9.0" } }, "source-map": { @@ -1939,10 +1939,10 @@ "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.10.2.tgz", "integrity": "sha512-TXtUUEYHuaTEbLZWIKUr5pmBuhDLy+8KYtPYdcV8qC+pOZL+NKqYwvWSRrVXHn+ZmRRAu8vJTAznH7Oag6RVRw==", "requires": { - "fast-deep-equal": "2.0.1", - "fast-json-stable-stringify": "2.0.0", - "json-schema-traverse": "0.4.1", - "uri-js": "4.2.2" + "fast-deep-equal": "^2.0.1", + "fast-json-stable-stringify": "^2.0.0", + "json-schema-traverse": "^0.4.1", + "uri-js": "^4.2.2" } }, "rxjs": { @@ -1950,7 +1950,7 @@ "resolved": "https://registry.npmjs.org/rxjs/-/rxjs-6.4.0.tgz", "integrity": "sha512-Z9Yfa11F6B9Sg/BK9MnqnQ+aQYicPLtilXBp2yUtDt2JRCE0h26d33EnfO3ZxoNxG0T92OUucP3Ct7cpfkdFfw==", "requires": { - "tslib": "1.10.0" + "tslib": "^1.9.0" } }, "source-map": { @@ -1969,8 +1969,8 @@ "@angular-devkit/architect": "0.803.3", "@angular-devkit/core": "8.3.3", "@angular-devkit/schematics": "8.3.3", - "inquirer": "6.5.1", - "minimist": "1.2.0", + "inquirer": "^6.3.1", + "minimist": "^1.2.0", "strip-json-comments": "2.0.1" }, "dependencies": { @@ -2013,10 +2013,10 @@ "integrity": "sha512-TXtUUEYHuaTEbLZWIKUr5pmBuhDLy+8KYtPYdcV8qC+pOZL+NKqYwvWSRrVXHn+ZmRRAu8vJTAznH7Oag6RVRw==", "dev": true, "requires": { - "fast-deep-equal": "2.0.1", - "fast-json-stable-stringify": "2.0.0", - "json-schema-traverse": "0.4.1", - "uri-js": "4.2.2" + "fast-deep-equal": "^2.0.1", + "fast-json-stable-stringify": "^2.0.0", + "json-schema-traverse": "^0.4.1", + "uri-js": "^4.2.2" } }, "rxjs": { @@ -2025,7 +2025,7 @@ "integrity": "sha512-Z9Yfa11F6B9Sg/BK9MnqnQ+aQYicPLtilXBp2yUtDt2JRCE0h26d33EnfO3ZxoNxG0T92OUucP3Ct7cpfkdFfw==", "dev": true, "requires": { - "tslib": "1.10.0" + "tslib": "^1.9.0" } }, "source-map": { @@ -2051,14 +2051,14 @@ "graphviz": "0.0.8", "ignore": "5.0.4", "npm-run-all": "4.1.5", - "opn": "5.5.0", + "opn": "^5.3.0", "prettier": "1.18.2", - "rxjs": "6.5.3", + "rxjs": "^6.4.0", "semver": "5.4.1", "strip-json-comments": "2.0.1", "tmp": "0.0.33", - "viz.js": "1.8.2", - "yargs": "11.1.0", + "viz.js": "^1.8.1", + "yargs": "^11.0.0", "yargs-parser": "10.0.0" }, "dependencies": { @@ -2081,7 +2081,7 @@ "integrity": "sha512-Z9Yfa11F6B9Sg/BK9MnqnQ+aQYicPLtilXBp2yUtDt2JRCE0h26d33EnfO3ZxoNxG0T92OUucP3Ct7cpfkdFfw==", "dev": true, "requires": { - "tslib": "1.10.0" + "tslib": "^1.9.0" } } } @@ -2102,7 +2102,7 @@ "integrity": "sha512-Z9Yfa11F6B9Sg/BK9MnqnQ+aQYicPLtilXBp2yUtDt2JRCE0h26d33EnfO3ZxoNxG0T92OUucP3Ct7cpfkdFfw==", "dev": true, "requires": { - "tslib": "1.10.0" + "tslib": "^1.9.0" } } } @@ -2113,10 +2113,10 @@ "integrity": "sha512-TXtUUEYHuaTEbLZWIKUr5pmBuhDLy+8KYtPYdcV8qC+pOZL+NKqYwvWSRrVXHn+ZmRRAu8vJTAznH7Oag6RVRw==", "dev": true, "requires": { - "fast-deep-equal": "2.0.1", - "fast-json-stable-stringify": "2.0.0", - "json-schema-traverse": "0.4.1", - "uri-js": "4.2.2" + "fast-deep-equal": "^2.0.1", + "fast-json-stable-stringify": "^2.0.0", + "json-schema-traverse": "^0.4.1", + "uri-js": "^4.2.2" } }, "ansi-regex": { @@ -2137,9 +2137,9 @@ "integrity": "sha512-4FG+RSG9DL7uEwRUZXZn3SS34DiDPfzP0VOiEwtUWlE+AR2EIg+hSyvrIgUUfhdgR/UkAeW2QHgeP+hWrXs7jQ==", "dev": true, "requires": { - "string-width": "2.1.1", - "strip-ansi": "4.0.0", - "wrap-ansi": "2.1.0" + "string-width": "^2.1.1", + "strip-ansi": "^4.0.0", + "wrap-ansi": "^2.0.0" } }, "cosmiconfig": { @@ -2148,10 +2148,10 @@ "integrity": "sha512-6e5vDdrXZD+t5v0L8CrurPeybg4Fmf+FCSYxXKYVAqLUtyCSbuyqE059d0kDthTNRzKVjL7QMgNpEUlsoYH3iQ==", "dev": true, "requires": { - "is-directory": "0.3.1", - "js-yaml": "3.13.1", - "parse-json": "4.0.0", - "require-from-string": "2.0.2" + "is-directory": "^0.3.1", + "js-yaml": "^3.9.0", + "parse-json": "^4.0.0", + "require-from-string": "^2.0.1" } }, "cross-spawn": { @@ -2160,9 +2160,9 @@ "integrity": "sha1-6L0O/uWPz/b4+UUQoKVUu/ojVEk=", "dev": true, "requires": { - "lru-cache": "4.1.5", - "shebang-command": "1.2.0", - "which": "1.3.1" + "lru-cache": "^4.0.1", + "shebang-command": "^1.2.0", + "which": "^1.2.9" } }, "execa": { @@ -2171,13 +2171,13 @@ "integrity": "sha1-lEvs00zEHuMqY6n68nrVpl/Fl3c=", "dev": true, "requires": { - "cross-spawn": "5.1.0", - "get-stream": "3.0.0", - "is-stream": "1.1.0", - "npm-run-path": "2.0.2", - "p-finally": "1.0.0", - "signal-exit": "3.0.2", - "strip-eof": "1.0.0" + "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" } }, "find-up": { @@ -2186,7 +2186,7 @@ "integrity": "sha1-RdG35QbHF93UgndaK3eSCjwMV6c=", "dev": true, "requires": { - "locate-path": "2.0.0" + "locate-path": "^2.0.0" } }, "fs-extra": { @@ -2195,9 +2195,9 @@ "integrity": "sha512-lk2cUCo8QzbiEWEbt7Cw3m27WMiRG321xsssbcIpfMhpRjrlC08WBOVQqj1/nQYYNnPtyIhP1oqLO3QwT2tPCw==", "dev": true, "requires": { - "graceful-fs": "4.2.2", - "jsonfile": "4.0.0", - "universalify": "0.1.2" + "graceful-fs": "^4.1.2", + "jsonfile": "^4.0.0", + "universalify": "^0.1.0" } }, "get-stream": { @@ -2224,8 +2224,8 @@ "integrity": "sha1-K1aLJl7slExtnA3pw9u7ygNUzY4=", "dev": true, "requires": { - "p-locate": "2.0.0", - "path-exists": "3.0.0" + "p-locate": "^2.0.0", + "path-exists": "^3.0.0" } }, "lru-cache": { @@ -2234,8 +2234,8 @@ "integrity": "sha512-sWZlbEP2OsHNkXrMl5GYk/jKk70MBng6UU4YI/qGDYbgf6YbP4EvmqISbXCoJiRKs+1bSpFHVgQxvJ17F2li5g==", "dev": true, "requires": { - "pseudomap": "1.0.2", - "yallist": "2.1.2" + "pseudomap": "^1.0.2", + "yallist": "^2.1.2" } }, "mem": { @@ -2244,7 +2244,7 @@ "integrity": "sha1-Xt1StIXKHZAP5kiVUFOZoN+kX3Y=", "dev": true, "requires": { - "mimic-fn": "1.2.0" + "mimic-fn": "^1.0.0" } }, "mimic-fn": { @@ -2259,9 +2259,9 @@ "integrity": "sha512-3sslG3zJbEYcaC4YVAvDorjGxc7tv6KVATnLPZONiljsUncvihe9BQoVCEs0RZ1kmf4Hk9OBqlZfJZWI4GanKA==", "dev": true, "requires": { - "execa": "0.7.0", - "lcid": "1.0.0", - "mem": "1.1.0" + "execa": "^0.7.0", + "lcid": "^1.0.0", + "mem": "^1.1.0" } }, "p-limit": { @@ -2270,7 +2270,7 @@ "integrity": "sha512-vvcXsLAJ9Dr5rQOPk7toZQZJApBl2K4J6dANSsEuh6QI41JYcsS/qhTGa9ErIUUgK3WNQoJYvylxvjqmiqEA9Q==", "dev": true, "requires": { - "p-try": "1.0.0" + "p-try": "^1.0.0" } }, "p-locate": { @@ -2279,7 +2279,7 @@ "integrity": "sha1-IKAQOyIqcMj9OcwuWAaA893l7EM=", "dev": true, "requires": { - "p-limit": "1.3.0" + "p-limit": "^1.1.0" } }, "p-try": { @@ -2294,8 +2294,8 @@ "integrity": "sha1-vjX1Qlvh9/bHRxhPmKeIy5lHfuA=", "dev": true, "requires": { - "error-ex": "1.3.2", - "json-parse-better-errors": "1.0.2" + "error-ex": "^1.3.1", + "json-parse-better-errors": "^1.0.1" } }, "semver": { @@ -2316,8 +2316,8 @@ "integrity": "sha512-nOqH59deCq9SRHlxq1Aw85Jnt4w6KvLKqWVik6oA9ZklXLNIOlqg4F2yrT1MVaTjAqvVwdfeZ7w7aCvJD7ugkw==", "dev": true, "requires": { - "is-fullwidth-code-point": "2.0.0", - "strip-ansi": "4.0.0" + "is-fullwidth-code-point": "^2.0.0", + "strip-ansi": "^4.0.0" } }, "strip-ansi": { @@ -2326,7 +2326,7 @@ "integrity": "sha1-qEeQIusaw2iocTibY1JixQXuNo8=", "dev": true, "requires": { - "ansi-regex": "3.0.0" + "ansi-regex": "^3.0.0" } }, "which-module": { @@ -2353,18 +2353,18 @@ "integrity": "sha512-NwW69J42EsCSanF8kyn5upxvjp5ds+t3+udGBeTbFnERA+lF541DDpMawzo4z6W/QrzNM18D+BPMiOBibnFV5A==", "dev": true, "requires": { - "cliui": "4.1.0", - "decamelize": "1.2.0", - "find-up": "2.1.0", - "get-caller-file": "1.0.3", - "os-locale": "2.1.0", - "require-directory": "2.1.1", - "require-main-filename": "1.0.1", - "set-blocking": "2.0.0", - "string-width": "2.1.1", - "which-module": "2.0.0", - "y18n": "3.2.1", - "yargs-parser": "9.0.2" + "cliui": "^4.0.0", + "decamelize": "^1.1.1", + "find-up": "^2.1.0", + "get-caller-file": "^1.0.1", + "os-locale": "^2.0.0", + "require-directory": "^2.1.1", + "require-main-filename": "^1.0.1", + "set-blocking": "^2.0.0", + "string-width": "^2.0.0", + "which-module": "^2.0.0", + "y18n": "^3.2.1", + "yargs-parser": "^9.0.2" }, "dependencies": { "yargs-parser": { @@ -2373,7 +2373,7 @@ "integrity": "sha1-nM9qQ0YP5O1Aqbto9I1DuKaMwHc=", "dev": true, "requires": { - "camelcase": "4.1.0" + "camelcase": "^4.1.0" } } } @@ -2384,7 +2384,7 @@ "integrity": "sha512-+DHejWujTVYeMHLff8U96rLc4uE4Emncoftvn5AjhB1Jw1pWxLzgBUT/WYbPrHmy6YPEBTZQx5myHhVcuuu64g==", "dev": true, "requires": { - "camelcase": "4.1.0" + "camelcase": "^4.1.0" } } } @@ -2421,7 +2421,7 @@ "integrity": "sha512-Z9Yfa11F6B9Sg/BK9MnqnQ+aQYicPLtilXBp2yUtDt2JRCE0h26d33EnfO3ZxoNxG0T92OUucP3Ct7cpfkdFfw==", "dev": true, "requires": { - "tslib": "1.10.0" + "tslib": "^1.9.0" } }, "semver": { @@ -2443,18 +2443,18 @@ "resolved": "https://registry.npmjs.org/@swimlane/ngx-charts/-/ngx-charts-12.0.1.tgz", "integrity": "sha512-Dmm46eTtyKtTkku1ZIB39q/AHvUGdGIRJc7jPUucJLoSPWeD7AjvE6OFiIZYP48IkPwV0R3OI/ZmKyNA3DPkRQ==", "requires": { - "d3": "4.13.0", - "d3-array": "1.2.4", - "d3-brush": "1.1.3", - "d3-color": "1.4.0", - "d3-force": "1.2.1", - "d3-format": "1.4.1", - "d3-hierarchy": "1.1.8", - "d3-interpolate": "1.3.2", - "d3-scale": "1.0.7", - "d3-selection": "1.4.0", - "d3-shape": "1.3.5", - "d3-time-format": "2.1.3" + "d3": "^4.10.2", + "d3-array": "^1.2.1", + "d3-brush": "^1.0.4", + "d3-color": "^1.0.3", + "d3-force": "^1.1.0", + "d3-format": "^1.2.0", + "d3-hierarchy": "^1.1.5", + "d3-interpolate": "^1.1.5", + "d3-scale": "^1.0.6", + "d3-selection": "^1.1.0", + "d3-shape": "^1.2.0", + "d3-time-format": "^2.1.0" } }, "@szmarczak/http-timer": { @@ -2463,7 +2463,7 @@ "integrity": "sha512-XIB2XbzHTN6ieIjfIMV9hlVcfPU26s2vafYWQcZHWXHOxiaRZYEDKEwdl129Zyg50+foYV2jCgtrqSA6qNuNSA==", "dev": true, "requires": { - "defer-to-connect": "1.0.2" + "defer-to-connect": "^1.0.1" } }, "@tootallnate/once": { @@ -2512,9 +2512,9 @@ "integrity": "sha512-1Bh06cbWJUHMC97acuD6UMG29nMt0Aqz1vF3guLfG+kHHJhy3AyohZFFxYk2f7Q1SQIrNwvncxAE0N/9s70F2w==", "dev": true, "requires": { - "@types/events": "3.0.0", - "@types/minimatch": "3.0.3", - "@types/node": "12.7.9" + "@types/events": "*", + "@types/minimatch": "*", + "@types/node": "*" } }, "@types/jasmine": { @@ -2529,7 +2529,7 @@ "integrity": "sha512-d9p31r7Nxk0ZH0U39PTH0hiDlJ+qNVGjlt1ucOoTUptxb2v+Y5VMnsxfwN+i3hK4yQnqBi3FMmoMFcd1JHDxdg==", "dev": true, "requires": { - "@types/jasmine": "3.4.2" + "@types/jasmine": "*" } }, "@types/json5": { @@ -2543,9 +2543,9 @@ "integrity": "sha512-mkJejrAacgignkBce2+qD9S4VncjEfAT0Dion0fRcqpav3Sd2KiLTHODZOXRP3S8b0ZY5sXr9meDB3P8MSH8Cg==", "dev": true, "requires": { - "@types/bluebird": "3.5.27", - "@types/node": "12.7.9", - "log4js": "3.0.6" + "@types/bluebird": "*", + "@types/node": "*", + "log4js": "^3.0.0" } }, "@types/marked": { @@ -2591,10 +2591,10 @@ "integrity": "sha512-3Wo2jNYwqgXcIz/rrq18AdOZUQB8cQ34CXZo+LUwPJNpvRAL86+Kc2wwI8mqpz9Cr1V+enIox5v+WZhy/p3h8w==", "dev": true, "requires": { - "@types/caseless": "0.12.2", - "@types/node": "12.7.9", - "@types/tough-cookie": "2.3.5", - "form-data": "2.5.1" + "@types/caseless": "*", + "@types/node": "*", + "@types/tough-cookie": "*", + "form-data": "^2.5.0" }, "dependencies": { "form-data": { @@ -2603,9 +2603,9 @@ "integrity": "sha512-m21N3WOmEEURgk6B9GLOE4RuWOFf28Lhh9qGYeNlGq4VDXUlJy2th2slBNU8Gp8EzloYZOibZJ7t5ecIrFSjVA==", "dev": true, "requires": { - "asynckit": "0.4.0", - "combined-stream": "1.0.8", - "mime-types": "2.1.24" + "asynckit": "^0.4.0", + "combined-stream": "^1.0.6", + "mime-types": "^2.1.12" } } } @@ -2616,7 +2616,7 @@ "integrity": "sha512-auApPaJf3NPfe18hSoJkp8EbZzer2ISk7o8mCC3M9he/a04+gbMF97NkpD2S8riMGvm4BMRI59/SZQSaLTKpsQ==", "dev": true, "requires": { - "@types/node": "12.7.9" + "@types/node": "*" } }, "@types/selenium-webdriver": { @@ -2712,7 +2712,7 @@ "dev": true, "requires": { "@webassemblyjs/ast": "1.8.5", - "mamacro": "0.0.3" + "mamacro": "^0.0.3" } }, "@webassemblyjs/helper-wasm-bytecode": { @@ -2739,7 +2739,7 @@ "integrity": "sha512-aaCvQYrvKbY/n6wKHb/ylAJr27GglahUO89CcGXMItrOBqRarUMxWLJgxm9PJNuKULwN5n1csT9bYoMeZOGF3g==", "dev": true, "requires": { - "@xtuc/ieee754": "1.2.0" + "@xtuc/ieee754": "^1.2.0" } }, "@webassemblyjs/leb128": { @@ -2877,7 +2877,7 @@ "integrity": "sha512-Il80Qs2WjYlJIBNzNkK6KYqlVMTbZLXgHx2oT0pU/fjRHyEp+PEfEPY0R3WCwAGVOtauxh1hOxNgIf5bv7dQpA==", "dev": true, "requires": { - "mime-types": "2.1.24", + "mime-types": "~2.1.24", "negotiator": "0.6.2" } }, @@ -2922,7 +2922,7 @@ "integrity": "sha512-salcGninV0nPrwpGNn4VTXBb1SOuXQBiqbrNXoeizJsHrsL6ERFM2Ne3JUSBWRE6aeNJI2ROP/WEEIDUiDe3cg==", "dev": true, "requires": { - "es6-promisify": "5.0.0" + "es6-promisify": "^5.0.0" } }, "agentkeepalive": { @@ -2969,7 +2969,7 @@ "resolved": "https://registry.npmjs.org/angular2-virtual-scroll/-/angular2-virtual-scroll-0.4.16.tgz", "integrity": "sha512-6NWk0DjCh4ebU8+LgfBoKYyp3McxDA/k5vTnEiV32VpVnyhN//eThZpVpggI1D2fJBqgTAY09C8v++qXHHLP7A==", "requires": { - "@tweenjs/tween.js": "17.4.0" + "@tweenjs/tween.js": "^17.1.0" } }, "ansi-align": { @@ -2978,7 +2978,7 @@ "integrity": "sha512-ZpClVKqXN3RGBmKibdfWzqCY4lnjEuoNzU5T0oEFpfd/z5qJHVarukridD4juLO2FXMiwUQxr9WqQtaYa8XRYw==", "dev": true, "requires": { - "string-width": "3.1.0" + "string-width": "^3.0.0" }, "dependencies": { "ansi-regex": { @@ -3005,9 +3005,9 @@ "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==", "dev": true, "requires": { - "emoji-regex": "7.0.3", - "is-fullwidth-code-point": "2.0.0", - "strip-ansi": "5.2.0" + "emoji-regex": "^7.0.1", + "is-fullwidth-code-point": "^2.0.0", + "strip-ansi": "^5.1.0" } }, "strip-ansi": { @@ -3016,7 +3016,7 @@ "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", "dev": true, "requires": { - "ansi-regex": "4.1.0" + "ansi-regex": "^4.1.0" } } } @@ -3033,7 +3033,7 @@ "integrity": "sha512-Cg3ymMAdN10wOk/VYfLV7KCQyv7EDirJ64500sU7n9UlmioEtDuU5Gd+hj73hXSU/ex7tHJSssmyftDdkMLO8Q==", "dev": true, "requires": { - "type-fest": "0.5.2" + "type-fest": "^0.5.2" } }, "ansi-gray": { @@ -3100,7 +3100,7 @@ "integrity": "sha1-126/jKlNJ24keja61EpLdKthGZE=", "dev": true, "requires": { - "default-require-extensions": "1.0.0" + "default-require-extensions": "^1.0.0" } }, "aproba": { @@ -3143,7 +3143,7 @@ "dev": true, "requires": { "ast-types-flow": "0.0.7", - "commander": "2.20.0" + "commander": "^2.11.0" } }, "arr-diff": { @@ -3295,7 +3295,7 @@ "integrity": "sha512-jxwzQpLQjSmWXgwaCZE9Nz+glAG01yF1QnWgbhGwHI5A6FRIEY6IVqtHhIepHqI7/kyEyQEagBC5mBEFlIYvdg==", "dev": true, "requires": { - "safer-buffer": "2.1.2" + "safer-buffer": "~2.1.0" } }, "asn1.js": { @@ -3315,7 +3315,7 @@ "integrity": "sha512-EDsgawzwoun2CZkCgtxJbv392v4nbk9XDD06zI+kQYoBM/3RBWLlEyJARDOmhAAosBjWACEkKL6S+lIZtcAubA==", "dev": true, "requires": { - "object-assign": "4.1.1", + "object-assign": "^4.1.1", "util": "0.10.3" }, "dependencies": { @@ -3359,7 +3359,7 @@ "integrity": "sha512-zflvls11DCy+dQWzTW2dzuilv8Z5X/pjfmZOWba6TNIVDm+2UDaJmXSOXlasHKfNBs8oo3M0aT50fDEWfKZjXg==", "dev": true, "requires": { - "lodash": "4.17.15" + "lodash": "^4.17.14" } }, "async-array-reduce": { @@ -3374,10 +3374,10 @@ "integrity": "sha512-uYkTP8dw2og1tu1nmza1n1CMW0qb8gWWlwqMmLb7MhBVs4BXrFziT6HXUd+/RlRA/i4H9AkofYloUbs1fwMqlw==", "dev": true, "requires": { - "end-of-stream": "1.4.3", - "once": "1.4.0", - "process-nextick-args": "2.0.1", - "stream-exhaust": "1.0.2" + "end-of-stream": "^1.1.0", + "once": "^1.3.2", + "process-nextick-args": "^2.0.0", + "stream-exhaust": "^1.0.1" } }, "async-each": { @@ -3417,13 +3417,13 @@ "integrity": "sha512-aVo5WxR3VyvyJxcJC3h4FKfwCQvQWb1tSI5VHNibddCVWrcD1NvlxEweg3TSgiPztMnWfjpy2FURKA2kvDE+Tw==", "dev": true, "requires": { - "browserslist": "4.6.6", - "caniuse-lite": "1.0.30000989", - "chalk": "2.4.2", - "normalize-range": "0.1.2", - "num2fraction": "1.2.2", - "postcss": "7.0.17", - "postcss-value-parser": "4.0.2" + "browserslist": "^4.6.3", + "caniuse-lite": "^1.0.30000980", + "chalk": "^2.4.2", + "normalize-range": "^0.1.2", + "num2fraction": "^1.2.2", + "postcss": "^7.0.17", + "postcss-value-parser": "^4.0.0" } }, "aws-sign2": { @@ -3522,10 +3522,10 @@ "integrity": "sha512-4BmWKtBOBm13uoUwd08UwjZlaw3O9GWf456R9j+5YykFZ6LUIjIKLc0zEZf+hauxPOJs96C8k6FvYD09vWzhYw==", "optional": true, "requires": { - "find-cache-dir": "2.1.0", - "loader-utils": "1.2.3", - "mkdirp": "0.5.1", - "pify": "4.0.1" + "find-cache-dir": "^2.0.0", + "loader-utils": "^1.0.2", + "mkdirp": "^0.5.1", + "pify": "^4.0.1" }, "dependencies": { "find-cache-dir": { @@ -3534,9 +3534,9 @@ "integrity": "sha512-Tq6PixE0w/VMFfCgbONnkiQIVol/JJL7nRMi20fqzA4NRs9AfeqMGeRdPi3wIhYkxjeBaWh2rxwapn5Tu3IqOQ==", "optional": true, "requires": { - "commondir": "1.0.1", - "make-dir": "2.1.0", - "pkg-dir": "3.0.0" + "commondir": "^1.0.1", + "make-dir": "^2.0.0", + "pkg-dir": "^3.0.0" } }, "pify": { @@ -3561,7 +3561,7 @@ "resolved": "https://registry.npmjs.org/babel-plugin-dynamic-import-node/-/babel-plugin-dynamic-import-node-2.3.0.tgz", "integrity": "sha512-o6qFkpeQEBxcqt0XYlWzAVxNCSCZdUgcR8IRlhD/8DylxjjO4foPcvTW0GGKa/cVt3rvxZ7o5ippJ+/0nvLhlQ==", "requires": { - "object.assign": "4.1.0" + "object.assign": "^4.1.0" } }, "babel-runtime": { @@ -3744,7 +3744,7 @@ "integrity": "sha1-pDAdOJtqQ/m2f/PKEaP2Y342Dp4=", "dev": true, "requires": { - "tweetnacl": "0.14.5" + "tweetnacl": "^0.14.3" } }, "better-assert": { @@ -3800,15 +3800,15 @@ "dev": true, "requires": { "bytes": "3.1.0", - "content-type": "1.0.4", + "content-type": "~1.0.4", "debug": "2.6.9", - "depd": "1.1.2", + "depd": "~1.1.2", "http-errors": "1.7.2", "iconv-lite": "0.4.24", - "on-finished": "2.3.0", + "on-finished": "~2.3.0", "qs": "6.7.0", "raw-body": "2.4.0", - "type-is": "1.6.18" + "type-is": "~1.6.17" }, "dependencies": { "bytes": { @@ -3845,14 +3845,14 @@ "integrity": "sha512-cU4J/+NodM3IHdSL2yN8bqYqnmlBTidDR4RC7nJs61ZmtGz8VZzM3HLQX0zY5mrSmPtR3xWwsq2jOUQqFZN8+A==", "dev": true, "requires": { - "ansi-align": "3.0.0", - "camelcase": "5.3.1", - "chalk": "2.4.2", - "cli-boxes": "2.2.0", - "string-width": "3.1.0", - "term-size": "1.2.0", - "type-fest": "0.3.1", - "widest-line": "2.0.1" + "ansi-align": "^3.0.0", + "camelcase": "^5.3.1", + "chalk": "^2.4.2", + "cli-boxes": "^2.2.0", + "string-width": "^3.0.0", + "term-size": "^1.2.0", + "type-fest": "^0.3.0", + "widest-line": "^2.0.0" }, "dependencies": { "ansi-regex": { @@ -3879,9 +3879,9 @@ "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==", "dev": true, "requires": { - "emoji-regex": "7.0.3", - "is-fullwidth-code-point": "2.0.0", - "strip-ansi": "5.2.0" + "emoji-regex": "^7.0.1", + "is-fullwidth-code-point": "^2.0.0", + "strip-ansi": "^5.1.0" } }, "strip-ansi": { @@ -3890,7 +3890,7 @@ "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", "dev": true, "requires": { - "ansi-regex": "4.1.0" + "ansi-regex": "^4.1.0" } }, "type-fest": { @@ -4019,9 +4019,9 @@ "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.6.6.tgz", "integrity": "sha512-D2Nk3W9JL9Fp/gIcWei8LrERCS+eXu9AM5cfXA8WEZ84lFks+ARnZ0q/R69m2SV3Wjma83QDDPxsNKXUwdIsyA==", "requires": { - "caniuse-lite": "1.0.30000989", - "electron-to-chromium": "1.3.273", - "node-releases": "1.1.33" + "caniuse-lite": "^1.0.30000984", + "electron-to-chromium": "^1.3.191", + "node-releases": "^1.1.25" } }, "browserstack": { @@ -4039,10 +4039,10 @@ "integrity": "sha512-fRaynjF0MvtyyfPRy2NFnVwxLyNtD28K/v9xRsIjUVf7xLc80NIm7Nfr3KXlFmWizhG91PL/UAOXlHkoxQjaNw==", "dev": true, "requires": { - "https-proxy-agent": "2.2.2", - "is-running": "2.1.0", - "ps-tree": "1.1.1", - "temp-fs": "0.9.9" + "https-proxy-agent": "^2.2.1", + "is-running": "^2.0.0", + "ps-tree": "=1.1.1", + "temp-fs": "^0.9.9" } }, "buffer": { @@ -4138,21 +4138,21 @@ "integrity": "sha512-ifKgxH2CKhJEg6tNdAwziu6Q33EvuG26tYcda6PT3WKisZcYDXsnEdnRv67Po3yCzFfaSoMjGZzJyD2c3DT1dg==", "dev": true, "requires": { - "bluebird": "3.7.0", - "chownr": "1.1.3", - "figgy-pudding": "3.5.1", - "glob": "7.1.4", - "graceful-fs": "4.2.2", - "infer-owner": "1.0.4", - "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.7.1", - "ssri": "6.0.1", - "unique-filename": "1.1.1", - "y18n": "4.0.0" + "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" }, "dependencies": { "glob": { @@ -4161,12 +4161,12 @@ "integrity": "sha512-hkLPepehmnKk41pUGm3sYxoFs/umurYfYJCerbXEyFIWcAzvpipAgVkBqqT9RBKMGjnq6kMuyYwha6csxbiM1A==", "dev": true, "requires": { - "fs.realpath": "1.0.0", - "inflight": "1.0.6", - "inherits": "2.0.4", - "minimatch": "3.0.4", - "once": "1.4.0", - "path-is-absolute": "1.0.1" + "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" } } } @@ -4193,13 +4193,13 @@ "integrity": "sha512-Oj3cAGPCqOZX7Rz64Uny2GYAZNliQSqfbePrgAQ1wKAihYmCUnraBtJtKcGR4xz7wF+LoJC+ssFZvv5BgF9Igg==", "dev": true, "requires": { - "clone-response": "1.0.2", - "get-stream": "5.1.0", - "http-cache-semantics": "4.0.3", - "keyv": "3.1.0", - "lowercase-keys": "2.0.0", - "normalize-url": "4.5.0", - "responselike": "1.0.2" + "clone-response": "^1.0.2", + "get-stream": "^5.1.0", + "http-cache-semantics": "^4.0.0", + "keyv": "^3.0.0", + "lowercase-keys": "^2.0.0", + "normalize-url": "^4.1.0", + "responselike": "^1.0.2" }, "dependencies": { "get-stream": { @@ -4208,7 +4208,7 @@ "integrity": "sha512-EXr1FOzrzTfGeL0gQdeFEvOMm2mzMOglyiOXSTpPC+iAjAKftbr3jpCMWynogwYnM+eSj9sHGc6wjIcDvYiygw==", "dev": true, "requires": { - "pump": "3.0.0" + "pump": "^3.0.0" } }, "http-cache-semantics": { @@ -4235,8 +4235,8 @@ "integrity": "sha512-LwZy+p3SFs1Pytd/jYct4wpv49HiYCqd9Rlc5ZVdk0V+8Yzv6jR5Blk3TRmPL1ft69TxP0IMZGJ+WPFU2BFhww==", "dev": true, "requires": { - "end-of-stream": "1.4.3", - "once": "1.4.0" + "end-of-stream": "^1.1.0", + "once": "^1.3.1" } } } @@ -4247,7 +4247,7 @@ "integrity": "sha1-hH4PzgoiN1CpoCfFSzNzGtMVQTQ=", "dev": true, "requires": { - "callsites": "2.0.0" + "callsites": "^2.0.0" } }, "caller-path": { @@ -4256,7 +4256,7 @@ "integrity": "sha1-Ro+DBE42mrIBD6xfBs7uFbsssfQ=", "dev": true, "requires": { - "caller-callsite": "2.0.0" + "caller-callsite": "^2.0.0" } }, "callsite": { @@ -4299,9 +4299,9 @@ "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz", "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==", "requires": { - "ansi-styles": "3.2.1", - "escape-string-regexp": "1.0.5", - "supports-color": "5.5.0" + "ansi-styles": "^3.2.1", + "escape-string-regexp": "^1.0.5", + "supports-color": "^5.3.0" }, "dependencies": { "supports-color": { @@ -4309,7 +4309,7 @@ "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", "requires": { - "has-flag": "3.0.0" + "has-flag": "^3.0.0" } } } @@ -4325,19 +4325,19 @@ "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-2.0.4.tgz", "integrity": "sha512-z9n7yt9rOvIJrMhvDtDictKrkFHeihkNl6uWMmZlmL6tJtX9Cs+87oK+teBx+JIgzvbX3yZHT3eF8vpbDxHJXQ==", "requires": { - "anymatch": "2.0.0", - "async-each": "1.0.3", - "braces": "2.3.2", - "fsevents": "1.2.9", - "glob-parent": "3.1.0", - "inherits": "2.0.4", - "is-binary-path": "1.0.1", - "is-glob": "4.0.1", - "lodash.debounce": "4.0.8", - "normalize-path": "2.1.1", - "path-is-absolute": "1.0.1", - "readdirp": "2.2.1", - "upath": "1.2.0" + "anymatch": "^2.0.0", + "async-each": "^1.0.0", + "braces": "^2.3.0", + "fsevents": "^1.2.2", + "glob-parent": "^3.1.0", + "inherits": "^2.0.1", + "is-binary-path": "^1.0.0", + "is-glob": "^4.0.0", + "lodash.debounce": "^4.0.8", + "normalize-path": "^2.1.1", + "path-is-absolute": "^1.0.0", + "readdirp": "^2.0.0", + "upath": "^1.0.5" } }, "chownr": { @@ -4352,7 +4352,7 @@ "integrity": "sha512-9e/zx1jw7B4CO+c/RXoCsfg/x1AfUBioy4owYH0bJprEYAx5hRFLRhWBqHAG57D0ZM4H7vxbP7bPe0VwhQRYDQ==", "dev": true, "requires": { - "tslib": "1.10.0" + "tslib": "^1.9.0" } }, "ci-info": { @@ -4432,7 +4432,7 @@ "integrity": "sha512-I/zHAwsKf9FqGoXM4WWRACob9+SNukZTd94DWF57E4toouRulbCxcUh6RKUEOQlYTHJnzkPMySvPNaaSLNfLZw==", "dev": true, "requires": { - "restore-cursor": "3.1.0" + "restore-cursor": "^3.1.0" } }, "cli-width": { @@ -4447,9 +4447,9 @@ "integrity": "sha1-EgYBU3qRbSmUD5NNo7SNWFo5IT0=", "dev": true, "requires": { - "string-width": "1.0.2", - "strip-ansi": "3.0.1", - "wrap-ansi": "2.1.0" + "string-width": "^1.0.1", + "strip-ansi": "^3.0.1", + "wrap-ansi": "^2.0.0" } }, "clone": { @@ -4470,9 +4470,9 @@ "integrity": "sha512-neHB9xuzh/wk0dIHweyAXv2aPGZIVk3pLMe+/RNzINf17fe0OG96QroktYAUm7SM1PBnzTabaLboqqxDyMU+SQ==", "dev": true, "requires": { - "is-plain-object": "2.0.4", - "kind-of": "6.0.2", - "shallow-clone": "3.0.1" + "is-plain-object": "^2.0.4", + "kind-of": "^6.0.2", + "shallow-clone": "^3.0.0" } }, "clone-response": { @@ -4481,7 +4481,7 @@ "integrity": "sha1-0dyXOSAxTfZ/vrlCI7TuNQI56Ws=", "dev": true, "requires": { - "mimic-response": "1.0.1" + "mimic-response": "^1.0.0" } }, "clone-stats": { @@ -4496,9 +4496,9 @@ "integrity": "sha512-2EF8zTQOxYq70Y4XKtorQupqF0m49MBz2/yf5Bj+MHjvpG3Hy7sImifnqD6UA+TKYxeSV+u6qqQPawN5UvnpKQ==", "dev": true, "requires": { - "inherits": "2.0.4", - "process-nextick-args": "2.0.1", - "readable-stream": "2.3.6" + "inherits": "^2.0.1", + "process-nextick-args": "^2.0.0", + "readable-stream": "^2.3.5" } }, "co": { @@ -4543,15 +4543,15 @@ "integrity": "sha512-izfUfhEOOgAizszPlEDxo71DK/C4wprZw0vkY6UWcOSTQvN1JyfXf9DXwaV7WX+/JC+hH0ShXfdtGLA9Rca7LA==", "dev": true, "requires": { - "app-root-path": "2.2.1", - "aria-query": "3.0.0", - "axobject-query": "2.0.2", - "css-selector-tokenizer": "0.7.1", - "cssauron": "1.4.0", - "damerau-levenshtein": "1.0.5", - "semver-dsl": "1.0.1", - "source-map": "0.5.7", - "sprintf-js": "1.1.2" + "app-root-path": "^2.2.1", + "aria-query": "^3.0.0", + "axobject-query": "^2.0.2", + "css-selector-tokenizer": "^0.7.1", + "cssauron": "^1.4.0", + "damerau-levenshtein": "^1.0.4", + "semver-dsl": "^1.0.1", + "source-map": "^0.5.7", + "sprintf-js": "^1.1.2" }, "dependencies": { "source-map": { @@ -4619,7 +4619,7 @@ "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==", "dev": true, "requires": { - "delayed-stream": "1.0.0" + "delayed-stream": "~1.0.0" } }, "commander": { @@ -4661,7 +4661,7 @@ "integrity": "sha512-BGHeLCK1GV7j1bSmQQAi26X+GgWcTjLr/0tzSvMCl3LH1w1IJ4PFSPoV5316b30cneTziC+B1a+3OjoSUcQYmw==", "dev": true, "requires": { - "mime-db": "1.40.0" + "mime-db": ">= 1.40.0 < 2" } }, "compression": { @@ -4670,13 +4670,13 @@ "integrity": "sha512-jaSIDzP9pZVS4ZfQ+TzvtiWhdpFhE2RDHz8QJkpX9SIpLq88VueF5jJw6t+6CUQcAoA6t+x89MLrWAqpfDE8iQ==", "dev": true, "requires": { - "accepts": "1.3.7", + "accepts": "~1.3.5", "bytes": "3.0.0", - "compressible": "2.0.17", + "compressible": "~2.0.16", "debug": "2.6.9", - "on-headers": "1.0.2", + "on-headers": "~1.0.2", "safe-buffer": "5.1.2", - "vary": "1.1.2" + "vary": "~1.1.2" } }, "concat-map": { @@ -4702,12 +4702,12 @@ "integrity": "sha512-CmquAXFBocrzaSM8mtGPMM/HiWmyIpr4CcJl/rgY2uCObZ/S7cKU0silxslqJejl+t/T9HS8E0PUNQD81JGUEQ==", "dev": true, "requires": { - "dot-prop": "4.2.0", - "graceful-fs": "4.2.2", - "make-dir": "1.3.0", - "unique-string": "1.0.0", - "write-file-atomic": "2.4.3", - "xdg-basedir": "3.0.0" + "dot-prop": "^4.1.0", + "graceful-fs": "^4.1.2", + "make-dir": "^1.0.0", + "unique-string": "^1.0.0", + "write-file-atomic": "^2.0.0", + "xdg-basedir": "^3.0.0" }, "dependencies": { "make-dir": { @@ -4716,7 +4716,7 @@ "integrity": "sha512-2w31R7SJtieJJnQtGc7RVL2StM2vGYVfqUOvUDxH6bC6aJTxPxTF0GnIgCyu7tjockiUWAYQRbxa7vKn34s5sQ==", "dev": true, "requires": { - "pify": "3.0.0" + "pify": "^3.0.0" } } } @@ -4780,7 +4780,7 @@ "resolved": "https://registry.npmjs.org/convert-source-map/-/convert-source-map-1.6.0.tgz", "integrity": "sha512-eFu7XigvxdZ1ETfbgPBohgyQ/Z++C0eEhTor0qRwBw9unw+L0/6V8wkSuGgzdThkiS5lSpdptOQPD8Ak40a+7A==", "requires": { - "safe-buffer": "5.1.2" + "safe-buffer": "~5.1.1" } }, "cookie": { @@ -4830,18 +4830,18 @@ "integrity": "sha512-YBuYGpSzoCHSSDGyHy6VJ7SHojKp6WHT4D7ItcQFNAYx2hrwkMe56e97xfVR0/ovDuMTrMffXUiltvQljtAGeg==", "dev": true, "requires": { - "cacache": "11.3.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": "1.9.1", - "webpack-log": "2.0.0" + "cacache": "^11.3.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.0", + "schema-utils": "^1.0.0", + "serialize-javascript": "^1.7.0", + "webpack-log": "^2.0.0" }, "dependencies": { "cacache": { @@ -4850,20 +4850,20 @@ "integrity": "sha512-p8WcneCytvzPxhDvYp31PD039vi77I12W+/KfR9S8AZbaiARFBCpsPJS+9uhWfeBfeAtW7o/4vt3MUqLkbY6nA==", "dev": true, "requires": { - "bluebird": "3.7.0", - "chownr": "1.1.3", - "figgy-pudding": "3.5.1", - "glob": "7.1.4", - "graceful-fs": "4.2.2", - "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.7.1", - "ssri": "6.0.1", - "unique-filename": "1.1.1", - "y18n": "4.0.0" + "bluebird": "^3.5.5", + "chownr": "^1.1.1", + "figgy-pudding": "^3.5.1", + "glob": "^7.1.4", + "graceful-fs": "^4.1.15", + "lru-cache": "^5.1.1", + "mississippi": "^3.0.0", + "mkdirp": "^0.5.1", + "move-concurrently": "^1.0.1", + "promise-inflight": "^1.0.1", + "rimraf": "^2.6.3", + "ssri": "^6.0.1", + "unique-filename": "^1.1.1", + "y18n": "^4.0.0" } }, "find-cache-dir": { @@ -4872,9 +4872,9 @@ "integrity": "sha512-Tq6PixE0w/VMFfCgbONnkiQIVol/JJL7nRMi20fqzA4NRs9AfeqMGeRdPi3wIhYkxjeBaWh2rxwapn5Tu3IqOQ==", "dev": true, "requires": { - "commondir": "1.0.1", - "make-dir": "2.1.0", - "pkg-dir": "3.0.0" + "commondir": "^1.0.1", + "make-dir": "^2.0.0", + "pkg-dir": "^3.0.0" } }, "glob": { @@ -4883,12 +4883,12 @@ "integrity": "sha512-hkLPepehmnKk41pUGm3sYxoFs/umurYfYJCerbXEyFIWcAzvpipAgVkBqqT9RBKMGjnq6kMuyYwha6csxbiM1A==", "dev": true, "requires": { - "fs.realpath": "1.0.0", - "inflight": "1.0.6", - "inherits": "2.0.4", - "minimatch": "3.0.4", - "once": "1.4.0", - "path-is-absolute": "1.0.1" + "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" } }, "globby": { @@ -4897,12 +4897,12 @@ "integrity": "sha1-+yzP+UAfhgCUXfral0QMypcrhoA=", "dev": true, "requires": { - "array-union": "1.0.2", - "dir-glob": "2.2.2", - "glob": "7.1.4", - "ignore": "3.3.10", - "pify": "3.0.0", - "slash": "1.0.0" + "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" } }, "normalize-path": { @@ -4923,8 +4923,8 @@ "resolved": "https://registry.npmjs.org/core-js-compat/-/core-js-compat-3.2.1.tgz", "integrity": "sha512-MwPZle5CF9dEaMYdDeWm73ao/IflDH+FjeJCWEADcEgFSE9TLimFKwJsfmkwzI8eC0Aj0mgvMDjeQjrElkz4/A==", "requires": { - "browserslist": "4.6.6", - "semver": "6.3.0" + "browserslist": "^4.6.6", + "semver": "^6.3.0" }, "dependencies": { "semver": { @@ -4945,10 +4945,10 @@ "integrity": "sha512-H65gsXo1SKjf8zmrJ67eJk8aIRKV5ff2D4uKZIBZShbhGSpEmsQOPW/SKMKYhSTrqR7ufy6RP69rPogdaPh/kA==", "dev": true, "requires": { - "import-fresh": "2.0.0", - "is-directory": "0.3.1", - "js-yaml": "3.13.1", - "parse-json": "4.0.0" + "import-fresh": "^2.0.0", + "is-directory": "^0.3.1", + "js-yaml": "^3.13.1", + "parse-json": "^4.0.0" }, "dependencies": { "parse-json": { @@ -4957,8 +4957,8 @@ "integrity": "sha1-vjX1Qlvh9/bHRxhPmKeIy5lHfuA=", "dev": true, "requires": { - "error-ex": "1.3.2", - "json-parse-better-errors": "1.0.2" + "error-ex": "^1.3.1", + "json-parse-better-errors": "^1.0.1" } } } @@ -5006,11 +5006,11 @@ "integrity": "sha512-eTVLrBSt7fjbDygz805pMnstIs2VTBNkRm0qxZd+M7A5XDdxVRWO5MxGBXZhjY4cqLYLdtrGqRf8mBPmzwSpWQ==", "dev": true, "requires": { - "nice-try": "1.0.5", - "path-key": "2.0.1", - "semver": "5.6.0", - "shebang-command": "1.2.0", - "which": "1.3.1" + "nice-try": "^1.0.4", + "path-key": "^2.0.1", + "semver": "^5.5.0", + "shebang-command": "^1.2.0", + "which": "^1.2.9" } }, "crypto-browserify": { @@ -5061,9 +5061,9 @@ "integrity": "sha1-hqdj9Y7k18L2sQLkdkBQ3n7ZDGs=", "dev": true, "requires": { - "regenerate": "1.4.0", - "regjsgen": "0.2.0", - "regjsparser": "0.1.5" + "regenerate": "^1.2.1", + "regjsgen": "^0.2.0", + "regjsparser": "^0.1.4" } }, "regjsgen": { @@ -5078,7 +5078,7 @@ "integrity": "sha1-fuj4Tcb6eS0/0K4ijSS9lJ6tIFw=", "dev": true, "requires": { - "jsesc": "0.5.0" + "jsesc": "~0.5.0" } } } @@ -5122,8 +5122,8 @@ "integrity": "sha512-m62ShEObQ39CfralilEQRjH6oAMtNCV1xJyEx5LpRYUVN+EviphDgUc/F3hnYbADmkiNs67Y+3ylmlG7Lnu+FA==", "dev": true, "requires": { - "es5-ext": "0.10.51", - "type": "1.2.0" + "es5-ext": "^0.10.50", + "type": "^1.0.1" } }, "d3": { @@ -5252,11 +5252,11 @@ "resolved": "https://registry.npmjs.org/d3-brush/-/d3-brush-1.1.3.tgz", "integrity": "sha512-v8bbYyCFKjyCzFk/tdWqXwDykY8YWqhXYjcYxfILIit085VZOpj4XJKOMccTsvWxgzSLMJQg5SiqHjslsipEDg==", "requires": { - "d3-dispatch": "1.0.3", - "d3-drag": "1.2.1", - "d3-interpolate": "1.3.2", - "d3-selection": "1.4.0", - "d3-transition": "1.1.1" + "d3-dispatch": "1", + "d3-drag": "1", + "d3-interpolate": "1", + "d3-selection": "1", + "d3-transition": "1" } }, "d3-chord": { @@ -5312,10 +5312,10 @@ "resolved": "https://registry.npmjs.org/d3-force/-/d3-force-1.2.1.tgz", "integrity": "sha512-HHvehyaiUlVo5CxBJ0yF/xny4xoaxFxDnBXNvNcfW9adORGZfyNF1dj6DGLKyk4Yh3brP/1h3rnDzdIAwL08zg==", "requires": { - "d3-collection": "1.0.4", - "d3-dispatch": "1.0.3", - "d3-quadtree": "1.0.3", - "d3-timer": "1.0.7" + "d3-collection": "1", + "d3-dispatch": "1", + "d3-quadtree": "1", + "d3-timer": "1" } }, "d3-format": { @@ -5404,7 +5404,7 @@ "resolved": "https://registry.npmjs.org/d3-shape/-/d3-shape-1.3.5.tgz", "integrity": "sha512-VKazVR3phgD+MUCldapHD7P9kcrvPcexeX/PkMJmkUov4JM8IxsSg1DvbYoYich9AtdTsa5nNk2++ImPiDiSxg==", "requires": { - "d3-path": "1.0.5" + "d3-path": "1" } }, "d3-time": { @@ -5513,7 +5513,7 @@ "integrity": "sha1-gKTdMjdIOEv6JICDYirt7Jgq3/M=", "dev": true, "requires": { - "mimic-response": "1.0.1" + "mimic-response": "^1.0.0" } }, "deep-equal": { @@ -5522,12 +5522,12 @@ "integrity": "sha512-ZbfWJq/wN1Z273o7mUSjILYqehAktR2NVoSrOukDkU9kg2v/Uv89yU4Cvz8seJeAmtN5oqiefKq8FPuXOboqLw==", "dev": true, "requires": { - "is-arguments": "1.0.4", - "is-date-object": "1.0.1", - "is-regex": "1.0.4", - "object-is": "1.0.1", - "object-keys": "1.1.1", - "regexp.prototype.flags": "1.2.0" + "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-extend": { @@ -5570,8 +5570,8 @@ "integrity": "sha512-h6sMrVB1VMWVrW13mSc6ia/DwYYw5MN6+exNu1OaJeFac5aSAvwM7lZ0NVfTABuSkQelr4h5oebg3KB1XPdjgA==", "dev": true, "requires": { - "execa": "1.0.0", - "ip-regex": "2.1.0" + "execa": "^1.0.0", + "ip-regex": "^2.1.0" } }, "default-require-extensions": { @@ -5580,7 +5580,7 @@ "integrity": "sha1-836hXT4T/9m0N9M+GnW1+5eHTLg=", "dev": true, "requires": { - "strip-bom": "2.0.0" + "strip-bom": "^2.0.0" } }, "default-resolution": { @@ -5600,7 +5600,7 @@ "resolved": "https://registry.npmjs.org/define-properties/-/define-properties-1.1.3.tgz", "integrity": "sha512-3MqfYKj2lLzdMSf8ZIZE/V+Zuy+BgD6f164e8K2w7dgnpKArBDerGYpM46IYYcjnkdPNMjPk9A6VFB8+3SKlXQ==", "requires": { - "object-keys": "1.1.1" + "object-keys": "^1.0.12" } }, "define-property": { @@ -5646,13 +5646,13 @@ "integrity": "sha512-QwGuEUouP2kVwQenAsOof5Fv8K9t3D8Ca8NxcXKrIpEHjTXK5J2nXLdP+ALI1cgv8wj7KuwBhTwBkOZSJKM5XQ==", "dev": true, "requires": { - "@types/glob": "7.1.1", - "globby": "6.1.0", - "is-path-cwd": "2.2.0", - "is-path-in-cwd": "2.1.0", - "p-map": "2.1.0", - "pify": "4.0.1", - "rimraf": "2.7.1" + "@types/glob": "^7.1.1", + "globby": "^6.1.0", + "is-path-cwd": "^2.0.0", + "is-path-in-cwd": "^2.0.0", + "p-map": "^2.0.0", + "pify": "^4.0.1", + "rimraf": "^2.6.3" }, "dependencies": { "globby": { @@ -5661,11 +5661,11 @@ "integrity": "sha1-9abXDoOV4hyFj7BInWTfAkJNUGw=", "dev": true, "requires": { - "array-union": "1.0.2", - "glob": "7.1.3", - "object-assign": "4.1.1", - "pify": "2.3.0", - "pinkie-promise": "2.0.1" + "array-union": "^1.0.1", + "glob": "^7.0.3", + "object-assign": "^4.0.1", + "pify": "^2.0.0", + "pinkie-promise": "^2.0.0" }, "dependencies": { "pify": { @@ -5688,7 +5688,7 @@ "integrity": "sha512-rNocXHgipO+rvnP6dk3zI20RpOtrAM/kzbB258Uw5BWr3TpXi861yzjo16Dn4hUox07iw5AyeMLHWsujkjzvRQ==", "dev": true, "requires": { - "is-path-inside": "2.1.0" + "is-path-inside": "^2.1.0" } }, "is-path-inside": { @@ -5697,7 +5697,7 @@ "integrity": "sha512-wiyhTzfDWsvwAW53OBWF5zuvaOGlZ6PwYxAbPVDhpm+gM09xKQGjBq/8uYN12aDvMxnAnq3dxTyoSoRNmg5YFg==", "dev": true, "requires": { - "path-is-inside": "1.0.2" + "path-is-inside": "^1.0.2" } }, "pify": { @@ -5720,10 +5720,10 @@ "integrity": "sha512-bdhJatRNYsJnOhSRx9Eej3ABBtxQQw/uz2RprpYL5R3jCC2XMYVBcQWwvQLl+iNDk4LCLEKhdIP3uZSqRWi/tw==", "dev": true, "requires": { - "async-each": "1.0.3", - "extend-shallow": "2.0.1", - "matched": "1.0.2", - "rimraf": "2.7.1" + "async-each": "^1.0.1", + "extend-shallow": "^2.0.1", + "matched": "^1.0.2", + "rimraf": "^2.6.1" }, "dependencies": { "extend-shallow": { @@ -5732,7 +5732,7 @@ "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", "dev": true, "requires": { - "is-extendable": "0.1.1" + "is-extendable": "^0.1.0" } } } @@ -5792,8 +5792,8 @@ "integrity": "sha1-f3Qt4Gb8dIvI24IFad3c5Jvw1FY=", "dev": true, "requires": { - "asap": "2.0.6", - "wrappy": "1.0.2" + "asap": "^2.0.0", + "wrappy": "1" } }, "di": { @@ -5825,7 +5825,7 @@ "integrity": "sha512-f9LBi5QWzIW3I6e//uxZoLBlUt9kcp66qo0sSCxL6YZKc75R1c4MFCoe/LaZiBGmgujvQdxc5Bn3QhfyvK5Hsw==", "dev": true, "requires": { - "path-type": "3.0.0" + "path-type": "^3.0.0" } }, "dns-equal": { @@ -5913,10 +5913,10 @@ "integrity": "sha512-07z8uv2wMyS51kKhD1KsdXJg5WQ6t93RneqRxUHnskXVtlYYkLqM0gqStQZ3pj073g687jPCHrqNfCzawLYh5g==", "dev": true, "requires": { - "end-of-stream": "1.4.3", - "inherits": "2.0.4", - "readable-stream": "2.3.6", - "stream-shift": "1.0.0" + "end-of-stream": "^1.0.0", + "inherits": "^2.0.1", + "readable-stream": "^2.0.0", + "stream-shift": "^1.0.0" } }, "each-props": { @@ -5935,8 +5935,8 @@ "integrity": "sha1-OoOpBOVDUyh4dMVkt1SThoSamMk=", "dev": true, "requires": { - "jsbn": "0.1.1", - "safer-buffer": "2.1.2" + "jsbn": "~0.1.0", + "safer-buffer": "^2.1.0" } }, "ee-first": { @@ -5956,13 +5956,13 @@ "integrity": "sha512-xvJINNLbTeWQjrl6X+7eQCrIy/YPv5XCpKW6kB5mKvtnGILoLDcySuwomfdzt0BMdLNVnuRNTuzKNHj0bva1Cg==", "dev": true, "requires": { - "bn.js": "4.11.8", - "brorand": "1.1.0", - "hash.js": "1.1.7", - "hmac-drbg": "1.0.1", - "inherits": "2.0.4", - "minimalistic-assert": "1.0.1", - "minimalistic-crypto-utils": "1.0.1" + "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": { @@ -5997,7 +5997,7 @@ "integrity": "sha512-cbNhPFS6MlYlWTGncSiDYbdqKhwWFy7kNeb1YSOG6K65i/wPTkLVCJQj0hXA4j0m5Da+hBWnqopEnu1FFelisQ==", "dev": true, "requires": { - "once": "1.4.0" + "once": "^1.4.0" } }, "engine.io": { @@ -6035,9 +6035,9 @@ "integrity": "sha512-nnWLa/NwZSt4KQJu51MYlCcSQ5g7INpOrOMt4XV8j4dqTXdmlUmSHQ8/oLC069ckre0fRsgfvsKwbTdtKLCDkA==", "dev": true, "requires": { - "async-limiter": "1.0.1", - "safe-buffer": "5.1.2", - "ultron": "1.1.1" + "async-limiter": "~1.0.0", + "safe-buffer": "~5.1.0", + "ultron": "~1.1.0" } } } @@ -6082,9 +6082,9 @@ "integrity": "sha512-nnWLa/NwZSt4KQJu51MYlCcSQ5g7INpOrOMt4XV8j4dqTXdmlUmSHQ8/oLC069ckre0fRsgfvsKwbTdtKLCDkA==", "dev": true, "requires": { - "async-limiter": "1.0.1", - "safe-buffer": "5.1.2", - "ultron": "1.1.1" + "async-limiter": "~1.0.0", + "safe-buffer": "~5.1.0", + "ultron": "~1.1.0" } } } @@ -6138,7 +6138,7 @@ "integrity": "sha512-7dFHNmqeFSEt2ZBsCriorKnn3Z2pj+fd9kmI6QoWw4//DL+icEBfc0U7qJCisqrTsKTjw4fNFy2pW9OqStD84g==", "dev": true, "requires": { - "is-arrayish": "0.2.1" + "is-arrayish": "^0.2.1" } }, "error-stack-parser": { @@ -6146,7 +6146,7 @@ "resolved": "https://registry.npmjs.org/error-stack-parser/-/error-stack-parser-2.0.4.tgz", "integrity": "sha512-fZ0KkoxSjLFmhW5lHbUT3tLwy3nX1qEzMYo8koY1vrsAco53CMT1djnBSeC/wUjTEZRhZl9iRw7PaMaxfJ4wzQ==", "requires": { - "stackframe": "1.1.0" + "stackframe": "^1.1.0" } }, "es-abstract": { @@ -6155,16 +6155,16 @@ "integrity": "sha512-DgoQmbpFNOofkjJtKwr87Ma5EW4Dc8fWhD0R+ndq7Oc456ivUfGOOP6oAZTTKl5/CcNMP+EN+e3/iUzgE0veZg==", "dev": true, "requires": { - "es-to-primitive": "1.2.0", - "function-bind": "1.1.1", - "has": "1.0.3", - "has-symbols": "1.0.0", - "is-callable": "1.1.4", - "is-regex": "1.0.4", - "object-inspect": "1.6.0", - "object-keys": "1.1.1", - "string.prototype.trimleft": "2.1.0", - "string.prototype.trimright": "2.1.0" + "es-to-primitive": "^1.2.0", + "function-bind": "^1.1.1", + "has": "^1.0.3", + "has-symbols": "^1.0.0", + "is-callable": "^1.1.4", + "is-regex": "^1.0.4", + "object-inspect": "^1.6.0", + "object-keys": "^1.1.1", + "string.prototype.trimleft": "^2.0.0", + "string.prototype.trimright": "^2.0.0" } }, "es-to-primitive": { @@ -6173,9 +6173,9 @@ "integrity": "sha512-qZryBOJjV//LaxLTV6UC//WewneB3LcXOL9NP++ozKVXsIIIpm/2c13UDiD9Jp2eThsecw9m3jPqDwTyobcdbg==", "dev": true, "requires": { - "is-callable": "1.1.4", - "is-date-object": "1.0.1", - "is-symbol": "1.0.2" + "is-callable": "^1.1.4", + "is-date-object": "^1.0.1", + "is-symbol": "^1.0.2" } }, "es5-ext": { @@ -6184,9 +6184,9 @@ "integrity": "sha512-oRpWzM2WcLHVKpnrcyB7OW8j/s67Ba04JCm0WnNv3RiABSvs7mrQlutB8DBv793gKcp0XENR8Il8WxGTlZ73gQ==", "dev": true, "requires": { - "es6-iterator": "2.0.3", - "es6-symbol": "3.1.2", - "next-tick": "1.0.0" + "es6-iterator": "~2.0.3", + "es6-symbol": "~3.1.1", + "next-tick": "^1.0.0" } }, "es6-iterator": { @@ -6248,8 +6248,8 @@ "integrity": "sha1-vwDvT9q2uhtG7Le2KbTH7VcVzHc=", "dev": true, "requires": { - "d": "1.0.1", - "es5-ext": "0.10.51" + "d": "1", + "es5-ext": "~0.10.14" } } } @@ -6260,8 +6260,8 @@ "integrity": "sha512-/ZypxQsArlv+KHpGvng52/Iz8by3EQPxhmbuz8yFG89N/caTFBSbcXONDw0aMjy827gQg26XAjP4uXFvnfINmQ==", "dev": true, "requires": { - "d": "1.0.1", - "es5-ext": "0.10.51" + "d": "^1.0.1", + "es5-ext": "^0.10.51" } }, "es6-weak-map": { @@ -6270,10 +6270,10 @@ "integrity": "sha512-p5um32HOTO1kP+w7PRnB+5lQ43Z6muuMuIMffvDN8ZB4GcnjLBV6zGStpbASIMk4DCAvEaamhe2zhyCb/QXXsA==", "dev": true, "requires": { - "d": "1.0.1", - "es5-ext": "0.10.51", - "es6-iterator": "2.0.3", - "es6-symbol": "3.1.2" + "d": "1", + "es5-ext": "^0.10.46", + "es6-iterator": "^2.0.3", + "es6-symbol": "^3.1.1" } }, "escape-html": { @@ -6408,7 +6408,7 @@ "integrity": "sha1-ZNo/fValRBLll5S9Ytw1KV6PKYc=", "dev": true, "requires": { - "restore-cursor": "1.0.1" + "restore-cursor": "^1.0.1" } }, "figures": { @@ -6417,8 +6417,8 @@ "integrity": "sha1-y+Hjr/zxzUS4DK3+0o3Hk6lwHS4=", "dev": true, "requires": { - "escape-string-regexp": "1.0.5", - "object-assign": "4.1.1" + "escape-string-regexp": "^1.0.5", + "object-assign": "^4.1.0" } }, "inquirer": { @@ -6427,19 +6427,19 @@ "integrity": "sha1-HvK/1jUE3wvHV4X/+MLEHfEvB34=", "dev": true, "requires": { - "ansi-escapes": "1.4.0", - "ansi-regex": "2.1.1", - "chalk": "1.1.3", - "cli-cursor": "1.0.2", - "cli-width": "2.2.0", - "figures": "1.7.0", - "lodash": "4.17.15", - "readline2": "1.0.1", - "run-async": "0.1.0", - "rx-lite": "3.1.2", - "string-width": "1.0.2", - "strip-ansi": "3.0.1", - "through": "2.3.8" + "ansi-escapes": "^1.1.0", + "ansi-regex": "^2.0.0", + "chalk": "^1.0.0", + "cli-cursor": "^1.0.1", + "cli-width": "^2.0.0", + "figures": "^1.3.5", + "lodash": "^4.3.0", + "readline2": "^1.0.1", + "run-async": "^0.1.0", + "rx-lite": "^3.1.2", + "string-width": "^1.0.1", + "strip-ansi": "^3.0.0", + "through": "^2.3.6" } }, "onetime": { @@ -6454,8 +6454,8 @@ "integrity": "sha1-NGYfRohjJ/7SmRR5FSJS35LapUE=", "dev": true, "requires": { - "exit-hook": "1.1.1", - "onetime": "1.1.0" + "exit-hook": "^1.0.0", + "onetime": "^1.0.0" } }, "run-async": { @@ -6464,7 +6464,7 @@ "integrity": "sha1-yK1KXhEGYeQCp9IbUw4AnyX444k=", "dev": true, "requires": { - "once": "1.4.0" + "once": "^1.3.0" } }, "shelljs": { @@ -6493,8 +6493,8 @@ "integrity": "sha512-p7VutNr1O/QrxysMo3E45FjYDTeXBy0iTltPFNSqKAIfjDSXC+4dj+qfyuD8bfAXrW/y6lW3O76VaYNPKfpKrg==", "dev": true, "requires": { - "esrecurse": "4.2.1", - "estraverse": "4.3.0" + "esrecurse": "^4.1.0", + "estraverse": "^4.1.1" } }, "espree": { @@ -6615,13 +6615,13 @@ "integrity": "sha512-adbxcyWV46qiHyvSp50TKt05tB4tK3HcmF7/nxfAdhnox83seTDbwnaqKO4sXRy7roHAIFqJP/Rw/AuEbX61LA==", "dev": true, "requires": { - "cross-spawn": "6.0.5", - "get-stream": "4.1.0", - "is-stream": "1.1.0", - "npm-run-path": "2.0.2", - "p-finally": "1.0.0", - "signal-exit": "3.0.2", - "strip-eof": "1.0.0" + "cross-spawn": "^6.0.0", + "get-stream": "^4.0.0", + "is-stream": "^1.1.0", + "npm-run-path": "^2.0.0", + "p-finally": "^1.0.0", + "signal-exit": "^3.0.0", + "strip-eof": "^1.0.0" } }, "exit": { @@ -6674,7 +6674,7 @@ "integrity": "sha1-l+gBqgUt8CRU3kawK/YhZCzchQI=", "dev": true, "requires": { - "homedir-polyfill": "1.0.3" + "homedir-polyfill": "^1.0.1" } }, "express": { @@ -6683,36 +6683,36 @@ "integrity": "sha512-mHJ9O79RqluphRrcw2X/GTh3k9tVv8YcoyY4Kkh4WDMUYKRZUq0h1o0w2rrrxBqM7VoeUVqgb27xlEMXTnYt4g==", "dev": true, "requires": { - "accepts": "1.3.7", + "accepts": "~1.3.7", "array-flatten": "1.1.1", "body-parser": "1.19.0", "content-disposition": "0.5.3", - "content-type": "1.0.4", + "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", + "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", + "methods": "~1.1.2", + "on-finished": "~2.3.0", + "parseurl": "~1.3.3", "path-to-regexp": "0.1.7", - "proxy-addr": "2.0.5", + "proxy-addr": "~2.0.5", "qs": "6.7.0", - "range-parser": "1.2.1", + "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", + "statuses": "~1.5.0", + "type-is": "~1.6.18", "utils-merge": "1.0.1", - "vary": "1.1.2" + "vary": "~1.1.2" }, "dependencies": { "array-flatten": { @@ -6760,9 +6760,9 @@ "integrity": "sha512-hMQ4CX1p1izmuLYyZqLMO/qGNw10wSv9QDCPfzXfyFrOaCSSoRfqE1Kf1s5an66J5JZC62NewG+mK49jOCtQew==", "dev": true, "requires": { - "chardet": "0.7.0", - "iconv-lite": "0.4.24", - "tmp": "0.0.33" + "chardet": "^0.7.0", + "iconv-lite": "^0.4.24", + "tmp": "^0.0.33" } }, "extglob": { @@ -6853,12 +6853,12 @@ "integrity": "sha512-wkIbV6qg37xTJwqSsdnIphL1e+LaGz4AIQqr00mIubMaEhv1/HEmJ0uuCGZRNRUkZZmOB5mJKO0ZUTVq+SxMQg==", "dev": true, "requires": { - "@nodelib/fs.stat": "2.0.2", - "@nodelib/fs.walk": "1.2.3", - "glob-parent": "5.1.0", - "is-glob": "4.0.1", - "merge2": "1.3.0", - "micromatch": "4.0.2" + "@nodelib/fs.stat": "^2.0.1", + "@nodelib/fs.walk": "^1.2.1", + "glob-parent": "^5.0.0", + "is-glob": "^4.0.1", + "merge2": "^1.2.3", + "micromatch": "^4.0.2" }, "dependencies": { "braces": { @@ -6867,7 +6867,7 @@ "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", "dev": true, "requires": { - "fill-range": "7.0.1" + "fill-range": "^7.0.1" } }, "fill-range": { @@ -6876,7 +6876,7 @@ "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", "dev": true, "requires": { - "to-regex-range": "5.0.1" + "to-regex-range": "^5.0.1" } }, "glob-parent": { @@ -6885,7 +6885,7 @@ "integrity": "sha512-qjtRgnIVmOfnKUE3NJAQEdk+lKrxfw8t5ke7SXtfMTHcjsBfOfWXCQfdb30zfDoZQ2IRSIiidmjtbHZPZ++Ihw==", "dev": true, "requires": { - "is-glob": "4.0.1" + "is-glob": "^4.0.1" } }, "is-number": { @@ -6900,8 +6900,8 @@ "integrity": "sha512-y7FpHSbMUMoyPbYUSzO6PaZ6FyRnQOpHuKwbo1G+Knck95XVU4QAiKdGEnj5wwoS7PlOgthX/09u5iFJ+aYf5Q==", "dev": true, "requires": { - "braces": "3.0.2", - "picomatch": "2.0.7" + "braces": "^3.0.1", + "picomatch": "^2.0.5" } }, "to-regex-range": { @@ -6910,7 +6910,7 @@ "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", "dev": true, "requires": { - "is-number": "7.0.0" + "is-number": "^7.0.0" } } } @@ -6938,7 +6938,7 @@ "integrity": "sha512-jmxqQ3Z/nXoeyDmWAzF9kH1aGZSis6e/SbfPmJpUnyZ0ogr6iscHQaml4wsEepEWSdtmpy+eVXmCRIMpxaXqOA==", "dev": true, "requires": { - "reusify": "1.0.4" + "reusify": "^1.0.0" } }, "faye-websocket": { @@ -6962,7 +6962,7 @@ "integrity": "sha512-HKri+WoWoUgr83pehn/SIgLOMZ9nAWC6dcGj26RY2R4F50u4+RTUz0RCrUlOV3nKRAICW1UGzyb+kcX2qK1S/g==", "dev": true, "requires": { - "escape-string-regexp": "1.0.5" + "escape-string-regexp": "^1.0.5" } }, "file-entry-cache": { @@ -6981,8 +6981,8 @@ "integrity": "sha512-+xZnaK5R8kBJrHK0/6HRlrKNamvVS5rjyuju+rnyxRGuwUJwpAMsVzUl5dz6rK8brkzjV6JpcFNjp6NqV0g1OQ==", "dev": true, "requires": { - "loader-utils": "1.2.3", - "schema-utils": "2.4.1" + "loader-utils": "^1.2.3", + "schema-utils": "^2.0.0" }, "dependencies": { "ajv": { @@ -6991,10 +6991,10 @@ "integrity": "sha512-TXtUUEYHuaTEbLZWIKUr5pmBuhDLy+8KYtPYdcV8qC+pOZL+NKqYwvWSRrVXHn+ZmRRAu8vJTAznH7Oag6RVRw==", "dev": true, "requires": { - "fast-deep-equal": "2.0.1", - "fast-json-stable-stringify": "2.0.0", - "json-schema-traverse": "0.4.1", - "uri-js": "4.2.2" + "fast-deep-equal": "^2.0.1", + "fast-json-stable-stringify": "^2.0.0", + "json-schema-traverse": "^0.4.1", + "uri-js": "^4.2.2" } }, "schema-utils": { @@ -7003,8 +7003,8 @@ "integrity": "sha512-RqYLpkPZX5Oc3fw/kHHHyP56fg5Y+XBpIpV8nCg0znIALfq3OH+Ea9Hfeac9BAMwG5IICltiZ0vxFvJQONfA5w==", "dev": true, "requires": { - "ajv": "6.10.2", - "ajv-keywords": "3.4.1" + "ajv": "^6.10.2", + "ajv-keywords": "^3.4.1" } } } @@ -7047,12 +7047,12 @@ "dev": true, "requires": { "debug": "2.6.9", - "encodeurl": "1.0.2", - "escape-html": "1.0.3", - "on-finished": "2.3.0", - "parseurl": "1.3.3", - "statuses": "1.5.0", - "unpipe": "1.0.0" + "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": { @@ -7061,9 +7061,9 @@ "integrity": "sha512-t7ulV1fmbxh5G9l/492O1p5+EBbr3uwpt6odhFTMc+nWyhmbloe+ja9BZ8pIBtqFWhOmCWVjx+pTW4zDkFoclw==", "dev": true, "requires": { - "commondir": "1.0.1", - "make-dir": "3.0.0", - "pkg-dir": "4.2.0" + "commondir": "^1.0.1", + "make-dir": "^3.0.0", + "pkg-dir": "^4.1.0" }, "dependencies": { "find-up": { @@ -7072,8 +7072,8 @@ "integrity": "sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==", "dev": true, "requires": { - "locate-path": "5.0.0", - "path-exists": "4.0.0" + "locate-path": "^5.0.0", + "path-exists": "^4.0.0" } }, "locate-path": { @@ -7082,7 +7082,7 @@ "integrity": "sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==", "dev": true, "requires": { - "p-locate": "4.1.0" + "p-locate": "^4.1.0" } }, "make-dir": { @@ -7091,7 +7091,7 @@ "integrity": "sha512-grNJDhb8b1Jm1qeqW5R/O63wUo4UXo2v2HMic6YT9i/HBlF93S8jkMgH7yugvY9ABDShH4VZMn8I+U8+fCNegw==", "dev": true, "requires": { - "semver": "6.3.0" + "semver": "^6.0.0" } }, "p-locate": { @@ -7100,7 +7100,7 @@ "integrity": "sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A==", "dev": true, "requires": { - "p-limit": "2.2.1" + "p-limit": "^2.2.0" } }, "path-exists": { @@ -7115,7 +7115,7 @@ "integrity": "sha512-HRDzbaKjC+AOWVXxAU/x54COGeIv9eb+6CkDSQoNTt4XyWoIJvuPsXizxu/Fr23EiekbtZwmh1IcIG/l/a10GQ==", "dev": true, "requires": { - "find-up": "4.1.0" + "find-up": "^4.0.0" } }, "semver": { @@ -7137,7 +7137,7 @@ "resolved": "https://registry.npmjs.org/find-up/-/find-up-3.0.0.tgz", "integrity": "sha512-1yD6RmLI1XBfxugvORwlck6f75tYL+iR0jqwsOrOxMZyGYqUuDhJ0l4AXdO1iX/FTs9cBAMEk1gWSEx1kSbylg==", "requires": { - "locate-path": "3.0.0" + "locate-path": "^3.0.0" } }, "findup-sync": { @@ -7233,10 +7233,10 @@ "integrity": "sha512-VwyB3Lkgacfik2vhqR4uv2rvebqmDvFu4jlN/C1RzWoJEo8I7z4Q404oiqYCkq41mni8EzQnm95emU9seckwtg==", "dev": true, "requires": { - "circular-json": "0.3.3", - "graceful-fs": "4.2.2", - "rimraf": "2.6.3", - "write": "0.2.1" + "circular-json": "^0.3.1", + "graceful-fs": "^4.1.2", + "rimraf": "~2.6.2", + "write": "^0.2.1" }, "dependencies": { "circular-json": { @@ -7251,7 +7251,7 @@ "integrity": "sha512-mwqeW5XsA2qAejG46gYdENaxXjx9onRNCfn7L0duuP4hCuTIi/QO7PDK07KJfp1d+izWPrzEJDcSqBa0OZQriA==", "dev": true, "requires": { - "glob": "7.1.3" + "glob": "^7.1.3" } } } @@ -7268,8 +7268,8 @@ "integrity": "sha512-3Z4XhFZ3992uIq0XOqb9AreonueSYphE6oYbpt5+3u06JWklbsPkNv3ZKkP9Bz/r+1MWCaMoSQ28P85+1Yc77w==", "dev": true, "requires": { - "inherits": "2.0.4", - "readable-stream": "2.3.6" + "inherits": "^2.0.3", + "readable-stream": "^2.3.6" } }, "follow-redirects": { @@ -7278,7 +7278,7 @@ "integrity": "sha512-CRcPzsSIbXyVDl0QI01muNDu69S8trU4jArW9LpOt2WtC6LyUJetcIrmfHsRBx7/Jb6GHJUiuqyYxPooFfNt6A==", "dev": true, "requires": { - "debug": "3.2.6" + "debug": "^3.0.0" }, "dependencies": { "debug": { @@ -7287,7 +7287,7 @@ "integrity": "sha512-mel+jf7nrtEl5Pn1Qx46zARXKDpBbvzezse7p7LqINmdoIk8PYP5SySaxEmYv6TZ0JyEKA1hsCId6DIhgITtWQ==", "dev": true, "requires": { - "ms": "2.1.2" + "ms": "^2.1.1" } }, "ms": { @@ -7323,14 +7323,14 @@ "resolved": "https://registry.npmjs.org/fork-ts-checker-webpack-plugin/-/fork-ts-checker-webpack-plugin-0.4.15.tgz", "integrity": "sha512-qNYuygh2GxXehBvQZ5rI5YlQFn+7ZV6kmkyD9Sgs33dWl73NZdUOB5aCp8v0EXJn176AhPrZP8YCMT3h01fs+g==", "requires": { - "babel-code-frame": "6.26.0", - "chalk": "2.4.2", - "chokidar": "2.0.4", - "lodash": "4.17.15", - "micromatch": "3.1.10", - "minimatch": "3.0.4", - "resolve": "1.12.0", - "tapable": "1.1.3" + "babel-code-frame": "^6.22.0", + "chalk": "^2.4.1", + "chokidar": "^2.0.4", + "lodash": "^4.17.11", + "micromatch": "^3.1.10", + "minimatch": "^3.0.4", + "resolve": "^1.5.0", + "tapable": "^1.0.0" } }, "form-data": { @@ -7339,9 +7339,9 @@ "integrity": "sha512-1lLKB2Mu3aGP1Q/2eCOx0fNbRMe7XdwktwOruhfqqd0rIJWwN4Dh+E3hrPSlDCXnSR7UtZ1N38rVXm+6+MEhJQ==", "dev": true, "requires": { - "asynckit": "0.4.0", - "combined-stream": "1.0.8", - "mime-types": "2.1.24" + "asynckit": "^0.4.0", + "combined-stream": "^1.0.6", + "mime-types": "^2.1.12" } }, "forwarded": { @@ -7395,9 +7395,9 @@ "integrity": "sha512-yhlQgA6mnOJUKOsRUFsgJdQCvkKhcz8tlZG5HBQfReYZy46OwLcY+Zia0mtdHsOo9y/hP+CxMN0TU9QxoOtG4g==", "dev": true, "requires": { - "graceful-fs": "4.2.2", - "jsonfile": "4.0.0", - "universalify": "0.1.2" + "graceful-fs": "^4.2.0", + "jsonfile": "^4.0.0", + "universalify": "^0.1.0" } }, "fs-minipass": { @@ -7406,7 +7406,7 @@ "integrity": "sha512-GWSSJGFy4e9GUeCcbIkED+bgAoFyj7XF1mV8rma3QW4NIqX9Kyx79N/PF61H5udOV3aY1IaMLs6pGbH71nlCTA==", "dev": true, "requires": { - "minipass": "2.9.0" + "minipass": "^2.6.0" } }, "fs-mkdirp-stream": { @@ -7443,8 +7443,8 @@ "integrity": "sha512-oeyj2H3EjjonWcFjD5NvZNE9Rqe4UW+nQBU2HNeKw0koVLEFIhtyETyAakeAM3de7Z/SW5kcA+fZUait9EApnw==", "optional": true, "requires": { - "nan": "2.14.0", - "node-pre-gyp": "0.12.0" + "nan": "^2.12.1", + "node-pre-gyp": "^0.12.0" }, "dependencies": { "abbrev": { @@ -7470,8 +7470,8 @@ "integrity": "sha512-5hYdAkZlcG8tOLujVDTgCT+uPX0VnpAH28gWsLfzpXYm7wP6mp5Q/gYyR7YQ0cKVJcXJnl3j2kpBan13PtQf6w==", "optional": true, "requires": { - "delegates": "1.0.0", - "readable-stream": "2.3.6" + "delegates": "^1.0.0", + "readable-stream": "^2.0.6" } }, "balanced-match": { @@ -7486,7 +7486,7 @@ "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", "optional": true, "requires": { - "balanced-match": "1.0.0", + "balanced-match": "^1.0.0", "concat-map": "0.0.1" } }, @@ -7526,7 +7526,7 @@ "integrity": "sha512-pYAIzeRo8J6KPEaJ0VWOh5Pzkbw/RetuzehGM7QRRX5he4fPHx2rdKMB256ehJCkX+XRQm16eZLqLNS8RSZXZw==", "optional": true, "requires": { - "ms": "2.1.1" + "ms": "^2.1.1" } }, "deep-extend": { @@ -7553,7 +7553,7 @@ "integrity": "sha512-JhBl0skXjUPCFH7x6x61gQxrKyXsxB5gcgePLZCwfyCGGsTISMoIeObbrvVeP6Xmyaudw4TT43qV2Gz+iyd2oQ==", "optional": true, "requires": { - "minipass": "2.3.5" + "minipass": "^2.2.1" } }, "fs.realpath": { @@ -7568,14 +7568,14 @@ "integrity": "sha1-LANAXHU4w51+s3sxcCLjJfsBi/c=", "optional": true, "requires": { - "aproba": "1.2.0", - "console-control-strings": "1.1.0", - "has-unicode": "2.0.1", - "object-assign": "4.1.1", - "signal-exit": "3.0.2", - "string-width": "1.0.2", - "strip-ansi": "3.0.1", - "wide-align": "1.1.3" + "aproba": "^1.0.3", + "console-control-strings": "^1.0.0", + "has-unicode": "^2.0.0", + "object-assign": "^4.1.0", + "signal-exit": "^3.0.0", + "string-width": "^1.0.1", + "strip-ansi": "^3.0.1", + "wide-align": "^1.1.0" } }, "glob": { @@ -7584,12 +7584,12 @@ "integrity": "sha512-vcfuiIxogLV4DlGBHIUOwI0IbrJ8HWPc4MU7HzviGeNho/UJDfi6B5p3sHeWIQ0KGIU0Jpxi5ZHxemQfLkkAwQ==", "optional": true, "requires": { - "fs.realpath": "1.0.0", - "inflight": "1.0.6", - "inherits": "2.0.3", - "minimatch": "3.0.4", - "once": "1.4.0", - "path-is-absolute": "1.0.1" + "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" } }, "has-unicode": { @@ -7604,7 +7604,7 @@ "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", "optional": true, "requires": { - "safer-buffer": "2.1.2" + "safer-buffer": ">= 2.1.2 < 3" } }, "ignore-walk": { @@ -7613,7 +7613,7 @@ "integrity": "sha512-DTVlMx3IYPe0/JJcYP7Gxg7ttZZu3IInhuEhbchuqneY9wWe5Ojy2mXLBaQFUQmo0AW2r3qG7m1mg86js+gnlQ==", "optional": true, "requires": { - "minimatch": "3.0.4" + "minimatch": "^3.0.4" } }, "inflight": { @@ -7622,8 +7622,8 @@ "integrity": "sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk=", "optional": true, "requires": { - "once": "1.4.0", - "wrappy": "1.0.2" + "once": "^1.3.0", + "wrappy": "1" } }, "inherits": { @@ -7644,7 +7644,7 @@ "integrity": "sha1-754xOG8DGn8NZDr4L95QxFfvAMs=", "optional": true, "requires": { - "number-is-nan": "1.0.1" + "number-is-nan": "^1.0.0" } }, "isarray": { @@ -7659,7 +7659,7 @@ "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==", "optional": true, "requires": { - "brace-expansion": "1.1.11" + "brace-expansion": "^1.1.7" } }, "minimist": { @@ -7674,8 +7674,8 @@ "integrity": "sha512-Gi1W4k059gyRbyVUZQ4mEqLm0YIUiGYfvxhF6SIlk3ui1WVxMTGfGdQ2SInh3PDrRTVvPKgULkpJtT4RH10+VA==", "optional": true, "requires": { - "safe-buffer": "5.1.2", - "yallist": "3.0.3" + "safe-buffer": "^5.1.2", + "yallist": "^3.0.0" } }, "minizlib": { @@ -7684,7 +7684,7 @@ "integrity": "sha512-7+4oTUOWKg7AuL3vloEWekXY2/D20cevzsrNT2kGWm+39J9hGTCBv8VI5Pm5lXZ/o3/mdR4f8rflAPhnQb8mPA==", "optional": true, "requires": { - "minipass": "2.3.5" + "minipass": "^2.2.1" } }, "mkdirp": { @@ -7708,9 +7708,9 @@ "integrity": "sha512-QBZu7aAFR0522EyaXZM0FZ9GLpq6lvQ3uq8gteiDUp7wKdy0lSd2hPlgFwVuW1CBkfEs9PfDQsQzZghLs/psdg==", "optional": true, "requires": { - "debug": "4.1.1", - "iconv-lite": "0.4.24", - "sax": "1.2.4" + "debug": "^4.1.0", + "iconv-lite": "^0.4.4", + "sax": "^1.2.4" } }, "node-pre-gyp": { @@ -7719,16 +7719,16 @@ "integrity": "sha512-4KghwV8vH5k+g2ylT+sLTjy5wmUOb9vPhnM8NHvRf9dHmnW/CndrFXy2aRPaPST6dugXSdHXfeaHQm77PIz/1A==", "optional": true, "requires": { - "detect-libc": "1.0.3", - "mkdirp": "0.5.1", - "needle": "2.3.0", - "nopt": "4.0.1", - "npm-packlist": "1.4.1", - "npmlog": "4.1.2", - "rc": "1.2.8", - "rimraf": "2.6.3", - "semver": "5.7.0", - "tar": "4.4.8" + "detect-libc": "^1.0.2", + "mkdirp": "^0.5.1", + "needle": "^2.2.1", + "nopt": "^4.0.1", + "npm-packlist": "^1.1.6", + "npmlog": "^4.0.2", + "rc": "^1.2.7", + "rimraf": "^2.6.1", + "semver": "^5.3.0", + "tar": "^4" } }, "nopt": { @@ -7737,8 +7737,8 @@ "integrity": "sha1-0NRoWv1UFRk8jHUFYC0NF81kR00=", "optional": true, "requires": { - "abbrev": "1.1.1", - "osenv": "0.1.5" + "abbrev": "1", + "osenv": "^0.1.4" } }, "npm-bundled": { @@ -7753,8 +7753,8 @@ "integrity": "sha512-+TcdO7HJJ8peiiYhvPxsEDhF3PJFGUGRcFsGve3vxvxdcpO2Z4Z7rkosRM0kWj6LfbK/P0gu3dzk5RU1ffvFcw==", "optional": true, "requires": { - "ignore-walk": "3.0.1", - "npm-bundled": "1.0.6" + "ignore-walk": "^3.0.1", + "npm-bundled": "^1.0.1" } }, "npmlog": { @@ -7763,10 +7763,10 @@ "integrity": "sha512-2uUqazuKlTaSI/dC8AzicUck7+IrEaOnN/e0jd3Xtt1KcGpwx30v50mL7oPyr/h9bL3E4aZccVwpwP+5W9Vjkg==", "optional": true, "requires": { - "are-we-there-yet": "1.1.5", - "console-control-strings": "1.1.0", - "gauge": "2.7.4", - "set-blocking": "2.0.0" + "are-we-there-yet": "~1.1.2", + "console-control-strings": "~1.1.0", + "gauge": "~2.7.3", + "set-blocking": "~2.0.0" } }, "number-is-nan": { @@ -7787,7 +7787,7 @@ "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=", "optional": true, "requires": { - "wrappy": "1.0.2" + "wrappy": "1" } }, "os-homedir": { @@ -7808,8 +7808,8 @@ "integrity": "sha512-0CWcCECdMVc2Rw3U5w9ZjqX6ga6ubk1xDVKxtBQPK7wis/0F2r9T6k4ydGYhecl7YUBxBVxhL5oisPsNxAPe2g==", "optional": true, "requires": { - "os-homedir": "1.0.2", - "os-tmpdir": "1.0.2" + "os-homedir": "^1.0.0", + "os-tmpdir": "^1.0.0" } }, "path-is-absolute": { @@ -7830,10 +7830,10 @@ "integrity": "sha512-y3bGgqKj3QBdxLbLkomlohkvsA8gdAiUQlSBJnBhfn+BPxg4bc62d8TcBW15wavDfgexCgccckhcZvywyQYPOw==", "optional": true, "requires": { - "deep-extend": "0.6.0", - "ini": "1.3.5", - "minimist": "1.2.0", - "strip-json-comments": "2.0.1" + "deep-extend": "^0.6.0", + "ini": "~1.3.0", + "minimist": "^1.2.0", + "strip-json-comments": "~2.0.1" }, "dependencies": { "minimist": { @@ -7850,13 +7850,13 @@ "integrity": "sha512-tQtKA9WIAhBF3+VLAseyMqZeBjW0AHJoxOtYqSUZNJxauErmLbVm2FW1y+J/YA9dUrAC39ITejlZWhVIwawkKw==", "optional": true, "requires": { - "core-util-is": "1.0.2", - "inherits": "2.0.3", - "isarray": "1.0.0", - "process-nextick-args": "2.0.0", - "safe-buffer": "5.1.2", - "string_decoder": "1.1.1", - "util-deprecate": "1.0.2" + "core-util-is": "~1.0.0", + "inherits": "~2.0.3", + "isarray": "~1.0.0", + "process-nextick-args": "~2.0.0", + "safe-buffer": "~5.1.1", + "string_decoder": "~1.1.1", + "util-deprecate": "~1.0.1" } }, "rimraf": { @@ -7865,7 +7865,7 @@ "integrity": "sha512-mwqeW5XsA2qAejG46gYdENaxXjx9onRNCfn7L0duuP4hCuTIi/QO7PDK07KJfp1d+izWPrzEJDcSqBa0OZQriA==", "optional": true, "requires": { - "glob": "7.1.3" + "glob": "^7.1.3" } }, "safe-buffer": { @@ -7909,9 +7909,9 @@ "integrity": "sha1-EYvfW4zcUaKn5w0hHgfisLmxB9M=", "optional": true, "requires": { - "code-point-at": "1.1.0", - "is-fullwidth-code-point": "1.0.0", - "strip-ansi": "3.0.1" + "code-point-at": "^1.0.0", + "is-fullwidth-code-point": "^1.0.0", + "strip-ansi": "^3.0.0" } }, "string_decoder": { @@ -7920,7 +7920,7 @@ "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==", "optional": true, "requires": { - "safe-buffer": "5.1.2" + "safe-buffer": "~5.1.0" } }, "strip-ansi": { @@ -7928,7 +7928,7 @@ "bundled": true, "optional": true, "requires": { - "ansi-regex": "2.1.1" + "ansi-regex": "^2.0.0" } }, "strip-json-comments": { @@ -7943,13 +7943,13 @@ "integrity": "sha512-LzHF64s5chPQQS0IYBn9IN5h3i98c12bo4NCO7e0sGM2llXQ3p2FGC5sdENN4cTW48O915Sh+x+EXx7XW96xYQ==", "optional": true, "requires": { - "chownr": "1.1.1", - "fs-minipass": "1.2.5", - "minipass": "2.3.5", - "minizlib": "1.2.1", - "mkdirp": "0.5.1", - "safe-buffer": "5.1.2", - "yallist": "3.0.3" + "chownr": "^1.1.1", + "fs-minipass": "^1.2.5", + "minipass": "^2.3.4", + "minizlib": "^1.1.1", + "mkdirp": "^0.5.0", + "safe-buffer": "^5.1.2", + "yallist": "^3.0.2" } }, "util-deprecate": { @@ -7964,7 +7964,7 @@ "integrity": "sha512-QGkOQc8XL6Bt5PwnsExKBPuMKBxnGxWWW3fU55Xt4feHozMUhdUMaBCk290qpm/wG5u/RSKzwdAC4i51YigihA==", "optional": true, "requires": { - "string-width": "1.0.2" + "string-width": "^1.0.2 || 2" } }, "wrappy": { @@ -7990,7 +7990,7 @@ "integrity": "sha512-eeB5GfMNeevm/GRYq20ShmsaGcmI81kIX2K9XQx5miC8KdHaC6Jm0qQ8ZNeGOi7wYB8OsdxKs+Y2oVuTFuVwKQ==", "dev": true, "requires": { - "is-property": "1.0.2" + "is-property": "^1.0.2" } }, "generate-object-property": { @@ -8020,7 +8020,7 @@ "integrity": "sha512-GMat4EJ5161kIy2HevLlr4luNjBgvmj413KaQA7jt4V8B4RDsfpHk7WQ9GVqfYyyx8OS/L66Kox+rJRNklLK7w==", "dev": true, "requires": { - "pump": "3.0.0" + "pump": "^3.0.0" }, "dependencies": { "pump": { @@ -8029,8 +8029,8 @@ "integrity": "sha512-LwZy+p3SFs1Pytd/jYct4wpv49HiYCqd9Rlc5ZVdk0V+8Yzv6jR5Blk3TRmPL1ft69TxP0IMZGJ+WPFU2BFhww==", "dev": true, "requires": { - "end-of-stream": "1.4.3", - "once": "1.4.0" + "end-of-stream": "^1.1.0", + "once": "^1.3.1" } } } @@ -8055,12 +8055,12 @@ "integrity": "sha512-vcfuiIxogLV4DlGBHIUOwI0IbrJ8HWPc4MU7HzviGeNho/UJDfi6B5p3sHeWIQ0KGIU0Jpxi5ZHxemQfLkkAwQ==", "dev": true, "requires": { - "fs.realpath": "1.0.0", - "inflight": "1.0.6", - "inherits": "2.0.4", - "minimatch": "3.0.4", - "once": "1.4.0", - "path-is-absolute": "1.0.1" + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.0.4", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" } }, "glob-parent": { @@ -8129,9 +8129,9 @@ "integrity": "sha512-sKzpEkf11GpOFuw0Zzjzmt4B4UZwjOcG757PPvrfhxcLFbq0wpsgpOqxpxtxFiCG4DtG93M6XRVbF2oGdev7bg==", "dev": true, "requires": { - "global-prefix": "1.0.2", - "is-windows": "1.0.2", - "resolve-dir": "1.0.1" + "global-prefix": "^1.0.1", + "is-windows": "^1.0.1", + "resolve-dir": "^1.0.0" } }, "global-prefix": { @@ -8140,11 +8140,11 @@ "integrity": "sha1-2/dDxsFJklk8ZVVoy2btMsASLr4=", "dev": true, "requires": { - "expand-tilde": "2.0.2", - "homedir-polyfill": "1.0.3", - "ini": "1.3.5", - "is-windows": "1.0.2", - "which": "1.3.1" + "expand-tilde": "^2.0.2", + "homedir-polyfill": "^1.0.1", + "ini": "^1.3.4", + "is-windows": "^1.0.1", + "which": "^1.2.14" } }, "globals": { @@ -8159,14 +8159,14 @@ "integrity": "sha512-sSs4inE1FB2YQiymcmTv6NWENryABjUNPeWhOvmn4SjtKybglsyPZxFB3U1/+L1bYi0rNZDqCLlHyLYDl1Pq5A==", "dev": true, "requires": { - "@types/glob": "7.1.1", - "array-union": "2.1.0", - "dir-glob": "3.0.1", - "fast-glob": "3.0.4", - "glob": "7.1.3", - "ignore": "5.1.4", - "merge2": "1.3.0", - "slash": "3.0.0" + "@types/glob": "^7.1.1", + "array-union": "^2.1.0", + "dir-glob": "^3.0.1", + "fast-glob": "^3.0.3", + "glob": "^7.1.3", + "ignore": "^5.1.1", + "merge2": "^1.2.3", + "slash": "^3.0.0" }, "dependencies": { "array-union": { @@ -8181,7 +8181,7 @@ "integrity": "sha512-WkrWp9GR4KXfKGYzOLmTuGVi1UWFfws377n9cc55/tb6DuqyF6pcQ5AbiHEshaDpY9v6oaSr2XCDidGmMwdzIA==", "dev": true, "requires": { - "path-type": "4.0.0" + "path-type": "^4.0.0" } }, "ignore": { @@ -8247,17 +8247,17 @@ "integrity": "sha512-R7eWptXuGYxwijs0eV+v3o6+XH1IqVK8dJOEecQfTmkncw9AV4dcw/Dhxi8MdlqPthxxpZyizMzyg8RTmEsG+Q==", "dev": true, "requires": { - "@sindresorhus/is": "0.14.0", - "@szmarczak/http-timer": "1.1.2", - "cacheable-request": "6.1.0", - "decompress-response": "3.3.0", - "duplexer3": "0.1.4", - "get-stream": "4.1.0", - "lowercase-keys": "1.0.1", - "mimic-response": "1.0.1", - "p-cancelable": "1.1.0", - "to-readable-stream": "1.0.0", - "url-parse-lax": "3.0.0" + "@sindresorhus/is": "^0.14.0", + "@szmarczak/http-timer": "^1.1.2", + "cacheable-request": "^6.0.0", + "decompress-response": "^3.3.0", + "duplexer3": "^0.1.4", + "get-stream": "^4.1.0", + "lowercase-keys": "^1.0.1", + "mimic-response": "^1.0.1", + "p-cancelable": "^1.0.0", + "to-readable-stream": "^1.0.0", + "url-parse-lax": "^3.0.0" } }, "graceful-fs": { @@ -8292,7 +8292,7 @@ "integrity": "sha512-SFKX67auSNoVR38N3L+nvsPjOE0bybKTYbkf5tRvushrAPQ9V75huw0ZxBkKVeRU9kqH3d6HA4xTckbwZ4ixmA==", "dev": true, "requires": { - "ansi-wrap": "0.1.0" + "ansi-wrap": "^0.1.0" } }, "gulp-cli": { @@ -8329,11 +8329,11 @@ "integrity": "sha512-oR3t8kn+ccHkSyRcBV5kBLPXrhqTh5d6wBAR7r7wqjNQNBhYvOwPedCwlAaGcNl1qSeXNDn6qOk1Qyxvx9Wrow==", "dev": true, "requires": { - "get-stream": "5.1.0", - "plugin-error": "1.0.1", - "through2": "3.0.1", - "vinyl": "2.2.0", - "yazl": "2.5.1" + "get-stream": "^5.1.0", + "plugin-error": "^1.0.1", + "through2": "^3.0.1", + "vinyl": "^2.1.0", + "yazl": "^2.5.1" }, "dependencies": { "get-stream": { @@ -8342,7 +8342,7 @@ "integrity": "sha512-EXr1FOzrzTfGeL0gQdeFEvOMm2mzMOglyiOXSTpPC+iAjAKftbr3jpCMWynogwYnM+eSj9sHGc6wjIcDvYiygw==", "dev": true, "requires": { - "pump": "3.0.0" + "pump": "^3.0.0" } }, "pump": { @@ -8351,8 +8351,8 @@ "integrity": "sha512-LwZy+p3SFs1Pytd/jYct4wpv49HiYCqd9Rlc5ZVdk0V+8Yzv6jR5Blk3TRmPL1ft69TxP0IMZGJ+WPFU2BFhww==", "dev": true, "requires": { - "end-of-stream": "1.4.3", - "once": "1.4.0" + "end-of-stream": "^1.1.0", + "once": "^1.3.1" } }, "through2": { @@ -8361,7 +8361,7 @@ "integrity": "sha512-M96dvTalPT3YbYLaKaCuwu+j06D/8Jfib0o/PxbVt6Amhv3dUAtW6rTV1jPgJSBG83I/e04Y6xkVdVhSRhi0ww==", "dev": true, "requires": { - "readable-stream": "2.3.6" + "readable-stream": "2 || 3" } } } @@ -8392,10 +8392,10 @@ "integrity": "sha512-7XlnO8yBXOdi7AzowjZssQr47Ctidqm7GbgARapOaqSN9HQhlClnOkR9HieGauIT3A8MBC6u9wPCXs97PCYpWg==", "dev": true, "requires": { - "neo-async": "2.6.1", - "optimist": "0.6.1", - "source-map": "0.6.1", - "uglify-js": "3.6.0" + "neo-async": "^2.6.0", + "optimist": "^0.6.1", + "source-map": "^0.6.1", + "uglify-js": "^3.1.4" }, "dependencies": { "source-map": { @@ -8473,7 +8473,7 @@ "integrity": "sha1-mqqe7b/7G6OZCnsAEPtnjuAIEgc=", "dev": true, "requires": { - "is-glob": "3.1.0" + "is-glob": "^3.0.0" }, "dependencies": { "is-glob": { @@ -8482,7 +8482,7 @@ "integrity": "sha1-e6WuJCF4BKxwcHuWkiVnSGzD6Eo=", "dev": true, "requires": { - "is-extglob": "2.1.1" + "is-extglob": "^2.1.0" } } } @@ -8564,7 +8564,7 @@ "integrity": "sha512-eSmmWE5bZTK2Nou4g0AI3zZ9rswp7GRKoKXS1BLUkvPviOqs4YTN1djQIqrXy9k5gEtdLPy86JjRwsNM9tnDcA==", "dev": true, "requires": { - "parse-passwd": "1.0.0" + "parse-passwd": "^1.0.0" } }, "hosted-git-info": { @@ -8609,10 +8609,10 @@ "integrity": "sha512-uUQBt3H/cSIVfch6i1EuPNy/YsRSOUBXTVfZ+yR7Zjez3qjBz6i9+i4zjNaoqcoFVI4lQJ5plg63TvGfRSDCRg==", "dev": true, "requires": { - "depd": "1.1.2", + "depd": "~1.1.2", "inherits": "2.0.3", "setprototypeof": "1.1.1", - "statuses": "1.5.0", + "statuses": ">= 1.5.0 < 2", "toidentifier": "1.0.0" }, "dependencies": { @@ -8636,9 +8636,9 @@ "integrity": "sha512-84I2iJM/n1d4Hdgc6y2+qY5mDaz2PUVjlg9znE9byl+q0uC3DeByqBGReQu5tpLK0TAqTIXScRUV+dg7+bUPpQ==", "dev": true, "requires": { - "eventemitter3": "4.0.0", - "follow-redirects": "1.9.0", - "requires-port": "1.0.0" + "eventemitter3": "^4.0.0", + "follow-redirects": "^1.0.0", + "requires-port": "^1.0.0" } }, "http-proxy-agent": { @@ -8668,10 +8668,10 @@ "integrity": "sha512-yHYTgWMQO8VvwNS22eLLloAkvungsKdKTLO8AJlftYIKNfJr3GK3zK0ZCfzDDGUBttdGc8xFy1mCitvNKQtC3Q==", "dev": true, "requires": { - "http-proxy": "1.18.0", - "is-glob": "4.0.1", - "lodash": "4.17.15", - "micromatch": "3.1.10" + "http-proxy": "^1.17.0", + "is-glob": "^4.0.0", + "lodash": "^4.17.11", + "micromatch": "^3.1.10" } }, "http-signature": { @@ -8697,8 +8697,8 @@ "integrity": "sha512-c8Ndjc9Bkpfx/vCJueCPy0jlP4ccCCSNDp8xwCZzPjKJUm+B+u9WX2x98Qx4n1PiMNTWo3D7KK5ifNV/yJyRzg==", "dev": true, "requires": { - "agent-base": "4.3.0", - "debug": "3.2.6" + "agent-base": "^4.3.0", + "debug": "^3.1.0" }, "dependencies": { "debug": { @@ -8707,7 +8707,7 @@ "integrity": "sha512-mel+jf7nrtEl5Pn1Qx46zARXKDpBbvzezse7p7LqINmdoIk8PYP5SySaxEmYv6TZ0JyEKA1hsCId6DIhgITtWQ==", "dev": true, "requires": { - "ms": "2.1.2" + "ms": "^2.1.1" } }, "ms": { @@ -8732,7 +8732,7 @@ "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", "requires": { - "safer-buffer": "2.1.2" + "safer-buffer": ">= 2.1.2 < 3" } }, "ieee754": { @@ -8759,7 +8759,7 @@ "integrity": "sha512-EXyErtpHbn75ZTsOADsfx6J/FPo6/5cjev46PXrcTpd8z3BoRkXgYu9/JVqrI7tusjmwCZutGeRJeU0Wo1e4Cw==", "dev": true, "requires": { - "minimatch": "3.0.4" + "minimatch": "^3.0.4" } }, "image-size": { @@ -8795,8 +8795,8 @@ "integrity": "sha1-2BNVwVYS04bGH53dOSLUMEgipUY=", "dev": true, "requires": { - "caller-path": "2.0.0", - "resolve-from": "3.0.0" + "caller-path": "^2.0.0", + "resolve-from": "^3.0.0" } }, "import-from": { @@ -8875,19 +8875,19 @@ "integrity": "sha512-uxNHBeQhRXIoHWTSNYUFhQVrHYFThIt6IVo2fFmSe8aBwdR3/w6b58hJpiL/fMukFkvGzjg+hSxFtwvVmKZmXw==", "dev": true, "requires": { - "ansi-escapes": "4.2.1", - "chalk": "2.4.2", - "cli-cursor": "3.1.0", - "cli-width": "2.2.0", - "external-editor": "3.1.0", - "figures": "3.0.0", - "lodash": "4.17.15", + "ansi-escapes": "^4.2.1", + "chalk": "^2.4.2", + "cli-cursor": "^3.1.0", + "cli-width": "^2.0.0", + "external-editor": "^3.0.3", + "figures": "^3.0.0", + "lodash": "^4.17.15", "mute-stream": "0.0.8", - "run-async": "2.3.0", - "rxjs": "6.5.3", - "string-width": "4.1.0", - "strip-ansi": "5.2.0", - "through": "2.3.8" + "run-async": "^2.2.0", + "rxjs": "^6.4.0", + "string-width": "^4.1.0", + "strip-ansi": "^5.1.0", + "through": "^2.3.6" }, "dependencies": { "ansi-regex": { @@ -8908,9 +8908,9 @@ "integrity": "sha512-NrX+1dVVh+6Y9dnQ19pR0pP4FiEIlUvdTGn8pw6CKTNq5sgib2nIhmUNT5TAmhWmvKr3WcxBcP3E8nWezuipuQ==", "dev": true, "requires": { - "emoji-regex": "8.0.0", - "is-fullwidth-code-point": "3.0.0", - "strip-ansi": "5.2.0" + "emoji-regex": "^8.0.0", + "is-fullwidth-code-point": "^3.0.0", + "strip-ansi": "^5.2.0" } }, "strip-ansi": { @@ -8919,7 +8919,7 @@ "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", "dev": true, "requires": { - "ansi-regex": "4.1.0" + "ansi-regex": "^4.1.0" } } } @@ -8930,8 +8930,8 @@ "integrity": "sha512-S1zBo1D6zcsyuC6PMmY5+55YMILQ9av8lotMx447Bq6SAgo/sDK6y6uUKmuYhW7eacnIhFfsPmCNYdDzsnnDCg==", "dev": true, "requires": { - "default-gateway": "4.2.0", - "ipaddr.js": "1.9.0" + "default-gateway": "^4.2.0", + "ipaddr.js": "^1.9.0" } }, "interpret": { @@ -9048,7 +9048,7 @@ "integrity": "sha512-YfJT7rkpQB0updsdHLGWrvhBJfcfzNNawYDNIyQXJz0IViGf75O8EBPKSdvw2rF+LGCsX4FZ8tcr3b19LcZq4w==", "dev": true, "requires": { - "ci-info": "2.0.0" + "ci-info": "^2.0.0" } }, "is-data-descriptor": { @@ -9131,7 +9131,7 @@ "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.1.tgz", "integrity": "sha512-5G0tKtBTFImOqDnLB2hG6Bp2qcKEFduo4tZu9MT/H6NQv/ghhy30o55ufafxJ/LdH79LLs2Kfrn85TLKyA7BUg==", "requires": { - "is-extglob": "2.1.1" + "is-extglob": "^2.1.1" } }, "is-installed-globally": { @@ -9162,11 +9162,11 @@ "integrity": "sha512-XTHBZSIIxNsIsZXg7XB5l8z/OBFosl1Wao4tXLpeC7eKU4Vm/kdop2azkPqULwnfGQjmeDIyey9g7afMMtdWAA==", "dev": true, "requires": { - "generate-function": "2.3.1", - "generate-object-property": "1.2.0", - "is-my-ip-valid": "1.0.0", - "jsonpointer": "4.0.1", - "xtend": "4.0.2" + "generate-function": "^2.0.0", + "generate-object-property": "^1.1.0", + "is-my-ip-valid": "^1.0.0", + "jsonpointer": "^4.0.0", + "xtend": "^4.0.0" } }, "is-negated-glob": { @@ -9306,7 +9306,7 @@ "integrity": "sha512-HS8bZ9ox60yCJLH9snBpIwv9pYUAkcuLhSA1oero1UB5y9aiQpRA8y2ex945AOtCZL1lJDeIk3G5LthswI46Lw==", "dev": true, "requires": { - "has-symbols": "1.0.0" + "has-symbols": "^1.0.0" } }, "is-typedarray": { @@ -9466,17 +9466,17 @@ "integrity": "sha512-kH5YRdqdbs5hiH4/Rr1Q0cSAGgjh3jTtg8vu9NLebBAoK3adVO4jk81J+TYOkTr2+Q4NLeb1ACvmEt65iG/Vbw==", "dev": true, "requires": { - "async": "2.6.3", - "fileset": "2.0.3", - "istanbul-lib-coverage": "1.2.1", - "istanbul-lib-hook": "1.2.2", - "istanbul-lib-instrument": "1.10.2", - "istanbul-lib-report": "1.1.5", - "istanbul-lib-source-maps": "1.2.6", - "istanbul-reports": "1.1.4", - "js-yaml": "3.13.1", - "mkdirp": "0.5.1", - "once": "1.4.0" + "async": "^2.1.4", + "fileset": "^2.0.2", + "istanbul-lib-coverage": "^1.1.2", + "istanbul-lib-hook": "^1.1.0", + "istanbul-lib-instrument": "^1.9.2", + "istanbul-lib-report": "^1.1.3", + "istanbul-lib-source-maps": "^1.2.3", + "istanbul-reports": "^1.1.4", + "js-yaml": "^3.7.0", + "mkdirp": "^0.5.1", + "once": "^1.4.0" } }, "istanbul-instrumenter-loader": { @@ -9538,7 +9538,7 @@ "integrity": "sha512-/Jmq7Y1VeHnZEQ3TL10VHyb564mn6VrQXHchON9Jf/AEcmQ3ZIiyD1BVzNOKTZf/G3gE+kiGK6SmpF9y3qGPLw==", "dev": true, "requires": { - "append-transform": "0.4.0" + "append-transform": "^0.4.0" } }, "istanbul-lib-instrument": { @@ -9547,13 +9547,13 @@ "integrity": "sha512-aWHxfxDqvh/ZlxR8BBaEPVSWDPUkGD63VjGQn3jcw8jCp7sHEMKcrj4xfJn/ABzdMEHiQNyvDQhqm5o8+SQg7A==", "dev": true, "requires": { - "babel-generator": "6.26.1", - "babel-template": "6.26.0", - "babel-traverse": "6.26.0", - "babel-types": "6.26.0", - "babylon": "6.18.0", - "istanbul-lib-coverage": "1.2.1", - "semver": "5.6.0" + "babel-generator": "^6.18.0", + "babel-template": "^6.16.0", + "babel-traverse": "^6.18.0", + "babel-types": "^6.18.0", + "babylon": "^6.18.0", + "istanbul-lib-coverage": "^1.2.1", + "semver": "^5.3.0" } }, "istanbul-lib-report": { @@ -9562,10 +9562,10 @@ "integrity": "sha512-UsYfRMoi6QO/doUshYNqcKJqVmFe9w51GZz8BS3WB0lYxAllQYklka2wP9+dGZeHYaWIdcXUx8JGdbqaoXRXzw==", "dev": true, "requires": { - "istanbul-lib-coverage": "1.2.1", - "mkdirp": "0.5.1", - "path-parse": "1.0.6", - "supports-color": "3.2.3" + "istanbul-lib-coverage": "^1.2.1", + "mkdirp": "^0.5.1", + "path-parse": "^1.0.5", + "supports-color": "^3.1.2" }, "dependencies": { "has-flag": { @@ -9580,7 +9580,7 @@ "integrity": "sha1-ZawFBLOVQXHYpklGsq48u4pfVPY=", "dev": true, "requires": { - "has-flag": "1.0.0" + "has-flag": "^1.0.0" } } } @@ -9591,11 +9591,11 @@ "integrity": "sha512-TtbsY5GIHgbMsMiRw35YBHGpZ1DVFEO19vxxeiDMYaeOFOCzfnYVxvl6pOUIZR4dtPhAGpSMup8OyF8ubsaqEg==", "dev": true, "requires": { - "debug": "3.2.6", - "istanbul-lib-coverage": "1.2.1", - "mkdirp": "0.5.1", - "rimraf": "2.7.1", - "source-map": "0.5.6" + "debug": "^3.1.0", + "istanbul-lib-coverage": "^1.2.1", + "mkdirp": "^0.5.1", + "rimraf": "^2.6.1", + "source-map": "^0.5.3" }, "dependencies": { "debug": { @@ -9604,7 +9604,7 @@ "integrity": "sha512-mel+jf7nrtEl5Pn1Qx46zARXKDpBbvzezse7p7LqINmdoIk8PYP5SySaxEmYv6TZ0JyEKA1hsCId6DIhgITtWQ==", "dev": true, "requires": { - "ms": "2.1.2" + "ms": "^2.1.1" } }, "ms": { @@ -9621,7 +9621,7 @@ "integrity": "sha512-DfSTVOTkuO+kRmbO8Gk650Wqm1WRGr6lrdi2EwDK1vxpS71vdlLd613EpzOKdIFioB5f/scJTjeWBnvd1FWejg==", "dev": true, "requires": { - "handlebars": "4.3.0" + "handlebars": "^4.0.3" } }, "jasmine": { @@ -9654,7 +9654,7 @@ "resolved": "https://registry.npmjs.org/jasmine-marbles/-/jasmine-marbles-0.6.0.tgz", "integrity": "sha512-1uzgjEesEeCb+r+v46qn5x326TiGqk5SUZa+A3O+XnMCjG/pGcUOhL9Xsg5L7gLC6RFHyWGTkB5fei4rcvIOiQ==", "requires": { - "lodash": "4.17.15" + "lodash": "^4.5.0" } }, "jasmine-spec-reporter": { @@ -9678,8 +9678,8 @@ "integrity": "sha512-51PE4haMSXcHohnSMdM42anbvZANYTqMrr52tVKPqqsPJMzoP6FYYDVqahX/HrAoKEKz3uUPzSvKs9A3qR4iVw==", "dev": true, "requires": { - "merge-stream": "2.0.0", - "supports-color": "6.1.0" + "merge-stream": "^2.0.0", + "supports-color": "^6.1.0" } }, "js-levenshtein": { @@ -9777,7 +9777,7 @@ "integrity": "sha1-h3Gq4HmbZAdrdmQPygWPnBDjPss=", "dev": true, "requires": { - "graceful-fs": "4.2.2" + "graceful-fs": "^4.1.6" } }, "jsonify": { @@ -9869,8 +9869,8 @@ "integrity": "sha512-mM8522psRCqzV+6LhomX5wgp25YVibjh8Wj23I5RPkPppSVSjyKD2A2mBJmWGa+KN7f2D6LNh9jkBCeyLktzjg==", "dev": true, "requires": { - "normalize-path": "3.0.0", - "picomatch": "2.0.7" + "normalize-path": "^3.0.0", + "picomatch": "^2.0.4" } }, "binary-extensions": { @@ -9894,14 +9894,14 @@ "integrity": "sha512-/j5PPkb5Feyps9e+jo07jUZGvkB5Aj953NrI4s8xSVScrAo/RHeILrtdb4uzR7N6aaFFxxJ+gt8mA8HfNpw76w==", "dev": true, "requires": { - "anymatch": "3.1.1", - "braces": "3.0.2", - "fsevents": "2.1.0", - "glob-parent": "5.1.0", - "is-binary-path": "2.1.0", - "is-glob": "4.0.1", - "normalize-path": "3.0.0", - "readdirp": "3.1.3" + "anymatch": "~3.1.1", + "braces": "~3.0.2", + "fsevents": "~2.1.0", + "glob-parent": "~5.1.0", + "is-binary-path": "~2.1.0", + "is-glob": "~4.0.1", + "normalize-path": "~3.0.0", + "readdirp": "~3.1.3" } }, "date-format": { @@ -9952,7 +9952,7 @@ "integrity": "sha512-qjtRgnIVmOfnKUE3NJAQEdk+lKrxfw8t5ke7SXtfMTHcjsBfOfWXCQfdb30zfDoZQ2IRSIiidmjtbHZPZ++Ihw==", "dev": true, "requires": { - "is-glob": "4.0.1" + "is-glob": "^4.0.1" } }, "is-binary-path": { @@ -10007,7 +10007,7 @@ "integrity": "sha512-ZOsfTGkjO2kqeR5Mzr5RYDbTGYneSkdNKX2fOX2P5jF7vMrd/GNnIAUtDldeHHumHUCQ3V05YfWUdxMPAsRu9Q==", "dev": true, "requires": { - "picomatch": "2.0.7" + "picomatch": "^2.0.4" } }, "source-map": { @@ -10057,7 +10057,7 @@ "integrity": "sha512-3dPs/n7vgz1rxxtynpzZTvb9y/GIaW8xjAwcIGttLbycqoFtI7yo1NGnQi6oFTherRE+GIhCAHZC4vEqWGhNvg==", "dev": true, "requires": { - "which": "1.3.1" + "which": "^1.2.1" } }, "karma-cli": { @@ -10066,7 +10066,7 @@ "integrity": "sha512-1Kb28UILg1ZsfqQmeELbPzuEb5C6GZJfVIk0qOr8LNYQuYWmAaqP16WpbpKEjhejDrDYyYOwwJXSZO6u7q5Pvw==", "dev": true, "requires": { - "resolve": "1.12.0" + "resolve": "^1.3.3" } }, "karma-coverage-istanbul-reporter": { @@ -10075,8 +10075,8 @@ "integrity": "sha512-UH0mXPJFJyK5uiK7EkwGtQ8f30lCBAfqRResnZ4pzLJ04SOp4SPlYkmwbbZ6iVJ6sQFVzlDUXlntBEsLRdgZpg==", "dev": true, "requires": { - "istanbul-api": "2.1.6", - "minimatch": "3.0.4" + "istanbul-api": "^2.1.6", + "minimatch": "^3.0.4" }, "dependencies": { "append-transform": { @@ -10085,7 +10085,7 @@ "integrity": "sha512-P009oYkeHyU742iSZJzZZywj4QRJdnTWffaKuJQLablCZ1uz6/cW4yaRgcDaoQ+uwOxxnt0gRUcwfsNP2ri0gw==", "dev": true, "requires": { - "default-require-extensions": "2.0.0" + "default-require-extensions": "^2.0.0" } }, "debug": { @@ -10094,7 +10094,7 @@ "integrity": "sha512-pYAIzeRo8J6KPEaJ0VWOh5Pzkbw/RetuzehGM7QRRX5he4fPHx2rdKMB256ehJCkX+XRQm16eZLqLNS8RSZXZw==", "dev": true, "requires": { - "ms": "2.1.2" + "ms": "^2.1.1" } }, "default-require-extensions": { @@ -10103,7 +10103,7 @@ "integrity": "sha1-9fj7sYp9bVCyH2QfZJ67Uiz+JPc=", "dev": true, "requires": { - "strip-bom": "3.0.0" + "strip-bom": "^3.0.0" } }, "istanbul-api": { @@ -10112,19 +10112,19 @@ "integrity": "sha512-x0Eicp6KsShG1k1rMgBAi/1GgY7kFGEBwQpw3PXGEmu+rBcBNhqU8g2DgY9mlepAsLPzrzrbqSgCGANnki4POA==", "dev": true, "requires": { - "async": "2.6.3", - "compare-versions": "3.5.1", - "fileset": "2.0.3", - "istanbul-lib-coverage": "2.0.5", - "istanbul-lib-hook": "2.0.7", - "istanbul-lib-instrument": "3.3.0", - "istanbul-lib-report": "2.0.8", - "istanbul-lib-source-maps": "3.0.6", - "istanbul-reports": "2.2.6", - "js-yaml": "3.13.1", - "make-dir": "2.1.0", - "minimatch": "3.0.4", - "once": "1.4.0" + "async": "^2.6.2", + "compare-versions": "^3.4.0", + "fileset": "^2.0.3", + "istanbul-lib-coverage": "^2.0.5", + "istanbul-lib-hook": "^2.0.7", + "istanbul-lib-instrument": "^3.3.0", + "istanbul-lib-report": "^2.0.8", + "istanbul-lib-source-maps": "^3.0.6", + "istanbul-reports": "^2.2.4", + "js-yaml": "^3.13.1", + "make-dir": "^2.1.0", + "minimatch": "^3.0.4", + "once": "^1.4.0" } }, "istanbul-lib-coverage": { @@ -10139,7 +10139,7 @@ "integrity": "sha512-vrRztU9VRRFDyC+aklfLoeXyNdTfga2EI3udDGn4cZ6fpSXpHLV9X6CHvfoMCPtggg8zvDDmC4b9xfu0z6/llA==", "dev": true, "requires": { - "append-transform": "1.0.0" + "append-transform": "^1.0.0" } }, "istanbul-lib-instrument": { @@ -10148,13 +10148,13 @@ "integrity": "sha512-5nnIN4vo5xQZHdXno/YDXJ0G+I3dAm4XgzfSVTPLQpj/zAV2dV6Juy0yaf10/zrJOJeHoN3fraFe+XRq2bFVZA==", "dev": true, "requires": { - "@babel/generator": "7.6.2", - "@babel/parser": "7.6.2", - "@babel/template": "7.6.0", - "@babel/traverse": "7.6.2", - "@babel/types": "7.6.1", - "istanbul-lib-coverage": "2.0.5", - "semver": "6.3.0" + "@babel/generator": "^7.4.0", + "@babel/parser": "^7.4.3", + "@babel/template": "^7.4.0", + "@babel/traverse": "^7.4.3", + "@babel/types": "^7.4.0", + "istanbul-lib-coverage": "^2.0.5", + "semver": "^6.0.0" } }, "istanbul-lib-report": { @@ -10163,9 +10163,9 @@ "integrity": "sha512-fHBeG573EIihhAblwgxrSenp0Dby6tJMFR/HvlerBsrCTD5bkUuoNtn3gVh29ZCS824cGGBPn7Sg7cNk+2xUsQ==", "dev": true, "requires": { - "istanbul-lib-coverage": "2.0.5", - "make-dir": "2.1.0", - "supports-color": "6.1.0" + "istanbul-lib-coverage": "^2.0.5", + "make-dir": "^2.1.0", + "supports-color": "^6.1.0" } }, "istanbul-lib-source-maps": { @@ -10174,11 +10174,11 @@ "integrity": "sha512-R47KzMtDJH6X4/YW9XTx+jrLnZnscW4VpNN+1PViSYTejLVPWv7oov+Duf8YQSPyVRUvueQqz1TcsC6mooZTXw==", "dev": true, "requires": { - "debug": "4.1.1", - "istanbul-lib-coverage": "2.0.5", - "make-dir": "2.1.0", - "rimraf": "2.7.1", - "source-map": "0.6.1" + "debug": "^4.1.1", + "istanbul-lib-coverage": "^2.0.5", + "make-dir": "^2.1.0", + "rimraf": "^2.6.3", + "source-map": "^0.6.1" } }, "istanbul-reports": { @@ -10187,7 +10187,7 @@ "integrity": "sha512-SKi4rnMyLBKe0Jy2uUdx28h8oG7ph2PPuQPvIAh31d+Ci+lSiEu4C+h3oBPuJ9+mPKhOyW0M8gY4U5NM1WLeXA==", "dev": true, "requires": { - "handlebars": "4.3.0" + "handlebars": "^4.1.2" } }, "ms": { @@ -10237,7 +10237,7 @@ "integrity": "sha512-RsBECncGO17KAoJCYXjv+ckIz+Ii9NCi+9enk+rq6XC81ezYkb4/RHE6CTXdA7IOJqoF3wcaLfVG0CPmE5ca6A==", "dev": true, "requires": { - "source-map-support": "0.5.13" + "source-map-support": "^0.5.5" } }, "karma-spec-reporter": { @@ -10246,7 +10246,7 @@ "integrity": "sha1-LpxyB+pyZ3EmAln4K+y1QyCeRAo=", "dev": true, "requires": { - "colors": "1.1.2" + "colors": "^1.1.2" } }, "keyv": { @@ -10265,9 +10265,9 @@ "dev": true }, "kind-of": { - "version": "6.0.2", - "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-6.0.2.tgz", - "integrity": "sha512-s5kLOcnH0XqDO+FvuaLX8DDjZ18CGFk7VygH40QoKPUQhW4e2rvM0rwUq0t8IQDOwYSeLK01U90OjzBTme2QqA==" + "version": "6.0.3", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-6.0.3.tgz", + "integrity": "sha512-dcS1ul+9tmeD95T+x28/ehLgd9mENa3LsvDTtzm3vyBEO7RPptvAD+t44WVXaUjTBRcrpFeFlC8WCruUR456hw==" }, "klaw-sync": { "version": "2.1.0", @@ -10300,7 +10300,7 @@ "integrity": "sha512-weT+r0kTkRQdCdYCNtkMwWXQTMEswKrFBkm4ckQOMVhhqhIMI1UT2hMj+1iigIhgSZm5gTmrRXBNoGUgaTY1xA==", "dev": true, "requires": { - "package-json": "6.5.0" + "package-json": "^6.3.0" } }, "lazystream": { @@ -10362,9 +10362,9 @@ "integrity": "sha512-bquCU89mO/yWLaUq0Clk7qCsKhsF/TZpJUzETRvJa9KSVEL9SO3ovCvdEHISBhrC81OwC8QSVX7E0bzElZj9cg==", "dev": true, "requires": { - "clone": "2.1.2", - "loader-utils": "1.2.3", - "pify": "4.0.1" + "clone": "^2.1.1", + "loader-utils": "^1.1.0", + "pify": "^4.0.1" }, "dependencies": { "pify": { @@ -10418,8 +10418,8 @@ "integrity": "sha512-7poZHRla+ae0eEButlwMrPpkXyhNVBf2EHePYWT0jyLnI6311/OXJkTI2sOIRungRpQgU2oDMpro5bSFPT5F0A==", "dev": true, "requires": { - "@types/webpack-sources": "0.1.5", - "webpack-sources": "1.4.3" + "@types/webpack-sources": "^0.1.5", + "webpack-sources": "^1.2.0" } }, "lie": { @@ -10495,8 +10495,8 @@ "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-3.0.0.tgz", "integrity": "sha512-7AO748wWnIhNqAuaty2ZWHkQHRSNfPVIsPIfwEOWO22AmaoVrWavlOcMR5nzTLNYvp36X220/maaRsrec1G65A==", "requires": { - "p-locate": "3.0.0", - "path-exists": "3.0.0" + "p-locate": "^3.0.0", + "path-exists": "^3.0.0" } }, "lodash": { @@ -10573,7 +10573,7 @@ "resolved": "https://registry.npmjs.org/loose-envify/-/loose-envify-1.4.0.tgz", "integrity": "sha512-lyuxPGr/Wfhrlem2CL/UcnUc1zcqKAImBDzukY7Y5F/yQiNdko6+fRLevlw1HgMySw7f611UIY408EtxRSoK3Q==", "requires": { - "js-tokens": "4.0.0" + "js-tokens": "^3.0.0 || ^4.0.0" } }, "lowercase-keys": { @@ -10588,7 +10588,7 @@ "integrity": "sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==", "dev": true, "requires": { - "yallist": "3.1.1" + "yallist": "^3.0.2" } }, "magic-string": { @@ -10596,7 +10596,7 @@ "resolved": "https://registry.npmjs.org/magic-string/-/magic-string-0.25.3.tgz", "integrity": "sha512-6QK0OpF/phMz0Q2AxILkX2mFhi7m+WMwTRg0LQKq/WBB0cDP4rYH3Wp4/d3OTXlrPLVJT/RFqj8tFeAR4nk8AA==", "requires": { - "sourcemap-codec": "1.4.6" + "sourcemap-codec": "^1.4.4" } }, "make-dir": { @@ -10604,8 +10604,8 @@ "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-2.1.0.tgz", "integrity": "sha512-LS9X+dc8KLxXCb8dni79fLIIUA5VyZoyjSMCwTluaXA0o27cCK0bhXkpgw+sTXVpPy/lSO57ilRixqk0vDmtRA==", "requires": { - "pify": "4.0.1", - "semver": "5.6.0" + "pify": "^4.0.1", + "semver": "^5.6.0" }, "dependencies": { "pify": { @@ -10627,17 +10627,17 @@ "integrity": "sha512-nFr/vpL1Jc60etMVKeaLOqfGjMMb3tAHFVJWxHOFCFS04Zmd7kGlMxo0l1tzfhoQje0/UPnd0X8OeGUiXXnfPA==", "dev": true, "requires": { - "agentkeepalive": "3.5.2", - "cacache": "12.0.2", - "http-cache-semantics": "3.8.1", - "http-proxy-agent": "2.1.0", - "https-proxy-agent": "2.2.2", - "lru-cache": "5.1.1", - "mississippi": "3.0.0", - "node-fetch-npm": "2.0.2", - "promise-retry": "1.1.1", - "socks-proxy-agent": "4.0.2", - "ssri": "6.0.1" + "agentkeepalive": "^3.4.1", + "cacache": "^12.0.0", + "http-cache-semantics": "^3.8.1", + "http-proxy-agent": "^2.1.0", + "https-proxy-agent": "^2.2.1", + "lru-cache": "^5.1.1", + "mississippi": "^3.0.0", + "node-fetch-npm": "^2.0.2", + "promise-retry": "^1.1.1", + "socks-proxy-agent": "^4.0.0", + "ssri": "^6.0.0" } }, "make-iterator": { @@ -10777,12 +10777,12 @@ "integrity": "sha512-7ivM1jFZVTOOS77QsR+TtYHH0ecdLclMkqbf5qiJdX2RorqfhsL65QHySPZgDE0ZjHoh+mQUNHTanNXIlzXd0Q==", "dev": true, "requires": { - "arr-union": "3.1.0", - "async-array-reduce": "0.2.1", - "glob": "7.1.3", - "has-glob": "1.0.0", - "is-valid-glob": "1.0.0", - "resolve-dir": "1.0.1" + "arr-union": "^3.1.0", + "async-array-reduce": "^0.2.1", + "glob": "^7.1.2", + "has-glob": "^1.0.0", + "is-valid-glob": "^1.0.0", + "resolve-dir": "^1.0.0" } }, "md5.js": { @@ -10808,9 +10808,9 @@ "integrity": "sha512-qvwipnozMohxLXG1pOqoLiZKNkC4r4qqRucSoDwXowsNGDSULiqFTRUF05vcZWnwJSG22qTsynQhxbaMtnX9gw==", "dev": true, "requires": { - "map-age-cleaner": "0.1.3", - "mimic-fn": "2.1.0", - "p-is-promise": "2.1.0" + "map-age-cleaner": "^0.1.3", + "mimic-fn": "^2.1.0", + "p-is-promise": "^2.1.0" } }, "memory-fs": { @@ -10927,10 +10927,10 @@ "integrity": "sha512-MNpRGbNA52q6U92i0qbVpQNsgk7LExy41MdAlG84FeytfDOtRIf/mCHdEgG8rpTKOaNKiqUnZdlptF469hxqOw==", "dev": true, "requires": { - "loader-utils": "1.2.3", + "loader-utils": "^1.1.0", "normalize-url": "1.9.1", - "schema-utils": "1.0.0", - "webpack-sources": "1.4.3" + "schema-utils": "^1.0.0", + "webpack-sources": "^1.1.0" } }, "minimalistic-assert": { @@ -10964,8 +10964,8 @@ "integrity": "sha512-wxfUjg9WebH+CUDX/CdbRlh5SmfZiy/hpkxaRI16Y9W56Pa75sWgd/rvFilSgrauD9NyFymP/+JFV3KwzIsJeg==", "dev": true, "requires": { - "safe-buffer": "5.1.2", - "yallist": "3.1.1" + "safe-buffer": "^5.1.2", + "yallist": "^3.0.0" } }, "minizlib": { @@ -10974,7 +10974,7 @@ "integrity": "sha512-6ZYMOEnmVsdCeTJVE0W9ZD+pVnE8h9Hma/iOwwRDsdQoePpoX56/8B6z3P9VNwppJuBKNRuFDRNRqRWexT9G9Q==", "dev": true, "requires": { - "minipass": "2.9.0" + "minipass": "^2.9.0" } }, "mississippi": { @@ -10983,16 +10983,16 @@ "integrity": "sha512-x471SsVjUtBRtcvd4BzKE9kFC+/2TeWgKCgw0bZcw1b9l2X3QX5vCWgF+KaZaYm87Ss//rHnWryupDrgLvmSkA==", "dev": true, "requires": { - "concat-stream": "1.6.2", - "duplexify": "3.7.1", - "end-of-stream": "1.4.3", - "flush-write-stream": "1.1.1", - "from2": "2.3.0", - "parallel-transform": "1.2.0", - "pump": "3.0.0", - "pumpify": "1.5.1", - "stream-each": "1.2.3", - "through2": "2.0.5" + "concat-stream": "^1.5.0", + "duplexify": "^3.4.2", + "end-of-stream": "^1.1.0", + "flush-write-stream": "^1.0.0", + "from2": "^2.1.0", + "parallel-transform": "^1.1.0", + "pump": "^3.0.0", + "pumpify": "^1.3.3", + "stream-each": "^1.1.0", + "through2": "^2.0.0" }, "dependencies": { "pump": { @@ -11001,8 +11001,8 @@ "integrity": "sha512-LwZy+p3SFs1Pytd/jYct4wpv49HiYCqd9Rlc5ZVdk0V+8Yzv6jR5Blk3TRmPL1ft69TxP0IMZGJ+WPFU2BFhww==", "dev": true, "requires": { - "end-of-stream": "1.4.3", - "once": "1.4.0" + "end-of-stream": "^1.1.0", + "once": "^1.3.1" } } } @@ -11057,7 +11057,7 @@ "resolved": "https://registry.npmjs.org/moment-timezone/-/moment-timezone-0.5.26.tgz", "integrity": "sha512-sFP4cgEKTCymBBKgoxZjYzlSovC20Y6J7y3nanDc5RoBIXKlZhoYwBoZGe3flwU6A372AcRwScH8KiwV6zjy1g==", "requires": { - "moment": "2.24.0" + "moment": ">= 2.9.0" } }, "move-concurrently": { @@ -11118,17 +11118,17 @@ "resolved": "https://registry.npmjs.org/nanomatch/-/nanomatch-1.2.13.tgz", "integrity": "sha512-fpoe2T0RbHwBTBUOftAfBPaDEi06ufaUai0mE6Yn1kacc3SnTErfb/h+X94VXzI64rKFHYImXSvdwGGCmwOqCA==", "requires": { - "arr-diff": "4.0.0", - "array-unique": "0.3.2", - "define-property": "2.0.2", - "extend-shallow": "3.0.2", - "fragment-cache": "0.2.1", - "is-windows": "1.0.2", - "kind-of": "6.0.2", - "object.pick": "1.3.0", - "regex-not": "1.0.2", - "snapdragon": "0.8.2", - "to-regex": "3.0.2" + "arr-diff": "^4.0.0", + "array-unique": "^0.3.2", + "define-property": "^2.0.2", + "extend-shallow": "^3.0.2", + "fragment-cache": "^0.2.1", + "is-windows": "^1.0.2", + "kind-of": "^6.0.2", + "object.pick": "^1.3.0", + "regex-not": "^1.0.0", + "snapdragon": "^0.8.1", + "to-regex": "^3.0.1" } }, "negotiator": { @@ -11155,33 +11155,33 @@ "integrity": "sha512-GT6QK5WAirQwALdeJPiXdgRd5PzRqcknb/C/G+cCDEbUFri4oGVmns2Nl4I0FGg/cRn6nXTxRiUunOSqZ3Lehw==", "dev": true, "requires": { - "ajv": "6.10.2", - "autoprefixer": "9.6.1", - "browserslist": "4.6.6", - "chalk": "2.4.2", - "chokidar": "3.2.1", - "clean-css": "4.2.1", - "commander": "3.0.2", - "fs-extra": "8.1.0", - "glob": "7.1.3", - "injection-js": "2.2.2", - "less": "3.9.0", - "less-plugin-npm-import": "2.1.0", - "node-sass-tilde-importer": "1.0.2", - "postcss": "7.0.17", - "postcss-url": "8.0.0", - "read-pkg-up": "5.0.0", - "rimraf": "3.0.0", - "rollup": "1.22.0", - "rollup-plugin-commonjs": "10.1.0", - "rollup-plugin-json": "4.0.0", - "rollup-plugin-node-resolve": "5.2.0", - "rollup-plugin-sourcemaps": "0.4.2", - "rxjs": "6.5.3", - "sass": "1.22.9", - "stylus": "0.54.5", - "terser": "4.1.4", - "update-notifier": "3.0.1" + "ajv": "^6.10.2", + "autoprefixer": "^9.6.0", + "browserslist": "^4.0.0", + "chalk": "^2.3.1", + "chokidar": "^3.0.0", + "clean-css": "^4.1.11", + "commander": "^3.0.0", + "fs-extra": "^8.0.0", + "glob": "^7.1.2", + "injection-js": "^2.2.1", + "less": "^3.8.0", + "less-plugin-npm-import": "^2.1.0", + "node-sass-tilde-importer": "^1.0.0", + "postcss": "^7.0.0", + "postcss-url": "^8.0.0", + "read-pkg-up": "^5.0.0", + "rimraf": "^3.0.0", + "rollup": "^1.12.1", + "rollup-plugin-commonjs": "^10.0.0", + "rollup-plugin-json": "^4.0.0", + "rollup-plugin-node-resolve": "^5.0.0", + "rollup-plugin-sourcemaps": "^0.4.2", + "rxjs": "^6.0.0", + "sass": "^1.17.3", + "stylus": "^0.54.5", + "terser": "^4.1.2", + "update-notifier": "^3.0.0" }, "dependencies": { "ajv": { @@ -11190,10 +11190,10 @@ "integrity": "sha512-TXtUUEYHuaTEbLZWIKUr5pmBuhDLy+8KYtPYdcV8qC+pOZL+NKqYwvWSRrVXHn+ZmRRAu8vJTAznH7Oag6RVRw==", "dev": true, "requires": { - "fast-deep-equal": "2.0.1", - "fast-json-stable-stringify": "2.0.0", - "json-schema-traverse": "0.4.1", - "uri-js": "4.2.2" + "fast-deep-equal": "^2.0.1", + "fast-json-stable-stringify": "^2.0.0", + "json-schema-traverse": "^0.4.1", + "uri-js": "^4.2.2" } }, "anymatch": { @@ -11202,8 +11202,8 @@ "integrity": "sha512-mM8522psRCqzV+6LhomX5wgp25YVibjh8Wj23I5RPkPppSVSjyKD2A2mBJmWGa+KN7f2D6LNh9jkBCeyLktzjg==", "dev": true, "requires": { - "normalize-path": "3.0.0", - "picomatch": "2.0.7" + "normalize-path": "^3.0.0", + "picomatch": "^2.0.4" } }, "binary-extensions": { @@ -11218,7 +11218,7 @@ "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", "dev": true, "requires": { - "fill-range": "7.0.1" + "fill-range": "^7.0.1" } }, "chokidar": { @@ -11227,14 +11227,14 @@ "integrity": "sha512-/j5PPkb5Feyps9e+jo07jUZGvkB5Aj953NrI4s8xSVScrAo/RHeILrtdb4uzR7N6aaFFxxJ+gt8mA8HfNpw76w==", "dev": true, "requires": { - "anymatch": "3.1.1", - "braces": "3.0.2", - "fsevents": "2.1.0", - "glob-parent": "5.1.0", - "is-binary-path": "2.1.0", - "is-glob": "4.0.1", - "normalize-path": "3.0.0", - "readdirp": "3.1.3" + "anymatch": "~3.1.1", + "braces": "~3.0.2", + "fsevents": "~2.1.0", + "glob-parent": "~5.1.0", + "is-binary-path": "~2.1.0", + "is-glob": "~4.0.1", + "normalize-path": "~3.0.0", + "readdirp": "~3.1.3" } }, "commander": { @@ -11249,7 +11249,7 @@ "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", "dev": true, "requires": { - "to-regex-range": "5.0.1" + "to-regex-range": "^5.0.1" } }, "fsevents": { @@ -11265,7 +11265,7 @@ "integrity": "sha512-qjtRgnIVmOfnKUE3NJAQEdk+lKrxfw8t5ke7SXtfMTHcjsBfOfWXCQfdb30zfDoZQ2IRSIiidmjtbHZPZ++Ihw==", "dev": true, "requires": { - "is-glob": "4.0.1" + "is-glob": "^4.0.1" } }, "is-binary-path": { @@ -11274,7 +11274,7 @@ "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==", "dev": true, "requires": { - "binary-extensions": "2.0.0" + "binary-extensions": "^2.0.0" } }, "is-number": { @@ -11295,10 +11295,10 @@ "integrity": "sha512-OOY5b7PAEFV0E2Fir1KOkxchnZNCdowAJgQ5NuxjpBKTRP3pQhwkrkxqQjeoKJ+fO7bCpmIZaogI4eZGDMEGOw==", "dev": true, "requires": { - "@babel/code-frame": "7.5.5", - "error-ex": "1.3.2", - "json-parse-better-errors": "1.0.2", - "lines-and-columns": "1.1.6" + "@babel/code-frame": "^7.0.0", + "error-ex": "^1.3.1", + "json-parse-better-errors": "^1.0.1", + "lines-and-columns": "^1.1.6" } }, "read-pkg": { @@ -11307,10 +11307,10 @@ "integrity": "sha512-Ug69mNOpfvKDAc2Q8DRpMjjzdtrnv9HcSMX+4VsZxD1aZ6ZzrIE7rlzXBtWTyhULSMKg076AW6WR5iZpD0JiOg==", "dev": true, "requires": { - "@types/normalize-package-data": "2.4.0", - "normalize-package-data": "2.5.0", - "parse-json": "5.0.0", - "type-fest": "0.6.0" + "@types/normalize-package-data": "^2.4.0", + "normalize-package-data": "^2.5.0", + "parse-json": "^5.0.0", + "type-fest": "^0.6.0" } }, "read-pkg-up": { @@ -11319,8 +11319,8 @@ "integrity": "sha512-XBQjqOBtTzyol2CpsQOw8LHV0XbDZVG7xMMjmXAJomlVY03WOBRmYgDJETlvcg0H63AJvPRwT7GFi5rvOzUOKg==", "dev": true, "requires": { - "find-up": "3.0.0", - "read-pkg": "5.2.0" + "find-up": "^3.0.0", + "read-pkg": "^5.0.0" } }, "readdirp": { @@ -11329,7 +11329,7 @@ "integrity": "sha512-ZOsfTGkjO2kqeR5Mzr5RYDbTGYneSkdNKX2fOX2P5jF7vMrd/GNnIAUtDldeHHumHUCQ3V05YfWUdxMPAsRu9Q==", "dev": true, "requires": { - "picomatch": "2.0.7" + "picomatch": "^2.0.4" } }, "rimraf": { @@ -11338,7 +11338,7 @@ "integrity": "sha512-NDGVxTsjqfunkds7CqsOiEnxln4Bo7Nddl3XhS4pXg5OzwkLqJ971ZVAAnB+DDLnF76N+VnDEiBHaVV8I06SUg==", "dev": true, "requires": { - "glob": "7.1.3" + "glob": "^7.1.3" } }, "to-regex-range": { @@ -11347,7 +11347,7 @@ "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", "dev": true, "requires": { - "is-number": "7.0.0" + "is-number": "^7.0.0" } }, "type-fest": { @@ -11363,7 +11363,7 @@ "resolved": "https://registry.npmjs.org/ngrx-store-localstorage/-/ngrx-store-localstorage-8.0.0.tgz", "integrity": "sha512-3AHqw1AeXDXU/Hgxlh5fjP/srgGuP8BJR2uQ6ViGLeKwMEQDZuQfwd8zu9+bPZcvIxbAcxjWoGCVJXYEkBlxmg==", "requires": { - "deepmerge": "3.3.0" + "deepmerge": "^3.2.0" } }, "ngx-moment": { @@ -11371,7 +11371,7 @@ "resolved": "https://registry.npmjs.org/ngx-moment/-/ngx-moment-3.4.0.tgz", "integrity": "sha512-GEqzSsu12VsXXP35aerlQpuZ1ienEYQZxHmp+RH7EuJD7hWamKgLOpmbiDI9Ij3KLW/UApvonYzZvyRSv3ea/w==", "requires": { - "tslib": "1.10.0" + "tslib": "^1.9.0" } }, "nice-try": { @@ -11409,29 +11409,29 @@ "integrity": "sha512-h/zcD8H9kaDZ9ALUWwlBUDo6TKF8a7qBSCSEGfjTVIYeqsioSKaAX+BN7NgiMGp6iSIXZ3PxgCu8KS3b71YK5Q==", "dev": true, "requires": { - "assert": "1.5.0", - "browserify-zlib": "0.2.0", - "buffer": "4.9.1", - "console-browserify": "1.1.0", - "constants-browserify": "1.0.0", - "crypto-browserify": "3.12.0", - "domain-browser": "1.2.0", - "events": "3.0.0", - "https-browserify": "1.0.0", - "os-browserify": "0.3.0", + "assert": "^1.1.1", + "browserify-zlib": "^0.2.0", + "buffer": "^4.3.0", + "console-browserify": "^1.1.0", + "constants-browserify": "^1.0.0", + "crypto-browserify": "^3.11.0", + "domain-browser": "^1.1.1", + "events": "^3.0.0", + "https-browserify": "^1.0.0", + "os-browserify": "^0.3.0", "path-browserify": "0.0.1", - "process": "0.11.10", - "punycode": "1.4.1", - "querystring-es3": "0.2.1", - "readable-stream": "2.3.6", - "stream-browserify": "2.0.2", - "stream-http": "2.8.3", - "string_decoder": "1.1.1", - "timers-browserify": "2.0.11", + "process": "^0.11.10", + "punycode": "^1.2.4", + "querystring-es3": "^0.2.0", + "readable-stream": "^2.3.3", + "stream-browserify": "^2.0.1", + "stream-http": "^2.7.2", + "string_decoder": "^1.0.0", + "timers-browserify": "^2.0.4", "tty-browserify": "0.0.0", - "url": "0.11.0", - "util": "0.11.1", - "vm-browserify": "1.1.0" + "url": "^0.11.0", + "util": "^0.11.0", + "vm-browserify": "^1.0.1" }, "dependencies": { "punycode": { @@ -11447,7 +11447,7 @@ "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-1.1.33.tgz", "integrity": "sha512-I0V30bWQEoHb+10W8oedVoUrdjW5wIkYm0w7vvcrPO95pZY738m1k77GF5sO0vKg5eXYg9oGtrMAETbgZGm11A==", "requires": { - "semver": "5.6.0" + "semver": "^5.3.0" } }, "node-sass-tilde-importer": { @@ -11474,10 +11474,10 @@ "integrity": "sha512-/5CMN3T0R4XTj4DcGaexo+roZSdSFW/0AOOTROrjxzCG1wrWXEsGbRKevjlIL+ZDE4sZlJr5ED4YW0yqmkK+eA==", "dev": true, "requires": { - "hosted-git-info": "2.8.4", - "resolve": "1.12.0", - "semver": "5.6.0", - "validate-npm-package-license": "3.0.4" + "hosted-git-info": "^2.1.4", + "resolve": "^1.10.0", + "semver": "2 || 3 || 4 || 5", + "validate-npm-package-license": "^3.0.1" } }, "normalize-path": { @@ -11500,10 +11500,10 @@ "integrity": "sha1-LMDWazHqIwNkWENuNiDYWVTGbDw=", "dev": true, "requires": { - "object-assign": "4.1.1", - "prepend-http": "1.0.4", - "query-string": "4.3.4", - "sort-keys": "1.1.2" + "object-assign": "^4.0.1", + "prepend-http": "^1.0.0", + "query-string": "^4.1.0", + "sort-keys": "^1.0.0" } }, "normalizr": { @@ -11544,8 +11544,8 @@ "integrity": "sha512-zTLo8UcVYtDU3gdeaFu2Xu0n0EvelfHDGuqtNIn5RO7yQj4H1TqNdBc/yZjxnWA0PVB8D3Woyp0i5B43JwQ6Vw==", "dev": true, "requires": { - "ignore-walk": "3.0.2", - "npm-bundled": "1.0.6" + "ignore-walk": "^3.0.1", + "npm-bundled": "^1.0.1" } }, "npm-pick-manifest": { @@ -11554,9 +11554,9 @@ "integrity": "sha512-wNprTNg+X5nf+tDi+hbjdHhM4bX+mKqv6XmPh7B5eG+QY9VARfQPfCEH013H5GqfNj6ee8Ij2fg8yk0mzps1Vw==", "dev": true, "requires": { - "figgy-pudding": "3.5.1", - "npm-package-arg": "6.1.0", - "semver": "5.6.0" + "figgy-pudding": "^3.5.1", + "npm-package-arg": "^6.0.0", + "semver": "^5.4.1" } }, "npm-registry-fetch": { @@ -11565,13 +11565,13 @@ "integrity": "sha512-1ZQ+yjnxc698R5h9Yje9CASapzAZr7aYDkJDdERg9xg2hOEY0vRJwskOaJAXq8N/eLavzvW4g564YAfq6zMn/A==", "dev": true, "requires": { - "JSONStream": "1.3.5", - "bluebird": "3.7.0", - "figgy-pudding": "3.5.1", - "lru-cache": "5.1.1", - "make-fetch-happen": "5.0.0", - "npm-package-arg": "6.1.0", - "safe-buffer": "5.2.0" + "JSONStream": "^1.3.4", + "bluebird": "^3.5.1", + "figgy-pudding": "^3.4.1", + "lru-cache": "^5.1.1", + "make-fetch-happen": "^5.0.0", + "npm-package-arg": "^6.1.0", + "safe-buffer": "^5.2.0" }, "dependencies": { "safe-buffer": { @@ -11842,7 +11842,7 @@ "integrity": "sha512-5NcSkPHhwTVFIQN+TUqXoS5+dlElHXdpAWu9I0HP20YOtIi+aZ0Ct82jdlILDxjLEAWwvm+qj1m6aEtsDVmm6Q==", "dev": true, "requires": { - "mimic-fn": "2.1.0" + "mimic-fn": "^2.1.0" } }, "open": { @@ -11851,7 +11851,7 @@ "integrity": "sha512-IFenVPgF70fSm1keSd2iDBIDIBZkroLeuffXq+wKTzTJlBpesFWojV9lb8mzOfaAzM1sr7HQHuO0vtV0zYekGg==", "dev": true, "requires": { - "is-wsl": "1.1.0" + "is-wsl": "^1.1.0" } }, "opn": { @@ -11860,7 +11860,7 @@ "integrity": "sha512-PqHpggC9bLV0VeWcdKhkpxY+3JTzetLSqTCWL/z/tFIbI6G8JCjondXklT1JinczLz2Xib62sSp0T/gKT4KksA==", "dev": true, "requires": { - "is-wsl": "1.1.0" + "is-wsl": "^1.1.0" } }, "optimist": { @@ -11985,7 +11985,7 @@ "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-2.2.1.tgz", "integrity": "sha512-85Tk+90UCVWvbDavCLKPOLC9vvY8OwEX/RtKF+/1OADJMVlFfEHOiMTPVyxg7mk/dKa+ipdHm0OUkTvCpMTuwg==", "requires": { - "p-try": "2.2.0" + "p-try": "^2.0.0" } }, "p-locate": { @@ -11993,7 +11993,7 @@ "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-3.0.0.tgz", "integrity": "sha512-x+12w/To+4GFfgJhBEpiDcLozRJGegY+Ei7/z0tSLkMmxGZNybVMSfWj9aJn8Z5Fc7dBUNJOOVgPv2H7IwulSQ==", "requires": { - "p-limit": "2.2.1" + "p-limit": "^2.0.0" } }, "p-map": { @@ -12008,7 +12008,7 @@ "integrity": "sha512-XE6G4+YTTkT2a0UWb2kjZe8xNwf8bIbnqpc/IS/idOBVhyves0mK5OJgeocjx7q5pvX/6m23xuzVPYT1uGM73w==", "dev": true, "requires": { - "retry": "0.12.0" + "retry": "^0.12.0" } }, "p-try": { @@ -12022,10 +12022,10 @@ "integrity": "sha512-k3bdm2n25tkyxcjSKzB5x8kfVxlMdgsbPr0GkZcwHsLpba6cBjqCt1KlcChKEvxHIcTB1FVMuwoijZ26xex5MQ==", "dev": true, "requires": { - "got": "9.6.0", - "registry-auth-token": "4.0.0", - "registry-url": "5.1.0", - "semver": "6.3.0" + "got": "^9.6.0", + "registry-auth-token": "^4.0.0", + "registry-url": "^5.0.0", + "semver": "^6.2.0" }, "dependencies": { "semver": { @@ -12042,34 +12042,34 @@ "integrity": "sha512-jAEP+Nqj4kyMWyNpfTU/Whx1jA7jEc5cCOlurm0/0oL+v8TAp1QSsK83N7bYe+2bEdFzMAtPG5TBebjzzGV0cA==", "dev": true, "requires": { - "bluebird": "3.7.0", - "cacache": "12.0.2", - "figgy-pudding": "3.5.1", - "get-stream": "4.1.0", - "glob": "7.1.3", - "infer-owner": "1.0.4", - "lru-cache": "5.1.1", - "make-fetch-happen": "5.0.0", - "minimatch": "3.0.4", - "minipass": "2.9.0", - "mississippi": "3.0.0", - "mkdirp": "0.5.1", - "normalize-package-data": "2.5.0", - "npm-package-arg": "6.1.0", - "npm-packlist": "1.4.4", - "npm-pick-manifest": "2.2.3", - "npm-registry-fetch": "4.0.1", - "osenv": "0.1.5", - "promise-inflight": "1.0.1", - "promise-retry": "1.1.1", - "protoduck": "5.0.1", - "rimraf": "2.7.1", - "safe-buffer": "5.1.2", - "semver": "5.6.0", - "ssri": "6.0.1", - "tar": "4.4.13", - "unique-filename": "1.1.1", - "which": "1.3.1" + "bluebird": "^3.5.3", + "cacache": "^12.0.2", + "figgy-pudding": "^3.5.1", + "get-stream": "^4.1.0", + "glob": "^7.1.3", + "infer-owner": "^1.0.4", + "lru-cache": "^5.1.1", + "make-fetch-happen": "^5.0.0", + "minimatch": "^3.0.4", + "minipass": "^2.3.5", + "mississippi": "^3.0.0", + "mkdirp": "^0.5.1", + "normalize-package-data": "^2.4.0", + "npm-package-arg": "^6.1.0", + "npm-packlist": "^1.1.12", + "npm-pick-manifest": "^2.2.3", + "npm-registry-fetch": "^4.0.0", + "osenv": "^0.1.5", + "promise-inflight": "^1.0.1", + "promise-retry": "^1.1.1", + "protoduck": "^5.0.1", + "rimraf": "^2.6.2", + "safe-buffer": "^5.1.2", + "semver": "^5.6.0", + "ssri": "^6.0.1", + "tar": "^4.4.8", + "unique-filename": "^1.1.1", + "which": "^1.3.1" }, "dependencies": { "npm-pick-manifest": { @@ -12078,9 +12078,9 @@ "integrity": "sha512-+IluBC5K201+gRU85vFlUwX3PFShZAbAgDNp2ewJdWMVSppdo/Zih0ul2Ecky/X7b51J7LrrUAP+XOmOCvYZqA==", "dev": true, "requires": { - "figgy-pudding": "3.5.1", - "npm-package-arg": "6.1.0", - "semver": "5.6.0" + "figgy-pudding": "^3.5.1", + "npm-package-arg": "^6.0.0", + "semver": "^5.4.1" } } } @@ -12097,9 +12097,9 @@ "integrity": "sha512-P2vSmIu38uIlvdcU7fDkyrxj33gTUy/ABO5ZUbGowxNCopBq/OoD42bP4UmMrJoPyk4Uqf0mu3mtWBhHCZD8yg==", "dev": true, "requires": { - "cyclist": "1.0.1", - "inherits": "2.0.4", - "readable-stream": "2.3.6" + "cyclist": "^1.0.1", + "inherits": "^2.0.3", + "readable-stream": "^2.1.5" } }, "parse-asn1": { @@ -12108,12 +12108,12 @@ "integrity": "sha512-jkMYn1dcJqF6d5CpU689bq7w/b5ALS9ROVSpQDPrZsqqesUJii9qutvoT5ltGedNXMO2e16YUWIghG9KxaViTQ==", "dev": true, "requires": { - "asn1.js": "4.10.1", - "browserify-aes": "1.2.0", - "create-hash": "1.2.0", - "evp_bytestokey": "1.0.3", - "pbkdf2": "3.0.17", - "safe-buffer": "5.1.2" + "asn1.js": "^4.0.0", + "browserify-aes": "^1.0.0", + "create-hash": "^1.1.0", + "evp_bytestokey": "^1.0.0", + "pbkdf2": "^3.0.3", + "safe-buffer": "^5.1.1" } }, "parse-filepath": { @@ -12317,7 +12317,7 @@ "resolved": "https://registry.npmjs.org/pkg-dir/-/pkg-dir-3.0.0.tgz", "integrity": "sha512-/E57AYkoeQ25qkxMj5PBOVgF8Kiu/h7cYS30Z5+R7WaiCCBfLq58ZI/dSeaEKb9WVJV5n/03QwrN3IeWIFllvw==", "requires": { - "find-up": "3.0.0" + "find-up": "^3.0.0" } }, "plugin-error": { @@ -12326,10 +12326,10 @@ "integrity": "sha512-L1zP0dk7vGweZME2i+EeakvUNqSrdiI3F91TwEoYiGrAfUXmVv6fJIq4g82PAXxNsWOp0J7ZqQy/3Szz0ajTxA==", "dev": true, "requires": { - "ansi-colors": "1.1.0", - "arr-diff": "4.0.0", - "arr-union": "3.1.0", - "extend-shallow": "3.0.2" + "ansi-colors": "^1.0.1", + "arr-diff": "^4.0.0", + "arr-union": "^3.1.0", + "extend-shallow": "^3.0.2" }, "dependencies": { "ansi-colors": { @@ -12338,7 +12338,7 @@ "integrity": "sha512-SFKX67auSNoVR38N3L+nvsPjOE0bybKTYbkf5tRvushrAPQ9V75huw0ZxBkKVeRU9kqH3d6HA4xTckbwZ4ixmA==", "dev": true, "requires": { - "ansi-wrap": "0.1.0" + "ansi-wrap": "^0.1.0" } } } @@ -12355,9 +12355,9 @@ "integrity": "sha512-ekRl7zD2qxYndYflwiryJwMioBI7LI7rVXg3EnLK3sjkouT5eOuhS3gS255XxBksa30VG8UPZYZCdgfGOfkSUg==", "dev": true, "requires": { - "async": "1.5.2", - "debug": "2.6.9", - "mkdirp": "0.5.1" + "async": "^1.5.2", + "debug": "^2.2.0", + "mkdirp": "0.5.x" }, "dependencies": { "async": { @@ -12379,9 +12379,9 @@ "integrity": "sha512-546ZowA+KZ3OasvQZHsbuEpysvwTZNGJv9EfyCQdsIDltPSWHAeTQ5fQy/Npi2ZDtLI3zs7Ps/p6wThErhm9fQ==", "dev": true, "requires": { - "chalk": "2.4.2", - "source-map": "0.6.1", - "supports-color": "6.1.0" + "chalk": "^2.4.2", + "source-map": "^0.6.1", + "supports-color": "^6.1.0" }, "dependencies": { "source-map": { @@ -12418,8 +12418,8 @@ "integrity": "sha512-4pV3JJVPLd5+RueiVVB+gFOAa7GWc25XQcMp86Zexzke69mKf6Nx9LRcQywdz7yZI9n1udOxmLuAwTBypypF8Q==", "dev": true, "requires": { - "cosmiconfig": "5.2.1", - "import-cwd": "2.1.0" + "cosmiconfig": "^5.0.0", + "import-cwd": "^2.0.0" } }, "postcss-loader": { @@ -12654,17 +12654,17 @@ "integrity": "sha512-XINj6b8CYuUYC93SG3xPkxlyUc3IJbD6Vvo75CVGuG9uzsefDzWQrhz0Lq8vbPxtb4d63CZdYophF8k8Or/YiA==", "dev": true, "requires": { - "adm-zip": "0.4.13", - "chalk": "1.1.3", - "del": "2.2.2", - "glob": "7.1.3", - "ini": "1.3.5", - "minimist": "1.2.0", - "q": "1.4.1", - "request": "2.88.0", - "rimraf": "2.7.1", - "semver": "5.6.0", - "xml2js": "0.4.22" + "adm-zip": "^0.4.9", + "chalk": "^1.1.1", + "del": "^2.2.0", + "glob": "^7.0.3", + "ini": "^1.3.4", + "minimist": "^1.2.0", + "q": "^1.4.1", + "request": "^2.87.0", + "rimraf": "^2.5.2", + "semver": "^5.3.0", + "xml2js": "^0.4.17" } } } @@ -12675,7 +12675,7 @@ "integrity": "sha512-t/7RxHXPH6cJtP0pRG6smSr9QJidhB+3kXu0KgXnbGYMgzEnUxRQ4/LDdfOwZEMyIh3/xHb8PX3t+lfL9z+YVQ==", "dev": true, "requires": { - "forwarded": "0.1.2", + "forwarded": "~0.1.2", "ipaddr.js": "1.9.0" } }, @@ -12778,8 +12778,8 @@ "integrity": "sha1-u7aTucqRXCMlFbIosaArYJBD2+s=", "dev": true, "requires": { - "object-assign": "4.1.1", - "strict-uri-encode": "1.1.0" + "object-assign": "^4.1.0", + "strict-uri-encode": "^1.0.0" } }, "querystring": { @@ -12851,8 +12851,8 @@ "integrity": "sha512-lzUVMuJ06HF4rYveaz9Tv0WRlUMxJ0Y1hgSkkgg+50iEdaI0TthyEDe08KIHb0XsF6rn8WYTqPCaGTZg3sX+qA==", "dev": true, "requires": { - "loader-utils": "1.2.3", - "schema-utils": "2.4.1" + "loader-utils": "^1.1.0", + "schema-utils": "^2.0.1" }, "dependencies": { "ajv": { @@ -12861,10 +12861,10 @@ "integrity": "sha512-TXtUUEYHuaTEbLZWIKUr5pmBuhDLy+8KYtPYdcV8qC+pOZL+NKqYwvWSRrVXHn+ZmRRAu8vJTAznH7Oag6RVRw==", "dev": true, "requires": { - "fast-deep-equal": "2.0.1", - "fast-json-stable-stringify": "2.0.0", - "json-schema-traverse": "0.4.1", - "uri-js": "4.2.2" + "fast-deep-equal": "^2.0.1", + "fast-json-stable-stringify": "^2.0.0", + "json-schema-traverse": "^0.4.1", + "uri-js": "^4.2.2" } }, "schema-utils": { @@ -12873,8 +12873,8 @@ "integrity": "sha512-RqYLpkPZX5Oc3fw/kHHHyP56fg5Y+XBpIpV8nCg0znIALfq3OH+Ea9Hfeac9BAMwG5IICltiZ0vxFvJQONfA5w==", "dev": true, "requires": { - "ajv": "6.10.2", - "ajv-keywords": "3.4.1" + "ajv": "^6.10.2", + "ajv-keywords": "^3.4.1" } } } @@ -12914,11 +12914,11 @@ "integrity": "sha512-KLhu8M1ZZNkMcrq1+0UJbR8Dii8KZUqB0Sha4mOx/bknfKI/fyrQVrG/YIt2UOtG667sD8+ee4EXMM91W9dC+A==", "dev": true, "requires": { - "glob": "7.1.3", - "graceful-fs": "4.2.2", - "json-parse-better-errors": "1.0.2", - "normalize-package-data": "2.5.0", - "slash": "1.0.0" + "glob": "^7.1.1", + "graceful-fs": "^4.1.2", + "json-parse-better-errors": "^1.0.1", + "normalize-package-data": "^2.0.0", + "slash": "^1.0.0" } }, "read-package-tree": { @@ -12927,9 +12927,9 @@ "integrity": "sha512-mLUDsD5JVtlZxjSlPPx1RETkNjjvQYuweKwNVt1Sn8kP5Jh44pvYuUHCp6xSVDZWbNxVxG5lyZJ921aJH61sTw==", "dev": true, "requires": { - "read-package-json": "2.1.0", - "readdir-scoped-modules": "1.1.0", - "util-promisify": "2.1.0" + "read-package-json": "^2.0.0", + "readdir-scoped-modules": "^1.0.0", + "util-promisify": "^2.1.0" } }, "read-pkg": { @@ -13013,10 +13013,10 @@ "integrity": "sha512-asaikDeqAQg7JifRsZn1NJZXo9E+VwlyCfbkZhwyISinqk5zNS6266HS5kah6P0SaQKGF6SkNnZVHUzHFYxYDw==", "dev": true, "requires": { - "debuglog": "1.0.1", - "dezalgo": "1.0.3", - "graceful-fs": "4.2.2", - "once": "1.4.0" + "debuglog": "^1.0.1", + "dezalgo": "^1.0.0", + "graceful-fs": "^4.1.2", + "once": "^1.3.0" } }, "readdirp": { @@ -13024,9 +13024,9 @@ "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-2.2.1.tgz", "integrity": "sha512-1JU/8q+VgFZyxwrJ+SVIOsh+KywWGpds3NTqikiKpDMZWScmAYyKIgqkO+ARvNWJfXeXR1zxz7aHF4u4CyH6vQ==", "requires": { - "graceful-fs": "4.2.2", - "micromatch": "3.1.10", - "readable-stream": "2.3.6" + "graceful-fs": "^4.1.11", + "micromatch": "^3.1.10", + "readable-stream": "^2.0.2" } }, "readline2": { @@ -13073,7 +13073,7 @@ "resolved": "https://registry.npmjs.org/regenerate-unicode-properties/-/regenerate-unicode-properties-8.1.0.tgz", "integrity": "sha512-LGZzkgtLY79GeXLm8Dp0BVLdQlWICzBnJz/ipWUgo59qBaZ+BHtq51P2q1uVZlppMuUAT37SDk39qUbjTWB7bA==", "requires": { - "regenerate": "1.4.0" + "regenerate": "^1.4.0" } }, "regenerator-runtime": { @@ -13087,7 +13087,7 @@ "resolved": "https://registry.npmjs.org/regenerator-transform/-/regenerator-transform-0.14.1.tgz", "integrity": "sha512-flVuee02C3FKRISbxhXl9mGzdbWUVHubl1SMaknjxkFB1/iqpJhArQUvRxOOPEc/9tAiX0BaQ28FJH10E4isSQ==", "requires": { - "private": "0.1.8" + "private": "^0.1.6" } }, "regex-not": { @@ -13105,7 +13105,7 @@ "integrity": "sha512-ztaw4M1VqgMwl9HlPpOuiYgItcHlunW0He2fE6eNfT6E/CF2FtYi9ofOYe4mKntstYk0Fyh/rDRBdS3AnxjlrA==", "dev": true, "requires": { - "define-properties": "1.1.3" + "define-properties": "^1.1.2" } }, "regexpu-core": { @@ -13113,12 +13113,12 @@ "resolved": "https://registry.npmjs.org/regexpu-core/-/regexpu-core-4.6.0.tgz", "integrity": "sha512-YlVaefl8P5BnFYOITTNzDvan1ulLOiXJzCNZxduTIosN17b87h3bvG9yHMoHaRuo88H4mQ06Aodj5VtYGGGiTg==", "requires": { - "regenerate": "1.4.0", - "regenerate-unicode-properties": "8.1.0", - "regjsgen": "0.5.0", - "regjsparser": "0.6.0", - "unicode-match-property-ecmascript": "1.0.4", - "unicode-match-property-value-ecmascript": "1.1.0" + "regenerate": "^1.4.0", + "regenerate-unicode-properties": "^8.1.0", + "regjsgen": "^0.5.0", + "regjsparser": "^0.6.0", + "unicode-match-property-ecmascript": "^1.0.4", + "unicode-match-property-value-ecmascript": "^1.1.0" } }, "registry-auth-token": { @@ -13127,8 +13127,8 @@ "integrity": "sha512-lpQkHxd9UL6tb3k/aHAVfnVtn+Bcs9ob5InuFLLEDqSqeq+AljB8GZW9xY0x7F+xYwEcjKe07nyoxzEYz6yvkw==", "dev": true, "requires": { - "rc": "1.2.8", - "safe-buffer": "5.1.2" + "rc": "^1.2.8", + "safe-buffer": "^5.0.1" } }, "registry-url": { @@ -13137,7 +13137,7 @@ "integrity": "sha512-8acYXXTI0AkQv6RAOjE3vOaIXZkT9wo4LOFbBKYQEEnnMNBpKqdUrI6S4NT0KPIo/WVvJ5tE/X5LF/TQUf0ekw==", "dev": true, "requires": { - "rc": "1.2.8" + "rc": "^1.2.8" } }, "regjsgen": { @@ -13150,7 +13150,7 @@ "resolved": "https://registry.npmjs.org/regjsparser/-/regjsparser-0.6.0.tgz", "integrity": "sha512-RQ7YyokLiQBomUJuUG8iGVvkgOLxwyZM8k6d3q5SAXpg4r5TZJZigKFvC6PpD+qQ98bCDC5YelPeA3EucDoNeQ==", "requires": { - "jsesc": "0.5.0" + "jsesc": "~0.5.0" } }, "remove-bom-buffer": { @@ -13221,9 +13221,9 @@ "integrity": "sha512-x8uSfKKK/5YiJ8JYsNMwa1PJYvHfPdUABVXd21ro09Nh5BRZ5ATuACwrqCNpktwVqVbFTk/cIGFepeMqY0oX3g==", "dev": true, "requires": { - "chalk": "2.4.2", - "glob": "7.1.4", - "yargs": "13.3.0" + "chalk": "^2.4.2", + "glob": "^7.1.4", + "yargs": "^13.3.0" }, "dependencies": { "ansi-regex": { @@ -13238,9 +13238,9 @@ "integrity": "sha512-PYeGSEmmHM6zvoef2w8TPzlrnNpXIjTipYK780YswmIP9vjxmd6Y2a3CB2Ks6/AU8NHjZugXvo8w3oWM2qnwXA==", "dev": true, "requires": { - "string-width": "3.1.0", - "strip-ansi": "5.2.0", - "wrap-ansi": "5.1.0" + "string-width": "^3.1.0", + "strip-ansi": "^5.2.0", + "wrap-ansi": "^5.1.0" } }, "emoji-regex": { @@ -13261,12 +13261,12 @@ "integrity": "sha512-hkLPepehmnKk41pUGm3sYxoFs/umurYfYJCerbXEyFIWcAzvpipAgVkBqqT9RBKMGjnq6kMuyYwha6csxbiM1A==", "dev": true, "requires": { - "fs.realpath": "1.0.0", - "inflight": "1.0.6", - "inherits": "2.0.4", - "minimatch": "3.0.4", - "once": "1.4.0", - "path-is-absolute": "1.0.1" + "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" } }, "is-fullwidth-code-point": { @@ -13287,9 +13287,9 @@ "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==", "dev": true, "requires": { - "emoji-regex": "7.0.3", - "is-fullwidth-code-point": "2.0.0", - "strip-ansi": "5.2.0" + "emoji-regex": "^7.0.1", + "is-fullwidth-code-point": "^2.0.0", + "strip-ansi": "^5.1.0" } }, "strip-ansi": { @@ -13298,7 +13298,7 @@ "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", "dev": true, "requires": { - "ansi-regex": "4.1.0" + "ansi-regex": "^4.1.0" } }, "which-module": { @@ -13313,9 +13313,9 @@ "integrity": "sha512-QC1/iN/2/RPVJ5jYK8BGttj5z83LmSKmvbvrXPNCLZSEb32KKVDJDl/MOt2N01qU2H/FkzEa9PKto1BqDjtd7Q==", "dev": true, "requires": { - "ansi-styles": "3.2.1", - "string-width": "3.1.0", - "strip-ansi": "5.2.0" + "ansi-styles": "^3.2.0", + "string-width": "^3.0.0", + "strip-ansi": "^5.0.0" } }, "yargs": { @@ -13324,16 +13324,16 @@ "integrity": "sha512-2eehun/8ALW8TLoIl7MVaRUrg+yCnenu8B4kBlRxj3GJGDKU1Og7sMXPNm1BYyM1DOJmTZ4YeN/Nwxv+8XJsUA==", "dev": true, "requires": { - "cliui": "5.0.0", - "find-up": "3.0.0", - "get-caller-file": "2.0.5", - "require-directory": "2.1.1", - "require-main-filename": "2.0.0", - "set-blocking": "2.0.0", - "string-width": "3.1.0", - "which-module": "2.0.0", - "y18n": "4.0.0", - "yargs-parser": "13.1.1" + "cliui": "^5.0.0", + "find-up": "^3.0.0", + "get-caller-file": "^2.0.1", + "require-directory": "^2.1.1", + "require-main-filename": "^2.0.0", + "set-blocking": "^2.0.0", + "string-width": "^3.0.0", + "which-module": "^2.0.0", + "y18n": "^4.0.0", + "yargs-parser": "^13.1.1" } }, "yargs-parser": { @@ -13342,8 +13342,8 @@ "integrity": "sha512-oVAVsHz6uFrg3XQheFII8ESO2ssAf9luWuAd6Wexsu4F3OtIW0o8IribPXYrD4WC24LWtPrJlGy87y5udK+dxQ==", "dev": true, "requires": { - "camelcase": "5.3.1", - "decamelize": "1.2.0" + "camelcase": "^5.0.0", + "decamelize": "^1.2.0" } } } @@ -13430,7 +13430,7 @@ "integrity": "sha1-lAhe9jWB7NPaqSREqP6U6CV3dR8=", "dev": true, "requires": { - "callsites": "0.2.0" + "callsites": "^0.2.0" } }, "callsites": { @@ -13463,7 +13463,7 @@ "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.12.0.tgz", "integrity": "sha512-B/dOmuoAik5bKcD6s6nXDCjzUKnaDvdkRyAk6rsmsKLipWj4797iothd7jmmUhWTfinVMU+wc56rYKsit2Qy4w==", "requires": { - "path-parse": "1.0.6" + "path-parse": "^1.0.6" } }, "resolve-cwd": { @@ -13481,8 +13481,8 @@ "integrity": "sha1-eaQGRMNivoLybv/nOcm7U4IEb0M=", "dev": true, "requires": { - "expand-tilde": "2.0.2", - "global-modules": "1.0.0" + "expand-tilde": "^2.0.0", + "global-modules": "^1.0.0" } }, "resolve-from": { @@ -13511,7 +13511,7 @@ "integrity": "sha1-kYcg7ztjHFZCvgaPFa3lpG9Loec=", "dev": true, "requires": { - "lowercase-keys": "1.0.1" + "lowercase-keys": "^1.0.0" } }, "restore-cursor": { @@ -13520,8 +13520,8 @@ "integrity": "sha512-l+sSefzHpj5qimhFSE5a8nufZYAM3sBSVMAPtYkmC+4EH2anSGaEMXSD0izRQbu9nfyQ9y5JrVmp7E8oZrUjvA==", "dev": true, "requires": { - "onetime": "5.1.0", - "signal-exit": "3.0.2" + "onetime": "^5.1.0", + "signal-exit": "^3.0.2" } }, "ret": { @@ -13553,7 +13553,7 @@ "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==", "dev": true, "requires": { - "glob": "7.1.3" + "glob": "^7.1.3" } }, "ripemd160": { @@ -13572,9 +13572,9 @@ "integrity": "sha512-x4l4ZrV/Mr/x/jvFTmwROdEAhbZjx16yDRTVSKWh/i4oJDuW2dVEbECT853mybYCz7BAitU8ElGlhx7dNjw3qQ==", "dev": true, "requires": { - "@types/estree": "0.0.39", - "@types/node": "12.7.9", - "acorn": "7.1.0" + "@types/estree": "*", + "@types/node": "*", + "acorn": "^7.1.0" }, "dependencies": { "acorn": { @@ -13591,11 +13591,11 @@ "integrity": "sha512-jlXbjZSQg8EIeAAvepNwhJj++qJWNJw1Cl0YnOqKtP5Djx+fFGkp3WRh+W0ASCaFG5w1jhmzDxgu3SJuVxPF4Q==", "dev": true, "requires": { - "estree-walker": "0.6.1", - "is-reference": "1.1.4", - "magic-string": "0.25.3", - "resolve": "1.12.0", - "rollup-pluginutils": "2.8.2" + "estree-walker": "^0.6.1", + "is-reference": "^1.1.2", + "magic-string": "^0.25.2", + "resolve": "^1.11.0", + "rollup-pluginutils": "^2.8.1" } }, "rollup-plugin-json": { @@ -13604,7 +13604,7 @@ "integrity": "sha512-hgb8N7Cgfw5SZAkb3jf0QXii6QX/FOkiIq2M7BAQIEydjHvTyxXHQiIzZaTFgx1GK0cRCHOCBHIyEkkLdWKxow==", "dev": true, "requires": { - "rollup-pluginutils": "2.8.2" + "rollup-pluginutils": "^2.5.0" } }, "rollup-plugin-node-resolve": { @@ -13614,10 +13614,10 @@ "dev": true, "requires": { "@types/resolve": "0.0.8", - "builtin-modules": "3.1.0", - "is-module": "1.0.0", - "resolve": "1.12.0", - "rollup-pluginutils": "2.8.2" + "builtin-modules": "^3.1.0", + "is-module": "^1.0.0", + "resolve": "^1.11.1", + "rollup-pluginutils": "^2.8.1" } }, "rollup-plugin-sourcemaps": { @@ -13645,7 +13645,7 @@ "integrity": "sha1-A3GrSuC91yDUFm19/aZP96RFpsA=", "dev": true, "requires": { - "is-promise": "2.1.0" + "is-promise": "^2.1.0" } }, "run-parallel": { @@ -13679,7 +13679,7 @@ "resolved": "https://registry.npmjs.org/rxjs/-/rxjs-6.5.3.tgz", "integrity": "sha512-wuYsAYYFdWTAnAaPoKGNhfpWwKZbJW+HgAJ+mImp+Epl7BG8oNWBCTyRM8gba9k4lk8BgWdoYm21Mo/RYhhbgA==", "requires": { - "tslib": "1.10.0" + "tslib": "^1.9.0" } }, "rxjs-spy": { @@ -13687,11 +13687,11 @@ "resolved": "https://registry.npmjs.org/rxjs-spy/-/rxjs-spy-7.5.1.tgz", "integrity": "sha512-dJ9mO4HvW2r16PsU15Qsc0RVkG7pFrfyCNTGx3vrxWje3kIgZ6QjMVnWblQxbniZ32lwLk/2x9+D2O6GhgXV/w==", "requires": { - "@types/circular-json": "0.4.0", - "@types/stacktrace-js": "0.0.32", - "circular-json": "0.5.9", - "error-stack-parser": "2.0.4", - "stacktrace-gps": "3.0.3" + "@types/circular-json": "^0.4.0", + "@types/stacktrace-js": "^0.0.32", + "circular-json": "^0.5.0", + "error-stack-parser": "^2.0.1", + "stacktrace-gps": "^3.0.2" } }, "rxjs-tslint": { @@ -13700,11 +13700,11 @@ "integrity": "sha512-NnOfqutNfdT7VQnQm32JLYh2gDZjc0gdWZFtrxf/czNGkLKJ1nOO6jbKAFI09W0f9lCtv6P2ozxjbQH8TSPPFQ==", "dev": true, "requires": { - "chalk": "2.4.2", - "optimist": "0.6.1", - "tslint": "5.20.0", - "tsutils": "2.29.0", - "typescript": "3.5.3" + "chalk": "^2.4.0", + "optimist": "^0.6.1", + "tslint": "^5.9.1", + "tsutils": "^2.25.0", + "typescript": ">=2.8.3" } }, "rxjs-websockets": { @@ -13736,7 +13736,7 @@ "integrity": "sha512-FzU1X2V8DlnqabrL4u7OBwD2vcOzNMongEJEx3xMEhWY/v26FFR3aG0hyeu2T965sfR0E9ufJwmG+Qjz78vFPQ==", "dev": true, "requires": { - "chokidar": "2.0.4" + "chokidar": ">=2.0.0 <4.0.0" } }, "sass-lint": { @@ -13745,20 +13745,20 @@ "integrity": "sha512-DSyah8/MyjzW2BWYmQWekYEKir44BpLqrCFsgs9iaWiVTcwZfwXHF586hh3D1n+/9ihUNMfd8iHAyb9KkGgs7Q==", "dev": true, "requires": { - "commander": "2.20.0", - "eslint": "2.13.1", + "commander": "^2.8.1", + "eslint": "^2.7.0", "front-matter": "2.1.2", - "fs-extra": "3.0.1", - "glob": "7.1.3", - "globule": "1.2.1", - "gonzales-pe-sl": "4.2.3", - "js-yaml": "3.13.1", - "known-css-properties": "0.3.0", - "lodash.capitalize": "4.2.1", - "lodash.kebabcase": "4.1.1", - "merge": "1.2.1", - "path-is-absolute": "1.0.1", - "util": "0.10.4" + "fs-extra": "^3.0.1", + "glob": "^7.0.0", + "globule": "^1.0.0", + "gonzales-pe-sl": "^4.2.3", + "js-yaml": "^3.5.4", + "known-css-properties": "^0.3.0", + "lodash.capitalize": "^4.1.0", + "lodash.kebabcase": "^4.0.0", + "merge": "^1.2.0", + "path-is-absolute": "^1.0.0", + "util": "^0.10.3" }, "dependencies": { "fs-extra": { @@ -13767,9 +13767,9 @@ "integrity": "sha1-N5TzeMWLNC6n27sjCVEJxLO2IpE=", "dev": true, "requires": { - "graceful-fs": "4.2.2", - "jsonfile": "3.0.1", - "universalify": "0.1.2" + "graceful-fs": "^4.1.2", + "jsonfile": "^3.0.0", + "universalify": "^0.1.0" } }, "inherits": { @@ -13784,7 +13784,7 @@ "integrity": "sha1-pezG9l9T9mLEQVx2daAzHQmS7GY=", "dev": true, "requires": { - "graceful-fs": "4.2.2" + "graceful-fs": "^4.1.6" } }, "util": { @@ -13804,11 +13804,11 @@ "integrity": "sha512-h8yUWaWtsbuIiOCgR9fd9c2lRXZ2uG+h8Dzg/AGNj+Hg/3TO8+BBAW9mEP+mh8ei+qBKqSJ0F1FLlYjNBc61OA==", "dev": true, "requires": { - "clone-deep": "4.0.1", - "loader-utils": "1.2.3", - "neo-async": "2.6.1", - "pify": "4.0.1", - "semver": "5.6.0" + "clone-deep": "^4.0.1", + "loader-utils": "^1.0.1", + "neo-async": "^2.5.0", + "pify": "^4.0.1", + "semver": "^5.5.0" }, "dependencies": { "pify": { @@ -13931,18 +13931,18 @@ "dev": true, "requires": { "debug": "2.6.9", - "depd": "1.1.2", - "destroy": "1.0.4", - "encodeurl": "1.0.2", - "escape-html": "1.0.3", - "etag": "1.8.1", + "depd": "~1.1.2", + "destroy": "~1.0.4", + "encodeurl": "~1.0.2", + "escape-html": "~1.0.3", + "etag": "~1.8.1", "fresh": "0.5.2", - "http-errors": "1.7.2", + "http-errors": "~1.7.2", "mime": "1.6.0", "ms": "2.1.1", - "on-finished": "2.3.0", - "range-parser": "1.2.1", - "statuses": "1.5.0" + "on-finished": "~2.3.0", + "range-parser": "~1.2.1", + "statuses": "~1.5.0" }, "dependencies": { "ms": { @@ -13980,10 +13980,10 @@ "integrity": "sha1-i1VoC7S+KDoLW/TqLjhYC+HZMg0=", "dev": true, "requires": { - "depd": "1.1.2", + "depd": "~1.1.2", "inherits": "2.0.3", "setprototypeof": "1.1.0", - "statuses": "1.5.0" + "statuses": ">= 1.4.0 < 2" } }, "inherits": { @@ -14006,9 +14006,9 @@ "integrity": "sha512-JMrvUwE54emCYWlTI+hGrGv5I8dEwmco/00EvkzIIsR7MqrHonbD9pO2MOfFnpFntl7ecpZs+3mW+XbQZu9QCg==", "dev": true, "requires": { - "encodeurl": "1.0.2", - "escape-html": "1.0.3", - "parseurl": "1.3.3", + "encodeurl": "~1.0.2", + "escape-html": "~1.0.3", + "parseurl": "~1.3.3", "send": "0.17.1" } }, @@ -14073,7 +14073,7 @@ "integrity": "sha512-/6KqX+GVUdqPuPPd2LxDDxzX6CAbjJehAAOKlNpqqUpAqPM6HeL8f+o3a+JsyGjn2lv0WY8UsTgUJjU9Ok55NA==", "dev": true, "requires": { - "kind-of": "6.0.2" + "kind-of": "^6.0.2" } }, "shebang-command": { @@ -14361,7 +14361,7 @@ "integrity": "sha512-D2y4bovYpzziGgbHYtGCMjlJM36vAl/y+xUyn1C+FVx8szd1E+86KwVw6XvYSzOP8iMpm1X0I4xJD+QtUb36OA==", "dev": true, "requires": { - "websocket-driver": "0.7.3" + "websocket-driver": ">=0.5.1" } }, "ms": { @@ -14378,7 +14378,7 @@ "integrity": "sha512-pCpjxQgOByDHLlNqlnh/mNSAxIUkyBBuwwhTcV+enZGbDaClPvHdvm6uvOwZfFJkam7cGhBNbb4JxiP8UZkRvQ==", "dev": true, "requires": { - "ip": "1.1.5", + "ip": "^1.1.5", "smart-buffer": "4.0.2" } }, @@ -14388,8 +14388,8 @@ "integrity": "sha512-NT6syHhI9LmuEMSK6Kd2V7gNv5KFZoLE7V5udWmn0de+3Mkj3UMA/AJPLyeNUVmElCurSHtUdM3ETpR3z770Wg==", "dev": true, "requires": { - "agent-base": "4.2.1", - "socks": "2.3.2" + "agent-base": "~4.2.1", + "socks": "~2.3.2" }, "dependencies": { "agent-base": { @@ -14398,7 +14398,7 @@ "integrity": "sha512-JVwXMr9nHYTUXsBFKUqhJwvlcYU/blreOEUkhNR2eXZIvwd+c+o5V4MgDPKWnMS/56awN3TRzIP+KoPn+roQtg==", "dev": true, "requires": { - "es6-promisify": "5.0.0" + "es6-promisify": "^5.0.0" } } } @@ -14409,7 +14409,7 @@ "integrity": "sha1-RBttTTRnmPG05J6JIK37oOVD+a0=", "dev": true, "requires": { - "is-plain-obj": "1.1.0" + "is-plain-obj": "^1.0.0" } }, "source-list-map": { @@ -14451,8 +14451,8 @@ "integrity": "sha512-SHSKFHadjVA5oR4PPqhtAVdcBWwRYVd6g6cAXnIbRiIwc2EhPrTuKUBdSLvlEKyIP3GCf89fltvcZiP9MMFA1w==", "dev": true, "requires": { - "buffer-from": "1.1.1", - "source-map": "0.6.1" + "buffer-from": "^1.0.0", + "source-map": "^0.6.0" }, "dependencies": { "source-map": { @@ -14485,8 +14485,8 @@ "integrity": "sha512-lr2EZCctC2BNR7j7WzJ2FpDznxky1sjfxvvYEyzxNyb6lZXHODmEoJeFu4JupYlkfha1KZpJyoqiJ7pgA1qq8Q==", "dev": true, "requires": { - "spdx-expression-parse": "3.0.0", - "spdx-license-ids": "3.0.5" + "spdx-expression-parse": "^3.0.0", + "spdx-license-ids": "^3.0.0" } }, "spdx-exceptions": { @@ -14517,11 +14517,11 @@ "integrity": "sha512-HeZS3PBdMA+sZSu0qwpCxl3DeALD5ASx8pAX0jZdKXSpPWbQ6SYGnlg3BBmYLx5LtiZrmkAZfErCm2oECBcioA==", "dev": true, "requires": { - "debug": "4.1.1", - "handle-thing": "2.0.0", - "http-deceiver": "1.2.7", - "select-hose": "2.0.0", - "spdy-transport": "3.0.0" + "debug": "^4.1.0", + "handle-thing": "^2.0.0", + "http-deceiver": "^1.2.7", + "select-hose": "^2.0.0", + "spdy-transport": "^3.0.0" }, "dependencies": { "debug": { @@ -14530,7 +14530,7 @@ "integrity": "sha512-pYAIzeRo8J6KPEaJ0VWOh5Pzkbw/RetuzehGM7QRRX5he4fPHx2rdKMB256ehJCkX+XRQm16eZLqLNS8RSZXZw==", "dev": true, "requires": { - "ms": "2.1.2" + "ms": "^2.1.1" } }, "ms": { @@ -14576,9 +14576,9 @@ "integrity": "sha512-jItXPLmrSR8jmTRmRWJXCnGJsfy85mB3Wd/uINMXA65yrnFo0cPClFIUWzo2najVNSl+mx7/4W8ttlLWJe99pQ==", "dev": true, "requires": { - "inherits": "2.0.4", - "string_decoder": "1.1.1", - "util-deprecate": "1.0.2" + "inherits": "^2.0.3", + "string_decoder": "^1.1.1", + "util-deprecate": "^1.0.1" } } } @@ -14589,7 +14589,7 @@ "integrity": "sha512-qVIkJvbtS9j/UeZumbdfz0vg+QfG/zxonAjzefZrqzkr7xOncLVXkeGbTpzd1gjCBM4PmVNkWlkeTVhgskAGSQ==", "dev": true, "requires": { - "chalk": "2.4.2" + "chalk": "^2.0.1" } }, "split": { @@ -14621,15 +14621,15 @@ "integrity": "sha512-HXXqVUq7+pcKeLqqZj6mHFUMvXtOJt1uoUx09pFW6011inTMxqI8BA8PM95myrIyyKwdnzjdFjLiE6KBPVtJIg==", "dev": true, "requires": { - "asn1": "0.2.4", - "assert-plus": "1.0.0", - "bcrypt-pbkdf": "1.0.2", - "dashdash": "1.14.1", - "ecc-jsbn": "0.1.2", - "getpass": "0.1.7", - "jsbn": "0.1.1", - "safer-buffer": "2.1.2", - "tweetnacl": "0.14.5" + "asn1": "~0.2.3", + "assert-plus": "^1.0.0", + "bcrypt-pbkdf": "^1.0.0", + "dashdash": "^1.12.0", + "ecc-jsbn": "~0.1.1", + "getpass": "^0.1.1", + "jsbn": "~0.1.0", + "safer-buffer": "^2.0.2", + "tweetnacl": "~0.14.0" } }, "ssri": { @@ -14638,7 +14638,7 @@ "integrity": "sha512-3Wge10hNcT1Kur4PDFwEieXSCMCJs/7WvSACcrMYrNp+b8kDL1/0wJch5Ni2WrtwEa2IO8OsVfeKIciKCDx/QA==", "dev": true, "requires": { - "figgy-pudding": "3.5.1" + "figgy-pudding": "^3.5.1" } }, "stack-trace": { @@ -14658,7 +14658,7 @@ "integrity": "sha512-51Rr7dXkyFUKNmhY/vqZWK+EvdsfFSRiQVtgHTFlAdNIYaDD7bVh21yBHXaNWAvTD+w+QSjxHg7/v6Tz4veExA==", "requires": { "source-map": "0.5.6", - "stackframe": "1.1.0" + "stackframe": "^1.1.0" } }, "static-extend": { @@ -14725,9 +14725,9 @@ "integrity": "sha1-N5TzeMWLNC6n27sjCVEJxLO2IpE=", "dev": true, "requires": { - "graceful-fs": "4.2.2", - "jsonfile": "3.0.1", - "universalify": "0.1.2" + "graceful-fs": "^4.1.2", + "jsonfile": "^3.0.0", + "universalify": "^0.1.0" } }, "jsonfile": { @@ -14736,7 +14736,7 @@ "integrity": "sha1-pezG9l9T9mLEQVx2daAzHQmS7GY=", "dev": true, "requires": { - "graceful-fs": "4.2.2" + "graceful-fs": "^4.1.6" } }, "mkdirp": { @@ -14873,8 +14873,8 @@ "integrity": "sha512-FJ6b7EgdKxxbDxc79cOlok6Afd++TTs5szo+zJTUyow3ycrRfJVE2pq3vcN53XexvKZu/DJMDfeI/qMiZTrjTw==", "dev": true, "requires": { - "define-properties": "1.1.3", - "function-bind": "1.1.1" + "define-properties": "^1.1.3", + "function-bind": "^1.1.1" } }, "string.prototype.trimright": { @@ -14883,8 +14883,8 @@ "integrity": "sha512-fXZTSV55dNBwv16uw+hh5jkghxSnc5oHq+5K/gXgizHwAvMetdAJlHqqoFC1FSDVPYWLkAKl2cxpUT41sV7nSg==", "dev": true, "requires": { - "define-properties": "1.1.3", - "function-bind": "1.1.1" + "define-properties": "^1.1.3", + "function-bind": "^1.1.1" } }, "string_decoder": { @@ -14936,8 +14936,8 @@ "integrity": "sha512-B0dOCFwv7/eY31a5PCieNwMgMhVGFe9w+rh7s/Bx8kfFkrth9zfTZquoYvdw8URgiqxObQKcpW51Ugz1HjfdZw==", "dev": true, "requires": { - "loader-utils": "1.2.3", - "schema-utils": "2.4.1" + "loader-utils": "^1.2.3", + "schema-utils": "^2.0.1" }, "dependencies": { "ajv": { @@ -14946,10 +14946,10 @@ "integrity": "sha512-TXtUUEYHuaTEbLZWIKUr5pmBuhDLy+8KYtPYdcV8qC+pOZL+NKqYwvWSRrVXHn+ZmRRAu8vJTAznH7Oag6RVRw==", "dev": true, "requires": { - "fast-deep-equal": "2.0.1", - "fast-json-stable-stringify": "2.0.0", - "json-schema-traverse": "0.4.1", - "uri-js": "4.2.2" + "fast-deep-equal": "^2.0.1", + "fast-json-stable-stringify": "^2.0.0", + "json-schema-traverse": "^0.4.1", + "uri-js": "^4.2.2" } }, "schema-utils": { @@ -14958,8 +14958,8 @@ "integrity": "sha512-RqYLpkPZX5Oc3fw/kHHHyP56fg5Y+XBpIpV8nCg0znIALfq3OH+Ea9Hfeac9BAMwG5IICltiZ0vxFvJQONfA5w==", "dev": true, "requires": { - "ajv": "6.10.2", - "ajv-keywords": "3.4.1" + "ajv": "^6.10.2", + "ajv-keywords": "^3.4.1" } } } @@ -15020,7 +15020,7 @@ "integrity": "sha512-qe1jfm1Mg7Nq/NSh6XE24gPXROEVsWHxC1LIx//XNlD9iw7YZQGjZNjYN7xGaEG6iKdA8EtNFW6R0gjnVXp+wQ==", "dev": true, "requires": { - "has-flag": "3.0.0" + "has-flag": "^3.0.0" } }, "sver-compat": { @@ -15149,13 +15149,13 @@ "integrity": "sha512-w2VwSrBoHa5BsSyH+KxEqeQBAllHhccyMFVHtGtdMpF4W7IRWfZjFiQceJPChOeTsSDVUpER2T8FA93pr0L+QA==", "dev": true, "requires": { - "chownr": "1.1.3", - "fs-minipass": "1.2.7", - "minipass": "2.9.0", - "minizlib": "1.3.3", - "mkdirp": "0.5.1", - "safe-buffer": "5.1.2", - "yallist": "3.1.1" + "chownr": "^1.1.1", + "fs-minipass": "^1.2.5", + "minipass": "^2.8.6", + "minizlib": "^1.2.1", + "mkdirp": "^0.5.0", + "safe-buffer": "^5.1.2", + "yallist": "^3.0.3" } }, "teeny-request": { @@ -15267,9 +15267,9 @@ "integrity": "sha1-6L0O/uWPz/b4+UUQoKVUu/ojVEk=", "dev": true, "requires": { - "lru-cache": "4.1.5", - "shebang-command": "1.2.0", - "which": "1.3.1" + "lru-cache": "^4.0.1", + "shebang-command": "^1.2.0", + "which": "^1.2.9" } }, "execa": { @@ -15278,13 +15278,13 @@ "integrity": "sha1-lEvs00zEHuMqY6n68nrVpl/Fl3c=", "dev": true, "requires": { - "cross-spawn": "5.1.0", - "get-stream": "3.0.0", - "is-stream": "1.1.0", - "npm-run-path": "2.0.2", - "p-finally": "1.0.0", - "signal-exit": "3.0.2", - "strip-eof": "1.0.0" + "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" } }, "get-stream": { @@ -15299,8 +15299,8 @@ "integrity": "sha512-sWZlbEP2OsHNkXrMl5GYk/jKk70MBng6UU4YI/qGDYbgf6YbP4EvmqISbXCoJiRKs+1bSpFHVgQxvJ17F2li5g==", "dev": true, "requires": { - "pseudomap": "1.0.2", - "yallist": "2.1.2" + "pseudomap": "^1.0.2", + "yallist": "^2.1.2" } }, "yallist": { @@ -15317,9 +15317,9 @@ "integrity": "sha512-+ZwXJvdSwbd60jG0Illav0F06GDJF0R4ydZ21Q3wGAFKoBGyJGo34F63vzJHgvYxc1ukOtIjvwEvl9MkjzM6Pg==", "dev": true, "requires": { - "commander": "2.20.0", - "source-map": "0.6.1", - "source-map-support": "0.5.13" + "commander": "^2.20.0", + "source-map": "~0.6.1", + "source-map-support": "~0.5.12" }, "dependencies": { "source-map": { @@ -15336,15 +15336,15 @@ "integrity": "sha512-ZXmmfiwtCLfz8WKZyYUuuHf3dMYEjg8NrjHMb0JqHVHVOSkzp3cW2/XG1fP3tRhqEqSzMwzzRQGtAPbs4Cncxg==", "dev": true, "requires": { - "cacache": "12.0.2", - "find-cache-dir": "2.1.0", - "is-wsl": "1.1.0", - "schema-utils": "1.0.0", - "serialize-javascript": "1.9.1", - "source-map": "0.6.1", - "terser": "4.1.4", - "webpack-sources": "1.4.3", - "worker-farm": "1.7.0" + "cacache": "^12.0.2", + "find-cache-dir": "^2.1.0", + "is-wsl": "^1.1.0", + "schema-utils": "^1.0.0", + "serialize-javascript": "^1.7.0", + "source-map": "^0.6.1", + "terser": "^4.1.2", + "webpack-sources": "^1.4.0", + "worker-farm": "^1.7.0" }, "dependencies": { "find-cache-dir": { @@ -15353,9 +15353,9 @@ "integrity": "sha512-Tq6PixE0w/VMFfCgbONnkiQIVol/JJL7nRMi20fqzA4NRs9AfeqMGeRdPi3wIhYkxjeBaWh2rxwapn5Tu3IqOQ==", "dev": true, "requires": { - "commondir": "1.0.1", - "make-dir": "2.1.0", - "pkg-dir": "3.0.0" + "commondir": "^1.0.1", + "make-dir": "^2.0.0", + "pkg-dir": "^3.0.0" } }, "source-map": { @@ -15384,8 +15384,8 @@ "integrity": "sha512-/mrRod8xqpA+IHSLyGCQ2s8SPHiCDEeQJSep1jqLYeEUClOFG2Qsh+4FU6G9VeqpZnGW/Su8LQGc4YKni5rYSQ==", "dev": true, "requires": { - "readable-stream": "2.3.6", - "xtend": "4.0.2" + "readable-stream": "~2.3.6", + "xtend": "~4.0.1" } }, "through2-filter": { @@ -15416,7 +15416,7 @@ "integrity": "sha512-60aV6sgJ5YEbzUdn9c8kYGIqOubPoUdqQCul3SBAsRCZ40s6Y5cMcrW4dt3/k/EsbLVJNl9n6Vz3fTc+k2GeKQ==", "dev": true, "requires": { - "setimmediate": "1.0.5" + "setimmediate": "^1.0.4" } }, "tmp": { @@ -15549,11 +15549,11 @@ "resolved": "https://registry.npmjs.org/ts-loader/-/ts-loader-5.3.1.tgz", "integrity": "sha512-fDDgpBH3SR8xlt2MasLdz3Yy611PQ/UY/KGyo7TgXhTRU/6sS8uGG0nJYnU1OdFBNKcoYbId1UTNaAOUn+i41g==", "requires": { - "chalk": "2.4.2", - "enhanced-resolve": "4.1.0", - "loader-utils": "1.2.3", - "micromatch": "3.1.10", - "semver": "5.6.0" + "chalk": "^2.3.0", + "enhanced-resolve": "^4.0.0", + "loader-utils": "^1.0.2", + "micromatch": "^3.1.4", + "semver": "^5.0.1" } }, "ts-md5": { @@ -15567,11 +15567,11 @@ "integrity": "sha512-5LpRN+mTiCs7lI5EtbXmF/HfMeCjzt7DH9CZwtkr6SywStrNQC723wG+aOWFiLNn7zT3kD/RnFqi3ZUfr4l5Qw==", "dev": true, "requires": { - "arg": "4.1.1", - "diff": "4.0.1", - "make-error": "1.3.5", - "source-map-support": "0.5.13", - "yn": "3.1.1" + "arg": "^4.1.0", + "diff": "^4.0.1", + "make-error": "^1.1.1", + "source-map-support": "^0.5.6", + "yn": "^3.0.0" } }, "tsconfig-paths": { @@ -15579,10 +15579,10 @@ "resolved": "https://registry.npmjs.org/tsconfig-paths/-/tsconfig-paths-3.9.0.tgz", "integrity": "sha512-dRcuzokWhajtZWkQsDVKbWyY+jgcLC5sqJhg2PSgf4ZkH2aHPvaOY8YWGhmjb68b5qqTfasSsDO9k7RUiEmZAw==", "requires": { - "@types/json5": "0.0.29", - "json5": "1.0.1", - "minimist": "1.2.0", - "strip-bom": "3.0.0" + "@types/json5": "^0.0.29", + "json5": "^1.0.1", + "minimist": "^1.2.0", + "strip-bom": "^3.0.0" }, "dependencies": { "strip-bom": { @@ -15597,9 +15597,9 @@ "resolved": "https://registry.npmjs.org/tsconfig-paths-webpack-plugin/-/tsconfig-paths-webpack-plugin-3.2.0.tgz", "integrity": "sha512-S/gOOPOkV8rIL4LurZ1vUdYCVgo15iX9ZMJ6wx6w2OgcpT/G4wMyHB6WM+xheSqGMrWKuxFul+aXpCju3wmj/g==", "requires": { - "chalk": "2.4.2", - "enhanced-resolve": "4.1.0", - "tsconfig-paths": "3.9.0" + "chalk": "^2.3.0", + "enhanced-resolve": "^4.0.0", + "tsconfig-paths": "^3.4.0" } }, "tsickle": { @@ -15608,9 +15608,9 @@ "integrity": "sha512-ufUZqLUNqh+kOfr52N/hJ5JbiDO32/CO7ZCteZBX9HA2kiejwEgDaJeJe1GAj2TIu683IgTA/LPKvlns6Liw0w==", "dev": true, "requires": { - "minimist": "1.2.0", - "mkdirp": "0.5.1", - "source-map": "0.7.3" + "minimist": "^1.2.0", + "mkdirp": "^0.5.1", + "source-map": "^0.7.3" }, "dependencies": { "source-map": { @@ -15632,19 +15632,19 @@ "integrity": "sha512-2vqIvkMHbnx8acMogAERQ/IuINOq6DFqgF8/VDvhEkBqQh/x6SP0Y+OHnKth9/ZcHQSroOZwUQSN18v8KKF0/g==", "dev": true, "requires": { - "@babel/code-frame": "7.5.5", - "builtin-modules": "1.1.1", - "chalk": "2.4.2", - "commander": "2.20.0", - "diff": "4.0.1", - "glob": "7.1.3", - "js-yaml": "3.13.1", - "minimatch": "3.0.4", - "mkdirp": "0.5.1", - "resolve": "1.12.0", - "semver": "5.6.0", - "tslib": "1.10.0", - "tsutils": "2.29.0" + "@babel/code-frame": "^7.0.0", + "builtin-modules": "^1.1.1", + "chalk": "^2.3.0", + "commander": "^2.12.1", + "diff": "^4.0.1", + "glob": "^7.1.1", + "js-yaml": "^3.13.1", + "minimatch": "^3.0.4", + "mkdirp": "^0.5.1", + "resolve": "^1.3.2", + "semver": "^5.3.0", + "tslib": "^1.8.0", + "tsutils": "^2.29.0" }, "dependencies": { "builtin-modules": { @@ -15661,7 +15661,7 @@ "integrity": "sha512-g5JVHCIJwzfISaXpXE1qvNalca5Jwob6FjI4AoPlqMusJ6ftFE7IkkFoMhVLRgK+4Kx3gkzb8UZK5t5yTTvEmA==", "dev": true, "requires": { - "tslib": "1.10.0" + "tslib": "^1.8.1" } }, "tty-browserify": { @@ -15713,7 +15713,7 @@ "dev": true, "requires": { "media-typer": "0.3.0", - "mime-types": "2.1.24" + "mime-types": "~2.1.24" } }, "typedarray": { @@ -15735,8 +15735,8 @@ "dev": true, "optional": true, "requires": { - "commander": "2.20.0", - "source-map": "0.6.1" + "commander": "~2.20.0", + "source-map": "~0.6.1" }, "dependencies": { "source-map": { @@ -15799,8 +15799,8 @@ "resolved": "https://registry.npmjs.org/unicode-match-property-ecmascript/-/unicode-match-property-ecmascript-1.0.4.tgz", "integrity": "sha512-L4Qoh15vTfntsn4P1zqnHulG0LdXgjSO035fEpdtp6YxXhMT51Q6vgM5lYdG/5X3MjS+k/Y9Xw4SFCY9IkR0rg==", "requires": { - "unicode-canonical-property-names-ecmascript": "1.0.4", - "unicode-property-aliases-ecmascript": "1.0.5" + "unicode-canonical-property-names-ecmascript": "^1.0.4", + "unicode-property-aliases-ecmascript": "^1.0.4" } }, "unicode-match-property-value-ecmascript": { @@ -15839,7 +15839,7 @@ "integrity": "sha512-zoWr9ObaxALD3DOPfjPSqxt4fnZiWblxHIgeWqW8x7UqDzEtHEQLzji2cuJYQFCU6KmoJikOYAZlrTHHebjx2w==", "dev": true, "requires": { - "imurmurhash": "0.1.4" + "imurmurhash": "^0.1.4" } }, "unique-stream": { @@ -15867,9 +15867,9 @@ "integrity": "sha512-gE91dtMvNkjO+kWsPstHRtSwHXz0l2axqptGYp5ceg4MsuurloM0PU3pdOfpb5zBXUvyjT4PwhWK2m39uczZuw==", "dev": true, "requires": { - "debug": "3.2.6", - "request": "2.88.0", - "uuid": "3.3.3" + "debug": "^3.0.0", + "request": "^2.88.0", + "uuid": "^3.0.0" }, "dependencies": { "debug": { @@ -15878,7 +15878,7 @@ "integrity": "sha512-mel+jf7nrtEl5Pn1Qx46zARXKDpBbvzezse7p7LqINmdoIk8PYP5SySaxEmYv6TZ0JyEKA1hsCId6DIhgITtWQ==", "dev": true, "requires": { - "ms": "2.1.2" + "ms": "^2.1.1" } }, "ms": { @@ -15948,18 +15948,18 @@ "integrity": "sha512-grrmrB6Zb8DUiyDIaeRTBCkgISYUgETNe7NglEbVsrLWXeESnlCSP50WfRSj/GmzMPl6Uchj24S/p80nP/ZQrQ==", "dev": true, "requires": { - "boxen": "3.2.0", - "chalk": "2.4.2", - "configstore": "4.0.0", - "has-yarn": "2.1.0", - "import-lazy": "2.1.0", - "is-ci": "2.0.0", - "is-installed-globally": "0.1.0", - "is-npm": "3.0.0", - "is-yarn-global": "0.3.0", - "latest-version": "5.1.0", - "semver-diff": "2.1.0", - "xdg-basedir": "3.0.0" + "boxen": "^3.0.0", + "chalk": "^2.0.1", + "configstore": "^4.0.0", + "has-yarn": "^2.1.0", + "import-lazy": "^2.1.0", + "is-ci": "^2.0.0", + "is-installed-globally": "^0.1.0", + "is-npm": "^3.0.0", + "is-yarn-global": "^0.3.0", + "latest-version": "^5.0.0", + "semver-diff": "^2.0.0", + "xdg-basedir": "^3.0.0" } }, "uri-js": { @@ -15999,8 +15999,8 @@ "integrity": "sha512-d3uaVyzDB9tQoSXFvuSUNFibTd9zxd2bkVrDRvF5TmvWWQwqE4lgYJ5m+x1DbecWkw+LK4RNl2CU1hHuOKPVlg==", "dev": true, "requires": { - "querystringify": "2.1.1", - "requires-port": "1.0.0" + "querystringify": "^2.1.1", + "requires-port": "^1.0.0" } }, "url-parse-lax": { @@ -16009,7 +16009,7 @@ "integrity": "sha1-FrXK/Afb42dsGxmZF3gj1lA6yww=", "dev": true, "requires": { - "prepend-http": "2.0.0" + "prepend-http": "^2.0.0" }, "dependencies": { "prepend-http": { @@ -16056,8 +16056,8 @@ "integrity": "sha512-sWZlbEP2OsHNkXrMl5GYk/jKk70MBng6UU4YI/qGDYbgf6YbP4EvmqISbXCoJiRKs+1bSpFHVgQxvJ17F2li5g==", "dev": true, "requires": { - "pseudomap": "1.0.2", - "yallist": "2.1.2" + "pseudomap": "^1.0.2", + "yallist": "^2.1.2" } }, "yallist": { @@ -16096,7 +16096,7 @@ "integrity": "sha1-PCI2R2xNMsX/PEcAKt18E7moKlM=", "dev": true, "requires": { - "object.getownpropertydescriptors": "2.0.3" + "object.getownpropertydescriptors": "^2.0.3" } }, "util.promisify": { @@ -16127,7 +16127,7 @@ "integrity": "sha512-amh9CCg3ZxkzQ48Mhcb8iX7xpAfYJgePHxWMQCBWECpOSqJUXgY26ncA61UTV0BkPqfhcy6mzwCIoP4ygxpW8w==", "dev": true, "requires": { - "homedir-polyfill": "1.0.3" + "homedir-polyfill": "^1.0.1" } }, "validate-npm-package-license": { @@ -16136,8 +16136,8 @@ "integrity": "sha512-DpKm2Ui/xN7/HQKCtpZxoRWBhZ9Z0kqtygG8XCgNQ8ZlDnxuQmWhj566j8fN4Cu3/JmbhsDo7fcAJq4s9h27Ew==", "dev": true, "requires": { - "spdx-correct": "3.1.0", - "spdx-expression-parse": "3.0.0" + "spdx-correct": "^3.0.0", + "spdx-expression-parse": "^3.0.0" } }, "validate-npm-package-name": { @@ -16178,12 +16178,12 @@ "integrity": "sha512-MBH+yP0kC/GQ5GwBqrTPTzEfiiLjta7hTtvQtbxBgTeSXsmKQRQecjibMbxIXzVT3Y9KJK+drOz1/k+vsu8Nkg==", "dev": true, "requires": { - "clone": "2.1.2", - "clone-buffer": "1.0.0", - "clone-stats": "1.0.0", - "cloneable-readable": "1.1.3", - "remove-trailing-separator": "1.1.0", - "replace-ext": "1.0.0" + "clone": "^2.1.1", + "clone-buffer": "^1.0.0", + "clone-stats": "^1.0.0", + "cloneable-readable": "^1.0.0", + "remove-trailing-separator": "^1.0.1", + "replace-ext": "^1.0.0" } }, "vinyl-fs": { @@ -16297,25 +16297,25 @@ "@webassemblyjs/helper-module-context": "1.8.5", "@webassemblyjs/wasm-edit": "1.8.5", "@webassemblyjs/wasm-parser": "1.8.5", - "acorn": "6.3.0", - "ajv": "6.10.2", - "ajv-keywords": "3.4.1", - "chrome-trace-event": "1.0.2", - "enhanced-resolve": "4.1.0", - "eslint-scope": "4.0.3", - "json-parse-better-errors": "1.0.2", - "loader-runner": "2.4.0", - "loader-utils": "1.2.3", - "memory-fs": "0.4.1", - "micromatch": "3.1.10", - "mkdirp": "0.5.1", - "neo-async": "2.6.1", - "node-libs-browser": "2.2.1", - "schema-utils": "1.0.0", - "tapable": "1.1.3", - "terser-webpack-plugin": "1.4.1", - "watchpack": "1.6.0", - "webpack-sources": "1.4.3" + "acorn": "^6.2.1", + "ajv": "^6.10.2", + "ajv-keywords": "^3.4.1", + "chrome-trace-event": "^1.0.2", + "enhanced-resolve": "^4.1.0", + "eslint-scope": "^4.0.3", + "json-parse-better-errors": "^1.0.2", + "loader-runner": "^2.4.0", + "loader-utils": "^1.2.3", + "memory-fs": "^0.4.1", + "micromatch": "^3.1.10", + "mkdirp": "^0.5.1", + "neo-async": "^2.6.1", + "node-libs-browser": "^2.2.1", + "schema-utils": "^1.0.0", + "tapable": "^1.1.3", + "terser-webpack-plugin": "^1.4.1", + "watchpack": "^1.6.0", + "webpack-sources": "^1.4.1" }, "dependencies": { "acorn": { @@ -16330,10 +16330,10 @@ "integrity": "sha512-TXtUUEYHuaTEbLZWIKUr5pmBuhDLy+8KYtPYdcV8qC+pOZL+NKqYwvWSRrVXHn+ZmRRAu8vJTAznH7Oag6RVRw==", "dev": true, "requires": { - "fast-deep-equal": "2.0.1", - "fast-json-stable-stringify": "2.0.0", - "json-schema-traverse": "0.4.1", - "uri-js": "4.2.2" + "fast-deep-equal": "^2.0.1", + "fast-json-stable-stringify": "^2.0.0", + "json-schema-traverse": "^0.4.1", + "uri-js": "^4.2.2" } } } @@ -16371,10 +16371,10 @@ "integrity": "sha512-qvDesR1QZRIAZHOE3iQ4CXLZZSQ1lAUsSpnQmlB1PBfoN/xdRjmge3Dok0W4IdaVLJOGJy3sGI4sZHwjRU0PCA==", "dev": true, "requires": { - "memory-fs": "0.4.1", - "mime": "2.4.4", - "range-parser": "1.2.1", - "webpack-log": "2.0.0" + "memory-fs": "^0.4.1", + "mime": "^2.4.2", + "range-parser": "^1.2.1", + "webpack-log": "^2.0.0" }, "dependencies": { "mime": { @@ -16392,37 +16392,37 @@ "dev": true, "requires": { "ansi-html": "0.0.7", - "bonjour": "3.5.0", - "chokidar": "2.1.8", - "compression": "1.7.4", - "connect-history-api-fallback": "1.6.0", - "debug": "4.1.1", - "del": "4.1.1", - "express": "4.17.1", - "html-entities": "1.2.1", - "http-proxy-middleware": "0.19.1", - "import-local": "2.0.0", - "internal-ip": "4.3.0", - "ip": "1.1.5", - "is-absolute-url": "3.0.3", - "killable": "1.0.1", - "loglevel": "1.6.4", - "opn": "5.5.0", - "p-retry": "3.0.1", - "portfinder": "1.0.24", - "schema-utils": "1.0.0", - "selfsigned": "1.10.7", - "semver": "6.3.0", - "serve-index": "1.9.1", + "bonjour": "^3.5.0", + "chokidar": "^2.1.6", + "compression": "^1.7.4", + "connect-history-api-fallback": "^1.6.0", + "debug": "^4.1.1", + "del": "^4.1.1", + "express": "^4.17.1", + "html-entities": "^1.2.1", + "http-proxy-middleware": "^0.19.1", + "import-local": "^2.0.0", + "internal-ip": "^4.3.0", + "ip": "^1.1.5", + "is-absolute-url": "^3.0.0", + "killable": "^1.0.1", + "loglevel": "^1.6.3", + "opn": "^5.5.0", + "p-retry": "^3.0.1", + "portfinder": "^1.0.21", + "schema-utils": "^1.0.0", + "selfsigned": "^1.10.4", + "semver": "^6.3.0", + "serve-index": "^1.9.1", "sockjs": "0.3.19", "sockjs-client": "1.3.0", - "spdy": "4.0.1", - "strip-ansi": "3.0.1", - "supports-color": "6.1.0", - "url": "0.11.0", - "webpack-dev-middleware": "3.7.0", - "webpack-log": "2.0.0", - "ws": "6.2.1", + "spdy": "^4.0.1", + "strip-ansi": "^3.0.1", + "supports-color": "^6.1.0", + "url": "^0.11.0", + "webpack-dev-middleware": "^3.7.0", + "webpack-log": "^2.0.0", + "ws": "^6.2.1", "yargs": "12.0.5" }, "dependencies": { @@ -16438,18 +16438,18 @@ "integrity": "sha512-ZmZUazfOzf0Nve7duiCKD23PFSCs4JPoYyccjUFF3aQkQadqBhfzhjkwBH2mNOG9cTBwhamM37EIsIkZw3nRgg==", "dev": true, "requires": { - "anymatch": "2.0.0", - "async-each": "1.0.3", - "braces": "2.3.2", - "fsevents": "1.2.9", - "glob-parent": "3.1.0", - "inherits": "2.0.4", - "is-binary-path": "1.0.1", - "is-glob": "4.0.1", - "normalize-path": "3.0.0", - "path-is-absolute": "1.0.1", - "readdirp": "2.2.1", - "upath": "1.2.0" + "anymatch": "^2.0.0", + "async-each": "^1.0.1", + "braces": "^2.3.2", + "fsevents": "^1.2.7", + "glob-parent": "^3.1.0", + "inherits": "^2.0.3", + "is-binary-path": "^1.0.0", + "is-glob": "^4.0.0", + "normalize-path": "^3.0.0", + "path-is-absolute": "^1.0.0", + "readdirp": "^2.2.1", + "upath": "^1.1.1" } }, "cliui": { @@ -16458,9 +16458,9 @@ "integrity": "sha512-4FG+RSG9DL7uEwRUZXZn3SS34DiDPfzP0VOiEwtUWlE+AR2EIg+hSyvrIgUUfhdgR/UkAeW2QHgeP+hWrXs7jQ==", "dev": true, "requires": { - "string-width": "2.1.1", - "strip-ansi": "4.0.0", - "wrap-ansi": "2.1.0" + "string-width": "^2.1.1", + "strip-ansi": "^4.0.0", + "wrap-ansi": "^2.0.0" }, "dependencies": { "strip-ansi": { @@ -16469,7 +16469,7 @@ "integrity": "sha1-qEeQIusaw2iocTibY1JixQXuNo8=", "dev": true, "requires": { - "ansi-regex": "3.0.0" + "ansi-regex": "^3.0.0" } } } @@ -16480,7 +16480,7 @@ "integrity": "sha512-pYAIzeRo8J6KPEaJ0VWOh5Pzkbw/RetuzehGM7QRRX5he4fPHx2rdKMB256ehJCkX+XRQm16eZLqLNS8RSZXZw==", "dev": true, "requires": { - "ms": "2.1.2" + "ms": "^2.1.1" } }, "invert-kv": { @@ -16501,7 +16501,7 @@ "integrity": "sha512-avPEb8P8EGnwXKClwsNUgryVjllcRqtMYa49NTsbQagYuT1DcXnl1915oxWjoyGrXR6zH/Y0Zc96xWsPcoDKeA==", "dev": true, "requires": { - "invert-kv": "2.0.0" + "invert-kv": "^2.0.0" } }, "mem": { @@ -16510,9 +16510,9 @@ "integrity": "sha512-qX2bG48pTqYRVmDB37rn/6PT7LcR8T7oAX3bf99u1Tt1nzxYfxkgqDwUwolPlXweM0XzBOBFzSx4kfp7KP1s/w==", "dev": true, "requires": { - "map-age-cleaner": "0.1.3", - "mimic-fn": "2.1.0", - "p-is-promise": "2.1.0" + "map-age-cleaner": "^0.1.1", + "mimic-fn": "^2.0.0", + "p-is-promise": "^2.0.0" } }, "ms": { @@ -16533,9 +16533,9 @@ "integrity": "sha512-Z8l3R4wYWM40/52Z+S265okfFj8Kt2cC2MKY+xNi3kFs+XGI7WXu/I309QQQYbRW4ijiZ+yxs9pqEhJh0DqW3Q==", "dev": true, "requires": { - "execa": "1.0.0", - "lcid": "2.0.0", - "mem": "4.3.0" + "execa": "^1.0.0", + "lcid": "^2.0.0", + "mem": "^4.0.0" } }, "semver": { @@ -16550,8 +16550,8 @@ "integrity": "sha512-nOqH59deCq9SRHlxq1Aw85Jnt4w6KvLKqWVik6oA9ZklXLNIOlqg4F2yrT1MVaTjAqvVwdfeZ7w7aCvJD7ugkw==", "dev": true, "requires": { - "is-fullwidth-code-point": "2.0.0", - "strip-ansi": "4.0.0" + "is-fullwidth-code-point": "^2.0.0", + "strip-ansi": "^4.0.0" }, "dependencies": { "strip-ansi": { @@ -16560,7 +16560,7 @@ "integrity": "sha1-qEeQIusaw2iocTibY1JixQXuNo8=", "dev": true, "requires": { - "ansi-regex": "3.0.0" + "ansi-regex": "^3.0.0" } } } @@ -16577,18 +16577,18 @@ "integrity": "sha512-Lhz8TLaYnxq/2ObqHDql8dX8CJi97oHxrjUcYtzKbbykPtVW9WB+poxI+NM2UIzsMgNCZTIf0AQwsjK5yMAqZw==", "dev": true, "requires": { - "cliui": "4.1.0", - "decamelize": "1.2.0", - "find-up": "3.0.0", - "get-caller-file": "1.0.3", - "os-locale": "3.1.0", - "require-directory": "2.1.1", - "require-main-filename": "1.0.1", - "set-blocking": "2.0.0", - "string-width": "2.1.1", - "which-module": "2.0.0", - "y18n": "4.0.0", - "yargs-parser": "11.1.1" + "cliui": "^4.0.0", + "decamelize": "^1.2.0", + "find-up": "^3.0.0", + "get-caller-file": "^1.0.1", + "os-locale": "^3.0.0", + "require-directory": "^2.1.1", + "require-main-filename": "^1.0.1", + "set-blocking": "^2.0.0", + "string-width": "^2.0.0", + "which-module": "^2.0.0", + "y18n": "^3.2.1 || ^4.0.0", + "yargs-parser": "^11.1.1" } }, "yargs-parser": { @@ -16597,8 +16597,8 @@ "integrity": "sha512-C6kB/WJDiaxONLJQnF8ccx9SEeoTTLek8RVbaOIsrAUS8VrBEXfmeSnCZxygc+XC2sNMBIwOOnfcxiynjHsVSQ==", "dev": true, "requires": { - "camelcase": "5.3.1", - "decamelize": "1.2.0" + "camelcase": "^5.0.0", + "decamelize": "^1.2.0" } } } @@ -16633,8 +16633,8 @@ "integrity": "sha512-lgTS3Xhv1lCOKo7SA5TjKXMjpSM4sBjNV5+q2bqesbSPs5FjGmU6jjtBSkX9b4qW87vDIsCIlUPOEhbZrMdjeQ==", "dev": true, "requires": { - "source-list-map": "2.0.1", - "source-map": "0.6.1" + "source-list-map": "^2.0.0", + "source-map": "~0.6.1" }, "dependencies": { "source-map": { @@ -16660,9 +16660,9 @@ "integrity": "sha512-bpxWlvbbB459Mlipc5GBzzZwhoZgGEZLuqPaR0INBGnPAY1vdBX6hPnoFXiw+3yWxDuHyQjO2oXTMyS8A5haFg==", "dev": true, "requires": { - "http-parser-js": "0.4.10", - "safe-buffer": "5.1.2", - "websocket-extensions": "0.1.3" + "http-parser-js": ">=0.4.0 <0.4.11", + "safe-buffer": ">=5.1.0", + "websocket-extensions": ">=0.1.1" } }, "websocket-extensions": { @@ -16746,7 +16746,7 @@ "integrity": "sha512-rvw3QTZc8lAxyVrqcSGVm5yP/IJ2UcB3U0graE3LCFoZ0Yn2x4EoVSqJKdB/T5M+FLcRPjz4TDacRf3OCfNUzw==", "dev": true, "requires": { - "errno": "0.1.7" + "errno": "~0.1.7" } }, "worker-plugin": { @@ -16755,7 +16755,7 @@ "integrity": "sha512-W5nRkw7+HlbsEt3qRP6MczwDDISjiRj2GYt9+bpe8A2La00TmJdwzG5bpdMXhRt1qcWmwAvl1TiKaHRa+XDS9Q==", "dev": true, "requires": { - "loader-utils": "1.2.3" + "loader-utils": "^1.1.0" } }, "wrap-ansi": { @@ -16789,9 +16789,9 @@ "integrity": "sha512-GaETH5wwsX+GcnzhPgKcKjJ6M2Cq3/iZp1WyY/X1CSqrW+jVNM9Y7D8EC2sM4ZG/V8wZlSniJnCKWPmBYAucRQ==", "dev": true, "requires": { - "graceful-fs": "4.2.2", - "imurmurhash": "0.1.4", - "signal-exit": "3.0.2" + "graceful-fs": "^4.1.11", + "imurmurhash": "^0.1.4", + "signal-exit": "^3.0.2" } }, "ws": { @@ -16800,7 +16800,7 @@ "integrity": "sha512-GIyAXC2cB7LjvpgMt9EKS2ldqr0MTrORaleiOno6TweZ6r3TKtoFQWay/2PceJ3RuBasOHzXNn5Lrw1X0bEjqA==", "dev": true, "requires": { - "async-limiter": "1.0.1" + "async-limiter": "~1.0.0" } }, "xdg-basedir": { @@ -16893,19 +16893,19 @@ "integrity": "sha1-a6MY6xaWFyf10oT46gA+jWFU0Mg=", "dev": true, "requires": { - "camelcase": "3.0.0", - "cliui": "3.2.0", - "decamelize": "1.2.0", - "get-caller-file": "1.0.3", - "os-locale": "1.4.0", - "read-pkg-up": "1.0.1", - "require-directory": "2.1.1", - "require-main-filename": "1.0.1", - "set-blocking": "2.0.0", - "string-width": "1.0.2", - "which-module": "1.0.0", - "y18n": "3.2.1", - "yargs-parser": "5.0.0" + "camelcase": "^3.0.0", + "cliui": "^3.2.0", + "decamelize": "^1.1.1", + "get-caller-file": "^1.0.1", + "os-locale": "^1.4.0", + "read-pkg-up": "^1.0.1", + "require-directory": "^2.1.1", + "require-main-filename": "^1.0.1", + "set-blocking": "^2.0.0", + "string-width": "^1.0.2", + "which-module": "^1.0.0", + "y18n": "^3.2.1", + "yargs-parser": "^5.0.0" }, "dependencies": { "camelcase": { @@ -16945,7 +16945,7 @@ "integrity": "sha512-phENi2PLiHnHb6QBVot+dJnaAZ0xosj7p3fWl+znIjBDlnMI2PsZCJZ306BPTFOaHf5qdDEI8x5qFrSOBN5vrw==", "dev": true, "requires": { - "buffer-crc32": "0.2.13" + "buffer-crc32": "~0.2.3" } }, "yeast": { diff --git a/package.json b/package.json index 32c0f16a1e..1b2e633add 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "stratos", - "version": "3.2.0", + "version": "3.2.1", "description": "Stratos Console", "main": "index.js", "scripts": { From 4ede9773000f352045bc880474087dfd8e02935b Mon Sep 17 00:00:00 2001 From: Richard Cox Date: Fri, 29 May 2020 10:07:07 +0100 Subject: [PATCH 10/40] First pass at changelog (additional SSO fix required) --- CHANGELOG.md | 11 +++++++++++ 1 file changed, 11 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 4922e56735..cb26710e89 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,5 +1,16 @@ # Change Log +## 3.2.1 + +[Full Changelog](https://github.com/cloudfoundry/stratos/compare/3.2.0...3.2.1) + +This release contains a number of fixes: + +**Fixes:** + +- User Button Misaligned [\#4316](https://github.com/cloudfoundry/stratos/issues/4316) + + ## 3.2.0 [Full Changelog](https://github.com/cloudfoundry/stratos/compare/3.1.0...3.2.0) From e970d88f002017b826019dfc248e966c79f98c49 Mon Sep 17 00:00:00 2001 From: Richard Cox Date: Fri, 29 May 2020 13:21:38 +0100 Subject: [PATCH 11/40] Update changelog --- CHANGELOG.md | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index cb26710e89..a998a95757 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -4,13 +4,14 @@ [Full Changelog](https://github.com/cloudfoundry/stratos/compare/3.2.0...3.2.1) -This release contains a number of fixes: +**Improvements:** + +- Apply SSO whitelist to additional places [\#4318](https://github.com/cloudfoundry/stratos/issues/4318) **Fixes:** - User Button Misaligned [\#4316](https://github.com/cloudfoundry/stratos/issues/4316) - ## 3.2.0 [Full Changelog](https://github.com/cloudfoundry/stratos/compare/3.1.0...3.2.0) From 80ad363ddeb6f9b7372cfcfbd74136355c442e83 Mon Sep 17 00:00:00 2001 From: Richard Cox Date: Mon, 1 Jun 2020 13:35:56 +0100 Subject: [PATCH 12/40] Fix package-lock.json --- package-lock.json | 14 +++++++------- 1 file changed, 7 insertions(+), 7 deletions(-) diff --git a/package-lock.json b/package-lock.json index 86c2a1c5e8..e6c33bafa9 100644 --- a/package-lock.json +++ b/package-lock.json @@ -319,7 +319,7 @@ "@schematics/update": "0.901.5", "@yarnpkg/lockfile": "1.1.0", "ansi-colors": "4.1.1", - "debug": "^4.1.1", + "debug": "4.1.1", "ini": "1.3.5", "inquirer": "7.1.0", "npm-package-arg": "8.0.1", @@ -6022,7 +6022,7 @@ }, "duplexer": { "version": "0.1.1", - "resolved": "https://registry.npmjs.org/duplexer/-/duplexer-0.1.1.tgz", + "resolved": "http://registry.npmjs.org/duplexer/-/duplexer-0.1.1.tgz", "integrity": "sha1-rOb/gIwc5mtX0ev5eXessCM0z8E=", "dev": true }, @@ -6716,7 +6716,7 @@ }, "event-stream": { "version": "3.3.4", - "resolved": "https://registry.npmjs.org/event-stream/-/event-stream-3.3.4.tgz", + "resolved": "http://registry.npmjs.org/event-stream/-/event-stream-3.3.4.tgz", "integrity": "sha1-SrTJoPWlTbkzi0w02Gv86PSzVXE=", "dev": true, "requires": { @@ -11197,7 +11197,7 @@ }, "map-stream": { "version": "0.1.0", - "resolved": "https://registry.npmjs.org/map-stream/-/map-stream-0.1.0.tgz", + "resolved": "http://registry.npmjs.org/map-stream/-/map-stream-0.1.0.tgz", "integrity": "sha1-5WqpTEyAVaFkBKBnS3jyFffI4ZQ=", "dev": true }, @@ -13535,7 +13535,7 @@ }, "pause-stream": { "version": "0.0.11", - "resolved": "https://registry.npmjs.org/pause-stream/-/pause-stream-0.0.11.tgz", + "resolved": "http://registry.npmjs.org/pause-stream/-/pause-stream-0.0.11.tgz", "integrity": "sha1-/lo0sMvOErWqaitAPuLnO2AvFEU=", "dev": true, "requires": { @@ -17051,7 +17051,7 @@ }, "split": { "version": "0.3.3", - "resolved": "https://registry.npmjs.org/split/-/split-0.3.3.tgz", + "resolved": "http://registry.npmjs.org/split/-/split-0.3.3.tgz", "integrity": "sha1-zQ7qXmOiEd//frDwkcQTPi0N0o8=", "dev": true, "requires": { @@ -17217,7 +17217,7 @@ }, "stream-combiner": { "version": "0.0.4", - "resolved": "https://registry.npmjs.org/stream-combiner/-/stream-combiner-0.0.4.tgz", + "resolved": "http://registry.npmjs.org/stream-combiner/-/stream-combiner-0.0.4.tgz", "integrity": "sha1-TV5DPBhSYd3mI8o/RMWGvPXErRQ=", "dev": true, "requires": { From 36670d206c9e0c509deb6bf3f885ab6e634d8405 Mon Sep 17 00:00:00 2001 From: Richard Cox Date: Mon, 1 Jun 2020 13:36:28 +0100 Subject: [PATCH 13/40] Remove v3.2.1 was .travis.taml --- .travis.yml | 1 - 1 file changed, 1 deletion(-) diff --git a/.travis.yml b/.travis.yml index 3d04b8ae8b..7f87e2619e 100644 --- a/.travis.yml +++ b/.travis.yml @@ -20,7 +20,6 @@ branches: - master - v2-master - e2e-tests - - v3.2.1 cache: directories: - "$HOME/.npm" From 6447b3bdddb5fded04f0ea992e696c7fd6220e18 Mon Sep 17 00:00:00 2001 From: Neil MacDougall Date: Mon, 1 Jun 2020 21:39:31 +0100 Subject: [PATCH 14/40] Remove need for --recreate-pods when upgrading --- deploy/ci/automation/helmtest.sh | 6 +++--- deploy/kubernetes/README.md | 2 +- deploy/kubernetes/build.sh | 1 + deploy/kubernetes/console/README.md | 4 +--- deploy/kubernetes/console/templates/__stratos.tpl | 5 +++++ deploy/kubernetes/console/templates/config-init.yaml | 2 ++ deploy/kubernetes/console/templates/database.yaml | 2 ++ deploy/kubernetes/console/templates/deployment.yaml | 5 +++++ 8 files changed, 20 insertions(+), 7 deletions(-) create mode 100644 deploy/kubernetes/console/templates/__stratos.tpl diff --git a/deploy/ci/automation/helmtest.sh b/deploy/ci/automation/helmtest.sh index 8c05ad703f..5a50bfac18 100755 --- a/deploy/ci/automation/helmtest.sh +++ b/deploy/ci/automation/helmtest.sh @@ -164,7 +164,7 @@ CHART_FILE="${KUBE_FOLDER}/console-${DEV_IMAGE_VERSION}.tgz" echo "Chart file path: ${CHART_FILE}" log "Upgrading using latest Helm Chart" -helm upgrade ${NAME} "${CHART_FILE}" --recreate-pods --debug --set consoleVersion=${DEV_IMAGE_VERSION} --set imagePullPolicy=Always +helm upgrade ${NAME} "${CHART_FILE}" --debug --set consoleVersion=${DEV_IMAGE_VERSION} --set imagePullPolicy=Always checkVersion console-${DEV_IMAGE_VERSION} waitForHelmRelease @@ -177,7 +177,7 @@ sed -i.bak -e 's/appVersion: '"${DEV_IMAGE_VERSION}"'/appVersion: 0.2.0/g' "${HE cat "${HELM_TMP}/Chart.yaml" log "Upgrading using latest Helm Chart (checking chart upgrade)" -helm upgrade ${NAME} "${HELM_TMP}" --recreate-pods --debug --set imagePullPolicy=Always +helm upgrade ${NAME} "${HELM_TMP}" --debug --set imagePullPolicy=Always waitForHelmRelease checkVersion console-0.2.0 @@ -204,7 +204,7 @@ helm install ${HELM_REPO_NAME}/console --name ${NAME} --namespace ${NAMESPACE} waitForHelmRelease log "Upgrading using --reuse-values" -helm upgrade ${NAME} "${HELM_TMP}" --recreate-pods --debug --reuse-values +helm upgrade ${NAME} "${HELM_TMP}" --debug --reuse-values waitForHelmRelease # Should have used same values as before diff --git a/deploy/kubernetes/README.md b/deploy/kubernetes/README.md index 35b76359f2..8e2a4c066f 100644 --- a/deploy/kubernetes/README.md +++ b/deploy/kubernetes/README.md @@ -220,7 +220,7 @@ $ helm repo update To update an instance, the following assumes your instance is called `my-console`, and overrides have been specified in a file called `overrides.yaml`. ``` -$ helm upgrade -f overrides.yaml my-console stratos/console --recreate-pods +$ helm upgrade -f overrides.yaml my-console stratos/console ``` After the upgrade, perform a `helm list` to ensure your console is the latest version. diff --git a/deploy/kubernetes/build.sh b/deploy/kubernetes/build.sh index c2e0c5e730..63cc4a0f63 100755 --- a/deploy/kubernetes/build.sh +++ b/deploy/kubernetes/build.sh @@ -236,6 +236,7 @@ pushd "${DEST_HELM_CHART_PATH}" > /dev/null rm -rf "${DEST_HELM_CHART_PATH}/**/*.orig" # Run customization script if there is one +# This can do things like provide a custom __stratos.tpl file if [ -f "${STRATOS_PATH}/custom-src/deploy/kubernetes/customize-helm.sh" ]; then printf "${YELLOW}${BOLD}Applying Helm Chart customizations${RESET}\n" "${STRATOS_PATH}/custom-src/deploy/kubernetes/customize-helm.sh" "${DEST_HELM_CHART_PATH}" diff --git a/deploy/kubernetes/console/README.md b/deploy/kubernetes/console/README.md index e4c920c061..cc4a7399d4 100644 --- a/deploy/kubernetes/console/README.md +++ b/deploy/kubernetes/console/README.md @@ -155,11 +155,9 @@ helm repo update To update an instance, the following assumes your instance is called `my-console`: ``` -helm upgrade my-console stratos/console --recreate-pods +helm upgrade my-console stratos/console ``` -> Note: You *must* use the `--recreate-pods` flag when upgrading - After the upgrade, perform a `helm list` to ensure your console is the latest version. ## Uninstalling diff --git a/deploy/kubernetes/console/templates/__stratos.tpl b/deploy/kubernetes/console/templates/__stratos.tpl new file mode 100644 index 0000000000..f1cc982504 --- /dev/null +++ b/deploy/kubernetes/console/templates/__stratos.tpl @@ -0,0 +1,5 @@ +# Customizations for the Helm Chart + +# Extra env vars for the Jetstream Pod in deployment.yaml +{{- define "stratosJetstreamEnv" }} +{{- end }} \ No newline at end of file diff --git a/deploy/kubernetes/console/templates/config-init.yaml b/deploy/kubernetes/console/templates/config-init.yaml index e6d2f3582a..0cc82ee389 100644 --- a/deploy/kubernetes/console/templates/config-init.yaml +++ b/deploy/kubernetes/console/templates/config-init.yaml @@ -102,6 +102,8 @@ spec: {{- end }} containers: - env: + - name: STRATOS_IMAGE_REF + value: "{{.Values.consoleVersion}}:{{ .Release.Revision }}" - name: "STRATOS_VOLUME_MIGRATION" value: "true" - name: "IS_UPGRADE" diff --git a/deploy/kubernetes/console/templates/database.yaml b/deploy/kubernetes/console/templates/database.yaml index 2f23173f49..5f64b162a7 100644 --- a/deploy/kubernetes/console/templates/database.yaml +++ b/deploy/kubernetes/console/templates/database.yaml @@ -60,6 +60,8 @@ spec: image: {{.Values.kube.registry.hostname}}/{{.Values.kube.organization}}/{{.Values.images.mariadb}}:{{.Values.consoleVersion}} imagePullPolicy: {{.Values.imagePullPolicy}} env: + - name: STRATOS_IMAGE_REF + value: "{{.Values.consoleVersion}}:{{ .Release.Revision }}" - name: MYSQL_ROOT_PASSWORD valueFrom: secretKeyRef: diff --git a/deploy/kubernetes/console/templates/deployment.yaml b/deploy/kubernetes/console/templates/deployment.yaml index 5446aefec3..6889e86bab 100644 --- a/deploy/kubernetes/console/templates/deployment.yaml +++ b/deploy/kubernetes/console/templates/deployment.yaml @@ -65,6 +65,8 @@ spec: name: https protocol: TCP env: + - name: STRATOS_IMAGE_REF + value: "{{.Values.consoleVersion}}:{{ .Release.Revision }}" - name: CONSOLE_CERT_PATH value: "/{{ .Release.Name }}-cert-volume" volumeMounts: @@ -75,6 +77,8 @@ spec: imagePullPolicy: {{.Values.imagePullPolicy}} name: proxy env: + - name: STRATOS_IMAGE_REF + value: "{{.Values.consoleVersion}}:{{ .Release.Revision }}" - name: STRATOS_HELM_RELEASE value: "{{ .Release.Name }}" - name: DB_USER @@ -277,6 +281,7 @@ spec: - name: UI_LIST_ALLOW_LOAD_MAXED value: {{ default "false" .Values.console.ui.listAllowLoadMaxed | quote }} {{- end }} + {{- include "stratosJetstreamEnv" . | indent 8 }} readinessProbe: httpGet: path: /pp/v1/ping From dedfe8e24a11296a150e8b8f2a9d07a87e776227 Mon Sep 17 00:00:00 2001 From: Neil MacDougall Date: Tue, 2 Jun 2020 09:15:23 +0100 Subject: [PATCH 15/40] WIP: Kubernetes terminal --- custom-src/deploy/kubernetes/custom-build.sh | 4 + .../kube-console/kube-console.component.html | 2 +- .../kube-console/kube-console.component.ts | 3 +- .../kubernetes-summary.component.html | 4 +- deploy/Dockerfile.kubeconsole.yml | 18 -- .../kube-terminal/Dockerfile.kubeconsole | 40 ++++ .../kube-terminal}/kubeconsole.bashrc | 35 ++- .../console/templates/deployment.yaml | 4 + .../ssh-viewer/ssh-viewer.component.html | 2 +- .../ssh-viewer/ssh-viewer.component.ts | 10 + src/jetstream/plugins/kubernetes/main.go | 9 +- .../{kube_console.go => terminal.go} | 210 ++++++++---------- .../plugins/kubernetes/terminal/terminal.go | 35 +++ 13 files changed, 230 insertions(+), 146 deletions(-) delete mode 100644 deploy/Dockerfile.kubeconsole.yml create mode 100644 deploy/containers/kube-terminal/Dockerfile.kubeconsole rename deploy/{ => containers/kube-terminal}/kubeconsole.bashrc (50%) rename src/jetstream/plugins/kubernetes/{kube_console.go => terminal.go} (79%) create mode 100644 src/jetstream/plugins/kubernetes/terminal/terminal.go diff --git a/custom-src/deploy/kubernetes/custom-build.sh b/custom-src/deploy/kubernetes/custom-build.sh index b850a85563..14e129d336 100644 --- a/custom-src/deploy/kubernetes/custom-build.sh +++ b/custom-src/deploy/kubernetes/custom-build.sh @@ -21,4 +21,8 @@ function custom_image_build() { # Build and push an image for the Helm Repo Sync Tool log "-- Building/publishing Monocular Chart Repo Sync Tool" patchAndPushImage stratos-chartsync Dockerfile "${STRATOS_PATH}/src/jetstream/plugins/monocular/chart-repo" + + # Build and push an image for the Kubernetes Terminal + log "-- Building/publishing Kubernetes Terminal" + patchAndPushImage stratos-kube-terminal Dockerfile.kubeconsole "${STRATOS_PATH}/deploy/containers/kube-terminal" } \ No newline at end of file diff --git a/custom-src/frontend/app/custom/kubernetes/kube-console/kube-console.component.html b/custom-src/frontend/app/custom/kubernetes/kube-console/kube-console.component.html index e8cfda74c0..905abb2f62 100644 --- a/custom-src/frontend/app/custom/kubernetes/kube-console/kube-console.component.html +++ b/custom-src/frontend/app/custom/kubernetes/kube-console/kube-console.component.html @@ -1,5 +1,5 @@ -

Kubernetes Console

+

Kubernetes Terminal

diff --git a/deploy/Dockerfile.kubeconsole.yml b/deploy/Dockerfile.kubeconsole.yml deleted file mode 100644 index 48c3f21443..0000000000 --- a/deploy/Dockerfile.kubeconsole.yml +++ /dev/null @@ -1,18 +0,0 @@ -FROM splatform/stratos-bk-base:opensuse as prod-build -USER root -RUN zypper in -y wget tar unzip nano vim curl - -RUN curl -LO https://storage.googleapis.com/kubernetes-release/release/$(curl -s https://storage.googleapis.com/kubernetes-release/release/stable.txt)/bin/linux/amd64/kubectl && \ - chmod +x ./kubectl && \ - mv ./kubectl /usr/local/bin/kubectl - -# Install Helm - -RUN curl -LO https://git.io/get_helm.sh && \ - chmod 700 get_helm.sh && \ - ./get_helm.sh - -ADD ./kubeconsole.bashrc /root/.bashrc -WORKDIR /root - -CMD exec /bin/bash -c "trap : TERM INT; sleep infinity & wait" diff --git a/deploy/containers/kube-terminal/Dockerfile.kubeconsole b/deploy/containers/kube-terminal/Dockerfile.kubeconsole new file mode 100644 index 0000000000..19f55f08b4 --- /dev/null +++ b/deploy/containers/kube-terminal/Dockerfile.kubeconsole @@ -0,0 +1,40 @@ +FROM splatform/stratos-bk-build-base:leap15_1 as terminal-builder +USER root +WORKDIR /root + +#RUN zypper in -y tar + +# Kubectl versions +RUN curl -L -o kubectl_1.18 https://storage.googleapis.com/kubernetes-release/release/v1.18.2/bin/linux/amd64/kubectl +RUN curl -L -o kubectl_1.17 https://storage.googleapis.com/kubernetes-release/release/v1.17.5/bin/linux/amd64/kubectl +#RUN curl -L -o kubectl_1.16 https://storage.googleapis.com/kubernetes-release/release/v1.16.9/bin/linux/amd64/kubectl +#RUN curl -L -o kubectl_1.15 https://storage.googleapis.com/kubernetes-release/release/v1.15.11/bin/linux/amd64/kubectl +RUN curl -L -o kubectl_1.14 https://storage.googleapis.com/kubernetes-release/release/v1.14.10/bin/linux/amd64/kubectl + +# Tar each one up, to save space in the image +RUN gzip kubectl_1.18 +RUN gzip kubectl_1.17 +RUN gzip kubectl_1.14 + +# Fetch Helm 3 package +RUN curl -L -o helm.tar.gz https://get.helm.sh/helm-v3.1.2-linux-amd64.tar.gz && \ + tar -xvf helm.tar.gz --strip-components=1 && \ + gzip helm + +RUN ls -al + +FROM splatform/stratos-bk-init-base:leap15_1 + +RUN zypper in -y gzip + +RUN mkdir /stratos + +# Copy the various kubectl versions + +COPY --from=terminal-builder /root/helm.gz /stratos/helm.gz +COPY --from=terminal-builder /root/kubectl* /stratos/ + +ADD ./kubeconsole.bashrc /root/.bashrc +WORKDIR /root + +CMD exec /bin/bash -c "trap : TERM INT; sleep infinity & wait" diff --git a/deploy/kubeconsole.bashrc b/deploy/containers/kube-terminal/kubeconsole.bashrc similarity index 50% rename from deploy/kubeconsole.bashrc rename to deploy/containers/kube-terminal/kubeconsole.bashrc index 2ac7e17679..10da69e597 100644 --- a/deploy/kubeconsole.bashrc +++ b/deploy/containers/kube-terminal/kubeconsole.bashrc @@ -6,18 +6,42 @@ RESET="\033[0m" BOLD="\033[1m" DIM="\033[2m" -echo -e "${BOLD}${GREEN}SUSE Cloud Application Platform${RESET}" +echo -e "${BOLD}${GREEN}SUSE Stratos Console${RESET}" echo "" -echo -e "${CYAN}Kubernetes Console${RESET}" +echo -e "${CYAN}Kubernetes Terminal${RESET}" echo "" +# Unpack helm comand +gunzip /stratos/helm.gz + +# Need to choose appropriate kubectl version +pushd /stratos > /dev/null +# Default to the newwest version that we have +USE=$(ls kubectl_* | sort -r | head -n1) +popd > /dev/null + +# If env var KUBERNETES_VERSION is set, then use it (major.minor only) +if [ -n "${KUBERNETES_VERSION}" ]; then + VERSION="kubectl_${KUBERNETES_VERSION}.gz" + if [ -f "/stratos/${VERSION}" ]; then + USE=${VERSION} + fi +fi + +gunzip /stratos/${USE} +VER=${USE::-3} +mv /stratos/${VER} /stratos/kubectl +chmod +x /stratos/kubectl +export PATH=/stratos:$PATH + export KUBECONFIG=/root/.stratos/kubeconfig export PS1="\033[92mstratos>\033[0m" alias k=kubectl -helm init --client-only > /dev/null -helm repo remove local > /dev/null -helm repo remove stable > /dev/null +# Helm shell completion +source <(helm completion bash) + +#helm repo remove stable > /dev/null if [ -f "/root/.stratos/helm-setup" ]; then echo "Setting up Helm repositories ..." @@ -26,7 +50,6 @@ if [ -f "/root/.stratos/helm-setup" ]; then echo "" fi - if [ -f "/root/.stratos/history" ]; then cat /root/.stratos/history > /root/.bash_history fi diff --git a/deploy/kubernetes/console/templates/deployment.yaml b/deploy/kubernetes/console/templates/deployment.yaml index c4123af521..6792a1ebf0 100644 --- a/deploy/kubernetes/console/templates/deployment.yaml +++ b/deploy/kubernetes/console/templates/deployment.yaml @@ -71,6 +71,10 @@ spec: env: - name: STRATOS_HELM_RELEASE value: "{{ .Release.Name }}" + - name: STRATOS_KUBERNETES_NAMESPACE + value: "{{ .Release.Namespace }}" + - name: STRATOS_KUBERNETES_TERMINAL_IMAGE + value: "{{.Values.kube.registry.hostname}}/{{.Values.kube.organization}}/stratos-kube-terminal:{{.Values.consoleVersion}}" - name: DB_USER valueFrom: secretKeyRef: diff --git a/src/frontend/packages/core/src/shared/components/ssh-viewer/ssh-viewer.component.html b/src/frontend/packages/core/src/shared/components/ssh-viewer/ssh-viewer.component.html index 44cac219ee..d5cee755ee 100644 --- a/src/frontend/packages/core/src/shared/components/ssh-viewer/ssh-viewer.component.html +++ b/src/frontend/packages/core/src/shared/components/ssh-viewer/ssh-viewer.component.html @@ -1,7 +1,7 @@
- Error occurred establishing SSH connection + {{ errorMessage || 'Error occurred establishing SSH connection' }}
Disconnected diff --git a/src/frontend/packages/core/src/shared/components/ssh-viewer/ssh-viewer.component.ts b/src/frontend/packages/core/src/shared/components/ssh-viewer/ssh-viewer.component.ts index 8d9d8e055c..c12df6d6a1 100644 --- a/src/frontend/packages/core/src/shared/components/ssh-viewer/ssh-viewer.component.ts +++ b/src/frontend/packages/core/src/shared/components/ssh-viewer/ssh-viewer.component.ts @@ -122,6 +122,11 @@ export class SshViewerComponent implements OnInit, OnDestroy { for (const c of data.split(' ')) { this.xterm.write(String.fromCharCode(parseInt(c, 16))); } + } else { + console.log('Error') + const eMsg = this.errorMessage; + //this.disconnect(); + this.errorMessage = eMsg; } }, (err) => { @@ -146,6 +151,11 @@ export class SshViewerComponent implements OnInit, OnDestroy { for (let i = 4; i < chars.length - 1; i++) { title += String.fromCharCode(parseInt(chars[i], 16)); } + if (title.length > 0 && title.charAt(0) === '!') { + this.errorMessage = title.substr(1); + console.log(this.errorMessage); + return true; + } this.message = title; } return false; diff --git a/src/jetstream/plugins/kubernetes/main.go b/src/jetstream/plugins/kubernetes/main.go index 75e530df4d..5e12edf9c6 100644 --- a/src/jetstream/plugins/kubernetes/main.go +++ b/src/jetstream/plugins/kubernetes/main.go @@ -14,12 +14,15 @@ import ( log "github.com/sirupsen/logrus" "github.com/cloudfoundry-incubator/stratos/src/jetstream/plugins/kubernetes/auth" + + "github.com/cloudfoundry-incubator/stratos/src/jetstream/plugins/kubernetes/terminal" ) // KubernetesSpecification is the endpoint that adds Kubernetes support to the backend type KubernetesSpecification struct { portalProxy interfaces.PortalProxy endpointType string + kubeTerminal *terminal.KubeTerminal } type KubeStatus struct { @@ -53,8 +56,10 @@ const ( kubeDashboardPluginConfigSetting = "kubeDashboardEnabled" ) +// Init creates a new instance of the Kubernetes plugin func Init(portalProxy interfaces.PortalProxy) (interfaces.StratosPlugin, error) { - return &KubernetesSpecification{portalProxy: portalProxy, endpointType: kubeEndpointType}, nil + kubeTerminal := terminal.NewKubeTerminal(portalProxy) + return &KubernetesSpecification{portalProxy: portalProxy, endpointType: kubeEndpointType, kubeTerminal: kubeTerminal}, nil } func (c *KubernetesSpecification) GetEndpointPlugin() (interfaces.EndpointPlugin, error) { @@ -160,7 +165,7 @@ func (c *KubernetesSpecification) AddSessionGroupRoutes(echoGroup *echo.Group) { echoGroup.GET("/helm/releases/:endpoint/:namespace/:name", c.GetRelease) // Kube Terminal - echoGroup.GET("/kubeconsole/:guid", c.KubeConsole) + echoGroup.GET("/kubeconsole/:guid", c.KubeTerminal) } diff --git a/src/jetstream/plugins/kubernetes/kube_console.go b/src/jetstream/plugins/kubernetes/terminal.go similarity index 79% rename from src/jetstream/plugins/kubernetes/kube_console.go rename to src/jetstream/plugins/kubernetes/terminal.go index 7f54775d01..c1a98d5496 100644 --- a/src/jetstream/plugins/kubernetes/kube_console.go +++ b/src/jetstream/plugins/kubernetes/terminal.go @@ -2,34 +2,33 @@ package kubernetes import ( "bytes" + "crypto/tls" "errors" "fmt" - "crypto/tls" + //"encoding/base64" "encoding/json" "net" "net/http" - "time" "strings" - //"io/ioutil" - //yaml "gopkg.in/yaml.v2" + "time" - uuid "github.com/satori/go.uuid" "github.com/labstack/echo" + uuid "github.com/satori/go.uuid" log "github.com/sirupsen/logrus" - "github.com/cloudfoundry-incubator/stratos/src/jetstream/repository/interfaces" "github.com/cloudfoundry-incubator/stratos/src/jetstream/plugins/kubernetes/auth" + "github.com/cloudfoundry-incubator/stratos/src/jetstream/repository/interfaces" restclient "k8s.io/client-go/rest" - //"k8s.io/kubernetes/pkg/client/unversioned/remotecommand" + "github.com/gorilla/websocket" - "k8s.io/client-go/kubernetes" v1 "k8s.io/api/core/v1" metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" + "k8s.io/client-go/kubernetes" corev1 "k8s.io/client-go/kubernetes/typed/core/v1" - "k8s.io/client-go/tools/remotecommand" scheme "k8s.io/client-go/kubernetes/scheme" + "k8s.io/client-go/tools/remotecommand" ) const ( @@ -47,13 +46,13 @@ var upgrader = websocket.Upgrader{ // PodCreationData stores the clients and names used ot create pod and secret type PodCreationData struct { - ClientSet *kubernetes.Clientset - Config *restclient.Config - Namespace string - PodClient corev1.PodInterface + ClientSet *kubernetes.Clientset + Config *restclient.Config + Namespace string + PodClient corev1.PodInterface SecretClient corev1.SecretInterface - PodName string - SecretName string + PodName string + SecretName string } // KeyCode - JSON object that is passed from the front-end to notify of a key press or a term resize @@ -63,7 +62,7 @@ type KeyCode struct { Rows int `json:"rows"` } -type TermianlSize struct { +type terminalSize struct { Width uint16 Height uint16 } @@ -73,7 +72,24 @@ const ( writeWait = 10 * time.Second ) -func (k *KubernetesSpecification) KubeConsole(c echo.Context) error { +// Determine if we Kube Terminal is supported +// TODO +// Check namespace +// Check we have secret for pod creation/deletion +// Image for the terminal + +// Environment variables that we expect to see +// STRATOS_KUBERNETES_NAMESPACE +// STRATOS_KUBERNETES_TERMINAL_IMAGE + +// For development, you can overide the token to use with the Kubernetes API, with: +// STRATOS_KUBERNETES_API_TOKEN + +// Start periodic routing to clean up pods that are orphaned +// TODO + +// KubeTerminal handles web-socket request to launch a Kubernetes Terminal +func (k *KubernetesSpecification) KubeTerminal(c echo.Context) error { c.Response().Status = 500 @@ -120,6 +136,9 @@ func (k *KubernetesSpecification) KubeConsole(c echo.Context) error { log.Error("ERROR creating secret or pod") log.Info(err) k.cleanupPodAndSecret(podData) + + // Send error message + sendProgressMessage(ws, "!"+err.Error()) return err } @@ -143,38 +162,37 @@ func (k *KubernetesSpecification) KubeConsole(c echo.Context) error { // return errors.New("Could not get config for this auth type") // } + // req, err := http.NewRequest("POST", target, nil) + // if err != nil { + // k.cleanupPodAndSecret(podData) + // return errors.New("Could not create new HTTP request") + // } - req, err := http.NewRequest("POST", target, nil) - if err != nil { - k.cleanupPodAndSecret(podData) - return errors.New("Could not create new HTTP request") - } - - // Set auth header so we log in if needed - if len(tokenRec.AuthToken) > 0 { - //req.Header.Add("Authorization", "Bearer "+tokenRec.AuthToken) - log.Info("Setting auth header") - } - - //req.Header.Add("Accept", "*/*") + // // TODO: Check response code + // log.Info(re) - log.Info("Config") - log.Info("Making request") - log.Info(req) + // // Set auth header so we log in if needed + // if len(tokenRec.AuthToken) > 0 { + // //req.Header.Add("Authorization", "Bearer "+tokenRec.AuthToken) + // log.Info("Setting auth header") + // } // endpointRequest := &interfaces.CNSIRequest{ // GUID: endpointGUID, // } + // TODO: Make generic + + // What is the token for ??? kubeAuthToken := &auth.KubeCertificate{} err = json.NewDecoder(strings.NewReader(tokenRec.AuthToken)).Decode(kubeAuthToken) if err != nil { - k.cleanupPodAndSecret(podData) + k.cleanupPodAndSecret(podData) return err } cert, err := kubeAuthToken.GetCerticate() if err != nil { - k.cleanupPodAndSecret(podData) + k.cleanupPodAndSecret(podData) return err } dial := (&net.Dialer{ @@ -182,6 +200,8 @@ func (k *KubernetesSpecification) KubeConsole(c echo.Context) error { KeepAlive: 30 * time.Second, }).Dial + // TODO: Not sure about the cert here? I don't think we need this like it is - wonder why I did that? + sslTransport := &http.Transport{ Proxy: http.ProxyFromEnvironment, Dial: dial, @@ -193,10 +213,11 @@ func (k *KubernetesSpecification) KubeConsole(c echo.Context) error { MaxIdleConnsPerHost: 6, // (default is 2) } - kubeCertClient := http.Client{} - kubeCertClient.Transport = sslTransport - kubeCertClient.Timeout = time.Duration(30) * time.Second + kubeHTTPClient := http.Client{} + kubeHTTPClient.Transport = sslTransport + kubeHTTPClient.Timeout = time.Duration(30) * time.Second + // This dialer does not use the kubeHttpClient - is it unused? dialer := &websocket.Dialer{ TLSClientConfig: &tls.Config{ InsecureSkipVerify: true, @@ -204,7 +225,7 @@ func (k *KubernetesSpecification) KubeConsole(c echo.Context) error { }, } - if strings.HasPrefix(target ,"https://") { + if strings.HasPrefix(target, "https://") { target = "wss://" + target[8:] } else { target = "ws://" + target[7:] @@ -217,48 +238,18 @@ func (k *KubernetesSpecification) KubeConsole(c echo.Context) error { defer wsConn.Close() } - // log.Info(err) - // log.Info(res) - // log.Info(wsConn) - - // if kubeAuthToken.Token != "" { - // req.Header.Set("Authorization", "bearer "+kubeAuthToken.Token) - // } - - //res, err := kubeCertClient.Do(req) + // Check res + log.Warn("== Status ===========================================") + log.Info(res.Status) + log.Info(res.StatusCode) - - kubeCertClient.CloseIdleConnections() - - //var res *http.Response - - // var client http.Client - // client = p.GetHttpClientForRequest(req, cnsiRecord.SkipSSLValidation) - // res, err = client.Do(req) - - // Find the auth provider for the auth type - default to oauthflow - // authHandler := p.GetAuthProvider(tokenRec.AuthType) - // if authHandler.Handler != nil { - // res, err = authHandler.Handler(endpointRequest, req) - // } else { - // res, err = p.DoOAuthFlowRequest(endpointRequest, req) - // } - log.Error(err) -// log.Error(res) + kubeHTTPClient.CloseIdleConnections() if err != nil { - log.Error("Failed to make request") k.cleanupPodAndSecret(podData) return errors.New("Could not make request") } - log.Error("=== Made request to exec endpoint OK") - log.Error(res) - - // Websockets next - //log.Info(wsConn) - - //done := make(chan struct{}) stdoutDone := make(chan struct{}) go pumpStdout(ws, wsConn, stdoutDone) @@ -266,9 +257,8 @@ func (k *KubernetesSpecification) KubeConsole(c echo.Context) error { for { _, r, err := ws.ReadMessage() if err != nil { - log.Error("Error reading message from web socket") - log.Warnf("%v+", err) - k.cleanupPodAndSecret(podData) + log.Errorf("Kubernetes terminal: error reading message from web socket: %+v", err) + k.cleanupPodAndSecret(podData) return err } @@ -276,36 +266,26 @@ func (k *KubernetesSpecification) KubeConsole(c echo.Context) error { json.Unmarshal(r, &res) if res.Cols == 0 { - - slice := make([]byte, 1) slice[0] = 0 slice = append(slice, []byte(res.Key)...) wsConn.WriteMessage(websocket.TextMessage, slice) } else { - // Terminal resize request - // if err := windowChange(session, res.Rows, res.Cols); err != nil { - // log.Error("Can not resize the PTY") - // } - log.Error("Terminal resize receieved") - - size := TermianlSize{ - Width: uint16(res.Cols), + size := terminalSize{ + Width: uint16(res.Cols), Height: uint16(res.Rows), } j, _ := json.Marshal(size) - log.Info(j) - resizeStream := []byte{4} slice := append(resizeStream, j...) - log.Info(slice) wsConn.WriteMessage(websocket.TextMessage, slice) } } // Cleanup - log.Info("*** Cleaning up.... ***") + log.Debug("Kubernetes Terminal is cleaning up") + // This is unreachable???? return k.cleanupPodAndSecret(podData) } @@ -321,7 +301,7 @@ func pumpStdout(ws *websocket.Conn, source *websocket.Conn, done chan struct{}) ws.SetWriteDeadline(time.Now().Add(writeWait)) bytes := fmt.Sprintf("% x\n", r[1:]) if err := ws.WriteMessage(websocket.TextMessage, []byte(bytes)); err != nil { - log.Error("App SSH Failed to write nessage") + log.Error("Kubernetes Terimnal failed to write nessage") ws.Close() break } @@ -343,7 +323,7 @@ func (k *KubernetesSpecification) createPod(c echo.Context, cnsiRecord interface result.Config = config - kubeConfig, err := k.GetKubeConfigForEndpoint(cnsiRecord.APIEndpoint.String(), tokenRecord) + kubeConfig, err := k.GetKubeConfigForEndpoint(cnsiRecord.APIEndpoint.String(), tokenRecord, namespace) if err != nil { return result, errors.New("Can not get Kubernetes config for specified endpoint") } @@ -360,11 +340,11 @@ func (k *KubernetesSpecification) createPod(c echo.Context, cnsiRecord interface // Create the secret secretSpec := &v1.Secret{ TypeMeta: metav1.TypeMeta{ - Kind: "secret", + Kind: "secret", APIVersion: "v1", }, ObjectMeta: metav1.ObjectMeta{ - Name: secretName, + Name: secretName, Namespace: namespace, }, Type: "Opaque", @@ -373,7 +353,7 @@ func (k *KubernetesSpecification) createPod(c echo.Context, cnsiRecord interface secretSpec.Data = make(map[string][]byte) secretSpec.Data["kubeconfig"] = []byte(kubeConfig) secretSpec.Data["history"] = []byte(history) - + // Get Helm repository script if we have Helm repositories helmSetup := getHelmRepoSetupScript(k.portalProxy) if len(helmSetup) > 0 { @@ -395,11 +375,11 @@ func (k *KubernetesSpecification) createPod(c echo.Context, cnsiRecord interface podSpec := &v1.Pod{ TypeMeta: metav1.TypeMeta{ - Kind: "pod", + Kind: "pod", APIVersion: "v1", }, ObjectMeta: metav1.ObjectMeta{ - Name: podName, + Name: podName, Namespace: namespace, }, } @@ -407,7 +387,7 @@ func (k *KubernetesSpecification) createPod(c echo.Context, cnsiRecord interface podSpec.ObjectMeta.Annotations = make(map[string]string) // Record the session ID - session, err :=k.portalProxy.GetSession(c) + session, err := k.portalProxy.GetSession(c) if err == nil { podSpec.ObjectMeta.Annotations["stratos-session"] = session.ID log.Infof("Session ID: %s", session.ID) @@ -435,7 +415,7 @@ func (k *KubernetesSpecification) createPod(c echo.Context, cnsiRecord interface volumesSpec[0].Secret = &v1.SecretVolumeSource{ SecretName: secretName, } - podSpec.Spec.Volumes = volumesSpec + podSpec.Spec.Volumes = volumesSpec // Create a new pod pod, err := podClient.Create(podSpec) @@ -446,7 +426,7 @@ func (k *KubernetesSpecification) createPod(c echo.Context, cnsiRecord interface result.PodClient = podClient result.PodName = podName - sendProgressMessage(ws, "Waiting for Kubernetes Console to start up ...") + sendProgressMessage(ws, "Waiting for Kubernetes Terminal to start up ...") statusOptions := metav1.GetOptions{} @@ -518,6 +498,7 @@ func sendProgressMessage(ws *websocket.Conn, progressMsg string) { } } +// Not used? func captureBashHistory(podData *PodCreationData) error { log.Warn("**** Trying to capture bash history from pod ****") @@ -530,14 +511,13 @@ func captureBashHistory(podData *PodCreationData) error { //cmd := []string{"bash", "-c", "\"cat $HISTFILE\""} cmd := []string{"cat", "/root/.bash_history"} - - req := podData.ClientSet.Core().RESTClient().Post(). + req := podData.ClientSet.CoreV1().RESTClient().Post(). Resource("pods"). Name(podData.PodName). Namespace(podData.Namespace). SubResource("exec"). Param("container", consoleContainerName) - + req.VersionedParams(&v1.PodExecOptions{ Container: consoleContainerName, Command: cmd, @@ -546,7 +526,7 @@ func captureBashHistory(podData *PodCreationData) error { Stderr: false, TTY: true, }, scheme.ParameterCodec) - + var stdout bytes.Buffer //err := execute("POST", req.URL(), nil, nil, &stdout, nil, false) @@ -561,10 +541,10 @@ func captureBashHistory(podData *PodCreationData) error { err = exec.Stream(remotecommand.StreamOptions{ //SupportedProtocols: remotecommandserver.SupportedStreamingProtocols, - Stdin: nil, - Stdout: &stdout, - Stderr: nil, - Tty: true, + Stdin: nil, + Stdout: &stdout, + Stderr: nil, + Tty: true, }) log.Error("Get Bash History") @@ -572,11 +552,11 @@ func captureBashHistory(podData *PodCreationData) error { log.Error(stdout.String()) history = stdout.String() - - // if options.PreserveWhitespace { - // return stdout.String(), stderr.String(), err - // } - // return strings.TrimSpace(stdout.String()), strings.TrimSpace(stderr.String()), err + + // if options.PreserveWhitespace { + // return stdout.String(), stderr.String(), err + // } + // return strings.TrimSpace(stdout.String()), strings.TrimSpace(stderr.String()), err return nil -} \ No newline at end of file +} diff --git a/src/jetstream/plugins/kubernetes/terminal/terminal.go b/src/jetstream/plugins/kubernetes/terminal/terminal.go new file mode 100644 index 0000000000..646fb46b45 --- /dev/null +++ b/src/jetstream/plugins/kubernetes/terminal/terminal.go @@ -0,0 +1,35 @@ +package terminal + +import ( + "github.com/cloudfoundry-incubator/stratos/src/jetstream/repository/interfaces" + "github.com/cloudfoundry-incubator/stratos/src/jetstream/repository/interfaces/config" + + log "github.com/sirupsen/logrus" +) + +// KubeTerminal supports spawning pods to provide a CLI environment to the user +type KubeTerminal struct { + Namespace string `configName:"STRATOS_KUBERNETES_NAMESPACE"` + Image string `configName:"STRATOS_KUBERNETES_TERMINAL_IMAGE"` + Token string `configName:"STRATOS_NAMESPACE"` +} + +// NewKubeTerminal checks that the environment is set up to support the Kube Terminal +func NewKubeTerminal(p interfaces.PortalProxy) *KubeTerminal { + + kt := &KubeTerminal{} + if err := config.Load(kt, p.Env().Lookup); err != nil { + log.Warn("Unable to load Kube Terminal configuration. %v", err) + return nil + } + + // Check that we have everything we need + if len(kt.Image) == 0 || len(kt.Namespace) == 0 || len(kt.Token) == 0 { + log.Warn("Kube Terminal configuration is not complete") + return nil + } + + log.Info("Kubernetes Terminal configured") + + return kt +} From ac7694234b13171a0840befa18e3acb7c86f5a06 Mon Sep 17 00:00:00 2001 From: Neil MacDougall Date: Tue, 2 Jun 2020 09:25:29 +0100 Subject: [PATCH 16/40] Add support for helm chart customizations --- deploy/kubernetes/build.sh | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/deploy/kubernetes/build.sh b/deploy/kubernetes/build.sh index 63cc4a0f63..6b2fcf7ada 100755 --- a/deploy/kubernetes/build.sh +++ b/deploy/kubernetes/build.sh @@ -221,6 +221,11 @@ fi log "-- Building Helm Chart" +# Apply any chart customizations added by a fork +if [ "${HAS_CUSTOM_BUILD}" == "true" ]; then + custom_helm_build +fi + # Don't change the chart in the repo, copy it and modify it locally SRC_HELM_CHART_PATH="${STRATOS_PATH}/deploy/kubernetes/console" From 799124c57544271726446a4b85b77db44fc57a97 Mon Sep 17 00:00:00 2001 From: Neil MacDougall Date: Tue, 2 Jun 2020 09:34:03 +0100 Subject: [PATCH 17/40] Removed change not needed --- deploy/kubernetes/build.sh | 5 ----- 1 file changed, 5 deletions(-) diff --git a/deploy/kubernetes/build.sh b/deploy/kubernetes/build.sh index 6b2fcf7ada..63cc4a0f63 100755 --- a/deploy/kubernetes/build.sh +++ b/deploy/kubernetes/build.sh @@ -221,11 +221,6 @@ fi log "-- Building Helm Chart" -# Apply any chart customizations added by a fork -if [ "${HAS_CUSTOM_BUILD}" == "true" ]; then - custom_helm_build -fi - # Don't change the chart in the repo, copy it and modify it locally SRC_HELM_CHART_PATH="${STRATOS_PATH}/deploy/kubernetes/console" From 7227dcaacf50d014fd9b6eb09f81788613ab6fb8 Mon Sep 17 00:00:00 2001 From: Neil MacDougall Date: Tue, 2 Jun 2020 09:39:50 +0100 Subject: [PATCH 18/40] Apply recreate-pods fix to SUSE additional containers --- custom-src/deploy/kubernetes/__stratos.tpl | 15 +++++++++++++++ custom-src/deploy/kubernetes/customize-helm.sh | 6 +++++- .../kubernetes/console/templates/chartsync.yaml | 3 +++ .../kubernetes/console/templates/deployment.yaml | 10 ---------- .../console/templates/fdbdoclayer/deployment.yaml | 4 ++++ 5 files changed, 27 insertions(+), 11 deletions(-) create mode 100644 custom-src/deploy/kubernetes/__stratos.tpl diff --git a/custom-src/deploy/kubernetes/__stratos.tpl b/custom-src/deploy/kubernetes/__stratos.tpl new file mode 100644 index 0000000000..3ebbce1efe --- /dev/null +++ b/custom-src/deploy/kubernetes/__stratos.tpl @@ -0,0 +1,15 @@ +# Customizations for the Helm Chart + +# Extra env vars for the Jetstream Pod in deployment.yaml +{{- define "stratosJetstreamEnv" }} +- name: MONOCULAR_CRT_PATH + value: "/etc/monocular-certs/tls.crt" +- name: MONOCULAR_KEY_PATH + value: "/etc/monocular-certs/tls.key" +- name: MONOCULAR_CA_CRT_PATH + value: "/etc/monocular-certs/ca.crt" +- name: FDB_URL + value: "mongodb://{{ .Release.Name }}-fdbdoclayer:27016" +- name: SYNC_SERVER_URL + value: "http://{{ .Release.Name }}-chartsync:8080" +{{- end }} \ No newline at end of file diff --git a/custom-src/deploy/kubernetes/customize-helm.sh b/custom-src/deploy/kubernetes/customize-helm.sh index 6284fe6c35..e57c571440 100755 --- a/custom-src/deploy/kubernetes/customize-helm.sh +++ b/custom-src/deploy/kubernetes/customize-helm.sh @@ -15,6 +15,11 @@ echo "Customizations folder: ${DIR}" echo "Chart folder : ${CHART_PATH}" echo "" +# =========================================================================================== +# Copy our customization helper over the default, empty one +# =========================================================================================== +cp "${DIR}/__stratos.tpl" "${CHART_PATH}/templates/__stratos.tpl" + # =========================================================================================== # Chart.yaml changes # =========================================================================================== @@ -45,7 +50,6 @@ echo -e "${CYAN}Patching README.md${RESET}" sed -i.bak -e 's@Stratos@SUSE Stratos Console@g' ${CHART_PATH}/README.md # Change first paragraph to include Kubernetes -console for Cloud Foundry. SRC="console for Cloud Foundry." DEST="console for Cloud Foundry and Kubernetes." sed -i.bak -e 's@'"${SRC}"'@'"${DEST}"'@g' ${CHART_PATH}/README.md diff --git a/deploy/kubernetes/console/templates/chartsync.yaml b/deploy/kubernetes/console/templates/chartsync.yaml index 2acedc0bca..1eeab2affd 100644 --- a/deploy/kubernetes/console/templates/chartsync.yaml +++ b/deploy/kubernetes/console/templates/chartsync.yaml @@ -34,6 +34,9 @@ spec: imagePullPolicy: {{.Values.imagePullPolicy}} command: ["/chartrepo"] args: ["serve", "--doclayer-url=mongodb://{{ .Release.Name }}-fdbdoclayer:27016", "--cafile=/etc/certs/ca.crt", "--certfile=/etc/certs/tls.crt", "--keyfile=/etc/certs/tls.key"] + env: + - name: STRATOS_IMAGE_REF + value: "{{.Values.consoleVersion}}:{{ .Release.Revision }}" ports: - name: endpoint containerPort: 8080 diff --git a/deploy/kubernetes/console/templates/deployment.yaml b/deploy/kubernetes/console/templates/deployment.yaml index 2b5f5a0949..60656b1d83 100644 --- a/deploy/kubernetes/console/templates/deployment.yaml +++ b/deploy/kubernetes/console/templates/deployment.yaml @@ -282,16 +282,6 @@ spec: value: {{ default "false" .Values.console.ui.listAllowLoadMaxed | quote }} {{- end }} {{- include "stratosJetstreamEnv" . | indent 8 }} - - name: MONOCULAR_CRT_PATH - value: "/etc/monocular-certs/tls.crt" - - name: MONOCULAR_KEY_PATH - value: "/etc/monocular-certs/tls.key" - - name: MONOCULAR_CA_CRT_PATH - value: "/etc/monocular-certs/ca.crt" - - name: FDB_URL - value: "mongodb://{{ .Release.Name }}-fdbdoclayer:27016" - - name: SYNC_SERVER_URL - value: "http://{{ .Release.Name }}-chartsync:8080" readinessProbe: httpGet: path: /pp/v1/ping diff --git a/deploy/kubernetes/console/templates/fdbdoclayer/deployment.yaml b/deploy/kubernetes/console/templates/fdbdoclayer/deployment.yaml index 63cdb0e7e8..4a1a02e06d 100644 --- a/deploy/kubernetes/console/templates/fdbdoclayer/deployment.yaml +++ b/deploy/kubernetes/console/templates/fdbdoclayer/deployment.yaml @@ -52,6 +52,8 @@ spec: image: {{.Values.kube.registry.hostname}}/{{.Values.kube.organization}}/{{.Values.images.fdbdoclayer}}:{{.Values.consoleVersion}} imagePullPolicy: {{.Values.imagePullPolicy}} env: + - name: STRATOS_IMAGE_REF + value: "{{.Values.consoleVersion}}:{{ .Release.Revision }}" - name: FDB_COORDINATOR value: {{ .Release.Name }}-fdbdoclayer - name: FDB_LISTEN_IP @@ -88,6 +90,8 @@ spec: image: {{.Values.kube.registry.hostname}}/{{.Values.kube.organization}}/{{.Values.images.fdbserver}}:{{.Values.consoleVersion}} imagePullPolicy: {{.Values.imagePullPolicy}} env: + - name: STRATOS_IMAGE_REF + value: "{{.Values.consoleVersion}}:{{ .Release.Revision }}" - name: FDB_COORDINATOR value: {{ .Release.Name }}-fdbdoclayer - name: FDB_LISTEN_IP From 76a13f5d149880b568293839bd4d28e45113d8bd Mon Sep 17 00:00:00 2001 From: Neil MacDougall Date: Wed, 3 Jun 2020 13:42:36 +0100 Subject: [PATCH 19/40] Kubernetes Terminal updates and tidy up --- deploy/common-build.sh | 15 +- .../console/templates/deployment.yaml | 3 + .../console/templates/service-account.yaml | 61 ++ src/jetstream/main.go | 3 + src/jetstream/plugins/kubernetes/api/api.go | 12 + src/jetstream/plugins/kubernetes/main.go | 9 +- src/jetstream/plugins/kubernetes/terminal.go | 562 ------------------ .../plugins/kubernetes/terminal/cleanup.go | 84 +++ .../plugins/kubernetes/terminal/helpers.go | 233 ++++++++ .../plugins/kubernetes/terminal/start.go | 254 ++++++++ .../plugins/kubernetes/terminal/terminal.go | 49 +- .../repository/interfaces/sessiondata.go | 4 + .../sessiondata/psql_sessiondata.go | 32 +- 13 files changed, 750 insertions(+), 571 deletions(-) create mode 100644 deploy/kubernetes/console/templates/service-account.yaml create mode 100644 src/jetstream/plugins/kubernetes/api/api.go delete mode 100644 src/jetstream/plugins/kubernetes/terminal.go create mode 100644 src/jetstream/plugins/kubernetes/terminal/cleanup.go create mode 100644 src/jetstream/plugins/kubernetes/terminal/helpers.go create mode 100644 src/jetstream/plugins/kubernetes/terminal/start.go diff --git a/deploy/common-build.sh b/deploy/common-build.sh index 2e8906adf4..11fb464ddc 100644 --- a/deploy/common-build.sh +++ b/deploy/common-build.sh @@ -43,7 +43,7 @@ function buildAndPublishImage { # Proxy support # Remove intermediate containers after a successful build -BUILD_ARGS="--rm=true --squash" +BUILD_ARGS="--rm=true" RUN_ARGS="" if [ -n "${http_proxy:-}" -o -n "${HTTP_PROXY:-}" ]; then BUILD_ARGS="${BUILD_ARGS} --build-arg http_proxy=${http_proxy:-${HTTP_PROXY}}" @@ -54,6 +54,19 @@ if [ -n "${https_proxy:-}" -o -n "${HTTPS_PROXY:-}" ]; then RUN_ARGS="${RUN_ARGS} -e https_proxy=${https_proxy:-${HTTPS_PROXY}}" fi +# Check if we can squash +echo "checking" +CAN_SQUASH=$(docker info 2>&1 | grep "Experimental: true" | cat) +echo "check" +if [ "${CAN_SQUASH}" == "1" ]; then + BUILD_ARGS="${BUILD_ARGS} --squash" + echo "Images will be squashed" +else + echo "Images will NOT be squashed" +fi + +echo "OK" + # Use correct sed command for Mac SED="sed -r" unamestr=`uname` diff --git a/deploy/kubernetes/console/templates/deployment.yaml b/deploy/kubernetes/console/templates/deployment.yaml index 6792a1ebf0..094a6ed78c 100644 --- a/deploy/kubernetes/console/templates/deployment.yaml +++ b/deploy/kubernetes/console/templates/deployment.yaml @@ -308,6 +308,9 @@ spec: imagePullSecrets: - name: {{.Values.dockerRegistrySecret}} {{- end }} + {{- if and (eq (printf "%s" .Values.kube.auth) "rbac") (.Capabilities.APIVersions.Has "rbac.authorization.k8s.io/v1") }} + serviceAccountName: "stratos" + {{- end }} volumes: - name: "{{ .Release.Name }}-encryption-key-volume" secret: diff --git a/deploy/kubernetes/console/templates/service-account.yaml b/deploy/kubernetes/console/templates/service-account.yaml new file mode 100644 index 0000000000..d8ef0e07d7 --- /dev/null +++ b/deploy/kubernetes/console/templates/service-account.yaml @@ -0,0 +1,61 @@ +{{- if and (eq (printf "%s" .Values.kube.auth) "rbac") (.Capabilities.APIVersions.Has "rbac.authorization.k8s.io/v1") }} +--- +# Service account main Stratos Deployment +# Allows it to create some resources in its namespace +apiVersion: "v1" +kind: "ServiceAccount" +metadata: + name: "stratos" + labels: + app.kubernetes.io/component: "stratos" + app.kubernetes.io/instance: "{{ .Release.Name }}" + app.kubernetes.io/name: "stratos" + app.kubernetes.io/version: "{{ .Chart.AppVersion }}" + helm.sh/chart: "{{ .Chart.Name }}-{{ .Chart.Version | replace "+" "_" }}" +--- +# Role "stratos-role" only used by account "stratos" +apiVersion: "rbac.authorization.k8s.io/v1" +kind: "Role" +metadata: + name: "stratos-role" + labels: + app.kubernetes.io/component: "stratos-role" + app.kubernetes.io/instance: "{{ .Release.Name }}" + app.kubernetes.io/name: "stratos" + app.kubernetes.io/version: "{{ .Chart.AppVersion }}" + helm.sh/chart: "{{ .Chart.Name }}-{{ .Chart.Version | replace "+" "_" }}" +rules: +- apiGroups: + - "" + resources: + - "secrets" + - "pods" + verbs: + - "create" + - "update" + - "get" + - "list" + - "delete" +- apiGroups: [""] + resources: ["pods/exec"] + verbs: ["create", "get"] +--- +# Role binding for service account "stratos" and role "stratos-role" +apiVersion: "rbac.authorization.k8s.io/v1" +kind: "RoleBinding" +metadata: + name: "stratos-role-binding" + labels: + app.kubernetes.io/component: "stratos-role-binding" + app.kubernetes.io/instance: "{{ .Release.Name }}" + app.kubernetes.io/name: "stratos" + app.kubernetes.io/version: "{{ .Chart.AppVersion }}" + helm.sh/chart: "{{ .Chart.Name }}-{{ .Chart.Version | replace "+" "_" }}" +subjects: +- kind: "ServiceAccount" + name: "stratos" +roleRef: + apiGroup: "rbac.authorization.k8s.io" + kind: "Role" + name: "stratos-role" +{{- end }} diff --git a/src/jetstream/main.go b/src/jetstream/main.go index 9b649da31d..f6f90c60c6 100644 --- a/src/jetstream/main.go +++ b/src/jetstream/main.go @@ -10,6 +10,7 @@ import ( "fmt" "io" "io/ioutil" + "math/rand" "net" "net/http" "os" @@ -115,6 +116,8 @@ func main() { } } + rand.Seed(time.Now().UnixNano()) + log.SetOutput(os.Stdout) log.Info("========================================") diff --git a/src/jetstream/plugins/kubernetes/api/api.go b/src/jetstream/plugins/kubernetes/api/api.go new file mode 100644 index 0000000000..b15fc1b3c3 --- /dev/null +++ b/src/jetstream/plugins/kubernetes/api/api.go @@ -0,0 +1,12 @@ +package api + +import ( + "github.com/cloudfoundry-incubator/stratos/src/jetstream/repository/interfaces" + + restclient "k8s.io/client-go/rest" +) + +type Kubernetes interface { + GetConfigForEndpoint(masterURL string, token interfaces.TokenRecord) (*restclient.Config, error) + GetKubeConfigForEndpoint(masterURL string, token interfaces.TokenRecord, namespace string) (string, error) +} diff --git a/src/jetstream/plugins/kubernetes/main.go b/src/jetstream/plugins/kubernetes/main.go index 5e12edf9c6..bca169a26c 100644 --- a/src/jetstream/plugins/kubernetes/main.go +++ b/src/jetstream/plugins/kubernetes/main.go @@ -59,7 +59,9 @@ const ( // Init creates a new instance of the Kubernetes plugin func Init(portalProxy interfaces.PortalProxy) (interfaces.StratosPlugin, error) { kubeTerminal := terminal.NewKubeTerminal(portalProxy) - return &KubernetesSpecification{portalProxy: portalProxy, endpointType: kubeEndpointType, kubeTerminal: kubeTerminal}, nil + kube := &KubernetesSpecification{portalProxy: portalProxy, endpointType: kubeEndpointType, kubeTerminal: kubeTerminal} + kubeTerminal.Kube = kube + return kube, nil } func (c *KubernetesSpecification) GetEndpointPlugin() (interfaces.EndpointPlugin, error) { @@ -136,6 +138,9 @@ func (c *KubernetesSpecification) Init() error { // Kube dashboard is enabled by Tech Preview mode c.portalProxy.GetConfig().PluginConfig[kubeDashboardPluginConfigSetting] = strconv.FormatBool(c.portalProxy.GetConfig().EnableTechPreview) + // Kick off the cleanup of any old kube terminal pods + c.kubeTerminal.StartCleanup() + return nil } @@ -165,7 +170,7 @@ func (c *KubernetesSpecification) AddSessionGroupRoutes(echoGroup *echo.Group) { echoGroup.GET("/helm/releases/:endpoint/:namespace/:name", c.GetRelease) // Kube Terminal - echoGroup.GET("/kubeconsole/:guid", c.KubeTerminal) + echoGroup.GET("/kubeconsole/:guid", c.kubeTerminal.Start) } diff --git a/src/jetstream/plugins/kubernetes/terminal.go b/src/jetstream/plugins/kubernetes/terminal.go deleted file mode 100644 index c1a98d5496..0000000000 --- a/src/jetstream/plugins/kubernetes/terminal.go +++ /dev/null @@ -1,562 +0,0 @@ -package kubernetes - -import ( - "bytes" - "crypto/tls" - "errors" - "fmt" - - //"encoding/base64" - "encoding/json" - "net" - "net/http" - "strings" - "time" - - "github.com/labstack/echo" - uuid "github.com/satori/go.uuid" - log "github.com/sirupsen/logrus" - - "github.com/cloudfoundry-incubator/stratos/src/jetstream/plugins/kubernetes/auth" - "github.com/cloudfoundry-incubator/stratos/src/jetstream/repository/interfaces" - restclient "k8s.io/client-go/rest" - - "github.com/gorilla/websocket" - v1 "k8s.io/api/core/v1" - metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" - "k8s.io/client-go/kubernetes" - corev1 "k8s.io/client-go/kubernetes/typed/core/v1" - - scheme "k8s.io/client-go/kubernetes/scheme" - "k8s.io/client-go/tools/remotecommand" -) - -const ( - consoleContainerName = "kube-console" -) - -var history = "" - -// TTY Resize, see: https://gitlab.cncf.ci/kubernetes/kubernetes/commit/3b21a9901bcd48bb452d3bf1a0cddc90dae142c4#9691a2f9b9c30711f0397221db0b9ac55ab0e2d1 - -// Allow connections from any Origin -var upgrader = websocket.Upgrader{ - CheckOrigin: func(r *http.Request) bool { return true }, -} - -// PodCreationData stores the clients and names used ot create pod and secret -type PodCreationData struct { - ClientSet *kubernetes.Clientset - Config *restclient.Config - Namespace string - PodClient corev1.PodInterface - SecretClient corev1.SecretInterface - PodName string - SecretName string -} - -// KeyCode - JSON object that is passed from the front-end to notify of a key press or a term resize -type KeyCode struct { - Key string `json:"key"` - Cols int `json:"cols"` - Rows int `json:"rows"` -} - -type terminalSize struct { - Width uint16 - Height uint16 -} - -const ( - // Time allowed to write a message to the peer. - writeWait = 10 * time.Second -) - -// Determine if we Kube Terminal is supported -// TODO -// Check namespace -// Check we have secret for pod creation/deletion -// Image for the terminal - -// Environment variables that we expect to see -// STRATOS_KUBERNETES_NAMESPACE -// STRATOS_KUBERNETES_TERMINAL_IMAGE - -// For development, you can overide the token to use with the Kubernetes API, with: -// STRATOS_KUBERNETES_API_TOKEN - -// Start periodic routing to clean up pods that are orphaned -// TODO - -// KubeTerminal handles web-socket request to launch a Kubernetes Terminal -func (k *KubernetesSpecification) KubeTerminal(c echo.Context) error { - - c.Response().Status = 500 - - log.Info("Kube Console backend request") - - endpointGUID := c.Param("guid") - userGUID := c.Get("user_id").(string) - - ///api/v1/namespaces/project-1/pods/pod-1-lmlzj/exec?command=/bin/bash&stdin=true&stderr=true&stdout=true&tty=true - - namespace := "stratos" - - // TODO: Refresh auth token - - // Upgrade the web socket for the incoming request - ws, pingTicker, err := interfaces.UpgradeToWebSocket(c) - if err != nil { - return err - } - defer ws.Close() - defer pingTicker.Stop() - - // Send a message to say that we are creating the pod - sendProgressMessage(ws, "Launching Kubernetes Console ... one moment please") - - var p = k.portalProxy - - cnsiRecord, err := p.GetCNSIRecord(endpointGUID) - if err != nil { - return errors.New("Could not get endpoint information") - } - - // Get token for this users - tokenRec, ok := p.GetCNSITokenRecord(endpointGUID, userGUID) - if !ok { - return errors.New("Could not get token") - } - - podData, err := k.createPod(c, cnsiRecord, tokenRec, namespace, ws) - // Clear progress message - sendProgressMessage(ws, "") - - if err != nil { - log.Error("ERROR creating secret or pod") - log.Info(err) - k.cleanupPodAndSecret(podData) - - // Send error message - sendProgressMessage(ws, "!"+err.Error()) - return err - } - - log.Info(podData.PodName) - - log.Info(tokenRec.AuthToken) - log.Info(tokenRec.AuthType) - - // Make the info call to the SSH endpoint info - // Currently this is not cached, so we must get it each time - apiEndpoint := cnsiRecord.APIEndpoint - log.Info(apiEndpoint) - // target := fmt.Sprintf("%s/api/v1/namespaces/kube-system/services/https:kubernetes-dashboard:/proxy/", apiEndpoint) - //target := fmt.Sprintf("%s/api/v1/namespaces/kube-system/services/http:kubernetes-dashboard:/proxy/", apiEndpoint) - // target := http://localhost:8001/api/v1/namespaces/kube-system/services/kubernetes-dashboard/proxy - target := fmt.Sprintf("%s/api/v1/namespaces/%s/pods/%s/exec?command=/bin/bash&stdin=true&stderr=true&stdout=true&tty=true", apiEndpoint, namespace, podData.PodName) - log.Info(target) - - // config, err := k.getConfig(&cnsiRecord, &tokenRec) - // if err != nil { - // return errors.New("Could not get config for this auth type") - // } - - // req, err := http.NewRequest("POST", target, nil) - // if err != nil { - // k.cleanupPodAndSecret(podData) - // return errors.New("Could not create new HTTP request") - // } - - // // TODO: Check response code - // log.Info(re) - - // // Set auth header so we log in if needed - // if len(tokenRec.AuthToken) > 0 { - // //req.Header.Add("Authorization", "Bearer "+tokenRec.AuthToken) - // log.Info("Setting auth header") - // } - - // endpointRequest := &interfaces.CNSIRequest{ - // GUID: endpointGUID, - // } - - // TODO: Make generic - - // What is the token for ??? - kubeAuthToken := &auth.KubeCertificate{} - err = json.NewDecoder(strings.NewReader(tokenRec.AuthToken)).Decode(kubeAuthToken) - if err != nil { - k.cleanupPodAndSecret(podData) - return err - } - cert, err := kubeAuthToken.GetCerticate() - if err != nil { - k.cleanupPodAndSecret(podData) - return err - } - dial := (&net.Dialer{ - Timeout: time.Duration(30) * time.Second, - KeepAlive: 30 * time.Second, - }).Dial - - // TODO: Not sure about the cert here? I don't think we need this like it is - wonder why I did that? - - sslTransport := &http.Transport{ - Proxy: http.ProxyFromEnvironment, - Dial: dial, - TLSHandshakeTimeout: 10 * time.Second, // 10 seconds is a sound default value (default is 0) - TLSClientConfig: &tls.Config{ - InsecureSkipVerify: true, - Certificates: []tls.Certificate{cert}, - }, - MaxIdleConnsPerHost: 6, // (default is 2) - } - - kubeHTTPClient := http.Client{} - kubeHTTPClient.Transport = sslTransport - kubeHTTPClient.Timeout = time.Duration(30) * time.Second - - // This dialer does not use the kubeHttpClient - is it unused? - dialer := &websocket.Dialer{ - TLSClientConfig: &tls.Config{ - InsecureSkipVerify: true, - Certificates: []tls.Certificate{cert}, - }, - } - - if strings.HasPrefix(target, "https://") { - target = "wss://" + target[8:] - } else { - target = "ws://" + target[7:] - } - - header := &http.Header{} - wsConn, res, err := dialer.Dial(target, *header) - - if err == nil { - defer wsConn.Close() - } - - // Check res - log.Warn("== Status ===========================================") - log.Info(res.Status) - log.Info(res.StatusCode) - - kubeHTTPClient.CloseIdleConnections() - - if err != nil { - k.cleanupPodAndSecret(podData) - return errors.New("Could not make request") - } - - stdoutDone := make(chan struct{}) - go pumpStdout(ws, wsConn, stdoutDone) - - // Read the input from the web socket and pipe it to the SSH client - for { - _, r, err := ws.ReadMessage() - if err != nil { - log.Errorf("Kubernetes terminal: error reading message from web socket: %+v", err) - k.cleanupPodAndSecret(podData) - return err - } - - res := KeyCode{} - json.Unmarshal(r, &res) - - if res.Cols == 0 { - slice := make([]byte, 1) - slice[0] = 0 - slice = append(slice, []byte(res.Key)...) - wsConn.WriteMessage(websocket.TextMessage, slice) - } else { - size := terminalSize{ - Width: uint16(res.Cols), - Height: uint16(res.Rows), - } - j, _ := json.Marshal(size) - resizeStream := []byte{4} - slice := append(resizeStream, j...) - wsConn.WriteMessage(websocket.TextMessage, slice) - } - } - - // Cleanup - log.Debug("Kubernetes Terminal is cleaning up") - - // This is unreachable???? - return k.cleanupPodAndSecret(podData) -} - -func pumpStdout(ws *websocket.Conn, source *websocket.Conn, done chan struct{}) { - //buffer := make([]byte, 32768) - for { - _, r, err := source.ReadMessage() - if err != nil { - log.Info(err) - ws.Close() - break - } - ws.SetWriteDeadline(time.Now().Add(writeWait)) - bytes := fmt.Sprintf("% x\n", r[1:]) - if err := ws.WriteMessage(websocket.TextMessage, []byte(bytes)); err != nil { - log.Error("Kubernetes Terimnal failed to write nessage") - ws.Close() - break - } - } -} - -func (k *KubernetesSpecification) createPod(c echo.Context, cnsiRecord interfaces.CNSIRecord, tokenRecord interfaces.TokenRecord, namespace string, ws *websocket.Conn) (*PodCreationData, error) { - - id := uuid.NewV4().String() - secretName := fmt.Sprintf("k8s-s-console-%s", id) - podName := fmt.Sprintf("k8s-p-console-%s", id) - - result := &PodCreationData{} - - config, err := k.GetConfigForEndpoint(cnsiRecord.APIEndpoint.String(), tokenRecord) - if err != nil { - return result, errors.New("Can not get Kubernetes config for specified endpoint") - } - - result.Config = config - - kubeConfig, err := k.GetKubeConfigForEndpoint(cnsiRecord.APIEndpoint.String(), tokenRecord, namespace) - if err != nil { - return result, errors.New("Can not get Kubernetes config for specified endpoint") - } - - kubeClient, err := kubernetes.NewForConfig(config) - if err != nil { - log.Error("Could not get kube client") - return result, err - } - - result.Namespace = namespace - result.ClientSet = kubeClient - - // Create the secret - secretSpec := &v1.Secret{ - TypeMeta: metav1.TypeMeta{ - Kind: "secret", - APIVersion: "v1", - }, - ObjectMeta: metav1.ObjectMeta{ - Name: secretName, - Namespace: namespace, - }, - Type: "Opaque", - } - - secretSpec.Data = make(map[string][]byte) - secretSpec.Data["kubeconfig"] = []byte(kubeConfig) - secretSpec.Data["history"] = []byte(history) - - // Get Helm repository script if we have Helm repositories - helmSetup := getHelmRepoSetupScript(k.portalProxy) - if len(helmSetup) > 0 { - secretSpec.Data["helm-setup"] = []byte(helmSetup) - } - - secretsClient := kubeClient.CoreV1().Secrets(namespace) - _, err = secretsClient.Create(secretSpec) - - if err != nil { - log.Warn("Unable to create Secret") - return result, err - } - - result.SecretClient = secretsClient - result.SecretName = secretName - - podClient := kubeClient.CoreV1().Pods(namespace) - - podSpec := &v1.Pod{ - TypeMeta: metav1.TypeMeta{ - Kind: "pod", - APIVersion: "v1", - }, - ObjectMeta: metav1.ObjectMeta{ - Name: podName, - Namespace: namespace, - }, - } - - podSpec.ObjectMeta.Annotations = make(map[string]string) - - // Record the session ID - session, err := k.portalProxy.GetSession(c) - if err == nil { - podSpec.ObjectMeta.Annotations["stratos-session"] = session.ID - log.Infof("Session ID: %s", session.ID) - } - - automount := false - - podSpec.Spec.AutomountServiceAccountToken = &automount - podSpec.Spec.RestartPolicy = "Never" - - volumeMountsSpec := make([]v1.VolumeMount, 1) - volumeMountsSpec[0].Name = "kubeconfig" - volumeMountsSpec[0].MountPath = "/root/.stratos" - volumeMountsSpec[0].ReadOnly = true - - containerSpec := make([]v1.Container, 1) - containerSpec[0].Name = consoleContainerName - containerSpec[0].Image = "nwmac/kubeconsole" - containerSpec[0].ImagePullPolicy = "Always" - containerSpec[0].VolumeMounts = volumeMountsSpec - podSpec.Spec.Containers = containerSpec - - volumesSpec := make([]v1.Volume, 1) - volumesSpec[0].Name = "kubeconfig" - volumesSpec[0].Secret = &v1.SecretVolumeSource{ - SecretName: secretName, - } - podSpec.Spec.Volumes = volumesSpec - - // Create a new pod - pod, err := podClient.Create(podSpec) - if err != nil { - return result, err - } - - result.PodClient = podClient - result.PodName = podName - - sendProgressMessage(ws, "Waiting for Kubernetes Terminal to start up ...") - - statusOptions := metav1.GetOptions{} - - // Wait for the pod to be running - ready := false - for { - status, err := podClient.Get(pod.Name, statusOptions) - - if err != nil { - break - } - - //log.Info(status.Status.Phase) - if status.Status.Phase == "Running" { - ready = true - } - - if ready { - break - } - - // Sleep - time.Sleep(2500 * time.Millisecond) - } - - return result, err -} - -func (k *KubernetesSpecification) cleanupPodAndSecret(podData *PodCreationData) error { - - if len(podData.PodName) > 0 { - captureBashHistory(podData) - podData.PodClient.Delete(podData.PodName, nil) - } - - if len(podData.SecretName) > 0 { - podData.SecretClient.Delete(podData.SecretName, nil) - } - - return nil -} - -func getHelmRepoSetupScript(portalProxy interfaces.PortalProxy) string { - - str := "" - - // Get all of the helm endpoints - endpoints, err := portalProxy.ListEndpoints() - if err != nil { - log.Error("Can not list Helm Repository endpoints") - return str - } - - for _, ep := range endpoints { - if ep.CNSIType == "helm" { - str += fmt.Sprintf("helm repo add %s %s > /dev/null\n", ep.Name, ep.APIEndpoint) - } - } - - return str -} - -func sendProgressMessage(ws *websocket.Conn, progressMsg string) { - // Send a message to say that we are creating the pod - msg := fmt.Sprintf("\033]2;%s\007", progressMsg) - bytes := fmt.Sprintf("% x\n", []byte(msg)) - if err := ws.WriteMessage(websocket.TextMessage, []byte(bytes)); err != nil { - log.Error("Could not send message to client to indicate console is starting") - } -} - -// Not used? -func captureBashHistory(podData *PodCreationData) error { - - log.Warn("**** Trying to capture bash history from pod ****") - - // Can we capture the history? - - // returning stdout, stderr and error. `options` allowed for - // additional parameters to be passed. - - //cmd := []string{"bash", "-c", "\"cat $HISTFILE\""} - cmd := []string{"cat", "/root/.bash_history"} - - req := podData.ClientSet.CoreV1().RESTClient().Post(). - Resource("pods"). - Name(podData.PodName). - Namespace(podData.Namespace). - SubResource("exec"). - Param("container", consoleContainerName) - - req.VersionedParams(&v1.PodExecOptions{ - Container: consoleContainerName, - Command: cmd, - Stdin: false, - Stdout: true, - Stderr: false, - TTY: true, - }, scheme.ParameterCodec) - - var stdout bytes.Buffer - //err := execute("POST", req.URL(), nil, nil, &stdout, nil, false) - - exec, err := remotecommand.NewSPDYExecutor(podData.Config, "POST", req.URL()) - if err != nil { - log.Error("Could not exec") - log.Error(err) - return err - } - - log.Warn("Attempting stream ......") - - err = exec.Stream(remotecommand.StreamOptions{ - //SupportedProtocols: remotecommandserver.SupportedStreamingProtocols, - Stdin: nil, - Stdout: &stdout, - Stderr: nil, - Tty: true, - }) - - log.Error("Get Bash History") - log.Error(err) - log.Error(stdout.String()) - - history = stdout.String() - - // if options.PreserveWhitespace { - // return stdout.String(), stderr.String(), err - // } - // return strings.TrimSpace(stdout.String()), strings.TrimSpace(stderr.String()), err - - return nil -} diff --git a/src/jetstream/plugins/kubernetes/terminal/cleanup.go b/src/jetstream/plugins/kubernetes/terminal/cleanup.go new file mode 100644 index 0000000000..8426385b4f --- /dev/null +++ b/src/jetstream/plugins/kubernetes/terminal/cleanup.go @@ -0,0 +1,84 @@ +package terminal + +import ( + "math/rand" + "strconv" + "time" + + log "github.com/sirupsen/logrus" + + metaV1 "k8s.io/apimachinery/pkg/apis/meta/v1" +) + +// Wait time in minutes after intiial wait +const waitPeriod = 10 + +// StartCleanup starts a backgrdound routine to cleanup orphaned pods +func (k *KubeTerminal) StartCleanup() { + go k.cleanup() +} + +func (k *KubeTerminal) cleanup() { + // Use a random initial wait before cleaning up + // If we had moer backends, this helps to ensure they are not all trying to cleanup at the same time + wait := rand.Intn(30) + + // Testing + wait = 0 + for { + time.Sleep(time.Duration(wait) * time.Minute) + log.Debug("Cleaning up stale Kubernetes Terminal pods and secrets ...") + + // Get all pods with a given label + podClient, secretClient, err := k.getClients() + if err == nil { + // Only want the pods that are kube terminals + options := metaV1.ListOptions{} + options.LabelSelector = "stratos-role=kube-terminal" + pods, err := podClient.List(options) + if err == nil { + for _, pod := range pods.Items { + if sessionID, ok := pod.Annotations["stratos-session"]; ok { + i, err := strconv.Atoi(sessionID) + if err == nil { + isValid, err := k.PortalProxy.GetSessionDataStore().IsValidSession(i) + if err == nil && !isValid { + log.Debugf("Deleting pod %s", pod.Name) + podClient.Delete(pod.Name, nil) + } + } + } + } + } else { + log.Debug("Kube Terminal Cleanup: Could not get pods") + log.Debug(err) + } + + // Only want the secrets that are kube terminals + secrets, err := secretClient.List(options) + if err == nil { + for _, secret := range secrets.Items { + if sessionID, ok := secret.Annotations["stratos-session"]; ok { + i, err := strconv.Atoi(sessionID) + if err == nil { + isValid, err := k.PortalProxy.GetSessionDataStore().IsValidSession(i) + if err == nil && !isValid { + log.Debugf("Deleting secret %s", secret.Name) + secretClient.Delete(secret.Name, nil) + } + } + } + } + } else { + log.Debug("Kube Terminal Cleanup: Could not get secrets") + log.Debug(err) + } + + } else { + log.Debug("Kube Terminal Cleanup: Could not get clients") + log.Debug(err) + } + + wait = waitPeriod + } +} diff --git a/src/jetstream/plugins/kubernetes/terminal/helpers.go b/src/jetstream/plugins/kubernetes/terminal/helpers.go new file mode 100644 index 0000000000..7e455a926b --- /dev/null +++ b/src/jetstream/plugins/kubernetes/terminal/helpers.go @@ -0,0 +1,233 @@ +package terminal + +import ( + "errors" + "fmt" + "strings" + "time" + + "github.com/labstack/echo" + uuid "github.com/satori/go.uuid" + log "github.com/sirupsen/logrus" + + "github.com/cloudfoundry-incubator/stratos/src/jetstream/plugins/kubernetes/auth" + "github.com/cloudfoundry-incubator/stratos/src/jetstream/repository/interfaces" + + "github.com/gorilla/websocket" + v1 "k8s.io/api/core/v1" + metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" + "k8s.io/client-go/kubernetes" + corev1 "k8s.io/client-go/kubernetes/typed/core/v1" +) + +// PodCreationData stores the clients and names used to create pod and secret +type PodCreationData struct { + Namespace string + PodClient corev1.PodInterface + SecretClient corev1.SecretInterface + PodName string + SecretName string +} + +func (k *KubeTerminal) getClients() (corev1.PodInterface, corev1.SecretInterface, error) { + + // Create a token record for Token Auth using the Service Account token + token := auth.NewKubeTokenAuthTokenRecord(k.PortalProxy, string(k.Token)) + config, err := k.Kube.GetConfigForEndpoint(k.APIServer, *token) + if err != nil { + return nil, nil, errors.New("Can not get Kubernetes config for specified endpoint") + } + kubeClient, err := kubernetes.NewForConfig(config) + if err != nil { + log.Error("Could not get kube client") + return nil, nil, err + } + + podClient := kubeClient.CoreV1().Pods(k.Namespace) + secretsClient := kubeClient.CoreV1().Secrets(k.Namespace) + return podClient, secretsClient, nil +} + +// Create a pod for a user to run the Kube terminal +func (k *KubeTerminal) createPod(c echo.Context, kubeConfig string, ws *websocket.Conn) (*PodCreationData, error) { + // Unique ID for the secret and pod name + id := uuid.NewV4().String() + id = strings.ReplaceAll(id, "-", "") + // Names for the secret and pod + secretName := fmt.Sprintf("terminal-%s", id) + podName := secretName + + podClient, secretClient, err := k.getClients() + + result := &PodCreationData{} + result.Namespace = k.Namespace + + // Get the session ID + sessionID := "" + session, err := k.PortalProxy.GetSession(c) + if err == nil { + sessionID = session.ID + } + + // Create the secret + secretSpec := &v1.Secret{ + TypeMeta: metav1.TypeMeta{ + Kind: "secret", + APIVersion: "v1", + }, + ObjectMeta: metav1.ObjectMeta{ + Name: secretName, + Namespace: k.Namespace, + }, + Type: "Opaque", + } + + setResourcMetadata(&secretSpec.ObjectMeta, sessionID) + + secretSpec.Data = make(map[string][]byte) + secretSpec.Data["kubeconfig"] = []byte(kubeConfig) + //secretSpec.Data["history"] = []byte(history) + + // Get Helm repository script if we have Helm repositories + helmSetup := getHelmRepoSetupScript(k.PortalProxy) + if len(helmSetup) > 0 { + secretSpec.Data["helm-setup"] = []byte(helmSetup) + } + + _, err = secretClient.Create(secretSpec) + if err != nil { + log.Warn("Unable to create Secret") + return result, err + } + + result.SecretClient = secretClient + result.SecretName = secretName + + // Pod + podSpec := &v1.Pod{ + TypeMeta: metav1.TypeMeta{ + Kind: "pod", + APIVersion: "v1", + }, + ObjectMeta: metav1.ObjectMeta{ + Name: podName, + Namespace: k.Namespace, + }, + } + + // Label the pod, so we can find it as a kube terminal pod + setResourcMetadata(&podSpec.ObjectMeta, sessionID) + + // Don't mount a service account token + off := false + podSpec.Spec.AutomountServiceAccountToken = &off + podSpec.Spec.EnableServiceLinks = &off + podSpec.Spec.RestartPolicy = "Never" + podSpec.Spec.DNSPolicy = "Default" + + volumeMountsSpec := make([]v1.VolumeMount, 1) + volumeMountsSpec[0].Name = "kubeconfig" + volumeMountsSpec[0].MountPath = "/root/.stratos" + volumeMountsSpec[0].ReadOnly = true + + containerSpec := make([]v1.Container, 1) + containerSpec[0].Name = consoleContainerName + containerSpec[0].Image = k.Image + containerSpec[0].ImagePullPolicy = "Always" + containerSpec[0].VolumeMounts = volumeMountsSpec + podSpec.Spec.Containers = containerSpec + + volumesSpec := make([]v1.Volume, 1) + volumesSpec[0].Name = "kubeconfig" + volumesSpec[0].Secret = &v1.SecretVolumeSource{ + SecretName: secretName, + } + podSpec.Spec.Volumes = volumesSpec + + // Create a new pod + pod, err := podClient.Create(podSpec) + if err != nil { + // Secret will get cleaned up by caller + return result, err + } + + result.PodClient = podClient + result.PodName = podName + + sendProgressMessage(ws, "Waiting for Kubernetes Terminal to start up ...") + + // Wait for the pod to be running + timeout := 60 + statusOptions := metav1.GetOptions{} + for { + status, err := podClient.Get(pod.Name, statusOptions) + if err == nil && status.Status.Phase == "Running" { + log.Error("Pod is ready !") + break; + } + + timeout = timeout - 1 + if timeout == 0 { + err = errors.New("Timed out waiting for pod to enter ready state") + break + } + + // Sleep + time.Sleep(1500 * time.Millisecond) + } + + return result, err +} + +func setResourcMetadata(metadata *metav1.ObjectMeta, sessionID string) { + // Label the kubeerntes resource, so we can find it as a kube terminal pod + metadata.Labels = make(map[string]string) + metadata.Labels[stratosRoleLabel] = stratosKubeTerminalRole + metadata.Annotations = make(map[string]string) + if len(sessionID) > 0 { + metadata.Annotations[stratosSessionAnnotation] = sessionID + } +} + +// Cleanup the pod and secret +func (k *KubeTerminal) cleanupPodAndSecret(podData *PodCreationData) error { + if len(podData.PodName) > 0 { + //captureBashHistory(podData) + podData.PodClient.Delete(podData.PodName, nil) + } + + if len(podData.SecretName) > 0 { + podData.SecretClient.Delete(podData.SecretName, nil) + } + + return nil +} + + +func getHelmRepoSetupScript(portalProxy interfaces.PortalProxy) string { + str := "" + + // Get all of the helm endpoints + endpoints, err := portalProxy.ListEndpoints() + if err != nil { + log.Error("Can not list Helm Repository endpoints") + return str + } + + for _, ep := range endpoints { + if ep.CNSIType == "helm" { + str += fmt.Sprintf("helm repo add %s %s > /dev/null\n", ep.Name, ep.APIEndpoint) + } + } + + return str +} + +func sendProgressMessage(ws *websocket.Conn, progressMsg string) { + // Send a message to say that we are creating the pod + msg := fmt.Sprintf("\033]2;%s\007", progressMsg) + bytes := fmt.Sprintf("% x\n", []byte(msg)) + if err := ws.WriteMessage(websocket.TextMessage, []byte(bytes)); err != nil { + log.Error("Could not send message to client to indicate terminal is starting") + } +} diff --git a/src/jetstream/plugins/kubernetes/terminal/start.go b/src/jetstream/plugins/kubernetes/terminal/start.go new file mode 100644 index 0000000000..b59e4e0a37 --- /dev/null +++ b/src/jetstream/plugins/kubernetes/terminal/start.go @@ -0,0 +1,254 @@ +package terminal + +import ( + "crypto/tls" + "errors" + "fmt" + + //"encoding/base64" + "encoding/json" + "net" + "net/http" + "strings" + "time" + + "github.com/labstack/echo" + log "github.com/sirupsen/logrus" + + "github.com/cloudfoundry-incubator/stratos/src/jetstream/repository/interfaces" + + "github.com/gorilla/websocket" +) + +// TTY Resize, see: https://gitlab.cncf.ci/kubernetes/kubernetes/commit/3b21a9901bcd48bb452d3bf1a0cddc90dae142c4#9691a2f9b9c30711f0397221db0b9ac55ab0e2d1 + +// Allow connections from any Origin +var upgrader = websocket.Upgrader{ + CheckOrigin: func(r *http.Request) bool { return true }, +} + + +// KeyCode - JSON object that is passed from the front-end to notify of a key press or a term resize +type KeyCode struct { + Key string `json:"key"` + Cols int `json:"cols"` + Rows int `json:"rows"` +} + +type terminalSize struct { + Width uint16 + Height uint16 +} + +const ( + // Time allowed to write a message to the peer. + writeWait = 10 * time.Second +) + + +// Determine if we Kube Terminal is supported +// TODO +// Check namespace +// Check we have secret for pod creation/deletion +// Image for the terminal + +// Environment variables that we expect to see +// STRATOS_KUBERNETES_NAMESPACE +// STRATOS_KUBERNETES_TERMINAL_IMAGE + +// For development, you can overide the token to use with the Kubernetes API, with: +// STRATOS_KUBERNETES_API_TOKEN + +// Start periodic routing to clean up pods that are orphaned +// TODO + +// Start handles web-socket request to launch a Kubernetes Terminal +func (k *KubeTerminal) Start(c echo.Context) error { + + var p = k.PortalProxy + + c.Response().Status = 500 + + log.Info("Kube Terminal backend request") + + // TODO: We will need these ?? + + endpointGUID := c.Param("guid") + userGUID := c.Get("user_id").(string) + + cnsiRecord, err := p.GetCNSIRecord(endpointGUID) + if err != nil { + return errors.New("Could not get endpoint information") + } + + // Get token for this user + tokenRecord, ok := p.GetCNSITokenRecord(endpointGUID, userGUID) + if !ok { + return errors.New("Could not get token") + } + + // This is the kube config for the kubernetes endpoint that we want configured in the Terminal + kubeConfig, err := k.Kube.GetKubeConfigForEndpoint(cnsiRecord.APIEndpoint.String(), tokenRecord, "") + if err != nil { + return errors.New("Can not get Kubernetes config for specified endpoint") + } + + // TODO: Refresh auth token + + // Upgrade the web socket for the incoming request + ws, pingTicker, err := interfaces.UpgradeToWebSocket(c) + if err != nil { + return err + } + defer ws.Close() + defer pingTicker.Stop() + + // We are now in web socket land - we don't want any middleware to change the HTTP response + c.Set("Stratos-WebSocket", "true") + + // Send a message to say that we are creating the pod + sendProgressMessage(ws, "Launching Kubernetes Terminal ... one moment please") + + podData, err := k.createPod(c, kubeConfig, ws) + + // Clear progress message + sendProgressMessage(ws, "") + + if err != nil { + log.Error("ERROR creating secret or pod") + log.Info(err) + k.cleanupPodAndSecret(podData) + + // Send error message + sendProgressMessage(ws, "!" + err.Error()) + return err + } + + // API Endpoint to SSH/exec into a container + target := fmt.Sprintf("%s/api/v1/namespaces/%s/pods/%s/exec?command=/bin/bash&stdin=true&stderr=true&stdout=true&tty=true", k.APIServer, k.Namespace, podData.PodName) + + + // TODO: Check use of kubeHTTPClient + + dial := (&net.Dialer{ + Timeout: time.Duration(30) * time.Second, + KeepAlive: 30 * time.Second, + }).Dial + + // TODO: Not sure about the cert here? I don't think we need this like it is - wonder why I did that? + + sslTransport := &http.Transport{ + Proxy: http.ProxyFromEnvironment, + Dial: dial, + TLSHandshakeTimeout: 10 * time.Second, // 10 seconds is a sound default value (default is 0) + TLSClientConfig: &tls.Config{ + InsecureSkipVerify: true, + //Certificates: []tls.Certificate{cert}, + }, + MaxIdleConnsPerHost: 6, // (default is 2) + } + + kubeHTTPClient := http.Client{} + kubeHTTPClient.Transport = sslTransport + kubeHTTPClient.Timeout = time.Duration(30) * time.Second + + + // This dialer does not use the kubeHttpClient - is it unused? + dialer := &websocket.Dialer{ + TLSClientConfig: &tls.Config{ + InsecureSkipVerify: true, + //Certificates: []tls.Certificate{cert}, + }, + } + + if strings.HasPrefix(target, "https://") { + target = "wss://" + target[8:] + } else { + target = "ws://" + target[7:] + } + + header := &http.Header{} + header.Add("Authorization", fmt.Sprintf("Bearer %s", string(k.Token))) + wsConn, res, err := dialer.Dial(target, *header) + + if err == nil { + defer wsConn.Close() + } + + // Check res + log.Warn("== Status ===========================================") + log.Info(res.Status) + log.Info(res.StatusCode) + + kubeHTTPClient.CloseIdleConnections() + + if err != nil { + k.cleanupPodAndSecret(podData) + return errors.New("Could not make request") + } + + stdoutDone := make(chan bool) + go pumpStdout(ws, wsConn, stdoutDone) + + // Read the input from the web socket and pipe it to the SSH client + for { + _, r, err := ws.ReadMessage() + if err != nil { + // Check to see if this was because the web socket was closed cleanly + closed := false + select { + case msg := <-stdoutDone: + closed = msg + } + if !closed { + log.Errorf("Kubernetes terminal: error reading message from web socket: %+v", err) + } + k.cleanupPodAndSecret(podData) + return err + } + + res := KeyCode{} + json.Unmarshal(r, &res) + + if res.Cols == 0 { + slice := make([]byte, 1) + slice[0] = 0 + slice = append(slice, []byte(res.Key)...) + wsConn.WriteMessage(websocket.TextMessage, slice) + } else { + size := terminalSize{ + Width: uint16(res.Cols), + Height: uint16(res.Rows), + } + j, _ := json.Marshal(size) + resizeStream := []byte{4} + slice := append(resizeStream, j...) + wsConn.WriteMessage(websocket.TextMessage, slice) + } + } + + // Cleanup + log.Debug("Kubernetes Terminal is cleaning up") + + // This is unreachable???? + return k.cleanupPodAndSecret(podData) +} + +func pumpStdout(ws *websocket.Conn, source *websocket.Conn, done chan bool) { + for { + _, r, err := source.ReadMessage() + if err != nil { + // Close + ws.Close() + done <- true + break + } + ws.SetWriteDeadline(time.Now().Add(writeWait)) + bytes := fmt.Sprintf("% x\n", r[1:]) + if err := ws.WriteMessage(websocket.TextMessage, []byte(bytes)); err != nil { + log.Errorf("Kubernetes Terimnal failed to write nessage: %+v", err) + ws.Close() + break + } + } +} diff --git a/src/jetstream/plugins/kubernetes/terminal/terminal.go b/src/jetstream/plugins/kubernetes/terminal/terminal.go index 646fb46b45..1654006ad7 100644 --- a/src/jetstream/plugins/kubernetes/terminal/terminal.go +++ b/src/jetstream/plugins/kubernetes/terminal/terminal.go @@ -1,35 +1,74 @@ package terminal import ( + "fmt" + "io/ioutil" + "github.com/cloudfoundry-incubator/stratos/src/jetstream/repository/interfaces" + "github.com/cloudfoundry-incubator/stratos/src/jetstream/plugins/kubernetes/api" "github.com/cloudfoundry-incubator/stratos/src/jetstream/repository/interfaces/config" log "github.com/sirupsen/logrus" ) +const ( + //serviceAccountTokenFile = "/var/run/secrets/kubernetes.io/serviceaccount/token" + serviceAccountTokenFile = "./token" + serviceHostEnvVar = "KUBERNETES_SERVICE_HOST" + servicePortEnvVar = "KUBERNETES_SERVICE_PORT" + + stratosRoleLabel = "stratos-role" + stratosKubeTerminalRole = "kube-terminal" + stratosSessionAnnotation = "stratos-session" + + consoleContainerName = "kube-terminal" +) + // KubeTerminal supports spawning pods to provide a CLI environment to the user type KubeTerminal struct { + PortalProxy interfaces.PortalProxy Namespace string `configName:"STRATOS_KUBERNETES_NAMESPACE"` Image string `configName:"STRATOS_KUBERNETES_TERMINAL_IMAGE"` - Token string `configName:"STRATOS_NAMESPACE"` + Token []byte + APIServer string + Kube api.Kubernetes } // NewKubeTerminal checks that the environment is set up to support the Kube Terminal func NewKubeTerminal(p interfaces.PortalProxy) *KubeTerminal { - kt := &KubeTerminal{} + kt := &KubeTerminal{ + PortalProxy: p, + } if err := config.Load(kt, p.Env().Lookup); err != nil { - log.Warn("Unable to load Kube Terminal configuration. %v", err) + log.Warnf("Unable to load Kube Terminal configuration. %v", err) return nil } // Check that we have everything we need - if len(kt.Image) == 0 || len(kt.Namespace) == 0 || len(kt.Token) == 0 { + if len(kt.Image) == 0 || len(kt.Namespace) == 0 { log.Warn("Kube Terminal configuration is not complete") return nil } - log.Info("Kubernetes Terminal configured") + // Read the Kubernetes API Endpoint + host, hostFound := p.Env().Lookup(serviceHostEnvVar) + port, portFound := p.Env().Lookup(servicePortEnvVar) + if !hostFound || !portFound { + log.Warn("Kubernetes API Server configuration not found (host and/or port env vars not set)") + return nil + } + kt.APIServer = fmt.Sprintf("https://%s:%s", host, port) + + // Read the Service Account Token + token, err := ioutil.ReadFile(serviceAccountTokenFile) + if err != nil { + log.Warnf("Unable to load Service Account token. %v", err) + return nil + } + + kt.Token = token + log.Debug("Kubernetes Terminal configured") return kt } diff --git a/src/jetstream/repository/interfaces/sessiondata.go b/src/jetstream/repository/interfaces/sessiondata.go index 4128f4280f..37884b576a 100644 --- a/src/jetstream/repository/interfaces/sessiondata.go +++ b/src/jetstream/repository/interfaces/sessiondata.go @@ -10,9 +10,13 @@ type SessionDataStore interface { SetValues(session, group string, values map[string]string, autoExpire bool) error DeleteValues(session, group string) error + IsValidSession(id int) (bool, error) + // Cleanup runs a background goroutine every interval that deletes expired sessions from the database Cleanup(interval time.Duration) (chan<- struct{}, <-chan struct{}) // StopCleanup stops the background cleanup from running StopCleanup(quit chan<- struct{}, done <-chan struct{}) + + } diff --git a/src/jetstream/repository/sessiondata/psql_sessiondata.go b/src/jetstream/repository/sessiondata/psql_sessiondata.go index d0c3e7c1a0..c60ad32b63 100644 --- a/src/jetstream/repository/sessiondata/psql_sessiondata.go +++ b/src/jetstream/repository/sessiondata/psql_sessiondata.go @@ -3,6 +3,8 @@ package sessiondata import ( "database/sql" "fmt" + "strconv" + "time" log "github.com/sirupsen/logrus" @@ -16,12 +18,15 @@ var insertSessionDataValue = `INSERT INTO session_data (session, groupName, name var deleteSessionGroupData = `DELETE FROM session_data WHERE session=$1 AND groupName=$2` -// Expire data for sessions that not longer exist +// Expire data for sessions that no longer exist var expireSessionData = `UPDATE session_data SET expired=true WHERE session NOT IN (SELECT id from sessions)` // Delete data for sessions that no longer exist var deleteSessionData = `DELETE FROM session_data WHERE expired=true AND keep_on_expire=false` +// Check if a session valid +var isValidSession = `SELECT id, expires_on from sessions WHERE id=$1` + // SessionDataRepository is a RDB-backed Session Data repository type SessionDataRepository struct { db *sql.DB @@ -40,6 +45,7 @@ func InitRepositoryProvider(databaseProvider string) { deleteSessionGroupData = datastore.ModifySQLStatement(deleteSessionGroupData, databaseProvider) expireSessionData = datastore.ModifySQLStatement(expireSessionData, databaseProvider) deleteSessionData = datastore.ModifySQLStatement(deleteSessionData, databaseProvider) + isValidSession = datastore.ModifySQLStatement(isValidSession, databaseProvider) } // GetValues returns all values from the config table as a map @@ -100,3 +106,27 @@ func (c *SessionDataRepository) SetValues(session, group string, values map[stri return nil } + +// IsValidSession - Determines if the given session ID is still valid (has not expired) +func (c *SessionDataRepository) IsValidSession(session int) (bool, error) { + var ( + id string + expiry time.Time + ) + + err := c.db.QueryRow(isValidSession, strconv.Itoa(session)).Scan(&id, &expiry) + + switch { + case err == sql.ErrNoRows: + // No record with this ID - session does not exist + return false, nil + case err != nil: + return false, fmt.Errorf("Error trying to find Session record: %v", err) + default: + // do nothing + } + + // Check if the session has expired + now := time.Now() + return expiry.After(now), nil +} From 891a8de996099847e27b384141c66a2b1116aac3 Mon Sep 17 00:00:00 2001 From: Neil MacDougall Date: Wed, 3 Jun 2020 14:39:16 +0100 Subject: [PATCH 20/40] Detect kube version --- .../plugins/kubernetes/terminal/helpers.go | 40 ++++++++++++++++++- .../plugins/kubernetes/terminal/start.go | 40 ++++--------------- .../plugins/kubernetes/terminal/terminal.go | 10 +++++ 3 files changed, 56 insertions(+), 34 deletions(-) diff --git a/src/jetstream/plugins/kubernetes/terminal/helpers.go b/src/jetstream/plugins/kubernetes/terminal/helpers.go index 7e455a926b..656f8893de 100644 --- a/src/jetstream/plugins/kubernetes/terminal/helpers.go +++ b/src/jetstream/plugins/kubernetes/terminal/helpers.go @@ -1,8 +1,10 @@ package terminal import ( + "encoding/json" "errors" "fmt" + "regexp" "strings" "time" @@ -49,7 +51,7 @@ func (k *KubeTerminal) getClients() (corev1.PodInterface, corev1.SecretInterface } // Create a pod for a user to run the Kube terminal -func (k *KubeTerminal) createPod(c echo.Context, kubeConfig string, ws *websocket.Conn) (*PodCreationData, error) { +func (k *KubeTerminal) createPod(c echo.Context, kubeConfig, kubeVersion string, ws *websocket.Conn) (*PodCreationData, error) { // Unique ID for the secret and pod name id := uuid.NewV4().String() id = strings.ReplaceAll(id, "-", "") @@ -135,6 +137,12 @@ func (k *KubeTerminal) createPod(c echo.Context, kubeConfig string, ws *websocke containerSpec[0].Image = k.Image containerSpec[0].ImagePullPolicy = "Always" containerSpec[0].VolumeMounts = volumeMountsSpec + + // Add env var for kube version + containerSpec[0].Env = make([]v1.EnvVar, 1) + containerSpec[0].Env[0].Name = "K8S_VERSION" + containerSpec[0].Env[0].Value = kubeVersion + podSpec.Spec.Containers = containerSpec volumesSpec := make([]v1.Volume, 1) @@ -162,7 +170,6 @@ func (k *KubeTerminal) createPod(c echo.Context, kubeConfig string, ws *websocke for { status, err := podClient.Get(pod.Name, statusOptions) if err == nil && status.Status.Phase == "Running" { - log.Error("Pod is ready !") break; } @@ -231,3 +238,32 @@ func sendProgressMessage(ws *websocket.Conn, progressMsg string) { log.Error("Could not send message to client to indicate terminal is starting") } } + +func (k *KubeTerminal) getKubeVersion(endpointID, userID string) (string, error) { + + response, err := k.PortalProxy.DoProxySingleRequest(endpointID, userID, "GET", "/api/v1/nodes", nil, nil) + if err != nil || response.StatusCode != 200 { + return "", errors.New("Could not fetch node list") + } + + var nodes v1.NodeList + err = json.Unmarshal(response.Response, &nodes) + if err != nil { + return "", errors.New("Could not unmarshal node list") + } + + if len(nodes.Items) > 0 { + version := nodes.Items[0].Status.NodeInfo.KubeletVersion + reg, err := regexp.Compile("[^0-9\\.]+") + if err == nil { + version = reg.ReplaceAllString(version, "") + } + parts := strings.Split(version, ".") + if len(parts) > 1 { + v := fmt.Sprintf("%s.%s", parts[0], parts[1]) + return v, nil + } + } + + return "", errors.New("Can not get Kubernetes version") +} diff --git a/src/jetstream/plugins/kubernetes/terminal/start.go b/src/jetstream/plugins/kubernetes/terminal/start.go index b59e4e0a37..42dcde7b60 100644 --- a/src/jetstream/plugins/kubernetes/terminal/start.go +++ b/src/jetstream/plugins/kubernetes/terminal/start.go @@ -45,33 +45,14 @@ const ( writeWait = 10 * time.Second ) - -// Determine if we Kube Terminal is supported -// TODO -// Check namespace -// Check we have secret for pod creation/deletion -// Image for the terminal - -// Environment variables that we expect to see -// STRATOS_KUBERNETES_NAMESPACE -// STRATOS_KUBERNETES_TERMINAL_IMAGE - -// For development, you can overide the token to use with the Kubernetes API, with: -// STRATOS_KUBERNETES_API_TOKEN - -// Start periodic routing to clean up pods that are orphaned -// TODO - // Start handles web-socket request to launch a Kubernetes Terminal func (k *KubeTerminal) Start(c echo.Context) error { var p = k.PortalProxy - c.Response().Status = 500 + //c.Response().Status = 500 - log.Info("Kube Terminal backend request") - - // TODO: We will need these ?? + log.Debug("Kube Terminal backend request") endpointGUID := c.Param("guid") userGUID := c.Get("user_id").(string) @@ -91,9 +72,11 @@ func (k *KubeTerminal) Start(c echo.Context) error { kubeConfig, err := k.Kube.GetKubeConfigForEndpoint(cnsiRecord.APIEndpoint.String(), tokenRecord, "") if err != nil { return errors.New("Can not get Kubernetes config for specified endpoint") - } + } - // TODO: Refresh auth token + // Determine the Kubernetes version + version, _ := k.getKubeVersion(endpointGUID, userGUID) + log.Debugf("Kubernetes Version: %s", version) // Upgrade the web socket for the incoming request ws, pingTicker, err := interfaces.UpgradeToWebSocket(c) @@ -109,14 +92,13 @@ func (k *KubeTerminal) Start(c echo.Context) error { // Send a message to say that we are creating the pod sendProgressMessage(ws, "Launching Kubernetes Terminal ... one moment please") - podData, err := k.createPod(c, kubeConfig, ws) + podData, err := k.createPod(c, kubeConfig, version, ws) // Clear progress message sendProgressMessage(ws, "") if err != nil { - log.Error("ERROR creating secret or pod") - log.Info(err) + log.Errorf("Kubernetes Terminal: Error creating secret or pod: %+v", err) k.cleanupPodAndSecret(podData) // Send error message @@ -127,7 +109,6 @@ func (k *KubeTerminal) Start(c echo.Context) error { // API Endpoint to SSH/exec into a container target := fmt.Sprintf("%s/api/v1/namespaces/%s/pods/%s/exec?command=/bin/bash&stdin=true&stderr=true&stdout=true&tty=true", k.APIServer, k.Namespace, podData.PodName) - // TODO: Check use of kubeHTTPClient dial := (&net.Dialer{ @@ -175,11 +156,6 @@ func (k *KubeTerminal) Start(c echo.Context) error { defer wsConn.Close() } - // Check res - log.Warn("== Status ===========================================") - log.Info(res.Status) - log.Info(res.StatusCode) - kubeHTTPClient.CloseIdleConnections() if err != nil { diff --git a/src/jetstream/plugins/kubernetes/terminal/terminal.go b/src/jetstream/plugins/kubernetes/terminal/terminal.go index 1654006ad7..c7645d0af7 100644 --- a/src/jetstream/plugins/kubernetes/terminal/terminal.go +++ b/src/jetstream/plugins/kubernetes/terminal/terminal.go @@ -3,6 +3,7 @@ package terminal import ( "fmt" "io/ioutil" + "regexp" "github.com/cloudfoundry-incubator/stratos/src/jetstream/repository/interfaces" "github.com/cloudfoundry-incubator/stratos/src/jetstream/plugins/kubernetes/api" @@ -69,6 +70,15 @@ func NewKubeTerminal(p interfaces.PortalProxy) *KubeTerminal { kt.Token = token + version := "v1.15+" + + reg, err := regexp.Compile("[^0-9\\.]+") + if err == nil { + version = reg.ReplaceAllString(version, "") + } + + log.Info(version) + log.Debug("Kubernetes Terminal configured") return kt } From b9f6fdcd8d674d747e4b5da6dfa73a8a4094dffc Mon Sep 17 00:00:00 2001 From: Neil MacDougall Date: Wed, 3 Jun 2020 16:32:56 +0100 Subject: [PATCH 21/40] Tidy up --- .../plugins/kubernetes/terminal/start.go | 30 +------------------ .../plugins/kubernetes/terminal/terminal.go | 25 +++++++--------- 2 files changed, 11 insertions(+), 44 deletions(-) diff --git a/src/jetstream/plugins/kubernetes/terminal/start.go b/src/jetstream/plugins/kubernetes/terminal/start.go index 42dcde7b60..fd7a1b8619 100644 --- a/src/jetstream/plugins/kubernetes/terminal/start.go +++ b/src/jetstream/plugins/kubernetes/terminal/start.go @@ -7,7 +7,6 @@ import ( //"encoding/base64" "encoding/json" - "net" "net/http" "strings" "time" @@ -109,31 +108,6 @@ func (k *KubeTerminal) Start(c echo.Context) error { // API Endpoint to SSH/exec into a container target := fmt.Sprintf("%s/api/v1/namespaces/%s/pods/%s/exec?command=/bin/bash&stdin=true&stderr=true&stdout=true&tty=true", k.APIServer, k.Namespace, podData.PodName) - // TODO: Check use of kubeHTTPClient - - dial := (&net.Dialer{ - Timeout: time.Duration(30) * time.Second, - KeepAlive: 30 * time.Second, - }).Dial - - // TODO: Not sure about the cert here? I don't think we need this like it is - wonder why I did that? - - sslTransport := &http.Transport{ - Proxy: http.ProxyFromEnvironment, - Dial: dial, - TLSHandshakeTimeout: 10 * time.Second, // 10 seconds is a sound default value (default is 0) - TLSClientConfig: &tls.Config{ - InsecureSkipVerify: true, - //Certificates: []tls.Certificate{cert}, - }, - MaxIdleConnsPerHost: 6, // (default is 2) - } - - kubeHTTPClient := http.Client{} - kubeHTTPClient.Transport = sslTransport - kubeHTTPClient.Timeout = time.Duration(30) * time.Second - - // This dialer does not use the kubeHttpClient - is it unused? dialer := &websocket.Dialer{ TLSClientConfig: &tls.Config{ @@ -150,14 +124,12 @@ func (k *KubeTerminal) Start(c echo.Context) error { header := &http.Header{} header.Add("Authorization", fmt.Sprintf("Bearer %s", string(k.Token))) - wsConn, res, err := dialer.Dial(target, *header) + wsConn, _, err := dialer.Dial(target, *header) if err == nil { defer wsConn.Close() } - kubeHTTPClient.CloseIdleConnections() - if err != nil { k.cleanupPodAndSecret(podData) return errors.New("Could not make request") diff --git a/src/jetstream/plugins/kubernetes/terminal/terminal.go b/src/jetstream/plugins/kubernetes/terminal/terminal.go index c7645d0af7..a77f792908 100644 --- a/src/jetstream/plugins/kubernetes/terminal/terminal.go +++ b/src/jetstream/plugins/kubernetes/terminal/terminal.go @@ -3,7 +3,6 @@ package terminal import ( "fmt" "io/ioutil" - "regexp" "github.com/cloudfoundry-incubator/stratos/src/jetstream/repository/interfaces" "github.com/cloudfoundry-incubator/stratos/src/jetstream/plugins/kubernetes/api" @@ -13,10 +12,11 @@ import ( ) const ( - //serviceAccountTokenFile = "/var/run/secrets/kubernetes.io/serviceaccount/token" - serviceAccountTokenFile = "./token" + serviceAccountTokenFile = "/var/run/secrets/kubernetes.io/serviceaccount/token" serviceHostEnvVar = "KUBERNETES_SERVICE_HOST" servicePortEnvVar = "KUBERNETES_SERVICE_PORT" + // For dev - read token from env var + serviceTokenEnvVar = "KUBE_TERMINAL_SERVICE_ACCOUNT_TOKEN" stratosRoleLabel = "stratos-role" stratosKubeTerminalRole = "kube-terminal" @@ -37,7 +37,6 @@ type KubeTerminal struct { // NewKubeTerminal checks that the environment is set up to support the Kube Terminal func NewKubeTerminal(p interfaces.PortalProxy) *KubeTerminal { - kt := &KubeTerminal{ PortalProxy: p, } @@ -64,21 +63,17 @@ func NewKubeTerminal(p interfaces.PortalProxy) *KubeTerminal { // Read the Service Account Token token, err := ioutil.ReadFile(serviceAccountTokenFile) if err != nil { - log.Warnf("Unable to load Service Account token. %v", err) - return nil + // Check env var + tkn, found := p.Env().Lookup(serviceTokenEnvVar) + if !found { + log.Warnf("Unable to load Service Account token. %v", err) + return nil + } + token = []byte(tkn) } kt.Token = token - version := "v1.15+" - - reg, err := regexp.Compile("[^0-9\\.]+") - if err == nil { - version = reg.ReplaceAllString(version, "") - } - - log.Info(version) - log.Debug("Kubernetes Terminal configured") return kt } From dc394aa15b01cc4f98def463e92df2a451e24bc6 Mon Sep 17 00:00:00 2001 From: Neil MacDougall Date: Wed, 3 Jun 2020 20:23:04 +0100 Subject: [PATCH 22/40] Bug fixes and further tidy up --- custom-src/deploy/kubernetes/__stratos.tpl | 4 ++ .../kube-console/kube-console.component.ts | 45 ++++--------- .../kube-terminal/kubeconsole.bashrc | 63 +++++++++++-------- .../plugins/kubernetes/terminal/cleanup.go | 9 +-- .../plugins/kubernetes/terminal/helpers.go | 9 +-- .../plugins/kubernetes/terminal/start.go | 25 ++++---- 6 files changed, 72 insertions(+), 83 deletions(-) diff --git a/custom-src/deploy/kubernetes/__stratos.tpl b/custom-src/deploy/kubernetes/__stratos.tpl index 3ebbce1efe..4999b0e2cb 100644 --- a/custom-src/deploy/kubernetes/__stratos.tpl +++ b/custom-src/deploy/kubernetes/__stratos.tpl @@ -12,4 +12,8 @@ value: "mongodb://{{ .Release.Name }}-fdbdoclayer:27016" - name: SYNC_SERVER_URL value: "http://{{ .Release.Name }}-chartsync:8080" +- name: STRATOS_KUBERNETES_NAMESPACE + value: "{{ .Release.Namespace }}" +- name: STRATOS_KUBERNETES_TERMINAL_IMAGE + value: "{{.Values.kube.registry.hostname}}/{{.Values.kube.organization}}/stratos-kube-terminal:{{.Values.consoleVersion}}" {{- end }} \ No newline at end of file diff --git a/custom-src/frontend/app/custom/kubernetes/kube-console/kube-console.component.ts b/custom-src/frontend/app/custom/kubernetes/kube-console/kube-console.component.ts index 8335425367..42a25a5a9a 100644 --- a/custom-src/frontend/app/custom/kubernetes/kube-console/kube-console.component.ts +++ b/custom-src/frontend/app/custom/kubernetes/kube-console/kube-console.component.ts @@ -2,9 +2,8 @@ import { Component, OnInit, ViewChild } from '@angular/core'; import { ActivatedRoute } from '@angular/router'; import { NEVER, Observable, Subject, Subscription } from 'rxjs'; import websocketConnect, { normalClosureMessage } from 'rxjs-websockets'; -import { catchError, tap, switchMap } from 'rxjs/operators'; +import { catchError, tap, switchMap, map } from 'rxjs/operators'; -import { IApp } from '../../../core/cf-api.types'; import { IHeaderBreadcrumb } from '../../../shared/components/page-header/page-header.types'; import { SshViewerComponent } from '../../../shared/components/ssh-viewer/ssh-viewer.component'; import { KubernetesEndpointService } from '../services/kubernetes-endpoint.service'; @@ -42,32 +41,14 @@ export class KubeConsoleComponent implements OnInit { public errorMessage: string; - public sshRoute: string; - public connected: boolean; public kubeSummaryLink: string; - private connection: Subscription; - - public instanceId: string; - public breadcrumbs$: Observable; @ViewChild('sshViewer', { static: false }) sshViewer: SshViewerComponent; - private getBreadcrumbs( - application: IApp, - ) { - return [ - { - breadcrumbs: [ - { value: application.name, routerLink: `/kubernetes/${application.cfGuid}/${application.guid}/instances` } - ] - }, - ]; - } - constructor( public kubeEndpointService: KubernetesEndpointService, ) { } @@ -75,9 +56,7 @@ export class KubeConsoleComponent implements OnInit { ngOnInit() { this.connectionStatus.next(0); const guid = this.kubeEndpointService.baseKube.guid; - this.kubeSummaryLink = ( - `/kubernetes/${guid}/summary` - ); + this.kubeSummaryLink = `/kubernetes/${guid}/summary`; if (!guid) { this.messages = NEVER; @@ -90,25 +69,27 @@ export class KubeConsoleComponent implements OnInit { `${protocol}://${host}/pp/v1/kubeconsole/${guid}` ); this.sshInput = new Subject(); - const connection = websocketConnect( - streamUrl - ); + const connection = websocketConnect(streamUrl); this.messages = connection.pipe( tap(() => this.connectionStatus.next(1)), switchMap(getResponse => getResponse(this.sshInput)), catchError((e: Error) => { - console.log(e.message); - if (e.message !== normalClosureMessage) { + if (e.message !== normalClosureMessage && !this.sshViewer.isConnected) { this.errorMessage = 'Error launching Kubernetes Terminal'; } return []; })); - // this.breadcrumbs$ = this.applicationService.waitForAppEntity$.pipe( - // map(app => this.getBreadcrumbs(app.entity.entity)), - // first() - // ); + // Breadcrumbs + this.breadcrumbs$ = this.kubeEndpointService.endpoint$.pipe( + map(endpoint => ([{ + breadcrumbs: [ + { value: endpoint.entity.name, routerLink: `/kubernetes/${endpoint.entity.guid}` }, + ] + }]) + ) + ); } } } diff --git a/deploy/containers/kube-terminal/kubeconsole.bashrc b/deploy/containers/kube-terminal/kubeconsole.bashrc index 10da69e597..362fc54adc 100644 --- a/deploy/containers/kube-terminal/kubeconsole.bashrc +++ b/deploy/containers/kube-terminal/kubeconsole.bashrc @@ -11,27 +11,31 @@ echo "" echo -e "${CYAN}Kubernetes Terminal${RESET}" echo "" -# Unpack helm comand -gunzip /stratos/helm.gz - -# Need to choose appropriate kubectl version -pushd /stratos > /dev/null -# Default to the newwest version that we have -USE=$(ls kubectl_* | sort -r | head -n1) -popd > /dev/null - -# If env var KUBERNETES_VERSION is set, then use it (major.minor only) -if [ -n "${KUBERNETES_VERSION}" ]; then - VERSION="kubectl_${KUBERNETES_VERSION}.gz" - if [ -f "/stratos/${VERSION}" ]; then - USE=${VERSION} +# Only do these on first run +if [ ! -f "/root/.stratos/firstrun" ]; then + # Unpack helm comand + gunzip /stratos/helm.gz + + # Need to choose appropriate kubectl version + pushd /stratos > /dev/null + # Default to the newwest version that we have + USE=$(ls kubectl_* | sort -r | head -n1) + popd > /dev/null + + # If env var K8S_VERSION is set, then use it (major.minor only) + if [ -n "${K8S_VERSION}" ]; then + VERSION="kubectl_${K8S_VERSION}.gz" + if [ -f "/stratos/${VERSION}" ]; then + USE=${VERSION} + fi fi + + gunzip /stratos/${USE} + VER=${USE::-3} + mv /stratos/${VER} /stratos/kubectl + chmod +x /stratos/kubectl fi -gunzip /stratos/${USE} -VER=${USE::-3} -mv /stratos/${VER} /stratos/kubectl -chmod +x /stratos/kubectl export PATH=/stratos:$PATH export KUBECONFIG=/root/.stratos/kubeconfig @@ -43,15 +47,17 @@ source <(helm completion bash) #helm repo remove stable > /dev/null -if [ -f "/root/.stratos/helm-setup" ]; then - echo "Setting up Helm repositories ..." - source "/root/.stratos/helm-setup" > /dev/null - helm repo update 2>&1 > /dev/null - echo "" -fi +if [ ! -f "/root/.stratos/firstrun" ]; then + if [ -f "/root/.stratos/helm-setup" ]; then + echo "Setting up Helm repositories ..." + source "/root/.stratos/helm-setup" > /dev/null + helm repo update 2>&1 > /dev/null + echo "" + fi -if [ -f "/root/.stratos/history" ]; then - cat /root/.stratos/history > /root/.bash_history + if [ -f "/root/.stratos/history" ]; then + cat /root/.stratos/history > /root/.bash_history + fi fi # Make Bash append rather than overwrite the history on disk: @@ -61,5 +67,10 @@ PROMPT_COMMAND='history -a' # Don't put duplicate lines in the history. export HISTCONTROL=ignoredups +touch /root/.stratos/firstrun + +# Remove any env vars matching KUBERNETES +unset `compgen -A variable | grep KUBERNETES` + echo "Ready" echo "" diff --git a/src/jetstream/plugins/kubernetes/terminal/cleanup.go b/src/jetstream/plugins/kubernetes/terminal/cleanup.go index 8426385b4f..d1d50aa559 100644 --- a/src/jetstream/plugins/kubernetes/terminal/cleanup.go +++ b/src/jetstream/plugins/kubernetes/terminal/cleanup.go @@ -10,21 +10,18 @@ import ( metaV1 "k8s.io/apimachinery/pkg/apis/meta/v1" ) -// Wait time in minutes after intiial wait +// Wait time in minutes after random intiial wait const waitPeriod = 10 -// StartCleanup starts a backgrdound routine to cleanup orphaned pods +// StartCleanup starts a background routine to cleanup orphaned pods func (k *KubeTerminal) StartCleanup() { go k.cleanup() } func (k *KubeTerminal) cleanup() { // Use a random initial wait before cleaning up - // If we had moer backends, this helps to ensure they are not all trying to cleanup at the same time + // If we had more than one backend, this helps to ensure they are not all trying to cleanup at the same time wait := rand.Intn(30) - - // Testing - wait = 0 for { time.Sleep(time.Duration(wait) * time.Minute) log.Debug("Cleaning up stale Kubernetes Terminal pods and secrets ...") diff --git a/src/jetstream/plugins/kubernetes/terminal/helpers.go b/src/jetstream/plugins/kubernetes/terminal/helpers.go index 656f8893de..8cc58caf43 100644 --- a/src/jetstream/plugins/kubernetes/terminal/helpers.go +++ b/src/jetstream/plugins/kubernetes/terminal/helpers.go @@ -58,9 +58,7 @@ func (k *KubeTerminal) createPod(c echo.Context, kubeConfig, kubeVersion string, // Names for the secret and pod secretName := fmt.Sprintf("terminal-%s", id) podName := secretName - podClient, secretClient, err := k.getClients() - result := &PodCreationData{} result.Namespace = k.Namespace @@ -88,7 +86,6 @@ func (k *KubeTerminal) createPod(c echo.Context, kubeConfig, kubeVersion string, secretSpec.Data = make(map[string][]byte) secretSpec.Data["kubeconfig"] = []byte(kubeConfig) - //secretSpec.Data["history"] = []byte(history) // Get Helm repository script if we have Helm repositories helmSetup := getHelmRepoSetupScript(k.PortalProxy) @@ -98,7 +95,7 @@ func (k *KubeTerminal) createPod(c echo.Context, kubeConfig, kubeVersion string, _, err = secretClient.Create(secretSpec) if err != nil { - log.Warn("Unable to create Secret") + log.Warnf("Kubernetes Terminal: Unable to create Secret: %+v", err) return result, err } @@ -155,6 +152,7 @@ func (k *KubeTerminal) createPod(c echo.Context, kubeConfig, kubeVersion string, // Create a new pod pod, err := podClient.Create(podSpec) if err != nil { + log.Warnf("Kubernetes Terminal: Unable to create Pod: %+v", err) // Secret will get cleaned up by caller return result, err } @@ -210,7 +208,6 @@ func (k *KubeTerminal) cleanupPodAndSecret(podData *PodCreationData) error { return nil } - func getHelmRepoSetupScript(portalProxy interfaces.PortalProxy) string { str := "" @@ -240,7 +237,6 @@ func sendProgressMessage(ws *websocket.Conn, progressMsg string) { } func (k *KubeTerminal) getKubeVersion(endpointID, userID string) (string, error) { - response, err := k.PortalProxy.DoProxySingleRequest(endpointID, userID, "GET", "/api/v1/nodes", nil, nil) if err != nil || response.StatusCode != 200 { return "", errors.New("Could not fetch node list") @@ -253,6 +249,7 @@ func (k *KubeTerminal) getKubeVersion(endpointID, userID string) (string, error) } if len(nodes.Items) > 0 { + // Get the version number - remove any 'v' perfix or '+' suffix version := nodes.Items[0].Status.NodeInfo.KubeletVersion reg, err := regexp.Compile("[^0-9\\.]+") if err == nil { diff --git a/src/jetstream/plugins/kubernetes/terminal/start.go b/src/jetstream/plugins/kubernetes/terminal/start.go index fd7a1b8619..c05b3b579a 100644 --- a/src/jetstream/plugins/kubernetes/terminal/start.go +++ b/src/jetstream/plugins/kubernetes/terminal/start.go @@ -46,23 +46,18 @@ const ( // Start handles web-socket request to launch a Kubernetes Terminal func (k *KubeTerminal) Start(c echo.Context) error { - - var p = k.PortalProxy - - //c.Response().Status = 500 - - log.Debug("Kube Terminal backend request") + log.Debug("Kube Terminal start request") endpointGUID := c.Param("guid") userGUID := c.Get("user_id").(string) - cnsiRecord, err := p.GetCNSIRecord(endpointGUID) + cnsiRecord, err := k.PortalProxy.GetCNSIRecord(endpointGUID) if err != nil { return errors.New("Could not get endpoint information") } // Get token for this user - tokenRecord, ok := p.GetCNSITokenRecord(endpointGUID, userGUID) + tokenRecord, ok := k.PortalProxy.GetCNSITokenRecord(endpointGUID, userGUID) if !ok { return errors.New("Could not get token") } @@ -138,6 +133,12 @@ func (k *KubeTerminal) Start(c echo.Context) error { stdoutDone := make(chan bool) go pumpStdout(ws, wsConn, stdoutDone) + // If the downstream connection is closed, close the other web socket as well + ws.SetCloseHandler(func (code int, text string) error { + wsConn.Close() + return nil + }) + // Read the input from the web socket and pipe it to the SSH client for { _, r, err := ws.ReadMessage() @@ -151,6 +152,7 @@ func (k *KubeTerminal) Start(c echo.Context) error { if !closed { log.Errorf("Kubernetes terminal: error reading message from web socket: %+v", err) } + log.Warn("Kube Terminal cleaning up ....") k.cleanupPodAndSecret(podData) return err } @@ -176,9 +178,8 @@ func (k *KubeTerminal) Start(c echo.Context) error { } // Cleanup - log.Debug("Kubernetes Terminal is cleaning up") + log.Error("Kubernetes Terminal is cleaning up") - // This is unreachable???? return k.cleanupPodAndSecret(podData) } @@ -187,15 +188,13 @@ func pumpStdout(ws *websocket.Conn, source *websocket.Conn, done chan bool) { _, r, err := source.ReadMessage() if err != nil { // Close - ws.Close() done <- true break } ws.SetWriteDeadline(time.Now().Add(writeWait)) bytes := fmt.Sprintf("% x\n", r[1:]) if err := ws.WriteMessage(websocket.TextMessage, []byte(bytes)); err != nil { - log.Errorf("Kubernetes Terimnal failed to write nessage: %+v", err) - ws.Close() + log.Errorf("Kubernetes Terminal failed to write message: %+v", err) break } } From 5899bfdf184ec2e27caa0778d267fd5cdcd770ac Mon Sep 17 00:00:00 2001 From: Neil MacDougall Date: Wed, 3 Jun 2020 20:31:08 +0100 Subject: [PATCH 23/40] Fix backend logging issue --- src/jetstream/plugins/kubernetes/terminal/start.go | 8 ++++++-- 1 file changed, 6 insertions(+), 2 deletions(-) diff --git a/src/jetstream/plugins/kubernetes/terminal/start.go b/src/jetstream/plugins/kubernetes/terminal/start.go index c05b3b579a..5c88b1d32f 100644 --- a/src/jetstream/plugins/kubernetes/terminal/start.go +++ b/src/jetstream/plugins/kubernetes/terminal/start.go @@ -127,7 +127,9 @@ func (k *KubeTerminal) Start(c echo.Context) error { if err != nil { k.cleanupPodAndSecret(podData) - return errors.New("Could not make request") + log.Warn("Kube Terminal: Could not connect to pod") + // No point returning an error - we've already upgraded to web sockets, so we can't use the HTTP response now + return nil } stdoutDone := make(chan bool) @@ -154,7 +156,9 @@ func (k *KubeTerminal) Start(c echo.Context) error { } log.Warn("Kube Terminal cleaning up ....") k.cleanupPodAndSecret(podData) - return err + + // No point returning an error - we've already upgraded to web sockets, so we can't use the HTTP response now + return nil } res := KeyCode{} From fc2212583206a62a12e67d32e8bbae2d93beca83 Mon Sep 17 00:00:00 2001 From: Neil MacDougall Date: Wed, 3 Jun 2020 20:32:03 +0100 Subject: [PATCH 24/40] Change log level to debug --- src/jetstream/plugins/kubernetes/terminal/start.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/jetstream/plugins/kubernetes/terminal/start.go b/src/jetstream/plugins/kubernetes/terminal/start.go index 5c88b1d32f..3a228b025b 100644 --- a/src/jetstream/plugins/kubernetes/terminal/start.go +++ b/src/jetstream/plugins/kubernetes/terminal/start.go @@ -154,7 +154,7 @@ func (k *KubeTerminal) Start(c echo.Context) error { if !closed { log.Errorf("Kubernetes terminal: error reading message from web socket: %+v", err) } - log.Warn("Kube Terminal cleaning up ....") + log.Debug("Kube Terminal cleaning up ....") k.cleanupPodAndSecret(podData) // No point returning an error - we've already upgraded to web sockets, so we can't use the HTTP response now From 1deac6d67ed8085c24578b8b28c64d98121e33ca Mon Sep 17 00:00:00 2001 From: Neil MacDougall Date: Thu, 4 Jun 2020 09:35:01 +0100 Subject: [PATCH 25/40] Change some naming to kube terminal --- .../kube-console.component.html | 0 .../kube-console.component.scss | 0 .../kube-console.component.spec.ts | 4 ++-- .../kube-console.component.ts | 2 +- .../app/custom/kubernetes/kubernetes.module.ts | 2 +- .../app/custom/kubernetes/kubernetes.routing.ts | 4 ++-- .../services/kubernetes-endpoint.service.ts | 6 ++++++ .../kubernetes-summary.component.html | 2 +- .../kubernetes-summary.component.ts | 4 ++-- src/jetstream/plugins/kubernetes/main.go | 11 ++++++++--- 10 files changed, 23 insertions(+), 12 deletions(-) rename custom-src/frontend/app/custom/kubernetes/{kube-console => kube-terminal}/kube-console.component.html (100%) rename custom-src/frontend/app/custom/kubernetes/{kube-console => kube-terminal}/kube-console.component.scss (100%) rename custom-src/frontend/app/custom/kubernetes/{kube-console => kube-terminal}/kube-console.component.spec.ts (89%) rename custom-src/frontend/app/custom/kubernetes/{kube-console => kube-terminal}/kube-console.component.ts (98%) diff --git a/custom-src/frontend/app/custom/kubernetes/kube-console/kube-console.component.html b/custom-src/frontend/app/custom/kubernetes/kube-terminal/kube-console.component.html similarity index 100% rename from custom-src/frontend/app/custom/kubernetes/kube-console/kube-console.component.html rename to custom-src/frontend/app/custom/kubernetes/kube-terminal/kube-console.component.html diff --git a/custom-src/frontend/app/custom/kubernetes/kube-console/kube-console.component.scss b/custom-src/frontend/app/custom/kubernetes/kube-terminal/kube-console.component.scss similarity index 100% rename from custom-src/frontend/app/custom/kubernetes/kube-console/kube-console.component.scss rename to custom-src/frontend/app/custom/kubernetes/kube-terminal/kube-console.component.scss diff --git a/custom-src/frontend/app/custom/kubernetes/kube-console/kube-console.component.spec.ts b/custom-src/frontend/app/custom/kubernetes/kube-terminal/kube-console.component.spec.ts similarity index 89% rename from custom-src/frontend/app/custom/kubernetes/kube-console/kube-console.component.spec.ts rename to custom-src/frontend/app/custom/kubernetes/kube-terminal/kube-console.component.spec.ts index 030ec22e95..85fda06bf6 100644 --- a/custom-src/frontend/app/custom/kubernetes/kube-console/kube-console.component.spec.ts +++ b/custom-src/frontend/app/custom/kubernetes/kube-terminal/kube-console.component.spec.ts @@ -1,5 +1,5 @@ -import { async, ComponentFixture, TestBed } from '@angular/core/testing'; -import { RouterTestingModule } from '@angular/router/testing'; +import { async, ComponentFixture, TestBed } from './@angular/core/testing'; +import { RouterTestingModule } from './@angular/router/testing'; import { TabNavService } from '../../../../tab-nav.service'; import { ApplicationServiceMock } from '../../../../test-framework/application-service-helper'; diff --git a/custom-src/frontend/app/custom/kubernetes/kube-console/kube-console.component.ts b/custom-src/frontend/app/custom/kubernetes/kube-terminal/kube-console.component.ts similarity index 98% rename from custom-src/frontend/app/custom/kubernetes/kube-console/kube-console.component.ts rename to custom-src/frontend/app/custom/kubernetes/kube-terminal/kube-console.component.ts index 42a25a5a9a..643fd02271 100644 --- a/custom-src/frontend/app/custom/kubernetes/kube-console/kube-console.component.ts +++ b/custom-src/frontend/app/custom/kubernetes/kube-terminal/kube-console.component.ts @@ -66,7 +66,7 @@ export class KubeConsoleComponent implements OnInit { const host = window.location.host; const protocol = window.location.protocol === 'https:' ? 'wss' : 'ws'; const streamUrl = ( - `${protocol}://${host}/pp/v1/kubeconsole/${guid}` + `${protocol}://${host}/pp/v1/kubeterminal/${guid}` ); this.sshInput = new Subject(); const connection = websocketConnect(streamUrl); diff --git a/custom-src/frontend/app/custom/kubernetes/kubernetes.module.ts b/custom-src/frontend/app/custom/kubernetes/kubernetes.module.ts index 840793c34c..2d33eeeda6 100644 --- a/custom-src/frontend/app/custom/kubernetes/kubernetes.module.ts +++ b/custom-src/frontend/app/custom/kubernetes/kubernetes.module.ts @@ -92,7 +92,7 @@ import { KubernetesNamespacesTabComponent } from './tabs/kubernetes-namespaces-t import { KubernetesNodesTabComponent } from './tabs/kubernetes-nodes-tab/kubernetes-nodes-tab.component'; import { KubernetesPodsTabComponent } from './tabs/kubernetes-pods-tab/kubernetes-pods-tab.component'; import { KubernetesSummaryTabComponent } from './tabs/kubernetes-summary-tab/kubernetes-summary.component'; -import { KubeConsoleComponent } from './kube-console/kube-console.component'; +import { KubeConsoleComponent } from './kube-terminal/kube-console.component'; /* tslint:enable */ diff --git a/custom-src/frontend/app/custom/kubernetes/kubernetes.routing.ts b/custom-src/frontend/app/custom/kubernetes/kubernetes.routing.ts index b30e712772..c7681161ce 100644 --- a/custom-src/frontend/app/custom/kubernetes/kubernetes.routing.ts +++ b/custom-src/frontend/app/custom/kubernetes/kubernetes.routing.ts @@ -1,4 +1,3 @@ -import { KubeConsoleComponent } from './kube-console/kube-console.component'; import { NgModule } from '@angular/core'; import { RouterModule, Routes } from '@angular/router'; @@ -24,6 +23,7 @@ import { KubernetesNodesTabComponent } from './tabs/kubernetes-nodes-tab/kuberne import { KubernetesPodsTabComponent } from './tabs/kubernetes-pods-tab/kubernetes-pods-tab.component'; import { KubernetesSummaryTabComponent } from './tabs/kubernetes-summary-tab/kubernetes-summary.component'; import { KubedashConfigurationComponent } from './kubernetes-dashboard/kubedash-configuration/kubedash-configuration.component'; +import { KubeConsoleComponent } from './kube-terminal/kube-console.component'; const kubernetes: Routes = [{ path: '', @@ -132,7 +132,7 @@ const kubernetes: Routes = [{ component: KubedashConfigurationComponent, }, { - path: ':endpointId/console', + path: ':endpointId/terminal', component: KubeConsoleComponent, data: { uiNoMargin: true diff --git a/custom-src/frontend/app/custom/kubernetes/services/kubernetes-endpoint.service.ts b/custom-src/frontend/app/custom/kubernetes/services/kubernetes-endpoint.service.ts index 612e762654..101e4c099d 100644 --- a/custom-src/frontend/app/custom/kubernetes/services/kubernetes-endpoint.service.ts +++ b/custom-src/frontend/app/custom/kubernetes/services/kubernetes-endpoint.service.ts @@ -56,6 +56,7 @@ export class KubernetesEndpointService { kubeDashboardStatus$: Observable; kubeDashboardLabel$: Observable; kubeDashboardConfigured$: Observable; + kubeTerminalEnabled$: Observable; constructor( public baseKube: BaseKubeGuid, @@ -190,6 +191,11 @@ export class KubernetesEndpointService { map(auth => auth.sessionData['plugin-config'].kubeDashboardEnabled === 'true') ); + this.kubeTerminalEnabled$ = this.store.select('auth').pipe( + filter(auth => !!auth.sessionData['plugin-config']), + map(auth => auth.sessionData['plugin-config'].kubeTerminalEnabled === 'true') + ); + const kubeDashboardStatus$ = this.entityServiceFactory.create( this.kubeGuid, new GetKubernetesDashboard(this.kubeGuid), diff --git a/custom-src/frontend/app/custom/kubernetes/tabs/kubernetes-summary-tab/kubernetes-summary.component.html b/custom-src/frontend/app/custom/kubernetes/tabs/kubernetes-summary-tab/kubernetes-summary.component.html index e744bedefc..4f3165dbe8 100644 --- a/custom-src/frontend/app/custom/kubernetes/tabs/kubernetes-summary-tab/kubernetes-summary.component.html +++ b/custom-src/frontend/app/custom/kubernetes/tabs/kubernetes-summary-tab/kubernetes-summary.component.html @@ -4,7 +4,7 @@ dashboard View Dashboard - diff --git a/custom-src/frontend/app/custom/kubernetes/tabs/kubernetes-summary-tab/kubernetes-summary.component.ts b/custom-src/frontend/app/custom/kubernetes/tabs/kubernetes-summary-tab/kubernetes-summary.component.ts index 51fcc2657e..26ee5df7e6 100644 --- a/custom-src/frontend/app/custom/kubernetes/tabs/kubernetes-summary-tab/kubernetes-summary.component.ts +++ b/custom-src/frontend/app/custom/kubernetes/tabs/kubernetes-summary-tab/kubernetes-summary.component.ts @@ -64,7 +64,7 @@ export class KubernetesSummaryTabComponent implements OnInit, OnDestroy { source: SafeResourceUrl; dashboardLink: string; - kubeConsoleLink: string; + kubeTerminalLink: string; public podCapacity$: Observable; public diskPressure$: Observable; @@ -158,7 +158,7 @@ export class KubernetesSummaryTabComponent implements OnInit, OnDestroy { warningText: `Nodes with unknown ready status found` }); this.dashboardLink = `/kubernetes/${guid}/dashboard`; - this.kubeConsoleLink = `/kubernetes/${guid}/console`; + this.kubeTerminalLink = `/kubernetes/${guid}/terminal`; this.kubeNodeVersions$ = this.kubeEndpointService.getNodeKubeVersions(nodes$).pipe(startWith('-')); diff --git a/src/jetstream/plugins/kubernetes/main.go b/src/jetstream/plugins/kubernetes/main.go index bca169a26c..8cccc03ce4 100644 --- a/src/jetstream/plugins/kubernetes/main.go +++ b/src/jetstream/plugins/kubernetes/main.go @@ -52,8 +52,10 @@ const ( kubeEndpointType = "k8s" defaultKubeClientID = "K8S_CLIENT" - // kubeDashboardPluginConfigSetting is config value send back to the client to indicate if the kube dashboard can be navigated to + // kubeDashboardPluginConfigSetting is config value sent back to the client to indicate if the kube dashboard ie enabled kubeDashboardPluginConfigSetting = "kubeDashboardEnabled" + // kubeTerminalPluginConfigSetting is config value sent back to the client to indicate if the kube terminal is enabled + kubeTerminalPluginConfigSetting ="kubeTerminalEnabled" ) // Init creates a new instance of the Kubernetes plugin @@ -138,6 +140,10 @@ func (c *KubernetesSpecification) Init() error { // Kube dashboard is enabled by Tech Preview mode c.portalProxy.GetConfig().PluginConfig[kubeDashboardPluginConfigSetting] = strconv.FormatBool(c.portalProxy.GetConfig().EnableTechPreview) + // Kube terminal is enabled by Tech Preview mode + c.portalProxy.GetConfig().PluginConfig[kubeTerminalPluginConfigSetting] = strconv.FormatBool(c.portalProxy.GetConfig().EnableTechPreview) + + // Kick off the cleanup of any old kube terminal pods c.kubeTerminal.StartCleanup() @@ -170,8 +176,7 @@ func (c *KubernetesSpecification) AddSessionGroupRoutes(echoGroup *echo.Group) { echoGroup.GET("/helm/releases/:endpoint/:namespace/:name", c.GetRelease) // Kube Terminal - echoGroup.GET("/kubeconsole/:guid", c.kubeTerminal.Start) - + echoGroup.GET("/kubeterminal/:guid", c.kubeTerminal.Start) } func (c *KubernetesSpecification) Info(apiEndpoint string, skipSSLValidation bool) (interfaces.CNSIRecord, interface{}, error) { From 0bb63256548f93941c67bd300007671e148ad4f5 Mon Sep 17 00:00:00 2001 From: Neil MacDougall Date: Thu, 4 Jun 2020 11:59:34 +0100 Subject: [PATCH 26/40] Close stream --- src/jetstream/plugins/kubernetes/terminal/start.go | 2 ++ 1 file changed, 2 insertions(+) diff --git a/src/jetstream/plugins/kubernetes/terminal/start.go b/src/jetstream/plugins/kubernetes/terminal/start.go index 3a228b025b..d0ac6ac276 100644 --- a/src/jetstream/plugins/kubernetes/terminal/start.go +++ b/src/jetstream/plugins/kubernetes/terminal/start.go @@ -192,6 +192,7 @@ func pumpStdout(ws *websocket.Conn, source *websocket.Conn, done chan bool) { _, r, err := source.ReadMessage() if err != nil { // Close + ws.Close() done <- true break } @@ -199,6 +200,7 @@ func pumpStdout(ws *websocket.Conn, source *websocket.Conn, done chan bool) { bytes := fmt.Sprintf("% x\n", r[1:]) if err := ws.WriteMessage(websocket.TextMessage, []byte(bytes)); err != nil { log.Errorf("Kubernetes Terminal failed to write message: %+v", err) + ws.Close() break } } From 745ae4b772970ef6087e3d2732e81ff7b10521fe Mon Sep 17 00:00:00 2001 From: Neil MacDougall Date: Thu, 4 Jun 2020 13:49:52 +0100 Subject: [PATCH 27/40] Run container as stratos user --- .../kube-terminal/Dockerfile.kubeconsole | 22 +++++++++++++++---- .../kube-terminal/kubeconsole.bashrc | 16 +++++++------- .../plugins/kubernetes/terminal/helpers.go | 2 +- 3 files changed, 27 insertions(+), 13 deletions(-) diff --git a/deploy/containers/kube-terminal/Dockerfile.kubeconsole b/deploy/containers/kube-terminal/Dockerfile.kubeconsole index 19f55f08b4..e62753397e 100644 --- a/deploy/containers/kube-terminal/Dockerfile.kubeconsole +++ b/deploy/containers/kube-terminal/Dockerfile.kubeconsole @@ -2,8 +2,6 @@ FROM splatform/stratos-bk-build-base:leap15_1 as terminal-builder USER root WORKDIR /root -#RUN zypper in -y tar - # Kubectl versions RUN curl -L -o kubectl_1.18 https://storage.googleapis.com/kubernetes-release/release/v1.18.2/bin/linux/amd64/kubectl RUN curl -L -o kubectl_1.17 https://storage.googleapis.com/kubernetes-release/release/v1.17.5/bin/linux/amd64/kubectl @@ -34,7 +32,23 @@ RUN mkdir /stratos COPY --from=terminal-builder /root/helm.gz /stratos/helm.gz COPY --from=terminal-builder /root/kubectl* /stratos/ -ADD ./kubeconsole.bashrc /root/.bashrc -WORKDIR /root +# Run as user 'stratos' +RUN useradd -ms /bin/bash stratos -K MAIL_DIR=/dev/null + +RUN chown -R stratos /stratos && \ + chgrp -R users /stratos + +# Remove a few packages +RUN zypper rm -y diffutils shadow util-linux fillup python3 openssl bind-utils kmod \ + python3-base python-rpm-macros pkg-config hostname GeoIP-data + +# Remove zypper +RUN zypper rm -y dirmngr && \ + rm -rf /usr/bin/rpm* + +USER stratos +WORKDIR /home/stratos + +ADD ./kubeconsole.bashrc /home/stratos/.bashrc CMD exec /bin/bash -c "trap : TERM INT; sleep infinity & wait" diff --git a/deploy/containers/kube-terminal/kubeconsole.bashrc b/deploy/containers/kube-terminal/kubeconsole.bashrc index 362fc54adc..e71545ed5a 100644 --- a/deploy/containers/kube-terminal/kubeconsole.bashrc +++ b/deploy/containers/kube-terminal/kubeconsole.bashrc @@ -12,7 +12,7 @@ echo -e "${CYAN}Kubernetes Terminal${RESET}" echo "" # Only do these on first run -if [ ! -f "/root/.stratos/firstrun" ]; then +if [ ! -f "/stratos/.firstrun" ]; then # Unpack helm comand gunzip /stratos/helm.gz @@ -38,7 +38,7 @@ fi export PATH=/stratos:$PATH -export KUBECONFIG=/root/.stratos/kubeconfig +export KUBECONFIG=${HOME}/.stratos/kubeconfig export PS1="\033[92mstratos>\033[0m" alias k=kubectl @@ -47,16 +47,16 @@ source <(helm completion bash) #helm repo remove stable > /dev/null -if [ ! -f "/root/.stratos/firstrun" ]; then - if [ -f "/root/.stratos/helm-setup" ]; then +if [ ! -f "/stratos/.firstrun" ]; then + if [ -f "${HOME}/.stratos/helm-setup" ]; then echo "Setting up Helm repositories ..." - source "/root/.stratos/helm-setup" > /dev/null + source "${HOME}/.stratos/helm-setup" > /dev/null helm repo update 2>&1 > /dev/null echo "" fi - if [ -f "/root/.stratos/history" ]; then - cat /root/.stratos/history > /root/.bash_history + if [ -f "${HOME}/.stratos/history" ]; then + cat ${HOME}/.stratos/history > ${HOME}/.bash_history fi fi @@ -67,7 +67,7 @@ PROMPT_COMMAND='history -a' # Don't put duplicate lines in the history. export HISTCONTROL=ignoredups -touch /root/.stratos/firstrun +touch "/stratos/.firstrun" # Remove any env vars matching KUBERNETES unset `compgen -A variable | grep KUBERNETES` diff --git a/src/jetstream/plugins/kubernetes/terminal/helpers.go b/src/jetstream/plugins/kubernetes/terminal/helpers.go index 8cc58caf43..b34caf9a9a 100644 --- a/src/jetstream/plugins/kubernetes/terminal/helpers.go +++ b/src/jetstream/plugins/kubernetes/terminal/helpers.go @@ -126,7 +126,7 @@ func (k *KubeTerminal) createPod(c echo.Context, kubeConfig, kubeVersion string, volumeMountsSpec := make([]v1.VolumeMount, 1) volumeMountsSpec[0].Name = "kubeconfig" - volumeMountsSpec[0].MountPath = "/root/.stratos" + volumeMountsSpec[0].MountPath = "/home/stratos/.stratos" volumeMountsSpec[0].ReadOnly = true containerSpec := make([]v1.Container, 1) From 108f3e243c386f47eed5b2139ced7ccc8544560a Mon Sep 17 00:00:00 2001 From: Neil MacDougall Date: Thu, 4 Jun 2020 14:38:15 +0100 Subject: [PATCH 28/40] Fix image build --- deploy/common-build.sh | 6 +----- .../kube-terminal/Dockerfile.kubeconsole | 16 ++++++++++------ deploy/kubernetes/build.sh | 5 +++-- 3 files changed, 14 insertions(+), 13 deletions(-) diff --git a/deploy/common-build.sh b/deploy/common-build.sh index 11fb464ddc..6f29c55d1e 100644 --- a/deploy/common-build.sh +++ b/deploy/common-build.sh @@ -55,9 +55,7 @@ if [ -n "${https_proxy:-}" -o -n "${HTTPS_PROXY:-}" ]; then fi # Check if we can squash -echo "checking" -CAN_SQUASH=$(docker info 2>&1 | grep "Experimental: true" | cat) -echo "check" +CAN_SQUASH=$(docker info 2>&1 | grep "Experimental: true" -c | cat) if [ "${CAN_SQUASH}" == "1" ]; then BUILD_ARGS="${BUILD_ARGS} --squash" echo "Images will be squashed" @@ -65,8 +63,6 @@ else echo "Images will NOT be squashed" fi -echo "OK" - # Use correct sed command for Mac SED="sed -r" unamestr=`uname` diff --git a/deploy/containers/kube-terminal/Dockerfile.kubeconsole b/deploy/containers/kube-terminal/Dockerfile.kubeconsole index e62753397e..1872e5abb0 100644 --- a/deploy/containers/kube-terminal/Dockerfile.kubeconsole +++ b/deploy/containers/kube-terminal/Dockerfile.kubeconsole @@ -5,13 +5,15 @@ WORKDIR /root # Kubectl versions RUN curl -L -o kubectl_1.18 https://storage.googleapis.com/kubernetes-release/release/v1.18.2/bin/linux/amd64/kubectl RUN curl -L -o kubectl_1.17 https://storage.googleapis.com/kubernetes-release/release/v1.17.5/bin/linux/amd64/kubectl -#RUN curl -L -o kubectl_1.16 https://storage.googleapis.com/kubernetes-release/release/v1.16.9/bin/linux/amd64/kubectl -#RUN curl -L -o kubectl_1.15 https://storage.googleapis.com/kubernetes-release/release/v1.15.11/bin/linux/amd64/kubectl +RUN curl -L -o kubectl_1.16 https://storage.googleapis.com/kubernetes-release/release/v1.16.9/bin/linux/amd64/kubectl +RUN curl -L -o kubectl_1.15 https://storage.googleapis.com/kubernetes-release/release/v1.15.11/bin/linux/amd64/kubectl RUN curl -L -o kubectl_1.14 https://storage.googleapis.com/kubernetes-release/release/v1.14.10/bin/linux/amd64/kubectl # Tar each one up, to save space in the image RUN gzip kubectl_1.18 RUN gzip kubectl_1.17 +RUN gzip kubectl_1.16 +RUN gzip kubectl_1.15 RUN gzip kubectl_1.14 # Fetch Helm 3 package @@ -21,9 +23,12 @@ RUN curl -L -o helm.tar.gz https://get.helm.sh/helm-v3.1.2-linux-amd64.tar.gz && RUN ls -al -FROM splatform/stratos-bk-init-base:leap15_1 +# Use small base image with very little in it +FROM splatform/stratos-base:leap15_1 -RUN zypper in -y gzip +# Use gzip from the builder image +COPY --from=terminal-builder /usr/bin/gunzip /usr/bin/ +COPY --from=terminal-builder /usr/bin/gzip /usr/bin/ RUN mkdir /stratos @@ -39,8 +44,7 @@ RUN chown -R stratos /stratos && \ chgrp -R users /stratos # Remove a few packages -RUN zypper rm -y diffutils shadow util-linux fillup python3 openssl bind-utils kmod \ - python3-base python-rpm-macros pkg-config hostname GeoIP-data +RUN zypper rm -y diffutils shadow fillup openssl # Remove zypper RUN zypper rm -y dirmngr && \ diff --git a/deploy/kubernetes/build.sh b/deploy/kubernetes/build.sh index 63cc4a0f63..a9eb81afe3 100755 --- a/deploy/kubernetes/build.sh +++ b/deploy/kubernetes/build.sh @@ -169,11 +169,10 @@ function patchDockerfile { if [ "${DOCKER_REG_DEFAULTS}" == "false" ]; then sed -i.bak "s@splatform@${DOCKER_REGISTRY}/${DOCKER_ORG}@g" "${FOLDER}/${PATCHED_DOCKER_FILE}" fi - sed -i.bak "s/opensuse/${BASE_IMAGE_TAG}/g" "${FOLDER}/${PATCHED_DOCKER_FILE}" + sed -i.bak "s/leap15_1/${BASE_IMAGE_TAG}/g" "${FOLDER}/${PATCHED_DOCKER_FILE}" popd > /dev/null 2>&1 } - # # MAIN ------------------------------------------------------------------------------------------- # @@ -219,6 +218,8 @@ if [ "${CHART_ONLY}" == "false" ]; then fi fi +custom_image_build + log "-- Building Helm Chart" # Don't change the chart in the repo, copy it and modify it locally From 7af08f21a2ff8b7e310863dca141ae99b62fe503 Mon Sep 17 00:00:00 2001 From: Neil MacDougall Date: Thu, 4 Jun 2020 14:41:54 +0100 Subject: [PATCH 29/40] Reanme docker file and add image to CI pipeline --- custom-src/deploy/kubernetes/custom-build.sh | 2 +- deploy/ci/suse-console-dev-releases.yml | 15 ++++++++++++++- ...erfile.kubeconsole => Dockerfile.kubeterminal} | 0 3 files changed, 15 insertions(+), 2 deletions(-) rename deploy/containers/kube-terminal/{Dockerfile.kubeconsole => Dockerfile.kubeterminal} (100%) diff --git a/custom-src/deploy/kubernetes/custom-build.sh b/custom-src/deploy/kubernetes/custom-build.sh index 14e129d336..35d6202755 100644 --- a/custom-src/deploy/kubernetes/custom-build.sh +++ b/custom-src/deploy/kubernetes/custom-build.sh @@ -24,5 +24,5 @@ function custom_image_build() { # Build and push an image for the Kubernetes Terminal log "-- Building/publishing Kubernetes Terminal" - patchAndPushImage stratos-kube-terminal Dockerfile.kubeconsole "${STRATOS_PATH}/deploy/containers/kube-terminal" + patchAndPushImage stratos-kube-terminal Dockerfile.kubeterminal "${STRATOS_PATH}/deploy/containers/kube-terminal" } \ No newline at end of file diff --git a/deploy/ci/suse-console-dev-releases.yml b/deploy/ci/suse-console-dev-releases.yml index 82b15b962c..ef2247a720 100644 --- a/deploy/ci/suse-console-dev-releases.yml +++ b/deploy/ci/suse-console-dev-releases.yml @@ -65,7 +65,13 @@ resources: username: ((docker-username)) password: ((docker-password)) repository: ((docker-repository))/stratos-chartsync - +- name: kube-terminal-image + type: docker-image + source: + username: ((docker-username)) + password: ((docker-password)) + repository: ((docker-repository))/stratos-kube-terminal + # Artifacts - name: image-tag type: s3 @@ -147,6 +153,13 @@ jobs: tag: image-tag/v2-alpha-tag patch_base_reg: ((patch-base-reg)) patch_base_tag: ((patch-base-tag)) + - put: kube-terminal-image + params: + dockerfile: stratos/deploy/containers/kube-terminal/Dockerfile.kubeterminal + build: stratos/deploy/containers/kube-terminal + tag: image-tag/v2-alpha-tag + patch_base_reg: ((patch-base-reg)) + patch_base_tag: ((patch-base-tag)) - do: - put: ui-image params: diff --git a/deploy/containers/kube-terminal/Dockerfile.kubeconsole b/deploy/containers/kube-terminal/Dockerfile.kubeterminal similarity index 100% rename from deploy/containers/kube-terminal/Dockerfile.kubeconsole rename to deploy/containers/kube-terminal/Dockerfile.kubeterminal From b935ca47a2a6a7823d67dd64e0e49614d36beac0 Mon Sep 17 00:00:00 2001 From: Neil MacDougall Date: Mon, 8 Jun 2020 11:49:06 +0100 Subject: [PATCH 30/40] Allow fork to add images to the imagelist file --- custom-src/deploy/kubernetes/imagelist.txt | 1 + deploy/common-build.sh | 1 - deploy/kubernetes/build.sh | 6 +++--- deploy/kubernetes/imagelist-gen.sh | 20 ++++++++++++++++++++ 4 files changed, 24 insertions(+), 4 deletions(-) create mode 100644 custom-src/deploy/kubernetes/imagelist.txt diff --git a/custom-src/deploy/kubernetes/imagelist.txt b/custom-src/deploy/kubernetes/imagelist.txt new file mode 100644 index 0000000000..e628bfa984 --- /dev/null +++ b/custom-src/deploy/kubernetes/imagelist.txt @@ -0,0 +1 @@ +stratos-kube-terminal:_VERSION_ \ No newline at end of file diff --git a/deploy/common-build.sh b/deploy/common-build.sh index 6f29c55d1e..913c159a17 100644 --- a/deploy/common-build.sh +++ b/deploy/common-build.sh @@ -113,7 +113,6 @@ function cleanup { echo "-- Cleaning up ${STRATOS_PATH}" rm -rf ${STRATOS_PATH}/dist rm -rf ${STRATOS_PATH}/node_modules - rm -rf ${STRATOS_PATH}/bower_components echo echo "-- Cleaning up ${STRATOS_PATH}/deploy/containers/nginx/dist" rm -rf ${STRATOS_PATH}/deploy/containers/nginx/dist diff --git a/deploy/kubernetes/build.sh b/deploy/kubernetes/build.sh index a9eb81afe3..1b19bf1408 100755 --- a/deploy/kubernetes/build.sh +++ b/deploy/kubernetes/build.sh @@ -183,7 +183,9 @@ popd > /dev/null 2>&1 echo "Base path: ${STRATOS_PATH}" # cleanup output, intermediate artifacts -cleanup +if [ "${CHART_ONLY}" == "false" ]; then + cleanup +fi # Clean any old patched docker files left if previously errored # rm -rf ${STRATOS_PATH}/deploy/Dockerfile.*.patched @@ -218,8 +220,6 @@ if [ "${CHART_ONLY}" == "false" ]; then fi fi -custom_image_build - log "-- Building Helm Chart" # Don't change the chart in the repo, copy it and modify it locally diff --git a/deploy/kubernetes/imagelist-gen.sh b/deploy/kubernetes/imagelist-gen.sh index 293c1a7ba4..5f42571609 100755 --- a/deploy/kubernetes/imagelist-gen.sh +++ b/deploy/kubernetes/imagelist-gen.sh @@ -12,6 +12,22 @@ __DIRNAME="$( cd "$( dirname "${BASH_SOURCE[0]}" )" && pwd )" printf "${BOLD}${CYAN}Generating ${YELLOW}imagelist.txt${RESET}\n" echo "" +STRATOS_FOLDER="$( cd "$( dirname "${BASH_SOURCE[0]}" )" && cd ../../ && pwd )" + +# Add any customizations +function addCustomizations() { + + if [ -f "${STRATOS_FOLDER}/custom-src/deploy/kubernetes/imagelist.txt" ];then + echo "Including custom imagelist contents" + cat "${STRATOS_FOLDER}/custom-src/deploy/kubernetes/imagelist.txt" >> ./imagelist.txt + + # Update version number + VERSION=$(grep -Po 'consoleVersion: \K(.*)' ./values.yaml) + echo "Image Version: ${VERSION}" + sed -i 's/_VERSION_/'"${VERSION}"'/g' imagelist.txt + fi +} + CHART_FOLDER=${1} @@ -37,6 +53,10 @@ ls -alR echo "" helm template -f ${__DIRNAME}/imagelist.values.yaml ${CHART_FOLDER} | grep "image:" | grep --extended --only-matching '([^"/[:space:]]+/)?[^"/[:space:]]+/[^:[:space:]]+:[a-zA-Z0-9\._-]+' | sort | uniq | awk -F'/' '{print $2}' > imagelist.txt + +# Add any customizations +addCustomizations + popd > /dev/null printf "${CYAN}" From da9610c575ef766950bf5613d63df3c3f05e1dd9 Mon Sep 17 00:00:00 2001 From: Richard Cox Date: Fri, 12 Jun 2020 14:16:00 +0100 Subject: [PATCH 31/40] Fix issue where no config caused panic --- src/jetstream/plugins/kubernetes/main.go | 7 ++++--- 1 file changed, 4 insertions(+), 3 deletions(-) diff --git a/src/jetstream/plugins/kubernetes/main.go b/src/jetstream/plugins/kubernetes/main.go index 8cccc03ce4..929c610494 100644 --- a/src/jetstream/plugins/kubernetes/main.go +++ b/src/jetstream/plugins/kubernetes/main.go @@ -55,14 +55,16 @@ const ( // kubeDashboardPluginConfigSetting is config value sent back to the client to indicate if the kube dashboard ie enabled kubeDashboardPluginConfigSetting = "kubeDashboardEnabled" // kubeTerminalPluginConfigSetting is config value sent back to the client to indicate if the kube terminal is enabled - kubeTerminalPluginConfigSetting ="kubeTerminalEnabled" + kubeTerminalPluginConfigSetting = "kubeTerminalEnabled" ) // Init creates a new instance of the Kubernetes plugin func Init(portalProxy interfaces.PortalProxy) (interfaces.StratosPlugin, error) { kubeTerminal := terminal.NewKubeTerminal(portalProxy) kube := &KubernetesSpecification{portalProxy: portalProxy, endpointType: kubeEndpointType, kubeTerminal: kubeTerminal} - kubeTerminal.Kube = kube + if kubeTerminal != nil { + kubeTerminal.Kube = kube + } return kube, nil } @@ -143,7 +145,6 @@ func (c *KubernetesSpecification) Init() error { // Kube terminal is enabled by Tech Preview mode c.portalProxy.GetConfig().PluginConfig[kubeTerminalPluginConfigSetting] = strconv.FormatBool(c.portalProxy.GetConfig().EnableTechPreview) - // Kick off the cleanup of any old kube terminal pods c.kubeTerminal.StartCleanup() From e798fc6793c55384df7e7673f3476abe8b6c1c71 Mon Sep 17 00:00:00 2001 From: Neil MacDougall Date: Tue, 23 Jun 2020 11:22:09 +0100 Subject: [PATCH 32/40] Indicate which commands are available --- deploy/containers/kube-terminal/kubeconsole.bashrc | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/deploy/containers/kube-terminal/kubeconsole.bashrc b/deploy/containers/kube-terminal/kubeconsole.bashrc index e71545ed5a..0a1edfdd56 100644 --- a/deploy/containers/kube-terminal/kubeconsole.bashrc +++ b/deploy/containers/kube-terminal/kubeconsole.bashrc @@ -72,5 +72,6 @@ touch "/stratos/.firstrun" # Remove any env vars matching KUBERNETES unset `compgen -A variable | grep KUBERNETES` -echo "Ready" +echo +echo -e "Ready - ${CYAN}kubectl${RESET} and ${CYAN}helm${RESET} commands are available" echo "" From f34497f1ee6e3cea9967fd456793d0d65f829753 Mon Sep 17 00:00:00 2001 From: Neil MacDougall Date: Tue, 23 Jun 2020 11:25:30 +0100 Subject: [PATCH 33/40] Fix merge issue --- deploy/kubernetes/imagelist-gen.sh | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/deploy/kubernetes/imagelist-gen.sh b/deploy/kubernetes/imagelist-gen.sh index 4114e81aae..1605cdb54d 100755 --- a/deploy/kubernetes/imagelist-gen.sh +++ b/deploy/kubernetes/imagelist-gen.sh @@ -52,16 +52,16 @@ printf "${CYAN}Chart folder contents:${RESET}\n" ls -alR echo "" -# Add any customizations -addCustomizations - helm template -f ${__DIRNAME}/imagelist.values.yaml ${CHART_FOLDER} | grep "image:" | grep --extended --only-matching '([^"/[:space:]]+/)?[^"/[:space:]]+/[^:[:space:]]+:[a-zA-Z0-9\._-]+' | sort | uniq | awk -F'/' '{print $2}' > imagelist.txt - if [ $? -ne 0 ]; then echo -e "${BOLD}${RED}ERROR: Failed to render Helm Chart in order to generate image list" exit 1 fi +# Add any customizations to the image list +# Mainly used if there are unreferenced images that need to be included +addCustomizations + popd > /dev/null printf "${CYAN}" From 53a4b2a2171eec7d620c3d5706e6d5497b73c591 Mon Sep 17 00:00:00 2001 From: Neil MacDougall Date: Tue, 23 Jun 2020 11:27:04 +0100 Subject: [PATCH 34/40] Remove commented-out code --- .../src/shared/components/ssh-viewer/ssh-viewer.component.ts | 2 -- 1 file changed, 2 deletions(-) diff --git a/src/frontend/packages/core/src/shared/components/ssh-viewer/ssh-viewer.component.ts b/src/frontend/packages/core/src/shared/components/ssh-viewer/ssh-viewer.component.ts index c12df6d6a1..8d9cedccb8 100644 --- a/src/frontend/packages/core/src/shared/components/ssh-viewer/ssh-viewer.component.ts +++ b/src/frontend/packages/core/src/shared/components/ssh-viewer/ssh-viewer.component.ts @@ -68,7 +68,6 @@ export class SshViewerComponent implements OnInit, OnDestroy { this.xterm = new Terminal(); this.xterm.loadAddon(this.xtermFitAddon); this.xterm.open(this.container.nativeElement); - // this.xtermFitAddon.fit(); this.resize(); this.xterm.onKey(e => { @@ -125,7 +124,6 @@ export class SshViewerComponent implements OnInit, OnDestroy { } else { console.log('Error') const eMsg = this.errorMessage; - //this.disconnect(); this.errorMessage = eMsg; } }, From 817b72948d9541e58f0370b5e95df6f1b4e4e966 Mon Sep 17 00:00:00 2001 From: Neil MacDougall Date: Tue, 23 Jun 2020 15:27:11 +0100 Subject: [PATCH 35/40] Fix unit test compilation --- .../kube-terminal/kube-console.component.spec.ts | 9 +++++---- 1 file changed, 5 insertions(+), 4 deletions(-) diff --git a/custom-src/frontend/app/custom/kubernetes/kube-terminal/kube-console.component.spec.ts b/custom-src/frontend/app/custom/kubernetes/kube-terminal/kube-console.component.spec.ts index 85fda06bf6..8671538bc9 100644 --- a/custom-src/frontend/app/custom/kubernetes/kube-terminal/kube-console.component.spec.ts +++ b/custom-src/frontend/app/custom/kubernetes/kube-terminal/kube-console.component.spec.ts @@ -1,9 +1,10 @@ -import { async, ComponentFixture, TestBed } from './@angular/core/testing'; -import { RouterTestingModule } from './@angular/router/testing'; +import { async, ComponentFixture, TestBed } from '@angular/core/testing'; +import { RouterTestingModule } from '@angular/router/testing'; +import { createBasicStoreModule } from '@stratosui/store/testing'; +import { ApplicationService } from '../../../../../cloud-foundry/src/features/applications/application.service'; +import { ApplicationServiceMock } from '../../../../../cloud-foundry/test-framework/application-service-helper'; import { TabNavService } from '../../../../tab-nav.service'; -import { ApplicationServiceMock } from '../../../../test-framework/application-service-helper'; -import { createBasicStoreModule } from '../../../../test-framework/store-test-helper'; import { CoreModule } from '../../../core/core.module'; import { SharedModule } from '../../../shared/shared.module'; import { KubeConsoleComponent } from './kube-console.component'; From 3ec93620868aa0eb1b4efba269c56533c2c458a5 Mon Sep 17 00:00:00 2001 From: Neil MacDougall Date: Wed, 24 Jun 2020 11:44:44 +0100 Subject: [PATCH 36/40] Bug fix for when terminal is not enabled --- src/jetstream/plugins/kubernetes/main.go | 8 ++++++-- 1 file changed, 6 insertions(+), 2 deletions(-) diff --git a/src/jetstream/plugins/kubernetes/main.go b/src/jetstream/plugins/kubernetes/main.go index 929c610494..2cee120fcc 100644 --- a/src/jetstream/plugins/kubernetes/main.go +++ b/src/jetstream/plugins/kubernetes/main.go @@ -146,7 +146,9 @@ func (c *KubernetesSpecification) Init() error { c.portalProxy.GetConfig().PluginConfig[kubeTerminalPluginConfigSetting] = strconv.FormatBool(c.portalProxy.GetConfig().EnableTechPreview) // Kick off the cleanup of any old kube terminal pods - c.kubeTerminal.StartCleanup() + if c.kubeTerminal != nil { + c.kubeTerminal.StartCleanup() + } return nil } @@ -177,7 +179,9 @@ func (c *KubernetesSpecification) AddSessionGroupRoutes(echoGroup *echo.Group) { echoGroup.GET("/helm/releases/:endpoint/:namespace/:name", c.GetRelease) // Kube Terminal - echoGroup.GET("/kubeterminal/:guid", c.kubeTerminal.Start) + if c.kubeTerminal != nil { + echoGroup.GET("/kubeterminal/:guid", c.kubeTerminal.Start) + } } func (c *KubernetesSpecification) Info(apiEndpoint string, skipSSLValidation bool) (interfaces.CNSIRecord, interface{}, error) { From 16326d56fb94a46a8341e26825c94cd32a19d860 Mon Sep 17 00:00:00 2001 From: Neil MacDougall Date: Wed, 24 Jun 2020 11:48:18 +0100 Subject: [PATCH 37/40] Only enable if tech preview is on --- .../plugins/kubernetes/terminal/terminal.go | 28 +++++++++++-------- 1 file changed, 17 insertions(+), 11 deletions(-) diff --git a/src/jetstream/plugins/kubernetes/terminal/terminal.go b/src/jetstream/plugins/kubernetes/terminal/terminal.go index a77f792908..83d259147b 100644 --- a/src/jetstream/plugins/kubernetes/terminal/terminal.go +++ b/src/jetstream/plugins/kubernetes/terminal/terminal.go @@ -4,8 +4,8 @@ import ( "fmt" "io/ioutil" - "github.com/cloudfoundry-incubator/stratos/src/jetstream/repository/interfaces" "github.com/cloudfoundry-incubator/stratos/src/jetstream/plugins/kubernetes/api" + "github.com/cloudfoundry-incubator/stratos/src/jetstream/repository/interfaces" "github.com/cloudfoundry-incubator/stratos/src/jetstream/repository/interfaces/config" log "github.com/sirupsen/logrus" @@ -13,13 +13,13 @@ import ( const ( serviceAccountTokenFile = "/var/run/secrets/kubernetes.io/serviceaccount/token" - serviceHostEnvVar = "KUBERNETES_SERVICE_HOST" - servicePortEnvVar = "KUBERNETES_SERVICE_PORT" + serviceHostEnvVar = "KUBERNETES_SERVICE_HOST" + servicePortEnvVar = "KUBERNETES_SERVICE_PORT" // For dev - read token from env var serviceTokenEnvVar = "KUBE_TERMINAL_SERVICE_ACCOUNT_TOKEN" - stratosRoleLabel = "stratos-role" - stratosKubeTerminalRole = "kube-terminal" + stratosRoleLabel = "stratos-role" + stratosKubeTerminalRole = "kube-terminal" stratosSessionAnnotation = "stratos-session" consoleContainerName = "kube-terminal" @@ -27,16 +27,22 @@ const ( // KubeTerminal supports spawning pods to provide a CLI environment to the user type KubeTerminal struct { - PortalProxy interfaces.PortalProxy - Namespace string `configName:"STRATOS_KUBERNETES_NAMESPACE"` - Image string `configName:"STRATOS_KUBERNETES_TERMINAL_IMAGE"` - Token []byte - APIServer string - Kube api.Kubernetes + PortalProxy interfaces.PortalProxy + Namespace string `configName:"STRATOS_KUBERNETES_NAMESPACE"` + Image string `configName:"STRATOS_KUBERNETES_TERMINAL_IMAGE"` + Token []byte + APIServer string + Kube api.Kubernetes } // NewKubeTerminal checks that the environment is set up to support the Kube Terminal func NewKubeTerminal(p interfaces.PortalProxy) *KubeTerminal { + // Only enabled in tech preview + if !p.GetConfig().EnableTechPreview { + log.Info("Kube Terminal not enabled - requires tech preview") + return nil + } + kt := &KubeTerminal{ PortalProxy: p, } From cd65b145a8c0cefde02f0a7f77b83d15bde21772 Mon Sep 17 00:00:00 2001 From: Neil MacDougall Date: Wed, 24 Jun 2020 16:49:25 +0100 Subject: [PATCH 38/40] Add dev doc and helper setup script --- build/tools/kube-terminal-dev.sh | 58 ++++++++++++++++++++++++++++++++ docs/suse/kube-terminal-dev.md | 21 ++++++++++++ 2 files changed, 79 insertions(+) create mode 100755 build/tools/kube-terminal-dev.sh create mode 100644 docs/suse/kube-terminal-dev.md diff --git a/build/tools/kube-terminal-dev.sh b/build/tools/kube-terminal-dev.sh new file mode 100755 index 0000000000..2ec5c69c3d --- /dev/null +++ b/build/tools/kube-terminal-dev.sh @@ -0,0 +1,58 @@ +#!/usr/bin/env bash + +# Colours +CYAN="\033[96m" +YELLOW="\033[93m" +RED="\033[91m" +RESET="\033[0m" +BOLD="\033[1m" + +# Program Paths: +PROG=$(basename ${BASH_SOURCE[0]}) +PROG_DIR="$( cd "$( dirname "${BASH_SOURCE[0]}" )" && pwd )" +STRATOS_DIR="$( cd "${PROG_DIR}/../.." && pwd )" + +echo "Creating Service Account" +SRC="${STRATOS_DIR}/deploy/kubernetes/console/templates/service-account.yaml" + +TEMPFILE=$(mktemp) +cp $SRC $TEMPFILE +sed -i.bak '/\s*helm/d' $TEMPFILE +sed -i.bak '/\s*app\.kubernetes\.io\/version/d' $TEMPFILE +sed -i.bak '/\s*app\.kubernetes\.io\/instance/d' $TEMPFILE +sed -i.bak '/\s*{{-/d' $TEMPFILE + +kubectl apply -f $TEMPFILE + +# Service account should be created - now need to get token +SECRET=$(kubectl get sa stratos -o json | jq -r '.secrets[0].name') +TOKEN=$(kubectl get secret $SECRET -o json | jq -r '.data.token') +echo "Token secret: $SECRET" +echo "Token $TOKEN" + +rm -f $TEMPFILE +rm -f $TEMPFILE.bak + +# Create a namespace +NS="stratos-dev" +kubectl get ns $NS > /dev/null +if [ $? -ne 0 ]; then + kubectl create ns $NS +fi + +CFG=${STRATOS_DIR}/src/jetstream/config.properties +touch $CFG + +echo -e "\n# Kubernetes Terminal Config for dev" >> $CFG +echo "STRATOS_KUBERNETES_NAMESPACE=stratos-dev" >> $CFG +echo "STRATOS_KUBERNETES_TERMINAL_IMAGE=splatform/stratos-kube-terminal:dev" >> $CFG +echo "KUBE_TERMINAL_SERVICE_ACCOUNT_TOKEN=$TOKEN" >> $CFG + +MKUBE=$(minikube ip) +if [ $? -eq 0 ]; then + echo "KUBERNETES_SERVICE_HOST=$MKUBE" >> $CFG + echo "KUBERNETES_SERVICE_PORT=8443" >> $CFG +else + echo "KUBERNETES_SERVICE_HOST=" >> $CFG + echo "KUBERNETES_SERVICE_PORT=8443" >> $CFG +fi diff --git a/docs/suse/kube-terminal-dev.md b/docs/suse/kube-terminal-dev.md new file mode 100644 index 0000000000..fe552b064c --- /dev/null +++ b/docs/suse/kube-terminal-dev.md @@ -0,0 +1,21 @@ +# Enabling the Kubernetes Terminal in local development + +You need a Kubernetes cluster with `kubectl` set up and configured with the kubeconfig file. + +Run the script `build/tools/kube-terminal-dev.sh` + +This script will: + +- Create a service account named `stratos` +- Create a namespace named `stratos-dev` +- Write environment variables to the `src/jetstream/config.properties` file + +If you have minikube running, the configuration for your Kubernetes API Server will be set correctly - otherwise +you will need to edit the `src/jetstream/config.properties` file and set these two variables: + +- `KUBERNETES_SERVICE_HOST` +- `KUBERNETES_SERVICE_PORT` + +The Jetstream backend should be configured. + +> Note: Ensure you set `ENABLE_TECH_PREVIEW=true` to enable the Kubernetes Terminal feature. \ No newline at end of file From cd0d6d6be53ceb8f6d9b0f57aa602aea15ce1d8a Mon Sep 17 00:00:00 2001 From: Neil MacDougall Date: Fri, 26 Jun 2020 11:41:06 +0100 Subject: [PATCH 39/40] Fix dev helper script --- build/tools/kube-terminal-dev.sh | 23 +++++++++++++---------- 1 file changed, 13 insertions(+), 10 deletions(-) diff --git a/build/tools/kube-terminal-dev.sh b/build/tools/kube-terminal-dev.sh index 2ec5c69c3d..ce1ab1549e 100755 --- a/build/tools/kube-terminal-dev.sh +++ b/build/tools/kube-terminal-dev.sh @@ -22,24 +22,27 @@ sed -i.bak '/\s*app\.kubernetes\.io\/version/d' $TEMPFILE sed -i.bak '/\s*app\.kubernetes\.io\/instance/d' $TEMPFILE sed -i.bak '/\s*{{-/d' $TEMPFILE -kubectl apply -f $TEMPFILE +# Create a namespace +NS="stratos-dev" +kubectl get ns $NS > /dev/null 2>&1 +if [ $? -ne 0 ]; then + kubectl create ns $NS +fi + +kubectl apply -n $NS -f $TEMPFILE +USER=stratos-dev-admin-user +USER=stratos # Service account should be created - now need to get token -SECRET=$(kubectl get sa stratos -o json | jq -r '.secrets[0].name') -TOKEN=$(kubectl get secret $SECRET -o json | jq -r '.data.token') +SECRET=$(kubectl get -n $NS sa $USER -o json | jq -r '.secrets[0].name') +TOKEN=$(kubectl get -n $NS secret $SECRET -o json | jq -r '.data.token') echo "Token secret: $SECRET" +TOKEN=$(echo $TOKEN | base64 -d -) echo "Token $TOKEN" rm -f $TEMPFILE rm -f $TEMPFILE.bak -# Create a namespace -NS="stratos-dev" -kubectl get ns $NS > /dev/null -if [ $? -ne 0 ]; then - kubectl create ns $NS -fi - CFG=${STRATOS_DIR}/src/jetstream/config.properties touch $CFG From 4c206ad49612fccd897ae7e45f05d6baf277ca6e Mon Sep 17 00:00:00 2001 From: Neil MacDougall Date: Fri, 26 Jun 2020 14:41:23 +0100 Subject: [PATCH 40/40] Address PR feedback --- .../plugins/kubernetes/terminal/cleanup.go | 14 ++++++++------ src/jetstream/plugins/kubernetes/terminal/start.go | 3 --- 2 files changed, 8 insertions(+), 9 deletions(-) diff --git a/src/jetstream/plugins/kubernetes/terminal/cleanup.go b/src/jetstream/plugins/kubernetes/terminal/cleanup.go index d1d50aa559..621921e1e5 100644 --- a/src/jetstream/plugins/kubernetes/terminal/cleanup.go +++ b/src/jetstream/plugins/kubernetes/terminal/cleanup.go @@ -22,6 +22,8 @@ func (k *KubeTerminal) cleanup() { // Use a random initial wait before cleaning up // If we had more than one backend, this helps to ensure they are not all trying to cleanup at the same time wait := rand.Intn(30) + log.Debug("Kubernetes Terminal cleanup will start in %d minutes", wait) + for { time.Sleep(time.Duration(wait) * time.Minute) log.Debug("Cleaning up stale Kubernetes Terminal pods and secrets ...") @@ -35,7 +37,7 @@ func (k *KubeTerminal) cleanup() { pods, err := podClient.List(options) if err == nil { for _, pod := range pods.Items { - if sessionID, ok := pod.Annotations["stratos-session"]; ok { + if sessionID, ok := pod.Annotations[stratosSessionAnnotation]; ok { i, err := strconv.Atoi(sessionID) if err == nil { isValid, err := k.PortalProxy.GetSessionDataStore().IsValidSession(i) @@ -55,7 +57,7 @@ func (k *KubeTerminal) cleanup() { secrets, err := secretClient.List(options) if err == nil { for _, secret := range secrets.Items { - if sessionID, ok := secret.Annotations["stratos-session"]; ok { + if sessionID, ok := secret.Annotations[stratosSessionAnnotation]; ok { i, err := strconv.Atoi(sessionID) if err == nil { isValid, err := k.PortalProxy.GetSessionDataStore().IsValidSession(i) @@ -67,13 +69,13 @@ func (k *KubeTerminal) cleanup() { } } } else { - log.Debug("Kube Terminal Cleanup: Could not get secrets") - log.Debug(err) + log.Warn("Kube Terminal Cleanup: Could not get secrets") + log.Warn(err) } } else { - log.Debug("Kube Terminal Cleanup: Could not get clients") - log.Debug(err) + log.Warn("Kube Terminal Cleanup: Could not get clients") + log.Warn(err) } wait = waitPeriod diff --git a/src/jetstream/plugins/kubernetes/terminal/start.go b/src/jetstream/plugins/kubernetes/terminal/start.go index d0ac6ac276..63c90fe519 100644 --- a/src/jetstream/plugins/kubernetes/terminal/start.go +++ b/src/jetstream/plugins/kubernetes/terminal/start.go @@ -26,7 +26,6 @@ var upgrader = websocket.Upgrader{ CheckOrigin: func(r *http.Request) bool { return true }, } - // KeyCode - JSON object that is passed from the front-end to notify of a key press or a term resize type KeyCode struct { Key string `json:"key"` @@ -103,11 +102,9 @@ func (k *KubeTerminal) Start(c echo.Context) error { // API Endpoint to SSH/exec into a container target := fmt.Sprintf("%s/api/v1/namespaces/%s/pods/%s/exec?command=/bin/bash&stdin=true&stderr=true&stdout=true&tty=true", k.APIServer, k.Namespace, podData.PodName) - // This dialer does not use the kubeHttpClient - is it unused? dialer := &websocket.Dialer{ TLSClientConfig: &tls.Config{ InsecureSkipVerify: true, - //Certificates: []tls.Certificate{cert}, }, }